|
|
@ -22,8 +22,8 @@ import 'box.dart'; |
|
|
|
import 'controller.dart'; |
|
|
|
import 'controller.dart'; |
|
|
|
import 'cursor.dart'; |
|
|
|
import 'cursor.dart'; |
|
|
|
import 'delegate.dart'; |
|
|
|
import 'delegate.dart'; |
|
|
|
import 'keyboard_listener.dart'; |
|
|
|
|
|
|
|
import 'editor.dart'; |
|
|
|
import 'editor.dart'; |
|
|
|
|
|
|
|
import 'keyboard_listener.dart'; |
|
|
|
|
|
|
|
|
|
|
|
class RawEditor extends StatefulWidget { |
|
|
|
class RawEditor extends StatefulWidget { |
|
|
|
final QuillController controller; |
|
|
|
final QuillController controller; |
|
|
@ -76,17 +76,17 @@ class RawEditor extends StatefulWidget { |
|
|
|
: assert(controller != null, 'controller cannot be null'), |
|
|
|
: assert(controller != null, 'controller cannot be null'), |
|
|
|
assert(focusNode != null, 'focusNode cannot be null'), |
|
|
|
assert(focusNode != null, 'focusNode cannot be null'), |
|
|
|
assert(scrollable || scrollController != null, |
|
|
|
assert(scrollable || scrollController != null, |
|
|
|
'scrollController cannot be null'), |
|
|
|
'scrollController cannot be null'), |
|
|
|
assert(selectionColor != null, 'selectionColor cannot be null'), |
|
|
|
assert(selectionColor != null, 'selectionColor cannot be null'), |
|
|
|
assert(enableInteractiveSelection != null, |
|
|
|
assert(enableInteractiveSelection != null, |
|
|
|
'enableInteractiveSelection cannot be null'), |
|
|
|
'enableInteractiveSelection cannot be null'), |
|
|
|
assert(showSelectionHandles != null, |
|
|
|
assert(showSelectionHandles != null, |
|
|
|
'showSelectionHandles cannot be null'), |
|
|
|
'showSelectionHandles cannot be null'), |
|
|
|
assert(readOnly != null, 'readOnly cannot be null'), |
|
|
|
assert(readOnly != null, 'readOnly cannot be null'), |
|
|
|
assert(maxHeight == null || maxHeight > 0, 'maxHeight cannot be null'), |
|
|
|
assert(maxHeight == null || maxHeight > 0, 'maxHeight cannot be null'), |
|
|
|
assert(minHeight == null || minHeight >= 0, 'minHeight cannot be null'), |
|
|
|
assert(minHeight == null || minHeight >= 0, 'minHeight cannot be null'), |
|
|
|
assert(maxHeight == null || minHeight == null || maxHeight >= minHeight, |
|
|
|
assert(maxHeight == null || minHeight == null || maxHeight >= minHeight, |
|
|
|
'maxHeight cannot be null'), |
|
|
|
'maxHeight cannot be null'), |
|
|
|
assert(autoFocus != null, 'autoFocus cannot be null'), |
|
|
|
assert(autoFocus != null, 'autoFocus cannot be null'), |
|
|
|
assert(toolbarOptions != null, 'toolbarOptions cannot be null'), |
|
|
|
assert(toolbarOptions != null, 'toolbarOptions cannot be null'), |
|
|
|
showCursor = showCursor ?? !readOnly, |
|
|
|
showCursor = showCursor ?? !readOnly, |
|
|
@ -134,11 +134,11 @@ class RawEditorState extends EditorState |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleCursorMovement( |
|
|
|
handleCursorMovement( |
|
|
|
LogicalKeyboardKey key, |
|
|
|
LogicalKeyboardKey key, |
|
|
|
bool wordModifier, |
|
|
|
bool wordModifier, |
|
|
|
bool lineModifier, |
|
|
|
bool lineModifier, |
|
|
|
bool shift, |
|
|
|
bool shift, |
|
|
|
) { |
|
|
|
) { |
|
|
|
if (wordModifier && lineModifier) { |
|
|
|
if (wordModifier && lineModifier) { |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
@ -202,7 +202,7 @@ class RawEditorState extends EditorState |
|
|
|
RenderEditableBox child = getRenderEditor().childAtPosition(originPosition); |
|
|
|
RenderEditableBox child = getRenderEditor().childAtPosition(originPosition); |
|
|
|
TextPosition localPosition = TextPosition( |
|
|
|
TextPosition localPosition = TextPosition( |
|
|
|
offset: |
|
|
|
offset: |
|
|
|
originPosition.offset - child.getContainer().getDocumentOffset()); |
|
|
|
originPosition.offset - child.getContainer().getDocumentOffset()); |
|
|
|
|
|
|
|
|
|
|
|
TextPosition position = upKey |
|
|
|
TextPosition position = upKey |
|
|
|
? child.getPositionAbove(localPosition) |
|
|
|
? child.getPositionAbove(localPosition) |
|
|
@ -219,10 +219,10 @@ class RawEditorState extends EditorState |
|
|
|
child.getOffsetForCaret(localPosition).dx, |
|
|
|
child.getOffsetForCaret(localPosition).dx, |
|
|
|
sibling |
|
|
|
sibling |
|
|
|
.getOffsetForCaret(TextPosition( |
|
|
|
.getOffsetForCaret(TextPosition( |
|
|
|
offset: upKey ? sibling.getContainer().length - 1 : 0)) |
|
|
|
offset: upKey ? sibling.getContainer().length - 1 : 0)) |
|
|
|
.dy); |
|
|
|
.dy); |
|
|
|
TextPosition siblingPosition = |
|
|
|
TextPosition siblingPosition = |
|
|
|
sibling.getPositionForOffset(finalOffset); |
|
|
|
sibling.getPositionForOffset(finalOffset); |
|
|
|
position = TextPosition( |
|
|
|
position = TextPosition( |
|
|
|
offset: sibling.getContainer().getDocumentOffset() + |
|
|
|
offset: sibling.getContainer().getDocumentOffset() + |
|
|
|
siblingPosition.offset); |
|
|
|
siblingPosition.offset); |
|
|
@ -271,7 +271,7 @@ class RawEditorState extends EditorState |
|
|
|
TextSelection textSelection = getRenderEditor().selectWordAtPosition( |
|
|
|
TextSelection textSelection = getRenderEditor().selectWordAtPosition( |
|
|
|
TextPosition( |
|
|
|
TextPosition( |
|
|
|
offset: |
|
|
|
offset: |
|
|
|
_nextCharacter(newSelection.extentOffset, plainText, false))); |
|
|
|
_nextCharacter(newSelection.extentOffset, plainText, false))); |
|
|
|
return newSelection.copyWith(extentOffset: textSelection.extentOffset); |
|
|
|
return newSelection.copyWith(extentOffset: textSelection.extentOffset); |
|
|
|
} else if (lineModifier) { |
|
|
|
} else if (lineModifier) { |
|
|
|
if (leftKey) { |
|
|
|
if (leftKey) { |
|
|
@ -292,7 +292,7 @@ class RawEditorState extends EditorState |
|
|
|
|
|
|
|
|
|
|
|
if (rightKey && newSelection.extentOffset < plainText.length) { |
|
|
|
if (rightKey && newSelection.extentOffset < plainText.length) { |
|
|
|
int nextExtent = |
|
|
|
int nextExtent = |
|
|
|
_nextCharacter(newSelection.extentOffset, plainText, true); |
|
|
|
_nextCharacter(newSelection.extentOffset, plainText, true); |
|
|
|
int distance = nextExtent - newSelection.extentOffset; |
|
|
|
int distance = nextExtent - newSelection.extentOffset; |
|
|
|
newSelection = newSelection.copyWith(extentOffset: nextExtent); |
|
|
|
newSelection = newSelection.copyWith(extentOffset: nextExtent); |
|
|
|
if (shift) { |
|
|
|
if (shift) { |
|
|
@ -303,7 +303,7 @@ class RawEditorState extends EditorState |
|
|
|
|
|
|
|
|
|
|
|
if (leftKey && newSelection.extentOffset > 0) { |
|
|
|
if (leftKey && newSelection.extentOffset > 0) { |
|
|
|
int previousExtent = |
|
|
|
int previousExtent = |
|
|
|
_previousCharacter(newSelection.extentOffset, plainText, true); |
|
|
|
_previousCharacter(newSelection.extentOffset, plainText, true); |
|
|
|
int distance = newSelection.extentOffset - previousExtent; |
|
|
|
int distance = newSelection.extentOffset - previousExtent; |
|
|
|
newSelection = newSelection.copyWith(extentOffset: previousExtent); |
|
|
|
newSelection = newSelection.copyWith(extentOffset: previousExtent); |
|
|
|
if (shift) { |
|
|
|
if (shift) { |
|
|
@ -521,7 +521,7 @@ class RawEditorState extends EditorState |
|
|
|
|
|
|
|
|
|
|
|
if (widget.scrollable) { |
|
|
|
if (widget.scrollable) { |
|
|
|
EdgeInsets baselinePadding = |
|
|
|
EdgeInsets baselinePadding = |
|
|
|
EdgeInsets.only(top: _styles.paragraph.verticalSpacing.item1); |
|
|
|
EdgeInsets.only(top: _styles.paragraph.verticalSpacing.item1); |
|
|
|
child = BaselineProxy( |
|
|
|
child = BaselineProxy( |
|
|
|
textStyle: _styles.paragraph.style, |
|
|
|
textStyle: _styles.paragraph.style, |
|
|
|
padding: baselinePadding, |
|
|
|
padding: baselinePadding, |
|
|
@ -536,8 +536,8 @@ class RawEditorState extends EditorState |
|
|
|
BoxConstraints constraints = widget.expands |
|
|
|
BoxConstraints constraints = widget.expands |
|
|
|
? BoxConstraints.expand() |
|
|
|
? BoxConstraints.expand() |
|
|
|
: BoxConstraints( |
|
|
|
: BoxConstraints( |
|
|
|
minHeight: widget.minHeight ?? 0.0, |
|
|
|
minHeight: widget.minHeight ?? 0.0, |
|
|
|
maxHeight: widget.maxHeight ?? double.infinity); |
|
|
|
maxHeight: widget.maxHeight ?? double.infinity); |
|
|
|
|
|
|
|
|
|
|
|
return QuillStyles( |
|
|
|
return QuillStyles( |
|
|
|
data: _styles, |
|
|
|
data: _styles, |
|
|
@ -740,7 +740,7 @@ class RawEditorState extends EditorState |
|
|
|
if (selection.isCollapsed) { |
|
|
|
if (selection.isCollapsed) { |
|
|
|
if (!forward && textBefore.isNotEmpty) { |
|
|
|
if (!forward && textBefore.isNotEmpty) { |
|
|
|
final int characterBoundary = |
|
|
|
final int characterBoundary = |
|
|
|
_previousCharacter(textBefore.length, textBefore, true); |
|
|
|
_previousCharacter(textBefore.length, textBefore, true); |
|
|
|
textBefore = textBefore.substring(0, characterBoundary); |
|
|
|
textBefore = textBefore.substring(0, characterBoundary); |
|
|
|
cursorPosition = characterBoundary; |
|
|
|
cursorPosition = characterBoundary; |
|
|
|
} |
|
|
|
} |
|
|
@ -750,7 +750,7 @@ class RawEditorState extends EditorState |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
TextSelection newSelection = |
|
|
|
TextSelection newSelection = |
|
|
|
TextSelection.collapsed(offset: cursorPosition); |
|
|
|
TextSelection.collapsed(offset: cursorPosition); |
|
|
|
String newText = textBefore + textAfter; |
|
|
|
String newText = textBefore + textAfter; |
|
|
|
int size = plainText.length - newText.length; |
|
|
|
int size = plainText.length - newText.length; |
|
|
|
widget.controller.replaceText( |
|
|
|
widget.controller.replaceText( |
|
|
@ -785,7 +785,7 @@ class RawEditorState extends EditorState |
|
|
|
|
|
|
|
|
|
|
|
textEditingValue = TextEditingValue( |
|
|
|
textEditingValue = TextEditingValue( |
|
|
|
text: |
|
|
|
text: |
|
|
|
selection.textBefore(plainText) + selection.textAfter(plainText), |
|
|
|
selection.textBefore(plainText) + selection.textAfter(plainText), |
|
|
|
selection: TextSelection.collapsed(offset: selection.start), |
|
|
|
selection: TextSelection.collapsed(offset: selection.start), |
|
|
|
); |
|
|
|
); |
|
|
|
} |
|
|
|
} |
|
|
@ -847,7 +847,11 @@ class RawEditorState extends EditorState |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
SchedulerBinding.instance.addPostFrameCallback( |
|
|
|
SchedulerBinding.instance.addPostFrameCallback( |
|
|
|
(Duration _) => _updateOrDisposeSelectionOverlayIfNeeded()); |
|
|
|
(Duration _) => _updateOrDisposeSelectionOverlayIfNeeded()); |
|
|
|
|
|
|
|
setState(() { |
|
|
|
|
|
|
|
// Use widget.controller.value in build() |
|
|
|
|
|
|
|
// Trigger build and updateChildren |
|
|
|
|
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
_updateOrDisposeSelectionOverlayIfNeeded() { |
|
|
|
_updateOrDisposeSelectionOverlayIfNeeded() { |
|
|
@ -897,7 +901,12 @@ class RawEditorState extends EditorState |
|
|
|
updateKeepAlive(); |
|
|
|
updateKeepAlive(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
_onChangedClipboardStatus() {} |
|
|
|
_onChangedClipboardStatus() { |
|
|
|
|
|
|
|
setState(() { |
|
|
|
|
|
|
|
// Inform the widget that the value of clipboardStatus has changed. |
|
|
|
|
|
|
|
// Trigger build and updateChildren |
|
|
|
|
|
|
|
}); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
bool _showCaretOnScreenScheduled = false; |
|
|
|
bool _showCaretOnScreenScheduled = false; |
|
|
|
|
|
|
|
|
|
|
@ -913,7 +922,7 @@ class RawEditorState extends EditorState |
|
|
|
final viewport = RenderAbstractViewport.of(getRenderEditor()); |
|
|
|
final viewport = RenderAbstractViewport.of(getRenderEditor()); |
|
|
|
assert(viewport != null); |
|
|
|
assert(viewport != null); |
|
|
|
final editorOffset = |
|
|
|
final editorOffset = |
|
|
|
getRenderEditor().localToGlobal(Offset(0.0, 0.0), ancestor: viewport); |
|
|
|
getRenderEditor().localToGlobal(Offset(0.0, 0.0), ancestor: viewport); |
|
|
|
final offsetInViewport = _scrollController.offset + editorOffset.dy; |
|
|
|
final offsetInViewport = _scrollController.offset + editorOffset.dy; |
|
|
|
|
|
|
|
|
|
|
|
final offset = getRenderEditor().getOffsetToRevealCursor( |
|
|
|
final offset = getRenderEditor().getOffsetToRevealCursor( |
|
|
@ -1053,4 +1062,4 @@ class _Editor extends MultiChildRenderObjectWidget { |
|
|
|
renderObject.onSelectionChanged = onSelectionChanged; |
|
|
|
renderObject.onSelectionChanged = onSelectionChanged; |
|
|
|
renderObject.setPadding(padding); |
|
|
|
renderObject.setPadding(padding); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|