Выбор нескольких строк в UIPickerView - PullRequest
13 голосов
/ 25 мая 2010

Я хочу выбрать несколько строк в UIPickerView, поэтому у меня была идея показать мои данные в таблице и добавить эту таблицу в качестве подпредставления для выбора. Я пробовал это, но не удалось.

Есть предложения как это сделать?

Ответы [ 7 ]

14 голосов
/ 22 января 2012

Вы можете сделать это без UITableView, используя только UITapGestureRecognizer:)

Кроме того, добавьте NSMutableArray *selectedItems где-нибудь в вашем .h файле.

- (UIView *)pickerView:(UIPickerView *)pickerView viewForRow:(NSInteger)row forComponent:(NSInteger)component reusingView:(UIView *)view {
    UITableViewCell *cell = (UITableViewCell *)view;

    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:nil];
        [cell setBackgroundColor:[UIColor clearColor]];
        [cell setBounds: CGRectMake(0, 0, cell.frame.size.width -20 , 44)];
        UITapGestureRecognizer *singleTapGestureRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(toggleSelection:)];
        singleTapGestureRecognizer.numberOfTapsRequired = 1;
        [cell addGestureRecognizer:singleTapGestureRecognizer];
    }

    if ([selectedItems indexOfObject:[NSNumber numberWithInt:row]] != NSNotFound) {
        [cell setAccessoryType:UITableViewCellAccessoryCheckmark];
    } else {
        [cell setAccessoryType:UITableViewCellAccessoryNone];
    }
    cell.textLabel.text = [datasource objectAtIndex:row];
    cell.tag = row;

    return cell;
}

- (void)toggleSelection:(UITapGestureRecognizer *)recognizer {
    NSNumber *row = [NSNumber numberWithInt:recognizer.view.tag];
    NSUInteger index = [selectedItems indexOfObject:row];
    if (index != NSNotFound) {
        [selectedItems removeObjectAtIndex:index];
        [(UITableViewCell *)(recognizer.view) setAccessoryType:UITableViewCellAccessoryNone];
    } else {
        [selectedItems addObject:row];
        [(UITableViewCell *)(recognizer.view) setAccessoryType:UITableViewCellAccessoryCheckmark];
    }
}
13 голосов
/ 25 мая 2010

«принятый» пользовательский интерфейс для множественного выбора на iPhone должен использовать UITableView с галочками (т.е. не использовать UIPickerView для множественного выбора).

Однако, если вам необходимо, здесь есть инструкции о том, как его подделать http://www.iphonedevsdk.com/forum/iphone-sdk-development/14634-uipickerview-multiple-selection.html, сделав пустой сборщик и поместив вид таблицы поверх него.

7 голосов
/ 18 января 2011

Реализован быстрый взлом для получения поведения множественного выбора UIPickerView (как в Mobile Safari) без добавления других представлений перед представлением выбора, если кому-то интересно: https://github.com/alexleutgoeb/ALPickerView

Улучшения высоко ценятся!

2 голосов
/ 18 августа 2017

Следующий код работает для iOS 10 . У меня не было возможности протестировать его на старых версиях, но я думаю, что он должен работать. Идея похожа на @ suda, но она добавляет один распознаватель касаний непосредственно в представление выбора вместо того, чтобы добавлять распознаватель касаний в каждую строку, так как это не работает на iOS7 +.

Для краткости я не включил полную реализацию протоколов UIPickerViewDataSource и UIPickerViewDelegate, только соответствующие части для реализации множественного выбора.

// 1. Conform to UIGestureRecognizerDelegate protocol
@interface MyViewController () <UIPickerViewDataSource, UIPickerViewDelegate, UIGestureRecognizerDelegate>

@property (nonatomic, strong) NSMutableArray *selectedArray;    // To store which rows are selected
@property (nonatomic, strong) NSArray *dataArray;           // Picker data
@property (weak, nonatomic) IBOutlet UIPickerView *pickerView;

@end

@implementation MyViewController

- (void)viewDidLoad 
{    
    [super viewDidLoad];

    self.selectedArray = [NSMutableArray array]; 
    self.dataArray = @[@"Option 1", @"Option 2", @"Option 3", @"Option 4"];

    // 2. Add tap recognizer to your picker view
    UITapGestureRecognizer *tapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(pickerTapped:)];
    tapRecognizer.delegate = self;
    [self.pickerView addGestureRecognizer:tapRecognizer];
}

- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer
{
    return true;
}

- (void)pickerTapped:(UITapGestureRecognizer *)tapRecognizer
{
    // 3. Find out wich row was tapped (idea based on https://stackoverflow.com/a/25719326) 
    if (tapRecognizer.state == UIGestureRecognizerStateEnded) {
        CGFloat rowHeight = [self.pickerView rowSizeForComponent:0].height;
        CGRect selectedRowFrame = CGRectInset(self.pickerView.bounds, 0.0, (CGRectGetHeight(self.pickerView.frame) - rowHeight) / 2.0 );
        BOOL userTappedOnSelectedRow = (CGRectContainsPoint(selectedRowFrame, [tapRecognizer locationInView:self.pickerView]));
        if (userTappedOnSelectedRow) {
            NSInteger selectedRow = [self.pickerView selectedRowInComponent:0];
            NSUInteger index = [self.selectedArray indexOfObject:[NSNumber numberWithInteger:selectedRow]];

            if (index != NSNotFound) {
                NSLog(@"Row %ld OFF", (long)selectedRow);
                [self.selectedArray removeObjectAtIndex:index];
            } else {
                NSLog(@"Row %ld ON",  (long)selectedRow);
                [self.selectedArray addObject:[NSNumber numberWithInteger:selectedRow]];
            }
            // I don't know why calling reloadAllComponents sometimes scrolls to the first row
            //[self.pickerView reloadAllComponents];
            // This workaround seems to work correctly:
            self.pickerView.dataSource = self;            
            NSLog(@"Rows reloaded");
        }
    }
}

- (UIView *)pickerView:(UIPickerView *)pickerView viewForRow:(NSInteger)row forComponent:(NSInteger)component reusingView:(UIView *)view 
{    
    // 4. Customize your Picker row as needed. 
    // This is a very simple view just to make the point

    UILabel *pickerLabel = (UILabel *)view;

    if (pickerLabel == nil) {
        pickerLabel = [[UILabel alloc] initWithFrame:CGRectMake(0.0, 0.0, 400, 32)];
    }

    BOOL isSelected = [self.selectedArray indexOfObject:[NSNumber numberWithInteger:row]] != NSNotFound;
    NSString *text = [self.dataArray objectAtIndex:row];
    text = [text stringByAppendingString:isSelected ? @"☒ " : @"☐ "];
    [pickerLabel setText:text];

    return pickerLabel;
}


// Do not forget to add the remaining methods to conform the UIPickerViewDataSource and UIPickerViewDelegate protocols!

@end
1 голос
/ 16 ноября 2015
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer 
{
       return true;
}

И вам нужно переопределить этот метод, в остальном, из ios9, распознавание жестов касания не будет работать.

0 голосов
/ 22 января 2019

Вот мой взгляд на эту проблему: https://github.com/aselivanov/UIMultiPicker. Я хотел точно такой же сборщик, что и мобильный Safari для обработки <select multiple>. Это выглядит очень похоже на UIPickerView (держу пари, они используют один и тот же код).

0 голосов
/ 10 сентября 2018

То же разрешение в Swift 4 для этого случая

  //set up tap gesture
        let tapGestureRecognaizer = UITapGestureRecognizer(target: self, action: #selector(pickerTapp))
        tapGestureRecognaizer.delegate = self
        tapGestureRecognaizer.numberOfTapsRequired = 2
        self.pickerView.addGestureRecognizer(tapGestureRecognaizer)

@objc func pickerTapp(tapGesture: UITapGestureRecognizer) {
    if tapGesture.state == .ended {
        // looking for frame selection row
        let selectedItem = dataSource2[pickerView.selectedRow(inComponent: 1)]
        let rowHeight = self.pickerView.rowSize(forComponent: 1).height
        let rowWidth = self.pickerView.rowSize(forComponent: 1).width
        let selectRowFrame = CGRect(x: CGFloat(Int(self.pickerView.frame.width) / self.pickerView.numberOfComponents), y: (self.pickerView.frame.height - rowHeight) / 2, width: rowWidth, height: rowHeight)

        let userTappedOnSelectedRow = selectRowFrame.contains(tapGesture.location(in: self.pickerView))
        // if tap to selection row ....
        if userTappedOnSelectedRow {
            if selectedArray.contains(selectedItem) {
                var index = 0
                for item in selectedArray {
                    if item == selectedItem {
                        selectedArray.remove(at: index)
                    } else {
                        index += 1
                    }
                }
            } else {
                selectedArray.append(selectedItem)
            }
        }

        //reload Data
        self.pickerView.dataSource = self
        self.pickerView.selectRow(selectedRow, inComponent: 1, animated: false)
        self.pickerView(self.pickerView, didSelectRow: selectedRow, inComponent: 1)
    }

полное разрешение: https://github.com/akonoplev/Features/blob/master/MultiSelectUIPickerView/MultiSelectUIPickerView/ViewController.swift

...