177 lines
5.2 KiB
Swift
177 lines
5.2 KiB
Swift
//
|
|
// Copyright 2021-2024 New Vector Ltd.
|
|
//
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
// Please see LICENSE in the repository root for full details.
|
|
//
|
|
|
|
@objc protocol RoomInputToolbarTextViewDelegate: AnyObject {
|
|
func textView(_ textView: RoomInputToolbarTextView, didChangeHeight height: CGFloat)
|
|
func textView(_ textView: RoomInputToolbarTextView, didReceivePasteForMediaFromSender sender: Any?)
|
|
}
|
|
|
|
@objcMembers
|
|
class RoomInputToolbarTextView: UITextView {
|
|
|
|
private var heightConstraint: NSLayoutConstraint!
|
|
private var pillViews = [UIView]()
|
|
|
|
weak var toolbarDelegate: RoomInputToolbarTextViewDelegate?
|
|
|
|
var placeholder: String? {
|
|
didSet {
|
|
setNeedsDisplay()
|
|
}
|
|
}
|
|
|
|
var placeholderColor: UIColor = UIColor(white: 0.8, alpha: 1.0) {
|
|
didSet {
|
|
setNeedsDisplay()
|
|
}
|
|
}
|
|
|
|
var minHeight: CGFloat = 30.0 {
|
|
didSet {
|
|
updateUI()
|
|
}
|
|
}
|
|
|
|
var maxHeight: CGFloat = 0.0 {
|
|
didSet {
|
|
updateUI()
|
|
}
|
|
}
|
|
|
|
override var text: String! {
|
|
willSet {
|
|
flushPills()
|
|
}
|
|
didSet {
|
|
updateUI()
|
|
}
|
|
}
|
|
|
|
override var attributedText: NSAttributedString! {
|
|
willSet {
|
|
flushPills()
|
|
}
|
|
didSet {
|
|
updateUI()
|
|
}
|
|
}
|
|
|
|
override init(frame: CGRect, textContainer: NSTextContainer?) {
|
|
super.init(frame: frame, textContainer: textContainer)
|
|
commonInit()
|
|
}
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
super.init(coder: aDecoder)
|
|
commonInit()
|
|
}
|
|
|
|
private func commonInit() {
|
|
contentMode = .redraw
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(textDidChange), name: UITextView.textDidChangeNotification, object: self)
|
|
|
|
if let heightConstraint = constraints.filter({ $0.firstAttribute == .height && $0.relation == .equal }).first {
|
|
self.heightConstraint = heightConstraint
|
|
} else {
|
|
heightConstraint = self.heightAnchor.constraint(equalToConstant: minHeight)
|
|
addConstraint(heightConstraint)
|
|
}
|
|
}
|
|
|
|
// MARK: - Overrides
|
|
|
|
override func layoutSubviews() {
|
|
super.layoutSubviews()
|
|
updateUI()
|
|
}
|
|
|
|
override func draw(_ rect: CGRect) {
|
|
super.draw(rect)
|
|
|
|
guard attributedText.length == 0, let placeholder = placeholder else {
|
|
return
|
|
}
|
|
|
|
var attributes: [NSAttributedString.Key: Any] = [.foregroundColor: placeholderColor]
|
|
if let font = font {
|
|
attributes[.font] = font
|
|
}
|
|
|
|
let frame = rect.inset(by: .init(top: textContainerInset.top,
|
|
left: textContainerInset.left + textContainer.lineFragmentPadding,
|
|
bottom: textContainerInset.bottom,
|
|
right: textContainerInset.right))
|
|
|
|
placeholder.draw(in: frame, withAttributes: attributes)
|
|
}
|
|
|
|
override var keyCommands: [UIKeyCommand]? {
|
|
return [UIKeyCommand(input: "\r", modifierFlags: [], action: #selector(keyCommandSelector(_:)))]
|
|
}
|
|
|
|
/// Overrides paste to handle images pasted from Safari, passing them up to the input toolbar.
|
|
/// This is required as the pasteboard contains both the image and the image's URL, with the
|
|
/// default implementation choosing to paste the URL and completely ignore the image data.
|
|
override func paste(_ sender: Any?) {
|
|
let pasteboard = MXKPasteboardManager.shared.pasteboard
|
|
let types = pasteboard.types.map { UTI(rawValue: $0) }
|
|
|
|
if types.contains(where: { $0.conforms(to: .image) }) {
|
|
toolbarDelegate?.textView(self, didReceivePasteForMediaFromSender: sender)
|
|
} else {
|
|
super.paste(sender)
|
|
}
|
|
}
|
|
|
|
// MARK: - Private
|
|
|
|
@objc private func textDidChange(notification: Notification) {
|
|
if let sender = notification.object as? RoomInputToolbarTextView, sender == self {
|
|
updateUI()
|
|
}
|
|
}
|
|
|
|
private func updateUI() {
|
|
var height = contentSize.height
|
|
height = minHeight > 0 ? max(height, minHeight) : height
|
|
height = maxHeight > 0 ? min(height, maxHeight) : height
|
|
|
|
// Update placeholder
|
|
self.setNeedsDisplay()
|
|
|
|
guard height != heightConstraint.constant else {
|
|
return
|
|
}
|
|
|
|
heightConstraint.constant = height
|
|
toolbarDelegate?.textView(self, didChangeHeight: height)
|
|
}
|
|
|
|
@objc private func keyCommandSelector(_ keyCommand: UIKeyCommand) {
|
|
guard keyCommand.input == "\r", let delegate = (self.delegate as? RoomInputToolbarView) else {
|
|
return
|
|
}
|
|
|
|
delegate.onTouchUp(inside: delegate.rightInputToolbarButton)
|
|
}
|
|
}
|
|
|
|
extension RoomInputToolbarTextView: PillViewFlusher {
|
|
func registerPillView(_ pillView: UIView) {
|
|
pillViews.append(pillView)
|
|
}
|
|
|
|
private func flushPills() {
|
|
for view in pillViews {
|
|
view.alpha = 0.0
|
|
view.removeFromSuperview()
|
|
}
|
|
pillViews.removeAll()
|
|
}
|
|
}
|