alchemy_cms 8.0.0.a → 8.0.0.b

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.

Potentially problematic release.


This version of alchemy_cms might be problematic. Click here for more details.

Files changed (182) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -0
  3. data/app/assets/builds/alchemy/admin/page-select.css +1 -1
  4. data/app/assets/builds/alchemy/admin.css +1 -1
  5. data/app/assets/builds/alchemy/dark-theme.css +1 -0
  6. data/app/assets/builds/alchemy/light-theme.css +1 -0
  7. data/app/assets/builds/alchemy/theme.css +1 -0
  8. data/app/assets/builds/alchemy/welcome.css +1 -1
  9. data/app/assets/builds/tinymce/skins/content/alchemy/content.min.css +1 -1
  10. data/app/assets/builds/tinymce/skins/content/alchemy-dark/content.min.css +1 -0
  11. data/app/assets/builds/tinymce/skins/ui/alchemy/skin.min.css +1 -1
  12. data/app/assets/builds/tinymce/skins/ui/alchemy-dark/content.min.css +1 -0
  13. data/app/assets/builds/tinymce/skins/ui/alchemy-dark/skin.min.css +1 -0
  14. data/app/assets/images/alchemy/element_icons/default.svg +1 -0
  15. data/app/assets/images/alchemy/icons-sprite.svg +1 -1
  16. data/app/components/alchemy/admin/element_select.rb +39 -0
  17. data/app/components/alchemy/ingredients/datetime_view.rb +4 -2
  18. data/app/controllers/alchemy/admin/attachments_controller.rb +2 -0
  19. data/app/controllers/alchemy/admin/elements_controller.rb +2 -0
  20. data/app/controllers/alchemy/admin/pages_controller.rb +2 -0
  21. data/app/controllers/alchemy/admin/pictures_controller.rb +19 -33
  22. data/app/controllers/alchemy/pages_controller.rb +19 -2
  23. data/app/controllers/concerns/alchemy/admin/resource_filter.rb +1 -0
  24. data/app/helpers/alchemy/admin/attachments_helper.rb +5 -5
  25. data/app/helpers/alchemy/pages_helper.rb +1 -1
  26. data/app/javascript/alchemy_admin/components/auto_submit.js +20 -0
  27. data/app/javascript/alchemy_admin/components/datepicker.js +8 -5
  28. data/app/javascript/alchemy_admin/components/element_editor/delete_element_button.js +3 -2
  29. data/app/javascript/alchemy_admin/components/element_editor.js +25 -15
  30. data/app/javascript/alchemy_admin/components/element_select.js +43 -0
  31. data/app/javascript/alchemy_admin/components/index.js +5 -0
  32. data/app/javascript/alchemy_admin/components/link_buttons.js +6 -2
  33. data/app/javascript/alchemy_admin/components/remote_select.js +5 -1
  34. data/app/javascript/alchemy_admin/components/tinymce.js +93 -16
  35. data/app/javascript/alchemy_admin/dialog.js +1 -1
  36. data/app/javascript/alchemy_admin/file_editors.js +1 -1
  37. data/app/javascript/alchemy_admin/image_loader.js +4 -2
  38. data/app/javascript/alchemy_admin/picture_editors.js +7 -4
  39. data/app/javascript/alchemy_admin/picture_selector.js +4 -4
  40. data/app/jobs/alchemy/delete_picture_job.rb +12 -0
  41. data/app/models/alchemy/attachment.rb +2 -9
  42. data/app/models/alchemy/element.rb +1 -0
  43. data/app/models/alchemy/element_definition.rb +30 -0
  44. data/app/models/alchemy/ingredient.rb +1 -1
  45. data/app/models/alchemy/language.rb +2 -7
  46. data/app/models/alchemy/page/page_naming.rb +4 -11
  47. data/app/models/alchemy/page/page_natures.rb +16 -11
  48. data/app/models/alchemy/page.rb +1 -6
  49. data/app/models/alchemy/page_definition.rb +1 -1
  50. data/app/models/alchemy/picture.rb +6 -17
  51. data/app/models/alchemy/site/layout.rb +1 -0
  52. data/app/models/alchemy/site.rb +1 -6
  53. data/app/models/alchemy/storage_adapter/dragonfly/picture_url.rb +7 -2
  54. data/app/models/alchemy/storage_adapter/dragonfly.rb +24 -2
  55. data/app/models/concerns/alchemy/relatable_resource.rb +28 -0
  56. data/app/stylesheets/alchemy/_custom-properties.scss +162 -0
  57. data/app/stylesheets/alchemy/_mixins.scss +12 -24
  58. data/app/stylesheets/alchemy/_themes.scss +540 -0
  59. data/app/stylesheets/alchemy/admin/archive.scss +28 -8
  60. data/app/stylesheets/alchemy/admin/attachments.scss +10 -33
  61. data/app/stylesheets/alchemy/admin/base.scss +4 -1
  62. data/app/stylesheets/alchemy/admin/buttons.scss +7 -32
  63. data/app/stylesheets/alchemy/admin/dialogs.scss +17 -7
  64. data/app/stylesheets/alchemy/admin/element-select.scss +11 -0
  65. data/app/stylesheets/alchemy/admin/elements.scss +94 -33
  66. data/app/stylesheets/alchemy/admin/filters.scss +8 -9
  67. data/app/stylesheets/alchemy/admin/flatpickr.scss +12 -27
  68. data/app/stylesheets/alchemy/admin/form_fields.scss +0 -15
  69. data/app/stylesheets/alchemy/admin/forms.scss +3 -8
  70. data/app/stylesheets/alchemy/admin/frame.scss +5 -7
  71. data/app/stylesheets/alchemy/admin/icons.scss +0 -9
  72. data/app/stylesheets/alchemy/admin/image_library.scss +13 -55
  73. data/app/stylesheets/alchemy/admin/navigation.scss +1 -11
  74. data/app/stylesheets/alchemy/admin/node-select.scss +1 -10
  75. data/app/stylesheets/alchemy/admin/notices.scss +5 -4
  76. data/app/stylesheets/alchemy/admin/page-select.scss +16 -0
  77. data/app/stylesheets/alchemy/admin/pagination.scss +1 -8
  78. data/app/stylesheets/alchemy/admin/preview_window.scss +12 -1
  79. data/app/stylesheets/alchemy/admin/resource_info.scss +106 -3
  80. data/app/stylesheets/alchemy/admin/search.scss +1 -1
  81. data/app/stylesheets/alchemy/admin/selects.scss +58 -31
  82. data/app/stylesheets/alchemy/admin/shoelace.scss +32 -62
  83. data/app/stylesheets/alchemy/admin/sitemap.scss +1 -1
  84. data/app/stylesheets/alchemy/admin/tables.scss +3 -3
  85. data/app/stylesheets/alchemy/admin/tags.scss +18 -35
  86. data/app/stylesheets/alchemy/admin/toolbar.scss +0 -6
  87. data/app/stylesheets/alchemy/admin/typography.scss +2 -5
  88. data/app/stylesheets/alchemy/admin.scss +1 -1
  89. data/app/stylesheets/alchemy/dark-theme.scss +5 -0
  90. data/app/stylesheets/alchemy/light-theme.scss +6 -0
  91. data/app/stylesheets/alchemy/theme.scss +13 -0
  92. data/app/stylesheets/tinymce/skins/content/alchemy/content.scss +8 -8
  93. data/app/stylesheets/tinymce/skins/content/alchemy-dark/content.scss +70 -0
  94. data/app/stylesheets/tinymce/skins/ui/alchemy/skin.scss +28 -43
  95. data/app/stylesheets/tinymce/skins/ui/alchemy-dark/content.scss +1 -0
  96. data/app/stylesheets/tinymce/skins/ui/alchemy-dark/skin.scss +3784 -0
  97. data/app/views/alchemy/admin/attachments/_files_list.html.erb +20 -10
  98. data/app/views/alchemy/admin/attachments/assign.js.erb +4 -3
  99. data/app/views/alchemy/admin/attachments/show.html.erb +55 -43
  100. data/app/views/alchemy/admin/crop.html.erb +1 -1
  101. data/app/views/alchemy/admin/elements/_form.html.erb +9 -9
  102. data/app/views/alchemy/admin/elements/_header.html.erb +4 -1
  103. data/app/views/alchemy/admin/ingredients/_picture_fields.html.erb +1 -1
  104. data/app/views/alchemy/admin/pages/info.html.erb +1 -1
  105. data/app/views/alchemy/admin/partials/_search_form.html.erb +1 -0
  106. data/app/views/alchemy/admin/pictures/_archive.html.erb +12 -22
  107. data/app/views/alchemy/admin/pictures/_archive_overlay.html.erb +1 -6
  108. data/app/views/alchemy/admin/pictures/_form.html.erb +1 -1
  109. data/app/views/alchemy/admin/pictures/_infos.html.erb +21 -52
  110. data/app/views/alchemy/admin/pictures/_library_sidebar.html.erb +7 -0
  111. data/app/views/alchemy/admin/pictures/_picture.html.erb +14 -20
  112. data/app/views/alchemy/admin/pictures/_picture_to_assign.html.erb +20 -16
  113. data/app/views/alchemy/admin/pictures/_sorting_select.html.erb +13 -0
  114. data/app/views/alchemy/admin/pictures/_tag_list.html.erb +1 -1
  115. data/app/views/alchemy/admin/pictures/edit_multiple.html.erb +1 -6
  116. data/app/views/alchemy/admin/pictures/index.html.erb +3 -12
  117. data/app/views/alchemy/admin/pictures/show.html.erb +10 -5
  118. data/app/views/alchemy/admin/resources/_filter_bar.html.erb +5 -15
  119. data/app/views/alchemy/admin/resources/_resource_usage_info.html.erb +36 -0
  120. data/app/views/alchemy/admin/styleguide/index.html.erb +118 -66
  121. data/app/views/alchemy/base/error_notice.html.erb +1 -1
  122. data/app/views/alchemy/ingredients/_page_editor.html.erb +0 -1
  123. data/app/views/alchemy/ingredients/_richtext_editor.html.erb +0 -1
  124. data/app/views/alchemy/ingredients/_select_editor.html.erb +1 -2
  125. data/app/views/layouts/alchemy/admin.html.erb +22 -17
  126. data/config/locales/alchemy.en.yml +26 -8
  127. data/db/migrate/20250905140323_add_created_at_index_to_pictures_and_attachments.rb +14 -0
  128. data/lib/alchemy/configurations/format_matchers.rb +1 -1
  129. data/lib/alchemy/configurations/main.rb +7 -0
  130. data/lib/alchemy/configurations/page_cache.rb +19 -0
  131. data/lib/alchemy/engine.rb +16 -7
  132. data/lib/alchemy/install/tasks.rb +0 -12
  133. data/lib/alchemy/name_conversions.rb +6 -0
  134. data/lib/alchemy/tasks/tidy.rb +18 -0
  135. data/lib/alchemy/test_support/factories/picture_factory.rb +1 -0
  136. data/lib/alchemy/test_support/relatable_resource_examples.rb +58 -0
  137. data/lib/alchemy/tinymce.rb +0 -1
  138. data/lib/alchemy/version.rb +1 -1
  139. data/lib/alchemy.rb +2 -11
  140. data/lib/generators/alchemy/install/install_generator.rb +21 -10
  141. data/lib/generators/alchemy/install/templates/alchemy.rb.tt +10 -6
  142. data/lib/tasks/alchemy/tidy.rake +6 -0
  143. data/lib/tasks/alchemy/usage.rake +2 -0
  144. data/vendor/assets/stylesheets/tinymce/skins/content/dark/content.min.css +1 -0
  145. data/vendor/assets/stylesheets/tinymce/skins/content/default/content.min.css +1 -0
  146. data/vendor/assets/stylesheets/tinymce/skins/ui/oxide/skin.min.css +1 -0
  147. data/vendor/assets/stylesheets/tinymce/skins/ui/oxide-dark/content.min.css +1 -0
  148. data/vendor/assets/stylesheets/tinymce/skins/ui/oxide-dark/skin.min.css +1 -0
  149. data/vendor/javascript/clipboard.min.js +1 -1
  150. data/vendor/javascript/cropperjs.min.js +1 -1
  151. data/vendor/javascript/handlebars.min.js +3 -3
  152. data/vendor/javascript/jquery.min.js +1 -1
  153. data/vendor/javascript/select2.min.js +3 -3
  154. data/vendor/javascript/shoelace.min.js +92 -76
  155. data/vendor/javascript/sortable.min.js +2 -2
  156. data/vendor/javascript/tinymce.min.js +1 -1
  157. data/vendor/javascript/ungap-custom-elements.min.js +2 -2
  158. metadata +46 -32
  159. data/CHANGELOG.md +0 -2100
  160. data/CODE_OF_CONDUCT.md +0 -13
  161. data/CONTRIBUTING.md +0 -73
  162. data/Gemfile +0 -78
  163. data/Rakefile +0 -102
  164. data/SECURITY.md +0 -13
  165. data/alchemy_cms.gemspec +0 -97
  166. data/app/assets/builds/alchemy/custom-properties.css +0 -1
  167. data/app/helpers/alchemy/admin/elements_helper.rb +0 -25
  168. data/app/stylesheets/alchemy/custom-properties.css +0 -244
  169. data/bin/importmap +0 -4
  170. data/bin/rails +0 -9
  171. data/bin/rspec +0 -3
  172. data/bin/setup +0 -30
  173. data/bin/start +0 -17
  174. data/bun.lockb +0 -0
  175. data/bundles/remixicon.mjs +0 -153
  176. data/bundles/shoelace.js +0 -12
  177. data/bundles/tinymce.js +0 -22
  178. data/eslint.config.js +0 -18
  179. data/lib/alchemy/upgrader/.keep +0 -0
  180. data/lib/alchemy/upgrader/tasks/.keep +0 -0
  181. data/rollup.config.mjs +0 -108
  182. data/vitest.config.js +0 -21
@@ -60,16 +60,26 @@
60
60
  object: attachment,
61
61
  file_attribute: 'file' %>
62
62
  <% end %>
63
- <% table.with_action(:destroy, Alchemy.t(:delete_file)) do |attachment| %>
64
- <%= link_to_confirm_dialog render_icon(:minus),
65
- Alchemy.t(:confirm_to_delete_file),
66
- alchemy.admin_attachment_path(
67
- id: attachment,
68
- q: search_filter_params[:q],
69
- page: params[:page],
70
- per_page: params[:per_page]
71
- ),
72
- class: "icon_button" %>
63
+ <% table.with_action(:destroy) do |attachment| %>
64
+ <% if attachment.deletable? %>
65
+ <sl-tooltip content="<%= Alchemy.t(:delete_file) %>">
66
+ <%= link_to_confirm_dialog render_icon(:minus),
67
+ Alchemy.t(:confirm_to_delete_file),
68
+ alchemy.admin_attachment_path(
69
+ id: attachment,
70
+ q: search_filter_params[:q],
71
+ page: params[:page],
72
+ per_page: params[:per_page]
73
+ ),
74
+ class: "icon_button" %>
75
+ </sl-tooltip>
76
+ <% else %>
77
+ <sl-tooltip content="<%= Alchemy.t(:in_use) %>">
78
+ <button class="icon_button disabled">
79
+ <%= render_icon(:minus) %>
80
+ </button>
81
+ </sl-tooltip>
82
+ <% end %>
73
83
  <% end %>
74
84
  <% table.with_action(:edit, Alchemy.t(:rename_file)) do |attachment| %>
75
85
  <%= link_to_dialog render_icon(:edit),
@@ -5,10 +5,11 @@
5
5
  $form_field.parent().find("> .file_name").text("<%= @attachment.name %>")
6
6
  $form_field.parent().find("> .file_icon").html("<%= j render_icon(@attachment.icon_css_class) %>")
7
7
  $form_field.parent().find("> .remove_file_link").removeClass("hidden")
8
- Alchemy.closeCurrentDialog(function() {
9
- var elementEditor = $form_field[0].closest("alchemy-element-editor")
8
+ Alchemy.closeCurrentDialog(() => {
9
+ const field = $form_field[0]
10
+ const elementEditor = field.closest("alchemy-element-editor")
10
11
  if (elementEditor) {
11
- elementEditor.setDirty()
12
+ elementEditor.setDirty(field)
12
13
  }
13
14
  })
14
15
  })()
@@ -1,45 +1,57 @@
1
- <div class="resource_info">
2
- <div class="value">
3
- <label>
4
- <%= render_icon @attachment.icon_css_class %>
5
- </label>
6
- <p><%= @attachment.file_name %></p>
7
- </div>
8
- <div class="value with-icon">
9
- <label><%= Alchemy::Attachment.human_attribute_name(:url) %></label>
10
- <p><%= @attachment.url %></p>
11
- <alchemy-clipboard-button
12
- content="<%= @attachment.url %>"
13
- success-text="<%= Alchemy.t("Copied to clipboard") %>"
14
- class="icon_button--right"
15
- ></alchemy-clipboard-button>
16
- </div>
17
- <div class="value with-icon">
18
- <label><%= Alchemy::Attachment.human_attribute_name(:download_url) %></label>
19
- <p><%= @attachment.url(download: true) %></p>
20
- <alchemy-clipboard-button
21
- content="<%= @attachment.url(download: true) %>"
22
- success-text="<%= Alchemy.t("Copied to clipboard") %>"
23
- class="icon_button--right"
24
- ></alchemy-clipboard-button>
25
- </div>
26
- </div>
1
+ <div class="resource-details">
2
+ <aside>
3
+ <div class="resource_info">
4
+ <div class="value">
5
+ <label>
6
+ <%= render_icon @attachment.icon_css_class %>
7
+ </label>
8
+ <p><%= @attachment.file_name %></p>
9
+ </div>
10
+ <div class="value with-icon">
11
+ <label><%= Alchemy::Attachment.human_attribute_name(:url) %></label>
12
+ <p><%= @attachment.url %></p>
13
+ <alchemy-clipboard-button
14
+ content="<%= @attachment.url %>"
15
+ success-text="<%= Alchemy.t("Copied to clipboard") %>"
16
+ class="icon_button--right"
17
+ ></alchemy-clipboard-button>
18
+ </div>
19
+ <div class="value with-icon">
20
+ <label><%= Alchemy::Attachment.human_attribute_name(:download_url) %></label>
21
+ <p><%= @attachment.url(download: true) %></p>
22
+ <alchemy-clipboard-button
23
+ content="<%= @attachment.url(download: true) %>"
24
+ success-text="<%= Alchemy.t("Copied to clipboard") %>"
25
+ class="icon_button--right"
26
+ ></alchemy-clipboard-button>
27
+ </div>
28
+ </div>
27
29
 
28
- <% case @attachment.file_mime_type %>
29
- <% when *Alchemy::Filetypes::IMAGE_FILE_TYPES %>
30
- <div class="attachment_preview_container image-preview">
31
- <%= image_tag(@attachment.url, class: "full_width") %>
32
- </div>
33
- <% when *Alchemy::Filetypes::AUDIO_FILE_TYPES %>
34
- <div class="attachment_preview_container player-preview">
35
- <%= audio_tag(@attachment.url, preload: "none", controls: true, class: "full_width") %>
36
- </div>
37
- <% when *Alchemy::Filetypes::VIDEO_FILE_TYPES %>
38
- <div class="attachment_preview_container player-preview">
39
- <%= video_tag(@attachment.url, preload: "metadata", controls: true, class: "full_width") %>
30
+ <hr>
31
+
32
+ <%= render "alchemy/admin/resources/resource_usage_info",
33
+ assignments: @assignments,
34
+ resource_name: :attachment %>
35
+ </aside>
36
+
37
+ <%# We need to use the `tag` helper for the div to be really empty for
38
+ the CSS `:empty` selector to work in order to hide this div if there is no preview %>
39
+ <%= tag.div(class: "resource-preview") do %>
40
+ <% case @attachment.file_mime_type %>
41
+ <% when *Alchemy::Filetypes::IMAGE_FILE_TYPES %>
42
+ <div class="attachment_preview_container">
43
+ <%= image_tag(@attachment.url) %>
44
+ </div>
45
+ <% when *Alchemy::Filetypes::AUDIO_FILE_TYPES %>
46
+ <div class="attachment_preview_container">
47
+ <%= audio_tag(@attachment.url, preload: "metadata", controls: true) %>
48
+ </div>
49
+ <% when *Alchemy::Filetypes::VIDEO_FILE_TYPES %>
50
+ <div class="attachment_preview_container">
51
+ <%= video_tag(@attachment.url, preload: "metadata", controls: true) %>
52
+ </div>
53
+ <% when "application/pdf" %>
54
+ <iframe src="<%= @attachment.url %>" frameborder="0"></iframe>
55
+ <% end %>
56
+ <% end %>
40
57
  </div>
41
- <% when "application/pdf" %>
42
- <iframe src="<%= @attachment.url %>" frameborder=0 class="full-iframe">
43
- Your browser does not support frames.
44
- </iframe>
45
- <% end %>
@@ -12,7 +12,7 @@
12
12
  </div>
13
13
  <form>
14
14
  <%= button_tag Alchemy.t(:apply), type: 'submit' %>
15
- <%= button_tag Alchemy.t('Reset Imagemask'), class: 'reset_mask', type: 'reset' %>
15
+ <%= button_tag Alchemy.t('Reset Imagemask'), class: 'reset_mask secondary', type: 'reset' %>
16
16
  </form>
17
17
  </div>
18
18
  <% end %>
@@ -6,15 +6,15 @@
6
6
  <%= turbo_frame_tag @element do %>
7
7
  <%= alchemy_form_for [:admin, @element], remote: false do |form| %>
8
8
  <%= form.hidden_field :page_version_id %>
9
- <%= form.input :name,
10
- label: Alchemy.t(:element_of_type),
11
- collection: elements_for_select(@elements),
12
- prompt: Alchemy.t(:select_element),
13
- selected: (@elements.first if @elements.count == 1),
14
- input_html: {is: 'alchemy-select', autofocus: true} %>
15
- <% if @elements.count == 1 %>
16
- <%= form.hidden_field :name, value: @elements.first.name %>
17
- <% end %>
9
+ <div class="input">
10
+ <%= form.label(:name, class: "control-label required") do %>
11
+ <%= Alchemy.t(:element_of_type) %><abbr>*</abbr>
12
+ <% end %>
13
+ <%= render Alchemy::Admin::ElementSelect.new(
14
+ @elements,
15
+ field_name: form.field_name(:name)
16
+ ) %>
17
+ </div>
18
18
  <%= form.hidden_field :parent_element_id, value: @parent_element.try(:id) %>
19
19
  <%= form.submit Alchemy.t(:add) %>
20
20
  <%- end -%>
@@ -5,7 +5,10 @@
5
5
  <% elsif element.deprecated? %>
6
6
  <%= hint_with_tooltip element.deprecation_notice %>
7
7
  <% else %>
8
- <%= render_icon('draggable', style: false, class: 'element', fixed_width: false) %>
8
+ <span class="icon element">
9
+ <%= element.definition.icon_file %>
10
+ </span>
11
+ <%= render_icon(:draggable, style: false, fixed_width: false) %>
9
12
  <% end %>
10
13
  </span>
11
14
  <span class="element-title">
@@ -15,7 +15,7 @@
15
15
  then css_classes
16
16
  else
17
17
  css_classes.map do |klass|
18
- [Alchemy.t(klass, scope: "picture_ingredients.css_classes", default: picture_editor.css_class.camelcase), klass]
18
+ [Alchemy.t(klass, scope: "picture_ingredients.css_classes", default: ingredient.css_class&.camelcase), klass]
19
19
  end
20
20
  end %>
21
21
  <%= f.input :css_class, collection: css_classes_collection, include_blank: false %>
@@ -15,7 +15,7 @@
15
15
  </div>
16
16
  <div class="value">
17
17
  <label><%= Alchemy::Page.human_attribute_name(:urlname) %></label>
18
- <p><%= "/#{@page.urlname}" %></p>
18
+ <p><%= @page.url_path %></p>
19
19
  </div>
20
20
  <div class="value">
21
21
  <label><%= Alchemy.t(:page_status) %></label>
@@ -8,6 +8,7 @@
8
8
  <%= render_icon('search') %>
9
9
  </button>
10
10
  <%= f.search_field resource_handler.search_field_name,
11
+ value: params.dig(:q, resource_handler.search_field_name),
11
12
  class: 'search_input_field',
12
13
  placeholder: Alchemy.t(:search) %>
13
14
  <%= link_to render_icon(:times, size: '1x'), url,
@@ -1,23 +1,19 @@
1
- <div id="library_sidebar">
2
- <%= render 'filter_bar' if resource_has_filters %>
1
+ <%= render "library_sidebar" %>
3
2
 
4
- <% if Alchemy::Picture.tag_counts.any? %>
5
- <div class="tag-list with_filter_bar<%= ' filtered' if search_filter_params[:tagged_with].present? %>">
6
- <%= render 'tag_list' %>
7
- </div>
8
- <% end %>
9
- </div>
3
+ <% forwarded_params = search_filter_params.merge(
4
+ page: params[:page],
5
+ size: @size.presence
6
+ ).to_h %>
10
7
 
11
- <%= form_tag delete_multiple_admin_pictures_path, method: :delete do %>
8
+ <%= form_tag delete_multiple_admin_pictures_path(forwarded_params), method: :delete do %>
12
9
  <div class="selected_item_tools hidden">
13
10
  <h3><%= Alchemy.t(:edit_selected_pictures) %></h3>
14
11
  <%= link_to(
15
12
  render_icon(:edit, size: '1x') + Alchemy.t("Edit"),
16
- edit_multiple_admin_pictures_path,
13
+ edit_multiple_admin_pictures_path(forwarded_params),
17
14
  class: 'button with_icon',
18
15
  title: Alchemy.t('Edit multiple pictures'),
19
- id: 'edit_multiple_pictures',
20
- style: 'float: none'
16
+ id: 'edit_multiple_pictures'
21
17
  ) %>
22
18
  <%= button_tag render_icon("delete-bin-2", size: '1x') + Alchemy.t("Delete"),
23
19
  'data-turbo-confirm': Alchemy.t(:confirm_to_delete_images_from_server),
@@ -25,21 +21,15 @@
25
21
  &nbsp;<%= Alchemy.t(:or) %>&nbsp;
26
22
  <%= link_to(
27
23
  render_icon(:close) + Alchemy.t("Clear selection"),
28
- admin_pictures_path(
29
- q: search_filter_params[:q],
30
- tagged_with: search_filter_params[:tagged_with],
31
- size: @size,
32
- filter: search_filter_params[:filter]
33
- ),
34
- class: 'secondary button with_icon',
35
- style: 'float: none'
24
+ admin_pictures_path(forwarded_params),
25
+ class: 'secondary button with_icon'
36
26
  ) %>
37
27
  </div>
38
- <% if @pictures.blank? and @recent_pictures.blank? and search_filter_params[:q].blank? %>
28
+ <% if @pictures.blank? && @recent_pictures.blank? && search_filter_params[:q].blank? %>
39
29
  <%= render_message do %>
40
30
  <%= Alchemy.t(:no_images_in_archive) %>
41
31
  <% end %>
42
- <% elsif @pictures.blank? and @recent_pictures.blank? %>
32
+ <% elsif @pictures.blank? && @recent_pictures.blank? %>
43
33
  <%= render_message do %>
44
34
  <%= Alchemy.t(:no_search_results) %>
45
35
  <% end %>
@@ -2,12 +2,7 @@
2
2
  <%= render 'filter_and_size_bar' %>
3
3
  </div>
4
4
  <div id="assign_image_list" class="with_padding<%= search_filter_params[:tagged_with].present? ? ' filtered' : '' %>">
5
- <div id="library_sidebar">
6
- <%= render 'filter_bar' if resource_has_filters %>
7
- <div class="tag-list">
8
- <%= render 'tag_list' %>
9
- </div>
10
- </div>
5
+ <%= render "library_sidebar" %>
11
6
  <% if @pictures.empty? %>
12
7
  <%= render_message do %>
13
8
  <% if search_filter_params.empty? %>
@@ -1,5 +1,5 @@
1
1
  <%= turbo_frame_tag(@picture) do %>
2
- <%= alchemy_form_for [alchemy, :admin, @picture] do |f| %>
2
+ <%= alchemy_form_for [alchemy, :admin, @picture], class: "picture-form" do |f| %>
3
3
  <%= f.input :name %>
4
4
  <%= render "alchemy/admin/pictures/picture_description_field", f: f %>
5
5
  <%= render Alchemy::Admin::TagsAutocomplete.new(additional_class: "input") do %>
@@ -1,55 +1,24 @@
1
- <div class="resource_info">
2
- <div class="picture-file-infos">
3
- <div class="value">
4
- <label><%= Alchemy::Picture.human_attribute_name(:image_file_name) %></label>
5
- <p><%= @picture.image_file_name %></p>
6
- </div>
7
- <div class="value">
8
- <label><%= Alchemy::Picture.human_attribute_name(:image_file_dimensions) %></label>
9
- <p><%= @picture.image_file_dimensions %>px</p>
10
- </div>
11
- <div class="value">
12
- <label><%= Alchemy::Picture.human_attribute_name(:image_file_size) %></label>
13
- <p><%= number_to_human_size @picture.image_file_size %></p>
14
- </div>
1
+ <div class="resource_info file-infos">
2
+ <div class="value">
3
+ <label><%= Alchemy::Picture.human_attribute_name(:image_file_name) %></label>
4
+ <p><%= @picture.image_file_name %></p>
15
5
  </div>
16
- </div>
17
-
18
- <div class="picture-usage-info resource_info">
19
- <h3>
20
- <%= Alchemy.t(:this_picture_is_used_on_these_pages) %>
21
- </h3>
22
- <div id="pictures_page_list">
23
- <% if @assignments.any? %>
24
- <ul>
25
- <% @assignments.group_by(&:page).each do |page, picture_ingredients| %>
26
- <% if page %>
27
- <li>
28
- <h3>
29
- <%= render_icon 'file' %>
30
- <p><%= link_to page.name, edit_admin_page_path(page) %></p>
31
- </h3>
32
- <ul class="list">
33
- <% picture_ingredients.group_by(&:element).each do |element, picture_ingredients| %>
34
- <li class="<%= cycle('even', 'odd') %>">
35
- <% page_link = link_to element.display_name_with_preview_text,
36
- edit_admin_page_path(page, anchor: "element_#{element.id}") %>
37
- <% ingredients = picture_ingredients.map { |p| Alchemy::IngredientEditor.new(p).translated_role }.to_sentence %>
38
- <%= render_icon('draggable', style: false) %>
39
- <p>
40
- <%== Alchemy.t(:pictures_in_page, page: page_link, pictures: ingredients) %>
41
- </p>
42
- </li>
43
- <% end %>
44
- </ul>
45
- </li>
46
- <% end %>
47
- <% end %>
48
- </ul>
49
- <% else %>
50
- <%= render_message do %>
51
- <%= Alchemy.t(:picture_not_in_use_yet) %>
52
- <% end %>
53
- <% end %>
6
+ <div class="value">
7
+ <label><%= Alchemy::Picture.human_attribute_name(:image_file_dimensions) %></label>
8
+ <p><%= @picture.image_file_dimensions %>px</p>
9
+ </div>
10
+ <div class="value">
11
+ <label><%= Alchemy::Picture.human_attribute_name(:image_file_size) %></label>
12
+ <p><%= number_to_human_size @picture.image_file_size %></p>
13
+ </div>
14
+ <div class="value">
15
+ <label><%= Alchemy::Picture.human_attribute_name(:created_at) %></label>
16
+ <p><%= l(@picture.created_at, format: :"alchemy.default") %></p>
54
17
  </div>
55
18
  </div>
19
+
20
+ <hr>
21
+
22
+ <%= render "alchemy/admin/resources/resource_usage_info",
23
+ assignments: @assignments,
24
+ resource_name: :picture %>
@@ -0,0 +1,7 @@
1
+ <div id="library_sidebar">
2
+ <%= render "sorting_select" %>
3
+ <%= render "filter_bar" if resource_has_filters %>
4
+ <div class="tag-list">
5
+ <%= render "tag_list" %>
6
+ </div>
7
+ </div>
@@ -1,38 +1,32 @@
1
+ <% picture_path = alchemy.admin_picture_path(
2
+ picture,
3
+ search_filter_params.merge(
4
+ page: params[:page],
5
+ size: @size.presence
6
+ ).to_h
7
+ ) %>
8
+
1
9
  <div class="picture_thumbnail <%= @size %>" id="picture_<%= picture.id %>" name="<%= picture.name %>">
2
10
  <span class="picture_tool select">
3
11
  <%= check_box_tag "picture_ids[]", picture.id %>
4
12
  </span>
5
13
  <% if picture.deletable? && can?(:destroy, picture) %>
6
- <span class="picture_tool delete">
14
+ <div class="picture_tool delete">
7
15
  <sl-tooltip content="<%= Alchemy.t('Delete image') %>">
8
16
  <%= link_to_confirm_dialog(
9
17
  render_icon("delete-bin-2"),
10
18
  Alchemy.t(:confirm_to_delete_image_from_server),
11
- alchemy.admin_picture_path(
12
- id: picture,
13
- q: search_filter_params[:q],
14
- page: params[:page],
15
- tagged_with: search_filter_params[:tagged_with],
16
- size: @size,
17
- filter: search_filter_params[:filter]
18
- )
19
+ picture_path
19
20
  ) -%>
20
21
  </sl-tooltip>
21
- </span>
22
+ </div>
22
23
  <% end %>
23
24
  <% picture_url = picture.thumbnail_url(size: preview_size(@size)) %>
24
- <% image = image_tag(picture_url || "alchemy/missing-image.svg", alt: picture.name) %>
25
- <% if can?(:edit, picture) && picture_url %>
25
+ <% image = picture_url ? image_tag(picture_url, alt: picture.name) : '<alchemy-icon name="file-damage"></alchemy-icon>'.html_safe %>
26
+ <% if can?(:edit, picture) %>
26
27
  <%= link_to(
27
28
  image,
28
- alchemy.admin_picture_path(
29
- id: picture,
30
- q: search_filter_params[:q],
31
- page: params[:page],
32
- tagged_with: search_filter_params[:tagged_with],
33
- size: @size,
34
- filter: search_filter_params[:filter]
35
- ),
29
+ picture_path,
36
30
  class: 'thumbnail_background'
37
31
  ) %>
38
32
  <% else %>
@@ -1,20 +1,24 @@
1
1
  <div class="picture_thumbnail" name="<%= picture_to_assign.name %>" id="assignable_<%= picture_to_assign.id %>">
2
- <sl-tooltip content="<%= Alchemy.t(:assign_image) %>">
3
- <%= link_to(
4
- image_tag(
5
- picture_to_assign.thumbnail_url(size: preview_size(size)) || "alchemy/missing-image.svg",
6
- alt: picture_to_assign.name
7
- ),
8
- alchemy.assign_admin_picture_path(
9
- id: picture_to_assign.id,
10
- form_field_id: @form_field_id
11
- ),
12
- remote: true,
13
- onclick: '$(self).attr("href", "#").off("click"); return false',
14
- method: 'put',
15
- class: 'thumbnail_background'
16
- ) %>
17
- </sl-tooltip>
2
+ <% if picture_to_assign.image_file %>
3
+ <sl-tooltip content="<%= Alchemy.t(:assign_image) %>">
4
+ <%= link_to(
5
+ image_tag(
6
+ picture_to_assign.thumbnail_url(size: preview_size(size)),
7
+ alt: picture_to_assign.name
8
+ ),
9
+ alchemy.assign_admin_picture_path(
10
+ id: picture_to_assign.id,
11
+ form_field_id: @form_field_id
12
+ ),
13
+ remote: true,
14
+ onclick: '$(self).attr("href", "#").off("click"); return false',
15
+ method: 'put',
16
+ class: 'thumbnail_background'
17
+ ) %>
18
+ </sl-tooltip>
19
+ <% else %>
20
+ <alchemy-icon name="file-damage"></alchemy-icon>
21
+ <% end %>
18
22
  <div class="picture_name" title="<%= picture_to_assign.name %>">
19
23
  <%= picture_to_assign.name %>
20
24
  </div>
@@ -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,5 +1,5 @@
1
1
  <% if Alchemy::Picture.tag_counts.any? %>
2
- <h3><%= Alchemy.t("Filter by tag") %></h3>
2
+ <label><%= Alchemy.t("Filter by tag") %></label>
3
3
  <%= render Alchemy::Admin::ListFilter.new(".tag-list li") %>
4
4
  <ul>
5
5
  <%= render_tag_list('Alchemy::Picture') %>
@@ -1,9 +1,4 @@
1
- <%= form_tag update_multiple_admin_pictures_path do %>
2
- <%= hidden_field_tag :q, search_filter_params[:q] %>
3
- <%= hidden_field_tag :size, @size %>
4
- <%= hidden_field_tag :tagged_with, search_filter_params[:tagged_with] %>
5
- <%= hidden_field_tag :filter, search_filter_params[:filter] %>
6
-
1
+ <%= form_tag update_multiple_admin_pictures_path(search_filter_params.merge(size: @size.presence)) do %>
7
2
  <% @pictures.pluck(:id).each do |id| %>
8
3
  <%= hidden_field_tag "picture_ids[]", id %>
9
4
  <% end %>
@@ -17,10 +17,7 @@
17
17
  <%= link_to(
18
18
  render_icon('zoom-out'),
19
19
  alchemy.admin_pictures_path(
20
- size: "small",
21
- q: search_filter_params[:q],
22
- filter: search_filter_params[:filter],
23
- tagged_with: search_filter_params[:tagged_with]
20
+ search_filter_params.merge(size: "small").to_h
24
21
  ),
25
22
  class: "icon_button"
26
23
  ) %>
@@ -29,10 +26,7 @@
29
26
  <%= link_to(
30
27
  render_icon('search'),
31
28
  alchemy.admin_pictures_path(
32
- size: "medium",
33
- q: search_filter_params[:q],
34
- filter: search_filter_params[:filter],
35
- tagged_with: search_filter_params[:tagged_with]
29
+ search_filter_params.merge(size: "medium").to_h
36
30
  ),
37
31
  class: "icon_button"
38
32
  ) %>
@@ -41,10 +35,7 @@
41
35
  <%= link_to(
42
36
  render_icon('zoom-in'),
43
37
  alchemy.admin_pictures_path(
44
- size: "large",
45
- q: search_filter_params[:q],
46
- filter: search_filter_params[:filter],
47
- tagged_with: search_filter_params[:tagged_with]
38
+ search_filter_params.merge(size: "large").to_h
48
39
  ),
49
40
  class: "icon_button"
50
41
  ) %>
@@ -1,13 +1,17 @@
1
1
  <div class="zoomed-picture-background">
2
- <%= image_tag @picture.url || "alchemy/missing-image.svg" %>
2
+ <% if @picture.image_file %>
3
+ <%= image_tag @picture.url %>
4
+ <% else %>
5
+ <alchemy-icon name="file-damage"></alchemy-icon>
6
+ <% end %>
3
7
  </div>
4
8
 
5
9
  <div class="picture-overlay-navigation">
6
10
  <% if @previous %>
7
11
  <%= link_to alchemy.admin_picture_path(
8
- id: @previous,
12
+ id: :previous,
9
13
  q: search_filter_params[:q],
10
- page: params[:page],
14
+ page: @previous,
11
15
  tagged_with: search_filter_params[:tagged_with],
12
16
  size: @size,
13
17
  filter: search_filter_params[:filter]
@@ -19,9 +23,9 @@
19
23
  <% end %>
20
24
  <% if @next %>
21
25
  <%= link_to alchemy.admin_picture_path(
22
- id: @next,
26
+ id: :next,
23
27
  q: search_filter_params[:q],
24
- page: params[:page],
28
+ page: @next,
25
29
  tagged_with: search_filter_params[:tagged_with],
26
30
  size: @size,
27
31
  filter: search_filter_params[:filter]
@@ -35,6 +39,7 @@
35
39
 
36
40
  <div class="picture-details-overlay">
37
41
  <%= render 'form' %>
42
+ <hr>
38
43
  <%= render 'infos' %>
39
44
  </div>
40
45
 
@@ -1,17 +1,7 @@
1
1
  <div id="filter_bar">
2
- <% alchemy_filters.each do |filter| %>
3
- <%= render filter.input_component(search_filter_params, @query) %>
4
- <% end %>
2
+ <alchemy-auto-submit>
3
+ <% alchemy_filters.each do |filter| %>
4
+ <%= render filter.input_component(search_filter_params, @query) %>
5
+ <% end %>
6
+ </alchemy-auto-submit>
5
7
  </div>
6
-
7
- <script type="module">
8
- // Still using jQuery here, because select2 does emit the event from
9
- // the correct element.
10
- $('#filter_bar').on('change', function(event) {
11
- // We need to dispatch a submit event, so that Turbo that listens
12
- // to it submits the search form us.
13
- const submitEvent = new Event("submit", { bubbles: true, cancelable: true });
14
- event.target.form.dispatchEvent(submitEvent);
15
- return false;
16
- });
17
- </script>