trestle 0.9.5 → 0.9.7

Sign up to get free protection for your applications and to get access to all the features.
Files changed (133) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/rspec.yml +28 -0
  3. data/.gitignore +3 -0
  4. data/CONTRIBUTING.md +38 -0
  5. data/Gemfile +9 -2
  6. data/README.md +6 -1
  7. data/app/assets/bundle/trestle/bundle.css +7 -7
  8. data/app/assets/bundle/trestle/bundle.js +40 -38
  9. data/app/assets/bundle/trestle/fa-brands-400.eot +0 -0
  10. data/app/assets/bundle/trestle/fa-brands-400.svg +41 -41
  11. data/app/assets/bundle/trestle/fa-brands-400.ttf +0 -0
  12. data/app/assets/bundle/trestle/fa-brands-400.woff +0 -0
  13. data/app/assets/bundle/trestle/fa-brands-400.woff2 +0 -0
  14. data/app/assets/bundle/trestle/fa-regular-400.eot +0 -0
  15. data/app/assets/bundle/trestle/fa-regular-400.svg +2 -2
  16. data/app/assets/bundle/trestle/fa-regular-400.ttf +0 -0
  17. data/app/assets/bundle/trestle/fa-regular-400.woff +0 -0
  18. data/app/assets/bundle/trestle/fa-regular-400.woff2 +0 -0
  19. data/app/assets/bundle/trestle/fa-solid-900.eot +0 -0
  20. data/app/assets/bundle/trestle/fa-solid-900.svg +13 -7
  21. data/app/assets/bundle/trestle/fa-solid-900.ttf +0 -0
  22. data/app/assets/bundle/trestle/fa-solid-900.woff +0 -0
  23. data/app/assets/bundle/trestle/fa-solid-900.woff2 +0 -0
  24. data/app/assets/bundle/trestle/flatpickr/ar-dz.js +1 -0
  25. data/app/assets/bundle/trestle/flatpickr/ar.js +1 -1
  26. data/app/assets/bundle/trestle/flatpickr/at.js +1 -1
  27. data/app/assets/bundle/trestle/flatpickr/az.js +1 -1
  28. data/app/assets/bundle/trestle/flatpickr/be.js +1 -1
  29. data/app/assets/bundle/trestle/flatpickr/bg.js +1 -1
  30. data/app/assets/bundle/trestle/flatpickr/bn.js +1 -1
  31. data/app/assets/bundle/trestle/flatpickr/bs.js +1 -1
  32. data/app/assets/bundle/trestle/flatpickr/cat.js +1 -1
  33. data/app/assets/bundle/trestle/flatpickr/ckb.js +1 -0
  34. data/app/assets/bundle/trestle/flatpickr/cs.js +1 -1
  35. data/app/assets/bundle/trestle/flatpickr/cy.js +1 -1
  36. data/app/assets/bundle/trestle/flatpickr/da.js +1 -1
  37. data/app/assets/bundle/trestle/flatpickr/de.js +1 -1
  38. data/app/assets/bundle/trestle/flatpickr/default.js +1 -1
  39. data/app/assets/bundle/trestle/flatpickr/eo.js +1 -1
  40. data/app/assets/bundle/trestle/flatpickr/es.js +1 -1
  41. data/app/assets/bundle/trestle/flatpickr/et.js +1 -1
  42. data/app/assets/bundle/trestle/flatpickr/fa.js +1 -1
  43. data/app/assets/bundle/trestle/flatpickr/fi.js +1 -1
  44. data/app/assets/bundle/trestle/flatpickr/fo.js +1 -1
  45. data/app/assets/bundle/trestle/flatpickr/fr.js +1 -1
  46. data/app/assets/bundle/trestle/flatpickr/ga.js +1 -1
  47. data/app/assets/bundle/trestle/flatpickr/gr.js +1 -1
  48. data/app/assets/bundle/trestle/flatpickr/he.js +1 -1
  49. data/app/assets/bundle/trestle/flatpickr/hi.js +1 -1
  50. data/app/assets/bundle/trestle/flatpickr/hr.js +1 -1
  51. data/app/assets/bundle/trestle/flatpickr/hu.js +1 -1
  52. data/app/assets/bundle/trestle/flatpickr/hy.js +1 -0
  53. data/app/assets/bundle/trestle/flatpickr/id.js +1 -1
  54. data/app/assets/bundle/trestle/flatpickr/is.js +1 -1
  55. data/app/assets/bundle/trestle/flatpickr/it.js +1 -1
  56. data/app/assets/bundle/trestle/flatpickr/ja.js +1 -1
  57. data/app/assets/bundle/trestle/flatpickr/ka.js +1 -1
  58. data/app/assets/bundle/trestle/flatpickr/km.js +1 -1
  59. data/app/assets/bundle/trestle/flatpickr/ko.js +1 -1
  60. data/app/assets/bundle/trestle/flatpickr/kz.js +1 -1
  61. data/app/assets/bundle/trestle/flatpickr/lt.js +1 -1
  62. data/app/assets/bundle/trestle/flatpickr/lv.js +1 -1
  63. data/app/assets/bundle/trestle/flatpickr/mk.js +1 -1
  64. data/app/assets/bundle/trestle/flatpickr/mn.js +1 -1
  65. data/app/assets/bundle/trestle/flatpickr/ms.js +1 -1
  66. data/app/assets/bundle/trestle/flatpickr/my.js +1 -1
  67. data/app/assets/bundle/trestle/flatpickr/nl.js +1 -1
  68. data/app/assets/bundle/trestle/flatpickr/nn.js +1 -0
  69. data/app/assets/bundle/trestle/flatpickr/no.js +1 -1
  70. data/app/assets/bundle/trestle/flatpickr/pa.js +1 -1
  71. data/app/assets/bundle/trestle/flatpickr/pl.js +1 -1
  72. data/app/assets/bundle/trestle/flatpickr/pt.js +1 -1
  73. data/app/assets/bundle/trestle/flatpickr/ro.js +1 -1
  74. data/app/assets/bundle/trestle/flatpickr/ru.js +1 -1
  75. data/app/assets/bundle/trestle/flatpickr/si.js +1 -1
  76. data/app/assets/bundle/trestle/flatpickr/sk.js +1 -1
  77. data/app/assets/bundle/trestle/flatpickr/sl.js +1 -1
  78. data/app/assets/bundle/trestle/flatpickr/sq.js +1 -1
  79. data/app/assets/bundle/trestle/flatpickr/sr-cyr.js +1 -1
  80. data/app/assets/bundle/trestle/flatpickr/sr.js +1 -1
  81. data/app/assets/bundle/trestle/flatpickr/sv.js +1 -1
  82. data/app/assets/bundle/trestle/flatpickr/th.js +1 -1
  83. data/app/assets/bundle/trestle/flatpickr/tr.js +1 -1
  84. data/app/assets/bundle/trestle/flatpickr/uk.js +1 -1
  85. data/app/assets/bundle/trestle/flatpickr/uz.js +1 -1
  86. data/app/assets/bundle/trestle/flatpickr/uz_latn.js +1 -1
  87. data/app/assets/bundle/trestle/flatpickr/vn.js +1 -1
  88. data/app/assets/bundle/trestle/flatpickr/zh-tw.js +1 -1
  89. data/app/assets/bundle/trestle/flatpickr/zh.js +1 -1
  90. data/app/assets/javascripts/trestle/i18n.js.erb +1 -1
  91. data/app/controllers/trestle/dashboard_controller.rb +2 -2
  92. data/app/helpers/trestle/avatar_helper.rb +6 -2
  93. data/app/helpers/trestle/container_helper.rb +12 -4
  94. data/app/helpers/trestle/params_helper.rb +4 -1
  95. data/app/helpers/trestle/url_helper.rb +1 -11
  96. data/app/views/trestle/flash/_alert.html.erb +3 -1
  97. data/app/views/trestle/flash/_debug.html.erb +13 -8
  98. data/app/views/trestle/flash/_flash.html.erb +1 -1
  99. data/config/routes.rb +1 -1
  100. data/frontend/css/components/_alerts.scss +31 -0
  101. data/frontend/css/components/_avatar.scss +14 -0
  102. data/frontend/css/components/_buttons.scss +0 -1
  103. data/frontend/css/components/_datepicker.scss +1 -0
  104. data/frontend/css/components/_modal.scss +2 -0
  105. data/frontend/css/components/_tags.scss +2 -0
  106. data/frontend/css/components/_timestamp.scss +1 -1
  107. data/frontend/css/layout/_sidebar.scss +3 -2
  108. data/frontend/css/variables/_bootstrap.scss +1 -0
  109. data/frontend/css/variables/_trestle.scss +20 -0
  110. data/frontend/js/components/dialog.js +8 -3
  111. data/frontend/theme/trestle/theme/_defaults.scss +4 -4
  112. data/lib/generators/trestle/install/templates/_theme.scss +1 -1
  113. data/lib/generators/trestle/install/templates/trestle.rb.erb +8 -2
  114. data/lib/trestle/adapters/active_record_adapter.rb +16 -0
  115. data/lib/trestle/configurable.rb +6 -0
  116. data/lib/trestle/configuration.rb +4 -5
  117. data/lib/trestle/form/automatic.rb +2 -0
  118. data/lib/trestle/form/builder.rb +4 -4
  119. data/lib/trestle/form/field.rb +1 -1
  120. data/lib/trestle/form/renderer.rb +4 -1
  121. data/lib/trestle/lazy.rb +56 -0
  122. data/lib/trestle/navigation/item.rb +1 -1
  123. data/lib/trestle/registry.rb +61 -0
  124. data/lib/trestle/reloader.rb +1 -1
  125. data/lib/trestle/table/automatic.rb +7 -0
  126. data/lib/trestle/table/column.rb +5 -1
  127. data/lib/trestle/version.rb +1 -1
  128. data/lib/trestle.rb +30 -14
  129. data/package.json +20 -23
  130. data/webpack.config.js +36 -48
  131. data/yarn.lock +2817 -4562
  132. metadata +11 -4
  133. data/.travis.yml +0 -33
@@ -2,6 +2,6 @@
2
2
  <%= render "trestle/flash/alert", html_class: "alert-success", icon: icon("alert-icon far fa-check-circle"), alert: normalize_flash_alert(flash[:message]) %>
3
3
  <% elsif flash[:error] -%>
4
4
  <%= render layout: "trestle/flash/alert", locals: { html_class: "alert-danger", icon: icon("alert-icon far fa-times-circle"), alert: normalize_flash_alert(flash[:error]) } do %>
5
- <%= render "trestle/flash/debug" if debug_form_errors? %>
5
+ <%= render "trestle/flash/debug", errors: instance.errors if debug_form_errors? %>
6
6
  <% end %>
7
7
  <% end -%>
data/config/routes.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  Trestle::Engine.routes.draw do
2
- Trestle.admins.each do |name, admin|
2
+ Trestle.registry.each do |admin|
3
3
  instance_eval(&admin.routes)
4
4
  end
5
5
 
@@ -32,13 +32,18 @@
32
32
 
33
33
  .alert-icon {
34
34
  margin-right: $alert-padding-x;
35
+
35
36
  font-size: 40px;
37
+ width: 40px;
38
+
39
+ text-align: center;
36
40
 
37
41
  @include media-breakpoint-down(sm) {
38
42
  margin-left: $alert-padding-x * 0.25;
39
43
  margin-right: $alert-padding-x * 0.75;
40
44
 
41
45
  font-size: 32px;
46
+ width: 32px;
42
47
  }
43
48
  }
44
49
 
@@ -74,10 +79,16 @@
74
79
  border-color: $border;
75
80
 
76
81
  a {
82
+ color: $color;
83
+
77
84
  &:hover, &:focus {
78
85
  color: mix($color, $background, 80%);
79
86
  }
80
87
  }
88
+
89
+ code {
90
+ color: $color;
91
+ }
81
92
  }
82
93
 
83
94
  .alert-success {
@@ -88,6 +99,26 @@
88
99
  @include alert-variant($alert-danger-bg, $alert-danger-border, $alert-danger-color);
89
100
  }
90
101
 
102
+ .alert-info {
103
+ @include alert-variant($alert-info-bg, $alert-info-border, $alert-info-color);
104
+ }
105
+
106
+ .alert-warning {
107
+ @include alert-variant($alert-warning-bg, $alert-warning-border, $alert-warning-color);
108
+ }
109
+
110
+ .alert-light {
111
+ @include alert-variant($alert-light-bg, $alert-light-border, $alert-light-color);
112
+ }
113
+
114
+ .alert-dark {
115
+ @include alert-variant($alert-dark-bg, $alert-dark-border, $alert-dark-color);
116
+ }
117
+
118
+ .alert-primary {
119
+ @include alert-variant($alert-primary-bg, $alert-primary-border, $alert-primary-color);
120
+ }
121
+
91
122
 
92
123
  // Debug errors
93
124
 
@@ -41,3 +41,17 @@
41
41
  font-weight: 500;
42
42
  }
43
43
  }
44
+
45
+ .avatar-sm {
46
+ width: 24px !important;
47
+ height: 24px !important;
48
+ }
49
+
50
+ .avatar-lg {
51
+ width: 64px !important;
52
+ height: 64px !important;
53
+
54
+ .avatar-fallback {
55
+ font-size: 1.5rem;
56
+ }
57
+ }
@@ -1,5 +1,4 @@
1
1
  .btn {
2
- border-width: 0;
3
2
  box-shadow: none;
4
3
 
5
4
  &.loading {
@@ -13,6 +13,7 @@
13
13
 
14
14
  &:hover, &:focus {
15
15
  color: #aaa;
16
+ text-decoration: none;
16
17
  }
17
18
 
18
19
  &::before {
@@ -24,6 +24,8 @@
24
24
  }
25
25
 
26
26
  &.background {
27
+ overflow-y: hidden;
28
+
27
29
  .modal-dialog {
28
30
  transform: scale(0.9);
29
31
  opacity: 0.75;
@@ -16,6 +16,8 @@
16
16
  }
17
17
 
18
18
  a.tag {
19
+ font-weight: inherit;
20
+
19
21
  &:hover, &:focus {
20
22
  text-decoration: none;
21
23
  color: $tag-hover-color;
@@ -4,7 +4,7 @@
4
4
 
5
5
  small {
6
6
  display: block;
7
- color: #666;
7
+ opacity: 0.75;
8
8
  }
9
9
 
10
10
  &.timestamp-inline,
@@ -16,11 +16,13 @@
16
16
  height: $header-height;
17
17
 
18
18
  display: flex;
19
- align-items: center;
19
+ align-items: stretch;
20
20
 
21
21
  .navbar-toggler {
22
22
  outline: none;
23
23
 
24
+ align-self: center;
25
+
24
26
  margin-left: $grid-gutter-width / 2;
25
27
  margin-right: $grid-gutter-width / 2;
26
28
 
@@ -44,7 +46,6 @@
44
46
  display: flex;
45
47
  align-items: center;
46
48
 
47
- height: 100%;
48
49
  padding: 10px 15px;
49
50
 
50
51
  font-size: 1.4rem;
@@ -7,6 +7,7 @@ $theme-colors: (
7
7
  "warning": #f3bd71,
8
8
  "danger": #e17572,
9
9
  "light": #bbbbbb,
10
+ "dark": #444444,
10
11
 
11
12
  // Backwards compatibility
12
13
  "default": #bbbbbb
@@ -101,6 +101,26 @@ $alert-danger-bg: #de7471;
101
101
  $alert-danger-color: white;
102
102
  $alert-danger-border: #d85956;
103
103
 
104
+ $alert-info-bg: darken(theme-color("info"), 10%);
105
+ $alert-info-color: white;
106
+ $alert-info-border: darken($alert-info-bg, 10%);
107
+
108
+ $alert-warning-bg: darken(theme-color("warning"), 5%);
109
+ $alert-warning-color: white;
110
+ $alert-warning-border: darken($alert-warning-bg, 5%);
111
+
112
+ $alert-light-bg: theme-color("light");
113
+ $alert-light-color: $body-color;
114
+ $alert-light-border: darken($alert-light-bg, 5%);
115
+
116
+ $alert-dark-bg: theme-color("dark");
117
+ $alert-dark-color: white;
118
+ $alert-dark-border: darken($alert-dark-bg, 10%);
119
+
120
+ $alert-primary-bg: darken(theme-color("primary"), 5%);
121
+ $alert-primary-color: white;
122
+ $alert-primary-border: darken($alert-primary-bg, 10%);
123
+
104
124
 
105
125
  // Modals
106
126
 
@@ -20,14 +20,19 @@ function createElement () {
20
20
  focus: false
21
21
  })
22
22
 
23
- // Remove dialog elements once hidden
24
23
  $el.on('hidden.bs.modal', function () {
24
+ // Restore modal-open class on body if background modals still visible
25
+ if ($el.prevAll('.modal.show').length) {
26
+ $(document.body).addClass('modal-open')
27
+ }
28
+
29
+ // Remove dialog elements once hidden
25
30
  $el.remove()
26
31
  })
27
32
 
28
33
  // Restore the next visible modal to the foreground
29
34
  $el.on('hide.bs.modal', function () {
30
- $el.prevAll('.modal.in').last().removeClass('background')
35
+ $el.prevAll('.modal.show').last().removeClass('background')
31
36
  })
32
37
 
33
38
  // Set X-Trestle-Dialog header on AJAX requests initiated from the dialog
@@ -120,7 +125,7 @@ export default class Dialog {
120
125
  this.$el.modal('show')
121
126
 
122
127
  // Background any existing visible modals
123
- this.$el.prevAll('.modal.in').addClass('background')
128
+ this.$el.prevAll('.modal.show').addClass('background')
124
129
  }
125
130
 
126
131
  hide () {
@@ -8,7 +8,7 @@ $theme-colors: (
8
8
  $theme-color-interval: 8% !default;
9
9
 
10
10
  // The yiq lightness value that determines when the lightness of color changes from "dark" to "light". Acceptable values are between 0 and 255.
11
- $yiq-contrasted-threshold: 150 !default;
11
+ $yiq-contrasted-threshold: 175 !default;
12
12
 
13
13
  // Customize the light and dark text colors for use in our YIQ color contrast function.
14
14
  $yiq-text-dark: #212529 !default;
@@ -115,9 +115,9 @@ $badge-focus-width: $input-btn-focus-width !default;
115
115
 
116
116
  // Alerts
117
117
 
118
- $alert-bg-level: -10 !default;
119
- $alert-border-level: -9 !default;
120
- $alert-color-level: 6 !default;
118
+ $alert-bg-level: 2 !default;
119
+ $alert-border-level: 3 !default;
120
+ $alert-color-level: -12 !default;
121
121
 
122
122
 
123
123
  // Progress bars
@@ -9,5 +9,5 @@
9
9
  //
10
10
  // $theme-colors: (
11
11
  // "primary": #337ab7,
12
- // "secondary": #b3b53d
12
+ // "secondary": #719dc3
13
13
  // );
@@ -15,7 +15,7 @@ Trestle.configure do |config|
15
15
  #
16
16
  # config.site_logo_small = "logo-small.png"
17
17
 
18
- # Speficy a favicon to be used within the admin.
18
+ # Specify a favicon to be used within the admin.
19
19
  #
20
20
  # config.favicon = "favicon.ico"
21
21
 
@@ -70,6 +70,7 @@ Trestle.configure do |config|
70
70
  # Specify helper modules to expose to the admin.
71
71
  #
72
72
  # config.helper :all
73
+ # config.helper -> { CustomHelper }
73
74
 
74
75
  # Register callbacks to run before, after or around all Trestle actions.
75
76
  #
@@ -103,6 +104,11 @@ Trestle.configure do |config|
103
104
  #
104
105
  # config.persistent_params << :query
105
106
 
107
+ # List of methods to try calling on an instance when displayed by the `display` helper.
108
+ # Defaults to [:display_name, :full_name, :name, :title, :username, :login, :email].
109
+ #
110
+ # config.display_methods.unshift(:admin_label)
111
+
106
112
  # Customize the default adapter class used by all admin resources.
107
113
  # See the documentation on Trestle::Adapters::Adapter for details on
108
114
  # the adapter methods that can be customized.
@@ -118,7 +124,7 @@ Trestle.configure do |config|
118
124
  # def render; end
119
125
  # end
120
126
  #
121
- # config.form_field :custom, CustomFormField
127
+ # config.form_field :custom, -> { CustomFormField }
122
128
 
123
129
  # == Debugging Options
124
130
  #
@@ -60,6 +60,8 @@ module Trestle
60
60
  Attribute::Association.new(column.name, class: -> { reflection.klass }, name: name, polymorphic: reflection.polymorphic?, type_column: reflection.foreign_type)
61
61
  elsif column.name.end_with?("_type") && (name = column.name.sub(/_type$/, '')) && (reflection = model.reflections[name])
62
62
  # Ignore type columns for polymorphic associations
63
+ elsif enum_column?(column)
64
+ Attribute.new(column.name, :enum, values: enum_values(column))
63
65
  else
64
66
  Attribute.new(column.name, column.type, array_column?(column) ? { array: true } : {})
65
67
  end
@@ -81,6 +83,20 @@ module Trestle
81
83
  def array_column?(column)
82
84
  column.respond_to?(:array?) && column.array?
83
85
  end
86
+
87
+ def enum_column?(column)
88
+ model.defined_enums.key?(column.name)
89
+ end
90
+
91
+ def enum_values(column)
92
+ model.defined_enums[column.name].map { |key, value|
93
+ [value, enum_human_name(column, key)]
94
+ }
95
+ end
96
+
97
+ def enum_human_name(column, value)
98
+ human_attribute_name([column.name.pluralize, value].join("."), default: value.humanize)
99
+ end
84
100
  end
85
101
  end
86
102
  end
@@ -15,6 +15,12 @@ module Trestle
15
15
  options.fetch(name) {
16
16
  if defaults.key?(name)
17
17
  value = defaults[name]
18
+
19
+ # Avoid references to the same instance of a default value
20
+ if value.respond_to?(:dup) && !value.is_a?(Proc)
21
+ value = value.dup
22
+ end
23
+
18
24
  assign(name, value)
19
25
  end
20
26
  }
@@ -54,18 +54,17 @@ module Trestle
54
54
  menus << Navigation::Block.new(&block)
55
55
  end
56
56
 
57
-
58
57
  ## Extension Options
59
58
 
60
59
  # [Internal] List of helper modules to include in all Trestle controllers
61
- option :helpers, []
60
+ option :helpers, Lazy::List.new
62
61
 
63
62
  # [Internal] Container module for block-defined helpers
64
63
  option :helper_module, Module.new
65
64
 
66
65
  # Register global helpers available to all Trestle admins
67
66
  def helper(*helpers, &block)
68
- self.helpers += helpers
67
+ self.helpers << helpers
69
68
  self.helper_module.module_eval(&block) if block_given?
70
69
  end
71
70
 
@@ -82,8 +81,8 @@ module Trestle
82
81
  option :default_adapter, Adapters.compose(Adapters::ActiveRecordAdapter, Adapters::DraperAdapter)
83
82
 
84
83
  # Register a custom form field class
85
- def form_field(name, klass)
86
- Form::Builder.register(name, klass)
84
+ def form_field(name, field)
85
+ Form::Builder.register(name, field)
87
86
  end
88
87
 
89
88
  # [Internal] List of registered hooks
@@ -32,6 +32,8 @@ module Trestle
32
32
  datetime_field attribute.name
33
33
  when :boolean
34
34
  check_box attribute.name
35
+ when :enum
36
+ collection_radio_buttons attribute.name, attribute.options[:values] || [], :first, :last
35
37
  when :json, :jsonb
36
38
  value = instance.public_send(attribute.name)
37
39
  text_area attribute.name, value: value.try(:to_json)
@@ -9,7 +9,7 @@ module Trestle
9
9
  undef_method :display
10
10
 
11
11
  cattr_accessor :fields
12
- self.fields = {}
12
+ self.fields = Lazy::Hash.new
13
13
 
14
14
  def errors(name)
15
15
  if object.respond_to?(:errors) && object.errors.respond_to?(:[])
@@ -19,14 +19,14 @@ module Trestle
19
19
  end
20
20
  end
21
21
 
22
- def self.register(name, klass)
22
+ def self.register(name, field)
23
23
  rename_existing_helper_method(name)
24
- self.fields[name] = klass
24
+ self.fields[name] = field
25
25
  end
26
26
 
27
27
  protected
28
28
  def respond_to_missing?(name, include_all=false)
29
- self.class.fields.has_key?(name) || super
29
+ self.class.fields.key?(name) || super
30
30
  end
31
31
 
32
32
  def method_missing(name, *args, &block)
@@ -45,7 +45,7 @@ module Trestle
45
45
  end
46
46
 
47
47
  def readonly?
48
- options[:readonly] || admin.readonly?
48
+ options[:readonly] || admin.try(:readonly?)
49
49
  end
50
50
 
51
51
  def normalize_options!
@@ -4,6 +4,9 @@ require "action_view/helpers"
4
4
  module Trestle
5
5
  class Form
6
6
  class Renderer
7
+ def self.ruby2_keywords(*)
8
+ end unless respond_to?(:ruby2_keywords, true)
9
+
7
10
  include ::ActionView::Context
8
11
  include ::ActionView::Helpers::CaptureHelper
9
12
 
@@ -40,7 +43,7 @@ module Trestle
40
43
  concat(result)
41
44
  end
42
45
 
43
- def method_missing(name, *args, &block)
46
+ ruby2_keywords def method_missing(name, *args, &block)
44
47
  target = @form.respond_to?(name) ? @form : @template
45
48
 
46
49
  if block_given? && !RAW_BLOCK_HELPERS.include?(name)
@@ -0,0 +1,56 @@
1
+ module Trestle
2
+ module Lazy
3
+ module Constantize
4
+ def constantize(value)
5
+ case value
6
+ when String
7
+ value.safe_constantize
8
+ when Proc
9
+ value.call
10
+ else
11
+ value
12
+ end
13
+ end
14
+ end
15
+
16
+ class List
17
+ include Enumerable
18
+ include Constantize
19
+
20
+ def initialize(*items)
21
+ @list = items
22
+ end
23
+
24
+ def each(&block)
25
+ @list.each do |item|
26
+ yield constantize(item)
27
+ end
28
+ end
29
+
30
+ def <<(items)
31
+ @list += Array(items)
32
+ end
33
+ end
34
+
35
+ class Hash
36
+ include Enumerable
37
+ include Constantize
38
+
39
+ delegate :[]=, :key?, to: :@hash
40
+
41
+ def initialize
42
+ @hash = {}
43
+ end
44
+
45
+ def each(&block)
46
+ @hash.each do |key, value|
47
+ yield key, constantize(value)
48
+ end
49
+ end
50
+
51
+ def [](key)
52
+ constantize(@hash[key])
53
+ end
54
+ end
55
+ end
56
+ end
@@ -17,7 +17,7 @@ module Trestle
17
17
  end
18
18
 
19
19
  def <=>(other)
20
- priority <=> other.priority
20
+ [priority, name] <=> [other.priority, other.name]
21
21
  end
22
22
 
23
23
  def priority
@@ -0,0 +1,61 @@
1
+ module Trestle
2
+ class Registry
3
+ include Enumerable
4
+
5
+ # The admins hash is left exposed for backwards compatibility
6
+ attr_reader :admins
7
+
8
+ def initialize
9
+ reset!
10
+ end
11
+
12
+ def each(&block)
13
+ @admins.values.sort_by(&:admin_name).each(&block)
14
+ end
15
+
16
+ def reset!
17
+ @admins = {}
18
+ @models = {}
19
+ end
20
+
21
+ def empty?
22
+ none?
23
+ end
24
+
25
+ def register(admin, register_model: true)
26
+ @admins[admin.admin_name] = admin
27
+
28
+ if register_model && register_admin_for_model_loookup?(admin)
29
+ @models[admin.model.name] ||= admin
30
+ end
31
+
32
+ admin
33
+ end
34
+
35
+ def lookup_admin(admin)
36
+ # Given object is already an admin class
37
+ return admin if admin.is_a?(Class) && admin < Trestle::Admin
38
+
39
+ @admins[admin.to_s]
40
+ end
41
+ alias lookup lookup_admin
42
+
43
+ def lookup_model(model)
44
+ # Lookup each class in the model's ancestor chain
45
+ while model
46
+ admin = @models[model.name]
47
+ return admin if admin
48
+
49
+ model = model.superclass
50
+ end
51
+
52
+ # No admin found
53
+ nil
54
+ end
55
+
56
+ private
57
+ def register_admin_for_model_loookup?(admin)
58
+ admin.respond_to?(:model) && !(admin.respond_to?(:singular?) && admin.singular?)
59
+ end
60
+ end
61
+ end
@@ -26,7 +26,7 @@ module Trestle
26
26
  end
27
27
 
28
28
  def clear
29
- Trestle.admins = {}
29
+ Trestle.registry.reset!
30
30
  end
31
31
 
32
32
  def load_paths
@@ -14,6 +14,13 @@ module Trestle
14
14
  case attribute.type
15
15
  when :association
16
16
  Column.new(attribute.association_name, sort: false)
17
+ when :enum
18
+ Column.new(attribute.name, align: :center, sort: attribute.name) do |instance|
19
+ if value = instance.public_send(attribute.name)
20
+ full_attribute_name = [attribute.name.to_s.pluralize, value].join(".")
21
+ status_tag(admin.human_attribute_name(full_attribute_name, default: value.humanize))
22
+ end
23
+ end
17
24
  else
18
25
  Column.new(attribute.name, link: index.zero?, align: (:center if [:datetime, :boolean].include?(attribute.type)))
19
26
  end
@@ -108,7 +108,7 @@ module Trestle
108
108
 
109
109
  def column_value(instance)
110
110
  if @column.block
111
- if defined?(Haml) && Haml::Helpers.block_is_haml?(@column.block)
111
+ if block_is_legacy_haml?
112
112
  # In order for table column blocks to work properly within Haml templates,
113
113
  # the _hamlout local variable needs to be defined in the scope of the block,
114
114
  # so that the Haml version of the capture method is used. Because we
@@ -131,6 +131,10 @@ module Trestle
131
131
  instance.send(@column.field)
132
132
  end
133
133
  end
134
+
135
+ def block_is_legacy_haml?
136
+ defined?(Haml) && Haml::Helpers.respond_to?(:block_is_haml?) && Haml::Helpers.block_is_haml?(@column.block)
137
+ end
134
138
  end
135
139
  end
136
140
  end
@@ -1,3 +1,3 @@
1
1
  module Trestle
2
- VERSION = "0.9.5"
2
+ VERSION = "0.9.7"
3
3
  end