Длинная нажатие на UitableView
-
29-09-2019 - |
Вопрос
Я хотел бы справиться с длинным нажатием на UITableViewCell
Чтобы распечатать «меню быстрого доступа». Кто -то уже сделал это?
В частности, жест узнает о UITableView
?
Решение
Сначала добавьте узнаватель жеста на данный момент в вид с таблицей:
UILongPressGestureRecognizer *lpgr = [[UILongPressGestureRecognizer alloc]
initWithTarget:self action:@selector(handleLongPress:)];
lpgr.minimumPressDuration = 2.0; //seconds
lpgr.delegate = self;
[self.myTableView addGestureRecognizer:lpgr];
[lpgr release];
Затем в обработке жестов:
-(void)handleLongPress:(UILongPressGestureRecognizer *)gestureRecognizer
{
CGPoint p = [gestureRecognizer locationInView:self.myTableView];
NSIndexPath *indexPath = [self.myTableView indexPathForRowAtPoint:p];
if (indexPath == nil) {
NSLog(@"long press on table view but not on a row");
} else if (gestureRecognizer.state == UIGestureRecognizerStateBegan) {
NSLog(@"long press on table view at row %ld", indexPath.row);
} else {
NSLog(@"gestureRecognizer.state = %ld", gestureRecognizer.state);
}
}
Вы должны быть осторожны с этим, чтобы это не мешало нормальному постукиванию пользователя по ячейке, а также отмечала, что handleLongPress
может выстрелить несколько раз (это будет связано с изменениями состояния распознавания жестов).
Другие советы
Я использовал ответ Анна-Каренины, и он работает почти отлично с очень серьезной ошибкой.
Если вы используете разделы, давление в разделе название раздела даст вам неправильный результат нажатия первой строки в этом разделе, я добавил фиксированную версию ниже (включая фильтрацию фиктивных вызовов на основе состояния жеста, в соответствии с Анна-Каренна предложение).
- (IBAction)handleLongPress:(UILongPressGestureRecognizer *)gestureRecognizer
{
if (gestureRecognizer.state == UIGestureRecognizerStateBegan) {
CGPoint p = [gestureRecognizer locationInView:self.tableView];
NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:p];
if (indexPath == nil) {
NSLog(@"long press on table view but not on a row");
} else {
UITableViewCell *cell = [self.tableView cellForRowAtIndexPath:indexPath];
if (cell.isHighlighted) {
NSLog(@"long press on table view at section %d row %d", indexPath.section, indexPath.row);
}
}
}
}
Вот разъясненная инструкция, объединяющая ответ Dawn Song и ответ Marmor.
Перетащите длинное распознавание жеста и бросьте в свою столовую ячейку. Он прыгнет в нижнюю часть списка слева.
Затем подключите распознаватель жеста так же, как вы подключили кнопку.
Добавить код из Marmor в обработчике действий
- (IBAction)handleLongPress:(UILongPressGestureRecognizer *)sender {
if (sender.state == UIGestureRecognizerStateBegan) {
CGPoint p = [sender locationInView:self.tableView];
NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:p];
if (indexPath == nil) {
NSLog(@"long press on table view but not on a row");
} else {
UITableViewCell *cell = [self.tableView cellForRowAtIndexPath:indexPath];
if (cell.isHighlighted) {
NSLog(@"long press on table view at section %d row %d", indexPath.section, indexPath.row);
}
}
}
}
Ответ в Swift 5 (продолжение ответа Рики в Swift)
Добавить
UIGestureRecognizerDelegate
к вашему ViewController
override func viewDidLoad() {
super.viewDidLoad()
//Long Press
let longPressGesture = UILongPressGestureRecognizer(target: self, action: #selector(handleLongPress))
longPressGesture.minimumPressDuration = 0.5
longPressGesture.delegate = self
self.tableView.addGestureRecognizer(longPressGesture)
}
И функция:
@objc func handleLongPress(longPressGesture: UILongPressGestureRecognizer) {
let p = longPressGesture.location(in: self.tableView)
let indexPath = self.tableView.indexPathForRow(at: p)
if indexPath == nil {
print("Long press on table view, not row.")
} else if longPressGesture.state == UIGestureRecognizer.State.began {
print("Long press on row, at \(indexPath!.row)")
}
}
Похоже, что более эффективно добавить распознаватель непосредственно в ячейку, как показано здесь:
Нажмите и удерживайте для ячейки TableView, тогда и сейчас
(прокрутите пример внизу)
Ответ в Swift:
Добавить делегата UIGestureRecognizerDelegate
к вашему uitableviewcontroller.
В пределах uitebleviewcontroller:
override func viewDidLoad() {
super.viewDidLoad()
let longPressGesture:UILongPressGestureRecognizer = UILongPressGestureRecognizer(target: self, action: "handleLongPress:")
longPressGesture.minimumPressDuration = 1.0 // 1 second press
longPressGesture.delegate = self
self.tableView.addGestureRecognizer(longPressGesture)
}
И функция:
func handleLongPress(longPressGesture:UILongPressGestureRecognizer) {
let p = longPressGesture.locationInView(self.tableView)
let indexPath = self.tableView.indexPathForRowAtPoint(p)
if indexPath == nil {
print("Long press on table view, not row.")
}
else if (longPressGesture.state == UIGestureRecognizerState.Began) {
print("Long press on row, at \(indexPath!.row)")
}
}
Я собрал небольшую категорию на UitableView на основе отличного ответа Анны Каренины.
Таким образом, у вас будет удобный метод делегата, как вы привыкли при работе с обычными видами таблицы. Проверьте это:
// UITableView+LongPress.h
#import <UIKit/UIKit.h>
@protocol UITableViewDelegateLongPress;
@interface UITableView (LongPress) <UIGestureRecognizerDelegate>
@property(nonatomic,assign) id <UITableViewDelegateLongPress> delegate;
- (void)addLongPressRecognizer;
@end
@protocol UITableViewDelegateLongPress <UITableViewDelegate>
- (void)tableView:(UITableView *)tableView didRecognizeLongPressOnRowAtIndexPath:(NSIndexPath *)indexPath;
@end
// UITableView+LongPress.m
#import "UITableView+LongPress.h"
@implementation UITableView (LongPress)
@dynamic delegate;
- (void)addLongPressRecognizer {
UILongPressGestureRecognizer *lpgr = [[UILongPressGestureRecognizer alloc]
initWithTarget:self action:@selector(handleLongPress:)];
lpgr.minimumPressDuration = 1.2; //seconds
lpgr.delegate = self;
[self addGestureRecognizer:lpgr];
}
- (void)handleLongPress:(UILongPressGestureRecognizer *)gestureRecognizer
{
CGPoint p = [gestureRecognizer locationInView:self];
NSIndexPath *indexPath = [self indexPathForRowAtPoint:p];
if (indexPath == nil) {
NSLog(@"long press on table view but not on a row");
}
else {
if (gestureRecognizer.state == UIGestureRecognizerStateBegan) {
// I am not sure why I need to cast here. But it seems to be alright.
[(id<UITableViewDelegateLongPress>)self.delegate tableView:self didRecognizeLongPressOnRowAtIndexPath:indexPath];
}
}
}
Если вы хотите использовать это в UitableViewController, вам, вероятно, нужно подкласс и соответствовать новому протоколу.
Это отлично подходит для меня, надеюсь, это поможет другим!
Swift 3 Ответ, используя современный синтаксис, включает в себя другие ответы и исключил ненужный код.
override func viewDidLoad() {
super.viewDidLoad()
let recognizer = UILongPressGestureRecognizer(target: self, action: #selector(tablePressed))
tableView.addGestureRecognizer(recognizer)
}
@IBAction func tablePressed(_ recognizer: UILongPressGestureRecognizer) {
let point = recognizer.location(in: tableView)
guard recognizer.state == .began,
let indexPath = tableView.indexPathForRow(at: point),
let cell = tableView.cellForRow(at: indexPath),
cell.isHighlighted
else {
return
}
// TODO
}
Просто добавьте uilongpressgesturerecognizer в данную ячейку -прототип в раскадровке, а затем потяните жест в файл .m ViewController, чтобы создать метод действия. Я сделал это, как сказал.
Используйте свойство Uitouch TimeStamp In Touchesbegan, чтобы запустить таймер или остановить его, когда Touchesend