iOS - UITableViewCell 中的 UIGestureRecognizer 类似于 Snapchat
iOS - UIGestureRecognizer In UITableViewCell Similar To Snapchat
我正在尝试在我的练习应用程序中实现与 Snapchat 非常相似的功能,您可以在其中向右或向左拖动 UITableViewCell
,并且在拖动时,视图后面的图像慢慢出现,直到它到达某个点,然后它开始一个 segue 或 PageViewController
类型 segue 到另一个视图控制器,你可以用来与你的朋友聊天。
起初,我尝试使用屏幕边缘平移手势识别器,但只有从屏幕边缘开始滑动时才有效。我需要能够从 UITableViewCell
内的任何地方滑动。
我需要的功能演示在 Snapchat 中,当你在你朋友的一个 table 单元格上慢慢向右滑动时,你会更清楚地看到它,它会慢慢显示消息图标的图像并最终引导转到另一个可以聊天的视图。
那么平移手势识别器就足够了吗?如果是这样,完成这项工作的最佳方法是什么?我看过有关平移手势识别器的教程,但我看不出在滑动一定距离后它最终如何导致另一个视图控制器。我想我可以将消息图标放在向右滑动时可能出现的显示的 table 单元格内容后面,但我如何才能实现如此流畅的功能?
学习这个真的会增加我流畅的用户体验。任何建议或方法将不胜感激,谢谢。
编辑:请在 Objective C 中留下答案。我不知道 Swift.
您可以为此创建自定义 TableViewCell。下面的代码应该可以满足您的要求。只需将代表添加到您的 ViewController
protocol TableViewCellDelegate {
func something()
}
class TableViewCell: UITableViewCell {
var startSegue = false
var label: UILabel // Or UIView, whatever you want to show
var delegate: TableViewCellDelegate? // Delegate to your ViewController to perform the segue
required init(coder aDecoder: NSCoder) {
fatalError("NSCoding not supported")
}
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
// Utility method for creating the label / view
func createLabel() -> UILabel {
let label = UILabel(frame: CGRect.nullRect)
// Add label customization here
return label
}
// Create "check" & "cross" labels for context cues
label = createLabel() // Create the label
label.text = "\u{2713}" // Add check symbol as text
super.init(style: style, reuseIdentifier: reuseIdentifier)
addSubview(label)
// Add a pan recognizer
var recognizer = UIPanGestureRecognizer(target: self, action: "handleSwipe:")
recognizer.delegate = self
addGestureRecognizer(recognizer)
}
let cueMargin: CGFloat = 10.0, cueWidth: CGFloat = 50.0
override func layoutSubviews() {
// The label starts out of view
label.frame = CGRect(x: bounds.size.width + cueMargin, y: 0, width: cueWidth, height: bounds.size.height)
}
func handleSwipe(recognizer: UIPanGestureRecognizer) {
let translation = recognizer.translationInView(self)
if recognizer.state == .Changed {
center = CGPointMake(center.x + translation.x, center.y)
startSegue = frame.origin.x < -frame.size.width / 2.0 // If swiped over 50%, return true
label.textColor = startSegue ? UIColor.redColor() : UIColor.whiteColor() // If swiped over 50%, become red
recognizer.setTranslation(CGPointZero, inView: self)
}
if recognizer.state == .Ended {
let originalFrame = CGRect(x: 0, y: frame.origin.y, width: bounds.size.width, height: bounds.size.height)
if delegate != nil {
startSegue ? delegate!.something() : UIView.animateWithDuration(0.2) { self.frame = originalFrame }
}
}
}
// Need this to handle the conflict between vertical swipes of tableviewgesture and pangesture
override func gestureRecognizerShouldBegin(gestureRecognizer: UIGestureRecognizer) -> Bool {
if let panGestureRecognizer = gestureRecognizer as? UIPanGestureRecognizer {
let velocity = panGestureRecognizer.velocityInView(superview!)
if fabs(velocity.x) >= fabs(velocity.y) { return true }
return false
}
return false
}
}
我从这个tutorial
得到这个
编辑:
在您的 ViewController 中,您必须 'register' 使用此 TableViewCell:
yourTableName.registerClass(TableViewCell.self, forCellReuseIdentifier: "cellIdentifier")
cellForRowAtIndexPath
的 TableViewDataSource
看起来像:
func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("cellIdentifier", forIndexPath: indexPath) as TableViewCell
cell.delegate = self
// Add stuff for your other labels like:
// "cell.name = item.name" etc
return cell
}
如果要将数据传回 ViewController
,只需使用委托即可。
我正在尝试在我的练习应用程序中实现与 Snapchat 非常相似的功能,您可以在其中向右或向左拖动 UITableViewCell
,并且在拖动时,视图后面的图像慢慢出现,直到它到达某个点,然后它开始一个 segue 或 PageViewController
类型 segue 到另一个视图控制器,你可以用来与你的朋友聊天。
起初,我尝试使用屏幕边缘平移手势识别器,但只有从屏幕边缘开始滑动时才有效。我需要能够从 UITableViewCell
内的任何地方滑动。
我需要的功能演示在 Snapchat 中,当你在你朋友的一个 table 单元格上慢慢向右滑动时,你会更清楚地看到它,它会慢慢显示消息图标的图像并最终引导转到另一个可以聊天的视图。
那么平移手势识别器就足够了吗?如果是这样,完成这项工作的最佳方法是什么?我看过有关平移手势识别器的教程,但我看不出在滑动一定距离后它最终如何导致另一个视图控制器。我想我可以将消息图标放在向右滑动时可能出现的显示的 table 单元格内容后面,但我如何才能实现如此流畅的功能?
学习这个真的会增加我流畅的用户体验。任何建议或方法将不胜感激,谢谢。
编辑:请在 Objective C 中留下答案。我不知道 Swift.
您可以为此创建自定义 TableViewCell。下面的代码应该可以满足您的要求。只需将代表添加到您的 ViewController
protocol TableViewCellDelegate {
func something()
}
class TableViewCell: UITableViewCell {
var startSegue = false
var label: UILabel // Or UIView, whatever you want to show
var delegate: TableViewCellDelegate? // Delegate to your ViewController to perform the segue
required init(coder aDecoder: NSCoder) {
fatalError("NSCoding not supported")
}
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
// Utility method for creating the label / view
func createLabel() -> UILabel {
let label = UILabel(frame: CGRect.nullRect)
// Add label customization here
return label
}
// Create "check" & "cross" labels for context cues
label = createLabel() // Create the label
label.text = "\u{2713}" // Add check symbol as text
super.init(style: style, reuseIdentifier: reuseIdentifier)
addSubview(label)
// Add a pan recognizer
var recognizer = UIPanGestureRecognizer(target: self, action: "handleSwipe:")
recognizer.delegate = self
addGestureRecognizer(recognizer)
}
let cueMargin: CGFloat = 10.0, cueWidth: CGFloat = 50.0
override func layoutSubviews() {
// The label starts out of view
label.frame = CGRect(x: bounds.size.width + cueMargin, y: 0, width: cueWidth, height: bounds.size.height)
}
func handleSwipe(recognizer: UIPanGestureRecognizer) {
let translation = recognizer.translationInView(self)
if recognizer.state == .Changed {
center = CGPointMake(center.x + translation.x, center.y)
startSegue = frame.origin.x < -frame.size.width / 2.0 // If swiped over 50%, return true
label.textColor = startSegue ? UIColor.redColor() : UIColor.whiteColor() // If swiped over 50%, become red
recognizer.setTranslation(CGPointZero, inView: self)
}
if recognizer.state == .Ended {
let originalFrame = CGRect(x: 0, y: frame.origin.y, width: bounds.size.width, height: bounds.size.height)
if delegate != nil {
startSegue ? delegate!.something() : UIView.animateWithDuration(0.2) { self.frame = originalFrame }
}
}
}
// Need this to handle the conflict between vertical swipes of tableviewgesture and pangesture
override func gestureRecognizerShouldBegin(gestureRecognizer: UIGestureRecognizer) -> Bool {
if let panGestureRecognizer = gestureRecognizer as? UIPanGestureRecognizer {
let velocity = panGestureRecognizer.velocityInView(superview!)
if fabs(velocity.x) >= fabs(velocity.y) { return true }
return false
}
return false
}
}
我从这个tutorial
得到这个编辑: 在您的 ViewController 中,您必须 'register' 使用此 TableViewCell:
yourTableName.registerClass(TableViewCell.self, forCellReuseIdentifier: "cellIdentifier")
cellForRowAtIndexPath
的 TableViewDataSource
看起来像:
func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("cellIdentifier", forIndexPath: indexPath) as TableViewCell
cell.delegate = self
// Add stuff for your other labels like:
// "cell.name = item.name" etc
return cell
}
如果要将数据传回 ViewController
,只需使用委托即可。