Перетащите UIView между UIViews
у меня есть объект UIView X, который содержится в объекте UIView A. Я хочу иметь возможность коснуться X и удалить его из объекта A и переместить его в объект B (другой UIView). Оба объекта A & B находятся внутри одного и того же super UIView.
A B
_____ _____
| | | |
| X | -> | |
|___| |___|
это то, что у меня есть до сих пор.
@implementation X_UIView
float deltaX;
float deltaY;
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
[self.superview.superview addSubview:self]; //pop dragged view outside of container view
CGPoint beginCenter = self.center;
UITouch * touch = [touches anyObject];
CGPoint touchPoint = [touch locationInView:self.superview];
deltaX = touchPoint.x - beginCenter.x;
deltaY = touchPoint.y - beginCenter.y;
}
- (void) touchesMoved:(NSSet*)touches withEvent:(UIEvent*)event {
UITouch * touch = [touches anyObject];
CGPoint touchPoint = [touch locationInView:self.superview];
// Set the correct center when touched
touchPoint.x -= deltaX;
touchPoint.y -= deltaY;
self.center = touchPoint;
}
- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
//discover view that event ended was over and add self as a subview.
}
@end
2 ответов
вызов [[touches anyObject] locationInView: self.superview]
чтобы получить точку под пальцем в контейнер. Тогда пошли self.superview -hitTest:withEvent:
чтобы узнать, что вид X находится внутри. Обратите внимание, что он всегда будет возвращать X, поэтому вам придется переопределить либо -pointIsInside:withEvent:
или -hitTest:withEvent:
чтобы вернуть nil, а вы отстаете. Этот вид kludge-причина, по которой я бы реализовал такое отслеживание в представлении контейнера, а не в перетаскиваемом представлении.
С iOS 11 вы можете решить свою проблему с помощью API перетаскивания. Следующий код Swift 4 показывает, как это сделать.
ViewContainer.Свифт!--12-->
import MobileCoreServices
import UIKit
enum ViewContainerError: Error {
case invalidType, unarchiveFailure
}
class ViewContainer: NSObject {
let view: UIView
required init(view: UIView) {
self.view = view
}
}
extension ViewContainer: NSItemProviderReading {
static var readableTypeIdentifiersForItemProvider = [kUTTypeData as String]
static func object(withItemProviderData data: Data, typeIdentifier: String) throws -> Self {
if typeIdentifier == kUTTypeData as String {
guard let view = NSKeyedUnarchiver.unarchiveObject(with: data) as? UIView else { throw ViewContainerError.unarchiveFailure }
return self.init(view: view)
} else {
throw ViewContainerError.invalidType
}
}
}
extension ViewContainer: NSItemProviderWriting {
static var writableTypeIdentifiersForItemProvider = [kUTTypeData as String]
func loadData(withTypeIdentifier typeIdentifier: String, forItemProviderCompletionHandler completionHandler: @escaping (Data?, Error?) -> Void) -> Progress? {
if typeIdentifier == kUTTypeData as String {
let data = NSKeyedArchiver.archivedData(withRootObject: view)
completionHandler(data, nil)
} else {
completionHandler(nil, ViewContainerError.invalidType)
}
return nil
}
}
ViewController.Свифт!--12-->
import UIKit
class ViewController: UIViewController {
let redView = UIView()
let greenView = UIView()
override func viewDidLoad() {
super.viewDidLoad()
let blueView = UIView()
blueView.backgroundColor = .blue
greenView.backgroundColor = .green
greenView.isUserInteractionEnabled = true
greenView.addSubview(blueView)
setConstraintsInSuperView(forView: blueView)
redView.backgroundColor = .red
redView.isUserInteractionEnabled = true
let greenViewDropInteraction = UIDropInteraction(delegate: self)
let greenViewDragInteraction = UIDragInteraction(delegate: self)
greenViewDragInteraction.isEnabled = true
redView.addInteraction(greenViewDragInteraction)
greenView.addInteraction(greenViewDropInteraction)
let redViewDropInteraction = UIDropInteraction(delegate: self)
let redViewDragInteraction = UIDragInteraction(delegate: self)
redViewDragInteraction.isEnabled = true
greenView.addInteraction(redViewDragInteraction)
redView.addInteraction(redViewDropInteraction)
let stackView = UIStackView(arrangedSubviews: [greenView, redView])
view.addSubview(stackView)
stackView.distribution = .fillEqually
stackView.frame = view.bounds
stackView.autoresizingMask = [.flexibleWidth, .flexibleHeight]
}
}
extension ViewController {
// MARK: - Helper methods
func setConstraintsInSuperView(forView subView: UIView) {
subView.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate(NSLayoutConstraint.constraints(withVisualFormat: "H:|-[subView]-|", options: [], metrics: nil, views: ["subView": subView]))
NSLayoutConstraint.activate(NSLayoutConstraint.constraints(withVisualFormat: "V:|-[subView]-|", options: [], metrics: nil, views: ["subView": subView]))
}
}
extension ViewController: UIDragInteractionDelegate {
func dragInteraction(_ interaction: UIDragInteraction, itemsForBeginning session: UIDragSession) -> [UIDragItem] {
guard let containedView = interaction.view?.subviews.first else { return [] }
let viewContainer = ViewContainer(view: containedView)
let itemProvider = NSItemProvider(object: viewContainer)
let item = UIDragItem(itemProvider: itemProvider)
item.localObject = viewContainer.view
return [item]
}
func dragInteraction(_ interaction: UIDragInteraction, sessionWillBegin session: UIDragSession) {
guard let containedView = interaction.view?.subviews.first else { return }
containedView.removeFromSuperview()
}
func dragInteraction(_ interaction: UIDragInteraction, previewForLifting item: UIDragItem, session: UIDragSession) -> UITargetedDragPreview? {
guard let containedView = interaction.view?.subviews.first else { return nil }
return UITargetedDragPreview(view: containedView)
}
func dragInteraction(_ interaction: UIDragInteraction, item: UIDragItem, willAnimateCancelWith animator: UIDragAnimating) {
animator.addCompletion { _ in
guard let containedView = item.localObject as? UIView else { return }
interaction.view!.addSubview(containedView)
self.setConstraintsInSuperView(forView: containedView)
}
}
func dragInteraction(_ interaction: UIDragInteraction, prefersFullSizePreviewsFor session: UIDragSession) -> Bool {
return true
}
}
extension ViewController: UIDropInteractionDelegate {
func dropInteraction(_ interaction: UIDropInteraction, canHandle session: UIDropSession) -> Bool {
return session.canLoadObjects(ofClass: ViewContainer.self) && session.items.count == 1
}
func dropInteraction(_ interaction: UIDropInteraction, sessionDidUpdate session: UIDropSession) -> UIDropProposal {
let dropLocation = session.location(in: view)
let operation: UIDropOperation
if interaction.view!.frame.contains(dropLocation) && session.localDragSession != nil {
operation = .move
} else {
operation = .cancel
}
return UIDropProposal(operation: operation)
}
func dropInteraction(_ interaction: UIDropInteraction, performDrop session: UIDropSession) {
session.loadObjects(ofClass: ViewContainer.self) { viewContainers in
guard let viewContainers = viewContainers as? [ViewContainer], let viewContainer = viewContainers.first else { return }
interaction.view!.addSubview(viewContainer.view)
self.setConstraintsInSuperView(forView: viewContainer.view)
}
}
}