notepad-plus-plus/scintilla/src/Decoration.cxx

312 lines
9.7 KiB
C++

/** @file Decoration.cxx
** Visual elements added over text.
**/
// Copyright 1998-2007 by Neil Hodgson <neilh@scintilla.org>
// The License.txt file describes the conditions under which this software may be distributed.
#include <cstddef>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cstdarg>
#include <stdexcept>
#include <string_view>
#include <vector>
#include <optional>
#include <algorithm>
#include <memory>
#include "ScintillaTypes.h"
#include "Debugging.h"
#include "Position.h"
#include "SplitVector.h"
#include "Partitioning.h"
#include "RunStyles.h"
#include "Decoration.h"
using namespace Scintilla::Internal;
namespace {
template <typename POS>
class Decoration : public IDecoration {
int indicator;
public:
RunStyles<POS, int> rs;
explicit Decoration(int indicator_) : indicator(indicator_) {
}
bool Empty() const noexcept override {
return (rs.Runs() == 1) && (rs.AllSameAs(0));
}
int Indicator() const noexcept override {
return indicator;
}
Sci::Position Length() const noexcept override {
return rs.Length();
}
int ValueAt(Sci::Position position) const noexcept override {
return rs.ValueAt(static_cast<POS>(position));
}
Sci::Position StartRun(Sci::Position position) const noexcept override {
return rs.StartRun(static_cast<POS>(position));
}
Sci::Position EndRun(Sci::Position position) const noexcept override {
return rs.EndRun(static_cast<POS>(position));
}
void SetValueAt(Sci::Position position, int value) override {
rs.SetValueAt(static_cast<POS>(position), value);
}
void InsertSpace(Sci::Position position, Sci::Position insertLength) override {
rs.InsertSpace(static_cast<POS>(position), static_cast<POS>(insertLength));
}
Sci::Position Runs() const noexcept override {
return rs.Runs();
}
};
template <typename POS>
class DecorationList : public IDecorationList {
int currentIndicator;
int currentValue;
Decoration<POS> *current; // Non-owning. Cached so FillRange doesn't have to search for each call.
Sci::Position lengthDocument;
// Ordered by indicator
std::vector<std::unique_ptr<Decoration<POS>>> decorationList;
std::vector<const IDecoration*> decorationView; // Read-only view of decorationList
bool clickNotified;
Decoration<POS> *DecorationFromIndicator(int indicator) noexcept;
Decoration<POS> *Create(int indicator, Sci::Position length);
void Delete(int indicator);
void DeleteAnyEmpty();
void SetView();
public:
DecorationList();
const std::vector<const IDecoration*> &View() const noexcept override {
return decorationView;
}
void SetCurrentIndicator(int indicator) override;
int GetCurrentIndicator() const noexcept override { return currentIndicator; }
void SetCurrentValue(int value) override;
int GetCurrentValue() const noexcept override { return currentValue; }
// Returns changed=true if some values may have changed
FillResult<Sci::Position> FillRange(Sci::Position position, int value, Sci::Position fillLength) override;
void InsertSpace(Sci::Position position, Sci::Position insertLength) override;
void DeleteRange(Sci::Position position, Sci::Position deleteLength) override;
void DeleteLexerDecorations() override;
int AllOnFor(Sci::Position position) const noexcept override;
int ValueAt(int indicator, Sci::Position position) noexcept override;
Sci::Position Start(int indicator, Sci::Position position) noexcept override;
Sci::Position End(int indicator, Sci::Position position) noexcept override;
bool ClickNotified() const noexcept override {
return clickNotified;
}
void SetClickNotified(bool notified) noexcept override {
clickNotified = notified;
}
};
template <typename POS>
DecorationList<POS>::DecorationList() : currentIndicator(0), currentValue(1), current(nullptr),
lengthDocument(0), clickNotified(false) {
}
template <typename POS>
Decoration<POS> *DecorationList<POS>::DecorationFromIndicator(int indicator) noexcept {
for (const std::unique_ptr<Decoration<POS>> &deco : decorationList) {
if (deco->Indicator() == indicator) {
return deco.get();
}
}
return nullptr;
}
template <typename POS>
Decoration<POS> *DecorationList<POS>::Create(int indicator, Sci::Position length) {
currentIndicator = indicator;
std::unique_ptr<Decoration<POS>> decoNew = std::make_unique<Decoration<POS>>(indicator);
decoNew->rs.InsertSpace(0, static_cast<POS>(length));
typename std::vector<std::unique_ptr<Decoration<POS>>>::iterator it = std::lower_bound(
decorationList.begin(), decorationList.end(), decoNew,
[](const std::unique_ptr<Decoration<POS>> &a, const std::unique_ptr<Decoration<POS>> &b) noexcept {
return a->Indicator() < b->Indicator();
});
typename std::vector<std::unique_ptr<Decoration<POS>>>::iterator itAdded =
decorationList.insert(it, std::move(decoNew));
SetView();
return itAdded->get();
}
template <typename POS>
void DecorationList<POS>::Delete(int indicator) {
decorationList.erase(std::remove_if(decorationList.begin(), decorationList.end(),
[indicator](const std::unique_ptr<Decoration<POS>> &deco) noexcept {
return deco->Indicator() == indicator;
}), decorationList.end());
current = nullptr;
SetView();
}
template <typename POS>
void DecorationList<POS>::SetCurrentIndicator(int indicator) {
currentIndicator = indicator;
current = DecorationFromIndicator(indicator);
currentValue = 1;
}
template <typename POS>
void DecorationList<POS>::SetCurrentValue(int value) {
currentValue = value ? value : 1;
}
template <typename POS>
FillResult<Sci::Position> DecorationList<POS>::FillRange(Sci::Position position, int value, Sci::Position fillLength) {
if (!current) {
current = DecorationFromIndicator(currentIndicator);
if (!current) {
current = Create(currentIndicator, lengthDocument);
}
}
// Converting result from POS to Sci::Position as callers not polymorphic.
const FillResult<POS> frInPOS = current->rs.FillRange(static_cast<POS>(position), value, static_cast<POS>(fillLength));
const FillResult<Sci::Position> fr { frInPOS.changed, frInPOS.position, frInPOS.fillLength };
if (current->Empty()) {
Delete(currentIndicator);
}
return fr;
}
template <typename POS>
void DecorationList<POS>::InsertSpace(Sci::Position position, Sci::Position insertLength) {
const bool atEnd = position == lengthDocument;
lengthDocument += insertLength;
for (const std::unique_ptr<Decoration<POS>> &deco : decorationList) {
deco->rs.InsertSpace(static_cast<POS>(position), static_cast<POS>(insertLength));
if (atEnd) {
deco->rs.FillRange(static_cast<POS>(position), 0, static_cast<POS>(insertLength));
}
}
}
template <typename POS>
void DecorationList<POS>::DeleteRange(Sci::Position position, Sci::Position deleteLength) {
lengthDocument -= deleteLength;
for (const std::unique_ptr<Decoration<POS>> &deco : decorationList) {
deco->rs.DeleteRange(static_cast<POS>(position), static_cast<POS>(deleteLength));
}
DeleteAnyEmpty();
if (decorationList.size() != decorationView.size()) {
// One or more empty decorations deleted so update view.
current = nullptr;
SetView();
}
}
template <typename POS>
void DecorationList<POS>::DeleteLexerDecorations() {
decorationList.erase(std::remove_if(decorationList.begin(), decorationList.end(),
[](const std::unique_ptr<Decoration<POS>> &deco) noexcept {
return deco->Indicator() < static_cast<int>(Scintilla::IndicatorNumbers::Container);
}), decorationList.end());
current = nullptr;
SetView();
}
template <typename POS>
void DecorationList<POS>::DeleteAnyEmpty() {
if (lengthDocument == 0) {
decorationList.clear();
} else {
decorationList.erase(std::remove_if(decorationList.begin(), decorationList.end(),
[](const std::unique_ptr<Decoration<POS>> &deco) noexcept {
return deco->Empty();
}), decorationList.end());
}
}
template <typename POS>
void DecorationList<POS>::SetView() {
decorationView.clear();
for (const std::unique_ptr<Decoration<POS>> &deco : decorationList) {
decorationView.push_back(deco.get());
}
}
template <typename POS>
int DecorationList<POS>::AllOnFor(Sci::Position position) const noexcept {
int mask = 0;
for (const std::unique_ptr<Decoration<POS>> &deco : decorationList) {
if (deco->rs.ValueAt(static_cast<POS>(position))) {
if (deco->Indicator() < static_cast<int>(Scintilla::IndicatorNumbers::Ime)) {
mask |= 1u << deco->Indicator();
}
}
}
return mask;
}
template <typename POS>
int DecorationList<POS>::ValueAt(int indicator, Sci::Position position) noexcept {
const Decoration<POS> *deco = DecorationFromIndicator(indicator);
if (deco) {
return deco->rs.ValueAt(static_cast<POS>(position));
}
return 0;
}
template <typename POS>
Sci::Position DecorationList<POS>::Start(int indicator, Sci::Position position) noexcept {
const Decoration<POS> *deco = DecorationFromIndicator(indicator);
if (deco) {
return deco->rs.StartRun(static_cast<POS>(position));
}
return 0;
}
template <typename POS>
Sci::Position DecorationList<POS>::End(int indicator, Sci::Position position) noexcept {
const Decoration<POS> *deco = DecorationFromIndicator(indicator);
if (deco) {
return deco->rs.EndRun(static_cast<POS>(position));
}
return 0;
}
}
namespace Scintilla::Internal {
std::unique_ptr<IDecoration> DecorationCreate(bool largeDocument, int indicator) {
if (largeDocument)
return std::make_unique<Decoration<Sci::Position>>(indicator);
else
return std::make_unique<Decoration<int>>(indicator);
}
std::unique_ptr<IDecorationList> DecorationListCreate(bool largeDocument) {
if (largeDocument)
return std::make_unique<DecorationList<Sci::Position>>();
else
return std::make_unique<DecorationList<int>>();
}
}