alchemy_cms 6.1.5 → 7.0.0.pre.b

Sign up to get free protection for your applications and to get access to all the features.
Files changed (363) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/brakeman-analysis.yml +2 -2
  3. data/.github/workflows/ci.yml +7 -10
  4. data/.github/workflows/lint.yml +17 -0
  5. data/.gitignore +1 -6
  6. data/.hound.yml +2 -3
  7. data/.rubocop.yml +4 -350
  8. data/.standard.yml +3 -0
  9. data/CHANGELOG.md +49 -19
  10. data/Gemfile +4 -2
  11. data/README.md +7 -9
  12. data/Rakefile +12 -7
  13. data/alchemy_cms.gemspec +3 -3
  14. data/app/assets/javascripts/alchemy/admin.js +0 -1
  15. data/app/assets/javascripts/alchemy/alchemy.dirty.js.coffee +1 -1
  16. data/app/assets/javascripts/alchemy/alchemy.element_editors.js.coffee +18 -32
  17. data/app/assets/javascripts/alchemy/alchemy.elements_window.js.coffee +2 -2
  18. data/app/assets/javascripts/alchemy/alchemy.gui.js.coffee +2 -2
  19. data/app/assets/javascripts/alchemy/alchemy.link_dialog.js.coffee +33 -30
  20. data/app/assets/stylesheets/alchemy/elements.scss +16 -35
  21. data/app/assets/stylesheets/alchemy/forms.scss +0 -4
  22. data/app/assets/stylesheets/alchemy/node-select.scss +2 -2
  23. data/app/components/alchemy/ingredients/audio_view.rb +37 -0
  24. data/app/components/alchemy/ingredients/base_view.rb +38 -0
  25. data/app/components/alchemy/ingredients/boolean_view.rb +13 -0
  26. data/app/components/alchemy/ingredients/datetime_view.rb +22 -0
  27. data/app/components/alchemy/ingredients/file_view.rb +40 -0
  28. data/app/components/alchemy/ingredients/headline_view.rb +20 -0
  29. data/app/components/alchemy/ingredients/html_view.rb +9 -0
  30. data/app/components/alchemy/ingredients/link_view.rb +25 -0
  31. data/app/components/alchemy/ingredients/node_view.rb +11 -0
  32. data/app/components/alchemy/ingredients/page_view.rb +15 -0
  33. data/app/components/alchemy/ingredients/picture_view.rb +108 -0
  34. data/app/components/alchemy/ingredients/richtext_view.rb +22 -0
  35. data/app/components/alchemy/ingredients/select_view.rb +6 -0
  36. data/app/components/alchemy/ingredients/text_view.rb +41 -0
  37. data/app/components/alchemy/ingredients/video_view.rb +39 -0
  38. data/app/controllers/alchemy/admin/attachments_controller.rb +3 -4
  39. data/app/controllers/alchemy/admin/base_controller.rb +7 -7
  40. data/app/controllers/alchemy/admin/clipboard_controller.rb +2 -2
  41. data/app/controllers/alchemy/admin/elements_controller.rb +33 -43
  42. data/app/controllers/alchemy/admin/languages_controller.rb +1 -1
  43. data/app/controllers/alchemy/admin/nodes_controller.rb +2 -2
  44. data/app/controllers/alchemy/admin/pages_controller.rb +11 -11
  45. data/app/controllers/alchemy/admin/pictures_controller.rb +15 -15
  46. data/app/controllers/alchemy/admin/resources_controller.rb +28 -46
  47. data/app/controllers/alchemy/admin/styleguide_controller.rb +1 -0
  48. data/app/controllers/alchemy/admin/tags_controller.rb +11 -11
  49. data/app/controllers/alchemy/api/base_controller.rb +2 -2
  50. data/app/controllers/alchemy/api/elements_controller.rb +11 -13
  51. data/app/controllers/alchemy/api/ingredients_controller.rb +1 -1
  52. data/app/controllers/alchemy/api/nodes_controller.rb +1 -1
  53. data/app/controllers/alchemy/api/pages_controller.rb +14 -12
  54. data/app/controllers/alchemy/attachments_controller.rb +3 -3
  55. data/app/controllers/alchemy/base_controller.rb +1 -1
  56. data/app/controllers/alchemy/messages_controller.rb +18 -18
  57. data/app/controllers/alchemy/pages_controller.rb +10 -16
  58. data/app/controllers/concerns/alchemy/admin/archive_overlay.rb +1 -0
  59. data/app/controllers/concerns/alchemy/admin/uploader_responses.rb +5 -7
  60. data/app/controllers/concerns/alchemy/legacy_page_redirects.rb +5 -5
  61. data/app/decorators/alchemy/element_editor.rb +13 -33
  62. data/app/decorators/alchemy/ingredient_editor.rb +6 -6
  63. data/app/helpers/alchemy/admin/attachments_helper.rb +1 -1
  64. data/app/helpers/alchemy/admin/base_helper.rb +21 -22
  65. data/app/helpers/alchemy/admin/elements_helper.rb +1 -3
  66. data/app/helpers/alchemy/admin/form_helper.rb +1 -1
  67. data/app/helpers/alchemy/admin/navigation_helper.rb +7 -7
  68. data/app/helpers/alchemy/admin/pages_helper.rb +2 -2
  69. data/app/helpers/alchemy/admin/tags_helper.rb +3 -3
  70. data/app/helpers/alchemy/base_helper.rb +2 -2
  71. data/app/helpers/alchemy/elements_block_helper.rb +13 -48
  72. data/app/helpers/alchemy/elements_helper.rb +15 -23
  73. data/app/helpers/alchemy/pages_helper.rb +11 -15
  74. data/app/helpers/alchemy/url_helper.rb +1 -1
  75. data/app/mailers/alchemy/messages_mailer.rb +1 -1
  76. data/app/models/alchemy/attachment.rb +12 -9
  77. data/app/models/alchemy/base_record.rb +3 -0
  78. data/app/models/alchemy/eager_loading.rb +6 -7
  79. data/app/models/alchemy/element/definitions.rb +1 -1
  80. data/app/models/alchemy/element/element_ingredients.rb +4 -11
  81. data/app/models/alchemy/element/presenters.rb +9 -25
  82. data/app/models/alchemy/element.rb +4 -19
  83. data/app/models/alchemy/elements_repository.rb +1 -1
  84. data/app/models/alchemy/image_cropper_settings.rb +2 -2
  85. data/app/models/alchemy/ingredient.rb +29 -16
  86. data/app/models/alchemy/ingredient_validator.rb +1 -1
  87. data/app/models/alchemy/ingredients/audio.rb +2 -0
  88. data/app/models/alchemy/ingredients/datetime.rb +3 -1
  89. data/app/models/alchemy/ingredients/file.rb +7 -0
  90. data/app/models/alchemy/ingredients/headline.rb +6 -0
  91. data/app/models/alchemy/ingredients/link.rb +2 -0
  92. data/app/models/alchemy/ingredients/node.rb +2 -0
  93. data/app/models/alchemy/ingredients/page.rb +2 -0
  94. data/app/models/alchemy/ingredients/picture.rb +29 -0
  95. data/app/models/alchemy/ingredients/richtext.rb +17 -8
  96. data/app/models/alchemy/ingredients/select.rb +1 -0
  97. data/app/models/alchemy/ingredients/text.rb +8 -0
  98. data/app/models/alchemy/ingredients/video.rb +2 -0
  99. data/app/models/alchemy/language/code.rb +1 -1
  100. data/app/models/alchemy/language.rb +4 -4
  101. data/app/models/alchemy/legacy_page_url.rb +1 -1
  102. data/app/models/alchemy/node.rb +11 -8
  103. data/app/models/alchemy/page/page_elements.rb +19 -40
  104. data/app/models/alchemy/page/page_layouts.rb +0 -14
  105. data/app/models/alchemy/page/page_naming.rb +4 -4
  106. data/app/models/alchemy/page/page_natures.rb +1 -11
  107. data/app/models/alchemy/page/page_scopes.rb +5 -5
  108. data/app/models/alchemy/page.rb +11 -19
  109. data/app/models/alchemy/picture/calculations.rb +2 -2
  110. data/app/models/alchemy/picture/transformations.rb +2 -32
  111. data/app/models/alchemy/picture/url.rb +5 -5
  112. data/app/models/alchemy/picture.rb +22 -20
  113. data/app/models/alchemy/picture_thumb/create.rb +7 -18
  114. data/app/models/alchemy/picture_thumb/file_store.rb +33 -0
  115. data/app/models/alchemy/picture_thumb.rb +11 -11
  116. data/app/models/alchemy/picture_variant.rb +2 -3
  117. data/app/models/alchemy/tag.rb +8 -0
  118. data/app/models/concerns/alchemy/picture_thumbnails.rb +8 -8
  119. data/app/serializers/alchemy/base_serializer.rb +1 -1
  120. data/app/serializers/alchemy/element_serializer.rb +1 -6
  121. data/app/serializers/alchemy/page_tree_serializer.rb +7 -7
  122. data/app/services/alchemy/delete_elements.rb +1 -7
  123. data/app/services/alchemy/duplicate_element.rb +2 -7
  124. data/app/services/alchemy/tag_validations.rb +1 -1
  125. data/app/views/alchemy/admin/elements/_element.html.erb +8 -22
  126. data/app/views/alchemy/admin/elements/create.js.erb +1 -1
  127. data/app/views/alchemy/admin/elements/fold.js.erb +2 -2
  128. data/app/views/alchemy/admin/elements/order.js.erb +1 -1
  129. data/app/views/alchemy/admin/elements/update.js.erb +1 -2
  130. data/app/views/alchemy/admin/pages/_external_link.html.erb +2 -2
  131. data/app/views/alchemy/admin/pages/_file_link.html.erb +2 -2
  132. data/app/views/alchemy/admin/pages/_internal_link.html.erb +2 -2
  133. data/app/views/alchemy/admin/pages/edit.html.erb +1 -4
  134. data/app/views/alchemy/admin/pages/update.js.erb +10 -4
  135. data/app/views/alchemy/admin/pictures/_filter_and_size_bar.html.erb +1 -3
  136. data/app/views/alchemy/admin/pictures/_infos.html.erb +4 -6
  137. data/app/views/alchemy/ingredients/_audio_view.html.erb +1 -14
  138. data/app/views/alchemy/ingredients/_boolean_editor.html.erb +1 -1
  139. data/app/views/alchemy/ingredients/_boolean_view.html.erb +1 -1
  140. data/app/views/alchemy/ingredients/_datetime_view.html.erb +3 -9
  141. data/app/views/alchemy/ingredients/_file_view.html.erb +3 -16
  142. data/app/views/alchemy/ingredients/_headline_editor.html.erb +1 -1
  143. data/app/views/alchemy/ingredients/_headline_view.html.erb +4 -10
  144. data/app/views/alchemy/ingredients/_html_editor.html.erb +1 -1
  145. data/app/views/alchemy/ingredients/_html_view.html.erb +1 -1
  146. data/app/views/alchemy/ingredients/_link_view.html.erb +4 -9
  147. data/app/views/alchemy/ingredients/_node_editor.html.erb +1 -1
  148. data/app/views/alchemy/ingredients/_node_view.html.erb +1 -1
  149. data/app/views/alchemy/ingredients/_page_view.html.erb +1 -4
  150. data/app/views/alchemy/ingredients/_picture_editor.html.erb +4 -4
  151. data/app/views/alchemy/ingredients/_picture_view.html.erb +4 -5
  152. data/app/views/alchemy/ingredients/_richtext_editor.html.erb +11 -2
  153. data/app/views/alchemy/ingredients/_richtext_view.html.erb +3 -3
  154. data/app/views/alchemy/ingredients/_select_editor.html.erb +2 -2
  155. data/app/views/alchemy/ingredients/_select_view.html.erb +1 -1
  156. data/app/views/alchemy/ingredients/_text_editor.html.erb +1 -1
  157. data/app/views/alchemy/ingredients/_text_view.html.erb +3 -19
  158. data/app/views/alchemy/ingredients/_video_view.html.erb +3 -18
  159. data/app/views/alchemy/ingredients/shared/_link_tools.html.erb +4 -3
  160. data/app/views/alchemy/ingredients/shared/_picture_tools.html.erb +1 -0
  161. data/app/views/alchemy/pages/_meta_data.html.erb +0 -1
  162. data/app/views/layouts/alchemy/admin.html.erb +10 -8
  163. data/bin/setup +37 -0
  164. data/bin/start +17 -0
  165. data/config/alchemy/config.yml +6 -6
  166. data/config/brakeman.ignore +56 -57
  167. data/config/initializers/assets.rb +1 -0
  168. data/config/initializers/dragonfly.rb +1 -0
  169. data/config/initializers/mime_types.rb +1 -0
  170. data/config/initializers/mini_profiler.rb +1 -0
  171. data/config/initializers/simple_form.rb +3 -2
  172. data/config/locales/alchemy.en.yml +98 -112
  173. data/config/routes.rb +22 -36
  174. data/config/spring.rb +1 -0
  175. data/db/migrate/20230121212637_alchemy_six_point_one.rb +248 -0
  176. data/db/migrate/20230505132743_add_indexes_to_alchemy_pictures.rb +6 -0
  177. data/lib/alchemy/admin/locale.rb +3 -3
  178. data/lib/alchemy/admin/preview_url.rb +2 -2
  179. data/lib/alchemy/auth_accessors.rb +1 -1
  180. data/lib/alchemy/cache_digests/template_tracker.rb +6 -7
  181. data/lib/alchemy/config.rb +3 -3
  182. data/lib/alchemy/controller_actions.rb +4 -4
  183. data/lib/alchemy/deprecation.rb +2 -1
  184. data/lib/alchemy/dragonfly/processors/thumbnail.rb +1 -1
  185. data/lib/alchemy/element_definition.rb +2 -2
  186. data/lib/alchemy/engine.rb +2 -1
  187. data/lib/alchemy/errors.rb +0 -11
  188. data/lib/alchemy/filetypes.rb +7 -7
  189. data/lib/alchemy/forms/builder.rb +4 -4
  190. data/lib/alchemy/hints.rb +10 -10
  191. data/lib/alchemy/i18n.rb +6 -4
  192. data/lib/alchemy/install/tasks.rb +2 -1
  193. data/lib/alchemy/name_conversions.rb +1 -1
  194. data/lib/alchemy/page_layout.rb +1 -1
  195. data/lib/alchemy/permissions.rb +5 -17
  196. data/lib/alchemy/resource.rb +10 -10
  197. data/lib/alchemy/resources_helper.rb +7 -7
  198. data/lib/alchemy/routing_constraints.rb +5 -5
  199. data/lib/alchemy/searchable_resource.rb +38 -0
  200. data/lib/alchemy/seeder.rb +4 -3
  201. data/lib/alchemy/shell.rb +2 -1
  202. data/lib/alchemy/taggable.rb +3 -2
  203. data/lib/alchemy/tasks/tidy.rb +1 -38
  204. data/lib/alchemy/test_support/capybara_helpers.rb +69 -0
  205. data/lib/alchemy/test_support/config_stubbing.rb +1 -0
  206. data/lib/alchemy/test_support/factories/element_factory.rb +2 -4
  207. data/lib/alchemy/test_support/factories/ingredient_factory.rb +1 -1
  208. data/lib/alchemy/test_support/factories/page_factory.rb +5 -3
  209. data/lib/alchemy/test_support/having_crop_action_examples.rb +9 -9
  210. data/lib/alchemy/test_support/having_picture_thumbnails_examples.rb +33 -33
  211. data/lib/alchemy/test_support/integration_helpers.rb +4 -3
  212. data/lib/alchemy/test_support/shared_contexts.rb +2 -1
  213. data/lib/alchemy/test_support/shared_dom_ids_examples.rb +10 -10
  214. data/lib/alchemy/test_support/shared_ingredient_examples.rb +13 -7
  215. data/lib/alchemy/test_support/shared_uploader_examples.rb +1 -0
  216. data/lib/alchemy/tinymce.rb +3 -43
  217. data/lib/alchemy/upgrader/seven_point_zero.rb +45 -0
  218. data/lib/alchemy/upgrader/tasks/.keep +0 -0
  219. data/lib/alchemy/upgrader.rb +9 -3
  220. data/lib/alchemy/version.rb +1 -1
  221. data/lib/alchemy.rb +0 -19
  222. data/lib/alchemy_cms.rb +2 -2
  223. data/lib/generators/alchemy/base.rb +3 -2
  224. data/lib/generators/alchemy/elements/elements_generator.rb +2 -2
  225. data/lib/generators/alchemy/elements/templates/view.html.erb +1 -10
  226. data/lib/generators/alchemy/elements/templates/view.html.haml +1 -9
  227. data/lib/generators/alchemy/elements/templates/view.html.slim +1 -9
  228. data/lib/generators/alchemy/ingredient/ingredient_generator.rb +1 -0
  229. data/lib/generators/alchemy/install/files/alchemy.en.yml +7 -8
  230. data/lib/generators/alchemy/install/files/application.html.erb +1 -1
  231. data/lib/generators/alchemy/install/install_generator.rb +20 -23
  232. data/lib/generators/alchemy/install/templates/elements.yml.tt +12 -12
  233. data/lib/generators/alchemy/module/module_generator.rb +1 -0
  234. data/lib/generators/alchemy/page_layouts/page_layouts_generator.rb +1 -0
  235. data/lib/generators/alchemy/site_layouts/site_layouts_generator.rb +1 -0
  236. data/lib/generators/alchemy/views/views_generator.rb +2 -1
  237. data/lib/tasks/alchemy/thumbnails.rake +6 -25
  238. data/lib/tasks/alchemy/tidy.rake +2 -12
  239. data/lib/tasks/alchemy/upgrade.rake +13 -49
  240. data/package/admin.js +2 -0
  241. data/package/dist/admin.js +16 -0
  242. data/package/dist/admin.js.map +7 -0
  243. data/package/src/datepicker.js +1 -0
  244. data/package/src/tinymce.js +142 -0
  245. data/package.json +5 -3
  246. metadata +41 -135
  247. data/app/assets/javascripts/alchemy/alchemy.tinymce.js.coffee +0 -93
  248. data/app/controllers/alchemy/admin/contents_controller.rb +0 -21
  249. data/app/controllers/alchemy/admin/essence_audios_controller.rb +0 -30
  250. data/app/controllers/alchemy/admin/essence_files_controller.rb +0 -31
  251. data/app/controllers/alchemy/admin/essence_pictures_controller.rb +0 -43
  252. data/app/controllers/alchemy/admin/essence_videos_controller.rb +0 -34
  253. data/app/controllers/alchemy/api/contents_controller.rb +0 -52
  254. data/app/decorators/alchemy/content_editor.rb +0 -119
  255. data/app/helpers/alchemy/admin/contents_helper.rb +0 -42
  256. data/app/helpers/alchemy/admin/essences_helper.rb +0 -31
  257. data/app/models/alchemy/content/factory.rb +0 -143
  258. data/app/models/alchemy/content.rb +0 -247
  259. data/app/models/alchemy/element/element_contents.rb +0 -200
  260. data/app/models/alchemy/element/element_essences.rb +0 -133
  261. data/app/models/alchemy/essence_audio.rb +0 -13
  262. data/app/models/alchemy/essence_boolean.rb +0 -20
  263. data/app/models/alchemy/essence_date.rb +0 -25
  264. data/app/models/alchemy/essence_file.rb +0 -49
  265. data/app/models/alchemy/essence_headline.rb +0 -41
  266. data/app/models/alchemy/essence_html.rb +0 -23
  267. data/app/models/alchemy/essence_link.rb +0 -21
  268. data/app/models/alchemy/essence_node.rb +0 -19
  269. data/app/models/alchemy/essence_page.rb +0 -17
  270. data/app/models/alchemy/essence_picture.rb +0 -67
  271. data/app/models/alchemy/essence_picture_view.rb +0 -90
  272. data/app/models/alchemy/essence_richtext.rb +0 -44
  273. data/app/models/alchemy/essence_select.rb +0 -19
  274. data/app/models/alchemy/essence_text.rb +0 -23
  275. data/app/models/alchemy/essence_video.rb +0 -13
  276. data/app/presenters/alchemy/picture_view.rb +0 -88
  277. data/app/serializers/alchemy/content_serializer.rb +0 -17
  278. data/app/serializers/alchemy/essence_boolean_serializer.rb +0 -10
  279. data/app/serializers/alchemy/essence_date_serializer.rb +0 -10
  280. data/app/serializers/alchemy/essence_file_serializer.rb +0 -13
  281. data/app/serializers/alchemy/essence_html_serializer.rb +0 -10
  282. data/app/serializers/alchemy/essence_link_serializer.rb +0 -13
  283. data/app/serializers/alchemy/essence_picture_serializer.rb +0 -28
  284. data/app/serializers/alchemy/essence_richtext_serializer.rb +0 -11
  285. data/app/serializers/alchemy/essence_select_serializer.rb +0 -10
  286. data/app/serializers/alchemy/essence_text_serializer.rb +0 -22
  287. data/app/views/alchemy/admin/contents/create.js.erb +0 -21
  288. data/app/views/alchemy/admin/essence_audios/edit.html.erb +0 -7
  289. data/app/views/alchemy/admin/essence_files/edit.html.erb +0 -21
  290. data/app/views/alchemy/admin/essence_pictures/destroy.js.erb +0 -5
  291. data/app/views/alchemy/admin/essence_pictures/edit.html.erb +0 -30
  292. data/app/views/alchemy/admin/essence_pictures/save_link.js.erb +0 -3
  293. data/app/views/alchemy/admin/essence_pictures/update.js.erb +0 -8
  294. data/app/views/alchemy/admin/essence_videos/edit.html.erb +0 -12
  295. data/app/views/alchemy/admin/pages/_tinymce_custom_config.html.erb +0 -13
  296. data/app/views/alchemy/essences/_essence_audio_editor.html.erb +0 -4
  297. data/app/views/alchemy/essences/_essence_audio_view.html.erb +0 -15
  298. data/app/views/alchemy/essences/_essence_boolean_editor.html.erb +0 -11
  299. data/app/views/alchemy/essences/_essence_boolean_view.html.erb +0 -2
  300. data/app/views/alchemy/essences/_essence_date_editor.html.erb +0 -16
  301. data/app/views/alchemy/essences/_essence_date_view.html.erb +0 -10
  302. data/app/views/alchemy/essences/_essence_file_editor.html.erb +0 -54
  303. data/app/views/alchemy/essences/_essence_file_view.html.erb +0 -18
  304. data/app/views/alchemy/essences/_essence_headline_editor.html.erb +0 -36
  305. data/app/views/alchemy/essences/_essence_headline_view.html.erb +0 -10
  306. data/app/views/alchemy/essences/_essence_html_editor.html.erb +0 -10
  307. data/app/views/alchemy/essences/_essence_html_view.html.erb +0 -2
  308. data/app/views/alchemy/essences/_essence_link_editor.html.erb +0 -30
  309. data/app/views/alchemy/essences/_essence_link_view.html.erb +0 -10
  310. data/app/views/alchemy/essences/_essence_node_editor.html.erb +0 -27
  311. data/app/views/alchemy/essences/_essence_node_view.html.erb +0 -1
  312. data/app/views/alchemy/essences/_essence_page_editor.html.erb +0 -26
  313. data/app/views/alchemy/essences/_essence_page_view.html.erb +0 -5
  314. data/app/views/alchemy/essences/_essence_picture_editor.html.erb +0 -59
  315. data/app/views/alchemy/essences/_essence_picture_view.html.erb +0 -6
  316. data/app/views/alchemy/essences/_essence_richtext_editor.html.erb +0 -14
  317. data/app/views/alchemy/essences/_essence_richtext_view.html.erb +0 -4
  318. data/app/views/alchemy/essences/_essence_select_editor.html.erb +0 -28
  319. data/app/views/alchemy/essences/_essence_select_view.html.erb +0 -2
  320. data/app/views/alchemy/essences/_essence_text_editor.html.erb +0 -29
  321. data/app/views/alchemy/essences/_essence_text_view.html.erb +0 -17
  322. data/app/views/alchemy/essences/_essence_video_editor.html.erb +0 -4
  323. data/app/views/alchemy/essences/_essence_video_view.html.erb +0 -19
  324. data/app/views/alchemy/essences/shared/_essence_picture_tools.html.erb +0 -59
  325. data/app/views/alchemy/essences/shared/_linkable_essence_tools.html.erb +0 -20
  326. data/app/views/alchemy/pages/show.rss.builder +0 -21
  327. data/db/migrate/20200226213334_alchemy_four_point_four.rb +0 -313
  328. data/db/migrate/20200423073425_create_alchemy_essence_nodes.rb +0 -11
  329. data/db/migrate/20200504210159_remove_site_id_from_nodes.rb +0 -28
  330. data/db/migrate/20200505215518_add_language_id_foreign_key_to_alchemy_pages.rb +0 -8
  331. data/db/migrate/20200511113603_add_menu_type_to_alchemy_nodes.rb +0 -27
  332. data/db/migrate/20200514091507_make_page_layoutpage_null_false.rb +0 -6
  333. data/db/migrate/20200519073500_remove_visible_from_alchemy_pages.rb +0 -24
  334. data/db/migrate/20200617110713_create_alchemy_picture_thumbs.rb +0 -22
  335. data/db/migrate/20200907111332_remove_tri_state_booleans.rb +0 -33
  336. data/db/migrate/20201207131309_create_page_versions.rb +0 -19
  337. data/db/migrate/20201207135820_add_page_version_id_to_alchemy_elements.rb +0 -76
  338. data/db/migrate/20210205143548_rename_public_on_and_public_until_on_alchemy_pages.rb +0 -10
  339. data/db/migrate/20210326105046_add_sanitized_body_to_alchemy_essence_richtexts.rb +0 -7
  340. data/db/migrate/20210406093436_add_alchemy_essence_headlines.rb +0 -12
  341. data/db/migrate/20210506135919_create_essence_audios.rb +0 -19
  342. data/db/migrate/20210506140258_create_essence_videos.rb +0 -23
  343. data/db/migrate/20210508091432_create_alchemy_ingredients.rb +0 -22
  344. data/db/migrate/20220514072456_restrict_on_delete_page_id_foreign_key_from_alchemy_nodes.rb +0 -13
  345. data/db/migrate/20220622130905_add_playsinline_to_alchemy_essence_videos.rb +0 -9
  346. data/lib/alchemy/essence.rb +0 -250
  347. data/lib/alchemy/test_support/essence_shared_examples.rb +0 -271
  348. data/lib/alchemy/test_support/factories/content_factory.rb +0 -20
  349. data/lib/alchemy/test_support/factories/essence_audio_factory.rb +0 -7
  350. data/lib/alchemy/test_support/factories/essence_file_factory.rb +0 -7
  351. data/lib/alchemy/test_support/factories/essence_page_factory.rb +0 -7
  352. data/lib/alchemy/test_support/factories/essence_picture_factory.rb +0 -11
  353. data/lib/alchemy/test_support/factories/essence_text_factory.rb +0 -7
  354. data/lib/alchemy/test_support/factories/essence_video_factory.rb +0 -7
  355. data/lib/alchemy/upgrader/five_point_zero.rb +0 -41
  356. data/lib/alchemy/upgrader/six_point_zero.rb +0 -21
  357. data/lib/alchemy/upgrader/tasks/add_page_versions.rb +0 -33
  358. data/lib/alchemy/upgrader/tasks/element_views_updater.rb +0 -34
  359. data/lib/alchemy/upgrader/tasks/harden_gutentag_migrations.rb +0 -29
  360. data/lib/alchemy/upgrader/tasks/ingredients_migrator.rb +0 -74
  361. data/lib/generators/alchemy/essence/essence_generator.rb +0 -49
  362. data/lib/generators/alchemy/essence/templates/editor.html.erb +0 -17
  363. data/lib/generators/alchemy/essence/templates/view.html.erb +0 -2
@@ -73,22 +73,22 @@ module Alchemy
73
73
  def render_elements(options = {}, &blk)
74
74
  options = {
75
75
  from_page: @page,
76
- render_format: "html",
76
+ render_format: "html"
77
77
  }.update(options)
78
78
 
79
79
  finder = options[:finder] || Alchemy::ElementsFinder.new(options)
80
80
 
81
81
  page_version = if @preview_mode
82
- options[:from_page]&.draft_version
83
- else
84
- options[:from_page]&.public_version
85
- end
82
+ options[:from_page]&.draft_version
83
+ else
84
+ options[:from_page]&.public_version
85
+ end
86
86
 
87
87
  elements = finder.elements(page_version: page_version)
88
88
 
89
89
  default_rendering = ->(element, i) { render_element(element, options, i + 1) }
90
90
  capture do
91
- if block_given?
91
+ if blk
92
92
  elements.map.with_index(&blk)
93
93
  else
94
94
  elements.map.with_index(&default_rendering)
@@ -112,9 +112,9 @@ module Alchemy
112
112
  #
113
113
  # # elements.yml
114
114
  # - name: headline
115
- # contents:
116
- # - name: text
117
- # type: EssenceText
115
+ # ingredients:
116
+ # - role: text
117
+ # type: Text
118
118
  #
119
119
  # Then your element view partial has to be named like:
120
120
  #
@@ -143,7 +143,7 @@ module Alchemy
143
143
  def render_element(element, options = {}, counter = 1)
144
144
  if element.nil?
145
145
  warning("Element is nil")
146
- render "alchemy/elements/view_not_found", { name: "nil" }
146
+ render "alchemy/elements/view_not_found", {name: "nil"}
147
147
  return
148
148
  end
149
149
 
@@ -155,8 +155,8 @@ module Alchemy
155
155
  locals: {
156
156
  element: element,
157
157
  counter: counter,
158
- options: options.except(:locals, :partial),
159
- }.merge(options[:locals] || {}),
158
+ options: options.except(:locals, :partial)
159
+ }.merge(options[:locals] || {})
160
160
  )
161
161
  rescue ActionView::MissingTemplate => e
162
162
  warning(%(
@@ -166,14 +166,6 @@ module Alchemy
166
166
  render "alchemy/elements/view_not_found", name: element.name
167
167
  end
168
168
 
169
- # Returns a string for the id attribute of a html element for the given element
170
- # @deprecated
171
- def element_dom_id(element)
172
- element&.dom_id
173
- end
174
-
175
- deprecate element_dom_id: "element.dom_id", deprecator: Alchemy::Deprecation
176
-
177
169
  # Renders the HTML tag attributes required for preview mode.
178
170
  def element_preview_code(element)
179
171
  tag_builder.tag_options(element_preview_code_attributes(element))
@@ -183,7 +175,7 @@ module Alchemy
183
175
  def element_preview_code_attributes(element)
184
176
  return {} unless element.present? && @preview_mode && element.page == @page
185
177
 
186
- { "data-alchemy-element" => element.id }
178
+ {"data-alchemy-element" => element.id}
187
179
  end
188
180
 
189
181
  # Returns the element's tags information as a string. Parameters and options
@@ -211,12 +203,12 @@ module Alchemy
211
203
  #
212
204
  def element_tags_attributes(element, options = {})
213
205
  options = {
214
- formatter: lambda { |tags| tags.join(" ") },
206
+ formatter: lambda { |tags| tags.join(" ") }
215
207
  }.merge(options)
216
208
 
217
209
  return {} if !element.taggable? || element.tag_list.blank?
218
210
 
219
- { "data-element-tags" => options[:formatter].call(element.tag_list) }
211
+ {"data-element-tags" => options[:formatter].call(element.tag_list)}
220
212
  end
221
213
  end
222
214
  end
@@ -5,10 +5,6 @@ module Alchemy
5
5
  include Alchemy::BaseHelper
6
6
  include Alchemy::ElementsHelper
7
7
 
8
- def picture_essence_caption(content)
9
- content.try(:essence).try(:caption)
10
- end
11
-
12
8
  # Renders links to language root pages of all published languages.
13
9
  #
14
10
  # @option options linkname [String] ('name')
@@ -28,7 +24,7 @@ module Alchemy
28
24
  linkname: "name",
29
25
  show_title: true,
30
26
  spacer: "",
31
- reverse: false,
27
+ reverse: false
32
28
  }.merge(options)
33
29
  languages = Language.on_current_site.published.with_root_page.order("name #{options[:reverse] ? "DESC" : "ASC"}")
34
30
  return nil if languages.count < 2
@@ -37,7 +33,7 @@ module Alchemy
37
33
  partial: "alchemy/language_links/language",
38
34
  collection: languages,
39
35
  spacer_template: "alchemy/language_links/spacer",
40
- locals: { languages: languages, options: options },
36
+ locals: {languages: languages, options: options}
41
37
  )
42
38
  end
43
39
 
@@ -83,7 +79,7 @@ module Alchemy
83
79
  def render_menu(menu_type, options = {})
84
80
  root_node = Alchemy::Node.roots.find_by(
85
81
  menu_type: menu_type,
86
- language: Alchemy::Language.current,
82
+ language: Alchemy::Language.current
87
83
  )
88
84
  if root_node.nil?
89
85
  warning("Menu with type #{menu_type} not found!")
@@ -114,12 +110,12 @@ module Alchemy
114
110
  page: @page,
115
111
  restricted_only: false,
116
112
  reverse: false,
117
- link_active_page: false,
113
+ link_active_page: false
118
114
  }.merge(options)
119
115
 
120
- pages = options[:page].
121
- self_and_ancestors.contentpages.
122
- published
116
+ pages = options[:page]
117
+ .self_and_ancestors.contentpages
118
+ .published
123
119
 
124
120
  if options.delete(:restricted_only)
125
121
  pages = pages.restricted
@@ -150,13 +146,13 @@ module Alchemy
150
146
  options = {
151
147
  prefix: "",
152
148
  suffix: "",
153
- separator: "",
149
+ separator: ""
154
150
  }.update(options)
155
151
  title_parts = [options[:prefix]]
156
- if response.status == 200
157
- title_parts << @page.title
152
+ title_parts << if response.status == 200
153
+ @page.title
158
154
  else
159
- title_parts << response.status
155
+ response.status
160
156
  end
161
157
  title_parts << options[:suffix]
162
158
  title_parts.reject(&:blank?).join(options[:separator]).html_safe
@@ -20,7 +20,7 @@ module Alchemy
20
20
  def show_page_path_params(page, optional_params = {})
21
21
  raise ArgumentError, "Page is nil" if page.nil?
22
22
 
23
- url_params = { urlname: page.urlname }.update(optional_params)
23
+ url_params = {urlname: page.urlname}.update(optional_params)
24
24
  prefix_locale?(page.language_code) ? url_params.update(locale: page.language_code) : url_params
25
25
  end
26
26
 
@@ -8,7 +8,7 @@ module Alchemy
8
8
  from: mail_from,
9
9
  to: mail_to,
10
10
  reply_to: message.try(:email),
11
- subject: subject,
11
+ subject: subject
12
12
  )
13
13
  end
14
14
  end
@@ -30,14 +30,17 @@ module Alchemy
30
30
 
31
31
  stampable stamper_class_name: Alchemy.user_class.name
32
32
 
33
- has_many :essence_files, class_name: "Alchemy::EssenceFile", foreign_key: "attachment_id"
34
- has_many :contents, through: :essence_files
35
- has_many :elements, through: :contents
33
+ has_many :file_ingredients,
34
+ class_name: "Alchemy::Ingredients::File",
35
+ foreign_key: "related_object_id",
36
+ inverse_of: :related_object
37
+
38
+ has_many :elements, through: :file_ingredients
36
39
  has_many :pages, through: :elements
37
40
 
38
41
  scope :by_file_type, ->(file_type) { where(file_mime_type: file_type) }
39
42
  scope :recent, -> { where("#{table_name}.created_at > ?", Time.current - 24.hours).order(:created_at) }
40
- scope :without_tag, -> { left_outer_joins(:taggings).where(gutentag_taggings: { id: nil }) }
43
+ scope :without_tag, -> { left_outer_joins(:taggings).where(gutentag_taggings: {id: nil}) }
41
44
 
42
45
  # We need to define this method here to have it available in the validations below.
43
46
  class << self
@@ -59,12 +62,12 @@ module Alchemy
59
62
  [
60
63
  {
61
64
  name: :by_file_type,
62
- values: distinct.pluck(:file_mime_type).map { |type| [Alchemy.t(type, scope: "mime_types"), type] }.sort_by(&:first),
65
+ values: distinct.pluck(:file_mime_type).map { |type| [Alchemy.t(type, scope: "mime_types"), type] }.sort_by(&:first)
63
66
  },
64
67
  {
65
68
  name: :misc,
66
- values: %w(recent last_upload without_tag),
67
- },
69
+ values: %w[recent last_upload without_tag]
70
+ }
68
71
  ]
69
72
  end
70
73
 
@@ -76,7 +79,7 @@ module Alchemy
76
79
  end
77
80
 
78
81
  def searchable_alchemy_resource_attributes
79
- %w(name file_name)
82
+ %w[name file_name]
80
83
  end
81
84
 
82
85
  def allowed_filetypes
@@ -103,7 +106,7 @@ module Alchemy
103
106
  {
104
107
  "name" => read_attribute(:file_name),
105
108
  "size" => read_attribute(:file_size),
106
- "error" => errors[:file].join,
109
+ "error" => errors[:file].join
107
110
  }
108
111
  end
109
112
 
@@ -1,10 +1,13 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  module Alchemy
3
4
  def self.table_name_prefix
4
5
  "alchemy_"
5
6
  end
6
7
 
7
8
  class BaseRecord < ActiveRecord::Base
9
+ extend Alchemy::SearchableResource
10
+
8
11
  self.abstract_class = true
9
12
  end
10
13
  end
@@ -20,19 +20,18 @@ module Alchemy
20
20
  [
21
21
  :tags,
22
22
  {
23
- language: :site,
23
+ :language => :site,
24
24
  version => {
25
25
  elements: [
26
26
  :page,
27
27
  :touchable_pages,
28
28
  :tags,
29
29
  {
30
- ingredients: :related_object,
31
- contents: :essence,
32
- },
33
- ],
34
- },
35
- },
30
+ ingredients: :related_object
31
+ }
32
+ ]
33
+ }
34
+ }
36
35
  ]
37
36
  end
38
37
  end
@@ -41,7 +41,7 @@ module Alchemy
41
41
  # The definition of this element.
42
42
  #
43
43
  def definition
44
- if definition = self.class.definition_by_name(name)
44
+ if (definition = self.class.definition_by_name(name))
45
45
  definition
46
46
  else
47
47
  log_warning "Could not find element definition for #{name}. " \
@@ -37,13 +37,6 @@ module Alchemy
37
37
  ingredients_by_type(type).first
38
38
  end
39
39
 
40
- # All ingredients from element by given role.
41
- def ingredients_by_role(role)
42
- ingredients.select do |ingredient|
43
- ingredient.role == Ingredient.normalize_type(role)
44
- end
45
- end
46
-
47
40
  # All ingredients from element by given type.
48
41
  def ingredients_by_type(type)
49
42
  ingredients.select do |ingredient|
@@ -133,7 +126,7 @@ module Alchemy
133
126
  # == Error message translation fallbacks
134
127
  #
135
128
  # In order to not translate every single ingredient for every element
136
- # you can provide default error messages per content name:
129
+ # you can provide default error messages per ingredient role:
137
130
  #
138
131
  # === Example
139
132
  #
@@ -166,9 +159,9 @@ module Alchemy
166
159
  scope: "ingredient_validations",
167
160
  default: [
168
161
  "fields.#{role}.#{error}".to_sym,
169
- "errors.#{error}".to_sym,
162
+ "errors.#{error}".to_sym
170
163
  ],
171
- field: Alchemy::Ingredient.translated_label_for(role, name),
164
+ field: Alchemy::Ingredient.translated_label_for(role, name)
172
165
  )
173
166
  end
174
167
  end
@@ -182,7 +175,7 @@ module Alchemy
182
175
  ingredient_definitions.each do |attributes|
183
176
  ingredients.build(
184
177
  role: attributes[:role],
185
- type: Alchemy::Ingredient.normalize_type(attributes[:type]),
178
+ type: Alchemy::Ingredient.normalize_type(attributes[:type])
186
179
  )
187
180
  end
188
181
  end
@@ -38,16 +38,15 @@ module Alchemy
38
38
 
39
39
  # Returns a preview text for element.
40
40
  #
41
- # It's taken from the first Content found in the +elements.yml+ definition file.
41
+ # It's taken from the first Ingredient found in the +elements.yml+ definition file.
42
42
  #
43
- # You can flag a Content as +as_element_title+ to take this as preview.
43
+ # You can flag a Ingredient as +as_element_title+ to take this as preview.
44
44
  #
45
45
  # @param maxlength [Fixnum] (60)
46
46
  # Length of characters after the text will be cut off.
47
47
  #
48
48
  def preview_text(maxlength = 60)
49
49
  preview_text_from_preview_ingredient(maxlength) ||
50
- preview_text_from_preview_content(maxlength) ||
51
50
  preview_text_from_nested_elements(maxlength)
52
51
  end
53
52
 
@@ -61,14 +60,14 @@ module Alchemy
61
60
  #
62
61
  # - name: funky_element
63
62
  # display_name: Funky Element
64
- # contents:
65
- # - name: headline
66
- # type: EssenceText
67
- # - name: text
68
- # type EssenceRichtext
69
- # as_element_title: true
63
+ # ingredients:
64
+ # - role: headline
65
+ # type: Text
66
+ # - role: text
67
+ # type: Richtext
68
+ # as_element_title: true
70
69
  #
71
- # With "I want to tell you a funky story" as stripped_body for the EssenceRichtext Content produces:
70
+ # With "I want to tell you a funky story" as stripped_body for the Richtext ingredient produces:
72
71
  #
73
72
  # Funky Element: I want to tell ...
74
73
  #
@@ -85,17 +84,6 @@ module Alchemy
85
84
  self.class.dom_id_class.new(self).call
86
85
  end
87
86
 
88
- # The content that's used for element's preview text.
89
- #
90
- # It tries to find one of element's contents that is defined +as_element_title+.
91
- # Takes element's first content if no content is defined +as_element_title+.
92
- #
93
- # @return (Alchemy::Content)
94
- #
95
- def preview_content
96
- @_preview_content ||= contents.detect(&:preview_content?) || contents.first
97
- end
98
-
99
87
  # The ingredient that's used for element's preview text.
100
88
  #
101
89
  # It tries to find one of element's ingredients that is defined +as_element_title+.
@@ -115,10 +103,6 @@ module Alchemy
115
103
  all_nested_elements.first.preview_text(maxlength)
116
104
  end
117
105
 
118
- def preview_text_from_preview_content(maxlength)
119
- preview_content.try!(:preview_text, maxlength)
120
- end
121
-
122
106
  def preview_text_from_preview_ingredient(maxlength)
123
107
  preview_ingredient&.preview_text(maxlength)
124
108
  end
@@ -21,10 +21,7 @@
21
21
  #
22
22
 
23
23
  require_dependency "alchemy/element/definitions"
24
- require_dependency "alchemy/element/dom_id"
25
- require_dependency "alchemy/element/element_contents"
26
24
  require_dependency "alchemy/element/element_ingredients"
27
- require_dependency "alchemy/element/element_essences"
28
25
  require_dependency "alchemy/element/presenters"
29
26
 
30
27
  module Alchemy
@@ -39,13 +36,12 @@ module Alchemy
39
36
  "amount",
40
37
  "autogenerate",
41
38
  "nestable_elements",
42
- "contents",
43
39
  "hint",
44
40
  "ingredients",
45
41
  "taggable",
46
42
  "compact",
47
43
  "message",
48
- "deprecated",
44
+ "deprecated"
49
45
  ].freeze
50
46
 
51
47
  # All Elements that share the same page version and parent element and are fixed or not are considered a list.
@@ -60,10 +56,6 @@ module Alchemy
60
56
 
61
57
  stampable stamper_class_name: Alchemy.user_class.name
62
58
 
63
- has_many :contents, dependent: :destroy, inverse_of: :element
64
-
65
- deprecate contents: :ingredients, deprecator: Alchemy::Deprecation
66
-
67
59
  before_destroy :delete_all_nested_elements
68
60
 
69
61
  has_many :all_nested_elements,
@@ -96,9 +88,7 @@ module Alchemy
96
88
  validates_presence_of :name, on: :create
97
89
  validates_format_of :name, on: :create, with: NAME_REGEXP
98
90
 
99
- attr_accessor :autogenerate_contents
100
91
  attr_accessor :autogenerate_nested_elements
101
- after_create :create_contents, unless: -> { autogenerate_contents == false }
102
92
  after_create :generate_nested_elements, unless: -> { autogenerate_nested_elements == false }
103
93
 
104
94
  after_update :touch_touchable_pages
@@ -106,12 +96,11 @@ module Alchemy
106
96
  scope :published, -> { where(public: true) }
107
97
  scope :hidden, -> { where(public: false) }
108
98
  scope :not_restricted, -> { joins(:page).merge(Page.not_restricted) }
109
- scope :available, -> { published }
110
99
  scope :named, ->(names) { where(name: names) }
111
100
  scope :excluded, ->(names) { where.not(name: names) }
112
101
  scope :fixed, -> { where(fixed: true) }
113
102
  scope :unfixed, -> { where(fixed: false) }
114
- scope :from_current_site, -> { where(Language.table_name => { site_id: Site.current || Site.default }).joins(page: "language") }
103
+ scope :from_current_site, -> { where(Language.table_name => {site_id: Site.current || Site.default}).joins(page: "language") }
115
104
  scope :folded, -> { where(folded: true) }
116
105
  scope :expanded, -> { where(folded: false) }
117
106
  scope :not_nested, -> { where(parent_element_id: nil) }
@@ -120,8 +109,6 @@ module Alchemy
120
109
 
121
110
  # Concerns
122
111
  include Definitions
123
- include ElementContents
124
- include ElementEssences
125
112
  include ElementIngredients
126
113
  include Presenters
127
114
 
@@ -158,7 +145,7 @@ module Alchemy
158
145
  @_dom_id_class = klass
159
146
  end
160
147
 
161
- # This methods does a copy of source and all depending contents and all of their depending essences.
148
+ # This methods does a copy of source and all its ingredients.
162
149
  #
163
150
  # == Options
164
151
  #
@@ -193,8 +180,6 @@ module Alchemy
193
180
 
194
181
  all_from_clipboard(clipboard).where(name: parent_element.definition["nestable_elements"])
195
182
  end
196
-
197
- deprecate available: :published, deprecator: Alchemy::Deprecation
198
183
  end
199
184
 
200
185
  # Returns next public element from same page.
@@ -278,7 +263,7 @@ module Alchemy
278
263
  # Elements are defined in the +config/alchemy/elements.yml+ file
279
264
  #
280
265
  # - name: article
281
- # contents:
266
+ # ingredients:
282
267
  # ...
283
268
  #
284
269
  # == Override the view
@@ -102,7 +102,7 @@ module Alchemy
102
102
  # Elements off setted by
103
103
  # @return [Alchemy::ElementRepository]
104
104
  def offset(offset)
105
- self.class.new elements[offset.to_i..-1]
105
+ self.class.new elements[offset.to_i..]
106
106
  end
107
107
 
108
108
  # Elements limitted by
@@ -21,7 +21,7 @@ module Alchemy
21
21
  min_size: large_enough? ? min_size : false,
22
22
  ratio: ratio,
23
23
  default_box: default_box,
24
- image_size: [image_width, image_height],
24
+ image_size: [image_width, image_height]
25
25
  }.freeze
26
26
  end
27
27
 
@@ -80,7 +80,7 @@ module Alchemy
80
80
  default_crop_from[0],
81
81
  default_crop_from[1],
82
82
  default_crop_from[0] + default_crop_size[0],
83
- default_crop_from[1] + default_crop_size[1],
83
+ default_crop_from[1] + default_crop_size[1]
84
84
  ]
85
85
  end
86
86
  end
@@ -13,11 +13,13 @@ module Alchemy
13
13
  belongs_to :element, touch: true, class_name: "Alchemy::Element", inverse_of: :ingredients
14
14
  belongs_to :related_object, polymorphic: true, optional: true
15
15
 
16
+ has_one :page, through: :element, class_name: "Alchemy::Page"
17
+
16
18
  after_initialize :set_default_value,
17
19
  if: -> { definition.key?(:default) && value.nil? }
18
20
 
19
21
  validates :type, presence: true
20
- validates :role, presence: true
22
+ validates :role, presence: true, uniqueness: {scope: :element_id, case_sensitive: false}
21
23
 
22
24
  validates_with Alchemy::IngredientValidator, on: :update, if: :has_validations?
23
25
 
@@ -73,14 +75,19 @@ module Alchemy
73
75
  Alchemy.t(
74
76
  role,
75
77
  scope: "ingredient_roles.#{element_name}",
76
- default: Alchemy.t("ingredient_roles.#{role}", default: role.humanize),
78
+ default: Alchemy.t("ingredient_roles.#{role}", default: role.humanize)
77
79
  )
78
80
  end
79
- end
80
81
 
81
- # Compatibility method for access from element
82
- def essence
83
- self
82
+ # Allow to define settings on the ingredient definition
83
+ def allow_settings(settings)
84
+ @allowed_settings = Array(settings)
85
+ end
86
+
87
+ # Allowed settings on the ingredient
88
+ def allowed_settings
89
+ @allowed_settings ||= []
90
+ end
84
91
  end
85
92
 
86
93
  # The value or the related object if present
@@ -93,16 +100,6 @@ module Alchemy
93
100
  definition[:settings] || {}
94
101
  end
95
102
 
96
- # Fetches value from settings
97
- #
98
- # @param key [Symbol] - The hash key you want to fetch the value from
99
- # @param options [Hash] - An optional Hash that can override the settings.
100
- # Normally passed as options hash into the content
101
- # editor view.
102
- def settings_value(key, options = {})
103
- settings.merge(options || {})[key.to_sym]
104
- end
105
-
106
103
  # Definition hash for this ingredient from +elements.yml+ file.
107
104
  #
108
105
  def definition
@@ -158,12 +155,28 @@ module Alchemy
158
155
  !!definition[:as_element_title]
159
156
  end
160
157
 
158
+ # The view component of the ingredient with mapped options.
159
+ #
160
+ # @param options [Hash] - Passed to the view component as keyword arguments
161
+ # @param html_options [Hash] - Passed to the view component
162
+ def as_view_component(options: {}, html_options: {})
163
+ view_component_class.new(self, **options, html_options: html_options)
164
+ end
165
+
161
166
  private
162
167
 
168
+ def view_component_class
169
+ @_view_component_class ||= "#{self.class.name}View".constantize
170
+ end
171
+
163
172
  def hint_translation_attribute
164
173
  role
165
174
  end
166
175
 
176
+ def hint_translation_scope
177
+ "ingredient_hints"
178
+ end
179
+
167
180
  def set_default_value
168
181
  self.value = default_value
169
182
  end
@@ -89,7 +89,7 @@ module Alchemy
89
89
  def duplicates
90
90
  ingredient.class
91
91
  .joins(:element).merge(Alchemy::Element.published)
92
- .where(Alchemy::Element.table_name => { name: ingredient.element.name })
92
+ .where(Alchemy::Element.table_name => {name: ingredient.element.name})
93
93
  .where(value: ingredient.value)
94
94
  .where.not(id: ingredient.id)
95
95
  end
@@ -11,6 +11,8 @@ module Alchemy
11
11
  :muted,
12
12
  :loop
13
13
 
14
+ allow_settings %i[except only]
15
+
14
16
  related_object_alias :attachment, class_name: "Alchemy::Attachment"
15
17
 
16
18
  delegate :name, to: :attachment, allow_nil: true
@@ -5,6 +5,8 @@ module Alchemy
5
5
  # A datetime value
6
6
  #
7
7
  class Datetime < Alchemy::Ingredient
8
+ allow_settings %i[date_format]
9
+
8
10
  def value
9
11
  ActiveRecord::Type::DateTime.new.cast(self[:value])
10
12
  end
@@ -13,7 +15,7 @@ module Alchemy
13
15
  def preview_text(_maxlength = nil)
14
16
  return "" unless value
15
17
 
16
- ::I18n.l(value, format: :'alchemy.essence_date')
18
+ ::I18n.l(value, format: :"alchemy.ingredient_date")
17
19
  end
18
20
  end
19
21
  end
@@ -12,6 +12,13 @@ module Alchemy
12
12
  :link_text,
13
13
  :title
14
14
 
15
+ allow_settings %i[
16
+ css_classes
17
+ except
18
+ link_text
19
+ only
20
+ ]
21
+
15
22
  related_object_alias :attachment, class_name: "Alchemy::Attachment"
16
23
 
17
24
  delegate :name, to: :attachment, allow_nil: true