-
-
Notifications
You must be signed in to change notification settings - Fork 897
/
Copy pathlists.dart
139 lines (119 loc) · 4.53 KB
/
lists.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
137
138
139
import 'dart:collection';
import 'package:collection/collection.dart';
import 'package:flutter_html/flutter_html.dart';
import 'package:list_counter/list_counter.dart';
class ListProcessing {
static StyledElement processLists(StyledElement tree) {
tree = _preprocessListMarkers(tree);
tree = _processListCounters(tree);
tree = _processListMarkers(tree);
return tree;
}
/// [_preprocessListMarkers] adds marker pseudo elements to the front of all list
/// items.
static StyledElement _preprocessListMarkers(StyledElement tree) {
tree.style.listStylePosition ??= ListStylePosition.outside;
if (tree.style.display == Display.listItem) {
// Add the marker pseudo-element if it doesn't exist
tree.style.marker ??= Marker(
content: Content.normal,
style: tree.style,
);
// Inherit styles from originating widget
tree.style.marker!.style =
tree.style.copyOnlyInherited(tree.style.marker!.style ?? Style());
// Add the implicit counter-increment on `list-item` if it isn't set
// explicitly already
tree.style.counterIncrement ??= {};
if (!tree.style.counterIncrement!.containsKey('list-item')) {
tree.style.counterIncrement!['list-item'] = 1;
}
}
// Add the counters to ol and ul types.
if (tree.name == 'ol' || tree.name == 'ul') {
// respect the `start` attribute in order to begin lists at !1.
final startAttribute = tree.attributes['start'];
final offset =
startAttribute == null ? 1 : int.tryParse(startAttribute) ?? 1;
tree.style.counterIncrement ??= {};
// the counter starts at 1 so set to 0 in order to be correct after first
// increment
tree.style.counterIncrement!['list-item'] = offset - 1;
tree.style.counterReset ??= {};
if (!tree.style.counterReset!.containsKey('list-item')) {
tree.style.counterReset!['list-item'] = 0;
}
}
for (var child in tree.children) {
_preprocessListMarkers(child);
}
return tree;
}
/// [_processListCounters] adds the appropriate counter values to each
/// StyledElement on the tree.
static StyledElement _processListCounters(StyledElement tree,
[ListQueue<Counter>? counters]) {
// Add the counters for the current scope.
tree.counters.addAll(counters?.deepCopy() ?? []);
// Create any new counters
if (tree.style.counterReset != null) {
tree.style.counterReset!.forEach((counterName, initialValue) {
tree.counters.add(Counter(counterName, initialValue ?? 0));
});
}
// Increment any counters that are to be incremented
if (tree.style.counterIncrement != null) {
tree.style.counterIncrement!.forEach((counterName, increment) {
tree.counters
.lastWhereOrNull(
(counter) => counter.name == counterName,
)
?.increment(increment ?? 1);
// If we didn't newly create the counter, increment the counter in the old copy as well.
if (tree.style.counterReset == null ||
!tree.style.counterReset!.containsKey(counterName)) {
counters
?.lastWhereOrNull(
(counter) => counter.name == counterName,
)
?.increment(increment ?? 1);
}
});
}
for (var element in tree.children) {
_processListCounters(element, tree.counters);
}
return tree;
}
/// [_processListMarkers] finally applies the marker content to the tree
/// as a [Marker] on the [Style] object.
static StyledElement _processListMarkers(StyledElement tree) {
if (tree.style.display == Display.listItem) {
final listStyleType = tree.style.listStyleType ?? ListStyleType.decimal;
final counterStyle = CounterStyleRegistry.lookup(
listStyleType.counterStyle,
);
String counterContent;
if (tree.style.marker?.content.isNormal ?? true) {
counterContent = counterStyle.generateMarkerContent(
tree.counters.lastOrNull?.value ?? 0,
);
} else if (!(tree.style.marker?.content.display ?? true)) {
counterContent = '';
} else {
counterContent = tree.style.marker?.content.replacementContent ??
counterStyle.generateMarkerContent(
tree.counters.lastOrNull?.value ?? 0,
);
}
tree.style.marker = Marker(
content: Content(counterContent),
style: tree.style.marker?.style,
);
}
for (var child in tree.children) {
_processListMarkers(child);
}
return tree;
}
}