Skip to content
This repository has been archived by the owner on Sep 27, 2024. It is now read-only.

Commit

Permalink
[iOS] Text View is now created by the Wrapper, and swapped in case it…
Browse files Browse the repository at this point in the history
…'s re-rendered (#931)

Co-authored-by: Stefan Ceriu <stefan.ceriu@gmail.com>
  • Loading branch information
Velin92 and stefanceriu authored Feb 8, 2024
1 parent b814ed4 commit 49307af
Show file tree
Hide file tree
Showing 7 changed files with 101 additions and 59 deletions.
2 changes: 1 addition & 1 deletion platforms/ios/example/Wysiwyg/Views/ContentView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ struct ContentView: View {
}
HStack {
Spacer()
if viewModel.textView.autocorrectionType == .yes {
if viewModel.textView?.autocorrectionType == .yes {
Image(systemName: "text.badge.checkmark")
.foregroundColor(.green)
.padding(.horizontal, 16)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -85,7 +85,7 @@ public struct WysiwygComposerView: View {

@ViewBuilder
private var placeholderView: some View {
if viewModel.isContentEmpty, !viewModel.textView.isDictationRunning {
if viewModel.isContentEmpty, viewModel.textView?.isDictationRunning != true {
Text(placeholder)
.font(Font(UIFont.preferredFont(forTextStyle: .body)))
.foregroundColor(placeholderColor)
Expand Down Expand Up @@ -121,8 +121,15 @@ struct UITextViewWrapper: UIViewRepresentable {
}

func makeUIView(context: Context) -> WysiwygTextView {
let textView = viewModel.textView

// Default text container have a slightly different behaviour
// than what iOS would use if textContainer is nil, this
// fixes issues with background color not working on newline characters.
let layoutManager = NSLayoutManager()
let textStorage = NSTextStorage()
let textContainer = NSTextContainer()
textStorage.addLayoutManager(layoutManager)
layoutManager.addTextContainer(textContainer)
let textView = WysiwygTextView(frame: .zero, textContainer: textContainer)
textView.accessibilityIdentifier = "WysiwygComposer"
textView.font = UIFont.preferredFont(forTextStyle: .body)
textView.autocapitalizationType = .sentences
Expand All @@ -137,12 +144,16 @@ struct UITextViewWrapper: UIViewRepresentable {
textView.clipsToBounds = false
textView.tintColor = UIColor.tintColor
textView.wysiwygDelegate = context.coordinator
viewModel.textView = textView
viewModel.updateCompressedHeightIfNeeded()

return textView
}

func updateUIView(_ uiView: WysiwygTextView, context: Context) { }
func updateUIView(_ uiView: WysiwygTextView, context: Context) {
if uiView !== viewModel.textView {
viewModel.textView = uiView
}
}

func makeCoordinator() -> Coordinator {
Coordinator(viewModel.replaceText,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,19 +26,17 @@ import UIKit
public class WysiwygComposerViewModel: WysiwygComposerViewModelProtocol, ObservableObject {
// MARK: - Public

/// The textView that the model manages.
public private(set) var textView = {
// Default text container have a slightly different behaviour
// than what iOS would use if textContainer is nil, this
// fixes issues with background color not working on nealine characters.
let layoutManager = NSLayoutManager()
let textStorage = NSTextStorage()
let textContainer = NSTextContainer()
textStorage.addLayoutManager(layoutManager)
layoutManager.addTextContainer(textContainer)
return WysiwygTextView(frame: .zero, textContainer: textContainer)
}()

/// The textView that the model currently manages.
public var textView: WysiwygTextView? {
didSet {
guard let textView else {
return
}
textView.linkTextAttributes[.foregroundColor] = parserStyle.linkColor
textView.apply(attributedContent)
}
}

/// The composer minimal height.
public let minHeight: CGFloat
/// The mention replacer defined by the hosting application.
Expand Down Expand Up @@ -79,7 +77,7 @@ public class WysiwygComposerViewModel: WysiwygComposerViewModelProtocol, Observa
public var parserStyle: HTMLParserStyle {
didSet {
// In case of a color change, this will refresh the attributed text
textView.linkTextAttributes[.foregroundColor] = parserStyle.linkColor
textView?.linkTextAttributes[.foregroundColor] = parserStyle.linkColor
let update = model.setContentFromHtml(html: content.html)
applyUpdate(update)
updateTextView()
Expand Down Expand Up @@ -141,7 +139,6 @@ public class WysiwygComposerViewModel: WysiwygComposerViewModelProtocol, Observa
self.parserStyle = parserStyle
self.mentionReplacer = mentionReplacer

textView.linkTextAttributes[.foregroundColor] = parserStyle.linkColor
model = ComposerModelWrapper()
model.delegate = self
// Publish composer empty state.
Expand All @@ -156,7 +153,10 @@ public class WysiwygComposerViewModel: WysiwygComposerViewModelProtocol, Observa
guard let self = self else { return }
// Improves a lot the user experience by keeping the selected range always visible when there are changes in the size.
DispatchQueue.main.async {
self.textView.scrollRangeToVisible(self.textView.selectedRange)
guard let textView = self.textView else {
return
}
textView.scrollRangeToVisible(textView.selectedRange)
}
}
.store(in: &cancellables)
Expand Down Expand Up @@ -221,7 +221,7 @@ public extension WysiwygComposerViewModel {
/// Clear the content of the composer.
func clearContent() {
if plainTextMode {
textView.attributedText = NSAttributedString(string: "", attributes: defaultTextAttributes)
textView?.attributedText = NSAttributedString(string: "", attributes: defaultTextAttributes)
updateCompressedHeightIfNeeded()
} else {
applyUpdate(model.clear())
Expand Down Expand Up @@ -283,6 +283,10 @@ public extension WysiwygComposerViewModel {

public extension WysiwygComposerViewModel {
func updateCompressedHeightIfNeeded() {
guard let textView else {
return
}

let idealTextHeight = textView
.sizeThatFits(CGSize(width: textView.bounds.size.width,
height: CGFloat.greatestFiniteMagnitude)
Expand All @@ -303,8 +307,8 @@ public extension WysiwygComposerViewModel {

// This fixes a bug where the attributed string keeps link and inline code formatting
// when they are the last formatting to be deleted
if textView.attributedText.length == 0 {
textView.typingAttributes = defaultTextAttributes
if textView?.attributedText.length == 0 {
textView?.typingAttributes = defaultTextAttributes
}

let update: ComposerUpdate
Expand Down Expand Up @@ -332,7 +336,7 @@ public extension WysiwygComposerViewModel {
case let .update(newState):
if newState[.link] != actionStates[.link] {
applyUpdate(update, skipTextViewUpdate: true)
textView.apply(attributedContent)
textView?.apply(attributedContent)
updateCompressedHeightIfNeeded()
return false
}
Expand All @@ -346,7 +350,7 @@ public extension WysiwygComposerViewModel {

func select(range: NSRange) {
do {
guard let text = textView.attributedText, !plainTextMode else { return }
guard let text = textView?.attributedText, !plainTextMode else { return }
let htmlSelection = try text.htmlRange(from: range)
Logger.viewModel.logDebug(["Sel(att): \(range)",
"Sel: \(htmlSelection)",
Expand All @@ -365,15 +369,17 @@ public extension WysiwygComposerViewModel {

func didUpdateText() {
if plainTextMode {
if textView.text.isEmpty != isContentEmpty {
isContentEmpty = textView.text.isEmpty
if let textView {
if textView.text.isEmpty != isContentEmpty {
isContentEmpty = textView.text.isEmpty
}
plainTextContent = textView.attributedText
}
plainTextContent = textView.attributedText
} else {
reconciliateIfNeeded()
applyPendingFormatsIfNeeded()
}

updateCompressedHeightIfNeeded()
}

Expand Down Expand Up @@ -405,7 +411,7 @@ public extension WysiwygComposerViewModel {

@available(iOS 16.0, *)
func getIdealSize(_ proposal: ProposedViewSize) -> CGSize {
guard let width = proposal.width else { return .zero }
guard let textView, let width = proposal.width else { return .zero }

let idealHeight = textView
.sizeThatFits(CGSize(width: width, height: CGFloat.greatestFiniteMagnitude))
Expand Down Expand Up @@ -437,7 +443,7 @@ private extension WysiwygComposerViewModel {
// Note: this makes replaceAll act like .keep on cases where we expect the text
// view to be properly updated by the system.
if !skipTextViewUpdate {
textView.apply(attributedContent)
textView?.apply(attributedContent)
updateCompressedHeightIfNeeded()
}
case let .select(startUtf16Codeunit: start,
Expand Down Expand Up @@ -543,7 +549,7 @@ private extension WysiwygComposerViewModel {
if let mentionReplacer {
attributed = mentionReplacer.postProcessMarkdown(in: attributed)
}
textView.attributedText = attributed
textView?.attributedText = attributed
updateCompressedHeightIfNeeded()
} else {
let update = model.setContentFromMarkdown(markdown: computeMarkdownContent())
Expand All @@ -556,7 +562,7 @@ private extension WysiwygComposerViewModel {
/// Reconciliate the content of the model with the content of the text view.
func reconciliateIfNeeded() {
do {
guard !textView.isDictationRunning,
guard let textView, !textView.isDictationRunning,
let replacement = try StringDiffer.replacement(from: attributedContent.text.htmlChars,
to: textView.attributedText.htmlChars) else {
return
Expand All @@ -580,7 +586,7 @@ private extension WysiwygComposerViewModel {
case StringDifferError.tooComplicated,
StringDifferError.insertionsDontMatchRemovals:
// Restore from the model, as otherwise the composer will enter a broken state
textView.apply(attributedContent)
textView?.apply(attributedContent)
updateCompressedHeightIfNeeded()
Logger.viewModel.logError(["Reconciliate failed, content has been restored from the model"],
functionName: #function)
Expand All @@ -600,7 +606,7 @@ private extension WysiwygComposerViewModel {
func applyPendingFormatsIfNeeded() {
guard hasPendingFormats else { return }

textView.apply(attributedContent)
textView?.apply(attributedContent)
updateCompressedHeightIfNeeded()
hasPendingFormats = false
}
Expand All @@ -609,6 +615,9 @@ private extension WysiwygComposerViewModel {
///
/// - Returns: A markdown string.
func computeMarkdownContent() -> String {
guard let textView else {
return ""
}
let markdownContent: String
if let mentionReplacer, let attributedText = textView.attributedText {
// `MentionReplacer` should restore altered content to valid markdown.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ import UIKit

public protocol WysiwygComposerViewModelProtocol: AnyObject {
/// The textView that the model manages.
var textView: WysiwygTextView { get }
var textView: WysiwygTextView? { get set }

/// Whether the current content of the composer is empty.
var isContentEmpty: Bool { get }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -103,10 +103,10 @@ extension WysiwygComposerViewModelTests {

private extension WysiwygComposerViewModelTests {
func assertAutoCorrectEnabled() {
XCTAssertEqual(viewModel.textView.autocorrectionType, .yes)
XCTAssertEqual(viewModel.textView?.autocorrectionType, .yes)
}

func assertAutocorrectDisabled() {
XCTAssertEqual(viewModel.textView.autocorrectionType, .no)
XCTAssertEqual(viewModel.textView?.autocorrectionType, .no)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,11 @@ extension WysiwygComposerViewModelTests {

func testMentionsStatBySettingRoomIDMention() {
viewModel.setMention(url: "https://matrix.to/#/!room:matrix.org", name: "Room", mentionType: .room)
XCTAssertEqual(viewModel.getMentionsState(), MentionsState(userIds: [], roomIds: ["!room:matrix.org"], roomAliases: [], hasAtRoomMention: false))
XCTAssertEqual(viewModel.getMentionsState(),
MentionsState(userIds: [],
roomIds: ["!room:matrix.org"],
roomAliases: [],
hasAtRoomMention: false))
}

func testMentionsStateBySettingRoomIDMentionFromContent() {
Expand Down
Loading

0 comments on commit 49307af

Please sign in to comment.