@salesforce/webapp-template-feature-react-global-search-experimental 1.76.1 → 1.78.0

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 (24) hide show
  1. package/dist/CHANGELOG.md +16 -0
  2. package/dist/force-app/main/default/webapplications/feature-react-global-search/package.json +3 -3
  3. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/api/objectInfoGraphQLService.ts +108 -156
  4. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/api/objectInfoService.ts +5 -10
  5. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/components/detail/UiApiDetailForm.tsx +2 -2
  6. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/hooks/useObjectInfoBatch.ts +1 -1
  7. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/filters/picklist.ts +5 -31
  8. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/objectInfo/objectInfo.ts +46 -163
  9. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/schema.d.ts +200 -0
  10. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/utils/graphQLObjectInfoAdapter.ts +37 -279
  11. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/utils/recordUtils.ts +4 -4
  12. package/dist/force-app/main/default/webapplications/feature-react-global-search/src/index.ts +0 -5
  13. package/dist/package.json +1 -1
  14. package/package.json +1 -1
  15. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/api/objectInfoGraphQLService.ts +108 -156
  16. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/api/objectInfoService.ts +5 -10
  17. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/components/detail/UiApiDetailForm.tsx +2 -2
  18. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/hooks/useObjectInfoBatch.ts +1 -1
  19. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/filters/picklist.ts +5 -31
  20. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/objectInfo/objectInfo.ts +46 -163
  21. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/types/schema.d.ts +200 -0
  22. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/utils/graphQLObjectInfoAdapter.ts +37 -279
  23. package/src/force-app/main/default/webapplications/feature-react-global-search/src/features/global-search/utils/recordUtils.ts +4 -4
  24. package/src/force-app/main/default/webapplications/feature-react-global-search/src/index.ts +0 -5
@@ -0,0 +1,200 @@
1
+ export type Maybe<T> = T | null;
2
+ export type InputMaybe<T> = Maybe<T>;
3
+ export type Exact<T extends { [key: string]: unknown }> = { [K in keyof T]: T[K] };
4
+ export type MakeOptional<T, K extends keyof T> = Omit<T, K> & { [SubKey in K]?: Maybe<T[SubKey]> };
5
+ export type MakeMaybe<T, K extends keyof T> = Omit<T, K> & { [SubKey in K]: Maybe<T[SubKey]> };
6
+ export type MakeEmpty<T extends { [key: string]: unknown }, K extends keyof T> = {
7
+ [_ in K]?: never;
8
+ };
9
+ export type Incremental<T> =
10
+ | T
11
+ | { [P in keyof T]?: P extends " $fragmentName" | "__typename" ? T[P] : never };
12
+ /** All built-in and custom scalars, mapped to their actual values */
13
+ export type Scalars = {
14
+ ID: { input: string; output: string };
15
+ String: { input: string; output: string };
16
+ Boolean: { input: boolean; output: boolean };
17
+ Int: { input: number; output: number };
18
+ Float: { input: number; output: number };
19
+ Base64: { input: string; output: string };
20
+ /** An arbitrary precision signed decimal */
21
+ BigDecimal: { input: number | string; output: number };
22
+ /** An arbitrary precision signed integer */
23
+ BigInteger: { input: number; output: number };
24
+ /** An 8-bit signed integer */
25
+ Byte: { input: number; output: number };
26
+ /** A UTF-16 code unit; a character on Unicode's BMP */
27
+ Char: { input: number; output: number };
28
+ Currency: { input: number | string; output: number };
29
+ Date: { input: string; output: string };
30
+ DateTime: { input: string; output: string };
31
+ Double: { input: number | string; output: number };
32
+ Email: { input: string; output: string };
33
+ EncryptedString: { input: string; output: string };
34
+ /** Can be set to an ID or a Reference to the result of another mutation operation. */
35
+ IdOrRef: { input: string; output: string };
36
+ JSON: { input: string; output: string };
37
+ Latitude: { input: number | string; output: number };
38
+ /** A 64-bit signed integer */
39
+ Long: { input: number; output: number };
40
+ LongTextArea: { input: string; output: string };
41
+ Longitude: { input: number | string; output: number };
42
+ MultiPicklist: { input: string; output: string };
43
+ Percent: { input: number | string; output: number };
44
+ PhoneNumber: { input: string; output: string };
45
+ Picklist: { input: string; output: string };
46
+ RichTextArea: { input: string; output: string };
47
+ /** A 16-bit signed integer */
48
+ Short: { input: number; output: number };
49
+ TextArea: { input: string; output: string };
50
+ Time: { input: string; output: string };
51
+ Url: { input: string; output: string };
52
+ };
53
+
54
+ export enum DataType {
55
+ Address = "ADDRESS",
56
+ Anytype = "ANYTYPE",
57
+ Base64 = "BASE64",
58
+ Boolean = "BOOLEAN",
59
+ Combobox = "COMBOBOX",
60
+ Complexvalue = "COMPLEXVALUE",
61
+ Currency = "CURRENCY",
62
+ Date = "DATE",
63
+ Datetime = "DATETIME",
64
+ Double = "DOUBLE",
65
+ Email = "EMAIL",
66
+ Encryptedstring = "ENCRYPTEDSTRING",
67
+ Int = "INT",
68
+ Json = "JSON",
69
+ Junctionidlist = "JUNCTIONIDLIST",
70
+ Location = "LOCATION",
71
+ Long = "LONG",
72
+ Multipicklist = "MULTIPICKLIST",
73
+ Percent = "PERCENT",
74
+ Phone = "PHONE",
75
+ Picklist = "PICKLIST",
76
+ Reference = "REFERENCE",
77
+ String = "STRING",
78
+ Textarea = "TEXTAREA",
79
+ Time = "TIME",
80
+ Url = "URL",
81
+ }
82
+
83
+ export enum FieldExtraTypeInfo {
84
+ ExternalLookup = "EXTERNAL_LOOKUP",
85
+ ImageUrl = "IMAGE_URL",
86
+ IndirectLookup = "INDIRECT_LOOKUP",
87
+ Personname = "PERSONNAME",
88
+ Plaintextarea = "PLAINTEXTAREA",
89
+ Richtextarea = "RICHTEXTAREA",
90
+ SwitchablePersonname = "SWITCHABLE_PERSONNAME",
91
+ }
92
+
93
+ /** Input for ObjectInfo and PickValues */
94
+ export type ObjectInfoInput = {
95
+ apiName: Scalars["String"]["input"];
96
+ fieldNames?: InputMaybe<Array<Scalars["String"]["input"]>>;
97
+ recordTypeIDs?: InputMaybe<Array<Scalars["ID"]["input"]>>;
98
+ };
99
+
100
+ export enum ResultOrder {
101
+ Asc = "ASC",
102
+ Desc = "DESC",
103
+ }
104
+
105
+ export type GetObjectInfosQueryVariables = Exact<{
106
+ apiNames: Array<Scalars["String"]["input"]> | Scalars["String"]["input"];
107
+ }>;
108
+
109
+ export type GetObjectInfosQuery = {
110
+ uiapi: {
111
+ objectInfos?: Array<{
112
+ ApiName: string;
113
+ label?: string | null;
114
+ labelPlural?: string | null;
115
+ nameFields: Array<string | null>;
116
+ defaultRecordTypeId?: string | null;
117
+ keyPrefix?: string | null;
118
+ layoutable: boolean;
119
+ queryable: boolean;
120
+ searchable: boolean;
121
+ updateable: boolean;
122
+ deletable: boolean;
123
+ createable: boolean;
124
+ custom: boolean;
125
+ mruEnabled: boolean;
126
+ feedEnabled: boolean;
127
+ fields: Array<
128
+ | {
129
+ ApiName: string;
130
+ label?: string | null;
131
+ dataType?: DataType | null;
132
+ relationshipName?: string | null;
133
+ reference: boolean;
134
+ compound: boolean;
135
+ compoundFieldName?: string | null;
136
+ compoundComponentName?: string | null;
137
+ controllingFields: Array<string | null>;
138
+ controllerName?: string | null;
139
+ referenceToInfos: Array<{ ApiName: string; nameFields: Array<string | null> } | null>;
140
+ }
141
+ | {
142
+ ApiName: string;
143
+ label?: string | null;
144
+ dataType?: DataType | null;
145
+ relationshipName?: string | null;
146
+ reference: boolean;
147
+ compound: boolean;
148
+ compoundFieldName?: string | null;
149
+ compoundComponentName?: string | null;
150
+ controllingFields: Array<string | null>;
151
+ controllerName?: string | null;
152
+ referenceToInfos: Array<{ ApiName: string; nameFields: Array<string | null> } | null>;
153
+ }
154
+ | null
155
+ >;
156
+ recordTypeInfos: Array<{
157
+ recordTypeId?: string | null;
158
+ name?: string | null;
159
+ master: boolean;
160
+ available: boolean;
161
+ defaultRecordTypeMapping: boolean;
162
+ } | null>;
163
+ themeInfo?: { color?: string | null; iconUrl?: string | null } | null;
164
+ childRelationships: Array<{
165
+ relationshipName?: string | null;
166
+ fieldName?: string | null;
167
+ childObjectApiName: string;
168
+ } | null>;
169
+ dependentFields: Array<{ controllingField: string } | null>;
170
+ } | null> | null;
171
+ };
172
+ };
173
+
174
+ export type GetPicklistValuesQueryVariables = Exact<{
175
+ objectInfoInputs: Array<ObjectInfoInput> | ObjectInfoInput;
176
+ }>;
177
+
178
+ export type GetPicklistValuesQuery = {
179
+ uiapi: {
180
+ objectInfos?: Array<{
181
+ ApiName: string;
182
+ fields: Array<
183
+ | {
184
+ ApiName: string;
185
+ picklistValuesByRecordTypeIDs?: Array<{
186
+ recordTypeID: string;
187
+ defaultValue?: { value?: string | null } | null;
188
+ picklistValues?: Array<{
189
+ label?: string | null;
190
+ value?: string | null;
191
+ validFor?: Array<number | null> | null;
192
+ }> | null;
193
+ } | null> | null;
194
+ }
195
+ | { ApiName: string }
196
+ | null
197
+ >;
198
+ } | null> | null;
199
+ };
200
+ };
@@ -10,259 +10,36 @@
10
10
 
11
11
  import type {
12
12
  ObjectInfoBatchResponse,
13
+ GetObjectInfosQueryObjectInfos,
14
+ GetPicklistValuesQueryObjectInfo,
15
+ GetPicklistValuesQueryField,
16
+ GetObjectInfosQueryObjectInfo,
13
17
  ObjectInfoResult,
14
- Field,
15
- RecordTypeInfo,
16
- ThemeInfo,
18
+ PicklistValue as GraphQLPicklistValue,
17
19
  } from "../types/objectInfo/objectInfo";
18
20
  import type { PicklistValue } from "../types/filters/picklist";
19
21
 
20
- type GraphQLNode = Record<string, unknown>;
21
-
22
- function getStr(node: GraphQLNode, key: string): string {
23
- const v = node[key] ?? node[key.charAt(0).toLowerCase() + key.slice(1)];
24
- return typeof v === "string" ? v : "";
25
- }
26
-
27
- function getNum(node: GraphQLNode, key: string): number {
28
- const v = node[key] ?? node[key.charAt(0).toLowerCase() + key.slice(1)];
29
- return typeof v === "number" ? v : 0;
30
- }
31
-
32
- function getBool(node: GraphQLNode, key: string): boolean {
33
- const v = node[key] ?? node[key.charAt(0).toLowerCase() + key.slice(1)];
34
- return typeof v === "boolean" ? v : false;
35
- }
36
-
37
- function getArr<T>(node: GraphQLNode, key: string, map: (item: unknown) => T): T[] {
38
- const v = node[key] ?? node[key.charAt(0).toLowerCase() + key.slice(1)];
39
- if (!Array.isArray(v)) return [];
40
- return v.map((item) => map(item));
41
- }
42
-
43
- function getObj(node: GraphQLNode, key: string): Record<string, unknown> {
44
- const v = node[key] ?? node[key.charAt(0).toLowerCase() + key.slice(1)];
45
- return v != null && typeof v === "object" && !Array.isArray(v)
46
- ? (v as Record<string, unknown>)
47
- : {};
48
- }
49
-
50
- /** GraphQL returns dependentFields as [DependentField]; REST expects Record. Normalize to object. */
51
- function toDependentFieldsRecord(v: unknown): Record<string, unknown> {
52
- if (v == null) return {};
53
- if (Array.isArray(v)) return {};
54
- return typeof v === "object" ? (v as Record<string, unknown>) : {};
55
- }
56
-
57
- /** Canonical dataType casing for formatters (Phone, FormattedAddress) and reference resolution (Reference). */
58
- const DATA_TYPE_CANONICAL: Record<string, string> = {
59
- phone: "Phone",
60
- email: "Email",
61
- url: "Url",
62
- address: "Address",
63
- reference: "Reference",
64
- };
65
-
66
- function normalizeDataType(raw: string): string {
67
- if (!raw) return raw;
68
- const lower = raw.toLowerCase();
69
- return DATA_TYPE_CANONICAL[lower] ?? raw;
70
- }
71
-
72
- /** Normalize GraphQL Field to our Field type (REST-compatible). */
73
- function mapField(fieldNode: unknown): [string, Field] {
74
- const n = (fieldNode != null && typeof fieldNode === "object" ? fieldNode : {}) as GraphQLNode;
75
- const apiName = getStr(n, "ApiName") || getStr(n, "apiName");
76
- const label = getStr(n, "label");
77
- const dataType = normalizeDataType(getStr(n, "dataType"));
78
- const relationshipName = getStr(n, "relationshipName") || null;
79
- const reference = getBool(n, "reference");
80
- const compound = getBool(n, "compound");
81
- const compoundFieldName = getStr(n, "compoundFieldName") || null;
82
- const compoundComponentName = getStr(n, "compoundComponentName") || null;
83
- const controllingFields = getArr(n, "controllingFields", String);
84
- const controllerName = getStr(n, "controllerName") || null;
85
- const referenceToInfos = getArr(n, "referenceToInfos", (item) => {
86
- const r = (item != null && typeof item === "object" ? item : {}) as GraphQLNode;
87
- return {
88
- apiName: getStr(r, "apiName") || getStr(r, "ApiName"),
89
- nameFields: getArr(r, "nameFields", String),
90
- };
91
- });
92
- return [
93
- apiName,
94
- {
95
- apiName,
96
- label,
97
- dataType,
98
- relationshipName,
99
- reference,
100
- compound,
101
- compoundFieldName,
102
- compoundComponentName,
103
- controllingFields,
104
- controllerName,
105
- referenceToInfos,
106
- calculated: getBool(n, "calculated"),
107
- createable: getBool(n, "createable"),
108
- custom: getBool(n, "custom"),
109
- defaultValue: n.defaultValue ?? null,
110
- defaultedOnCreate: getBool(n, "defaultedOnCreate"),
111
- digits: getNum(n, "digits"),
112
- externalId: getBool(n, "externalId"),
113
- extraTypeInfo: getStr(n, "extraTypeInfo") || null,
114
- filterable: getBool(n, "filterable"),
115
- filteredLookupInfo: null,
116
- highScaleNumber: getBool(n, "highScaleNumber"),
117
- htmlFormatted: getBool(n, "htmlFormatted"),
118
- inlineHelpText: getStr(n, "inlineHelpText") || null,
119
- length: getNum(n, "length"),
120
- maskType: getStr(n, "maskType") || null,
121
- nameField: getBool(n, "nameField"),
122
- polymorphicForeignKey: getBool(n, "polymorphicForeignKey"),
123
- precision: getNum(n, "precision"),
124
- required: getBool(n, "required"),
125
- scale: getNum(n, "scale"),
126
- searchPrefilterable: getBool(n, "searchPrefilterable"),
127
- sortable: getBool(n, "sortable"),
128
- unique: getBool(n, "unique"),
129
- updateable: getBool(n, "updateable"),
130
- referenceTargetField: getStr(n, "referenceTargetField") || null,
131
- } as Field,
132
- ];
133
- }
134
-
135
- /** Normalize GraphQL recordTypeInfo to RecordTypeInfo. */
136
- function mapRecordTypeInfo(rtNode: unknown): [string, RecordTypeInfo] {
137
- const n = (rtNode != null && typeof rtNode === "object" ? rtNode : {}) as GraphQLNode;
138
- const recordTypeId = getStr(n, "recordTypeId");
139
- return [
140
- recordTypeId,
141
- {
142
- recordTypeId,
143
- name: getStr(n, "name"),
144
- master: getBool(n, "master"),
145
- available: getBool(n, "available"),
146
- defaultRecordTypeMapping: getBool(n, "defaultRecordTypeMapping"),
147
- } as RecordTypeInfo,
148
- ];
149
- }
150
-
151
- function toFieldsMap(fieldsNode: unknown): Record<string, Field> {
152
- if (fieldsNode == null) return {};
153
- if (Array.isArray(fieldsNode)) {
154
- return Object.fromEntries((fieldsNode as unknown[]).map(mapField).filter(([k]) => k));
155
- }
156
- if (typeof fieldsNode === "object" && !Array.isArray(fieldsNode)) {
157
- const entries = Object.entries(fieldsNode as Record<string, unknown>).map(([, v]) =>
158
- mapField(v),
159
- );
160
- return Object.fromEntries(entries.filter(([k]) => k));
161
- }
162
- return {};
163
- }
164
-
165
- /**
166
- * Ensures compound parent fields (e.g. BillingAddress, ShippingAddress) exist in the fields map
167
- * with dataType "Address" so layout transform can set item.dataType for FormattedAddress.
168
- * GraphQL may only return component fields (BillingStreet, etc.) with compoundFieldName.
169
- */
170
- function ensureCompoundParentFields(fields: Record<string, Field>): Record<string, Field> {
171
- const result = { ...fields };
172
- for (const field of Object.values(fields)) {
173
- const compoundName = field.compoundFieldName;
174
- if (!compoundName || result[compoundName] != null) continue;
175
- const dataType =
176
- compoundName.endsWith("Address") || compoundName.endsWith("address") ? "Address" : "String";
177
- result[compoundName] = {
178
- ...field,
179
- apiName: compoundName,
180
- label: compoundName,
181
- dataType,
182
- compound: true,
183
- compoundFieldName: null,
184
- compoundComponentName: null,
185
- } as Field;
186
- }
187
- return result;
188
- }
189
-
190
- function toRecordTypeInfosMap(rtNode: unknown): Record<string, RecordTypeInfo> {
191
- if (rtNode == null) return {};
192
- if (Array.isArray(rtNode)) {
193
- return Object.fromEntries((rtNode as unknown[]).map(mapRecordTypeInfo).filter(([k]) => k));
194
- }
195
- if (typeof rtNode === "object" && !Array.isArray(rtNode)) {
196
- const entries = Object.entries(rtNode as Record<string, unknown>).map(([, v]) =>
197
- mapRecordTypeInfo(v),
198
- );
199
- return Object.fromEntries(entries.filter(([k]) => k));
200
- }
201
- return {};
202
- }
203
-
204
- /** Map a single GraphQL ObjectInfo node to ObjectInfoResult (REST-compatible). */
205
- export function graphQLObjectInfoToObjectInfoResult(node: GraphQLNode): ObjectInfoResult {
206
- const themeInfoNode = getObj(node, "themeInfo");
207
- const themeInfo: ThemeInfo = {
208
- color: getStr(themeInfoNode, "color"),
209
- iconUrl: getStr(themeInfoNode, "iconUrl"),
210
- };
211
-
212
- const childRelationships = getArr(node, "childRelationships", (item) => {
213
- const c = (item != null && typeof item === "object" ? item : {}) as GraphQLNode;
214
- return {
215
- childObjectApiName: getStr(c, "childObjectApiName"),
216
- fieldName: getStr(c, "fieldName"),
217
- junctionIdListNames: getArr(c, "junctionIdListNames", String),
218
- junctionReferenceTo: getArr(c, "junctionReferenceTo", String),
219
- relationshipName: getStr(c, "relationshipName"),
220
- };
221
- });
222
-
223
- const apiName = getStr(node, "ApiName") || getStr(node, "apiName");
224
- const fieldsNode = node.fields ?? node.Fields;
225
- const recordTypeInfosNode = node.recordTypeInfos ?? node.RecordTypeInfos;
226
-
22
+ export function graphQLObjectInfoToObjectInfoResult(
23
+ objectInfo: GetObjectInfosQueryObjectInfo,
24
+ ): ObjectInfoResult {
25
+ const { fields, ...rest } = objectInfo;
26
+ const fieldsRecord = Object.fromEntries((fields ?? []).map((field) => [field?.ApiName, field]));
227
27
  return {
228
- apiName,
229
- label: getStr(node, "label"),
230
- labelPlural: getStr(node, "labelPlural"),
231
- nameFields: getArr(node, "nameFields", String),
232
- defaultRecordTypeId: getStr(node, "defaultRecordTypeId") || "012000000000000AAA",
233
- keyPrefix: getStr(node, "keyPrefix"),
234
- layoutable: getBool(node, "layoutable"),
235
- queryable: getBool(node, "queryable"),
236
- searchable: getBool(node, "searchable"),
237
- updateable: getBool(node, "updateable"),
238
- deletable: getBool(node, "deletable"),
239
- createable: getBool(node, "createable"),
240
- custom: getBool(node, "custom"),
241
- mruEnabled: getBool(node, "mruEnabled"),
242
- feedEnabled: getBool(node, "feedEnabled"),
243
- fields: ensureCompoundParentFields(toFieldsMap(fieldsNode)),
244
- recordTypeInfos: toRecordTypeInfosMap(recordTypeInfosNode),
245
- themeInfo,
246
- childRelationships,
247
- associateEntityType: getStr(node, "associateEntityType") || null,
248
- associateParentEntity: getStr(node, "associateParentEntity") || null,
249
- compactLayoutable: getBool(node, "compactLayoutable"),
250
- searchLayoutable: getBool(node, "searchLayoutable"),
251
- dependentFields: toDependentFieldsRecord(node.dependentFields ?? node.DependentFields),
252
- eTag: getStr(node, "eTag"),
253
- } as ObjectInfoResult;
28
+ ...rest,
29
+ fields: fieldsRecord,
30
+ };
254
31
  }
255
32
 
256
33
  /** Convert GraphQL objectInfos array to ObjectInfoBatchResponse (REST shape). */
257
34
  export function graphQLObjectInfosToBatchResponse(
258
- nodes: GraphQLNode[],
35
+ objectInfos: GetObjectInfosQueryObjectInfos,
259
36
  _requestedApiNames: string[],
260
37
  ): ObjectInfoBatchResponse {
261
- const results = nodes.map((node) => ({
262
- result: graphQLObjectInfoToObjectInfoResult(node),
38
+ const results = objectInfos.map((objectInfo) => ({
39
+ result: graphQLObjectInfoToObjectInfoResult(objectInfo!),
263
40
  statusCode: 200,
264
41
  }));
265
- return { results } as ObjectInfoBatchResponse;
42
+ return { results };
266
43
  }
267
44
 
268
45
  /**
@@ -270,50 +47,31 @@ export function graphQLObjectInfosToBatchResponse(
270
47
  * Uses picklistValuesByRecordTypeIDs; prefers the given recordTypeId or first available.
271
48
  */
272
49
  export function extractPicklistValuesFromGraphQLObjectInfo(
273
- node: GraphQLNode,
50
+ objectInfo: NonNullable<GetPicklistValuesQueryObjectInfo>,
274
51
  fieldName: string,
275
52
  recordTypeId?: string,
276
53
  ): PicklistValue[] {
277
- const fieldsNode = node.fields ?? node.Fields;
278
- if (fieldsNode == null || typeof fieldsNode !== "object") return [];
279
- const fieldMap = Array.isArray(fieldsNode)
280
- ? Object.fromEntries(
281
- (fieldsNode as unknown[]).map((f: unknown) => {
282
- const n = (f != null && typeof f === "object" ? f : {}) as GraphQLNode;
283
- const k = getStr(n, "ApiName") || getStr(n, "apiName");
284
- return [k, n];
285
- }),
286
- )
287
- : (fieldsNode as Record<string, GraphQLNode>);
54
+ const fields = objectInfo.fields;
55
+ if (fields == null) return [];
56
+ const fieldMap: Record<string, GetPicklistValuesQueryField> = Object.fromEntries(
57
+ fields.map((field) => {
58
+ return [field?.ApiName, field];
59
+ }),
60
+ );
288
61
  const field = fieldMap[fieldName];
289
- if (!field || typeof field !== "object") return [];
290
- const picklistData = field.picklistValuesByRecordTypeIDs ?? field.PicklistValuesByRecordTypeIDs;
291
- if (!picklistData || !Array.isArray(picklistData)) return [];
62
+ if (!field) return [];
63
+ const picklistData =
64
+ "picklistValuesByRecordTypeIDs" in field ? field.picklistValuesByRecordTypeIDs : undefined;
65
+ if (!picklistData) return [];
292
66
  const rtId = recordTypeId ?? "012000000000000AAA";
293
- const arr = picklistData as Array<Record<string, unknown>>;
294
- const byRecordType = arr.find((p) => (p.recordTypeID ?? p.recordTypeId) === rtId);
295
- if (!byRecordType) {
296
- const first = arr[0];
297
- if (!first) return [];
298
- return mapPicklistValues(first);
299
- }
300
- return mapPicklistValues(byRecordType);
67
+ const byRecordType = picklistData.find((p) => p!.recordTypeID === rtId) ?? picklistData[0];
68
+ return mapPicklistValues(byRecordType?.picklistValues ?? []);
301
69
  }
302
70
 
303
- function mapPicklistValues(byRecordType: Record<string, unknown>): PicklistValue[] {
304
- const raw = byRecordType.picklistValues;
305
- if (!Array.isArray(raw)) return [];
306
- return raw.map((item: unknown) => {
307
- const n = (item != null && typeof item === "object" ? item : {}) as Record<string, unknown>;
308
- return {
309
- label: typeof n.label === "string" ? n.label : "",
310
- value: typeof n.value === "string" ? n.value : "",
311
- validFor: Array.isArray(n.validFor) ? n.validFor : undefined,
312
- attributes: (n.attributes != null && typeof n.attributes === "object"
313
- ? n.attributes
314
- : undefined) as Record<string, unknown> | undefined,
315
- defaultValue: typeof n.defaultValue === "boolean" ? n.defaultValue : undefined,
316
- active: typeof n.active === "boolean" ? n.active : undefined,
317
- } as PicklistValue;
318
- });
71
+ function mapPicklistValues(values: GraphQLPicklistValue[]): PicklistValue[] {
72
+ return values.map((item) => ({
73
+ ...item,
74
+ value: item.value ?? "",
75
+ label: item.label ?? "",
76
+ }));
319
77
  }
@@ -49,13 +49,13 @@ const getFetchableFieldsFromLayoutItem = function (
49
49
 
50
50
  // add field: fieldType
51
51
  const fieldMetadata = metadata.fields[comp.apiName];
52
- fields[comp.apiName] = fieldMetadata ? fieldMetadata.dataType : "";
52
+ fields[comp.apiName] = fieldMetadata?.dataType ?? "";
53
53
 
54
54
  // add relatedField if one exists (Id field -> add relationship name so we request Owner.Name)
55
55
  if (comp.apiName in metadata.fields) {
56
56
  const relationshipName = fieldMetadata?.relationshipName;
57
57
  if (relationshipName) {
58
- fields[relationshipName] = fieldMetadata.dataType;
58
+ fields[relationshipName] = fieldMetadata.dataType ?? "";
59
59
 
60
60
  relationFieldMap[comp.apiName] = relationshipName;
61
61
  }
@@ -65,7 +65,7 @@ const getFetchableFieldsFromLayoutItem = function (
65
65
  const idField = findIdFieldForRelationship(metadata, comp.apiName);
66
66
  if (idField) {
67
67
  const idMeta = metadata.fields[idField];
68
- fields[idField] = idMeta ? idMeta.dataType : "";
68
+ fields[idField] = idMeta?.dataType ?? "";
69
69
  relationFieldMap[idField] = comp.apiName;
70
70
  }
71
71
  }
@@ -137,7 +137,7 @@ export const calculateFieldsToFetch = function (
137
137
  const fields = getFetchableFieldsFromLayout(metadata, layout, relationFieldMap);
138
138
  let fieldsToFetch = Object.keys(fields);
139
139
  if (shouldPrefixedWithEntityName) {
140
- fieldsToFetch = fieldsToFetch.map((field) => `${metadata.apiName}.${field}`);
140
+ fieldsToFetch = fieldsToFetch.map((field) => `${metadata.ApiName}.${field}`);
141
141
  }
142
142
  // populate field types for o11y logging
143
143
  const fieldTypes = Object.values(fields).filter((fieldType) => fieldType !== "");
@@ -87,13 +87,8 @@ export type {
87
87
  FiltersResponse,
88
88
  } from "./features/global-search/types/filters/filters";
89
89
 
90
- export { PicklistValueArraySchema } from "./features/global-search/types/filters/picklist";
91
90
  export type { PicklistValue } from "./features/global-search/types/filters/picklist";
92
91
 
93
- export {
94
- ObjectInfoBatchResponseSchema,
95
- ObjectInfoResultSchema,
96
- } from "./features/global-search/types/objectInfo/objectInfo";
97
92
  export type {
98
93
  ObjectInfoBatchResponse,
99
94
  ObjectInfoResult,
package/dist/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@salesforce/webapp-template-base-sfdx-project-experimental",
3
- "version": "1.76.1",
3
+ "version": "1.78.0",
4
4
  "description": "Base SFDX project template",
5
5
  "private": true,
6
6
  "files": [
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@salesforce/webapp-template-feature-react-global-search-experimental",
3
- "version": "1.76.1",
3
+ "version": "1.78.0",
4
4
  "description": "Global search feature for Salesforce objects with filtering and pagination",
5
5
  "license": "SEE LICENSE IN LICENSE.txt",
6
6
  "author": "",