2022-01-04 23:07:50 +00:00
|
|
|
// @file ScintillaDocument.cpp
|
2013-08-28 00:44:27 +00:00
|
|
|
// Wrapper for Scintilla document object so it can be manipulated independently.
|
|
|
|
// Copyright (c) 2011 Archaeopteryx Software, Inc. d/b/a Wingware
|
|
|
|
|
2015-06-07 21:19:26 +00:00
|
|
|
#include <stdexcept>
|
2019-05-04 18:14:48 +00:00
|
|
|
#include <string_view>
|
2013-08-28 00:44:27 +00:00
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
2022-01-04 23:07:50 +00:00
|
|
|
#include <set>
|
|
|
|
#include <optional>
|
2019-05-04 18:14:48 +00:00
|
|
|
#include <memory>
|
2013-08-28 00:44:27 +00:00
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
#include "ScintillaTypes.h"
|
|
|
|
#include "ScintillaMessages.h"
|
|
|
|
#include "ScintillaStructures.h"
|
2013-08-28 00:44:27 +00:00
|
|
|
#include "ScintillaDocument.h"
|
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
#include "Debugging.h"
|
|
|
|
#include "Geometry.h"
|
2013-08-28 00:44:27 +00:00
|
|
|
#include "Platform.h"
|
|
|
|
|
2019-05-04 18:14:48 +00:00
|
|
|
#include "ILoader.h"
|
2013-08-28 00:44:27 +00:00
|
|
|
#include "ILexer.h"
|
|
|
|
#include "Scintilla.h"
|
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
#include "CharacterCategoryMap.h"
|
2019-05-04 18:14:48 +00:00
|
|
|
#include "Position.h"
|
|
|
|
#include "UniqueString.h"
|
2013-08-28 00:44:27 +00:00
|
|
|
#include "SplitVector.h"
|
|
|
|
#include "Partitioning.h"
|
|
|
|
#include "RunStyles.h"
|
|
|
|
#include "ContractionState.h"
|
|
|
|
#include "CellBuffer.h"
|
|
|
|
#include "KeyMap.h"
|
|
|
|
#include "Indicator.h"
|
|
|
|
#include "LineMarker.h"
|
|
|
|
#include "Style.h"
|
|
|
|
#include "ViewStyle.h"
|
|
|
|
#include "CharClassify.h"
|
|
|
|
#include "Decoration.h"
|
|
|
|
#include "CaseFolder.h"
|
|
|
|
#include "Document.h"
|
|
|
|
|
2015-06-07 21:19:26 +00:00
|
|
|
using namespace Scintilla;
|
2022-01-04 23:07:50 +00:00
|
|
|
using namespace Scintilla::Internal;
|
2015-06-07 21:19:26 +00:00
|
|
|
|
2013-08-28 00:44:27 +00:00
|
|
|
class WatcherHelper : public DocWatcher {
|
2019-05-04 18:14:48 +00:00
|
|
|
ScintillaDocument *owner;
|
2013-08-28 00:44:27 +00:00
|
|
|
public:
|
2019-05-04 18:14:48 +00:00
|
|
|
explicit WatcherHelper(ScintillaDocument *owner_);
|
|
|
|
|
|
|
|
void NotifyModifyAttempt(Document *doc, void *userData) override;
|
|
|
|
void NotifySavePoint(Document *doc, void *userData, bool atSavePoint) override;
|
|
|
|
void NotifyModified(Document *doc, DocModification mh, void *userData) override;
|
|
|
|
void NotifyDeleted(Document *doc, void *userData) noexcept override;
|
2021-02-21 04:53:09 +00:00
|
|
|
void NotifyStyleNeeded(Document *doc, void *userData, Sci::Position endPos) override;
|
2019-05-04 18:14:48 +00:00
|
|
|
void NotifyLexerChanged(Document *doc, void *userData) override;
|
2022-01-04 23:07:50 +00:00
|
|
|
void NotifyErrorOccurred(Document *doc, void *userData, Status status) override;
|
2013-08-28 00:44:27 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
WatcherHelper::WatcherHelper(ScintillaDocument *owner_) : owner(owner_) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void WatcherHelper::NotifyModifyAttempt(Document *, void *) {
|
2019-05-04 18:14:48 +00:00
|
|
|
emit owner->modify_attempt();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WatcherHelper::NotifySavePoint(Document *, void *, bool atSavePoint) {
|
2019-05-04 18:14:48 +00:00
|
|
|
emit owner->save_point(atSavePoint);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WatcherHelper::NotifyModified(Document *, DocModification mh, void *) {
|
|
|
|
int length = mh.length;
|
|
|
|
if (!mh.text)
|
|
|
|
length = 0;
|
|
|
|
QByteArray ba = QByteArray::fromRawData(mh.text, length);
|
2022-01-04 23:07:50 +00:00
|
|
|
emit owner->modified(mh.position, static_cast<int>(mh.modificationType), ba, length,
|
|
|
|
mh.linesAdded, mh.line, static_cast<int>(mh.foldLevelNow), static_cast<int>(mh.foldLevelPrev));
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-04 18:14:48 +00:00
|
|
|
void WatcherHelper::NotifyDeleted(Document *, void *) noexcept {
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-04 18:14:48 +00:00
|
|
|
void WatcherHelper::NotifyStyleNeeded(Document *, void *, Sci::Position endPos) {
|
|
|
|
emit owner->style_needed(endPos);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WatcherHelper::NotifyLexerChanged(Document *, void *) {
|
2019-05-04 18:14:48 +00:00
|
|
|
emit owner->lexer_changed();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
void WatcherHelper::NotifyErrorOccurred(Document *, void *, Status status) {
|
|
|
|
emit owner->error_occurred(static_cast<int>(status));
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ScintillaDocument::ScintillaDocument(QObject *parent, void *pdoc_) :
|
2021-02-21 04:53:09 +00:00
|
|
|
QObject(parent), pdoc(pdoc_), docWatcher(nullptr) {
|
2013-08-28 00:44:27 +00:00
|
|
|
if (!pdoc) {
|
2022-01-04 23:07:50 +00:00
|
|
|
pdoc = new Document(DocumentOption::Default);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
docWatcher = new WatcherHelper(this);
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->AddRef();
|
|
|
|
(static_cast<Document *>(pdoc))->AddWatcher(docWatcher, pdoc);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ScintillaDocument::~ScintillaDocument() {
|
|
|
|
Document *doc = static_cast<Document *>(pdoc);
|
|
|
|
if (doc) {
|
|
|
|
doc->RemoveWatcher(docWatcher, doc);
|
|
|
|
doc->Release();
|
|
|
|
}
|
2021-02-21 04:53:09 +00:00
|
|
|
pdoc = nullptr;
|
2013-08-28 00:44:27 +00:00
|
|
|
delete docWatcher;
|
2021-02-21 04:53:09 +00:00
|
|
|
docWatcher = nullptr;
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void *ScintillaDocument::pointer() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return pdoc;
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::line_from_position(int pos) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->LineFromPosition(pos);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::is_cr_lf(int pos) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->IsCrLf(pos);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::delete_chars(int pos, int len) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->DeleteChars(pos, len);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::undo() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->Undo();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::redo() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->Redo();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::can_undo() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->CanUndo();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::can_redo() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->CanRedo();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::delete_undo_history() {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->DeleteUndoHistory();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::set_undo_collection(bool collect_undo) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->SetUndoCollection(collect_undo);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::is_collecting_undo() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->IsCollectingUndo();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::begin_undo_action() {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->BeginUndoAction();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::end_undo_action() {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->EndUndoAction();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::set_save_point() {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->SetSavePoint();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::is_save_point() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->IsSavePoint();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::set_read_only(bool read_only) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->SetReadOnly(read_only);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::is_read_only() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->IsReadOnly();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::insert_string(int position, QByteArray &str) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->InsertString(position, str.data(), str.size());
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray ScintillaDocument::get_char_range(int position, int length) {
|
2019-05-04 18:14:48 +00:00
|
|
|
Document *doc = static_cast<Document *>(pdoc);
|
2013-08-28 00:44:27 +00:00
|
|
|
|
|
|
|
if (position < 0 || length <= 0 || position + length > doc->Length())
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
QByteArray ba(length, '\0');
|
|
|
|
doc->GetCharRange(ba.data(), position, length);
|
|
|
|
return ba;
|
|
|
|
}
|
|
|
|
|
|
|
|
char ScintillaDocument::style_at(int position) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->StyleAt(position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::line_start(int lineno) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->LineStart(lineno);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::line_end(int lineno) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->LineEnd(lineno);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::line_end_position(int pos) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->LineEndPosition(pos);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::length() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->Length();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::lines_total() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->LinesTotal();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-04 18:14:48 +00:00
|
|
|
void ScintillaDocument::start_styling(int position) {
|
|
|
|
(static_cast<Document *>(pdoc))->StartStyling(position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScintillaDocument::set_style_for(int length, char style) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->SetStyleFor(length, style);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::get_end_styled() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->GetEndStyled();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::ensure_styled_to(int position) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->EnsureStyledTo(position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::set_current_indicator(int indic) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->decorations->SetCurrentIndicator(indic);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::decoration_fill_range(int position, int value, int fillLength) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->DecorationFillRange(position, value, fillLength);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::decorations_value_at(int indic, int position) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->decorations->ValueAt(indic, position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::decorations_start(int indic, int position) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->decorations->Start(indic, position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::decorations_end(int indic, int position) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->decorations->End(indic, position);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int ScintillaDocument::get_code_page() {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->CodePage();
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::set_code_page(int code_page) {
|
2019-05-04 18:14:48 +00:00
|
|
|
(static_cast<Document *>(pdoc))->dbcsCodePage = code_page;
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2015-06-07 21:19:26 +00:00
|
|
|
int ScintillaDocument::get_eol_mode() {
|
2022-01-04 23:07:50 +00:00
|
|
|
return static_cast<int>((static_cast<Document *>(pdoc))->eolMode);
|
2015-06-07 21:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScintillaDocument::set_eol_mode(int eol_mode) {
|
2022-01-04 23:07:50 +00:00
|
|
|
(static_cast<Document *>(pdoc))->eolMode = static_cast<EndOfLine>(eol_mode);
|
2015-06-07 21:19:26 +00:00
|
|
|
}
|
|
|
|
|
2013-08-28 00:44:27 +00:00
|
|
|
int ScintillaDocument::move_position_outside_char(int pos, int move_dir, bool check_line_end) {
|
2019-05-04 18:14:48 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->MovePositionOutsideChar(pos, move_dir, check_line_end);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
|
2015-06-07 21:19:26 +00:00
|
|
|
int ScintillaDocument::get_character(int pos) {
|
2021-02-21 04:53:09 +00:00
|
|
|
return (static_cast<Document *>(pdoc))->GetCharacterAndWidth(pos, nullptr);
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|