From 495238ddcdd4a0fba468775d2c2ca1e0b8cbd61e Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 14:03:06 -0300 Subject: [PATCH 01/74] New More Formatting Identifier --- Aztec/Classes/GUI/FormatBar/FormattingIdentifier.swift | 1 + 1 file changed, 1 insertion(+) diff --git a/Aztec/Classes/GUI/FormatBar/FormattingIdentifier.swift b/Aztec/Classes/GUI/FormatBar/FormattingIdentifier.swift index 0f30ad335..067d0a00d 100644 --- a/Aztec/Classes/GUI/FormatBar/FormattingIdentifier.swift +++ b/Aztec/Classes/GUI/FormatBar/FormattingIdentifier.swift @@ -11,6 +11,7 @@ public enum FormattingIdentifier: String { case blockquote = "blockquote" case link = "link" case media = "media" + case more = "more" case sourcecode = "sourcecode" case header = "header" case header1 = "header1" From 0ad59655650555ef41c6c45685f33d1352e1ed47 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Thu, 16 Mar 2017 17:18:48 +0000 Subject: [PATCH 02/74] Add test to verify crash. --- AztecTests/TextStorageTests.swift | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index aac4b585a..740307d96 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -272,4 +272,18 @@ class TextStorageTests: XCTestCase XCTAssertNil(storage.attachment(withId: identifier)) } } + + /// This test verifies if we can delete all the content from a storage object that has html with a comment + /// + func testDeleteAllSelectionWhenContentHasComments() { + let storage = TextStorage() + let commentString = "This is a comment" + let html = "" + storage.setHTML(html, withDefaultFontDescriptor: UIFont.systemFont(ofSize: 14).fontDescriptor) + storage.replaceCharacters(in: NSRange(location: 0, length: 1), with: NSAttributedString(string: "")) + + let resultHTML = storage.getHTML() + + XCTAssertEqual(html, resultHTML) + } } From 284c97022506f69b9fcd63d0c27d3d73fe596e5b Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 14:23:44 -0300 Subject: [PATCH 03/74] NSAttributedString+Attachments: New Helper --- .../Classes/Extensions/NSAttributedString+Attachments.swift | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift b/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift index 9db800bec..cc90fe0ae 100644 --- a/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift +++ b/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift @@ -6,6 +6,11 @@ import UIKit // extension NSAttributedString { + /// Indicates the Attributed String Length of a single TextAttachment + /// + static let lengthOfTextAttachment = NSAttributedString(attachment: NSTextAttachment()).length + + /// Loads any NSTextAttachment's lazy file reference, into a UIImage instance, in memory. /// func loadLazyAttachments() { From eb4149bf53499ea2853fa7d833197a28268b9048 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 14:29:37 -0300 Subject: [PATCH 04/74] TextStorage: New insertMoreAttachment helper --- Aztec/Classes/TextKit/TextStorage.swift | 21 +++++++++++++++++++-- 1 file changed, 19 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index baf711543..55b168494 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -177,10 +177,10 @@ open class TextStorage: NSTextStorage { attributedString.enumerateAttribute(NSAttachmentAttributeName, in: fullRange, options: []) { (object, range, stop) in - guard let object = object else { + guard let object = object, !(object is MoreAttachment) else { return } - + guard let attachmentsDelegate = attachmentsDelegate else { assertionFailure("This class can't really handle not having an image provider set.") return @@ -741,6 +741,23 @@ open class TextStorage: NSTextStorage { } } + /// Inserts the MoreAttachment at the specified position + /// + open func insertMoreAttachment(at position: Int) -> MoreAttachment { + let message = "MORE" + let label = NSLocalizedString("MORE", comment: "Text for the center of the more divider") + + let attachment = MoreAttachment() + attachment.message = message + attachment.label = NSAttributedString(string: label, attributes: [:]) + + let payload = NSAttributedString(attachment: attachment) + let target = NSMakeRange(position, 0) + replaceCharacters(in: target, with: payload) + + return attachment + } + // MARK: - Toggle Attributes From 51df040696f7c4df68ead322400eeb19351f29cf Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 14:29:51 -0300 Subject: [PATCH 05/74] TextView: New insertMoreAttachment Helper --- Aztec/Classes/TextKit/TextView.swift | 26 ++++++++++++++++++++++++-- 1 file changed, 24 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 887619bde..4c79037fd 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -734,7 +734,7 @@ open class TextView: UITextView { /// open func insertImage(sourceURL url: URL, atPosition position: Int, placeHolderImage: UIImage?, identifier: String = UUID().uuidString) -> TextAttachment { let attachment = storage.insertImage(sourceURL: url, atPosition: position, placeHolderImage: placeHolderImage ?? defaultMissingImage, identifier: identifier) - let length = NSAttributedString(attachment:NSTextAttachment()).length + let length = NSAttributedString.lengthOfTextAttachment textStorage.addAttributes(typingAttributes, range: NSMakeRange(position, length)) selectedRange = NSMakeRange(position+length, 0) delegate?.textViewDidChange?(self) @@ -930,7 +930,29 @@ open class TextView: UITextView { /// open func refreshLayoutFor(attachment: TextAttachment) { layoutManager.invalidateLayoutForAttachment(attachment) - } + } + + + // MARK: - More + + /// Inserts the More Comment at the specified position. + /// + /// - Parameter position: The character position at which to insert the more attachment. + /// + /// - Returns: the attachment object that can be used for further calls + /// + @discardableResult + open func insertMoreAttachment(at position: Int) -> MoreAttachment { + let attachment = storage.insertMoreAttachment(at: position) + let attachmentRange = NSRange(location: position, length: NSAttributedString.lengthOfTextAttachment) + + storage.addAttributes(typingAttributes, range: attachmentRange) + + selectedRange = NSMakeRange(attachmentRange.endLocation, 0) + delegate?.textViewDidChange?(self) + + return attachment + } } From b7b5f8f9d9b551dc5f6fd16ea84dd4334fe2a52d Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 14:31:17 -0300 Subject: [PATCH 06/74] EditorDemoController: Wiring insertMore API --- Example/Example/EditorDemoController.swift | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index 7c648aa54..42ceff9a7 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -393,6 +393,8 @@ extension EditorDemoController : Aztec.FormatBarDelegate { toggleEditingMode() case .header, .header1, .header2, .header3, .header4, .header5, .header6: toggleHeader() + case .more: + insertMoreAttachment() } updateFormatBar() @@ -500,6 +502,11 @@ extension EditorDemoController : Aztec.FormatBarDelegate { showLinkDialog(forURL: linkURL, title: linkTitle, range: linkRange) } + func insertMoreAttachment() { + let position = richTextView.selectedRange.location + richTextView.insertMoreAttachment(at: position) + } + func showLinkDialog(forURL url: URL?, title: String?, range: NSRange) { let isInsertingNewLink = (url == nil) @@ -659,7 +666,8 @@ extension EditorDemoController : Aztec.FormatBarDelegate { FormatBarItem(image: Gridicon.iconOfType(.quote), identifier: .blockquote), FormatBarItem(image: Gridicon.iconOfType(.listUnordered), identifier: .unorderedlist), FormatBarItem(image: Gridicon.iconOfType(.listOrdered), identifier: .orderedlist), - FormatBarItem(image: Gridicon.iconOfType(.link), identifier: .link) + FormatBarItem(image: Gridicon.iconOfType(.link), identifier: .link), + FormatBarItem(image: Gridicon.iconOfType(.readMore), identifier: .more) ] } From 78a658784a02447997610da11d3364f3eec50077 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 16:45:39 -0300 Subject: [PATCH 07/74] DOMString: New insertComment Helper --- Aztec/Classes/Libxml2/DOMString.swift | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/Aztec/Classes/Libxml2/DOMString.swift b/Aztec/Classes/Libxml2/DOMString.swift index 0ee874e2d..36e4cd9db 100644 --- a/Aztec/Classes/Libxml2/DOMString.swift +++ b/Aztec/Classes/Libxml2/DOMString.swift @@ -510,6 +510,25 @@ extension Libxml2 { rootNode.replaceCharacters(in: range, with: descriptor) } + /// Inserts an HTML Comment at the specified range. + /// + /// - Parameters: + /// - comment: the comment to be stored. + /// - range: the range to apply the style to. + /// + func insertComment(comment: String, at range: NSRange) { + performAsyncUndoable { [weak self] in + self?.insertCommentSynchronously(comment: comment, at: range) + } + } + + private func insertCommentSynchronously(comment: String, at range: NSRange) { + let descriptor = CommentNodeDescriptor(comment: comment) + + rootNode.replaceCharacters(in: range, with: descriptor) + } + + // MARK: - Styles to HTML elements /// Applies a standard HTML element to the specified range. From 78d287c3193baaeddc020b6a824b3901eefc27ef Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 16:46:02 -0300 Subject: [PATCH 08/74] TextStorage: Comment Diffing Support --- Aztec/Classes/TextKit/TextStorage.swift | 33 +++++++++++++++++++------ 1 file changed, 26 insertions(+), 7 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 32db0eaa1..d0246aea7 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -339,6 +339,9 @@ open class TextStorage: NSTextStorage { /// - targetValue: the new value of the attribute /// private func processAttributesDifference(in domRange: NSRange, key: String, sourceValue: Any?, targetValue: Any?) { + let isLineAttachment = sourceValue is LineAttachment || targetValue is LineAttachment + let isMoreAttachment = sourceValue is MoreAttachment || targetValue is MoreAttachment + switch(key) { case NSFontAttributeName: let sourceFont = sourceValue as? UIFont @@ -355,14 +358,17 @@ open class TextStorage: NSTextStorage { let targetStyle = targetValue as? NSNumber processUnderlineDifferences(in: domRange, betweenOriginal: sourceStyle, andNew: targetStyle) - case NSAttachmentAttributeName: - if sourceValue is LineAttachment || targetValue is LineAttachment { - let sourceAttachment = sourceValue as? LineAttachment - let targetAttachment = targetValue as? LineAttachment + case NSAttachmentAttributeName where isLineAttachment: + let sourceAttachment = sourceValue as? LineAttachment + let targetAttachment = targetValue as? LineAttachment - processLineAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) - return - } + processLineAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) + case NSAttachmentAttributeName where isMoreAttachment: + let sourceAttachment = sourceValue as? MoreAttachment + let targetAttachment = targetValue as? MoreAttachment + + processMoreAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) + case NSAttachmentAttributeName: let sourceAttachment = sourceValue as? TextAttachment let targetAttachment = targetValue as? TextAttachment @@ -457,6 +463,19 @@ open class TextStorage: NSTextStorage { } } + private func processMoreAttachmentDifferences(in range: NSRange, betweenOriginal original: MoreAttachment?, andNew new: MoreAttachment?) { + + let add = original == nil && new != nil + let remove = original != nil && new == nil + + if add { + dom.insertComment(comment: "Comment", at: range) + } else if remove { + dom.remove(element: .hr, at: range) + } + } + + /// Processes differences in the italic trait of two font objects, and applies them to the DOM /// in the specified range. /// From b54c1f3fdc0acb2b48e91b356bef146b16767fbe Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 16:46:14 -0300 Subject: [PATCH 09/74] CommentNodeDescriptor: Adding default node name --- .../Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift b/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift index f25d6742a..4c692fed7 100644 --- a/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift +++ b/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift @@ -8,6 +8,7 @@ extension Libxml2 { /// - Creating it. /// class CommentNodeDescriptor: NodeDescriptor { + let defaultCommentName = "comment" let comment: String // MARK: - CustomReflectable @@ -18,9 +19,9 @@ extension Libxml2 { } } - init(name: String, comment: String) { + init(comment: String) { self.comment = comment - super.init(name: name) + super.init(name: defaultCommentName) } } } From 6ca554de0b635d820840c059353f9d5f0a2a19fb Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 16:47:36 -0300 Subject: [PATCH 10/74] TextStorage: Adds TODO --- Aztec/Classes/TextKit/TextStorage.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index d0246aea7..438fd1ae0 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -471,7 +471,8 @@ open class TextStorage: NSTextStorage { if add { dom.insertComment(comment: "Comment", at: range) } else if remove { - dom.remove(element: .hr, at: range) +// TODO: FIXME. Proper removeComment support +// dom.remove(element: .hr, at: range) } } From 89d497d9460d1870c35922f5cc1b1c86cda2bf5a Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 17:28:34 -0300 Subject: [PATCH 11/74] CommentNodeDescriptor: Renames constant --- Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift b/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift index 4c692fed7..6fbbb592b 100644 --- a/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift +++ b/Aztec/Classes/Libxml2/Descriptors/CommentNodeDescriptor.swift @@ -8,7 +8,7 @@ extension Libxml2 { /// - Creating it. /// class CommentNodeDescriptor: NodeDescriptor { - let defaultCommentName = "comment" + let nodeName = "comment" let comment: String // MARK: - CustomReflectable @@ -21,7 +21,7 @@ extension Libxml2 { init(comment: String) { self.comment = comment - super.init(name: defaultCommentName) + super.init(name: nodeName) } } } From df5a5ef2f839ac44891baefd0ec32783c4d1bc63 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 17:30:40 -0300 Subject: [PATCH 12/74] DOMString: Normalizes replacement helper names --- Aztec/Classes/Libxml2/DOMString.swift | 28 +++++++++++++-------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/Aztec/Classes/Libxml2/DOMString.swift b/Aztec/Classes/Libxml2/DOMString.swift index 36e4cd9db..020031198 100644 --- a/Aztec/Classes/Libxml2/DOMString.swift +++ b/Aztec/Classes/Libxml2/DOMString.swift @@ -472,19 +472,19 @@ extension Libxml2 { // MARK: - Images - /// Applies an image to the specified range + /// Replaces the specified range with a given image. /// /// - Parameters: - /// - imageURL: the URL for the img src attribute /// - range: the range to insert the image + /// - imageURL: the URL for the img src attribute /// - func insertImage(imageURL: URL, replacing range:NSRange) { + func replace(_ range: NSRange, with imageURL: URL) { performAsyncUndoable { [weak self] in - self?.insertImageSynchronously(imageURL: imageURL, replacing: range) + self?.replaceSynchronously(range, with: imageURL) } } - private func insertImageSynchronously(imageURL: URL, replacing range: NSRange) { + private func replaceSynchronously(_ range: NSRange, with imageURL: URL) { let imageURLString = imageURL.absoluteString let attributes = [Libxml2.StringAttribute(name:"src", value: imageURLString)] @@ -493,36 +493,36 @@ extension Libxml2 { rootNode.replaceCharacters(in: range, with: descriptor) } - /// Applies horizontal ruler to the specified range. + /// Replaces the specified range with a Horizontal Ruler. /// /// - Parameters: /// - range: the range to apply the style to. /// - func insertHorizontalRuler(at range: NSRange) { + func replaceWithHorizontalRuler(_ range: NSRange) { performAsyncUndoable { [weak self] in - self?.insertHorizontalRulerSynchronously(at: range) + self?.replaceSynchronouslyWithHorizontalRuler(range) } } - private func insertHorizontalRulerSynchronously(at range: NSRange) { + private func replaceSynchronouslyWithHorizontalRuler(_ range: NSRange) { let descriptor = ElementNodeDescriptor(elementType: .hr) rootNode.replaceCharacters(in: range, with: descriptor) } - /// Inserts an HTML Comment at the specified range. + /// Replaces the specified range with a Comment. /// /// - Parameters: - /// - comment: the comment to be stored. /// - range: the range to apply the style to. + /// - comment: the comment to be stored. /// - func insertComment(comment: String, at range: NSRange) { + func replace(_ range: NSRange, with comment: String) { performAsyncUndoable { [weak self] in - self?.insertCommentSynchronously(comment: comment, at: range) + self?.replaceSynchronously(range, with: comment) } } - private func insertCommentSynchronously(comment: String, at range: NSRange) { + private func replaceSynchronously(_ range: NSRange, with comment: String) { let descriptor = CommentNodeDescriptor(comment: comment) rootNode.replaceCharacters(in: range, with: descriptor) From 0ec98170362fc0dde447350361396bfe1866f66a Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 16 Mar 2017 17:30:54 -0300 Subject: [PATCH 13/74] TextStorage: Wires new DOM API --- Aztec/Classes/TextKit/TextStorage.swift | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 438fd1ae0..5e97592c9 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -445,7 +445,7 @@ open class TextStorage: NSTextStorage { return } - dom.insertImage(imageURL: urlToAdd, replacing: range) + dom.replace(range, with: urlToAdd) } else if removeImageUrl { dom.removeImage(spanning: range) } @@ -457,7 +457,7 @@ open class TextStorage: NSTextStorage { let remove = original != nil && new == nil if add { - dom.insertHorizontalRuler(at: range) + dom.replaceWithHorizontalRuler(range) } else if remove { dom.remove(element: .hr, at: range) } @@ -469,7 +469,8 @@ open class TextStorage: NSTextStorage { let remove = original != nil && new == nil if add { - dom.insertComment(comment: "Comment", at: range) +// TODO: Load with CommentAttachment's payload + dom.replace(range, with: "Comment") } else if remove { // TODO: FIXME. Proper removeComment support // dom.remove(element: .hr, at: range) From 4b74cceb9f6494b9abd80d7cffe0ac2f52845243 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Fri, 17 Mar 2017 10:57:49 +0000 Subject: [PATCH 14/74] Update gridicon to version 0.5 --- Example/Cartfile.resolved | 2 +- WordPress-Aztec-iOS.podspec | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Example/Cartfile.resolved b/Example/Cartfile.resolved index 85d110d48..7fbad5f3b 100644 --- a/Example/Cartfile.resolved +++ b/Example/Cartfile.resolved @@ -1 +1 @@ -github "Automattic/Gridicons-iOS" "8582ace3eae40bfb56f5e341c76295ccc830eaed" +github "Automattic/Gridicons-iOS" "977cbfaa88d35e2fdaceef496be211cbe4578f8b" diff --git a/WordPress-Aztec-iOS.podspec b/WordPress-Aztec-iOS.podspec index 0afdd1512..6a52695f4 100644 --- a/WordPress-Aztec-iOS.podspec +++ b/WordPress-Aztec-iOS.podspec @@ -38,6 +38,6 @@ Pod::Spec.new do |s| s.xcconfig = {'HEADER_SEARCH_PATHS' => '$(SDKROOT)/usr/include/libxml2'} s.preserve_paths = 'Aztec/Modulemaps/libxml2/*' - s.dependency 'Gridicons', '0.4' + s.dependency 'Gridicons', '0.5' end From ce9e463580a73dc3f34a4de69ffab6a44bc3283a Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 10:21:01 -0300 Subject: [PATCH 15/74] TextStorage: Addressing TODO's --- Aztec/Classes/TextKit/MoreAttachment.swift | 6 ++++++ Aztec/Classes/TextKit/TextStorage.swift | 13 ++++--------- 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/Aztec/Classes/TextKit/MoreAttachment.swift b/Aztec/Classes/TextKit/MoreAttachment.swift index ae6e14d2d..9e0347530 100644 --- a/Aztec/Classes/TextKit/MoreAttachment.swift +++ b/Aztec/Classes/TextKit/MoreAttachment.swift @@ -21,6 +21,12 @@ open class MoreAttachment: NSTextAttachment } } + /// Comment's Text. + /// This is a temporary helper property, and will be removed as soon as we merge MoreAttachment + CommentAttachment. + /// + let text = "MORE" + + open var message: String = "" // MARK: - NSTextAttachmentContainer diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 5e97592c9..7f5f7e6f2 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -465,15 +465,10 @@ open class TextStorage: NSTextStorage { private func processMoreAttachmentDifferences(in range: NSRange, betweenOriginal original: MoreAttachment?, andNew new: MoreAttachment?) { - let add = original == nil && new != nil - let remove = original != nil && new == nil - - if add { -// TODO: Load with CommentAttachment's payload - dom.replace(range, with: "Comment") - } else if remove { -// TODO: FIXME. Proper removeComment support -// dom.remove(element: .hr, at: range) + if let newAttachment = new, original == nil { + dom.replace(range, with: newAttachment.text) + } else if let _ = original, new == nil { + dom.replaceCharacters(inRange: range, withString: String(), preferLeftNode: false) } } From 81ed091733b82484116cc4fecfb4382d44297572 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 10:28:39 -0300 Subject: [PATCH 16/74] HTMLNodeToNSAttributedString: More tag is now case insensitive --- Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift index 47b6e15fb..485da22f7 100644 --- a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift +++ b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift @@ -99,7 +99,7 @@ class HMTLNodeToNSAttributedString: SafeConverter { /// fileprivate func convertCommentNode(_ node: CommentNode, inheritingAttributes inheritedAttributes: [String:Any]) -> NSAttributedString { let moreLabel = "more" - if node.comment.hasPrefix(moreLabel) { + if node.comment.lowercased().hasPrefix(moreLabel) { var attributes = inheritedAttributes; let moreAttachment = MoreAttachment() let index = moreLabel.endIndex From ddcd0134094c4870978490ad9a8afe7aa753afe5 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 10:29:01 -0300 Subject: [PATCH 17/74] MoreAttachment: Updates constant --- Aztec/Classes/TextKit/MoreAttachment.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/MoreAttachment.swift b/Aztec/Classes/TextKit/MoreAttachment.swift index 9e0347530..e65c86e0f 100644 --- a/Aztec/Classes/TextKit/MoreAttachment.swift +++ b/Aztec/Classes/TextKit/MoreAttachment.swift @@ -24,7 +24,7 @@ open class MoreAttachment: NSTextAttachment /// Comment's Text. /// This is a temporary helper property, and will be removed as soon as we merge MoreAttachment + CommentAttachment. /// - let text = "MORE" + let text = "more" open var message: String = "" From 18ff18c7c150983a8bfd27121490ea6fb02d3ee6 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 11:02:10 -0300 Subject: [PATCH 18/74] TextStorage: Addressing comments --- Aztec/Classes/TextKit/TextStorage.swift | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 9f1571759..3400c3716 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -467,8 +467,8 @@ open class TextStorage: NSTextStorage { if let newAttachment = new, original == nil { dom.replace(range, with: newAttachment.text) - } else if let _ = original, new == nil { - dom.replaceCharacters(inRange: range, withString: String(), preferLeftNode: false) + } else if original != nil, new == nil { + deleteCharacters(in: range) } } @@ -789,6 +789,7 @@ open class TextStorage: NSTextStorage { /// Inserts the MoreAttachment at the specified position /// + @discardableResult open func insertMoreAttachment(at position: Int) -> MoreAttachment { let message = "MORE" let label = NSLocalizedString("MORE", comment: "Text for the center of the more divider") From 192fc59fcd30907a74c53a8e8c83040196ce891e Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 11:04:04 -0300 Subject: [PATCH 19/74] TextStorageTests: Updates unit tests --- AztecTests/TextStorageTests.swift | 39 +++++++++++++++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index cd1942af0..7cbc8585e 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -286,6 +286,20 @@ class TextStorageTests: XCTestCase XCTAssertEqual(html, "
") } + /// This test check if the insertion of antwo horizontal ruler works correctly and the hr tag(s) are inserted + /// + func testInsertTwoHorizontalRulersGeneratesExpectedHTML() { + let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + + storage.insertHorizontalRuler(at: NSRange.zero) + storage.insertHorizontalRuler(at: NSRange.zero) + let html = storage.getHTML() + + XCTAssertEqual(html, "

") + } + /// This test check if the insertion of an horizontal ruler over an image attachment works correctly and the hr tag is inserted /// func testInsertHorizontalRulerOverImage() { @@ -299,4 +313,29 @@ class TextStorageTests: XCTestCase XCTAssertEqual(html, "
") } + + + /// This test check if the insertion of a More Attachment works correctly and the tag is inserted + /// + func testInsertMoreAttachmentCorretlyGeneratesHTMLComment() { + let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + + storage.insertMoreAttachment(at: 0) + let html = storage.getHTML() + + XCTAssertEqual(html, "") + } + + /// This test check if the insertion of a More Attachment works correctly and the tag is inserted + /// + func testInsertTwoMoreAttachmentsDoNotCrashTheEditor() { + let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + + storage.insertMoreAttachment(at: 0) + storage.insertMoreAttachment(at: 0) + } } From ddf2239e6a9c9deaa10a95d2b0cd6f5e2f827b3e Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 11:50:55 -0300 Subject: [PATCH 20/74] TextStorage: Updates Attachment Diff Logic --- Aztec/Classes/TextKit/TextStorage.swift | 17 +++++------------ 1 file changed, 5 insertions(+), 12 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 3400c3716..5a749b033 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -453,23 +453,16 @@ open class TextStorage: NSTextStorage { private func processLineAttachmentDifferences(in range: NSRange, betweenOriginal original: LineAttachment?, andNew new: LineAttachment?) { - let add = original == nil && new != nil - let remove = original != nil && new == nil - - if add { - dom.replaceWithHorizontalRuler(range) - } else if remove { - dom.remove(element: .hr, at: range) - } + dom.replaceWithHorizontalRuler(range) } private func processMoreAttachmentDifferences(in range: NSRange, betweenOriginal original: MoreAttachment?, andNew new: MoreAttachment?) { - if let newAttachment = new, original == nil { - dom.replace(range, with: newAttachment.text) - } else if original != nil, new == nil { - deleteCharacters(in: range) + guard let newAttachment = new else { + return } + + dom.replace(range, with: newAttachment.text) } From c9ff0d0d27053ac67710096427584f44dfa3d83e Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 13:45:02 -0300 Subject: [PATCH 21/74] CommentNode: Implementing deleteCharacters --- Aztec/Classes/Libxml2/DOM/Data/CommentNode.swift | 8 ++++++++ AztecTests/TextStorageTests.swift | 4 ++++ 2 files changed, 12 insertions(+) diff --git a/Aztec/Classes/Libxml2/DOM/Data/CommentNode.swift b/Aztec/Classes/Libxml2/DOM/Data/CommentNode.swift index c7d79c5a1..4c476fb8a 100644 --- a/Aztec/Classes/Libxml2/DOM/Data/CommentNode.swift +++ b/Aztec/Classes/Libxml2/DOM/Data/CommentNode.swift @@ -35,5 +35,13 @@ extension Libxml2 { override func text() -> String { return String(.newline) } + + override func deleteCharacters(inRange range: NSRange) { + guard range.location == 0 && range.length == length() else { + return + } + + removeFromParent() + } } } diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 7cbc8585e..2eb7e38c0 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -337,5 +337,9 @@ class TextStorageTests: XCTestCase storage.insertMoreAttachment(at: 0) storage.insertMoreAttachment(at: 0) + + let html = storage.getHTML() + + XCTAssertEqual(html, "") } } From a628fe00559b7ec37e005c62bfa5283996c28fba Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 14:11:47 -0300 Subject: [PATCH 22/74] TextView: Updates Horizontal Ruler Public API --- Aztec/Classes/TextKit/TextStorage.swift | 4 ++-- Aztec/Classes/TextKit/TextView.swift | 4 ++-- AztecTests/TextStorageTests.swift | 8 ++++---- Example/Example/EditorDemoController.swift | 2 +- 4 files changed, 9 insertions(+), 9 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 5a749b033..6b38ab8e9 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -700,7 +700,7 @@ open class TextStorage: NSTextStorage { /// /// - Parameter range: the range where the element will be inserted /// - func insertHorizontalRuler(at range: NSRange) { + func replaceRangeWithHorizontalRuler(_ range: NSRange) { let line = LineAttachment() let attachmentString = NSAttributedString(attachment: line) @@ -783,7 +783,7 @@ open class TextStorage: NSTextStorage { /// Inserts the MoreAttachment at the specified position /// @discardableResult - open func insertMoreAttachment(at position: Int) -> MoreAttachment { + open func replaceRangeWithMoreAttachment(_ range: NSRange) -> MoreAttachment { let message = "MORE" let label = NSLocalizedString("MORE", comment: "Text for the center of the more divider") diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 5dded66c9..8eddf4931 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -529,8 +529,8 @@ open class TextView: UITextView { /// /// - Parameter range: the range where the ruler will be inserted /// - open func replaceWithHorizontalRuler(at range: NSRange) { - storage.insertHorizontalRuler(at: range) + open func replaceRangeWithHorizontalRuler(_ range: NSRange) { + storage.replaceRangeWithHorizontalRuler(range) let length = NSAttributedString(attachment:NSTextAttachment()).length textStorage.addAttributes(typingAttributes, range: NSMakeRange(range.location, length)) selectedRange = NSMakeRange(range.location + length, 0) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 2eb7e38c0..0eee9fa5a 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -275,7 +275,7 @@ class TextStorageTests: XCTestCase /// This test check if the insertion of an horizontal ruler works correctly and the hr tag is inserted /// - func testInsertHorizontalRuler() { + func testReplaceRangeWithHorizontalRuler() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate @@ -288,13 +288,13 @@ class TextStorageTests: XCTestCase /// This test check if the insertion of antwo horizontal ruler works correctly and the hr tag(s) are inserted /// - func testInsertTwoHorizontalRulersGeneratesExpectedHTML() { + func testReplaceRangeWithHorizontalRulerGeneratesExpectedHTMLWhenExecutedSequentially() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.insertHorizontalRuler(at: NSRange.zero) - storage.insertHorizontalRuler(at: NSRange.zero) + storage.replaceRangeWithHorizontalRuler(.zero) + storage.replaceRangeWithHorizontalRuler(.zero) let html = storage.getHTML() XCTAssertEqual(html, "

") diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index a25f8a607..e2cc22896 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -421,7 +421,7 @@ extension EditorDemoController : Aztec.FormatBarDelegate { } func insertHorizontalRuler() { - richTextView.replaceWithHorizontalRuler(at: richTextView.selectedRange) + richTextView.replaceRangeWithHorizontalRuler(richTextView.selectedRange) } func toggleHeader() { From 012b6584a20d21d2eccf95ddb74435e1e2459a01 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 17 Mar 2017 14:12:02 -0300 Subject: [PATCH 23/74] TextView: Updates Insert More Public API --- Aztec/Classes/TextKit/TextStorage.swift | 3 +-- Aztec/Classes/TextKit/TextView.swift | 8 ++++---- AztecTests/TextStorageTests.swift | 16 ++++++++-------- Example/Example/EditorDemoController.swift | 3 +-- 4 files changed, 14 insertions(+), 16 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 6b38ab8e9..c44288388 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -792,8 +792,7 @@ open class TextStorage: NSTextStorage { attachment.label = NSAttributedString(string: label, attributes: [:]) let payload = NSAttributedString(attachment: attachment) - let target = NSMakeRange(position, 0) - replaceCharacters(in: target, with: payload) + replaceCharacters(in: range, with: payload) return attachment } diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 8eddf4931..1ca6b65a7 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -1007,14 +1007,14 @@ open class TextView: UITextView { /// Inserts the More Comment at the specified position. /// - /// - Parameter position: The character position at which to insert the more attachment. + /// - Parameter range: The character range that must be replaced by a More Attachment. /// /// - Returns: the attachment object that can be used for further calls /// @discardableResult - open func insertMoreAttachment(at position: Int) -> MoreAttachment { - let attachment = storage.insertMoreAttachment(at: position) - let attachmentRange = NSRange(location: position, length: NSAttributedString.lengthOfTextAttachment) + open func replaceRangeWithMoreAttachment(_ range: NSRange) -> MoreAttachment { + let attachment = storage.replaceRangeWithMoreAttachment(range) + let attachmentRange = NSRange(location: range.location, length: NSAttributedString.lengthOfTextAttachment) storage.addAttributes(typingAttributes, range: attachmentRange) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 0eee9fa5a..47663bb3f 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -280,7 +280,7 @@ class TextStorageTests: XCTestCase let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.insertHorizontalRuler(at: NSRange.zero) + storage.replaceRangeWithHorizontalRuler(.zero) let html = storage.getHTML() XCTAssertEqual(html, "
") @@ -302,13 +302,13 @@ class TextStorageTests: XCTestCase /// This test check if the insertion of an horizontal ruler over an image attachment works correctly and the hr tag is inserted /// - func testInsertHorizontalRulerOverImage() { + func testReplaceRangeWithHorizontalRulerRulerOverImage() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate let _ = storage.insertImage(sourceURL: URL(string: "https://wordpress.com")!, atPosition: 0, placeHolderImage: UIImage()) - storage.insertHorizontalRuler(at: NSRange(location: 0, length:1)) + storage.replaceRangeWithHorizontalRuler(NSRange(location: 0, length:1)) let html = storage.getHTML() XCTAssertEqual(html, "
") @@ -317,12 +317,12 @@ class TextStorageTests: XCTestCase /// This test check if the insertion of a More Attachment works correctly and the tag is inserted /// - func testInsertMoreAttachmentCorretlyGeneratesHTMLComment() { + func testReplaceRangeWithMoreAttachmentGeneratesExpectedHTMLComment() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.insertMoreAttachment(at: 0) + storage.replaceRangeWithMoreAttachment(.zero) let html = storage.getHTML() XCTAssertEqual(html, "") @@ -330,13 +330,13 @@ class TextStorageTests: XCTestCase /// This test check if the insertion of a More Attachment works correctly and the tag is inserted /// - func testInsertTwoMoreAttachmentsDoNotCrashTheEditor() { + func testReplaceRangeWithMoreAttachmentDoNotCrashTheEditorWhenCalledSequentially() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.insertMoreAttachment(at: 0) - storage.insertMoreAttachment(at: 0) + storage.replaceRangeWithMoreAttachment(.zero) + storage.replaceRangeWithMoreAttachment(.zero) let html = storage.getHTML() diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index e2cc22896..2a9c37bef 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -493,8 +493,7 @@ extension EditorDemoController : Aztec.FormatBarDelegate { } func insertMoreAttachment() { - let position = richTextView.selectedRange.location - richTextView.insertMoreAttachment(at: position) + richTextView.replaceRangeWithMoreAttachment(richTextView.selectedRange) } func showLinkDialog(forURL url: URL?, title: String?, range: NSRange) { From 9a18874aa955ffc7791d1d05b94cc9d0c5e8ed16 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Mon, 20 Mar 2017 10:58:55 -0300 Subject: [PATCH 24/74] TextStorage: Updates public API --- Aztec/Classes/TextKit/TextStorage.swift | 11 +++++++---- Aztec/Classes/TextKit/TextView.swift | 7 ++----- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index c44288388..f0268d7b9 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -783,16 +783,19 @@ open class TextStorage: NSTextStorage { /// Inserts the MoreAttachment at the specified position /// @discardableResult - open func replaceRangeWithMoreAttachment(_ range: NSRange) -> MoreAttachment { + open func replaceRangeWithMoreAttachment(_ range: NSRange, attributes: [String: Any]) -> MoreAttachment { let message = "MORE" let label = NSLocalizedString("MORE", comment: "Text for the center of the more divider") - + let attachment = MoreAttachment() attachment.message = message attachment.label = NSAttributedString(string: label, attributes: [:]) - let payload = NSAttributedString(attachment: attachment) - replaceCharacters(in: range, with: payload) + let stringWithAttachment = NSAttributedString(attachment: attachment) + replaceCharacters(in: range, with: stringWithAttachment) + + let attachmentRange = NSRange(location: range.location, length: NSAttributedString.lengthOfTextAttachment) + addAttributes(attributes, range: attachmentRange) return attachment } diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 1ca6b65a7..1e1ded514 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -1013,12 +1013,9 @@ open class TextView: UITextView { /// @discardableResult open func replaceRangeWithMoreAttachment(_ range: NSRange) -> MoreAttachment { - let attachment = storage.replaceRangeWithMoreAttachment(range) - let attachmentRange = NSRange(location: range.location, length: NSAttributedString.lengthOfTextAttachment) + let attachment = storage.replaceRangeWithMoreAttachment(range, attributes: typingAttributes) - storage.addAttributes(typingAttributes, range: attachmentRange) - - selectedRange = NSMakeRange(attachmentRange.endLocation, 0) + selectedRange = NSMakeRange(range.location + NSAttributedString.lengthOfTextAttachment, 0) delegate?.textViewDidChange?(self) return attachment From 2bf036e44461656cf5b49db9f828fe9b4c3868a0 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Mon, 20 Mar 2017 10:59:05 -0300 Subject: [PATCH 25/74] DOMString: Renames helper method --- Aztec/Classes/Libxml2/DOMString.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Aztec/Classes/Libxml2/DOMString.swift b/Aztec/Classes/Libxml2/DOMString.swift index 020031198..d58d2d786 100644 --- a/Aztec/Classes/Libxml2/DOMString.swift +++ b/Aztec/Classes/Libxml2/DOMString.swift @@ -493,18 +493,18 @@ extension Libxml2 { rootNode.replaceCharacters(in: range, with: descriptor) } - /// Replaces the specified range with a Horizontal Ruler. + /// Replaces the specified range with a Horizontal Ruler Style. /// /// - Parameters: /// - range: the range to apply the style to. /// func replaceWithHorizontalRuler(_ range: NSRange) { performAsyncUndoable { [weak self] in - self?.replaceSynchronouslyWithHorizontalRuler(range) + self?.replaceSynchronouslyWithHorizontalRulerStyle(range) } } - private func replaceSynchronouslyWithHorizontalRuler(_ range: NSRange) { + private func replaceSynchronouslyWithHorizontalRulerStyle(_ range: NSRange) { let descriptor = ElementNodeDescriptor(elementType: .hr) rootNode.replaceCharacters(in: range, with: descriptor) From 9c77b66e3cd50222a0dda683e03f1ff1d2761a29 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Mon, 20 Mar 2017 11:03:40 -0300 Subject: [PATCH 26/74] TextStorageTests: Fixing unit tests --- AztecTests/TextStorageTests.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 47663bb3f..f008f40bc 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -322,7 +322,7 @@ class TextStorageTests: XCTestCase let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.replaceRangeWithMoreAttachment(.zero) + storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) let html = storage.getHTML() XCTAssertEqual(html, "") @@ -335,8 +335,8 @@ class TextStorageTests: XCTestCase let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.replaceRangeWithMoreAttachment(.zero) - storage.replaceRangeWithMoreAttachment(.zero) + storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) + storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) let html = storage.getHTML() From c2a42a026943564689624ccc18bb63ca7d4f8d17 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Mon, 20 Mar 2017 11:24:51 -0300 Subject: [PATCH 27/74] MoreAttachment: Not case insensitive anymore --- Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift | 4 ++-- Aztec/Classes/TextKit/MoreAttachment.swift | 2 +- AztecTests/TextStorageTests.swift | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift index 485da22f7..c23e47bb6 100644 --- a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift +++ b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift @@ -98,8 +98,8 @@ class HMTLNodeToNSAttributedString: SafeConverter { /// - Returns: the converted node as an `NSAttributedString`. /// fileprivate func convertCommentNode(_ node: CommentNode, inheritingAttributes inheritedAttributes: [String:Any]) -> NSAttributedString { - let moreLabel = "more" - if node.comment.lowercased().hasPrefix(moreLabel) { + let moreLabel = "MORE" + if node.comment.hasPrefix(moreLabel) { var attributes = inheritedAttributes; let moreAttachment = MoreAttachment() let index = moreLabel.endIndex diff --git a/Aztec/Classes/TextKit/MoreAttachment.swift b/Aztec/Classes/TextKit/MoreAttachment.swift index e65c86e0f..9e0347530 100644 --- a/Aztec/Classes/TextKit/MoreAttachment.swift +++ b/Aztec/Classes/TextKit/MoreAttachment.swift @@ -24,7 +24,7 @@ open class MoreAttachment: NSTextAttachment /// Comment's Text. /// This is a temporary helper property, and will be removed as soon as we merge MoreAttachment + CommentAttachment. /// - let text = "more" + let text = "MORE" open var message: String = "" diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 01f3bb838..dabd307fe 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -324,7 +324,7 @@ class TextStorageTests: XCTestCase storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) let html = storage.getHTML() - XCTAssertEqual(html, "") + XCTAssertEqual(html, "") } /// This test check if the insertion of a More Attachment works correctly and the tag is inserted @@ -339,7 +339,7 @@ class TextStorageTests: XCTestCase let html = storage.getHTML() - XCTAssertEqual(html, "") + XCTAssertEqual(html, "") } /// This test verifies if we can delete all the content from a storage object that has html with a comment From 3132e6745fc165c1bca8ed37d7efdab68daf745f Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Mon, 20 Mar 2017 21:54:53 +0000 Subject: [PATCH 28/74] Add the possibility to parse and create spans for colors. --- Aztec.xcodeproj/project.pbxproj | 4 +++ .../HTMLNodeToNSAttributedString.swift | 27 ++++++++++++++++++- .../Classes/Extensions/UIColor+Parsers.swift | 27 +++++++++++++++++++ .../StandardAttributeFormatter.swift | 6 +++++ .../DOM/Data/StandardElementType.swift | 1 + Example/Example/SampleContent/content.html | 1 + 6 files changed, 65 insertions(+), 1 deletion(-) create mode 100644 Aztec/Classes/Extensions/UIColor+Parsers.swift diff --git a/Aztec.xcodeproj/project.pbxproj b/Aztec.xcodeproj/project.pbxproj index 6dedda5a2..5180d8b37 100644 --- a/Aztec.xcodeproj/project.pbxproj +++ b/Aztec.xcodeproj/project.pbxproj @@ -103,6 +103,7 @@ FF7C89A81E3A2B7C000472A8 /* Blockquote.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89A71E3A2B7C000472A8 /* Blockquote.swift */; }; FF7C89AC1E3A47F1000472A8 /* StandardAttributeFormatter.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */; }; FF7C89B01E3BC52F000472A8 /* NSAttributedString+FontTraits.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89AF1E3BC52F000472A8 /* NSAttributedString+FontTraits.swift */; }; + FF7DCB471E80837D00AB77CB /* UIColor+Parsers.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7DCB461E80837D00AB77CB /* UIColor+Parsers.swift */; }; FFA61E891DF18F3D00B71BF6 /* ParagraphStyle.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFA61E881DF18F3D00B71BF6 /* ParagraphStyle.swift */; }; FFA61EC21DF6C1C900B71BF6 /* NSAttributedString+Archive.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFA61EC11DF6C1C900B71BF6 /* NSAttributedString+Archive.swift */; }; FFD0FEB71DAE59A700430586 /* NSLayoutManager+Attachments.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFD0FEB61DAE59A700430586 /* NSLayoutManager+Attachments.swift */; }; @@ -239,6 +240,7 @@ FF7C89A71E3A2B7C000472A8 /* Blockquote.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Blockquote.swift; sourceTree = ""; }; FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = StandardAttributeFormatter.swift; sourceTree = ""; }; FF7C89AF1E3BC52F000472A8 /* NSAttributedString+FontTraits.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSAttributedString+FontTraits.swift"; sourceTree = ""; }; + FF7DCB461E80837D00AB77CB /* UIColor+Parsers.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIColor+Parsers.swift"; sourceTree = ""; }; FFA61E881DF18F3D00B71BF6 /* ParagraphStyle.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ParagraphStyle.swift; sourceTree = ""; }; FFA61EC11DF6C1C900B71BF6 /* NSAttributedString+Archive.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSAttributedString+Archive.swift"; sourceTree = ""; }; FFD0FEB61DAE59A700430586 /* NSLayoutManager+Attachments.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSLayoutManager+Attachments.swift"; sourceTree = ""; }; @@ -426,6 +428,7 @@ B5C99D3E1E72E2E700335355 /* UIStackView+Helpers.swift */, 599F25271D8BC9A1002871D6 /* UITextView+Helpers.swift */, F1A218141E02D5B3000AF5EB /* UndoManager.swift */, + FF7DCB461E80837D00AB77CB /* UIColor+Parsers.swift */, ); path = Extensions; sourceTree = ""; @@ -785,6 +788,7 @@ 599F254C1D8BC9A1002871D6 /* UITextView+Helpers.swift in Sources */, FFA61E891DF18F3D00B71BF6 /* ParagraphStyle.swift in Sources */, F1FA0E751E6EF267009D98EE /* DOMLogic.swift in Sources */, + FF7DCB471E80837D00AB77CB /* UIColor+Parsers.swift in Sources */, FFD436961E300EF800A0E26F /* FontFormatter.swift in Sources */, F1FA0E771E6EF29B009D98EE /* DOMInspector.swift in Sources */, 599F25451D8BC9A1002871D6 /* Libxml2.swift in Sources */, diff --git a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift index 47b6e15fb..eba1fec7f 100644 --- a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift +++ b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift @@ -211,6 +211,15 @@ class HMTLNodeToNSAttributedString: SafeConverter { attributeValue = attachment } + if node.isNodeType(.span) { + if let elementStyle = node.valueForStringAttribute(named: "style") { + let styles = parseStyle(style: elementStyle) + if !styles.isEmpty, let colorString = styles["color"] { + attributeValue = UIColor(hexString: colorString) + } + } + } + for (key, formatter) in elementToFormattersMap { if node.isNodeType(key) { if let standardValueFormatter = formatter as? StandardAttributeFormatter, @@ -240,6 +249,22 @@ class HMTLNodeToNSAttributedString: SafeConverter { .h3: HeaderFormatter(headerLevel: .h3), .h4: HeaderFormatter(headerLevel: .h4), .h5: HeaderFormatter(headerLevel: .h5), - .h6: HeaderFormatter(headerLevel: .h6) + .h6: HeaderFormatter(headerLevel: .h6), + .span: ColorFormatter() ] + + func parseStyle(style: String) -> [String: String] { + var stylesDictionary = [String: String]() + let styleAttributes = style.components(separatedBy: ";") + for sytleAttribute in styleAttributes { + let keyValue = sytleAttribute.components(separatedBy: ":") + guard keyValue.count == 2, + let key = keyValue.first?.trimmingCharacters(in: CharacterSet.whitespaces), + let value = keyValue.last?.trimmingCharacters(in: CharacterSet.whitespaces) else { + continue + } + stylesDictionary[key] = value + } + return stylesDictionary + } } diff --git a/Aztec/Classes/Extensions/UIColor+Parsers.swift b/Aztec/Classes/Extensions/UIColor+Parsers.swift new file mode 100644 index 000000000..7d181ba4a --- /dev/null +++ b/Aztec/Classes/Extensions/UIColor+Parsers.swift @@ -0,0 +1,27 @@ +import UIKit + +extension UIColor { + + /// Creates a color based on a hexString. If the string is not a valid hexColor it return nils + /// Example of colors: #FF0000, #00FF0000 + /// + convenience init?(hexString: String) { + + let hex = hexString.trimmingCharacters(in: CharacterSet.alphanumerics.inverted) + var int = UInt32() + Scanner(string: hex).scanHexInt32(&int) + let a, r, g, b: UInt32 + switch hex.characters.count { + case 3: // RGB (12-bit) + (a, r, g, b) = (255, (int >> 8) * 17, (int >> 4 & 0xF) * 17, (int & 0xF) * 17) + case 6: // RGB (24-bit) + (a, r, g, b) = (255, int >> 16, int >> 8 & 0xFF, int & 0xFF) + case 8: // ARGB (32-bit) + (a, r, g, b) = (int >> 24, int >> 16 & 0xFF, int >> 8 & 0xFF, int & 0xFF) + default: + return nil + } + self.init(red: CGFloat(r) / 255, green: CGFloat(g) / 255, blue: CGFloat(b) / 255, alpha: CGFloat(a) / 255) + } + +} diff --git a/Aztec/Classes/Formatters/StandardAttributeFormatter.swift b/Aztec/Classes/Formatters/StandardAttributeFormatter.swift index 8605bb0ae..ae6a404d1 100644 --- a/Aztec/Classes/Formatters/StandardAttributeFormatter.swift +++ b/Aztec/Classes/Formatters/StandardAttributeFormatter.swift @@ -70,3 +70,9 @@ class HRFormatter: StandardAttributeFormatter { } } +class ColorFormatter: StandardAttributeFormatter { + init(color: UIColor = .black) { + super.init(elementType: .span, attributeKey: NSForegroundColorAttributeName, attributeValue: color) + } +} + diff --git a/Aztec/Classes/Libxml2/DOM/Data/StandardElementType.swift b/Aztec/Classes/Libxml2/DOM/Data/StandardElementType.swift index db16e2f16..2da1e5a9a 100644 --- a/Aztec/Classes/Libxml2/DOM/Data/StandardElementType.swift +++ b/Aztec/Classes/Libxml2/DOM/Data/StandardElementType.swift @@ -35,6 +35,7 @@ extension Libxml2 { case p = "p" case pre = "pre" case s = "s" + case span = "span" case strike = "strike" case strong = "strong" case table = "table" diff --git a/Example/Example/SampleContent/content.html b/Example/Example/SampleContent/content.html index 9ca412942..7b7b276ec 100644 --- a/Example/Example/SampleContent/content.html +++ b/Example/Example/SampleContent/content.html @@ -14,6 +14,7 @@

Character Styles

Italic text
Underlined text
Strikethrough
+Colors
I'm a link!
From 824a4f82ca34470901d1bbd7fa38ca7c96c6399c Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Tue, 21 Mar 2017 10:29:09 -0300 Subject: [PATCH 29/74] Fixes a crash. --- Aztec/Classes/TextKit/TextStorage.swift | 2 +- AztecTests/TextViewTests.swift | 20 +++++++++++++++++++- 2 files changed, 20 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index f0268d7b9..01e900f2f 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -263,7 +263,7 @@ open class TextStorage: NSTextStorage { dom.deleteBlockSeparator(at: targetDomRange.location) } - applyStylesToDom(from: preprocessedString, startingAt: range.location) + applyStylesToDom(from: domString, startingAt: range.location) } detectAttachmentRemoved(in: range) diff --git a/AztecTests/TextViewTests.swift b/AztecTests/TextViewTests.swift index 826cbc82f..e91e28f29 100644 --- a/AztecTests/TextViewTests.swift +++ b/AztecTests/TextViewTests.swift @@ -2,7 +2,7 @@ import XCTest @testable import Aztec import Gridicons -class AztecVisualtextViewTests: XCTestCase { +class AztecVisualTextViewTests: XCTestCase { override func setUp() { super.setUp() @@ -16,6 +16,12 @@ class AztecVisualtextViewTests: XCTestCase { // MARK: - TextView construction + func createEmptyTextView() -> Aztec.TextView { + let richTextView = Aztec.TextView(defaultFont: UIFont.systemFont(ofSize: 14), defaultMissingImage: Gridicon.iconOfType(.attachment)) + + return richTextView + } + func createTextView(withHTML html: String) -> Aztec.TextView { let richTextView = Aztec.TextView(defaultFont: UIFont.systemFont(ofSize: 14), defaultMissingImage: Gridicon.iconOfType(.attachment)) @@ -242,6 +248,18 @@ class AztecVisualtextViewTests: XCTestCase { XCTAssert(!textView.formatIdentifiersSpanningRange(range).contains(.unorderedlist)) } + /// This test was created to prevent regressions related to this issue: + /// https://github.com/wordpress-mobile/WordPress-Aztec-iOS/issues/350 + /// + func testToggleBlockquoteAndStrikethrough() { + let textView = createEmptyTextView() + + textView.toggleStrikethrough(range: NSRange.zero) + textView.toggleBlockquote(range: NSRange.zero) + + // The test not crashing would be successful. + } + // MARK: - Test Attributes Exist func check(textView: TextView, range:NSRange, forIndentifier identifier: FormattingIdentifier) -> Bool { From 8c93ba3d0021416923780f75f61f1e8d579f20c9 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 13:35:56 +0000 Subject: [PATCH 30/74] Add tests for parsing colors in hex. --- Aztec.xcodeproj/project.pbxproj | 4 ++ .../Classes/Extensions/UIColor+Parsers.swift | 6 +- .../Extensions/UIColorHexParserTests.swift | 61 +++++++++++++++++++ 3 files changed, 69 insertions(+), 2 deletions(-) create mode 100644 AztecTests/Extensions/UIColorHexParserTests.swift diff --git a/Aztec.xcodeproj/project.pbxproj b/Aztec.xcodeproj/project.pbxproj index 5180d8b37..3d0c525c3 100644 --- a/Aztec.xcodeproj/project.pbxproj +++ b/Aztec.xcodeproj/project.pbxproj @@ -104,6 +104,7 @@ FF7C89AC1E3A47F1000472A8 /* StandardAttributeFormatter.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */; }; FF7C89B01E3BC52F000472A8 /* NSAttributedString+FontTraits.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89AF1E3BC52F000472A8 /* NSAttributedString+FontTraits.swift */; }; FF7DCB471E80837D00AB77CB /* UIColor+Parsers.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7DCB461E80837D00AB77CB /* UIColor+Parsers.swift */; }; + FF7DCB4B1E815F9400AB77CB /* UIColorHexParserTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7DCB4A1E815F9400AB77CB /* UIColorHexParserTests.swift */; }; FFA61E891DF18F3D00B71BF6 /* ParagraphStyle.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFA61E881DF18F3D00B71BF6 /* ParagraphStyle.swift */; }; FFA61EC21DF6C1C900B71BF6 /* NSAttributedString+Archive.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFA61EC11DF6C1C900B71BF6 /* NSAttributedString+Archive.swift */; }; FFD0FEB71DAE59A700430586 /* NSLayoutManager+Attachments.swift in Sources */ = {isa = PBXBuildFile; fileRef = FFD0FEB61DAE59A700430586 /* NSLayoutManager+Attachments.swift */; }; @@ -241,6 +242,7 @@ FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = StandardAttributeFormatter.swift; sourceTree = ""; }; FF7C89AF1E3BC52F000472A8 /* NSAttributedString+FontTraits.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSAttributedString+FontTraits.swift"; sourceTree = ""; }; FF7DCB461E80837D00AB77CB /* UIColor+Parsers.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIColor+Parsers.swift"; sourceTree = ""; }; + FF7DCB4A1E815F9400AB77CB /* UIColorHexParserTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = UIColorHexParserTests.swift; path = Extensions/UIColorHexParserTests.swift; sourceTree = ""; }; FFA61E881DF18F3D00B71BF6 /* ParagraphStyle.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ParagraphStyle.swift; sourceTree = ""; }; FFA61EC11DF6C1C900B71BF6 /* NSAttributedString+Archive.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSAttributedString+Archive.swift"; sourceTree = ""; }; FFD0FEB61DAE59A700430586 /* NSLayoutManager+Attachments.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "NSLayoutManager+Attachments.swift"; sourceTree = ""; }; @@ -605,6 +607,7 @@ F111DF101E3BAC8E003FB794 /* NSAttributedStringAttributeRangesTests.swift */, F18733C71DA09737005AEB80 /* NSRangeComparisonTests.swift */, FF152D8D1E68552A00FF596C /* StringRangeConversionTests.swift */, + FF7DCB4A1E815F9400AB77CB /* UIColorHexParserTests.swift */, ); name = Extensions; sourceTree = ""; @@ -828,6 +831,7 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + FF7DCB4B1E815F9400AB77CB /* UIColorHexParserTests.swift in Sources */, 594C9D741D8BE6C700D74542 /* InNodeConverterTests.swift in Sources */, B5F84B631E706B720089A76C /* NSAttributedStringAnalyzerTests.swift in Sources */, 59FEA0751D8BDFA700D138DF /* NodeTests.swift in Sources */, diff --git a/Aztec/Classes/Extensions/UIColor+Parsers.swift b/Aztec/Classes/Extensions/UIColor+Parsers.swift index 7d181ba4a..9b7ea4709 100644 --- a/Aztec/Classes/Extensions/UIColor+Parsers.swift +++ b/Aztec/Classes/Extensions/UIColor+Parsers.swift @@ -2,14 +2,16 @@ import UIKit extension UIColor { - /// Creates a color based on a hexString. If the string is not a valid hexColor it return nils + /// Creates a color based on a hexString. If the string is not a valid hexColor it return nil /// Example of colors: #FF0000, #00FF0000 /// convenience init?(hexString: String) { let hex = hexString.trimmingCharacters(in: CharacterSet.alphanumerics.inverted) var int = UInt32() - Scanner(string: hex).scanHexInt32(&int) + if !Scanner(string: hex).scanHexInt32(&int) { + return nil + } let a, r, g, b: UInt32 switch hex.characters.count { case 3: // RGB (12-bit) diff --git a/AztecTests/Extensions/UIColorHexParserTests.swift b/AztecTests/Extensions/UIColorHexParserTests.swift new file mode 100644 index 000000000..c8de37b83 --- /dev/null +++ b/AztecTests/Extensions/UIColorHexParserTests.swift @@ -0,0 +1,61 @@ +import XCTest +@testable import Aztec + +class UIColorHexParserTests: XCTestCase { + + override func setUp() { + super.setUp() + // Put setup code here. This method is called before the invocation of each test method in the class. + } + + override func tearDown() { + // Put teardown code here. This method is called after the invocation of each test method in the class. + super.tearDown() + } + + func testParseOf24bitsHexColors() { + var color = UIColor(hexString: "#FF0000") + + XCTAssertEqual(color, UIColor.red) + + color = UIColor(hexString: "#00FF00") + + XCTAssertEqual(color, UIColor.green) + + color = UIColor(hexString: "#0000FF") + + XCTAssertEqual(color, UIColor.blue) + + color = UIColor(hexString: "#FFFFFF") + + XCTAssertEqual(color, UIColor.init(colorLiteralRed: 1, green: 1, blue: 1, alpha: 1)) + } + + func testParseOf32bitsHexColors() { + var color = UIColor(hexString: "#FFFF0000") + + XCTAssertEqual(color, UIColor.red) + + color = UIColor(hexString: "#FF00FF00") + + XCTAssertEqual(color, UIColor.green) + + color = UIColor(hexString: "#FF0000FF") + + XCTAssertEqual(color, UIColor.blue) + + color = UIColor(hexString: "#FFFFFFFF") + + XCTAssertEqual(color, UIColor.init(colorLiteralRed: 1, green: 1, blue: 1, alpha: 1)) + } + + func testFailingColor() { + var color = UIColor(hexString: "#") + + XCTAssertEqual(color, nil) + + color = UIColor(hexString: "#ZZZZZZ") + + XCTAssertEqual(color, nil) + } +} From c8f69dca979e1ca9ed697df37a37e3e5d569eeb5 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 12:40:23 +0000 Subject: [PATCH 31/74] Propagate formatters application range changes correctly --- .../Formatters/AttributeFormatter.swift | 31 ++++++++++++------- 1 file changed, 19 insertions(+), 12 deletions(-) diff --git a/Aztec/Classes/Formatters/AttributeFormatter.swift b/Aztec/Classes/Formatters/AttributeFormatter.swift index 07d04ac07..0cd0abb76 100644 --- a/Aztec/Classes/Formatters/AttributeFormatter.swift +++ b/Aztec/Classes/Formatters/AttributeFormatter.swift @@ -21,7 +21,9 @@ protocol AttributeFormatter { /// - text: Text that should be formatted. /// - range: Segment of text which format should be toggled. /// - @discardableResult func toggle(in text: NSMutableAttributedString, at range: NSRange) -> NSRange? + /// - Returns: the full range where the toggle was applied + /// + @discardableResult func toggle(in text: NSMutableAttributedString, at range: NSRange) -> NSRange /// Apply or removes formatter attributes to the provided attribute dictionary and returns it. /// @@ -49,11 +51,11 @@ protocol AttributeFormatter { /// Applies the Formatter's Attributes into a given string, at the specified range. /// - func applyAttributes(to string: NSMutableAttributedString, at range: NSRange) + @discardableResult func applyAttributes(to string: NSMutableAttributedString, at range: NSRange) -> NSRange /// Removes the Formatter's Attributes from a given string, at the specified range. /// - func removeAttributes(from string: NSMutableAttributedString, at range: NSRange) + @discardableResult func removeAttributes(from string: NSMutableAttributedString, at range: NSRange) -> NSRange /// Checks if the attribute is present in a dictionary of attributes. /// @@ -111,13 +113,15 @@ extension AttributeFormatter { /// Applies the Formatter's Attributes into a given string, at the specified range. /// - func applyAttributes(to text: NSMutableAttributedString, at range: NSRange) { + /// - Returns: the full range where the attributes where applied + /// + @discardableResult func applyAttributes(to text: NSMutableAttributedString, at range: NSRange) -> NSRange { var rangeToApply = applicationRange(for: range, in: text) if worksInEmptyRange() && ( rangeToApply.length == 0 || text.length == 0) { let placeholder = placeholderForEmptyLine(using: placeholderAttributes) text.insert(placeholder, at: rangeToApply.location) - rangeToApply = NSMakeRange(text.length - 1, 1) + rangeToApply = NSMakeRange(rangeToApply.location, placeholder.length) } text.enumerateAttributes(in: rangeToApply, options: []) { (attributes, range, stop) in @@ -125,11 +129,15 @@ extension AttributeFormatter { let attributes = apply(to: currentAttributes) text.addAttributes(attributes, range: range) } + + return rangeToApply } /// Removes the Formatter's Attributes from a given string, at the specified range. /// - func removeAttributes(from text: NSMutableAttributedString, at range: NSRange) { + /// - Returns: the full range where the attributes where removed + /// + @discardableResult func removeAttributes(from text: NSMutableAttributedString, at range: NSRange) -> NSRange { let rangeToApply = applicationRange(for: range, in: text) text.enumerateAttributes(in: rangeToApply, options: []) { (attributes, range, stop) in let currentAttributes = text.attributes(at: range.location, effectiveRange: nil) @@ -144,22 +152,21 @@ extension AttributeFormatter { text.addAttributes(attributes, range: range) } + return rangeToApply } /// Toggles the Attribute Format, into a given string, at the specified range. /// @discardableResult - func toggle(in text: NSMutableAttributedString, at range: NSRange) -> NSRange? { + func toggle(in text: NSMutableAttributedString, at range: NSRange) -> NSRange { //We decide if we need to apply or not the attribute based on the value on the initial position of the range let shouldApply = shouldApplyAttributes(to: text, at: range) if shouldApply { - applyAttributes(to: text, at: range) + return applyAttributes(to: text, at: range) } else { - removeAttributes(from: text, at: range) - } - - return nil + return removeAttributes(from: text, at: range) + } } } From 46f5b5a931855198c64e3446d8e2478a2825a6d5 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 12:55:24 +0000 Subject: [PATCH 32/74] Fix range change on toggle. --- Aztec/Classes/TextKit/TextStorage.swift | 2 +- Aztec/Classes/TextKit/TextView.swift | 5 ++--- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 01e900f2f..176586f0a 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -665,7 +665,7 @@ open class TextStorage: NSTextStorage { } // MARK: - Styles: Toggling - @discardableResult func toggle(formatter: AttributeFormatter, at range: NSRange) -> NSRange? { + @discardableResult func toggle(formatter: AttributeFormatter, at range: NSRange) -> NSRange { let applicationRange = formatter.applicationRange(for: range, in: self) if applicationRange.length == 0, !formatter.worksInEmptyRange() { return applicationRange diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 1e1ded514..9f0af55eb 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -426,9 +426,8 @@ open class TextView: UITextView { // MARK: - Formatting func toggle(formatter: AttributeFormatter, atRange range: NSRange) { - let newSelectedRange = storage.toggle(formatter: formatter, at: range) - selectedRange = newSelectedRange ?? selectedRange - if selectedRange.length == 0 { + let applicationRange = storage.toggle(formatter: formatter, at: range) + if applicationRange.length == 0 { typingAttributes = formatter.toggle(in: typingAttributes) } else { // NOTE: We are making sure that the selectedRange location is inside the string From 221047c0bd533a680a82b4f3116ab557aed67200 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 11:02:05 -0300 Subject: [PATCH 33/74] MoreAttachment: Comment Node Text is now lowercase --- Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift | 5 ++--- Aztec/Classes/TextKit/MoreAttachment.swift | 2 +- Aztec/Classes/TextKit/TextStorage.swift | 6 +----- 3 files changed, 4 insertions(+), 9 deletions(-) diff --git a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift index c23e47bb6..756dc90d5 100644 --- a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift +++ b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift @@ -98,11 +98,10 @@ class HMTLNodeToNSAttributedString: SafeConverter { /// - Returns: the converted node as an `NSAttributedString`. /// fileprivate func convertCommentNode(_ node: CommentNode, inheritingAttributes inheritedAttributes: [String:Any]) -> NSAttributedString { - let moreLabel = "MORE" - if node.comment.hasPrefix(moreLabel) { + if node.comment.hasPrefix(MoreAttachment.commentNodeText) { var attributes = inheritedAttributes; let moreAttachment = MoreAttachment() - let index = moreLabel.endIndex + let index = MoreAttachment.commentNodeText.endIndex moreAttachment.message = node.comment.substring(from: index) moreAttachment.label = NSAttributedString(string: NSLocalizedString("MORE", comment: "Text for the center of the more divider"), attributes: defaultAttributes) attributes[NSAttachmentAttributeName] = moreAttachment diff --git a/Aztec/Classes/TextKit/MoreAttachment.swift b/Aztec/Classes/TextKit/MoreAttachment.swift index 9e0347530..9466ad28d 100644 --- a/Aztec/Classes/TextKit/MoreAttachment.swift +++ b/Aztec/Classes/TextKit/MoreAttachment.swift @@ -24,7 +24,7 @@ open class MoreAttachment: NSTextAttachment /// Comment's Text. /// This is a temporary helper property, and will be removed as soon as we merge MoreAttachment + CommentAttachment. /// - let text = "MORE" + static let commentNodeText = "more" open var message: String = "" diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 01e900f2f..92fa54128 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -458,11 +458,7 @@ open class TextStorage: NSTextStorage { private func processMoreAttachmentDifferences(in range: NSRange, betweenOriginal original: MoreAttachment?, andNew new: MoreAttachment?) { - guard let newAttachment = new else { - return - } - - dom.replace(range, with: newAttachment.text) + dom.replace(range, with: MoreAttachment.commentNodeText) } From 4b920ce981cea2c14646e0a95a6893f4b02d393b Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 14:13:07 +0000 Subject: [PATCH 34/74] Add unit tests for range checking. --- AztecTests/StringRangeConversionTests.swift | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/AztecTests/StringRangeConversionTests.swift b/AztecTests/StringRangeConversionTests.swift index 13ade761c..dd9be98f9 100644 --- a/AztecTests/StringRangeConversionTests.swift +++ b/AztecTests/StringRangeConversionTests.swift @@ -139,5 +139,19 @@ class StringRangeConversionTests: XCTestCase { XCTAssertEqual("Hello ", wordCaptured) } + func testLocationBeforeAtLimits() { + // test at start of string + let string = "" + let nonExistentLocation = string.location(before: 0) + XCTAssertNil(nonExistentLocation) + } + + func testLocationAfterAtLimits() { + // test at start of string + let string = "" + let nonExistentLocation = string.location(after: 0) + XCTAssertNil(nonExistentLocation) + } + } From ee7628c39d9199c4659accc67db1576d969eb0f7 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 14:15:10 +0000 Subject: [PATCH 35/74] Add fix for access at the edges on after and before. --- Aztec/Classes/Extensions/String+RangeConversion.swift | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/Extensions/String+RangeConversion.swift b/Aztec/Classes/Extensions/String+RangeConversion.swift index f9e1cd5b1..c3ba519c6 100644 --- a/Aztec/Classes/Extensions/String+RangeConversion.swift +++ b/Aztec/Classes/Extensions/String+RangeConversion.swift @@ -33,7 +33,9 @@ extension String } func location(after: Int) -> Int? { - guard let currentIndex = indexFromLocation(after) else { + guard let currentIndex = indexFromLocation(after), + currentIndex != endIndex + else { return nil } let afterIndex = index(after: currentIndex) @@ -42,9 +44,12 @@ extension String } func location(before: Int) -> Int? { - guard let currentIndex = indexFromLocation(before) else { + guard let currentIndex = indexFromLocation(before), + currentIndex != startIndex + else { return nil } + let beforeIndex = index(before: currentIndex) let before16 = beforeIndex.samePosition(in: utf16) return utf16.distance(from: utf16.startIndex, to: before16) From 13d68751b782fccf1cb7d804782209ba469a96d4 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 14:25:21 +0000 Subject: [PATCH 36/74] Improved code layout. --- Aztec/Classes/Extensions/String+RangeConversion.swift | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/Aztec/Classes/Extensions/String+RangeConversion.swift b/Aztec/Classes/Extensions/String+RangeConversion.swift index c3ba519c6..508f36066 100644 --- a/Aztec/Classes/Extensions/String+RangeConversion.swift +++ b/Aztec/Classes/Extensions/String+RangeConversion.swift @@ -33,9 +33,7 @@ extension String } func location(after: Int) -> Int? { - guard let currentIndex = indexFromLocation(after), - currentIndex != endIndex - else { + guard let currentIndex = indexFromLocation(after), currentIndex != endIndex else { return nil } let afterIndex = index(after: currentIndex) @@ -44,9 +42,7 @@ extension String } func location(before: Int) -> Int? { - guard let currentIndex = indexFromLocation(before), - currentIndex != startIndex - else { + guard let currentIndex = indexFromLocation(before), currentIndex != startIndex else { return nil } From 3cfe7ef2fc960af5d0d3684cf918050972f7796b Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Tue, 21 Mar 2017 14:26:12 +0000 Subject: [PATCH 37/74] Add test for crash. --- AztecTests/TextViewTests.swift | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/AztecTests/TextViewTests.swift b/AztecTests/TextViewTests.swift index e91e28f29..f1d90108f 100644 --- a/AztecTests/TextViewTests.swift +++ b/AztecTests/TextViewTests.swift @@ -527,4 +527,13 @@ class AztecVisualTextViewTests: XCTestCase { XCTAssertEqual(textView.getHTML(), "\(linkTitle)") } + + func testToggleBlockquoteWriteOneCharAndDelete() { + let textView = createEmptyTextView() + + textView.toggleBlockquote(range: NSRange.zero) + textView.insertText("A") + textView.deleteBackward() + // The test not crashing would be successful. + } } From e532727f190636fe054402ca431b6bc29e58eb05 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 11:41:25 -0300 Subject: [PATCH 38/74] TextStorageTests: Fixes Tests --- AztecTests/TextStorageTests.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index dabd307fe..01f3bb838 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -324,7 +324,7 @@ class TextStorageTests: XCTestCase storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) let html = storage.getHTML() - XCTAssertEqual(html, "") + XCTAssertEqual(html, "") } /// This test check if the insertion of a More Attachment works correctly and the tag is inserted @@ -339,7 +339,7 @@ class TextStorageTests: XCTestCase let html = storage.getHTML() - XCTAssertEqual(html, "") + XCTAssertEqual(html, "") } /// This test verifies if we can delete all the content from a storage object that has html with a comment From 534ac3e0a2150873909632e7eb236f097ff0a560 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:46:54 -0300 Subject: [PATCH 39/74] Nukes MoreAttachment --- Aztec/Classes/TextKit/MoreAttachment.swift | 86 ---------------------- 1 file changed, 86 deletions(-) delete mode 100644 Aztec/Classes/TextKit/MoreAttachment.swift diff --git a/Aztec/Classes/TextKit/MoreAttachment.swift b/Aztec/Classes/TextKit/MoreAttachment.swift deleted file mode 100644 index 9466ad28d..000000000 --- a/Aztec/Classes/TextKit/MoreAttachment.swift +++ /dev/null @@ -1,86 +0,0 @@ -import Foundation -import UIKit - -/// Custom text attachment. -/// -open class MoreAttachment: NSTextAttachment -{ - fileprivate var glyphImage: UIImage? - - /// The color to use when drawing progress indicators - /// - open var color: UIColor = UIColor.gray - - /// A message to display overlaid on top of the image - /// - open var label: NSAttributedString = NSAttributedString(string: "MORE") { - willSet { - if newValue != label { - glyphImage = nil - } - } - } - - /// Comment's Text. - /// This is a temporary helper property, and will be removed as soon as we merge MoreAttachment + CommentAttachment. - /// - static let commentNodeText = "more" - - - open var message: String = "" - - // MARK: - NSTextAttachmentContainer - - override open func image(forBounds imageBounds: CGRect, textContainer: NSTextContainer?, characterIndex charIndex: Int) -> UIImage? { - - if let cachedImage = glyphImage, imageBounds.size.equalTo(cachedImage.size) { - return cachedImage - } - - glyphImage = glyph(forBounds: imageBounds) - - return glyphImage - } - - fileprivate func glyph(forBounds bounds: CGRect) -> UIImage? { - - let size = bounds.size - - UIGraphicsBeginImageContextWithOptions(bounds.size, false, 0) - - let colorMessage = NSMutableAttributedString(attributedString: label) - colorMessage.addAttribute(NSForegroundColorAttributeName, value: color, range: label.rangeOfEntireString) - let textRect = colorMessage.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) - let textPosition = CGPoint(x: ((size.width - textRect.width) / 2), y: ((size.height - textRect.height) / 2) ) - colorMessage.draw(in: CGRect(origin: textPosition , size: CGSize(width: size.width, height: textRect.size.height))) - - let path = UIBezierPath() - - let dashWidth: CGFloat = 8.0 - let dashes: [ CGFloat ] = [ dashWidth, dashWidth ] - path.setLineDash(dashes, count: dashes.count, phase: 0.0) - path.lineWidth = 2.0 - let centerY = round(size.height / 2.0) - path.move(to: CGPoint(x:0, y: centerY)) - path.addLine(to: CGPoint(x: ((size.width - textRect.width) / 2) - dashWidth, y: centerY)) - - path.move(to: CGPoint(x:((size.width + textRect.width) / 2) + dashWidth, y: centerY)) - path.addLine(to: CGPoint(x: size.width, y: centerY)) - - color.setStroke() - path.stroke() - - let result = UIGraphicsGetImageFromCurrentImageContext() - UIGraphicsEndImageContext() - return result; - } - - override open func attachmentBounds(for textContainer: NSTextContainer?, proposedLineFragment lineFrag: CGRect, glyphPosition position: CGPoint, characterIndex charIndex: Int) -> CGRect { - - let padding = textContainer?.lineFragmentPadding ?? 0 - let width = lineFrag.width - padding * 2 - let height:CGFloat = 44.0 - - return CGRect(origin: CGPoint.zero, size: CGSize(width: width, height: height)) - } -} From 6dc4584d6f0d5640f2f8d3a2daf34a8c57309ffe Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:50:53 -0300 Subject: [PATCH 40/74] EditorDemoController: Wiring new Insert Comment API --- Example/Example/EditorDemoController.swift | 50 +++++++++++++--------- 1 file changed, 29 insertions(+), 21 deletions(-) diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index 2a9c37bef..3c1c902df 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -6,16 +6,12 @@ import UIKit import MobileCoreServices class EditorDemoController: UIViewController { - static let margin = CGFloat(20) - static let defaultContentFont = UIFont.systemFont(ofSize: 14) - fileprivate var mediaErrorMode = false - lazy var headers: [HeaderFormatter.HeaderType] = [.none, .h1, .h2, .h3, .h4, .h5, .h6] + fileprivate var mediaErrorMode = false fileprivate(set) lazy var richTextView: Aztec.TextView = { - let defaultMissingImage = Gridicon.iconOfType(.image) - let textView = Aztec.TextView(defaultFont: type(of: self).defaultContentFont, defaultMissingImage: defaultMissingImage) + let textView = Aztec.TextView(defaultFont: Constants.defaultContentFont, defaultMissingImage: Constants.defaultMissingImage) let toolbar = self.createToolbar(htmlMode: false) @@ -167,24 +163,24 @@ class EditorDemoController: UIViewController { private func configureConstraints() { NSLayoutConstraint.activate([ - titleTextField.leftAnchor.constraint(equalTo: view.leftAnchor, constant: type(of: self).margin), - titleTextField.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -type(of: self).margin), - titleTextField.topAnchor.constraint(equalTo: view.topAnchor, constant: type(of: self).margin), + titleTextField.leftAnchor.constraint(equalTo: view.leftAnchor, constant: Constants.margin), + titleTextField.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -Constants.margin), + titleTextField.topAnchor.constraint(equalTo: view.topAnchor, constant: Constants.margin), titleTextField.heightAnchor.constraint(equalToConstant: titleTextField.font!.lineHeight) ]) NSLayoutConstraint.activate([ - separatorView.leftAnchor.constraint(equalTo: view.leftAnchor, constant: type(of: self).margin), - separatorView.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -type(of: self).margin), - separatorView.topAnchor.constraint(equalTo: titleTextField.bottomAnchor, constant: type(of: self).margin), + separatorView.leftAnchor.constraint(equalTo: view.leftAnchor, constant: Constants.margin), + separatorView.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -Constants.margin), + separatorView.topAnchor.constraint(equalTo: titleTextField.bottomAnchor, constant: Constants.margin), separatorView.heightAnchor.constraint(equalToConstant: separatorView.frame.height) ]) NSLayoutConstraint.activate([ - richTextView.leftAnchor.constraint(equalTo: view.leftAnchor, constant: type(of: self).margin), - richTextView.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -type(of: self).margin), - richTextView.topAnchor.constraint(equalTo: separatorView.bottomAnchor, constant: type(of: self).margin), - richTextView.bottomAnchor.constraint(equalTo: view.bottomAnchor, constant: -type(of: self).margin) + richTextView.leftAnchor.constraint(equalTo: view.leftAnchor, constant: Constants.margin), + richTextView.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -Constants.margin), + richTextView.topAnchor.constraint(equalTo: separatorView.bottomAnchor, constant: Constants.margin), + richTextView.bottomAnchor.constraint(equalTo: view.bottomAnchor, constant: -Constants.margin) ]) NSLayoutConstraint.activate([ @@ -198,7 +194,7 @@ class EditorDemoController: UIViewController { private func configureDefaultProperties(for textView: UITextView, using formatBar: Aztec.FormatBar, accessibilityLabel: String) { textView.accessibilityLabel = accessibilityLabel - textView.font = EditorDemoController.defaultContentFont + textView.font = Constants.defaultContentFont textView.inputAccessoryView = formatBar textView.keyboardDismissMode = .interactive textView.textColor = UIColor.darkText @@ -430,17 +426,17 @@ extension EditorDemoController : Aztec.FormatBarDelegate { changeRichTextInputView(to: nil) return } - let headerOptions = headers.map { (headerType) -> NSAttributedString in + let headerOptions = Constants.headers.map { (headerType) -> NSAttributedString in NSAttributedString(string: headerType.description, attributes:[NSFontAttributeName: UIFont.systemFont(ofSize: headerType.fontSize)]) } let headerPicker = OptionsTableView(frame: CGRect(x: 0, y: 0, width: self.view.frame.size.width, height: 200), options: headerOptions) headerPicker.autoresizingMask = [.flexibleHeight, .flexibleWidth] headerPicker.onSelect = { selected in - self.richTextView.toggleHeader(self.headers[selected], range: self.richTextView.selectedRange) + self.richTextView.toggleHeader(Constants.headers[selected], range: self.richTextView.selectedRange) self.changeRichTextInputView(to: nil) } - if let selectedHeader = headers.index(of:self.headerLevelForSelectedText()) { + if let selectedHeader = Constants.headers.index(of: self.headerLevelForSelectedText()) { headerPicker.selectRow(at: IndexPath(row: selectedHeader, section: 0), animated: false, scrollPosition: .top) } changeRichTextInputView(to: headerPicker) @@ -493,7 +489,7 @@ extension EditorDemoController : Aztec.FormatBarDelegate { } func insertMoreAttachment() { - richTextView.replaceRangeWithMoreAttachment(richTextView.selectedRange) + richTextView.replaceRangeWithCommentAttachment(richTextView.selectedRange, text: Constants.moreAttachmentText) } func showLinkDialog(forURL url: URL?, title: String?, range: NSRange) { @@ -879,3 +875,15 @@ private extension EditorDemoController present(navigationController, animated: true, completion: nil) } } + + +extension EditorDemoController { + + struct Constants { + static let defaultContentFont = UIFont.systemFont(ofSize: 14) + static let defaultMissingImage = Gridicon.iconOfType(.image) + static let headers: [HeaderFormatter.HeaderType] = [.none, .h1, .h2, .h3, .h4, .h5, .h6] + static let margin = CGFloat(20) + static let moreAttachmentText = "more" + } +} From 98617463180b97752983daff9d6529821652e9d9 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:51:13 -0300 Subject: [PATCH 41/74] NSAttributedString+Attachments: New Convenience Initializer --- .../NSAttributedString+Attachments.swift | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift b/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift index cc90fe0ae..e6e82a5ba 100644 --- a/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift +++ b/Aztec/Classes/Extensions/NSAttributedString+Attachments.swift @@ -11,6 +11,22 @@ extension NSAttributedString static let lengthOfTextAttachment = NSAttributedString(attachment: NSTextAttachment()).length + /// String containing the NSTextAttachment Character + /// + static let textAttachmentString = String(UnicodeScalar(NSAttachmentCharacter)!) + + + + /// Helper Initializer: returns an Attributed String, with the specified attachment, styled with a given + /// collection of attributes. + /// + convenience init(attachment: NSTextAttachment, attributes: [String: Any]) { + var attributesWithAttachment = attributes + attributesWithAttachment[NSAttachmentAttributeName] = attachment + + self.init(string: NSAttributedString.textAttachmentString, attributes: attributesWithAttachment) + } + /// Loads any NSTextAttachment's lazy file reference, into a UIImage instance, in memory. /// func loadLazyAttachments() { From 5afa134ed1ed7f5f32c1118d8d66d5833316be82 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:51:46 -0300 Subject: [PATCH 42/74] HTMLNodeToNSAttributedString: Wiring CommentAttachment Conversion --- .../HTMLNodeToNSAttributedString.swift | 16 +++++----------- 1 file changed, 5 insertions(+), 11 deletions(-) diff --git a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift index 4922f1912..508d1c098 100644 --- a/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift +++ b/Aztec/Classes/Converters/HTMLNodeToNSAttributedString.swift @@ -97,17 +97,11 @@ class HMTLNodeToNSAttributedString: SafeConverter { /// /// - Returns: the converted node as an `NSAttributedString`. /// - fileprivate func convertCommentNode(_ node: CommentNode, inheritingAttributes inheritedAttributes: [String:Any]) -> NSAttributedString { - if node.comment.hasPrefix(MoreAttachment.commentNodeText) { - var attributes = inheritedAttributes; - let moreAttachment = MoreAttachment() - let index = MoreAttachment.commentNodeText.endIndex - moreAttachment.message = node.comment.substring(from: index) - moreAttachment.label = NSAttributedString(string: NSLocalizedString("MORE", comment: "Text for the center of the more divider"), attributes: defaultAttributes) - attributes[NSAttachmentAttributeName] = moreAttachment - return NSAttributedString(string:String(UnicodeScalar(NSAttachmentCharacter)!), attributes: attributes) - } - return NSAttributedString(string: node.text(), attributes: inheritedAttributes) + fileprivate func convertCommentNode(_ node: CommentNode, inheritingAttributes attributes: [String:Any]) -> NSAttributedString { + let attachment = CommentAttachment() + attachment.text = node.comment + + return NSAttributedString(attachment: attachment, attributes: attributes) } /// Converts an `ElementNode` to `NSAttributedString`. From d38251f0216e70706f67909058e56cd4006232b0 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:52:56 -0300 Subject: [PATCH 43/74] TextView: Updates Public API --- Aztec/Classes/TextKit/TextView.swift | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 9f0af55eb..2d80059ae 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -1004,15 +1004,17 @@ open class TextView: UITextView { // MARK: - More - /// Inserts the More Comment at the specified position. + /// Inserts an HTML Comment at the specified position. /// - /// - Parameter range: The character range that must be replaced by a More Attachment. + /// - Parameters: + /// - range: The character range that must be replaced with a Comment Attachment. + /// - text: The Comment Attachment's Text. /// /// - Returns: the attachment object that can be used for further calls /// @discardableResult - open func replaceRangeWithMoreAttachment(_ range: NSRange) -> MoreAttachment { - let attachment = storage.replaceRangeWithMoreAttachment(range, attributes: typingAttributes) + open func replaceRangeWithCommentAttachment(_ range: NSRange, text: String) -> CommentAttachment { + let attachment = storage.replaceRangeWithCommentAttachment(range, text: text, attributes: typingAttributes) selectedRange = NSMakeRange(range.location + NSAttributedString.lengthOfTextAttachment, 0) delegate?.textViewDidChange?(self) From 472e55cd195fd8f53d94ee9e618eb340e38c6a18 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 12:53:42 -0300 Subject: [PATCH 44/74] Implements CommentAttachment --- Aztec.xcodeproj/project.pbxproj | 8 +- Aztec/Classes/TextKit/CommentAttachment.swift | 82 +++++++++++++++++++ 2 files changed, 86 insertions(+), 4 deletions(-) create mode 100644 Aztec/Classes/TextKit/CommentAttachment.swift diff --git a/Aztec.xcodeproj/project.pbxproj b/Aztec.xcodeproj/project.pbxproj index 3d0c525c3..5fade849c 100644 --- a/Aztec.xcodeproj/project.pbxproj +++ b/Aztec.xcodeproj/project.pbxproj @@ -48,6 +48,7 @@ 59FEA0781D8BDFA700D138DF /* HTMLToAttributedStringTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 59FEA06A1D8BDFA700D138DF /* HTMLToAttributedStringTests.swift */; }; 59FEA07A1D8BDFA700D138DF /* InHTMLConverterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 59FEA06C1D8BDFA700D138DF /* InHTMLConverterTests.swift */; }; B551A4A01E770B3800EE3A7F /* UIFont+Emoji.swift in Sources */ = {isa = PBXBuildFile; fileRef = B551A49F1E770B3800EE3A7F /* UIFont+Emoji.swift */; }; + B572AC281E817CFE008948C2 /* CommentAttachment.swift in Sources */ = {isa = PBXBuildFile; fileRef = B572AC271E817CFE008948C2 /* CommentAttachment.swift */; }; B577DC651E7B18E90012A1F8 /* NodeDescriptor.swift in Sources */ = {isa = PBXBuildFile; fileRef = B577DC641E7B18E90012A1F8 /* NodeDescriptor.swift */; }; B577DC671E7B18F80012A1F8 /* CommentNodeDescriptor.swift in Sources */ = {isa = PBXBuildFile; fileRef = B577DC661E7B18F80012A1F8 /* CommentNodeDescriptor.swift */; }; B59C9F9F1DF74BB80073B1D6 /* UIFont+Traits.swift in Sources */ = {isa = PBXBuildFile; fileRef = B59C9F9E1DF74BB80073B1D6 /* UIFont+Traits.swift */; }; @@ -98,7 +99,6 @@ F1FFB2A11E6058930015ACB8 /* DOMStringTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F1FFB2A01E6058930015ACB8 /* DOMStringTests.swift */; }; FF13CD4D1E5C8067000FF10E /* HeaderFormatter.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF13CD4C1E5C8067000FF10E /* HeaderFormatter.swift */; }; FF152D8E1E68552A00FF596C /* StringRangeConversionTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF152D8D1E68552A00FF596C /* StringRangeConversionTests.swift */; }; - FF7A1C4F1E560A1F00C4C7C8 /* MoreAttachment.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7A1C4E1E560A1F00C4C7C8 /* MoreAttachment.swift */; }; FF7A1C511E5651EA00C4C7C8 /* LineAttachment.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7A1C501E5651EA00C4C7C8 /* LineAttachment.swift */; }; FF7C89A81E3A2B7C000472A8 /* Blockquote.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89A71E3A2B7C000472A8 /* Blockquote.swift */; }; FF7C89AC1E3A47F1000472A8 /* StandardAttributeFormatter.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */; }; @@ -182,6 +182,7 @@ 59FEA06C1D8BDFA700D138DF /* InHTMLConverterTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = InHTMLConverterTests.swift; sourceTree = ""; }; 59FEA06D1D8BDFA700D138DF /* InNodeConverterTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = InNodeConverterTests.swift; sourceTree = ""; }; B551A49F1E770B3800EE3A7F /* UIFont+Emoji.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIFont+Emoji.swift"; sourceTree = ""; }; + B572AC271E817CFE008948C2 /* CommentAttachment.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CommentAttachment.swift; sourceTree = ""; }; B577DC641E7B18E90012A1F8 /* NodeDescriptor.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = NodeDescriptor.swift; path = Descriptors/NodeDescriptor.swift; sourceTree = ""; }; B577DC661E7B18F80012A1F8 /* CommentNodeDescriptor.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = CommentNodeDescriptor.swift; path = Descriptors/CommentNodeDescriptor.swift; sourceTree = ""; }; B59C9F9E1DF74BB80073B1D6 /* UIFont+Traits.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "UIFont+Traits.swift"; sourceTree = ""; }; @@ -236,7 +237,6 @@ FF5B98E41DC355B400571CA4 /* LICENSE */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = LICENSE; sourceTree = SOURCE_ROOT; }; FF7A1C481E51EFB600C4C7C8 /* WordPress-Aztec-iOS.podspec */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = "WordPress-Aztec-iOS.podspec"; sourceTree = ""; }; FF7A1C4A1E51F05700C4C7C8 /* CONTRIBUTING.md */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = net.daringfireball.markdown; path = CONTRIBUTING.md; sourceTree = ""; }; - FF7A1C4E1E560A1F00C4C7C8 /* MoreAttachment.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MoreAttachment.swift; sourceTree = ""; }; FF7A1C501E5651EA00C4C7C8 /* LineAttachment.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = LineAttachment.swift; sourceTree = ""; }; FF7C89A71E3A2B7C000472A8 /* Blockquote.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Blockquote.swift; sourceTree = ""; }; FF7C89AB1E3A47F1000472A8 /* StandardAttributeFormatter.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = StandardAttributeFormatter.swift; sourceTree = ""; }; @@ -457,8 +457,8 @@ 599F252E1D8BC9A1002871D6 /* TextKit */ = { isa = PBXGroup; children = ( + B572AC271E817CFE008948C2 /* CommentAttachment.swift */, 599F25301D8BC9A1002871D6 /* TextAttachment.swift */, - FF7A1C4E1E560A1F00C4C7C8 /* MoreAttachment.swift */, FF7A1C501E5651EA00C4C7C8 /* LineAttachment.swift */, B5BC4FF51DA2D76600614582 /* TextList.swift */, 599F25311D8BC9A1002871D6 /* TextStorage.swift */, @@ -755,6 +755,7 @@ 599F25471D8BC9A1002871D6 /* HTMLConstants.swift in Sources */, 599F25371D8BC9A1002871D6 /* InAttributeConverter.swift in Sources */, 599F25531D8BC9A1002871D6 /* TextStorage.swift in Sources */, + B572AC281E817CFE008948C2 /* CommentAttachment.swift in Sources */, F1FA0E861E6EF514009D98EE /* Node.swift in Sources */, 599F253C1D8BC9A1002871D6 /* OutHTMLAttributeConverter.swift in Sources */, 599F254B1D8BC9A1002871D6 /* String+RangeConversion.swift in Sources */, @@ -785,7 +786,6 @@ 599F254A1D8BC9A1002871D6 /* NSAttributedString+Attachments.swift in Sources */, B5B96DAB1E01B2F300791315 /* UIPasteboard+Helpers.swift in Sources */, F17D64AE1E4230A400D09FED /* VisualOnlyAttribute.swift in Sources */, - FF7A1C4F1E560A1F00C4C7C8 /* MoreAttachment.swift in Sources */, F15C9B881DD58D8B00833C39 /* ElementNodeDescriptor.swift in Sources */, FF7C89A81E3A2B7C000472A8 /* Blockquote.swift in Sources */, 599F254C1D8BC9A1002871D6 /* UITextView+Helpers.swift in Sources */, diff --git a/Aztec/Classes/TextKit/CommentAttachment.swift b/Aztec/Classes/TextKit/CommentAttachment.swift new file mode 100644 index 000000000..b24dcac71 --- /dev/null +++ b/Aztec/Classes/TextKit/CommentAttachment.swift @@ -0,0 +1,82 @@ +import Foundation +import UIKit + +/// Custom text attachment. +/// +open class CommentAttachment: NSTextAttachment { + + /// + /// + fileprivate var glyphImage: UIImage? + + /// The color to use when drawing progress indicators + /// + open var color = UIColor.gray + + /// A message to display overlaid on top of the image + /// + open var label = NSAttributedString(string: "MORE") { + willSet { + if newValue != label { + glyphImage = nil + } + } + } + open var text: String = "" + + + // MARK: - NSTextAttachmentContainer + + override open func image(forBounds imageBounds: CGRect, textContainer: NSTextContainer?, characterIndex charIndex: Int) -> UIImage? { + + if let cachedImage = glyphImage, imageBounds.size.equalTo(cachedImage.size) { + return cachedImage + } + + glyphImage = glyph(forBounds: imageBounds) + + return glyphImage + } + + fileprivate func glyph(forBounds bounds: CGRect) -> UIImage? { + + let size = bounds.size + + UIGraphicsBeginImageContextWithOptions(bounds.size, false, 0) + + let colorMessage = NSMutableAttributedString(attributedString: label) + colorMessage.addAttribute(NSForegroundColorAttributeName, value: color, range: label.rangeOfEntireString) + let textRect = colorMessage.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) + let textPosition = CGPoint(x: ((size.width - textRect.width) / 2), y: ((size.height - textRect.height) / 2) ) + colorMessage.draw(in: CGRect(origin: textPosition , size: CGSize(width: size.width, height: textRect.size.height))) + + let path = UIBezierPath() + + let dashWidth: CGFloat = 8.0 + let dashes: [ CGFloat ] = [ dashWidth, dashWidth ] + path.setLineDash(dashes, count: dashes.count, phase: 0.0) + path.lineWidth = 2.0 + let centerY = round(size.height / 2.0) + path.move(to: CGPoint(x:0, y: centerY)) + path.addLine(to: CGPoint(x: ((size.width - textRect.width) / 2) - dashWidth, y: centerY)) + + path.move(to: CGPoint(x:((size.width + textRect.width) / 2) + dashWidth, y: centerY)) + path.addLine(to: CGPoint(x: size.width, y: centerY)) + + color.setStroke() + path.stroke() + + let result = UIGraphicsGetImageFromCurrentImageContext() + UIGraphicsEndImageContext() + return result; + } + + override open func attachmentBounds(for textContainer: NSTextContainer?, proposedLineFragment lineFrag: CGRect, glyphPosition position: CGPoint, characterIndex charIndex: Int) -> CGRect { + + let padding = textContainer?.lineFragmentPadding ?? 0 + let width = lineFrag.width - padding * 2 + let height:CGFloat = 44.0 + + return CGRect(origin: CGPoint.zero, size: CGSize(width: width, height: height)) + } +} From 9e7ff88048a49a6e1f6c11601e16173a0bcbd456 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 14:08:51 -0300 Subject: [PATCH 45/74] TextStorage: Wiring CommentAttachment --- Aztec/Classes/TextKit/TextStorage.swift | 42 +++++++++++-------------- 1 file changed, 19 insertions(+), 23 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 1e06fc9f7..9090370ff 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -177,7 +177,7 @@ open class TextStorage: NSTextStorage { attributedString.enumerateAttribute(NSAttachmentAttributeName, in: fullRange, options: []) { (object, range, stop) in - guard let object = object, !(object is LineAttachment), !(object is MoreAttachment) else { + guard let object = object, !(object is LineAttachment), !(object is CommentAttachment) else { return } @@ -340,7 +340,7 @@ open class TextStorage: NSTextStorage { /// private func processAttributesDifference(in domRange: NSRange, key: String, sourceValue: Any?, targetValue: Any?) { let isLineAttachment = sourceValue is LineAttachment || targetValue is LineAttachment - let isMoreAttachment = sourceValue is MoreAttachment || targetValue is MoreAttachment + let isCommentAttachment = sourceValue is CommentAttachment || targetValue is CommentAttachment switch(key) { case NSFontAttributeName: @@ -363,11 +363,11 @@ open class TextStorage: NSTextStorage { let targetAttachment = targetValue as? LineAttachment processLineAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) - case NSAttachmentAttributeName where isMoreAttachment: - let sourceAttachment = sourceValue as? MoreAttachment - let targetAttachment = targetValue as? MoreAttachment + case NSAttachmentAttributeName where isCommentAttachment: + let sourceAttachment = sourceValue as? CommentAttachment + let targetAttachment = targetValue as? CommentAttachment - processMoreAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) + processCommentAttachmentDifferences(in: domRange, betweenOriginal: sourceAttachment, andNew: targetAttachment) case NSAttachmentAttributeName: let sourceAttachment = sourceValue as? TextAttachment let targetAttachment = targetValue as? TextAttachment @@ -456,9 +456,12 @@ open class TextStorage: NSTextStorage { dom.replaceWithHorizontalRuler(range) } - private func processMoreAttachmentDifferences(in range: NSRange, betweenOriginal original: MoreAttachment?, andNew new: MoreAttachment?) { + private func processCommentAttachmentDifferences(in range: NSRange, betweenOriginal original: CommentAttachment?, andNew new: CommentAttachment?) { + guard let newAttachment = new else { + return + } - dom.replace(range, with: MoreAttachment.commentNodeText) + dom.replace(range, with: newAttachment.text) } @@ -605,7 +608,7 @@ open class TextStorage: NSTextStorage { private func canAppendToNodeRepresentedByCharacter(atIndex index: Int) -> Bool { return !hasNewLine(atIndex: index) && !hasHorizontalLine(atIndex: index) - && !hasMoreMarker(atIndex: index) + && !hasCommentMarker(atIndex: index) && !hasVisualOnlyElement(atIndex: index) } @@ -627,9 +630,9 @@ open class TextStorage: NSTextStorage { return true } - private func hasMoreMarker(atIndex index: Int) -> Bool { + private func hasCommentMarker(atIndex index: Int) -> Bool { guard let attachment = attribute(NSAttachmentAttributeName, at: index, effectiveRange: nil), - attachment is MoreAttachment else { + attachment is CommentAttachment else { return false } @@ -776,23 +779,16 @@ open class TextStorage: NSTextStorage { } } - /// Inserts the MoreAttachment at the specified position + /// Inserts the Comment Attachment at the specified position /// @discardableResult - open func replaceRangeWithMoreAttachment(_ range: NSRange, attributes: [String: Any]) -> MoreAttachment { - let message = "MORE" - let label = NSLocalizedString("MORE", comment: "Text for the center of the more divider") - - let attachment = MoreAttachment() - attachment.message = message - attachment.label = NSAttributedString(string: label, attributes: [:]) + open func replaceRangeWithCommentAttachment(_ range: NSRange, text: String, attributes: [String: Any]) -> CommentAttachment { + let attachment = CommentAttachment() + attachment.text = text - let stringWithAttachment = NSAttributedString(attachment: attachment) + let stringWithAttachment = NSAttributedString(attachment: attachment, attributes: attributes) replaceCharacters(in: range, with: stringWithAttachment) - let attachmentRange = NSRange(location: range.location, length: NSAttributedString.lengthOfTextAttachment) - addAttributes(attributes, range: attachmentRange) - return attachment } From da9711e637e7fd92114429e29783d024c3af6d51 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 14:36:25 -0300 Subject: [PATCH 46/74] Fixing TextStorageTests --- AztecTests/TextStorageTests.swift | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index 01f3bb838..dfa96fd77 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -314,32 +314,32 @@ class TextStorageTests: XCTestCase XCTAssertEqual(html, "
") } - /// This test check if the insertion of a More Attachment works correctly and the tag is inserted + /// This test check if the insertion of a Comment Attachment works correctly and the expected tag gets inserted /// - func testReplaceRangeWithMoreAttachmentGeneratesExpectedHTMLComment() { + func testReplaceRangeWithCommentAttachmentGeneratesExpectedHTMLComment() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) + storage.replaceRangeWithCommentAttachment(.zero, text: "more", attributes: [:]) let html = storage.getHTML() XCTAssertEqual(html, "") } - /// This test check if the insertion of a More Attachment works correctly and the tag is inserted + /// This test check if the insertion of a Comment Attachment works correctly and the expected tag gets inserted /// - func testReplaceRangeWithMoreAttachmentDoNotCrashTheEditorWhenCalledSequentially() { + func testReplaceRangeWithCommentAttachmentDoNotCrashTheEditorWhenCalledSequentially() { let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() storage.attachmentsDelegate = mockDelegate - storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) - storage.replaceRangeWithMoreAttachment(.zero, attributes: [:]) + storage.replaceRangeWithCommentAttachment(.zero, text: "more", attributes: [:]) + storage.replaceRangeWithCommentAttachment(.zero, text: "some other comment should go here", attributes: [:]) let html = storage.getHTML() - XCTAssertEqual(html, "") + XCTAssertEqual(html, "") } /// This test verifies if we can delete all the content from a storage object that has html with a comment From 08ccbe6ad2e4c3135fed7aa53b0a216a223499f8 Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Tue, 21 Mar 2017 14:57:57 -0300 Subject: [PATCH 47/74] Adds the logic to merge list items on newline deletion. --- .../Libxml2/DOM/Data/ElementNode.swift | 19 +++++++++++++++++++ .../Classes/Libxml2/DOM/Logic/DOMEditor.swift | 14 +++++++++++++- 2 files changed, 32 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/Libxml2/DOM/Data/ElementNode.swift b/Aztec/Classes/Libxml2/DOM/Data/ElementNode.swift index 96b5dbe5e..373f0eec3 100644 --- a/Aztec/Classes/Libxml2/DOM/Data/ElementNode.swift +++ b/Aztec/Classes/Libxml2/DOM/Data/ElementNode.swift @@ -1447,6 +1447,25 @@ extension Libxml2 { } } + func unwrapChildren(first amount: Int) { + assert(children.count >= amount) + + guard let parent = parent else { + assertionFailure("Cannot execute this method if a parent isn't set.") + return + } + + let myIndex = parent.indexOf(childNode: self) + + for _ in 0...(amount - 1) { + parent.insert(children[0], at: myIndex) + } + + if children.count == 0 { + removeFromParent() + } + } + /// Unwraps the receiver's children from the receiver. /// @discardableResult diff --git a/Aztec/Classes/Libxml2/DOM/Logic/DOMEditor.swift b/Aztec/Classes/Libxml2/DOM/Logic/DOMEditor.swift index 02d824b51..14bf35acc 100644 --- a/Aztec/Classes/Libxml2/DOM/Logic/DOMEditor.swift +++ b/Aztec/Classes/Libxml2/DOM/Logic/DOMEditor.swift @@ -324,7 +324,19 @@ extension Libxml2 { if let rightElement = rightSibling as? ElementNode, rightElement.isBlockLevelElement() { - finalRightNodes = rightElement.unwrapChildren() + if rightElement.children.count > 0, + let rightChildElement = rightElement.children[0] as? ElementNode, + rightChildElement.isBlockLevelElement() { + + rightElement.unwrapChildren(first: 1) + + // This case was designed for lists. They need to unwrap the first list element + // from both the ul / ol and the li elements. + // + finalRightNodes = rightChildElement.unwrapChildren() + } else { + finalRightNodes = rightElement.unwrapChildren() + } } else { finalRightNodes = [rightSibling] } From ecc5c36f97dcac1676fd9de61e97b898ad1fff94 Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Tue, 21 Mar 2017 15:03:37 -0300 Subject: [PATCH 48/74] Added a new test for the newline deletion before lists. --- AztecTests/TextViewTests.swift | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/AztecTests/TextViewTests.swift b/AztecTests/TextViewTests.swift index f1d90108f..043a4cfe6 100644 --- a/AztecTests/TextViewTests.swift +++ b/AztecTests/TextViewTests.swift @@ -484,6 +484,37 @@ class AztecVisualTextViewTests: XCTestCase { XCTAssertEqual(textView.getHTML(), "

HelloWorld!

") } + /// Tests that deleting a newline works by merging the component around it. + /// + /// Input: + /// - Initial HTML: "List
  • first
  • second
  • " + /// - Deletion range: (loc: 4, len 1) + /// - Second deletion range: (loc: 9, len: 1) + /// + /// Output: + /// - Final HTML: "Listfirstsecond" + /// + func testDeleteNewline5() { + + let textView = createTextView(withHTML: "List
    • first
    • second
    • ") + + let rangeStart = textView.position(from: textView.beginningOfDocument, offset: 4)! + let rangeEnd = textView.position(from: rangeStart, offset: 1)! + let range = textView.textRange(from: rangeStart, to: rangeEnd)! + + textView.replace(range, withText: "") + + XCTAssertEqual(textView.getHTML(), "Listfirst
      • second
      ") + + let rangeStart2 = textView.position(from: textView.beginningOfDocument, offset: 9)! + let rangeEnd2 = textView.position(from: rangeStart2, offset: 1)! + let range2 = textView.textRange(from: rangeStart2, to: rangeEnd2)! + + textView.replace(range2, withText: "") + + XCTAssertEqual(textView.getHTML(), "Listfirstsecond") + } + /// Tests that deleting a newline works at the end of text with paragraph with header before works. /// /// Input: From 9250d74aa97df0b71fb86eb4b658863f1fb2145a Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 16:30:20 -0300 Subject: [PATCH 49/74] TextStorage: attachmentsDelegate is now IUO --- Aztec/Classes/TextKit/TextStorage.swift | 20 ++++++-------------- 1 file changed, 6 insertions(+), 14 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 9090370ff..2ca78a2a0 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -90,7 +90,7 @@ open class TextStorage: NSTextStorage { // MARK: - Attachments - var attachmentsDelegate: TextStorageAttachmentsDelegate? + var attachmentsDelegate: TextStorageAttachmentsDelegate! open func TextAttachments() -> [TextAttachment] { let range = NSMakeRange(0, length) @@ -171,18 +171,13 @@ open class TextStorage: NSTextStorage { /// - Returns: the preprocessed string. /// fileprivate func preprocessAttachmentsForInsertion(_ attributedString: NSAttributedString) -> NSAttributedString { - + assert(attachmentsDelegate != nil) + let fullRange = NSRange(location: 0, length: attributedString.length) let finalString = NSMutableAttributedString(attributedString: attributedString) attributedString.enumerateAttribute(NSAttachmentAttributeName, in: fullRange, options: []) { (object, range, stop) in - - guard let object = object, !(object is LineAttachment), !(object is CommentAttachment) else { - return - } - - guard let attachmentsDelegate = attachmentsDelegate else { - assertionFailure("This class can't really handle not having an image provider set.") + guard let object = object, !(object is LineAttachment) else { return } @@ -217,7 +212,7 @@ open class TextStorage: NSTextStorage { fileprivate func detectAttachmentRemoved(in range:NSRange) { textStore.enumerateAttachmentsOfType(TextAttachment.self, range: range) { (attachment, range, stop) in - self.attachmentsDelegate?.storage(self, deletedAttachmentWithID: attachment.identifier) + self.attachmentsDelegate.storage(self, deletedAttachmentWithID: attachment.identifier) } } @@ -859,10 +854,7 @@ extension TextStorage: TextAttachmentImageProvider { onSuccess success: @escaping (UIImage) -> (), onFailure failure: @escaping () -> ()) -> UIImage { - guard let attachmentsDelegate = attachmentsDelegate else { - fatalError("This class doesn't really support not having an attachments delegate set.") - } - + assert(attachmentsDelegate != nil) return attachmentsDelegate.storage(self, attachment: textAttachment, imageForURL: url, onSuccess: success, onFailure: failure) } From 9935e1e62a4a30cd6c24aee2a2998618473eac11 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:47:02 -0300 Subject: [PATCH 50/74] CommentAttachment: Implements Delegate Mechanism --- Aztec/Classes/TextKit/CommentAttachment.swift | 90 +++++++++---------- 1 file changed, 40 insertions(+), 50 deletions(-) diff --git a/Aztec/Classes/TextKit/CommentAttachment.swift b/Aztec/Classes/TextKit/CommentAttachment.swift index b24dcac71..c53eebc02 100644 --- a/Aztec/Classes/TextKit/CommentAttachment.swift +++ b/Aztec/Classes/TextKit/CommentAttachment.swift @@ -1,82 +1,72 @@ import Foundation import UIKit -/// Custom text attachment. + +/// Comment Attachment's Delegate Helpers /// -open class CommentAttachment: NSTextAttachment { +protocol CommentAttachmentDelegate: class { + /// Returns the Bounds that should be used to render the current attachment. + /// + /// - Parameters: + /// - commentAttachment: The Comment to be rendered + /// - fragment: Current Line Fragment Bounds + /// + /// - Returns: CGRect specifiying the Attachment Bounds. + /// + func commentAttachment(_ commentAttachment: CommentAttachment, boundsForLineFragment fragment: CGRect) -> CGRect + + /// Returns the Image Representation for a given attachment. + /// + /// - Parameters: + /// - commentAttachment: The Comment to be rendered + /// - size: The Canvas Size /// + /// - Returns: Optional UIImage instance, representing a given comment. + /// + func commentAttachment(_ commentAttachment: CommentAttachment, imageForSize size: CGSize) -> UIImage? +} + + +/// Comment Attachments: Represents an HTML Comment +/// +open class CommentAttachment: NSTextAttachment { + + /// Internal Cached Image /// fileprivate var glyphImage: UIImage? - /// The color to use when drawing progress indicators + /// Delegate /// - open var color = UIColor.gray + weak var delegate: CommentAttachmentDelegate? /// A message to display overlaid on top of the image /// - open var label = NSAttributedString(string: "MORE") { - willSet { - if newValue != label { - glyphImage = nil - } + open var text: String = "" { + didSet { + glyphImage = nil } } - open var text: String = "" // MARK: - NSTextAttachmentContainer override open func image(forBounds imageBounds: CGRect, textContainer: NSTextContainer?, characterIndex charIndex: Int) -> UIImage? { - if let cachedImage = glyphImage, imageBounds.size.equalTo(cachedImage.size) { return cachedImage } - glyphImage = glyph(forBounds: imageBounds) + glyphImage = delegate?.commentAttachment(self, imageForSize: imageBounds.size) return glyphImage } - fileprivate func glyph(forBounds bounds: CGRect) -> UIImage? { - - let size = bounds.size - - UIGraphicsBeginImageContextWithOptions(bounds.size, false, 0) - - let colorMessage = NSMutableAttributedString(attributedString: label) - colorMessage.addAttribute(NSForegroundColorAttributeName, value: color, range: label.rangeOfEntireString) - let textRect = colorMessage.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) - let textPosition = CGPoint(x: ((size.width - textRect.width) / 2), y: ((size.height - textRect.height) / 2) ) - colorMessage.draw(in: CGRect(origin: textPosition , size: CGSize(width: size.width, height: textRect.size.height))) - - let path = UIBezierPath() - - let dashWidth: CGFloat = 8.0 - let dashes: [ CGFloat ] = [ dashWidth, dashWidth ] - path.setLineDash(dashes, count: dashes.count, phase: 0.0) - path.lineWidth = 2.0 - let centerY = round(size.height / 2.0) - path.move(to: CGPoint(x:0, y: centerY)) - path.addLine(to: CGPoint(x: ((size.width - textRect.width) / 2) - dashWidth, y: centerY)) - - path.move(to: CGPoint(x:((size.width + textRect.width) / 2) + dashWidth, y: centerY)) - path.addLine(to: CGPoint(x: size.width, y: centerY)) - - color.setStroke() - path.stroke() - - let result = UIGraphicsGetImageFromCurrentImageContext() - UIGraphicsEndImageContext() - return result; - } - override open func attachmentBounds(for textContainer: NSTextContainer?, proposedLineFragment lineFrag: CGRect, glyphPosition position: CGPoint, characterIndex charIndex: Int) -> CGRect { + guard let bounds = delegate?.commentAttachment(self, boundsForLineFragment: lineFrag) else { + assertionFailure("Could not determine Comment Attachment Size") + return .zero + } - let padding = textContainer?.lineFragmentPadding ?? 0 - let width = lineFrag.width - padding * 2 - let height:CGFloat = 44.0 - - return CGRect(origin: CGPoint.zero, size: CGSize(width: width, height: height)) + return bounds } } From a456d052814f6b1d94238cc915c1ef4fb9cc3970 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:48:17 -0300 Subject: [PATCH 51/74] TextStorage: Implements CommentAttachmentDelegate Protocol --- Aztec/Classes/TextKit/TextStorage.swift | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 2ca78a2a0..bf8ae9b89 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -846,6 +846,9 @@ open class TextStorage: NSTextStorage { } } + +// MARK: - TextStorage: TextAttachmentImageProvider Conformance +// extension TextStorage: TextAttachmentImageProvider { func textAttachment( @@ -859,3 +862,19 @@ extension TextStorage: TextAttachmentImageProvider { } } + + +// MARK: - TextStorage: CommentAttachmentDelegate Conformance +// +extension TextStorage: CommentAttachmentDelegate { + + func commentAttachment(_ commentAttachment: CommentAttachment, imageForSize size: CGSize) -> UIImage? { + assert(attachmentsDelegate != nil) + return attachmentsDelegate.storage(self, imageForComment: commentAttachment, with: size) + } + + func commentAttachment(_ commentAttachment: CommentAttachment, boundsForLineFragment fragment: CGRect) -> CGRect { + assert(attachmentsDelegate != nil) + return attachmentsDelegate.storage(self, boundsForComment: commentAttachment, with: fragment) + } +} From 1801506f21884b2655dbf424ac586cb4fd9f41dd Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:48:44 -0300 Subject: [PATCH 52/74] TextStorage: Wiring CommentAttachment Delegate on setHTML --- Aztec/Classes/TextKit/TextStorage.swift | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index bf8ae9b89..30c94f43b 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -839,9 +839,13 @@ open class TextStorage: NSTextStorage { let originalLength = textStore.length textStore = NSMutableAttributedString(attributedString: attributedString) - textStore.enumerateAttachmentsOfType(TextAttachment.self) { [weak self] (attachment, range, stop) in + textStore.enumerateAttachmentsOfType(TextAttachment.self) { [weak self] (attachment, _, _) in attachment.imageProvider = self } + textStore.enumerateAttachmentsOfType(CommentAttachment.self) { [weak self] (attachment, _, _) in + attachment.delegate = self + } + edited([.editedAttributes, .editedCharacters], range: NSRange(location: 0, length: originalLength), changeInLength: textStore.length - originalLength) } } From 0c1e740f0aa2f6b30df0a3c0649db1a3bfc58a1b Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:51:56 -0300 Subject: [PATCH 53/74] TextStorage: Adds new AttachmentDelegate Methods --- Aztec/Classes/TextKit/TextStorage.swift | 25 ++++++++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 30c94f43b..6f385a08c 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -40,7 +40,30 @@ protocol TextStorageAttachmentsDelegate { /// - Parameters: /// - textView: The textView where the attachment was removed. /// - attachmentID: The attachment identifier of the media removed. - func storage(_ storage: TextStorage, deletedAttachmentWithID attachmentID: String); + /// + func storage(_ storage: TextStorage, deletedAttachmentWithID attachmentID: String) + + /// Provides the Bounds required to represent a given attachment, within a specified line fragment. + /// + /// - Parameters: + /// - storage: The storage that is requesting the bounds. + /// - attachment: CommentAttachment about to be rendered. + /// - lineFragment: Line Fragment in which the glyph would be rendered. + /// + /// - Returns: Rect specifying the Bounds for the comment attachment + /// + func storage(_ storage: TextStorage, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect + + /// Provides the (Optional) Image Representation of the specified size, for a given Attachment. + /// + /// - Parameters: + /// - storage: The storage that is requesting the bounds. + /// - attachment: CommentAttachment about to be rendered. + /// - size: Expected Image Size + /// + /// - Returns: (Optional) UIImage representation of the Comment Attachment. + /// + func storage(_ storage: TextStorage, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? } /// Custom NSTextStorage From 9053d7455092296d96eeca7341058d2fa18166cc Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:53:38 -0300 Subject: [PATCH 54/74] TextStorage: Normalizes Comments --- Aztec/Classes/TextKit/TextStorage.swift | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 6f385a08c..d53d117e7 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -7,13 +7,14 @@ protocol TextStorageAttachmentsDelegate { /// Provides images for attachments that are part of the storage /// - /// - parameter storage: The storage that is requesting the image. - /// - parameter attachment: The attachment that is requesting the image. - /// - parameter url: url for the image. - /// - parameter success: a callback block to be invoked with the image fetched from the url. - /// - parameter failure: a callback block to be invoked when an error occurs when fetching the image. + /// - Parameters: + /// - storage: The storage that is requesting the image. + /// - attachment: The attachment that is requesting the image. + /// - url: url for the image. + /// - success: Callback block to be invoked with the image fetched from the url. + /// - failure: Callback block to be invoked when an error occurs when fetching the image. /// - /// - returns: returns a temporary UIImage to be used while the request is happening + /// - Returns: returns a temporary UIImage to be used while the request is happening /// func storage( _ storage: TextStorage, @@ -28,8 +29,8 @@ protocol TextStorageAttachmentsDelegate { /// delegate can specify an URL where that image is available. /// /// - Parameters: - /// - storage: The storage that is requesting the image. - /// - image: The image that was added to the storage. + /// - storage: The storage that is requesting the image. + /// - image: The image that was added to the storage. /// /// - Returns: the requested `NSURL` where the image is stored. /// From 60134d3661f9947969951416aa59bb664569c1a3 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Tue, 21 Mar 2017 17:56:38 -0300 Subject: [PATCH 55/74] TextView: Implements TextViewCommentsDelegate Protocol --- Aztec/Classes/TextKit/TextView.swift | 58 +++++++++++++++++++++++++++- 1 file changed, 57 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 2d80059ae..373e29dbb 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -3,6 +3,9 @@ import UIKit import Foundation import Gridicons + +// MARK: - TextViewMediaDelegate +// public protocol TextViewMediaDelegate: class { /// This method requests from the delegate the image at the specified URL. @@ -62,6 +65,37 @@ public protocol TextViewMediaDelegate: class { func textView(_ textView: TextView, deselectedAttachment attachment: TextAttachment, atPosition position: CGPoint) } + +// MARK: - TextViewCommentsDelegate +// +public protocol TextViewCommentsDelegate: class { + + /// Provides the Bounds required to represent a given attachment, within a specified line fragment. + /// + /// - Parameters: + /// - textView: The textView that is requesting the bounds. + /// - attachment: CommentAttachment about to be rendered. + /// - lineFragment: Line Fragment in which the glyph would be rendered. + /// + /// - Returns: Rect specifying the Bounds for the comment attachment + /// + func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect + + /// Provides the (Optional) Image Representation of the specified size, for a given Attachment. + /// + /// - Parameters: + /// - textView: The textView that is requesting the bounds. + /// - attachment: CommentAttachment about to be rendered. + /// - size: Expected Image Size + /// + /// - Returns: (Optional) UIImage representation of the Comment Attachment. + /// + func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage +} + + +// MARK: - TextViewFormattingDelegate +// public protocol TextViewFormattingDelegate: class { /// Called a text view command toggled a style. @@ -71,6 +105,9 @@ public protocol TextViewFormattingDelegate: class { func textViewCommandToggledAStyle() } + +// MARK: - TextView +// open class TextView: UITextView { typealias ElementNode = Libxml2.ElementNode @@ -83,6 +120,10 @@ open class TextView: UITextView { /// open weak var mediaDelegate: TextViewMediaDelegate? + // MARK: - Properties: Comment Attachments + + open weak var commentsDelegate: TextViewCommentsDelegate? + // MARK: - Properties: Formatting open weak var formattingDelegate: TextViewFormattingDelegate? @@ -1048,7 +1089,6 @@ extension TextView: TextStorageAttachmentsDelegate { } func storage(_ storage: TextStorage, urlForAttachment attachment: TextAttachment) -> URL { - guard let mediaDelegate = mediaDelegate else { fatalError("This class requires a media delegate to be set.") } @@ -1059,6 +1099,22 @@ extension TextView: TextStorageAttachmentsDelegate { func storage(_ storage: TextStorage, deletedAttachmentWithID attachmentID: String) { mediaDelegate?.textView(self, deletedAttachmentWithID: attachmentID) } + + func storage(_ storage: TextStorage, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { + guard let commentsDelegate = commentsDelegate else { + fatalError("This class requires a comments delegate to be set.") + } + + return commentsDelegate.textView(self, imageForComment: attachment, with: size) + } + + func storage(_ storage: TextStorage, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { + guard let commentsDelegate = commentsDelegate else { + fatalError("This class requires a comments delegate to be set.") + } + + return commentsDelegate.textView(self, boundsForComment: attachment, with: lineFragment) + } } // MARK: - UIGestureRecognizerDelegate From 589154ba8cd110ab1cff4130ab4d7a93455eb3ac Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Wed, 22 Mar 2017 09:21:18 -0300 Subject: [PATCH 56/74] Fixes and improves a unit test. --- AztecTests/TextViewTests.swift | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/AztecTests/TextViewTests.swift b/AztecTests/TextViewTests.swift index 043a4cfe6..81c67d433 100644 --- a/AztecTests/TextViewTests.swift +++ b/AztecTests/TextViewTests.swift @@ -487,16 +487,17 @@ class AztecVisualTextViewTests: XCTestCase { /// Tests that deleting a newline works by merging the component around it. /// /// Input: - /// - Initial HTML: "List
      • first
      • second
      • " + /// - Initial HTML: "List
        • first
        • second
        • third
        " /// - Deletion range: (loc: 4, len 1) /// - Second deletion range: (loc: 9, len: 1) + /// - Third deletion range: (loc: 15, len: 1) /// /// Output: /// - Final HTML: "Listfirstsecond" /// func testDeleteNewline5() { - let textView = createTextView(withHTML: "List
        • first
        • second
        • ") + let textView = createTextView(withHTML: "List
          • first
          • second
          • third
          ") let rangeStart = textView.position(from: textView.beginningOfDocument, offset: 4)! let rangeEnd = textView.position(from: rangeStart, offset: 1)! @@ -504,7 +505,7 @@ class AztecVisualTextViewTests: XCTestCase { textView.replace(range, withText: "") - XCTAssertEqual(textView.getHTML(), "Listfirst
          • second
          ") + XCTAssertEqual(textView.getHTML(), "Listfirst
          • second
          • third
          ") let rangeStart2 = textView.position(from: textView.beginningOfDocument, offset: 9)! let rangeEnd2 = textView.position(from: rangeStart2, offset: 1)! @@ -512,7 +513,15 @@ class AztecVisualTextViewTests: XCTestCase { textView.replace(range2, withText: "") - XCTAssertEqual(textView.getHTML(), "Listfirstsecond") + XCTAssertEqual(textView.getHTML(), "Listfirstsecond
          • third
          ") + + let rangeStart3 = textView.position(from: textView.beginningOfDocument, offset: 15)! + let rangeEnd3 = textView.position(from: rangeStart3, offset: 1)! + let range3 = textView.textRange(from: rangeStart3, to: rangeEnd3)! + + textView.replace(range3, withText: "") + + XCTAssertEqual(textView.getHTML(), "Listfirstsecondthird") } /// Tests that deleting a newline works at the end of text with paragraph with header before works. From c836f6ddfdfd7ca4515050ce46514a796c48dc07 Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Wed, 22 Mar 2017 10:46:12 -0300 Subject: [PATCH 57/74] Fixes a serious issue with the root node de-synchronizing. --- Aztec/Classes/Libxml2/DOMString.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Aztec/Classes/Libxml2/DOMString.swift b/Aztec/Classes/Libxml2/DOMString.swift index d58d2d786..844a50358 100644 --- a/Aztec/Classes/Libxml2/DOMString.swift +++ b/Aztec/Classes/Libxml2/DOMString.swift @@ -113,9 +113,10 @@ extension Libxml2 { } catch { fatalError("Could not convert the HTML.") } - + domQueue.sync { self.rootNode = output.rootNode + self.domEditor = DOMEditor(with: output.rootNode) } return output.attributedString From f6b9552c917122f3841112c895597ae804436caa Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 12:24:01 -0300 Subject: [PATCH 58/74] TextAttachment: Renames delegate --- Aztec/Classes/TextKit/TextAttachment.swift | 12 +++++------- Aztec/Classes/TextKit/TextStorage.swift | 10 +++++----- 2 files changed, 10 insertions(+), 12 deletions(-) diff --git a/Aztec/Classes/TextKit/TextAttachment.swift b/Aztec/Classes/TextKit/TextAttachment.swift index 05382e6dd..1f8381cb7 100644 --- a/Aztec/Classes/TextKit/TextAttachment.swift +++ b/Aztec/Classes/TextKit/TextAttachment.swift @@ -1,7 +1,7 @@ import Foundation import UIKit -protocol TextAttachmentImageProvider { +protocol TextAttachmentDelegate { func textAttachment( _ textAttachment: TextAttachment, imageForURL url: URL, @@ -127,7 +127,7 @@ open class TextAttachment: NSTextAttachment fileprivate var glyphImage: UIImage? - var imageProvider: TextAttachmentImageProvider? + var delegate: TextAttachmentDelegate? var isFetchingImage: Bool = false @@ -343,7 +343,7 @@ open class TextAttachment: NSTextAttachment func updateImage(inTextContainer textContainer: NSTextContainer? = nil) { - guard let imageProvider = imageProvider else { + guard let delegate = delegate else { assertionFailure("This class doesn't really support not having an updater set.") return } @@ -354,9 +354,7 @@ open class TextAttachment: NSTextAttachment isFetchingImage = true - let image = imageProvider.textAttachment(self, - imageForURL: url, - onSuccess: { [weak self] (image) in + let image = delegate.textAttachment(self, imageForURL: url, onSuccess: { [weak self] image in guard let strongSelf = self else { return } @@ -364,7 +362,7 @@ open class TextAttachment: NSTextAttachment strongSelf.isFetchingImage = false strongSelf.image = image strongSelf.invalidateLayout(inTextContainer: textContainer) - }, onFailure: { [weak self]() in + }, onFailure: { [weak self] _ in guard let strongSelf = self else { return diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index d53d117e7..2b9ed3411 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -702,7 +702,7 @@ open class TextStorage: NSTextStorage { /// func insertImage(sourceURL url: URL, atPosition position:Int, placeHolderImage: UIImage, identifier: String = UUID().uuidString) -> TextAttachment { let attachment = TextAttachment(identifier: identifier) - attachment.imageProvider = self + attachment.delegate = self attachment.url = url attachment.image = placeHolderImage @@ -864,7 +864,7 @@ open class TextStorage: NSTextStorage { let originalLength = textStore.length textStore = NSMutableAttributedString(attributedString: attributedString) textStore.enumerateAttachmentsOfType(TextAttachment.self) { [weak self] (attachment, _, _) in - attachment.imageProvider = self + attachment.delegate = self } textStore.enumerateAttachmentsOfType(CommentAttachment.self) { [weak self] (attachment, _, _) in attachment.delegate = self @@ -875,9 +875,9 @@ open class TextStorage: NSTextStorage { } -// MARK: - TextStorage: TextAttachmentImageProvider Conformance +// MARK: - TextStorage: TextAttachmentDelegate Methods // -extension TextStorage: TextAttachmentImageProvider { +extension TextStorage: TextAttachmentDelegate { func textAttachment( _ textAttachment: TextAttachment, @@ -892,7 +892,7 @@ extension TextStorage: TextAttachmentImageProvider { } -// MARK: - TextStorage: CommentAttachmentDelegate Conformance +// MARK: - TextStorage: CommentAttachmentDelegate Methods // extension TextStorage: CommentAttachmentDelegate { From 6ec6bbc4236fd919dd581b343c47e52626161931 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 12:26:08 -0300 Subject: [PATCH 59/74] TextAttachment: Weak delegate --- Aztec/Classes/TextKit/TextAttachment.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Aztec/Classes/TextKit/TextAttachment.swift b/Aztec/Classes/TextKit/TextAttachment.swift index 1f8381cb7..a59354e37 100644 --- a/Aztec/Classes/TextKit/TextAttachment.swift +++ b/Aztec/Classes/TextKit/TextAttachment.swift @@ -1,7 +1,7 @@ import Foundation import UIKit -protocol TextAttachmentDelegate { +protocol TextAttachmentDelegate: class { func textAttachment( _ textAttachment: TextAttachment, imageForURL url: URL, @@ -127,7 +127,7 @@ open class TextAttachment: NSTextAttachment fileprivate var glyphImage: UIImage? - var delegate: TextAttachmentDelegate? + weak var delegate: TextAttachmentDelegate? var isFetchingImage: Bool = false From 5b4b3b7ad422edb55a88d06614b7bee5affecc61 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 12:27:37 -0300 Subject: [PATCH 60/74] TextStorage: Attachment Loop Cleanup --- Aztec/Classes/TextKit/TextStorage.swift | 50 +++++++++++++------------ 1 file changed, 26 insertions(+), 24 deletions(-) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 2b9ed3411..2318c4f91 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -1,6 +1,7 @@ import Foundation import UIKit + /// Implemented by a class taking care of handling attachments for the storage. /// protocol TextStorageAttachmentsDelegate { @@ -67,6 +68,7 @@ protocol TextStorageAttachmentsDelegate { func storage(_ storage: TextStorage, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? } + /// Custom NSTextStorage /// open class TextStorage: NSTextStorage { @@ -201,34 +203,34 @@ open class TextStorage: NSTextStorage { let finalString = NSMutableAttributedString(attributedString: attributedString) attributedString.enumerateAttribute(NSAttachmentAttributeName, in: fullRange, options: []) { (object, range, stop) in - guard let object = object, !(object is LineAttachment) else { - return - } - - guard let attachment = object as? NSTextAttachment else { + guard let textAttachment = object as? NSTextAttachment else { assertionFailure("We expected a text attachment object.") return } - - guard let image = attachment.image else { - // We only suppot image attachments for now. All other attachment types are - // stripped for safety. - // - finalString.removeAttribute(NSAttachmentAttributeName, range: range) - return - } - - if let textAttachment = attachment as? TextAttachment { - textAttachment.imageProvider = self - return + + switch textAttachment { + case _ as LineAttachment: + break + case let attachment as CommentAttachment: + attachment.delegate = self + case let attachment as TextAttachment: + attachment.delegate = self + default: + guard let image = textAttachment.image else { + // We only suppot image attachments for now. All other attachment types are + /// stripped for safety. + // + finalString.removeAttribute(NSAttachmentAttributeName, range: range) + return + } + + let replacementAttachment = TextAttachment() + replacementAttachment.delegate = self + replacementAttachment.image = image + replacementAttachment.url = attachmentsDelegate.storage(self, urlForAttachment: replacementAttachment) + + finalString.addAttribute(NSAttachmentAttributeName, value: replacementAttachment, range: range) } - - let replacementAttachment = TextAttachment() - replacementAttachment.imageProvider = self - replacementAttachment.image = image - replacementAttachment.url = attachmentsDelegate.storage(self, urlForAttachment: replacementAttachment) - - finalString.addAttribute(NSAttachmentAttributeName, value: replacementAttachment, range: range) } return finalString From ad1bfe41cb9650ae90fa9d32af50d7e42185a969 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:08:28 -0300 Subject: [PATCH 61/74] TextStorage: Failsafe --- Aztec/Classes/TextKit/TextStorage.swift | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/Aztec/Classes/TextKit/TextStorage.swift b/Aztec/Classes/TextKit/TextStorage.swift index 2318c4f91..fb32bd749 100644 --- a/Aztec/Classes/TextKit/TextStorage.swift +++ b/Aztec/Classes/TextKit/TextStorage.swift @@ -203,6 +203,10 @@ open class TextStorage: NSTextStorage { let finalString = NSMutableAttributedString(attributedString: attributedString) attributedString.enumerateAttribute(NSAttachmentAttributeName, in: fullRange, options: []) { (object, range, stop) in + guard let object = object else { + return + } + guard let textAttachment = object as? NSTextAttachment else { assertionFailure("We expected a text attachment object.") return From 5aad13d7cfcef88e0f4dc963ab34dd76365ab668 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:08:37 -0300 Subject: [PATCH 62/74] TextView: Updates Delegate Signature --- Aztec/Classes/TextKit/TextView.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Aztec/Classes/TextKit/TextView.swift b/Aztec/Classes/TextKit/TextView.swift index 373e29dbb..350b7e2a5 100644 --- a/Aztec/Classes/TextKit/TextView.swift +++ b/Aztec/Classes/TextKit/TextView.swift @@ -90,7 +90,7 @@ public protocol TextViewCommentsDelegate: class { /// /// - Returns: (Optional) UIImage representation of the Comment Attachment. /// - func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage + func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? } From 55ae50e6f6cc804dabfacc03b3b1d0e5137def6e Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:08:46 -0300 Subject: [PATCH 63/74] Implements MoreAttachmentRender --- Example/Example/MoreAttachmentRender.swift | 96 ++++++++++++++++++++++ 1 file changed, 96 insertions(+) create mode 100644 Example/Example/MoreAttachmentRender.swift diff --git a/Example/Example/MoreAttachmentRender.swift b/Example/Example/MoreAttachmentRender.swift new file mode 100644 index 000000000..e1533f945 --- /dev/null +++ b/Example/Example/MoreAttachmentRender.swift @@ -0,0 +1,96 @@ +// +// MoreAttachmentRender.swift +// AztecExample +// +// Created by Jorge Leandro Perez on 3/22/17. +// Copyright © 2017 Automattic Inc. All rights reserved. +// + +import Foundation +import UIKit +import Aztec + + +// MARK: - MoreAttachmentRender: Renders More Comments! +// +class MoreAttachmentRender { + + /// Attachment to be rendered + /// + let attachment: CommentAttachment + + /// Text Color + /// + var textColor = UIColor.gray + + + /// Default Initializer: Returns *nil* whenever the Attachment's text is not *more*. + /// This render is expected to only work with `` comments! + /// + init?(attachment: CommentAttachment) { + self.attachment = attachment + + guard attachment.text == Constants.defaultCommentText else { + return nil + } + } +} + + +// MARK: - TextViewCommentsDelegate Methods +// +extension MoreAttachmentRender: TextViewCommentsDelegate { + + func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { + UIGraphicsBeginImageContextWithOptions(size, false, 0) + + let label = attachment.text.uppercased() + let attributes = [NSForegroundColorAttributeName: textColor] + let colorMessage = NSAttributedString(string: label, attributes: attributes) + + let textRect = colorMessage.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) + let textPosition = CGPoint(x: ((size.width - textRect.width) * 0.5), y: ((size.height - textRect.height) * 0.5)) + colorMessage.draw(in: CGRect(origin: textPosition , size: CGSize(width: size.width, height: textRect.size.height))) + + let path = UIBezierPath() + + let dashes = [ Constants.defaultDashCount, Constants.defaultDashCount ] + path.setLineDash(dashes, count: dashes.count, phase: 0.0) + path.lineWidth = Constants.defaultDashWidth + + let centerY = round(size.height * 0.5) + path.move(to: CGPoint(x: 0, y: centerY)) + path.addLine(to: CGPoint(x: ((size.width - textRect.width) * 0.5) - Constants.defaultDashWidth, y: centerY)) + + path.move(to: CGPoint(x:((size.width + textRect.width) * 0.5) + Constants.defaultDashWidth, y: centerY)) + path.addLine(to: CGPoint(x: size.width, y: centerY)) + + textColor.setStroke() + path.stroke() + + let result = UIGraphicsGetImageFromCurrentImageContext() + UIGraphicsEndImageContext() + + return result + } + + func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { + let padding = textView.textContainer.lineFragmentPadding + let width = lineFragment.width - padding * 2 + + return CGRect(origin: .zero, size: CGSize(width: width, height: Constants.defaultHeight)) + } +} + + +// MARK: - Constants +// +extension MoreAttachmentRender { + + struct Constants { + static let defaultDashCount = CGFloat(8.0) + static let defaultDashWidth = CGFloat(2.0) + static let defaultHeight = CGFloat(44.0) + static let defaultCommentText = "more" + } +} From 9b775fac2f50878a347ef249b97661fbb743ef99 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:08:54 -0300 Subject: [PATCH 64/74] Implements CommentAttachmentRender --- Example/Example/CommentAttachmentRender.swift | 77 +++++++++++++++++++ 1 file changed, 77 insertions(+) create mode 100644 Example/Example/CommentAttachmentRender.swift diff --git a/Example/Example/CommentAttachmentRender.swift b/Example/Example/CommentAttachmentRender.swift new file mode 100644 index 000000000..9ec377127 --- /dev/null +++ b/Example/Example/CommentAttachmentRender.swift @@ -0,0 +1,77 @@ +// +// CommentAttachmentRender.swift +// AztecExample +// +// Created by Jorge Leandro Perez on 3/22/17. +// Copyright © 2017 Automattic Inc. All rights reserved. +// + +import Foundation +import UIKit +import Aztec + + +class CommentAttachmentRender { + + /// Attachment to be rendered + /// + let attachment: CommentAttachment + + /// Text Color + /// + var textColor = UIColor.gray + + + /// Default Initializer + /// + init?(attachment: CommentAttachment) { + self.attachment = attachment + } +} + + +// MARK: - TextViewCommentsDelegate Methods +// +extension CommentAttachmentRender: TextViewCommentsDelegate { + + func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { + UIGraphicsBeginImageContextWithOptions(size, false, 0) + + let message = messageAttributedString() + let targetRect = boundingRect(for: message, size: size) + + message.draw(in: targetRect) + + let result = UIGraphicsGetImageFromCurrentImageContext() + UIGraphicsEndImageContext() + + return result + } + + func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { + let message = messageAttributedString() + let size = CGSize(width: lineFragment.size.width, height: lineFragment.size.height) + let targetRect = boundingRect(for: message, size: size) + + return targetRect + } + + + + func boundingRect(for message: NSAttributedString, size: CGSize) -> CGRect { + let targetBounds = message.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) + let targetPosition = CGPoint(x: ((size.width - targetBounds.width) * 0.5), y: ((size.height - targetBounds.height) * 0.5)) + + return CGRect(origin: targetPosition, size: targetBounds.size) + + } + + func messageAttributedString() -> NSAttributedString { + let attributes: [String: Any] = [ + NSForegroundColorAttributeName: textColor, + NSFontAttributeName: UIFont.systemFont(ofSize: 14) + ] + + return NSAttributedString(string: attachment.text.uppercased(), attributes: attributes) + } +} From d3badb13e62bf1bfb268fd64141f7cc4c0d6a0a1 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:09:01 -0300 Subject: [PATCH 65/74] Updates Project --- Example/AztecExample.xcodeproj/project.pbxproj | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/Example/AztecExample.xcodeproj/project.pbxproj b/Example/AztecExample.xcodeproj/project.pbxproj index 7daa74605..419dede88 100644 --- a/Example/AztecExample.xcodeproj/project.pbxproj +++ b/Example/AztecExample.xcodeproj/project.pbxproj @@ -18,6 +18,8 @@ 607FACDB1AFB9204008FA782 /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 607FACD91AFB9204008FA782 /* Main.storyboard */; }; 607FACDD1AFB9204008FA782 /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 607FACDC1AFB9204008FA782 /* Images.xcassets */; }; 607FACE01AFB9204008FA782 /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 607FACDE1AFB9204008FA782 /* LaunchScreen.xib */; }; + B570B1C91E82D332008CF41E /* MoreAttachmentRender.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */; }; + B570B1CC1E82D343008CF41E /* CommentAttachmentRender.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */; }; E63EF92B1D36A60B00B5BA4B /* EditorDemoController.swift in Sources */ = {isa = PBXBuildFile; fileRef = E63EF92A1D36A60B00B5BA4B /* EditorDemoController.swift */; }; FF6691C21E76CF9200C6A703 /* OptionsTableView.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF6691C11E76CF9200C6A703 /* OptionsTableView.swift */; }; FF9AF5481DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = FF9AF5471DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard */; }; @@ -93,6 +95,8 @@ 607FACDF1AFB9204008FA782 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; }; 607FACE51AFB9204008FA782 /* AztecExample-Tests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = "AztecExample-Tests.xctest"; sourceTree = BUILT_PRODUCTS_DIR; }; 607FACEA1AFB9204008FA782 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; + B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MoreAttachmentRender.swift; sourceTree = ""; }; + B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CommentAttachmentRender.swift; sourceTree = ""; }; E63EF92A1D36A60B00B5BA4B /* EditorDemoController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = EditorDemoController.swift; sourceTree = ""; }; FF6691C11E76CF9200C6A703 /* OptionsTableView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = OptionsTableView.swift; sourceTree = ""; }; FF9AF5471DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; path = AttachmentDetailsViewController.storyboard; sourceTree = ""; }; @@ -174,6 +178,7 @@ 607FACD21AFB9204008FA782 /* Example for WordPress-Aztec-iOS */ = { isa = PBXGroup; children = ( + B570B1CD1E82D349008CF41E /* Renders */, 599F256F1D8BCF57002871D6 /* AppDelegate.swift */, 59D287391D8C599B00B99C80 /* AttachmentDetailsViewController.swift */, FF9AF5471DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard */, @@ -214,6 +219,15 @@ name = "Supporting Files"; sourceTree = ""; }; + B570B1CD1E82D349008CF41E /* Renders */ = { + isa = PBXGroup; + children = ( + B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */, + B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */, + ); + name = Renders; + sourceTree = ""; + }; /* End PBXGroup section */ /* Begin PBXNativeTarget section */ @@ -352,8 +366,10 @@ 599F25701D8BCF57002871D6 /* AppDelegate.swift in Sources */, 59D2873B1D8C599B00B99C80 /* AttachmentDetailsViewController.swift in Sources */, FF6691C21E76CF9200C6A703 /* OptionsTableView.swift in Sources */, + B570B1CC1E82D343008CF41E /* CommentAttachmentRender.swift in Sources */, E63EF92B1D36A60B00B5BA4B /* EditorDemoController.swift in Sources */, 607FACD81AFB9204008FA782 /* ViewController.swift in Sources */, + B570B1C91E82D332008CF41E /* MoreAttachmentRender.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; From 8ec0ab2736476f4b51bf960ab27fa275990f1adc Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 14:09:22 -0300 Subject: [PATCH 66/74] EditorDemoController: Wiring Comments Delegate --- Example/Example/EditorDemoController.swift | 38 ++++++++++++++++++++-- 1 file changed, 36 insertions(+), 2 deletions(-) diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index 3c1c902df..9fc2d953a 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -21,6 +21,7 @@ class EditorDemoController: UIViewController { textView.delegate = self textView.formattingDelegate = self textView.mediaDelegate = self + textView.commentsDelegate = self return textView }() @@ -665,8 +666,41 @@ extension EditorDemoController : Aztec.FormatBarDelegate { } -extension EditorDemoController: TextViewMediaDelegate -{ + +extension EditorDemoController: TextViewCommentsDelegate { + + /// + /// + func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { + if let render = MoreAttachmentRender(attachment: attachment) { + return render.textView(textView, imageForComment: attachment, with: size) + } + + if let render = CommentAttachmentRender(attachment: attachment) { + return render.textView(textView, imageForComment: attachment, with: size) + } + + return nil + } + + /// + /// + func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { + if let render = MoreAttachmentRender(attachment: attachment) { + return render.textView(textView, boundsForComment: attachment, with: lineFragment) + } + + if let render = CommentAttachmentRender(attachment: attachment) { + return render.textView(textView, boundsForComment: attachment, with: lineFragment) + } + + return .zero + } +} + + +extension EditorDemoController: TextViewMediaDelegate { + func textView(_ textView: TextView, imageAtUrl url: URL, onSuccess success: @escaping (UIImage) -> Void, onFailure failure: @escaping (Void) -> Void) -> UIImage { let task = URLSession.shared.dataTask(with: url, completionHandler: { (data, urlResponse, error) in From 0329e169a679674c0bb686a6dc0940ee0785222f Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 15:10:09 -0300 Subject: [PATCH 67/74] CommentAttachmentRender: Properly rendering inline text --- AztecTests/TextStorageTests.swift | 8 ++++++ Example/Example/CommentAttachmentRender.swift | 27 ++++++++++++------- Example/Example/EditorDemoController.swift | 8 ++---- 3 files changed, 28 insertions(+), 15 deletions(-) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index dfa96fd77..d5c5086f6 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -111,6 +111,14 @@ class TextStorageTests: XCTestCase func storage(_ storage: TextStorage, attachment: TextAttachment, imageForURL url: URL, onSuccess success: @escaping (UIImage) -> (), onFailure failure: @escaping () -> ()) -> UIImage { return UIImage() } + + func storage(_ storage: TextStorage, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { + return .zero + } + + func storage(_ storage: TextStorage, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { + return UIImage() + } } func testRemovalOfAttachment() { diff --git a/Example/Example/CommentAttachmentRender.swift b/Example/Example/CommentAttachmentRender.swift index 9ec377127..027cb2981 100644 --- a/Example/Example/CommentAttachmentRender.swift +++ b/Example/Example/CommentAttachmentRender.swift @@ -13,19 +13,23 @@ import Aztec class CommentAttachmentRender { - /// Attachment to be rendered + /// Comment Attachment Text /// - let attachment: CommentAttachment + let defaultText = NSLocalizedString("[COMMENT]", comment: "Comment Attachment Label") /// Text Color /// var textColor = UIColor.gray + /// Text Font + /// + var textFont: UIFont + /// Default Initializer /// - init?(attachment: CommentAttachment) { - self.attachment = attachment + init?(font: UIFont) { + self.textFont = font } } @@ -50,28 +54,33 @@ extension CommentAttachmentRender: TextViewCommentsDelegate { func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { let message = messageAttributedString() + let size = CGSize(width: lineFragment.size.width, height: lineFragment.size.height) - let targetRect = boundingRect(for: message, size: size) + var rect = boundingRect(for: message, size: size) + rect.origin.y = textFont.descender - return targetRect + return rect.integral } +} +// MARK: - Private Methods +// +private extension CommentAttachmentRender { func boundingRect(for message: NSAttributedString, size: CGSize) -> CGRect { let targetBounds = message.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) let targetPosition = CGPoint(x: ((size.width - targetBounds.width) * 0.5), y: ((size.height - targetBounds.height) * 0.5)) return CGRect(origin: targetPosition, size: targetBounds.size) - } func messageAttributedString() -> NSAttributedString { let attributes: [String: Any] = [ NSForegroundColorAttributeName: textColor, - NSFontAttributeName: UIFont.systemFont(ofSize: 14) + NSFontAttributeName: textFont ] - return NSAttributedString(string: attachment.text.uppercased(), attributes: attributes) + return NSAttributedString(string: defaultText, attributes: attributes) } } diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index 9fc2d953a..2a1cd305f 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -669,28 +669,24 @@ extension EditorDemoController : Aztec.FormatBarDelegate { extension EditorDemoController: TextViewCommentsDelegate { - /// - /// func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { if let render = MoreAttachmentRender(attachment: attachment) { return render.textView(textView, imageForComment: attachment, with: size) } - if let render = CommentAttachmentRender(attachment: attachment) { + if let render = CommentAttachmentRender(font: Constants.defaultContentFont) { return render.textView(textView, imageForComment: attachment, with: size) } return nil } - /// - /// func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { if let render = MoreAttachmentRender(attachment: attachment) { return render.textView(textView, boundsForComment: attachment, with: lineFragment) } - if let render = CommentAttachmentRender(attachment: attachment) { + if let render = CommentAttachmentRender(font: Constants.defaultContentFont) { return render.textView(textView, boundsForComment: attachment, with: lineFragment) } From 316b199521229dd10a32b057e3bca31e3c7f98e2 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Wed, 22 Mar 2017 15:36:52 -0300 Subject: [PATCH 68/74] TextStorageTests: Fixing Unit Tests --- AztecTests/TextStorageTests.swift | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/AztecTests/TextStorageTests.swift b/AztecTests/TextStorageTests.swift index d5c5086f6..a8658b5a3 100644 --- a/AztecTests/TextStorageTests.swift +++ b/AztecTests/TextStorageTests.swift @@ -22,7 +22,9 @@ class TextStorageTests: XCTestCase let attributes = [ NSFontAttributeName: UIFont.boldSystemFont(ofSize: 10) ] + let mockDelegate = MockAttachmentsDelegate() let storage = TextStorage() + storage.attachmentsDelegate = mockDelegate storage.append(NSAttributedString(string: "foo")) storage.append(NSAttributedString(string: "bar", attributes: attributes)) storage.append(NSAttributedString(string: "baz")) @@ -43,7 +45,9 @@ class TextStorageTests: XCTestCase let attributes = [ NSFontAttributeName: UIFont.boldSystemFont(ofSize: 10) ] + let mockDelegate = MockAttachmentsDelegate() let storage = TextStorage() + storage.attachmentsDelegate = mockDelegate storage.append(NSAttributedString(string: "foo")) storage.append(NSAttributedString(string: "bar", attributes: attributes)) storage.append(NSAttributedString(string: "baz")) @@ -57,7 +61,9 @@ class TextStorageTests: XCTestCase let attributes = [ NSFontAttributeName: UIFont.boldSystemFont(ofSize: 10) ] + let mockDelegate = MockAttachmentsDelegate() let storage = TextStorage() + storage.attachmentsDelegate = mockDelegate storage.append(NSAttributedString(string: "foo")) storage.append(NSAttributedString(string: "bar", attributes: attributes)) storage.append(NSAttributedString(string: "baz")) @@ -81,8 +87,8 @@ class TextStorageTests: XCTestCase } func testDelegateCallbackWhenAttachmentRemoved() { - let storage = TextStorage() let mockDelegate = MockAttachmentsDelegate() + let storage = TextStorage() storage.attachmentsDelegate = mockDelegate let attachment = storage.insertImage(sourceURL: URL(string:"test://")!, atPosition: 0, placeHolderImage: UIImage()) @@ -159,7 +165,9 @@ class TextStorageTests: XCTestCase } func testBlockquoteToggle() { + let mockDelegate = MockAttachmentsDelegate() let storage = TextStorage() + storage.attachmentsDelegate = mockDelegate storage.append(NSAttributedString(string: "Apply a blockquote")) let blockquoteFormatter = BlockquoteFormatter() storage.toggle(formatter: blockquoteFormatter, at: storage.rangeOfEntireString) @@ -177,6 +185,9 @@ class TextStorageTests: XCTestCase func testLinkInsert() { let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + storage.append(NSAttributedString(string: "Apply a link")) let linkFormatter = LinkFormatter() linkFormatter.attributeValue = URL(string: "www.wordpress.com")! @@ -195,6 +206,9 @@ class TextStorageTests: XCTestCase func testHeaderToggle() { let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + storage.append(NSAttributedString(string: "Apply a header")) let formatter = HeaderFormatter(headerLevel: .h1) storage.toggle(formatter: formatter, at: storage.rangeOfEntireString) @@ -354,6 +368,9 @@ class TextStorageTests: XCTestCase /// func testDeleteAllSelectionWhenContentHasComments() { let storage = TextStorage() + let mockDelegate = MockAttachmentsDelegate() + storage.attachmentsDelegate = mockDelegate + let commentString = "This is a comment" let html = "" storage.setHTML(html, withDefaultFontDescriptor: UIFont.systemFont(ofSize: 14).fontDescriptor) From a40a4f5b327616c15d1535f4907338e23f277342 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 23 Mar 2017 10:49:30 -0300 Subject: [PATCH 69/74] Renames CommentAttachmentRender > Renderer --- ...chmentRender.swift => CommentAttachmentRenderer.swift} | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) rename Example/Example/{CommentAttachmentRender.swift => CommentAttachmentRenderer.swift} (92%) diff --git a/Example/Example/CommentAttachmentRender.swift b/Example/Example/CommentAttachmentRenderer.swift similarity index 92% rename from Example/Example/CommentAttachmentRender.swift rename to Example/Example/CommentAttachmentRenderer.swift index 027cb2981..a88f5cb1b 100644 --- a/Example/Example/CommentAttachmentRender.swift +++ b/Example/Example/CommentAttachmentRenderer.swift @@ -1,5 +1,5 @@ // -// CommentAttachmentRender.swift +// CommentAttachmentRenderer.swift // AztecExample // // Created by Jorge Leandro Perez on 3/22/17. @@ -11,7 +11,7 @@ import UIKit import Aztec -class CommentAttachmentRender { +class CommentAttachmentRenderer { /// Comment Attachment Text /// @@ -36,7 +36,7 @@ class CommentAttachmentRender { // MARK: - TextViewCommentsDelegate Methods // -extension CommentAttachmentRender: TextViewCommentsDelegate { +extension CommentAttachmentRenderer: TextViewCommentsDelegate { func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { UIGraphicsBeginImageContextWithOptions(size, false, 0) @@ -66,7 +66,7 @@ extension CommentAttachmentRender: TextViewCommentsDelegate { // MARK: - Private Methods // -private extension CommentAttachmentRender { +private extension CommentAttachmentRenderer { func boundingRect(for message: NSAttributedString, size: CGSize) -> CGRect { let targetBounds = message.boundingRect(with: size, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) From c5770c6e13bdcbacb07ef7b79c3fc5994ea8fb46 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 23 Mar 2017 10:49:41 -0300 Subject: [PATCH 70/74] Renames MoreAttachmentRender > Renderer --- ...chmentRender.swift => MoreAttachmentRenderer.swift} | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) rename Example/Example/{MoreAttachmentRender.swift => MoreAttachmentRenderer.swift} (93%) diff --git a/Example/Example/MoreAttachmentRender.swift b/Example/Example/MoreAttachmentRenderer.swift similarity index 93% rename from Example/Example/MoreAttachmentRender.swift rename to Example/Example/MoreAttachmentRenderer.swift index e1533f945..aac722030 100644 --- a/Example/Example/MoreAttachmentRender.swift +++ b/Example/Example/MoreAttachmentRenderer.swift @@ -1,5 +1,5 @@ // -// MoreAttachmentRender.swift +// MoreAttachmentRenderer.swift // AztecExample // // Created by Jorge Leandro Perez on 3/22/17. @@ -11,9 +11,9 @@ import UIKit import Aztec -// MARK: - MoreAttachmentRender: Renders More Comments! +// MARK: - MoreAttachmentRenderer: Renders More Comments! // -class MoreAttachmentRender { +class MoreAttachmentRenderer { /// Attachment to be rendered /// @@ -39,7 +39,7 @@ class MoreAttachmentRender { // MARK: - TextViewCommentsDelegate Methods // -extension MoreAttachmentRender: TextViewCommentsDelegate { +extension MoreAttachmentRenderer: TextViewCommentsDelegate { func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { UIGraphicsBeginImageContextWithOptions(size, false, 0) @@ -85,7 +85,7 @@ extension MoreAttachmentRender: TextViewCommentsDelegate { // MARK: - Constants // -extension MoreAttachmentRender { +extension MoreAttachmentRenderer { struct Constants { static let defaultDashCount = CGFloat(8.0) From e403744a875aa3adaba8f23a5b95d4ea46571fb7 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 23 Mar 2017 10:51:06 -0300 Subject: [PATCH 71/74] Updates Demo Project --- Example/AztecExample.xcodeproj/project.pbxproj | 16 ++++++++-------- Example/Example/EditorDemoController.swift | 8 ++++---- 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/Example/AztecExample.xcodeproj/project.pbxproj b/Example/AztecExample.xcodeproj/project.pbxproj index 419dede88..6e9c10116 100644 --- a/Example/AztecExample.xcodeproj/project.pbxproj +++ b/Example/AztecExample.xcodeproj/project.pbxproj @@ -18,8 +18,8 @@ 607FACDB1AFB9204008FA782 /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 607FACD91AFB9204008FA782 /* Main.storyboard */; }; 607FACDD1AFB9204008FA782 /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 607FACDC1AFB9204008FA782 /* Images.xcassets */; }; 607FACE01AFB9204008FA782 /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 607FACDE1AFB9204008FA782 /* LaunchScreen.xib */; }; - B570B1C91E82D332008CF41E /* MoreAttachmentRender.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */; }; - B570B1CC1E82D343008CF41E /* CommentAttachmentRender.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */; }; + B570B1C91E82D332008CF41E /* MoreAttachmentRenderer.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1C81E82D332008CF41E /* MoreAttachmentRenderer.swift */; }; + B570B1CC1E82D343008CF41E /* CommentAttachmentRenderer.swift in Sources */ = {isa = PBXBuildFile; fileRef = B570B1CB1E82D343008CF41E /* CommentAttachmentRenderer.swift */; }; E63EF92B1D36A60B00B5BA4B /* EditorDemoController.swift in Sources */ = {isa = PBXBuildFile; fileRef = E63EF92A1D36A60B00B5BA4B /* EditorDemoController.swift */; }; FF6691C21E76CF9200C6A703 /* OptionsTableView.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF6691C11E76CF9200C6A703 /* OptionsTableView.swift */; }; FF9AF5481DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = FF9AF5471DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard */; }; @@ -95,8 +95,8 @@ 607FACDF1AFB9204008FA782 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; }; 607FACE51AFB9204008FA782 /* AztecExample-Tests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = "AztecExample-Tests.xctest"; sourceTree = BUILT_PRODUCTS_DIR; }; 607FACEA1AFB9204008FA782 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; - B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MoreAttachmentRender.swift; sourceTree = ""; }; - B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CommentAttachmentRender.swift; sourceTree = ""; }; + B570B1C81E82D332008CF41E /* MoreAttachmentRenderer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MoreAttachmentRenderer.swift; sourceTree = ""; }; + B570B1CB1E82D343008CF41E /* CommentAttachmentRenderer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CommentAttachmentRenderer.swift; sourceTree = ""; }; E63EF92A1D36A60B00B5BA4B /* EditorDemoController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = EditorDemoController.swift; sourceTree = ""; }; FF6691C11E76CF9200C6A703 /* OptionsTableView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = OptionsTableView.swift; sourceTree = ""; }; FF9AF5471DB0E4E200C42ED3 /* AttachmentDetailsViewController.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; path = AttachmentDetailsViewController.storyboard; sourceTree = ""; }; @@ -222,8 +222,8 @@ B570B1CD1E82D349008CF41E /* Renders */ = { isa = PBXGroup; children = ( - B570B1CB1E82D343008CF41E /* CommentAttachmentRender.swift */, - B570B1C81E82D332008CF41E /* MoreAttachmentRender.swift */, + B570B1CB1E82D343008CF41E /* CommentAttachmentRenderer.swift */, + B570B1C81E82D332008CF41E /* MoreAttachmentRenderer.swift */, ); name = Renders; sourceTree = ""; @@ -366,10 +366,10 @@ 599F25701D8BCF57002871D6 /* AppDelegate.swift in Sources */, 59D2873B1D8C599B00B99C80 /* AttachmentDetailsViewController.swift in Sources */, FF6691C21E76CF9200C6A703 /* OptionsTableView.swift in Sources */, - B570B1CC1E82D343008CF41E /* CommentAttachmentRender.swift in Sources */, + B570B1CC1E82D343008CF41E /* CommentAttachmentRenderer.swift in Sources */, E63EF92B1D36A60B00B5BA4B /* EditorDemoController.swift in Sources */, 607FACD81AFB9204008FA782 /* ViewController.swift in Sources */, - B570B1C91E82D332008CF41E /* MoreAttachmentRender.swift in Sources */, + B570B1C91E82D332008CF41E /* MoreAttachmentRenderer.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; diff --git a/Example/Example/EditorDemoController.swift b/Example/Example/EditorDemoController.swift index 2a1cd305f..963783bec 100644 --- a/Example/Example/EditorDemoController.swift +++ b/Example/Example/EditorDemoController.swift @@ -670,11 +670,11 @@ extension EditorDemoController : Aztec.FormatBarDelegate { extension EditorDemoController: TextViewCommentsDelegate { func textView(_ textView: TextView, imageForComment attachment: CommentAttachment, with size: CGSize) -> UIImage? { - if let render = MoreAttachmentRender(attachment: attachment) { + if let render = MoreAttachmentRenderer(attachment: attachment) { return render.textView(textView, imageForComment: attachment, with: size) } - if let render = CommentAttachmentRender(font: Constants.defaultContentFont) { + if let render = CommentAttachmentRenderer(font: Constants.defaultContentFont) { return render.textView(textView, imageForComment: attachment, with: size) } @@ -682,11 +682,11 @@ extension EditorDemoController: TextViewCommentsDelegate { } func textView(_ textView: TextView, boundsForComment attachment: CommentAttachment, with lineFragment: CGRect) -> CGRect { - if let render = MoreAttachmentRender(attachment: attachment) { + if let render = MoreAttachmentRenderer(attachment: attachment) { return render.textView(textView, boundsForComment: attachment, with: lineFragment) } - if let render = CommentAttachmentRender(font: Constants.defaultContentFont) { + if let render = CommentAttachmentRenderer(font: Constants.defaultContentFont) { return render.textView(textView, boundsForComment: attachment, with: lineFragment) } From d1aa6a967cfc3eeac364e64cd6ad386fd71df5fe Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Thu, 23 Mar 2017 11:55:46 -0300 Subject: [PATCH 72/74] Nukes Comments --- Example/Example/CommentAttachmentRenderer.swift | 8 -------- Example/Example/MoreAttachmentRenderer.swift | 8 -------- 2 files changed, 16 deletions(-) diff --git a/Example/Example/CommentAttachmentRenderer.swift b/Example/Example/CommentAttachmentRenderer.swift index a88f5cb1b..8add6fe1b 100644 --- a/Example/Example/CommentAttachmentRenderer.swift +++ b/Example/Example/CommentAttachmentRenderer.swift @@ -1,11 +1,3 @@ -// -// CommentAttachmentRenderer.swift -// AztecExample -// -// Created by Jorge Leandro Perez on 3/22/17. -// Copyright © 2017 Automattic Inc. All rights reserved. -// - import Foundation import UIKit import Aztec diff --git a/Example/Example/MoreAttachmentRenderer.swift b/Example/Example/MoreAttachmentRenderer.swift index aac722030..7bf6dbc71 100644 --- a/Example/Example/MoreAttachmentRenderer.swift +++ b/Example/Example/MoreAttachmentRenderer.swift @@ -1,11 +1,3 @@ -// -// MoreAttachmentRenderer.swift -// AztecExample -// -// Created by Jorge Leandro Perez on 3/22/17. -// Copyright © 2017 Automattic Inc. All rights reserved. -// - import Foundation import UIKit import Aztec From d6925ce21030c91723504f8783c9fcd63a53c539 Mon Sep 17 00:00:00 2001 From: Jorge Leandro Perez Date: Fri, 24 Mar 2017 11:03:59 -0300 Subject: [PATCH 73/74] Marking AttachmentRenderer(s) as Final --- Example/Example/CommentAttachmentRenderer.swift | 2 +- Example/Example/MoreAttachmentRenderer.swift | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Example/Example/CommentAttachmentRenderer.swift b/Example/Example/CommentAttachmentRenderer.swift index 8add6fe1b..c801d52ca 100644 --- a/Example/Example/CommentAttachmentRenderer.swift +++ b/Example/Example/CommentAttachmentRenderer.swift @@ -3,7 +3,7 @@ import UIKit import Aztec -class CommentAttachmentRenderer { +final class CommentAttachmentRenderer { /// Comment Attachment Text /// diff --git a/Example/Example/MoreAttachmentRenderer.swift b/Example/Example/MoreAttachmentRenderer.swift index 7bf6dbc71..247c37d87 100644 --- a/Example/Example/MoreAttachmentRenderer.swift +++ b/Example/Example/MoreAttachmentRenderer.swift @@ -5,7 +5,7 @@ import Aztec // MARK: - MoreAttachmentRenderer: Renders More Comments! // -class MoreAttachmentRenderer { +final class MoreAttachmentRenderer { /// Attachment to be rendered /// From 6f104929a72f43737cda35df34490d0849cad9c3 Mon Sep 17 00:00:00 2001 From: Sergio Estevao Date: Mon, 27 Mar 2017 20:50:08 +0100 Subject: [PATCH 74/74] Update pod spec version. --- WordPress-Aztec-iOS.podspec | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/WordPress-Aztec-iOS.podspec b/WordPress-Aztec-iOS.podspec index bcf2c0d91..ebb6fcdba 100644 --- a/WordPress-Aztec-iOS.podspec +++ b/WordPress-Aztec-iOS.podspec @@ -8,7 +8,7 @@ Pod::Spec.new do |s| s.name = 'WordPress-Aztec-iOS' - s.version = '0.5a7' + s.version = '0.5a7.1' s.summary = 'The native HTML Editor.' # This description is used to generate tags and improve search results.