alchemy_cms 8.0.12 → 8.1.0

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 (286) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +14 -10
  3. data/app/assets/builds/alchemy/admin.css +1 -1
  4. data/app/assets/builds/alchemy/dark-theme.css +1 -1
  5. data/app/assets/builds/alchemy/light-theme.css +1 -1
  6. data/app/assets/builds/alchemy/preview.min.js +1 -1
  7. data/app/assets/builds/alchemy/theme.css +1 -1
  8. data/app/{views/alchemy/admin/elements/_element.html.erb → components/alchemy/admin/element_editor.html.erb} +34 -29
  9. data/app/components/alchemy/admin/element_editor.rb +115 -0
  10. data/app/components/alchemy/admin/element_select.rb +12 -9
  11. data/app/components/alchemy/admin/ingredient_editor.rb +54 -0
  12. data/app/components/alchemy/admin/list_filter.rb +16 -5
  13. data/app/components/alchemy/admin/page_node.html.erb +214 -0
  14. data/app/components/alchemy/admin/page_node.rb +70 -0
  15. data/app/components/alchemy/admin/picture_thumbnail.rb +36 -0
  16. data/app/components/alchemy/admin/publish_page_button.html.erb +15 -0
  17. data/app/components/alchemy/admin/publish_page_button.rb +54 -0
  18. data/app/{helpers/alchemy/admin/tags_helper.rb → components/alchemy/admin/tags_list.rb} +19 -11
  19. data/app/components/alchemy/admin/toolbar_button.rb +17 -13
  20. data/app/components/alchemy/ingredients/audio_editor.rb +8 -0
  21. data/app/components/alchemy/ingredients/base_editor.rb +222 -0
  22. data/app/components/alchemy/ingredients/boolean_editor.rb +21 -0
  23. data/app/components/alchemy/ingredients/color_editor.rb +80 -0
  24. data/app/components/alchemy/ingredients/color_view.rb +13 -0
  25. data/app/components/alchemy/ingredients/datetime_editor.rb +28 -0
  26. data/app/components/alchemy/ingredients/file_editor.rb +69 -0
  27. data/app/components/alchemy/ingredients/headline_editor.rb +88 -0
  28. data/app/components/alchemy/ingredients/html_editor.rb +11 -0
  29. data/app/components/alchemy/ingredients/link_editor.rb +29 -0
  30. data/app/components/alchemy/ingredients/node_editor.rb +23 -0
  31. data/app/components/alchemy/ingredients/number_editor.rb +28 -0
  32. data/app/components/alchemy/ingredients/page_editor.rb +19 -0
  33. data/app/components/alchemy/ingredients/picture_editor.rb +81 -0
  34. data/app/components/alchemy/ingredients/richtext_editor.rb +31 -0
  35. data/app/components/alchemy/ingredients/select_editor.rb +37 -0
  36. data/app/components/alchemy/ingredients/select_view.rb +7 -0
  37. data/app/components/alchemy/ingredients/text_editor.rb +41 -0
  38. data/app/components/alchemy/ingredients/video_editor.rb +8 -0
  39. data/app/controllers/alchemy/admin/attachments_controller.rb +8 -15
  40. data/app/controllers/alchemy/admin/base_controller.rb +7 -18
  41. data/app/controllers/alchemy/admin/clipboard_controller.rb +15 -11
  42. data/app/controllers/alchemy/admin/dashboard_controller.rb +2 -2
  43. data/app/controllers/alchemy/admin/elements_controller.rb +34 -32
  44. data/app/controllers/alchemy/admin/ingredients_controller.rb +1 -0
  45. data/app/controllers/alchemy/admin/languages_controller.rb +0 -3
  46. data/app/controllers/alchemy/admin/layoutpages_controller.rb +2 -1
  47. data/app/controllers/alchemy/admin/legacy_page_urls_controller.rb +1 -1
  48. data/app/controllers/alchemy/admin/nodes_controller.rb +24 -1
  49. data/app/controllers/alchemy/admin/pages_controller.rb +36 -42
  50. data/app/controllers/alchemy/admin/pictures_controller.rb +4 -28
  51. data/app/controllers/alchemy/admin/resources_controller.rb +1 -1
  52. data/app/controllers/alchemy/api/ingredients_controller.rb +1 -1
  53. data/app/controllers/alchemy/api/pages_controller.rb +5 -3
  54. data/app/controllers/alchemy/base_controller.rb +6 -6
  55. data/app/controllers/alchemy/pages_controller.rb +12 -6
  56. data/app/controllers/concerns/alchemy/admin/archive_overlay.rb +0 -1
  57. data/app/controllers/concerns/alchemy/admin/clipboard.rb +57 -0
  58. data/app/controllers/concerns/alchemy/admin/uploader_responses.rb +2 -2
  59. data/app/controllers/concerns/alchemy/site_redirects.rb +1 -1
  60. data/app/decorators/alchemy/ingredient_editor.rb +37 -4
  61. data/app/helpers/alchemy/admin/base_helper.rb +10 -6
  62. data/app/helpers/alchemy/admin/ingredients_helper.rb +6 -3
  63. data/app/helpers/alchemy/base_helper.rb +1 -1
  64. data/app/helpers/alchemy/pages_helper.rb +1 -1
  65. data/app/javascript/alchemy_admin/components/action.js +5 -1
  66. data/app/javascript/alchemy_admin/components/color_select.js +73 -0
  67. data/app/javascript/alchemy_admin/components/element_editor/delete_element_button.js +11 -3
  68. data/app/javascript/alchemy_admin/components/element_editor/publish_element_button.js +7 -2
  69. data/app/javascript/alchemy_admin/components/element_editor.js +11 -12
  70. data/app/javascript/alchemy_admin/components/element_select.js +39 -17
  71. data/app/javascript/alchemy_admin/components/elements_window.js +0 -2
  72. data/app/javascript/alchemy_admin/components/file_editor.js +26 -0
  73. data/app/javascript/alchemy_admin/components/index.js +9 -0
  74. data/app/javascript/alchemy_admin/components/list_filter.js +57 -8
  75. data/app/javascript/alchemy_admin/components/message.js +9 -3
  76. data/app/javascript/alchemy_admin/components/page_node.js +119 -0
  77. data/app/javascript/alchemy_admin/{page_publication_fields.js → components/page_publication_fields.js} +9 -8
  78. data/app/javascript/alchemy_admin/{picture_editors.js → components/picture_editor.js} +30 -45
  79. data/app/javascript/alchemy_admin/components/picture_thumbnail.js +107 -0
  80. data/app/javascript/alchemy_admin/components/publish_page_button.js +41 -0
  81. data/app/javascript/alchemy_admin/components/select.js +3 -1
  82. data/app/javascript/alchemy_admin/components/sitemap.js +210 -0
  83. data/app/javascript/alchemy_admin/{sortable_elements.js → components/sortable_elements.js} +22 -25
  84. data/app/javascript/alchemy_admin/components/tinymce.js +10 -5
  85. data/app/javascript/alchemy_admin/components/uploader.js +30 -0
  86. data/app/javascript/alchemy_admin/image_overlay.js +0 -2
  87. data/app/javascript/alchemy_admin/initializer.js +0 -3
  88. data/app/javascript/alchemy_admin/link_dialog.js +1 -6
  89. data/app/javascript/alchemy_admin/templates/compiled.js +1 -1
  90. data/app/javascript/alchemy_admin/utils/ajax.js +15 -3
  91. data/app/javascript/alchemy_admin.js +0 -6
  92. data/app/models/alchemy/attachment.rb +4 -44
  93. data/app/models/alchemy/element/definitions.rb +1 -2
  94. data/app/models/alchemy/element/element_ingredients.rb +6 -2
  95. data/app/models/alchemy/element.rb +54 -13
  96. data/app/models/alchemy/element_definition.rb +4 -1
  97. data/app/models/alchemy/elements_repository.rb +6 -0
  98. data/app/models/alchemy/folded_page.rb +2 -2
  99. data/app/models/alchemy/ingredient.rb +38 -1
  100. data/app/models/alchemy/ingredient_definition.rb +4 -1
  101. data/app/models/alchemy/ingredient_validator.rb +6 -2
  102. data/app/models/alchemy/ingredients/color.rb +10 -0
  103. data/app/models/alchemy/ingredients/headline.rb +2 -17
  104. data/app/models/alchemy/ingredients/picture.rb +4 -4
  105. data/app/models/alchemy/ingredients/select.rb +19 -0
  106. data/app/models/alchemy/language/code.rb +0 -1
  107. data/app/models/alchemy/node.rb +28 -1
  108. data/app/models/alchemy/page/page_naming.rb +0 -7
  109. data/app/models/alchemy/page/page_natures.rb +7 -3
  110. data/app/models/alchemy/page/page_scopes.rb +13 -1
  111. data/app/models/alchemy/page/publisher.rb +14 -2
  112. data/app/models/alchemy/page.rb +102 -23
  113. data/app/models/alchemy/page_definition.rb +4 -1
  114. data/app/models/alchemy/page_version.rb +22 -6
  115. data/app/models/alchemy/picture.rb +10 -11
  116. data/app/models/alchemy/picture_variant.rb +1 -3
  117. data/app/models/alchemy/resource.rb +1 -1
  118. data/app/models/alchemy/storage_adapter/active_storage.rb +14 -2
  119. data/app/models/alchemy/storage_adapter/dragonfly.rb +12 -0
  120. data/app/models/alchemy/storage_adapter.rb +2 -0
  121. data/app/models/concerns/alchemy/picture_thumbnails.rb +4 -4
  122. data/app/models/concerns/alchemy/publishable.rb +54 -0
  123. data/app/models/concerns/alchemy/relatable_resource.rb +4 -14
  124. data/app/serializers/alchemy/page_tree_serializer.rb +11 -31
  125. data/app/services/alchemy/copy_page.rb +17 -0
  126. data/app/services/alchemy/duplicate_element.rb +1 -1
  127. data/app/services/alchemy/page_tree_preloader.rb +105 -0
  128. data/app/stylesheets/alchemy/_extends.scss +3 -9
  129. data/app/stylesheets/alchemy/_mixins.scss +3 -1
  130. data/app/stylesheets/alchemy/_themes.scss +19 -10
  131. data/app/stylesheets/alchemy/admin/archive.scss +1 -0
  132. data/app/stylesheets/alchemy/admin/base.scss +5 -2
  133. data/app/stylesheets/alchemy/admin/buttons.scss +3 -3
  134. data/app/stylesheets/alchemy/admin/element-select.scss +18 -0
  135. data/app/stylesheets/alchemy/admin/elements.scss +123 -23
  136. data/app/stylesheets/alchemy/admin/errors.scss +1 -1
  137. data/app/stylesheets/alchemy/admin/flash.scss +6 -4
  138. data/app/stylesheets/alchemy/admin/images.scss +9 -5
  139. data/app/stylesheets/alchemy/admin/list_filter.scss +4 -4
  140. data/app/stylesheets/alchemy/admin/navigation.scss +1 -1
  141. data/app/stylesheets/alchemy/admin/notices.scss +1 -2
  142. data/app/stylesheets/alchemy/admin/selects.scss +36 -21
  143. data/app/stylesheets/alchemy/admin/shoelace.scss +14 -1
  144. data/app/stylesheets/alchemy/admin/sitemap.scss +11 -3
  145. data/app/stylesheets/alchemy/admin/tags.scss +3 -1
  146. data/app/stylesheets/alchemy/admin/toolbar.scss +1 -1
  147. data/app/views/alchemy/_edit_mode.html.erb +1 -1
  148. data/app/views/alchemy/_menubar.html.erb +1 -1
  149. data/app/views/alchemy/admin/attachments/_archive_overlay.html.erb +35 -31
  150. data/app/views/alchemy/admin/attachments/_files_list.html.erb +1 -1
  151. data/app/views/alchemy/admin/attachments/_library_sidebar.html.erb +6 -0
  152. data/app/views/alchemy/admin/attachments/_overlay_file_list.html.erb +1 -1
  153. data/app/views/alchemy/admin/attachments/_replace_button.html.erb +1 -8
  154. data/app/views/alchemy/admin/attachments/_sorting_select.html.erb +13 -0
  155. data/app/views/alchemy/admin/attachments/_tag_list.html.erb +2 -3
  156. data/app/views/alchemy/admin/attachments/index.html.erb +5 -11
  157. data/app/views/alchemy/admin/attachments/show.html.erb +1 -1
  158. data/app/views/alchemy/admin/clipboard/_button.html.erb +1 -0
  159. data/app/views/alchemy/admin/clipboard/index.html.erb +4 -5
  160. data/app/views/alchemy/admin/clipboard/insert.turbo_stream.erb +1 -1
  161. data/app/views/alchemy/admin/crop.html.erb +5 -7
  162. data/app/views/alchemy/admin/dashboard/widgets/_locked_pages.html.erb +1 -1
  163. data/app/views/alchemy/admin/elements/_add_nested_element_form.html.erb +6 -6
  164. data/app/views/alchemy/admin/elements/_fixed_element.html.erb +1 -1
  165. data/app/views/alchemy/admin/elements/_footer.html.erb +7 -1
  166. data/app/views/alchemy/admin/elements/_header.html.erb +5 -5
  167. data/app/views/alchemy/admin/elements/_toolbar.html.erb +33 -8
  168. data/app/views/alchemy/admin/elements/create.turbo_stream.erb +10 -10
  169. data/app/views/alchemy/admin/elements/index.html.erb +29 -16
  170. data/app/views/alchemy/admin/elements/new.html.erb +2 -2
  171. data/app/views/alchemy/admin/ingredients/update.turbo_stream.erb +3 -5
  172. data/app/views/alchemy/admin/leave.html.erb +1 -1
  173. data/app/views/alchemy/admin/nodes/_node.html.erb +19 -0
  174. data/app/views/alchemy/admin/nodes/edit.html.erb +1 -1
  175. data/app/views/alchemy/admin/nodes/index.html.erb +3 -1
  176. data/app/views/alchemy/admin/nodes/new.html.erb +14 -1
  177. data/app/views/alchemy/admin/pages/_current_page.html.erb +3 -1
  178. data/app/views/alchemy/admin/pages/_form.html.erb +21 -9
  179. data/app/views/alchemy/admin/pages/_page_status.html.erb +1 -1
  180. data/app/views/alchemy/admin/pages/_publication_fields.html.erb +28 -26
  181. data/app/views/alchemy/admin/pages/_table.html.erb +0 -7
  182. data/app/views/alchemy/admin/pages/_toolbar.html.erb +3 -6
  183. data/app/views/alchemy/admin/pages/edit.html.erb +5 -11
  184. data/app/views/alchemy/admin/pages/flush.turbo_stream.erb +2 -0
  185. data/app/views/alchemy/admin/pages/fold.turbo_stream.erb +5 -0
  186. data/app/views/alchemy/admin/pages/index.html.erb +5 -3
  187. data/app/views/alchemy/admin/pages/new.html.erb +2 -12
  188. data/app/views/alchemy/admin/pages/publish.turbo_stream.erb +12 -0
  189. data/app/views/alchemy/admin/pages/tree.html.erb +13 -0
  190. data/app/views/alchemy/admin/pages/update.turbo_stream.erb +5 -16
  191. data/app/views/alchemy/admin/partials/_flash_notices.html.erb +1 -1
  192. data/app/views/alchemy/admin/partials/{_remote_search_form.html.erb → _overlay_search_form.html.erb} +1 -2
  193. data/app/views/alchemy/admin/partials/_paste_from_clipboard_form.html.erb +12 -0
  194. data/app/views/alchemy/admin/pictures/_archive_overlay.html.erb +24 -21
  195. data/app/views/alchemy/admin/pictures/_filter_and_size_bar.html.erb +18 -26
  196. data/app/views/alchemy/admin/pictures/_picture.html.erb +12 -16
  197. data/app/views/alchemy/admin/pictures/_picture_to_assign.html.erb +3 -6
  198. data/app/views/alchemy/admin/pictures/_tag_list.html.erb +2 -3
  199. data/app/views/alchemy/admin/pictures/index.html.erb +0 -1
  200. data/app/views/alchemy/admin/pictures/update.turbo_stream.erb +1 -1
  201. data/app/views/alchemy/admin/resources/_resource_usage_info.html.erb +1 -1
  202. data/app/views/alchemy/admin/resources/_tag_list.html.erb +2 -3
  203. data/app/views/alchemy/admin/styleguide/index.html.erb +25 -20
  204. data/app/views/alchemy/admin/tags/edit.html.erb +1 -1
  205. data/app/views/alchemy/admin/tinymce/_setup.html.erb +2 -2
  206. data/app/views/alchemy/admin/uploader/_button.html.erb +1 -15
  207. data/app/views/alchemy/attachments/show.html.erb +1 -1
  208. data/app/views/alchemy/base/permission_denied.js.erb +1 -1
  209. data/app/views/alchemy/ingredients/shared/_anchor.html.erb +9 -7
  210. data/app/views/alchemy/ingredients/shared/_link_tools.html.erb +12 -5
  211. data/app/views/alchemy/ingredients/shared/_picture_tools.html.erb +10 -11
  212. data/app/views/alchemy/language_links/_spacer.html.erb +1 -1
  213. data/app/views/alchemy/messages_mailer/new.html.erb +1 -1
  214. data/app/views/alchemy/welcome.html.erb +1 -1
  215. data/config/locales/alchemy.en.yml +12 -3
  216. data/config/routes.rb +2 -2
  217. data/db/migrate/20230123112425_add_searchable_to_alchemy_pages.rb +1 -1
  218. data/db/migrate/20230505132743_add_indexes_to_alchemy_pictures.rb +1 -1
  219. data/db/migrate/20231113104432_create_page_mutexes.rb +1 -1
  220. data/db/migrate/20240314105244_create_alchemy_picture_descriptions.rb +1 -1
  221. data/db/migrate/20250626160259_add_unique_index_to_picture_descriptions.rb +1 -1
  222. data/db/migrate/20250905140323_add_created_at_index_to_pictures_and_attachments.rb +1 -1
  223. data/db/migrate/20251106150010_convert_select_value_for_multiple.rb +11 -0
  224. data/db/migrate/20260102121232_add_metadata_to_page_versions.rb +9 -0
  225. data/db/migrate/20260115164704_add_publication_timestamps_to_alchemy_elements.rb +30 -0
  226. data/db/migrate/20260115164705_add_index_to_element_publication_timestamps.rb +13 -0
  227. data/lib/alchemy/ability_helper.rb +1 -3
  228. data/lib/alchemy/auth_accessors.rb +51 -117
  229. data/lib/alchemy/configuration.rb +1 -2
  230. data/lib/alchemy/configurations/main.rb +63 -0
  231. data/lib/alchemy/controller_actions.rb +2 -3
  232. data/lib/alchemy/engine.rb +9 -12
  233. data/lib/alchemy/error_tracking/error_logger.rb +1 -1
  234. data/lib/alchemy/errors.rb +1 -1
  235. data/lib/alchemy/logger.rb +34 -4
  236. data/lib/alchemy/name_conversions.rb +0 -6
  237. data/lib/alchemy/seeder.rb +2 -2
  238. data/lib/alchemy/tasks/usage.rb +4 -4
  239. data/lib/alchemy/test_support/factories/page_version_factory.rb +3 -0
  240. data/lib/alchemy/test_support/having_picture_thumbnails_examples.rb +30 -0
  241. data/lib/alchemy/test_support/shared_ingredient_editor_examples.rb +26 -6
  242. data/lib/alchemy/test_support/shared_publishable_examples.rb +114 -0
  243. data/lib/alchemy/upgrader/eight_one.rb +56 -0
  244. data/lib/alchemy/upgrader.rb +9 -1
  245. data/lib/alchemy/version.rb +1 -1
  246. data/lib/alchemy.rb +1 -4
  247. data/lib/alchemy_cms.rb +0 -1
  248. data/lib/generators/alchemy/elements/templates/view.html.erb +3 -3
  249. data/lib/generators/alchemy/ingredient/ingredient_generator.rb +6 -8
  250. data/lib/generators/alchemy/ingredient/templates/editor_component.rb.tt +22 -0
  251. data/lib/generators/alchemy/page_layouts/templates/layout.html.erb +1 -1
  252. data/lib/generators/alchemy/site_layouts/templates/layout.html.erb +1 -1
  253. data/lib/tasks/alchemy/upgrade.rake +21 -7
  254. data/vendor/javascript/shoelace.min.js +713 -31
  255. data/vendor/javascript/tinymce.min.js +1 -1
  256. metadata +104 -84
  257. data/app/decorators/alchemy/element_editor.rb +0 -90
  258. data/app/helpers/alchemy/admin/pictures_helper.rb +0 -14
  259. data/app/javascript/alchemy_admin/file_editors.js +0 -28
  260. data/app/javascript/alchemy_admin/image_loader.js +0 -54
  261. data/app/javascript/alchemy_admin/page_sorter.js +0 -71
  262. data/app/javascript/alchemy_admin/sitemap.js +0 -154
  263. data/app/javascript/alchemy_admin/templates/page_folder.hbs +0 -3
  264. data/app/views/alchemy/admin/attachments/archive_overlay.js.erb +0 -4
  265. data/app/views/alchemy/admin/pages/_page.html.erb +0 -163
  266. data/app/views/alchemy/admin/pages/_sitemap.html.erb +0 -30
  267. data/app/views/alchemy/admin/pages/flush.js.erb +0 -2
  268. data/app/views/alchemy/admin/pictures/archive_overlay.js.erb +0 -5
  269. data/app/views/alchemy/admin/pictures/index.js.erb +0 -2
  270. data/app/views/alchemy/ingredients/_audio_editor.html.erb +0 -5
  271. data/app/views/alchemy/ingredients/_boolean_editor.html.erb +0 -11
  272. data/app/views/alchemy/ingredients/_datetime_editor.html.erb +0 -20
  273. data/app/views/alchemy/ingredients/_file_editor.html.erb +0 -52
  274. data/app/views/alchemy/ingredients/_headline_editor.html.erb +0 -44
  275. data/app/views/alchemy/ingredients/_html_editor.html.erb +0 -8
  276. data/app/views/alchemy/ingredients/_link_editor.html.erb +0 -30
  277. data/app/views/alchemy/ingredients/_node_editor.html.erb +0 -13
  278. data/app/views/alchemy/ingredients/_number_editor.html.erb +0 -24
  279. data/app/views/alchemy/ingredients/_page_editor.html.erb +0 -13
  280. data/app/views/alchemy/ingredients/_picture_editor.html.erb +0 -59
  281. data/app/views/alchemy/ingredients/_richtext_editor.html.erb +0 -15
  282. data/app/views/alchemy/ingredients/_select_editor.html.erb +0 -31
  283. data/app/views/alchemy/ingredients/_text_editor.html.erb +0 -29
  284. data/app/views/alchemy/ingredients/_video_editor.html.erb +0 -5
  285. data/lib/generators/alchemy/ingredient/templates/editor.html.erb +0 -14
  286. /data/{lib → app/models}/alchemy/permissions.rb +0 -0
@@ -0,0 +1,13 @@
1
+ <div id="sorting_select">
2
+ <div class="filter-input">
3
+ <alchemy-auto-submit>
4
+ <%= label_tag "q[s]", t(".label") %>
5
+ <%= select_tag "q[s]", options_for_select([
6
+ [t(".by_latest"), "created_at desc"],
7
+ [t(".alphabetical"), "name asc"]
8
+ ], search_filter_params.dig(:q, :s)),
9
+ form: "resource_search",
10
+ class: "full_width" %>
11
+ </alchemy-auto-submit>
12
+ </div>
13
+ </div>
@@ -1,14 +1,13 @@
1
1
  <% if Alchemy::Attachment.tag_counts.any? %>
2
2
  <h3><%= Alchemy.t("Filter by tag") %></h3>
3
- <%= render Alchemy::Admin::ListFilter.new(".tag-list li") %>
3
+ <%= render Alchemy::Admin::ListFilter.new(".tag-list li", placeholder: Alchemy.t("Filter tags")) %>
4
4
  <ul>
5
- <%= render_tag_list('Alchemy::Attachment') %>
5
+ <%= render Alchemy::Admin::TagsList.new('Alchemy::Attachment') %>
6
6
  </ul>
7
7
  <% if search_filter_params[:tagged_with].present? %>
8
8
  <%= link_to(
9
9
  render_icon(:times, size: '1x') + Alchemy.t('Remove tag filter'),
10
10
  url_for(search_filter_params.except(:tagged_with)),
11
- remote: request.xhr?,
12
11
  class: 'secondary button small with_icon'
13
12
  ) %>
14
13
  <% end %>
@@ -3,26 +3,20 @@
3
3
  <% if can? :create, Alchemy::Attachment %>
4
4
  <div class="toolbar_button">
5
5
  <%= render 'alchemy/admin/uploader/button',
6
- redirect_url: alchemy.admin_attachments_path,
6
+ redirect_url: alchemy.admin_attachments_path(q: {
7
+ last_upload: true
8
+ }),
7
9
  object: Alchemy::Attachment.new,
8
10
  file_attribute: 'file' %>
9
11
  </div>
10
12
  <% end %>
11
13
  </div>
14
+
12
15
  <%= render 'alchemy/admin/partials/search_form' %>
13
16
  <% end %>
14
17
 
15
18
  <div id="archive_all" class="with_tag_filter resources-table-wrapper">
16
19
  <%= render 'alchemy/admin/resources/table_header' %>
17
20
  <%= render 'files_list' %>
18
-
19
- <div id="library_sidebar">
20
- <%= render 'filter_bar' if resource_has_filters %>
21
-
22
- <% if Alchemy::Attachment.tag_counts.any? %>
23
- <div class="tag-list with_filter_bar<%= ' filtered' if search_filter_params[:tagged_with].present? %>">
24
- <%= render 'tag_list' %>
25
- </div>
26
- <% end %>
27
- </div>
21
+ <%= render 'library_sidebar' %>
28
22
  </div>
@@ -51,7 +51,7 @@
51
51
  <%= video_tag(@attachment.url, preload: "metadata", controls: true) %>
52
52
  </div>
53
53
  <% when "application/pdf" %>
54
- <iframe src="<%= @attachment.url %>" frameborder="0"></iframe>
54
+ <iframe src="<%= @attachment.url %>" frameborder="0" title="Alchemy.t('File preview')"></iframe>
55
55
  <% end %>
56
56
  <% end %>
57
57
  </div>
@@ -8,5 +8,6 @@
8
8
  size: "400x305"
9
9
  },
10
10
  id: "clipboard_button",
11
+ tooltip_placement: local_assigns.fetch(:tooltip_placement, "top-start"),
11
12
  if_permitted_to: [:index, :alchemy_admin_clipboard]
12
13
  ) %>
@@ -1,18 +1,17 @@
1
1
  <%= turbo_frame_tag :clipboard_items do %>
2
- <% if @clipboard_items.blank? %>
2
+ <% if clipboard_items.blank? %>
3
3
  <%= render_message do %>
4
4
  <%= Alchemy.t('No items in your clipboard') %>
5
5
  <% end %>
6
6
  <% else %>
7
7
  <%= render_message do %>
8
- <%= Alchemy.t('Add items from clipboard via "Add Element" button') %>
8
+ <%= Alchemy.t("Add items from clipboard via \"Add #{remarkable_type.singularize.capitalize}\" button") %>
9
9
  <% end %>
10
-
11
10
  <ul>
12
- <% @clipboard_items.each do |item| %>
11
+ <% clipboard_items.each do |item| %>
13
12
  <% item_class = item.class.name.demodulize.underscore.pluralize %>
14
13
  <li id="clipboard_item_<%= item.id %>" class="<%= item_class %>">
15
- <% if item_class == 'pages' %>
14
+ <% if %[pages nodes].include? item_class %>
16
15
  <%= render_icon(:file) %>
17
16
  <%= truncate(item.name, length: 50) %>
18
17
  <% else %>
@@ -1,6 +1,6 @@
1
1
  <% if @item.class.name == 'Alchemy::Element' && params[:remove] -%>
2
2
  <% if @item.fixed? -%>
3
- <alchemy-action name="removeFixedElement" params="[<%= @item.id %>]" />
3
+ <alchemy-action name="removeFixedElement" params="[<%= @item.id %>]"></alchemy-action>
4
4
  <% else -%>
5
5
  <%= turbo_stream.remove "element_#{@item.id}" %>
6
6
  <% end -%>
@@ -1,15 +1,15 @@
1
- <div id="jscropper">
2
1
  <% if @no_image_notice %>
3
2
  <%= render_message :warn do %>
4
3
  <%= @no_image_notice %>
5
4
  <% end %>
6
5
  <% else %>
6
+ <div id="jscropper">
7
7
  <%= render_message do %>
8
8
  <%= simple_format Alchemy.t(:explain_cropping) %>
9
9
  <% end %>
10
- <div class="thumbnail_background">
11
- <%= image_tag @picture.url(flatten: true), id: 'imageToCrop' %>
12
- </div>
10
+ <%= content_tag "alchemy-picture-thumbnail", nil,
11
+ src: @picture.url(flatten: true),
12
+ id: "imageToCrop" %>
13
13
  <form>
14
14
  <%= button_tag Alchemy.t(:apply), type: 'submit' %>
15
15
  <%= button_tag Alchemy.t('Reset Imagemask'), class: 'reset_mask secondary', type: 'reset' %>
@@ -19,11 +19,9 @@
19
19
  <% if @settings %>
20
20
  <script type="module">
21
21
  import ImageCropper from "alchemy_admin/image_cropper";
22
- import ImageLoader from "alchemy_admin/image_loader";
23
22
 
24
- const image = document.getElementById("imageToCrop");
23
+ const image = document.getElementById("imageToCrop")?.querySelector("img");
25
24
 
26
- new ImageLoader(image);
27
25
  new ImageCropper(
28
26
  image,
29
27
  <%= @settings[:default_box].to_json %>,
@@ -37,7 +37,7 @@
37
37
  </td>
38
38
  <td>
39
39
  <% if (current_alchemy_user.id == page.locked_by) || can?(:manage, Alchemy::Current.site) %>
40
- <%= form_tag(alchemy.unlock_admin_page_path(page, :redirect_to => alchemy.admin_dashboard_path)) do %>
40
+ <%= form_tag(alchemy.unlock_admin_page_path(page, :redirect_to => alchemy.admin_dashboard_url)) do %>
41
41
  <button class="icon_button small" title="<%= Alchemy.t(:explain_unlocking) %>">
42
42
  <%= render_icon(:close, size: '1x') %>
43
43
  </button>
@@ -1,25 +1,25 @@
1
1
  <%= content_tag :div, class: 'add-nested-element', data: { element_id: element.id } do %>
2
2
  <% if element.nestable_elements.length == 1 &&
3
3
  (nestable_element = element.nestable_elements.first) &&
4
- Alchemy::Element.all_from_clipboard_for_parent_element(get_clipboard("elements"), element).none?
4
+ Alchemy::Element.all_from_clipboard_for_parent_element(clipboard, element).none?
5
5
  %>
6
6
  <%= turbo_frame_tag("new_nested_element_#{element.id}") do %>
7
7
  <%= form_for [:admin, Alchemy::Element.new(name: nestable_element)], html: { class: 'add-nested-element-form', id: nil } do |f| %>
8
8
  <%= f.hidden_field :name %>
9
9
  <%= f.hidden_field :page_version_id, value: element.page_version_id %>
10
10
  <%= f.hidden_field :parent_element_id, value: element.id %>
11
- <button class="add-nestable-element-button" is="alchemy-button">
11
+ <%= button_tag class: "add-nestable-element-button", is: "alchemy-button", disabled: cannot?(:create, Alchemy::Element) do %>
12
12
  <%= Alchemy.t(:add_nested_element, name: Alchemy.t(nestable_element.to_sym, scope: 'element_names')) %>
13
- </button>
13
+ <% end %>
14
14
  <% end %>
15
15
  <% end %>
16
16
  <% else %>
17
17
  <%= link_to_dialog (nestable_element ? Alchemy.t(:add_nested_element, name: Alchemy.t(nestable_element.to_sym, scope: 'element_names')) : Alchemy.t("New Element")),
18
- alchemy.new_admin_element_path(
18
+ can?(:create, Alchemy::Element) ? alchemy.new_admin_element_path(
19
19
  parent_element_id: element.id,
20
20
  page_version_id: element.page_version_id
21
- ), {
22
- size: "320x125",
21
+ ) : nil, {
22
+ size: "380x125",
23
23
  title: Alchemy.t("New Element")
24
24
  }, class: "button add-nestable-element-button" %>
25
25
  <% end %>
@@ -2,5 +2,5 @@
2
2
  <%= element.display_name %>
3
3
  </sl-tab>
4
4
  <sl-tab-panel id="fixed_element_<%= element.id %>" name="fixed-element-<%= element.id %>" style="--padding: 0" class="scrollable-elements">
5
- <%= render Alchemy::ElementEditor.new(element) %>
5
+ <%= render Alchemy::Admin::ElementEditor.new(element: element) %>
6
6
  </sl-tab-panel>
@@ -5,7 +5,13 @@
5
5
  </p>
6
6
  <% end %>
7
7
 
8
- <button type="submit" form="element_<%= element.id %>_form" is="alchemy-button" data-turbo="false">
8
+ <button
9
+ form="element_<%= element.id %>_form"
10
+ is="alchemy-button"
11
+ data-turbo="false"
12
+ type="submit"
13
+ <%= "disabled" if cannot?(:update, element) %>
14
+ >
9
15
  <%= Alchemy.t(:save) %>
10
16
  </button>
11
17
  </div>
@@ -1,15 +1,15 @@
1
1
  <div class="element-header<%= ' has-hint' if element.has_hint? %>">
2
- <span class="element-handle">
2
+ <span class="element-handle<%= ' draggable' if can?(:order, element) %>">
3
3
  <% if element.definition.blank? %>
4
- <%= hint_with_tooltip Alchemy.t(:element_definition_missing) %>
4
+ <%= hint_with_tooltip Alchemy.t(:element_definition_missing), icon_class: "element-icon" %>
5
5
  <% elsif element.deprecated? %>
6
- <%= hint_with_tooltip element.deprecation_notice %>
6
+ <%= hint_with_tooltip element.deprecation_notice, icon_class: "element-icon" %>
7
7
  <% else %>
8
- <span class="icon element">
8
+ <span class="element-icon">
9
9
  <%= element.definition.icon_file %>
10
10
  </span>
11
- <%= render_icon(:draggable, style: false, fixed_width: false) %>
12
11
  <% end %>
12
+ <%= render_icon(:draggable, style: false, fixed_width: false, class: "handle-icon") %>
13
13
  </span>
14
14
  <span class="element-title">
15
15
  <span class="preview_text_element_name"><%= element.display_name %></span>
@@ -1,34 +1,59 @@
1
1
  <% remarkable_type = "elements" %>
2
2
  <div class="element-toolbar">
3
- <sl-tooltip placement="top-start" content="<%= Alchemy.t(:copy_element) %>">
3
+ <sl-tooltip
4
+ placement="top-start"
5
+ content="<%= Alchemy.t(:copy_element) %>"
6
+ <%= "disabled" if cannot?(:create, element) %>
7
+ >
4
8
  <%= button_to(
5
9
  render_icon("file-copy"),
6
10
  alchemy.insert_admin_clipboard_path(remarkable_type: remarkable_type, remarkable_id: element.id),
7
11
  "data-turbo-method": :post,
8
- class: "icon_button"
12
+ class: "icon_button",
13
+ disabled: cannot?(:create, element)
9
14
  ) %>
10
15
  </sl-tooltip>
11
- <sl-tooltip content="<%= Alchemy.t(:cut_element) %>">
16
+ <sl-tooltip
17
+ content="<%= Alchemy.t(:cut_element) %>"
18
+ <%= "disabled" if cannot?(:create, element) %>
19
+ >
12
20
  <%= button_to(
13
21
  render_icon("scissors-cut"),
14
22
  alchemy.insert_admin_clipboard_path(remarkable_type: remarkable_type, remarkable_id: element.id, remove: true),
15
23
  "data-turbo-method": :post,
16
- class: "icon_button"
24
+ class: "icon_button",
25
+ disabled: cannot?(:create, element)
17
26
  ) %>
18
27
  </sl-tooltip>
19
- <sl-tooltip content="<%= Alchemy.t("Delete element") %>">
28
+ <sl-tooltip
29
+ content="<%= Alchemy.t("Delete element") %>"
30
+ <%= "disabled" if cannot?(:destroy, element) %>
31
+ >
20
32
  <alchemy-delete-element-button
21
33
  href="<%= alchemy.admin_element_path(element) %>"
22
34
  message="<%= Alchemy.t(:confirm_to_delete_element) %>"
23
35
  >
24
- <button type="button" class="icon_button" data-turbo="false">
36
+ <button
37
+ class="icon_button"
38
+ data-turbo="false"
39
+ type="button"
40
+ <%= "disabled" if cannot?(:destroy, element) %>
41
+ >
25
42
  <%= render_icon('delete-bin-2') %>
26
43
  </button>
27
44
  </alchemy-delete-element-button>
28
45
  </sl-tooltip>
29
- <sl-tooltip content="<%= element.public? ? Alchemy.t(:hide_element) : Alchemy.t(:show_element) %>" placement="top-end">
46
+ <sl-tooltip
47
+ content="<%= element.public? ? Alchemy.t(:hide_element) : Alchemy.t(:show_element) %>"
48
+ placement="top-end"
49
+ <%= "disabled" if cannot?(:update, element) %>
50
+ >
30
51
  <alchemy-publish-element-button>
31
- <sl-switch <%= element.public? ? nil : "checked" %> size="small">
52
+ <sl-switch
53
+ size="small"
54
+ <%= "checked" if !element.public? %>
55
+ <%= "disabled" if cannot?(:update, element) %>
56
+ >
32
57
  <%= t(".hide") %>
33
58
  </sl-switch>
34
59
  </alchemy-publish-element-button>
@@ -1,11 +1,3 @@
1
- <% opts = {
2
- partial: "alchemy/admin/elements/element",
3
- locals: {
4
- element: Alchemy::ElementEditor.new(@element),
5
- created: true
6
- }
7
- } %>
8
-
9
1
  <% if @element.fixed? %>
10
2
  <% target = "fixed_element_#{@element.id}" %>
11
3
  <%= turbo_stream.append "fixed-elements",
@@ -22,15 +14,23 @@
22
14
  <% end %>
23
15
 
24
16
  <% if @insert_at_top %>
25
- <%= turbo_stream.prepend target, **opts %>
17
+ <%= turbo_stream.prepend target do %>
18
+ <%= render Alchemy::Admin::ElementEditor.new(element: @element, created: true) %>
19
+ <% end %>
26
20
  <% else %>
27
- <%= turbo_stream.append target, **opts %>
21
+ <%= turbo_stream.append target do %>
22
+ <%= render Alchemy::Admin::ElementEditor.new(element: @element, created: true) %>
23
+ <% end %>
28
24
  <% end %>
29
25
 
30
26
  <%= turbo_stream.replace "clipboard_button",
31
27
  partial: "alchemy/admin/clipboard/button",
32
28
  locals: {remarkable_type: "elements"} %>
33
29
 
30
+ <%= turbo_stream.replace "publish_page_button" do %>
31
+ <%= render Alchemy::Admin::PublishPageButton.new(page: @element.page) %>
32
+ <% end %>
33
+
34
34
  <alchemy-growl>
35
35
  <%= Alchemy.t(:successfully_added_element) %>
36
36
  </alchemy-growl>
@@ -9,16 +9,26 @@
9
9
  label: Alchemy.t("New Element"),
10
10
  dialog_options: {
11
11
  title: Alchemy.t("New Element"),
12
- size: "320x125"
12
+ size: "380x125"
13
13
  },
14
+ tooltip_placement: "bottom-start",
14
15
  if_permitted_to: [:create, Alchemy::Element]
15
16
  ) %>
16
- <%= render "alchemy/admin/clipboard/button", remarkable_type: "elements" %>
17
- <sl-tooltip content="<%= Alchemy.t("Collapse all elements") %>" placement="top-end" class="right">
18
- <button id="collapse-all-elements-button" class="icon_button">
19
- <alchemy-icon name="contract-up-down"></alchemy-icon>
20
- </button>
21
- </sl-tooltip>
17
+ <%= render "alchemy/admin/clipboard/button", remarkable_type: "elements", tooltip_placement: "bottom-start" %>
18
+ <div class="right">
19
+ <%= render Alchemy::Admin::ListFilter.new(
20
+ "alchemy-element-editor",
21
+ name_attribute: "filter-text",
22
+ placeholder: Alchemy.t("Filter elements"),
23
+ hotkey: "alt+f"
24
+ ) %>
25
+ <div class="toolbar_spacer"></div>
26
+ <sl-tooltip content="<%= Alchemy.t("Collapse all elements") %>" placement="bottom-end">
27
+ <button id="collapse-all-elements-button" class="icon_button">
28
+ <alchemy-icon name="contract-up-down"></alchemy-icon>
29
+ </button>
30
+ </sl-tooltip>
31
+ </div>
22
32
  </div>
23
33
  <% if @page.element_definitions.any?(&:fixed) %>
24
34
  <sl-tab-group id="fixed-elements">
@@ -28,24 +38,27 @@
28
38
  <%= render partial: "fixed_element", collection: @fixed_elements, as: :element %>
29
39
  <sl-tab-panel
30
40
  name="main-content-elements"
31
- class="sortable-elements scrollable-elements"
32
- data-droppable-elements="<%= @page.element_definition_names.join(' ') %>"
33
- data-element-name="main-content-elements"
34
- id="main-content-elements"
35
41
  style="--padding: 0"
36
42
  >
37
- <%= render @elements.map { |element| Alchemy::ElementEditor.new(element) } %>
43
+ <alchemy-sortable-elements
44
+ id="main-content-elements"
45
+ class="scrollable-elements"
46
+ data-droppable-elements="<%= @page.element_definition_names.join(' ') %>"
47
+ data-element-name="main-content-elements"
48
+ >
49
+ <%= render Alchemy::Admin::ElementEditor.with_collection(@elements) %>
50
+ </alchemy-sortable-elements>
38
51
  </sl-tab-panel>
39
52
  </sl-tab-group>
40
53
  <% else %>
41
- <div
54
+ <alchemy-sortable-elements
42
55
  id="main-content-elements"
43
- class="sortable-elements scrollable-elements"
56
+ class="scrollable-elements"
44
57
  data-droppable-elements="<%= @page.element_definition_names.join(' ') %>"
45
58
  data-element-name="main-content-elements"
46
59
  >
47
- <%= render @elements.map { |element| Alchemy::ElementEditor.new(element) } %>
48
- </div>
60
+ <%= render Alchemy::Admin::ElementEditor.with_collection(@elements) %>
61
+ </alchemy-sortable-elements>
49
62
  <% end %>
50
63
  </alchemy-elements-window>
51
64
  <% end %>
@@ -1,4 +1,4 @@
1
- <%- if @clipboard_items.blank? -%>
1
+ <%- if clipboard_items.blank? -%>
2
2
  <%= render 'form' %>
3
3
  <%- else -%>
4
4
  <sl-tab-group id="overlay_tabs">
@@ -14,7 +14,7 @@
14
14
  <div class="input select">
15
15
  <label for="paste_from_clipboard" class="control-label"><%= Alchemy.t("Element") %></label>
16
16
  <%= select_tag 'paste_from_clipboard',
17
- clipboard_select_tag_options(@clipboard_items),
17
+ clipboard_select_tag_options(clipboard_items),
18
18
  is: 'alchemy-select' %>
19
19
  </div>
20
20
  <%= f.submit Alchemy.t(:paste) %>
@@ -4,9 +4,7 @@
4
4
  <alchemy-action name="updateAnchorIcon" params="<%= [@ingredient.id, @ingredient.dom_id.present?].to_json %>"></alchemy-action>
5
5
  <% end %>
6
6
  <% if @ingredient.type == "Alchemy::Ingredients::Picture" && @ingredient.css_class.present? %>
7
- <%= turbo_stream.replace_all "[data-ingredient-id='#{@ingredient.id}'] .picture_ingredient_css_class",
8
- partial: "alchemy/ingredients/shared/picture_css_class",
9
- locals: {
10
- css_class: Alchemy::IngredientEditor.new(@ingredient).css_class
11
- } %>
7
+ <%= turbo_stream.replace @ingredient do %>
8
+ <%= render Alchemy::Ingredients::PictureEditor.new(@ingredient) %>
9
+ <% end %>
12
10
  <% end %>
@@ -1,7 +1,7 @@
1
1
  <%= render_message do %>
2
2
  <%= Alchemy.t("You are about to leave Alchemy") %>
3
3
  <% end %>
4
- <%= form_tag Alchemy.logout_path, method: Alchemy.logout_method do %>
4
+ <%= form_tag Alchemy.config.logout_path, method: Alchemy.config.logout_method do %>
5
5
  <div class="buttons">
6
6
  <%= link_to Alchemy.t("stay logged in"),
7
7
  alchemy.root_path,
@@ -40,6 +40,25 @@
40
40
  ) %>
41
41
  </sl-tooltip>
42
42
  <% end %>
43
+ <% if can?(:copy, node) %>
44
+ <sl-tooltip content="<%= Alchemy.t(:copy_node) %>">
45
+ <%= link_to(
46
+ render_icon("file-copy"),
47
+ alchemy.insert_admin_clipboard_path(
48
+ remarkable_type: :nodes,
49
+ remarkable_id: node.id,
50
+ ),
51
+ "data-turbo-method": :post,
52
+ class: "icon_button"
53
+ ) %>
54
+ </sl-tooltip>
55
+ <% else %>
56
+ <div class="sitemap_tool disabled">
57
+ <sl-tooltip content="<%= Alchemy.t("Your user role does not allow you to copy this node") %>" class="like-hint-tooltip" placement="bottom-start">
58
+ <%= render_icon("file-copy") %>
59
+ </sl-tooltip>
60
+ </div>
61
+ <% end %>
43
62
  <% if can?(:destroy, node) %>
44
63
  <sl-tooltip content="<%= node.root? ? Alchemy.t(:delete_menu) : Alchemy.t(:delete_node) %>">
45
64
  <%= link_to_confirm_dialog(
@@ -1 +1 @@
1
- <%= render 'form', node: @node, button_label: Alchemy.t(:save) %>
1
+ <%= render 'form', node: @node, button_label: Alchemy.t(:save) %>
@@ -18,11 +18,13 @@
18
18
  },
19
19
  if_permitted_to: [:create, Alchemy::Node]
20
20
  ) if @root_nodes.any? %>
21
+ <div class="toolbar_spacer"></div>
22
+ <%= render "alchemy/admin/clipboard/button", remarkable_type: "nodes" %>
21
23
  </div>
22
24
  <% end %>
23
25
 
24
26
  <div id="archive_all">
25
- <h1 id="node_filter_result"></h1>
27
+ <h1 id="node_filter_result">Node filter display</h1>
26
28
  <% if @root_nodes.any? %>
27
29
  <% @root_nodes.each do |root_node| %>
28
30
  <ul class="nodes_tree list">
@@ -1 +1,14 @@
1
- <%= render 'form', node: @node, button_label: Alchemy.t(:create) %>
1
+ <%- if clipboard_items.blank? -%>
2
+ <%= render 'form', node: @node, button_label: Alchemy.t(:create) %>
3
+ <%- else -%>
4
+ <sl-tab-group id="overlay_tabs">
5
+ <sl-tab slot="nav" panel="create_node_tab"><%= Alchemy.t('New node') %></sl-tab>
6
+ <sl-tab slot="nav" panel="paste_node_tab"><%= Alchemy.t('Paste from clipboard') %></sl-tab>
7
+ <sl-tab-panel name="create_node_tab">
8
+ <%= render 'form', node: @node, button_label: Alchemy.t(:create) %>
9
+ </sl-tab-panel>
10
+ <sl-tab-panel name="paste_node_tab">
11
+ <%= render 'alchemy/admin/partials/paste_from_clipboard_form', item: @node %>
12
+ </sl-tab-panel>
13
+ </sl-tab-group>
14
+ <%- end -%>
@@ -1,4 +1,6 @@
1
1
  <div class="page_status_and_name" id="locked_page_<%= current_page.id %>">
2
2
  <%= render 'alchemy/admin/pages/page_infos', page: current_page %>
3
- <%= render 'alchemy/admin/pages/page_status', page: current_page %>
3
+ <%= render 'alchemy/admin/pages/page_status',
4
+ published: local_assigns.fetch(:published, false),
5
+ page: current_page %>
4
6
  </div>
@@ -19,9 +19,14 @@
19
19
 
20
20
  <%= f.input :name, autofocus: true %>
21
21
  <%= f.input :urlname, as: 'string', input_html: {value: @page.slug}, label: Alchemy::Page.human_attribute_name(:slug) %>
22
- <alchemy-char-counter max-chars="60">
23
- <%= f.input :title %>
24
- </alchemy-char-counter>
22
+
23
+ <%= f.fields_for :draft_version, @page.draft_version do |v| %>
24
+ <alchemy-char-counter max-chars="60">
25
+ <%= v.input :title, input_html: {
26
+ disabled: @page.attribute_fixed?(:title)
27
+ } %>
28
+ </alchemy-char-counter>
29
+ <% end %>
25
30
 
26
31
  <% if Alchemy.config.show_page_searchable_checkbox %>
27
32
  <div class="input check_boxes">
@@ -40,13 +45,20 @@
40
45
  </div>
41
46
  </div>
42
47
 
43
- <alchemy-char-counter max-chars="160">
44
- <%= f.input :meta_description, as: 'text' %>
45
- </alchemy-char-counter>
48
+ <%= f.fields_for :draft_version, @page.draft_version do |v| %>
49
+ <alchemy-char-counter max-chars="160">
50
+ <%= v.input :meta_description, as: 'text', input_html: {
51
+ disabled: @page.attribute_fixed?(:meta_description)
52
+ } %>
53
+ </alchemy-char-counter>
46
54
 
47
- <%= f.input :meta_keywords,
48
- as: 'text',
49
- hint: Alchemy.t('pages.update.comma_seperated') %>
55
+ <%= v.input :meta_keywords,
56
+ as: 'text',
57
+ hint: Alchemy.t('pages.update.comma_seperated'),
58
+ input_html: {
59
+ disabled: @page.attribute_fixed?(:meta_keywords)
60
+ } %>
61
+ <% end %>
50
62
 
51
63
  <%= render Alchemy::Admin::TagsAutocomplete.new do %>
52
64
  <%= f.input :tag_list, input_html: { value: f.object.tag_list.join(",") } %>
@@ -1,5 +1,5 @@
1
1
  <span class="page_status">
2
- <% if page.public? %>
2
+ <% if local_assigns[:published] || page.public? %>
3
3
  <%= render_icon(:cloud, size: "1x") %>
4
4
  <% else %>
5
5
  <%= render_icon("cloud-off", size: "1x", class: "disabled") %>