1
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

【Swift】TableViewで検索したもののみ表示する

Last updated at Posted at 2021-03-31

はじめに

今回は検索した文字列のみTableViewで表示させてます。以下のようなものを作ってみます。
ezgif.com-gif-maker.gif

GitHub

実装

1.Main.storyboardで以下のようにUITextFieldUITableViewを配置する。
スクリーンショット 2021-03-31 20.01.16.png
2.文字列を入れておく配列と、検索されたものを入れておく配列を用意する。

ViewController
private var datas = [String]()
private var filterdDatas = [String]()

3.先ほど用意した配列にviewDidLoad()でデータを格納する。

ViewController
datas.append("Swift")
datas.append("Java")
datas.append("Ruby")
datas.append("C++")
datas.append("C")
datas.append("C#")
datas.append("Python")
datas.append("Perl")
datas.append("JavaScript")
datas.append("PHP")
datas.append("Scratch")
datas.append("Scala")
datas.append("COBOL")
datas.append("Curl")
datas.append("Dart")
datas.append("HTML")
datas.append("CSS")

4.delegateを使う

ViewController
tableView.delegate = self
tableView.dataSource = self
tableView.tableFooterView = UIView() // 不要なセパレーターが表示されなくなります
textField.delegate = self

5.UITableViewDelegateを実装

ViewController
extension ViewController: UITableViewDelegate {
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return 60 // 適当に高さを60にしておきます
    }
}

6.UITableViewDataSourceを実装

ViewController
extension ViewController: UITableViewDataSource {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        if !filterdDatas.isEmpty {
            return filterdDatas.count
        }
        return datas.count
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)
        cell.textLabel?.text = filterdDatas.isEmpty ? datas[indexPath.row] : filterdDatas[indexPath.row]
        return cell
    }
}

filterdDatasが空だったら、datas.countを、何かしら入っていたらfilterdDatas.countを返してあげます。

7.UITextFieldDelegateを実装

ViewController
extension ViewController: UITextFieldDelegate {
    func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
        if let text = textField.text {
            (string.count == 0) ? filterText(String(text.dropLast())) : filterText(text+string)
        }
        return true
    }
    
    func filterText(_ query: String) {
        filterdDatas.removeAll()
        datas.forEach { data in
            if data.lowercased().starts(with: query.lowercased()) {
                filterdDatas.append(data)
            }
        }
        tableView.reloadData()
    }
}

textFieldのデリゲートメソッドであるshouldChangeCharactersInは文字が入力される度に処理が呼ばれるメソッドです(参考記事)
(string.count == 0) ? filterText(String(text.dropLast())) : filterText(text+string)stringは入力した文字が一文字です。これが0つまり、textFieldの文字を消去した時にtext.dropLast()します。(参考記事)
lowercased()で大文字を全て小文字に変換しています。(参考記事)
starts(with: )でシーケンスの最初の要素が別のシーケンスの要素と同じであるかどうかをブール値で返します。つまり、大文字小文字を区別せずに、入力された値のquerydataが完全に含まれていればtrueになります。
そして、文字が入力されるたびにtableViewを更新してあげます。

8.フラグを立てる
このままだと、検索結果がない時には全て表示されてしまいます。 
なので、フラグを立てて該当する文字があるかどうか判断します。

ViewController
private var isFilterd = false

そして、tableView.reloadData()の下にisFilterd = trueを追加して、numberOfRowsInSectionreturn isFilterd ? 0 : datas.countに変更します。

おわりに

おわりです。

1
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?