HornsAndHooves-publify_core 10.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/CHANGELOG.md +102 -0
- data/MIT-LICENSE +21 -0
- data/README.md +9 -0
- data/app/assets/fonts/bootstrap/glyphicons-halflings-regular.eot +0 -0
- data/app/assets/fonts/bootstrap/glyphicons-halflings-regular.svg +288 -0
- data/app/assets/fonts/bootstrap/glyphicons-halflings-regular.ttf +0 -0
- data/app/assets/fonts/bootstrap/glyphicons-halflings-regular.woff +0 -0
- data/app/assets/fonts/bootstrap/glyphicons-halflings-regular.woff2 +0 -0
- data/app/assets/fonts/open-sans-bold.woff +0 -0
- data/app/assets/fonts/open-sans.woff +0 -0
- data/app/assets/images/admin/loading.gif +0 -0
- data/app/assets/images/admin/typologo.gif +0 -0
- data/app/assets/images/calendar_date_select/calendar.gif +0 -0
- data/app/assets/images/close.gif +0 -0
- data/app/assets/images/closelabel.gif +0 -0
- data/app/assets/images/go.png +0 -0
- data/app/assets/images/loading.gif +0 -0
- data/app/assets/images/nextlabel.gif +0 -0
- data/app/assets/images/overlay.png +0 -0
- data/app/assets/images/powered.gif +0 -0
- data/app/assets/images/prevlabel.gif +0 -0
- data/app/assets/images/spinner-blue.gif +0 -0
- data/app/assets/images/spinner.gif +0 -0
- data/app/assets/images/thumb_blank.jpg +0 -0
- data/app/assets/javascripts/application.js +2 -0
- data/app/assets/javascripts/bootstrap/affix.js +164 -0
- data/app/assets/javascripts/bootstrap/alert.js +95 -0
- data/app/assets/javascripts/bootstrap/button.js +125 -0
- data/app/assets/javascripts/bootstrap/collapse.js +212 -0
- data/app/assets/javascripts/bootstrap/dropdown.js +165 -0
- data/app/assets/javascripts/bootstrap/modal.js +358 -0
- data/app/assets/javascripts/bootstrap/tab.js +155 -0
- data/app/assets/javascripts/bootstrap/transition.js +59 -0
- data/app/assets/javascripts/bootstrap-sprockets.js +8 -0
- data/app/assets/javascripts/cookies.js +60 -0
- data/app/assets/javascripts/datetimepicker.js +1470 -0
- data/app/assets/javascripts/lang/da_DK.js +17 -0
- data/app/assets/javascripts/lang/default.js +21 -0
- data/app/assets/javascripts/lang/en_US.js +21 -0
- data/app/assets/javascripts/lang/fr_FR.js +21 -0
- data/app/assets/javascripts/lang/nl_NL.js +21 -0
- data/app/assets/javascripts/lang/zh_TW.js +17 -0
- data/app/assets/javascripts/lightbox.js +350 -0
- data/app/assets/javascripts/observe.js +28 -0
- data/app/assets/javascripts/publify.js +11 -0
- data/app/assets/javascripts/publify_admin.js +104 -0
- data/app/assets/javascripts/quicktags.js +448 -0
- data/app/assets/javascripts/set-timeago-lang.js +3 -0
- data/app/assets/javascripts/sidebar.js +28 -0
- data/app/assets/javascripts/spinnable.js +5 -0
- data/app/assets/javascripts/tagmanager.js +481 -0
- data/app/assets/javascripts/typeahead.js +1139 -0
- data/app/assets/stylesheets/_bootstrap-compass.scss +9 -0
- data/app/assets/stylesheets/_bootstrap-mincer.scss +19 -0
- data/app/assets/stylesheets/_bootstrap-sprockets.scss +9 -0
- data/app/assets/stylesheets/_bootstrap.scss +43 -0
- data/app/assets/stylesheets/accounts.css.scss +7 -0
- data/app/assets/stylesheets/administration_structure.css.scss +255 -0
- data/app/assets/stylesheets/bootstrap/_alerts.scss +73 -0
- data/app/assets/stylesheets/bootstrap/_button-groups.scss +242 -0
- data/app/assets/stylesheets/bootstrap/_buttons.scss +168 -0
- data/app/assets/stylesheets/bootstrap/_close.scss +37 -0
- data/app/assets/stylesheets/bootstrap/_code.scss +69 -0
- data/app/assets/stylesheets/bootstrap/_component-animations.scss +38 -0
- data/app/assets/stylesheets/bootstrap/_dropdowns.scss +213 -0
- data/app/assets/stylesheets/bootstrap/_forms.scss +586 -0
- data/app/assets/stylesheets/bootstrap/_glyphicons.scss +307 -0
- data/app/assets/stylesheets/bootstrap/_grid.scss +94 -0
- data/app/assets/stylesheets/bootstrap/_labels.scss +66 -0
- data/app/assets/stylesheets/bootstrap/_media.scss +66 -0
- data/app/assets/stylesheets/bootstrap/_mixins.scss +37 -0
- data/app/assets/stylesheets/bootstrap/_modals.scss +150 -0
- data/app/assets/stylesheets/bootstrap/_navbar.scss +479 -0
- data/app/assets/stylesheets/bootstrap/_navs.scss +242 -0
- data/app/assets/stylesheets/bootstrap/_normalize.scss +427 -0
- data/app/assets/stylesheets/bootstrap/_pagination.scss +86 -0
- data/app/assets/stylesheets/bootstrap/_panels.scss +222 -0
- data/app/assets/stylesheets/bootstrap/_print.scss +99 -0
- data/app/assets/stylesheets/bootstrap/_scaffolding.scss +160 -0
- data/app/assets/stylesheets/bootstrap/_tables.scss +234 -0
- data/app/assets/stylesheets/bootstrap/_theme.scss +224 -0
- data/app/assets/stylesheets/bootstrap/_type.scss +296 -0
- data/app/assets/stylesheets/bootstrap/_utilities.scss +55 -0
- data/app/assets/stylesheets/bootstrap/_variables.scss +654 -0
- data/app/assets/stylesheets/bootstrap/_wells.scss +29 -0
- data/app/assets/stylesheets/bootstrap/mixins/_alerts.scss +15 -0
- data/app/assets/stylesheets/bootstrap/mixins/_background-variant.scss +12 -0
- data/app/assets/stylesheets/bootstrap/mixins/_border-radius.scss +18 -0
- data/app/assets/stylesheets/bootstrap/mixins/_buttons.scss +56 -0
- data/app/assets/stylesheets/bootstrap/mixins/_center-block.scss +7 -0
- data/app/assets/stylesheets/bootstrap/mixins/_clearfix.scss +22 -0
- data/app/assets/stylesheets/bootstrap/mixins/_forms.scss +82 -0
- data/app/assets/stylesheets/bootstrap/mixins/_gradients.scss +56 -0
- data/app/assets/stylesheets/bootstrap/mixins/_grid-framework.scss +81 -0
- data/app/assets/stylesheets/bootstrap/mixins/_grid.scss +122 -0
- data/app/assets/stylesheets/bootstrap/mixins/_hide-text.scss +21 -0
- data/app/assets/stylesheets/bootstrap/mixins/_image.scss +28 -0
- data/app/assets/stylesheets/bootstrap/mixins/_labels.scss +12 -0
- data/app/assets/stylesheets/bootstrap/mixins/_nav-divider.scss +10 -0
- data/app/assets/stylesheets/bootstrap/mixins/_nav-vertical-align.scss +9 -0
- data/app/assets/stylesheets/bootstrap/mixins/_opacity.scss +7 -0
- data/app/assets/stylesheets/bootstrap/mixins/_pagination.scss +24 -0
- data/app/assets/stylesheets/bootstrap/mixins/_panels.scss +20 -0
- data/app/assets/stylesheets/bootstrap/mixins/_reset-filter.scss +8 -0
- data/app/assets/stylesheets/bootstrap/mixins/_resize.scss +6 -0
- data/app/assets/stylesheets/bootstrap/mixins/_responsive-visibility.scss +17 -0
- data/app/assets/stylesheets/bootstrap/mixins/_size.scss +10 -0
- data/app/assets/stylesheets/bootstrap/mixins/_tab-focus.scss +9 -0
- data/app/assets/stylesheets/bootstrap/mixins/_table-row.scss +28 -0
- data/app/assets/stylesheets/bootstrap/mixins/_text-emphasis.scss +12 -0
- data/app/assets/stylesheets/bootstrap/mixins/_text-overflow.scss +8 -0
- data/app/assets/stylesheets/bootstrap/mixins/_vendor-prefixes.scss +222 -0
- data/app/assets/stylesheets/coderay.css +135 -0
- data/app/assets/stylesheets/datetimepicker.css +306 -0
- data/app/assets/stylesheets/lightbox.css +63 -0
- data/app/assets/stylesheets/publify.css.scss +9 -0
- data/app/assets/stylesheets/publify_admin.css.scss +10 -0
- data/app/assets/stylesheets/rss.css +53 -0
- data/app/assets/stylesheets/sidebar_admin.css.scss +36 -0
- data/app/assets/stylesheets/tagmanager.css +102 -0
- data/app/assets/stylesheets/user-styles.css +29 -0
- data/app/controllers/accounts_controller.rb +4 -0
- data/app/controllers/admin/base_controller.rb +27 -0
- data/app/controllers/admin/content_controller.rb +194 -0
- data/app/controllers/admin/dashboard_controller.rb +33 -0
- data/app/controllers/admin/feedback_controller.rb +160 -0
- data/app/controllers/admin/notes_controller.rb +73 -0
- data/app/controllers/admin/pages_controller.rb +73 -0
- data/app/controllers/admin/post_types_controller.rb +57 -0
- data/app/controllers/admin/profiles_controller.rb +57 -0
- data/app/controllers/admin/redirects_controller.rb +51 -0
- data/app/controllers/admin/resources_controller.rb +36 -0
- data/app/controllers/admin/seo_controller.rb +46 -0
- data/app/controllers/admin/settings_controller.rb +53 -0
- data/app/controllers/admin/sidebar_controller.rb +67 -0
- data/app/controllers/admin/tags_controller.rb +55 -0
- data/app/controllers/admin/themes_controller.rb +31 -0
- data/app/controllers/admin/users_controller.rb +67 -0
- data/app/controllers/articles_controller.rb +215 -0
- data/app/controllers/authors_controller.rb +30 -0
- data/app/controllers/base_controller.rb +43 -0
- data/app/controllers/comments_controller.rb +67 -0
- data/app/controllers/content_controller.rb +24 -0
- data/app/controllers/feedback_controller.rb +13 -0
- data/app/controllers/notes_controller.rb +35 -0
- data/app/controllers/setup_controller.rb +61 -0
- data/app/controllers/tags_controller.rb +50 -0
- data/app/controllers/text_controller.rb +7 -0
- data/app/controllers/textfilter_controller.rb +5 -0
- data/app/controllers/theme_controller.rb +63 -0
- data/app/controllers/xml_controller.rb +20 -0
- data/app/helpers/admin/base_helper.rb +87 -0
- data/app/helpers/admin/feedback_helper.rb +53 -0
- data/app/helpers/articles_helper.rb +10 -0
- data/app/helpers/authors_helper.rb +30 -0
- data/app/helpers/base_helper.rb +257 -0
- data/app/helpers/blog_helper.rb +14 -0
- data/app/helpers/xml_helper.rb +18 -0
- data/app/jobs/application_job.rb +5 -0
- data/app/mailers/notification_mailer.rb +40 -0
- data/app/models/ability.rb +53 -0
- data/app/models/archives_sidebar.rb +45 -0
- data/app/models/article/factory.rb +64 -0
- data/app/models/article.rb +308 -0
- data/app/models/blog.rb +280 -0
- data/app/models/comment.rb +57 -0
- data/app/models/config_manager.rb +83 -0
- data/app/models/content.rb +118 -0
- data/app/models/content_base.rb +88 -0
- data/app/models/feedback.rb +250 -0
- data/app/models/meta_sidebar.rb +10 -0
- data/app/models/note.rb +152 -0
- data/app/models/page.rb +42 -0
- data/app/models/page_sidebar.rb +14 -0
- data/app/models/ping.rb +5 -0
- data/app/models/post_type.rb +16 -0
- data/app/models/redirect.rb +51 -0
- data/app/models/redirection.rb +6 -0
- data/app/models/resource.rb +12 -0
- data/app/models/search_sidebar.rb +9 -0
- data/app/models/sidebar.rb +142 -0
- data/app/models/static_sidebar.rb +22 -0
- data/app/models/tag.rb +71 -0
- data/app/models/tag_sidebar.rb +30 -0
- data/app/models/text_filter.rb +141 -0
- data/app/models/trackback.rb +46 -0
- data/app/models/trigger.rb +34 -0
- data/app/models/user.rb +142 -0
- data/app/services/title_builder.rb +88 -0
- data/app/uploaders/resource_uploader.rb +63 -0
- data/app/views/accounts/confirm.html.erb +7 -0
- data/app/views/admin/content/_article_list.html.erb +26 -0
- data/app/views/admin/content/_form.html.erb +145 -0
- data/app/views/admin/content/autosave.js.erb +5 -0
- data/app/views/admin/content/edit.html.erb +3 -0
- data/app/views/admin/content/index.html.erb +48 -0
- data/app/views/admin/content/index.js.erb +1 -0
- data/app/views/admin/content/new.html.erb +3 -0
- data/app/views/admin/dashboard/_comment.html.erb +18 -0
- data/app/views/admin/dashboard/_comments.html.erb +10 -0
- data/app/views/admin/dashboard/_drafts.html.erb +25 -0
- data/app/views/admin/dashboard/_overview.html.erb +21 -0
- data/app/views/admin/dashboard/_welcome.html.erb +28 -0
- data/app/views/admin/dashboard/index.html.erb +16 -0
- data/app/views/admin/feedback/_button.html.erb +19 -0
- data/app/views/admin/feedback/_feedback.html.erb +7 -0
- data/app/views/admin/feedback/_ham.html.erb +17 -0
- data/app/views/admin/feedback/_spam.html.erb +23 -0
- data/app/views/admin/feedback/article.html.erb +69 -0
- data/app/views/admin/feedback/edit.html.erb +48 -0
- data/app/views/admin/feedback/ham.js +1 -0
- data/app/views/admin/feedback/index.html.erb +53 -0
- data/app/views/admin/feedback/spam.js +1 -0
- data/app/views/admin/migrations/show.html.erb +39 -0
- data/app/views/admin/notes/_form.html.erb +37 -0
- data/app/views/admin/notes/_header.html.erb +6 -0
- data/app/views/admin/notes/_list.html.erb +13 -0
- data/app/views/admin/notes/_note.html.erb +14 -0
- data/app/views/admin/notes/edit.html.erb +11 -0
- data/app/views/admin/notes/index.html.erb +11 -0
- data/app/views/admin/notes/show.html.erb +14 -0
- data/app/views/admin/pages/_form.html.erb +101 -0
- data/app/views/admin/pages/_pages.html.erb +21 -0
- data/app/views/admin/pages/edit.html.erb +1 -0
- data/app/views/admin/pages/index.html.erb +17 -0
- data/app/views/admin/pages/new.html.erb +1 -0
- data/app/views/admin/post_types/_index_and_form.html.erb +65 -0
- data/app/views/admin/post_types/edit.html.erb +1 -0
- data/app/views/admin/post_types/index.html.erb +1 -0
- data/app/views/admin/profiles/index.html.erb +10 -0
- data/app/views/admin/redirects/_index_and_form.html.erb +68 -0
- data/app/views/admin/redirects/edit.html.erb +1 -0
- data/app/views/admin/redirects/index.html.erb +1 -0
- data/app/views/admin/resources/index.html.erb +72 -0
- data/app/views/admin/seo/_general.html.erb +104 -0
- data/app/views/admin/seo/_permalinks.html.erb +53 -0
- data/app/views/admin/seo/_titles.html.erb +209 -0
- data/app/views/admin/seo/show.html.erb +28 -0
- data/app/views/admin/settings/display.html.erb +101 -0
- data/app/views/admin/settings/feedback.html.erb +118 -0
- data/app/views/admin/settings/index.html.erb +73 -0
- data/app/views/admin/settings/write.html.erb +69 -0
- data/app/views/admin/shared/_edit.html.erb +4 -0
- data/app/views/admin/shared/_menu.html.erb +121 -0
- data/app/views/admin/shared/_twitter_alert.html.erb +3 -0
- data/app/views/admin/sidebar/_available.html.erb +6 -0
- data/app/views/admin/sidebar/_available.json.erb +6 -0
- data/app/views/admin/sidebar/_config.html.erb +27 -0
- data/app/views/admin/sidebar/_target.html.erb +9 -0
- data/app/views/admin/sidebar/_target_sidebar.html.erb +20 -0
- data/app/views/admin/sidebar/destroy.js.erb +1 -0
- data/app/views/admin/sidebar/index.html.erb +31 -0
- data/app/views/admin/sidebar/sortable.js.erb +3 -0
- data/app/views/admin/sidebar/update.js.erb +2 -0
- data/app/views/admin/tags/_index_and_form.html.erb +59 -0
- data/app/views/admin/tags/edit.html.erb +1 -0
- data/app/views/admin/tags/index.html.erb +1 -0
- data/app/views/admin/themes/index.html.erb +27 -0
- data/app/views/admin/users/_form.html.erb +215 -0
- data/app/views/admin/users/edit.html.erb +8 -0
- data/app/views/admin/users/index.html.erb +39 -0
- data/app/views/admin/users/new.html.erb +8 -0
- data/app/views/archives_sidebar/_content.html.erb +13 -0
- data/app/views/articles/_archives_article.html.erb +9 -0
- data/app/views/articles/_article.html.erb +9 -0
- data/app/views/articles/_article_author.html.erb +4 -0
- data/app/views/articles/_article_collection.html.erb +8 -0
- data/app/views/articles/_article_content.html.erb +5 -0
- data/app/views/articles/_article_excerpt.html.erb +15 -0
- data/app/views/articles/_article_links.html.erb +12 -0
- data/app/views/articles/_comment.html.erb +1 -0
- data/app/views/articles/_comment_errors.html.erb +2 -0
- data/app/views/articles/_comment_form.html.erb +48 -0
- data/app/views/articles/_comment_list.html.erb +5 -0
- data/app/views/articles/_comment_preview.html.erb +4 -0
- data/app/views/articles/_full_article_content.html.erb +4 -0
- data/app/views/articles/_password_form.html.erb +10 -0
- data/app/views/articles/_trackback.html.erb +6 -0
- data/app/views/articles/archives.html.erb +15 -0
- data/app/views/articles/comment.js.erb +5 -0
- data/app/views/articles/comment_failed.js.erb +3 -0
- data/app/views/articles/error.html.erb +3 -0
- data/app/views/articles/feedback_atom_feed.atom.builder +10 -0
- data/app/views/articles/feedback_rss_feed.rss.builder +21 -0
- data/app/views/articles/index.html.erb +1 -0
- data/app/views/articles/index_atom_feed.atom.builder +9 -0
- data/app/views/articles/index_rss_feed.rss.builder +20 -0
- data/app/views/articles/live_search.html.erb +10 -0
- data/app/views/articles/read.html.erb +43 -0
- data/app/views/articles/search.html.erb +8 -0
- data/app/views/articles/trackback.xml.builder +7 -0
- data/app/views/articles/view_page.html.erb +3 -0
- data/app/views/authors/show.html.erb +30 -0
- data/app/views/authors/show_atom_feed.atom.builder +9 -0
- data/app/views/authors/show_rss_feed.rss.builder +20 -0
- data/app/views/comments/_comment.html.erb +17 -0
- data/app/views/comments/preview.html.erb +1 -0
- data/app/views/comments/preview.js.erb +3 -0
- data/app/views/devise/mailer/reset_password_instructions.html.erb +11 -0
- data/app/views/devise/passwords/edit.html.erb +28 -0
- data/app/views/devise/passwords/new.html.erb +20 -0
- data/app/views/devise/registrations/new.html.erb +36 -0
- data/app/views/devise/sessions/new.html.erb +32 -0
- data/app/views/devise/shared/_links.html.erb +11 -0
- data/app/views/errors/404.html.erb +2 -0
- data/app/views/feedback/index.atom.builder +9 -0
- data/app/views/feedback/index.rss.builder +20 -0
- data/app/views/layouts/accounts.html.erb +33 -0
- data/app/views/layouts/administration.html.erb +37 -0
- data/app/views/layouts/default.html.erb +33 -0
- data/app/views/layouts/editor.html.erb +17 -0
- data/app/views/meta_sidebar/_content.html.erb +8 -0
- data/app/views/notes/_note.html.erb +16 -0
- data/app/views/notes/error.html.erb +3 -0
- data/app/views/notes/index.html.erb +15 -0
- data/app/views/notes/show.html.erb +5 -0
- data/app/views/notes/show_in_reply.html.erb +18 -0
- data/app/views/notification_mailer/_mail_footer.html.erb +5 -0
- data/app/views/notification_mailer/_mail_header.html.erb +3 -0
- data/app/views/notification_mailer/article.html.erb +7 -0
- data/app/views/notification_mailer/comment.html.erb +12 -0
- data/app/views/notification_mailer/notif_user.html.erb +9 -0
- data/app/views/page_sidebar/_content.html.erb +12 -0
- data/app/views/search_sidebar/_content.html.erb +10 -0
- data/app/views/settings/done.html.erb +2 -0
- data/app/views/settings/install.html.erb +12 -0
- data/app/views/setup/index.html.erb +17 -0
- data/app/views/shared/_atom_header.atom.builder +6 -0
- data/app/views/shared/_atom_item_article.atom.builder +36 -0
- data/app/views/shared/_atom_item_comment.atom.builder +10 -0
- data/app/views/shared/_atom_item_trackback.atom.builder +10 -0
- data/app/views/shared/_flash.erb +10 -0
- data/app/views/shared/_google_analytics.html.erb +8 -0
- data/app/views/shared/_page_header.html.erb +23 -0
- data/app/views/shared/_rss_item_article.rss.builder +32 -0
- data/app/views/shared/_rss_item_comment.rss.builder +9 -0
- data/app/views/shared/_rss_item_trackback.rss.builder +9 -0
- data/app/views/sidebar/_row.html.erb +1 -0
- data/app/views/sidebar/_sidebar.html.erb +5 -0
- data/app/views/sidebar/display_plugins.html.erb +5 -0
- data/app/views/sidebar/show.html.erb +1 -0
- data/app/views/static_sidebar/_content.html.erb +2 -0
- data/app/views/tag_sidebar/_content.html.erb +10 -0
- data/app/views/tags/index.html.erb +15 -0
- data/app/views/tags/show.html.erb +1 -0
- data/app/views/theme/static_view_test.html.erb +1 -0
- data/app/views/xml/_googlesitemap_item_article.googlesitemap.builder +7 -0
- data/app/views/xml/_googlesitemap_item_category.googlesitemap.builder +6 -0
- data/app/views/xml/_googlesitemap_item_page.googlesitemap.builder +6 -0
- data/app/views/xml/_googlesitemap_item_tag.googlesitemap.builder +6 -0
- data/app/views/xml/sitemap.googlesitemap.builder +9 -0
- data/config/i18n-tasks.yml +49 -0
- data/config/initializers/devise.rb +271 -0
- data/config/initializers/mime_types.rb +7 -0
- data/config/locales/da.yml +769 -0
- data/config/locales/de.yml +769 -0
- data/config/locales/en.yml +770 -0
- data/config/locales/es-MX.yml +769 -0
- data/config/locales/fr.yml +769 -0
- data/config/locales/he.yml +769 -0
- data/config/locales/it.yml +769 -0
- data/config/locales/ja.yml +769 -0
- data/config/locales/lt.yml +783 -0
- data/config/locales/nb-NO.yml +769 -0
- data/config/locales/nl.yml +769 -0
- data/config/locales/pl.yml +797 -0
- data/config/locales/pt-BR.yml +769 -0
- data/config/locales/ro.yml +782 -0
- data/config/locales/ru.yml +797 -0
- data/config/locales/sidebars.da.yml +20 -0
- data/config/locales/sidebars.de.yml +20 -0
- data/config/locales/sidebars.en.yml +20 -0
- data/config/locales/sidebars.es-MX.yml +20 -0
- data/config/locales/sidebars.fr.yml +20 -0
- data/config/locales/sidebars.he.yml +20 -0
- data/config/locales/sidebars.it.yml +20 -0
- data/config/locales/sidebars.ja.yml +20 -0
- data/config/locales/sidebars.lt.yml +20 -0
- data/config/locales/sidebars.nb-NO.yml +20 -0
- data/config/locales/sidebars.nl.yml +20 -0
- data/config/locales/sidebars.pl.yml +20 -0
- data/config/locales/sidebars.pt-BR.yml +20 -0
- data/config/locales/sidebars.ro.yml +20 -0
- data/config/locales/sidebars.ru.yml +20 -0
- data/config/locales/sidebars.zh-CN.yml +20 -0
- data/config/locales/sidebars.zh-TW.yml +20 -0
- data/config/locales/zh-CN.yml +768 -0
- data/config/locales/zh-TW.yml +769 -0
- data/config/routes.rb +174 -0
- data/db/migrate/113_initial_schema.rb +210 -0
- data/db/migrate/114_fixes_buggy_articles_and_notes.rb +56 -0
- data/db/migrate/115_drops_categories_for_tags.rb +37 -0
- data/db/migrate/20150207131657_add_missing_indexes.rb +23 -0
- data/db/migrate/20150807134129_simplify_redirect_relations.rb +45 -0
- data/db/migrate/20150808052637_add_blog_ids.rb +39 -0
- data/db/migrate/20150808191127_add_blog_id_to_redirects.rb +19 -0
- data/db/migrate/20150810094754_add_blog_id_to_tags.rb +19 -0
- data/db/migrate/20160108111120_add_devise_to_users.rb +57 -0
- data/db/migrate/20160108184201_move_last_connection_to_last_sign_in_at.rb +20 -0
- data/db/migrate/20160110094906_remove_profiles_rights.rb +18 -0
- data/db/migrate/20160605103918_replace_profile_id_with_string.rb +32 -0
- data/db/migrate/20160605154632_remove_profiles.rb +31 -0
- data/db/migrate/20160701061851_demand_blog_id_on_contents.rb +13 -0
- data/db/migrate/20160701062604_add_blog_id_to_resources.rb +32 -0
- data/db/migrate/20170528093024_move_resources_to_content.rb +9 -0
- data/db/migrate/20170528094923_move_tags_to_content.rb +10 -0
- data/db/migrate/20170528201606_remove_separate_published_flag.rb +9 -0
- data/db/migrate/20170605071626_remove_extra_state_columns_from_feedback.rb +10 -0
- data/db/migrate/20170702105201_remove_published_at_from_feedback.rb +9 -0
- data/db/migrate/20190208151235_add_text_filter_name_fields.rb +11 -0
- data/db/migrate/20190208152646_move_text_filter_to_name.rb +99 -0
- data/db/migrate/20190209155717_remove_text_filter_ids.rb +21 -0
- data/db/migrate/20190209160610_remove_text_filters.rb +19 -0
- data/db/seeds.rb +25 -0
- data/lib/email_notify.rb +28 -0
- data/lib/format.rb +9 -0
- data/lib/publify_core/engine.rb +31 -0
- data/lib/publify_core/lang.rb +9 -0
- data/lib/publify_core/testing_support/dns_mock.rb +15 -0
- data/lib/publify_core/testing_support/factories.rb +242 -0
- data/lib/publify_core/testing_support/feed_assertions.rb +48 -0
- data/lib/publify_core/testing_support/fixtures/exploit.svg +4 -0
- data/lib/publify_core/testing_support/fixtures/fakepng.png +1 -0
- data/lib/publify_core/testing_support/fixtures/otherfile.txt +1 -0
- data/lib/publify_core/testing_support/fixtures/testfile.png +0 -0
- data/lib/publify_core/testing_support/fixtures/testfile.txt +1 -0
- data/lib/publify_core/testing_support/upload_fixtures.rb +15 -0
- data/lib/publify_core/version.rb +5 -0
- data/lib/publify_core.rb +46 -0
- data/lib/publify_guid.rb +11 -0
- data/lib/publify_plugins.rb +77 -0
- data/lib/publify_textfilter_markdown.rb +57 -0
- data/lib/publify_textfilter_none.rb +16 -0
- data/lib/publify_textfilter_smartypants.rb +16 -0
- data/lib/publify_textfilter_textile.rb +23 -0
- data/lib/publify_textfilter_twitterfilter.rb +39 -0
- data/lib/publify_time.rb +34 -0
- data/lib/sidebar_field.rb +127 -0
- data/lib/sidebar_registry.rb +35 -0
- data/lib/spam_protection.rb +103 -0
- data/lib/tasks/i18n.rake +9 -0
- data/lib/tasks/manifest.rake +30 -0
- data/lib/tasks/publify_core_tasks.rake +6 -0
- data/lib/text_filter_plugin.rb +183 -0
- data/lib/theme.rb +72 -0
- data/lib/transforms.rb +47 -0
- data/themes/plain/about.markdown +4 -0
- data/themes/plain/javascripts/theme.js +0 -0
- data/themes/plain/preview.png +0 -0
- data/themes/plain/stylesheets/theme.css +57 -0
- metadata +1062 -0
@@ -0,0 +1,257 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Methods added to this helper will be available to all templates in the application.
|
4
|
+
require "digest/sha1"
|
5
|
+
|
6
|
+
module BaseHelper
|
7
|
+
include BlogHelper
|
8
|
+
|
9
|
+
# Need to rewrite this one, quick hack to test my changes.
|
10
|
+
attr_reader :page_title
|
11
|
+
|
12
|
+
def render_sidebars
|
13
|
+
rendered_sidebars = Sidebar.order(:active_position).map do |sb|
|
14
|
+
@sidebar = sb
|
15
|
+
sb.parse_request(content_array, params)
|
16
|
+
render_sidebar(sb)
|
17
|
+
end
|
18
|
+
safe_join rendered_sidebars
|
19
|
+
rescue => e
|
20
|
+
logger.error e
|
21
|
+
logger.error e.backtrace.join("\n")
|
22
|
+
I18n.t("errors.render_sidebar")
|
23
|
+
end
|
24
|
+
|
25
|
+
def render_sidebar(sidebar)
|
26
|
+
render_to_string(partial: sidebar.content_partial, locals: sidebar.to_locals_hash,
|
27
|
+
layout: false)
|
28
|
+
end
|
29
|
+
|
30
|
+
def themeable_stylesheet_link_tag(name)
|
31
|
+
src = this_blog.current_theme.path + "/stylesheets/#{name}.css"
|
32
|
+
stylesheet_link_tag "/stylesheets/theme/#{name}.css" if File.exist? src
|
33
|
+
end
|
34
|
+
|
35
|
+
def themeable_javascript_include_tag(name)
|
36
|
+
src = this_blog.current_theme.path + "/javascripts/#{name}.js"
|
37
|
+
javascript_include_tag "/javascripts/theme/#{name}.js" if File.exist? src
|
38
|
+
end
|
39
|
+
|
40
|
+
def render_to_string(*args, &block)
|
41
|
+
controller.send(:render_to_string, *args, &block)
|
42
|
+
end
|
43
|
+
|
44
|
+
def link_to_permalink(item, title, anchor = nil, style = nil, nofollow = nil,
|
45
|
+
only_path = false)
|
46
|
+
options = {}
|
47
|
+
options[:class] = style if style
|
48
|
+
options[:rel] = "nofollow" if nofollow
|
49
|
+
url = item.permalink_url(anchor, only_path)
|
50
|
+
if url
|
51
|
+
link_to title, url, options
|
52
|
+
else
|
53
|
+
title
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
def avatar_tag(options = {})
|
58
|
+
begin
|
59
|
+
avatar_class = this_blog.plugin_avatar.constantize
|
60
|
+
rescue NameError
|
61
|
+
return ""
|
62
|
+
end
|
63
|
+
return "" unless avatar_class.respond_to?(:get_avatar)
|
64
|
+
|
65
|
+
avatar_class.get_avatar(options)
|
66
|
+
end
|
67
|
+
|
68
|
+
def meta_tag(name, value)
|
69
|
+
tag :meta, name: name, content: value if value.present?
|
70
|
+
end
|
71
|
+
|
72
|
+
def markup_help_popup(markup, text)
|
73
|
+
if markup && markup.commenthelp.size > 1
|
74
|
+
link_to(text,
|
75
|
+
url_for(controller: "articles", action: "markup_help", id: markup.name),
|
76
|
+
onclick: "return popup(this, 'Publify Markup Help')")
|
77
|
+
else
|
78
|
+
""
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def onhover_show_admin_tools(type, id = nil)
|
83
|
+
admin_id = "#admin_#{[type, id].compact.join("_")}"
|
84
|
+
tag = []
|
85
|
+
tag << %{ onmouseover="if (getCookie('publify_user_profile') == 'admin')\
|
86
|
+
{ $('#{admin_id}').show(); }" }
|
87
|
+
tag << %{ onmouseout="$('#{admin_id}').hide();" }
|
88
|
+
safe_join(tag, " ")
|
89
|
+
end
|
90
|
+
|
91
|
+
def feed_title
|
92
|
+
if @feed_title.present?
|
93
|
+
@feed_title
|
94
|
+
elsif @page_title.present?
|
95
|
+
@page_title
|
96
|
+
else
|
97
|
+
this_blog.blog_name
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
def html(content, what = :all, _deprecated = false)
|
102
|
+
content.html(what)
|
103
|
+
end
|
104
|
+
|
105
|
+
def display_user_avatar(user, size = "avatar", klass = "alignleft")
|
106
|
+
if user.resource.present?
|
107
|
+
avatar_path = case size
|
108
|
+
when "thumb"
|
109
|
+
user.resource.upload.thumb.url
|
110
|
+
when "medium"
|
111
|
+
user.resource.upload.medium.url
|
112
|
+
when "large"
|
113
|
+
user.resource.upload.large.url
|
114
|
+
else
|
115
|
+
user.resource.upload.avatar.url
|
116
|
+
end
|
117
|
+
return if avatar_path.nil?
|
118
|
+
|
119
|
+
avatar_url = this_blog.file_url(avatar_path)
|
120
|
+
elsif user.twitter_profile_image.present?
|
121
|
+
avatar_url = user.twitter_profile_image
|
122
|
+
end
|
123
|
+
return unless avatar_url
|
124
|
+
|
125
|
+
image_tag(avatar_url, alt: user.nickname, class: klass)
|
126
|
+
end
|
127
|
+
|
128
|
+
def author_picture(status)
|
129
|
+
return if status.user.twitter_profile_image.blank?
|
130
|
+
|
131
|
+
image_tag(status.user.twitter_profile_image, class: "alignleft",
|
132
|
+
alt: status.user.nickname)
|
133
|
+
end
|
134
|
+
|
135
|
+
def page_header_includes
|
136
|
+
content_array.map(&:whiteboard).map do |w|
|
137
|
+
w.select { |k, _v| k =~ /^page_header_/ }.map do |_, v|
|
138
|
+
v = v.chomp
|
139
|
+
# trim the same number of spaces from the beginning of each line
|
140
|
+
# this way plugins can indent nicely without making ugly source output
|
141
|
+
spaces = /\A[ \t]*/.match(v)[0].gsub(/\t/, " ")
|
142
|
+
# add 2 spaces to line up with the assumed position of the surrounding tags
|
143
|
+
v.gsub!(/^#{spaces}/, " ")
|
144
|
+
end
|
145
|
+
end.flatten.uniq.join("\n")
|
146
|
+
end
|
147
|
+
|
148
|
+
def feed_atom
|
149
|
+
feed_for("atom")
|
150
|
+
end
|
151
|
+
|
152
|
+
def feed_rss
|
153
|
+
feed_for("rss")
|
154
|
+
end
|
155
|
+
|
156
|
+
def content_array
|
157
|
+
if @articles
|
158
|
+
@articles
|
159
|
+
elsif @article
|
160
|
+
[@article]
|
161
|
+
elsif @page
|
162
|
+
[@page]
|
163
|
+
else
|
164
|
+
[]
|
165
|
+
end
|
166
|
+
end
|
167
|
+
|
168
|
+
def display_date(date)
|
169
|
+
l(date, format: this_blog.date_format)
|
170
|
+
end
|
171
|
+
|
172
|
+
def display_time(time)
|
173
|
+
time.strftime(this_blog.time_format)
|
174
|
+
end
|
175
|
+
|
176
|
+
def display_date_and_time(timestamp)
|
177
|
+
return if timestamp.blank?
|
178
|
+
|
179
|
+
if this_blog.date_format == "setting_date_format_distance_of_time_in_words"
|
180
|
+
timeago_tag timestamp, date_only: false
|
181
|
+
else
|
182
|
+
"#{display_date(timestamp)} #{t("helper.at")} #{display_time(timestamp)}"
|
183
|
+
end
|
184
|
+
end
|
185
|
+
|
186
|
+
def show_meta_keyword
|
187
|
+
return unless this_blog.use_meta_keyword
|
188
|
+
|
189
|
+
meta_tag "keywords", @keywords if @keywords.present?
|
190
|
+
end
|
191
|
+
|
192
|
+
def stop_index_robots?(blog)
|
193
|
+
stop = (params[:year].present? || params[:page].present?)
|
194
|
+
stop = blog.unindex_tags if controller_name == "tags"
|
195
|
+
stop = blog.unindex_categories if controller_name == "categories"
|
196
|
+
stop
|
197
|
+
end
|
198
|
+
|
199
|
+
def get_reply_context_url(reply)
|
200
|
+
link_to(reply["user"]["name"],
|
201
|
+
reply["user"]["entities"]["url"]["urls"][0]["expanded_url"])
|
202
|
+
rescue
|
203
|
+
link_to(reply["user"]["name"], "https://twitter.com/#{reply["user"]["name"]}")
|
204
|
+
end
|
205
|
+
|
206
|
+
def get_reply_context_twitter_link(reply)
|
207
|
+
link_to(display_date_and_time(reply["created_at"].to_time.in_time_zone),
|
208
|
+
"https://twitter.com/#{reply["user"]["screen_name"]}/status/#{reply["id_str"]}")
|
209
|
+
end
|
210
|
+
|
211
|
+
private
|
212
|
+
|
213
|
+
def feed_for(type)
|
214
|
+
if params[:action] == "search"
|
215
|
+
url_for(only_path: false, format: type, q: params[:q])
|
216
|
+
elsif !@article.nil?
|
217
|
+
@article.feed_url(type)
|
218
|
+
elsif !@auto_discovery_url_atom.nil?
|
219
|
+
instance_variable_get("@auto_discovery_url_#{type}")
|
220
|
+
end
|
221
|
+
end
|
222
|
+
|
223
|
+
# fetches appropriate html content for RSS and ATOM feeds. Checks for:
|
224
|
+
# - article being password protected
|
225
|
+
# - hiding extended content on RSS. In this case if there is an excerpt we
|
226
|
+
# show the excerpt, or else we show the body
|
227
|
+
def fetch_html_content_for_feeds(item, this_blog)
|
228
|
+
if item.password_protected?
|
229
|
+
"<p>This article is password protected. Please " \
|
230
|
+
"<a href='#{item.permalink_url}'>fill in your password</a> to read it</p>"
|
231
|
+
elsif this_blog.hide_extended_on_rss
|
232
|
+
if item.excerpt? && !item.excerpt.empty?
|
233
|
+
item.excerpt
|
234
|
+
else
|
235
|
+
html(item, :body)
|
236
|
+
end
|
237
|
+
else
|
238
|
+
html(item, :all)
|
239
|
+
end
|
240
|
+
end
|
241
|
+
|
242
|
+
def nofollowify_links(string)
|
243
|
+
if this_blog.dofollowify
|
244
|
+
string
|
245
|
+
else
|
246
|
+
string.gsub(/<a(.*?)>/i, '<a\1 rel="nofollow">')
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
250
|
+
def nofollowified_link_to(text, url)
|
251
|
+
if this_blog.dofollowify
|
252
|
+
link_to(text, url)
|
253
|
+
else
|
254
|
+
link_to(text, url, rel: "nofollow")
|
255
|
+
end
|
256
|
+
end
|
257
|
+
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module BlogHelper
|
4
|
+
# The base URL for this request, calculated by looking up the URL for the main
|
5
|
+
# blog index page.
|
6
|
+
def blog_base_url
|
7
|
+
url_for(controller: "/articles", action: "index").gsub(%r{/$}, "")
|
8
|
+
end
|
9
|
+
|
10
|
+
# Find the blog whose base_url matches the current location.
|
11
|
+
def this_blog
|
12
|
+
@this_blog ||= Blog.find_blog(blog_base_url)
|
13
|
+
end
|
14
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module XmlHelper
|
4
|
+
def collection_lastmod(collection)
|
5
|
+
article_updated = collection.contents.order(updated_at: :desc).first
|
6
|
+
article_published = collection.contents.order(published_at: :desc).first
|
7
|
+
|
8
|
+
times = []
|
9
|
+
times.push article_updated.updated_at if article_updated
|
10
|
+
times.push article_published.updated_at if article_published
|
11
|
+
|
12
|
+
if times.empty?
|
13
|
+
Time.zone.at(0).xmlschema
|
14
|
+
else
|
15
|
+
times.max.xmlschema
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,40 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class NotificationMailer < ApplicationMailer
|
4
|
+
helper :base
|
5
|
+
layout nil
|
6
|
+
|
7
|
+
def article(article, user)
|
8
|
+
@user = user
|
9
|
+
@blog = article.blog
|
10
|
+
@article = article
|
11
|
+
build_mail @blog, @user, "New article: #{article.title}"
|
12
|
+
end
|
13
|
+
|
14
|
+
def comment(comment, user)
|
15
|
+
@user = user
|
16
|
+
@blog = comment.blog
|
17
|
+
@comment = comment
|
18
|
+
build_mail @blog, @user, "New comment on #{comment.article.title}"
|
19
|
+
end
|
20
|
+
|
21
|
+
def notif_user(user)
|
22
|
+
@user = user
|
23
|
+
# TODO: Make user blog-dependent
|
24
|
+
@blog = Blog.first
|
25
|
+
build_mail @blog, @user, "Welcome to Publify"
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
def make_subject(blog, subject)
|
31
|
+
"[#{blog.blog_name}] #{subject}"
|
32
|
+
end
|
33
|
+
|
34
|
+
def build_mail(blog, user, subject)
|
35
|
+
headers["X-Mailer"] = "Publify #{PublifyCore::VERSION}"
|
36
|
+
mail(from: blog.email_from,
|
37
|
+
to: user.email,
|
38
|
+
subject: make_subject(blog, subject))
|
39
|
+
end
|
40
|
+
end
|
@@ -0,0 +1,53 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "cancancan"
|
4
|
+
|
5
|
+
class Ability
|
6
|
+
include CanCan::Ability
|
7
|
+
|
8
|
+
def initialize(user)
|
9
|
+
return unless user
|
10
|
+
|
11
|
+
case user.profile
|
12
|
+
when "admin"
|
13
|
+
add_admin_abilities
|
14
|
+
add_publisher_abilities
|
15
|
+
add_contributor_abilities
|
16
|
+
when "publisher"
|
17
|
+
add_publisher_abilities
|
18
|
+
add_contributor_abilities
|
19
|
+
when "contributor"
|
20
|
+
add_contributor_abilities
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
private
|
25
|
+
|
26
|
+
def add_admin_abilities
|
27
|
+
can :manage, "admin/migrations"
|
28
|
+
can :manage, "admin/seo"
|
29
|
+
can :manage, "admin/settings"
|
30
|
+
can :manage, "admin/sidebar"
|
31
|
+
can :manage, "admin/textfilters"
|
32
|
+
can :manage, "admin/themes"
|
33
|
+
can :manage, "admin/users"
|
34
|
+
end
|
35
|
+
|
36
|
+
def add_publisher_abilities
|
37
|
+
can :manage, "admin/content"
|
38
|
+
can :manage, "admin/feedback"
|
39
|
+
can :manage, "admin/notes"
|
40
|
+
can :manage, "admin/pages"
|
41
|
+
can :manage, "admin/post_types"
|
42
|
+
can :manage, "admin/redirects"
|
43
|
+
can :manage, "admin/resources"
|
44
|
+
can :manage, "admin/tags"
|
45
|
+
|
46
|
+
can :manage, "articles"
|
47
|
+
end
|
48
|
+
|
49
|
+
def add_contributor_abilities
|
50
|
+
can :manage, "admin/dashboard"
|
51
|
+
can :manage, "admin/profiles"
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class ArchivesSidebar < Sidebar
|
4
|
+
description "Displays links to monthly archives"
|
5
|
+
setting :title, "Archives"
|
6
|
+
setting :show_count, true, label: "Show article counts", input_type: :checkbox
|
7
|
+
setting :count, 10, label: "Number of Months"
|
8
|
+
|
9
|
+
attr_accessor :archives
|
10
|
+
|
11
|
+
def self.date_funcs
|
12
|
+
@date_funcs ||=
|
13
|
+
if /SQLite3Adapter/.match?(Content.connection.class.name)
|
14
|
+
["strftime('%Y', published_at) as year", "strftime('%m', published_at) as month"]
|
15
|
+
else
|
16
|
+
["extract(year from published_at) as year",
|
17
|
+
"extract(month from published_at) as month"]
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
def parse_request(_contents, _params)
|
22
|
+
# The original query that was here instantiated every article and every
|
23
|
+
# tag, and then sorted through them just to do a 'group by date'.
|
24
|
+
# Unfortunately, there's no universally-supported way to do this
|
25
|
+
# across all three of our supported DBs. So, we resort to a bit of
|
26
|
+
# DB-specific code.
|
27
|
+
date_funcs = self.class.date_funcs
|
28
|
+
|
29
|
+
article_counts = Article.published.select("count(*) as count", *date_funcs).
|
30
|
+
group(:year, :month).reorder("year desc", "month desc").limit(count.to_i)
|
31
|
+
|
32
|
+
@archives = article_counts.map do |entry|
|
33
|
+
month = entry.month.to_i
|
34
|
+
year = entry.year.to_i
|
35
|
+
{
|
36
|
+
name: I18n.l(Date.new(year, month), format: "%B %Y"),
|
37
|
+
month: month,
|
38
|
+
year: year,
|
39
|
+
article_count: entry.count,
|
40
|
+
}
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
SidebarRegistry.register_sidebar ArchivesSidebar
|
@@ -0,0 +1,64 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Article::Factory
|
4
|
+
attr_reader :blog, :user
|
5
|
+
|
6
|
+
def initialize(blog, user)
|
7
|
+
@blog = blog
|
8
|
+
@user = user
|
9
|
+
end
|
10
|
+
|
11
|
+
def default
|
12
|
+
blog.articles.build.tap do |art|
|
13
|
+
art.allow_comments = blog.default_allow_comments
|
14
|
+
art.allow_pings = blog.default_allow_pings
|
15
|
+
art.text_filter_name = default_text_filter
|
16
|
+
art.state = "draft"
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
def get_or_build_from(id)
|
21
|
+
return blog.articles.find(id) if id.present?
|
22
|
+
|
23
|
+
default
|
24
|
+
end
|
25
|
+
|
26
|
+
def match_permalink_format(path, format)
|
27
|
+
article_params = extract_params(path, format)
|
28
|
+
requested_article(article_params) if article_params
|
29
|
+
end
|
30
|
+
|
31
|
+
def requested_article(params = {})
|
32
|
+
params[:title] ||= params[:article_id]
|
33
|
+
Article.requested_article(params)
|
34
|
+
end
|
35
|
+
|
36
|
+
def extract_params(path, format)
|
37
|
+
specs = format.split("/")
|
38
|
+
specs.delete("")
|
39
|
+
parts = path.split("/")
|
40
|
+
parts.delete("")
|
41
|
+
|
42
|
+
return if parts.length != specs.length
|
43
|
+
|
44
|
+
specs.zip(parts).reduce({}) do |result, (spec, item)|
|
45
|
+
if spec =~ /(.*)%(.*)%(.*)/
|
46
|
+
before_format = Regexp.last_match[1]
|
47
|
+
format_string = Regexp.last_match[2]
|
48
|
+
after_format = Regexp.last_match[3]
|
49
|
+
item =~ /^#{before_format}(.*)#{after_format}$/
|
50
|
+
break unless Regexp.last_match
|
51
|
+
|
52
|
+
value = Regexp.last_match[1]
|
53
|
+
result[format_string.to_sym] = value
|
54
|
+
elsif spec != item
|
55
|
+
break
|
56
|
+
end
|
57
|
+
result
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def default_text_filter
|
62
|
+
user.text_filter_name || blog.text_filter
|
63
|
+
end
|
64
|
+
end
|