From 876f3941c93ac54c6f081859fb352bb9d7b30d7e Mon Sep 17 00:00:00 2001 From: Mauro <34335419+Velin92@users.noreply.github.com> Date: Fri, 17 Nov 2023 17:29:49 +0100 Subject: [PATCH] [iOS] Fix for dictation not working properly (#875) --- .../WysiwygComposerView.swift | 2 +- .../WysiwygComposerViewModel.swift | 3 +- .../WysiwygComposerView/WysiwygTextView.swift | 34 ++++++++++++++++++- 3 files changed, 36 insertions(+), 3 deletions(-) diff --git a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerView.swift b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerView.swift index 65c09993d..1af9090f6 100644 --- a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerView.swift +++ b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerView.swift @@ -88,7 +88,7 @@ public struct WysiwygComposerView: View { @ViewBuilder private var placeholderView: some View { - if viewModel.isContentEmpty { + if viewModel.isContentEmpty, !viewModel.textView.isDictationRunning { Text(placeholder) .font(Font(UIFont.preferredFont(forTextStyle: .body))) .foregroundColor(placeholderColor) diff --git a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerViewModel.swift b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerViewModel.swift index 3a55f2d9d..de9c667b2 100644 --- a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerViewModel.swift +++ b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygComposerViewModel.swift @@ -556,7 +556,8 @@ private extension WysiwygComposerViewModel { /// Reconciliate the content of the model with the content of the text view. func reconciliateIfNeeded() { do { - guard let replacement = try StringDiffer.replacement(from: attributedContent.text.htmlChars, + guard !textView.isDictationRunning, + let replacement = try StringDiffer.replacement(from: attributedContent.text.htmlChars, to: textView.attributedText.htmlChars) else { return } diff --git a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygTextView.swift b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygTextView.swift index c6a532ab0..312022866 100644 --- a/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygTextView.swift +++ b/platforms/ios/lib/WysiwygComposer/Sources/WysiwygComposer/Components/WysiwygComposerView/WysiwygTextView.swift @@ -44,6 +44,8 @@ protocol WysiwygTextViewDelegate: AnyObject { public protocol MentionDisplayHelper { } public class WysiwygTextView: UITextView { + private(set) var isDictationRunning = false + /// Internal delegate for the text view. weak var wysiwygDelegate: WysiwygTextViewDelegate? @@ -53,12 +55,42 @@ public class WysiwygTextView: UITextView { override public init(frame: CGRect, textContainer: NSTextContainer?) { super.init(frame: frame, textContainer: textContainer) - contentMode = .redraw + commonInit() } required init?(coder: NSCoder) { super.init(coder: coder) + commonInit() + } + + private func commonInit() { contentMode = .redraw + NotificationCenter.default.addObserver(self, + selector: #selector(textInputCurrentInputModeDidChange), + name: UITextInputMode.currentInputModeDidChangeNotification, + object: nil) + } + + @objc private func textInputCurrentInputModeDidChange(notification: Notification) { + // We don't care about the input mode if this is not the first responder + guard isFirstResponder else { + return + } + + guard let inputMode = textInputMode?.primaryLanguage, + inputMode == "dictation" else { + isDictationRunning = false + return + } + isDictationRunning = true + } + + override public func dictationRecordingDidEnd() { + isDictationRunning = false + } + + override public func dictationRecognitionFailed() { + isDictationRunning = false } /// Register a pill view that has been added through `NSTextAttachmentViewProvider`.