summaryrefslogtreecommitdiff
path: root/Sora/Views/SearchSuggestionsView.swift
blob: 2f2d5729542143fbfed49d6e0b29b3249e408e1e (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
import SwiftUI

struct SearchSuggestionsView: View {
  var items: [Either<BooruTag, BooruSearchQuery>]
  @Binding var searchText: String

  private var lastSearchTag: String {
    String(searchText.split(separator: " ").last ?? "").lowercased()
  }

  private var filteredItems: [Either<BooruTag, BooruSearchQuery>] {
    let matchCanidateTag = lastSearchTag
    let matchingTagsLimit = 50

    guard !matchCanidateTag.isEmpty else { return [] }

    var seenTags = Set<String>()
    var matchingTags: [Either<BooruTag, BooruSearchQuery>] = []

    matchingTags.reserveCapacity(matchingTagsLimit)

    for item in items {
      if matchingTags.count >= matchingTagsLimit { break }

      switch item {
      case .left(let tag):
        let lowercasedName = tag.name.lowercased()

        if lowercasedName.contains(matchCanidateTag),
          seenTags.insert(lowercasedName).inserted
        {
          matchingTags.append(item)
        }

      case .right(let query):
        for tagName in query.tags {
          let lowercasedTagName = tagName.lowercased()

          if lowercasedTagName.contains(matchCanidateTag),
            seenTags.insert(lowercasedTagName).inserted
          {
            matchingTags.append(item)

            break
          }
        }
      }
    }

    return matchingTags
  }

  var body: some View {
    ForEach(filteredItems, id: \.self) { item in
      switch item {
      case .left(let tag):
        Button {
          if let range = searchText.range(of: lastSearchTag, options: .backwards) {
            searchText.replaceSubrange(range, with: tag.name)
          }
        } label: {
          Text(tag.name)
        }

      case .right(let query):
        Button {
          if let range = searchText.range(of: lastSearchTag, options: .backwards),
            let matchingTag = query.tags.first(where: { $0.lowercased().contains(lastSearchTag) })
          {
            searchText.replaceSubrange(range, with: matchingTag)
          }
        } label: {
          if let matchingTag = query.tags.first(where: { $0.lowercased().contains(lastSearchTag) })
          {
            Text(matchingTag)
          } else {
            Text(query.tags.first ?? "")
          }
        }
      }
    }
  }
}