-
Notifications
You must be signed in to change notification settings - Fork 84
/
Copy pathListCommand.swift
166 lines (146 loc) · 7.68 KB
/
ListCommand.swift
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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
//
// ListCommand.swift
// Proton
//
// Created by Rajdeep Kwatra on 28/5/20.
// Copyright © 2020 Rajdeep Kwatra. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
import Foundation
import UIKit
public enum Indentation {
case indent
case outdent
}
/// Describes the formatting of a line of text. While general purpose in nature, this is
/// used by `EditorListFormattingProvider` for providing formatting for lists.
public struct LineFormatting {
/// Indentation of line
public let indentation: CGFloat
/// Vertical spacing before the line
public let spacingBefore: CGFloat
/// Vertical spacing after the line
public let spacingAfter: CGFloat?
/// Initializes
/// - Parameters:
/// - indentation: Indentation for each line of text
/// - spacingBefore: Vertical spacing before line of text
/// - spacingAfter: Vertical spacing after line of text
public init(indentation: CGFloat, spacingBefore: CGFloat, spacingAfter: CGFloat? = nil) {
self.indentation = indentation
self.spacingBefore = spacingBefore
self.spacingAfter = spacingAfter
}
}
/// Command that can be used to toggle list attributes of selected range of text.
/// If the length of selected range of text is 0, the attributes are applied on the current line of text.
public class ListCommand: EditorCommand {
public init() { }
/// Name of the command
public var name: CommandName {
return CommandName("listCommand")
}
/// Value to be set for attribute `.listItem` when applying to a range of text.
/// This value is returned back by `ListFormattingProvider` when querying list marker for a given index.
/// This may be used to store info that helps generate appropriate markers for e.g. storing context related
/// to bullet vs ordered lists.
/// - Note:
/// When set to nil before running `execute`, it removes list formatting from the selected range of text.
public var attributeValue: Any?
/// Executes the command with value of `attributeValue` for `.listItem` attribute. If the `attributeValue` is nil, executing
/// removed list formatting from the selected range of text.
/// - Parameter editor: Editor to execute the command on.
public func execute(on editor: EditorView) {
var selectedRange = editor.selectedRange
// Adjust to span entire line range if the selection starts in the middle of the line
if let currentLine = editor.contentLinesInRange(NSRange(location: selectedRange.location, length: 0)).first,
currentLine.range.length > 0 {
let location = currentLine.range.location
var length = max(currentLine.range.length, selectedRange.length + (selectedRange.location - currentLine.range.location))
let range = NSRange(location: location, length: length)
if editor.contentLength > range.endLocation,
editor.attributedText.substring(from: NSRange(location: range.endLocation, length: 1)) == "\n" {
length += 1
}
selectedRange = NSRange(location: location, length: length)
}
guard selectedRange.length > 0 else {
if editor.isEmpty ||
editor.attributedText.attribute(.listItem, at: max(0, editor.selectedRange.location - 1), effectiveRange: nil) == nil {
ListTextProcessor().createListItemInANewLine(editor: editor, editedRange: selectedRange, indentMode: .indent, attributeValue: attributeValue)
} else {
ListTextProcessor().exitList(editor: editor)
}
return
}
guard let attrValue = attributeValue else {
let paragraphStyle = editor.paragraphStyle
editor.addAttributes([
.paragraphStyle: paragraphStyle
], at: selectedRange)
editor.removeAttribute(.listItem, at: selectedRange)
editor.typingAttributes[.listItem] = nil
cleanupIfNeeded(editor: editor)
return
}
// Fix the list attribute on the trailing `\n` in previous line, if previous line has a listItem attribute applied
if let previousLine = editor.previousContentLine(from: selectedRange.location),
let listValue = editor.attributedText.attribute(.listItem, at: previousLine.range.endLocation - 1, effectiveRange: nil),
editor.attributedText.attribute(.listItem, at: previousLine.range.endLocation, effectiveRange: nil) == nil {
editor.addAttribute(.listItem, value: listValue, at: NSRange(location: previousLine.range.endLocation, length: 1))
}
editor.attributedText.enumerateAttribute(.paragraphStyle, in: selectedRange, options: []) { (value, range, _) in
let paraStyle = value as? NSParagraphStyle
let mutableStyle = ListTextProcessor().updatedParagraphStyle(paraStyle: paraStyle, listLineFormatting: editor.listLineFormatting, indentMode: .indent, defaultParaStyle: editor.paragraphStyle)
editor.addAttribute(.paragraphStyle, value: mutableStyle ?? editor.paragraphStyle, at: range)
}
editor.addAttribute(.listItem, value: attrValue, at: selectedRange)
editor.typingAttributes[.listItem] = attrValue
attributeValue = nil
}
/// Executes the command with value of `attributeValue` for `.listItem` attribute.
/// - Parameters:
/// - editor: Editor to execute the command on.
/// - attributeValue: Value of `.listItem` attribute. Use nil to remove list formatting.
public func execute(on editor: EditorView, attributeValue: Any?) {
self.attributeValue = attributeValue
execute(on: editor)
}
// Cleanup any dangling lists after the parent of this is removed as being a list item
private func cleanupIfNeeded(editor: EditorView) {
guard let nextContentLine = editor.nextContentLine(from: editor.selectedRange.endLocation),
nextContentLine.text.attributeOrNil(.listItem, at: 0) != nil,
let listToUpdateRange = editor.attributedText.rangeOf(attribute: .listItem, startingLocation: editor.selectedRange.endLocation) else { return }
let listIndent = editor.listLineFormatting.indentation
var levelToSet = 0
var prevStyle: NSParagraphStyle?
editor.attributedText.enumerateAttribute(.paragraphStyle, in: listToUpdateRange) { value, range, stop in
levelToSet = 0
if let paraStyle = (value as? NSParagraphStyle)?.mutableParagraphStyle {
let previousLevel = Int(prevStyle?.firstLineHeadIndent ?? 0)/Int(listIndent)
let currentLevel = Int(paraStyle.firstLineHeadIndent)/Int(listIndent)
if currentLevel - previousLevel > 1 {
levelToSet = previousLevel + 1
let indentation = CGFloat(levelToSet) * listIndent
paraStyle.firstLineHeadIndent = indentation
paraStyle.headIndent = indentation
editor.addAttribute(.paragraphStyle, value: paraStyle, at: range)
prevStyle = paraStyle
} else {
prevStyle = value as? NSParagraphStyle
}
}
}
}
}