LNUserSearchViewController.swift 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. //
  2. // LNUserSearchViewController.swift
  3. // Lanu
  4. //
  5. // Created by OneeChan on 2025/12/12.
  6. //
  7. import Foundation
  8. import UIKit
  9. import SnapKit
  10. import MJRefresh
  11. extension UIView {
  12. func pushToUserSearch() {
  13. let vc = LNUserSearchViewController()
  14. navigationController?.pushViewController(vc, animated: true)
  15. }
  16. }
  17. class LNUserSearchViewController: LNViewController {
  18. private let searchInput = UITextField()
  19. private let historyView = LNUserSearchHistoryView()
  20. private let emptyView = LNNoMoreDataView()
  21. private let tableView = UITableView()
  22. private var curKeyword: String?
  23. private var nextTag: String?
  24. private var curList: [LNGameMateSearchResultVO] = []
  25. override func viewDidLoad() {
  26. super.viewDidLoad()
  27. setupViews()
  28. }
  29. }
  30. extension LNUserSearchViewController {
  31. private func searchUser() {
  32. guard let curKeyword, !curKeyword.isEmpty else {
  33. tableView.mj_header?.endRefreshing()
  34. tableView.mj_footer?.endRefreshingWithNoMoreData()
  35. return
  36. }
  37. view.endEditing(true)
  38. historyView.isHidden = true
  39. historyView.addRecord(curKeyword)
  40. tableView.isHidden = false
  41. LNGameMateManager.shared.searchGameMate(keyword: curKeyword, next: nextTag ?? "")
  42. { [weak self] list, next in
  43. guard let self else { return }
  44. guard let list else {
  45. tableView.mj_header?.endRefreshing()
  46. tableView.mj_footer?.endRefreshingWithNoMoreData()
  47. if curList.isEmpty {
  48. emptyView.showNetworkError()
  49. }
  50. return
  51. }
  52. if nextTag?.isEmpty != false {
  53. curList = list
  54. } else {
  55. curList.append(contentsOf: list)
  56. }
  57. nextTag = next
  58. tableView.reloadData()
  59. if curList.isEmpty {
  60. emptyView.showNoData(tips: .init(key: "A00244"))
  61. } else {
  62. emptyView.hide()
  63. }
  64. self.tableView.mj_header?.endRefreshing()
  65. if next?.isEmpty != false {
  66. tableView.mj_footer?.endRefreshingWithNoMoreData()
  67. } else {
  68. tableView.mj_footer?.endRefreshing()
  69. }
  70. }
  71. }
  72. func reportExposure() {
  73. guard tableView.contentOffset.y >= 0 else {
  74. return
  75. }
  76. guard let indexs = tableView.indexPathsForVisibleRows,
  77. !indexs.isEmpty else {
  78. return
  79. }
  80. var items: [LNGameMateSearchResultVO] = []
  81. for index in indexs {
  82. items.append(curList[index.row])
  83. }
  84. LNStatisticManager.shared.reportExposure(uids: items.map({ $0.userNo })) { _ in }
  85. }
  86. }
  87. extension LNUserSearchViewController: UITableViewDataSource, UITableViewDelegate {
  88. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  89. curList.count
  90. }
  91. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  92. let cell = tableView.dequeueReusableCell(
  93. withIdentifier: LNUserSearchItemCell.className,
  94. for: indexPath) as! LNUserSearchItemCell
  95. let item = curList[indexPath.row]
  96. cell.update(item)
  97. return cell
  98. }
  99. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  100. if searchInput.isFirstResponder {
  101. view.endEditing(true)
  102. }
  103. }
  104. func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
  105. reportExposure()
  106. }
  107. func scrollViewDidEndScrollingAnimation(_ scrollView: UIScrollView) {
  108. reportExposure()
  109. }
  110. func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) {
  111. if !decelerate {
  112. reportExposure()
  113. }
  114. }
  115. }
  116. extension LNUserSearchViewController: LNUserSearchHistoryViewDelegate {
  117. func onUserSearchHistoryView(view: LNUserSearchHistoryView, didClick history: String) {
  118. searchInput.text = history
  119. curKeyword = history
  120. nextTag = nil
  121. curList.removeAll()
  122. tableView.reloadData()
  123. tableView.mj_header?.beginRefreshing()
  124. }
  125. }
  126. extension LNUserSearchViewController: UITextFieldDelegate {
  127. func textFieldShouldReturn(_ textField: UITextField) -> Bool {
  128. textField.resignFirstResponder()
  129. guard let text = textField.text?.trimmingCharacters(in: .whitespacesAndNewlines),
  130. !text.isEmpty else { return true }
  131. curKeyword = text
  132. nextTag = nil
  133. curList.removeAll()
  134. tableView.reloadData()
  135. tableView.mj_header?.beginRefreshing()
  136. return true
  137. }
  138. }
  139. extension LNUserSearchViewController {
  140. private func setupViews() {
  141. setupNavBar()
  142. let history = buildHistory()
  143. view.addSubview(history)
  144. history.snp.makeConstraints { make in
  145. make.horizontalEdges.equalToSuperview().inset(16)
  146. make.top.equalToSuperview().offset(12)
  147. }
  148. let list = buildList()
  149. view.addSubview(list)
  150. list.snp.makeConstraints { make in
  151. make.horizontalEdges.equalToSuperview().inset(16)
  152. make.top.equalToSuperview().offset(12)
  153. make.bottom.equalToSuperview()
  154. }
  155. view.onTap { [weak self] in
  156. guard let self else { return }
  157. view.endEditing(true)
  158. }
  159. }
  160. private func setupNavBar() {
  161. let search = UIButton()
  162. search.setTitle(.init(key: "A00245"), for: .normal)
  163. search.setTitleColor(.text_5, for: .normal)
  164. search.titleLabel?.font = .heading_h4
  165. search.setContentHuggingPriority(.defaultHigh, for: .horizontal)
  166. search.setContentCompressionResistancePriority(.defaultHigh, for: .horizontal)
  167. search.addAction(UIAction(handler: { [weak self] _ in
  168. guard let self else { return }
  169. _ = textFieldShouldReturn(searchInput)
  170. }), for: .touchUpInside)
  171. setRightButton(search)
  172. let input = buildSearchInput()
  173. setTitleView(input)
  174. }
  175. private func buildSearchInput() -> UIView {
  176. let container = UIView()
  177. container.backgroundColor = .fill_2
  178. container.layer.cornerRadius = 18
  179. container.snp.makeConstraints { make in
  180. make.height.equalTo(36)
  181. make.width.equalTo(view.bounds.width).priority(.medium)
  182. }
  183. let ic = UIImageView()
  184. ic.image = .icMagnifyingglass
  185. container.addSubview(ic)
  186. ic.snp.makeConstraints { make in
  187. make.leading.equalToSuperview().offset(10)
  188. make.centerY.equalToSuperview()
  189. make.width.height.equalTo(18)
  190. }
  191. searchInput.font = .body_s
  192. searchInput.placeholder = .init(key: "A00246")
  193. searchInput.clearButtonMode = .always
  194. searchInput.returnKeyType = .search
  195. searchInput.enablesReturnKeyAutomatically = true
  196. searchInput.delegate = self
  197. searchInput.becomeFirstResponder()
  198. searchInput.addAction(UIAction(handler: { [weak self] _ in
  199. guard let self else { return }
  200. if searchInput.text?.isEmpty != false {
  201. historyView.isHidden = false
  202. tableView.isHidden = true
  203. }
  204. }), for: .editingChanged)
  205. container.addSubview(searchInput)
  206. searchInput.snp.makeConstraints { make in
  207. make.leading.equalTo(ic.snp.trailing).offset(8)
  208. make.centerY.equalToSuperview()
  209. make.trailing.equalToSuperview().offset(-10)
  210. }
  211. return container
  212. }
  213. private func buildList() -> UIView {
  214. let header = MJRefreshNormalHeader { [weak self] in
  215. guard let self else { return }
  216. nextTag = nil
  217. searchUser()
  218. }
  219. header.lastUpdatedTimeLabel?.isHidden = true
  220. header.stateLabel?.isHidden = true
  221. header.endRefreshingCompletionBlock = { [weak self] in
  222. guard let self else { return }
  223. reportExposure()
  224. }
  225. tableView.mj_header = header
  226. let footer = MJRefreshAutoNormalFooter { [weak self] in
  227. guard let self else { return }
  228. searchUser()
  229. }
  230. footer.setTitle("", for: .noMoreData)
  231. footer.setTitle(.init(key: "A00046"), for: .idle)
  232. footer.endRefreshingCompletionBlock = { [weak self] in
  233. guard let self else { return }
  234. reportExposure()
  235. }
  236. tableView.mj_footer = footer
  237. tableView.isHidden = true
  238. tableView.dataSource = self
  239. tableView.delegate = self
  240. tableView.separatorStyle = .none
  241. tableView.showsVerticalScrollIndicator = false
  242. tableView.showsHorizontalScrollIndicator = false
  243. tableView.register(
  244. LNUserSearchItemCell.self,
  245. forCellReuseIdentifier: LNUserSearchItemCell.className
  246. )
  247. tableView.addSubview(emptyView)
  248. emptyView.snp.makeConstraints { make in
  249. make.centerX.equalToSuperview()
  250. make.centerY.equalToSuperview().multipliedBy(0.6)
  251. }
  252. return tableView
  253. }
  254. private func buildHistory() -> UIView {
  255. historyView.isHidden = false
  256. historyView.delegate = self
  257. return historyView
  258. }
  259. }
  260. #if DEBUG
  261. import SwiftUI
  262. struct LNIMSearchViewControllerPreview: UIViewControllerRepresentable {
  263. func makeUIViewController(context: Context) -> some UIViewController {
  264. LNUserSearchViewController()
  265. }
  266. func updateUIViewController(_ uiViewController: UIViewControllerType, context: Context) {
  267. }
  268. }
  269. #Preview(body: {
  270. LNIMSearchViewControllerPreview()
  271. })
  272. #endif