@kaspernj/api-maker 1.0.2011 → 1.0.2013

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 (413) hide show
  1. package/{.eslintrc.cjs → .eslintrc.js} +10 -1
  2. package/README.md +35 -0
  3. package/android/build.gradle +43 -0
  4. package/android/src/main/AndroidManifest.xml +2 -0
  5. package/android/src/main/java/expo/modules/api_maker/ApiMakerModule.kt +50 -0
  6. package/android/src/main/java/expo/modules/api_maker/ApiMakerView.kt +30 -0
  7. package/build/api.js +120 -0
  8. package/build/attribute-not-loaded-error.js +2 -0
  9. package/build/base-component.js +5 -0
  10. package/build/base-error.js +33 -0
  11. package/build/base-model/attribute.js +25 -0
  12. package/build/base-model/column.js +11 -0
  13. package/build/base-model/reflection.js +14 -0
  14. package/build/base-model/scope.js +11 -0
  15. package/build/base-model.js +828 -0
  16. package/build/bootstrap/attribute-row.js +135 -0
  17. package/build/bootstrap/attribute-rows.js +31 -0
  18. package/build/bootstrap/card.js +138 -0
  19. package/build/bootstrap/checkbox.js +97 -0
  20. package/build/bootstrap/checkboxes.js +183 -0
  21. package/build/bootstrap/index.js +2 -0
  22. package/build/bootstrap/input.js +129 -0
  23. package/build/bootstrap/invalid-feedback.js +27 -0
  24. package/build/bootstrap/paginate.js +150 -0
  25. package/build/bootstrap/radio-buttons.js +72 -0
  26. package/build/bootstrap/select.js +88 -0
  27. package/build/bootstrap/sort-link.js +131 -0
  28. package/build/cable-connection-pool.js +128 -0
  29. package/build/cable-subscription-pool.js +207 -0
  30. package/build/cable-subscription.js +21 -0
  31. package/build/cache-key-generator.js +81 -0
  32. package/build/can-can.js +134 -0
  33. package/build/channels-consumer.js +9 -0
  34. package/build/collection-loader.js +48 -0
  35. package/build/collection.js +280 -0
  36. package/build/command-submit-data.js +81 -0
  37. package/build/commands-pool.js +195 -0
  38. package/build/compose.js +9 -0
  39. package/build/config.js +62 -0
  40. package/build/custom-error.js +5 -0
  41. package/build/data-set-to-attributes.js +10 -0
  42. package/build/deserializer.js +56 -0
  43. package/build/destroy-error.js +5 -0
  44. package/build/devise.js +112 -0
  45. package/build/draggable-sort/controller.js +136 -0
  46. package/build/draggable-sort/index.js +114 -0
  47. package/build/draggable-sort/item.js +188 -0
  48. package/build/error-logger.js +101 -0
  49. package/build/error-messages.js +13 -0
  50. package/build/event-connection.js +32 -0
  51. package/build/event-emitter-listener.js +15 -0
  52. package/build/event-model-class.js +24 -0
  53. package/build/events.js +5 -0
  54. package/build/flash-message.js +79 -0
  55. package/build/form.js +83 -0
  56. package/build/history-expo.js +20 -0
  57. package/build/history-react-native.js +20 -0
  58. package/build/history.js +3 -0
  59. package/build/index.js +3 -0
  60. package/build/inputs/attachment.js +133 -0
  61. package/build/inputs/auto-submit.js +36 -0
  62. package/build/inputs/checkbox.js +138 -0
  63. package/build/inputs/checkboxes.js +112 -0
  64. package/build/inputs/id-for-component.js +15 -0
  65. package/build/inputs/input-wrapper.js +19 -0
  66. package/build/inputs/input.js +243 -0
  67. package/build/inputs/money.js +179 -0
  68. package/build/inputs/name-for-component.js +13 -0
  69. package/build/inputs/select.js +93 -0
  70. package/build/instance-of-class-name.js +23 -0
  71. package/build/is-expo.js +22 -0
  72. package/build/key-value-store.js +53 -0
  73. package/build/link.js +73 -0
  74. package/build/logger.js +30 -0
  75. package/build/modal.js +38 -0
  76. package/build/model-class-require.js +13 -0
  77. package/build/model-events.js +30 -0
  78. package/build/model-name.js +28 -0
  79. package/build/model-prop-type.js +101 -0
  80. package/build/model-recipes-loader.js +32 -0
  81. package/build/model-recipes-model-loader.js +357 -0
  82. package/build/models-response-reader.js +40 -0
  83. package/build/models.js +8 -0
  84. package/build/money-formatter.js +78 -0
  85. package/build/not-loaded-error.js +2 -0
  86. package/build/params.js +61 -0
  87. package/build/preloaded.js +38 -0
  88. package/build/resize-observer.js +11 -0
  89. package/build/result.js +13 -0
  90. package/build/router/route.js +270 -0
  91. package/build/router/switch.js +72 -0
  92. package/build/router.js +59 -0
  93. package/build/routes-native.js +202 -0
  94. package/build/routes.js +50 -0
  95. package/build/run-last.js +35 -0
  96. package/build/serializer.js +63 -0
  97. package/build/services.js +20 -0
  98. package/build/session-status-updater.js +134 -0
  99. package/build/source-maps-loader.js +184 -0
  100. package/build/super-admin/config-reader.js +85 -0
  101. package/build/super-admin/edit-page/edit-attribute-checkbox.js +85 -0
  102. package/build/super-admin/edit-page/edit-attribute-content.js +55 -0
  103. package/build/super-admin/edit-page/edit-attribute-input.js +79 -0
  104. package/build/super-admin/edit-page/edit-attribute.js +77 -0
  105. package/build/super-admin/edit-page.js +123 -0
  106. package/build/super-admin/has-edit-config.js +12 -0
  107. package/build/super-admin/index-page.js +25 -0
  108. package/build/super-admin/index.js +181 -0
  109. package/build/super-admin/layout/header/index.js +161 -0
  110. package/build/super-admin/layout/index.js +164 -0
  111. package/build/super-admin/layout/menu/index.js +206 -0
  112. package/build/super-admin/layout/menu/menu-content.js +46 -0
  113. package/build/super-admin/layout/menu/menu-item.js +84 -0
  114. package/build/super-admin/layout/no-access.js +27 -0
  115. package/build/super-admin/model-class-table.js +61 -0
  116. package/build/super-admin/models.js +8 -0
  117. package/build/super-admin/show-nav.js +51 -0
  118. package/build/super-admin/show-page/belongs-to-attribute-row.js +29 -0
  119. package/build/super-admin/show-page/index.js +141 -0
  120. package/build/super-admin/show-reflection-actions.js +47 -0
  121. package/build/super-admin/show-reflection-link.js +48 -0
  122. package/build/super-admin/show-reflection-page.js +48 -0
  123. package/build/table/column-content.js +117 -0
  124. package/build/table/column-identifier.js +18 -0
  125. package/build/table/column-visible.js +7 -0
  126. package/build/table/components/column.js +21 -0
  127. package/build/table/components/flat-list.js +21 -0
  128. package/build/table/components/header.js +26 -0
  129. package/build/table/components/row.js +26 -0
  130. package/build/table/filters/attribute-element.js +44 -0
  131. package/build/table/filters/filter-form.js +402 -0
  132. package/build/table/filters/filter.js +90 -0
  133. package/build/table/filters/index.js +184 -0
  134. package/build/table/filters/load-search-modal.js +175 -0
  135. package/build/table/filters/reflection-element.js +37 -0
  136. package/build/table/filters/save-search-modal.js +88 -0
  137. package/build/table/filters/scope-element.js +42 -0
  138. package/build/table/header-column-content.js +68 -0
  139. package/build/table/header-column.js +144 -0
  140. package/build/table/header-select.js +105 -0
  141. package/build/table/model-callback-args.js +10 -0
  142. package/build/table/model-column.js +97 -0
  143. package/build/table/model-row.js +193 -0
  144. package/build/table/select-calculator.js +63 -0
  145. package/build/table/settings/column-row.js +97 -0
  146. package/build/table/settings/download-action.js +79 -0
  147. package/build/table/settings/index.js +92 -0
  148. package/build/table/table-settings.js +219 -0
  149. package/build/table/table.js +972 -0
  150. package/build/table/use-sorting.js +34 -0
  151. package/build/table/widths.js +75 -0
  152. package/build/table/worker-plugins-check-all-checkbox.js +125 -0
  153. package/build/table/worker-plugins-checkbox.js +115 -0
  154. package/build/translated-attributes.js +10 -0
  155. package/build/translated-collections.js +12 -0
  156. package/build/updated-attribute.js +76 -0
  157. package/build/url-encode.js +15 -0
  158. package/build/use-breakpoint.js +71 -0
  159. package/build/use-can-can.js +55 -0
  160. package/build/use-collection.js +225 -0
  161. package/build/use-created-event.js +52 -0
  162. package/build/use-current-user.js +76 -0
  163. package/build/use-destroyed-event.js +60 -0
  164. package/build/use-event-emitter.js +13 -0
  165. package/build/use-event-listener.js +16 -0
  166. package/build/use-input.js +137 -0
  167. package/build/use-model-event.js +60 -0
  168. package/build/use-model.js +169 -0
  169. package/build/use-resize-observer.js +23 -0
  170. package/build/use-router.js +104 -0
  171. package/build/use-screen-layout.js +39 -0
  172. package/build/use-styles.js +41 -0
  173. package/build/use-updated-event.js +60 -0
  174. package/build/use-validation-errors.js +30 -0
  175. package/build/utils/card.js +51 -0
  176. package/build/utils/checkbox.js +78 -0
  177. package/build/utils/checkboxes.js +163 -0
  178. package/build/utils/default-style.js +18 -0
  179. package/build/utils/icon.js +23 -0
  180. package/build/utils/invalid-feedback.js +19 -0
  181. package/build/utils/modal.js +62 -0
  182. package/build/utils/text.js +44 -0
  183. package/build/validation-error.js +28 -0
  184. package/build/validation-errors.js +98 -0
  185. package/build/with-api-maker.js +29 -0
  186. package/build/with-collection.js +14 -0
  187. package/build/with-current-user.js +10 -0
  188. package/build/with-model.js +15 -0
  189. package/build/with-router.js +22 -0
  190. package/expo-module.config.json +17 -0
  191. package/ios/ApiMaker.podspec +29 -0
  192. package/ios/ApiMakerModule.swift +48 -0
  193. package/ios/ApiMakerView.swift +38 -0
  194. package/package.json +49 -46
  195. package/src/{api.mjs → api.js} +38 -37
  196. package/src/base-component.jsx +5 -0
  197. package/src/{base-error.mjs → base-error.js} +4 -6
  198. package/src/base-model/attribute.js +33 -0
  199. package/src/base-model/column.js +13 -0
  200. package/src/base-model/reflection.js +15 -0
  201. package/src/base-model/scope.js +12 -0
  202. package/src/{base-model.mjs → base-model.js} +302 -144
  203. package/src/bootstrap/attribute-row.jsx +135 -0
  204. package/src/bootstrap/attribute-rows.jsx +27 -0
  205. package/src/bootstrap/card.jsx +149 -0
  206. package/src/bootstrap/checkbox.jsx +86 -0
  207. package/src/bootstrap/checkboxes.jsx +185 -0
  208. package/src/bootstrap/index.js +0 -0
  209. package/src/bootstrap/input.jsx +173 -0
  210. package/src/bootstrap/invalid-feedback.jsx +31 -0
  211. package/src/bootstrap/paginate.jsx +187 -0
  212. package/src/bootstrap/radio-buttons.jsx +87 -0
  213. package/src/bootstrap/select.jsx +110 -0
  214. package/src/bootstrap/sort-link.jsx +106 -0
  215. package/src/{cable-connection-pool.mjs → cable-connection-pool.js} +16 -36
  216. package/src/{cable-subscription-pool.mjs → cable-subscription-pool.js} +26 -21
  217. package/src/{cable-subscription.mjs → cable-subscription.js} +6 -4
  218. package/src/cache-key-generator.js +100 -0
  219. package/src/{can-can.mjs → can-can.js} +13 -12
  220. package/src/channels-consumer.js +10 -0
  221. package/src/collection-loader.jsx +58 -224
  222. package/src/{collection.mjs → collection.js} +98 -44
  223. package/src/{command-submit-data.mjs → command-submit-data.js} +2 -7
  224. package/src/{commands-pool.mjs → commands-pool.js} +57 -41
  225. package/src/compose.js +11 -0
  226. package/src/{config.mjs → config.js} +6 -2
  227. package/src/{custom-error.mjs → custom-error.js} +1 -1
  228. package/src/data-set-to-attributes.js +13 -0
  229. package/src/{deserializer.mjs → deserializer.js} +3 -3
  230. package/src/destroy-error.js +7 -0
  231. package/src/devise.js +129 -0
  232. package/src/draggable-sort/controller.js +137 -0
  233. package/src/draggable-sort/index.jsx +108 -0
  234. package/src/draggable-sort/item.jsx +174 -0
  235. package/src/{error-logger.mjs → error-logger.js} +12 -7
  236. package/src/{error-messages.mjs → error-messages.js} +1 -0
  237. package/src/event-connection.jsx +1 -1
  238. package/src/event-emitter-listener.jsx +8 -27
  239. package/src/event-model-class.jsx +2 -4
  240. package/src/events.js +7 -0
  241. package/src/flash-message.js +70 -0
  242. package/src/form.jsx +91 -0
  243. package/src/history-expo.js +23 -0
  244. package/src/history-react-native.js +25 -0
  245. package/src/history.js +3 -0
  246. package/src/index.js +3 -0
  247. package/src/inputs/attachment.jsx +108 -0
  248. package/src/inputs/auto-submit.js +37 -0
  249. package/src/inputs/checkbox.jsx +125 -0
  250. package/src/inputs/checkboxes.jsx +116 -0
  251. package/src/inputs/id-for-component.js +15 -0
  252. package/src/inputs/input-wrapper.jsx +16 -0
  253. package/src/inputs/input.jsx +260 -0
  254. package/src/inputs/money.jsx +187 -0
  255. package/src/inputs/name-for-component.js +15 -0
  256. package/src/inputs/select.jsx +104 -0
  257. package/src/is-expo.js +18 -0
  258. package/src/{key-value-store.mjs → key-value-store.js} +1 -1
  259. package/src/link.jsx +54 -9
  260. package/src/logger.js +38 -0
  261. package/src/modal.jsx +37 -0
  262. package/src/model-class-require.js +17 -0
  263. package/src/{model-events.mjs → model-events.js} +1 -1
  264. package/src/{model-name.mjs → model-name.js} +8 -8
  265. package/src/{model-recipes-loader.mjs → model-recipes-loader.js} +1 -1
  266. package/src/{model-recipes-model-loader.mjs → model-recipes-model-loader.js} +5 -5
  267. package/src/{models-response-reader.mjs → models-response-reader.js} +3 -4
  268. package/src/models.js +7 -0
  269. package/src/{models.mjs.erb → models.js.erb} +6 -3
  270. package/src/{money-formatter.mjs → money-formatter.js} +5 -4
  271. package/src/{params.mjs → params.js} +17 -7
  272. package/src/{preloaded.mjs → preloaded.js} +2 -2
  273. package/src/resize-observer.jsx +10 -0
  274. package/src/result.js +13 -0
  275. package/src/router/route.jsx +246 -0
  276. package/src/router/switch.jsx +76 -0
  277. package/src/router.jsx +23 -21
  278. package/src/{routes-native.mjs → routes-native.js} +12 -6
  279. package/src/{routes.mjs → routes.js} +1 -1
  280. package/src/run-last.js +39 -0
  281. package/src/{serializer.mjs → serializer.js} +2 -2
  282. package/src/{services.mjs → services.js} +1 -1
  283. package/src/session-status-updater.js +172 -0
  284. package/src/{source-maps-loader.mjs → source-maps-loader.js} +54 -28
  285. package/src/super-admin/config-reader.jsx +93 -0
  286. package/src/super-admin/edit-page/edit-attribute-checkbox.jsx +81 -0
  287. package/src/super-admin/edit-page/edit-attribute-content.jsx +57 -0
  288. package/src/super-admin/edit-page/edit-attribute-input.jsx +71 -0
  289. package/src/super-admin/edit-page/edit-attribute.jsx +81 -0
  290. package/src/super-admin/edit-page.jsx +117 -0
  291. package/src/super-admin/has-edit-config.js +15 -0
  292. package/src/super-admin/index-page.jsx +23 -0
  293. package/src/super-admin/index.jsx +197 -0
  294. package/src/super-admin/layout/header/index.jsx +144 -0
  295. package/src/super-admin/layout/header/style.scss +45 -0
  296. package/src/super-admin/layout/index.jsx +154 -0
  297. package/src/super-admin/layout/menu/index.jsx +171 -0
  298. package/src/super-admin/layout/menu/menu-content.jsx +43 -0
  299. package/src/super-admin/layout/menu/menu-item.jsx +70 -0
  300. package/src/super-admin/layout/menu/style.scss +11 -0
  301. package/src/super-admin/layout/no-access.jsx +26 -0
  302. package/src/super-admin/model-class-table.jsx +66 -0
  303. package/src/super-admin/models.js +11 -0
  304. package/src/super-admin/show-nav.jsx +44 -0
  305. package/src/super-admin/show-page/belongs-to-attribute-row.jsx +30 -0
  306. package/src/super-admin/show-page/index.jsx +141 -0
  307. package/src/super-admin/show-reflection-actions.jsx +49 -0
  308. package/src/super-admin/show-reflection-link.jsx +40 -0
  309. package/src/super-admin/show-reflection-page.jsx +47 -0
  310. package/src/super-admin/stylesheets/variables.scss +11 -0
  311. package/src/table/column-content.jsx +122 -0
  312. package/src/table/column-identifier.js +23 -0
  313. package/src/table/column-visible.js +7 -0
  314. package/src/table/components/column.jsx +19 -0
  315. package/src/table/components/flat-list.jsx +19 -0
  316. package/src/table/components/header.jsx +21 -0
  317. package/src/table/components/row.jsx +23 -0
  318. package/src/table/filters/attribute-element.jsx +47 -0
  319. package/src/table/filters/filter-form.jsx +407 -0
  320. package/src/table/filters/filter.jsx +70 -0
  321. package/src/table/filters/index.jsx +170 -0
  322. package/src/table/filters/load-search-modal.jsx +146 -0
  323. package/src/table/filters/reflection-element.jsx +38 -0
  324. package/src/table/filters/save-search-modal.jsx +74 -0
  325. package/src/table/filters/scope-element.jsx +44 -0
  326. package/src/table/header-column-content.jsx +55 -0
  327. package/src/table/header-column.jsx +129 -0
  328. package/src/table/header-select.jsx +73 -0
  329. package/src/table/model-callback-args.js +10 -0
  330. package/src/table/model-column.jsx +82 -0
  331. package/src/table/model-row.jsx +136 -0
  332. package/src/table/select-calculator.js +65 -0
  333. package/src/table/settings/column-row.jsx +93 -0
  334. package/src/table/settings/download-action.jsx +68 -0
  335. package/src/table/settings/index.jsx +65 -0
  336. package/src/table/table-settings.js +263 -0
  337. package/src/table/table.jsx +950 -0
  338. package/src/table/use-sorting.js +35 -0
  339. package/src/table/variables.scss +11 -0
  340. package/src/table/widths.jsx +87 -0
  341. package/src/table/worker-plugins-check-all-checkbox.jsx +112 -0
  342. package/src/table/worker-plugins-checkbox.jsx +104 -0
  343. package/src/translated-attributes.js +11 -0
  344. package/src/{translated-collections.mjs → translated-collections.js} +3 -2
  345. package/src/updated-attribute.jsx +1 -1
  346. package/src/url-encode.js +18 -0
  347. package/src/use-breakpoint.js +87 -0
  348. package/src/use-can-can.js +55 -0
  349. package/src/use-collection.js +280 -0
  350. package/src/use-created-event.js +55 -0
  351. package/src/use-current-user.js +88 -0
  352. package/src/use-destroyed-event.js +60 -0
  353. package/src/use-event-emitter.js +15 -0
  354. package/src/use-event-listener.js +19 -0
  355. package/src/use-input.js +157 -0
  356. package/src/use-model-event.js +60 -0
  357. package/src/use-model.js +189 -0
  358. package/src/use-resize-observer.js +24 -0
  359. package/src/use-router.jsx +128 -0
  360. package/src/use-screen-layout.js +49 -0
  361. package/src/use-styles.js +50 -0
  362. package/src/use-updated-event.js +60 -0
  363. package/src/use-validation-errors.js +33 -0
  364. package/src/utils/card.jsx +49 -0
  365. package/src/utils/checkbox.jsx +80 -0
  366. package/src/utils/checkboxes.jsx +158 -0
  367. package/src/utils/default-style.jsx +25 -0
  368. package/src/utils/icon.jsx +21 -0
  369. package/src/utils/invalid-feedback.jsx +20 -0
  370. package/src/utils/modal.jsx +60 -0
  371. package/src/utils/text.jsx +38 -0
  372. package/src/{validation-error.mjs → validation-error.js} +9 -5
  373. package/src/{validation-errors.mjs → validation-errors.js} +15 -27
  374. package/src/with-api-maker.jsx +31 -0
  375. package/src/with-collection.jsx +17 -0
  376. package/src/with-current-user.jsx +7 -25
  377. package/src/with-model.jsx +16 -0
  378. package/src/with-router.jsx +7 -129
  379. package/__tests__/base-model.test.js +0 -71
  380. package/__tests__/cable-connection-pool.test.js +0 -227
  381. package/__tests__/cable-subscription-pool.test.js +0 -26
  382. package/__tests__/can-can.test.js +0 -34
  383. package/__tests__/collection.test.js +0 -51
  384. package/__tests__/custom-error.test.js +0 -13
  385. package/__tests__/model-name.test.js +0 -34
  386. package/__tests__/model-prop-type.test.js +0 -113
  387. package/__tests__/params.test.js +0 -40
  388. package/__tests__/routes-native.test.js +0 -103
  389. package/__tests__/routes.test.js +0 -46
  390. package/__tests__/serializer.test.js +0 -30
  391. package/__tests__/support/task.js +0 -27
  392. package/__tests__/support/user.js +0 -32
  393. package/index.js +0 -1
  394. package/jest.config.js +0 -4
  395. package/src/can-can-loader.jsx +0 -54
  396. package/src/channels-consumer.mjs +0 -3
  397. package/src/destroy-error.mjs +0 -7
  398. package/src/devise.mjs +0 -122
  399. package/src/event-created.jsx +0 -65
  400. package/src/event-destroyed.jsx +0 -29
  401. package/src/event-listener.jsx +0 -38
  402. package/src/event-updated.jsx +0 -74
  403. package/src/logger.mjs +0 -25
  404. package/src/model-class-require.mjs +0 -10
  405. package/src/model-load-wrapper.jsx +0 -116
  406. package/src/result.mjs +0 -29
  407. package/src/session-status-updater.mjs +0 -116
  408. package/webpack.config.js +0 -15
  409. /package/src/{attribute-not-loaded-error.mjs → attribute-not-loaded-error.js} +0 -0
  410. /package/src/{instance-of-class-name.mjs → instance-of-class-name.js} +0 -0
  411. /package/src/{model-prop-type.mjs → model-prop-type.js} +0 -0
  412. /package/src/{not-loaded-error.mjs → not-loaded-error.js} +0 -0
  413. /package/src/{translated-collections-data.mjs.erb → translated-collections-data.js.erb} +0 -0
@@ -0,0 +1,23 @@
1
+ import BaseComponent from "../../base-component"
2
+ import memo from "set-state-compare/src/memo"
3
+ import React from "react"
4
+ import {shapeComponent} from "set-state-compare/src/shape-component"
5
+ import useBreakpoint from "../../use-breakpoint"
6
+ import {View} from "react-native"
7
+
8
+ export default memo(shapeComponent(class SharedTableRow extends BaseComponent {
9
+ render() {
10
+ const {style, ...restProps} = this.props
11
+ const {name: breakpoint, smDown} = useBreakpoint()
12
+ const actualStyle = Object.assign(
13
+ {
14
+ flexDirection: breakpoint == "sm" || smDown ? "column" : "row"
15
+ },
16
+ style
17
+ )
18
+
19
+ return (
20
+ <View style={actualStyle} {...restProps} />
21
+ )
22
+ }
23
+ }))
@@ -0,0 +1,47 @@
1
+ import BaseComponent from "../../base-component"
2
+ import {digg} from "diggerize"
3
+ import PropTypes from "prop-types"
4
+ import PropTypesExact from "prop-types-exact"
5
+ import memo from "set-state-compare/src/memo"
6
+ import {Pressable} from "react-native"
7
+ import React from "react"
8
+ import {shapeComponent} from "set-state-compare/src/shape-component"
9
+ import Text from "../../utils/text"
10
+
11
+ export default memo(shapeComponent(class AttributeElement extends BaseComponent {
12
+ static propTypes = PropTypesExact({
13
+ active: PropTypes.bool.isRequired,
14
+ attribute: PropTypes.object.isRequired,
15
+ modelClassName: PropTypes.string.isRequired,
16
+ fikter: PropTypes.object,
17
+ onClick: PropTypes.func.isRequired
18
+ })
19
+
20
+ render() {
21
+ const {active, attribute, modelClassName} = this.p
22
+ const style = {}
23
+
24
+ if (active) style.fontWeight = "bold"
25
+
26
+ return (
27
+ <Pressable
28
+ dataSet={{
29
+ class: "attribute-element",
30
+ attributeName: digg(attribute, "attributeName"),
31
+ modelClass: modelClassName
32
+ }}
33
+ onPress={this.tt.onAttributeClicked}
34
+ >
35
+ <Text style={style}>
36
+ {digg(attribute, "humanName")}
37
+ </Text>
38
+ </Pressable>
39
+ )
40
+ }
41
+
42
+ onAttributeClicked = (e) => {
43
+ e.preventDefault()
44
+
45
+ this.p.onClick({attribute: this.p.attribute})
46
+ }
47
+ }))
@@ -0,0 +1,407 @@
1
+ import {useMemo, useRef} from "react"
2
+ import {ActivityIndicator, View} from "react-native"
3
+ import AttributeElement from "./attribute-element"
4
+ import BaseComponent from "../../base-component"
5
+ import {digg, digs} from "diggerize"
6
+ import * as inflection from "inflection"
7
+ import {Form} from "../../form"
8
+ import Input from "../../inputs/input"
9
+ import memo from "set-state-compare/src/memo"
10
+ import Params from "../../params"
11
+ import PropTypes from "prop-types"
12
+ import PropTypesExact from "prop-types-exact"
13
+ import React from "react"
14
+ import ReflectionElement from "./reflection-element"
15
+ import ScopeElement from "./scope-element"
16
+ import Select from "../../inputs/select"
17
+ import Services from "../../services"
18
+ import {shapeComponent} from "set-state-compare/src/shape-component"
19
+ import Text from "../../utils/text"
20
+ import useI18n from "i18n-on-steroids/src/use-i18n"
21
+
22
+ export default memo(shapeComponent(class ApiMakerTableFiltersFilterForm extends BaseComponent {
23
+ static propTypes = PropTypesExact({
24
+ filter: PropTypes.object,
25
+ modelClass: PropTypes.func.isRequired,
26
+ onApplyClicked: PropTypes.func.isRequired,
27
+ querySearchName: PropTypes.string.isRequired
28
+ })
29
+
30
+ setup() {
31
+ const {t} = useI18n({namespace: "js.api_maker.table.filters.filter_form"})
32
+
33
+ this.useStates({
34
+ associations: null,
35
+ attribute: undefined,
36
+ actualCurrentModelClass: () => ({modelClass: this.p.modelClass}),
37
+ loading: 0,
38
+ modelClassName: digg(this.p.modelClass.modelClassData(), "className"),
39
+ path: [],
40
+ predicate: undefined,
41
+ predicates: undefined,
42
+ ransackableAttributes: undefined,
43
+ ransackableScopes: undefined,
44
+ scope: this.props.filter.sc,
45
+ value: this.props.filter.v
46
+ })
47
+
48
+ this.setInstance({
49
+ t,
50
+ valueInputRef: useRef(),
51
+ })
52
+
53
+ useMemo(() => {
54
+ this.loadRansackPredicates()
55
+
56
+ if (this.props.filter.v) {
57
+ this.loadInitialValuesWithLoadingIndicator()
58
+ }
59
+ }, [])
60
+
61
+ useMemo(() => {
62
+ this.loadAssociations()
63
+ }, [this.s.modelClassName])
64
+ }
65
+
66
+ currentModelClass = () => digg(this.s.actualCurrentModelClass, "modelClass")
67
+
68
+ parseAssociationData(result) {
69
+ const associations = result.associations.map(({human_name, model_class_name, reflection_name, resource}) => ({
70
+ humanName: human_name,
71
+ modelClassName: model_class_name,
72
+ reflectionName: inflection.camelize(reflection_name, true),
73
+ resource
74
+ }))
75
+ const ransackableAttributes = digg(result, "ransackable_attributes").map(({attribute_name: attributeName, human_name: humanName}) => ({
76
+ attributeName, humanName
77
+ }))
78
+ const ransackableScopes = digg(result, "ransackable_scopes")
79
+
80
+ return {associations, ransackableAttributes, ransackableScopes}
81
+ }
82
+
83
+ async loadAssociations() {
84
+ this.increaseLoading()
85
+
86
+ try {
87
+ if (!this.s.modelClassName) throw new Error("'modelClassName' not set in state")
88
+
89
+ const result = await Services.current().sendRequest("Models::Associations", {model_class_name: this.s.modelClassName})
90
+ const {associations, ransackableAttributes, ransackableScopes} = this.parseAssociationData(result)
91
+
92
+ this.setState({associations, ransackableAttributes, ransackableScopes})
93
+ } finally {
94
+ this.decreaseLoading()
95
+ }
96
+ }
97
+
98
+ decreaseLoading = () => this.setState((prevState) => ({loading: prevState.loading - 1}))
99
+ increaseLoading = () => this.setState((prevState) => ({loading: prevState.loading + 1}))
100
+
101
+ async loadInitialValuesWithLoadingIndicator() {
102
+ try {
103
+ this.increaseLoading()
104
+ await this.loadInitialValues()
105
+ } finally {
106
+ this.decreaseLoading()
107
+ }
108
+ }
109
+
110
+ async loadInitialValues() {
111
+ if (!this.s.modelClassName) throw new Error("'modelClassName' not set in state")
112
+
113
+ let result = await Services.current().sendRequest("Models::Associations", {model_class_name: this.s.modelClassName})
114
+ let data = this.parseAssociationData(result)
115
+ let modelClassName = this.s.modelClassName
116
+ const path = []
117
+
118
+ for (const pathPart of this.props.filter.p) {
119
+ const reflection = data.associations.find((association) => digg(association, "reflectionName") == inflection.camelize(pathPart, true))
120
+
121
+ if (!reflection) throw new Error(`Couldn't find association by that name ${this.s.modelClassName}#${pathPart}`)
122
+
123
+ modelClassName = digg(reflection, "modelClassName")
124
+
125
+ if (!modelClassName) {
126
+ const pathNames = path.map((pathPart) => pathPart.name()).join(".")
127
+
128
+ throw new Error(`No model class name from ${pathNames}.${reflection.name()}`)
129
+ }
130
+
131
+ result = await Services.current().sendRequest("Models::Associations", {model_class_name: modelClassName})
132
+ data = this.parseAssociationData(result)
133
+
134
+ path.push(reflection)
135
+ }
136
+
137
+ const {ransackableAttributes} = data
138
+ const attribute = this.p.filter.a
139
+ const ransackableAttribute = ransackableAttributes.find((ransackableAttribute) => digg(ransackableAttribute, "attributeName") == attribute)
140
+
141
+ this.setState({attribute: ransackableAttribute, modelClassName, path})
142
+ }
143
+
144
+ async loadRansackPredicates() {
145
+ this.increaseLoading()
146
+
147
+ try {
148
+ const response = await Services.current().sendRequest("Ransack::Predicates")
149
+ const predicates = digg(response, "predicates")
150
+ let currentPredicate
151
+
152
+ if (this.props.filter.pre) {
153
+ currentPredicate = predicates.find((predicate) => predicate.name == this.props.filter.pre)
154
+ }
155
+
156
+ this.setState({
157
+ predicate: currentPredicate,
158
+ predicates
159
+ })
160
+ } finally {
161
+ this.decreaseLoading()
162
+ }
163
+ }
164
+
165
+ render() {
166
+ const {t, valueInputRef} = this.tt
167
+ const {attribute, path, predicate, predicates, scope, value} = this.s
168
+ let submitEnabled = false
169
+
170
+ if (attribute && predicate) {
171
+ submitEnabled = true
172
+ } else if (scope) {
173
+ submitEnabled = true
174
+ }
175
+
176
+ return (
177
+ <View dataSet={{class: "api-maker--table--filters--filter-form"}} style={{minWidth: 50, minHeight: 50}}>
178
+ <Form onSubmit={this.tt.onSubmit}>
179
+ <View style={{flexDirection: "row"}}>
180
+ {path.map(({humanName, reflectionName}, pathPartIndex) =>
181
+ <View key={`${pathPartIndex}-${reflectionName}`} style={{flexDirection: "row"}}>
182
+ {pathPartIndex > 0 &&
183
+ <Text style={{marginRight: 5, marginLeft: 5}}>
184
+ -
185
+ </Text>
186
+ }
187
+ <Text>
188
+ {humanName}
189
+ </Text>
190
+ </View>
191
+ )}
192
+ </View>
193
+ <View style={{flexDirection: "row"}}>
194
+ <View>
195
+ {this.s.associations && this.sortedReflectionsByName(this.s.associations).map((reflection) =>
196
+ <ReflectionElement
197
+ key={reflection.reflectionName}
198
+ modelClassName={this.s.modelClassName}
199
+ onClick={this.tt.onReflectionClicked}
200
+ reflection={reflection}
201
+ />
202
+ )}
203
+ </View>
204
+ <View>
205
+ {this.s.ransackableAttributes && this.sortedAttributesByName(this.s.ransackableAttributes)?.map((attribute) =>
206
+ <AttributeElement
207
+ active={attribute.attributeName == this.s.attribute?.attributeName}
208
+ attribute={attribute}
209
+ key={attribute.attributeName}
210
+ modelClassName={this.s.modelClassName}
211
+ onClick={this.tt.onAttributeClicked}
212
+ />
213
+ )}
214
+ {this.s.ransackableScopes?.map((scope) =>
215
+ <ScopeElement
216
+ active={scope == this.s.scope}
217
+ key={scope}
218
+ scope={scope}
219
+ onScopeClicked={this.tt.onScopeClicked}
220
+ />
221
+ )}
222
+ </View>
223
+ <View>
224
+ {predicates && !this.s.scope &&
225
+ <Select
226
+ className="predicate-select"
227
+ defaultValue={predicate?.name}
228
+ includeBlank
229
+ onChange={this.tt.onPredicateChanged}
230
+ options={predicates.map((predicate) => digg(predicate, "name"))}
231
+ />
232
+ }
233
+ </View>
234
+ <View>
235
+ {((attribute && predicate) || scope) &&
236
+ <Input className="value-input" defaultValue={value} inputRef={valueInputRef} />
237
+ }
238
+ </View>
239
+ </View>
240
+ <View>
241
+ <button className="apply-filter-button" disabled={!submitEnabled}>
242
+ {t(".apply", {defaultValue: "Apply"})}
243
+ </button>
244
+ </View>
245
+ </Form>
246
+ {this.s.loading > 0 &&
247
+ <View
248
+ style={{
249
+ alignItems: "center",
250
+ justifyContent: "center",
251
+ position: "absolute",
252
+ width: "100%",
253
+ height: "100%"
254
+ }}
255
+ >
256
+ <ActivityIndicator size="large" />
257
+ </View>
258
+ }
259
+ </View>
260
+ )
261
+ }
262
+
263
+ currentModelClassFromPath(path) {
264
+ const {modelClass} = this.p
265
+ let currentModelClass = modelClass
266
+
267
+ for (const pathPart of path) {
268
+ const camelizedPathPart = inflection.camelize(pathPart, true)
269
+ const association = currentModelClass.ransackableAssociations().find((reflection) => reflection.name() == camelizedPathPart)
270
+
271
+ if (!association) {
272
+ const ransackableAssociationNames = currentModelClass.ransackableAssociations().map((reflection) => reflection.name()).join(", ")
273
+
274
+ throw new Error(`Could not find a Ransackable association by that name: ${camelizedPathPart} in ${ransackableAssociationNames}`)
275
+ }
276
+
277
+ currentModelClass = association.modelClass()
278
+ }
279
+
280
+ return currentModelClass
281
+ }
282
+
283
+ currentPathParts() {
284
+ const {modelClass} = this.p
285
+ const {path} = this.s
286
+ const result = []
287
+ let currentModelClass = modelClass
288
+
289
+ result.push({
290
+ modelClass,
291
+ translation: modelClass.modelName().human({count: 2})
292
+ })
293
+
294
+ for (const pathPart of path) {
295
+ const camelizedPathPart = inflection.camelize(pathPart, true)
296
+ const pathPartTranslation = currentModelClass.humanAttributeName(camelizedPathPart)
297
+
298
+ currentModelClass = currentModelClass.ransackableAssociations().find((reflection) => reflection.name() == camelizedPathPart).modelClass()
299
+
300
+ result.push({
301
+ modelClass: currentModelClass,
302
+ translation: pathPartTranslation
303
+ })
304
+ }
305
+
306
+ return result
307
+ }
308
+
309
+ onAttributeClicked = ({attribute}) => {
310
+ this.setState({
311
+ attribute,
312
+ scope: undefined
313
+ })
314
+ }
315
+
316
+ onPredicateChanged = (e) => {
317
+ const chosenPredicateName = digg(e, "target", "value")
318
+ const predicate = this.s.predicates.find((predicate) => predicate.name == chosenPredicateName)
319
+
320
+ this.setState({predicate})
321
+ }
322
+
323
+ onReflectionClicked = ({reflection}) => {
324
+ const newPath = this.s.path.concat([reflection])
325
+
326
+ this.setState({
327
+ associations: null,
328
+ attribute: undefined,
329
+ actualCurrentModelClass: {modelClass: digg(reflection, "resource")},
330
+ modelClassName: digg(reflection, "modelClassName"),
331
+ path: newPath,
332
+ predicate: undefined
333
+ })
334
+ }
335
+
336
+ onScopeClicked = ({scope}) => {
337
+ this.setState({
338
+ attribute: undefined,
339
+ predicate: undefined,
340
+ scope
341
+ })
342
+ }
343
+
344
+ onSubmit = () => {
345
+ const {filter, querySearchName} = this.p
346
+ const {attribute, path, predicate, scope} = this.s
347
+ const {filterIndex} = digs(filter, "filterIndex")
348
+ const searchParams = Params.parse()[querySearchName] || {}
349
+ const value = digg(this.tt.valueInputRef, "current", "value")
350
+ const p = path.map((reflection) => inflection.underscore(reflection.reflectionName))
351
+ const newSearchParams = {
352
+ p,
353
+ v: value
354
+ }
355
+
356
+ if (attribute) {
357
+ newSearchParams.a = digg(attribute, "attributeName")
358
+ newSearchParams.pre = digg(predicate, "name")
359
+ } else if (scope) {
360
+ newSearchParams.sc = scope
361
+ } else {
362
+ throw new Error("Dont know if should search for attribute or scope?")
363
+ }
364
+
365
+ searchParams[filterIndex] = JSON.stringify(newSearchParams)
366
+
367
+ const newParams = {}
368
+
369
+ newParams[querySearchName] = searchParams
370
+
371
+ Params.changeParams(newParams)
372
+
373
+ this.props.onApplyClicked()
374
+ }
375
+
376
+ reflectionsWithModelClass(reflections) {
377
+ return reflections.filter((reflection) => {
378
+ try {
379
+ reflection.modelClass()
380
+
381
+ return true
382
+ } catch (error) {
383
+ return false
384
+ }
385
+ })
386
+ }
387
+
388
+ sortedAttributesByName(attributes) {
389
+ return attributes.sort((a, b) =>
390
+ digg(a, "humanName")
391
+ .toLowerCase()
392
+ .localeCompare(
393
+ digg(b, "humanName").toLowerCase()
394
+ )
395
+ )
396
+ }
397
+
398
+ sortedReflectionsByName(reflections) {
399
+ return reflections.sort((a, b) =>
400
+ digg(a, "humanName")
401
+ .toLowerCase()
402
+ .localeCompare(
403
+ digg(b, "humanName").toLowerCase()
404
+ )
405
+ )
406
+ }
407
+ }))
@@ -0,0 +1,70 @@
1
+ import {Pressable, View} from "react-native"
2
+ import BaseComponent from "../../base-component"
3
+ import Icon from "../../utils/icon"
4
+ import PropTypes from "prop-types"
5
+ import PropTypesExact from "prop-types-exact"
6
+ import memo from "set-state-compare/src/memo"
7
+ import React from "react"
8
+ import {shapeComponent} from "set-state-compare/src/shape-component"
9
+ import Text from "../../utils/text"
10
+
11
+ export default memo(shapeComponent(class ApiMakerTableFilter extends BaseComponent {
12
+ static defaultProps = {
13
+ a: null,
14
+ pre: null
15
+ }
16
+
17
+ static propTypes = PropTypesExact({
18
+ a: PropTypes.string,
19
+ filterIndex: PropTypes.number.isRequired,
20
+ onClick: PropTypes.func.isRequired,
21
+ onRemoveClicked: PropTypes.func.isRequired,
22
+ p: PropTypes.array.isRequired,
23
+ pre: PropTypes.string,
24
+ sc: PropTypes.string,
25
+ v: PropTypes.string.isRequired
26
+ })
27
+
28
+ render() {
29
+ const {p, v} = this.p
30
+ const {a, pre, sc} = this.props
31
+
32
+ return (
33
+ <View style={{alignItems: "center", flexDirection: "row", backgroundColor: "grey", paddingVertical: 10, paddingHorizontal: 6}}>
34
+ <Pressable
35
+ dataSet={{
36
+ attribute: a,
37
+ class: "filter-label",
38
+ scope: sc
39
+ }}
40
+ onPress={this.tt.onFilterPressed}
41
+ >
42
+ <Text>
43
+ {p.length > 0 &&
44
+ `${p.join(".")}.`
45
+ }
46
+ {a} {sc} {pre} {v}
47
+ </Text>
48
+ </Pressable>
49
+ <Pressable dataSet={{class: "remove-filter-button"}} onPress={this.tt.onRemoveFilterPressed} style={{marginLeft: 6}}>
50
+ <Icon name="remove" />
51
+ </Pressable>
52
+ </View>
53
+ )
54
+ }
55
+
56
+ onFilterPressed = (e) => {
57
+ e.preventDefault()
58
+
59
+ const {a, filterIndex, p, pre, v} = this.p
60
+ const {sc} = this.props
61
+
62
+ this.props.onClick({a, filterIndex, p, pre, sc, v})
63
+ }
64
+
65
+ onRemoveFilterPressed = (e) => {
66
+ e.preventDefault()
67
+
68
+ this.props.onRemoveClicked({filterIndex: this.p.filterIndex})
69
+ }
70
+ }))