alchemy_cms 5.2.1 → 6.0.0.pre.b4

Sign up to get free protection for your applications and to get access to all the features.
Files changed (302) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/ci.yml +6 -14
  3. data/.gitignore +0 -1
  4. data/.hound.yml +1 -1
  5. data/.rubocop.yml +46 -4
  6. data/CHANGELOG.md +121 -8
  7. data/Gemfile +4 -2
  8. data/README.md +19 -4
  9. data/Rakefile +37 -23
  10. data/alchemy_cms.gemspec +78 -65
  11. data/app/assets/javascripts/alchemy/admin.js +0 -2
  12. data/app/assets/javascripts/alchemy/alchemy.base.js.coffee +0 -27
  13. data/app/assets/javascripts/alchemy/alchemy.confirm_dialog.js.coffee +2 -1
  14. data/app/assets/javascripts/alchemy/alchemy.dialog.js.coffee +1 -1
  15. data/app/assets/javascripts/alchemy/alchemy.dragndrop.js.coffee +0 -25
  16. data/app/assets/javascripts/alchemy/alchemy.element_editors.js.coffee +1 -1
  17. data/app/assets/javascripts/alchemy/alchemy.elements_window.js.coffee +2 -0
  18. data/app/assets/javascripts/alchemy/alchemy.fixed_elements.js +1 -1
  19. data/app/assets/javascripts/alchemy/alchemy.gui.js.coffee +3 -1
  20. data/app/assets/javascripts/alchemy/alchemy.image_overlay.coffee +1 -1
  21. data/app/assets/javascripts/alchemy/alchemy.link_dialog.js.coffee +40 -27
  22. data/app/assets/javascripts/alchemy/templates/node_folder.hbs +1 -1
  23. data/app/assets/stylesheets/alchemy/_extends.scss +15 -2
  24. data/app/assets/stylesheets/alchemy/admin.scss +1 -1
  25. data/app/assets/stylesheets/alchemy/archive.scss +20 -5
  26. data/app/assets/stylesheets/alchemy/buttons.scss +0 -4
  27. data/app/assets/stylesheets/alchemy/elements.scss +73 -61
  28. data/app/assets/stylesheets/alchemy/fonts.scss +0 -0
  29. data/app/assets/stylesheets/alchemy/images.scss +8 -0
  30. data/app/assets/stylesheets/alchemy/node-select.scss +4 -3
  31. data/app/assets/stylesheets/alchemy/page-select.scss +1 -0
  32. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce-small.svg +0 -0
  33. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce-small.ttf +0 -0
  34. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce-small.woff +0 -0
  35. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce.svg +0 -0
  36. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce.ttf +0 -0
  37. data/app/assets/stylesheets/tinymce/skins/alchemy/fonts/tinymce.woff +0 -0
  38. data/app/assets/stylesheets/tinymce/skins/alchemy/img/anchor.gif +0 -0
  39. data/app/assets/stylesheets/tinymce/skins/alchemy/img/loader.gif +0 -0
  40. data/app/assets/stylesheets/tinymce/skins/alchemy/img/object.gif +0 -0
  41. data/app/assets/stylesheets/tinymce/skins/alchemy/img/trans.gif +0 -0
  42. data/app/assets/stylesheets/tinymce/skins/alchemy/skin.min.css.scss +0 -0
  43. data/app/controllers/alchemy/admin/attachments_controller.rb +8 -4
  44. data/app/controllers/alchemy/admin/base_controller.rb +5 -7
  45. data/app/controllers/alchemy/admin/elements_controller.rb +59 -34
  46. data/app/controllers/alchemy/admin/essence_audios_controller.rb +30 -0
  47. data/app/controllers/alchemy/admin/essence_files_controller.rb +0 -14
  48. data/app/controllers/alchemy/admin/essence_pictures_controller.rb +8 -79
  49. data/app/controllers/alchemy/admin/essence_videos_controller.rb +33 -0
  50. data/app/controllers/alchemy/admin/ingredients_controller.rb +30 -0
  51. data/app/controllers/alchemy/admin/layoutpages_controller.rb +0 -1
  52. data/app/controllers/alchemy/admin/pages_controller.rb +7 -22
  53. data/app/controllers/alchemy/admin/pictures_controller.rb +56 -17
  54. data/app/controllers/alchemy/admin/resources_controller.rb +84 -10
  55. data/app/controllers/alchemy/api/elements_controller.rb +13 -4
  56. data/app/controllers/alchemy/api/pages_controller.rb +4 -3
  57. data/app/controllers/concerns/alchemy/admin/archive_overlay.rb +13 -3
  58. data/app/controllers/concerns/alchemy/admin/crop_action.rb +26 -0
  59. data/app/decorators/alchemy/element_editor.rb +26 -1
  60. data/app/decorators/alchemy/ingredient_editor.rb +158 -0
  61. data/app/helpers/alchemy/admin/elements_helper.rb +1 -0
  62. data/app/helpers/alchemy/admin/essences_helper.rb +1 -1
  63. data/app/helpers/alchemy/admin/ingredients_helper.rb +42 -0
  64. data/app/helpers/alchemy/elements_block_helper.rb +29 -6
  65. data/app/helpers/alchemy/elements_helper.rb +12 -5
  66. data/app/helpers/alchemy/pages_helper.rb +3 -11
  67. data/app/jobs/alchemy/base_job.rb +11 -0
  68. data/app/jobs/alchemy/publish_page_job.rb +11 -0
  69. data/app/models/alchemy/attachment.rb +24 -7
  70. data/app/models/alchemy/content/factory.rb +23 -27
  71. data/app/models/alchemy/content.rb +1 -6
  72. data/app/models/alchemy/element/definitions.rb +29 -27
  73. data/app/models/alchemy/element/element_contents.rb +131 -122
  74. data/app/models/alchemy/element/element_essences.rb +111 -98
  75. data/app/models/alchemy/element/element_ingredients.rb +184 -0
  76. data/app/models/alchemy/element/presenters.rb +104 -85
  77. data/app/models/alchemy/element.rb +39 -86
  78. data/app/models/alchemy/elements_repository.rb +126 -0
  79. data/app/models/alchemy/essence_audio.rb +12 -0
  80. data/app/models/alchemy/essence_headline.rb +40 -0
  81. data/app/models/alchemy/essence_picture.rb +4 -116
  82. data/app/models/alchemy/essence_richtext.rb +12 -0
  83. data/app/models/alchemy/essence_video.rb +12 -0
  84. data/app/models/alchemy/image_cropper_settings.rb +87 -0
  85. data/app/models/alchemy/ingredient.rb +183 -0
  86. data/app/models/alchemy/ingredient_validator.rb +97 -0
  87. data/app/models/alchemy/ingredients/audio.rb +29 -0
  88. data/app/models/alchemy/ingredients/boolean.rb +21 -0
  89. data/app/models/alchemy/ingredients/datetime.rb +20 -0
  90. data/app/models/alchemy/ingredients/file.rb +30 -0
  91. data/app/models/alchemy/ingredients/headline.rb +42 -0
  92. data/app/models/alchemy/ingredients/html.rb +19 -0
  93. data/app/models/alchemy/ingredients/link.rb +16 -0
  94. data/app/models/alchemy/ingredients/node.rb +23 -0
  95. data/app/models/alchemy/ingredients/page.rb +23 -0
  96. data/app/models/alchemy/ingredients/picture.rb +41 -0
  97. data/app/models/alchemy/ingredients/richtext.rb +57 -0
  98. data/app/models/alchemy/ingredients/select.rb +10 -0
  99. data/app/models/alchemy/ingredients/text.rb +17 -0
  100. data/app/models/alchemy/ingredients/video.rb +33 -0
  101. data/app/models/alchemy/language.rb +0 -11
  102. data/app/models/alchemy/page/fixed_attributes.rb +53 -51
  103. data/app/models/alchemy/page/page_elements.rb +186 -205
  104. data/app/models/alchemy/page/page_naming.rb +66 -64
  105. data/app/models/alchemy/page/page_natures.rb +131 -143
  106. data/app/models/alchemy/page/page_scopes.rb +117 -102
  107. data/app/models/alchemy/page/publisher.rb +50 -0
  108. data/app/models/alchemy/page/url_path.rb +1 -1
  109. data/app/models/alchemy/page.rb +79 -36
  110. data/app/models/alchemy/page_version.rb +58 -0
  111. data/app/models/alchemy/picture/calculations.rb +2 -8
  112. data/app/models/alchemy/picture/preprocessor.rb +2 -0
  113. data/app/models/alchemy/picture/transformations.rb +24 -96
  114. data/app/models/alchemy/picture.rb +17 -39
  115. data/app/models/concerns/alchemy/picture_thumbnails.rb +181 -0
  116. data/app/models/concerns/alchemy/touch_elements.rb +2 -2
  117. data/app/presenters/alchemy/picture_view.rb +88 -0
  118. data/app/serializers/alchemy/element_serializer.rb +5 -0
  119. data/app/serializers/alchemy/page_tree_serializer.rb +3 -2
  120. data/app/services/alchemy/delete_elements.rb +44 -0
  121. data/app/services/alchemy/duplicate_element.rb +56 -0
  122. data/app/views/alchemy/admin/attachments/_archive_overlay.html.erb +2 -3
  123. data/app/views/alchemy/admin/attachments/_file_to_assign.html.erb +3 -3
  124. data/app/views/alchemy/admin/attachments/assign.js.erb +11 -0
  125. data/app/views/alchemy/admin/attachments/index.html.erb +2 -3
  126. data/app/views/alchemy/admin/crop.html.erb +36 -0
  127. data/app/views/alchemy/admin/elements/_element.html.erb +14 -10
  128. data/app/views/alchemy/admin/elements/{_element_footer.html.erb → _footer.html.erb} +0 -0
  129. data/app/views/alchemy/admin/elements/{_new_element_form.html.erb → _form.html.erb} +1 -1
  130. data/app/views/alchemy/admin/elements/{_element_header.html.erb → _header.html.erb} +1 -1
  131. data/app/views/alchemy/admin/elements/{_element_toolbar.html.erb → _toolbar.html.erb} +5 -6
  132. data/app/views/alchemy/admin/elements/create.js.erb +1 -1
  133. data/app/views/alchemy/admin/elements/{trash.js.erb → destroy.js.erb} +2 -6
  134. data/app/views/alchemy/admin/elements/fold.js.erb +2 -2
  135. data/app/views/alchemy/admin/elements/new.html.erb +3 -3
  136. data/app/views/alchemy/admin/elements/order.js.erb +0 -17
  137. data/app/views/alchemy/admin/elements/update.js.erb +3 -2
  138. data/app/views/alchemy/admin/essence_audios/edit.html.erb +7 -0
  139. data/app/views/alchemy/admin/essence_pictures/update.js.erb +0 -1
  140. data/app/views/alchemy/admin/essence_videos/edit.html.erb +11 -0
  141. data/app/views/alchemy/admin/ingredients/_audio_fields.html.erb +4 -0
  142. data/app/views/alchemy/admin/ingredients/_file_fields.html.erb +18 -0
  143. data/app/views/alchemy/admin/ingredients/_picture_fields.html.erb +25 -0
  144. data/app/views/alchemy/admin/ingredients/_video_fields.html.erb +8 -0
  145. data/app/views/alchemy/admin/ingredients/edit.html.erb +4 -0
  146. data/app/views/alchemy/admin/layoutpages/edit.html.erb +0 -5
  147. data/app/views/alchemy/admin/nodes/_node.html.erb +2 -2
  148. data/app/views/alchemy/admin/pages/_anchor_link.html.erb +1 -1
  149. data/app/views/alchemy/admin/pages/_external_link.html.erb +1 -1
  150. data/app/views/alchemy/admin/pages/_file_link.html.erb +1 -1
  151. data/app/views/alchemy/admin/pages/_form.html.erb +0 -6
  152. data/app/views/alchemy/admin/pages/_internal_link.html.erb +1 -1
  153. data/app/views/alchemy/admin/pages/_tinymce_custom_config.html.erb +5 -2
  154. data/app/views/alchemy/admin/pages/_toolbar.html.erb +1 -1
  155. data/app/views/alchemy/admin/pages/edit.html.erb +37 -25
  156. data/app/views/alchemy/admin/pages/index.html.erb +2 -9
  157. data/app/views/alchemy/admin/partials/_remote_search_form.html.erb +2 -4
  158. data/app/views/alchemy/admin/partials/_routes.html.erb +7 -11
  159. data/app/views/alchemy/admin/partials/_search_form.html.erb +9 -0
  160. data/app/views/alchemy/admin/pictures/_archive.html.erb +1 -1
  161. data/app/views/alchemy/admin/pictures/_archive_overlay.html.erb +1 -1
  162. data/app/views/alchemy/admin/pictures/_filter_and_size_bar.html.erb +5 -7
  163. data/app/views/alchemy/admin/pictures/_infos.html.erb +0 -1
  164. data/app/views/alchemy/admin/pictures/_picture_to_assign.html.erb +4 -4
  165. data/app/views/alchemy/admin/pictures/assign.js.erb +10 -0
  166. data/app/views/alchemy/admin/pictures/index.html.erb +8 -3
  167. data/app/views/alchemy/admin/resources/_filter.html.erb +12 -0
  168. data/app/views/alchemy/admin/resources/_filter_bar.html.erb +14 -17
  169. data/app/views/alchemy/admin/resources/_form.html.erb +3 -0
  170. data/app/views/alchemy/admin/resources/_table_header.html.erb +15 -0
  171. data/app/views/alchemy/admin/resources/index.html.erb +3 -11
  172. data/app/views/alchemy/essences/_essence_audio_editor.html.erb +4 -0
  173. data/app/views/alchemy/essences/_essence_audio_view.html.erb +15 -0
  174. data/app/views/alchemy/essences/_essence_file_editor.html.erb +15 -6
  175. data/app/views/alchemy/essences/_essence_headline_editor.html.erb +36 -0
  176. data/app/views/alchemy/essences/_essence_headline_view.html.erb +10 -0
  177. data/app/views/alchemy/essences/_essence_link_editor.html.erb +8 -4
  178. data/app/views/alchemy/essences/_essence_picture_editor.html.erb +27 -12
  179. data/app/views/alchemy/essences/_essence_picture_view.html.erb +3 -3
  180. data/app/views/alchemy/essences/_essence_text_editor.html.erb +12 -4
  181. data/app/views/alchemy/essences/_essence_video_editor.html.erb +4 -0
  182. data/app/views/alchemy/essences/_essence_video_view.html.erb +18 -0
  183. data/app/views/alchemy/essences/shared/_essence_picture_tools.html.erb +21 -16
  184. data/app/views/alchemy/essences/shared/_linkable_essence_tools.html.erb +2 -2
  185. data/app/views/alchemy/ingredients/_audio_editor.html.erb +5 -0
  186. data/app/views/alchemy/ingredients/_audio_view.html.erb +14 -0
  187. data/app/views/alchemy/ingredients/_boolean_editor.html.erb +11 -0
  188. data/app/views/alchemy/ingredients/_boolean_view.html.erb +1 -0
  189. data/app/views/alchemy/ingredients/_datetime_editor.html.erb +17 -0
  190. data/app/views/alchemy/ingredients/_datetime_view.html.erb +9 -0
  191. data/app/views/alchemy/ingredients/_file_editor.html.erb +52 -0
  192. data/app/views/alchemy/ingredients/_file_view.html.erb +17 -0
  193. data/app/views/alchemy/ingredients/_headline_editor.html.erb +30 -0
  194. data/app/views/alchemy/ingredients/_headline_view.html.erb +9 -0
  195. data/app/views/alchemy/ingredients/_html_editor.html.erb +8 -0
  196. data/app/views/alchemy/ingredients/_html_view.html.erb +1 -0
  197. data/app/views/alchemy/ingredients/_link_editor.html.erb +24 -0
  198. data/app/views/alchemy/ingredients/_link_view.html.erb +9 -0
  199. data/app/views/alchemy/ingredients/_node_editor.html.erb +26 -0
  200. data/app/views/alchemy/ingredients/_node_view.html.erb +1 -0
  201. data/app/views/alchemy/ingredients/_page_editor.html.erb +25 -0
  202. data/app/views/alchemy/ingredients/_page_view.html.erb +4 -0
  203. data/app/views/alchemy/ingredients/_picture_editor.html.erb +60 -0
  204. data/app/views/alchemy/ingredients/_picture_view.html.erb +5 -0
  205. data/app/views/alchemy/ingredients/_richtext_editor.html.erb +12 -0
  206. data/app/views/alchemy/ingredients/_richtext_view.html.erb +3 -0
  207. data/app/views/alchemy/ingredients/_select_editor.html.erb +30 -0
  208. data/app/views/alchemy/ingredients/_select_view.html.erb +1 -0
  209. data/app/views/alchemy/ingredients/_text_editor.html.erb +20 -0
  210. data/app/views/alchemy/ingredients/_text_view.html.erb +16 -0
  211. data/app/views/alchemy/ingredients/_video_editor.html.erb +5 -0
  212. data/app/views/alchemy/ingredients/_video_view.html.erb +17 -0
  213. data/app/views/alchemy/ingredients/shared/_link_tools.html.erb +20 -0
  214. data/app/views/alchemy/ingredients/shared/_picture_tools.html.erb +57 -0
  215. data/config/brakeman.ignore +66 -159
  216. data/config/initializers/dragonfly.rb +10 -0
  217. data/config/locales/alchemy.en.yml +108 -64
  218. data/config/routes.rb +17 -22
  219. data/db/migrate/20201207131309_create_page_versions.rb +19 -0
  220. data/db/migrate/20201207135820_add_page_version_id_to_alchemy_elements.rb +76 -0
  221. data/db/migrate/20210205143548_rename_public_on_and_public_until_on_alchemy_pages.rb +10 -0
  222. data/db/migrate/20210326105046_add_sanitized_body_to_alchemy_essence_richtexts.rb +7 -0
  223. data/db/migrate/20210406093436_add_alchemy_essence_headlines.rb +12 -0
  224. data/db/migrate/20210506135919_create_essence_audios.rb +19 -0
  225. data/db/migrate/20210506140258_create_essence_videos.rb +23 -0
  226. data/db/migrate/20210508091432_create_alchemy_ingredients.rb +22 -0
  227. data/lib/alchemy/admin/preview_url.rb +2 -0
  228. data/lib/alchemy/deprecation.rb +1 -1
  229. data/lib/alchemy/dragonfly/processors/auto_orient.rb +18 -0
  230. data/lib/alchemy/dragonfly/processors/crop_resize.rb +35 -0
  231. data/lib/alchemy/elements_finder.rb +14 -60
  232. data/lib/alchemy/essence.rb +1 -2
  233. data/lib/alchemy/forms/builder.rb +21 -1
  234. data/lib/alchemy/hints.rb +8 -4
  235. data/lib/alchemy/page_layout.rb +0 -13
  236. data/lib/alchemy/permissions.rb +30 -29
  237. data/lib/alchemy/resource.rb +13 -3
  238. data/lib/alchemy/resource_filter.rb +40 -0
  239. data/lib/alchemy/resources_helper.rb +1 -16
  240. data/lib/alchemy/tasks/tidy.rb +29 -0
  241. data/lib/alchemy/test_support/essence_shared_examples.rb +0 -1
  242. data/lib/alchemy/test_support/factories/element_factory.rb +8 -8
  243. data/lib/alchemy/test_support/factories/essence_audio_factory.rb +7 -0
  244. data/lib/alchemy/test_support/factories/essence_video_factory.rb +7 -0
  245. data/lib/alchemy/test_support/factories/ingredient_factory.rb +25 -0
  246. data/lib/alchemy/test_support/factories/page_factory.rb +20 -1
  247. data/lib/alchemy/test_support/factories/page_version_factory.rb +23 -0
  248. data/lib/alchemy/test_support/having_crop_action_examples.rb +170 -0
  249. data/lib/alchemy/test_support/having_picture_thumbnails_examples.rb +646 -0
  250. data/lib/alchemy/test_support/shared_ingredient_editor_examples.rb +21 -0
  251. data/lib/alchemy/test_support/shared_ingredient_examples.rb +75 -0
  252. data/lib/alchemy/test_support.rb +2 -11
  253. data/lib/alchemy/tinymce.rb +21 -0
  254. data/lib/alchemy/upgrader/six_point_zero.rb +21 -0
  255. data/lib/alchemy/upgrader/tasks/add_page_versions.rb +33 -0
  256. data/lib/alchemy/upgrader/tasks/ingredients_migrator.rb +62 -0
  257. data/lib/alchemy/version.rb +1 -1
  258. data/lib/alchemy_cms.rb +1 -0
  259. data/lib/generators/alchemy/elements/elements_generator.rb +1 -0
  260. data/lib/generators/alchemy/elements/templates/view.html.erb +9 -0
  261. data/lib/generators/alchemy/elements/templates/view.html.haml +9 -0
  262. data/lib/generators/alchemy/elements/templates/view.html.slim +9 -0
  263. data/lib/generators/alchemy/ingredient/ingredient_generator.rb +38 -0
  264. data/lib/generators/alchemy/ingredient/templates/editor.html.erb +14 -0
  265. data/lib/generators/alchemy/ingredient/templates/model.rb.tt +13 -0
  266. data/lib/generators/alchemy/ingredient/templates/view.html.erb +1 -0
  267. data/lib/generators/alchemy/install/templates/dragonfly.rb.tt +1 -1
  268. data/lib/generators/alchemy/menus/templates/node.html.erb +1 -1
  269. data/lib/generators/alchemy/menus/templates/node.html.haml +1 -1
  270. data/lib/generators/alchemy/menus/templates/node.html.slim +1 -1
  271. data/lib/generators/alchemy/menus/templates/wrapper.html.erb +1 -1
  272. data/lib/generators/alchemy/menus/templates/wrapper.html.haml +1 -1
  273. data/lib/generators/alchemy/menus/templates/wrapper.html.slim +1 -1
  274. data/lib/tasks/alchemy/thumbnails.rake +2 -0
  275. data/lib/tasks/alchemy/tidy.rake +12 -0
  276. data/lib/tasks/alchemy/upgrade.rake +26 -0
  277. data/package/admin.js +11 -1
  278. data/package/src/__tests__/i18n.spec.js +23 -0
  279. data/package/src/file_editors.js +28 -0
  280. data/package/src/i18n.js +1 -3
  281. data/package/src/image_cropper.js +103 -0
  282. data/package/src/image_loader.js +58 -0
  283. data/package/src/node_tree.js +5 -5
  284. data/package/src/picture_editors.js +169 -0
  285. data/package/src/utils/__tests__/ajax.spec.js +20 -12
  286. data/package/src/utils/ajax.js +8 -3
  287. data/package.json +3 -2
  288. data/vendor/assets/javascripts/jquery_plugins/jquery.Jcrop.min.js +3 -18
  289. data/vendor/assets/stylesheets/jquery.Jcrop.min.scss +2 -28
  290. metadata +289 -53
  291. data/app/assets/javascripts/alchemy/alchemy.image_cropper.js.coffee +0 -44
  292. data/app/assets/javascripts/alchemy/alchemy.trash_window.js.coffee +0 -30
  293. data/app/assets/stylesheets/alchemy/trash.scss +0 -8
  294. data/app/controllers/alchemy/admin/trash_controller.rb +0 -44
  295. data/app/views/alchemy/admin/attachments/_filter_bar.html.erb +0 -29
  296. data/app/views/alchemy/admin/essence_files/assign.js.erb +0 -3
  297. data/app/views/alchemy/admin/essence_pictures/assign.js.erb +0 -4
  298. data/app/views/alchemy/admin/essence_pictures/crop.html.erb +0 -48
  299. data/app/views/alchemy/admin/pictures/_filter_bar.html.erb +0 -30
  300. data/app/views/alchemy/admin/trash/clear.js.erb +0 -4
  301. data/app/views/alchemy/admin/trash/index.html.erb +0 -31
  302. data/lib/alchemy/test_support/factories.rb +0 -20
@@ -17,6 +17,7 @@ module Alchemy
17
17
  acts_as_essence preview_text_column: "stripped_body"
18
18
 
19
19
  before_save :strip_content
20
+ before_save :sanitize_content
20
21
 
21
22
  def has_tinymce?
22
23
  true
@@ -27,5 +28,16 @@ module Alchemy
27
28
  def strip_content
28
29
  self.stripped_body = Rails::Html::FullSanitizer.new.sanitize(body)
29
30
  end
31
+
32
+ def sanitize_content
33
+ self.sanitized_body = Rails::Html::SafeListSanitizer.new.sanitize(
34
+ body,
35
+ content_sanitizer_settings
36
+ )
37
+ end
38
+
39
+ def content_sanitizer_settings
40
+ content&.settings&.fetch(:sanitizer, {}) || {}
41
+ end
30
42
  end
31
43
  end
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class EssenceVideo < ActiveRecord::Base
5
+ acts_as_essence(
6
+ ingredient_column: :attachment,
7
+ preview_text_method: :name,
8
+ )
9
+
10
+ belongs_to :attachment, optional: true
11
+ end
12
+ end
@@ -0,0 +1,87 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ # Settings for the graphical JS image cropper
5
+ class ImageCropperSettings
6
+ attr_reader :render_size, :default_crop_from, :default_crop_size, :fixed_ratio, :image_width, :image_height
7
+
8
+ def initialize(render_size:, default_crop_from:, default_crop_size:, fixed_ratio:, image_width:, image_height:)
9
+ @render_size = render_size || [0, 0]
10
+ @fixed_ratio = fixed_ratio
11
+ @image_width = image_width.to_i
12
+ @image_height = image_height.to_i
13
+ @default_crop_from = default_crop_from || [0, 0]
14
+ @default_crop_size = default_crop_size || [@image_width, @image_height]
15
+ end
16
+
17
+ def to_h
18
+ return {} if image_width.zero? || image_height.zero?
19
+
20
+ {
21
+ min_size: large_enough? ? min_size : false,
22
+ ratio: ratio,
23
+ default_box: default_box,
24
+ image_size: [image_width, image_height],
25
+ }.freeze
26
+ end
27
+
28
+ def [](key)
29
+ to_h[key]
30
+ end
31
+
32
+ private
33
+
34
+ def ratio
35
+ return false if fixed_ratio == false
36
+ return ratio_from_size if fixed_ratio.nil?
37
+
38
+ Float(fixed_ratio)
39
+ end
40
+
41
+ # Only returns an array of width and height if image is large enough
42
+ # or false to disable min size option of the image cropper
43
+ def large_enough?
44
+ return true if render_size.any?(&:zero?)
45
+
46
+ image_width >= render_size[0] && image_height >= render_size[1]
47
+ end
48
+
49
+ # Infers the aspect ratio from size or fixed_ratio. If you don't want a fixed
50
+ # aspect ratio, don't specify a size or only width or height.
51
+ #
52
+ def ratio_from_size
53
+ if render_size.none?(&:zero?)
54
+ render_size[0].to_f / render_size[1]
55
+ elsif [image_width, image_height].none?(&:zero?)
56
+ image_width.to_f / image_height
57
+ else
58
+ false
59
+ end
60
+ end
61
+
62
+ # Infers the minimum width or height
63
+ # if the aspect ratio and one dimension is specified.
64
+ #
65
+ def min_size
66
+ return render_size unless ratio
67
+
68
+ if render_size[1].zero?
69
+ [render_size[0], (render_size[0] / ratio).to_i]
70
+ else
71
+ [(render_size[1] * ratio).to_i, render_size[1]]
72
+ end
73
+ end
74
+
75
+ # Given a point and a mask, this function returns the area on the
76
+ # underlying canvas as a Hash of two points
77
+ #
78
+ def default_box
79
+ [
80
+ default_crop_from[0],
81
+ default_crop_from[1],
82
+ default_crop_from[0] + default_crop_size[0],
83
+ default_crop_from[1] + default_crop_size[1],
84
+ ]
85
+ end
86
+ end
87
+ end
@@ -0,0 +1,183 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class Ingredient < BaseRecord
5
+ class DefinitionError < StandardError; end
6
+
7
+ include Hints
8
+
9
+ self.table_name = "alchemy_ingredients"
10
+
11
+ belongs_to :element, touch: true, class_name: "Alchemy::Element", inverse_of: :ingredients
12
+ belongs_to :related_object, polymorphic: true, optional: true
13
+
14
+ before_validation(on: :create) { self.value ||= default_value }
15
+
16
+ validates :type, presence: true
17
+ validates :role, presence: true
18
+
19
+ validates_with Alchemy::IngredientValidator, on: :update, if: :has_validations?
20
+
21
+ scope :audios, -> { where(type: "Alchemy::Ingredients::Audio") }
22
+ scope :booleans, -> { where(type: "Alchemy::Ingredients::Boolean") }
23
+ scope :datetimes, -> { where(type: "Alchemy::Ingredients::Datetime") }
24
+ scope :files, -> { where(type: "Alchemy::Ingredients::File") }
25
+ scope :headlines, -> { where(type: "Alchemy::Ingredients::Headline") }
26
+ scope :htmls, -> { where(type: "Alchemy::Ingredients::Html") }
27
+ scope :links, -> { where(type: "Alchemy::Ingredients::Link") }
28
+ scope :nodes, -> { where(type: "Alchemy::Ingredients::Node") }
29
+ scope :pages, -> { where(type: "Alchemy::Ingredients::Page") }
30
+ scope :pictures, -> { where(type: "Alchemy::Ingredients::Picture") }
31
+ scope :richtexts, -> { where(type: "Alchemy::Ingredients::Richtext") }
32
+ scope :selects, -> { where(type: "Alchemy::Ingredients::Select") }
33
+ scope :texts, -> { where(type: "Alchemy::Ingredients::Text") }
34
+ scope :videos, -> { where(type: "Alchemy::Ingredients::Video") }
35
+
36
+ class << self
37
+ # Defines getter and setter method aliases for related object
38
+ #
39
+ # @param [String|Symbol] The name of the alias
40
+ # @param [String] The class name of the related object
41
+ def related_object_alias(name, class_name:)
42
+ alias_method name, :related_object
43
+ alias_method "#{name}=", :related_object=
44
+
45
+ # Somehow Rails STI does not allow us to use `alias_method` for the related_object_id
46
+ define_method "#{name}_id" do
47
+ related_object_id
48
+ end
49
+
50
+ define_method "#{name}_id=" do |id|
51
+ self.related_object_id = id
52
+ self.related_object_type = class_name
53
+ end
54
+ end
55
+
56
+ # Modulize ingredient type
57
+ #
58
+ # Makes sure the passed ingredient type is in the +Alchemy::Ingredients+
59
+ # module namespace.
60
+ #
61
+ # If you add custom ingredient class,
62
+ # put them in the +Alchemy::Ingredients+ module namespace
63
+ # @param [String] Ingredient class name
64
+ # @return [String]
65
+ def normalize_type(ingredient_type)
66
+ "Alchemy::Ingredients::#{ingredient_type.to_s.classify.demodulize}"
67
+ end
68
+
69
+ def translated_label_for(role, element_name = nil)
70
+ Alchemy.t(
71
+ role,
72
+ scope: "ingredient_roles.#{element_name}",
73
+ default: Alchemy.t("ingredient_roles.#{role}", default: role.humanize),
74
+ )
75
+ end
76
+ end
77
+
78
+ # Compatibility method for access from element
79
+ def essence
80
+ self
81
+ end
82
+
83
+ # The value or the related object if present
84
+ def value
85
+ related_object || self[:value]
86
+ end
87
+
88
+ # Settings for this ingredient from the +elements.yml+ definition.
89
+ def settings
90
+ definition[:settings] || {}
91
+ end
92
+
93
+ # Fetches value from settings
94
+ #
95
+ # @param key [Symbol] - The hash key you want to fetch the value from
96
+ # @param options [Hash] - An optional Hash that can override the settings.
97
+ # Normally passed as options hash into the content
98
+ # editor view.
99
+ def settings_value(key, options = {})
100
+ settings.merge(options || {})[key.to_sym]
101
+ end
102
+
103
+ # Definition hash for this ingredient from +elements.yml+ file.
104
+ #
105
+ def definition
106
+ return {} unless element
107
+
108
+ element.ingredient_definition_for(role) || {}
109
+ end
110
+
111
+ # The first 30 characters of the value
112
+ #
113
+ # Used by the Element#preview_text method.
114
+ #
115
+ # @param [Integer] max_length (30)
116
+ #
117
+ def preview_text(maxlength = 30)
118
+ value.to_s[0..maxlength - 1]
119
+ end
120
+
121
+ # Cross DB adapter data accessor that works
122
+ def data
123
+ @_data ||= (self[:data] || {}).with_indifferent_access
124
+ end
125
+
126
+ # The path to the view partial of the ingredient
127
+ # @return [String]
128
+ def to_partial_path
129
+ "alchemy/ingredients/#{partial_name}_view"
130
+ end
131
+
132
+ # The demodulized underscored class name of the ingredient
133
+ # @return [String]
134
+ def partial_name
135
+ self.class.name.demodulize.underscore
136
+ end
137
+
138
+ # @return [Boolean]
139
+ def has_validations?
140
+ !!definition[:validate]
141
+ end
142
+
143
+ # @return [Boolean]
144
+ def has_hint?
145
+ !!definition[:hint]
146
+ end
147
+
148
+ # @return [Boolean]
149
+ def deprecated?
150
+ !!definition[:deprecated]
151
+ end
152
+
153
+ # @return [Boolean]
154
+ def has_tinymce?
155
+ false
156
+ end
157
+
158
+ # @return [Boolean]
159
+ def preview_ingredient?
160
+ !!definition[:as_element_title]
161
+ end
162
+
163
+ private
164
+
165
+ def hint_translation_attribute
166
+ role
167
+ end
168
+
169
+ # Returns the default value from ingredient definition
170
+ #
171
+ # If the value is a symbol it gets passed through i18n
172
+ # inside the +alchemy.default_ingredient_texts+ scope
173
+ def default_value
174
+ default = definition[:default]
175
+ case default
176
+ when Symbol
177
+ Alchemy.t(default, scope: :default_ingredient_texts)
178
+ else
179
+ default
180
+ end
181
+ end
182
+ end
183
+ end
@@ -0,0 +1,97 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ # Ingredient Validations:
5
+ #
6
+ # Ingredient validations can be set inside the +config/elements.yml+ file.
7
+ #
8
+ # Supported validations are:
9
+ #
10
+ # * presence
11
+ # * uniqueness
12
+ # * format
13
+ #
14
+ # *) format needs to come with a regex or a predefined matcher string as its value.
15
+ #
16
+ # There are already predefined format matchers listed in the +config/alchemy/config.yml+ file.
17
+ # It is also possible to add own format matchers there.
18
+ #
19
+ # Example of format matchers in +config/alchemy/config.yml+:
20
+ #
21
+ # format_matchers:
22
+ # email: !ruby/regexp '/\A[^@\s]+@([^@\s]+\.)+[^@\s]+\z/'
23
+ # url: !ruby/regexp '/\A[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,5}(:[0-9]{1,5})?(\/.*)?\z/ix'
24
+ # ssl: !ruby/regexp '/https:\/\/[\S]+/'
25
+ #
26
+ # Example of an element definition with ingredient validations:
27
+ #
28
+ # - name: person
29
+ # ingredients:
30
+ # - role: name
31
+ # type: Text
32
+ # validate: [presence]
33
+ # - role: email
34
+ # type: Text
35
+ # validate: [format: 'email']
36
+ # - role: homepage
37
+ # type: Text
38
+ # validate: [format: !ruby/regexp '^[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,5}(:[0-9]{1,5})?(\/.*)?$']
39
+ #
40
+ # Example of an element definition with chained validations.
41
+ #
42
+ # - name: person
43
+ # ingredients:
44
+ # - role: name
45
+ # type: Text
46
+ # validate: [presence, uniqueness, format: 'name']
47
+ #
48
+ class IngredientValidator < ActiveModel::Validator
49
+ def validate(ingredient)
50
+ @ingredient = ingredient
51
+ validations.each do |validation|
52
+ if validation.respond_to?(:keys)
53
+ validation.map do |key, value|
54
+ send("validate_#{key}", value)
55
+ end
56
+ else
57
+ send("validate_#{validation}")
58
+ end
59
+ end
60
+ end
61
+
62
+ private
63
+
64
+ attr_reader :ingredient
65
+
66
+ def validations
67
+ ingredient.definition.fetch(:validate, [])
68
+ end
69
+
70
+ def validate_presence(*)
71
+ if ingredient.value.blank?
72
+ ingredient.errors.add(:value, :blank)
73
+ end
74
+ end
75
+
76
+ def validate_uniqueness(*)
77
+ if duplicates.any?
78
+ ingredient.errors.add(:value, :taken)
79
+ end
80
+ end
81
+
82
+ def validate_format(format)
83
+ matcher = Alchemy::Config.get("format_matchers")[format] || format
84
+ if !ingredient.value.to_s.match?(Regexp.new(matcher))
85
+ ingredient.errors.add(:value, :invalid)
86
+ end
87
+ end
88
+
89
+ def duplicates
90
+ ingredient.class
91
+ .joins(:element).merge(Alchemy::Element.available)
92
+ .where(Alchemy::Element.table_name => { name: ingredient.element.name })
93
+ .where(value: ingredient.value)
94
+ .where.not(id: ingredient.id)
95
+ end
96
+ end
97
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ module Ingredients
5
+ # A audio attachment
6
+ #
7
+ class Audio < Alchemy::Ingredient
8
+ store_accessor :data,
9
+ :autoplay,
10
+ :controls,
11
+ :muted,
12
+ :loop
13
+
14
+ related_object_alias :attachment, class_name: "Alchemy::Attachment"
15
+
16
+ delegate :name, to: :attachment, allow_nil: true
17
+
18
+ # The first 30 characters of the attachments name
19
+ #
20
+ # Used by the Element#preview_text method.
21
+ #
22
+ # @param [Integer] max_length (30)
23
+ #
24
+ def preview_text(max_length = 30)
25
+ name.to_s[0..max_length - 1]
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ module Ingredients
5
+ # A boolean value
6
+ #
7
+ class Boolean < Alchemy::Ingredient
8
+ def value
9
+ ActiveRecord::Type::Boolean.new.cast(self[:value])
10
+ end
11
+
12
+ # The localized value
13
+ #
14
+ # Used by the Element#preview_text method.
15
+ #
16
+ def preview_text(_max_length = nil)
17
+ Alchemy.t(value, scope: "ingredient_values.boolean")
18
+ end
19
+ end
20
+ end
21
+ end