alchemy_cms 4.4.4 → 5.0.0.beta2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (386) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/ci.yml +56 -18
  3. data/.github/workflows/stale.yml +1 -1
  4. data/.gitignore +20 -2
  5. data/.hound.yml +2 -1
  6. data/.prettierrc +6 -0
  7. data/.rubocop.yml +33 -20
  8. data/CHANGELOG.md +121 -0
  9. data/Gemfile +24 -22
  10. data/README.md +31 -19
  11. data/Rakefile +10 -8
  12. data/alchemy_cms.gemspec +7 -5
  13. data/app/assets/javascripts/alchemy/admin.js +1 -2
  14. data/app/assets/javascripts/alchemy/alchemy.base.js.coffee +2 -1
  15. data/app/assets/javascripts/alchemy/alchemy.element_editors.js.coffee +0 -2
  16. data/app/assets/javascripts/alchemy/alchemy.link_dialog.js.coffee +5 -5
  17. data/app/assets/javascripts/alchemy/alchemy.page_sorter.js +17 -17
  18. data/app/assets/javascripts/alchemy/node_select.js +39 -0
  19. data/app/assets/javascripts/alchemy/templates/index.js +2 -0
  20. data/app/assets/javascripts/alchemy/templates/node.hbs +16 -0
  21. data/app/assets/javascripts/alchemy/templates/node_folder.hbs +3 -0
  22. data/app/assets/javascripts/alchemy/templates/page.hbs +1 -1
  23. data/app/assets/stylesheets/alchemy/_mixins.scss +2 -3
  24. data/app/assets/stylesheets/alchemy/_variables.scss +2 -2
  25. data/app/assets/stylesheets/alchemy/admin.scss +3 -2
  26. data/app/assets/stylesheets/alchemy/base.scss +0 -1
  27. data/app/assets/stylesheets/alchemy/elements.scss +1 -1
  28. data/app/assets/stylesheets/alchemy/forms.scss +5 -0
  29. data/app/assets/stylesheets/alchemy/lists.scss +0 -8
  30. data/app/assets/stylesheets/alchemy/node-select.scss +43 -0
  31. data/app/assets/stylesheets/alchemy/nodes.scss +6 -1
  32. data/app/assets/stylesheets/alchemy/sitemap.scss +63 -21
  33. data/app/assets/stylesheets/alchemy/tables.scss +1 -24
  34. data/app/controllers/alchemy/admin/attachments_controller.rb +8 -7
  35. data/app/controllers/alchemy/admin/base_controller.rb +13 -33
  36. data/app/controllers/alchemy/admin/clipboard_controller.rb +5 -4
  37. data/app/controllers/alchemy/admin/contents_controller.rb +1 -2
  38. data/app/controllers/alchemy/admin/dashboard_controller.rb +10 -9
  39. data/app/controllers/alchemy/admin/elements_controller.rb +20 -20
  40. data/app/controllers/alchemy/admin/essence_pictures_controller.rb +12 -14
  41. data/app/controllers/alchemy/admin/languages_controller.rb +35 -2
  42. data/app/controllers/alchemy/admin/layoutpages_controller.rb +5 -2
  43. data/app/controllers/alchemy/admin/nodes_controller.rb +5 -14
  44. data/app/controllers/alchemy/admin/pages_controller.rb +50 -63
  45. data/app/controllers/alchemy/admin/pictures_controller.rb +16 -16
  46. data/app/controllers/alchemy/admin/resources_controller.rb +21 -13
  47. data/app/controllers/alchemy/admin/sites_controller.rb +18 -0
  48. data/app/controllers/alchemy/admin/styleguide_controller.rb +1 -0
  49. data/app/controllers/alchemy/admin/tags_controller.rb +5 -3
  50. data/app/controllers/alchemy/admin/trash_controller.rb +6 -6
  51. data/app/controllers/alchemy/api/base_controller.rb +2 -2
  52. data/app/controllers/alchemy/api/contents_controller.rb +4 -4
  53. data/app/controllers/alchemy/api/elements_controller.rb +8 -8
  54. data/app/controllers/alchemy/api/nodes_controller.rb +65 -0
  55. data/app/controllers/alchemy/api/pages_controller.rb +15 -22
  56. data/app/controllers/alchemy/attachments_controller.rb +5 -5
  57. data/app/controllers/alchemy/base_controller.rb +10 -9
  58. data/app/controllers/alchemy/messages_controller.rb +16 -23
  59. data/app/controllers/alchemy/pages_controller.rb +13 -11
  60. data/app/controllers/concerns/alchemy/admin/archive_overlay.rb +3 -2
  61. data/app/controllers/concerns/alchemy/admin/current_language.rb +23 -0
  62. data/app/controllers/concerns/alchemy/admin/uploader_responses.rb +5 -5
  63. data/app/controllers/concerns/alchemy/legacy_page_redirects.rb +4 -4
  64. data/app/controllers/concerns/alchemy/page_redirects.rb +2 -9
  65. data/app/controllers/concerns/alchemy/site_redirects.rb +2 -2
  66. data/app/decorators/alchemy/content_editor.rb +55 -0
  67. data/app/decorators/alchemy/element_editor.rb +39 -0
  68. data/app/helpers/alchemy/admin/attachments_helper.rb +6 -6
  69. data/app/helpers/alchemy/admin/base_helper.rb +36 -35
  70. data/app/helpers/alchemy/admin/contents_helper.rb +3 -3
  71. data/app/helpers/alchemy/admin/elements_helper.rb +3 -88
  72. data/app/helpers/alchemy/admin/essences_helper.rb +8 -117
  73. data/app/helpers/alchemy/admin/form_helper.rb +1 -1
  74. data/app/helpers/alchemy/admin/navigation_helper.rb +24 -23
  75. data/app/helpers/alchemy/admin/pages_helper.rb +12 -12
  76. data/app/helpers/alchemy/admin/pictures_helper.rb +4 -6
  77. data/app/helpers/alchemy/admin/tags_helper.rb +8 -7
  78. data/app/helpers/alchemy/base_helper.rb +13 -8
  79. data/app/helpers/alchemy/elements_block_helper.rb +2 -31
  80. data/app/helpers/alchemy/elements_helper.rb +12 -58
  81. data/app/helpers/alchemy/pages_helper.rb +24 -174
  82. data/app/helpers/alchemy/url_helper.rb +2 -1
  83. data/app/mailers/alchemy/base_mailer.rb +1 -1
  84. data/app/mailers/alchemy/messages_mailer.rb +1 -1
  85. data/app/models/alchemy/attachment.rb +20 -18
  86. data/app/models/alchemy/base_record.rb +2 -5
  87. data/app/models/alchemy/content.rb +33 -52
  88. data/app/models/alchemy/content/factory.rb +24 -31
  89. data/app/models/alchemy/element.rb +45 -53
  90. data/app/models/alchemy/element/definitions.rb +4 -4
  91. data/app/models/alchemy/element/element_contents.rb +9 -6
  92. data/app/models/alchemy/element/element_essences.rb +4 -3
  93. data/app/models/alchemy/element/presenters.rb +3 -2
  94. data/app/models/alchemy/element_to_page.rb +1 -1
  95. data/app/models/alchemy/essence_boolean.rb +1 -3
  96. data/app/models/alchemy/essence_date.rb +2 -3
  97. data/app/models/alchemy/essence_file.rb +4 -4
  98. data/app/models/alchemy/essence_html.rb +1 -3
  99. data/app/models/alchemy/essence_link.rb +1 -3
  100. data/app/models/alchemy/essence_node.rb +18 -0
  101. data/app/models/alchemy/essence_page.rb +3 -16
  102. data/app/models/alchemy/essence_picture.rb +17 -16
  103. data/app/models/alchemy/essence_picture_view.rb +7 -6
  104. data/app/models/alchemy/essence_richtext.rb +1 -3
  105. data/app/models/alchemy/essence_select.rb +1 -3
  106. data/app/models/alchemy/essence_text.rb +0 -2
  107. data/app/models/alchemy/folded_page.rb +1 -0
  108. data/app/models/alchemy/language.rb +21 -35
  109. data/app/models/alchemy/language/code.rb +4 -4
  110. data/app/models/alchemy/legacy_page_url.rb +1 -1
  111. data/app/models/alchemy/message.rb +3 -3
  112. data/app/models/alchemy/node.rb +55 -9
  113. data/app/models/alchemy/page.rb +53 -123
  114. data/app/models/alchemy/page/fixed_attributes.rb +3 -2
  115. data/app/models/alchemy/page/page_elements.rb +35 -44
  116. data/app/models/alchemy/page/page_naming.rb +20 -70
  117. data/app/models/alchemy/page/page_natures.rb +7 -34
  118. data/app/models/alchemy/page/page_scopes.rb +23 -29
  119. data/app/models/alchemy/page/url_path.rb +64 -0
  120. data/app/models/alchemy/picture.rb +40 -30
  121. data/app/models/alchemy/picture/preprocessor.rb +26 -0
  122. data/app/models/alchemy/picture/transformations.rb +9 -7
  123. data/app/models/alchemy/picture/url.rb +5 -5
  124. data/app/models/alchemy/site.rb +6 -36
  125. data/app/models/alchemy/site/layout.rb +2 -2
  126. data/app/models/concerns/alchemy/touch_elements.rb +24 -0
  127. data/app/serializers/alchemy/content_serializer.rb +0 -3
  128. data/app/serializers/alchemy/essence_boolean_serializer.rb +3 -3
  129. data/app/serializers/alchemy/essence_date_serializer.rb +3 -3
  130. data/app/serializers/alchemy/essence_file_serializer.rb +4 -2
  131. data/app/serializers/alchemy/essence_html_serializer.rb +3 -3
  132. data/app/serializers/alchemy/essence_link_serializer.rb +3 -3
  133. data/app/serializers/alchemy/essence_picture_serializer.rb +5 -4
  134. data/app/serializers/alchemy/essence_richtext_serializer.rb +3 -3
  135. data/app/serializers/alchemy/essence_select_serializer.rb +3 -3
  136. data/app/serializers/alchemy/essence_text_serializer.rb +5 -4
  137. data/app/serializers/alchemy/node_serializer.rb +14 -0
  138. data/app/serializers/alchemy/page_serializer.rb +2 -1
  139. data/app/serializers/alchemy/page_tree_serializer.rb +11 -14
  140. data/app/views/alchemy/admin/attachments/_archive_overlay.html.erb +1 -2
  141. data/app/views/alchemy/admin/attachments/_file_to_assign.html.erb +1 -2
  142. data/app/views/alchemy/admin/attachments/_files_list.html.erb +4 -4
  143. data/app/views/alchemy/admin/contents/create.js.erb +1 -3
  144. data/app/views/alchemy/admin/elements/_element.html.erb +9 -18
  145. data/app/views/alchemy/admin/elements/create.js.erb +1 -1
  146. data/app/views/alchemy/admin/elements/fold.js.erb +1 -1
  147. data/app/views/alchemy/admin/elements/index.html.erb +3 -3
  148. data/app/views/alchemy/admin/essence_files/assign.js.erb +1 -6
  149. data/app/views/alchemy/admin/essence_files/edit.html.erb +1 -1
  150. data/app/views/alchemy/admin/essence_pictures/assign.js.erb +1 -7
  151. data/app/views/alchemy/admin/essence_pictures/crop.html.erb +1 -1
  152. data/app/views/alchemy/admin/essence_pictures/edit.html.erb +7 -7
  153. data/app/views/alchemy/admin/essence_pictures/update.js.erb +1 -1
  154. data/app/views/alchemy/admin/languages/_form.html.erb +5 -5
  155. data/app/views/alchemy/admin/languages/_table.html.erb +3 -3
  156. data/app/views/alchemy/admin/languages/edit.html.erb +1 -0
  157. data/app/views/alchemy/admin/languages/index.html.erb +23 -4
  158. data/app/views/alchemy/admin/languages/new.html.erb +1 -0
  159. data/app/views/alchemy/admin/layoutpages/index.html.erb +6 -2
  160. data/app/views/alchemy/admin/nodes/_form.html.erb +23 -15
  161. data/app/views/alchemy/admin/nodes/_node.html.erb +5 -19
  162. data/app/views/alchemy/admin/nodes/index.html.erb +3 -18
  163. data/app/views/alchemy/admin/pages/_create_language_form.html.erb +0 -8
  164. data/app/views/alchemy/admin/pages/_form.html.erb +1 -2
  165. data/app/views/alchemy/admin/pages/_new_page_form.html.erb +1 -0
  166. data/app/views/alchemy/admin/pages/_page.html.erb +14 -25
  167. data/app/views/alchemy/admin/pages/_page_infos.html.erb +0 -4
  168. data/app/views/alchemy/admin/pages/_sitemap.html.erb +6 -0
  169. data/app/views/alchemy/admin/pages/edit.html.erb +1 -1
  170. data/app/views/alchemy/admin/pages/info.html.erb +0 -9
  171. data/app/views/alchemy/admin/pages/unlock.js.erb +13 -6
  172. data/app/views/alchemy/admin/partials/_remote_search_form.html.erb +0 -2
  173. data/app/views/alchemy/admin/partials/_routes.html.erb +8 -0
  174. data/app/views/alchemy/admin/pictures/_filter_and_size_bar.html.erb +1 -5
  175. data/app/views/alchemy/admin/pictures/_overlay_picture_list.html.erb +1 -1
  176. data/app/views/alchemy/admin/pictures/_picture_to_assign.html.erb +1 -2
  177. data/app/views/alchemy/admin/pictures/index.html.erb +18 -3
  178. data/app/views/alchemy/admin/resources/_resource.html.erb +1 -1
  179. data/app/views/alchemy/admin/resources/_table.html.erb +2 -2
  180. data/app/views/alchemy/admin/sites/_form.html.erb +8 -0
  181. data/app/views/alchemy/admin/sites/edit.html.erb +1 -0
  182. data/app/views/alchemy/admin/sites/index.html.erb +21 -9
  183. data/app/views/alchemy/admin/sites/new.html.erb +1 -0
  184. data/app/views/alchemy/admin/tags/index.html.erb +2 -2
  185. data/app/views/alchemy/essences/_essence_boolean_editor.html.erb +10 -12
  186. data/app/views/alchemy/essences/_essence_date_editor.html.erb +11 -8
  187. data/app/views/alchemy/essences/_essence_file_editor.html.erb +16 -17
  188. data/app/views/alchemy/essences/_essence_html_editor.html.erb +8 -5
  189. data/app/views/alchemy/essences/_essence_link_editor.html.erb +18 -15
  190. data/app/views/alchemy/essences/_essence_node_editor.html.erb +27 -0
  191. data/app/views/alchemy/essences/_essence_node_view.html.erb +1 -0
  192. data/app/views/alchemy/essences/_essence_page_editor.html.erb +14 -11
  193. data/app/views/alchemy/essences/_essence_picture_editor.html.erb +22 -20
  194. data/app/views/alchemy/essences/_essence_richtext_editor.html.erb +10 -7
  195. data/app/views/alchemy/essences/_essence_select_editor.html.erb +12 -16
  196. data/app/views/alchemy/essences/_essence_text_editor.html.erb +18 -17
  197. data/app/views/alchemy/essences/shared/_essence_picture_tools.html.erb +6 -11
  198. data/app/views/alchemy/pages/show.rss.builder +3 -2
  199. data/app/views/layouts/alchemy/admin.html.erb +1 -0
  200. data/babel.config.js +12 -0
  201. data/bin/rails +5 -4
  202. data/config/alchemy/config.yml +23 -22
  203. data/config/brakeman.ignore +1 -1
  204. data/config/initializers/assets.rb +2 -1
  205. data/config/initializers/dragonfly.rb +2 -1
  206. data/config/initializers/mime_types.rb +1 -0
  207. data/config/initializers/mini_profiler.rb +3 -2
  208. data/config/initializers/simple_form.rb +6 -6
  209. data/config/locales/alchemy.en.yml +37 -14
  210. data/config/routes.rb +32 -28
  211. data/config/spring.rb +3 -2
  212. data/db/migrate/20200226213334_alchemy_four_point_four.rb +313 -0
  213. data/db/migrate/20200423073425_create_alchemy_essence_nodes.rb +11 -0
  214. data/db/migrate/20200504210159_remove_site_id_from_nodes.rb +28 -0
  215. data/db/migrate/20200505215518_add_language_id_foreign_key_to_alchemy_pages.rb +8 -0
  216. data/db/migrate/20200511113603_add_menu_type_to_alchemy_nodes.rb +27 -0
  217. data/db/migrate/20200514091507_make_page_layoutpage_null_false.rb +6 -0
  218. data/db/migrate/20200519073500_remove_visible_from_alchemy_pages.rb +24 -0
  219. data/lib/alchemy/admin/locale.rb +3 -1
  220. data/lib/alchemy/admin/preview_url.rb +85 -0
  221. data/lib/alchemy/auth_accessors.rb +8 -7
  222. data/lib/alchemy/cache_digests/template_tracker.rb +5 -4
  223. data/lib/alchemy/config.rb +26 -2
  224. data/lib/alchemy/configuration_methods.rb +3 -1
  225. data/lib/alchemy/controller_actions.rb +6 -5
  226. data/lib/alchemy/deprecation.rb +2 -1
  227. data/lib/alchemy/elements_finder.rb +5 -5
  228. data/lib/alchemy/engine.rb +23 -8
  229. data/lib/alchemy/errors.rb +0 -7
  230. data/lib/alchemy/essence.rb +17 -16
  231. data/lib/alchemy/filetypes.rb +5 -5
  232. data/lib/alchemy/forms/builder.rb +4 -4
  233. data/lib/alchemy/hints.rb +1 -1
  234. data/lib/alchemy/i18n.rb +2 -1
  235. data/lib/alchemy/modules.rb +12 -12
  236. data/lib/alchemy/name_conversions.rb +5 -5
  237. data/lib/alchemy/on_page_layout/callbacks_runner.rb +1 -0
  238. data/lib/alchemy/page_layout.rb +15 -12
  239. data/lib/alchemy/paths.rb +1 -1
  240. data/lib/alchemy/permissions.rb +7 -6
  241. data/lib/alchemy/resource.rb +23 -13
  242. data/lib/alchemy/resources_helper.rb +12 -18
  243. data/lib/alchemy/routing_constraints.rb +1 -1
  244. data/lib/alchemy/seeder.rb +42 -14
  245. data/lib/alchemy/shell.rb +13 -10
  246. data/lib/alchemy/taggable.rb +1 -0
  247. data/lib/alchemy/tasks/tidy.rb +4 -3
  248. data/lib/alchemy/test_support/config_stubbing.rb +1 -0
  249. data/lib/alchemy/test_support/essence_shared_examples.rb +72 -72
  250. data/lib/alchemy/test_support/factories.rb +1 -1
  251. data/lib/alchemy/test_support/factories/attachment_factory.rb +5 -5
  252. data/lib/alchemy/test_support/factories/content_factory.rb +6 -6
  253. data/lib/alchemy/test_support/factories/dummy_user_factory.rb +7 -7
  254. data/lib/alchemy/test_support/factories/element_factory.rb +9 -9
  255. data/lib/alchemy/test_support/factories/essence_file_factory.rb +3 -3
  256. data/lib/alchemy/test_support/factories/essence_page_factory.rb +3 -3
  257. data/lib/alchemy/test_support/factories/essence_picture_factory.rb +4 -4
  258. data/lib/alchemy/test_support/factories/essence_text_factory.rb +3 -3
  259. data/lib/alchemy/test_support/factories/language_factory.rb +21 -14
  260. data/lib/alchemy/test_support/factories/node_factory.rb +8 -8
  261. data/lib/alchemy/test_support/factories/page_factory.rb +15 -27
  262. data/lib/alchemy/test_support/factories/picture_factory.rb +5 -5
  263. data/lib/alchemy/test_support/factories/site_factory.rb +7 -6
  264. data/lib/alchemy/test_support/integration_helpers.rb +1 -0
  265. data/lib/alchemy/test_support/shared_contexts.rb +5 -4
  266. data/lib/alchemy/test_support/shared_uploader_examples.rb +4 -3
  267. data/lib/alchemy/tinymce.rb +15 -13
  268. data/lib/alchemy/upgrader.rb +8 -7
  269. data/lib/alchemy/upgrader/five_point_zero.rb +41 -0
  270. data/lib/alchemy/upgrader/tasks/element_views_updater.rb +4 -4
  271. data/lib/alchemy/upgrader/tasks/harden_gutentag_migrations.rb +29 -0
  272. data/lib/alchemy/version.rb +1 -1
  273. data/lib/alchemy_cms.rb +52 -50
  274. data/lib/{rails/generators → generators}/alchemy/base.rb +5 -4
  275. data/lib/{rails/generators → generators}/alchemy/elements/elements_generator.rb +13 -9
  276. data/lib/{rails/generators → generators}/alchemy/elements/templates/view.html.erb +0 -0
  277. data/lib/{rails/generators → generators}/alchemy/elements/templates/view.html.haml +0 -0
  278. data/lib/{rails/generators → generators}/alchemy/elements/templates/view.html.slim +0 -0
  279. data/lib/{rails/generators → generators}/alchemy/essence/essence_generator.rb +15 -13
  280. data/lib/generators/alchemy/essence/templates/editor.html.erb +17 -0
  281. data/lib/{rails/generators → generators}/alchemy/essence/templates/view.html.erb +0 -0
  282. data/lib/{rails/generators → generators}/alchemy/install/files/_article.html.erb +0 -0
  283. data/lib/{rails/generators → generators}/alchemy/install/files/_standard.html.erb +0 -0
  284. data/lib/{rails/generators → generators}/alchemy/install/files/alchemy.en.yml +0 -0
  285. data/lib/generators/alchemy/install/files/alchemy_admin.js +1 -0
  286. data/lib/{rails/generators → generators}/alchemy/install/files/all.css +0 -0
  287. data/lib/{rails/generators → generators}/alchemy/install/files/all.js +0 -0
  288. data/lib/{rails/generators → generators}/alchemy/install/files/application.html.erb +0 -0
  289. data/lib/{rails/generators → generators}/alchemy/install/files/article.scss +0 -0
  290. data/lib/generators/alchemy/install/install_generator.rb +110 -0
  291. data/lib/{rails/generators → generators}/alchemy/install/templates/dragonfly.rb.tt +0 -0
  292. data/lib/{rails/generators → generators}/alchemy/install/templates/elements.yml.tt +0 -0
  293. data/lib/generators/alchemy/install/templates/menus.yml.tt +8 -0
  294. data/lib/{rails/generators → generators}/alchemy/install/templates/page_layouts.yml.tt +0 -0
  295. data/lib/{rails/generators → generators}/alchemy/menus/menus_generator.rb +5 -5
  296. data/lib/{rails/generators → generators}/alchemy/menus/templates/node.html.erb +1 -4
  297. data/lib/{rails/generators → generators}/alchemy/menus/templates/node.html.haml +1 -4
  298. data/lib/{rails/generators → generators}/alchemy/menus/templates/node.html.slim +1 -4
  299. data/lib/{rails/generators → generators}/alchemy/menus/templates/wrapper.html.erb +1 -1
  300. data/lib/{rails/generators → generators}/alchemy/menus/templates/wrapper.html.haml +2 -2
  301. data/lib/{rails/generators → generators}/alchemy/menus/templates/wrapper.html.slim +2 -2
  302. data/lib/{rails/generators → generators}/alchemy/module/module_generator.rb +3 -2
  303. data/lib/{rails/generators → generators}/alchemy/module/templates/ability.rb.tt +0 -0
  304. data/lib/{rails/generators → generators}/alchemy/module/templates/controller.rb.tt +0 -0
  305. data/lib/{rails/generators → generators}/alchemy/module/templates/module_config.rb.tt +0 -0
  306. data/lib/{rails/generators → generators}/alchemy/page_layouts/page_layouts_generator.rb +5 -4
  307. data/lib/{rails/generators → generators}/alchemy/page_layouts/templates/layout.html.erb +0 -0
  308. data/lib/{rails/generators → generators}/alchemy/page_layouts/templates/layout.html.haml +0 -0
  309. data/lib/{rails/generators → generators}/alchemy/page_layouts/templates/layout.html.slim +0 -0
  310. data/lib/{rails/generators → generators}/alchemy/site_layouts/site_layouts_generator.rb +4 -2
  311. data/lib/{rails/generators → generators}/alchemy/site_layouts/templates/layout.html.erb +0 -0
  312. data/lib/{rails/generators → generators}/alchemy/site_layouts/templates/layout.html.haml +0 -0
  313. data/lib/{rails/generators → generators}/alchemy/site_layouts/templates/layout.html.slim +0 -0
  314. data/lib/{rails/generators → generators}/alchemy/views/views_generator.rb +7 -6
  315. data/lib/kaminari/scoped_pagination_url_helper.rb +1 -0
  316. data/lib/tasks/alchemy/db.rake +3 -19
  317. data/lib/tasks/alchemy/install.rake +3 -2
  318. data/lib/tasks/alchemy/tidy.rake +9 -8
  319. data/lib/tasks/alchemy/upgrade.rake +28 -105
  320. data/package.json +45 -0
  321. data/package/admin.js +14 -0
  322. data/package/src/__tests__/i18n.spec.js +70 -0
  323. data/package/src/i18n.js +48 -0
  324. data/package/src/node_tree.js +72 -0
  325. data/package/src/translations.js +32 -0
  326. data/package/src/utils/__tests__/ajax.spec.js +124 -0
  327. data/package/src/utils/__tests__/events.spec.js +38 -0
  328. data/package/src/utils/ajax.js +48 -0
  329. data/package/src/utils/events.js +16 -0
  330. data/vendor/assets/fonts/fa-regular-400.eot +0 -0
  331. data/vendor/assets/fonts/fa-regular-400.svg +798 -358
  332. data/vendor/assets/fonts/fa-regular-400.ttf +0 -0
  333. data/vendor/assets/fonts/fa-regular-400.woff +0 -0
  334. data/vendor/assets/fonts/fa-regular-400.woff2 +0 -0
  335. data/vendor/assets/fonts/fa-solid-900.eot +0 -0
  336. data/vendor/assets/fonts/fa-solid-900.svg +4933 -1408
  337. data/vendor/assets/fonts/fa-solid-900.ttf +0 -0
  338. data/vendor/assets/fonts/fa-solid-900.woff +0 -0
  339. data/vendor/assets/fonts/fa-solid-900.woff2 +0 -0
  340. data/vendor/assets/javascripts/jquery_plugins/jquery.ui.nestedSortable.js +1 -2
  341. data/vendor/assets/stylesheets/fontawesome/_core.scss +5 -0
  342. data/vendor/assets/stylesheets/fontawesome/_fixed-width.scss +1 -1
  343. data/vendor/assets/stylesheets/fontawesome/_icons.scss +651 -2
  344. data/vendor/assets/stylesheets/fontawesome/_mixins.scss +0 -1
  345. data/vendor/assets/stylesheets/fontawesome/_rotated-flipped.scss +3 -2
  346. data/vendor/assets/stylesheets/fontawesome/_stacked.scss +1 -1
  347. data/vendor/assets/stylesheets/fontawesome/_variables.scss +662 -9
  348. data/vendor/assets/stylesheets/fontawesome/fontawesome.scss +2 -2
  349. data/vendor/assets/stylesheets/fontawesome/regular.scss +23 -0
  350. data/vendor/assets/stylesheets/fontawesome/solid.scss +24 -0
  351. metadata +131 -83
  352. data/app/assets/javascripts/alchemy/alchemy.i18n.js.coffee +0 -32
  353. data/app/assets/javascripts/alchemy/alchemy.translations.js.coffee +0 -29
  354. data/app/helpers/alchemy/essences_helper.rb +0 -119
  355. data/app/models/concerns/alchemy/content_touching.rb +0 -23
  356. data/app/serializers/alchemy/legacy_element_serializer.rb +0 -15
  357. data/app/views/alchemy/admin/contents/_missing.html.erb +0 -17
  358. data/app/views/alchemy/admin/pages/_menu_fields.html.erb +0 -33
  359. data/app/views/alchemy/admin/pages/configure_external.html.erb +0 -32
  360. data/app/views/alchemy/elements/_editor_not_found.html.erb +0 -4
  361. data/app/views/alchemy/navigation/_image_link.html.erb +0 -14
  362. data/app/views/alchemy/navigation/_link.html.erb +0 -19
  363. data/app/views/alchemy/navigation/_renderer.html.erb +0 -29
  364. data/db/migrate/20180226123013_alchemy_four_point_zero.rb +0 -363
  365. data/db/migrate/20180227224537_migrate_tags_to_gutentag.rb +0 -41
  366. data/db/migrate/20180519204655_add_fixed_to_alchemy_elements.rb +0 -6
  367. data/db/migrate/20191016073858_create_alchemy_essence_pages.rb +0 -8
  368. data/db/migrate/20191029212236_create_alchemy_nodes.rb +0 -24
  369. data/db/migrate/20200226081535_add_site_id_to_alchemy_nodes.rb +0 -15
  370. data/lib/alchemy/ssl_protection.rb +0 -32
  371. data/lib/alchemy/tasks/helpers.rb +0 -81
  372. data/lib/alchemy/test_support/controller_requests.rb +0 -93
  373. data/lib/alchemy/upgrader/four_point_four.rb +0 -52
  374. data/lib/alchemy/upgrader/four_point_one.rb +0 -42
  375. data/lib/alchemy/upgrader/four_point_two.rb +0 -85
  376. data/lib/alchemy/upgrader/tasks/cells_migration.rb +0 -43
  377. data/lib/alchemy/upgrader/tasks/cells_upgrader.rb +0 -148
  378. data/lib/alchemy/upgrader/tasks/element_partial_name_variable_updater.rb +0 -28
  379. data/lib/alchemy/upgrader/tasks/harden_acts_as_taggable_on_migrations.rb +0 -27
  380. data/lib/alchemy/upgrader/tasks/picture_gallery_migration.rb +0 -65
  381. data/lib/alchemy/upgrader/tasks/picture_gallery_upgrader.rb +0 -210
  382. data/lib/rails/generators/alchemy/essence/templates/editor.html.erb +0 -15
  383. data/lib/rails/generators/alchemy/install/install_generator.rb +0 -60
  384. data/lib/tasks/alchemy/convert.rake +0 -95
  385. data/vendor/assets/stylesheets/fontawesome/fa-regular.scss +0 -22
  386. data/vendor/assets/stylesheets/fontawesome/fa-solid.scss +0 -23
@@ -21,20 +21,17 @@ module Alchemy
21
21
 
22
22
  # All pages locked by given user
23
23
  #
24
- scope :locked_by, ->(user) {
25
- if user.class.respond_to? :primary_key
26
- locked.where(locked_by: user.send(user.class.primary_key))
27
- end
28
- }
24
+ scope :locked_by,
25
+ ->(user) {
26
+ if user.class.respond_to? :primary_key
27
+ locked.where(locked_by: user.send(user.class.primary_key))
28
+ end
29
+ }
29
30
 
30
31
  # All not locked pages
31
32
  #
32
33
  scope :not_locked, -> { where(locked_at: nil, locked_by: nil) }
33
34
 
34
- # All visible pages
35
- #
36
- scope :visible, -> { where(visible: true) }
37
-
38
35
  # All not restricted pages
39
36
  #
40
37
  scope :not_restricted, -> { where(restricted: false) }
@@ -45,29 +42,27 @@ module Alchemy
45
42
 
46
43
  # All pages that are a published language root
47
44
  #
48
- scope :public_language_roots, -> {
49
- published.language_roots.where(
50
- language_code: Language.published.pluck(:language_code)
51
- )
52
- }
45
+ scope :public_language_roots,
46
+ -> {
47
+ published.language_roots.where(
48
+ language_code: Language.published.pluck(:language_code),
49
+ )
50
+ }
53
51
 
54
52
  # Last 5 pages that where recently edited by given user
55
53
  #
56
- scope :all_last_edited_from, ->(user) {
57
- where(updater_id: user.id).order('updated_at DESC').limit(5)
58
- }
54
+ scope :all_last_edited_from,
55
+ ->(user) {
56
+ where(updater_id: user.id).order("updated_at DESC").limit(5)
57
+ }
59
58
 
60
59
  # Returns all pages that have the given +language_id+
61
60
  #
62
- scope :with_language, ->(language_id) {
63
- where(language_id: language_id)
64
- }
61
+ scope :with_language, ->(language_id) { where(language_id: language_id) }
65
62
 
66
63
  # Returns all content pages.
67
64
  #
68
- scope :contentpages, -> {
69
- where(layoutpage: [false, nil]).where.not(parent_id: nil)
70
- }
65
+ scope :contentpages, -> { where(layoutpage: [false, nil]) }
71
66
 
72
67
  # Returns all public contentpages that are not locked.
73
68
  #
@@ -79,9 +74,7 @@ module Alchemy
79
74
  #
80
75
  # Used for flushing all pages caches at once.
81
76
  #
82
- scope :flushable_layoutpages, -> {
83
- not_locked.layoutpages.where.not(parent_id: Page.unscoped.root.id)
84
- }
77
+ scope :flushable_layoutpages, -> { not_locked.layoutpages }
85
78
 
86
79
  # All searchable pages
87
80
  #
@@ -89,9 +82,10 @@ module Alchemy
89
82
 
90
83
  # All pages from +Alchemy::Site.current+
91
84
  #
92
- scope :from_current_site, -> {
93
- where(Language.table_name => {site_id: Site.current || Site.default}).joins(:language)
94
- }
85
+ scope :from_current_site,
86
+ -> {
87
+ where(Language.table_name => { site_id: Site.current || Site.default }).joins(:language)
88
+ }
95
89
 
96
90
  # All pages for xml sitemap
97
91
  #
@@ -0,0 +1,64 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class Page
5
+ # = The url_path for this page
6
+ #
7
+ # Use this to build relative links to this page
8
+ #
9
+ # It takes several circumstances into account:
10
+ #
11
+ # 1. It returns just a slash for language root pages of the default langauge
12
+ # 2. It returns a url path with a leading slash for regular pages
13
+ # 3. It returns a url path with a leading slash and language code prefix for pages not having the default language
14
+ # 4. It returns a url path with a leading slash and the language code for language root pages of a non-default language
15
+ #
16
+ # == Examples
17
+ #
18
+ # Using Rails' link_to helper
19
+ #
20
+ # link_to page.url
21
+ #
22
+ class UrlPath
23
+ ROOT_PATH = "/"
24
+
25
+ def initialize(page)
26
+ @page = page
27
+ @language = @page.language
28
+ @site = @language.site
29
+ end
30
+
31
+ def call
32
+ if @page.language_root?
33
+ language_root_path
34
+ elsif @site.languages.select(&:public?).length > 1
35
+ page_path_with_language_prefix
36
+ else
37
+ page_path_with_leading_slash
38
+ end
39
+ end
40
+
41
+ private
42
+
43
+ def language_root_path
44
+ @language.default? ? ROOT_PATH : language_path
45
+ end
46
+
47
+ def page_path_with_language_prefix
48
+ @language.default? ? page_path : language_path + page_path
49
+ end
50
+
51
+ def page_path_with_leading_slash
52
+ @page.language_root? ? ROOT_PATH : page_path
53
+ end
54
+
55
+ def language_path
56
+ "/#{@page.language_code}"
57
+ end
58
+
59
+ def page_path
60
+ "/#{@page.urlname}"
61
+ end
62
+ end
63
+ end
64
+ end
@@ -22,17 +22,23 @@
22
22
 
23
23
  module Alchemy
24
24
  class Picture < BaseRecord
25
+ THUMBNAIL_SIZES = {
26
+ small: "80x60",
27
+ medium: "160x120",
28
+ large: "240x180",
29
+ }.with_indifferent_access.freeze
30
+
25
31
  CONVERTIBLE_FILE_FORMATS = %w(gif jpg jpeg png).freeze
26
32
 
27
33
  include Alchemy::NameConversions
28
34
  include Alchemy::Taggable
29
- include Alchemy::ContentTouching
35
+ include Alchemy::TouchElements
30
36
  include Alchemy::Picture::Transformations
31
37
  include Alchemy::Picture::Url
32
38
 
33
39
  has_many :essence_pictures,
34
- class_name: 'Alchemy::EssencePicture',
35
- foreign_key: 'picture_id',
40
+ class_name: "Alchemy::EssencePicture",
41
+ foreign_key: "picture_id",
36
42
  inverse_of: :ingredient_association
37
43
 
38
44
  has_many :contents, through: :essence_pictures
@@ -50,24 +56,37 @@ module Alchemy
50
56
  raise PictureInUseError, Alchemy.t(:cannot_delete_picture_notice) % { name: name }
51
57
  end
52
58
 
59
+ # Image preprocessing class
60
+ def self.preprocessor_class
61
+ @_preprocessor_class ||= Preprocessor
62
+ end
63
+
64
+ # Set a image preprocessing class
65
+ #
66
+ # # config/initializers/alchemy.rb
67
+ # Alchemy::Picture.preprocessor_class = My::ImagePreprocessor
68
+ #
69
+ def self.preprocessor_class=(klass)
70
+ @_preprocessor_class = klass
71
+ end
72
+
53
73
  # Enables Dragonfly image processing
54
74
  dragonfly_accessor :image_file, app: :alchemy_pictures do
55
75
  # Preprocess after uploading the picture
56
- after_assign do |p|
57
- resize = Config.get(:preprocess_image_resize)
58
- p.thumb!(resize) if resize.present?
76
+ after_assign do |image|
77
+ self.class.preprocessor_class.new(image).call
59
78
  end
60
79
  end
61
80
 
62
81
  # We need to define this method here to have it available in the validations below.
63
82
  class << self
64
83
  def allowed_filetypes
65
- Config.get(:uploader).fetch('allowed_filetypes', {}).fetch('alchemy/pictures', [])
84
+ Config.get(:uploader).fetch("allowed_filetypes", {}).fetch("alchemy/pictures", [])
66
85
  end
67
86
  end
68
87
 
69
88
  validates_presence_of :image_file
70
- validates_size_of :image_file, maximum: Config.get(:uploader)['file_size_limit'].megabytes
89
+ validates_size_of :image_file, maximum: Config.get(:uploader)["file_size_limit"].megabytes
71
90
  validates_property :format,
72
91
  of: :image_file,
73
92
  in: allowed_filetypes,
@@ -76,21 +95,10 @@ module Alchemy
76
95
 
77
96
  stampable stamper_class_name: Alchemy.user_class_name
78
97
 
79
- scope :named, ->(name) {
80
- where("#{table_name}.name LIKE ?", "%#{name}%")
81
- }
82
-
83
- scope :recent, -> {
84
- where("#{table_name}.created_at > ?", Time.current - 24.hours).order(:created_at)
85
- }
86
-
87
- scope :deletable, -> {
88
- where("#{table_name}.id NOT IN (SELECT picture_id FROM #{EssencePicture.table_name})")
89
- }
90
-
91
- scope :without_tag, -> {
92
- left_outer_joins(:taggings).where(gutentag_taggings: {id: nil})
93
- }
98
+ scope :named, ->(name) { where("#{table_name}.name LIKE ?", "%#{name}%") }
99
+ scope :recent, -> { where("#{table_name}.created_at > ?", Time.current - 24.hours).order(:created_at) }
100
+ scope :deletable, -> { where("#{table_name}.id NOT IN (SELECT picture_id FROM #{EssencePicture.table_name})") }
101
+ scope :without_tag, -> { left_outer_joins(:taggings).where(gutentag_taggings: { id: nil }) }
94
102
 
95
103
  # Class methods
96
104
 
@@ -102,6 +110,7 @@ module Alchemy
102
110
  def last_upload
103
111
  last_picture = Picture.last
104
112
  return Picture.all unless last_picture
113
+
105
114
  Picture.where(upload_hash: last_picture.upload_hash)
106
115
  end
107
116
 
@@ -123,11 +132,11 @@ module Alchemy
123
132
  pictures.order(:name)
124
133
  end
125
134
 
126
- def filtered_by(filter = '')
135
+ def filtered_by(filter = "")
127
136
  case filter
128
- when 'recent' then recent
129
- when 'last_upload' then last_upload
130
- when 'without_tag' then without_tag
137
+ when "recent" then recent
138
+ when "last_upload" then last_upload
139
+ when "without_tag" then without_tag
131
140
  else
132
141
  all
133
142
  end
@@ -166,7 +175,7 @@ module Alchemy
166
175
  {
167
176
  name: image_file_name,
168
177
  size: image_file_size,
169
- error: errors[:image_file].join
178
+ error: errors[:image_file].join,
170
179
  }
171
180
  end
172
181
 
@@ -176,7 +185,7 @@ module Alchemy
176
185
  if name.blank?
177
186
  "image_#{id}"
178
187
  else
179
- ::CGI.escape(name.gsub(/\.(gif|png|jpe?g|tiff?)/i, '').tr('.', ' '))
188
+ ::CGI.escape(name.gsub(/\.(gif|png|jpe?g|tiff?)/i, "").tr(".", " "))
180
189
  end
181
190
  end
182
191
 
@@ -190,6 +199,7 @@ module Alchemy
190
199
  #
191
200
  def humanized_name
192
201
  return "" if image_file_name.blank?
202
+
193
203
  convert_to_humanized_name(image_file_name, suffix)
194
204
  end
195
205
 
@@ -213,7 +223,7 @@ module Alchemy
213
223
  #
214
224
  def convertible?
215
225
  Config.get(:image_output_format) &&
216
- Config.get(:image_output_format) != 'original' &&
226
+ Config.get(:image_output_format) != "original" &&
217
227
  has_convertible_format?
218
228
  end
219
229
 
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class Picture < BaseRecord
5
+ class Preprocessor
6
+ def initialize(image_file)
7
+ @image_file = image_file
8
+ end
9
+
10
+ # Preprocess images after upload
11
+ #
12
+ # Define preprocessing options in the Alchemy::Config
13
+ #
14
+ # preprocess_image_resize [String] - Downsizing example: '1000x1000>'
15
+ #
16
+ def call
17
+ max_image_size = Alchemy::Config.get(:preprocess_image_resize)
18
+ image_file.thumb!(max_image_size) if max_image_size.present?
19
+ end
20
+
21
+ private
22
+
23
+ attr_reader :image_file
24
+ end
25
+ end
26
+ end
@@ -49,6 +49,7 @@ module Alchemy
49
49
  #
50
50
  def crop(size, crop_from = nil, crop_size = nil, upsample = false)
51
51
  raise "No size given!" if size.empty?
52
+
52
53
  render_to = sizes_from_string(size)
53
54
  if crop_from && crop_size
54
55
  top_left = point_from_string(crop_from)
@@ -79,7 +80,7 @@ module Alchemy
79
80
  height = 0 if height.nil?
80
81
  {
81
82
  width: width,
82
- height: height
83
+ height: height,
83
84
  }
84
85
  end
85
86
 
@@ -109,7 +110,7 @@ module Alchemy
109
110
  def image_size
110
111
  {
111
112
  width: image_file_width,
112
- height: image_file_height
113
+ height: image_file_height,
113
114
  }
114
115
  end
115
116
 
@@ -118,6 +119,7 @@ module Alchemy
118
119
  #
119
120
  def can_be_cropped_to(string, upsample = false)
120
121
  return true if upsample
122
+
121
123
  is_bigger_than sizes_from_string(string)
122
124
  end
123
125
 
@@ -147,7 +149,7 @@ module Alchemy
147
149
  y = 0 if y.nil?
148
150
  {
149
151
  x: x,
150
- y: y
152
+ y: y,
151
153
  }
152
154
  end
153
155
 
@@ -158,7 +160,7 @@ module Alchemy
158
160
  def get_top_left_crop_corner(dimensions)
159
161
  {
160
162
  x: (image_file_width - dimensions[:width]) / 2,
161
- y: (image_file_height - dimensions[:height]) / 2
163
+ y: (image_file_height - dimensions[:height]) / 2,
162
164
  }
163
165
  end
164
166
 
@@ -182,7 +184,7 @@ module Alchemy
182
184
  def size_when_fitting(target, dimensions = get_base_dimensions)
183
185
  zoom = [
184
186
  dimensions[:width].to_f / target[:width],
185
- dimensions[:height].to_f / target[:height]
187
+ dimensions[:height].to_f / target[:height],
186
188
  ].max
187
189
 
188
190
  if zoom == 0.0
@@ -205,7 +207,7 @@ module Alchemy
205
207
  x1: point[:x],
206
208
  y1: point[:y],
207
209
  x2: point[:x] + mask[:width],
208
- y2: point[:y] + mask[:height]
210
+ y2: point[:y] + mask[:height],
209
211
  }
210
212
  end
211
213
 
@@ -252,7 +254,7 @@ module Alchemy
252
254
  def reduce_to_image(dimensions)
253
255
  {
254
256
  width: [dimensions[:width], image_file_width].min,
255
- height: [dimensions[:height], image_file_height].min
257
+ height: [dimensions[:height], image_file_height].min,
256
258
  }
257
259
  end
258
260
  end
@@ -12,7 +12,7 @@ module Alchemy
12
12
  :format,
13
13
  :quality,
14
14
  :size,
15
- :upsample
15
+ :upsample,
16
16
  ]
17
17
 
18
18
  # Returns a path to picture for use inside a image_tag helper.
@@ -66,8 +66,8 @@ module Alchemy
66
66
  end
67
67
 
68
68
  options = {
69
- flatten: target_format != 'gif' && image_file_format == 'gif'
70
- }.merge(options)
69
+ flatten: target_format != "gif" && image_file_format == "gif",
70
+ }.with_indifferent_access.merge(options)
71
71
 
72
72
  encoding_options = []
73
73
 
@@ -77,13 +77,13 @@ module Alchemy
77
77
  end
78
78
 
79
79
  if options[:flatten]
80
- encoding_options << '-flatten'
80
+ encoding_options << "-flatten"
81
81
  end
82
82
 
83
83
  convertion_needed = target_format != image_file_format || encoding_options.present?
84
84
 
85
85
  if has_convertible_format? && convertion_needed
86
- image = image.encode(target_format, encoding_options.join(' '))
86
+ image = image.encode(target_format, encoding_options.join(" "))
87
87
  end
88
88
 
89
89
  image
@@ -23,10 +23,12 @@ module Alchemy
23
23
  # associations
24
24
  has_many :languages
25
25
 
26
- scope :published, -> { where(public: true) }
26
+ before_destroy if: -> { languages.any? } do
27
+ errors.add(:languages, :still_present)
28
+ throw(:abort)
29
+ end
27
30
 
28
- # Callbacks
29
- before_create :create_default_language!, unless: -> { languages.any? }
31
+ scope :published, -> { where(public: true) }
30
32
 
31
33
  # concerns
32
34
  include Alchemy::Site::Layout
@@ -64,7 +66,7 @@ module Alchemy
64
66
  end
65
67
 
66
68
  def default
67
- Site.first || create_default_site!
69
+ Site.first
68
70
  end
69
71
 
70
72
  def find_for_host(host)
@@ -81,38 +83,6 @@ module Alchemy
81
83
  site.aliases.split.include?(host) if site.aliases.present?
82
84
  end
83
85
  end
84
-
85
- private
86
-
87
- def create_default_site!
88
- default_site = Alchemy::Config.get(:default_site)
89
- if default_site
90
- create!(name: default_site['name'], host: default_site['host'])
91
- else
92
- raise DefaultSiteNotFoundError
93
- end
94
- end
95
- end
96
-
97
- private
98
-
99
- # If no languages are present, create a default language based
100
- # on the host app's Alchemy configuration.
101
- def create_default_language!
102
- default_language = Alchemy::Config.get(:default_language)
103
- if default_language
104
- languages.build(
105
- name: default_language['name'],
106
- language_code: default_language['code'],
107
- locale: default_language['code'],
108
- frontpage_name: default_language['frontpage_name'],
109
- page_layout: default_language['page_layout'],
110
- public: true,
111
- default: true
112
- )
113
- else
114
- raise DefaultLanguageNotFoundError
115
- end
116
86
  end
117
87
  end
118
88
  end