|
21 | 21 | let totalResults = 0; |
22 | 22 | let userResultsById: Record<string, Models.User<Record<string, unknown>>> = {}; // use a hash map so we can quickly look up a user by id |
23 | 23 | let selected: Record<string, Models.Target> = {}; |
24 | | - let selectedSize = 0; |
25 | | - let selectedUsers = 0; |
26 | | - let hasSelection = false; |
27 | 24 |
|
28 | 25 | function reset() { |
29 | 26 | offset = 0; |
|
38 | 35 |
|
39 | 36 | async function request() { |
40 | 37 | if (!show) return; |
| 38 | +
|
41 | 39 | const queries = [Query.limit(5), Query.offset(offset)]; |
42 | 40 |
|
43 | 41 | if (providerType === MessagingProviderType.Email) { |
|
52 | 50 | }); |
53 | 51 |
|
54 | 52 | totalResults = response.total; |
55 | | - userResultsById = {}; |
56 | | - response.users.forEach((user) => { |
57 | | - if (providerType !== null) { |
58 | | - user.targets = user.targets.filter( |
59 | | - (target) => target.providerType === providerType |
60 | | - ); |
61 | | - } |
62 | | - userResultsById = { |
63 | | - ...userResultsById, |
64 | | - [user.$id]: user |
65 | | - }; |
66 | | - }); |
| 53 | +
|
| 54 | + userResultsById = Object.fromEntries( |
| 55 | + response.users.map((user) => { |
| 56 | + const filteredUser = |
| 57 | + providerType !== null |
| 58 | + ? { |
| 59 | + ...user, |
| 60 | + targets: user.targets.filter( |
| 61 | + (target) => target.providerType === providerType |
| 62 | + ) |
| 63 | + } |
| 64 | + : user; |
| 65 | + return [user.$id, filteredUser]; |
| 66 | + }) |
| 67 | + ); |
67 | 68 | } |
68 | 69 |
|
69 | | - function onUserSelection(event: CustomEvent<boolean>, userId: string) { |
| 70 | + function onUserSelection(event: CustomEvent<boolean | 'indeterminate'>, userId: string) { |
70 | 71 | const user = userResultsById[userId]; |
| 72 | + const shouldSelect = event.detail === 'indeterminate' || event.detail === true; |
71 | 73 |
|
72 | | - if (event.detail) { |
73 | | - user.targets.forEach((target) => { |
74 | | - selected = { |
75 | | - ...selected, |
76 | | - [target.$id]: target |
77 | | - }; |
78 | | - }); |
79 | | - } else { |
80 | | - user.targets.forEach((target) => { |
81 | | - const { [target.$id]: _, ...rest } = selected; |
82 | | - selected = rest; |
83 | | - }); |
84 | | - } |
| 74 | + const updatedSelected = { ...selected }; |
| 75 | +
|
| 76 | + user.targets.forEach((target) => { |
| 77 | + if (shouldSelect) { |
| 78 | + updatedSelected[target.$id] = target; |
| 79 | + } else { |
| 80 | + delete updatedSelected[target.$id]; |
| 81 | + } |
| 82 | + }); |
| 83 | +
|
| 84 | + selected = updatedSelected; |
85 | 85 | } |
86 | 86 |
|
87 | 87 | function onTargetSelection(event: CustomEvent<boolean>, target: Models.Target) { |
88 | 88 | if (event.detail) { |
89 | | - selected = { |
90 | | - ...selected, |
91 | | - [target.$id]: target |
92 | | - }; |
| 89 | + selected = { ...selected, [target.$id]: target }; |
93 | 90 | } else { |
94 | 91 | const { [target.$id]: _, ...rest } = selected; |
95 | 92 | selected = rest; |
|
107 | 104 | request(); |
108 | 105 | } |
109 | 106 |
|
110 | | - $: { |
111 | | - selectedSize = 0; |
112 | | - const users = new Set(); |
113 | | - for (const s in selected) { |
114 | | - const target = selected[s]; |
115 | | - users.add(target.userId); |
116 | | - selectedSize++; |
117 | | - } |
118 | | - selectedUsers = users.size; |
119 | | - } |
120 | | -
|
| 107 | + $: selectedTargets = Object.values(selected); |
| 108 | + $: selectedSize = selectedTargets.length; |
| 109 | + $: selectedUsers = new Set(selectedTargets.map((target) => target.userId)).size; |
121 | 110 | $: hasSelection = selectedSize > 0; |
122 | 111 |
|
| 112 | + $: parentStates = Object.fromEntries( |
| 113 | + Object.entries(userResultsById).map(([userId, user]) => { |
| 114 | + const selectedCount = user.targets.filter((target) => selected[target.$id]).length; |
| 115 | + const totalCount = user.targets.length; |
| 116 | +
|
| 117 | + let state: boolean | 'indeterminate'; |
| 118 | + if (selectedCount === 0) { |
| 119 | + state = false; |
| 120 | + } else if (selectedCount === totalCount) { |
| 121 | + state = true; |
| 122 | + } else { |
| 123 | + state = 'indeterminate'; |
| 124 | + } |
| 125 | +
|
| 126 | + return [userId, state]; |
| 127 | + }) |
| 128 | + ); |
| 129 | +
|
| 130 | + $: targetSelectionStates = Object.fromEntries( |
| 131 | + Object.values(userResultsById) |
| 132 | + .flatMap((user) => user.targets) |
| 133 | + .map((target) => [target.$id, !!selected[target.$id]]) |
| 134 | + ); |
| 135 | +
|
123 | 136 | $: if (show) { |
124 | 137 | selected = targetsById; |
125 | 138 | } |
|
140 | 153 | ).length} |
141 | 154 | <Accordion |
142 | 155 | selectable |
143 | | - title={user.name |
144 | | - ? user.name |
145 | | - : user.email |
146 | | - ? user.email |
147 | | - : user.phone |
148 | | - ? user.phone |
149 | | - : userId} |
| 156 | + title={user.name || user.email || user.phone || userId} |
150 | 157 | badge={user.targets.length === 0 |
151 | 158 | ? '0 targets' |
152 | 159 | : `${selectedCount}/${user.targets.length} targets`} |
153 | 160 | disabled={!user.targets.length} |
154 | | - checked={selectedCount > 0 && selectedCount === user.targets.length} |
| 161 | + checked={parentStates[userId]} |
155 | 162 | on:change={(event) => onUserSelection(event, userId)}> |
156 | 163 | {#each user.targets as target} |
157 | 164 | <Layout.Stack direction="row"> |
158 | 165 | <Selector.Checkbox |
159 | 166 | id={target.$id} |
160 | 167 | size="s" |
161 | | - checked={!!selected[target.$id]} |
| 168 | + checked={targetSelectionStates[target.$id] || false} |
162 | 169 | on:change={(event) => onTargetSelection(event, target)}> |
163 | 170 | </Selector.Checkbox> |
164 | 171 | <div class="u-inline-flex u-gap-8"> |
|
0 commit comments