comfortable_media_surfer 3.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.codeclimate.yml +4 -0
- data/.gitattributes +2 -0
- data/.github/issue_template.md +19 -0
- data/.github/pull_request_template.md +6 -0
- data/.github/workflows/coveralls.yml +40 -0
- data/.github/workflows/rubyonrails.yml +51 -0
- data/.gitignore +21 -0
- data/.rubocop.yml +75 -0
- data/.travis.yml +44 -0
- data/CONTRIBUTING.md +35 -0
- data/Gemfile +38 -0
- data/LICENSE +20 -0
- data/README.md +109 -0
- data/Rakefile +5 -0
- data/app/assets/config/manifest.js +0 -0
- data/app/assets/fonts/comfy/admin/cms/lib/redactor-font.eot +0 -0
- data/app/assets/javascripts/comfy/admin/cms/application.js +39 -0
- data/app/assets/javascripts/comfy/admin/cms/base.js +46 -0
- data/app/assets/javascripts/comfy/admin/cms/categories.js +17 -0
- data/app/assets/javascripts/comfy/admin/cms/codemirror.js +31 -0
- data/app/assets/javascripts/comfy/admin/cms/custom.js +1 -0
- data/app/assets/javascripts/comfy/admin/cms/diff.js +10 -0
- data/app/assets/javascripts/comfy/admin/cms/file_link.js +67 -0
- data/app/assets/javascripts/comfy/admin/cms/file_upload.js +194 -0
- data/app/assets/javascripts/comfy/admin/cms/files_modal.js +40 -0
- data/app/assets/javascripts/comfy/admin/cms/page_fragments.js +22 -0
- data/app/assets/javascripts/comfy/admin/cms/slugify.js +34 -0
- data/app/assets/javascripts/comfy/admin/cms/sortable_list.js +40 -0
- data/app/assets/javascripts/comfy/admin/cms/timepicker.js +30 -0
- data/app/assets/javascripts/comfy/admin/cms/wysiwyg.js +72 -0
- data/app/assets/javascripts/comfy/vendor/Moxie.swf +0 -0
- data/app/assets/javascripts/comfy/vendor/Moxie.xap +0 -0
- data/app/assets/javascripts/comfy/vendor/bootstrap.bundle.min.js +7 -0
- data/app/assets/javascripts/comfy/vendor/bootstrap.bundle.min.js.map +1 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/addon/edit/closetag.js +169 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/mode/css/css.js +832 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/mode/htmlmixed/htmlmixed.js +152 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/mode/javascript/javascript.js +875 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/mode/markdown/markdown.js +861 -0
- data/app/assets/javascripts/comfy/vendor/codemirror/mode/xml/xml.js +394 -0
- data/app/assets/javascripts/comfy/vendor/codemirror.js +9653 -0
- data/app/assets/javascripts/comfy/vendor/diff/diff_match_patch.min.js +49 -0
- data/app/assets/javascripts/comfy/vendor/diff/pretty_text_diff.js +76 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ar.js +52 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/bg.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/bn.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ca.js +83 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/cs.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/cy.js +92 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/da.js +71 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/de.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/default.js +80 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/eo.js +73 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/es.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/et.js +73 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/fa.js +68 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/fi.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/fr.js +75 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/gr.js +74 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/he.js +57 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/hi.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/hr.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/hu.js +72 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/id.js +61 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/it.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ja.js +57 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ko.js +60 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/lt.js +72 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/lv.js +67 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/mk.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/mn.js +67 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ms.js +68 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/my.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/nl.js +75 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/no.js +73 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/pa.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/pl.js +68 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/pt-BR.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/pt.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ro.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/ru.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/si.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/sk.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/sl.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/sq.js +65 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/sr.js +69 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/sv.js +70 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/th.js +72 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/tr.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/uk.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/vn.js +66 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/zh-CN.js +61 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr/i18n/zh-TW.js +61 -0
- data/app/assets/javascripts/comfy/vendor/flatpickr.min.js +2 -0
- data/app/assets/javascripts/comfy/vendor/fontawesome.js +5 -0
- data/app/assets/javascripts/comfy/vendor/moxie.min.js +16 -0
- data/app/assets/javascripts/comfy/vendor/plupload.dev.js +2497 -0
- data/app/assets/javascripts/comfy/vendor/redactor/definedlinks.js +53 -0
- data/app/assets/javascripts/comfy/vendor/redactor/filemanager.js +64 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ar.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/az.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ba.js +79 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/bg.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/by.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ca.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/cs.js +87 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/da.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/de.js +80 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/el.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/en.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/eo.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/es.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/es_AR.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/fa.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/fi.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/fr.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ge.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/gr.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/he.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/hr.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/hu.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/id.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/it.js +78 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ja.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/kn.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ko.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/lt.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/lv.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/mk.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/nb.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/nl.js +80 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/pl.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/pt-BR.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/pt.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ro.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ru.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sk.js +76 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sl.js +79 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sq.js +79 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sr-CIR.js +79 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sr-LAT.js +79 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/sv.js +74 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/th.js +74 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/tr.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/ua.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/uk.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/vi.js +74 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/zh-CN.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/i18n/zh-TW.js +75 -0
- data/app/assets/javascripts/comfy/vendor/redactor/imagemanager.js +57 -0
- data/app/assets/javascripts/comfy/vendor/redactor/table.js +337 -0
- data/app/assets/javascripts/comfy/vendor/redactor/video.js +77 -0
- data/app/assets/javascripts/comfy/vendor/redactor.js +9561 -0
- data/app/assets/javascripts/comfy/vendor/sortable.min.js +5 -0
- data/app/assets/stylesheets/comfy/admin/cms/application.sass +11 -0
- data/app/assets/stylesheets/comfy/admin/cms/base.sass +256 -0
- data/app/assets/stylesheets/comfy/admin/cms/codemirror_overrides.sass +19 -0
- data/app/assets/stylesheets/comfy/admin/cms/custom.sass +1 -0
- data/app/assets/stylesheets/comfy/admin/cms/redactor_overrides.sass +22 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_alert.scss +51 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_badge.scss +54 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_breadcrumb.scss +41 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_button-group.scss +163 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_buttons.scss +137 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_card.scss +289 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_carousel.scss +197 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_close.scss +41 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_code.scss +48 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_custom-forms.scss +507 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_dropdown.scss +191 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_forms.scss +330 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_functions.scss +86 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_grid.scss +52 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_images.scss +42 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_input-group.scss +193 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_jumbotron.scss +17 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_list-group.scss +149 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_media.scss +8 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_mixins.scss +47 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_modal.scss +229 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_nav.scss +120 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_navbar.scss +294 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_pagination.scss +73 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_popover.scss +171 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_print.scss +141 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_progress.scss +43 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_reboot.scss +483 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_root.scss +19 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_spinners.scss +55 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_tables.scss +185 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_toasts.scss +44 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_tooltip.scss +115 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_transitions.scss +20 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_type.scss +125 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_utilities.scss +17 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/_variables.scss +1123 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/bootstrap-grid.scss +29 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/bootstrap-reboot.scss +12 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/bootstrap.scss +44 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_alert.scss +13 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_background-variant.scss +21 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_badge.scss +17 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_border-radius.scss +63 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_box-shadow.scss +20 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_breakpoints.scss +123 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_buttons.scss +107 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_caret.scss +62 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_clearfix.scss +7 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_deprecate.scss +10 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_float.scss +14 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_forms.scss +192 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_gradients.scss +45 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_grid-framework.scss +66 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_grid.scss +51 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_hover.scss +37 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_image.scss +36 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_list-group.scss +21 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_lists.scss +7 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_nav-divider.scss +10 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_pagination.scss +22 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_reset-text.scss +17 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_resize.scss +6 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_screen-reader.scss +33 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_size.scss +7 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_table-row.scss +39 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_text-emphasis.scss +16 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_text-hide.scss +11 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_text-truncate.scss +8 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_transition.scss +16 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/mixins/_visibility.scss +8 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_align.scss +8 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_background.scss +19 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_borders.scss +75 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_clearfix.scss +3 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_display.scss +26 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_embed.scss +39 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_flex.scss +51 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_float.scss +11 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_overflow.scss +5 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_position.scss +32 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_screenreaders.scss +11 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_shadows.scss +6 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_sizing.scss +20 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_spacing.scss +73 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_stretched-link.scss +19 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_text.scss +72 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/utilities/_visibility.scss +13 -0
- data/app/assets/stylesheets/comfy/vendor/bootstrap/vendor/_rfs.scss +204 -0
- data/app/assets/stylesheets/comfy/vendor/codemirror.css +346 -0
- data/app/assets/stylesheets/comfy/vendor/flatpickr.min.css +13 -0
- data/app/assets/stylesheets/comfy/vendor/fontawesome.css +343 -0
- data/app/assets/stylesheets/comfy/vendor/redactor.css +948 -0
- data/app/controllers/application_controller.rb +5 -0
- data/app/controllers/comfy/admin/base_controller.rb +20 -0
- data/app/controllers/comfy/admin/cms/base_controller.rb +54 -0
- data/app/controllers/comfy/admin/cms/categories_controller.rb +38 -0
- data/app/controllers/comfy/admin/cms/files_controller.rb +139 -0
- data/app/controllers/comfy/admin/cms/layouts_controller.rb +68 -0
- data/app/controllers/comfy/admin/cms/pages_controller.rb +163 -0
- data/app/controllers/comfy/admin/cms/revisions/base_controller.rb +48 -0
- data/app/controllers/comfy/admin/cms/revisions/layout_controller.rb +16 -0
- data/app/controllers/comfy/admin/cms/revisions/page_controller.rb +28 -0
- data/app/controllers/comfy/admin/cms/revisions/snippet_controller.rb +16 -0
- data/app/controllers/comfy/admin/cms/revisions/translation_controller.rb +29 -0
- data/app/controllers/comfy/admin/cms/sites_controller.rb +68 -0
- data/app/controllers/comfy/admin/cms/snippets_controller.rb +65 -0
- data/app/controllers/comfy/admin/cms/translations_controller.rb +105 -0
- data/app/controllers/comfy/cms/assets_controller.rb +38 -0
- data/app/controllers/comfy/cms/base_controller.rb +26 -0
- data/app/controllers/comfy/cms/content_controller.rb +90 -0
- data/app/controllers/concerns/comfy/paginate.rb +11 -0
- data/app/controllers/concerns/comfy/reorder_action.rb +20 -0
- data/app/helpers/comfy/admin/cms_helper.rb +50 -0
- data/app/helpers/comfy/cms_helper.rb +71 -0
- data/app/models/comfy/cms/categorization.rb +14 -0
- data/app/models/comfy/cms/category.rb +22 -0
- data/app/models/comfy/cms/file.rb +65 -0
- data/app/models/comfy/cms/fragment.rb +58 -0
- data/app/models/comfy/cms/layout.rb +114 -0
- data/app/models/comfy/cms/page.rb +173 -0
- data/app/models/comfy/cms/revision.rb +10 -0
- data/app/models/comfy/cms/site.rb +93 -0
- data/app/models/comfy/cms/snippet.rb +43 -0
- data/app/models/comfy/cms/translation.rb +42 -0
- data/app/models/concerns/comfy/cms/with_categories.rb +51 -0
- data/app/models/concerns/comfy/cms/with_fragments.rb +123 -0
- data/app/views/comfy/admin/cms/categories/_categories.html.haml +4 -0
- data/app/views/comfy/admin/cms/categories/_edit.html.haml +7 -0
- data/app/views/comfy/admin/cms/categories/_form.html.haml +5 -0
- data/app/views/comfy/admin/cms/categories/_index.html.haml +28 -0
- data/app/views/comfy/admin/cms/categories/_show.html.haml +15 -0
- data/app/views/comfy/admin/cms/categories/create.js.erb +7 -0
- data/app/views/comfy/admin/cms/categories/destroy.js.erb +8 -0
- data/app/views/comfy/admin/cms/categories/edit.js.erb +3 -0
- data/app/views/comfy/admin/cms/categories/update.js.erb +6 -0
- data/app/views/comfy/admin/cms/files/_file.html.haml +31 -0
- data/app/views/comfy/admin/cms/files/_form.html.haml +26 -0
- data/app/views/comfy/admin/cms/files/_modal.html.haml +3 -0
- data/app/views/comfy/admin/cms/files/destroy.js.erb +4 -0
- data/app/views/comfy/admin/cms/files/edit.html.haml +5 -0
- data/app/views/comfy/admin/cms/files/index.html.haml +31 -0
- data/app/views/comfy/admin/cms/files/new.html.haml +5 -0
- data/app/views/comfy/admin/cms/fragments/_form_fragment_attachments.html.haml +23 -0
- data/app/views/comfy/admin/cms/fragments/_form_fragments.html.haml +38 -0
- data/app/views/comfy/admin/cms/layouts/_form.html.haml +20 -0
- data/app/views/comfy/admin/cms/layouts/_index_branch.html.haml +26 -0
- data/app/views/comfy/admin/cms/layouts/edit.html.haml +9 -0
- data/app/views/comfy/admin/cms/layouts/index.html.haml +10 -0
- data/app/views/comfy/admin/cms/layouts/new.html.haml +5 -0
- data/app/views/comfy/admin/cms/pages/_form.html.haml +31 -0
- data/app/views/comfy/admin/cms/pages/_index_branch.html.haml +44 -0
- data/app/views/comfy/admin/cms/pages/edit.html.haml +10 -0
- data/app/views/comfy/admin/cms/pages/index.html.haml +13 -0
- data/app/views/comfy/admin/cms/pages/new.html.haml +5 -0
- data/app/views/comfy/admin/cms/pages/toggle_branch.js.erb +13 -0
- data/app/views/comfy/admin/cms/revisions/_sidebar.html.haml +2 -0
- data/app/views/comfy/admin/cms/revisions/show.html.haml +36 -0
- data/app/views/comfy/admin/cms/sites/_form.html.haml +15 -0
- data/app/views/comfy/admin/cms/sites/edit.html.haml +5 -0
- data/app/views/comfy/admin/cms/sites/index.html.haml +28 -0
- data/app/views/comfy/admin/cms/sites/new.html.haml +5 -0
- data/app/views/comfy/admin/cms/snippets/_form.html.haml +19 -0
- data/app/views/comfy/admin/cms/snippets/edit.html.haml +9 -0
- data/app/views/comfy/admin/cms/snippets/index.html.haml +33 -0
- data/app/views/comfy/admin/cms/snippets/new.html.haml +5 -0
- data/app/views/comfy/admin/cms/translations/_form.html.haml +20 -0
- data/app/views/comfy/admin/cms/translations/_sidebar.html.haml +19 -0
- data/app/views/comfy/admin/cms/translations/edit.html.haml +10 -0
- data/app/views/comfy/admin/cms/translations/new.html.haml +5 -0
- data/app/views/kaminari/comfy/_first_page.html.haml +3 -0
- data/app/views/kaminari/comfy/_gap.html.haml +2 -0
- data/app/views/kaminari/comfy/_last_page.html.haml +3 -0
- data/app/views/kaminari/comfy/_next_page.html.haml +2 -0
- data/app/views/kaminari/comfy/_page.html.haml +2 -0
- data/app/views/kaminari/comfy/_paginator.html.haml +11 -0
- data/app/views/kaminari/comfy/_prev_page.html.haml +2 -0
- data/app/views/layouts/comfy/admin/cms/_body.html.haml +14 -0
- data/app/views/layouts/comfy/admin/cms/_flash.html.haml +4 -0
- data/app/views/layouts/comfy/admin/cms/_footer_js.html.haml +6 -0
- data/app/views/layouts/comfy/admin/cms/_head.html.haml +19 -0
- data/app/views/layouts/comfy/admin/cms/_left.html.haml +46 -0
- data/app/views/layouts/comfy/admin/cms/_right.html.haml +5 -0
- data/app/views/layouts/comfy/admin/cms.html.haml +4 -0
- data/bin/bundle +3 -0
- data/bin/rails +4 -0
- data/bin/rake +4 -0
- data/bin/setup +36 -0
- data/bin/update +31 -0
- data/bin/yarn +11 -0
- data/comfortable_media_surfer.gemspec +37 -0
- data/config/application.rb +59 -0
- data/config/boot.rb +5 -0
- data/config/cable.yml +10 -0
- data/config/cms_routes.rb +8 -0
- data/config/database.yml +40 -0
- data/config/environment.rb +7 -0
- data/config/environments/development.rb +80 -0
- data/config/environments/production.rb +99 -0
- data/config/environments/test.rb +74 -0
- data/config/initializers/assets.rb +14 -0
- data/config/initializers/comfortable_media_surfer.rb +122 -0
- data/config/initializers/content_security_policy.rb +27 -0
- data/config/initializers/filter_parameter_logging.rb +10 -0
- data/config/initializers/inflections.rb +18 -0
- data/config/initializers/json.rb +9 -0
- data/config/initializers/new_framework_defaults_7_0.rb +144 -0
- data/config/initializers/new_framework_defaults_7_1.rb +225 -0
- data/config/initializers/permissions_policy.rb +15 -0
- data/config/locales/ar.yml +262 -0
- data/config/locales/ca.yml +262 -0
- data/config/locales/cs.yml +262 -0
- data/config/locales/da.yml +262 -0
- data/config/locales/de.yml +262 -0
- data/config/locales/en.yml +262 -0
- data/config/locales/es.yml +262 -0
- data/config/locales/fi.yml +262 -0
- data/config/locales/fr.yml +262 -0
- data/config/locales/gr.yml +262 -0
- data/config/locales/hr.yml +262 -0
- data/config/locales/it.yml +262 -0
- data/config/locales/ja.yml +262 -0
- data/config/locales/nb.yml +262 -0
- data/config/locales/nl.yml +262 -0
- data/config/locales/pl.yml +262 -0
- data/config/locales/pt-BR.yml +262 -0
- data/config/locales/ru.yml +262 -0
- data/config/locales/sk.yml +262 -0
- data/config/locales/sv.yml +262 -0
- data/config/locales/tr.yml +262 -0
- data/config/locales/uk.yml +262 -0
- data/config/locales/zh-CN.yml +262 -0
- data/config/locales/zh-TW.yml +262 -0
- data/config/storage.yml +35 -0
- data/config.ru +6 -0
- data/db/cms_seeds/sample-site/files/_default.jpg.yml +5 -0
- data/db/cms_seeds/sample-site/files/default.jpg +0 -0
- data/db/cms_seeds/sample-site/layouts/default/content.html +16 -0
- data/db/cms_seeds/sample-site/layouts/default/nested/content.html +14 -0
- data/db/cms_seeds/sample-site/pages/index/child_a/content.html +7 -0
- data/db/cms_seeds/sample-site/pages/index/child_b/content.html +6 -0
- data/db/cms_seeds/sample-site/pages/index/cms logo.png +0 -0
- data/db/cms_seeds/sample-site/pages/index/content.es.html +6 -0
- data/db/cms_seeds/sample-site/pages/index/content.fr.html +6 -0
- data/db/cms_seeds/sample-site/pages/index/content.html +25 -0
- data/db/cms_seeds/sample-site/pages/index/footer.png +0 -0
- data/db/cms_seeds/sample-site/pages/index/header.png +0 -0
- data/db/cms_seeds/sample-site/snippets/default.html +8 -0
- data/db/migrate/00_create_active_storage_tables.active_storage.rb +50 -0
- data/db/migrate/01_create_cms.rb +143 -0
- data/gemfiles/6.1.gemfile +31 -0
- data/gemfiles/7.0.gemfile +31 -0
- data/gemfiles/7.1.gemfile +31 -0
- data/gemfiles/7.2.gemfile +31 -0
- data/lib/comfortable_media_surfer/access_control/admin_authentication.rb +22 -0
- data/lib/comfortable_media_surfer/access_control/admin_authorization.rb +10 -0
- data/lib/comfortable_media_surfer/access_control/public_authentication.rb +10 -0
- data/lib/comfortable_media_surfer/access_control/public_authorization.rb +10 -0
- data/lib/comfortable_media_surfer/configuration.rb +138 -0
- data/lib/comfortable_media_surfer/content/block.rb +12 -0
- data/lib/comfortable_media_surfer/content/params_parser.rb +149 -0
- data/lib/comfortable_media_surfer/content/renderer.rb +146 -0
- data/lib/comfortable_media_surfer/content/tag.rb +48 -0
- data/lib/comfortable_media_surfer/content/tags/asset.rb +57 -0
- data/lib/comfortable_media_surfer/content/tags/audio.rb +39 -0
- data/lib/comfortable_media_surfer/content/tags/breadcrumbs.rb +43 -0
- data/lib/comfortable_media_surfer/content/tags/checkbox.rb +27 -0
- data/lib/comfortable_media_surfer/content/tags/children.rb +52 -0
- data/lib/comfortable_media_surfer/content/tags/date.rb +19 -0
- data/lib/comfortable_media_surfer/content/tags/datetime.rb +42 -0
- data/lib/comfortable_media_surfer/content/tags/file.rb +70 -0
- data/lib/comfortable_media_surfer/content/tags/file_link.rb +58 -0
- data/lib/comfortable_media_surfer/content/tags/files.rb +41 -0
- data/lib/comfortable_media_surfer/content/tags/fragment.rb +64 -0
- data/lib/comfortable_media_surfer/content/tags/helper.rb +53 -0
- data/lib/comfortable_media_surfer/content/tags/image.rb +56 -0
- data/lib/comfortable_media_surfer/content/tags/markdown.rb +22 -0
- data/lib/comfortable_media_surfer/content/tags/mixins/file_content.rb +42 -0
- data/lib/comfortable_media_surfer/content/tags/number.rb +18 -0
- data/lib/comfortable_media_surfer/content/tags/page_file_link.rb +86 -0
- data/lib/comfortable_media_surfer/content/tags/partial.rb +47 -0
- data/lib/comfortable_media_surfer/content/tags/siblings.rb +66 -0
- data/lib/comfortable_media_surfer/content/tags/snippet.rb +36 -0
- data/lib/comfortable_media_surfer/content/tags/template.rb +42 -0
- data/lib/comfortable_media_surfer/content/tags/text.rb +18 -0
- data/lib/comfortable_media_surfer/content/tags/textarea.rb +19 -0
- data/lib/comfortable_media_surfer/content/tags/wysiwyg.rb +19 -0
- data/lib/comfortable_media_surfer/content/tags.rb +29 -0
- data/lib/comfortable_media_surfer/content.rb +11 -0
- data/lib/comfortable_media_surfer/engine.rb +29 -0
- data/lib/comfortable_media_surfer/error.rb +24 -0
- data/lib/comfortable_media_surfer/extensions/acts_as_tree.rb +114 -0
- data/lib/comfortable_media_surfer/extensions/has_revisions.rb +68 -0
- data/lib/comfortable_media_surfer/extensions.rb +8 -0
- data/lib/comfortable_media_surfer/form_builder.rb +43 -0
- data/lib/comfortable_media_surfer/render_methods.rb +106 -0
- data/lib/comfortable_media_surfer/routing.rb +79 -0
- data/lib/comfortable_media_surfer/seeds/file/exporter.rb +37 -0
- data/lib/comfortable_media_surfer/seeds/file/importer.rb +59 -0
- data/lib/comfortable_media_surfer/seeds/layout/exporter.rb +38 -0
- data/lib/comfortable_media_surfer/seeds/layout/importer.rb +60 -0
- data/lib/comfortable_media_surfer/seeds/page/exporter.rb +88 -0
- data/lib/comfortable_media_surfer/seeds/page/importer.rb +217 -0
- data/lib/comfortable_media_surfer/seeds/snippet/exporter.rb +32 -0
- data/lib/comfortable_media_surfer/seeds/snippet/importer.rb +47 -0
- data/lib/comfortable_media_surfer/seeds.rb +112 -0
- data/lib/comfortable_media_surfer/version.rb +5 -0
- data/lib/comfortable_media_surfer/view_hooks.rb +33 -0
- data/lib/comfortable_media_surfer.rb +56 -0
- data/lib/generators/comfy/cms/README +28 -0
- data/lib/generators/comfy/cms/assets_generator.rb +16 -0
- data/lib/generators/comfy/cms/cms_generator.rb +67 -0
- data/lib/generators/comfy/cms/controllers_generator.rb +15 -0
- data/lib/generators/comfy/cms/models_generator.rb +15 -0
- data/lib/generators/comfy/cms/views_generator.rb +15 -0
- data/lib/generators/comfy/scaffold/scaffold_generator.rb +79 -0
- data/lib/generators/comfy/scaffold/templates/controller.rb.tt +62 -0
- data/lib/generators/comfy/scaffold/templates/migration.rb.tt +11 -0
- data/lib/generators/comfy/scaffold/templates/model.rb.tt +21 -0
- data/lib/generators/comfy/scaffold/templates/tests/controller.rb.tt +115 -0
- data/lib/generators/comfy/scaffold/templates/tests/fixture.yml.tt +4 -0
- data/lib/generators/comfy/scaffold/templates/tests/model.rb.tt +26 -0
- data/lib/generators/comfy/scaffold/templates/views/_form.haml.tt +7 -0
- data/lib/generators/comfy/scaffold/templates/views/edit.haml.tt +5 -0
- data/lib/generators/comfy/scaffold/templates/views/index.haml.tt +21 -0
- data/lib/generators/comfy/scaffold/templates/views/new.haml.tt +5 -0
- data/lib/generators/comfy/scaffold/templates/views/show.haml.tt +4 -0
- data/lib/tasks/cms_seeds.rake +40 -0
- data/public/favicon.ico +0 -0
- data/rakelib/create_release.rake +81 -0
- metadata +797 -0
@@ -0,0 +1,106 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::RenderMethods
|
4
|
+
def self.included(base)
|
5
|
+
# If application controller doesn't have template associated with it
|
6
|
+
# CMS will attempt to find one. This is so you don't have to explicitly
|
7
|
+
# call render cms_page: '/something'
|
8
|
+
base.rescue_from 'ActionView::MissingTemplate' do |e|
|
9
|
+
render cms_page: request.path
|
10
|
+
rescue ComfortableMediaSurfer::MissingPage, ComfortableMediaSurfer::MissingSite
|
11
|
+
raise e
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
# Now you can render cms_page simply by calling:
|
16
|
+
# render cms_page: '/path/to/page'
|
17
|
+
# This way application controllers can use CMS content while populating
|
18
|
+
# instance variables that can be used in partials (that are included by
|
19
|
+
# by the cms page and/or layout)
|
20
|
+
#
|
21
|
+
# Or how about not worrying about setting up CMS pages and rendering
|
22
|
+
# application view using a CMS layout?
|
23
|
+
# render cms_layout: 'layout_slug', cms_fragments: {
|
24
|
+
# fragment_identifier_a: 'content text',
|
25
|
+
# fragment_identifier_b: {template: 'path/to/template' },
|
26
|
+
# fragment_identifier_c: {partial: 'path/to/partial' }
|
27
|
+
# }
|
28
|
+
#
|
29
|
+
# This way you are populating page block content and rendering
|
30
|
+
# an instantialized CMS page.
|
31
|
+
#
|
32
|
+
# Site is loaded automatically based on the request. However you can force
|
33
|
+
# it by passing :cms_site parameter with site's slug. For example:
|
34
|
+
# render cms_page: '/path/to/page', cms_site: 'default'
|
35
|
+
#
|
36
|
+
def render(options = {}, locals = {}, &block)
|
37
|
+
return super unless options.is_a?(Hash)
|
38
|
+
|
39
|
+
if (site_identifier = options.delete(:cms_site)) && !(@cms_site = Comfy::Cms::Site.find_by_identifier(site_identifier))
|
40
|
+
raise ComfortableMediaSurfer::MissingSite, site_identifier
|
41
|
+
end
|
42
|
+
|
43
|
+
if ((page_path = options.delete(:cms_page)) || (layout_identifier = options.delete(:cms_layout))) && !@cms_site ||= Comfy::Cms::Site.find_site(
|
44
|
+
request.host_with_port.downcase, request.fullpath
|
45
|
+
)
|
46
|
+
raise ComfortableMediaSurfer::MissingSite, "#{request.host.downcase}/#{request.fullpath}"
|
47
|
+
end
|
48
|
+
|
49
|
+
if page_path
|
50
|
+
render_cms_page(page_path, options, locals, &block)
|
51
|
+
elsif layout_identifier
|
52
|
+
render_cms_layout(layout_identifier, options, locals, &block)
|
53
|
+
else
|
54
|
+
super
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
def render_cms_page(path, options = {}, locals = {}, &block)
|
59
|
+
path.gsub!(%r{^/#{@cms_site.path}}, '') if @cms_site.path.present?
|
60
|
+
|
61
|
+
unless (@cms_page = @cms_site.pages.find_by_full_path(path))
|
62
|
+
raise ComfortableMediaSurfer::MissingPage, path
|
63
|
+
end
|
64
|
+
|
65
|
+
@cms_page.translate!
|
66
|
+
|
67
|
+
@cms_layout = @cms_page.layout
|
68
|
+
if (cms_fragments = options.delete(:cms_fragments)).present?
|
69
|
+
cms_fragments.each do |identifier, value|
|
70
|
+
content = value.is_a?(Hash) ? render_to_string(value.merge(layout: false)) : value.to_s
|
71
|
+
page_fragment = @cms_page.fragments.detect { |f| f.identifier == identifier.to_s } ||
|
72
|
+
@cms_page.fragments.build(identifier: identifier.to_s)
|
73
|
+
page_fragment.content = content
|
74
|
+
end
|
75
|
+
end
|
76
|
+
cms_app_layout = @cms_layout.app_layout
|
77
|
+
options[:layout] ||= cms_app_layout.blank? ? nil : cms_app_layout
|
78
|
+
options[:inline] = @cms_page.render
|
79
|
+
|
80
|
+
render(options, locals, &block)
|
81
|
+
end
|
82
|
+
|
83
|
+
def render_cms_layout(identifier, options = {}, locals = {}, &block)
|
84
|
+
unless (@cms_layout = @cms_site.layouts.find_by_identifier(identifier))
|
85
|
+
raise ComfortableMediaSurfer::MissingLayout, identifier
|
86
|
+
end
|
87
|
+
|
88
|
+
cms_app_layout = @cms_layout.app_layout
|
89
|
+
cms_page = @cms_site.pages.build(layout: @cms_layout)
|
90
|
+
cms_fragments =
|
91
|
+
options.delete(:cms_fragments) || { content: render_to_string({ layout: false }.merge(options)) }
|
92
|
+
|
93
|
+
cms_fragments.each do |frag_identifier, value|
|
94
|
+
content = value.is_a?(Hash) ? render_to_string(value.merge(layout: false)) : value.to_s
|
95
|
+
cms_page.fragments.build(identifier: frag_identifier.to_s, content: content)
|
96
|
+
end
|
97
|
+
options[:layout] ||= cms_app_layout.blank? ? nil : cms_app_layout
|
98
|
+
options[:inline] = cms_page.render
|
99
|
+
|
100
|
+
render(options, locals, &block)
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
ActiveSupport.on_load :action_controller_base do
|
105
|
+
include ComfortableMediaSurfer::RenderMethods
|
106
|
+
end
|
@@ -0,0 +1,79 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Routing
|
4
|
+
class ActionDispatch::Routing::Mapper
|
5
|
+
def comfy_route_cms_admin(path: 'admin')
|
6
|
+
scope module: :comfy, as: :comfy do
|
7
|
+
scope module: :admin do
|
8
|
+
namespace :cms, as: :admin_cms, path: path, except: :show do
|
9
|
+
get '/', to: 'base#jump'
|
10
|
+
|
11
|
+
concern :with_revisions do |options|
|
12
|
+
resources :revisions, options.merge(only: %i[index show]) do
|
13
|
+
patch :revert, on: :member
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
concern :with_reorder do
|
18
|
+
put :reorder, on: :collection
|
19
|
+
end
|
20
|
+
|
21
|
+
concern :with_form_fragments do
|
22
|
+
get :form_fragments, on: :member
|
23
|
+
end
|
24
|
+
|
25
|
+
resources :sites do
|
26
|
+
resources :pages do
|
27
|
+
concerns :with_reorder
|
28
|
+
concerns :with_form_fragments
|
29
|
+
concerns :with_revisions, controller: 'revisions/page'
|
30
|
+
|
31
|
+
get :toggle_branch, on: :member
|
32
|
+
|
33
|
+
resources :translations, except: [:index] do
|
34
|
+
concerns :with_form_fragments
|
35
|
+
concerns :with_revisions, controller: 'revisions/translation'
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
39
|
+
resources :files, concerns: [:with_reorder]
|
40
|
+
|
41
|
+
resources :layouts do
|
42
|
+
concerns :with_reorder
|
43
|
+
concerns :with_revisions, controller: 'revisions/layout'
|
44
|
+
end
|
45
|
+
|
46
|
+
resources :snippets do
|
47
|
+
concerns :with_reorder
|
48
|
+
concerns :with_revisions, controller: 'revisions/snippet'
|
49
|
+
end
|
50
|
+
|
51
|
+
resources :categories
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
class ActionDispatch::Routing::Mapper
|
60
|
+
def comfy_route_cms(options = {})
|
61
|
+
ComfortableMediaSurfer.configuration.public_cms_path = options[:path]
|
62
|
+
|
63
|
+
scope module: :comfy, as: :comfy do
|
64
|
+
namespace :cms, path: options[:path] do
|
65
|
+
get 'cms-css/:site_id/:identifier(/:cache_buster)' => 'assets#render_css', as: 'render_css'
|
66
|
+
get 'cms-js/:site_id/:identifier(/:cache_buster)' => 'assets#render_js', as: 'render_js'
|
67
|
+
|
68
|
+
get '(*cms_path)' => 'content#show', as: 'render_page', action: '/:format'
|
69
|
+
end
|
70
|
+
end
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
class ActionDispatch::Routing::Mapper
|
75
|
+
def comfy_route(identifier, options = {})
|
76
|
+
send("comfy_route_#{identifier}", **options)
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::File
|
4
|
+
class Exporter < ComfortableMediaSurfer::Seeds::Exporter
|
5
|
+
def initialize(from, to = from)
|
6
|
+
super
|
7
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, to, 'files/')
|
8
|
+
end
|
9
|
+
|
10
|
+
def export!
|
11
|
+
prepare_folder!(path)
|
12
|
+
|
13
|
+
site.files.each do |file|
|
14
|
+
file_path = File.join(path, file.attachment.filename.to_s)
|
15
|
+
|
16
|
+
# writing attributes
|
17
|
+
::File.write(::File.join(path, "_#{file.attachment.filename}.yml"), {
|
18
|
+
'label' => file.label,
|
19
|
+
'description' => file.description,
|
20
|
+
'categories' => file.categories.map(&:label)
|
21
|
+
}.to_yaml)
|
22
|
+
|
23
|
+
# writing content
|
24
|
+
begin
|
25
|
+
::File.binwrite(::File.join(path, ::File.basename(file_path)), file.attachment.download)
|
26
|
+
rescue Errno::ENOENT, OpenURI::HTTPError
|
27
|
+
message = "[CMS SEEDS] No physical File \t #{file.attachment.filename}"
|
28
|
+
ComfortableMediaSurfer.logger.warn(message)
|
29
|
+
next
|
30
|
+
end
|
31
|
+
|
32
|
+
message = "[CMS SEEDS] Exported File \t #{file.attachment.filename}"
|
33
|
+
ComfortableMediaSurfer.logger.info(message)
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
@@ -0,0 +1,59 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::File
|
4
|
+
class Importer < ComfortableMediaSurfer::Seeds::Importer
|
5
|
+
def initialize(from, to = from)
|
6
|
+
super
|
7
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, from, 'files/')
|
8
|
+
end
|
9
|
+
|
10
|
+
def import!
|
11
|
+
Dir["#{path}[^_]*"].each do |file_path|
|
12
|
+
filename = ::File.basename(file_path)
|
13
|
+
|
14
|
+
file = site.files.with_attached_attachment
|
15
|
+
.where('active_storage_blobs.filename' => filename).references(:blob).first ||
|
16
|
+
site.files.new
|
17
|
+
|
18
|
+
# We need to track actual file and its attributes
|
19
|
+
fresh_file = false
|
20
|
+
|
21
|
+
if File.exist?(attrs_path = File.join(path, "_#{filename}.yml")) && fresh_seed?(file, attrs_path)
|
22
|
+
fresh_file = true
|
23
|
+
|
24
|
+
attrs = YAML.safe_load_file(attrs_path)
|
25
|
+
category_ids = category_names_to_ids(file, attrs.delete('categories'))
|
26
|
+
file.attributes = attrs.merge(
|
27
|
+
category_ids: category_ids
|
28
|
+
)
|
29
|
+
end
|
30
|
+
|
31
|
+
if fresh_seed?(file, file_path)
|
32
|
+
fresh_file = true
|
33
|
+
|
34
|
+
file_handler = File.open(file_path)
|
35
|
+
file.file = {
|
36
|
+
io: file_handler,
|
37
|
+
filename: filename,
|
38
|
+
content_type: MimeMagic.by_magic(file_handler)
|
39
|
+
}
|
40
|
+
end
|
41
|
+
|
42
|
+
if fresh_file
|
43
|
+
if file.save
|
44
|
+
message = "[CMS SEEDS] Imported File \t #{file_path}"
|
45
|
+
ComfortableMediaSurfer.logger.info(message)
|
46
|
+
else
|
47
|
+
message = "[CMS SEEDS] Failed to import File \n#{file.errors.inspect}"
|
48
|
+
ComfortableMediaSurfer.logger.warn(message)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
seed_ids << file.id
|
53
|
+
end
|
54
|
+
|
55
|
+
# cleaning up
|
56
|
+
site.files.where('id NOT IN (?)', seed_ids).destroy_all
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
@@ -0,0 +1,38 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::Layout
|
4
|
+
class Exporter < ComfortableMediaSurfer::Seeds::Exporter
|
5
|
+
def initialize(from, to = from)
|
6
|
+
super
|
7
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, to, 'layouts/')
|
8
|
+
end
|
9
|
+
|
10
|
+
def export!
|
11
|
+
prepare_folder!(path)
|
12
|
+
|
13
|
+
site.layouts.each do |layout|
|
14
|
+
layout_path = File.join(path, layout.ancestors.reverse.collect(&:identifier), layout.identifier)
|
15
|
+
FileUtils.mkdir_p(layout_path)
|
16
|
+
|
17
|
+
path = ::File.join(layout_path, 'content.html')
|
18
|
+
data = []
|
19
|
+
|
20
|
+
attrs = {
|
21
|
+
'label' => layout.label,
|
22
|
+
'app_layout' => layout.app_layout,
|
23
|
+
'position' => layout.position
|
24
|
+
}.to_yaml
|
25
|
+
|
26
|
+
data << { header: 'attributes', content: attrs }
|
27
|
+
data << { header: 'content', content: layout.content }
|
28
|
+
data << { header: 'js', content: layout.js }
|
29
|
+
data << { header: 'css', content: layout.css }
|
30
|
+
|
31
|
+
write_file_content(path, data)
|
32
|
+
|
33
|
+
message = "[CMS SEEDS] Exported Layout \t #{layout.identifier}"
|
34
|
+
ComfortableMediaSurfer.logger.info(message)
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -0,0 +1,60 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::Layout
|
4
|
+
class Importer < ComfortableMediaSurfer::Seeds::Importer
|
5
|
+
def initialize(from, to = from)
|
6
|
+
super
|
7
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, from, 'layouts/')
|
8
|
+
end
|
9
|
+
|
10
|
+
def import!(path = self.path, parent = nil)
|
11
|
+
Dir["#{path}*/"].each do |layout_path|
|
12
|
+
import_layout(layout_path, parent)
|
13
|
+
end
|
14
|
+
|
15
|
+
# cleaning up
|
16
|
+
site.layouts.where('id NOT IN (?)', seed_ids).destroy_all
|
17
|
+
end
|
18
|
+
|
19
|
+
private
|
20
|
+
|
21
|
+
def import_layout(path, parent)
|
22
|
+
identifier = path.split('/').last
|
23
|
+
|
24
|
+
# reading file content in, resulting in a hash
|
25
|
+
content_path = File.join(path, 'content.html')
|
26
|
+
content_hash = parse_file_content(content_path)
|
27
|
+
|
28
|
+
# parsing attributes section
|
29
|
+
attributes_yaml = content_hash.delete('attributes')
|
30
|
+
attrs = YAML.safe_load(attributes_yaml)
|
31
|
+
|
32
|
+
layout = site.layouts.where(identifier: identifier).first_or_initialize
|
33
|
+
layout.parent = parent
|
34
|
+
|
35
|
+
if fresh_seed?(layout, content_path)
|
36
|
+
layout.attributes = attrs.merge(
|
37
|
+
app_layout: attrs['app_layout'] || parent.try(:app_layout),
|
38
|
+
content: content_hash['content'],
|
39
|
+
js: content_hash['js'],
|
40
|
+
css: content_hash['css']
|
41
|
+
)
|
42
|
+
|
43
|
+
if layout.save
|
44
|
+
message = "[CMS SEEDS] Imported Layout \t #{layout.identifier}"
|
45
|
+
ComfortableMediaSurfer.logger.info(message)
|
46
|
+
else
|
47
|
+
message = "[CMS SEEDS] Failed to import Layout \n#{layout.errors.inspect}"
|
48
|
+
ComfortableMediaSurfer.logger.warn(message)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
seed_ids << layout.id
|
53
|
+
|
54
|
+
# importing child layouts (if there are any)
|
55
|
+
Dir["#{path}*/"].each do |layout_path|
|
56
|
+
import_layout(layout_path, layout)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
@@ -0,0 +1,88 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::Page
|
4
|
+
class Exporter < ComfortableMediaSurfer::Seeds::Exporter
|
5
|
+
def initialize(from, to = from)
|
6
|
+
super
|
7
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, to, 'pages/')
|
8
|
+
end
|
9
|
+
|
10
|
+
def export!
|
11
|
+
prepare_folder!(path)
|
12
|
+
|
13
|
+
site.pages.each do |page|
|
14
|
+
page.slug = 'index' if page.slug.blank?
|
15
|
+
page_path = File.join(path, page.ancestors.reverse.map { |p| p.slug.blank? ? 'index' : p.slug }, page.slug)
|
16
|
+
FileUtils.mkdir_p(page_path)
|
17
|
+
|
18
|
+
path = ::File.join(page_path, 'content.html')
|
19
|
+
data = []
|
20
|
+
|
21
|
+
attrs = {
|
22
|
+
'label' => page.label,
|
23
|
+
'layout' => page.layout.try(:identifier),
|
24
|
+
'target_page' => page.target_page.try(:full_path),
|
25
|
+
'categories' => page.categories.map(&:label),
|
26
|
+
'is_published' => page.is_published,
|
27
|
+
'position' => page.position
|
28
|
+
}.to_yaml
|
29
|
+
|
30
|
+
data << { header: 'attributes', content: attrs }
|
31
|
+
data += fragments_data(page, page_path)
|
32
|
+
|
33
|
+
write_file_content(path, data)
|
34
|
+
|
35
|
+
message = "[CMS SEEDS] Exported Page \t #{page.full_path}"
|
36
|
+
ComfortableMediaSurfer.logger.info(message)
|
37
|
+
|
38
|
+
export_translations(page, page_path)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
private
|
43
|
+
|
44
|
+
def export_translations(page, page_path)
|
45
|
+
page.translations.each do |translation|
|
46
|
+
path = ::File.join(page_path, "content.#{translation.locale}.html")
|
47
|
+
data = []
|
48
|
+
|
49
|
+
attrs = {
|
50
|
+
'label' => translation.label,
|
51
|
+
'layout' => translation.layout.try(:identifier),
|
52
|
+
'is_published' => page.is_published
|
53
|
+
}.to_yaml
|
54
|
+
|
55
|
+
data << { header: 'attributes', content: attrs }
|
56
|
+
data += fragments_data(translation, page_path)
|
57
|
+
|
58
|
+
write_file_content(path, data)
|
59
|
+
|
60
|
+
message = "[CMS SEEDS] Exported Translation \t #{translation.locale}"
|
61
|
+
ComfortableMediaSurfer.logger.info(message)
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
# Collecting fragment data and writing attachment files to disk
|
66
|
+
def fragments_data(record, page_path)
|
67
|
+
record.fragments.collect do |frag|
|
68
|
+
header = "#{frag.tag} #{frag.identifier}"
|
69
|
+
content =
|
70
|
+
case frag.tag
|
71
|
+
when 'datetime', 'date'
|
72
|
+
frag.datetime
|
73
|
+
when 'checkbox'
|
74
|
+
frag.boolean
|
75
|
+
when 'file', 'files'
|
76
|
+
frag.attachments.map do |attachment|
|
77
|
+
::File.binwrite(::File.join(page_path, attachment.filename.to_s), attachment.download)
|
78
|
+
attachment.filename
|
79
|
+
end.join("\n")
|
80
|
+
else
|
81
|
+
frag.content
|
82
|
+
end
|
83
|
+
|
84
|
+
{ header: header, content: content }
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
@@ -0,0 +1,217 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ComfortableMediaSurfer::Seeds::Page
|
4
|
+
class Importer < ComfortableMediaSurfer::Seeds::Importer
|
5
|
+
# tracking target page linking. Since we might be linking to something that
|
6
|
+
# doesn't exist yet, we'll defer linking to the end of import
|
7
|
+
attr_accessor :target_pages
|
8
|
+
|
9
|
+
def initialize(from, to = from)
|
10
|
+
super
|
11
|
+
self.path = ::File.join(ComfortableMediaSurfer.config.seeds_path, from, 'pages/')
|
12
|
+
end
|
13
|
+
|
14
|
+
def import!
|
15
|
+
import_page(File.join(path, 'index/'), nil)
|
16
|
+
|
17
|
+
link_target_pages
|
18
|
+
|
19
|
+
# Remove pages not found in seeds
|
20
|
+
site.pages.where('id NOT IN (?)', seed_ids).destroy_all
|
21
|
+
end
|
22
|
+
|
23
|
+
private
|
24
|
+
|
25
|
+
# Recursive function that will be called for each child page (subfolder)
|
26
|
+
def import_page(path, parent)
|
27
|
+
slug = path.split('/').last
|
28
|
+
|
29
|
+
# setting page record
|
30
|
+
page =
|
31
|
+
if parent.present?
|
32
|
+
child = site.pages.where(slug: slug).first_or_initialize
|
33
|
+
child.parent = parent
|
34
|
+
child
|
35
|
+
else
|
36
|
+
site.pages.root || site.pages.new(slug: slug)
|
37
|
+
end
|
38
|
+
|
39
|
+
content_path = File.join(path, 'content.html')
|
40
|
+
|
41
|
+
# If file is newer than page record we'll process it
|
42
|
+
if fresh_seed?(page, content_path)
|
43
|
+
|
44
|
+
# reading file content in, resulting in a hash
|
45
|
+
fragments_hash = parse_file_content(content_path)
|
46
|
+
|
47
|
+
# parsing attributes section
|
48
|
+
attributes_yaml = fragments_hash.delete('attributes')
|
49
|
+
attrs = YAML.safe_load(attributes_yaml)
|
50
|
+
|
51
|
+
# applying attributes
|
52
|
+
layout = site.layouts.find_by(identifier: attrs.delete('layout')) || parent.try(:layout)
|
53
|
+
category_ids = category_names_to_ids(page, attrs.delete('categories'))
|
54
|
+
target_page = attrs.delete('target_page')
|
55
|
+
|
56
|
+
page.attributes = attrs.merge(
|
57
|
+
layout: layout,
|
58
|
+
category_ids: category_ids
|
59
|
+
)
|
60
|
+
|
61
|
+
# applying fragments
|
62
|
+
old_frag_identifiers = page.fragments.pluck(:identifier)
|
63
|
+
|
64
|
+
new_frag_identifiers, fragments_attributes =
|
65
|
+
construct_fragments_attributes(fragments_hash, page, path)
|
66
|
+
page.fragments_attributes = fragments_attributes
|
67
|
+
|
68
|
+
if page.save
|
69
|
+
message = "[CMS SEEDS] Imported Page \t #{page.full_path}"
|
70
|
+
ComfortableMediaSurfer.logger.info(message)
|
71
|
+
|
72
|
+
# defering target page linking
|
73
|
+
if target_page.present?
|
74
|
+
self.target_pages ||= {}
|
75
|
+
self.target_pages[page.id] = target_page
|
76
|
+
end
|
77
|
+
|
78
|
+
# cleaning up old fragments
|
79
|
+
page.fragments.where(identifier: old_frag_identifiers - new_frag_identifiers).destroy_all
|
80
|
+
|
81
|
+
else
|
82
|
+
message = "[CMS SEEDS] Failed to import Page \n#{page.errors.inspect}"
|
83
|
+
ComfortableMediaSurfer.logger.warn(message)
|
84
|
+
end
|
85
|
+
end
|
86
|
+
|
87
|
+
import_translations(path, page)
|
88
|
+
|
89
|
+
# Tracking what page from seeds we're working with. So we can remove pages
|
90
|
+
# that are no longer in seeds
|
91
|
+
seed_ids << page.id
|
92
|
+
|
93
|
+
# importing child pages (if there are any)
|
94
|
+
Dir["#{path}*/"].each do |page_path|
|
95
|
+
import_page(page_path, page)
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
# Importing translations for given page. They look like `content.locale.html`
|
100
|
+
def import_translations(path, page)
|
101
|
+
old_translations = page.translations.pluck(:locale)
|
102
|
+
new_translations = []
|
103
|
+
|
104
|
+
Dir["#{path}content.*.html"].each do |file_path|
|
105
|
+
locale = File.basename(file_path).match(%r{content\.(\w+)\.html})[1]
|
106
|
+
new_translations << locale
|
107
|
+
|
108
|
+
translation = page.translations.where(locale: locale).first_or_initialize
|
109
|
+
|
110
|
+
next unless fresh_seed?(translation, file_path)
|
111
|
+
|
112
|
+
# reading file content in, resulting in a hash
|
113
|
+
fragments_hash = parse_file_content(file_path)
|
114
|
+
|
115
|
+
# parsing attributes section
|
116
|
+
attributes_yaml = fragments_hash.delete('attributes')
|
117
|
+
attrs = YAML.safe_load(attributes_yaml)
|
118
|
+
|
119
|
+
# applying attributes
|
120
|
+
layout = site.layouts.find_by(identifier: attrs.delete('layout')) || page.try(:layout)
|
121
|
+
translation.attributes = attrs.merge(
|
122
|
+
layout: layout
|
123
|
+
)
|
124
|
+
|
125
|
+
# applying fragments
|
126
|
+
old_frag_identifiers = translation.fragments.pluck(:identifier)
|
127
|
+
|
128
|
+
new_frag_identifiers, fragments_attributes =
|
129
|
+
construct_fragments_attributes(fragments_hash, translation, path)
|
130
|
+
translation.fragments_attributes = fragments_attributes
|
131
|
+
|
132
|
+
if translation.save
|
133
|
+
message = "[CMS SEEDS] Imported Translation \t #{locale}"
|
134
|
+
ComfortableMediaSurfer.logger.info(message)
|
135
|
+
|
136
|
+
# cleaning up old fragments
|
137
|
+
frags_to_remove = old_frag_identifiers - new_frag_identifiers
|
138
|
+
translation.fragments.where(identifier: frags_to_remove).destroy_all
|
139
|
+
|
140
|
+
else
|
141
|
+
message = "[CMS SEEDS] Failed to import Translation \n#{locale}"
|
142
|
+
ComfortableMediaSurfer.logger.warn(message)
|
143
|
+
end
|
144
|
+
end
|
145
|
+
|
146
|
+
# Cleaning up removed translations
|
147
|
+
translations_to_remove = old_translations - new_translations
|
148
|
+
page.translations.where(locale: translations_to_remove).destroy_all
|
149
|
+
end
|
150
|
+
|
151
|
+
# Constructing frag attributes hash that can be assigned to page or translation
|
152
|
+
# also returning list of frag identifiers so we can destroy old ones
|
153
|
+
def construct_fragments_attributes(hash, record, path)
|
154
|
+
frag_identifiers = []
|
155
|
+
frag_attributes = hash.collect do |frag_header, frag_content|
|
156
|
+
tag, identifier = frag_header.split
|
157
|
+
frag_hash = {
|
158
|
+
identifier: identifier,
|
159
|
+
tag: tag
|
160
|
+
}
|
161
|
+
|
162
|
+
# tracking fragments that need removing later
|
163
|
+
frag_identifiers << identifier
|
164
|
+
|
165
|
+
# based on tag we need to cram content in proper place and proper format
|
166
|
+
case tag
|
167
|
+
when 'date', 'datetime'
|
168
|
+
frag_hash[:datetime] = frag_content
|
169
|
+
when 'checkbox'
|
170
|
+
frag_hash[:boolean] = frag_content
|
171
|
+
when 'file', 'files'
|
172
|
+
files, file_ids_destroy = files_content(record, identifier, path, frag_content)
|
173
|
+
frag_hash[:files] = files
|
174
|
+
frag_hash[:file_ids_destroy] = file_ids_destroy
|
175
|
+
else
|
176
|
+
frag_hash[:content] = frag_content
|
177
|
+
end
|
178
|
+
|
179
|
+
frag_hash
|
180
|
+
end
|
181
|
+
|
182
|
+
[frag_identifiers, frag_attributes]
|
183
|
+
end
|
184
|
+
|
185
|
+
# Preparing fragment attachments. Returns hashes with file data for
|
186
|
+
# ActiveStorage and a list of ids of old attachements to destroy
|
187
|
+
def files_content(record, identifier, path, frag_content)
|
188
|
+
# preparing attachments
|
189
|
+
files = frag_content.split("\n").collect do |filename|
|
190
|
+
file_handler = File.open(File.join(path, filename))
|
191
|
+
{
|
192
|
+
io: file_handler,
|
193
|
+
filename: filename,
|
194
|
+
content_type: MimeMagic.by_magic(file_handler)
|
195
|
+
}
|
196
|
+
end
|
197
|
+
|
198
|
+
# ensuring that old attachments get removed
|
199
|
+
ids_destroy = []
|
200
|
+
if (frag = record.fragments.find_by(identifier: identifier))
|
201
|
+
ids_destroy = frag.attachments.pluck(:id)
|
202
|
+
end
|
203
|
+
|
204
|
+
[files, ids_destroy]
|
205
|
+
end
|
206
|
+
|
207
|
+
def link_target_pages
|
208
|
+
return unless self.target_pages.present?
|
209
|
+
|
210
|
+
self.target_pages.each do |page_id, target|
|
211
|
+
if (target = site.pages.find_by(full_path: target))
|
212
|
+
@site.pages.find(page_id).update_column(:target_page_id, target.id)
|
213
|
+
end
|
214
|
+
end
|
215
|
+
end
|
216
|
+
end
|
217
|
+
end
|