-
Notifications
You must be signed in to change notification settings - Fork 122
/
Copy pathmark.ts
111 lines (101 loc) · 3.62 KB
/
mark.ts
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
import {compareDeep} from "./comparedeep"
import {Attrs, MarkType, Schema} from "./schema"
/// A mark is a piece of information that can be attached to a node,
/// such as it being emphasized, in code font, or a link. It has a
/// type and optionally a set of attributes that provide further
/// information (such as the target of the link). Marks are created
/// through a `Schema`, which controls which types exist and which
/// attributes they have.
export class Mark {
/// @internal
constructor(
/// The type of this mark.
readonly type: MarkType,
/// The attributes associated with this mark.
readonly attrs: Attrs
) {}
/// Given a set of marks, create a new set which contains this one as
/// well, in the right position. If this mark is already in the set,
/// the set itself is returned. If any marks that are set to be
/// [exclusive](#model.MarkSpec.excludes) with this mark are present,
/// those are replaced by this one.
addToSet(set: readonly Mark[]): readonly Mark[] {
let copy, placed = false
for (let i = 0; i < set.length; i++) {
let other = set[i]
if (this.eq(other)) return set
if (this.type.excludes(other.type)) {
if (!copy) copy = set.slice(0, i)
} else if (other.type.excludes(this.type)) {
return set
} else {
if (!placed && other.type.rank > this.type.rank) {
if (!copy) copy = set.slice(0, i)
copy.push(this)
placed = true
}
if (copy) copy.push(other)
}
}
if (!copy) copy = set.slice()
if (!placed) copy.push(this)
return copy
}
/// Remove this mark from the given set, returning a new set. If this
/// mark is not in the set, the set itself is returned.
removeFromSet(set: readonly Mark[]): readonly Mark[] {
for (let i = 0; i < set.length; i++)
if (this.eq(set[i]))
return set.slice(0, i).concat(set.slice(i + 1))
return set
}
/// Test whether this mark is in the given set of marks.
isInSet(set: readonly Mark[]) {
for (let i = 0; i < set.length; i++)
if (this.eq(set[i])) return true
return false
}
/// Test whether this mark has the same type and attributes as
/// another mark.
eq(other: Mark) {
return this == other ||
(this.type == other.type && compareDeep(this.attrs, other.attrs))
}
/// Convert this mark to a JSON-serializeable representation.
toJSON(): any {
let obj: any = {type: this.type.name}
for (let _ in this.attrs) {
obj.attrs = this.attrs
break
}
return obj
}
/// Deserialize a mark from JSON.
static fromJSON(schema: Schema, json: any) {
if (!json) throw new RangeError("Invalid input for Mark.fromJSON")
let type = schema.marks[json.type]
if (!type) throw new RangeError(`There is no mark type ${json.type} in this schema`)
let mark = type.create(json.attrs)
type.checkAttrs(mark.attrs)
return mark
}
/// Test whether two sets of marks are identical.
static sameSet(a: readonly Mark[], b: readonly Mark[]) {
if (a == b) return true
if (a.length != b.length) return false
for (let i = 0; i < a.length; i++)
if (!a[i].eq(b[i])) return false
return true
}
/// Create a properly sorted mark set from null, a single mark, or an
/// unsorted array of marks.
static setFrom(marks?: Mark | readonly Mark[] | null): readonly Mark[] {
if (!marks || Array.isArray(marks) && marks.length == 0) return Mark.none
if (marks instanceof Mark) return [marks]
let copy = marks.slice()
copy.sort((a, b) => a.type.rank - b.type.rank)
return copy
}
/// The empty set of marks.
static none: readonly Mark[] = []
}