pages_core 3.5.1 → 3.6.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (312) hide show
  1. checksums.yaml +5 -5
  2. data/README.md +7 -13
  3. data/app/assets/javascripts/pages/{admin.es6.jsx → admin.jsx} +2 -4
  4. data/app/assets/javascripts/pages/admin/components.jsx +2 -0
  5. data/app/assets/javascripts/pages/admin/components/attachment.jsx +130 -0
  6. data/app/assets/javascripts/pages/admin/components/attachment_editor.jsx +131 -0
  7. data/app/assets/javascripts/pages/admin/components/attachments.jsx +211 -0
  8. data/app/assets/javascripts/pages/admin/components/date_range_select.jsx +174 -0
  9. data/app/assets/javascripts/pages/admin/components/drag_uploader.jsx +174 -0
  10. data/app/assets/javascripts/pages/admin/components/editable_image.jsx +57 -0
  11. data/app/assets/javascripts/pages/admin/components/file_upload_button.jsx +44 -0
  12. data/app/assets/javascripts/pages/admin/components/focal_point.jsx +82 -0
  13. data/app/assets/javascripts/pages/admin/components/grid_image.jsx +124 -0
  14. data/app/assets/javascripts/pages/admin/components/image_editor.jsx +496 -0
  15. data/app/assets/javascripts/pages/admin/components/image_grid.jsx +301 -0
  16. data/app/assets/javascripts/pages/admin/components/image_uploader.jsx +171 -0
  17. data/app/assets/javascripts/pages/admin/components/modal.jsx +48 -0
  18. data/app/assets/javascripts/pages/admin/components/modal_store.jsx +20 -0
  19. data/app/assets/javascripts/pages/admin/components/page_dates.jsx +58 -0
  20. data/app/assets/javascripts/pages/admin/components/page_files.jsx +14 -0
  21. data/app/assets/javascripts/pages/admin/components/page_images.jsx +16 -0
  22. data/app/assets/javascripts/pages/admin/components/{page_tree.es6.jsx → page_tree.jsx} +7 -37
  23. data/app/assets/javascripts/pages/admin/components/{page_tree_node.es6.jsx → page_tree_node.jsx} +32 -10
  24. data/app/assets/javascripts/pages/admin/components/page_tree_store.jsx +203 -0
  25. data/app/assets/javascripts/pages/admin/components/rich_text_area.jsx +63 -0
  26. data/app/assets/javascripts/pages/admin/components/rich_text_toolbar.jsx +58 -0
  27. data/app/assets/javascripts/pages/admin/components/toast.jsx +37 -0
  28. data/app/assets/javascripts/pages/admin/components/toast_store.jsx +52 -0
  29. data/app/assets/javascripts/pages/admin/features/{content_tabs.es6.jsx → content_tabs.jsx} +11 -2
  30. data/app/assets/javascripts/pages/admin/features/{edit_page.es6.jsx → edit_page.jsx} +7 -51
  31. data/app/assets/javascripts/pages/admin/features/rich_text.jsx +14 -0
  32. data/app/assets/javascripts/pages/admin/features/{tag_editor.es6.jsx → tag_editor.jsx} +0 -0
  33. data/app/assets/javascripts/pages/admin/lib/{tree.es6.jsx → tree.jsx} +0 -0
  34. data/app/assets/javascripts/pages/{login_form.es6.jsx → login_form.jsx} +0 -0
  35. data/app/assets/stylesheets/pages/admin.scss +9 -13
  36. data/app/assets/stylesheets/pages/admin/components/archive.scss +6 -0
  37. data/app/assets/stylesheets/pages/admin/components/attachments.scss +130 -0
  38. data/app/assets/stylesheets/pages/admin/components/buttons.scss +18 -0
  39. data/app/assets/stylesheets/pages/admin/components/forms.scss +99 -21
  40. data/app/assets/stylesheets/pages/admin/components/header.scss +16 -28
  41. data/app/assets/stylesheets/pages/admin/components/image_editor.scss +195 -0
  42. data/app/assets/stylesheets/pages/admin/components/image_grid.scss +181 -0
  43. data/app/assets/stylesheets/pages/admin/components/image_uploader.scss +53 -0
  44. data/app/assets/stylesheets/pages/admin/components/links.scss +1 -1
  45. data/app/assets/stylesheets/pages/admin/components/list_table.scss +8 -5
  46. data/app/assets/stylesheets/pages/admin/components/login.scss +2 -8
  47. data/app/assets/stylesheets/pages/admin/components/modal.scss +91 -0
  48. data/app/assets/stylesheets/pages/admin/components/page_tree.scss +12 -6
  49. data/app/assets/stylesheets/pages/admin/components/pagination.scss +34 -16
  50. data/app/assets/stylesheets/pages/admin/components/sidebar.scss +9 -6
  51. data/app/assets/stylesheets/pages/admin/components/tag_editor.scss +20 -15
  52. data/app/assets/stylesheets/pages/admin/components/textarea.scss +1 -71
  53. data/app/assets/stylesheets/pages/admin/components/toast.scss +51 -0
  54. data/app/assets/stylesheets/pages/admin/components/toolbar.scss +108 -0
  55. data/app/assets/stylesheets/pages/admin/controllers/pages.scss +9 -21
  56. data/app/assets/stylesheets/pages/admin/controllers/users.scss +2 -2
  57. data/app/assets/stylesheets/pages/admin/vars.scss +26 -4
  58. data/app/controller_dummies/admin/admin_controller.rb +0 -2
  59. data/app/controller_dummies/application_controller.rb +2 -4
  60. data/app/controller_dummies/attachments_controller.rb +2 -0
  61. data/app/controller_dummies/frontend_controller.rb +0 -2
  62. data/app/controller_dummies/images_controller.rb +0 -2
  63. data/app/controller_dummies/page_files_controller.rb +0 -2
  64. data/app/controller_dummies/pages_controller.rb +0 -2
  65. data/app/controller_dummies/sitemaps_controller.rb +0 -2
  66. data/app/controllers/admin/attachments_controller.rb +48 -0
  67. data/app/controllers/admin/categories_controller.rb +2 -5
  68. data/app/controllers/admin/images_controller.rb +25 -19
  69. data/app/controllers/admin/invites_controller.rb +16 -26
  70. data/app/controllers/admin/pages_controller.rb +50 -32
  71. data/app/controllers/admin/password_resets_controller.rb +11 -18
  72. data/app/controllers/admin/users_controller.rb +16 -22
  73. data/app/controllers/concerns/pages_core/admin/news_page_controller.rb +21 -14
  74. data/app/controllers/concerns/pages_core/authentication.rb +0 -2
  75. data/app/controllers/concerns/pages_core/domain_based_cache.rb +0 -2
  76. data/app/controllers/concerns/pages_core/error_renderer.rb +33 -0
  77. data/app/controllers/concerns/pages_core/policies_helper.rb +9 -13
  78. data/app/controllers/concerns/pages_core/preview_pages_controller.rb +3 -5
  79. data/app/controllers/concerns/pages_core/process_titler.rb +1 -3
  80. data/app/controllers/concerns/pages_core/rss_controller.rb +0 -2
  81. data/app/controllers/errors_controller.rb +52 -26
  82. data/app/controllers/pages_core/admin_controller.rb +22 -13
  83. data/app/controllers/pages_core/attachments_controller.rb +36 -0
  84. data/app/controllers/pages_core/{application_controller.rb → base_controller.rb} +16 -5
  85. data/app/controllers/pages_core/frontend/page_files_controller.rb +5 -24
  86. data/app/controllers/pages_core/frontend/pages_controller.rb +4 -8
  87. data/app/controllers/pages_core/frontend_controller.rb +0 -2
  88. data/app/controllers/pages_core/images_controller.rb +0 -2
  89. data/app/controllers/pages_core/sitemaps_controller.rb +3 -5
  90. data/app/controllers/sessions_controller.rb +3 -15
  91. data/app/formatters/pages_core/html_formatter.rb +60 -16
  92. data/app/formatters/pages_core/link_renderer.rb +15 -0
  93. data/app/helpers/admin/admin_helper.rb +0 -2
  94. data/app/helpers/admin/menu_helper.rb +2 -4
  95. data/app/helpers/admin/pages_helper.rb +47 -9
  96. data/app/helpers/application_helper.rb +0 -2
  97. data/app/helpers/frontend_helper.rb +0 -2
  98. data/app/helpers/pages_core/admin/admin_helper.rb +75 -20
  99. data/app/helpers/pages_core/admin/form_builder.rb +36 -0
  100. data/app/helpers/pages_core/admin/labelled_field_helper.rb +6 -8
  101. data/app/helpers/pages_core/admin/tag_editor_helper.rb +0 -2
  102. data/app/helpers/pages_core/application_helper.rb +1 -2
  103. data/app/helpers/pages_core/attachments_helper.rb +36 -0
  104. data/app/helpers/pages_core/form_builder.rb +7 -11
  105. data/app/helpers/pages_core/frontend_helper.rb +0 -6
  106. data/app/helpers/pages_core/head_tags_helper.rb +8 -4
  107. data/app/helpers/pages_core/images_helper.rb +0 -2
  108. data/app/helpers/pages_core/meta_tags_helper.rb +3 -5
  109. data/app/helpers/pages_core/open_graph_tags_helper.rb +1 -3
  110. data/app/helpers/pages_core/page_path_helper.rb +14 -9
  111. data/app/jobs/pages_core/autopublish_job.rb +0 -2
  112. data/app/jobs/pages_core/sweep_cache_job.rb +0 -2
  113. data/app/mailers/admin_mailer.rb +3 -16
  114. data/app/models/attachment.rb +76 -0
  115. data/app/models/autopublisher.rb +3 -3
  116. data/app/models/category.rb +0 -3
  117. data/app/models/concerns/pages_core/has_roles.rb +1 -2
  118. data/app/models/concerns/pages_core/humanizable_param.rb +4 -4
  119. data/app/models/concerns/pages_core/page_model/attachments.rb +39 -0
  120. data/app/models/concerns/pages_core/page_model/autopublishable.rb +0 -2
  121. data/app/models/concerns/pages_core/page_model/dated_page.rb +59 -0
  122. data/app/models/concerns/pages_core/page_model/images.rb +12 -15
  123. data/app/models/concerns/pages_core/page_model/localizable.rb +10 -3
  124. data/app/models/concerns/pages_core/page_model/pathable.rb +8 -10
  125. data/app/models/concerns/pages_core/page_model/redirectable.rb +0 -2
  126. data/app/models/concerns/pages_core/page_model/sortable.rb +1 -3
  127. data/app/models/concerns/pages_core/page_model/status.rb +1 -3
  128. data/app/models/concerns/pages_core/page_model/templateable.rb +2 -4
  129. data/app/models/concerns/pages_core/page_model/tree.rb +24 -5
  130. data/app/models/concerns/pages_core/sweepable.rb +0 -2
  131. data/app/models/concerns/pages_core/taggable.rb +4 -3
  132. data/app/models/image.rb +1 -0
  133. data/app/models/invite.rb +0 -10
  134. data/app/models/page.rb +17 -21
  135. data/app/models/page_builder.rb +0 -2
  136. data/app/models/page_category.rb +0 -2
  137. data/app/models/page_exporter.rb +87 -0
  138. data/app/models/page_file.rb +24 -48
  139. data/app/models/page_image.rb +3 -37
  140. data/app/models/page_path.rb +0 -2
  141. data/app/models/password_reset_token.rb +0 -4
  142. data/app/models/role.rb +15 -2
  143. data/app/models/tag.rb +15 -6
  144. data/app/models/tagging.rb +1 -3
  145. data/app/models/user.rb +29 -25
  146. data/app/policies/page_file_policy.rb +13 -17
  147. data/app/policies/page_image_policy.rb +13 -17
  148. data/app/policies/page_policy.rb +26 -26
  149. data/app/policies/policy.rb +2 -8
  150. data/app/policies/user_policy.rb +32 -32
  151. data/app/serializers/admin/attachment_serializer.rb +29 -0
  152. data/app/serializers/admin/image_serializer.rb +53 -6
  153. data/app/serializers/admin/page_file_serializer.rb +6 -0
  154. data/app/serializers/admin/page_image_serializer.rb +1 -1
  155. data/app/serializers/page_export_serializer.rb +30 -0
  156. data/app/serializers/page_file_export_serializer.rb +4 -0
  157. data/app/serializers/page_image_export_serializer.rb +40 -0
  158. data/app/serializers/page_image_serializer.rb +2 -0
  159. data/app/services/pages_core/create_user_service.rb +36 -0
  160. data/app/services/pages_core/invite_service.rb +41 -0
  161. data/app/views/admin/images/show.json.jbuilder +6 -0
  162. data/app/views/admin/pages/_edit_content.html.erb +7 -0
  163. data/app/views/admin/pages/_edit_files.html.erb +8 -0
  164. data/app/views/admin/pages/_edit_images.html.erb +8 -95
  165. data/app/views/admin/pages/_edit_options.html.erb +7 -15
  166. data/app/views/admin/pages/_list_item.html.erb +50 -0
  167. data/app/views/admin/pages/deleted.html.erb +42 -0
  168. data/app/views/admin/pages/edit.html.erb +9 -94
  169. data/app/views/admin/pages/index.html.erb +9 -12
  170. data/app/views/admin/pages/new.html.erb +2 -1
  171. data/app/views/admin/pages/news.html.erb +59 -45
  172. data/app/views/admin/password_resets/show.html.erb +6 -9
  173. data/app/views/admin/users/_access_control.html.erb +4 -1
  174. data/app/views/admin/users/_list.html.erb +12 -7
  175. data/app/views/admin/users/edit.html.erb +5 -11
  176. data/app/views/admin/users/login.html.erb +58 -15
  177. data/app/views/admin/users/show.html.erb +1 -1
  178. data/app/views/admin_mailer/invite.text.erb +1 -1
  179. data/app/views/admin_mailer/password_reset.text.erb +1 -1
  180. data/app/views/errors/401.html.erb +6 -0
  181. data/app/views/errors/403.html.erb +1 -1
  182. data/app/views/errors/500.html.erb +11 -6
  183. data/app/views/errors/500_critical.html.erb +1 -1
  184. data/app/views/feeds/pages.rss.builder +1 -3
  185. data/app/views/layouts/admin.html.erb +83 -83
  186. data/app/views/layouts/admin/_analytics.html.erb +1 -3
  187. data/app/views/layouts/admin/_header.html.erb +2 -2
  188. data/app/views/layouts/errors.html.erb +3 -7
  189. data/config/locales/en.yml +12 -0
  190. data/config/routes.rb +38 -55
  191. data/db/migrate/20111219033112_create_pages_tables.rb +6 -8
  192. data/db/migrate/20120627033112_rename_textbits.rb +1 -3
  193. data/db/migrate/20121010055412_drop_removed_tables.rb +1 -3
  194. data/db/migrate/20130130053932_add_queue_to_delayed_jobs.rb +1 -3
  195. data/db/migrate/20130303053932_remove_filter_from_localizations.rb +1 -3
  196. data/db/migrate/20130303160632_remove_imagesets.rb +1 -3
  197. data/db/migrate/20130303161732_remove_sms_subscribers.rb +1 -3
  198. data/db/migrate/20130823133208_update_page_redirect_to.rb +1 -3
  199. data/db/migrate/20140203183900_create_roles.rb +1 -1
  200. data/db/migrate/20140414150500_change_locale_names.rb +1 -3
  201. data/db/migrate/20140604142100_remove_openid_url.rb +1 -1
  202. data/db/migrate/20140920231700_convert_images_to_dis.rb +1 -1
  203. data/db/migrate/20140922124600_convert_page_files_to_dis.rb +1 -1
  204. data/db/migrate/20141004003100_create_password_reset_tokens.rb +1 -1
  205. data/db/migrate/20141006181300_remove_user_cruft.rb +1 -1
  206. data/db/migrate/20141007173000_create_invites.rb +1 -1
  207. data/db/migrate/20150204130800_update_delayed_job_table.rb +1 -1
  208. data/db/migrate/20150401131300_localize_images.rb +1 -1
  209. data/db/migrate/20150520174300_add_meta_image_to_page.rb +1 -1
  210. data/db/migrate/20150904164200_add_pinned_to_tags.rb +1 -1
  211. data/db/migrate/20151002174800_create_page_paths.rb +2 -2
  212. data/db/migrate/20151021103400_drop_binaries_table.rb +1 -1
  213. data/db/migrate/20151204151000_remove_page_content_order.rb +1 -1
  214. data/db/migrate/20160330220900_rename_pages_categories.rb +1 -1
  215. data/db/migrate/20160405202700_change_localization_limit.rb +1 -1
  216. data/db/migrate/20170716040500_remove_page_comments.rb +23 -0
  217. data/db/migrate/20170716213400_remove_sessions.rb +15 -0
  218. data/db/migrate/20180207134000_add_dates_to_pages.rb +11 -0
  219. data/db/migrate/20190211154800_create_attachments.rb +73 -0
  220. data/lib/pages_core.rb +6 -9
  221. data/lib/pages_core/admin_menu_item.rb +0 -2
  222. data/lib/pages_core/archive_finder.rb +21 -15
  223. data/lib/pages_core/attachment_embedder.rb +38 -0
  224. data/lib/pages_core/cache_sweeper.rb +14 -23
  225. data/lib/pages_core/configuration.rb +0 -2
  226. data/lib/pages_core/configuration/base.rb +0 -2
  227. data/lib/pages_core/configuration/pages.rb +2 -8
  228. data/lib/pages_core/digest_verifier.rb +70 -0
  229. data/lib/pages_core/engine.rb +6 -13
  230. data/lib/pages_core/extensions.rb +0 -3
  231. data/lib/pages_core/extensions/string_extensions.rb +0 -2
  232. data/lib/pages_core/page_path_constraint.rb +0 -2
  233. data/lib/pages_core/pages_plugin.rb +0 -2
  234. data/lib/pages_core/plugin.rb +0 -2
  235. data/lib/pages_core/pub_sub.rb +36 -0
  236. data/lib/pages_core/templates.rb +0 -2
  237. data/lib/pages_core/templates/block_configuration.rb +1 -3
  238. data/lib/pages_core/templates/configuration.rb +88 -10
  239. data/lib/pages_core/templates/configuration_handler.rb +6 -4
  240. data/lib/pages_core/templates/configuration_proxy.rb +4 -2
  241. data/lib/pages_core/templates/controller_actions.rb +0 -2
  242. data/lib/pages_core/templates/template_configuration.rb +41 -37
  243. data/lib/pages_core/version.rb +1 -3
  244. data/lib/rails/generators/pages_core/frontend/frontend_generator.rb +10 -17
  245. data/lib/rails/generators/pages_core/frontend/templates/application.js.erb +1 -2
  246. data/lib/rails/generators/pages_core/frontend/templates/{application.css.scss.erb → application.scss.erb} +0 -0
  247. data/lib/rails/generators/pages_core/frontend/templates/{base.css.scss.erb → base.scss.erb} +0 -0
  248. data/lib/rails/generators/pages_core/frontend/templates/{breakpoints.css.scss.erb → breakpoints.scss.erb} +0 -0
  249. data/lib/rails/generators/pages_core/frontend/templates/clearfix.scss.erb +7 -0
  250. data/lib/rails/generators/pages_core/frontend/templates/layout.html.erb +0 -3
  251. data/lib/rails/generators/pages_core/install/install_generator.rb +4 -15
  252. data/lib/rails/generators/pages_core/install/templates/application_controller.rb +1 -3
  253. data/lib/rails/generators/pages_core/install/templates/application_helper.rb +0 -2
  254. data/lib/rails/generators/pages_core/install/templates/cache_sweeper_initializer.rb +0 -5
  255. data/lib/rails/generators/pages_core/install/templates/delayed_job_initializer.rb +0 -2
  256. data/lib/rails/generators/pages_core/install/templates/frontend_controller.rb +0 -2
  257. data/lib/rails/generators/pages_core/install/templates/frontend_helper.rb +0 -2
  258. data/lib/rails/generators/pages_core/install/templates/page_templates_initializer.rb +7 -15
  259. data/lib/rails/generators/pages_core/install/templates/pages_controller.rb +0 -2
  260. data/lib/rails/generators/pages_core/install/templates/pages_initializer.rb +2 -19
  261. data/lib/rails/generators/pages_core/rspec/rspec_generator.rb +2 -4
  262. data/lib/rails/generators/pages_core/rspec/templates/factories.rb +1 -1
  263. data/lib/rails/generators/pages_core/rspec/templates/spec_helper.rb +4 -13
  264. data/lib/tasks/pages.rake +0 -62
  265. data/lib/tasks/pages/cache.rake +6 -2
  266. data/lib/tasks/pages/export.rake +9 -0
  267. data/lib/tasks/pages/page_paths.rake +0 -2
  268. data/lib/tasks/pages/update.rake +0 -2
  269. data/template.rb +3 -3
  270. data/vendor/assets/javascripts/ReactCrop.min.js +1 -0
  271. data/vendor/assets/javascripts/reflux.min.js +1 -1
  272. data/vendor/assets/stylesheets/ReactCrop.css +167 -0
  273. metadata +200 -175
  274. data/app/assets/javascripts/pages/admin/components.es6.jsx +0 -1
  275. data/app/assets/javascripts/pages/admin/components/page_tree_actions.es6.jsx +0 -8
  276. data/app/assets/javascripts/pages/admin/components/page_tree_store.es6.jsx +0 -161
  277. data/app/assets/javascripts/pages/admin/features/editable_image.es6.jsx +0 -145
  278. data/app/assets/javascripts/pages/admin/features/modal.es6.jsx +0 -90
  279. data/app/assets/javascripts/pages/admin/features/page_images.es6.jsx +0 -338
  280. data/app/assets/javascripts/pages/admin/features/rich_text.es6.jsx +0 -124
  281. data/app/assets/javascripts/pages/admin/lib/ajax_extensions.es6.jsx +0 -21
  282. data/app/assets/javascripts/pages/admin/lib/center_on_screen.es6.jsx +0 -22
  283. data/app/assets/stylesheets/pages/admin/components/editable_image.scss +0 -18
  284. data/app/assets/stylesheets/pages/admin/components/images.scss +0 -155
  285. data/app/assets/stylesheets/pages/admin/print.scss +0 -17
  286. data/app/controllers/admin/page_comments_controller.rb +0 -61
  287. data/app/controllers/admin/page_files_controller.rb +0 -79
  288. data/app/controllers/admin/page_images_controller.rb +0 -111
  289. data/app/controllers/concerns/pages_core/add_comments_controller.rb +0 -67
  290. data/app/controllers/concerns/pages_core/exception_handler.rb +0 -137
  291. data/app/controllers/concerns/pages_core/search_pages_controller.rb +0 -40
  292. data/app/helpers/pages_core/login_helper.rb +0 -14
  293. data/app/indices/page_file_index.rb +0 -9
  294. data/app/indices/page_index.rb +0 -29
  295. data/app/indices/user_index.rb +0 -11
  296. data/app/models/concerns/pages_core/page_model/commentable.rb +0 -29
  297. data/app/models/concerns/pages_core/page_model/searchable.rb +0 -41
  298. data/app/models/page_comment.rb +0 -18
  299. data/app/serializers/page_tree_serializer.rb +0 -15
  300. data/app/views/admin/pages/_edit_comments.html.erb +0 -37
  301. data/app/views/admin/pages/_pagelisting.html.erb +0 -63
  302. data/app/views/admin/users/_login_form.html.erb +0 -47
  303. data/app/views/admin_mailer/comment_notification.text.erb +0 -7
  304. data/lib/pages_core/extensions/hash_extensions.rb +0 -23
  305. data/lib/pages_core/file_embedder.rb +0 -40
  306. data/lib/pages_core/paginates.rb +0 -102
  307. data/lib/rails/generators/pages_core/frontend/templates/hidpi.css.scss.erb +0 -8
  308. data/lib/rails/generators/pages_core/install/templates/thinking_sphinx.yml +0 -12
  309. data/lib/tasks/db.rake +0 -96
  310. data/lib/tasks/pages/assets.rake +0 -65
  311. data/vendor/assets/javascripts/jquery.dimscreen.js +0 -77
  312. data/vendor/assets/javascripts/jquery.fieldselection.js +0 -59
@@ -0,0 +1,48 @@
1
+ class Modal extends Reflux.Component {
2
+ constructor(props) {
3
+ super(props);
4
+ this.store = ModalStore;
5
+ this.closeModal = this.closeModal.bind(this);
6
+ this.handleKeypress = this.handleKeypress.bind(this);
7
+ }
8
+
9
+ componentDidMount() {
10
+ window.addEventListener("keypress", this.handleKeypress);
11
+ }
12
+
13
+ componentWillUnmount() {
14
+ window.removeEventListener("keypress", this.handleKeypress);
15
+ }
16
+
17
+ closeModal(evt) {
18
+ evt.stopPropagation();
19
+ evt.preventDefault();
20
+ ModalActions.close();
21
+ }
22
+
23
+ handleKeypress(evt) {
24
+ if (this.state.component && (evt.key == "Escape" || evt.keyCode === 27)) {
25
+ this.closeModal(evt);
26
+ }
27
+ }
28
+
29
+ render() {
30
+ let component = this.state.component;
31
+
32
+ if (component) {
33
+ document.body.classList.add("modal");
34
+ } else {
35
+ document.body.classList.remove("modal");
36
+ return (<div className="modal-wrapper"></div>);
37
+ }
38
+
39
+ return (
40
+ <div className="modal-wrapper open">
41
+ <div className="background" onClick={this.closeModal} />
42
+ <div className="modal">
43
+ {component}
44
+ </div>
45
+ </div>
46
+ );
47
+ }
48
+ }
@@ -0,0 +1,20 @@
1
+ var ModalActions = Reflux.createActions([
2
+ "open",
3
+ "close"
4
+ ]);
5
+
6
+ class ModalStore extends Reflux.Store {
7
+ constructor() {
8
+ super();
9
+ this.state = { component: null };
10
+ this.listenToMany(ModalActions);
11
+ }
12
+
13
+ onOpen(component) {
14
+ this.setState({component: component});
15
+ }
16
+
17
+ onClose() {
18
+ this.setState({component: null});
19
+ }
20
+ }
@@ -0,0 +1,58 @@
1
+ class PageDates extends React.Component {
2
+ constructor(props) {
3
+ super(props);
4
+ this.state = {
5
+ has_dates: (props.starts_at ? true : false),
6
+ all_day: !!props.all_day
7
+ };
8
+
9
+ this.toggleAllDay = this.toggleAllDay.bind(this);
10
+ this.toggleHasDates = this.toggleHasDates.bind(this);
11
+ }
12
+
13
+ toggleHasDates() {
14
+ this.setState({ has_dates: !this.state.has_dates });
15
+ }
16
+
17
+ toggleAllDay() {
18
+ this.setState({ all_day: !this.state.all_day });
19
+ }
20
+
21
+ timeToString(time) {
22
+ return time.toTimeString().slice(0, 5);
23
+ }
24
+
25
+ render() {
26
+ return (
27
+ <div className="page-dates field">
28
+ <input type="hidden"
29
+ name="page[all_day]"
30
+ value={(this.state.has_dates && this.state.all_day) ? "1" : "0"} />
31
+ <label>
32
+ Dates
33
+ </label>
34
+ <div className="toggles">
35
+ <label className="has-dates-toggle">
36
+ <input type="checkbox"
37
+ checked={this.state.has_dates}
38
+ onChange={this.toggleHasDates} />
39
+ Enabled
40
+ </label>
41
+ <label className={!this.state.has_dates && "disabled"}>
42
+ <input type="checkbox"
43
+ disabled={!this.state.has_dates}
44
+ checked={this.state.all_day}
45
+ onChange={this.toggleAllDay} />
46
+ All day event
47
+ </label>
48
+ </div>
49
+ <DateRangeSelect objectName="page"
50
+ startsAt={this.props.starts_at}
51
+ endsAt={this.props.ends_at}
52
+ disabled={!this.state.has_dates}
53
+ disableTime={this.state.all_day}
54
+ />
55
+ </div>
56
+ );
57
+ }
58
+ }
@@ -0,0 +1,14 @@
1
+ class PageFiles extends React.Component {
2
+ render() {
3
+ return (
4
+ <div className="page-files">
5
+ <Attachments attribute="page[page_files_attributes]"
6
+ showEmbed={true}
7
+ locale={this.props.locale}
8
+ locales={this.props.locales}
9
+ csrf_token={this.props.csrf_token}
10
+ records={this.props.records} />
11
+ </div>
12
+ );
13
+ }
14
+ }
@@ -0,0 +1,16 @@
1
+ class PageImages extends React.Component {
2
+ render() {
3
+ return (
4
+ <div className="page-images">
5
+ <ImageGrid attribute="page[page_images_attributes]"
6
+ primaryAttribute="page[image_id]"
7
+ enablePrimary={true}
8
+ showEmbed={true}
9
+ locale={this.props.locale}
10
+ locales={this.props.locales}
11
+ csrf_token={this.props.csrf_token}
12
+ records={this.props.records} />
13
+ </div>
14
+ );
15
+ }
16
+ }
@@ -25,42 +25,18 @@
25
25
  SOFTWARE.
26
26
  */
27
27
 
28
- class PageTree extends React.Component {
28
+ class PageTree extends Reflux.Component {
29
29
  constructor(props) {
30
30
  super(props);
31
- this.state = this.init(this.props);
32
-
33
- if (props.tree) {
34
- PageTreeActions.init(this.props.tree);
35
- }
31
+ this.state = {
32
+ dragging: this.initDragging()
33
+ };
34
+ this.store = PageTreeStore;
36
35
  }
37
36
 
38
37
  componentDidMount() {
39
- let self = this;
40
- this.unsubscribe = PageTreeStore.listen(
41
- newTree => self.setState({tree: newTree})
42
- );
43
- }
44
-
45
- componentWillReceiveProps(nextProps) {
46
- if (!this._updated) {
47
- this.setState(this.init(nextProps));
48
- if (nextProps.tree) {
49
- PageTreeActions.init(this.props.tree);
50
- }
51
- } else {
52
- this._updated = false;
53
- }
54
- }
55
-
56
- componentWillUnmount() {
57
- this.unsubscribe();
58
- }
59
-
60
- init(props) {
61
- return {
62
- dragging: this.initDragging()
63
- };
38
+ PageTreeActions.setCSRFToken(this.props.csrf_token);
39
+ PageTreeActions.init(this.props);
64
40
  }
65
41
 
66
42
  initDragging() {
@@ -101,7 +77,6 @@ class PageTree extends React.Component {
101
77
  }
102
78
 
103
79
  render() {
104
- var self = this;
105
80
  var tree = this.state.tree;
106
81
  var dragging = this.state.dragging;
107
82
 
@@ -320,11 +295,6 @@ class PageTree extends React.Component {
320
295
  }
321
296
  }
322
297
 
323
- PageTree.propTypes = {
324
- tree: React.PropTypes.object.isRequired,
325
- paddingLeft: React.PropTypes.number
326
- };
327
-
328
298
  PageTree.defaultProps = {
329
299
  paddingLeft: 15
330
300
  };
@@ -31,6 +31,11 @@ class PageTreeNode extends React.Component {
31
31
  this.state = { newName: props.index.node.name };
32
32
  }
33
33
 
34
+ permitted(action) {
35
+ return this.node().permissions &&
36
+ this.node().permissions.indexOf(action) != -1;
37
+ }
38
+
34
39
  actions() {
35
40
  let statusLabel = (this.node().status != 2) ? "Publish" : "Hide";
36
41
  let statusIcon = (this.node().status != 2) ? "check" : "ban";
@@ -53,25 +58,25 @@ class PageTreeNode extends React.Component {
53
58
  } else {
54
59
  return (
55
60
  <span className="actions">
56
- {this.button(statusLabel, {
61
+ {this.permitted("edit") && this.button(statusLabel, {
57
62
  className: "toggle-status",
58
63
  icon: statusIcon,
59
64
  onClick: e => this.toggleStatus()
60
65
  })}
61
66
 
62
- {this.button("Rename", {
67
+ {this.permitted("edit") && this.button("Rename", {
63
68
  className: "edit",
64
69
  icon: "pencil",
65
70
  onClick: e => this.edit()
66
71
  })}
67
72
 
68
- {this.button("Delete", {
73
+ {this.permitted("edit") && this.button("Delete", {
69
74
  className: "delete",
70
75
  icon: "trash",
71
76
  onClick: e => this.deletePage()
72
77
  })}
73
78
 
74
- {this.button("Add child", {
79
+ {this.permitted("create") && this.button("Add child", {
75
80
  className: "add",
76
81
  icon: "plus",
77
82
  onClick: e => this.props.addChild(this.props.index)
@@ -90,7 +95,9 @@ class PageTreeNode extends React.Component {
90
95
  }
91
96
  }
92
97
 
93
- if (!node.collapsed && this.visibleChildren().length > 0) {
98
+ if (!node.collapsed &&
99
+ this.permitted("create") &&
100
+ (node.root || this.visibleChildren().length > 0)) {
94
101
  return (
95
102
  this.button("Add page here", {
96
103
  className: "add add-inline",
@@ -188,7 +195,9 @@ class PageTreeNode extends React.Component {
188
195
  }
189
196
 
190
197
  collapsedLabel() {
191
- if (this.node().collapsed && this.node().children.length > 0) {
198
+ if (this.node().collapsed &&
199
+ this.node().children &&
200
+ this.node().children.length > 0) {
192
201
  let pluralized = (this.node().children.length == 1) ? "item" : "items";
193
202
  return (
194
203
  <span className="collapsed-label">
@@ -231,16 +240,17 @@ class PageTreeNode extends React.Component {
231
240
  let props = this.props;
232
241
  let index = props.index;
233
242
  let dragging = props.dragging;
243
+ let editing = this.node().editing;
234
244
  var classnames = "node";
235
245
 
236
- var node = this.node().editing ? this.renderEditNode() : this.renderNode();
246
+ var node = editing ? this.renderEditNode() : this.renderNode();
237
247
 
238
248
  if (index.id === dragging) {
239
249
  classnames = "node placeholder";
240
250
  }
241
251
 
242
252
  let handleMouseDown = function (e) {
243
- if (props.onDragStart) {
253
+ if (self.permitted("edit") && !editing && props.onDragStart) {
244
254
  props.onDragStart(props.index.id, self.refs.inner, e);
245
255
  }
246
256
  }
@@ -309,6 +319,7 @@ class PageTreeNode extends React.Component {
309
319
  let index = this.props.index;
310
320
  let node = index.node;
311
321
 
322
+ var dateLabel = "";
312
323
  var pageName = <span className="name">{this.pageName()}</span>;
313
324
  var className = "page";
314
325
 
@@ -318,12 +329,18 @@ class PageTreeNode extends React.Component {
318
329
  className = `page status-${this.node().status}`;
319
330
  }
320
331
 
321
- if (node.id && node.locale) {
332
+ if (node.id && node.locale && this.permitted("edit")) {
322
333
  pageName = <a href={this.editUrl(node)} className="name">
323
334
  {this.pageName()}
324
335
  </a>;
325
336
  }
326
337
 
338
+ if (node.id && node.starts_at) {
339
+ dateLabel = <span className="date">
340
+ {node.starts_at}
341
+ </span>
342
+ }
343
+
327
344
  if (node.news_page) {
328
345
  iconClass = "fa fa-newspaper-o icon";
329
346
  } else if (node.pinned) {
@@ -334,6 +351,7 @@ class PageTreeNode extends React.Component {
334
351
  <div className={className}>
335
352
  <i className={iconClass}></i>
336
353
  {pageName}
354
+ {dateLabel}
337
355
  {this.statusLabel()}
338
356
  {this.collapsedLabel()}
339
357
  {this.actions()}
@@ -369,6 +387,10 @@ class PageTreeNode extends React.Component {
369
387
  }
370
388
 
371
389
  visibleChildren() {
372
- return this.node().children.filter(p => p.status != 4);
390
+ if (this.node().children) {
391
+ return this.node().children.filter(p => p.status != 4);
392
+ } else {
393
+ return [];
394
+ }
373
395
  }
374
396
  }
@@ -0,0 +1,203 @@
1
+ var PageTreeActions = Reflux.createActions([
2
+ "addChild",
3
+ "init",
4
+ "movedPage",
5
+ "setCSRFToken",
6
+ "toggleCollapsed",
7
+ "updatePage",
8
+ "updateTree"
9
+ ]);
10
+
11
+ class PageTreeStore extends Reflux.Store {
12
+ constructor() {
13
+ super();
14
+ this.state = { tree: null };
15
+ this.listenToMany(PageTreeActions);
16
+ }
17
+
18
+ collapsedState() {
19
+ if (window && window.localStorage &&
20
+ typeof(window.localStorage.collapsedPages) != "undefined") {
21
+ return JSON.parse(window.localStorage.collapsedPages);
22
+ }
23
+ return {};
24
+ }
25
+
26
+ applyCollapsed(tree) {
27
+ let store = this;
28
+ let collapsedState = this.collapsedState();
29
+ let walk = function (id) {
30
+ var index = tree.getIndex(id);
31
+ var node = index.node;
32
+ if (collapsedState.hasOwnProperty(node.id)) {
33
+ node.collapsed = collapsedState[node.id];
34
+ } else if (node.news_page) {
35
+ node.collapsed = true;
36
+ } else if (store.depth(tree, index) > 1) {
37
+ node.collapsed = true;
38
+ }
39
+ if (index.children && index.children.length) {
40
+ index.children.forEach(c => walk(c));
41
+ }
42
+ };
43
+ walk(1);
44
+ }
45
+
46
+ depth(tree, index) {
47
+ var depth = 0;
48
+ var pointer = index;
49
+ while (pointer = tree.getIndex(pointer.parent)) {
50
+ depth += 1;
51
+ }
52
+ return depth;
53
+ }
54
+
55
+ createPage(index, attributes) {
56
+ let store = this;
57
+ let url = `/admin/${index.node.locale}/pages.json`;
58
+ $.post(url, { page: attributes }, function (response) {
59
+ store.updateNode(index, response);
60
+ });
61
+ }
62
+
63
+ movePage(index, parent, position) {
64
+ let data = {
65
+ parent_id: parent.node.id,
66
+ position: position
67
+ };
68
+ let url = `/admin/${index.node.locale}/pages/${index.node.id}/move.json`;
69
+ this.performUpdate(index, url, data);
70
+ }
71
+
72
+ updatePage(index, attributes) {
73
+ let url = `/admin/${index.node.locale}/pages/${index.node.id}.json`;
74
+ this.performUpdate(index, url, { page: attributes });
75
+ }
76
+
77
+ performUpdate(index, url, data) {
78
+ let store = this;
79
+
80
+ let xhr = new XMLHttpRequest();
81
+ xhr.open("PUT", url);
82
+ xhr.setRequestHeader("Content-Type","application/json; charset=utf-8");
83
+ xhr.setRequestHeader("X-CSRF-Token", this.state.csrf_token);
84
+ xhr.addEventListener("load", function () {
85
+ if (xhr.readyState == 4 && xhr.status == "200") {
86
+ store.updateNode(index, JSON.parse(xhr.responseText));
87
+ }
88
+ });
89
+ xhr.send(JSON.stringify(data));
90
+ }
91
+
92
+ updateNode(index, attributes) {
93
+ for (var attr in attributes) {
94
+ if (attributes.hasOwnProperty(attr)) {
95
+ index.node[attr] = attributes[attr];
96
+ }
97
+ }
98
+ this.setState({tree: tree});
99
+ }
100
+
101
+ getInitialState() {
102
+ return this.state.tree;
103
+ }
104
+
105
+ setCollapsed(id, value) {
106
+ var node = this.state.tree.getIndex(id).node;
107
+ node.collapsed = value;
108
+ this.storeCollapsed(id, node.collapsed);
109
+ this.state.tree.updateNodesPosition();
110
+ }
111
+
112
+ storeCollapsed(id, newState) {
113
+ let node = this.state.tree.getIndex(id).node;
114
+ var store = this.collapsedState();
115
+ store[node.id] = newState;
116
+ window.localStorage.collapsedPages = JSON.stringify(store);
117
+ }
118
+
119
+ reorderChildren(id) {
120
+ let tree = this.state.tree;
121
+ var index = this.state.tree.getIndex(id);
122
+ var node = index.node;
123
+ if (!node.news_page) {
124
+ return;
125
+ }
126
+ index.children = index.children.sort(function (a, b) {
127
+ var aNode = tree.getIndex(a).node;
128
+ var bNode = tree.getIndex(b).node;
129
+ if (aNode.pinned == bNode.pinned) {
130
+ return new Date(bNode.published_at) - new Date(aNode.published_at);
131
+ } else {
132
+ return aNode.pinned ? -1 : 1;
133
+ }
134
+ });
135
+ tree.updateNodesPosition();
136
+ }
137
+
138
+ onInit(props) {
139
+ let pages = props.pages;
140
+
141
+ // Build tree
142
+ let parentMap = pages.reduce((m, page) => {
143
+ let id = page.parent_page_id;
144
+ m[id] = [...(m[id] || []), page];
145
+ return m;
146
+ }, {});
147
+
148
+ pages.forEach((p) => p.children = parentMap[p.id] || []);
149
+
150
+ tree = new Tree({ name: "All Pages",
151
+ locale: props.locale,
152
+ permissions: props.permissions,
153
+ root: true,
154
+ children: parentMap[null] });
155
+ this.applyCollapsed(tree);
156
+ tree.updateNodesPosition();
157
+ this.setState({tree: tree});
158
+ }
159
+
160
+ onMovedPage(id) {
161
+ let tree = this.state.tree;
162
+ let index = tree.getIndex(id);
163
+ this.reorderChildren(index.parent);
164
+
165
+ parent = tree.getIndex(index.parent);
166
+ position = parent.children.indexOf(id) + 1;
167
+
168
+ this.movePage(index, parent, position);
169
+ this.setState({tree: tree});
170
+ }
171
+
172
+ onAddChild(id, attributes) {
173
+ let tree = this.state.tree;
174
+ var index = tree.append(attributes, id);
175
+ this.reorderChildren(id);
176
+ this.setCollapsed(id, false);
177
+ this.createPage(index, attributes);
178
+ this.setState({tree: tree});
179
+ }
180
+
181
+ onSetCSRFToken(token) {
182
+ this.setState({ csrf_token: token });
183
+ }
184
+
185
+ onToggleCollapsed(id) {
186
+ let tree = this.state.tree;
187
+ var node = tree.getIndex(id).node;
188
+ this.setCollapsed(id, !node.collapsed);
189
+ this.setState({tree: tree});
190
+ }
191
+
192
+ onUpdatePage(id, attributes) {
193
+ let tree = this.state.tree;
194
+ var index = tree.getIndex(id);
195
+ this.updateNode(index, attributes);
196
+ this.updatePage(index, attributes);
197
+ }
198
+
199
+ onUpdateTree(newTree) {
200
+ tree = newTree;
201
+ this.setState({tree: tree});
202
+ }
203
+ }