alchemy_cms 5.0.0.rc2 → 5.1.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (107) hide show
  1. checksums.yaml +4 -4
  2. data/.github/PULL_REQUEST_TEMPLATE.md +1 -1
  3. data/.github/workflows/stale.yml +1 -1
  4. data/.gitignore +1 -0
  5. data/.travis.yml +48 -0
  6. data/CHANGELOG.md +64 -2
  7. data/CONTRIBUTING.md +2 -2
  8. data/Gemfile +3 -3
  9. data/README.md +2 -2
  10. data/alchemy_cms.gemspec +3 -2
  11. data/app/assets/images/alchemy/missing-image.svg +1 -0
  12. data/app/assets/javascripts/alchemy/alchemy.element_editors.js.coffee +1 -4
  13. data/app/assets/javascripts/alchemy/alchemy.preview.js.coffee +0 -3
  14. data/app/assets/javascripts/alchemy/alchemy.preview_window.js.coffee +29 -4
  15. data/app/assets/stylesheets/alchemy/_variables.scss +3 -0
  16. data/app/assets/stylesheets/alchemy/archive.scss +23 -17
  17. data/app/assets/stylesheets/alchemy/buttons.scss +26 -15
  18. data/app/assets/stylesheets/alchemy/errors.scss +1 -1
  19. data/app/assets/stylesheets/alchemy/navigation.scss +7 -10
  20. data/app/assets/stylesheets/alchemy/pagination.scss +1 -1
  21. data/app/assets/stylesheets/alchemy/search.scss +12 -2
  22. data/app/assets/stylesheets/alchemy/selects.scss +4 -2
  23. data/app/assets/stylesheets/alchemy/tables.scss +38 -9
  24. data/app/assets/stylesheets/alchemy/tags.scss +19 -31
  25. data/app/controllers/alchemy/admin/pages_controller.rb +59 -9
  26. data/app/controllers/alchemy/admin/pictures_controller.rb +13 -6
  27. data/app/controllers/alchemy/admin/resources_controller.rb +3 -3
  28. data/app/controllers/alchemy/pages_controller.rb +49 -14
  29. data/app/helpers/alchemy/admin/base_helper.rb +0 -42
  30. data/app/helpers/alchemy/admin/navigation_helper.rb +2 -1
  31. data/app/helpers/alchemy/url_helper.rb +2 -2
  32. data/app/models/alchemy/attachment.rb +21 -1
  33. data/app/models/alchemy/attachment/url.rb +40 -0
  34. data/app/models/alchemy/essence_file.rb +1 -1
  35. data/app/models/alchemy/essence_picture.rb +4 -4
  36. data/app/models/alchemy/essence_picture_view.rb +10 -4
  37. data/app/models/alchemy/page.rb +24 -1
  38. data/app/models/alchemy/page/page_natures.rb +2 -0
  39. data/app/models/alchemy/page/url_path.rb +8 -6
  40. data/app/models/alchemy/picture.rb +58 -2
  41. data/app/models/alchemy/picture/calculations.rb +55 -0
  42. data/app/models/alchemy/picture/transformations.rb +5 -49
  43. data/app/models/alchemy/picture/url.rb +28 -75
  44. data/app/models/alchemy/picture_thumb.rb +57 -0
  45. data/app/models/alchemy/picture_thumb/create.rb +39 -0
  46. data/app/models/alchemy/picture_thumb/signature.rb +23 -0
  47. data/app/models/alchemy/picture_thumb/uid.rb +22 -0
  48. data/app/models/alchemy/picture_variant.rb +114 -0
  49. data/app/models/alchemy/site/layout.rb +30 -2
  50. data/app/views/alchemy/admin/attachments/show.html.erb +8 -8
  51. data/app/views/alchemy/admin/dashboard/index.html.erb +13 -16
  52. data/app/views/alchemy/admin/essence_pictures/crop.html.erb +1 -1
  53. data/app/views/alchemy/admin/essence_pictures/edit.html.erb +2 -2
  54. data/app/views/alchemy/admin/layoutpages/edit.html.erb +4 -6
  55. data/app/views/alchemy/admin/pages/_create_language_form.html.erb +19 -29
  56. data/app/views/alchemy/admin/pages/_form.html.erb +4 -6
  57. data/app/views/alchemy/admin/pages/_new_page_form.html.erb +12 -2
  58. data/app/views/alchemy/admin/pages/_page_layout_filter.html.erb +29 -0
  59. data/app/views/alchemy/admin/pages/_table.html.erb +27 -0
  60. data/app/views/alchemy/admin/pages/_table_row.html.erb +107 -0
  61. data/app/views/alchemy/admin/pages/_toolbar.html.erb +77 -0
  62. data/app/views/alchemy/admin/pages/edit.html.erb +9 -1
  63. data/app/views/alchemy/admin/pages/index.html.erb +41 -74
  64. data/app/views/alchemy/admin/pages/list/_table.html.erb +31 -0
  65. data/app/views/alchemy/admin/pages/unlock.js.erb +2 -2
  66. data/app/views/alchemy/admin/pages/update.js.erb +19 -10
  67. data/app/views/alchemy/admin/partials/_remote_search_form.html.erb +14 -13
  68. data/app/views/alchemy/admin/partials/_search_form.html.erb +8 -8
  69. data/app/views/alchemy/admin/pictures/_archive.html.erb +1 -1
  70. data/app/views/alchemy/admin/pictures/_form.html.erb +1 -1
  71. data/app/views/alchemy/admin/pictures/_picture.html.erb +3 -3
  72. data/app/views/alchemy/admin/pictures/_picture_to_assign.html.erb +1 -1
  73. data/app/views/alchemy/admin/pictures/edit_multiple.html.erb +1 -1
  74. data/app/views/alchemy/admin/pictures/index.html.erb +1 -1
  75. data/app/views/alchemy/admin/pictures/show.html.erb +3 -3
  76. data/app/views/alchemy/admin/resources/_filter_bar.html.erb +13 -11
  77. data/app/views/alchemy/admin/resources/_per_page_select.html.erb +3 -3
  78. data/app/views/alchemy/admin/resources/index.html.erb +24 -22
  79. data/app/views/alchemy/admin/tags/index.html.erb +14 -15
  80. data/app/views/alchemy/base/500.html.erb +11 -13
  81. data/app/views/alchemy/essences/_essence_file_view.html.erb +4 -4
  82. data/config/alchemy/config.yml +15 -11
  83. data/config/alchemy/modules.yml +12 -12
  84. data/config/locales/alchemy.en.yml +6 -4
  85. data/config/routes.rb +1 -1
  86. data/db/migrate/20200617110713_create_alchemy_picture_thumbs.rb +22 -0
  87. data/db/migrate/20200907111332_remove_tri_state_booleans.rb +33 -0
  88. data/lib/alchemy.rb +66 -0
  89. data/lib/alchemy/admin/preview_url.rb +2 -0
  90. data/lib/alchemy/auth_accessors.rb +12 -5
  91. data/lib/alchemy/config.rb +1 -3
  92. data/lib/alchemy/engine.rb +7 -6
  93. data/lib/alchemy/modules.rb +11 -1
  94. data/lib/alchemy/permissions.rb +1 -0
  95. data/lib/alchemy/test_support/factories/picture_factory.rb +0 -1
  96. data/lib/alchemy/test_support/factories/picture_thumb_factory.rb +12 -0
  97. data/lib/alchemy/version.rb +1 -1
  98. data/lib/alchemy_cms.rb +2 -3
  99. data/lib/generators/alchemy/install/files/alchemy.en.yml +2 -2
  100. data/lib/generators/alchemy/install/templates/dragonfly.rb.tt +5 -5
  101. data/lib/tasks/alchemy/thumbnails.rake +37 -0
  102. metadata +41 -13
  103. data/.github/workflows/ci.yml +0 -134
  104. data/.github/workflows/greetings.yml +0 -13
  105. data/app/controllers/concerns/alchemy/locale_redirects.rb +0 -40
  106. data/app/controllers/concerns/alchemy/page_redirects.rb +0 -68
  107. data/lib/alchemy/userstamp.rb +0 -12
@@ -11,16 +11,18 @@ module Alchemy
11
11
  before_action :load_resource,
12
12
  only: [:show, :edit, :update, :destroy, :info]
13
13
 
14
+ before_action :set_size, only: [:index, :show, :edit_multiple]
15
+
14
16
  authorize_resource class: Alchemy::Picture
15
17
 
16
18
  def index
17
- @size = params[:size].present? ? params[:size] : "medium"
18
19
  @query = Picture.ransack(search_filter_params[:q])
19
20
  @pictures = Picture.search_by(
20
21
  search_filter_params,
21
22
  @query,
22
23
  items_per_page,
23
24
  )
25
+ @pictures = @pictures.includes(:thumbs)
24
26
 
25
27
  if in_overlay?
26
28
  archive_overlay
@@ -115,7 +117,7 @@ module Alchemy
115
117
 
116
118
  def items_per_page
117
119
  if in_overlay?
118
- case params[:size]
120
+ case @size
119
121
  when "small" then 25
120
122
  when "large" then 4
121
123
  else
@@ -124,19 +126,24 @@ module Alchemy
124
126
  else
125
127
  cookies[:alchemy_pictures_per_page] = params[:per_page] ||
126
128
  cookies[:alchemy_pictures_per_page] ||
127
- pictures_per_page_for_size(params[:size])
129
+ pictures_per_page_for_size
128
130
  end
129
131
  end
130
132
 
131
133
  def items_per_page_options
132
- per_page = pictures_per_page_for_size(@size)
134
+ per_page = pictures_per_page_for_size
133
135
  [per_page, per_page * 2, per_page * 4]
134
136
  end
135
137
 
136
138
  private
137
139
 
138
- def pictures_per_page_for_size(size)
139
- case size
140
+ def set_size
141
+ @size = params[:size] || session[:alchemy_pictures_size] || "medium"
142
+ session[:alchemy_pictures_size] = @size
143
+ end
144
+
145
+ def pictures_per_page_for_size
146
+ case @size
140
147
  when "small" then 60
141
148
  when "large" then 12
142
149
  else
@@ -148,15 +148,15 @@ module Alchemy
148
148
 
149
149
  def common_search_filter_includes
150
150
  [
151
- {q: [
151
+ { q: [
152
152
  resource_handler.search_field_name,
153
153
  :s,
154
- ]},
154
+ ] },
155
155
  :tagged_with,
156
156
  :filter,
157
157
  :page,
158
158
  :per_page,
159
- ].freeze
159
+ ]
160
160
  end
161
161
 
162
162
  def items_per_page
@@ -13,7 +13,10 @@ module Alchemy
13
13
 
14
14
  # Redirecting concerns. Order is important here!
15
15
  include SiteRedirects
16
- include LocaleRedirects
16
+
17
+ before_action :enforce_no_locale,
18
+ if: :locale_prefix_not_allowed?,
19
+ only: [:index, :show]
17
20
 
18
21
  before_action :load_index_page, only: [:index]
19
22
  before_action :load_page, only: [:show]
@@ -21,11 +24,13 @@ module Alchemy
21
24
  # Legacy page redirects need to run after the page was loaded and before we render 404.
22
25
  include LegacyPageRedirects
23
26
 
24
- # From here on, we need a +@page+ to work with!
25
- before_action :page_not_found!, if: -> { @page.blank? }, only: [:index, :show]
27
+ # From here on, we need a published +@page+ to work with!
28
+ before_action :page_not_found!, unless: -> { @page&.public? }, only: [:index, :show]
26
29
 
27
- # Page redirects need to run after the page was loaded and we're sure to have a +@page+ set.
28
- include PageRedirects
30
+ # Page redirects need to run after the page was loaded and we're sure to have a public +@page+ set.
31
+ before_action :enforce_locale,
32
+ if: :locale_prefix_missing?,
33
+ only: [:index, :show]
29
34
 
30
35
  # We only need to set the +@root_page+ if we are sure that no more redirects happen.
31
36
  before_action :set_root_page, only: [:index, :show]
@@ -66,12 +71,8 @@ module Alchemy
66
71
  # descendant it finds. If no public page can be found it renders a 404 error.
67
72
  #
68
73
  def show
69
- if redirect_url.present?
70
- redirect_permanently_to redirect_url
71
- else
72
- authorize! :show, @page
73
- render_page if render_fresh_page?
74
- end
74
+ authorize! :show, @page
75
+ render_page if render_fresh_page?
75
76
  end
76
77
 
77
78
  # Renders a search engine compatible xml sitemap.
@@ -84,13 +85,25 @@ module Alchemy
84
85
 
85
86
  private
86
87
 
88
+ # Redirects to requested action without locale prefixed
89
+ def enforce_no_locale
90
+ redirect_permanently_to additional_params.merge(locale: nil)
91
+ end
92
+
93
+ # Is the requested locale allowed?
94
+ #
95
+ # If Alchemy is not in multi language mode or the requested locale is the default locale,
96
+ # then we want to redirect to a non prefixed url.
97
+ #
98
+ def locale_prefix_not_allowed?
99
+ params[:locale].present? && !multi_language? ||
100
+ params[:locale].presence == ::I18n.default_locale.to_s
101
+ end
102
+
87
103
  # == Loads index page
88
104
  #
89
105
  # Loads the current public language root page.
90
106
  #
91
- # If the root page is not public it redirects to the first published child.
92
- # This can be configured via +redirect_to_public_child+ [default: true]
93
- #
94
107
  # If no index page and no admin users are present we show the "Welcome to Alchemy" page.
95
108
  #
96
109
  def load_index_page
@@ -116,6 +129,28 @@ module Alchemy
116
129
  )
117
130
  end
118
131
 
132
+ def enforce_locale
133
+ redirect_permanently_to page_locale_redirect_url(locale: Language.current.code)
134
+ end
135
+
136
+ def locale_prefix_missing?
137
+ multi_language? && params[:locale].blank? && !default_locale?
138
+ end
139
+
140
+ def default_locale?
141
+ Language.current.code.to_sym == ::I18n.default_locale.to_sym
142
+ end
143
+
144
+ # Page url with or without locale while keeping all additional params
145
+ def page_locale_redirect_url(options = {})
146
+ options = {
147
+ locale: prefix_locale? ? @page.language_code : nil,
148
+ urlname: @page.urlname,
149
+ }.merge(options)
150
+
151
+ alchemy.show_page_path additional_params.merge(options)
152
+ end
153
+
119
154
  # Redirects to given url with 301 status
120
155
  def redirect_permanently_to(url)
121
156
  redirect_to url, status: :moved_permanently
@@ -272,48 +272,6 @@ module Alchemy
272
272
  end
273
273
  end
274
274
 
275
- # Renders the toolbar shown on top of the records.
276
- #
277
- # == Example
278
- #
279
- # <% label_title = Alchemy.t("Create #{resource_name}", default: Alchemy.t('Create')) %>
280
- # <% toolbar(
281
- # buttons: [
282
- # {
283
- # icon: :plus,
284
- # label: label_title,
285
- # url: new_resource_path,
286
- # title: label_title,
287
- # hotkey: 'alt+n',
288
- # dialog_options: {
289
- # title: label_title,
290
- # size: "430x400"
291
- # },
292
- # if_permitted_to: [:create, resource_model]
293
- # }
294
- # ]
295
- # ) %>
296
- #
297
- # @option options [Array] :buttons ([])
298
- # Pass an Array with button options. They will be passed to {#toolbar_button} helper.
299
- # @option options [Boolean] :search (true)
300
- # Show searchfield.
301
- #
302
- def toolbar(options = {})
303
- defaults = {
304
- buttons: [],
305
- search: true,
306
- }
307
- options = defaults.merge(options)
308
- content_for(:toolbar) do
309
- content = <<-CONTENT.strip_heredoc
310
- #{options[:buttons].map { |button_options| toolbar_button(button_options) }.join}
311
- #{render("alchemy/admin/partials/search_form", url: options[:search_url]) if options[:search]}
312
- CONTENT
313
- content.html_safe
314
- end
315
- end
316
-
317
275
  # (internal) Used by upload form
318
276
  def new_asset_path_with_session_information(asset_type)
319
277
  session_key = Rails.application.config.session_options[:key]
@@ -175,7 +175,8 @@ module Alchemy
175
175
  # Returns true if the given entry's controller is current controller
176
176
  #
177
177
  def is_entry_controller_active?(entry)
178
- entry["controller"].gsub(/\A\//, "") == params[:controller]
178
+ entry["controller"].gsub(/\A\//, "") == params[:controller] ||
179
+ entry.fetch("nested_controllers", []).include?(params[:controller])
179
180
  end
180
181
 
181
182
  # Returns true if the given entry's action is current controllers action
@@ -26,12 +26,12 @@ module Alchemy
26
26
 
27
27
  # Returns the path for downloading an alchemy attachment
28
28
  def download_alchemy_attachment_path(attachment)
29
- alchemy.download_attachment_path(attachment, attachment.urlname)
29
+ alchemy.download_attachment_path(attachment, attachment.slug)
30
30
  end
31
31
 
32
32
  # Returns the url for downloading an alchemy attachment
33
33
  def download_alchemy_attachment_url(attachment)
34
- alchemy.download_attachment_url(attachment, attachment.urlname)
34
+ alchemy.download_attachment_url(attachment, attachment.slug)
35
35
  end
36
36
 
37
37
  # Returns the full url containing host, page and anchor for the given element
@@ -37,6 +37,20 @@ module Alchemy
37
37
 
38
38
  # We need to define this method here to have it available in the validations below.
39
39
  class << self
40
+ # The class used to generate URLs for attachments
41
+ #
42
+ # @see Alchemy::Attachment::Url
43
+ def url_class
44
+ @_url_class ||= Alchemy::Attachment::Url
45
+ end
46
+
47
+ # Set a different attachment url class
48
+ #
49
+ # @see Alchemy::Attachment::Url
50
+ def url_class=(klass)
51
+ @_url_class = klass
52
+ end
53
+
40
54
  def searchable_alchemy_resource_attributes
41
55
  %w(name file_name)
42
56
  end
@@ -76,8 +90,14 @@ module Alchemy
76
90
  }
77
91
  end
78
92
 
93
+ def url(options = {})
94
+ if file
95
+ self.class.url_class.new(self).call(options)
96
+ end
97
+ end
98
+
79
99
  # An url save filename without format suffix
80
- def urlname
100
+ def slug
81
101
  CGI.escape(file_name.gsub(/\.#{extension}$/, "").tr(".", " "))
82
102
  end
83
103
 
@@ -0,0 +1,40 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Alchemy
4
+ class Attachment < BaseRecord
5
+ # The class representing an URL to an attachment
6
+ #
7
+ # Set a different one
8
+ #
9
+ # Alchemy::Attachment.url_class = MyRemoteUrlClass
10
+ #
11
+ class Url
12
+ def initialize(attachment)
13
+ @attachment = attachment
14
+ end
15
+
16
+ # The attachment url
17
+ #
18
+ # @param [Hash] options
19
+ # @option options [Symbol] :download return a URL for downloading the attachment
20
+ # @option options [Symbol] :name The filename
21
+ # @option options [Symbol] :format The file extension
22
+ #
23
+ # @return [String]
24
+ #
25
+ def call(options = {})
26
+ if options.delete(:download)
27
+ routes.download_attachment_path(@attachment, options)
28
+ else
29
+ routes.show_attachment_path(@attachment, options)
30
+ end
31
+ end
32
+
33
+ private
34
+
35
+ def routes
36
+ Alchemy::Engine.routes.url_helpers
37
+ end
38
+ end
39
+ end
40
+ end
@@ -23,7 +23,7 @@ module Alchemy
23
23
 
24
24
  routes.download_attachment_path(
25
25
  id: attachment.id,
26
- name: attachment.urlname,
26
+ name: attachment.slug,
27
27
  format: attachment.suffix,
28
28
  )
29
29
  end
@@ -62,7 +62,7 @@ module Alchemy
62
62
  def picture_url(options = {})
63
63
  return if picture.nil?
64
64
 
65
- picture.url picture_url_options.merge(options)
65
+ picture.url(picture_url_options.merge(options)) || "missing-image.png"
66
66
  end
67
67
 
68
68
  # Picture rendering options
@@ -103,7 +103,7 @@ module Alchemy
103
103
  format: picture.image_file_format,
104
104
  }
105
105
 
106
- picture.url(options)
106
+ picture.url(options) || "alchemy/missing-image.svg"
107
107
  end
108
108
 
109
109
  # The name of the picture used as preview text in element editor views.
@@ -140,10 +140,10 @@ module Alchemy
140
140
  # Show image cropping link for content
141
141
  def allow_image_cropping?
142
142
  content && content.settings[:crop] && picture &&
143
- picture.can_be_cropped_to(
143
+ picture.can_be_cropped_to?(
144
144
  content.settings[:size],
145
145
  content.settings[:upsample],
146
- )
146
+ ) && !!picture.image_file
147
147
  end
148
148
 
149
149
  def crop_values_present?
@@ -44,18 +44,20 @@ module Alchemy
44
44
  end
45
45
  end
46
46
 
47
- private
48
-
49
47
  def caption
50
48
  return unless show_caption?
51
49
 
52
50
  @_caption ||= content_tag(:figcaption, essence.caption)
53
51
  end
54
52
 
53
+ def src
54
+ essence.picture_url(options.except(*DEFAULT_OPTIONS.keys))
55
+ end
56
+
55
57
  def img_tag
56
58
  @_img_tag ||= image_tag(
57
- essence.picture_url(options.except(*DEFAULT_OPTIONS.keys)), {
58
- alt: essence.alt_tag.presence,
59
+ src, {
60
+ alt: alt_text,
59
61
  title: essence.title.presence,
60
62
  class: caption ? nil : essence.css_class.presence,
61
63
  srcset: srcset.join(", ").presence,
@@ -79,5 +81,9 @@ module Alchemy
79
81
  width.present? ? "#{url} #{width}w" : "#{url} #{height}h"
80
82
  end
81
83
  end
84
+
85
+ def alt_text
86
+ essence.alt_tag.presence || html_options.delete(:alt) || essence.picture.name&.humanize
87
+ end
82
88
  end
83
89
  end
@@ -149,6 +149,29 @@ module Alchemy
149
149
  # Class methods
150
150
  #
151
151
  class << self
152
+ # The url_path class
153
+ # @see Alchemy::Page::UrlPath
154
+ def url_path_class
155
+ @_url_path_class ||= Alchemy::Page::UrlPath
156
+ end
157
+
158
+ # Set a custom url path class
159
+ #
160
+ # # config/initializers/alchemy.rb
161
+ # Alchemy::Page.url_path_class = MyPageUrlPathClass
162
+ #
163
+ def url_path_class=(klass)
164
+ @_url_path_class = klass
165
+ end
166
+
167
+ def alchemy_resource_filters
168
+ %w[published not_public restricted]
169
+ end
170
+
171
+ def searchable_alchemy_resource_attributes
172
+ %w[name urlname title]
173
+ end
174
+
152
175
  # Used to store the current page previewed in the edit page template.
153
176
  #
154
177
  def current_preview=(page)
@@ -298,7 +321,7 @@ module Alchemy
298
321
  #
299
322
  # @see Alchemy::Page::UrlPath#call
300
323
  def url_path
301
- Alchemy::Page::UrlPath.new(self).call
324
+ self.class.url_path_class.new(self).call
302
325
  end
303
326
 
304
327
  # The page's view partial is dependent from its page layout
@@ -17,6 +17,8 @@ module Alchemy
17
17
  definition["taggable"] == true
18
18
  end
19
19
 
20
+ deprecate :taggable?, deprecator: Alchemy::Deprecation
21
+
20
22
  def rootpage?
21
23
  !new_record? && parent_id.blank?
22
24
  end