@sourceloop/search-client 5.0.2 → 5.2.2
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.
- package/README.md +99 -2
- package/esm2020/lib/lib-configuration.mjs +1 -1
- package/esm2020/lib/search/promise-api-adapter.service.mjs +26 -0
- package/esm2020/lib/search/search.component.mjs +55 -27
- package/esm2020/lib/search-lib.module.mjs +6 -6
- package/esm2020/lib/types.mjs +6 -2
- package/esm2020/public-api.mjs +2 -2
- package/fesm2015/sourceloop-search-client.mjs +85 -34
- package/fesm2015/sourceloop-search-client.mjs.map +1 -1
- package/fesm2020/sourceloop-search-client.mjs +88 -34
- package/fesm2020/sourceloop-search-client.mjs.map +1 -1
- package/lib/lib-configuration.d.ts +2 -1
- package/lib/search/promise-api-adapter.service.d.ts +7 -0
- package/lib/search/search.component.d.ts +9 -5
- package/lib/types.d.ts +5 -0
- package/package.json +5 -5
package/README.md
CHANGED
|
@@ -2,13 +2,15 @@
|
|
|
2
2
|
|
|
3
3
|
An Angular module that exports a component that can enable users to search over configured models using the search microservice provided in the sourceloop microservice catalog.
|
|
4
4
|
|
|
5
|
-
##
|
|
5
|
+
## Angular Module
|
|
6
|
+
|
|
7
|
+
### Installation
|
|
6
8
|
|
|
7
9
|
```sh
|
|
8
10
|
npm i @sourceloop/search-client
|
|
9
11
|
```
|
|
10
12
|
|
|
11
|
-
|
|
13
|
+
### Usage
|
|
12
14
|
|
|
13
15
|
Create a new Application using Angular CLI and import the SearchLibModule and add it to the imports array of the module. Also create a new service that implements the ISearchService interface exported by the search library. This service will be used by the exported component to make API calls whenever needed. You will have to update the providers section of your module with { provide: SEARCH_SERVICE_TOKEN, useExisting: Your_Service_Name }
|
|
14
16
|
Your module will then look something like this
|
|
@@ -152,3 +154,98 @@ To use the default icons you will have to import the following in your styles.sc
|
|
|
152
154
|
```
|
|
153
155
|
|
|
154
156
|
You can also choose to use your own icons by providing classes for icons in the configuration.
|
|
157
|
+
|
|
158
|
+
## Web Component
|
|
159
|
+
|
|
160
|
+
This library is also available as a [Web Component](https://developer.mozilla.org/en-US/docs/Web/Web_Components) so users of frameworks like React and Vue can also integrate this search element in their application with minimal effort.
|
|
161
|
+
|
|
162
|
+
### Installation
|
|
163
|
+
|
|
164
|
+
```sh
|
|
165
|
+
npm i @sourceloop/search-client
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
In the node modules you can find two files relevant to the element - `element/search-element.js` and `element/style.css`. How you serve and include these files in your non Angular project depend on the framework that you are using. For example, for Vanilla JS and HTML you can simply import the js and styles in your HTML ->
|
|
169
|
+
|
|
170
|
+
```html
|
|
171
|
+
<script type="text/javascript" src="search-element.js"></script>
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
### Usage
|
|
175
|
+
|
|
176
|
+
The web component accepts all the same inputs and services as the regular Angular Module, but instead of passing them through bindings and DI, you pass them as properties of the element as shown below.
|
|
177
|
+
|
|
178
|
+
```html
|
|
179
|
+
<!DOCTYPE html>
|
|
180
|
+
<html>
|
|
181
|
+
<head>
|
|
182
|
+
...
|
|
183
|
+
<link rel="stylesheet" href="assets/icomoon/style.css" />
|
|
184
|
+
<link rel="stylesheet" href="styles.css" />
|
|
185
|
+
...
|
|
186
|
+
</head>
|
|
187
|
+
<body>
|
|
188
|
+
<sourceloop-search-element></sourceloop-search-element>
|
|
189
|
+
<script type="text/javascript" src="search-element.js"></script>
|
|
190
|
+
<script>
|
|
191
|
+
document.addEventListener('DOMContentLoaded', () => {
|
|
192
|
+
const element = document.querySelector('sourceloop-search-element');
|
|
193
|
+
// Code to set inputs of the component
|
|
194
|
+
element.searchProvider = {
|
|
195
|
+
searchApiRequestWithPromise: () =>
|
|
196
|
+
Promise.resolve([
|
|
197
|
+
{
|
|
198
|
+
name: 'Test',
|
|
199
|
+
description: 'Test',
|
|
200
|
+
rank: 0.4,
|
|
201
|
+
source: 'ToDo',
|
|
202
|
+
},
|
|
203
|
+
{
|
|
204
|
+
name: 'Akshat',
|
|
205
|
+
description: 'Dubey',
|
|
206
|
+
rank: 0.4,
|
|
207
|
+
source: 'User',
|
|
208
|
+
},
|
|
209
|
+
]),
|
|
210
|
+
recentSearchApiRequestWithPromise: () => Promise.resolve([]),
|
|
211
|
+
};
|
|
212
|
+
element.config = new SearchConfiguration({
|
|
213
|
+
displayPropertyName: 'name',
|
|
214
|
+
models: [
|
|
215
|
+
{
|
|
216
|
+
name: 'ToDo',
|
|
217
|
+
displayName: 'List',
|
|
218
|
+
imageUrl: 'https://picsum.photos/id/1000/50',
|
|
219
|
+
},
|
|
220
|
+
{
|
|
221
|
+
name: 'User',
|
|
222
|
+
displayName: 'Users',
|
|
223
|
+
imageUrl: 'https://picsum.photos/id/1/50',
|
|
224
|
+
},
|
|
225
|
+
],
|
|
226
|
+
order: [`name ASC`, `description DESC`],
|
|
227
|
+
hideCategorizeButton: false,
|
|
228
|
+
placeholder: 'Search Programs, Projects or Dashboards',
|
|
229
|
+
categorizeResults: true,
|
|
230
|
+
saveInRecents: true,
|
|
231
|
+
limit: 4,
|
|
232
|
+
});
|
|
233
|
+
});
|
|
234
|
+
</script>
|
|
235
|
+
</body>
|
|
236
|
+
</html>
|
|
237
|
+
```
|
|
238
|
+
|
|
239
|
+
Note that the instance of `SearchService` passed to the element is following a different interface -
|
|
240
|
+
|
|
241
|
+
```ts
|
|
242
|
+
export interface ISearchServiceWithPromises<T extends IReturnType> {
|
|
243
|
+
searchApiRequestWithPromise(
|
|
244
|
+
requestParameters: ISearchQuery,
|
|
245
|
+
saveInRecents: boolean,
|
|
246
|
+
): Promise<T[]>;
|
|
247
|
+
recentSearchApiRequestWithPromise?(): Promise<ISearchQuery[]>;
|
|
248
|
+
}
|
|
249
|
+
```
|
|
250
|
+
|
|
251
|
+
This facilitates the use of the `Web Component` without relying on [rxjs](https://rxjs.dev/). You can still use the `Observable` based service if you want by importing the rxjs library manually.
|
|
@@ -60,4 +60,4 @@ function setIconClasses(d) {
|
|
|
60
60
|
recentSearchIconClass: d.recentSearchIconClass ?? 'icomoon Search',
|
|
61
61
|
};
|
|
62
62
|
}
|
|
63
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lib-configuration.js","sourceRoot":"","sources":["../../../src/lib/lib-configuration.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) 2023 Sourcefuse Technologies\n//\n// This software is released under the MIT License.\n// https://opensource.org/licenses/MIT\nimport {IDefaultReturnType, IModel} from './types';\nexport class Configuration<T = IDefaultReturnType> {\n  /** property to be displayed in the results */\n  displayPropertyName: keyof T;\n  /** list of model configuration to be render and categorize search results */\n  models: IModel[];\n  /** max number of results (based on limitByType option) */\n  limit?: number;\n  /** apply limit on individual models, or on overall results */\n  limitByType?: boolean;\n  /** apply a particular ordering on results */\n  order?: string[];\n  /** offset for results in case limit is used */\n  offset?: number;\n  /** save the search query in recent history */\n  saveInRecents?: boolean;\n  /** a placeholder to display in the search box */\n  placeholder?: string;\n  /** a function to generate placeholder, overrides the placeholder property */\n  placeholderFunction?: (input: string, category: string) => string;\n  /** categorize results on the basis of models provided */\n  categorizeResults?: boolean;\n  /** hides the recent search list */\n  hideRecentSearch?: boolean;\n  /** hide the category selection button */\n  hideCategorizeButton?: boolean;\n  /** save value in recent search only on enter or change in category,\n   *  if false, also saved on typing */\n  saveInRecentsOnlyOnEnter?: boolean;\n  /** search only on enter key or when category is changed */\n  searchOnlyOnEnter?: boolean;\n  noResultMessage?: string;\n  searchIconClass?: string;\n  crossIconClass?: string;\n  dropDownButtonIconClass?: string;\n  recentSearchIconClass?: string;\n\n  constructor(d: Configuration<T>) {\n    checkForError(d);\n    this.displayPropertyName = d.displayPropertyName;\n    this.models = d.models;\n\n    /* IRequestParameters - will be given default values before call is made in case undefined/null,\n    otherwise there ! is used on which sonar gives code smell */\n    this.limit = d.limit;\n    this.limitByType = d.limitByType;\n    this.order = d.order;\n    this.offset = d.offset;\n    this.saveInRecents = d.saveInRecents;\n\n    const displayTexts = setDisplayText(d);\n    this.noResultMessage = displayTexts.noResultMessage;\n    this.placeholder = displayTexts.placeholder;\n    this.placeholderFunction = displayTexts.placeholderFunction;\n\n    const searchConfig = setSearchConfig(d);\n    this.categorizeResults = searchConfig.categorizeResults;\n    this.hideRecentSearch = searchConfig.hideRecentSearch;\n    this.hideCategorizeButton = searchConfig.hideCategorizeButton;\n    this.saveInRecentsOnlyOnEnter = searchConfig.saveInRecentsOnlyOnEnter;\n    this.searchOnlyOnEnter = searchConfig.searchOnlyOnEnter;\n\n    const classes = setIconClasses(d);\n    this.searchIconClass = classes.searchIconClass;\n    this.crossIconClass = classes.crossIconClass;\n    this.dropDownButtonIconClass = classes.dropDownButtonIconClass;\n    this.recentSearchIconClass = classes.recentSearchIconClass;\n  }\n}\nfunction checkForError<T>(d: Configuration<T>) {\n  if (\n    d.categorizeResults === false &&\n    (d.hideCategorizeButton === false || d.hideCategorizeButton === undefined)\n  ) {\n    throw new Error(\n      'You must provide hideCategorizeButton:true as categorizeResults is false',\n    );\n  }\n  if (d.saveInRecents === false && d.saveInRecentsOnlyOnEnter === true) {\n    throw new Error(\n      'You must provide saveInRecents:true for saveInRecentsOnlyOnEnter:true',\n    );\n  }\n}\nfunction setDisplayText<T>(d: Configuration<T>) {\n  return {\n    placeholder: d.placeholder ?? 'Search',\n    noResultMessage: d.noResultMessage ?? 'No result found',\n    placeholderFunction: d.placeholderFunction,\n  };\n}\nfunction setSearchConfig<T>(d: Configuration<T>) {\n  return {\n    categorizeResults: d.categorizeResults ?? true,\n    hideRecentSearch: d.hideRecentSearch ?? false,\n    hideCategorizeButton: d.hideCategorizeButton ?? false,\n    saveInRecentsOnlyOnEnter: d.saveInRecentsOnlyOnEnter ?? false,\n    searchOnlyOnEnter: d.searchOnlyOnEnter ?? false,\n  };\n}\nfunction setIconClasses<T>(d: Configuration<T>) {\n  return {\n    searchIconClass: d.searchIconClass ?? 'icomoon Search',\n    crossIconClass: d.crossIconClass ?? 'icomoon close',\n    dropDownButtonIconClass: d.dropDownButtonIconClass ?? 'icomoon arrow_down',\n    recentSearchIconClass: d.recentSearchIconClass ?? 'icomoon Search',\n  };\n}\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { from } from 'rxjs';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class PromiseApiAdapterService {
|
|
5
|
+
adapt(instance) {
|
|
6
|
+
// this is a workaround for the fact that the recentSearchApiRequestWithPromise
|
|
7
|
+
// method is optional in the ISearchServiceWithPromises interface
|
|
8
|
+
// and type system is not able maintain the type information of a property
|
|
9
|
+
const recentSearchMethod = instance.recentSearchApiRequestWithPromise;
|
|
10
|
+
return {
|
|
11
|
+
searchApiRequest: (requestParameters, saveInRecents) => from(instance.searchApiRequestWithPromise(requestParameters, saveInRecents)),
|
|
12
|
+
...(recentSearchMethod && {
|
|
13
|
+
recentSearchApiRequest: () => from(recentSearchMethod()),
|
|
14
|
+
}),
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
PromiseApiAdapterService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: PromiseApiAdapterService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
19
|
+
PromiseApiAdapterService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: PromiseApiAdapterService, providedIn: 'root' });
|
|
20
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: PromiseApiAdapterService, decorators: [{
|
|
21
|
+
type: Injectable,
|
|
22
|
+
args: [{
|
|
23
|
+
providedIn: 'root',
|
|
24
|
+
}]
|
|
25
|
+
}] });
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvbWlzZS1hcGktYWRhcHRlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9zZWFyY2gvcHJvbWlzZS1hcGktYWRhcHRlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFDLElBQUksRUFBQyxNQUFNLE1BQU0sQ0FBQzs7QUFVMUIsTUFBTSxPQUFPLHdCQUF3QjtJQUNuQyxLQUFLLENBQUMsUUFBdUM7UUFDM0MsK0VBQStFO1FBQy9FLGlFQUFpRTtRQUNqRSwwRUFBMEU7UUFDMUUsTUFBTSxrQkFBa0IsR0FBRyxRQUFRLENBQUMsaUNBQWlDLENBQUM7UUFDdEUsT0FBTztZQUNMLGdCQUFnQixFQUFFLENBQUMsaUJBQWlCLEVBQUUsYUFBYSxFQUFFLEVBQUUsQ0FDckQsSUFBSSxDQUNGLFFBQVEsQ0FBQywyQkFBMkIsQ0FDbEMsaUJBQWlCLEVBQ2pCLGFBQWEsQ0FDZCxDQUNGO1lBQ0gsR0FBRyxDQUFDLGtCQUFrQixJQUFJO2dCQUN4QixzQkFBc0IsRUFBRSxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLEVBQUUsQ0FBQzthQUN6RCxDQUFDO1NBQ0gsQ0FBQztJQUNKLENBQUM7O3NIQWxCVSx3QkFBd0I7MEhBQXhCLHdCQUF3QixjQUZ2QixNQUFNOzRGQUVQLHdCQUF3QjtrQkFIcEMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0luamVjdGFibGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtmcm9tfSBmcm9tICdyeGpzJztcbmltcG9ydCB7XG4gIElSZXR1cm5UeXBlLFxuICBJU2VhcmNoU2VydmljZSxcbiAgSVNlYXJjaFNlcnZpY2VXaXRoUHJvbWlzZXMsXG59IGZyb20gJy4uL3R5cGVzJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuZXhwb3J0IGNsYXNzIFByb21pc2VBcGlBZGFwdGVyU2VydmljZTxUIGV4dGVuZHMgSVJldHVyblR5cGU+IHtcbiAgYWRhcHQoaW5zdGFuY2U6IElTZWFyY2hTZXJ2aWNlV2l0aFByb21pc2VzPFQ+KTogSVNlYXJjaFNlcnZpY2U8VD4ge1xuICAgIC8vIHRoaXMgaXMgYSB3b3JrYXJvdW5kIGZvciB0aGUgZmFjdCB0aGF0IHRoZSByZWNlbnRTZWFyY2hBcGlSZXF1ZXN0V2l0aFByb21pc2VcbiAgICAvLyBtZXRob2QgaXMgb3B0aW9uYWwgaW4gdGhlIElTZWFyY2hTZXJ2aWNlV2l0aFByb21pc2VzIGludGVyZmFjZVxuICAgIC8vIGFuZCB0eXBlIHN5c3RlbSBpcyBub3QgYWJsZSBtYWludGFpbiB0aGUgdHlwZSBpbmZvcm1hdGlvbiBvZiBhIHByb3BlcnR5XG4gICAgY29uc3QgcmVjZW50U2VhcmNoTWV0aG9kID0gaW5zdGFuY2UucmVjZW50U2VhcmNoQXBpUmVxdWVzdFdpdGhQcm9taXNlO1xuICAgIHJldHVybiB7XG4gICAgICBzZWFyY2hBcGlSZXF1ZXN0OiAocmVxdWVzdFBhcmFtZXRlcnMsIHNhdmVJblJlY2VudHMpID0+XG4gICAgICAgIGZyb20oXG4gICAgICAgICAgaW5zdGFuY2Uuc2VhcmNoQXBpUmVxdWVzdFdpdGhQcm9taXNlKFxuICAgICAgICAgICAgcmVxdWVzdFBhcmFtZXRlcnMsXG4gICAgICAgICAgICBzYXZlSW5SZWNlbnRzLFxuICAgICAgICAgICksXG4gICAgICAgICksXG4gICAgICAuLi4ocmVjZW50U2VhcmNoTWV0aG9kICYmIHtcbiAgICAgICAgcmVjZW50U2VhcmNoQXBpUmVxdWVzdDogKCkgPT4gZnJvbShyZWNlbnRTZWFyY2hNZXRob2QoKSksXG4gICAgICB9KSxcbiAgICB9O1xuICB9XG59XG4iXX0=
|