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
@@ -25,7 +25,7 @@ module Alchemy
25
25
  THUMBNAIL_SIZES = {
26
26
  small: "80x60",
27
27
  medium: "160x120",
28
- large: "240x180",
28
+ large: "240x180"
29
29
  }.with_indifferent_access.freeze
30
30
 
31
31
  CONVERTIBLE_FILE_FORMATS = %w[gif jpg jpeg png webp].freeze
@@ -38,7 +38,7 @@ module Alchemy
38
38
  :format,
39
39
  :quality,
40
40
  :size,
41
- :upsample,
41
+ :upsample
42
42
  ]
43
43
 
44
44
  include Alchemy::Logger
@@ -47,17 +47,16 @@ module Alchemy
47
47
  include Alchemy::TouchElements
48
48
  include Calculations
49
49
 
50
- has_many :essence_pictures,
51
- class_name: "Alchemy::EssencePicture",
52
- foreign_key: "picture_id",
53
- inverse_of: :ingredient_association
50
+ has_many :picture_ingredients,
51
+ class_name: "Alchemy::Ingredients::Picture",
52
+ foreign_key: "related_object_id",
53
+ inverse_of: :related_object
54
54
 
55
- has_many :contents, through: :essence_pictures
56
- has_many :elements, through: :contents
55
+ has_many :elements, through: :picture_ingredients
57
56
  has_many :pages, through: :elements
58
57
  has_many :thumbs, class_name: "Alchemy::PictureThumb", dependent: :destroy
59
58
 
60
- # Raise error, if picture is in use (aka. assigned to an EssencePicture)
59
+ # Raise error, if picture is in use (aka. assigned to an Picture ingredient)
61
60
  #
62
61
  # === CAUTION
63
62
  #
@@ -65,7 +64,7 @@ module Alchemy
65
64
  # to ensure this runs before Dragonfly's before_destroy callback.
66
65
  #
67
66
  before_destroy unless: :deletable? do
68
- raise PictureInUseError, Alchemy.t(:cannot_delete_picture_notice) % { name: name }
67
+ raise PictureInUseError, Alchemy.t(:cannot_delete_picture_notice) % {name: name}
69
68
  end
70
69
 
71
70
  # Image preprocessing class
@@ -114,8 +113,11 @@ module Alchemy
114
113
 
115
114
  scope :named, ->(name) { where("#{table_name}.name LIKE ?", "%#{name}%") }
116
115
  scope :recent, -> { where("#{table_name}.created_at > ?", Time.current - 24.hours).order(:created_at) }
117
- scope :deletable, -> { where("#{table_name}.id NOT IN (SELECT picture_id FROM #{EssencePicture.table_name})") }
118
- scope :without_tag, -> { left_outer_joins(:taggings).where(gutentag_taggings: { id: nil }) }
116
+ scope :deletable,
117
+ -> {
118
+ where("#{table_name}.id NOT IN (SELECT related_object_id FROM alchemy_ingredients WHERE related_object_type = 'Alchemy::Picture')")
119
+ }
120
+ scope :without_tag, -> { left_outer_joins(:taggings).where(gutentag_taggings: {id: nil}) }
119
121
  scope :by_file_format, ->(format) { where(image_file_format: format) }
120
122
 
121
123
  # Class methods
@@ -140,17 +142,17 @@ module Alchemy
140
142
  [
141
143
  {
142
144
  name: :by_file_format,
143
- values: @_file_formats,
145
+ values: @_file_formats
144
146
  },
145
147
  {
146
148
  name: :misc,
147
- values: %w(recent last_upload without_tag deletable),
148
- },
149
+ values: %w[recent last_upload without_tag deletable]
150
+ }
149
151
  ]
150
152
  end
151
153
 
152
154
  def searchable_alchemy_resource_attributes
153
- %w(name image_file_name)
155
+ %w[name image_file_name]
154
156
  end
155
157
 
156
158
  def last_upload
@@ -182,7 +184,7 @@ module Alchemy
182
184
  options.except(*TRANSFORMATION_OPTIONS).merge(
183
185
  basename: name,
184
186
  ext: variant.render_format,
185
- name: name,
187
+ name: name
186
188
  )
187
189
  )
188
190
  rescue ::Dragonfly::Job::Fetch::NotFound => e
@@ -210,7 +212,7 @@ module Alchemy
210
212
  {
211
213
  name: image_file_name,
212
214
  size: image_file_size,
213
- error: errors[:image_file].join,
215
+ error: errors[:image_file].join
214
216
  }
215
217
  end
216
218
 
@@ -280,10 +282,10 @@ module Alchemy
280
282
  pages.any? && pages.not_restricted.blank?
281
283
  end
282
284
 
283
- # Returns true if picture is not assigned to any EssencePicture.
285
+ # Returns true if picture is not assigned to any Picture ingredient.
284
286
  #
285
287
  def deletable?
286
- essence_pictures.empty?
288
+ picture_ingredients.empty?
287
289
  end
288
290
 
289
291
  # A size String from original image file values.
@@ -2,9 +2,11 @@
2
2
 
3
3
  module Alchemy
4
4
  class PictureThumb < BaseRecord
5
- # Stores the render result of a Alchemy::PictureVariant
6
- # in the configured Dragonfly datastore
7
- # (Default: Dragonfly::FileDataStore)
5
+ # Creates a Alchemy::PictureThumb
6
+ #
7
+ # Stores the processes result of a Alchemy::PictureVariant
8
+ # in the configured +Alchemy::PictureThumb.storage_class+
9
+ # (Default: {Alchemy::PictureThumb::FileStore})
8
10
  #
9
11
  class Create
10
12
  class << self
@@ -24,26 +26,13 @@ module Alchemy
24
26
  thumb.uid = uid
25
27
  end
26
28
  begin
27
- # process the image
28
- image = variant.image
29
- # store the processed image
30
- image.to_file(server_path(uid)).close
31
- rescue RuntimeError => e
29
+ Alchemy::PictureThumb.storage_class.call(variant, uid)
30
+ rescue => e
32
31
  ErrorTracking.notification_handler.call(e)
33
32
  # destroy the thumb if processing or storing fails
34
33
  @thumb&.destroy
35
34
  end
36
35
  end
37
-
38
- private
39
-
40
- # Alchemys dragonfly datastore config seperates the storage path from the public server
41
- # path for security reasons. The Dragonfly FileDataStorage does not support that,
42
- # so we need to build the path on our own.
43
- def server_path(uid)
44
- dragonfly_app = ::Dragonfly.app(:alchemy_pictures)
45
- "#{dragonfly_app.datastore.server_root}/#{uid}"
46
- end
47
36
  end
48
37
  end
49
38
  end
@@ -0,0 +1,33 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class PictureThumb < BaseRecord
5
+ # Stores the render result of a Alchemy::PictureVariant
6
+ # in the configured Dragonfly datastore
7
+ # (Default: Dragonfly::FileDataStore)
8
+ #
9
+ class FileStore
10
+ class << self
11
+ # @param [Alchemy::PictureVariant] variant the to be rendered image
12
+ # @param [String] uid The Unique Image Identifier the image is stored at
13
+ #
14
+ def call(variant, uid)
15
+ # process the image
16
+ image = variant.image
17
+ # store the processed image
18
+ image.to_file(server_path(uid)).close
19
+ end
20
+
21
+ private
22
+
23
+ # Alchemys dragonfly datastore config seperates the storage path from the public server
24
+ # path for security reasons. The Dragonfly FileDataStorage does not support that,
25
+ # so we need to build the path on our own.
26
+ def server_path(uid)
27
+ dragonfly_app = ::Dragonfly.app(:alchemy_pictures)
28
+ "#{dragonfly_app.datastore.server_root}/#{uid}"
29
+ end
30
+ end
31
+ end
32
+ end
33
+ end
@@ -7,7 +7,7 @@ module Alchemy
7
7
  # different thumbnail store (ie. a remote file storage).
8
8
  #
9
9
  # config/initializers/alchemy.rb
10
- # Alchemy::PictureThumb.generator_class = My::ThumbnailGenerator
10
+ # Alchemy::PictureThumb.storage_class = My::ThumbnailStore
11
11
  #
12
12
  class PictureThumb < BaseRecord
13
13
  belongs_to :picture, class_name: "Alchemy::Picture"
@@ -16,18 +16,18 @@ module Alchemy
16
16
  validates :uid, presence: true
17
17
 
18
18
  class << self
19
- # Thumbnail generator class
19
+ # Thumbnail storage class
20
20
  #
21
- # @see Alchemy::PictureThumb::Create
22
- def generator_class
23
- @_generator_class ||= Alchemy::PictureThumb::Create
21
+ # @see Alchemy::PictureThumb::FileStore
22
+ def storage_class
23
+ @_storage_class ||= Alchemy::PictureThumb::FileStore
24
24
  end
25
25
 
26
- # Set a thumbnail generator class
26
+ # Set a thumbnail storage class
27
27
  #
28
- # @see Alchemy::PictureThumb::Create
29
- def generator_class=(klass)
30
- @_generator_class = klass
28
+ # @see Alchemy::PictureThumb::FileStore
29
+ def storage_class=(klass)
30
+ @_storage_class = klass
31
31
  end
32
32
 
33
33
  # Upfront generation of picture thumbnails
@@ -42,14 +42,14 @@ module Alchemy
42
42
  Alchemy::Picture::THUMBNAIL_SIZES.values.each do |size|
43
43
  variant = Alchemy::PictureVariant.new(picture, {
44
44
  size: size,
45
- flatten: true,
45
+ flatten: true
46
46
  })
47
47
  signature = Alchemy::PictureThumb::Signature.call(variant)
48
48
  thumb = find_by(signature: signature)
49
49
  next if thumb
50
50
 
51
51
  uid = Alchemy::PictureThumb::Uid.call(signature, variant)
52
- generator_class.call(variant, signature, uid)
52
+ Alchemy::PictureThumb::Create.call(variant, signature, uid)
53
53
  end
54
54
  end
55
55
  end
@@ -54,8 +54,7 @@ module Alchemy
54
54
  raise MissingImageFileError, "Missing image file for #{picture.inspect}" if image.nil?
55
55
 
56
56
  image = processed_image(image, @options)
57
- image = encoded_image(image, @options)
58
- image
57
+ encoded_image(image, @options)
59
58
  rescue MissingImageFileError, WrongImageFormatError => e
60
59
  log_warning(e.message)
61
60
  nil
@@ -88,7 +87,7 @@ module Alchemy
88
87
  end
89
88
 
90
89
  options = {
91
- flatten: !render_format.in?(ANIMATED_IMAGE_FORMATS) && picture.image_file_format == "gif",
90
+ flatten: !render_format.in?(ANIMATED_IMAGE_FORMATS) && picture.image_file_format == "gif"
92
91
  }.with_indifferent_access.merge(options)
93
92
 
94
93
  encoding_options = []
@@ -13,6 +13,14 @@
13
13
  # The original Tag model is Gutentag::Tag
14
14
  module Alchemy
15
15
  class Tag < Gutentag::Tag
16
+ def self.ransackable_attributes(_auth_object = nil)
17
+ %w[created_at id name taggings_count updated_at]
18
+ end
19
+
20
+ def self.ransackable_associations(_auth_object = nil)
21
+ %w[taggings]
22
+ end
23
+
16
24
  # Replaces tag with new tag on all models tagged with tag.
17
25
  def self.replace(tag, new_tag)
18
26
  tag.taggings.collect(&:taggable).each do |taggable|
@@ -20,7 +20,7 @@ module Alchemy
20
20
  #
21
21
  # === Example:
22
22
  #
23
- # essence_picture.picture_url(size: '200x300', crop: true, format: 'gif')
23
+ # picture_view.picture_url(size: '200x300', crop: true, format: 'gif')
24
24
  # # '/pictures/1/show/200x300/crop/cats.gif?sh=765rfghj'
25
25
  #
26
26
  # @option options size [String]
@@ -56,7 +56,7 @@ module Alchemy
56
56
  crop: crop,
57
57
  crop_from: crop && crop_from.presence || nil,
58
58
  crop_size: crop && crop_size.presence || nil,
59
- size: settings[:size],
59
+ size: settings[:size]
60
60
  }.with_indifferent_access
61
61
  end
62
62
 
@@ -84,7 +84,7 @@ module Alchemy
84
84
  crop_from: crop && crop_from.presence || default_crop_from&.join("x"),
85
85
  crop_size: crop && crop_size.presence || default_crop_size&.join("x"),
86
86
  flatten: true,
87
- format: picture&.image_file_format || "jpg",
87
+ format: picture&.image_file_format || "jpg"
88
88
  }
89
89
  end
90
90
 
@@ -96,16 +96,16 @@ module Alchemy
96
96
  default_crop_size: default_crop_size,
97
97
  fixed_ratio: settings[:fixed_ratio],
98
98
  image_width: picture&.image_file_width,
99
- image_height: picture&.image_file_height,
99
+ image_height: picture&.image_file_height
100
100
  ).to_h
101
101
  end
102
102
 
103
- # Show image cropping link for content
103
+ # Show image cropping link for ingredient
104
104
  def allow_image_cropping?
105
105
  settings[:crop] && picture &&
106
106
  picture.can_be_cropped_to?(
107
107
  settings[:size],
108
- settings[:upsample],
108
+ settings[:upsample]
109
109
  ) && !!picture.image_file
110
110
  end
111
111
 
@@ -124,7 +124,7 @@ module Alchemy
124
124
  def thumbnail_zoom_factor(mask)
125
125
  [
126
126
  mask[0].to_f / (image_file_width || 1),
127
- mask[1].to_f / (image_file_height || 1),
127
+ mask[1].to_f / (image_file_height || 1)
128
128
  ].max
129
129
  end
130
130
 
@@ -134,7 +134,7 @@ module Alchemy
134
134
 
135
135
  [
136
136
  ((image_file_width || 0) - default_crop_size[0]) / 2,
137
- ((image_file_height || 0) - default_crop_size[1]) / 2,
137
+ ((image_file_height || 0) - default_crop_size[1]) / 2
138
138
  ].map(&:round)
139
139
  end
140
140
 
@@ -14,7 +14,7 @@ module Alchemy
14
14
  # The attributes to be serialized. See ActiveModel::Serialization.
15
15
  # By default, serialize all columns from the AR object.
16
16
  def attributes
17
- Hash[object.class.column_names.map { |c| [c, nil] }]
17
+ object.class.column_names.map { |c| [c, nil] }.to_h
18
18
  end
19
19
 
20
20
  private
@@ -10,16 +10,11 @@ module Alchemy
10
10
  :tag_list,
11
11
  :created_at,
12
12
  :updated_at,
13
- :ingredients,
14
- :content_ids,
15
13
  :dom_id,
16
14
  :display_name
17
15
 
18
16
  has_many :nested_elements
19
-
20
- def ingredients
21
- object.contents.collect(&:serialize)
22
- end
17
+ has_many :ingredients
23
18
 
24
19
  def display_name
25
20
  object.display_name_with_preview_text
@@ -3,13 +3,13 @@
3
3
  module Alchemy
4
4
  class PageTreeSerializer < BaseSerializer
5
5
  def attributes
6
- { "pages" => nil }
6
+ {"pages" => nil}
7
7
  end
8
8
 
9
9
  def pages
10
10
  tree = []
11
- path = [{ id: object.parent_id, children: tree }]
12
- page_list = object.self_and_descendants.includes(:public_version, { language: :site })
11
+ path = [{id: object.parent_id, children: tree}]
12
+ page_list = object.self_and_descendants.includes(:public_version, {language: :site})
13
13
  base_level = object.level - 1
14
14
  # Load folded pages in advance
15
15
  folded_user_pages = FoldedPage.folded_for_user(opts[:user]).pluck(:page_id)
@@ -61,7 +61,7 @@ module Alchemy
61
61
  level: level,
62
62
  root: page.root?,
63
63
  root_or_leaf: page.root? || page.leaf?,
64
- children: [],
64
+ children: []
65
65
  }
66
66
 
67
67
  if opts[:elements]
@@ -75,7 +75,7 @@ module Alchemy
75
75
  locked: page.locked?,
76
76
  locked_notice: page.locked? ? Alchemy.t("This page is locked", name: page.locker_name) : nil,
77
77
  permissions: page_permissions(page, opts[:ability]),
78
- status_titles: page_status_titles(page),
78
+ status_titles: page_status_titles(page)
79
79
  })
80
80
  else
81
81
  p_hash
@@ -98,14 +98,14 @@ module Alchemy
98
98
  copy: ability.can?(:copy, page),
99
99
  destroy: ability.can?(:destroy, page),
100
100
  create: ability.can?(:create, Alchemy::Page),
101
- edit_content: ability.can?(:edit_content, page),
101
+ edit_content: ability.can?(:edit_content, page)
102
102
  }
103
103
  end
104
104
 
105
105
  def page_status_titles(page)
106
106
  {
107
107
  public: page.status_title(:public),
108
- restricted: page.status_title(:restricted),
108
+ restricted: page.status_title(:restricted)
109
109
  }
110
110
  end
111
111
  end
@@ -3,6 +3,7 @@
3
3
  module Alchemy
4
4
  class DeleteElements
5
5
  class WouldLeaveOrphansError < StandardError; end
6
+
6
7
  attr_reader :elements
7
8
 
8
9
  def initialize(elements)
@@ -14,13 +15,6 @@ module Alchemy
14
15
  raise WouldLeaveOrphansError
15
16
  end
16
17
 
17
- contents = Alchemy::Content.where(element_id: elements.map(&:id))
18
- contents.group_by(&:essence_type)
19
- .transform_values! { |value| value.map(&:essence_id) }
20
- .each do |class_name, ids|
21
- class_name.constantize.where(id: ids).delete_all
22
- end
23
- contents.delete_all
24
18
  Gutentag::Tagging.where(taggable: elements).delete_all
25
19
  delete_elements
26
20
  end
@@ -10,7 +10,7 @@ module Alchemy
10
10
  "id",
11
11
  "folded",
12
12
  "updated_at",
13
- "updater_id",
13
+ "updater_id"
14
14
  ].freeze
15
15
 
16
16
  attr_reader :source_element, :repository
@@ -25,20 +25,15 @@ module Alchemy
25
25
  .except(*SKIPPED_ATTRIBUTES_ON_COPY)
26
26
  .merge(differences)
27
27
  .merge(
28
- autogenerate_contents: false,
29
28
  autogenerate_ingredients: false,
30
29
  autogenerate_nested_elements: false,
31
- tags: source_element.tags,
30
+ tags: source_element.tags
32
31
  )
33
32
 
34
33
  new_element = Element.new(attributes)
35
34
  new_element.ingredients = source_element.ingredients.map(&:dup)
36
35
  new_element.save!
37
36
 
38
- source_element.contents.map do |content|
39
- Content.copy(content, element: new_element)
40
- end
41
-
42
37
  nested_elements = repository.children_of(source_element)
43
38
  Element.acts_as_list_no_update do
44
39
  nested_elements.each.with_index(1) do |nested_element, position|
@@ -11,7 +11,7 @@ module Alchemy
11
11
  end
12
12
 
13
13
  def call
14
- klass.validates :name, presence: true, uniqueness: { case_sensitive: true }
14
+ klass.validates :name, presence: true, uniqueness: {case_sensitive: true}
15
15
  end
16
16
 
17
17
  private
@@ -20,7 +20,7 @@
20
20
 
21
21
  <% if element.editable? %>
22
22
  <%= form_for [alchemy, :admin, element], remote: true,
23
- html: {id: "element_#{element.id}_form".html_safe, class: 'element-content'} do |f| %>
23
+ html: {id: "element_#{element.id}_form".html_safe, class: 'element-body'} do |f| %>
24
24
 
25
25
  <div id="element_<%= element.id %>_errors" class="element_errors"></div>
26
26
 
@@ -31,35 +31,18 @@
31
31
 
32
32
  <!-- Each ingredient group -->
33
33
  <% element.ingredients.select { |i| i.definition[:group] }.group_by { |i| i.definition[:group] }.each do |group, ingredients| %>
34
- <div class="content-group">
35
- <%= link_to '#', id: "element_#{element.id}_content_group_#{group.parameterize.underscore}_header", class: 'content-group-header', data: { toggle_content_group: true } do %>
34
+ <div class="ingredient-group">
35
+ <%= link_to '#', id: "element_#{element.id}_ingredient_group_#{group.parameterize.underscore}_header", class: 'ingredient-group-header', data: { toggle_ingredient_group: true } do %>
36
36
  <%= element.translated_group group %>
37
- <i class="content-group-expand icon fa-fw fa-angle-down fas"></i>
37
+ <i class="ingredient-group-expand icon fa-fw fa-angle-down fas"></i>
38
38
  <% end %>
39
- <%= content_tag :div, id: "element_#{element.id}_content_group_#{group.parameterize.underscore}", class: 'content-group-contents' do %>
39
+ <%= content_tag :div, id: "element_#{element.id}_ingredient_group_#{group.parameterize.underscore}", class: 'ingredient-group-ingredients' do %>
40
40
  <%= render ingredients, element_form: f %>
41
41
  <% end %>
42
42
  </div>
43
43
  <% end %>
44
44
  </div>
45
45
  <% end %>
46
- <!-- Contents -->
47
- <div id="element_<%= element.id %>_content" class="element-content-editors">
48
- <%= render element.contents.select { |c| !c.definition[:group] } %>
49
-
50
- <!-- Each content group -->
51
- <% element.contents.select { |c| c.definition[:group] }.group_by { |c| c.definition[:group] }.each do |group, contents| %>
52
- <div class="content-group">
53
- <%= link_to '#', id: "element_#{element.id}_content_group_#{group.parameterize.underscore}_header", class: 'content-group-header', data: { toggle_content_group: true } do %>
54
- <%= element.translated_group group %>
55
- <i class="content-group-expand icon fa-fw fa-angle-down fas"></i>
56
- <% end %>
57
- <%= content_tag :div, id: "element_#{element.id}_content_group_#{group.parameterize.underscore}", class: 'content-group-contents' do %>
58
- <%= render contents, element_form: f %>
59
- <% end %>
60
- </div>
61
- <% end %>
62
- </div>
63
46
 
64
47
  <% if element.taggable? %>
65
48
  <div class="autocomplete_tag_list">
@@ -73,6 +56,9 @@
73
56
  <% end %>
74
57
  <% end %>
75
58
 
59
+ <%# We need to render nested elements even if the element is folded,
60
+ because we need the element present in the DOM for the feature
61
+ "click element in the preview => load and expand element editor". %>
76
62
  <% if element.nestable_elements.any? %>
77
63
  <div class="nestable-elements">
78
64
  <%= content_tag :div,
@@ -34,7 +34,7 @@
34
34
 
35
35
  Alchemy.growl('<%= Alchemy.t(:successfully_added_element) %>');
36
36
  Alchemy.closeCurrentDialog();
37
- Alchemy.Tinymce.init(<%= (@element.richtext_contents_ids + @element.richtext_ingredients_ids).to_json %>);
37
+ Alchemy.Tinymce.init(<%= @element.richtext_ingredients_ids.to_json %>);
38
38
  Alchemy.PreviewWindow.refresh(function() {
39
39
  Alchemy.ElementEditors.focusElementPreview(<%= @element.id %>);
40
40
  });
@@ -14,12 +14,12 @@
14
14
 
15
15
  <% if @element.folded? -%>
16
16
 
17
- Alchemy.Tinymce.remove(<%= (@element.richtext_contents_ids + @element.richtext_ingredients_ids).to_json %>);
17
+ Alchemy.Tinymce.remove(<%= @element.richtext_ingredients_ids.to_json %>);
18
18
 
19
19
  <% else -%>
20
20
 
21
21
  $el.trigger('FocusElementEditor.Alchemy');
22
- Alchemy.Tinymce.init(<%= (@element.richtext_contents_ids + @element.richtext_ingredients_ids).to_json %>);
22
+ Alchemy.Tinymce.init(<%= @element.richtext_ingredients_ids.to_json %>);
23
23
  Alchemy.GUI.initElement($el);
24
24
  Alchemy.SortableElements(
25
25
  <%= @page.id %>,
@@ -1,5 +1,5 @@
1
1
  (function() {
2
- <% if @parent_element && @parent_element.contents.empty? %>
2
+ <% if @parent_element && @parent_element.ingredients.empty? %>
3
3
  var $parent = $('#element_<%= @parent_element.id %>');
4
4
  Alchemy.ElementEditors.setTitle(
5
5
  $parent,
@@ -1,7 +1,7 @@
1
1
  (function() {
2
2
  var $el = $('#element_<%= @element.id %>');
3
3
  var $errors = $('#element_<%= @element.id %>_errors');
4
- $('> .element-content .content_editor, > .element-content .ingredient-editor', $el).removeClass('validation_failed');
4
+ $('> .element-body .ingredient-editor', $el).removeClass('validation_failed');
5
5
 
6
6
  <%- if @element_validated -%>
7
7
 
@@ -20,7 +20,6 @@
20
20
  Alchemy.growl('<%= j @notice %>', 'warn');
21
21
  $errors.html('<%= j @error_message %><ul><li><%== j @error_messages.join("</li><li>") %></li></ul>');
22
22
  $errors.show();
23
- $('<%= @element.contents_with_errors.map { |content| "#" + content.dom_id }.join(", ") %>').addClass('validation_failed');
24
23
  $('<%== @element.ingredients_with_errors.map { |ingredient| "[data-ingredient-id=\"#{ingredient.id}\"]" }.join(", ") %>').addClass('validation_failed');
25
24
  Alchemy.Buttons.enable($el);
26
25
 
@@ -8,8 +8,8 @@
8
8
  <ul></ul>
9
9
  </div>
10
10
  <div class="input text">
11
- <label for="external_url" class="control-label">URL</label>
12
- <%= text_field_tag "external_url" %>
11
+ <label for="external_link" class="control-label">URL</label>
12
+ <%= text_field_tag "external_link" %>
13
13
  </div>
14
14
  <div class="input text">
15
15
  <label for="external_link_title" class="control-label">
@@ -3,10 +3,10 @@
3
3
  <h3><%= Alchemy.t(:choose_file_to_link) %></h3>
4
4
  <% end %>
5
5
  <div class="input select">
6
- <label for="public_filename" class="control-label">
6
+ <label for="file_link" class="control-label">
7
7
  <%= Alchemy.t(:file) %>
8
8
  </label>
9
- <%= select_tag "public_filename",
9
+ <%= select_tag "file_link",
10
10
  options_for_select(@attachments),
11
11
  prompt: Alchemy.t('Please choose'),
12
12
  class: "alchemy_selectbox" %>
@@ -4,10 +4,10 @@
4
4
  <p><%= Alchemy.t(:internal_link_page_elements_explanation) %></p>
5
5
  <% end %>
6
6
  <div class="input select">
7
- <label for="page_urlname" class="control-label">
7
+ <label for="internal_link" class="control-label">
8
8
  <%= Alchemy.t(:page) %>
9
9
  </label>
10
- <input type="text" id="page_urlname" class="alchemy_selectbox full_width">
10
+ <input type="text" id="internal_link" class="alchemy_selectbox full_width">
11
11
  </div>
12
12
  <div class="input select">
13
13
  <label for="element_anchor" class="control-label">
@@ -138,9 +138,6 @@
138
138
  <% end %>
139
139
 
140
140
  <% content_for :javascripts do %>
141
- <% if Alchemy::Tinymce.custom_configs_present?(@page) %>
142
- <%= render 'tinymce_custom_config' %>
143
- <% end %>
144
141
 
145
142
  <% content_for :javascript_includes do %>
146
143
  <meta name="turbolinks-cache-control" content="no-cache">
@@ -197,7 +194,7 @@
197
194
  Alchemy.SortableElements(<%= @page.id %>, '<%= form_authenticity_token %>');
198
195
  Alchemy.ElementEditors.init();
199
196
  Alchemy.SelectBox('.element-editor');
200
- Alchemy.Tinymce.init(<%= (@page.richtext_contents_ids + @page.richtext_ingredients_ids).to_json %>);
197
+ Alchemy.Tinymce.init(<%= @page.richtext_ingredients_ids.to_json %>);
201
198
  $('#fixed-elements').tabs().tabs('paging', {
202
199
  follow: true,
203
200
  followOnSelect: true,