// Copyright 2019 Google LLC. // Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. #ifndef editor_DEFINED #define editor_DEFINED #include "experimental/editor/stringslice.h" #include "include/core/SkColor.h" #include "include/core/SkFont.h" #include "include/core/SkString.h" #include "include/core/SkTextBlob.h" #include #include #include class SkCanvas; class SkShaper; // TODO: modulize this; editor::Editor becomes SkEditor ? namespace editor { class Editor { struct TextLine; public: // used to load a new file. void setText(const char* text, size_t len); // total height in canvas display units. int getHeight() const { return fHeight; } // spacing around the text, in canvas display units. int getMargin() const { return fMargin; } // set display width in canvas display units void setWidth(int w); // may force re-shape // get/set current font (used for shaping and displaying text) const SkFont& font() const { return fFont; } void setFont(SkFont font); size_t lineCount() const { return fLines.size(); } const StringSlice& line(size_t i) const { return SkASSERT(i < fLines.size()), fLines[i].fText; } struct Text { const std::vector& fLines; struct Iterator { std::vector::const_iterator fPtr; const StringSlice& operator*() { return fPtr->fText; } void operator++() { ++fPtr; } bool operator!=(const Iterator& other) const { return fPtr != other.fPtr; } }; Iterator begin() const { return Iterator{fLines.begin()}; } Iterator end() const { return Iterator{fLines.end()}; } }; Text text() const { return Text{fLines}; } // get size of line in canvas display units. int lineHeight(size_t index) const { return fLines[index].fHeight; } struct TextPosition { size_t fTextByteIndex = SIZE_MAX; // index into UTF-8 representation of line. size_t fParagraphIndex = SIZE_MAX; // logical line, based on hard newline characters. }; enum class Movement { kNowhere, kLeft, kUp, kRight, kDown, kHome, kEnd, }; TextPosition move(Editor::Movement move, Editor::TextPosition pos) const; TextPosition getPosition(SkIPoint); SkRect getLocation(TextPosition); // insert into current text. TextPosition insert(TextPosition, const char* utf8Text, size_t byteLen); // remove text between two positions TextPosition remove(TextPosition, TextPosition); StringSlice copy(TextPosition, TextPosition) const; struct PaintOpts { SkColor4f fBackgroundColor = {1, 1, 1, 1}; SkColor4f fForegroundColor = {0, 0, 0, 1}; // TODO: maybe have multiple selections and cursors, each with separate colors. SkColor4f fSelectionColor = {0.729f, 0.827f, 0.988f, 1}; SkColor4f fCursorColor = {1, 0, 0, 1}; TextPosition fSelectionBegin; TextPosition fSelectionEnd; TextPosition fCursor; }; void paint(SkCanvas* canvas, PaintOpts); private: // TODO: rename this to TextParagraph. fLines to fParas. struct TextLine { StringSlice fText; sk_sp fBlob; std::vector fCursorPos; std::vector fLineEndOffsets; SkIPoint fOrigin = {0, 0}; int fHeight = 0; bool fShaped = false; TextLine(StringSlice t) : fText(std::move(t)) {} TextLine() {} }; std::vector fLines; int fMargin = 10; int fWidth = 0; int fHeight = 0; SkFont fFont; SkRect fSpaceBounds = {0, 0, 0, 0}; bool fNeedsReshape = false; static void Shape(TextLine*, SkShaper*, float width, const SkFont&, SkRect); void markDirty(TextLine*); void markAllDirty() { for (auto& l : fLines) { this->markDirty(&l); } } void reshapeAll(); }; } // namespace editor static inline bool operator==(const editor::Editor::TextPosition& u, const editor::Editor::TextPosition& v) { return u.fParagraphIndex == v.fParagraphIndex && u.fTextByteIndex == v.fTextByteIndex; } static inline bool operator!=(const editor::Editor::TextPosition& u, const editor::Editor::TextPosition& v) { return !(u == v); } static inline bool operator<(const editor::Editor::TextPosition& u, const editor::Editor::TextPosition& v) { return u.fParagraphIndex < v.fParagraphIndex || (u.fParagraphIndex == v.fParagraphIndex && u.fTextByteIndex < v.fTextByteIndex); } #endif // editor_DEFINED