it-roy-ru.com

UIRefreshControl без UITableViewController

Просто любопытно, так как это не сразу кажется возможным, но есть ли хитрый способ использовать новый класс iOS 6 UIRefreshControl без использования подкласса UITableViewController?

Я часто использую UIViewController с подпунктом UITableView и подчиняюсь UITableViewDataSource и UITableViewDelegate вместо того, чтобы использовать UITableViewController напрямую.

305
Keller

По догадкам, основываясь на вдохновении DrummerB, я попытался просто добавить экземпляр UIRefreshControl в качестве подпредставления к своему UITableView. И это волшебным образом работает!

UIRefreshControl *refreshControl = [[UIRefreshControl alloc] init];
[refreshControl addTarget:self action:@selector(handleRefresh:) forControlEvents:UIControlEventValueChanged];
[self.myTableView addSubview:refreshControl];

Это добавляет UIRefreshControl над вашим табличным представлением и работает, как и ожидалось, без использования UITableViewController :)


Правка: Это выше по-прежнему работает, но, как отмечали некоторые, есть небольшое "заикание" при добавлении UIRefreshControl таким образом. Решением этой проблемы является создание экземпляра UITableViewController, а затем установка для него ваших UIRefreshControl и UITableView, т.е.

UITableViewController *tableViewController = [[UITableViewController alloc] init];
tableViewController.tableView = self.myTableView;

self.refreshControl = [[UIRefreshControl alloc] init];
[self.refreshControl addTarget:self action:@selector(getConnections) forControlEvents:UIControlEventValueChanged];
tableViewController.refreshControl = self.refreshControl;
386
Keller

Чтобы устранить заикание, вызванное принятым ответом, вы можете назначить свое UITableViewUITableViewController.

_tableViewController = [[UITableViewController alloc]initWithStyle:UITableViewStylePlain];
[self addChildViewController:_tableViewController];

_tableViewController.refreshControl = [UIRefreshControl new];
[_tableViewController.refreshControl addTarget:self action:@selector(loadStream) forControlEvents:UIControlEventValueChanged];

_theTableView = _tableViewController.tableView;

EDIT:

Способ добавить UIRefreshControl без UITableViewController без заикания и сохранить анимацию Nice после обновления данных в табличном представлении.

UIRefreshControl *refreshControl = [UIRefreshControl new];
[refreshControl addTarget:self action:@selector(handleRefresh:) forControlEvents:UIControlEventValueChanged];
[self.theTableView addSubview:refreshControl];
[self.theTableView sendSubviewToBack:refreshControl];

Позже при обработке обновленных данных ...

- (void)handleRefresh:(UIRefreshControl *)refreshControl {
    [self.theTableView reloadData];
    [self.theTableView layoutIfNeeded];
    [refreshControl endRefreshing];
}
94
Piotr Tomasik

То, что вы попробуете, это использовать представление контейнера внутри ViewController, который вы используете. Вы можете определить чистый подкласс UITableViewController с выделенным табличным представлением и поместить его в ViewController.

21
Tomohisa Takaoka

Ну, UIRefreshControl - это подкласс UIView, так что вы можете использовать его самостоятельно. Я не уверен, хотя, как это делает себя. Рендеринг может просто зависеть от фрейма, но он также может полагаться на UIScrollView или UITableViewController.

В любом случае, это будет скорее взлом, чем элегантное решение. Я рекомендую вам взглянуть на один из доступных сторонних клонов или написать свой собственный.

ODRefreshControl

enter image description here

SlimeRefresh

enter image description here

18
DrummerB

Попробуйте отложить вызов метода refreshControl -endRefresh на долю секунды после того, как tableView перезагрузит свое содержимое, либо с помощью -performSelector:withObject:afterDelay: от NSObject, либо dispatch_after из GCD.

Я создал категорию на UIRefreshControl для этого:

@implementation UIRefreshControl (Delay)

- (void)endRefreshingAfterDelay:(NSTimeInterval)delay {
    dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delay * NSEC_PER_SEC));
    dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
        [self endRefreshing];
    });
}

@end

Я проверил это, и это также работает на коллекцию просмотров. Я заметил, что задержка в 0,01 секунды достаточно:

// My data refresh process here while the refresh control 'isRefreshing'
[self.tableView reloadData];
[self.refreshControl endRefreshingAfterDelay:.01];
7
boliva

IOS 10 Swift 3.0

это просто

import UIKit

class ViewControllerA: UIViewController, UITableViewDataSource, UITableViewDelegate {

    @IBOutlet weak var myTableView: UITableView!

    override func viewDidLoad() {
        super.viewDidLoad()

        myTableView.delegate = self
        myTableView.dataSource = self

        if #available(iOS 10.0, *) {
            let refreshControl = UIRefreshControl()
            let title = NSLocalizedString("PullToRefresh", comment: "Pull to refresh")
            refreshControl.attributedTitle = NSAttributedString(string: title)
            refreshControl.addTarget(self,
                                     action: #selector(refreshOptions(sender:)),
                                     for: .valueChanged)
            myTableView.refreshControl = refreshControl
        }
    }

    @objc private func refreshOptions(sender: UIRefreshControl) {
        // Perform actions to refresh the content
        // ...
        // and then dismiss the control
        sender.endRefreshing()
    }

    // MARK: - Table view data source

    func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 12
    }


    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "reuseIdentifier", for: indexPath)

        cell.textLabel?.text = "Cell \(String(indexPath.row))"
        return cell
    }

}

enter image description here

Если вы хотите узнать о iOS 10 UIRefreshControl, прочитайте ( здесь .

6
Ashok R

Добавление элемента управления обновлением в качестве подпредставления создает пустое пространство над заголовками раздела.

Вместо этого я встроил UITableViewController в свой UIViewController, затем изменил свое свойство tableView, чтобы оно указывало на встроенное, и альт! Минимальные изменения кода. :-)

Шаги:

  1. Создайте новый UITableViewController в раскадровке и вставьте его в свой оригинальный UIViewController
  2. Замените @IBOutlet weak var tableView: UITableView! на тот, что во вновь встроенном UITableViewController, как показано ниже

class MyViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {
    weak var tableView: UITableView!

    override func viewDidLoad() {
        super.viewDidLoad()

        let tableViewController = self.childViewControllers.first as! UITableViewController
        tableView = tableViewController.tableView
        tableView.dataSource = self
        tableView.delegate = self

        // Now we can (properly) add the refresh control
        let refreshControl = UIRefreshControl()
        refreshControl.addTarget(self, action: "handleRefresh:", forControlEvents: .ValueChanged)
        tableViewController.refreshControl = refreshControl
    }

    ...
}
3
cbh2000

Для Swift 2.2.

Сначала создайте UIRefreshControl ().

var refreshControl : UIRefreshControl!

В вашем методе viewDidLoad () добавьте:

refreshControl = UIRefreshControl()
    refreshControl.attributedTitle = NSAttributedString(string: "Refreshing..")
    refreshControl.addTarget(self, action: #selector(YourUIViewController.refresh(_:)), forControlEvents: UIControlEvents.ValueChanged)
    self.tableView.addSubview(refreshControl)

И сделать функцию обновления

func refresh(refreshControl: UIRefreshControl) {

    // do something ...

    // reload tableView
    self.tableView.reloadData()

    // End refreshing
    refreshControl.endRefreshing()
}
2
stakahop

Попробуй это,

Вышеуказанные решения хороши, но tableView.refreshControl доступен только для UITableViewController до iOS 9.x и поставляется в UITableView начиная с iOS 10.x и далее.

Написано в Swift 3 -

let refreshControl = UIRefreshControl()
refreshControl.addTarget(self, action: #selector(FeedViewController.loadNewData), for: UIControlEvents.valueChanged)
// Fix for the RefreshControl Not appearing in background
tableView?.addSubview(refreshControl)
tableView.sendSubview(toBack: refreshControl)
0
Ankit Kumar Gupta

Вот еще одно решение, которое немного отличается.

Мне пришлось использовать его из-за некоторых проблем с иерархией представлений, которые у меня были: я создавал некоторые функциональные возможности, которые требовали передачи представлений в разные места в иерархии представлений, которые ломались при использовании табличного представления UITableViewController, поскольку табличное представление является корневым представлением UITableViewController ( self.view), а не просто обычное представление, оно создало противоречивые иерархии контроллера/представления и вызвало сбой.

По сути, создайте свой собственный подкласс UITableViewController и переопределите loadView, чтобы назначить self.view другое представление, и переопределите свойство tableView, чтобы вернуть отдельный просмотр таблицы.

например:

@interface MyTableVC : UITableViewController
@end

@interface MyTableVC ()
@property (nonatomic, strong) UITableView *separateTableView;
@end

@implementation MyTableVC

- (void)loadView {
    self.view = [[UIView alloc] initWithFrame:CGRectZero];
}

- (UITableView *)tableView {
    return self.separateTableView;
}

- (void)setTableView:(UITableView *)tableView {
    self.separateTableView = tableView;
}

@end

В сочетании с решением Келлера это будет более надежным в том смысле, что tableView теперь является обычным представлением, а не корневым представлением VC, и будет более устойчивым к изменению иерархий представлений. Пример использования этого способа:

MyTableVC *tableViewController = [[MyTableVC alloc] init];
tableViewController.tableView = self.myTableView;

self.refreshControl = [[UIRefreshControl alloc] init];
[self.refreshControl addTarget:self action:@selector(getConnections) forControlEvents:UIControlEventValueChanged];
tableViewController.refreshControl = self.refreshControl;

Существует еще одно возможное использование для этого:

Так как создание подклассов таким образом отделяет self.view от self.tableView, теперь возможно использовать этот UITableViewController как обычный контроллер и добавлять другие подпредставления в self.view без странностей добавления подпредставлений в UITableView, поэтому можно рассмотреть возможность создания их просматривать контроллеры непосредственно подкласс UITableViewController вместо того, чтобы иметь дочерние UITableViewController.

Некоторые вещи, на которые стоит обратить внимание:

Так как мы переопределяем свойство tableView без вызова super, могут быть некоторые вещи, на которые следует обратить внимание, и которые необходимо обработать при необходимости Например, установка tableview в моем примере выше не добавит tableview к self.view и не установит кадр, который вы, возможно, захотите сделать. Кроме того, в этой реализации нет табличного представления по умолчанию, предоставленного вам при создании экземпляра класса, что также можно рассмотреть при добавлении. Я не включаю это здесь, потому что это в каждом конкретном случае, и это решение действительно хорошо согласуется с решением Келлера.

0
psilencer