forked from flutter/assets-for-api-docs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutils.dart
136 lines (120 loc) · 5.14 KB
/
utils.dart
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
/// This defines a colored placeholder with padding, used to represent a
/// generic widget in diagrams.
class Hole extends StatelessWidget {
const Hole({
Key? key,
this.color = const Color(0xFFFFFFFF),
}) : super(key: key);
final Color color;
@override
Widget build(BuildContext context) {
return AspectRatio(
aspectRatio: 1.0,
child: Padding(
padding: const EdgeInsets.all(4.0),
child: Placeholder(
strokeWidth: 2.0,
color: color,
),
),
);
}
}
/// This is a struct to represent a text label in the [LabelPainterWidget].
class Label {
const Label(this.key, this.text, this.anchor);
final GlobalKey key;
final String text;
final FractionalOffset anchor;
}
/// The will take a list of locations that a label should point to, defined
/// by the [Label] structure.
class LabelPainterWidget extends StatelessWidget {
/// Creates a widget that paints labels in defined locations.
///
/// All parameters are required and must not be null.
LabelPainterWidget({
required GlobalKey key,
required List<Label> labels,
required GlobalKey heroKey,
}) : painter = LabelPainter(labels: labels, heroKey: heroKey, canvasKey: key),
super(key: key);
final LabelPainter painter;
@override
Widget build(BuildContext context) => CustomPaint(painter: painter);
}
/// The custom painter that [LabelPainterWidget] uses to paint the list of
/// labels it is given.
class LabelPainter extends CustomPainter {
LabelPainter({
required this.labels,
required this.heroKey,
required this.canvasKey,
}) : _painters = <Label, TextPainter>{} {
for (final Label label in labels) {
final TextPainter painter = TextPainter(
textDirection: TextDirection.ltr,
text: TextSpan(text: label.text, style: _labelTextStyle),
);
painter.layout();
_painters[label] = painter;
}
}
final List<Label> labels;
final GlobalKey heroKey;
final GlobalKey canvasKey;
final Map<Label, TextPainter> _painters;
static const TextStyle _labelTextStyle = TextStyle(color: Color(0xFF000000));
static const double margin = 16.0;
@override
void paint(Canvas canvas, Size size) {
final RenderBox hero = heroKey.currentContext!.findRenderObject() as RenderBox;
final RenderBox diagram = canvasKey.currentContext!.findRenderObject() as RenderBox;
final Paint dotPaint = Paint();
final Paint linePaint = Paint()..strokeWidth = 2.0;
final Offset heroTopLeft = diagram.globalToLocal(hero.localToGlobal(Offset.zero));
for (final Label label in labels) {
final RenderBox box = label.key.currentContext!.findRenderObject() as RenderBox;
final Offset anchor = diagram.globalToLocal(box.localToGlobal(label.anchor.alongSize(box.size)));
final Offset anchorOnHero = anchor - heroTopLeft;
final FractionalOffset relativeAnchor = FractionalOffset.fromOffsetAndSize(anchorOnHero, hero.size);
final double distanceToTop = anchorOnHero.dy;
final double distanceToBottom = hero.size.height - anchorOnHero.dy;
final double distanceToLeft = anchorOnHero.dx;
final double distanceToRight = hero.size.width - anchorOnHero.dx;
Offset labelPosition;
Offset textPosition = Offset.zero;
final TextPainter painter = _painters[label]!;
if (distanceToTop <= distanceToLeft && distanceToTop <= distanceToRight && distanceToTop <= distanceToBottom) {
labelPosition = Offset(anchor.dx + (relativeAnchor.dx - 0.5) * margin, heroTopLeft.dy - margin);
textPosition = Offset(labelPosition.dx - painter.width / 2.0, labelPosition.dy - painter.height);
} else if (distanceToBottom < distanceToLeft && distanceToBottom < distanceToRight && distanceToTop > distanceToBottom) {
labelPosition = Offset(anchor.dx, heroTopLeft.dy + hero.size.height + margin);
textPosition = Offset(labelPosition.dx - painter.width / 2.0, labelPosition.dy);
} else if (distanceToLeft < distanceToRight) {
labelPosition = Offset(heroTopLeft.dx - margin, anchor.dy);
textPosition = Offset(labelPosition.dx - painter.width - 2.0, labelPosition.dy - painter.height / 2.0);
} else if (distanceToLeft > distanceToRight) {
labelPosition = Offset(heroTopLeft.dx + hero.size.width + margin, anchor.dy);
textPosition = Offset(labelPosition.dx, labelPosition.dy - painter.height / 2.0);
} else {
labelPosition = Offset(anchor.dx, heroTopLeft.dy - margin * 2.0);
textPosition = Offset(anchor.dx - painter.width / 2.0, anchor.dy - margin - painter.height);
}
canvas.drawCircle(anchor, 4.0, dotPaint);
canvas.drawLine(anchor, labelPosition, linePaint);
painter.paint(canvas, textPosition);
}
}
@override
bool shouldRepaint(LabelPainter oldDelegate) {
return labels != oldDelegate.labels || canvasKey != oldDelegate.canvasKey;
}
@override
bool hitTest(Offset position) => false;
}