您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
454 行
16 KiB
454 行
16 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.gestures;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.service;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
using Canvas = Unity.UIWidgets.ui.Canvas;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
|
|
namespace Unity.UIWidgets.rendering {
|
|
public enum TextOverflow {
|
|
/// Clip the overflowing text to fix its container.
|
|
clip,
|
|
|
|
/// Fade the overflowing text to transparent.
|
|
fade,
|
|
|
|
/// Use an ellipsis to indicate that the text has overflowed.
|
|
ellipsis,
|
|
}
|
|
|
|
|
|
public class RenderParagraph : RenderBox {
|
|
static readonly string _kEllipsis = "\u2026";
|
|
|
|
bool _softWrap;
|
|
|
|
TextOverflow _overflow;
|
|
readonly TextPainter _textPainter;
|
|
bool _hasVisualOverflow = false;
|
|
|
|
List<TextBox> _selectionRects;
|
|
|
|
public RenderParagraph(TextSpan text,
|
|
TextAlign textAlign = TextAlign.left,
|
|
TextDirection textDirection = TextDirection.ltr,
|
|
bool softWrap = true,
|
|
TextOverflow overflow = TextOverflow.clip,
|
|
float textScaleFactor = 1.0f,
|
|
int? maxLines = null,
|
|
Action onSelectionChanged = null,
|
|
Color selectionColor = null
|
|
) {
|
|
D.assert(maxLines == null || maxLines > 0);
|
|
this._softWrap = softWrap;
|
|
this._overflow = overflow;
|
|
this._textPainter = new TextPainter(
|
|
text,
|
|
textAlign,
|
|
textDirection,
|
|
textScaleFactor,
|
|
maxLines,
|
|
overflow == TextOverflow.ellipsis ? _kEllipsis : ""
|
|
);
|
|
|
|
this._selection = null;
|
|
this.onSelectionChanged = onSelectionChanged;
|
|
this.selectionColor = selectionColor;
|
|
|
|
this._resetHoverHandler();
|
|
}
|
|
|
|
public Action onSelectionChanged;
|
|
public Color selectionColor;
|
|
|
|
public TextSelection selection {
|
|
get { return this._selection; }
|
|
set {
|
|
if (this._selection == value) {
|
|
return;
|
|
}
|
|
|
|
this._selection = value;
|
|
this._selectionRects = null;
|
|
this.markNeedsPaint();
|
|
}
|
|
}
|
|
|
|
public TextSpan text {
|
|
get { return this._textPainter.text; }
|
|
|
|
set {
|
|
Debug.Assert(value != null);
|
|
switch (this._textPainter.text.compareTo(value)) {
|
|
case RenderComparison.identical:
|
|
case RenderComparison.metadata:
|
|
return;
|
|
case RenderComparison.function:
|
|
this._textPainter.text = value;
|
|
break;
|
|
case RenderComparison.paint:
|
|
this._textPainter.text = value;
|
|
this.markNeedsPaint();
|
|
break;
|
|
case RenderComparison.layout:
|
|
this._textPainter.text = value;
|
|
this.markNeedsLayout();
|
|
break;
|
|
}
|
|
|
|
this._resetHoverHandler();
|
|
}
|
|
}
|
|
|
|
public TextAlign textAlign {
|
|
get { return this._textPainter.textAlign; }
|
|
set {
|
|
if (this._textPainter.textAlign == value) {
|
|
return;
|
|
}
|
|
|
|
this._textPainter.textAlign = value;
|
|
this.markNeedsPaint();
|
|
}
|
|
}
|
|
|
|
public TextDirection? textDirection {
|
|
get { return this._textPainter.textDirection; }
|
|
set {
|
|
if (this._textPainter.textDirection == value) {
|
|
return;
|
|
}
|
|
|
|
this._textPainter.textDirection = this.textDirection;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
protected Offset getOffsetForCaret(TextPosition position, Rect caretPrototype) {
|
|
D.assert(this._textPainter != null);
|
|
return this._textPainter.getOffsetForCaret(position, caretPrototype);
|
|
}
|
|
|
|
public bool softWrap {
|
|
get { return this._softWrap; }
|
|
set {
|
|
if (this._softWrap == value) {
|
|
return;
|
|
}
|
|
|
|
this._softWrap = value;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
public TextOverflow overflow {
|
|
get { return this._overflow; }
|
|
set {
|
|
if (this._overflow == value) {
|
|
return;
|
|
}
|
|
|
|
this._overflow = value;
|
|
this._textPainter.ellipsis = value == TextOverflow.ellipsis ? _kEllipsis : null;
|
|
// _textPainter.e
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
public float textScaleFactor {
|
|
get { return this._textPainter.textScaleFactor; }
|
|
set {
|
|
if (Mathf.Abs(this._textPainter.textScaleFactor - value) < 0.00000001) {
|
|
return;
|
|
}
|
|
|
|
this._textPainter.textScaleFactor = value;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
public int? maxLines {
|
|
get { return this._textPainter.maxLines; }
|
|
set {
|
|
D.assert(this.maxLines == null || this.maxLines > 0);
|
|
if (this._textPainter.maxLines == value) {
|
|
return;
|
|
}
|
|
|
|
this._textPainter.maxLines = value;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
public Size textSize {
|
|
get { return this._textPainter.size; }
|
|
}
|
|
|
|
protected override float computeMinIntrinsicWidth(float height) {
|
|
this._layoutText();
|
|
return this._textPainter.minIntrinsicWidth;
|
|
}
|
|
|
|
protected override float computeMaxIntrinsicWidth(float height) {
|
|
this._layoutText();
|
|
return this._textPainter.maxIntrinsicWidth;
|
|
}
|
|
|
|
float _computeIntrinsicHeight(float width) {
|
|
this._layoutText(minWidth: width, maxWidth: width);
|
|
return this._textPainter.height;
|
|
}
|
|
|
|
protected override float computeMinIntrinsicHeight(float width) {
|
|
return this._computeIntrinsicHeight(width);
|
|
}
|
|
|
|
protected override float computeMaxIntrinsicHeight(float width) {
|
|
return this._computeIntrinsicHeight(width);
|
|
}
|
|
|
|
protected override float? computeDistanceToActualBaseline(TextBaseline baseline) {
|
|
this._layoutTextWithConstraints(this.constraints);
|
|
return this._textPainter.computeDistanceToActualBaseline(baseline);
|
|
}
|
|
|
|
|
|
protected override bool hitTestSelf(Offset position) {
|
|
return true;
|
|
}
|
|
|
|
bool _hasFocus = false;
|
|
bool _listenerAttached = false;
|
|
|
|
public bool hasFocus {
|
|
get { return this._hasFocus; }
|
|
set {
|
|
if (this._hasFocus == value) {
|
|
return;
|
|
}
|
|
|
|
this._hasFocus = value;
|
|
|
|
if (this._hasFocus) {
|
|
D.assert(!this._listenerAttached);
|
|
RawKeyboard.instance.addListener(this._handleKeyEvent);
|
|
this._listenerAttached = true;
|
|
}
|
|
else {
|
|
this.selection = null;
|
|
D.assert(this._listenerAttached);
|
|
RawKeyboard.instance.removeListener(this._handleKeyEvent);
|
|
this._listenerAttached = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
TextSpan _previousHoverSpan;
|
|
bool _pointerHoverInside;
|
|
bool _hasHoverRecognizer;
|
|
|
|
void _resetHoverHandler() {
|
|
this._hasHoverRecognizer = this._textPainter.text.hasHoverRecognizer;
|
|
this._previousHoverSpan = null;
|
|
this._pointerHoverInside = false;
|
|
}
|
|
|
|
void _handleKeyEvent(RawKeyEvent keyEvent) {
|
|
//only allow KCommand.copy
|
|
if (keyEvent is RawKeyUpEvent) {
|
|
return;
|
|
}
|
|
|
|
if (this.selection.isCollapsed) {
|
|
return;
|
|
}
|
|
|
|
KeyCode pressedKeyCode = keyEvent.data.unityEvent.keyCode;
|
|
int modifiers = (int) keyEvent.data.unityEvent.modifiers;
|
|
bool ctrl = (modifiers & (int) EventModifiers.Control) > 0;
|
|
bool cmd = (modifiers & (int) EventModifiers.Command) > 0;
|
|
bool cKey = pressedKeyCode == KeyCode.C;
|
|
bool isMac = SystemInfo.operatingSystemFamily == OperatingSystemFamily.MacOSX;
|
|
|
|
KeyCommand? kcmd = keyEvent is RawKeyCommandEvent
|
|
? ((RawKeyCommandEvent) keyEvent).command
|
|
: ((ctrl || (isMac && cmd)) && cKey)
|
|
? KeyCommand.Copy
|
|
: (KeyCommand?) null;
|
|
|
|
if (kcmd == KeyCommand.Copy) {
|
|
Clipboard.setData(
|
|
new ClipboardData(text: this.selection.textInside(this.text.toPlainText()))
|
|
);
|
|
}
|
|
}
|
|
|
|
public override void detach() {
|
|
if (this._listenerAttached) {
|
|
RawKeyboard.instance.removeListener(this._handleKeyEvent);
|
|
}
|
|
|
|
base.detach();
|
|
}
|
|
|
|
TextSelection _selection;
|
|
|
|
public void selectPositionAt(Offset from = null, Offset to = null, SelectionChangedCause? cause = null) {
|
|
D.assert(cause != null);
|
|
D.assert(from != null);
|
|
if (true) {
|
|
TextPosition fromPosition =
|
|
this._textPainter.getPositionForOffset(this.globalToLocal(from));
|
|
TextPosition toPosition = to == null
|
|
? null
|
|
: this._textPainter.getPositionForOffset(this.globalToLocal(to));
|
|
|
|
int baseOffset = fromPosition.offset;
|
|
int extentOffset = fromPosition.offset;
|
|
if (toPosition != null) {
|
|
baseOffset = Math.Min(fromPosition.offset, toPosition.offset);
|
|
extentOffset = Math.Max(fromPosition.offset, toPosition.offset);
|
|
}
|
|
|
|
TextSelection newSelection = new TextSelection(
|
|
baseOffset: baseOffset,
|
|
extentOffset: extentOffset,
|
|
affinity: fromPosition.affinity);
|
|
|
|
if (newSelection != this._selection) {
|
|
this._handleSelectionChanged(newSelection, cause.Value);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void _handleSelectionChanged(TextSelection selection,
|
|
SelectionChangedCause cause) {
|
|
this.selection = selection;
|
|
this.onSelectionChanged?.Invoke();
|
|
}
|
|
|
|
|
|
void _handlePointerHover(PointerEvent evt) {
|
|
if (!this._hasHoverRecognizer) {
|
|
return;
|
|
}
|
|
|
|
if (evt is PointerEnterEvent) {
|
|
this._pointerHoverInside = true;
|
|
}
|
|
else if (evt is PointerLeaveEvent) {
|
|
this._pointerHoverInside = false;
|
|
this._previousHoverSpan?.hoverRecognizer?.OnPointerLeave?.Invoke();
|
|
this._previousHoverSpan = null;
|
|
}
|
|
else if (evt is PointerHoverEvent && this._pointerHoverInside) {
|
|
this._layoutTextWithConstraints(this.constraints);
|
|
Offset offset = this.globalToLocal(evt.position);
|
|
TextPosition position = this._textPainter.getPositionForOffset(offset);
|
|
TextSpan span = this._textPainter.text.getSpanForPosition(position);
|
|
|
|
if (this._previousHoverSpan != span) {
|
|
this._previousHoverSpan?.hoverRecognizer?.OnPointerLeave?.Invoke();
|
|
span?.hoverRecognizer?.OnPointerEnter?.Invoke((PointerHoverEvent) evt);
|
|
this._previousHoverSpan = span;
|
|
}
|
|
}
|
|
}
|
|
|
|
public override void handleEvent(PointerEvent evt, HitTestEntry entry) {
|
|
D.assert(this.debugHandleEvent(evt, entry));
|
|
if (evt is PointerDownEvent) {
|
|
this._layoutTextWithConstraints(this.constraints);
|
|
Offset offset = ((BoxHitTestEntry) entry).localPosition;
|
|
TextPosition position = this._textPainter.getPositionForOffset(offset);
|
|
TextSpan span = this._textPainter.text.getSpanForPosition(position);
|
|
span?.recognizer?.addPointer((PointerDownEvent) evt);
|
|
return;
|
|
}
|
|
|
|
this._handlePointerHover(evt);
|
|
}
|
|
|
|
protected override void performLayout() {
|
|
this._layoutTextWithConstraints(this.constraints);
|
|
var textSize = this._textPainter.size;
|
|
var didOverflowHeight = this._textPainter.didExceedMaxLines;
|
|
this.size = this.constraints.constrain(textSize);
|
|
|
|
var didOverflowWidth = this.size.width < textSize.width;
|
|
this._hasVisualOverflow = didOverflowWidth || didOverflowHeight;
|
|
|
|
this._selectionRects = null;
|
|
}
|
|
|
|
public override void paint(PaintingContext context, Offset offset) {
|
|
this._layoutTextWithConstraints(this.constraints);
|
|
var canvas = context.canvas;
|
|
|
|
if (this._hasVisualOverflow) {
|
|
var bounds = offset & this.size;
|
|
canvas.save();
|
|
canvas.clipRect(bounds);
|
|
}
|
|
|
|
if (this._selection != null && this.selectionColor != null && this._selection.isValid) {
|
|
if (!this._selection.isCollapsed) {
|
|
this._selectionRects =
|
|
this._selectionRects ?? this._textPainter.getBoxesForSelection(this._selection);
|
|
this._paintSelection(canvas, offset);
|
|
}
|
|
}
|
|
|
|
this._textPainter.paint(canvas, offset);
|
|
if (this._hasVisualOverflow) {
|
|
canvas.restore();
|
|
}
|
|
}
|
|
|
|
|
|
void _paintSelection(Canvas canvas, Offset effectiveOffset) {
|
|
D.assert(this._selectionRects != null);
|
|
D.assert(this.selectionColor != null);
|
|
var paint = new Paint {color = this.selectionColor};
|
|
|
|
Path barPath = new Path();
|
|
foreach (var box in this._selectionRects) {
|
|
barPath.addRect(box.toRect().shift(effectiveOffset));
|
|
}
|
|
|
|
canvas.drawPath(barPath, paint);
|
|
}
|
|
|
|
void _layoutText(float minWidth = 0.0f, float maxWidth = float.PositiveInfinity) {
|
|
var widthMatters = this.softWrap || this.overflow == TextOverflow.ellipsis;
|
|
this._textPainter.layout(minWidth, widthMatters ? maxWidth : float.PositiveInfinity);
|
|
}
|
|
|
|
void _layoutTextWithConstraints(BoxConstraints constraints) {
|
|
this._layoutText(minWidth: constraints.minWidth, maxWidth: constraints.maxWidth);
|
|
}
|
|
|
|
public override List<DiagnosticsNode> debugDescribeChildren() {
|
|
return new List<DiagnosticsNode> {
|
|
this.text.toDiagnosticsNode(name: "text", style: DiagnosticsTreeStyle.transition)
|
|
};
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new EnumProperty<TextAlign>("textAlign", this.textAlign));
|
|
properties.add(new EnumProperty<TextDirection?>("textDirection", this.textDirection));
|
|
properties.add(new FlagProperty("softWrap", value: this.softWrap, ifTrue: "wrapping at box width",
|
|
ifFalse: "no wrapping except at line break characters", showName: true));
|
|
properties.add(new EnumProperty<TextOverflow>("overflow", this.overflow));
|
|
properties.add(new FloatProperty("textScaleFactor", this.textScaleFactor, defaultValue: 1.0f));
|
|
properties.add(new IntProperty("maxLines", this.maxLines, ifNull: "unlimited"));
|
|
}
|
|
}
|
|
}
|