namirasoft-account-react 1.4.418 → 1.4.421

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.
Files changed (74) hide show
  1. package/dist/IEntityInfo.d.ts +3 -1
  2. package/dist/NSACacheService.d.ts +3 -0
  3. package/dist/NSACacheService.js +13 -0
  4. package/dist/NSACacheService.js.map +1 -1
  5. package/dist/NSAFilterOperators.d.ts +28 -0
  6. package/dist/NSAFilterOperators.js +50 -0
  7. package/dist/NSAFilterOperators.js.map +1 -0
  8. package/dist/components/NSAQuickFilter/NSAFilterBoxBase.d.ts +14 -0
  9. package/dist/components/NSAQuickFilter/NSAFilterBoxBase.js +40 -0
  10. package/dist/components/NSAQuickFilter/NSAFilterBoxBase.js.map +1 -0
  11. package/dist/components/NSAQuickFilter/NSAFilterBoxBoolean.d.ts +12 -0
  12. package/dist/components/NSAQuickFilter/NSAFilterBoxBoolean.js +30 -0
  13. package/dist/components/NSAQuickFilter/NSAFilterBoxBoolean.js.map +1 -0
  14. package/dist/components/NSAQuickFilter/NSAFilterBoxDate.d.ts +12 -0
  15. package/dist/components/NSAQuickFilter/NSAFilterBoxDate.js +58 -0
  16. package/dist/components/NSAQuickFilter/NSAFilterBoxDate.js.map +1 -0
  17. package/dist/components/NSAQuickFilter/NSAFilterBoxDateTime.d.ts +12 -0
  18. package/dist/components/NSAQuickFilter/NSAFilterBoxDateTime.js +58 -0
  19. package/dist/components/NSAQuickFilter/NSAFilterBoxDateTime.js.map +1 -0
  20. package/dist/components/NSAQuickFilter/NSAFilterBoxEnum.d.ts +22 -0
  21. package/dist/components/NSAQuickFilter/NSAFilterBoxEnum.js +52 -0
  22. package/dist/components/NSAQuickFilter/NSAFilterBoxEnum.js.map +1 -0
  23. package/dist/components/NSAQuickFilter/NSAFilterBoxNumber.d.ts +22 -0
  24. package/dist/components/NSAQuickFilter/NSAFilterBoxNumber.js +67 -0
  25. package/dist/components/NSAQuickFilter/NSAFilterBoxNumber.js.map +1 -0
  26. package/dist/components/NSAQuickFilter/NSAFilterBoxString.d.ts +13 -0
  27. package/dist/components/NSAQuickFilter/NSAFilterBoxString.js +47 -0
  28. package/dist/components/NSAQuickFilter/NSAFilterBoxString.js.map +1 -0
  29. package/dist/components/NSAQuickFilter/NSAFilterBoxTime.d.ts +12 -0
  30. package/dist/components/NSAQuickFilter/NSAFilterBoxTime.js +58 -0
  31. package/dist/components/NSAQuickFilter/NSAFilterBoxTime.js.map +1 -0
  32. package/dist/components/NSAQuickFilter/NSAQuickFilterBar.d.ts +28 -0
  33. package/dist/components/NSAQuickFilter/NSAQuickFilterBar.js +74 -0
  34. package/dist/components/NSAQuickFilter/NSAQuickFilterBar.js.map +1 -0
  35. package/dist/components/NSAQuickFilter/NSAQuickFilterDialog.d.ts +43 -0
  36. package/dist/components/NSAQuickFilter/NSAQuickFilterDialog.js +114 -0
  37. package/dist/components/NSAQuickFilter/NSAQuickFilterDialog.js.map +1 -0
  38. package/dist/components/NSAReorderDialog.d.ts +30 -0
  39. package/dist/components/NSAReorderDialog.js +69 -0
  40. package/dist/components/NSAReorderDialog.js.map +1 -0
  41. package/dist/components/NSAReorderDialog.module.css +50 -0
  42. package/dist/components/NSASortDialog.d.ts +21 -0
  43. package/dist/components/NSASortDialog.js +61 -0
  44. package/dist/components/NSASortDialog.js.map +1 -0
  45. package/dist/components/NSATable.d.ts +95 -0
  46. package/dist/components/NSATable.js +335 -0
  47. package/dist/components/NSATable.js.map +1 -0
  48. package/dist/layouts/NSASectionList.d.ts +7 -37
  49. package/dist/layouts/NSASectionList.js +17 -247
  50. package/dist/layouts/NSASectionList.js.map +1 -1
  51. package/dist/layouts/NSASectionList.module.css +15 -18
  52. package/package.json +2 -2
  53. package/src/IEntityInfo.ts +3 -1
  54. package/src/NSACacheService.ts +20 -0
  55. package/src/NSAFilterOperators.ts +93 -0
  56. package/src/components/NSAQuickFilter/NSAFilterBoxBase.tsx +56 -0
  57. package/src/components/NSAQuickFilter/NSAFilterBoxBoolean.tsx +55 -0
  58. package/src/components/NSAQuickFilter/NSAFilterBoxDate.tsx +105 -0
  59. package/src/components/NSAQuickFilter/NSAFilterBoxDateTime.tsx +106 -0
  60. package/src/components/NSAQuickFilter/NSAFilterBoxEnum.module.css +45 -0
  61. package/src/components/NSAQuickFilter/NSAFilterBoxEnum.tsx +80 -0
  62. package/src/components/NSAQuickFilter/NSAFilterBoxNumber.tsx +137 -0
  63. package/src/components/NSAQuickFilter/NSAFilterBoxString.tsx +82 -0
  64. package/src/components/NSAQuickFilter/NSAFilterBoxTime.tsx +102 -0
  65. package/src/components/NSAQuickFilter/NSAQuickFilterBar.module.css +20 -0
  66. package/src/components/NSAQuickFilter/NSAQuickFilterBar.tsx +115 -0
  67. package/src/components/NSAQuickFilter/NSAQuickFilterDialog.module.css +75 -0
  68. package/src/components/NSAQuickFilter/NSAQuickFilterDialog.tsx +261 -0
  69. package/src/components/NSAReorderDialog.module.css +50 -0
  70. package/src/components/NSAReorderDialog.tsx +150 -0
  71. package/src/components/NSASortDialog.tsx +118 -0
  72. package/src/components/NSATable.tsx +490 -0
  73. package/src/layouts/NSASectionList.module.css +15 -18
  74. package/src/layouts/NSASectionList.tsx +43 -395
@@ -0,0 +1,55 @@
1
+ import { FilterItemOperator } from "namirasoft-core";
2
+ import { NSBoxBoolean } from "namirasoft-site-react";
3
+ import { createRef } from "react";
4
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
5
+
6
+ interface NSAFilterBoxBooleanState
7
+ {
8
+ value: boolean | null;
9
+ }
10
+
11
+ export class NSAFilterBoxBoolean extends NSAFilterBoxBase<NSAFilterBoxBaseProps, NSAFilterBoxBooleanState>
12
+ {
13
+ private containerRef = createRef<HTMLDivElement>();
14
+
15
+ constructor(props: NSAFilterBoxBaseProps)
16
+ {
17
+ super(props);
18
+ this.state = { value: this.isValueless() ? false : null };
19
+ this.handleChange = this.handleChange.bind(this);
20
+ }
21
+
22
+ private isValueless(): boolean
23
+ {
24
+ const name = this.props.config.operatorName;
25
+ return !!name && FilterItemOperator.getByName(name).count === 0;
26
+ }
27
+
28
+
29
+ private handleChange(value: boolean | null)
30
+ {
31
+ this.setState({ value });
32
+ if (this.isValueless())
33
+ this.props.onChanged(value === true ? this.buildFilterItem() : null);
34
+ else if (value !== null)
35
+ this.props.onChanged(this.buildFilterItem(String(Number(value))));
36
+ else
37
+ this.props.onChanged(null);
38
+ }
39
+
40
+ override render(): JSX.Element
41
+ {
42
+ return (
43
+ <div ref={this.containerRef} style={{ width: "100%" }}>
44
+ <NSBoxBoolean
45
+ title={this.getLabel()}
46
+ placeholder={this.getLabel()}
47
+ required={false}
48
+ hideHeader={true}
49
+ triState={!this.isValueless()}
50
+ onChanged={e => this.handleChange(e.getValue())}
51
+ />
52
+ </div>
53
+ );
54
+ }
55
+ }
@@ -0,0 +1,105 @@
1
+ import { FilterItem, FilterItemOperator, SortItem, TimeOperation } from "namirasoft-core";
2
+ import { NSBoxDate, NSBoxDateRange, NSBoxEntity } from "namirasoft-site-react";
3
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
4
+ import { NSAFilterBoxBoolean } from "./NSAFilterBoxBoolean";
5
+
6
+ export interface NSAFilterBoxDateProps extends NSAFilterBoxBaseProps
7
+ {
8
+ getValues: (search: string) => Promise<{ value: string; title: string }[]>;
9
+ }
10
+
11
+ type ValueItem = { value: string; title: string };
12
+
13
+ const ENTITY_OPERATORS = ["equals"];
14
+
15
+ export class NSAFilterBoxDate extends NSAFilterBoxBase<NSAFilterBoxDateProps>
16
+ {
17
+ private betweenActive = false;
18
+
19
+ private list = async (
20
+ filters: FilterItem[] | null,
21
+ _page: number | null,
22
+ _size: number | null,
23
+ _sorts: SortItem[]
24
+ ): Promise<{ count: number; rows: ValueItem[] }> =>
25
+ {
26
+ const searchFilter = filters?.find(f => f.values.length > 0);
27
+ const search = searchFilter?.values[0] ?? "";
28
+ const rows = await this.props.getValues(search);
29
+ return { count: rows.length, rows };
30
+ };
31
+
32
+ override render(): JSX.Element | null
33
+ {
34
+ const { config, onChanged } = this.props;
35
+ const op = config.operatorName
36
+ ? FilterItemOperator.getByName(config.operatorName)
37
+ : null;
38
+
39
+ if (op && op.count === 0)
40
+ return <NSAFilterBoxBoolean config={config} onChanged={onChanged} />;
41
+
42
+ if (config.operatorName === "between")
43
+ return (
44
+ <NSBoxDateRange
45
+ title={this.getLabel()}
46
+ placeholder={this.getLabel()}
47
+ required={false}
48
+ hideHeader={true}
49
+ preset={true}
50
+ onChanged={e =>
51
+ {
52
+ const val = e.getValue();
53
+ if (val)
54
+ {
55
+ this.betweenActive = true;
56
+ onChanged(this.buildBetweenFilterItems(
57
+ TimeOperation.toDBFormat(val.from),
58
+ TimeOperation.toDBFormat(val.to)
59
+ ));
60
+ }
61
+ else if (this.betweenActive)
62
+ {
63
+ this.betweenActive = false;
64
+ onChanged(null);
65
+ }
66
+ }}
67
+ />
68
+ );
69
+
70
+ if (config.operatorName && ENTITY_OPERATORS.includes(config.operatorName))
71
+ return (
72
+ <NSBoxEntity<ValueItem>
73
+ title={this.getLabel()}
74
+ placeholder={this.getLabel()}
75
+ required={false}
76
+ hideHeader={true}
77
+ multiple={false}
78
+ list={this.list}
79
+ table_name={config.column.table.name}
80
+ getValue={item => item.value}
81
+ getTitle={item => item.title}
82
+ getSort={() => null}
83
+ onChanged={e =>
84
+ {
85
+ const val = e.getValueOne();
86
+ onChanged(val ? this.buildFilterItem(val) : null);
87
+ }}
88
+ />
89
+ );
90
+
91
+ return (
92
+ <NSBoxDate
93
+ title={this.getLabel()}
94
+ placeholder={this.getLabel()}
95
+ required={false}
96
+ hideHeader={true}
97
+ onChanged={e =>
98
+ {
99
+ const val = e.getValue();
100
+ onChanged(val ? this.buildFilterItem(TimeOperation.toDBFormat(val)) : null);
101
+ }}
102
+ />
103
+ );
104
+ }
105
+ }
@@ -0,0 +1,106 @@
1
+ import { FilterItem, FilterItemOperator, SortItem, TimeOperation } from "namirasoft-core";
2
+ import { NSBoxDateTime, NSBoxDateTimeRange, NSBoxEntity } from "namirasoft-site-react";
3
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
4
+ import { NSAFilterBoxBoolean } from "./NSAFilterBoxBoolean";
5
+
6
+ export interface NSAFilterBoxDateTimeProps extends NSAFilterBoxBaseProps
7
+ {
8
+ getValues: (search: string) => Promise<{ value: string; title: string }[]>;
9
+ }
10
+
11
+ type ValueItem = { value: string; title: string };
12
+
13
+ const ENTITY_OPERATORS = ["equals"];
14
+
15
+ export class NSAFilterBoxDateTime extends NSAFilterBoxBase<NSAFilterBoxDateTimeProps>
16
+ {
17
+ private betweenActive = false;
18
+
19
+ private list = async (
20
+ filters: FilterItem[] | null,
21
+ _page: number | null,
22
+ _size: number | null,
23
+ _sorts: SortItem[]
24
+ ): Promise<{ count: number; rows: ValueItem[] }> =>
25
+ {
26
+ const searchFilter = filters?.find(f => f.values.length > 0);
27
+ const search = searchFilter?.values[0] ?? "";
28
+ const rows = await this.props.getValues(search);
29
+ return { count: rows.length, rows };
30
+ };
31
+
32
+ override render(): JSX.Element | null
33
+ {
34
+ const { config, onChanged } = this.props;
35
+ const op = config.operatorName
36
+ ? FilterItemOperator.getByName(config.operatorName)
37
+ : null;
38
+
39
+ if (op && op.count === 0)
40
+ return <NSAFilterBoxBoolean config={config} onChanged={onChanged} />;
41
+
42
+ if (config.operatorName === "between")
43
+ return (
44
+ <NSBoxDateTimeRange
45
+ title={this.getLabel()}
46
+ placeholder={this.getLabel()}
47
+ required={false}
48
+ hideHeader={true}
49
+ preset={true}
50
+ onChanged={e =>
51
+ {
52
+ const val = e.getValue();
53
+ if (val)
54
+ {
55
+ this.betweenActive = true;
56
+ onChanged(this.buildBetweenFilterItems(
57
+ TimeOperation.toDBFormat(val.from),
58
+ TimeOperation.toDBFormat(val.to)
59
+ ));
60
+ }
61
+ else if (this.betweenActive)
62
+ {
63
+ this.betweenActive = false;
64
+ onChanged(null);
65
+ }
66
+ }}
67
+ />
68
+ );
69
+
70
+ if (config.operatorName && ENTITY_OPERATORS.includes(config.operatorName))
71
+ return (
72
+ <NSBoxEntity<ValueItem>
73
+ title={this.getLabel()}
74
+ placeholder={this.getLabel()}
75
+ required={false}
76
+ hideHeader={true}
77
+ multiple={false}
78
+ list={this.list}
79
+ table_name={config.column.table.name}
80
+ getValue={item => item.value}
81
+ getTitle={item => item.title}
82
+ getSort={() => null}
83
+ onChanged={e =>
84
+ {
85
+ const val = e.getValueOne();
86
+ onChanged(val ? this.buildFilterItem(val) : null);
87
+ }}
88
+ />
89
+ );
90
+
91
+ return (
92
+ <NSBoxDateTime
93
+ title={this.getLabel()}
94
+ placeholder={this.getLabel()}
95
+ required={false}
96
+ hideHeader={true}
97
+ defaultValue={undefined}
98
+ onChanged={e =>
99
+ {
100
+ const val = e.getValue();
101
+ onChanged(val ? this.buildFilterItem(TimeOperation.toDBFormat(val)) : null);
102
+ }}
103
+ />
104
+ );
105
+ }
106
+ }
@@ -0,0 +1,45 @@
1
+ .title {
2
+ font-size: 13px;
3
+ font-weight: 400;
4
+ color: #6b6b6b;
5
+ display: flex;
6
+ align-items: center;
7
+ white-space: nowrap;
8
+ flex-shrink: 0;
9
+ margin-right: 2px;
10
+ }
11
+
12
+ .container {
13
+ display: flex;
14
+ flex-wrap: nowrap;
15
+ align-items: center;
16
+ gap: 6px;
17
+ height: 48px;
18
+ width: 100%;
19
+ box-sizing: border-box;
20
+ padding: 0 10px;
21
+ border: 1px solid #a0a0a0;
22
+ border-radius: 8px;
23
+ background-color: #fff;
24
+ overflow-x: auto;
25
+ overflow-y: hidden;
26
+ }
27
+
28
+ .option {
29
+ padding: 4px 12px;
30
+ border-radius: 16px;
31
+ border: 1px solid #a0a0a0;
32
+ background-color: #fff;
33
+ color: #333;
34
+ cursor: pointer;
35
+ font-size: 13px;
36
+ font-family: inherit;
37
+ flex-shrink: 0;
38
+ white-space: nowrap;
39
+ }
40
+
41
+ .option_active {
42
+ border-color: #1976d2;
43
+ background-color: #1976d2;
44
+ color: #fff;
45
+ }
@@ -0,0 +1,80 @@
1
+ import { FilterItemOperator } from "namirasoft-core";
2
+ import { NSLoading } from "namirasoft-site-react";
3
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
4
+ import { NSAFilterBoxBoolean } from "./NSAFilterBoxBoolean";
5
+ import Styles from "./NSAFilterBoxEnum.module.css";
6
+
7
+ export interface NSAFilterBoxEnumProps extends NSAFilterBoxBaseProps
8
+ {
9
+ getValues: (search: string) => Promise<{ value: string; title: string }[]>;
10
+ }
11
+
12
+ type ValueItem = { value: string; title: string };
13
+
14
+ type State = {
15
+ loading: boolean;
16
+ values: ValueItem[];
17
+ selectedValues: string[];
18
+ };
19
+
20
+ export class NSAFilterBoxEnum extends NSAFilterBoxBase<NSAFilterBoxEnumProps, State>
21
+ {
22
+ constructor(props: NSAFilterBoxEnumProps)
23
+ {
24
+ super(props);
25
+ this.state = { loading: false, values: [], selectedValues: [] };
26
+ }
27
+
28
+ override async componentDidMount()
29
+ {
30
+ this.setState({ loading: true });
31
+ const values = await this.props.getValues("");
32
+ this.setState({ loading: false, values });
33
+ }
34
+
35
+ override render(): JSX.Element | null
36
+ {
37
+ const { config, onChanged } = this.props;
38
+ const op = config.operatorName
39
+ ? FilterItemOperator.getByName(config.operatorName)
40
+ : null;
41
+
42
+ if (op && op.count === 0)
43
+ return <NSAFilterBoxBoolean config={config} onChanged={onChanged} />;
44
+
45
+ if (config.operatorName === "equals")
46
+ {
47
+ return (
48
+ <div className={Styles.container}>
49
+ <span className={Styles.title}>{this.getLabel()}</span>
50
+ {
51
+ this.state.loading
52
+ ? <NSLoading classList={["p-2"]} small={true} hideTitle={true} hideDescription={true} />
53
+ : this.state.values.map(item =>
54
+ {
55
+ const isActive = this.state.selectedValues.includes(item.value);
56
+ return (
57
+ <button
58
+ key={item.value}
59
+ className={`${Styles.option} ${isActive ? Styles.option_active : ""}`}
60
+ onClick={() =>
61
+ {
62
+ const next = isActive
63
+ ? this.state.selectedValues.filter(v => v !== item.value)
64
+ : [...this.state.selectedValues, item.value];
65
+ this.setState({ selectedValues: next });
66
+ onChanged(next.length > 0 ? next.map(v => this.buildFilterItem(v)) : null);
67
+ }}
68
+ >
69
+ {item.title}
70
+ </button>
71
+ );
72
+ })
73
+ }
74
+ </div>
75
+ );
76
+ }
77
+
78
+ return null;
79
+ }
80
+ }
@@ -0,0 +1,137 @@
1
+ import { FilterItem, FilterItemOperator, SetTimeouService, SortItem } from "namirasoft-core";
2
+ import { NSBoxDouble, NSBoxEntity, NSRow } from "namirasoft-site-react";
3
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
4
+ import { NSAFilterBoxBoolean } from "./NSAFilterBoxBoolean";
5
+
6
+ export interface NSAFilterBoxNumberProps extends NSAFilterBoxBaseProps
7
+ {
8
+ getValues: (search: string) => Promise<{ value: string; title: string }[]>;
9
+ }
10
+
11
+ interface NSAFilterBoxNumberState
12
+ {
13
+ valueFrom: number | null;
14
+ valueTo: number | null;
15
+ }
16
+
17
+ type ValueItem = { value: string; title: string };
18
+
19
+ const ENTITY_OPERATORS = ["equals", "notequals"];
20
+ const DEBOUNCE_MS = 1000;
21
+
22
+ export class NSAFilterBoxNumber extends NSAFilterBoxBase<NSAFilterBoxNumberProps, NSAFilterBoxNumberState>
23
+ {
24
+ private debounce = new SetTimeouService();
25
+ private betweenDebounce = new SetTimeouService();
26
+ private betweenActive = false;
27
+
28
+ constructor(props: NSAFilterBoxNumberProps)
29
+ {
30
+ super(props);
31
+ this.state = { valueFrom: null, valueTo: null };
32
+ }
33
+
34
+ private emitDebounced(value: number | null): void
35
+ {
36
+ this.debounce.setTimeoutIfNotCalledAgain(
37
+ () => this.props.onChanged(value !== null ? this.buildFilterItem(String(value)) : null),
38
+ DEBOUNCE_MS
39
+ );
40
+ }
41
+
42
+ private handleBetweenChange(patch: Partial<NSAFilterBoxNumberState>): void
43
+ {
44
+ this.setState(patch as NSAFilterBoxNumberState, () =>
45
+ {
46
+ const { valueFrom, valueTo } = this.state;
47
+ this.betweenDebounce.setTimeoutIfNotCalledAgain(() =>
48
+ {
49
+ if (valueFrom !== null && valueTo !== null)
50
+ {
51
+ this.betweenActive = true;
52
+ this.props.onChanged(this.buildBetweenFilterItems(String(valueFrom), String(valueTo)));
53
+ }
54
+ else if (this.betweenActive)
55
+ {
56
+ this.betweenActive = false;
57
+ this.props.onChanged(null);
58
+ }
59
+ }, DEBOUNCE_MS);
60
+ });
61
+ }
62
+
63
+ private list = async (
64
+ filters: FilterItem[] | null,
65
+ _page: number | null,
66
+ _size: number | null,
67
+ _sorts: SortItem[]
68
+ ): Promise<{ count: number; rows: ValueItem[] }> =>
69
+ {
70
+ const searchFilter = filters?.find(f => f.values.length > 0);
71
+ const search = searchFilter?.values[0] ?? "";
72
+ const rows = await this.props.getValues(search);
73
+ return { count: rows.length, rows };
74
+ };
75
+
76
+ override render(): JSX.Element | null
77
+ {
78
+ const { config, onChanged } = this.props;
79
+ const op = config.operatorName
80
+ ? FilterItemOperator.getByName(config.operatorName)
81
+ : null;
82
+
83
+ if (op && op.count === 0)
84
+ return <NSAFilterBoxBoolean config={config} onChanged={onChanged} />;
85
+
86
+ if (config.operatorName === "between")
87
+ return (
88
+ <NSRow>
89
+ <NSBoxDouble
90
+ title={`${this.getLabel()} From`}
91
+ placeholder={`${this.getLabel()} From`}
92
+ required={false}
93
+ hideHeader={true}
94
+ onChanged={e => this.handleBetweenChange({ valueFrom: e.getValue() })}
95
+ />
96
+ <NSBoxDouble
97
+ title={`${this.getLabel()} To`}
98
+ placeholder={`${this.getLabel()} To`}
99
+ required={false}
100
+ hideHeader={true}
101
+ onChanged={e => this.handleBetweenChange({ valueTo: e.getValue() })}
102
+ />
103
+ </NSRow>
104
+ );
105
+
106
+ if (config.operatorName && ENTITY_OPERATORS.includes(config.operatorName))
107
+ return (
108
+ <NSBoxEntity<ValueItem>
109
+ title={this.getLabel()}
110
+ placeholder={this.getLabel()}
111
+ required={false}
112
+ hideHeader={true}
113
+ multiple={false}
114
+ list={this.list}
115
+ table_name={config.column.table.name}
116
+ getValue={item => item.value}
117
+ getTitle={item => item.title}
118
+ getSort={() => null}
119
+ onChanged={e =>
120
+ {
121
+ const val = e.getValueOne();
122
+ onChanged(val ? this.buildFilterItem(val) : null);
123
+ }}
124
+ />
125
+ );
126
+
127
+ return (
128
+ <NSBoxDouble
129
+ title={this.getLabel()}
130
+ placeholder={this.getLabel()}
131
+ required={false}
132
+ hideHeader={true}
133
+ onChanged={e => this.emitDebounced(e.getValue())}
134
+ />
135
+ );
136
+ }
137
+ }
@@ -0,0 +1,82 @@
1
+ import { FilterItem, FilterItemOperator, SetTimeouService, SortItem } from "namirasoft-core";
2
+ import { NSBoxEntity, NSBoxString } from "namirasoft-site-react";
3
+ import { NSAFilterBoxBase, NSAFilterBoxBaseProps } from "./NSAFilterBoxBase";
4
+ import { NSAFilterBoxBoolean } from "./NSAFilterBoxBoolean";
5
+
6
+ export interface NSAFilterBoxStringProps extends NSAFilterBoxBaseProps
7
+ {
8
+ getValues: (search: string) => Promise<{ value: string; title: string }[]>;
9
+ }
10
+
11
+ type ValueItem = { value: string; title: string };
12
+
13
+ const ENTITY_OPERATORS = ["equals", "notequals"];
14
+ const DEBOUNCE_MS = 1000;
15
+
16
+ export class NSAFilterBoxString extends NSAFilterBoxBase<NSAFilterBoxStringProps>
17
+ {
18
+ private debounce = new SetTimeouService();
19
+
20
+ private list = async (
21
+ filters: FilterItem[] | null,
22
+ _page: number | null,
23
+ _size: number | null,
24
+ _sorts: SortItem[]
25
+ ): Promise<{ count: number; rows: ValueItem[] }> =>
26
+ {
27
+ const searchFilter = filters?.find(f => f.values.length > 0);
28
+ const search = searchFilter?.values[0] ?? "";
29
+ const rows = await this.props.getValues(search);
30
+ return { count: rows.length, rows };
31
+ };
32
+
33
+ private emitDebounced(value: string | null): void
34
+ {
35
+ this.debounce.setTimeoutIfNotCalledAgain(
36
+ () => this.props.onChanged(value ? this.buildFilterItem(value) : null),
37
+ DEBOUNCE_MS
38
+ );
39
+ }
40
+
41
+ override render(): JSX.Element | null
42
+ {
43
+ const { config, onChanged } = this.props;
44
+ const op = config.operatorName
45
+ ? FilterItemOperator.getByName(config.operatorName)
46
+ : null;
47
+
48
+ if (op && op.count === 0)
49
+ return <NSAFilterBoxBoolean config={config} onChanged={onChanged} />;
50
+
51
+ if (config.operatorName && ENTITY_OPERATORS.includes(config.operatorName))
52
+ return (
53
+ <NSBoxEntity<ValueItem>
54
+ title={this.getLabel()}
55
+ placeholder={this.getLabel()}
56
+ required={false}
57
+ hideHeader={true}
58
+ multiple={false}
59
+ list={this.list}
60
+ table_name={config.column.table.name}
61
+ getValue={item => item.value}
62
+ getTitle={item => item.title}
63
+ getSort={() => null}
64
+ onChanged={e =>
65
+ {
66
+ const val = e.getValueOne();
67
+ onChanged(val ? this.buildFilterItem(val) : null);
68
+ }}
69
+ />
70
+ );
71
+
72
+ return (
73
+ <NSBoxString
74
+ title={this.getLabel()}
75
+ placeholder={this.getLabel()}
76
+ required={false}
77
+ hideHeader={true}
78
+ onChanged={e => this.emitDebounced(e.getValue())}
79
+ />
80
+ );
81
+ }
82
+ }