@webiny/react-properties 0.0.0-unstable.79032b23a5 → 0.0.0-unstable.7be00a75a9

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.
@@ -0,0 +1,40 @@
1
+ type DevToolsView = "browse" | "raw";
2
+ interface DevToolsSectionProps {
3
+ name: string;
4
+ data: unknown;
5
+ /**
6
+ * Group name for the sidebar. Items with the same group appear
7
+ * under the same header. Defaults to "Sections".
8
+ *
9
+ * @example
10
+ * ```tsx
11
+ * <DevToolsSection name="Article" group="CMS" data={model} />
12
+ * <DevToolsSection name="Author" group="CMS" data={author} />
13
+ * ```
14
+ */
15
+ group?: string;
16
+ /**
17
+ * Which views to show in the detail panel.
18
+ * - `"browse"` — split view with root keys on the left, value tree on the right
19
+ * - `"raw"` — full JSON tree view
20
+ *
21
+ * Accepts a single view or an array. First item is the default tab.
22
+ * @default ["browse", "raw"]
23
+ */
24
+ views?: DevToolsView | DevToolsView[];
25
+ }
26
+ /**
27
+ * Registers a named section in the Webiny DevTools extension.
28
+ * Renders nothing — purely a data registration side-effect.
29
+ *
30
+ * When the component unmounts (e.g., route change), the section
31
+ * is automatically removed from DevTools.
32
+ *
33
+ * @example
34
+ * ```tsx
35
+ * <DevToolsSection name="CMS Model" data={model} />
36
+ * <DevToolsSection name="Article" group="CMS" data={model} views="raw" />
37
+ * ```
38
+ */
39
+ export declare function DevToolsSection({ name, data, group, views }: DevToolsSectionProps): null;
40
+ export {};
@@ -0,0 +1,54 @@
1
+ import { useEffect } from "react";
2
+ import { getHook } from "./useDebugConfig.js";
3
+ function DevToolsSection({ name, data, group, views }) {
4
+ const dataKey = safeStringify(data);
5
+ useEffect(()=>{
6
+ if ("development" !== process.env.NODE_ENV) return;
7
+ const normalizedViews = views ? Array.isArray(views) ? views : [
8
+ views
9
+ ] : [
10
+ "browse",
11
+ "raw"
12
+ ];
13
+ let plainData;
14
+ try {
15
+ plainData = JSON.parse(dataKey);
16
+ } catch {
17
+ plainData = data;
18
+ }
19
+ const hook = getHook();
20
+ hook.sections[name] = {
21
+ data: plainData,
22
+ group: group || "Sections",
23
+ views: normalizedViews,
24
+ updatedAt: Date.now()
25
+ };
26
+ hook.revision++;
27
+ return ()=>{
28
+ if (window.__WEBINY_DEVTOOLS_HOOK__) {
29
+ delete window.__WEBINY_DEVTOOLS_HOOK__.sections[name];
30
+ window.__WEBINY_DEVTOOLS_HOOK__.revision++;
31
+ }
32
+ };
33
+ }, [
34
+ name,
35
+ dataKey,
36
+ group,
37
+ Array.isArray(views) ? views.join(",") : views
38
+ ]);
39
+ return null;
40
+ }
41
+ function safeStringify(value) {
42
+ try {
43
+ return JSON.stringify(value, (_key, v)=>{
44
+ if ("function" == typeof v) return `[Function: ${v.name || "anonymous"}]`;
45
+ if (void 0 === v) return "[undefined]";
46
+ return v;
47
+ });
48
+ } catch {
49
+ return String(value);
50
+ }
51
+ }
52
+ export { DevToolsSection };
53
+
54
+ //# sourceMappingURL=DevToolsSection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevToolsSection.js","sources":["../src/DevToolsSection.tsx"],"sourcesContent":["import { useEffect } from \"react\";\nimport { getHook } from \"./useDebugConfig.js\";\n\ntype DevToolsView = \"browse\" | \"raw\";\n\ninterface DevToolsSectionProps {\n name: string;\n data: unknown;\n /**\n * Group name for the sidebar. Items with the same group appear\n * under the same header. Defaults to \"Sections\".\n *\n * @example\n * ```tsx\n * <DevToolsSection name=\"Article\" group=\"CMS\" data={model} />\n * <DevToolsSection name=\"Author\" group=\"CMS\" data={author} />\n * ```\n */\n group?: string;\n /**\n * Which views to show in the detail panel.\n * - `\"browse\"` — split view with root keys on the left, value tree on the right\n * - `\"raw\"` — full JSON tree view\n *\n * Accepts a single view or an array. First item is the default tab.\n * @default [\"browse\", \"raw\"]\n */\n views?: DevToolsView | DevToolsView[];\n}\n\n/**\n * Registers a named section in the Webiny DevTools extension.\n * Renders nothing — purely a data registration side-effect.\n *\n * When the component unmounts (e.g., route change), the section\n * is automatically removed from DevTools.\n *\n * @example\n * ```tsx\n * <DevToolsSection name=\"CMS Model\" data={model} />\n * <DevToolsSection name=\"Article\" group=\"CMS\" data={model} views=\"raw\" />\n * ```\n */\nexport function DevToolsSection({ name, data, group, views }: DevToolsSectionProps) {\n // Snapshot the data on every render to get a stable, plain value.\n // This also ensures that if data is a MobX observable, we capture\n // the current state as a plain object (no proxies on the hook).\n const dataKey = safeStringify(data);\n\n useEffect(() => {\n if (process.env.NODE_ENV !== \"development\") {\n return;\n }\n\n const normalizedViews: DevToolsView[] = views\n ? Array.isArray(views)\n ? views\n : [views]\n : [\"browse\", \"raw\"];\n\n let plainData: unknown;\n try {\n plainData = JSON.parse(dataKey);\n } catch {\n plainData = data;\n }\n\n const hook = getHook();\n hook.sections[name] = {\n data: plainData,\n group: group || \"Sections\",\n views: normalizedViews,\n updatedAt: Date.now()\n };\n hook.revision++;\n\n return () => {\n if (window.__WEBINY_DEVTOOLS_HOOK__) {\n delete window.__WEBINY_DEVTOOLS_HOOK__.sections[name];\n window.__WEBINY_DEVTOOLS_HOOK__.revision++;\n }\n };\n }, [name, dataKey, group, Array.isArray(views) ? views.join(\",\") : views]);\n\n return null;\n}\n\nfunction safeStringify(value: unknown): string {\n try {\n return JSON.stringify(value, (_key, v) => {\n if (typeof v === \"function\") {\n return `[Function: ${v.name || \"anonymous\"}]`;\n }\n if (typeof v === \"undefined\") {\n return \"[undefined]\";\n }\n return v;\n });\n } catch {\n return String(value);\n }\n}\n"],"names":["DevToolsSection","name","data","group","views","dataKey","safeStringify","useEffect","process","normalizedViews","Array","plainData","JSON","hook","getHook","Date","window","value","_key","v","String"],"mappings":";;AA2CO,SAASA,gBAAgB,EAAEC,IAAI,EAAEC,IAAI,EAAEC,KAAK,EAAEC,KAAK,EAAwB;IAI9E,MAAMC,UAAUC,cAAcJ;IAE9BK,UAAU;QACN,IAAIC,AAAyB,kBAAzBA,QAAQ,GAAG,CAAC,QAAQ,EACpB;QAGJ,MAAMC,kBAAkCL,QAClCM,MAAM,OAAO,CAACN,SACVA,QACA;YAACA;SAAM,GACX;YAAC;YAAU;SAAM;QAEvB,IAAIO;QACJ,IAAI;YACAA,YAAYC,KAAK,KAAK,CAACP;QAC3B,EAAE,OAAM;YACJM,YAAYT;QAChB;QAEA,MAAMW,OAAOC;QACbD,KAAK,QAAQ,CAACZ,KAAK,GAAG;YAClB,MAAMU;YACN,OAAOR,SAAS;YAChB,OAAOM;YACP,WAAWM,KAAK,GAAG;QACvB;QACAF,KAAK,QAAQ;QAEb,OAAO;YACH,IAAIG,OAAO,wBAAwB,EAAE;gBACjC,OAAOA,OAAO,wBAAwB,CAAC,QAAQ,CAACf,KAAK;gBACrDe,OAAO,wBAAwB,CAAC,QAAQ;YAC5C;QACJ;IACJ,GAAG;QAACf;QAAMI;QAASF;QAAOO,MAAM,OAAO,CAACN,SAASA,MAAM,IAAI,CAAC,OAAOA;KAAM;IAEzE,OAAO;AACX;AAEA,SAASE,cAAcW,KAAc;IACjC,IAAI;QACA,OAAOL,KAAK,SAAS,CAACK,OAAO,CAACC,MAAMC;YAChC,IAAI,AAAa,cAAb,OAAOA,GACP,OAAO,CAAC,WAAW,EAAEA,EAAE,IAAI,IAAI,YAAY,CAAC,CAAC;YAEjD,IAAI,AAAa,WAANA,GACP,OAAO;YAEX,OAAOA;QACX;IACJ,EAAE,OAAM;QACJ,OAAOC,OAAOH;IAClB;AACJ"}
package/Properties.d.ts CHANGED
@@ -1,17 +1,28 @@
1
1
  import React from "react";
2
+ import { PropertyStore } from "./domain/index.js";
3
+ export interface ConnectToPropertiesProps {
4
+ name: string;
5
+ children: React.ReactNode;
6
+ }
7
+ export declare const ConnectToProperties: ({ name, children }: ConnectToPropertiesProps) => React.JSX.Element;
2
8
  export interface Property {
3
9
  id: string;
4
10
  parent: string;
5
11
  name: string;
6
12
  value?: unknown;
7
13
  array?: boolean;
14
+ $isFirst?: boolean;
15
+ $isLast?: boolean;
8
16
  }
9
17
  interface AddPropertyOptions {
10
18
  after?: string;
11
19
  before?: string;
20
+ priority?: number;
12
21
  }
13
22
  interface PropertiesContext {
14
- properties: Property[];
23
+ name?: string;
24
+ store: PropertyStore;
25
+ getAncestor(name: string): PropertiesContext | undefined;
15
26
  getObject<T = unknown>(): T;
16
27
  addProperty(property: Property, options?: AddPropertyOptions): void;
17
28
  removeProperty(id: string): void;
@@ -19,10 +30,14 @@ interface PropertiesContext {
19
30
  }
20
31
  declare const PropertiesContext: React.Context<PropertiesContext | undefined>;
21
32
  interface PropertiesProps {
33
+ name?: string;
22
34
  onChange?(properties: Property[]): void;
35
+ children: React.ReactNode;
23
36
  }
24
- export declare const Properties: React.FC<PropertiesProps>;
37
+ export declare const Properties: ({ name, onChange, children }: PropertiesProps) => React.JSX.Element;
25
38
  export declare function useProperties(): PropertiesContext;
39
+ export declare function useMaybeProperties(): PropertiesContext | undefined;
40
+ export declare function useAncestorByName(name: string | undefined): PropertiesContext | undefined;
26
41
  interface PropertyProps {
27
42
  id?: string;
28
43
  name: string;
@@ -34,11 +49,12 @@ interface PropertyProps {
34
49
  remove?: boolean;
35
50
  parent?: string;
36
51
  root?: boolean;
52
+ children?: React.ReactNode;
37
53
  }
38
54
  export declare function useParentProperty(): Property | undefined;
39
55
  interface AncestorMatch {
40
56
  [key: string]: string | boolean | number | null | undefined;
41
57
  }
42
58
  export declare function useAncestor(params: AncestorMatch): Property | undefined;
43
- export declare const Property: React.FC<PropertyProps>;
59
+ export declare const Property: ({ id, name, value, children, after, before, replace, remove, array, root, parent }: PropertyProps) => React.JSX.Element | null;
44
60
  export {};
package/Properties.js CHANGED
@@ -1,193 +1,140 @@
1
- "use strict";
2
-
3
- var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard").default;
4
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
8
- exports.Property = exports.Properties = void 0;
9
- exports.useAncestor = useAncestor;
10
- exports.useParentProperty = useParentProperty;
11
- exports.useProperties = useProperties;
12
- var _objectSpread2 = _interopRequireDefault(require("@babel/runtime/helpers/objectSpread2"));
13
- var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
14
- var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
15
- var _react = _interopRequireWildcard(require("react"));
16
- var _utils = require("./utils");
17
- function removeByParent(id, properties) {
18
- return properties.filter(function (prop) {
19
- return prop.parent === id;
20
- }).reduce(function (acc, item) {
21
- return removeByParent(item.id, acc.filter(function (prop) {
22
- return prop.id !== item.id;
23
- }));
24
- }, properties);
25
- }
26
- var PropertiesContext = /*#__PURE__*/(0, _react.createContext)(undefined);
27
- var Properties = function Properties(_ref) {
28
- var onChange = _ref.onChange,
29
- children = _ref.children;
30
- var _useState = (0, _react.useState)([]),
31
- _useState2 = (0, _slicedToArray2.default)(_useState, 2),
32
- properties = _useState2[0],
33
- setProperties = _useState2[1];
34
- (0, _react.useEffect)(function () {
35
- if (onChange) {
36
- onChange(properties);
37
- }
38
- }, [properties]);
39
- var context = (0, _react.useMemo)(function () {
40
- return {
41
- properties: properties,
42
- getObject: function getObject() {
43
- return (0, _utils.toObject)(properties);
44
- },
45
- addProperty: function addProperty(property) {
46
- var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
47
- setProperties(function (properties) {
48
- // If a property with this ID already exists, merge the two properties.
49
- var index = properties.findIndex(function (prop) {
50
- return prop.id === property.id;
51
- });
52
- if (index > -1) {
53
- return [].concat((0, _toConsumableArray2.default)(properties.slice(0, index)), [(0, _objectSpread2.default)((0, _objectSpread2.default)({}, properties[index]), property)], (0, _toConsumableArray2.default)(properties.slice(index + 1)));
54
- }
55
- if (options.after) {
56
- var _index = properties.findIndex(function (prop) {
57
- return prop.id === options.after;
58
- });
59
- if (_index > -1) {
60
- return [].concat((0, _toConsumableArray2.default)(properties.slice(0, _index + 1)), [property], (0, _toConsumableArray2.default)(properties.slice(_index + 1)));
61
- }
62
- }
63
- if (options.before) {
64
- var _index2 = properties.findIndex(function (prop) {
65
- return prop.id === options.before;
66
- });
67
- if (_index2 > -1) {
68
- return [].concat((0, _toConsumableArray2.default)(properties.slice(0, _index2)), [property], (0, _toConsumableArray2.default)(properties.slice(_index2)));
69
- }
70
- }
71
- return [].concat((0, _toConsumableArray2.default)(properties), [property]);
72
- });
73
- },
74
- removeProperty: function removeProperty(id) {
75
- setProperties(function (properties) {
76
- return removeByParent(id, properties.filter(function (prop) {
77
- return prop.id !== id;
78
- }));
79
- });
80
- },
81
- replaceProperty: function replaceProperty(id, property) {
82
- setProperties(function (properties) {
83
- var toReplace = properties.findIndex(function (prop) {
84
- return prop.id === id;
85
- });
86
- if (toReplace > -1) {
87
- // Replace the property and remove all remaining child properties.
88
- return removeByParent(id, [].concat((0, _toConsumableArray2.default)(properties.slice(0, toReplace)), [property], (0, _toConsumableArray2.default)(properties.slice(toReplace + 1))));
89
- }
90
- return properties;
1
+ import react, { createContext, useContext, useEffect, useMemo, useRef } from "react";
2
+ import { getUniqueId, toObject } from "./utils.js";
3
+ import { PropertyStore } from "./domain/index.js";
4
+ import { usePropertyPriority } from "./PropertyPriority.js";
5
+ const PropertiesTargetContext = /*#__PURE__*/ createContext(void 0);
6
+ const ConnectToProperties = ({ name, children })=>/*#__PURE__*/ react.createElement(PropertiesTargetContext.Provider, {
7
+ value: name
8
+ }, children);
9
+ const PropertiesContext = /*#__PURE__*/ createContext(void 0);
10
+ const Properties = ({ name, onChange, children })=>{
11
+ const storeRef = useRef(null);
12
+ if (!storeRef.current) storeRef.current = new PropertyStore();
13
+ const store = storeRef.current;
14
+ let parent;
15
+ try {
16
+ parent = useProperties();
17
+ } catch {}
18
+ useEffect(()=>{
19
+ if (!onChange) return;
20
+ return store.subscribe((properties)=>{
21
+ onChange(properties);
91
22
  });
92
- }
93
- };
94
- }, [properties]);
95
- return /*#__PURE__*/_react.default.createElement(PropertiesContext.Provider, {
96
- value: context
97
- }, children);
23
+ }, [
24
+ store,
25
+ onChange
26
+ ]);
27
+ const context = useMemo(()=>({
28
+ name,
29
+ store,
30
+ getAncestor (ancestorName) {
31
+ if (!parent) return;
32
+ return parent && parent.name === ancestorName ? parent : parent.getAncestor(ancestorName);
33
+ },
34
+ getObject () {
35
+ return toObject(store.allProperties);
36
+ },
37
+ addProperty (property, options = {}) {
38
+ store.addProperty(property, options);
39
+ },
40
+ removeProperty (id) {
41
+ store.removeProperty(id);
42
+ },
43
+ replaceProperty (id, property) {
44
+ store.replaceProperty(id, property);
45
+ }
46
+ }), [
47
+ store
48
+ ]);
49
+ return /*#__PURE__*/ react.createElement(PropertiesContext.Provider, {
50
+ value: context
51
+ }, children);
98
52
  };
99
- exports.Properties = Properties;
100
53
  function useProperties() {
101
- var properties = (0, _react.useContext)(PropertiesContext);
102
- if (!properties) {
103
- throw Error("Properties context provider is missing!");
104
- }
105
- return properties;
54
+ const context = useContext(PropertiesContext);
55
+ if (!context) throw Error("Properties context provider is missing!");
56
+ return context;
57
+ }
58
+ function useMaybeProperties() {
59
+ return useContext(PropertiesContext);
60
+ }
61
+ function useAncestorByName(name) {
62
+ const parent = useMaybeProperties();
63
+ return useMemo(()=>{
64
+ if (!name || !parent) return;
65
+ if (parent.name === name) return parent;
66
+ return parent.getAncestor(name);
67
+ }, [
68
+ name
69
+ ]);
106
70
  }
107
- var PropertyContext = /*#__PURE__*/(0, _react.createContext)(undefined);
71
+ const PropertyContext = /*#__PURE__*/ createContext(void 0);
108
72
  function useParentProperty() {
109
- return (0, _react.useContext)(PropertyContext);
73
+ return useContext(PropertyContext);
110
74
  }
111
75
  function useAncestor(params) {
112
- var property = useParentProperty();
113
- var _useProperties = useProperties(),
114
- properties = _useProperties.properties;
115
- var matchOrGetAncestor = function matchOrGetAncestor(property, params) {
116
- var matchedProps = properties.filter(function (prop) {
117
- return prop.parent === property.id;
118
- }).filter(function (prop) {
119
- return prop.name in params && prop.value === params[prop.name];
120
- });
121
- if (matchedProps.length === Object.keys(params).length) {
122
- return property;
123
- }
124
- var newParent = property.parent ? properties.find(function (prop) {
125
- return prop.id === property.parent;
126
- }) : undefined;
127
- return newParent ? matchOrGetAncestor(newParent, params) : undefined;
128
- };
129
- return property ? matchOrGetAncestor(property, params) : undefined;
76
+ const property = useParentProperty();
77
+ const { store } = useProperties();
78
+ const matchOrGetAncestor = (property, params)=>{
79
+ const children = store.getChildrenOf(property.id);
80
+ const matchedProps = children.filter((prop)=>prop.name in params && prop.value === params[prop.name]);
81
+ if (matchedProps.length === Object.keys(params).length) return property;
82
+ const newParent = property.parent ? store.getById(property.parent) : void 0;
83
+ return newParent ? matchOrGetAncestor(newParent, params) : void 0;
84
+ };
85
+ return property ? matchOrGetAncestor(property, params) : void 0;
130
86
  }
131
- var Property = function Property(_ref2) {
132
- var id = _ref2.id,
133
- name = _ref2.name,
134
- value = _ref2.value,
135
- children = _ref2.children,
136
- _ref2$after = _ref2.after,
137
- after = _ref2$after === void 0 ? undefined : _ref2$after,
138
- _ref2$before = _ref2.before,
139
- before = _ref2$before === void 0 ? undefined : _ref2$before,
140
- _ref2$replace = _ref2.replace,
141
- replace = _ref2$replace === void 0 ? undefined : _ref2$replace,
142
- _ref2$remove = _ref2.remove,
143
- remove = _ref2$remove === void 0 ? false : _ref2$remove,
144
- _ref2$array = _ref2.array,
145
- array = _ref2$array === void 0 ? false : _ref2$array,
146
- _ref2$root = _ref2.root,
147
- root = _ref2$root === void 0 ? false : _ref2$root,
148
- _ref2$parent = _ref2.parent,
149
- parent = _ref2$parent === void 0 ? undefined : _ref2$parent;
150
- var uniqueId = (0, _react.useMemo)(function () {
151
- return id || (0, _utils.getUniqueId)();
152
- }, []);
153
- var parentProperty = useParentProperty();
154
- var properties = useProperties();
155
- if (!properties) {
156
- throw Error("<Properties> provider is missing higher in the hierarchy!");
157
- }
158
- var addProperty = properties.addProperty,
159
- removeProperty = properties.removeProperty,
160
- replaceProperty = properties.replaceProperty;
161
- var parentId = parent ? parent : root ? "" : (parentProperty === null || parentProperty === void 0 ? void 0 : parentProperty.id) || "";
162
- var property = {
163
- id: uniqueId,
164
- name: name,
165
- value: value,
166
- parent: parentId,
167
- array: array
168
- };
169
- (0, _react.useEffect)(function () {
170
- if (remove) {
171
- removeProperty(uniqueId);
172
- return;
173
- }
174
- if (replace) {
175
- replaceProperty(replace, property);
176
- return;
177
- }
178
- addProperty(property, {
179
- after: after,
180
- before: before
181
- });
182
- return function () {
183
- removeProperty(uniqueId);
87
+ const Property = ({ id, name, value, children, after, before, replace, remove = false, array = false, root = false, parent })=>{
88
+ const targetName = useContext(PropertiesTargetContext);
89
+ const uniqueId = useMemo(()=>id || getUniqueId(), []);
90
+ const parentProperty = useParentProperty();
91
+ const immediateProperties = useProperties();
92
+ const ancestorByName = useAncestorByName(targetName);
93
+ const previousValue = useRef(value);
94
+ const priority = usePropertyPriority();
95
+ const properties = targetName && ancestorByName ? ancestorByName : immediateProperties;
96
+ if (!properties) throw Error("<Properties> provider is missing higher in the hierarchy!");
97
+ const { addProperty, removeProperty, replaceProperty, store: propertyStore } = properties;
98
+ const parentId = parent ? parent : root ? "" : parentProperty?.id || "";
99
+ const property = {
100
+ id: uniqueId,
101
+ name,
102
+ value,
103
+ parent: parentId,
104
+ array
184
105
  };
185
- }, []);
186
- if (children) {
187
- return /*#__PURE__*/_react.default.createElement(PropertyContext.Provider, {
188
- value: property
106
+ if (!remove) propertyStore.registerLookup(property);
107
+ useEffect(()=>{
108
+ if (remove) return void removeProperty(uniqueId);
109
+ if (replace) return void replaceProperty(replace, property);
110
+ const $isFirst = "$first" === before;
111
+ const $isLast = "$last" === after;
112
+ addProperty({
113
+ ...property,
114
+ $isFirst,
115
+ $isLast
116
+ }, {
117
+ after,
118
+ before,
119
+ priority
120
+ });
121
+ return ()=>{
122
+ removeProperty(uniqueId);
123
+ };
124
+ }, []);
125
+ useEffect(()=>{
126
+ if (previousValue.current !== value) {
127
+ previousValue.current = value;
128
+ if (!remove && !replace) replaceProperty(uniqueId, property);
129
+ }
130
+ }, [
131
+ value
132
+ ]);
133
+ if (children) return /*#__PURE__*/ react.createElement(PropertyContext.Provider, {
134
+ value: property
189
135
  }, children);
190
- }
191
- return null;
136
+ return null;
192
137
  };
193
- exports.Property = Property;
138
+ export { ConnectToProperties, Properties, Property, useAncestor, useAncestorByName, useMaybeProperties, useParentProperty, useProperties };
139
+
140
+ //# sourceMappingURL=Properties.js.map
package/Properties.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"names":["removeByParent","id","properties","filter","prop","parent","reduce","acc","item","PropertiesContext","createContext","undefined","Properties","onChange","children","useState","setProperties","useEffect","context","useMemo","getObject","toObject","addProperty","property","options","index","findIndex","slice","after","before","removeProperty","replaceProperty","toReplace","useProperties","useContext","Error","PropertyContext","useParentProperty","useAncestor","params","matchOrGetAncestor","matchedProps","name","value","length","Object","keys","newParent","find","Property","replace","remove","array","root","uniqueId","getUniqueId","parentProperty","parentId"],"sources":["Properties.tsx"],"sourcesContent":["import React, { createContext, useContext, useEffect, useMemo, useState } from \"react\";\nimport { getUniqueId, toObject } from \"./utils\";\n\nexport interface Property {\n id: string;\n parent: string;\n name: string;\n value?: unknown;\n array?: boolean;\n}\n\nfunction removeByParent(id: string, properties: Property[]): Property[] {\n return properties\n .filter(prop => prop.parent === id)\n .reduce((acc, item) => {\n return removeByParent(\n item.id,\n acc.filter(prop => prop.id !== item.id)\n );\n }, properties);\n}\n\ninterface AddPropertyOptions {\n after?: string;\n before?: string;\n}\n\ninterface PropertiesContext {\n properties: Property[];\n getObject<T = unknown>(): T;\n addProperty(property: Property, options?: AddPropertyOptions): void;\n removeProperty(id: string): void;\n replaceProperty(id: string, property: Property): void;\n}\n\nconst PropertiesContext = createContext<PropertiesContext | undefined>(undefined);\n\ninterface PropertiesProps {\n onChange?(properties: Property[]): void;\n}\n\nexport const Properties: React.FC<PropertiesProps> = ({ onChange, children }) => {\n const [properties, setProperties] = useState<Property[]>([]);\n\n useEffect(() => {\n if (onChange) {\n onChange(properties);\n }\n }, [properties]);\n\n const context: PropertiesContext = useMemo(\n () => ({\n properties,\n getObject<T>() {\n return toObject(properties) as T;\n },\n addProperty(property, options = {}) {\n setProperties(properties => {\n // If a property with this ID already exists, merge the two properties.\n const index = properties.findIndex(prop => prop.id === property.id);\n if (index > -1) {\n return [\n ...properties.slice(0, index),\n { ...properties[index], ...property },\n ...properties.slice(index + 1)\n ];\n }\n\n if (options.after) {\n const index = properties.findIndex(prop => prop.id === options.after);\n if (index > -1) {\n return [\n ...properties.slice(0, index + 1),\n property,\n ...properties.slice(index + 1)\n ];\n }\n }\n\n if (options.before) {\n const index = properties.findIndex(prop => prop.id === options.before);\n if (index > -1) {\n return [\n ...properties.slice(0, index),\n property,\n ...properties.slice(index)\n ];\n }\n }\n\n return [...properties, property];\n });\n },\n removeProperty(id) {\n setProperties(properties => {\n return removeByParent(\n id,\n properties.filter(prop => prop.id !== id)\n );\n });\n },\n replaceProperty(id, property) {\n setProperties(properties => {\n const toReplace = properties.findIndex(prop => prop.id === id);\n\n if (toReplace > -1) {\n // Replace the property and remove all remaining child properties.\n return removeByParent(id, [\n ...properties.slice(0, toReplace),\n property,\n ...properties.slice(toReplace + 1)\n ]);\n }\n return properties;\n });\n }\n }),\n [properties]\n );\n\n return <PropertiesContext.Provider value={context}>{children}</PropertiesContext.Provider>;\n};\n\nexport function useProperties() {\n const properties = useContext(PropertiesContext);\n if (!properties) {\n throw Error(\"Properties context provider is missing!\");\n }\n\n return properties;\n}\n\ninterface PropertyProps {\n id?: string;\n name: string;\n value?: unknown;\n array?: boolean;\n after?: string;\n before?: string;\n replace?: string;\n remove?: boolean;\n parent?: string;\n root?: boolean;\n}\n\nconst PropertyContext = createContext<Property | undefined>(undefined);\n\nexport function useParentProperty() {\n return useContext(PropertyContext);\n}\n\ninterface AncestorMatch {\n [key: string]: string | boolean | number | null | undefined;\n}\n\nexport function useAncestor(params: AncestorMatch) {\n const property = useParentProperty();\n const { properties } = useProperties();\n\n const matchOrGetAncestor = (\n property: Property,\n params: AncestorMatch\n ): Property | undefined => {\n const matchedProps = properties\n .filter(prop => prop.parent === property.id)\n .filter(prop => prop.name in params && prop.value === params[prop.name]);\n\n if (matchedProps.length === Object.keys(params).length) {\n return property;\n }\n\n const newParent = property.parent\n ? properties.find(prop => prop.id === property.parent)\n : undefined;\n\n return newParent ? matchOrGetAncestor(newParent, params) : undefined;\n };\n\n return property ? matchOrGetAncestor(property, params) : undefined;\n}\n\nexport const Property: React.FC<PropertyProps> = ({\n id,\n name,\n value,\n children,\n after = undefined,\n before = undefined,\n replace = undefined,\n remove = false,\n array = false,\n root = false,\n parent = undefined\n}) => {\n const uniqueId = useMemo(() => id || getUniqueId(), []);\n const parentProperty = useParentProperty();\n const properties = useProperties();\n\n if (!properties) {\n throw Error(\"<Properties> provider is missing higher in the hierarchy!\");\n }\n\n const { addProperty, removeProperty, replaceProperty } = properties;\n const parentId = parent ? parent : root ? \"\" : parentProperty?.id || \"\";\n const property = { id: uniqueId, name, value, parent: parentId, array };\n\n useEffect(() => {\n if (remove) {\n removeProperty(uniqueId);\n return;\n }\n\n if (replace) {\n replaceProperty(replace, property);\n return;\n }\n\n addProperty(property, { after, before });\n\n return () => {\n removeProperty(uniqueId);\n };\n }, []);\n\n if (children) {\n return <PropertyContext.Provider value={property}>{children}</PropertyContext.Provider>;\n }\n\n return null;\n};\n"],"mappings":"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"}
1
+ {"version":3,"file":"Properties.js","sources":["../src/Properties.tsx"],"sourcesContent":["import React, { createContext, useContext, useEffect, useMemo, useRef } from \"react\";\nimport { getUniqueId, toObject } from \"./utils.js\";\nimport { PropertyStore } from \"./domain/index.js\";\nimport { usePropertyPriority } from \"./PropertyPriority.js\";\n\nconst PropertiesTargetContext = createContext<string | undefined>(undefined);\n\nexport interface ConnectToPropertiesProps {\n name: string;\n children: React.ReactNode;\n}\n\nexport const ConnectToProperties = ({ name, children }: ConnectToPropertiesProps) => {\n return (\n <PropertiesTargetContext.Provider value={name}>{children}</PropertiesTargetContext.Provider>\n );\n};\n\nexport interface Property {\n id: string;\n parent: string;\n name: string;\n value?: unknown;\n array?: boolean;\n $isFirst?: boolean;\n $isLast?: boolean;\n}\n\ninterface AddPropertyOptions {\n after?: string;\n before?: string;\n priority?: number;\n}\n\ninterface PropertiesContext {\n name?: string;\n store: PropertyStore;\n getAncestor(name: string): PropertiesContext | undefined;\n getObject<T = unknown>(): T;\n addProperty(property: Property, options?: AddPropertyOptions): void;\n removeProperty(id: string): void;\n replaceProperty(id: string, property: Property): void;\n}\n\nconst PropertiesContext = createContext<PropertiesContext | undefined>(undefined);\n\ninterface PropertiesProps {\n name?: string;\n onChange?(properties: Property[]): void;\n children: React.ReactNode;\n}\n\nexport const Properties = ({ name, onChange, children }: PropertiesProps) => {\n const storeRef = useRef<PropertyStore | null>(null);\n if (!storeRef.current) {\n storeRef.current = new PropertyStore();\n }\n const store = storeRef.current;\n\n let parent: PropertiesContext;\n\n try {\n parent = useProperties();\n } catch {\n // Do nothing, if there's no parent.\n }\n\n useEffect(() => {\n if (!onChange) {\n return;\n }\n\n return store.subscribe(properties => {\n onChange(properties);\n });\n }, [store, onChange]);\n\n // Context value is stable — it never changes after mount.\n // Children never re-render due to context changes.\n const context: PropertiesContext = useMemo(\n () => ({\n name,\n store,\n getAncestor(ancestorName: string) {\n if (!parent) {\n return undefined;\n }\n\n return parent && parent.name === ancestorName\n ? parent\n : parent.getAncestor(ancestorName);\n },\n getObject<T>() {\n return toObject(store.allProperties) as T;\n },\n addProperty(property, options = {}) {\n store.addProperty(property, options);\n },\n removeProperty(id) {\n store.removeProperty(id);\n },\n replaceProperty(id, property) {\n store.replaceProperty(id, property);\n }\n }),\n [store]\n );\n\n return <PropertiesContext.Provider value={context}>{children}</PropertiesContext.Provider>;\n};\n\nexport function useProperties() {\n const context = useContext(PropertiesContext);\n if (!context) {\n throw Error(\"Properties context provider is missing!\");\n }\n\n return context;\n}\n\nexport function useMaybeProperties() {\n return useContext(PropertiesContext);\n}\n\nexport function useAncestorByName(name: string | undefined) {\n const parent = useMaybeProperties();\n\n return useMemo(() => {\n if (!name || !parent) {\n return undefined;\n }\n\n if (parent.name === name) {\n return parent;\n }\n\n return parent.getAncestor(name);\n }, [name]);\n}\n\ninterface PropertyProps {\n id?: string;\n name: string;\n value?: unknown;\n array?: boolean;\n after?: string;\n before?: string;\n replace?: string;\n remove?: boolean;\n parent?: string;\n root?: boolean;\n children?: React.ReactNode;\n}\n\nconst PropertyContext = createContext<Property | undefined>(undefined);\n\nexport function useParentProperty() {\n return useContext(PropertyContext);\n}\n\ninterface AncestorMatch {\n [key: string]: string | boolean | number | null | undefined;\n}\n\nexport function useAncestor(params: AncestorMatch) {\n const property = useParentProperty();\n const { store } = useProperties();\n\n const matchOrGetAncestor = (\n property: Property,\n params: AncestorMatch\n ): Property | undefined => {\n const children = store.getChildrenOf(property.id);\n const matchedProps = children.filter(\n prop => prop.name in params && prop.value === params[prop.name]\n );\n\n if (matchedProps.length === Object.keys(params).length) {\n return property;\n }\n\n const newParent = property.parent ? store.getById(property.parent) : undefined;\n\n return newParent ? matchOrGetAncestor(newParent, params) : undefined;\n };\n\n return property ? matchOrGetAncestor(property, params) : undefined;\n}\n\nexport const Property = ({\n id,\n name,\n value,\n children,\n after = undefined,\n before = undefined,\n replace = undefined,\n remove = false,\n array = false,\n root = false,\n parent = undefined\n}: PropertyProps) => {\n const targetName = useContext(PropertiesTargetContext);\n const uniqueId = useMemo(() => id || getUniqueId(), []);\n const parentProperty = useParentProperty();\n const immediateProperties = useProperties();\n const ancestorByName = useAncestorByName(targetName);\n const previousValue = useRef(value);\n const priority = usePropertyPriority();\n\n const properties = targetName && ancestorByName ? ancestorByName : immediateProperties;\n\n if (!properties) {\n throw Error(\"<Properties> provider is missing higher in the hierarchy!\");\n }\n\n const { addProperty, removeProperty, replaceProperty, store: propertyStore } = properties;\n const parentId = parent ? parent : root ? \"\" : parentProperty?.id || \"\";\n const property = { id: uniqueId, name, value, parent: parentId, array };\n\n // Register in the synchronous lookup during render so useAncestor can find this property.\n if (!remove) {\n propertyStore.registerLookup(property);\n }\n\n useEffect(() => {\n if (remove) {\n removeProperty(uniqueId);\n return;\n }\n\n if (replace) {\n replaceProperty(replace, property);\n return;\n }\n\n const $isFirst = before === \"$first\";\n const $isLast = after === \"$last\";\n\n addProperty({ ...property, $isFirst, $isLast }, { after, before, priority });\n\n return () => {\n removeProperty(uniqueId);\n };\n }, []);\n\n useEffect(() => {\n if (previousValue.current !== value) {\n previousValue.current = value;\n if (!remove && !replace) {\n replaceProperty(uniqueId, property);\n }\n }\n }, [value]);\n\n if (children) {\n return <PropertyContext.Provider value={property}>{children}</PropertyContext.Provider>;\n }\n\n return null;\n};\n"],"names":["PropertiesTargetContext","createContext","undefined","ConnectToProperties","name","children","PropertiesContext","Properties","onChange","storeRef","useRef","PropertyStore","store","parent","useProperties","useEffect","properties","context","useMemo","ancestorName","toObject","property","options","id","useContext","Error","useMaybeProperties","useAncestorByName","PropertyContext","useParentProperty","useAncestor","params","matchOrGetAncestor","matchedProps","prop","Object","newParent","Property","value","after","before","replace","remove","array","root","targetName","uniqueId","getUniqueId","parentProperty","immediateProperties","ancestorByName","previousValue","priority","usePropertyPriority","addProperty","removeProperty","replaceProperty","propertyStore","parentId","$isFirst","$isLast"],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import React from "react";
2
+ interface PropertyPriorityProviderProps {
3
+ priority: number;
4
+ children: React.ReactNode;
5
+ }
6
+ export declare const PropertyPriorityProvider: ({ priority, children }: PropertyPriorityProviderProps) => React.JSX.Element;
7
+ export declare function usePropertyPriority(): number;
8
+ export {};
@@ -0,0 +1,11 @@
1
+ import react, { createContext, useContext } from "react";
2
+ const PropertyPriorityContext = /*#__PURE__*/ createContext(0);
3
+ const PropertyPriorityProvider = ({ priority, children })=>/*#__PURE__*/ react.createElement(PropertyPriorityContext.Provider, {
4
+ value: priority
5
+ }, children);
6
+ function usePropertyPriority() {
7
+ return useContext(PropertyPriorityContext);
8
+ }
9
+ export { PropertyPriorityProvider, usePropertyPriority };
10
+
11
+ //# sourceMappingURL=PropertyPriority.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PropertyPriority.js","sources":["../src/PropertyPriority.tsx"],"sourcesContent":["import React, { createContext, useContext } from \"react\";\n\nconst PropertyPriorityContext = createContext(0);\n\ninterface PropertyPriorityProviderProps {\n priority: number;\n children: React.ReactNode;\n}\n\nexport const PropertyPriorityProvider = ({ priority, children }: PropertyPriorityProviderProps) => {\n return (\n <PropertyPriorityContext.Provider value={priority}>\n {children}\n </PropertyPriorityContext.Provider>\n );\n};\n\nexport function usePropertyPriority(): number {\n return useContext(PropertyPriorityContext);\n}\n"],"names":["PropertyPriorityContext","createContext","PropertyPriorityProvider","priority","children","usePropertyPriority","useContext"],"mappings":";AAEA,MAAMA,0BAA0B,WAAHA,GAAGC,cAAc;AAOvC,MAAMC,2BAA2B,CAAC,EAAEC,QAAQ,EAAEC,QAAQ,EAAiC,GACnF,WAAP,GACI,oBAACJ,wBAAwB,QAAQ;QAAC,OAAOG;OACpCC;AAKN,SAASC;IACZ,OAAOC,WAAWN;AACtB"}
package/README.md CHANGED
@@ -1,65 +1,11 @@
1
- # React Properties
1
+ # @webiny/react-properties
2
2
 
3
- [![](https://img.shields.io/npm/dw/@webiny/react-properties.svg)](https://www.npmjs.com/package/@webiny/react-properties)
4
- [![](https://img.shields.io/npm/v/@webiny/react-properties.svg)](https://www.npmjs.com/package/@webiny/react-properties)
5
- [![code style: prettier](https://img.shields.io/badge/code_style-prettier-ff69b4.svg?style=flat-square)](https://github.com/prettier/prettier)
6
- [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg?style=flat-square)](http://makeapullrequest.com)
3
+ > [!NOTE]
4
+ > This package is part of the [Webiny](https://www.webiny.com) monorepo.
5
+ > It’s **included in every Webiny project by default** and is not meant to be used as a standalone package.
7
6
 
8
- A tiny React properties framework, to build dynamic data objects using React components, which can be customized after initial creation. The usage is very similar to how you write XML data structures, but in this case you're using actual React.
7
+ 📘 **Documentation:** [https://www.webiny.com/docs](https://www.webiny.com/docs)
9
8
 
10
- ## Basic Example
9
+ ---
11
10
 
12
- ```jsx
13
- import React, { useCallback } from "react";
14
- import { Properties, Property, toObject } from "@webiny/react-properties";
15
-
16
- const View = () => {
17
- const onChange = useCallback(properties => {
18
- console.log(toObject(properties));
19
- }, []);
20
-
21
- return (
22
- <Properties onChange={onChange}>
23
- <Property name={"group"}>
24
- <Property name={"name"} value={"layout"} />
25
- <Property name={"label"} value={"Layout"} />
26
- <Property name={"toolbar"}>
27
- <Property name={"name"} value={"basic"} />
28
- </Property>
29
- </Property>
30
- <Property name={"group"}>
31
- <Property name={"name"} value={"heroes"} />
32
- <Property name={"label"} value={"Heroes"} />
33
- <Property name={"toolbar"}>
34
- <Property name={"name"} value={"heroes"} />
35
- </Property>
36
- </Property>
37
- </Properties>
38
- );
39
- };
40
- ```
41
-
42
- Output:
43
-
44
- ```json
45
- {
46
- "group": [
47
- {
48
- "name": "layout",
49
- "label": "Layout",
50
- "toolbar": {
51
- "name": "basic"
52
- }
53
- },
54
- {
55
- "name": "heroes",
56
- "label": "Heroes",
57
- "toolbar": {
58
- "name": "heroes"
59
- }
60
- }
61
- ]
62
- }
63
- ```
64
-
65
- For more examples, check out the test files.
11
+ _This README file is automatically generated during the publish process._