trestle 0.9.0 → 0.9.8

Sign up to get free protection for your applications and to get access to all the features.
Files changed (235) 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 +12 -5
  6. data/README.md +9 -2
  7. data/app/assets/bundle/trestle/bundle.css +9 -9
  8. data/app/assets/bundle/trestle/bundle.js +50 -48
  9. data/app/assets/bundle/trestle/fa-brands-400.eot +0 -0
  10. data/app/assets/bundle/trestle/fa-brands-400.svg +818 -597
  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 +93 -95
  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 +1209 -842
  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 -0
  86. data/app/assets/bundle/trestle/flatpickr/uz_latn.js +1 -0
  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/admin.js +1 -0
  91. data/app/assets/javascripts/trestle/i18n.js.erb +8 -0
  92. data/{lib → app/controllers/concerns}/trestle/controller/breadcrumbs.rb +1 -1
  93. data/app/controllers/concerns/trestle/controller/title.rb +20 -0
  94. data/app/controllers/concerns/trestle/controller/toolbars.rb +30 -0
  95. data/app/controllers/concerns/trestle/resource/controller/actions.rb +133 -0
  96. data/app/controllers/concerns/trestle/resource/controller/data_methods.rb +52 -0
  97. data/app/controllers/concerns/trestle/resource/controller/redirection.rb +23 -0
  98. data/app/controllers/concerns/trestle/resource/controller/toolbar.rb +11 -0
  99. data/app/controllers/trestle/admin_controller.rb +31 -0
  100. data/app/controllers/trestle/application_controller.rb +12 -0
  101. data/app/controllers/trestle/dashboard_controller.rb +2 -2
  102. data/app/controllers/trestle/resource_controller.rb +6 -0
  103. data/app/helpers/trestle/avatar_helper.rb +6 -2
  104. data/app/helpers/trestle/container_helper.rb +12 -4
  105. data/app/helpers/trestle/grid_helper.rb +51 -7
  106. data/app/helpers/trestle/hook_helper.rb +1 -25
  107. data/app/helpers/trestle/i18n_helper.rb +8 -15
  108. data/app/helpers/trestle/layout_helper.rb +20 -0
  109. data/app/helpers/trestle/navigation_helper.rb +23 -0
  110. data/app/helpers/trestle/params_helper.rb +4 -1
  111. data/app/helpers/trestle/table_helper.rb +22 -3
  112. data/app/helpers/trestle/title_helper.rb +11 -0
  113. data/app/helpers/trestle/toolbars_helper.rb +0 -11
  114. data/app/helpers/trestle/url_helper.rb +1 -11
  115. data/app/views/layouts/trestle/admin.html.erb +6 -6
  116. data/app/views/trestle/application/_dialog.html.erb +1 -1
  117. data/app/views/trestle/application/_header.html.erb +1 -3
  118. data/app/views/trestle/flash/_alert.html.erb +3 -1
  119. data/app/views/trestle/flash/_debug.html.erb +13 -8
  120. data/app/views/trestle/flash/_flash.html.erb +1 -1
  121. data/app/views/trestle/resource/_scopes.html.erb +17 -8
  122. data/app/views/trestle/resource/edit.html.erb +2 -2
  123. data/app/views/trestle/resource/index.html.erb +3 -3
  124. data/app/views/trestle/resource/new.html.erb +1 -1
  125. data/app/views/trestle/resource/show.html.erb +2 -2
  126. data/app/views/trestle/shared/_sidebar.html.erb +5 -6
  127. data/app/views/trestle/table/_pagination.html.erb +1 -1
  128. data/config/locales/de.rb +18 -0
  129. data/config/locales/de.yml +101 -0
  130. data/config/locales/vi.rb +18 -0
  131. data/config/locales/vi.yml +101 -0
  132. data/config/routes.rb +1 -1
  133. data/frontend/css/components/_alerts.scss +31 -0
  134. data/frontend/css/components/_avatar.scss +14 -0
  135. data/frontend/css/components/_buttons.scss +0 -1
  136. data/frontend/css/components/_datepicker.scss +1 -0
  137. data/frontend/css/components/_fields.scss +6 -0
  138. data/frontend/css/components/_modal.scss +2 -0
  139. data/frontend/css/components/_scopes.scss +66 -21
  140. data/frontend/css/components/_table.scss +20 -0
  141. data/frontend/css/components/_tags.scss +2 -0
  142. data/frontend/css/components/_timestamp.scss +1 -1
  143. data/frontend/css/layout/_content.scss +5 -2
  144. data/frontend/css/layout/_sidebar.scss +9 -3
  145. data/frontend/css/variables/_bootstrap.scss +1 -0
  146. data/frontend/css/variables/_trestle.scss +25 -5
  147. data/frontend/js/components/confirmation.js +4 -2
  148. data/frontend/js/components/dialog.js +15 -4
  149. data/frontend/js/components/form.js +9 -0
  150. data/frontend/js/components/navigation.js +63 -0
  151. data/frontend/js/components/pagination.js +51 -0
  152. data/frontend/js/components/sidebar.js +8 -11
  153. data/frontend/js/components/table.js +33 -2
  154. data/frontend/js/components/tabs.js +3 -3
  155. data/frontend/js/core/events.js +7 -3
  156. data/frontend/js/index.js +1 -0
  157. data/frontend/theme/trestle/theme/_defaults.scss +4 -4
  158. data/frontend/theme/trestle/theme/bootstrap/_buttons.scss +3 -0
  159. data/lib/generators/trestle/install/install_generator.rb +6 -2
  160. data/lib/generators/trestle/install/templates/_custom.css +8 -0
  161. data/lib/generators/trestle/install/templates/_theme.scss +1 -1
  162. data/lib/generators/trestle/install/templates/trestle.rb.erb +8 -2
  163. data/lib/generators/trestle/resource/resource_generator.rb +15 -1
  164. data/lib/generators/trestle/resource/templates/admin.rb.erb +15 -8
  165. data/lib/trestle/adapters/active_record_adapter.rb +17 -1
  166. data/lib/trestle/adapters.rb +1 -1
  167. data/lib/trestle/admin/builder.rb +7 -3
  168. data/lib/trestle/admin.rb +18 -8
  169. data/lib/trestle/configurable.rb +16 -0
  170. data/lib/trestle/configuration.rb +7 -8
  171. data/lib/trestle/debug_errors.rb +24 -0
  172. data/lib/trestle/engine.rb +3 -10
  173. data/lib/trestle/evaluation_context.rb +11 -3
  174. data/lib/trestle/form/automatic.rb +4 -2
  175. data/lib/trestle/form/builder.rb +6 -7
  176. data/lib/trestle/form/field.rb +44 -17
  177. data/lib/trestle/form/fields/check_box_helpers.rb +1 -1
  178. data/lib/trestle/form/fields/collection_select.rb +1 -1
  179. data/lib/trestle/form/fields/date_picker.rb +9 -3
  180. data/lib/trestle/form/fields/date_select.rb +1 -1
  181. data/lib/trestle/form/fields/datetime_select.rb +1 -1
  182. data/lib/trestle/form/fields/file_field.rb +1 -1
  183. data/lib/trestle/form/fields/form_group.rb +18 -5
  184. data/lib/trestle/form/fields/grouped_collection_select.rb +1 -1
  185. data/lib/trestle/form/fields/range_field.rb +2 -4
  186. data/lib/trestle/form/fields/select.rb +2 -3
  187. data/lib/trestle/form/fields/static_field.rb +7 -3
  188. data/lib/trestle/form/fields/tag_select.rb +3 -1
  189. data/lib/trestle/form/fields/time_select.rb +1 -1
  190. data/lib/trestle/form/fields/time_zone_select.rb +1 -1
  191. data/lib/trestle/form/fields.rb +6 -38
  192. data/lib/trestle/form/renderer.rb +11 -4
  193. data/lib/trestle/form.rb +5 -7
  194. data/lib/trestle/hook/helpers.rb +33 -0
  195. data/lib/trestle/hook/set.rb +32 -0
  196. data/lib/trestle/hook.rb +5 -31
  197. data/lib/trestle/lazy.rb +56 -0
  198. data/lib/trestle/navigation/group.rb +5 -2
  199. data/lib/trestle/navigation/item.rb +7 -3
  200. data/lib/trestle/navigation.rb +3 -6
  201. data/lib/trestle/registry.rb +61 -0
  202. data/lib/trestle/reloader.rb +24 -12
  203. data/lib/trestle/resource/builder.rb +4 -3
  204. data/lib/trestle/resource/collection.rb +1 -5
  205. data/lib/trestle/resource/toolbar.rb +42 -0
  206. data/lib/trestle/resource.rb +8 -9
  207. data/lib/trestle/scopes/block.rb +9 -8
  208. data/lib/trestle/scopes/definition.rb +22 -0
  209. data/lib/trestle/scopes/scope.rb +6 -2
  210. data/lib/trestle/scopes.rb +38 -11
  211. data/lib/trestle/tab.rb +2 -0
  212. data/lib/trestle/table/actions_column.rb +6 -6
  213. data/lib/trestle/table/automatic.rb +10 -3
  214. data/lib/trestle/table/builder.rb +4 -4
  215. data/lib/trestle/table/column.rb +34 -25
  216. data/lib/trestle/table/row.rb +13 -13
  217. data/lib/trestle/table/select_column.rb +19 -7
  218. data/lib/trestle/table.rb +17 -12
  219. data/lib/trestle/toolbar.rb +4 -11
  220. data/lib/trestle/version.rb +1 -1
  221. data/lib/trestle.rb +62 -48
  222. data/package.json +21 -24
  223. data/trestle.gemspec +6 -6
  224. data/webpack.config.js +40 -49
  225. data/yarn.lock +3044 -4730
  226. metadata +66 -22
  227. data/.travis.yml +0 -27
  228. data/lib/trestle/admin/controller.rb +0 -28
  229. data/lib/trestle/application_controller.rb +0 -12
  230. data/lib/trestle/resource/controller.rb +0 -174
  231. /data/{lib → app/controllers/concerns}/trestle/controller/callbacks.rb +0 -0
  232. /data/{lib → app/controllers/concerns}/trestle/controller/dialog.rb +0 -0
  233. /data/{lib → app/controllers/concerns}/trestle/controller/helpers.rb +0 -0
  234. /data/{lib → app/controllers/concerns}/trestle/controller/layout.rb +0 -0
  235. /data/{lib → app/controllers/concerns}/trestle/controller/location.rb +0 -0
@@ -0,0 +1,101 @@
1
+ vi:
2
+ trestle:
3
+ title: "Trestle Admin"
4
+ footer: "Powered by Trestle"
5
+ version: "Version"
6
+
7
+ helpers:
8
+ page_entries_info:
9
+ one_page:
10
+ display_entries:
11
+ zero: "Không có %{entry_name} được tìm thấy"
12
+ one: "Hiển thị <strong>1</strong> %{entry_name}"
13
+ other: "Hiển thị <strong>tất cả %{count}</strong> %{entry_name}"
14
+
15
+ more_pages:
16
+ display_entries: "Hiển thị %{entry_name} <strong>%{first}&nbsp;-&nbsp;%{last}</strong> của <b>%{total}</b>"
17
+
18
+ onboarding:
19
+ welcome: "Welcome to Trestle"
20
+ no_admins: "To begin, please create an admin within <code>app/admin</code>."
21
+ no_template: "To customize this template, please create <code>%{path}</code>."
22
+ no_form: "Please define a form block or create a <code>_form.html</code> partial."
23
+
24
+ ui:
25
+ toggle_navigation: "Toggle navigation"
26
+ toggle_sidebar: "Toggle sidebar"
27
+ toggle_dropdown: "Toggle dropdown"
28
+
29
+ dialog:
30
+ error: "Yêu cầu chưa hoàn thành."
31
+
32
+ confirmation:
33
+ title: "Bạn có chắc?"
34
+ delete: "Xoá"
35
+ cancel: "Huỷ"
36
+
37
+ file:
38
+ choose_file: "Chọn tập tin..."
39
+ browse: "Tải lên"
40
+
41
+ admin:
42
+ titles:
43
+ index: "Danh sách %{pluralized_model_name}"
44
+ new: "Tạo mới %{model_name}"
45
+ edit: Chỉnh sửa %{model_name}"
46
+
47
+ buttons:
48
+ new: "Tạo mới %{model_name}"
49
+ save: "Lưu %{model_name}"
50
+ delete: "Xoá %{model_name}"
51
+ show: "Xem %{model_name}"
52
+ edit: "Chỉnh sửa %{model_name}"
53
+ ok: "Đồng ý"
54
+
55
+ breadcrumbs:
56
+ home: "Trang Chủ"
57
+
58
+ flash:
59
+ create:
60
+ success:
61
+ title: "Thành công!"
62
+ message: "%{lowercase_model_name} được tạo mới thành công!"
63
+
64
+ failure:
65
+ title: "Cảnh báo!"
66
+ message: "Vui lòng chính xác các thông tin bên dưới!"
67
+
68
+ update:
69
+ success:
70
+ title: "Thành công!"
71
+ message: "%{lowercase_model_name} đã được cập nhật thành công!"
72
+
73
+ failure:
74
+ title: "Cảnh báo!"
75
+ message: "Vui lòng chính xác các thông tin bên dưới!."
76
+
77
+ destroy:
78
+ success:
79
+ title: "Thành công!"
80
+ message: "%{lowercase_model_name} đã được xoá thành công!"
81
+
82
+ failure:
83
+ title: "Cảnh báo!"
84
+ message: "Không thể xoá %{lowercase_model_name}."
85
+
86
+ table:
87
+ headers:
88
+ id: "ID"
89
+
90
+ form:
91
+ select:
92
+ prompt: "- Chọn %{attribute_name} -"
93
+
94
+ format:
95
+ blank: "Chưa có"
96
+
97
+ datepicker:
98
+ formats:
99
+ date: "m/d/Y"
100
+ datetime: "m/d/Y h:i K"
101
+ time: "h:i K"
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 {
@@ -12,6 +12,12 @@
12
12
  }
13
13
  }
14
14
 
15
+ ul.invalid-feedback {
16
+ list-style: none;
17
+ margin-bottom: 0;
18
+ padding-left: 0;
19
+ }
20
+
15
21
  .form-text {
16
22
  color: $form-text-color;
17
23
  font-size: 0.9rem;
@@ -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;
@@ -1,44 +1,89 @@
1
1
  .scopes {
2
- list-style: none;
3
-
4
- margin-top: -0.25rem;
5
2
  margin-bottom: 0;
6
3
 
7
- padding: 0;
4
+ dd {
5
+ margin-bottom: 0;
6
+ }
7
+
8
+ &.grouped {
9
+ @include media-breakpoint-up(md) {
10
+ display: grid;
11
+ grid-template-columns: auto 1fr;
12
+ }
13
+
14
+
15
+ @include media-breakpoint-down(sm) {
16
+ .scope-group-empty:first-child {
17
+ display: none;
18
+ }
19
+ }
20
+
21
+ background: #fafafa;
22
+
23
+ dd {
24
+ padding: 0.375rem 0.5rem 0.375rem;
25
+ }
26
+ }
27
+
28
+ &.columns {
29
+ grid-template-rows: auto 1fr;
30
+ grid-auto-flow: column;
31
+
32
+ overflow-x: auto;
33
+ }
34
+ }
35
+
36
+ .scope-group {
37
+ line-height: 2;
38
+ font-size: 0.85rem;
39
+ padding: 0.5rem 1rem 0.25rem 0.75rem;
40
+
41
+ background: #f7f7f7;
42
+ }
43
+
44
+ .scope-list {
45
+ list-style: none;
8
46
 
9
47
  display: flex;
10
48
  flex-wrap: wrap;
11
49
 
50
+ position: relative;
51
+ align-items: center;
52
+
53
+ margin-bottom: 0;
54
+ padding: 0;
55
+
12
56
  li {
13
- margin-top: 0.25rem;
14
- margin-right: 0.35rem;
57
+ margin: 0.125rem 0.35rem 0.125rem 0;
15
58
 
16
59
  &:last-child {
17
60
  margin-right: 0;
18
61
  }
19
62
  }
63
+ }
20
64
 
21
- a {
22
- background-color: $scope-bg;
65
+ .scope {
66
+ display: inline-block;
23
67
 
24
- font-size: 0.8rem;
25
- font-weight: normal;
26
- white-space: nowrap;
68
+ background-color: $scope-bg;
27
69
 
28
- border-radius: 50rem;
29
- padding: $scope-padding;
70
+ font-size: 0.8rem;
71
+ font-weight: normal;
72
+ white-space: nowrap;
30
73
 
31
- &, &:hover, &:focus {
32
- color: $scope-color;
33
- }
74
+ border-radius: 50rem;
75
+ padding: $scope-padding;
34
76
 
35
- &:hover, &:focus {
36
- background-color: $scope-hover-bg;
37
- text-decoration: none;
38
- }
77
+ &, &:hover, &:focus {
78
+ color: $scope-color;
79
+ }
80
+
81
+ &:hover, &:focus {
82
+ background-color: $scope-hover-bg;
83
+ text-decoration: none;
39
84
  }
40
85
 
41
- li.active a {
86
+ &.active {
42
87
  &, &:hover, &:focus, strong {
43
88
  color: $scope-active-color;
44
89
  }
@@ -24,6 +24,10 @@
24
24
  .main-content-sidebar & {
25
25
  @include table-container-background($content-sidebar-bg);
26
26
  }
27
+
28
+ &.loading {
29
+ opacity: 0.5;
30
+ }
27
31
  }
28
32
 
29
33
  .trestle-table {
@@ -64,6 +68,22 @@
64
68
  .select-row {
65
69
  width: 2.5em;
66
70
  text-align: center;
71
+
72
+ .custom-control {
73
+ padding-left: 1.5rem;
74
+ }
75
+
76
+ .custom-control-input {
77
+ width: 1.5rem;
78
+ height: 1.5rem;
79
+ }
80
+
81
+ .custom-control-label {
82
+ &::before,
83
+ &::after {
84
+ left: -1.325rem;
85
+ }
86
+ }
67
87
  }
68
88
 
69
89
  .actions {
@@ -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,
@@ -105,8 +105,7 @@
105
105
 
106
106
  .utilities {
107
107
  display: flex;
108
- justify-content: space-between;
109
-
108
+ justify-content: flex-start;
110
109
  margin-bottom: 10px;
111
110
  }
112
111
 
@@ -146,6 +145,10 @@
146
145
  padding: 15px;
147
146
  }
148
147
 
148
+ .utilities {
149
+ margin-bottom: 10px;
150
+ }
151
+
149
152
  .main-content-header,
150
153
  .main-content-footer {
151
154
  flex-direction: column;
@@ -16,11 +16,15 @@
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
+
24
+ align-self: center;
25
+
23
26
  margin-left: $grid-gutter-width / 2;
27
+ margin-right: $grid-gutter-width / 2;
24
28
 
25
29
  background: $sidebar-mobile-toggle-bg;
26
30
  border: $sidebar-mobile-toggle-border;
@@ -42,7 +46,6 @@
42
46
  display: flex;
43
47
  align-items: center;
44
48
 
45
- height: 100%;
46
49
  padding: 10px 15px;
47
50
 
48
51
  font-size: 1.4rem;
@@ -155,8 +158,11 @@
155
158
 
156
159
  .app-sidebar-title {
157
160
  justify-content: center;
158
- margin-right: 65px;
159
161
  padding: 10px 5px;
162
+
163
+ // Match right margin with navbar toggler width:
164
+ // (margin + border + font-size * icon-width + padding)
165
+ margin-right: calc(#{$grid-gutter-width} + #{$border-width} + #{$navbar-toggler-font-size} * 1.5 + #{$navbar-toggler-padding-x} * 2);
160
166
  }
161
167
 
162
168
  .toggle-sidebar {
@@ -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
@@ -77,14 +77,14 @@ $tag-hover-border: darken(theme-color("primary"), 10%);
77
77
 
78
78
  // Scopes
79
79
 
80
- $scope-padding: 0.25rem 0.5rem;
80
+ $scope-padding: 0.25rem 0.675rem;
81
81
 
82
- $scope-color: rgba(black, 0.3);
83
- $scope-bg: rgba(black, 0.025);
82
+ $scope-color: rgba(black, 0.4);
83
+ $scope-bg: rgba(black, 0.035);
84
84
 
85
- $scope-label-color: rgba(black, 0.4);
85
+ $scope-label-color: rgba(black, 0.5);
86
86
 
87
- $scope-hover-color: rgba(black, 0.3);
87
+ $scope-hover-color: rgba(black, 0.5);
88
88
  $scope-hover-bg: rgba(black, 0.05);
89
89
 
90
90
  $scope-active-color: white;
@@ -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
 
@@ -15,11 +15,13 @@ init(function (root) {
15
15
  copyAttributes: 'href target data-remote data-method data-url data-params data-type'
16
16
  }
17
17
 
18
- const CONFIRM = { ...DEFAULTS,
18
+ const CONFIRM = {
19
+ ...DEFAULTS,
19
20
  rootSelector: '[data-toggle="confirm"]'
20
21
  }
21
22
 
22
- const DELETE = { ...DEFAULTS,
23
+ const DELETE = {
24
+ ...DEFAULTS,
23
25
  rootSelector: '[data-toggle="confirm-delete"]',
24
26
  btnOkClass: 'btn btn-sm btn-danger',
25
27
  btnOkLabel: i18n['trestle.confirmation.delete'] || 'Delete'
@@ -15,16 +15,24 @@ const TEMPLATE = `
15
15
  function createElement () {
16
16
  const $el = $(TEMPLATE).appendTo('body')
17
17
 
18
- $el.modal({ show: false })
18
+ $el.modal({
19
+ show: false,
20
+ focus: false
21
+ })
19
22
 
20
- // Remove dialog elements once hidden
21
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
22
30
  $el.remove()
23
31
  })
24
32
 
25
33
  // Restore the next visible modal to the foreground
26
34
  $el.on('hide.bs.modal', function () {
27
- $el.prevAll('.modal.in').last().removeClass('background')
35
+ $el.prevAll('.modal.show').last().removeClass('background')
28
36
  })
29
37
 
30
38
  // Set X-Trestle-Dialog header on AJAX requests initiated from the dialog
@@ -117,7 +125,7 @@ export default class Dialog {
117
125
  this.$el.modal('show')
118
126
 
119
127
  // Background any existing visible modals
120
- this.$el.prevAll('.modal.in').addClass('background')
128
+ this.$el.prevAll('.modal.show').addClass('background')
121
129
  }
122
130
 
123
131
  hide () {
@@ -125,6 +133,9 @@ export default class Dialog {
125
133
  }
126
134
  }
127
135
 
136
+ // Expose showError as Trestle.Dialog.showError
137
+ Dialog.showError = showError
138
+
128
139
  $(document).on('click', '[data-behavior="dialog"]', function (e) {
129
140
  e.preventDefault()
130
141
 
@@ -22,6 +22,9 @@ init(function (root) {
22
22
 
23
23
  $form
24
24
  .on('ajax:send', function (e) {
25
+ // Only run when triggered directly on form
26
+ if (e.target !== this) return
27
+
25
28
  // Disable submit buttons
26
29
  $(this).find(':submit').prop('disabled', true)
27
30
 
@@ -30,6 +33,9 @@ init(function (root) {
30
33
  if (button) { $(button).addClass('loading') }
31
34
  })
32
35
  .on('ajax:complete', function (e) {
36
+ // Only run when triggered directly on form
37
+ if (e.target !== this) return
38
+
33
39
  const xhr = e.detail[0]
34
40
 
35
41
  // Reset submit buttons
@@ -64,6 +70,9 @@ init(function (root) {
64
70
  }
65
71
  })
66
72
  .on('ajax:success', function (e) {
73
+ // Only run when triggered directly on form
74
+ if (e.target !== this) return
75
+
67
76
  const xhr = e.detail[2]
68
77
 
69
78
  const $context = $(this).closest('[data-context]')
@@ -0,0 +1,63 @@
1
+ import cookie from '../core/cookie'
2
+
3
+ const PREFIX = 'trestle:navigation'
4
+ const SEPARATOR = ','
5
+
6
+ class NavigationCookieStore {
7
+ addGroup (id, state) {
8
+ this.updateState(state, (groups) => groups.add(id))
9
+ }
10
+
11
+ removeGroup (id, state) {
12
+ this.updateState(state, (groups) => groups.delete(id))
13
+ }
14
+
15
+ updateState (state, callback) {
16
+ const groups = this.getState(state)
17
+ callback(groups)
18
+ this.saveState(state, groups)
19
+ }
20
+
21
+ getState (state) {
22
+ const str = cookie.get(`${PREFIX}:${state}`)
23
+ return new Set(str.length ? str.split(SEPARATOR) : [])
24
+ }
25
+
26
+ saveState (state, groups) {
27
+ cookie.set(`${PREFIX}:${state}`, [...groups].join(SEPARATOR))
28
+ }
29
+ }
30
+
31
+ class Navigation {
32
+ constructor () {
33
+ this.store = new NavigationCookieStore()
34
+ }
35
+
36
+ toggle (list) {
37
+ if (list.hasClass('collapsed')) {
38
+ this.expand(list)
39
+ } else {
40
+ this.collapse(list)
41
+ }
42
+ }
43
+
44
+ expand (list) {
45
+ list.removeClass('collapsed')
46
+
47
+ const id = list.data('group')
48
+
49
+ this.store.addGroup(id, 'expanded')
50
+ this.store.removeGroup(id, 'collapsed')
51
+ }
52
+
53
+ collapse (list) {
54
+ list.addClass('collapsed')
55
+
56
+ const id = list.data('group')
57
+
58
+ this.store.addGroup(id, 'collapsed')
59
+ this.store.removeGroup(id, 'expanded')
60
+ }
61
+ }
62
+
63
+ export default new Navigation()