问题描述
我在 ViewController 顶部创建了一个自定义 textField 栏来过滤 UITableView
中的数据值.由于我有嵌套类型的 JSON,因此无法正确了解如何为其使用过滤器.
I have created a custom textField bar on top of the ViewController to filter data values in UITableView
. As I have a nested type of JSON so couldn't properly get how to use filter for it.
- 我想将
textField
实现为带有 TableView 的搜索栏.附上截图. - 我需要过滤的值是
pickList ->文本字段
textSearchChange
为文本搜索添加了函数.
- I want to implement
textField
as a search bar with TableView. screenshot attached. - Value which I need to filter is
pickList -> textField
textSearchChange
function added for text search.
数据是部分明智的,然后是值,并且已经在 tableView 中显示.
Data is section wise and then values and is already displaying in tableView.
型号:
struct SectionList : Codable {
let title : String?
var items : [Item]?
}
struct PickListData: Codable {
let items: [Item]?
}
struct Item : Codable {
let actionType : Int?
var textField : String?
var pickList: [SectionList]?
var selection: [Item]?
let selectedValue: [String]?
let version: Int?
let masterId: Int?
let itemValue: String?
}
视图控制器代码:
import UIKit
class ViewController: UIViewController, UITextFieldDelegate {
@IBOutlet weak var searchTxt: UITextField!
@IBOutlet weak var tableView: UITableView!
var AppData: Item?
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
tableView.delegate = self
tableView.dataSource = self
searchTxt.delegate = self
readDataList()
}
func readDataList(){
if let url = Bundle.main.url(forResource: "list", withExtension: "json") {
do {
let data = try Data(contentsOf: url)
let decoder = JSONDecoder()
let response = try decoder.decode(PickListData.self, from: data)
let res = response.items?.filter { $0.actionType == 101}
AppData = res?.first
print(AppData)
self.tableView.reloadData()
} catch {
print("error:\(error)")
}
}
}
@IBAction func textSearchChange(_ sender: UITextField) {
print("search")
}
}
extension ViewController: UITableViewDelegate, UITableViewDataSource {
func numberOfSections(in tableView: UITableView) -> Int {
return AppData?.pickList?.count ?? 0
}
func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
return AppData?.pickList?[section].title
}
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return AppData?.pickList?[section].items?.count ?? 0
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = UITableViewCell()
let dic = AppData?.pickList?[indexPath.section].items?[indexPath.row]//.pickList?[indexPath.row].title
//AppData?[indexPath.section].pickList[indexPath.row].items
//print(dic)
cell.textLabel?.text = dic?.textField
return cell
}
}
JSON 数据:
{
"items": [
{
"actionType": 101,
"version": 3,
"pickList": [
{
"title": "Sayaç yeri seçimi",
"items": [
{
"textField": "Sayaç Yeri Seçiniz",
"itemValue": "0"
},
{
"textField": "Sayaç daire girişinde",
"itemValue": "1"
},
{
"textField": "Sayaç apt. girişinde",
"itemValue": "2"
},
{
"textField": "Sayaç bodrumda",
"itemValue": "3"
},
{
"textField": "Sayaç çatı katında",
"itemValue": "4"
},
{
"textField": "Sayaç bahçede (Müstakil)",
"itemValue": "5"
},
{
"textField": "Sayaç bina dışında",
"itemValue": "6"
},
{
"textField": "Sayaç balkonda",
"itemValue": "7"
},
{
"textField": "Sayaç daire içinde",
"itemValue": "8"
},
{
"textField": "Sayaç istasyon içinde",
"itemValue": "9"
}
]
}
]
}
]
}
更新代码: 但这是错误的我想用 TextField
Updated Code: but this is wrong I want to search with TextField
func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
let searchText = textField.text! + string
Filterdata = (AppData?.pickList?.filter({(($0.title!).localizedCaseInsensitiveContains(searchText))}))!
if(Filterdata.count == 0){
isSearch = false
}else{
isSearch = true
}
self.tableView.reloadData();
return true
}
图片:
推荐答案
请按照以下步骤操作以获得可行的解决方案
Please follow this steps for the working solution
* 第 1 步 -> 替换结构
struct SectionList : Codable {
let title : String?
var items : [RowItems]?
mutating func filterData(string: String) {
self.items = self.items?.filter({ (item) -> Bool in
if item.textField?.contains(string) == true {
return true
}
return false
})
}
}
struct Item : Codable {
let actionType : Int?
var pickList: [SectionList]?
let version: Int?
mutating func filterData(string: String) {
guard var tempList = pickList else { return }
for index in 0..<tempList.count {
tempList[index].filterData(string: string)
}
pickList = tempList
}
}
* 第 2 步 -> 再创建一个变量来保存原始数据
var globalAppData: Item?
* 第 3 步 -> 在解析 json 时为 globalAppData 赋值
let res = response.items?.filter { $0.actionType == 101}
AppData = res?.first
globalAppData = AppData
* 第 4 步 -> 为文本更改添加观察者
searchTxt.addTarget(self, action: #selector(textSearchChange(_:)), for: .editingChanged)
* 第 5 步 -> 替换 textDidChange 方法
@IBAction func textSearchChange(_ sender: UITextField){
var tempData = globalAppData
if let text = sender.text, text.isEmpty == false {
tempData?.filterData(string: text)
}
AppData = tempData
self.tableView.reloadData()
}
* 可选步骤 6 ->
如果您想要不区分大小写的搜索,请替换此行
If you want case insensitive search then replace this line
item.textField?.contains(string)
与
item.textField?.lowercased().contains(string.lowercased())
这篇关于如何实现 TextField 搜索栏来过滤 tableview 值 Swift的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!