我使用Below方法检测UITextView
中图像的点按。
`func textView(_ textView: UITextView, shouldInteractWith textAttachment: NSTextAttachment, in characterRange: NSRange, interaction: UITextItemInteraction) -> Bool`
此方法仅在textView.isEditable = false
时调用。
然后我在UITapGestureRecognizer
上添加UITextView
,并在用户点按UITextView
内的图片时调用。但是那点我不知道如何在用户点击哪个图像时如果内部UITextView
内有多个图像。我也得到了UITextView
x和y的点击位置,但不知道我怎么能得到文字,或者它是不是这些点的图像
let TapGesture = UITapGestureRecognizer(target: self, action: #selector(tapDetected(sender:)))
TapGesture.delegate = self
textView.addGestureRecognizer(TapGesture)`
我还尝试在textView.addSubview
中添加一个视图。但是,如果用户想要在此子视图之前或之后键入文本,我也不知道如何更改其位置,就像它的行为与NSAttributedString Images
相同,相应地更改其位置文本。
let imgRect : UIBezierPath = UIBezierPath(rect: CGRect(x: 0, y: 0, width: 30, height: 30))
textView.textContainer.exclusionPaths = [imgRect]
let spacerView : UIView = UIView.init(frame: CGRect(x: 0, y: 0, width: 30, height: 30))
spacerView.backgroundColor = .red
textView.addSubview(spacerView)
任何人都可以告诉我如何在编辑为真时检测到图像上的点击。或者任何人都知道如何在NSAttributedString
图像上添加动作(addTarget)。我还检查了iOS默认的Notes
应用程序,他们正在做我需要的同样的事情。这个功能背后的主要原因我想在UiTextView
中添加附加视频缩略图选项,当用户在键入时点按视频缩略图时,视频将自动在播放器中播放。我附加了从手机录制的视频,这是我的project。
我需要与视频下方完全相同的功能
由于
import UIKit
class ViewController: UIViewController,UITextViewDelegate,UIGestureRecognizerDelegate {
@IBOutlet var textView: UITextView!
@IBOutlet var imageView: UIImageView!
override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
textView.resignFirstResponder()
print("touchesBegan")
}
override func viewDidLoad() {
super.viewDidLoad()
let TapGesture = UITapGestureRecognizer(target: self, action: #selector(tapDetected(sender:)))
TapGesture.delegate = self
textView.addGestureRecognizer(TapGesture)
let imgRect : UIBezierPath = UIBezierPath(rect: CGRect(x: 0, y: 0, width: 30, height: 30))
textView.textContainer.exclusionPaths = [imgRect]
let spacerView : UIView = UIView.init(frame: CGRect(x: 0, y: 0, width: 30, height: 30))
spacerView.backgroundColor = .red
textView.addSubview(spacerView)
textView.attributedText.addObserver(self, forKeyPath: "image", options: .new, context: nil)
textView.attributedText.addObserver(self, forKeyPath: "image", options: .initial, context: nil)
textView.attributedText.addObserver(self, forKeyPath: "image", options: .old, context: nil)
textView.attributedText.addObserver(self, forKeyPath: "image", options: .prior, context: nil)
}
func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool {
return true
}
@IBAction func addImage(_ sender: Any) {
var attributedString :NSMutableAttributedString!
attributedString = NSMutableAttributedString(attributedString:textView.attributedText)
let textAttachment = NSTextAttachment()
textAttachment.image = UIImage(named: "taylor")
let oldWidth = textAttachment.image!.size.width;
//I'm subtracting 10px to make the image display nicely, accounting
//for the padding inside the textView
let scaleFactor = (oldWidth / (textView.frame.size.width - 10))
textAttachment.image = UIImage(cgImage: textAttachment.image!.cgImage!, scale: scaleFactor, orientation: .up)
let attrStringWithImage = NSAttributedString(attachment: textAttachment)
attributedString.append(attrStringWithImage)
textView.attributedText = attributedString;
}
@objc func tapDetected(sender: UITapGestureRecognizer) {
print("Tap On Image")
print("Tap Location",sender.location(in: sender.view))
guard case let senderView = sender.view, (senderView is UITextView) else {
return
}
// calculate layout manager touch location
let textView = senderView as! UITextView, // we sure this is an UITextView, so force casting it
layoutManager = textView.layoutManager
var location = sender.location(in: textView)
location.x -= textView.textContainerInset.left
location.y -= textView.textContainerInset.top
print("location",location)
let textContainer = textView.textContainer,
characterIndex = layoutManager.characterIndex(for: location, in: textContainer, fractionOfDistanceBetweenInsertionPoints: nil),
textStorage = textView.textStorage
guard characterIndex < textStorage.length else {
return
}
}
func textViewDidChange(_ textView: UITextView) {
print("textViewDidChange")
}
func textViewShouldBeginEditing(_ textView: UITextView) -> Bool {
print("textViewShouldBeginEditing")
return true
}
func textViewDidBeginEditing(_ textView: UITextView) {
print("textViewDidBeginEditing")
}
func textViewDidEndEditing(_ textView: UITextView) {
print("textViewDidBeginEditing")
}
func textViewShouldEndEditing(_ textView: UITextView) -> Bool {
print("textViewShouldEndEditing")
return true
}
func textViewDidChangeSelection(_ textView: UITextView) {
print("textViewDidChangeSelection")
print("selectedText", textView.selectedRange.location)
print("textView.attributedText.containsAttachments(in: textView.selectedRange",textView.attributedText.containsAttachments(in: textView.selectedRange))
print("textView.attributedText.attributedSubstring(from: textView.selectedRange)",textView.attributedText.attributedSubstring(from: textView.selectedRange))
let img = textView.getParts()
for i in img {
if let image = i as? UIImage {
imageView.image = image
}
}
}
override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
print("observeValueobserveValueobserveValueobserveValueobserveValue keyPath \(String(describing: keyPath)) change \(String(describing: change)) context \(String(describing: context)) ")
}
func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
print("textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String)")
return true
}
func textView(_ textView: UITextView, shouldInteractWith URL: URL, in characterRange: NSRange, interaction: UITextItemInteraction) -> Bool {
print("textView(_ textView: UITextView, shouldInteractWith URL: URL, in characterRange: NSRange, interaction: UITextItemInteraction) -> Bool ")
return true
}
func textView(_ textView: UITextView, shouldInteractWith textAttachment: NSTextAttachment, in characterRange: NSRange, interaction: UITextItemInteraction) -> Bool {
print("textView(_ textView: UITextView, shouldInteractWith textAttachment: NSTextAttachment, in characterRange: NSRange, interaction: UITextItemInteraction) -> Bool")
imageView.image = textAttachment.image
return true
}
func textView(_ textView: UITextView, shouldInteractWith textAttachment: NSTextAttachment, in characterRange: NSRange) -> Bool {
print("textView(_ textView: UITextView, shouldInteractWith textAttachment: NSTextAttachment, in characterRange: NSRange) -> Bool")
return true
}
}
extension UITextView {
func getParts() -> [AnyObject] {
var parts = [AnyObject]()
let attributedString = self.attributedText
let range = self.selectedRange//NSMakeRange(0, (attributedString?.length)!)
attributedString?.enumerateAttributes(in: range, options: NSAttributedString.EnumerationOptions(rawValue: 0)) { (object, range, stop) in
if object.keys.contains(NSAttributedStringKey.attachment) {
if let attachment = object[NSAttributedStringKey.attachment] as? NSTextAttachment {
if let image = attachment.image {
parts.append(image)
} else if let image = attachment.image(forBounds: attachment.bounds, textContainer: nil, characterIndex: range.location) {
parts.append(image)
}
}
} else {
let stringValue : String = attributedString!.attributedSubstring(from: range).string
if (!stringValue.trimmingCharacters(in: .whitespaces).isEmpty) {
parts.append(stringValue as AnyObject)
}
}
}
return parts
}
}
答案 0 :(得分:0)
首先,创建一个新的NSAttributedStringKey,您将使用它来识别图像附件。然后使用图像创建NSTextAttachment,将其包装在NSMutableAttributedString中并向其添加自定义属性。最后将包装器添加到完整的NSAttributedString并附加UITapGestureRecognizer。
然后在UITapGestureRecognizer的选择器中查找该自定义标记。
大多数位代码:
extension NSAttributedStringKey {
static let imagePath = NSAttributedStringKey(rawValue: "imagePath")
}
...然后在设置文本显示时
let fullString = NSMutableAttributedString()
let imageAttachment = NSTextAttachment()
imageAttachment.image = image
let imageAttributedString: NSMutableAttributedString = NSAttributedString(attachment: imageAttachment).mutableCopy() as! NSMutableAttributedString
let customAttribute = [ NSAttributedStringKey.imagePath: imagePath ]
imageAttributedString.addAttributes(customAttribute, range: NSRange(location: 0, length: imageAttributedString.length))
fullString.append(imageAttributedString)
然后在点击操作调用的函数中:
@objc func onImageTap(_ sender: UITapGestureRecognizer) {
let textView = sender.view as! UITextView
let layoutManager = textView.layoutManager
// location of tap in textView coordinates
var location = sender.location(in: textView)
location.x -= textView.textContainerInset.left;
location.y -= textView.textContainerInset.top;
// character index at tap location
let characterIndex = layoutManager.characterIndex(for: location, in: textView.textContainer, fractionOfDistanceBetweenInsertionPoints: nil)
// if index is valid
if characterIndex < textView.textStorage.length {
// check if the tap location has the custom attribute
let attributeValue = textView.attributedText.attribute(NSAttributedStringKey.imagePath, at: characterIndex, effectiveRange: nil) as? String
if let value = attributeValue {
print("You tapped on \(NSAttributedStringKey.imagePath) and the value is: \(value)")
}
}
}
从那里你知道水龙头在图像中,你有图像框内的坐标,所以你可以使用这个组合来找出图像中的位置。
答案 1 :(得分:0)
由于查尔斯的答案很有暗示性,所以我想让我接近。
我的方式与他没有太大不同,而是添加一个新的属性键,我使用原始的“附件”作为获取图像的键。
因此,创建一个图像阵列,每次添加/删除图像时都要更新该阵列(以确保正确的图像顺序)。
创建用于查看图像的图像查看器(可以从Internet搜索)。
使用Charles答案来检测图像上的点击(我使用“附件”键而不是自定义键)。
点击图像后,打开图像查看器并显示当前图像,应将图像数组解析为图像查看器,以便图像查看器可以正确的顺序显示图像。
这是我的代码的一部分:
@objc func tapOnImage(_ sender: UITapGestureRecognizer) {
let textView = sender.view as! UITextView
let layoutManager = textView.layoutManager
var location = sender.location(in: textView)
location.x -= textView.textContainerInset.left
location.y -= memtextViewoView.textContainerInset.top
let characterIndex = layoutManager.characterIndex(for: location,
in: textView.textContainer,
fractionOfDistanceBetweenInsertionPoints: nil)
if characterIndex < textView.textStorage.length {
let attachment = textView.attributedText.attribute(NSAttributedStringKey.attachment,
at: characterIndex,
effectiveRange: nil) as? NSTextAttachment
if let attachImage = attachment {
print("tap on image: ", attachImage.image)
}
}
}
从上面的代码中,您可以发现单击不同的图像时,控制台将显示不同的对象,从那里您可以使用该图像执行所需的任何操作。
我希望这可以帮助那些坚持此类问题的人。
顺便说一句,我正在xcode 9.2上使用swift 4.1