From 1b85d67e0e52cade30a3c9d5592308626b1e838b Mon Sep 17 00:00:00 2001 From: Kevin Ansfield <kevin@lookingsideways.co.uk> Date: Wed, 18 May 2016 10:29:59 +0100 Subject: [PATCH 1/2] Remove split Ghost-Admin code --- core/client/.ember-cli | 9 - core/client/.gitignore | 20 - core/client/.jscsrc | 14 - core/client/.jshintrc | 36 - core/client/.watchmanconfig | 3 - core/client/app/README.md | 30 - core/client/app/_config.yml | 15 - core/client/app/adapters/application.js | 9 - core/client/app/adapters/base.js | 54 -- .../app/adapters/embedded-relation-adapter.js | 132 ---- core/client/app/adapters/setting.js | 19 - core/client/app/adapters/tag.js | 4 - core/client/app/adapters/user.js | 23 - core/client/app/app.js | 20 - core/client/app/authenticators/oauth2.js | 28 - core/client/app/authorizers/oauth2.js | 3 - .../app/components/gh-activating-list-item.js | 22 - core/client/app/components/gh-alert.js | 40 - core/client/app/components/gh-alerts.js | 22 - core/client/app/components/gh-app.js | 15 - core/client/app/components/gh-blog-url.js | 12 - core/client/app/components/gh-cm-editor.js | 47 -- .../client/app/components/gh-content-cover.js | 30 - .../components/gh-content-preview-content.js | 22 - .../components/gh-content-view-container.js | 15 - .../app/components/gh-datetime-input.js | 33 - .../app/components/gh-dropdown-button.js | 24 - core/client/app/components/gh-dropdown.js | 99 --- core/client/app/components/gh-ed-editor.js | 52 -- core/client/app/components/gh-ed-preview.js | 110 --- .../app/components/gh-editor-save-button.js | 50 -- core/client/app/components/gh-editor.js | 123 --- .../client/app/components/gh-error-message.js | 36 - core/client/app/components/gh-feature-flag.js | 45 -- core/client/app/components/gh-file-upload.js | 37 - .../client/app/components/gh-file-uploader.js | 161 ---- core/client/app/components/gh-form-group.js | 5 - .../app/components/gh-fullscreen-modal.js | 85 --- .../gh-image-uploader-with-preview.js | 39 - .../app/components/gh-image-uploader.js | 226 ------ .../app/components/gh-infinite-scroll.js | 12 - core/client/app/components/gh-input.js | 8 - core/client/app/components/gh-light-table.js | 27 - core/client/app/components/gh-main.js | 13 - core/client/app/components/gh-menu-toggle.js | 42 -- core/client/app/components/gh-nav-menu.js | 48 -- core/client/app/components/gh-navigation.js | 39 - .../app/components/gh-navitem-url-input.js | 145 ---- core/client/app/components/gh-navitem.js | 55 -- core/client/app/components/gh-notification.js | 56 -- .../client/app/components/gh-notifications.js | 17 - .../app/components/gh-popover-button.js | 26 - core/client/app/components/gh-popover.js | 11 - .../app/components/gh-posts-list-item.js | 85 --- .../client/app/components/gh-profile-image.js | 142 ---- core/client/app/components/gh-search-input.js | 206 ----- .../app/components/gh-search-input/trigger.js | 43 -- .../client/app/components/gh-select-native.js | 42 -- core/client/app/components/gh-selectize.js | 124 --- core/client/app/components/gh-skip-link.js | 36 - core/client/app/components/gh-spin-button.js | 59 -- .../app/components/gh-subscribers-table.js | 17 - core/client/app/components/gh-tab-pane.js | 34 - core/client/app/components/gh-tab.js | 35 - core/client/app/components/gh-tabs-manager.js | 85 --- .../app/components/gh-tag-settings-form.js | 136 ---- core/client/app/components/gh-tag.js | 12 - .../gh-tags-management-container.js | 54 -- core/client/app/components/gh-textarea.js | 8 - .../app/components/gh-trim-focus-input.js | 41 - core/client/app/components/gh-url-preview.js | 35 - core/client/app/components/gh-user-active.js | 31 - core/client/app/components/gh-user-invited.js | 69 -- .../gh-validation-status-container.js | 26 - core/client/app/components/gh-view-title.js | 14 - core/client/app/components/modals/base.js | 56 -- .../client/app/components/modals/copy-html.js | 9 - .../app/components/modals/delete-all.js | 49 -- .../app/components/modals/delete-post.js | 55 -- .../components/modals/delete-subscriber.js | 23 - .../app/components/modals/delete-tag.js | 27 - .../app/components/modals/delete-user.js | 19 - .../components/modals/import-subscribers.js | 43 -- .../app/components/modals/invite-new-user.js | 125 --- .../app/components/modals/leave-editor.js | 12 - .../app/components/modals/markdown-help.js | 4 - .../app/components/modals/new-subscriber.js | 32 - .../app/components/modals/re-authenticate.js | 68 -- .../app/components/modals/transfer-owner.js | 17 - .../app/components/modals/upload-image.js | 101 --- core/client/app/controllers/about.js | 21 - core/client/app/controllers/application.js | 57 -- core/client/app/controllers/editor/edit.js | 14 - core/client/app/controllers/editor/new.js | 25 - core/client/app/controllers/error.js | 20 - .../app/controllers/post-settings-menu.js | 491 ------------ core/client/app/controllers/posts.js | 88 --- core/client/app/controllers/reset.js | 75 -- .../app/controllers/settings/apps/index.js | 14 - .../app/controllers/settings/apps/slack.js | 75 -- .../controllers/settings/code-injection.js | 19 - .../app/controllers/settings/general.js | 253 ------- core/client/app/controllers/settings/labs.js | 89 --- .../app/controllers/settings/navigation.js | 101 --- core/client/app/controllers/settings/tags.js | 43 -- .../app/controllers/settings/tags/tag.js | 81 -- core/client/app/controllers/setup.js | 23 - core/client/app/controllers/setup/three.js | 236 ------ core/client/app/controllers/setup/two.js | 152 ---- core/client/app/controllers/signin.js | 114 --- core/client/app/controllers/signup.js | 96 --- core/client/app/controllers/subscribers.js | 162 ---- core/client/app/controllers/team/index.js | 33 - core/client/app/controllers/team/user.js | 425 ----------- .../client/app/helpers/gh-count-characters.js | 25 - .../app/helpers/gh-count-down-characters.js | 28 - core/client/app/helpers/gh-count-words.js | 20 - core/client/app/helpers/gh-format-html.js | 26 - core/client/app/helpers/gh-format-markdown.js | 34 - core/client/app/helpers/gh-format-timeago.js | 16 - core/client/app/helpers/gh-path.js | 55 -- core/client/app/helpers/gh-user-can-admin.js | 16 - core/client/app/helpers/highlighted-text.js | 7 - core/client/app/helpers/is-equal.js | 13 - core/client/app/helpers/is-not.js | 11 - core/client/app/html/apps.html | 305 -------- core/client/app/html/permalinks.html | 250 ------ core/client/app/html/themes.html | 273 ------- core/client/app/index.html | 56 -- .../app/initializers/ember-simple-auth.js | 17 - .../app/initializers/trailing-history.js | 23 - .../jquery-ajax-oauth-prefilter.js | 21 - core/client/app/mirage/config.js | 436 ----------- .../app/mirage/factories/notification.js | 9 - core/client/app/mirage/factories/post.js | 23 - core/client/app/mirage/factories/role.js | 12 - core/client/app/mirage/factories/setting.js | 13 - .../client/app/mirage/factories/subscriber.js | 21 - core/client/app/mirage/factories/tag.js | 23 - core/client/app/mirage/factories/user.js | 27 - core/client/app/mirage/fixtures/roles.js | 43 -- core/client/app/mirage/fixtures/settings.js | 218 ------ core/client/app/mirage/scenarios/default.js | 8 - core/client/app/mixins/404-handler.js | 23 - core/client/app/mixins/active-link-wrapper.js | 32 - core/client/app/mixins/body-event-listener.js | 49 -- .../app/mixins/current-user-settings.js | 25 - core/client/app/mixins/dropdown-mixin.js | 17 - core/client/app/mixins/ed-editor-api.js | 142 ---- core/client/app/mixins/ed-editor-scroll.js | 100 --- core/client/app/mixins/ed-editor-shortcuts.js | 173 ----- .../app/mixins/editor-base-controller.js | 424 ----------- core/client/app/mixins/editor-base-route.js | 144 ---- core/client/app/mixins/infinite-scroll.js | 43 -- core/client/app/mixins/pagination.js | 118 --- .../app/mixins/settings-menu-controller.js | 34 - core/client/app/mixins/settings-save.js | 17 - core/client/app/mixins/shortcuts-route.js | 49 -- core/client/app/mixins/shortcuts.js | 79 -- core/client/app/mixins/slug-url.js | 16 - core/client/app/mixins/style-body.js | 32 - core/client/app/mixins/text-input.js | 25 - core/client/app/mixins/validation-engine.js | 162 ---- core/client/app/mixins/validation-state.js | 34 - core/client/app/models/navigation-item.js | 27 - core/client/app/models/notification.js | 9 - core/client/app/models/post.js | 87 --- core/client/app/models/role.js | 20 - core/client/app/models/setting.js | 28 - core/client/app/models/slack-integration.js | 19 - core/client/app/models/subscriber.js | 23 - core/client/app/models/tag.js | 23 - core/client/app/models/user.js | 121 --- core/client/app/resolver.js | 3 - core/client/app/router.js | 70 -- core/client/app/routes/about.js | 36 - core/client/app/routes/application.js | 132 ---- core/client/app/routes/authenticated.js | 6 - core/client/app/routes/editor/edit.js | 65 -- core/client/app/routes/editor/index.js | 10 - core/client/app/routes/editor/new.js | 50 -- core/client/app/routes/error404.js | 15 - core/client/app/routes/mobile-index-route.js | 37 - core/client/app/routes/posts.js | 100 --- core/client/app/routes/posts/index.js | 57 -- core/client/app/routes/posts/post.js | 77 -- core/client/app/routes/reset.js | 33 - core/client/app/routes/settings/apps.js | 20 - core/client/app/routes/settings/apps/slack.js | 19 - .../app/routes/settings/code-injection.js | 27 - core/client/app/routes/settings/general.js | 26 - core/client/app/routes/settings/labs.js | 22 - core/client/app/routes/settings/navigation.js | 46 -- core/client/app/routes/settings/tags.js | 104 --- core/client/app/routes/settings/tags/index.js | 20 - core/client/app/routes/settings/tags/new.js | 21 - core/client/app/routes/settings/tags/tag.js | 20 - core/client/app/routes/setup.js | 59 -- core/client/app/routes/setup/index.js | 10 - core/client/app/routes/setup/one.js | 64 -- core/client/app/routes/setup/three.js | 12 - core/client/app/routes/signin.js | 45 -- core/client/app/routes/signout.js | 25 - core/client/app/routes/signup.js | 77 -- core/client/app/routes/subscribers.js | 54 -- core/client/app/routes/subscribers/import.js | 9 - core/client/app/routes/subscribers/new.js | 37 - core/client/app/routes/team/index.js | 32 - core/client/app/routes/team/user.js | 58 -- core/client/app/serializers/application.js | 27 - core/client/app/serializers/post.js | 56 -- core/client/app/serializers/setting.js | 46 -- core/client/app/serializers/tag.js | 20 - core/client/app/serializers/user.js | 29 - core/client/app/services/ajax.js | 72 -- core/client/app/services/config.js | 44 -- core/client/app/services/dropdown.js | 20 - core/client/app/services/feature.js | 84 --- core/client/app/services/ghost-paths.js | 8 - core/client/app/services/media-queries.js | 48 -- core/client/app/services/notifications.js | 187 ----- core/client/app/services/session.js | 24 - core/client/app/services/slug-generator.js | 29 - core/client/app/session-stores/application.js | 10 - core/client/app/styles/app.css | 48 -- core/client/app/styles/components/badges.css | 74 -- .../app/styles/components/dropdowns.css | 149 ---- core/client/app/styles/components/modals.css | 183 ----- .../app/styles/components/notifications.css | 219 ------ .../app/styles/components/pagination.css | 111 --- .../client/app/styles/components/popovers.css | 59 -- .../app/styles/components/power-select.css | 123 --- .../app/styles/components/selectize.css | 405 ---------- .../app/styles/components/settings-menu.css | 180 ----- .../app/styles/components/splitbuttons.css | 62 -- .../client/app/styles/components/uploader.css | 162 ---- core/client/app/styles/csscomb.json | 235 ------ core/client/app/styles/layouts/about.css | 123 --- core/client/app/styles/layouts/apps.css | 252 ------- core/client/app/styles/layouts/auth.css | 56 -- core/client/app/styles/layouts/content.css | 289 ------- core/client/app/styles/layouts/editor.css | 403 ---------- core/client/app/styles/layouts/error.css | 88 --- core/client/app/styles/layouts/flow.css | 464 ------------ core/client/app/styles/layouts/main.css | 560 -------------- core/client/app/styles/layouts/packages.css | 376 --------- core/client/app/styles/layouts/settings.css | 163 ---- .../client/app/styles/layouts/subscribers.css | 69 -- core/client/app/styles/layouts/tags.css | 129 ---- core/client/app/styles/layouts/user.css | 217 ------ core/client/app/styles/layouts/users.css | 202 ----- core/client/app/styles/patterns/_shame.css | 7 - core/client/app/styles/patterns/buttons.css | 237 ------ core/client/app/styles/patterns/forms.css | 343 --------- core/client/app/styles/patterns/global.css | 441 ----------- core/client/app/styles/patterns/icons.css | 266 ------- core/client/app/styles/patterns/labels.css | 119 --- core/client/app/styles/patterns/navlist.css | 71 -- core/client/app/styles/patterns/tables.css | 79 -- core/client/app/templates/-import-errors.hbs | 7 - core/client/app/templates/-user-list-item.hbs | 18 - core/client/app/templates/about.hbs | 44 -- core/client/app/templates/application.hbs | 28 - .../components/gh-activating-list-item.hbs | 1 - .../app/templates/components/gh-alert.hbs | 4 - .../app/templates/components/gh-alerts.hbs | 3 - .../app/templates/components/gh-app.hbs | 1 - .../app/templates/components/gh-blog-url.hbs | 1 - .../components/gh-content-preview-content.hbs | 1 - .../components/gh-content-view-container.hbs | 1 - .../components/gh-datetime-input.hbs | 5 - .../templates/components/gh-ed-preview.hbs | 14 - .../components/gh-editor-save-button.hbs | 22 - .../app/templates/components/gh-editor.hbs | 50 -- .../templates/components/gh-error-message.hbs | 1 - .../templates/components/gh-feature-flag.hbs | 3 - .../templates/components/gh-file-upload.hbs | 4 - .../templates/components/gh-file-uploader.hbs | 20 - .../components/gh-fullscreen-modal.hbs | 11 - .../gh-image-uploader-with-preview.hbs | 16 - .../components/gh-image-uploader.hbs | 43 -- .../components/gh-infinite-scroll.hbs | 1 - .../templates/components/gh-menu-toggle.hbs | 1 - .../templates/components/gh-modal-dialog.hbs | 18 - .../app/templates/components/gh-nav-menu.hbs | 69 -- .../templates/components/gh-navigation.hbs | 1 - .../app/templates/components/gh-navitem.hbs | 26 - .../templates/components/gh-notification.hbs | 4 - .../templates/components/gh-notifications.hbs | 3 - .../components/gh-posts-list-item.hbs | 1 - .../templates/components/gh-profile-image.hbs | 17 - .../templates/components/gh-search-input.hbs | 13 - .../components/gh-search-input/trigger.hbs | 12 - .../templates/components/gh-select-native.hbs | 14 - .../templates/components/gh-spin-button.hbs | 9 - .../gh-subscribers-table-delete-cell.hbs | 1 - .../components/gh-subscribers-table.hbs | 17 - .../components/gh-tag-settings-form.hbs | 88 --- .../app/templates/components/gh-tag.hbs | 8 - .../gh-tags-management-container.hbs | 1 - .../templates/components/gh-url-preview.hbs | 1 - .../templates/components/gh-user-active.hbs | 1 - .../templates/components/gh-user-invited.hbs | 1 - .../templates/components/gh-view-title.hbs | 2 - .../templates/components/modals/copy-html.hbs | 8 - .../components/modals/delete-all.hbs | 13 - .../components/modals/delete-post.hbs | 17 - .../components/modals/delete-subscriber.hbs | 13 - .../components/modals/delete-tag.hbs | 17 - .../components/modals/delete-user.hbs | 17 - .../components/modals/import-subscribers.hbs | 47 -- .../components/modals/invite-new-user.hbs | 41 - .../components/modals/leave-editor.hbs | 18 - .../components/modals/markdown-help.hbs | 81 -- .../components/modals/new-subscriber.hbs | 29 - .../components/modals/re-authenticate.hbs | 16 - .../components/modals/transfer-owner.hbs | 16 - .../components/modals/upload-image.hbs | 20 - core/client/app/templates/editor/edit.hbs | 48 -- core/client/app/templates/error.hbs | 27 - .../app/templates/post-settings-menu.hbs | 144 ---- core/client/app/templates/posts.hbs | 53 -- core/client/app/templates/posts/index.hbs | 8 - core/client/app/templates/posts/post.hbs | 14 - core/client/app/templates/reset.hbs | 18 - core/client/app/templates/settings/apps.hbs | 3 - .../app/templates/settings/apps/index.hbs | 33 - .../app/templates/settings/apps/slack.hbs | 42 -- .../app/templates/settings/code-injection.hbs | 30 - .../client/app/templates/settings/general.hbs | 137 ---- core/client/app/templates/settings/labs.hbs | 66 -- .../app/templates/settings/navigation.hbs | 19 - core/client/app/templates/settings/tags.hbs | 27 - .../app/templates/settings/tags/index.hbs | 6 - .../app/templates/settings/tags/tag.hbs | 11 - core/client/app/templates/setup.hbs | 27 - core/client/app/templates/setup/one.hbs | 12 - core/client/app/templates/setup/three.hbs | 19 - core/client/app/templates/setup/two.hbs | 44 -- core/client/app/templates/signin.hbs | 22 - core/client/app/templates/signup.hbs | 43 -- core/client/app/templates/subscribers.hbs | 49 -- .../app/templates/subscribers/import.hbs | 3 - core/client/app/templates/subscribers/new.hbs | 4 - core/client/app/templates/team/index.hbs | 82 -- core/client/app/templates/team/user.hbs | 197 ----- .../app/transforms/facebook-url-user.js | 21 - core/client/app/transforms/moment-date.js | 18 - .../app/transforms/navigation-settings.js | 41 - core/client/app/transforms/raw.js | 11 - core/client/app/transforms/slack-settings.js | 37 - .../client/app/transforms/twitter-url-user.js | 21 - core/client/app/transitions.js | 16 - core/client/app/utils/ajax.js | 49 -- core/client/app/utils/bound-one-way.js | 31 - core/client/app/utils/caja-sanitizers.js | 26 - core/client/app/utils/ctrl-or-cmd.js | 1 - core/client/app/utils/date-formatting.js | 39 - core/client/app/utils/document-title.js | 58 -- core/client/app/utils/ed-image-manager.js | 40 - core/client/app/utils/editor-shortcuts.js | 31 - core/client/app/utils/ghost-paths.js | 55 -- core/client/app/utils/isFinite.js | 7 - core/client/app/utils/isNumber.js | 8 - core/client/app/utils/link-component.js | 20 - core/client/app/utils/random-password.js | 8 - core/client/app/utils/text-field.js | 7 - core/client/app/utils/titleize.js | 16 - core/client/app/utils/validator-extensions.js | 19 - core/client/app/utils/window-proxy.js | 9 - core/client/app/utils/word-count.js | 18 - core/client/app/validators/base.js | 37 - core/client/app/validators/invite-user.js | 17 - core/client/app/validators/nav-item.js | 36 - core/client/app/validators/new-user.js | 35 - core/client/app/validators/post.js | 37 - core/client/app/validators/reset.js | 21 - core/client/app/validators/setting.js | 47 -- core/client/app/validators/setup.js | 14 - core/client/app/validators/signin.js | 48 -- core/client/app/validators/signup.js | 3 - .../app/validators/slack-integration.js | 24 - core/client/app/validators/subscriber.js | 19 - core/client/app/validators/tag-settings.js | 56 -- core/client/app/validators/user.js | 81 -- core/client/bower.json | 30 - core/client/config/deprecation-workflow.js | 6 - core/client/config/environment.js | 54 -- core/client/ember-cli-build.js | 85 --- core/client/lib/.jshintrc | 4 - core/client/lib/asset-delivery/index.js | 22 - core/client/lib/asset-delivery/package.json | 6 - core/client/package.json | 71 -- .../client/public/assets/fonts/ghosticons.eot | Bin 22584 -> 0 bytes .../client/public/assets/fonts/ghosticons.svg | 83 -- .../client/public/assets/fonts/ghosticons.ttf | Bin 22420 -> 0 bytes .../public/assets/fonts/ghosticons.woff | Bin 22496 -> 0 bytes core/client/public/assets/img/404-ghost.png | Bin 3681 -> 0 bytes .../client/public/assets/img/404-ghost@2x.png | Bin 7958 -> 0 bytes core/client/public/assets/img/ghost-logo.png | Bin 8638 -> 0 bytes core/client/public/assets/img/ghosticon.jpg | Bin 2499 -> 0 bytes .../public/assets/img/install-welcome.png | Bin 240204 -> 0 bytes .../public/assets/img/invite-placeholder.png | Bin 7860 -> 0 bytes core/client/public/assets/img/large.png | Bin 1912 -> 0 bytes core/client/public/assets/img/loadingcat.gif | Bin 20207 -> 0 bytes core/client/public/assets/img/medium.png | Bin 400 -> 0 bytes core/client/public/assets/img/slackicon.png | Bin 18136 -> 0 bytes core/client/public/assets/img/small.png | Bin 426 -> 0 bytes .../public/assets/img/touch-icon-ipad.png | Bin 494 -> 0 bytes .../public/assets/img/touch-icon-iphone.png | Bin 640 -> 0 bytes core/client/public/assets/img/user-cover.png | Bin 23563 -> 0 bytes core/client/public/assets/img/user-image.png | Bin 2421 -> 0 bytes core/client/public/assets/img/users.png | Bin 49253 -> 0 bytes core/client/testem.js | 14 - core/client/tests/.jscsrc | 4 - core/client/tests/.jshintrc | 58 -- .../tests/acceptance/authentication-test.js | 195 ----- .../tests/acceptance/password-reset-test.js | 107 --- .../tests/acceptance/posts/post-test.js | 71 -- .../tests/acceptance/settings/apps-test.js | 89 --- .../settings/code-injection-test.js | 91 --- .../tests/acceptance/settings/general-test.js | 325 -------- .../tests/acceptance/settings/labs-test.js | 95 --- .../acceptance/settings/navigation-test.js | 224 ------ .../tests/acceptance/settings/slack-test.js | 121 --- .../tests/acceptance/settings/tags-test.js | 305 -------- core/client/tests/acceptance/setup-test.js | 402 ---------- core/client/tests/acceptance/signin-test.js | 131 ---- .../tests/acceptance/subscribers-test.js | 256 ------- core/client/tests/acceptance/team-test.js | 479 ------------ core/client/tests/helpers/adapter-error.js | 19 - core/client/tests/helpers/destroy-app.js | 5 - .../tests/helpers/module-for-acceptance.js | 23 - core/client/tests/helpers/resolver.js | 11 - core/client/tests/helpers/start-app.js | 21 - core/client/tests/index.html | 60 -- .../tests/integration/adapters/tag-test.js | 63 -- .../tests/integration/adapters/user-test.js | 86 --- .../integration/components/gh-alert-test.js | 46 -- .../integration/components/gh-alerts-test.js | 58 -- .../components/gh-cm-editor-test.js | 53 -- .../components/gh-feature-flag-test.js | 60 -- .../components/gh-file-uploader-test.js | 98 --- .../components/gh-image-uploader-test.js | 244 ------ .../gh-image-uploader-with-preview-test.js | 48 -- .../components/gh-navigation-test.js | 75 -- .../integration/components/gh-navitem-test.js | 114 --- .../components/gh-navitem-url-input-test.js | 481 ------------ .../components/gh-notification-test.js | 44 -- .../components/gh-notifications-test.js | 44 -- .../components/gh-profile-image-test.js | 152 ---- .../components/gh-search-input-test.js | 26 - .../components/gh-subscribers-table-test.js | 26 - .../components/gh-tag-settings-form-test.js | 322 -------- .../gh-tags-management-container-test.js | 33 - .../gh-validation-status-container-test.js | 72 -- .../modals/delete-subscriber-test.js | 30 - .../modals/import-subscribers-test.js | 30 - .../components/modals/new-subscriber-test.js | 30 - .../components/transfer-owner-test.js | 39 - .../tests/integration/services/ajax-test.js | 131 ---- .../integration/services/feature-test.js | 207 ----- .../services/slug-generator-test.js | 62 -- core/client/tests/test-helper.js | 11 - core/client/tests/unit/.gitkeep | 0 .../tests/unit/components/gh-alert-test.js | 33 - .../tests/unit/components/gh-app-test.js | 28 - .../gh-content-preview-content-test.js | 28 - .../components/gh-editor-save-button-test.js | 32 - .../tests/unit/components/gh-editor-test.js | 34 - .../unit/components/gh-file-uploader-test.js | 313 -------- .../unit/components/gh-image-uploader-test.js | 344 --------- .../components/gh-infinite-scroll-test.js | 28 - .../components/gh-navitem-url-input-test.js | 39 - .../unit/components/gh-notification-test.js | 50 -- .../components/gh-posts-list-item-test.js | 28 - .../unit/components/gh-select-native-test.js | 28 - .../unit/components/gh-selectize-test.js | 40 - .../unit/components/gh-spin-button-test.js | 28 - .../components/gh-trim-focus-input_test.js | 47 -- .../unit/components/gh-url-preview_test.js | 42 -- .../unit/components/gh-user-active-test.js | 28 - .../unit/components/gh-user-invited-test.js | 28 - .../controllers/post-settings-menu-test.js | 712 ------------------ .../unit/controllers/settings/general-test.js | 93 --- .../controllers/settings/navigation-test.js | 181 ----- .../unit/controllers/subscribers-test.js | 21 - .../unit/helpers/gh-user-can-admin-test.js | 59 -- .../unit/helpers/highlighted-text-test.js | 18 - .../tests/unit/helpers/is-equal-test.js | 18 - core/client/tests/unit/helpers/is-not-test.js | 18 - .../tests/unit/mixins/infinite-scroll-test.js | 18 - .../unit/mixins/validation-engine-test.js | 42 -- .../tests/unit/models/navigation-item-test.js | 67 -- core/client/tests/unit/models/post-test.js | 84 --- core/client/tests/unit/models/role-test.js | 25 - core/client/tests/unit/models/setting-test.js | 12 - .../tests/unit/models/subscriber-test.js | 20 - core/client/tests/unit/models/tag-test.js | 12 - core/client/tests/unit/models/user-test.js | 141 ---- .../tests/unit/routes/subscribers-test.js | 20 - .../unit/routes/subscribers/import-test.js | 21 - .../tests/unit/routes/subscribers/new-test.js | 20 - .../client/tests/unit/services/config-test.js | 34 - .../tests/unit/services/notifications-test.js | 419 ----------- .../unit/transforms/facebook-url-user-test.js | 32 - .../transforms/navigation-settings-test.js | 42 -- .../unit/transforms/slack-settings-test.js | 37 - .../unit/transforms/twitter-url-user-test.js | 32 - .../tests/unit/utils/ghost-paths-test.js | 58 -- .../tests/unit/validators/nav-item-test.js | 101 --- .../unit/validators/slack-integration-test.js | 66 -- .../tests/unit/validators/subscriber-test.js | 77 -- .../unit/validators/tag-settings-test.js | 306 -------- 514 files changed, 33625 deletions(-) delete mode 100644 core/client/.ember-cli delete mode 100644 core/client/.gitignore delete mode 100644 core/client/.jscsrc delete mode 100644 core/client/.jshintrc delete mode 100644 core/client/.watchmanconfig delete mode 100644 core/client/app/README.md delete mode 100644 core/client/app/_config.yml delete mode 100644 core/client/app/adapters/application.js delete mode 100644 core/client/app/adapters/base.js delete mode 100644 core/client/app/adapters/embedded-relation-adapter.js delete mode 100644 core/client/app/adapters/setting.js delete mode 100644 core/client/app/adapters/tag.js delete mode 100644 core/client/app/adapters/user.js delete mode 100755 core/client/app/app.js delete mode 100644 core/client/app/authenticators/oauth2.js delete mode 100644 core/client/app/authorizers/oauth2.js delete mode 100644 core/client/app/components/gh-activating-list-item.js delete mode 100644 core/client/app/components/gh-alert.js delete mode 100644 core/client/app/components/gh-alerts.js delete mode 100644 core/client/app/components/gh-app.js delete mode 100644 core/client/app/components/gh-blog-url.js delete mode 100644 core/client/app/components/gh-cm-editor.js delete mode 100644 core/client/app/components/gh-content-cover.js delete mode 100644 core/client/app/components/gh-content-preview-content.js delete mode 100644 core/client/app/components/gh-content-view-container.js delete mode 100644 core/client/app/components/gh-datetime-input.js delete mode 100644 core/client/app/components/gh-dropdown-button.js delete mode 100644 core/client/app/components/gh-dropdown.js delete mode 100644 core/client/app/components/gh-ed-editor.js delete mode 100644 core/client/app/components/gh-ed-preview.js delete mode 100644 core/client/app/components/gh-editor-save-button.js delete mode 100644 core/client/app/components/gh-editor.js delete mode 100644 core/client/app/components/gh-error-message.js delete mode 100644 core/client/app/components/gh-feature-flag.js delete mode 100644 core/client/app/components/gh-file-upload.js delete mode 100644 core/client/app/components/gh-file-uploader.js delete mode 100644 core/client/app/components/gh-form-group.js delete mode 100644 core/client/app/components/gh-fullscreen-modal.js delete mode 100644 core/client/app/components/gh-image-uploader-with-preview.js delete mode 100644 core/client/app/components/gh-image-uploader.js delete mode 100644 core/client/app/components/gh-infinite-scroll.js delete mode 100644 core/client/app/components/gh-input.js delete mode 100644 core/client/app/components/gh-light-table.js delete mode 100644 core/client/app/components/gh-main.js delete mode 100644 core/client/app/components/gh-menu-toggle.js delete mode 100644 core/client/app/components/gh-nav-menu.js delete mode 100644 core/client/app/components/gh-navigation.js delete mode 100644 core/client/app/components/gh-navitem-url-input.js delete mode 100644 core/client/app/components/gh-navitem.js delete mode 100644 core/client/app/components/gh-notification.js delete mode 100644 core/client/app/components/gh-notifications.js delete mode 100644 core/client/app/components/gh-popover-button.js delete mode 100644 core/client/app/components/gh-popover.js delete mode 100644 core/client/app/components/gh-posts-list-item.js delete mode 100644 core/client/app/components/gh-profile-image.js delete mode 100644 core/client/app/components/gh-search-input.js delete mode 100644 core/client/app/components/gh-search-input/trigger.js delete mode 100644 core/client/app/components/gh-select-native.js delete mode 100644 core/client/app/components/gh-selectize.js delete mode 100644 core/client/app/components/gh-skip-link.js delete mode 100644 core/client/app/components/gh-spin-button.js delete mode 100644 core/client/app/components/gh-subscribers-table.js delete mode 100644 core/client/app/components/gh-tab-pane.js delete mode 100644 core/client/app/components/gh-tab.js delete mode 100644 core/client/app/components/gh-tabs-manager.js delete mode 100644 core/client/app/components/gh-tag-settings-form.js delete mode 100644 core/client/app/components/gh-tag.js delete mode 100644 core/client/app/components/gh-tags-management-container.js delete mode 100644 core/client/app/components/gh-textarea.js delete mode 100644 core/client/app/components/gh-trim-focus-input.js delete mode 100644 core/client/app/components/gh-url-preview.js delete mode 100644 core/client/app/components/gh-user-active.js delete mode 100644 core/client/app/components/gh-user-invited.js delete mode 100644 core/client/app/components/gh-validation-status-container.js delete mode 100644 core/client/app/components/gh-view-title.js delete mode 100644 core/client/app/components/modals/base.js delete mode 100644 core/client/app/components/modals/copy-html.js delete mode 100644 core/client/app/components/modals/delete-all.js delete mode 100644 core/client/app/components/modals/delete-post.js delete mode 100644 core/client/app/components/modals/delete-subscriber.js delete mode 100644 core/client/app/components/modals/delete-tag.js delete mode 100644 core/client/app/components/modals/delete-user.js delete mode 100644 core/client/app/components/modals/import-subscribers.js delete mode 100644 core/client/app/components/modals/invite-new-user.js delete mode 100644 core/client/app/components/modals/leave-editor.js delete mode 100644 core/client/app/components/modals/markdown-help.js delete mode 100644 core/client/app/components/modals/new-subscriber.js delete mode 100644 core/client/app/components/modals/re-authenticate.js delete mode 100644 core/client/app/components/modals/transfer-owner.js delete mode 100644 core/client/app/components/modals/upload-image.js delete mode 100644 core/client/app/controllers/about.js delete mode 100644 core/client/app/controllers/application.js delete mode 100644 core/client/app/controllers/editor/edit.js delete mode 100644 core/client/app/controllers/editor/new.js delete mode 100644 core/client/app/controllers/error.js delete mode 100644 core/client/app/controllers/post-settings-menu.js delete mode 100644 core/client/app/controllers/posts.js delete mode 100644 core/client/app/controllers/reset.js delete mode 100644 core/client/app/controllers/settings/apps/index.js delete mode 100644 core/client/app/controllers/settings/apps/slack.js delete mode 100644 core/client/app/controllers/settings/code-injection.js delete mode 100644 core/client/app/controllers/settings/general.js delete mode 100644 core/client/app/controllers/settings/labs.js delete mode 100644 core/client/app/controllers/settings/navigation.js delete mode 100644 core/client/app/controllers/settings/tags.js delete mode 100644 core/client/app/controllers/settings/tags/tag.js delete mode 100644 core/client/app/controllers/setup.js delete mode 100644 core/client/app/controllers/setup/three.js delete mode 100644 core/client/app/controllers/setup/two.js delete mode 100644 core/client/app/controllers/signin.js delete mode 100644 core/client/app/controllers/signup.js delete mode 100644 core/client/app/controllers/subscribers.js delete mode 100644 core/client/app/controllers/team/index.js delete mode 100644 core/client/app/controllers/team/user.js delete mode 100644 core/client/app/helpers/gh-count-characters.js delete mode 100644 core/client/app/helpers/gh-count-down-characters.js delete mode 100644 core/client/app/helpers/gh-count-words.js delete mode 100644 core/client/app/helpers/gh-format-html.js delete mode 100644 core/client/app/helpers/gh-format-markdown.js delete mode 100644 core/client/app/helpers/gh-format-timeago.js delete mode 100644 core/client/app/helpers/gh-path.js delete mode 100644 core/client/app/helpers/gh-user-can-admin.js delete mode 100644 core/client/app/helpers/highlighted-text.js delete mode 100644 core/client/app/helpers/is-equal.js delete mode 100644 core/client/app/helpers/is-not.js delete mode 100644 core/client/app/html/apps.html delete mode 100644 core/client/app/html/permalinks.html delete mode 100644 core/client/app/html/themes.html delete mode 100644 core/client/app/index.html delete mode 100644 core/client/app/initializers/ember-simple-auth.js delete mode 100644 core/client/app/initializers/trailing-history.js delete mode 100644 core/client/app/instance-initializers/jquery-ajax-oauth-prefilter.js delete mode 100644 core/client/app/mirage/config.js delete mode 100644 core/client/app/mirage/factories/notification.js delete mode 100644 core/client/app/mirage/factories/post.js delete mode 100644 core/client/app/mirage/factories/role.js delete mode 100644 core/client/app/mirage/factories/setting.js delete mode 100644 core/client/app/mirage/factories/subscriber.js delete mode 100644 core/client/app/mirage/factories/tag.js delete mode 100644 core/client/app/mirage/factories/user.js delete mode 100644 core/client/app/mirage/fixtures/roles.js delete mode 100644 core/client/app/mirage/fixtures/settings.js delete mode 100644 core/client/app/mirage/scenarios/default.js delete mode 100644 core/client/app/mixins/404-handler.js delete mode 100644 core/client/app/mixins/active-link-wrapper.js delete mode 100644 core/client/app/mixins/body-event-listener.js delete mode 100644 core/client/app/mixins/current-user-settings.js delete mode 100644 core/client/app/mixins/dropdown-mixin.js delete mode 100644 core/client/app/mixins/ed-editor-api.js delete mode 100644 core/client/app/mixins/ed-editor-scroll.js delete mode 100644 core/client/app/mixins/ed-editor-shortcuts.js delete mode 100644 core/client/app/mixins/editor-base-controller.js delete mode 100644 core/client/app/mixins/editor-base-route.js delete mode 100644 core/client/app/mixins/infinite-scroll.js delete mode 100644 core/client/app/mixins/pagination.js delete mode 100644 core/client/app/mixins/settings-menu-controller.js delete mode 100644 core/client/app/mixins/settings-save.js delete mode 100644 core/client/app/mixins/shortcuts-route.js delete mode 100644 core/client/app/mixins/shortcuts.js delete mode 100644 core/client/app/mixins/slug-url.js delete mode 100644 core/client/app/mixins/style-body.js delete mode 100644 core/client/app/mixins/text-input.js delete mode 100644 core/client/app/mixins/validation-engine.js delete mode 100644 core/client/app/mixins/validation-state.js delete mode 100644 core/client/app/models/navigation-item.js delete mode 100644 core/client/app/models/notification.js delete mode 100644 core/client/app/models/post.js delete mode 100644 core/client/app/models/role.js delete mode 100644 core/client/app/models/setting.js delete mode 100644 core/client/app/models/slack-integration.js delete mode 100644 core/client/app/models/subscriber.js delete mode 100644 core/client/app/models/tag.js delete mode 100644 core/client/app/models/user.js delete mode 100644 core/client/app/resolver.js delete mode 100644 core/client/app/router.js delete mode 100644 core/client/app/routes/about.js delete mode 100644 core/client/app/routes/application.js delete mode 100644 core/client/app/routes/authenticated.js delete mode 100644 core/client/app/routes/editor/edit.js delete mode 100644 core/client/app/routes/editor/index.js delete mode 100644 core/client/app/routes/editor/new.js delete mode 100644 core/client/app/routes/error404.js delete mode 100644 core/client/app/routes/mobile-index-route.js delete mode 100644 core/client/app/routes/posts.js delete mode 100644 core/client/app/routes/posts/index.js delete mode 100644 core/client/app/routes/posts/post.js delete mode 100644 core/client/app/routes/reset.js delete mode 100644 core/client/app/routes/settings/apps.js delete mode 100644 core/client/app/routes/settings/apps/slack.js delete mode 100644 core/client/app/routes/settings/code-injection.js delete mode 100644 core/client/app/routes/settings/general.js delete mode 100644 core/client/app/routes/settings/labs.js delete mode 100644 core/client/app/routes/settings/navigation.js delete mode 100644 core/client/app/routes/settings/tags.js delete mode 100644 core/client/app/routes/settings/tags/index.js delete mode 100644 core/client/app/routes/settings/tags/new.js delete mode 100644 core/client/app/routes/settings/tags/tag.js delete mode 100644 core/client/app/routes/setup.js delete mode 100644 core/client/app/routes/setup/index.js delete mode 100644 core/client/app/routes/setup/one.js delete mode 100644 core/client/app/routes/setup/three.js delete mode 100644 core/client/app/routes/signin.js delete mode 100644 core/client/app/routes/signout.js delete mode 100644 core/client/app/routes/signup.js delete mode 100644 core/client/app/routes/subscribers.js delete mode 100644 core/client/app/routes/subscribers/import.js delete mode 100644 core/client/app/routes/subscribers/new.js delete mode 100644 core/client/app/routes/team/index.js delete mode 100644 core/client/app/routes/team/user.js delete mode 100644 core/client/app/serializers/application.js delete mode 100644 core/client/app/serializers/post.js delete mode 100644 core/client/app/serializers/setting.js delete mode 100644 core/client/app/serializers/tag.js delete mode 100644 core/client/app/serializers/user.js delete mode 100644 core/client/app/services/ajax.js delete mode 100644 core/client/app/services/config.js delete mode 100644 core/client/app/services/dropdown.js delete mode 100644 core/client/app/services/feature.js delete mode 100644 core/client/app/services/ghost-paths.js delete mode 100644 core/client/app/services/media-queries.js delete mode 100644 core/client/app/services/notifications.js delete mode 100644 core/client/app/services/session.js delete mode 100644 core/client/app/services/slug-generator.js delete mode 100644 core/client/app/session-stores/application.js delete mode 100644 core/client/app/styles/app.css delete mode 100644 core/client/app/styles/components/badges.css delete mode 100644 core/client/app/styles/components/dropdowns.css delete mode 100644 core/client/app/styles/components/modals.css delete mode 100644 core/client/app/styles/components/notifications.css delete mode 100644 core/client/app/styles/components/pagination.css delete mode 100644 core/client/app/styles/components/popovers.css delete mode 100644 core/client/app/styles/components/power-select.css delete mode 100644 core/client/app/styles/components/selectize.css delete mode 100644 core/client/app/styles/components/settings-menu.css delete mode 100644 core/client/app/styles/components/splitbuttons.css delete mode 100644 core/client/app/styles/components/uploader.css delete mode 100644 core/client/app/styles/csscomb.json delete mode 100644 core/client/app/styles/layouts/about.css delete mode 100644 core/client/app/styles/layouts/apps.css delete mode 100644 core/client/app/styles/layouts/auth.css delete mode 100644 core/client/app/styles/layouts/content.css delete mode 100644 core/client/app/styles/layouts/editor.css delete mode 100644 core/client/app/styles/layouts/error.css delete mode 100644 core/client/app/styles/layouts/flow.css delete mode 100644 core/client/app/styles/layouts/main.css delete mode 100644 core/client/app/styles/layouts/packages.css delete mode 100644 core/client/app/styles/layouts/settings.css delete mode 100644 core/client/app/styles/layouts/subscribers.css delete mode 100644 core/client/app/styles/layouts/tags.css delete mode 100644 core/client/app/styles/layouts/user.css delete mode 100644 core/client/app/styles/layouts/users.css delete mode 100644 core/client/app/styles/patterns/_shame.css delete mode 100644 core/client/app/styles/patterns/buttons.css delete mode 100644 core/client/app/styles/patterns/forms.css delete mode 100644 core/client/app/styles/patterns/global.css delete mode 100755 core/client/app/styles/patterns/icons.css delete mode 100644 core/client/app/styles/patterns/labels.css delete mode 100644 core/client/app/styles/patterns/navlist.css delete mode 100644 core/client/app/styles/patterns/tables.css delete mode 100644 core/client/app/templates/-import-errors.hbs delete mode 100644 core/client/app/templates/-user-list-item.hbs delete mode 100644 core/client/app/templates/about.hbs delete mode 100644 core/client/app/templates/application.hbs delete mode 100644 core/client/app/templates/components/gh-activating-list-item.hbs delete mode 100644 core/client/app/templates/components/gh-alert.hbs delete mode 100644 core/client/app/templates/components/gh-alerts.hbs delete mode 100644 core/client/app/templates/components/gh-app.hbs delete mode 100644 core/client/app/templates/components/gh-blog-url.hbs delete mode 100644 core/client/app/templates/components/gh-content-preview-content.hbs delete mode 100644 core/client/app/templates/components/gh-content-view-container.hbs delete mode 100644 core/client/app/templates/components/gh-datetime-input.hbs delete mode 100644 core/client/app/templates/components/gh-ed-preview.hbs delete mode 100644 core/client/app/templates/components/gh-editor-save-button.hbs delete mode 100644 core/client/app/templates/components/gh-editor.hbs delete mode 100644 core/client/app/templates/components/gh-error-message.hbs delete mode 100644 core/client/app/templates/components/gh-feature-flag.hbs delete mode 100644 core/client/app/templates/components/gh-file-upload.hbs delete mode 100644 core/client/app/templates/components/gh-file-uploader.hbs delete mode 100644 core/client/app/templates/components/gh-fullscreen-modal.hbs delete mode 100644 core/client/app/templates/components/gh-image-uploader-with-preview.hbs delete mode 100644 core/client/app/templates/components/gh-image-uploader.hbs delete mode 100644 core/client/app/templates/components/gh-infinite-scroll.hbs delete mode 100644 core/client/app/templates/components/gh-menu-toggle.hbs delete mode 100644 core/client/app/templates/components/gh-modal-dialog.hbs delete mode 100644 core/client/app/templates/components/gh-nav-menu.hbs delete mode 100644 core/client/app/templates/components/gh-navigation.hbs delete mode 100644 core/client/app/templates/components/gh-navitem.hbs delete mode 100644 core/client/app/templates/components/gh-notification.hbs delete mode 100644 core/client/app/templates/components/gh-notifications.hbs delete mode 100644 core/client/app/templates/components/gh-posts-list-item.hbs delete mode 100644 core/client/app/templates/components/gh-profile-image.hbs delete mode 100644 core/client/app/templates/components/gh-search-input.hbs delete mode 100644 core/client/app/templates/components/gh-search-input/trigger.hbs delete mode 100644 core/client/app/templates/components/gh-select-native.hbs delete mode 100644 core/client/app/templates/components/gh-spin-button.hbs delete mode 100644 core/client/app/templates/components/gh-subscribers-table-delete-cell.hbs delete mode 100644 core/client/app/templates/components/gh-subscribers-table.hbs delete mode 100644 core/client/app/templates/components/gh-tag-settings-form.hbs delete mode 100644 core/client/app/templates/components/gh-tag.hbs delete mode 100644 core/client/app/templates/components/gh-tags-management-container.hbs delete mode 100644 core/client/app/templates/components/gh-url-preview.hbs delete mode 100644 core/client/app/templates/components/gh-user-active.hbs delete mode 100644 core/client/app/templates/components/gh-user-invited.hbs delete mode 100644 core/client/app/templates/components/gh-view-title.hbs delete mode 100644 core/client/app/templates/components/modals/copy-html.hbs delete mode 100644 core/client/app/templates/components/modals/delete-all.hbs delete mode 100644 core/client/app/templates/components/modals/delete-post.hbs delete mode 100644 core/client/app/templates/components/modals/delete-subscriber.hbs delete mode 100644 core/client/app/templates/components/modals/delete-tag.hbs delete mode 100644 core/client/app/templates/components/modals/delete-user.hbs delete mode 100644 core/client/app/templates/components/modals/import-subscribers.hbs delete mode 100644 core/client/app/templates/components/modals/invite-new-user.hbs delete mode 100644 core/client/app/templates/components/modals/leave-editor.hbs delete mode 100644 core/client/app/templates/components/modals/markdown-help.hbs delete mode 100644 core/client/app/templates/components/modals/new-subscriber.hbs delete mode 100644 core/client/app/templates/components/modals/re-authenticate.hbs delete mode 100644 core/client/app/templates/components/modals/transfer-owner.hbs delete mode 100644 core/client/app/templates/components/modals/upload-image.hbs delete mode 100644 core/client/app/templates/editor/edit.hbs delete mode 100644 core/client/app/templates/error.hbs delete mode 100644 core/client/app/templates/post-settings-menu.hbs delete mode 100644 core/client/app/templates/posts.hbs delete mode 100644 core/client/app/templates/posts/index.hbs delete mode 100644 core/client/app/templates/posts/post.hbs delete mode 100644 core/client/app/templates/reset.hbs delete mode 100644 core/client/app/templates/settings/apps.hbs delete mode 100644 core/client/app/templates/settings/apps/index.hbs delete mode 100644 core/client/app/templates/settings/apps/slack.hbs delete mode 100644 core/client/app/templates/settings/code-injection.hbs delete mode 100644 core/client/app/templates/settings/general.hbs delete mode 100644 core/client/app/templates/settings/labs.hbs delete mode 100644 core/client/app/templates/settings/navigation.hbs delete mode 100644 core/client/app/templates/settings/tags.hbs delete mode 100644 core/client/app/templates/settings/tags/index.hbs delete mode 100644 core/client/app/templates/settings/tags/tag.hbs delete mode 100644 core/client/app/templates/setup.hbs delete mode 100644 core/client/app/templates/setup/one.hbs delete mode 100644 core/client/app/templates/setup/three.hbs delete mode 100644 core/client/app/templates/setup/two.hbs delete mode 100644 core/client/app/templates/signin.hbs delete mode 100644 core/client/app/templates/signup.hbs delete mode 100644 core/client/app/templates/subscribers.hbs delete mode 100644 core/client/app/templates/subscribers/import.hbs delete mode 100644 core/client/app/templates/subscribers/new.hbs delete mode 100644 core/client/app/templates/team/index.hbs delete mode 100644 core/client/app/templates/team/user.hbs delete mode 100644 core/client/app/transforms/facebook-url-user.js delete mode 100644 core/client/app/transforms/moment-date.js delete mode 100644 core/client/app/transforms/navigation-settings.js delete mode 100644 core/client/app/transforms/raw.js delete mode 100644 core/client/app/transforms/slack-settings.js delete mode 100644 core/client/app/transforms/twitter-url-user.js delete mode 100644 core/client/app/transitions.js delete mode 100644 core/client/app/utils/ajax.js delete mode 100644 core/client/app/utils/bound-one-way.js delete mode 100644 core/client/app/utils/caja-sanitizers.js delete mode 100644 core/client/app/utils/ctrl-or-cmd.js delete mode 100644 core/client/app/utils/date-formatting.js delete mode 100644 core/client/app/utils/document-title.js delete mode 100644 core/client/app/utils/ed-image-manager.js delete mode 100644 core/client/app/utils/editor-shortcuts.js delete mode 100644 core/client/app/utils/ghost-paths.js delete mode 100644 core/client/app/utils/isFinite.js delete mode 100644 core/client/app/utils/isNumber.js delete mode 100644 core/client/app/utils/link-component.js delete mode 100644 core/client/app/utils/random-password.js delete mode 100644 core/client/app/utils/text-field.js delete mode 100644 core/client/app/utils/titleize.js delete mode 100644 core/client/app/utils/validator-extensions.js delete mode 100644 core/client/app/utils/window-proxy.js delete mode 100644 core/client/app/utils/word-count.js delete mode 100644 core/client/app/validators/base.js delete mode 100644 core/client/app/validators/invite-user.js delete mode 100644 core/client/app/validators/nav-item.js delete mode 100644 core/client/app/validators/new-user.js delete mode 100644 core/client/app/validators/post.js delete mode 100644 core/client/app/validators/reset.js delete mode 100644 core/client/app/validators/setting.js delete mode 100644 core/client/app/validators/setup.js delete mode 100644 core/client/app/validators/signin.js delete mode 100644 core/client/app/validators/signup.js delete mode 100644 core/client/app/validators/slack-integration.js delete mode 100644 core/client/app/validators/subscriber.js delete mode 100644 core/client/app/validators/tag-settings.js delete mode 100644 core/client/app/validators/user.js delete mode 100644 core/client/bower.json delete mode 100644 core/client/config/deprecation-workflow.js delete mode 100644 core/client/config/environment.js delete mode 100644 core/client/ember-cli-build.js delete mode 100644 core/client/lib/.jshintrc delete mode 100644 core/client/lib/asset-delivery/index.js delete mode 100644 core/client/lib/asset-delivery/package.json delete mode 100644 core/client/package.json delete mode 100755 core/client/public/assets/fonts/ghosticons.eot delete mode 100755 core/client/public/assets/fonts/ghosticons.svg delete mode 100755 core/client/public/assets/fonts/ghosticons.ttf delete mode 100755 core/client/public/assets/fonts/ghosticons.woff delete mode 100644 core/client/public/assets/img/404-ghost.png delete mode 100644 core/client/public/assets/img/404-ghost@2x.png delete mode 100644 core/client/public/assets/img/ghost-logo.png delete mode 100644 core/client/public/assets/img/ghosticon.jpg delete mode 100644 core/client/public/assets/img/install-welcome.png delete mode 100644 core/client/public/assets/img/invite-placeholder.png delete mode 100644 core/client/public/assets/img/large.png delete mode 100644 core/client/public/assets/img/loadingcat.gif delete mode 100644 core/client/public/assets/img/medium.png delete mode 100644 core/client/public/assets/img/slackicon.png delete mode 100644 core/client/public/assets/img/small.png delete mode 100644 core/client/public/assets/img/touch-icon-ipad.png delete mode 100644 core/client/public/assets/img/touch-icon-iphone.png delete mode 100644 core/client/public/assets/img/user-cover.png delete mode 100644 core/client/public/assets/img/user-image.png delete mode 100644 core/client/public/assets/img/users.png delete mode 100644 core/client/testem.js delete mode 100644 core/client/tests/.jscsrc delete mode 100644 core/client/tests/.jshintrc delete mode 100644 core/client/tests/acceptance/authentication-test.js delete mode 100644 core/client/tests/acceptance/password-reset-test.js delete mode 100644 core/client/tests/acceptance/posts/post-test.js delete mode 100644 core/client/tests/acceptance/settings/apps-test.js delete mode 100644 core/client/tests/acceptance/settings/code-injection-test.js delete mode 100644 core/client/tests/acceptance/settings/general-test.js delete mode 100644 core/client/tests/acceptance/settings/labs-test.js delete mode 100644 core/client/tests/acceptance/settings/navigation-test.js delete mode 100644 core/client/tests/acceptance/settings/slack-test.js delete mode 100644 core/client/tests/acceptance/settings/tags-test.js delete mode 100644 core/client/tests/acceptance/setup-test.js delete mode 100644 core/client/tests/acceptance/signin-test.js delete mode 100644 core/client/tests/acceptance/subscribers-test.js delete mode 100644 core/client/tests/acceptance/team-test.js delete mode 100644 core/client/tests/helpers/adapter-error.js delete mode 100644 core/client/tests/helpers/destroy-app.js delete mode 100644 core/client/tests/helpers/module-for-acceptance.js delete mode 100644 core/client/tests/helpers/resolver.js delete mode 100644 core/client/tests/helpers/start-app.js delete mode 100644 core/client/tests/index.html delete mode 100644 core/client/tests/integration/adapters/tag-test.js delete mode 100644 core/client/tests/integration/adapters/user-test.js delete mode 100644 core/client/tests/integration/components/gh-alert-test.js delete mode 100644 core/client/tests/integration/components/gh-alerts-test.js delete mode 100644 core/client/tests/integration/components/gh-cm-editor-test.js delete mode 100644 core/client/tests/integration/components/gh-feature-flag-test.js delete mode 100644 core/client/tests/integration/components/gh-file-uploader-test.js delete mode 100644 core/client/tests/integration/components/gh-image-uploader-test.js delete mode 100644 core/client/tests/integration/components/gh-image-uploader-with-preview-test.js delete mode 100644 core/client/tests/integration/components/gh-navigation-test.js delete mode 100644 core/client/tests/integration/components/gh-navitem-test.js delete mode 100644 core/client/tests/integration/components/gh-navitem-url-input-test.js delete mode 100644 core/client/tests/integration/components/gh-notification-test.js delete mode 100644 core/client/tests/integration/components/gh-notifications-test.js delete mode 100644 core/client/tests/integration/components/gh-profile-image-test.js delete mode 100644 core/client/tests/integration/components/gh-search-input-test.js delete mode 100644 core/client/tests/integration/components/gh-subscribers-table-test.js delete mode 100644 core/client/tests/integration/components/gh-tag-settings-form-test.js delete mode 100644 core/client/tests/integration/components/gh-tags-management-container-test.js delete mode 100644 core/client/tests/integration/components/gh-validation-status-container-test.js delete mode 100644 core/client/tests/integration/components/modals/delete-subscriber-test.js delete mode 100644 core/client/tests/integration/components/modals/import-subscribers-test.js delete mode 100644 core/client/tests/integration/components/modals/new-subscriber-test.js delete mode 100644 core/client/tests/integration/components/transfer-owner-test.js delete mode 100644 core/client/tests/integration/services/ajax-test.js delete mode 100644 core/client/tests/integration/services/feature-test.js delete mode 100644 core/client/tests/integration/services/slug-generator-test.js delete mode 100644 core/client/tests/test-helper.js delete mode 100644 core/client/tests/unit/.gitkeep delete mode 100644 core/client/tests/unit/components/gh-alert-test.js delete mode 100644 core/client/tests/unit/components/gh-app-test.js delete mode 100644 core/client/tests/unit/components/gh-content-preview-content-test.js delete mode 100644 core/client/tests/unit/components/gh-editor-save-button-test.js delete mode 100644 core/client/tests/unit/components/gh-editor-test.js delete mode 100644 core/client/tests/unit/components/gh-file-uploader-test.js delete mode 100644 core/client/tests/unit/components/gh-image-uploader-test.js delete mode 100644 core/client/tests/unit/components/gh-infinite-scroll-test.js delete mode 100644 core/client/tests/unit/components/gh-navitem-url-input-test.js delete mode 100644 core/client/tests/unit/components/gh-notification-test.js delete mode 100644 core/client/tests/unit/components/gh-posts-list-item-test.js delete mode 100644 core/client/tests/unit/components/gh-select-native-test.js delete mode 100644 core/client/tests/unit/components/gh-selectize-test.js delete mode 100644 core/client/tests/unit/components/gh-spin-button-test.js delete mode 100644 core/client/tests/unit/components/gh-trim-focus-input_test.js delete mode 100644 core/client/tests/unit/components/gh-url-preview_test.js delete mode 100644 core/client/tests/unit/components/gh-user-active-test.js delete mode 100644 core/client/tests/unit/components/gh-user-invited-test.js delete mode 100644 core/client/tests/unit/controllers/post-settings-menu-test.js delete mode 100644 core/client/tests/unit/controllers/settings/general-test.js delete mode 100644 core/client/tests/unit/controllers/settings/navigation-test.js delete mode 100644 core/client/tests/unit/controllers/subscribers-test.js delete mode 100644 core/client/tests/unit/helpers/gh-user-can-admin-test.js delete mode 100644 core/client/tests/unit/helpers/highlighted-text-test.js delete mode 100644 core/client/tests/unit/helpers/is-equal-test.js delete mode 100644 core/client/tests/unit/helpers/is-not-test.js delete mode 100644 core/client/tests/unit/mixins/infinite-scroll-test.js delete mode 100644 core/client/tests/unit/mixins/validation-engine-test.js delete mode 100644 core/client/tests/unit/models/navigation-item-test.js delete mode 100644 core/client/tests/unit/models/post-test.js delete mode 100644 core/client/tests/unit/models/role-test.js delete mode 100644 core/client/tests/unit/models/setting-test.js delete mode 100644 core/client/tests/unit/models/subscriber-test.js delete mode 100644 core/client/tests/unit/models/tag-test.js delete mode 100644 core/client/tests/unit/models/user-test.js delete mode 100644 core/client/tests/unit/routes/subscribers-test.js delete mode 100644 core/client/tests/unit/routes/subscribers/import-test.js delete mode 100644 core/client/tests/unit/routes/subscribers/new-test.js delete mode 100644 core/client/tests/unit/services/config-test.js delete mode 100644 core/client/tests/unit/services/notifications-test.js delete mode 100644 core/client/tests/unit/transforms/facebook-url-user-test.js delete mode 100644 core/client/tests/unit/transforms/navigation-settings-test.js delete mode 100644 core/client/tests/unit/transforms/slack-settings-test.js delete mode 100644 core/client/tests/unit/transforms/twitter-url-user-test.js delete mode 100644 core/client/tests/unit/utils/ghost-paths-test.js delete mode 100644 core/client/tests/unit/validators/nav-item-test.js delete mode 100644 core/client/tests/unit/validators/slack-integration-test.js delete mode 100644 core/client/tests/unit/validators/subscriber-test.js delete mode 100644 core/client/tests/unit/validators/tag-settings-test.js diff --git a/core/client/.ember-cli b/core/client/.ember-cli deleted file mode 100644 index 59bb55fe90..0000000000 --- a/core/client/.ember-cli +++ /dev/null @@ -1,9 +0,0 @@ -{ - /** - Ember CLI sends analytics information by default. The data is completely - anonymous, but there are times when you might want to disable this behavior. - - Setting `disableAnalytics` to true will prevent any data from being sent. - */ - "disableAnalytics": true -} diff --git a/core/client/.gitignore b/core/client/.gitignore deleted file mode 100644 index 7fa0a2c1c3..0000000000 --- a/core/client/.gitignore +++ /dev/null @@ -1,20 +0,0 @@ -# See http://help.github.com/ignore-files/ for more about ignoring files. - -# compiled output -/dist -/tmp - -# dependencies -/node_modules -/bower_components - -# misc -/connect.lock -/coverage/* -/libpeerconnection.log -npm-debug.log -testem.log - -# built by grunt -public/assets/img/contributors/ -app/templates/-contributors.hbs diff --git a/core/client/.jscsrc b/core/client/.jscsrc deleted file mode 100644 index 15197078ce..0000000000 --- a/core/client/.jscsrc +++ /dev/null @@ -1,14 +0,0 @@ -{ - "preset": "ember-suave", - "validateIndentation": 4, - "disallowSpacesInFunction": null, - "disallowSpacesInNamedFunctionExpression": { - "beforeOpeningRoundBrace": true - }, - "disallowSpacesInFunctionDeclaration": { - "beforeOpeningRoundBrace": true - }, - "disallowSpacesInsideObjectBrackets": "all", - "requireCommentsToIncludeAccess": null, - "requireSpacesInsideObjectBrackets": null -} diff --git a/core/client/.jshintrc b/core/client/.jshintrc deleted file mode 100644 index c753d6085b..0000000000 --- a/core/client/.jshintrc +++ /dev/null @@ -1,36 +0,0 @@ -{ - "predef": [ - "server", - "document", - "window", - "-Promise", - "-Notification", - "validator", - "moment" - ], - "browser": true, - "boss": true, - "curly": true, - "debug": false, - "devel": true, - "eqeqeq": true, - "evil": true, - "forin": false, - "immed": false, - "laxbreak": false, - "newcap": true, - "noarg": true, - "noempty": false, - "nonew": false, - "nomen": false, - "onevar": false, - "plusplus": false, - "regexp": false, - "undef": true, - "sub": true, - "strict": false, - "white": false, - "eqnull": true, - "esnext": true, - "unused": true -} diff --git a/core/client/.watchmanconfig b/core/client/.watchmanconfig deleted file mode 100644 index e7834e3e4f..0000000000 --- a/core/client/.watchmanconfig +++ /dev/null @@ -1,3 +0,0 @@ -{ - "ignore_dirs": ["tmp", "dist"] -} diff --git a/core/client/app/README.md b/core/client/app/README.md deleted file mode 100644 index 125ed2f578..0000000000 --- a/core/client/app/README.md +++ /dev/null @@ -1,30 +0,0 @@ -# Ghost Admin Client - -Ember.js application used as a client-side admin for the [Ghost](http://ghost.org) blogging platform. This readme is a work in progress guide aimed at explaining the specific nuances of the Ghost Ember app to contributors whose main focus is on this side of things. - - -## CSS - -We use pure CSS, which is pre-processed for backwards compatibility by [Myth](http://myth.io). We do not follow any strict CSS framework, however our general style is pretty similar to BEM. - -Styles are primarily broken up into 4 main categories: - -* **Patterns** - are base level visual styles for HTML elements (eg. Buttons) -* **Components** - are groups of patterns used to create a UI component (eg. Modals) -* **Layouts** - are groups of components used to create application screens (eg. Settings) - -All of these separate files are subsequently imported and compiled in `app.css`. - - -## Front End Standards - -* 4 spaces for HTML & CSS indentation. Never tabs. -* Double quotes only, never single quotes. -* Use tags and elements appropriate for an HTML5 doctype (including self-closing tags) -* Adhere to the [Recess CSS](http://markdotto.com/2011/11/29/css-property-order/) property order. -* Always a space after a property's colon (.e.g, display: block; and not display:block;). -* End all lines with a semi-colon. -* For multiple, comma-separated selectors, place each selector on its own line. -* Use js- prefixed classes for JavaScript hooks into the DOM, and never use these in CSS as per [Slightly Obtrusive JavaSript](http://ozmm.org/posts/slightly_obtrusive_javascript.html) -* Avoid over-nesting CSS. Never nest more than 3 levels deep. -* Use comments to explain "why" not "what" (Good: This requires a z-index in order to appear above mobile navigation. Bad: This is a thing which is always on top!) diff --git a/core/client/app/_config.yml b/core/client/app/_config.yml deleted file mode 100644 index 4c390fe3be..0000000000 --- a/core/client/app/_config.yml +++ /dev/null @@ -1,15 +0,0 @@ -# Dependencies -markdown: kramdown -highlighter: highlighter - -# Permalinks -permalink: pretty - -# Server -source: docs -destination: _gh_pages -host: 0.0.0.0 -port: 9001 -baseurl: -url: http://localhost:9001 -encoding: UTF-8 diff --git a/core/client/app/adapters/application.js b/core/client/app/adapters/application.js deleted file mode 100644 index 924f2e53c1..0000000000 --- a/core/client/app/adapters/application.js +++ /dev/null @@ -1,9 +0,0 @@ -import EmbeddedRelationAdapter from 'ghost/adapters/embedded-relation-adapter'; - -export default EmbeddedRelationAdapter.extend({ - - shouldBackgroundReloadRecord() { - return false; - } - -}); diff --git a/core/client/app/adapters/base.js b/core/client/app/adapters/base.js deleted file mode 100644 index c5d42c0ce9..0000000000 --- a/core/client/app/adapters/base.js +++ /dev/null @@ -1,54 +0,0 @@ -import Ember from 'ember'; -import RESTAdapter from 'ember-data/adapters/rest'; -import ghostPaths from 'ghost/utils/ghost-paths'; -import DataAdapterMixin from 'ember-simple-auth/mixins/data-adapter-mixin'; - -const { - inject: {service} -} = Ember; - -export default RESTAdapter.extend(DataAdapterMixin, { - authorizer: 'authorizer:oauth2', - - host: window.location.origin, - namespace: ghostPaths().apiRoot.slice(1), - - session: service(), - - shouldBackgroundReloadRecord() { - return false; - }, - - query(store, type, query) { - let id; - - if (query.id) { - id = query.id; - delete query.id; - } - - return this.ajax(this.buildURL(type.modelName, id), 'GET', {data: query}); - }, - - buildURL() { - // Ensure trailing slashes - let url = this._super(...arguments); - - if (url.slice(-1) !== '/') { - url += '/'; - } - - return url; - }, - - handleResponse(status) { - if (status === 401) { - if (this.get('session.isAuthenticated')) { - this.get('session').invalidate(); - return; // prevent error from bubbling because invalidate is async - } - } - - return this._super(...arguments); - } -}); diff --git a/core/client/app/adapters/embedded-relation-adapter.js b/core/client/app/adapters/embedded-relation-adapter.js deleted file mode 100644 index f971405c3d..0000000000 --- a/core/client/app/adapters/embedded-relation-adapter.js +++ /dev/null @@ -1,132 +0,0 @@ -import Ember from 'ember'; -import BaseAdapter from 'ghost/adapters/base'; - -const {get, isNone} = Ember; - -// EmbeddedRelationAdapter will augment the query object in calls made to -// DS.Store#findRecord, findAll, query, and queryRecord with the correct "includes" -// (?include=relatedType) by introspecting on the provided subclass of the DS.Model. -// In cases where there is no query object (DS.Model#save, or simple finds) the URL -// that is built will be augmented with ?include=... where appropriate. -// -// Example: -// If a model has an embedded hasMany relation, the related type will be included: -// roles: DS.hasMany('role', { embedded: 'always' }) => ?include=roles - -export default BaseAdapter.extend({ - find(store, type, id, snapshot) { - return this.ajax(this.buildIncludeURL(store, type.modelName, id, snapshot, 'find'), 'GET'); - }, - - findRecord(store, type, id, snapshot) { - return this.ajax(this.buildIncludeURL(store, type.modelName, id, snapshot, 'findRecord'), 'GET'); - }, - - findAll(store, type, sinceToken) { - let query, url; - - if (sinceToken) { - query = {since: sinceToken}; - } - - url = this.buildIncludeURL(store, type.modelName, null, null, 'findAll'); - - return this.ajax(url, 'GET', {data: query}); - }, - - query(store, type, query) { - return this._super(store, type, this.buildQuery(store, type.modelName, query)); - }, - - queryRecord(store, type, query) { - return this._super(store, type, this.buildQuery(store, type.modelName, query)); - }, - - createRecord(store, type, snapshot) { - return this.saveRecord(store, type, snapshot, {method: 'POST'}, 'createRecord'); - }, - - updateRecord(store, type, snapshot) { - let options = { - method: 'PUT', - id: get(snapshot, 'id') - }; - - return this.saveRecord(store, type, snapshot, options, 'updateRecord'); - }, - - saveRecord(store, type, snapshot, options, requestType) { - let _options = options || {}; - let url = this.buildIncludeURL(store, type.modelName, _options.id, snapshot, requestType); - let payload = this.preparePayload(store, type, snapshot); - - return this.ajax(url, _options.method, payload); - }, - - preparePayload(store, type, snapshot) { - let serializer = store.serializerFor(type.modelName); - let payload = {}; - - serializer.serializeIntoHash(payload, type, snapshot); - - return {data: payload}; - }, - - buildIncludeURL(store, modelName, id, snapshot, requestType, query) { - let includes = this.getEmbeddedRelations(store, modelName); - let url = this.buildURL(modelName, id, snapshot, requestType, query); - - if (includes.length) { - url += `?include=${includes.join(',')}`; - } - - return url; - }, - - buildQuery(store, modelName, options) { - let deDupe = {}; - let toInclude = this.getEmbeddedRelations(store, modelName); - let query = options || {}; - - if (toInclude.length) { - // If this is a find by id, build a query object and attach the includes - if (typeof options === 'string' || typeof options === 'number') { - query = {}; - query.id = options; - query.include = toInclude.join(','); - } else if (typeof options === 'object' || isNone(options)) { - // If this is a find all (no existing query object) build one and attach - // the includes. - // If this is a find with an existing query object then merge the includes - // into the existing object. Existing properties and includes are preserved. - query = query || {}; - toInclude = toInclude.concat(query.include ? query.include.split(',') : []); - - toInclude.forEach((include) => { - deDupe[include] = true; - }); - - query.include = Object.keys(deDupe).join(','); - } - } - - return query; - }, - - getEmbeddedRelations(store, modelName) { - let model = store.modelFor(modelName); - let ret = []; - - // Iterate through the model's relationships and build a list - // of those that need to be pulled in via "include" from the API - model.eachRelationship(function (name, meta) { - if (meta.kind === 'hasMany' && - Object.prototype.hasOwnProperty.call(meta.options, 'embedded') && - meta.options.embedded === 'always') { - ret.push(name); - } - }); - - return ret; - } -}); diff --git a/core/client/app/adapters/setting.js b/core/client/app/adapters/setting.js deleted file mode 100644 index ccc266e885..0000000000 --- a/core/client/app/adapters/setting.js +++ /dev/null @@ -1,19 +0,0 @@ -import ApplicationAdapter from 'ghost/adapters/application'; - -export default ApplicationAdapter.extend({ - updateRecord(store, type, record) { - let data = {}; - let serializer = store.serializerFor(type.modelName); - - // remove the fake id that we added onto the model. - delete record.id; - - // use the SettingSerializer to transform the model back into - // an array of settings objects like the API expects - serializer.serializeIntoHash(data, type, record); - - // use the ApplicationAdapter's buildURL method but do not - // pass in an id. - return this.ajax(this.buildURL(type.modelName), 'PUT', {data}); - } -}); diff --git a/core/client/app/adapters/tag.js b/core/client/app/adapters/tag.js deleted file mode 100644 index 5dc33f7e89..0000000000 --- a/core/client/app/adapters/tag.js +++ /dev/null @@ -1,4 +0,0 @@ -import ApplicationAdapter from 'ghost/adapters/application'; -import SlugUrl from 'ghost/mixins/slug-url'; - -export default ApplicationAdapter.extend(SlugUrl); diff --git a/core/client/app/adapters/user.js b/core/client/app/adapters/user.js deleted file mode 100644 index 578f737cc6..0000000000 --- a/core/client/app/adapters/user.js +++ /dev/null @@ -1,23 +0,0 @@ -import ApplicationAdapter from 'ghost/adapters/application'; -import SlugUrl from 'ghost/mixins/slug-url'; - -export default ApplicationAdapter.extend(SlugUrl, { - find(store, type, id) { - return this.findQuery(store, type, {id, status: 'all'}); - }, - - // TODO: This is needed because the API currently expects you to know the - // status of the record before retrieving by ID. Quick fix is to always - // include status=all in the query - findRecord(store, type, id, snapshot) { - let url = this.buildIncludeURL(store, type.modelName, id, snapshot, 'findRecord'); - - url += '&status=all'; - - return this.ajax(url, 'GET'); - }, - - findAll(store, type, id) { - return this.query(store, type, {id, status: 'all'}); - } -}); diff --git a/core/client/app/app.js b/core/client/app/app.js deleted file mode 100755 index ed5c211677..0000000000 --- a/core/client/app/app.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; -import Resolver from './resolver'; -import loadInitializers from 'ember-load-initializers'; -import 'ghost/utils/link-component'; -import 'ghost/utils/text-field'; -import config from './config/environment'; - -const {Application} = Ember; - -Ember.MODEL_FACTORY_INJECTIONS = true; - -let App = Application.extend({ - Resolver, - modulePrefix: config.modulePrefix, - podModulePrefix: config.podModulePrefix -}); - -loadInitializers(App, config.modulePrefix); - -export default App; diff --git a/core/client/app/authenticators/oauth2.js b/core/client/app/authenticators/oauth2.js deleted file mode 100644 index b5380973b0..0000000000 --- a/core/client/app/authenticators/oauth2.js +++ /dev/null @@ -1,28 +0,0 @@ -import Ember from 'ember'; -import Authenticator from 'ember-simple-auth/authenticators/oauth2-password-grant'; - -const { - computed, - inject: {service} -} = Ember; - -export default Authenticator.extend({ - config: service(), - ghostPaths: service(), - - serverTokenEndpoint: computed('ghostPaths.apiRoot', function () { - return `${this.get('ghostPaths.apiRoot')}/authentication/token`; - }), - - serverTokenRevocationEndpoint: computed('ghostPaths.apiRoot', function () { - return `${this.get('ghostPaths.apiRoot')}/authentication/revoke`; - }), - - makeRequest(url, data) { - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - data.client_id = this.get('config.clientId'); - data.client_secret = this.get('config.clientSecret'); - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - return this._super(url, data); - } -}); diff --git a/core/client/app/authorizers/oauth2.js b/core/client/app/authorizers/oauth2.js deleted file mode 100644 index cdd9c12760..0000000000 --- a/core/client/app/authorizers/oauth2.js +++ /dev/null @@ -1,3 +0,0 @@ -import Oauth2Bearer from 'ember-simple-auth/authorizers/oauth2-bearer'; - -export default Oauth2Bearer; diff --git a/core/client/app/components/gh-activating-list-item.js b/core/client/app/components/gh-activating-list-item.js deleted file mode 100644 index 06fbec9b74..0000000000 --- a/core/client/app/components/gh-activating-list-item.js +++ /dev/null @@ -1,22 +0,0 @@ -import Ember from 'ember'; - -const {Component, run} = Ember; - -export default Component.extend({ - tagName: 'li', - classNameBindings: ['active'], - active: false, - linkClasses: null, - - click() { - this.$('a').blur(); - }, - - actions: { - setActive(value) { - run.schedule('afterRender', this, function () { - this.set('active', value); - }); - } - } -}); diff --git a/core/client/app/components/gh-alert.js b/core/client/app/components/gh-alert.js deleted file mode 100644 index 02419bf622..0000000000 --- a/core/client/app/components/gh-alert.js +++ /dev/null @@ -1,40 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: 'article', - classNames: ['gh-alert'], - classNameBindings: ['typeClass'], - - notifications: service(), - - typeClass: computed('message.type', function () { - let type = this.get('message.type'); - let classes = ''; - let typeMapping; - - typeMapping = { - success: 'green', - error: 'red', - warn: 'yellow', - info: 'blue' - }; - - if (typeMapping[type] !== undefined) { - classes += `gh-alert-${typeMapping[type]}`; - } - - return classes; - }), - - actions: { - closeNotification() { - this.get('notifications').closeNotification(this.get('message')); - } - } -}); diff --git a/core/client/app/components/gh-alerts.js b/core/client/app/components/gh-alerts.js deleted file mode 100644 index 179318dc21..0000000000 --- a/core/client/app/components/gh-alerts.js +++ /dev/null @@ -1,22 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service}, - observer -} = Ember; -const {alias} = computed; - -export default Component.extend({ - tagName: 'aside', - classNames: 'gh-alerts', - - notifications: service(), - - messages: alias('notifications.alerts'), - - messageCountObserver: observer('messages.[]', function () { - this.sendAction('notify', this.get('messages').length); - }) -}); diff --git a/core/client/app/components/gh-app.js b/core/client/app/components/gh-app.js deleted file mode 100644 index 0fa2af8b83..0000000000 --- a/core/client/app/components/gh-app.js +++ /dev/null @@ -1,15 +0,0 @@ -import Ember from 'ember'; - -const {Component, observer} = Ember; - -export default Component.extend({ - classNames: ['gh-app'], - - showSettingsMenu: false, - - toggleSettingsMenuBodyClass: observer('showSettingsMenu', function () { - let showSettingsMenu = this.get('showSettingsMenu'); - - Ember.$('body').toggleClass('settings-menu-expanded', showSettingsMenu); - }) -}); diff --git a/core/client/app/components/gh-blog-url.js b/core/client/app/components/gh-blog-url.js deleted file mode 100644 index 285d2cb398..0000000000 --- a/core/client/app/components/gh-blog-url.js +++ /dev/null @@ -1,12 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: '', - - config: service() -}); diff --git a/core/client/app/components/gh-cm-editor.js b/core/client/app/components/gh-cm-editor.js deleted file mode 100644 index 785ba4aa13..0000000000 --- a/core/client/app/components/gh-cm-editor.js +++ /dev/null @@ -1,47 +0,0 @@ -/* global CodeMirror */ -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - classNameBindings: ['isFocused:focused'], - - value: '', // make sure a value exists - isFocused: false, - - // options for the editor - lineNumbers: true, - indentUnit: 4, - mode: 'htmlmixed', - theme: 'xq-light', - - _editor: null, // reference to CodeMirror editor - - didInsertElement() { - this._super(...arguments); - - let options = this.getProperties('lineNumbers', 'indentUnit', 'mode', 'theme'); - let editor = new CodeMirror(this.get('element'), options); - - editor.getDoc().setValue(this.get('value')); - - // events - editor.on('focus', Ember.run.bind(this, 'set', 'isFocused', true)); - editor.on('blur', Ember.run.bind(this, 'set', 'isFocused', false)); - editor.on('change', () => { - Ember.run(this, function () { - this.set('value', editor.getDoc().getValue()); - }); - }); - - this._editor = editor; - }, - - willDestroyElement() { - this._super(...arguments); - - let editor = this._editor.getWrapperElement(); - editor.parentNode.removeChild(editor); - this._editor = null; - } -}); diff --git a/core/client/app/components/gh-content-cover.js b/core/client/app/components/gh-content-cover.js deleted file mode 100644 index 05ec3b4a1a..0000000000 --- a/core/client/app/components/gh-content-cover.js +++ /dev/null @@ -1,30 +0,0 @@ -/* - -Implements a div for covering the page content -when in a menu context that, for example, -should be closed when the user clicks elsewhere. - -Example: -``` -{{gh-content-cover onClick="closeMenus" onMouseEnter="closeAutoNav"}} -``` -**/ - -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - classNames: ['content-cover'], - - onClick: null, - onMouseEnter: null, - - click() { - this.sendAction('onClick'); - }, - - mouseEnter() { - this.sendAction('onMouseEnter'); - } -}); diff --git a/core/client/app/components/gh-content-preview-content.js b/core/client/app/components/gh-content-preview-content.js deleted file mode 100644 index 873a78ec62..0000000000 --- a/core/client/app/components/gh-content-preview-content.js +++ /dev/null @@ -1,22 +0,0 @@ -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - classNames: ['content-preview-content'], - - content: null, - - didReceiveAttrs(options) { - this._super(...arguments); - - // adjust when didReceiveAttrs gets both newAttrs and oldAttrs - if (options.newAttrs.content && this.get('content') !== options.newAttrs.content.value) { - let el = this.$(); - - if (el) { - el.closest('.content-preview').scrollTop(0); - } - } - } -}); diff --git a/core/client/app/components/gh-content-view-container.js b/core/client/app/components/gh-content-view-container.js deleted file mode 100644 index eff63ba31b..0000000000 --- a/core/client/app/components/gh-content-view-container.js +++ /dev/null @@ -1,15 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: ['gh-view', 'content-view-container'], - - mediaQueries: service(), - previewIsHidden: computed.reads('mediaQueries.maxWidth900') -}); diff --git a/core/client/app/components/gh-datetime-input.js b/core/client/app/components/gh-datetime-input.js deleted file mode 100644 index de4e34d502..0000000000 --- a/core/client/app/components/gh-datetime-input.js +++ /dev/null @@ -1,33 +0,0 @@ -import Ember from 'ember'; -import TextInputMixin from 'ghost/mixins/text-input'; -import boundOneWay from 'ghost/utils/bound-one-way'; -import {formatDate} from 'ghost/utils/date-formatting'; -import {invokeAction} from 'ember-invoke-action'; - -const {Component} = Ember; - -export default Component.extend(TextInputMixin, { - tagName: 'span', - classNames: 'input-icon icon-calendar', - - datetime: boundOneWay('value'), - inputClass: null, - inputId: null, - inputName: null, - - didReceiveAttrs() { - let datetime = this.get('datetime') || moment(); - - if (!this.get('update')) { - throw new Error(`You must provide an \`update\` action to \`{{${this.templateName}}}\`.`); - } - - this.set('datetime', formatDate(datetime)); - }, - - focusOut() { - let datetime = this.get('datetime'); - - invokeAction(this, 'update', datetime); - } -}); diff --git a/core/client/app/components/gh-dropdown-button.js b/core/client/app/components/gh-dropdown-button.js deleted file mode 100644 index a9429be827..0000000000 --- a/core/client/app/components/gh-dropdown-button.js +++ /dev/null @@ -1,24 +0,0 @@ -import Ember from 'ember'; -import DropdownMixin from 'ghost/mixins/dropdown-mixin'; - -const { - Component, - inject: {service} -} = Ember; - -export default Component.extend(DropdownMixin, { - tagName: 'button', - attributeBindings: 'role', - role: 'button', - - // matches with the dropdown this button toggles - dropdownName: null, - - dropdown: service(), - - // Notify dropdown service this dropdown should be toggled - click(event) { - this._super(event); - this.get('dropdown').toggleDropdown(this.get('dropdownName'), this); - } -}); diff --git a/core/client/app/components/gh-dropdown.js b/core/client/app/components/gh-dropdown.js deleted file mode 100644 index b7f01a589f..0000000000 --- a/core/client/app/components/gh-dropdown.js +++ /dev/null @@ -1,99 +0,0 @@ -import Ember from 'ember'; -import DropdownMixin from 'ghost/mixins/dropdown-mixin'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend(DropdownMixin, { - classNames: 'dropdown', - classNameBindings: ['fadeIn:fade-in-scale:fade-out', 'isOpen:open:closed'], - - name: null, - closeOnClick: false, - - // Helps track the user re-opening the menu while it's fading out. - closing: false, - - // Helps track whether the dropdown is open or closes, or in a transition to either - isOpen: false, - - // Managed the toggle between the fade-in and fade-out classes - fadeIn: computed('isOpen', 'closing', function () { - return this.get('isOpen') && !this.get('closing'); - }), - - dropdown: service(), - - open() { - this.set('isOpen', true); - this.set('closing', false); - this.set('button.isOpen', true); - }, - - close() { - this.set('closing', true); - - if (this.get('button')) { - this.set('button.isOpen', false); - } - - this.$().on('animationend webkitAnimationEnd oanimationend MSAnimationEnd', (event) => { - if (event.originalEvent.animationName === 'fade-out') { - Ember.run(this, function () { - if (this.get('closing')) { - this.set('isOpen', false); - this.set('closing', false); - } - }); - } - }); - }, - - // Called by the dropdown service when any dropdown button is clicked. - toggle(options) { - let isClosing = this.get('closing'); - let isOpen = this.get('isOpen'); - let name = this.get('name'); - let targetDropdownName = options.target; - let button = this.get('button'); - - if (name === targetDropdownName && (!isOpen || isClosing)) { - if (!button) { - button = options.button; - this.set('button', button); - } - this.open(); - } else if (isOpen) { - this.close(); - } - }, - - click(event) { - this._super(event); - - if (this.get('closeOnClick')) { - return this.close(); - } - }, - - didInsertElement() { - let dropdownService = this.get('dropdown'); - - this._super(...arguments); - - dropdownService.on('close', this, this.close); - dropdownService.on('toggle', this, this.toggle); - }, - - willDestroyElement() { - let dropdownService = this.get('dropdown'); - - this._super(...arguments); - - dropdownService.off('close', this, this.close); - dropdownService.off('toggle', this, this.toggle); - } -}); diff --git a/core/client/app/components/gh-ed-editor.js b/core/client/app/components/gh-ed-editor.js deleted file mode 100644 index a927458bc1..0000000000 --- a/core/client/app/components/gh-ed-editor.js +++ /dev/null @@ -1,52 +0,0 @@ -import Ember from 'ember'; -import EditorAPI from 'ghost/mixins/ed-editor-api'; -import EditorShortcuts from 'ghost/mixins/ed-editor-shortcuts'; -import EditorScroll from 'ghost/mixins/ed-editor-scroll'; -import {invokeAction} from 'ember-invoke-action'; - -const {TextArea, run} = Ember; - -export default TextArea.extend(EditorAPI, EditorShortcuts, EditorScroll, { - focus: false, - - /** - * Tell the controller about focusIn events, will trigger an autosave on a new document - */ - focusIn() { - this.sendAction('onFocusIn'); - }, - - /** - * Sets the focus of the textarea if needed - */ - setFocus() { - if (this.get('focus')) { - this.$().val(this.$().val()).focus(); - } - }, - - /** - * Sets up properties at render time - */ - didInsertElement() { - this._super(...arguments); - - this.setFocus(); - - invokeAction(this, 'setEditor', this); - - run.scheduleOnce('afterRender', this, this.afterRenderEvent); - }, - - afterRenderEvent() { - if (this.get('focus') && this.get('focusCursorAtEnd')) { - this.setSelection('end'); - } - }, - - actions: { - toggleCopyHTMLModal(generatedHTML) { - invokeAction(this, 'toggleCopyHTMLModal', generatedHTML); - } - } -}); diff --git a/core/client/app/components/gh-ed-preview.js b/core/client/app/components/gh-ed-preview.js deleted file mode 100644 index f28d47d198..0000000000 --- a/core/client/app/components/gh-ed-preview.js +++ /dev/null @@ -1,110 +0,0 @@ -import Ember from 'ember'; -import {formatMarkdown} from 'ghost/helpers/gh-format-markdown'; - -const { - Component, - run, - uuid -} = Ember; - -export default Component.extend({ - _scrollWrapper: null, - - previewHTML: '', - - init() { - this._super(...arguments); - this.set('imageUploadComponents', Ember.A([])); - this.buildPreviewHTML(); - }, - - didInsertElement() { - this._super(...arguments); - this._scrollWrapper = this.$().closest('.entry-preview-content'); - this.adjustScrollPosition(this.get('scrollPosition')); - }, - - didReceiveAttrs(attrs) { - this._super(...arguments); - - if (!attrs.oldAttrs) { - return; - } - - if (attrs.newAttrs.scrollPosition && attrs.newAttrs.scrollPosition.value !== attrs.oldAttrs.scrollPosition.value) { - this.adjustScrollPosition(attrs.newAttrs.scrollPosition.value); - } - - if (attrs.newAttrs.markdown.value !== attrs.oldAttrs.markdown.value) { - run.throttle(this, this.buildPreviewHTML, 30, false); - } - }, - - adjustScrollPosition(scrollPosition) { - let scrollWrapper = this._scrollWrapper; - - if (scrollWrapper) { - scrollWrapper.scrollTop(scrollPosition); - } - }, - - buildPreviewHTML() { - let markdown = this.get('markdown'); - let html = formatMarkdown([markdown]).string; - let template = document.createElement('template'); - template.innerHTML = html; - let fragment = template.content; - let dropzones = fragment.querySelectorAll('.js-drop-zone'); - let components = this.get('imageUploadComponents'); - - if (dropzones.length !== components.length) { - components = Ember.A([]); - this.set('imageUploadComponents', components); - } - - [...dropzones].forEach((oldEl, i) => { - let el = oldEl.cloneNode(true); - let component = components[i]; - let uploadTarget = el.querySelector('.js-upload-target'); - let id = uuid(); - let destinationElementId = `image-uploader-${id}`; - let src; - - if (uploadTarget) { - src = uploadTarget.getAttribute('src'); - } - - if (component) { - component.set('destinationElementId', destinationElementId); - component.set('src', src); - } else { - let imageUpload = Ember.Object.create({ - destinationElementId, - id, - src, - index: i - }); - - this.get('imageUploadComponents').pushObject(imageUpload); - } - - el.id = destinationElementId; - el.innerHTML = ''; - el.classList.remove('image-uploader'); - - oldEl.parentNode.replaceChild(el, oldEl); - }); - - this.set('previewHTML', fragment); - }, - - actions: { - updateImageSrc(index, url) { - this.attrs.updateImageSrc(index, url); - }, - - updateHeight() { - this.attrs.updateHeight(this.$().height()); - } - } -}); diff --git a/core/client/app/components/gh-editor-save-button.js b/core/client/app/components/gh-editor-save-button.js deleted file mode 100644 index 9b04b131eb..0000000000 --- a/core/client/app/components/gh-editor-save-button.js +++ /dev/null @@ -1,50 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: ['splitbtn', 'js-publish-splitbutton'], - classNameBindings: ['isNew:unsaved'], - - isNew: null, - isPublished: null, - willPublish: null, - postOrPage: null, - submitting: false, - - // Tracks whether we're going to change the state of the post on save - isDangerous: computed('isPublished', 'willPublish', function () { - return this.get('isPublished') !== this.get('willPublish'); - }), - - publishText: computed('isPublished', 'postOrPage', function () { - return this.get('isPublished') ? `Update ${this.get('postOrPage')}` : 'Publish Now'; - }), - - draftText: computed('isPublished', function () { - return this.get('isPublished') ? 'Unpublish' : 'Save Draft'; - }), - - deleteText: computed('postOrPage', function () { - return `Delete ${this.get('postOrPage')}`; - }), - - saveText: computed('willPublish', 'publishText', 'draftText', function () { - return this.get('willPublish') ? this.get('publishText') : this.get('draftText'); - }), - - actions: { - save() { - this.sendAction('save'); - }, - - setSaveType(saveType) { - this.sendAction('setSaveType', saveType); - }, - - delete() { - this.sendAction('delete'); - } - } -}); diff --git a/core/client/app/components/gh-editor.js b/core/client/app/components/gh-editor.js deleted file mode 100644 index 5b0fec6988..0000000000 --- a/core/client/app/components/gh-editor.js +++ /dev/null @@ -1,123 +0,0 @@ -import Ember from 'ember'; -import ShortcutsMixin from 'ghost/mixins/shortcuts'; -import imageManager from 'ghost/utils/ed-image-manager'; -import editorShortcuts from 'ghost/utils/editor-shortcuts'; -import {invokeAction} from 'ember-invoke-action'; - -const {Component, computed, run} = Ember; -const {equal} = computed; - -export default Component.extend(ShortcutsMixin, { - tagName: 'section', - classNames: ['view-container', 'view-editor'], - - activeTab: 'markdown', - editor: null, - editorDisabled: undefined, - editorScrollInfo: null, // updated when gh-ed-editor component scrolls - height: null, // updated when markdown is rendered - shouldFocusEditor: false, - showCopyHTMLModal: false, - copyHTMLModalContent: null, - - shortcuts: editorShortcuts, - - markdownActive: equal('activeTab', 'markdown'), - previewActive: equal('activeTab', 'preview'), - - // HTML Preview listens to scrollPosition and updates its scrollTop value - // This property receives scrollInfo from the textEditor, and height from the preview pane, and will update the - // scrollPosition value such that when either scrolling or typing-at-the-end of the text editor the preview pane - // stays in sync - scrollPosition: computed('editorScrollInfo', 'height', function () { - let scrollInfo = this.get('editorScrollInfo'); - let {$previewContent, $previewViewPort} = this; - - if (!scrollInfo || !$previewContent || !$previewViewPort) { - return 0; - } - - let previewHeight = $previewContent.height() - $previewViewPort.height(); - let previewPosition, ratio; - - ratio = previewHeight / scrollInfo.diff; - previewPosition = scrollInfo.top * ratio; - - return previewPosition; - }), - - didInsertElement() { - this._super(...arguments); - this.registerShortcuts(); - run.scheduleOnce('afterRender', this, this._cacheElements); - }, - - willDestroyElement() { - invokeAction(this, 'onTeardown'); - - this.removeShortcuts(); - }, - - _cacheElements() { - // cache these elements for use in other methods - this.$previewViewPort = this.$('.js-entry-preview-content'); - this.$previewContent = this.$('.js-rendered-markdown'); - }, - - actions: { - selectTab(tab) { - this.set('activeTab', tab); - }, - - updateScrollInfo(scrollInfo) { - this.set('editorScrollInfo', scrollInfo); - }, - - updateHeight(height) { - this.set('height', height); - }, - - // set from a `sendAction` on the gh-ed-editor component, - // so that we get a reference for handling uploads. - setEditor(editor) { - this.set('editor', editor); - }, - - disableEditor() { - this.set('editorDisabled', true); - }, - - enableEditor() { - this.set('editorDisabled', undefined); - }, - - // The actual functionality is implemented in utils/ed-editor-shortcuts - editorShortcut(options) { - if (this.editor.$().is(':focus')) { - this.editor.shortcut(options.type); - } - }, - - // Match the uploaded file to a line in the editor, and update that line with a path reference - // ensuring that everything ends up in the correct place and format. - handleImgUpload(imageIndex, newSrc) { - let editor = this.get('editor'); - let editorValue = editor.getValue(); - let replacement = imageManager.getSrcRange(editorValue, imageIndex); - let cursorPosition; - - if (replacement) { - cursorPosition = replacement.start + newSrc.length + 1; - if (replacement.needsParens) { - newSrc = `(${newSrc})`; - } - editor.replaceSelection(newSrc, replacement.start, replacement.end, cursorPosition); - } - }, - - toggleCopyHTMLModal(generatedHTML) { - this.set('copyHTMLModalContent', generatedHTML); - this.toggleProperty('showCopyHTMLModal'); - } - } -}); diff --git a/core/client/app/components/gh-error-message.js b/core/client/app/components/gh-error-message.js deleted file mode 100644 index 281fe3c226..0000000000 --- a/core/client/app/components/gh-error-message.js +++ /dev/null @@ -1,36 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed, isEmpty} = Ember; - -/** - * Renders one random error message when passed a DS.Errors object - * and a property name. The message will be one of the ones associated with - * that specific property. If there are no errors associated with the property, - * nothing will be rendered. - * @param {DS.Errors} errors The DS.Errors object - * @param {string} property The property name - */ -export default Component.extend({ - tagName: 'p', - classNames: ['response'], - - errors: null, - property: '', - - isVisible: computed.notEmpty('errors'), - - message: computed('errors.[]', 'property', function () { - let property = this.get('property'); - let errors = this.get('errors'); - let messages = []; - let index; - - if (!isEmpty(errors) && errors.get(property)) { - errors.get(property).forEach((error) => { - messages.push(error); - }); - index = Math.floor(Math.random() * messages.length); - return messages[index].message; - } - }) -}); diff --git a/core/client/app/components/gh-feature-flag.js b/core/client/app/components/gh-feature-flag.js deleted file mode 100644 index e744e0f943..0000000000 --- a/core/client/app/components/gh-feature-flag.js +++ /dev/null @@ -1,45 +0,0 @@ -import Ember from 'ember'; - -const { - computed, - inject: {service}, - Component -} = Ember; - -const FeatureFlagComponent = Component.extend({ - tagName: 'label', - classNames: 'checkbox', - attributeBindings: ['for'], - _flagValue: null, - - feature: service(), - - init() { - this._super(...arguments); - - this.set('_flagValue', this.get(`feature.${this.get('flag')}`)); - }, - - value: computed('_flagValue', { - get() { - return this.get('_flagValue'); - }, - set(key, value) { - return this.set(`feature.${this.get('flag')}`, value); - } - }), - - for: computed('flag', function () { - return `labs-${this.get('flag')}`; - }), - - name: computed('flag', function () { - return `labs[${this.get('flag')}]`; - }) -}); - -FeatureFlagComponent.reopenClass({ - positionalParams: ['flag'] -}); - -export default FeatureFlagComponent; diff --git a/core/client/app/components/gh-file-upload.js b/core/client/app/components/gh-file-upload.js deleted file mode 100644 index e65b8817fd..0000000000 --- a/core/client/app/components/gh-file-upload.js +++ /dev/null @@ -1,37 +0,0 @@ -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - _file: null, - - uploadButtonText: 'Text', - uploadButtonDisabled: true, - - onUpload: null, - onAdd: null, - - shouldResetForm: true, - - change(event) { - this.set('uploadButtonDisabled', false); - this.sendAction('onAdd'); - this._file = event.target.files[0]; - }, - - actions: { - upload() { - if (!this.get('uploadButtonDisabled') && this._file) { - this.sendAction('onUpload', this._file); - } - - // Prevent double post by disabling the button. - this.set('uploadButtonDisabled', true); - - // Reset form - if (this.get('shouldResetForm')) { - this.$().closest('form')[0].reset(); - } - } - } -}); diff --git a/core/client/app/components/gh-file-uploader.js b/core/client/app/components/gh-file-uploader.js deleted file mode 100644 index f17feb3e5b..0000000000 --- a/core/client/app/components/gh-file-uploader.js +++ /dev/null @@ -1,161 +0,0 @@ -import Ember from 'ember'; -import { invoke, invokeAction } from 'ember-invoke-action'; -import { - RequestEntityTooLargeError, - UnsupportedMediaTypeError -} from 'ghost/services/ajax'; - -const { - Component, - computed, - inject: {service}, - isBlank, - run -} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: ['gh-image-uploader'], - classNameBindings: ['dragClass'], - - labelText: 'Select or drag-and-drop a file', - url: null, - paramName: 'file', - - file: null, - response: null, - - dragClass: null, - failureMessage: null, - uploadPercentage: 0, - - ajax: service(), - - formData: computed('file', function () { - let paramName = this.get('paramName'); - let file = this.get('file'); - let formData = new FormData(); - - formData.append(paramName, file); - - return formData; - }), - - progressStyle: computed('uploadPercentage', function () { - let percentage = this.get('uploadPercentage'); - let width = ''; - - if (percentage > 0) { - width = `${percentage}%`; - } else { - width = '0'; - } - - return Ember.String.htmlSafe(`width: ${width}`); - }), - - dragOver(event) { - if (!event.dataTransfer) { - return; - } - - // this is needed to work around inconsistencies with dropping files - // from Chrome's downloads bar - let eA = event.dataTransfer.effectAllowed; - event.dataTransfer.dropEffect = (eA === 'move' || eA === 'linkMove') ? 'move' : 'copy'; - - event.stopPropagation(); - event.preventDefault(); - - this.set('dragClass', '--drag-over'); - }, - - dragLeave(event) { - event.preventDefault(); - this.set('dragClass', null); - }, - - drop(event) { - event.preventDefault(); - this.set('dragClass', null); - if (event.dataTransfer.files) { - invoke(this, 'fileSelected', event.dataTransfer.files); - } - }, - - generateRequest() { - let ajax = this.get('ajax'); - let formData = this.get('formData'); - let url = this.get('url'); - - invokeAction(this, 'uploadStarted'); - - ajax.post(url, { - data: formData, - processData: false, - contentType: false, - dataType: 'text', - xhr: () => { - let xhr = new window.XMLHttpRequest(); - - xhr.upload.addEventListener('progress', (event) => { - this._uploadProgress(event); - }, false); - - return xhr; - } - }).then((response) => { - this._uploadSuccess(JSON.parse(response)); - }).catch((error) => { - this._uploadFailed(error); - }).finally(() => { - invokeAction(this, 'uploadFinished'); - }); - }, - - _uploadProgress(event) { - if (event.lengthComputable) { - run(() => { - let percentage = Math.round((event.loaded / event.total) * 100); - this.set('uploadPercentage', percentage); - }); - } - }, - - _uploadSuccess(response) { - invokeAction(this, 'uploadSuccess', response); - invoke(this, 'reset'); - }, - - _uploadFailed(error) { - let message; - - if (error instanceof UnsupportedMediaTypeError) { - message = 'The file type you uploaded is not supported.'; - } else if (error instanceof RequestEntityTooLargeError) { - message = 'The file you uploaded was larger than the maximum file size your server allows.'; - } else if (error.errors && !isBlank(error.errors[0].message)) { - message = error.errors[0].message; - } else { - message = 'Something went wrong :('; - } - - this.set('failureMessage', message); - invokeAction(this, 'uploadFailed', error); - }, - - actions: { - fileSelected(fileList) { - this.set('file', fileList[0]); - run.schedule('actions', this, function () { - this.generateRequest(); - }); - }, - - reset() { - this.set('file', null); - this.set('uploadPercentage', 0); - this.set('failureMessage', null); - } - } -}); diff --git a/core/client/app/components/gh-form-group.js b/core/client/app/components/gh-form-group.js deleted file mode 100644 index 5445314cc5..0000000000 --- a/core/client/app/components/gh-form-group.js +++ /dev/null @@ -1,5 +0,0 @@ -import ValidationStatusContainer from 'ghost/components/gh-validation-status-container'; - -export default ValidationStatusContainer.extend({ - classNames: 'form-group' -}); diff --git a/core/client/app/components/gh-fullscreen-modal.js b/core/client/app/components/gh-fullscreen-modal.js deleted file mode 100644 index 21740a1538..0000000000 --- a/core/client/app/components/gh-fullscreen-modal.js +++ /dev/null @@ -1,85 +0,0 @@ -import Ember from 'ember'; -import LiquidTether from 'liquid-tether/components/liquid-tether'; -import {invokeAction} from 'ember-invoke-action'; - -const { - RSVP: {Promise}, - inject: {service}, - isBlank, - on, - run -} = Ember; -const emberA = Ember.A; - -const FullScreenModalComponent = LiquidTether.extend({ - to: 'fullscreen-modal', - target: 'document.body', - targetModifier: 'visible', - targetAttachment: 'top center', - attachment: 'top center', - tetherClass: 'fullscreen-modal', - overlayClass: 'fullscreen-modal-background', - modalPath: 'unknown', - - dropdown: service(), - - init() { - this._super(...arguments); - this.modalPath = `modals/${this.get('modal')}`; - }, - - setTetherClass: on('init', function () { - let tetherClass = this.get('tetherClass'); - let modifiers = (this.get('modifier') || '').split(' '); - let tetherClasses = emberA([tetherClass]); - - modifiers.forEach((modifier) => { - if (!isBlank(modifier)) { - let className = `${tetherClass}-${modifier}`; - tetherClasses.push(className); - } - }); - - this.set('tetherClass', tetherClasses.join(' ')); - }), - - closeDropdowns: on('didInsertElement', function () { - run.schedule('afterRender', this, function () { - this.get('dropdown').closeDropdowns(); - }); - }), - - actions: { - close() { - // Because we return the promise from invokeAction, we have - // to check if "close" exists first - if (this.get('close')) { - return invokeAction(this, 'close'); - } - - return new Promise((resolve) => { - resolve(); - }); - }, - - confirm() { - if (this.get('confirm')) { - return invokeAction(this, 'confirm'); - } - - return new Promise((resolve) => { - resolve(); - }); - }, - - clickOverlay() { - this.send('close'); - } - } -}); - -FullScreenModalComponent.reopenClass({ - positionalParams: ['modal'] -}); - -export default FullScreenModalComponent; diff --git a/core/client/app/components/gh-image-uploader-with-preview.js b/core/client/app/components/gh-image-uploader-with-preview.js deleted file mode 100644 index aba65c3261..0000000000 --- a/core/client/app/components/gh-image-uploader-with-preview.js +++ /dev/null @@ -1,39 +0,0 @@ -import Ember from 'ember'; - -const { - Component -} = Ember; - -export default Component.extend({ - actions: { - update() { - if (typeof this.attrs.update === 'function') { - this.attrs.update(...arguments); - } - }, - - onInput() { - if (typeof this.attrs.onInput === 'function') { - this.attrs.onInput(...arguments); - } - }, - - uploadStarted() { - if (typeof this.attrs.uploadStarted === 'function') { - this.attrs.uploadStarted(...arguments); - } - }, - - uploadFinished() { - if (typeof this.attrs.uploadFinished === 'function') { - this.attrs.uploadFinished(...arguments); - } - }, - - formChanged() { - if (typeof this.attrs.formChanged === 'function') { - this.attrs.formChanged(...arguments); - } - } - } -}); diff --git a/core/client/app/components/gh-image-uploader.js b/core/client/app/components/gh-image-uploader.js deleted file mode 100644 index 1141b38fc4..0000000000 --- a/core/client/app/components/gh-image-uploader.js +++ /dev/null @@ -1,226 +0,0 @@ -import Ember from 'ember'; -import ghostPaths from 'ghost/utils/ghost-paths'; -import {RequestEntityTooLargeError, UnsupportedMediaTypeError} from 'ghost/services/ajax'; - -const { - Component, - computed, - inject: {service}, - isBlank, - run -} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: ['gh-image-uploader'], - classNameBindings: ['dragClass'], - - image: null, - text: 'Upload an image', - saveButton: true, - - dragClass: null, - failureMessage: null, - file: null, - formType: 'upload', - url: null, - uploadPercentage: 0, - - ajax: service(), - config: service(), - - // TODO: this wouldn't be necessary if the server could accept direct - // file uploads - formData: computed('file', function () { - let file = this.get('file'); - let formData = new FormData(); - - formData.append('uploadimage', file); - - return formData; - }), - - progressStyle: computed('uploadPercentage', function () { - let percentage = this.get('uploadPercentage'); - let width = ''; - - if (percentage > 0) { - width = `${percentage}%`; - } else { - width = '0'; - } - - return Ember.String.htmlSafe(`width: ${width}`); - }), - - canShowUploadForm: computed('config.fileStorage', function () { - return this.get('config.fileStorage') !== false; - }), - - showUploadForm: computed('formType', function () { - let canShowUploadForm = this.get('canShowUploadForm'); - let formType = this.get('formType'); - - return formType === 'upload' && canShowUploadForm; - }), - - didReceiveAttrs() { - let image = this.get('image'); - this.set('url', image); - }, - - dragOver(event) { - let showUploadForm = this.get('showUploadForm'); - - if (!event.dataTransfer) { - return; - } - - // this is needed to work around inconsistencies with dropping files - // from Chrome's downloads bar - let eA = event.dataTransfer.effectAllowed; - event.dataTransfer.dropEffect = (eA === 'move' || eA === 'linkMove') ? 'move' : 'copy'; - - event.stopPropagation(); - event.preventDefault(); - - if (showUploadForm) { - this.set('dragClass', '--drag-over'); - } - }, - - dragLeave(event) { - let showUploadForm = this.get('showUploadForm'); - - event.preventDefault(); - - if (showUploadForm) { - this.set('dragClass', null); - } - }, - - drop(event) { - let showUploadForm = this.get('showUploadForm'); - - event.preventDefault(); - - this.set('dragClass', null); - - if (showUploadForm) { - if (event.dataTransfer.files) { - this.send('fileSelected', event.dataTransfer.files); - } - } - }, - - uploadStarted() { - if (typeof this.attrs.uploadStarted === 'function') { - this.attrs.uploadStarted(); - } - }, - - uploadProgress(event) { - if (event.lengthComputable) { - run(() => { - let percentage = Math.round((event.loaded / event.total) * 100); - this.set('uploadPercentage', percentage); - }); - } - }, - - uploadFinished() { - if (typeof this.attrs.uploadFinished === 'function') { - this.attrs.uploadFinished(); - } - }, - - uploadSuccess(response) { - this.set('url', response); - this.send('saveUrl'); - this.send('reset'); - }, - - uploadFailed(error) { - let message; - - if (error instanceof UnsupportedMediaTypeError) { - message = 'The image type you uploaded is not supported. Please use .PNG, .JPG, .GIF, .SVG.'; - } else if (error instanceof RequestEntityTooLargeError) { - message = 'The image you uploaded was larger than the maximum file size your server allows.'; - } else if (error.errors && !isBlank(error.errors[0].message)) { - message = error.errors[0].message; - } else { - message = 'Something went wrong :('; - } - - this.set('failureMessage', message); - }, - - generateRequest() { - let ajax = this.get('ajax'); - let formData = this.get('formData'); - let url = `${ghostPaths().apiRoot}/uploads/`; - - this.uploadStarted(); - - ajax.post(url, { - data: formData, - processData: false, - contentType: false, - dataType: 'text', - xhr: () => { - let xhr = new window.XMLHttpRequest(); - - xhr.upload.addEventListener('progress', (event) => { - this.uploadProgress(event); - }, false); - - return xhr; - } - }).then((response) => { - let url = JSON.parse(response); - this.uploadSuccess(url); - }).catch((error) => { - this.uploadFailed(error); - }).finally(() => { - this.uploadFinished(); - }); - }, - - actions: { - fileSelected(fileList) { - this.set('file', fileList[0]); - run.schedule('actions', this, function () { - this.generateRequest(); - }); - }, - - onInput(url) { - this.set('url', url); - - if (typeof this.attrs.onInput === 'function') { - this.attrs.onInput(url); - } - }, - - reset() { - this.set('file', null); - this.set('uploadPercentage', 0); - }, - - switchForm(formType) { - this.set('formType', formType); - - if (typeof this.attrs.formChanged === 'function') { - run.scheduleOnce('afterRender', this, function () { - this.attrs.formChanged(formType); - }); - } - }, - - saveUrl() { - let url = this.get('url'); - this.attrs.update(url); - } - } -}); diff --git a/core/client/app/components/gh-infinite-scroll.js b/core/client/app/components/gh-infinite-scroll.js deleted file mode 100644 index 56746ba1f3..0000000000 --- a/core/client/app/components/gh-infinite-scroll.js +++ /dev/null @@ -1,12 +0,0 @@ -import Ember from 'ember'; -import InfiniteScrollMixin from 'ghost/mixins/infinite-scroll'; - -const {Component} = Ember; - -export default Component.extend(InfiniteScrollMixin, { - actions: { - checkScroll() { - this._checkScroll(); - } - } -}); diff --git a/core/client/app/components/gh-input.js b/core/client/app/components/gh-input.js deleted file mode 100644 index 43d8ef131e..0000000000 --- a/core/client/app/components/gh-input.js +++ /dev/null @@ -1,8 +0,0 @@ -import Ember from 'ember'; -import TextInputMixin from 'ghost/mixins/text-input'; - -const {TextField} = Ember; - -export default TextField.extend(TextInputMixin, { - classNames: 'gh-input' -}); diff --git a/core/client/app/components/gh-light-table.js b/core/client/app/components/gh-light-table.js deleted file mode 100644 index 010f311ae2..0000000000 --- a/core/client/app/components/gh-light-table.js +++ /dev/null @@ -1,27 +0,0 @@ -import Ember from 'ember'; -import LightTable from 'ember-light-table/components/light-table'; - -const {$, run} = Ember; - -export default LightTable.extend({ - - // HACK: infinite pagination was not triggering when scrolling very fast - // as the throttle triggers before scrolling into the buffer area but - // the scroll finishes before the throttle timeout. Adding a debounce that - // does the same thing means that we are guaranteed a final trigger when - // scrolling stops - // - // An issue has been opened upstream, this can be removed if it gets fixed - // https://github.com/offirgolan/ember-light-table/issues/15 - - _setupScrollEvents() { - $(this.get('touchMoveContainer')).on('touchmove.light-table', run.bind(this, this._scrollHandler, '_touchmoveTimer')); - $(this.get('scrollContainer')).on('scroll.light-table', run.bind(this, this._scrollHandler, '_scrollTimer')); - $(this.get('scrollContainer')).on('scroll.light-table', run.bind(this, this._scrollHandler, '_scrollDebounce')); - }, - - _scrollHandler(timer) { - this.set(timer, run.debounce(this, this._onScroll, 100)); - this.set(timer, run.throttle(this, this._onScroll, 100)); - } -}); diff --git a/core/client/app/components/gh-main.js b/core/client/app/components/gh-main.js deleted file mode 100644 index e4bd8916df..0000000000 --- a/core/client/app/components/gh-main.js +++ /dev/null @@ -1,13 +0,0 @@ -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - tagName: 'main', - classNames: ['gh-main'], - ariaRole: 'main', - - mouseEnter() { - this.sendAction('onMouseEnter'); - } -}); diff --git a/core/client/app/components/gh-menu-toggle.js b/core/client/app/components/gh-menu-toggle.js deleted file mode 100644 index 8e5ba3ad9e..0000000000 --- a/core/client/app/components/gh-menu-toggle.js +++ /dev/null @@ -1,42 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -/* - This cute little component has two jobs. - - On desktop, it toggles autoNav behaviour. It tracks - that state via the maximise property, and uses the - state to render the appropriate icon. - - On mobile, it renders a closing icon, and clicking it - closes the mobile menu -*/ -export default Component.extend({ - classNames: ['gh-menu-toggle'], - - mediaQueries: service(), - isMobile: computed.reads('mediaQueries.isMobile'), - maximise: false, - - iconClass: computed('maximise', 'isMobile', function () { - if (this.get('maximise') && !this.get('isMobile')) { - return 'icon-maximise'; - } else { - return 'icon-minimise'; - } - }), - - click() { - if (this.get('isMobile')) { - this.sendAction('mobileAction'); - } else { - this.toggleProperty('maximise'); - this.sendAction('desktopAction'); - } - } -}); diff --git a/core/client/app/components/gh-nav-menu.js b/core/client/app/components/gh-nav-menu.js deleted file mode 100644 index b3098d167f..0000000000 --- a/core/client/app/components/gh-nav-menu.js +++ /dev/null @@ -1,48 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - inject: {service}, - computed -} = Ember; - -export default Component.extend({ - tagName: 'nav', - classNames: ['gh-nav'], - classNameBindings: ['open'], - - open: false, - - navMenuIcon: computed('ghostPaths.subdir', function () { - let url = `${this.get('ghostPaths.subdir')}/ghost/img/ghosticon.jpg`; - - return Ember.String.htmlSafe(`background-image: url(${url})`); - }), - - config: service(), - session: service(), - ghostPaths: service(), - feature: service(), - - mouseEnter() { - this.sendAction('onMouseEnter'); - }, - - actions: { - toggleAutoNav() { - this.sendAction('toggleMaximise'); - }, - - showMarkdownHelp() { - this.sendAction('showMarkdownHelp'); - }, - - closeMobileMenu() { - this.sendAction('closeMobileMenu'); - }, - - openAutoNav() { - this.sendAction('openAutoNav'); - } - } -}); diff --git a/core/client/app/components/gh-navigation.js b/core/client/app/components/gh-navigation.js deleted file mode 100644 index 53fedb3d1f..0000000000 --- a/core/client/app/components/gh-navigation.js +++ /dev/null @@ -1,39 +0,0 @@ -import Ember from 'ember'; - -const {Component, run} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: 'gh-view', - - didInsertElement() { - let navContainer = this.$('.js-gh-blognav'); - let navElements = '.gh-blognav-item:not(.gh-blognav-item:last-child)'; - // needed because jqueryui sortable doesn't trigger babel's autoscoping - let _this = this; - - this._super(...arguments); - - navContainer.sortable({ - handle: '.gh-blognav-grab', - items: navElements, - - start(event, ui) { - run(() => { - ui.item.data('start-index', ui.item.index()); - }); - }, - - update(event, ui) { - run(() => { - _this.sendAction('moveItem', ui.item.data('start-index'), ui.item.index()); - }); - } - }); - }, - - willDestroyElement() { - this._super(...arguments); - this.$('.ui-sortable').sortable('destroy'); - } -}); diff --git a/core/client/app/components/gh-navitem-url-input.js b/core/client/app/components/gh-navitem-url-input.js deleted file mode 100644 index df1928c6e0..0000000000 --- a/core/client/app/components/gh-navitem-url-input.js +++ /dev/null @@ -1,145 +0,0 @@ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -const {TextField, computed, run} = Ember; - -let joinUrlParts = function (url, path) { - if (path[0] !== '/' && url.slice(-1) !== '/') { - path = `/${path}`; - } else if (path[0] === '/' && url.slice(-1) === '/') { - path = path.slice(1); - } - - return url + path; -}; - -let isRelative = function (url) { - // "protocol://", "//example.com", "scheme:", "#anchor", & invalid paths - // should all be treated as absolute - return !url.match(/\s/) && !validator.isURL(url) && !url.match(/^(\/\/|#|[a-zA-Z0-9\-]+:)/); -}; - -export default TextField.extend({ - classNames: 'gh-input', - - isBaseUrl: computed('baseUrl', 'value', function () { - return this.get('baseUrl') === this.get('value'); - }), - - didReceiveAttrs() { - this._super(...arguments); - - let baseUrl = this.get('baseUrl'); - let url = this.get('url'); - - // if we have a relative url, create the absolute url to be displayed in the input - if (isRelative(url)) { - url = joinUrlParts(baseUrl, url); - } - - this.set('value', url); - }, - - focusIn(event) { - this.set('hasFocus', true); - - if (this.get('isBaseUrl')) { - // position the cursor at the end of the input - run.next(function (el) { - let {length} = el.value; - - el.setSelectionRange(length, length); - }, event.target); - } - }, - - keyDown(event) { - // delete the "placeholder" value all at once - if (this.get('isBaseUrl') && (event.keyCode === 8 || event.keyCode === 46)) { - this.set('value', ''); - - event.preventDefault(); - } - - // CMD-S - if (event.keyCode === 83 && event.metaKey) { - this.notifyUrlChanged(); - } - }, - - keyPress(event) { - invokeAction(this, 'clearErrors'); - - // enter key - if (event.keyCode === 13) { - this.notifyUrlChanged(); - } - - return true; - }, - - focusOut() { - this.set('hasFocus', false); - - this.notifyUrlChanged(); - }, - - notifyUrlChanged() { - let url = this.get('value').trim(); - let urlParts = document.createElement('a'); - let baseUrl = this.get('baseUrl'); - let baseUrlParts = document.createElement('a'); - - // ensure value property is trimmed - this.set('value', url); - - // leverage the browser's native URI parsing - urlParts.href = url; - baseUrlParts.href = baseUrl; - - // if we have an email address, add the mailto: - if (validator.isEmail(url)) { - url = `mailto:${url}`; - this.set('value', url); - } - - // if we have a relative url, create the absolute url to be displayed in the input - if (isRelative(url)) { - url = joinUrlParts(baseUrl, url); - this.set('value', url); - } - - // get our baseUrl relativity checks in order - let isOnSameHost = urlParts.host === baseUrlParts.host; - let isAnchorLink = url.match(/^#/); - let isRelativeToBasePath = urlParts.pathname.indexOf(baseUrlParts.pathname) === 0; - - // if our pathname is only missing a trailing / mark it as relative - if (`${urlParts.pathname}/` === baseUrlParts.pathname) { - isRelativeToBasePath = true; - } - - // if relative to baseUrl, remove the base url before sending to action - if (!isAnchorLink && isOnSameHost && isRelativeToBasePath) { - url = url.replace(/^[a-zA-Z0-9\-]+:/, ''); - url = url.replace(/^\/\//, ''); - url = url.replace(baseUrlParts.host, ''); - url = url.replace(baseUrlParts.pathname, ''); - - // handle case where url path is same as baseUrl path but missing trailing slash - if (urlParts.pathname.slice(-1) !== '/') { - url = url.replace(baseUrlParts.pathname.slice(0, -1), ''); - } - - if (!url.match(/^\//)) { - url = `/${url}`; - } - - if (!url.match(/\/$/) && !url.match(/[\.#\?]/)) { - url = `${url}/`; - } - } - - this.sendAction('change', url); - } -}); diff --git a/core/client/app/components/gh-navitem.js b/core/client/app/components/gh-navitem.js deleted file mode 100644 index 341ea1a9da..0000000000 --- a/core/client/app/components/gh-navitem.js +++ /dev/null @@ -1,55 +0,0 @@ -import Ember from 'ember'; -import ValidationState from 'ghost/mixins/validation-state'; -import SortableItem from 'ember-sortable/mixins/sortable-item'; - -const {Component, computed, run} = Ember; -const {alias, readOnly} = computed; - -export default Component.extend(ValidationState, SortableItem, { - classNames: 'gh-blognav-item', - classNameBindings: ['errorClass', 'navItem.isNew::gh-blognav-item--sortable'], - - new: false, - handle: '.gh-blognav-grab', - - model: alias('navItem'), - errors: readOnly('navItem.errors'), - - errorClass: computed('hasError', function () { - if (this.get('hasError')) { - return 'gh-blognav-item--error'; - } - }), - - keyPress(event) { - // enter key - if (event.keyCode === 13 && this.get('navItem.isNew')) { - event.preventDefault(); - run.scheduleOnce('actions', this, function () { - this.send('addItem'); - }); - } - }, - - actions: { - addItem() { - this.sendAction('addItem'); - }, - - deleteItem(item) { - this.sendAction('deleteItem', item); - }, - - updateUrl(value) { - this.sendAction('updateUrl', value, this.get('navItem')); - }, - - clearLabelErrors() { - this.get('navItem.errors').remove('label'); - }, - - clearUrlErrors() { - this.get('navItem.errors').remove('url'); - } - } -}); diff --git a/core/client/app/components/gh-notification.js b/core/client/app/components/gh-notification.js deleted file mode 100644 index 27787e8ad9..0000000000 --- a/core/client/app/components/gh-notification.js +++ /dev/null @@ -1,56 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: 'article', - classNames: ['gh-notification', 'gh-notification-passive'], - classNameBindings: ['typeClass'], - - message: null, - - notifications: service(), - - typeClass: computed('message.type', function () { - let type = this.get('message.type'); - let classes = ''; - let typeMapping; - - typeMapping = { - success: 'green', - error: 'red', - warn: 'yellow' - }; - - if (typeMapping[type] !== undefined) { - classes += `gh-notification-${typeMapping[type]}`; - } - - return classes; - }), - - didInsertElement() { - this._super(...arguments); - - this.$().on('animationend webkitAnimationEnd oanimationend MSAnimationEnd', (event) => { - if (event.originalEvent.animationName === 'fade-out') { - this.get('notifications').closeNotification(this.get('message')); - } - }); - }, - - willDestroyElement() { - this._super(...arguments); - this.$().off('animationend webkitAnimationEnd oanimationend MSAnimationEnd'); - }, - - actions: { - closeNotification() { - this.get('notifications').closeNotification(this.get('message')); - } - } -}); diff --git a/core/client/app/components/gh-notifications.js b/core/client/app/components/gh-notifications.js deleted file mode 100644 index a293bbcf35..0000000000 --- a/core/client/app/components/gh-notifications.js +++ /dev/null @@ -1,17 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; -const {alias} = computed; - -export default Component.extend({ - tagName: 'aside', - classNames: 'gh-notifications', - - notifications: service(), - - messages: alias('notifications.notifications') -}); diff --git a/core/client/app/components/gh-popover-button.js b/core/client/app/components/gh-popover-button.js deleted file mode 100644 index ac852680ff..0000000000 --- a/core/client/app/components/gh-popover-button.js +++ /dev/null @@ -1,26 +0,0 @@ -import Ember from 'ember'; -import DropdownButton from 'ghost/components/gh-dropdown-button'; - -const { - inject: {service} -} = Ember; - -function K() { - return this; -} - -export default DropdownButton.extend({ - dropdown: service(), - - click: K, - - mouseEnter() { - this._super(...arguments); - this.get('dropdown').toggleDropdown(this.get('popoverName'), this); - }, - - mouseLeave() { - this._super(...arguments); - this.get('dropdown').toggleDropdown(this.get('popoverName'), this); - } -}); diff --git a/core/client/app/components/gh-popover.js b/core/client/app/components/gh-popover.js deleted file mode 100644 index c724003f0b..0000000000 --- a/core/client/app/components/gh-popover.js +++ /dev/null @@ -1,11 +0,0 @@ -import Ember from 'ember'; -import GhostDropdown from 'ghost/components/gh-dropdown'; - -const { - inject: {service} -} = Ember; - -export default GhostDropdown.extend({ - classNames: 'ghost-popover', - dropdown: service() -}); diff --git a/core/client/app/components/gh-posts-list-item.js b/core/client/app/components/gh-posts-list-item.js deleted file mode 100644 index 51c92c81c0..0000000000 --- a/core/client/app/components/gh-posts-list-item.js +++ /dev/null @@ -1,85 +0,0 @@ -import Ember from 'ember'; -import ActiveLinkWrapper from 'ghost/mixins/active-link-wrapper'; -import {invokeAction} from 'ember-invoke-action'; - -const { - $, - Component, - computed, - inject: {service} -} = Ember; -const {alias, equal} = computed; - -export default Component.extend(ActiveLinkWrapper, { - tagName: 'li', - classNameBindings: ['isFeatured:featured', 'isPage:page'], - - post: null, - previewIsHidden: false, - - isFeatured: alias('post.featured'), - isPage: alias('post.page'), - isPublished: equal('post.status', 'published'), - - ghostPaths: service(), - - authorName: computed('post.author.name', 'post.author.email', function () { - return this.get('post.author.name') || this.get('post.author.email'); - }), - - authorAvatar: computed('post.author.image', function () { - return this.get('post.author.image') || `${this.get('ghostPaths.subdir')}/ghost/img/user-image.png`; - }), - - authorAvatarBackground: computed('authorAvatar', function () { - return Ember.String.htmlSafe(`background-image: url(${this.get('authorAvatar')})`); - }), - - click() { - this.sendAction('onClick', this.get('post')); - }, - - doubleClick() { - this.sendAction('onDoubleClick', this.get('post')); - }, - - didInsertElement() { - this._super(...arguments); - this.addObserver('active', this, this.scrollIntoView); - }, - - willDestroyElement() { - this._super(...arguments); - this.removeObserver('active', this, this.scrollIntoView); - if (this.get('post.isDeleted') && this.get('onDelete')) { - invokeAction(this, 'onDelete'); - } - }, - - scrollIntoView() { - if (!this.get('active')) { - return; - } - - let element = this.$(); - let offset = element.offset().top; - let elementHeight = element.height(); - let container = $('.js-content-scrollbox'); - let containerHeight = container.height(); - let currentScroll = container.scrollTop(); - let isBelowTop, isAboveBottom, isOnScreen; - - isAboveBottom = offset < containerHeight; - isBelowTop = offset > elementHeight; - - isOnScreen = isBelowTop && isAboveBottom; - - if (!isOnScreen) { - // Scroll so that element is centered in container - // 40 is the amount of padding on the container - container.clearQueue().animate({ - scrollTop: currentScroll + offset - 40 - containerHeight / 2 - }); - } - } -}); diff --git a/core/client/app/components/gh-profile-image.js b/core/client/app/components/gh-profile-image.js deleted file mode 100644 index 00f038e8bb..0000000000 --- a/core/client/app/components/gh-profile-image.js +++ /dev/null @@ -1,142 +0,0 @@ -import Ember from 'ember'; -import AjaxService from 'ember-ajax/services/ajax'; -import {NotFoundError} from 'ghost/services/ajax'; - -const { - Component, - computed, - inject: {service}, - isBlank, - run -} = Ember; - -const {notEmpty} = computed; - -/** - * A component to manage a user profile image. By default it just handles picture uploads, - * but if passed a bound 'email' property it will render the user's gravatar image - * - * Example: {{gh-profile-image email=controllerEmailProperty setImage="controllerActionName" debounce=500}} - * - * @param {int} size The size of the image to render - * @param {String} email Reference to a bound email object if gravatar image behavior is desired. - * @param {String|action} setImage The string name of the action on the controller to be called when an image is added. - * @param {int} debounce Period to wait after changes to email before attempting to load gravatar - * @property {Boolean} hasUploadedImage Whether or not the user has uploaded an image (whether or not to show the default image/gravatar image) - * @property {String} defaultImage String containing the background-image css property of the default user profile image - * @property {String} imageBackground String containing the background-image css property with the gravatar url - */ -export default Component.extend({ - email: '', - size: 90, - debounce: 300, - - validEmail: '', - hasUploadedImage: false, - fileStorage: true, - ajax: AjaxService.create(), - config: service(), - - ghostPaths: service(), - displayGravatar: notEmpty('validEmail'), - - init() { - this._super(...arguments); - // Fire this immediately in case we're initialized with a valid email - this.trySetValidEmail(); - }, - - defaultImage: computed('ghostPaths', function () { - let url = `${this.get('ghostPaths.subdir')}/ghost/img/user-image.png`; - return Ember.String.htmlSafe(`background-image: url(${url})`); - }), - - trySetValidEmail() { - if (!this.get('isDestroyed')) { - let email = this.get('email'); - this.set('validEmail', validator.isEmail(email) ? email : ''); - } - }, - - didReceiveAttrs(attrs) { - this._super(...arguments); - let timeout = parseInt(attrs.newAttrs.throttle || this.get('debounce')); - run.debounce(this, 'trySetValidEmail', timeout); - }, - - imageBackground: computed('validEmail', 'size', function () { - let email = this.get('validEmail'); - let size = this.get('size'); - let style = ''; - - if (!isBlank(email)) { - let gravatarUrl = `//www.gravatar.com/avatar/${window.md5(email)}?s=${size}&d=404`; - - this.get('ajax').request(gravatarUrl) - .catch((error) => { - let defaultImageUrl = `url("${this.get('ghostPaths.subdir')}/ghost/img/user-image.png")`; - - if (error instanceof NotFoundError) { - this.$('.placeholder-img')[0].style.backgroundImage = Ember.String.htmlSafe(defaultImageUrl); - } else { - this.$('.placeholder-img')[0].style.backgroundImage = 'url()'; - } - }); - - style = `background-image: url(${gravatarUrl})`; - } - return Ember.String.htmlSafe(style); - }), - - didInsertElement() { - let size = this.get('size'); - let uploadElement = this.$('.js-file-input'); - - this._super(...arguments); - - // while theoretically the 'add' and 'processalways' functions could be - // added as properties of the hash passed to fileupload(), for some reason - // they needed to be placed in an on() call for the add method to work correctly - uploadElement.fileupload({ - url: this.get('ghostPaths.url').api('uploads'), - dropZone: this.$('.js-img-dropzone'), - previewMaxHeight: size, - previewMaxWidth: size, - previewCrop: true, - maxNumberOfFiles: 1, - autoUpload: false - }) - .on('fileuploadadd', run.bind(this, this.queueFile)) - .on('fileuploadprocessalways', run.bind(this, this.triggerPreview)); - }, - - willDestroyElement() { - let $input = this.$('.js-file-input'); - - this._super(...arguments); - - if ($input.length && $input.data()['blueimp-fileupload']) { - $input.fileupload('destroy'); - } - }, - - queueFile(e, data) { - let fileName = data.files[0].name; - - if ((/\.(gif|jpe?g|png|svg?z)$/i).test(fileName)) { - this.sendAction('setImage', data); - } - }, - - triggerPreview(e, data) { - let file = data.files[data.index]; - - if (file.preview) { - this.set('hasUploadedImage', true); - // necessary jQuery code because file.preview is a raw DOM object - // potential todo: rename 'gravatar-img' class in the CSS to be something - // that both the gravatar and the image preview can use that's not so confusing - this.$('.js-img-preview').empty().append(this.$(file.preview).addClass('gravatar-img')); - } - } -}); diff --git a/core/client/app/components/gh-search-input.js b/core/client/app/components/gh-search-input.js deleted file mode 100644 index 93372cd32f..0000000000 --- a/core/client/app/components/gh-search-input.js +++ /dev/null @@ -1,206 +0,0 @@ -/* global key */ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; - -const { - Component, - RSVP, - computed, - run, - inject: {service}, - isBlank, - isEmpty -} = Ember; - -export function computedGroup(category) { - return computed('content', 'currentSearch', function () { - if (!this.get('currentSearch') || !this.get('content')) { - return []; - } - - return this.get('content').filter((item) => { - let search = new RegExp(this.get('currentSearch'), 'ig'); - - return (item.category === category) && - item.title.match(search); - }); - }); -} - -export default Component.extend({ - - selection: null, - content: [], - isLoading: false, - contentExpiry: 10 * 1000, - contentExpiresAt: false, - currentSearch: '', - - posts: computedGroup('Posts'), - pages: computedGroup('Pages'), - users: computedGroup('Users'), - tags: computedGroup('Tags'), - - _store: service('store'), - _routing: service('-routing'), - ajax: service(), - - refreshContent() { - let promises = []; - let now = new Date(); - let contentExpiry = this.get('contentExpiry'); - let contentExpiresAt = this.get('contentExpiresAt'); - - if (this.get('isLoading') || contentExpiresAt > now) { - return RSVP.resolve(); - } - - this.set('isLoading', true); - this.set('content', []); - promises.pushObject(this._loadPosts()); - promises.pushObject(this._loadUsers()); - promises.pushObject(this._loadTags()); - - return RSVP.all(promises).then(() => { }).finally(() => { - this.set('isLoading', false); - this.set('contentExpiresAt', new Date(now.getTime() + contentExpiry)); - }); - }, - - groupedContent: computed('posts', 'pages', 'users', 'tags', function () { - let groups = []; - - if (!isEmpty(this.get('posts'))) { - groups.pushObject({groupName: 'Posts', options: this.get('posts')}); - } - - if (!isEmpty(this.get('pages'))) { - groups.pushObject({groupName: 'Pages', options: this.get('pages')}); - } - - if (!isEmpty(this.get('users'))) { - groups.pushObject({groupName: 'Users', options: this.get('users')}); - } - - if (!isEmpty(this.get('tags'))) { - groups.pushObject({groupName: 'Tags', options: this.get('tags')}); - } - - return groups; - }), - - _loadPosts() { - let store = this.get('_store'); - let postsUrl = `${store.adapterFor('post').urlForQuery({}, 'post')}/`; - let postsQuery = {fields: 'id,title,page', limit: 'all', status: 'all', staticPages: 'all'}; - let content = this.get('content'); - - return this.get('ajax').request(postsUrl, {data: postsQuery}).then((posts) => { - - content.pushObjects(posts.posts.map((post) => { - return { - id: `post.${post.id}`, - title: post.title, - category: post.page ? 'Pages' : 'Posts' - }; - })); - }); - }, - - _loadUsers() { - let store = this.get('_store'); - let usersUrl = `${store.adapterFor('user').urlForQuery({}, 'user')}/`; - let usersQuery = {fields: 'name,slug', limit: 'all'}; - let content = this.get('content'); - - return this.get('ajax').request(usersUrl, {data: usersQuery}).then((users) => { - content.pushObjects(users.users.map((user) => { - return { - id: `user.${user.slug}`, - title: user.name, - category: 'Users' - }; - })); - }); - }, - - _loadTags() { - let store = this.get('_store'); - let tagsUrl = `${store.adapterFor('tag').urlForQuery({}, 'tag')}/`; - let tagsQuery = {fields: 'name,slug', limit: 'all'}; - let content = this.get('content'); - - return this.get('ajax').request(tagsUrl, {data: tagsQuery}).then((tags) => { - content.pushObjects(tags.tags.map((tag) => { - return { - id: `tag.${tag.slug}`, - title: tag.name, - category: 'Tags' - }; - })); - }); - }, - - _performSearch(term, resolve, reject) { - if (isBlank(term)) { - return resolve([]); - } - - this.refreshContent().then(() => { - this.set('currentSearch', term); - - return resolve(this.get('groupedContent')); - }).catch(reject); - }, - - _setKeymasterScope() { - key.setScope('search-input'); - }, - - _resetKeymasterScope() { - key.setScope('default'); - }, - - willDestroy() { - this._super(...arguments); - this._resetKeymasterScope(); - }, - - actions: { - openSelected(selected) { - if (!selected) { - return; - } - - if (selected.category === 'Posts' || selected.category === 'Pages') { - let id = selected.id.replace('post.', ''); - this.get('_routing.router').transitionTo('editor.edit', id); - } - - if (selected.category === 'Users') { - let id = selected.id.replace('user.', ''); - this.get('_routing.router').transitionTo('team.user', id); - } - - if (selected.category === 'Tags') { - let id = selected.id.replace('tag.', ''); - this.get('_routing.router').transitionTo('settings.tags.tag', id); - } - }, - - onFocus() { - this._setKeymasterScope(); - }, - - onBlur() { - this._resetKeymasterScope(); - }, - - search(term) { - return new RSVP.Promise((resolve, reject) => { - run.debounce(this, this._performSearch, term, resolve, reject, 200); - }); - } - } - -}); diff --git a/core/client/app/components/gh-search-input/trigger.js b/core/client/app/components/gh-search-input/trigger.js deleted file mode 100644 index 6559b36d2e..0000000000 --- a/core/client/app/components/gh-search-input/trigger.js +++ /dev/null @@ -1,43 +0,0 @@ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -const {run, isBlank, Component} = Ember; - -export default Component.extend({ - open() { - this.get('select.actions').open(); - }, - - close() { - this.get('select.actions').close(); - }, - - actions: { - captureMouseDown(e) { - e.stopPropagation(); - }, - - search(term) { - if (isBlank(term) === this.get('select.isOpen')) { - run.scheduleOnce('afterRender', this, isBlank(term) ? this.close : this.open); - } - - invokeAction(this, 'select.actions.search', term); - }, - - focusInput() { - this.$('input')[0].focus(); - }, - - resetInput() { - this.$('input').val(''); - }, - - handleKeydown(e) { - let select = this.get('select'); - if (!select.isOpen) { - e.stopPropagation(); - } - } - } -}); diff --git a/core/client/app/components/gh-select-native.js b/core/client/app/components/gh-select-native.js deleted file mode 100644 index 64e1af11d6..0000000000 --- a/core/client/app/components/gh-select-native.js +++ /dev/null @@ -1,42 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed} = Ember; -const {reads} = computed; - -function K() { - return this; -} - -export default Component.extend({ - content: null, - prompt: null, - optionValuePath: 'id', - optionLabelPath: 'title', - selection: null, - action: K, // action to fire on change - - // shadow the passed-in `selection` to avoid - // leaking changes to it via a 2-way binding - _selection: reads('selection'), - - actions: { - change() { - // jscs:disable requireArrayDestructuring - let selectEl = this.$('select')[0]; - // jscs:enable requireArrayDestructuring - let {selectedIndex} = selectEl; - - // decrement index by 1 if we have a prompt - let hasPrompt = !!this.get('prompt'); - let contentIndex = hasPrompt ? selectedIndex - 1 : selectedIndex; - - let selection = this.get('content').objectAt(contentIndex); - - // set the local, shadowed selection to avoid leaking - // changes to `selection` out via 2-way binding - this.set('_selection', selection); - - this.sendAction('action', selection); - } - } -}); diff --git a/core/client/app/components/gh-selectize.js b/core/client/app/components/gh-selectize.js deleted file mode 100644 index 3f3cfa0b5e..0000000000 --- a/core/client/app/components/gh-selectize.js +++ /dev/null @@ -1,124 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import EmberSelectizeComponent from 'ember-cli-selectize/components/ember-selectize'; - -const {computed, isArray, isBlank, get, run} = Ember; -const emberA = Ember.A; - -export default EmberSelectizeComponent.extend({ - - selectizeOptions: computed(function () { - let options = this._super(...arguments); - - options.onChange = run.bind(this, '_onChange'); - - return options; - }), - - /** - * Event callback that is triggered when user creates a tag - * - modified to pass the caret position to the action - */ - _create(input, callback) { - let caret = this._selectize.caretPos; - - // Delete user entered text - this._selectize.setTextboxValue(''); - // Send create action - - // allow the observers and computed properties to run first - run.schedule('actions', this, function () { - this.sendAction('create-item', input, caret); - }); - // We cancel the creation here, so it's up to you to include the created element - // in the content and selection property - callback(null); - }, - - _addSelection(obj) { - let _valuePath = this.get('_valuePath'); - let val = get(obj, _valuePath); - let caret = this._selectize.caretPos; - - // caret position is always 1 more than the desired index as this method - // is called after selectize has inserted the item and the caret has moved - // to the right - caret = caret - 1; - - this.get('selection').insertAt(caret, obj); - - run.schedule('actions', this, function () { - this.sendAction('add-item', obj); - this.sendAction('add-value', val); - }); - }, - - _onChange(args) { - let selection = Ember.get(this, 'selection'); - let valuePath = Ember.get(this, '_valuePath'); - let reorderedSelection = emberA([]); - - if (!args || !selection || !isArray(selection) || args.length !== get(selection, 'length')) { - return; - } - - // exit if we're not dealing with the same objects as the selection - let objectsHaveChanged = selection.any(function (obj) { - return args.indexOf(get(obj, valuePath)) === -1; - }); - - if (objectsHaveChanged) { - return; - } - - // exit if the order is still the same - let orderIsSame = selection.every(function (obj, idx) { - return get(obj, valuePath) === args[idx]; - }); - - if (orderIsSame) { - return; - } - - // we have a re-order, update the selection - args.forEach((value) => { - let obj = selection.find(function (item) { - return `${get(item, valuePath)}` === value; - }); - - if (obj) { - reorderedSelection.addObject(obj); - } - }); - - this.set('selection', reorderedSelection); - }, - - _preventOpeningWhenBlank() { - let openOnFocus = this.get('openOnFocus'); - - if (!openOnFocus) { - run.schedule('afterRender', this, function () { - let selectize = this._selectize; - if (selectize) { - selectize.on('dropdown_open', function () { - if (isBlank(selectize.$control_input.val())) { - selectize.close(); - } - }); - selectize.on('type', function (filter) { - if (isBlank(filter)) { - selectize.close(); - } - }); - } - }); - } - }, - - didInsertElement() { - this._super(...arguments); - this._preventOpeningWhenBlank(); - } - -}); diff --git a/core/client/app/components/gh-skip-link.js b/core/client/app/components/gh-skip-link.js deleted file mode 100644 index 960ba08b81..0000000000 --- a/core/client/app/components/gh-skip-link.js +++ /dev/null @@ -1,36 +0,0 @@ -/*jshint scripturl:true*/ -import Ember from 'ember'; - -const {$, Component} = Ember; - -export default Component.extend({ - tagName: 'a', - anchor: '', - classNames: ['sr-only', 'sr-only-focusable'], - // Add attributes to component for href - // href should be set to retain anchor properties - // such as pointer cursor and text underline - attributeBindings: ['href'], - // Used so that upon clicking on the link - // anchor behaviors or ignored - href: Ember.String.htmlSafe('javascript:;'), - - click() { - let anchor = this.get('anchor'); - let $el = Ember.$(anchor); - - if ($el) { - // Scrolls to the top of main content or whatever - // is passed to the anchor attribute - Ember.$('body').scrollTop($el.offset().top); - - // This sets focus on the content which was skipped to - // upon losing focus, the tabindex should be removed - // so that normal keyboard navigation picks up from focused - // element - Ember.$($el).attr('tabindex', -1).on('blur focusout', function () { - $(this).removeAttr('tabindex'); - }).focus(); - } - } -}); diff --git a/core/client/app/components/gh-spin-button.js b/core/client/app/components/gh-spin-button.js deleted file mode 100644 index 39e203586b..0000000000 --- a/core/client/app/components/gh-spin-button.js +++ /dev/null @@ -1,59 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed, observer, run} = Ember; -const {equal} = computed; - -export default Component.extend({ - tagName: 'button', - buttonText: '', - submitting: false, - showSpinner: false, - showSpinnerTimeout: null, - autoWidth: true, - - // Disable Button when isLoading equals true - attributeBindings: ['disabled', 'type', 'tabindex'], - - // Must be set on the controller - disabled: equal('showSpinner', true), - - click() { - if (this.get('action')) { - this.sendAction('action'); - return false; - } - return true; - }, - - toggleSpinner: observer('submitting', function () { - let submitting = this.get('submitting'); - let timeout = this.get('showSpinnerTimeout'); - - if (submitting) { - this.set('showSpinner', true); - this.set('showSpinnerTimeout', run.later(this, function () { - if (!this.get('submitting')) { - this.set('showSpinner', false); - } - this.set('showSpinnerTimeout', null); - }, 1000)); - } else if (!submitting && timeout === null) { - this.set('showSpinner', false); - } - }), - - setSize: observer('showSpinner', function () { - if (this.get('showSpinner') && this.get('autoWidth')) { - this.$().width(this.$().width()); - this.$().height(this.$().height()); - } else { - this.$().width(''); - this.$().height(''); - } - }), - - willDestroy() { - this._super(...arguments); - run.cancel(this.get('showSpinnerTimeout')); - } -}); diff --git a/core/client/app/components/gh-subscribers-table.js b/core/client/app/components/gh-subscribers-table.js deleted file mode 100644 index cde011adff..0000000000 --- a/core/client/app/components/gh-subscribers-table.js +++ /dev/null @@ -1,17 +0,0 @@ -import Ember from 'ember'; - -export default Ember.Component.extend({ - classNames: ['subscribers-table'], - - table: null, - - actions: { - onScrolledToBottom() { - let loadNextPage = this.get('loadNextPage'); - - if (!this.get('isLoading')) { - loadNextPage(); - } - } - } -}); diff --git a/core/client/app/components/gh-tab-pane.js b/core/client/app/components/gh-tab-pane.js deleted file mode 100644 index 63faec69ac..0000000000 --- a/core/client/app/components/gh-tab-pane.js +++ /dev/null @@ -1,34 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed} = Ember; -const {alias} = computed; - -// See gh-tabs-manager.js for use -export default Component.extend({ - classNameBindings: ['active'], - - tabsManager: computed(function () { - return this.nearestWithProperty('isTabsManager'); - }), - - tab: computed('tabsManager.tabs.[]', 'tabsManager.tabPanes.[]', function () { - let index = this.get('tabsManager.tabPanes').indexOf(this); - let tabs = this.get('tabsManager.tabs'); - - return tabs && tabs.objectAt(index); - }), - - active: alias('tab.active'), - - willRender() { - this._super(...arguments); - // Register with the tabs manager - this.get('tabsManager').registerTabPane(this); - }, - - willDestroyElement() { - this._super(...arguments); - // Deregister with the tabs manager - this.get('tabsManager').unregisterTabPane(this); - } -}); diff --git a/core/client/app/components/gh-tab.js b/core/client/app/components/gh-tab.js deleted file mode 100644 index efd6c79279..0000000000 --- a/core/client/app/components/gh-tab.js +++ /dev/null @@ -1,35 +0,0 @@ -import Ember from 'ember'; - -const {Component, computed} = Ember; - -// See gh-tabs-manager.js for use -export default Component.extend({ - tabsManager: computed(function () { - return this.nearestWithProperty('isTabsManager'); - }), - - active: computed('tabsManager.activeTab', function () { - return this.get('tabsManager.activeTab') === this; - }), - - index: computed('tabsManager.tabs.[]', function () { - return this.get('tabsManager.tabs').indexOf(this); - }), - - // Select on click - click() { - this.get('tabsManager').select(this); - }, - - willRender() { - this._super(...arguments); - // register the tabs with the tab manager - this.get('tabsManager').registerTab(this); - }, - - willDestroyElement() { - this._super(...arguments); - // unregister the tabs with the tab manager - this.get('tabsManager').unregisterTab(this); - } -}); diff --git a/core/client/app/components/gh-tabs-manager.js b/core/client/app/components/gh-tabs-manager.js deleted file mode 100644 index 32c023cd42..0000000000 --- a/core/client/app/components/gh-tabs-manager.js +++ /dev/null @@ -1,85 +0,0 @@ -import Ember from 'ember'; - -const {Component} = Ember; - -/** -Heavily inspired by ic-tabs (https://github.com/instructure/ic-tabs) - -Three components work together for smooth tabbing. -1. tabs-manager (gh-tabs) -2. tab (gh-tab) -3. tab-pane (gh-tab-pane) - -## Usage: -The tabs-manager must wrap all tab and tab-pane components, -but they can be nested at any level. - -A tab and its pane are tied together via their order. -So, the second tab within a tab manager will activate -the second pane within that manager. - -```hbs -{{#gh-tabs-manager}} - {{#gh-tab}} - First tab - {{/gh-tab}} - {{#gh-tab}} - Second tab - {{/gh-tab}} - - .... - {{#gh-tab-pane}} - First pane - {{/gh-tab-pane}} - {{#gh-tab-pane}} - Second pane - {{/gh-tab-pane}} -{{/gh-tabs-manager}} -``` -## Options: - -the tabs-manager will send a "selected" action whenever one of its -tabs is clicked. -```hbs -{{#gh-tabs-manager selected="myAction"}} - .... -{{/gh-tabs-manager}} -``` - -## Styling: -Both tab and tab-pane elements have an "active" -class applied when they are active. - -*/ -export default Component.extend({ - activeTab: null, - tabs: [], - tabPanes: [], - - // Used by children to find this tabsManager - isTabsManager: true, - - // Called when a gh-tab is clicked. - select(tab) { - this.set('activeTab', tab); - this.sendAction('selected'); - }, - - // Register tabs and their panes to allow for - // interaction between components. - registerTab(tab) { - this.get('tabs').addObject(tab); - }, - - unregisterTab(tab) { - this.get('tabs').removeObject(tab); - }, - - registerTabPane(tabPane) { - this.get('tabPanes').addObject(tabPane); - }, - - unregisterTabPane(tabPane) { - this.get('tabPanes').removeObject(tabPane); - } -}); diff --git a/core/client/app/components/gh-tag-settings-form.js b/core/client/app/components/gh-tag-settings-form.js deleted file mode 100644 index b892cf3140..0000000000 --- a/core/client/app/components/gh-tag-settings-form.js +++ /dev/null @@ -1,136 +0,0 @@ -/* global key */ -import Ember from 'ember'; -import boundOneWay from 'ghost/utils/bound-one-way'; -import {invokeAction} from 'ember-invoke-action'; - -const { - Component, - Handlebars, - computed, - get, - inject: {service} -} = Ember; -const {reads} = computed; - -export default Component.extend({ - - tag: null, - - scratchName: boundOneWay('tag.name'), - scratchSlug: boundOneWay('tag.slug'), - scratchDescription: boundOneWay('tag.description'), - scratchMetaTitle: boundOneWay('tag.metaTitle'), - scratchMetaDescription: boundOneWay('tag.metaDescription'), - - isViewingSubview: false, - - config: service(), - mediaQueries: service(), - - isMobile: reads('mediaQueries.maxWidth600'), - - title: computed('tag.isNew', function () { - if (this.get('tag.isNew')) { - return 'New Tag'; - } else { - return 'Tag Settings'; - } - }), - - seoTitle: computed('scratchName', 'scratchMetaTitle', function () { - let metaTitle = this.get('scratchMetaTitle') || ''; - - metaTitle = metaTitle.length > 0 ? metaTitle : this.get('scratchName'); - - if (metaTitle && metaTitle.length > 70) { - metaTitle = metaTitle.substring(0, 70).trim(); - metaTitle = Handlebars.Utils.escapeExpression(metaTitle); - metaTitle = Ember.String.htmlSafe(`${metaTitle}…`); - } - - return metaTitle; - }), - - seoURL: computed('scratchSlug', function () { - let blogUrl = this.get('config.blogUrl'); - let seoSlug = this.get('scratchSlug') || ''; - - let seoURL = `${blogUrl}/tag/${seoSlug}`; - - // only append a slash to the URL if the slug exists - if (seoSlug) { - seoURL += '/'; - } - - if (seoURL.length > 70) { - seoURL = seoURL.substring(0, 70).trim(); - seoURL = Ember.String.htmlSafe(`${seoURL}…`); - } - - return seoURL; - }), - - seoDescription: computed('scratchDescription', 'scratchMetaDescription', function () { - let metaDescription = this.get('scratchMetaDescription') || ''; - - metaDescription = metaDescription.length > 0 ? metaDescription : this.get('scratchDescription'); - - if (metaDescription && metaDescription.length > 156) { - metaDescription = metaDescription.substring(0, 156).trim(); - metaDescription = Handlebars.Utils.escapeExpression(metaDescription); - metaDescription = Ember.String.htmlSafe(`${metaDescription}…`); - } - - return metaDescription; - }), - - didReceiveAttrs(attrs) { - this._super(...arguments); - - if (get(attrs, 'newAttrs.tag.value.id') !== get(attrs, 'oldAttrs.tag.value.id')) { - this.reset(); - } - }, - - reset() { - this.set('isViewingSubview', false); - if (this.$()) { - this.$('.settings-menu-pane').scrollTop(0); - } - }, - - focusIn() { - key.setScope('tag-settings-form'); - }, - - focusOut() { - key.setScope('default'); - }, - - actions: { - setProperty(property, value) { - invokeAction(this, 'setProperty', property, value); - }, - - setCoverImage(image) { - invokeAction(this, 'setProperty', 'image', image); - }, - - clearCoverImage() { - invokeAction(this, 'setProperty', 'image', ''); - }, - - openMeta() { - this.set('isViewingSubview', true); - }, - - closeMeta() { - this.set('isViewingSubview', false); - }, - - deleteTag() { - invokeAction(this, 'showDeleteTagModal'); - } - } - -}); diff --git a/core/client/app/components/gh-tag.js b/core/client/app/components/gh-tag.js deleted file mode 100644 index f4692ac4a6..0000000000 --- a/core/client/app/components/gh-tag.js +++ /dev/null @@ -1,12 +0,0 @@ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -export default Ember.Component.extend({ - willDestroyElement() { - this._super(...arguments); - - if (this.get('tag.isDeleted') && this.get('onDelete')) { - invokeAction(this, 'onDelete'); - } - } -}); diff --git a/core/client/app/components/gh-tags-management-container.js b/core/client/app/components/gh-tags-management-container.js deleted file mode 100644 index 4be8a86127..0000000000 --- a/core/client/app/components/gh-tags-management-container.js +++ /dev/null @@ -1,54 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service}, - isBlank, - observer, - run -} = Ember; -const {equal, reads} = computed; - -export default Component.extend({ - classNames: ['view-container'], - classNameBindings: ['isMobile'], - - mediaQueries: service(), - - tags: null, - selectedTag: null, - - isMobile: reads('mediaQueries.maxWidth600'), - isEmpty: equal('tags.length', 0), - - init() { - this._super(...arguments); - run.schedule('actions', this, this.fireMobileChangeActions); - }, - - displaySettingsPane: computed('isEmpty', 'selectedTag', 'isMobile', function () { - let isEmpty = this.get('isEmpty'); - let selectedTag = this.get('selectedTag'); - let isMobile = this.get('isMobile'); - - // always display settings pane for blank-slate on mobile - if (isMobile && isEmpty) { - return true; - } - - // display list if no tag is selected on mobile - if (isMobile && isBlank(selectedTag)) { - return false; - } - - // default to displaying settings pane - return true; - }), - - fireMobileChangeActions: observer('isMobile', function () { - if (!this.get('isMobile')) { - this.sendAction('leftMobile'); - } - }) -}); diff --git a/core/client/app/components/gh-textarea.js b/core/client/app/components/gh-textarea.js deleted file mode 100644 index e668dd220f..0000000000 --- a/core/client/app/components/gh-textarea.js +++ /dev/null @@ -1,8 +0,0 @@ -import Ember from 'ember'; -import TextInputMixin from 'ghost/mixins/text-input'; - -const {TextArea} = Ember; - -export default TextArea.extend(TextInputMixin, { - classNames: 'gh-input' -}); diff --git a/core/client/app/components/gh-trim-focus-input.js b/core/client/app/components/gh-trim-focus-input.js deleted file mode 100644 index 14edd47123..0000000000 --- a/core/client/app/components/gh-trim-focus-input.js +++ /dev/null @@ -1,41 +0,0 @@ -/*global device*/ -import Ember from 'ember'; - -const {TextField, computed} = Ember; - -export default TextField.extend({ - focus: true, - classNames: 'gh-input', - attributeBindings: ['autofocus'], - - autofocus: computed(function () { - if (this.get('focus')) { - return (device.ios()) ? false : 'autofocus'; - } - - return false; - }), - - _focusField() { - // This fix is required until Mobile Safari has reliable - // autofocus, select() or focus() support - if (this.get('focus') && !device.ios()) { - this.$().val(this.$().val()).focus(); - } - }, - - _trimValue() { - let text = this.$().val(); - this.$().val(text.trim()); - }, - - didInsertElement() { - this._super(...arguments); - this._focusField(); - }, - - focusOut() { - this._super(...arguments); - this._trimValue(); - } -}); diff --git a/core/client/app/components/gh-url-preview.js b/core/client/app/components/gh-url-preview.js deleted file mode 100644 index 48ded92e39..0000000000 --- a/core/client/app/components/gh-url-preview.js +++ /dev/null @@ -1,35 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -/* -Example usage: -{{gh-url-preview prefix="tag" slug=theSlugValue tagName="p" classNames="description"}} -*/ -export default Component.extend({ - classNames: 'ghost-url-preview', - prefix: null, - slug: null, - - config: service(), - - url: computed('slug', function () { - // Get the blog URL and strip the scheme - let blogUrl = this.get('config.blogUrl'); - // Remove `http[s]://` - let noSchemeBlogUrl = blogUrl.substr(blogUrl.indexOf('://') + 3); - - // Get the prefix and slug values - let prefix = this.get('prefix') ? `${this.get('prefix')}/` : ''; - let slug = this.get('slug') ? `${this.get('slug')}/` : ''; - - // Join parts of the URL together with slashes - let theUrl = `${noSchemeBlogUrl}/${prefix}${slug}`; - - return theUrl; - }) -}); diff --git a/core/client/app/components/gh-user-active.js b/core/client/app/components/gh-user-active.js deleted file mode 100644 index 13e7735bdb..0000000000 --- a/core/client/app/components/gh-user-active.js +++ /dev/null @@ -1,31 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: '', - - user: null, - - ghostPaths: service(), - - userDefault: computed('ghostPaths', function () { - return `${this.get('ghostPaths.subdir')}/ghost/img/user-image.png`; - }), - - userImageBackground: computed('user.image', 'userDefault', function () { - let url = this.get('user.image') || this.get('userDefault'); - - return Ember.String.htmlSafe(`background-image: url(${url})`); - }), - - lastLogin: computed('user.lastLogin', function () { - let lastLogin = this.get('user.lastLogin'); - - return lastLogin ? lastLogin.fromNow() : '(Never)'; - }) -}); diff --git a/core/client/app/components/gh-user-invited.js b/core/client/app/components/gh-user-invited.js deleted file mode 100644 index 45a89c2ccb..0000000000 --- a/core/client/app/components/gh-user-invited.js +++ /dev/null @@ -1,69 +0,0 @@ -import Ember from 'ember'; - -const { - Component, - computed, - inject: {service} -} = Ember; - -export default Component.extend({ - tagName: '', - - user: null, - isSending: false, - - notifications: service(), - - createdAt: computed('user.createdAt', function () { - let createdAt = this.get('user.createdAt'); - - return createdAt ? createdAt.fromNow() : ''; - }), - - actions: { - resend() { - let user = this.get('user'); - let notifications = this.get('notifications'); - - this.set('isSending', true); - user.resendInvite().then((result) => { - let notificationText = `Invitation resent! (${user.get('email')})`; - - // If sending the invitation email fails, the API will still return a status of 201 - // but the user's status in the response object will be 'invited-pending'. - if (result.users[0].status === 'invited-pending') { - notifications.showAlert('Invitation email was not sent. Please try resending.', {type: 'error', key: 'invite.resend.not-sent'}); - } else { - user.set('status', result.users[0].status); - notifications.showNotification(notificationText, {key: 'invite.resend.success'}); - } - }).catch((error) => { - notifications.showAPIError(error, {key: 'invite.resend'}); - }).finally(() => { - this.set('isSending', false); - }); - }, - - revoke() { - let user = this.get('user'); - let email = user.get('email'); - let notifications = this.get('notifications'); - - // reload the user to get the most up-to-date information - user.reload().then(() => { - if (user.get('invited')) { - user.destroyRecord().then(() => { - let notificationText = `Invitation revoked. (${email})`; - notifications.showNotification(notificationText, {key: 'invite.revoke.success'}); - }).catch((error) => { - notifications.showAPIError(error, {key: 'invite.revoke'}); - }); - } else { - // if the user is no longer marked as "invited", then show a warning and reload the route - this.sendAction('reload'); - notifications.showAlert('This user has already accepted the invitation.', {type: 'error', delayed: true, key: 'invite.revoke.already-accepted'}); - } - }); - } - } -}); diff --git a/core/client/app/components/gh-validation-status-container.js b/core/client/app/components/gh-validation-status-container.js deleted file mode 100644 index 5ddfb3ba13..0000000000 --- a/core/client/app/components/gh-validation-status-container.js +++ /dev/null @@ -1,26 +0,0 @@ -import Ember from 'ember'; -import ValidationStateMixin from 'ghost/mixins/validation-state'; - -const {Component, computed} = Ember; - -/** - * Handles the CSS necessary to show a specific property state. When passed a - * DS.Errors object and a property name, if the DS.Errors object has errors for - * the specified property, it will change the CSS to reflect the error state - * @param {DS.Errors} errors The DS.Errors object - * @param {string} property Name of the property - */ -export default Component.extend(ValidationStateMixin, { - classNameBindings: ['errorClass'], - - errorClass: computed('property', 'hasError', 'hasValidated.[]', function () { - let hasValidated = this.get('hasValidated'); - let property = this.get('property'); - - if (hasValidated && hasValidated.contains(property)) { - return this.get('hasError') ? 'error' : 'success'; - } else { - return ''; - } - }) -}); diff --git a/core/client/app/components/gh-view-title.js b/core/client/app/components/gh-view-title.js deleted file mode 100644 index 0c70b0c9d2..0000000000 --- a/core/client/app/components/gh-view-title.js +++ /dev/null @@ -1,14 +0,0 @@ -import Ember from 'ember'; - -const {Component} = Ember; - -export default Component.extend({ - tagName: 'h2', - classNames: ['view-title'], - - actions: { - openMobileMenu() { - this.sendAction('openMobileMenu'); - } - } -}); diff --git a/core/client/app/components/modals/base.js b/core/client/app/components/modals/base.js deleted file mode 100644 index b77cdace4c..0000000000 --- a/core/client/app/components/modals/base.js +++ /dev/null @@ -1,56 +0,0 @@ -/* global key */ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -const {Component, run} = Ember; - -export default Component.extend({ - tagName: 'section', - classNames: 'modal-content', - - _previousKeymasterScope: null, - - _setupShortcuts() { - run(function () { - document.activeElement.blur(); - }); - this._previousKeymasterScope = key.getScope(); - - key('enter', 'modal', () => { - this.send('confirm'); - }); - - key('escape', 'modal', () => { - this.send('closeModal'); - }); - - key.setScope('modal'); - }, - - _removeShortcuts() { - key.unbind('enter', 'modal'); - key.unbind('escape', 'modal'); - - key.setScope(this._previousKeymasterScope); - }, - - didInsertElement() { - this._super(...arguments); - this._setupShortcuts(); - }, - - willDestroyElement() { - this._super(...arguments); - this._removeShortcuts(); - }, - - actions: { - confirm() { - throw new Error('You must override the "confirm" action in your modal component'); - }, - - closeModal() { - invokeAction(this, 'closeModal'); - } - } -}); diff --git a/core/client/app/components/modals/copy-html.js b/core/client/app/components/modals/copy-html.js deleted file mode 100644 index 078b0b12be..0000000000 --- a/core/client/app/components/modals/copy-html.js +++ /dev/null @@ -1,9 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; - -const {computed} = Ember; -const {alias} = computed; - -export default ModalComponent.extend({ - generatedHtml: alias('model') -}); diff --git a/core/client/app/components/modals/delete-all.js b/core/client/app/components/modals/delete-all.js deleted file mode 100644 index 11b1748df2..0000000000 --- a/core/client/app/components/modals/delete-all.js +++ /dev/null @@ -1,49 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; - -const { - inject: {service} -} = Ember; - -export default ModalComponent.extend({ - - submitting: false, - - ghostPaths: service(), - notifications: service(), - store: service(), - ajax: service(), - - _deleteAll() { - let deleteUrl = this.get('ghostPaths.url').api('db'); - return this.get('ajax').del(deleteUrl); - }, - - _unloadData() { - this.get('store').unloadAll('post'); - this.get('store').unloadAll('tag'); - }, - - _showSuccess() { - this.get('notifications').showAlert('All content deleted from database.', {type: 'success', key: 'all-content.delete.success'}); - }, - - _showFailure(error) { - this.get('notifications').showAPIError(error, {key: 'all-content.delete'}); - }, - - actions: { - confirm() { - this.set('submitting', true); - - this._deleteAll().then(() => { - this._unloadData(); - this._showSuccess(); - }).catch((error) => { - this._showFailure(error); - }).finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/delete-post.js b/core/client/app/components/modals/delete-post.js deleted file mode 100644 index a452805887..0000000000 --- a/core/client/app/components/modals/delete-post.js +++ /dev/null @@ -1,55 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; - -const { - computed, - inject: {service} -} = Ember; - -const {alias} = computed; - -export default ModalComponent.extend({ - - submitting: false, - - post: alias('model'), - - notifications: service(), - routing: service('-routing'), - - _deletePost() { - let post = this.get('post'); - - // definitely want to clear the data store and post of any unsaved, - // client-generated tags - post.updateTags(); - - return post.destroyRecord(); - }, - - _success() { - // clear any previous error messages - this.get('notifications').closeAlerts('post.delete'); - - // redirect to content screen - this.get('routing').transitionTo('posts'); - }, - - _failure() { - this.get('notifications').showAlert('Your post could not be deleted. Please try again.', {type: 'error', key: 'post.delete.failed'}); - }, - - actions: { - confirm() { - this.set('submitting', true); - - this._deletePost().then(() => { - this._success(); - }, () => { - this._failure(); - }).finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/delete-subscriber.js b/core/client/app/components/modals/delete-subscriber.js deleted file mode 100644 index ec8b58c765..0000000000 --- a/core/client/app/components/modals/delete-subscriber.js +++ /dev/null @@ -1,23 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; -import {invokeAction} from 'ember-invoke-action'; - -const {computed} = Ember; -const {alias} = computed; - -export default ModalComponent.extend({ - - submitting: false, - - subscriber: alias('model'), - - actions: { - confirm() { - this.set('submitting', true); - - invokeAction(this, 'confirm').finally(() => { - this.set('submitting', false); - }); - } - } -}); diff --git a/core/client/app/components/modals/delete-tag.js b/core/client/app/components/modals/delete-tag.js deleted file mode 100644 index 3e36d72edc..0000000000 --- a/core/client/app/components/modals/delete-tag.js +++ /dev/null @@ -1,27 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; -import {invokeAction} from 'ember-invoke-action'; - -const {computed} = Ember; -const {alias} = computed; - -export default ModalComponent.extend({ - - submitting: false, - - tag: alias('model'), - - postInflection: computed('tag.count.posts', function () { - return this.get('tag.count.posts') > 1 ? 'posts' : 'post'; - }), - - actions: { - confirm() { - this.set('submitting', true); - - invokeAction(this, 'confirm').finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/delete-user.js b/core/client/app/components/modals/delete-user.js deleted file mode 100644 index 801adac23d..0000000000 --- a/core/client/app/components/modals/delete-user.js +++ /dev/null @@ -1,19 +0,0 @@ -import ModalComponent from 'ghost/components/modals/base'; -import {invokeAction} from 'ember-invoke-action'; - -export default ModalComponent.extend({ - - submitting: false, - - user: null, - - actions: { - confirm() { - this.set('submitting', true); - - invokeAction(this, 'confirm').finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/import-subscribers.js b/core/client/app/components/modals/import-subscribers.js deleted file mode 100644 index 272c643be8..0000000000 --- a/core/client/app/components/modals/import-subscribers.js +++ /dev/null @@ -1,43 +0,0 @@ -import Ember from 'ember'; -import { invokeAction } from 'ember-invoke-action'; -import ModalComponent from 'ghost/components/modals/base'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -const {computed} = Ember; - -export default ModalComponent.extend({ - labelText: 'Select or drag-and-drop a CSV File', - - response: null, - closeDisabled: false, - - uploadUrl: computed(function () { - return `${ghostPaths().apiRoot}/subscribers/csv/`; - }), - - actions: { - uploadStarted() { - this.set('closeDisabled', true); - }, - - uploadFinished() { - this.set('closeDisabled', false); - }, - - uploadSuccess(response) { - this.set('response', response.meta.stats); - // invoke the passed in confirm action - invokeAction(this, 'confirm'); - }, - - confirm() { - // noop - we don't want the enter key doing anything - }, - - closeModal() { - if (!this.get('closeDisabled')) { - this._super(...arguments); - } - } - } -}); diff --git a/core/client/app/components/modals/invite-new-user.js b/core/client/app/components/modals/invite-new-user.js deleted file mode 100644 index 14bd9cb99a..0000000000 --- a/core/client/app/components/modals/invite-new-user.js +++ /dev/null @@ -1,125 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - RSVP: {Promise}, - inject: {service}, - run -} = Ember; -const emberA = Ember.A; - -export default ModalComponent.extend(ValidationEngine, { - classNames: 'modal-content invite-new-user', - - role: null, - roles: null, - authorRole: null, - submitting: false, - - validationType: 'inviteUser', - - notifications: service(), - store: service(), - - init() { - this._super(...arguments); - - // populate roles and set initial value for the dropdown - run.schedule('afterRender', this, function () { - this.get('store').query('role', {permissions: 'assign'}).then((roles) => { - let authorRole = roles.findBy('name', 'Author'); - - this.set('roles', roles); - this.set('authorRole', authorRole); - - if (!this.get('role')) { - this.set('role', authorRole); - } - }); - }); - }, - - willDestroyElement() { - this._super(...arguments); - // TODO: this should not be needed, ValidationEngine acts as a - // singleton and so it's errors and hasValidated state stick around - this.get('errors').clear(); - this.set('hasValidated', emberA()); - }, - - validate() { - let email = this.get('email'); - - // TODO: either the validator should check the email's existence or - // the API should return an appropriate error when attempting to save - return new Promise((resolve, reject) => { - return this._super().then(() => { - this.get('store').findAll('user', {reload: true}).then((result) => { - let invitedUser = result.findBy('email', email); - - if (invitedUser) { - this.get('errors').clear('email'); - if (invitedUser.get('status') === 'invited' || invitedUser.get('status') === 'invited-pending') { - this.get('errors').add('email', 'A user with that email address was already invited.'); - } else { - this.get('errors').add('email', 'A user with that email address already exists.'); - } - - // TODO: this shouldn't be needed, ValidationEngine doesn't mark - // properties as validated when validating an entire object - this.get('hasValidated').addObject('email'); - reject(); - } else { - resolve(); - } - }); - }, () => { - // TODO: this shouldn't be needed, ValidationEngine doesn't mark - // properties as validated when validating an entire object - this.get('hasValidated').addObject('email'); - reject(); - }); - }); - }, - - actions: { - setRole(role) { - this.set('role', role); - }, - - confirm() { - let email = this.get('email'); - let role = this.get('role'); - let notifications = this.get('notifications'); - let newUser; - - this.validate().then(() => { - this.set('submitting', true); - - newUser = this.get('store').createRecord('user', { - email, - role, - status: 'invited' - }); - - newUser.save().then(() => { - let notificationText = `Invitation sent! (${email})`; - - // If sending the invitation email fails, the API will still return a status of 201 - // but the user's status in the response object will be 'invited-pending'. - if (newUser.get('status') === 'invited-pending') { - notifications.showAlert('Invitation email was not sent. Please try resending.', {type: 'error', key: 'invite.send.failed'}); - } else { - notifications.showNotification(notificationText, {key: 'invite.send.success'}); - } - }).catch((errors) => { - newUser.deleteRecord(); - notifications.showErrors(errors, {key: 'invite.send'}); - }).finally(() => { - this.send('closeModal'); - }); - }); - } - } -}); diff --git a/core/client/app/components/modals/leave-editor.js b/core/client/app/components/modals/leave-editor.js deleted file mode 100644 index 98dd690641..0000000000 --- a/core/client/app/components/modals/leave-editor.js +++ /dev/null @@ -1,12 +0,0 @@ -import ModalComponent from 'ghost/components/modals/base'; -import {invokeAction} from 'ember-invoke-action'; - -export default ModalComponent.extend({ - actions: { - confirm() { - invokeAction(this, 'confirm').finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/markdown-help.js b/core/client/app/components/modals/markdown-help.js deleted file mode 100644 index 7827870b6b..0000000000 --- a/core/client/app/components/modals/markdown-help.js +++ /dev/null @@ -1,4 +0,0 @@ -import ModalComponent from 'ghost/components/modals/base'; - -export default ModalComponent.extend({ -}); diff --git a/core/client/app/components/modals/new-subscriber.js b/core/client/app/components/modals/new-subscriber.js deleted file mode 100644 index 246d24b7f7..0000000000 --- a/core/client/app/components/modals/new-subscriber.js +++ /dev/null @@ -1,32 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; - -export default ModalComponent.extend({ - actions: { - updateEmail(newEmail) { - this.set('model.email', newEmail); - this.set('model.hasValidated', Ember.A()); - this.get('model.errors').clear(); - }, - - confirm() { - let confirmAction = this.get('confirm'); - - this.set('submitting', true); - - confirmAction().then(() => { - this.send('closeModal'); - }).catch((errors) => { - let [error] = errors; - if (error && error.match(/email/i)) { - this.get('model.errors').add('email', error); - this.get('model.hasValidated').pushObject('email'); - } - }).finally(() => { - if (!this.get('isDestroying') && !this.get('isDestroyed')) { - this.set('submitting', false); - } - }); - } - } -}); diff --git a/core/client/app/components/modals/re-authenticate.js b/core/client/app/components/modals/re-authenticate.js deleted file mode 100644 index 9c59e12ef1..0000000000 --- a/core/client/app/components/modals/re-authenticate.js +++ /dev/null @@ -1,68 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - $, - computed, - inject: {service} -} = Ember; - -export default ModalComponent.extend(ValidationEngine, { - validationType: 'signin', - - submitting: false, - authenticationError: null, - - notifications: service(), - session: service(), - - identification: computed('session.user.email', function () { - return this.get('session.user.email'); - }), - - _authenticate() { - let session = this.get('session'); - let authStrategy = 'authenticator:oauth2'; - let identification = this.get('identification'); - let password = this.get('password'); - - session.set('skipAuthSuccessHandler', true); - - this.toggleProperty('submitting'); - - return session.authenticate(authStrategy, identification, password).finally(() => { - this.toggleProperty('submitting'); - session.set('skipAuthSuccessHandler', undefined); - }); - }, - - actions: { - confirm() { - // Manually trigger events for input fields, ensuring legacy compatibility with - // browsers and password managers that don't send proper events on autofill - $('#login').find('input').trigger('change'); - - this.set('authenticationError', null); - - this.validate({property: 'signin'}).then(() => { - this._authenticate().then(() => { - this.get('notifications').closeAlerts('post.save'); - this.send('closeModal'); - }).catch((error) => { - if (error && error.errors) { - error.errors.forEach((err) => { - err.message = Ember.String.htmlSafe(err.message); - }); - - this.get('errors').add('password', 'Incorrect password'); - this.get('hasValidated').pushObject('password'); - this.set('authenticationError', error.errors[0].message); - } - }); - }, () => { - this.get('hasValidated').pushObject('password'); - }); - } - } -}); diff --git a/core/client/app/components/modals/transfer-owner.js b/core/client/app/components/modals/transfer-owner.js deleted file mode 100644 index 8753c22f71..0000000000 --- a/core/client/app/components/modals/transfer-owner.js +++ /dev/null @@ -1,17 +0,0 @@ -import ModalComponent from 'ghost/components/modals/base'; -import {invokeAction} from 'ember-invoke-action'; - -export default ModalComponent.extend({ - user: null, - submitting: false, - - actions: { - confirm() { - this.set('submitting', true); - - invokeAction(this, 'confirm').finally(() => { - this.send('closeModal'); - }); - } - } -}); diff --git a/core/client/app/components/modals/upload-image.js b/core/client/app/components/modals/upload-image.js deleted file mode 100644 index 8c063cc5b8..0000000000 --- a/core/client/app/components/modals/upload-image.js +++ /dev/null @@ -1,101 +0,0 @@ -import Ember from 'ember'; -import ModalComponent from 'ghost/components/modals/base'; -import cajaSanitizers from 'ghost/utils/caja-sanitizers'; - -const { - computed, - inject: {service}, - isEmpty -} = Ember; - -export default ModalComponent.extend({ - model: null, - submitting: false, - - url: '', - newUrl: '', - - config: service(), - notifications: service(), - - image: computed('model.model', 'model.imageProperty', { - get() { - let imageProperty = this.get('model.imageProperty'); - - return this.get(`model.model.${imageProperty}`); - }, - - set(key, value) { - let model = this.get('model.model'); - let imageProperty = this.get('model.imageProperty'); - - return model.set(imageProperty, value); - } - }), - - didReceiveAttrs() { - let image = this.get('image'); - this.set('url', image); - this.set('newUrl', image); - }, - - // TODO: should validation be handled in the gh-image-uploader component? - // pro - consistency everywhere, simplification here - // con - difficult if the "save" is happening externally as it does here - // - // maybe it should be handled at the model level? - // - automatically present everywhere - // - file uploads should always result in valid urls so it should only - // affect the url input form - keyDown() { - this._setErrorState(false); - }, - - _setErrorState(state) { - if (state) { - this.$('.url').addClass('error'); - } else { - this.$('.url').removeClass('error'); - } - }, - - _validateUrl(url) { - if (!isEmpty(url) && !cajaSanitizers.url(url)) { - this._setErrorState(true); - return {message: 'Image URI is not valid'}; - } - - return true; - }, - // end validation - - actions: { - fileUploaded(url) { - this.set('url', url); - this.set('newUrl', url); - }, - - removeImage() { - this.set('url', ''); - this.set('newUrl', ''); - }, - - confirm() { - let model = this.get('model.model'); - let newUrl = this.get('newUrl'); - let result = this._validateUrl(newUrl); - let notifications = this.get('notifications'); - - if (result === true) { - this.set('submitting', true); - this.set('image', newUrl); - - model.save().catch((err) => { - notifications.showAPIError(err, {key: 'image.upload'}); - }).finally(() => { - this.send('closeModal'); - }); - } - } - } -}); diff --git a/core/client/app/controllers/about.js b/core/client/app/controllers/about.js deleted file mode 100644 index 5f6c869462..0000000000 --- a/core/client/app/controllers/about.js +++ /dev/null @@ -1,21 +0,0 @@ -import Ember from 'ember'; - -const { - Controller, - computed -} = Ember; - -export default Controller.extend({ - updateNotificationCount: 0, - - actions: { - updateNotificationChange(count) { - this.set('updateNotificationCount', count); - } - }, - - copyrightYear: computed(function () { - let date = new Date(); - return date.getFullYear(); - }) -}); diff --git a/core/client/app/controllers/application.js b/core/client/app/controllers/application.js deleted file mode 100644 index 4a26ce9466..0000000000 --- a/core/client/app/controllers/application.js +++ /dev/null @@ -1,57 +0,0 @@ -import Ember from 'ember'; - -const { - Controller, - computed, - inject: {service} -} = Ember; - -export default Controller.extend({ - dropdown: service(), - session: service(), - - showNavMenu: computed('currentPath', 'session.isAuthenticated', function () { - return (this.get('currentPath') !== 'error404' || this.get('session.isAuthenticated')) && - !this.get('currentPath').match(/(signin|signup|setup|reset)/); - }), - - topNotificationCount: 0, - showMobileMenu: false, - showSettingsMenu: false, - showMarkdownHelpModal: false, - - autoNav: false, - autoNavOpen: computed('autoNav', { - get() { - return false; - }, - set(key, value) { - if (this.get('autoNav')) { - return value; - } - return false; - } - }), - - actions: { - topNotificationChange(count) { - this.set('topNotificationCount', count); - }, - - toggleAutoNav() { - this.toggleProperty('autoNav'); - }, - - openAutoNav() { - this.set('autoNavOpen', true); - }, - - closeAutoNav() { - this.set('autoNavOpen', false); - }, - - closeMobileMenu() { - this.set('showMobileMenu', false); - } - } -}); diff --git a/core/client/app/controllers/editor/edit.js b/core/client/app/controllers/editor/edit.js deleted file mode 100644 index 7cb7b57523..0000000000 --- a/core/client/app/controllers/editor/edit.js +++ /dev/null @@ -1,14 +0,0 @@ -import Ember from 'ember'; -import EditorControllerMixin from 'ghost/mixins/editor-base-controller'; - -const {Controller} = Ember; - -export default Controller.extend(EditorControllerMixin, { - showDeletePostModal: false, - - actions: { - toggleDeletePostModal() { - this.toggleProperty('showDeletePostModal'); - } - } -}); diff --git a/core/client/app/controllers/editor/new.js b/core/client/app/controllers/editor/new.js deleted file mode 100644 index 6dc2ea5040..0000000000 --- a/core/client/app/controllers/editor/new.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; -import EditorControllerMixin from 'ghost/mixins/editor-base-controller'; - -const {Controller} = Ember; - -function K() { - return this; -} - -export default Controller.extend(EditorControllerMixin, { - // Overriding autoSave on the base controller, as the new controller shouldn't be autosaving - autoSave: K, - actions: { - /** - * Redirect to editor after the first save - */ - save(options) { - return this._super(options).then((model) => { - if (model.get('id')) { - this.replaceRoute('editor.edit', model); - } - }); - } - } -}); diff --git a/core/client/app/controllers/error.js b/core/client/app/controllers/error.js deleted file mode 100644 index fae8a96ac5..0000000000 --- a/core/client/app/controllers/error.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; - -const {Controller, computed} = Ember; - -export default Controller.extend({ - - stack: false, - - code: computed('content.status', function () { - return this.get('content.status') > 200 ? this.get('content.status') : 500; - }), - - message: computed('content.statusText', function () { - if (this.get('code') === 404) { - return 'Page not found'; - } - - return this.get('content.statusText') !== 'error' ? this.get('content.statusText') : 'Internal Server Error'; - }) -}); diff --git a/core/client/app/controllers/post-settings-menu.js b/core/client/app/controllers/post-settings-menu.js deleted file mode 100644 index e83e69e4fe..0000000000 --- a/core/client/app/controllers/post-settings-menu.js +++ /dev/null @@ -1,491 +0,0 @@ -import Ember from 'ember'; -import {parseDateString} from 'ghost/utils/date-formatting'; -import SettingsMenuMixin from 'ghost/mixins/settings-menu-controller'; -import boundOneWay from 'ghost/utils/bound-one-way'; -import isNumber from 'ghost/utils/isNumber'; - -const { - $, - ArrayProxy, - Controller, - Handlebars, - PromiseProxyMixin, - RSVP, - computed, - guidFor, - inject: {service, controller}, - isArray, - isBlank, - observer, - run -} = Ember; - -export default Controller.extend(SettingsMenuMixin, { - debounceId: null, - lastPromise: null, - selectedAuthor: null, - - application: controller(), - config: service(), - ghostPaths: service(), - notifications: service(), - session: service(), - slugGenerator: service(), - - initializeSelectedAuthor: observer('model', function () { - return this.get('model.author').then((author) => { - this.set('selectedAuthor', author); - return author; - }); - }), - - authors: computed(function () { - // Loaded asynchronously, so must use promise proxies. - let deferred = {}; - - deferred.promise = this.store.query('user', {limit: 'all'}).then((users) => { - return users.rejectBy('id', 'me').sortBy('name'); - }).then((users) => { - return users.filter((user) => { - return user.get('active'); - }); - }); - - return ArrayProxy - .extend(PromiseProxyMixin) - .create(deferred); - }), - - slugValue: boundOneWay('model.slug'), - - // Requests slug from title - generateAndSetSlug(destination) { - let title = this.get('model.titleScratch'); - let afterSave = this.get('lastPromise'); - let promise; - - // Only set an "untitled" slug once per post - if (title === '(Untitled)' && this.get('model.slug')) { - return; - } - - promise = RSVP.resolve(afterSave).then(() => { - return this.get('slugGenerator').generateSlug('post', title).then((slug) => { - if (!isBlank(slug)) { - this.set(destination, slug); - } - }).catch(() => { - // Nothing to do (would be nice to log this somewhere though), - // but a rejected promise needs to be handled here so that a resolved - // promise is returned. - }); - }); - - this.set('lastPromise', promise); - }, - - metaTitleScratch: boundOneWay('model.metaTitle'), - metaDescriptionScratch: boundOneWay('model.metaDescription'), - - seoTitle: computed('model.titleScratch', 'metaTitleScratch', function () { - let metaTitle = this.get('metaTitleScratch') || ''; - - metaTitle = metaTitle.length > 0 ? metaTitle : this.get('model.titleScratch'); - - if (metaTitle.length > 70) { - metaTitle = metaTitle.substring(0, 70).trim(); - metaTitle = Handlebars.Utils.escapeExpression(metaTitle); - metaTitle = Ember.String.htmlSafe(`${metaTitle}…`); - } - - return metaTitle; - }), - - seoDescription: computed('model.scratch', 'metaDescriptionScratch', function () { - let metaDescription = this.get('metaDescriptionScratch') || ''; - let html = ''; - let el, placeholder; - - if (metaDescription.length > 0) { - placeholder = metaDescription; - } else { - el = $('.rendered-markdown'); - - // Get rendered markdown - if (el !== undefined && el.length > 0) { - html = el.clone(); - html.find('.js-drop-zone').remove(); - html = html[0].innerHTML; - } - - // Strip HTML - placeholder = $('<div />', {html}).text(); - // Replace new lines and trim - placeholder = placeholder.replace(/\n+/g, ' ').trim(); - } - - if (placeholder.length > 156) { - // Limit to 156 characters - placeholder = placeholder.substring(0, 156).trim(); - placeholder = Handlebars.Utils.escapeExpression(placeholder); - placeholder = Ember.String.htmlSafe(`${placeholder}…`); - } - - return placeholder; - }), - - seoURL: computed('model.slug', 'config.blogUrl', function () { - let blogUrl = this.get('config.blogUrl'); - let seoSlug = this.get('model.slug') ? this.get('model.slug') : ''; - let seoURL = `${blogUrl}/${seoSlug}`; - - // only append a slash to the URL if the slug exists - if (seoSlug) { - seoURL += '/'; - } - - if (seoURL.length > 70) { - seoURL = seoURL.substring(0, 70).trim(); - seoURL = Ember.String.htmlSafe(`${seoURL}…`); - } - - return seoURL; - }), - - // observe titleScratch, keeping the post's slug in sync - // with it until saved for the first time. - addTitleObserver: observer('model', function () { - if (this.get('model.isNew') || this.get('model.title') === '(Untitled)') { - this.addObserver('model.titleScratch', this, 'titleObserver'); - } - }), - - titleObserver() { - let title = this.get('model.title'); - let debounceId; - - // generate a slug if a post is new and doesn't have a title yet or - // if the title is still '(Untitled)' and the slug is unaltered. - if ((this.get('model.isNew') && !title) || title === '(Untitled)') { - debounceId = run.debounce(this, 'generateAndSetSlug', 'model.slug', 700); - } - - this.set('debounceId', debounceId); - }, - - // live-query of all tags for tag input autocomplete - availableTags: computed(function () { - return this.get('store').filter('tag', {limit: 'all'}, () => { - return true; - }); - }), - - showErrors(errors) { - errors = isArray(errors) ? errors : [errors]; - this.get('notifications').showErrors(errors); - }, - - actions: { - discardEnter() { - return false; - }, - - togglePage() { - this.toggleProperty('model.page'); - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (this.get('model.isNew')) { - return; - } - - this.get('model').save().catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - toggleFeatured() { - this.toggleProperty('model.featured'); - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (this.get('model.isNew')) { - return; - } - - this.get('model').save(this.get('saveOptions')).catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - /** - * triggered by user manually changing slug - */ - updateSlug(newSlug) { - let slug = this.get('model.slug'); - - newSlug = newSlug || slug; - newSlug = newSlug && newSlug.trim(); - - // Ignore unchanged slugs or candidate slugs that are empty - if (!newSlug || slug === newSlug) { - // reset the input to its previous state - this.set('slugValue', slug); - - return; - } - - this.get('slugGenerator').generateSlug('post', newSlug).then((serverSlug) => { - // If after getting the sanitized and unique slug back from the API - // we end up with a slug that matches the existing slug, abort the change - if (serverSlug === slug) { - return; - } - - // Because the server transforms the candidate slug by stripping - // certain characters and appending a number onto the end of slugs - // to enforce uniqueness, there are cases where we can get back a - // candidate slug that is a duplicate of the original except for - // the trailing incrementor (e.g., this-is-a-slug and this-is-a-slug-2) - - // get the last token out of the slug candidate and see if it's a number - let slugTokens = serverSlug.split('-'); - let check = Number(slugTokens.pop()); - - // if the candidate slug is the same as the existing slug except - // for the incrementor then the existing slug should be used - if (isNumber(check) && check > 0) { - if (slug === slugTokens.join('-') && serverSlug !== newSlug) { - this.set('slugValue', slug); - - return; - } - } - - this.set('model.slug', serverSlug); - - if (this.hasObserverFor('model.titleScratch')) { - this.removeObserver('model.titleScratch', this, 'titleObserver'); - } - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (this.get('model.isNew')) { - return; - } - - return this.get('model').save(); - }).catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - /** - * Parse user's set published date. - * Action sent by post settings menu view. - * (#1351) - */ - setPublishedAt(userInput) { - if (!userInput) { - // Clear out the publishedAt field for a draft - if (this.get('model.isDraft')) { - this.set('model.publishedAt', null); - } - - return; - } - - let newPublishedAt = parseDateString(userInput); - let publishedAt = moment(this.get('model.publishedAt')); - let errMessage = ''; - - // Clear previous errors - this.get('model.errors').remove('post-setting-date'); - - // Validate new Published date - if (!newPublishedAt.isValid()) { - errMessage = 'Published Date must be a valid date with format: ' + - 'DD MMM YY @ HH:mm (e.g. 6 Dec 14 @ 15:00)'; - } else if (newPublishedAt.diff(new Date(), 'h') > 0) { - errMessage = 'Published Date cannot currently be in the future.'; - } - - // If errors, notify and exit. - if (errMessage) { - this.get('model.errors').add('post-setting-date', errMessage); - return; - } - - // Validation complete, update the view - this.set('model.publishedAt', newPublishedAt); - - // Don't save the date if the user didn't actually changed the date - if (publishedAt && publishedAt.isSame(newPublishedAt)) { - return; - } - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (this.get('model.isNew')) { - return; - } - - this.get('model').save().catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - setMetaTitle(metaTitle) { - let property = 'metaTitle'; - let model = this.get('model'); - let currentTitle = model.get(property) || ''; - - // Only update if the title has changed - if (currentTitle === metaTitle) { - return; - } - - model.set(property, metaTitle); - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (model.get('isNew')) { - return; - } - - model.save(); - }, - - setMetaDescription(metaDescription) { - let property = 'metaDescription'; - let model = this.get('model'); - let currentDescription = model.get(property) || ''; - - // Only update if the description has changed - if (currentDescription === metaDescription) { - return; - } - - model.set(property, metaDescription); - - // If this is a new post. Don't save the model. Defer the save - // to the user pressing the save button - if (model.get('isNew')) { - return; - } - - model.save(); - }, - - setCoverImage(image) { - this.set('model.image', image); - - if (this.get('model.isNew')) { - return; - } - - this.get('model').save().catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - clearCoverImage() { - this.set('model.image', ''); - - if (this.get('model.isNew')) { - return; - } - - this.get('model').save().catch((errors) => { - this.showErrors(errors); - this.get('model').rollbackAttributes(); - }); - }, - - resetPubDate() { - this.set('publishedAtValue', ''); - }, - - closeNavMenu() { - this.get('application').send('closeNavMenu'); - }, - - changeAuthor(newAuthor) { - let author = this.get('model.author'); - let model = this.get('model'); - - // return if nothing changed - if (newAuthor.get('id') === author.get('id')) { - return; - } - - model.set('author', newAuthor); - - // if this is a new post (never been saved before), don't try to save it - if (this.get('model.isNew')) { - return; - } - - model.save().catch((errors) => { - this.showErrors(errors); - this.set('selectedAuthor', author); - model.rollbackAttributes(); - }); - }, - - addTag(tagName, index) { - let currentTags = this.get('model.tags'); - let currentTagNames = currentTags.map((tag) => { - return tag.get('name').toLowerCase(); - }); - let availableTagNames, - tagToAdd; - - tagName = tagName.trim(); - - // abort if tag is already selected - if (currentTagNames.contains(tagName.toLowerCase())) { - return; - } - - this.get('availableTags').then((availableTags) => { - availableTagNames = availableTags.map((tag) => { - return tag.get('name').toLowerCase(); - }); - - // find existing tag or create new - if (availableTagNames.contains(tagName.toLowerCase())) { - tagToAdd = availableTags.find((tag) => { - return tag.get('name').toLowerCase() === tagName.toLowerCase(); - }); - } else { - tagToAdd = this.get('store').createRecord('tag', { - name: tagName - }); - - // we need to set a UUID so that selectize has a unique value - // it will be ignored when sent to the server - tagToAdd.set('uuid', guidFor(tagToAdd)); - } - - // push tag onto post relationship - if (tagToAdd) { - this.get('model.tags').insertAt(index, tagToAdd); - } - }); - }, - - removeTag(tag) { - this.get('model.tags').removeObject(tag); - - if (tag.get('isNew')) { - tag.destroyRecord(); - } - } - } -}); diff --git a/core/client/app/controllers/posts.js b/core/client/app/controllers/posts.js deleted file mode 100644 index 6b34a9a87e..0000000000 --- a/core/client/app/controllers/posts.js +++ /dev/null @@ -1,88 +0,0 @@ -import Ember from 'ember'; - -const {Controller, compare, computed} = Ember; -const {equal} = computed; - -// a custom sort function is needed in order to sort the posts list the same way the server would: -// status: ASC -// publishedAt: DESC -// updatedAt: DESC -// id: DESC -function comparator(item1, item2) { - let updated1 = item1.get('updatedAt'); - let updated2 = item2.get('updatedAt'); - let idResult, - publishedAtResult, - statusResult, - updatedAtResult; - - // when `updatedAt` is undefined, the model is still - // being written to with the results from the server - if (item1.get('isNew') || !updated1) { - return -1; - } - - if (item2.get('isNew') || !updated2) { - return 1; - } - - idResult = compare(parseInt(item1.get('id')), parseInt(item2.get('id'))); - statusResult = compare(item1.get('status'), item2.get('status')); - updatedAtResult = compare(updated1.valueOf(), updated2.valueOf()); - publishedAtResult = publishedAtCompare(item1, item2); - - if (statusResult === 0) { - if (publishedAtResult === 0) { - if (updatedAtResult === 0) { - // This should be DESC - return idResult * -1; - } - // This should be DESC - return updatedAtResult * -1; - } - // This should be DESC - return publishedAtResult * -1; - } - - return statusResult; -} - -function publishedAtCompare(item1, item2) { - let published1 = item1.get('publishedAt'); - let published2 = item2.get('publishedAt'); - - if (!published1 && !published2) { - return 0; - } - - if (!published1 && published2) { - return -1; - } - - if (!published2 && published1) { - return 1; - } - - return compare(published1.valueOf(), published2.valueOf()); -} - -export default Controller.extend({ - - showDeletePostModal: false, - - // See PostsRoute's shortcuts - postListFocused: equal('keyboardFocus', 'postList'), - postContentFocused: equal('keyboardFocus', 'postContent'), - - sortedPosts: computed('model.@each.status', 'model.@each.publishedAt', 'model.@each.isNew', 'model.@each.updatedAt', function () { - let postsArray = this.get('model').toArray(); - - return postsArray.sort(comparator); - }), - - actions: { - toggleDeletePostModal() { - this.toggleProperty('showDeletePostModal'); - } - } -}); diff --git a/core/client/app/controllers/reset.js b/core/client/app/controllers/reset.js deleted file mode 100644 index e1a94d8e04..0000000000 --- a/core/client/app/controllers/reset.js +++ /dev/null @@ -1,75 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - Controller, - computed, - inject: {service} -} = Ember; - -export default Controller.extend(ValidationEngine, { - newPassword: '', - ne2Password: '', - token: '', - submitting: false, - flowErrors: '', - - validationType: 'reset', - - ghostPaths: service(), - notifications: service(), - session: service(), - ajax: service(), - - email: computed('token', function () { - // The token base64 encodes the email (and some other stuff), - // each section is divided by a '|'. Email comes second. - return atob(this.get('token')).split('|')[1]; - }), - - // Used to clear sensitive information - clearData() { - this.setProperties({ - newPassword: '', - ne2Password: '', - token: '' - }); - }, - - actions: { - submit() { - let credentials = this.getProperties('newPassword', 'ne2Password', 'token'); - - this.set('flowErrors', ''); - this.get('hasValidated').addObjects(['newPassword', 'ne2Password']); - this.validate().then(() => { - let authUrl = this.get('ghostPaths.url').api('authentication', 'passwordreset'); - this.toggleProperty('submitting'); - this.get('ajax').put(authUrl, { - data: { - passwordreset: [credentials] - } - }).then((resp) => { - this.toggleProperty('submitting'); - this.get('notifications').showAlert(resp.passwordreset[0].message, {type: 'warn', delayed: true, key: 'password.reset'}); - this.get('session').authenticate('authenticator:oauth2', this.get('email'), credentials.newPassword); - }).catch((error) => { - this.get('notifications').showAPIError(error, {key: 'password.reset'}); - this.toggleProperty('submitting'); - }); - }).catch((error) => { - if (this.get('errors.newPassword')) { - this.set('flowErrors', this.get('errors.newPassword')[0].message); - } - - if (this.get('errors.ne2Password')) { - this.set('flowErrors', this.get('errors.ne2Password')[0].message); - } - - if (this.get('errors.length') === 0) { - throw error; - } - }); - } - } -}); diff --git a/core/client/app/controllers/settings/apps/index.js b/core/client/app/controllers/settings/apps/index.js deleted file mode 100644 index 5017fbc1d7..0000000000 --- a/core/client/app/controllers/settings/apps/index.js +++ /dev/null @@ -1,14 +0,0 @@ -import Ember from 'ember'; - -const { - computed, - inject: {controller} -} = Ember; - -const {alias} = computed; - -export default Ember.Controller.extend({ - appsController: controller('settings.apps'), - - slack: alias('appsController.model.slack.firstObject') -}); diff --git a/core/client/app/controllers/settings/apps/slack.js b/core/client/app/controllers/settings/apps/slack.js deleted file mode 100644 index 26877cd8a6..0000000000 --- a/core/client/app/controllers/settings/apps/slack.js +++ /dev/null @@ -1,75 +0,0 @@ -import Ember from 'ember'; -import { invoke } from 'ember-invoke-action'; - -const { - Controller, - computed: {empty}, - inject: {service} -} = Ember; - -export default Controller.extend({ - ghostPaths: service(), - ajax: service(), - notifications: service(), - - // will be set by route - settings: null, - - isSaving: false, - savePromise: null, - isSendingTest: false, - - testNotificationDisabled: empty('model.url'), - - actions: { - sendTestNotification() { - let notifications = this.get('notifications'); - let slackApi = this.get('ghostPaths.url').api('slack', 'test'); - - if (this.get('isSendingTest')) { - return; - } - - this.set('isSendingTest', true); - - invoke(this, 'save').then(() => { - this.get('ajax').post(slackApi).then(() => { - notifications.showAlert('Check your slack channel test message.', {type: 'info', key: 'slack-test.send.success'}); - }).catch((error) => { - notifications.showAPIError(error, {key: 'slack-test:send'}); - }); - }).catch(() => { - // noop - error already handled in .save - }).finally(() => { - this.set('isSendingTest', false); - }); - }, - - updateURL(value) { - this.set('model.url', value); - this.get('model.errors').clear(); - }, - - save() { - let slack = this.get('model'); - let settings = this.get('settings'); - - if (this.get('isSaving')) { - return; - } - - return slack.validate().then(() => { - settings.get('slack').clear().pushObject(slack); - - this.set('isSaving', true); - - return settings.save().catch((err) => { - this.get('notifications').showErrors(err); - throw err; - }).finally(() => { - this.set('isSaving', false); - }); - }); - } - } -}); diff --git a/core/client/app/controllers/settings/code-injection.js b/core/client/app/controllers/settings/code-injection.js deleted file mode 100644 index 576b611bb8..0000000000 --- a/core/client/app/controllers/settings/code-injection.js +++ /dev/null @@ -1,19 +0,0 @@ -import Ember from 'ember'; -import SettingsSaveMixin from 'ghost/mixins/settings-save'; - -const { - Controller, - inject: {service} -} = Ember; - -export default Controller.extend(SettingsSaveMixin, { - notifications: service(), - - save() { - let notifications = this.get('notifications'); - - return this.get('model').save().catch((error) => { - notifications.showAPIError(error, {key: 'code-injection.save'}); - }); - } -}); diff --git a/core/client/app/controllers/settings/general.js b/core/client/app/controllers/settings/general.js deleted file mode 100644 index 92938e2ea7..0000000000 --- a/core/client/app/controllers/settings/general.js +++ /dev/null @@ -1,253 +0,0 @@ -import Ember from 'ember'; -import SettingsSaveMixin from 'ghost/mixins/settings-save'; -import randomPassword from 'ghost/utils/random-password'; - -const { - Controller, - computed, - inject: {service}, - observer, - run -} = Ember; - -export default Controller.extend(SettingsSaveMixin, { - - showUploadLogoModal: false, - showUploadCoverModal: false, - - notifications: service(), - config: service(), - _scratchFacebook: null, - _scratchTwitter: null, - - selectedTheme: computed('model.activeTheme', 'themes', function () { - let activeTheme = this.get('model.activeTheme'); - let themes = this.get('themes'); - let selectedTheme; - - themes.forEach((theme) => { - if (theme.name === activeTheme) { - selectedTheme = theme; - } - }); - - return selectedTheme; - }), - - logoImageSource: computed('model.logo', function () { - return this.get('model.logo') || ''; - }), - - coverImageSource: computed('model.cover', function () { - return this.get('model.cover') || ''; - }), - - isDatedPermalinks: computed('model.permalinks', { - set(key, value) { - this.set('model.permalinks', value ? '/:year/:month/:day/:slug/' : '/:slug/'); - - let slugForm = this.get('model.permalinks'); - return slugForm !== '/:slug/'; - }, - - get() { - let slugForm = this.get('model.permalinks'); - - return slugForm !== '/:slug/'; - } - }), - - themes: computed(function () { - return this.get('model.availableThemes').reduce(function (themes, t) { - let theme = {}; - - theme.name = t.name; - theme.label = t.package ? `${t.package.name} - ${t.package.version}` : t.name; - theme.package = t.package; - theme.active = !!t.active; - - themes.push(theme); - - return themes; - }, []); - }).readOnly(), - - generatePassword: observer('model.isPrivate', function () { - this.get('model.errors').remove('password'); - if (this.get('model.isPrivate') && this.get('model.hasDirtyAttributes')) { - this.get('model').set('password', randomPassword()); - } - }), - - save() { - let notifications = this.get('notifications'); - let config = this.get('config'); - - return this.get('model').save().then((model) => { - config.set('blogTitle', model.get('title')); - - // this forces the document title to recompute after - // a blog title change - this.send('collectTitleTokens', []); - - return model; - }).catch((error) => { - if (error) { - notifications.showAPIError(error, {key: 'settings.save'}); - } - throw error; - }); - }, - - actions: { - checkPostsPerPage() { - let postsPerPage = this.get('model.postsPerPage'); - - if (postsPerPage < 1 || postsPerPage > 1000 || isNaN(postsPerPage)) { - this.set('model.postsPerPage', 5); - } - }, - - setTheme(theme) { - this.set('model.activeTheme', theme.name); - }, - - toggleUploadCoverModal() { - this.toggleProperty('showUploadCoverModal'); - }, - - toggleUploadLogoModal() { - this.toggleProperty('showUploadLogoModal'); - }, - - validateFacebookUrl() { - let newUrl = this.get('_scratchFacebook'); - let oldUrl = this.get('model.facebook'); - let errMessage = ''; - - if (newUrl === '') { - // Clear out the Facebook url - this.set('model.facebook', ''); - this.get('model.errors').remove('facebook'); - return; - } - - // _scratchFacebook will be null unless the user has input something - if (!newUrl) { - newUrl = oldUrl; - } - - // If new url didn't change, exit - if (newUrl === oldUrl) { - this.get('model.errors').remove('facebook'); - return; - } - - if (newUrl.match(/(?:facebook\.com\/)(\S+)/) || newUrl.match(/([a-z\d\.]+)/i)) { - let username = []; - - if (newUrl.match(/(?:facebook\.com\/)(\S+)/)) { - [ , username ] = newUrl.match(/(?:facebook\.com\/)(\S+)/); - } else { - [ , username ] = newUrl.match(/(?:https\:\/\/|http\:\/\/)?(?:www\.)?(?:\w+\.\w+\/+)?(\S+)/mi); - } - - // check if we have a /page/username or without - if (username.match(/^(?:\/)?(pages?\/\S+)/mi)) { - // we got a page url, now save the username without the / in the beginning - - [ , username ] = username.match(/^(?:\/)?(pages?\/\S+)/mi); - } else if (username.match(/^(http|www)|(\/)/) || !username.match(/^([a-z\d\.]{5,50})$/mi)) { - errMessage = !username.match(/^([a-z\d\.]{5,50})$/mi) ? 'Your Page name is not a valid Facebook Page name' : 'The URL must be in a format like https://www.facebook.com/yourPage'; - - this.get('model.errors').add('facebook', errMessage); - this.get('model.hasValidated').pushObject('facebook'); - return; - } - - newUrl = `https://www.facebook.com/${username}`; - this.set('model.facebook', newUrl); - - this.get('model.errors').remove('facebook'); - this.get('model.hasValidated').pushObject('facebook'); - - // User input is validated - return this.save().then(() => { - this.set('model.facebook', ''); - run.schedule('afterRender', this, function () { - this.set('model.facebook', newUrl); - }); - }); - } else { - errMessage = 'The URL must be in a format like ' + - 'https://www.facebook.com/yourPage'; - this.get('model.errors').add('facebook', errMessage); - this.get('model.hasValidated').pushObject('facebook'); - return; - } - }, - - validateTwitterUrl() { - let newUrl = this.get('_scratchTwitter'); - let oldUrl = this.get('model.twitter'); - let errMessage = ''; - - if (newUrl === '') { - // Clear out the Twitter url - this.set('model.twitter', ''); - this.get('model.errors').remove('twitter'); - return; - } - - // _scratchTwitter will be null unless the user has input something - if (!newUrl) { - newUrl = oldUrl; - } - - // If new url didn't change, exit - if (newUrl === oldUrl) { - this.get('model.errors').remove('twitter'); - return; - } - - if (newUrl.match(/(?:twitter\.com\/)(\S+)/) || newUrl.match(/([a-z\d\.]+)/i)) { - let username = []; - - if (newUrl.match(/(?:twitter\.com\/)(\S+)/)) { - [ , username] = newUrl.match(/(?:twitter\.com\/)(\S+)/); - } else { - [username] = newUrl.match(/([^/]+)\/?$/mi); - } - - // check if username starts with http or www and show error if so - if (username.match(/^(http|www)|(\/)/) || !username.match(/^[a-z\d\.\_]{1,15}$/mi)) { - errMessage = !username.match(/^[a-z\d\.\_]{1,15}$/mi) ? 'Your Username is not a valid Twitter Username' : 'The URL must be in a format like https://twitter.com/yourUsername'; - - this.get('model.errors').add('twitter', errMessage); - this.get('model.hasValidated').pushObject('twitter'); - return; - } - - newUrl = `https://twitter.com/${username}`; - this.set('model.twitter', newUrl); - - this.get('model.errors').remove('twitter'); - this.get('model.hasValidated').pushObject('twitter'); - - // User input is validated - return this.save().then(() => { - this.set('model.twitter', ''); - run.schedule('afterRender', this, function () { - this.set('model.twitter', newUrl); - }); - }); - } else { - errMessage = 'The URL must be in a format like ' + - 'https://twitter.com/yourUsername'; - this.get('model.errors').add('twitter', errMessage); - this.get('model.hasValidated').pushObject('twitter'); - return; - } - } - } -}); diff --git a/core/client/app/controllers/settings/labs.js b/core/client/app/controllers/settings/labs.js deleted file mode 100644 index 911e941a92..0000000000 --- a/core/client/app/controllers/settings/labs.js +++ /dev/null @@ -1,89 +0,0 @@ -import Ember from 'ember'; - -const { - $, - Controller, - inject: {service}, - isArray -} = Ember; - -export default Controller.extend({ - uploadButtonText: 'Import', - importErrors: '', - submitting: false, - showDeleteAllModal: false, - - ghostPaths: service(), - notifications: service(), - session: service(), - ajax: service(), - - actions: { - onUpload(file) { - let formData = new FormData(); - let notifications = this.get('notifications'); - let currentUserId = this.get('session.user.id'); - let dbUrl = this.get('ghostPaths.url').api('db'); - - this.set('uploadButtonText', 'Importing'); - this.set('importErrors', ''); - - formData.append('importfile', file); - - this.get('ajax').post(dbUrl, { - data: formData, - dataType: 'json', - cache: false, - contentType: false, - processData: false - }).then(() => { - // Clear the store, so that all the new data gets fetched correctly. - this.store.unloadAll(); - // Reload currentUser and set session - this.set('session.user', this.store.findRecord('user', currentUserId)); - // TODO: keep as notification, add link to view content - notifications.showNotification('Import successful.', {key: 'import.upload.success'}); - }).catch((response) => { - if (response && response.errors && isArray(response.errors)) { - this.set('importErrors', response.errors); - } - - notifications.showAlert('Import Failed', {type: 'error', key: 'import.upload.failed'}); - }).finally(() => { - this.set('uploadButtonText', 'Import'); - }); - }, - - exportData() { - let dbUrl = this.get('ghostPaths.url').api('db'); - let accessToken = this.get('session.data.authenticated.access_token'); - let downloadURL = `${dbUrl}?access_token=${accessToken}`; - let iframe = $('#iframeDownload'); - - if (iframe.length === 0) { - iframe = $('<iframe>', {id: 'iframeDownload'}).hide().appendTo('body'); - } - - iframe.attr('src', downloadURL); - }, - - sendTestEmail() { - let notifications = this.get('notifications'); - let emailUrl = this.get('ghostPaths.url').api('mail', 'test'); - - this.toggleProperty('submitting'); - - this.get('ajax').post(emailUrl).then(() => { - notifications.showAlert('Check your email for the test message.', {type: 'info', key: 'test-email.send.success'}); - this.toggleProperty('submitting'); - }).catch((error) => { - notifications.showAPIError(error, {key: 'test-email:send'}); - this.toggleProperty('submitting'); - }); - }, - - toggleDeleteAllModal() { - this.toggleProperty('showDeleteAllModal'); - } - } -}); diff --git a/core/client/app/controllers/settings/navigation.js b/core/client/app/controllers/settings/navigation.js deleted file mode 100644 index ee91b84478..0000000000 --- a/core/client/app/controllers/settings/navigation.js +++ /dev/null @@ -1,101 +0,0 @@ -import Ember from 'ember'; -import SettingsSaveMixin from 'ghost/mixins/settings-save'; -import NavigationItem from 'ghost/models/navigation-item'; - -const { - Controller, - RSVP, - computed, - inject: {service} -} = Ember; - -export default Controller.extend(SettingsSaveMixin, { - config: service(), - notifications: service(), - - newNavItem: null, - - blogUrl: computed('config.blogUrl', function () { - let url = this.get('config.blogUrl'); - - return url.slice(-1) !== '/' ? `${url}/` : url; - }), - - init() { - this._super(...arguments); - this.set('newNavItem', NavigationItem.create({isNew: true})); - }, - - save() { - let navItems = this.get('model.navigation'); - let newNavItem = this.get('newNavItem'); - let notifications = this.get('notifications'); - let validationPromises = []; - - if (!newNavItem.get('isBlank')) { - validationPromises.pushObject(this.send('addItem')); - } - - navItems.map((item) => { - validationPromises.pushObject(item.validate()); - }); - - return RSVP.all(validationPromises).then(() => { - return this.get('model').save().catch((err) => { - notifications.showErrors(err); - }); - }).catch(() => { - // TODO: noop - needed to satisfy spinner button - }); - }, - - addNewNavItem() { - let navItems = this.get('model.navigation'); - let newNavItem = this.get('newNavItem'); - - newNavItem.set('isNew', false); - navItems.pushObject(newNavItem); - this.set('newNavItem', NavigationItem.create({isNew: true})); - }, - - actions: { - addItem() { - let newNavItem = this.get('newNavItem'); - - // If the url sent through is blank (user never edited the url) - if (newNavItem.get('url') === '') { - newNavItem.set('url', '/'); - } - - return newNavItem.validate().then(() => { - this.addNewNavItem(); - }); - }, - - deleteItem(item) { - if (!item) { - return; - } - - let navItems = this.get('model.navigation'); - - navItems.removeObject(item); - }, - - reorderItems(navItems) { - this.set('model.navigation', navItems); - }, - - updateUrl(url, navItem) { - if (!navItem) { - return; - } - - navItem.set('url', url); - }, - - reset() { - this.set('newNavItem', NavigationItem.create({isNew: true})); - } - } -}); diff --git a/core/client/app/controllers/settings/tags.js b/core/client/app/controllers/settings/tags.js deleted file mode 100644 index 6bd1050cf8..0000000000 --- a/core/client/app/controllers/settings/tags.js +++ /dev/null @@ -1,43 +0,0 @@ -import Ember from 'ember'; - -const { - computed, - inject: {controller} -} = Ember; -const {alias, equal, sort} = computed; - -export default Ember.Controller.extend({ - - tagController: controller('settings.tags.tag'), - - selectedTag: alias('tagController.tag'), - - tagListFocused: equal('keyboardFocus', 'tagList'), - tagContentFocused: equal('keyboardFocus', 'tagContent'), - - // TODO: replace with ordering by page count once supported by the API - tags: sort('model', function (a, b) { - let idA = +a.get('id'); - let idB = +b.get('id'); - - if (idA > idB) { - return 1; - } else if (idA < idB) { - return -1; - } - - return 0; - }), - - actions: { - leftMobile() { - let firstTag = this.get('tags.firstObject'); - // redirect to first tag if possible so that you're not left with - // tag settings blank slate when switching from portrait to landscape - if (firstTag && !this.get('tagController.tag')) { - this.transitionToRoute('settings.tags.tag', firstTag); - } - } - } - -}); diff --git a/core/client/app/controllers/settings/tags/tag.js b/core/client/app/controllers/settings/tags/tag.js deleted file mode 100644 index 085f70ed18..0000000000 --- a/core/client/app/controllers/settings/tags/tag.js +++ /dev/null @@ -1,81 +0,0 @@ -import Ember from 'ember'; - -const { - Controller, - computed, - inject: {service, controller} -} = Ember; -const {alias} = computed; - -export default Controller.extend({ - - showDeleteTagModal: false, - - tag: alias('model'), - isMobile: alias('tagsController.isMobile'), - - applicationController: controller('application'), - tagsController: controller('settings.tags'), - notifications: service(), - - _saveTagProperty(propKey, newValue) { - let tag = this.get('tag'); - let currentValue = tag.get(propKey); - - newValue = newValue.trim(); - - // Quit if there was no change - if (newValue === currentValue) { - return; - } - - tag.set(propKey, newValue); - // TODO: This is required until .validate/.save mark fields as validated - tag.get('hasValidated').addObject(propKey); - - tag.save().then((savedTag) => { - // replace 'new' route with 'tag' route - this.replaceRoute('settings.tags.tag', savedTag); - }).catch((error) => { - if (error) { - this.get('notifications').showAPIError(error, {key: 'tag.save'}); - } - }); - }, - - _deleteTag() { - let tag = this.get('tag'); - - return tag.destroyRecord().then(() => { - this._deleteTagSuccess(); - }, (error) => { - this._deleteTagFailure(error); - }); - }, - - _deleteTagSuccess() { - let currentRoute = this.get('applicationController.currentRouteName') || ''; - - if (currentRoute.match(/^settings\.tags/)) { - this.transitionToRoute('settings.tags.index'); - } - }, - - _deleteTagFailure(error) { - this.get('notifications').showAPIError(error, {key: 'tag.delete'}); - }, - - actions: { - setProperty(propKey, value) { - this._saveTagProperty(propKey, value); - }, - - toggleDeleteTagModal() { - this.toggleProperty('showDeleteTagModal'); - }, - - deleteTag() { - return this._deleteTag(); - } - } -}); diff --git a/core/client/app/controllers/setup.js b/core/client/app/controllers/setup.js deleted file mode 100644 index 17afc33e9f..0000000000 --- a/core/client/app/controllers/setup.js +++ /dev/null @@ -1,23 +0,0 @@ -import Ember from 'ember'; - -const { - Controller, - computed, - get, - inject: {service, controller} -} = Ember; -const {match} = computed; - -export default Controller.extend({ - appController: controller('application'), - ghostPaths: service(), - - showBackLink: match('appController.currentRouteName', /^setup\.(two|three)$/), - - backRoute: computed('appController.currentRouteName', function () { - let appController = this.get('appController'); - let currentRoute = get(appController, 'currentRouteName'); - - return currentRoute === 'setup.two' ? 'setup.one' : 'setup.two'; - }) -}); diff --git a/core/client/app/controllers/setup/three.js b/core/client/app/controllers/setup/three.js deleted file mode 100644 index efce9ccf8e..0000000000 --- a/core/client/app/controllers/setup/three.js +++ /dev/null @@ -1,236 +0,0 @@ -import Ember from 'ember'; -import DS from 'ember-data'; - -const { - Controller, - RSVP, - computed, - inject: {service, controller}, - run -} = Ember; -const {Errors} = DS; -const {alias} = computed; -const emberA = Ember.A; - -export default Controller.extend({ - notifications: service(), - two: controller('setup/two'), - - errors: Errors.create(), - hasValidated: emberA(), - users: '', - ownerEmail: alias('two.email'), - submitting: false, - - usersArray: computed('users', function () { - let errors = this.get('errors'); - let users = this.get('users').split('\n').filter(function (email) { - return email.trim().length > 0; - }); - - // remove "no users to invite" error if we have users - if (users.uniq().length > 0 && errors.get('users.length') === 1) { - if (errors.get('users.firstObject').message.match(/no users/i)) { - errors.remove('users'); - } - } - - return users.uniq(); - }), - - validUsersArray: computed('usersArray', 'ownerEmail', function () { - let ownerEmail = this.get('ownerEmail'); - - return this.get('usersArray').filter(function (user) { - return validator.isEmail(user) && user !== ownerEmail; - }); - }), - - invalidUsersArray: computed('usersArray', 'ownerEmail', function () { - let ownerEmail = this.get('ownerEmail'); - - return this.get('usersArray').reject((user) => { - return validator.isEmail(user) || user === ownerEmail; - }); - }), - - validationResult: computed('invalidUsersArray', function () { - let errors = []; - - this.get('invalidUsersArray').forEach((user) => { - errors.push({ - user, - error: 'email' - }); - }); - - if (errors.length === 0) { - // ensure we aren't highlighting fields when everything is fine - this.get('errors').clear(); - return true; - } else { - return errors; - } - }), - - validate() { - let errors = this.get('errors'); - let validationResult = this.get('validationResult'); - let property = 'users'; - - errors.clear(); - - // If property isn't in the `hasValidated` array, add it to mark that this field can show a validation result - this.get('hasValidated').addObject(property); - - if (validationResult === true) { - return true; - } - - validationResult.forEach((error) => { - // Only one error type here so far, but one day the errors might be more detailed - switch (error.error) { - case 'email': - errors.add(property, `${error.user} is not a valid email.`); - } - }); - - return false; - }, - - buttonText: computed('errors.users', 'validUsersArray', 'invalidUsersArray', function () { - let usersError = this.get('errors.users.firstObject.message'); - let validNum = this.get('validUsersArray').length; - let invalidNum = this.get('invalidUsersArray').length; - let userCount; - - if (usersError && usersError.match(/no users/i)) { - return usersError; - } - - if (invalidNum > 0) { - userCount = invalidNum === 1 ? 'email address' : 'email addresses'; - return `${invalidNum} invalid ${userCount}`; - } - - if (validNum > 0) { - userCount = validNum === 1 ? 'user' : 'users'; - userCount = `${validNum} ${userCount}`; - } else { - userCount = 'some users'; - } - - return `Invite ${userCount}`; - }), - - buttonClass: computed('validationResult', 'usersArray.length', function () { - if (this.get('validationResult') === true && this.get('usersArray.length') > 0) { - return 'btn-green'; - } else { - return 'btn-minor'; - } - }), - - authorRole: computed(function () { - return this.store.findAll('role', {reload: true}).then((roles) => { - return roles.findBy('name', 'Author'); - }); - }), - - _transitionAfterSubmission() { - if (!this._hasTransitioned) { - this._hasTransitioned = true; - this.transitionToRoute('posts.index'); - } - }, - - actions: { - validate() { - this.validate(); - }, - - invite() { - let users = this.get('usersArray'); - let notifications = this.get('notifications'); - let invitationsString, submissionTimeout; - - if (this.validate() && users.length > 0) { - this.set('submitting', true); - this._hasTransitioned = false; - - // wait for 4 seconds, otherwise transition anyway - submissionTimeout = run.later(this, function () { - this._transitionAfterSubmission(); - }, 4000); - - this.get('authorRole').then((authorRole) => { - RSVP.Promise.all( - users.map((user) => { - let newUser = this.store.createRecord('user', { - email: user, - status: 'invited', - role: authorRole - }); - - return newUser.save().then(() => { - return { - email: user, - success: newUser.get('status') === 'invited' - }; - }).catch(() => { - return { - email: user, - success: false - }; - }); - }) - ).then((invites) => { - let erroredEmails = []; - let successCount = 0; - let message; - - run.cancel(submissionTimeout); - - invites.forEach((invite) => { - if (invite.success) { - successCount++; - } else { - erroredEmails.push(invite.email); - } - }); - - if (erroredEmails.length > 0) { - invitationsString = erroredEmails.length > 1 ? ' invitations: ' : ' invitation: '; - message = `Failed to send ${erroredEmails.length} ${invitationsString}`; - message += erroredEmails.join(', '); - message += ". Please check your email configuration, see <a href=\'http://support.ghost.org/mail\' target=\'_blank\'>http://support.ghost.org/mail</a> for instructions"; - - message = Ember.String.htmlSafe(message); - notifications.showAlert(message, {type: 'error', delayed: successCount > 0, key: 'signup.send-invitations.failed'}); - } - - if (successCount > 0) { - // pluralize - invitationsString = successCount > 1 ? 'invitations' : 'invitation'; - notifications.showAlert(`${successCount} ${invitationsString} sent!`, {type: 'success', delayed: true, key: 'signup.send-invitations.success'}); - } - - this.set('submitting', false); - - run.schedule('actions', this, function () { - this.send('loadServerNotifications'); - this._transitionAfterSubmission(); - }); - }); - }); - } else if (users.length === 0) { - this.get('errors').add('users', 'No users to invite'); - } - }, - - skipInvite() { - this.send('loadServerNotifications'); - this.transitionToRoute('posts.index'); - } - } -}); diff --git a/core/client/app/controllers/setup/two.js b/core/client/app/controllers/setup/two.js deleted file mode 100644 index 5f9ca41686..0000000000 --- a/core/client/app/controllers/setup/two.js +++ /dev/null @@ -1,152 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - Controller, - RSVP: {Promise}, - inject: {service, controller}, - isArray -} = Ember; - -export default Controller.extend(ValidationEngine, { - size: 90, - blogTitle: null, - name: null, - email: '', - password: null, - image: null, - blogCreated: false, - submitting: false, - flowErrors: '', - - ghostPaths: service(), - notifications: service(), - application: controller(), - config: service(), - session: service(), - ajax: service(), - - // ValidationEngine settings - validationType: 'setup', - - /** - * Uploads the given data image, then sends the changed user image property to the server - * @param {Object} user User object, returned from the 'setup' api call - * @return {Ember.RSVP.Promise} A promise that takes care of both calls - */ - sendImage(user) { - let image = this.get('image'); - - return new Promise((resolve, reject) => { - image.formData = {}; - image.submit() - .success((response) => { - let usersUrl = this.get('ghostPaths.url').api('users', user.id.toString()); - user.image = response; - this.get('ajax').put(usersUrl, { - data: { - users: [user] - } - }).then(resolve).catch(reject); - }) - .error(reject); - }); - }, - - _handleSaveError(resp) { - this.toggleProperty('submitting'); - - if (resp && resp.errors && isArray(resp.errors)) { - this.set('flowErrors', resp.errors[0].message); - } else { - this.get('notifications').showAPIError(resp, {key: 'setup.blog-details'}); - } - }, - - _handleAuthenticationError(error) { - this.toggleProperty('submitting'); - if (error && error.errors) { - this.set('flowErrors', error.errors[0].message); - } else { - // Connection errors don't return proper status message, only req.body - this.get('notifications').showAlert('There was a problem on the server.', {type: 'error', key: 'setup.authenticate.failed'}); - } - }, - - afterAuthentication(result) { - if (this.get('image')) { - this.sendImage(result.users[0]) - .then(() => { - this.toggleProperty('submitting'); - this.transitionToRoute('setup.three'); - }).catch((resp) => { - this.toggleProperty('submitting'); - this.get('notifications').showAPIError(resp, {key: 'setup.blog-details'}); - }); - } else { - this.toggleProperty('submitting'); - this.transitionToRoute('setup.three'); - } - }, - - actions: { - preValidate(model) { - // Only triggers validation if a value has been entered, preventing empty errors on focusOut - if (this.get(model)) { - this.validate({property: model}); - } - }, - - setup() { - let setupProperties = ['blogTitle', 'name', 'email', 'password']; - let data = this.getProperties(setupProperties); - let config = this.get('config'); - let method = this.get('blogCreated') ? 'put' : 'post'; - - this.toggleProperty('submitting'); - this.set('flowErrors', ''); - - this.get('hasValidated').addObjects(setupProperties); - this.validate().then(() => { - let authUrl = this.get('ghostPaths.url').api('authentication', 'setup'); - this.get('ajax')[method](authUrl, { - data: { - setup: [{ - name: data.name, - email: data.email, - password: data.password, - blogTitle: data.blogTitle - }] - } - }).then((result) => { - config.set('blogTitle', data.blogTitle); - - // don't try to login again if we are already logged in - if (this.get('session.isAuthenticated')) { - return this.afterAuthentication(result); - } - - // Don't call the success handler, otherwise we will be redirected to admin - this.set('session.skipAuthSuccessHandler', true); - this.get('session').authenticate('authenticator:oauth2', this.get('email'), this.get('password')).then(() => { - this.set('blogCreated', true); - return this.afterAuthentication(result); - }).catch((error) => { - this._handleAuthenticationError(error); - }).finally(() => { - this.set('session.skipAuthSuccessHandler', undefined); - }); - }).catch((error) => { - this._handleSaveError(error); - }); - }).catch(() => { - this.toggleProperty('submitting'); - this.set('flowErrors', 'Please fill out the form to setup your blog.'); - }); - }, - - setImage(image) { - this.set('image', image); - } - } -}); diff --git a/core/client/app/controllers/signin.js b/core/client/app/controllers/signin.js deleted file mode 100644 index 026696b432..0000000000 --- a/core/client/app/controllers/signin.js +++ /dev/null @@ -1,114 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - $, - Controller, - inject: {service, controller}, - isArray -} = Ember; - -export default Controller.extend(ValidationEngine, { - submitting: false, - loggingIn: false, - authProperties: ['identification', 'password'], - - ghostPaths: service(), - notifications: service(), - session: service(), - application: controller(), - ajax: service(), - flowErrors: '', - - // ValidationEngine settings - validationType: 'signin', - - actions: { - authenticate() { - let model = this.get('model'); - let authStrategy = 'authenticator:oauth2'; - - // Authentication transitions to posts.index, we can leave spinner running unless there is an error - this.get('session').authenticate(authStrategy, model.get('identification'), model.get('password')).catch((error) => { - this.toggleProperty('loggingIn'); - - if (error && error.errors) { - error.errors.forEach((err) => { - err.message = err.message.htmlSafe(); - }); - - this.set('flowErrors', error.errors[0].message.string); - - if (error.errors[0].message.string.match(/user with that email/)) { - this.get('model.errors').add('identification', ''); - } - - if (error.errors[0].message.string.match(/password is incorrect/)) { - this.get('model.errors').add('password', ''); - } - } else { - // Connection errors don't return proper status message, only req.body - this.get('notifications').showAlert('There was a problem on the server.', {type: 'error', key: 'session.authenticate.failed'}); - } - }); - }, - - validateAndAuthenticate() { - this.set('flowErrors', ''); - // Manually trigger events for input fields, ensuring legacy compatibility with - // browsers and password managers that don't send proper events on autofill - $('#login').find('input').trigger('change'); - - // This is a bit dirty, but there's no other way to ensure the properties are set as well as 'signin' - this.get('hasValidated').addObjects(this.authProperties); - this.validate({property: 'signin'}).then(() => { - this.toggleProperty('loggingIn'); - this.send('authenticate'); - }).catch((error) => { - if (error) { - this.get('notifications').showAPIError(error, {key: 'signin.authenticate'}); - } else { - this.set('flowErrors', 'Please fill out the form to sign in.'); - } - }); - }, - - forgotten() { - let email = this.get('model.identification'); - let notifications = this.get('notifications'); - - this.set('flowErrors', ''); - // This is a bit dirty, but there's no other way to ensure the properties are set as well as 'forgotPassword' - this.get('hasValidated').addObject('identification'); - this.validate({property: 'forgotPassword'}).then(() => { - let forgottenUrl = this.get('ghostPaths.url').api('authentication', 'passwordreset'); - this.toggleProperty('submitting'); - - this.get('ajax').post(forgottenUrl, { - data: { - passwordreset: [{email}] - } - }).then(() => { - this.toggleProperty('submitting'); - notifications.showAlert('Please check your email for instructions.', {type: 'info', key: 'forgot-password.send.success'}); - }).catch((resp) => { - this.toggleProperty('submitting'); - - if (resp && resp.errors && isArray(resp.errors)) { - let [{message}] = resp.errors; - - this.set('flowErrors', message); - - if (message.match(/no user with that email/)) { - this.get('model.errors').add('identification', ''); - } - } else { - notifications.showAPIError(resp, {defaultErrorText: 'There was a problem with the reset, please try again.', key: 'forgot-password.send'}); - } - }); - }).catch(() => { - this.set('flowErrors', 'We need your email address to reset your password!'); - }); - } - } -}); diff --git a/core/client/app/controllers/signup.js b/core/client/app/controllers/signup.js deleted file mode 100644 index f578d672a0..0000000000 --- a/core/client/app/controllers/signup.js +++ /dev/null @@ -1,96 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - Controller, - RSVP: {Promise}, - inject: {service}, - isArray -} = Ember; - -export default Controller.extend(ValidationEngine, { - // ValidationEngine settings - validationType: 'signup', - - submitting: false, - flowErrors: '', - image: null, - - ghostPaths: service(), - config: service(), - notifications: service(), - session: service(), - ajax: service(), - - sendImage() { - let image = this.get('image'); - - this.get('session.user').then((user) => { - return new Promise((resolve, reject) => { - image.formData = {}; - image.submit() - .success((response) => { - let usersUrl = this.get('ghostPaths.url').api('users', user.id.toString()); - user.image = response; - this.get('ajax').put(usersUrl, { - data: { - users: [user] - } - }).then(resolve).catch(reject); - }) - .error(reject); - }); - }); - }, - - actions: { - signup() { - let model = this.get('model'); - let setupProperties = ['name', 'email', 'password', 'token']; - let data = model.getProperties(setupProperties); - let image = this.get('image'); - let notifications = this.get('notifications'); - - this.set('flowErrors', ''); - - this.get('hasValidated').addObjects(setupProperties); - this.validate().then(() => { - let authUrl = this.get('ghostPaths.url').api('authentication', 'invitation'); - this.toggleProperty('submitting'); - this.get('ajax').post(authUrl, { - dataType: 'json', - data: { - invitation: [{ - name: data.name, - email: data.email, - password: data.password, - token: data.token - }] - } - }).then(() => { - this.get('session').authenticate('authenticator:oauth2', this.get('model.email'), this.get('model.password')).then(() => { - if (image) { - this.sendImage(); - } - }).catch((resp) => { - notifications.showAPIError(resp, {key: 'signup.complete'}); - }); - }).catch((resp) => { - this.toggleProperty('submitting'); - - if (resp && resp.errors && isArray(resp.errors)) { - this.set('flowErrors', resp.errors[0].message); - } else { - notifications.showAPIError(resp, {key: 'signup.complete'}); - } - }); - }).catch(() => { - this.set('flowErrors', 'Please fill out the form to complete your sign-up'); - }); - }, - - setImage(image) { - this.set('image', image); - } - } -}); diff --git a/core/client/app/controllers/subscribers.js b/core/client/app/controllers/subscribers.js deleted file mode 100644 index 1c39bcf0f2..0000000000 --- a/core/client/app/controllers/subscribers.js +++ /dev/null @@ -1,162 +0,0 @@ -import Ember from 'ember'; -import Table from 'ember-light-table'; -import PaginationMixin from 'ghost/mixins/pagination'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -const { - $, - assign, - computed, - inject: {service} -} = Ember; - -export default Ember.Controller.extend(PaginationMixin, { - - queryParams: ['order', 'direction'], - order: 'created_at', - direction: 'desc', - - paginationModel: 'subscriber', - - total: 0, - table: null, - subscriberToDelete: null, - - session: service(), - - // paginationSettings is replaced by the pagination mixin so we need a - // getter/setter CP here so that we don't lose the dynamic order param - paginationSettings: computed('order', 'direction', { - get() { - let order = this.get('order'); - let direction = this.get('direction'); - - let currentSettings = this._paginationSettings || { - limit: 30 - }; - - return assign({}, currentSettings, { - order: `${order} ${direction}` - }); - }, - set(key, value) { - this._paginationSettings = value; - return value; - } - }), - - columns: computed('order', 'direction', function () { - let order = this.get('order'); - let direction = this.get('direction'); - - return [{ - label: 'Subscriber', - valuePath: 'email', - sorted: order === 'email', - ascending: direction === 'asc' - }, { - label: 'Subscription Date', - valuePath: 'createdAt', - format(value) { - return value.format('MMMM DD, YYYY'); - }, - sorted: order === 'created_at', - ascending: direction === 'asc' - }, { - label: 'Status', - valuePath: 'status', - sorted: order === 'status', - ascending: direction === 'asc' - }, { - label: '', - sortable: false, - cellComponent: 'gh-subscribers-table-delete-cell', - align: 'right' - }]; - }), - - initializeTable() { - this.set('table', new Table(this.get('columns'), this.get('subscribers'))); - }, - - // capture the total from the server any time we fetch a new page - didReceivePaginationMeta(meta) { - if (meta && meta.pagination) { - this.set('total', meta.pagination.total); - } - }, - - actions: { - loadFirstPage() { - let table = this.get('table'); - - return this._super(...arguments).then((results) => { - table.addRows(results); - return results; - }); - }, - - loadNextPage() { - let table = this.get('table'); - - return this._super(...arguments).then((results) => { - table.addRows(results); - return results; - }); - }, - - sortByColumn(column) { - let table = this.get('table'); - - if (column.sorted) { - this.setProperties({ - order: column.get('valuePath').trim().underscore(), - direction: column.ascending ? 'asc' : 'desc' - }); - table.setRows([]); - this.send('loadFirstPage'); - } - }, - - addSubscriber(subscriber) { - this.get('table').insertRowAt(0, subscriber); - this.incrementProperty('total'); - }, - - deleteSubscriber(subscriber) { - this.set('subscriberToDelete', subscriber); - }, - - confirmDeleteSubscriber() { - let subscriber = this.get('subscriberToDelete'); - - return subscriber.destroyRecord().then(() => { - this.set('subscriberToDelete', null); - this.get('table').removeRow(subscriber); - this.decrementProperty('total'); - }); - }, - - cancelDeleteSubscriber() { - this.set('subscriberToDelete', null); - }, - - reset() { - this.get('table').setRows([]); - this.send('loadFirstPage'); - }, - - exportData() { - let exportUrl = ghostPaths().url.api('subscribers/csv'); - let accessToken = this.get('session.data.authenticated.access_token'); - let downloadURL = `${exportUrl}?access_token=${accessToken}`; - let iframe = $('#iframeDownload'); - - if (iframe.length === 0) { - iframe = $('<iframe>', {id: 'iframeDownload'}).hide().appendTo('body'); - } - - iframe.attr('src', downloadURL); - } - } -}); diff --git a/core/client/app/controllers/team/index.js b/core/client/app/controllers/team/index.js deleted file mode 100644 index 9763a3d4a9..0000000000 --- a/core/client/app/controllers/team/index.js +++ /dev/null @@ -1,33 +0,0 @@ -import Ember from 'ember'; - -const { - Controller, - computed, - inject: {service} -} = Ember; -const {alias, filter} = computed; - -export default Controller.extend({ - - showInviteUserModal: false, - - users: alias('model'), - - session: service(), - - activeUsers: filter('users', function (user) { - return /^active|warn-[1-4]|locked$/.test(user.get('status')); - }), - - invitedUsers: filter('users', function (user) { - let status = user.get('status'); - - return status === 'invited' || status === 'invited-pending'; - }), - - actions: { - toggleInviteUserModal() { - this.toggleProperty('showInviteUserModal'); - } - } -}); diff --git a/core/client/app/controllers/team/user.js b/core/client/app/controllers/team/user.js deleted file mode 100644 index 580a3ffc69..0000000000 --- a/core/client/app/controllers/team/user.js +++ /dev/null @@ -1,425 +0,0 @@ -import Ember from 'ember'; -import isNumber from 'ghost/utils/isNumber'; -import boundOneWay from 'ghost/utils/bound-one-way'; -import { invoke } from 'ember-invoke-action'; - -const { - Controller, - RSVP, - computed, - inject: {service}, - run, - isArray -} = Ember; -const {alias, and, not, or, readOnly} = computed; - -export default Controller.extend({ - submitting: false, - lastPromise: null, - showDeleteUserModal: false, - showTransferOwnerModal: false, - showUploadCoverModal: false, - showUplaodImageModal: false, - _scratchFacebook: null, - _scratchTwitter: null, - - ajax: service(), - dropdown: service(), - ghostPaths: service(), - notifications: service(), - session: service(), - slugGenerator: service(), - - user: alias('model'), - currentUser: alias('session.user'), - - email: readOnly('model.email'), - slugValue: boundOneWay('model.slug'), - - isNotOwnersProfile: not('user.isOwner'), - isAdminUserOnOwnerProfile: and('currentUser.isAdmin', 'user.isOwner'), - canAssignRoles: or('currentUser.isAdmin', 'currentUser.isOwner'), - canMakeOwner: and('currentUser.isOwner', 'isNotOwnProfile', 'user.isAdmin'), - rolesDropdownIsVisible: and('isNotOwnProfile', 'canAssignRoles', 'isNotOwnersProfile'), - userActionsAreVisible: or('deleteUserActionIsVisible', 'canMakeOwner'), - - isNotOwnProfile: computed('user.id', 'currentUser.id', function () { - return this.get('user.id') !== this.get('currentUser.id'); - }), - - deleteUserActionIsVisible: computed('currentUser', 'canAssignRoles', 'user', function () { - if ((this.get('canAssignRoles') && this.get('isNotOwnProfile') && !this.get('user.isOwner')) || - (this.get('currentUser.isEditor') && (this.get('isNotOwnProfile') || - this.get('user.isAuthor')))) { - return true; - } - }), - - // duplicated in gh-user-active -- find a better home and consolidate? - userDefault: computed('ghostPaths', function () { - return `${this.get('ghostPaths.subdir')}/ghost/img/user-image.png`; - }), - - userImageBackground: computed('user.image', 'userDefault', function () { - let url = this.get('user.image') || this.get('userDefault'); - - return Ember.String.htmlSafe(`background-image: url(${url})`); - }), - // end duplicated - - coverDefault: computed('ghostPaths', function () { - return `${this.get('ghostPaths.subdir')}/ghost/img/user-cover.png`; - }), - - coverImageBackground: computed('user.cover', 'coverDefault', function () { - let url = this.get('user.cover') || this.get('coverDefault'); - - return Ember.String.htmlSafe(`background-image: url(${url})`); - }), - - coverTitle: computed('user.name', function () { - return `${this.get('user.name')}'s Cover Image`; - }), - - roles: computed(function () { - return this.store.query('role', {permissions: 'assign'}); - }), - - _deleteUser() { - if (this.get('deleteUserActionIsVisible')) { - let user = this.get('user'); - return user.destroyRecord(); - } - }, - - _deleteUserSuccess() { - this.get('notifications').closeAlerts('user.delete'); - this.store.unloadAll('post'); - this.transitionToRoute('team'); - }, - - _deleteUserFailure() { - this.get('notifications').showAlert('The user could not be deleted. Please try again.', {type: 'error', key: 'user.delete.failed'}); - }, - - actions: { - changeRole(newRole) { - this.set('model.role', newRole); - }, - - save() { - let user = this.get('user'); - let slugValue = this.get('slugValue'); - let afterUpdateSlug = this.get('lastPromise'); - let promise, - slugChanged; - - if (user.get('slug') !== slugValue) { - slugChanged = true; - user.set('slug', slugValue); - } - - this.toggleProperty('submitting'); - - promise = RSVP.resolve(afterUpdateSlug).then(() => { - return user.save({format: false}); - }).then((model) => { - let currentPath, - newPath; - - // If the user's slug has changed, change the URL and replace - // the history so refresh and back button still work - if (slugChanged) { - currentPath = window.history.state.path; - - newPath = currentPath.split('/'); - newPath[newPath.length - 2] = model.get('slug'); - newPath = newPath.join('/'); - - window.history.replaceState({path: newPath}, '', newPath); - } - - this.toggleProperty('submitting'); - this.get('notifications').closeAlerts('user.update'); - - return model; - }).catch((errors) => { - if (errors) { - this.get('notifications').showErrors(errors, {key: 'user.update'}); - } - - this.toggleProperty('submitting'); - }); - - this.set('lastPromise', promise); - return promise; - }, - - deleteUser() { - return this._deleteUser().then(() => { - this._deleteUserSuccess(); - }, () => { - this._deleteUserFailure(); - }); - }, - - toggleDeleteUserModal() { - if (this.get('deleteUserActionIsVisible')) { - this.toggleProperty('showDeleteUserModal'); - } - }, - - password() { - let user = this.get('user'); - - if (user.get('isPasswordValid')) { - user.saveNewPassword().then((model) => { - // Clear properties from view - user.setProperties({ - password: '', - newPassword: '', - ne2Password: '' - }); - - this.get('notifications').showAlert('Password updated.', {type: 'success', key: 'user.change-password.success'}); - - return model; - }).catch((errors) => { - this.get('notifications').showAPIError(errors, {key: 'user.change-password'}); - }); - } else { - // TODO: switch to in-line validation - this.get('notifications').showErrors(user.get('passwordValidationErrors'), {key: 'user.change-password'}); - } - }, - - updateSlug(newSlug) { - let afterSave = this.get('lastPromise'); - let promise; - - promise = RSVP.resolve(afterSave).then(() => { - let slug = this.get('model.slug'); - - newSlug = newSlug || slug; - newSlug = newSlug.trim(); - - // Ignore unchanged slugs or candidate slugs that are empty - if (!newSlug || slug === newSlug) { - this.set('slugValue', slug); - - return; - } - - return this.get('slugGenerator').generateSlug('user', newSlug).then((serverSlug) => { - // If after getting the sanitized and unique slug back from the API - // we end up with a slug that matches the existing slug, abort the change - if (serverSlug === slug) { - return; - } - - // Because the server transforms the candidate slug by stripping - // certain characters and appending a number onto the end of slugs - // to enforce uniqueness, there are cases where we can get back a - // candidate slug that is a duplicate of the original except for - // the trailing incrementor (e.g., this-is-a-slug and this-is-a-slug-2) - - // get the last token out of the slug candidate and see if it's a number - let slugTokens = serverSlug.split('-'); - let check = Number(slugTokens.pop()); - - // if the candidate slug is the same as the existing slug except - // for the incrementor then the existing slug should be used - if (isNumber(check) && check > 0) { - if (slug === slugTokens.join('-') && serverSlug !== newSlug) { - this.set('slugValue', slug); - - return; - } - } - - this.set('slugValue', serverSlug); - }); - }); - - this.set('lastPromise', promise); - }, - - validateFacebookUrl() { - let newUrl = this.get('_scratchFacebook'); - let oldUrl = this.get('user.facebook'); - let errMessage = ''; - - if (newUrl === '') { - // Clear out the Facebook url - this.set('user.facebook', ''); - this.get('user.errors').remove('facebook'); - return; - } - - // _scratchFacebook will be null unless the user has input something - if (!newUrl) { - newUrl = oldUrl; - } - - // If new url didn't change, exit - if (newUrl === oldUrl) { - this.get('user.errors').remove('facebook'); - return; - } - - // TODO: put the validation here into a validator - if (newUrl.match(/(?:facebook\.com\/)(\S+)/) || newUrl.match(/([a-z\d\.]+)/i)) { - let username = []; - - if (newUrl.match(/(?:facebook\.com\/)(\S+)/)) { - [ , username ] = newUrl.match(/(?:facebook\.com\/)(\S+)/); - } else { - [ , username ] = newUrl.match(/(?:https\:\/\/|http\:\/\/)?(?:www\.)?(?:\w+\.\w+\/+)?(\S+)/mi); - } - - // check if we have a /page/username or without - if (username.match(/^(?:\/)?(pages?\/\S+)/mi)) { - // we got a page url, now save the username without the / in the beginning - - [ , username ] = username.match(/^(?:\/)?(pages?\/\S+)/mi); - } else if (username.match(/^(http|www)|(\/)/) || !username.match(/^([a-z\d\.]{5,50})$/mi)) { - errMessage = !username.match(/^([a-z\d\.]{5,50})$/mi) ? 'Your Username is not a valid Facebook Username' : 'The URL must be in a format like https://www.facebook.com/yourUsername'; - - this.get('user.errors').add('facebook', errMessage); - this.get('user.hasValidated').pushObject('facebook'); - return; - } - - newUrl = `https://www.facebook.com/${username}`; - this.set('user.facebook', newUrl); - - this.get('user.errors').remove('facebook'); - this.get('user.hasValidated').pushObject('facebook'); - - // User input is validated - invoke(this, 'save').then(() => { - // necessary to update the value in the input field - this.set('user.facebook', ''); - run.schedule('afterRender', this, function () { - this.set('user.facebook', newUrl); - }); - }); - } else { - errMessage = 'The URL must be in a format like ' + - 'https://www.facebook.com/yourUsername'; - this.get('user.errors').add('facebook', errMessage); - this.get('user.hasValidated').pushObject('facebook'); - return; - } - }, - - validateTwitterUrl() { - let newUrl = this.get('_scratchTwitter'); - let oldUrl = this.get('user.twitter'); - let errMessage = ''; - - if (newUrl === '') { - // Clear out the Twitter url - this.set('user.twitter', ''); - this.get('user.errors').remove('twitter'); - return; - } - - // _scratchTwitter will be null unless the user has input something - if (!newUrl) { - newUrl = oldUrl; - } - - // If new url didn't change, exit - if (newUrl === oldUrl) { - this.get('user.errors').remove('twitter'); - return; - } - - // TODO: put the validation here into a validator - if (newUrl.match(/(?:twitter\.com\/)(\S+)/) || newUrl.match(/([a-z\d\.]+)/i)) { - let username = []; - - if (newUrl.match(/(?:twitter\.com\/)(\S+)/)) { - [ , username] = newUrl.match(/(?:twitter\.com\/)(\S+)/); - } else { - [username] = newUrl.match(/([^/]+)\/?$/mi); - } - - // check if username starts with http or www and show error if so - if (username.match(/^(http|www)|(\/)/) || !username.match(/^[a-z\d\.\_]{1,15}$/mi)) { - errMessage = !username.match(/^[a-z\d\.\_]{1,15}$/mi) ? 'Your Username is not a valid Twitter Username' : 'The URL must be in a format like https://twitter.com/yourUsername'; - - this.get('user.errors').add('twitter', errMessage); - this.get('user.hasValidated').pushObject('twitter'); - return; - } - - newUrl = `https://twitter.com/${username}`; - this.set('user.twitter', newUrl); - - this.get('user.errors').remove('twitter'); - this.get('user.hasValidated').pushObject('twitter'); - - // User input is validated - invoke(this, 'save').then(() => { - // necessary to update the value in the input field - this.set('user.twitter', ''); - run.schedule('afterRender', this, function () { - this.set('user.twitter', newUrl); - }); - }); - } else { - errMessage = 'The URL must be in a format like ' + - 'https://twitter.com/yourUsername'; - this.get('user.errors').add('twitter', errMessage); - this.get('user.hasValidated').pushObject('twitter'); - return; - } - }, - - transferOwnership() { - let user = this.get('user'); - let url = this.get('ghostPaths.url').api('users', 'owner'); - - this.get('dropdown').closeDropdowns(); - - return this.get('ajax').put(url, { - data: { - owner: [{ - id: user.get('id') - }] - } - }).then((response) => { - // manually update the roles for the users that just changed roles - // because store.pushPayload is not working with embedded relations - if (response && isArray(response.users)) { - response.users.forEach((userJSON) => { - let user = this.store.peekRecord('user', userJSON.id); - let role = this.store.peekRecord('role', userJSON.roles[0].id); - - user.set('role', role); - }); - } - - this.get('notifications').showAlert(`Ownership successfully transferred to ${user.get('name')}`, {type: 'success', key: 'owner.transfer.success'}); - }).catch((error) => { - this.get('notifications').showAPIError(error, {key: 'owner.transfer'}); - }); - }, - - toggleTransferOwnerModal() { - if (this.get('canMakeOwner')) { - this.toggleProperty('showTransferOwnerModal'); - } - }, - - toggleUploadCoverModal() { - this.toggleProperty('showUploadCoverModal'); - }, - - toggleUploadImageModal() { - this.toggleProperty('showUploadImageModal'); - } - } -}); diff --git a/core/client/app/helpers/gh-count-characters.js b/core/client/app/helpers/gh-count-characters.js deleted file mode 100644 index 7fc9bccbfe..0000000000 --- a/core/client/app/helpers/gh-count-characters.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - if (!params || !params.length) { - return; - } - - let el = document.createElement('span'); - let content = params[0] || ''; - let {length} = content; - - el.className = 'word-count'; - - if (length > 180) { - el.style.color = '#E25440'; - } else { - el.style.color = '#9E9D95'; - } - - el.innerHTML = 200 - length; - - return Ember.String.htmlSafe(el.outerHTML); -}); diff --git a/core/client/app/helpers/gh-count-down-characters.js b/core/client/app/helpers/gh-count-down-characters.js deleted file mode 100644 index dd937fb655..0000000000 --- a/core/client/app/helpers/gh-count-down-characters.js +++ /dev/null @@ -1,28 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - if (!params || params.length < 2) { - return; - } - - let el = document.createElement('span'); - let [content, maxCharacters] = params; - let length; - - content = content || ''; - length = content.length; - - el.className = 'word-count'; - - if (length > maxCharacters) { - el.style.color = '#E25440'; - } else { - el.style.color = '#9FBB58'; - } - - el.innerHTML = length; - - return Ember.String.htmlSafe(el.outerHTML); -}); diff --git a/core/client/app/helpers/gh-count-words.js b/core/client/app/helpers/gh-count-words.js deleted file mode 100644 index 2d33cea1af..0000000000 --- a/core/client/app/helpers/gh-count-words.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; -import counter from 'ghost/utils/word-count'; - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - if (!params || !params.length) { - return; - } - - let markdown = params[0] || ''; - - if (/^\s*$/.test(markdown)) { - return '0 words'; - } - - let count = counter(markdown); - - return count + (count === 1 ? ' word' : ' words'); -}); diff --git a/core/client/app/helpers/gh-format-html.js b/core/client/app/helpers/gh-format-html.js deleted file mode 100644 index c01945ebc5..0000000000 --- a/core/client/app/helpers/gh-format-html.js +++ /dev/null @@ -1,26 +0,0 @@ -/* global html_sanitize*/ -import Ember from 'ember'; -import cajaSanitizers from 'ghost/utils/caja-sanitizers'; - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - if (!params || !params.length) { - return; - } - - let escapedhtml = params[0] || ''; - - // replace script and iFrame - escapedhtml = escapedhtml.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, - '<pre class="js-embed-placeholder">Embedded JavaScript</pre>'); - escapedhtml = escapedhtml.replace(/<iframe\b[^<]*(?:(?!<\/iframe>)<[^<]*)*<\/iframe>/gi, - '<pre class="iframe-embed-placeholder">Embedded iFrame</pre>'); - - // sanitize HTML - // jscs:disable requireCamelCaseOrUpperCaseIdentifiers - escapedhtml = html_sanitize(escapedhtml, cajaSanitizers.url, cajaSanitizers.id); - // jscs:enable requireCamelCaseOrUpperCaseIdentifiers - - return Ember.String.htmlSafe(escapedhtml); -}); diff --git a/core/client/app/helpers/gh-format-markdown.js b/core/client/app/helpers/gh-format-markdown.js deleted file mode 100644 index 7e09cddbf8..0000000000 --- a/core/client/app/helpers/gh-format-markdown.js +++ /dev/null @@ -1,34 +0,0 @@ -/* global Showdown, html_sanitize*/ -import Ember from 'ember'; -import cajaSanitizers from 'ghost/utils/caja-sanitizers'; - -const {Helper} = Ember; - -let showdown = new Showdown.converter({extensions: ['ghostimagepreview', 'ghostgfm', 'footnotes', 'highlight']}); - -export function formatMarkdown(params) { - if (!params || !params.length) { - return; - } - - let markdown = params[0] || ''; - let escapedhtml = ''; - - // convert markdown to HTML - escapedhtml = showdown.makeHtml(markdown); - - // replace script and iFrame - escapedhtml = escapedhtml.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, - '<pre class="js-embed-placeholder">Embedded JavaScript</pre>'); - escapedhtml = escapedhtml.replace(/<iframe\b[^<]*(?:(?!<\/iframe>)<[^<]*)*<\/iframe>/gi, - '<pre class="iframe-embed-placeholder">Embedded iFrame</pre>'); - - // sanitize html - // jscs:disable requireCamelCaseOrUpperCaseIdentifiers - escapedhtml = html_sanitize(escapedhtml, cajaSanitizers.url, cajaSanitizers.id); - // jscs:enable requireCamelCaseOrUpperCaseIdentifiers - - return Ember.String.htmlSafe(escapedhtml); -} - -export default Helper.helper(formatMarkdown); diff --git a/core/client/app/helpers/gh-format-timeago.js b/core/client/app/helpers/gh-format-timeago.js deleted file mode 100644 index 97bf5d18c4..0000000000 --- a/core/client/app/helpers/gh-format-timeago.js +++ /dev/null @@ -1,16 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - if (!params || !params.length) { - return; - } - - let [timeago] = params; - - return moment(timeago).fromNow(); - // stefanpenner says cool for small number of timeagos. - // For large numbers moment sucks => single Ember.Object based clock better - // https://github.com/manuelmitasch/ghost-admin-ember-demo/commit/fba3ab0a59238290c85d4fa0d7c6ed1be2a8a82e#commitcomment-5396524 -}); diff --git a/core/client/app/helpers/gh-path.js b/core/client/app/helpers/gh-path.js deleted file mode 100644 index 3e9f416a81..0000000000 --- a/core/client/app/helpers/gh-path.js +++ /dev/null @@ -1,55 +0,0 @@ -import Ember from 'ember'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -// Handlebars Helper {{gh-path}} -// Usage: Assume 'http://www.myghostblog.org/myblog/' -// {{gh-path}} or {{gh-path 'blog'}} for Ghost's root (/myblog/) -// {{gh-path 'admin'}} for Ghost's admin root (/myblog/ghost/) -// {{gh-path 'api'}} for Ghost's api root (/myblog/ghost/api/v0.1/) -// {{gh-path 'admin' '/assets/hi.png'}} for resolved url (/myblog/ghost/assets/hi.png) - -const {Helper} = Ember; - -export default Helper.helper(function (params) { - let paths = ghostPaths(); - let [path, url] = params; - let base; - - if (!path) { - path = 'blog'; - } - - if (!/^(blog|admin|api)$/.test(path)) { - url = path; - path = 'blog'; - } - - switch (path.toString()) { - case 'blog': - base = paths.blogRoot; - break; - case 'admin': - base = paths.adminRoot; - break; - case 'api': - base = paths.apiRoot; - break; - default: - base = paths.blogRoot; - break; - } - - // handle leading and trailing slashes - - base = base[base.length - 1] !== '/' ? `${base}/` : base; - - if (url && url.length > 0) { - if (url[0] === '/') { - url = url.substr(1); - } - - base = base + url; - } - - return Ember.String.htmlSafe(base); -}); diff --git a/core/client/app/helpers/gh-user-can-admin.js b/core/client/app/helpers/gh-user-can-admin.js deleted file mode 100644 index bc611addb0..0000000000 --- a/core/client/app/helpers/gh-user-can-admin.js +++ /dev/null @@ -1,16 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -// Handlebars Helper {{gh-user-can-admin}} group users by admin and owner using if, or group them author using unless -// Usage: call helper as with aparameter of session.user -// e.g - {{#if (gh-user-can-admin session.user)}} 'block content' {{/if}} -// @param session.user - -export function ghUserCanAdmin(params) { - return !!(params[0].get('isOwner') || params[0].get('isAdmin')); -} - -export default Helper.helper(function (params) { - return ghUserCanAdmin(params); -}); diff --git a/core/client/app/helpers/highlighted-text.js b/core/client/app/helpers/highlighted-text.js deleted file mode 100644 index d3562c7356..0000000000 --- a/core/client/app/helpers/highlighted-text.js +++ /dev/null @@ -1,7 +0,0 @@ -import Ember from 'ember'; - -export function highlightedText([text, termToHighlight]) { - return Ember.String.htmlSafe(text.replace(new RegExp(termToHighlight, 'ig'), '<span class="highlight">$&</span>')); -} - -export default Ember.Helper.helper(highlightedText); diff --git a/core/client/app/helpers/is-equal.js b/core/client/app/helpers/is-equal.js deleted file mode 100644 index a3f28e703d..0000000000 --- a/core/client/app/helpers/is-equal.js +++ /dev/null @@ -1,13 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -export function isEqual(params) { - let [lhs, rhs] = params; - - return lhs === rhs; -} - -export default Helper.helper(function (params) { - return isEqual(params); -}); diff --git a/core/client/app/helpers/is-not.js b/core/client/app/helpers/is-not.js deleted file mode 100644 index e434481716..0000000000 --- a/core/client/app/helpers/is-not.js +++ /dev/null @@ -1,11 +0,0 @@ -import Ember from 'ember'; - -const {Helper} = Ember; - -export function isNot(params) { - return !params; -} - -export default Helper.helper(function (params) { - return isNot(params); -}); diff --git a/core/client/app/html/apps.html b/core/client/app/html/apps.html deleted file mode 100644 index ec8783bdd8..0000000000 --- a/core/client/app/html/apps.html +++ /dev/null @@ -1,305 +0,0 @@ -<!DOCTYPE html> -<html lang="en"> - -<head> - <meta charset="UTF-8"> - <title>Apps</title> - <link rel="stylesheet" type="text/css" href="https://fonts.googleapis.com/css?family=Open+Sans:400,300,700" /> - <link rel="stylesheet" href="file:///Users/John/Sites/Ghost/core/built/assets/vendor.css" /> - <link rel="stylesheet" href="file:///Users/John/Sites/Ghost/core/built/assets/ghost.css" /> -</head> - -<body class="ember-application settings settings-view-general" data-apps="false" data-filestorage="true" data-blogurl="http://127.0.0.1:2368"> - -<div id="ember725" class="ember-view gh-app"><a class="sr-only sr-only-focusable" href="#gh-main">Skip to main content</a> - -<aside id="ember749" class="ember-view gh-alerts"><!----></aside> - -<div class="gh-viewport"> - - <nav class="gh-nav"> -<header id="ember783" class="ember-view gh-nav-menu closed" role="button"> <div class="gh-nav-menu-icon" style="background-image: url(https://s3.amazonaws.com/f.cl.ly/items/3I0g431b2b3q00253K1V/d16dc430c9c4f5c09d6ca09be3e5c72fdb21c1ac.png)"></div> - <div class="gh-nav-menu-details"> - <div class="gh-nav-menu-details-blog">Ghost</div> - <div class="gh-nav-menu-details-user">John Thing</div> - </div> - <i class="icon-arrow"></i> -</header><div id="ember792" class="ember-view ghost-dropdown dropdown closed fade-out"> <ul class="dropdown-menu dropdown-triangle-top js-user-menu-dropdown-menu" role="menu" style="right:-50%;left:auto;margin-right:40px"> - <li role="presentation"><a id="ember810" class="ember-view gh-nav-menu-about dropdown-item js-nav-item" href="/ghost/settings/about/" tabindex="-1"><i class="icon-gh"></i> About Ghost</a></li> - <li class="divider"></li> - <li role="presentation"><a id="ember811" class="ember-view dropdown-item user-menu-profile js-nav-item" href="/ghost/settings/users/john/" tabindex="-1"><i class="icon-user"></i> Your Profile</a></li> - <li class="divider"></li> - <li role="presentation"><a id="ember812" class="ember-view dropdown-item user-menu-signout" href="/ghost/signout/" tabindex="-1"><i class="icon-signout"></i> Sign Out</a></li> - </ul> -</div> <section class="gh-nav-body"> - <section class="gh-nav-search"> - <input class="gh-nav-search-input gh-input" type="text" placeholder="Search"> - <button class="gh-nav-search-button"><i class="icon-search"></i><span class="sr-only">Search</span></button> - </section> - <ul class="gh-nav-list gh-nav-main"> - <li><a id="ember796" class="ember-view gh-nav-main-editor" href="/ghost/editor/"><i class="icon-pen"></i>New Post</a></li> - <li><a id="ember797" class="ember-view gh-nav-main-content" href="/ghost/"><i class="icon-content"></i>Content</a></li> - <li><a id="ember798" class="ember-view gh-nav-main-users" href="/ghost/settings/users/"><i class="icon-team"></i>Team</a></li> - </ul> - <ul class="gh-nav-list gh-nav-settings"> - <li class="gh-nav-list-h">Settings</li> - <li><a id="ember799" class="ember-view gh-nav-settings-general" href="/ghost/settings/general/"><i class="icon-settings"></i>General</a></li> - <li><a id="ember800" class="ember-view gh-nav-settings-navigation" href="/ghost/settings/navigation/"><i class="icon-compass"></i>Navigation</a></li> - <li><a id="ember801" class="ember-view gh-nav-settings-tags" href="/ghost/settings/tags/"><i class="icon-tag"></i>Tags</a></li> - <li><a id="ember802" class="ember-view gh-nav-settings-code-injection" href="/ghost/settings/code-injection/"><i class="icon-code"></i>Code Injection</a></li> - <li><a id="ember803" class="ember-view gh-nav-settings-labs" href="/ghost/settings/labs/"><i class="icon-labs"></i>Labs</a></li> - <li><a id="ember804" class="ember-view gh-nav-settings-apps active" href="/ghost/settings/apps/"><i class="icon-apps"></i>Apps</a></li> - </ul> - </section> - <footer class="gh-nav-footer"> - <div class="gh-autonav-toggle"> - <i class="icon-minimise"></i> - </div> - <a class="gh-nav-footer-sitelink" target="_blank" href="http://localhost:2368/">View blog <i class="icon-external"></i></a> - <div class="gh-help-menu"> -<div id="ember804" class="ember-view closed" role="button"> <div class="gh-help-button"> - <i class="icon-question"><span class="hidden">Help</span></i> - </div> -</div><div id="ember805" class="ember-view ghost-dropdown dropdown closed fade-out"> <ul class="dropdown-menu dropdown-triangle-bottom" role="menu"> - <li role="presentation"><a class="dropdown-item help-menu-support" role="menuitem" tabindex="-1" href="http://support.ghost.org/" target="_blank"><i class="icon-ambulance"></i> Support Center</a></li> - <li role="presentation"><a class="dropdown-item help-menu-tweet" role="menuitem" tabindex="-1" href="https://twitter.com/intent/tweet?text=%40TryGhost+Hi%21+Can+you+help+me+with+&related=TryGhost" target="_blank" onclick="window.open(this.href, 'twitter-share', 'width=550,height=235');return false;"><i class="icon-twitter"></i> Tweet @TryGhost!</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-how-to" role="menuitem" tabindex="-1" href="http://support.ghost.org/how-to-use-ghost/" target="_blank"><i class="icon-book"></i> How to Use Ghost</a></li> - <li role="presentation"><a class="dropdown-item help-menu-markdown" role="menuitem" tabindex="-1" href="" data-ember-action="1036"><i class="icon-markdown"></i> Markdown Help</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-wishlist" role="menuitem" tabindex="-1" href="http://ideas.ghost.org/" target="_blank"><i class="icon-idea"></i> Wishlist</a></li> - </ul> -</div> </div> - </footer> -</nav> - - - <main id="gh-main" class="gh-main" role="main" data-notification-count="0"> - <section id="ember1308" class="ember-view gh-view js-settings-content"><header class="view-header"> - <h2 class="view-title">Apps</h2> - <section class="view-actions"> - <input id="blog-title" class="gh-input package-filter" type="text" name="general[title]" placeholder="Filter apps by name"> - - </section> -</header> - -<section class="view-content"> - - <div class="package-grid package-grid-apps"> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-enable package-controls-button" href="#"><i class="icon-play"></i>Enable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-app" data-href="/app/"> - <div class="package-card-content"> - <figure class="package-card-app-icon" style="background-image:url(https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg)"></figure> - <div class="package-card-meta"> - <h3 class="package-card-app-title">Ghost</h3> - <div class="package-card-stats"> - <div class="package-downloads"> - <i class="icon-download"></i> - <span class="package-download-count">14,301</span> - </div> - </div> - <p class="package-card-app-desc">A UI theme that offers endless possibilities: one Atom, many Isotopes ;) Isotope adapts to match any syntax theme.</p> - </div> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-settings package-controls-button" href="#"><i class="icon-settings"></i>Settings</a> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - <a class="package-disable package-controls-button" href="#"><i class="icon-pause"></i>Disable</a> - </div> - </footer> - </article> - </div> - - </div> - -</section> - </main> - - <aside id="ember766" class="ember-view gh-notifications"><!----></aside> - - <!----> - <!----> - -</div></div> - -</body> -</html> diff --git a/core/client/app/html/permalinks.html b/core/client/app/html/permalinks.html deleted file mode 100644 index f4c3c3c58b..0000000000 --- a/core/client/app/html/permalinks.html +++ /dev/null @@ -1,250 +0,0 @@ -<!DOCTYPE html> -<html lang="en"> - -<head> - <meta charset="UTF-8"> - <title>Permalinks</title> - <link rel="stylesheet" href="../assets/css/ghost.min.css"> - - <!-- Styles not needed for this. Just to get around various bits that Ember handles for us. --> - <style> - @media (max-width: 900px) { - .settings-nav { - top: 0; - left: -100%; - } - .settings-content { - margin-left: 0; - } - } - </style> -</head> - -<body class="ember-application settings settings-view-general" data-apps="false" data-filestorage="true" data-blogurl="http://127.0.0.1:2368"> - - <div id="container"><a class="sr-only sr-only-focusable" href="#gh-main">Skip to main content</a> - - <nav class="global-nav" role="navigation"> - - <a class="nav-item ghost-logo" href="/" title="/"> - <div class="nav-label"><i class="icon-ghost"></i> - <span>Visit blog</span> - </div> - </a> - - <a class="nav-item nav-content js-nav-item" href="/ghost/"> - <div class="nav-label"><i class="icon-content"></i> Content</div> - </a> - - <a class="nav-item nav-new js-nav-item" href="/ghost/editor/"> - <div class="nav-label"><i class="icon-add"></i> New Post</div> - </a> - - - <a class="nav-item nav-settings js-nav-item active" href="/ghost/settings/"> - <div class="nav-label"><i class="icon-settings2"></i> Settings</div> - </a> - - - - - <div class="nav-item user-menu" data-href="#"> - <div class=" nav-label"> - <div class="image"> - <img src="../../../core/shared/img/user-image.png" alt="Paul Davis's profile picture"> - </div> - <div class="name"> - Paul Davis <i class="icon-chevron-down"></i> - <small>Profile & Settings</small> - </div> - </div> - </div> - - </nav> - - <div class="nav-cover js-nav-cover"></div> - - <main id="gh-main" class="viewport" role="main" data-notification-count="0" data-bindattr-448="448"> - <aside class="notifications top"> - </aside> - <aside class="notifications bottom"> - </aside> - <div> - <header class="page-header"> - <button class="menu-button js-menu-button" data-ember-action="1221"> - <span class="sr-only">Menu</span> - </button> - <h2 class="page-title">Settings</h2> - </header> - - <div class="page-content"> - <nav class="settings-nav js-settings-nav"> - <ul> - <li class="settings-nav-general icon-settings active"><a class="active" href="/ghost/settings/general/">General</a> - </li> - <li class="settings-nav-users icon-users"><a href="/ghost/settings/users/">Users</a> - </li> - <li class="settings-nav-about icon-pacman"><a href="/ghost/settings/about/">About</a> - </li> - </ul> - </nav> - - <section class="settings-content js-settings-content fade-in"> - - <header class="settings-view-header"> - <a class="btn btn-default btn-back active" href="/ghost/settings/">Back</a> - <h2 class="page-title">General</h2> - <section class="page-actions"> - <button type="button" class="btn btn-blue" data-ember-action="1271">Save</button> - </section> - </header> - - <section class="content settings-general"> - <form id="settings-general" novalidate="novalidate"> - - <div class="form-group"> - <label for="blog-title">Blog Title</label> - <input id="blog-title" class="" type="text" name="general[title]" value="Paul Davis"> - <p>The name of your blog</p> - </div> - - <div class="form-group"> - <label for="blog-logo">Blog Logo</label> - <button type="button" class="btn btn-green js-modal-logo" data-ember-action="1277">Upload Image</button> - <p>Display a sexy logo for your publication</p> - </div> - - <fieldset> - - <div class="form-group"> - <label for="postsPerPage">Posts per page</label> - <input id="postsPerPage" class="" type="number" name="general[postsPerPage]" min="1" max="1000" value="5"> - <p>How many posts should be displayed on each page</p> - </div> - - <div class="form-group"> - <label for="blog-permalinks">Permalinks</label> - <label class="permalink-input-wrapper input" tabindex="0"> - - <div class="permalink-fake-input"> - <span class="permalink-domain">http://myblog.ghost.io</span> - <span class="permalink-parameter label label-default">journal</span> - <span class="permalink-parameter label label-default">year</span> - <span class="permalink-parameter label label-default">hello</span> - <span class="permalink-parameter label label-default">agaian</span> - <span class="permalink-parameter label label-default">this</span> - <span class="permalink-parameter label label-default">is</span> - <span class="permalink-parameter label label-default">nice</span> - <span class="permalink-parameter label label-default">yo</span> - <span class="permalink-parameter label label-default">yo</span> - <span class="permalink-parameter label label-default">yo</span> - <input id="blog-permalinks" class="permalink-input" type="text" name="general[permalinks]"> - </div> - - <div class="popover"> - <button class="permalink-help button icon-question hover-me"> - <span class="hidden">URL Structure Formatting</span> - </button> - <div class="popover-item closed popover-triangle-bottom"> - <div class="popover-title">URL Structure Formatting</div> - <div class="popover-desc">You can use dynamic variables in this field.</div> - <div class="popover-body"> - <p> - <b>%t</b> - The title of your post (or page)<br> - <b>%c</b> - The tag which your post is categorised in<br> - <b>%y</b> - The year your post was published<br> - <b>%m</b> - The month your post was published<br> - <b>%d</b> - The day your post was published - </p> - </div> - </div> - </div> - - </label> - <p>The default URL structure for your blog</p> - </div> - - <div class="form-group for-select"> - <label for="activeTheme">Theme</label> - <span class="gh-select" data-bindattr-1286="1286" tabindex="0" data-select-text="Casper - 1.1.0"> - <select id="activeTheme" class="ember-select" name="general[activeTheme]"> - <option value="casper">Casper - 1.1.0</option> - <option value="hayleybrowne">HJB - 0.5</option> - <option value="mono">Mono - 0.1.0</option> - <option value="pad-gs">PAD-GS - 0.8</option> - <option value="pauladamdavis">PAD - 0.8</option> - </select> - </span> - <p>Select a theme for your blog</p> - </div> - </fieldset> - </form> - </section> - </section> - </div> - </div> - </main> - - </div> - -</body> -</html> - -<script src="../../../bower_components/jquery/dist/jquery.js"></script> -<script> - $(function(){ - - $('.permalink-input-wrapper').on('click', function(){ - $(this).addClass('focus'); - $('#blog-permalinks').focus(); - }); - - $('#blog-permalinks').on('blur', function(){ - $('.permalink-input-wrapper').removeClass('focus'); - }); - - $('.hover-me').hover(function(){ - $(this).next('.popover-item').addClass('open fade-in').removeClass('closed fade-out'); - }, function(){ - $(this).next('.popover-item').removeClass('fade-in').addClass('fade-out'); - $(this).next('.popover-item').on('animationend webkitAnimationEnd oanimationend MSAnimationEnd', function (event) { - if (event.originalEvent.animationName === 'fade-out') { - $(this).removeClass('open fade-out').addClass('closed'); - } - }); - }); - - - function convertToSlug(Text){ - return Text.toLowerCase().replace(/[^\w ]+/g,'').replace(/ +/g,'-').trim(); - } - - $('#blog-permalinks').on('keyup', function(e){ - // 188 = comma - // Append the new parameter when pressing comma - if (e.keyCode === 188) { - var input_val = convertToSlug($(this).val()); - - var the_param = '<span class="permalink-parameter label label-default">' + input_val + '</span>'; - - // If a parameter exists, append after it. - // Else append after the domain - if ($('.permalink-parameter').length) { - $('.permalink-parameter:last').after(the_param); - } else { - $('.permalink-domain').after(the_param); - } - - - $(this).val(''); - } - - // 8 = backspace - // When backspace is pressed AND the input is empty, delete the last parameter - if (e.keyCode === 8 && $(this).val() == '') { - $('.permalink-parameter:last').remove(); - } - }); - - }); -</script> \ No newline at end of file diff --git a/core/client/app/html/themes.html b/core/client/app/html/themes.html deleted file mode 100644 index b696c1c381..0000000000 --- a/core/client/app/html/themes.html +++ /dev/null @@ -1,273 +0,0 @@ -<!DOCTYPE html> -<html lang="en"> - -<head> - <meta charset="UTF-8"> - <title>Themes</title> - <link rel="stylesheet" type="text/css" href="https://fonts.googleapis.com/css?family=Open+Sans:400,300,700" /> - <link rel="stylesheet" href="file:///Users/John/Sites/Ghost/core/built/assets/vendor.css" /> - <link rel="stylesheet" href="file:///Users/John/Sites/Ghost/core/built/assets/ghost.css" /> -</head> - -<body class="ember-application settings settings-view-general" data-apps="false" data-filestorage="true" data-blogurl="http://127.0.0.1:2368"> - -<div id="ember725" class="ember-view gh-app"><a class="sr-only sr-only-focusable" href="#gh-main">Skip to main content</a> - -<aside id="ember749" class="ember-view gh-alerts"><!----></aside> - -<div class="gh-viewport"> - - <nav class="gh-nav"> -<header id="ember783" class="ember-view gh-nav-menu closed" role="button"> <div class="gh-nav-menu-icon" style="background-image: url(https://s3.amazonaws.com/f.cl.ly/items/3I0g431b2b3q00253K1V/d16dc430c9c4f5c09d6ca09be3e5c72fdb21c1ac.png)"></div> - <div class="gh-nav-menu-details"> - <div class="gh-nav-menu-details-blog">Ghost</div> - <div class="gh-nav-menu-details-user">John Thing</div> - </div> - <i class="icon-arrow"></i> -</header><div id="ember792" class="ember-view ghost-dropdown dropdown closed fade-out"> <ul class="dropdown-menu dropdown-triangle-top js-user-menu-dropdown-menu" role="menu" style="right:-50%;left:auto;margin-right:40px"> - <li role="presentation"><a id="ember810" class="ember-view gh-nav-menu-about dropdown-item js-nav-item" href="/ghost/settings/about/" tabindex="-1"><i class="icon-gh"></i> About Ghost</a></li> - <li class="divider"></li> - <li role="presentation"><a id="ember811" class="ember-view dropdown-item user-menu-profile js-nav-item" href="/ghost/settings/users/john/" tabindex="-1"><i class="icon-user"></i> Your Profile</a></li> - <li class="divider"></li> - <li role="presentation"><a id="ember812" class="ember-view dropdown-item user-menu-signout" href="/ghost/signout/" tabindex="-1"><i class="icon-signout"></i> Sign Out</a></li> - </ul> -</div> <section class="gh-nav-body"> - <section class="gh-nav-search"> - <input class="gh-nav-search-input gh-input" type="text" placeholder="Search"> - <button class="gh-nav-search-button"><i class="icon-search"></i><span class="sr-only">Search</span></button> - </section> - <ul class="gh-nav-list gh-nav-main"> - <li><a id="ember796" class="ember-view gh-nav-main-editor" href="/ghost/editor/"><i class="icon-pen"></i>New Post</a></li> - <li><a id="ember797" class="ember-view gh-nav-main-content" href="/ghost/"><i class="icon-content"></i>Content</a></li> - <li><a id="ember798" class="ember-view gh-nav-main-users" href="/ghost/settings/users/"><i class="icon-team"></i>Team</a></li> - </ul> - <ul class="gh-nav-list gh-nav-settings"> - <li class="gh-nav-list-h">Settings</li> - <li><a id="ember799" class="ember-view gh-nav-settings-general" href="/ghost/settings/general/"><i class="icon-settings"></i>General</a></li> - <li><a id="ember800" class="ember-view gh-nav-settings-navigation" href="/ghost/settings/navigation/"><i class="icon-compass"></i>Navigation</a></li> - <li><a id="ember801" class="ember-view gh-nav-settings-tags" href="/ghost/settings/tags/"><i class="icon-tag"></i>Tags</a></li> - <li><a id="ember802" class="ember-view gh-nav-settings-code-injection" href="/ghost/settings/code-injection/"><i class="icon-code"></i>Code Injection</a></li> - <li><a id="ember803" class="ember-view gh-nav-settings-labs" href="/ghost/settings/labs/"><i class="icon-labs"></i>Labs</a></li> - <li><a id="ember804" class="ember-view gh-nav-settings-apps active" href="/ghost/settings/apps/"><i class="icon-design"></i>Themes</a></li> - </ul> - </section> - <footer class="gh-nav-footer"> - <div class="gh-autonav-toggle"> - <i class="icon-minimise"></i> - </div> - <a class="gh-nav-footer-sitelink" target="_blank" href="http://localhost:2368/">View blog <i class="icon-external"></i></a> - <div class="gh-help-menu"> -<div id="ember804" class="ember-view closed" role="button"> <div class="gh-help-button"> - <i class="icon-question"><span class="hidden">Help</span></i> - </div> -</div><div id="ember805" class="ember-view ghost-dropdown dropdown closed fade-out"> <ul class="dropdown-menu dropdown-triangle-bottom" role="menu"> - <li role="presentation"><a class="dropdown-item help-menu-support" role="menuitem" tabindex="-1" href="http://support.ghost.org/" target="_blank"><i class="icon-ambulance"></i> Support Center</a></li> - <li role="presentation"><a class="dropdown-item help-menu-tweet" role="menuitem" tabindex="-1" href="https://twitter.com/intent/tweet?text=%40TryGhost+Hi%21+Can+you+help+me+with+&related=TryGhost" target="_blank" onclick="window.open(this.href, 'twitter-share', 'width=550,height=235');return false;"><i class="icon-twitter"></i> Tweet @TryGhost!</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-how-to" role="menuitem" tabindex="-1" href="http://support.ghost.org/how-to-use-ghost/" target="_blank"><i class="icon-book"></i> How to Use Ghost</a></li> - <li role="presentation"><a class="dropdown-item help-menu-markdown" role="menuitem" tabindex="-1" href="" data-ember-action="1036"><i class="icon-markdown"></i> Markdown Help</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-wishlist" role="menuitem" tabindex="-1" href="http://ideas.ghost.org/" target="_blank"><i class="icon-idea"></i> Wishlist</a></li> - </ul> -</div> </div> - </footer> -</nav> - - - <main id="gh-main" class="gh-main" role="main" data-notification-count="0"> - <section id="ember1308" class="ember-view gh-view js-settings-content"><header class="view-header"> - <h2 class="view-title">Themes</h2> - <section class="view-actions"> - <input id="blog-title" class="gh-input package-filter" type="text" name="general[title]" placeholder="Filter themes by name"> - - </section> -</header> - -<section class="view-content"> - - <form> - <div class="form-group for-select"> - <label for="activeTheme">Active Theme</label> - <span class="gh-select" tabindex="0" data-select-text="CodingHorror - 1.0.0"> - <select id="activeTheme" class="ember-view ember-select" name="general[activeTheme]"><!----> - <option id="ember1360" class="ember-view" value="Casper-master">CasperPR - 1.0.1</option> - <option id="ember1363" class="ember-view" value="blogdotghost">Blog - 1.0</option> - <option id="ember1366" class="ember-view" value="casper">Casper - 1.2.4</option><option id="ember1369" class="ember-view" value="codinghorror">CodingHorror - 1.0.0</option> - <option id="ember1372" class="ember-view" value="codinghorror.zip">codinghorror.zip</option> - <option id="ember1375" class="ember-view" value="john">john</option> - </select> - </span> - <p>Select a theme for your blog</p> - </div> - </form> - - <div class="package-grid package-grid-themes"> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - <div class="package-grid-cell"> - <article class="package-card-theme"> - <div class="package-card-content"> - <a class="package-card-theme-image" href="/theme/"> - <img src="http://cl.ly/b4FN/casper2.jpg" alt="theme" /> - <div class="package-card-theme-overlay"> - <h3 class="package-card-theme-title">Casper and a Really Long Name</h3> - </div> - </a> - </div> - <footer class="package-card-footer"> - <a class="package-developer" href="#" target="_blank"> - <img src="https://pbs.twimg.com/profile_images/439941058404028416/3nRmjK6v.jpeg" alt="developer" /> - <span class="package-developer-name">GhostFoundation1</span> - </a> - <div class="package-controls"> - <a class="package-uninstall package-controls-button" href="#"><i class="icon-trash"></i>Uninstall</a> - </div> - </footer> - </article> - </div> - - </div> - -</section> - </main> - - <aside id="ember766" class="ember-view gh-notifications"><!----></aside> - - <!----> - <!----> - -</div></div> - -</body> -</html> diff --git a/core/client/app/index.html b/core/client/app/index.html deleted file mode 100644 index fddadc8673..0000000000 --- a/core/client/app/index.html +++ /dev/null @@ -1,56 +0,0 @@ -<!doctype html> -<!--[if (IE 8)&!(IEMobile)]><html class="no-js lt-ie9" lang="en"><![endif]--> -<!--[if (gte IE 9)| IEMobile |!(IE)]><!--><html class="no-js" lang="en"><!--<![endif]--> -<head> - <meta http-equiv="Content-Type" content="text/html" charset="UTF-8" /> - <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" /> - - <title>Ghost Admin</title> - - {{content-for 'head'}} - - <meta name="HandheldFriendly" content="True" /> - <meta name="MobileOptimized" content="320" /> - <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1, minimal-ui" /> - <meta name="pinterest" content="nopin" /> - - <meta http-equiv="cleartype" content="on" /> - <meta name="apple-mobile-web-app-capable" content="yes" /> - <meta name="apple-mobile-web-app-status-bar-style" content="black" /> - <meta name="apple-mobile-web-app-title" content="Ghost" /> - - <link rel="shortcut icon" href="{{asset "favicon.ico"}}" /> - <link rel="apple-touch-icon-precomposed" href="{{asset "img/touch-icon-iphone.png" ghost="true"}}" /> - <link rel="apple-touch-icon-precomposed" sizes="76x76" href="{{asset "img/touch-icon-ipad.png" ghost="true"}}" /> - <link rel="apple-touch-icon-precomposed" sizes="120x120" href="{{asset "img/small.png" ghost="true"}}" /> - <link rel="apple-touch-icon-precomposed" sizes="152x152" href="{{asset "img/medium.png" ghost="true"}}" /> - - <meta name="application-name" content="Ghost" /> - <meta name="msapplication-TileColor" content="#ffffff" /> - <meta name="msapplication-square70x70logo" content="{{asset "img/small.png" ghost="true"}}" /> - <meta name="msapplication-square150x150logo" content="{{asset "img/medium.png" ghost="true"}}" /> - <meta name="msapplication-square310x310logo" content="{{asset "img/large.png" ghost="true"}}" /> - - {{#each configuration as |config key|}} - <meta name="env-{{key}}" content="{{config.value}}" data-type="{{config.type}}" /> - {{/each}} - - {{#if configuration.useGoogleFonts.value}} - <link rel="stylesheet" type="text/css" href="//fonts.googleapis.com/css?family=Open+Sans:400,300,700" /> - {{/if}} - - <link rel="stylesheet" href="{{asset "vendor.css" ghost="true" minifyInProduction="true"}}" /> - <link rel="stylesheet" href="{{asset "ghost.css" ghost="true" minifyInProduction="true"}}" /> - - {{content-for 'head-footer'}} -</head> -<body> -{{content-for 'body'}} - -{{content-for 'body-footer'}} - -{{! Dem scripts }} -<script src="{{asset "vendor.js" ghost="true" minifyInProduction="true"}}"></script> -<script src="{{asset "ghost.js" ghost="true" minifyInProduction="true"}}"></script> -</body> -</html> diff --git a/core/client/app/initializers/ember-simple-auth.js b/core/client/app/initializers/ember-simple-auth.js deleted file mode 100644 index 8d43833f8c..0000000000 --- a/core/client/app/initializers/ember-simple-auth.js +++ /dev/null @@ -1,17 +0,0 @@ -import ENV from '../config/environment'; -import ghostPaths from '../utils/ghost-paths'; -import Configuration from 'ember-simple-auth/configuration'; -import setupSession from 'ember-simple-auth/initializers/setup-session'; -import setupSessionService from 'ember-simple-auth/initializers/setup-session-service'; - -export default { - name: 'ember-simple-auth', - initialize(registry) { - let config = ENV['ember-simple-auth'] || {}; - config.baseURL = ghostPaths().adminRoot; - Configuration.load(config); - - setupSession(registry); - setupSessionService(registry); - } -}; diff --git a/core/client/app/initializers/trailing-history.js b/core/client/app/initializers/trailing-history.js deleted file mode 100644 index 73493e4637..0000000000 --- a/core/client/app/initializers/trailing-history.js +++ /dev/null @@ -1,23 +0,0 @@ -import Ember from 'ember'; - -const {HistoryLocation} = Ember; - -let trailingHistory = HistoryLocation.extend({ - formatURL() { - let url = this._super(...arguments); - - if (url.indexOf('?') > 0) { - return url.replace(/([^\/])\?/, '$1/?'); - } else { - return url.replace(/\/?$/, '/'); - } - } -}); - -export default { - name: 'registerTrailingLocationHistory', - - initialize(application) { - application.register('location:trailing-history', trailingHistory); - } -}; diff --git a/core/client/app/instance-initializers/jquery-ajax-oauth-prefilter.js b/core/client/app/instance-initializers/jquery-ajax-oauth-prefilter.js deleted file mode 100644 index f88a36afef..0000000000 --- a/core/client/app/instance-initializers/jquery-ajax-oauth-prefilter.js +++ /dev/null @@ -1,21 +0,0 @@ -import Ember from 'ember'; - -const {assign} = Ember; - -export default { - name: 'jquery-ajax-oauth-prefilter', - after: 'ember-simple-auth', - - initialize(application) { - let session = application.lookup('service:session'); - - Ember.$.ajaxPrefilter(function (options) { - session.authorize('authorizer:oauth2', function (headerName, headerValue) { - let headerObject = {}; - - headerObject[headerName] = headerValue; - options.headers = assign(options.headers || {}, headerObject); - }); - }); - } -}; diff --git a/core/client/app/mirage/config.js b/core/client/app/mirage/config.js deleted file mode 100644 index 742bd4491d..0000000000 --- a/core/client/app/mirage/config.js +++ /dev/null @@ -1,436 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import Mirage from 'ember-cli-mirage'; - -const {$, isBlank} = Ember; - -function paginatedResponse(modelName, allModels, request) { - let page = +request.queryParams.page || 1; - let limit = request.queryParams.limit || 15; - let pages, models, next, prev; - - allModels = allModels || []; - - if (limit === 'all') { - models = allModels; - pages = 1; - } else { - limit = +limit; - - let start = (page - 1) * limit; - let end = start + limit; - - models = allModels.slice(start, end); - pages = Math.ceil(allModels.length / limit); - - if (start > 0) { - prev = page - 1; - } - - if (end < allModels.length) { - next = page + 1; - } - } - - return { - meta: { - pagination: { - page, - limit, - pages, - total: allModels.length, - next: next || null, - prev: prev || null - } - }, - [modelName]: models - }; -} - -function mockSubscribers(server) { - server.get('/subscribers/', function (db, request) { - let response = paginatedResponse('subscribers', db.subscribers, request); - return response; - }); - - server.post('/subscribers/', function (db, request) { - let [attrs] = JSON.parse(request.requestBody).subscribers; - let [subscriber] = db.subscribers.where({email: attrs.email}); - - if (subscriber) { - return new Mirage.Response(422, {}, { - errors: [{ - errorType: 'ValidationError', - message: 'Email already exists.', - property: 'email' - }] - }); - } else { - attrs.created_at = new Date(); - attrs.created_by = 0; - - subscriber = db.subscribers.insert(attrs); - - return { - subscriber - }; - } - }); - - server.put('/subscribers/:id/', function (db, request) { - let {id} = request.params; - let [attrs] = JSON.parse(request.requestBody).subscribers; - let subscriber = db.subscribers.update(id, attrs); - - return { - subscriber - }; - }); - - server.del('/subscribers/:id/', function (db, request) { - db.subscribers.remove(request.params.id); - - return new Mirage.Response(204, {}, {}); - }); - - server.post('/subscribers/csv/', function (/*db, request*/) { - // NB: we get a raw FormData object with no way to inspect it in Chrome - // until version 50 adds the additional read methods - // https://developer.mozilla.org/en-US/docs/Web/API/FormData#Browser_compatibility - - server.createList('subscriber', 50); - - return { - meta: { - stats: { - imported: 50, - duplicates: 3, - invalid: 2 - } - } - }; - }); -} - -export default function () { - // this.urlPrefix = ''; // make this `http://localhost:8080`, for example, if your API is on a different server - this.namespace = 'ghost/api/v0.1'; // make this `api`, for example, if your API is namespaced - this.timing = 400; // delay for each request, automatically set to 0 during testing - - // Mock endpoints here to override real API requests during development - mockSubscribers(this); - - // keep this line, it allows all other API requests to hit the real server - this.passthrough(); - - // add any external domains to make sure those get passed through too - this.passthrough('https://count.ghost.org/'); - this.passthrough('http://www.gravatar.com/**'); -} - -// Mock all endpoints here as there is no real API during testing -export function testConfig() { - // this.urlPrefix = ''; // make this `http://localhost:8080`, for example, if your API is on a different server - this.namespace = 'ghost/api/v0.1'; // make this `api`, for example, if your API is namespaced - // this.timing = 400; // delay for each request, automatically set to 0 during testing - - /* Authentication ------------------------------------------------------- */ - - this.post('/authentication/token', function () { - return { - access_token: '5JhTdKI7PpoZv4ROsFoERc6wCHALKFH5jxozwOOAErmUzWrFNARuH1q01TYTKeZkPW7FmV5MJ2fU00pg9sm4jtH3Z1LjCf8D6nNqLYCfFb2YEKyuvG7zHj4jZqSYVodN2YTCkcHv6k8oJ54QXzNTLIDMlCevkOebm5OjxGiJpafMxncm043q9u1QhdU9eee3zouGRMVVp8zkKVoo5zlGMi3zvS2XDpx7xsfk8hKHpUgd7EDDQxmMueifWv7hv6n', - expires_in: 3600, - refresh_token: 'XP13eDjwV5mxOcrq1jkIY9idhdvN3R1Br5vxYpYIub2P5Hdc8pdWMOGmwFyoUshiEB62JWHTl8H1kACJR18Z8aMXbnk5orG28br2kmVgtVZKqOSoiiWrQoeKTqrRV0t7ua8uY5HdDUaKpnYKyOdpagsSPn3WEj8op4vHctGL3svOWOjZhq6F2XeVPMR7YsbiwBE8fjT3VhTB3KRlBtWZd1rE0Qo2EtSplWyjGKv1liAEiL0ndQoLeeSOCH4rTP7', - token_type: 'Bearer' - }; - }); - - this.post('/authentication/passwordreset', function (db, request) { - // jscs:disable requireObjectDestructuring - let {passwordreset} = $.deparam(request.requestBody); - let email = passwordreset[0].email; - // jscs:enable requireObjectDestructuring - - if (email === 'unknown@example.com') { - return new Mirage.Response(404, {}, { - errors: [ - { - message: 'There is no user with that email address.', - errorType: 'NotFoundError' - } - ] - }); - } else { - return { - passwordreset: [ - {message: 'Check your email for further instructions.'} - ] - }; - } - }); - - /* Download Count ------------------------------------------------------- */ - - let downloadCount = 0; - this.get('https://count.ghost.org/', function () { - downloadCount++; - return { - count: downloadCount - }; - }); - - /* Notifications -------------------------------------------------------- */ - - this.get('/notifications/', 'notifications'); - - /* Posts ---------------------------------------------------------------- */ - - this.post('/posts/', function (db, request) { - let [attrs] = JSON.parse(request.requestBody).posts; - let post; - - if (isBlank(attrs.slug) && !isBlank(attrs.title)) { - attrs.slug = attrs.title.dasherize(); - } - - // NOTE: this does not use the post factory to fill in blank fields - post = db.posts.insert(attrs); - - return { - posts: [post] - }; - }); - - this.get('/posts/', function (db, request) { - // TODO: handle status/staticPages/author params - let response = paginatedResponse('posts', db.posts, request); - return response; - }); - - this.del('/posts/:id/', function (db, request) { - db.posts.remove(request.params.id); - - return new Mirage.Response(204, {}, {}); - }); - - /* Roles ---------------------------------------------------------------- */ - - this.get('/roles/', function (db, request) { - if (request.queryParams.permissions === 'assign') { - let roles = db.roles.find([1,2,3]); - return {roles}; - } - - return { - roles: db.roles - }; - }); - - /* Settings ------------------------------------------------------------- */ - - this.get('/settings/', function (db, request) { - let filters = request.queryParams.type.split(','); - let settings = []; - - filters.forEach((filter) => { - settings.pushObjects(db.settings.where({type: filter})); - }); - - return { - settings, - meta: { - filters: { - type: request.queryParams.type - } - } - }; - }); - - this.put('/settings/', function (db, request) { - let newSettings = JSON.parse(request.requestBody).settings; - - db.settings.remove(); - db.settings.insert(newSettings); - - return { - meta: {}, - settings: db.settings - }; - }); - - /* Apps - Slack Test Notification --------------------------------------------------------- */ - - this.post('/slack/test', function () { - return {}; - }); - - /* Slugs ---------------------------------------------------------------- */ - - this.get('/slugs/post/:slug/', function (db, request) { - return { - slugs: [ - {slug: Ember.String.dasherize(decodeURIComponent(request.params.slug))} - ] - }; - }); - - this.get('/slugs/user/:slug/', function (db, request) { - return { - slugs: [ - {slug: Ember.String.dasherize(decodeURIComponent(request.params.slug))} - ] - }; - }); - - /* Setup ---------------------------------------------------------------- */ - - this.post('/authentication/setup', function (db, request) { - let [attrs] = $.deparam(request.requestBody).setup; - let [role] = db.roles.where({name: 'Owner'}); - let user; - - // create owner role unless already exists - if (!role) { - role = db.roles.insert({name: 'Owner'}); - } - attrs.roles = [role]; - - if (!isBlank(attrs.email)) { - attrs.slug = attrs.email.split('@')[0].dasherize(); - } - - // NOTE: this does not use the user factory to fill in blank fields - user = db.users.insert(attrs); - - delete user.roles; - - return { - users: [user] - }; - }); - - this.get('/authentication/setup/', function () { - return { - setup: [ - {status: true} - ] - }; - }); - - /* Subscribers ---------------------------------------------------------- */ - - mockSubscribers(this); - - /* Tags ----------------------------------------------------------------- */ - - this.post('/tags/', function (db, request) { - let [attrs] = JSON.parse(request.requestBody).tags; - let tag; - - if (isBlank(attrs.slug) && !isBlank(attrs.name)) { - attrs.slug = attrs.name.dasherize(); - } - - // NOTE: this does not use the tag factory to fill in blank fields - tag = db.tags.insert(attrs); - - return { - tag - }; - }); - - this.get('/tags/', function (db, request) { - let response = paginatedResponse('tags', db.tags, request); - // TODO: remove post_count unless requested? - return response; - }); - - this.get('/tags/slug/:slug/', function (db, request) { - let [tag] = db.tags.where({slug: request.params.slug}); - - // TODO: remove post_count unless requested? - - return { - tag - }; - }); - - this.put('/tags/:id/', function (db, request) { - let {id} = request.params; - let [attrs] = JSON.parse(request.requestBody).tags; - let record = db.tags.update(id, attrs); - - return { - tag: record - }; - }); - - this.del('/tags/:id/', function (db, request) { - db.tags.remove(request.params.id); - - return new Mirage.Response(204, {}, {}); - }); - - /* Users ---------------------------------------------------------------- */ - - this.post('/users/', function (db, request) { - let [attrs] = JSON.parse(request.requestBody).users; - let user; - - if (!isBlank(attrs.email)) { - attrs.slug = attrs.email.split('@')[0].dasherize(); - } - - // NOTE: this does not use the user factory to fill in blank fields - user = db.users.insert(attrs); - - return { - users: [user] - }; - }); - - // /users/me = Always return the user with ID=1 - this.get('/users/me', function (db) { - return { - users: [db.users.find(1)] - }; - }); - - this.get('/users/', 'users'); - - this.get('/users/slug/:slug/', function (db, request) { - let user = db.users.where({slug: request.params.slug}); - - return { - users: user - }; - }); - - this.del('/users/:id/', function (db, request) { - db.users.remove(request.params.id); - - return new Mirage.Response(204, {}, {}); - }); - - this.get('/users/:id', function (db, request) { - return { - users: [db.users.find(request.params.id)] - }; - }); - - this.put('/users/:id/', function (db, request) { - let {id} = request.params; - let [attrs] = JSON.parse(request.requestBody).users; - let record = db.users.update(id, attrs); - - return { - user: record - }; - }); - - /* External sites ------------------------------------------------------- */ - - this.get('http://www.gravatar.com/avatar/:md5', function () { - return ''; - }, 200); -} diff --git a/core/client/app/mirage/factories/notification.js b/core/client/app/mirage/factories/notification.js deleted file mode 100644 index 6029e3d7c6..0000000000 --- a/core/client/app/mirage/factories/notification.js +++ /dev/null @@ -1,9 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - dismissible: true, - message: 'This is an alert', - status: 'alert', - type: 'error' -}); diff --git a/core/client/app/mirage/factories/post.js b/core/client/app/mirage/factories/post.js deleted file mode 100644 index d61a4c3ec5..0000000000 --- a/core/client/app/mirage/factories/post.js +++ /dev/null @@ -1,23 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - uuid(i) { return `post-${i}`; }, - description(i) { return `Title for post ${i}.`; }, - slug(i) { return `post-${i}`; }, - markdown(i) { return `Markdown for post ${i}.`; }, - html(i) { return `HTML for post ${i}.`; }, - image(i) { return `/content/images/2015/10/post-${i}.jpg`; }, - featured() { return false; }, - page() { return false; }, - status(i) { return `/content/images/2015/10/post-${i}.jpg`; }, - meta_description(i) { return `Meta description for post ${i}.`; }, - meta_title(i) { return `Meta Title for post ${i}`; }, - author_id() { return 1; }, - updated_at() { return '2015-10-19T16:25:07.756Z'; }, - updated_by() { return 1; }, - published_at() { return '2015-10-19T16:25:07.756Z'; }, - published_by() { return 1; }, - created_at() { return '2015-09-11T09:44:29.871Z'; }, - created_by() { return 1; } -}); diff --git a/core/client/app/mirage/factories/role.js b/core/client/app/mirage/factories/role.js deleted file mode 100644 index 9a66a474c2..0000000000 --- a/core/client/app/mirage/factories/role.js +++ /dev/null @@ -1,12 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - created_at() { return '2013-11-25T14:48:11.000Z'; }, - created_by() { return 1; }, - description(i) { return `Role ${i}`; }, - name() { return ''; }, - updated_at() { return '2013-11-25T14:48:11.000Z'; }, - updated_by() { return 1; }, - uuid(i) { return `role-${i}`; } -}); diff --git a/core/client/app/mirage/factories/setting.js b/core/client/app/mirage/factories/setting.js deleted file mode 100644 index 3cf6ff9dad..0000000000 --- a/core/client/app/mirage/factories/setting.js +++ /dev/null @@ -1,13 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - uuid(i) { return `setting-${i}`; }, - key(i) { return `setting-${i}`; }, - value() { return null; }, - type() { return 'blog'; }, - created_at() { return '2015-01-12T18:29:01.000Z'; }, - created_by() { return 1; }, - updated_at() { return '2015-10-27T17:39:58.288Z'; }, - updated_by() { return 1; } -}); diff --git a/core/client/app/mirage/factories/subscriber.js b/core/client/app/mirage/factories/subscriber.js deleted file mode 100644 index 711d4e1f64..0000000000 --- a/core/client/app/mirage/factories/subscriber.js +++ /dev/null @@ -1,21 +0,0 @@ -import Mirage, {faker} from 'ember-cli-mirage'; - -let randomDate = function randomDate(start = moment().subtract(30, 'days').toDate(), end = new Date()) { - return new Date(start.getTime() + Math.random() * (end.getTime() - start.getTime())); -}; - -let statuses = ['pending', 'subscribed']; - -// jscs:disable requireBlocksOnNewline -// jscs:disable requireCamelCaseOrUpperCaseIdentifiers -export default Mirage.Factory.extend({ - uuid(i) { return `subscriber-${i}`; }, - name() { return `${faker.name.firstName()} ${faker.name.lastName()}`; }, - email() { return faker.internet.email(); }, - status() { return statuses[Math.floor(Math.random() * statuses.length)]; }, - created_at() { return randomDate(); }, - updated_at: null, - created_by: 0, - updated_by: null, - unsubscribed_at: null -}); diff --git a/core/client/app/mirage/factories/tag.js b/core/client/app/mirage/factories/tag.js deleted file mode 100644 index 7a17d574bb..0000000000 --- a/core/client/app/mirage/factories/tag.js +++ /dev/null @@ -1,23 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - created_at() { return '2015-09-11T09:44:29.871Z'; }, - created_by() { return 1; }, - description(i) { return `Description for tag ${i}.`; }, - hidden() { return false; }, - image(i) { return `/content/images/2015/10/tag-${i}.jpg`; }, - meta_description(i) { return `Meta description for tag ${i}.`; }, - meta_title(i) { return `Meta Title for tag ${i}`; }, - name(i) { return `Tag ${i}`; }, - parent() { return null; }, - slug(i) { return `tag-${i}`; }, - updated_at() { return '2015-10-19T16:25:07.756Z'; }, - updated_by() { return 1; }, - uuid(i) { return `tag-${i}`; }, - count() { - return { - posts: 1 - }; - }, -}); diff --git a/core/client/app/mirage/factories/user.js b/core/client/app/mirage/factories/user.js deleted file mode 100644 index 9c40ba69cc..0000000000 --- a/core/client/app/mirage/factories/user.js +++ /dev/null @@ -1,27 +0,0 @@ -/* jscs:disable */ -import Mirage from 'ember-cli-mirage'; - -export default Mirage.Factory.extend({ - accessibility() { return null; }, - bio() { return null; }, - cover() { return null; }, - created_at() { return '2015-09-02T13:41:50.000Z'; }, - created_by() { return null; }, - email(i) { return `user-${i}@example.com`; }, - image() { return '//www.gravatar.com/avatar/3ae045bc198a157401827c8455cd7c99?s=250&d=mm&r=x'; }, - language() { return 'en_US'; }, - last_login() { return '2015-11-02T16:12:05.000Z'; }, - location() { return null; }, - meta_description() { return null; }, - meta_title() { return null; }, - name(i) { return `User ${i}`; }, - slug(i) { return `user-${i}`; }, - status() { return 'active'; }, - tour() { return null; }, - updated_at() { return '2015-11-02T16:12:05.000Z'; }, - updated_by() { return '2015-09-02T13:41:50.000Z'; }, - uuid(i) { return `user-${i}`; }, - website() { return 'http://example.com'; }, - - roles() { return []; } -}); diff --git a/core/client/app/mirage/fixtures/roles.js b/core/client/app/mirage/fixtures/roles.js deleted file mode 100644 index 6b29208fc5..0000000000 --- a/core/client/app/mirage/fixtures/roles.js +++ /dev/null @@ -1,43 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -export default [ - { - id: 1, - uuid: 'b2576c4e-fa4e-41d4-8236-ced75f735222', - name: 'Administrator', - description: 'Administrators', - created_at: '2015-11-13T16:01:29.131Z', - created_by: 1, - updated_at: '2015-11-13T16:01:29.131Z', - updated_by: 1 - }, - { - id: 2, - uuid: '6ee03efb-322e-4f6e-9c91-bc228b5eec6b', - name: 'Editor', - description: 'Editors', - created_at: '2015-11-13T16:01:29.131Z', - created_by: 1, - updated_at: '2015-11-13T16:01:29.131Z', - updated_by: 1 - }, - { - id: 3, - uuid: 'de481b62-63f8-42c7-b5b9-6c5f5a877f53', - name: 'Author', - description: 'Authors', - created_at: '2015-11-13T16:01:29.131Z', - created_by: 1, - updated_at: '2015-11-13T16:01:29.131Z', - updated_by: 1 - }, - { - id: 4, - uuid: 'ac8cbaf6-e6be-4129-b0fb-ec9ddfa61056', - name: 'Owner', - description: 'Blog Owner', - created_at: '2015-11-13T16:01:29.132Z', - created_by: 1, - updated_at: '2015-11-13T16:01:29.132Z', - updated_by: 1 - } -]; diff --git a/core/client/app/mirage/fixtures/settings.js b/core/client/app/mirage/fixtures/settings.js deleted file mode 100644 index 472c9a00e3..0000000000 --- a/core/client/app/mirage/fixtures/settings.js +++ /dev/null @@ -1,218 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -export default [ - { - created_at: '2015-09-11T09:44:30.805Z', - created_by: 1, - id: 1, - key: 'title', - type: 'blog', - updated_at: '2015-10-04T16:26:05.195Z', - updated_by: 1, - uuid: '39e16daf-43fa-4bf0-87d4-44948ba8bf4c', - value: 'Test Blog' - }, - { - created_at: '2015-09-11T09:44:30.806Z', - created_by: 1, - id: 2, - key: 'description', - type: 'blog', - updated_at: '2015-10-04T16:26:05.198Z', - updated_by: 1, - uuid: 'e6c8b636-6925-4c4a-a5d9-1dc0870fb8ea', - value: 'Thoughts, stories and ideas.' - }, - { - id: 3, - uuid: '4339ce48-b485-418a-acc2-1d34cf17a5e3', - key: 'logo', - value: '/content/images/2013/Nov/logo.png', - type: 'blog', - created_at: '2013-11-25T14:48:11.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.273Z', - updated_by: 1 - }, - { - id: 4, - uuid: 'e41b6c2a-7f72-45ea-96d8-ee016f06d78b', - key: 'cover', - value: '/content/images/2014/Feb/cover.jpg', - type: 'blog', - created_at: '2013-11-25T14:48:11.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.276Z', - updated_by: 1 - }, - { - id: 5, - uuid: '4558457e-9f61-47a5-9d45-8b83829bf1cf', - key: 'defaultLang', - value: 'en_US', - type: 'blog', - created_at: '2013-11-25T14:48:11.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.278Z', - updated_by: 1 - }, - { - created_at: '2015-09-11T09:44:30.809Z', - created_by: 1, - id: 6, - key: 'postsPerPage', - type: 'blog', - updated_at: '2015-10-04T16:26:05.211Z', - updated_by: 1, - uuid: '775e6ca1-bcc3-4347-a53d-15d5d76c04a4', - value: '5' - }, - { - id: 7, - uuid: '3c93b240-d22b-473f-9063-537023e06c2d', - key: 'forceI18n', - value: 'true', - type: 'blog', - created_at: '2013-11-25T14:48:11.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.280Z', - updated_by: 1 - }, - { - id: 8, - uuid: '4e58389f-f173-4387-b28c-0435623882ad', - key: 'activeTheme', - value: 'casper', - type: 'theme', - created_at: '2013-11-25T14:48:11.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.284Z', - updated_by: 1 - }, - { - id: 9, - uuid: '8052c2bf-9c19-4d6c-8944-7465321d00be', - key: 'permalinks', - value: '/:slug/', - type: 'blog', - created_at: '2014-01-14T12:01:51.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.282Z', - updated_by: 1 - }, - { - created_at: '2015-09-11T09:44:30.809Z', - created_by: 1, - id: 10, - key: 'ghost_head', - type: 'blog', - updated_at: '2015-09-23T13:32:49.858Z', - updated_by: 1, - uuid: 'df7f3151-bc08-4a77-be9d-dd315b630d51', - value: '' - }, - { - created_at: '2015-09-11T09:44:30.809Z', - created_by: 1, - id: 11, - key: 'ghost_foot', - type: 'blog', - updated_at: '2015-09-23T13:32:49.858Z', - updated_by: 1, - uuid: '0649d45e-828b-4dd0-8381-3dff6d1d5ddb', - value: '' - }, - { - id: 12, - uuid: 'd806f358-7996-4c74-b153-8876959c4b70', - key: 'labs', - value: '{"codeInjectionUI":true,"subscribers":true}', - type: 'blog', - created_at: '2015-01-12T18:29:01.000Z', - created_by: 1, - updated_at: '2015-10-27T17:39:58.288Z', - updated_by: 1 - }, - { - created_at: '2015-09-11T09:44:30.810Z', - created_by: 1, - id: 13, - key: 'navigation', - type: 'blog', - updated_at: '2015-09-23T13:32:49.868Z', - updated_by: 1, - uuid: '4cc51d1c-fcbd-47e6-a71b-fdd1abb223fc', - value: JSON.stringify([ - {label: 'Home', url: '/'}, - {label: 'About', url: '/about'} - ]) - }, - { - created_at: '2015-09-11T09:44:30.810Z', - created_by: 1, - id: 14, - key: 'isPrivate', - type: 'blog', - updated_at: '2015-09-23T13:32:49.868Z', - updated_by: 1, - uuid: 'e306ec3e-d079-11e5-ab30-625662870761', - value: false - }, - { - created_at: '2015-09-11T09:44:30.810Z', - created_by: 1, - id: 15, - key: 'password', - type: 'blog', - updated_at: '2015-09-23T13:32:49.868Z', - updated_by: 1, - uuid: 'f8e8cbda-d079-11e5-ab30-625662870761', - value: '' - }, - { - created_at: '2016-05-05T15:04:03.115Z', - created_by: 1, - id: 17, - key: 'slack', - type: 'blog', - updated_at: '2016-05-05T18:33:09.168Z', - updated_by: 1, - uuid: 'dd4ebaa8-dedb-40ff-a663-ec64a92d4111', - value: '[{"url":""}]' - }, - { - created_at: '2016-05-05T15:40:12.133Z', - created_by: 1, - id: 23, - key: 'facebook', - type: 'blog', - updated_at: '2016-05-08T15:20:25.953Z', - updated_by: 1, - uuid: 'd4387e5c-3230-46dd-a89b-0d8a40365c35', - value: 'test' - }, - { - created_at: '2016-05-05T15:40:12.134Z', - created_by: 1, - id: 24, - key: 'twitter', - type: 'blog', - updated_at: '2016-05-08T15:20:25.954Z', - updated_by: 1, - uuid: '5130441f-e4c7-4750-9692-a22d841ab049', - value: '@test' - }, - { - key: 'availableThemes', - value: [ - { - name: 'casper', - package: { - name: 'Blog', - version: '1.0' - }, - active: true - } - ], - type: 'theme' - } -]; diff --git a/core/client/app/mirage/scenarios/default.js b/core/client/app/mirage/scenarios/default.js deleted file mode 100644 index b43931e3b9..0000000000 --- a/core/client/app/mirage/scenarios/default.js +++ /dev/null @@ -1,8 +0,0 @@ -export default function (server) { - // Seed your development database using your factories. This - // data will not be loaded in your tests. - - // server.createList('contact', 10); - - server.createList('subscriber', 125); -} diff --git a/core/client/app/mixins/404-handler.js b/core/client/app/mixins/404-handler.js deleted file mode 100644 index ef378910b2..0000000000 --- a/core/client/app/mixins/404-handler.js +++ /dev/null @@ -1,23 +0,0 @@ -import Ember from 'ember'; - -export default Ember.Mixin.create({ - actions: { - error(error, transition) { - if (error.errors && error.errors[0].errorType === 'NotFoundError') { - transition.abort(); - - let routeInfo = transition.handlerInfos[transition.handlerInfos.length - 1]; - let router = this.get('router'); - let params = []; - - for (let key of Object.keys(routeInfo.params)) { - params.push(routeInfo.params[key]); - } - - return this.transitionTo('error404', router.generate(routeInfo.name, ...params).replace('/ghost/', '').replace(/^\//g, '')); - } - - return this._super(...arguments); - } - } -}); diff --git a/core/client/app/mixins/active-link-wrapper.js b/core/client/app/mixins/active-link-wrapper.js deleted file mode 100644 index fa8a2831c5..0000000000 --- a/core/client/app/mixins/active-link-wrapper.js +++ /dev/null @@ -1,32 +0,0 @@ -// logic borrowed from https://github.com/alexspeller/ember-cli-active-link-wrapper/blob/master/addon/components/active-link.js - -import Ember from 'ember'; - -const {computed, run} = Ember; -const emberA = Ember.A; - -export default Ember.Mixin.create({ - - classNameBindings: ['active'], - - childLinkViews: [], - - active: computed('childLinkViews.@each.active', function () { - return emberA(this.get('childLinkViews')).isAny('active'); - }), - - didRender() { - this._super(...arguments); - - run.schedule('afterRender', this, function () { - let childLinkElements = this.$('a.ember-view'); - - let childLinkViews = childLinkElements.toArray().map((view) => - this._viewRegistry[view.id] - ); - - this.set('childLinkViews', childLinkViews); - }); - } - -}); diff --git a/core/client/app/mixins/body-event-listener.js b/core/client/app/mixins/body-event-listener.js deleted file mode 100644 index 1a975a93d1..0000000000 --- a/core/client/app/mixins/body-event-listener.js +++ /dev/null @@ -1,49 +0,0 @@ -import Ember from 'ember'; - -const {$, Mixin, run} = Ember; - -function K() { - return this; -} - -// Code modified from Addepar/ember-widgets -// https://github.com/Addepar/ember-widgets/blob/master/src/mixins.coffee#L39 - -export default Mixin.create({ - bodyElementSelector: 'html', - bodyClick: K, - - init() { - this._super(...arguments); - - return run.next(this, this._setupDocumentHandlers); - }, - - willDestroy() { - this._super(...arguments); - - return this._removeDocumentHandlers(); - }, - - _setupDocumentHandlers() { - if (this._clickHandler) { - return; - } - - this._clickHandler = () => { - return this.bodyClick(); - }; - - return $(this.get('bodyElementSelector')).on('click', this._clickHandler); - }, - - _removeDocumentHandlers() { - $(this.get('bodyElementSelector')).off('click', this._clickHandler); - this._clickHandler = null; - }, - - // http://stackoverflow.com/questions/152975/how-to-detect-a-click-outside-an-element - click(event) { - return event.stopPropagation(); - } -}); diff --git a/core/client/app/mixins/current-user-settings.js b/core/client/app/mixins/current-user-settings.js deleted file mode 100644 index 75f02af752..0000000000 --- a/core/client/app/mixins/current-user-settings.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; - -const {Mixin} = Ember; - -export default Mixin.create({ - transitionAuthor() { - return (user) => { - if (user.get('isAuthor')) { - return this.transitionTo('team.user', user); - } - - return user; - }; - }, - - transitionEditor() { - return (user) => { - if (user.get('isEditor')) { - return this.transitionTo('team'); - } - - return user; - }; - } -}); diff --git a/core/client/app/mixins/dropdown-mixin.js b/core/client/app/mixins/dropdown-mixin.js deleted file mode 100644 index dfb885f54c..0000000000 --- a/core/client/app/mixins/dropdown-mixin.js +++ /dev/null @@ -1,17 +0,0 @@ -import Ember from 'ember'; - -const {Evented, Mixin} = Ember; - -/* - Dropdowns and their buttons are evented and do not propagate clicks. -*/ -export default Mixin.create(Evented, { - classNameBindings: ['isOpen:open:closed'], - isOpen: false, - - click(event) { - this._super(event); - - return event.stopPropagation(); - } -}); diff --git a/core/client/app/mixins/ed-editor-api.js b/core/client/app/mixins/ed-editor-api.js deleted file mode 100644 index 64aa610fae..0000000000 --- a/core/client/app/mixins/ed-editor-api.js +++ /dev/null @@ -1,142 +0,0 @@ -import Ember from 'ember'; - -const { - Mixin, - run -} = Ember; - -export default Mixin.create({ - /** - * Get Value - * - * Get the full contents of the textarea - * - * @returns {String} - */ - getValue() { - return this.readDOMAttr('value'); - }, - - /** - * Get Selection - * - * Return the currently selected text from the textarea - * - * @returns {Selection} - */ - getSelection() { - return this.$().getSelection(); - }, - - /** - * Get Line To Cursor - * - * Fetch the string of characters from the start of the given line up to the cursor - * @returns {{text: string, start: number}} - */ - getLineToCursor() { - let selection = this.$().getSelection(); - let value = this.getValue(); - let lineStart; - - // Normalise newlines - value = value.replace('\r\n', '\n'); - - // We want to look at the characters behind the cursor - lineStart = value.lastIndexOf('\n', selection.start - 1) + 1; - - return { - text: value.substring(lineStart, selection.start), - start: lineStart - }; - }, - - /** - * Get Line - * - * Return the string of characters for the line the cursor is currently on - * - * @returns {{text: string, start: number, end: number}} - */ - getLine() { - let selection = this.$().getSelection(); - let value = this.getValue(); - let lineStart, - lineEnd; - - // Normalise newlines - value = value.replace('\r\n', '\n'); - - // We want to look at the characters behind the cursor - lineStart = value.lastIndexOf('\n', selection.start - 1) + 1; - lineEnd = value.indexOf('\n', selection.start); - lineEnd = lineEnd === -1 ? value.length - 1 : lineEnd; - - return { - // jscs:disable - text: value.substring(lineStart, lineEnd).replace(/^\n/, ''), - // jscs:enable - start: lineStart, - end: lineEnd - }; - }, - - /** - * Set Selection - * - * Set the section of text in the textarea that should be selected by the cursor - * - * @param {number} start - * @param {number} end - */ - setSelection(start, end) { - let $textarea = this.$(); - - if (start === 'end') { - start = $textarea.val().length; - } - - end = end || start; - - $textarea.setSelection(start, end); - }, - - /** - * Replace Selection - * - * @param {String} replacement - the string to replace with - * @param {number} replacementStart - where to start replacing - * @param {number} [replacementEnd] - when to stop replacing, defaults to replacementStart - * @param {String|boolean|Object} [cursorPosition] - where to put the cursor after replacing - * - * Cursor position after replacement defaults to the end of the replacement. - * Providing selectionStart only will cause the cursor to be placed there, or alternatively a range can be selected - * by providing selectionEnd. - */ - replaceSelection(replacement, replacementStart, replacementEnd, cursorPosition) { - run.schedule('afterRender', this, function () { - let $textarea = this.$(); - - cursorPosition = cursorPosition || 'collapseToEnd'; - replacementEnd = replacementEnd || replacementStart; - - $textarea.setSelection(replacementStart, replacementEnd); - - if (['select', 'collapseToStart', 'collapseToEnd'].indexOf(cursorPosition) !== -1) { - $textarea.replaceSelectedText(replacement, cursorPosition); - } else { - $textarea.replaceSelectedText(replacement); - if (cursorPosition.hasOwnProperty('start')) { - $textarea.setSelection(cursorPosition.start, cursorPosition.end); - } else { - $textarea.setSelection(cursorPosition, cursorPosition); - } - } - - $textarea.focus(); - // Tell the editor it has changed, as programmatic replacements won't trigger this automatically - this._elementValueDidChange(); - this.sendAction('onChange'); - }); - } -}); diff --git a/core/client/app/mixins/ed-editor-scroll.js b/core/client/app/mixins/ed-editor-scroll.js deleted file mode 100644 index df483b6006..0000000000 --- a/core/client/app/mixins/ed-editor-scroll.js +++ /dev/null @@ -1,100 +0,0 @@ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -const {Mixin, run} = Ember; - -export default Mixin.create({ - /** - * Determine if the cursor is at the end of the textarea - */ - isCursorAtEnd() { - let selection = this.$().getSelection(); - let value = this.getValue(); - let linesAtEnd = 3; - let match, - stringAfterCursor; - - stringAfterCursor = value.substring(selection.end); - match = stringAfterCursor.match(/\n/g); - - if (!match || match.length < linesAtEnd) { - return true; - } - - return false; - }, - - /** - * Build an object that represents the scroll state - */ - getScrollInfo() { - let scroller = this.get('element'); - let scrollInfo = { - top: scroller.scrollTop, - height: scroller.scrollHeight, - clientHeight: scroller.clientHeight, - diff: scroller.scrollHeight - scroller.clientHeight, - padding: 50, - isCursorAtEnd: this.isCursorAtEnd() - }; - - return scrollInfo; - }, - - /** - * Calculate if we're within scrollInfo.padding of the end of the document, and scroll the rest of the way - */ - adjustScrollPosition() { - // If we're receiving change events from the end of the document, i.e the user is typing-at-the-end, update the - // scroll position to ensure both panels stay in view and in sync - let scrollInfo = this.getScrollInfo(); - - if (scrollInfo.isCursorAtEnd && (scrollInfo.diff >= scrollInfo.top) && - (scrollInfo.diff < scrollInfo.top + scrollInfo.padding)) { - scrollInfo.top += scrollInfo.padding; - // Scroll the left pane - this.$().scrollTop(scrollInfo.top); - } - }, - - /** - * Send the scrollInfo for scrollEvents to the view so that the preview pane can be synced - */ - scrollHandler() { - this.set('scrollThrottle', run.throttle(this, () => { - invokeAction(this, 'updateScrollInfo', this.getScrollInfo()); - }, 10)); - }, - - /** - * once the element is in the DOM bind to the events which control scroll behaviour - */ - attachScrollHandlers() { - let $el = this.$(); - - $el.on('keypress', run.bind(this, this.adjustScrollPosition)); - - $el.on('scroll', run.bind(this, this.scrollHandler)); - }, - - /** - * once the element has been removed from the DOM unbind from the events which control scroll behaviour - */ - detachScrollHandlers() { - this.$().off('keypress'); - this.$().off('scroll'); - run.cancel(this.get('scrollThrottle')); - }, - - didInsertElement() { - this._super(...arguments); - - this.attachScrollHandlers(); - }, - - willDestroyElement() { - this._super(...arguments); - - this.detachScrollHandlers(); - } -}); diff --git a/core/client/app/mixins/ed-editor-shortcuts.js b/core/client/app/mixins/ed-editor-shortcuts.js deleted file mode 100644 index e204f2870b..0000000000 --- a/core/client/app/mixins/ed-editor-shortcuts.js +++ /dev/null @@ -1,173 +0,0 @@ -/* global moment, Showdown */ -import Ember from 'ember'; -import titleize from 'ghost/utils/titleize'; - -const {Mixin} = Ember; - -// Used for simple, noncomputational replace-and-go! shortcuts. -// See default case in shortcut function below. -let simpleShortcutSyntax = { - bold: { - regex: '**|**', - cursor: '|' - }, - italic: { - regex: '*|*', - cursor: '|' - - }, - strike: { - regex: '~~|~~', - cursor: '|' - }, - code: { - regex: '`|`', - cursor: '|' - }, - blockquote: { - regex: '> |', - cursor: '|', - newline: true - }, - list: { - regex: '* |', - cursor: '|', - newline: true - }, - link: { - regex: '[|](http://)', - cursor: 'http://' - }, - image: { - regex: '', - cursor: 'http://', - newline: true - } -}; - -let shortcuts = { - simple(type, replacement, selection, line) { - let startIndex = 0; - let shortcut; - - if (simpleShortcutSyntax.hasOwnProperty(type)) { - shortcut = simpleShortcutSyntax[type]; - // insert the markdown - replacement.text = shortcut.regex.replace('|', selection.text); - - // add a newline if needed - if (shortcut.newline && line.text !== '') { - startIndex = 1; - replacement.text = `\n${replacement.text}`; - } - - // handle cursor position - if (selection.text === '' && shortcut.cursor === '|') { - // the cursor should go where | was - replacement.position = startIndex + replacement.start + shortcut.regex.indexOf(shortcut.cursor); - } else if (shortcut.cursor !== '|') { - // the cursor should select the string which matches shortcut.cursor - replacement.position = { - start: replacement.start + replacement.text.indexOf(shortcut.cursor) - }; - replacement.position.end = replacement.position.start + shortcut.cursor.length; - } - } - - return replacement; - }, - - cycleHeaderLevel(replacement, line) { - let match = line.text.match(/^#+/); - let currentHeaderLevel, - hashPrefix; - - if (!match) { - currentHeaderLevel = 1; - } else { - currentHeaderLevel = match[0].length; - } - - if (currentHeaderLevel > 2) { - currentHeaderLevel = 1; - } - - hashPrefix = new Array(currentHeaderLevel + 2).join('#'); - - replacement.text = `${hashPrefix} ${line.text.replace(/^#* /, '')}`; - - replacement.start = line.start; - replacement.end = line.end; - - return replacement; - }, - - copyHTML(editor, selection) { - let converter = new Showdown.converter(); - let generatedHTML; - - if (selection.text) { - generatedHTML = converter.makeHtml(selection.text); - } else { - generatedHTML = converter.makeHtml(editor.getValue()); - } - - // Talk to the editor - editor.send('toggleCopyHTMLModal', generatedHTML); - }, - - currentDate(replacement) { - replacement.text = moment(new Date()).format('D MMMM YYYY'); - return replacement; - }, - - uppercase(replacement, selection) { - replacement.text = selection.text.toLocaleUpperCase(); - return replacement; - }, - - lowercase(replacement, selection) { - replacement.text = selection.text.toLocaleLowerCase(); - return replacement; - }, - - titlecase(replacement, selection) { - replacement.text = titleize(selection.text); - return replacement; - } -}; - -export default Mixin.create({ - shortcut(type) { - let selection = this.getSelection(); - let replacement = { - start: selection.start, - end: selection.end, - position: 'collapseToEnd' - }; - - switch (type) { - // This shortcut is special as it needs to send an action - case 'copyHTML': - shortcuts.copyHTML(this, selection); - break; - case 'cycleHeaderLevel': - replacement = shortcuts.cycleHeaderLevel(replacement, this.getLine()); - break; - // These shortcuts all process the basic information - case 'currentDate': - case 'uppercase': - case 'lowercase': - case 'titlecase': - replacement = shortcuts[type](replacement, selection, this.getLineToCursor()); - break; - // All the of basic formatting shortcuts work with a regex - default: - replacement = shortcuts.simple(type, replacement, selection, this.getLineToCursor()); - } - - if (replacement.text) { - this.replaceSelection(replacement.text, replacement.start, replacement.end, replacement.position); - } - } -}); diff --git a/core/client/app/mixins/editor-base-controller.js b/core/client/app/mixins/editor-base-controller.js deleted file mode 100644 index 01086f057b..0000000000 --- a/core/client/app/mixins/editor-base-controller.js +++ /dev/null @@ -1,424 +0,0 @@ -import Ember from 'ember'; -import PostModel from 'ghost/models/post'; -import boundOneWay from 'ghost/utils/bound-one-way'; - -const { - Mixin, - RSVP: {resolve}, - computed, - inject: {service, controller}, - observer, - run -} = Ember; -const {alias} = computed; - -// this array will hold properties we need to watch -// to know if the model has been changed (`controller.hasDirtyAttributes`) -const watchedProps = ['model.scratch', 'model.titleScratch', 'model.hasDirtyAttributes', 'model.tags.[]']; - -PostModel.eachAttribute(function (name) { - watchedProps.push(`model.${name}`); -}); - -export default Mixin.create({ - _autoSaveId: null, - _timedSaveId: null, - submitting: false, - - showLeaveEditorModal: false, - showReAuthenticateModal: false, - - postSettingsMenuController: controller('post-settings-menu'), - notifications: service(), - - init() { - this._super(...arguments); - window.onbeforeunload = () => { - return this.get('hasDirtyAttributes') ? this.unloadDirtyMessage() : null; - }; - }, - - shouldFocusTitle: alias('model.isNew'), - shouldFocusEditor: false, - - autoSave: observer('model.scratch', function () { - // Don't save just because we swapped out models - if (this.get('model.isDraft') && !this.get('model.isNew')) { - let autoSaveId, - saveOptions, - timedSaveId; - - saveOptions = { - silent: true, - backgroundSave: true - }; - - timedSaveId = run.throttle(this, 'send', 'save', saveOptions, 60000, false); - this._timedSaveId = timedSaveId; - - autoSaveId = run.debounce(this, 'send', 'save', saveOptions, 3000); - this._autoSaveId = autoSaveId; - } - }), - - /** - * By default, a post will not change its publish state. - * Only with a user-set value (via setSaveType action) - * can the post's status change. - */ - willPublish: boundOneWay('model.isPublished'), - - // set by the editor route and `hasDirtyAttributes`. useful when checking - // whether the number of tags has changed for `hasDirtyAttributes`. - previousTagNames: null, - - tagNames: computed('model.tags.@each.name', function () { - return this.get('model.tags').mapBy('name'); - }), - - postOrPage: computed('model.page', function () { - return this.get('model.page') ? 'Page' : 'Post'; - }), - - // compares previousTagNames to tagNames - tagNamesEqual() { - let tagNames = this.get('tagNames'); - let previousTagNames = this.get('previousTagNames'); - let hashCurrent, - hashPrevious; - - // beware! even if they have the same length, - // that doesn't mean they're the same. - if (tagNames.length !== previousTagNames.length) { - return false; - } - - // instead of comparing with slow, nested for loops, - // perform join on each array and compare the strings - hashCurrent = tagNames.join(''); - hashPrevious = previousTagNames.join(''); - - return hashCurrent === hashPrevious; - }, - - // a hook created in editor-base-route's setupController - modelSaved() { - let model = this.get('model'); - - // safer to updateTags on save in one place - // rather than in all other places save is called - model.updateTags(); - - // set previousTagNames to current tagNames for hasDirtyAttributes check - this.set('previousTagNames', this.get('tagNames')); - - // `updateTags` triggers `hasDirtyAttributes => true`. - // for a saved model it would otherwise be false. - - // if the two "scratch" properties (title and content) match the model, then - // it's ok to set hasDirtyAttributes to false - if (model.get('titleScratch') === model.get('title') && - model.get('scratch') === model.get('markdown')) { - this.set('hasDirtyAttributes', false); - } - }, - - // an ugly hack, but necessary to watch all the model's properties - // and more, without having to be explicit and do it manually - hasDirtyAttributes: computed.apply(Ember, watchedProps.concat({ - get() { - let model = this.get('model'); - - if (!model) { - return false; - } - - let markdown = model.get('markdown'); - let title = model.get('title'); - let titleScratch = model.get('titleScratch'); - let scratch = this.get('model.scratch'); - let changedAttributes; - - if (!this.tagNamesEqual()) { - return true; - } - - if (titleScratch !== title) { - return true; - } - - // since `scratch` is not model property, we need to check - // it explicitly against the model's markdown attribute - if (markdown !== scratch) { - return true; - } - - // if the Adapter failed to save the model isError will be true - // and we should consider the model still dirty. - if (model.get('isError')) { - return true; - } - - // models created on the client always return `hasDirtyAttributes: true`, - // so we need to see which properties have actually changed. - if (model.get('isNew')) { - changedAttributes = Object.keys(model.changedAttributes()); - - if (changedAttributes.length) { - return true; - } - - return false; - } - - // even though we use the `scratch` prop to show edits, - // which does *not* change the model's `hasDirtyAttributes` property, - // `hasDirtyAttributes` will tell us if the other props have changed, - // as long as the model is not new (model.isNew === false). - return model.get('hasDirtyAttributes'); - }, - set(key, value) { - return value; - } - })), - - // used on window.onbeforeunload - unloadDirtyMessage() { - return '==============================\n\n' + - 'Hey there! It looks like you\'re in the middle of writing' + - ' something and you haven\'t saved all of your content.' + - '\n\nSave before you go!\n\n' + - '=============================='; - }, - - // TODO: This has to be moved to the I18n localization file. - // This structure is supposed to be close to the i18n-localization which will be used soon. - messageMap: { - errors: { - post: { - published: { - published: 'Update failed.', - draft: 'Saving failed.' - }, - draft: { - published: 'Publish failed.', - draft: 'Saving failed.' - } - - } - }, - - success: { - post: { - published: { - published: 'Updated.', - draft: 'Saved.' - }, - draft: { - published: 'Published!', - draft: 'Saved.' - } - } - } - }, - - // TODO: Update for new notification click-action API - showSaveNotification(prevStatus, status, delay) { - let message = this.messageMap.success.post[prevStatus][status]; - let notifications = this.get('notifications'); - let type, path; - - if (status === 'published') { - type = this.get('postOrPage'); - path = this.get('model.absoluteUrl'); - } else { - type = 'Preview'; - path = this.get('model.previewUrl'); - } - - message += ` <a href="${path}" target="_blank">View ${type}</a>`; - - notifications.showNotification(message.htmlSafe(), {delayed: delay}); - }, - - showErrorAlert(prevStatus, status, errors, delay) { - let message = this.messageMap.errors.post[prevStatus][status]; - let notifications = this.get('notifications'); - let error; - - function isString(str) { - /*global toString*/ - return toString.call(str) === '[object String]'; - } - - if (errors && isString(errors)) { - error = errors; - } else if (errors && errors[0] && isString(errors[0])) { - error = errors[0]; - } else if (errors && errors[0] && errors[0].message && isString(errors[0].message)) { - error = errors[0].message; - } else { - error = 'Unknown Error'; - } - - message += `<br />${error}`; - message = Ember.String.htmlSafe(message); - - notifications.showAlert(message, {type: 'error', delayed: delay, key: 'post.save'}); - }, - - actions: { - cancelTimers() { - let autoSaveId = this._autoSaveId; - let timedSaveId = this._timedSaveId; - - if (autoSaveId) { - run.cancel(autoSaveId); - this._autoSaveId = null; - } - - if (timedSaveId) { - run.cancel(timedSaveId); - this._timedSaveId = null; - } - }, - - save(options) { - let prevStatus = this.get('model.status'); - let isNew = this.get('model.isNew'); - let psmController = this.get('postSettingsMenuController'); - let promise, status; - - options = options || {}; - - this.toggleProperty('submitting'); - - if (options.backgroundSave) { - // do not allow a post's status to be set to published by a background save - status = 'draft'; - } else { - status = this.get('willPublish') ? 'published' : 'draft'; - } - - this.send('cancelTimers'); - - // Set the properties that are indirected - // set markdown equal to what's in the editor, minus the image markers. - this.set('model.markdown', this.get('model.scratch')); - this.set('model.status', status); - - // Set a default title - if (!this.get('model.titleScratch').trim()) { - this.set('model.titleScratch', '(Untitled)'); - } - - this.set('model.title', this.get('model.titleScratch')); - this.set('model.metaTitle', psmController.get('metaTitleScratch')); - this.set('model.metaDescription', psmController.get('metaDescriptionScratch')); - - if (!this.get('model.slug')) { - // Cancel any pending slug generation that may still be queued in the - // run loop because we need to run it before the post is saved. - run.cancel(psmController.get('debounceId')); - - psmController.generateAndSetSlug('model.slug'); - } - - promise = resolve(psmController.get('lastPromise')).then(() => { - return this.get('model').save(options).then((model) => { - if (!options.silent) { - this.showSaveNotification(prevStatus, model.get('status'), isNew ? true : false); - } - - this.toggleProperty('submitting'); - return model; - }); - }).catch((errors) => { - if (!options.silent) { - errors = errors || this.get('model.errors.messages'); - this.showErrorAlert(prevStatus, this.get('model.status'), errors); - } - - this.set('model.status', prevStatus); - - this.toggleProperty('submitting'); - return this.get('model'); - }); - - psmController.set('lastPromise', promise); - - return promise; - }, - - setSaveType(newType) { - if (newType === 'publish') { - this.set('willPublish', true); - } else if (newType === 'draft') { - this.set('willPublish', false); - } - }, - - autoSaveNew() { - if (this.get('model.isNew')) { - this.send('save', {silent: true, backgroundSave: true}); - } - }, - - toggleLeaveEditorModal(transition) { - this.set('leaveEditorTransition', transition); - this.toggleProperty('showLeaveEditorModal'); - }, - - leaveEditor() { - let transition = this.get('leaveEditorTransition'); - let model = this.get('model'); - - if (!transition) { - this.get('notifications').showAlert('Sorry, there was an error in the application. Please let the Ghost team know what happened.', {type: 'error'}); - return; - } - - // definitely want to clear the data store and post of any unsaved, client-generated tags - model.updateTags(); - - if (model.get('isNew')) { - // the user doesn't want to save the new, unsaved post, so delete it. - model.deleteRecord(); - } else { - // roll back changes on model props - model.rollbackAttributes(); - } - - // setting hasDirtyAttributes to false here allows willTransition on the editor route to succeed - this.set('hasDirtyAttributes', false); - - // since the transition is now certain to complete, we can unset window.onbeforeunload here - window.onbeforeunload = null; - - return transition.retry(); - }, - - updateTitle() { - let currentTitle = this.model.get('title'); - let newTitle = this.model.get('titleScratch').trim(); - - if (currentTitle === newTitle) { - return; - } - - if (this.get('model.isDraft') && !this.get('model.isNew')) { - // this is preferrable to setting hasDirtyAttributes to false manually - this.model.set('title', newTitle); - - this.send('save', { - silent: true, - backgroundSave: true - }); - } - }, - - toggleReAuthenticateModal() { - this.toggleProperty('showReAuthenticateModal'); - } - } -}); diff --git a/core/client/app/mixins/editor-base-route.js b/core/client/app/mixins/editor-base-route.js deleted file mode 100644 index 01d35785d8..0000000000 --- a/core/client/app/mixins/editor-base-route.js +++ /dev/null @@ -1,144 +0,0 @@ -import Ember from 'ember'; -import ShortcutsRoute from 'ghost/mixins/shortcuts-route'; -import styleBody from 'ghost/mixins/style-body'; -import ctrlOrCmd from 'ghost/utils/ctrl-or-cmd'; - -const {$, Mixin, RSVP, run} = Ember; - -let generalShortcuts = {}; -generalShortcuts[`${ctrlOrCmd}+alt+p`] = 'publish'; -generalShortcuts['alt+shift+z'] = 'toggleZenMode'; - -export default Mixin.create(styleBody, ShortcutsRoute, { - classNames: ['editor'], - - shortcuts: generalShortcuts, - - actions: { - save() { - let selectedElement = $(document.activeElement); - - if (selectedElement.is('input[type="text"]')) { - selectedElement.trigger('focusout'); - } - - run.scheduleOnce('actions', this, function () { - this.get('controller').send('save'); - }); - }, - - publish() { - let controller = this.get('controller'); - - controller.send('setSaveType', 'publish'); - controller.send('save'); - }, - - toggleZenMode() { - Ember.$('body').toggleClass('zen'); - }, - - willTransition(transition) { - let controller = this.get('controller'); - let scratch = controller.get('model.scratch'); - let controllerIsDirty = controller.get('hasDirtyAttributes'); - let model = controller.get('model'); - let state = model.getProperties('isDeleted', 'isSaving', 'hasDirtyAttributes', 'isNew'); - let deletedWithoutChanges, - fromNewToEdit; - - // if a save is in-flight we don't know whether or not it's safe to leave - // so we abort the transition and retry after the save has completed. - if (state.isSaving) { - transition.abort(); - return run.later(this, function () { - RSVP.resolve(controller.get('lastPromise')).then(() => { - transition.retry(); - }); - }, 100); - } - - fromNewToEdit = this.get('routeName') === 'editor.new' && - transition.targetName === 'editor.edit' && - transition.intent.contexts && - transition.intent.contexts[0] && - transition.intent.contexts[0].id === model.get('id'); - - deletedWithoutChanges = state.isDeleted && - (state.isSaving || !state.hasDirtyAttributes); - - if (!fromNewToEdit && !deletedWithoutChanges && controllerIsDirty) { - transition.abort(); - controller.send('toggleLeaveEditorModal', transition); - return; - } - - // The controller may hold model state that will be lost in the transition, - // so we need to apply it now. - if (fromNewToEdit && controllerIsDirty) { - if (scratch !== model.get('markdown')) { - model.set('markdown', scratch); - } - } - - if (state.isNew) { - model.deleteRecord(); - } - - // since the transition is now certain to complete.. - window.onbeforeunload = null; - - // remove model-related listeners created in editor-base-route - this.detachModelHooks(controller, model); - } - }, - - renderTemplate(controller, model) { - this._super(controller, model); - - this.render('post-settings-menu', { - model, - into: 'application', - outlet: 'settings-menu' - }); - }, - - attachModelHooks(controller, model) { - // this will allow us to track when the model is saved and update the controller - // so that we can be sure controller.hasDirtyAttributes is correct, without having to update the - // controller on each instance of `model.save()`. - // - // another reason we can't do this on `model.save().then()` is because the post-settings-menu - // also saves the model, and passing messages is difficult because we have two - // types of editor controllers, and the PSM also exists on the posts.post route. - // - // The reason we can't just keep this functionality in the editor controller is - // because we need to remove these handlers on `willTransition` in the editor route. - model.on('didCreate', controller, controller.get('modelSaved')); - model.on('didUpdate', controller, controller.get('modelSaved')); - }, - - detachModelHooks(controller, model) { - model.off('didCreate', controller, controller.get('modelSaved')); - model.off('didUpdate', controller, controller.get('modelSaved')); - }, - - setupController(controller, model) { - let tags = model.get('tags'); - - model.set('scratch', model.get('markdown')); - model.set('titleScratch', model.get('title')); - - this._super(...arguments); - - if (tags) { - // used to check if anything has changed in the editor - controller.set('previousTagNames', tags.mapBy('name')); - } else { - controller.set('previousTagNames', []); - } - - // attach model-related listeners created in editor-base-route - this.attachModelHooks(controller, model); - } -}); diff --git a/core/client/app/mixins/infinite-scroll.js b/core/client/app/mixins/infinite-scroll.js deleted file mode 100644 index 68d9ec706c..0000000000 --- a/core/client/app/mixins/infinite-scroll.js +++ /dev/null @@ -1,43 +0,0 @@ -import Ember from 'ember'; - -const {Mixin, run} = Ember; - -export default Mixin.create({ - isLoading: false, - triggerPoint: 100, - - /** - * Determines if we are past a scroll point where we need to fetch the next page - */ - _checkScroll() { - let element = this.get('element'); - let triggerPoint = this.get('triggerPoint'); - let isLoading = this.get('isLoading'); - - // If we haven't passed our threshold or we are already fetching content, exit - if (isLoading || (element.scrollTop + element.clientHeight + triggerPoint <= element.scrollHeight)) { - return; - } - - this.sendAction('fetch'); - }, - - didInsertElement() { - this._super(...arguments); - - let el = this.get('element'); - - el.onscroll = run.bind(this, this._checkScroll); - - // run on load, on the offchance that the initial load - // did not fill the view. - this._checkScroll(); - }, - - willDestroyElement() { - this._super(...arguments); - - // turn off the scroll handler - this.get('element').onscroll = null; - } -}); diff --git a/core/client/app/mixins/pagination.js b/core/client/app/mixins/pagination.js deleted file mode 100644 index f316f6eb39..0000000000 --- a/core/client/app/mixins/pagination.js +++ /dev/null @@ -1,118 +0,0 @@ -import Ember from 'ember'; -import getRequestErrorMessage from 'ghost/utils/ajax'; - -const { - Mixin, - computed, - RSVP, - inject: {service} -} = Ember; - -let defaultPaginationSettings = { - page: 1, - limit: 15 -}; - -export default Mixin.create({ - notifications: service(), - - paginationModel: null, - paginationSettings: null, - - // add a hook so that routes/controllers can do something with the meta data - paginationMeta: computed({ - get() { - return this._paginationMeta; - }, - set(key, value) { - if (this.didReceivePaginationMeta) { - this.didReceivePaginationMeta(value); - } - this._paginationMeta = value; - return value; - } - }), - - init() { - let paginationSettings = this.get('paginationSettings'); - let settings = Ember.assign({}, defaultPaginationSettings, paginationSettings); - - this._super(...arguments); - this.set('paginationSettings', settings); - this.set('paginationMeta', {}); - }, - - /** - * Takes an ajax response, concatenates any error messages, then generates an error notification. - * @param {jqXHR} response The jQuery ajax reponse object. - * @return - */ - reportLoadError(response) { - let message = 'A problem was encountered while loading more records'; - - if (response) { - // Get message from response - message += `: ${getRequestErrorMessage(response, true)}`; - } else { - message += '.'; - } - - this.get('notifications').showAlert(message, {type: 'error', key: 'pagination.load.failed'}); - }, - - loadFirstPage() { - let paginationSettings = this.get('paginationSettings'); - let modelName = this.get('paginationModel'); - - this.set('paginationSettings.page', 1); - - this.set('isLoading', true); - - return this.get('store').query(modelName, paginationSettings).then((results) => { - this.set('paginationMeta', results.meta); - return results; - }).catch((response) => { - this.reportLoadError(response); - }).finally(() => { - this.set('isLoading', false); - }); - }, - - actions: { - loadFirstPage() { - return this.loadFirstPage(); - }, - - /** - * Loads the next paginated page of posts into the ember-data store. Will cause the posts list UI to update. - * @return - */ - loadNextPage() { - let store = this.get('store'); - let modelName = this.get('paginationModel'); - let metadata = this.get('paginationMeta'); - let nextPage = metadata.pagination && metadata.pagination.next; - let paginationSettings = this.get('paginationSettings'); - - if (nextPage && !this.get('isLoading')) { - this.set('isLoading', true); - this.set('paginationSettings.page', nextPage); - - return store.query(modelName, paginationSettings).then((results) => { - this.set('paginationMeta', results.meta); - return results; - }).catch((response) => { - this.reportLoadError(response); - }).finally(() => { - this.set('isLoading', false); - }); - } else { - return RSVP.resolve([]); - } - }, - - resetPagination() { - this.set('paginationSettings.page', 1); - } - } -}); diff --git a/core/client/app/mixins/settings-menu-controller.js b/core/client/app/mixins/settings-menu-controller.js deleted file mode 100644 index 915dc8c479..0000000000 --- a/core/client/app/mixins/settings-menu-controller.js +++ /dev/null @@ -1,34 +0,0 @@ -import Ember from 'ember'; - -const { - Mixin, - computed, - inject: {controller} -} = Ember; - -export default Mixin.create({ - application: controller(), - - isViewingSubview: computed('application.showSettingsMenu', { - get() { - return false; - }, - set(key, value) { - // Not viewing a subview if we can't even see the PSM - if (!this.get('application.showSettingsMenu')) { - return false; - } - return value; - } - }), - - actions: { - showSubview() { - this.set('isViewingSubview', true); - }, - - closeSubview() { - this.set('isViewingSubview', false); - } - } -}); diff --git a/core/client/app/mixins/settings-save.js b/core/client/app/mixins/settings-save.js deleted file mode 100644 index d666c71520..0000000000 --- a/core/client/app/mixins/settings-save.js +++ /dev/null @@ -1,17 +0,0 @@ -import Ember from 'ember'; - -const {Mixin} = Ember; - -export default Mixin.create({ - submitting: false, - - actions: { - save() { - this.set('submitting', true); - - this.save().then(() => { - this.set('submitting', false); - }); - } - } -}); diff --git a/core/client/app/mixins/shortcuts-route.js b/core/client/app/mixins/shortcuts-route.js deleted file mode 100644 index f80eb5ef1d..0000000000 --- a/core/client/app/mixins/shortcuts-route.js +++ /dev/null @@ -1,49 +0,0 @@ -import Ember from 'ember'; -import ShortcutsMixin from 'ghost/mixins/shortcuts'; - -const {Mixin} = Ember; - -/** - * Only routes can implement shortcuts. - * If you need to trigger actions on the controller, - * simply call them with `this.get('controller').send('action')`. - * - * To implement shortcuts, add this mixin to your `extend()`, - * and implement a `shortcuts` hash. - * In this hash, keys are shortcut combinations and values are route action names. - * (see [keymaster docs](https://github.com/madrobby/keymaster/blob/master/README.markdown)), - * - * ```javascript - * shortcuts: { - * 'ctrl+s, command+s': 'save', - * 'ctrl+alt+z': 'toggleZenMode' - * } - * ``` - * For more complex actions, shortcuts can instead have their value - * be an object like {action, options} - * ```javascript - * shortcuts: { - * 'ctrl+k': {action: 'markdownShortcut', options: 'createLink'} - * } - * ``` - * You can set the scope of your shortcut by passing a scope property. - * ```javascript - * shortcuts : { - * 'enter': {action : 'confirmModal', scope: 'modal'} - * } - * ``` - * If you don't specify a scope, we use a default scope called "default". - * To have all your shortcut work in all scopes, give it the scope "all". - * Find out more at the keymaster docs - */ -export default Mixin.create(ShortcutsMixin, { - activate() { - this._super(...arguments); - this.registerShortcuts(); - }, - - deactivate() { - this._super(...arguments); - this.removeShortcuts(); - } -}); diff --git a/core/client/app/mixins/shortcuts.js b/core/client/app/mixins/shortcuts.js deleted file mode 100644 index 7c0bba40cf..0000000000 --- a/core/client/app/mixins/shortcuts.js +++ /dev/null @@ -1,79 +0,0 @@ -/* global key */ -import Ember from 'ember'; - -const {Mixin, run, typeOf} = Ember; - -// Configure KeyMaster to respond to all shortcuts, -// even inside of -// input, textarea, and select. -key.filter = function () { - return true; -}; - -key.setScope('default'); -/** - * Only routes can implement shortcuts. - * If you need to trigger actions on the controller, - * simply call them with `this.get('controller').send('action')`. - * - * To implement shortcuts, add this mixin to your `extend()`, - * and implement a `shortcuts` hash. - * In this hash, keys are shortcut combinations and values are route action names. - * (see [keymaster docs](https://github.com/madrobby/keymaster/blob/master/README.markdown)), - * - * ```javascript - * shortcuts: { - * 'ctrl+s, command+s': 'save', - * 'ctrl+alt+z': 'toggleZenMode' - * } - * ``` - * For more complex actions, shortcuts can instead have their value - * be an object like {action, options} - * ```javascript - * shortcuts: { - * 'ctrl+k': {action: 'markdownShortcut', options: 'createLink'} - * } - * ``` - * You can set the scope of your shortcut by passing a scope property. - * ```javascript - * shortcuts : { - * 'enter': {action : 'confirmModal', scope: 'modal'} - * } - * ``` - * If you don't specify a scope, we use a default scope called "default". - * To have all your shortcut work in all scopes, give it the scope "all". - * Find out more at the keymaster docs - */ -export default Mixin.create({ - registerShortcuts() { - let shortcuts = this.get('shortcuts'); - - Object.keys(shortcuts).forEach((shortcut) => { - let scope = shortcuts[shortcut].scope || 'default'; - let action = shortcuts[shortcut]; - let options; - - if (typeOf(action) !== 'string') { - options = action.options; - action = action.action; - } - - key(shortcut, scope, (event) => { - // stop things like ctrl+s from actually opening a save dialogue - event.preventDefault(); - run(this, function () { - this.send(action, options); - }); - }); - }); - }, - - removeShortcuts() { - let shortcuts = this.get('shortcuts'); - - Object.keys(shortcuts).forEach((shortcut) => { - let scope = shortcuts[shortcut].scope || 'default'; - key.unbind(shortcut, scope); - }); - } -}); diff --git a/core/client/app/mixins/slug-url.js b/core/client/app/mixins/slug-url.js deleted file mode 100644 index 7bf9fb7559..0000000000 --- a/core/client/app/mixins/slug-url.js +++ /dev/null @@ -1,16 +0,0 @@ -import Ember from 'ember'; - -const {isBlank} = Ember; - -export default Ember.Mixin.create({ - buildURL(_modelName, _id, _snapshot, _requestType, query) { - let url = this._super(...arguments); - - if (query && !isBlank(query.slug)) { - url += `slug/${query.slug}/`; - delete query.slug; - } - - return url; - } -}); diff --git a/core/client/app/mixins/style-body.js b/core/client/app/mixins/style-body.js deleted file mode 100644 index bbda2f116e..0000000000 --- a/core/client/app/mixins/style-body.js +++ /dev/null @@ -1,32 +0,0 @@ -import Ember from 'ember'; - -const {Mixin, run} = Ember; - -// mixin used for routes that need to set a css className on the body tag -export default Mixin.create({ - activate() { - let cssClasses = this.get('classNames'); - - this._super(...arguments); - - if (cssClasses) { - run.schedule('afterRender', null, function () { - cssClasses.forEach((curClass) => { - Ember.$('body').addClass(curClass); - }); - }); - } - }, - - deactivate() { - let cssClasses = this.get('classNames'); - - this._super(...arguments); - - run.schedule('afterRender', null, function () { - cssClasses.forEach((curClass) => { - Ember.$('body').removeClass(curClass); - }); - }); - } -}); diff --git a/core/client/app/mixins/text-input.js b/core/client/app/mixins/text-input.js deleted file mode 100644 index 0c84af81d5..0000000000 --- a/core/client/app/mixins/text-input.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; - -const {Mixin} = Ember; - -export default Mixin.create({ - selectOnClick: false, - stopEnterKeyDownPropagation: false, - - click(event) { - if (this.get('selectOnClick')) { - event.currentTarget.select(); - } - }, - - keyDown(event) { - // stop event propagation when pressing "enter" - // most useful in the case when undesired (global) keyboard shortcuts are getting triggered while interacting - // with this particular input element. - if (this.get('stopEnterKeyDownPropagation') && event.keyCode === 13) { - event.stopPropagation(); - - return true; - } - } -}); diff --git a/core/client/app/mixins/validation-engine.js b/core/client/app/mixins/validation-engine.js deleted file mode 100644 index fdb6e254eb..0000000000 --- a/core/client/app/mixins/validation-engine.js +++ /dev/null @@ -1,162 +0,0 @@ -import Ember from 'ember'; -import DS from 'ember-data'; -import Model from 'ember-data/model'; -import getRequestErrorMessage from 'ghost/utils/ajax'; - -import InviteUserValidator from 'ghost/validators/invite-user'; -import NavItemValidator from 'ghost/validators/nav-item'; -import PostValidator from 'ghost/validators/post'; -import ResetValidator from 'ghost/validators/reset'; -import SettingValidator from 'ghost/validators/setting'; -import SetupValidator from 'ghost/validators/setup'; -import SigninValidator from 'ghost/validators/signin'; -import SignupValidator from 'ghost/validators/signup'; -import SlackIntegrationValidator from 'ghost/validators/slack-integration'; -import SubscriberValidator from 'ghost/validators/subscriber'; -import TagSettingsValidator from 'ghost/validators/tag-settings'; -import UserValidator from 'ghost/validators/user'; - -import ValidatorExtensions from 'ghost/utils/validator-extensions'; - -const {Mixin, RSVP, isArray} = Ember; -const {Errors} = DS; -const emberA = Ember.A; - -// our extensions to the validator library -ValidatorExtensions.init(); - -/** -* The class that gets this mixin will receive these properties and functions. -* It will be able to validate any properties on itself (or the model it passes to validate()) -* with the use of a declared validator. -*/ -export default Mixin.create({ - // these validators can be passed a model to validate when the class that - // mixes in the ValidationEngine declares a validationType equal to a key on this object. - // the model is either passed in via `this.validate({ model: object })` - // or by calling `this.validate()` without the model property. - // in that case the model will be the class that the ValidationEngine - // was mixed into, i.e. the controller or Ember Data model. - validators: { - inviteUser: InviteUserValidator, - navItem: NavItemValidator, - post: PostValidator, - reset: ResetValidator, - setting: SettingValidator, - setup: SetupValidator, - signin: SigninValidator, - signup: SignupValidator, - slackIntegration: SlackIntegrationValidator, - subscriber: SubscriberValidator, - tag: TagSettingsValidator, - user: UserValidator - }, - - // This adds the Errors object to the validation engine, and shouldn't affect - // ember-data models because they essentially use the same thing - errors: null, - - // Store whether a property has been validated yet, so that we know whether or not - // to show error / success validation for a field - hasValidated: null, - - init() { - this._super(...arguments); - this.set('errors', Errors.create()); - this.set('hasValidated', emberA()); - }, - - /** - * Passes the model to the validator specified by validationType. - * Returns a promise that will resolve if validation succeeds, and reject if not. - * Some options can be specified: - * - * `model: Object` - you can specify the model to be validated, rather than pass the default value of `this`, - * the class that mixes in this mixin. - * - * `property: String` - you can specify a specific property to validate. If - * no property is specified, the entire model will be - * validated - */ - validate(opts) { - let model = this; - let hasValidated, - type, - validator; - - opts = opts || {}; - - if (opts.model) { - model = opts.model; - } else if (this instanceof Model) { - model = this; - } else if (this.get('model')) { - model = this.get('model'); - } - - type = this.get('validationType') || model.get('validationType'); - validator = this.get(`validators.${type}`) || model.get(`validators.${type}`); - hasValidated = this.get('hasValidated'); - - opts.validationType = type; - - return new RSVP.Promise((resolve, reject) => { - let passed; - - if (!type || !validator) { - return reject([`The validator specified, "${type}", did not exist!`]); - } - - if (opts.property) { - // If property isn't in `hasValidated`, add it to mark that this field can show a validation result - hasValidated.addObject(opts.property); - model.get('errors').remove(opts.property); - } else { - model.get('errors').clear(); - } - - passed = validator.check(model, opts.property); - - return (passed) ? resolve() : reject(); - }); - }, - - /** - * The primary goal of this method is to override the `save` method on Ember Data models. - * This allows us to run validation before actually trying to save the model to the server. - * You can supply options to be passed into the `validate` method, since the ED `save` method takes no options. - */ - save(options) { - let {_super} = this; - - options = options || {}; - options.wasSave = true; - - // model.destroyRecord() calls model.save() behind the scenes. - // in that case, we don't need validation checks or error propagation, - // because the model itself is being destroyed. - if (this.get('isDeleted')) { - return this._super(...arguments); - } - - // If validation fails, reject with validation errors. - // If save to the server fails, reject with server response. - return this.validate(options).then(() => { - return _super.call(this, options); - }).catch((result) => { - // server save failed or validator type doesn't exist - if (result && !isArray(result)) { - // return the array of errors from the server - result = getRequestErrorMessage(result); - } - - return RSVP.reject(result); - }); - }, - - actions: { - validate(property) { - this.validate({property}); - } - } -}); diff --git a/core/client/app/mixins/validation-state.js b/core/client/app/mixins/validation-state.js deleted file mode 100644 index b2c3d0e8ed..0000000000 --- a/core/client/app/mixins/validation-state.js +++ /dev/null @@ -1,34 +0,0 @@ -import Ember from 'ember'; - -const {Mixin, computed, isEmpty} = Ember; -const emberA = Ember.A; - -export default Mixin.create({ - - errors: null, - property: '', - hasValidated: emberA(), - - hasError: computed('errors.[]', 'property', 'hasValidated.[]', function () { - let property = this.get('property'); - let errors = this.get('errors'); - let hasValidated = this.get('hasValidated'); - - // if we aren't looking at a specific property we always want an error class - if (!property && !isEmpty(errors)) { - return true; - } - - // If we haven't yet validated this field, there is no validation class needed - if (!hasValidated || !hasValidated.contains(property)) { - return false; - } - - if (errors) { - return errors.get(property); - } - - return false; - }) - -}); diff --git a/core/client/app/models/navigation-item.js b/core/client/app/models/navigation-item.js deleted file mode 100644 index 372246105e..0000000000 --- a/core/client/app/models/navigation-item.js +++ /dev/null @@ -1,27 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - computed, - isBlank -} = Ember; - -export default Ember.Object.extend(ValidationEngine, { - label: '', - url: '', - isNew: false, - - validationType: 'navItem', - - isComplete: computed('label', 'url', function () { - let {label, url} = this.getProperties('label', 'url'); - - return !isBlank(label) && !isBlank(url); - }), - - isBlank: computed('label', 'url', function () { - let {label, url} = this.getProperties('label', 'url'); - - return isBlank(label) && isBlank(url); - }) -}); diff --git a/core/client/app/models/notification.js b/core/client/app/models/notification.js deleted file mode 100644 index 7b0c8ffdc1..0000000000 --- a/core/client/app/models/notification.js +++ /dev/null @@ -1,9 +0,0 @@ -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; - -export default Model.extend({ - dismissible: attr('boolean'), - status: attr('string'), - type: attr('string'), - message: attr('string') -}); diff --git a/core/client/app/models/post.js b/core/client/app/models/post.js deleted file mode 100644 index a1c094e032..0000000000 --- a/core/client/app/models/post.js +++ /dev/null @@ -1,87 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; -import { belongsTo, hasMany } from 'ember-data/relationships'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - computed, - inject: {service} -} = Ember; -const {equal} = computed; - -export default Model.extend(ValidationEngine, { - validationType: 'post', - - uuid: attr('string'), - title: attr('string', {defaultValue: ''}), - slug: attr('string'), - markdown: attr('string', {defaultValue: ''}), - html: attr('string'), - image: attr('string'), - featured: attr('boolean', {defaultValue: false}), - page: attr('boolean', {defaultValue: false}), - status: attr('string', {defaultValue: 'draft'}), - language: attr('string', {defaultValue: 'en_US'}), - metaTitle: attr('string'), - metaDescription: attr('string'), - author: belongsTo('user', {async: true}), - authorId: attr('number'), - updatedAt: attr('moment-date'), - updatedBy: attr(), - publishedAt: attr('moment-date'), - publishedBy: belongsTo('user', {async: true}), - createdAt: attr('moment-date'), - createdBy: attr(), - tags: hasMany('tag', { - embedded: 'always', - async: false - }), - url: attr('string'), - - config: service(), - ghostPaths: service(), - - absoluteUrl: computed('url', 'ghostPaths.url', 'config.blogUrl', function () { - let blogUrl = this.get('config.blogUrl'); - let postUrl = this.get('url'); - return this.get('ghostPaths.url').join(blogUrl, postUrl); - }), - - previewUrl: computed('uuid', 'ghostPaths.url', 'config.blogUrl', 'config.routeKeywords.preview', function () { - let blogUrl = this.get('config.blogUrl'); - let uuid = this.get('uuid'); - let previewKeyword = this.get('config.routeKeywords.preview'); - // New posts don't have a preview - if (!uuid) { - return ''; - } - return this.get('ghostPaths.url').join(blogUrl, previewKeyword, uuid); - }), - - scratch: null, - titleScratch: null, - - // Computed post properties - - isPublished: equal('status', 'published'), - isDraft: equal('status', 'draft'), - - // remove client-generated tags, which have `id: null`. - // Ember Data won't recognize/update them automatically - // when returned from the server with ids. - // https://github.com/emberjs/data/issues/1829 - updateTags() { - let tags = this.get('tags'); - let oldTags = tags.filterBy('id', null); - - tags.removeObjects(oldTags); - oldTags.invoke('deleteRecord'); - }, - - isAuthoredByUser(user) { - return parseInt(user.get('id'), 10) === parseInt(this.get('authorId'), 10); - } - -}); diff --git a/core/client/app/models/role.js b/core/client/app/models/role.js deleted file mode 100644 index 6d93ba775b..0000000000 --- a/core/client/app/models/role.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; - -const {computed} = Ember; - -export default Model.extend({ - uuid: attr('string'), - name: attr('string'), - description: attr('string'), - createdAt: attr('moment-date'), - updatedAt: attr('moment-date'), - createdBy: attr(), - updatedBy: attr(), - - lowerCaseName: computed('name', function () { - return this.get('name').toLocaleLowerCase(); - }) -}); diff --git a/core/client/app/models/setting.js b/core/client/app/models/setting.js deleted file mode 100644 index 22a1884944..0000000000 --- a/core/client/app/models/setting.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -export default Model.extend(ValidationEngine, { - validationType: 'setting', - - title: attr('string'), - description: attr('string'), - logo: attr('string'), - cover: attr('string'), - defaultLang: attr('string'), - postsPerPage: attr('number'), - forceI18n: attr('boolean'), - permalinks: attr('string'), - activeTheme: attr('string'), - availableThemes: attr(), - ghost_head: attr('string'), - ghost_foot: attr('string'), - facebook: attr('facebook-url-user'), - twitter: attr('twitter-url-user'), - labs: attr('string'), - navigation: attr('navigation-settings'), - isPrivate: attr('boolean'), - password: attr('string'), - slack: attr('slack-settings') -}); diff --git a/core/client/app/models/slack-integration.js b/core/client/app/models/slack-integration.js deleted file mode 100644 index 6b4931dccb..0000000000 --- a/core/client/app/models/slack-integration.js +++ /dev/null @@ -1,19 +0,0 @@ -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - computed, - isBlank -} = Ember; - -export default Ember.Object.extend(ValidationEngine, { - // values entered here will act as defaults - url: '', - - validationType: 'slackIntegration', - - isActive: computed('url', function () { - let url = this.get('url'); - return !isBlank(url); - }) -}); diff --git a/core/client/app/models/subscriber.js b/core/client/app/models/subscriber.js deleted file mode 100644 index de8a7f90d6..0000000000 --- a/core/client/app/models/subscriber.js +++ /dev/null @@ -1,23 +0,0 @@ -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; -import {belongsTo} from 'ember-data/relationships'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -export default Model.extend(ValidationEngine, { - validationType: 'subscriber', - - uuid: attr('string'), - name: attr('string'), - email: attr('string'), - status: attr('string'), - subscribedUrl: attr('string'), - subscribedReferrer: attr('string'), - unsubscribedUrl: attr('string'), - unsubscribedAt: attr('moment-date'), - createdAt: attr('moment-date'), - updatedAt: attr('moment-date'), - createdBy: attr('number'), - updatedBy: attr('number'), - - post: belongsTo('post') -}); diff --git a/core/client/app/models/tag.js b/core/client/app/models/tag.js deleted file mode 100644 index ef5f03f7a1..0000000000 --- a/core/client/app/models/tag.js +++ /dev/null @@ -1,23 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -export default Model.extend(ValidationEngine, { - validationType: 'tag', - - uuid: attr('string'), - name: attr('string'), - slug: attr('string'), - description: attr('string'), - parent: attr(), - metaTitle: attr('string'), - metaDescription: attr('string'), - image: attr('string'), - hidden: attr('boolean'), - createdAt: attr('moment-date'), - updatedAt: attr('moment-date'), - createdBy: attr(), - updatedBy: attr(), - count: attr('raw') -}); diff --git a/core/client/app/models/user.js b/core/client/app/models/user.js deleted file mode 100644 index e3d3dab14f..0000000000 --- a/core/client/app/models/user.js +++ /dev/null @@ -1,121 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import Model from 'ember-data/model'; -import attr from 'ember-data/attr'; -import { hasMany } from 'ember-data/relationships'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const { - computed, - inject: {service} -} = Ember; -const {equal, empty} = computed; - -export default Model.extend(ValidationEngine, { - validationType: 'user', - - uuid: attr('string'), - name: attr('string'), - slug: attr('string'), - email: attr('string'), - image: attr('string'), - cover: attr('string'), - bio: attr('string'), - website: attr('string'), - location: attr('string'), - accessibility: attr('string'), - status: attr('string'), - language: attr('string', {defaultValue: 'en_US'}), - metaTitle: attr('string'), - metaDescription: attr('string'), - lastLogin: attr('moment-date'), - createdAt: attr('moment-date'), - createdBy: attr('number'), - updatedAt: attr('moment-date'), - updatedBy: attr('number'), - roles: hasMany('role', { - embedded: 'always', - async: false - }), - count: attr('raw'), - facebook: attr('facebook-url-user'), - twitter: attr('twitter-url-user'), - - ghostPaths: service(), - ajax: service(), - - // TODO: Once client-side permissions are in place, - // remove the hard role check. - isAuthor: equal('role.name', 'Author'), - isEditor: equal('role.name', 'Editor'), - isAdmin: equal('role.name', 'Administrator'), - isOwner: equal('role.name', 'Owner'), - - isPasswordValid: empty('passwordValidationErrors.[]'), - - active: computed('status', function () { - return ['active', 'warn-1', 'warn-2', 'warn-3', 'warn-4', 'locked'].indexOf(this.get('status')) > -1; - }), - - invited: computed('status', function () { - return ['invited', 'invited-pending'].indexOf(this.get('status')) > -1; - }), - - pending: equal('status', 'invited-pending'), - - passwordValidationErrors: computed('password', 'newPassword', 'ne2Password', function () { - let validationErrors = []; - - if (!validator.equals(this.get('newPassword'), this.get('ne2Password'))) { - validationErrors.push({message: 'Your new passwords do not match'}); - } - - if (!validator.isLength(this.get('newPassword'), 8)) { - validationErrors.push({message: 'Your password is not long enough. It must be at least 8 characters long.'}); - } - - return validationErrors; - }), - - role: computed('roles', { - get() { - return this.get('roles.firstObject'); - }, - set(key, value) { - // Only one role per user, so remove any old data. - this.get('roles').clear(); - this.get('roles').pushObject(value); - - return value; - } - }), - - saveNewPassword() { - let url = this.get('ghostPaths.url').api('users', 'password'); - - return this.get('ajax').put(url, { - data: { - password: [{ - user_id: this.get('id'), - oldPassword: this.get('password'), - newPassword: this.get('newPassword'), - ne2Password: this.get('ne2Password') - }] - } - }); - }, - - resendInvite() { - let fullUserData = this.toJSON(); - let userData = { - email: fullUserData.email, - roles: fullUserData.roles - }; - let inviteUrl = this.get('ghostPaths.url').api('users'); - - return this.get('ajax').post(inviteUrl, { - data: JSON.stringify({users: [userData]}), - contentType: 'application/json' - }); - } -}); diff --git a/core/client/app/resolver.js b/core/client/app/resolver.js deleted file mode 100644 index 2fb563d6c0..0000000000 --- a/core/client/app/resolver.js +++ /dev/null @@ -1,3 +0,0 @@ -import Resolver from 'ember-resolver'; - -export default Resolver; diff --git a/core/client/app/router.js b/core/client/app/router.js deleted file mode 100644 index 5541d72996..0000000000 --- a/core/client/app/router.js +++ /dev/null @@ -1,70 +0,0 @@ -import Ember from 'ember'; -import ghostPaths from 'ghost/utils/ghost-paths'; -import documentTitle from 'ghost/utils/document-title'; -import config from './config/environment'; - -const { - inject: {service}, - on -} = Ember; - -const Router = Ember.Router.extend({ - location: config.locationType, // use HTML5 History API instead of hash-tag based URLs - rootURL: ghostPaths().adminRoot, // admin interface lives under sub-directory /ghost - - notifications: service(), - - displayDelayedNotifications: on('didTransition', function () { - this.get('notifications').displayDelayed(); - }) -}); - -documentTitle(); - -Router.map(function () { - this.route('setup', function () { - this.route('one'); - this.route('two'); - this.route('three'); - }); - - this.route('signin'); - this.route('signout'); - this.route('signup', {path: '/signup/:token'}); - this.route('reset', {path: '/reset/:token'}); - this.route('about', {path: '/about'}); - - this.route('posts', {path: '/'}, function () { - this.route('post', {path: ':post_id'}); - }); - - this.route('editor', function () { - this.route('new', {path: ''}); - this.route('edit', {path: ':post_id'}); - }); - - this.route('team', {path: '/team'}, function () { - this.route('user', {path: ':user_slug'}); - }); - - this.route('settings.general', {path: '/settings/general'}); - this.route('settings.tags', {path: '/settings/tags'}, function () { - this.route('tag', {path: ':tag_slug'}); - this.route('new'); - }); - this.route('settings.labs', {path: '/settings/labs'}); - this.route('settings.code-injection', {path: '/settings/code-injection'}); - this.route('settings.navigation', {path: '/settings/navigation'}); - this.route('settings.apps', {path: '/settings/apps'}, function () { - this.route('slack', {path: 'slack'}); - }); - - this.route('subscribers', function() { - this.route('new'); - this.route('import'); - }); - - this.route('error404', {path: '/*path'}); -}); - -export default Router; diff --git a/core/client/app/routes/about.js b/core/client/app/routes/about.js deleted file mode 100644 index 36a7a9bc4f..0000000000 --- a/core/client/app/routes/about.js +++ /dev/null @@ -1,36 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import styleBody from 'ghost/mixins/style-body'; - -const { - inject: {service} -} = Ember; - -export default AuthenticatedRoute.extend(styleBody, { - titleToken: 'About', - - classNames: ['view-about'], - - ghostPaths: service(), - ajax: service(), - - cachedConfig: false, - - model() { - let cachedConfig = this.get('cachedConfig'); - let configUrl = this.get('ghostPaths.url').api('configuration', 'about'); - - if (cachedConfig) { - return cachedConfig; - } - - return this.get('ajax').request(configUrl) - .then((configurationResponse) => { - let [cachedConfig] = configurationResponse.configuration; - - this.set('cachedConfig', cachedConfig); - - return cachedConfig; - }); - } -}); diff --git a/core/client/app/routes/application.js b/core/client/app/routes/application.js deleted file mode 100644 index 5753a8b132..0000000000 --- a/core/client/app/routes/application.js +++ /dev/null @@ -1,132 +0,0 @@ -import Ember from 'ember'; -import AuthConfiguration from 'ember-simple-auth/configuration'; -import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin'; -import ShortcutsRoute from 'ghost/mixins/shortcuts-route'; -import ctrlOrCmd from 'ghost/utils/ctrl-or-cmd'; -import windowProxy from 'ghost/utils/window-proxy'; - -const { - Route, - inject: {service}, - run -} = Ember; - -function K() { - return this; -} - -let shortcuts = {}; - -shortcuts.esc = {action: 'closeMenus', scope: 'all'}; -shortcuts[`${ctrlOrCmd}+s`] = {action: 'save', scope: 'all'}; - -export default Route.extend(ApplicationRouteMixin, ShortcutsRoute, { - shortcuts, - - config: service(), - feature: service(), - dropdown: service(), - notifications: service(), - - afterModel(model, transition) { - this._super(...arguments); - - if (this.get('session.isAuthenticated')) { - this.set('appLoadTransition', transition); - transition.send('loadServerNotifications'); - - // return the feature loading promise so that we block until settings - // are loaded in order for synchronous access everywhere - return this.get('feature').fetch(); - } - }, - - title(tokens) { - return `${tokens.join(' - ')} - ${this.get('config.blogTitle')}`; - }, - - sessionAuthenticated() { - if (this.get('session.skipAuthSuccessHandler')) { - return; - } - - // standard ESA post-sign-in redirect - this._super(...arguments); - - // trigger post-sign-in background behaviour - this.get('session.user').then((user) => { - this.send('signedIn', user); - }); - }, - - sessionInvalidated() { - let transition = this.get('appLoadTransition'); - - if (transition) { - transition.send('authorizationFailed'); - } else { - run.scheduleOnce('routerTransitions', this, function () { - this.send('authorizationFailed'); - }); - } - }, - - actions: { - openMobileMenu() { - this.controller.set('showMobileMenu', true); - }, - - openSettingsMenu() { - this.controller.set('showSettingsMenu', true); - }, - - closeMenus() { - this.get('dropdown').closeDropdowns(); - this.controller.setProperties({ - showSettingsMenu: false, - showMobileMenu: false - }); - }, - - didTransition() { - this.set('appLoadTransition', null); - this.send('closeMenus'); - }, - - signedIn() { - this.get('notifications').clearAll(); - this.send('loadServerNotifications', true); - }, - - invalidateSession() { - this.get('session').invalidate().catch((error) => { - this.get('notifications').showAlert(error.message, {type: 'error', key: 'session.invalidate.failed'}); - }); - }, - - authorizationFailed() { - windowProxy.replaceLocation(AuthConfiguration.baseURL); - }, - - loadServerNotifications(isDelayed) { - if (this.get('session.isAuthenticated')) { - this.get('session.user').then((user) => { - if (!user.get('isAuthor') && !user.get('isEditor')) { - this.store.findAll('notification', {reload: true}).then((serverNotifications) => { - serverNotifications.forEach((notification) => { - this.get('notifications').handleNotification(notification, isDelayed); - }); - }); - } - }); - } - }, - - toggleMarkdownHelpModal() { - this.get('controller').toggleProperty('showMarkdownHelpModal'); - }, - - // noop default for unhandled save (used from shortcuts) - save: K - } -}); diff --git a/core/client/app/routes/authenticated.js b/core/client/app/routes/authenticated.js deleted file mode 100644 index 621e9f6acb..0000000000 --- a/core/client/app/routes/authenticated.js +++ /dev/null @@ -1,6 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin'; - -const {Route} = Ember; - -export default Route.extend(AuthenticatedRouteMixin); diff --git a/core/client/app/routes/editor/edit.js b/core/client/app/routes/editor/edit.js deleted file mode 100644 index cd589093e5..0000000000 --- a/core/client/app/routes/editor/edit.js +++ /dev/null @@ -1,65 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import base from 'ghost/mixins/editor-base-route'; -import NotFoundHandler from 'ghost/mixins/404-handler'; -import isNumber from 'ghost/utils/isNumber'; -import isFinite from 'ghost/utils/isFinite'; - -export default AuthenticatedRoute.extend(base, NotFoundHandler, { - titleToken: 'Editor', - - beforeModel(transition) { - this.set('_transitionedFromNew', transition.data.fromNew); - - this._super(...arguments); - }, - - model(params) { - let postId, - query; - - postId = Number(params.post_id); - - if (!isNumber(postId) || !isFinite(postId) || postId % 1 !== 0 || postId <= 0) { - return this.transitionTo('error404', `editor/${params.post_id}`); - } - - query = { - id: postId, - status: 'all', - staticPages: 'all' - }; - - return this.store.query('post', query).then((records) => { - let post = records.get('firstObject'); - - if (post) { - return post; - } - - return this.replaceWith('posts.index'); - }); - }, - - afterModel(post) { - this._super(...arguments); - - return this.get('session.user').then((user) => { - if (user.get('isAuthor') && !post.isAuthoredByUser(user)) { - return this.replaceWith('posts.index'); - } - }); - }, - - setupController(controller) { - this._super(...arguments); - - controller.set('shouldFocusEditor', this.get('_transitionedFromNew')); - }, - - actions: { - authorizationFailed() { - this.get('controller').send('toggleReAuthenticateModal'); - } - } -}); diff --git a/core/client/app/routes/editor/index.js b/core/client/app/routes/editor/index.js deleted file mode 100644 index e395525dcd..0000000000 --- a/core/client/app/routes/editor/index.js +++ /dev/null @@ -1,10 +0,0 @@ -import Ember from 'ember'; - -const {Route} = Ember; - -export default Route.extend({ - beforeModel() { - this._super(...arguments); - this.transitionTo('editor.new'); - } -}); diff --git a/core/client/app/routes/editor/new.js b/core/client/app/routes/editor/new.js deleted file mode 100644 index 0da50705a0..0000000000 --- a/core/client/app/routes/editor/new.js +++ /dev/null @@ -1,50 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import base from 'ghost/mixins/editor-base-route'; - -export default AuthenticatedRoute.extend(base, { - titleToken: 'Editor', - - model() { - return this.get('session.user').then((user) => { - return this.store.createRecord('post', { - author: user - }); - }); - }, - - renderTemplate(controller, model) { - this.render('editor/edit', { - controller, - model - }); - - this.render('post-settings-menu', { - model, - into: 'application', - outlet: 'settings-menu' - }); - }, - - setupController() { - let psm = this.controllerFor('post-settings-menu'); - - // make sure there are no titleObserver functions hanging around - // from previous posts - psm.removeObserver('titleScratch', psm, 'titleObserver'); - - // Ensure that the PSM Publish Date selector resets - psm.send('resetPubDate'); - - this._super(...arguments); - }, - - actions: { - willTransition(transition) { - // decorate the transition object so the editor.edit route - // knows this was the previous active route - transition.data.fromNew = true; - - this._super(...arguments); - } - } -}); diff --git a/core/client/app/routes/error404.js b/core/client/app/routes/error404.js deleted file mode 100644 index d94af975c4..0000000000 --- a/core/client/app/routes/error404.js +++ /dev/null @@ -1,15 +0,0 @@ -import Ember from 'ember'; - -const {Route} = Ember; - -export default Route.extend({ - controllerName: 'error', - templateName: 'error', - titleToken: 'Error', - - model() { - return { - status: 404 - }; - } -}); diff --git a/core/client/app/routes/mobile-index-route.js b/core/client/app/routes/mobile-index-route.js deleted file mode 100644 index c41f07c372..0000000000 --- a/core/client/app/routes/mobile-index-route.js +++ /dev/null @@ -1,37 +0,0 @@ -import Ember from 'ember'; - -const { - Route, - addObserver, - inject: {service}, - removeObserver, - K -} = Ember; - -// Routes that extend MobileIndexRoute need to implement -// desktopTransition, a function which is called when -// the user resizes to desktop levels. -export default Route.extend({ - desktopTransition: K, - _callDesktopTransition: null, - - mediaQueries: service(), - - activate() { - this._super(...arguments); - this._callDesktopTransition = () => { - if (!this.get('mediaQueries.isMobile')) { - this.desktopTransition(); - } - }; - addObserver(this, 'mediaQueries.isMobile', this._callDesktopTransition); - }, - - deactivate() { - this._super(...arguments); - if (this._callDesktopTransition) { - removeObserver(this, 'mediaQueries.isMobile', this._callDesktopTransition); - this._callDesktopTransition = null; - } - } -}); diff --git a/core/client/app/routes/posts.js b/core/client/app/routes/posts.js deleted file mode 100644 index 5edfb161a2..0000000000 --- a/core/client/app/routes/posts.js +++ /dev/null @@ -1,100 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import ShortcutsRoute from 'ghost/mixins/shortcuts-route'; -import PaginationMixin from 'ghost/mixins/pagination'; - -export default AuthenticatedRoute.extend(ShortcutsRoute, PaginationMixin, { - titleToken: 'Content', - - paginationModel: 'post', - paginationSettings: { - status: 'all', - staticPages: 'all' - }, - - model() { - let paginationSettings = this.get('paginationSettings'); - - return this.get('session.user').then((user) => { - if (user.get('isAuthor')) { - paginationSettings.filter = paginationSettings.filter ? - `${paginationSettings.filter}+author:${user.get('slug')}` : `author:${user.get('slug')}`; - } - - return this.loadFirstPage().then(() => { - // using `.filter` allows the template to auto-update when new models are pulled in from the server. - // we just need to 'return true' to allow all models by default. - return this.store.filter('post', (post) => { - if (user.get('isAuthor')) { - return post.isAuthoredByUser(user); - } - - return true; - }); - }); - }); - }, - - stepThroughPosts(step) { - let currentPost = this.get('controller.currentPost'); - let posts = this.get('controller.sortedPosts'); - let length = posts.get('length'); - let newPosition = posts.indexOf(currentPost) + step; - - // if we are on the first or last item - // just do nothing (desired behavior is to not - // loop around) - if (newPosition >= length) { - return; - } else if (newPosition < 0) { - return; - } - - this.transitionTo('posts.post', posts.objectAt(newPosition)); - }, - - scrollContent(amount) { - let content = Ember.$('.js-content-preview'); - let scrolled = content.scrollTop(); - - content.scrollTop(scrolled + 50 * amount); - }, - - shortcuts: { - 'up, k': 'moveUp', - 'down, j': 'moveDown', - left: 'focusList', - right: 'focusContent', - c: 'newPost' - }, - - actions: { - focusList() { - this.controller.set('keyboardFocus', 'postList'); - }, - - focusContent() { - this.controller.set('keyboardFocus', 'postContent'); - }, - - newPost() { - this.transitionTo('editor.new'); - }, - - moveUp() { - if (this.controller.get('postContentFocused')) { - this.scrollContent(-1); - } else { - this.stepThroughPosts(-1); - } - }, - - moveDown() { - if (this.controller.get('postContentFocused')) { - this.scrollContent(1); - } else { - this.stepThroughPosts(1); - } - } - } -}); diff --git a/core/client/app/routes/posts/index.js b/core/client/app/routes/posts/index.js deleted file mode 100644 index 78b5a9e449..0000000000 --- a/core/client/app/routes/posts/index.js +++ /dev/null @@ -1,57 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin'; -import MobileIndexRoute from 'ghost/routes/mobile-index-route'; - -const { - computed, - inject: {service} -} = Ember; -const {reads} = computed; - -export default MobileIndexRoute.extend(AuthenticatedRouteMixin, { - noPosts: false, - - mediaQueries: service(), - isMobile: reads('mediaQueries.isMobile'), - - // Transition to a specific post if we're not on mobile - beforeModel() { - this._super(...arguments); - if (!this.get('isMobile')) { - return this.goToPost(); - } - }, - - setupController(controller) { - controller.set('noPosts', this.get('noPosts')); - this._super(...arguments); - }, - - goToPost() { - // the store has been populated by PostsRoute - let posts = this.store.peekAll('post'); - let post; - - return this.get('session.user').then((user) => { - post = posts.find(function (post) { - // Authors can only see posts they've written - if (user.get('isAuthor')) { - return post.isAuthoredByUser(user); - } - - return true; - }); - - if (post) { - return this.transitionTo('posts.post', post); - } - - this.set('noPosts', true); - }); - }, - - // Mobile posts route callback - desktopTransition() { - this.goToPost(); - } -}); diff --git a/core/client/app/routes/posts/post.js b/core/client/app/routes/posts/post.js deleted file mode 100644 index a7643a21eb..0000000000 --- a/core/client/app/routes/posts/post.js +++ /dev/null @@ -1,77 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import ShortcutsRoute from 'ghost/mixins/shortcuts-route'; -import NotFoundHandler from 'ghost/mixins/404-handler'; -import isNumber from 'ghost/utils/isNumber'; -import isFinite from 'ghost/utils/isFinite'; - -export default AuthenticatedRoute.extend(ShortcutsRoute, NotFoundHandler, { - model(params) { - let post, - postId, - query; - - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - postId = Number(params.post_id); - - if (!isNumber(postId) || !isFinite(postId) || postId % 1 !== 0 || postId <= 0) { - return this.transitionTo('error404', params.post_id); - } - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - - post = this.store.peekRecord('post', postId); - if (post) { - return post; - } - - query = { - id: postId, - status: 'all', - staticPages: 'all' - }; - - return this.store.query('post', query).then((records) => { - let post = records.get('firstObject'); - - if (post) { - return post; - } - - return this.replaceWith('posts.index'); - }); - }, - - afterModel(post) { - return this.get('session.user').then((user) => { - if (user.get('isAuthor') && !post.isAuthoredByUser(user)) { - return this.replaceWith('posts.index'); - } - }); - }, - - setupController(controller, model) { - this._super(controller, model); - - this.controllerFor('posts').set('currentPost', model); - }, - - shortcuts: { - 'enter, o': 'openEditor', - 'command+backspace, ctrl+backspace': 'deletePost' - }, - - actions: { - openEditor(post) { - post = post || this.get('controller.model'); - - if (!post) { - return; - } - - this.transitionTo('editor.edit', post.get('id')); - }, - - deletePost() { - this.controllerFor('posts').send('toggleDeletePostModal'); - } - } -}); diff --git a/core/client/app/routes/reset.js b/core/client/app/routes/reset.js deleted file mode 100644 index 66ffcdc062..0000000000 --- a/core/client/app/routes/reset.js +++ /dev/null @@ -1,33 +0,0 @@ -import Ember from 'ember'; -import Configuration from 'ember-simple-auth/configuration'; -import styleBody from 'ghost/mixins/style-body'; - -const { - Route, - inject: {service} -} = Ember; - -export default Route.extend(styleBody, { - classNames: ['ghost-reset'], - - notifications: service(), - session: service(), - - beforeModel() { - this._super(...arguments); - if (this.get('session.isAuthenticated')) { - this.get('notifications').showAlert('You can\'t reset your password while you\'re signed in.', {type: 'warn', delayed: true, key: 'password.reset.signed-in'}); - this.transitionTo(Configuration.routeAfterAuthentication); - } - }, - - setupController(controller, params) { - controller.token = params.token; - }, - - // Clear out any sensitive information - deactivate() { - this._super(...arguments); - this.controller.clearData(); - } -}); diff --git a/core/client/app/routes/settings/apps.js b/core/client/app/routes/settings/apps.js deleted file mode 100644 index 97d672d2b2..0000000000 --- a/core/client/app/routes/settings/apps.js +++ /dev/null @@ -1,20 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - Apps', - - classNames: ['settings-view-apps'], - - beforeModel() { - this._super(...arguments); - return this.get('session.user') - .then(this.transitionAuthor()) - .then(this.transitionEditor()); - }, - - model() { - return this.store.queryRecord('setting', {type: 'blog,theme,private'}); - } -}); diff --git a/core/client/app/routes/settings/apps/slack.js b/core/client/app/routes/settings/apps/slack.js deleted file mode 100644 index a0fd3801bf..0000000000 --- a/core/client/app/routes/settings/apps/slack.js +++ /dev/null @@ -1,19 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - Apps - Slack', - - classNames: ['settings-view-apps-slack'], - - model() { - return this.modelFor('settings.apps').get('slack.firstObject'); - }, - - setupController(controller) { - this._super(...arguments); - - controller.set('settings', this.modelFor('settings.apps')); - } -}); diff --git a/core/client/app/routes/settings/code-injection.js b/core/client/app/routes/settings/code-injection.js deleted file mode 100644 index b293ad41bd..0000000000 --- a/core/client/app/routes/settings/code-injection.js +++ /dev/null @@ -1,27 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - Code Injection', - classNames: ['settings-view-code'], - - beforeModel() { - this._super(...arguments); - return this.get('session.user') - .then(this.transitionAuthor()) - .then(this.transitionEditor()); - }, - - model() { - return this.store.query('setting', {type: 'blog,theme,private'}).then((records) => { - return records.get('firstObject'); - }); - }, - - actions: { - save() { - this.get('controller').send('save'); - } - } -}); diff --git a/core/client/app/routes/settings/general.js b/core/client/app/routes/settings/general.js deleted file mode 100644 index 56e5be0fc4..0000000000 --- a/core/client/app/routes/settings/general.js +++ /dev/null @@ -1,26 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - General', - - classNames: ['settings-view-general'], - - beforeModel() { - this._super(...arguments); - return this.get('session.user') - .then(this.transitionAuthor()) - .then(this.transitionEditor()); - }, - - model() { - return this.store.queryRecord('setting', {type: 'blog,theme,private'}); - }, - - actions: { - save() { - this.get('controller').send('save'); - } - } -}); diff --git a/core/client/app/routes/settings/labs.js b/core/client/app/routes/settings/labs.js deleted file mode 100644 index 0facca73d3..0000000000 --- a/core/client/app/routes/settings/labs.js +++ /dev/null @@ -1,22 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import styleBody from 'ghost/mixins/style-body'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - Labs', - - classNames: ['settings'], - - beforeModel() { - this._super(...arguments); - return this.get('session.user') - .then(this.transitionAuthor()) - .then(this.transitionEditor()); - }, - - model() { - return this.store.query('setting', {type: 'blog,theme,private'}).then((records) => { - return records.get('firstObject'); - }); - } -}); diff --git a/core/client/app/routes/settings/navigation.js b/core/client/app/routes/settings/navigation.js deleted file mode 100644 index dca80d7b94..0000000000 --- a/core/client/app/routes/settings/navigation.js +++ /dev/null @@ -1,46 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; - -const {$} = Ember; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, { - titleToken: 'Settings - Navigation', - - classNames: ['settings-view-navigation'], - - beforeModel() { - this._super(...arguments); - return this.get('session.user') - .then(this.transitionAuthor()); - }, - - model() { - return this.store.query('setting', {type: 'blog,theme,private'}).then((records) => { - return records.get('firstObject'); - }); - }, - - setupController() { - this._super(...arguments); - this.get('controller').send('reset'); - }, - - actions: { - save() { - // since shortcuts are run on the route, we have to signal to the components - // on the page that we're about to save. - $('.page-actions .btn-blue').focus(); - - this.get('controller').send('save'); - }, - - willTransition() { - // reset the model so that our CPs re-calc and unsaved changes aren't - // persisted across transitions - this.set('controller.model', null); - return this._super(...arguments); - } - } -}); diff --git a/core/client/app/routes/settings/tags.js b/core/client/app/routes/settings/tags.js deleted file mode 100644 index 58fdaec991..0000000000 --- a/core/client/app/routes/settings/tags.js +++ /dev/null @@ -1,104 +0,0 @@ -/* global key */ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import ShortcutsRoute from 'ghost/mixins/shortcuts-route'; -import PaginationMixin from 'ghost/mixins/pagination'; - -export default AuthenticatedRoute.extend(CurrentUserSettings, PaginationMixin, ShortcutsRoute, { - titleToken: 'Settings - Tags', - - paginationModel: 'tag', - paginationSettings: { - include: 'count.posts', - limit: 15 - }, - - shortcuts: { - 'up, k': 'moveUp', - 'down, j': 'moveDown', - left: 'focusList', - right: 'focusContent', - c: 'newTag' - }, - - beforeModel() { - this._super(...arguments); - - return this.get('session.user') - .then(this.transitionAuthor()); - }, - - model() { - return this.loadFirstPage().then(() => { - return this.store.filter('tag', (tag) => { - return !tag.get('isNew'); - }); - }); - }, - - deactivate() { - this._super(...arguments); - this.send('resetShortcutsScope'); - this.send('resetPagination'); - }, - - stepThroughTags(step) { - let currentTag = this.modelFor('settings.tags.tag'); - let tags = this.get('controller.tags'); - let length = tags.get('length'); - - if (currentTag && length) { - let newPosition = tags.indexOf(currentTag) + step; - - if (newPosition >= length) { - return; - } else if (newPosition < 0) { - return; - } - - this.transitionTo('settings.tags.tag', tags.objectAt(newPosition)); - } - }, - - scrollContent(amount) { - let content = Ember.$('.tag-settings-pane'); - let scrolled = content.scrollTop(); - - content.scrollTop(scrolled + 50 * amount); - }, - - actions: { - moveUp() { - if (this.controller.get('tagContentFocused')) { - this.scrollContent(-1); - } else { - this.stepThroughTags(-1); - } - }, - - moveDown() { - if (this.controller.get('tagContentFocused')) { - this.scrollContent(1); - } else { - this.stepThroughTags(1); - } - }, - - focusList() { - this.set('controller.keyboardFocus', 'tagList'); - }, - - focusContent() { - this.set('controller.keyboardFocus', 'tagContent'); - }, - - newTag() { - this.transitionTo('settings.tags.new'); - }, - - resetShortcutsScope() { - key.setScope('default'); - } - } -}); diff --git a/core/client/app/routes/settings/tags/index.js b/core/client/app/routes/settings/tags/index.js deleted file mode 100644 index b86f114db2..0000000000 --- a/core/client/app/routes/settings/tags/index.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; - -const { - inject: {service} -} = Ember; - -export default AuthenticatedRoute.extend({ - mediaQueries: service(), - - beforeModel() { - let firstTag = this.modelFor('settings.tags').get('firstObject'); - - this._super(...arguments); - - if (firstTag && !this.get('mediaQueries.maxWidth600')) { - this.transitionTo('settings.tags.tag', firstTag); - } - } -}); diff --git a/core/client/app/routes/settings/tags/new.js b/core/client/app/routes/settings/tags/new.js deleted file mode 100644 index d19a359cd2..0000000000 --- a/core/client/app/routes/settings/tags/new.js +++ /dev/null @@ -1,21 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; - -export default AuthenticatedRoute.extend({ - - controllerName: 'settings.tags.tag', - - model() { - return this.store.createRecord('tag'); - }, - - renderTemplate() { - this.render('settings.tags.tag'); - }, - - // reset the model so that mobile screens react to an empty selectedTag - deactivate() { - this._super(...arguments); - this.set('controller.model', null); - } - -}); diff --git a/core/client/app/routes/settings/tags/tag.js b/core/client/app/routes/settings/tags/tag.js deleted file mode 100644 index 6bfba4cb06..0000000000 --- a/core/client/app/routes/settings/tags/tag.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import NotFoundHandler from 'ghost/mixins/404-handler'; - -export default AuthenticatedRoute.extend(NotFoundHandler, { - - model(params) { - return this.store.queryRecord('tag', {slug: params.tag_slug}); - }, - - serialize(model) { - return {tag_slug: model.get('slug')}; - }, - - // reset the model so that mobile screens react to an empty selectedTag - deactivate() { - this._super(...arguments); - this.set('controller.model', null); - } -}); diff --git a/core/client/app/routes/setup.js b/core/client/app/routes/setup.js deleted file mode 100644 index 4f11e75ca2..0000000000 --- a/core/client/app/routes/setup.js +++ /dev/null @@ -1,59 +0,0 @@ -import Ember from 'ember'; -import Configuration from 'ember-simple-auth/configuration'; -import styleBody from 'ghost/mixins/style-body'; - -const { - Route, - inject: {service} -} = Ember; - -export default Route.extend(styleBody, { - titleToken: 'Setup', - - classNames: ['ghost-setup'], - - ghostPaths: service('ghost-paths'), - session: service(), - ajax: service(), - - // use the beforeModel hook to check to see whether or not setup has been - // previously completed. If it has, stop the transition into the setup page. - beforeModel() { - this._super(...arguments); - - if (this.get('session.isAuthenticated')) { - this.transitionTo(Configuration.routeIfAlreadyAuthenticated); - return; - } - - let authUrl = this.get('ghostPaths.url').api('authentication', 'setup'); - - // If user is not logged in, check the state of the setup process via the API - return this.get('ajax').request(authUrl) - .then((result) => { - let [setup] = result.setup; - - if (setup.status) { - return this.transitionTo('signin'); - } else { - let controller = this.controllerFor('setup/two'); - if (setup.title) { - controller.set('blogTitle', setup.title.replace(/'/gim, '\'')); - } - - if (setup.name) { - controller.set('name', setup.name.replace(/'/gim, '\'')); - } - - if (setup.email) { - controller.set('email', setup.email); - } - } - }); - }, - - deactivate() { - this._super(...arguments); - this.controllerFor('setup/two').set('password', ''); - } -}); diff --git a/core/client/app/routes/setup/index.js b/core/client/app/routes/setup/index.js deleted file mode 100644 index 2b6133aedb..0000000000 --- a/core/client/app/routes/setup/index.js +++ /dev/null @@ -1,10 +0,0 @@ -import Ember from 'ember'; - -const {Route} = Ember; - -export default Route.extend({ - beforeModel() { - this._super(...arguments); - this.transitionTo('setup.one'); - } -}); diff --git a/core/client/app/routes/setup/one.js b/core/client/app/routes/setup/one.js deleted file mode 100644 index 234228bc53..0000000000 --- a/core/client/app/routes/setup/one.js +++ /dev/null @@ -1,64 +0,0 @@ -import Ember from 'ember'; -import AjaxService from 'ember-ajax/services/ajax'; - -const { - Route, - inject: {service}, - run -} = Ember; - -let DownloadCountPoller = Ember.Object.extend({ - url: null, - count: '', - runId: null, - - ajax: AjaxService.create(), - - init() { - this._super(...arguments); - this.downloadCounter(); - this.poll(); - }, - - poll() { - let interval = Ember.testing ? 20 : 2000; - let runId = run.later(this, function () { - this.downloadCounter(); - if (!Ember.testing) { - this.poll(); - } - }, interval); - - this.set('runId', runId); - }, - - downloadCounter() { - this.get('ajax').request(this.get('url')).then((data) => { - let pattern = /(-?\d+)(\d{3})/; - let count = data.count.toString(); - - while (pattern.test(count)) { - count = count.replace(pattern, '$1,$2'); - } - - this.set('count', count); - }).catch(() => { - this.set('count', ''); - }); - } -}); - -export default Route.extend({ - ghostPaths: service('ghost-paths'), - - model() { - return DownloadCountPoller.create({url: this.get('ghostPaths.count')}); - }, - - resetController(controller, isExiting) { - if (isExiting) { - run.cancel(controller.get('model.runId')); - controller.set('model', null); - } - } -}); diff --git a/core/client/app/routes/setup/three.js b/core/client/app/routes/setup/three.js deleted file mode 100644 index 25e22b1cc6..0000000000 --- a/core/client/app/routes/setup/three.js +++ /dev/null @@ -1,12 +0,0 @@ -import Ember from 'ember'; - -const {Route} = Ember; - -export default Route.extend({ - beforeModel() { - this._super(...arguments); - if (!this.controllerFor('setup.two').get('blogCreated')) { - this.transitionTo('setup.two'); - } - } -}); diff --git a/core/client/app/routes/signin.js b/core/client/app/routes/signin.js deleted file mode 100644 index 89c978762c..0000000000 --- a/core/client/app/routes/signin.js +++ /dev/null @@ -1,45 +0,0 @@ -import Ember from 'ember'; -import styleBody from 'ghost/mixins/style-body'; -import Configuration from 'ember-simple-auth/configuration'; -import DS from 'ember-data'; - -const { - Route, - inject: {service} -} = Ember; -const {Errors} = DS; - -export default Route.extend(styleBody, { - titleToken: 'Sign In', - - classNames: ['ghost-login'], - - session: service(), - - beforeModel() { - this._super(...arguments); - - if (this.get('session.isAuthenticated')) { - this.transitionTo(Configuration.routeIfAlreadyAuthenticated); - } - }, - - model() { - return Ember.Object.create({ - identification: '', - password: '', - errors: Errors.create() - }); - }, - - // the deactivate hook is called after a route has been exited. - deactivate() { - let controller = this.controllerFor('signin'); - - this._super(...arguments); - - // clear the properties that hold the credentials when we're no longer on the signin screen - controller.set('model.identification', ''); - controller.set('model.password', ''); - } -}); diff --git a/core/client/app/routes/signout.js b/core/client/app/routes/signout.js deleted file mode 100644 index ab1a1abea8..0000000000 --- a/core/client/app/routes/signout.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import styleBody from 'ghost/mixins/style-body'; - -const { - canInvoke, - inject: {service} -} = Ember; - -export default AuthenticatedRoute.extend(styleBody, { - titleToken: 'Sign Out', - - classNames: ['ghost-signout'], - - notifications: service(), - - afterModel(model, transition) { - this.get('notifications').clearAll(); - if (canInvoke(transition, 'send')) { - transition.send('invalidateSession'); - } else { - this.send('invalidateSession'); - } - } -}); diff --git a/core/client/app/routes/signup.js b/core/client/app/routes/signup.js deleted file mode 100644 index c1f6d0d82c..0000000000 --- a/core/client/app/routes/signup.js +++ /dev/null @@ -1,77 +0,0 @@ -import Ember from 'ember'; -import DS from 'ember-data'; -import Configuration from 'ember-simple-auth/configuration'; -import styleBody from 'ghost/mixins/style-body'; - -const { - Route, - RSVP: {Promise}, - inject: {service} -} = Ember; -const {Errors} = DS; - -export default Route.extend(styleBody, { - classNames: ['ghost-signup'], - - ghostPaths: service('ghost-paths'), - notifications: service(), - session: service(), - ajax: service(), - - beforeModel() { - this._super(...arguments); - - if (this.get('session.isAuthenticated')) { - this.get('notifications').showAlert('You need to sign out to register as a new user.', {type: 'warn', delayed: true, key: 'signup.create.already-authenticated'}); - this.transitionTo(Configuration.routeIfAlreadyAuthenticated); - } - }, - - model(params) { - let model = Ember.Object.create(); - let re = /^(?:[A-Za-z0-9_\-]{4})*(?:[A-Za-z0-9_\-]{2}|[A-Za-z0-9_\-]{3})?$/; - let email, - tokenText; - - return new Promise((resolve) => { - if (!re.test(params.token)) { - this.get('notifications').showAlert('Invalid token.', {type: 'error', delayed: true, key: 'signup.create.invalid-token'}); - - return resolve(this.transitionTo('signin')); - } - - tokenText = atob(params.token); - email = tokenText.split('|')[1]; - - model.set('email', email); - model.set('token', params.token); - model.set('errors', Errors.create()); - - let authUrl = this.get('ghostPaths.url').api('authentication', 'invitation'); - - return this.get('ajax').request(authUrl, { - dataType: 'json', - data: { - email - } - }).then((response) => { - if (response && response.invitation && response.invitation[0].valid === false) { - this.get('notifications').showAlert('The invitation does not exist or is no longer valid.', {type: 'warn', delayed: true, key: 'signup.create.invalid-invitation'}); - - return resolve(this.transitionTo('signin')); - } - - resolve(model); - }).catch(() => { - resolve(model); - }); - }); - }, - - deactivate() { - this._super(...arguments); - - // clear the properties that hold the sensitive data from the controller - this.controllerFor('signup').setProperties({email: '', password: '', token: ''}); - } -}); diff --git a/core/client/app/routes/subscribers.js b/core/client/app/routes/subscribers.js deleted file mode 100644 index 76ed5febd4..0000000000 --- a/core/client/app/routes/subscribers.js +++ /dev/null @@ -1,54 +0,0 @@ -import Ember from 'ember'; -import AuthenticatedRoute from 'ghost/routes/authenticated'; - -const { - RSVP, - inject: {service} -} = Ember; - -export default AuthenticatedRoute.extend({ - titleToken: 'Subscribers', - - feature: service(), - - // redirect if subscribers is disabled or user isn't owner/admin - beforeModel() { - this._super(...arguments); - let promises = { - user: this.get('session.user'), - subscribers: this.get('feature.subscribers') - }; - - return RSVP.hash(promises).then((hash) => { - let {user, subscribers} = hash; - - if (!subscribers || !(user.get('isOwner') || user.get('isAdmin'))) { - return this.transitionTo('posts'); - } - }); - }, - - setupController(controller) { - this._super(...arguments); - controller.initializeTable(); - controller.send('loadFirstPage'); - }, - - resetController(controller, isExiting) { - this._super(...arguments); - if (isExiting) { - controller.set('order', 'created_at'); - controller.set('direction', 'desc'); - } - }, - - actions: { - addSubscriber(subscriber) { - this.get('controller').send('addSubscriber', subscriber); - }, - - reset() { - this.get('controller').send('reset'); - } - } -}); diff --git a/core/client/app/routes/subscribers/import.js b/core/client/app/routes/subscribers/import.js deleted file mode 100644 index 1c9fa472e1..0000000000 --- a/core/client/app/routes/subscribers/import.js +++ /dev/null @@ -1,9 +0,0 @@ -import Ember from 'ember'; - -export default Ember.Route.extend({ - actions: { - cancel() { - this.transitionTo('subscribers'); - } - } -}); diff --git a/core/client/app/routes/subscribers/new.js b/core/client/app/routes/subscribers/new.js deleted file mode 100644 index 86665899d4..0000000000 --- a/core/client/app/routes/subscribers/new.js +++ /dev/null @@ -1,37 +0,0 @@ -import Ember from 'ember'; - -export default Ember.Route.extend({ - model() { - return this.get('store').createRecord('subscriber'); - }, - - deactivate() { - let subscriber = this.controller.get('model'); - - this._super(...arguments); - - if (subscriber.get('isNew')) { - this.rollbackModel(); - } - }, - - rollbackModel() { - let subscriber = this.controller.get('model'); - subscriber.rollbackAttributes(); - }, - - actions: { - save() { - let subscriber = this.controller.get('model'); - return subscriber.save().then((saved) => { - this.send('addSubscriber', saved); - return saved; - }); - }, - - cancel() { - this.rollbackModel(); - this.transitionTo('subscribers'); - } - } -}); diff --git a/core/client/app/routes/team/index.js b/core/client/app/routes/team/index.js deleted file mode 100644 index f96514b8ba..0000000000 --- a/core/client/app/routes/team/index.js +++ /dev/null @@ -1,32 +0,0 @@ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import PaginationMixin from 'ghost/mixins/pagination'; -import styleBody from 'ghost/mixins/style-body'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, PaginationMixin, { - titleToken: 'Team', - - classNames: ['view-team'], - - paginationModel: 'user', - paginationSettings: { - status: 'active', - limit: 20 - }, - - model() { - this.loadFirstPage(); - - return this.store.query('user', {limit: 'all', status: 'invited'}).then(() => { - return this.store.filter('user', () => { - return true; - }); - }); - }, - - actions: { - reload() { - this.refresh(); - } - } -}); diff --git a/core/client/app/routes/team/user.js b/core/client/app/routes/team/user.js deleted file mode 100644 index a6f464ea69..0000000000 --- a/core/client/app/routes/team/user.js +++ /dev/null @@ -1,58 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import AuthenticatedRoute from 'ghost/routes/authenticated'; -import CurrentUserSettings from 'ghost/mixins/current-user-settings'; -import styleBody from 'ghost/mixins/style-body'; -import NotFoundHandler from 'ghost/mixins/404-handler'; - -export default AuthenticatedRoute.extend(styleBody, CurrentUserSettings, NotFoundHandler, { - titleToken: 'Team - User', - - classNames: ['team-view-user'], - - model(params) { - return this.store.queryRecord('user', {slug: params.user_slug, include: 'count.posts'}); - }, - - serialize(model) { - return {user_slug: model.get('slug')}; - }, - - afterModel(user) { - this._super(...arguments); - - return this.get('session.user').then((currentUser) => { - let isOwnProfile = user.get('id') === currentUser.get('id'); - let isAuthor = currentUser.get('isAuthor'); - let isEditor = currentUser.get('isEditor'); - - if (isAuthor && !isOwnProfile) { - this.transitionTo('team.user', currentUser); - } else if (isEditor && !isOwnProfile && !user.get('isAuthor')) { - this.transitionTo('team'); - } - }); - }, - - deactivate() { - let model = this.modelFor('team.user'); - - // we want to revert any unsaved changes on exit - if (model && model.get('hasDirtyAttributes')) { - model.rollbackAttributes(); - } - - model.get('errors').clear(); - - this._super(...arguments); - }, - - actions: { - didTransition() { - this.modelFor('team.user').get('errors').clear(); - }, - - save() { - this.get('controller').send('save'); - } - } -}); diff --git a/core/client/app/serializers/application.js b/core/client/app/serializers/application.js deleted file mode 100644 index 9d15cf539d..0000000000 --- a/core/client/app/serializers/application.js +++ /dev/null @@ -1,27 +0,0 @@ -import Ember from 'ember'; -import RESTSerializer from 'ember-data/serializers/rest'; - -const { - decamelize -} = Ember.String; - -export default RESTSerializer.extend({ - serializeIntoHash(hash, type, record, options) { - // Our API expects an id on the posted object - options = options || {}; - options.includeId = true; - - // We have a plural root in the API - let root = Ember.String.pluralize(type.modelName); - let data = this.serialize(record, options); - - // Don't ever pass uuid's - delete data.uuid; - - hash[root] = [data]; - }, - - keyForAttribute(attr) { - return decamelize(attr); - } -}); diff --git a/core/client/app/serializers/post.js b/core/client/app/serializers/post.js deleted file mode 100644 index e80f54ff85..0000000000 --- a/core/client/app/serializers/post.js +++ /dev/null @@ -1,56 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import ApplicationSerializer from 'ghost/serializers/application'; -import EmbeddedRecordsMixin from 'ember-data/serializers/embedded-records-mixin'; - -export default ApplicationSerializer.extend(EmbeddedRecordsMixin, { - // settings for the EmbeddedRecordsMixin. - attrs: { - tags: {embedded: 'always'} - }, - - normalize(model, hash, prop) { - // this is to enable us to still access the raw authorId - // without requiring an extra get request (since it is an - // async relationship). - if ((prop === 'post' || prop === 'posts') && hash.author !== undefined) { - hash.author_id = hash.author; - } - - return this._super(...arguments); - }, - - normalizeSingleResponse(store, primaryModelClass, payload) { - let root = this.keyForAttribute(primaryModelClass.modelName); - let pluralizedRoot = Ember.String.pluralize(primaryModelClass.modelName); - - payload[root] = payload[pluralizedRoot][0]; - delete payload[pluralizedRoot]; - - return this._super(...arguments); - }, - - normalizeArrayResponse() { - return this._super(...arguments); - }, - - serializeIntoHash(hash, type, record, options) { - options = options || {}; - options.includeId = true; - - // We have a plural root in the API - let root = Ember.String.pluralize(type.modelName); - let data = this.serialize(record, options); - - // Properties that exist on the model but we don't want sent in the payload - - delete data.uuid; - delete data.html; - // Inserted locally as a convenience. - delete data.author_id; - // Read-only virtual property. - delete data.url; - - hash[root] = [data]; - } -}); diff --git a/core/client/app/serializers/setting.js b/core/client/app/serializers/setting.js deleted file mode 100644 index 634ae484c2..0000000000 --- a/core/client/app/serializers/setting.js +++ /dev/null @@ -1,46 +0,0 @@ -import Ember from 'ember'; -import ApplicationSerializer from 'ghost/serializers/application'; - -export default ApplicationSerializer.extend({ - serializeIntoHash(hash, type, record, options) { - // Settings API does not want ids - options = options || {}; - options.includeId = false; - - let root = Ember.String.pluralize(type.modelName); - let data = this.serialize(record, options); - let payload = []; - - delete data.id; - - Object.keys(data).forEach((k) => { - payload.push({key: k, value: data[k]}); - }); - - hash[root] = payload; - }, - - normalizeArrayResponse(store, primaryModelClass, _payload, id, requestType) { - let payload = {settings: [this._extractObjectFromArrayPayload(_payload)]}; - return this._super(store, primaryModelClass, payload, id, requestType); - }, - - normalizeSingleResponse(store, primaryModelClass, _payload, id, requestType) { - let payload = {setting: this._extractObjectFromArrayPayload(_payload)}; - return this._super(store, primaryModelClass, payload, id, requestType); - }, - - keyForAttribute(attr) { - return attr; - }, - - _extractObjectFromArrayPayload(_payload) { - let payload = {id: '0'}; - - _payload.settings.forEach((setting) => { - payload[setting.key] = setting.value; - }); - - return payload; - } -}); diff --git a/core/client/app/serializers/tag.js b/core/client/app/serializers/tag.js deleted file mode 100644 index 4e33b03cbe..0000000000 --- a/core/client/app/serializers/tag.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import ApplicationSerializer from 'ghost/serializers/application'; - -export default ApplicationSerializer.extend({ - serializeIntoHash(hash, type, record, options) { - options = options || {}; - options.includeId = true; - - let root = Ember.String.pluralize(type.modelName); - let data = this.serialize(record, options); - - // Properties that exist on the model but we don't want sent in the payload - - delete data.uuid; - delete data.count; - - hash[root] = [data]; - } -}); diff --git a/core/client/app/serializers/user.js b/core/client/app/serializers/user.js deleted file mode 100644 index 79587bbe02..0000000000 --- a/core/client/app/serializers/user.js +++ /dev/null @@ -1,29 +0,0 @@ -import Ember from 'ember'; -import ApplicationSerializer from 'ghost/serializers/application'; -import EmbeddedRecordsMixin from 'ember-data/serializers/embedded-records-mixin'; - -export default ApplicationSerializer.extend(EmbeddedRecordsMixin, { - attrs: { - roles: {embedded: 'always'} - }, - - extractSingle(store, primaryType, payload) { - let root = this.keyForAttribute(primaryType.modelName); - let pluralizedRoot = Ember.String.pluralize(primaryType.modelName); - - payload[root] = payload[pluralizedRoot][0]; - delete payload[pluralizedRoot]; - - return this._super(...arguments); - }, - - normalizeSingleResponse(store, primaryModelClass, payload) { - let root = this.keyForAttribute(primaryModelClass.modelName); - let pluralizedRoot = Ember.String.pluralize(primaryModelClass.modelName); - - payload[root] = payload[pluralizedRoot][0]; - delete payload[pluralizedRoot]; - - return this._super(...arguments); - } -}); diff --git a/core/client/app/services/ajax.js b/core/client/app/services/ajax.js deleted file mode 100644 index c8558bf14f..0000000000 --- a/core/client/app/services/ajax.js +++ /dev/null @@ -1,72 +0,0 @@ -import Ember from 'ember'; -import AjaxService from 'ember-ajax/services/ajax'; -import {AjaxError} from 'ember-ajax/errors'; - -const {inject, computed} = Ember; - -export function RequestEntityTooLargeError(errors) { - AjaxError.call(this, errors, 'Request was rejected because it\'s larger than the maximum file size the server allows'); -} - -export function UnsupportedMediaTypeError(errors) { - AjaxError.call(this, errors, 'Request was rejected because it contains an unknown or unsupported file type.'); -} - -// TODO: remove once upgraded to ember-ajax 2.0 -export function NotFoundError(errors) { - AjaxError.call(this, errors, 'Resource was not found.'); -} - -NotFoundError.prototype = Object.create(AjaxError.prototype); - -export default AjaxService.extend({ - session: inject.service(), - - headers: computed('session.isAuthenticated', function () { - let session = this.get('session'); - - if (session.get('isAuthenticated')) { - let headers = {}; - - session.authorize('authorizer:oauth2', (headerName, headerValue) => { - headers[headerName] = headerValue; - }); - - return headers; - } else { - return []; - } - }), - - handleResponse(status, headers, payload) { - if (this.isRequestEntityTooLarge(status, headers, payload)) { - return new RequestEntityTooLargeError(payload.errors); - } else if (this.isUnsupportedMediaType(status, headers, payload)) { - return new UnsupportedMediaTypeError(payload.errors); - } else if (this.isNotFoundError(status, headers, payload)) { - return new NotFoundError(payload.errors); - } - - return this._super(...arguments); - }, - - normalizeErrorResponse(status, headers, payload) { - if (payload && typeof payload === 'object') { - payload.errors = payload.error || payload.errors || payload.message || undefined; - } - - return this._super(status, headers, payload); - }, - - isRequestEntityTooLarge(status/*, headers, payload */) { - return status === 413; - }, - - isUnsupportedMediaType(status/*, headers, payload */) { - return status === 415; - }, - - isNotFoundError(status) { - return status === 404; - } -}); diff --git a/core/client/app/services/config.js b/core/client/app/services/config.js deleted file mode 100644 index b71b3bc2d8..0000000000 --- a/core/client/app/services/config.js +++ /dev/null @@ -1,44 +0,0 @@ -import Ember from 'ember'; - -const {Service, _ProxyMixin, computed} = Ember; - -function isNumeric(num) { - return Ember.$.isNumeric(num); -} - -function _mapType(val, type) { - if (val === '') { - return null; - } else if (type === 'bool') { - return (val === 'true') ? true : false; - } else if (type === 'int' && isNumeric(val)) { - return +val; - } else if (type === 'json') { - try { - return JSON.parse(val); - } catch (e) { - return val; - } - } else { // assume string if type is null or matches nothing else - return val; - } -} - -export default Service.extend(_ProxyMixin, { - content: computed(function () { - let metaConfigTags = Ember.$('meta[name^="env-"]'); - let config = {}; - - metaConfigTags.each((i, el) => { - let key = el.name; - let value = el.content; - let type = el.getAttribute('data-type'); - - let propertyName = key.substring(4); - - config[propertyName] = _mapType(value, type); - }); - - return config; - }) -}); diff --git a/core/client/app/services/dropdown.js b/core/client/app/services/dropdown.js deleted file mode 100644 index 4941faeac0..0000000000 --- a/core/client/app/services/dropdown.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; -// This is used by the dropdown initializer (and subsequently popovers) to manage closing & toggling -import BodyEventListener from 'ghost/mixins/body-event-listener'; - -const {Service, Evented} = Ember; - -export default Service.extend(Evented, BodyEventListener, { - bodyClick(event) { - /*jshint unused:false */ - this.closeDropdowns(); - }, - - closeDropdowns() { - this.trigger('close'); - }, - - toggleDropdown(dropdownName, dropdownButton) { - this.trigger('toggle', {target: dropdownName, button: dropdownButton}); - } -}); diff --git a/core/client/app/services/feature.js b/core/client/app/services/feature.js deleted file mode 100644 index fca3ef705a..0000000000 --- a/core/client/app/services/feature.js +++ /dev/null @@ -1,84 +0,0 @@ -import Ember from 'ember'; - -const { - Service, - computed, - inject: {service}, - set -} = Ember; - -const EmberError = Ember.Error; - -export function feature(name) { - return computed(`config.${name}`, `labs.${name}`, { - get() { - if (this.get(`config.${name}`)) { - return this.get(`config.${name}`); - } - - return this.get(`labs.${name}`) || false; - }, - set(key, value) { - this.update(key, value); - return value; - } - }); -} - -export default Service.extend({ - store: service(), - config: service(), - notifications: service(), - - publicAPI: feature('publicAPI'), - subscribers: feature('subscribers'), - - _settings: null, - - labs: computed('_settings.labs', function () { - let labs = this.get('_settings.labs'); - - try { - return JSON.parse(labs) || {}; - } catch (e) { - return {}; - } - }), - - fetch() { - return this.get('store').queryRecord('setting', {type: 'blog,theme,private'}).then((settings) => { - this.set('_settings', settings); - return true; - }); - }, - - update(key, value) { - let settings = this.get('_settings'); - let labs = this.get('labs'); - - // set the new labs key value - set(labs, key, value); - // update the 'labs' key of the settings model - settings.set('labs', JSON.stringify(labs)); - - return settings.save().then(() => { - // return the labs key value that we get from the server - this.notifyPropertyChange('labs'); - return this.get(`labs.${key}`); - - }).catch((errors) => { - settings.rollbackAttributes(); - this.notifyPropertyChange('labs'); - - // we'll always have an errors object unless we hit a - // validation error - if (!errors) { - throw new EmberError(`Validation of the feature service settings model failed when updating labs.`); - } - - this.get('notifications').showErrors(errors); - - return this.get(`labs.${key}`); - }); - } -}); diff --git a/core/client/app/services/ghost-paths.js b/core/client/app/services/ghost-paths.js deleted file mode 100644 index 620c250d9a..0000000000 --- a/core/client/app/services/ghost-paths.js +++ /dev/null @@ -1,8 +0,0 @@ -import Ember from 'ember'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -const {Service, _ProxyMixin} = Ember; - -export default Service.extend(_ProxyMixin, { - content: ghostPaths() -}); diff --git a/core/client/app/services/media-queries.js b/core/client/app/services/media-queries.js deleted file mode 100644 index 2414ad7a2a..0000000000 --- a/core/client/app/services/media-queries.js +++ /dev/null @@ -1,48 +0,0 @@ -import Ember from 'ember'; - -const {Service, run} = Ember; - -const MEDIA_QUERIES = { - maxWidth600: '(max-width: 600px)', - isMobile: '(max-width: 800px)', - maxWidth900: '(max-width: 900px)', - maxWidth1000: '(max-width: 1000px)' -}; - -export default Service.extend({ - init() { - this._super(...arguments); - this._handlers = []; - this.loadQueries(MEDIA_QUERIES); - }, - - loadQueries(queries) { - Object.keys(queries).forEach((key) => { - this.loadQuery(key, queries[key]); - }); - }, - - loadQuery(key, queryString) { - let query = window.matchMedia(queryString); - - this.set(key, query.matches); - - let handler = run.bind(this, () => { - let lastValue = this.get(key); - let newValue = query.matches; - if (lastValue !== newValue) { - this.set(key, query.matches); - } - }); - query.addListener(handler); - this._handlers.push([query, handler]); - }, - - willDestroy() { - this._handlers.forEach(([query, handler]) => { - query.removeListener(handler); - }); - this._super(...arguments); - } - -}); diff --git a/core/client/app/services/notifications.js b/core/client/app/services/notifications.js deleted file mode 100644 index 09666650b4..0000000000 --- a/core/client/app/services/notifications.js +++ /dev/null @@ -1,187 +0,0 @@ -import Ember from 'ember'; -import {AjaxError} from 'ember-ajax/errors'; - -const { - Service, - computed: {filter}, - get, - set, - isArray -} = Ember; -const emberA = Ember.A; - -// Notification keys take the form of "noun.verb.message", eg: -// -// "invite.resend.api-error" -// "user.invite.already-invited" -// -// The "noun.verb" part will be used as the "key base" in duplicate checks -// to avoid stacking of multiple error messages whilst leaving enough -// specificity to re-use keys for i18n lookups - -export default Service.extend({ - delayedNotifications: emberA(), - content: emberA(), - - alerts: filter('content', function (notification) { - let status = get(notification, 'status'); - return status === 'alert'; - }), - - notifications: filter('content', function (notification) { - let status = get(notification, 'status'); - return status === 'notification'; - }), - - handleNotification(message, delayed) { - // If this is an alert message from the server, treat it as html safe - if (typeof message.toJSON === 'function' && message.get('status') === 'alert') { - message.set('message', message.get('message').htmlSafe()); - } - - if (!get(message, 'status')) { - set(message, 'status', 'notification'); - } - - // close existing duplicate alerts/notifications to avoid stacking - if (get(message, 'key')) { - this._removeItems(get(message, 'status'), get(message, 'key')); - } - - if (!delayed) { - this.get('content').pushObject(message); - } else { - this.get('delayedNotifications').pushObject(message); - } - }, - - showAlert(message, options) { - options = options || {}; - - this.handleNotification({ - message, - status: 'alert', - type: options.type, - key: options.key - }, options.delayed); - }, - - showNotification(message, options) { - options = options || {}; - - if (!options.doNotCloseNotifications) { - this.closeNotifications(); - } else { - // TODO: this should be removed along with showErrors - options.key = undefined; - } - - this.handleNotification({ - message, - status: 'notification', - type: options.type, - key: options.key - }, options.delayed); - }, - - // TODO: review whether this can be removed once no longer used by validations - showErrors(errors, options) { - options = options || {}; - options.type = options.type || 'error'; - // TODO: getting keys from the server would be useful here (necessary for i18n) - options.key = (options.key && `${options.key}.api-error`) || 'api-error'; - - if (!options.doNotCloseNotifications) { - this.closeNotifications(); - } - - // ensure all errors that are passed in get shown - options.doNotCloseNotifications = true; - - for (let i = 0; i < errors.length; i += 1) { - this.showNotification(errors[i].message || errors[i], options); - } - }, - - showAPIError(resp, options) { - options = options || {}; - options.type = options.type || 'error'; - // TODO: getting keys from the server would be useful here (necessary for i18n) - options.key = (options.key && `${options.key}.api-error`) || 'api-error'; - - if (!options.doNotCloseNotifications) { - this.closeNotifications(); - } - - options.defaultErrorText = options.defaultErrorText || 'There was a problem on the server, please try again.'; - - if (resp instanceof AjaxError) { - resp = resp.errors; - } - - if (resp && isArray(resp) && resp.length) { // Array of errors - this.showErrors(resp, options); - } else if (resp && resp.detail) { // ember-ajax provided error message - this.showAlert(resp.detail, options); - } else { // text error or no error - this.showAlert(resp || options.defaultErrorText, options); - } - }, - - displayDelayed() { - this.delayedNotifications.forEach((message) => { - this.get('content').pushObject(message); - }); - this.delayedNotifications = []; - }, - - closeNotification(notification) { - let content = this.get('content'); - - if (typeof notification.toJSON === 'function') { - notification.deleteRecord(); - notification.save().finally(() => { - content.removeObject(notification); - }); - } else { - content.removeObject(notification); - } - }, - - closeNotifications(key) { - this._removeItems('notification', key); - }, - - closeAlerts(key) { - this._removeItems('alert', key); - }, - - clearAll() { - this.get('content').clear(); - }, - - _removeItems(status, key) { - if (key) { - let keyBase = this._getKeyBase(key); - // TODO: keys should only have . special char but we should - // probably use a better regexp escaping function/polyfill - let escapedKeyBase = keyBase.replace('.', '\\.'); - let keyRegex = new RegExp(`^${escapedKeyBase}`); - - this.set('content', this.get('content').reject((item) => { - let itemKey = get(item, 'key'); - let itemStatus = get(item, 'status'); - - return itemStatus === status && (itemKey && itemKey.match(keyRegex)); - })); - } else { - this.set('content', this.get('content').rejectBy('status', status)); - } - }, - - // take a key and return the first two elements, eg: - // "invite.revoke.failed" => "invite.revoke" - _getKeyBase(key) { - return key.split('.').slice(0, 2).join('.'); - } -}); diff --git a/core/client/app/services/session.js b/core/client/app/services/session.js deleted file mode 100644 index bb63134491..0000000000 --- a/core/client/app/services/session.js +++ /dev/null @@ -1,24 +0,0 @@ -import Ember from 'ember'; -import SessionService from 'ember-simple-auth/services/session'; - -const { - computed, - inject: {service} -} = Ember; - -export default SessionService.extend({ - store: service(), - feature: service(), - - user: computed(function () { - return this.get('store').findRecord('user', 'me'); - }), - - authenticate() { - return this._super(...arguments).then((authResult) => { - return this.get('feature').fetch().then(() => { - return authResult; - }); - }); - } -}); diff --git a/core/client/app/services/slug-generator.js b/core/client/app/services/slug-generator.js deleted file mode 100644 index d835f8983e..0000000000 --- a/core/client/app/services/slug-generator.js +++ /dev/null @@ -1,29 +0,0 @@ -import Ember from 'ember'; - -const { - RSVP: {resolve}, - inject: {service}, - Service -} = Ember; - -export default Service.extend({ - ghostPaths: service(), - ajax: service(), - - generateSlug(slugType, textToSlugify) { - let url; - - if (!textToSlugify) { - return resolve(''); - } - - url = this.get('ghostPaths.url').api('slugs', slugType, encodeURIComponent(textToSlugify)); - - return this.get('ajax').request(url).then((response) => { - let [firstSlug] = response.slugs; - let {slug} = firstSlug; - - return slug; - }); - } -}); diff --git a/core/client/app/session-stores/application.js b/core/client/app/session-stores/application.js deleted file mode 100644 index 61920f330c..0000000000 --- a/core/client/app/session-stores/application.js +++ /dev/null @@ -1,10 +0,0 @@ -import AdaptiveStore from 'ember-simple-auth/session-stores/adaptive'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -const paths = ghostPaths(); -const keyName = `ghost${(paths.subdir.indexOf('/') === 0 ? `-${paths.subdir.substr(1)}` : ``) }:session`; - -export default AdaptiveStore.extend({ - localStorageKey: keyName, - cookieName: keyName -}); diff --git a/core/client/app/styles/app.css b/core/client/app/styles/app.css deleted file mode 100644 index e02694df9e..0000000000 --- a/core/client/app/styles/app.css +++ /dev/null @@ -1,48 +0,0 @@ -/* Stop: Normalize. -/* ---------------------------------------------------------- */ -@import "../../bower_components/normalize.css/normalize.css"; - - -/* Patterns: Groups of Styles -/* ---------------------------------------------------------- */ -@import "patterns/global.css"; -@import "patterns/_shame.css"; -@import "patterns/icons.css"; -@import "patterns/forms.css"; -@import "patterns/buttons.css"; -@import "patterns/labels.css"; -@import "patterns/tables.css"; -@import "patterns/navlist.css"; - - -/* Components: Groups of Patterns -/* ---------------------------------------------------------- */ -@import "components/modals.css"; -@import "components/notifications.css"; -@import "components/uploader.css"; -@import "components/splitbuttons.css"; -@import "components/dropdowns.css"; -@import "components/pagination.css"; -@import "components/badges.css"; -@import "components/popovers.css"; -@import "components/settings-menu.css"; -@import "components/selectize.css"; -@import "components/power-select.css"; - - -/* Layouts: Groups of Components -/* ---------------------------------------------------------- */ -@import "layouts/main.css"; -@import "layouts/flow.css"; -@import "layouts/auth.css"; -@import "layouts/content.css"; -@import "layouts/editor.css"; -@import "layouts/settings.css"; -@import "layouts/users.css"; -@import "layouts/user.css"; -@import "layouts/about.css"; -@import "layouts/tags.css"; -@import "layouts/error.css"; -@import "layouts/apps.css"; -@import "layouts/packages.css"; -@import "layouts/subscribers.css"; diff --git a/core/client/app/styles/components/badges.css b/core/client/app/styles/components/badges.css deleted file mode 100644 index 202f6f4eee..0000000000 --- a/core/client/app/styles/components/badges.css +++ /dev/null @@ -1,74 +0,0 @@ -/* Badges -/* ---------------------------------------------------------- */ - -.badge { - display: inline-block; - padding: 2px 4px; - min-width: 10px; - background-color: #777; - border-radius: 10px; - box-shadow: 0 0 0 1px #777; - color: #fff; - vertical-align: baseline; - text-align: center; - white-space: nowrap; - font-size: 1rem; - line-height: 1; - font-weight: inherit; -} - -.badge:empty { - display: none; -} - -.btn .badge { - position: relative; - top: -1px; -} - -.btn-xs .badge { - top: 0; - padding: 1px 5px; -} - -p .badge { - position: relative; - top: -2px; - margin-left: -1px; - padding: 0 4px 3px; -} - -.nav-pills > li > a > .badge { - margin-left: 3px; -} - -.nav-item .badge { - margin-top: -5px; - margin-left: 3px; -} - -a.badge:hover, -a.badge:focus { - color: #fff; - text-decoration: none; - cursor: pointer; -} - - -/* Colour variations -/* ---------------------------------------------------------- */ - -.badge.badge-blue { - background-color: var(--blue); - box-shadow: 0 0 0 1px var(--blue); -} - -.badge.badge-green { - background-color: var(--green); - box-shadow: 0 0 0 1px var(--green); -} - -.badge.badge-red { - background-color: var(--red); - box-shadow: 0 0 0 1px var(--red); -} diff --git a/core/client/app/styles/components/dropdowns.css b/core/client/app/styles/components/dropdowns.css deleted file mode 100644 index adc6d76867..0000000000 --- a/core/client/app/styles/components/dropdowns.css +++ /dev/null @@ -1,149 +0,0 @@ -/* Dropdowns -/* ---------------------------------------------------------- */ - -.dropdown { - position: relative; - z-index: 1000; -} - -.dropdown-toggle:focus { - outline: 0; -} - -.dropdown-menu { - position: absolute; - top: 100%; - left: 0; - float: left; - margin: 2px 0 0; - padding: 10px; - min-width: 200px; - border: #dfe1e3 1px solid; - background-color: #fff; - background-clip: padding-box; - border-radius: 4px; - box-shadow: rgba(0, 0, 0, 0.10) 0 2px 6px; - list-style: none; - text-align: left; - text-transform: none; - letter-spacing: 0; - font-size: 1.4rem; - font-weight: normal; -} - -.dropdown-menu.pull-right { - right: 0; - left: auto; -} - -.dropdown-menu .divider { - overflow: hidden; - margin: 8px 0; - height: 1px; - background: color(#dfe1e3 lightness(+5%)); -} - -.dropdown-menu > li > a, -.dropdown-menu > li > button { - display: flex; - align-items: center; - clear: both; - padding: 6px 10px; - width: 100%; - border-radius: 3px; - color: color(var(--darkgrey) lightness(+20%)); - text-align: left; - white-space: nowrap; - font-size: 1.3rem; - line-height: 1em; - font-weight: normal; - transition: none; -} - -.dropdown-menu i { - margin-right: 10px; - font-size: 14px; - line-height: 1em; -} - -@media (max-width: 500px) { - .dropdown-menu > li > a, - .dropdown-menu > li > button { - padding: 7px 8px; - font-size: 1.5rem; - } - .dropdown-menu i { - font-size: 16px; - } -} - - -/* States -/* ---------------------------------------------------------- */ - -/* Hover/Focus */ -.dropdown-menu > li > a:hover, -.dropdown-menu > li > a:focus, -.dropdown-menu > li > button:hover, -.dropdown-menu > li > button:focus { - background: color(var(--blue) alpha(-85%)); - color: var(--darkgrey); - text-decoration: none; -} - -/* Active */ -.dropdown-menu > .active > a, -.dropdown-menu > .active > a:hover, -.dropdown-menu > .active > a:focus, -.dropdown-menu > .active > button, -.dropdown-menu > .active > button:hover, -.dropdown-menu > .active > button:focus { - outline: 0; - background-color: #428bca; - color: #fff; - text-decoration: none; -} - -/* Disabled */ -.dropdown-menu > .disabled > a, -.dropdown-menu > .disabled > a:hover, -.dropdown-menu > .disabled > a:focus, -.dropdown-menu > .disabled > button, -.dropdown-menu > .disabled > button:hover, -.dropdown-menu > .disabled > button:focus { - color: #777; -} - -.dropdown-menu > .disabled > a:hover, -.dropdown-menu > .disabled > a:focus, -.dropdown-menu > .disabled > button:hover, -.dropdown-menu > .disabled > button:focus { - background-color: transparent; - background-image: none; - text-decoration: none; - cursor: not-allowed; -} - - -/* Open / Close -/* ---------------------------------------------------------- */ - -.open > .dropdown-menu { - display: block; -} - -.open > a { - outline: 0; -} - -.closed > .dropdown-menu { - display: none; -} - - -/* Selectize -/* ---------------------------------------------------------- */ - -.selectize-dropdown { - z-index: 200; -} diff --git a/core/client/app/styles/components/modals.css b/core/client/app/styles/components/modals.css deleted file mode 100644 index fd52dbe1a7..0000000000 --- a/core/client/app/styles/components/modals.css +++ /dev/null @@ -1,183 +0,0 @@ -/* Modals -/* ---------------------------------------------------------- */ - - -/* Fullscreen Modal -/* ---------------------------------------------------------- */ - -.fullscreen-modal-liquid-target { - overflow-y: auto; - height: 100vh; -} - -.fullscreen-modal-background { - position: fixed; - top: 0; - right: 0; - bottom: 0; - left: 0; - z-index: 0; - background: rgba(0, 0, 0, 0.6); -} - -.fullscreen-modal { - padding-top: 30px; - padding-bottom: 30px; - max-width: 550px; - width: 100%; - pointer-events: auto; -} - -@media (max-width: 900px) { - .fullscreen-modal { - padding: 10px; - } -} - -/* Modifiers -/* ---------------------------------------------------------- */ - -.fullscreen-modal-wide { - width: 550px; -} - -@media (max-width: 900px) { - .fullscreen-modal-wide { - width: 100%; - } -} - -.fullscreen-modal-action { - padding: 60px 0 30px; -} - -@media (max-width: 900px) { - .fullscreen-modal-action { - padding: 30px 0; - } -} - - -/* The modal -/* ---------------------------------------------------------- */ - - -/* Modal content -/* ---------------------------------------------------------- */ - -.modal-content { - position: relative; - padding: 18px; - background-color: #fff; - background-clip: padding-box; - border-radius: var(--border-radius); - box-shadow: rgba(0, 0, 0, 0.2) 0 0 0 6px; -} - -.modal-content * { - user-select: text; -} - -.modal-content .close { - position: absolute; - top: 19px; - right: 19px; - z-index: 9999; - margin: 0; - padding: 0; - width: 16px; - border: none; - color: #808284; -} - -.modal-content .close:hover { - color: var(--darkgrey); -} - -.modal-header { - position: relative; - margin-bottom: 18px; -} - -.modal-header h1 { - display: inline-block; - margin: 0 25px 0 0; - font-size: 1.85em; - font-weight: 100; -} - -.modal-body { - position: relative; - overflow-y: auto; -} - -.modal-body .red { - color: var(--red); -} - -.modal-body > *:first-child { - margin-top: 0; -} - -.modal-body > *:last-child { - margin-bottom: 0; -} - -.modal-footer { - display: flex; - justify-content: flex-end; - margin-top: 20px; -} - -.modal-footer button { - margin-left: 8px; - min-width: 100px; - text-align: center; -} - -.modal-footer button:first-of-type { - margin-left: 0; -} - -.modal-body .gh-image-uploader { - margin: 0; -} - - -/* Content Modifiers -/* ---------------------------------------------------------- */ - - -/* Login styles */ -.modal-body .login-form { - display: block; -} - -.modal-body .login-form .password-wrap input { - width: 100%; -} - -@media (max-width: 900px) { - .modal-body .login-form { - margin: 0 auto; - max-width: 264px; - } - .modal-body .login-form .password-wrap { - margin: 0 auto 1em; - width: 100%; - } - .modal-body .login-form .btn { - margin: 0; - margin-bottom: 1em; - width: 100%; - } -} - -@media (min-width: 901px) { - .modal-body .login-form { - display: flex; - } - .modal-body .login-form .password-wrap { - flex: 1; - } -} diff --git a/core/client/app/styles/components/notifications.css b/core/client/app/styles/components/notifications.css deleted file mode 100644 index 2c4f641edf..0000000000 --- a/core/client/app/styles/components/notifications.css +++ /dev/null @@ -1,219 +0,0 @@ -/* Notifications -/* ---------------------------------------------------------- */ - -/* Base notification style */ -.gh-notifications { - position: fixed; - bottom: 20px; - left: 20px; - z-index: 7000; - display: flex; - flex-direction: column; -} - -/* Base notification style */ -.gh-notification { - position: relative; - display: flex; - margin-top: 5px; - padding: 4px; - width: 220px; - border: #dfe1e3 1px solid; - background: rgba(255,255,255,0.9); - border-radius: 4px; - box-shadow: rgba(0,0,0,0.06) 0 1px 7px; - color: #808284; - font-size: 1.2rem; - line-height: 1.4em; -} - -.gh-notification:hover { - cursor: pointer; -} - -.gh-notification-content { - flex-grow: 1; - padding: 10px 15px; - border-radius: 3px; - transition: background 0.2s ease; -} - -.gh-notification em { - color: var(--blue); - font-style: normal; -} - -.gh-notification:hover .gh-notification-content { - background: color(var(--blue) lightness(+34%)); -} - -.gh-notification-close { - position: absolute; - top: 0; - right: 0; - padding: 6px 6px 5px 5px; - background: #fff; - border-radius: 0 4px 0 4px; - font-size: 7px; - line-height: 5px; -} - -.gh-notification-close:hover { - background: #fff; - color: var(--red); -} - -.gh-notification-passive { - animation: fade-out; - animation-delay: 5s; - animation-iteration-count: 1; -} - -.gh-notification-passive:hover { - animation: fade-in; -} - -/* Red notification -/* ---------------------------------------------------------- */ - -.gh-notification-red { - border: color(var(--red) lightness(+28%)) 1px solid; -} - -.gh-notification-red em { - color: var(--red); -} - -.gh-notification-red:hover .gh-notification-content { - background: color(var(--red) lightness(+40%)); -} - -/* Green notification -/* ---------------------------------------------------------- */ - -.gh-notification-green { - border: color(var(--green) lightness(+28%)) 1px solid; -} - -.gh-notification-green em { - color: var(--green); -} - -.gh-notification-green:hover .gh-notification-content { - background: color(var(--green) lightness(+40%)); -} - -/* Alerts -/* ---------------------------------------------------------- */ - -/* Alert wrapper, top of screen */ -.gh-alerts { - flex-shrink: 0; - display: flex; - flex-direction: column; -} - -/* Base alert style */ -.gh-alert { - z-index: 1000; - flex-grow: 1; - display: flex; - justify-content: space-between; - align-items: center; - padding: 14px 15px; - border-bottom: #dfe1e3 1px solid; -} - -.gh-alert-content { - font-size: 1.4rem; - line-height: 1.3em; - font-weight: 200; - user-select: text; -} - -.gh-alert a { - text-decoration: underline; - font-weight: 400; - user-select: text; -} - -.gh-alert-close { - flex-shrink: 0; - margin-left: 20px; - padding: 5px; - font-size: 10px; - line-height: 10px; -} - -.gh-alert-close:hover { - color: var(--red); -} - - -/* Blue alert -/* ---------------------------------------------------------- */ - -.gh-alert-blue { - border-bottom: color(var(--blue) lightness(-10%)) 1px solid; - background: var(--blue); - color: #fff; -} -.gh-alert-blue a { - color: #fff; -} -.gh-alert-blue .gh-alert-close:hover { - color: #fff; -} - -/* Red alert -/* ---------------------------------------------------------- */ - -.gh-alert-red { - border-bottom: color(var(--red) lightness(-10%)) 1px solid; - background: var(--red); - color: #fff; -} -.gh-alert-red a { - color: #fff; -} -.gh-alert-red .gh-alert-close:hover { - color: #fff; -} - -/* Green alert -/* ---------------------------------------------------------- */ - -.gh-alert-green { - border-bottom: color(var(--green) lightness(-7%)) 1px solid; - background: var(--green); - color: #fff; -} -.gh-alert-green a { - color: #fff; -} -.gh-alert-green .gh-alert-close:hover { - color: #fff; -} - -/* Black alert -/* ---------------------------------------------------------- */ - -.gh-alert-black { - border-bottom: color(var(--darkgrey) lightness(-10%)) 1px solid; - background: var(--darkgrey); - color: #fff; -} -.gh-alert-black a { - color: #fff; -} -.gh-alert-black .gh-alert-close:hover { - color: #fff; -} - -/* Yellow alert -/* ---------------------------------------------------------- */ - -.gh-alert-yellow { - border-bottom: #e9ebb6 1px solid; - background: #fdffb6; -} diff --git a/core/client/app/styles/components/pagination.css b/core/client/app/styles/components/pagination.css deleted file mode 100644 index 882fca7506..0000000000 --- a/core/client/app/styles/components/pagination.css +++ /dev/null @@ -1,111 +0,0 @@ -/* Pagination -/* ---------------------------------------------------------- */ - -.pagination { - display: inline-block; - margin: 20px 0; - padding-left: 0; - border-radius: var(--border-radius); -} - -.pagination > li { - display: inline; -} - -.pagination > li > a, -.pagination > li > span { - position: relative; - float: left; - margin-left: -1px; - padding: 6px 12px; - border: 1px solid #dfe1e3; - background-color: #fff; - color: var(--blue); - text-decoration: none; - line-height: 1.42857143; -} - -.pagination > li:first-child > a, -.pagination > li:first-child > span { - margin-left: 0; - border-top-left-radius: var(--border-radius); - border-bottom-left-radius: var(--border-radius); -} - -.pagination > li:last-child > a, -.pagination > li:last-child > span { - border-top-right-radius: var(--border-radius); - border-bottom-right-radius: var(--border-radius); -} - -.pagination > li > a:hover, -.pagination > li > a:focus, -.pagination > li > span:hover, -.pagination > li > span:focus { - background-color: #eee; - color: #2a6496; -} - -.pagination > .active > a, -.pagination > .active > a:hover, -.pagination > .active > a:focus, -.pagination > .active > span, -.pagination > .active > span:hover, -.pagination > .active > span:focus { - z-index: 2; - background-color: #428bca; - color: #fff; - cursor: default; -} - -.pagination > .disabled > span, -.pagination > .disabled > span:hover, -.pagination > .disabled > span:focus, -.pagination > .disabled > a, -.pagination > .disabled > a:hover, -.pagination > .disabled > a:focus { - border-color: #ddd; - background-color: #fff; - color: #777; - cursor: not-allowed; -} - - -/* Sizing -/* ---------------------------------------------------------- */ - -.pagination-lg > li > a, -.pagination-lg > li > span { - padding: 10px 16px; - font-size: 18px; -} - -.pagination-lg > li:first-child > a, -.pagination-lg > li:first-child > span { - border-top-left-radius: 6px; - border-bottom-left-radius: 6px; -} - -.pagination-lg > li:last-child > a, -.pagination-lg > li:last-child > span { - border-top-right-radius: 6px; - border-bottom-right-radius: 6px; -} - -.pagination-sm > li > a, -.pagination-sm > li > span { - padding: 5px 10px; - font-size: 12px; -} - -.pagination-sm > li:first-child > a, -.pagination-sm > li:first-child > span { - border-top-left-radius: 3px; - border-bottom-left-radius: 3px; -} - -.pagination-sm > li:last-child > a, -.pagination-sm > li:last-child > span { - border-top-right-radius: 3px; - border-bottom-right-radius: 3px; -} diff --git a/core/client/app/styles/components/popovers.css b/core/client/app/styles/components/popovers.css deleted file mode 100644 index ba456af432..0000000000 --- a/core/client/app/styles/components/popovers.css +++ /dev/null @@ -1,59 +0,0 @@ -/* Popovers -/* ---------------------------------------------------------- */ - -.popover-item { - position: relative; - display: inline-block; - padding: 11px 26px 13px 16px; - min-width: 300px; - max-width: 400px; - background: var(--darkgrey); - border-radius: 6px; - color: var(--midgrey); - font-size: 1.2rem; -} - -.popover-title { - color: #fff; - font-size: 1.4rem; - font-weight: 300; -} - -.popover-desc { - margin-top: -4px; -} - -.popover-body { - margin-top: 11px; - line-height: 1.7; -} - -.popover-body b { - color: #fff; -} - -.popover-body > *:last-child { - margin: 0; -} - - -/* Open / Close -/* ---------------------------------------------------------- */ - -.popover { - position: relative; - display: inline-block; -} - -.popover .popover-item { - position: absolute; - z-index: 20; -} - -.popover .popover-item.open { - display: block; -} - -.popover .popover-item.closed { - display: none; -} diff --git a/core/client/app/styles/components/power-select.css b/core/client/app/styles/components/power-select.css deleted file mode 100644 index 97be3b7d50..0000000000 --- a/core/client/app/styles/components/power-select.css +++ /dev/null @@ -1,123 +0,0 @@ -.ember-power-select-trigger { - border: 1px solid #dfe1e3; - border-radius: var(--border-radius); - color: #666; -} - -.ember-power-select-search { - padding: 2px 0 3px 0 !important; -} - -.ember-basic-dropdown--opened > .ember-power-select-trigger, -.ember-power-select-trigger[aria-expanded="true"], -.ember-power-select-search input { - outline: 0; - border-color: #b1b1b1; -} - -.ember-power-select-dropdown { - position: absolute; - z-index: 1000; - box-sizing: border-box; - margin: -1px 0 0 0; - border: 1px solid #b1b1b1; - border-top: 0 none; - background: #fff; - border-radius: 0 0 var(--border-radius) var(--border-radius); - box-shadow: 0 3px 6px rgba(0, 0, 0, 0.1); -} - -.ember-power-select-options:not([role="group"]) { - max-height: 200px; -} - -.ember-power-select-search input { - display: inline-block !important; - margin: 0 1px !important; - padding: 0 !important; - min-height: 0 !important; - max-width: 100% !important; - max-height: none !important; - border: 0 none !important; - background: none !important; - box-shadow: none !important; - text-indent: 0 !important; - font-size: 1.3rem; - line-height: inherit !important; -} - -.ember-power-select-group { - float: left; - box-sizing: border-box; - width: 100%; - border-top: 0 none; - border-right: 1px solid #f2f2f2; -} - -.ember-power-select-group .ember-power-select-group-name { - position: relative; - display: inline-block; - padding: 7px 8px; - background: #fff; - color: var(--midgrey); - font-size: 0.85em; - font-weight: normal; - cursor: default; -} - -.ember-power-select-group .ember-power-select-group-name:after { - content: ""; - position: absolute; - top: 52%; - left: calc(100% + 3px); - display: block; - width: calc(189px - 100%); - height: 1px; - border-bottom: #dfe1e3 1px solid; -} - -@media (max-width: 800px) { - .ember-power-select-group .ember-power-select-group-name:after { - width: calc(224px - 100%); - } -} - -@media (max-width: 500px) { - .ember-power-select-group .ember-power-select-group-name:after { - width: calc(80vw - 45px - 100%); - } -} - -.ember-power-select-group:first-of-type .ember-power-select-group-name { - margin-bottom: 7px; - padding-top: 0; - padding-bottom: 0; -} - -.ember-power-select-group .ember-power-select-option { - overflow: hidden; - padding: 7px 8px; - line-height: 1.35em; - cursor: pointer; -} - -.ember-power-select-group .ember-power-select-option .highlight { - background: #fff3b8; - border-radius: 1px; -} - -.ember-power-select-group .ember-power-select-option[aria-current="true"] { - background: color(var(--blue) alpha(-85%)); - color: var(--darkgrey); -} - -/* - HACK: ember-power-select has no separate class for the loading message - Issue: https://github.com/cibernox/ember-power-select/issues/479 - */ -.ember-power-select-dropdown > .ember-power-select-options > .ember-power-select-option:first-of-type, -.ember-power-select-option--no-matches-message { - padding: 7px 8px; - color: var(--midgrey); - font-size: 0.9em; -} diff --git a/core/client/app/styles/components/selectize.css b/core/client/app/styles/components/selectize.css deleted file mode 100644 index c7a25db99f..0000000000 --- a/core/client/app/styles/components/selectize.css +++ /dev/null @@ -1,405 +0,0 @@ -.selectize-control.plugin-drag_drop.multi > .selectize-input > div.ui-sortable-placeholder { - visibility: visible !important; - border: 0 none !important; - background: #f2f2f2 !important; - background: rgba(0, 0, 0, 0.06) !important; -} - -.selectize-control.plugin-drag_drop .ui-sortable-placeholder:after { - content: "!"; - visibility: hidden; -} - -.selectize-control.plugin-drag_drop .ui-sortable-helper { - box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2); -} - -.selectize-dropdown-header { - position: relative; - padding: 5px 8px; - border-bottom: 1px solid #d0d0d0; - background: #f8f8f8; - border-radius: var(--border-radius) var(--border-radius) 0 0; -} - -.selectize-dropdown-header-close { - position: absolute; - top: 50%; - right: 8px; - margin-top: -12px; - color: #303030; - font-size: 20px !important; - line-height: 20px; - opacity: 0.4; -} - -.selectize-dropdown-header-close:hover { - color: #000; -} - -.selectize-dropdown.plugin-optgroup_columns .optgroup { - float: left; - box-sizing: border-box; - border-top: 0 none; - border-right: 1px solid #f2f2f2; -} - -.selectize-dropdown.plugin-optgroup_columns .optgroup:last-child { - border-right: 0 none; -} - -.selectize-dropdown.plugin-optgroup_columns .optgroup:before { - display: none; -} - -.selectize-dropdown.plugin-optgroup_columns .optgroup-header { - border-top: 0 none; -} - -.selectize-control.plugin-remove_button [data-value] { - position: relative; - padding-right: 20px !important; -} - -.selectize-control.plugin-remove_button [data-value] .remove { - position: absolute; - top: 0; - right: 0; - bottom: 0; - z-index: 1; - display: flex; - justify-content: center; - align-items: center; - box-sizing: border-box; - width: 17px; - border-radius: 0 2px 2px 0; - color: inherit; - vertical-align: middle; - text-align: center; - text-decoration: none; - font-size: 12px; - font-weight: bold; -} - -.selectize-control.plugin-remove_button [data-value] .remove:hover { - background: rgba(0, 0, 0, 0.05); -} - -.selectize-control.plugin-remove_button [data-value].active .remove { - border-left-color: #00578d; -} - -.selectize-control.plugin-remove_button .disabled [data-value] .remove:hover { - background: none; -} - -.selectize-control.plugin-remove_button .disabled [data-value] .remove { - border-left-color: #aaa; -} - -.selectize-control { - position: relative; -} - -.selectize-dropdown, -.selectize-input, -.selectize-input input { - color: #303030; - font-family: inherit; - font-size: 1.4rem; -} - -.selectize-input, -.selectize-control.single .selectize-input.input-active { - display: inline-block; - background: #fff; - cursor: text; -} - -.selectize-input { - position: relative; - z-index: 1; - display: inline-block; - overflow: hidden; - box-sizing: border-box; - padding: 8px 10px; - width: 100%; - height: 37px; - border: 1px solid #dfe1e3; - border-radius: var(--border-radius); - color: #666; - transition: border-color 0.15s linear; -} - -.selectize-input.focus { - border-color: #b1b1b1; -} - -.selectize-control.multi .selectize-input.has-items { - padding: 5px 8px 2px; - height: auto; -} - -.selectize-input.full { - background-color: #fff; -} - -.selectize-input.disabled, -.selectize-input.disabled * { - cursor: default !important; -} - -.selectize-input.dropdown-active { - border-radius: var(--border-radius) var(--border-radius) 0 0; -} - -.selectize-input > * { - display: -moz-inline-stack; - display: inline-block; - vertical-align: baseline; - zoom: 1; - - *display: inline; -} - -.selectize-control.multi .selectize-input > div { - margin: 0 3px 3px 0; - padding: 1px 4px; - background: var(--blue); - color: #fff; - cursor: pointer; -} - - -/* Active tag - selected state when tag is clicked */ -.selectize-control.multi .selectize-input > div.active { - background: color(var(--blue) lightness(-10%)); - color: #fff; -} - -.selectize-control.multi .selectize-input.disabled > div, -.selectize-control.multi .selectize-input.disabled > div.active { - border: 1px solid #aaa; - background: #d2d2d2; - color: #fff; -} - -.selectize-input > input { - display: inline-block !important; - margin: 0 1px !important; - padding: 0 !important; - min-height: 0 !important; - max-width: 100% !important; - max-height: none !important; - border: 0 none !important; - background: none !important; - box-shadow: none !important; - text-indent: 0 !important; - line-height: inherit !important; -} - -.selectize-input > input:-ms-clear { - display: none; -} - -.selectize-input > input:focus { - outline: none !important; -} - -.selectize-input:after { - content: " "; - display: block; - clear: left; -} -.selectize-input.dropdown-active:before { - content: " "; - position: absolute; - right: 0; - bottom: 0; - left: 0; - display: block; - height: 1px; - background: #f0f0f0; -} - -.selectize-dropdown { - position: absolute; - z-index: 1000; - box-sizing: border-box; - margin: -1px 0 0 0; - border: 1px solid #b1b1b1; - border-top: 0 none; - background: #fff; - border-radius: 0 0 var(--border-radius) var(--border-radius); - box-shadow: 0 3px 6px rgba(0, 0, 0, 0.1); -} - -.selectize-dropdown [data-selectable] { - overflow: hidden; - cursor: pointer; -} - -.selectize-dropdown [data-selectable] .highlight { - background: #fff3b8; - border-radius: 1px; -} - -.selectize-dropdown [data-selectable], -.selectize-dropdown .optgroup-header, -.selectize-dropdown .dropdown-empty-message { - padding: 7px 8px; -} - -.selectize-dropdown .optgroup-header { - background: #fff; - color: #303030; - cursor: default; -} - -.selectize-dropdown .active { - background: color(var(--blue) alpha(-85%)); - color: var(--darkgrey); -} - -.selectize-dropdown .active.create { - color: #666; -} - -.selectize-dropdown .create { - color: rgba(48, 48, 48, 0.5); -} - -.selectize-dropdown-content { - overflow-x: hidden; - overflow-y: auto; - max-height: 200px; -} - -.selectize-control.single .selectize-input, -.selectize-control.single .selectize-input input { - cursor: pointer; -} - -.selectize-control.single .selectize-input.input-active, -.selectize-control.single .selectize-input.input-active input { - cursor: text; -} - -.selectize-control.single .selectize-input:after { - content: " "; - position: absolute; - top: 50%; - right: 15px; - display: block; - margin-top: -3px; - width: 0; - height: 0; - border-width: 5px 5px 0 5px; - border-style: solid; - border-color: #808080 transparent transparent transparent; -} - -.selectize-control.single .selectize-input.dropdown-active:after { - margin-top: -4px; - border-width: 0 5px 5px 5px; - border-color: transparent transparent #808080 transparent; -} - -.selectize-control.rtl.single .selectize-input:after { - right: auto; - left: 15px; -} - -.selectize-control.rtl .selectize-input > input { - margin: 0 4px 0 -2px !important; -} - -.selectize-control .selectize-input.disabled { - background-color: #fafafa; - opacity: 0.5; -} - -.selectize-control.multi .selectize-input.has-items { - padding-right: 5px; - padding-left: 5px; -} - -.selectize-control.multi .selectize-input.disabled [data-value] { - background: none; - box-shadow: none; - color: #999; - text-shadow: none; -} - -.selectize-control.multi .selectize-input.disabled [data-value], -.selectize-control.multi .selectize-input.disabled [data-value] .remove { - border-color: #e6e6e6; -} - -.selectize-control.multi .selectize-input.disabled [data-value] .remove { - background: none; -} - -.selectize-control.multi .selectize-input [data-value] { - background: var(--blue); - border-radius: 3px; -} - -.selectize-control.multi .selectize-input [data-value].active { - background: color(var(--blue) lightness(-10%)); -} - -.selectize-control.single .selectize-input { - background: #f9f9f9; -} - -.selectize-control.single .selectize-input, -.selectize-dropdown.single { - border-color: #b8b8b8; -} - -.optgroup:first-of-type .optgroup-header { - margin-bottom: 7px; - padding-top: 0; - padding-bottom: 0; -} - -.selectize-dropdown .optgroup-header { - position: relative; - display: inline-block; - padding-top: 7px; - background: #fff; - color: var(--midgrey); - font-size: 0.85em; -} - -.selectize-dropdown .optgroup-header:after { - content: ""; - position: absolute; - top: 52%; - left: calc(100% + 3px); - display: block; - width: calc(189px - 100%); - height: 1px; - border-bottom: #dfe1e3 1px solid; -} -@media (max-width: 800px) { - .selectize-dropdown .optgroup-header:after { - width: calc(224px - 100%); - } -} -@media (max-width: 500px) { - .selectize-dropdown .optgroup-header:after { - width: calc(80vw - 45px - 100%); - } -} - -.selectize-dropdown .option { - line-height: 1.35em; -} - -.dropdown-empty-message { - position: relative; - color: var(--midgrey); - font-size: 0.9em; -} diff --git a/core/client/app/styles/components/settings-menu.css b/core/client/app/styles/components/settings-menu.css deleted file mode 100644 index 25ccc8c140..0000000000 --- a/core/client/app/styles/components/settings-menu.css +++ /dev/null @@ -1,180 +0,0 @@ -/* Settings Menu -/* ---------------------------------------------------------- */ - - -/* Container -/* ---------------------------------------------------------- */ - -.settings-menu-container { - position: fixed; - top: 0; - right: 0; - bottom: 0; - z-index: 500; - overflow: hidden; - max-width: 100%; - width: 350px; - border-left: #dfe1e3 1px solid; - background: #fff; - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); - transform: translate3d(350px, 0px, 0px); -} - -.settings-menu-expanded .settings-menu-container { - transform: translate3d(0, 0px, 0px); -} - -.settings-menu-container .settings-menu-pane { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - overflow: auto; - -webkit-overflow-scrolling: touch; - opacity: 1; - transform: translate3d(0, 0px, 0px); -} - -@media (min-width: 901px) { - .settings-menu-container .settings-menu-pane { - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); - } -} - -.settings-menu-container .settings-menu-pane.settings-menu-pane-out-left { - transform: translate3d(-100%, 0px, 0px); -} - -.settings-menu-container .settings-menu-pane.settings-menu-pane-out-right { - transform: translate3d(100%, 0px, 0px); -} - -.settings-menu-container .settings-menu-pane.settings-menu-pane-in { - transform: translate3d(0, 0px, 0px); -} - - -/* Header -/* ---------------------------------------------------------- */ - -.settings-menu-header { - position: relative; - display: flex; - justify-content: space-between; - align-items: center; - padding: 15px 24px; -} - -.settings-menu-header h4 { - margin: 0; - font-size: 1.6rem; - line-height: 1.375; - font-weight: normal; -} - -.settings-menu-header .close { - margin-right: -15px; - padding: 10px 15px; - font-size: 12px; - line-height: 12px; -} - -.settings-menu-header.subview h4 { - text-align: center; -} - -.settings-menu-header.subview .back { - margin-left: -15px; - padding: 10px 15px; - font-size: 14px; - line-height: 14px; -} - - - - - -/* Content -/* ---------------------------------------------------------- */ - -.settings-menu-content { - padding: 0 24px 24px; -} - -.settings-menu-content .gh-image-uploader { - margin: 0 0 1.6rem 0; -} - -.settings-menu-content .gh-image-uploader .description { - font-size: 1.4rem; -} - -.settings-menu-content .gh-image-uploader form { - padding: 35px 45px; -} - -.settings-menu-content .gh-image-uploader.--with-image { - margin-top: 0; - min-height: 50px; - max-height: 250px; - width: auto; -} - -.settings-menu-content textarea { - height: 108px; -} - -.settings-menu-content .tag-delete-button { - padding-left: 0; - color: var(--red); -} - -.settings-menu-content .tag-delete-button:before { - position: relative; - top: -1px; - margin-right: 4px; -} - -.settings-menu-content .tag-delete-button:hover, -.settings-menu-content .tag-delete-button:hover:before { - color: color(var(--red) lightness(-10%)); -} - -.settings-menu-content .nav-list { - margin-top: 3rem; -} - -.settings-menu-content .word-count { - font-weight: bold; -} - -.ghost-url-preview { - /* Preview never wider than input */ - overflow: hidden; - width: 98%; - text-overflow: ellipsis; - white-space: nowrap; -} - - -/* Background -/* ---------------------------------------------------------- */ - -.settings-menu-expanded .content-cover, -.mobile-menu-expanded .content-cover { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - z-index: 900; - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); - /* Not off the screen, to give a parallax effect */ -} -.settings-menu-expanded .content-cover { - transform: translate3d(-350px, 0px, 0px); -} -.mobile-menu-expanded .content-cover { - transform: translate3d(235px, 0px, 0px); -} diff --git a/core/client/app/styles/components/splitbuttons.css b/core/client/app/styles/components/splitbuttons.css deleted file mode 100644 index 33c6017e4a..0000000000 --- a/core/client/app/styles/components/splitbuttons.css +++ /dev/null @@ -1,62 +0,0 @@ -/* Splitbuttons -/* ---------------------------------------------------------- */ - -.splitbtn { - position: relative; - display: inline-block; - vertical-align: middle; - /* Flatten out the right side */ - /* Flatten out the left side */ -} - -.splitbtn .btn { - position: relative; - float: left; - /* Prevent double border between buttons */ - /* Make sure the hovered element is always on - // top so overlap from .btn + btn. invisible */ -} - -.splitbtn .btn + .btn { - margin-left: -1px; -} - -.splitbtn .btn:hover, -.splitbtn .btn:focus, -.splitbtn .btn:active, -.splitbtn .btn.active { - z-index: 2; -} - -.splitbtn .btn:first-child { - margin-left: 0; -} - -.splitbtn .btn:first-child:not(:last-child):not(.dropdown-toggle) { - border-top-right-radius: 0; - border-bottom-right-radius: 0; -} - -.splitbtn .dropdown-toggle { - padding-right: 12px; - padding-left: 12px; - border-top-left-radius: 0; - border-bottom-left-radius: 0; - /* This is the additional dropdown arrow, to the right of the button. */ -} - -.splitbtn .dropdown-toggle.btn-sm { - padding-right: 10px; - padding-left: 10px; - height: 31px; -} - -.splitbtn .dropdown-toggle.btn-lg { - padding-right: 16px; - padding-left: 16px; -} - -.splitbtn .dropdown-toggle .options { - color: #fff; - text-align: center; -} diff --git a/core/client/app/styles/components/uploader.css b/core/client/app/styles/components/uploader.css deleted file mode 100644 index 67974824bc..0000000000 --- a/core/client/app/styles/components/uploader.css +++ /dev/null @@ -1,162 +0,0 @@ -/* Image Uploader -/* ---------------------------------------------------------- */ - -.gh-image-uploader { - position: relative; - display: flex; - flex-direction: column; - align-items: center; - overflow: hidden; - margin: 1.6em 0; - min-height: 130px; - width: 100%; - background: #f6f7f8; - border-radius: 4px; - color: #808284; - text-align: center; -} - -.gh-image-uploader.--drag-over { - border: 2px solid; -} - -.gh-image-uploader.--with-image { - background: rgba(0, 0, 0, 0.1); - border-radius: 2px; -} - -.gh-image-uploader img { - display: block; - margin: 0 auto; - max-width: 100%; - line-height: 0; -} - -.gh-image-uploader .image-cancel { - position: absolute; - top: 10px; - right: 10px; - z-index: 300; - display: block; - padding: 8px; - background: rgba(0, 0, 0, 0.6); - border-radius: var(--border-radius); - box-shadow: rgba(255, 255, 255, 0.2) 0 0 0 1px; - color: #fff; - text-decoration: none; - font-size: 13px; - line-height: 10px; -} - -.gh-image-uploader .upload-form { - flex-grow: 1; - display: flex; - flex-direction: row; -} - -.gh-image-uploader .x-file-input { - flex-grow: 1; - display: flex; -} - -.gh-image-uploader .x-file-input label { - flex-grow: 1; - display: flex; - align-items: center; - outline: none; -} - -.gh-image-uploader .description { - width: 100%; - text-align: center; - font-size: 1.6rem; -} - -.gh-image-uploader .image-upload, -.gh-image-uploader .image-url { - position: absolute; - bottom: 0; - left: 0; - display: block; - padding: 10px; - color: var(--midgrey); - text-decoration: none; - font-size: 14px; - line-height: 12px; -} - -.gh-image-uploader a { - color: var(--midgrey); - text-decoration: none; -} - -.gh-image-uploader a:hover { - color: var(--darkgrey); -} -.gh-image-uploader .image-upload:hover, -.gh-image-uploader .image-url:hover { - cursor: pointer; -} - -.gh-image-uploader form { - padding: 55px 60px; - width: 100%; -} - -.gh-image-uploader input.url { - margin: 0 0 10px 0; - padding: 9px 7px; - outline: 0; - background: #fff; - vertical-align: middle; - font: -webkit-small-control; - font-size: 1.4rem; -} - -.gh-image-uploader input.url + .btn.btn-blue { - color: #fff; -} - -.gh-image-uploader .image-cancel:hover { - background: var(--red); - color: #fff; - cursor: pointer; -} - -.gh-image-uploader .progress-container { - flex-grow: 1; - display: flex; - flex-direction: row; - align-items: center; - width: 100%; -} - -.gh-image-uploader .progress { - overflow: hidden; - margin: 0 auto; - width: 60%; - background: linear-gradient(to bottom, #f5f5f5, #f9f9f9); - border-radius: 12px; - box-shadow: rgba(0, 0, 0, 0.1) 0 1px 2px inset; -} - -.gh-image-uploader .failed { - margin: 1em 2em; - font-size: 16px; -} - -.gh-image-uploader .bar { - height: 12px; - background: var(--blue); -} - -.gh-image-uploader .bar.fail { - width: 100% !important; - background: var(--red); -} - -/* Try Again button */ -.gh-image-uploader .btn-green:last-child { - margin-top: 1em; - margin-bottom: 3em; -} diff --git a/core/client/app/styles/csscomb.json b/core/client/app/styles/csscomb.json deleted file mode 100644 index 624f4c19f9..0000000000 --- a/core/client/app/styles/csscomb.json +++ /dev/null @@ -1,235 +0,0 @@ -{ - "remove-empty-rulesets": true, - "always-semicolon": true, - "color-case": "lower", - "block-indent": " ", - "color-shorthand": true, - "element-case": "lower", - "eof-newline": true, - "leading-zero": true, - "quotes": "double", - "space-before-colon": "", - "space-after-colon": " ", - "space-before-combinator": " ", - "space-after-combinator": " ", - "space-between-declarations": "\n", - "space-before-opening-brace": " ", - "space-after-opening-brace": "\n", - "space-after-selector-delimiter": "\n", - "space-before-selector-delimiter": "", - "space-before-closing-brace": "\n", - "strip-spaces": true, - "tab-size": 4, - "unitless-zero": true, - "sort-order": [ [ - "content", - "visibility", - "position", - "top", - "right", - "bottom", - "left", - "z-index", - "order", - "flex", - "flex-grow", - "flex-shrink", - "flex-basis", - "align-self", - "display", - "flex-flow", - "flex-direction", - "flex-wrap", - "justify-content", - "align-items", - "align-content", - "flex-order", - "flex-pack", - "flex-align", - "float", - "clear", - "overflow", - "overflow-x", - "overflow-y", - "-webkit-overflow-scrolling", - "clip", - "box-sizing", - "margin", - "margin-top", - "margin-right", - "margin-bottom", - "margin-left", - "padding", - "padding-top", - "padding-right", - "padding-bottom", - "padding-left", - "min-width", - "min-height", - "max-width", - "max-height", - "width", - "height", - "outline", - "outline-width", - "outline-style", - "outline-color", - "outline-offset", - "border", - "border-spacing", - "border-collapse", - "border-width", - "border-style", - "border-color", - "border-top", - "border-top-width", - "border-top-style", - "border-top-color", - "border-right", - "border-right-width", - "border-right-style", - "border-right-color", - "border-bottom", - "border-bottom-width", - "border-bottom-style", - "border-bottom-color", - "border-left", - "border-left-width", - "border-left-style", - "border-left-color", - "border-image", - "border-image-source", - "border-image-slice", - "border-image-width", - "border-image-outset", - "border-image-repeat", - "border-top-image", - "border-right-image", - "border-bottom-image", - "border-left-image", - "border-corner-image", - "border-top-left-image", - "border-top-right-image", - "border-bottom-right-image", - "border-bottom-left-image", - "background", - "filter:progid:DXImageTransform.Microsoft.AlphaImageLoader", - "background-color", - "background-image", - "background-attachment", - "background-position", - "background-position-x", - "background-position-y", - "background-clip", - "background-origin", - "background-size", - "background-repeat", - "border-radius", - "border-top-left-radius", - "border-top-right-radius", - "border-bottom-right-radius", - "border-bottom-left-radius", - "box-decoration-break", - "box-shadow", - "color", - "table-layout", - "caption-side", - "empty-cells", - "list-style", - "list-style-position", - "list-style-type", - "list-style-image", - "quotes", - "counter-increment", - "counter-reset", - "vertical-align", - "text-align", - "text-align-last", - "text-decoration", - "text-emphasis", - "text-emphasis-position", - "text-emphasis-style", - "text-emphasis-color", - "text-indent", - "text-justify", - "text-outline", - "text-transform", - "text-wrap", - "text-overflow", - "text-overflow-ellipsis", - "text-overflow-mode", - "text-shadow", - "white-space", - "word-spacing", - "word-wrap", - "word-break", - "tab-size", - "hyphens", - "letter-spacing", - "font", - "font-family", - "font-size", - "line-height", - "font-weight", - "font-style", - "font-variant", - "font-size-adjust", - "font-stretch", - "text-rendering", - "font-feature-settings", - "user-select", - "src", - "opacity", - "filter:progid:DXImageTransform.Microsoft.Alpha(Opacity", - "filter", - "resize", - "cursor", - "nav-index", - "nav-up", - "nav-right", - "nav-down", - "nav-left", - "transition", - "transition-delay", - "transition-timing-function", - "transition-duration", - "transition-property", - "transform", - "transform-origin", - "animation", - "animation-name", - "animation-duration", - "animation-play-state", - "animation-timing-function", - "animation-delay", - "animation-iteration-count", - "animation-direction", - "animation-fill-mode", - "pointer-events", - "unicode-bidi", - "direction", - "columns", - "column-span", - "column-width", - "column-count", - "column-fill", - "column-gap", - "column-rule", - "column-rule-width", - "column-rule-style", - "column-rule-color", - "break-before", - "break-inside", - "break-after", - "page-break-before", - "page-break-inside", - "page-break-after", - "orphans", - "widows", - "zoom", - "max-zoom", - "min-zoom", - "user-zoom", - "orientation" - ] ] -} diff --git a/core/client/app/styles/layouts/about.css b/core/client/app/styles/layouts/about.css deleted file mode 100644 index 57b3b10da2..0000000000 --- a/core/client/app/styles/layouts/about.css +++ /dev/null @@ -1,123 +0,0 @@ -/* About /ghost/settings/about/ -/* ---------------------------------------------------------- */ - -.gh-logo { - position: relative; - width: 120px; - height: auto; -} - -.gh-env-details { - display: flex; - align-items: center; - margin: 1em 0; -} - -.gh-env-list { - margin: 0; - padding: 0 40px 0 0; - list-style: none; -} - -.gh-env-help { - max-width: 200px; -} - -.gh-env-help .btn { - display: block; - margin: 5px 0; -} - -@media (max-width: 670px) { - .gh-env-details { - flex-direction: column; - align-items: flex-start; - } - .gh-env-help { - margin: 1em 0; - max-width: none; - } - .gh-env-help .btn { - display: inline-block; - } -} - -.gh-credits { - margin: 2em 0; - max-width: 650px; - color: var(--midgrey); - font-size: 1.8rem; - font-weight: 200; -} - -@media (max-width: 890px) { - .gh-credits { - max-width: 460px; - } -} - -.gh-credits h2 { - font-size: 2.4rem; -} - - -/* Contributors -/* ---------------------------------------------------------- */ - -.gh-contributors { - display: flex; - flex-wrap: wrap; - margin: 1em 0; -} - -.gh-contributors a { - position: relative; - display: block; - margin: 0 10px 10px 0; - width: 60px; - height: 60px; -} - -.gh-contributors img { - border-radius: 100%; -} - -.gh-contributors a:before { - content: attr(title); - position: absolute; - top: -20px; - left: 50%; - padding: 2px 6px; - background: var(--darkgrey); - border-radius: var(--border-radius); - color: #fff; - font-size: 1rem; - line-height: 1.3em; - opacity: 0; - transition: opacity 0.15s ease-in-out; - transform: translateX(-50%); - pointer-events: none; -} - -.gh-contributors a:after { - content: ""; - position: absolute; - top: -6px; - left: 50%; - opacity: 0; - transition: opacity 0.15s ease-in-out; - transform: translateX(-50%); -} - -.gh-contributors a:hover:before, -.gh-contributors a:hover:after { - opacity: 1; -} - -/* Copyright Info -/* ---------------------------------------------------------- */ - -.gh-copyright-info { - color: var(--midgrey); - font-size: 1.2rem; -} diff --git a/core/client/app/styles/layouts/apps.css b/core/client/app/styles/layouts/apps.css deleted file mode 100644 index 55b3633536..0000000000 --- a/core/client/app/styles/layouts/apps.css +++ /dev/null @@ -1,252 +0,0 @@ -/* Apps -/* ---------------------------------------------------------- */ - -.apps-filter { - border-radius: 5px; -} -@media (max-width: 1460px) { - .apps-filter { - max-width: 700px; - } -} - -/* Main Layout -/* ---------------------------------------------------------- */ - -.apps-grid-title { - display: block; - margin-bottom: 5px; - color: var(--midgrey); -} - -.apps-grid { - display: flex; - flex-flow: row wrap; - align-items: flex-start; - max-width: 1200px; - border: rgba(0,0,0,0.1) 1px solid; - border-radius: 5px; -} - -.apps-grid-note { - display: block; - margin-top: 5px; - color: var(--midgrey); - font-size: 1.2rem; - font-style: italic; -} - -/* Apps Card -/* ---------------------------------------------------------- */ - -.apps-grid-cell { - flex: 1 1 100%; -} -.apps-grid-cell:hover { - background: #f5f7f8; -} - -.apps-card-app { - overflow: hidden; - padding: 14px; - height: 75px; - border-top: rgba(0,0,0,0.1) 1px solid; - transition: background 0.3s ease; -} - -.apps-card-app:first-of-type { - border-top: none; -} - -.apps-card-content { - position: relative; - display: flex; -} - -.apps-card-content > .btn { - position: absolute; - top: 5px; - right: 20px; -} -.apps-card-content > .apps-configured { - position: absolute; - top: 13px; - right: 29px; - display: flex; - color: var(--midgrey); - white-space: nowrap; -} - -.apps-configured > span { - padding-right: 14px; - font-size: 1.3rem; -} - -.apps-configured > i { - position: absolute; - top: 2px; - font-size: 1.1rem; -} - -.apps-card-app-icon { - flex: 0 0 47px; - margin: 0 15px 0 0; - width: 47px; - height: 47px; - background-position: center center; - background-size: cover; - border-radius: 15%; -} - -.apps-card-meta { - position: relative; - display: flex; - flex-direction: column; -} - -.apps-card-app-title { - overflow: hidden; - margin: 4px 0 0 0; - padding: 0 70px 0 0; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 1.7rem; - line-height: 1.15em; - font-weight: bold; -} - - -/* Apps Card Meta -/* ---------------------------------------------------------- */ - -.apps-card-app-desc { - display: -webkit-box; - overflow: hidden; - margin: 0; - padding: 0; - max-height: 4.2rem; - color: var(--midgrey); - text-overflow: ellipsis; - font-size: 1.4rem; - line-height: 1.3em; - font-weight: 200; - - -webkit-line-clamp: 2; - -webkit-box-orient: vertical; -} -@media (min-width: 600px) and (max-width: 1460px) { - .apps-card-app-desc { - padding-right: 80px; - } -} - - -/* Apps Card Footer -/* ---------------------------------------------------------- */ - -.apps-card-footer { - display: flex; - justify-content: space-between; - align-items: center; - margin-top: 15px; - width: 100%; -} -/* Slack -/* ---------------------------------------------------------- */ - -.app-grid { - display: flex; - flex-wrap: wrap; - justify-content: flex-start; - align-items: center; - align-content: flex-start; - margin-bottom: 30px; -} -.app-icon { - position: relative; - flex: 0 0 60px; - margin: 0 15px 0 0; - width: 60px; - height: 60px; - background-position: center center; - background-size: cover; - border-radius: 10%; -} - -.app-cell h3 { - margin: 0; - color: #393939; - font-size: 2.4rem; -} - -.app-cell p { - margin: 0; - margin-bottom: 5px; - color: #a3a3a3; - font-size: 1.6rem; - line-height: 1.4em; -} - -.app-subtitle { - max-width: 550px; - color: #666363; - font-size: 1.6rem; -} -.app-config-form .btn-grey { - margin-top: 1.6em; - background-color: #e8e8e8; - box-shadow: none; - font-size: 1.1rem; -} - -.app-config-form > .btn-grey:hover, -.app-config-form > .btn-grey:focus { - border-color: rgb(223, 225, 227); -} -/* Media Queries -/* ---------------------------------------------------------- */ - -@media (max-width: 800px) { - .apps-grid-apps { - overflow: hidden; - margin: 0 0 4vw 0; - border: #dfe1e3 1px solid; - border-radius: 5px; - } - - .apps-card-app { - margin: 0; - border: none; - border-top: #dfe1e3 1px solid; - border-radius: 0; - } - .apps-grid-cell:first-of-type .apps-card-app { - border-top: none; - } -} - -@media (max-width: 760px) { - .apps-card-app { - padding: 15px; - } - .apps-card-app .apps-card-footer { - justify-content: flex-end; - } - .apps-card-theme .apps-card-footer { - margin: 0; - padding: 15px; - } -} - -@media (max-width: 540px) { - .apps-card-footer { - justify-content: flex-end; - } - .apps-card-app .apps-card-footer { - flex-direction: column; - align-items: flex-start; - } - .apps-card-content > .apps-configured { - right: 15px; - } -} diff --git a/core/client/app/styles/layouts/auth.css b/core/client/app/styles/layouts/auth.css deleted file mode 100644 index a7ebf8de59..0000000000 --- a/core/client/app/styles/layouts/auth.css +++ /dev/null @@ -1,56 +0,0 @@ -/* Sign in -/* ---------------------------------------------------------- */ - -.gh-signin { - position: relative; - margin: 30px auto; - padding: 40px; - max-width: 400px; - width: 100%; - border: #dae1e3 1px solid; - background: #f8fbfd; - border-radius: 5px; - text-align: left; -} - -.gh-signin .form-group { - margin-bottom: 1.5rem; -} - -.gh-signin .btn { - margin: 0; - padding: 12px; -} - -.forgotten-wrap { - position: relative; -} - -.forgotten-wrap input { - padding-right: 7rem; -} - -.forgotten-wrap .forgotten-link { - position: absolute; - top: 10px; - right: 1px; - bottom: 10px; - padding: 0 11px 0 12px; - border-left: #dae1e3 1px solid; - border-radius: 0; - text-transform: none; - letter-spacing: 0; - font-size: 1.1rem; - line-height: 1.2rem; -} - -.forgotten-link:hover { - border-left: #dae1e3 1px solid; - color: color(var(--blue) lightness(-20%)); - text-decoration: none; -} - -.forgotten-link:active { - box-shadow: none; - text-decoration: underline; -} diff --git a/core/client/app/styles/layouts/content.css b/core/client/app/styles/layouts/content.css deleted file mode 100644 index a0bc8a9247..0000000000 --- a/core/client/app/styles/layouts/content.css +++ /dev/null @@ -1,289 +0,0 @@ -/* Content /ghost/ - -/* Show/Hide on Mobile // TODO: What the fuck does that mean? -/* ---------------------------------------------------------- */ - -.content-list.show-menu { - display: block; -} - -.content-list.show-content { - display: none; -} - -.content-preview.show-menu { - display: none; -} - -.content-preview.show-content { - display: block; -} - - -/* Content List (Left pane) -/* ---------------------------------------------------------- */ - -.content-list { - position: absolute; - top: 0; - bottom: 0; - left: 0; - padding: 15px; - width: 33%; - border-right: #dfe1e3 1px solid; - background: #fff; -} - -@media (max-width: 900px) { - .content-list { - right: 0; - z-index: 500; - width: auto; - border: none; - } -} - -.content-list .content-list-content { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - overflow: auto; - -webkit-overflow-scrolling: touch; -} - -.content-list .entry-title { - font-size: 1.6rem; - line-height: 1.4em; - font-weight: normal; -} - -.content-list .entry-meta { - margin-top: 14px; - line-height: 18px; -} - -.content-list .avatar { - position: relative; - float: left; - margin-right: 14px; - width: 18px; - height: 18px; - background-position: center center; - background-size: cover; - border-radius: 18px; -} - -.content-list .avatar img { - position: absolute; - top: 0; - left: 0; - width: 100%; - height: 100%; - opacity: 0; -} - -.content-list .status, -.content-list .author { - font-size: 1.3rem; - font-weight: 300; - transition: opacity 0.15s linear; -} - -.content-list .avatar:hover + .author + .status { - opacity: 0; -} - -.content-list .avatar:hover + .author { - opacity: 1; -} - -.content-list .author { - position: absolute; - bottom: 22px; - left: 56px; - opacity: 0; -} - -.content-list .status .draft { - color: var(--red); -} - -.content-list .status .scheduled { - color: var(--orange); -} - -.content-list ol { - margin: 0; - padding: 0; - list-style: none; -} - -.content-list li { - position: relative; - margin: 0; - padding: 0; - border-bottom: #dfe1e3 1px solid; -} - -.content-list li a { - display: block; - padding: 19px 20px 22px 24px; - color: rgba(0, 0, 0, 0.5); -} - -.content-list li a:hover { - text-decoration: none; -} - -@media (max-width: 400px) { - .content-list li a { - padding: 15px; - } -} - -@media (max-width: 900px) { - .content-list li a { - padding-right: 40px; - } -} - -@media (min-width: 901px) { - .content-list li a:after { - display: none; - } -} - -@media (min-width: 901px) { - .content-list .active a { - border-left: var(--blue) 3px solid; - } -} - - -/* Preview (Right pane) -/* ---------------------------------------------------------- */ - -.content-preview-content .content-preview-title a { - position: relative; - color: var(--darkgrey); - text-decoration: none; -} - -.content-preview { - position: absolute; - top: 0; - right: 0; - bottom: 0; - overflow: auto; - -webkit-overflow-scrolling: touch; - padding: 15px; - width: 67%; - background: #fff; -} - -@media (max-width: 900px) { - .content-preview { - display: none; - overflow: visible; - width: 100%; - border: none; - } -} - -.content-preview .content-preview-content { - padding: 5%; - word-break: break-word; - hyphens: auto; -} - -@media (max-width: 900px) { - .content-preview .content-preview-content { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - overflow: auto; - -webkit-overflow-scrolling: touch; - } -} - -.content-preview .content-preview-content .wrapper { - margin: 0 auto; - max-width: 700px; -} - -.content-preview .post-controls { - position: absolute; - top: 20px; - right: 25px; -} - -.content-preview .post-controls .post-edit { - padding-top: 0; - padding-right: 0; - border: none; - font-size: 18px; -} -.content-preview .post-controls .post-edit:hover { - color: var(--darkgrey); -} - -.content-preview img { - width: 100%; - height: auto; -} - - -/* Empty State -/* ---------------------------------------------------------- */ - -.no-posts-box { - position: relative; - z-index: 600; - display: flex; - justify-content: center; - align-items: center; - margin: 0 auto; - padding: 0; - height: 90%; -} - -.no-posts-box .no-posts { - display: flex; - flex-direction: column; - align-items: center; -} - -.no-posts-box .no-posts h3 { - margin-bottom: 20px; - color: var(--midgrey); - font-size: 2em; - font-weight: 200; -} - - -/* Keyboard Focus Effects -/* ---------------------------------------------------------- */ - - -/* This has to be a pseudo element to sit over the top of everything else in the content list */ -.content-list.keyboard-focused:before, -.tag-list-content.keyboard-focused:before, -.tag-settings.keyboard-focused:before { - content: ""; - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - z-index: 500; - animation: keyboard-focus-style-fade-out 1.5s 1 forwards; - pointer-events: none; -} - -.content-preview.keyboard-focused { - animation: keyboard-focus-style-fade-out 1.5s 1 forwards; -} diff --git a/core/client/app/styles/layouts/editor.css b/core/client/app/styles/layouts/editor.css deleted file mode 100644 index 6f15605baa..0000000000 --- a/core/client/app/styles/layouts/editor.css +++ /dev/null @@ -1,403 +0,0 @@ -/* Editor /ghost/editor/ -/* ---------------------------------------------------------- */ - - -/* Title -/* ---------------------------------------------------------- */ - -.gh-editor-title { - flex-grow: 1; -} - -.gh-editor-title input { - margin: 0; - padding: 0; - width: 100%; - border: 0; - background: transparent; - color: var(--darkgrey); - letter-spacing: -1px; - font-size: 2.6rem; - font-weight: bold; -} - -.gh-editor-title input:focus { - outline: 0; -} - -.editor-options .dropdown-menu { - top: 35px; - right: 0; - left: auto; -} - - -/* Container & Headers -/* ---------------------------------------------------------- */ - -.view-editor { - display: flex; -} - -.editor .entry-preview { - border-left: #dfe1e3 1px solid; -} - -.editor .entry-markdown, -.editor .entry-preview { - position: relative; /*TODO: Remove*/ - display: flex; - flex-direction: column; - width: 50%; -} - -/* Content areas at the top, and fill available space */ -.editor .entry-markdown-content, -.editor .entry-preview-content { - order: 1; - flex-grow: 1; -} - -/* Headers at the bottom, and fixed height */ -.editor .floatingheader { - order: 2; - flex-shrink: 0; - display: flex; - justify-content: space-between; - align-items: center; - padding: 5px 15px; - height: 40px; - border-top: #dfe1e3 1px solid; - color: var(--midgrey); - font-size: 1.2rem; - line-height: 1em; -} -.editor .floatingheader a { - padding: 5px 15px; - color: var(--midgrey); -} -.editor .floatingheader a.active { - font-weight: bold; -} -.editor .floatingheader a:first-of-type { - padding-left: 0; -} -.editor .floatingheader a:last-of-type { - padding-right: 0; -} -.editor .floatingheader span a:not(:first-of-type) { - border-left: 1px solid #dfe1e3; -} -.editor .floatingheader .mobile-tabs { - display: none; -} - -/* Switch to 1 col editor on small screens */ -@media (max-width: 1000px) { - .editor .entry-markdown, - .editor .entry-preview { - width: 100%; - border-left: none; - } - /* We can't use display:none here as we want to keep widths/heights - * so that scrolling is kept in sync */ - .editor .entry-markdown:not(.active), - .editor .entry-preview:not(.active) { - visibility: hidden; - position: absolute; - z-index: -1; - height: 100%; - } - .editor .floatingheader .mobile-tabs { - display: inline; - } - .editor .floatingheader .desktop-tabs { - display: none; - } -} - - -/* Editor (Left pane) -/* ---------------------------------------------------------- */ - -.editor .entry-markdown-content { - position: relative; - flex-grow: 1; -} - -.editor .markdown-editor { - /* Legacy absolute positioning */ - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - overflow: auto; - -webkit-overflow-scrolling: touch; - padding: 21px 20px 36px 20px; - max-width: 100%; - height: 100%; - border: 0; - color: color(var(--darkgrey) lightness(+10%)); - font-family: var(--font-family-mono); - font-size: 1.6rem; - line-height: 2.5rem; - resize: none; -} - -.editor .markdown-editor:focus { - outline: 0; -} - -@media (max-width: 450px) { - .editor .markdown-editor { - padding: 15px; - } -} - - -/* FFF: Fucking Firefox Fixes -/* ---------------------------------------------------------- */ - -@-moz-document url-prefix() { - .editor .markdown-editor { - top: 40px; - padding-top: 0; - padding-bottom: 0; - height: calc(100% - 40px); - } -} - - -/* Preview (Right pane) -/* ---------------------------------------------------------- */ - -.editor .entry-preview-content { - flex-grow: 1; - overflow: auto; - -webkit-overflow-scrolling: touch; - padding: 19px 20px 37px 20px; - word-break: break-word; - hyphens: auto; - cursor: default; -} - -/* The styles for the actual content inside the preview */ -.entry-preview-content, -.content-preview-content { - font-size: 1.8rem; - line-height: 1.5em; - font-weight: 200; -} - -.entry-preview-content *, -.content-preview-content * { - user-select: text; -} - -.entry-preview-content a, -.content-preview-content a { - color: var(--blue); - text-decoration: underline; -} - -.entry-preview-content sup a, -.content-preview-content sup a { - text-decoration: none; -} - -.entry-preview-content .btn, -.content-preview-content .btn { - color: #dfe1e3; - text-decoration: none; -} - -.entry-preview-content .img-placeholder, -.content-preview-content .img-placeholder { - position: relative; - height: 100px; - border: 5px dashed #dfe1e3; -} - -.entry-preview-content .img-placeholder span, -.content-preview-content .img-placeholder span { - position: absolute; - top: 50%; - display: block; - margin-top: -15px; - width: 100%; - height: 30px; - text-align: center; -} - -.entry-preview-content a.image-edit, -.content-preview-content a.image-edit { - width: 16px; - height: 16px; -} - -.entry-preview-content img, -.content-preview-content img { - margin: 0 auto; - max-width: 100%; - height: auto; -} - -/* Placeholder objects for <script> & <iframe> */ -.js-embed-placeholder, -.iframe-embed-placeholder { - padding: 100px 20px; - border: none; - background: #f9f9f9; - text-align: center; - font-family: var(--font-family); - font-size: 1.6rem; - font-weight: bold; -} - -/* Tags input CSS (TODO: needs some revision) -/* ------------------------------------------------------ */ -.tags-input-list { - display: flex; - flex-wrap: wrap; - margin: 0; - padding: 0; - list-style-type: none; -} - -.tags-input-list li { - flex: 1 0 auto; -} - -.label-tag { - margin-right: 0.3em; - padding: 0.2em 0.6em 0.3em; - background-color: var(--darkgrey); - border-radius: 0.25em; - color: var(--lightgrey); - text-align: center; - font-weight: 300; -} - -.label-tag.highlight { - background: var(--midgrey); - color: #fff; -} - -.tag-input { - margin-top: 5px; - border: none; - font-weight: 300; - cursor: default; -} - -.tag-input:focus { - outline: 0; -} - -.publish-bar-actions { - flex: 1 0 auto; - align-self: auto; - display: flex; - text-align: right; -} - -.post-settings { - position: relative; - display: inline-block; - padding: 15px; - color: var(--midgrey); - transition: all 0.15s ease-out 0s; -} - -.post-settings:hover, -.post-settings.active { - color: var(--darkgrey); -} - -.post-settings i { - width: 16px; - height: 16px; - font-size: 16px; - line-height: 16px; -} - -.post-settings-menu .dropdown-menu { - top: auto; - right: 100%; - bottom: 100%; - left: auto; -} - -.post-view-link { - position: absolute; - top: 1px; - right: 0; - font-size: 1.3rem; -} - -.post-view-link i { - display: inline; - font-size: 10px; -} - - -/* Post settings meta -/* ---------------------------------------------------------- */ - -/* Google Imitation */ -.seo-preview { - font-family: Arial, sans-serif; -} - -.seo-preview-title { - color: #1e0fbe; - text-overflow: ellipses; - word-wrap: break-word; - font-size: 1.8rem; - line-height: 2.16rem; - - -webkit-text-overflow: ellipsis; -} - -.seo-preview-link { - margin: 1px 0 2px 0; - color: #006621; - word-wrap: break-word; - font-size: 1.3rem; - line-height: 1.6rem; -} - -.seo-preview-description { - color: #545454; - word-wrap: break-word; - font-size: 1.3rem; - line-height: 1.4; -} - - -/* Markdown Help Icon + Modal -/* ---------------------------------------------------------- */ - -.markdown-help-icon { - font-size: 16px; -} - -.markdown-help-icon:hover, -.markdown-help-label:hover { - cursor: help; -} - -.modal-markdown-help-table { - margin: 0 0 20px; - width: 100%; -} - -.modal-markdown-help-table td, -.modal-markdown-help-table th { - padding: 8px 0; -} - -.modal-markdown-help-table th { - text-align: left; -} diff --git a/core/client/app/styles/layouts/error.css b/core/client/app/styles/layouts/error.css deleted file mode 100644 index 18f20d2773..0000000000 --- a/core/client/app/styles/layouts/error.css +++ /dev/null @@ -1,88 +0,0 @@ -/* Error /ghost/404/ -/* ---------------------------------------------------------- */ - -.error-content { - flex-grow: 1; - display: flex; - justify-content: center; - align-items: center; - user-select: text; -} - -.error-details { - display: flex; - align-items: center; - margin-bottom: 4rem; -} - -.error-ghost { - margin: 15px; - height: 115px; -} - -@media (max-width: 630px) { - .error-ghost { - display: none; - } -} - -.error-code { - margin: 0; - color: #979797; - font-size: 7.8rem; - line-height: 0.9em; -} - - -.error-description { - margin: 0; - padding: 0; - border: none; - color: #979797; - font-size: 1.9rem; - font-weight: 300; -} - -.error-message { - display: flex; - flex-direction: column; - margin: 15px; -} - -.error-message a { - margin-top: 5px; - font-size: 1.4rem; - line-height: 1; -} - -/* Stack trace -/* ---------------------------------------------------------- */ - -.error-stack { - margin: 1rem auto; - padding: 2rem; - max-width: 800px; - background-color: rgba(255, 255, 255, 0.3); -} - -.error-stack-list { - margin: 0; - padding: 0; - list-style-type: none; -} - -.error-stack-list li { - display: block; -} - -.error-stack-list li:before { - content: "\21AA"; - display: inline-block; - margin-right: 0.5rem; - color: #bbb; - font-size: 1.2rem; -} - -.error-stack-function { - font-weight: bold; -} diff --git a/core/client/app/styles/layouts/flow.css b/core/client/app/styles/layouts/flow.css deleted file mode 100644 index 0a8082b4dc..0000000000 --- a/core/client/app/styles/layouts/flow.css +++ /dev/null @@ -1,464 +0,0 @@ -/* Full screen workflow -/* ---------------------------------------------------------- */ - -.gh-flow { - flex-grow: 1; - display: flex; - flex-direction: column; - overflow-y: auto; - min-height: 100%; -} - -.gh-flow-head { - flex-shrink: 0; - display: flex; - justify-content: space-between; - padding-top: 4vh; - padding-bottom: 20px; -} - -.gh-flow-content-wrap { - flex-grow: 1; - flex-shrink: 0; - display: flex; - justify-content: center; - align-items: center; - margin: 0 5%; - padding-bottom: 8vh; -} - -.gh-flow-back { - position: absolute; - top: 0; - left: 0; - display: flex; - align-items: center; - margin: 0 0 0 3%; - padding: 2px 9px 2px 5px; - border: transparent 1px solid; - border-radius: 4px; - color: #7d878a; - font-weight: 100; - transition: all 0.3s ease; -} - -.gh-flow-back i { - margin-right: 4px; - font-size: 12px; - line-height: 8px; -} - -.gh-flow-back:hover { - border: #dae1e3 1px solid; -} - -.gh-flow-nav { - position: relative; - flex: 1; -} - -.gh-flow-nav ol { - display: flex; - justify-content: space-between; - margin: 0 auto; - padding: 0; - width: 160px; - list-style: none; -} - -.gh-flow-nav li { - margin: 0; -} - -.gh-flow-nav .divider { - align-self: center; - width: 22px; - height: 2px; - background-image: linear-gradient(to right, var(--green) 33%, rgba(255, 255, 255, 0) 0%); - background-position: bottom; - background-size: 6px 2px; - background-repeat: repeat-x; -} - -.gh-flow-nav .active ~ .divider { - background-image: linear-gradient(to right, #e3e3e3 33%, rgba(255, 255, 255, 0) 0%); -} - -.gh-flow-nav .step { - display: flex; - justify-content: center; - align-items: center; - width: 30px; - height: 30px; - border: transparent 2px solid; - background: var(--green); - border-radius: 100%; - color: #fff; - vertical-align: middle; - text-align: center; - text-align: center; - font-size: 1.3rem; - line-height: 1; -} - -.gh-flow-nav .step .num { - display: none; -} - -.gh-flow-nav .step i { - width: 26px; - height: 26px; - font-size: 26px; -} - -.gh-flow-nav .active ~ li:not(divider) .step { - border: #e3e3e3 2px solid; - background: transparent; - color: #cdcdcd; -} - -.gh-flow-nav .active ~ li:not(divider) .step .num { - display: block; -} - -.gh-flow-nav .active ~ li:not(divider) .step i { - display: none; -} - -.gh-flow-nav .active .step { - border: var(--green) 2px solid; - background: transparent; - color: color(var(--green) lightness(-10%)); - cursor: default; -} - -.gh-flow-nav .active .step .num { - display: block; -} - -.gh-flow-nav .active .step i { - display: none; -} - -.gh-flow-nav .done { - border: none; - background: var(--green); - color: #fff; -} - - -.gh-flow-content { - display: flex; - flex-direction: column; - max-width: 700px; - width: 100%; - color: var(--midgrey); - text-align: center; - font-size: 1.9rem; - line-height: 1.5em; - font-weight: 100; -} - -@media (max-width: 500px) { - .gh-flow-content { - font-size: 4vw; - } -} - -.gh-flow-content header { - margin: 0 auto; - max-width: 520px; -} - -.gh-flow-content h1 { - letter-spacing: -1px; - font-size: 4.2rem; - font-weight: 100; -} - -@media (max-width: 600px) { - .gh-flow-content h1 { - font-size: 7vw; - } -} - -.gh-flow-content strong { - font-weight: 400; -} - -.gh-flow-content em { - color: var(--blue); - font-weight: 400; - font-style: normal; -} - -.gh-flow-content .gh-flow-screenshot { - display: flex; - align-items: center; - margin: 0; - height: 45vh; -} - -.gh-flow-content .gh-flow-screenshot img { - position: relative; - left: -3%; - flex-shrink: 0; - display: block; - margin: 0 auto; - max-height: 100%; -} - -@media (max-width: 860px) { - .gh-flow-content .gh-flow-screenshot img { - left: 0; - } -} -@media (max-width: 600px) { - .gh-flow-content .gh-flow-screenshot { - height: auto; - } -} - -.gh-flow-content .btn { - display: block; - margin: 20px auto 0; - max-width: 400px; -} - -.gh-flow-content .gh-flow-skip { - display: inline-block; - margin-top: 5px; - color: #7d878a; - font-size: 1.2rem; -} - -.gh-flow-content .gh-flow-create { - position: relative; - margin: 70px auto 30px; - padding: 50px 40px 25px; - max-width: 400px; - width: 100%; - border: #dae1e3 1px solid; - background: #f8fbfd; - border-radius: 5px; - text-align: left; -} - -.gh-flow-content .account-image { - position: absolute; - top: -50px; - left: 50%; - overflow: hidden; - margin: 0; - margin-left: -50px; - padding: 4px; - width: 100px; - height: 100px; - border: #d1d9db 1px solid; - background: #fff; - border-radius: 100%; - text-align: center; -} - -.gh-flow-content .account-image:hover .edit-account-image { - opacity: 1; -} - -.gh-flow-content .edit-account-image { - position: absolute; - top: 4px; - right: 4px; - bottom: 4px; - left: 4px; - width: calc(100% - 8px); - background: rgba(87, 163, 232, 0.7); - border-radius: 100%; - color: #fff; - text-decoration: none; - text-transform: uppercase; - font-size: 3rem; - line-height: 90px; - opacity: 0; - transition: opacity 0.3s ease; -} - -.gh-flow-content .placeholder-img { - display: block; - width: 90px; - height: 90px; - background-color: #f8fbfd; - background-position: center center; - background-size: cover; - border-radius: 100%; - animation: fade-in 1s; -} - -.gh-flow-content .gravatar-img { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - display: block; - box-sizing: content-box; - width: calc(100% - 8px); - width: 90px; - height: 90px; - border: #fff 4px solid; - background-position: center center; - background-size: cover; - border-radius: 100%; - animation: fade-in 1s; -} - -.gh-flow-content .file-uploader { - position: absolute; - right: 0; - margin: 0; - font-size: 23px; - opacity: 0; - cursor: pointer; - transform: scale(14); - transform-origin: right; - direction: ltr; -} - -.gh-flow-content .form-group { - margin-bottom: 2.5rem; -} - -.gh-flow-content .form-group label { - margin: 0; - font-size: 1.4rem; - font-weight: 400; -} - -.gh-flow-content .form-group a { - text-decoration: underline; -} - -.gh-flow-content input { - padding: 10px; - border: #dae1e3 1px solid; - font-size: 1.6rem; - line-height: 1.4em; - font-weight: 100; -} - -.gh-flow-content .pw-strength { - position: absolute; - top: 50%; - right: 1px; - margin-top: -11px; - padding: 0 10px; - height: 24px; - background: rgba(255, 255, 255, 0.9); -} - -.gh-flow-content .pw-strength-dot { - display: block; - margin-top: 2px; - width: 3px; - height: 3px; - background-color: #d9e0e3; - border-radius: 100%; -} - -.gh-flow-content .pw-strength-dot:first-child { - margin-top: 0; -} - -.gh-flow-content .pw-strength-activedot { - background-color: var(--red); -} - -.gh-flow-content .input-icon[class*="icon-"]:before { - transform: translateY(-49%); -} - -.gh-flow-content .gh-flow-invite { - position: relative; - margin: 0 auto; - max-width: 400px; - width: 100%; - text-align: left; -} - -.gh-flow-invite label { - display: flex; - justify-content: space-between; - align-items: center; -} - -.gh-flow-invite label i { - width: 14px; - height: 14px; - font-size: 14px; - line-height: 1.4em; -} - -.gh-flow-invite textarea { - background: url(img/invite-placeholder.png) 8px 10px no-repeat; - background-size: 202px 48px; - box-shadow: none; /* Remove some default styling for Firefox (required attribute) */ -} - -.gh-flow-invite textarea:valid { - background: none; -} - -@media (max-width: 460px) { - .gh-flow-content .gh-flow-invite label i { - display: none; - } -} - -.gh-flow-content .gh-flow-faces { - margin-bottom: 2vw; - width: 100%; -} - -.gh-flow-content textarea { - width: 100%; - height: 160px; - font-size: 1.6rem; - font-weight: 100; -} - -.gh-flow-content .response { - position: absolute; - right: 0; - bottom: -25px; - margin: 0; - color: #a6b0b3; - text-align: right; - font-size: 1.2rem; -} - -.gh-flow-content .success .input-icon:before { - color: var(--green); -} - -.error, -.error-content, -.main-error { - user-select: text; -} - -.gh-flow-content .error input { - border-color: var(--red); -} - -.gh-flow-content .error .input-icon:before { - color: var(--red); -} - -.gh-flow-content .error .response { - color: var(--red); -} - -.gh-flow-content .main-error { - margin-top: 5px; - color: var(--red); - font-size: 1.3rem; -} diff --git a/core/client/app/styles/layouts/main.css b/core/client/app/styles/layouts/main.css deleted file mode 100644 index 4d5f995fad..0000000000 --- a/core/client/app/styles/layouts/main.css +++ /dev/null @@ -1,560 +0,0 @@ -/* Global Layout -/* ---------------------------------------------------------- */ - -/* - Ember's app container, set height so that .gh-app and .gh-viewport - don't need to use 100vh where bottom of screen gets covered by iOS menus - http://nicolas-hoizey.com/2015/02/viewport-height-is-taller-than-the-visible-part-of-the-document-in-some-mobile-browsers.html - - TODO: Once we have routable components it should be possible to remove this - by moving the gh-app component functionality into the application component - which would remove the extra div that this targets. -*/ -body > .ember-view:not(.liquid-target-container) { - height: 100%; -} - -/* Main viewport, contains main content, and alerts */ -.gh-app { - display: flex; - flex-direction: column; - overflow: hidden; - height: 100%; -} - -/* Content viewport, contains everything else */ -.gh-viewport { - flex-grow: 1; - display: flex; - overflow: hidden; - max-height: 100%; -} - -.gh-main { - position: relative; - flex-grow: 1; - display: flex; - background: #fff; -} - -/* Flexbox fix. https://github.com/TryGhost/Ghost/issues/5804#issuecomment-141416812 */ -.gh-main > section { - width: 1px; -} - - -/* Global Nav -/* ---------------------------------------------------------- */ - -.gh-nav { - position: relative; - z-index: 800; - flex: 0 0 235px; - display: flex; - flex-direction: column; - min-width: 0; /* TODO: This is a bullshit Firefox hack */ - border-right: #dfe1e3 1px solid; - background: #f5f7f8; - transform: translateX(0); -} - -.gh-nav-menu { - flex-shrink: 0; - display: flex; - align-items: center; - padding: 15px; - cursor: pointer; -} - -.gh-nav-menu i { - margin-right: 8px; - width: 11px; - height: 11px; - font-size: 11px; - line-height: 11px; - transition: margin-top 0.2s ease; -} - -.gh-nav-menu-icon { - flex-shrink: 0; - margin-right: 10px; - width: 34px; - height: 34px; - background-color: #222; - background-size: 34px; - border-radius: 4px; -} - -.gh-nav-menu-details { - flex-grow: 1; - padding-right: 10px; - min-width: 0; /* TODO: This is a bullshit Firefox hack */ -} - -.gh-nav-menu-details-blog { - overflow: hidden; - margin-bottom: 1px; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 1.5rem; - line-height: 1.3em; - font-weight: 600; -} - -.gh-nav-menu-details-user { - overflow: hidden; - color: var(--midgrey); - text-overflow: ellipsis; - white-space: nowrap; - font-size: 1.2rem; - line-height: 1.2em; -} - -.gh-nav-body { - flex-grow: 1; - overflow-y: auto; -} - -.gh-nav-search { - position: relative; - margin: 0 15px 10px; -} - -.gh-nav-search .selectize-control { - display: flex; -} - -.gh-nav-search-input .selectize-input { - padding: 4px 8px; - padding-right: 30px; - height: auto; -} -.gh-nav-search-input .selectize-input, -.gh-nav-search-input .selectize-input input, -.gh-nav-search-input .selectize-dropdown { - font-size: 1.3rem; -} - -.gh-nav-search .selectize-input.dropdown-active { - border-bottom: #fff 1px solid; -} - -.gh-nav-search .selectize-input.dropdown-active:before { - display: none; -} - -.gh-nav-search .selectize-dropdown-content { - max-height: calc(100vh - 150px); -} - -.gh-nav-search-button { - position: absolute; - top: 0; - right: 0; - bottom: 0; - z-index: 1; - padding: 0 8px 0 5px; -} - -.gh-nav-search-button i { - width: 16px; - height: 16px; - color: var(--midgrey); - font-size: 13px; - line-height: 13px; - transition: color 0.2s ease; - transform: rotate(90deg); -} - -.gh-nav-search-button:hover i { - color: var(--darkgrey); -} - -.gh-nav-list { - margin: 0; - padding: 0 15px 0 0; - list-style: none; - font-size: 1.3rem; - line-height: 1.5em; -} - -.gh-nav-list-h { - overflow: hidden; - margin-top: 15px; - padding: 5px 10px 5px 15px; - color: #808284; - text-transform: uppercase; - text-overflow: ellipsis; - white-space: nowrap; - letter-spacing: 1px; - font-size: 1.2rem; - line-height: 1.1em; -} - -.gh-nav-list a { - display: flex; - align-items: center; - padding: 5px 10px 5px 15px; - border-radius: 0 4px 4px 0; - color: var(--darkgrey); - transition: none; -} - -.gh-nav-list .active { - background: color(var(--blue) lightness(+10%)); - color: #fff; -} - -.gh-nav-list a:not(.active):hover { - background: color(var(--blue) alpha(-85%)); - color: var(--darkgrey); -} - -.gh-nav-list i { - margin-right: 8px; - width: 15px; - height: 15px; - color: rgba(0,0,0,0.6); - text-align: center; - font-size: 15px; - line-height: 1; -} - -.gh-nav-list .active i { - color: #fff; -} - -.gh-nav-list a:not(.active):hover i { - color: var(--darkgrey); -} - -.gh-nav-footer { - flex-shrink: 0; - display: flex; - align-items: center; - height: 40px; - border-top: #dfe1e3 1px solid; - color: var(--midgrey); -} - -.gh-nav-footer-sitelink { - flex-grow: 1; - padding: 12px; - color: color(var(--midgrey) lightness(-10%)); - text-align: center; - text-transform: uppercase; - font-size: 1rem; - line-height: 1; - font-weight: 200; -} - -.gh-nav-footer-sitelink i { - margin-left: 5px; - font-size: 1rem; -} - -.gh-nav-footer-sitelink:hover { - color: var(--blue); -} - - -/* Mobile Nav -/* ---------------------------------------------------------- */ - -.gh-mobilemenu-button { - display: none; -} - -@media (max-width: 800px) { - .view-header { - padding-left: 0 !important; - } - - .gh-mobilemenu-button { - flex-shrink: 0; - display: block; - margin: 0; - padding: 24px; - font-size: 18px; - line-height: 18px; - } - - .gh-mobilemenu-button .icon-gh { - margin: 0; - } - - /* Hide the nav */ - .gh-nav { - position: absolute; - top: 0; - left: 0; - width: 270px; - height: 100%; - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); - transform: translate3d(-270px, 0px, 0px); - } - .mobile-menu-expanded .gh-nav { - transform: translate3d(0,0,0); - } - - .gh-nav-list a { - padding: 7px 10px 7px 15px; - border-radius: 0 4px 4px 0; - } -} - -@media (max-width: 500px) { - .gh-mobilemenu-button { - padding: 24px 15px 24px 16px; - } - - .gh-nav { - width: 80vw; - transform: translate3d(-80vw, 0px, 0px); - } - .mobile-menu-expanded .gh-nav { - transform: translate3d(0,0,0); - } - .mobile-menu-expanded .content-cover { - transform: translate3d(80vw, 0, 0); - } - - .gh-nav-search-input .selectize-input, - .gh-nav-search-input .selectize-input input, - .gh-nav-search-input .selectize-dropdown { - font-size: 1.5rem; - } - - .gh-nav-list { - font-size: 1.5rem; - } - - .gh-nav-list-h { - font-size: 1.4rem; - } - - .gh-nav-list i { - margin-right: 8px; - width: 17px; - height: 17px; - text-align: center; - font-size: 17px; - } -} - - -/* Auto Nav - Opens and closes like OSX dock -/* ---------------------------------------------------------- */ - -.gh-menu-toggle { - display: flex; - justify-content: center; - align-items: center; - padding: 5px 10px; - width: 45px; - height: 27px; - border-right: #dfe1e3 1px solid; - line-height: 1; - cursor: pointer; -} - -.gh-menu-toggle:hover { - cursor: pointer; -} - -.gh-menu-toggle i { - transition: all 0.2s ease; -} - -.gh-menu-toggle:hover i { - color: var(--blue); -} - -/* Autonav is tricky, because hit areas of translated elements aren't in sync - with the visible element we need to add the hover behaviour to a small, - non-moving element. The following code positions our hit area and transitions - it in-sync with it's container so it always sticks to the left of the viewport - then hides off-canvas when required as display:none breaks transitions. */ - -.gh-autonav-toggle { - position: absolute; - top: 0; - right: 0; - display: none; - width: 15px; - height: 100%; - transition: transform 0.20s; - transform: translate3d(0,0,0); -} - -.gh-autonav .gh-autonav-toggle { - display: block; -} - -.gh-nav.open .gh-autonav-toggle { - transition: transform 0.15s; - transform: translate3d(-235px,0,0); -} - -@media (min-width: 801px) { - /* Hide the nav */ - .gh-autonav .gh-nav { - position: absolute; - top: 0; - left: 0; - z-index: 1000; - width: 235px; - height: 100%; - transition: transform 0.20s; - /* translate3d for GPU accelerated animation - http://bit.ly/1EY1Xhx */ - transform: translate3d(-220px,0,0); - } - - /* THE FUTURE: Super sexy background blur for Webkit - http://cl.ly/b1rG */ - @supports (-webkit-backdrop-filter: none) or (backdrop-filter: none) { - .gh-autonav .gh-nav { - background: rgba(246,246,246, 0.7); - - -webkit-backdrop-filter: blur(10px); - backdrop-filter: blur(10px); - } - } - - /* Bring it back on hover */ - .gh-autonav .gh-nav.open { - transition: transform 0.15s; - transform: translate3d(0,0,0); - } - - /* Move main content over for the closed-nav trigger bar */ - .gh-autonav .gh-main { - margin-left: 15px; - } -} - - -/* Help (?) Menu -/* ---------------------------------------------------------- */ - -.gh-help-menu { - display: flex; - align-items: center; - border-left: #dfe1e3 1px solid; - cursor: pointer; -} - -.gh-help-button { - padding: 5px 15px; - color: var(--midgrey); - text-align: center; - font-size: 1.4rem; - line-height: 1.2em; - transition: all 0.5s; -} - -.gh-help-menu:hover .gh-help-button { - color: var(--blue); - transition: all 0.3s; -} - -.gh-help-menu .dropdown { - bottom: 215px; - left: -180px; -} - -.gh-help-menu .dropdown.fade-in-scale { - animation-duration: 0.1s; -} - -.gh-help-menu .dropdown.fade-out { - animation-duration: 0.01s; -} - - -/* Container for App View -/* ---------------------------------------------------------- */ - -.gh-view { - flex-grow: 1; - display: flex; - flex-direction: column; -} - -.view-header { - flex-shrink: 0; - display: flex; - justify-content: space-between; - align-items: center; - padding: 0 20px; - height: 65px; - border-bottom: #dfe1e3 1px solid; -} - -.view-title { - display: flex; - align-items: center; - overflow: hidden; - margin: 0; - padding: 0; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 2rem; - line-height: 1.2em; - font-weight: 400; -} - -.view-title a { - color: inherit; -} - -.view-title a:hover { - color: var(--blue); -} - -.view-title i { - margin: 0 10px; - color: #818181; - font-size: 14px; - line-height: 12px; -} - -.view-actions { - flex-shrink: 0; - display: flex; - align-items: center; -} - -.view-actions .btn { - margin-left: 8px; -} - -.view-container, -.view-content { - position: relative; - flex-grow: 1; - overflow-x: hidden; - overflow-y: auto; - -webkit-overflow-scrolling: touch; -} - -.view-content { - padding: 20px; -} - -.view-content p, -.view-content p * { - user-select: text; -} - -@media (max-width: 400px) { - .view-header { - padding: 0 7px; - height: 50px; - } - .view-title { - font-size: 1.8rem; - } - .view-content { - padding: 15px; - } -} diff --git a/core/client/app/styles/layouts/packages.css b/core/client/app/styles/layouts/packages.css deleted file mode 100644 index 5794a5ea62..0000000000 --- a/core/client/app/styles/layouts/packages.css +++ /dev/null @@ -1,376 +0,0 @@ -/* Packages - Themes / Apps -/* ---------------------------------------------------------- */ - -.package-filter { - border-radius: 5px; -} -@media (max-width: 1460px) { - .package-filter { - max-width: 700px; - } -} - - -/* Main Layout -/* ---------------------------------------------------------- */ - -.package-grid { - display: flex; - flex-flow: row wrap; - align-items: space-between; - margin: -10px -10px 4vw -10px; - max-width: 1200px; -} - -/* 3 col themes */ -.package-grid-themes .package-grid-cell { - flex: 0 0 33.3333%; -} - -/* 2 col themes */ -@media (max-width: 1240px) { - .package-grid-themes .package-grid-cell { - flex: 0 0 100%; - } -} - -/* 1 col themes */ -@media (max-width: 800px) { - .package-grid-themes .package-grid-cell { - flex: 1 1 100%; - } -} - -/* 2 col apps */ -.package-grid-apps .package-grid-cell { - flex: 0 0 100%; -} - -/* 1 col apps */ -@media (max-width: 1200px) { - .package-grid-apps .package-grid-cell { - flex: 1 1 100%; - } -} - - -/* Package Card Theme -/* ---------------------------------------------------------- */ - -.package-card-theme { - overflow: hidden; - margin: 10px; - border: rgba(0,0,0,0.1) 1px solid; - border-radius: 5px; -} - -.package-index .package-card-theme, -.package-featured .package-card-theme { - flex: 1 1 240px; -} - -.package-card-theme-image { - position: relative; - display: block; -} - -.package-card-theme-image:hover img { - filter: grayscale(0.5) blur(1px); - - -webkit-filter: grayscale(0.5) blur(1px); -} - -.package-card-theme-image:hover .package-card-theme-overlay { - opacity: 1; - transition: all 0.2s ease; -} - -.package-card-theme-image img { - display: block; - max-width: 100%; - line-height: 0; -} - -.package-card-theme-overlay { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - display: flex; - justify-content: center; - align-items: center; - padding: 10%; - background: rgba(0,20,40,0.2); - text-align: center; - opacity: 0; - transition: opacity 0.4s ease; -} - -.package-card-theme-title { - color: #fff; - font-size: 2rem; - line-height: 1.15em; - font-weight: 100; -} - -.package-card-theme .package-card-footer { - margin: 0; - padding: 16px 20px; - border-top: rgba(0,0,0,0.1) 1px solid; -} - -/* Package Card App -/* ---------------------------------------------------------- */ - -.package-card-app { - overflow: hidden; - margin: 10px; - padding: 14px; - height: 75px; - /*max-width: 700px;*/ - border: rgba(0,0,0,0.1) 1px solid; - border-radius: 5px; - transition: background 0.3s ease; -} - -.package-card-app:hover { - background: rgba(0,20,60,0.03); - cursor: pointer; - transition: background 0.1s ease; -} - -.package-card-content { - position: relative; - display: flex; -} - -.package-card-content .btn { - position: absolute; - right: 20px; -} - -.package-card-app-icon { - flex: 0 0 47px; - margin: 0 15px 0 0; - width: 47px; - height: 47px; - background-position: center center; - background-size: cover; - border-radius: 15%; -} - -.package-card-meta { - position: relative; - display: flex; - flex-direction: column; -} - -.package-card-app-title { - overflow: hidden; - margin: 0 0 4px 0; - padding: 0 70px 0 0; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 1.7rem; - font-weight: normal; -} - - -/* Package Card Meta -/* ---------------------------------------------------------- */ - -.package-card-stats { - position: absolute; - top: -5px; - right: 0; - display: flex; - align-items: center; -} - -.package-downloads { - display: flex; - align-items: center; - height: 26px; - border: transparent 1px solid; - color: var(--midgrey); - font-size: 13px; - line-height: 24px; -} - -.package-downloads:hover { - cursor: default; -} - -.package-downloads i { - margin-right: 5px; - font-size: 15px; -} - - -.package-download-count { - font-size: 13px; -} - -.package-card-app-desc { - display: -webkit-box; - overflow: hidden; - margin: 0; - padding: 0; - max-height: 4.2rem; - color: var(--midgrey); - text-overflow: ellipsis; - font-size: 1.4rem; - line-height: 1.3em; - font-weight: 200; - - -webkit-line-clamp: 2; - -webkit-box-orient: vertical; -} -@media (min-width: 600px) and (max-width: 1460px) { - .package-card-app-desc { - padding-right: 80px; - } -} - - -/* Package Card Footer -/* ---------------------------------------------------------- */ - -.package-card-footer { - display: flex; - justify-content: space-between; - align-items: center; - margin-top: 15px; - width: 100%; -} - -.package-developer { - display: flex; - align-items: center; - color: var(--midgrey); -} - -.package-developer:hover { - color: var(--blue); -} - -.package-developer img { - flex-shrink: 0; - margin-right: 6px; - width: 20px; - height: 20px; - border-radius: 100%; -} - -.package-developer-name { - overflow: hidden; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 1.4rem; -} - -.package-controls { - flex-shrink: 0; - display: flex; - overflow: hidden; - border: rgba(0,0,0,0.1) 1px solid; - border-radius: 4px; -} - -.package-controls-button { - display: flex; - align-items: center; - padding: 7px 12px; - border-left: rgba(0,0,0,0.1) 1px solid; - background: #fff; - color: var(--midgrey); - font-size: 1.3rem; - line-height: 1; - transition: none; -} - -.package-controls-button:first-child { - border: none; -} - -.package-controls-button:hover { - color: var(--darkgrey); -} - -.package-controls-button i { - margin-right: 5px; - width: 11px; - height: 11px; - font-size: 11px; -} - -.package-disable { - border-right: var(--green) 3px solid; -} - -.package-enable { - border-right: var(--red) 3px solid; -} - - -/* Media Queries -/* ---------------------------------------------------------- */ - -@media (max-width: 800px) { - .package-grid-apps { - overflow: hidden; - margin: 0 0 4vw 0; - border: #dfe1e3 1px solid; - border-radius: 5px; - } - - .package-card-app { - margin: 0; - border: none; - border-top: #dfe1e3 1px solid; - border-radius: 0; - } - .package-grid-cell:first-of-type .package-card-app { - border-top: none; - } -} - -@media (max-width: 760px) { - .package-card-app { - padding: 15px; - } - .package-card-app .package-developer { - display: none; - } - .package-card-app .package-card-footer { - justify-content: flex-end; - } - .package-card-theme .package-card-footer { - margin: 0; - padding: 15px; - } -} - -@media (max-width: 600px) { - .package-grid { - margin: -10px -10px 4vw -10px; - border: none; - } - .package-grid-apps { - margin: -10px -20px 4vw -20px; - } -} - -@media (max-width: 540px) { - .package-card-footer { - justify-content: flex-end; - } - .package-card-app .package-card-footer { - flex-direction: column; - align-items: flex-start; - } - .package-card-footer .package-developer { - display: none; - } -} diff --git a/core/client/app/styles/layouts/settings.css b/core/client/app/styles/layouts/settings.css deleted file mode 100644 index f3911976a0..0000000000 --- a/core/client/app/styles/layouts/settings.css +++ /dev/null @@ -1,163 +0,0 @@ -/* Settings -/* ---------------------------------------------------------- */ - - -/* Navigation -/* ---------------------------------------------------------- */ - -.gh-blognav { - margin: 20px 0; -} - -.gh-blognav-item { - display: flex; - align-items: center; - margin-bottom: 10px; - padding: 0 20px; -} - -.gh-blognav-item--error { - margin-bottom: calc(1em + 10px); -} - -.gh-blognav-item .response { - position: absolute; - margin-bottom: 0; -} - -.gh-blognav-grab { - padding: 0 16px 0 0; - width: 16px; - color: #d1d1d1; - text-indent: -4px; - font-size: 16px; - cursor: move; -} - -.gh-blognav-line { - display: flex; - width: 100%; -} - -.gh-blognav-label { - flex-grow: 1; - margin-right: 10px; -} - -.gh-blognav-url { - flex-grow: 3; -} - -.gh-blognav-delete { - padding: 8px 0 8px 10px; - color: #c1c1c1; - font-size: 14px; - transition: color 0.1s linear; -} - -.gh-blognav-delete:hover, -.gh-blognav-delete:focus { - color: var(--red); -} - -.gh-blognav-add { - margin-right: -2px; - margin-left: 10px; - width: 16px; - height: 16px; - background: var(--green); - border-radius: 2px; - color: #fff; - text-align: center; - font-size: 10px; - line-height: 8px; - transition: background 0.1s linear; -} - -.gh-blognav-add:hover, -.gh-blognav-add:focus { - background: color(var(--green) lightness(-10%)); -} - -.gh-blognav-item:not(.gh-blognav-item--sortable) { - padding-left: calc(16px + 20px); - /* icon-grab + nav-item padding) */ -} - -/* Remove space between inputs on smaller screens */ -@media (max-width: 800px) { - .gh-blognav-label { - margin-right: -1px; - } - .gh-blognav-label input { - border-right-color: #eaeaea; - border-radius: 4px 0 0 4px; - } - .gh-blognav-url input { - border-left-color: #eaeaea; - border-radius: 0 4px 4px 0; - } - .gh-blognav-item input:focus { - position: relative; - z-index: 100; - } -} - - -/* Code Injection -/* ---------------------------------------------------------- */ - -.settings-code { - max-width: 700px; -} - -.settings-code p { - margin: 0 0 4px 0; -} - -.settings-code code { - vertical-align: middle; -} - -.settings-code-editor { - padding: 0; - min-width: 250px; - min-height: 300px; - max-width: 680px; - width: 100%; - height: auto; - border: 1px solid #e0dfd7; - border-radius: var(--border-radius); - line-height: 22px; - transition: border-color 0.15s linear; - - -webkit-appearance: none; -} - -.settings-code-editor.focused { - outline: 0; - border-color: var(--midgrey); -} - -.settings-code-editor .CodeMirror { - border-radius: inherit; -} - -.settings-code-editor .cm-s-xq-light span.cm-meta { - color: #000; -} - - - -/* Labs -/* ---------------------------------------------------------- */ - -#startupload { - line-height: inherit; -} - -@media (max-width: 400px) { - #startupload { - margin-top: 5px; - } -} diff --git a/core/client/app/styles/layouts/subscribers.css b/core/client/app/styles/layouts/subscribers.css deleted file mode 100644 index 4cb4efc1ce..0000000000 --- a/core/client/app/styles/layouts/subscribers.css +++ /dev/null @@ -1,69 +0,0 @@ -/* Subscribers Management /ghost/subscribers/ -/* ---------------------------------------------------------- */ - -.view-subscribers .view-container { - display: flex; - flex-direction: row; - min-height: 100%; -} - - -/* Table (left pane) -/* ---------------------------------------------------------- */ - -.subscribers-table { - flex-grow: 1; - overflow-y: auto; - padding: 0 12px; /* ember-light-table has 8px padding on cells */ - max-height: 100%; -} - -.subscribers-table table { - margin: 0; -} - -.subscribers-table table .btn { - visibility: hidden; -} - -.subscribers-table table tr:hover .btn { - visibility: visible; -} - -.subscribers-table tbody td:last-of-type { - padding-top: 0; - padding-bottom: 0; - padding-left: 0; -} - - -/* Sidebar (right pane) -/* ---------------------------------------------------------- */ - -.subscribers-sidebar { - width: 350px; - border-left: 1px solid #dfe1e3; -} - -.subscribers-import-buttons { - display: flex; - flex-direction: row; -} - -.subscribers-import-buttons .btn { - flex-grow: 1; - margin-right: 10px; -} - -.subscribers-import-buttons .btn:last-of-type { - margin-right: 0; -} - - -/* Import modal -/* ---------------------------------------------------------- */ - -.subscribers-import-results { - margin: 0; - width: auto; -} diff --git a/core/client/app/styles/layouts/tags.css b/core/client/app/styles/layouts/tags.css deleted file mode 100644 index 54f47bc438..0000000000 --- a/core/client/app/styles/layouts/tags.css +++ /dev/null @@ -1,129 +0,0 @@ -/* Tag Management /ghost/settings/tags/ -/* ---------------------------------------------------------- */ - - -/* Tag -/* ---------------------------------------------------------- */ - -.settings-tag { - position: relative; - display: block; - padding: 0 45px 0 0; - border-bottom: 1px solid #dfe1e3; -} - -.settings-tag .tag-edit-button { - display: block; - padding: 20px; - width: calc(100% + 45px); - text-align: left; -} - -.settings-tag .tag-edit-button.active { - border-left: 3px solid; -} - -.settings-tag .label { - display: inline-block; - overflow: hidden; - max-width: 100%; - vertical-align: middle; - text-overflow: ellipsis; - white-space: nowrap; -} - -.settings-tag .label-alt { - text-transform: uppercase; -} - -.settings-tag .tag-title { - color: var(--darkgrey); - font-size: 16px; - font-weight: normal; -} - -.settings-tag .tag-description { - margin: 0; - color: color(#dfe1e3 lightness(-10%)); - word-wrap: break-word; - font-size: 13px; -} - -.settings-tag .tags-count { - position: absolute; - top: 20px; - right: 12px; - color: color(#dfe1e3 lightness(-10%)); - font-size: 16px; -} - -/* Tag List (Left pane) -/* ---------------------------------------------------------- */ - -.tag-list { - position: absolute; - top: 0; - bottom: 0; - left: 0; - overflow: auto; - max-width: calc(100% - 350px); - width: 66%; - border-right: #dfe1e3 1px solid; - background: #fff; -} - -@media (max-width: 600px) { - .tag-list { - max-width: 100%; - width: 100%; - } - - .settings-tag .tag-edit-button.active { - border-left: none; - } -} - -/* Tag Settings (Right pane) -/* ---------------------------------------------------------- */ - -.tag-settings { - position: absolute; - top: 0; - right: 0; - bottom: 0; - overflow-x: hidden; - overflow-y: auto; - -webkit-overflow-scrolling: touch; - min-width: 350px; - width: 34%; - border: none; - background: #fff; - transform: none; -} - -.tag-settings .no-posts { - padding: 1em; -} - -.tag-settings .no-posts h3 { - text-align: center; -} - -.tag-settings .settings-menu-pane { - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); -} - -@media (max-width: 600px) { - .tag-settings { - min-width: 0; - width: 100%; - transition: transform 0.4s cubic-bezier(0.1, 0.7, 0.1, 1); - transform: translate3d(100%, 0px, 0px); - - transform-style: preserve-3d; - } - - .tag-settings-in { - transform: translate3d(0px, 0px, 0px); - } -} diff --git a/core/client/app/styles/layouts/user.css b/core/client/app/styles/layouts/user.css deleted file mode 100644 index 20c90c6d15..0000000000 --- a/core/client/app/styles/layouts/user.css +++ /dev/null @@ -1,217 +0,0 @@ -/* User profile /ghost/settings/users/<user>/ -/* ---------------------------------------------------------- */ - - -/* User actions menu -/* ---------------------------------------------------------- */ - -.user-actions-cog { - margin-right: 10px; - padding: 9px 11px; -} - -.user-actions-menu { - top: calc(100% + 17px); - right: 0; - left: auto; -} - -.user-actions-menu.fade-out { - animation-duration: 0.01s; -} - - -/* Layout -/* ---------------------------------------------------------- */ - -.content.settings-user { - padding: 0; -} - -@media (min-width: 901px) { - .content.settings-user { - padding: 0 40px; - } -} - -.user-cover { - position: relative; - overflow: hidden; - margin: 0; - width: auto; - height: 300px; - background: #fafafa no-repeat center center; - background-size: cover; -} - -@media (max-width: 900px) { - .user-cover { - margin: 0; - } -} - -.user-cover:after { - /* Gradient overlay */ - content: ""; - position: absolute; - right: 0; - bottom: 0; - left: 0; - height: 110px; - background: linear-gradient(transparent, rgba(0, 0, 0, 0.18)); -} - -.user-cover-edit { - position: absolute; - top: 30px; - left: 35px; - z-index: 2; - min-height: 37px; - height: 37px; - border-width: 0; - background: rgba(0, 0, 0, 0.3); - border-radius: var(--border-radius); - color: rgba(255, 255, 255, 0.8); - transition: color 0.3s ease, background 0.3s ease; -} - -.user-cover-edit:hover { - background: rgba(0, 0, 0, 0.5); - color: #fff; -} - - -/* Edit user -/* ---------------------------------------------------------- */ - -@media (min-width: 651px) { - .first-form-group { - margin-right: 20px; - padding-left: 40px; - } - .first-form-group input { - max-width: 100%; - } -} - -.user-details-top { - position: relative; -} - -@media (max-width: 650px) { - .user-details-top { - margin-top: 40px; - margin-bottom: 0; - } -} - -@media (min-width: 651px) { - .user-details-top { - margin-top: -91px; - margin-bottom: 0; - padding: 0; - } - .user-details-top p { - color: #fff; - } - .user-details-top label[for="user-name"] { - color: transparent; - } - .user-details-top .user-name { - border-color: #fff; - } -} - -.user-profile { - position: relative; - z-index: 1; -} - -@media (min-width: 651px) { - .user-profile { - padding-right: 20px; - padding-left: 143px; - } -} - -@media (max-width: 650px) { - .user-profile fieldset { - padding: 0 40px; - } -} - -@media (max-width: 550px) { - .user-profile fieldset { - padding: 0 15px; - } -} - -.user-profile textarea { - min-width: 240px; -} - - -/* Profile picture -/* ---------------------------------------------------------- */ - -.user-image { - position: absolute; - z-index: 2; - display: block; - float: left; - overflow: hidden; - margin-right: 20px; - margin-left: -6px; - padding: 3px; - width: 126px; - height: 126px; - background: #fff; - border-radius: 100%; - text-align: center; -} - -@media (min-width: 651px) { - .user-image { - top: -19px; - left: -98px; - } -} - -@media (max-width: 650px) { - .user-image { - top: -135px; - left: 50%; - margin-right: 0; - margin-left: -63px; - } -} - -.user-image .img { - display: block; - width: 120px; - height: 120px; - background-position: center center; - background-size: cover; - border-radius: 100%; -} - -.user-image:hover .edit-user-image { - opacity: 1; -} - -.edit-user-image { - position: absolute; - top: 3px; - right: 3px; - bottom: 3px; - left: 3px; - width: calc(100% - 6px); - background: rgba(0, 0, 0, 0.5); - border-radius: 100%; - color: #fff; - text-decoration: none; - text-transform: uppercase; - line-height: 120px; - opacity: 0; - transition: opacity 0.3s ease; -} diff --git a/core/client/app/styles/layouts/users.css b/core/client/app/styles/layouts/users.css deleted file mode 100644 index f5275e2217..0000000000 --- a/core/client/app/styles/layouts/users.css +++ /dev/null @@ -1,202 +0,0 @@ -/* Users /ghost/settings/users/ -/* ---------------------------------------------------------- */ - -.users-list-wrapper { - overflow: auto; - height: 100%; -} - - -/* User list -/* ---------------------------------------------------------- */ - -.invited-users { - margin-bottom: 34px; -} - -.user-list-title { - margin-bottom: 14px; - color: #a1a1a1; - font-size: 13px; - font-weight: normal; -} - -.user-list-item { - display: flex; - justify-content: start; - align-items: center; - padding: 0 15px; - height: 68px; - border-top: 1px solid #dfe1e3; -} - -/* Only apply these styles to anchor tags (pending invited are divs) */ -a.user-list-item { - text-decoration: none; -} - -@media (min-width: 601px) { - a.user-list-item:hover { - background: color(#dfe1e3 lightness(+10%)); - } - a.user-list-item:last-of-type:hover { - box-shadow: inset 0 -1px 0 #dfe1e3; - } -} - -.user-list-item-icon { - position: relative; - display: block; - overflow: hidden; - width: 35px; - height: 35px; - background: #dfe1e3; - border-radius: 100%; - color: transparent; - font-size: 0; -} - -.user-list-item-icon:before { - position: absolute; - top: 50%; - right: 0; - left: 0; - margin-top: -7px; - color: var(--midgrey); - text-align: center; - font-size: 14px; -} - -.user-list-item-figure { - position: relative; - display: block; - width: 35px; - height: 35px; - background-position: center center; - background-size: cover; - border-radius: 35px; -} - -.user-list-item-figure img { - position: absolute; - top: 0; - left: 0; - width: 100%; - height: 100%; - opacity: 0; -} - -.user-list-item-body { - flex: 1 1 auto; - align-items: stretch; - padding-left: 15px; - line-height: 1; -} - -.user-list-item-body .name { - display: inline-block; - color: var(--darkgrey); - font-size: 15px; - font-weight: 400; -} - -.user-list-item-body .description { - display: inline-block; - margin-top: 3px; - color: #a1a1a1; - white-space: nowrap; - font-size: 12px; -} - -.user-list-item-body .description-error { - display: inline-block; - margin-top: 3px; - color: var(--red); - white-space: nowrap; - font-size: 12px; -} - -.user-list-item-aside .user-list-action:not(:first-of-type) { - margin-left: 20px; -} - -.user-list-item-aside .role-label { - float: left; - margin-top: -1px; -} - -.user-list-item-aside .role-label + .role-label { - margin-left: 5px; -} - -.user-list-action { - text-decoration: underline; - text-transform: uppercase; - font-size: 11px; -} - - -/* Role Labels -/* ---------------------------------------------------------- */ - -.role-label { - display: inline-block; - padding: 6px 8px; - background: #eee; - color: rgba(0, 0, 0, 0.5); - text-transform: uppercase; - letter-spacing: 0.1em; - font-size: 9px; - line-height: 1; - font-weight: 400; -} - -.role-label.owner { - background: var(--darkgrey); - color: rgba(255, 255, 255, 0.8); -} - -.role-label.administrator { - background: var(--red); - color: rgba(255, 255, 255, 0.8); -} - -.role-label.editor { - background: var(--blue); - color: rgba(255, 255, 255, 0.8); -} - - -/* User invitation modal -/* ---------------------------------------------------------- */ - -.invite-new-user .form-group { - margin-bottom: 0; - padding: 0; -} - -.invite-new-user .form-group label { - position: static; - display: block; - text-align: left; -} - -.invite-new-user .form-group:nth-of-type(1) { - float: left; - width: 60%; -} - -.invite-new-user .form-group:nth-of-type(2) { - float: left; - margin-left: 5%; - width: 35%; -} - -.invite-new-user .form-group input { - width: 100%; -} - -.invite-new-user .btn-green { - margin: 0; - width: 100%; -} diff --git a/core/client/app/styles/patterns/_shame.css b/core/client/app/styles/patterns/_shame.css deleted file mode 100644 index a4a5638182..0000000000 --- a/core/client/app/styles/patterns/_shame.css +++ /dev/null @@ -1,7 +0,0 @@ -/* Shame -/* ---------------------------------------------------------- */ -/* TODO: Kill with fire */ - - -/* Animations -/* ---------------------------------------------------------- */ diff --git a/core/client/app/styles/patterns/buttons.css b/core/client/app/styles/patterns/buttons.css deleted file mode 100644 index dadefc5274..0000000000 --- a/core/client/app/styles/patterns/buttons.css +++ /dev/null @@ -1,237 +0,0 @@ -/* Buttons -/* ---------------------------------------------------------- */ - -/* Base button style */ -.btn { - display: inline-block; - margin-bottom: 0; - padding: 9px 15px; - border: #dfe1e3 1px solid; - background: #fff; - background-image: none; /* Reset unusual Firefox-on-Android default style; see https://github.com/necolas/normalize.css/issues/214 */ - border-radius: var(--border-radius); - color: #808284; - text-align: center; - text-transform: uppercase; - text-shadow: none; - white-space: nowrap; - letter-spacing: 1px; - font-size: 1.1rem; - line-height: 1.428571429; - font-weight: 300; - cursor: pointer; - transition: color 0.2s ease, - background 0.2s ease, - border-color 0.2s ease; -} - -/* When hovered or clicked */ -.btn:hover, -.btn:focus { - border-color: var(--blue); - color: color(var(--blue) lightness(-10%)); - text-decoration: none; -} - -/* When focused with keyboard */ -.btn:focus, -.btn:active:focus, -.btn.active:focus { - outline: thin dotted; - outline: 0 auto -webkit-focus-ring-color; - outline-offset: -2px; -} - -/* When clicked */ -.btn:active, -.btn.active { - outline: 0; - background-image: none; - box-shadow: inset 0 1px 2px rgba(0, 0, 0, 0.2); -} - -/* When disabled */ -.btn.disabled, -.btn[disabled], -fieldset[disabled] .btn { - box-shadow: none; - opacity: 0.65; - cursor: not-allowed; - pointer-events: none; -} - -.btn i { - display: inline-block; - vertical-align: middle; -} - -.btn-hover-green:hover, -.btn-hover-green:active, -.btn-hover-green:focus { - border-color: var(--green); - color: color(var(--green) lightness(-10%)); -} - - -/* Blue button -/* ---------------------------------------------------------- */ - -.btn-blue { - border-color: color(var(--blue) lightness(-10%)); - background: var(--blue); - color: #fff; -} - -.btn-blue:hover, -.btn-blue:active, -.btn-blue:focus { - border-color: color(var(--blue) lightness(-20%)); - background: color(var(--blue) lightness(-10%)); - color: #fff; -} - - -/* Green button -/* ---------------------------------------------------------- */ - -.btn-green { - border-color: color(var(--green) lightness(-10%)); - background: var(--green); - color: #fff; -} - -.btn-green:hover, -.btn-green:active, -.btn-green:focus { - border-color: color(var(--green) lightness(-20%)); - background: color(var(--green) lightness(-10%)); - color: #fff; -} - - -/* Red button -/* ---------------------------------------------------------- */ - -.btn-red { - border-color: color(var(--red) lightness(-10%)); - background: var(--red); - color: #fff; -} - -.btn-red:hover, -.btn-red:active, -.btn-red:focus { - border-color: color(var(--red) lightness(-20%)); - background: color(var(--red) lightness(-10%)); - color: #fff; -} - - -/* Link button -/* ---------------------------------------------------------- */ - -/* For styling a button like a link */ -.btn-link { - border-color: transparent; - background: transparent; - color: var(--blue); -} - -.btn-link:hover, -.btn-link:active, -.btn-link:focus { - border-color: transparent; - background: transparent; - color: var(--blue); - text-decoration: underline; -} - -.btn-link.disabled, -.btn-link[disabled] { - box-shadow: none; - color: #b2b2b2; - opacity: 0.65; - cursor: not-allowed; - pointer-events: none; -} - - -/* Minor button -/* ---------------------------------------------------------- */ - -/* For buttons with a small/insignificant action for -// example a "cancel" button. Style is de-emphasised. */ -.btn-minor { - padding: 8px 15px; - text-transform: none; - letter-spacing: 0; - font-size: 1.2rem; -} - -.btn-minor:hover, -.btn-minor:active, -.btn-minor:focus { - border-color: #c1c1c1; - background: #fff; - box-shadow: none; - color: #808284; -} - - -/* Button size variations -/* ---------------------------------------------------------- */ - -.btn-lg { - padding: 12px 18px; - border-radius: 4px; - font-size: 1.4rem; - line-height: 1.33; -} - -.btn-sm { - padding: 7px 10px; - border-radius: 2px; - font-size: 1rem; - line-height: 1.5; -} - -.btn-block { - display: block; - width: 100%; -} - -/* Vertically space out multiple block buttons */ -.btn-block + .btn-block { - margin-top: 5px; -} - -/* Specificity overrides */ -input[type="submit"].btn-block, -input[type="reset"].btn-block, -input[type="button"].btn-block { - width: 100%; -} - -/* Spin Buttons! -/* ---------------------------------------------------------- */ -.spinner { - position: relative; - display: inline-block; - box-sizing: border-box; - margin: -2px 0; - width: 14px; - height: 14px; - border: rgba(0,0,0,0.2) solid 4px; - border-radius: 100px; - animation: spin 1s linear infinite; -} - -.spinner:before { - content: ""; - display: block; - margin-top: 6px; - width: 4px; - height: 4px; - background: rgba(0,0,0,0.6); - border-radius: 100px; -} diff --git a/core/client/app/styles/patterns/forms.css b/core/client/app/styles/patterns/forms.css deleted file mode 100644 index ea414a136d..0000000000 --- a/core/client/app/styles/patterns/forms.css +++ /dev/null @@ -1,343 +0,0 @@ -/* Forms -/* ---------------------------------------------------------- */ - -form label { - display: block; - color: var(--darkgrey); - font-size: 1.3rem; - font-weight: bold; - user-select: text; -} - -form .word-count { - float: right; - font-weight: bold; -} - -fieldset { - margin: 0 0 3em 0; - padding: 0; - border: none; - user-select: text; -} - -legend { - display: block; - margin: 2em 0; - width: 100%; - border-bottom: #dfe1e3 1px solid; - color: #b1b1b1; - font-size: 1.2em; - line-height: 2.0em; - user-select: text; -} - -input { - user-select: text; -} - -.error .response { - color: var(--red); -} - - -/* Form Groups -/* ---------------------------------------------------------- */ - -.form-group { - position: relative; - margin-bottom: 1.6em; - max-width: 500px; - width: 100%; - user-select: text; -} - -.form-group p { - margin: 4px 0 0 0; - color: #b1b1b1; - font-size: 1.3rem; -} - -.form-group h3 { - margin-bottom: 1.6em; - font-size: 1.5rem; -} - -.form-group label { - margin-bottom: 4px; -} - -@media (max-width: 550px) { - .form-group { - max-width: 100%; - } -} - - -/* Input Icons -/* ---------------------------------------------------------- */ - -.input-icon[class*="icon-"] { - position: relative; - display: block; -} - -.input-icon[class*="icon-"] input[type="email"], -.input-icon[class*="icon-"] input[type="number"], -.input-icon[class*="icon-"] input[type="password"], -.input-icon[class*="icon-"] input[type="search"], -.input-icon[class*="icon-"] input[type="tel"], -.input-icon[class*="icon-"] input[type="text"], -.input-icon[class*="icon-"] input[type="url"], -.input-icon[class*="icon-"] input[type="date"] { - padding-left: 3.2rem; -} - -.input-icon[class*="icon-"] .gh-select select { - padding-left: 3.2rem; -} - -.input-icon[class*="icon-"]:before { - position: absolute; - top: 50%; - left: 1.1rem; - z-index: 100; - font-size: 1.3rem; - transform: translateY(-52%); -} - - -/* Inputs -/* ---------------------------------------------------------- */ - -.gh-input, -.gh-select, -select { - display: block; - padding: 8px 10px; - width: 100%; - border: 1px solid #dfe1e3; - border-radius: var(--border-radius); - color: #666; - font-size: 1.4rem; - font-weight: normal; - user-select: text; - transition: border-color 0.15s linear; - - -webkit-appearance: none; -} - -.gh-select, -select { - cursor: pointer; -} - -.gh-input.error, -.error .gh-input, -.gh-select.error, -select.error { - border-color: var(--red); -} - -.gh-input:focus, -.gh-input.focus, -.gh-select:focus, -select:focus { - outline: 0; - border-color: #b1b1b1; -} - -textarea { - min-width: 250px; - min-height: 10rem; - max-width: 500px; - width: 100%; - height: auto; - line-height: 1.5; - user-select: text; - resize: vertical; -} - - -/* Radio / Checkboxes -/* ---------------------------------------------------------- */ - -.for-radio label, -.for-checkbox label { - display: block; - padding-bottom: 4px; - cursor: pointer; -} - -.for-radio label p, -.for-checkbox label p { - overflow: auto; - color: #000; - font-weight: normal; -} - -.for-radio label:hover p, -.for-checkbox label:hover p { - color: var(--midgrey); -} - -.for-radio input, -.for-checkbox input { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: -9999px; -} - -.for-radio .input-toggle-component, -.for-checkbox .input-toggle-component { - position: relative; - top: 1px; - display: inline-block; - float: left; - margin-right: 7px; - width: 18px; - height: 18px; - border: 1px solid #dfe1e3; - background: #f7f7f7; -} - -.for-checkbox label:hover input:not(:checked) + .input-toggle-component, -.for-radio label:hover input:not(:checked) + .input-toggle-component { - border-color: var(--lightgrey); -} - -.for-checkbox .input-toggle-component { - border-radius: var(--border-radius); -} - -.for-checkbox .input-toggle-component { - transition: background 0.15s ease-in-out, border-color 0.15s ease-in-out; -} - -.for-checkbox .input-toggle-component:before { - content: ""; - position: absolute; - top: 4px; - left: 3px; - width: 10px; - height: 6px; - border: 2px solid #fff; - border-top: none; - border-right: none; - opacity: 0; - transition: opacity 0.15s ease-in-out; - transform: rotate(-45deg); -} - -.for-checkbox input:checked + .input-toggle-component { - border-color: color(var(--green) lightness(-10%)); - background: var(--green); -} - -.for-checkbox input:checked + .input-toggle-component:before { - opacity: 1; -} - -.for-radio .input-toggle-component { - border-radius: 100px; -} - -.for-radio .input-toggle-component { - transition: background 0.15s ease-in-out, border-color 0.15s ease-in-out; -} - -.for-radio .input-toggle-component:before { - content: ""; - position: absolute; - top: 4px; - left: 4px; - width: 8px; - height: 8px; - background: #fff; - border-radius: 100%; - opacity: 0; - transition: opacity 0.15s ease-in-out; -} - -.for-radio input:checked + .input-toggle-component { - border-color: color(var(--green) lightness(-10%)); - background: var(--green); -} - -.for-radio input:checked + .input-toggle-component:before { - opacity: 1; -} - - -/* Select -/* ---------------------------------------------------------- */ - -.gh-select { - position: relative; - display: block; - overflow: hidden; - padding: 0; - max-width: 100%; - width: 100%; - border-width: 0; -} - -.gh-select:after { - content: "\e00f"; - position: absolute; - top: 50%; - right: 1.2rem; - margin-top: -0.5em; - text-transform: none !important; - font-family: "ghosticons" !important; - font-size: 1rem; - line-height: 1; - font-weight: normal !important; - font-style: normal !important; - font-variant: normal !important; - pointer-events: none; - - speak: none; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -.gh-select select { - padding: 8px 10px; - outline: none; - background: #fff; - text-indent: 0.01px; - text-overflow: ""; - line-height: normal; - - appearance: none; - -webkit-appearance: none; - -moz-appearance: window; -} - -.gh-select select::-ms-expand { - display: none; -} - -.gh-select select:-moz-focusring { - color: transparent; - text-shadow: 0 0 0 #000; -} - - -/* FFF: Fucking Firefox Fixes -/* ---------------------------------------------------------- */ - -@-moz-document url-prefix() { - .gh-select { - border-width: 1px; - } - .gh-select select { - padding: 7px 10px 7px 8px; - } - .gh-select:focus { - border-color: #b1b1b1; - } -} diff --git a/core/client/app/styles/patterns/global.css b/core/client/app/styles/patterns/global.css deleted file mode 100644 index 5121d910f3..0000000000 --- a/core/client/app/styles/patterns/global.css +++ /dev/null @@ -1,441 +0,0 @@ -/* Global styles -/* ---------------------------------------------------------- */ - - -/* Variables -/* ---------------------------------------------------------- */ - -:root { - /* Colours */ - --darkgrey: #242628; - --midgrey: #7d878a; - --lightgrey: #e2edf2; - --blue: #5ba4e5; - --red: #e25440; - --orange: #f2a925; - --green: #9fbb58; - /* Style values */ - --border-radius: 4px; - --font-family: "Open Sans", sans-serif; - --font-family-mono: Consolas, "Liberation Mono", Menlo, Courier, monospace; -} - -/* Colour classes -/* ---------------------------------------------------------- */ -.darkgrey { - color: var(--darkgrey); -} - -.midgrey { - color: var(--midgrey); -} - -.lightgrey { - color: var(--lightgrey); -} - -.blue { - color: var(--blue); -} - -.red { - color: var(--red); -} - -.orange { - color: var(--orange); -} - -.green { - color: var(--green); -} - -/* Layout -/* ---------------------------------------------------------- */ - -*, -*:before, -*:after { - box-sizing: border-box; - user-select: none; -} - -html { - overflow: hidden; - width: 100%; - /* Prevent elastic scrolling on the whole page */ - height: 100%; - font: 62.5%/1.65 "Open Sans", sans-serif; - - -webkit-tap-highlight-color: rgba(0, 0, 0, 0); -} - -body { - overflow: auto; - overflow-x: hidden; - width: 100%; - /* Prevent elastic scrolling on the whole page */ - height: 100%; - color: color(var(--darkgrey) lightness(+10%)); - font-size: 1.4rem; -} - -::selection { - background: color(var(--blue) lightness(+20%)); -} - - -/* Text -/* ---------------------------------------------------------- */ - -h1, -h2, -h3, -h4, -h5, -h6 { - margin: 0 0 0.3em 0; - color: var(--darkgrey); - line-height: 1.15em; - text-rendering: optimizeLegibility; -} - -h1 { - text-indent: -1px; - font-size: 5rem; -} - -h2 { - text-indent: -1px; - font-size: 4.2rem; -} - -h3 { - font-size: 3.8rem; -} - -h4 { - font-size: 3.1rem; -} - -h5 { - font-size: 2.8rem; -} - -h6 { - font-size: 2.2rem; -} - -p, -ul, -ol, -dl { - margin: 0 0 1.7em 0; -} - -ol, -ul { - padding-left: 2.5em; -} - -ol ol, -ul ul, -ul ol, -ol ul { - margin: 0 0 0.4em 0; - padding-left: 2em; - font-size: 0.9em; -} - -mark { - background-color: #fdffb6; -} - -a { - color: var(--blue); - text-decoration: none; - transition: background 0.3s, color 0.3s; -} - -a:hover { - text-decoration: none; - transition: background 0.1s, color 0.1s; -} - -a.highlight { - color: var(--orange); - font-weight: bold; -} - -hr { - display: block; - margin: 3.2em 0; - padding: 0; - height: 1px; - border: 0; - border-top: 1px solid #dfe1e3; -} - -dl { - margin: 1.6em 0; -} - -dl dt { - float: left; - clear: left; - overflow: hidden; - margin-bottom: 1em; - width: 180px; - text-align: right; - text-overflow: ellipsis; - white-space: nowrap; - font-weight: bold; -} - -dl dd { - margin-bottom: 1em; - margin-left: 200px; -} - -blockquote { - margin: 1.6em 0; - padding: 0 1.6em 0 1.6em; - border-left: #dfe1e3 0.6em solid;; -} - -blockquote p { - margin: 0.8em 0; - font-size: 1.2em; - font-weight: 300; -} - -blockquote small { - display: inline-block; - margin: 0.8em 0 0.8em 1.5em; - color: var(--midgrey); - font-size: 0.9em; -} -/* Quotation marks */ -blockquote small:before { - content: "\2014 \00A0"; -} - -blockquote cite { - font-weight: bold; -} -blockquote cite a { - font-weight: normal; -} - -.markdown, -pre, -code, -tt { - font-family: var(--font-family-mono); -} - -code, -tt { - padding: 0.2rem 0.3rem 0.1rem; - border: color(#f5f7f8 lightness(-10%)) 1px solid; - background: #f5f7f8; - border-radius: 2px; - color: #c25; - vertical-align: middle; - white-space: pre-wrap; - font-size: 0.8em; - line-height: 1em; -} - -pre { - overflow: auto; - margin: 1.6em 0; - padding: 10px; - width: 100%; - border: color(#f5f7f8 lightness(-10%)) 1px solid; - background: #f5f7f8; - border-radius: 3px; - white-space: pre; - font-family: var(--font-family-mono); - font-size: 0.9em;; -} - -pre code, -pre tt { - padding: 0; - border: none; - background: transparent; - color: inherit; - white-space: pre-wrap; - font-size: inherit; -} - -kbd { - display: inline-block; - margin-bottom: 0.4em; - padding: 1px 8px; - border: #ccc 1px solid; - background: #f4f4f4; - border-radius: 4px; - box-shadow: 0 1px 0 rgba(0, 0, 0, 0.2), - 0 1px 0 0 #fff inset; - color: var(--darkgrey); - text-shadow: #fff 0 1px 0; - font-size: 0.9em; - font-weight: bold; -} - -button { - padding: 0; - outline: none; - border: none; - background: transparent; - box-shadow: none; -} - -i { - display: block; -} - -img { - max-width: 100%; -} - - -/* Utilities -/* ---------------------------------------------------------- */ - -.clearfix, -.clearfix:after { - content: ""; - display: table; - clear: both; -} - -.wrapper { - position: relative; -} - -.show { - display: block !important; -} - -.hidden { - visibility: hidden !important; - display: none !important; -} - -.invisible { - visibility: hidden; -} - -.sr-only { - position: absolute; - overflow: hidden; - clip: rect(0, 0, 0, 0); - margin: -1px; - padding: 0; - width: 1px; - height: 1px; - border: 0; -} - -.sr-only-focusable:focus { - z-index: 900; - overflow: visible; - clip: auto; - margin: 0; - padding: 0 10px; - width: auto; - height: auto; - background-color: #f5f5f5; - color: #333; - text-decoration: none; - line-height: 49px; - font-weight: bold; -} - -.right { - float: right; -} - -.left { - float: left; -} - -.vertical { - display: table-cell; - vertical-align: middle; -} - - -/* Animations -/* ---------------------------------------------------------- */ - -@keyframes fade-in { - from { - opacity: 0; - } - to { - opacity: 1; - } -} - -@keyframes fade-in-snap { - to { - opacity: 1; - } -} - -@keyframes fade-in-scale { - from { - opacity: 0; - transform: scale(0.95); - } - to { - opacity: 1; - transform: scale(1); - } -} - -@keyframes fade-out { - from { - opacity: 1; - } - to { - opacity: 0; - } -} - -@keyframes keyboard-focus-style-fade-out { - from { - box-shadow: inset 0 0 30px 1px color(var(--midgrey) lightness(+20%)); - } - to { - box-shadow: none; - } -} - -@keyframes spin { - from { - transform: rotate(0deg); - } - to { - transform: rotate(360deg); - } -} - -.fade-in { - animation: fade-in 0.2s; - animation-fill-mode: forwards; -} - -.fade-in-scale { - animation: fade-in-scale 0.2s; - animation-fill-mode: forwards; -} - -.fade-out { - animation: fade-out 0.5s; - animation-fill-mode: forwards; -} diff --git a/core/client/app/styles/patterns/icons.css b/core/client/app/styles/patterns/icons.css deleted file mode 100755 index 4f398af7a0..0000000000 --- a/core/client/app/styles/patterns/icons.css +++ /dev/null @@ -1,266 +0,0 @@ -/* Icons -/* ---------------------------------------------------------- */ -/* CUSTOM GENERATED FILE */ -/* DO NOT UPDATE WITHOUT TALKING TO @JOHNONOLAN FIRST */ - -@font-face { - font-family: "ghosticons"; - font-weight: normal; - font-style: normal; - src: url("fonts/ghosticons.eot"); - src: url("fonts/ghosticons.eot?#iefix") format("embedded-opentype"), - url("fonts/ghosticons.woff") format("woff"), - url("fonts/ghosticons.ttf") format("truetype"), - url("fonts/ghosticons.svg#ghosticons") format("svg"); -} - -[data-icon]:before { - content: attr(data-icon); - text-transform: none !important; - font-family: "ghosticons" !important; - line-height: 1; - font-weight: normal !important; - font-style: normal !important; - font-variant: normal !important; - - speak: none; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -[class^="icon-"]:before, -[class*=" icon-"]:before { - text-transform: none !important; - font-family: "ghosticons" !important; - line-height: 1; - font-weight: normal !important; - font-style: normal !important; - font-variant: normal !important; - - speak: none; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -.icon-user:before { - content: "\e000"; -} -.icon-search:before { - content: "\e001"; -} -.icon-design:before { - content: "\e003"; -} -.icon-content:before { - content: "\e004"; -} -.icon-x:before { - content: "\e005"; -} -.icon-add:before { - content: "\e006"; -} -.icon-labs:before { - content: "\e007"; -} -.icon-settings:before { - content: "\e008"; -} -.icon-front-end:before { - content: "\e00a"; -} -.icon-dash:before { - content: "\e00b"; -} -.icon-tag:before { - content: "\e009"; -} -.icon-compass:before { - content: "\e002"; -} -.icon-code:before { - content: "\e00c"; -} -.icon-team:before { - content: "\e00d"; -} -.icon-idea:before { - content: "\e00e"; -} -.icon-arrow:before, -.icon-ascending:before, -.icon-descending:before { - content: "\e00f"; -} -.icon-ascending:before { - display: inline-block; - transform: rotate(180deg); -} -.icon-pen:before { - content: "\e010"; -} -.icon-clip:before { - content: "\e011"; -} -.icon-trash:before { - content: "\e012"; -} -.icon-edit:before { - content: "\e013"; -} -.icon-new:before { - content: "\e014"; -} -.icon-lock:before { - content: "\e015"; -} -.icon-link:before { - content: "\e016"; -} -.icon-chat:before { - content: "\e017"; -} -.icon-smiley:before { - content: "\e018"; -} -.icon-star:before { - content: "\e019"; -} -.icon-rss:before { - content: "\e01b"; -} -.icon-hotspot:before { - content: "\e01d"; -} -.icon-mail:before { - content: "\e01e"; -} -.icon-sound-on:before { - content: "\e01f"; -} -.icon-sound-off:before { - content: "\e020"; -} -.icon-download:before { - content: "\e021"; -} -.icon-upload:before { - content: "\e022"; -} -.icon-bell:before { - content: "\e023"; -} -.icon-shop:before { - content: "\e01c"; -} -.icon-box:before { - content: "\e024"; -} -.icon-connections:before { - content: "\e025"; -} -.icon-arrow-right:before { - content: "\e01a"; -} -.icon-arrow-left:before { - content: "\e026"; -} -.icon-arrow-up:before { - content: "\e027"; -} -.icon-ghost:before { - content: "\e028"; -} -.icon-dice:before { - content: "\e029"; -} -.icon-ambulance:before { - content: "\e02a"; -} -.icon-calendar:before { - content: "\e02b"; -} -.icon-folder:before { - content: "\e02c"; -} -.icon-pulse:before { - content: "\e02d"; -} -.icon-photos:before { - content: "\e02e"; -} -.icon-legal:before { - content: "\e02f"; -} -.icon-letter:before { - content: "\e030"; -} -.icon-grid:before { - content: "\e031"; -} -.icon-list:before { - content: "\e032"; -} -.icon-blog:before { - content: "\e033"; -} -.icon-question:before { - content: "\e034"; -} -.icon-error:before { - content: "\e035"; -} -.icon-markdown:before { - content: "\e037"; -} -.icon-external:before { - content: "\e038"; -} -.icon-arrow2:before { - content: "\e039"; -} -.icon-arrow2-up:before { - content: "\e03a"; -} -.icon-arrow2-left:before { - content: "\e03b"; -} -.icon-arrow2-right:before { - content: "\e03c"; -} -.icon-x2:before { - content: "\e03d"; -} -.icon-gh:before { - content: "\e03e"; -} -.icon-signout:before { - content: "\e036"; -} -.icon-minimise:before { - content: "\e03f"; -} -.icon-maximise:before { - content: "\e040"; -} -.icon-book:before { - content: "\e041"; -} -.icon-twitter:before { - content: "\e042"; -} -.icon-check:before { - content: "\e043"; -} -.icon-grab:before { - content: "\e044"; -} -.icon-add2:before { - content: "\e045"; -} -.icon-pause:before { - content: "\e046"; -} -.icon-play:before { - content: "\e047"; -} diff --git a/core/client/app/styles/patterns/labels.css b/core/client/app/styles/patterns/labels.css deleted file mode 100644 index db95a7d78c..0000000000 --- a/core/client/app/styles/patterns/labels.css +++ /dev/null @@ -1,119 +0,0 @@ -/* Labels -/* ---------------------------------------------------------- */ - -.label { - display: inline; - padding: 0.2em 0.6em 0.3em; - border-radius: 0.25em; - color: #fff; - vertical-align: baseline; - text-align: center; - white-space: nowrap; - font-size: 75%; - line-height: 1; - font-weight: 300; -} - -.label:empty { - display: none; -} - -.btn .label { - position: relative; - top: -1px; -} - -h1 .label, -h2 .label, -h3 .label, -h4 .label, -h5 .label, -h6 .label { - position: relative; - top: -0.18em; - display: inline-block; - padding: 0.2em 0.5em 0.25em; - font-size: 70%; - line-height: 70%; -} - -a.label:hover, -a.label:focus { - color: #fff; - text-decoration: none; - cursor: pointer; -} - - -/* Label Colours -/* ---------------------------------------------------------- */ - -.label-default { - background-color: #a1adb3; - color: #fff; -} - -.label-default[href] { - color: #fff; -} - -.label-default[href]:hover, -.label-default[href]:focus { - background-color: color(#a1adb3 lightness(-10%)); -} - -.label-alt { - background-color: #666; - color: #fff; -} - -.label-alt[href] { - color: #fff; -} - -.label-alt[href]:hover, -.label-alt[href]:focus { - background-color: color(#666 lightness(-10%)); -} - -.label-blue { - background-color: var(--blue); - color: #fff; -} - -.label-blue[href] { - color: #fff; -} - -.label-blue[href]:hover, -.label-blue[href]:focus { - background-color: color(var(--blue) lightness(-10%)); -} - -.label-green { - background-color: var(--green); - color: #fff; -} - -.label-green[href] { - color: #fff; -} - -.label-green[href]:hover, -.label-green[href]:focus { - background-color: color(var(--green) lightness(-10%)); -} - -.label-red { - background-color: var(--red); - color: #fff; -} - -.label-red[href] { - color: #fff; -} - -.label-red[href]:hover, -.label-red[href]:focus { - background-color: color(var(--red) lightness(-10%)); -} diff --git a/core/client/app/styles/patterns/navlist.css b/core/client/app/styles/patterns/navlist.css deleted file mode 100644 index 69a048751a..0000000000 --- a/core/client/app/styles/patterns/navlist.css +++ /dev/null @@ -1,71 +0,0 @@ -/* Nav Lists -/* ---------------------------------------------------------- */ - -.nav-list { - padding: 0; - max-width: 500px; - border: 1px solid #e0dfd7; - background: #fff; - border-radius: var(--border-radius); -} - -.nav-list.nav-list-block { - max-width: none; -} - -.nav-list-item { - position: relative; - display: block; - padding: 8px 40px 8px 12px; - color: var(--darkgrey); -} - -.nav-list-item:hover { - background: color(#dfe1e3 lightness(+10%)); - cursor: pointer; -} - -.nav-list-item:first-of-type { - border-top-left-radius: var(--border-radius); - border-top-right-radius: var(--border-radius); -} - -.nav-list-item:last-of-type { - border-bottom-right-radius: var(--border-radius); - border-bottom-left-radius: var(--border-radius); -} - -.nav-list-item:not(:last-of-type) { - border-bottom: 1px solid #e0dfd7; -} - -.nav-list-item button { - text-align: left; -} - -.nav-list-item a { - color: var(--darkgrey); -} - -.nav-list-item b { - display: block; - font-size: 1.6rem; - line-height: 1.375; - font-weight: normal; -} - -.nav-list-item span { - display: block; - color: var(--midgrey); - font-size: 1.1rem; - line-height: 1.375; -} - -.nav-list-item i { - position: absolute; - top: 50%; - right: 10px; - margin-top: -0.9rem; - color: var(--midgrey); - font-size: 1.4rem; -} diff --git a/core/client/app/styles/patterns/tables.css b/core/client/app/styles/patterns/tables.css deleted file mode 100644 index f2680b6b4e..0000000000 --- a/core/client/app/styles/patterns/tables.css +++ /dev/null @@ -1,79 +0,0 @@ -/* Tables -/* ---------------------------------------------------------- */ - -/* Base */ -table, -.table { - margin: 1.6em 0; - max-width: 100%; - width: 100%; - background-color: transparent; -} - -table th, -table td, -.table th, -.table td { - padding: 8px; - vertical-align: middle; - text-align: left; - line-height: 20px; - user-select: text; -} - - -/* Default Table -/* ---------------------------------------------------------- */ - -.table th, -.table td { - border-top: 1px solid #dfe1e3; -} - -.table th { - color: var(--midgrey); -} - -.table caption + thead tr:first-child th, -.table caption + thead tr:first-child td, -.table colgroup + thead tr:first-child th, -.table colgroup + thead tr:first-child td, -.table thead:first-child tr:first-child th, -.table thead:first-child tr:first-child td { - border-top: 0; -} - -.table tbody + tbody { - border-top: 2px solid #dfe1e3; -} - -.table table table { - background-color: #fff; -} - -.table tbody > tr:nth-child(odd) > td, -.table tbody > tr:nth-child(odd) > th { - background-color: color(#dfe1e3 lightness(+10%)); -} - -.table.plain tbody > tr:nth-child(odd) > td, -.table.plain tbody > tr:nth-child(odd) > th { - background: transparent; -} - -/* Ember Light Table -/* ---------------------------------------------------------- */ - -.ember-light-table th { - white-space: nowrap; -} - -.ember-light-table .lt-column .lt-sort-icon { - float: none; - margin-left: 0.3rem; -} - -.lt-sort-icon.icon-ascending:before, -.lt-sort-icon.icon-descending:before { - font-size: 0.6em; -} diff --git a/core/client/app/templates/-import-errors.hbs b/core/client/app/templates/-import-errors.hbs deleted file mode 100644 index abc767e207..0000000000 --- a/core/client/app/templates/-import-errors.hbs +++ /dev/null @@ -1,7 +0,0 @@ -{{#if importErrors}} -<table class="table"> -{{#each importErrors as |error|}} - <tr><td>{{error.message}}</td></tr> -{{/each}} -</table> -{{/if}} diff --git a/core/client/app/templates/-user-list-item.hbs b/core/client/app/templates/-user-list-item.hbs deleted file mode 100644 index 7b5eda3b14..0000000000 --- a/core/client/app/templates/-user-list-item.hbs +++ /dev/null @@ -1,18 +0,0 @@ -<span class="user-list-item-figure" style={{component.userImageBackground}}> - <span class="hidden">Photo of {{user.name}}</span> -</span> - -<div class="user-list-item-body"> - <span class="name"> - {{user.name}} - </span> - <br> - <span class="description">Last seen: {{component.lastLogin}}</span> -</div> -<aside class="user-list-item-aside"> - {{#unless session.user.isAuthor}} - {{#each user.roles as |role|}} - <span class="role-label {{role.lowerCaseName}}">{{role.name}}</span> - {{/each}} - {{/unless}} -</aside> diff --git a/core/client/app/templates/about.hbs b/core/client/app/templates/about.hbs deleted file mode 100644 index 02c0c2367f..0000000000 --- a/core/client/app/templates/about.hbs +++ /dev/null @@ -1,44 +0,0 @@ -<section class="gh-view js-settings-content"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>About Ghost</span>{{/gh-view-title}} - </header> - <section class="view-content"> - <header class="gh-about-header"> - <img class="gh-logo" src="{{gh-path 'admin' '/img/ghost-logo.png'}}" alt="Ghost" /> - {{!-- TODO: fix about notifications --}} - {{gh-notifications location="about-upgrade" notify="updateNotificationChange"}} - </header> - - <section class="gh-env-details"> - <ul class="gh-env-list"> - <li class="gh-env-list-version"><strong>Version</strong> {{model.version}}</li> - <li><strong>Environment</strong> {{model.environment}}</li> - <li class="gh-env-list-database-type"><strong>Database</strong> {{model.database}}</li> - <li><strong>Mail</strong> {{#if model.mail}}{{model.mail}}{{else}}Native{{/if}}</li> - </ul> - <div class="gh-env-help"> - <a href="http://support.ghost.org" class="btn btn-minor" target="_blank">User Documentation</a> - <a href="https://ghost.org/slack/" class="btn btn-minor" target="_blank">Get Help With Ghost</a> - </div> - </section> - - <section class="gh-credits"> - <h2>The People Who Made it Possible</h2> - - <section class="gh-contributors"> - {{partial "contributors"}} - </section> - - <p>Ghost is built by an incredible group of contributors from all over the world. Here are just a few of the people who helped create the version you’re using right now.</p> - - <a href="https://ghost.org/about/contribute/" class="btn btn-blue btn-lg">Find out how you can get involved</a> - - </section> - - <footer class="gh-copyright-info"> - Copyright 2013 - {{copyrightYear}} Ghost Foundation, released under the <a href="https://github.com/TryGhost/Ghost/blob/master/LICENSE">MIT license</a>. - <br> - <a href="https://ghost.org/">Ghost</a> is a trademark of the <a href="https://ghost.org/about/">Ghost Foundation</a>. - </footer> - </section> -</section> diff --git a/core/client/app/templates/application.hbs b/core/client/app/templates/application.hbs deleted file mode 100644 index eea9080b04..0000000000 --- a/core/client/app/templates/application.hbs +++ /dev/null @@ -1,28 +0,0 @@ -{{#gh-app showSettingsMenu=showSettingsMenu}} - {{#gh-skip-link anchor=".gh-main"}}Skip to main content{{/gh-skip-link}} - - {{gh-alerts notify="topNotificationChange"}} - - <div class="gh-viewport {{if autoNav 'gh-autonav'}} {{if showSettingsMenu 'settings-menu-expanded'}} {{if showMobileMenu 'mobile-menu-expanded'}}"> - {{#if showNavMenu}} - {{gh-nav-menu open=autoNavOpen toggleMaximise="toggleAutoNav" openAutoNav="openAutoNav" showMarkdownHelp="toggleMarkdownHelpModal" closeMobileMenu="closeMobileMenu"}} - {{/if}} - - {{#gh-main onMouseEnter="closeAutoNav" data-notification-count=topNotificationCount}} - {{outlet}} - {{/gh-main}} - - - {{gh-notifications}} - - {{gh-content-cover onClick="closeMenus" onMouseEnter="closeAutoNav"}} - - {{outlet "settings-menu"}} - </div>{{!gh-viewport}} -{{/gh-app}} - -{{#if showMarkdownHelpModal}} - {{gh-fullscreen-modal "markdown-help" - close=(route-action "toggleMarkdownHelpModal") - modifier="wide"}} -{{/if}} diff --git a/core/client/app/templates/components/gh-activating-list-item.hbs b/core/client/app/templates/components/gh-activating-list-item.hbs deleted file mode 100644 index b32091944f..0000000000 --- a/core/client/app/templates/components/gh-activating-list-item.hbs +++ /dev/null @@ -1 +0,0 @@ -{{#link-to route alternateActive=(action "setActive") classNameBindings="linkClasses"}}{{title}}{{yield}}{{/link-to}} diff --git a/core/client/app/templates/components/gh-alert.hbs b/core/client/app/templates/components/gh-alert.hbs deleted file mode 100644 index 299f731c00..0000000000 --- a/core/client/app/templates/components/gh-alert.hbs +++ /dev/null @@ -1,4 +0,0 @@ -<div class="gh-alert-content"> - {{message.message}} -</div> -<button class="gh-alert-close icon-x" {{action "closeNotification"}}><span class="hidden">Close</span></button> \ No newline at end of file diff --git a/core/client/app/templates/components/gh-alerts.hbs b/core/client/app/templates/components/gh-alerts.hbs deleted file mode 100644 index e7920538a3..0000000000 --- a/core/client/app/templates/components/gh-alerts.hbs +++ /dev/null @@ -1,3 +0,0 @@ -{{#each messages as |message|}} - {{gh-alert message=message}} -{{/each}} \ No newline at end of file diff --git a/core/client/app/templates/components/gh-app.hbs b/core/client/app/templates/components/gh-app.hbs deleted file mode 100644 index 889d9eeadc..0000000000 --- a/core/client/app/templates/components/gh-app.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield}} diff --git a/core/client/app/templates/components/gh-blog-url.hbs b/core/client/app/templates/components/gh-blog-url.hbs deleted file mode 100644 index 457738f620..0000000000 --- a/core/client/app/templates/components/gh-blog-url.hbs +++ /dev/null @@ -1 +0,0 @@ -{{{config.blogUrl}}} \ No newline at end of file diff --git a/core/client/app/templates/components/gh-content-preview-content.hbs b/core/client/app/templates/components/gh-content-preview-content.hbs deleted file mode 100644 index 889d9eeadc..0000000000 --- a/core/client/app/templates/components/gh-content-preview-content.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield}} diff --git a/core/client/app/templates/components/gh-content-view-container.hbs b/core/client/app/templates/components/gh-content-view-container.hbs deleted file mode 100644 index e76a0b3886..0000000000 --- a/core/client/app/templates/components/gh-content-view-container.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield previewIsHidden}} diff --git a/core/client/app/templates/components/gh-datetime-input.hbs b/core/client/app/templates/components/gh-datetime-input.hbs deleted file mode 100644 index 6b6e5ea887..0000000000 --- a/core/client/app/templates/components/gh-datetime-input.hbs +++ /dev/null @@ -1,5 +0,0 @@ -{{gh-input value=datetime - id=inputId - class=inputClass - name=inputName - stopEnterKeyDownPropagation="true"}} diff --git a/core/client/app/templates/components/gh-ed-preview.hbs b/core/client/app/templates/components/gh-ed-preview.hbs deleted file mode 100644 index e54e56c1d3..0000000000 --- a/core/client/app/templates/components/gh-ed-preview.hbs +++ /dev/null @@ -1,14 +0,0 @@ -{{previewHTML}} - -{{#each imageUploadComponents as |uploader|}} - {{#ember-wormhole to=uploader.destinationElementId}} - {{gh-image-uploader-with-preview - image=uploader.src - text="Upload an image" - update=(action "updateImageSrc" uploader.index) - remove=(action "updateImageSrc" uploader.index "") - uploadStarted=uploadStarted - uploadFinished=uploadFinished - formChanged=(action "updateHeight")}} - {{/ember-wormhole}} -{{/each}} diff --git a/core/client/app/templates/components/gh-editor-save-button.hbs b/core/client/app/templates/components/gh-editor-save-button.hbs deleted file mode 100644 index 40608cccea..0000000000 --- a/core/client/app/templates/components/gh-editor-save-button.hbs +++ /dev/null @@ -1,22 +0,0 @@ -{{#gh-spin-button type="button" classNameBindings=":btn :btn-sm :js-publish-button isDangerous:btn-red:btn-blue" action="save" submitting=submitting}}{{saveText}}{{/gh-spin-button}} - -{{#gh-dropdown-button dropdownName="post-save-menu" classNameBindings=":btn :btn-sm isDangerous:btn-red:btn-blue btnopen:active :dropdown-toggle :up"}} - <i class="options icon-arrow2"></i> - <span class="sr-only">Toggle Settings Menu</span> -{{/gh-dropdown-button}} -{{#gh-dropdown name="post-save-menu" closeOnClick="true" classNames="editor-options"}} - <ul class="dropdown-menu dropdown-triangle-bottom-right"> - <li class="post-save-publish {{if willPublish 'active'}}"> - <a {{action "setSaveType" "publish"}} href="#">{{publishText}}</a> - </li> - <li class="post-save-draft {{unless willPublish 'active'}}"> - <a {{action "setSaveType" "draft"}} href="#">{{draftText}}</a> - </li> - {{#unless isNew}} - <li class="divider delete"></li> - <li class="delete"> - <a {{action "delete"}} href="#">{{deleteText}}</a> - </li> - {{/unless}} - </ul> -{{/gh-dropdown}} diff --git a/core/client/app/templates/components/gh-editor.hbs b/core/client/app/templates/components/gh-editor.hbs deleted file mode 100644 index 569a43ac0c..0000000000 --- a/core/client/app/templates/components/gh-editor.hbs +++ /dev/null @@ -1,50 +0,0 @@ -<section class="entry-markdown js-entry-markdown {{if markdownActive 'active'}}"> - <header class="floatingheader"> - <span class="desktop-tabs"><a class="markdown-help-label" href="" title="Markdown Help" {{action (route-action "toggleMarkdownHelpModal")}}>Markdown</a></span> - <span class="mobile-tabs"> - <a href="#" {{action 'selectTab' 'markdown'}} class="{{if markdownActive 'active'}}">Markdown</a> - <a href="#" {{action 'selectTab' 'preview'}} class="{{if previewActive 'active'}}">Preview</a> - </span> - <a class="markdown-help-icon" href="" title="Markdown Help" {{action (route-action "toggleMarkdownHelpModal")}}><i class="icon-markdown"></i></a> - </header> - <section id="entry-markdown-content" class="entry-markdown-content"> - {{gh-ed-editor classNames="markdown-editor js-markdown-editor" - tabindex="1" - spellcheck="true" - value=value - setEditor=(action "setEditor") - updateScrollInfo=(action "updateScrollInfo") - toggleCopyHTMLModal=(action "toggleCopyHTMLModal") - onFocusIn=editorFocused - height=height - focus=shouldFocusEditor - readonly=editorDisabled}} - </section> -</section> - -<section class="entry-preview js-entry-preview {{if previewActive 'active'}}"> - <header class="floatingheader"> - <span class="desktop-tabs"><a target="_blank" href="{{previewUrl}}">Preview</a></span> - <span class="mobile-tabs"> - <a href="#" {{action 'selectTab' 'markdown'}} class="{{if markdownActive 'active'}}">Markdown</a> - <a href="#" {{action 'selectTab' 'preview'}} class="{{if previewActive 'active'}}">Preview</a> - </span> - <span class="entry-word-count">{{gh-count-words value}}</span> - </header> - <section class="entry-preview-content js-entry-preview-content"> - {{gh-ed-preview classNames="rendered-markdown js-rendered-markdown" - markdown=value - scrollPosition=scrollPosition - updateHeight=(action "updateHeight") - uploadStarted=(action "disableEditor") - uploadFinished=(action "enableEditor") - updateImageSrc=(action "handleImgUpload")}} - </section> -</section> - -{{#if showCopyHTMLModal}} - {{gh-fullscreen-modal "copy-html" - model=copyHTMLModalContent - close=(action "toggleCopyHTMLModal") - modifier="action"}} -{{/if}} diff --git a/core/client/app/templates/components/gh-error-message.hbs b/core/client/app/templates/components/gh-error-message.hbs deleted file mode 100644 index 75af56131a..0000000000 --- a/core/client/app/templates/components/gh-error-message.hbs +++ /dev/null @@ -1 +0,0 @@ -{{message}} diff --git a/core/client/app/templates/components/gh-feature-flag.hbs b/core/client/app/templates/components/gh-feature-flag.hbs deleted file mode 100644 index b67a90b479..0000000000 --- a/core/client/app/templates/components/gh-feature-flag.hbs +++ /dev/null @@ -1,3 +0,0 @@ -{{input id=for name=name type="checkbox" checked=value}} -<span class="input-toggle-component"></span> -<p>{{{yield}}}</p> diff --git a/core/client/app/templates/components/gh-file-upload.hbs b/core/client/app/templates/components/gh-file-upload.hbs deleted file mode 100644 index 408f340a15..0000000000 --- a/core/client/app/templates/components/gh-file-upload.hbs +++ /dev/null @@ -1,4 +0,0 @@ -<input data-url="upload" class="gh-input btn-block" type="file" name="importfile" accept="{{options.acceptEncoding}}"> -<button type="submit" class="btn btn-green btn-block" id="startupload" disabled={{uploadButtonDisabled}} {{action "upload"}}> - {{uploadButtonText}} -</button> diff --git a/core/client/app/templates/components/gh-file-uploader.hbs b/core/client/app/templates/components/gh-file-uploader.hbs deleted file mode 100644 index a7927844fc..0000000000 --- a/core/client/app/templates/components/gh-file-uploader.hbs +++ /dev/null @@ -1,20 +0,0 @@ -{{#if file}} - {{!-- Upload in progress! --}} - {{#if failureMessage}} - <div class="failed">{{failureMessage}}</div> - {{/if}} - <div class="progress-container"> - <div class="progress"> - <div class="bar {{if failureMessage "fail"}}" style={{progressStyle}}></div> - </div> - </div> - {{#if failureMessage}} - <button class="btn btn-green" {{action "reset"}}>Try Again</button> - {{/if}} -{{else}} - <div class="upload-form"> - {{#x-file-input multiple=false alt=labelText action=(action 'fileSelected') accept="text/csv"}} - <div class="description">{{labelText}}</div> - {{/x-file-input}} - </div> -{{/if}} diff --git a/core/client/app/templates/components/gh-fullscreen-modal.hbs b/core/client/app/templates/components/gh-fullscreen-modal.hbs deleted file mode 100644 index 06ecdf5f79..0000000000 --- a/core/client/app/templates/components/gh-fullscreen-modal.hbs +++ /dev/null @@ -1,11 +0,0 @@ -<div class="liquid-tether-overlay {{overlayClass}} {{if on-overlay-click 'clickable'}}" {{action 'clickOverlay'}}></div> -<div class="liquid-tether {{tetherClass}}"> - {{#if hasBlock}} - {{yield}} - {{else}} - {{component modalPath - model=model - confirm=(action 'confirm') - closeModal=(action 'close')}} - {{/if}} -</div> diff --git a/core/client/app/templates/components/gh-image-uploader-with-preview.hbs b/core/client/app/templates/components/gh-image-uploader-with-preview.hbs deleted file mode 100644 index 9481d4c482..0000000000 --- a/core/client/app/templates/components/gh-image-uploader-with-preview.hbs +++ /dev/null @@ -1,16 +0,0 @@ -{{#if image}} - <div class="gh-image-uploader --with-image"> - <div><img src={{image}}></div> - <a class="image-cancel icon-trash" title="Delete" {{action remove}}> - <span class="hidden">Delete</span> - </a> - </div> -{{else}} - {{gh-image-uploader - text=text - update=(action 'update') - onInput=(action 'onInput') - uploadStarted=(action 'uploadStarted') - uploadFinished=(action 'uploadFinished') - formChanged=(action 'formChanged')}} -{{/if}} diff --git a/core/client/app/templates/components/gh-image-uploader.hbs b/core/client/app/templates/components/gh-image-uploader.hbs deleted file mode 100644 index 09349492c5..0000000000 --- a/core/client/app/templates/components/gh-image-uploader.hbs +++ /dev/null @@ -1,43 +0,0 @@ -{{#if file}} - {{!-- Upload in progress! --}} - {{#if failureMessage}} - <div class="failed">{{failureMessage}}</div> - {{/if}} - <div class="progress-container"> - <div class="progress"> - <div class="bar {{if failureMessage "fail"}}" style={{progressStyle}}></div> - </div> - </div> - {{#if failureMessage}} - <button class="btn btn-green" {{action "reset"}}>Try Again</button> - {{/if}} -{{else}} - {{#if showUploadForm}} - {{!-- file selection/drag-n-drop --}} - <div class="upload-form"> - {{#x-file-input multiple=false alt=text action=(action 'fileSelected') accept="image/gif,image/jpg,image/jpeg,image/png,image/svg+xml"}} - <div class="description">{{text}}</div> - {{/x-file-input}} - </div> - - <a class="image-url" {{action 'switchForm' 'url-input'}}> - <i class="icon-link"><span class="hidden">URL</span></i> - </a> - {{else}} - {{!-- URL input --}} - <form class="url-form"> - {{one-way-input class="url gh-input" placeholder="http://" value=url update=(action "onInput") onenter=(action "saveUrl")}} - {{#if saveButton}} - <button class="btn btn-blue gh-input" {{action 'saveUrl'}}>Save</button> - {{else}} - <div class="description">{{text}}</div> - {{/if}} - </form> - - {{#if canShowUploadForm}} - <a class="image-upload icon-photos" title="Add image" {{action 'switchForm' 'upload'}}> - <span class="hidden">Upload</span> - </a> - {{/if}} - {{/if}} -{{/if}} diff --git a/core/client/app/templates/components/gh-infinite-scroll.hbs b/core/client/app/templates/components/gh-infinite-scroll.hbs deleted file mode 100644 index adc76bdb1f..0000000000 --- a/core/client/app/templates/components/gh-infinite-scroll.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield (action "checkScroll")}} diff --git a/core/client/app/templates/components/gh-menu-toggle.hbs b/core/client/app/templates/components/gh-menu-toggle.hbs deleted file mode 100644 index dae2476648..0000000000 --- a/core/client/app/templates/components/gh-menu-toggle.hbs +++ /dev/null @@ -1 +0,0 @@ -<i class="{{iconClass}}" role="button"></i> diff --git a/core/client/app/templates/components/gh-modal-dialog.hbs b/core/client/app/templates/components/gh-modal-dialog.hbs deleted file mode 100644 index f024e0b893..0000000000 --- a/core/client/app/templates/components/gh-modal-dialog.hbs +++ /dev/null @@ -1,18 +0,0 @@ -<div class="modal-container js-modal-container" {{action "closeModal"}}> - <article class="{{klass}} js-modal"> - <section class="modal-content" {{action "noBubble" bubbles=false preventDefault=false}}> - {{#if title}}<header class="modal-header"><h1>{{title}}</h1></header>{{/if}} - {{#if showClose}}<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a>{{/if}} - <section class="modal-body"> - {{yield}} - </section> - {{#if confirm}} - <footer class="modal-footer"> - {{! Buttons must be on one line to prevent white-space errors }} - <button type="button" class="{{rejectButtonClass}} btn-minor js-button-reject" {{action "confirm" "reject"}}>{{confirm.reject.text}}</button><button type="button" class="{{acceptButtonClass}} js-button-accept" {{action "confirm" "accept"}}>{{confirm.accept.text}}</button> - </footer> - {{/if}} - </section> - </article> -</div> -<div class="modal-background js-modal-background"></div> diff --git a/core/client/app/templates/components/gh-nav-menu.hbs b/core/client/app/templates/components/gh-nav-menu.hbs deleted file mode 100644 index a789ef8bd1..0000000000 --- a/core/client/app/templates/components/gh-nav-menu.hbs +++ /dev/null @@ -1,69 +0,0 @@ -{{#gh-dropdown-button tagName="header" class="gh-nav-menu" dropdownName="user-menu"}} - <div class="gh-nav-menu-icon" style={{navMenuIcon}}></div> - <div class="gh-nav-menu-details"> - <div class="gh-nav-menu-details-blog">{{config.blogTitle}}</div> - <div class="gh-nav-menu-details-user">{{session.user.name}}</div> - </div> - <i class="icon-arrow"></i> -{{/gh-dropdown-button}} -{{#gh-dropdown tagName="div" name="user-menu" closeOnClick="true"}} - <ul class="dropdown-menu dropdown-triangle-top js-user-menu-dropdown-menu" role="menu" style="right:-20px;left:auto;"> - <li role="presentation">{{#link-to "about" classNames="gh-nav-menu-about dropdown-item js-nav-item" role="menuitem" tabindex="-1"}}<i class="icon-shop"></i> About Ghost{{/link-to}}</li> - <li class="divider"></li> - <li role="presentation">{{#link-to "team.user" session.user.slug classNames="dropdown-item user-menu-profile js-nav-item" role="menuitem" tabindex="-1"}}<i class="icon-user"></i> Your Profile{{/link-to}}</li> - <li role="presentation">{{#link-to "signout" classNames="dropdown-item user-menu-signout" role="menuitem" tabindex="-1"}}<i class="icon-signout"></i> Sign Out{{/link-to}}</li> - </ul> -{{/gh-dropdown}} -<section class="gh-nav-body"> - <section class="gh-nav-search"> - {{gh-search-input class="gh-nav-search-input"}} - </section> - <ul class="gh-nav-list gh-nav-main"> - {{!<li><i class="icon-dash"></i>Dashboard</li>}} - <li>{{#link-to "editor.new" classNames="gh-nav-main-editor"}}<i class="icon-pen"></i>New Post{{/link-to}}</li> - <li>{{#link-to "posts" classNames="gh-nav-main-content"}}<i class="icon-content"></i>Content{{/link-to}}</li> - {{!<li><a href="#"><i class="icon-user"></i>My Posts</a></li>}} - <li>{{#link-to "team" classNames="gh-nav-main-users"}}<i class="icon-team"></i>Team{{/link-to}}</li> - {{!<li><a href="#"><i class="icon-idea"></i>Ideas</a></li>}} - {{#if feature.subscribers}} - {{#if (gh-user-can-admin session.user)}} - <li>{{#link-to "subscribers" classNames="gh-nav-main-subscribers"}}<i class="icon-mail"></i>Subscribers{{/link-to}}</li> - {{/if}} - {{/if}} - </ul> - {{#if (gh-user-can-admin session.user)}} - <ul class="gh-nav-list gh-nav-settings"> - <li class="gh-nav-list-h">Settings</li> - <li>{{#link-to "settings.general" classNames="gh-nav-settings-general"}}<i class="icon-settings"></i>General{{/link-to}}</li> - {{!<li><i class="icon-design"></i>Themes</li>}} - <li>{{#link-to "settings.navigation" classNames="gh-nav-settings-navigation"}}<i class="icon-compass"></i>Navigation{{/link-to}}</li> - <li>{{#link-to "settings.tags" classNames="gh-nav-settings-tags"}}<i class="icon-tag"></i>Tags{{/link-to}}</li> - <li>{{#link-to "settings.code-injection" classNames="gh-nav-settings-code-injection"}}<i class="icon-code"></i>Code Injection{{/link-to}}</li> - <li>{{#link-to "settings.apps" classNames="gh-nav-settings-apps"}}<i class="icon-box"></i>Apps{{/link-to}}</li> - <li>{{#link-to "settings.labs" classNames="gh-nav-settings-labs"}}<i class="icon-labs"></i>Labs{{/link-to}}</li> - </ul> - {{/if}} -</section> -<footer class="gh-nav-footer"> - {{gh-menu-toggle desktopAction="toggleAutoNav" mobileAction="closeMobileMenu"}} - <a class="gh-nav-footer-sitelink" href="{{config.blogUrl}}/" target="_blank">View blog</a> - <div class="gh-help-menu"> - {{#gh-dropdown-button dropdownName="help-menu" tagName="div"}} - <div class="gh-help-button"> - <i class="icon-question"><span class="hidden">Help</span></i> - </div> - {{/gh-dropdown-button}} - {{#gh-dropdown tagName="div" name="help-menu" closeOnClick="true"}} - <ul class="dropdown-menu dropdown-triangle-bottom" role="menu"> - <li role="presentation"><a class="dropdown-item help-menu-support" role="menuitem" tabindex="-1" href="http://support.ghost.org/" target="_blank"><i class="icon-ambulance"></i> Support Center</a></li> - <li role="presentation"><a class="dropdown-item help-menu-tweet" role="menuitem" tabindex="-1" href="https://twitter.com/intent/tweet?text=%40TryGhost+Hi%21+Can+you+help+me+with+&related=TryGhost" target="_blank" onclick="window.open(this.href, 'twitter-share', 'width=550,height=235');return false;"><i class="icon-twitter"></i> Tweet @TryGhost!</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-how-to" role="menuitem" tabindex="-1" href="http://support.ghost.org/how-to-use-ghost/" target="_blank"><i class="icon-book"></i> How to Use Ghost</a></li> - <li role="presentation"><a class="dropdown-item help-menu-markdown" role="menuitem" tabindex="-1" href="" {{action "showMarkdownHelp"}}><i class="icon-markdown"></i> Markdown Help</a></li> - <li class="divider"></li> - <li role="presentation"><a class="dropdown-item help-menu-wishlist" role="menuitem" tabindex="-1" href="http://ideas.ghost.org/" target="_blank"><i class="icon-idea"></i> Wishlist</a></li> - </ul> - {{/gh-dropdown}} - </div>{{! .help-menu }} -</footer> -<div class="gh-autonav-toggle" {{action "openAutoNav" on="mouseEnter"}}></div> diff --git a/core/client/app/templates/components/gh-navigation.hbs b/core/client/app/templates/components/gh-navigation.hbs deleted file mode 100644 index 889d9eeadc..0000000000 --- a/core/client/app/templates/components/gh-navigation.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield}} diff --git a/core/client/app/templates/components/gh-navitem.hbs b/core/client/app/templates/components/gh-navitem.hbs deleted file mode 100644 index 6ecc4ba421..0000000000 --- a/core/client/app/templates/components/gh-navitem.hbs +++ /dev/null @@ -1,26 +0,0 @@ -{{#unless navItem.isNew}} - <span class="gh-blognav-grab icon-grab"> - <span class="sr-only">Reorder</span> - </span> -{{/unless}} - -<div class="gh-blognav-line"> - {{#gh-validation-status-container tagName="span" class="gh-blognav-label" errors=navItem.errors property="label" hasValidated=navItem.hasValidated}} - {{gh-trim-focus-input focus=navItem.last placeholder="Label" value=navItem.label keyPress=(action "clearLabelErrors")}} - {{gh-error-message errors=navItem.errors property="label"}} - {{/gh-validation-status-container}} - {{#gh-validation-status-container tagName="span" class="gh-blognav-url" errors=navItem.errors property="url" hasValidated=navItem.hasValidated}} - {{gh-navitem-url-input baseUrl=baseUrl url=navItem.url isNew=navItem.isNew change="updateUrl" clearErrors=(action "clearUrlErrors")}} - {{gh-error-message errors=navItem.errors property="url"}} - {{/gh-validation-status-container}} -</div> - -{{#if navItem.isNew}} - <button type="button" class="gh-blognav-add" {{action "addItem"}}> - <i class="icon-add2"><span class="sr-only">Add</span></i> - </button> -{{else}} - <button type="button" class="gh-blognav-delete" {{action "deleteItem" navItem}}> - <i class="icon-trash"><span class="sr-only">Delete</span></i> - </button> -{{/if}} diff --git a/core/client/app/templates/components/gh-notification.hbs b/core/client/app/templates/components/gh-notification.hbs deleted file mode 100644 index 8dc1a21626..0000000000 --- a/core/client/app/templates/components/gh-notification.hbs +++ /dev/null @@ -1,4 +0,0 @@ -<div class="gh-notification-content"> - {{message.message}} -</div> -<button class="gh-notification-close icon-x" {{action "closeNotification"}}><span class="hidden">Close</span></button> \ No newline at end of file diff --git a/core/client/app/templates/components/gh-notifications.hbs b/core/client/app/templates/components/gh-notifications.hbs deleted file mode 100644 index 4d4a955bc6..0000000000 --- a/core/client/app/templates/components/gh-notifications.hbs +++ /dev/null @@ -1,3 +0,0 @@ -{{#each messages as |message|}} - {{gh-notification message=message}} -{{/each}} diff --git a/core/client/app/templates/components/gh-posts-list-item.hbs b/core/client/app/templates/components/gh-posts-list-item.hbs deleted file mode 100644 index f4a0b59217..0000000000 --- a/core/client/app/templates/components/gh-posts-list-item.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield this}} diff --git a/core/client/app/templates/components/gh-profile-image.hbs b/core/client/app/templates/components/gh-profile-image.hbs deleted file mode 100644 index 0a3ca9b4b1..0000000000 --- a/core/client/app/templates/components/gh-profile-image.hbs +++ /dev/null @@ -1,17 +0,0 @@ -<figure class="account-image js-file-upload"> - {{#unless hasUploadedImage}} - <div class="placeholder-img" style={{defaultImage}}></div> - <div id="account-image" class="gravatar-img" style={{imageBackground}}> - <span class="sr-only">User image</span> - </div> - {{/unless}} - - <div class="js-img-preview"></div> - - <span class="edit-account-image js-img-dropzone"> - <i class="icon-photos"> - <span class="sr-only">Upload an image</span> - </i> - </span> - {{#if fileStorage}}<input type="file" class="file-uploader js-file-input" name="uploadimage">{{/if}} -</figure> diff --git a/core/client/app/templates/components/gh-search-input.hbs b/core/client/app/templates/components/gh-search-input.hbs deleted file mode 100644 index 04a0fef6cb..0000000000 --- a/core/client/app/templates/components/gh-search-input.hbs +++ /dev/null @@ -1,13 +0,0 @@ -{{#power-select - search=(action "search") - onchange=(action "openSelected") - placeholder="Search" - onopen=(action "onFocus") - onclose=(action "onBlur") - searchEnabled=false - triggerComponent="gh-search-input/trigger" - renderInPlace=true - loadingMessage="Loading" - as |name term|}} - {{highlighted-text name.title term}} -{{/power-select}} diff --git a/core/client/app/templates/components/gh-search-input/trigger.hbs b/core/client/app/templates/components/gh-search-input/trigger.hbs deleted file mode 100644 index f9f613ab1c..0000000000 --- a/core/client/app/templates/components/gh-search-input/trigger.hbs +++ /dev/null @@ -1,12 +0,0 @@ -<div class="ember-power-select-search" onmousedown={{action "captureMouseDown"}}> - <input type="search" autocomplete="off" - autocorrect="off" autocapitalize="off" - value={{if extra.labelPath (get selected extra.labelPath) selected}} - spellcheck="false" role="combobox" - placeholder={{placeholder}} - oninput={{action 'search' value="target.value"}} - onmousedown={{action "captureMouseDown"}} - onkeydown={{action "handleKeydown"}} - onblur={{action "resetInput"}}> - <button class="gh-nav-search-button" {{action "focusInput"}}><i class="icon-search"></i><span class="sr-only">Search</span></button> -</div> diff --git a/core/client/app/templates/components/gh-select-native.hbs b/core/client/app/templates/components/gh-select-native.hbs deleted file mode 100644 index 153a4012f6..0000000000 --- a/core/client/app/templates/components/gh-select-native.hbs +++ /dev/null @@ -1,14 +0,0 @@ -<select {{action "change" on="change"}}> - {{#if prompt}} - <option disabled selected={{is-not selection}}> - {{prompt}} - </option> - {{/if}} - - {{#each content as |item|}} - <option value="{{get item optionValuePath}}" - selected={{is-equal item selection}}> - {{get item optionLabelPath}} - </option> - {{/each}} -</select> diff --git a/core/client/app/templates/components/gh-spin-button.hbs b/core/client/app/templates/components/gh-spin-button.hbs deleted file mode 100644 index eb93ed58a1..0000000000 --- a/core/client/app/templates/components/gh-spin-button.hbs +++ /dev/null @@ -1,9 +0,0 @@ -{{#if showSpinner}} - <span class="spinner"></span> -{{else}} - {{#if buttonText}} - {{buttonText}} - {{else}} - {{{yield}}} - {{/if}} -{{/if}} \ No newline at end of file diff --git a/core/client/app/templates/components/gh-subscribers-table-delete-cell.hbs b/core/client/app/templates/components/gh-subscribers-table-delete-cell.hbs deleted file mode 100644 index 0ea48d332e..0000000000 --- a/core/client/app/templates/components/gh-subscribers-table-delete-cell.hbs +++ /dev/null @@ -1 +0,0 @@ -<button class="btn btn-minor btn-sm" {{action tableActions.delete row.content}}><i class="icon-trash"></i></button> diff --git a/core/client/app/templates/components/gh-subscribers-table.hbs b/core/client/app/templates/components/gh-subscribers-table.hbs deleted file mode 100644 index 0ab317fc78..0000000000 --- a/core/client/app/templates/components/gh-subscribers-table.hbs +++ /dev/null @@ -1,17 +0,0 @@ -{{#gh-light-table table scrollContainer=".subscribers-table" scrollBuffer=100 onScrolledToBottom=(action 'onScrolledToBottom') as |t|}} - {{t.head onColumnClick=(action sortByColumn) iconAscending="icon-ascending" iconDescending="icon-descending"}} - - {{#t.body canSelect=false tableActions=(hash delete=(action delete)) as |body|}} - {{#if isLoading}} - {{#body.loader}} - Loading... - {{/body.loader}} - {{else}} - {{#if table.isEmpty}} - {{#body.no-data}} - No subscribers found. - {{/body.no-data}} - {{/if}} - {{/if}} - {{/t.body}} -{{/gh-light-table}} diff --git a/core/client/app/templates/components/gh-tag-settings-form.hbs b/core/client/app/templates/components/gh-tag-settings-form.hbs deleted file mode 100644 index 83651ca560..0000000000 --- a/core/client/app/templates/components/gh-tag-settings-form.hbs +++ /dev/null @@ -1,88 +0,0 @@ -<div class="{{if isViewingSubview 'settings-menu-pane-out-left' 'settings-menu-pane-in'}} settings-menu settings-menu-pane tag-settings-pane"> - <div class="settings-menu-header {{if isMobile 'subview'}}"> - {{#if isMobile}} - {{#link-to 'settings.tags' class="back icon-arrow-left settings-menu-header-action"}}<span class="hidden">Back</span>{{/link-to}} - <h4>{{title}}</h4> - <div style="width:23px;">{{!flexbox space-between}}</div> - {{else}} - <h4>{{title}}</h4> - {{/if}} - </div> - <div class="settings-menu-content"> - {{gh-image-uploader-with-preview - image=tag.image - text="Add tag image" - update=(action "setCoverImage") - remove=(action "clearCoverImage")}} - <form> - {{#gh-form-group errors=tag.errors hasValidated=tag.hasValidated property="name"}} - <label for="tag-name">Name</label> - {{gh-input id="tag-name" name="name" type="text" value=scratchName focus-out=(action 'setProperty' 'name')}} - {{gh-error-message errors=tag.errors property="name"}} - {{/gh-form-group}} - - {{#gh-form-group errors=tag.errors hasValidated=tag.hasValidated property="slug"}} - <label for="tag-slug">URL</label> - {{gh-input id="tag-slug" name="slug" type="text" value=scratchSlug focus-out=(action 'setProperty' 'slug')}} - {{gh-url-preview prefix="tag" slug=scratchSlug tagName="p" classNames="description"}} - {{gh-error-message errors=activeTag.errors property="slug"}} - {{/gh-form-group}} - - {{#gh-form-group errors=tag.errors hasValidated=tag.hasValidated property="description"}} - <label for="tag-description">Description</label> - {{gh-textarea id="tag-description" name="description" value=scratchDescription focus-out=(action 'setProperty' 'description')}} - {{gh-error-message errors=tag.errors property="description"}} - <p>Maximum: <b>200</b> characters. You’ve used {{gh-count-down-characters scratchDescription 200}}</p> - {{/gh-form-group}} - - <ul class="nav-list nav-list-block"> - <li class="nav-list-item" {{action 'openMeta'}}> - <button type="button" class="meta-data-button"> - <b>Meta Data</b> - <span>Extra content for SEO and social media.</span> - </button> - <i class="icon-arrow-right"></i> - </li> - </ul> - - {{#unless tag.isNew}} - <button type="button" class="btn btn-link btn-sm tag-delete-button" {{action "deleteTag"}}><i class="icon-trash"></i> Delete Tag</button> - {{/unless}} - </form> - </div> -</div>{{! .settings-menu-pane }} - -<div class="{{if isViewingSubview 'settings-menu-pane-in' 'settings-menu-pane-out-right'}} settings-menu settings-menu-pane tag-meta-settings-pane"> - <div class="settings-menu-header subview"> - <button {{action "closeMeta"}} class="back icon-arrow-left settings-menu-header-action"><span class="hidden">Back</span></button> - <h4>Meta Data</h4> - <div style="width:23px;">{{!flexbox space-between}}</div> - </div> - - <div class="settings-menu-content"> - <form> - {{#gh-form-group errors=tag.errors hasValidated=tag.hasValidated property="metaTitle"}} - <label for="meta-title">Meta Title</label> - {{gh-input id="meta-title" name="metaTitle" type="text" value=scratchMetaTitle focus-out=(action 'setProperty' 'metaTitle')}} - {{gh-error-message errors=tag.errors property="metaTitle"}} - <p>Recommended: <b>70</b> characters. You’ve used {{gh-count-down-characters scratchMetaTitle 70}}</p> - {{/gh-form-group}} - - {{#gh-form-group errors=tag.errors hasValidated=tag.hasValidated property="metaDescription"}} - <label for="meta-description">Meta Description</label> - {{gh-textarea id="meta-description" name="metaDescription" value=scratchMetaDescription focus-out=(action 'setProperty' 'metaDescription')}} - {{gh-error-message errors=tag.errors property="metaDescription"}} - <p>Recommended: <b>156</b> characters. You’ve used {{gh-count-down-characters scratchMetaDescription 156}}</p> - {{/gh-form-group}} - - <div class="form-group"> - <label>Search Engine Result Preview</label> - <div class="seo-preview"> - <div class="seo-preview-title">{{seoTitle}}</div> - <div class="seo-preview-link">{{seoURL}}</div> - <div class="seo-preview-description">{{seoDescription}}</div> - </div> - </div> - </form> - </div> -</div> diff --git a/core/client/app/templates/components/gh-tag.hbs b/core/client/app/templates/components/gh-tag.hbs deleted file mode 100644 index 92ba536b36..0000000000 --- a/core/client/app/templates/components/gh-tag.hbs +++ /dev/null @@ -1,8 +0,0 @@ -<div class="settings-tag"> - {{#link-to 'settings.tags.tag' tag class="tag-edit-button"}} - <span class="tag-title">{{tag.name}}</span> - <span class="label label-default">/{{tag.slug}}</span> - <p class="tag-description">{{tag.description}}</p> - <span class="tags-count">{{tag.count.posts}}</span> - {{/link-to}} -</div> diff --git a/core/client/app/templates/components/gh-tags-management-container.hbs b/core/client/app/templates/components/gh-tags-management-container.hbs deleted file mode 100644 index f4a0b59217..0000000000 --- a/core/client/app/templates/components/gh-tags-management-container.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield this}} diff --git a/core/client/app/templates/components/gh-url-preview.hbs b/core/client/app/templates/components/gh-url-preview.hbs deleted file mode 100644 index 54c93ee14b..0000000000 --- a/core/client/app/templates/components/gh-url-preview.hbs +++ /dev/null @@ -1 +0,0 @@ -{{url}} diff --git a/core/client/app/templates/components/gh-user-active.hbs b/core/client/app/templates/components/gh-user-active.hbs deleted file mode 100644 index f4a0b59217..0000000000 --- a/core/client/app/templates/components/gh-user-active.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield this}} diff --git a/core/client/app/templates/components/gh-user-invited.hbs b/core/client/app/templates/components/gh-user-invited.hbs deleted file mode 100644 index f4a0b59217..0000000000 --- a/core/client/app/templates/components/gh-user-invited.hbs +++ /dev/null @@ -1 +0,0 @@ -{{yield this}} diff --git a/core/client/app/templates/components/gh-view-title.hbs b/core/client/app/templates/components/gh-view-title.hbs deleted file mode 100644 index 19cce3579e..0000000000 --- a/core/client/app/templates/components/gh-view-title.hbs +++ /dev/null @@ -1,2 +0,0 @@ -<button {{action "openMobileMenu"}} class="gh-mobilemenu-button" role="presentation"><i class="icon-gh"><span class="sr-only">Menu</span></i></button> -{{yield}} diff --git a/core/client/app/templates/components/modals/copy-html.hbs b/core/client/app/templates/components/modals/copy-html.hbs deleted file mode 100644 index dbbfd78e4f..0000000000 --- a/core/client/app/templates/components/modals/copy-html.hbs +++ /dev/null @@ -1,8 +0,0 @@ -<header class="modal-header"> - <h1>Generated HTML</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - {{textarea value=generatedHtml rows="6"}} -</div> diff --git a/core/client/app/templates/components/modals/delete-all.hbs b/core/client/app/templates/components/modals/delete-all.hbs deleted file mode 100644 index 5a12e068b5..0000000000 --- a/core/client/app/templates/components/modals/delete-all.hbs +++ /dev/null @@ -1,13 +0,0 @@ -<header class="modal-header"> - <h1>Would you really like to delete all content from your blog?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <p>This is permanent! No backups, no restores, no magic undo button. <br /> We warned you, ok?</p> -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Delete{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/delete-post.hbs b/core/client/app/templates/components/modals/delete-post.hbs deleted file mode 100644 index 0c9ab67823..0000000000 --- a/core/client/app/templates/components/modals/delete-post.hbs +++ /dev/null @@ -1,17 +0,0 @@ -<header class="modal-header"> - <h1>Are you sure you want to delete this post?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <p> - You're about to delete "<strong>{{post.title}}</strong>".<br /> - This is permanent! No backups, no restores, no magic undo button.<br /> - We warned you, ok? - </p> -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Delete{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/delete-subscriber.hbs b/core/client/app/templates/components/modals/delete-subscriber.hbs deleted file mode 100644 index 19973a543b..0000000000 --- a/core/client/app/templates/components/modals/delete-subscriber.hbs +++ /dev/null @@ -1,13 +0,0 @@ -<header class="modal-header"> - <h1>Are you sure?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <strong>WARNING:</strong> All data for this subscriber will be deleted. There is no way to recover this. -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Delete{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/delete-tag.hbs b/core/client/app/templates/components/modals/delete-tag.hbs deleted file mode 100644 index be144fdd9e..0000000000 --- a/core/client/app/templates/components/modals/delete-tag.hbs +++ /dev/null @@ -1,17 +0,0 @@ -<header class="modal-header"> - <h1>Are you sure you want to delete this tag?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <strong>WARNING:</strong> - {{#if tag.post_count}} - <span class="red">This tag is attached to {{tag.count.posts}} {{postInflection}}.</span> - {{/if}} - You're about to delete "<strong>{{tag.name}}</strong>". This is permanent! No backups, no restores, no magic undo button. We warned you, ok? -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Delete{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/delete-user.hbs b/core/client/app/templates/components/modals/delete-user.hbs deleted file mode 100644 index f74faccc12..0000000000 --- a/core/client/app/templates/components/modals/delete-user.hbs +++ /dev/null @@ -1,17 +0,0 @@ -<header class="modal-header"> - <h1>Are you sure you want to delete this user?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - {{#if user.count.posts}} - <strong>WARNING:</strong> <span class="red">This user is the author of {{pluralize user.count.posts 'post'}}.</span> All posts and user data will be deleted. There is no way to recover this. - {{else}} - <strong>WARNING:</strong> All user data will be deleted. There is no way to recover this. - {{/if}} -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Delete{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/import-subscribers.hbs b/core/client/app/templates/components/modals/import-subscribers.hbs deleted file mode 100644 index e04b2aa9c7..0000000000 --- a/core/client/app/templates/components/modals/import-subscribers.hbs +++ /dev/null @@ -1,47 +0,0 @@ -<header class="modal-header"> - <h1> - {{#if response}} - Import Successful - {{else}} - Import Subscribers - {{/if}} - </h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - {{#liquid-if response class="fade-transition"}} - <table class="subscribers-import-results"> - <tr> - <td>Imported:</td> - <td align="left">{{response.imported}}</td> - </tr> - {{#if response.duplicates}} - <tr> - <td>Duplicates:</td> - <td align="left">{{response.duplicates}}</td> - </tr> - {{/if}} - {{#if response.invalid}} - <tr> - <td>Invalid:</td> - <td align="left">{{response.invalid}}</td> - </tr> - {{/if}} - </table> - {{else}} - {{gh-file-uploader - url=uploadUrl - paramName="subscribersfile" - labelText="Select or drag-and-drop a CSV file." - uploadStarted=(action 'uploadStarted') - uploadFinished=(action 'uploadFinished') - uploadSuccess=(action 'uploadSuccess')}} - {{/liquid-if}} -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} disabled={{closeDisabled}} class="btn btn-default btn-minor"> - {{#if response}}Close{{else}}Cancel{{/if}} - </button> -</div> diff --git a/core/client/app/templates/components/modals/invite-new-user.hbs b/core/client/app/templates/components/modals/invite-new-user.hbs deleted file mode 100644 index 8be5e093fd..0000000000 --- a/core/client/app/templates/components/modals/invite-new-user.hbs +++ /dev/null @@ -1,41 +0,0 @@ -<header class="modal-header"> - <h1>Invite a New User</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <fieldset> - {{#gh-form-group errors=errors hasValidated=hasValidated property="email"}} - <label for="new-user-email">Email Address</label> - {{gh-input enter="sendInvite" - class="email" - id="new-user-email" - type="email" - placeholder="Email Address" - name="email" - autofocus="autofocus" - autocapitalize="off" - autocorrect="off" - value=email - focusOut=(action "validate" "email")}} - {{gh-error-message errors=errors property="email"}} - {{/gh-form-group}} - - <div class="form-group for-select"> - <label for="new-user-role">Role</label> - <span class="gh-select" tabindex="0"> - {{gh-select-native id="new-user-role" - content=roles - optionValuePath="id" - optionLabelPath="name" - selection=role - action="setRole" - }} - </span> - </div> - </fieldset> -</div> - -<div class="modal-footer"> - {{#gh-spin-button action="confirm" class="btn btn-green" submitting=submitting}}Send invitation now{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/leave-editor.hbs b/core/client/app/templates/components/modals/leave-editor.hbs deleted file mode 100644 index 15527ba94a..0000000000 --- a/core/client/app/templates/components/modals/leave-editor.hbs +++ /dev/null @@ -1,18 +0,0 @@ -<header class="modal-header"> - <h1>Are you sure you want to leave this page?</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <p> - Hey there! It looks like you're in the middle of writing something and - you haven't saved all of your content. - </p> - - <p>Save before you go!</p> -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Stay</button> - <button {{action "confirm"}} class="btn btn-red">Leave</button> -</div> diff --git a/core/client/app/templates/components/modals/markdown-help.hbs b/core/client/app/templates/components/modals/markdown-help.hbs deleted file mode 100644 index 79dcb4ffb1..0000000000 --- a/core/client/app/templates/components/modals/markdown-help.hbs +++ /dev/null @@ -1,81 +0,0 @@ -<header class="modal-header"> - <h1>Markdown Help</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <section class="markdown-help-container"> - <table class="modal-markdown-help-table"> - <thead> - <tr> - <th>Markdown</th> - <th>Result</th> - <th>Shortcut</th> - </tr> - </thead> - <tbody> - <tr> - <td>**text**</td> - <td><strong>Bold</strong></td> - <td>Ctrl/⌘ + B </td> - </tr> - <tr> - <td>*text*</td> - <td><em>Emphasize</em></td> - <td>Ctrl/⌘ + I</td> - </tr> - <tr> - <td>~~text~~</td> - <td><del>Strike-through</del></td> - <td>Ctrl + Alt + U</td> - </tr> - <tr> - <td>[title](http://)</td> - <td><a href="#">Link</a></td> - <td>Ctrl/⌘ + K</td> - </tr> - <tr> - <td>`code`</td> - <td><code>Inline Code</code></td> - <td>Ctrl/⌘ + Shift + K</td> - </tr> - <tr> - <td></td> - <td>Image</td> - <td>Ctrl/⌘ + Shift + I</td> - </tr> - <tr> - <td>* item</td> - <td>List</td> - <td>Ctrl + L</td> - </tr> - <tr> - <td>> quote</td> - <td>Blockquote</td> - <td>Ctrl + Q</td> - </tr> - <tr> - <td>==Highlight==</td> - <td><mark>Highlight</mark></td> - <td></td> - </tr> - <tr> - <td># Heading</td> - <td>H1</td> - <td></td> - </tr> - <tr> - <td>## Heading</td> - <td>H2</td> - <td>Ctrl/⌘ + H</td> - </tr> - <tr> - <td>### Heading</td> - <td>H3</td> - <td>Ctrl/⌘ + H (x2)</td> - </tr> - </tbody> - </table> - For further Markdown syntax reference: <a href="http://support.ghost.org/markdown-guide/" target="_blank">Markdown Documentation</a> - </section> -</div> diff --git a/core/client/app/templates/components/modals/new-subscriber.hbs b/core/client/app/templates/components/modals/new-subscriber.hbs deleted file mode 100644 index 7891c38536..0000000000 --- a/core/client/app/templates/components/modals/new-subscriber.hbs +++ /dev/null @@ -1,29 +0,0 @@ -<header class="modal-header"> - <h1>Add a Subscriber</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <fieldset> - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="email"}} - <label for="new-subscriber-email">Email Address</label> - <input type="email" - value={{model.email}} - oninput={{action "updateEmail" value="target.value"}} - id="new-subscriber-email" - class="gh-input email" - placeholder="Email Address" - name="email" - autofocus="autofocus" - autocapitalize="off" - autocorrect="off"> - {{gh-error-message errors=model.errors property="email"}} - {{/gh-form-group}} - - </fieldset> -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-green" submitting=submitting}}Add{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/re-authenticate.hbs b/core/client/app/templates/components/modals/re-authenticate.hbs deleted file mode 100644 index d5dcc0d8ef..0000000000 --- a/core/client/app/templates/components/modals/re-authenticate.hbs +++ /dev/null @@ -1,16 +0,0 @@ -<header class="modal-header"> - <h1>Please re-authenticate</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body {{if authenticationError 'error'}}"> - <form id="login" class="login-form" method="post" novalidate="novalidate" {{action "confirm" on="submit"}}> - {{#gh-validation-status-container class="password-wrap" errors=errors property="password" hasValidated=hasValidated}} - {{input class="gh-input password" type="password" placeholder="Password" name="password" value=password}} - {{/gh-validation-status-container}} - {{#gh-spin-button class="btn btn-blue" type="submit" submitting=submitting}}Log in{{/gh-spin-button}} - </form> - {{#if authenticationError}} - <p class="response">{{authenticationError}}</p> - {{/if}} -</div> diff --git a/core/client/app/templates/components/modals/transfer-owner.hbs b/core/client/app/templates/components/modals/transfer-owner.hbs deleted file mode 100644 index 6932b0251e..0000000000 --- a/core/client/app/templates/components/modals/transfer-owner.hbs +++ /dev/null @@ -1,16 +0,0 @@ -<header class="modal-header"> - <h1>Transfer Ownership</h1> -</header> -<a class="close icon-x" href="" title="Close" {{action "closeModal"}}><span class="hidden">Close</span></a> - -<div class="modal-body"> - <p> - Are you sure you want to transfer the ownership of this blog? - You will not be able to undo this action. - </p> -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-red" submitting=submitting}}Yep - I'm sure{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/components/modals/upload-image.hbs b/core/client/app/templates/components/modals/upload-image.hbs deleted file mode 100644 index 6b6fd2613e..0000000000 --- a/core/client/app/templates/components/modals/upload-image.hbs +++ /dev/null @@ -1,20 +0,0 @@ -<div class="modal-body"> - {{#if url}} - <div class="gh-image-uploader --with-image"> - <div><img src={{url}}></div> - <a class="image-cancel icon-trash" title="Delete" {{action 'removeImage'}}> - <span class="hidden">Delete</span> - </a> - </div> - {{else}} - {{gh-image-uploader image=newUrl - saveButton=false - update=(action 'fileUploaded') - onInput=(action (mut newUrl))}} - {{/if}} -</div> - -<div class="modal-footer"> - <button {{action "closeModal"}} class="btn btn-default btn-minor">Cancel</button> - {{#gh-spin-button action="confirm" class="btn btn-blue right js-button-accept" submitting=submitting}}Save{{/gh-spin-button}} -</div> diff --git a/core/client/app/templates/editor/edit.hbs b/core/client/app/templates/editor/edit.hbs deleted file mode 100644 index 42f254ae6e..0000000000 --- a/core/client/app/templates/editor/edit.hbs +++ /dev/null @@ -1,48 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title classNames="gh-editor-title" openMobileMenu="openMobileMenu"}} - {{gh-trim-focus-input type="text" id="entry-title" placeholder="Your Post Title" value=model.titleScratch tabindex="1" focus=shouldFocusTitle focus-out="updateTitle" }} - {{/gh-view-title}} - <section class="view-actions"> - <button type="button" class="post-settings" title="Post Settings" {{action "openSettingsMenu"}}> - <i class="icon-settings"></i> - </button> - {{gh-editor-save-button - isPublished=model.isPublished - willPublish=willPublish - postOrPage=postOrPage - isNew=model.isNew - save="save" - setSaveType="setSaveType" - delete="toggleDeletePostModal" - submitting=submitting - }} - </section> - </header> - - {{gh-editor value=model.scratch - shouldFocusEditor=shouldFocusEditor - previewUrl=model.previewUrl - editorFocused=(action "autoSaveNew") - onTeardown=(action "cancelTimers")}} -</section> - -{{#if showDeletePostModal}} - {{gh-fullscreen-modal "delete-post" - model=model - close=(action "toggleDeletePostModal") - modifier="action wide"}} -{{/if}} - -{{#if showLeaveEditorModal}} - {{gh-fullscreen-modal "leave-editor" - confirm=(action "leaveEditor") - close=(action "toggleLeaveEditorModal") - modifier="action wide"}} -{{/if}} - -{{#if showReAuthenticateModal}} - {{gh-fullscreen-modal "re-authenticate" - close=(action "toggleReAuthenticateModal") - modifier="action wide"}} -{{/if}} diff --git a/core/client/app/templates/error.hbs b/core/client/app/templates/error.hbs deleted file mode 100644 index 20b16f46c0..0000000000 --- a/core/client/app/templates/error.hbs +++ /dev/null @@ -1,27 +0,0 @@ -<div class="gh-view"> - <section class="error-content error-404 js-error-container"> - <section class="error-details"> - <img class="error-ghost" src="{{gh-path 'admin' '/img/404-ghost@2x.png'}}" srcset="{{gh-path 'admin' '/img/404-ghost.png'}} 1x, {{gh-path 'admin' '/img/404-ghost@2x.png'}} 2x" /> - <section class="error-message"> - <h1 class="error-code">{{code}}</h1> - <h2 class="error-description">{{message}}</h2> - </section> - </section> - </section> - - {{#if stack}} - <section class="error-stack"> - <h3>Stack Trace</h3> - <p><strong>{{message}}</strong></p> - <ul class="error-stack-list"> - {{#each stack as |item|}} - <li> - at - {{#if item.function}}<em class="error-stack-function">{{item.function}}</em>{{/if}} - <span class="error-stack-file">({{item.at}})</span> - </li> - {{/each}} - </ul> - </section> - {{/if}} -</div> diff --git a/core/client/app/templates/post-settings-menu.hbs b/core/client/app/templates/post-settings-menu.hbs deleted file mode 100644 index fb2f6704ba..0000000000 --- a/core/client/app/templates/post-settings-menu.hbs +++ /dev/null @@ -1,144 +0,0 @@ -{{#gh-tabs-manager selected="showSubview" id="entry-controls" class="settings-menu-container"}} -<div id="entry-controls"> - <div class="{{if isViewingSubview 'settings-menu-pane-out-left' 'settings-menu-pane-in'}} settings-menu settings-menu-pane"> - <div class="settings-menu-header"> - <h4>Post Settings</h4> - <button class="close icon-x settings-menu-header-action" {{action "closeMenus"}}><span class="hidden">Close</span></button> - </div> - <div class="settings-menu-content"> - {{gh-image-uploader-with-preview - image=model.image - text="Add post image" - update=(action "setCoverImage") - remove=(action "clearCoverImage")}} - <form> - <div class="form-group"> - <label for="url">Post URL</label> - {{#if model.isPublished}} - <a class="post-view-link" target="_blank" href="{{model.absoluteUrl}}"> - View post <i class="icon-external"></i> - </a> - {{else}} - <a class="post-view-link" target="_blank" href="{{model.previewUrl}}"> - Preview <i class="icon-external"></i> - </a> - {{/if}} - - <span class="input-icon icon-link"> - {{gh-input class="post-setting-slug" id="url" value=slugValue name="post-setting-slug" focus-out="updateSlug" stopEnterKeyDownPropagation="true"}} - </span> - {{gh-url-preview slug=slugValue tagName="p" classNames="description"}} - </div> - - {{#gh-form-group errors=model.errors property="post-setting-date"}} - <label for="post-setting-date">Publish Date</label> - {{gh-datetime-input value=model.publishedAt - update=(action "setPublishedAt") - inputClass="post-setting-date" - inputId="post-setting-date" - inputName="post-setting-date"}} - {{gh-error-message errors=model.errors property="post-setting-date"}} - {{/gh-form-group}} - - <div class="form-group"> - <label for="tag-input">Tags</label> - {{gh-selectize - id="tag-input" - multiple=true - selection=model.tags - content=availableTags - optionValuePath="content.uuid" - optionLabelPath="content.name" - openOnFocus=false - create-item="addTag" - remove-item="removeTag" - plugins="remove_button, drag_drop"}} - </div> - - {{#unless session.user.isAuthor}} - <div class="form-group for-select"> - <label for="author-list">Author</label> - <span class="input-icon icon-user"> - <span class="gh-select" tabindex="0"> - {{gh-select-native - name="post-setting-author" - id="author-list" - content=authors - optionValuePath="id" - optionLabelPath="name" - selection=selectedAuthor - action="changeAuthor" - }} - </span> - </span> - </div> - {{/unless}} - - <ul class="nav-list nav-list-block"> - {{#gh-tab tagName="li" classNames="nav-list-item"}} - <button type="button"> - <b>Meta Data</b> - <span>Extra content for SEO and social media.</span> - </button> - <i class="icon-arrow-right"></i> - {{/gh-tab}} - </ul> - - <div class="form-group for-checkbox"> - <label class="checkbox" for="static-page" {{action "togglePage" bubbles="false"}}> - {{input type="checkbox" name="static-page" id="static-page" class="gh-input post-setting-static-page" checked=model.page}} - <span class="input-toggle-component"></span> - <p>Turn this post into a static page</p> - </label> - - <label class="checkbox" for="featured" {{action "toggleFeatured" bubbles="false"}}> - {{input type="checkbox" name="featured" id="featured" class="gh-input post-setting-featured" checked=model.featured}} - <span class="input-toggle-component"></span> - <p>Feature this post</p> - </label> - </div> - - </form> - </div>{{! .settings-menu-content }} - </div>{{! .post-settings-menu }} - - <div class="{{if isViewingSubview 'settings-menu-pane-in' 'settings-menu-pane-out-right'}} settings-menu settings-menu-pane"> - {{#gh-tab-pane}} - {{#if isViewingSubview}} - <div class="settings-menu-header subview"> - <button {{action "closeSubview"}} class="back icon-arrow-left settings-menu-header-action"><span class="hidden">Back</span></button> - <h4>Meta Data</h4> - <div style="width:23px;">{{!flexbox space-between}}</div> - </div> - - <div class="settings-menu-content"> - <form {{action "discardEnter" on="submit"}}> - {{#gh-form-group errors=model.errors property="metaTitle"}} - <label for="meta-title">Meta Title</label> - {{gh-input class="post-setting-meta-title" id="meta-title" value=metaTitleScratch name="post-setting-meta-title" focus-out="setMetaTitle" stopEnterKeyDownPropagation="true"}} - <p>Recommended: <b>70</b> characters. You’ve used {{gh-count-down-characters metaTitleScratch 70}}</p> - {{gh-error-message errors=model.errors property="metaTitle"}} - {{/gh-form-group}} - - {{#gh-form-group errors=model.errors property="metaDescription"}} - <label for="meta-description">Meta Description</label> - {{gh-textarea class="gh-input post-setting-meta-description" id="meta-description" value=metaDescriptionScratch name="post-setting-meta-description" focus-out="setMetaDescription" stopEnterKeyDownPropagation="true"}} - <p>Recommended: <b>156</b> characters. You’ve used {{gh-count-down-characters metaDescriptionScratch 156}}</p> - {{gh-error-message errors=model.errors property="metaDescription"}} - {{/gh-form-group}} - - <div class="form-group"> - <label>Search Engine Result Preview</label> - <div class="seo-preview"> - <div class="seo-preview-title">{{seoTitle}}</div> - <div class="seo-preview-link">{{seoURL}}</div> - <div class="seo-preview-description">{{seoDescription}}</div> - </div> - </div> - </form> - </div>{{! .settings-menu-content }} - {{/if}} - {{/gh-tab-pane}} - </div> -</div> -{{/gh-tabs-manager}} diff --git a/core/client/app/templates/posts.hbs b/core/client/app/templates/posts.hbs deleted file mode 100644 index c42ae4e199..0000000000 --- a/core/client/app/templates/posts.hbs +++ /dev/null @@ -1,53 +0,0 @@ -{{#gh-content-view-container as |previewIsHidden|}} -<header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Content</span>{{/gh-view-title}} - <section class="view-actions"> - {{#link-to "editor.new" class="btn btn-green" title="New Post"}}New Post{{/link-to}} - </section> -</header> - -<div class="view-container"> - <section class="content-list js-content-list {{if postListFocused 'keyboard-focused'}}"> - {{#gh-infinite-scroll tagName="section" classNames="content-list-content js-content-scrollbox" fetch="loadNextPage" as |checkScroll|}} - <ol class="posts-list"> - {{#each sortedPosts key="id" as |post|}} - {{#gh-posts-list-item post=post onDoubleClick="openEditor" onDelete=(action checkScroll) as |component|}} - {{#link-to (if previewIsHidden 'editor.edit' 'posts.post') post.id class="permalink" title="Edit this post"}} - <h3 class="entry-title">{{post.title}}</h3> - <section class="entry-meta"> - <span class="avatar" style={{component.authorAvatarBackground}}> - <img src="{{component.authorAvatar}}" title="{{component.authorName}}"> - </span> - <span class="author">{{component.authorName}}</span> - <span class="status"> - {{#if component.isPublished}} - {{#if post.page}} - <span class="page">Page</span> - {{else}} - <time datetime="{{post.publishedAt}}" class="date published"> - Published {{gh-format-timeago post.publishedAt}} - </time> - {{/if}} - {{else}} - <span class="draft">Draft</span> - {{/if}} - </span> - </section> - {{/link-to}} - {{/gh-posts-list-item}} - {{/each}} - </ol> - {{/gh-infinite-scroll}} - </section> - <section class="content-preview js-content-preview {{if postContentFocused 'keyboard-focused'}}"> - {{outlet}} - </section> -</div> -{{/gh-content-view-container}} - -{{#if showDeletePostModal}} - {{gh-fullscreen-modal "delete-post" - model=currentPost - close=(action "toggleDeletePostModal") - modifier="action wide"}} -{{/if}} diff --git a/core/client/app/templates/posts/index.hbs b/core/client/app/templates/posts/index.hbs deleted file mode 100644 index 20c64ac37e..0000000000 --- a/core/client/app/templates/posts/index.hbs +++ /dev/null @@ -1,8 +0,0 @@ -<div class="no-posts-box"> - {{#if noPosts}} - <div class="no-posts"> - <h3>You Haven't Written Any Posts Yet!</h3> - {{#link-to "editor.new"}}<button type="button" class="btn btn-green btn-lg" title="New Post">Write a new Post</button>{{/link-to}} - </div> - {{/if}} -</div> diff --git a/core/client/app/templates/posts/post.hbs b/core/client/app/templates/posts/post.hbs deleted file mode 100644 index 7625f5d150..0000000000 --- a/core/client/app/templates/posts/post.hbs +++ /dev/null @@ -1,14 +0,0 @@ -<section class="post-controls"> - {{#link-to "editor.edit" model.id class="btn btn-minor post-edit" title="Edit this post"}}<i class="icon-edit"></i>{{/link-to}} -</section> - -{{#gh-content-preview-content tagName="section" content=model}} - <div class="wrapper"> - <h1 class="content-preview-title"> - {{#link-to "editor.edit" model.id}} - {{model.title}} - {{/link-to}} - </h1> - {{gh-format-html model.html}} - </div> -{{/gh-content-preview-content}} diff --git a/core/client/app/templates/reset.hbs b/core/client/app/templates/reset.hbs deleted file mode 100644 index 2d646a2796..0000000000 --- a/core/client/app/templates/reset.hbs +++ /dev/null @@ -1,18 +0,0 @@ -<div class="gh-flow"> - <div class="gh-flow-content-wrap"> - <section class="gh-flow-content fade-in"> - <form id="reset" class="gh-signin" method="post" novalidate="novalidate" {{action "submit" on="submit"}}> - {{#gh-form-group errors=errors hasValidated=hasValidated property="newPassword"}} - {{gh-input type="password" name="newpassword" placeholder="Password" class="password" autocorrect="off" autofocus="autofocus" value=newPassword}} - {{/gh-form-group}} - {{#gh-form-group errors=errors hasValidated=hasValidated property="ne2Password"}} - {{gh-input type="password" name="ne2password" placeholder="Confirm Password" class="password" autocorrect="off" autofocus="autofocus" value=ne2Password}} - {{/gh-form-group}} - - {{#gh-spin-button class="btn btn-blue btn-block" type="submit" submitting=submitting autoWidth="false"}}Reset Password{{/gh-spin-button}} - </form> - - <p class="main-error">{{{flowErrors}}}</p> - </section> - </div> -</div> diff --git a/core/client/app/templates/settings/apps.hbs b/core/client/app/templates/settings/apps.hbs deleted file mode 100644 index f7e76622da..0000000000 --- a/core/client/app/templates/settings/apps.hbs +++ /dev/null @@ -1,3 +0,0 @@ -<section class="gh-view"> - {{outlet}} -</section> diff --git a/core/client/app/templates/settings/apps/index.hbs b/core/client/app/templates/settings/apps/index.hbs deleted file mode 100644 index 065beb3445..0000000000 --- a/core/client/app/templates/settings/apps/index.hbs +++ /dev/null @@ -1,33 +0,0 @@ -<header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span style="padding-left:1px">Apps</span>{{/gh-view-title}} -</header> - -<section class="view-container"> - <section class="view-content"> - <span class="apps-grid-title">Available integrations</span> - <div class="apps-grid"> - <div class="apps-grid-cell"> - {{#link-to "settings.apps.slack" id="slack-link"}} - <article class="apps-card-app"> - <div class="apps-card-content"> - <figure class="apps-card-app-icon" style="background-image:url({{gh-path 'admin' '/img/slackicon.png'}})"></figure> - <div class="apps-card-meta"> - <h3 class="apps-card-app-title">Slack</h3> - <p class="apps-card-app-desc">A team communication tool</p> - </div> - <div class="apps-configured"> - {{#if slack.isActive}} - <span class="green">Active</span> - {{else}} - <span>Configure</span> - {{/if}} - <i class="icon-arrow-right"></i> - </div> - </div> - </article> - {{/link-to}} - </div> - </div> - <p class="apps-grid-note">(More coming soon!)</p> - </section> -</section> diff --git a/core/client/app/templates/settings/apps/slack.hbs b/core/client/app/templates/settings/apps/slack.hbs deleted file mode 100644 index 3decad5e1a..0000000000 --- a/core/client/app/templates/settings/apps/slack.hbs +++ /dev/null @@ -1,42 +0,0 @@ -<header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span style="padding-left:1px">{{#link-to "settings.apps.index"}}Apps{{/link-to}} <i class="icon-arrow-right" style="display:inline"></i> Slack</span>{{/gh-view-title}} - <section class="view-actions"> - {{#gh-spin-button id="saveSlackIntegration" class="btn btn-green" action=(action "save") submitting=isSaving}} - Save - {{/gh-spin-button}} - </section> -</header> -<section class="view-container"> - <section class="view-content"> - <section class="app-grid"> - <div class="app-cell"> - <img class="app-icon" src="{{gh-path 'admin' '/img/slackicon.png'}}" /> - </div> - <div class="app-cell"> - <h3>Slack</h3> - <p>A messaging app for teams</p> - </div> - </section> - <section class="app-subtitle"> - <p>Automatically send newly published posts to a channel in Slack.</p> - </section> - - <form class="app-config-form" id="slack-settings" novalidate="novalidate" {{action "save" on="submit"}}> - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="url"}} - <label for="url">Webhook URL</label> - {{one-way-input model.url class="gh-input" name="slack[url]" update=(action "updateURL") onenter=(action "save") placeholder="https://hooks.slack.com/services/..."}} - {{#unless model.errors.url}} - <p>Set up a new incoming webhook <a href="https://my.slack.com/apps/new/A0F7XDUAZ-incoming-webhooks" target="_blank">here</a>, and grab the URL.</p> - {{else}} - {{gh-error-message errors=model.errors property="url"}} - {{/unless}} - {{/gh-form-group}} - </form> - - <form class="app-config-form"> - {{#gh-spin-button id="sendTestNotification" class="btn btn-grey" disabled=testNotificationDisabled action=(action "sendTestNotification") submitting=isSendingTest}} - Send Test Notification - {{/gh-spin-button}} - </form> - </section> -</section> diff --git a/core/client/app/templates/settings/code-injection.hbs b/core/client/app/templates/settings/code-injection.hbs deleted file mode 100644 index 6e8d86ee64..0000000000 --- a/core/client/app/templates/settings/code-injection.hbs +++ /dev/null @@ -1,30 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Code Injection</span>{{/gh-view-title}} - <section class="view-actions"> - {{#gh-spin-button class="btn btn-blue" action="save" submitting=submitting}}Save{{/gh-spin-button}} - </section> - </header> - - <section class="view-content"> - <form id="settings-code" novalidate="novalidate"> - <fieldset> - <p> - Ghost allows you to inject code into the top and bottom of your theme files without editing them. This allows for quick modifications to insert useful things like tracking codes and meta tags. - </p> - - <div class="form-group settings-code"> - <label for="ghost-head">Blog Header</label> - <p>Code here will be injected into the <code>\{{ghost_head}}</code> tag on every page of your blog</p> - {{gh-cm-editor id="ghost-head" class="gh-input settings-code-editor" name="codeInjection[ghost_head]" type="text" value=model.ghost_head}} - </div> - - <div class="form-group settings-code"> - <label for="ghost-foot">Blog Footer</label> - <p>Code here will be injected into the <code>\{{ghost_foot}}</code> tag on every page of your blog</p> - {{gh-cm-editor id="ghost-foot" class="gh-input settings-code-editor" name="codeInjection[ghost_foot]" type="text" value=model.ghost_foot}} - </div> - </fieldset> - </form> - </section> -</section> diff --git a/core/client/app/templates/settings/general.hbs b/core/client/app/templates/settings/general.hbs deleted file mode 100644 index 495791a206..0000000000 --- a/core/client/app/templates/settings/general.hbs +++ /dev/null @@ -1,137 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>General</span>{{/gh-view-title}} - <section class="view-actions"> - {{#gh-spin-button class="btn btn-blue" action="save" submitting=submitting}}Save{{/gh-spin-button}} - </section> - </header> - - <section class="view-content"> - <form id="settings-general" novalidate="novalidate"> - <fieldset> - - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="title"}} - <label for="blog-title">Blog Title</label> - {{gh-input id="blog-title" class="gh-input" name="general[title]" type="text" value=model.title focusOut=(action "validate" "title" target=model)}} - {{gh-error-message errors=model.errors property="title"}} - <p>The name of your blog</p> - {{/gh-form-group}} - - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="description" class="description-container"}} - <label for="blog-description">Blog Description</label> - {{gh-textarea id="blog-description" class="gh-input" name="general[description]" value=model.description focusOut=(action "validate" "description" target=model)}} - {{gh-error-message errors=model.errors property="description"}} - <p> - Describe what your blog is about - {{gh-count-characters model.description}} - </p> - {{/gh-form-group}} - </fieldset> - - <div class="form-group"> - <label>Blog Logo</label> - {{#if model.logo}} - <img class="blog-logo" src="{{model.logo}}" alt="logo" role="button" {{action "toggleUploadLogoModal"}}> - {{else}} - <button type="button" class="btn btn-green js-modal-logo" {{action "toggleUploadLogoModal"}}>Upload Image</button> - {{/if}} - <p>Display a logo for your publication</p> - - {{#if showUploadLogoModal}} - {{gh-fullscreen-modal "upload-image" - model=(hash model=model imageProperty="logo") - close=(action "toggleUploadLogoModal") - modifier="action wide"}} - {{/if}} - </div> - - <div class="form-group"> - <label>Blog Cover</label> - {{#if model.cover}} - <img class="blog-cover" src="{{model.cover}}" alt="cover photo" role="button" {{action "toggleUploadCoverModal"}}> - {{else}} - <button type="button" class="btn btn-green js-modal-cover" {{action "toggleUploadCoverModal"}}>Upload Image</button> - {{/if}} - <p>Display a cover image on your site</p> - - {{#if showUploadCoverModal}} - {{gh-fullscreen-modal "upload-image" - model=(hash model=model imageProperty="cover") - close=(action "toggleUploadCoverModal") - modifier="action wide"}} - {{/if}} - </div> - - <fieldset> - - <div class="form-group"> - <label for="postsPerPage">Posts per page</label> - {{! `pattern` brings up numeric keypad allowing any number of digits}} - {{gh-input id="postsPerPage" class="gh-input" name="general[postsPerPage]" focus-out="checkPostsPerPage" value=model.postsPerPage min="1" max="1000" type="number" pattern="[0-9]*"}} - <p>How many posts should be displayed on each page</p> - </div> - - <div class="form-group for-checkbox"> - <label for="permalinks">Dated Permalinks</label> - <label class="checkbox" for="permalinks"> - {{input id="permalinks" class="gh-input" name="general[permalinks]" type="checkbox" checked=isDatedPermalinks}} - <span class="input-toggle-component"></span> - <p>Include the date in your post URLs</p> - </label> - </div> - - <div class="form-group for-select"> - <label for="activeTheme">Theme</label> - <span class="gh-select" data-select-text="{{selectedTheme.label}}" tabindex="0"> - {{gh-select-native - id="activeTheme" - name="general[activeTheme]" - content=themes - optionValuePath="name" - optionLabelPath="label" - selection=selectedTheme - action="setTheme" - }} - </span> - <p>Select a theme for your blog</p> - </div> - - <div class="form-group"> - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="facebook"}} - <label for="facebook">Facebook Page</label> - <input value={{model.facebook}} oninput={{action (mut _scratchFacebook) value="target.value"}} {{action "validateFacebookUrl" on="focusOut"}} type="url" class="gh-input" id="facebook" name="general[facebook]" placeholder="https://www.facebook.com/ghost" autocorrect="off" /> - {{gh-error-message errors=model.errors property="facebook"}} - <p>URL of your blog's Facebook Page</p> - {{/gh-form-group}} - </div> - <div class="form-group"> - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="twitter"}} - <label for="twitter">Twitter Profile</label> - <input value={{model.twitter}} oninput={{action (mut _scratchTwitter) value="target.value"}} {{action "validateTwitterUrl" on="focusOut"}} type="url" class="gh-input" id="facebook" name="general[twitter]" placeholder="https://twitter.com/tryghost" autocorrect="off" /> - {{gh-error-message errors=model.errors property="twitter"}} - <p>URL of your blog's Twitter profile</p> - {{/gh-form-group}} - </div> - - <div class="form-group for-checkbox"> - <label for="isPrivate">Make this blog private</label> - <label class="checkbox" for="isPrivate"> - {{input id="isPrivate" name="general[isPrivate]" type="checkbox" - checked=model.isPrivate}} - <span class="input-toggle-component"></span> - <p>Enable password protection</p> - </label> - </div> - - {{#if model.isPrivate}} - {{#gh-form-group errors=model.errors hasValidated=model.hasValidated property="password"}} - {{gh-input name="general[password]" type="text" value=model.password focusOut=(action "validate" "password" target=model)}} - {{gh-error-message errors=model.errors property="password"}} - <p>This password will be needed to access your blog. All search engine optimization and social features are now disabled. This password is stored in plaintext.</p> - {{/gh-form-group}} - {{/if}} - </fieldset> - - </form> - </section> -</section> diff --git a/core/client/app/templates/settings/labs.hbs b/core/client/app/templates/settings/labs.hbs deleted file mode 100644 index df721163fe..0000000000 --- a/core/client/app/templates/settings/labs.hbs +++ /dev/null @@ -1,66 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Labs</span>{{/gh-view-title}} - </header> - - <section class="view-content settings-debug"> - <p><strong>Important note:</strong> Labs is a testing ground for experimental features which aren't quite ready for primetime. They may change, break or inexplicably disappear at any time.</p> - <form id="settings-export"> - <fieldset> - <div class="form-group"> - <label>Export</label> - <button type="button" class="btn btn-blue" {{action "exportData"}}>Export</button> - <p>Export the blog settings and data.</p> - </div> - </fieldset> - </form> - <form id="settings-import" enctype="multipart/form-data"> - <fieldset> - <div class="form-group"> - <label>Import</label> - {{partial "import-errors"}} - {{gh-file-upload id="importfile" classNames="flex" uploadButtonText=uploadButtonText onUpload="onUpload"}} - <p>Import from another Ghost installation. If you import a user, this will replace the current user & log you out.</p> - </div> - </fieldset> - </form> - <form id="settings-resetdb"> - <fieldset> - <div class="form-group"> - <label>Delete all Content</label> - <button type="button" class="btn btn-red js-delete" {{action "toggleDeleteAllModal"}}>Delete</button> - <p>Delete all posts and tags from the database.</p> - </div> - </fieldset> - </form> - <form id="settings-testmail"> - <fieldset> - <div class="form-group"> - <label>Send a test email</label> - {{#gh-spin-button id="sendtestemail" class="btn btn-blue" action="sendTestEmail" submitting=submitting}}Send{{/gh-spin-button}} - <p>Sends a test email to your address.</p> - </div> - </fieldset> - </form> - <hr> - <form> - <fieldset> - <div class="form-group for-checkbox"> - <h3>Enable Beta Features</h3> - {{#gh-feature-flag "publicAPI"}} - Public API - For full instructions, read the <a href="http://support.ghost.org/public-api-beta/">developer guide</a>. - {{/gh-feature-flag}} - {{#gh-feature-flag "subscribers"}} - Subscribers - Allow visitors to subscribe to e-mail updates of your new posts - {{/gh-feature-flag}} - </div> - </fieldset> - </form> - </section> -</section> - -{{#if showDeleteAllModal}} - {{gh-fullscreen-modal "delete-all" - close=(action "toggleDeleteAllModal") - modifier="action wide"}} -{{/if}} diff --git a/core/client/app/templates/settings/navigation.hbs b/core/client/app/templates/settings/navigation.hbs deleted file mode 100644 index 9426fafda1..0000000000 --- a/core/client/app/templates/settings/navigation.hbs +++ /dev/null @@ -1,19 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Navigation</span>{{/gh-view-title}} - <section class="view-actions"> - {{#gh-spin-button class="btn btn-blue" action="save" submitting=submitting}}Save{{/gh-spin-button}} - </section> - </header> - - <section class="view-container"> - <form id="settings-navigation" class="gh-blognav" novalidate="novalidate"> - {{#sortable-group onChange=(action 'reorderItems') as |group|}} - {{#each model.navigation as |navItem|}} - {{gh-navitem navItem=navItem baseUrl=blogUrl addItem="addItem" deleteItem="deleteItem" updateUrl="updateUrl" group=group}} - {{/each}} - {{/sortable-group}} - {{gh-navitem navItem=newNavItem baseUrl=blogUrl addItem="addItem" updateUrl="updateUrl"}} - </form> - </section> -</section> diff --git a/core/client/app/templates/settings/tags.hbs b/core/client/app/templates/settings/tags.hbs deleted file mode 100644 index f67750a544..0000000000 --- a/core/client/app/templates/settings/tags.hbs +++ /dev/null @@ -1,27 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Tags</span>{{/gh-view-title}} - <section class="view-actions"> - {{#link-to "settings.tags.new" class="btn btn-green" title="New Tag"}}New Tag{{/link-to}} - {{!-- <button type="button" class="btn btn-green" {{action "newTag"}}>New Tag</button> --}} - </section> - </header> - - {{#gh-tags-management-container tags=tags selectedTag=selectedTag enteredMobile="enteredMobile" leftMobile="leftMobile" as |container|}} - {{#gh-infinite-scroll - fetch="loadNextPage" - isLoading=isLoading - classNames="tag-list" - as |checkScroll| - }} - <section class="tag-list-content settings-tags {{if tagListFocused 'keyboard-focused'}}"> - {{#each tags as |tag|}} - {{gh-tag tag=tag onDelete=(action checkScroll)}} - {{/each}} - </section> - {{/gh-infinite-scroll}} - <section class="settings-menu-container tag-settings {{if tagContentFocused 'keyboard-focused'}} {{if container.displaySettingsPane 'tag-settings-in'}}"> - {{outlet}} - </section> - {{/gh-tags-management-container}} -</section> diff --git a/core/client/app/templates/settings/tags/index.hbs b/core/client/app/templates/settings/tags/index.hbs deleted file mode 100644 index 42f93a98e4..0000000000 --- a/core/client/app/templates/settings/tags/index.hbs +++ /dev/null @@ -1,6 +0,0 @@ -<div class="no-posts-box"> - <div class="no-posts"> - <h3>You haven't added any Tags yet!</h3> - {{#link-to "settings.tags.new"}}<button type="button" class="btn btn-green btn-lg" title="New Tag">Add a Tag</button>{{/link-to}} - </div> -</div> diff --git a/core/client/app/templates/settings/tags/tag.hbs b/core/client/app/templates/settings/tags/tag.hbs deleted file mode 100644 index ccbfb121a8..0000000000 --- a/core/client/app/templates/settings/tags/tag.hbs +++ /dev/null @@ -1,11 +0,0 @@ -{{gh-tag-settings-form tag=tag - setProperty=(action "setProperty") - showDeleteTagModal=(action "toggleDeleteTagModal")}} - -{{#if showDeleteTagModal}} - {{gh-fullscreen-modal "delete-tag" - model=tag - confirm=(action "deleteTag") - close=(action "toggleDeleteTagModal") - modifier="action wide"}} -{{/if}} diff --git a/core/client/app/templates/setup.hbs b/core/client/app/templates/setup.hbs deleted file mode 100644 index a0153448ee..0000000000 --- a/core/client/app/templates/setup.hbs +++ /dev/null @@ -1,27 +0,0 @@ -<div class="gh-flow"> - <header class="gh-flow-head"> - <nav class="gh-flow-nav"> - {{#if showBackLink}} - {{#link-to backRoute classNames="gh-flow-back"}}<i class="icon-arrow-left"></i> Back{{/link-to}} - {{/if}} - <ol> - {{#gh-activating-list-item route="setup.one" linkClasses="step"}} - <i class="icon-check"></i><span class="num">1</span> - {{/gh-activating-list-item}} - <li class="divider"></li> - {{#gh-activating-list-item route="setup.two" linkClasses="step"}} - <i class="icon-check"></i><span class="num">2</span> - {{/gh-activating-list-item}} - <li class="divider"></li> - {{#gh-activating-list-item route="setup.three" linkClasses="step"}} - <i class="icon-check"></i><span class="num">3</span> - {{/gh-activating-list-item}} - </ol> - </nav> - </header> - <div class="gh-flow-content-wrap"> - <section class="gh-flow-content"> - {{outlet}} - </section> - </div> -</div> diff --git a/core/client/app/templates/setup/one.hbs b/core/client/app/templates/setup/one.hbs deleted file mode 100644 index 9081de39ca..0000000000 --- a/core/client/app/templates/setup/one.hbs +++ /dev/null @@ -1,12 +0,0 @@ -<header> - <h1>Welcome to <strong>Ghost</strong>!</h1> - <p>All over the world, people have started <em>{{model.count}}</em> incredible blogs with Ghost. Today, we’re starting yours.</p> -</header> - -<figure class="gh-flow-screenshot"> - <img src="{{gh-path 'admin' 'img/install-welcome.png'}}" alt="Ghost screenshot" /> -</figure> - -{{#link-to "setup.two" classNames="btn btn-green btn-lg"}} - Create your account <i class="icon-chevron"></i> -{{/link-to}} diff --git a/core/client/app/templates/setup/three.hbs b/core/client/app/templates/setup/three.hbs deleted file mode 100644 index 5a5c09e706..0000000000 --- a/core/client/app/templates/setup/three.hbs +++ /dev/null @@ -1,19 +0,0 @@ -<header> - <h1>Invite your team</h1> - <p>Ghost works best when shared with others. Collaborate, get feedback on your posts & work together on ideas.</p> -</header> - -<div><img class="gh-flow-faces" src="{{gh-path 'admin' 'img/users.png'}}" alt="" /></div> - -<form class="gh-flow-invite"> - {{#gh-form-group errors=errors hasValidated=hasValidated property="users"}} - <label for="users">Enter one email address per line, we’ll handle the rest! <i class="icon-mail"></i></label> - {{gh-textarea name="users" value=users required="required" focusOut=(action "validate")}} - {{/gh-form-group}} - - {{#gh-spin-button type="submit" action="invite" classNameBindings=":btn :btn-default :btn-lg :btn-block buttonClass" submitting=submitting autoWidth="false"}}{{buttonText}}{{/gh-spin-button}} -</form> - -<button class="gh-flow-skip" {{action "skipInvite"}}> - I'll do this later, take me to my blog! -</button> diff --git a/core/client/app/templates/setup/two.hbs b/core/client/app/templates/setup/two.hbs deleted file mode 100644 index 8c53c6b02f..0000000000 --- a/core/client/app/templates/setup/two.hbs +++ /dev/null @@ -1,44 +0,0 @@ -<header> - <h1>Create your account</h1> -</header> - -<form id="setup" class="gh-flow-create"> - {{!-- Horrible hack to prevent Chrome from incorrectly auto-filling inputs --}} - <input style="display:none;" type="text" name="fakeusernameremembered"/> - <input style="display:none;" type="password" name="fakepasswordremembered"/> - - {{gh-profile-image fileStorage=config.fileStorage email=email setImage="setImage"}} - {{#gh-form-group errors=errors hasValidated=hasValidated property="email"}} - <label for="email-address">Email address</label> - <span class="input-icon icon-mail"> - {{gh-trim-focus-input tabindex="1" type="email" name="email" placeholder="Eg. john@example.com" autocorrect="off" value=email focusOut=(action "preValidate" "email")}} - </span> - {{gh-error-message errors=errors property="email"}} - {{/gh-form-group}} - {{#gh-form-group errors=errors hasValidated=hasValidated property="name"}} - <label for="full-name">Full name</label> - <span class="input-icon icon-user"> - {{gh-input tabindex="2" type="text" name="name" placeholder="Eg. John H. Watson" autocorrect="off" value=name focusOut=(action "preValidate" "name")}} - </span> - {{gh-error-message errors=errors property="name"}} - {{/gh-form-group}} - {{#gh-form-group errors=errors hasValidated=hasValidated property="password"}} - <label for="password">Password</label> - <span class="input-icon icon-lock"> - {{gh-input tabindex="3" type="password" name="password" placeholder="At least 8 characters" autocorrect="off" value=password focusOut=(action "preValidate" "password")}} - </span> - {{gh-error-message errors=errors property="password"}} - {{/gh-form-group}} - {{#gh-form-group errors=errors hasValidated=hasValidated property="blogTitle"}} - <label for="blog-title">Blog title</label> - <span class="input-icon icon-content"> - {{gh-input tabindex="4" type="text" name="blog-title" placeholder="Eg. The Daily Awesome" autocorrect="off" value=blogTitle focusOut=(action "preValidate" "blogTitle")}} - </span> - {{gh-error-message errors=errors property="blogTitle"}} - {{/gh-form-group}} - {{#gh-spin-button type="submit" tabindex="5" class="btn btn-green btn-lg btn-block" action="setup" submitting=submitting autoWidth="false"}} - Last step: Invite your team <i class="icon-chevron"></i> - {{/gh-spin-button}} -</form> - -<p class="main-error">{{{flowErrors}}}</p> diff --git a/core/client/app/templates/signin.hbs b/core/client/app/templates/signin.hbs deleted file mode 100644 index 009df68653..0000000000 --- a/core/client/app/templates/signin.hbs +++ /dev/null @@ -1,22 +0,0 @@ -<div class="gh-flow"> - <div class="gh-flow-content-wrap"> - <section class="gh-flow-content"> - <form id="login" class="gh-signin" method="post" novalidate="novalidate"> - {{#gh-form-group errors=model.errors hasValidated=hasValidated property="identification"}} - <span class="input-icon icon-mail"> - {{gh-trim-focus-input class="gh-input email" type="email" placeholder="Email Address" name="identification" autocapitalize="off" autocorrect="off" tabindex="1" focusOut=(action "validate" "identification") value=model.identification}} - </span> - {{/gh-form-group}} - {{#gh-form-group errors=model.errors hasValidated=hasValidated property="password"}} - <span class="input-icon icon-lock forgotten-wrap"> - {{gh-input class="password" type="password" placeholder="Password" name="password" tabindex="2" value=model.password autocorrect="off"}} - {{#gh-spin-button class="forgotten-link btn btn-link" type="button" action="forgotten" tabindex="4" submitting=submitting autoWidth="true"}}Forgot?{{/gh-spin-button}} - </span> - {{/gh-form-group}} - {{#gh-spin-button class="login btn btn-blue btn-block" type="submit" action="validateAndAuthenticate" tabindex="3" submitting=loggingIn autoWidth="false"}}Sign in{{/gh-spin-button}} - </form> - - <p class="main-error">{{{flowErrors}}}</p> - </section> - </div> -</div> diff --git a/core/client/app/templates/signup.hbs b/core/client/app/templates/signup.hbs deleted file mode 100644 index c30e91bfc3..0000000000 --- a/core/client/app/templates/signup.hbs +++ /dev/null @@ -1,43 +0,0 @@ -<div class="gh-flow"> - - <div class="gh-flow-content-wrap"> - <section class="gh-flow-content"> - <header> - <h1>Create your account</h1> - </header> - - <form id="signup" class="gh-flow-create" method="post" novalidate="novalidate"> - {{!-- Hack to stop Chrome's broken auto-fills --}} - <input style="display:none;" type="text" name="fakeusernameremembered"/> - <input style="display:none;" type="password" name="fakepasswordremembered"/> - - {{gh-profile-image fileStorage=config.fileStorage email=model.email setImage="setImage"}} - {{#gh-form-group errors=model.errors hasValidated=hasValidated property="email"}} - <label for="email-address">Email address</label> - <span class="input-icon icon-mail"> - {{gh-input type="email" name="email" placeholder="Eg. john@example.com" enter=(action "signup") disabled="disabled" autocorrect="off" value=model.email focusOut=(action "validate" "email")}} - </span> - {{gh-error-message errors=model.errors property="email"}} - {{/gh-form-group}} - {{#gh-form-group errors=model.errors hasValidated=hasValidated property="name"}} - <label for="full-name">Full name</label> - <span class="input-icon icon-user"> - {{gh-trim-focus-input tabindex="1" type="text" name="name" placeholder="Eg. John H. Watson" enter=(action "signup") autocorrect="off" value=model.name focusOut=(action "validate" "name")}} - </span> - {{gh-error-message errors=model.errors property="name"}} - {{/gh-form-group}} - {{#gh-form-group errors=model.errors hasValidated=hasValidated property="password"}} - <label for="password">Password</label> - <span class="input-icon icon-lock"> - {{gh-input tabindex="2" type="password" name="password" enter=(action "signup") autocorrect="off" value=model.password focusOut=(action "validate" "password")}} - </span> - {{gh-error-message errors=model.errors property="password"}} - {{/gh-form-group}} - </form> - - {{#gh-spin-button tabindex="3" type="submit" class="btn btn-green btn-lg btn-block" action="signup" submitting=submitting autoWidth="false"}}Create Account{{/gh-spin-button}} - <p class="main-error">{{{flowErrors}}}</p> - </section> - </div> - -</div> diff --git a/core/client/app/templates/subscribers.hbs b/core/client/app/templates/subscribers.hbs deleted file mode 100644 index b3bd45ee39..0000000000 --- a/core/client/app/templates/subscribers.hbs +++ /dev/null @@ -1,49 +0,0 @@ -<section class="gh-view view-subscribers"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Subscribers</span>{{/gh-view-title}} - <div class="view-actions"> - {{#link-to "subscribers.new" class="btn btn-green"}}Add Subscriber{{/link-to}} - </div> - </header> - - <section class="view-container"> - {{gh-subscribers-table - table=table - isLoading=isLoading - loadNextPage=(action 'loadNextPage') - sortByColumn=(action 'sortByColumn') - delete=(action 'deleteSubscriber')}} - - <div class="subscribers-sidebar"> - <div class="settings-menu-header"> - <h4>Import Subscribers</h4> - </div> - <div class="settings-menu-content subscribers-import-buttons"> - {{#link-to "subscribers.import" class="btn btn-hover-green"}}Import CSV{{/link-to}} - <a {{action 'exportData'}} class="btn">Export CSV</a> - </div> - - <div class="settings-menu-header"> - <h4>Quick Stats</h4> - </div> - <div class="settings-menu-content"> - <ul> - <li> - Total Subscribers: - <span id="total-subscribers">{{total}}</span> - </li> - </ul> - </div> - </div> - </section> -</section> - -{{#if subscriberToDelete}} - {{gh-fullscreen-modal "delete-subscriber" - model=subscriberToDelete - confirm=(action "confirmDeleteSubscriber") - close=(action "cancelDeleteSubscriber") - modifier="action wide"}} -{{/if}} - -{{outlet}} diff --git a/core/client/app/templates/subscribers/import.hbs b/core/client/app/templates/subscribers/import.hbs deleted file mode 100644 index 35ede97005..0000000000 --- a/core/client/app/templates/subscribers/import.hbs +++ /dev/null @@ -1,3 +0,0 @@ -{{gh-fullscreen-modal "import-subscribers" - confirm=(route-action "reset") - close=(route-action "cancel")}} diff --git a/core/client/app/templates/subscribers/new.hbs b/core/client/app/templates/subscribers/new.hbs deleted file mode 100644 index 46de448405..0000000000 --- a/core/client/app/templates/subscribers/new.hbs +++ /dev/null @@ -1,4 +0,0 @@ -{{gh-fullscreen-modal "new-subscriber" - model=model - confirm=(route-action "save") - close=(route-action "cancel")}} diff --git a/core/client/app/templates/team/index.hbs b/core/client/app/templates/team/index.hbs deleted file mode 100644 index cd8e3f697a..0000000000 --- a/core/client/app/templates/team/index.hbs +++ /dev/null @@ -1,82 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}}<span>Team</span>{{/gh-view-title}} - {{!-- Do not show Invite user button to authors --}} - {{#unless session.user.isAuthor}} - <section class="view-actions"> - <button class="btn btn-green" {{action "toggleInviteUserModal"}} >Invite People</button> - </section> - - {{#if showInviteUserModal}} - {{gh-fullscreen-modal "invite-new-user" - close=(action "toggleInviteUserModal") - modifier="action"}} - {{/if}} - {{/unless}} - </header> - - {{#gh-infinite-scroll - fetch="loadNextPage" - isLoading=isLoading - tagName="section" - classNames="view-content team" - }} - {{!-- Do not show invited users to authors --}} - {{#unless session.user.isAuthor}} - {{#if invitedUsers}} - <section class="user-list invited-users"> - <h4 class="user-list-title">Invited users</h4> - {{#each invitedUsers as |user|}} - {{#gh-user-invited user=user reload="reload" as |component|}} - <div class="user-list-item"> - <span class="user-list-item-icon icon-mail">ic</span> - <div class="user-list-item-body"> - <span class="name">{{user.email}}</span><br> - {{#if user.pending}} - <span class="description-error"> - Invitation not sent - please try again - </span> - {{else}} - <span class="description"> - Invitation sent: {{component.createdAt}} - </span> - {{/if}} - </div> - <aside class="user-list-item-aside"> - {{#if component.isSending}} - <span>Sending Invite...</span> - {{else}} - <a class="user-list-action" href="#" {{action "revoke" target=component}}> - Revoke - </a> - <a class="user-list-action" href="#" {{action "resend" target=component}}> - Resend - </a> - {{/if}} - </aside> - </div> - {{/gh-user-invited}} - {{/each}} - </section> - {{/if}} - {{/unless}} - - <section class="user-list active-users"> - <h4 class="user-list-title">Active users</h4> - {{#each activeUsers key="id" as |user|}} - {{!-- For authors only shows users as a list, otherwise show users with links to user page --}} - {{#unless session.user.isAuthor}} - {{#gh-user-active user=user as |component|}} - {{#link-to 'team.user' user.slug class="user-list-item"}} - {{partial 'user-list-item'}} - {{/link-to}} - {{/gh-user-active}} - {{else}} - {{#gh-user-active user=user as |component|}} - <li class="ember-view active user-list-item">{{partial 'user-list-item'}}</li> - {{/gh-user-active}} - {{/unless}} - {{/each}} - </section> - {{/gh-infinite-scroll}} -</section> diff --git a/core/client/app/templates/team/user.hbs b/core/client/app/templates/team/user.hbs deleted file mode 100644 index a9db1c4206..0000000000 --- a/core/client/app/templates/team/user.hbs +++ /dev/null @@ -1,197 +0,0 @@ -<section class="gh-view"> - <header class="view-header"> - {{#gh-view-title openMobileMenu="openMobileMenu"}} - {{link-to "Team" "team"}} - <i class="icon-arrow-right"></i> <span>{{user.name}}</span> - {{/gh-view-title}} - <section class="view-actions"> - {{#if userActionsAreVisible}} - <span class="dropdown"> - {{#gh-dropdown-button dropdownName="user-actions-menu" classNames="btn btn-default only-has-icon user-actions-cog" title="User Actions"}} - <i class="icon-settings"></i> - <span class="hidden">User Settings</span> - {{/gh-dropdown-button}} - {{#gh-dropdown name="user-actions-menu" tagName="ul" classNames="user-actions-menu dropdown-menu dropdown-triangle-top-right"}} - {{#if canMakeOwner}} - <li> - <button {{action "toggleTransferOwnerModal"}}> - Make Owner - </button> - {{#if showTransferOwnerModal}} - {{gh-fullscreen-modal "transfer-owner" - confirm=(action "transferOwnership") - close=(action "toggleTransferOwnerModal") - modifier="action wide"}} - {{/if}} - </li> - {{/if}} - {{#if deleteUserActionIsVisible}} - <li> - <button {{action "toggleDeleteUserModal"}} class="delete"> - Delete User - </button> - {{#if showDeleteUserModal}} - {{gh-fullscreen-modal "delete-user" - confirm=(action "deleteUser") - close=(action "toggleDeleteUserModal") - modifier="action wide"}} - {{/if}} - </li> - {{/if}} - {{/gh-dropdown}} - </span> - {{/if}} - - {{#gh-spin-button class="btn btn-blue" action="save" submitting=submitting}}Save{{/gh-spin-button}} - </section> - </header> - - <div class="view-container settings-user"> - - <figure class="user-cover" style={{coverImageBackground}}> - <button class="btn btn-default user-cover-edit" {{action "toggleUploadCoverModal"}}>Change Cover</button> - {{#if showUploadCoverModal}} - {{gh-fullscreen-modal "upload-image" - model=(hash model=user imageProperty="cover") - close=(action "toggleUploadCoverModal") - modifier="action wide"}} - {{/if}} - </figure> - - <form class="user-profile" novalidate="novalidate" autocomplete="off"> - - {{!-- Horrible hack to prevent Chrome from incorrectly auto-filling inputs --}} - <input style="display:none;" type="text" name="fakeusernameremembered"/> - <input style="display:none;" type="password" name="fakepasswordremembered"/> - - <fieldset class="user-details-top"> - - <figure class="user-image"> - <div id="user-image" class="img" style={{userImageBackground}}><span class="hidden">{{user.name}}"s Picture</span></div> - <button type="button" {{action "toggleUploadImageModal"}} class="edit-user-image">Edit Picture</button> - {{#if showUploadImageModal}} - {{gh-fullscreen-modal "upload-image" - model=(hash model=user imageProperty="image") - close=(action "toggleUploadImageModal") - modifier="action wide"}} - {{/if}} - </figure> - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="name" class="first-form-group"}} - <label for="user-name">Full Name</label> - {{input value=user.name id="user-name" class="gh-input user-name" placeholder="Full Name" autocorrect="off" focusOut=(action "validate" "name" target=user)}} - {{#if user.errors.name}} - {{gh-error-message errors=user.errors property="name"}} - {{else}} - <p>Use your real name so people can recognise you</p> - {{/if}} - {{/gh-form-group}} - - </fieldset> - - <fieldset class="user-details-bottom"> - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="slug"}} - <label for="user-slug">Slug</label> - {{gh-input class="gh-input user-name" id="user-slug" value=slugValue name="user" focus-out="updateSlug" placeholder="Slug" selectOnClick="true" autocorrect="off"}} - <p>{{gh-blog-url}}/author/{{slugValue}}</p> - {{gh-error-message errors=user.errors property="slug"}} - {{/gh-form-group}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="email"}} - <label for="user-email">Email</label> - {{!-- Administrators only see text of Owner's email address but not input --}} - {{#unless isAdminUserOnOwnerProfile}} - {{input type="email" value=user.email id="user-email" name="email" class="gh-input" placeholder="Email Address" autocapitalize="off" autocorrect="off" autocomplete="off" focusOut=(action "validate" "email" target=user)}} - {{gh-error-message errors=user.errors property="email"}} - {{else}} - <span>{{user.email}}</span> - {{/unless}} - <p>Used for notifications</p> - {{/gh-form-group}} - - {{#if rolesDropdownIsVisible}} - <div class="form-group"> - <label for="user-role">Role</label> - <span class="gh-select" tabindex="0"> - {{gh-select-native id="new-user-role" - content=roles - optionValuePath="id" - optionLabelPath="name" - selection=model.role - action="changeRole" - }} - </span> - <p>What permissions should this user have?</p> - </div> - {{/if}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="location"}} - <label for="user-location">Location</label> - {{input type="text" value=user.location id="user-location" class="gh-input" focusOut=(action "validate" "location" target=user)}} - {{gh-error-message errors=user.errors property="location"}} - <p>Where in the world do you live?</p> - {{/gh-form-group}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="website"}} - <label for="user-website">Website</label> - {{input type="url" value=user.website id="user-website" class="gh-input" autocapitalize="off" autocorrect="off" autocomplete="off" focusOut=(action "validate" "website" target=user)}} - {{gh-error-message errors=user.errors property="website"}} - <p>Have a website or blog other than this one? Link it!</p> - {{/gh-form-group}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="facebook"}} - <label for="user-facebook">Facebook Profile</label> - <input value={{user.facebook}} oninput={{action (mut _scratchFacebook) value="target.value"}} {{action "validateFacebookUrl" on="focusOut"}} type="url" class="gh-input" id="user-facebook" name="user[facebook]" placeholder="https://www.facebook.com/username" autocorrect="off" /> - {{gh-error-message errors=user.errors property="facebook"}} - <p>URL of your personal Facebook Profile</p> - {{/gh-form-group}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="twitter"}} - <label for="user-twitter">Twitter Profile</label> - <input value={{user.twitter}} oninput={{action (mut _scratchTwitter) value="target.value"}} {{action "validateTwitterUrl" on="focusOut"}} type="url" class="gh-input" id="user-twitter" name="user[twitter]" placeholder="https://twitter.com/username" autocorrect="off" /> - {{gh-error-message errors=user.errors property="twitter"}} - <p>URL of your personal Twitter profile</p> - {{/gh-form-group}} - - {{#gh-form-group errors=user.errors hasValidated=user.hasValidated property="bio" class="bio-container"}} - <label for="user-bio">Bio</label> - {{textarea id="user-bio" class="gh-input" value=user.bio focusOut=(action "validate" "bio" target=user)}} - {{gh-error-message errors=user.errors property="bio"}} - <p> - Write about you, in 200 characters or less. - {{gh-count-characters user.bio}} - </p> - {{/gh-form-group}} - - <hr /> - - </fieldset> - {{!-- If an administrator is viewing Owner's profile then hide inputs for change password --}} - {{#unless isAdminUserOnOwnerProfile}} - <fieldset> - {{#unless isNotOwnProfile}} - <div class="form-group"> - <label for="user-password-old">Old Password</label> - {{input value=user.password type="password" id="user-password-old" class="gh-input"}} - </div> - {{/unless}} - - <div class="form-group"> - <label for="user-password-new">New Password</label> - {{input value=user.newPassword type="password" id="user-password-new" class="gh-input"}} - </div> - - <div class="form-group"> - <label for="user-new-password-verification">Verify Password</label> - {{input value=user.ne2Password type="password" id="user-new-password-verification" class="gh-input"}} - </div> - <div class="form-group"> - <button type="button" class="btn btn-red button-change-password" {{action "password"}}>Change Password</button> - </div> - - </fieldset> - {{/unless}} - </form> - </div> -</section> diff --git a/core/client/app/transforms/facebook-url-user.js b/core/client/app/transforms/facebook-url-user.js deleted file mode 100644 index bb2f5e7825..0000000000 --- a/core/client/app/transforms/facebook-url-user.js +++ /dev/null @@ -1,21 +0,0 @@ -import Transform from 'ember-data/transform'; - -export default Transform.extend({ - deserialize(serialized) { - if (serialized) { - let [ , user ] = serialized.match(/(\S+)/); - - return `https://www.facebook.com/${user}`; - } - return serialized; - }, - - serialize(deserialized) { - if (deserialized) { - let [ , user] = deserialized.match(/(?:https:\/\/)(?:www\.)(?:facebook\.com)\/(?:#!\/)?(\w+\/?\S+)/mi); - - return user; - } - return deserialized; - } -}); diff --git a/core/client/app/transforms/moment-date.js b/core/client/app/transforms/moment-date.js deleted file mode 100644 index 6dae62cad8..0000000000 --- a/core/client/app/transforms/moment-date.js +++ /dev/null @@ -1,18 +0,0 @@ -/* global moment */ -import Transform from 'ember-data/transform'; - -export default Transform.extend({ - deserialize(serialized) { - if (serialized) { - return moment(serialized); - } - return serialized; - }, - - serialize(deserialized) { - if (deserialized) { - return moment(deserialized).toDate(); - } - return deserialized; - } -}); diff --git a/core/client/app/transforms/navigation-settings.js b/core/client/app/transforms/navigation-settings.js deleted file mode 100644 index 07dc4aa386..0000000000 --- a/core/client/app/transforms/navigation-settings.js +++ /dev/null @@ -1,41 +0,0 @@ -import Ember from 'ember'; -import Transform from 'ember-data/transform'; -import NavigationItem from 'ghost/models/navigation-item'; - -const {isArray} = Ember; -const emberA = Ember.A; - -export default Transform.extend({ - deserialize(serialized) { - let navItems, settingsArray; - - try { - settingsArray = JSON.parse(serialized) || []; - } catch (e) { - settingsArray = []; - } - - navItems = settingsArray.map((itemDetails) => { - return NavigationItem.create(itemDetails); - }); - - return emberA(navItems); - }, - - serialize(deserialized) { - let settingsArray; - - if (isArray(deserialized)) { - settingsArray = deserialized.map((item) => { - let label = item.get('label').trim(); - let url = item.get('url').trim(); - - return {label, url}; - }).compact(); - } else { - settingsArray = []; - } - - return JSON.stringify(settingsArray); - } -}); diff --git a/core/client/app/transforms/raw.js b/core/client/app/transforms/raw.js deleted file mode 100644 index e366f928c9..0000000000 --- a/core/client/app/transforms/raw.js +++ /dev/null @@ -1,11 +0,0 @@ -import Transform from 'ember-data/transform'; - -export default Transform.extend({ - deserialize(serialized) { - return serialized; - }, - - serialize(deserialized) { - return deserialized; - } -}); diff --git a/core/client/app/transforms/slack-settings.js b/core/client/app/transforms/slack-settings.js deleted file mode 100644 index 4d242d79c0..0000000000 --- a/core/client/app/transforms/slack-settings.js +++ /dev/null @@ -1,37 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import Transform from 'ember-data/transform'; -import SlackObject from 'ghost/models/slack-integration'; - -const {isArray} = Ember; -const emberA = Ember.A; - -export default Transform.extend({ - deserialize(serialized) { - let slackObj, settingsArray; - try { - settingsArray = JSON.parse(serialized) || []; - } catch (e) { - settingsArray = []; - } - - slackObj = settingsArray.map((itemDetails) => { - return SlackObject.create(itemDetails); - }); - return emberA(slackObj); - }, - - serialize(deserialized) { - let settingsArray; - if (isArray(deserialized)) { - settingsArray = deserialized.map((item) => { - let url = (item.get('url') || '').trim(); - - return {url}; - }).compact(); - } else { - settingsArray = []; - } - return JSON.stringify(settingsArray); - } -}); diff --git a/core/client/app/transforms/twitter-url-user.js b/core/client/app/transforms/twitter-url-user.js deleted file mode 100644 index ac2eae1903..0000000000 --- a/core/client/app/transforms/twitter-url-user.js +++ /dev/null @@ -1,21 +0,0 @@ -import Transform from 'ember-data/transform'; - -export default Transform.extend({ - deserialize(serialized) { - if (serialized) { - let [ , user ] = serialized.match(/@?([^\/]*)/); - - return `https://twitter.com/${user}`; - } - return serialized; - }, - - serialize(deserialized) { - if (deserialized) { - let [ , user] = deserialized.match(/(?:https:\/\/)(?:twitter\.com)\/(?:#!\/)?@?([^\/]*)/); - - return `@${user}`; - } - return deserialized; - } -}); diff --git a/core/client/app/transitions.js b/core/client/app/transitions.js deleted file mode 100644 index 23d7ce920b..0000000000 --- a/core/client/app/transitions.js +++ /dev/null @@ -1,16 +0,0 @@ -import { target } from 'liquid-tether'; - -export default function () { - this.transition( - target('fullscreen-modal'), - this.toValue(({isVisible}) => isVisible), - // this.use('tether', [modal options], [background options]) - this.use('tether', ['fade', {duration: 150}], ['fade', {duration: 150}]), - this.reverse('tether', ['fade', {duration: 80}], ['fade', {duration: 150}]) - ); - - this.transition( - this.hasClass('fade-transition'), - this.use('crossFade', {duration: 100}) - ); -} diff --git a/core/client/app/utils/ajax.js b/core/client/app/utils/ajax.js deleted file mode 100644 index de659142da..0000000000 --- a/core/client/app/utils/ajax.js +++ /dev/null @@ -1,49 +0,0 @@ -import Ember from 'ember'; - -const {isArray} = Ember; - -// TODO: this should be removed and instead have our app serializer properly -// process the response so that errors can be tied to the model - -// Used in API request fail handlers to parse a standard api error -// response json for the message to display -export default function getRequestErrorMessage(request, performConcat) { - let message, - msgDetail; - - // Can't really continue without a request - if (!request) { - return null; - } - - // Seems like a sensible default - message = request.statusText; - - // If a non 200 response - if (request.status !== 200) { - try { - // Try to parse out the error, or default to 'Unknown' - if (request.errors && isArray(request.errors)) { - message = request.errors.map((errorItem) => { - return errorItem.message; - }); - } else { - message = request.error || 'Unknown Error'; - } - } catch (e) { - msgDetail = request.status ? `${request.status} - ${request.statusText}` : 'Server was not available'; - message = `The server returned an error (${msgDetail}).`; - } - } - - if (performConcat && isArray(message)) { - message = message.join('<br />'); - } - - // return an array of errors by default - if (!performConcat && typeof message === 'string') { - message = [message]; - } - - return message; -} diff --git a/core/client/app/utils/bound-one-way.js b/core/client/app/utils/bound-one-way.js deleted file mode 100644 index 74b3844752..0000000000 --- a/core/client/app/utils/bound-one-way.js +++ /dev/null @@ -1,31 +0,0 @@ -import Ember from 'ember'; - -const {computed} = Ember; - -/** - * Defines a property similarly to `Ember.computed.oneway`, - * save that while a `oneway` loses its binding upon being set, - * the `BoundOneWay` will continue to listen for upstream changes. - * - * This is an ideal tool for working with values inside of {{input}} - * elements. - * @param {*} upstream - * @param {function} transform a function to transform the **upstream** value. - */ -export default function (upstream, transform) { - if (typeof transform !== 'function') { - // default to the identity function - transform = function (value) { - return value; - }; - } - - return computed(upstream, { - get() { - return transform(this.get(upstream)); - }, - set(key, value) { - return value; - } - }); -} diff --git a/core/client/app/utils/caja-sanitizers.js b/core/client/app/utils/caja-sanitizers.js deleted file mode 100644 index d55198856e..0000000000 --- a/core/client/app/utils/caja-sanitizers.js +++ /dev/null @@ -1,26 +0,0 @@ -/** - * google-caja uses url() and id() to verify if the values are allowed. - */ -/** - * Check if URL is allowed - * URLs are allowed if they start with http://, https://, or /. - */ -let url = function (url) { - url = url.toString().replace(/['"]+/g, ''); - if (/^https?:\/\//.test(url) || /^\//.test(url)) { - return url; - } -}; - -/** - * Check if ID is allowed - * All ids are allowed at the moment. - */ -let id = function (id) { - return id; -}; - -export default { - url, - id -}; diff --git a/core/client/app/utils/ctrl-or-cmd.js b/core/client/app/utils/ctrl-or-cmd.js deleted file mode 100644 index 8384620b9b..0000000000 --- a/core/client/app/utils/ctrl-or-cmd.js +++ /dev/null @@ -1 +0,0 @@ -export default navigator.userAgent.indexOf('Mac') !== -1 ? 'command' : 'ctrl'; diff --git a/core/client/app/utils/date-formatting.js b/core/client/app/utils/date-formatting.js deleted file mode 100644 index bc4d1411ed..0000000000 --- a/core/client/app/utils/date-formatting.js +++ /dev/null @@ -1,39 +0,0 @@ -/* global moment */ -// jscs: disable disallowSpacesInsideParentheses - -const parseDateFormats = ['DD MMM YY @ HH:mm', 'DD MMM YY HH:mm', - 'D MMM YY @ HH:mm', 'D MMM YY HH:mm', - 'DD MMM YYYY @ HH:mm', 'DD MMM YYYY HH:mm', - 'D MMM YYYY @ HH:mm', 'D MMM YYYY HH:mm', - 'DD/MM/YY @ HH:mm', 'DD/MM/YY HH:mm', - 'DD/MM/YYYY @ HH:mm', 'DD/MM/YYYY HH:mm', - 'DD-MM-YY @ HH:mm', 'DD-MM-YY HH:mm', - 'DD-MM-YYYY @ HH:mm', 'DD-MM-YYYY HH:mm', - 'YYYY-MM-DD @ HH:mm', 'YYYY-MM-DD HH:mm', - 'DD MMM @ HH:mm', 'DD MMM HH:mm', - 'D MMM @ HH:mm', 'D MMM HH:mm']; - -const displayDateFormat = 'DD MMM YY @ HH:mm'; - -// Add missing timestamps -function verifyTimeStamp(dateString) { - if (dateString && !dateString.slice(-5).match(/\d+:\d\d/)) { - dateString += ' 12:00'; - } - return dateString; -} - -// Parses a string to a Moment -function parseDateString(value) { - return value ? moment(verifyTimeStamp(value), parseDateFormats, true) : undefined; -} - -// Formats a Date or Moment -function formatDate(value) { - return verifyTimeStamp(value ? moment(value).format(displayDateFormat) : ''); -} - -export { - parseDateString, - formatDate -}; diff --git a/core/client/app/utils/document-title.js b/core/client/app/utils/document-title.js deleted file mode 100644 index 504c6dd29b..0000000000 --- a/core/client/app/utils/document-title.js +++ /dev/null @@ -1,58 +0,0 @@ -import Ember from 'ember'; - -const {Route, Router, isArray, on} = Ember; - -export default function () { - Route.reopen({ - // `titleToken` can either be a static string or a function - // that accepts a model object and returns a string (or array - // of strings if there are multiple tokens). - titleToken: null, - - // `title` can either be a static string or a function - // that accepts an array of tokens and returns a string - // that will be the document title. The `collectTitleTokens` action - // stops bubbling once a route is encountered that has a `title` - // defined. - title: null, - - actions: { - collectTitleTokens(tokens) { - let {titleToken} = this; - let finalTitle; - - if (typeof this.titleToken === 'function') { - titleToken = this.titleToken(this.currentModel); - } - - if (isArray(titleToken)) { - tokens.unshift(...titleToken); - } else if (titleToken) { - tokens.unshift(titleToken); - } - - if (this.title) { - if (typeof this.title === 'function') { - finalTitle = this.title(tokens); - } else { - finalTitle = this.title; - } - - this.router.setTitle(finalTitle); - } else { - return true; - } - } - } - }); - - Router.reopen({ - updateTitle: on('didTransition', function () { - this.send('collectTitleTokens', []); - }), - - setTitle(title) { - window.document.title = title; - } - }); -} diff --git a/core/client/app/utils/ed-image-manager.js b/core/client/app/utils/ed-image-manager.js deleted file mode 100644 index 539e8b976e..0000000000 --- a/core/client/app/utils/ed-image-manager.js +++ /dev/null @@ -1,40 +0,0 @@ -const imageMarkdownRegex = /^!(?:\[([^\n\]]*)\])(?:\(([^\n\]]*)\))?$/gim; - -// Process the markdown content and find all of the locations where there is an image markdown block -function parse(stringToParse) { - let images = []; - let m; - - while ((m = imageMarkdownRegex.exec(stringToParse)) !== null) { - images.push(m); - } - - return images; -} - -// Figure out the start and end of the selection range for the src in the markdown, so we know what to replace -function getSrcRange(content, index) { - let images = parse(content); - let replacement = {}; - - if (index > -1) { - // [1] matches the alt text, and 2 matches the url between the () - // if the () are missing entirely, which is valid, [2] will be undefined and we'll need to treat this case - // a little differently - if (images[index][2] === undefined) { - replacement.needsParens = true; - replacement.start = content.indexOf(']', images[index].index) + 1; - replacement.end = replacement.start; - } else { - replacement.start = content.indexOf('(', images[index].index) + 1; - replacement.end = replacement.start + images[index][2].length; - } - return replacement; - } - - return false; -} - -export default { - getSrcRange -}; diff --git a/core/client/app/utils/editor-shortcuts.js b/core/client/app/utils/editor-shortcuts.js deleted file mode 100644 index 54b62879b1..0000000000 --- a/core/client/app/utils/editor-shortcuts.js +++ /dev/null @@ -1,31 +0,0 @@ -// # Editor shortcuts -// Loaded by gh-editor component -// This map is used to ensure the right action is called by each shortcut -import ctrlOrCmd from 'ghost/utils/ctrl-or-cmd'; - -let shortcuts = {}; - -// Markdown Shortcuts - -// Text -shortcuts['ctrl+alt+u'] = {action: 'editorShortcut', options: {type: 'strike'}}; -shortcuts[`${ctrlOrCmd}+b`] = {action: 'editorShortcut', options: {type: 'bold'}}; -shortcuts[`${ctrlOrCmd}+i`] = {action: 'editorShortcut', options: {type: 'italic'}}; - -shortcuts['ctrl+u'] = {action: 'editorShortcut', options: {type: 'uppercase'}}; -shortcuts['ctrl+shift+u'] = {action: 'editorShortcut', options: {type: 'lowercase'}}; -shortcuts['ctrl+alt+shift+u'] = {action: 'editorShortcut', options: {type: 'titlecase'}}; -shortcuts[`${ctrlOrCmd}+shift+c`] = {action: 'editorShortcut', options: {type: 'copyHTML'}}; -shortcuts[`${ctrlOrCmd}+h`] = {action: 'editorShortcut', options: {type: 'cycleHeaderLevel'}}; - -// Formatting -shortcuts['ctrl+q'] = {action: 'editorShortcut', options: {type: 'blockquote'}}; -shortcuts['ctrl+l'] = {action: 'editorShortcut', options: {type: 'list'}}; - -// Insert content -shortcuts['ctrl+shift+1'] = {action: 'editorShortcut', options: {type: 'currentDate'}}; -shortcuts[`${ctrlOrCmd}+k`] = {action: 'editorShortcut', options: {type: 'link'}}; -shortcuts[`${ctrlOrCmd}+shift+i`] = {action: 'editorShortcut', options: {type: 'image'}}; -shortcuts[`${ctrlOrCmd}+shift+k`] = {action: 'editorShortcut', options: {type: 'code'}}; - -export default shortcuts; diff --git a/core/client/app/utils/ghost-paths.js b/core/client/app/utils/ghost-paths.js deleted file mode 100644 index 5be07bb682..0000000000 --- a/core/client/app/utils/ghost-paths.js +++ /dev/null @@ -1,55 +0,0 @@ -let makeRoute = function (root, args) { - let slashAtStart = /^\//; - let slashAtEnd = /\/$/; - let parts = Array.prototype.slice.call(args, 0); - let route = root.replace(slashAtEnd, ''); - - parts.forEach((part) => { - if (part) { - route = [route, part.replace(slashAtStart, '').replace(slashAtEnd, '')].join('/'); - } - }); - - return route += '/'; -}; - -export default function () { - let path = window.location.pathname; - let subdir = path.substr(0, path.search('/ghost/')); - let adminRoot = `${subdir}/ghost`; - let apiRoot = `${subdir}/ghost/api/v0.1`; - - function assetUrl(src) { - return subdir + src; - } - - return { - adminRoot, - apiRoot, - subdir, - blogRoot: `${subdir}/`, - count: 'https://count.ghost.org/', - - url: { - admin() { - return makeRoute(adminRoot, arguments); - }, - - api() { - return makeRoute(apiRoot, arguments); - }, - - join() { - if (arguments.length > 1) { - return makeRoute(arguments[0], Array.prototype.slice.call(arguments, 1)); - } else if (arguments.length === 1) { - let [arg] = arguments; - return arg.slice(-1) === '/' ? arg : `${arg}/`; - } - return '/'; - }, - - asset: assetUrl - } - }; -} diff --git a/core/client/app/utils/isFinite.js b/core/client/app/utils/isFinite.js deleted file mode 100644 index 23e1dd9f42..0000000000 --- a/core/client/app/utils/isFinite.js +++ /dev/null @@ -1,7 +0,0 @@ -/* globals window */ - -// isFinite function from lodash - -export default function (value) { - return window.isFinite(value) && !window.isNaN(parseFloat(value)); -} diff --git a/core/client/app/utils/isNumber.js b/core/client/app/utils/isNumber.js deleted file mode 100644 index 0688a5baa3..0000000000 --- a/core/client/app/utils/isNumber.js +++ /dev/null @@ -1,8 +0,0 @@ -// isNumber function from lodash - -const {toString} = Object.prototype; - -export default function (value) { - return typeof value === 'number' || - value && typeof value === 'object' && toString.call(value) === '[object Number]' || false; -} diff --git a/core/client/app/utils/link-component.js b/core/client/app/utils/link-component.js deleted file mode 100644 index 64367a5b96..0000000000 --- a/core/client/app/utils/link-component.js +++ /dev/null @@ -1,20 +0,0 @@ -import Ember from 'ember'; -import {invokeAction} from 'ember-invoke-action'; - -const {LinkComponent, computed} = Ember; - -LinkComponent.reopen({ - active: computed('attrs.params', '_routing.currentState', function () { - let isActive = this._super(...arguments); - - if (typeof this.get('alternateActive') === 'function') { - invokeAction(this, 'alternateActive', isActive); - } - - return isActive; - }), - - activeClass: computed('tagName', function () { - return this.get('tagName') === 'button' ? '' : 'active'; - }) -}); diff --git a/core/client/app/utils/random-password.js b/core/client/app/utils/random-password.js deleted file mode 100644 index 0c54928785..0000000000 --- a/core/client/app/utils/random-password.js +++ /dev/null @@ -1,8 +0,0 @@ -/* global generatePassword */ - -export default function () { - let word = generatePassword(6); - let randomN = Math.floor(Math.random() * 1000); - - return word + randomN; -} diff --git a/core/client/app/utils/text-field.js b/core/client/app/utils/text-field.js deleted file mode 100644 index 4287d37c08..0000000000 --- a/core/client/app/utils/text-field.js +++ /dev/null @@ -1,7 +0,0 @@ -import Ember from 'ember'; - -const {TextField} = Ember; - -TextField.reopen({ - attributeBindings: ['autofocus'] -}); diff --git a/core/client/app/utils/titleize.js b/core/client/app/utils/titleize.js deleted file mode 100644 index ee06358247..0000000000 --- a/core/client/app/utils/titleize.js +++ /dev/null @@ -1,16 +0,0 @@ -import Ember from 'ember'; -const lowerWords = ['of', 'a', 'the', 'and', 'an', 'or', 'nor', 'but', 'is', 'if', - 'then', 'else', 'when', 'at', 'from', 'by', 'on', 'off', 'for', - 'in', 'out', 'over', 'to', 'into', 'with']; - -export default function (input) { - let words = input.split(' ').map((word, index) => { - if (index === 0 || lowerWords.indexOf(word) === -1) { - word = Ember.String.capitalize(word); - } - - return word; - }); - - return words.join(' '); -} diff --git a/core/client/app/utils/validator-extensions.js b/core/client/app/utils/validator-extensions.js deleted file mode 100644 index 825b17a634..0000000000 --- a/core/client/app/utils/validator-extensions.js +++ /dev/null @@ -1,19 +0,0 @@ -import Ember from 'ember'; - -const {isBlank} = Ember; - -function init() { - // Provide a few custom validators - // - validator.extend('empty', function (str) { - return isBlank(str); - }); - - validator.extend('notContains', function (str, badString) { - return str.indexOf(badString) === -1; - }); -} - -export default { - init -}; diff --git a/core/client/app/utils/window-proxy.js b/core/client/app/utils/window-proxy.js deleted file mode 100644 index 48b98a6c30..0000000000 --- a/core/client/app/utils/window-proxy.js +++ /dev/null @@ -1,9 +0,0 @@ -export default { - changeLocation(url) { - window.location = url; - }, - - replaceLocation(url) { - window.location.replace(url); - } -}; diff --git a/core/client/app/utils/word-count.js b/core/client/app/utils/word-count.js deleted file mode 100644 index fef7438a46..0000000000 --- a/core/client/app/utils/word-count.js +++ /dev/null @@ -1,18 +0,0 @@ -// jscs: disable - -export default function (s) { - // replaces previous XRegExp("[^\\s\\d\\p{L}]", 'g') that was causing - // issues when browsers added more es6 support and was a 63KB minified - // dependency whereas this is ~8KB - // unicode list taken from https://github.com/slevithan/xregexp/blob/master/src/addons/unicode-categories.js - let nonANumLetters = new RegExp("[^\\s\\dA-Za-z\xAA\xB5\xBA\xC0-\xD6\xD8-\xF6\xF8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u037F\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u052F\u0531-\u0556\u0559\u0561-\u0587\u05D0-\u05EA\u05F0-\u05F2\u0620-\u064A\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE\u06EF\u06FA-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07CA-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u08A0-\u08B4\u0904-\u0939\u093D\u0950\u0958-\u0961\u0971-\u0980\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09F0\u09F1\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0AF9\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B71\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C39\u0C3D\u0C58-\u0C5A\u0C60\u0C61\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CF1\u0CF2\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D5F-\u0D61\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0EDC-\u0EDF\u0F00\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1380-\u138F\u13A0-\u13F5\u13F8-\u13FD\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u16F1-\u16F8\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u1820-\u1877\u1880-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191E\u1950-\u196D\u1970-\u1974\u1980-\u19AB\u19B0-\u19C9\u1A00-\u1A16\u1A20-\u1A54\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B83-\u1BA0\u1BAE\u1BAF\u1BBA-\u1BE5\u1C00-\u1C23\u1C4D-\u1C4F\u1C5A-\u1C7D\u1CE9-\u1CEC\u1CEE-\u1CF1\u1CF5\u1CF6\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2071\u207F\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2183\u2184\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005\u3006\u3031-\u3035\u303B\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312D\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FD5\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA61F\uA62A\uA62B\uA640-\uA66E\uA67F-\uA69D\uA6A0-\uA6E5\uA717-\uA71F\uA722-\uA788\uA78B-\uA7AD\uA7B0-\uA7B7\uA7F7-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA840-\uA873\uA882-\uA8B3\uA8F2-\uA8F7\uA8FB\uA8FD\uA90A-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF\uA9E0-\uA9E4\uA9E6-\uA9EF\uA9FA-\uA9FE\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA60-\uAA76\uAA7A\uAA7E-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uAB30-\uAB5A\uAB5C-\uAB65\uAB70-\uABE2\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]", 'g'); - - s = s.replace(/<(.|\n)*?>/g, ' '); // strip tags - s = s.replace(nonANumLetters, ''); // ignore non-alphanumeric letters - s = s.replace(/(^\s*)|(\s*$)/gi, ''); // exclude starting and ending white-space - s = s.replace(/\n /gi, ' '); // convert newlines to spaces - s = s.replace(/\n+/gi, ' '); - s = s.replace(/[ ]{2,}/gi, ' '); // convert 2 or more spaces to 1 - - return s.split(' ').length; -} diff --git a/core/client/app/validators/base.js b/core/client/app/validators/base.js deleted file mode 100644 index b753ba835e..0000000000 --- a/core/client/app/validators/base.js +++ /dev/null @@ -1,37 +0,0 @@ -import Ember from 'ember'; - -/** - * Base validator that all validators should extend - * Handles checking of individual properties or the entire model - */ -export default Ember.Object.extend({ - properties: [], - passed: false, - - /** - * When passed a model and (optionally) a property name, - * checks it against a list of validation functions - * @param {Ember.Object} model Model to validate - * @param {string} prop Property name to check - * @return {boolean} True if the model passed all (or one) validation(s), - * false if not - */ - check(model, prop) { - this.set('passed', true); - - if (prop && this[prop]) { - this[prop](model); - } else { - this.get('properties').forEach((property) => { - if (this[property]) { - this[property](model); - } - }); - } - return this.get('passed'); - }, - - invalidate() { - this.set('passed', false); - } -}); diff --git a/core/client/app/validators/invite-user.js b/core/client/app/validators/invite-user.js deleted file mode 100644 index a43b88aba7..0000000000 --- a/core/client/app/validators/invite-user.js +++ /dev/null @@ -1,17 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['email'], - - email(model) { - let email = model.get('email'); - - if (validator.empty(email)) { - model.get('errors').add('email', 'Please enter an email.'); - this.invalidate(); - } else if (!validator.isEmail(email)) { - model.get('errors').add('email', 'Invalid Email.'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/nav-item.js b/core/client/app/validators/nav-item.js deleted file mode 100644 index 081ab53ba1..0000000000 --- a/core/client/app/validators/nav-item.js +++ /dev/null @@ -1,36 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['label', 'url'], - - label(model) { - let label = model.get('label'); - let hasValidated = model.get('hasValidated'); - - if (validator.empty(label)) { - model.get('errors').add('label', 'You must specify a label'); - this.invalidate(); - } - - hasValidated.addObject('label'); - }, - - url(model) { - let url = model.get('url'); - let hasValidated = model.get('hasValidated'); - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - let validatorOptions = {require_protocol: true}; - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - let urlRegex = new RegExp(/^(\/|#|[a-zA-Z0-9\-]+:)/); - - if (validator.empty(url)) { - model.get('errors').add('url', 'You must specify a URL or relative path'); - this.invalidate(); - } else if (url.match(/\s/) || (!validator.isURL(url, validatorOptions) && !url.match(urlRegex))) { - model.get('errors').add('url', 'You must specify a valid URL or relative path'); - this.invalidate(); - } - - hasValidated.addObject('url'); - } -}); diff --git a/core/client/app/validators/new-user.js b/core/client/app/validators/new-user.js deleted file mode 100644 index f743e6f1aa..0000000000 --- a/core/client/app/validators/new-user.js +++ /dev/null @@ -1,35 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.extend({ - properties: ['name', 'email', 'password'], - - name(model) { - let name = model.get('name'); - - if (!validator.isLength(name, 1)) { - model.get('errors').add('name', 'Please enter a name.'); - this.invalidate(); - } - }, - - email(model) { - let email = model.get('email'); - - if (validator.empty(email)) { - model.get('errors').add('email', 'Please enter an email.'); - this.invalidate(); - } else if (!validator.isEmail(email)) { - model.get('errors').add('email', 'Invalid Email.'); - this.invalidate(); - } - }, - - password(model) { - let password = model.get('password'); - - if (!validator.isLength(password, 8)) { - model.get('errors').add('password', 'Password must be at least 8 characters long'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/post.js b/core/client/app/validators/post.js deleted file mode 100644 index b661b2f916..0000000000 --- a/core/client/app/validators/post.js +++ /dev/null @@ -1,37 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['title', 'metaTitle', 'metaDescription'], - - title(model) { - let title = model.get('title'); - - if (validator.empty(title)) { - model.get('errors').add('title', 'You must specify a title for the post.'); - this.invalidate(); - } - - if (!validator.isLength(title, 0, 150)) { - model.get('errors').add('title', 'Title cannot be longer than 150 characters.'); - this.invalidate(); - } - }, - - metaTitle(model) { - let metaTitle = model.get('metaTitle'); - - if (!validator.isLength(metaTitle, 0, 150)) { - model.get('errors').add('metaTitle', 'Meta Title cannot be longer than 150 characters.'); - this.invalidate(); - } - }, - - metaDescription(model) { - let metaDescription = model.get('metaDescription'); - - if (!validator.isLength(metaDescription, 0, 200)) { - model.get('errors').add('metaDescription', 'Meta Description cannot be longer than 200 characters.'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/reset.js b/core/client/app/validators/reset.js deleted file mode 100644 index 8c8992ba0c..0000000000 --- a/core/client/app/validators/reset.js +++ /dev/null @@ -1,21 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['newPassword'], - - newPassword(model) { - let p1 = model.get('newPassword'); - let p2 = model.get('ne2Password'); - - if (validator.empty(p1)) { - model.get('errors').add('newPassword', 'Please enter a password.'); - this.invalidate(); - } else if (!validator.isLength(p1, 8)) { - model.get('errors').add('newPassword', 'The password is not long enough.'); - this.invalidate(); - } else if (!validator.equals(p1, p2)) { - model.get('errors').add('ne2Password', 'The two new passwords don\'t match.'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/setting.js b/core/client/app/validators/setting.js deleted file mode 100644 index 09774d8eb0..0000000000 --- a/core/client/app/validators/setting.js +++ /dev/null @@ -1,47 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['title', 'description', 'password', 'postsPerPage'], - title(model) { - let title = model.get('title'); - - if (!validator.isLength(title, 0, 150)) { - model.get('errors').add('title', 'Title is too long'); - this.invalidate(); - } - }, - - description(model) { - let desc = model.get('description'); - - if (!validator.isLength(desc, 0, 200)) { - model.get('errors').add('description', 'Description is too long'); - this.invalidate(); - } - }, - - password(model) { - let isPrivate = model.get('isPrivate'); - let password = model.get('password'); - - if (isPrivate && password === '') { - model.get('errors').add('password', 'Password must be supplied'); - this.invalidate(); - } - }, - - postsPerPage(model) { - let postsPerPage = model.get('postsPerPage'); - - if (!validator.isInt(postsPerPage)) { - model.get('errors').add('postsPerPage', 'Posts per page must be a number'); - this.invalidate(); - } else if (postsPerPage > 1000) { - model.get('errors').add('postsPerPage', 'The maximum number of posts per page is 1000'); - this.invalidate(); - } else if (postsPerPage < 1) { - model.get('errors').add('postsPerPage', 'The minimum number of posts per page is 1'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/setup.js b/core/client/app/validators/setup.js deleted file mode 100644 index 62b380b5f3..0000000000 --- a/core/client/app/validators/setup.js +++ /dev/null @@ -1,14 +0,0 @@ -import NewUserValidator from 'ghost/validators/new-user'; - -export default NewUserValidator.create({ - properties: ['name', 'email', 'password', 'blogTitle'], - - blogTitle(model) { - let blogTitle = model.get('blogTitle'); - - if (!validator.isLength(blogTitle, 1)) { - model.get('errors').add('blogTitle', 'Please enter a blog title.'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/signin.js b/core/client/app/validators/signin.js deleted file mode 100644 index 46dd6aaaa6..0000000000 --- a/core/client/app/validators/signin.js +++ /dev/null @@ -1,48 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['identification', 'signin', 'forgotPassword'], - invalidMessage: 'Email address is not valid', - - identification(model) { - let id = model.get('identification'); - - if (!validator.empty(id) && !validator.isEmail(id)) { - model.get('errors').add('identification', this.get('invalidMessage')); - this.invalidate(); - } - }, - - signin(model) { - let id = model.get('identification'); - let password = model.get('password'); - - model.get('errors').clear(); - - if (validator.empty(id)) { - model.get('errors').add('identification', 'Please enter an email'); - this.invalidate(); - } - - if (!validator.empty(id) && !validator.isEmail(id)) { - model.get('errors').add('identification', this.get('invalidMessage')); - this.invalidate(); - } - - if (validator.empty(password)) { - model.get('errors').add('password', 'Please enter a password'); - this.invalidate(); - } - }, - - forgotPassword(model) { - let id = model.get('identification'); - - model.get('errors').clear(); - - if (validator.empty(id) || !validator.isEmail(id)) { - model.get('errors').add('identification', this.get('invalidMessage')); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/signup.js b/core/client/app/validators/signup.js deleted file mode 100644 index 0e0a537719..0000000000 --- a/core/client/app/validators/signup.js +++ /dev/null @@ -1,3 +0,0 @@ -import NewUserValidator from 'ghost/validators/new-user'; - -export default NewUserValidator.create(); diff --git a/core/client/app/validators/slack-integration.js b/core/client/app/validators/slack-integration.js deleted file mode 100644 index de746098d2..0000000000 --- a/core/client/app/validators/slack-integration.js +++ /dev/null @@ -1,24 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['url'], - - url(model) { - let url = model.get('url'); - let hasValidated = model.get('hasValidated'); - - let urlRegex = new RegExp(/(^https:\/\/hooks\.slack\.com\/services\/)(\S+)/); - - if (!validator.empty(url) && !url.match(urlRegex)) { - model.get('errors').add( - 'url', - 'The URL must be in a format like ' + - 'https://hooks.slack.com/services/<your personal key>' - ); - - this.invalidate(); - } - - hasValidated.addObject('url'); - } -}); diff --git a/core/client/app/validators/subscriber.js b/core/client/app/validators/subscriber.js deleted file mode 100644 index 3bb3ba44a5..0000000000 --- a/core/client/app/validators/subscriber.js +++ /dev/null @@ -1,19 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['email'], - - email(model) { - let email = model.get('email'); - - if (validator.empty(email)) { - model.get('errors').add('email', 'Please enter an email.'); - model.get('hasValidated').pushObject('email'); - this.invalidate(); - } else if (!validator.isEmail(email)) { - model.get('errors').add('email', 'Invalid email.'); - model.get('hasValidated').pushObject('email'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/tag-settings.js b/core/client/app/validators/tag-settings.js deleted file mode 100644 index 6cdc1f5022..0000000000 --- a/core/client/app/validators/tag-settings.js +++ /dev/null @@ -1,56 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['name', 'slug', 'description', 'metaTitle', 'metaDescription'], - - name(model) { - let name = model.get('name'); - - if (validator.empty(name)) { - model.get('errors').add('name', 'You must specify a name for the tag.'); - this.invalidate(); - } else if (name.match(/^,/)) { - model.get('errors').add('name', 'Tag names can\'t start with commas.'); - this.invalidate(); - } else if (!validator.isLength(name, 0, 150)) { - model.get('errors').add('name', 'Tag names cannot be longer than 150 characters.'); - this.invalidate(); - } - }, - - slug(model) { - let slug = model.get('slug'); - - if (!validator.isLength(slug, 0, 150)) { - model.get('errors').add('slug', 'URL cannot be longer than 150 characters.'); - this.invalidate(); - } - }, - - description(model) { - let description = model.get('description'); - - if (!validator.isLength(description, 0, 200)) { - model.get('errors').add('description', 'Description cannot be longer than 200 characters.'); - this.invalidate(); - } - }, - - metaTitle(model) { - let metaTitle = model.get('metaTitle'); - - if (!validator.isLength(metaTitle, 0, 150)) { - model.get('errors').add('metaTitle', 'Meta Title cannot be longer than 150 characters.'); - this.invalidate(); - } - }, - - metaDescription(model) { - let metaDescription = model.get('metaDescription'); - - if (!validator.isLength(metaDescription, 0, 200)) { - model.get('errors').add('metaDescription', 'Meta Description cannot be longer than 200 characters.'); - this.invalidate(); - } - } -}); diff --git a/core/client/app/validators/user.js b/core/client/app/validators/user.js deleted file mode 100644 index 0d89a9ec65..0000000000 --- a/core/client/app/validators/user.js +++ /dev/null @@ -1,81 +0,0 @@ -import BaseValidator from './base'; - -export default BaseValidator.create({ - properties: ['name', 'bio', 'email', 'location', 'website', 'roles'], - - isActive(model) { - return (model.get('status') === 'active'); - }, - - name(model) { - let name = model.get('name'); - - if (this.isActive(model)) { - if (validator.empty(name)) { - model.get('errors').add('name', 'Please enter a name.'); - this.invalidate(); - } else if (!validator.isLength(name, 0, 150)) { - model.get('errors').add('name', 'Name is too long'); - this.invalidate(); - } - } - }, - - bio(model) { - let bio = model.get('bio'); - - if (this.isActive(model)) { - if (!validator.isLength(bio, 0, 200)) { - model.get('errors').add('bio', 'Bio is too long'); - this.invalidate(); - } - } - }, - - email(model) { - let email = model.get('email'); - - if (!validator.isEmail(email)) { - model.get('errors').add('email', 'Please supply a valid email address'); - this.invalidate(); - } - }, - - location(model) { - let location = model.get('location'); - - if (this.isActive(model)) { - if (!validator.isLength(location, 0, 150)) { - model.get('errors').add('location', 'Location is too long'); - this.invalidate(); - } - } - }, - - website(model) { - let website = model.get('website'); - - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - if (this.isActive(model)) { - if (!validator.empty(website) && - (!validator.isURL(website, {require_protocol: false}) || - !validator.isLength(website, 0, 2000))) { - - model.get('errors').add('website', 'Website is not a valid url'); - this.invalidate(); - } - } - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - }, - - roles(model) { - if (!this.isActive(model)) { - let roles = model.get('roles'); - - if (roles.length < 1) { - model.get('errors').add('role', 'Please select a role'); - this.invalidate(); - } - } - } -}); diff --git a/core/client/bower.json b/core/client/bower.json deleted file mode 100644 index 22d4713fb3..0000000000 --- a/core/client/bower.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "name": "ghost", - "dependencies": { - "blueimp-md5": "2.3.0", - "codemirror": "5.13.2", - "devicejs": "0.2.7", - "ember": "2.5.1", - "ember-cli-shims": "0.1.1", - "ember-cli-test-loader": "0.2.2", - "ember-mocha": "0.8.11", - "Faker": "3.1.0", - "fastclick": "1.0.6", - "google-caja": "5669.0.0", - "jquery-deparam": "0.5.1", - "jquery-file-upload": "9.5.6", - "jquery-ui": "1.11.4", - "jqueryui-touch-punch": "furf/jquery-ui-touch-punch#4bc009145202d9c7483ba85f3a236a8f3470354d", - "jquery.simulate.drag-sortable": "0.1.0", - "keymaster": "1.6.3", - "lodash": "3.7.0", - "moment": "2.12.0", - "normalize.css": "3.0.3", - "password-generator": "2.0.2", - "pretender": "1.0.0", - "rangyinputs": "1.2.0", - "selectize": "~0.12.1", - "showdown-ghost": "0.3.6", - "validator-js": "3.39.0" - } -} diff --git a/core/client/config/deprecation-workflow.js b/core/client/config/deprecation-workflow.js deleted file mode 100644 index a8b4f6553f..0000000000 --- a/core/client/config/deprecation-workflow.js +++ /dev/null @@ -1,6 +0,0 @@ -window.deprecationWorkflow = window.deprecationWorkflow || {}; -window.deprecationWorkflow.config = { - workflow: [ - {handler: 'silence', matchMessage: 'You modified (-join-classes "ember-view" "form-group" (-normalize-class "errorClass" errorClass activeClass=undefined inactiveClass=undefined)) twice in a single render. This was unreliable in Ember 1.x and will be removed in Ember 3.0'} - ] -}; diff --git a/core/client/config/environment.js b/core/client/config/environment.js deleted file mode 100644 index 3889a71ff3..0000000000 --- a/core/client/config/environment.js +++ /dev/null @@ -1,54 +0,0 @@ -/* jshint node: true */ -/* jscs:disable */ - -module.exports = function (environment) { - var ENV = { - modulePrefix: 'ghost', - environment: environment, - baseURL: '/', - locationType: 'trailing-history', - EmberENV: { - FEATURES: { - // Here you can enable experimental features on an ember canary build - // e.g. 'with-controller': true - } - }, - - APP: { - // Here you can pass flags/options to your application instance - // when it is created - }, - - 'ember-simple-auth': { - authenticationRoute: 'signin', - routeAfterAuthentication: 'posts', - routeIfAlreadyAuthenticated: 'posts' - } - }; - - if (environment === 'development') { - // ENV.APP.LOG_RESOLVER = true; - ENV.APP.LOG_ACTIVE_GENERATION = true; - ENV.APP.LOG_TRANSITIONS = true; - ENV.APP.LOG_TRANSITIONS_INTERNAL = true; - ENV.APP.LOG_VIEW_LOOKUPS = true; - // Enable mirage here in order to mock API endpoints during development - ENV['ember-cli-mirage'] = { - enabled: false - }; - } - - if (environment === 'test') { - // Testem prefers this... - ENV.baseURL = '/'; - ENV.locationType = 'none'; - - // keep test console output quieter - ENV.APP.LOG_ACTIVE_GENERATION = false; - ENV.APP.LOG_VIEW_LOOKUPS = false; - - ENV.APP.rootElement = '#ember-testing'; - } - - return ENV; -}; diff --git a/core/client/ember-cli-build.js b/core/client/ember-cli-build.js deleted file mode 100644 index a78f4af043..0000000000 --- a/core/client/ember-cli-build.js +++ /dev/null @@ -1,85 +0,0 @@ -/* jscs:disable */ -/* global require, module */ - -var EmberApp = require('ember-cli/lib/broccoli/ember-app'), - environment = EmberApp.env(), - isProduction = environment === 'production', - mythCompress = isProduction || environment === 'test', - disabled = {enabled: false}, - assetLocation; - -assetLocation = function (fileName) { - if (isProduction) { - fileName = fileName.replace('.', '.min.'); - } - return '/assets/' + fileName; -}; - -module.exports = function (defaults) { - var app = new EmberApp(defaults, { - babel: { - optional: ['es6.spec.symbols'], - includePolyfill: true - }, - outputPaths: { - app: { - js: assetLocation('ghost.js') - }, - vendor: { - js: assetLocation('vendor.js'), - css: assetLocation('vendor.css') - } - }, - mythOptions: { - source: './app/styles/app.css', - inputFile: 'app.css', - browsers: 'last 2 versions', - // @TODO: enable sourcemaps for development without including them in the release - sourcemap: false, - compress: mythCompress, - outputFile: isProduction ? 'ghost.min.css' : 'ghost.css' - }, - hinting: false, - fingerprint: disabled, - 'ember-cli-selectize': { - theme: false - } - }); - - // 'dem Scripts - app.import('bower_components/validator-js/validator.js'); - app.import('bower_components/rangyinputs/rangyinputs-jquery-src.js'); - app.import('bower_components/showdown-ghost/src/showdown.js'); - app.import('bower_components/showdown-ghost/src/extensions/ghostgfm.js'); - app.import('bower_components/showdown-ghost/src/extensions/ghostimagepreview.js'); - app.import('bower_components/showdown-ghost/src/extensions/footnotes.js'); - app.import('bower_components/showdown-ghost/src/extensions/highlight.js'); - app.import('bower_components/moment/moment.js'); - app.import('bower_components/keymaster/keymaster.js'); - app.import('bower_components/devicejs/lib/device.js'); - app.import('bower_components/jquery-ui/jquery-ui.js'); - app.import('bower_components/jquery-file-upload/js/jquery.fileupload.js'); - app.import('bower_components/blueimp-load-image/js/load-image.all.min.js'); - app.import('bower_components/jquery-file-upload/js/jquery.fileupload-process.js'); - app.import('bower_components/jquery-file-upload/js/jquery.fileupload-image.js'); - app.import('bower_components/google-caja/html-css-sanitizer-bundle.js'); - app.import('bower_components/jqueryui-touch-punch/jquery.ui.touch-punch.js'); - app.import('bower_components/codemirror/lib/codemirror.js'); - app.import('bower_components/codemirror/mode/htmlmixed/htmlmixed.js'); - app.import('bower_components/codemirror/mode/xml/xml.js'); - app.import('bower_components/codemirror/mode/css/css.js'); - app.import('bower_components/codemirror/mode/javascript/javascript.js'); - app.import('bower_components/password-generator/lib/password-generator.js'); - app.import('bower_components/blueimp-md5/js/md5.js'); - - if (app.env === 'test') { - app.import(app.bowerDirectory + '/jquery.simulate.drag-sortable/jquery.simulate.drag-sortable.js', {type: 'test'}); - app.import(app.bowerDirectory + '/jquery-deparam/jquery-deparam.js', {type: 'test'}); - } - - // 'dem Styles - app.import('bower_components/codemirror/lib/codemirror.css'); - app.import('bower_components/codemirror/theme/xq-light.css'); - - return app.toTree(); -}; diff --git a/core/client/lib/.jshintrc b/core/client/lib/.jshintrc deleted file mode 100644 index 839c191fa9..0000000000 --- a/core/client/lib/.jshintrc +++ /dev/null @@ -1,4 +0,0 @@ -{ - "node": true, - "browser": false -} diff --git a/core/client/lib/asset-delivery/index.js b/core/client/lib/asset-delivery/index.js deleted file mode 100644 index 6215a3bb52..0000000000 --- a/core/client/lib/asset-delivery/index.js +++ /dev/null @@ -1,22 +0,0 @@ -/* jscs:disable */ -module.exports = { - name: 'asset-delivery', - postBuild: function (results) { - var fs = this.project.require('fs-extra'), - walkSync = this.project.require('walk-sync'), - assetsIn = results.directory + '/assets', - templateOut = '../server/views/default.hbs', - assetsOut = '../built/assets', - assets = walkSync(assetsIn); - - fs.ensureDirSync(assetsOut); - - fs.copySync(results.directory + '/index.html', templateOut, {clobber: true}); - - assets.forEach(function (relativePath) { - if (relativePath.slice(-1) === '/') { return; } - - fs.copySync(assetsIn + '/' + relativePath, assetsOut + '/' + relativePath, {clobber:true}); - }); - } -}; diff --git a/core/client/lib/asset-delivery/package.json b/core/client/lib/asset-delivery/package.json deleted file mode 100644 index 954fad24cb..0000000000 --- a/core/client/lib/asset-delivery/package.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "name": "asset-delivery", - "keywords": [ - "ember-addon" - ] -} diff --git a/core/client/package.json b/core/client/package.json deleted file mode 100644 index 232668cf96..0000000000 --- a/core/client/package.json +++ /dev/null @@ -1,71 +0,0 @@ -{ - "name": "ghost", - "version": "0.0.0", - "description": "Small description for ghost goes here", - "private": true, - "directories": { - "doc": "doc", - "test": "tests" - }, - "scripts": { - "start": "ember server", - "build": "ember build", - "test": "ember test" - }, - "repository": "", - "engines": { - "node": ">= 0.10.0" - }, - "author": "", - "license": "MIT", - "devDependencies": { - "broccoli-asset-rev": "2.4.2", - "ember-ajax": "0.7.1", - "ember-cli": "2.5.0", - "ember-cli-app-version": "1.0.0", - "ember-cli-babel": "5.1.6", - "ember-cli-content-security-policy": "0.4.0", - "ember-cli-dependency-checker": "1.2.0", - "ember-cli-deprecation-workflow": "0.2.0", - "ember-cli-fastclick": "1.0.3", - "ember-cli-htmlbars": "1.0.3", - "ember-cli-htmlbars-inline-precompile": "0.3.1", - "ember-cli-jshint": "1.0.0", - "ember-cli-mirage": "0.1.13", - "ember-cli-mocha": "0.10.1", - "ember-cli-pretender": "0.6.0", - "ember-cli-release": "0.2.8", - "ember-cli-selectize": "0.4.3", - "ember-cli-sri": "2.1.0", - "ember-cli-uglify": "1.2.0", - "ember-data": "2.5.2", - "ember-data-filter": "1.13.0", - "ember-export-application-global": "1.0.5", - "ember-invoke-action": "1.3.0", - "ember-light-table": "0.1.9", - "ember-load-initializers": "0.5.1", - "ember-myth": "0.1.1", - "ember-one-way-controls": "0.6.2", - "ember-power-select": "0.9.2", - "ember-resolver": "2.0.3", - "ember-route-action-helper": "0.3.0", - "ember-simple-auth": "1.1.0", - "ember-sinon": "0.5.0", - "ember-sortable": "1.7.0", - "ember-suave": "2.0.1", - "ember-watson": "0.7.0", - "ember-wormhole": "0.3.5", - "emberx-file-input": "1.0.0", - "fs-extra": "0.16.3", - "glob": "^4.0.5", - "liquid-fire": "0.23.0", - "liquid-tether": "1.0.0", - "loader.js": "4.0.1", - "walk-sync": "^0.1.3" - }, - "ember-addon": { - "paths": [ - "lib/asset-delivery" - ] - } -} diff --git a/core/client/public/assets/fonts/ghosticons.eot b/core/client/public/assets/fonts/ghosticons.eot deleted file mode 100755 index 3f125d7c7624a73b0742b053bab8567f6a143caa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 22584 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 diff --git a/core/client/public/assets/fonts/ghosticons.svg b/core/client/public/assets/fonts/ghosticons.svg deleted file mode 100755 index 2ad3574486..0000000000 --- a/core/client/public/assets/fonts/ghosticons.svg +++ /dev/null @@ -1,83 +0,0 @@ -<?xml version="1.0" standalone="no"?> -<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" > -<svg xmlns="http://www.w3.org/2000/svg"> -<metadata>Generated by IcoMoon</metadata> -<defs> -<font id="icomoon" horiz-adv-x="1024"> -<font-face units-per-em="1024" ascent="960" descent="-64" /> -<missing-glyph horiz-adv-x="1024" /> -<glyph unicode=" " d="" /> -<glyph unicode="" glyph-name="uniE000" d="M512 960c-282 0-512-230-512-512 0-136 52-264 148-360 98-98 226-152 364-152s268 54 364 152c0 0 0 0 0 0 96 96 148 224 148 360 0 282-230 512-512 512zM512-22c-118 0-228 44-314 122 44 24 108 46 172 70 22 8 44 16 64 24 8 2 14 10 14 20v106c0 10-6 18-14 20-2 0-50 20-50 108 0 12-8 22-20 22-2 0-6 8-6 20s4 20 4 22c12 0 22 10 22 22 0 4-2 10-6 22-6 22-22 82-8 102 2 2 6 6 16 6 6 0 10-2 16-2 10-2 22 4 24 16 6 28 56 48 118 48s112-20 118-48c10-40-8-94-16-120-4-12-6-18-6-24 0-12 8-22 20-22 2-2 6-10 6-22s-4-20-4-20c-12 0-22-10-22-22 0-88-48-108-50-108-8-2-14-10-14-20v-106c0-10 6-18 14-20 22-10 46-18 68-26 64-24 126-46 168-68-86-78-196-122-314-122zM858 130c-44 28-110 52-184 78-18 6-38 14-56 20v78c22 14 60 48 64 126 16 10 28 32 28 58s-10 46-24 56c12 32 30 92 18 142-16 60-96 80-160 80-58 0-128-16-152-62-30 2-46-12-56-22-26-36-8-102 2-138-14-10-24-30-24-56s12-48 28-58c4-78 42-112 64-126v-78c-18-6-34-12-50-18-70-26-142-52-190-80-80 88-124 200-124 318 0 258 212 470 470 470s470-212 470-470c0-118-44-230-124-318z" /> -<glyph unicode="" glyph-name="uniE001" d="M1018-28l-384 384c0 0 0 0 0 0 56 64 92 148 92 242 0 200-164 362-364 362s-362-162-362-362c0-200 162-364 362-364 94 0 178 36 242 92 0 0 0 0 0 0l384-384c4-4 10-6 14-6 6 0 12 2 16 6 8 8 8 22 0 30zM362 278c-176 0-320 142-320 320 0 176 144 320 320 320 178 0 320-144 320-320 0-178-142-320-320-320z" /> -<glyph unicode="" glyph-name="uniE002" d="M716 680l-534-278c-8-4-14-14-10-24 2-8 10-16 20-16h234v-234c0-10 8-18 16-20 2-2 4-2 6-2 8 0 16 4 18 12l278 534c4 8 4 18-4 24-6 8-16 8-24 4zM470 216v168c0 12-10 22-22 22h-168l396 206zM512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470z" /> -<glyph unicode="" glyph-name="uniE003" d="M874 768c-96 96-226 150-362 150s-266-54-362-150c-200-200-200-526 0-724 20-20 46-32 76-32 28 0 54 12 74 32 42 40 42 108 0 150-4 4-6 10-6 16 0 4 2 10 6 14 8 8 22 8 30 0l182-180c48-50 112-76 182-76 68 0 132 26 180 76 200 198 200 524 0 724zM844 74c-40-40-94-64-150-64-58 0-112 24-152 64l-180 180c-12 12-28 20-46 20s-34-8-46-20c-12-12-18-28-18-44 0-18 6-34 18-46 26-24 26-66 0-90-24-24-66-24-90 0-182 182-182 480 0 664 88 88 206 136 332 136s244-48 332-136c182-184 182-482 0-664zM694 532c22 0 44 8 60 24s24 38 24 60c0 24-8 44-24 60-32 34-90 34-122 0-16-16-24-36-24-60 0-22 8-44 24-60s38-24 62-24zM662 646c8 8 20 14 32 14 10 0 22-6 30-14s12-18 12-30c0-10-4-22-12-30-16-16-46-16-62 0-8 8-12 20-12 30 0 12 4 22 12 30zM856 450c-24 24-66 24-90 0-26-24-26-64 0-90 12-12 28-18 44-18 18 0 34 6 46 18s18 28 18 46c0 16-6 32-18 44zM826 390c-8-8-22-8-30 0s-8 22 0 30c4 4 8 6 14 6s12-2 16-6c4-4 6-8 6-14s-2-12-6-16zM692 300c-28 0-54-10-74-30s-32-48-32-76c0-28 12-56 32-76s46-30 76-30c28 0 54 10 74 30s32 48 32 76c0 28-12 56-32 76s-46 30-76 30zM738 148c-24-24-66-24-90 0-12 12-18 30-18 46 0 18 6 34 18 46s28 18 46 18c16 0 32-6 44-18s20-28 20-46c0-16-8-34-20-46zM376 722c-16-16-24-38-24-60 0-24 8-44 24-60s38-26 60-26c24 0 44 10 60 26 18 16 26 38 26 60s-10 44-26 60c-32 32-88 32-120 0zM466 632c-16-16-44-16-60 0-8 8-12 18-12 30s4 22 12 30c8 8 20 12 30 12 12 0 22-4 30-12s14-18 14-30c0-12-6-22-14-30z" /> -<glyph unicode="" glyph-name="uniE004" d="M918 960h-812c-12 0-20-10-20-22v-980c0-12 8-22 20-22h812c12 0 20 10 20 22v980c0 12-8 22-20 22zM896-22h-768v940h106v-44h-20v-84h84v84h-20v44h128v-44h-22v-84h86v84h-22v44h128v-44h-22v-84h86v84h-22v44h128v-44h-20v-84h84v84h-20v44h106zM746 662h-468c-12 0-22-10-22-22s10-22 22-22h468c12 0 22 10 22 22s-10 22-22 22zM746 534h-468c-12 0-22-10-22-22s10-22 22-22h468c12 0 22 10 22 22s-10 22-22 22zM746 406h-468c-12 0-22-10-22-22s10-22 22-22h468c12 0 22 10 22 22s-10 22-22 22zM746 278h-468c-12 0-22-10-22-22s10-22 22-22h468c12 0 22 10 22 22s-10 22-22 22zM746 150h-468c-12 0-22-10-22-22s10-22 22-22h468c12 0 22 10 22 22s-10 22-22 22z" /> -<glyph unicode="" glyph-name="uniE005" d="M542 448l476 474c10 8 10 22 0 30-8 10-22 10-30 0l-476-474-474 476c-8 10-22 10-30 0-10-8-10-22 0-30l474-476-476-474c-10-8-10-22 0-30 4-4 8-8 14-8s12 4 16 8l476 474 474-476c4-4 10-6 16-6s10 2 14 6c10 8 10 22 0 30z" /> -<glyph unicode="" glyph-name="uniE006" d="M960 490h-448v448c0 12-8 22-20 22 0 0 0 0 0 0-12 0-22-10-22-22v-448h-448c-12 0-22-8-22-20s10-22 22-22h448v-448c0-12 8-22 20-22s22 10 22 22v448h448c12 0 22 10 22 22s-10 20-22 20z" /> -<glyph unicode="" glyph-name="uniE007" d="M962 264c-52 52-128 70-198 46l-134 134 134 136c20-6 40-10 62-10 52 0 100 20 136 56 56 56 70 136 40 208-2 6-8 10-16 12-6 2-14 0-18-6l-76-78h-60v66l76 76c4 4 6 12 6 18-2 8-6 14-14 16-24 10-48 16-74 16-52 0-100-20-136-56-52-52-70-128-46-198l-136-134-134 134c24 70 6 146-46 198-56 56-136 72-208 40-6-2-12-8-12-16-2-6 0-14 6-18l78-76v-66h-66l-76 78c-4 6-12 8-20 6-6-2-12-6-14-12-32-72-16-154 40-210 52-52 128-68 198-44l134-136-134-134c-20 6-42 10-62 10-52 0-100-20-136-56-56-56-72-138-40-210 2-8 8-12 14-14 8 0 14 2 20 6l76 76h66v-60l-78-76c-6-4-8-12-6-20 0-6 6-12 12-14 24-10 50-16 74-16 52 0 100 20 136 56 50 52 68 128 44 198l134 134 136-134c-24-70-6-146 46-198 36-36 86-56 136-56 24 0 50 4 74 16 8 2 12 8 14 14 0 8-2 14-6 20l-76 76v60h60l76-76c4-4 12-6 18-6 8 2 14 6 16 14 32 72 16 154-40 210zM274 620c-6 6-16 6-24 4-56-24-120-12-164 30-34 36-50 84-42 130l58-58c4-4 8-6 14-6h98c12 0 20 8 20 20v98c0 4-2 10-6 14l-58 56c46 8 92-6 128-40 42-44 56-108 32-164-4-8-2-18 4-24l144-144-60-62zM330 186c24-56 12-122-30-164-36-34-84-50-130-42l58 58c4 4 6 8 6 14v90c0 12-8 22-20 22h-98c-6 0-10-2-14-6l-58-56c-8 48 8 96 42 132 28 28 66 44 106 44 20 0 40-4 58-12 8-4 18-2 24 4l410 410c6 6 8 16 4 24-24 56-12 120 32 164 28 28 66 42 106 42 8 0 18 0 26-2l-56-56c-4-4-6-10-6-14v-98c0-12 8-20 20-20h92c4 0 10 2 14 6l56 58c8-46-6-92-40-128-28-28-66-44-106-44-20 0-40 4-58 12-8 4-18 2-24-4l-410-410c-6-6-8-16-4-24zM972 102l-56 56c-4 4-10 6-14 6h-92c-12 0-20-10-20-22v-90c0-6 2-10 6-14l56-58c-48-8-96 8-132 42-44 44-56 108-32 164 4 8 2 18-4 24l-146 144 62 60 144-144c6-6 16-8 24-4 56 24 120 10 164-32 34-36 50-84 40-132z" /> -<glyph unicode="" glyph-name="uniE008" d="M1002 534h-122c-8 34-18 70-32 98l86 88c4 4 6 8 6 14s-2 12-6 16l-120 120c-8 8-22 8-30 0l-88-86c-28 14-64 24-98 32v122c0 12-10 22-22 22h-128c-12 0-22-10-22-22v-122c-34-8-70-18-98-32l-88 86c-8 8-22 8-30 0l-120-120c-8-8-8-22 0-30l86-88c-14-28-24-64-32-98h-122c-12 0-22-10-22-22v-128c0-12 10-22 22-22h122c8-34 18-70 32-98l-86-88c-4-4-6-8-6-14s2-12 6-16l120-120c8-8 22-8 30 0l88 86c28-14 64-24 98-32v-122c0-12 10-22 22-22h128c12 0 22 10 22 22v122c34 8 70 18 98 32l88-86c8-8 22-8 30 0l120 120c8 8 8 22 0 30l-86 88c14 28 24 64 32 98h122c12 0 22 10 22 22v128c0 12-10 22-22 22zM982 406h-118c-10 0-18-8-20-16-6-24-22-88-40-120-6-8-4-18 2-26l84-82-92-92-82 84c-8 6-18 8-26 2-32-18-96-34-120-40-8-2-16-10-16-20v-118h-84v118c0 10-8 18-16 20-24 6-88 22-120 40-8 6-18 4-26-2l-82-84-92 92 84 82c6 8 8 18 2 26-18 32-34 96-40 120-2 8-10 16-20 16h-118v84h118c10 0 18 8 20 16 6 24 22 88 40 120 6 8 4 18-2 26l-84 82 92 92 82-84c8-6 18-8 26-2 32 18 96 34 118 40 10 2 18 10 18 20v118h84v-118c0-10 8-18 16-20 38-10 92-24 120-40 8-6 18-4 26 2l82 84 92-92-84-82c-6-8-8-18-2-26 18-32 34-96 40-118 2-10 10-18 20-18h118zM512 662c-118 0-214-96-214-214s96-214 214-214c118 0 214 96 214 214s-96 214-214 214zM512 278c-94 0-170 76-170 170s76 170 170 170c94 0 170-76 170-170s-76-170-170-170z" /> -<glyph unicode="" glyph-name="uniE009" d="M1018 334l-620 620c-4 4-8 6-14 6h-362c-6 0-12-2-16-6s-6-10-6-16v-362c0-6 2-10 6-14l620-620c4-4 10-6 14-6 2 0 4 0 6 0 8 2 12 8 16 16l80 266 266 80c8 4 14 8 16 16s0 16-6 20zM720 256c-8-4-14-8-16-16l-74-242-588 588v332h332l588-588zM234 832c-58 0-106-48-106-106 0-60 48-108 106-108 60 0 108 48 108 108 0 58-48 106-108 106zM234 662c-34 0-64 28-64 64 0 34 30 64 64 64 36 0 64-30 64-64 0-36-28-64-64-64z" /> -<glyph unicode="" glyph-name="uniE00A" d="M1002 832h-980c-12 0-22-10-22-22v-768c0-12 10-20 22-20h980c12 0 22 8 22 20v768c0 12-10 22-22 22zM982 64h-940v726h940zM106 534h812c12 0 20 8 20 20v128c0 12-8 22-20 22h-812c-12 0-20-10-20-22v-128c0-12 8-20 20-20zM128 662h768v-86h-768zM448 448h-256c-12 0-22-10-22-22s10-20 22-20h256c12 0 22 8 22 20s-10 22-22 22zM448 362h-298c-12 0-22-8-22-20s10-22 22-22h298c12 0 22 10 22 22s-10 20-22 20zM448 278h-298c-12 0-22-10-22-22s10-22 22-22h298c12 0 22 10 22 22s-10 22-22 22zM448 192h-298c-12 0-22-10-22-22s10-20 22-20h298c12 0 22 8 22 20s-10 22-22 22zM874 448h-298c-12 0-22-10-22-22s10-20 22-20h298c12 0 22 8 22 20s-10 22-22 22zM874 362h-298c-12 0-22-8-22-20s10-22 22-22h298c12 0 22 10 22 22s-10 20-22 20zM874 278h-298c-12 0-22-10-22-22s10-22 22-22h298c12 0 22 10 22 22s-10 22-22 22zM832 192h-256c-12 0-22-10-22-22s10-20 22-20h256c12 0 22 8 22 20s-10 22-22 22z" /> -<glyph unicode="" glyph-name="uniE00B" d="M512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470zM784 720c-8 8-22 8-30 0l-198-198c-14 6-28 12-44 12-48 0-86-38-86-86s38-86 86-86c48 0 86 38 86 86 0 16-6 30-12 44l198 198c8 8 8 22 0 30zM512 406c-24 0-42 18-42 42s18 42 42 42c24 0 42-18 42-42s-18-42-42-42zM210 440c0 12-8 22-20 22h-86c-12 0-22-10-22-22s10-22 22-22h86c12 0 20 10 20 22zM208 346l-78-34c-12-4-16-16-12-28 4-8 12-12 20-12 2 0 4 0 8 2l78 32c12 4 16 16 12 28-4 10-16 16-28 12zM894 312l-78 34c-12 4-24-2-28-12-4-12 0-24 12-28l78-32c2-2 6-2 8-2 8 0 16 4 20 12 4 12 0 24-12 28zM918 470h-86c0 0 0 0 0 0-12 0-22-10-22-22s10-22 22-22h86c12 0 20 10 20 22s-8 22-20 22zM808 550c2 0 6 0 8 0l78 34c12 4 16 16 12 28-4 10-16 16-28 10l-78-32c-12-4-16-16-12-28 4-8 12-12 20-12zM626 724c2-2 6-2 8-2 8 0 16 6 20 14l32 78c6 12 0 24-10 28-12 4-24 0-28-12l-34-78c-4-12 2-24 12-28zM510 738c12 0 20 10 20 22v86c0 12-8 20-20 20s-22-8-22-20v-86c0-12 10-22 22-22zM370 736c4-8 12-14 20-14 2 0 6 0 8 2 10 4 16 16 12 28l-34 78c-4 12-16 16-28 12-10-4-16-16-10-28zM300 690l-60 60c-8 8-22 8-30 0s-8-22 0-30l60-60c4-4 10-8 16-8s10 4 14 8c10 8 10 20 0 30zM224 590l-78 32c-12 6-24 0-28-10-4-12 0-24 12-28l78-34c2 0 6 0 8 0 8 0 16 4 20 12 4 12 0 24-12 28zM704 278h-384c-12 0-22-10-22-22v-128c0-12 10-22 22-22h384c12 0 22 10 22 22v128c0 12-10 22-22 22zM682 150h-340v84h340z" /> -<glyph unicode="" glyph-name="uniE00C" d="M400 826c-10 8-22 8-32 0l-362-362c-8-10-8-22 0-32l362-362c6-4 10-6 16-6s10 2 16 6c8 8 8 22 0 30l-348 348 348 348c8 8 8 22 0 30zM1018 464l-362 362c-10 8-22 8-32 0-8-8-8-22 0-30l348-348-348-348c-8-8-8-22 0-30 6-4 10-6 16-6s10 2 16 6l362 362c8 10 8 22 0 32z" /> -<glyph unicode="" glyph-name="uniE00D" d="M256 490h214v44c0 32-26 66-58 74l-92 26v34c38 28 64 76 64 132 0 88-66 160-150 160-82 0-148-72-148-160 0-56 24-104 64-132v-34l-94-26c-32-8-56-42-56-74v-44zM234 918c48 0 90-36 102-84-6-2-14-4-24-2s-28 6-38 24c-2 6-8 10-14 12-8 0-14-2-20-6-30-32-78-34-106-24 16 46 54 80 100 80zM128 796c38-12 86-8 124 20 14-14 32-24 54-26 4-2 10-2 16-2s12 2 18 2c-2-60-50-110-106-110-58 0-104 52-106 116zM68 568l124 34v42c14-4 28-6 42-6 16 0 30 2 44 6v-42l124-34c12-4 24-20 24-34h-384c0 14 12 30 26 34zM968 608l-94 26v34c40 28 64 76 64 132 0 88-66 160-148 160-84 0-150-72-150-160 0-56 26-104 64-132v-34l-92-26c-32-8-58-42-58-74v-44h470v44c0 32-24 66-56 74zM790 918c48 0 88-36 102-84-8-2-16-4-26-2s-28 6-36 24c-4 6-10 10-16 12-6 0-14-2-18-6-32-32-78-34-108-24 16 46 56 80 102 80zM682 796c38-12 86-8 126 20 12-14 32-24 52-26 6 0 10-2 16-2s14 2 20 4c-4-62-50-112-106-112-58 0-106 52-108 116zM810 534h-212c0 14 12 30 24 34l124 34v42c14-4 28-6 44-6 14 0 28 2 42 6v-42l124-34c14-4 26-20 26-34zM690 54l-92 26v32c38 30 64 78 64 132 0 90-68 162-150 162s-150-72-150-162c0-54 26-102 64-132v-32l-92-26c-32-10-56-42-56-76v-42h468v42c0 34-24 66-56 76zM512 362c48 0 88-34 102-82-8-2-16-4-26-2s-26 6-36 24c-4 6-8 10-16 10-6 2-12 0-18-6-30-30-78-34-106-22 14 46 54 78 100 78zM406 242c38-12 86-8 124 18 14-12 32-22 52-24 4 0 10-2 14-2 8 0 16 2 22 4-2-62-50-112-106-112-58 0-104 52-106 116zM534-22h-214c0 14 12 30 26 34l124 36v42c12-4 28-6 42-6s30 2 42 6v-42l124-36c14-4 26-20 26-34zM292 304c8 10 8 22 0 32l-106 106c-8 8-22 8-30 0s-8-22 0-30l106-108c4-4 10-6 16-6 4 0 10 2 14 6zM838 442l-106-106c-8-10-8-22 0-32 4-4 10-6 14-6 6 0 12 2 16 6l106 108c8 8 8 22 0 30s-22 8-30 0z" /> -<glyph unicode="" glyph-name="uniE00E" d="M512 746c-152 0-278-124-278-276 0-122 78-228 192-264v-36c0-12 10-20 22-20h128c12 0 22 8 22 20v36c114 36 192 142 192 264 0 152-126 276-278 276zM570 242c-8-2-16-10-16-20v-30h-84v30c0 10-8 18-16 20-104 28-176 120-176 228 0 128 104 234 234 234s234-106 234-234c0-108-72-200-176-228zM534 22h-44c-12 0-20-10-20-22s8-22 20-22h44c12 0 20 10 20 22s-8 22-20 22zM576 106h-128c-12 0-22-8-22-20s10-22 22-22h128c12 0 22 10 22 22s-10 20-22 20zM512 790c12 0 22 8 22 20v86c0 12-10 22-22 22s-22-10-22-22v-86c0-12 10-20 22-20zM938 490h-84c-12 0-22-8-22-20s10-22 22-22h84c12 0 22 10 22 22s-10 20-22 20zM170 490h-84c-12 0-22-8-22-20s10-22 22-22h84c12 0 22 10 22 22s-10 20-22 20zM256 696c4-4 10-6 14-6 6 0 12 2 16 6 8 8 8 22 0 30l-90 90c-10 8-22 8-32 0-8-8-8-22 0-30zM754 690c4 0 10 2 14 6l90 90c10 8 10 22 0 30-8 8-20 8-30 0l-90-90c-8-8-8-22 0-30 4-4 10-6 16-6z" /> -<glyph unicode="" glyph-name="uniE00F" d="M1016 744c-8 8-22 8-30-2l-474-542-474 542c-8 10-22 10-30 2-10-8-10-20-2-30l490-560c4-6 10-8 16-8s12 2 16 8l490 560c8 10 8 22-2 30z" /> -<glyph unicode="" glyph-name="uniE010" d="M912 726l-122 122c-8 8-22 8-30 0l-16-18c-18 12-38 18-58 18-30 0-56-10-76-30l-242-242c-8-8-8-22 0-30 4-4 10-6 16-6 4 0 10 2 14 6l242 240c18 20 48 24 72 14l-192-194-424-422c0 0 0 0 0-2-2 0-2-2-4-6l-90-210c-4-8-2-18 4-24 4-4 10-6 16-6 2 0 6 0 8 2l210 90c0 0 0 0 0 0v0c4 2 6 2 6 4 2 0 2 0 2 0l422 424 242 240c4 4 6 10 6 16s-2 10-6 14zM62-2l58 134 76-76zM232 78l-90 90 392 392 90-90zM654 500l-90 90 196 198 16 14 90-90z" /> -<glyph unicode="" glyph-name="uniE011" d="M930 778c-48 48-112 76-180 76-70 0-134-28-182-76l-512-512c-74-74-74-196 0-272 38-36 86-56 136-56s98 20 136 56l490 492c20 20 30 48 30 76s-10 54-30 76c-20 20-48 30-76 30s-56-10-76-30l-362-364c-8-8-8-22 0-30s22-8 30 0l364 362c24 26 66 26 90 0 12-12 18-28 18-44 0-18-6-34-18-46l-490-492c-58-58-154-58-212 0-58 60-58 154 0 212l512 512c40 40 94 62 152 62 56 0 110-22 150-62 84-84 84-218 0-302l-362-362c-8-8-8-22 0-30s22-8 30 0l362 362c100 100 100 262 0 362z" /> -<glyph unicode="" glyph-name="uniE012" d="M982 832h-278v106c0 12-10 22-22 22h-340c-12 0-22-10-22-22v-106h-278c-12 0-20-10-20-22s8-20 20-20h108v-832c0-12 8-22 20-22h684c12 0 20 10 20 22v832h108c12 0 20 8 20 20s-8 22-20 22zM362 918h300v-86h-300zM832-22h-640v812h640zM342 662c-12 0-22-10-22-22v-490c0-12 10-22 22-22s20 10 20 22v490c0 12-8 22-20 22zM512 662c-12 0-22-10-22-22v-490c0-12 10-22 22-22s22 10 22 22v490c0 12-10 22-22 22zM662 640v-490c0-12 8-22 20-22s22 10 22 22v490c0 12-10 22-22 22s-20-10-20-22z" /> -<glyph unicode="" glyph-name="uniE013" d="M832 512c-12 0-22-10-22-22v-512h-768v854h662c12 0 22 10 22 22s-10 20-22 20h-682c-12 0-22-8-22-20v-896c0-12 10-22 22-22h810c12 0 22 10 22 22v532c0 12-10 22-22 22zM1020 918c-12 26-36 42-62 42-16 0-32-6-44-20l-408-406c0-2-2-4-4-6l-60-120c-4-10-2-20 4-26 4-4 10-6 16-6 2 0 6 0 10 2l120 60c2 2 4 4 6 4l346 348c0 0 0 0 0 0s0 0 0 0l60 60c20 20 26 44 16 68zM570 476l-60-30 30 60 328 330 32-32zM974 880l-44-44-30 30 44 44c12 12 30 6 36-8 4-8 2-14-6-22z" /> -<glyph unicode="" glyph-name="uniE014" d="M848 662l-122 122c-4 4-10 6-14 6-6 0-12-2-16-6l-376-378c0 0 0 0 0 0-2 0-2-2-2-2-2-2-2-2-4-4v-2c0 0 0 0 0 0l-90-210c-4-8-2-18 4-24 4-4 10-6 16-6 2 0 6 0 8 2l210 90c0 0 0 0 0 0h2c2 2 4 4 6 6l298 298c0 0 0 0 0 0s0 0 0 0l80 78c8 8 8 22 0 30zM454 300l-90 90 270 270 90-92zM342 354l76-76-134-58zM754 600l-90 90 48 48 90-90zM512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470z" /> -<glyph unicode="" glyph-name="uniE015" d="M512 362c-24 0-42-18-42-42 0-16 8-30 20-36v-114c0-12 10-20 22-20s22 8 22 20v114c12 6 20 20 20 36 0 24-18 42-42 42zM768 576v128c0 142-114 256-256 256s-256-114-256-256v-128h-128v-640h768v640zM298 704c0 118 96 214 214 214s214-96 214-214v-128h-428zM854-22h-684v556h684z" /> -<glyph unicode="" glyph-name="uniE016" d="M682 346c-28 0-54 12-74 32l-4 4c-8 8-8 22 0 30s22 8 30 0l4-4c24-24 66-24 90 0l170 170c40 40 40 108 0 150l-76 74c-20 20-46 32-76 32-28 0-54-12-74-32l-170-168c-24-26-24-66 0-90l4-4c8-8 8-22 0-30s-22-8-30 0l-4 4c-42 40-42 108 0 150l170 170c28 28 64 42 104 42s78-14 106-42l76-76c58-58 58-152 0-212l-170-168c-20-20-46-32-76-32zM278-40c-40 0-78 14-106 42l-76 76c-58 58-58 152 0 210l170 170c20 20 46 32 76 32 28 0 54-12 74-32l4-4c8-8 8-22 0-30s-22-8-30 0l-4 4c-24 24-66 24-90 0l-170-170c-40-42-40-108 0-150l76-76c20-20 46-30 76-30 28 0 54 10 74 30l170 170c24 24 24 66 0 90l-4 4c-8 8-8 22 0 30s22 8 30 0l4-4c42-42 42-108 0-150l-170-170c-28-28-64-42-104-42zM346 230c-6 0-10 2-16 6-8 10-8 22 0 30l332 332c10 10 22 10 32 0 8-8 8-20 0-30l-332-332c-6-4-10-6-16-6z" /> -<glyph unicode="" glyph-name="uniE017" d="M248 382l-156-62 54 96c6 10 4 20-4 28-64 50-100 122-100 198 0 152 154 276 342 276s342-124 342-276h42c0 176-172 318-384 318s-384-142-384-318c0-82 36-162 100-220l-76-134c-4-8-4-18 2-24 4-6 10-8 16-8 4 0 6 0 8 2l208 82c18-6 36-10 56-12l6 42c-20 4-40 8-58 14-4 0-10 0-14-2zM944 102c52 46 80 108 80 176 0 150-146 276-320 276s-320-126-320-276c0-154 140-282 306-282 0 0 0 0 0 0 42 0 82 8 122 24l162-62c2 0 4 0 8 0 6 0 12 2 16 6 6 6 6 16 2 24zM818 62c-2 2-4 2-8 2-2 0-6 0-8-2-36-16-74-24-112-24 0 0 0 0 0 0-142 0-264 110-264 240 0 126 128 234 278 234s278-108 278-234c0-62-26-114-78-154-8-6-10-18-6-26l40-80z" /> -<glyph unicode="" glyph-name="uniE018" d="M512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470zM832 426c-12 0-22-8-22-20 0-166-134-300-298-300s-298 134-298 300c0 12-10 20-22 20s-22-8-22-20c0-188 154-342 342-342s342 154 342 342c0 12-10 20-22 20zM234 512c12 0 22 10 22 22 0 34 28 64 64 64s64-30 64-64c0-12 10-22 22-22s20 10 20 22c0 58-48 106-106 106s-106-48-106-106c0-12 8-22 20-22zM618 512c12 0 22 10 22 22 0 34 28 64 64 64s64-30 64-64c0-12 10-22 22-22s20 10 20 22c0 58-48 106-106 106s-106-48-106-106c0-12 8-22 20-22z" /> -<glyph unicode="" glyph-name="uniE019" d="M1022 582c-2 10-10 16-20 16h-346l-124 348c-2 8-10 14-20 14s-18-6-20-14l-124-348h-346c-10 0-18-6-20-16-4-8 0-18 6-22l288-226-124-370c-4-8 0-18 8-24 6-6 18-6 24 0l308 226 308-226c4-2 8-4 12-4s8 2 12 4c8 6 12 16 8 24l-124 370 288 226c8 4 10 14 6 22zM690 358c-6-6-10-14-6-24l106-320-266 196c-4 2-8 4-12 4s-8-2-12-4l-266-196 106 320c4 10 0 18-6 24l-250 196h300c10 0 18 6 20 14l108 306 108-306c2-8 10-14 20-14h300z" /> -<glyph unicode="" glyph-name="uniE01A" d="M806 464l-560 490c-10 8-22 8-30-2-8-8-8-22 2-30l542-474-542-474c-10-8-10-22-2-30 4-6 10-8 16-8 4 0 10 2 14 6l560 490c6 4 8 10 8 16s-2 12-8 16z" /> -<glyph unicode="" glyph-name="uniE01B" d="M0 576v-42c330 0 598-268 598-598h42c0 352-288 640-640 640zM0 960v-42c542 0 982-440 982-982h42c0 564-460 1024-1024 1024zM128 192c-70 0-128-58-128-128s58-128 128-128c70 0 128 58 128 128s-58 128-128 128zM128-22c-48 0-86 38-86 86s38 86 86 86c48 0 86-38 86-86s-38-86-86-86z" /> -<glyph unicode="" glyph-name="uniE01C" d="M534 362h-342c-12 0-22-8-22-20v-256c0-12 10-22 22-22h342c12 0 20 10 20 22v256c0 12-8 20-20 20zM512 106h-298v214h298zM1024 640c0 4 0 6-2 8 0 0 0 0 0 0 0 2 0 2 0 2l-86 170c-4 8-10 12-18 12h-812c-8 0-14-4-18-12l-86-170c0 0 0-2 0-2s0 0 0 0c-2-2-2-6-2-8 0 0 0 0 0 0v-42c0-60 34-112 86-136v-504c0-12 8-22 20-22h812c12 0 20 10 20 22v504c52 24 86 76 86 136v42c0 0 0 0 0 0zM968 662h-162l-32 128h130zM768 518c-28-42-80-70-128-70-40 0-78 16-106 46v124h234zM256 618h234v-124c-28-30-66-46-106-46-48 0-100 28-128 70zM490 790v-128h-228l32 128zM730 790l32-128h-228v128zM120 790h130l-32-128h-162zM42 598v20h172v-96c-24-22-44-32-64-32-60 0-108 48-108 108zM640-22v342h170v-342zM896-22h-42v364c0 12-10 20-22 20h-214c-12 0-20-8-20-20v-364h-470v472c8-2 14-2 22-2 34 0 62 16 82 32 38-46 96-74 152-74 48 0 92 18 128 48 36-30 80-48 128-48 56 0 114 28 152 74 22-16 48-32 82-32 8 0 16 0 22 2zM982 598c0-60-48-108-108-108-20 0-40 10-64 32v96h172zM768 128c0-12-10-22-22-22s-20 10-20 22c0 12 8 22 20 22s22-10 22-22z" /> -<glyph unicode="" glyph-name="uniE01D" d="M874 874c-8 8-22 8-30 0s-8-22 0-30c88-88 138-206 138-332s-50-244-138-332c-8-8-8-22 0-30 4-4 10-6 14-6 6 0 12 2 16 6 96 96 150 226 150 362s-54 266-150 362zM754 754c-8 8-22 8-30 0-10-10-10-22 0-30 56-58 86-132 86-212s-30-154-86-212c-10-8-10-22 0-30 4-4 8-6 14-6s12 2 16 6c64 66 100 150 100 242s-36 176-100 242zM42 512c0 126 50 244 138 332 8 8 8 22 0 30s-22 8-30 0c-96-96-150-226-150-362s54-266 150-362c4-4 10-6 16-6 4 0 10 2 14 6 8 8 8 22 0 30-88 88-138 206-138 332zM214 512c0 80 30 154 86 212 10 8 10 22 0 30-8 8-22 8-30 0-64-66-100-150-100-242s36-176 100-242c4-4 10-6 16-6s10 2 14 6c10 8 10 22 0 30-56 58-86 132-86 212zM512 662c-82 0-150-68-150-150 0-76 56-138 128-148v-342c0-12 10-22 22-22s22 10 22 22v342c72 10 128 72 128 148 0 82-68 150-150 150zM512 406c-58 0-106 48-106 106s48 106 106 106c58 0 106-48 106-106s-48-106-106-106z" /> -<glyph unicode="" glyph-name="uniE01E" d="M896 746h-768c-48 0-86-38-86-84v-470c0-48 38-86 86-86h768c48 0 86 38 86 86v470c0 46-38 84-86 84zM896 704c2 0 6 0 8 0l-392-314-392 314c2 0 6 0 8 0zM938 192c0-24-18-42-42-42h-768c-24 0-42 18-42 42v470c0 4 0 8 2 12l410-328c4-4 10-4 14-4s10 0 14 4l410 328c2-4 2-8 2-12z" /> -<glyph unicode="" glyph-name="uniE01F" d="M542 916c-8 4-18 2-24-4l-292-294h-140c-48 0-86-38-86-84v-128c0-48 38-86 86-86h140l292-292c4-4 10-6 16-6 2 0 4 0 8 0 8 4 12 12 12 20v854c0 8-4 16-12 20zM512 94l-262 262c-4 4-10 6-16 6h-148c-24 0-44 20-44 44v128c0 22 20 42 44 42h148c6 0 12 2 16 6l262 262zM760 742c-8 8-22 6-30-4-8-8-6-22 4-30 72-60 120-154 120-238 0-86-48-180-120-240-10-8-12-22-4-30 4-6 10-8 16-8s10 2 14 4c82 70 136 176 136 274 0 96-54 204-136 272zM674 636c-10 6-22 4-30-4-6-10-4-24 4-30 36-26 78-70 78-132 0-64-42-108-78-132-8-8-10-22-4-30 4-6 10-10 18-10 4 0 8 2 12 4 60 44 94 104 94 168 0 62-34 122-94 166z" /> -<glyph unicode="" glyph-name="uniE020" d="M234 320c12 0 22 10 22 22s-10 20-22 20h-106c-24 0-42 20-42 44v128c0 22 18 42 42 42h150c4 0 10 2 14 6l262 262v-182c0-12 10-22 22-22s22 10 22 22v234c0 8-6 16-14 20s-16 2-24-4l-292-294h-140c-48 0-86-38-86-84v-128c0-48 38-86 86-86zM704 490c-12 0-22-8-22-20 0-48-38-86-84-86v22c0 12-10 20-22 20s-22-8-22-20v-312l-112 114c-8 8-22 8-30 0-8-10-8-22 0-32l148-148c4-4 10-6 16-6 2 0 6 0 8 0 8 4 14 12 14 20v300c70 0 128 56 128 128 0 12-10 20-22 20zM976 912c-10 8-22 8-32 0l-852-854c-8-8-8-22 0-30 4-4 10-6 14-6 6 0 12 2 16 6l854 852c8 10 8 22 0 32z" /> -<glyph unicode="" glyph-name="uniE021" d="M912 572c-40 38-94 60-150 62-54 100-158 164-272 164-154 0-282-116-304-266-44 4-86-10-122-38-40-34-64-84-64-136 0-50 16-92 46-122 52-50 128-52 138-52 0 0 0 0 0 0h82c12 0 22 10 22 20 0 12-10 22-22 22h-82c0 0-66 0-108 40-22 22-34 54-34 92 0 40 18 78 50 102 30 26 70 36 110 26l24-4 2 24c8 140 124 250 262 250 102 0 196-60 240-152l6-14 14 2c50 2 96-16 132-52 36-34 56-80 56-130 0-154-146-182-164-184h-60c-12 0-22-10-22-22 0-10 10-20 22-20h66c2 0 202 28 202 226 0 60-26 118-70 162zM604 122l-92-92v376c0 12-10 20-22 20s-20-8-20-20v-376l-92 92c-8 8-22 8-30 0s-8-22 0-30l128-128c2-2 4-4 6-6 4 0 6 0 8 0 4 0 6 0 8 0 4 2 6 4 8 6l128 128c8 8 8 22 0 30s-22 8-30 0z" /> -<glyph unicode="" glyph-name="uniE022" d="M912 572c-40 38-94 60-150 62-54 100-158 164-272 164-154 0-282-116-304-266-44 4-86-10-122-38-40-34-64-84-64-136 0-50 16-92 46-122 52-50 128-52 138-52 0 0 0 0 0 0h122c12 0 22 10 22 20 0 12-10 22-22 22h-122c0 0-66 0-108 40-22 22-34 54-34 92 0 40 18 78 50 102 30 26 70 36 110 26l24-4 2 24c8 140 124 250 262 250 102 0 196-60 240-152l6-14 14 2c50 2 96-16 132-52 36-34 56-80 56-130 0-154-146-182-164-184h-100c-12 0-22-10-22-22 0-10 10-20 22-20h106c2 0 202 28 202 226 0 60-26 118-70 162zM506 464c-2 2-4 2-8 4s-10 2-16 0c-2-2-4-2-6-4l-128-128c-8-10-8-22 0-32 8-8 22-8 30 0l92 92v-418c0-12 8-20 20-20s22 8 22 20v418l92-92c4-4 10-6 14-6 6 0 12 2 16 6 8 10 8 22 0 32z" /> -<glyph unicode="" glyph-name="uniE023" d="M1002 106c-58 0-106 48-106 108v234c0 156-96 296-240 356-16 66-74 114-144 114-68 0-130-48-146-114-142-60-238-200-238-356v-234c0-60-48-108-106-108-12 0-22-8-22-20s10-22 22-22h342c10-72 72-128 148-128s138 56 148 128h342c12 0 22 10 22 22s-10 20-22 20zM512-22c-52 0-94 38-104 86h208c-10-48-52-86-104-86zM126 106c28 28 44 66 44 108v234c0 142 90 270 224 320 6 2 12 8 12 16 8 52 54 90 106 90s98-38 106-90c0-8 6-14 12-16 134-50 224-178 224-320v-234c0-42 16-80 44-108z" /> -<glyph unicode="" glyph-name="uniE024" d="M1024 726v0c0 2 0 4-2 6 0 0 0 2 0 2 0 2-2 4-4 6 0 0 0 0 0 0-2 2-2 2-4 4 0 0-2 0-2 0s0 0 0 0l-492 214c-4 2-12 2-16 0l-492-214c0 0 0 0 0 0s-2 0-2 0c-2-2-2-2-4-4 0 0 0 0 0 0-2-2-4-4-4-6 0 0 0-2 0-2-2-2-2-4-2-6v0c0 0 0 0 0 0v-556c0-8 6-16 12-18l492-214c0 0 0 0 0 0 2-2 6-2 8-2s6 0 8 2c0 0 0 0 0 0l492 214c6 2 12 10 12 18v556c0 0 0 0 0 0zM512 916l438-190-198-86-396 208zM512 536l-438 190 232 100 396-208zM42 692l448-194v-508l-448 194zM982 184l-448-194v508l448 194zM392 446l-234 106c-6 4-14 4-20 0s-10-12-10-18v-214c0-8 4-16 12-20l236-106c2-2 4-2 8-2s8 2 12 4c6 4 10 10 10 18v212c0 10-6 16-14 20zM362 246l-192 88v166l192-88z" /> -<glyph unicode="" glyph-name="uniE025" d="M938 426c-28 0-56-14-70-38l-116 48c10 24 16 50 16 76 0 46-16 86-42 120l170 170c12-8 26-12 42-12 48 0 86 38 86 84 0 48-38 86-86 86-46 0-84-38-84-86 0-16 4-30 12-42l-170-170c-34 26-74 42-120 42-72 0-134-40-166-98l-240 104c0 4 0 10 0 16 0 46-38 84-84 84-48 0-86-38-86-84 0-48 38-86 86-86 26 0 48 12 64 32l242-104c-6-18-8-36-8-56 0-46 16-88 42-120l-298-298c-12 8-26 12-42 12-48 0-86-38-86-84 0-48 38-86 86-86 46 0 84 38 84 86 0 16-4 30-12 42l298 298c28-22 62-36 98-40v-218c-36-10-64-44-64-82 0-48 38-86 86-86s86 38 86 86c0 38-28 72-64 82v218c54 6 102 34 132 76l124-50c0-2 0-4 0-6 0-48 38-86 84-86 48 0 86 38 86 86 0 46-38 84-86 84zM86 682c-24 0-44 20-44 44 0 22 20 42 44 42 22 0 42-20 42-42 0-24-20-44-42-44zM938 918c24 0 44-20 44-44 0-22-20-42-44-42-22 0-42 20-42 42 0 24 20 44 42 44zM86-22c-24 0-44 20-44 44 0 22 20 42 44 42 22 0 42-20 42-42 0-24-20-44-42-44zM618 22c0-24-18-44-42-44s-42 20-42 44c0 22 18 42 42 42s42-20 42-42zM576 362c-82 0-150 68-150 150s68 150 150 150c82 0 150-68 150-150s-68-150-150-150zM938 298c-22 0-42 20-42 44 0 22 20 42 42 42 24 0 44-20 44-42 0-24-20-44-44-44z" /> -<glyph unicode="" glyph-name="uniE026" d="M806-26l-542 474 542 474c10 8 10 22 2 30-8 10-20 10-30 2l-560-490c-6-4-8-10-8-16s2-12 8-16l560-490c4-4 10-6 14-6 6 0 12 2 16 8 8 8 8 22-2 30z" /> -<glyph unicode="" glyph-name="uniE027" d="M1018 182l-490 560c-8 10-24 10-32 0l-490-560c-8-10-8-22 2-30 8-8 22-8 30 2l474 542 474-542c4-6 10-8 16-8s10 2 14 6c10 8 10 20 2 30z" /> -<glyph unicode="" glyph-name="uniE028" d="M512 960c-212 0-384-172-384-384v-576c0-36 28-64 64-64h22c34 0 64 28 64 64 0 12 8 22 20 22s22-10 22-22c0-36 28-64 64-64h42c36 0 64 28 64 64 0 12 10 22 22 22s22-10 22-22c0-36 28-64 64-64h42c36 0 64 28 64 64 0 12 10 22 22 22s20-10 20-22c0-36 30-64 64-64h22c36 0 64 28 64 64v576c0 212-172 384-384 384zM854 0c0-12-10-22-22-22h-22c-12 0-20 10-20 22 0 36-30 64-64 64-36 0-64-28-64-64 0-12-10-22-22-22h-42c-12 0-22 10-22 22 0 36-28 64-64 64s-64-28-64-64c0-12-10-22-22-22h-42c-12 0-22 10-22 22 0 36-28 64-64 64-34 0-64-28-64-64 0-12-8-22-20-22h-22c-12 0-22 10-22 22v576c0 188 154 342 342 342s342-154 342-342zM640 662c-48 0-86-38-86-86v-128c0-48 38-86 86-86s86 38 86 86v128c0 48-38 86-86 86zM682 448c0-24-18-42-42-42s-42 18-42 42c22 0 42 20 42 42 0 24-20 44-42 44v42c0 24 18 42 42 42s42-18 42-42zM384 662c-48 0-86-38-86-86v-128c0-48 38-86 86-86s86 38 86 86v128c0 48-38 86-86 86zM426 448c0-24-18-42-42-42s-42 18-42 42c22 0 42 20 42 42 0 24-20 44-42 44v42c0 24 18 42 42 42s42-18 42-42z" /> -<glyph unicode="" glyph-name="uniE029" d="M362 576c-22 0-42-20-42-42 0-24 20-44 42-44 24 0 44 20 44 44 0 22-20 42-44 42zM576 234c0-22-20-42-42-42-24 0-44 20-44 42 0 24 20 44 44 44 22 0 42-20 42-44zM298 128c0-24-18-42-42-42s-42 18-42 42c0 24 18 42 42 42s42-18 42-42zM170 342c0-24-18-44-42-44s-42 20-42 44c0 22 18 42 42 42s42-20 42-42zM512 128c0-24-20-42-42-42-24 0-44 18-44 42s20 42 44 42c22 0 42-18 42-42zM640 342c0-24-20-44-42-44-24 0-44 20-44 44 0 22 20 42 44 42 22 0 42-20 42-42zM854 682c0-22-20-42-44-42-22 0-42 20-42 42 0 24 20 44 42 44 24 0 44-20 44-44zM854 470c0-24-20-44-44-44-22 0-42 20-42 44 0 22 20 42 42 42 24 0 44-20 44-42zM938 834c0 0 0 0 0 0 0 6-2 10-6 14 0 0-2 0-2 0s0 2 0 2c-2 0-4 2-6 2 0 0 0 0 0 0l-342 108c-4 0-8 0-12 0l-342-108c0 0 0 0 0 0-2 0-4-2-6-2 0 0 0-2 0-2-2 0-2 0-2 0-4-4-6-8-6-14 0 0 0 0 0 0 0-2 0-2 0-2v-216l-200-62c0 0 0 0 0 0-2-2-4-2-4-4-2 0-2 0-2 0s0 0 0 0c-4-4-6-8-8-14 0 0 0 0 0-2 0 0 0 0 0 0v-428c0-8 6-16 12-18l342-150c0 0 0 0 2 0 2-2 4-2 6-2 4 0 6 0 8 2 0 0 0 0 2 0l340 150c8 2 14 10 14 18v200l200 80c8 2 12 10 12 20v426c0 0 0 0 0 2zM576 916l276-86-276-104-276 104zM362 428l-274 104 274 86 276-86zM256 802l298-112v-88l-184 58c-6 2-10 2-14 0l-100-30zM42 502l300-112v-400l-300 130zM682 120l-298-130v400l298 112zM896 420l-170-68v182c0 0 0 0 0 0 0 2 0 2-2 2 0 6-2 10-6 14 0 0 0 0 0 0-2 0-2 0-2 0-2 2-4 2-6 4 0 0 0 0 0 0l-112 34v102l298 112zM576 790c24 0 42 18 42 42s-18 42-42 42c-24 0-42-18-42-42s18-42 42-42z" /> -<glyph unicode="" glyph-name="uniE02A" d="M1002 662h-404v20c0 60-48 108-108 108-58 0-106-48-106-108v-106h-170c-8 0-16-4-20-10l-104-190-84-84c-4-4-6-10-6-14v-236c0-12 10-20 22-20h86c10-50 54-86 106-86 50 0 94 36 104 86h388c10-50 54-86 104-86 52 0 96 36 106 86h86c12 0 22 8 22 20v598c0 12-10 22-22 22zM982 150h-556v468h556zM706 64h-388c-4 16-10 30-20 42h428c-10-12-16-26-20-42zM426 682c0 36 30 64 64 64 36 0 64-28 64-64v-20h-128zM122 348c2 0 2 2 4 4l100 182h158v-384h-342v118zM42 106h86c-8-12-16-26-20-42h-66zM214-22c-36 0-64 30-64 64 0 36 28 64 64 64 34 0 64-28 64-64 0-34-30-64-64-64zM810-22c-34 0-64 30-64 64 0 36 30 64 64 64 36 0 64-28 64-64 0-34-28-64-64-64zM916 64c-4 16-12 30-20 42h86v-42zM182 322c4-2 6-2 10-2 8 0 16 4 20 12l58 116h50c12 0 22 10 22 22s-10 20-22 20h-64c-8 0-16-4-20-12l-64-128c-4-10 0-22 10-28zM554 426v-84c0-12 10-22 22-22h64v-64c0-12 10-22 22-22h84c12 0 22 10 22 22v64h64c12 0 22 10 22 22v84c0 12-10 22-22 22h-64v64c0 12-10 22-22 22h-84c-12 0-22-10-22-22v-64h-64c-12 0-22-10-22-22zM598 406h64c12 0 20 8 20 20v64h44v-64c0-12 8-20 20-20h64v-44h-64c-12 0-20-8-20-20v-64h-44v64c0 12-8 20-20 20h-64z" /> -<glyph unicode="" glyph-name="uniE02B" d="M1002 874h-148v64c0 12-10 22-22 22h-128c-12 0-22-10-22-22v-64h-340v64c0 12-10 22-22 22h-128c-12 0-22-10-22-22v-64h-148c-12 0-22-8-22-20v-896c0-12 10-22 22-22h980c12 0 22 10 22 22v896c0 12-10 20-22 20zM726 918h84v-128h-84zM214 918h84v-128h-84zM170 832v-64c0-12 10-22 22-22h128c12 0 22 10 22 22v64h340v-64c0-12 10-22 22-22h128c12 0 22 10 22 22v64h128v-170h-940v170zM42-22v640h940v-640zM918 448c12 0 20 10 20 22s-8 20-20 20h-150v64c0 12-10 22-22 22s-20-10-20-22v-64h-192v64c0 12-10 22-22 22s-22-10-22-22v-64h-192v64c0 12-8 22-20 22s-22-10-22-22v-64h-150c-12 0-20-8-20-20s8-22 20-22h150v-128h-150c-12 0-20-10-20-22s8-20 20-20h150v-128h-150c-12 0-20-10-20-22s8-22 20-22h150v-64c0-12 10-20 22-20s20 8 20 20v64h192v-64c0-12 10-20 22-20s22 8 22 20v64h192v-64c0-12 8-20 20-20s22 8 22 20v64h150c12 0 20 10 20 22s-8 22-20 22h-150v128h150c12 0 20 8 20 20s-8 22-20 22h-150v128zM298 448h192v-128h-192zM298 150v128h192v-128zM726 150h-192v128h192zM726 320h-192v128h192z" /> -<glyph unicode="" glyph-name="uniE02C" d="M1002 704h-106v64c0 12-10 22-22 22h-532v64c0 12-10 20-22 20h-298c-12 0-22-8-22-20v-704c0-72 58-128 128-128h746c84 0 150 66 150 148v512c0 12-10 22-22 22zM214 682v-532c0-48-38-86-86-86s-86 38-86 86v682h256v-64c0-12 10-22 22-22h534v-42h-620c-12 0-20-10-20-22zM982 170c0-58-48-106-108-106h-650c20 22 32 52 32 86v512h726z" /> -<glyph unicode="" glyph-name="uniE02D" d="M1002 406h-240l-80 200c-4 8-14 14-22 12-10 0-18-8-20-16l-104-486-110 698c-2 10-10 18-20 18s-18-6-22-16l-122-410h-240c-12 0-22-10-22-22s10-22 22-22h256c8 0 18 6 20 16l100 336 114-718c2-10 10-18 20-18 0 0 2 0 2 0 10 0 18 8 20 18l114 528 58-148c4-8 12-14 20-14h256c12 0 22 10 22 22s-10 22-22 22z" /> -<glyph unicode="" glyph-name="uniE02E" d="M810 918h-768v-768h768zM768 874v-554h-92l-122 306c-4 8-12 14-20 14s-16-4-20-12l-92-204-86 102c-4 4-10 8-18 8-6-2-12-6-16-12l-122-202h-94v554zM630 320h-400l92 156 88-106c4-6 12-8 20-8 6 2 14 6 16 14l86 188zM86 192v86h682v-86zM876 666c-10 2-22-8-22-18-2-12 8-24 20-24l62-6-54-594-616 56c-12 0-22-8-22-20-2-12 6-22 18-24l660-60 62 680zM256 618c48 0 86 38 86 86s-38 86-86 86c-48 0-86-38-86-86s38-86 86-86zM256 746c24 0 42-18 42-42s-18-42-42-42c-24 0-42 18-42 42s18 42 42 42z" /> -<glyph unicode="" glyph-name="uniE02F" d="M1024 346c-2 2-2 2-2 4 0 0 0 0 0 0l-136 312h32c12 0 20 8 20 20s-8 22-20 22h-346c-8 36-42 64-82 64-38 0-72-28-82-64h-302c-12 0-20-10-20-22s8-20 20-20h32l-136-312c0 0 0 0 0 0 0-2 0-2-2-4 0-2 0-4 0-4s0 0 0 0c0-94 76-172 170-172s172 78 172 172c0 0 0 0 0 0s-2 2-2 4c0 2 0 2 0 4 0 0 0 0 0 0l-136 312h204c8-30 32-54 62-62v-622h-108c-12 0-20-8-20-20s8-22 20-22h300c12 0 20 10 20 22s-8 20-20 20h-150v622c30 8 54 32 60 62h248l-136-312c0 0 0 0 0 0 0-2 0-2 0-4s-2-4-2-4c0 0 0 0 0 0 0-94 78-172 172-172s170 78 170 172c0 0 0 0 0 0s0 2 0 4zM170 630l118-268h-234zM170 214c-62 0-116 46-126 106h252c-10-60-62-106-126-106zM490 640c-22 0-42 20-42 42 0 24 20 44 42 44 24 0 44-20 44-44 0-22-20-42-44-42zM970 362h-234l118 268zM854 214c-64 0-116 46-126 106h252c-10-60-64-106-126-106z" /> -<glyph unicode="" glyph-name="uniE030" d="M1014 678c-12 16-32 26-54 26h-512c-44 0-92-34-110-78l-120-298c-10-24-8-48 4-66s32-28 54-28h512c44 0 92 34 110 78l120 298c10 26 8 50-4 68zM978 626l-120-298c-10-28-42-50-70-50h-512c-6 0-14 0-18 8-4 6-4 16 0 26l120 298c10 28 42 52 70 52h512c6 0 14-2 18-8s4-16 0-28zM894 616l-290-162-154 158c-8 8-22 8-30 0s-8-22 0-30l166-170c4-4 10-6 14-6s8 0 12 2l302 170c10 6 14 20 8 30s-18 14-28 8zM460 444l-148-106c-10-8-12-20-6-30 4-6 10-10 18-10 4 0 8 2 12 4l150 108c8 6 12 20 4 30-6 8-20 10-30 4zM748 438c-6 10-20 14-30 6-10-6-14-18-8-28l64-106c4-8 12-12 18-12 4 0 8 2 12 4 10 6 14 18 8 28zM22 662h212c12 0 22 8 22 20s-10 22-22 22h-212c-12 0-22-10-22-22s10-20 22-20zM106 320h-84c-12 0-22-10-22-22s10-20 22-20h84c12 0 22 8 22 20s-10 22-22 22zM150 534h-128c-12 0-22-10-22-22s10-22 22-22h128c12 0 20 10 20 22s-8 22-20 22z" /> -<glyph unicode="" glyph-name="uniE031" d="M618 918h-256c-12 0-20-10-20-22v-256c0-12 8-22 20-22h256c12 0 22 10 22 22v256c0 12-10 22-22 22zM598 662h-214v212h214zM278 918h-256c-12 0-22-10-22-22v-256c0-12 10-22 22-22h256c12 0 20 10 20 22v256c0 12-8 22-20 22zM256 662h-214v212h214zM960 918h-256c-12 0-22-10-22-22v-256c0-12 10-22 22-22h256c12 0 22 10 22 22v256c0 12-10 22-22 22zM938 662h-212v212h212zM618 576h-256c-12 0-20-10-20-22v-256c0-12 8-20 20-20h256c12 0 22 8 22 20v256c0 12-10 22-22 22zM598 320h-214v214h214zM278 576h-256c-12 0-22-10-22-22v-256c0-12 10-20 22-20h256c12 0 20 8 20 20v256c0 12-8 22-20 22zM256 320h-214v214h214zM960 576h-256c-12 0-22-10-22-22v-256c0-12 10-20 22-20h256c12 0 22 8 22 20v256c0 12-10 22-22 22zM938 320h-212v214h212zM618 234h-256c-12 0-20-8-20-20v-256c0-12 8-22 20-22h256c12 0 22 10 22 22v256c0 12-10 20-22 20zM598-22h-214v214h214zM278 234h-256c-12 0-22-8-22-20v-256c0-12 10-22 22-22h256c12 0 20 10 20 22v256c0 12-8 20-20 20zM256-22h-214v214h214zM960 234h-256c-12 0-22-8-22-20v-256c0-12 10-22 22-22h256c12 0 22 10 22 22v256c0 12-10 20-22 20zM938-22h-212v214h212z" /> -<glyph unicode="" glyph-name="uniE032" d="M1002 918h-980c-12 0-22-10-22-22v-214c0-12 10-20 22-20h980c12 0 22 8 22 20v214c0 12-10 22-22 22zM982 704h-940v170h940zM1002 576h-980c-12 0-22-10-22-22v-212c0-12 10-22 22-22h980c12 0 22 10 22 22v212c0 12-10 22-22 22zM982 362h-940v172h940zM1002 234h-980c-12 0-22-8-22-20v-214c0-12 10-22 22-22h980c12 0 22 10 22 22v214c0 12-10 20-22 20zM982 22h-940v170h940z" /> -<glyph unicode="" glyph-name="uniE033" d="M1002 960h-980c-12 0-22-10-22-22v-980c0-12 10-22 22-22h980c12 0 22 10 22 22v980c0 12-10 22-22 22zM982-22h-940v940h940zM576 662h298c12 0 22 8 22 20s-10 22-22 22h-298c-12 0-22-10-22-22s10-20 22-20zM448 534h426c12 0 22 8 22 20s-10 22-22 22h-426c-12 0-22-10-22-22s10-20 22-20zM448 406h426c12 0 22 8 22 20s-10 22-22 22h-426c-12 0-22-10-22-22s10-20 22-20zM448 278h426c12 0 22 8 22 20s-10 22-22 22h-426c-12 0-22-10-22-22s10-20 22-20zM448 150h298c12 0 22 8 22 20s-10 22-22 22h-298c-12 0-22-10-22-22s10-20 22-20zM150 618h170c12 0 22 10 22 22v170c0 12-10 22-22 22h-170c-12 0-22-10-22-22v-170c0-12 10-22 22-22zM170 790h128v-128h-128zM150 320h170c12 0 22 10 22 22v212c0 12-10 22-22 22h-170c-12 0-22-10-22-22v-212c0-12 10-22 22-22zM170 534h128v-172h-128zM150 64h170c12 0 22 10 22 22v170c0 12-10 22-22 22h-170c-12 0-22-10-22-22v-170c0-12 10-22 22-22zM170 234h128v-128h-128z" /> -<glyph unicode="" glyph-name="uniE034" d="M554 192c0-24-18-42-42-42s-42 18-42 42c0 24 18 42 42 42s42-18 42-42zM512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470zM512 704c-82 0-150-66-150-150 0-12 10-20 22-20s22 8 22 20c0 60 48 108 106 108s106-48 106-108c0-58-48-106-106-106-12 0-22-10-22-22v-128c0-12 10-20 22-20s22 8 22 20v108c72 12 128 74 128 148 0 84-68 150-150 150z" /> -<glyph unicode="" glyph-name="uniE035" d="M498 960h-8c-274-4-494-228-490-500 4-270 216-482 482-482h10c130 2 254 56 348 150s144 220 142 350c-6 270-218 482-484 482zM810 158c-86-86-200-134-320-136h-8c-242 0-434 192-440 440-4 246 198 450 450 456h6c244 0 436-194 440-440 2-120-44-234-128-320zM490 384c12 0 22 10 22 22v298c0 12-10 22-22 22s-20-10-20-22v-298c0-12 8-22 20-22zM490 298c0 0 0 0 0 0-24 0-42-20-42-42 0-24 20-42 42-42 0 0 2 0 2 0 22 0 42 20 42 42-2 24-20 42-44 42z" /> -<glyph unicode="" glyph-name="uniE036" d="M576 190c0 0 0 0 0 0-12 0-20-8-20-20l-2-150-512 2v852h514v-150c0-12 10-22 22-22 0 0 0 0 0 0 12 0 20 10 20 22l2 170c0 12-10 22-22 22l-556 2c0 0 0 0 0 0-6 0-12-2-16-6s-6-10-6-16v-896c0-12 10-22 22-22h554c12 0 22 10 22 20v172c0 12-10 20-22 20zM1022 434c-2 4-4 8-8 10l-208 210c-8 8-22 8-30 0s-8-22 0-30l174-176h-714c-12 0-22-10-22-22s10-22 22-22l718 2-180-180c-8-8-8-20 0-30 4-4 10-6 16-6s10 2 14 6l214 214c4 4 6 10 6 14s0 8-2 10z" /> -<glyph unicode="" glyph-name="uniE037" d="M950 768h-876c-40 0-74-34-74-74v-492c0-40 34-74 74-74h876c40 0 74 34 74 74v492c0 40-34 74-74 74zM576 256h-128v192l-96-124-96 124v-192h-128v384h128l96-128 96 128h128zM768 224l-160 224h96v192h128v-192h96z" /> -<glyph unicode="" glyph-name="uniE038" d="M768 192h-512v510l128 2v128h-256v-768h768v320h-128zM512 832l128-128-192-192 128-128 192 192 128-128v384z" /> -<glyph unicode="" glyph-name="uniE039" d="M1008 762c-18 14-44 12-60-6l-436-522-436 522c-16 18-42 20-60 6-18-16-22-42-6-60l470-562c8-10 20-16 32-16s24 6 32 16l470 562c16 18 12 44-6 60z" /> -<glyph unicode="" glyph-name="uniE03A" d="M1014 196l-470 560c-16 20-48 20-64 0l-470-560c-16-18-12-46 6-60 18-16 44-14 60 4l436 522 436-522c10-10 22-14 34-14 10 0 18 2 26 10 18 14 22 42 6 60z" /> -<glyph unicode="" glyph-name="uniE03B" d="M298 448l522 436c18 16 20 42 6 60-16 18-42 22-60 6l-562-470c-10-8-16-20-16-32s6-24 16-32l560-470c8-6 18-10 28-10 12 0 24 6 34 16 14 18 12 44-6 60z" /> -<glyph unicode="" glyph-name="uniE03C" d="M820 480l-562 470c-18 16-44 12-60-6-14-18-12-44 6-60l522-436-522-436c-18-16-20-42-6-60 10-10 22-16 34-16 10 0 20 4 26 10l562 470c10 8 16 20 16 32s-6 24-16 32z" /> -<glyph unicode="" glyph-name="uniE03D" d="M572 448l440 440c16 16 16 42 0 60-18 16-44 16-60 0l-440-440-440 440c-16 16-42 16-60 0-16-18-16-44 0-60l440-440-440-440c-16-16-16-42 0-60 8-8 20-12 30-12 12 0 22 4 30 12l440 440 440-440c8-8 18-12 30-12 10 0 22 4 30 12 16 18 16 44 0 60z" /> -<glyph unicode="" glyph-name="uniE03E" d="M38 960h520c20 0 38-18 38-38v-130c0-22-18-40-38-40h-520c-20 0-38 18-38 40v130c0 20 18 38 38 38zM38 546h948c20 0 38-18 38-40v-130c0-20-18-38-38-38h-948c-20 0-38 18-38 38v130c0 22 18 40 38 40zM854 960h130c20 0 38-18 38-38v-130c0-22-18-40-38-40h-130c-22 0-40 18-40 40v130c0 20 18 38 40 38zM38 144h324c22 0 40-18 40-40v-130c0-20-18-38-40-38h-324c-20 0-38 18-38 38v130c0 22 18 40 38 40zM660 146h324c20 0 38-18 38-40v-130c0-20-18-38-38-38h-324c-22 0-40 18-40 38v130c0 22 18 40 40 40z" /> -<glyph unicode="" glyph-name="uniE03F" d="M1002 490h-674l136 134c8 10 8 22 0 32-10 8-22 8-32 0l-170-172c-2-2-4-4-4-6-2-6-2-12 0-16 0-4 2-6 4-8l170-170c6-4 10-6 16-6s10 2 16 6c8 8 8 22 0 30l-136 134h674c12 0 22 10 22 22s-10 20-22 20zM22 960c-12 0-22-10-22-22v-980c0-12 10-22 22-22s20 10 20 22v980c0 12-8 22-20 22z" /> -<glyph unicode="" glyph-name="uniE040" d="M1022 434c0 4-2 6-4 8l-170 170c-10 8-22 8-32 0-8-8-8-22 0-30l136-134h-418c-12 0-22-10-22-22s10-20 22-20h418l-136-134c-8-10-8-22 0-32 6-4 10-6 16-6s10 2 16 6l170 172c2 2 4 4 4 6 2 6 2 12 0 16zM320 960h-298c-12 0-22-10-22-22v-980c0-12 10-22 22-22h298c12 0 22 10 22 22v980c0 12-10 22-22 22zM298-22h-256v940h256z" /> -<glyph unicode="" glyph-name="uniE041" d="M1002 790h-64v64c0 12-8 20-20 20h-256c-62 0-104-20-128-58-24 38-68 58-128 58h-256c-12 0-22-8-22-20v-64h-64c-12 0-22-10-22-22v-682c0-12 10-22 22-22h256c38 0 128-10 128-106 0-12 10-22 22-22h128c12 0 20 10 20 22 0 96 90 106 128 106h256c12 0 22 10 22 22v682c0 12-10 22-22 22zM662 832h234v-640h-234c-46 0-82-12-108-34v568c0 74 34 106 108 106zM170 832h236c74 0 106-32 106-106v-568c-24 22-60 34-106 34h-236zM982 106h-236c-94 0-158-48-168-128h-88c-10 80-74 128-170 128h-234v640h42v-576c0-12 10-20 22-20h256c74 0 106-34 106-108 0-12 10-20 22-20s20 8 20 20c0 74 34 108 108 108h256c12 0 20 8 20 20v576h44z" /> -<glyph unicode="" glyph-name="uniE042" d="M962 726c-26-36-56-68-92-94 0-6 0-14 0-24 0-50-6-100-22-148-14-50-36-98-66-142-28-46-64-86-104-122-42-34-90-62-148-82-58-22-120-32-184-32-104 0-198 28-284 84 14-2 28-4 44-4 86 0 162 28 230 80-40 0-76 12-108 36s-54 54-66 92c14-2 26-4 36-4 16 0 32 2 48 8-42 8-78 30-106 62-28 34-42 74-42 118v2c26-14 54-22 84-22-26 16-46 38-60 64-16 28-22 58-22 88 0 34 8 66 24 94 46-56 102-102 168-136s138-54 214-58c-4 16-6 30-6 44 0 50 18 94 54 130s80 54 132 54c52 0 98-20 134-58 42 8 80 22 118 44-14-44-42-78-82-102 36 4 70 14 106 28z" /> -<glyph unicode="" glyph-name="uniE043" d="M710 612l-304-284-92 92c-8 8-22 8-30 0s-8-22 0-30l106-106c4-4 10-6 16-6 4 0 10 2 14 6l320 298c8 8 8 22 0 30s-20 8-30 0zM512 960c-282 0-512-230-512-512s230-512 512-512c282 0 512 230 512 512s-230 512-512 512zM512-22c-258 0-470 212-470 470s212 470 470 470c258 0 470-212 470-470s-212-470-470-470z" /> -<glyph unicode="" glyph-name="uniE044" d="M292 82h146v-146h-146zM586 82h146v-146h-146zM292 374h146v-146h-146zM586 374h146v-146h-146zM292 668h146v-146h-146zM586 668h146v-146h-146zM292 960h146v-146h-146zM586 960h146v-146h-146z" /> -<glyph unicode="" glyph-name="uniE045" d="M982 490h-426v428c0 22-20 42-42 42 0 0 0 0 0 0-24 0-44-20-44-42v-428h-428c-22 0-42-18-42-42s20-42 42-42h428v-428c0-22 18-42 42-42v0c24 0 42 20 42 42l2 428h426c24 0 42 18 42 42s-18 42-42 42z" /> -<glyph unicode="" glyph-name="uniE046" d="M448 960h-214c-12 0-20-10-20-22v-980c0-12 8-22 20-22h214c12 0 22 10 22 22v980c0 12-10 22-22 22zM426-22h-170v940h170zM790 960h-214c-12 0-22-10-22-22v-980c0-12 10-22 22-22h214c12 0 20 10 20 22v980c0 12-8 22-20 22zM768-22h-170v940h170z" /> -<glyph unicode="" glyph-name="uniE047" d="M1012 468l-982 490c-6 4-14 2-20-2s-10-10-10-18v-980c0-8 4-14 10-18 4-2 8-4 12-4 2 0 6 0 8 2l982 490c8 4 12 12 12 20s-4 16-12 20zM42-8v912l912-456z" /> -<glyph unicode="" glyph-name="uniE900" d="M153.425 390.755h-40.693v74.581h42.054v-73.614c3.128 2.261 6.242 4.437 9.3 6.485l2.665 8.249 20.466 63.303c4.026 12.43 9.679 24.007 16.622 34.603h-196.131v-255.644h37.453c18.53 61.98 66.462 111.63 108.263 142.037v0zM845.901 28.458c31.898 8.718 75.72-27.947 103.34-86.215l55.422 23.836c-21.869 59.776-17.885 116.551 9.398 134.628l-55.941 92.459h-153.768l41.549-164.708zM917.904 95.218c12.765 5.504 27.48-0.555 32.908-13.454 5.429-12.956-0.547-27.918-13.312-33.422-12.737-5.461-27.48 0.583-32.908 13.497-5.415 12.956 0.547 27.89 13.312 33.38v0zM662.044 897.536c-16.047 38.357-59.701 56.22-97.504 39.95-37.804-16.256-55.436-60.53-39.389-98.844l116.638-278.699h161.582l-141.326 337.593zM1016.292 248.732v255.644h-552.385l-10.436-32.284c10.044-29.17 10.464-60.956 0.743-91.108l-23.903-73.884-6.481-19.996c3.296-12.729 5.555-25.543 7.014-38.357l585.447-0.014zM911.227 465.337v-74.581h-42.026v74.581h42.026zM533.006 330.368h-42.026v134.969h42.026v-134.969zM659.084 390.755h-42.040v74.581h42.040v-74.581zM785.12 330.368h-41.998v134.969h41.998v-134.969zM354.858 287.843l0.786 2.432c-31.365 43.691-71.245 57.913-109.989 61.454-10.352 0.839-20.578 0.725-30.355 0.014-28.349-15.673-135.070-90.496-107.815-187.435 38.589-137.244-56.587-150.5-75.692-160.284-9.595-4.921 29.275-57.884 102.793-48.725 11.25 1.394 23.3 4.252 36.149 8.96 95.723 35.143 239.251 165.063 184.124 323.584v0zM295.382 496.569c-20.845-7.609-38.225-24.548-45.645-47.531l-20.494-63.36c6.186 0.028 12.512-0.142 18.965-0.64 39.796-2.745 83.996-18.475 119.584-57.159l23.889 73.884c7.477 23.040 3.395 47.047-8.935 65.749l134.93 416.939c7.911 24.491-5.19 50.816-29.345 58.837-6.635 2.204-13.396 2.802-19.919 2.005-17.127-2.133-32.347-14.009-38.098-31.758l-134.93-416.967z" /> -</font></defs></svg> \ No newline at end of file diff --git a/core/client/public/assets/fonts/ghosticons.ttf b/core/client/public/assets/fonts/ghosticons.ttf deleted file mode 100755 index 66b002a5c4c5556ec84c8590cd436ce7a67454ff..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 22420 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 Z?MED30)B3qz>WA)-A=4J<kZjl{9oMl!$JT6 diff --git a/core/client/public/assets/fonts/ghosticons.woff b/core/client/public/assets/fonts/ghosticons.woff deleted file mode 100755 index 858d1792918e63c2219df8d2306091f8107f4aaa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 22496 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 diff --git a/core/client/public/assets/img/404-ghost.png b/core/client/public/assets/img/404-ghost.png deleted file mode 100644 index 3475674dfd69dbf886fb9e5664e771a3385ac5a1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3681 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 diff --git a/core/client/public/assets/img/404-ghost@2x.png b/core/client/public/assets/img/404-ghost@2x.png deleted file mode 100644 index 9260c48514db86b5d15a0794cf7c8c39e148829c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7958 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	J&KE34TAY^D25^7*9(A95p~!hd*X8O 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 diff --git a/core/client/public/assets/img/ghost-logo.png b/core/client/public/assets/img/ghost-logo.png deleted file mode 100644 index a6b693c3494ca0742bedfbde1944ad04a67ae8f2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8638 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 bKltB%u70@lanttiFA1uTG!^UQt-}5vcjV>y diff --git a/core/client/public/assets/img/ghosticon.jpg b/core/client/public/assets/img/ghosticon.jpg deleted file mode 100644 index 2eec85d6b646822eab012cb6935b310b0c6ec7e8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2499 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 MU%%M@!57y3FYB}%XaE2J diff --git a/core/client/public/assets/img/install-welcome.png b/core/client/public/assets/img/install-welcome.png deleted file mode 100644 index cac617a747695b11a83690c1aba3941f196c7d3b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 240204 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�F%V-nFrAIP~s29<A$sgoeE6zMS5?1f-h0Y;GugKb+ZnV_2J+m6c_@%AAo> zEkh6y<H`;IOf>E95It@u`sF3vpIOoW$U(6kAf&QqLrNleuw_mjQc-wirRB^g3Eus9 z3=y6fw{7+Hbfl(IP*EHSB=nqA-UwG6VLQM5!M%955hff=_`tx!jgUmOD1X|(RSqP; ztBhqRAs%X1r=crzB}^jBS>Qu-=xNq?Ko?+cA`3NtmwGGU{_iv>Ghk1VYK`Anoe6<) zyzAsxK*%tKyX&u9bc(#9@So<h={&zmOnbJD&YfFqzNRtCA!?F&-n>QguI10$Cf`}^ zMT@7eTD%*&OjbXkRA(+Pk1xu9Zhch<v~FE)U#yPSE9e2W!2Fspy<0?MlrD1w68fNZ zVpNkE3nBs6C3fw3idQSk=z(Wb7Xua@MyzyBvj^k%Hh&LCEbNb=)?FQ)&N<jy|IQ=L z$67nOSJDmWiG#X=pD;W#gXgja9dtzRcl~V7hp!LGYL>dYI`e+6<*lDrYavY^oOF{7 znRgn5ThDV>JVxd%)_C?*ir09lcYglV)eTFWmDQK{`^W3~tSO$ZJfXeYpZl=2I0N2f z3+?QN<wUR{Pt8p%lf=I^exG74w-|bE+u2Q)E<As%Gi?1Q7E~~B0=rKyCPKVwiQN}( zI>QiRGh>9UFuwxI*HHNcIXR;=KuOW=`q-1^_&-<tPL@_L8CHt>O>2dT@13?-tz{X& zlp7R`q~_8XtcYRKoXQc!J|yXqULky|*xvHVxQIWirtRD1D%5_yn?_J{wKe}Twe;#& zG(JAQ?fkTV51Xs2d;O!bibBqg=2l<xLmM9-Y<aR|Kzj5depJJ;N6*N38EfO!P+P*M zBBw*9Od}Rx4p-8p^+;&gdM_=Z)p$YUA7fd<XUppd2|)aY*QeNaNQ+@?d9fc>Uqhr6 zBR3U-h<czb5oNBgOaMR=7G@o)iUH9@a|sggOc#RyKNGiQbHi+qYSJ<AfWE2agd5M^ z)=s1rxA#|hi&xGhp(NA@3LM<YAR0z#E=(*>B-Yc#)u)N&xPTtL>deumrY5obQnCBd zp2s##yY<IMi@%@jQGeKlb{A!cGjcNbiQf3HKB4SIS8*r`F32-#;Qgu?#^jn`|LnVO zH@0-Uv8$OgxN+#WG4&<NGBNXR>}ic<9w6x8M0ta`{)C}Q!jR;Jck~d=0v4$mJ>~J0 z(6~SuKlIhLz!`q}2Tq^l^yRE8&S{V4p!*_)EuYFrpB_ruYgy)@^TRnrK|!LVcM}8- z2Z=ul3rJ|-PUf3b{z)D|I2i_t`Xd%<p}@};S*TOuG^C?*<oaj$?3|TKVQ<EW1)3Y% zJrB7C`E=Vx;rel`?g6m3KBXOV)*lUXHyTeSbf9i>R((f=Z`)a&v)&ev99P#a8Y8sH zMw$vA<V=qrxQt7mJ9U6tD277y+YT|K-2wvwY`1qtwOO~ekD!Yw1aY&vtJv@w9m9OF zI^tig#;a_<e~;~Vow+me5C2Oi5bwz@+~*DZHfpUK&)M{8c5ZQbcC4`Hc5{1s0loR~ z{U|A|%N&jOzO3xqp>bzeY|Jw0sBO>geQ?XZZ*tH#ZP?pEw<VJx=sF8msVC;=je0UJ zz774h$8VD4b1p6}x~~;eAH!2{&Go<4AXrkB!nRAo*WdTHFQ@I*URU&p8hnb|+^g+^ zFaZb@(|lljg;Wa{-Z7AV3Dgok2BR{z6LF{2FCTluj^K-JX$ZY}P=)lhqr3ws^dN89 z`(gVk@90P{;|;6F$+tPlOl4>KK291gl(s$rlr+3p<?@a^2r37zp!Hr^8Y<dHcNE4Z zJl5(&g;L;;-D4CD)|QiiQtMUT)QfC88gvB=du5HSH6L;ge_6#Z(&W^5Hqi?^Cvfyp zDF%f_$B3zM8AP0Hp+sT5zs_ASeHWKwUMczNk36I)Hqj+^+W{yNG!ms@p;PI23;jn& z&TU<WtBD(beV&S*=6mk0ik|lz8kE6cq`-Xdm!Mu`VD<pMXnu@Xo1nNwH<J8Y$LLrw zmXAJ{bst9Yvtg`%aSwb?t(>1$@`h-Aoq+y?BzULK-bs>_VjybjF%V}R%URhwa2YQ9 zx6;;?8x%&K7aVk?)!)CK3VwAh8_k42<Q>j^{3fv$yvd=Id*CH1j*k~vB*1X_0~XbR z2|tj;5Cx&9o^GV%z<jp1km0IS5*UE-eRpl`5^R{dv(rZFCop&2+i@D|To~(pFnQc; zI+HOO*edhqzk&|ClqvV66tj+9A=!ijm{7aV(`6mReJ1%~NV8{`y+~xSmJjAeNo(L} zraMMbfID6$Z$HJ35&hlt!>*D+Ieft@v}+Z@6?JrdJ-AC^yJO4>;Nl)kS;^4Q9UWL+ zPIO<w&KoPv=}k3(VQf0<xuCHU?|>gjqnxIk{w6L@(+u*eD4k%S4B5nlwvbxpW5szX zm9<(5_2`jvc?8TEY;$^jytU{5Z~swYfWA3py%vqT`+Pf90R$xZJVKl%gZsHJidOy) zRo@xNX8g6CL}+4E1T|U&p{QM|OT}zr?_INYDXLbL(Aa7OHEY+1(nXcnYS*rssVX%a zwW$$r{?GHg@6-2uxbq?T^t(^a@0{zLb6u)-c?ltebm?fgY;E&Qb93<1$4^g_SkIRE z%61rxK<xvh;15$DzZG>BT{w91koQIecBG1~#C4(CsRZ1P8bzbCxKgt4=(%I7v`iI% zMM5WOLlmowYt{aUPIwFAzK{Gu&sbZdUgsU`;mGKiEMX);6Bw|7x@lLyCD!|_U5Q^= zH@pNWdl@qPWdMQly?XQVzO8C;iY#7M5Zgq@Icc~~-dH0KINvH|&s2|)1F5;Ugk^Ij z-eq+~2)j?Pir!DC7=FuR(?r>Lc(z|xB4Ji(dv-Lz1_ge}+HWtG+S0y~JnF+O^Gu0~ z33q2@$A0Ehx)Rg<D*y9#;$_;r*C7eajR&Ji<UdmFe9Ntu@{&M#alPvu_NfcsTsGf6 z9=u{Ucz1a|t4u0pf1#U8!0#(_x^p(?4mN@j7igUic%XC{h`|W4=^OsxGkPEW;``KA z;4LH=k)$UyzuX%eRP?-_rq+s)C{O7<i~JrIt+}d2{t{Qh=OLMlwW>2W9>3=A#8Er# z9Z*+4Q+4O4BoN=jmf-iO&VG_nB3oa}Q@MB+JvWz(cK?CPV%IDIi7(kS#9fH-XJif_ zwKskKH1ptGtO;TuP9j6G`14!)1RG*e!X+RSZHUVna(HFD==Jo;qenAu2F+Za`nU)i zYQEhyjmu%_a~-`_1eH^;xF>1mN&NmGV8(PvN{ZvDyk~RhUhtR6Z=F*`L&vCglH9`U zKXELeS>KqCGp>J&#N$c^yayIUMS*Z6Jg86e5c=QwEj$#|6J~$$b|ie3sPM45`S54_ z9A&>H>~uik;6}yR5GmNzG2Yb?;QOL;>isb7^A4EA>e8T0_XC7}uH&up;2e%eg41fk z*wM!?*~?PXl0A(R)azF=`qcxzPc$DE1=_~dLJ)#hYO8~;I4B$fgO0fBDt!0};gptU z0N^{XUV$`7e0dp(@_6nHgBg6mMQ@$5;=7t0FSpt?cF#J|3Fh9D+304H3)jgUly7Wq zekuEKo*{O`c$P^SCY~&7B>F*McES0sL5<D>xfEHKxyQBNK8#xkP$+zqjlG;dM@N4` z|4zBwn_#bd7g_N2iOgh=zjmv_i?4-<EU>E*v|#Jq&M6z_>Z_i5yF%p8KqJLaRUyb9 z7dSXS4IOBW3IAjH!9X+fBH*^z6?_B=E1ijmgSDPo^C(2~8JFJvegrH18uHD^UW=Cz z<sdBDqA&!OW)4#MFyff<tm6koFL?)>fPaBY1A@~~nVWYK5#+kqs3#(q3Ke66A>sK_ z)Mor6JbL};s)%4zOv>)Ut(`VPNgtcr#^ZMl$s#-)XLeOa%A;*NCLn!I%UW~s!2lx5 zwbw6}g%nt*oQvq%T>ND2jlM4Leqqx#I{!MeH=1POqvlMh!Y?^*<{mIEeyy<V^m46r zxV(Z8`s~A5NA%Qgffp$f&8uwZYBujR$tg&s4<<lFXj|EcJ~0;(eR67I`};XAtdsUP zavnBwgH*-7UR#KdA0PDy@Zb8mk(1E@v-SMMGyQcWNa15$96GaMr*mpb*OO>(c58>N zi|l`2F4OWEgeCS;)JSsjuw@4E+egpI1mZn2((H?SRJP)S<Xj)0zZP@p?%KcfU~2yz z<u0RnFDgy(gnhpV);)LWe9bpXN7#v8OW5CSKBcf5w6O2vBj3FO7Npfm(_Y9`A{jg% z6c;2E+yXjj%EchzNG33kTs8}?9gUUM`!mZhKO$ho`PNz@Z10Vd)jp}yJ?O^inwBk< z62Q(OR@+cp%&M!Lk;m`YO?(3g9!}@ZLXhG^{-}PdRuC+~+TqXv3Rk(6>0fdvWM9>3 z2P@MYVmoMy%>qhB&>o@|T!OZ951NCX5_zoNeke9qrGwHy8`6vW*T7K7l!~FVWuY)& z6kPLVW?(6#_LCBO$fMJVv%hbYf+!>I^(U!He^Zl`{$?(DGV*JY8^2M9IGcQ3#FT?8 zxS$X$Cia|R<`3~$N=4`*O?H2gGw;jgrZ-;xuj5-)t+M16j%k6e56`I~d8I#Cg#3u@ zKL2HxBP;8=tGr5@Gc6yJ<|U1MVXoEQ^+sE7;Txi{NwXyjj!J}dyPapS?wK!2gXflV z@X4BkiO$*(C@YFyI|2aiOGc?MX^WJDBeI6on%_Wraa)<;vTe6FZimyA07hB?td~QZ zzeLgqrF(Un#@Ii&7=gIKO3IqJ6$utKvWcH?+M*l^gy_e5&()j1oTzqLefzt>uCUNZ zqITWALdwRE%9}_GRW=dL5ET@t<?3!``&ItHi(_*TeK&5x=D6AxC=1soUrQ;@Iye<m zD;`S_a{?xx4|h&<uzV>YxfU5jjvL?5zQJ;@1<8SR?X)u_bM&hYUL+yBi(n#8>vr|d zOr)e8!NN0x&o$n!-wEAvyC47Y$+wqJL^Uo3ycj_rY`ng2RVUNAv-9h{?S<kf(&hXU z_=Zg|jppjmpv?&(kC`8>xIiXbX}{omI?Y~HM>C*wbZ90s3lcc}{X17q^lWNbkSfmf zU&)_tA@0k`2~`(7zvbj-yp-~bpUIXUETWJfxD*7?xg;5Tct_sm55p8+?H&{E88%U# ze3_TKL^j1ue=gwQal^6K$lj|DYfFQlS%mL{IJpXYpqQ*eZHTk6p`1ko_=_tF$Y<$X z!qZ23)WTAnLPv8b?X|=f5zVws(k@Q$Ye*4r3n{>;_V4-6KCg;I<rClCK50BK2HpxT z&YFHg+k9s2iZHCWrs2mTvk~>M<=B7L)3pEB+5;1!=E7T#ECXV*mH42ki;Wlth9*1m z0p3L)0uMzFeypC96{YeSTI5O`o>?dx9zNTezx?Bz#^X<kk(a@s;|n-s;vUQPJRAvo z)N%PHcef!^q2o|&n0;pbmtTY4t`tui`7!G(j8eAA2L~I#{M`kcSsuhJmAHR-fP|@B zdgWgN1pmNuSuKfts}{hPq(=q`LDeUY43DQD;{(e@FAlic5F?kjS|k68-yP}sN$2-; zj>s_@{K$!ss8e+w7U;6VJY*GCFx;!E(KU+Ui`J>RC-EoU$$cV2oaUW*RZPIyg*Zgh z{^|ACS7|?3Hh#((nQaKQUy?g#x8&R`ZKj1{7~3Pue<59w5uq*IkMA=WRSjc7K&c9@ z|NUN-p1sDM9obT5CbPLnAx+=!mmseOaErv6qI@3|Cj-9Z2A&&0>cheJz1z(j8~%_k zUpQ9)wr0&5v5fyx{ifyc_gH+AQni<zq?9BCb0cxUF0HUfAial8{_}Nbwck-bANnCl zB@=*W5*;O~UX4}~ufF~i@5`J_d?w4}tI$#wC@S1wuOb}#Ly@lGShaXqStZ$3G)m-P zVy>yFq3L*ysK8J2yvxyEUE$fL99gI^_JNx-!lO!#71HzN>iIonS9xdIh%IG%Y+^#e z=y@Df;Q6snXkOOCx#oKh>pJ^=>yHL(PnHzIC>(r>Va?`}5u%YrDRhb$zdZGT#3VtG zhUTBLO#7kcswS4$b9v=j%`-j6v`|c%Az;^>YaZic41$BhwcSEJHmY!^Wa+|U9v|}@ zyzEZw3QefC&<*^Kr($+}T}TbW`*$h|j^i?dDhHu?5orh)d2u9v+q0j=;27A;9+=8h zfrdQlkH^(^k^Y)zW@&#U`D1W~eQHR^h4(43tV#hq(SJRhZ1^s1?)U^`Oif?UtQWRK z7hN0x4D(s9Pz<kUKU+)+y$jilrf1#?|C<kp`}-l?vFNpQbo0$H3K~!Fk6&%4aSqvL zG;|g{^;4wxZq=$dJ+bGl?LBj;Fn?0E0C>G5@6hNE&}{h}ZfW+Lu`h;91x!3eJ_dhm z2wV~d^;2c1Wtc6KJ5S%Up$u(ao8}Az*ID#8#JgnF`ZrxY-f^tJF(ae?UpP3qDo)Id zGgcK~z3Gis=z+)x1Mt$0H2~;v0)FZH1!M$E1Y6YzB%2G}|K(%AO|A+e*=tPWE9LjL zgPK^X<^{c1zF(>uEOU{Ds$J5~9-g-rr9Xe>H2lu`R;3k!eQguRp%CE0pzUXBuJ-=T z%S&`{I@MAza7~jx8K(L|Kv<&}seW<z@$+V_drhIo<CO{lGlh|G5S)K>*0rmcPpaIB z=DI!VKB<uR+>bn6D@8y&LF3vH)wxa?9EK^|+jar(sW@re--_9C%PkB9lmSCj)Z{ z3y0ldC*5I`93`o<jHa-Mc%d6RyKk2dG@M_KDOfvW+r0gqMcqYxoP_)C7g3V$G#x@+ z@n7yYj6CQa|Lx!m@H?nCHgVQCJYMm0(lGrdY6sP<w6SLy>W$kCJZ%cLJe&VMKcD7W z#M92|NmB?yn0<X6G2x=od_lPN1M=OiWZc)ij3^lB<5#h;AhD9$tas+3yuEN%-;Nh& zFR*N;8xr1krCqV`h?od$g28+R^@M_aIFVRSI8$;aJ<$4j-qq4<F!gfD<g|RklN;5Q zk~elWZ<;32&ece|Gj`TQRiG<8KKK4Njj}{8WZT=;A46DA`S~3u+DXz==xo#8FMmFS zHI#piPwA<N*}&NdQW<Cy97x{Gjyj6IbULq4c(S!h8^XRdy;<9V9LBjF>}RwbE%T$& zx%ulYo9;M8-9I`iYdJi+j9r3`l3aAuY$%BrFJ25AeefUCg2sV1{|-Ll&!~j-nykNS zPhTX|?6(ixCg%Le3d%F8a3jRg)VzYzzVqWUjUaV6t@?alMO2Z(d)NrVU=>3n*;k+v zv?}{pvzR<tSOp#9F~U!a0tQ8h?_~P(o2JV@pI?>R;A8F{yWNG3u-{XMcVrbHQHa?B z(CrpM#E(x|nw9<19bFaN^vC-n{##q#hr8|FdUs31&PV}De|z;x-4ujKw^>-~)x9>8 zZW_vm>7Ra7v##A)cqrLA@tp68LL_#=;8T4c?KA734Q`KO`&P7+bV;GT$}@ixO~Ie7 zg&pD|2q?JZ-Mgv751}cB*IBPon(%JZK;^BZxo{AYWiup&sxkdvb|#Vm4_?j7(I*NZ zama!yOsopRMW#d()wG*zhbOp+K|bEpF$%C{WZ<5;V9&Kcq0D<tw@i`acMH%g)A~rK zFoEa$zS(-wPU8K{wq>uBWV&{1p2+Lzm4BpfC3)GZJLqurx8{|udWB__l&)>S?8KSa z4Oxw9^}dFd8Py$>Mc0jG9Fr&m84vX~_>!2<b~ifSoUP6TF&fs+jFpICB9})%pX68x zUb>P!uSlddPC-2sAf`dQO4p>{=+C^|9Qlrc?b9vwnOkvhsqOP4v$HiS+_$l{HQ#bH zM=|@)dCoNlT{VfSVyd7Ae5A*3(x0A^!6rtJU)WB!T0-2nXBom_G!iHX@Ih+h^#{NN zPE-6Gaa!Q}(gil-ugNp#fNYCbedX&(MepDbA9m16t(E3KGIyHTHekY%1M7o@QO}tA zxHayB+fbu+)}xn?V=i|Comb}3<c|CFdZh><Qv7+vlt>8wF+m_fPOc}Pb!I(d++!hW zj?bs*RM1gq{_Ltvcktum?j8&!dGM@fcRTvh@<C)^{Yy2JjrjS->h)L9iLUj0@kOyk zz5M@qZ=TOog=qZ(@W!QaDvW?>lz;#d`OfO1I{{GTDhB|nl-_g7o%yu8!b=)2KBtb) z)@C~T!OP}`)&bNwia$ykadcmnNpcFhof~`us$Q-oCeNPv)vGSQ)EFi9d!Lwn$DoL8 zPA9`!QRJ^KG-YCaleLy%Rzh0p)F+79XIf{BqlH49M&1gTJKZNT?_j4@Ulfj|mEciA zm_GB|Rnt|kd^Y|fPdtH7b~|=Qq5vB!iUbzkJ*duTi;nKD@a1ZEy75w%`GC3Ql>9L4 zbUo~N=xqNFWr`a7^g@rug1B(`XD7?~7dY?<GNs0MY1{v()5=3Ja2A^;0U+pP9utqi zCuhX7#s<FN_Wdwe@J81uB(`ll6=f&Co6USAU}<A~0fWHivTR3RyN(<Ih=>Vb0_Dr- z4b<JT0isQ=NRmXRIa1E=LoPAiULn!DRY-i-P5fLp%dPi0bfN|4OaT3y9pQ%B`eSL+ zuP(?MM5-&09y}<!d3tuzSmYT<gABv<<`3w-efQrcAtd6fZdHUbB(P=XaFre%r6~~% z<xNs1x4J4dnb{&;<s3g<S~f6pC&U%#7kcuLP<4t?9Zm@wd8O?yIf?#8ktLZin8<sj zV#W`g2x<1deh$Lk=m%B~XztSf+~y@hy<!`bH^+&Wx&QQG^Vueaa-G>@x;Bs)n~%DJ zmHM71S%72U7m(V5BJ@z7k<U?fT!Mb>)%2hO9mk}1Cr5mg`K_(Zq_bT;szm(|RokB8 zbygR4Fr?I!y=8Q(L@Jh62NVOVMxxSiYA6;wHe_&Gwe_Nv6q70d`sqc2wKSl<)NauM z@2!ro@YvhqR`!)iv5u^Rw(PEmfvPRrTue~<7!W;B)r(yl8*gpd=!3?6a)(sgGunv5 zG&1eqiZ6c7<cU}8YNW1oJg2Wna5}H^V09#!;4lp=L>o`z(2L#9j{j>`ny}{;U`Rwq zS~`rK?Z;9|NN1zDHUwGWrm1rw&R<$&GP_Wj8~9lK>FwMN_r94rh>DSshR3a7(a}U+ zvTTwU`KugReMZ~pRUD3fakhs__;#^U)9D;Sm*Ls4>cwIaKRu!tXxul#DAOphZH20( zO#Cy)r&nB2A$+$ILko*w;%E74icVhq^?-dF{DXD#FOgV%R2`<MisLO=+qR5-Vfy4` zBVaD{fOKFqJ`}n=Ce=dOCUe*tk5fH3%+N?IRRVXh+a-;|TBV=ChQX|Vv0lNx>T>V$ z*k0|^^|ufIyl=3Ur+u7`+B7*rnkh@xQJRg!c7C64r~bI<I~`My%`+ApupT3~nccYQ z=tZyI_I5Zvao$ZqclpBsk5*Eo({A#6$9B%UTq~A_kuPr**y%?yXiM{-@lXC^FW|Cp z21vQU=r~Iw>E;aS4Kvu|muA3aen*Snk~bkW-{geShZ(e4s8>}IpVCJAP}ugKTi%t5 zxIV5FWre0)S8`#0W?7%yc+M#)wC*YZS&F8=o1{tI5iS;<A|FhV$G`8HFF6?}<?Xk< ztm^07;dJ|OxZh8e0DbXJo4O*u!5qCnJ=if++gB!I6?k@9IJlk51gLtiB-Rds11LPi z0LlrEh<7jZqxB1fWc1}9*HsC<O$ev!GT+z_sGU)|bpFrxmYh<;;#lwbG2@4GAt_=? zXJMOLJ6UG~V?wb(7n+27AHDf5Cfvq}O0%|nU5gZn@E5_FiwcT`ANnTA!tN-k3)Eki z8`_51-mzyHT;V<(_icOL`r*D1zW}_|T5jT#0S5H0{9^I=BL|q!TC3ztUg6`9eBPfI z-sf8DHhq!Qx@qZngGcWi*KqJ}-c~OK6Ec2{yI0ZzK^?hko99gY*J~H<+Gv&|+h!L2 zhRunF{)~S{!}?<{$a0OdqbmvxSSy-Ryi^<@&oZ+`ie~PhtmQ~G2OnQEdcDH=`TDCU zE$?i?G%qX6|Co61RjIyFE2KhO9bQBe5ort`m5utCxB>KVV9>Ht&>_Be`Z3ysqvcJa zvWY~G`RMwO0(LnomPM){82xPBRG8%z6~J#^IuS8eyK>y+E>2V)Y#b)HE=HpW_+25U zmoRd^cINw^0O1|;7m3&ITj-*CjDMMPBQqC|Qt6TW_4k+s=9voty?eo)i$6VH<y?%- z{tVtM(t%`ba7xqfe{lJw&8|23lCW$%{^&KKQP*(qn&Uk8F~R>YtDBk9kXt1NJ#00U zp%)R3)B#w%Z9_$VkanfJmY8s!=I#@R8-1-*ne>FeU;7P<wRRuOI@)xZt8w4Q{PDkM zw13PyhS$V>OVq_S6}<khT<w1zzYj-x2fqB?v);bIACfp)6-f{I&MhY*3^Snmn41#M z@-K3Rsn_{sUD+Eia5y<*AbV+rzo;+*@fnq(-<o_`n@WpmMf7~aRU{#R1skfFc5&{= z@+Efv4dmW|gd=c`lXvu_ah$g2qbno*P>d{BMl$ZEbhwO}3H5U8M2MNdTD{bju_7e+ zP;mFcR>$#yk<RhyJ!Sm0T$YJAjOo-exQ9^CD7<-O4`gZDXEi{#GZ}ax#VA{MWo=Z$ zkKB)OvjyNYNVH1cSeI4)S%<~1^e?xQY6Q%g{DOPlay)5!^<+aPfVxLs3QWI{=EZgn zUAh0&SjXapv0=KKw8{fvp1!VsEUCGzjqZvN095y9Q6bx7NXwXzSLR6ffC$n1Jc1$^ z)0n&>f#;(apt9p_TVoX(Q~ROWrN;1vCl}OTY@mg*{72-B$?zhv3t-Dpq^UBV4SD|V z0q0I5auUHd#jf1FqGcn>d*^#gSYQY>!a0IdwPAqwb2`Sr=^G5HBJ^pMHiu2t(WzJ! zJ)>|Jl(u2|Y~Ce=^${`JH5F+r+;%~GQfKmXgLHhhenvi{#M_?yw54#Ib?Sv~r<TOk zkmG5KkOo`MT5S>-)YChhaLH^<q1er1Pup=m$#RAM^T|1?<A&JQ=N%JwuR`jN%)DMH z2m=Y#Tlzl<T;I@juRy}siK6#wRd;Ap!Dc(A)(cUOh-5VbD-Ab?;9bs!$!`CT88%B! z7MFC!+Bt7FxColuU#oUvoRR4oRV2TC(RuBgWU9a5_}7(@sQ<~P{_BJ9$+KsyiY5Fg zv=c!9Y7d~#BHWA=i<-E)3|13C07&VV%SBKEv=QOly^LjJqdSv2QyfBYoq%!$pLcqt zFJ30!^1mgLODOBzkTEY^9n_E=CmBcSzAk!Vn!!ub5{ztRL*HK$zLw2c`f5GD`E-Bo zWQo|_QQ<=HH?AqXl1?ywK4K?U;i`tg)*=zf%Qqgr!Wo-@3k8uT@)gsqa%8beQl}GG zMa9XGR$8<HdVwU2<SG`7>h2ypTRl4r3q9VCKkE)Vo}=`K{k5f5ktYLElx>b<lB6~k z5gyTs0JdNGx@jH0{dpi;<qfGGV$-Cpj9^i=lklunO!d|nB2R3OKa7aDR{4NTtE7O@ zqJ5_hiB<d38K#Bj)C9OCpv01AUQ2Ds#ei*{35@c+o$ZNCT1RX*&p+At)%@h$GH=<2 zk0sqbKlI4X`!9)3yKVyfMmqhC3G>6emL2wq-fQm?WDZEwS?aO-6+Q}2e~+F2{(lpw z|D;h{j*>zp_{G^qNGYN(OAV_XWaAas<<bwsKO#buhXLuF$s0HkJKbWv?!T1zwyoz` z1)rVeR>xh&YvxF_GPf!bnT)})CcopjBj7W}VuiP|y}EB(rl1tkt-rVHn^$O;4R+W! zE9FwQ0{-xx?+#l1ItaicS)qmOjW4aW8fUpd*xsG%y6IX*sd!@plQ;sZiYC9Ln)NYI zU!c@v@HGXN-T=9X9XZ{apZC7WWO+<oEj|1Fkeako$aA54of9Tb0W2)i02zcrxd^+> zyC_<I97FIJMcZO?iSo)5zZ?V2FEk@AJ3Zgu$q@E-D;XqPsVYlyrhO=S38pEW059|# z@69%AN<ei+A<6PKfqyk4yMRop8S~zE65s7|?)u)W^;gJFE;*O*QAuX{bLos%G|5vn z`v;!Cn3xg770{m}6(Fzkb=Gmx{(n;`$AxZNO#~2+g+jkK8_3)UpfJCza>19hM-l)L zoB|G?46~NU_-}`VaPBxbL@08%X+uR-Hs0p|Dl{%a#sx7t71!k@o)_nQMXu^GKImGJ zYXy_;`_il8HAklg97E5%5BEvkdih3o3I4T=BVR9HA0rt9EFt^_zQRx>7W2;cn>#IB z%m`jPgCkhptwn79bb9e|G%7uVnFdjvLc3b3x5L?SaC#VcwzW_e85x})vNMoztWewH z!Ot;-L$%^EAR|cZ>&DP0?d#q2)i>*Q)+*btfNTVPJNjR8yxD_dtSc`(tL%$I06%Cd zDs<jwudJhl)5$3~0aflXA~bHWww2FE1Vc&Xr?F1r(YzlNgTFW?P;hu-tUHxcGpd|1 z$gn6K@=CmzwOcm-NbZ8iL1#^pd@$n^hyOjqg}csHny&IlpmAs7Pgav~K)-7qx_m}r zcT0TZ8d`&Z2Pf3Tp(C(Z65&(j%cC4>{aL4Ue6&n0K6eKuhy;?C08sON>svdex@=!A z$V<z8TYV!!@EvlQfgCSYfIwU4MTZ*~%@bcfZsk<o_{%S*C#)YXEm1ZPsjNpzA%TG* znHjHA-2`r&M^tDVRm1{&(m)!POv&Zt(mFsnOoPsKL6x<0MF}#-2#!IWhKpbV!Ak$7 zm_E5Y4GS_dvD<}2el%s5MN0`91=mEYpAIOUE_FmR^Ubv!+n&8SQ)oG=r0SJsgvlq- zrWUY3fDg;Q7GlA$J5S6al1&5(kQRIOiI@OY#IlXRG?R5xQ%sh%kjXE41eh^}p||r} z&Z~3;7ME<@!2U4gjazlSN2`4Ix^3!@P8NlhN48Buxh$<ErUn<Wt~1me(d|pLjPBE$ zlf@aGa-&6_PMWf0`=}8OD?u4``Fm~mi_rW#(vJ2C3!J*g6&o8H-=$yt6C}2ZKfL^5 zhbT~y#KRoi#b&T5T6j5fnvl$<XN(L__ESL6VPU3X_F_L?UybTsSUuZ&c;>^^M4UNW z98h?6UVE(OjW0ce<8=V3-|M}T+Y)K!cx;st7#0|Kwj<?lyRo+Zv|YAqlx8*dg|4KG zkhU@Wdh34*uKD>*Y9jlzL6W@Ur$d6so8Xh#-l;|;7h0DMV}y{~+P+Qn<zvQx`w4WG zr44ATS2&U+SYLkc$f7$J!C^>RoFyb%vX`L_%Hg${%+C}|$KQmVCK*wNTK=xv{%Sd~ zJ^k&XMBNZiMj}-5o^c7Gx3|;5{INF3iJP{?^{abV=g0TCS-!T)TMt&@X1;qIKdu4c zW{QTFR6c(|2R4yF@B13}g#qwyueHR0)79h8nKJEA3l{XFvbs;RGlFc6{Lsu^rUIqe z!BsLQV_wQ!S_fzl|1)D1TKb52zdrMoHlKYtXZJgMf@9G%G|l(_b$uslHTL;OJk37; zqL}pZAi(1JBGfGX(Hpqg<w{EW2ZYOGTGytqz?0n@w|QHRjxLHJD@x$~+PXTZCw}2N z9TnXRAyihZ+qdt0|244dd@%0CN9Pa#TCEjE>N6zET}mutPKrJ_NXoyn^EOt^4Jc;u zNEh+Ce1aRI&>4%xTUTGHpIVkmrpi(2i6#@u&4wmW)fGsyZi_SD5&g8TICATn&5O|6 zms(CYh*VND$|;e$B;<VaNeXttw_1G`W|Is!DO^3|dYJyG>0syk1g}|PU2W`%3{v^m zBOFTArr*{kObU4Ra=S99b;#W?Dse<9<b?shz2?Z%M_@NxUs|>-8SSQ}FBH{y_^?VR zN5(qp&Qspyzk#{SmjBdRhwrA&KLfKUduXCY!ucQH+B0HeK}02&grN$=;+|U+3^sva z<zirOMpX6Knt0RM2F3fRt|jPTYqMpNboyH=^vJocY;DDQYCJUhUhJn!=Pj%+@tu#h z4LfH3!yKKIl%%-yYk@M~U-vL{um6p^v4(r04+na3s5tveul?bccgVrk{KVPr%%AUy z;vJ-#oc$*=Ufs#`{FYdEmuwU|pp5Y22p2Yk!p~HELaBgq$G3{074#Uu%Er1}U`sw8 z)WwOB-0m;eRpiE2l#geR@EE2eI|{`ZU0<>tOH1v1dh+3H|2V{5Ic)FOOg`mA>JHDA zv_^~d#St(92cyBpo>~_}T?61=BObl}WpPLZ!Ih9FA0}1V5x>(X1Rc<MXtTmV*N$`W zFmid!V`z#_;70aZx^oKxYk@Ihxr*g5I2$B0K1VI6y&M4&_-G{ic4VVaby-H^R;TC` zId1wHsmkppMkjrGqU4ykMMEOHF-ohDFQ(BgR49INyyRms?USpgbmOt!`~P)b$A!C8 z??sD=Mg$Ay!c`h-X@x56V)Sy=tB~48kB`9><#J%~C0UKjP%2f8$TgNYLgU{+PUf;d z-%s-?QfDWBC`(ddln*Twz0kkoId2GJpF%=Hj^^I%{AoU;Y<XYm54m}JgBYfKO8pWS z5IwfgO_>Ni8JF5<4!lJoHPwb=ZtyN0Ff(@t{iRS2a}N0U^vV|8EV4`8iwc)VL}r_& zL{Yyh@a<i^m>yO@d4_+{N0sHJ)n}Id>vYxe5L#@%tUfHW6beePABCRA0{;V`4i~dv z3&?Uh7=3-7<(_4jGT_N|9k-I>z}$exhbxE2VTxzAXAi^v=AXugZ7GE9)P>)t|Loms z3b|r)JB3i|tdE3-WiHR+Gl;Z}4M21ae=Au<6%JRcc3M8#dl0h4hD!L3qw7hl)Vh8^ zUwd>#1r=U71}d4qVGkelvt`RKMuUn>T;8mXnA_uHE8fri3Aq~lF{7{;`rO?zow^)V zg}#uR@t0LmRp1>*tnA{a!0U+}3!|4O72ZurXte)R#B?P!wej~ve#=#_e~Kfr!%{zS zD{oI7Z`Ug?a)lRYMC(Fv7tmTx?kl|101@y<2|3kLpVMOFPrTO?a`q`V(b0l8v23By zkNLjYSl)Emg)G_UOO(k!e=>p&m%I!$lDwD?lK6Oa%Yqui4!_<Gb^qe1&su<;xGAQr zF35s1{aL$~vH0pbG#0kYoM)6AFI>BX5V3enq@6aC5;m5SfC;*&1V&;di#@oy$I2dO z{|VcgKixV!?97qc-uCVl9uk&f80us~9|oj&)NgCsG-qtZ!Kx2~riY$Ei<W1s#yz^W zZ_Fnt1CE?I8=d#>66wJqJh!q+r*Dw&w5{v{-tlLIl~)N6JV+!|fF7<(SYl`MlXzK> zc*jVEuzIR6Rp{t{jq=RX8yzf{TmmmrusW-~f;5mf(KaDf0ahM2m{r)T!WxvZY3>2c zSsD;E1C`v;w;ztZZ&D7z&ep?rn9mO9PS;z`DqD_z%_ux9a|;LwoaCQ>D@_hyJ(da! zysAR3iR$;MUiA_WJ@fgyPuUu1{=RW}bAHPwZ@K&Uc+AJmbG4$!SkM2azN(>U+>gM# z8+9?)vaT2Vj%kB}K%<ISq=}Kq1FO{G+x7wH(1_0^{zRVg#1RZ~#GmPdxJe}~?8nVR zP+ps6ECRvN<()0Y?sP*ULGthqWxOT$cxk|lO0knqDLcD8#XoExVf=s_VDaeFS32R> z21cC{)F@&hUyo0YkIr%U@v&;Q2)m5xb@7l<xFQ0Cd_<tz>UqKe29fK-d()B=sNR8n zTF?(Z^IRE}N{jgtU}7+t4wUXzxBiq|b&ovSf2rI1h6BWiR(9I!u2gKnfcftz>b`oH zxbwz4M&|Bc3see~ZqBsd{_Ji~5@3`6jqd)}bF28hgphc@><eX?@z1QnRWe{2dgF9} zF)g+n;|@0n$AO6{DE3n088?t4o*t?<%ihN+sZ&ty){>;ybx0u{>m^;bb8~WNs5CYy zeX0|=`m0JuYVwg@bakZL*L%#}qly|rR5fKHX$CCKG#yXQtuWN%EvuQ7WH8FokDbma zQWYy9*xccH%ljDu+?%aREKL5qu?bg)w{9p$093v%t6$PMuo&1d=Fk1*2#~tU@AlrA z;Tp~~n$?ffnfGZ}*dbNf<Vi4<QZa0AUq>i?vDcm;tO9u0c-fAIe(~CdMW(g7jTH0> zFd@_GuVqE+<9lm{9;vs_8BVMq^De5#D66A?6lsIgu|NY=h)s=@N_%z#mPb$q8pA0s zWm7J~Ic<Gn{7rHU_F40$jJi_*hv?Yf2C>8-``+Pi8=7QLx{S5%4(ATF&gieOYsoPW z_Oe=To4p|sdR=@jED$l~HBBaz0%{Vj{R?y?efgN?ooC33huP?IH_g)Yu75a%=BF0) zc0QU}z90~TY1C-LX^%gj^s6mtgXsj|ViVLkhl@2ZmoB%VxzCX%N%n-Im_W+?3|iO< z0E2hvV*?E{@ux!&;phaI5ewO)DguBsFh~W=zvle$lhMM&K3z-2Rl@A$42e=-O&_xe z#b4ALLWbW7h~M9HsZe>PIz7D8_`B>9AD<G>ug$GfVqHrJF=t=vK&TcF>Th$7+B*;P z>8b2ML|V&Fe4jnf)}!?mtWL91n``~l#xvDS_cH>lvoa%i_w4|~_(Ykm^;W6Yn>Akv z8XSI|3K_FUWL)5f8ly*eF2Cy24tVKs!#`k@+qLFNO@|F<d~q?JKC5pl_KpPa(jAGX z+1)D`2P>j(shD&ug4*8vzosNRW2holf#f${6QBY&OtdfoS|HG1XIrf)874C^Q+ZXH zUmJx*iN5Qr_p~C8J-(*ho+zrLvJ(EW1Ar?kGu82z$^I5Jbng^(flCqcyY$t?SGV*l zM2*;y1*1A*F0pa?EC^wkC;Q6lYiRf=31n%+j|wP*>qKDEC$g>@gJ39P)zIALWpH?( zg@RngPv+umkzvO`Qt2EVW=Y<7mGo%i58uW<G3=-;>}2V((wt6!hq;<*qfJkxXd`mA z$ZY<(Z}pvL9&OFn81?I;^tO+TAut*k@XLpXC`mO*h;qHeq6HHhwJCH_H4b&h>MI}t z=5<pg?W%<o28v;qCi`SX)y_ycUfXpY=OxAO$adWh9vNJ#Cq(tj9tGy|onMg4<#=J7 z#48z4dnG`<M?d)=7>Mtsr|3Ksqyuo(5dyZt5HW?}BiGo{%Q5B#o;1m*q!GNni5<i9 zIF9F&oc!adC4zp;uI~s=arX0x1c$s)Ms(@ZA$b$_Rh!oq4Hhf>Twgh#RT#Qu=(3b+ zXCZ~FXfcE$5WZi&##q2lz%+%fepF}8Fhh@x7K@1kMS>Nr;==vTl|<_5z&;(`sO#bc z;iX^Dv5qnws+kDYPwC>FR(YAerV*eBgF!zy!0mi=hwshCu!E-xq&;eYUvD{HK3jdC z@a+e}Zatd~S(NYT9VkCzD?ic5zB>w#&Yf{E^oV-N#{W5{z*_Lpvm#;szIq*Vnm$&{ z3SH{OXc=(2Z1vQxr4B!U1*$I^G4bify86k3ULk9B3oEAeB5P@LdZXW%{jm+N-FEX) zt&t9)ez!~K6ODe;>xE4>^`GY=9WW;)F=#Ezt%O&#x-Q#a4X$v~S3LBY`X^U@0e@@2 zc+DhS)E<&-ml_=O8PLTkg8_^NyREts*ev0=cj5pb;WHpHicPl5K?us8ZQRz$6>OS& z86x4xS<V|v>W9Iw6*25`(mLwwg^bYG86|k{s~FwzPix8nmyPiB29V{8zhY6wnFMZJ z*QVOHhv`xAS|Z!q){u9!;mViUfP-_S;+gTeS{5cGCV<8ro>^^)wbrD4ATO$sQaa#3 zA)Zm@LL1NCgdL^kQ~sO<oDsvelukKppYEk!bCAUY$1F?xf^}5jFfSEV{6(IfHEDGj z`a;Q2{<OoXtaA>5ZSoiT<Wgon<6t28rMh!janB)8XvO#hqDZsPGF4}=%AM))g*?$4 zjv_om0THfM>VDrq4;(l<A^xK#{a&_?c>mVMv_~>V{caW-bt4;2t1>hRjtAn3>cx{J z!F{!jZD)5eeQjc_TlU{~|NqezG}UBh==tYlycx5KR7aXUsaI~O7?+Nq$HC0F>{tth zM4*dBY!hl=`i?9U?a{Pr)-E;L$!gLVF>J(JcUpQi0K1rtBMA<XD(cJfC;|@mU#?@f zy*y)WFxc}LSI@QY$0jBDtx^8Rx&kj@BD4N59-?s9pHAJJHMXS6=jQ2x(&+;6;Lq9i z*<TK&(_TIL=ELpy2%p&bM{Z}f$x7m3^=Bvw1iYJW)lsFr`S^Zpe%a!H{F~8hi1|lg zn<(i_4_g>cMMX{q&GIM;$#T=xr>=6tVKNi;V9#&O$=?s7ES>0^Xbk5DtLnA=0}Z_M zaq3Qrg!l)f<1>wE1M@nC(@g;?kaSl;g&0uz8`bhvxh_q#L!F9&d~MY|d7o09SJ+`s zcwRZ!&`2(p6*C$Kf`YzxP4Fh`LvLs?a&&qqVv0btqM$ZYn-l_)Uq?Y??S_TJ46uEC z-Xe>Gb-JJea6UUFO7lymMuM+~4p`0|lsT-3M+!M$Q8UQhAWoBpTVvqgMr|dszXmqq z$9#_tw-(OkTPSbJ7Q-m{XG4^XvtMCcjE%Qhw#NNw<Sm|Tt7HP+<GqCze|B6v&;VJJ zk<h#u8&zB8aa2&d+TIk=0ZfC=*;dE@T&ssi@Dn@^O@@5Dv|>btF%WZqRv3b#OsJsb z3YcfI2F!|w@F$zAdlao^<EO{H`n)W=@PCa58R`0?serM)I9YP)s<-$;O<nwN_|c-i zw3fNu=E8@65*`U5-0WZ=FdgPTO$+B7(v!qN^&`EXVNI#cA2RvbpEga)_DE!*;l#Z) zEdAFstE$KJdM@y8S7}lBunfu&vk6c2Fc{-S4_B(9`G@ndfQcF@U42X(Y%{hpd(xko zb<n?<10WU~gN33*3{l^{B$rl}A+y}wxyNC&Cl9#8*sITjtLsQK<!LdR(my&&IC?o* zZ<=V0H-8V=(r5}(I#W0s2-~5K4;)j$t8J;Uzx|p~Rd~?inV{EQiy&pZhkAx-Ox>09 zv5>~VWtBHh9t8ya?5Ev*7`-^HN+}?w4-L;Fo$HdHLIBI@g!Dgk#W;L5PbdW#6@1nj zRK}U}L$8NK_&wlcWzyn|t(_aA;r6W~&KyiiBz3z@kNYZ+myWwryLlI?OHXHR$)2VW z-3VwqGnVLu8gV1tfiFpe|I|7k(-07o3S*&0l$wD&ON6F2BBn9q<PMN2GmR6G`;g%^ zJ_^T)0;xd!7;>KgQ?7#i1&@xxGRs<#)=p*;1+CZ&j|h*?>sE~FCnvjc$sU{RkN2cp z41pIBMj*Hx1OxZwkze#>hxJ%6ZwFtd1uL!xnpWe$`GBw1tPH63bR^axAVP!3--*yS zF&Faqx?<?@X<Z9tujTZw6vbCF9cN}jWzzZeF_Ai^WEPL@OP>u!Bdj++rc974HMOcw zn|E_L$dDl=$-obaB@BQR6HU}2A?)~vx+)h9Kek?2x}-$%{ChE&;(81Bor@P;E?tkn zlp7X^6{?VfEo4jG=bD>x^t#;~WFFtu8KZ1?Dd@2G-FH}#eL-FIX{xMmPa68?u=cCL z+u#a%Xn9OvlidtEq5yRlttQ=Db<3?lj5=QN$vA?;J^{5kWheyoRR>7P(6;X|o@$T3 zVjon)hU`CRx?R_XqE1mv)=+OXwd$@uu6v&O$0~3<r*y4`Gzz^Kv>^fBJep7mZirD$ zh*L08)A171Sy{S3EB_)nnR?yC)LihJK2I`AB6%HcufO>oc6Pcz1y@@BMu$zE`I-^^ zBKYe(SidKDwt7g^$y7J>R;mhY6-nV31akMxLTvINSKfqv@_6WV3UdGyLxh!mYGW!H zgg6-92;Rf0v9q>LXleRV&<P&%7OFOncHG0SDf+buDJ}q5@ipV6c>Vhd<UW4Ir#^K# zbIn;xLA;9O$v->4%mR){%?S0<)m1?6-;KA8O_e3Cp~nAPAzu6V3YT5FrYrQL|2YG` zFdcbK{vKobi!1bg=ibhYXwwRh^oi-Rs{tF%Wony16dML5zAdoZGlmcGZk;o%rB9GS zwoWBgupyB_ryrZaG&WCG^-5CKZ^XYAAu*ZaL55>~wVH(T-_g%KO>}UhT9!B*l=c-Z zqQpRvNWM_+_8Ic=z8&Bl8o*@l#K>wldmk>{R?1ND>3*JcSNWp#<lkFH!0x{_3Bkwv z15%-X;}woim8is|JbcVV(w+MmHF)FBoT=07M5#P~Kuiy-@6~S#NQ+2?^Mg9J&A+zB zAMXF;&;u#B10!7_QZewLyafjRn~i8;f@2Ju8^3<V<FC)^p<VU)Z@b)ARXTCVVbV?E zyEnYUX}R^duI@M<sWHF2h<;TnkT`PfbC=hX*{6S(%8a^J#$*+=$C%?8jMyCgf04Wt zUK4NK=_L=JeEPTdtS4m5%~b-5Ar-sR^G_#0YE0n~?K|%~^zDi~2*UatWp>OqjS4^q zL;hr3iNFQ_mhX0Y0vh~{Dl_pKTWUH!lL}YO^2{eh*;IdVt8?%C>`|`%U{=`u;WZJH zr?BR6^+AxsB(%rkr-k6X{CpB3le3y<->@PIKw8IQ1f@+x(^1B*>X=3g8O$3Q5b4yq z$eEx)hq9>I$RtP?cXQZ=o_5YCj5TlY@$NsI``#3EOM|9-Voso5LJ;aremrn%ma$lP zUI^$3$uB(Nm+Kw(AOMIMN!~&nm|baSHT#zT^#$-WNFO>WCyc;BM0gCbVC}%IS?F^n zx�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 diff --git a/core/client/public/assets/img/invite-placeholder.png b/core/client/public/assets/img/invite-placeholder.png deleted file mode 100644 index 269e16e8c079e8310bb1c1abc438316050d09492..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7860 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 diff --git a/core/client/public/assets/img/large.png b/core/client/public/assets/img/large.png deleted file mode 100644 index 57a0db9e504efdedd1c7b28ebe399410274e21f4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1912 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 bivG(7yn9Ek;WNF!9~Bae3Pp0Z9lr1vXsd6S diff --git a/core/client/public/assets/img/loadingcat.gif b/core/client/public/assets/img/loadingcat.gif deleted file mode 100644 index 014d484414ff95b222e61e3d0d270ecb0e0b9740..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 20207 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 diff --git a/core/client/public/assets/img/medium.png b/core/client/public/assets/img/medium.png deleted file mode 100644 index 5ab6efc2c21efb4182ff0aa5e4b6f575d48f3493..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 400 zcmeAS@N?(olHy`uVBq!ia0vp^GeDSw4M<8HQcz@IU<~(kaSW-r_4e*T&ISVkS4ZWt zH?BGkdm=t>XX5YbW=rFmARssUa+3EX#;%@q42_Bg0uoHl2?_=*Z3i3@ICvTx4+uyA zDRgn5Gy(C7tXE$joG$zG%I3G#^m+f^efKr6nFu!*qQ9qqt4bm744^$|=6x`#o;Y`d zI46@cn%vxt;+>pGPGe#QI*o-*^o?uWihuu;zgGHSv*l*@5f*e$fprq+S&TsV_mbtK Wi-NLz=<d4=K;Y@>=d#Wzp$P!ffp|{< diff --git a/core/client/public/assets/img/slackicon.png b/core/client/public/assets/img/slackicon.png deleted file mode 100644 index 56fc146b6e66a678c9c78c7403f8e12847becfa0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18136 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 diff --git a/core/client/public/assets/img/small.png b/core/client/public/assets/img/small.png deleted file mode 100644 index 962c9cd48ff2a0e7c8012db90eb3c29ec937c2c6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 426 zcmV;b0agBqP)<h;3K|Lk000e1NJLTq004LZ004Lh0{{R34!XwR0001iP)t-s|NsAY zg^Xr)exa(bnx(2}cz~3jq-uJB+~478cz{)BaJ9U^j+&rTWp0<DriPQ5SZQ*(z{HiH zrInzjzQo9wqNn5L=ZTe@S7&frYjoJ#-RJ4*m!YTG+}^9Rx0a!&n4_qNlbNr$yj*Q{ zSZZ=wYjld1oVmfoke#B4m79u|n+A7h&Hw-a<Vi$9RA}Dq*vSrpFcgJh1VkAG6>$XT z`TZZz#HL+<U6|aUB>%U(=O$+bP3t%S000000C=0_p3UxNw&J}S<ukG3>BzN-60>zU ztI=pGR_wVwo4#+hf`J;j9>o;@e{nG$*j(;r>v>Y6aUoV5XMxS*XtuUjHOgXsxd47| zlK5XpCSt{D)c-=1wsAUr(9W?)F~uGIoPPEL0RX{E+hVS>`J$c)UfP=1)K#UkYE{n! zuc^(`DIDr-7`A#WQcSUs<|FEIt+VyAo(VppHlK~ljm|b@JrjI31^@s6008K?7dvDU Uf-#+Ip8x;=07*qoM6N<$g3@`;k^lez diff --git a/core/client/public/assets/img/touch-icon-ipad.png b/core/client/public/assets/img/touch-icon-ipad.png deleted file mode 100644 index 0b8bdf6a8548ae01458847180ab168b734403c76..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 494 zcmV<K0TKR*P)<h;3K|Lk000e1NJLTq002w?002w~0{{R3@JXQ=0001uP)t-s|NsAZ zhK=;~_4M`jdWer~e1mz1j`Q^OZG3}#ijZu3gZ}>hZG43D^z?g)kZpa0eT$KOiji)8 zgl>I>VR3qIeuZFhd47wMaDRsK^Yn3mhi`p_VRCwMfroN`hjD?2fsK=LfQa<;^>BZO zVsd(6a(RG_lYWhp_4W30fQN5?hV=IK^Y!&|frxN^g)Iks8vp<S97#k$R9M69mfLQ_ zAPj~(U}RfKAf$!1Nsl`p-~aKpl(9O~D%By<RLS2};tR_b|8ILVA_#&Yh%;f#GtK~C zzvCE>-zUS0N=SgH4co{4hY(f4P|5PjsT+WDFL@i!miJ1zD#ukOl)L~)9jf8RN~r|< zzhwDzVy^(UxeaZ;-&s4E;@aeXVXgt@`7u=Xury}jIa(SW-d=+=o_a`wNcZMnK0RHu z<*X1#C}t&EwEFT08Y|l0cIsFw)tgDlhHZ1m2N+ZuL_cqS12<+T(r0Y1H`#vrZ`N^y zW5+gFD9v`>M&{clE|&yv>M~ST>H)l&o(!A^ZWY>Gxt~)3dmbDPm)h^z$j;aCh`}B% kz?%nYkT(f}Ac!;M6W#z7<DDq&#Q*>R07*qoM6N<$f(GI2{{R30 diff --git a/core/client/public/assets/img/touch-icon-iphone.png b/core/client/public/assets/img/touch-icon-iphone.png deleted file mode 100644 index bafed226a52755670a8950310cc0f122f8df4745..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 640 zcmV-`0)PF9P)<h;3K|Lk000e1NJLTq002Ay002A)0{{R34I(K(0003FP)t-s|NsAW zgNk8tddbhx|Ns8_`umEOoO+0lX?TEUc7BJHng9O&e2I|x`uhI<|4m?NjhUZ)i;{PQ zi(_+qo29CkqNu~k%+b}@Vsm?VhK>6B`^nGJd4-K>d4Y(On^tFVjF+AN{{K^DZfJLZ zYkGl+m74zl|8jwdOkZeccYb+?j%0Lv`T6=}b$xe)jC+ZYoTjW#Vrz+&o3pyUo2060 zd4Z|0wXnClhLf24{QPHke^6s=gprrg*4S8Ta%_8pbApLwb$$E$`*((oZF__I`}&HP zoc#U$Wp#X3Xm4V2dzGK1e~Xfcm70r}omFRW!^+H3Wo~+ij+mmT&ehk5l$v&hjLOi` z!pX~chK!YuJ^%m!0WC>HK~zY`?UdI}!$1&47qAn+yRn__Bu?+W7f9&6cas1A5j@Ho zkc`Ac5f8wgw=>dcR;!syh(I6^e2s??$%drzpUU4O`DxO{xp_7wR_XZhN^AHfM6*KE znGMd?g*vrLY<6zmQw1|wGpU=?c-eooO7h8(-qTQ%^+a}^(>vR~gURll-CG?q*%L5w zo1E+M-V;o|d~h8GP(K!7@(1h`9&^BC7WvB?779k!97Ka5Os^7C!>bO$Y7Dg`ai4d^ zf!7z8e!6+7XD{_(zZLs1sw1#bm0t+zFzk~Y76zb$!VoNHpia^bgHo#r+H94;vI%{W z7%Er#p#4f2jGEda0az<`LA%8oSSF1NKpwsAdIz*qZ-XT%Dtg~Tf1daG)I9xhp)U{! a{v~g%9v`3H27wj;0000<MNUMnLSTZGphElr diff --git a/core/client/public/assets/img/user-cover.png b/core/client/public/assets/img/user-cover.png deleted file mode 100644 index 8de60640dd77431a246609da873922765aa3efb5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 23563 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 C!t4?N diff --git a/core/client/public/assets/img/user-image.png b/core/client/public/assets/img/user-image.png deleted file mode 100644 index 40e0d8226e6d1e0067f08045986c51919f1cfafb..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2421 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 Fe*m_Z4#@xj diff --git a/core/client/public/assets/img/users.png b/core/client/public/assets/img/users.png deleted file mode 100644 index 7b5df56d7c553605f57dcc53cf12c6db0d68afc2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 49253 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�Mpst%yQ<r| zkP!?HUC2MSEBSBx(hNf#7gp~oy2rl?6nO|%cOjL7w@m^N;{yB<ma3hXeU96^zJ|-8 z+8@`9ThQR#5>!hqpE-DVrhkmM+y)?Y+Pp^o19lPRPT70y%9~8Wb7Z>LwH&3}9yz&p zCk(g$`%0W>UI89<{ana@pf1s`DQKW;zV!y<h*eq!^P^@jC!7~r_mOjIZw`der>NN~ zJCR2l!e4y;=9f)4*-BhJ|H!(LLIuT*A(6_N;!kPfO1l#^pSMh^0uT&BQ+pFR1Eb=8 z&*v|`FW_cak9_Jxp5EIDwRa;xiKBstg}(Lgh)>P0%}2ga>&owN6|AhlP(W`R0uc~f z*TF+jaw0D0?kB0hA4GuYl_qVj@A9?N#XO1Ksxf^coLdze%Hk-+<G%BKOAxRs9q-w= zIiFS7zP%HQII1#J?*sU3x1mEf-+vZ2vRG@zsju*`z!aacAro?(qr54RZ=&!vD8Ri- z^P5MyQ#Y23JAJ3k$J`%-&2p10Fq@B3^e{K0shoFp)LJgmcj2dtst4>y<ey?n$NO59 z_Ik__H|W~xa4x0aNai-4wO(a+CQ!0yiY8?4_c2hi^H)%S#<ttee}f&EAXuixP9hZd z%{2Yn+G0`QG;8(q+<u+DATJ>5?HRxdVyfvTw&QQ^5*A1y1$(knh2jGOTO*shIFGY! zyofczNeNWxLn{Uiv#xoQ(M5|-)x;-oZUHyQ_{aI}_4`pHc``T4J^fA?r2qTqnPKnW z%7J~Hu!ir5l;ocK6fz^|tuapf0X4-i7ev~n4X3?cy#6JoqfxwQ!o0w5HWMi3Mo2A< zlU@D3M+7I<P2j3`zCQx%&Fm`eE-vswfx!-gycZeNwTTK!)nEgs_lQ3~^w#ZRY9_Cw z1F!xqcPCQ8+Ql~{xU_ym`l}Z+T`&Ul)~kS_etz!(o$O<PX{li8uT@O2nD1S8xY<~f zUG{3V!WnP~zp))my@7iNbh&%58CQXjFrSC9vyF9B_9T^X5^!0?0sBs6PaztJ>0H7k zFwvTi!Q262@p|{`YcK>?XtBo~i5$J&{UN&}tk!%<4e4jV&Z~Lk{N^;%1LlDYHH9vV zHtJEgzbZNHLIK68#lS-2*!mvQ#BI~ZX?VjqChq$|V9htT0`I=>KIDF@z&s{7&cOup zVt|}(3Gu~+1c<=LnV}Fcb#ngWugV?GT|2_YYQ!on64nZSzsAF_!~uxN4QzCO5IL-O zAs;8alF!X-QsTE+j-jOxTv94<B&9i<SVz^KDn1)zk+(Fe1`fZe@Q>|8Is@mpBEy-l z**gjxK}!I$_?`waS?s%>*B}~WMZ$zul3AZ_9wezhRXC_C*s}hQRSBI?c-aNBV4C** zyFht(al`k%02_HbvcBMfMF|n9o7~=c*vs1B{=@w<YLA`>y1OTWA$HRHrq%hWQ2Cp9 zEy=_-9T@V!EUXPf1ucDD1=yy(SDSRu8$7UGex1o*2(gjt>G(93_S&jm{cWB|a=$1A z0i;_?P`lXC^6bU5$bl>Ht02BQCY&c$ItU=2qo^erRB;+RIkRI}%d)NohBbgYnHBWf z$oMuo7vtQ`Fvl_>I=f>_{Rs}Ja)2TLTiK^kw;Z5#9glblQZ9lX#4m6G2YPVs4Ql>& z)<-C&;nRxvS6nM5BB=Wx2TU_{J21lKsjxnZ6bYLb8Cue85vo5q8?!8Z2R~g^+=QPt zAif?C99}gZL#|yXBnYuMl8c8&^s5kc5zlz-yMZMC+uFQEpLud=MogN0A+3aIAK6}c z(DrvF+IQz`0oS9CMK+%%6mz*Q@^>J-LUoT`wG)=2$)2#9wWGc<ro9S7ZjE;PBla)i zB8Va0EvXj_hA8PLNpKfu>QAR%PjGEnYp2RMJfYBHr-aJk58E0BH|HXe>-cH#)0?Uo zA+>@glF;z|X$fZdODhbz+yEXYq31~fmLFbJfwR)SYqV9Cn#sTzKEK%3*QJU7w*x6v z9o>Z_iljgS=fBfqP_P{u=yxE(at|VE=F|@^qg>Xn`9VYj4k(8#cJBsPyX@IS8iqgv z5S}-;gIYcLTCSp$@{$+;YO5gY7GGCQ7XM04rb_}!*5{SC=2OrbBifUH7}J4tA_D5c zUH9UaB&&y?4r<x#H2=^Ob_zqO4W6i&N2g0@T7QQk3egv<$exI1_IQviII$LibCUzp zdgEVBbL^?Sx-XfpG`^sKNat&~^e+7xA1x&cQtq)ncz40V6A%J*n@xU)nans)cSkyJ zMTx*hL$+3^tZfgWF1FVHng$o9iT@w39J*it`ynHqw&7Lpa4FV(fd$GXlnz4NFM>md zN-5M?T{}Lr*g@Sr(hZIrgMQH?zJxx?Cw~5}3nyNIb8|CxYa@|GlGns<!62VV=k4|< z2LBPa61fMj_5p^M+s>ufh4Cjmpi4OXBFQz_{CcoBA1XqD57|6!i7(GI5PuJp((I8i zU8^t%kOkM4`sI=Y>3+_p3%F>-xILJVE@U|-8)ynYYFj97vojPxK&FDf{0F*E2yJc@ zoDn(MK!sQjzovY^ZaViCHQy23P-H}85mN28VG2OJos2WaM|XN_2St#`O?B!G@OSpG z+jxGPWk#z$>yUQWp{m$W+)!ZR5l}kfAc5GZ6vzI$r%I$W#N2_u{C-3soUa`Ymi@^G z^06<?y2BS*2~m5foU0oS{P~lf=9tkRai9OMrQSoeEPMYSasTb{fmWFP&Tm{ZuCrRy zVwXIXF5#)5RORFjq`z%(eYu{@4vi?Zk^spn%ftwD8`cf8eZd~y3Bj{daZ+1Z18tuS zrH&EanBWf<V^qB}PHpbnfgy0XYa*+OOd2Dcd%HJ}t`?i@+I!jiVQBpRzl}PNaN(p} z`fauOgKcJt8k9<k#Qs($MuZ748I(Gm^u~4|IjM9BA@3$7l2T7B8I_m95r~0}Y4_FD zIfvd$;(L+YJA5p>mD`Wxr-Jno&?D#=zzslFY2Ov1r5R>lMCD{caAH5+bROQPV}SFr z5r|Q{^qFKyrTv)__3RZh6p%memA!$|4oz!|%PS;nm*5`T=y&^zF?~)0ox}shSagF^ zd`7`FU~X`;^P_m&<xUC8bH$G#P7ILzpG*lGvIyvefD|XOT(vI@)FWe_?{7V6B;LjK zHJ}zs$EP3=^HU<bP(|JUADwCDewW?A;gfRK{U3W`9MX>6#*+G7Qq};)xdWX0a!CCh zK=dxnpJuE_V#W?xWE_aAWSlr!gS{GhmQD9m1Urza2YpC15h~NOtaGWg;JyFdus0a_ z7zGT<>@@$Ju(!B#L>bAOw{hc9A%PVAJWG_jH@SFMZT>J!O+h-Q1ZybSP`hRg7;9~Z z)G=MY%C}+O7Q+$hZUX}dug1s1fgPrlpzWGi6h-7ht}EGi7cun4Y(X0M!~Yoc$j{(; zS9Sw4ga7$Zi(pXMOq+-?IpmQfzPY;xNFJuwh;<-crH|m~1L>2JBG64g`}SYf<*W-X z9n!B8-hM~aYm$LLQ`8D$9k8BpLyT+U6$9CSG-#mVlcI_r3f8`0v}dBoO#ETM--jzw z&6`NqL!I<WhzOi>2ygLl-WYu|Igceu+plf9f+`AWsq(cclx!cN{=2^;?g?~Bkxbv; zEa-Q^mW9Qfg@;<&n+&fN3G2zZf1(LM!o`HddwmG*Qk;D86^sNYEyQcOdWfN9-mZm^ z)zvd})Nprbv3dBmB1_6B5>tNtW9tQeSdo5R(pbX>!WY&ufH-!%rwgN|(6FV0i#{5u z8szu9S{}{1tG@VAiNxb&G$3!VeQo;H;joNh<p0|WxV0%<x$;>L&4W1~rTMwvr#E*K zvHv>1`};R`BrF~L1u<ZZp29S(IiP18wCGFs`L$;FNQ`9T2sx%l@8W0Mao_+~2=03A zLb1TqH(G9+vbz!W>xwUE6r-3OXXimxK+AFSpX9Y-?8V^JeLj6KLUy~yV5PwNV)j)D zZbuvq+)En;Ct{o3((87+blhyL?Tcj$!$(%AyWgN0L&hkQ!@AH!3r&H<XFvUDmq3z; z4M3naaY41mZHr$4{Q&m!6Wwm^@j`W2{5@4Yf<Suu?%lKC@iNB$YP9@@1-?5jG$*4J z75q;&qrb4@*pKQFI_d_aqeBBZxldM4B>mI|2L4yz0P#i#W?5_(v$NA!bU&(vY64Ga zgRP9wp90sB{Y*96Q|63BD8~lQ3|wXNMPv>u{<7=TFi4rb0}o`%1>Jk_=uTxznohLN z-UYNUNqJXRCJ^90$!b$v&6=rvFC&t9kjUlemOGdmB96J+5tA#~g#-pwz*Tbh%U22A zXVRS0FqStnWCOUw!;1e|J`t|&1V;oQ@Wwlk4XKoPk2k(t%Bz3g#(9)Ie`D0K+Yz4Y zM5?UR7ffF-_yQPF0@>DR$7oq*EJIk4=3V9sTdS<#|6PiK;rrcHYjb|?6E1cV+KeeZ z!#AD4D%%jTA3S(Txm!vJHgxX91Gy+nEzgx<;`P+DRfhD(*M#f=QB?w9@Nw#24Q4p^ zdMMBE42}1LpeJ6|Zo)(sk>ATP^PnQYb=kRA#H<naRuV(!8)8Q@;#HY@h2IJ_puY-p z^}Xl<LLqs}A>mCmV_Boewe_EkwV{zBL}l?e!*5NA#$Wjb`Zg%TPw#J!%dDe?uqW;U z$a>tMKWe+qhO6%ds|vnODb$o}y%L3_{cYFj4`|mne3lxxxf<tF3>y@rd`KDcSpGS> z)m2Kxen`V^LrqZ5MW?4<wk76qSmXJ0{@Ve58ajcQ+yOg{!3(9hgv->kjLulU9I^Gv zWz@|)5U_g*2LI%`0zjn0HVZQ^x(p}4l(Ou7qiu3j&1eW>`1RWyo^FX&#L#j*DUgP* z3BJ~jL<N6Or!praYpqSoTdp)7BZ|241`V7y^~U}I7<}Os%UUrGIE&h@k;rVVLysp( z81NdT7G{b*<w;y+F3$3>y(C?J8fb*+y)%K5o$nC<N-Pk7^yw|CXDo;k#;SREZ^tzY zyh8!EEr=_K!QnFb0~n$Z0HyP28Q+?e_#<P3<kc`k2x$av{O|SBLz9-8>r1{|1>*)L za&|$BX8E2KV+Y@cvryxTz9j)c)Ts|lCFX#B^&2fsif3QI?;o7iZ2O{HOFn3zT57^m zthsfoK0)mVP@`I9TX$4h+MkT~c$P-RisLK5#{P&mP+O$Idas;eB<ydgBb7_!9*iJ2 z;_iYeovm)wqJd1Qo7VK^oHMq>;%*h^U4^;$P{2i^ngbv(C*@d2R`5pCR|2CG2}nJO zqA`JLO@2eBFxMh<DYWOA5%|v!;@NDLUyC?&YBO(^P6*oB<8JG}9e6GcCV;Cr*l>fP zyY21CFLHt5RWi;jfdAdhPP6l54-L|n1`F&fyh4SG9R<*u^N%AaqMKk9S!$!NaS890 zc5?|nE(`l{+d(m>VgOXU|A~mk58#jWqogIA3R6e7{vr@)pedT;b-g(*@7K2auDYT= zPm3zN&=$kbXAB=Dp$#K>zdwu4kU}X(d;VR@6wa$n3zd_Uxx)`GeSyc;2GCp=Vhez^ z4|eprR=`oK(t3Mjy#juMI*c2H0C1#WBYmXs|8$be->Q)538*N`69iEsZtD$qNP^@p zWP`A<BueX9X^CqZ0)1wONp#66SW5HN7Pllrc$J-r@*;e!>eSw}N#M_2U$fm{OzE2$ zK4P8kUKf~GKFV+v@{*4Fs+PX=fYCm&S~oSs>dfp{CIA+8n|0&ez4gBQw{w6lr?k_M zE^gT5`UGT_0T8@qX)zIS)O6HM4gXUQdwleZ#34ni`V{j*H(B4_mEZwqfKs4Z+rt|T zn6ekUzWubKiNZ>XRcye{y`VbqKJ70AZoMrG(C1D@_scj_7C<Y0%Si<Gg*{TYH@FG1 zh3N`Zk}}fZe6N?q1u2(S_>V&EwNpznTbjIH%mMhk2qE3zWMxQ$K-3RwO2C?$n~~mG zpI4ORzPtg>YdgQ$1XM8mn7&H3T85hNG25^9k2E6<SA23=2Fynf-;%evlGt9-h*Y^l zkuc>Q9`5WO_R(sa5I-<PXh_A&6nEt);y)dCWkFhBpU7+826wpI5<Kb)0F18^q5gYr zdCIi=Jn%M7wLyG6=rq}!!fC9z$xsab@{54>Pd56u7m7H^6320DwzWEwZRk?8&txmj zSSHX0Q%%2!=!<UZdB2uAha5jR=0K;`G9JIgW)DKFoIylukO<tny!2ZgV@*%Z0)JEV zn0`IZ-TNFcm1C$sEBHO}mFw{V5EvkPx%7jwy~f{B`U)gnay9lIA3xr6JU<Qs20jl- z`~XC8uM6MIMGgC1pO@eFt6`Rr1<KDU*a0P7D+ezW7r5i`F{t;Ed=Nk}tG|-5jyjC$ zV3`RMW;pTqsxDcjzq%SxdhU;4EjLd}<mEgI9)MMxLWkZfDwk!BeE20H(_~&f@MCS& z`+z^%lu8$>ejmkUpV3>NWu(%9G@C>%t)X6jRPDTj$vd;GA0LN%Zc~@wN;GE4Kad{v zI^b+k$z1)-^yApm($WNE{W;lp;%R{K*(s&w%z7;Q&Cy#Cue}d@Ietl`03D{0vT6)K zQ$sg$2G_@5ULR*sWgKKl<&tkbJGT=I_aI-(H~{REeaXbL>75|}NA;`A^wL3B=%_DK z)Qp}+r~#{@$}|3tL|*ImdoXCDl9WgE6zKR0zw*k5rTpI4WC<VxgDs6Fs{WiL&P27= zS^f>9JjOg2r{92gz%Eh4{KoR#-KD{BmB+Ie$1a<m&I~6##O9I5Gnsy+<kQkX0gI~E zz|z`y9VL&D*W{(JZ+W1YEToq<Kg%oh0l=P7p{=2>2F(B7om`~xy3h+dfF>wBqJ)`t zP{XPhuLtzIZjHCbCgMB-5o@G*2hTi@)OLOGX6MU;<+bl7t09cyi}L6o1j!Rn!j|~w z17I`p;XQld#)XNu_}NZW>j0K22%IdG1k7J}XMf<i<%0-7Q72%T;o%iKV|x{c55e{F zH3a_utqa0yA^O#Ga`F~Q4<JbZkn4D!m*X+$4Pb*kODe`M`ojS^LNhZv8cFkbY@H4s z)n6mpIV6yOFf3N*-p5`nqwyr#04EcJx_buUIWH^H%&spM4W&(cNL~XBxWLIpU_#F0 zNcgMX+m7{<b;Lf)N?083D;nog0#`dRH#qlAH|-oY_<x1*-E2n>t{u+_-gdy=Td=JE z28@e<SS`=|&+i|W+q-SiekBzL059NKQpurnI^#=4L1&0W8=(|U+OHv0etWQF3$*0X z-;|T4tZr#EhP)O_Gp);Cqv%;LUFs2+fMtONaeOrV6qWHY_tKj;tnZQmkNJ!P*2K>i zRXhVQIy#2sCNXwOtm>S}Z>%wi)=rT~fW5I2Hvn2{{u2EfnPe165qUtTSMnoAx`$!* zFo!U9%#5}J<Rg+#HUN6O;>T-mh~<KkxmTn=(QwnMhg8+ago@ul7k%6T;V+rn3?F7< zgE@HA(}lzAhPvD#3G=q^8$X{)C-Qf$aA$rs<zfT{Wg|D)32j^h&Y3S02wI?yL2uYC zM=a1}<S{+l>i1mM_ErFcFTCNX2=fxFy&8uG65d?<o}dKtpAY5Z&5tXZ(X!-UJ8^;) zU--<8GCq;LOJcOCrVuG8AA@rTYXN|q7<4vE^%DX0Qb}?NW&z#HyY74+AU)v7%O^YA zTgAhh%m_YauBC)3Q@)Dw$|BPJFRSH`og(h!4rIYjK9AJ0joJx2JS`FT9JubJo<0Gv zlJvp-OCqpqiP0;urH6io!paz&xGu%mp`5p0vLLHjlD}nIeqef1*1qbi?mY&AW-npJ zII>p*(h8ZGYY0S)va+uq^JgnLWqezs^4AdWo9MTnRUpxlMWF^Nlz>mcEoP0jVepq> z(kiX<%dm^Z$PAHwxkV(uKke>c9;;0Mfs7fjR7#lJ6!x|D2oCT#8ez{-+qKo$N(`ST zvd~nS{~v(a`VX|mPjN=y`=xmH@dcRq&xUB5^OM=vm0%ZPGXPQ|JXl{5n{V2>*zBky z(3bfg{x!CvUJwA2z5KY2Nu<XX3TlY~-3-Bu(j9FD`J=8cek<ESG!m(jsU=wc$S<^a zp>30*>!@W;keL?1aLm}~zYNUoV&rJ{>WyT~(Ahj!O9E=$t*V|DgifQef|7C{5T!!4 zW&L23`H%Nk3+Lu9DlRzpRa5l~mzZWNW8LXfZIP676_ht|T?Q4S)|UWEu;!n2YB4k+ zSEaO46}{$lNK){hhS(oT-AR@ZTyW&xxJ0UPx{w3e!8sk-Jd3hUsC}Y>FuV)-V<uxZ zKZX`ds6b2;-i)1zN-T9iGMpGNS63GQ?>WcU1B(a}-q_A{n$K>#!wT|QT|LK=YSeRh z7cChZx@0-=S^-G&nF&)B0gN5Qr8e-=3tdf5D*Cf`h0?UPI0zWTux#q`s8w_}ffP#S z<P-5C0Z}%^T}xsrw2mbG5kBHp@3!_Gk?ni3=>(WQcf1zs>~w7D*DWSq-~r%jca!tl zN)6Qn_aq_L!iWS0h+%7dq}<Z};gSsCUBTz?TfFxhq!k_d{^82*=(yghR`~Nh0S(x5 z{sb|JgC!mXwBYVRyRYD<2hpWJqVL*vr*s$_|5x-c6E}(W3_!3o!1N^4WClF}RAnBD z>E}Vx?)pp3H5Y2{tEyIdm}#x-6_F`FZ`y?lPqbrwh8zDvN<M$v&f@)VIDk3-x9Py! z&jm3y@woxgwqPPsu#A{^nAdH*_{Y?mKWEg%CT3^U_69r(sbQ0EjBZZpts!u@MC$%L zK3~@~_SyUwT0+Pq#co6YuH)Vva|xV#|CW5wI@t2Vjn;_y3xBH&G1|l7;FIcV4X*Ol zZ-{aHMdPnqqoZcYkF>2B{r6FmK(Ezf_iGlvLaljQ6@c_BMJWL#(b1E&Ji+8bt<7a` zUuJp(0xw{mvJnn+QJ($a<Fc-a8k71u3`Mppj{r38_c?##pIP&h<5CzehVj-M2L-&$ z(_A49=8lP7)t*a+;><%`@>_e}wxvtRWS_{PJvWbkyU{q3Yn`)R&BWQPi`Jz7HfrQ? zJO<3~g1=weXfnNZIhZ1Q0buT}QJS_DhCTF|E#kQxqG6IuP!_2<r145U3&3%v9}TS1 z9tI!^V(W;3T8Amd_>Eq={#$RP8GEQz0?_Oq&!BXCv#TuETd`;$aMZDW%mNJr*jnGy ztg>(bNk-hi1=`{8$6pye=)Ph=U@k9mc|?J{x8y$}4erWAf0*H22)^bTCb;P10>G_5 zt{6(tLw^*DqQf#QPqCMm%&$t$Cy;`Jpvlpg!8tw-o?n{&hvNQzi{h<X-6W)MX^!6s z?#N!^B<7k3yoN{%PJhTZ5^E!xx2NNJ@rbM?@&wc@qa_fXrK)fI!;T9z#AiioWYpKy z=(4^2<`*jBA^?e0BrYb=4!>Fd*zdHMF6_W<^vczEaG!<(^dKH&v`Gn;X=zGYK&`aZ zp<4k}+0NO3vqe~$@XmxG!QeeE4oD6ALuzW53iaCR+IuOsgd2sdKGmGn@`!!SR571{ zBU$CSgRLm3x;fZq%=OLg>vKi31Vp8Pz|F3xT@R4t3OOK)Di(dO#?JLx$0MScfuYRy zoz{;a18oP3ucO)Gk`kr(r3Mn_kpg8Ng<h$zlX(cc#6+XJI=5IJD{gKzevT@L%;bgf z%NFX?tv;bk-2@>-Q#JqkMt?S-z$l=9$wmH7HY*T8i)1Sw-q^=b1Qzd}{GwDzM=nw9 zSr63=@6l6{+=6C;6QhhKAGR)Enb}lbp~m&G=28=<VFhily2tE6@k1waH`osLxk&&T zJy&a`K3-%^S}gWh+T>bir88P&SeQxF2PDv()edBj&xUUGwEJg>d<m$5zl4<82dPyM zV1hFYKdaSCe)gyS6XAtlMf^!;QXHm*4Ag65?zoSkPC*O<06hiXfs?CeC(I}cOM71J zCH0vZ{mfPF&8qIi#=+Tih}!(m!Hjk#8;xE|c<vIBa3fd6owLPqm}oWNl-L!j9ISeQ zC&PCqaBp8mJ+0_4-H)q^;1V_C!KW1xlND|UL&=h96T!EmV#}pg3DaL$ZTGdQ3E5qU zh`|9l5myZV^T=N0;}^Py$Ra++pU;aBKdpz$zVyfKU0%so3CcF&H51j;RghcLlQ9MC zPsN)5N)KOYa|nLfI`6RBF)v2V7W7_}p9-oMw|EF%5$|9ChPjO*b+mnNFpX5sPLr43 zk=pjdjTcHOzi9Q{^HI8+cB*V0@G-0FMK2&r-mH4gOidtF>9B!0($uD7se#l6mD4%g zR-b60!HrU2k&=v(yY)J9xAodEDDb6Zj@#)Ixc4VI=9Jc7y={@5{y1pWuVgoww-?DL zf;(RWk7@8->i3h`@4%xX0YWO&t~mpTT{~Qd>R=F(bmXRUql)0us`0L^Q>l*8n*ui9 zRAD@(S&g~?tiL|7o?)s-b*hRcVu@40_Zg`AxyU12rtH`l*Ad6Xa;IqhZbVnUnS#=V z7fUQQMV!&*U#HwNnyr5!U;Ff4h`BtON;M9?AwZWG+Q=k^rh6JUsrzjk^*;Sv<MO)@ z%b$^n^O2p;*n6S-idCe%ztK7nNHAc1|0nns8liE6zVeGF0uDc|L8?Ca9?YeOF^+zb zXlmBCVs%~Npw?(){Dv@)cU21aWEJ63{V0@^fA9UXWAV5<2^a0PAX~|1*DFv>;G(;* z2@+Y>y`zMu(Mb}Y7TNV=gC&$bH_QJzaapE9nj0atnd)Xc(CW`wR)#2Xbzem&eDy2E z6#F3vLE~+(ITi>#2v*B4g^QiY$Sg<OFT)<c@XEgGOCwPlOIrRe?T?7=YAUSsR434R zrT66;p$*mka?Oz5Bf0iQjy;NDQV+t_mB#8MW?ApIRVGN&uFP;#`3O6bX;+BVVk^6B zv1m1YLBoDLal?9gZpUIk&zG+3HEdy*_opib5?}b8epC5m1-VjXW5g>(peKNiMB?<a z-Gt=0%&_O^wkZ;?$vNBi<GdDv*pzYnl1};a@vr0UtImu>?~BIRmz6x?)bBnnez0F- zEwF2#Up3b1tlmz^c{2e@JYzfob^LJQppy3Jj=l*u`%SsBB!SO*kKwtr|Hb_qVpU<s z8*$4_qz5JQG>?+(r~O4S#97$!Z{$~QaC{h5BhJTZ5{wXLi$hvA$qtzIiM9(9O_uqk zWr$U5{P>S2G|(TfZUpNP@g_o*>P#CpqaSIzm@7s>)#7v|`%>i)?yi(@pZw<e!>@w* zZjbU3>LZ;tWz%={xE5GZ<r%Ns>@(kOg&6Uql1+N3#>tkXj5(A%I64-&v}#=3P$qKd zPApX}H@!gtXYkxq%uS_TjD}(`Pn~~>9V*bA*!Ky{Wph}g&1Vmxj|z!DXUfS;j6|<w z3O3nZjw_SUgNCvgh6_Tx6)<or1AeX6ICgf2oHpId3GRJ!XKlnt%x@R}xSB<Oi6HAl zuIhbNAd$yF(xEes>`>|8tVFbUNUa-uKyB4m=Oet=DP0pd!w0=Cce~u)wwl>mf6JOD z!syEWNslyS)NS$u8y5@vpYudlU58FPcqe005&haK=!Nm(Mp<C~-(jSgS14Lwh{_>$ zFyfJTgLo>U<hS^9T)Uf2^Xf_&Dc&hAD9Xm@l5YH;xOo4-Fw^7P^;n@i_rn(H;T}Zl zW#|uhMJ>zsfte31RoYu^qq5jgcMe@`T}AX$n9bViLY%7quJ4~B```x4{$~2DZ(&&? zVEV8G17Vl0d6Jto1Wuk`<Yynkl2uxDTAg<rI6?g0SC{sK4t{Y;14Wo83}N!c4_Wnf zTJ33qOu{(tHe?oi1tDoxt;OYXfgPWDYLgJxSxF|uI^nWdaw(~jBW|%s7&-l&&L!&y zQQ86z+BXnysa9MIkDi_j-<w0J-M&S8ea^$)*O0~PC#|RJhbJ1p$j}^Vp7a;0eZ*1Z zyV3;!@G{?cR@WVXSPVGrody@NAEh(yoTCkQU$PWxNJ73IB3VroKm9<xF+L;$lsJFh z&g$zi%4@ZJNUUMIOw;+I)Sc957V$=N3eVZb>aPCz2FLeib@4>JrwEgp8A$Oe8ZQ_3 z%^PF(yPwPKGTk_%#+Ln?w9a!t=VIm%(U<aO&G@s*1i(ctGz1}X_|9u96MKG9IZxB` zXVcyjM9FJFp`JhFJd`k@m}V7?ZQ)kRvb%7NR4a%YH>$3#=Rk>{Z|6!Md&Xn!0ea&l z!i#^%DE|rdruV)0zJ(HKMvq@@1TU@eKk|pG6TkBd2dR0czq6~{8eECG?z4p0L7gx| zI~7xFVV8`L;&TGFXc6pXc6Q?A^fDq1UCAE3`7MEOtQsVrkkbu2?g&Kgb^YJNzxtLz zUIx}ZvB(a|&OaCnaujMq^?7xiOm2m7VFx=jkKzxf$jr7|vN#2c+qJBPo+YQYd+F?m z$GZvfR@du}=avIzOGO?M`Y`JUH!O2=F7xp@8+VwR5&<_lq!U>tt%nprNCxE1aeRB0 zzJPfWrC0FuEd3H+ak;{Kr<#n<EW8E*tH$BhU0*4HK!wRKIQWHZ=Cx>!e`-D6rHg1a zA58frRfwkC9vN-Tbx<u|(WTrJD?c6l8_Xy6;1sE%iMx?m>Om}Z{ub>Nhu93BS#Q>S z+8>Yh{fqPnfep!`26;+_dgObvUI~*Vw+}5`MA!Arcg^lEgX0*QgqoVzNDoHtHfJ@@ z0#FK!7E^8r#o-liZ{Nqo$58QWzKZtHb$fDaI?z0swKVbGEV?C;4ZA1sI;<K>|A624 z6R`eB{Krc;xY0MCpjYiaRBErY5sL5oe49x%daTX?@pk!Of2T$BrY2e38=&i1cue%= zbn=tv2k<dVNhlo`)Qv_WS`ci|*<`ams{-PMrX~?C9zOXGAt?uW)%BT|_d{zj0_xup zvllQ$KqbPf{Q()w3l*WKg8X97$2@a;UK@PgII-ksfB3Y0$0$?mms}yG`*cyyYrk6J z%K3^B;TmQ4X(t;OFX|zfLT0X}7JD6`9S3`s0Y|Cg2Q7ve%e@>oQa5ZoNNh(XO&xF> zda}lg<G`ZXG_G`sqXE1AYBA%VmS4aRo*pI|$67tBu|N@>-gZ4_d!zn49`s0CwjS30 zwBWa(vsZ)-7JHQ1G0J={m!xb^GDjU5B=Uo{#fsMdvygsXeW_E=R=Xw}*hjq6fZY}j zE8muxUqu7EGPm30%*e&7$4qE$mAolpmalsYqk)7}dlzZ4M~RE={*Z#o>F{DUTIW#| z=d2o&YP_$YYbz7?KJ&-lw|HdKvuv#XPDEs!TAKQL0pqx3Jk-JLbPaD`B$8Q`v@%D1 zEcpDEmL6d-!5l)M_2zl?5H_xYoA!#Fb*<TfS40B`B&?dAG$dmwoc*gx*XDKFq!BEu zZ~p!MtV~;_uzK7v(K&vZQ8h%|`{l#<ILAOR!i{~#?_%$nrQm(m*kTJTl_dblbHZSc zMGG^HAy%D~d7@(^UkS5}+pr~%aJC!rN$5h3sLL{|g=(+PHxvW+r)@0WfK`O~0To#e z6w=N6Oe+JAt};_`2IKK{zX|lLhzg^nnKTem`%abGm7H$$^sgPwP2n1B8~+*At>0{m z^kn75^4Jv;?}I4LpDaZxC?@%?hKnIR6HRwKZr!IN9ThqP+!cx9_YEL46`P$j<!;AH z*_|l4r;M~i2@NFj3Repwj7dt|U3f0^2n{6!MEA#MbC{A<QE2L??CJT1Uc2WW>4tzM z>t*qFaRL@-l!8-De}T)WG2X~fsljVkU^2<gEJCFaYl(|7Z`Q5(bVi`k{oIysD-dzC z$BC;M=NuyKxbz;0OnNKrZRLY1H92Qa4}}hZ)C?R@K+Q4?cWf;8w9<!fr=*=iUtv3i z($bj2#0uKIf8wU{AKgB$w|9lxi$yjxYt7>=lDs`SGaIA8Gl~WIr~sd0lNKYpCj+)8 zhd^|IS+o!@_B!j<kzGatTBr_tI_K0v9kI+Cz!LGr1KFKeO2e75C-)aW@ydp&+jPtI z(|h?BsfrlNJ*@`%2G<WkOEI}1x1+_pR(&FPc8q@;lG7mJ6v?pgF8)7I?#W5@EYv6) zFC<)<adFgA)?WPmHWeCLG!Yns0tc&3DMbif4JVlLTAc}AA=-r)XaDD^WdvbnErC8! z5HNHttL*)dMe3G!a%E450v&R5tWm|UAzP&)Sz6V~;_uKPIk6kvX#9PSxD<RbXcf8@ zc6<A)>7PGw4fzdv+UL=EZyv<9{ur};iIk@qQP_2~jK{AVsJ85S2|hoUEC70mH$hu; z_@b2qv6FoDDOL|^ge({tY4NlMBh7!~r{P;j<T&;!2k$gNnT@d=eJ120V4d(mT<32h zJwXk>e%`(F*`;cNFT5{w4m$|3#{Wu4cY<UoG(5m0UJ7+|`h$$zr;TD^@?^gFo)E>l zg<ho^Endhbj1jw8<L7(v1)X?D?teV==g|7>H3H{0ugvaVy&!l6+gw{=<@bR{ah9r| zh?EAerDuswNme(bADIhLM!tUxT7C8uEidqw>lQ0?oKk9gC@lL|(KkqpRBBPMRtnWZ zrN^2I4KAM2o!l4Q^`<&ub})C~BICPnk26W-UvvyDblPZ=m+O{c2;+L<{FmUX6oacl z_2nC)wC5d5FujxbO-o%|3ytBTP}D^qwj~n_-yVj8hqmX<W+nDUSmN+wuL2Js8oA0y zX!kA-WL6h&7!!S$Vf!t7Aj&F{2~fJuk7m{X=z~}7C7B@*i=A>!CCvl#vtc>$h#*+E zry(vjF(}if<cbrhSm31UK?4Oum(mdONp>s4em$gcD~s6`a;+kU_-giJ<+5CS-QKg# z;-g!;84C#wZo_A04swxJuQ$&`?Y9fh(0S1;<59JvBu31w@v5_*_;pT1hZ^Xh#-L9T zj@&rJBT<|#B>SYpE{wq@%wcV%)<1Pq`m$ymb5GuY!LCdqwuN$u)ifWEwdp*>4-KA( zs8-^d+>E2GF2J37r&q*TIh9ry_@0J5=dE@D91z^<%F`w*kw>;5Evt;TN0gnVZt<}P zk>ve<WyS^d3q*Mdl0EFAFB@Vgi_oC6swwQGkIs5LB;+NYeEG2H7D$n!p&JeQsM{l= zYZrd~8Mj;P6EEa2)UT5>GS|I6Y?%Bvu%H<~hJn-1j*M>gGWptlPCUmY0OI>k-TQ~p zyQ`DujeQG+VY~cQ*9d#sI@t@maQr%EK^9=AzUos=##MEikjXMG4!!H8suuqV3H1-Y z^Xfm>8Md(-LVe$XEE|tgjZJ%u)>EO?c>V1)Q}6m!WfjZMZ+nlQvD^g494yI;f)J(z ztR0}%if@Q$Gw$8I+u5+^e{;{QcY`Mj<{wM`l?BCKk1-af&#M@*Gydh*4>i#r{@uU$ zV*xt=p+cNskj|xydSHDa{RhLEJ~HPIcMe$(18<*v9np~ofj~S4sCk{{LbCkbVCw~A z4Z=k0@TEqa8N@AzR@TnkL~`ix<f(V%kv;uBVxrL?sS7sSm0X(I_Z)vBc5h_w_Byo4 zaG~i*>T|ARDV(n4!`-Qj$EvZ{kZ7nI6#WaL8@mqs6dwC3G+khsWx*T<=iypst^+!d zD>{KqN2ib$4V(-E(Koi@BJHk(^-&~6>ncvmm1kOP-peI^#Sm}+D&(^DD?Q+UU_^+a zoSYuz`>Ea}UZ2rTd(J5Q_AEphh!QBNv-@kKyRVQFBt4%eN9x>@E4ul9xfcR<?GX*A zPpI~<#S+Mi$D1GhkN*QV>*Som+yzxPD9d!I`G|=Hgy0(J1OXW*re-h2BGYN)x5Q3< z8fO+VuMq~waJzp{eE<^U23GzXzF#}3d+p2<i1^I!#c%oW$M4zC=c#=ps*HAu?Mo?1 zLhX7vO}3T?;3PcyRNwe{=1n^2SMCm;#v1LXfYLIWMGg+)iO0Rs+xJyzG!5f!pM`ty zntKygCmCOXC0QK2F4M;aaNB0}f3nVe4(aiSTUy?)t<D>r-YUYBB<4J35ba%m)4klz z{sZ+vb~8pNf52G*H<w#^du{S6TWfjBgBlVJKUQ9us?f!%S$;vAd)RVt{2KpP!JsPv z3h3h3#9-7k5>=g!oF@Jy*mdmK0l|el*cvX_ZSo$lK{#|5t0`{%;pWC5#E%?qNV+#| zhV|QLeL)}$u$0}iwfe-0nFcePs$dz?O>BHveLUEXkVnh1@_O;#&x7sHBP$q6_xrhe z_cLM4i0G`-*uA1vJ(HWP84@V~FF9Mez4v>#Jw_@e^R!JKS}{lctwQopQo~f99`6Vo zZ;v9C*5L4X+|G3Z$K-)lr8#Z<MGV1@uKfqZICm3+ug!uHC;V_nByyx7xTT`_!Ofae zIWC{*!N9103~QAsVG8P<m#&_Gx31Q$zA!YUdlQE31pc_Jh(dGOYPavBfeU;B3AeGe z@1$ujFcE$EUGOZhcGcJ+HMM>)Zu0lfv<W*@-Mi7IE2^p9_Pl(>T;Mpn$}qnxI}hyo zS6<}IzzcG%=8u++X#E_hKY+yWyp4a+Lv2kli&p&HwRf0(b6bY<-Yjw$n)HFDE>VC4 z4YXo~A$}+#cuNp}1jjl+6w9{-X2J9|WAQ0V5<5WXmk1-@bEBVnlR-bZUB%k(3(B?+ zPsIEaAMa`W#46q|U8hv4@VcLj_0qK6h{J|gTcuK0sZ;}~Vr-%ZdZpH3r_+)6VY0ta z)@Y5!&_2!uBC@o8SJ_0rTt^<JLFTfO&H~D<$c9x<mUC=wE=(q+%747nR73}@jfz9! zdPO#Q71|(*ea`Q`90VKNlQL3vr@O!Oqt7eJ`8y}>TS^aQVW#fcNvXWn6Qo0<LIqt` z^9T8%2wgU@UiL7r@q6{Tri%OC`i0)lEAZ_Iz~_PG_DP3dju#MQU%sJ(SJ1<n9`R|w zA1c;}Uw~k=D+H!pc#VwziERcT{xwo}uB{FxM~A)a#|Zhe72^(PXSZ1lkNa;x0q9jt zX~OpudRyXL{odt^Xna8sa=KZ?;%`M77C9XU4Bb?wPlv^MR-^65eS79Vy13{qHcVeW zyL&j!@2g8+Y_-|Qv0PF5@bED8`nmS1{sd_4oJ4Z*bvvUMwnrmvml}2fI>-C|T6{^c z;`1Sbu!UNUXMr{ruT3D{1Ae&^4omcIW~PKd@La<0;C@roF2Un}KMXHZQ<K9kz&l%; z-swhvFnT&fjl^P7NM)DFLI0L>MO6qs<^>Y1((2E;b}w!P8D*Z%ydaj02Nh?lUa@za zy-OBeJNqrJQU`-j|4cbc^x$s(WzzFE2pWhW6zV1jUY<h@fIVqkrujW8!B#xx-7@oc zNC~W453+&4?fM)8@-^d0EZ~`jGcTE3^Ov|kd~geSjRy&IDP@?Kc_|ZqtSePs_q*SP zK_k^!%U#0ZyXcu+)|)h2l`)}D(#*lH&FpXa(X&q}cmFJiya*1#SWv)R06l>8l%RcB z8GFliVz%zl_&Z&~Mku_P&-FmxnL@QZJ;AlMsvh?hE8BB!ojrxdZ}5xxZ;&8}XT{+0 zKJ2;LLm6WY1IiE>IwcovqcS$%bk8DoqOBoFL>k3yjTP4^r%efiKLYUtet*pV7P^Pb z1o7LR;C33q?~}gS_7C;2*Y3!%9T9}zzdWYXYRCovPQvd7JKv3h=g=y<?1{vU2&;ha z=knHxqnb}84lH76UGD)u&2gt!55!7Jar#}J5i{?OE~g_&%MGo#W{4Myd@kJ3ryqi# zgo+ikXrwqMNS>Lb+u!tU7c-epZ)n4fD|Jh*?B1Q6zL`%z&+IWEL;>OaQW)-z@DbVK zC3U5;%R`@&*1I&A6-7z<MpDYKP0jc_*&C1Fbxy?dF>UXS)`1BZwm;%Y_{YsJj5l1A z(gevTD9rnVJ%=2h@v+i;h--}quLs;%o*xf_7R@ip%Y^@?70#{2CZ?#zS+S_K*f_*y zrmrp!o*jN{{qmNsgsgaJYX9~bWs2_kkuKR;lVz(jRL541|3^1sBQ2eV&YK*wl|Rpb z;-p@ND_tT5OnvoJEzrmwryk=sq0Q3r5-;+jo%36_dMZN^!wGOqk7<~!lL3#eV1bQ0 zryj12!dFfVYHC^oj`8tA#{N3RhqfbL{Y~uUWBwW9foar6boBGfZ~dM2@Sp!|{u^%K z>B@GGK6%ggZyZbBe=SDvcS3ZGiuncwIubUlW7Lm8r~Ba^psO-A^KivA)PAbG;zvM8 zBGvE@Z2T_8p@HP+LwRX60ey)=0_R;Qr4N2;&vClMa>^*f{9FAkO9O-T(h!o+BVU_> z5k`}Bigz1m-H7<_tcRUsxcR9C$z&cxjIA`zK_R1*g8zF0<PbE|^8(?>T&gmq<={E$ z6<>=3dSGlqp<BBw7fX5iC+pPy?0j}uUT^X7t?tD|n-Wf>4PY;5+{70xMF`;x&pcy` zIBd1}J@Fq^ldpI`ObF0O-AUdkhv8kE&@2!6DUaczkO$i8hxSnwxty>5UjPa{^};ZU zYMf{k&s@t%byddt0oO|J1x?_b@>HtzGbD~cb4}2AcopsLUH+3iXzaM|QYn+dT#<ir z0d^O}f@99El<{nH_tbuQdH}qN&cw;nY)0}W0qY||IiN8wZ(u5@Vu_CCblI?=zqM(& zjfLpiciWfD!1mj&UZDxe8Hsl;?dU<%jc+*5q#;pOzQQ0O#X5q_q<Ryc&>R}lJKRY_ z^{sIcDk-nzYF&Zv#Mb$2BVLXml(=S+QTuS`qUe(y=s!T-J2_&Eg27=@?J}i?gZ)Fp zlpV(*507D$2u6haUN?z+Uwbaj&WdVLjp&#p^j??QM&X3Dw1-N*Rs!c1QmdL3-1?x2 zHc_-Y@`j1OjL{I}xMTKT<wRp~v?wXA`sLf1fkbyQ798`I(qtKWptz!H=bTUE?1Aju zT%0^ml$w2lwGyWc7i1)yGyR`(!ZjlU{eAn)1^?uLj=FRa#MexMZ<$Z1Sv-~R+w>b& zSWtc5jMwfSM7@Vb^KQIxyNI2^hTcT4vio78yzbRpI*X-DnLNa)K%lXZT!bK2lCyR8 zGscM`6K@gj8Flud8|X>*VFN?dJ={gzRpiO;lF&Qpcbm_uY)W;twApplWEjdEd{=T* zocrF`qi8Z1JSj&^Lo<guJ&d^ER5nOqD9;_2P!{WJrLe0LE_mPNq{wimR3{@sA=GeK z$+4`aUQR9se%^Saq@UQ#mi>$*AjK8xtwl5$df?gBlk)dKY*aWNo+hYM(JY%_AT41K z=a;jvvUdB{jk3YD&p-Nq7cX5HdmrzBz?I7ujs5wJed{FIE<%V&zzs98Qo#v+$z-L3 zk-|QKK{s#T+_Zib{5${8fBfg?P?Gr6X{!iY99&coHoywaJ7M#V;E`c9{=0-k7H3K! z*%O|vOyWD{3H7)pcq3b`GIzCM2gkK5<pHst4${a^WzFrBgesgid8V5Lp(;;AOh#T_ zFm<}Rsl7|ui6;vjq-GMdS|&@OSP<etB86T2R+bl0q=$P3P#qGI=qa!aLtxY^Ol^K; zLI$)6&S+*+Mca>x$4xETWq05!8B<<Sk^E>3JS90%qFqHzp|sO#bXykxSdv;myxbtE zV>xT<chCD={^o}Tc;i5>+Xd=WG|MI%;nG{*fAsL%fBZ!%GPv;9nV$as6W%8(A?fm! zORp~)iDiOeHWpK56%_5ZC$@>NjQ7>wViL!xdh1)-3My)9Sgj;j??&exf)v7x=oxJ4 zOJjso?QmX1Z*G=Ql`oq*zCGGTG>Le(rEFrz)4GO+5<&4`ZR`bNJsk*LM^j9w4GoV_ z@Nm~$nydh|$2CYRFXCADLCKqJ-`Lttk**$+K3$s{HJ1=s;z#0x6@&#Ldvw^HDer#s z;{2)5y{xuwv_c>#KiOMyg0T-GhIBCEG)<!k(Q5K3(#eQWh%mfWCdtXo#RLvs+Ot)n z%v^a95?<Ag>L!GRQ;G2QbQfuxZ+G)7Ce07=byNAH?CwYP5T=CkgsYH{Sy*!V%*ok5 zek~Tg{XFAekDfjO+$L3!3TzYuZ<>e8$^`iyc+*t$*Mmfgv8m}Oz>1=mVFa&CQWLs= zG9Rpsy(|$muCI_PXt6EgrqFa|Q+ZQ;nY<#E*vlUp6=FfGoKa|giLkcrKB^t=3Vm<q zI~>DG>d_u`OK1?-RNfr<ASQwm;z8M_o{Vr1j*?CIOU9&77g~)(ezl<3kdPF&Ta5^8 zBYSH-)JsmmjlPei)I9Q5m&mw}acKt&!l}^1-6#tTq4sT=S(%v7G4S;CH1Vp^l-FXI z;OGJAz9d(*LentOrMbMVsBbi!{{<#(5A=Ag+UJPsA&54Drnd&<=j!EUV!RuKfnWUf zzs7Lem!JIioN3>zTBdRC{28csEczYln%IIOL4RwrklTAA@9mWb2u|=h-+dw!3r_p? zvkMo`K_O-`UpJz8N2tp%sdMwOnyNikPb%#DLql1dc#VQ&wNs%6v4rZRiD}L%GURMs z1B+4ZFz;QvJl7E>0Ema`kcj8<6pJb5UAEl5XrcP8Bl-<qXJpn1j>f4-e5M=ow%Erm z)`wOtTfAk%8scWgP;|OBHOej_()!D@hFCE{*zmy(u!Ot4pxYJ|btJE<!;_}KHpLAj zmK2(IO-xLRRkcUmmg-LFgY3M8OYR<qLN{nmNg~?CPi2zaygaPn;AIQ(<qlh;A53vi zdmD6uvN6H&HRBp3Z~Kc|=dfyfh>wbsilTZ5VvS31F$Ekv%IX=Al(2y#1pzP2N(j7Z z7J3I`ASmeeHfJD{;QXJbfA{6bGyeVM$|dSh6vffSrMijy8LkJ>J+wIE5q;h<8>+oF zP6{qw+|VqhLOmzdZR%?<&<Wm<jR^|m*p0>@!&Sl!ev63mM7q!ov;@{SNFnh@b%Og+ ztwjso+VaI3Ppo!Jga~08hF>4ueA4bq639PM@Iiaze)Gr%?PUwzTVOEB#e*`3r>My@ zlv_Kgk_64&`PH;((`L;0Y0><jb+k2(8LoSIW|y=~&|9*okO-T-Hb-^Mbe6BsoIhvQ zO!(Y=drZjPy=YaC+$fctf|Kqkv&ejOCkH77CChy_<xOv87CI>*8x`&1rZP!+T?1Bt z3_LYeOub-k=v4uc;EmDJ!}86kiNPHb2k%3}4SxLxRt=AjiB#c3MD-9PvEk8j;grjy zksITsASTH7G{46bOi=>fj_#o)xeOb@(Pfy73OhCjQu0kGUR;}CE#@YS_E^Z(a^x*c zuE9og>}LC7&@_7aSaWK;HxnbZ9I8Viu{F0-L;TU)eC?p#g7LOQbJ2}MdWBBM1yLYq z17X_<G_WJ)o;*9xx}ZCc&su<5aKeXf-?F}8>iix=_olWknTMiWa1iD@MYa^$%QU7< zvRN}{XsOTBU$^+I*#_tH22+Uy{j5!cZ<$Bjo$?Ym>fYWbC-nF0u3Eih!7p=XPyV3U zvt~+l^Gqy<+Y+=SNM*3WrNX3RG&M<F%IXsA;ULxn6y|TYaZj)!7o8~%Bo+_yx_Z&3 zO>%>HG+Z2eIPDn1!3(BaM9o1~6C903<r0_z!STWS5fV21`SsHnHasIUQ)O~pR1ZO7 z>!x0M>m=uEiiU%+PkyI!+NsvDa8g3+ouu;)H<w{@#rT&*Rli>NA-LFKZbOr(Ff4A^ zwRcy`kvHGMh$q-CcAZIfbPfh2qWZP^C>#vkPl%^5fHd-wi$6vvA^wPZbya=AlxC{C z-t*SsrtrJ+&p1xZ!=MMp4;XAwpZfD|UOi7<@rUeFPfgCDOob$9;o+gvB(OQNXD(ax ztKp`l*DbgBS(}KC3UNHQ`{?d98&)iu|I3`|)2EAl^D`%pqx>((-b48}RLJ<rjej-@ z2j4sF*hVjr!mUT|ks@!kN8JGt6J&$Nd;=#lydW5RMT%W14OEs4d_227i~o%wCQD08 zo$}7VI!~dKYxKO7>L5vzD;pS?6e;FAR6SPJ<Lezj_hC{jc6cinu28S3tC6$bh=g8N zkWMu?h~}MC#_8-K!9#}-<C*{iJ|C>haTB3t!uD!esIwX>G5D46(HU(^<}0MtC7*Tl zl3M)1X`^{QU%o_hfjC>Ju5PBc!S=$wT(_$oanjAoQlA@6*A4;`)LgW1fduyS-c9m# zb(?%sPfh-z43$twJd{DEv1-YpY0}%gxwF?UQ$Mg{wcV+muod&TVtmSE!&Yrg^<U@y z^wV@{-|go)vx}>0$cHD8fI`OmC2|KBx3;$y$10jiNRcU~B%XXV@8!8uqO9R0xcI#B zMhr38o#T6^1oZjgN|;|vrud|~uFvjb*kp69?@>T8ffhRkWdK%Igo_f$Ab6_RT}qKb z$fJj*l2isk{h+lZ10UpYYl8Q@i!7#{SK&?AsFVpdlk&qsU)?IZA0fUN=RFTaw~$s> z25V|3qxb_d@a87pwA#B+go<wp(^#}{*OoQs5A1ThdLa1WiMSVMvt6!~``xS#xLx3J zEyeLd)U#8bw~t!?Ze*^%aoJ+c1=5?P$&Td2SW?O4<r{=**bHTo>X@ML-07ox=g*%% zO&*&*eYu78tod5<yvHxU{9>?uBP3lycmf%&I0fd;qF1$NdE<u+j(K~Q4~t$+nodUf zp=G}e!$Gq4=hseRNXmB2{hM$Z9lNzN`2`-%OHkOsQhN@DO$NHEn!^&xz9idXH!w&A zfK^jp7xU<#a^N6%d2tR!$RNskCY$vSlBDb;oyDqlTR9qv8)7l_O<0&Rc9Izw6vBdX z%r!x#IPY^v7dwP16wAovM;Ii6D)h#NSbwLp2Mjbd7B0ZI?b*G}+u63Np75k)NbYR% z*uX(j>X;Ie5)Ab=jg6SuvuDf9^_J|ryY_PE=6mTs&6y|LJN)$1Prv>48+?uR+RJ>s zToIl?HfF6P674eBE$phC_DoN_VZrf;(2HH23tyGcUCrt2MRP}HgF_}u)k&iRJzrx; z%D|l-35lv;S7a(@T6HA9q_^-yWvB@Y#J0VVzEA}}aXbumY?L*kS5TE8NTn^s<F?Ys zAb9-Cb9Jm{@+U7wb+_cOPO=0!HdLmJ!oe_0;V)pfbS2s<XT4)K*2#rhTHC8$&8;tw z$9adND1LdYm*^SWIBj*6#lg-`&K)vZIjIZ?&JBW*51Qz|etx&8xPXYA-y>^<V(j4{ znRHC`&0H^cC-dFg1eLEciw89f&aHcp4{%V|xp~9&sOdk=l;KT&{`u!|<#jf#wtHk# z(ZC>XCqot?^y`mAyQ8;@O^q!T>7dK&!Z8z^;9ZH&+p>M&Yr~f(UWfBKk%mG}1Ie>$ z;~dMX>#&7o`&Z7IEF0`dd8xEH#VJb?(cY{3bg^x(N=g;?4h*8XBM+1(aV5uw7G6fs z3uThQF;8~CO?p0v$w`UI^Nt+J2S){WUK?v;&Ir$bUY4S?#SSEjavf1?^J;?3aG&oC zMe$Qk{J~&zQWK&c*<3q(aG!zRmPHHI7ZBgJuGikX&%*lZ#jwzz`sP;Zh(FY@9Lk*p z)-eHVbAI*wkooSVS{g!!hz#Pv1zYw38RWx3@HJ;Wr4LG;J|-KS7=*u2-E^nU(|b4a zii!!>+Xzo!r$2u~tVdtl%q*&iOwWFpn|22jL*2w-g2J_g$@aNvx6`v96<0*pH8&9< zmLF*-Fff%Db3CED99v#iP*OTB#;PD(xxn3aDm?|H{WbdSOAK2I?^e&|Df_U5Ql80e zWKBL;UYwWYd{y~m@ST3p{!B@3GQE|Y#fwp@K`b_zBjm^&Rhw|5F56jnDUu{R<t=sz zPn+nnXOOSYcEWkKK@fkSBOWGPo{8sq*V<m+P)~~pd~(<N*wKSVrl$JF<~vOHj`_MR z+ZK?*mao$0H%i}l&n{!rg9j~dTVD?J_pYd^#hv&A=n6)|PYu(dOfVf&QAMS@v;Dc_ z2lThB6$`ng6%YQZt-tnO&IC9JzVh%Bd1Qk^ZS&d{#}Amje0D!0JDUJ_0u{E$)bLy} zoX%;hYA7l!2~5qp7m@UbCvDfOn9Y3mj@elre-XAyM8tydwael!KobO8<Vo8dne<0$ z*8RfLpelMXG#|zClhssPiOcEX4Q%0{zw0Z0f<LIhqx=a4Ws94bLQ=h?r>9}t(i?{j zQOYx-E_Ni!v`go-L&@M|*Bh$Bf`XplaK1P*uAbdWCNhZT9ah#qI7EW+P?zzdp`m2* z9u)#Xl)Tt=q&YXrnw_ShHXqmWF_b+6!~3w25u9*9CaWnoGug-Om5uc!K!K)aW;={b z^-V1J;B12H8JnvwTu1^qxLI%KRBeRj>l>LFnwlOuWO?oKIp<flNr|!bOjT+`Fr0d7 z7!PH9Cm+np&2@Kvb^ZMDy@tB$RxOowl+2nnOG2VIS8L6hJLzL&Z~`0zUu=9qPT8Oc zTe(!rSby^ct3wWUkCKv;AW9lnkDwFb36kp1;iyyHSYA>dnUn9Bm~I`Bc+@A>&@o0w zkevy2Kbk>B{3QY*Linn{G<|8i#WoF^?-OeTT>%{e-BMB!4P68sr5d$)7fLmVrH!aG z`CwsbncmF5V2I3YtVT1nH#Jr(|D^O`W+mA|cvoW!N3f-Cw)e4xgOI@x9hQh|L{B~l z8okQ$qU;ElxaY_5fCUAe_Vj3G@XPX|oQ4)2(Mb?WHHfWs5I^}~eNphlKWyRfa)VS8 zbdu@8U^S76#iLY%iZHm*S7M?-UL%8^m6qV^?)dnw_4(7s4jtHMY-+m0&_qu_6)2&v zTDzX;WKeVQ5@}nlZ?Z?v(A3bxbl<+cr%wD~bMva3lYK&bOl^IAqk`r6kRKMqfv1M~ zP{v7JxsvLd=%@%M`zL?`518%Rwn1A<?m{`g{`#vVL=OrRufCDMCxesW;P%%|3%6QI z1R3YfmGe7;B68E3<rapz7f&9xdtwtB?61Nw<d%*e^johe{dt?RHdyTDin{Eg(va-D zSIHT-Vv<h=$L|A4pkvGyv9L^p`_Oim#GUU<c!Fmv2w$%_?wrqf{iC!F9py#a4V?yE z2OS9A2%QOC3LUG+RD)eeLxH)eHu>PSgF7)qW(T!r@v?n!G0T$dIi-mV7Hh1T*AMK% zvYBgs25F>HQ=XwzgIF(igiAg+T0@|!tRN{Q&F4|<li%^i1cmtHLn)p%d2s<1WrYnb zZA63=DAge5^R9*9$p`6;RRWjUJa%PPN2;T0SGtzc<VH4!G-ME^8dQWSDCsAN0n_56 zUOcurcj}M*`z#EMO@L|v+maTCP37ASOo$E#*XwM@sy9Gp2;IBO$YhVj9;@R=@7q|1 zhXvL#FuNXLw_{9K)D$t4dELs>F%^`SMMZ|WINIH_zHs{JKA>P*H)t=>TBM+{B33*I zTE9h`P5A-ELg66z>f<kG%$R|>Y*6xAJU|Q~W#Nf~X16Y%2904@NI+IzzCvY&O`~`M z{mv^)$E29#gLN!=QCV1e_G2Jf0dafWV)W$UVq)DrMlU<=1UKO!-moBirL4FU?h0-` zbT4#rO!6t{aOn107G?!cAUpxh5k;5^;~XJ`0YhqL_{-n$rq2af*<6)trMncvW_me1 zRpv8l@99T#L;+Bq0LcgInwratbMq5}(|jMrJ|zMWoEj9Sdfv^8^D8UNsBI)h9Xu{d zHK+iC<b$j>q9z}NJE5_}nsDvZQl8qG@(M3k&dD!Yi{qKhMxv8J5PzT>XoYZjs62_) zykj={xxGAn;)tP%iJp-eR^)3cU!lE*xMWa6Q&U9TG$yWRyl1D8$&tec?VmrcsH>MF zJOz94heF^Wd`XT1ij$O9*9HgrJ-Tz%>X5m~jx8HjFVj$ZH5{?x!FluM!OOIC53IeH zI~5#Ud)ftF1K{tcpMFxPY*1Q*?p(cM$?k0%e&1(&`|4S559i#%BCH-C&@1#SuTaA( zsK0^D0WDix>IDdiRVWLa%AI0#;$kfu;;r$pb*qoJj*YdHBtBXZ4v!!%^<q(3NCQXq zY)~k=N}<-N%(##7eo}u0hTyD?_6Ub^-NVbi!>WPiTHn_&lvkS-vQnbMl7g%ce@W0F z*-%+gR(^b7n%@(s@<Cu4DAMC>Po#Q3$ffy|=B3rp8xf?S+9s1p%;F@BCMLTo2QHj& zo57|%!=t4loynvtS=)|Pe$Np!=@iN3<*Xwd>g9A`{~jd(K|$ZP+nBg;5NfI@v#$UY z8k?EfJ$qcol3Gfvh^eQhxX}z1i!cC!HT9rt`+dLh#?{MlGGb=VoH-TF`{kEk_%GAa zJGAzG?qoQ)?ySdx1?v1FCm;`0vTRUVTeVDUzp?&fn`<B~l=(RhBjJ)#Y+z_up6OOu zmjfuuIcB>`K-gsWirJYNcbuJYSNR(?_|VL_V_xIyji}J!p}?c5tjm@5nL;XYG%plw zAP`f@jyx?b6~oMfjG&u1JTjTi{|u`J2Dm#aFU4mkA{2@>JQOFXuB$CdkIo3NO?Y{h ztTa#zE3~sp_kC256k1hXfd{C8Acdk0Dhfg#RYDM}l6q!%dYOHAK~8KSfr{Og#d_I? z=Le>g6jV170u+Lt6dm!Dz~vdzb9qKfYeL{_(;Zv3?le*+>iQraT)mc<vOz6H%Laub zLL>7nn>M&PJZoS}n)X*gC0-Q>2VrL{vjiRDTisvVo9xh8sIHE;K?B-A359-(^^cD6 zK|UN@cixLn21QE|=ggUd!wIu}<LamPZd5cdkZ_4)l8RbpO?0vqPC(dX52l$E#GP(Q zxTolC8GJ;3+-dkYeD;Gw%w~x1u4UFroE#eET~d;%s(A5(vj?$aXB8e7ad=YF$B2HO z+_6Gw4+U;U5us3mb*V_STUA+>6YWV^AUGu`e11GD%&EL2pWfPzAO(pHx5bMqm@Nqf zHEwZbH-pOdy448=Y@PEe%8K;HITkF9al7T4qwCD#e{B=jhHC+3uVYJTxz%O#7993D zC`nPpEqsbUYRd~Q95Na^ysN9LZ_?eVeA>0$z*HT77)Pva*s7<Jz5*yY_5}=dH)rPL zh<(I;3KM^*DRd~q_E-i9x{HG?gbd?(+)Ne^&YnG6=qr~PpIduBZwwA@xDv8JeW8T3 z5ht=ip|)(XhRwBewUTLHk#I=`k_st{0+m%Iy3=;xmJA9y)Cf+EJz5uky%W3G5O^zi zLwH+B3J~MLJ!m`1s*}V%Rya8$?UK?Ax|*YkeC_4+bdyEjVuH`={jcNjyadS*<!_q) ztdBA>GnGXsx={`drI|8CyY=kW!gN}q(-p)4MSJ{<)A@-(bxkY;DadNQ_^3Qt$>b#D zSG&fR-V7|;?_N70xBFy2?OBo*>5Ic<t|;35x=)_i?{Dr}f5pGdF{T73tPnJ$1BwwY zA(QiPSDG28yHQ*4<gUJOq0UZ2<ufns^&5#r1{W_~rV`Dd*5W1O@R62AM0kjZN8H^% zptzByrs$z8Rh<OjO~))v@jrAX;5?{OG8M~PW^rTH8E=4tn{Fp;zLz%D8~*(B&-j-O zPOzOi>+(y>P{BBS8Wc5`aMI-#xTq?jo_t}p+rnsOX|cbT$6u5o?zlO@2L2-a^-!7_ zycOY^4`~^53*As91ibM<c89Y0U}ib}>a9nZz`@q2`#3x`S!#VFf79%4o>k7{fW+a5 zP$=1;`h*g&AS4rvefkFif08@x(V^@}*E%K(K?+h@w~BPT6^-nO%!;RBB}Y7~#nLpP zZsl2)6cNysfQtn#FUs?ZcY$mT+2>Y!FQg<mwXC?VNhQw$i9eKY9lZk*Xa@7Pu3sfI zS{8EKDI0qkn=jJPBoG|b*<ql3IJjJUwb0_RiK%H*9Pt!ebW2SMLs`N)2|$|kx2z+| z=;7BIhJO*E@>_Y*4Z@1G&v<QjWNde0sxQ)->XiTl`EU?&5fY*kw3jcAj*6J-fkI3p z`W;uHwMlSfDz8p<jM+jsG&b2?Vz!0QOcP=cLXN_sxU-e<mulj#LP-;F_!sz3_(S+# z@v#TtWkjTblN}yhc}=R&$I3&s8dYR)8OCbwR>G#dmdy)FEkEvE_3oUYqbnyq+d2DV zd`4PXt(h)o(@a+Ul3rY`96~WTj0lAapu8d~;Jq($+74>LcsnZu`(#9fPmkus2LN3~ zkb;byLV3E~3Py`xa@jfG3O#|STY3BskBa#4Kougw>?rRG-bI+S1spa5u5qPBwaTM} zC~J+XjMvpCviwNP#luEZRuNdcUisu}<2F5_u$`4_)+xV#Hf`TA)%V%DacwoTWpc4Q zMUOEx<qYM)Au)WgzJ&)$Gy<XJG4UXj8x|e^yXI!1V9r6UjRqoK46(@~B9!rI%a>>h z)ep8IJb?pi6`2REp3U;68IS;zQP#a<4eHs=lPV$fqLHscgQQnYZ3?XCSjTSjh%bdy zMB~l|wxzni$`xF^PvdJ`F+p~meO!3ZL%)n&o7T(IF5Q_Q>)xh87hUixPcN%iAQ>Db z6o?F}j1MY8v|CGWNO8Y~0H6Gx;BX!iH6lnsLQbK0-7frY-wP?*>CD86qY3&EkII<v z;7;Wt!t_YrGhW4tXbzCm>j7o)`8A4uD3G-V4Sp4xh$J>44tGQ!+f`DWWa)~P_y-z` z7HQ}zx1cjPR6gOS8Wd`m&Kw_?(z;z4#qQLUHk3uzNw{&rif|b#A@iVUiQmGd+7qf3 zuQ0tJ;`P88AwrobvO&I??$ntmT<Su20zXU0lzcEL-5T*VqI+`2O~Gf{($Ry)yvhwq zFv*Z?R}v3NtAI!%;Rl1$FXe)R+t!4JhL^{<bgE8CQ6K9F=d|YmsmDCZKgdHY&d3D^ z&8!4^iFG@;tVl5tSiJ*CyQE43(QY%QqTS-`#MmdlBd&*;KRl417>s$X5(Ft!&gB(X zj4QpY;ev04A|ToXKj>Z^5f<8!@LbVrTuOxhNzYOxwCE47s@Pmb>E%H&iblT54eFi= zNxjOXPe?T=B;DD{RVt~rv2fucoYRbr+x3;-GwU{Q75%NCpi5q96~EY>a#G2wDRU@K zRca?OIdLHY%h`xEO`SdO;b6$v@AaCKP7BCrc28_jvNODH-GpE5Hn2IaF}et@5#3#5 z^cpy=N|Tx+lh7cU2DbXavPSl0|1vR0f(;vFhiH~p#@@uprSY<Nfm={qYNTIdtb?NE zfe>2d7ZP>bH^*Xpamf#U(Np$DY0T1fXqfPZyPX>YQ_Hc{4ECX@MjRN3cCq4vxilYy z^AzD3{?8jYZHNq#j1RV9tsjIdoLfPqx)?<eOX!yFHE=`Lt^h;`!OIB|fyX_{mD!xp z8qK~H8Rc|@<H<SiM)}}G{a}LMOWdZ_Td%Vn&m3cg_8R5)iN496MH(7nzuVng*Wuy9 z_C{%=OHIk6S!n)2t3&3bL}p9Pu1tV~o9-l$vZavv6fGNgw#lhsatmD%StEz%6}pc? zd32XcUQ`j2Q)TMfApS7$I7fHxXEJiMYe7Qa6XKU4FOq#ga%5O2R8-`|zmWNW)Rtfi z%0vhjb`FY%%8Yy6Y9%`1*p<l+2e0@~^|m*H${>jqCN)w~B?SB!=m+JDFQ&J4Kp720 z@f7E2J`Wq%T=}OSs&|L;O{UBamp2A3jY{Tde20Z8Bs3gDGME(M|GP(}s+yzk-1IuO zw1I<-Vz&o{@Z*dRwh9J^ad>C;@4_tz8!fxBN*kXL0|I+>jm(wbAse>pO7ivAYOiQ+ z>r&>>Qd7b1?adp(kgG~MW3lEUQX;dJ#~mlY!R-!oC?`(J#zM($P){gF?}|(YgGF6% zjSr6BkFXj!JS5=&g7P>;6>r|O$ejw!E3r%p0fprySs>cn&vakaOZXAyuJw>F>49H{ zYfv17Vr549RK~b~29TfGy*1vhhW0uq$_rwIJ%eda{L?Oa7w&hfkv)<8(%)oh)V7^s zLdiU8itFVnrHX13<5S1fDo1^itWZb9?-cDRo;D3FJfzhnnRWNeTfexH$+;O=rYepm z=uoQ!6Eav9<GRDSN#)Hs?NcddJ`FVKaJ$}d`SnSv3$-*P!zt?Ox`w98XJL9q<{DaB z_yZW~Dv?rxYb{wS$@eohHXa&M>g-cf)o5n5y^~bfJ+@guQd@nUHUSQTYp&HLWn<%_ z!_m()&W)e<6Jm;;VssEzBZp^Ky+&i3tf;~zR?7TDj&z8nm71e%--U@n0c3^>NgHo@ zMHbxov_R}T?{}{&sAkIcXh0SU;!rfg@j2;atCLWa9*xMIavbEDNUKXegVd`sFLovM z&4)dz@jwQ_&0HH;aWCa0gAk#7OzEM-Q#k}5WR^FeNk>X+f2-R~3X~#o*_w@8RD#2F zb{eYVERP&y{Zma>>@bkz>+RjUx2H$h+2qw!IGV?@X&x}$NlI9zuDNWC3{HT9%M4GF ztfd+n8ew5!=+~JLHuyHKzoa4>F*Opn18qw&JqV3zvZ4w{T8S*jfD7(YqqGa|I61&f z6)r7xi}qiX@XhzQ)yu#KK@d8EWb!yaFZD<l=U7!;L3?xvVLPRHX?~gvTHQgU)y22< z4-PSSvLqR_bg#wLx^u@jTTVC#;(A=Q$jG%ZwYVM$<Vk6reFL`Fk4aG!Vew*>5HZPP zX_b^}aP?YM^~0*Q>&1S(!-o&|_p2ahl$vTsGpDOpd--Bg!a<FVMnd7>>SK0fYUbj_ zi&Iik(C;#V4Ad@Z*$)v}Bf2}rZmVfdMB|yPr~>i~%6<%3y<7^8Ht!o<$3|SkL2&J* zQncu|?{#0*N&e=?ysGNhQWQx5DJV*yK?s9<3lculhUV4;hw}*7Dbr)0{$5L$qFMqA zX?2Nm2|Y9{m^@0X7uGiIb*snWx+70;F-`)CC)eXjEvZKWc~Y9=v-?t=-&GY$3%+K< zW;_s#CCgXnn^>r%2jFuTE)r9U#LCKQa8NZ_K2;0{M@EJ%EG$R~2bUO~`v3<o1@nKJ zH{OULO&eFOT2)wBh<=d?XKfudt_iylUn8zN$M0Zr^U+8qE2=1}XBoTFWj_Pl2$uq- zS)9c?83aH3T8b9^eqY-6e-c(A;iCZ&C?bOdvI)s{r}F&7U_|Sb@0lU5B%F6pX9lXx zpz%x;1FadWO(H%B7c<AbtMIw*$Q)dZtAOIkV)=4<3lhka&tPYA@7_JhkMP>-HmC{? zf`epFwFE<R@e(~%6d@V&&AMWuX0KnrK0K_lSbjB?k7mH4w{PDj9~@k9)J`ZI+-6@x zjs`-4Oa_C2evS!hoxQA}%wveM5z+nA4?`7PG=_<2kdjGk$Q2Y8Jmn*Ial^IRlIG^K zD<c3LeBGq0HAkWrE%^6n;zj8{A~LH&>|v?17ghctp7n`~bF61`<7`hLRHp)eXjw}a zp9hVDw7LYe;gL6-PVx1)o<)@6dFhcYxEOB%#S{FJe;JfzN7rDawD9n72<gLlWbv4E z4vHqLjtnkVNl6w-Vv032H3I?yR6B~)a59={G}`jz%gF=>wJ(PYg@fQ4t2Pn6bv8CO zN-mv)409WKGn8Iok60VAJh|-2@CYIl#FkXB+q*?<%EK}$F*pER@skLp86747865ND zPb6y5A#2&*A2MFmbJ|7ZUFsi18QBRu>6)jWbcLyrh|{U8GD-Mru|yt03W6FuICy*_ z9&S{2CH|KlX@iUL8BjdIUE)foEKJPn1i>{Ly~MT3%1Rv_9ZAme+jg6%mJcppwOR#a zP)o&R5PYht%^NpvEGa2b`4CdW&u9i2nvszaNzmXT&9(P(r-Fk^cOE6MB67uwxVSj< z<4YQAXvsh!#zYAvu5C_2BbFRTrnVm8B}(8v|EMQMF;?QFcDg_RL-%<xVc?(>Ls#o( zF)f<&sqPEePn$uh<x>ckP)H*gCY{i_KrrLtUz|pePE}5Obg+)eLPVa#1`n}GScoxb z<ndO7ME>AnJO>mD`YvaN(BNXYL_y_8NMN0vokx#~M-8ef)GNpbS1LEYSZm2rm6O43 zyNrc?H%m)P9*>8I@uP;T(F`HRAW0)bHIj8Z_+;=c91IftQPw{Eh@kB<F)?XsYC=D| zB(v_mmXNHIh`!<312g`BNN_Y($z(!9jImTw9Jr<+k)6u?im#X!=^DuM{)auPCkI=h zP+~F}C}Ji|x~i%&MCerO#n}mn$dlVfgoPM~q*V}Q>5)9R80P`Sg8nMH1cefl`##Lh z&Q4ujUBsb&>57%A#bjf?cGDK+K#XN8we^hm;Gn%ibe#_$K2%|nL0rSZF#};5$kT{j z!*{opARJsX9u97}8cM`QL84O>N=)8WWF)KlwF7Mn!f-hB^okb{1CB;1`D`@AxMN(j z<4GUv;NZPvQ5;QcE&+-ZPxdWaMK$X5OF7>k;Oscz(hw3Zk#N!ph8Pzlg&;hq>arvb zh{#g`RS1R{!!-<C#t>-fktDd75CO$}H*jgJVscO@F}W=%DM>`qHb94XfMwvDw(Bok zxKN=GWA*xts;W&q!J$g>I7GX2X-rHEu7-^oK1VZ?$pn!aapB<dLytwk!L7Ds#A?pU zl`9hx63~w<>5ak7bxSltFb+Qs5#J3h>1bq<`^dx(F|KH2>pHWrJOIr9PB=uH@fTg| zP~wq6@Td8@zkE3*9F#js`ax(ZzXk#lE|GcCy$>-?^LdCXQWpJSU(}<$5l<|_o|}i- znFYTz33_D`=wKY^XcXXN<p0{x-`UX5#lY9qz}IcJk30A-_$T~3{0aOy{9kwtyfVPi zDA3V3$iW!?dWgL#yjA$KJ@B^h=I{}?YI4HMbBM@O#NgqMg&0EtC}J%=VuOo`5>PxY z^I&W#3MD4J(dqP^JEyFTyJ`ClJn%B`?FJ@GmMKITrlG01S#K9knyF`OJ}&L!wrw>v zHFz5{YPcQEAX2-3|GuI@vZAVyTnH2a2f-IET}_m>SXfxFSSX&6D%Lx|i^#ozs2m<V zH2X9}e4{Z*jd?f3<iUBV3gg}X_Fcjun%QwelEE=Q`9gT37SEIa%^?TYsBozd374oa z=>{9Q?X)LHm4o~uACL0BkQb%^?10|^&hzrz>EWyI7NF}AwEcC+Hm7hMhsZ6jq7}cm zL~n)<f{%j_g^z|W0AIt)Yp0LfZeWXm6aglN*_s1MjDDyjj9A-1M?{_?_BO=0*Cx1> za7&Nq;9>#=6!ZPKcU2n-C8mJ&_4Ps0HlZQL+Uqvpgqn?|AcT-xb#*Ms57b1~Gcw0d z8#n8I0JYbzU+?bj#^Hcb!})+NEiEOUgiw9q+Phhz;ouToO9CKHZSBCoK=i9hL91y_ z@JKR6curOBmS|YnL__0{QZ_s?5}MP1@$P@mX(148>2JD@bOOUc@U8kn+O=6n{&zkd z#_8=x(N$Ejfq}s^Pa90|7AW$7{RP{bf@;st)xi5L_;U@^eI2^hF+wTGPcc4<5yQ5+ z1aEiu*Yot+F$xf08-f}Uq==*R5F-&{SzcNfBJxzS)S3<>cQ)aco~eS1i4{;h=0^fK zg-yL^C{o7?3k&u2^(Ur5Te=KC;Ow1*(9m?{nsxF+42a&<*<paU)~#N@3BKFbty{CR zvvE6A)bKx=;ZDQX*B2sN2~(+9d#fNB6bc7dS>C}l-c(lyp54*WfqqOWYlFkR8I`Xb zW49tyr__@eO((Vd`M^*&8iCZbd}i<AfELUS0G3xK60~2w)P0yiY%&OLUp=`NGsq<6 zc~4W%=IDwpGy*AT4MnN(c2<%Tz@V2V0LUPo*TZMW82)pJqUxFzz1cBbXA~X!>46Xu z;+jW=hcC<`o=U+Dvjc7W`w^s|obh^x?<a6c5L`^Sfa2ZmY@dRTVWiTf;K^=q(AL&g zLqkLGz^{*ww1bmN`hw_BbU_snSP%qyIBXR7@>SYUd!)a=AJ5}NjWDYSk9Ya<Wnxq# z(LX8{4z9oGhuiOOXlPheRD^y*k&Vu~?uM|OGl<zK$Ahy@w072@QAcqb8hPWD-?7^b zi(|l%+Y<@eMGwN0!7;y|qWkmb<C?a^Ot!bub8dc`!t3fsNtY;mWa{fJh;fT~XgP+f zA|LM!wKaoGe?Yq+-qy`u&pBk<Xv{BKr=;PAPGLH(f!hJ0g5D4Wl@L-L@pO+MG)#PT zo>fzfh98x!yt)5GNDHBZi-{Fbd^5I{(~Sz;QQE4is*H_|`KSIZ`n&PT(MB`$8JVx! zvJK>olI0fE0$tv=-6ptgj@9|gWo2dfA0cYQ8qoRq`MY-Q!Z{pVw*RhJIH>bDACGlZ zmM{16@){XIXC!3fteO??o@9(pI|_^=PB5roMx&92jBm}IL)I~Jt13*6j>Bh`mq|hf z$NapT?#z!qv~Jy68Q$bXa7zWdAB{I;U@fiH!3if31XMIMd>yvcBWPD{f%ot*3gJgK z8Xp!>)&<&aA_f-|D4_VDUrTl)lIS8EH8|<Jy1Fi2yco{@Yc_7dEluOQu8}#^PSIY! z5mbSiT8kGh(tx5AKm&mrLJabTt-4Up1kc~!z~X>^aOm(bG0JVJ5pF<3&$_v}EnBt> zw{URP8Sja3@M7SE-{|eHS?XH&YQlvJ7r0z5`o%;h)CPxp(ktv8X(-AV`#9p1vS<B6 zD4r3S5X)@nwe*!kRk42E2X?l5AF0UTnE!t7Lj!;NmsEe(*n=HX(m8^1u_GJLDGKVt zUE+45wMmh+wl24IH=)6XOwhXfM{LvD^j(M)TuhXJ;$3ddZn^D4!)UQXHqICfhNY$D z(iN-l4+jbD7p~u8HW8_^s~SNEG>5~vbm<a4YZ<S(lO`4pf-lxF!^symGBV1^$w9x8 z$iy12s}9vN#<m_Yr?U!;D`X<M{2P~?_Fb;h9|@e8y7Rty2hMc1kNzk5;NbY8E4oi| zCHz~E>#D7H7>zDuLak?@lUDcutx%Y|d!kt-GX)JWWK&sf>%h5)R-yzK6C$AaYD{ZW z2Z|#`Hl@YH#Qc8xJVJPi@WWPT5>wI$R}fr{&;z=zuI|j4GqR<tRFExPv0(xkoC*%E zw7917ZwoQn2?+`4ml2s%*4$PSmU9-JbL8=mtdmS`J{n8NB(&DfzGu-DyB??LKTMX; z^{gJhqh@PY>D`Z){4WyD^>&!NYpK~EVqa_>P+Zg8iAEGMvBuz*24tK<s}&~h6sMnB z;XF8uLimx1x1^eGsHKZ@=mumE=K;m%BU+1DeP}o#6KfD<9G8@G=)`G+^RRNuKdhqT z5{8EfnW#;T1Y;U9Yg$=Z;RFt9ZMT#V4z4}puEOsL<;<g_QI0cYvRf0c5QJ~&%p-Nx zk<1f})?74xkjcX3+xs3xl<#_!yyKyG*W0E$Mh5lo#YysdLi;;DxOBdV*Ie~0s%Yv! zqX?N)R>#iqPdkoQF4?hQtX@i)1BfxvSVAVbfz~FquIE%Jhn#|U1^ekRPCgT*7epqV z4Gj%N#Ki9V{RBdL^0>v}V<C~zP#BuHBSDQcW4f}k^5Vsd<LYy(!j@Xx5CR7;2!Vr} z?`Npw8=9J$(r7ePbp#nE#^M!4=U;Q6p$zWgJu>eSgPVuO3^F;5$?JI>QD*QsMFa@b zzZ;`B_jA2XoAmD|ke&>_YqHS+;Dh>?-6#LA>;6S0jCM3~kjZS-tPJ0@qi7YA5({om zD0$J{&qLz}naqf=fZ2N`3L!!%)Pkt$2r_wWU|=vPEW-TI9|-eF>t+Xz`UZvc^%Gto zuST*l-O|$X`0-<?x~nQUxbmc{gm7@X6LY~LO{IM?2)+h=C6s$cKYYkAxAq=JO4-xb zak}U@B=mrkRQib5-GIgkG7OE^-TykV#_V~za8hP@QE#WI{(~fnkimDY-hUoi&Hqo} zry#U!;Gk+h$gr{6=Io%%b7)Z$>rl}pwcNS84`D%M*jNpx-zJ{B!v*0#;SN1>Cyray zg2MKZVPn0$ect{-CYDDK@{`1k_8oHf@$2gDCdboPBjuRx>FM$G^n}P+6;va+nJ6I~ z1lQWQTN%}pSFO5x_b#1IM?ZGR6l(p09r<<M-YFL7Xf)=Y$>zDW?!5!(fP_p@);BmD zoYQ#7A!{NgyXZnGEI6UzX13|hNj1Pv(Fj3?xv|@-qw{SXXj{=5H`zUs_U6?EquheX z6k)86LEF@}-EN2mesE(CwtadVy9<>ZL596`c6Ip%hM+)W(LuzXegW+rq)J(%M($x< zSXg-F%9SNcmMBS`w?um~8598rSD0Q<==a&be}6<o#K6D+`hi2HOoLJzaYc7PF-OhQ zkz?#uTH(#c)|@y0LBadTlwlBim0QQX6I>!X)|d)@(YqkNgf%pbrXDf{pPG6(spYTT z5={^geCu06vMxd;;x}*5)I+A=sn<8+SJ-hRpoQXC@JtwoR@pT${00pPWD2$6;o+pz z^fQ;QBNQm;Cof!qDlUUVB;@jHR2bIX-Q9kEetY-sRdiir%`N6q!NE0Fj*9piTefU@ z`t&K13L;aj^$&KI(1W1<26YZdff;`kH3akxpzLPIRAcQugMOI}zdL19jS233k?xdO z!{nh{OURUJBX5SQn$kmaE<42NP!kKfC7MDR;`Sak8VtykY~{`UkCS<Z?rcgiK{GG* z%ZxU9+aMYM$dqgqRn<@J9Uz33%1qF3@4-j5_9f*NB!19plp5BMk<i1#!`$3lQPoIH zFHZpn1BCv>Z@QC+;iqrfwCT~KN2R5u==Td5uB@w%T~zNMlzEJ-r=x(3-wWz|I(u2@ zStG-VRk1o86RHoq$|N@vG`CHE8daWG&mDP#)Dy^5yUev_RC#)*9;Qkz-8lWoyj#_* zOe7UVrdk^qdXrGoeKWR|+~LM1o@^kYd5k`^*dbG`_4f6}CMI9BzD+eOc<$<rn7D-Q zo*vRaVl`@xY!-_JNePc0J-T@DVhr)zV*Mjh!NK(xeC01QVrXdi?Af!Tq9XKzg$(z{ zr{-dMP(<!U5=0zB6T)-P1N7|ZZ9-2L8E&nD+5RfN`rxZ9GBLqDcIi;BIj5F8GJ=jt z$na&YosH=gZeA$|NSZW5HzMyw1tW0?Wfeq*KT4QX(+zbw3`nFUH1}pdNa8^OVzk&H z!<#iVH+u&JA3t}A+)VKI)8{>W0~kySo%PfxKC&SnUs_t)y?gg`b#*bIgNroR+)kAo z4ldcYSLPZcXV0Du4GraTx#;H!8R2YrWT3G%JH6tCZ>lA+jz-=o`%=qo>72}=;Xd>n zkrBumxjg|{^eeuFhEGz72n2UNN<QtL=blmrxbqE?P9Vd*wY1lPmM|dWlmm_M5y!3x z#xVu=fe8+vg%26-ZD{08KC{;$v+Y<gA@dUIyELB;<9OtCRC5r5M@9f+Hn9Ri!cSki zPO_AOP>hAnU{dgbs-ZE>Hy)1{A0Pkl;X^YsGYt(5*|6O*%li_-L7m6>QoZxGZQG!@ zN?>4MU0oe|Sjb3Yy#sCaEoqsR4#8Pgj<MTtSY0G2^LR$(%Q|*SPd{Su$Ovx(L&I6s ztq^p4)-&JmNea$^;0f0p+nCCPvgXcSRE-fCA+3L~yMmcSD}3mlXi6lJAZ;6v&CDt< zsLqTYA~J$nTkmjeWfuhG9u4H^;TQ<+bZtHv!hvLhDYZS_11MA<86l0+%B968-G6Fp ze(-l90ztC_NAEs<78M)c%x<Cbb3|}(isRG308~2m_xHbj`?iIJ#q#CLCDjgU>qrO( zS6SW>@jgH&ckSAB;lc%HXJ^nh0=PsE1sUmWcw~^>US7_K0Ciqy_Gy>+UCKWZImhn^ z$vT-@{=BR)thuceM2_gWAS0_mRo@auyKhFr&A^gF4p}>K00j5hXI%9wbWN^>sNvp$ zVf2iUk;$NNM_FTBT=6rX(d>j<@CF23;&wxE6##;btwlq_XzfErD(e{>$!Y9$&2GOE z-MZID#i7S~F3pGhTOd`UcYa4vQ|}-uI*g1&2H_o5HMRc1q1SC50CB@J5Hztoa_#m# z5D8XP)r^cFE<b7r4oYf}rZ6KTBOo9E68xP%e;(9@YuBz_xNxD6nBQ}Lli;BLgjHd? z6LaCx)nl&*@(3L8`0?Wq-3;|o5)u+>YHEgtkV*m>vj24Va~Rybaz<o!wR3!t4X6v< zlZ+i`TQDlKXcU}vBChC8R`qKjk_>KcS6>V2b7Yh?IG~g>+9UEAAlAF+QvjI?cRyC3 zv=CV2e)|j%Fg}VX56Y$&G;lk6Q3Xb1RI!1f-Ud!VZfyXF3jNalfOLK;s5P8q9-4hQ zrQE5E5f9lCQNTVj3T-b3Mv7RyAtjy9Qg|05TMzoR>~d8U3WPL>M*=xlV_ILNw?&qB zRj~U95Di2|4TEK+k;#lpOn%|$Y<2#MfrX`NY2fZXmXIgW_LUQe92*)LXw{o+2oB0S z;yXGz=yW>B43m?SAqk+budh#dqEB{<M`p828p|QM$tjiP8ebkz+!IpT6H(C{Th*K1 z&|g%~0Rq_C+KM0`GGx-vyOF-Z_VykIyS=QTC8LrVQ$i1h+93IL-nlhyK>8pG7(A=W z3H~#`&I|q*yr6;^4KD^UV0(9C-#|O6v4{+rScAfd?Djsub}@xbo~d>Au~qlOO0W4B zop#Fwr6CjsF@2T>Nem(WcLMc;fMt_sX;4kr@@1x#Ywl&=!n+}*FKCr+DYapF3{V9! zxILg-L_LfQ8Gl{lbupp-Z+>l1Qt8X^9BbdSqafwODn{5n(IP1GEUoZCMx{q-BaOi= zg?MMw?Z}V`p6Os+Dq;1-Rdojxb-HA?12?=K#|3nFJgDVxKnnoGy}oQH5@O=PHuPXa z1>rs3?0r6L&;UZ%vQ==)<)~Jhc&=?)n_F&sa7kxkO*e4B_CB=IAw%}ghKEOh6{clo zLopKwKt6H)GHCY{C=A?x7*HSt3w!wore<W(8yUmHC?rLUAvlN(88T$ZkRg-Hpq3(J zGwd6LjUT#(MTQK+nKj(q$K!O=(>Vo|%;duQa1a}2Rr>(rORaEDDRW3Dc>ypH)PGPV z86tvnYWx5%0{*LQ&SLQ@+PYaFQAFK{3>h+H$na_1Jw5gG#+>|uNLrk?e~^QV`y<<z zH|{<>cl8Do3qEXhX0jhTetPU*j-EPq_KLOj-3O0s?O(aL0|bnSiOtT-ucy<yy3qxg bEcX8a$#FMIPQS>S00000NkvXXu0mjf9wX?m diff --git a/core/client/testem.js b/core/client/testem.js deleted file mode 100644 index bea2f315f6..0000000000 --- a/core/client/testem.js +++ /dev/null @@ -1,14 +0,0 @@ -/*jshint node:true*/ -module.exports = { - 'framework': 'mocha', - 'test_page': 'tests/index.html?hidepassed', - 'disable_watching': true, - 'launch_in_ci': [ - 'Chrome', - 'Firefox' - ], - 'launch_in_dev': [ - 'Chrome', - 'Firefox' - ] -}; diff --git a/core/client/tests/.jscsrc b/core/client/tests/.jscsrc deleted file mode 100644 index 411c02d939..0000000000 --- a/core/client/tests/.jscsrc +++ /dev/null @@ -1,4 +0,0 @@ -{ - "preset": "../.jscsrc", - "requireBlocksOnNewline": null -} diff --git a/core/client/tests/.jshintrc b/core/client/tests/.jshintrc deleted file mode 100644 index fbdf3a678f..0000000000 --- a/core/client/tests/.jshintrc +++ /dev/null @@ -1,58 +0,0 @@ -{ - "predef": [ - "server", - "authenticateSession", - "invalidateSession", - "currentSession", - "document", - "window", - "location", - "setTimeout", - "$", - "-Promise", - "define", - "console", - "visit", - "exists", - "fillIn", - "click", - "keyEvent", - "triggerEvent", - "find", - "findWithAssert", - "wait", - "DS", - "andThen", - "currentURL", - "currentPath", - "currentRouteName", - "expect" - ], - "mocha": true, - "node": false, - "browser": false, - "boss": true, - "curly": false, - "debug": false, - "devel": false, - "eqeqeq": true, - "evil": true, - "expr": true, - "forin": false, - "immed": false, - "laxbreak": false, - "newcap": true, - "noarg": true, - "noempty": false, - "nonew": false, - "nomen": false, - "onevar": false, - "plusplus": false, - "regexp": false, - "undef": true, - "sub": true, - "strict": false, - "white": false, - "eqnull": true, - "esnext": true -} diff --git a/core/client/tests/acceptance/authentication-test.js b/core/client/tests/acceptance/authentication-test.js deleted file mode 100644 index f6b7ba87a6..0000000000 --- a/core/client/tests/acceptance/authentication-test.js +++ /dev/null @@ -1,195 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; -import { authenticateSession, currentSession, invalidateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import Mirage from 'ember-cli-mirage'; -import windowProxy from 'ghost/utils/window-proxy'; -import ghostPaths from 'ghost/utils/ghost-paths'; - -const Ghost = ghostPaths(); - -describe('Acceptance: Authentication', function () { - let application, - originalReplaceLocation; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - describe('general page', function () { - beforeEach(function () { - originalReplaceLocation = windowProxy.replaceLocation; - windowProxy.replaceLocation = function (url) { - visit(url); - }; - - server.loadFixtures(); - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - }); - - afterEach(function () { - windowProxy.replaceLocation = originalReplaceLocation; - }); - - it('invalidates session on 401 API response', function () { - // return a 401 when attempting to retrieve users - server.get('/users/', (db, request) => { - return new Mirage.Response(401, {}, { - errors: [ - {message: 'Access denied.', errorType: 'UnauthorizedError'} - ] - }); - }); - - authenticateSession(application); - visit('/team'); - - andThen(() => { - expect(currentURL(), 'url after 401').to.equal('/signin'); - }); - }); - - it('doesn\'t show navigation menu on invalid url when not authenticated', function () { - invalidateSession(application); - - visit('/'); - - andThen(() => { - expect(currentURL(), 'current url').to.equal('/signin'); - expect(find('nav.gh-nav').length, 'nav menu presence').to.equal(0); - }); - - visit('/signin/invalidurl/'); - - andThen(() => { - expect(currentURL(), 'url after invalid url').to.equal('/signin/invalidurl/'); - expect(currentPath(), 'path after invalid url').to.equal('error404'); - expect(find('nav.gh-nav').length, 'nav menu presence').to.equal(0); - }); - }); - - it('shows nav menu on invalid url when authenticated', function () { - authenticateSession(application); - visit('/signin/invalidurl/'); - - andThen(() => { - expect(currentURL(), 'url after invalid url').to.equal('/signin/invalidurl/'); - expect(currentPath(), 'path after invalid url').to.equal('error404'); - expect(find('nav.gh-nav').length, 'nav menu presence').to.equal(1); - }); - }); - }); - - describe('editor', function () { - let origDebounce = Ember.run.debounce; - let origThrottle = Ember.run.throttle; - - // we don't want the autosave interfering in this test - beforeEach(function () { - Ember.run.debounce = function () { }; - Ember.run.throttle = function () { }; - }); - - it('displays re-auth modal attempting to save with invalid session', function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - // simulate an invalid session when saving the edited post - server.put('/posts/:id/', (db, request) => { - let post = db.posts.find(request.params.id); - let [attrs] = JSON.parse(request.requestBody).posts; - - if (attrs.markdown === 'Edited post body') { - return new Mirage.Response(401, {}, { - errors: [ - {message: 'Access denied.', errorType: 'UnauthorizedError'} - ] - }); - } else { - return { - posts: [post] - }; - } - }); - - server.loadFixtures(); - authenticateSession(application); - - visit('/editor'); - - // create the post - fillIn('#entry-title', 'Test Post'); - fillIn('textarea.markdown-editor', 'Test post body'); - click('.js-publish-button'); - - andThen(() => { - // we shouldn't have a modal at this point - expect(find('.modal-container #login').length, 'modal exists').to.equal(0); - // we also shouldn't have any alerts - expect(find('.gh-alert').length, 'no of alerts').to.equal(0); - }); - - // update the post - fillIn('textarea.markdown-editor', 'Edited post body'); - click('.js-publish-button'); - - andThen(() => { - // we should see a re-auth modal - expect(find('.fullscreen-modal #login').length, 'modal exists').to.equal(1); - }); - }); - - // don't clobber debounce/throttle for future tests - afterEach(function () { - Ember.run.debounce = origDebounce; - Ember.run.throttle = origThrottle; - }); - }); - - it('adds auth headers to jquery ajax', function (done) { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.post('/uploads', (db, request) => { - return request; - }); - server.loadFixtures(); - - // jscs:disable requireCamelCaseOrUpperCaseIdentifiers - authenticateSession(application, { - access_token: 'test_token', - expires_in: 3600, - token_type: 'Bearer' - }); - // jscs:enable requireCamelCaseOrUpperCaseIdentifiers - - // necessary to visit a page to fully boot the app in testing - visit('/').andThen(() => { - $.ajax({ - type: 'POST', - url: `${Ghost.apiRoot}/uploads/`, - data: {test: 'Test'} - }).then((request) => { - expect(request.requestHeaders.Authorization, 'Authorization header') - .to.exist; - expect(request.requestHeaders.Authorization, 'Authotization header content') - .to.equal('Bearer test_token'); - }).always(() => { - done(); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/password-reset-test.js b/core/client/tests/acceptance/password-reset-test.js deleted file mode 100644 index e6ff9bb822..0000000000 --- a/core/client/tests/acceptance/password-reset-test.js +++ /dev/null @@ -1,107 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; - -describe('Acceptance: Password Reset', function() { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - describe('request reset', function () { - it('is successful with valid data', function () { - visit('/signin'); - fillIn('input[name="identification"]', 'test@example.com'); - click('.forgotten-link'); - - andThen(function () { - // an alert with instructions is displayed - expect(find('.gh-alert-blue').length, 'alert count') - .to.equal(1); - }); - }); - - it('shows error messages with invalid data', function () { - visit('/signin'); - - // no email provided - click('.forgotten-link'); - - andThen(function () { - // email field is invalid - expect( - find('input[name="identification"]').closest('.form-group').hasClass('error'), - 'email field has error class (no email)' - ).to.be.true; - - // password field is valid - expect( - find('input[name="password"]').closest('.form-group').hasClass('error'), - 'password field has error class (no email)' - ).to.be.false; - - // error message shown - expect(find('p.main-error').text().trim(), 'error message') - .to.equal('We need your email address to reset your password!'); - }); - - // invalid email provided - fillIn('input[name="identification"]', 'test'); - click('.forgotten-link'); - - andThen(function () { - // email field is invalid - expect( - find('input[name="identification"]').closest('.form-group').hasClass('error'), - 'email field has error class (invalid email)' - ).to.be.true; - - // password field is valid - expect( - find('input[name="password"]').closest('.form-group').hasClass('error'), - 'password field has error class (invalid email)' - ).to.be.false; - - // error message - expect(find('p.main-error').text().trim(), 'error message') - .to.equal('We need your email address to reset your password!'); - }); - - // unknown email provided - fillIn('input[name="identification"]', 'unknown@example.com'); - click('.forgotten-link'); - - andThen(function () { - // email field is invalid - expect( - find('input[name="identification"]').closest('.form-group').hasClass('error'), - 'email field has error class (unknown email)' - ).to.be.true; - - // password field is valid - expect( - find('input[name="password"]').closest('.form-group').hasClass('error'), - 'password field has error class (unknown email)' - ).to.be.false; - - // error message - expect(find('p.main-error').text().trim(), 'error message') - .to.equal('There is no user with that email address.'); - }); - }); - }); - - // TODO: add tests for the change password screen -}); diff --git a/core/client/tests/acceptance/posts/post-test.js b/core/client/tests/acceptance/posts/post-test.js deleted file mode 100644 index cdbbae526c..0000000000 --- a/core/client/tests/acceptance/posts/post-test.js +++ /dev/null @@ -1,71 +0,0 @@ -/* jshint expr:true */ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import { errorOverride, errorReset } from 'ghost/tests/helpers/adapter-error'; -import Mirage from 'ember-cli-mirage'; - -describe('Acceptance: Posts - Post', function() { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - // load the settings fixtures - // TODO: this should always be run for acceptance tests - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('can visit post route', function () { - let posts = server.createList('post', 3); - - visit('/'); - - andThen(() => { - expect(find('.posts-list li').length, 'post list count').to.equal(3); - - // if we're in "desktop" size, we should redirect and highlight - if (find('.content-preview:visible').length) { - expect(currentURL(), 'currentURL').to.equal(`/${posts[0].id}`); - expect(find('.posts-list li').first().hasClass('active'), 'highlights latest post').to.be.true; - } - }); - }); - - it('redirects to 404 when post does not exist', function () { - server.get('/posts/200/', function () { - return new Mirage.Response(404, {'Content-Type': 'application/json'}, {errors: [{message: 'Post not found.', errorType: 'NotFoundError'}]}); - }); - - errorOverride(); - - visit('/200'); - - andThen(() => { - errorReset(); - expect(currentPath()).to.equal('error404'); - expect(currentURL()).to.equal('/200'); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/settings/apps-test.js b/core/client/tests/acceptance/settings/apps-test.js deleted file mode 100644 index ed38bbc15b..0000000000 --- a/core/client/tests/acceptance/settings/apps-test.js +++ /dev/null @@ -1,89 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -const {run} = Ember; - -describe('Acceptance: Settings - Apps', function () { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/apps'); - - andThen(function() { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/apps'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects to team page when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/apps'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it redirects to Slack when clicking on the grid', function () { - visit('/settings/apps'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/apps'); - - }); - - click('#slack-link'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/apps/slack'); - }); - - }); - - }); -}); diff --git a/core/client/tests/acceptance/settings/code-injection-test.js b/core/client/tests/acceptance/settings/code-injection-test.js deleted file mode 100644 index 8caaf78c4b..0000000000 --- a/core/client/tests/acceptance/settings/code-injection-test.js +++ /dev/null @@ -1,91 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -describe('Acceptance: Settings - Code-Injection', function() { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/code-injection'); - - andThen(function() { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/code-injection'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects to team page when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/code-injection'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it renders, loads editors correctly', function () { - visit('/settings/code-injection'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/code-injection'); - - // has correct page title - expect(document.title, 'page title').to.equal('Settings - Code Injection - Test Blog'); - - // highlights nav menu - expect($('.gh-nav-settings-code-injection').hasClass('active'), 'highlights nav menu item') - .to.be.true; - - expect(find('.view-header .view-actions .btn-blue').text().trim(), 'save button text').to.equal('Save'); - - expect(find('#ghost-head .CodeMirror').length, 'ghost head codemirror element').to.equal(1); - expect($('#ghost-head .CodeMirror').hasClass('cm-s-xq-light'), 'ghost head editor theme').to.be.true; - - expect(find('#ghost-foot .CodeMirror').length, 'ghost head codemirror element').to.equal(1); - expect($('#ghost-foot .CodeMirror').hasClass('cm-s-xq-light'), 'ghost head editor theme').to.be.true; - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/settings/general-test.js b/core/client/tests/acceptance/settings/general-test.js deleted file mode 100644 index c4f10f41db..0000000000 --- a/core/client/tests/acceptance/settings/general-test.js +++ /dev/null @@ -1,325 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -const {run} = Ember; - -describe('Acceptance: Settings - General', function () { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/general'); - - andThen(function() { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/general'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects to team page when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/general'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it renders, shows image uploader modals', function () { - visit('/settings/general'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/general'); - - // has correct page title - expect(document.title, 'page title').to.equal('Settings - General - Test Blog'); - - // highlights nav menu - expect($('.gh-nav-settings-general').hasClass('active'), 'highlights nav menu item') - .to.be.true; - - expect(find('.view-header .view-actions .btn-blue').text().trim(), 'save button text').to.equal('Save'); - - // initial postsPerPage should be 5 - expect(find('input#postsPerPage').val(), 'post per page value').to.equal('5'); - - expect(find('input#permalinks').prop('checked'), 'date permalinks checkbox').to.be.false; - }); - - fillIn('#settings-general input[name="general[title]"]', 'New Blog Title'); - click('.view-header .btn.btn-blue'); - - andThen(() => { - expect(document.title, 'page title').to.equal('Settings - General - New Blog Title'); - }); - - click('.blog-logo'); - - andThen(() => { - expect(find('.fullscreen-modal .modal-content .gh-image-uploader').length, 'modal selector').to.equal(1); - }); - - click('.fullscreen-modal .modal-content .gh-image-uploader .image-cancel'); - - andThen(() => { - expect(find('.fullscreen-modal .modal-content .gh-image-uploader .description').text()).to.equal('Upload an image'); - }); - - // click cancel button - click('.fullscreen-modal .modal-footer .btn.btn-minor'); - - andThen(() => { - expect(find('.fullscreen-modal').length).to.equal(0); - }); - - click('.blog-cover'); - - andThen(() => { - expect(find('.fullscreen-modal .modal-content .gh-image-uploader').length, 'modal selector').to.equal(1); - }); - - click('.fullscreen-modal .modal-footer .js-button-accept'); - - andThen(() => { - expect(find('.fullscreen-modal').length).to.equal(0); - }); - - // renders theme selector correctly - andThen(() => { - expect(find('#activeTheme select option').length, 'available themes').to.equal(1); - expect(find('#activeTheme select option').text().trim()).to.equal('Blog - 1.0'); - }); - - // handles private blog settings correctly - andThen(() => { - expect(find('input#isPrivate').prop('checked'), 'isPrivate checkbox').to.be.false; - }); - - click('input#isPrivate'); - - andThen(() => { - expect(find('input#isPrivate').prop('checked'), 'isPrivate checkbox').to.be.true; - expect(find('#settings-general input[name="general[password]"]').length, 'password input').to.equal(1); - expect(find('#settings-general input[name="general[password]"]').val(), 'password default value').to.not.equal(''); - }); - - fillIn('#settings-general input[name="general[password]"]', ''); - triggerEvent('#settings-general input[name="general[password]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal('Password must be supplied'); - }); - - fillIn('#settings-general input[name="general[password]"]', 'asdfg'); - triggerEvent('#settings-general input[name="general[password]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - // validates a facebook url correctly - - andThen(() => { - // loads fixtures and performs transform - expect(find('input[name="general[facebook]"]').val(), 'initial facebook value') - .to.equal('https://www.facebook.com/test'); - }); - - triggerEvent('#settings-general input[name="general[facebook]"]', 'focus'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - // regression test: we still have a value after the input is - // focused and then blurred without any changes - expect(find('input[name="general[facebook]"]').val(), 'facebook value after blur with no change') - .to.equal('https://www.facebook.com/test'); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'facebook.com/username'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/username'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'facebook.com/pages/some-facebook-page/857469375913?ref=ts'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/pages/some-facebook-page/857469375913?ref=ts'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', '*(&*(%%))'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal('The URL must be in a format like https://www.facebook.com/yourPage'); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'http://github.com/username'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/username'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'http://github.com/pages/username'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/pages/username'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'testuser'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/testuser'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'ab99'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal('Your Page name is not a valid Facebook Page name'); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'page/ab99'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/page/ab99'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[facebook]"]', 'page/*(&*(%%))'); - triggerEvent('#settings-general input[name="general[facebook]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[facebook]"]').val()).to.be.equal('https://www.facebook.com/page/*(&*(%%))'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - // validates a twitter url correctly - - andThen(() => { - // loads fixtures and performs transform - expect(find('input[name="general[twitter]"]').val(), 'initial twitter value') - .to.equal('https://twitter.com/test'); - }); - - triggerEvent('#settings-general input[name="general[twitter]"]', 'focus'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - // regression test: we still have a value after the input is - // focused and then blurred without any changes - expect(find('input[name="general[twitter]"]').val(), 'twitter value after blur with no change') - .to.equal('https://twitter.com/test'); - }); - - fillIn('#settings-general input[name="general[twitter]"]', 'twitter.com/username'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[twitter]"]').val()).to.be.equal('https://twitter.com/username'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[twitter]"]', '*(&*(%%))'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal('The URL must be in a format like https://twitter.com/yourUsername'); - }); - - fillIn('#settings-general input[name="general[twitter]"]', 'http://github.com/username'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[twitter]"]').val()).to.be.equal('https://twitter.com/username'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - fillIn('#settings-general input[name="general[twitter]"]', 'thisusernamehasmorethan15characters'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal('Your Username is not a valid Twitter Username'); - }); - - fillIn('#settings-general input[name="general[twitter]"]', 'testuser'); - triggerEvent('#settings-general input[name="general[twitter]"]', 'blur'); - - andThen(() => { - expect(find('#settings-general input[name="general[twitter]"]').val()).to.be.equal('https://twitter.com/testuser'); - expect(find('#settings-general .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - }); - - }); -}); diff --git a/core/client/tests/acceptance/settings/labs-test.js b/core/client/tests/acceptance/settings/labs-test.js deleted file mode 100644 index 0b3a5cb591..0000000000 --- a/core/client/tests/acceptance/settings/labs-test.js +++ /dev/null @@ -1,95 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -describe('Acceptance: Settings - Labs', function() { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/labs'); - - andThen(function() { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/labs'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects to team page when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/labs'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it renders, loads modals correctly', function () { - visit('/settings/labs'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/labs'); - - // has correct page title - expect(document.title, 'page title').to.equal('Settings - Labs - Test Blog'); - - // highlights nav menu - expect($('.gh-nav-settings-labs').hasClass('active'), 'highlights nav menu item') - .to.be.true; - }); - - click('#settings-resetdb .js-delete'); - - andThen(() => { - expect(find('.fullscreen-modal .modal-content').length, 'modal element').to.equal(1); - }); - - click('.fullscreen-modal .modal-footer .btn.btn-minor'); - - andThen(() => { - expect(find('.fullscreen-modal').length, 'modal element').to.equal(0); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/settings/navigation-test.js b/core/client/tests/acceptance/settings/navigation-test.js deleted file mode 100644 index 6f47ca8127..0000000000 --- a/core/client/tests/acceptance/settings/navigation-test.js +++ /dev/null @@ -1,224 +0,0 @@ -/* jshint expr:true */ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -describe('Acceptance: Settings - Navigation', function () { - let application; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/navigation'); - - andThen(function () { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/navigation'); - - andThen(function () { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - // load the settings fixtures - // TODO: this should always be run for acceptance tests - server.loadFixtures(); - - authenticateSession(application); - }); - - it('can visit /settings/navigation', function () { - visit('/settings/navigation'); - - andThen(function () { - expect(currentPath()).to.equal('settings.navigation'); - - // fixtures contain two nav items, check for three rows as we - // should have one extra that's blank - expect( - find('.gh-blognav-item').length, - 'navigation items count' - ).to.equal(3); - }); - }); - - it('saves navigation settings', function () { - visit('/settings/navigation'); - fillIn('.gh-blognav-label:first input', 'Test'); - fillIn('.gh-blognav-url:first input', '/test'); - triggerEvent('.gh-blognav-url:first input', 'blur'); - - click('.btn-blue'); - - andThen(function () { - let [navSetting] = server.db.settings.where({key: 'navigation'}); - - expect(navSetting.value).to.equal('[{"label":"Test","url":"/test/"},{"label":"About","url":"/about"}]'); - - // don't test against .error directly as it will pick up failed - // tests "pre.error" elements - expect(find('span.error').length, 'error fields count').to.equal(0); - expect(find('.gh-alert').length, 'alerts count').to.equal(0); - expect(find('.response:visible').length, 'validation errors count') - .to.equal(0); - }); - }); - - it('validates new item correctly on save', function () { - visit('/settings/navigation'); - - click('.btn-blue'); - - andThen(function () { - expect( - find('.gh-blognav-item').length, - 'number of nav items after saving with blank new item' - ).to.equal(3); - }); - - fillIn('.gh-blognav-label:last input', 'Test'); - fillIn('.gh-blognav-url:last input', 'http://invalid domain/'); - triggerEvent('.gh-blognav-url:last input', 'blur'); - - click('.btn-blue'); - - andThen(function () { - expect( - find('.gh-blognav-item').length, - 'number of nav items after saving with invalid new item' - ).to.equal(3); - - expect( - find('.gh-blognav-item:last .error').length, - 'number of invalid fields in new item' - ).to.equal(1); - }); - }); - - it('clears unsaved settings when navigating away', function () { - visit('/settings/navigation'); - fillIn('.gh-blognav-label:first input', 'Test'); - triggerEvent('.gh-blognav-label:first input', 'blur'); - - andThen(function () { - expect(find('.gh-blognav-label:first input').val()).to.equal('Test'); - }); - - visit('/settings/code-injection'); - visit('/settings/navigation'); - - andThen(function () { - expect(find('.gh-blognav-label:first input').val()).to.equal('Home'); - }); - }); - - it('can add and remove items', function (done) { - visit('/settings/navigation'); - - click('.gh-blognav-add'); - - andThen(function () { - expect( - find('.gh-blognav-label:last .response').is(':visible'), - 'blank label has validation error' - ).to.be.true; - }); - - fillIn('.gh-blognav-label:last input', 'New'); - triggerEvent('.gh-blognav-label:last input', 'keypress', {}); - - andThen(function () { - expect( - find('.gh-blognav-label:last .response').is(':visible'), - 'label validation is visible after typing' - ).to.be.false; - }); - - fillIn('.gh-blognav-url:last input', '/new'); - triggerEvent('.gh-blognav-url:last input', 'keypress', {}); - triggerEvent('.gh-blognav-url:last input', 'blur'); - - andThen(function () { - expect( - find('.gh-blognav-url:last .response').is(':visible'), - 'url validation is visible after typing' - ).to.be.false; - - expect( - find('.gh-blognav-url:last input').val() - ).to.equal(`${window.location.protocol}//${window.location.host}/new/`); - }); - - click('.gh-blognav-add'); - - andThen(function () { - expect( - find('.gh-blognav-item').length, - 'number of nav items after successful add' - ).to.equal(4); - - expect( - find('.gh-blognav-label:last input').val(), - 'new item label value after successful add' - ).to.be.blank; - - expect( - find('.gh-blognav-url:last input').val(), - 'new item url value after successful add' - ).to.equal(`${window.location.protocol}//${window.location.host}/`); - - expect( - find('.gh-blognav-item .response:visible').length, - 'number or validation errors shown after successful add' - ).to.equal(0); - }); - - click('.gh-blognav-item:first .gh-blognav-delete'); - - andThen(function () { - expect( - find('.gh-blognav-item').length, - 'number of nav items after successful remove' - ).to.equal(3); - }); - - click('.btn-blue'); - - andThen(function () { - let [navSetting] = server.db.settings.where({key: 'navigation'}); - - expect(navSetting.value).to.equal('[{"label":"About","url":"/about"},{"label":"New","url":"/new/"}]'); - - done(); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/settings/slack-test.js b/core/client/tests/acceptance/settings/slack-test.js deleted file mode 100644 index 4f42a88ab4..0000000000 --- a/core/client/tests/acceptance/settings/slack-test.js +++ /dev/null @@ -1,121 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import Mirage from 'ember-cli-mirage'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -const {run} = Ember; - -describe('Acceptance: Settings - Apps - Slack', function () { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/apps/slack'); - - andThen(function() { - expect(currentURL(), 'currentURL').to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/apps/slack'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects to team page when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/apps/slack'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it validates and saves a slack url properly', function () { - visit('/settings/apps/slack'); - - andThen(() => { - // has correct url - expect(currentURL(), 'currentURL').to.equal('/settings/apps/slack'); - - }); - - fillIn('#slack-settings input[name="slack[url]"]', 'notacorrecturl'); - click('#saveSlackIntegration'); - - andThen(() => { - expect(find('#slack-settings .error .response').text().trim(), 'inline validation response') - .to.equal('The URL must be in a format like https://hooks.slack.com/services/<your personal key>'); - }); - - fillIn('#slack-settings input[name="slack[url]"]', 'https://hooks.slack.com/services/1275958430'); - click('#sendTestNotification'); - - andThen(() => { - expect(find('.gh-alert-blue').length, 'modal element').to.equal(1); - expect(find('#slack-settings .error .response').text().trim(), 'inline validation response') - .to.equal(''); - }); - - andThen(() => { - server.put('/settings/', function (db, request) { - return new Mirage.Response(402, {}, { - errors: [ - { - errorType: 'ValidationError', - message: 'Test error' - } - ] - }); - }); - }); - - click('.gh-alert-blue .gh-alert-close'); - click('#sendTestNotification'); - - andThen(() => { - let [lastRequest] = server.pretender.handledRequests.slice(-1); - expect(lastRequest.url).to.not.match(/\/slack\/test/); - expect(find('.gh-alert-blue').length, 'check slack alert after api validation error').to.equal(0); - }); - }); - - }); -}); diff --git a/core/client/tests/acceptance/settings/tags-test.js b/core/client/tests/acceptance/settings/tags-test.js deleted file mode 100644 index 35334c4b3a..0000000000 --- a/core/client/tests/acceptance/settings/tags-test.js +++ /dev/null @@ -1,305 +0,0 @@ -/* jshint expr:true */ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../../helpers/start-app'; -import destroyApp from '../../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import { errorOverride, errorReset } from 'ghost/tests/helpers/adapter-error'; -import Mirage from 'ember-cli-mirage'; - -const {run} = Ember; - -// Grabbed from keymaster's testing code because Ember's `keyEvent` helper -// is for some reason not triggering the events in a way that keymaster detects: -// https://github.com/madrobby/keymaster/blob/master/test/keymaster.html#L31 -const modifierMap = { - 16: 'shiftKey', - 18: 'altKey', - 17: 'ctrlKey', - 91: 'metaKey' -}; -let keydown = function (code, modifiers, el) { - let event = document.createEvent('Event'); - event.initEvent('keydown', true, true); - event.keyCode = code; - if (modifiers && modifiers.length > 0) { - for (let i in modifiers) { - event[modifierMap[modifiers[i]]] = true; - } - } - (el || document).dispatchEvent(event); -}; -let keyup = function (code, el) { - let event = document.createEvent('Event'); - event.initEvent('keyup', true, true); - event.keyCode = code; - (el || document).dispatchEvent(event); -}; - -describe('Acceptance: Settings - Tags', function () { - let application; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/settings/tags'); - - andThen(() => { - expect(currentURL()).to.equal('/signin'); - }); - }); - - it('redirects to team page when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - visit('/settings/navigation'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - // load the settings fixtures - // TODO: this should always be run for acceptance tests - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it renders, can be navigated, can edit, create & delete tags', function () { - let tag1 = server.create('tag'); - let tag2 = server.create('tag'); - - visit('/settings/tags'); - - andThen(() => { - // it redirects to first tag - expect(currentURL(), 'currentURL').to.equal(`/settings/tags/${tag1.slug}`); - - // it has correct page title - expect(document.title, 'page title').to.equal('Settings - Tags - Test Blog'); - - // it highlights nav menu - expect($('.gh-nav-settings-tags').hasClass('active'), 'highlights nav menu item') - .to.be.true; - - // it lists all tags - expect(find('.settings-tags .settings-tag').length, 'tag list count') - .to.equal(2); - expect(find('.settings-tags .settings-tag:first .tag-title').text(), 'tag list item title') - .to.equal(tag1.name); - - // it highlights selected tag - expect(find(`a[href="/settings/tags/${tag1.slug}"]`).hasClass('active'), 'highlights selected tag') - .to.be.true; - - // it shows selected tag form - expect(find('.tag-settings-pane h4').text(), 'settings pane title') - .to.equal('Tag Settings'); - expect(find('.tag-settings-pane input[name="name"]').val(), 'loads correct tag into form') - .to.equal(tag1.name); - }); - - // click the second tag in the list - click('.tag-edit-button:last'); - - andThen(() => { - // it navigates to selected tag - expect(currentURL(), 'url after clicking tag').to.equal(`/settings/tags/${tag2.slug}`); - - // it highlights selected tag - expect(find(`a[href="/settings/tags/${tag2.slug}"]`).hasClass('active'), 'highlights selected tag') - .to.be.true; - - // it shows selected tag form - expect(find('.tag-settings-pane input[name="name"]').val(), 'loads correct tag into form') - .to.equal(tag2.name); - }); - - andThen(() => { - // simulate up arrow press - run(() => { - keydown(38); - keyup(38); - }); - }); - - andThen(() => { - // it navigates to previous tag - expect(currentURL(), 'url after keyboard up arrow').to.equal(`/settings/tags/${tag1.slug}`); - - // it highlights selected tag - expect(find(`a[href="/settings/tags/${tag1.slug}"]`).hasClass('active'), 'selects previous tag') - .to.be.true; - }); - - andThen(() => { - // simulate down arrow press - run(() => { - keydown(40); - keyup(40); - }); - }); - - andThen(() => { - // it navigates to previous tag - expect(currentURL(), 'url after keyboard down arrow').to.equal(`/settings/tags/${tag2.slug}`); - - // it highlights selected tag - expect(find(`a[href="/settings/tags/${tag2.slug}"]`).hasClass('active'), 'selects next tag') - .to.be.true; - }); - - // trigger save - fillIn('.tag-settings-pane input[name="name"]', 'New Name'); - triggerEvent('.tag-settings-pane input[name="name"]', 'blur'); - - andThen(() => { - // check we update with the data returned from the server - expect(find('.settings-tags .settings-tag:last .tag-title').text(), 'tag list updates on save') - .to.equal('New Name'); - expect(find('.tag-settings-pane input[name="name"]').val(), 'settings form updates on save') - .to.equal('New Name'); - }); - - // start new tag - click('.view-actions .btn-green'); - - andThen(() => { - // it navigates to the new tag route - expect(currentURL(), 'new tag URL').to.equal('/settings/tags/new'); - - // it displays the new tag form - expect(find('.tag-settings-pane h4').text(), 'settings pane title') - .to.equal('New Tag'); - - // all fields start blank - find('.tag-settings-pane input, .tag-settings-pane textarea').each(function () { - expect($(this).val(), `input field for ${$(this).attr('name')}`) - .to.be.blank; - }); - }); - - // save new tag - fillIn('.tag-settings-pane input[name="name"]', 'New Tag'); - triggerEvent('.tag-settings-pane input[name="name"]', 'blur'); - - andThen(() => { - // it redirects to the new tag's URL - expect(currentURL(), 'URL after tag creation').to.equal('/settings/tags/new-tag'); - - // it adds the tag to the list and selects - expect(find('.settings-tags .settings-tag').length, 'tag list count after creation') - .to.equal(3); - expect(find('.settings-tags .settings-tag:last .tag-title').text(), 'new tag list item title') - .to.equal('New Tag'); - expect(find('a[href="/settings/tags/new-tag"]').hasClass('active'), 'highlights new tag') - .to.be.true; - }); - - // delete tag - click('.tag-delete-button'); - click('.fullscreen-modal .btn-red'); - - andThen(() => { - // it redirects to the first tag - expect(currentURL(), 'URL after tag deletion').to.equal(`/settings/tags/${tag1.slug}`); - - // it removes the tag from the list - expect(find('.settings-tags .settings-tag').length, 'tag list count after deletion') - .to.equal(2); - }); - }); - - it('loads tag via slug when accessed directly', function () { - server.createList('tag', 2); - - visit('/settings/tags/tag-1'); - - andThen(() => { - expect(currentURL(), 'URL after direct load').to.equal('/settings/tags/tag-1'); - - // it loads all other tags - expect(find('.settings-tags .settings-tag').length, 'tag list count after direct load') - .to.equal(2); - - // selects tag in list - expect(find('a[href="/settings/tags/tag-1"]').hasClass('active'), 'highlights requested tag') - .to.be.true; - - // shows requested tag in settings pane - expect(find('.tag-settings-pane input[name="name"]').val(), 'loads correct tag into form') - .to.equal('Tag 1'); - }); - }); - - it('has infinite scroll pagination of tags list', function () { - server.createList('tag', 32); - - visit('settings/tags/tag-0'); - - andThen(() => { - // it loads first page - expect(find('.settings-tags .settings-tag').length, 'tag list count on first load') - .to.equal(15); - - find('.tag-list').scrollTop(find('.tag-list-content').height()); - }); - - triggerEvent('.tag-list', 'scroll'); - - andThen(() => { - // it loads the second page - expect(find('.settings-tags .settings-tag').length, 'tag list count on second load') - .to.equal(30); - - find('.tag-list').scrollTop(find('.tag-list-content').height()); - }); - - triggerEvent('.tag-list', 'scroll'); - - andThen(() => { - // it loads the final page - expect(find('.settings-tags .settings-tag').length, 'tag list count on third load') - .to.equal(32); - }); - }); - - it('redirects to 404 when tag does not exist', function () { - server.get('/tags/slug/unknown/', function () { - return new Mirage.Response(404, {'Content-Type': 'application/json'}, {errors: [{message: 'Tag not found.', errorType: 'NotFoundError'}]}); - }); - - errorOverride(); - - visit('settings/tags/unknown'); - - andThen(() => { - errorReset(); - expect(currentPath()).to.equal('error404'); - expect(currentURL()).to.equal('/settings/tags/unknown'); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/setup-test.js b/core/client/tests/acceptance/setup-test.js deleted file mode 100644 index cc42e0bcfb..0000000000 --- a/core/client/tests/acceptance/setup-test.js +++ /dev/null @@ -1,402 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from 'ghost/tests/helpers/start-app'; -import destroyApp from 'ghost/tests/helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import Mirage from 'ember-cli-mirage'; - -describe('Acceptance: Setup', function () { - let application; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - it('redirects if already authenticated', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - - visit('/setup/one'); - andThen(() => { - expect(currentURL()).to.equal('/'); - }); - - visit('/setup/two'); - andThen(() => { - expect(currentURL()).to.equal('/'); - }); - - visit('/setup/three'); - andThen(() => { - expect(currentURL()).to.equal('/'); - }); - }); - - it('redirects to signin if already set up', function () { - // mimick an already setup blog - server.get('/authentication/setup/', function () { - return { - setup: [ - {status: true} - ] - }; - }); - - invalidateSession(application); - - visit('/setup'); - andThen(() => { - expect(currentURL()).to.equal('/signin'); - }); - }); - - describe('with a new blog', function () { - beforeEach(function () { - // mimick a new blog - server.get('/authentication/setup/', function () { - return { - setup: [ - {status: false} - ] - }; - }); - }); - - it('has a successful happy path', function () { - invalidateSession(application); - server.loadFixtures('roles'); - - visit('/setup'); - - andThen(() => { - // it redirects to step one - expect(currentURL(), 'url after accessing /setup') - .to.equal('/setup/one'); - - // it highlights first step - expect(find('.gh-flow-nav .step:first-of-type').hasClass('active')) - .to.be.true; - expect(find('.gh-flow-nav .step:nth-of-type(2)').hasClass('active')) - .to.be.false; - expect(find('.gh-flow-nav .step:nth-of-type(3)').hasClass('active')) - .to.be.false; - - // it displays download count (count increments for each ajax call - // and polling is disabled in testing so our count should be "2" - - // 1 for first load and 1 for first poll) - expect(find('.gh-flow-content em').text()).to.equal('2'); - }); - - click('.btn-green'); - - andThen(() => { - // it transitions to step two - expect(currentURL(), 'url after clicking "Create your account"') - .to.equal('/setup/two'); - - // email field is focused by default - // NOTE: $('x').is(':focus') doesn't work in phantomjs CLI runner - // https://github.com/ariya/phantomjs/issues/10427 - expect(find('[name="email"]').get(0) === document.activeElement, 'email field has focus') - .to.be.true; - }); - - click('.btn-green'); - - andThen(() => { - // it marks fields as invalid - expect(find('.form-group.error').length, 'number of invalid fields') - .to.equal(4); - - // it displays error messages - expect(find('.error .response').length, 'number of in-line validation messages') - .to.equal(4); - - // it displays main error - expect(find('.main-error').length, 'main error is displayed') - .to.equal(1); - }); - - // enter valid details and submit - fillIn('[name="email"]', 'test@example.com'); - fillIn('[name="name"]', 'Test User'); - fillIn('[name="password"]', 'password'); - fillIn('[name="blog-title"]', 'Blog Title'); - click('.btn-green'); - - andThen(() => { - // it transitions to step 3 - expect(currentURL(), 'url after submitting step two') - .to.equal('/setup/three'); - - // submit button is "disabled" - expect(find('button[type="submit"]').hasClass('btn-green'), 'invite button with no emails is white') - .to.be.false; - }); - - // fill in a valid email - fillIn('[name="users"]', 'new-user@example.com'); - - andThen(() => { - // submit button is "enabled" - expect(find('button[type="submit"]').hasClass('btn-green'), 'invite button is green with valid email address') - .to.be.true; - }); - - // submit the invite form - click('button[type="submit"]'); - - andThen(() => { - // it redirects to the home / "content" screen - expect(currentURL(), 'url after submitting invites') - .to.equal('/'); - - // it displays success alert - expect(find('.gh-alert-green').length, 'number of success alerts') - .to.equal(1); - }); - }); - - it('handles validation errors in step 2', function () { - let postCount = 0; - - invalidateSession(application); - server.loadFixtures('roles'); - - server.post('/authentication/setup', function () { - postCount++; - - // validation error - if (postCount === 1) { - return new Mirage.Response(422, {}, { - errors: [ - { - errorType: 'ValidationError', - message: 'Server response message' - } - ] - }); - } - - // server error - if (postCount === 2) { - return new Mirage.Response(500, {}, null); - } - }); - - visit('/setup/two'); - click('.btn-green'); - - andThen(() => { - // non-server validation - expect(find('.main-error').text().trim(), 'error text') - .to.not.be.blank; - }); - - fillIn('[name="email"]', 'test@example.com'); - fillIn('[name="name"]', 'Test User'); - fillIn('[name="password"]', 'password'); - fillIn('[name="blog-title"]', 'Blog Title'); - - // first post - simulated validation error - click('.btn-green'); - - andThen(() => { - expect(find('.main-error').text().trim(), 'error text') - .to.equal('Server response message'); - }); - - // second post - simulated server error - click('.btn-green'); - - andThen(() => { - expect(find('.main-error').text().trim(), 'error text') - .to.be.blank; - - expect(find('.gh-alert-red').length, 'number of alerts') - .to.equal(1); - }); - }); - - it('handles invalid origin error on step 2', function () { - // mimick the API response for an invalid origin - server.post('/authentication/token', function () { - return new Mirage.Response(401, {}, { - errors: [ - { - errorType: 'UnauthorizedError', - message: 'Access Denied from url: unknown.com. Please use the url configured in config.js.' - } - ] - }); - }); - - invalidateSession(application); - server.loadFixtures('roles'); - - visit('/setup/two'); - fillIn('[name="email"]', 'test@example.com'); - fillIn('[name="name"]', 'Test User'); - fillIn('[name="password"]', 'password'); - fillIn('[name="blog-title"]', 'Blog Title'); - click('.btn-green'); - - andThen(() => { - // button should not be spinning - expect(find('.btn-green .spinner').length, 'button has spinner') - .to.equal(0); - // we should show an error message - expect(find('.main-error').text(), 'error text') - .to.equal('Access Denied from url: unknown.com. Please use the url configured in config.js.'); - }); - }); - - it('handles validation errors in step 3', function () { - let input = '[name="users"]'; - let postCount = 0; - let button, formGroup, user; - - invalidateSession(application); - server.loadFixtures('roles'); - - server.post('/users', function (db, request) { - let [params] = JSON.parse(request.requestBody).users; - - postCount++; - - // invalid - if (postCount === 1) { - return new Mirage.Response(422, {}, { - errors: [ - { - errorType: 'ValidationError', - message: 'Dummy validation error' - } - ] - }); - } - - // valid - user = db.users.insert(params); - return { - users: [user] - }; - }); - - // complete step 2 so we can access step 3 - visit('/setup/two'); - fillIn('[name="email"]', 'test@example.com'); - fillIn('[name="name"]', 'Test User'); - fillIn('[name="password"]', 'password'); - fillIn('[name="blog-title"]', 'Blog Title'); - click('.btn-green'); - - // default field/button state - andThen(() => { - formGroup = find('.gh-flow-invite .form-group'); - button = find('.gh-flow-invite button[type="submit"]'); - - expect(formGroup.hasClass('error'), 'default field has error class') - .to.be.false; - - expect(button.text().trim(), 'default button text') - .to.equal('Invite some users'); - - expect(button.hasClass('btn-minor'), 'default button is disabled') - .to.be.true; - }); - - // no users submitted state - click('.gh-flow-invite button[type="submit"]'); - - andThen(() => { - expect(formGroup.hasClass('error'), 'no users submitted field has error class') - .to.be.true; - - expect(button.text().trim(), 'no users submitted button text') - .to.equal('No users to invite'); - - expect(button.hasClass('btn-minor'), 'no users submitted button is disabled') - .to.be.true; - }); - - // single invalid email - fillIn(input, 'invalid email'); - triggerEvent(input, 'blur'); - - andThen(() => { - expect(formGroup.hasClass('error'), 'invalid field has error class') - .to.be.true; - - expect(button.text().trim(), 'single invalid button text') - .to.equal('1 invalid email address'); - - expect(button.hasClass('btn-minor'), 'invalid email button is disabled') - .to.be.true; - }); - - // multiple invalid emails - fillIn(input, 'invalid email\nanother invalid address'); - triggerEvent(input, 'blur'); - - andThen(() => { - expect(button.text().trim(), 'multiple invalid button text') - .to.equal('2 invalid email addresses'); - }); - - // single valid email - fillIn(input, 'invited@example.com'); - triggerEvent(input, 'blur'); - - andThen(() => { - expect(formGroup.hasClass('error'), 'valid field has error class') - .to.be.false; - - expect(button.text().trim(), 'single valid button text') - .to.equal('Invite 1 user'); - - expect(button.hasClass('btn-green'), 'valid email button is enabled') - .to.be.true; - }); - - // multiple valid emails - fillIn(input, 'invited1@example.com\ninvited2@example.com'); - triggerEvent(input, 'blur'); - - andThen(() => { - expect(button.text().trim(), 'multiple valid button text') - .to.equal('Invite 2 users'); - }); - - // submit invitations with simulated failure on 1 invite - click('.btn-green'); - - andThen(() => { - // it redirects to the home / "content" screen - expect(currentURL(), 'url after submitting invites') - .to.equal('/'); - - // it displays success alert - expect(find('.gh-alert-green').length, 'number of success alerts') - .to.equal(1); - - // it displays failure alert - expect(find('.gh-alert-red').length, 'number of failure alerts') - .to.equal(1); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/signin-test.js b/core/client/tests/acceptance/signin-test.js deleted file mode 100644 index 511f90a752..0000000000 --- a/core/client/tests/acceptance/signin-test.js +++ /dev/null @@ -1,131 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import Mirage from 'ember-cli-mirage'; - -describe('Acceptance: Signin', function() { - let application; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - it('redirects if already authenticated', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - authenticateSession(application); - - visit('/signin'); - andThen(() => { - expect(currentURL(), 'current url').to.equal('/'); - }); - }); - - describe('when attempting to signin', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - server.post('/authentication/token', function (db, request) { - // jscs:disable requireCamelCaseOrUpperCaseIdentifiers - let { - grant_type: grantType, - username, - password, - client_id: clientId - } = $.deparam(request.requestBody); - - expect(grantType, 'grant type').to.equal('password'); - expect(username, 'username').to.equal('test@example.com'); - expect(clientId, 'client id').to.equal('ghost-admin'); - - if (password === 'testpass') { - return { - access_token: '5JhTdKI7PpoZv4ROsFoERc6wCHALKFH5jxozwOOAErmUzWrFNARuH1q01TYTKeZkPW7FmV5MJ2fU00pg9sm4jtH3Z1LjCf8D6nNqLYCfFb2YEKyuvG7zHj4jZqSYVodN2YTCkcHv6k8oJ54QXzNTLIDMlCevkOebm5OjxGiJpafMxncm043q9u1QhdU9eee3zouGRMVVp8zkKVoo5zlGMi3zvS2XDpx7xsfk8hKHpUgd7EDDQxmMueifWv7hv6n', - expires_in: 3600, - refresh_token: 'XP13eDjwV5mxOcrq1jkIY9idhdvN3R1Br5vxYpYIub2P5Hdc8pdWMOGmwFyoUshiEB62JWHTl8H1kACJR18Z8aMXbnk5orG28br2kmVgtVZKqOSoiiWrQoeKTqrRV0t7ua8uY5HdDUaKpnYKyOdpagsSPn3WEj8op4vHctGL3svOWOjZhq6F2XeVPMR7YsbiwBE8fjT3VhTB3KRlBtWZd1rE0Qo2EtSplWyjGKv1liAEiL0ndQoLeeSOCH4rTP7', - token_type: 'Bearer' - }; - } else { - return new Mirage.Response(401, {}, { - errors: [{ - errorType: 'UnauthorizedError', - message: 'Invalid Password' - }] - }); - } - // jscs:enable requireCamelCaseOrUpperCaseIdentifiers - }); - }); - - it('errors correctly', function () { - invalidateSession(application); - - visit('/signin'); - - andThen(() => { - expect(currentURL(), 'signin url').to.equal('/signin'); - - expect(find('input[name="identification"]').length, 'email input field') - .to.equal(1); - expect(find('input[name="password"]').length, 'password input field') - .to.equal(1); - }); - - click('.btn-blue'); - - andThen(() => { - expect(find('.form-group.error').length, 'number of invalid fields') - .to.equal(2); - - expect(find('.main-error').length, 'main error is displayed') - .to.equal(1); - }); - - fillIn('[name="identification"]', 'test@example.com'); - fillIn('[name="password"]', 'invalid'); - click('.btn-blue'); - - andThen(() => { - expect(currentURL(), 'current url').to.equal('/signin'); - - expect(find('.main-error').length, 'main error is displayed') - .to.equal(1); - - expect(find('.main-error').text().trim(), 'main error text') - .to.equal('Invalid Password'); - }); - }); - - it('submits successfully', function () { - invalidateSession(application); - - visit('/signin'); - - andThen(() => { - expect(currentURL(), 'current url').to.equal('/signin'); - }); - - fillIn('[name="identification"]', 'test@example.com'); - fillIn('[name="password"]', 'testpass'); - click('.btn-blue'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/'); - }); - }); - }); -}); diff --git a/core/client/tests/acceptance/subscribers-test.js b/core/client/tests/acceptance/subscribers-test.js deleted file mode 100644 index 9c828b737c..0000000000 --- a/core/client/tests/acceptance/subscribers-test.js +++ /dev/null @@ -1,256 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; - -describe('Acceptance: Subscribers', function() { - let application; - - beforeEach(function() { - application = startApp(); - }); - - afterEach(function() { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/subscribers'); - - andThen(function () { - expect(currentURL()).to.equal('/signin'); - }); - }); - - it('redirects editors to posts', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role]}); - - authenticateSession(application); - visit('/subscribers'); - - andThen(function () { - expect(currentURL()).to.equal('/'); - expect(find('.gh-nav-main a:contains("Subscribers")').length, 'sidebar link is visible') - .to.equal(0); - }); - }); - - it('redirects authors to posts', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role]}); - - authenticateSession(application); - visit('/subscribers'); - - andThen(function () { - expect(currentURL()).to.equal('/'); - expect(find('.gh-nav-main a:contains("Subscribers")').length, 'sidebar link is visible') - .to.equal(0); - }); - }); - - describe('an admin', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Administrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('can manage subscribers', function () { - server.createList('subscriber', 40); - - authenticateSession(application); - visit('/'); - click('.gh-nav-main a:contains("Subscribers")'); - - andThen(function() { - // it navigates to the correct page - expect(currentPath()).to.equal('subscribers.index'); - - // it has correct page title - expect(document.title, 'page title') - .to.equal('Subscribers - Test Blog'); - - // it loads the first page - expect(find('.subscribers-table .lt-body .lt-row').length, 'number of subscriber rows') - .to.equal(30); - - // it shows the total number of subscribers - expect(find('#total-subscribers').text().trim(), 'displayed subscribers total') - .to.equal('40'); - - // it defaults to sorting by created_at desc - let [lastRequest] = server.pretender.handledRequests.slice(-1); - expect(lastRequest.queryParams.order).to.equal('created_at desc'); - - let createdAtHeader = find('.subscribers-table th:contains("Subscription Date")'); - expect(createdAtHeader.hasClass('is-sorted'), 'createdAt column is sorted') - .to.be.true; - expect(createdAtHeader.find('.icon-descending').length, 'createdAt column has descending icon') - .to.equal(1); - }); - - // click the column to re-order - click('th:contains("Subscription Date")'); - - andThen(function () { - // it flips the directions and re-fetches - let [lastRequest] = server.pretender.handledRequests.slice(-1); - expect(lastRequest.queryParams.order).to.equal('created_at asc'); - - let createdAtHeader = find('.subscribers-table th:contains("Subscription Date")'); - expect(createdAtHeader.find('.icon-ascending').length, 'createdAt column has ascending icon') - .to.equal(1); - - // scroll to the bottom of the table to simulate infinite scroll - find('.subscribers-table').scrollTop(find('.subscribers-table .ember-light-table').height()); - }); - - // trigger infinite scroll - triggerEvent('.subscribers-table', 'scroll'); - - andThen(function () { - // it loads the next page - expect(find('.subscribers-table .lt-body .lt-row').length, 'number of subscriber rows after infinite-scroll') - .to.equal(40); - }); - - // click the add subscriber button - click('.btn:contains("Add Subscriber")'); - - andThen(function () { - // it displays the add subscriber modal - expect(find('.fullscreen-modal').length, 'add subscriber modal displayed') - .to.equal(1); - }); - - // cancel the modal - click('.fullscreen-modal .btn:contains("Cancel")'); - - andThen(function () { - // it closes the add subscriber modal - expect(find('.fullscreen-modal').length, 'add subscriber modal displayed after cancel') - .to.equal(0); - }); - - // save a new subscriber - click('.btn:contains("Add Subscriber")'); - fillIn('.fullscreen-modal input[name="email"]', 'test@example.com'); - click('.fullscreen-modal .btn:contains("Add")'); - - andThen(function () { - // the add subscriber modal is closed - expect(find('.fullscreen-modal').length, 'add subscriber modal displayed after save') - .to.equal(0); - - // the subscriber is added to the table - expect(find('.subscribers-table .lt-body .lt-row:first-of-type .lt-cell:first-of-type').text().trim(), 'first email in list after addition') - .to.equal('test@example.com'); - - // the table is scrolled to the top - // TODO: implement scroll to new record after addition - // expect(find('.subscribers-table').scrollTop(), 'scroll position after addition') - // .to.equal(0); - - // the subscriber total is updated - expect(find('#total-subscribers').text().trim(), 'subscribers total after addition') - .to.equal('41'); - }); - - // saving a duplicate subscriber - click('.btn:contains("Add Subscriber")'); - fillIn('.fullscreen-modal input[name="email"]', 'test@example.com'); - click('.fullscreen-modal .btn:contains("Add")'); - - andThen(function () { - // the validation error is displayed - expect(find('.fullscreen-modal .error .response').text().trim(), 'duplicate email validation') - .to.equal('Email already exists.'); - - // the subscriber is not added to the table - expect(find('.lt-cell:contains(test@example.com)').length, 'number of "test@example.com rows"') - .to.equal(1); - - // the subscriber total is unchanged - expect(find('#total-subscribers').text().trim(), 'subscribers total after failed add') - .to.equal('41'); - }); - - // deleting a subscriber - click('.fullscreen-modal .btn:contains("Cancel")'); - click('.subscribers-table tbody tr:first-of-type button:last-of-type'); - - andThen(function () { - // it displays the delete subscriber modal - expect(find('.fullscreen-modal').length, 'delete subscriber modal displayed') - .to.equal(1); - }); - - // cancel the modal - click('.fullscreen-modal .btn:contains("Cancel")'); - - andThen(function () { - // return pauseTest(); - // it closes the add subscriber modal - expect(find('.fullscreen-modal').length, 'delete subscriber modal displayed after cancel') - .to.equal(0); - }); - - click('.subscribers-table tbody tr:first-of-type button:last-of-type'); - click('.fullscreen-modal .btn:contains("Delete")'); - - andThen(function () { - // the add subscriber modal is closed - expect(find('.fullscreen-modal').length, 'delete subscriber modal displayed after confirm') - .to.equal(0); - - // the subscriber is removed from the table - expect(find('.subscribers-table .lt-body .lt-row:first-of-type .lt-cell:first-of-type').text().trim(), 'first email in list after addition') - .to.not.equal('test@example.com'); - - // the subscriber total is updated - expect(find('#total-subscribers').text().trim(), 'subscribers total after addition') - .to.equal('40'); - }); - - // click the import subscribers button - click('.btn:contains("Import CSV")'); - - andThen(function () { - // it displays the import subscribers modal - expect(find('.fullscreen-modal').length, 'import subscribers modal displayed') - .to.equal(1); - }); - - // cancel the modal - click('.fullscreen-modal .btn:contains("Cancel")'); - - andThen(function () { - // it closes the import subscribers modal - expect(find('.fullscreen-modal').length, 'import subscribers modal displayed after cancel') - .to.equal(0); - }); - - // TODO: how to simulate file upload? - - // re-open import modal - // upload a file - // modal title changes - // modal button changes - // table is reset - // close modal - }); - }); -}); diff --git a/core/client/tests/acceptance/team-test.js b/core/client/tests/acceptance/team-test.js deleted file mode 100644 index 62910d1c2d..0000000000 --- a/core/client/tests/acceptance/team-test.js +++ /dev/null @@ -1,479 +0,0 @@ -/* jshint expr:true */ -import { - describe, - it, - beforeEach, - afterEach -} from 'mocha'; -import { expect } from 'chai'; -import Ember from 'ember'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; -import { invalidateSession, authenticateSession } from 'ghost/tests/helpers/ember-simple-auth'; -import { errorOverride, errorReset } from 'ghost/tests/helpers/adapter-error'; -import Mirage from 'ember-cli-mirage'; - -describe('Acceptance: Team', function () { - let application; - - beforeEach(function () { - application = startApp(); - }); - - afterEach(function () { - destroyApp(application); - }); - - it('redirects to signin when not authenticated', function () { - invalidateSession(application); - visit('/team'); - - andThen(function () { - expect(currentURL()).to.equal('/signin'); - }); - }); - - it('redirects correctly when authenticated as author', function () { - let role = server.create('role', {name: 'Author'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - server.create('user', {slug: 'no-access'}); - - authenticateSession(application); - visit('/team/no-access'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-user'); - }); - }); - - it('redirects correctly when authenticated as editor', function () { - let role = server.create('role', {name: 'Editor'}); - let user = server.create('user', {roles: [role], slug: 'test-user'}); - - server.create('user', {slug: 'no-access'}); - - authenticateSession(application); - visit('/team/no-access'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team'); - }); - }); - - describe('when logged in', function () { - beforeEach(function () { - let role = server.create('role', {name: 'Admininstrator'}); - let user = server.create('user', {roles: [role]}); - - server.loadFixtures(); - - return authenticateSession(application); - }); - - it('it renders and navigates correctly', function () { - let user1 = server.create('user'); - let user2 = server.create('user'); - - visit('/team'); - - andThen(() => { - // doesn't do any redirecting - expect(currentURL(), 'currentURL').to.equal('/team'); - - // it has correct page title - expect(document.title, 'page title').to.equal('Team - Test Blog'); - - // it shows 3 users in list (includes currently logged in user) - expect(find('.user-list .user-list-item').length, 'user list count') - .to.equal(3); - - click('.user-list-item:last'); - - andThen(() => { - // url is correct - expect(currentURL(), 'url after clicking user').to.equal(`/team/${user2.slug}`); - - // title is correct - expect(document.title, 'title after clicking user').to.equal('Team - User - Test Blog'); - - // view title should exist and be linkable and active - expect(find('.view-title a[href="/team"]').hasClass('active'), 'has linkable url back to team main page') - .to.be.true; - }); - - click('.view-title a'); - - andThen(() => { - // url should be /team again - expect(currentURL(), 'url after clicking back').to.equal('/team'); - }); - }); - }); - - describe('invite new user', function () { - let emailInputField = '.fullscreen-modal input[name="email"]'; - - // @TODO: Evaluate after the modal PR goes in - it('modal loads correctly', function () { - visit('/team'); - - andThen(() => { - // url is correct - expect(currentURL(), 'currentURL').to.equal('/team'); - - // invite user button exists - expect(find('.view-actions .btn-green').html(), 'invite people button text') - .to.equal('Invite People'); - }); - - click('.view-actions .btn-green'); - - andThen(() => { - let roleOptions = find('#new-user-role select option'); - - function checkOwnerExists() { - for (let i in roleOptions) { - if (roleOptions[i].tagName === 'option' && roleOptions[i].text === 'Owner') { - return true; - } - } - return false; - } - - function checkSelectedIsAuthor() { - for (let i in roleOptions) { - if (roleOptions[i].selected) { - return roleOptions[i].text === 'Author'; - } - } - return false; - } - - // should be 3 available roles - expect(roleOptions.length, 'number of available roles').to.equal(3); - - expect(checkOwnerExists(), 'owner role isn\'t available').to.be.false; - expect(checkSelectedIsAuthor(), 'author role is selected initially').to.be.true; - }); - }); - - it('sends an invite correctly', function () { - visit('/team'); - - andThen(() => { - expect(find('.user-list.invited-users .user-list-item').length, 'number of invited users').to.equal(0); - }); - - click('.view-actions .btn-green'); - click(emailInputField); - triggerEvent(emailInputField, 'blur'); - - andThen(() => { - expect(find('.modal-body .form-group:first').hasClass('error'), 'email input has error status').to.be.true; - expect(find('.modal-body .form-group:first .response').text()).to.contain('Please enter an email.'); - }); - - fillIn(emailInputField, 'test@example.com'); - click('.fullscreen-modal .btn-green'); - - andThen(() => { - expect(find('.user-list.invited-users .user-list-item').length, 'number of invited users').to.equal(1); - expect(find('.user-list.invited-users .user-list-item:first .name').text(), 'name of invited user').to.equal('test@example.com'); - }); - - click('.user-list.invited-users .user-list-item:first .user-list-item-aside .user-list-action:contains("Revoke")'); - - andThen(() => { - expect(find('.user-list.invited-users .user-list-item').length, 'number of invited users').to.equal(0); - }); - }); - - it('fails sending an invite correctly', function () { - server.create('user', {email: 'test1@example.com'}); - server.create('user', {email: 'test2@example.com', status: 'invited'}); - - visit('/team'); - - // check our users lists are what we expect - andThen(() => { - expect(find('.user-list.invited-users .user-list-item').length, 'number of invited users') - .to.equal(1); - // number of active users is 2 because of the logged-in user - expect(find('.user-list.active-users .user-list-item').length, 'number of active users') - .to.equal(2); - }); - - // click the "invite new user" button to open the modal - click('.view-actions .btn-green'); - - // fill in and submit the invite user modal with an existing user - fillIn(emailInputField, 'test1@example.com'); - click('.fullscreen-modal .btn-green'); - - andThen(() => { - // check the inline-validation - expect(find('.fullscreen-modal .error .response').text().trim(), 'inviting existing user error') - .to.equal('A user with that email address already exists.'); - }); - - // fill in and submit the invite user modal with an invited user - fillIn(emailInputField, 'test2@example.com'); - click('.fullscreen-modal .btn-green'); - - andThen(() => { - // check the inline-validation - expect(find('.fullscreen-modal .error .response').text().trim(), 'inviting invited user error') - .to.equal('A user with that email address was already invited.'); - - // ensure that there's been no change in our user lists - expect(find('.user-list.invited-users .user-list-item').length, 'number of invited users after failed invites') - .to.equal(1); - expect(find('.user-list.active-users .user-list-item').length, 'number of active users after failed invites') - .to.equal(2); - }); - }); - }); - - describe('existing user', function () { - let user; - - beforeEach(function () { - server.create('user', { - slug: 'test-1', - name: 'Test User', - facebook: 'test', - twitter: '@test' - }); - - server.loadFixtures(); - }); - - it('input fields reset and validate correctly', function () { - // test user name - visit('/team/test-1'); - - andThen(() => { - expect(currentURL(), 'currentURL').to.equal('/team/test-1'); - expect(find('.user-details-top .first-form-group input.user-name').val(), 'current user name').to.equal('Test User'); - }); - - // test empty user name - fillIn('.user-details-top .first-form-group input.user-name', ''); - triggerEvent('.user-details-top .first-form-group input.user-name', 'blur'); - - andThen(() => { - expect(find('.user-details-top .first-form-group').hasClass('error'), 'username input is in error state with blank input').to.be.true; - }); - - // test too long user name - fillIn('.user-details-top .first-form-group input.user-name', new Array(160).join('a')); - triggerEvent('.user-details-top .first-form-group input.user-name', 'blur'); - - andThen(() => { - expect(find('.user-details-top .first-form-group').hasClass('error'), 'username input is in error state with too long input').to.be.true; - }); - - // reset name field - fillIn('.user-details-top .first-form-group input.user-name', 'Test User'); - - andThen(() => { - expect(find('.user-details-bottom input[name="user"]').val(), 'slug value is default').to.equal('test-1'); - }); - - fillIn('.user-details-bottom input[name="user"]', ''); - triggerEvent('.user-details-bottom input[name="user"]', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom input[name="user"]').val(), 'slug value is reset to original upon empty string').to.equal('test-1'); - }); - - fillIn('.user-details-bottom input[name="user"]', 'white space'); - triggerEvent('.user-details-bottom input[name="user"]', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom input[name="user"]').val(), 'slug value is correctly dasherized').to.equal('white-space'); - }); - - fillIn('.user-details-bottom input[name="email"]', 'thisisnotanemail'); - triggerEvent('.user-details-bottom input[name="email"]', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(2)').hasClass('error'), 'email input should be in error state with invalid email').to.be.true; - }); - - fillIn('.user-details-bottom input[name="email"]', 'test@example.com'); - fillIn('#user-location', new Array(160).join('a')); - triggerEvent('#user-location', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(3)').hasClass('error'), 'location input should be in error state').to.be.true; - }); - - fillIn('#user-location', ''); - fillIn('#user-website', 'thisisntawebsite'); - triggerEvent('#user-website', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(4)').hasClass('error'), 'website input should be in error state').to.be.true; - }); - - // Testing Facebook input - - andThen(() => { - // displays initial value - expect(find('#user-facebook').val(), 'initial facebook value') - .to.equal('https://www.facebook.com/test'); - }); - - triggerEvent('#user-facebook', 'focus'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - // regression test: we still have a value after the input is - // focused and then blurred without any changes - expect(find('#user-facebook').val(), 'facebook value after blur with no change') - .to.equal('https://www.facebook.com/test'); - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', ')(*&%^%)'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.true; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'pages/)(*&%^%)'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('#user-facebook').val()).to.be.equal('https://www.facebook.com/pages/)(*&%^%)'); - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.false; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'testing'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('#user-facebook').val()).to.be.equal('https://www.facebook.com/testing'); - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.false; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'somewebsite.com/pages/some-facebook-page/857469375913?ref=ts'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('#user-facebook').val()).to.be.equal('https://www.facebook.com/pages/some-facebook-page/857469375913?ref=ts'); - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.false; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'test'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.true; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'http://twitter.com/testuser'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('#user-facebook').val()).to.be.equal('https://www.facebook.com/testuser'); - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.false; - }); - - fillIn('#user-facebook', ''); - fillIn('#user-facebook', 'facebook.com/testing'); - triggerEvent('#user-facebook', 'blur'); - - andThen(() => { - expect(find('#user-facebook').val()).to.be.equal('https://www.facebook.com/testing'); - expect(find('.user-details-bottom .form-group:nth-of-type(5)').hasClass('error'), 'facebook input should be in error state').to.be.false; - }); - - // Testing Twitter input - - andThen(() => { - // loads fixtures and performs transform - expect(find('#user-twitter').val(), 'initial twitter value') - .to.equal('https://twitter.com/test'); - }); - - triggerEvent('#user-twitter', 'focus'); - triggerEvent('#user-twitter', 'blur'); - - andThen(() => { - // regression test: we still have a value after the input is - // focused and then blurred without any changes - expect(find('#user-twitter').val(), 'twitter value after blur with no change') - .to.equal('https://twitter.com/test'); - }); - - fillIn('#user-twitter', ''); - fillIn('#user-twitter', ')(*&%^%)'); - triggerEvent('#user-twitter', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(6)').hasClass('error'), 'twitter input should be in error state').to.be.true; - }); - - fillIn('#user-twitter', ''); - fillIn('#user-twitter', 'name'); - triggerEvent('#user-twitter', 'blur'); - - andThen(() => { - expect(find('#user-twitter').val()).to.be.equal('https://twitter.com/name'); - expect(find('.user-details-bottom .form-group:nth-of-type(6)').hasClass('error'), 'twitter input should be in error state').to.be.false; - }); - - fillIn('#user-twitter', ''); - fillIn('#user-twitter', 'http://github.com/user'); - triggerEvent('#user-twitter', 'blur'); - - andThen(() => { - expect(find('#user-twitter').val()).to.be.equal('https://twitter.com/user'); - expect(find('.user-details-bottom .form-group:nth-of-type(6)').hasClass('error'), 'twitter input should be in error state').to.be.false; - }); - - fillIn('#user-twitter', ''); - fillIn('#user-twitter', 'twitter.com/user'); - triggerEvent('#user-twitter', 'blur'); - - andThen(() => { - expect(find('#user-twitter').val()).to.be.equal('https://twitter.com/user'); - expect(find('.user-details-bottom .form-group:nth-of-type(6)').hasClass('error'), 'twitter input should be in error state').to.be.false; - }); - - fillIn('#user-website', ''); - fillIn('#user-bio', new Array(210).join('a')); - triggerEvent('#user-bio', 'blur'); - - andThen(() => { - expect(find('.user-details-bottom .form-group:nth-of-type(7)').hasClass('error'), 'bio input should be in error state').to.be.true; - }); - }); - }); - - it('redirects to 404 when tag does not exist', function () { - server.get('/users/slug/unknown/', function () { - return new Mirage.Response(404, {'Content-Type': 'application/json'}, {errors: [{message: 'User not found.', errorType: 'NotFoundError'}]}); - }); - - errorOverride(); - - visit('/team/unknown'); - - andThen(() => { - errorReset(); - expect(currentPath()).to.equal('error404'); - expect(currentURL()).to.equal('/team/unknown'); - }); - }); - }); -}); diff --git a/core/client/tests/helpers/adapter-error.js b/core/client/tests/helpers/adapter-error.js deleted file mode 100644 index 567938f1ee..0000000000 --- a/core/client/tests/helpers/adapter-error.js +++ /dev/null @@ -1,19 +0,0 @@ -import Ember from 'ember'; - -// This is needed for testing error responses in acceptance tests -// See http://williamsbdev.com/posts/testing-rsvp-errors-handled-globally/ - -let originalException; -let originalLoggerError; - -export function errorOverride() { - originalException = Ember.Test.adapter.exception; - originalLoggerError = Ember.Logger.error; - Ember.Test.adapter.exception = function () {}; - Ember.Logger.error = function () {}; -} - -export function errorReset() { - Ember.Test.adapter.exception = originalException; - Ember.Logger.error = originalLoggerError; -} diff --git a/core/client/tests/helpers/destroy-app.js b/core/client/tests/helpers/destroy-app.js deleted file mode 100644 index 13d4989501..0000000000 --- a/core/client/tests/helpers/destroy-app.js +++ /dev/null @@ -1,5 +0,0 @@ -import Ember from 'ember'; - -export default function destroyApp(application) { - Ember.run(application, 'destroy'); -} diff --git a/core/client/tests/helpers/module-for-acceptance.js b/core/client/tests/helpers/module-for-acceptance.js deleted file mode 100644 index d9d22b6958..0000000000 --- a/core/client/tests/helpers/module-for-acceptance.js +++ /dev/null @@ -1,23 +0,0 @@ -import { module } from 'qunit'; -import startApp from '../helpers/start-app'; -import destroyApp from '../helpers/destroy-app'; - -export default function (name, options = {}) { - module(name, { - beforeEach() { - this.application = startApp(); - - if (options.beforeEach) { - options.beforeEach(...arguments); - } - }, - - afterEach() { - if (options.afterEach) { - options.afterEach(...arguments); - } - - destroyApp(this.application); - } - }); -} diff --git a/core/client/tests/helpers/resolver.js b/core/client/tests/helpers/resolver.js deleted file mode 100644 index 50e9179f54..0000000000 --- a/core/client/tests/helpers/resolver.js +++ /dev/null @@ -1,11 +0,0 @@ -import Resolver from '../../resolver'; -import config from '../../config/environment'; - -const resolver = Resolver.create(); - -resolver.namespace = { - modulePrefix: config.modulePrefix, - podModulePrefix: config.podModulePrefix -}; - -export default resolver; diff --git a/core/client/tests/helpers/start-app.js b/core/client/tests/helpers/start-app.js deleted file mode 100644 index 5b04a29d0f..0000000000 --- a/core/client/tests/helpers/start-app.js +++ /dev/null @@ -1,21 +0,0 @@ -import Ember from 'ember'; -import Application from '../../app'; -import config from '../../config/environment'; - -const {assign, run} = Ember; - -export default function startApp(attrs) { - let attributes = assign({}, config.APP); - let application; - - // use defaults, but you can override; - attributes = assign(attributes, attrs); - - run(function () { - application = Application.create(attributes); - application.setupForTesting(); - application.injectTestHelpers(); - }); - - return application; -} diff --git a/core/client/tests/index.html b/core/client/tests/index.html deleted file mode 100644 index dc9021f1ca..0000000000 --- a/core/client/tests/index.html +++ /dev/null @@ -1,60 +0,0 @@ -<!DOCTYPE html> -<html> - <head> - <meta charset="utf-8"> - <meta http-equiv="X-UA-Compatible" content="IE=edge"> - <title>Ghost Tests</title> - <meta name="description" content=""> - <meta name="viewport" content="width=device-width, initial-scale=1"> - - {{content-for 'head'}} - {{content-for 'test-head'}} - - <meta name="env-fileStorage" content="true" /> - <meta name="env-apps" content="false" /> - <meta name="env-blogUrl" content="http://localhost:7357/" /> - <meta name="env-blogTitle" content="Test Blog" /> - <meta name="env-routeKeywords" content="{"tag":"tag","author":"author","page":"page","preview":"p","private":"private"}" /> - <meta name="env-clientId" content="ghost-admin" /> - <meta name="env-clientSecret" content="5076dc643873" /> - - <link rel="stylesheet" href="assets/vendor.css"> - <link rel="stylesheet" href="assets/ghost.css"> - <link rel="stylesheet" href="assets/test-support.css"> - - {{content-for 'head-footer'}} - {{content-for 'test-head-footer'}} - - <style> - /* fix to ensure we use full viewport height when testing */ - #ember-testing { - height: 100%; - } - #ember-testing > div { - height: 100%; - } - /* fix firefox not supporting `zoom: 50%` */ - _::-moz-range-track, body:last-child #ember-testing { - -moz-transform-origin: 0 0; - -moz-transform: scale(0.5); - width: 200%; - height: 200%; - } - </style> - </head> - <body> - - {{content-for 'body'}} - {{content-for 'test-body'}} - - <script src="testem.js" integrity=""></script> - <script src="assets/vendor.js"></script> - <script src="assets/test-support.js"></script> - <script src="assets/ghost.js"></script> - <script src="assets/tests.js"></script> - <script src="assets/test-loader.js"></script> - - {{content-for 'body-footer'}} - {{content-for 'test-body-footer'}} - </body> -</html> diff --git a/core/client/tests/integration/adapters/tag-test.js b/core/client/tests/integration/adapters/tag-test.js deleted file mode 100644 index 839cd82e6e..0000000000 --- a/core/client/tests/integration/adapters/tag-test.js +++ /dev/null @@ -1,63 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; -import Pretender from 'pretender'; - -describeModule( - 'adapter:tag', - 'Integration: Adapter: tag', - { - integration: true - }, - function () { - let server, store; - - beforeEach(function () { - store = this.container.lookup('service:store'); - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('loads tags from regular endpoint when all are fetched', function (done) { - server.get('/ghost/api/v0.1/tags/', function () { - return [200, {'Content-Type': 'application/json'}, JSON.stringify({tags: [{ - id: 1, - name: 'Tag 1', - slug: 'tag-1' - }, { - id: 2, - name: 'Tag 2', - slug: 'tag-2' - }]})]; - }); - - store.findAll('tag', {reload: true}).then((tags) => { - expect(tags).to.be.ok; - expect(tags.objectAtContent(0).get('name')).to.equal('Tag 1'); - done(); - }); - }); - - it('loads tag from slug endpoint when single tag is queried and slug is passed in', function (done) { - server.get('/ghost/api/v0.1/tags/slug/tag-1/', function () { - return [200, {'Content-Type': 'application/json'}, JSON.stringify({tags: [{ - id: 1, - slug: 'tag-1', - name: 'Tag 1' - }]})]; - }); - - store.queryRecord('tag', {slug: 'tag-1'}).then((tag) => { - expect(tag).to.be.ok; - expect(tag.get('name')).to.equal('Tag 1'); - done(); - }); - }); - } -); diff --git a/core/client/tests/integration/adapters/user-test.js b/core/client/tests/integration/adapters/user-test.js deleted file mode 100644 index b9c2c884ac..0000000000 --- a/core/client/tests/integration/adapters/user-test.js +++ /dev/null @@ -1,86 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; -import Pretender from 'pretender'; - -describeModule( - 'adapter:user', - 'Integration: Adapter: user', - { - integration: true - }, - function () { - let server, store; - - beforeEach(function () { - store = this.container.lookup('service:store'); - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('loads users from regular endpoint when all are fetched', function (done) { - server.get('/ghost/api/v0.1/users/', function () { - return [200, {'Content-Type': 'application/json'}, JSON.stringify({users: [{ - id: 1, - name: 'User 1', - slug: 'user-1' - }, { - id: 2, - name: 'User 2', - slug: 'user-2' - }]})]; - }); - - store.findAll('user', {reload: true}).then((users) => { - expect(users).to.be.ok; - expect(users.objectAtContent(0).get('name')).to.equal('User 1'); - done(); - }); - }); - - it('loads user from slug endpoint when single user is queried and slug is passed in', function (done) { - server.get('/ghost/api/v0.1/users/slug/user-1/', function () { - return [200, {'Content-Type': 'application/json'}, JSON.stringify({users: [{ - id: 1, - slug: 'user-1', - name: 'User 1' - }]})]; - }); - - store.queryRecord('user', {slug: 'user-1'}).then((user) => { - expect(user).to.be.ok; - expect(user.get('name')).to.equal('User 1'); - done(); - }); - }); - - it('handles "include" parameter when querying single user via slug', function (done) { - server.get('/ghost/api/v0.1/users/slug/user-1/', (request) => { - let params = request.queryParams; - expect(params.include, 'include query').to.equal('roles,count.posts'); - - return [200, {'Content-Type': 'application/json'}, JSON.stringify({users: [{ - id: 1, - slug: 'user-1', - name: 'User 1', - count: { - posts: 5 - } - }]})]; - }); - - store.queryRecord('user', {slug: 'user-1', include: 'count.posts'}).then((user) => { - expect(user).to.be.ok; - expect(user.get('name')).to.equal('User 1'); - expect(user.get('count.posts')).to.equal(5); - done(); - }); - }); - } -); diff --git a/core/client/tests/integration/components/gh-alert-test.js b/core/client/tests/integration/components/gh-alert-test.js deleted file mode 100644 index aa976aa4a4..0000000000 --- a/core/client/tests/integration/components/gh-alert-test.js +++ /dev/null @@ -1,46 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; - -describeComponent( - 'gh-alert', - 'Integration: Component: gh-alert', - { - integration: true - }, - function () { - it('renders', function () { - this.set('message', {message: 'Test message', type: 'success'}); - - this.render(hbs`{{gh-alert message=message}}`); - - expect(this.$('article.gh-alert')).to.have.length(1); - let $alert = this.$('.gh-alert'); - - expect($alert.text()).to.match(/Test message/); - }); - - it('maps message types to CSS classes', function () { - this.set('message', {message: 'Test message', type: 'success'}); - - this.render(hbs`{{gh-alert message=message}}`); - let $alert = this.$('.gh-alert'); - - this.set('message.type', 'success'); - expect($alert.hasClass('gh-alert-green'), 'success class isn\'t green').to.be.true; - - this.set('message.type', 'error'); - expect($alert.hasClass('gh-alert-red'), 'success class isn\'t red').to.be.true; - - this.set('message.type', 'warn'); - expect($alert.hasClass('gh-alert-yellow'), 'success class isn\'t yellow').to.be.true; - - this.set('message.type', 'info'); - expect($alert.hasClass('gh-alert-blue'), 'success class isn\'t blue').to.be.true; - }); - } -); diff --git a/core/client/tests/integration/components/gh-alerts-test.js b/core/client/tests/integration/components/gh-alerts-test.js deleted file mode 100644 index 4c66eb89ef..0000000000 --- a/core/client/tests/integration/components/gh-alerts-test.js +++ /dev/null @@ -1,58 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -const {run} = Ember; -const emberA = Ember.A; - -let notificationsStub = Ember.Service.extend({ - alerts: emberA() -}); - -describeComponent( - 'gh-alerts', - 'Integration: Component: gh-alerts', - { - integration: true - }, - function () { - beforeEach(function () { - this.register('service:notifications', notificationsStub); - this.inject.service('notifications', {as: 'notifications'}); - - this.set('notifications.alerts', [ - {message: 'First', type: 'error'}, - {message: 'Second', type: 'warn'} - ]); - }); - - it('renders', function () { - this.render(hbs`{{gh-alerts}}`); - expect(this.$('.gh-alerts').length).to.equal(1); - expect(this.$('.gh-alerts').children().length).to.equal(2); - - this.set('notifications.alerts', emberA()); - expect(this.$('.gh-alerts').children().length).to.equal(0); - }); - - it('triggers "notify" action when message count changes', function () { - let expectedCount = 0; - - // test double for notify action - this.set('notify', (count) => expect(count).to.equal(expectedCount)); - - this.render(hbs`{{gh-alerts notify=(action notify)}}`); - - expectedCount = 3; - this.get('notifications.alerts').pushObject({message: 'Third', type: 'success'}); - - expectedCount = 0; - this.set('notifications.alerts', emberA()); - }); - } -); diff --git a/core/client/tests/integration/components/gh-cm-editor-test.js b/core/client/tests/integration/components/gh-cm-editor-test.js deleted file mode 100644 index c88b8ad703..0000000000 --- a/core/client/tests/integration/components/gh-cm-editor-test.js +++ /dev/null @@ -1,53 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -const {run} = Ember; - -describeComponent( - 'gh-cm-editor', - 'Integration: Component: gh-cm-editor', - { - integration: true - }, - function () { - it('handles editor events', function () { - this.set('text', ''); - - this.render(hbs`{{gh-cm-editor class="gh-input" value=text}}`); - let input = this.$('.gh-input'); - - expect(input.hasClass('focused'), 'has focused class on first render') - .to.be.false; - - run(() => { - input.find('textarea').trigger('focus'); - }); - - expect(input.hasClass('focused'), 'has focused class after focus') - .to.be.true; - - run(() => { - input.find('textarea').trigger('blur'); - }); - - expect(input.hasClass('focused'), 'loses focused class on blur') - .to.be.false; - - run(() => { - // access CodeMirror directly as it doesn't pick up changes - // to the textarea - let cm = input.find('.CodeMirror').get(0).CodeMirror; - cm.setValue('Testing'); - }); - - expect(this.get('text'), 'text value after CM editor change') - .to.equal('Testing'); - }); - } -); diff --git a/core/client/tests/integration/components/gh-feature-flag-test.js b/core/client/tests/integration/components/gh-feature-flag-test.js deleted file mode 100644 index 30813c1fb7..0000000000 --- a/core/client/tests/integration/components/gh-feature-flag-test.js +++ /dev/null @@ -1,60 +0,0 @@ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import wait from 'ember-test-helpers/wait'; - -const featureStub = Ember.Service.extend({ - testFlag: true -}); - -describeComponent( - 'gh-feature-flag', - 'Integration: Component: gh-feature-flag', - { - integration: true - }, - function() { - let server; - - beforeEach(function () { - this.register('service:feature', featureStub); - this.inject.service('feature', {as: 'feature'}); - }); - - it('renders properties correctly', function () { - this.render(hbs`{{gh-feature-flag "testFlag"}}`); - expect(this.$()).to.have.length(1); - expect(this.$('label').attr('for')).to.equal(this.$('input[type="checkbox"]').attr('id')); - }); - - it('renders correctly when flag is set to true', function () { - this.render(hbs`{{gh-feature-flag "testFlag"}}`); - expect(this.$()).to.have.length(1); - expect(this.$('label input[type="checkbox"]').prop('checked')).to.be.true; - }); - - it('renders correctly when flag is set to false', function () { - this.set('feature.testFlag', false); - - this.render(hbs`{{gh-feature-flag "testFlag"}}`); - expect(this.$()).to.have.length(1); - - expect(this.$('label input[type="checkbox"]').prop('checked')).to.be.false; - }); - - it('updates to reflect changes in flag property', function () { - this.render(hbs`{{gh-feature-flag "testFlag"}}`); - expect(this.$()).to.have.length(1); - - expect(this.$('label input[type="checkbox"]').prop('checked')).to.be.true; - - this.$('label').click(); - - expect(this.$('label input[type="checkbox"]').prop('checked')).to.be.false; - }); - } -); diff --git a/core/client/tests/integration/components/gh-file-uploader-test.js b/core/client/tests/integration/components/gh-file-uploader-test.js deleted file mode 100644 index c5f30cb24c..0000000000 --- a/core/client/tests/integration/components/gh-file-uploader-test.js +++ /dev/null @@ -1,98 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; - -const {run} = Ember; - -const stubSuccessfulUpload = function (server, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [200, {'Content-Type': 'application/json'}, '"/content/images/test.png"']; - }, delay); -}; - -const stubFailedUpload = function (server, code, error, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [code, {'Content-Type': 'application/json'}, JSON.stringify({ - errors: [{ - errorType: error, - message: `Error: ${error}` - }] - })]; - }, delay); -}; - -describeComponent( - 'gh-file-uploader', - 'Integration: Component: gh-file-uploader', - { - integration: true - }, - function() { - let server; - - beforeEach(function () { - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('renders', function() { - this.render(hbs`{{gh-file-uploader}}`); - - expect(this.$('label').text().trim(), 'default label') - .to.equal('Select or drag-and-drop a file'); - }); - - it('renders form with supplied label text', function () { - this.set('labelText', 'My label'); - this.render(hbs`{{gh-file-uploader labelText=labelText}}`); - - expect(this.$('label').text().trim(), 'label') - .to.equal('My label'); - }); - - it('generates request to supplied endpoint', function (done) { - stubSuccessfulUpload(server); - this.set('uploadUrl', '/ghost/api/v0.1/uploads/'); - - this.render(hbs`{{gh-file-uploader url=uploadUrl}}`); - this.$('input[type="file"]').trigger('change'); - - wait().then(() => { - expect(server.handledRequests.length).to.equal(1); - expect(server.handledRequests[0].url).to.equal('/ghost/api/v0.1/uploads/'); - done(); - }); - }); - - it('handles drag over/leave', function () { - this.render(hbs`{{gh-file-uploader}}`); - - run(() => { - let dragover = Ember.$.Event('dragover', { - dataTransfer: { - files: [] - } - }); - this.$('.gh-image-uploader').trigger(dragover); - }); - - expect(this.$('.gh-image-uploader').hasClass('--drag-over'), 'has drag-over class').to.be.true; - - run(() => { - this.$('.gh-image-uploader').trigger('dragleave'); - }); - - expect(this.$('.gh-image-uploader').hasClass('--drag-over'), 'has drag-over class').to.be.false; - }); - } -); diff --git a/core/client/tests/integration/components/gh-image-uploader-test.js b/core/client/tests/integration/components/gh-image-uploader-test.js deleted file mode 100644 index ea3513d3d6..0000000000 --- a/core/client/tests/integration/components/gh-image-uploader-test.js +++ /dev/null @@ -1,244 +0,0 @@ -/* jshint expr:true */ -import Ember from 'ember'; -import sinon from 'sinon'; -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; - -const {run} = Ember; - -const keyCodes = { - enter: 13 -}; - -const configStub = Ember.Service.extend({ - fileStorage: true -}); - -const sessionStub = Ember.Service.extend({ - isAuthenticated: false, - authorize(authorizer, block) { - if (this.get('isAuthenticated')) { - block('Authorization', 'Bearer token'); - } - } -}); - -const stubSuccessfulUpload = function (server, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [200, {'Content-Type': 'application/json'}, '"/content/images/test.png"']; - }, delay); -}; - -const stubFailedUpload = function (server, code, error, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [code, {'Content-Type': 'application/json'}, JSON.stringify({ - errors: [{ - errorType: error, - message: `Error: ${error}` - }] - })]; - }, delay); -}; - -describeComponent( - 'gh-image-upload', - 'Integration: Component: gh-image-uploader', - { - integration: true - }, - function() { - let server; - - beforeEach(function () { - this.register('service:config', configStub); - this.register('service:session', sessionStub); - this.inject.service('config', {as: 'configService'}); - this.inject.service('session', {as: 'sessionService'}); - this.set('update', function () {}); - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('renders', function() { - this.set('image', 'http://example.com/test.png'); - this.render(hbs`{{gh-image-uploader image=image}}`); - expect(this.$()).to.have.length(1); - }); - - it('defaults to upload form', function () { - this.render(hbs`{{gh-image-uploader image=image}}`); - expect(this.$('input[type="file"]').length).to.equal(1); - }); - - it('defaults to url form with no filestorage config', function () { - this.set('configService.fileStorage', false); - this.render(hbs`{{gh-image-uploader image=image}}`); - expect(this.$('input[type="file"]').length).to.equal(0); - expect(this.$('input[type="text"].url').length).to.equal(1); - }); - - it('can switch between form types', function () { - this.render(hbs`{{gh-image-uploader image=image}}`); - expect(this.$('input[type="file"]').length).to.equal(1); - expect(this.$('input[type="text"].url').length).to.equal(0); - - this.$('a.image-url').click(); - - expect(this.$('input[type="file"]').length, 'upload form is visible after switch to url form') - .to.equal(0); - expect(this.$('input[type="text"].url').length, 'url form is visible after switch to url form') - .to.equal(1); - - this.$('a.image-upload').click(); - - expect(this.$('input[type="file"]').length, 'upload form is visible after switch to upload form') - .to.equal(1); - expect(this.$('input[type="text"].url').length, 'url form is visible after switch to upload form') - .to.equal(0); - }); - - it('triggers formChanged action when switching between forms', function () { - let formChanged = sinon.spy(); - this.set('formChanged', formChanged); - - this.render(hbs`{{gh-image-uploader image=image formChanged=(action formChanged)}}`); - - this.$('a.image-url').click(); - this.$('a.image-upload').click(); - - expect(formChanged.calledTwice).to.be.true; - expect(formChanged.firstCall.args[0]).to.equal('url-input'); - expect(formChanged.secondCall.args[0]).to.equal('upload'); - }); - - describe('file upload form', function () { - it('renders form with supplied text', function () { - this.render(hbs`{{gh-image-uploader image=image text="text test"}}`); - expect(this.$('.description').text().trim()).to.equal('text test'); - }); - - it('generates request to correct endpoint', function (done) { - stubSuccessfulUpload(server); - - this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`); - this.$('input[type="file"]').trigger('change'); - - wait().then(() => { - expect(server.handledRequests.length).to.equal(1); - expect(server.handledRequests[0].url).to.equal('/ghost/api/v0.1/uploads/'); - expect(server.handledRequests[0].requestHeaders.Authorization).to.be.undefined; - done(); - }); - }); - - it('adds authentication headers to request', function (done) { - stubSuccessfulUpload(server); - - this.get('sessionService').set('isAuthenticated', true); - - this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`); - this.$('input[type="file"]').trigger('change'); - - wait().then(() => { - let [request] = server.handledRequests; - expect(request.requestHeaders.Authorization).to.equal('Bearer token'); - done(); - }); - }); - - it('handles drag over/leave', function () { - stubSuccessfulUpload(server); - - this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`); - - run(() => { - let dragover = Ember.$.Event('dragover', { - dataTransfer: { - files: [] - } - }); - this.$('.gh-image-uploader').trigger(dragover); - }); - - expect(this.$('.gh-image-uploader').hasClass('--drag-over'), 'has drag-over class').to.be.true; - - run(() => { - this.$('.gh-image-uploader').trigger('dragleave'); - }); - - expect(this.$('.gh-image-uploader').hasClass('--drag-over'), 'has drag-over class').to.be.false; - }); - }); - - describe('URL input form', function () { - beforeEach(function () { - this.set('configService.fileStorage', false); - }); - - it('displays save button by default', function () { - this.set('image', 'http://example.com/test.png'); - this.render(hbs`{{gh-image-uploader image=image text="text test"}}`); - expect(this.$('button').length).to.equal(1); - expect(this.$('input[type="text"]').val()).to.equal('http://example.com/test.png'); - }); - - it('can render without a save button', function () { - this.render(hbs`{{gh-image-uploader image=image saveButton=false text="text test"}}`); - expect(this.$('button').length).to.equal(0); - expect(this.$('.description').text().trim()).to.equal('text test'); - }); - - it('fires update action when save button clicked', function () { - let update = sinon.spy(); - this.set('update', update); - - this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`); - - this.$('input[type="text"]').val('saved url'); - this.$('input[type="text"]').change(); - this.$('button.btn-blue').click(); - - expect(update.calledOnce).to.be.true; - expect(update.firstCall.args[0]).to.equal('saved url'); - }); - - it('fires onInput action when typing URL', function () { - let onInput = sinon.spy(); - this.set('onInput', onInput); - - this.render(hbs`{{gh-image-uploader image=image onInput=(action onInput)}}`); - - this.$('input[type="text"]').val('input url'); - this.$('input[type="text"]').change(); - - expect(onInput.calledOnce).to.be.true; - expect(onInput.firstCall.args[0]).to.equal('input url'); - }); - - it('saves on enter key', function () { - let update = sinon.spy(); - this.set('update', update); - - this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`); - - this.$('input[type="text"]').val('saved url'); - this.$('input[type="text"]').change(); - this.$('input[type="text"]').trigger( - $.Event('keyup', {keyCode: keyCodes.enter, which: keyCodes.enter}) - ); - - expect(update.calledOnce).to.be.true; - expect(update.firstCall.args[0]).to.equal('saved url'); - }); - }); - } -); diff --git a/core/client/tests/integration/components/gh-image-uploader-with-preview-test.js b/core/client/tests/integration/components/gh-image-uploader-with-preview-test.js deleted file mode 100644 index 45d122e40f..0000000000 --- a/core/client/tests/integration/components/gh-image-uploader-with-preview-test.js +++ /dev/null @@ -1,48 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import sinon from 'sinon'; - -const {run} = Ember; - -describeComponent( - 'gh-image-uploader-with-preview', - 'Integration: Component: gh-image-uploader-with-preview', - { - integration: true - }, - function() { - it('renders image if provided', function() { - this.set('image', 'http://example.com/test.png'); - - this.render(hbs`{{gh-image-uploader-with-preview image=image}}`); - - expect(this.$('.gh-image-uploader.--with-image').length).to.equal(1); - expect(this.$('img').attr('src')).to.equal('http://example.com/test.png'); - }); - - it('renders upload form when no image provided', function () { - this.render(hbs`{{gh-image-uploader-with-preview image=image}}`); - - expect(this.$('input[type="file"]').length).to.equal(1); - }); - - it('triggers remove action when delete icon is clicked', function () { - let remove = sinon.spy(); - this.set('remove', remove); - this.set('image', 'http://example.com/test.png'); - - this.render(hbs`{{gh-image-uploader-with-preview image=image remove=remove}}`); - run(() => { - this.$('.icon-trash').click(); - }); - - expect(remove.calledOnce).to.be.true; - }); - } -); diff --git a/core/client/tests/integration/components/gh-navigation-test.js b/core/client/tests/integration/components/gh-navigation-test.js deleted file mode 100644 index a407cd8e25..0000000000 --- a/core/client/tests/integration/components/gh-navigation-test.js +++ /dev/null @@ -1,75 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeComponent, it } from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import NavItem from 'ghost/models/navigation-item'; - -const {run} = Ember; - -describeComponent( - 'gh-navigation', - 'Integration: Component: gh-navigation', - { - integration: true - }, - function () { - it('renders', function () { - this.render(hbs`{{#gh-navigation}}<div class="js-gh-blognav"><div class="gh-blognav-item"></div></div>{{/gh-navigation}}`); - expect(this.$('section.gh-view')).to.have.length(1); - expect(this.$('.ui-sortable')).to.have.length(1); - }); - - it('triggers reorder action', function () { - let navItems = []; - let expectedOldIndex = -1; - let expectedNewIndex = -1; - - navItems.pushObject(NavItem.create({label: 'First', url: '/first'})); - navItems.pushObject(NavItem.create({label: 'Second', url: '/second'})); - navItems.pushObject(NavItem.create({label: 'Third', url: '/third'})); - navItems.pushObject(NavItem.create({label: '', url: '', last: true})); - this.set('navigationItems', navItems); - this.set('blogUrl', 'http://localhost:2368'); - - this.on('moveItem', (oldIndex, newIndex) => { - expect(oldIndex).to.equal(expectedOldIndex); - expect(newIndex).to.equal(expectedNewIndex); - }); - - run(() => { - this.render(hbs ` - {{#gh-navigation moveItem="moveItem"}} - <form id="settings-navigation" class="gh-blognav js-gh-blognav" novalidate="novalidate"> - {{#each navigationItems as |navItem|}} - {{gh-navitem navItem=navItem baseUrl=blogUrl addItem="addItem" deleteItem="deleteItem" updateUrl="updateUrl"}} - {{/each}} - </form> - {{/gh-navigation}}`); - }); - - // check it renders the nav item rows - expect(this.$('.gh-blognav-item')).to.have.length(4); - - // move second item up one - expectedOldIndex = 1; - expectedNewIndex = 0; - run(() => { - Ember.$(this.$('.gh-blognav-item')[1]).simulateDragSortable({ - move: -1, - handle: '.gh-blognav-grab' - }); - }); - - // move second item down one - expectedOldIndex = 1; - expectedNewIndex = 2; - run(() => { - Ember.$(this.$('.gh-blognav-item')[1]).simulateDragSortable({ - move: 1, - handle: '.gh-blognav-grab' - }); - }); - }); - } -); diff --git a/core/client/tests/integration/components/gh-navitem-test.js b/core/client/tests/integration/components/gh-navitem-test.js deleted file mode 100644 index 1ae7b09f49..0000000000 --- a/core/client/tests/integration/components/gh-navitem-test.js +++ /dev/null @@ -1,114 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeComponent, it } from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import NavItem from 'ghost/models/navigation-item'; - -const {run} = Ember; - -describeComponent( - 'gh-navitem', - 'Integration: Component: gh-navitem', - { - integration: true - }, - function () { - beforeEach(function () { - this.set('baseUrl', 'http://localhost:2368'); - }); - - it('renders', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url'})); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl}}`); - let $item = this.$('.gh-blognav-item'); - - expect($item.find('.gh-blognav-grab').length).to.equal(1); - expect($item.find('.gh-blognav-label').length).to.equal(1); - expect($item.find('.gh-blognav-url').length).to.equal(1); - expect($item.find('.gh-blognav-delete').length).to.equal(1); - - // doesn't show any errors - expect($item.hasClass('gh-blognav-item--error')).to.be.false; - expect($item.find('.error').length).to.equal(0); - expect($item.find('.response:visible').length).to.equal(0); - }); - - it('doesn\'t show drag handle for new items', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url', isNew: true})); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl}}`); - let $item = this.$('.gh-blognav-item'); - - expect($item.find('.gh-blognav-grab').length).to.equal(0); - }); - - it('shows add button for new items', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url', isNew: true})); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl}}`); - let $item = this.$('.gh-blognav-item'); - - expect($item.find('.gh-blognav-add').length).to.equal(1); - expect($item.find('.gh-blognav-delete').length).to.equal(0); - }); - - it('triggers delete action', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url'})); - - let deleteActionCallCount = 0; - this.on('deleteItem', (navItem) => { - expect(navItem).to.equal(this.get('navItem')); - deleteActionCallCount++; - }); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl deleteItem="deleteItem"}}`); - this.$('.gh-blognav-delete').trigger('click'); - - expect(deleteActionCallCount).to.equal(1); - }); - - it('triggers add action', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url', isNew: true})); - - let addActionCallCount = 0; - this.on('add', () => { - addActionCallCount++; - }); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl addItem="add"}}`); - this.$('.gh-blognav-add').trigger('click'); - - expect(addActionCallCount).to.equal(1); - }); - - it('triggers update action', function () { - this.set('navItem', NavItem.create({label: 'Test', url: '/url'})); - - let updateActionCallCount = 0; - this.on('update', () => { - updateActionCallCount++; - }); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl updateUrl="update"}}`); - this.$('.gh-blognav-url input').trigger('blur'); - - expect(updateActionCallCount).to.equal(1); - }); - - it('displays inline errors', function () { - this.set('navItem', NavItem.create({label: '', url: ''})); - this.get('navItem').validate(); - - this.render(hbs`{{gh-navitem navItem=navItem baseUrl=baseUrl}}`); - let $item = this.$('.gh-blognav-item'); - - expect($item.hasClass('gh-blognav-item--error')).to.be.true; - expect($item.find('.gh-blognav-label').hasClass('error')).to.be.true; - expect($item.find('.gh-blognav-label .response').text().trim()).to.equal('You must specify a label'); - expect($item.find('.gh-blognav-url').hasClass('error')).to.be.true; - expect($item.find('.gh-blognav-url .response').text().trim()).to.equal('You must specify a URL or relative path'); - }); - } -); diff --git a/core/client/tests/integration/components/gh-navitem-url-input-test.js b/core/client/tests/integration/components/gh-navitem-url-input-test.js deleted file mode 100644 index 81a51a72e4..0000000000 --- a/core/client/tests/integration/components/gh-navitem-url-input-test.js +++ /dev/null @@ -1,481 +0,0 @@ -/* jshint scripturl:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -const {run} = Ember; - -// we want baseUrl to match the running domain so relative URLs are -// handled as expected (browser auto-sets the domain when using a.href) -let currentUrl = `${window.location.protocol}//${window.location.host}/`; - -describeComponent( - 'gh-navitem-url-input', - 'Integration: Component: gh-navitem-url-input', { - integration: true - }, - function () { - beforeEach(function () { - // set defaults - this.set('baseUrl', currentUrl); - this.set('url', ''); - this.set('isNew', false); - this.on('clearErrors', function () { - return null; - }); - }); - - it('renders correctly with blank url', function () { - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expect($input).to.have.length(1); - expect($input.hasClass('gh-input')).to.be.true; - expect($input.val()).to.equal(currentUrl); - }); - - it('renders correctly with relative urls', function () { - this.set('url', '/about'); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expect($input.val()).to.equal(`${currentUrl}about`); - - this.set('url', '/about#contact'); - expect($input.val()).to.equal(`${currentUrl}about#contact`); - }); - - it('renders correctly with absolute urls', function () { - this.set('url', 'https://example.com:2368/#test'); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expect($input.val()).to.equal('https://example.com:2368/#test'); - - this.set('url', 'mailto:test@example.com'); - expect($input.val()).to.equal('mailto:test@example.com'); - - this.set('url', 'tel:01234-5678-90'); - expect($input.val()).to.equal('tel:01234-5678-90'); - - this.set('url', '//protocol-less-url.com'); - expect($input.val()).to.equal('//protocol-less-url.com'); - - this.set('url', '#anchor'); - expect($input.val()).to.equal('#anchor'); - }); - - it('deletes base URL on backspace', function () { - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expect($input.val()).to.equal(currentUrl); - run(() => { - // TODO: why is ember's keyEvent helper not available here? - let e = Ember.$.Event('keydown'); - e.keyCode = 8; - $input.trigger(e); - }); - expect($input.val()).to.equal(''); - }); - - it('deletes base URL on delete', function () { - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expect($input.val()).to.equal(currentUrl); - run(() => { - // TODO: why is ember's keyEvent helper not available here? - let e = Ember.$.Event('keydown'); - e.keyCode = 46; - $input.trigger(e); - }); - expect($input.val()).to.equal(''); - }); - - it('adds base url to relative urls on blur', function () { - this.on('updateUrl', () => { - return null; - }); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - run(() => { - $input.val('/about').trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - - expect($input.val()).to.equal(`${currentUrl}about`); - }); - - it('adds "mailto:" to email addresses on blur', function () { - this.on('updateUrl', () => { - return null; - }); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - run(() => { - $input.val('test@example.com').trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - - expect($input.val()).to.equal('mailto:test@example.com'); - - // ensure we don't double-up on the mailto: - run(() => { - $input.trigger('blur'); - }); - expect($input.val()).to.equal('mailto:test@example.com'); - }); - - it('doesn\'t add base url to invalid urls on blur', function () { - this.on('updateUrl', () => { - return null; - }); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let changeValue = function (value) { - run(() => { - $input.val(value).trigger('input').trigger('blur'); - }); - }; - - changeValue('with spaces'); - expect($input.val()).to.equal('with spaces'); - - changeValue('/with spaces'); - expect($input.val()).to.equal('/with spaces'); - }); - - it('doesn\'t mangle invalid urls on blur', function () { - this.on('updateUrl', () => { - return null; - }); - this.render(hbs` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - run(() => { - $input.val(`${currentUrl} /test`).trigger('input').trigger('blur'); - }); - - expect($input.val()).to.equal(`${currentUrl} /test`); - }); - - it('triggers "change" action on blur', function () { - let changeActionCallCount = 0; - this.on('updateUrl', () => { - changeActionCallCount++; - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - $input.trigger('blur'); - - expect(changeActionCallCount).to.equal(1); - }); - - it('triggers "change" action on enter', function () { - let changeActionCallCount = 0; - this.on('updateUrl', () => { - changeActionCallCount++; - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - run(() => { - // TODO: why is ember's keyEvent helper not available here? - let e = Ember.$.Event('keypress'); - e.keyCode = 13; - $input.trigger(e); - }); - - expect(changeActionCallCount).to.equal(1); - }); - - it('triggers "change" action on CMD-S', function () { - let changeActionCallCount = 0; - this.on('updateUrl', () => { - changeActionCallCount++; - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - run(() => { - // TODO: why is ember's keyEvent helper not available here? - let e = Ember.$.Event('keydown'); - e.keyCode = 83; - e.metaKey = true; - $input.trigger(e); - }); - - expect(changeActionCallCount).to.equal(1); - }); - - it('sends absolute urls straight through to change action', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = url; - run(() => { - $input.val(url).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('http://example.com'); - testUrl('http://example.com/'); - testUrl('https://example.com'); - testUrl('//example.com'); - testUrl('//localhost:1234'); - testUrl('#anchor'); - testUrl('mailto:test@example.com'); - testUrl('tel:12345-567890'); - testUrl('javascript:alert("testing");'); - }); - - it('strips base url from relative urls before sending to change action', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = `/${url}`; - run(() => { - $input.val(`${currentUrl}${url}`).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('about/'); - testUrl('about#contact'); - testUrl('test/nested/'); - }); - - it('handles links to subdomains of blog domain', function () { - let expectedUrl = ''; - - this.set('baseUrl', 'http://example.com/'); - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - expectedUrl = 'http://test.example.com/'; - run(() => { - $input.val(expectedUrl).trigger('input').trigger('blur'); - }); - expect($input.val()).to.equal(expectedUrl); - }); - - it('adds trailing slash to relative URL', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = `/${url}/`; - run(() => { - $input.val(`${currentUrl}${url}`).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('about'); - testUrl('test/nested'); - }); - - it('does not add trailing slash on relative URL with [.?#]', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = `/${url}`; - run(() => { - $input.val(`${currentUrl}${url}`).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('about#contact'); - testUrl('test/nested.svg'); - testUrl('test?gho=sties'); - testUrl('test/nested?sli=mer'); - }); - - it('does not add trailing slash on non-relative URLs', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = `/${url}`; - run(() => { - $input.val(`${currentUrl}${url}`).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('http://woo.ff/test'); - testUrl('http://me.ow:2342/nested/test'); - testUrl('https://wro.om/car#race'); - testUrl('https://kabo.om/explosion?really=now'); - }); - - describe('with sub-folder baseUrl', function () { - beforeEach(function () { - this.set('baseUrl', `${currentUrl}blog/`); - }); - - it('handles URLs relative to base url', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = url; - run(() => { - $input.val(`${currentUrl}blog${url}`).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl('/about/'); - testUrl('/about#contact'); - testUrl('/test/nested/'); - }); - - it('handles URLs relative to base host', function () { - let expectedUrl = ''; - - this.on('updateUrl', (url) => { - expect(url).to.equal(expectedUrl); - }); - - this.render(hbs ` - {{gh-navitem-url-input baseUrl=baseUrl url=url isNew=isNew change="updateUrl" clearErrors=(action "clearErrors")}} - `); - let $input = this.$('input'); - - let testUrl = (url) => { - expectedUrl = url; - run(() => { - $input.val(url).trigger('input'); - }); - run(() => { - $input.trigger('blur'); - }); - }; - - testUrl(`http://${window.location.host}`); - testUrl(`https://${window.location.host}`); - testUrl(`http://${window.location.host}/`); - testUrl(`https://${window.location.host}/`); - testUrl(`http://${window.location.host}/test`); - testUrl(`https://${window.location.host}/test`); - testUrl(`http://${window.location.host}/#test`); - testUrl(`https://${window.location.host}/#test`); - testUrl(`http://${window.location.host}/another/folder`); - testUrl(`https://${window.location.host}/another/folder`); - }); - }); - } -); diff --git a/core/client/tests/integration/components/gh-notification-test.js b/core/client/tests/integration/components/gh-notification-test.js deleted file mode 100644 index 37e59d7d0f..0000000000 --- a/core/client/tests/integration/components/gh-notification-test.js +++ /dev/null @@ -1,44 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; - -describeComponent( - 'gh-notification', - 'Integration: Component: gh-notification', - { - integration: true - }, - function () { - it('renders', function () { - this.set('message', {message: 'Test message', type: 'success'}); - - this.render(hbs`{{gh-notification message=message}}`); - - expect(this.$('article.gh-notification')).to.have.length(1); - let $notification = this.$('.gh-notification'); - - expect($notification.hasClass('gh-notification-passive')).to.be.true; - expect($notification.text()).to.match(/Test message/); - }); - - it('maps message types to CSS classes', function () { - this.set('message', {message: 'Test message', type: 'success'}); - - this.render(hbs`{{gh-notification message=message}}`); - let $notification = this.$('.gh-notification'); - - this.set('message.type', 'success'); - expect($notification.hasClass('gh-notification-green'), 'success class isn\'t green').to.be.true; - - this.set('message.type', 'error'); - expect($notification.hasClass('gh-notification-red'), 'success class isn\'t red').to.be.true; - - this.set('message.type', 'warn'); - expect($notification.hasClass('gh-notification-yellow'), 'success class isn\'t yellow').to.be.true; - }); - } -); diff --git a/core/client/tests/integration/components/gh-notifications-test.js b/core/client/tests/integration/components/gh-notifications-test.js deleted file mode 100644 index 25afe951eb..0000000000 --- a/core/client/tests/integration/components/gh-notifications-test.js +++ /dev/null @@ -1,44 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -const {Service, run} = Ember; -const emberA = Ember.A; - -let notificationsStub = Service.extend({ - notifications: emberA() -}); - -describeComponent( - 'gh-notifications', - 'Integration: Component: gh-notifications', - { - integration: true - }, - function () { - beforeEach(function () { - this.register('service:notifications', notificationsStub); - this.inject.service('notifications', {as: 'notifications'}); - - this.set('notifications.notifications', [ - {message: 'First', type: 'error'}, - {message: 'Second', type: 'warn'} - ]); - }); - - it('renders', function () { - this.render(hbs`{{gh-notifications}}`); - expect(this.$('.gh-notifications').length).to.equal(1); - - expect(this.$('.gh-notifications').children().length).to.equal(2); - - this.set('notifications.notifications', emberA()); - expect(this.$('.gh-notifications').children().length).to.equal(0); - }); - } -); diff --git a/core/client/tests/integration/components/gh-profile-image-test.js b/core/client/tests/integration/components/gh-profile-image-test.js deleted file mode 100644 index 0f919b774b..0000000000 --- a/core/client/tests/integration/components/gh-profile-image-test.js +++ /dev/null @@ -1,152 +0,0 @@ -/* jshint expr:true */ -/* global md5 */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; - -const {run} = Ember; - -let pathsStub = Ember.Service.extend({ - url: { - api() { - return ''; - }, - asset(src) { - return src; - } - } -}); - -const stubKnownGravatar = function (server) { - server.get('http://www.gravatar.com/avatar/:md5', function () { - return [200, {'Content-Type': 'image/png'}, '']; - }); -}; - -const stubUnknownGravatar = function (server) { - server.get('http://www.gravatar.com/avatar/:md5', function () { - return [404, {}, '']; - }); -}; - -describeComponent( - 'gh-profile-image', - 'Integration: Component: gh-profile-image', - { - integration: true - }, - function () { - let server; - - beforeEach(function () { - this.register('service:ghost-paths', pathsStub); - this.inject.service('ghost-paths', {as: 'ghost-paths'}); - - server = new Pretender(); - stubKnownGravatar(server); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('renders', function () { - this.set('email', ''); - - this.render(hbs` - {{gh-profile-image email=email}} - `); - - expect(this.$()).to.have.length(1); - }); - - it('renders and tears down ok with fileStorage:false', function () { - this.set('fileStorage', false); - - this.render(hbs` - {{gh-profile-image fileStorage=fileStorage}} - `); - - expect(this.$()).to.have.length(1); - expect(this.$('input')).to.have.length(0); - }), - - it('renders default image if no email supplied', function () { - this.set('email', null); - - this.render(hbs` - {{gh-profile-image email=email size=100 debounce=50}} - `); - - expect(this.$('.gravatar-img').attr('style'), 'gravatar image style') - .to.be.blank; - }); - - it('renders the gravatar if valid email supplied', function (done) { - let email = 'test@example.com'; - let expectedUrl = `//www.gravatar.com/avatar/${md5(email)}?s=100&d=404`; - - this.set('email', email); - - this.render(hbs` - {{gh-profile-image email=email size=100 debounce=50}} - `); - - // wait for the ajax request to complete - wait().then(() => { - expect(this.$('.gravatar-img').attr('style'), 'gravatar image style') - .to.equal(`background-image: url(${expectedUrl})`); - done(); - }); - }); - - it('doesn\'t add background url if gravatar image doesn\'t exist', function (done) { - stubUnknownGravatar(server); - - this.render(hbs` - {{gh-profile-image email="test@example.com" size=100 debounce=50}} - `); - - wait().then(() => { - expect(this.$('.gravatar-img').attr('style'), 'gravatar image style') - .to.be.blank; - done(); - }); - }); - - it('throttles gravatar loading as email is changed', function (done) { - let email = 'test@example.com'; - let expectedUrl = `//www.gravatar.com/avatar/${md5(email)}?s=100&d=404`; - - this.set('email', 'test'); - - this.render(hbs` - {{gh-profile-image email=email size=100 debounce=300}} - `); - - run(() => { - this.set('email', email); - }); - - expect(this.$('.gravatar-img').attr('style'), '.gravatar-img background not immediately changed on email change') - .to.be.blank; - - run.later(this, function () { - expect(this.$('.gravatar-img').attr('style'), '.gravatar-img background still not changed before debounce timeout') - .to.be.blank; - }, 250); - - run.later(this, function () { - expect(this.$('.gravatar-img').attr('style'), '.gravatar-img background changed after debounce timeout') - .to.equal(`background-image: url(${expectedUrl})`); - done(); - }, 400); - }); - } -); diff --git a/core/client/tests/integration/components/gh-search-input-test.js b/core/client/tests/integration/components/gh-search-input-test.js deleted file mode 100644 index 14051b0a87..0000000000 --- a/core/client/tests/integration/components/gh-search-input-test.js +++ /dev/null @@ -1,26 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -const {run} = Ember; - -describeComponent( - 'gh-search-input', - 'Integration: Component: gh-search-input', - { - integration: true - }, - function () { - it('renders', function () { - // renders the component on the page - this.render(hbs`{{gh-search-input}}`); - - expect(this.$('.ember-power-select-search input')).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/gh-subscribers-table-test.js b/core/client/tests/integration/components/gh-subscribers-table-test.js deleted file mode 100644 index 5a976d8f5b..0000000000 --- a/core/client/tests/integration/components/gh-subscribers-table-test.js +++ /dev/null @@ -1,26 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Table from 'ember-light-table'; - -describeComponent( - 'gh-subscribers-table', - 'Integration: Component: gh-subscribers-table', - { - integration: true - }, - function() { - it('renders', function() { - this.set('table', new Table([], [])); - this.set('sortByColumn', function () {}); - this.set('delete', function () {}); - - this.render(hbs`{{gh-subscribers-table table=table sortByColumn=(action sortByColumn) delete=(action delete)}}`); - expect(this.$()).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/gh-tag-settings-form-test.js b/core/client/tests/integration/components/gh-tag-settings-form-test.js deleted file mode 100644 index 97ab8c9ddc..0000000000 --- a/core/client/tests/integration/components/gh-tag-settings-form-test.js +++ /dev/null @@ -1,322 +0,0 @@ -/* jshint expr:true */ -/* jscs:disable requireTemplateStringsForConcatenation */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import DS from 'ember-data'; - -const {run} = Ember; - -let configStub = Ember.Service.extend({ - blogUrl: 'http://localhost:2368' -}); - -let mediaQueriesStub = Ember.Service.extend({ - maxWidth600: false -}); - -describeComponent( - 'gh-tag-settings-form', - 'Integration: Component: gh-tag-settings-form', - { - integration: true - }, - function () { - beforeEach(function () { - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - let tag = Ember.Object.create({ - id: 1, - name: 'Test', - slug: 'test', - description: 'Description.', - metaTitle: 'Meta Title', - metaDescription: 'Meta description', - errors: DS.Errors.create(), - hasValidated: [] - }); - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - - this.set('tag', tag); - this.set('actions.setProperty', function (property, value) { - // this should be overridden if a call is expected - console.error(`setProperty called '${property}: ${value}'`); - }); - - this.register('service:config', configStub); - this.inject.service('config', {as: 'config'}); - - this.register('service:media-queries', mediaQueriesStub); - this.inject.service('media-queries', {as: 'mediaQueries'}); - }); - - it('renders', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - expect(this.$()).to.have.length(1); - }); - - it('has the correct title', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - expect(this.$('.tag-settings-pane h4').text(), 'existing tag title').to.equal('Tag Settings'); - - this.set('tag.isNew', true); - expect(this.$('.tag-settings-pane h4').text(), 'new tag title').to.equal('New Tag'); - }); - - it('renders main settings', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - expect(this.$('.gh-image-uploader').length, 'displays image uploader').to.equal(1); - expect(this.$('input[name="name"]').val(), 'name field value').to.equal('Test'); - expect(this.$('input[name="slug"]').val(), 'slug field value').to.equal('test'); - expect(this.$('textarea[name="description"]').val(), 'description field value').to.equal('Description.'); - expect(this.$('input[name="metaTitle"]').val(), 'metaTitle field value').to.equal('Meta Title'); - expect(this.$('textarea[name="metaDescription"]').val(), 'metaDescription field value').to.equal('Meta description'); - }); - - it('can switch between main/meta settings', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - expect(this.$('.tag-settings-pane').hasClass('settings-menu-pane-in'), 'main settings are displayed by default').to.be.true; - expect(this.$('.tag-meta-settings-pane').hasClass('settings-menu-pane-out-right'), 'meta settings are hidden by default').to.be.true; - - run(() => { - this.$('.meta-data-button').click(); - }); - - expect(this.$('.tag-settings-pane').hasClass('settings-menu-pane-out-left'), 'main settings are hidden after clicking Meta Data button').to.be.true; - expect(this.$('.tag-meta-settings-pane').hasClass('settings-menu-pane-in'), 'meta settings are displayed after clicking Meta Data button').to.be.true; - - run(() => { - this.$('.back').click(); - }); - - expect(this.$('.tag-settings-pane').hasClass('settings-menu-pane-in'), 'main settings are displayed after clicking "back"').to.be.true; - expect(this.$('.tag-meta-settings-pane').hasClass('settings-menu-pane-out-right'), 'meta settings are hidden after clicking "back"').to.be.true; - }); - - it('has one-way binding for properties', function () { - this.set('actions.setProperty', function () { - // noop - }); - - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - run(() => { - this.$('input[name="name"]').val('New name'); - this.$('input[name="slug"]').val('new-slug'); - this.$('textarea[name="description"]').val('New description'); - this.$('input[name="metaTitle"]').val('New metaTitle'); - this.$('textarea[name="metaDescription"]').val('New metaDescription'); - }); - - expect(this.get('tag.name'), 'tag name').to.equal('Test'); - expect(this.get('tag.slug'), 'tag slug').to.equal('test'); - expect(this.get('tag.description'), 'tag description').to.equal('Description.'); - expect(this.get('tag.metaTitle'), 'tag metaTitle').to.equal('Meta Title'); - expect(this.get('tag.metaDescription'), 'tag metaDescription').to.equal('Meta description'); - }); - - it('triggers setProperty action on blur of all fields', function () { - let expectedProperty = ''; - let expectedValue = ''; - - this.set('actions.setProperty', function (property, value) { - expect(property, 'property').to.equal(expectedProperty); - expect(value, 'value').to.equal(expectedValue); - }); - - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - expectedProperty = 'name'; - expectedValue = 'new-slug'; - run(() => { - this.$('input[name="name"]').val('New name'); - }); - - expectedProperty = 'url'; - expectedValue = 'new-slug'; - run(() => { - this.$('input[name="slug"]').val('new-slug'); - }); - - expectedProperty = 'description'; - expectedValue = 'New description'; - run(() => { - this.$('textarea[name="description"]').val('New description'); - }); - - expectedProperty = 'metaTitle'; - expectedValue = 'New metaTitle'; - run(() => { - this.$('input[name="metaTitle"]').val('New metaTitle'); - }); - - expectedProperty = 'metaDescription'; - expectedValue = 'New metaDescription'; - run(() => { - this.$('textarea[name="metaDescription"]').val('New metaDescription'); - }); - }); - - it('displays error messages for validated fields', function () { - let errors = this.get('tag.errors'); - let hasValidated = this.get('tag.hasValidated'); - - errors.add('name', 'must be present'); - hasValidated.push('name'); - - errors.add('slug', 'must be present'); - hasValidated.push('slug'); - - errors.add('description', 'is too long'); - hasValidated.push('description'); - - errors.add('metaTitle', 'is too long'); - hasValidated.push('metaTitle'); - - errors.add('metaDescription', 'is too long'); - hasValidated.push('metaDescription'); - - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - let nameFormGroup = this.$('input[name="name"]').closest('.form-group'); - expect(nameFormGroup.hasClass('error'), 'name form group has error state').to.be.true; - expect(nameFormGroup.find('.response').length, 'name form group has error message').to.equal(1); - - let slugFormGroup = this.$('input[name="slug"]').closest('.form-group'); - expect(slugFormGroup.hasClass('error'), 'slug form group has error state').to.be.true; - expect(slugFormGroup.find('.response').length, 'slug form group has error message').to.equal(1); - - let descriptionFormGroup = this.$('textarea[name="description"]').closest('.form-group'); - expect(descriptionFormGroup.hasClass('error'), 'description form group has error state').to.be.true; - - let metaTitleFormGroup = this.$('input[name="metaTitle"]').closest('.form-group'); - expect(metaTitleFormGroup.hasClass('error'), 'metaTitle form group has error state').to.be.true; - expect(metaTitleFormGroup.find('.response').length, 'metaTitle form group has error message').to.equal(1); - - let metaDescriptionFormGroup = this.$('textarea[name="metaDescription"]').closest('.form-group'); - expect(metaDescriptionFormGroup.hasClass('error'), 'metaDescription form group has error state').to.be.true; - expect(metaDescriptionFormGroup.find('.response').length, 'metaDescription form group has error message').to.equal(1); - }); - - it('displays char count for text fields', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - let descriptionFormGroup = this.$('textarea[name="description"]').closest('.form-group'); - expect(descriptionFormGroup.find('.word-count').text(), 'description char count').to.equal('12'); - - let metaDescriptionFormGroup = this.$('textarea[name="metaDescription"]').closest('.form-group'); - expect(metaDescriptionFormGroup.find('.word-count').text(), 'description char count').to.equal('16'); - }); - - it('renders SEO title preview', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - expect(this.$('.seo-preview-title').text(), 'displays meta title if present').to.equal('Meta Title'); - - run(() => { - this.set('tag.metaTitle', ''); - }); - expect(this.$('.seo-preview-title').text(), 'falls back to tag name without metaTitle').to.equal('Test'); - - run(() => { - this.set('tag.name', (new Array(151).join('x'))); - }); - let expectedLength = 70 + '…'.length; - expect(this.$('.seo-preview-title').text().length, 'cuts title to max 70 chars').to.equal(expectedLength); - }); - - it('renders SEO URL preview', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - expect(this.$('.seo-preview-link').text(), 'adds url and tag prefix').to.equal('http://localhost:2368/tag/test/'); - - run(() => { - this.set('tag.slug', (new Array(151).join('x'))); - }); - let expectedLength = 70 + '…'.length; - expect(this.$('.seo-preview-link').text().length, 'cuts slug to max 70 chars').to.equal(expectedLength); - }); - - it('renders SEO description preview', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - expect(this.$('.seo-preview-description').text(), 'displays meta description if present').to.equal('Meta description'); - - run(() => { - this.set('tag.metaDescription', ''); - }); - expect(this.$('.seo-preview-description').text(), 'falls back to tag description without metaDescription').to.equal('Description.'); - - run(() => { - this.set('tag.description', (new Array(200).join('x'))); - }); - let expectedLength = 156 + '…'.length; - expect(this.$('.seo-preview-description').text().length, 'cuts description to max 156 chars').to.equal(expectedLength); - }); - - it('resets if a new tag is received', function () { - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - run(() => { - this.$('.meta-data-button').click(); - }); - expect(this.$('.tag-meta-settings-pane').hasClass('settings-menu-pane-in'), 'meta data pane is shown').to.be.true; - - run(() => { - this.set('tag', Ember.Object.create({id: '2'})); - }); - expect(this.$('.tag-settings-pane').hasClass('settings-menu-pane-in'), 'resets to main settings').to.be.true; - }); - - it('triggers delete tag modal on delete click', function (done) { - // TODO: will time out if this isn't hit, there's probably a better - // way of testing this - this.set('actions.openModal', () => { - done(); - }); - - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty') showDeleteTagModal=(action 'openModal')}} - `); - - run(() => { - this.$('.tag-delete-button').click(); - }); - }); - - it('shows settings.tags arrow link on mobile', function () { - this.set('mediaQueries.maxWidth600', true); - - this.render(hbs` - {{gh-tag-settings-form tag=tag setProperty=(action 'setProperty')}} - `); - - expect(this.$('.tag-settings-pane .settings-menu-header .settings-menu-header-action').length, 'settings.tags link is shown').to.equal(1); - }); - } -); diff --git a/core/client/tests/integration/components/gh-tags-management-container-test.js b/core/client/tests/integration/components/gh-tags-management-container-test.js deleted file mode 100644 index 68e4e05e03..0000000000 --- a/core/client/tests/integration/components/gh-tags-management-container-test.js +++ /dev/null @@ -1,33 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; - -describeComponent( - 'gh-tags-management-container', - 'Integration: Component: gh-tags-management-container', - { - integration: true - }, - function () { - it('renders', function () { - this.set('tags', []); - this.set('selectedTag', null); - this.on('enteredMobile', function () { - // noop - }); - this.on('leftMobile', function () { - // noop - }); - - this.render(hbs` - {{#gh-tags-management-container tags=tags selectedTag=selectedTag enteredMobile="enteredMobile" leftMobile="leftMobile"}}{{/gh-tags-management-container}} - `); - expect(this.$()).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/gh-validation-status-container-test.js b/core/client/tests/integration/components/gh-validation-status-container-test.js deleted file mode 100644 index 1507020172..0000000000 --- a/core/client/tests/integration/components/gh-validation-status-container-test.js +++ /dev/null @@ -1,72 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import DS from 'ember-data'; - -describeComponent( - 'gh-validation-status-container', - 'Integration: Component: gh-validation-status-container', - { - integration: true - }, - function () { - beforeEach(function () { - let testObject = new Ember.Object(); - testObject.set('name', 'Test'); - testObject.set('hasValidated', []); - testObject.set('errors', DS.Errors.create()); - - this.set('testObject', testObject); - }); - - it('has no success/error class by default', function () { - this.render(hbs` - {{#gh-validation-status-container class="gh-test" property="name" errors=testObject.errors hasValidated=testObject.hasValidated}} - {{/gh-validation-status-container}} - `); - expect(this.$('.gh-test')).to.have.length(1); - expect(this.$('.gh-test').hasClass('success')).to.be.false; - expect(this.$('.gh-test').hasClass('error')).to.be.false; - }); - - it('has success class when valid', function () { - this.get('testObject.hasValidated').push('name'); - - this.render(hbs` - {{#gh-validation-status-container class="gh-test" property="name" errors=testObject.errors hasValidated=testObject.hasValidated}} - {{/gh-validation-status-container}} - `); - expect(this.$('.gh-test')).to.have.length(1); - expect(this.$('.gh-test').hasClass('success')).to.be.true; - expect(this.$('.gh-test').hasClass('error')).to.be.false; - }); - - it('has error class when invalid', function () { - this.get('testObject.hasValidated').push('name'); - this.get('testObject.errors').add('name', 'has error'); - - this.render(hbs` - {{#gh-validation-status-container class="gh-test" property="name" errors=testObject.errors hasValidated=testObject.hasValidated}} - {{/gh-validation-status-container}} - `); - expect(this.$('.gh-test')).to.have.length(1); - expect(this.$('.gh-test').hasClass('success')).to.be.false; - expect(this.$('.gh-test').hasClass('error')).to.be.true; - }); - - it('still renders if hasValidated is undefined', function () { - this.set('testObject.hasValidated', undefined); - - this.render(hbs` - {{#gh-validation-status-container class="gh-test" property="name" errors=testObject.errors hasValidated=testObject.hasValidated}} - {{/gh-validation-status-container}} - `); - expect(this.$('.gh-test')).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/modals/delete-subscriber-test.js b/core/client/tests/integration/components/modals/delete-subscriber-test.js deleted file mode 100644 index f34ad9d337..0000000000 --- a/core/client/tests/integration/components/modals/delete-subscriber-test.js +++ /dev/null @@ -1,30 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; - -describeComponent( - 'modals/delete-subscriber', - 'Integration: Component: modals/delete-subscriber', - { - integration: true - }, - function() { - it('renders', function() { - // Set any properties with this.set('myProperty', 'value'); - // Handle any actions with this.on('myAction', function(val) { ... }); - // Template block usage: - // this.render(hbs` - // {{#modals/delete-subscriber}} - // template content - // {{/modals/delete-subscriber}} - // `); - - this.render(hbs`{{modals/delete-subscriber}}`); - expect(this.$()).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/modals/import-subscribers-test.js b/core/client/tests/integration/components/modals/import-subscribers-test.js deleted file mode 100644 index 52dcf93b34..0000000000 --- a/core/client/tests/integration/components/modals/import-subscribers-test.js +++ /dev/null @@ -1,30 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; - -describeComponent( - 'modals/import-subscribers', - 'Integration: Component: modals/import-subscribers', - { - integration: true - }, - function() { - it('renders', function() { - // Set any properties with this.set('myProperty', 'value'); - // Handle any actions with this.on('myAction', function(val) { ... }); - // Template block usage: - // this.render(hbs` - // {{#modals/import-subscribers}} - // template content - // {{/modals/import-subscribers}} - // `); - - this.render(hbs`{{modals/import-subscribers}}`); - expect(this.$()).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/modals/new-subscriber-test.js b/core/client/tests/integration/components/modals/new-subscriber-test.js deleted file mode 100644 index be34ec9a98..0000000000 --- a/core/client/tests/integration/components/modals/new-subscriber-test.js +++ /dev/null @@ -1,30 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; - -describeComponent( - 'modals/new-subscriber', - 'Integration: Component: modals/new-subscriber', - { - integration: true - }, - function() { - it('renders', function() { - // Set any properties with this.set('myProperty', 'value'); - // Handle any actions with this.on('myAction', function(val) { ... }); - // Template block usage: - // this.render(hbs` - // {{#modals/new-subscriber}} - // template content - // {{/modals/new-subscriber}} - // `); - - this.render(hbs`{{modals/new-subscriber}}`); - expect(this.$()).to.have.length(1); - }); - } -); diff --git a/core/client/tests/integration/components/transfer-owner-test.js b/core/client/tests/integration/components/transfer-owner-test.js deleted file mode 100644 index 77ccad7960..0000000000 --- a/core/client/tests/integration/components/transfer-owner-test.js +++ /dev/null @@ -1,39 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import hbs from 'htmlbars-inline-precompile'; -import Ember from 'ember'; -import sinon from 'sinon'; - -const {RSVP, run} = Ember; - -describeComponent( - 'transfer-owner', - 'Integration: Component: modals/transfer-owner', - { - integration: true - }, - function() { - it('triggers confirm action', function() { - let confirm = sinon.stub(); - let closeModal = sinon.spy(); - - confirm.returns(RSVP.resolve({})); - - this.on('confirm', confirm); - this.on('closeModal', closeModal); - - this.render(hbs`{{modals/transfer-owner confirm=(action 'confirm') closeModal=(action 'closeModal')}}`); - - run(() => { - this.$('.btn.btn-red').click(); - }); - - expect(confirm.calledOnce, 'confirm called').to.be.true; - expect(closeModal.calledOnce, 'closeModal called').to.be.true; - }); - } -); diff --git a/core/client/tests/integration/services/ajax-test.js b/core/client/tests/integration/services/ajax-test.js deleted file mode 100644 index 23e45488df..0000000000 --- a/core/client/tests/integration/services/ajax-test.js +++ /dev/null @@ -1,131 +0,0 @@ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; -import Pretender from 'pretender'; -import {AjaxError, UnauthorizedError} from 'ember-ajax/errors'; -import {RequestEntityTooLargeError, UnsupportedMediaTypeError} from 'ghost/services/ajax'; - -function stubAjaxEndpoint(server, response = {}, code = 500) { - server.get('/test/', function () { - return [ - code, - {'Content-Type': 'application/json'}, - JSON.stringify(response) - ]; - }); -} - -describeModule( - 'service:ajax', - 'Integration: Service: ajax', - { - integration: true - }, - function () { - let server; - - beforeEach(function () { - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('correctly parses single message response text', function (done) { - let error = {message: 'Test Error'}; - stubAjaxEndpoint(server, error); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true(); - }).catch((error) => { - expect(error.errors).to.equal('Test Error'); - done(); - }); - }); - - it('correctly parses single error response text', function (done) { - let error = {error: 'Test Error'}; - stubAjaxEndpoint(server, error); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true(); - }).catch((error) => { - expect(error.errors).to.equal('Test Error'); - done(); - }); - }); - - it('correctly parses multiple error messages', function (done) { - let error = {errors: ['First Error', 'Second Error']}; - stubAjaxEndpoint(server, error); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true(); - }).catch((error) => { - expect(error.errors).to.deep.equal(['First Error', 'Second Error']); - done(); - }); - }); - - it('returns default error object for non built-in error', function (done) { - stubAjaxEndpoint(server, {}); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true; - }).catch((error) => { - expect(error).to.be.instanceOf(AjaxError); - done(); - }); - }); - - it('returns known error object for built-in errors', function (done) { - stubAjaxEndpoint(server, '', 401); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true; - }).catch((error) => { - expect(error).to.be.instanceOf(UnauthorizedError); - done(); - }); - }); - - it('returns RequestEntityTooLargeError object for 413 errors', function (done) { - stubAjaxEndpoint(server, {}, 413); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true; - }).catch((error) => { - expect(error).to.be.instanceOf(RequestEntityTooLargeError); - done(); - }); - }); - - it('returns UnsupportedMediaTypeError object for 415 errors', function (done) { - stubAjaxEndpoint(server, {}, 415); - - let ajax = this.subject(); - - ajax.request('/test/').then(() => { - expect(false).to.be.true; - }).catch((error) => { - expect(error).to.be.instanceOf(UnsupportedMediaTypeError); - done(); - }); - }); - } -); diff --git a/core/client/tests/integration/services/feature-test.js b/core/client/tests/integration/services/feature-test.js deleted file mode 100644 index 00d2dac7c5..0000000000 --- a/core/client/tests/integration/services/feature-test.js +++ /dev/null @@ -1,207 +0,0 @@ -import { - describeModule, - it -} from 'ember-mocha'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; -import FeatureService, {feature} from 'ghost/services/feature'; -import Ember from 'ember'; -import { errorOverride, errorReset } from 'ghost/tests/helpers/adapter-error'; - -const {RSVP, merge, run} = Ember; -const EmberError = Ember.Error; - -function stubSettings(server, labs, validSave = true, validSettings = true) { - let settings = [ - { - id: '1', - type: 'blog', - key: 'labs', - value: JSON.stringify(labs) - } - ]; - - if (validSettings) { - settings.push({ - id: '2', - type: 'blog', - key: 'postsPerPage', - value: 1 - }); - } - - server.get('/ghost/api/v0.1/settings/', function () { - return [200, {'Content-Type': 'application/json'}, JSON.stringify({settings})]; - }); - - server.put('/ghost/api/v0.1/settings/', function (request) { - let statusCode = (validSave) ? 200 : 400; - let response = (validSave) ? request.requestBody : JSON.stringify({ - errors: [{ - message: 'Test Error' - }] - }); - - return [statusCode, {'Content-Type': 'application/json'}, response]; - }); -} - -function addTestFlag() { - FeatureService.reopen({ - testFlag: feature('testFlag') - }); -} - -describeModule( - 'service:feature', - 'Integration: Service: feature', - { - integration: true - }, - function () { - let server; - - beforeEach(function () { - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('loads labs settings correctly', function (done) { - stubSettings(server, {testFlag: true}); - addTestFlag(); - - let service = this.subject(); - - service.fetch().then(() => { - expect(service.get('testFlag')).to.be.true; - done(); - }); - }); - - it('returns false for set flag with config false and labs false', function (done) { - stubSettings(server, {testFlag: false}); - addTestFlag(); - - let service = this.subject(); - service.get('config').set('testFlag', false); - - service.fetch().then(() => { - expect(service.get('labs.testFlag')).to.be.false; - expect(service.get('testFlag')).to.be.false; - done(); - }); - }); - - it('returns true for set flag with config true and labs false', function (done) { - stubSettings(server, {testFlag: false}); - addTestFlag(); - - let service = this.subject(); - service.get('config').set('testFlag', true); - - service.fetch().then(() => { - expect(service.get('labs.testFlag')).to.be.false; - expect(service.get('testFlag')).to.be.true; - done(); - }); - }); - - it('returns true for set flag with config false and labs true', function (done) { - stubSettings(server, {testFlag: true}); - addTestFlag(); - - let service = this.subject(); - service.get('config').set('testFlag', false); - - service.fetch().then(() => { - expect(service.get('labs.testFlag')).to.be.true; - expect(service.get('testFlag')).to.be.true; - done(); - }); - }); - - it('returns true for set flag with config true and labs true', function (done) { - stubSettings(server, {testFlag: true}); - addTestFlag(); - - let service = this.subject(); - service.get('config').set('testFlag', true); - - service.fetch().then(() => { - expect(service.get('labs.testFlag')).to.be.true; - expect(service.get('testFlag')).to.be.true; - done(); - }); - }); - - it('saves correctly', function (done) { - stubSettings(server, {testFlag: false}); - addTestFlag(); - - let service = this.subject(); - - service.fetch().then(() => { - expect(service.get('testFlag')).to.be.false; - - run(() => { - service.set('testFlag', true); - }); - - return wait().then(() => { - expect(server.handlers[1].numberOfCalls).to.equal(1); - expect(service.get('testFlag')).to.be.true; - done(); - }); - }); - }); - - it('notifies for server errors', function (done) { - stubSettings(server, {testFlag: false}, false); - addTestFlag(); - - let service = this.subject(); - - service.fetch().then(() => { - expect(service.get('testFlag')).to.be.false; - - run(() => { - service.set('testFlag', true); - }); - - return wait().then(() => { - expect(server.handlers[1].numberOfCalls).to.equal(1); - expect(service.get('notifications.notifications').length).to.equal(1); - expect(service.get('testFlag')).to.be.false; - done(); - }); - }); - }); - - it('notifies for validation errors', function (done) { - stubSettings(server, {testFlag: false}, true, false); - addTestFlag(); - - let service = this.subject(); - - service.fetch().then(() => { - expect(service.get('testFlag')).to.be.false; - - run(() => { - expect(() => { - service.set('testFlag', true); - }, EmberError, 'threw validation error'); - }); - - return wait().then(() => { - // ensure validation is happening before the API is hit - expect(server.handlers[1].numberOfCalls).to.equal(0); - expect(service.get('testFlag')).to.be.false; - done(); - }); - }); - }); - } -); diff --git a/core/client/tests/integration/services/slug-generator-test.js b/core/client/tests/integration/services/slug-generator-test.js deleted file mode 100644 index c1f5adf6db..0000000000 --- a/core/client/tests/integration/services/slug-generator-test.js +++ /dev/null @@ -1,62 +0,0 @@ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; -import Pretender from 'pretender'; -import Ember from 'ember'; - -const {dasherize} = Ember.String; - -function stubSlugEndpoint(server, type, slug) { - server.get('/ghost/api/v0.1/slugs/:type/:slug/', function (request) { - expect(request.params.type).to.equal(type); - expect(request.params.slug).to.equal(slug); - - return [ - 200, - {'Content-Type': 'application/json'}, - JSON.stringify({slugs: [{slug: dasherize(slug)}]}) - ]; - }); -} - -describeModule( - 'service:slug-generator', - 'Integration: Service: slug-generator', - { - integration: true - }, - function () { - let server; - - beforeEach(function () { - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('returns empty if no slug is provided', function (done) { - let service = this.subject(); - - service.generateSlug('post', '').then(function (slug) { - expect(slug).to.equal(''); - done(); - }); - }); - - it('calls correct endpoint and returns correct data', function (done) { - let rawSlug = 'a test post'; - stubSlugEndpoint(server, 'post', rawSlug); - - let service = this.subject(); - - service.generateSlug('post', rawSlug).then(function (slug) { - expect(slug).to.equal(dasherize(rawSlug)); - done(); - }); - }); - } -); diff --git a/core/client/tests/test-helper.js b/core/client/tests/test-helper.js deleted file mode 100644 index 1c9e9bcb19..0000000000 --- a/core/client/tests/test-helper.js +++ /dev/null @@ -1,11 +0,0 @@ -import resolver from './helpers/resolver'; -import { setResolver } from 'ember-mocha'; - -setResolver(resolver); - -/* jshint ignore:start */ -mocha.setup({ - timeout: 15000, - slow: 500 -}); -/* jshint ignore:end */ diff --git a/core/client/tests/unit/.gitkeep b/core/client/tests/unit/.gitkeep deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/core/client/tests/unit/components/gh-alert-test.js b/core/client/tests/unit/components/gh-alert-test.js deleted file mode 100644 index c446fa063f..0000000000 --- a/core/client/tests/unit/components/gh-alert-test.js +++ /dev/null @@ -1,33 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} -from 'ember-mocha'; -import sinon from 'sinon'; - -describeComponent( - 'gh-alert', - 'Unit: Component: gh-alert', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('closes notification through notifications service', function () { - let component = this.subject(); - let notifications = {}; - let notification = {message: 'Test close', type: 'success'}; - - notifications.closeNotification = sinon.spy(); - component.set('notifications', notifications); - component.set('message', notification); - - this.$().find('button').click(); - - expect(notifications.closeNotification.calledWith(notification)).to.be.true; - }); - } -); diff --git a/core/client/tests/unit/components/gh-app-test.js b/core/client/tests/unit/components/gh-app-test.js deleted file mode 100644 index 8b7e17ad18..0000000000 --- a/core/client/tests/unit/components/gh-app-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-app', - 'Unit: Component: gh-app', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-content-preview-content-test.js b/core/client/tests/unit/components/gh-content-preview-content-test.js deleted file mode 100644 index 24369ac87c..0000000000 --- a/core/client/tests/unit/components/gh-content-preview-content-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-content-preview-content', - 'Unit: Component: gh-content-preview-content', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-editor-save-button-test.js b/core/client/tests/unit/components/gh-editor-save-button-test.js deleted file mode 100644 index eb48c72fdc..0000000000 --- a/core/client/tests/unit/components/gh-editor-save-button-test.js +++ /dev/null @@ -1,32 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-editor-save-button', - 'Unit: Component: gh-editor-save-button', - { - unit: true, - needs: [ - 'component:gh-dropdown-button', - 'component:gh-dropdown', - 'component:gh-spin-button', - 'service:dropdown' - ] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-editor-test.js b/core/client/tests/unit/components/gh-editor-test.js deleted file mode 100644 index 60b4d613b5..0000000000 --- a/core/client/tests/unit/components/gh-editor-test.js +++ /dev/null @@ -1,34 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-editor', - 'Unit: Component: gh-editor', - { - unit: true, - // specify the other units that are required for this test - needs: [ - 'component:gh-ed-editor', - 'component:gh-ed-preview', - 'helper:gh-count-words', - 'helper:route-action', - 'service:notifications' - ] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-file-uploader-test.js b/core/client/tests/unit/components/gh-file-uploader-test.js deleted file mode 100644 index 715555c755..0000000000 --- a/core/client/tests/unit/components/gh-file-uploader-test.js +++ /dev/null @@ -1,313 +0,0 @@ -/* jshint expr:true */ -/* global Blob */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import Ember from 'ember'; -import sinon from 'sinon'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; - -const {run} = Ember; - -const createFile = function (content = ['test'], options = {}) { - let { - name, - type, - lastModifiedDate - } = options; - - let file = new Blob(content, {type: type ? type : 'text/plain'}); - file.name = name ? name : 'text.txt'; - - return file; -}; - -const stubSuccessfulUpload = function (server, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [200, {'Content-Type': 'application/json'}, '"/content/images/test.png"']; - }, delay); -}; - -const stubFailedUpload = function (server, code, error, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [code, {'Content-Type': 'application/json'}, JSON.stringify({ - errors: [{ - errorType: error, - message: `Error: ${error}` - }] - })]; - }, delay); -}; - -describeComponent( - 'gh-file-uploader', - 'Unit: Component: gh-file-uploader', - { - needs: [ - 'service:ajax', - 'service:session', // used by ajax service - 'service:feature', - 'component:x-file-input' - ], - unit: true - }, - function() { - let server, url; - - beforeEach(function () { - server = new Pretender(); - url = '/ghost/api/v0.1/uploads/'; - }); - - afterEach(function () { - server.shutdown(); - }); - - it('renders', function() { - // creates the component instance - let component = this.subject(); - // renders the component on the page - this.render(); - expect(component).to.be.ok; - expect(this.$()).to.have.length(1); - }); - - it('fires uploadSuccess action on successful upload', function (done) { - let uploadSuccess = sinon.spy(); - let component = this.subject({url, uploadSuccess}); - let file = createFile(); - - stubSuccessfulUpload(server); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadSuccess.calledOnce).to.be.true; - expect(uploadSuccess.firstCall.args[0]).to.equal('/content/images/test.png'); - done(); - }); - }); - - it('fires uploadStarted action on upload start', function (done) { - let uploadStarted = sinon.spy(); - let component = this.subject({url, uploadStarted}); - let file = createFile(); - - stubSuccessfulUpload(server); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadStarted.calledOnce).to.be.true; - done(); - }); - }); - - it('fires uploadFinished action on successful upload', function (done) { - let uploadFinished = sinon.spy(); - let component = this.subject({url, uploadFinished}); - let file = createFile(); - - stubSuccessfulUpload(server); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadFinished.calledOnce).to.be.true; - done(); - }); - }); - - it('fires uploadFinished action on failed upload', function (done) { - let uploadFinished = sinon.spy(); - let component = this.subject({url, uploadFinished}); - let file = createFile(); - - stubFailedUpload(server); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadFinished.calledOnce).to.be.true; - done(); - }); - }); - - it('displays invalid file type error', function (done) { - let component = this.subject({url}); - let file = createFile(); - - stubFailedUpload(server, 415, 'UnsupportedMediaTypeError'); - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The file type you uploaded is not supported/); - expect(this.$('.btn-green').length, 'reset button is displayed').to.equal(1); - expect(this.$('.btn-green').text()).to.equal('Try Again'); - done(); - }); - }); - - it('displays file too large for server error', function (done) { - let component = this.subject({url}); - let file = createFile(); - - stubFailedUpload(server, 413, 'RequestEntityTooLargeError'); - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The file you uploaded was larger/); - done(); - }); - }); - - it('handles file too large error directly from the web server', function (done) { - let component = this.subject({url}); - let file = createFile(); - - server.post('/ghost/api/v0.1/uploads/', function () { - return [413, {}, '']; - }); - - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The file you uploaded was larger/); - done(); - }); - }); - - it('displays other server-side error with message', function (done) { - let component = this.subject({url}); - let file = createFile(); - - stubFailedUpload(server, 400, 'UnknownError'); - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/Error: UnknownError/); - done(); - }); - }); - - it('handles unknown failure', function (done) { - let component = this.subject({url}); - let file = createFile(); - - server.post('/ghost/api/v0.1/uploads/', function () { - return [500, {'Content-Type': 'application/json'}, '']; - }); - - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/Something went wrong/); - done(); - }); - }); - - it('can be reset after a failed upload', function (done) { - let component = this.subject({url}); - let file = createFile(); - - stubFailedUpload(server, 400, 'UnknownError'); - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - run(() => { - this.$('.btn-green').click(); - }); - }); - - wait().then(() => { - expect(this.$('input[type="file"]').length).to.equal(1); - done(); - }); - }); - - it('displays upload progress', function (done) { - let component = this.subject({url, uploadFinished: done}); - let file = createFile(); - - // pretender fires a progress event every 50ms - stubSuccessfulUpload(server, 150); - this.render(); - - run(() => { - component.send('fileSelected', [file]); - }); - - // after 75ms we should have had one progress event - run.later(this, function () { - expect(this.$('.progress .bar').length).to.equal(1); - let [_, percentageWidth] = this.$('.progress .bar').attr('style').match(/width: (\d+)%?/); - expect(percentageWidth).to.be.above(0); - expect(percentageWidth).to.be.below(100); - }, 75); - }); - - it('triggers file upload on file drop', function (done) { - let uploadSuccess = sinon.spy(); - let component = this.subject({url, uploadSuccess}); - let file = createFile(); - let drop = Ember.$.Event('drop', { - dataTransfer: { - files: [file] - } - }); - - stubSuccessfulUpload(server); - this.render(); - - run(() => { - this.$().trigger(drop); - }); - - wait().then(() => { - expect(uploadSuccess.calledOnce).to.be.true; - expect(uploadSuccess.firstCall.args[0]).to.equal('/content/images/test.png'); - done(); - }); - }); - } -); diff --git a/core/client/tests/unit/components/gh-image-uploader-test.js b/core/client/tests/unit/components/gh-image-uploader-test.js deleted file mode 100644 index 30bfb0c88a..0000000000 --- a/core/client/tests/unit/components/gh-image-uploader-test.js +++ /dev/null @@ -1,344 +0,0 @@ -/* jshint expr:true */ -/* global Blob */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import Ember from 'ember'; -import sinon from 'sinon'; -import Pretender from 'pretender'; -import wait from 'ember-test-helpers/wait'; - -const {run} = Ember; - -const createFile = function (content = ['test'], options = {}) { - let { - name, - type, - lastModifiedDate - } = options; - - let file = new Blob(content, {type: type ? type : 'text/plain'}); - file.name = name ? name : 'text.txt'; - - return file; -}; - -const stubSuccessfulUpload = function (server, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [200, {'Content-Type': 'application/json'}, '"/content/images/test.png"']; - }, delay); -}; - -const stubFailedUpload = function (server, code, error, delay = 0) { - server.post('/ghost/api/v0.1/uploads/', function () { - return [code, {'Content-Type': 'application/json'}, JSON.stringify({ - errors: [{ - errorType: error, - message: `Error: ${error}` - }] - })]; - }, delay); -}; - -describeComponent( - 'gh-image-uploader', - 'Unit: Component: gh-image-uploader', - { - needs: [ - 'service:config', - 'service:session', - 'service:ajax', - 'service:feature', - 'component:x-file-input', - 'component:one-way-input' - ], - unit: true - }, - function() { - let server; - - beforeEach(function () { - server = new Pretender(); - }); - - afterEach(function () { - server.shutdown(); - }); - - it('renders', function() { - // creates the component instance - let component = this.subject(); - // renders the component on the page - this.render(); - expect(component).to.be.ok; - expect(this.$()).to.have.length(1); - }); - - it('fires update action on successful upload', function (done) { - let component = this.subject(); - let update = sinon.spy(); - let file = createFile(); - - stubSuccessfulUpload(server); - - this.render(); - component.attrs.update = update; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(update.calledOnce).to.be.true; - expect(update.firstCall.args[0]).to.equal('/content/images/test.png'); - done(); - }); - }); - - it('fires uploadStarted action on upload start', function (done) { - let component = this.subject(); - let uploadStarted = sinon.spy(); - let file = createFile(); - - stubSuccessfulUpload(server); - - this.render(); - component.attrs.update = () => {}; - component.attrs.uploadStarted = uploadStarted; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadStarted.calledOnce).to.be.true; - done(); - }); - }); - - it('fires uploadFinished action on successful upload', function (done) { - let component = this.subject(); - let uploadFinished = sinon.spy(); - let file = createFile(); - - stubSuccessfulUpload(server); - - this.render(); - component.attrs.update = () => {}; - component.attrs.uploadFinished = uploadFinished; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadFinished.calledOnce).to.be.true; - done(); - }); - }); - - it('fires uploadFinished action on failed upload', function (done) { - let component = this.subject(); - let uploadFinished = sinon.spy(); - let file = createFile(); - - stubFailedUpload(server); - - this.render(); - component.attrs.update = () => {}; - component.attrs.uploadFinished = uploadFinished; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(uploadFinished.calledOnce).to.be.true; - done(); - }); - }); - - it('displays invalid file type error', function (done) { - let component = this.subject(); - let file = createFile(); - - stubFailedUpload(server, 415, 'UnsupportedMediaTypeError'); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The image type you uploaded is not supported/); - expect(this.$('.btn-green').length, 'reset button is displayed').to.equal(1); - expect(this.$('.btn-green').text()).to.equal('Try Again'); - done(); - }); - }); - - it('displays file too large for server error', function (done) { - let component = this.subject(); - let file = createFile(); - - stubFailedUpload(server, 413, 'RequestEntityTooLargeError'); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The image you uploaded was larger/); - done(); - }); - }); - - it('handles file too large error directly from the web server', function (done) { - let component = this.subject(); - let file = createFile(); - - server.post('/ghost/api/v0.1/uploads/', function () { - return [413, {}, '']; - }); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/The image you uploaded was larger/); - done(); - }); - }); - - it('displays other server-side error with message', function (done) { - let component = this.subject(); - let file = createFile(); - - stubFailedUpload(server, 400, 'UnknownError'); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/Error: UnknownError/); - done(); - }); - }); - - it('handles unknown failure', function (done) { - let component = this.subject(); - let file = createFile(); - - server.post('/ghost/api/v0.1/uploads/', function () { - return [500, {'Content-Type': 'application/json'}, '']; - }); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - expect(this.$('.failed').length, 'error message is displayed').to.equal(1); - expect(this.$('.failed').text()).to.match(/Something went wrong/); - done(); - }); - }); - - it('can be reset after a failed upload', function (done) { - let component = this.subject(); - let file = createFile(); - - stubFailedUpload(server, 400, 'UnknownError'); - - this.render(); - component.attrs.update = () => {}; - - run(() => { - component.send('fileSelected', [file]); - }); - - wait().then(() => { - run(() => { - this.$('.btn-green').click(); - }); - }); - - wait().then(() => { - expect(this.$('input[type="file"]').length).to.equal(1); - done(); - }); - }); - - it('displays upload progress', function (done) { - let component = this.subject(); - let file = createFile(); - - // pretender fires a progress event every 50ms - stubSuccessfulUpload(server, 150); - - this.render(); - component.attrs.update = () => {}; - component.attrs.uploadFinished = done; - - run(() => { - component.send('fileSelected', [file]); - }); - - // after 75ms we should have had one progress event - run.later(this, function () { - expect(this.$('.progress .bar').length).to.equal(1); - let [_, percentageWidth] = this.$('.progress .bar').attr('style').match(/width: (\d+)%?/); - expect(percentageWidth).to.be.above(0); - expect(percentageWidth).to.be.below(100); - }, 75); - }); - - it('triggers file upload on file drop', function (done) { - let component = this.subject(); - let file = createFile(); - let update = sinon.spy(); - let drop = Ember.$.Event('drop', { - dataTransfer: { - files: [file] - } - }); - - stubSuccessfulUpload(server); - - this.render(); - component.attrs.update = update; - - run(() => { - this.$().trigger(drop); - }); - - wait().then(() => { - expect(update.calledOnce).to.be.true; - expect(update.firstCall.args[0]).to.equal('/content/images/test.png'); - done(); - }); - }); - } -); diff --git a/core/client/tests/unit/components/gh-infinite-scroll-test.js b/core/client/tests/unit/components/gh-infinite-scroll-test.js deleted file mode 100644 index 872f04e7dc..0000000000 --- a/core/client/tests/unit/components/gh-infinite-scroll-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-infinite-scroll', - 'Unit: Component: gh-infinite-scroll', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-navitem-url-input-test.js b/core/client/tests/unit/components/gh-navitem-url-input-test.js deleted file mode 100644 index 3534bcfcc0..0000000000 --- a/core/client/tests/unit/components/gh-navitem-url-input-test.js +++ /dev/null @@ -1,39 +0,0 @@ -/* jshint expr:true */ -import Ember from 'ember'; -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -const {run} = Ember; - -describeComponent( - 'gh-navitem-url-input', - 'Unit: Component: gh-navitem-url-input', - { - unit: true - }, - function () { - it('identifies a URL as the base URL', function () { - let component = this.subject({ - url: '', - baseUrl: 'http://example.com/' - }); - - this.render(); - - run(function () { - component.set('value', 'http://example.com/'); - }); - - expect(component.get('isBaseUrl')).to.be.ok; - - run(function () { - component.set('value', 'http://example.com/go/'); - }); - - expect(component.get('isBaseUrl')).to.not.be.ok; - }); - } -); diff --git a/core/client/tests/unit/components/gh-notification-test.js b/core/client/tests/unit/components/gh-notification-test.js deleted file mode 100644 index 08522a5853..0000000000 --- a/core/client/tests/unit/components/gh-notification-test.js +++ /dev/null @@ -1,50 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} -from 'ember-mocha'; -import sinon from 'sinon'; - -describeComponent( - 'gh-notification', - 'Unit: Component: gh-notification', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('closes notification through notifications service', function () { - let component = this.subject(); - let notifications = {}; - let notification = {message: 'Test close', type: 'success'}; - - notifications.closeNotification = sinon.spy(); - component.set('notifications', notifications); - component.set('message', notification); - - this.$().find('button').click(); - - expect(notifications.closeNotification.calledWith(notification)).to.be.true; - }); - - it('closes notification when animationend event is triggered', function (done) { - let component = this.subject(); - let notifications = {}; - let notification = {message: 'Test close', type: 'success'}; - - notifications.closeNotification = sinon.spy(); - component.set('notifications', notifications); - component.set('message', notification); - - // shorten the animation delay to speed up test - this.$().css('animation-delay', '0.1s'); - setTimeout(function () { - expect(notifications.closeNotification.calledWith(notification)).to.be.true; - done(); - }, 150); - }); - } -); diff --git a/core/client/tests/unit/components/gh-posts-list-item-test.js b/core/client/tests/unit/components/gh-posts-list-item-test.js deleted file mode 100644 index 73ebaee077..0000000000 --- a/core/client/tests/unit/components/gh-posts-list-item-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-posts-list-item', - 'Unit: Component: gh-posts-list-item', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-select-native-test.js b/core/client/tests/unit/components/gh-select-native-test.js deleted file mode 100644 index 258a456345..0000000000 --- a/core/client/tests/unit/components/gh-select-native-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-select-native', - 'Unit: Component: gh-select-native', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-selectize-test.js b/core/client/tests/unit/components/gh-selectize-test.js deleted file mode 100644 index 92474e8622..0000000000 --- a/core/client/tests/unit/components/gh-selectize-test.js +++ /dev/null @@ -1,40 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; -import Ember from 'ember'; - -const {run} = Ember; -const emberA = Ember.A; - -describeComponent( - 'gh-selectize', - 'Unit: Component: gh-selectize', - { - // Specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'], - unit: true - }, - function () { - it('re-orders selection when selectize order is changed', function () { - let component = this.subject(); - - run(() => { - component.set('content', emberA(['item 1', 'item 2', 'item 3'])); - component.set('selection', emberA(['item 2', 'item 3'])); - component.set('multiple', true); - }); - - this.render(); - - run(() => { - component._selectize.setValue(['item 3', 'item 2']); - }); - - expect(component.get('value'), 'component value').to.deep.equal(['item 3', 'item 2']); - expect(component.get('selection'), 'component selection').to.deep.equal(['item 3', 'item 2']); - }); - } -); diff --git a/core/client/tests/unit/components/gh-spin-button-test.js b/core/client/tests/unit/components/gh-spin-button-test.js deleted file mode 100644 index bb20ec1a4a..0000000000 --- a/core/client/tests/unit/components/gh-spin-button-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-spin-button', - 'Unit: Component: gh-spin-button', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-trim-focus-input_test.js b/core/client/tests/unit/components/gh-trim-focus-input_test.js deleted file mode 100644 index f10d9181fc..0000000000 --- a/core/client/tests/unit/components/gh-trim-focus-input_test.js +++ /dev/null @@ -1,47 +0,0 @@ -import Ember from 'ember'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-trim-focus-input', - 'Unit: Component: gh-trim-focus-input', - { - unit: true - }, - function () { - it('trims value on focusOut', function () { - let component = this.subject({ - value: 'some random stuff ' - }); - - this.render(); - - component.$().focusout(); - expect(component.$().val()).to.equal('some random stuff'); - }); - - it('does not have the autofocus attribute if not set to focus', function () { - let component = this.subject({ - value: 'some text', - focus: false - }); - - this.render(); - - expect(component.$().attr('autofocus')).to.not.be.ok; - }); - - it('has the autofocus attribute if set to focus', function () { - let component = this.subject({ - value: 'some text', - focus: true - }); - - this.render(); - - expect(component.$().attr('autofocus')).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/components/gh-url-preview_test.js b/core/client/tests/unit/components/gh-url-preview_test.js deleted file mode 100644 index 9a72ec36ae..0000000000 --- a/core/client/tests/unit/components/gh-url-preview_test.js +++ /dev/null @@ -1,42 +0,0 @@ -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-url-preview', - 'Unit: Component: gh-url-preview', - { - unit: true - }, - function () { - it('generates the correct preview URL with a prefix', function () { - let component = this.subject({ - prefix: 'tag', - slug: 'test-slug', - tagName: 'p', - classNames: 'test-class', - - config: {blogUrl: 'http://my-ghost-blog.com'} - }); - - this.render(); - - expect(component.get('url')).to.equal('my-ghost-blog.com/tag/test-slug/'); - }); - - it('generates the correct preview URL without a prefix', function () { - let component = this.subject({ - slug: 'test-slug', - tagName: 'p', - classNames: 'test-class', - - config: {blogUrl: 'http://my-ghost-blog.com'} - }); - - this.render(); - - expect(component.get('url')).to.equal('my-ghost-blog.com/test-slug/'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-user-active-test.js b/core/client/tests/unit/components/gh-user-active-test.js deleted file mode 100644 index e27ba9655d..0000000000 --- a/core/client/tests/unit/components/gh-user-active-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-user-active', - 'Unit: Component: gh-user-active', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/components/gh-user-invited-test.js b/core/client/tests/unit/components/gh-user-invited-test.js deleted file mode 100644 index 8cea19978b..0000000000 --- a/core/client/tests/unit/components/gh-user-invited-test.js +++ /dev/null @@ -1,28 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describeComponent, - it -} from 'ember-mocha'; - -describeComponent( - 'gh-user-invited', - 'Unit: Component: gh-user-invited', - { - unit: true - // specify the other units that are required for this test - // needs: ['component:foo', 'helper:bar'] - }, - function () { - it('renders', function () { - // creates the component instance - let component = this.subject(); - - expect(component._state).to.equal('preRender'); - - // renders the component on the page - this.render(); - expect(component._state).to.equal('inDOM'); - }); - } -); diff --git a/core/client/tests/unit/controllers/post-settings-menu-test.js b/core/client/tests/unit/controllers/post-settings-menu-test.js deleted file mode 100644 index ce07942e3f..0000000000 --- a/core/client/tests/unit/controllers/post-settings-menu-test.js +++ /dev/null @@ -1,712 +0,0 @@ -/* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ -import Ember from 'ember'; -import { - describeModule, - it -} from 'ember-mocha'; - -const {run} = Ember; - -function K() { - return this; -} - -describeModule( - 'controller:post-settings-menu', - 'Unit: Controller: post-settings-menu', - { - needs: ['controller:application', 'service:notifications', 'service:slug-generator'] - }, - - function () { - it('slugValue is one-way bound to model.slug', function () { - let controller = this.subject({ - model: Ember.Object.create({ - slug: 'a-slug' - }) - }); - - expect(controller.get('model.slug')).to.equal('a-slug'); - expect(controller.get('slugValue')).to.equal('a-slug'); - - run(function () { - controller.set('model.slug', 'changed-slug'); - - expect(controller.get('slugValue')).to.equal('changed-slug'); - }); - - run(function () { - controller.set('slugValue', 'changed-directly'); - - expect(controller.get('model.slug')).to.equal('changed-slug'); - expect(controller.get('slugValue')).to.equal('changed-directly'); - }); - - run(function () { - // test that the one-way binding is still in place - controller.set('model.slug', 'should-update'); - - expect(controller.get('slugValue')).to.equal('should-update'); - }); - }); - - it('metaTitleScratch is one-way bound to model.metaTitle', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaTitle: 'a title' - }) - }); - - expect(controller.get('model.metaTitle')).to.equal('a title'); - expect(controller.get('metaTitleScratch')).to.equal('a title'); - - run(function () { - controller.set('model.metaTitle', 'a different title'); - - expect(controller.get('metaTitleScratch')).to.equal('a different title'); - }); - - run(function () { - controller.set('metaTitleScratch', 'changed directly'); - - expect(controller.get('model.metaTitle')).to.equal('a different title'); - expect(controller.get('metaTitleScratch')).to.equal('changed directly'); - }); - - run(function () { - // test that the one-way binding is still in place - controller.set('model.metaTitle', 'should update'); - - expect(controller.get('metaTitleScratch')).to.equal('should update'); - }); - }); - - it('metaDescriptionScratch is one-way bound to model.metaDescription', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaDescription: 'a description' - }) - }); - - expect(controller.get('model.metaDescription')).to.equal('a description'); - expect(controller.get('metaDescriptionScratch')).to.equal('a description'); - - run(function () { - controller.set('model.metaDescription', 'a different description'); - - expect(controller.get('metaDescriptionScratch')).to.equal('a different description'); - }); - - run(function () { - controller.set('metaDescriptionScratch', 'changed directly'); - - expect(controller.get('model.metaDescription')).to.equal('a different description'); - expect(controller.get('metaDescriptionScratch')).to.equal('changed directly'); - }); - - run(function () { - // test that the one-way binding is still in place - controller.set('model.metaDescription', 'should update'); - - expect(controller.get('metaDescriptionScratch')).to.equal('should update'); - }); - }); - - describe('seoTitle', function () { - it('should be the metaTitle if one exists', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaTitle: 'a meta-title', - titleScratch: 'should not be used' - }) - }); - - expect(controller.get('seoTitle')).to.equal('a meta-title'); - }); - - it('should default to the title if an explicit meta-title does not exist', function () { - let controller = this.subject({ - model: Ember.Object.create({ - titleScratch: 'should be the meta-title' - }) - }); - - expect(controller.get('seoTitle')).to.equal('should be the meta-title'); - }); - - it('should be the metaTitle if both title and metaTitle exist', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaTitle: 'a meta-title', - titleScratch: 'a title' - }) - }); - - expect(controller.get('seoTitle')).to.equal('a meta-title'); - }); - - it('should revert to the title if explicit metaTitle is removed', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaTitle: 'a meta-title', - titleScratch: 'a title' - }) - }); - - expect(controller.get('seoTitle')).to.equal('a meta-title'); - - run(function () { - controller.set('model.metaTitle', ''); - - expect(controller.get('seoTitle')).to.equal('a title'); - }); - }); - - it('should truncate to 70 characters with an appended ellipsis', function () { - let longTitle = new Array(100).join('a'); - let controller = this.subject({ - model: Ember.Object.create() - }); - - expect(longTitle.length).to.equal(99); - - run(function () { - let expected = `${longTitle.substr(0, 70)}…`; - - controller.set('metaTitleScratch', longTitle); - - expect(controller.get('seoTitle').toString().length).to.equal(78); - expect(controller.get('seoTitle').toString()).to.equal(expected); - }); - }); - }); - - describe('seoDescription', function () { - it('should be the metaDescription if one exists', function () { - let controller = this.subject({ - model: Ember.Object.create({ - metaDescription: 'a description' - }) - }); - - expect(controller.get('seoDescription')).to.equal('a description'); - }); - - it.skip('should be generated from the rendered markdown if not explicitly set', function () { - // can't test right now because the rendered markdown is being pulled - // from the DOM via jquery - }); - - it('should truncate to 156 characters with an appended ellipsis', function () { - let longDescription = new Array(200).join('a'); - let controller = this.subject({ - model: Ember.Object.create() - }); - - expect(longDescription.length).to.equal(199); - - run(function () { - let expected = `${longDescription.substr(0, 156)}…`; - - controller.set('metaDescriptionScratch', longDescription); - - expect(controller.get('seoDescription').toString().length).to.equal(164); - expect(controller.get('seoDescription').toString()).to.equal(expected); - }); - }); - }); - - describe('seoURL', function () { - it('should be the URL of the blog if no post slug exists', function () { - let controller = this.subject({ - config: Ember.Object.create({blogUrl: 'http://my-ghost-blog.com'}), - model: Ember.Object.create() - }); - - expect(controller.get('seoURL')).to.equal('http://my-ghost-blog.com/'); - }); - - it('should be the URL of the blog plus the post slug', function () { - let controller = this.subject({ - config: Ember.Object.create({blogUrl: 'http://my-ghost-blog.com'}), - model: Ember.Object.create({slug: 'post-slug'}) - }); - - expect(controller.get('seoURL')).to.equal('http://my-ghost-blog.com/post-slug/'); - }); - - it('should update when the post slug changes', function () { - let controller = this.subject({ - config: Ember.Object.create({blogUrl: 'http://my-ghost-blog.com'}), - model: Ember.Object.create({slug: 'post-slug'}) - }); - - expect(controller.get('seoURL')).to.equal('http://my-ghost-blog.com/post-slug/'); - - run(function () { - controller.set('model.slug', 'changed-slug'); - - expect(controller.get('seoURL')).to.equal('http://my-ghost-blog.com/changed-slug/'); - }); - }); - - it('should truncate a long URL to 70 characters with an appended ellipsis', function () { - let blogURL = 'http://my-ghost-blog.com'; - let longSlug = new Array(75).join('a'); - let controller = this.subject({ - config: Ember.Object.create({blogUrl: blogURL}), - model: Ember.Object.create({slug: longSlug}) - }); - let expected; - - expect(longSlug.length).to.equal(74); - - expected = `${blogURL}/${longSlug}/`; - expected = `${expected.substr(0, 70)}…`; - - expect(controller.get('seoURL').toString().length).to.equal(78); - expect(controller.get('seoURL').toString()).to.equal(expected); - }); - }); - - describe('togglePage', function () { - it('should toggle the page property', function () { - let controller = this.subject({ - model: Ember.Object.create({ - page: false, - isNew: true - }) - }); - - expect(controller.get('model.page')).to.not.be.ok; - - run(function () { - controller.send('togglePage'); - - expect(controller.get('model.page')).to.be.ok; - }); - }); - - it('should not save the post if it is still new', function () { - let controller = this.subject({ - model: Ember.Object.create({ - page: false, - isNew: true, - save() { - this.incrementProperty('saved'); - return Ember.RSVP.resolve(); - } - }) - }); - - run(function () { - controller.send('togglePage'); - - expect(controller.get('model.page')).to.be.ok; - expect(controller.get('model.saved')).to.not.be.ok; - }); - }); - - it('should save the post if it is not new', function () { - let controller = this.subject({ - model: Ember.Object.create({ - page: false, - isNew: false, - save() { - this.incrementProperty('saved'); - return Ember.RSVP.resolve(); - } - }) - }); - - run(function () { - controller.send('togglePage'); - - expect(controller.get('model.page')).to.be.ok; - expect(controller.get('model.saved')).to.equal(1); - }); - }); - }); - - describe('toggleFeatured', function () { - it('should toggle the featured property', function () { - let controller = this.subject({ - model: Ember.Object.create({ - featured: false, - isNew: true - }) - }); - - run(function () { - controller.send('toggleFeatured'); - - expect(controller.get('model.featured')).to.be.ok; - }); - }); - - it('should not save the post if it is still new', function () { - let controller = this.subject({ - model: Ember.Object.create({ - featured: false, - isNew: true, - save() { - this.incrementProperty('saved'); - return Ember.RSVP.resolve(); - } - }) - }); - - run(function () { - controller.send('toggleFeatured'); - - expect(controller.get('model.featured')).to.be.ok; - expect(controller.get('model.saved')).to.not.be.ok; - }); - }); - - it('should save the post if it is not new', function () { - let controller = this.subject({ - model: Ember.Object.create({ - featured: false, - isNew: false, - save() { - this.incrementProperty('saved'); - return Ember.RSVP.resolve(); - } - }) - }); - - run(function () { - controller.send('toggleFeatured'); - - expect(controller.get('model.featured')).to.be.ok; - expect(controller.get('model.saved')).to.equal(1); - }); - }); - }); - - describe('generateAndSetSlug', function () { - it('should generate a slug and set it on the destination', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - return Ember.RSVP.resolve(`${str}-slug`); - } - }), - model: Ember.Object.create({slug: ''}) - }); - - run(function () { - controller.set('model.titleScratch', 'title'); - controller.generateAndSetSlug('model.slug'); - - expect(controller.get('model.slug')).to.equal(''); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('title-slug'); - - done(); - }).catch(done); - }); - }); - - it('should not set the destination if the title is "(Untitled)" and the post already has a slug', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - return Ember.RSVP.resolve(`${str}-slug`); - } - }), - model: Ember.Object.create({ - slug: 'whatever' - }) - }); - - expect(controller.get('model.slug')).to.equal('whatever'); - - run(function () { - controller.set('model.titleScratch', 'title'); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('whatever'); - - done(); - }).catch(done); - }); - }); - }); - - describe('titleObserver', function () { - it('should invoke generateAndSetSlug if the post is new and a title has not been set', function (done) { - let controller = this.subject({ - model: Ember.Object.create({isNew: true}), - invoked: 0, - generateAndSetSlug() { - this.incrementProperty('invoked'); - } - }); - - expect(controller.get('invoked')).to.equal(0); - expect(controller.get('model.title')).to.not.be.ok; - - run(function () { - controller.set('model.titleScratch', 'test'); - - controller.titleObserver(); - - // since titleObserver invokes generateAndSetSlug with a delay of 700ms - // we need to make sure this assertion runs after that. - // probably a better way to handle this? - run.later(function () { - expect(controller.get('invoked')).to.equal(1); - - done(); - }, 800); - }); - }); - - it('should invoke generateAndSetSlug if the post title is "(Untitled)"', function (done) { - let controller = this.subject({ - model: Ember.Object.create({ - isNew: false, - title: '(Untitled)' - }), - invoked: 0, - generateAndSetSlug() { - this.incrementProperty('invoked'); - } - }); - - expect(controller.get('invoked')).to.equal(0); - expect(controller.get('model.title')).to.equal('(Untitled)'); - - run(function () { - controller.set('model.titleScratch', 'test'); - - controller.titleObserver(); - - // since titleObserver invokes generateAndSetSlug with a delay of 700ms - // we need to make sure this assertion runs after that. - // probably a better way to handle this? - run.later(function () { - expect(controller.get('invoked')).to.equal(1); - - done(); - }, 800); - }); - }); - - it('should not invoke generateAndSetSlug if the post is new but has a title', function (done) { - let controller = this.subject({ - model: Ember.Object.create({ - isNew: true, - title: 'a title' - }), - invoked: 0, - generateAndSetSlug() { - this.incrementProperty('invoked'); - } - }); - - expect(controller.get('invoked')).to.equal(0); - expect(controller.get('model.title')).to.equal('a title'); - - run(function () { - controller.set('model.titleScratch', 'test'); - - controller.titleObserver(); - - // since titleObserver invokes generateAndSetSlug with a delay of 700ms - // we need to make sure this assertion runs after that. - // probably a better way to handle this? - run.later(function () { - expect(controller.get('invoked')).to.equal(0); - - done(); - }, 800); - }); - }); - }); - - describe('updateSlug', function () { - it('should reset slugValue to the previous slug when the new slug is blank or unchanged', function () { - let controller = this.subject({ - model: Ember.Object.create({ - slug: 'slug' - }) - }); - - run(function () { - // unchanged - controller.set('slugValue', 'slug'); - controller.send('updateSlug', controller.get('slugValue')); - - expect(controller.get('model.slug')).to.equal('slug'); - expect(controller.get('slugValue')).to.equal('slug'); - }); - - run(function () { - // unchanged after trim - controller.set('slugValue', 'slug '); - controller.send('updateSlug', controller.get('slugValue')); - - expect(controller.get('model.slug')).to.equal('slug'); - expect(controller.get('slugValue')).to.equal('slug'); - }); - - run(function () { - // blank - controller.set('slugValue', ''); - controller.send('updateSlug', controller.get('slugValue')); - - expect(controller.get('model.slug')).to.equal('slug'); - expect(controller.get('slugValue')).to.equal('slug'); - }); - }); - - it('should not set a new slug if the server-generated slug matches existing slug', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - let promise = Ember.RSVP.resolve(str.split('#')[0]); - this.set('lastPromise', promise); - return promise; - } - }), - model: Ember.Object.create({ - slug: 'whatever' - }) - }); - - run(function () { - controller.set('slugValue', 'whatever#slug'); - controller.send('updateSlug', controller.get('slugValue')); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('whatever'); - - done(); - }).catch(done); - }); - }); - - it('should not set a new slug if the only change is to the appended increment value', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - let sanitizedStr = str.replace(/[^a-zA-Z]/g, ''); - let promise = Ember.RSVP.resolve(`${sanitizedStr}-2`); - this.set('lastPromise', promise); - return promise; - } - }), - model: Ember.Object.create({ - slug: 'whatever' - }) - }); - - run(function () { - controller.set('slugValue', 'whatever!'); - controller.send('updateSlug', controller.get('slugValue')); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('whatever'); - - done(); - }).catch(done); - }); - }); - - it('should set the slug if the new slug is different', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - let promise = Ember.RSVP.resolve(str); - this.set('lastPromise', promise); - return promise; - } - }), - model: Ember.Object.create({ - slug: 'whatever', - save: K - }) - }); - - run(function () { - controller.set('slugValue', 'changed'); - controller.send('updateSlug', controller.get('slugValue')); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('changed'); - - done(); - }).catch(done); - }); - }); - - it('should save the post when the slug changes and the post is not new', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - let promise = Ember.RSVP.resolve(str); - this.set('lastPromise', promise); - return promise; - } - }), - model: Ember.Object.create({ - slug: 'whatever', - saved: 0, - isNew: false, - save() { - this.incrementProperty('saved'); - } - }) - }); - - run(function () { - controller.set('slugValue', 'changed'); - controller.send('updateSlug', controller.get('slugValue')); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('changed'); - expect(controller.get('model.saved')).to.equal(1); - - done(); - }).catch(done); - }); - }); - - it('should not save the post when the slug changes and the post is new', function (done) { - let controller = this.subject({ - slugGenerator: Ember.Object.create({ - generateSlug(slugType, str) { - let promise = Ember.RSVP.resolve(str); - this.set('lastPromise', promise); - return promise; - } - }), - model: Ember.Object.create({ - slug: 'whatever', - saved: 0, - isNew: true, - save() { - this.incrementProperty('saved'); - } - }) - }); - - run(function () { - controller.set('slugValue', 'changed'); - controller.send('updateSlug', controller.get('slugValue')); - - Ember.RSVP.resolve(controller.get('lastPromise')).then(function () { - expect(controller.get('model.slug')).to.equal('changed'); - expect(controller.get('model.saved')).to.equal(0); - - done(); - }).catch(done); - }); - }); - }); - } -); diff --git a/core/client/tests/unit/controllers/settings/general-test.js b/core/client/tests/unit/controllers/settings/general-test.js deleted file mode 100644 index 763b0f9fc5..0000000000 --- a/core/client/tests/unit/controllers/settings/general-test.js +++ /dev/null @@ -1,93 +0,0 @@ -import Ember from 'ember'; -import { - describeModule, - it -} from 'ember-mocha'; - -const {run} = Ember; - -describeModule( - 'controller:settings/general', - 'Unit: Controller: settings/general', - { - needs: ['service:notifications'] - }, - - function () { - it('isDatedPermalinks should be correct', function () { - let controller = this.subject({ - model: Ember.Object.create({ - permalinks: '/:year/:month/:day/:slug/' - }) - }); - - expect(controller.get('isDatedPermalinks')).to.be.ok; - - run(function () { - controller.set('model.permalinks', '/:slug/'); - - expect(controller.get('isDatedPermalinks')).to.not.be.ok; - }); - }); - - it('setting isDatedPermalinks should switch between dated and slug', function () { - let controller = this.subject({ - model: Ember.Object.create({ - permalinks: '/:year/:month/:day/:slug/' - }) - }); - - run(function () { - controller.set('isDatedPermalinks', false); - - expect(controller.get('isDatedPermalinks')).to.not.be.ok; - expect(controller.get('model.permalinks')).to.equal('/:slug/'); - }); - - run(function () { - controller.set('isDatedPermalinks', true); - - expect(controller.get('isDatedPermalinks')).to.be.ok; - expect(controller.get('model.permalinks')).to.equal('/:year/:month/:day/:slug/'); - }); - }); - - it('themes should be correct', function () { - let themes = []; - let controller; - - themes.push({ - name: 'casper', - active: true, - package: { - name: 'Casper', - version: '1.1.5' - } - }); - - themes.push({ - name: 'rasper', - package: { - name: 'Rasper', - version: '1.0.0' - } - }); - - controller = this.subject({ - model: Ember.Object.create({ - availableThemes: themes - }) - }); - - themes = controller.get('themes'); - expect(themes).to.be.an.Array; - expect(themes.length).to.equal(2); - expect(themes.objectAt(0).name).to.equal('casper'); - expect(themes.objectAt(0).active).to.be.ok; - expect(themes.objectAt(0).label).to.equal('Casper - 1.1.5'); - expect(themes.objectAt(1).name).to.equal('rasper'); - expect(themes.objectAt(1).active).to.not.be.ok; - expect(themes.objectAt(1).label).to.equal('Rasper - 1.0.0'); - }); - } -); diff --git a/core/client/tests/unit/controllers/settings/navigation-test.js b/core/client/tests/unit/controllers/settings/navigation-test.js deleted file mode 100644 index 4ddfde3cbe..0000000000 --- a/core/client/tests/unit/controllers/settings/navigation-test.js +++ /dev/null @@ -1,181 +0,0 @@ -/* jshint expr:true */ -import { expect, assert } from 'chai'; -import { describeModule, it } from 'ember-mocha'; -import Ember from 'ember'; -import NavItem from 'ghost/models/navigation-item'; - -const {run} = Ember; - -const navSettingJSON = `[ - {"label":"Home","url":"/"}, - {"label":"JS Test","url":"javascript:alert('hello');"}, - {"label":"About","url":"/about"}, - {"label":"Sub Folder","url":"/blah/blah"}, - {"label":"Telephone","url":"tel:01234-567890"}, - {"label":"Mailto","url":"mailto:test@example.com"}, - {"label":"External","url":"https://example.com/testing?query=test#anchor"}, - {"label":"No Protocol","url":"//example.com"} -]`; - -describeModule( - 'controller:settings/navigation', - 'Unit: Controller: settings/navigation', - { - // Specify the other units that are required for this test. - needs: ['service:config', 'service:notifications', 'model:navigation-item'] - }, - function () { - it('blogUrl: captures config and ensures trailing slash', function () { - let ctrl = this.subject(); - ctrl.set('config.blogUrl', 'http://localhost:2368/blog'); - expect(ctrl.get('blogUrl')).to.equal('http://localhost:2368/blog/'); - }); - - it('init: creates a new navigation item', function () { - let ctrl = this.subject(); - - run(() => { - expect(ctrl.get('newNavItem')).to.exist; - expect(ctrl.get('newNavItem.isNew')).to.be.true; - }); - }); - - it('blogUrl: captures config and ensures trailing slash', function () { - let ctrl = this.subject(); - ctrl.set('config.blogUrl', 'http://localhost:2368/blog'); - expect(ctrl.get('blogUrl')).to.equal('http://localhost:2368/blog/'); - }); - - it('save: validates nav items', function (done) { - let ctrl = this.subject(); - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: [ - NavItem.create({label: 'First', url: '/'}), - NavItem.create({label: '', url: '/second'}), - NavItem.create({label: 'Third', url: ''}) - ]})); - // blank item won't get added because the last item is incomplete - expect(ctrl.get('model.navigation.length')).to.equal(3); - - ctrl.save().then(function passedValidation() { - assert(false, 'navigationItems weren\'t validated on save'); - done(); - }).catch(function failedValidation() { - let navItems = ctrl.get('model.navigation'); - expect(navItems[0].get('errors').toArray()).to.be.empty; - expect(navItems[1].get('errors.firstObject.attribute')).to.equal('label'); - expect(navItems[2].get('errors.firstObject.attribute')).to.equal('url'); - done(); - }); - }); - }); - - it('save: ignores blank last item when saving', function (done) { - let ctrl = this.subject(); - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: [ - NavItem.create({label: 'First', url: '/'}), - NavItem.create({label: '', url: ''}) - ]})); - - expect(ctrl.get('model.navigation.length')).to.equal(2); - - ctrl.save().then(function passedValidation() { - assert(false, 'navigationItems weren\'t validated on save'); - done(); - }).catch(function failedValidation() { - let navItems = ctrl.get('model.navigation'); - expect(navItems[0].get('errors').toArray()).to.be.empty; - done(); - }); - }); - }); - - it('action - addItem: adds item to navigationItems', function () { - let ctrl = this.subject(); - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: [ - NavItem.create({label: 'First', url: '/first', last: true}) - ]})); - }); - - expect(ctrl.get('model.navigation.length')).to.equal(1); - - ctrl.set('newNavItem.label', 'New'); - ctrl.set('newNavItem.url', '/new'); - - run(() => { - ctrl.send('addItem'); - }); - - expect(ctrl.get('model.navigation.length')).to.equal(2); - expect(ctrl.get('model.navigation.lastObject.label')).to.equal('New'); - expect(ctrl.get('model.navigation.lastObject.url')).to.equal('/new'); - expect(ctrl.get('model.navigation.lastObject.isNew')).to.be.false; - expect(ctrl.get('newNavItem.label')).to.be.blank; - expect(ctrl.get('newNavItem.url')).to.be.blank; - expect(ctrl.get('newNavItem.isNew')).to.be.true; - }); - - it('action - addItem: doesn\'t insert new item if last object is incomplete', function () { - let ctrl = this.subject(); - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: [ - NavItem.create({label: '', url: '', last: true}) - ]})); - expect(ctrl.get('model.navigation.length')).to.equal(1); - ctrl.send('addItem'); - expect(ctrl.get('model.navigation.length')).to.equal(1); - }); - }); - - it('action - deleteItem: removes item from navigationItems', function () { - let ctrl = this.subject(); - let navItems = [ - NavItem.create({label: 'First', url: '/first'}), - NavItem.create({label: 'Second', url: '/second', last: true}) - ]; - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: navItems})); - expect(ctrl.get('model.navigation').mapBy('label')).to.deep.equal(['First', 'Second']); - ctrl.send('deleteItem', ctrl.get('model.navigation.firstObject')); - expect(ctrl.get('model.navigation').mapBy('label')).to.deep.equal(['Second']); - }); - }); - - it('action - reorderItems: updates navigationItems list', function () { - let ctrl = this.subject(); - let navItems = [ - NavItem.create({label: 'First', url: '/first'}), - NavItem.create({label: 'Second', url: '/second', last: true}) - ]; - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: navItems})); - expect(ctrl.get('model.navigation').mapBy('label')).to.deep.equal(['First', 'Second']); - ctrl.send('reorderItems', navItems.reverseObjects()); - expect(ctrl.get('model.navigation').mapBy('label')).to.deep.equal(['Second', 'First']); - }); - }); - - it('action - updateUrl: updates URL on navigationItem', function () { - let ctrl = this.subject(); - let navItems = [ - NavItem.create({label: 'First', url: '/first'}), - NavItem.create({label: 'Second', url: '/second', last: true}) - ]; - - run(() => { - ctrl.set('model', Ember.Object.create({navigation: navItems})); - expect(ctrl.get('model.navigation').mapBy('url')).to.deep.equal(['/first', '/second']); - ctrl.send('updateUrl', '/new', ctrl.get('model.navigation.firstObject')); - expect(ctrl.get('model.navigation').mapBy('url')).to.deep.equal(['/new', '/second']); - }); - }); - } -); diff --git a/core/client/tests/unit/controllers/subscribers-test.js b/core/client/tests/unit/controllers/subscribers-test.js deleted file mode 100644 index d56e8ce2e4..0000000000 --- a/core/client/tests/unit/controllers/subscribers-test.js +++ /dev/null @@ -1,21 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; - -describeModule( - 'controller:subscribers', - 'Unit: Controller: subscribers', - { - needs: ['service:notifications'] - }, - function() { - // Replace this with your real tests. - it('exists', function() { - let controller = this.subject(); - expect(controller).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/helpers/gh-user-can-admin-test.js b/core/client/tests/unit/helpers/gh-user-can-admin-test.js deleted file mode 100644 index 2a6a6ed61b..0000000000 --- a/core/client/tests/unit/helpers/gh-user-can-admin-test.js +++ /dev/null @@ -1,59 +0,0 @@ -import { - describeModule, - it -} from 'ember-mocha'; -import { ghUserCanAdmin } from 'ghost/helpers/gh-user-can-admin'; - -describe('Unit: Helper: gh-user-can-admin', function () { - // Mock up roles and test for truthy - describe('Owner role', function () { - let user = { - get(role) { - if (role === 'isOwner') { - return true; - } else if (role === 'isAdmin') { - return false; - } - } - }; - - it(' - can be Admin', function () { - let result = ghUserCanAdmin([user]); - expect(result).to.equal(true); - }); - }); - - describe('Administrator role', function () { - let user = { - get(role) { - if (role === 'isOwner') { - return false; - } else if (role === 'isAdmin') { - return true; - } - } - }; - - it(' - can be Admin', function () { - let result = ghUserCanAdmin([user]); - expect(result).to.equal(true); - }); - }); - - describe('Editor and Author roles', function () { - let user = { - get(role) { - if (role === 'isOwner') { - return false; - } else if (role === 'isAdmin') { - return false; - } - } - }; - - it(' - cannot be Admin', function () { - let result = ghUserCanAdmin([user]); - expect(result).to.equal(false); - }); - }); -}); diff --git a/core/client/tests/unit/helpers/highlighted-text-test.js b/core/client/tests/unit/helpers/highlighted-text-test.js deleted file mode 100644 index 2075efca21..0000000000 --- a/core/client/tests/unit/helpers/highlighted-text-test.js +++ /dev/null @@ -1,18 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import { - highlightedText -} from 'ghost/helpers/highlighted-text'; - -describe('Unit: Helper: highlighted-text', function() { - - it('works', function() { - let result = highlightedText(['Test', 'e']); - expect(result).to.be.an('object'); - expect(result.string).to.equal('T<span class="highlight">e</span>st'); - }); -}); diff --git a/core/client/tests/unit/helpers/is-equal-test.js b/core/client/tests/unit/helpers/is-equal-test.js deleted file mode 100644 index d7cffefa8d..0000000000 --- a/core/client/tests/unit/helpers/is-equal-test.js +++ /dev/null @@ -1,18 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describe, - it -} from 'mocha'; -import { - isEqual -} from 'ghost/helpers/is-equal'; - -describe('Unit: Helper: is-equal', function () { - // Replace this with your real tests. - it('works', function () { - let result = isEqual([42, 42]); - - expect(result).to.be.ok; - }); -}); diff --git a/core/client/tests/unit/helpers/is-not-test.js b/core/client/tests/unit/helpers/is-not-test.js deleted file mode 100644 index 7e2cc1f754..0000000000 --- a/core/client/tests/unit/helpers/is-not-test.js +++ /dev/null @@ -1,18 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describe, - it -} from 'mocha'; -import { - isNot -} from 'ghost/helpers/is-not'; - -describe('Unit: Helper: is-not', function () { - // Replace this with your real tests. - it('works', function () { - let result = isNot(false); - - expect(result).to.be.ok; - }); -}); diff --git a/core/client/tests/unit/mixins/infinite-scroll-test.js b/core/client/tests/unit/mixins/infinite-scroll-test.js deleted file mode 100644 index 88ba7f1ae8..0000000000 --- a/core/client/tests/unit/mixins/infinite-scroll-test.js +++ /dev/null @@ -1,18 +0,0 @@ -/* jshint expr:true */ -import {expect} from 'chai'; -import { - describe, - it -} from 'mocha'; -import Ember from 'ember'; -import InfiniteScrollMixin from 'ghost/mixins/infinite-scroll'; - -describe('Unit: Mixin: infinite-scroll', function () { - // Replace this with your real tests. - it('works', function () { - let InfiniteScrollObject = Ember.Object.extend(InfiniteScrollMixin); - let subject = InfiniteScrollObject.create(); - - expect(subject).to.be.ok; - }); -}); diff --git a/core/client/tests/unit/mixins/validation-engine-test.js b/core/client/tests/unit/mixins/validation-engine-test.js deleted file mode 100644 index 63e6f5a685..0000000000 --- a/core/client/tests/unit/mixins/validation-engine-test.js +++ /dev/null @@ -1,42 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import Ember from 'ember'; -import ValidationEngineMixin from 'ghost/mixins/validation-engine'; - -describe('ValidationEngineMixin', function () { - // Replace this with your real tests. - // it('works', function () { - // var ValidationEngineObject = Ember.Object.extend(ValidationEngineMixin); - // var subject = ValidationEngineObject.create(); - // expect(subject).to.be.ok; - // }); - - describe('#validate', function () { - it('loads the correct validator'); - it('rejects if the validator doesn\'t exist'); - it('resolves with valid object'); - it('rejects with invalid object'); - it('clears all existing errors'); - - describe('with a specified property', function () { - it('resolves with valid property'); - it('rejects with invalid property'); - it('adds property to hasValidated array'); - it('clears existing error on specified property'); - }); - - it('handles a passed in model'); - it('uses this.model if available'); - }); - - describe('#save', function () { - it('calls validate'); - it('rejects with validation errors'); - it('calls object\'s #save if validation passes'); - it('skips validation if it\'s a deletion'); - }); -}); diff --git a/core/client/tests/unit/models/navigation-item-test.js b/core/client/tests/unit/models/navigation-item-test.js deleted file mode 100644 index 524adaa072..0000000000 --- a/core/client/tests/unit/models/navigation-item-test.js +++ /dev/null @@ -1,67 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModule, it } from 'ember-mocha'; - -describeModule( - 'model:navigation-item', - 'Unit: Model: navigation-item', - { - // Specify the other units that are required for this test. - needs: [] - }, - function() { - it('isComplete is true when label and url are filled', function () { - let model = this.subject(); - - model.set('label', 'test'); - model.set('url', 'test'); - - expect(model.get('isComplete')).to.be.true; - }); - - it('isComplete is false when label is blank', function () { - let model = this.subject(); - - model.set('label', ''); - model.set('url', 'test'); - - expect(model.get('isComplete')).to.be.false; - }); - - it('isComplete is false when url is blank', function () { - let model = this.subject(); - - model.set('label', 'test'); - model.set('url', ''); - - expect(model.get('isComplete')).to.be.false; - }); - - it('isBlank is true when label and url are blank', function () { - let model = this.subject(); - - model.set('label', ''); - model.set('url', ''); - - expect(model.get('isBlank')).to.be.true; - }); - - it('isBlank is false when label is present', function () { - let model = this.subject(); - - model.set('label', 'test'); - model.set('url', ''); - - expect(model.get('isBlank')).to.be.false; - }); - - it('isBlank is false when url is present', function () { - let model = this.subject(); - - model.set('label', ''); - model.set('url', 'test'); - - expect(model.get('isBlank')).to.be.false; - }); - } -); diff --git a/core/client/tests/unit/models/post-test.js b/core/client/tests/unit/models/post-test.js deleted file mode 100644 index 7b50d8ad4d..0000000000 --- a/core/client/tests/unit/models/post-test.js +++ /dev/null @@ -1,84 +0,0 @@ -import Ember from 'ember'; -import { - describeModel, - it -} from 'ember-mocha'; - -describeModel( - 'post', - 'Unit: Model: post', - { - needs: ['model:user', 'model:tag', 'model:role'] - }, - - function () { - it('has a validation type of "post"', function () { - let model = this.subject(); - - expect(model.validationType).to.equal('post'); - }); - - it('isPublished and isDraft are correct', function () { - let model = this.subject({ - status: 'published' - }); - - expect(model.get('isPublished')).to.be.ok; - expect(model.get('isDraft')).to.not.be.ok; - - Ember.run(function () { - model.set('status', 'draft'); - - expect(model.get('isPublished')).to.not.be.ok; - expect(model.get('isDraft')).to.be.ok; - }); - }); - - it('isAuthoredByUser is correct', function () { - /* jscs:disable requireCamelCaseOrUpperCaseIdentifiers */ - let model = this.subject({ - authorId: 15 - }); - /* jscs:enable requireCamelCaseOrUpperCaseIdentifiers */ - let user = Ember.Object.create({id: '15'}); - - expect(model.isAuthoredByUser(user)).to.be.ok; - - Ember.run(function () { - model.set('authorId', 1); - - expect(model.isAuthoredByUser(user)).to.not.be.ok; - }); - }); - - it('updateTags removes and deletes old tags', function () { - let model = this.subject(); - - Ember.run(this, function () { - let modelTags = model.get('tags'); - let tag1 = this.store().createRecord('tag', {id: '1'}); - let tag2 = this.store().createRecord('tag', {id: '2'}); - let tag3 = this.store().createRecord('tag'); - - // During testing a record created without an explicit id will get - // an id of 'fixture-n' instead of null - tag3.set('id', null); - - modelTags.pushObject(tag1); - modelTags.pushObject(tag2); - modelTags.pushObject(tag3); - - expect(model.get('tags.length')).to.equal(3); - - model.updateTags(); - - expect(model.get('tags.length')).to.equal(2); - expect(model.get('tags.firstObject.id')).to.equal('1'); - expect(model.get('tags').objectAt(1).get('id')).to.equal('2'); - expect(tag1.get('isDeleted')).to.not.be.ok; - expect(tag2.get('isDeleted')).to.not.be.ok; - expect(tag3.get('isDeleted')).to.be.ok; - }); - }); - } -); diff --git a/core/client/tests/unit/models/role-test.js b/core/client/tests/unit/models/role-test.js deleted file mode 100644 index 4305257889..0000000000 --- a/core/client/tests/unit/models/role-test.js +++ /dev/null @@ -1,25 +0,0 @@ -import Ember from 'ember'; -import { - describeModel, - it -} from 'ember-mocha'; - -const {run} = Ember; - -describeModel('role', 'Unit: Model: role', function () { - it('provides a lowercase version of the name', function () { - let model = this.subject({ - name: 'Author' - }); - - expect(model.get('name')).to.equal('Author'); - expect(model.get('lowerCaseName')).to.equal('author'); - - run(function () { - model.set('name', 'Editor'); - - expect(model.get('name')).to.equal('Editor'); - expect(model.get('lowerCaseName')).to.equal('editor'); - }); - }); -}); diff --git a/core/client/tests/unit/models/setting-test.js b/core/client/tests/unit/models/setting-test.js deleted file mode 100644 index 1ee59225db..0000000000 --- a/core/client/tests/unit/models/setting-test.js +++ /dev/null @@ -1,12 +0,0 @@ -import { - describeModel, - it -} from 'ember-mocha'; - -describeModel('setting', 'Unit: Model: setting', function () { - it('has a validation type of "setting"', function () { - let model = this.subject(); - - expect(model.get('validationType')).to.equal('setting'); - }); -}); diff --git a/core/client/tests/unit/models/subscriber-test.js b/core/client/tests/unit/models/subscriber-test.js deleted file mode 100644 index d5a2ff3627..0000000000 --- a/core/client/tests/unit/models/subscriber-test.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModel, it } from 'ember-mocha'; - -describeModel( - 'subscriber', - 'Unit: Model: subscriber', - { - // Specify the other units that are required for this test. - needs: ['model:post'] - }, - function() { - // Replace this with your real tests. - it('exists', function() { - let model = this.subject(); - // var store = this.store(); - expect(model).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/models/tag-test.js b/core/client/tests/unit/models/tag-test.js deleted file mode 100644 index 3a961109f6..0000000000 --- a/core/client/tests/unit/models/tag-test.js +++ /dev/null @@ -1,12 +0,0 @@ -import { - describeModel, - it -} from 'ember-mocha'; - -describeModel('tag', 'Unit: Model: tag', function () { - it('has a validation type of "tag"', function () { - let model = this.subject(); - - expect(model.get('validationType')).to.equal('tag'); - }); -}); diff --git a/core/client/tests/unit/models/user-test.js b/core/client/tests/unit/models/user-test.js deleted file mode 100644 index 60f7376736..0000000000 --- a/core/client/tests/unit/models/user-test.js +++ /dev/null @@ -1,141 +0,0 @@ -import Ember from 'ember'; -import { - describeModel, - it -} from 'ember-mocha'; - -const {run} = Ember; - -describeModel( - 'user', - 'Unit: Model: user', - { - needs: ['model:role', 'serializer:application', 'serializer:user'] - }, - - function () { - it('has a validation type of "user"', function () { - let model = this.subject(); - - expect(model.get('validationType')).to.equal('user'); - }); - - it('active property is correct', function () { - let model = this.subject({ - status: 'active' - }); - - expect(model.get('active')).to.be.ok; - - ['warn-1', 'warn-2', 'warn-3', 'warn-4', 'locked'].forEach(function (status) { - run(() => { model.set('status', status); }); - expect(model.get('status')).to.be.ok; - }); - - run(() => { model.set('status', 'inactive'); }); - expect(model.get('active')).to.not.be.ok; - - run(() => { model.set('status', 'invited'); }); - expect(model.get('active')).to.not.be.ok; - }); - - it('invited property is correct', function () { - let model = this.subject({ - status: 'invited' - }); - - expect(model.get('invited')).to.be.ok; - - run(() => { model.set('status', 'invited-pending'); }); - expect(model.get('invited')).to.be.ok; - - run(() => { model.set('status', 'active'); }); - expect(model.get('invited')).to.not.be.ok; - - run(() => { model.set('status', 'inactive'); }); - expect(model.get('invited')).to.not.be.ok; - }); - - it('pending property is correct', function () { - let model = this.subject({ - status: 'invited-pending' - }); - - expect(model.get('pending')).to.be.ok; - - run(() => { model.set('status', 'invited'); }); - expect(model.get('pending')).to.not.be.ok; - - run(() => { model.set('status', 'inactive'); }); - expect(model.get('pending')).to.not.be.ok; - }); - - it('role property is correct', function () { - let model = this.subject(); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Author'}}}); - model.get('roles').pushObject(role); - }); - expect(model.get('role.name')).to.equal('Author'); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Editor'}}}); - model.set('role', role); - }); - expect(model.get('role.name')).to.equal('Editor'); - }); - - it('isAuthor property is correct', function () { - let model = this.subject(); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Author'}}}); - model.set('role', role); - }); - expect(model.get('isAuthor')).to.be.ok; - expect(model.get('isEditor')).to.not.be.ok; - expect(model.get('isAdmin')).to.not.be.ok; - expect(model.get('isOwner')).to.not.be.ok; - }); - - it('isEditor property is correct', function () { - let model = this.subject(); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Editor'}}}); - model.set('role', role); - }); - expect(model.get('isEditor')).to.be.ok; - expect(model.get('isAuthor')).to.not.be.ok; - expect(model.get('isAdmin')).to.not.be.ok; - expect(model.get('isOwner')).to.not.be.ok; - }); - - it('isAdmin property is correct', function () { - let model = this.subject(); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Administrator'}}}); - model.set('role', role); - }); - expect(model.get('isAdmin')).to.be.ok; - expect(model.get('isAuthor')).to.not.be.ok; - expect(model.get('isEditor')).to.not.be.ok; - expect(model.get('isOwner')).to.not.be.ok; - }); - - it('isOwner property is correct', function () { - let model = this.subject(); - - run(() => { - let role = this.store().push({data: {id: 1, type: 'role', attributes: {name: 'Owner'}}}); - model.set('role', role); - }); - expect(model.get('isOwner')).to.be.ok; - expect(model.get('isAuthor')).to.not.be.ok; - expect(model.get('isAdmin')).to.not.be.ok; - expect(model.get('isEditor')).to.not.be.ok; - }); - } -); diff --git a/core/client/tests/unit/routes/subscribers-test.js b/core/client/tests/unit/routes/subscribers-test.js deleted file mode 100644 index 1df5526729..0000000000 --- a/core/client/tests/unit/routes/subscribers-test.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; - -describeModule( - 'route:subscribers', - 'Unit: Route: subscribers', - { - needs: ['service:notifications'] - }, - function() { - it('exists', function() { - let route = this.subject(); - expect(route).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/routes/subscribers/import-test.js b/core/client/tests/unit/routes/subscribers/import-test.js deleted file mode 100644 index 87c71a2341..0000000000 --- a/core/client/tests/unit/routes/subscribers/import-test.js +++ /dev/null @@ -1,21 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; - -describeModule( - 'route:subscribers/import', - 'SubscribersImportRoute', - { - // Specify the other units that are required for this test. - needs: ['service:notifications'] - }, - function() { - it('exists', function() { - let route = this.subject(); - expect(route).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/routes/subscribers/new-test.js b/core/client/tests/unit/routes/subscribers/new-test.js deleted file mode 100644 index 87ed684a63..0000000000 --- a/core/client/tests/unit/routes/subscribers/new-test.js +++ /dev/null @@ -1,20 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; - -describeModule( - 'route:subscribers/new', - 'Unit: Route: subscribers/new', - { - needs: ['service:notifications'] - }, - function() { - it('exists', function() { - let route = this.subject(); - expect(route).to.be.ok; - }); - } -); diff --git a/core/client/tests/unit/services/config-test.js b/core/client/tests/unit/services/config-test.js deleted file mode 100644 index 47a7104d52..0000000000 --- a/core/client/tests/unit/services/config-test.js +++ /dev/null @@ -1,34 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; - -import Ember from 'ember'; - -describeModule( - 'service:config', - 'Unit: Service: config', - { - // Specify the other units that are required for this test. - // needs: ['service:foo'] - }, - function () { - // Replace this with your real tests. - it('exists', function () { - let service = this.subject(); - expect(service).to.be.ok; - }); - - it('correctly parses a client secret', function () { - Ember.$('<meta>').attr('name', 'env-clientSecret') - .attr('content', '23e435234423') - .appendTo('head'); - - let service = this.subject(); - - expect(service.get('clientSecret')).to.equal('23e435234423'); - }); - } -); diff --git a/core/client/tests/unit/services/notifications-test.js b/core/client/tests/unit/services/notifications-test.js deleted file mode 100644 index caebbc7708..0000000000 --- a/core/client/tests/unit/services/notifications-test.js +++ /dev/null @@ -1,419 +0,0 @@ -/* jshint expr:true */ -import Ember from 'ember'; -import sinon from 'sinon'; -import { expect } from 'chai'; -import { - describeModule, - it -} from 'ember-mocha'; -import {AjaxError, InvalidError} from 'ember-ajax/errors'; - -const {run, get} = Ember; -const emberA = Ember.A; - -describeModule( - 'service:notifications', - 'Unit: Service: notifications', - { - // Specify the other units that are required for this test. - // needs: ['model:notification'] - }, - function () { - beforeEach(function () { - this.subject().set('content', emberA()); - this.subject().set('delayedNotifications', emberA()); - }); - - it('filters alerts/notifications', function () { - let notifications = this.subject(); - - // wrapped in run-loop to enure alerts/notifications CPs are updated - run(() => { - notifications.showAlert('Alert'); - notifications.showNotification('Notification'); - }); - - expect(notifications.get('alerts.length')).to.equal(1); - expect(notifications.get('alerts.firstObject.message')).to.equal('Alert'); - - expect(notifications.get('notifications.length')).to.equal(1); - expect(notifications.get('notifications.firstObject.message')).to.equal('Notification'); - }); - - it('#handleNotification deals with DS.Notification notifications', function () { - let notifications = this.subject(); - let notification = Ember.Object.create({message: '<h1>Test</h1>', status: 'alert'}); - - notification.toJSON = function () {}; - - notifications.handleNotification(notification); - - notification = notifications.get('alerts')[0]; - - // alerts received from the server should be marked html safe - expect(notification.get('message')).to.have.property('toHTML'); - }); - - it('#handleNotification defaults to notification if no status supplied', function () { - let notifications = this.subject(); - - notifications.handleNotification({message: 'Test'}, false); - - expect(notifications.get('content')) - .to.deep.include({message: 'Test', status: 'notification'}); - }); - - it('#showAlert adds POJO alerts', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAlert('Test Alert', {type: 'error'}); - }); - - expect(notifications.get('alerts')) - .to.deep.include({message: 'Test Alert', status: 'alert', type: 'error', key: undefined}); - }); - - it('#showAlert adds delayed notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('Test Alert', {type: 'error', delayed: true}); - }); - - expect(notifications.get('delayedNotifications')) - .to.deep.include({message: 'Test Alert', status: 'notification', type: 'error', key: undefined}); - }); - - // in order to cater for complex keys that are suitable for i18n - // we split on the second period and treat the resulting base as - // the key for duplicate checking - it('#showAlert clears duplicates', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAlert('Kept'); - notifications.showAlert('Duplicate', {key: 'duplicate.key.fail'}); - }); - - expect(notifications.get('alerts.length')).to.equal(2); - - run(() => { - notifications.showAlert('Duplicate with new message', {key: 'duplicate.key.success'}); - }); - - expect(notifications.get('alerts.length')).to.equal(2); - expect(notifications.get('alerts.lastObject.message')).to.equal('Duplicate with new message'); - }); - - it('#showNotification adds POJO notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('Test Notification', {type: 'success'}); - }); - - expect(notifications.get('notifications')) - .to.deep.include({message: 'Test Notification', status: 'notification', type: 'success', key: undefined}); - }); - - it('#showNotification adds delayed notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('Test Notification', {delayed: true}); - }); - - expect(notifications.get('delayedNotifications')) - .to.deep.include({message: 'Test Notification', status: 'notification', type: undefined, key: undefined}); - }); - - it('#showNotification clears existing notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('First'); - notifications.showNotification('Second'); - }); - - expect(notifications.get('notifications.length')).to.equal(1); - expect(notifications.get('notifications')) - .to.deep.equal([{message: 'Second', status: 'notification', type: undefined, key: undefined}]); - }); - - it('#showNotification keeps existing notifications if doNotCloseNotifications option passed', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('First'); - notifications.showNotification('Second', {doNotCloseNotifications: true}); - }); - - expect(notifications.get('notifications.length')).to.equal(2); - }); - - // TODO: review whether this can be removed once it's no longer used by validations - it('#showErrors adds multiple notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showErrors([ - {message: 'First'}, - {message: 'Second'} - ]); - }); - - expect(notifications.get('notifications')).to.deep.equal([ - {message: 'First', status: 'notification', type: 'error', key: undefined}, - {message: 'Second', status: 'notification', type: 'error', key: undefined} - ]); - }); - - it('#showAPIError adds single json response error', function () { - let notifications = this.subject(); - let error = new AjaxError('Single error'); - - run(() => { - notifications.showAPIError(error); - }); - - let notification = notifications.get('alerts.firstObject'); - expect(get(notification, 'message')).to.equal('Single error'); - expect(get(notification, 'status')).to.equal('alert'); - expect(get(notification, 'type')).to.equal('error'); - expect(get(notification, 'key')).to.equal('api-error'); - }); - - // used to display validation errors returned from the server - it('#showAPIError adds multiple json response errors', function () { - let notifications = this.subject(); - let error = new AjaxError(['First error', 'Second error']); - - run(() => { - notifications.showAPIError(error); - }); - - expect(notifications.get('notifications')).to.deep.equal([ - {message: 'First error', status: 'notification', type: 'error', key: undefined}, - {message: 'Second error', status: 'notification', type: 'error', key: undefined} - ]); - }); - - it('#showAPIError displays default error text if response has no error/message', function () { - let notifications = this.subject(); - let resp = false; - - run(() => { notifications.showAPIError(resp); }); - - expect(notifications.get('content')).to.deep.equal([ - {message: 'There was a problem on the server, please try again.', status: 'alert', type: 'error', key: 'api-error'} - ]); - - notifications.set('content', emberA()); - - run(() => { - notifications.showAPIError(resp, {defaultErrorText: 'Overridden default'}); - }); - expect(notifications.get('content')).to.deep.equal([ - {message: 'Overridden default', status: 'alert', type: 'error', key: 'api-error'} - ]); - }); - - it('#showAPIError sets correct key when passed a base key', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAPIError('Test', {key: 'test.alert'}); - }); - - expect(notifications.get('alerts.firstObject.key')).to.equal('test.alert.api-error'); - }); - - it('#showAPIError sets correct key when not passed a key', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAPIError('Test'); - }); - - expect(notifications.get('alerts.firstObject.key')).to.equal('api-error'); - }); - - it('#showAPIError parses errors from ember-ajax correctly', function () { - let notifications = this.subject(); - let error = new InvalidError('Test Error'); - - run(() => { - notifications.showAPIError(error); - }); - - let notification = notifications.get('alerts.firstObject'); - expect(get(notification, 'message')).to.equal('Test Error'); - expect(get(notification, 'status')).to.equal('alert'); - expect(get(notification, 'type')).to.equal('error'); - expect(get(notification, 'key')).to.equal('api-error'); - }); - - it('#displayDelayed moves delayed notifications into content', function () { - let notifications = this.subject(); - - run(() => { - notifications.showNotification('First', {delayed: true}); - notifications.showNotification('Second', {delayed: true}); - notifications.showNotification('Third', {delayed: false}); - notifications.displayDelayed(); - }); - - expect(notifications.get('notifications')).to.deep.equal([ - {message: 'Third', status: 'notification', type: undefined, key: undefined}, - {message: 'First', status: 'notification', type: undefined, key: undefined}, - {message: 'Second', status: 'notification', type: undefined, key: undefined} - ]); - }); - - it('#closeNotification removes POJO notifications', function () { - let notification = {message: 'Close test', status: 'notification'}; - let notifications = this.subject(); - - run(() => { - notifications.handleNotification(notification); - }); - - expect(notifications.get('notifications')) - .to.include(notification); - - run(() => { - notifications.closeNotification(notification); - }); - - expect(notifications.get('notifications')) - .to.not.include(notification); - }); - - it('#closeNotification removes and deletes DS.Notification records', function () { - let notification = Ember.Object.create({message: 'Close test', status: 'alert'}); - let notifications = this.subject(); - - notification.toJSON = function () {}; - notification.deleteRecord = function () {}; - sinon.spy(notification, 'deleteRecord'); - notification.save = function () { - return { - finally(callback) { - return callback(notification); - } - }; - }; - sinon.spy(notification, 'save'); - - run(() => { notifications.handleNotification(notification); }); - - expect(notifications.get('alerts')).to.include(notification); - - run(() => { notifications.closeNotification(notification); }); - - expect(notification.deleteRecord.calledOnce).to.be.true; - expect(notification.save.calledOnce).to.be.true; - - expect(notifications.get('alerts')).to.not.include(notification); - }); - - it('#closeNotifications only removes notifications', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAlert('First alert'); - notifications.showNotification('First notification'); - notifications.showNotification('Second notification', {doNotCloseNotifications: true}); - }); - - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); - expect(notifications.get('notifications.length'), 'notifications count').to.equal(2); - - run(() => { notifications.closeNotifications(); }); - - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); - expect(notifications.get('notifications.length'), 'notifications count').to.equal(0); - }); - - it('#closeNotifications only closes notifications with specified key', function () { - let notifications = this.subject(); - - run(() => { - notifications.showAlert('First alert'); - // using handleNotification as showNotification will auto-prune - // duplicates and keys will be removed if doNotCloseNotifications - // is true - notifications.handleNotification({message: 'First notification', key: 'test.close', status: 'notification'}); - notifications.handleNotification({message: 'Second notification', key: 'test.keep', status: 'notification'}); - notifications.handleNotification({message: 'Third notification', key: 'test.close', status: 'notification'}); - }); - - run(() => { - notifications.closeNotifications('test.close'); - }); - - expect(notifications.get('notifications.length'), 'notifications count').to.equal(1); - expect(notifications.get('notifications.firstObject.message'), 'notification message').to.equal('Second notification'); - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); - }); - - it('#clearAll removes everything without deletion', function () { - let notifications = this.subject(); - let notificationModel = Ember.Object.create({message: 'model'}); - - notificationModel.toJSON = function () {}; - notificationModel.deleteRecord = function () {}; - sinon.spy(notificationModel, 'deleteRecord'); - notificationModel.save = function () { - return { - finally(callback) { - return callback(notificationModel); - } - }; - }; - sinon.spy(notificationModel, 'save'); - - notifications.handleNotification(notificationModel); - notifications.handleNotification({message: 'pojo'}); - - notifications.clearAll(); - - expect(notifications.get('content')).to.be.empty; - expect(notificationModel.deleteRecord.called).to.be.false; - expect(notificationModel.save.called).to.be.false; - }); - - it('#closeAlerts only removes alerts', function () { - let notifications = this.subject(); - - notifications.showNotification('First notification'); - notifications.showAlert('First alert'); - notifications.showAlert('Second alert'); - - run(() => { - notifications.closeAlerts(); - }); - - expect(notifications.get('alerts.length')).to.equal(0); - expect(notifications.get('notifications.length')).to.equal(1); - }); - - it('#closeAlerts closes only alerts with specified key', function () { - let notifications = this.subject(); - - notifications.showNotification('First notification'); - notifications.showAlert('First alert', {key: 'test.close'}); - notifications.showAlert('Second alert', {key: 'test.keep'}); - notifications.showAlert('Third alert', {key: 'test.close'}); - - run(() => { - notifications.closeAlerts('test.close'); - }); - - expect(notifications.get('alerts.length')).to.equal(1); - expect(notifications.get('alerts.firstObject.message')).to.equal('Second alert'); - expect(notifications.get('notifications.length')).to.equal(1); - }); - } -); diff --git a/core/client/tests/unit/transforms/facebook-url-user-test.js b/core/client/tests/unit/transforms/facebook-url-user-test.js deleted file mode 100644 index dfef25bcc9..0000000000 --- a/core/client/tests/unit/transforms/facebook-url-user-test.js +++ /dev/null @@ -1,32 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModule, it } from 'ember-mocha'; -import Ember from 'ember'; - -const emberA = Ember.A; - -describeModule( - 'transform:facebook-url-user', - 'Unit: Transform: facebook-url-user', - { - // Specify the other units that are required for this test. - // needs: ['transform:foo'] - }, - function() { - it('deserializes facebook url', function () { - let transform = this.subject(); - let serialized = 'testuser'; - let result = transform.deserialize(serialized); - - expect(result).to.equal('https://www.facebook.com/testuser'); - }); - - it('serializes url to facebook username', function () { - let transform = this.subject(); - let deserialized = 'https://www.facebook.com/testuser'; - let result = transform.serialize(deserialized); - - expect(result).to.equal('testuser'); - }); - } -); diff --git a/core/client/tests/unit/transforms/navigation-settings-test.js b/core/client/tests/unit/transforms/navigation-settings-test.js deleted file mode 100644 index 3e6e82d270..0000000000 --- a/core/client/tests/unit/transforms/navigation-settings-test.js +++ /dev/null @@ -1,42 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModule, it } from 'ember-mocha'; -import Ember from 'ember'; -import NavigationItem from 'ghost/models/navigation-item'; - -const emberA = Ember.A; - -describeModule( - 'transform:navigation-settings', - 'Unit: Transform: navigation-settings', - { - // Specify the other units that are required for this test. - // needs: ['transform:foo'] - }, - function() { - it('deserializes navigation json', function () { - let transform = this.subject(); - let serialized = '[{"label":"One","url":"/one"},{"label":"Two","url":"/two"}]'; - let result = transform.deserialize(serialized); - - expect(result.length).to.equal(2); - expect(result[0]).to.be.instanceof(NavigationItem); - expect(result[0].get('label')).to.equal('One'); - expect(result[0].get('url')).to.equal('/one'); - expect(result[1]).to.be.instanceof(NavigationItem); - expect(result[1].get('label')).to.equal('Two'); - expect(result[1].get('url')).to.equal('/two'); - }); - - it('serializes array of NavigationItems', function () { - let transform = this.subject(); - let deserialized = emberA([ - NavigationItem.create({label: 'One', url: '/one'}), - NavigationItem.create({label: 'Two', url: '/two'}) - ]); - let result = transform.serialize(deserialized); - - expect(result).to.equal('[{"label":"One","url":"/one"},{"label":"Two","url":"/two"}]'); - }); - } -); diff --git a/core/client/tests/unit/transforms/slack-settings-test.js b/core/client/tests/unit/transforms/slack-settings-test.js deleted file mode 100644 index dc154c06d1..0000000000 --- a/core/client/tests/unit/transforms/slack-settings-test.js +++ /dev/null @@ -1,37 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModule, it } from 'ember-mocha'; -import Ember from 'ember'; -import SlackIntegration from 'ghost/models/slack-integration'; - -const emberA = Ember.A; - -describeModule( - 'transform:slack-settings', - 'Unit: Transform: slack-settings', - { - // Specify the other units that are required for this test. - // needs: ['transform:foo'] - }, - function() { - it('deserializes settings json', function () { - let transform = this.subject(); - let serialized = '[{"url":"http://myblog.com/blogpost1"}]'; - let result = transform.deserialize(serialized); - - expect(result.length).to.equal(1); - expect(result[0]).to.be.instanceof(SlackIntegration); - expect(result[0].get('url')).to.equal('http://myblog.com/blogpost1'); - }); - - it('serializes array of Slack settings', function () { - let transform = this.subject(); - let deserialized = emberA([ - SlackIntegration.create({url: 'http://myblog.com/blogpost1'}) - ]); - let result = transform.serialize(deserialized); - - expect(result).to.equal('[{"url":"http://myblog.com/blogpost1"}]'); - }); - } -); diff --git a/core/client/tests/unit/transforms/twitter-url-user-test.js b/core/client/tests/unit/transforms/twitter-url-user-test.js deleted file mode 100644 index 10a4268adc..0000000000 --- a/core/client/tests/unit/transforms/twitter-url-user-test.js +++ /dev/null @@ -1,32 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { describeModule, it } from 'ember-mocha'; -import Ember from 'ember'; - -const emberA = Ember.A; - -describeModule( - 'transform:twitter-url-user', - 'Unit: Transform: twitter-url-user', - { - // Specify the other units that are required for this test. - // needs: ['transform:foo'] - }, - function() { - it('deserializes twitter url', function () { - let transform = this.subject(); - let serialized = '@testuser'; - let result = transform.deserialize(serialized); - - expect(result).to.equal('https://twitter.com/testuser'); - }); - - it('serializes url to twitter username', function () { - let transform = this.subject(); - let deserialized = 'https://twitter.com/testuser'; - let result = transform.serialize(deserialized); - - expect(result).to.equal('@testuser'); - }); - } -); diff --git a/core/client/tests/unit/utils/ghost-paths-test.js b/core/client/tests/unit/utils/ghost-paths-test.js deleted file mode 100644 index a2b4f05de2..0000000000 --- a/core/client/tests/unit/utils/ghost-paths-test.js +++ /dev/null @@ -1,58 +0,0 @@ -import ghostPaths from 'ghost/utils/ghost-paths'; - -describe('Unit: Util: ghost-paths', function () { - describe('join', function () { - let {join} = ghostPaths().url; - - it('should join two or more paths, normalizing slashes', function () { - let path; - - path = join('/one/', '/two/'); - expect(path).to.equal('/one/two/'); - - path = join('/one', '/two/'); - expect(path).to.equal('/one/two/'); - - path = join('/one/', 'two/'); - expect(path).to.equal('/one/two/'); - - path = join('/one/', 'two/', '/three/'); - expect(path).to.equal('/one/two/three/'); - - path = join('/one/', 'two', 'three/'); - expect(path).to.equal('/one/two/three/'); - }); - - it('should not change the slash at the beginning', function () { - let path; - - path = join('one/'); - expect(path).to.equal('one/'); - path = join('one/', 'two'); - expect(path).to.equal('one/two/'); - path = join('/one/', 'two'); - expect(path).to.equal('/one/two/'); - path = join('one/', 'two', 'three'); - expect(path).to.equal('one/two/three/'); - path = join('/one/', 'two', 'three'); - expect(path).to.equal('/one/two/three/'); - }); - - it('should always return a slash at the end', function () { - let path; - - path = join(); - expect(path).to.equal('/'); - path = join(''); - expect(path).to.equal('/'); - path = join('one'); - expect(path).to.equal('one/'); - path = join('one/'); - expect(path).to.equal('one/'); - path = join('one', 'two'); - expect(path).to.equal('one/two/'); - path = join('one', 'two/'); - expect(path).to.equal('one/two/'); - }); - }); -}); diff --git a/core/client/tests/unit/validators/nav-item-test.js b/core/client/tests/unit/validators/nav-item-test.js deleted file mode 100644 index 11adff632b..0000000000 --- a/core/client/tests/unit/validators/nav-item-test.js +++ /dev/null @@ -1,101 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import validator from 'ghost/validators/nav-item'; -import NavItem from 'ghost/models/navigation-item'; - -const testInvalidUrl = function (url) { - let navItem = NavItem.create({url}); - - validator.check(navItem, 'url'); - - expect(validator.get('passed'), `"${url}" passed`).to.be.false; - expect(navItem.get('errors').errorsFor('url')).to.deep.equal([{ - attribute: 'url', - message: 'You must specify a valid URL or relative path' - }]); - expect(navItem.get('hasValidated')).to.include('url'); -}; - -const testValidUrl = function (url) { - let navItem = NavItem.create({url}); - - validator.check(navItem, 'url'); - - expect(validator.get('passed'), `"${url}" failed`).to.be.true; - expect(navItem.get('hasValidated')).to.include('url'); -}; - -describe('Unit: Validator: nav-item', function () { - it('requires label presence', function () { - let navItem = NavItem.create(); - - validator.check(navItem, 'label'); - - expect(validator.get('passed')).to.be.false; - expect(navItem.get('errors').errorsFor('label')).to.deep.equal([{ - attribute: 'label', - message: 'You must specify a label' - }]); - expect(navItem.get('hasValidated')).to.include('label'); - }); - - it('requires url presence', function () { - let navItem = NavItem.create(); - - validator.check(navItem, 'url'); - - expect(validator.get('passed')).to.be.false; - expect(navItem.get('errors').errorsFor('url')).to.deep.equal([{ - attribute: 'url', - message: 'You must specify a URL or relative path' - }]); - expect(navItem.get('hasValidated')).to.include('url'); - }); - - it('fails on invalid url values', function () { - let invalidUrls = [ - 'test@example.com', - '/has spaces', - 'no-leading-slash', - 'http://example.com/with spaces' - ]; - - invalidUrls.forEach(function (url) { - testInvalidUrl(url); - }); - }); - - it('passes on valid url values', function () { - let validUrls = [ - 'http://localhost:2368', - 'http://localhost:2368/some-path', - 'https://localhost:2368/some-path', - '//localhost:2368/some-path', - 'http://localhost:2368/#test', - 'http://localhost:2368/?query=test&another=example', - 'http://localhost:2368/?query=test&another=example#test', - 'tel:01234-567890', - 'mailto:test@example.com', - 'http://some:user@example.com:1234', - '/relative/path' - ]; - - validUrls.forEach(function (url) { - testValidUrl(url); - }); - }); - - it('validates url and label by default', function () { - let navItem = NavItem.create(); - - validator.check(navItem); - - expect(navItem.get('errors').errorsFor('label')).to.not.be.empty; - expect(navItem.get('errors').errorsFor('url')).to.not.be.empty; - expect(validator.get('passed')).to.be.false; - }); -}); diff --git a/core/client/tests/unit/validators/slack-integration-test.js b/core/client/tests/unit/validators/slack-integration-test.js deleted file mode 100644 index 1912cdd304..0000000000 --- a/core/client/tests/unit/validators/slack-integration-test.js +++ /dev/null @@ -1,66 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import validator from 'ghost/validators/slack-integration'; -import SlackObject from 'ghost/models/slack-integration'; - -const testInvalidUrl = function (url) { - let slackObject = SlackObject.create({url}); - - validator.check(slackObject, 'url'); - - expect(validator.get('passed'), `"${url}" passed`).to.be.false; - expect(slackObject.get('errors').errorsFor('url')).to.deep.equal([{ - attribute: 'url', - message: 'The URL must be in a format like https://hooks.slack.com/services/<your personal key>' - }]); - expect(slackObject.get('hasValidated')).to.include('url'); -}; - -const testValidUrl = function (url) { - let slackObject = SlackObject.create({url}); - - validator.check(slackObject, 'url'); - - expect(validator.get('passed'), `"${url}" failed`).to.be.true; - expect(slackObject.get('hasValidated')).to.include('url'); -}; - -describe('Unit: Validator: slack-integration', function () { - it('fails on invalid url values', function () { - let invalidUrls = [ - 'test@example.com', - '/has spaces', - 'no-leading-slash', - 'http://example.com/with spaces' - ]; - - invalidUrls.forEach(function (url) { - testInvalidUrl(url); - }); - }); - - it('passes on valid url values', function () { - let validUrls = [ - 'https://hooks.slack.com/services/;alskdjf', - 'https://hooks.slack.com/services/123445678', - 'https://hooks.slack.com/services/some_webhook' - ]; - - validUrls.forEach(function (url) { - testValidUrl(url); - }); - }); - - it('validates url by default', function () { - let slackObject = SlackObject.create(); - - validator.check(slackObject); - - expect(slackObject.get('errors').errorsFor('url')).to.be.empty; - expect(validator.get('passed')).to.be.true; - }); -}); diff --git a/core/client/tests/unit/validators/subscriber-test.js b/core/client/tests/unit/validators/subscriber-test.js deleted file mode 100644 index 0f8ba5fb4a..0000000000 --- a/core/client/tests/unit/validators/subscriber-test.js +++ /dev/null @@ -1,77 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import Ember from 'ember'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const {run} = Ember; - -const Subscriber = Ember.Object.extend(ValidationEngine, { - validationType: 'subscriber', - - email: null -}); - -describe('Unit: Validator: subscriber', function () { - it('validates email by default', function () { - let subscriber = Subscriber.create({}); - let properties = subscriber.get('validators.subscriber.properties'); - - console.log(subscriber); - - expect(properties, 'properties').to.include('email'); - }); - - it('passes with a valid email', function () { - let subscriber = Subscriber.create({email: 'test@example.com'}); - let passed = false; - - run(() => { - subscriber.validate({property: 'email'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(subscriber.get('hasValidated'), 'hasValidated').to.include('email'); - }); - - it('validates email presence', function () { - let subscriber = Subscriber.create({}); - let passed = false; - - run(() => { - subscriber.validate({property: 'email'}).then(() => { - passed = true; - }); - }); - - let emailErrors = subscriber.get('errors').errorsFor('email').get(0); - expect(emailErrors.attribute, 'errors.email.attribute').to.equal('email'); - expect(emailErrors.message, 'errors.email.message').to.equal('Please enter an email.'); - - expect(passed, 'passed').to.be.false; - expect(subscriber.get('hasValidated'), 'hasValidated').to.include('email'); - }); - - it('validates email', function () { - let subscriber = Subscriber.create({email: 'foo'}); - let passed = false; - - run(() => { - subscriber.validate({property: 'email'}).then(() => { - passed = true; - }); - }); - - let emailErrors = subscriber.get('errors').errorsFor('email').get(0); - expect(emailErrors.attribute, 'errors.email.attribute').to.equal('email'); - expect(emailErrors.message, 'errors.email.message').to.equal('Invalid email.'); - - expect(passed, 'passed').to.be.false; - expect(subscriber.get('hasValidated'), 'hasValidated').to.include('email'); - }); -}); diff --git a/core/client/tests/unit/validators/tag-settings-test.js b/core/client/tests/unit/validators/tag-settings-test.js deleted file mode 100644 index cd9d77baa1..0000000000 --- a/core/client/tests/unit/validators/tag-settings-test.js +++ /dev/null @@ -1,306 +0,0 @@ -/* jshint expr:true */ -import { expect } from 'chai'; -import { - describe, - it -} from 'mocha'; -import sinon from 'sinon'; -import Ember from 'ember'; -// import validator from 'ghost/validators/tag-settings'; -import ValidationEngine from 'ghost/mixins/validation-engine'; - -const {run} = Ember; - -const Tag = Ember.Object.extend(ValidationEngine, { - validationType: 'tag', - - name: null, - description: null, - metaTitle: null, - metaDescription: null -}); - -// TODO: These tests have way too much duplication, consider creating test -// helpers for validations - -// TODO: Move testing of validation-engine behaviour into validation-engine-test -// and replace these tests with specific validator tests - -describe('Unit: Validator: tag-settings', function () { - it('validates all fields by default', function () { - let tag = Tag.create({}); - let properties = tag.get('validators.tag.properties'); - - // TODO: This is checking implementation details rather than expected - // behaviour. Replace once we have consistent behaviour (see below) - expect(properties, 'properties').to.include('name'); - expect(properties, 'properties').to.include('slug'); - expect(properties, 'properties').to.include('description'); - expect(properties, 'properties').to.include('metaTitle'); - expect(properties, 'properties').to.include('metaDescription'); - - // TODO: .validate (and by extension .save) doesn't currently affect - // .hasValidated - it would be good to make this consistent. - // The following tests currently fail: - // - // run(() => { - // tag.validate(); - // }); - // - // expect(tag.get('hasValidated'), 'hasValidated').to.include('name'); - // expect(tag.get('hasValidated'), 'hasValidated').to.include('description'); - // expect(tag.get('hasValidated'), 'hasValidated').to.include('metaTitle'); - // expect(tag.get('hasValidated'), 'hasValidated').to.include('metaDescription'); - }); - - it('passes with valid name', function () { - // longest valid name - let tag = Tag.create({name: (new Array(151).join('x'))}); - let passed = false; - - expect(tag.get('name').length, 'name length').to.equal(150); - - run(() => { - tag.validate({property: 'name'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(tag.get('hasValidated'), 'hasValidated').to.include('name'); - }); - - it('validates name presence', function () { - let tag = Tag.create(); - let passed = false; - let nameErrors; - - // TODO: validator is currently a singleton meaning state leaks - // between all objects that use it. Each object should either - // get it's own validator instance or validator objects should not - // contain state. The following currently fails: - // - // let validator = tag.get('validators.tag') - // expect(validator.get('passed'), 'passed').to.be.false; - - run(() => { - tag.validate({property: 'name'}).then(() => { - passed = true; - }); - }); - - nameErrors = tag.get('errors').errorsFor('name').get(0); - expect(nameErrors.attribute, 'errors.name.attribute').to.equal('name'); - expect(nameErrors.message, 'errors.name.message').to.equal('You must specify a name for the tag.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('name'); - }); - - it('validates names starting with a comma', function () { - let tag = Tag.create({name: ',test'}); - let passed = false; - let nameErrors; - - run(() => { - tag.validate({property: 'name'}).then(() => { - passed = true; - }); - }); - - nameErrors = tag.get('errors').errorsFor('name').get(0); - expect(nameErrors.attribute, 'errors.name.attribute').to.equal('name'); - expect(nameErrors.message, 'errors.name.message').to.equal('Tag names can\'t start with commas.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('name'); - }); - - it('validates name length', function () { - // shortest invalid name - let tag = Tag.create({name: (new Array(152).join('x'))}); - let passed = false; - let nameErrors; - - expect(tag.get('name').length, 'name length').to.equal(151); - - run(() => { - tag.validate({property: 'name'}).then(() => { - passed = true; - }); - }); - - nameErrors = tag.get('errors').errorsFor('name')[0]; - expect(nameErrors.attribute, 'errors.name.attribute').to.equal('name'); - expect(nameErrors.message, 'errors.name.message').to.equal('Tag names cannot be longer than 150 characters.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('name'); - }); - - it('passes with valid slug', function () { - // longest valid slug - let tag = Tag.create({slug: (new Array(151).join('x'))}); - let passed = false; - - expect(tag.get('slug').length, 'slug length').to.equal(150); - - run(() => { - tag.validate({property: 'slug'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(tag.get('hasValidated'), 'hasValidated').to.include('slug'); - }); - - it('validates slug length', function () { - // shortest invalid slug - let tag = Tag.create({slug: (new Array(152).join('x'))}); - let passed = false; - let slugErrors; - - expect(tag.get('slug').length, 'slug length').to.equal(151); - - run(() => { - tag.validate({property: 'slug'}).then(() => { - passed = true; - }); - }); - - slugErrors = tag.get('errors').errorsFor('slug')[0]; - expect(slugErrors.attribute, 'errors.slug.attribute').to.equal('slug'); - expect(slugErrors.message, 'errors.slug.message').to.equal('URL cannot be longer than 150 characters.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('slug'); - }); - - it('passes with a valid description', function () { - // longest valid description - let tag = Tag.create({description: (new Array(201).join('x'))}); - let passed = false; - - expect(tag.get('description').length, 'description length').to.equal(200); - - run(() => { - tag.validate({property: 'description'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(tag.get('hasValidated'), 'hasValidated').to.include('description'); - }); - - it('validates description length', function () { - // shortest invalid description - let tag = Tag.create({description: (new Array(202).join('x'))}); - let passed = false; - let errors; - - expect(tag.get('description').length, 'description length').to.equal(201); - - run(() => { - tag.validate({property: 'description'}).then(() => { - passed = true; - }); - }); - - errors = tag.get('errors').errorsFor('description')[0]; - expect(errors.attribute, 'errors.description.attribute').to.equal('description'); - expect(errors.message, 'errors.description.message').to.equal('Description cannot be longer than 200 characters.'); - - // TODO: tag.errors appears to be a singleton and previous errors are - // not cleared despite creating a new tag object - // - // console.log(JSON.stringify(tag.get('errors'))); - // expect(tag.get('errors.length')).to.equal(1); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('description'); - }); - - // TODO: we have both metaTitle and metaTitle property names on the - // model/validator respectively - this should be standardised - it('passes with a valid metaTitle', function () { - // longest valid metaTitle - let tag = Tag.create({metaTitle: (new Array(151).join('x'))}); - let passed = false; - - expect(tag.get('metaTitle').length, 'metaTitle length').to.equal(150); - - run(() => { - tag.validate({property: 'metaTitle'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(tag.get('hasValidated'), 'hasValidated').to.include('metaTitle'); - }); - - it('validates metaTitle length', function () { - // shortest invalid metaTitle - let tag = Tag.create({metaTitle: (new Array(152).join('x'))}); - let passed = false; - let errors; - - expect(tag.get('metaTitle').length, 'metaTitle length').to.equal(151); - - run(() => { - tag.validate({property: 'metaTitle'}).then(() => { - passed = true; - }); - }); - - errors = tag.get('errors').errorsFor('metaTitle')[0]; - expect(errors.attribute, 'errors.metaTitle.attribute').to.equal('metaTitle'); - expect(errors.message, 'errors.metaTitle.message').to.equal('Meta Title cannot be longer than 150 characters.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('metaTitle'); - }); - - // TODO: we have both metaDescription and metaDescription property names on - // the model/validator respectively - this should be standardised - it('passes with a valid metaDescription', function () { - // longest valid description - let tag = Tag.create({metaDescription: (new Array(201).join('x'))}); - let passed = false; - - expect(tag.get('metaDescription').length, 'metaDescription length').to.equal(200); - - run(() => { - tag.validate({property: 'metaDescription'}).then(() => { - passed = true; - }); - }); - - expect(passed, 'passed').to.be.true; - expect(tag.get('hasValidated'), 'hasValidated').to.include('metaDescription'); - }); - - it('validates metaDescription length', function () { - // shortest invalid metaDescription - let tag = Tag.create({metaDescription: (new Array(202).join('x'))}); - let passed = false; - let errors; - - expect(tag.get('metaDescription').length, 'metaDescription length').to.equal(201); - - run(() => { - tag.validate({property: 'metaDescription'}).then(() => { - passed = true; - }); - }); - - errors = tag.get('errors').errorsFor('metaDescription')[0]; - expect(errors.attribute, 'errors.metaDescription.attribute').to.equal('metaDescription'); - expect(errors.message, 'errors.metaDescription.message').to.equal('Meta Description cannot be longer than 200 characters.'); - - expect(passed, 'passed').to.be.false; - expect(tag.get('hasValidated'), 'hasValidated').to.include('metaDescription'); - }); -}); From 76826a385d95448376719b03f8375e49c0985092 Mon Sep 17 00:00:00 2001 From: Kevin Ansfield <kevin@lookingsideways.co.uk> Date: Wed, 18 May 2016 12:36:12 +0100 Subject: [PATCH 2/2] Add Admin-Client as submodule at `core/client` no issue - import the now separated [Ghost-Admin](https://github.com/TryGhost/Ghost-Admin) project as a submodule in it's original location of `core/client` --- .gitmodules | 3 +++ core/client | 1 + 2 files changed, 4 insertions(+) create mode 160000 core/client diff --git a/.gitmodules b/.gitmodules index 63888090db..f100604701 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,3 +1,6 @@ [submodule "content/themes/casper"] path = content/themes/casper url = https://github.com/TryGhost/Casper.git +[submodule "core/client"] + path = core/client + url = https://github.com/TryGhost/Ghost-Admin.git diff --git a/core/client b/core/client new file mode 160000 index 0000000000..39622c4e28 --- /dev/null +++ b/core/client @@ -0,0 +1 @@ +Subproject commit 39622c4e284554d0ac1f6dbe3de90a6e6943a6ed