@webiny/app-record-locking 0.0.0-unstable.06b2ede40f

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (147) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +12 -0
  3. package/components/HeadlessCmsActionsAcoCell.d.ts +2 -0
  4. package/components/HeadlessCmsActionsAcoCell.js +60 -0
  5. package/components/HeadlessCmsActionsAcoCell.js.map +1 -0
  6. package/components/HeadlessCmsContentEntry/ContentEntryGuard.d.ts +9 -0
  7. package/components/HeadlessCmsContentEntry/ContentEntryGuard.js +40 -0
  8. package/components/HeadlessCmsContentEntry/ContentEntryGuard.js.map +1 -0
  9. package/components/HeadlessCmsContentEntry/ContentEntryLocker.d.ts +18 -0
  10. package/components/HeadlessCmsContentEntry/ContentEntryLocker.js +106 -0
  11. package/components/HeadlessCmsContentEntry/ContentEntryLocker.js.map +1 -0
  12. package/components/HeadlessCmsContentEntry/HeadlessCmsContentEntry.d.ts +2 -0
  13. package/components/HeadlessCmsContentEntry/HeadlessCmsContentEntry.js +90 -0
  14. package/components/HeadlessCmsContentEntry/HeadlessCmsContentEntry.js.map +1 -0
  15. package/components/HeadlessCmsContentEntry/index.d.ts +1 -0
  16. package/components/HeadlessCmsContentEntry/index.js +3 -0
  17. package/components/HeadlessCmsContentEntry/index.js.map +1 -0
  18. package/components/LockedRecord/LockedRecord.d.ts +6 -0
  19. package/components/LockedRecord/LockedRecord.js +66 -0
  20. package/components/LockedRecord/LockedRecord.js.map +1 -0
  21. package/components/LockedRecord/LockedRecordForceUnlock.d.ts +9 -0
  22. package/components/LockedRecord/LockedRecordForceUnlock.js +75 -0
  23. package/components/LockedRecord/LockedRecordForceUnlock.js.map +1 -0
  24. package/components/LockedRecord/index.d.ts +1 -0
  25. package/components/LockedRecord/index.js +3 -0
  26. package/components/LockedRecord/index.js.map +1 -0
  27. package/components/RecordLockingProvider.d.ts +7 -0
  28. package/components/RecordLockingProvider.js +125 -0
  29. package/components/RecordLockingProvider.js.map +1 -0
  30. package/components/decorators/UseContentEntriesListHookDecorator.d.ts +2 -0
  31. package/components/decorators/UseContentEntriesListHookDecorator.js +21 -0
  32. package/components/decorators/UseContentEntriesListHookDecorator.js.map +1 -0
  33. package/components/decorators/UseRecordsDecorator.d.ts +9 -0
  34. package/components/decorators/UseRecordsDecorator.js +25 -0
  35. package/components/decorators/UseRecordsDecorator.js.map +1 -0
  36. package/components/decorators/UseSaveEntryDecorator.d.ts +2 -0
  37. package/components/decorators/UseSaveEntryDecorator.js +55 -0
  38. package/components/decorators/UseSaveEntryDecorator.js.map +1 -0
  39. package/components/permissionRenderer/RecordLockingPermissions.d.ts +7 -0
  40. package/components/permissionRenderer/RecordLockingPermissions.js +71 -0
  41. package/components/permissionRenderer/RecordLockingPermissions.js.map +1 -0
  42. package/components/permissionRenderer/index.d.ts +2 -0
  43. package/components/permissionRenderer/index.js +21 -0
  44. package/components/permissionRenderer/index.js.map +1 -0
  45. package/domain/RecordLocking.d.ts +28 -0
  46. package/domain/RecordLocking.js +320 -0
  47. package/domain/RecordLocking.js.map +1 -0
  48. package/domain/RecordLockingClient.d.ts +12 -0
  49. package/domain/RecordLockingClient.js +19 -0
  50. package/domain/RecordLockingClient.js.map +1 -0
  51. package/domain/RecordLockingGetLockRecord.d.ts +11 -0
  52. package/domain/RecordLockingGetLockRecord.js +21 -0
  53. package/domain/RecordLockingGetLockRecord.js.map +1 -0
  54. package/domain/RecordLockingGetLockedEntryLockRecord.d.ts +11 -0
  55. package/domain/RecordLockingGetLockedEntryLockRecord.js +19 -0
  56. package/domain/RecordLockingGetLockedEntryLockRecord.js.map +1 -0
  57. package/domain/RecordLockingIsEntryLocked.d.ts +11 -0
  58. package/domain/RecordLockingIsEntryLocked.js +19 -0
  59. package/domain/RecordLockingIsEntryLocked.js.map +1 -0
  60. package/domain/RecordLockingListLockRecords.d.ts +12 -0
  61. package/domain/RecordLockingListLockRecords.js +31 -0
  62. package/domain/RecordLockingListLockRecords.js.map +1 -0
  63. package/domain/RecordLockingLockEntry.d.ts +11 -0
  64. package/domain/RecordLockingLockEntry.js +15 -0
  65. package/domain/RecordLockingLockEntry.js.map +1 -0
  66. package/domain/RecordLockingUnlockEntry.d.ts +11 -0
  67. package/domain/RecordLockingUnlockEntry.js +19 -0
  68. package/domain/RecordLockingUnlockEntry.js.map +1 -0
  69. package/domain/RecordLockingUpdateEntryLock.d.ts +11 -0
  70. package/domain/RecordLockingUpdateEntryLock.js +19 -0
  71. package/domain/RecordLockingUpdateEntryLock.js.map +1 -0
  72. package/domain/abstractions/IRecordLocking.d.ts +25 -0
  73. package/domain/abstractions/IRecordLocking.js +3 -0
  74. package/domain/abstractions/IRecordLocking.js.map +1 -0
  75. package/domain/abstractions/IRecordLockingClient.d.ts +6 -0
  76. package/domain/abstractions/IRecordLockingClient.js +3 -0
  77. package/domain/abstractions/IRecordLockingClient.js.map +1 -0
  78. package/domain/abstractions/IRecordLockingGetLockRecord.d.ts +12 -0
  79. package/domain/abstractions/IRecordLockingGetLockRecord.js +3 -0
  80. package/domain/abstractions/IRecordLockingGetLockRecord.js.map +1 -0
  81. package/domain/abstractions/IRecordLockingGetLockedEntryLockRecord.d.ts +12 -0
  82. package/domain/abstractions/IRecordLockingGetLockedEntryLockRecord.js +3 -0
  83. package/domain/abstractions/IRecordLockingGetLockedEntryLockRecord.js.map +1 -0
  84. package/domain/abstractions/IRecordLockingIsEntryLocked.d.ts +8 -0
  85. package/domain/abstractions/IRecordLockingIsEntryLocked.js +3 -0
  86. package/domain/abstractions/IRecordLockingIsEntryLocked.js.map +1 -0
  87. package/domain/abstractions/IRecordLockingListLockRecords.d.ts +19 -0
  88. package/domain/abstractions/IRecordLockingListLockRecords.js +3 -0
  89. package/domain/abstractions/IRecordLockingListLockRecords.js.map +1 -0
  90. package/domain/abstractions/IRecordLockingLockEntry.d.ts +12 -0
  91. package/domain/abstractions/IRecordLockingLockEntry.js +3 -0
  92. package/domain/abstractions/IRecordLockingLockEntry.js.map +1 -0
  93. package/domain/abstractions/IRecordLockingUnlockEntry.d.ts +13 -0
  94. package/domain/abstractions/IRecordLockingUnlockEntry.js +3 -0
  95. package/domain/abstractions/IRecordLockingUnlockEntry.js.map +1 -0
  96. package/domain/abstractions/IRecordLockingUpdateEntryLock.d.ts +12 -0
  97. package/domain/abstractions/IRecordLockingUpdateEntryLock.js +3 -0
  98. package/domain/abstractions/IRecordLockingUpdateEntryLock.js.map +1 -0
  99. package/domain/graphql/fields.d.ts +2 -0
  100. package/domain/graphql/fields.js +30 -0
  101. package/domain/graphql/fields.js.map +1 -0
  102. package/domain/graphql/getLockRecord.d.ts +12 -0
  103. package/domain/graphql/getLockRecord.js +18 -0
  104. package/domain/graphql/getLockRecord.js.map +1 -0
  105. package/domain/graphql/getLockedEntryLockRecord.d.ts +12 -0
  106. package/domain/graphql/getLockedEntryLockRecord.js +18 -0
  107. package/domain/graphql/getLockedEntryLockRecord.js.map +1 -0
  108. package/domain/graphql/isEntryLocked.d.ts +12 -0
  109. package/domain/graphql/isEntryLocked.js +16 -0
  110. package/domain/graphql/isEntryLocked.js.map +1 -0
  111. package/domain/graphql/listLockRecords.d.ts +17 -0
  112. package/domain/graphql/listLockRecords.js +26 -0
  113. package/domain/graphql/listLockRecords.js.map +1 -0
  114. package/domain/graphql/lockEntry.d.ts +12 -0
  115. package/domain/graphql/lockEntry.js +20 -0
  116. package/domain/graphql/lockEntry.js.map +1 -0
  117. package/domain/graphql/unlockEntry.d.ts +12 -0
  118. package/domain/graphql/unlockEntry.js +18 -0
  119. package/domain/graphql/unlockEntry.js.map +1 -0
  120. package/domain/graphql/updateEntryLock.d.ts +12 -0
  121. package/domain/graphql/updateEntryLock.js +18 -0
  122. package/domain/graphql/updateEntryLock.js.map +1 -0
  123. package/domain/utils/createRecordLockingClient.d.ts +3 -0
  124. package/domain/utils/createRecordLockingClient.js +12 -0
  125. package/domain/utils/createRecordLockingClient.js.map +1 -0
  126. package/domain/utils/createRecordLockingError.d.ts +6 -0
  127. package/domain/utils/createRecordLockingError.js +16 -0
  128. package/domain/utils/createRecordLockingError.js.map +1 -0
  129. package/hooks/index.d.ts +2 -0
  130. package/hooks/index.js +4 -0
  131. package/hooks/index.js.map +1 -0
  132. package/hooks/usePermission.d.ts +3 -0
  133. package/hooks/usePermission.js +21 -0
  134. package/hooks/usePermission.js.map +1 -0
  135. package/hooks/useRecordLocking.d.ts +2 -0
  136. package/hooks/useRecordLocking.js +12 -0
  137. package/hooks/useRecordLocking.js.map +1 -0
  138. package/index.d.ts +7 -0
  139. package/index.js +29 -0
  140. package/index.js.map +1 -0
  141. package/package.json +52 -0
  142. package/types.d.ts +75 -0
  143. package/types.js +3 -0
  144. package/types.js.map +1 -0
  145. package/utils/createCacheKey.d.ts +3 -0
  146. package/utils/createCacheKey.js +15 -0
  147. package/utils/createCacheKey.js.map +1 -0
@@ -0,0 +1 @@
1
+ export * from "./LockedRecord";
@@ -0,0 +1,3 @@
1
+ export * from "./LockedRecord";
2
+
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":[],"sources":["index.ts"],"sourcesContent":["export * from \"./LockedRecord\";\n"],"mappings":"AAAA","ignoreList":[]}
@@ -0,0 +1,7 @@
1
+ import React from "react";
2
+ import type { IPossiblyRecordLockingRecord, IRecordLockingContext } from "../types";
3
+ export interface IRecordLockingProviderProps {
4
+ children: React.ReactNode;
5
+ }
6
+ export declare const RecordLockingContext: React.Context<IRecordLockingContext<IPossiblyRecordLockingRecord>>;
7
+ export declare const RecordLockingProvider: (props: IRecordLockingProviderProps) => React.JSX.Element;
@@ -0,0 +1,125 @@
1
+ import React, { useCallback, useMemo, useState } from "react";
2
+ import { useApolloClient } from "@apollo/react-hooks";
3
+ import { createRecordLocking } from "../domain/RecordLocking";
4
+ import { useStateIfMounted } from "@webiny/app-admin";
5
+ export const RecordLockingContext = /*#__PURE__*/React.createContext({});
6
+ const isSameArray = (existingRecords, newRecords) => {
7
+ if (existingRecords.length !== newRecords.length) {
8
+ return false;
9
+ }
10
+ return existingRecords.every((record, index) => {
11
+ const newRecord = newRecords[index];
12
+ if (!newRecord) {
13
+ return false;
14
+ }
15
+ return newRecord.id === record.id && newRecord.savedOn === record.savedOn;
16
+ });
17
+ };
18
+ const getData = records => {
19
+ return records.map(record => ({
20
+ id: record.id,
21
+ savedOn: record.savedOn
22
+ }));
23
+ };
24
+ export const RecordLockingProvider = props => {
25
+ const client = useApolloClient();
26
+ const [loading, setLoading] = useState(false);
27
+ const recordLocking = useMemo(() => {
28
+ return createRecordLocking({
29
+ client,
30
+ setLoading
31
+ });
32
+ }, []);
33
+ const [error, setError] = useStateIfMounted(null);
34
+ const [records, setRecords] = useStateIfMounted([]);
35
+ const setRecordsIfNeeded = useCallback(newRecords => {
36
+ const sameArray = isSameArray(getData(records), getData(newRecords));
37
+ if (sameArray) {
38
+ return;
39
+ }
40
+ setRecords(newRecords);
41
+ }, [records]);
42
+ const value = {
43
+ async updateEntryLock(params) {
44
+ const result = await recordLocking.updateEntryLock(params);
45
+ if (result.error) {
46
+ setError(result.error);
47
+ return result;
48
+ }
49
+ const target = result.data;
50
+ if (!target?.id) {
51
+ const error = {
52
+ message: "No data returned from server.",
53
+ code: "NO_DATA"
54
+ };
55
+ setError(error);
56
+ return {
57
+ error,
58
+ data: null
59
+ };
60
+ }
61
+ setRecords(prev => {
62
+ return prev.map(item => {
63
+ if (item.entryId === target.id) {
64
+ return {
65
+ ...item,
66
+ $locked: result.data
67
+ };
68
+ }
69
+ return item;
70
+ });
71
+ });
72
+ return result;
73
+ },
74
+ async unlockEntry(params) {
75
+ return await recordLocking.unlockEntry(params);
76
+ },
77
+ async unlockEntryForce(params) {
78
+ return await recordLocking.unlockEntry(params, true);
79
+ },
80
+ isLockExpired(input) {
81
+ return recordLocking.isLockExpired(input);
82
+ },
83
+ isRecordLocked(record) {
84
+ if (!record) {
85
+ return false;
86
+ }
87
+ return recordLocking.isRecordLocked(record);
88
+ },
89
+ getLockRecordEntry(id) {
90
+ return recordLocking.getLockRecordEntry(id);
91
+ },
92
+ removeEntryLock(params) {
93
+ return recordLocking.removeEntryLock(params);
94
+ },
95
+ async fetchLockRecord(params) {
96
+ try {
97
+ return await recordLocking.fetchLockRecord(params);
98
+ } catch (ex) {
99
+ return {
100
+ data: null,
101
+ error: ex
102
+ };
103
+ }
104
+ },
105
+ async fetchLockedEntryLockRecord(params) {
106
+ return recordLocking.fetchLockedEntryLockRecord(params);
107
+ },
108
+ async setRecords(folderId, type, newRecords) {
109
+ setRecordsIfNeeded(newRecords);
110
+ const result = await recordLocking.setRecords(folderId, type, newRecords);
111
+ if (!result) {
112
+ return;
113
+ }
114
+ setRecords(result);
115
+ },
116
+ error,
117
+ records,
118
+ loading
119
+ };
120
+ return /*#__PURE__*/React.createElement(RecordLockingContext.Provider, Object.assign({}, props, {
121
+ value: value
122
+ }));
123
+ };
124
+
125
+ //# sourceMappingURL=RecordLockingProvider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","useCallback","useMemo","useState","useApolloClient","createRecordLocking","useStateIfMounted","RecordLockingContext","createContext","isSameArray","existingRecords","newRecords","length","every","record","index","newRecord","id","savedOn","getData","records","map","RecordLockingProvider","props","client","loading","setLoading","recordLocking","error","setError","setRecords","setRecordsIfNeeded","sameArray","value","updateEntryLock","params","result","target","data","message","code","prev","item","entryId","$locked","unlockEntry","unlockEntryForce","isLockExpired","input","isRecordLocked","getLockRecordEntry","removeEntryLock","fetchLockRecord","ex","fetchLockedEntryLockRecord","folderId","type","createElement","Provider","Object","assign"],"sources":["RecordLockingProvider.tsx"],"sourcesContent":["import React, { useCallback, useMemo, useState } from \"react\";\nimport { useApolloClient } from \"@apollo/react-hooks\";\nimport { createRecordLocking } from \"~/domain/RecordLocking\";\nimport type {\n IFetchLockedEntryLockRecordParams,\n IFetchLockRecordParams,\n IPossiblyRecordLockingRecord,\n IRecordLockingContext,\n IRecordLockingError,\n IUnlockEntryParams,\n IUpdateEntryLockParams\n} from \"~/types\";\nimport { useStateIfMounted } from \"@webiny/app-admin\";\n\nexport interface IRecordLockingProviderProps {\n children: React.ReactNode;\n}\n\nexport const RecordLockingContext = React.createContext({} as unknown as IRecordLockingContext);\n\nconst isSameArray = (\n existingRecords: Pick<IPossiblyRecordLockingRecord[\"data\"], \"id\" | \"savedOn\">[],\n newRecords: Pick<IPossiblyRecordLockingRecord[\"data\"], \"id\" | \"savedOn\">[]\n): boolean => {\n if (existingRecords.length !== newRecords.length) {\n return false;\n }\n return existingRecords.every((record, index) => {\n const newRecord = newRecords[index];\n if (!newRecord) {\n return false;\n }\n\n return newRecord.id === record.id && newRecord.savedOn === record.savedOn;\n });\n};\n\nconst getData = (records: IPossiblyRecordLockingRecord[]) => {\n return records.map(record => ({\n id: record.id,\n savedOn: record.savedOn\n }));\n};\n\nexport const RecordLockingProvider = (props: IRecordLockingProviderProps) => {\n const client = useApolloClient();\n\n const [loading, setLoading] = useState(false);\n\n const recordLocking = useMemo(() => {\n return createRecordLocking({\n client,\n setLoading\n });\n }, []);\n\n const [error, setError] = useStateIfMounted<IRecordLockingError | null>(null);\n\n const [records, setRecords] = useStateIfMounted<IPossiblyRecordLockingRecord[]>([]);\n\n const setRecordsIfNeeded = useCallback(\n (newRecords: IPossiblyRecordLockingRecord[]) => {\n const sameArray = isSameArray(getData(records), getData(newRecords));\n if (sameArray) {\n return;\n }\n setRecords(newRecords);\n },\n [records]\n );\n\n const value: IRecordLockingContext = {\n async updateEntryLock(params: IUpdateEntryLockParams) {\n const result = await recordLocking.updateEntryLock(params);\n if (result.error) {\n setError(result.error);\n return result;\n }\n const target = result.data;\n if (!target?.id) {\n const error = {\n message: \"No data returned from server.\",\n code: \"NO_DATA\"\n };\n setError(error);\n return {\n error,\n data: null\n };\n }\n\n setRecords(prev => {\n return prev.map(item => {\n if (item.entryId === target.id) {\n return {\n ...item,\n $locked: result.data\n };\n }\n return item;\n });\n });\n return result;\n },\n async unlockEntry(params: IUnlockEntryParams) {\n return await recordLocking.unlockEntry(params);\n },\n async unlockEntryForce(params: IUnlockEntryParams) {\n return await recordLocking.unlockEntry(params, true);\n },\n isLockExpired(input: Date | string): boolean {\n return recordLocking.isLockExpired(input);\n },\n isRecordLocked(record) {\n if (!record) {\n return false;\n }\n return recordLocking.isRecordLocked(record);\n },\n getLockRecordEntry(id: string) {\n return recordLocking.getLockRecordEntry(id);\n },\n removeEntryLock(params: IUnlockEntryParams) {\n return recordLocking.removeEntryLock(params);\n },\n async fetchLockRecord(params: IFetchLockRecordParams) {\n try {\n return await recordLocking.fetchLockRecord(params);\n } catch (ex) {\n return {\n data: null,\n error: ex\n };\n }\n },\n async fetchLockedEntryLockRecord(params: IFetchLockedEntryLockRecordParams) {\n return recordLocking.fetchLockedEntryLockRecord(params);\n },\n async setRecords(folderId, type, newRecords) {\n setRecordsIfNeeded(newRecords);\n\n const result = await recordLocking.setRecords(folderId, type, newRecords);\n if (!result) {\n return;\n }\n setRecords(result);\n },\n error,\n records,\n loading\n };\n\n return <RecordLockingContext.Provider {...props} value={value} />;\n};\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ /// <reference types="react" />
2
+ export declare const UseContentEntriesListHookDecorator: () => import("react").JSX.Element;
@@ -0,0 +1,21 @@
1
+ import { useEffect } from "react";
2
+ import { ContentEntryListConfig } from "@webiny/app-headless-cms";
3
+ import { useRecordLocking } from "../../hooks";
4
+ export const UseContentEntriesListHookDecorator = ContentEntryListConfig.ContentEntries.useContentEntriesList.createDecorator(originalHook => {
5
+ return function RecordLockingUseContentEntriesList() {
6
+ const value = originalHook();
7
+ const recordLocking = useRecordLocking();
8
+ useEffect(() => {
9
+ if (!value.records) {
10
+ return;
11
+ }
12
+ recordLocking.setRecords(value.folderId, value.modelId, value.records);
13
+ }, [value.folderId, value.modelId, value.records, recordLocking]);
14
+ return {
15
+ ...value,
16
+ records: recordLocking.records
17
+ };
18
+ };
19
+ });
20
+
21
+ //# sourceMappingURL=UseContentEntriesListHookDecorator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["useEffect","ContentEntryListConfig","useRecordLocking","UseContentEntriesListHookDecorator","ContentEntries","useContentEntriesList","createDecorator","originalHook","RecordLockingUseContentEntriesList","value","recordLocking","records","setRecords","folderId","modelId"],"sources":["UseContentEntriesListHookDecorator.ts"],"sourcesContent":["import { useEffect } from \"react\";\nimport { ContentEntryListConfig } from \"@webiny/app-headless-cms\";\nimport { useRecordLocking } from \"~/hooks\";\n\nexport const UseContentEntriesListHookDecorator =\n ContentEntryListConfig.ContentEntries.useContentEntriesList.createDecorator(originalHook => {\n return function RecordLockingUseContentEntriesList() {\n const value = originalHook();\n const recordLocking = useRecordLocking();\n\n useEffect(() => {\n if (!value.records) {\n return;\n }\n recordLocking.setRecords(value.folderId, value.modelId, value.records);\n }, [value.folderId, value.modelId, value.records, recordLocking]);\n\n return {\n ...value,\n records: recordLocking.records\n };\n };\n });\n"],"mappings":"AAAA,SAASA,SAAS,QAAQ,OAAO;AACjC,SAASC,sBAAsB,QAAQ,0BAA0B;AACjE,SAASC,gBAAgB;AAEzB,OAAO,MAAMC,kCAAkC,GAC3CF,sBAAsB,CAACG,cAAc,CAACC,qBAAqB,CAACC,eAAe,CAACC,YAAY,IAAI;EACxF,OAAO,SAASC,kCAAkCA,CAAA,EAAG;IACjD,MAAMC,KAAK,GAAGF,YAAY,CAAC,CAAC;IAC5B,MAAMG,aAAa,GAAGR,gBAAgB,CAAC,CAAC;IAExCF,SAAS,CAAC,MAAM;MACZ,IAAI,CAACS,KAAK,CAACE,OAAO,EAAE;QAChB;MACJ;MACAD,aAAa,CAACE,UAAU,CAACH,KAAK,CAACI,QAAQ,EAAEJ,KAAK,CAACK,OAAO,EAAEL,KAAK,CAACE,OAAO,CAAC;IAC1E,CAAC,EAAE,CAACF,KAAK,CAACI,QAAQ,EAAEJ,KAAK,CAACK,OAAO,EAAEL,KAAK,CAACE,OAAO,EAAED,aAAa,CAAC,CAAC;IAEjE,OAAO;MACH,GAAGD,KAAK;MACRE,OAAO,EAAED,aAAa,CAACC;IAC3B,CAAC;EACL,CAAC;AACL,CAAC,CAAC","ignoreList":[]}
@@ -0,0 +1,9 @@
1
+ /// <reference types="react" />
2
+ /**
3
+ * When record locking mechanism is checking for records equality, it compares record `id` and `savedOn`.
4
+ * When you're updating ACO cache, you're just passing the new content entry value(s), and sometimes you might
5
+ * be doing it without the `savedOn`, and that causes record locking to keep its old state.
6
+ *
7
+ * This decorator ensures that calls to `updateRecordInCache` always include a `savedOn` timestamp.
8
+ */
9
+ export declare const UseRecordsDecorator: () => import("react").JSX.Element;
@@ -0,0 +1,25 @@
1
+ import { useRecords } from "@webiny/app-aco";
2
+
3
+ /**
4
+ * When record locking mechanism is checking for records equality, it compares record `id` and `savedOn`.
5
+ * When you're updating ACO cache, you're just passing the new content entry value(s), and sometimes you might
6
+ * be doing it without the `savedOn`, and that causes record locking to keep its old state.
7
+ *
8
+ * This decorator ensures that calls to `updateRecordInCache` always include a `savedOn` timestamp.
9
+ */
10
+ export const UseRecordsDecorator = useRecords.createDecorator(baseHook => {
11
+ return folderId => {
12
+ const hook = baseHook(folderId);
13
+ return {
14
+ ...hook,
15
+ updateRecordInCache(record) {
16
+ hook.updateRecordInCache({
17
+ ...record,
18
+ savedOn: new Date().toISOString()
19
+ });
20
+ }
21
+ };
22
+ };
23
+ });
24
+
25
+ //# sourceMappingURL=UseRecordsDecorator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["useRecords","UseRecordsDecorator","createDecorator","baseHook","folderId","hook","updateRecordInCache","record","savedOn","Date","toISOString"],"sources":["UseRecordsDecorator.tsx"],"sourcesContent":["import { useRecords } from \"@webiny/app-aco\";\n\n/**\n * When record locking mechanism is checking for records equality, it compares record `id` and `savedOn`.\n * When you're updating ACO cache, you're just passing the new content entry value(s), and sometimes you might\n * be doing it without the `savedOn`, and that causes record locking to keep its old state.\n *\n * This decorator ensures that calls to `updateRecordInCache` always include a `savedOn` timestamp.\n */\nexport const UseRecordsDecorator = useRecords.createDecorator(baseHook => {\n return (folderId?: string) => {\n const hook = baseHook(folderId);\n\n return {\n ...hook,\n updateRecordInCache(record: any) {\n hook.updateRecordInCache({\n ...record,\n savedOn: new Date().toISOString()\n });\n }\n };\n };\n});\n"],"mappings":"AAAA,SAASA,UAAU,QAAQ,iBAAiB;;AAE5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,MAAMC,mBAAmB,GAAGD,UAAU,CAACE,eAAe,CAACC,QAAQ,IAAI;EACtE,OAAQC,QAAiB,IAAK;IAC1B,MAAMC,IAAI,GAAGF,QAAQ,CAACC,QAAQ,CAAC;IAE/B,OAAO;MACH,GAAGC,IAAI;MACPC,mBAAmBA,CAACC,MAAW,EAAE;QAC7BF,IAAI,CAACC,mBAAmB,CAAC;UACrB,GAAGC,MAAM;UACTC,OAAO,EAAE,IAAIC,IAAI,CAAC,CAAC,CAACC,WAAW,CAAC;QACpC,CAAC,CAAC;MACN;IACJ,CAAC;EACL,CAAC;AACL,CAAC,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ /// <reference types="react" />
2
+ export declare const UseSaveEntryDecorator: () => import("react").JSX.Element;
@@ -0,0 +1,55 @@
1
+ import { useCallback } from "react";
2
+ import { useRecordLocking } from "../../hooks";
3
+ import { ContentEntryEditorConfig, useModel } from "@webiny/app-headless-cms";
4
+ import { useSnackbar } from "@webiny/app-admin";
5
+ const {
6
+ ContentEntry: {
7
+ ContentEntryForm
8
+ }
9
+ } = ContentEntryEditorConfig;
10
+ export const UseSaveEntryDecorator = ContentEntryForm.useContentEntryForm.createDecorator(originalHook => {
11
+ return function useRecordLockingUseSave() {
12
+ const hook = originalHook();
13
+ const {
14
+ fetchLockedEntryLockRecord,
15
+ updateEntryLock
16
+ } = useRecordLocking();
17
+ const {
18
+ showSnackbar
19
+ } = useSnackbar();
20
+ const {
21
+ model
22
+ } = useModel();
23
+ const {
24
+ entry
25
+ } = hook;
26
+ const saveEntry = useCallback(async (...params) => {
27
+ if (!entry.id) {
28
+ return hook.saveEntry(...params);
29
+ }
30
+ const result = await fetchLockedEntryLockRecord({
31
+ id: entry.id,
32
+ $lockingType: model.modelId
33
+ });
34
+ if (result?.lockedBy) {
35
+ const lockedBy = result.lockedBy;
36
+ showSnackbar(`It seems that the entry is locked by ${lockedBy.displayName || lockedBy.id}. You can't save your changes.`);
37
+ return null;
38
+ }
39
+ const saveResult = await hook.saveEntry(...params);
40
+ if (saveResult) {
41
+ await updateEntryLock({
42
+ id: saveResult.id,
43
+ $lockingType: model.modelId
44
+ });
45
+ }
46
+ return saveResult;
47
+ }, [entry?.id, model.modelId, updateEntryLock]);
48
+ return {
49
+ ...hook,
50
+ saveEntry
51
+ };
52
+ };
53
+ });
54
+
55
+ //# sourceMappingURL=UseSaveEntryDecorator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["useCallback","useRecordLocking","ContentEntryEditorConfig","useModel","useSnackbar","ContentEntry","ContentEntryForm","UseSaveEntryDecorator","useContentEntryForm","createDecorator","originalHook","useRecordLockingUseSave","hook","fetchLockedEntryLockRecord","updateEntryLock","showSnackbar","model","entry","saveEntry","params","id","result","$lockingType","modelId","lockedBy","displayName","saveResult"],"sources":["UseSaveEntryDecorator.tsx"],"sourcesContent":["import { useCallback } from \"react\";\nimport { useRecordLocking } from \"~/hooks\";\nimport { ContentEntryEditorConfig, useModel } from \"@webiny/app-headless-cms\";\nimport { useSnackbar } from \"@webiny/app-admin\";\n\nconst {\n ContentEntry: { ContentEntryForm }\n} = ContentEntryEditorConfig;\n\ntype SaveEntry = ReturnType<typeof ContentEntryForm.useContentEntryForm>[\"saveEntry\"];\n\nexport const UseSaveEntryDecorator = ContentEntryForm.useContentEntryForm.createDecorator(\n originalHook => {\n return function useRecordLockingUseSave() {\n const hook = originalHook();\n const { fetchLockedEntryLockRecord, updateEntryLock } = useRecordLocking();\n const { showSnackbar } = useSnackbar();\n const { model } = useModel();\n\n const { entry } = hook;\n\n const saveEntry: SaveEntry = useCallback(\n async (...params) => {\n if (!entry.id) {\n return hook.saveEntry(...params);\n }\n\n const result = await fetchLockedEntryLockRecord({\n id: entry.id,\n $lockingType: model.modelId\n });\n\n if (result?.lockedBy) {\n const lockedBy = result.lockedBy;\n showSnackbar(\n `It seems that the entry is locked by ${\n lockedBy.displayName || lockedBy.id\n }. You can't save your changes.`\n );\n return null;\n }\n\n const saveResult = await hook.saveEntry(...params);\n if (saveResult) {\n await updateEntryLock({\n id: saveResult.id,\n $lockingType: model.modelId\n });\n }\n\n return saveResult;\n },\n [entry?.id, model.modelId, updateEntryLock]\n );\n\n return {\n ...hook,\n saveEntry\n };\n };\n }\n);\n"],"mappings":"AAAA,SAASA,WAAW,QAAQ,OAAO;AACnC,SAASC,gBAAgB;AACzB,SAASC,wBAAwB,EAAEC,QAAQ,QAAQ,0BAA0B;AAC7E,SAASC,WAAW,QAAQ,mBAAmB;AAE/C,MAAM;EACFC,YAAY,EAAE;IAAEC;EAAiB;AACrC,CAAC,GAAGJ,wBAAwB;AAI5B,OAAO,MAAMK,qBAAqB,GAAGD,gBAAgB,CAACE,mBAAmB,CAACC,eAAe,CACrFC,YAAY,IAAI;EACZ,OAAO,SAASC,uBAAuBA,CAAA,EAAG;IACtC,MAAMC,IAAI,GAAGF,YAAY,CAAC,CAAC;IAC3B,MAAM;MAAEG,0BAA0B;MAAEC;IAAgB,CAAC,GAAGb,gBAAgB,CAAC,CAAC;IAC1E,MAAM;MAAEc;IAAa,CAAC,GAAGX,WAAW,CAAC,CAAC;IACtC,MAAM;MAAEY;IAAM,CAAC,GAAGb,QAAQ,CAAC,CAAC;IAE5B,MAAM;MAAEc;IAAM,CAAC,GAAGL,IAAI;IAEtB,MAAMM,SAAoB,GAAGlB,WAAW,CACpC,OAAO,GAAGmB,MAAM,KAAK;MACjB,IAAI,CAACF,KAAK,CAACG,EAAE,EAAE;QACX,OAAOR,IAAI,CAACM,SAAS,CAAC,GAAGC,MAAM,CAAC;MACpC;MAEA,MAAME,MAAM,GAAG,MAAMR,0BAA0B,CAAC;QAC5CO,EAAE,EAAEH,KAAK,CAACG,EAAE;QACZE,YAAY,EAAEN,KAAK,CAACO;MACxB,CAAC,CAAC;MAEF,IAAIF,MAAM,EAAEG,QAAQ,EAAE;QAClB,MAAMA,QAAQ,GAAGH,MAAM,CAACG,QAAQ;QAChCT,YAAY,CACR,wCACIS,QAAQ,CAACC,WAAW,IAAID,QAAQ,CAACJ,EAAE,gCAE3C,CAAC;QACD,OAAO,IAAI;MACf;MAEA,MAAMM,UAAU,GAAG,MAAMd,IAAI,CAACM,SAAS,CAAC,GAAGC,MAAM,CAAC;MAClD,IAAIO,UAAU,EAAE;QACZ,MAAMZ,eAAe,CAAC;UAClBM,EAAE,EAAEM,UAAU,CAACN,EAAE;UACjBE,YAAY,EAAEN,KAAK,CAACO;QACxB,CAAC,CAAC;MACN;MAEA,OAAOG,UAAU;IACrB,CAAC,EACD,CAACT,KAAK,EAAEG,EAAE,EAAEJ,KAAK,CAACO,OAAO,EAAET,eAAe,CAC9C,CAAC;IAED,OAAO;MACH,GAAGF,IAAI;MACPM;IACJ,CAAC;EACL,CAAC;AACL,CACJ,CAAC","ignoreList":[]}
@@ -0,0 +1,7 @@
1
+ import React from "react";
2
+ import type { RecordLockingSecurityPermission } from "../../types";
3
+ export interface RecordLockingPermissionsProps {
4
+ value: RecordLockingSecurityPermission[];
5
+ onChange: (value: RecordLockingSecurityPermission[]) => void;
6
+ }
7
+ export declare const RecordLockingPermissions: ({ value, onChange }: RecordLockingPermissionsProps) => React.JSX.Element;
@@ -0,0 +1,71 @@
1
+ import React, { useCallback, useMemo } from "react";
2
+ import { i18n } from "@webiny/app/i18n";
3
+ import { gridWithPaddingClass, PermissionInfo } from "@webiny/app-admin/components/Permissions";
4
+ import { Form } from "@webiny/form";
5
+ import { Grid, Select } from "@webiny/admin-ui";
6
+ const t = i18n.ns("app-record-locking/components/permissionRenderer");
7
+ const RECORD_LOCKING_PERMISSION = "recordLocking";
8
+ export const RecordLockingPermissions = ({
9
+ value,
10
+ onChange
11
+ }) => {
12
+ const onFormChange = useCallback(data => {
13
+ const newValue = value.filter(p => {
14
+ return p.name.startsWith(RECORD_LOCKING_PERMISSION) === false;
15
+ });
16
+ if (!data.canForceUnlock || data.canForceUnlock === "no") {
17
+ onChange(newValue);
18
+ return;
19
+ }
20
+ onChange([...newValue, {
21
+ name: "recordLocking",
22
+ canForceUnlock: "yes"
23
+ }]);
24
+ }, [value]);
25
+ const formData = useMemo(() => {
26
+ if (!Array.isArray(value)) {
27
+ return {};
28
+ }
29
+ const hasFullAccess = value.some(item => item.name === "*");
30
+ if (hasFullAccess) {
31
+ return {
32
+ canForceUnlock: "yes"
33
+ };
34
+ }
35
+ const permissions = value.filter(item => item.name.startsWith(RECORD_LOCKING_PERMISSION));
36
+ if (!permissions.length || !permissions.some(item => !!item.canForceUnlock)) {
37
+ return {};
38
+ }
39
+ return {
40
+ canForceUnlock: "yes"
41
+ };
42
+ }, []);
43
+ return /*#__PURE__*/React.createElement(Form, {
44
+ data: formData,
45
+ onChange: onFormChange
46
+ }, ({
47
+ Bind
48
+ }) => {
49
+ return /*#__PURE__*/React.createElement(React.Fragment, null, /*#__PURE__*/React.createElement(Grid, {
50
+ className: gridWithPaddingClass
51
+ }, /*#__PURE__*/React.createElement(Grid.Column, {
52
+ span: 6
53
+ }, /*#__PURE__*/React.createElement(PermissionInfo, {
54
+ title: t`Advanced Record Locking`
55
+ })), /*#__PURE__*/React.createElement(Grid.Column, {
56
+ span: 6
57
+ }, /*#__PURE__*/React.createElement(Bind, {
58
+ name: "canForceUnlock"
59
+ }, /*#__PURE__*/React.createElement(Select, {
60
+ options: [{
61
+ label: t`No Access`,
62
+ value: "no"
63
+ }, {
64
+ label: t`Full Access`,
65
+ value: "yes"
66
+ }]
67
+ })))));
68
+ });
69
+ };
70
+
71
+ //# sourceMappingURL=RecordLockingPermissions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","useCallback","useMemo","i18n","gridWithPaddingClass","PermissionInfo","Form","Grid","Select","t","ns","RECORD_LOCKING_PERMISSION","RecordLockingPermissions","value","onChange","onFormChange","data","newValue","filter","p","name","startsWith","canForceUnlock","formData","Array","isArray","hasFullAccess","some","item","permissions","length","createElement","Bind","Fragment","className","Column","span","title","options","label"],"sources":["RecordLockingPermissions.tsx"],"sourcesContent":["import React, { useCallback, useMemo } from \"react\";\nimport { i18n } from \"@webiny/app/i18n\";\nimport { gridWithPaddingClass, PermissionInfo } from \"@webiny/app-admin/components/Permissions\";\nimport { Form } from \"@webiny/form\";\nimport type { RecordLockingSecurityPermission } from \"~/types\";\nimport { Grid, Select } from \"@webiny/admin-ui\";\n\nconst t = i18n.ns(\"app-record-locking/components/permissionRenderer\");\n\nconst RECORD_LOCKING_PERMISSION = \"recordLocking\";\n\nexport interface RecordLockingPermissionsProps {\n value: RecordLockingSecurityPermission[];\n onChange: (value: RecordLockingSecurityPermission[]) => void;\n}\n\nexport const RecordLockingPermissions = ({ value, onChange }: RecordLockingPermissionsProps) => {\n const onFormChange = useCallback(\n (data: RecordLockingSecurityPermission) => {\n const newValue = value.filter(p => {\n return p.name.startsWith(RECORD_LOCKING_PERMISSION) === false;\n });\n\n if (!data.canForceUnlock || data.canForceUnlock === \"no\") {\n onChange(newValue);\n return;\n }\n\n onChange([\n ...newValue,\n {\n name: \"recordLocking\",\n canForceUnlock: \"yes\"\n }\n ]);\n },\n [value]\n );\n\n const formData = useMemo(() => {\n if (!Array.isArray(value)) {\n return {};\n }\n\n const hasFullAccess = value.some(item => item.name === \"*\");\n\n if (hasFullAccess) {\n return {\n canForceUnlock: \"yes\"\n };\n }\n\n const permissions = value.filter(item => item.name.startsWith(RECORD_LOCKING_PERMISSION));\n\n if (!permissions.length || !permissions.some(item => !!item.canForceUnlock)) {\n return {};\n }\n\n return {\n canForceUnlock: \"yes\"\n };\n }, []);\n\n return (\n <Form<RecordLockingSecurityPermission> data={formData} onChange={onFormChange}>\n {({ Bind }) => {\n return (\n <>\n <Grid className={gridWithPaddingClass}>\n <Grid.Column span={6}>\n <PermissionInfo title={t`Advanced Record Locking`} />\n </Grid.Column>\n <Grid.Column span={6}>\n <Bind name={\"canForceUnlock\"}>\n <Select\n options={[\n { label: t`No Access`, value: \"no\" },\n { label: t`Full Access`, value: \"yes\" }\n ]}\n ></Select>\n </Bind>\n </Grid.Column>\n </Grid>\n </>\n );\n }}\n </Form>\n );\n};\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import type { AdminAppPermissionRendererPlugin } from "@webiny/app-admin/types";
2
+ export declare const recordLockingPermissionRenderer: AdminAppPermissionRendererPlugin;
@@ -0,0 +1,21 @@
1
+ import React from "react";
2
+ import { ReactComponent as LockIcon } from "@webiny/icons/lock.svg";
3
+ import { RecordLockingPermissions } from "./RecordLockingPermissions";
4
+ import { Accordion } from "@webiny/admin-ui";
5
+ export const recordLockingPermissionRenderer = {
6
+ type: "admin-app-permissions-renderer",
7
+ name: "admin-app-permissions-renderer-record-locking",
8
+ render(props) {
9
+ return /*#__PURE__*/React.createElement(Accordion.Item, {
10
+ icon: /*#__PURE__*/React.createElement(Accordion.Item.Icon, {
11
+ label: "Record Locking",
12
+ icon: /*#__PURE__*/React.createElement(LockIcon, null)
13
+ }),
14
+ title: "Record Locking",
15
+ description: "Manage Record Locking permissions.",
16
+ "data-testid": "permission.recordLocking"
17
+ }, /*#__PURE__*/React.createElement(RecordLockingPermissions, props));
18
+ }
19
+ };
20
+
21
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","ReactComponent","LockIcon","RecordLockingPermissions","Accordion","recordLockingPermissionRenderer","type","name","render","props","createElement","Item","icon","Icon","label","title","description"],"sources":["index.tsx"],"sourcesContent":["import React from \"react\";\n\nimport type { AdminAppPermissionRendererPlugin } from \"@webiny/app-admin/types\";\nimport { ReactComponent as LockIcon } from \"@webiny/icons/lock.svg\";\nimport { RecordLockingPermissions } from \"./RecordLockingPermissions\";\nimport { Accordion } from \"@webiny/admin-ui\";\n\nexport const recordLockingPermissionRenderer: AdminAppPermissionRendererPlugin = {\n type: \"admin-app-permissions-renderer\",\n name: \"admin-app-permissions-renderer-record-locking\",\n render(props) {\n return (\n <Accordion.Item\n icon={<Accordion.Item.Icon label={\"Record Locking\"} icon={<LockIcon />} />}\n title={\"Record Locking\"}\n description={\"Manage Record Locking permissions.\"}\n data-testid={\"permission.recordLocking\"}\n >\n <RecordLockingPermissions {...props} />\n </Accordion.Item>\n );\n }\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AAGzB,SAASC,cAAc,IAAIC,QAAQ,QAAQ,wBAAwB;AACnE,SAASC,wBAAwB;AACjC,SAASC,SAAS,QAAQ,kBAAkB;AAE5C,OAAO,MAAMC,+BAAiE,GAAG;EAC7EC,IAAI,EAAE,gCAAgC;EACtCC,IAAI,EAAE,+CAA+C;EACrDC,MAAMA,CAACC,KAAK,EAAE;IACV,oBACIT,KAAA,CAAAU,aAAA,CAACN,SAAS,CAACO,IAAI;MACXC,IAAI,eAAEZ,KAAA,CAAAU,aAAA,CAACN,SAAS,CAACO,IAAI,CAACE,IAAI;QAACC,KAAK,EAAE,gBAAiB;QAACF,IAAI,eAAEZ,KAAA,CAAAU,aAAA,CAACR,QAAQ,MAAE;MAAE,CAAE,CAAE;MAC3Ea,KAAK,EAAE,gBAAiB;MACxBC,WAAW,EAAE,oCAAqC;MAClD,eAAa;IAA2B,gBAExChB,KAAA,CAAAU,aAAA,CAACP,wBAAwB,EAAKM,KAAQ,CAC1B,CAAC;EAEzB;AACJ,CAAC","ignoreList":[]}
@@ -0,0 +1,28 @@
1
+ import type { IRecordLocking } from "./abstractions/IRecordLocking";
2
+ import type { ApolloClient } from "apollo-client";
3
+ import type { IRecordLockingGetLockRecord } from "./abstractions/IRecordLockingGetLockRecord";
4
+ import type { IRecordLockingIsEntryLocked } from "./abstractions/IRecordLockingIsEntryLocked";
5
+ import type { IRecordLockingListLockRecords } from "./abstractions/IRecordLockingListLockRecords";
6
+ import type { IRecordLockingLockEntry } from "./abstractions/IRecordLockingLockEntry";
7
+ import type { IRecordLockingUnlockEntry } from "./abstractions/IRecordLockingUnlockEntry";
8
+ import type { IRecordLockingError, IRecordLockingRecord } from "../types";
9
+ import type { IRecordLockingUpdateEntryLock } from "./abstractions/IRecordLockingUpdateEntryLock";
10
+ import type { IRecordLockingGetLockedEntryLockRecord } from "./abstractions/IRecordLockingGetLockedEntryLockRecord";
11
+ export interface ICreateRecordLockingParams {
12
+ client: ApolloClient<any>;
13
+ setLoading: (loading: boolean) => void;
14
+ }
15
+ export interface IRecordLockingParams {
16
+ setLoading: (loading: boolean) => void;
17
+ getLockRecord: IRecordLockingGetLockRecord;
18
+ getLockedEntryLockRecord: IRecordLockingGetLockedEntryLockRecord;
19
+ isEntryLocked: IRecordLockingIsEntryLocked;
20
+ listLockRecords: IRecordLockingListLockRecords;
21
+ lockEntry: IRecordLockingLockEntry;
22
+ unlockEntry: IRecordLockingUnlockEntry;
23
+ updateEntryLock: IRecordLockingUpdateEntryLock;
24
+ }
25
+ export interface IOnErrorCb {
26
+ (error: IRecordLockingError): void;
27
+ }
28
+ export declare const createRecordLocking: <T extends IRecordLockingRecord>(config: ICreateRecordLockingParams) => IRecordLocking;