@abcagency/hc-ui-components 1.3.17 → 1.3.18

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,195 +1,239 @@
1
- import { getDistinctItemsByProximity } from '~/util/mapUtil';
2
-
3
- export const getFilterOptions = (listings, filteredListings, field, excludeZeroCount = null) => {
4
- const options = new Set();
5
- listings.forEach(listing => {
6
- if (listing.fields[field]) {
7
- options.add(listing.fields[field]);
8
- }
9
- });
10
-
11
- const optionCounts = {};
12
- options.forEach(option => {
13
- optionCounts[option] = 0;
14
- });
15
-
16
- filteredListings.forEach(listing => {
17
- const value = listing.fields[field];
18
- if (value && optionCounts.hasOwnProperty(value)) {
19
- optionCounts[value] += 1;
20
- }
21
- });
22
-
23
- return Array.from(options)
24
- .sort()
25
- .map(option => ({
26
- name: option,
27
- count: optionCounts[option] || 0
28
- }))
29
- .filter(option => !(excludeZeroCount === true && option.count === 0));
30
- };
31
-
32
- export const getSpecialFeatureOptions = (listings, filteredListings, siteConfig, favorites) => {
33
- const specialFeatures = siteConfig.specialFeatures;
34
- const featureCounts = Object.keys(specialFeatures).sort().reduce((acc, key) => {
35
- acc[specialFeatures[key]] = 0;
36
- return acc;
37
- }, {});
38
-
39
- filteredListings.forEach(listing => {
40
- Object.entries(specialFeatures).forEach(([featureKey, featureName]) => {
41
- if (listing.fields[featureKey] === 1) {
42
- featureCounts[featureName] += 1;
43
- }
44
- });
45
- });
46
-
47
- const specialFeatureOptions = Object.entries(featureCounts).map(([name, count]) => ({
48
- name,
49
- count
50
- }));
51
-
52
- for (let option of specialFeatureOptions) {
53
- if (option.name === 'Favorite') {
54
- option.count = filteredListings.filter(x => favorites.includes(x.id)).length;
55
- }
56
- }
57
-
58
- return specialFeatureOptions;
59
- };
60
-
61
- const getPointsOfInterestOptions = pointsOfInterestNames => {
62
- return Object.entries(pointsOfInterestNames).sort().map(([key, name]) => ({
63
- key,
64
- name
65
- }));
66
- };
67
-
68
- export const generateFilterOptions = (
69
- filteredListings,
70
- allListings,
71
- siteConfig,
72
- filterOptions,
73
- parentField = null,
74
- favorites
75
- ) => {
76
- if (allListings.length > 0) {
77
- const dynamicFilters = siteConfig.fieldFiltersShown.map(fieldName => {
78
- if (fieldName === parentField && filterOptions?.filters) {
79
- return filterOptions.filters.find(filter => filter.id === fieldName);
80
- }
81
- if (fieldName == "specialFeatures") {
82
- return {
83
- id: fieldName,
84
- title: siteConfig.fieldNames[fieldName],
85
- items: getSpecialFeatureOptions(allListings, filteredListings, siteConfig, favorites).sort()
86
- };
87
- }
88
- return {
89
- id: fieldName,
90
- title: siteConfig.fieldNames[fieldName],
91
- items: getFilterOptions(allListings, filteredListings, fieldName)
92
- };
93
- });
94
-
95
- const locations =
96
- siteConfig.locationFiltersShown.map((fieldName, index) => {
97
- if (index === 0 && filterOptions?.locations) {
98
- return filterOptions.locations.find(filter => filter.id === fieldName);
99
- }
100
- return {
101
- id: fieldName,
102
- title: siteConfig.fieldNames[fieldName],
103
- items: getFilterOptions(allListings, filteredListings, fieldName, true)
104
- };
105
- });
106
-
107
- const pointsOfInterest = {
108
- id: "pointsOfInterest",
109
- title: siteConfig.pointsOfInterestConfig.title,
110
- items: getPointsOfInterestOptions(
111
- siteConfig.pointsOfInterestConfig.pointsOfInterestNames
112
- )
113
- };
114
-
115
- return {
116
- filters: dynamicFilters,
117
- locations: locations,
118
- pointsOfInterest: pointsOfInterest
119
- };
120
- }
121
-
122
- return null;
123
- };
124
-
125
- export const applyFilters = (
126
- allListings,
127
- selectedFilters,
128
- query,
129
- listingEntities,
130
- favorites,
131
- siteConfig
132
- ) => {
133
- let results = allListings;
134
- let invertedSpecialFeaturesMap;
135
- if (siteConfig.specialFeatures) {
136
- invertedSpecialFeaturesMap = Object.entries(siteConfig.specialFeatures).reduce((acc, [key, value]) => {
137
- acc[value] = key;
138
- return acc;
139
- }, {});
140
-
141
- }
142
- const hasFavorite = !!selectedFilters.specialFeatures && !!selectedFilters.specialFeatures.Favorite;
143
-
144
- if (hasFavorite && selectedFilters.specialFeatures.Favorite == true) {
145
- results = results.filter(x => favorites.includes(x.id));
146
- }
147
- var favorite;
148
- if (hasFavorite) {
149
- favorite = selectedFilters.specialFeatures.Favorite;
150
- delete selectedFilters.specialFeatures.Favorite;
151
- }
152
- for (const [field, filterItems] of Object.entries(selectedFilters)) {
153
- const formattedField = field;
154
- if (field === "pointsOfInterest") continue;
155
- if (field === "specialFeatures" && invertedSpecialFeaturesMap && Object.keys(filterItems).length > 0) {
156
- results = results.filter(listing => {
157
- return Object.entries(filterItems).some(([filterName, filterValue]) => {
158
- const listingFieldName = invertedSpecialFeaturesMap[filterName];
159
- return filterValue && listing.fields[listingFieldName] === 1;
160
- });
161
- });
162
- } else if (Object.keys(filterItems).length > 0) {
163
- results = results.filter(listing =>
164
- filterItems.hasOwnProperty(listing.fields[formattedField])
165
- );
166
- }
167
- }
168
- if (query) {
169
- results = results.filter(listing =>
170
- Object.values(listing.fields)?.some(value =>
171
- value?.toString().toLowerCase().includes(query.toLowerCase())
172
- )
173
- );
174
- }
175
- const distinctItems = getDistinctItemsByProximity(results, listingEntities);
176
- if (hasFavorite) {
177
- selectedFilters.specialFeatures.Favorite = favorite;
178
- }
179
- return { filteredListings: results, mapItems: distinctItems };
180
- };
181
-
182
- export const filterListingsByLocation = (
183
- allListings,
184
- selectedLocation,
185
- listingEntities
186
- ) => {
187
- let results = allListings;
188
- if (selectedLocation !== null) {
189
- results = results.filter(item =>
190
- selectedLocation.items.hasOwnProperty(item.id)
191
- );
192
- }
193
- const mapItems = getDistinctItemsByProximity(results, listingEntities);
194
- return { filteredListings: results, mapItems: mapItems };
195
- };
1
+ import { getDistinctItemsByProximity } from '~/util/mapUtil';
2
+
3
+ import Fuse from 'fuse.js';
4
+
5
+ export const getFilterOptions = (listings, filteredListings, field, excludeZeroCount = null) => {
6
+ const options = new Set();
7
+ listings.forEach(listing => {
8
+ if (listing.fields[field]) {
9
+ options.add(listing.fields[field]);
10
+ }
11
+ });
12
+
13
+ const optionCounts = {};
14
+ options.forEach(option => {
15
+ optionCounts[option] = 0;
16
+ });
17
+
18
+ filteredListings.forEach(listing => {
19
+ const value = listing.fields[field];
20
+ if (value && optionCounts.hasOwnProperty(value)) {
21
+ optionCounts[value] += 1;
22
+ }
23
+ });
24
+
25
+ return Array.from(options)
26
+ .sort()
27
+ .map(option => ({
28
+ name: option,
29
+ count: optionCounts[option] || 0
30
+ }))
31
+ .filter(option => !(excludeZeroCount === true && option.count === 0));
32
+ };
33
+
34
+ export const getSpecialFeatureOptions = (listings, filteredListings, siteConfig, favorites) => {
35
+ const specialFeatures = siteConfig.specialFeatures;
36
+ const featureCounts = Object.keys(specialFeatures).sort().reduce((acc, key) => {
37
+ acc[specialFeatures[key]] = 0;
38
+ return acc;
39
+ }, {});
40
+
41
+ filteredListings.forEach(listing => {
42
+ Object.entries(specialFeatures).forEach(([featureKey, featureName]) => {
43
+ if (listing.fields[featureKey] === 1) {
44
+ featureCounts[featureName] += 1;
45
+ }
46
+ });
47
+ });
48
+
49
+ const specialFeatureOptions = Object.entries(featureCounts).map(([name, count]) => ({
50
+ name,
51
+ count
52
+ }));
53
+
54
+ for (let option of specialFeatureOptions) {
55
+ if (option.name === 'Favorite') {
56
+ option.count = filteredListings.filter(x => favorites.includes(x.id)).length;
57
+ }
58
+ }
59
+
60
+ return specialFeatureOptions;
61
+ };
62
+
63
+ const getPointsOfInterestOptions = pointsOfInterestNames => {
64
+ return Object.entries(pointsOfInterestNames).sort().map(([key, name]) => ({
65
+ key,
66
+ name
67
+ }));
68
+ };
69
+
70
+ export const generateFilterOptions = (
71
+ filteredListings,
72
+ allListings,
73
+ siteConfig,
74
+ filterOptions,
75
+ parentField = null,
76
+ favorites
77
+ ) => {
78
+ if (allListings.length > 0) {
79
+ const dynamicFilters = siteConfig.fieldFiltersShown.map(fieldName => {
80
+ if (fieldName === parentField && filterOptions?.filters) {
81
+ return filterOptions.filters.find(filter => filter.id === fieldName);
82
+ }
83
+ if (fieldName == "specialFeatures") {
84
+ return {
85
+ id: fieldName,
86
+ title: siteConfig.fieldNames[fieldName],
87
+ items: getSpecialFeatureOptions(allListings, filteredListings, siteConfig, favorites).sort()
88
+ };
89
+ }
90
+ return {
91
+ id: fieldName,
92
+ title: siteConfig.fieldNames[fieldName],
93
+ items: getFilterOptions(allListings, filteredListings, fieldName)
94
+ };
95
+ });
96
+
97
+ const locations =
98
+ siteConfig.locationFiltersShown.map((fieldName, index) => {
99
+ if (index === 0 && filterOptions?.locations) {
100
+ return filterOptions.locations.find(filter => filter.id === fieldName);
101
+ }
102
+ return {
103
+ id: fieldName,
104
+ title: siteConfig.fieldNames[fieldName],
105
+ items: getFilterOptions(allListings, filteredListings, fieldName, true)
106
+ };
107
+ });
108
+
109
+ const pointsOfInterest = {
110
+ id: "pointsOfInterest",
111
+ title: siteConfig.pointsOfInterestConfig.title,
112
+ items: getPointsOfInterestOptions(
113
+ siteConfig.pointsOfInterestConfig.pointsOfInterestNames
114
+ )
115
+ };
116
+
117
+ return {
118
+ filters: dynamicFilters,
119
+ locations: locations,
120
+ pointsOfInterest: pointsOfInterest
121
+ };
122
+ }
123
+
124
+ return null;
125
+ };
126
+
127
+ export const applyFilters = (
128
+ allListings,
129
+ selectedFilters,
130
+ query,
131
+ listingEntities,
132
+ favorites,
133
+ siteConfig
134
+ ) => {
135
+ let results = allListings;
136
+ let invertedSpecialFeaturesMap;
137
+ if (siteConfig.specialFeatures) {
138
+ invertedSpecialFeaturesMap = Object.entries(siteConfig.specialFeatures).reduce((acc, [key, value]) => {
139
+ acc[value] = key;
140
+ return acc;
141
+ }, {});
142
+
143
+ }
144
+ const hasFavorite = !!selectedFilters.specialFeatures && !!selectedFilters.specialFeatures.Favorite;
145
+
146
+ if (hasFavorite && selectedFilters.specialFeatures.Favorite == true) {
147
+ results = results.filter(x => favorites.includes(x.id));
148
+ }
149
+ var favorite;
150
+ if (hasFavorite) {
151
+ favorite = selectedFilters.specialFeatures.Favorite;
152
+ delete selectedFilters.specialFeatures.Favorite;
153
+ }
154
+ for (const [field, filterItems] of Object.entries(selectedFilters)) {
155
+ const formattedField = field;
156
+ if (field === "pointsOfInterest") continue;
157
+ if (field === "specialFeatures" && invertedSpecialFeaturesMap && Object.keys(filterItems).length > 0) {
158
+ results = results.filter(listing => {
159
+ return Object.entries(filterItems).some(([filterName, filterValue]) => {
160
+ const listingFieldName = invertedSpecialFeaturesMap[filterName];
161
+ return filterValue && listing.fields[listingFieldName] === 1;
162
+ });
163
+ });
164
+ } else if (Object.keys(filterItems).length > 0) {
165
+ results = results.filter(listing =>
166
+ filterItems.hasOwnProperty(listing.fields[formattedField])
167
+ );
168
+ }
169
+ }
170
+ if (query) {
171
+ results = searchResults(results, query);
172
+ }
173
+ const distinctItems = getDistinctItemsByProximity(results, listingEntities);
174
+ if (hasFavorite) {
175
+ selectedFilters.specialFeatures.Favorite = favorite;
176
+ }
177
+ return { filteredListings: results, mapItems: distinctItems };
178
+ };
179
+
180
+ function searchResults(results, query) {
181
+ const fields = [
182
+ 'id',
183
+ 'fields.posted',
184
+ 'fields.subtitle',
185
+ 'fields.education',
186
+ 'fields.position',
187
+ 'fields.category',
188
+ 'fields.categoryclass',
189
+ 'fields.shift',
190
+ 'fields.citystate',
191
+ 'fields.city',
192
+ 'fields.state',
193
+ 'fields.schedule',
194
+ 'fields.customflag1',
195
+ 'fields.bonus',
196
+ 'fields.remote',
197
+ 'fields.useclientjoburl',
198
+ 'fields.datecreated',
199
+ 'fields.datelastedited'
200
+ ];
201
+
202
+ const options = {
203
+ includeScore: true,
204
+ threshold: 0.3,
205
+ keys: fields
206
+ };
207
+
208
+ const fuse = new Fuse(results, options);
209
+ const lowerCaseQuery = query.toLowerCase();
210
+ const queryTerms = lowerCaseQuery.split(' ');
211
+
212
+ const exactIdMatch = results.find(result => result.id.toString() === query);
213
+ if (exactIdMatch) {
214
+ return [exactIdMatch];
215
+ }
216
+
217
+ const fuseQuery = queryTerms.map(term => ({
218
+ $or: fields.map(field => ({ [field]: term }))
219
+ }));
220
+
221
+ const fuseResults = fuse.search({ $and: fuseQuery });
222
+
223
+ return fuseResults.map(result => result.item);
224
+ }
225
+
226
+ export const filterListingsByLocation = (
227
+ allListings,
228
+ selectedLocation,
229
+ listingEntities
230
+ ) => {
231
+ let results = allListings;
232
+ if (selectedLocation !== null) {
233
+ results = results.filter(item =>
234
+ selectedLocation.items.hasOwnProperty(item.id)
235
+ );
236
+ }
237
+ const mapItems = getDistinctItemsByProximity(results, listingEntities);
238
+ return { filteredListings: results, mapItems: mapItems };
239
+ };