@webiny/app-headless-cms-common 0.0.0-unstable.e53eceafb5 → 0.0.0-unstable.e6f0dc8ca7
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/Fields/ErrorBoundary.d.ts +1 -1
- package/Fields/ErrorBoundary.js +1 -1
- package/Fields/ErrorBoundary.js.map +1 -1
- package/Fields/FieldElement.d.ts +2 -2
- package/Fields/FieldElement.js +15 -9
- package/Fields/FieldElement.js.map +1 -1
- package/Fields/FieldElementError.js +1 -2
- package/Fields/FieldElementError.js.map +1 -1
- package/Fields/FieldRulesProvider.d.ts +9 -0
- package/Fields/FieldRulesProvider.js +20 -0
- package/Fields/FieldRulesProvider.js.map +1 -0
- package/Fields/Fields.d.ts +1 -1
- package/Fields/Fields.js +137 -16
- package/Fields/Fields.js.map +1 -1
- package/Fields/LayoutDescriptorCell.d.ts +12 -0
- package/Fields/LayoutDescriptorCell.js +47 -0
- package/Fields/LayoutDescriptorCell.js.map +1 -0
- package/Fields/evaluateExpression.d.ts +23 -0
- package/Fields/evaluateExpression.js +102 -0
- package/Fields/evaluateExpression.js.map +1 -0
- package/Fields/fieldOptions.d.ts +36 -0
- package/Fields/fieldOptions.js +113 -0
- package/Fields/fieldOptions.js.map +1 -0
- package/Fields/index.d.ts +8 -3
- package/Fields/index.js +8 -3
- package/Fields/index.js.map +1 -1
- package/Fields/layoutFieldRenderers/AlertFieldRenderer.d.ts +7 -0
- package/Fields/layoutFieldRenderers/AlertFieldRenderer.js +13 -0
- package/Fields/layoutFieldRenderers/AlertFieldRenderer.js.map +1 -0
- package/Fields/layoutFieldRenderers/SeparatorFieldRenderer.d.ts +7 -0
- package/Fields/layoutFieldRenderers/SeparatorFieldRenderer.js +18 -0
- package/Fields/layoutFieldRenderers/SeparatorFieldRenderer.js.map +1 -0
- package/Fields/layoutFieldRenderers/TabsFieldRenderer.d.ts +12 -0
- package/Fields/layoutFieldRenderers/TabsFieldRenderer.js +72 -0
- package/Fields/layoutFieldRenderers/TabsFieldRenderer.js.map +1 -0
- package/Fields/operatorOptions.d.ts +10 -0
- package/Fields/operatorOptions.js +92 -0
- package/Fields/operatorOptions.js.map +1 -0
- package/Fields/useBind.d.ts +2 -1
- package/Fields/useBind.js +12 -8
- package/Fields/useBind.js.map +1 -1
- package/Fields/useFieldRules.d.ts +32 -0
- package/Fields/useFieldRules.js +153 -0
- package/Fields/useFieldRules.js.map +1 -0
- package/Fields/useRenderPlugins.d.ts +2 -1
- package/Fields/useRenderPlugins.js.map +1 -1
- package/ModelFieldProvider/CanEditField.d.ts +5 -0
- package/ModelFieldProvider/CanEditField.js +13 -0
- package/ModelFieldProvider/CanEditField.js.map +1 -0
- package/ModelFieldProvider/ModelFieldContext.d.ts +16 -13
- package/ModelFieldProvider/ModelFieldContext.js.map +1 -1
- package/ModelFieldProvider/index.d.ts +3 -2
- package/ModelFieldProvider/index.js +3 -2
- package/ModelFieldProvider/index.js.map +1 -1
- package/ModelFieldProvider/useModelField.d.ts +2 -3
- package/ModelFieldProvider/useModelField.js +2 -2
- package/ModelFieldProvider/useModelField.js.map +1 -1
- package/ModelProvider/ModelContext.d.ts +1 -1
- package/ModelProvider/ModelContext.js +7 -1
- package/ModelProvider/ModelContext.js.map +1 -1
- package/ModelProvider/index.d.ts +2 -2
- package/ModelProvider/index.js +2 -2
- package/ModelProvider/index.js.map +1 -1
- package/ModelProvider/useModel.d.ts +1 -1
- package/ModelProvider/useModel.js +1 -1
- package/ModelProvider/useModel.js.map +1 -1
- package/README.md +6 -13
- package/constants.d.ts +1 -1
- package/constants.js +1 -1
- package/constants.js.map +1 -1
- package/createFieldsList.d.ts +5 -1
- package/createFieldsList.js +4 -0
- package/createFieldsList.js.map +1 -1
- package/createValidationContainer.d.ts +2 -2
- package/createValidationContainer.js +5 -8
- package/createValidationContainer.js.map +1 -1
- package/createValidators.d.ts +2 -2
- package/createValidators.js +2 -2
- package/createValidators.js.map +1 -1
- package/entries.graphql.d.ts +12 -1
- package/entries.graphql.js +57 -16
- package/entries.graphql.js.map +1 -1
- package/exports/admin/cms/model.d.ts +1 -0
- package/exports/admin/cms/model.js +3 -0
- package/exports/admin/cms/model.js.map +1 -0
- package/exports/admin/cms.d.ts +1 -0
- package/exports/admin/cms.js +3 -0
- package/exports/admin/cms.js.map +1 -0
- package/getModelTitleFieldId.d.ts +1 -1
- package/getModelTitleFieldId.js.map +1 -1
- package/index.d.ts +10 -10
- package/index.js +10 -10
- package/index.js.map +1 -1
- package/normalizeIcon.d.ts +3 -0
- package/normalizeIcon.js +10 -0
- package/normalizeIcon.js.map +1 -0
- package/package.json +27 -29
- package/prepareFormData.d.ts +1 -1
- package/prepareFormData.js +1 -1
- package/prepareFormData.js.map +1 -1
- package/types/index.d.ts +92 -37
- package/types/index.js +1 -16
- package/types/index.js.map +1 -1
- package/types/model.d.ts +71 -18
- package/types/model.js +28 -1
- package/types/model.js.map +1 -1
- package/types/validation.d.ts +2 -2
- package/types/validation.js.map +1 -1
package/Fields/ErrorBoundary.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["React","FieldElementError","ErrorBoundary","Component","constructor","props","state","hasError","error","undefined","getDerivedStateFromError","componentDidCatch","errorInfo","field","console","groupCollapsed","fieldId","id","type","log","groupEnd","render","createElement","title","message","description","children"],"sources":["ErrorBoundary.tsx"],"sourcesContent":["import type { ErrorInfo } from \"react\";\nimport React from \"react\";\nimport type { CmsModelField } from \"~/types\";\nimport { FieldElementError } from \"./FieldElementError\";\n\ntype State =\n | {\n hasError: true;\n error: Error;\n }\n | { hasError: false; error: undefined };\n\ninterface Props {\n field: CmsModelField;\n [key: string]: any;\n}\n\nexport class ErrorBoundary extends React.Component<Props, State> {\n constructor(props: Props) {\n super(props);\n this.state = {\n hasError: false,\n error: undefined\n };\n }\n\n static getDerivedStateFromError(error: Error) {\n return {\n hasError: true,\n error\n };\n }\n\n public override componentDidCatch(error: Error, errorInfo: ErrorInfo) {\n const { field } = this.props;\n if (!field) {\n return;\n }\n console.groupCollapsed(\n `%cFIELD ERROR%c: An error occurred while rendering model field \"${field.fieldId}\" (${field.id}) of type \"${field.type}\".`,\n \"color:red\",\n \"color:default\"\n );\n console.log(\"Field definition\", field);\n console.error(error, errorInfo);\n console.groupEnd();\n }\n\n public override render() {\n if (this.state.hasError) {\n return (\n <FieldElementError\n title={`Error: ${this.state.error.message}`}\n description={\"See developer console for more details.\"}\n />\n );\n }\n\n return this.props.children;\n }\n}\n"],"mappings":"AACA,OAAOA,KAAK,MAAM,OAAO;AAEzB,SAASC,iBAAiB;AAc1B,OAAO,MAAMC,aAAa,SAASF,KAAK,CAACG,SAAS,CAAe;EAC7DC,WAAWA,CAACC,KAAY,EAAE;IACtB,KAAK,CAACA,KAAK,CAAC;IACZ,IAAI,CAACC,KAAK,GAAG;MACTC,QAAQ,EAAE,KAAK;MACfC,KAAK,EAAEC;IACX,CAAC;EACL;EAEA,OAAOC,wBAAwBA,CAACF,KAAY,EAAE;IAC1C,OAAO;MACHD,QAAQ,EAAE,IAAI;MACdC;IACJ,CAAC;EACL;EAEgBG,iBAAiBA,CAACH,KAAY,EAAEI,SAAoB,EAAE;IAClE,MAAM;MAAEC;IAAM,CAAC,GAAG,IAAI,CAACR,KAAK;IAC5B,IAAI,CAACQ,KAAK,EAAE;MACR;IACJ;IACAC,OAAO,CAACC,cAAc,CAClB,mEAAmEF,KAAK,CAACG,OAAO,MAAMH,KAAK,CAACI,EAAE,cAAcJ,KAAK,CAACK,IAAI,IAAI,EAC1H,WAAW,EACX,eACJ,CAAC;IACDJ,OAAO,CAACK,GAAG,CAAC,kBAAkB,EAAEN,KAAK,CAAC;IACtCC,OAAO,CAACN,KAAK,CAACA,KAAK,EAAEI,SAAS,CAAC;IAC/BE,OAAO,CAACM,QAAQ,CAAC,CAAC;EACtB;EAEgBC,MAAMA,CAAA,EAAG;IACrB,IAAI,IAAI,CAACf,KAAK,CAACC,QAAQ,EAAE;MACrB,oBACIP,KAAA,CAAAsB,aAAA,CAACrB,iBAAiB;QACdsB,KAAK,EAAE,UAAU,IAAI,CAACjB,KAAK,CAACE,KAAK,CAACgB,OAAO,EAAG;QAC5CC,WAAW,EAAE;MAA0C,CAC1D,CAAC;IAEV;IAEA,OAAO,IAAI,CAACpB,KAAK,CAACqB,QAAQ;EAC9B;AACJ","ignoreList":[]}
|
|
1
|
+
{"version":3,"names":["React","FieldElementError","ErrorBoundary","Component","constructor","props","state","hasError","error","undefined","getDerivedStateFromError","componentDidCatch","errorInfo","field","console","groupCollapsed","fieldId","id","type","log","groupEnd","render","createElement","title","message","description","children"],"sources":["ErrorBoundary.tsx"],"sourcesContent":["import type { ErrorInfo } from \"react\";\nimport React from \"react\";\nimport type { CmsModelField } from \"~/types/index.js\";\nimport { FieldElementError } from \"./FieldElementError.js\";\n\ntype State =\n | {\n hasError: true;\n error: Error;\n }\n | { hasError: false; error: undefined };\n\ninterface Props {\n field: CmsModelField;\n [key: string]: any;\n}\n\nexport class ErrorBoundary extends React.Component<Props, State> {\n constructor(props: Props) {\n super(props);\n this.state = {\n hasError: false,\n error: undefined\n };\n }\n\n static getDerivedStateFromError(error: Error) {\n return {\n hasError: true,\n error\n };\n }\n\n public override componentDidCatch(error: Error, errorInfo: ErrorInfo) {\n const { field } = this.props;\n if (!field) {\n return;\n }\n console.groupCollapsed(\n `%cFIELD ERROR%c: An error occurred while rendering model field \"${field.fieldId}\" (${field.id}) of type \"${field.type}\".`,\n \"color:red\",\n \"color:default\"\n );\n console.log(\"Field definition\", field);\n console.error(error, errorInfo);\n console.groupEnd();\n }\n\n public override render() {\n if (this.state.hasError) {\n return (\n <FieldElementError\n title={`Error: ${this.state.error.message}`}\n description={\"See developer console for more details.\"}\n />\n );\n }\n\n return this.props.children;\n }\n}\n"],"mappings":"AACA,OAAOA,KAAK,MAAM,OAAO;AAEzB,SAASC,iBAAiB;AAc1B,OAAO,MAAMC,aAAa,SAASF,KAAK,CAACG,SAAS,CAAe;EAC7DC,WAAWA,CAACC,KAAY,EAAE;IACtB,KAAK,CAACA,KAAK,CAAC;IACZ,IAAI,CAACC,KAAK,GAAG;MACTC,QAAQ,EAAE,KAAK;MACfC,KAAK,EAAEC;IACX,CAAC;EACL;EAEA,OAAOC,wBAAwBA,CAACF,KAAY,EAAE;IAC1C,OAAO;MACHD,QAAQ,EAAE,IAAI;MACdC;IACJ,CAAC;EACL;EAEgBG,iBAAiBA,CAACH,KAAY,EAAEI,SAAoB,EAAE;IAClE,MAAM;MAAEC;IAAM,CAAC,GAAG,IAAI,CAACR,KAAK;IAC5B,IAAI,CAACQ,KAAK,EAAE;MACR;IACJ;IACAC,OAAO,CAACC,cAAc,CAClB,mEAAmEF,KAAK,CAACG,OAAO,MAAMH,KAAK,CAACI,EAAE,cAAcJ,KAAK,CAACK,IAAI,IAAI,EAC1H,WAAW,EACX,eACJ,CAAC;IACDJ,OAAO,CAACK,GAAG,CAAC,kBAAkB,EAAEN,KAAK,CAAC;IACtCC,OAAO,CAACN,KAAK,CAACA,KAAK,EAAEI,SAAS,CAAC;IAC/BE,OAAO,CAACM,QAAQ,CAAC,CAAC;EACtB;EAEgBC,MAAMA,CAAA,EAAG;IACrB,IAAI,IAAI,CAACf,KAAK,CAACC,QAAQ,EAAE;MACrB,oBACIP,KAAA,CAAAsB,aAAA,CAACrB,iBAAiB;QACdsB,KAAK,EAAE,UAAU,IAAI,CAACjB,KAAK,CAACE,KAAK,CAACgB,OAAO,EAAG;QAC5CC,WAAW,EAAE;MAA0C,CAC1D,CAAC;IAEV;IAEA,OAAO,IAAI,CAACpB,KAAK,CAACqB,QAAQ;EAC9B;AACJ","ignoreList":[]}
|
package/Fields/FieldElement.d.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import type {
|
|
3
|
-
declare
|
|
2
|
+
import type { BindComponent, CmsEditorContentModel, CmsModelField } from "../types/index.js";
|
|
3
|
+
declare module "react" {
|
|
4
4
|
namespace JSX {
|
|
5
5
|
interface IntrinsicElements {
|
|
6
6
|
"hcms-model-field": {
|
package/Fields/FieldElement.js
CHANGED
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import get from "lodash/get";
|
|
2
|
+
import get from "lodash/get.js";
|
|
3
3
|
import { makeDecoratable } from "@webiny/app-admin";
|
|
4
|
-
import { i18n } from "@webiny/app/i18n";
|
|
5
|
-
import Label from "./Label";
|
|
6
|
-
import { useBind } from "./useBind";
|
|
7
|
-
import { useRenderPlugins } from "./useRenderPlugins";
|
|
8
|
-
import { ModelFieldProvider, useModelField } from "../ModelFieldProvider";
|
|
9
|
-
import { ErrorBoundary } from "./ErrorBoundary";
|
|
4
|
+
import { i18n } from "@webiny/app/i18n/index.js";
|
|
5
|
+
import Label from "./Label.js";
|
|
6
|
+
import { useBind } from "./useBind.js";
|
|
7
|
+
import { useRenderPlugins } from "./useRenderPlugins.js";
|
|
8
|
+
import { ModelFieldProvider, useModelField } from "../ModelFieldProvider/index.js";
|
|
9
|
+
import { ErrorBoundary } from "./ErrorBoundary.js";
|
|
10
10
|
const t = i18n.ns("app-headless-cms/admin/components/content-form");
|
|
11
11
|
const RenderField = props => {
|
|
12
12
|
const renderPlugins = useRenderPlugins();
|
|
@@ -28,9 +28,15 @@ const RenderField = props => {
|
|
|
28
28
|
contentModel
|
|
29
29
|
}));
|
|
30
30
|
}
|
|
31
|
-
const
|
|
31
|
+
const fieldRendererName = get(field, "renderer.name");
|
|
32
|
+
if (!fieldRendererName) {
|
|
33
|
+
return t`Cannot render "{fieldName}" field - field renderer not defined.`({
|
|
34
|
+
fieldName: /*#__PURE__*/React.createElement("strong", null, field.fieldId)
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
const renderPlugin = renderPlugins.find(plugin => plugin.renderer.rendererName === fieldRendererName);
|
|
32
38
|
if (!renderPlugin) {
|
|
33
|
-
return t`Cannot render "{fieldName}" field - field renderer missing.`({
|
|
39
|
+
return t`Cannot render "{fieldName}" field - field renderer missing in Admin UI plugins.`({
|
|
34
40
|
fieldName: /*#__PURE__*/React.createElement("strong", null, field.fieldId)
|
|
35
41
|
});
|
|
36
42
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["React","get","makeDecoratable","i18n","Label","useBind","useRenderPlugins","ModelFieldProvider","useModelField","ErrorBoundary","t","ns","RenderField","props","renderPlugins","Bind","contentModel","field","getBind","renderer","createElement","Fragment","
|
|
1
|
+
{"version":3,"names":["React","get","makeDecoratable","i18n","Label","useBind","useRenderPlugins","ModelFieldProvider","useModelField","ErrorBoundary","t","ns","RenderField","props","renderPlugins","Bind","contentModel","field","getBind","renderer","createElement","Fragment","fieldRendererName","fieldName","fieldId","renderPlugin","find","plugin","rendererName","render","FieldElement","id","type","RenderFieldElement"],"sources":["FieldElement.tsx"],"sourcesContent":["import React from \"react\";\nimport get from \"lodash/get.js\";\nimport { makeDecoratable } from \"@webiny/app-admin\";\nimport { i18n } from \"@webiny/app/i18n/index.js\";\nimport Label from \"./Label.js\";\nimport { useBind } from \"./useBind.js\";\nimport { useRenderPlugins } from \"./useRenderPlugins.js\";\nimport { ModelFieldProvider, useModelField } from \"../ModelFieldProvider/index.js\";\nimport type { BindComponent, CmsEditorContentModel, CmsModelField } from \"~/types/index.js\";\nimport { ErrorBoundary } from \"./ErrorBoundary.js\";\n\nconst t = i18n.ns(\"app-headless-cms/admin/components/content-form\");\n\ndeclare module \"react\" {\n namespace JSX {\n interface IntrinsicElements {\n \"hcms-model-field\": {\n \"data-id\": string;\n \"data-type\": string;\n \"data-field-id\": string;\n children: React.ReactNode;\n };\n }\n }\n}\n\ntype RenderFieldProps = Omit<FieldElementProps, \"field\">;\n\nconst RenderField = (props: RenderFieldProps) => {\n const renderPlugins = useRenderPlugins();\n const { Bind, contentModel } = props;\n const { field } = useModelField();\n const getBind = useBind({ Bind });\n\n if (typeof field.renderer === \"function\") {\n return <>{field.renderer({ field, getBind, Label, contentModel })}</>;\n }\n\n const fieldRendererName = get(field, \"renderer.name\");\n if (!fieldRendererName) {\n return t`Cannot render \"{fieldName}\" field - field renderer not defined.`({\n fieldName: <strong>{field.fieldId}</strong>\n });\n }\n\n const renderPlugin = renderPlugins.find(\n plugin => plugin.renderer.rendererName === fieldRendererName\n );\n\n if (!renderPlugin) {\n return t`Cannot render \"{fieldName}\" field - field renderer missing in Admin UI plugins.`({\n fieldName: <strong>{field.fieldId}</strong>\n });\n }\n\n return <>{renderPlugin.renderer.render({ field, getBind, Label, contentModel })}</>;\n};\n\nexport interface FieldElementProps {\n field: CmsModelField;\n Bind: BindComponent;\n contentModel: CmsEditorContentModel;\n}\n\nexport const FieldElement = makeDecoratable(\n \"FieldElement\",\n ({ field, ...props }: FieldElementProps) => {\n if (!field) {\n return null;\n }\n\n return (\n <hcms-model-field\n data-id={field.id}\n data-field-id={field.fieldId}\n data-type={field.type}\n >\n <ErrorBoundary field={field}>\n <ModelFieldProvider field={field}>\n <RenderField {...props} />\n </ModelFieldProvider>\n </ErrorBoundary>\n </hcms-model-field>\n );\n }\n);\n\n/**\n * @deprecated Use `FieldElement` instead.\n */\nexport const RenderFieldElement = FieldElement;\n\n/**\n * @deprecated Use `FieldElementProps` instead.\n */\nexport type RenderFieldElementProps = FieldElementProps;\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,OAAOC,GAAG,MAAM,eAAe;AAC/B,SAASC,eAAe,QAAQ,mBAAmB;AACnD,SAASC,IAAI,QAAQ,2BAA2B;AAChD,OAAOC,KAAK;AACZ,SAASC,OAAO;AAChB,SAASC,gBAAgB;AACzB,SAASC,kBAAkB,EAAEC,aAAa;AAE1C,SAASC,aAAa;AAEtB,MAAMC,CAAC,GAAGP,IAAI,CAACQ,EAAE,CAAC,gDAAgD,CAAC;AAiBnE,MAAMC,WAAW,GAAIC,KAAuB,IAAK;EAC7C,MAAMC,aAAa,GAAGR,gBAAgB,CAAC,CAAC;EACxC,MAAM;IAAES,IAAI;IAAEC;EAAa,CAAC,GAAGH,KAAK;EACpC,MAAM;IAAEI;EAAM,CAAC,GAAGT,aAAa,CAAC,CAAC;EACjC,MAAMU,OAAO,GAAGb,OAAO,CAAC;IAAEU;EAAK,CAAC,CAAC;EAEjC,IAAI,OAAOE,KAAK,CAACE,QAAQ,KAAK,UAAU,EAAE;IACtC,oBAAOnB,KAAA,CAAAoB,aAAA,CAAApB,KAAA,CAAAqB,QAAA,QAAGJ,KAAK,CAACE,QAAQ,CAAC;MAAEF,KAAK;MAAEC,OAAO;MAAEd,KAAK;MAAEY;IAAa,CAAC,CAAI,CAAC;EACzE;EAEA,MAAMM,iBAAiB,GAAGrB,GAAG,CAACgB,KAAK,EAAE,eAAe,CAAC;EACrD,IAAI,CAACK,iBAAiB,EAAE;IACpB,OAAOZ,CAAC,iEAAiE,CAAC;MACtEa,SAAS,eAAEvB,KAAA,CAAAoB,aAAA,iBAASH,KAAK,CAACO,OAAgB;IAC9C,CAAC,CAAC;EACN;EAEA,MAAMC,YAAY,GAAGX,aAAa,CAACY,IAAI,CACnCC,MAAM,IAAIA,MAAM,CAACR,QAAQ,CAACS,YAAY,KAAKN,iBAC/C,CAAC;EAED,IAAI,CAACG,YAAY,EAAE;IACf,OAAOf,CAAC,iFAAiF,CAAC;MACtFa,SAAS,eAAEvB,KAAA,CAAAoB,aAAA,iBAASH,KAAK,CAACO,OAAgB;IAC9C,CAAC,CAAC;EACN;EAEA,oBAAOxB,KAAA,CAAAoB,aAAA,CAAApB,KAAA,CAAAqB,QAAA,QAAGI,YAAY,CAACN,QAAQ,CAACU,MAAM,CAAC;IAAEZ,KAAK;IAAEC,OAAO;IAAEd,KAAK;IAAEY;EAAa,CAAC,CAAI,CAAC;AACvF,CAAC;AAQD,OAAO,MAAMc,YAAY,GAAG5B,eAAe,CACvC,cAAc,EACd,CAAC;EAAEe,KAAK;EAAE,GAAGJ;AAAyB,CAAC,KAAK;EACxC,IAAI,CAACI,KAAK,EAAE;IACR,OAAO,IAAI;EACf;EAEA,oBACIjB,KAAA,CAAAoB,aAAA;IACI,WAASH,KAAK,CAACc,EAAG;IAClB,iBAAed,KAAK,CAACO,OAAQ;IAC7B,aAAWP,KAAK,CAACe;EAAK,gBAEtBhC,KAAA,CAAAoB,aAAA,CAACX,aAAa;IAACQ,KAAK,EAAEA;EAAM,gBACxBjB,KAAA,CAAAoB,aAAA,CAACb,kBAAkB;IAACU,KAAK,EAAEA;EAAM,gBAC7BjB,KAAA,CAAAoB,aAAA,CAACR,WAAW,EAAKC,KAAQ,CACT,CACT,CACD,CAAC;AAE3B,CACJ,CAAC;;AAED;AACA;AACA;AACA,OAAO,MAAMoB,kBAAkB,GAAGH,YAAY;;AAE9C;AACA;AACA","ignoreList":[]}
|
|
@@ -11,8 +11,7 @@ const StyledError = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "produc
|
|
|
11
11
|
styles: "border:2px solid red;background-color:#f87e7e;border-radius:5px;padding:5px 10px"
|
|
12
12
|
} : {
|
|
13
13
|
name: "1ybf6v5",
|
|
14
|
-
styles: "border:2px solid red;background-color:#f87e7e;border-radius:5px;padding:5px 10px",
|
|
15
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkZpZWxkRWxlbWVudEVycm9yLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHOEIiLCJmaWxlIjoiRmllbGRFbGVtZW50RXJyb3IudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gXCJyZWFjdFwiO1xuaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5cbmNvbnN0IFN0eWxlZEVycm9yID0gc3R5bGVkLmRpdmBcbiAgICBib3JkZXI6IDJweCBzb2xpZCByZWQ7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI2Y4N2U3ZTtcbiAgICBib3JkZXItcmFkaXVzOiA1cHg7XG4gICAgcGFkZGluZzogNXB4IDEwcHg7XG5gO1xuXG5pbnRlcmZhY2UgRmllbGRFbGVtZW50RXJyb3JQcm9wcyB7XG4gICAgdGl0bGU6IHN0cmluZztcbiAgICBkZXNjcmlwdGlvbjogc3RyaW5nO1xufVxuXG5jb25zdCBzaG93RXJyb3IgPSBwcm9jZXNzLmVudi5OT0RFX0VOViA9PT0gXCJkZXZlbG9wbWVudFwiO1xuXG5leHBvcnQgY29uc3QgRmllbGRFbGVtZW50RXJyb3IgPSAocHJvcHM6IEZpZWxkRWxlbWVudEVycm9yUHJvcHMpID0+IHtcbiAgICBpZiAoIXNob3dFcnJvcikge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICByZXR1cm4gKFxuICAgICAgICA8U3R5bGVkRXJyb3I+XG4gICAgICAgICAgICA8aDU+e3Byb3BzLnRpdGxlfTwvaDU+XG4gICAgICAgICAgICA8cD57cHJvcHMuZGVzY3JpcHRpb259PC9wPlxuICAgICAgICA8L1N0eWxlZEVycm9yPlxuICAgICk7XG59O1xuIl19 */",
|
|
14
|
+
styles: "border:2px solid red;background-color:#f87e7e;border-radius:5px;padding:5px 10px/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkZpZWxkRWxlbWVudEVycm9yLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHOEIiLCJmaWxlIjoiRmllbGRFbGVtZW50RXJyb3IudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gXCJyZWFjdFwiO1xuaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5cbmNvbnN0IFN0eWxlZEVycm9yID0gc3R5bGVkLmRpdmBcbiAgICBib3JkZXI6IDJweCBzb2xpZCByZWQ7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI2Y4N2U3ZTtcbiAgICBib3JkZXItcmFkaXVzOiA1cHg7XG4gICAgcGFkZGluZzogNXB4IDEwcHg7XG5gO1xuXG5pbnRlcmZhY2UgRmllbGRFbGVtZW50RXJyb3JQcm9wcyB7XG4gICAgdGl0bGU6IHN0cmluZztcbiAgICBkZXNjcmlwdGlvbjogc3RyaW5nO1xufVxuXG5jb25zdCBzaG93RXJyb3IgPSBwcm9jZXNzLmVudi5OT0RFX0VOViA9PT0gXCJkZXZlbG9wbWVudFwiO1xuXG5leHBvcnQgY29uc3QgRmllbGRFbGVtZW50RXJyb3IgPSAocHJvcHM6IEZpZWxkRWxlbWVudEVycm9yUHJvcHMpID0+IHtcbiAgICBpZiAoIXNob3dFcnJvcikge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICByZXR1cm4gKFxuICAgICAgICA8U3R5bGVkRXJyb3I+XG4gICAgICAgICAgICA8aDU+e3Byb3BzLnRpdGxlfTwvaDU+XG4gICAgICAgICAgICA8cD57cHJvcHMuZGVzY3JpcHRpb259PC9wPlxuICAgICAgICA8L1N0eWxlZEVycm9yPlxuICAgICk7XG59O1xuIl19 */",
|
|
16
15
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
17
16
|
});
|
|
18
17
|
const showError = process.env.NODE_ENV === "development";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["React","StyledError","_styled","process","env","NODE_ENV","target","label","name","styles","
|
|
1
|
+
{"version":3,"names":["React","StyledError","_styled","process","env","NODE_ENV","target","label","name","styles","toString","_EMOTION_STRINGIFIED_CSS_ERROR__","showError","FieldElementError","props","createElement","title","description"],"sources":["FieldElementError.tsx"],"sourcesContent":["import React from \"react\";\nimport styled from \"@emotion/styled\";\n\nconst StyledError = styled.div`\n border: 2px solid red;\n background-color: #f87e7e;\n border-radius: 5px;\n padding: 5px 10px;\n`;\n\ninterface FieldElementErrorProps {\n title: string;\n description: string;\n}\n\nconst showError = process.env.NODE_ENV === \"development\";\n\nexport const FieldElementError = (props: FieldElementErrorProps) => {\n if (!showError) {\n return null;\n }\n\n return (\n <StyledError>\n <h5>{props.title}</h5>\n <p>{props.description}</p>\n </StyledError>\n );\n};\n"],"mappings":";;AAAA,OAAOA,KAAK,MAAM,OAAO;AAGzB,MAAMC,WAAW,gBAAAC,OAAA,QAAAC,OAAA,CAAAC,GAAA,CAAAC,QAAA;EAAAC,MAAA;AAAA;EAAAA,MAAA;EAAAC,KAAA;AAAA,GAAAJ,OAAA,CAAAC,GAAA,CAAAC,QAAA;EAAAG,IAAA;EAAAC,MAAA;AAAA;EAAAD,IAAA;EAAAC,MAAA;EAAAC,QAAA,EAAAC;AAAA,EAKhB;AAOD,MAAMC,SAAS,GAAGT,OAAO,CAACC,GAAG,CAACC,QAAQ,KAAK,aAAa;AAExD,OAAO,MAAMQ,iBAAiB,GAAIC,KAA6B,IAAK;EAChE,IAAI,CAACF,SAAS,EAAE;IACZ,OAAO,IAAI;EACf;EAEA,oBACIZ,KAAA,CAAAe,aAAA,CAACd,WAAW,qBACRD,KAAA,CAAAe,aAAA,aAAKD,KAAK,CAACE,KAAU,CAAC,eACtBhB,KAAA,CAAAe,aAAA,YAAID,KAAK,CAACG,WAAe,CAChB,CAAC;AAEtB,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import type { EffectiveFieldRules } from "./useFieldRules.js";
|
|
3
|
+
interface FieldRulesProviderProps {
|
|
4
|
+
rules: EffectiveFieldRules;
|
|
5
|
+
children: React.ReactNode;
|
|
6
|
+
}
|
|
7
|
+
export declare const FieldRulesProvider: ({ rules, children }: FieldRulesProviderProps) => React.JSX.Element;
|
|
8
|
+
export declare const useParentRules: () => EffectiveFieldRules;
|
|
9
|
+
export {};
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import React, { createContext, useContext } from "react";
|
|
2
|
+
const FieldRulesContext = /*#__PURE__*/createContext({
|
|
3
|
+
canView: true,
|
|
4
|
+
canEdit: true,
|
|
5
|
+
hidden: false,
|
|
6
|
+
disabled: false
|
|
7
|
+
});
|
|
8
|
+
export const FieldRulesProvider = ({
|
|
9
|
+
rules,
|
|
10
|
+
children
|
|
11
|
+
}) => {
|
|
12
|
+
return /*#__PURE__*/React.createElement(FieldRulesContext.Provider, {
|
|
13
|
+
value: rules
|
|
14
|
+
}, children);
|
|
15
|
+
};
|
|
16
|
+
export const useParentRules = () => {
|
|
17
|
+
return useContext(FieldRulesContext);
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
//# sourceMappingURL=FieldRulesProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["React","createContext","useContext","FieldRulesContext","canView","canEdit","hidden","disabled","FieldRulesProvider","rules","children","createElement","Provider","value","useParentRules"],"sources":["FieldRulesProvider.tsx"],"sourcesContent":["import React, { createContext, useContext } from \"react\";\nimport type { EffectiveFieldRules } from \"./useFieldRules.js\";\n\nconst FieldRulesContext = createContext<EffectiveFieldRules>({\n canView: true,\n canEdit: true,\n hidden: false,\n disabled: false\n});\n\ninterface FieldRulesProviderProps {\n rules: EffectiveFieldRules;\n children: React.ReactNode;\n}\n\nexport const FieldRulesProvider = ({ rules, children }: FieldRulesProviderProps) => {\n return <FieldRulesContext.Provider value={rules}>{children}</FieldRulesContext.Provider>;\n};\n\nexport const useParentRules = (): EffectiveFieldRules => {\n return useContext(FieldRulesContext);\n};\n"],"mappings":"AAAA,OAAOA,KAAK,IAAIC,aAAa,EAAEC,UAAU,QAAQ,OAAO;AAGxD,MAAMC,iBAAiB,gBAAGF,aAAa,CAAsB;EACzDG,OAAO,EAAE,IAAI;EACbC,OAAO,EAAE,IAAI;EACbC,MAAM,EAAE,KAAK;EACbC,QAAQ,EAAE;AACd,CAAC,CAAC;AAOF,OAAO,MAAMC,kBAAkB,GAAGA,CAAC;EAAEC,KAAK;EAAEC;AAAkC,CAAC,KAAK;EAChF,oBAAOV,KAAA,CAAAW,aAAA,CAACR,iBAAiB,CAACS,QAAQ;IAACC,KAAK,EAAEJ;EAAM,GAAEC,QAAqC,CAAC;AAC5F,CAAC;AAED,OAAO,MAAMI,cAAc,GAAGA,CAAA,KAA2B;EACrD,OAAOZ,UAAU,CAACC,iBAAiB,CAAC;AACxC,CAAC","ignoreList":[]}
|
package/Fields/Fields.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import type {
|
|
2
|
+
import type { BindComponent, CmsEditorContentModel, CmsEditorFieldsLayout, CmsModelField } from "../types/index.js";
|
|
3
3
|
interface FieldsProps {
|
|
4
4
|
Bind: BindComponent;
|
|
5
5
|
contentModel: CmsEditorContentModel;
|
package/Fields/Fields.js
CHANGED
|
@@ -1,10 +1,135 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import { Grid } from "@webiny/admin-ui";
|
|
3
|
-
import { FieldElement } from "./FieldElement";
|
|
4
|
-
import { FieldElementError } from "./FieldElementError";
|
|
2
|
+
import { Alert, Grid } from "@webiny/admin-ui";
|
|
3
|
+
import { FieldElement } from "./FieldElement.js";
|
|
4
|
+
import { FieldElementError } from "./FieldElementError.js";
|
|
5
|
+
import { isLayoutField } from "../types/model.js";
|
|
6
|
+
import { LayoutDescriptorCell } from "./LayoutDescriptorCell.js";
|
|
7
|
+
import { useAuthentication } from "@webiny/app-admin";
|
|
8
|
+
import { FieldRulesProvider, useParentRules } from "./FieldRulesProvider.js";
|
|
9
|
+
import { evaluateAccessControlRules, useFieldEffectiveRules } from "./useFieldRules.js";
|
|
5
10
|
const getFieldById = (fields, id) => {
|
|
6
11
|
return fields.find(field => field.id === id) || null;
|
|
7
12
|
};
|
|
13
|
+
const LayoutNotDefined = () => {
|
|
14
|
+
return /*#__PURE__*/React.createElement(Alert, {
|
|
15
|
+
type: "warning"
|
|
16
|
+
}, "You are missing the layout definition in your code content model. Please ensure you have the layout property correctly defined.");
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Renders a single layout descriptor cell with rules + permissions.
|
|
21
|
+
*/
|
|
22
|
+
|
|
23
|
+
const LayoutCell = ({
|
|
24
|
+
cell,
|
|
25
|
+
Bind,
|
|
26
|
+
fields,
|
|
27
|
+
contentModel,
|
|
28
|
+
gridClassName
|
|
29
|
+
}) => {
|
|
30
|
+
const isLayout = isLayoutField(cell);
|
|
31
|
+
const rules = useFieldEffectiveRules(isLayout ? cell : {});
|
|
32
|
+
if (!isLayout) {
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
if (!rules.canView || rules.hidden) {
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
return /*#__PURE__*/React.createElement(FieldRulesProvider, {
|
|
39
|
+
rules: rules
|
|
40
|
+
}, /*#__PURE__*/React.createElement(LayoutDescriptorCell, {
|
|
41
|
+
field: cell,
|
|
42
|
+
Bind: Bind,
|
|
43
|
+
fields: fields,
|
|
44
|
+
contentModel: contentModel,
|
|
45
|
+
gridClassName: gridClassName
|
|
46
|
+
}));
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Renders a single data field cell with rules + permissions.
|
|
51
|
+
*/
|
|
52
|
+
|
|
53
|
+
const FieldCell = ({
|
|
54
|
+
id,
|
|
55
|
+
field,
|
|
56
|
+
span,
|
|
57
|
+
Bind,
|
|
58
|
+
contentModel
|
|
59
|
+
}) => {
|
|
60
|
+
const rules = useFieldEffectiveRules(field ?? {});
|
|
61
|
+
if (!rules.canView || rules.hidden) {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
return /*#__PURE__*/React.createElement(Grid.Column, {
|
|
65
|
+
span: span
|
|
66
|
+
}, field ? /*#__PURE__*/React.createElement(FieldRulesProvider, {
|
|
67
|
+
rules: rules
|
|
68
|
+
}, /*#__PURE__*/React.createElement(FieldElement, {
|
|
69
|
+
field: field,
|
|
70
|
+
Bind: Bind,
|
|
71
|
+
contentModel: contentModel
|
|
72
|
+
})) : /*#__PURE__*/React.createElement(FieldElementError, {
|
|
73
|
+
title: `Missing field with id "${id}"!`,
|
|
74
|
+
description: "Make sure field layout contains the correct field ids (hint: check for typos)."
|
|
75
|
+
}));
|
|
76
|
+
};
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Renders a single row, delegating each cell to FieldCell or LayoutCell.
|
|
80
|
+
*/
|
|
81
|
+
|
|
82
|
+
const RowRenderer = ({
|
|
83
|
+
row,
|
|
84
|
+
fields,
|
|
85
|
+
Bind,
|
|
86
|
+
contentModel,
|
|
87
|
+
gridClassName
|
|
88
|
+
}) => {
|
|
89
|
+
const {
|
|
90
|
+
identity
|
|
91
|
+
} = useAuthentication();
|
|
92
|
+
const parentRules = useParentRules();
|
|
93
|
+
|
|
94
|
+
// Count visible string cells for column span calculation.
|
|
95
|
+
// This count is approximate based on access control rules only (not entry value rules),
|
|
96
|
+
// because entry value rules require per-field hooks and we can't call them in a filter.
|
|
97
|
+
// The actual visibility is enforced in FieldCell.
|
|
98
|
+
const visibleStringCells = row.filter(c => {
|
|
99
|
+
if (typeof c !== "string") {
|
|
100
|
+
return false;
|
|
101
|
+
}
|
|
102
|
+
const f = getFieldById(fields, c);
|
|
103
|
+
if (!f) {
|
|
104
|
+
return true;
|
|
105
|
+
}
|
|
106
|
+
const acPerms = evaluateAccessControlRules(f, identity);
|
|
107
|
+
return parentRules.canView && acPerms.canView;
|
|
108
|
+
});
|
|
109
|
+
const span = visibleStringCells.length > 0 ? Math.floor(12 / visibleStringCells.length) : 12;
|
|
110
|
+
return /*#__PURE__*/React.createElement(React.Fragment, null, row.map(cell => {
|
|
111
|
+
if (isLayoutField(cell)) {
|
|
112
|
+
return /*#__PURE__*/React.createElement(LayoutCell, {
|
|
113
|
+
key: cell.id,
|
|
114
|
+
cell: cell,
|
|
115
|
+
Bind: Bind,
|
|
116
|
+
fields: fields,
|
|
117
|
+
contentModel: contentModel,
|
|
118
|
+
gridClassName: gridClassName
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
const id = cell;
|
|
122
|
+
const field = getFieldById(fields, id);
|
|
123
|
+
return /*#__PURE__*/React.createElement(FieldCell, {
|
|
124
|
+
key: id,
|
|
125
|
+
id: id,
|
|
126
|
+
field: field,
|
|
127
|
+
span: span,
|
|
128
|
+
Bind: Bind,
|
|
129
|
+
contentModel: contentModel
|
|
130
|
+
});
|
|
131
|
+
}));
|
|
132
|
+
};
|
|
8
133
|
export const Fields = ({
|
|
9
134
|
Bind,
|
|
10
135
|
fields,
|
|
@@ -12,23 +137,19 @@ export const Fields = ({
|
|
|
12
137
|
contentModel,
|
|
13
138
|
gridClassName
|
|
14
139
|
}) => {
|
|
140
|
+
if (contentModel.plugin && fields.length > 0 && layout.length === 0) {
|
|
141
|
+
return /*#__PURE__*/React.createElement(LayoutNotDefined, null);
|
|
142
|
+
}
|
|
15
143
|
return /*#__PURE__*/React.createElement(Grid, {
|
|
16
144
|
className: gridClassName
|
|
17
145
|
}, layout.map((row, rowIndex) => /*#__PURE__*/React.createElement(React.Fragment, {
|
|
18
146
|
key: rowIndex
|
|
19
|
-
},
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
field: field,
|
|
26
|
-
Bind: Bind,
|
|
27
|
-
contentModel: contentModel
|
|
28
|
-
}) : /*#__PURE__*/React.createElement(FieldElementError, {
|
|
29
|
-
title: `Missing field with id "${id}"!`,
|
|
30
|
-
description: "Make sure field layout contains the correct field ids (hint: check for typos)."
|
|
31
|
-
}));
|
|
147
|
+
}, /*#__PURE__*/React.createElement(RowRenderer, {
|
|
148
|
+
row: row,
|
|
149
|
+
fields: fields,
|
|
150
|
+
Bind: Bind,
|
|
151
|
+
contentModel: contentModel,
|
|
152
|
+
gridClassName: gridClassName
|
|
32
153
|
}))));
|
|
33
154
|
};
|
|
34
155
|
|
package/Fields/Fields.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["React","Grid","FieldElement","FieldElementError","getFieldById","fields","id","find","field","Fields","Bind","layout","contentModel","gridClassName","createElement","className","map","row","rowIndex","Fragment","key","Column","span","Math","floor","length","title","description"],"sources":["Fields.tsx"],"sourcesContent":["import React from \"react\";\nimport { Grid, type ColumnProps } from \"@webiny/admin-ui\";\nimport { FieldElement } from \"./FieldElement\";\nimport { FieldElementError } from \"./FieldElementError\";\nimport type {\n CmsEditorContentModel,\n CmsModelField,\n CmsEditorFieldsLayout,\n BindComponent\n} from \"~/types\";\n\ninterface FieldsProps {\n Bind: BindComponent;\n contentModel: CmsEditorContentModel;\n fields: CmsModelField[];\n layout: CmsEditorFieldsLayout;\n gridClassName?: string;\n}\n\nconst getFieldById = (fields: CmsModelField[], id: string): CmsModelField | null => {\n return fields.find(field => field.id === id) || null;\n};\n\nexport const Fields = ({ Bind, fields, layout, contentModel, gridClassName }: FieldsProps) => {\n return (\n <Grid className={gridClassName}>\n {layout.map((row, rowIndex) => (\n <React.Fragment key={rowIndex}>\n {row.map(id => {\n const field = getFieldById(fields, id) as CmsModelField;\n\n return (\n <Grid.Column\n span={Math.floor(12 / row.length) as ColumnProps[\"span\"]}\n key={id}\n >\n {field ? (\n <FieldElement\n field={field}\n Bind={Bind}\n contentModel={contentModel}\n />\n ) : (\n <FieldElementError\n title={`Missing field with id \"${id}\"!`}\n description={\n \"Make sure field layout contains the correct field ids (hint: check for typos).\"\n }\n />\n )}\n </Grid.Column>\n );\n })}\n </React.Fragment>\n ))}\n </Grid>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,SAASC,IAAI,QAA0B,kBAAkB;AACzD,SAASC,YAAY;AACrB,SAASC,iBAAiB;AAgB1B,MAAMC,YAAY,GAAGA,CAACC,MAAuB,EAAEC,EAAU,KAA2B;EAChF,OAAOD,MAAM,CAACE,IAAI,CAACC,KAAK,IAAIA,KAAK,CAACF,EAAE,KAAKA,EAAE,CAAC,IAAI,IAAI;AACxD,CAAC;AAED,OAAO,MAAMG,MAAM,GAAGA,CAAC;EAAEC,IAAI;EAAEL,MAAM;EAAEM,MAAM;EAAEC,YAAY;EAAEC;AAA2B,CAAC,KAAK;EAC1F,oBACIb,KAAA,CAAAc,aAAA,CAACb,IAAI;IAACc,SAAS,EAAEF;EAAc,GAC1BF,MAAM,CAACK,GAAG,CAAC,CAACC,GAAG,EAAEC,QAAQ,kBACtBlB,KAAA,CAAAc,aAAA,CAACd,KAAK,CAACmB,QAAQ;IAACC,GAAG,EAAEF;EAAS,GACzBD,GAAG,CAACD,GAAG,CAACV,EAAE,IAAI;IACX,MAAME,KAAK,GAAGJ,YAAY,CAACC,MAAM,EAAEC,EAAE,CAAkB;IAEvD,oBACIN,KAAA,CAAAc,aAAA,CAACb,IAAI,CAACoB,MAAM;MACRC,IAAI,EAAEC,IAAI,CAACC,KAAK,CAAC,EAAE,GAAGP,GAAG,CAACQ,MAAM,CAAyB;MACzDL,GAAG,EAAEd;IAAG,GAEPE,KAAK,gBACFR,KAAA,CAAAc,aAAA,CAACZ,YAAY;MACTM,KAAK,EAAEA,KAAM;MACbE,IAAI,EAAEA,IAAK;MACXE,YAAY,EAAEA;IAAa,CAC9B,CAAC,gBAEFZ,KAAA,CAAAc,aAAA,CAACX,iBAAiB;MACduB,KAAK,EAAE,0BAA0BpB,EAAE,IAAK;MACxCqB,WAAW,EACP;IACH,CACJ,CAEI,CAAC;EAEtB,CAAC,CACW,CACnB,CACC,CAAC;AAEf,CAAC","ignoreList":[]}
|
|
1
|
+
{"version":3,"names":["React","Alert","Grid","FieldElement","FieldElementError","isLayoutField","LayoutDescriptorCell","useAuthentication","FieldRulesProvider","useParentRules","evaluateAccessControlRules","useFieldEffectiveRules","getFieldById","fields","id","find","field","LayoutNotDefined","createElement","type","LayoutCell","cell","Bind","contentModel","gridClassName","isLayout","rules","canView","hidden","FieldCell","span","Column","title","description","RowRenderer","row","identity","parentRules","visibleStringCells","filter","c","f","acPerms","length","Math","floor","Fragment","map","key","Fields","layout","plugin","className","rowIndex"],"sources":["Fields.tsx"],"sourcesContent":["import React from \"react\";\nimport { Alert, type ColumnProps, Grid } from \"@webiny/admin-ui\";\nimport { FieldElement } from \"./FieldElement.js\";\nimport { FieldElementError } from \"./FieldElementError.js\";\nimport type {\n BindComponent,\n CmsEditorContentModel,\n CmsEditorFieldsLayout,\n CmsModelField\n} from \"~/types/index.js\";\nimport type { CmsEditorLayoutCell } from \"~/types/model.js\";\nimport { isLayoutField } from \"~/types/model.js\";\nimport { LayoutDescriptorCell } from \"./LayoutDescriptorCell.js\";\nimport { useAuthentication } from \"@webiny/app-admin\";\nimport { FieldRulesProvider, useParentRules } from \"./FieldRulesProvider.js\";\nimport { evaluateAccessControlRules, useFieldEffectiveRules } from \"./useFieldRules.js\";\n\ninterface FieldsProps {\n Bind: BindComponent;\n contentModel: CmsEditorContentModel;\n fields: CmsModelField[];\n layout: CmsEditorFieldsLayout;\n gridClassName?: string;\n}\n\nconst getFieldById = (fields: CmsModelField[], id: string): CmsModelField | null => {\n return fields.find(field => field.id === id) || null;\n};\n\nconst LayoutNotDefined = () => {\n return (\n <Alert type={\"warning\"}>\n You are missing the layout definition in your code content model. Please ensure you have\n the layout property correctly defined.\n </Alert>\n );\n};\n\n/**\n * Renders a single layout descriptor cell with rules + permissions.\n */\ninterface LayoutCellProps {\n cell: CmsEditorLayoutCell;\n Bind: BindComponent;\n fields: CmsModelField[];\n contentModel: CmsEditorContentModel;\n gridClassName?: string;\n}\n\nconst LayoutCell = ({ cell, Bind, fields, contentModel, gridClassName }: LayoutCellProps) => {\n const isLayout = isLayoutField(cell);\n const rules = useFieldEffectiveRules(isLayout ? cell : {});\n\n if (!isLayout) {\n return null;\n }\n\n if (!rules.canView || rules.hidden) {\n return null;\n }\n\n return (\n <FieldRulesProvider rules={rules}>\n <LayoutDescriptorCell\n field={cell}\n Bind={Bind}\n fields={fields}\n contentModel={contentModel}\n gridClassName={gridClassName}\n />\n </FieldRulesProvider>\n );\n};\n\n/**\n * Renders a single data field cell with rules + permissions.\n */\ninterface FieldCellProps {\n id: string;\n field: CmsModelField | null;\n span: ColumnProps[\"span\"];\n Bind: BindComponent;\n contentModel: CmsEditorContentModel;\n}\n\nconst FieldCell = ({ id, field, span, Bind, contentModel }: FieldCellProps) => {\n const rules = useFieldEffectiveRules(field ?? {});\n\n if (!rules.canView || rules.hidden) {\n return null;\n }\n\n return (\n <Grid.Column span={span}>\n {field ? (\n <FieldRulesProvider rules={rules}>\n <FieldElement field={field} Bind={Bind} contentModel={contentModel} />\n </FieldRulesProvider>\n ) : (\n <FieldElementError\n title={`Missing field with id \"${id}\"!`}\n description={\n \"Make sure field layout contains the correct field ids (hint: check for typos).\"\n }\n />\n )}\n </Grid.Column>\n );\n};\n\n/**\n * Renders a single row, delegating each cell to FieldCell or LayoutCell.\n */\ninterface RowRendererProps {\n row: CmsEditorLayoutCell[];\n fields: CmsModelField[];\n Bind: BindComponent;\n contentModel: CmsEditorContentModel;\n gridClassName?: string;\n}\n\nconst RowRenderer = ({ row, fields, Bind, contentModel, gridClassName }: RowRendererProps) => {\n const { identity } = useAuthentication();\n const parentRules = useParentRules();\n\n // Count visible string cells for column span calculation.\n // This count is approximate based on access control rules only (not entry value rules),\n // because entry value rules require per-field hooks and we can't call them in a filter.\n // The actual visibility is enforced in FieldCell.\n const visibleStringCells = row.filter(c => {\n if (typeof c !== \"string\") {\n return false;\n }\n const f = getFieldById(fields, c);\n if (!f) {\n return true;\n }\n const acPerms = evaluateAccessControlRules(f, identity);\n return parentRules.canView && acPerms.canView;\n });\n\n const span =\n visibleStringCells.length > 0\n ? (Math.floor(12 / visibleStringCells.length) as ColumnProps[\"span\"])\n : (12 as ColumnProps[\"span\"]);\n\n return (\n <>\n {row.map(cell => {\n if (isLayoutField(cell)) {\n return (\n <LayoutCell\n key={cell.id}\n cell={cell}\n Bind={Bind}\n fields={fields}\n contentModel={contentModel}\n gridClassName={gridClassName}\n />\n );\n }\n\n const id = cell as string;\n const field = getFieldById(fields, id);\n\n return (\n <FieldCell\n key={id}\n id={id}\n field={field}\n span={span}\n Bind={Bind}\n contentModel={contentModel}\n />\n );\n })}\n </>\n );\n};\n\nexport const Fields = ({ Bind, fields, layout, contentModel, gridClassName }: FieldsProps) => {\n if (contentModel.plugin && fields.length > 0 && layout.length === 0) {\n return <LayoutNotDefined />;\n }\n\n return (\n <Grid className={gridClassName}>\n {layout.map((row, rowIndex) => (\n <React.Fragment key={rowIndex}>\n <RowRenderer\n row={row}\n fields={fields}\n Bind={Bind}\n contentModel={contentModel}\n gridClassName={gridClassName}\n />\n </React.Fragment>\n ))}\n </Grid>\n );\n};\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import type { CmsLayoutField } from "../types/model.js";
|
|
3
|
+
import type { BindComponent, CmsEditorContentModel, CmsModelField } from "../types/index.js";
|
|
4
|
+
interface LayoutDescriptorCellProps {
|
|
5
|
+
field: CmsLayoutField;
|
|
6
|
+
Bind: BindComponent;
|
|
7
|
+
fields: CmsModelField[];
|
|
8
|
+
contentModel: CmsEditorContentModel;
|
|
9
|
+
gridClassName?: string;
|
|
10
|
+
}
|
|
11
|
+
export declare const LayoutDescriptorCell: ({ field, Bind, fields, contentModel, gridClassName }: LayoutDescriptorCellProps) => React.JSX.Element | null;
|
|
12
|
+
export {};
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { plugins } from "@webiny/plugins";
|
|
3
|
+
import { SeparatorFieldRenderer } from "./layoutFieldRenderers/SeparatorFieldRenderer.js";
|
|
4
|
+
import { AlertFieldRenderer } from "./layoutFieldRenderers/AlertFieldRenderer.js";
|
|
5
|
+
import { TabsFieldRenderer } from "./layoutFieldRenderers/TabsFieldRenderer.js";
|
|
6
|
+
export const LayoutDescriptorCell = ({
|
|
7
|
+
field,
|
|
8
|
+
Bind,
|
|
9
|
+
fields,
|
|
10
|
+
contentModel,
|
|
11
|
+
gridClassName
|
|
12
|
+
}) => {
|
|
13
|
+
switch (field.type) {
|
|
14
|
+
case "separator":
|
|
15
|
+
return /*#__PURE__*/React.createElement(SeparatorFieldRenderer, {
|
|
16
|
+
field: field
|
|
17
|
+
});
|
|
18
|
+
case "alert":
|
|
19
|
+
return /*#__PURE__*/React.createElement(AlertFieldRenderer, {
|
|
20
|
+
field: field
|
|
21
|
+
});
|
|
22
|
+
case "tabs":
|
|
23
|
+
return /*#__PURE__*/React.createElement(TabsFieldRenderer, {
|
|
24
|
+
field: field,
|
|
25
|
+
Bind: Bind,
|
|
26
|
+
fields: fields,
|
|
27
|
+
contentModel: contentModel,
|
|
28
|
+
gridClassName: gridClassName
|
|
29
|
+
});
|
|
30
|
+
default:
|
|
31
|
+
{
|
|
32
|
+
const rendererPlugin = plugins.byType("cms-layout-field-renderer").find(p => p.fieldType === field.type);
|
|
33
|
+
if (rendererPlugin) {
|
|
34
|
+
return rendererPlugin.render({
|
|
35
|
+
field,
|
|
36
|
+
Bind,
|
|
37
|
+
fields,
|
|
38
|
+
contentModel,
|
|
39
|
+
gridClassName
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
return null;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
//# sourceMappingURL=LayoutDescriptorCell.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["React","plugins","SeparatorFieldRenderer","AlertFieldRenderer","TabsFieldRenderer","LayoutDescriptorCell","field","Bind","fields","contentModel","gridClassName","type","createElement","rendererPlugin","byType","find","p","fieldType","render"],"sources":["LayoutDescriptorCell.tsx"],"sourcesContent":["import React from \"react\";\nimport type {\n CmsAlertLayoutField,\n CmsLayoutField,\n CmsSeparatorLayoutField,\n CmsTabLayoutField\n} from \"~/types/model.js\";\nimport type {\n BindComponent,\n CmsEditorContentModel,\n CmsModelLayoutFieldRendererPlugin,\n CmsModelField\n} from \"~/types/index.js\";\nimport { plugins } from \"@webiny/plugins\";\nimport { SeparatorFieldRenderer } from \"./layoutFieldRenderers/SeparatorFieldRenderer.js\";\nimport { AlertFieldRenderer } from \"./layoutFieldRenderers/AlertFieldRenderer.js\";\nimport { TabsFieldRenderer } from \"./layoutFieldRenderers/TabsFieldRenderer.js\";\n\ninterface LayoutDescriptorCellProps {\n field: CmsLayoutField;\n Bind: BindComponent;\n fields: CmsModelField[];\n contentModel: CmsEditorContentModel;\n gridClassName?: string;\n}\n\nexport const LayoutDescriptorCell = ({\n field,\n Bind,\n fields,\n contentModel,\n gridClassName\n}: LayoutDescriptorCellProps) => {\n switch (field.type) {\n case \"separator\":\n return <SeparatorFieldRenderer field={field as CmsSeparatorLayoutField} />;\n case \"alert\":\n return <AlertFieldRenderer field={field as CmsAlertLayoutField} />;\n case \"tabs\":\n return (\n <TabsFieldRenderer\n field={field as CmsTabLayoutField}\n Bind={Bind}\n fields={fields}\n contentModel={contentModel}\n gridClassName={gridClassName}\n />\n );\n default: {\n const rendererPlugin = plugins\n .byType<CmsModelLayoutFieldRendererPlugin>(\"cms-layout-field-renderer\")\n .find(p => p.fieldType === field.type);\n if (rendererPlugin) {\n return rendererPlugin.render({\n field,\n Bind,\n fields,\n contentModel,\n gridClassName\n });\n }\n return null;\n }\n }\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AAazB,SAASC,OAAO,QAAQ,iBAAiB;AACzC,SAASC,sBAAsB;AAC/B,SAASC,kBAAkB;AAC3B,SAASC,iBAAiB;AAU1B,OAAO,MAAMC,oBAAoB,GAAGA,CAAC;EACjCC,KAAK;EACLC,IAAI;EACJC,MAAM;EACNC,YAAY;EACZC;AACuB,CAAC,KAAK;EAC7B,QAAQJ,KAAK,CAACK,IAAI;IACd,KAAK,WAAW;MACZ,oBAAOX,KAAA,CAAAY,aAAA,CAACV,sBAAsB;QAACI,KAAK,EAAEA;MAAiC,CAAE,CAAC;IAC9E,KAAK,OAAO;MACR,oBAAON,KAAA,CAAAY,aAAA,CAACT,kBAAkB;QAACG,KAAK,EAAEA;MAA6B,CAAE,CAAC;IACtE,KAAK,MAAM;MACP,oBACIN,KAAA,CAAAY,aAAA,CAACR,iBAAiB;QACdE,KAAK,EAAEA,KAA2B;QAClCC,IAAI,EAAEA,IAAK;QACXC,MAAM,EAAEA,MAAO;QACfC,YAAY,EAAEA,YAAa;QAC3BC,aAAa,EAAEA;MAAc,CAChC,CAAC;IAEV;MAAS;QACL,MAAMG,cAAc,GAAGZ,OAAO,CACzBa,MAAM,CAAoC,2BAA2B,CAAC,CACtEC,IAAI,CAACC,CAAC,IAAIA,CAAC,CAACC,SAAS,KAAKX,KAAK,CAACK,IAAI,CAAC;QAC1C,IAAIE,cAAc,EAAE;UAChB,OAAOA,cAAc,CAACK,MAAM,CAAC;YACzBZ,KAAK;YACLC,IAAI;YACJC,MAAM;YACNC,YAAY;YACZC;UACJ,CAAC,CAAC;QACN;QACA,OAAO,IAAI;MACf;EACJ;AACJ,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export type Operator = "==" | "!=" | ">" | "<" | ">=" | "<=" | "contains" | "startsWith" | "endsWith" | "isEmpty" | "isNotEmpty";
|
|
2
|
+
export interface ParsedExpression {
|
|
3
|
+
target: string;
|
|
4
|
+
operator: Operator;
|
|
5
|
+
value: string | number | boolean | null;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Resolve a fieldPath that may contain `$` placeholders using the bindParentName.
|
|
9
|
+
*
|
|
10
|
+
* The bindParentName gives us the concrete indices for the current list context.
|
|
11
|
+
* For example, if fieldPath is `panorama.hotspots.$.title` and bindParentName
|
|
12
|
+
* is `panorama.hotspots.2`, the result is `panorama.hotspots.2.title`.
|
|
13
|
+
*
|
|
14
|
+
* Supports multiple nested `$` segments for deeply nested list objects.
|
|
15
|
+
*/
|
|
16
|
+
export declare function resolveFieldPath(fieldPath: string, bindParentName?: string): string;
|
|
17
|
+
/**
|
|
18
|
+
* Evaluate a parsed expression against form values.
|
|
19
|
+
*
|
|
20
|
+
* If the resolved fieldPath ends with `.length`, it reads the array at the parent
|
|
21
|
+
* path and uses its length as the left-hand value.
|
|
22
|
+
*/
|
|
23
|
+
export declare function evaluateExpression(parsed: ParsedExpression, getFormValue: (path: string) => unknown): boolean;
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Resolve a fieldPath that may contain `$` placeholders using the bindParentName.
|
|
3
|
+
*
|
|
4
|
+
* The bindParentName gives us the concrete indices for the current list context.
|
|
5
|
+
* For example, if fieldPath is `panorama.hotspots.$.title` and bindParentName
|
|
6
|
+
* is `panorama.hotspots.2`, the result is `panorama.hotspots.2.title`.
|
|
7
|
+
*
|
|
8
|
+
* Supports multiple nested `$` segments for deeply nested list objects.
|
|
9
|
+
*/
|
|
10
|
+
export function resolveFieldPath(fieldPath, bindParentName) {
|
|
11
|
+
if (!fieldPath.includes("$")) {
|
|
12
|
+
return fieldPath;
|
|
13
|
+
}
|
|
14
|
+
if (!bindParentName) {
|
|
15
|
+
// No parent context, replace $ with 0 as fallback
|
|
16
|
+
return fieldPath.replace(/\$/g, "0");
|
|
17
|
+
}
|
|
18
|
+
const pathParts = fieldPath.split(".");
|
|
19
|
+
const parentParts = bindParentName.split(".");
|
|
20
|
+
|
|
21
|
+
// Build a map of prefix -> index from the parent path.
|
|
22
|
+
// E.g., parentName "panorama.hotspots.2.items.1" gives us:
|
|
23
|
+
// "panorama.hotspots" -> "2"
|
|
24
|
+
// "panorama.hotspots.2.items" -> "1"
|
|
25
|
+
const indexMap = new Map();
|
|
26
|
+
const prefixParts = [];
|
|
27
|
+
for (const part of parentParts) {
|
|
28
|
+
if (/^\d+$/.test(part)) {
|
|
29
|
+
indexMap.set(prefixParts.join("."), part);
|
|
30
|
+
} else {
|
|
31
|
+
prefixParts.push(part);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// Walk through pathParts and replace each `$` with the corresponding index
|
|
36
|
+
const resolved = [];
|
|
37
|
+
const currentPrefix = [];
|
|
38
|
+
for (const part of pathParts) {
|
|
39
|
+
if (part === "$") {
|
|
40
|
+
const key = currentPrefix.join(".");
|
|
41
|
+
const index = indexMap.get(key) ?? "0";
|
|
42
|
+
resolved.push(index);
|
|
43
|
+
} else {
|
|
44
|
+
currentPrefix.push(part);
|
|
45
|
+
resolved.push(part);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
return resolved.join(".");
|
|
49
|
+
}
|
|
50
|
+
function compareValues(operator, val, rhs) {
|
|
51
|
+
switch (operator) {
|
|
52
|
+
case "==":
|
|
53
|
+
return val == rhs;
|
|
54
|
+
case "!=":
|
|
55
|
+
return val != rhs;
|
|
56
|
+
case ">":
|
|
57
|
+
return Number(val) > Number(rhs);
|
|
58
|
+
case "<":
|
|
59
|
+
return Number(val) < Number(rhs);
|
|
60
|
+
case ">=":
|
|
61
|
+
return Number(val) >= Number(rhs);
|
|
62
|
+
case "<=":
|
|
63
|
+
return Number(val) <= Number(rhs);
|
|
64
|
+
case "contains":
|
|
65
|
+
return String(val ?? "").includes(String(rhs ?? ""));
|
|
66
|
+
case "startsWith":
|
|
67
|
+
return String(val ?? "").startsWith(String(rhs ?? ""));
|
|
68
|
+
case "endsWith":
|
|
69
|
+
return String(val ?? "").endsWith(String(rhs ?? ""));
|
|
70
|
+
case "isEmpty":
|
|
71
|
+
return val == null || val === "" || Array.isArray(val) && val.length === 0;
|
|
72
|
+
case "isNotEmpty":
|
|
73
|
+
return val != null && val !== "" && !(Array.isArray(val) && val.length === 0);
|
|
74
|
+
default:
|
|
75
|
+
return false;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Evaluate a parsed expression against form values.
|
|
81
|
+
*
|
|
82
|
+
* If the resolved fieldPath ends with `.length`, it reads the array at the parent
|
|
83
|
+
* path and uses its length as the left-hand value.
|
|
84
|
+
*/
|
|
85
|
+
export function evaluateExpression(parsed, getFormValue) {
|
|
86
|
+
const {
|
|
87
|
+
target,
|
|
88
|
+
operator,
|
|
89
|
+
value: rhs
|
|
90
|
+
} = parsed;
|
|
91
|
+
let val;
|
|
92
|
+
if (target.endsWith(".length")) {
|
|
93
|
+
const arrayPath = target.slice(0, -".length".length);
|
|
94
|
+
const arr = getFormValue(arrayPath);
|
|
95
|
+
val = Array.isArray(arr) ? arr.length : 0;
|
|
96
|
+
} else {
|
|
97
|
+
val = getFormValue(target);
|
|
98
|
+
}
|
|
99
|
+
return compareValues(operator, val, rhs);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
//# sourceMappingURL=evaluateExpression.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["resolveFieldPath","fieldPath","bindParentName","includes","replace","pathParts","split","parentParts","indexMap","Map","prefixParts","part","test","set","join","push","resolved","currentPrefix","key","index","get","compareValues","operator","val","rhs","Number","String","startsWith","endsWith","Array","isArray","length","evaluateExpression","parsed","getFormValue","target","value","arrayPath","slice","arr"],"sources":["evaluateExpression.ts"],"sourcesContent":["export type Operator =\n | \"==\"\n | \"!=\"\n | \">\"\n | \"<\"\n | \">=\"\n | \"<=\"\n | \"contains\"\n | \"startsWith\"\n | \"endsWith\"\n | \"isEmpty\"\n | \"isNotEmpty\";\n\nexport interface ParsedExpression {\n target: string;\n operator: Operator;\n value: string | number | boolean | null;\n}\n\n/**\n * Resolve a fieldPath that may contain `$` placeholders using the bindParentName.\n *\n * The bindParentName gives us the concrete indices for the current list context.\n * For example, if fieldPath is `panorama.hotspots.$.title` and bindParentName\n * is `panorama.hotspots.2`, the result is `panorama.hotspots.2.title`.\n *\n * Supports multiple nested `$` segments for deeply nested list objects.\n */\nexport function resolveFieldPath(fieldPath: string, bindParentName?: string): string {\n if (!fieldPath.includes(\"$\")) {\n return fieldPath;\n }\n\n if (!bindParentName) {\n // No parent context, replace $ with 0 as fallback\n return fieldPath.replace(/\\$/g, \"0\");\n }\n\n const pathParts = fieldPath.split(\".\");\n const parentParts = bindParentName.split(\".\");\n\n // Build a map of prefix -> index from the parent path.\n // E.g., parentName \"panorama.hotspots.2.items.1\" gives us:\n // \"panorama.hotspots\" -> \"2\"\n // \"panorama.hotspots.2.items\" -> \"1\"\n const indexMap = new Map<string, string>();\n const prefixParts: string[] = [];\n for (const part of parentParts) {\n if (/^\\d+$/.test(part)) {\n indexMap.set(prefixParts.join(\".\"), part);\n } else {\n prefixParts.push(part);\n }\n }\n\n // Walk through pathParts and replace each `$` with the corresponding index\n const resolved: string[] = [];\n const currentPrefix: string[] = [];\n for (const part of pathParts) {\n if (part === \"$\") {\n const key = currentPrefix.join(\".\");\n const index = indexMap.get(key) ?? \"0\";\n resolved.push(index);\n } else {\n currentPrefix.push(part);\n resolved.push(part);\n }\n }\n\n return resolved.join(\".\");\n}\n\nfunction compareValues(\n operator: string,\n val: unknown,\n rhs: string | number | boolean | null\n): boolean {\n switch (operator) {\n case \"==\":\n return val == rhs;\n case \"!=\":\n return val != rhs;\n case \">\":\n return Number(val) > Number(rhs);\n case \"<\":\n return Number(val) < Number(rhs);\n case \">=\":\n return Number(val) >= Number(rhs);\n case \"<=\":\n return Number(val) <= Number(rhs);\n case \"contains\":\n return String(val ?? \"\").includes(String(rhs ?? \"\"));\n case \"startsWith\":\n return String(val ?? \"\").startsWith(String(rhs ?? \"\"));\n case \"endsWith\":\n return String(val ?? \"\").endsWith(String(rhs ?? \"\"));\n case \"isEmpty\":\n return val == null || val === \"\" || (Array.isArray(val) && val.length === 0);\n case \"isNotEmpty\":\n return val != null && val !== \"\" && !(Array.isArray(val) && val.length === 0);\n default:\n return false;\n }\n}\n\n/**\n * Evaluate a parsed expression against form values.\n *\n * If the resolved fieldPath ends with `.length`, it reads the array at the parent\n * path and uses its length as the left-hand value.\n */\nexport function evaluateExpression(\n parsed: ParsedExpression,\n getFormValue: (path: string) => unknown\n): boolean {\n const { target, operator, value: rhs } = parsed;\n\n let val: unknown;\n if (target.endsWith(\".length\")) {\n const arrayPath = target.slice(0, -\".length\".length);\n const arr = getFormValue(arrayPath);\n val = Array.isArray(arr) ? arr.length : 0;\n } else {\n val = getFormValue(target);\n }\n\n return compareValues(operator, val, rhs);\n}\n"],"mappings":"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","ignoreList":[]}
|