Unverified Commit e1174eb0 authored by Ian Hickson's avatar Ian Hickson Committed by GitHub

TextPainter RTL (#12791)

parent 19227a99
This diff is collapsed.
...@@ -22,9 +22,9 @@ ui.Picture paint(ui.Rect paintBounds) { ...@@ -22,9 +22,9 @@ ui.Picture paint(ui.Rect paintBounds) {
canvas.drawRect(new ui.Rect.fromLTRB(-100.0, -100.0, 100.0, 100.0), canvas.drawRect(new ui.Rect.fromLTRB(-100.0, -100.0, 100.0, 100.0),
new ui.Paint()..color = const ui.Color.fromARGB(255, 0, 255, 0)); new ui.Paint()..color = const ui.Color.fromARGB(255, 0, 255, 0));
// The paint method of Pargraph draws the contents of the paragraph unto the // The paint method of Paragraph draws the contents of the paragraph onto the
// given canvas. // given canvas.
canvas.drawParagraph(paragraph, new ui.Offset(paragraph.width / -2.0, (paragraph.width / 2.0) - 125)); canvas.drawParagraph(paragraph, new ui.Offset(-paragraph.width / 2.0, (paragraph.width / 2.0) - 125.0));
return recorder.endRecording(); return recorder.endRecording();
} }
......
...@@ -1013,7 +1013,7 @@ class MessageProperty extends DiagnosticsProperty<Null> { ...@@ -1013,7 +1013,7 @@ class MessageProperty extends DiagnosticsProperty<Null> {
/// ///
/// The [name], `message`, and [level] arguments must not be null. /// The [name], `message`, and [level] arguments must not be null.
MessageProperty(String name, String message, { MessageProperty(String name, String message, {
DiagnosticLevel level : DiagnosticLevel.info, DiagnosticLevel level: DiagnosticLevel.info,
}) : assert(name != null), }) : assert(name != null),
assert(message != null), assert(message != null),
assert(level != null), assert(level != null),
...@@ -1032,11 +1032,12 @@ class StringProperty extends DiagnosticsProperty<String> { ...@@ -1032,11 +1032,12 @@ class StringProperty extends DiagnosticsProperty<String> {
/// The [showName], [quoted], and [level] arguments must not be null. /// The [showName], [quoted], and [level] arguments must not be null.
StringProperty(String name, String value, { StringProperty(String name, String value, {
String description, String description,
String tooltip,
bool showName: true, bool showName: true,
Object defaultValue: kNoDefaultValue, Object defaultValue: kNoDefaultValue,
this.quoted: true, this.quoted: true,
String ifEmpty, String ifEmpty,
DiagnosticLevel level : DiagnosticLevel. info, DiagnosticLevel level: DiagnosticLevel.info,
}) : assert(showName != null), }) : assert(showName != null),
assert(quoted != null), assert(quoted != null),
assert(level != null), assert(level != null),
...@@ -1045,12 +1046,13 @@ class StringProperty extends DiagnosticsProperty<String> { ...@@ -1045,12 +1046,13 @@ class StringProperty extends DiagnosticsProperty<String> {
value, value,
description: description, description: description,
defaultValue: defaultValue, defaultValue: defaultValue,
tooltip: tooltip,
showName: showName, showName: showName,
ifEmpty: ifEmpty, ifEmpty: ifEmpty,
level: level, level: level,
); );
/// Whether the description is enclosed in double quotes. /// Whether the value is enclosed in double quotes.
final bool quoted; final bool quoted;
@override @override
...@@ -1144,7 +1146,7 @@ class DoubleProperty extends _NumProperty<double> { ...@@ -1144,7 +1146,7 @@ class DoubleProperty extends _NumProperty<double> {
String unit, String unit,
String tooltip, String tooltip,
Object defaultValue: kNoDefaultValue, Object defaultValue: kNoDefaultValue,
bool showName : true, bool showName: true,
DiagnosticLevel level: DiagnosticLevel.info, DiagnosticLevel level: DiagnosticLevel.info,
}) : assert(showName != null), }) : assert(showName != null),
assert(level != null), assert(level != null),
......
...@@ -28,11 +28,14 @@ export 'dart:ui' show ...@@ -28,11 +28,14 @@ export 'dart:ui' show
Size, Size,
StrokeCap, StrokeCap,
StrokeJoin, StrokeJoin,
TextAffinity,
TextAlign, TextAlign,
TextBaseline, TextBaseline,
TextBox,
TextDecoration, TextDecoration,
TextDecorationStyle, TextDecorationStyle,
TextDirection, TextDirection,
TextPosition,
TileMode, TileMode,
VertexMode, VertexMode,
VoidCallback, VoidCallback,
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:ui' as ui show Paragraph, ParagraphBuilder, ParagraphConstraints, ParagraphStyle, TextBox; import 'dart:ui' as ui show Paragraph, ParagraphBuilder, ParagraphConstraints, ParagraphStyle;
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/gestures.dart'; import 'package:flutter/gestures.dart';
...@@ -11,6 +11,8 @@ import 'package:flutter/services.dart'; ...@@ -11,6 +11,8 @@ import 'package:flutter/services.dart';
import 'basic_types.dart'; import 'basic_types.dart';
import 'text_span.dart'; import 'text_span.dart';
export 'package:flutter/services.dart' show TextRange, TextSelection;
final String _kZeroWidthSpace = new String.fromCharCode(0x200B); final String _kZeroWidthSpace = new String.fromCharCode(0x200B);
/// An object that paints a [TextSpan] tree into a [Canvas]. /// An object that paints a [TextSpan] tree into a [Canvas].
...@@ -289,13 +291,13 @@ class TextPainter { ...@@ -289,13 +291,13 @@ class TextPainter {
/// Valid only after [layout] has been called. /// Valid only after [layout] has been called.
double computeDistanceToActualBaseline(TextBaseline baseline) { double computeDistanceToActualBaseline(TextBaseline baseline) {
assert(!_needsLayout); assert(!_needsLayout);
assert(baseline != null);
switch (baseline) { switch (baseline) {
case TextBaseline.alphabetic: case TextBaseline.alphabetic:
return _paragraph.alphabeticBaseline; return _paragraph.alphabeticBaseline;
case TextBaseline.ideographic: case TextBaseline.ideographic:
return _paragraph.ideographicBaseline; return _paragraph.ideographicBaseline;
} }
assert(baseline != null);
return null; return null;
} }
...@@ -381,10 +383,10 @@ class TextPainter { ...@@ -381,10 +383,10 @@ class TextPainter {
if (prevCodeUnit == null) if (prevCodeUnit == null)
return null; return null;
final int prevRuneOffset = _isUtf16Surrogate(prevCodeUnit) ? offset - 2 : offset - 1; final int prevRuneOffset = _isUtf16Surrogate(prevCodeUnit) ? offset - 2 : offset - 1;
final List<ui.TextBox> boxes = _paragraph.getBoxesForRange(prevRuneOffset, offset); final List<TextBox> boxes = _paragraph.getBoxesForRange(prevRuneOffset, offset);
if (boxes.isEmpty) if (boxes.isEmpty)
return null; return null;
final ui.TextBox box = boxes[0]; final TextBox box = boxes[0];
final double caretEnd = box.end; final double caretEnd = box.end;
final double dx = box.direction == TextDirection.rtl ? caretEnd : caretEnd - caretPrototype.width; final double dx = box.direction == TextDirection.rtl ? caretEnd : caretEnd - caretPrototype.width;
return new Offset(dx, box.top); return new Offset(dx, box.top);
...@@ -395,10 +397,10 @@ class TextPainter { ...@@ -395,10 +397,10 @@ class TextPainter {
if (nextCodeUnit == null) if (nextCodeUnit == null)
return null; return null;
final int nextRuneOffset = _isUtf16Surrogate(nextCodeUnit) ? offset + 2 : offset + 1; final int nextRuneOffset = _isUtf16Surrogate(nextCodeUnit) ? offset + 2 : offset + 1;
final List<ui.TextBox> boxes = _paragraph.getBoxesForRange(offset, nextRuneOffset); final List<TextBox> boxes = _paragraph.getBoxesForRange(offset, nextRuneOffset);
if (boxes.isEmpty) if (boxes.isEmpty)
return null; return null;
final ui.TextBox box = boxes[0]; final TextBox box = boxes[0];
final double caretStart = box.start; final double caretStart = box.start;
final double dx = box.direction == TextDirection.rtl ? caretStart - caretPrototype.width : caretStart; final double dx = box.direction == TextDirection.rtl ? caretStart - caretPrototype.width : caretStart;
return new Offset(dx, box.top); return new Offset(dx, box.top);
...@@ -443,6 +445,7 @@ class TextPainter { ...@@ -443,6 +445,7 @@ class TextPainter {
Offset getOffsetForCaret(TextPosition position, Rect caretPrototype) { Offset getOffsetForCaret(TextPosition position, Rect caretPrototype) {
assert(!_needsLayout); assert(!_needsLayout);
final int offset = position.offset; final int offset = position.offset;
assert(position.affinity != null);
switch (position.affinity) { switch (position.affinity) {
case TextAffinity.upstream: case TextAffinity.upstream:
return _getOffsetFromUpstream(offset, caretPrototype) return _getOffsetFromUpstream(offset, caretPrototype)
...@@ -453,7 +456,6 @@ class TextPainter { ...@@ -453,7 +456,6 @@ class TextPainter {
?? _getOffsetFromUpstream(offset, caretPrototype) ?? _getOffsetFromUpstream(offset, caretPrototype)
?? _emptyOffset; ?? _emptyOffset;
} }
assert(position.affinity != null);
return null; return null;
} }
...@@ -462,7 +464,7 @@ class TextPainter { ...@@ -462,7 +464,7 @@ class TextPainter {
/// A given selection might have more than one rect if this text painter /// A given selection might have more than one rect if this text painter
/// contains bidirectional text because logically contiguous text might not be /// contains bidirectional text because logically contiguous text might not be
/// visually contiguous. /// visually contiguous.
List<ui.TextBox> getBoxesForSelection(TextSelection selection) { List<TextBox> getBoxesForSelection(TextSelection selection) {
assert(!_needsLayout); assert(!_needsLayout);
return _paragraph.getBoxesForRange(selection.start, selection.end); return _paragraph.getBoxesForRange(selection.start, selection.end);
} }
......
...@@ -681,8 +681,8 @@ class TextStyle extends Diagnosticable { ...@@ -681,8 +681,8 @@ class TextStyle extends Diagnosticable {
defaultValue: null, defaultValue: null,
)); ));
styles.add(new EnumProperty<FontStyle>('${prefix}style', fontStyle, defaultValue: null)); styles.add(new EnumProperty<FontStyle>('${prefix}style', fontStyle, defaultValue: null));
styles.add(new DoubleProperty('${prefix}letterSpacing', letterSpacing, unit: 'x', defaultValue: null)); styles.add(new DoubleProperty('${prefix}letterSpacing', letterSpacing, defaultValue: null));
styles.add(new DoubleProperty('${prefix}wordSpacing', wordSpacing, unit: 'x', defaultValue: null)); styles.add(new DoubleProperty('${prefix}wordSpacing', wordSpacing, defaultValue: null));
styles.add(new EnumProperty<TextBaseline>('${prefix}baseline', textBaseline, defaultValue: null)); styles.add(new EnumProperty<TextBaseline>('${prefix}baseline', textBaseline, defaultValue: null));
styles.add(new DoubleProperty('${prefix}height', height, unit: 'x', defaultValue: null)); styles.add(new DoubleProperty('${prefix}height', height, unit: 'x', defaultValue: null));
if (decoration != null || decorationColor != null || decorationStyle != null) { if (decoration != null || decorationColor != null || decorationStyle != null) {
......
...@@ -362,6 +362,11 @@ class RenderEditable extends RenderBox { ...@@ -362,6 +362,11 @@ class RenderEditable extends RenderBox {
/// and the returned list is of length two. In this case, however, the two /// and the returned list is of length two. In this case, however, the two
/// points might actually be co-located (e.g., because of a bidirectional /// points might actually be co-located (e.g., because of a bidirectional
/// selection that contains some text but whose ends meet in the middle). /// selection that contains some text but whose ends meet in the middle).
///
/// See also:
///
/// * [getLocalRectForCaret], which is the equivalent but for
/// a [TextPosition] rather than a [TextSelection].
List<TextSelectionPoint> getEndpointsForSelection(TextSelection selection) { List<TextSelectionPoint> getEndpointsForSelection(TextSelection selection) {
assert(constraints != null); assert(constraints != null);
_layoutText(constraints.maxWidth); _layoutText(constraints.maxWidth);
...@@ -385,14 +390,30 @@ class RenderEditable extends RenderBox { ...@@ -385,14 +390,30 @@ class RenderEditable extends RenderBox {
} }
/// Returns the position in the text for the given global coordinate. /// Returns the position in the text for the given global coordinate.
///
/// See also:
///
/// * [getLocalRectForCaret], which is the reverse operation, taking
/// a [TextPosition] and returning a [Rect].
/// * [TextPainter.getPositionForOffset], which is the equivalent method
/// for a [TextPainter] object.
TextPosition getPositionForPoint(Offset globalPosition) { TextPosition getPositionForPoint(Offset globalPosition) {
_layoutText(constraints.maxWidth); _layoutText(constraints.maxWidth);
globalPosition += -_paintOffset; globalPosition += -_paintOffset;
return _textPainter.getPositionForOffset(globalToLocal(globalPosition)); return _textPainter.getPositionForOffset(globalToLocal(globalPosition));
} }
/// Returns the Rect in local coordinates for the caret at the given text /// Returns the [Rect] in local coordinates for the caret at the given text
/// position. /// position.
///
/// See also:
///
/// * [getPositionForPoint], which is the reverse operation, taking
/// an [Offset] in global coordinates and returning a [TextPosition].
/// * [getEndpointsForSelection], which is the equivalent but for
/// a selection rather than a particular text position.
/// * [TextPainter.getOffsetForCaret], the equivalent method for a
/// [TextPainter] object.
Rect getLocalRectForCaret(TextPosition caretPosition) { Rect getLocalRectForCaret(TextPosition caretPosition) {
_layoutText(constraints.maxWidth); _layoutText(constraints.maxWidth);
final Offset caretOffset = _textPainter.getOffsetForCaret(caretPosition, _caretPrototype); final Offset caretOffset = _textPainter.getOffsetForCaret(caretPosition, _caretPrototype);
......
This diff is collapsed.
...@@ -134,9 +134,10 @@ Future<Null> benchmarkWidgets(WidgetTesterCallback callback) { ...@@ -134,9 +134,10 @@ Future<Null> benchmarkWidgets(WidgetTesterCallback callback) {
/// that have not yet resolved. /// that have not yet resolved.
void expect(dynamic actual, dynamic matcher, { void expect(dynamic actual, dynamic matcher, {
String reason, String reason,
dynamic skip, // true or a String
}) { }) {
TestAsyncUtils.guardSync(); TestAsyncUtils.guardSync();
test_package.expect(actual, matcher, reason: reason); test_package.expect(actual, matcher, reason: reason, skip: skip);
} }
/// Assert that `actual` matches `matcher`. /// Assert that `actual` matches `matcher`.
......
...@@ -496,6 +496,7 @@ void main() { ...@@ -496,6 +496,7 @@ void main() {
'--non-interactive', '--non-interactive',
'--enable-checked-mode', '--enable-checked-mode',
'--use-test-fonts', '--use-test-fonts',
// '--enable-txt', // enable this to test libtxt rendering
'--packages=$packages', '--packages=$packages',
testPath, testPath,
]); ]);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment