├── .editorconfig ├── .ember-cli ├── .eslintignore ├── .eslintrc.js ├── .gitignore ├── .netlifyheaders ├── .netlifyredirects ├── .travis.yml ├── .watchmanconfig ├── README.md ├── app.json ├── app ├── adapters │ ├── application.js │ └── version.js ├── app.js ├── components │ ├── .gitkeep │ ├── chapter-links.js │ ├── dropdown-header.js │ ├── guides-article.js │ ├── link-to.js │ ├── search-input.js │ ├── search-result.js │ └── table-of-contents.js ├── controllers │ ├── .gitkeep │ ├── application.js │ └── version.js ├── helpers │ ├── .gitkeep │ └── shorten-version.js ├── index.html ├── initializers │ └── regiser-showdown-extenions.js ├── locations │ └── trailing-history.js ├── models │ ├── .gitkeep │ ├── content.js │ ├── page.js │ └── version.js ├── resolver.js ├── router.js ├── routes │ ├── .gitkeep │ ├── application.js │ ├── error.js │ ├── index.js │ ├── version.js │ └── version │ │ ├── index.js │ │ └── show.js ├── services │ ├── head-data.js │ ├── page.js │ └── search.js ├── styles │ ├── _code-block-file-name.scss │ ├── _prism-overrides.scss │ ├── _search-input.scss │ ├── _table-of-contents.scss │ └── app.scss └── templates │ ├── application.hbs │ ├── components │ ├── .gitkeep │ ├── chapter-links.hbs │ ├── dropdown-header.hbs │ ├── guides-article.hbs │ ├── search-input.hbs │ ├── search-result.hbs │ └── table-of-contents.hbs │ ├── error.hbs │ ├── version.hbs │ └── version │ ├── index.hbs │ └── show.hbs ├── appveyor.yml ├── config ├── deploy.js ├── environment.js └── targets.js ├── ember-cli-build.js ├── lib └── content-guides-generator │ ├── index.js │ └── package.json ├── package-lock.json ├── package.json ├── public ├── fonts │ ├── config.json │ ├── fontello.eot │ ├── fontello.svg │ ├── fontello.ttf │ ├── fontello.woff │ └── fontello.woff2 ├── images │ └── logos │ │ ├── ember.png │ │ └── search-by-algolia.svg └── robots.txt ├── static.json ├── testem.js ├── tests ├── acceptance │ ├── cookbook-test.js │ ├── current-url-test.js │ ├── error-page-test.js │ ├── meta-data-test.js │ ├── previous-next-links-test.js │ ├── table-of-contents-test.js │ ├── version-menu-test.js │ └── visual-regression-test.js ├── helpers │ ├── .gitkeep │ └── start-app.js ├── index.html ├── integration │ ├── components │ │ ├── dropdown-header-test.js │ │ └── search-input-test.js │ └── helpers │ │ └── shorten-version-test.js ├── test-helper.js └── unit │ └── services │ ├── head-data-test.js │ └── search-test.js └── vendor ├── .gitkeep └── ember-cli-spinkit ├── styles └── spinkit-cube-grid.css └── templates └── components └── spinkit-cube-grid.hbs /.editorconfig: -------------------------------------------------------------------------------- 1 | # EditorConfig helps developers define and maintain consistent 2 | # coding styles between different editors and IDEs 3 | # editorconfig.org 4 | 5 | root = true 6 | 7 | 8 | [*] 9 | end_of_line = lf 10 | charset = utf-8 11 | trim_trailing_whitespace = true 12 | insert_final_newline = true 13 | indent_style = space 14 | indent_size = 2 15 | 16 | [*.hbs] 17 | insert_final_newline = false 18 | 19 | [*.{diff,md}] 20 | trim_trailing_whitespace = false 21 | -------------------------------------------------------------------------------- /.ember-cli: -------------------------------------------------------------------------------- 1 | { 2 | /** 3 | Ember CLI sends analytics information by default. The data is completely 4 | anonymous, but there are times when you might want to disable this behavior. 5 | 6 | Setting `disableAnalytics` to true will prevent any data from being sent. 7 | */ 8 | "disableAnalytics": false 9 | } 10 | -------------------------------------------------------------------------------- /.eslintignore: -------------------------------------------------------------------------------- 1 | /blueprints/*/files/**/*.js 2 | # unconventional js 3 | /blueprints/*/files/ 4 | /vendor/ 5 | 6 | # compiled output 7 | /dist/ 8 | /tmp/ 9 | 10 | # dependencies 11 | /bower_components/ 12 | 13 | # misc 14 | /coverage/ 15 | !.* 16 | 17 | # ember-try 18 | /.node_modules.ember-try/ 19 | /bower.json.ember-try 20 | /package.json.ember-try 21 | -------------------------------------------------------------------------------- /.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | parserOptions: { 4 | ecmaVersion: 2017, 5 | sourceType: 'module' 6 | }, 7 | plugins: [ 8 | 'ember' 9 | ], 10 | extends: [ 11 | 'eslint:recommended', 12 | 'plugin:ember/recommended' 13 | ], 14 | env: { 15 | browser: true 16 | }, 17 | rules: { 18 | }, 19 | globals: { 20 | Prism: true, 21 | compareVersions: true 22 | }, 23 | overrides: [ 24 | // node files 25 | { 26 | files: [ 27 | 'ember-cli-build.js', 28 | 'testem.js', 29 | 'blueprints/*/index.js', 30 | 'config/**/*.js', 31 | 'lib/*/index.js' 32 | ], 33 | parserOptions: { 34 | sourceType: 'script', 35 | ecmaVersion: 2015 36 | }, 37 | env: { 38 | browser: false, 39 | node: true 40 | } 41 | }, 42 | 43 | // test files 44 | { 45 | files: ['tests/**/*.js'], 46 | excludedFiles: ['tests/dummy/**/*.js'], 47 | env: { 48 | embertest: true 49 | } 50 | } 51 | ] 52 | }; 53 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/ignore-files/ for more about ignoring files. 2 | 3 | # compiled output 4 | /dist 5 | /tmp 6 | 7 | # dependencies 8 | /node_modules 9 | /bower_components 10 | 11 | # misc 12 | /.sass-cache 13 | /connect.lock 14 | /coverage/* 15 | /libpeerconnection.log 16 | npm-debug.log* 17 | yarn-error.log 18 | testem.log 19 | .DS_Store 20 | 21 | # ember-try 22 | .node_modules.ember-try/ 23 | bower.json.ember-try 24 | package.json.ember-try 25 | 26 | # Deployment credentials 27 | config/credentials.json 28 | 29 | .vscode 30 | -------------------------------------------------------------------------------- /.netlifyheaders: -------------------------------------------------------------------------------- 1 | /*/ 2 | Cache-Control: public, max-age=3600 3 | 4 | /content/* 5 | Cache-Control: public, max-age=3600 6 | 7 | /assets/* 8 | Cache-Control: public, max-age=31536000 9 | -------------------------------------------------------------------------------- /.netlifyredirects: -------------------------------------------------------------------------------- 1 | / /release/ 2 | /current/* /release/:splat 3 | /* /_empty.html 200 4 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | --- 2 | language: node_js 3 | node_js: 4 | - "8" 5 | 6 | sudo: required 7 | dist: trusty 8 | 9 | branches: 10 | only: 11 | - master 12 | 13 | addons: 14 | chrome: stable 15 | 16 | cache: 17 | directories: 18 | - $HOME/.npm 19 | 20 | env: 21 | global: 22 | # See https://git.io/vdao3 for details. 23 | - JOBS=1 24 | - PERCY_TOKEN=877df6aad8486060f69a34864b6cd33f870633743b23411343737c46a875a762 25 | 26 | before_install: 27 | - npm config set spin false 28 | 29 | script: 30 | - npm run lint:js 31 | - ember build --prod 32 | - npm test 33 | 34 | after_failure: 35 | - wget https://gist.github.com/sivakumar-kailasam/730c6d2e5d0847cf9e0bc17df386c852/raw/discord_notifier.sh 36 | - chmod +x discord_notifier.sh 37 | - ./discord_notifier.sh 38 | -------------------------------------------------------------------------------- /.watchmanconfig: -------------------------------------------------------------------------------- 1 | { 2 | "ignore_dirs": ["tmp", "dist"] 3 | } 4 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # [deprecated] Ember Guides App 2 | 3 | This repository was the source for the Ember App that used to power the [Ember.js Guides](https://guides.emberjs.com/release/). This work has since been extracted into [Guidemaker](https://github.com/empress/guidemaker) and the [Guidemaker Ember Template](https://github.com/ember-learn/guidemaker-ember-template). 4 | 5 | The source for the guides is still avaliable in the same place in the [Guides Source](https://github.com/ember-learn/guides-source) repository. 6 | 7 | ## Contributing 8 | 9 | All further work will be done on [Guidemaker](https://github.com/empress/guidemaker), the [Guidemaker Ember Template](https://github.com/ember-learn/guidemaker-ember-template) and [Guides Source](https://github.com/ember-learn/guides-source). 10 | 11 | If you are interested in contributing then you can check out the [contributing instructions for the Guides Source](https://github.com/ember-learn/guides-source/blob/master/CONTRIBUTING.md). 12 | 13 | If you have questions you can join the #dev-ember-learning channel in the [Ember Community Discord](https://discordapp.com/invite/zT3asNS). 14 | -------------------------------------------------------------------------------- /app.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "guides-app", 3 | "description": "The ember.js guides app", 4 | "scripts": { 5 | }, 6 | "env": { 7 | }, 8 | "formation": { 9 | "web": { 10 | "quantity": 1 11 | } 12 | }, 13 | "addons": [ 14 | 15 | ], 16 | "buildpacks": [ 17 | { 18 | "url": "https://github.com/stonecircle/heroku-buildpack-ember-static" 19 | } 20 | ] 21 | } 22 | -------------------------------------------------------------------------------- /app/adapters/application.js: -------------------------------------------------------------------------------- 1 | import DS from 'ember-data'; 2 | 3 | export default DS.JSONAPIAdapter.extend({ 4 | buildURL(modelName, id, snapshot, requestType, query) { 5 | let url; 6 | 7 | if (requestType === 'queryRecord') { 8 | url = [modelName, query.version, `${query.path}.json`]; 9 | } else if(requestType === 'query' && modelName === 'page') { 10 | url = ['content', query.version, 'pages.json']; 11 | } else { 12 | return this._super(...arguments); 13 | } 14 | 15 | let host = this.host; 16 | let prefix = this.urlPrefix(); 17 | 18 | if (prefix) { url.unshift(prefix); } 19 | 20 | url = url.join('/'); 21 | if (!host && url && url.charAt(0) !== '/') { 22 | url = '/' + url; 23 | } 24 | 25 | return url; 26 | }, 27 | }); 28 | -------------------------------------------------------------------------------- /app/adapters/version.js: -------------------------------------------------------------------------------- 1 | import DS from 'ember-data'; 2 | 3 | export default DS.JSONAPIAdapter.extend({ 4 | buildURL() { 5 | let url = ['content', 'versions.json']; 6 | let host = this.host; 7 | let prefix = this.urlPrefix(); 8 | 9 | if (prefix) { url.unshift(prefix); } 10 | 11 | url = url.join('/'); 12 | if (!host && url && url.charAt(0) !== '/') { 13 | url = '/' + url; 14 | } 15 | 16 | return url; 17 | }, 18 | }); 19 | -------------------------------------------------------------------------------- /app/app.js: -------------------------------------------------------------------------------- 1 | import Application from '@ember/application'; 2 | import Resolver from './resolver'; 3 | import loadInitializers from 'ember-load-initializers'; 4 | import config from './config/environment'; 5 | 6 | const App = Application.extend({ 7 | modulePrefix: config.modulePrefix, 8 | podModulePrefix: config.podModulePrefix, 9 | Resolver 10 | }); 11 | 12 | loadInitializers(App, config.modulePrefix); 13 | 14 | export default App; 15 | -------------------------------------------------------------------------------- /app/components/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/components/.gitkeep -------------------------------------------------------------------------------- /app/components/chapter-links.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import { get, computed } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | 5 | export default Component.extend({ 6 | tagName: 'footer', 7 | 8 | page: service(), 9 | 10 | nextSectionPage: computed('page.nextSection.pages.[]', function() { 11 | let pages = get(this, 'page.nextSection.pages'); 12 | 13 | if(pages && pages.length) { 14 | return pages[0]; 15 | } 16 | }), 17 | previousSectionPage: computed('page.previousSection.pages.[]', function() { 18 | let pages = get(this, 'page.previousSection.pages'); 19 | 20 | if(pages && pages.length) { 21 | return pages[pages.length - 1]; 22 | } 23 | }) 24 | }); 25 | -------------------------------------------------------------------------------- /app/components/dropdown-header.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | 3 | export default Component.extend({ 4 | classNames: ['ds-suggestion'] 5 | }); 6 | -------------------------------------------------------------------------------- /app/components/guides-article.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import { inject as service } from '@ember/service'; 3 | 4 | export default Component.extend({ 5 | tagName: 'article', 6 | classNames: 'chapter', 7 | page: service(), 8 | didRender() { 9 | 10 | let nodeList = this.$('pre:not(.no-line-numbers) > code'); 11 | 12 | if (nodeList) { 13 | nodeList.each((index, code) => { 14 | code.parentNode.classList.add('line-numbers'); 15 | }); 16 | } 17 | 18 | let filenameNodeList = this.$('pre > code[data-filename]'); 19 | 20 | if (filenameNodeList) { 21 | filenameNodeList.each((index, code) => { 22 | if (code.parentNode.parentNode.classList.contains('filename')) { 23 | //do nothing 24 | return; 25 | } 26 | 27 | let filename = code.attributes['data-filename'].value; 28 | let match = filename.match(/\.(\w+)$/); 29 | 30 | let ext = ''; 31 | 32 | if (match && match[1]) { 33 | ext = match[1]; 34 | } 35 | 36 | this.$(code.parentNode).wrap(`
`); 37 | 38 | this.$(code.parentNode.parentNode).prepend(this.$(`${code.attributes['data-filename'].value}`)); 39 | this.$(code.parentNode.parentNode).prepend('
'); 40 | }); 41 | } 42 | 43 | let allHeaders = document.querySelectorAll("h1, h2, h3, h4, h5, h6") 44 | 45 | for (var element of allHeaders) { 46 | if (element.id) { 47 | element.className = 'anchorable-toc' 48 | let link = document.createElement('a'); 49 | link.className = 'toc-anchor'; 50 | link.href = `#${element.id}`; 51 | element.insertBefore(link, element.firstElementChild); 52 | } 53 | } 54 | 55 | Prism.highlightAll(); 56 | 57 | /** 58 | * Prism doesn't support diff & a secondary language highlighting. 59 | * 60 | * So first, we let prism convert the content of `` blocks 61 | * from a string into a different dom structure 62 | * by calling `Prism.highlightAll()` 63 | * 64 | * In the following block, we add + & - symbols to the lines inside the 65 | * code block based on the data-diff attribute on the code tag. 66 | * e.g., data-diff="-4,+5,+6,+7" 67 | * 68 | **/ 69 | filenameNodeList.each((_, codeBlock) => { 70 | 71 | let diffInfo = codeBlock.attributes['data-diff'] ? codeBlock.attributes["data-diff"].value.split(',') : []; 72 | 73 | if (diffInfo.length === 0) { 74 | return; 75 | } 76 | 77 | let lines = codeBlock.innerHTML.split('\n'); 78 | 79 | diffInfo.forEach(pD => { 80 | let operator = pD[0]; 81 | let lineNo = +(pD.replace(operator, '')); 82 | let text = lines[lineNo - 1]; 83 | if (operator === '+') { 84 | lines[lineNo - 1] = `+${text}`; 85 | } else { 86 | lines[lineNo - 1] = `-${text}`; 87 | } 88 | }); 89 | codeBlock.innerHTML = lines.join('\n'); 90 | }) 91 | 92 | } 93 | }); 94 | -------------------------------------------------------------------------------- /app/components/link-to.js: -------------------------------------------------------------------------------- 1 | import LinkComponent from '@ember/routing/link-component'; 2 | 3 | export default LinkComponent.extend({ 4 | click() { 5 | if(window.scrollTo) { 6 | window.scrollTo(0,0); 7 | } 8 | 9 | if(document) { 10 | document.getElementById('toc-toggle').checked = false; 11 | } 12 | } 13 | }); 14 | -------------------------------------------------------------------------------- /app/components/search-input.js: -------------------------------------------------------------------------------- 1 | import { getOwner } from '@ember/application'; 2 | import Component from '@ember/component'; 3 | import { get, set } from '@ember/object'; 4 | import { and } from '@ember/object/computed'; 5 | import { inject as service } from '@ember/service'; 6 | import { task, timeout } from 'ember-concurrency'; 7 | 8 | const SEARCH_DEBOUNCE_PERIOD = 300; 9 | const SEARCH_CLOSE_PERIOD = 200; 10 | 11 | export default Component.extend({ 12 | classNames: ['search-input'], 13 | 14 | // page: service(), 15 | searchService: service('search'), 16 | 17 | _resultTetherConstraints: Object.freeze([ 18 | { 19 | to: 'window', 20 | pin: ['left','right'] 21 | } 22 | ]), 23 | 24 | _focused: false, 25 | 26 | init() { 27 | this._super(...arguments); 28 | const config = getOwner(this).resolveRegistration('config:environment'); 29 | this.deprecationsGuideURL = config['deprecationsGuideURL']; 30 | }, 31 | 32 | showDropdown: and('query', '_focused'), 33 | 34 | search: task(function * (query) { 35 | 36 | yield timeout(SEARCH_DEBOUNCE_PERIOD); 37 | 38 | set(this, 'query', query); 39 | 40 | // Hide and don't run query if there's no search query 41 | if (!query) { 42 | return set(this, '_focused', false); 43 | } 44 | 45 | // ensure search results are visible if the menu was previously closed above 46 | set(this, '_focused', true); 47 | 48 | yield get(this, 'searchService.search').perform(query, this.projectVersion); 49 | 50 | }).restartable(), 51 | 52 | closeMenu: task(function * () { 53 | yield timeout(SEARCH_CLOSE_PERIOD); 54 | 55 | set(this, '_focused', false); 56 | }), 57 | 58 | actions: { 59 | onfocus() { 60 | set(this, '_focused', true); 61 | }, 62 | 63 | onblur() { 64 | this.get('closeMenu').perform(); 65 | } 66 | 67 | } 68 | }); 69 | -------------------------------------------------------------------------------- /app/components/search-result.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import { computed } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | 5 | export default Component.extend({ 6 | classNames: ['ds-suggestion'], 7 | 8 | page: service(), 9 | sectionTitle: computed('result.path', function() { 10 | let sectionId = this.result.path.split('/')[0]; 11 | 12 | let section = this.page.pages.find((page) => page.id === sectionId); 13 | return section.title; 14 | }), 15 | 16 | pageHeading: computed('result._highlightResult.headings.[]', function() { 17 | return this.result._highlightResult.headings[0]; 18 | }), 19 | 20 | remainingHeadings: computed('result._highlightResult.headings.[]', function() { 21 | return this.result._highlightResult.headings; 22 | }) 23 | }); 24 | -------------------------------------------------------------------------------- /app/components/table-of-contents.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import { computed } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | 5 | export default Component.extend({ 6 | fastboot: service(), 7 | 8 | level: '0', 9 | tagName: 'ol', 10 | tocLevel: computed('level', function() { 11 | return `toc-level-${this.level}`; 12 | }), 13 | classNameBindings: ['tocLevel'], 14 | }); 15 | -------------------------------------------------------------------------------- /app/controllers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/controllers/.gitkeep -------------------------------------------------------------------------------- /app/controllers/application.js: -------------------------------------------------------------------------------- 1 | import Controller from '@ember/controller'; 2 | import { inject as service } from '@ember/service'; 3 | 4 | export default Controller.extend({ 5 | page: service(), 6 | }); 7 | -------------------------------------------------------------------------------- /app/controllers/version.js: -------------------------------------------------------------------------------- 1 | import Controller, { 2 | inject as controller, 3 | } from '@ember/controller'; 4 | import { get, computed } from '@ember/object'; 5 | import { alias } from '@ember/object/computed'; 6 | import { inject as service } from '@ember/service'; 7 | 8 | export default Controller.extend({ 9 | page: service(), 10 | application: controller(), 11 | 12 | pages: alias('model.pages'), 13 | 14 | versions: computed('application.model.allVersions.[]', function () { 15 | let allVersions = get(this, 'application.model.allVersions'); 16 | 17 | return allVersions.sort(compareVersions).reverse(); 18 | }), 19 | 20 | actions: { 21 | selectVersion(version) { 22 | // Navigate to same section/page if it exists 23 | const path = get(this, 'page.currentPage.url'); 24 | this.store.queryRecord('content', {version, path}).then(() => { 25 | this.transitionToRoute(`/${version}/${path}`); 26 | }).catch(() => { 27 | this.transitionToRoute('version', version); 28 | }) 29 | } 30 | } 31 | }); 32 | -------------------------------------------------------------------------------- /app/helpers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/helpers/.gitkeep -------------------------------------------------------------------------------- /app/helpers/shorten-version.js: -------------------------------------------------------------------------------- 1 | import { helper } from '@ember/component/helper'; 2 | 3 | export function shortenVersion([version='']) { 4 | return version.slice(version.indexOf('v') + 1 || 0, version.lastIndexOf('.') === version.indexOf('.') ? version.length : version.lastIndexOf('.')); 5 | } 6 | 7 | export default helper(shortenVersion); 8 | -------------------------------------------------------------------------------- /app/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | {{content-for "head"}} 9 | 10 | 11 | 12 | 13 | 14 | {{content-for "head-footer"}} 15 | 16 | 17 | {{content-for "body"}} 18 | 19 | 20 | 21 | 22 | {{content-for "body-footer"}} 23 | 24 | 25 | -------------------------------------------------------------------------------- /app/initializers/regiser-showdown-extenions.js: -------------------------------------------------------------------------------- 1 | import showdown from 'showdown'; 2 | import showdownSectionGroups from 'showdown-section-groups'; 3 | 4 | export function initialize() { 5 | showdown.subParser('ellipsis', function (text, options, globals) { 6 | text = globals.converter._dispatch('ellipsis.before', text, options, globals); 7 | text = globals.converter._dispatch('ellipsis.after', text, options, globals); 8 | return text; 9 | }); 10 | 11 | showdown.extension('showdown-section-groups', showdownSectionGroups); 12 | 13 | showdown.subParser('githubCodeBlocks', function (text, options, globals) { 14 | 'use strict'; 15 | 16 | // early exit if option is not enabled 17 | if (!options.ghCodeBlocks) { 18 | return text; 19 | } 20 | 21 | text = globals.converter._dispatch('githubCodeBlocks.before', text, options, globals); 22 | 23 | text += '¨0'; 24 | 25 | text = text.replace(/(?:^|\n)```(.*)\n([\s\S]*?)\n```/g, function (wholeMatch, languageBlock, codeblock) { 26 | var end = (options.omitExtraWLInCodeBlocks) ? '' : '\n'; 27 | 28 | // First parse the github code block 29 | codeblock = showdown.subParser('encodeCode')(codeblock, options, globals); 30 | codeblock = showdown.subParser('detab')(codeblock, options, globals); 31 | codeblock = codeblock.replace(/^\n+/g, ''); // trim leading newlines 32 | codeblock = codeblock.replace(/\n+$/g, ''); // trim trailing whitespace 33 | 34 | var match = languageBlock.match(/(\w+)(\s+{(.*)})?/); 35 | var languageString = ''; 36 | var attributeString = ''; 37 | 38 | if(match && match[1]) { 39 | languageString = ' class="' + match[1] + ' language-' + match[1] + '"'; 40 | } 41 | 42 | if (match && match[3]) { 43 | attributeString = match[3]; 44 | } 45 | 46 | codeblock = '
' + codeblock + end + '
'; 47 | 48 | codeblock = showdown.subParser('hashBlock')(codeblock, options, globals); 49 | 50 | // Since GHCodeblocks can be false positives, we need to 51 | // store the primitive text and the parsed text in a global var, 52 | // and then return a token 53 | return '\n\n¨G' + (globals.ghCodeBlocks.push({text: wholeMatch, codeblock: codeblock}) - 1) + 'G\n\n'; 54 | }); 55 | 56 | // attacklab: strip sentinel 57 | text = text.replace(/¨0/, ''); 58 | 59 | return globals.converter._dispatch('githubCodeBlocks.after', text, options, globals); 60 | }); 61 | } 62 | 63 | export default { 64 | name: 'register-showdown-extensions', 65 | initialize 66 | }; 67 | -------------------------------------------------------------------------------- /app/locations/trailing-history.js: -------------------------------------------------------------------------------- 1 | import HistoryLocation from '@ember/routing/history-location'; 2 | 3 | export default HistoryLocation.extend({ 4 | formatURL() { 5 | let url = this._super(...arguments); 6 | 7 | if (url.includes('#')) { 8 | return url.replace(/([^/])#(.*)/, '$1/#$2'); 9 | } else { 10 | return url.replace(/\/?$/, '/'); 11 | } 12 | } 13 | }); 14 | -------------------------------------------------------------------------------- /app/models/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/models/.gitkeep -------------------------------------------------------------------------------- /app/models/content.js: -------------------------------------------------------------------------------- 1 | import DS from 'ember-data'; 2 | 3 | export default DS.Model.extend({ 4 | title: DS.attr(), 5 | content: DS.attr(), 6 | description: DS.attr(), 7 | canonical: DS.attr(), 8 | redirect: DS.attr(), 9 | }); 10 | -------------------------------------------------------------------------------- /app/models/page.js: -------------------------------------------------------------------------------- 1 | import DS from 'ember-data'; 2 | 3 | export default DS.Model.extend({ 4 | title: DS.attr('string'), 5 | pages: DS.attr(), 6 | skipToc: DS.attr('boolean'), 7 | }); 8 | -------------------------------------------------------------------------------- /app/models/version.js: -------------------------------------------------------------------------------- 1 | import DS from 'ember-data'; 2 | 3 | export default DS.Model.extend({ 4 | allVersions: DS.attr(), 5 | currentVersion: DS.attr('string'), 6 | }); 7 | -------------------------------------------------------------------------------- /app/resolver.js: -------------------------------------------------------------------------------- 1 | import Resolver from 'ember-resolver'; 2 | 3 | export default Resolver; 4 | -------------------------------------------------------------------------------- /app/router.js: -------------------------------------------------------------------------------- 1 | import EmberRouter from '@ember/routing/router'; 2 | 3 | import { get } from '@ember/object'; 4 | import { inject as service } from '@ember/service'; 5 | import { scheduleOnce } from '@ember/runloop'; 6 | 7 | import config from './config/environment'; 8 | 9 | const Router = EmberRouter.extend({ 10 | location: config.locationType, 11 | rootURL: config.rootURL, 12 | 13 | metrics: service(), 14 | fastboot: service(), 15 | 16 | didTransition() { 17 | this._super(...arguments); 18 | this._trackPage(); 19 | }, 20 | 21 | _trackPage() { 22 | if(get(this, 'fastboot.isFastBoot')) { 23 | return; 24 | } 25 | 26 | scheduleOnce('afterRender', this, () => { 27 | const page = this.url; 28 | const title = this.getWithDefault('currentRouteName', 'unknown'); 29 | 30 | // this is constant for this app and is only used to identify page views in the GA dashboard 31 | const hostname = 'guides.emberjs.com'; 32 | 33 | this.metrics.trackPage({ page, title, hostname }); 34 | }); 35 | }, 36 | }); 37 | 38 | Router.map(function() { 39 | this.route('version', { path: ':version' }, function() { 40 | this.route('show', { path: '*path' }); 41 | }); 42 | }); 43 | 44 | export default Router; 45 | -------------------------------------------------------------------------------- /app/routes/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/routes/.gitkeep -------------------------------------------------------------------------------- /app/routes/application.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | 3 | export default Route.extend({ 4 | model() { 5 | return this.store.findRecord('version', 'versions'); 6 | } 7 | }); 8 | -------------------------------------------------------------------------------- /app/routes/error.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | 3 | export default Route.extend({ 4 | classNames: ['x404'], 5 | }); 6 | -------------------------------------------------------------------------------- /app/routes/index.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | 3 | export default Route.extend({ 4 | redirect() { 5 | this.transitionTo('version', 'release'); 6 | } 7 | }); 8 | -------------------------------------------------------------------------------- /app/routes/version.js: -------------------------------------------------------------------------------- 1 | import { get, set } from '@ember/object'; 2 | import Route from '@ember/routing/route'; 3 | import { inject as service } from '@ember/service'; 4 | import { hash } from 'rsvp'; 5 | 6 | export default Route.extend({ 7 | page: service(), 8 | model(params) { 9 | let applicationModel = this.modelFor('application'); 10 | let currentVersion = get(applicationModel, 'currentVersion'); 11 | let version = params.version; 12 | 13 | if (params.version === 'release') { 14 | version = currentVersion; 15 | } 16 | 17 | return hash({ 18 | pages: this.store.query('page', { version }), 19 | allVersions: get(applicationModel, 'allVersions'), 20 | currentVersion, 21 | version: version, 22 | }); 23 | }, 24 | 25 | afterModel(model) { 26 | set(this.page, 'pages', get(model, 'pages')); 27 | } 28 | }); 29 | -------------------------------------------------------------------------------- /app/routes/version/index.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | import { inject as service } from '@ember/service'; 3 | import { get, set } from '@ember/object'; 4 | import { hash } from 'rsvp'; 5 | 6 | export default Route.extend({ 7 | page: service(), 8 | model() { 9 | let { version, currentVersion } = this.modelFor('version'); 10 | 11 | return hash({ 12 | content: this.store.queryRecord('content', { 13 | path: 'index', 14 | version, 15 | }), 16 | pages: this.store.query('page', { version }), 17 | version, 18 | currentVersion, 19 | }); 20 | }, 21 | 22 | afterModel(model) { 23 | let content = get(model, 'content'); 24 | set(this.page, 'content', content); 25 | let version = get(model, 'version'); 26 | set(this.page, 'currentVersion', version); 27 | } 28 | }); 29 | -------------------------------------------------------------------------------- /app/routes/version/show.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | import { inject as service } from '@ember/service'; 3 | import { get, set } from '@ember/object'; 4 | import { hash } from 'rsvp'; 5 | 6 | export default Route.extend({ 7 | page: service(), 8 | model(params) { 9 | const path = params.path.replace(/\/$/, ''); 10 | 11 | if (path === 'index') { 12 | return this.transitionTo('version'); 13 | } 14 | 15 | if (path.endsWith('/index')) { 16 | return this.transitionTo('version.show', path.replace(/\/index$/, '')) 17 | } 18 | 19 | const { 20 | version, 21 | currentVersion, 22 | pages, 23 | } = this.modelFor('version'); 24 | 25 | let contentPromise = this.store.queryRecord('content', { 26 | path, 27 | version 28 | }) 29 | .catch((e) => { 30 | if (['404', '403'].includes(get(e, 'errors.0.status'))) { 31 | return this.store.queryRecord('content', { 32 | path: `${path}/index`, 33 | version 34 | }); 35 | } 36 | throw e; 37 | }); 38 | 39 | return hash({ 40 | content: contentPromise, 41 | pages, 42 | path, 43 | version, 44 | currentVersion, 45 | }) 46 | }, 47 | afterModel(model) { 48 | if(model.content.redirect) { 49 | this.transitionTo('version.show', model.content.redirect) 50 | } 51 | 52 | let content = get(model, 'content'); 53 | set(this.page, 'content', content); 54 | let version = get(model, 'version'); 55 | set(this.page, 'currentVersion', version); 56 | } 57 | }); 58 | -------------------------------------------------------------------------------- /app/services/head-data.js: -------------------------------------------------------------------------------- 1 | import HeadData from 'ember-meta/services/head-data'; 2 | import { computed } from '@ember/object'; 3 | import { getOwner } from '@ember/application'; 4 | import { inject as service } from '@ember/service'; 5 | import { isPresent } from '@ember/utils'; 6 | 7 | import config from '../config/environment'; 8 | 9 | export default HeadData.extend({ 10 | page: service(), 11 | currentRouteModel: computed('routeName', function() { 12 | return getOwner(this).lookup(`route:${this.routeName}`).get('currentModel.content'); 13 | }), 14 | 15 | title: computed('routeName', 'page.{currentPage,currentSection}', function() { 16 | if(!this.page.currentPage || !this.page.currentSection) { 17 | return 'Ember Guides'; 18 | } 19 | 20 | return `${this.page.currentPage.title} - ${this.page.currentSection.title} - Ember Guides` 21 | }), 22 | 23 | description: computed('routeName', function() { 24 | return this.getWithDefault('currentRouteModel.description', config['ember-meta'].description); 25 | }), 26 | 27 | slug: computed('routeName', function() { 28 | // if there is no current model 29 | if (!this.currentRouteModel) { 30 | return null; 31 | } 32 | 33 | if(this.currentRouteModel.id === 'index') { 34 | return this.page.currentVersion; 35 | } 36 | 37 | return `${this.page.currentVersion}/${this.currentRouteModel.id.replace(/\/index$/, '')}`; 38 | }), 39 | 40 | canonical: computed('routeName', function() { 41 | // if there is no current model 42 | if (!this.currentRouteModel) { 43 | return null; 44 | } 45 | 46 | if (isPresent(this.currentRouteModel.canonical)) { 47 | return this.currentRouteModel.canonical; 48 | } 49 | 50 | let slug; 51 | 52 | if (this.currentRouteModel.id === 'index') { 53 | slug = 'release'; 54 | } else { 55 | slug = `release/${this.currentRouteModel.id.replace(/\/index$/, '')}` 56 | } 57 | 58 | return `${config['ember-meta'].url}${slug}/`; 59 | }), 60 | }); 61 | -------------------------------------------------------------------------------- /app/services/page.js: -------------------------------------------------------------------------------- 1 | import Service from '@ember/service'; 2 | import { inject as service } from '@ember/service'; 3 | import { get, set, computed } from '@ember/object'; 4 | 5 | export default Service.extend({ 6 | router: service(), 7 | fastboot: service(), 8 | headData: service(), 9 | 10 | currentSection: computed('router.currentURL', 'pages.[]', 'content.id', function() { 11 | let tocSections = this.pages; 12 | 13 | let contentId = get(this, 'content.id'); 14 | 15 | if(!tocSections || !contentId) { return; } 16 | 17 | let section = contentId.split('/')[0] 18 | let currentSection = tocSections.find((tocSection) => tocSection.id === section); 19 | 20 | if(!currentSection) { 21 | return; 22 | } 23 | 24 | // eslint-disable-next-line ember/no-side-effects 25 | set(this, 'metaSection', get(currentSection, 'title')); 26 | 27 | return currentSection; 28 | }), 29 | 30 | /** 31 | * Find the TOC item that matches the current visible content. This is needed because the title comes 32 | * from the TOC and not the content. Also we use this to compute nextPage and previousPage 33 | * @return {Promise} the current page as a POJO 34 | */ 35 | currentPage: computed('router.currentURL', 'currentSection.pages', 'content.id', function() { 36 | let currentSection = this.currentSection; 37 | 38 | if(!currentSection) { return; } 39 | 40 | // special case for the index section - there should always be only exactly 1 page in the "index" section 41 | if (currentSection.id === 'index') { 42 | return get(currentSection, 'pages')[0]; 43 | } 44 | 45 | let pages = get(currentSection, 'pages'); 46 | 47 | let currentPage = pages.find((page) => page.url === get(this, 'content.id')); 48 | 49 | if(!currentPage) { 50 | return; 51 | } 52 | 53 | // eslint-disable-next-line ember/no-side-effects 54 | set(this, 'metaPage', get(currentPage, 'title')); 55 | 56 | return currentPage; 57 | }), 58 | 59 | isFirstPage: computed('currentSection', 'currentPage', function() { 60 | let currentSection = this.currentSection; 61 | 62 | if(!currentSection) { return; } 63 | 64 | let pages = get(currentSection, 'pages'); 65 | if(pages) { 66 | return pages.indexOf(this.currentPage) === 0; 67 | } 68 | }), 69 | 70 | isLastPage: computed('currentSection', 'currentPage', function() { 71 | let currentSection = this.currentSection; 72 | 73 | if(!currentSection) { return; } 74 | 75 | let pages = get(currentSection, 'pages'); 76 | if(pages) { 77 | return pages.indexOf(this.currentPage) === (pages.length-1); 78 | } 79 | }), 80 | 81 | previousPage: computed('currentSection.pages', 'currentPage.url', function() { 82 | let currentSection = this.currentSection; 83 | let currentPage = this.currentPage; 84 | 85 | if(!currentSection || !currentPage) { return; } 86 | 87 | let pages = get(currentSection, 'pages'); 88 | 89 | if(pages) { 90 | let currentLocalPage = pages.find((page) => page.url === currentPage.url); 91 | let index = pages.indexOf(currentLocalPage); 92 | 93 | if (index > 0) { 94 | return pages[index - 1]; 95 | } 96 | } 97 | }), 98 | 99 | nextPage: computed('currentSection.pages', 'currentPage.url', function() { 100 | let currentSection = this.currentSection; 101 | let currentPage = this.currentPage; 102 | 103 | if(!currentSection || !currentPage) { return; } 104 | 105 | let pages = get(currentSection, 'pages'); 106 | 107 | if(pages) { 108 | let currentLocalPage = pages.find((page) => page.url === currentPage.url); 109 | let index = pages.indexOf(currentLocalPage); 110 | 111 | if (index < pages.length-1) { 112 | return pages[index + 1]; 113 | } 114 | } 115 | }), 116 | 117 | previousSection: computed('currentSection', 'pages.[]', function() { 118 | let currentSection = this.currentSection; 119 | 120 | if(!currentSection) { return; } 121 | 122 | let pages = this.pages; 123 | 124 | if (pages) { 125 | let page = pages.content.find((content) => content.id === currentSection.id); 126 | 127 | let index = pages.content.indexOf(page); 128 | 129 | if (index > 0) { 130 | return pages.objectAt(index-1); 131 | } 132 | } 133 | }), 134 | 135 | nextSection: computed('currentSection', 'pages.[]', function() { 136 | let currentSection = this.currentSection; 137 | 138 | if(!currentSection) { return; } 139 | 140 | let pages = this.pages; 141 | 142 | if (pages) { 143 | let page = pages.content.find((content) => content.id === currentSection.id); 144 | 145 | let index = pages.content.indexOf(page); 146 | 147 | if (index < get(pages, 'length') - 1) { 148 | return pages.objectAt(index + 1); 149 | } 150 | } 151 | }), 152 | }); 153 | -------------------------------------------------------------------------------- /app/services/search.js: -------------------------------------------------------------------------------- 1 | import Service from '@ember/service'; 2 | import { task } from 'ember-concurrency'; 3 | import { get, set } from '@ember/object'; 4 | import { A as emberArray } from '@ember/array'; 5 | import { denodeify } from 'rsvp'; 6 | import algoliasearch from 'algoliasearch'; 7 | import { getOwner } from '@ember/application'; 8 | 9 | export default Service.extend({ 10 | 11 | results: emberArray(), 12 | 13 | search: task(function * (query, projectVersion) { 14 | const searchObj = { 15 | hitsPerPage: 15, 16 | restrictSearchableAttributes: ['content'], 17 | facetFilters: [[`version:${projectVersion}`]], 18 | query 19 | }; 20 | 21 | return set(this, 'results', yield this.doSearch(searchObj)); 22 | 23 | }).restartable(), 24 | 25 | doSearch(searchObj) { 26 | const config = getOwner(this).resolveRegistration('config:environment'); 27 | const { algoliaId, algoliaKey } = config['ember-algolia']; 28 | 29 | const client = algoliasearch(algoliaId, algoliaKey); 30 | const index = client.initIndex('ember-guides'); 31 | const searchFunction = denodeify(index.search.bind(index)); 32 | 33 | return searchFunction(searchObj).then((results) => { 34 | return get(results, 'hits'); 35 | }); 36 | } 37 | }); 38 | -------------------------------------------------------------------------------- /app/styles/_code-block-file-name.scss: -------------------------------------------------------------------------------- 1 | div.filename { 2 | background-color: #292929; 3 | } 4 | 5 | pre[class*="language-"] { 6 | background-color: #151515; 7 | } 8 | 9 | .filename { 10 | border-radius: 0.3em; 11 | } 12 | 13 | .ribbon { 14 | margin-top: 0.33em; 15 | float: right; 16 | height: 20px; 17 | width: 52px; 18 | background: 0 0 no-repeat; 19 | background-size: 52px 20px; 20 | } 21 | 22 | .filename > span { 23 | font-family: Menlo, "DejaVu Sans Mono", "Bitstream Vera Sans Mono", Courier, monospace; 24 | font-size: 0.8em; 25 | color: lightgrey; 26 | display: block; 27 | padding: 5px 0 0 10px; 28 | } 29 | 30 | .filename.javascript .ribbon, 31 | .filename.js .ribbon { 32 | background-image: url("../images/ribbon-js.svg"); 33 | } 34 | 35 | .filename.html .ribbon { 36 | background-image: url("../images/ribbon-html.svg"); 37 | } 38 | 39 | .filename.handlebars .ribbon, 40 | .filename.hbs .ribbon { 41 | background-image: url("../images/ribbon-hbs.svg"); 42 | } 43 | 44 | code { 45 | -webkit-font-feature-settings: "kern", "tnum"; 46 | -moz-font-feature-settings: "kern", "tnum"; 47 | -ms-font-feature-settings: "kern", "tnum"; 48 | font-feature-settings: "kern", "tnum"; 49 | } 50 | 51 | code:not([class*="language-"]) { 52 | background-color: #F8E7CF; 53 | border-radius: 3px; 54 | font-family: Menlo, "DejaVu Sans Mono", "Bitstream Vera Sans Mono", Courier, monospace; 55 | font-size: 0.9em; 56 | padding: 0.2em 0.5em; 57 | margin: 0 0.1em; 58 | } 59 | -------------------------------------------------------------------------------- /app/styles/_prism-overrides.scss: -------------------------------------------------------------------------------- 1 | .token.regex, 2 | .token.important, 3 | .token.variable { 4 | color: #C3F590; 5 | } 6 | 7 | code > .diff-insertion { 8 | background-color: rgba(93, 125, 93, 0.5); 9 | 10 | .token.property, 11 | .token.tag, 12 | .token.constant, 13 | .token.symbol, 14 | .token.deleted { 15 | color: #ff95bb; 16 | } 17 | } 18 | 19 | code > .diff-deletion { 20 | background-color: rgba(144, 84, 84, .7); 21 | 22 | .token.property, 23 | .token.tag, 24 | .token.constant, 25 | .token.symbol, 26 | .token.deleted { 27 | color: #ffaac8; 28 | } 29 | } 30 | 31 | code .diff-operator { 32 | user-select: none; 33 | } 34 | 35 | code .token.comment { 36 | color: #E6E6E6; 37 | } 38 | 39 | code { 40 | .token.property, 41 | .token.tag, 42 | .token.constant, 43 | .token.symbol, 44 | .token.deleted { 45 | color: #ff6fa3; 46 | } 47 | } 48 | -------------------------------------------------------------------------------- /app/styles/_search-input.scss: -------------------------------------------------------------------------------- 1 | 2 | // Typography 3 | $base-font-family: 'Source Sans Pro', sans-serif; 4 | $heading-font-family: $base-font-family; 5 | $monospace-font-family: Menlo, "DejaVu Sans Mono", "Bitstream Vera Sans Mono", Courier, monospace; 6 | $footer-font-family: proxima-nova, sans-serif; 7 | 8 | // Font Sizes 9 | $base-font-size: 1rem; 10 | $small-font-size: 0.75rem; 11 | $large-font-size: 1.5rem; 12 | 13 | // Line height 14 | $base-line-height: 1.5; 15 | $heading-line-height: 1.2; 16 | 17 | // Spacing 18 | $base-spacing: $base-line-height * 1em; 19 | $small-spacing: $base-spacing / 2; 20 | $large-spacing: $base-spacing * 2; 21 | $top-spacing: $base-spacing * 3.333; // 80px 22 | 23 | 24 | // Colors 25 | $ember-orange: #dd6a58; 26 | $light-brown: #b67d47; 27 | $brown: #865931; 28 | $medium-gray: #999; 29 | $dark-gray: #444545; 30 | $tan: #fffdf9; 31 | $white: #fff; 32 | $black: #000; 33 | $creme: #FFFBF5; 34 | $linen: #f9e7e4; 35 | $near-black: #444; 36 | 37 | $base-background-color: #FDFDFD; 38 | $sidebar-background-color: #FFFDF9; 39 | 40 | // Code Highlighting 41 | $highlight-yellow: #F5E090; 42 | $highlight-green: #C3F590; 43 | $highlight-red: #EC605E; 44 | $highlight-blue: #90D7F5; 45 | $highlight-cyan: #78CEC8; 46 | $highlight-added: #5D7D5D; 47 | $highlight-removed: #905454; 48 | $code-background: $dark-gray; 49 | $code-header-background: shade($code-background, 40%); 50 | // $code-line-number-background: mix($code-background, $code-header-background); 51 | 52 | $highlight-colors: ( 53 | attribute-name: $highlight-red, 54 | comment: $medium-gray, 55 | content: $highlight-cyan, 56 | function: $highlight-red, 57 | key: $highlight-red, 58 | keyword: $highlight-yellow, 59 | local-variable: $highlight-red, 60 | string: $highlight-cyan, 61 | tag: $highlight-red, 62 | ); 63 | 64 | // Font Colors 65 | $base-font-color: $near-black; 66 | $dark-font-color: $dark-gray; 67 | $light-font-color: $light-brown; 68 | $action-color: $ember-orange; 69 | 70 | // Border 71 | $base-border-radius: 3px; 72 | $base-border-color: #F8E7CF; 73 | $base-border: 1px solid $base-border-color; 74 | 75 | // Forms 76 | $form-box-shadow: inset 0 1px 3px rgba(#000, 0.06); 77 | $form-box-shadow-focus: $form-box-shadow, 0 0 5px adjust-color($action-color, $lightness: -5%, $alpha: -0.3); 78 | 79 | // Animations 80 | $duration: 200ms; 81 | 82 | // Z-indices 83 | $base-z-index: 0; 84 | 85 | 86 | $mobile-portrait-screen: 30em; // 480px 87 | $medium-screen: 40em; // 640px 88 | $large-screen: 54em; // 864px 89 | 90 | 91 | // https://github.com/algolia/docsearch/blob/master/src/styles/main.scss 92 | $color-border: darken($base-border-color, 10%); 93 | $color-border-light: $base-border-color; 94 | $color-category-header-background: $ember-orange; 95 | $color-highlight-header-background: lighten($color-category-header-background, 15%); 96 | $color-highlight-text: $highlight-red; 97 | $color-selected-background: white; 98 | $color-selected-text: darken($action-color, 15%); 99 | $color-left-column-bg: $sidebar-background-color; 100 | $color-left-column: $base-font-color; 101 | 102 | $breakpoint-medium: $medium-screen; 103 | $breakpoint-large: $large-screen; 104 | 105 | $dropdown-min-width-medium: 100%; 106 | $dropdown-min-width-large: 600px; 107 | 108 | // The dropdown adapts to screen size, to provide three different displays. 109 | // - A simple list of matching results 110 | // - Same list, but with text snippetting added if size is large enough 111 | // - Adding a second colum to let the content breath if enough room available 112 | 113 | .search-input { 114 | input { 115 | width: 100%; 116 | height: 35px; 117 | margin-bottom: 0; 118 | padding-left: 2.5em; 119 | border: none; 120 | border-radius: 35px; 121 | outline: none; 122 | appearance: none; 123 | background: url("/images/search-icon.svg") rgba(255,255,255,0.1) 10px 10px no-repeat; 124 | box-shadow: none; 125 | line-height: 35px; 126 | font-size: 0.9rem; 127 | color: #fff; 128 | } 129 | input, 130 | input::placeholder { 131 | font-size: 0.9rem; 132 | color: #fff; 133 | } 134 | input::-webkit-search-cancel-button { 135 | margin-right: 10px; 136 | } 137 | } 138 | 139 | .ds-dropdown-results { 140 | z-index: 10; 141 | 142 | a { 143 | color: #000000; 144 | } 145 | } 146 | // Main autocomplete wrapper 147 | .ds-dropdown-menu { 148 | background-color: #ffffff; 149 | border-radius: 4px; 150 | box-shadow: 0 1px 2px rgba(0,0,0,0.5); 151 | color: black; 152 | display: block; 153 | font-size: 12.8px; 154 | margin: 6px 0 0; 155 | text-align: left; 156 | } 157 | 158 | // No Results 159 | .algolia-docsearch-suggestion--noresults { 160 | padding: 0 5px 10px 10px; 161 | a { 162 | color: $ember-orange; 163 | text-decoration: none; 164 | } 165 | } 166 | 167 | // Each suggestion 168 | .algolia-docsearch-suggestion { 169 | color: #333; 170 | cursor: pointer; 171 | overflow: hidden; 172 | border-bottom: 1px solid $color-border; 173 | } 174 | 175 | // Main category headers 176 | .algolia-docsearch-suggestion--category-header { 177 | display: none; 178 | border: 2px solid white; 179 | background: $color-category-header-background; 180 | color: white; 181 | font-weight: 600; 182 | padding: 5px 10px; 183 | text-align: left; 184 | // Only show it when flagged as "__main" 185 | .algolia-docsearch-suggestion__main & { 186 | display: block; 187 | } 188 | } 189 | 190 | // Highlight 191 | .algolia-docsearch-suggestion--subcategory-column-text em, .algolia-docsearch-suggestion--title em { 192 | padding: 0; 193 | font-style: normal; 194 | color: $color-highlight-text; 195 | background: none; 196 | font-weight: 600; 197 | // Highlight the background in header 198 | .algolia-docsearch-suggestion--category-header & { 199 | color: inherit; 200 | background: $color-highlight-header-background; 201 | } 202 | } 203 | 204 | // Selected suggestion 205 | .aa-cursor .algolia-docsearch-suggestion--content { 206 | color: $color-selected-text; 207 | } 208 | .aa-cursor .algolia-docsearch-suggestion { 209 | background: $color-selected-background; 210 | } 211 | 212 | // The secondary column is hidden on small screens 213 | .algolia-docsearch-suggestion--subcategory-column { 214 | display: none; 215 | } 216 | // The text snippet is hidden on small screens 217 | .algolia-docsearch-suggestion--text { 218 | display: none; 219 | 220 | // If text parent node has --no-results 221 | // we should display the content 222 | .algolia-docsearch-suggestion--no-results & { 223 | display: block; 224 | } 225 | } 226 | 227 | .algolia-docsearch-suggestion--content { 228 | padding: 3px 5px; 229 | } 230 | 231 | .algolia-docsearch-suggestion--subcategory-inline { 232 | display: inline-block; 233 | font-weight: bold; 234 | &:after { 235 | content: " › "; 236 | } 237 | } 238 | .algolia-docsearch-suggestion--title { 239 | display: inline; 240 | } 241 | 242 | // Footer 243 | .powered-by-algolia { 244 | display: flex; 245 | flex-direction: row-reverse; 246 | padding: .5em; 247 | } 248 | 249 | // BREAKPOINT 1: 250 | // Screen is big enough to display the text snippets 251 | @media (min-width: $breakpoint-medium) { 252 | .ds-dropdown-menu { 253 | min-width: $dropdown-min-width-medium; 254 | } 255 | .algolia-docsearch-suggestion--text { 256 | display: block; 257 | font-size: .9em; 258 | padding: 2px 0; 259 | } 260 | } 261 | 262 | // BREAKPOINT 2: 263 | // Screen is big enough to display results in two columns 264 | @media (min-width: $breakpoint-large) { 265 | .ds-dropdown-menu { 266 | width: $dropdown-min-width-large; 267 | } 268 | .algolia-docsearch-suggestion { 269 | display: table; 270 | width: 100%; 271 | border-bottom: 1px solid $color-border-light; 272 | } 273 | .algolia-docsearch-suggestion--subcategory-column { 274 | border-right: 1px solid $color-border-light; 275 | background: $color-left-column-bg; 276 | color: $color-left-column; 277 | display: table-cell; 278 | overflow: hidden; 279 | padding: 3px 7px 3px 5px; 280 | text-align: right; 281 | text-overflow: ellipsis; 282 | vertical-align: top; 283 | 284 | width: 135px; // Hardcoded 285 | max-width: 135px; // Hardcoded 286 | min-width: 135px; // Hardcoded 287 | } 288 | 289 | .algolia-docsearch-suggestion--subcategory-column-text { 290 | display: none; 291 | 292 | .algolia-docsearch-suggestion__secondary & { 293 | display: block; 294 | } 295 | } 296 | .algolia-docsearch-suggestion--content { 297 | display: table-cell; 298 | padding: 3px 10px; 299 | } 300 | .algolia-docsearch-suggestion--subcategory-inline { 301 | display: none; 302 | } 303 | .algolia-docsearch-suggestion--title { 304 | font-weight: 600; 305 | } 306 | .algolia-docsearch-suggestion--text { 307 | display: block; 308 | font-weight: normal; 309 | padding: 2px; 310 | } 311 | } 312 | -------------------------------------------------------------------------------- /app/styles/_table-of-contents.scss: -------------------------------------------------------------------------------- 1 | @media only percy { 2 | .version-select { 3 | display: none; 4 | } 5 | 6 | .sidebar .ember-power-select-trigger { 7 | display: none; 8 | } 9 | } 10 | 11 | .anchorable-toc { 12 | position: relative; 13 | margin-left: -22px; 14 | padding-left: 22px; 15 | } 16 | 17 | .anchorable-toc:hover a.toc-anchor { 18 | display: block; 19 | } 20 | 21 | a.toc-anchor { 22 | display: none; 23 | position: absolute; 24 | text-decoration: none; 25 | border: none; 26 | width: 30px; 27 | height: 13px; 28 | background: url("../images/link.png") no-repeat; 29 | background-size: 18px 9px; 30 | left: 0; 31 | opacity: 0.5; 32 | top: 50%; 33 | margin-top: -5px; 34 | } 35 | 36 | label[for="toc-toggle"] { 37 | cursor: pointer; 38 | font-size: 1.25em; 39 | position: relative; 40 | } 41 | 42 | label[for="toc-toggle"]:hover { 43 | color: #dd6a58; 44 | } 45 | 46 | label[for="toc-toggle"]:hover:after { 47 | border-color: #dd6a58 transparent transparent transparent; 48 | } 49 | 50 | .toc-toggle:checked ~ label[for="toc-toggle"]:hover:after { 51 | border-color: transparent transparent #dd6a58 transparent; 52 | } 53 | 54 | label[for="toc-toggle"]:after { 55 | content: ''; 56 | border-color: #b67d47 transparent transparent transparent; 57 | border-style: solid; 58 | border-width: 5px 4px 0 4px; 59 | height: 0; 60 | margin-left: -4px; 61 | margin-top: -2px; 62 | position: absolute; 63 | right: 6px; 64 | top: 50%; 65 | width: 0; 66 | } 67 | 68 | .toc-toggle:checked ~ label[for="toc-toggle"]:after { 69 | border-color: transparent transparent #b67d47 transparent; 70 | border-width: 0 4px 5px 4px; 71 | } 72 | @media screen and (min-width: 53.75em) { 73 | label[for="toc-toggle"] { 74 | display: none; 75 | } 76 | } 77 | 78 | .toc-container { 79 | border-top: 1px solid #F8E7CF; 80 | overflow: hidden; 81 | display: none; 82 | } 83 | 84 | .toc-toggle:checked ~ .toc-container { 85 | display: block; 86 | } 87 | @media screen and (min-width: 53.75em) { 88 | .toc-container { 89 | display: block; 90 | } 91 | 92 | .toc-container[style] { 93 | display: block !important; 94 | } 95 | } 96 | 97 | .toc-container a { 98 | text-decoration: none; 99 | } 100 | 101 | li.toc-level-0 { 102 | margin-bottom: 0.5em; 103 | } 104 | 105 | .toc-container { 106 | .cp-Panel-toggle { 107 | color: #9b2918; 108 | display: block; 109 | line-height: 1.25; 110 | padding: 0.33em 0; 111 | } 112 | 113 | .cp-Panel-toggle:hover { 114 | color: darken(#9b2918, 5); 115 | } 116 | } 117 | 118 | li.toc-level-0 > div > a { 119 | font-weight: bold; 120 | margin: 0.33em 0; 121 | } 122 | 123 | ol.toc-level-1 { 124 | border-left: 1px solid #74B0CE; 125 | font-size: 0.9rem; 126 | } 127 | 128 | // ol.toc-level-1:not(.selected) { 129 | // display: none; 130 | // } 131 | 132 | ol.toc-level-1 li { 133 | border-left: 3px solid transparent; 134 | transition: border-width 200ms, margin-right 200ms; 135 | } 136 | 137 | ol.toc-level-1 li.selected, 138 | ol.toc-level-1 li:hover { 139 | border-left-color: #74B0CE; 140 | } 141 | 142 | ol.toc-level-1 li:hover { 143 | border-left-width: 6px; 144 | margin-right: -3px; 145 | } 146 | 147 | ol.toc-level-1 a { 148 | color: #444; 149 | padding: 0.33em 0 0.33em 1.618em; 150 | display: block; 151 | line-height: 1.25; 152 | } 153 | 154 | ol.toc-level-1 .selected a { 155 | font-weight: 500; 156 | } 157 | -------------------------------------------------------------------------------- /app/styles/app.scss: -------------------------------------------------------------------------------- 1 | @font-face { 2 | font-family: 'fontello'; 3 | src: url("../fonts/fontello.eot?60624967"); 4 | src: url("../fonts/fontello.eot?60624967#iefix") format("embedded-opentype"),url("../fonts/fontello.woff?60624967") format("woff"),url("../fonts/fontello.ttf?60624967") format("truetype"),url("../fonts/fontello.svg?60624967#fontello") format("svg"); 5 | font-weight: normal; 6 | font-style: normal; 7 | } 8 | 9 | [class^="icon-"]:before, 10 | [class*=" icon-"]:before { 11 | font-family: "fontello"; 12 | font-style: normal; 13 | font-weight: normal; 14 | speak: none; 15 | display: inline-block; 16 | text-decoration: inherit; 17 | width: 1em; 18 | margin-right: 0.2em; 19 | text-align: center; 20 | font-variant: normal; 21 | text-transform: none; 22 | line-height: 1em; 23 | margin-left: 0.2em; 24 | } 25 | 26 | .icon-search:before { 27 | content: '\e804'; 28 | } 29 | 30 | .icon-cancel:before { 31 | content: '\e802'; 32 | } 33 | 34 | .icon-link:before { 35 | content: '\e803'; 36 | } 37 | 38 | .icon-pencil:before { 39 | content: '\e801'; 40 | } 41 | 42 | .icon-fork:before { 43 | content: '\e800'; 44 | } 45 | 46 | .icon-github:before { 47 | content: '\e806'; 48 | } 49 | 50 | .icon-gplus:before { 51 | content: '\e807'; 52 | } 53 | 54 | .icon-twitter:before { 55 | content: '\e805'; 56 | } 57 | 58 | .select2-container { 59 | box-sizing: border-box; 60 | display: inline-block; 61 | margin: 0; 62 | position: relative; 63 | vertical-align: middle; 64 | } 65 | 66 | .select2-container .select2-selection--single { 67 | box-sizing: border-box; 68 | cursor: pointer; 69 | display: block; 70 | height: 28px; 71 | user-select: none; 72 | -webkit-user-select: none; 73 | } 74 | 75 | .select2-container .select2-selection--single .select2-selection__rendered { 76 | display: block; 77 | padding-left: 8px; 78 | padding-right: 20px; 79 | overflow: hidden; 80 | text-overflow: ellipsis; 81 | white-space: nowrap; 82 | } 83 | 84 | .select2-container[dir="rtl"] .select2-selection--single .select2-selection__rendered { 85 | padding-right: 8px; 86 | padding-left: 20px; 87 | } 88 | 89 | .select2-container .select2-selection--multiple { 90 | box-sizing: border-box; 91 | cursor: pointer; 92 | display: block; 93 | min-height: 32px; 94 | user-select: none; 95 | -webkit-user-select: none; 96 | } 97 | 98 | .select2-container .select2-selection--multiple .select2-selection__rendered { 99 | display: inline-block; 100 | overflow: hidden; 101 | padding-left: 8px; 102 | text-overflow: ellipsis; 103 | white-space: nowrap; 104 | } 105 | 106 | .select2-container .select2-search--inline { 107 | float: left; 108 | } 109 | 110 | .select2-container .select2-search--inline .select2-search__field { 111 | box-sizing: border-box; 112 | border: none; 113 | font-size: 100%; 114 | margin-top: 5px; 115 | } 116 | 117 | .select2-container .select2-search--inline .select2-search__field::-webkit-search-cancel-button { 118 | -webkit-appearance: none; 119 | } 120 | 121 | .select2-dropdown { 122 | background-color: white; 123 | border: 1px solid #aaa; 124 | border-radius: 4px; 125 | box-sizing: border-box; 126 | display: block; 127 | position: absolute; 128 | left: -100000px; 129 | width: 100%; 130 | z-index: 1051; 131 | } 132 | 133 | .select2-results { 134 | display: block; 135 | } 136 | 137 | .select2-results__options { 138 | list-style: none; 139 | margin: 0; 140 | padding: 0; 141 | } 142 | 143 | .select2-results__option { 144 | padding: 6px; 145 | user-select: none; 146 | -webkit-user-select: none; 147 | } 148 | 149 | .select2-results__option[aria-selected] { 150 | cursor: pointer; 151 | } 152 | 153 | .select2-container--open .select2-dropdown { 154 | left: 0; 155 | } 156 | 157 | .select2-container--open .select2-dropdown--above { 158 | border-bottom: none; 159 | border-bottom-left-radius: 0; 160 | border-bottom-right-radius: 0; 161 | } 162 | 163 | .select2-container--open .select2-dropdown--below { 164 | border-top: none; 165 | border-top-left-radius: 0; 166 | border-top-right-radius: 0; 167 | } 168 | 169 | .select2-search--dropdown { 170 | display: block; 171 | padding: 4px; 172 | } 173 | 174 | .select2-search--dropdown .select2-search__field { 175 | padding: 4px; 176 | width: 100%; 177 | box-sizing: border-box; 178 | } 179 | 180 | .select2-search--dropdown .select2-search__field::-webkit-search-cancel-button { 181 | -webkit-appearance: none; 182 | } 183 | 184 | .select2-search--dropdown.select2-search--hide { 185 | display: none; 186 | } 187 | 188 | .select2-close-mask { 189 | border: 0; 190 | margin: 0; 191 | padding: 0; 192 | display: block; 193 | position: fixed; 194 | left: 0; 195 | top: 0; 196 | min-height: 100%; 197 | min-width: 100%; 198 | height: auto; 199 | width: auto; 200 | opacity: 0; 201 | z-index: 99; 202 | background-color: #fff; 203 | filter:alpha(opacity=0); 204 | } 205 | 206 | .select2-container--default .select2-selection--single { 207 | background-color: #fff; 208 | border: 1px solid #aaa; 209 | border-radius: 4px; 210 | } 211 | 212 | .select2-container--default .select2-selection--single .select2-selection__rendered { 213 | color: #444; 214 | line-height: 28px; 215 | } 216 | 217 | .select2-container--default .select2-selection--single .select2-selection__clear { 218 | cursor: pointer; 219 | float: right; 220 | font-weight: bold; 221 | } 222 | 223 | .select2-container--default .select2-selection--single .select2-selection__placeholder { 224 | color: #999; 225 | } 226 | 227 | .select2-container--default .select2-selection--single .select2-selection__arrow { 228 | height: 26px; 229 | position: absolute; 230 | top: 1px; 231 | right: 1px; 232 | width: 20px; 233 | } 234 | 235 | .select2-container--default .select2-selection--single .select2-selection__arrow b { 236 | border-color: #888 transparent transparent transparent; 237 | border-style: solid; 238 | border-width: 5px 4px 0 4px; 239 | height: 0; 240 | left: 50%; 241 | margin-left: -4px; 242 | margin-top: -2px; 243 | position: absolute; 244 | top: 50%; 245 | width: 0; 246 | } 247 | 248 | .select2-container--default[dir="rtl"] .select2-selection--single .select2-selection__clear { 249 | float: left; 250 | } 251 | 252 | .select2-container--default[dir="rtl"] .select2-selection--single .select2-selection__arrow { 253 | left: 1px; 254 | right: auto; 255 | } 256 | 257 | .select2-container--default.select2-container--disabled .select2-selection--single { 258 | background-color: #eee; 259 | cursor: default; 260 | } 261 | 262 | .select2-container--default.select2-container--disabled .select2-selection--single .select2-selection__clear { 263 | display: none; 264 | } 265 | 266 | .select2-container--default.select2-container--open .select2-selection--single .select2-selection__arrow b { 267 | border-color: transparent transparent #888 transparent; 268 | border-width: 0 4px 5px 4px; 269 | } 270 | 271 | .select2-container--default .select2-selection--multiple { 272 | background-color: white; 273 | border: 1px solid #aaa; 274 | border-radius: 4px; 275 | cursor: text; 276 | } 277 | 278 | .select2-container--default .select2-selection--multiple .select2-selection__rendered { 279 | box-sizing: border-box; 280 | list-style: none; 281 | margin: 0; 282 | padding: 0 5px; 283 | width: 100%; 284 | } 285 | 286 | .select2-container--default .select2-selection--multiple .select2-selection__placeholder { 287 | color: #999; 288 | margin-top: 5px; 289 | float: left; 290 | } 291 | 292 | .select2-container--default .select2-selection--multiple .select2-selection__clear { 293 | cursor: pointer; 294 | float: right; 295 | font-weight: bold; 296 | margin-top: 5px; 297 | margin-right: 10px; 298 | } 299 | 300 | .select2-container--default .select2-selection--multiple .select2-selection__choice { 301 | background-color: #e4e4e4; 302 | border: 1px solid #aaa; 303 | border-radius: 4px; 304 | cursor: default; 305 | float: left; 306 | margin-right: 5px; 307 | margin-top: 5px; 308 | padding: 0 5px; 309 | } 310 | 311 | .select2-container--default .select2-selection--multiple .select2-selection__choice__remove { 312 | color: #999; 313 | cursor: pointer; 314 | display: inline-block; 315 | font-weight: bold; 316 | margin-right: 2px; 317 | } 318 | 319 | .select2-container--default .select2-selection--multiple .select2-selection__choice__remove:hover { 320 | color: #333; 321 | } 322 | 323 | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice, 324 | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__placeholder { 325 | float: right; 326 | } 327 | 328 | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice { 329 | margin-left: 5px; 330 | margin-right: auto; 331 | } 332 | 333 | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice__remove { 334 | margin-left: 2px; 335 | margin-right: auto; 336 | } 337 | 338 | .select2-container--default.select2-container--focus .select2-selection--multiple { 339 | border: solid black 1px; 340 | outline: 0; 341 | } 342 | 343 | .select2-container--default.select2-container--disabled .select2-selection--multiple { 344 | background-color: #eee; 345 | cursor: default; 346 | } 347 | 348 | .select2-container--default.select2-container--disabled .select2-selection__choice__remove { 349 | display: none; 350 | } 351 | 352 | .select2-container--default.select2-container--open.select2-container--above .select2-selection--multiple, 353 | .select2-container--default.select2-container--open.select2-container--above .select2-selection--single { 354 | border-top-left-radius: 0; 355 | border-top-right-radius: 0; 356 | } 357 | 358 | .select2-container--default.select2-container--open.select2-container--below .select2-selection--multiple, 359 | .select2-container--default.select2-container--open.select2-container--below .select2-selection--single { 360 | border-bottom-left-radius: 0; 361 | border-bottom-right-radius: 0; 362 | } 363 | 364 | .select2-container--default .select2-search--dropdown .select2-search__field { 365 | border: 1px solid #aaa; 366 | } 367 | 368 | .select2-container--default .select2-search--inline .select2-search__field { 369 | background: transparent; 370 | border: none; 371 | outline: 0; 372 | } 373 | 374 | .select2-container--default .select2-results > .select2-results__options { 375 | max-height: 200px; 376 | overflow-y: auto; 377 | } 378 | 379 | .select2-container--default .select2-results__option[role=group] { 380 | padding: 0; 381 | } 382 | 383 | .select2-container--default .select2-results__option[aria-disabled=true] { 384 | color: #999; 385 | } 386 | 387 | .select2-container--default .select2-results__option[aria-selected=true] { 388 | background-color: #ddd; 389 | } 390 | 391 | .select2-container--default .select2-results__option .select2-results__option { 392 | padding-left: 1em; 393 | } 394 | 395 | .select2-container--default .select2-results__option .select2-results__option .select2-results__group { 396 | padding-left: 0; 397 | } 398 | 399 | .select2-container--default .select2-results__option .select2-results__option .select2-results__option { 400 | margin-left: -1em; 401 | padding-left: 2em; 402 | } 403 | 404 | .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option { 405 | margin-left: -2em; 406 | padding-left: 3em; 407 | } 408 | 409 | .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { 410 | margin-left: -3em; 411 | padding-left: 4em; 412 | } 413 | 414 | .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { 415 | margin-left: -4em; 416 | padding-left: 5em; 417 | } 418 | 419 | .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { 420 | margin-left: -5em; 421 | padding-left: 6em; 422 | } 423 | 424 | .select2-container--default .select2-results__option--highlighted[aria-selected] { 425 | background-color: #5897fb; 426 | color: white; 427 | } 428 | 429 | .select2-container--default .select2-results__group { 430 | cursor: default; 431 | display: block; 432 | padding: 6px; 433 | } 434 | 435 | .select2-container--classic .select2-selection--single { 436 | background-color: #f6f6f6; 437 | border: 1px solid #aaa; 438 | border-radius: 4px; 439 | outline: 0; 440 | background-image: -webkit-linear-gradient(top, #fff 50%, #eee 100%); 441 | background-image: -o-linear-gradient(top, #fff 50%, #eee 100%); 442 | background-image: linear-gradient(to bottom, #fff 50%, #eee 100%); 443 | background-repeat: repeat-x; 444 | filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffff', endColorstr='#eeeeee', GradientType=0); 445 | } 446 | 447 | .select2-container--classic .select2-selection--single:focus { 448 | border: 1px solid #5897fb; 449 | } 450 | 451 | .select2-container--classic .select2-selection--single .select2-selection__rendered { 452 | color: #444; 453 | line-height: 28px; 454 | } 455 | 456 | .select2-container--classic .select2-selection--single .select2-selection__clear { 457 | cursor: pointer; 458 | float: right; 459 | font-weight: bold; 460 | margin-right: 10px; 461 | } 462 | 463 | .select2-container--classic .select2-selection--single .select2-selection__placeholder { 464 | color: #999; 465 | } 466 | 467 | .select2-container--classic .select2-selection--single .select2-selection__arrow { 468 | background-color: #ddd; 469 | border: none; 470 | border-left: 1px solid #aaa; 471 | border-top-right-radius: 4px; 472 | border-bottom-right-radius: 4px; 473 | height: 26px; 474 | position: absolute; 475 | top: 1px; 476 | right: 1px; 477 | width: 20px; 478 | background-image: -webkit-linear-gradient(top, #eee 50%, #ccc 100%); 479 | background-image: -o-linear-gradient(top, #eee 50%, #ccc 100%); 480 | background-image: linear-gradient(to bottom, #eee 50%, #ccc 100%); 481 | background-repeat: repeat-x; 482 | filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#cccccc', GradientType=0); 483 | } 484 | 485 | .select2-container--classic .select2-selection--single .select2-selection__arrow b { 486 | border-color: #888 transparent transparent transparent; 487 | border-style: solid; 488 | border-width: 5px 4px 0 4px; 489 | height: 0; 490 | left: 50%; 491 | margin-left: -4px; 492 | margin-top: -2px; 493 | position: absolute; 494 | top: 50%; 495 | width: 0; 496 | } 497 | 498 | .select2-container--classic[dir="rtl"] .select2-selection--single .select2-selection__clear { 499 | float: left; 500 | } 501 | 502 | .select2-container--classic[dir="rtl"] .select2-selection--single .select2-selection__arrow { 503 | border: none; 504 | border-right: 1px solid #aaa; 505 | border-radius: 0; 506 | border-top-left-radius: 4px; 507 | border-bottom-left-radius: 4px; 508 | left: 1px; 509 | right: auto; 510 | } 511 | 512 | .select2-container--classic.select2-container--open .select2-selection--single { 513 | border: 1px solid #5897fb; 514 | } 515 | 516 | .select2-container--classic.select2-container--open .select2-selection--single .select2-selection__arrow { 517 | background: transparent; 518 | border: none; 519 | } 520 | 521 | .select2-container--classic.select2-container--open .select2-selection--single .select2-selection__arrow b { 522 | border-color: transparent transparent #888 transparent; 523 | border-width: 0 4px 5px 4px; 524 | } 525 | 526 | .select2-container--classic.select2-container--open.select2-container--above .select2-selection--single { 527 | border-top: none; 528 | border-top-left-radius: 0; 529 | border-top-right-radius: 0; 530 | background-image: -webkit-linear-gradient(top, #fff 0%, #eee 50%); 531 | background-image: -o-linear-gradient(top, #fff 0%, #eee 50%); 532 | background-image: linear-gradient(to bottom, #fff 0%, #eee 50%); 533 | background-repeat: repeat-x; 534 | filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffff', endColorstr='#eeeeee', GradientType=0); 535 | } 536 | 537 | .select2-container--classic.select2-container--open.select2-container--below .select2-selection--single { 538 | border-bottom: none; 539 | border-bottom-left-radius: 0; 540 | border-bottom-right-radius: 0; 541 | background-image: -webkit-linear-gradient(top, #eee 50%, #fff 100%); 542 | background-image: -o-linear-gradient(top, #eee 50%, #fff 100%); 543 | background-image: linear-gradient(to bottom, #eee 50%, #fff 100%); 544 | background-repeat: repeat-x; 545 | filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#ffffff', GradientType=0); 546 | } 547 | 548 | .select2-container--classic .select2-selection--multiple { 549 | background-color: white; 550 | border: 1px solid #aaa; 551 | border-radius: 4px; 552 | cursor: text; 553 | outline: 0; 554 | } 555 | 556 | .select2-container--classic .select2-selection--multiple:focus { 557 | border: 1px solid #5897fb; 558 | } 559 | 560 | .select2-container--classic .select2-selection--multiple .select2-selection__rendered { 561 | list-style: none; 562 | margin: 0; 563 | padding: 0 5px; 564 | } 565 | 566 | .select2-container--classic .select2-selection--multiple .select2-selection__clear { 567 | display: none; 568 | } 569 | 570 | .select2-container--classic .select2-selection--multiple .select2-selection__choice { 571 | background-color: #e4e4e4; 572 | border: 1px solid #aaa; 573 | border-radius: 4px; 574 | cursor: default; 575 | float: left; 576 | margin-right: 5px; 577 | margin-top: 5px; 578 | padding: 0 5px; 579 | } 580 | 581 | .select2-container--classic .select2-selection--multiple .select2-selection__choice__remove { 582 | color: #888; 583 | cursor: pointer; 584 | display: inline-block; 585 | font-weight: bold; 586 | margin-right: 2px; 587 | } 588 | 589 | .select2-container--classic .select2-selection--multiple .select2-selection__choice__remove:hover { 590 | color: #555; 591 | } 592 | 593 | .select2-container--classic[dir="rtl"] .select2-selection--multiple .select2-selection__choice { 594 | float: right; 595 | } 596 | 597 | .select2-container--classic[dir="rtl"] .select2-selection--multiple .select2-selection__choice { 598 | margin-left: 5px; 599 | margin-right: auto; 600 | } 601 | 602 | .select2-container--classic[dir="rtl"] .select2-selection--multiple .select2-selection__choice__remove { 603 | margin-left: 2px; 604 | margin-right: auto; 605 | } 606 | 607 | .select2-container--classic.select2-container--open .select2-selection--multiple { 608 | border: 1px solid #5897fb; 609 | } 610 | 611 | .select2-container--classic.select2-container--open.select2-container--above .select2-selection--multiple { 612 | border-top: none; 613 | border-top-left-radius: 0; 614 | border-top-right-radius: 0; 615 | } 616 | 617 | .select2-container--classic.select2-container--open.select2-container--below .select2-selection--multiple { 618 | border-bottom: none; 619 | border-bottom-left-radius: 0; 620 | border-bottom-right-radius: 0; 621 | } 622 | 623 | .select2-container--classic .select2-search--dropdown .select2-search__field { 624 | border: 1px solid #aaa; 625 | outline: 0; 626 | } 627 | 628 | .select2-container--classic .select2-search--inline .select2-search__field { 629 | outline: 0; 630 | } 631 | 632 | .select2-container--classic .select2-dropdown { 633 | background-color: white; 634 | border: 1px solid transparent; 635 | } 636 | 637 | .select2-container--classic .select2-dropdown--above { 638 | border-bottom: none; 639 | } 640 | 641 | .select2-container--classic .select2-dropdown--below { 642 | border-top: none; 643 | } 644 | 645 | .select2-container--classic .select2-results > .select2-results__options { 646 | max-height: 200px; 647 | overflow-y: auto; 648 | } 649 | 650 | .select2-container--classic .select2-results__option[role=group] { 651 | padding: 0; 652 | } 653 | 654 | .select2-container--classic .select2-results__option[aria-disabled=true] { 655 | color: grey; 656 | } 657 | 658 | .select2-container--classic .select2-results__option--highlighted[aria-selected] { 659 | background-color: #3875d7; 660 | color: white; 661 | } 662 | 663 | .select2-container--classic .select2-results__group { 664 | cursor: default; 665 | display: block; 666 | padding: 6px; 667 | } 668 | 669 | .select2-container--classic.select2-container--open .select2-dropdown { 670 | border-color: #5897fb; 671 | } 672 | 673 | button, 674 | input[type="button"], 675 | input[type="reset"], 676 | input[type="submit"], 677 | button { 678 | -webkit-appearance: none; 679 | -moz-appearance: none; 680 | -ms-appearance: none; 681 | -o-appearance: none; 682 | appearance: none; 683 | -webkit-font-smoothing: antialiased; 684 | -moz-osx-font-smoothing: grayscale; 685 | background-color: #dd6a58; 686 | border-radius: 3px; 687 | border: none; 688 | color: #fff; 689 | cursor: pointer; 690 | display: inline-block; 691 | font-family: "Roboto", "Helvetica Neue", "Helvetica", sans-serif; 692 | font-size: 1rem; 693 | font-weight: 600; 694 | line-height: 1; 695 | padding: 0.75em 1em; 696 | text-decoration: none; 697 | user-select: none; 698 | vertical-align: middle; 699 | white-space: nowrap; 700 | } 701 | 702 | button:focus, 703 | button:hover, 704 | input[type="button"]:hover, 705 | input[type="button"]:focus, 706 | input[type="reset"]:hover, 707 | input[type="reset"]:focus, 708 | input[type="submit"]:hover, 709 | input[type="submit"]:focus, 710 | button:hover, 711 | button:focus { 712 | background-color: #c13c27; 713 | color: #fff; 714 | } 715 | 716 | button:disabled, 717 | input[type="button"]:disabled, 718 | input[type="reset"]:disabled, 719 | input[type="submit"]:disabled, 720 | button:disabled { 721 | cursor: not-allowed; 722 | opacity: 0.5; 723 | } 724 | 725 | fieldset { 726 | background-color: #fefdfc; 727 | border: 1px solid #F8E7CF; 728 | margin: 0 0 0.75em; 729 | padding: 1.5em; 730 | } 731 | 732 | input, 733 | label, 734 | select { 735 | display: block; 736 | font-family: "Roboto", "Helvetica Neue", "Helvetica", sans-serif; 737 | font-size: 1rem; 738 | } 739 | 740 | label { 741 | font-weight: 600; 742 | margin-bottom: 0.375em; 743 | } 744 | 745 | label.required::after { 746 | content: "*"; 747 | } 748 | 749 | label abbr { 750 | display: none; 751 | } 752 | 753 | input[type="color"], 754 | input[type="date"], 755 | input[type="datetime"], 756 | input[type="datetime-local"], 757 | input[type="email"], 758 | input[type="month"], 759 | input[type="number"], 760 | input[type="password"], 761 | input[type="tel"], 762 | input[type="text"], 763 | input[type="time"], 764 | input[type="url"], 765 | input[type="week"], 766 | input:not([type]), 767 | select[multiple=multiple], 768 | textarea, 769 | textarea { 770 | background-color: #FDFDFD; 771 | border: 1px solid #F8E7CF; 772 | border-radius: 3px; 773 | box-shadow: inset 0 1px 3px rgba(0,0,0,0.06); 774 | box-sizing: border-box; 775 | font-family: "Roboto", "Helvetica Neue", "Helvetica", sans-serif; 776 | font-size: 1rem; 777 | margin-bottom: 0.75em; 778 | padding: 0.5em; 779 | transition: border-color; 780 | width: 100%; 781 | } 782 | 783 | input[type="color"]:hover, 784 | input[type="date"]:hover, 785 | input[type="datetime"]:hover, 786 | input[type="datetime-local"]:hover, 787 | input[type="email"]:hover, 788 | input[type="month"]:hover, 789 | input[type="number"]:hover, 790 | input[type="password"]:hover, 791 | input[type="search"]:hover, 792 | input[type="tel"]:hover, 793 | input[type="text"]:hover, 794 | input[type="time"]:hover, 795 | input[type="url"]:hover, 796 | input[type="week"]:hover, 797 | input:not([type]):hover, 798 | select[multiple=multiple]:hover, 799 | textarea:hover, 800 | textarea:hover { 801 | border-color: #f2d1a2; 802 | } 803 | 804 | input[type="color"]:focus, 805 | input[type="date"]:focus, 806 | input[type="datetime"]:focus, 807 | input[type="datetime-local"]:focus, 808 | input[type="email"]:focus, 809 | input[type="month"]:focus, 810 | input[type="number"]:focus, 811 | input[type="password"]:focus, 812 | input[type="search"]:focus, 813 | input[type="tel"]:focus, 814 | input[type="text"]:focus, 815 | input[type="time"]:focus, 816 | input[type="url"]:focus, 817 | input[type="week"]:focus, 818 | input:not([type]):focus, 819 | select[multiple=multiple]:focus, 820 | textarea:focus, 821 | textarea:focus { 822 | border-color: #dd6a58; 823 | box-shadow: inset 0 1px 3px rgba(0,0,0,0.06),0 0 5px rgba(217,87,67,0.7); 824 | } 825 | 826 | textarea { 827 | resize: vertical; 828 | } 829 | 830 | input[type="search"] { 831 | -webkit-appearance: none; 832 | -moz-appearance: none; 833 | -ms-appearance: none; 834 | -o-appearance: none; 835 | appearance: none; 836 | } 837 | 838 | input[type="checkbox"], 839 | input[type="radio"] { 840 | display: inline; 841 | margin-right: 0.375em; 842 | } 843 | 844 | input[type="file"] { 845 | padding-bottom: 0.75em; 846 | width: 100%; 847 | } 848 | 849 | select { 850 | margin-bottom: 1.5em; 851 | max-width: 100%; 852 | width: auto; 853 | } 854 | 855 | ol, 856 | ul { 857 | list-style-type: none; 858 | margin: 0; 859 | padding: 0; 860 | } 861 | 862 | article ul { 863 | list-style-type: disc; 864 | margin-bottom: 0.75em; 865 | padding-left: 1.5em; 866 | } 867 | 868 | article ol { 869 | list-style-type: none; 870 | margin-bottom: 1.5em; 871 | } 872 | 873 | article ol > li { 874 | counter-increment: customlistcounter; 875 | clear: both; 876 | padding: 0.75em 0 0.75em 2em; 877 | position: relative; 878 | z-index: 1; 879 | } 880 | 881 | article ol > li::before { 882 | position: absolute; 883 | top: 0.95em; 884 | left: 0; 885 | height: 2em; 886 | width: 2em; 887 | content: counter(customlistcounter) " "; 888 | border: 2px solid #F8E7CF; 889 | border-radius: 50%; 890 | color: #b67d47; 891 | float: left; 892 | font-size: 0.75rem; 893 | line-height: 2; 894 | overflow: hidden; 895 | text-align: center; 896 | } 897 | 898 | article ol:first-child { 899 | counter-reset: customlistcounter; 900 | } 901 | 902 | .chapter li { 903 | margin: 0.75em 0; 904 | } 905 | 906 | .chapter ul+.highlight { 907 | margin-top: 1.5em; 908 | } 909 | 910 | dl { 911 | margin-bottom: 0.75em; 912 | } 913 | 914 | dl dt { 915 | font-weight: bold; 916 | margin-top: 0.75em; 917 | } 918 | 919 | dl dd { 920 | margin: 0; 921 | } 922 | 923 | table { 924 | -webkit-font-feature-settings: "kern", "liga", "tnum"; 925 | -moz-font-feature-settings: "kern", "liga", "tnum"; 926 | -ms-font-feature-settings: "kern", "liga", "tnum"; 927 | font-feature-settings: "kern", "liga", "tnum"; 928 | border-collapse: collapse; 929 | margin: 0.75em 0; 930 | table-layout: fixed; 931 | width: 100%; 932 | } 933 | 934 | thead { 935 | background-color: #f9e7e4; 936 | } 937 | 938 | th { 939 | border-bottom: 1px solid #eec68c; 940 | font-weight: 600; 941 | text-align: left; 942 | } 943 | 944 | td { 945 | border-bottom: 1px solid #F8E7CF; 946 | } 947 | 948 | td, 949 | th { 950 | padding: 0.75em; 951 | } 952 | 953 | td, 954 | th, 955 | tr { 956 | vertical-align: middle; 957 | } 958 | /*! HiDPI v2.0.1 | MIT License | git.io/hidpi */ 959 | 960 | h1 code, 961 | h2 code, 962 | h3 code, 963 | h4 code, 964 | h5 code, 965 | h6 code { 966 | text-transform: none; 967 | } 968 | 969 | 970 | p { 971 | margin: 1.5em 0 0.75em; 972 | line-height: 1.5em; 973 | } 974 | 975 | a { 976 | color: #c7341f; 977 | transition: color 0.1s linear; 978 | text-decoration: underline; 979 | } 980 | 981 | a:active, 982 | a:focus, 983 | a:hover { 984 | color: #9b2918; 985 | cursor: pointer; 986 | } 987 | 988 | a:active { 989 | outline: none; 990 | } 991 | 992 | hr { 993 | border-bottom: 1px solid #F8E7CF; 994 | border-left: none; 995 | border-right: none; 996 | border-top: none; 997 | margin: 1.5em 0; 998 | } 999 | 1000 | img, 1001 | picture { 1002 | margin: 0; 1003 | max-width: 100%; 1004 | } 1005 | 1006 | em { 1007 | font-style: italic; 1008 | } 1009 | 1010 | strong { 1011 | font-weight: bold; 1012 | } 1013 | 1014 | .ds-dropdown-menu { 1015 | background-color: #FFF; 1016 | border-radius: 4px; 1017 | margin: 6px 0 0; 1018 | text-align: left; 1019 | box-shadow: 0 1px 2px rgba(0,0,0,0.5); 1020 | } 1021 | 1022 | .algolia-docsearch-suggestion { 1023 | color: #333; 1024 | cursor: pointer; 1025 | overflow: hidden; 1026 | border-bottom: 1px solid #f2d1a2; 1027 | } 1028 | 1029 | .algolia-docsearch-suggestion--category-header { 1030 | display: none; 1031 | border: 2px solid white; 1032 | background: #dd6a58; 1033 | color: white; 1034 | font-weight: 600; 1035 | padding: 5px 10px; 1036 | text-align: left; 1037 | } 1038 | 1039 | .algolia-docsearch-suggestion__main .algolia-docsearch-suggestion--category-header { 1040 | display: block; 1041 | } 1042 | 1043 | .algolia-docsearch-suggestion--highlight { 1044 | padding: 0; 1045 | color: #EC605E; 1046 | background: none; 1047 | font-weight: 600; 1048 | } 1049 | 1050 | .algolia-docsearch-suggestion--category-header .algolia-docsearch-suggestion--highlight { 1051 | color: inherit; 1052 | background: #eaa398; 1053 | } 1054 | 1055 | .aa-cursor .algolia-docsearch-suggestion--content { 1056 | color: #c13c27; 1057 | } 1058 | 1059 | .aa-cursor .algolia-docsearch-suggestion { 1060 | background: white; 1061 | } 1062 | 1063 | .algolia-docsearch-suggestion--subcategory-column { 1064 | display: none; 1065 | } 1066 | 1067 | .algolia-docsearch-suggestion--text { 1068 | display: none; 1069 | } 1070 | 1071 | .algolia-docsearch-suggestion--no-results .algolia-docsearch-suggestion--text { 1072 | display: block; 1073 | } 1074 | 1075 | .algolia-docsearch-suggestion--content { 1076 | padding: 3px 5px; 1077 | } 1078 | 1079 | .algolia-docsearch-suggestion--subcategory-inline { 1080 | display: inline-block; 1081 | font-weight: bold; 1082 | } 1083 | 1084 | .algolia-docsearch-suggestion--subcategory-inline:after { 1085 | content: " › "; 1086 | } 1087 | 1088 | .algolia-docsearch-suggestion--title { 1089 | display: inline; 1090 | } 1091 | 1092 | .algolia-docsearch-footer { 1093 | border-top: 1px solid #F8E7CF; 1094 | text-align: right; 1095 | font-size: 12px; 1096 | padding: 4px 2px 0 0; 1097 | color: #333; 1098 | } 1099 | 1100 | .algolia-docsearch-footer .algolia-docsearch-footer--logo { 1101 | display: inline-block !important; 1102 | width: 45px; 1103 | height: 16px; 1104 | text-indent: 101%; 1105 | overflow: hidden; 1106 | white-space: nowrap; 1107 | background-image: url(""); 1108 | background-repeat: no-repeat; 1109 | background-size: contain; 1110 | vertical-align: middle; 1111 | } 1112 | @media (min-width: 40em) { 1113 | .ds-dropdown-menu { 1114 | min-width: 100%; 1115 | } 1116 | 1117 | .algolia-docsearch-suggestion--text { 1118 | display: block; 1119 | font-size: 0.9em; 1120 | padding: 2px 0; 1121 | } 1122 | } 1123 | @media (min-width: 53.75em) { 1124 | .ds-dropdown-menu { 1125 | min-width: 600px; 1126 | } 1127 | 1128 | .algolia-docsearch-suggestion { 1129 | display: table; 1130 | width: 100%; 1131 | border-bottom: 1px solid #F8E7CF; 1132 | } 1133 | 1134 | .algolia-docsearch-suggestion--subcategory-column { 1135 | border-right: 1px solid #F8E7CF; 1136 | background: #FFFDF9; 1137 | color: #444; 1138 | display: table-cell; 1139 | overflow: hidden; 1140 | padding: 3px 7px 3px 5px; 1141 | text-align: right; 1142 | text-overflow: ellipsis; 1143 | vertical-align: top; 1144 | width: 135px; 1145 | max-width: 135px; 1146 | min-width: 135px; 1147 | } 1148 | 1149 | .algolia-docsearch-suggestion--subcategory-column-text { 1150 | display: none; 1151 | } 1152 | 1153 | .algolia-docsearch-suggestion__secondary .algolia-docsearch-suggestion--subcategory-column-text { 1154 | display: block; 1155 | } 1156 | 1157 | .algolia-docsearch-suggestion--content { 1158 | display: table-cell; 1159 | padding: 3px 10px; 1160 | } 1161 | 1162 | .algolia-docsearch-suggestion--subcategory-inline { 1163 | display: none; 1164 | } 1165 | 1166 | .algolia-docsearch-suggestion--title { 1167 | font-weight: 600; 1168 | } 1169 | 1170 | .algolia-docsearch-suggestion--text { 1171 | display: block; 1172 | font-weight: normal; 1173 | padding: 2px; 1174 | } 1175 | } 1176 | 1177 | *, 1178 | *:after { 1179 | box-sizing: border-box; 1180 | } 1181 | 1182 | body { 1183 | -webkit-font-smoothing: antialiased; 1184 | -moz-osx-font-smoothing: grayscale; 1185 | background-color: #FDFDFD; 1186 | margin: 0; 1187 | } 1188 | @media screen and (min-width: 53.75em) { 1189 | main { 1190 | display: flex; 1191 | } 1192 | } 1193 | 1194 | .container { 1195 | max-width: 60em; 1196 | margin-left: auto; 1197 | margin-right: auto; 1198 | padding: 0 1em; 1199 | } 1200 | 1201 | .container::after { 1202 | clear: both; 1203 | content: ""; 1204 | display: table; 1205 | } 1206 | 1207 | .sidebar { 1208 | padding-top: 0.75em; 1209 | background-color: #FFFDF9; 1210 | border-bottom: 1px solid #F8E7CF; 1211 | position: relative; 1212 | z-index: 1; 1213 | } 1214 | @media screen and (min-width: 53.75em) { 1215 | .sidebar { 1216 | float: left; 1217 | display: block; 1218 | margin-right: 2.35765%; 1219 | width: 27.49666%; 1220 | border-bottom: 0; 1221 | border-right: 1px solid #F8E7CF; 1222 | padding: 3em 1.618em 3em 0; 1223 | } 1224 | 1225 | .sidebar:last-child { 1226 | margin-right: 0; 1227 | } 1228 | 1229 | .sidebar::before { 1230 | position: absolute; 1231 | top: 0; 1232 | right: 0; 1233 | bottom: 0; 1234 | left: -100vw; 1235 | content: ""; 1236 | display: block; 1237 | background-color: #FFFDF9; 1238 | z-index: -1; 1239 | } 1240 | } 1241 | 1242 | .sidebar .ember-power-select-trigger { 1243 | margin-bottom: 1em; 1244 | } 1245 | 1246 | article { 1247 | padding: 3em 0; 1248 | @media screen and (min-width: 53.75em) { 1249 | float: left; 1250 | display: block; 1251 | margin-right: 2.35765%; 1252 | width: 70.14568%; 1253 | } 1254 | &:last-child { 1255 | margin-right: 0; 1256 | } 1257 | } 1258 | 1259 | article .edit-page { 1260 | height: 18px; 1261 | width: 24px; 1262 | color: #865931; 1263 | display: inline-block; 1264 | float: right; 1265 | font-size: 1rem; 1266 | margin: 0.5em 0 0; 1267 | opacity: 0.4; 1268 | overflow: hidden; 1269 | } 1270 | 1271 | article .edit-page:hover { 1272 | opacity: 1; 1273 | border-bottom: 0; 1274 | } 1275 | article, h1 { 1276 | margin-top: 0; 1277 | } 1278 | 1279 | article hr + h2, 1280 | article hr + h3, 1281 | article hr + div > h2:first-child, 1282 | article hr + div > h3:first-child { 1283 | margin-top: 0 1284 | } 1285 | 1286 | article h4 { 1287 | margin-top: 1.5em; 1288 | } 1289 | 1290 | article .video { 1291 | position: relative; 1292 | width: 100%; 1293 | height: 0; 1294 | padding-bottom: 56.25%; 1295 | margin-top: 2em; 1296 | } 1297 | 1298 | article .video iframe { 1299 | position: absolute; 1300 | top: 0; 1301 | left: 0; 1302 | width: 100%; 1303 | height: 100%; 1304 | } 1305 | 1306 | article footer { 1307 | border-top: 1px solid #F8E7CF; 1308 | margin-top: 5em; 1309 | padding: 1.5em 0; 1310 | } 1311 | 1312 | article footer::after { 1313 | clear: both; 1314 | content: ""; 1315 | display: table; 1316 | } 1317 | 1318 | article footer a { 1319 | text-decoration: underline; 1320 | } 1321 | 1322 | article footer a::after, 1323 | article footer a::before { 1324 | display: inline-block; 1325 | font-weight: 700; 1326 | margin: 0 0.5em; 1327 | transition: transform 200ms; 1328 | } 1329 | 1330 | article footer .previous-guide { 1331 | float: left; 1332 | } 1333 | 1334 | article footer .previous-guide::before { 1335 | content: "\2039"; 1336 | } 1337 | 1338 | article footer .previous-guide:hover::before { 1339 | transform: translateX(-0.5em); 1340 | } 1341 | 1342 | article footer .next-guide { 1343 | float: right; 1344 | } 1345 | 1346 | article footer .next-guide::after { 1347 | content: "\203A"; 1348 | } 1349 | 1350 | article footer .next-guide:hover::after { 1351 | transform: translateX(0.5em); 1352 | } 1353 | 1354 | article.x404 { 1355 | width: 80%; 1356 | margin: 0 auto; 1357 | } 1358 | 1359 | article.x404 img { 1360 | float: left; 1361 | margin: 1em 0.5em; 1362 | max-width: 300px; 1363 | } 1364 | 1365 | .back-to-top { 1366 | -webkit-transition: opacity 200ms; 1367 | -moz-transition: opacity 200ms; 1368 | transition: opacity 200ms; 1369 | color: #dd6a58; 1370 | display: none; 1371 | margin-left: 1em; 1372 | text-align: center; 1373 | } 1374 | @media screen and (max-width: 74em) { 1375 | .back-to-top { 1376 | height: 1.5em; 1377 | width: 1.5em; 1378 | overflow: hidden; 1379 | } 1380 | } 1381 | @media screen and (max-width: 63em) { 1382 | .back-to-top { 1383 | height: 0; 1384 | } 1385 | } 1386 | 1387 | @import 'table-of-contents'; 1388 | 1389 | .old-version-warning { 1390 | background-color: #dd6a58; 1391 | color: #FFFBF5; 1392 | padding: 0.5em; 1393 | border-radius: 3px; 1394 | margin-bottom: 40px; 1395 | } 1396 | 1397 | .old-version-warning .btn { 1398 | float: right; 1399 | font-weight: bold; 1400 | font-size: 12px; 1401 | color: #FFFBF5; 1402 | padding: 0.3em 0.8em; 1403 | background-color: #e69082; 1404 | border-radius: 3px; 1405 | } 1406 | 1407 | .old-version-warning .btn:hover { 1408 | background: #d4442e; 1409 | } 1410 | 1411 | .ds-dropdown-menu { 1412 | right: 0 !important; 1413 | font-size: 0.8rem; 1414 | border-color: #aaa; 1415 | } 1416 | @media (min-width: 53.75em) { 1417 | .ds-dropdown-menu { 1418 | left: auto !important; 1419 | } 1420 | } 1421 | @font-face { 1422 | font-family: 'fontello'; 1423 | src: url("../fonts/fontello.eot?32151418"); 1424 | src: url("../fonts/fontello.eot?32151418#iefix") format("embedded-opentype"),url("../fonts/fontello.woff2?32151418") format("woff2"),url("../fonts/fontello.woff?32151418") format("woff"),url("../fonts/fontello.ttf?32151418") format("truetype"),url("../fonts/fontello.svg?32151418#fontello") format("svg"); 1425 | font-weight: normal; 1426 | font-style: normal; 1427 | } 1428 | 1429 | [class^="icon-"]:before, 1430 | [class*=" icon-"]:before { 1431 | font-family: "fontello"; 1432 | font-style: normal; 1433 | font-weight: normal; 1434 | speak: none; 1435 | display: inline-block; 1436 | text-decoration: inherit; 1437 | width: 1em; 1438 | margin-right: 0.2em; 1439 | text-align: center; 1440 | font-variant: normal; 1441 | text-transform: none; 1442 | line-height: 1em; 1443 | margin-left: 0.2em; 1444 | -webkit-font-smoothing: antialiased; 1445 | -moz-osx-font-smoothing: grayscale; 1446 | } 1447 | 1448 | .icon-search:before { 1449 | content: '\e800'; 1450 | } 1451 | 1452 | .icon-cancel:before { 1453 | content: '\e801'; 1454 | } 1455 | 1456 | .icon-link:before { 1457 | content: '\e802'; 1458 | } 1459 | 1460 | .icon-pencil:before { 1461 | content: '\e803'; 1462 | } 1463 | 1464 | .icon-fork:before { 1465 | content: '\e804'; 1466 | } 1467 | 1468 | .icon-github:before { 1469 | content: '\e805'; 1470 | } 1471 | 1472 | .icon-twitter:before { 1473 | content: '\e806'; 1474 | } 1475 | 1476 | .icon-gplus:before { 1477 | content: '\e807'; 1478 | } 1479 | 1480 | .icon-attention-circled:before { 1481 | content: '\e808'; 1482 | } 1483 | 1484 | .icon-link-ext:before { 1485 | content: '\f08e'; 1486 | } 1487 | 1488 | .note { 1489 | background: rgba(0,0,0,0.2); 1490 | } 1491 | 1492 | .note::before { 1493 | content: "Note"; 1494 | } 1495 | 1496 | .previous-guide { 1497 | float: left; 1498 | border-bottom: 1px solid transparent; 1499 | } 1500 | 1501 | .next-guide { 1502 | float: right; 1503 | border-bottom: 1px solid transparent; 1504 | } 1505 | 1506 | .visually-hidden { 1507 | border: 0 !important; 1508 | clip: rect(0 0 0 0) !important; 1509 | height: 1px !important; 1510 | margin: -1px !important; 1511 | overflow: hidden !important; 1512 | padding: 0 !important; 1513 | position: absolute !important; 1514 | width: 1px !important; 1515 | } 1516 | 1517 | .footer a { 1518 | text-decoration: none; 1519 | } 1520 | 1521 | @import "ember-power-select"; 1522 | @import "code-block-file-name"; 1523 | @import "prism-overrides"; 1524 | @import "search-input"; 1525 | -------------------------------------------------------------------------------- /app/templates/application.hbs: -------------------------------------------------------------------------------- 1 | {{head-layout}} 2 | 3 | {{#es-header}} 4 | {{#es-navbar}} 5 |
6 | {{search-input projectVersion=page.currentVersion}} 7 |
8 | {{/es-navbar}} 9 | {{/es-header}} 10 | 11 |
12 | {{outlet}} 13 |
14 | 15 | {{es-footer}} 16 | -------------------------------------------------------------------------------- /app/templates/components/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/app/templates/components/.gitkeep -------------------------------------------------------------------------------- /app/templates/components/chapter-links.hbs: -------------------------------------------------------------------------------- 1 | {{#if page.isFirstPage}} 2 | {{#if page.previousSection}} 3 | {{#link-to 'version.show' previousSectionPage.url class="previous-guide"}}{{previousSectionPage.title}}{{/link-to}} 4 | {{/if}} 5 | {{else}} 6 | {{#if page.previousPage}} 7 | {{#link-to 'version.show' page.previousPage.url class="previous-guide"}}{{page.previousPage.title}}{{/link-to}} 8 | {{/if}} 9 | {{/if}} 10 | 11 | 12 | {{#if page.isLastPage}} 13 | {{#if page.nextSection}} 14 | {{#link-to 'version.show' nextSectionPage.url class="next-guide"}}We've finished covering {{page.currentSection.title}}. Next up: {{page.nextSection.title}} - {{nextSectionPage.title}}{{/link-to}} 15 | {{/if}} 16 | {{else}} 17 | {{#link-to 'version.show' page.nextPage.url class="next-guide"}}{{page.nextPage.title}}{{/link-to}} 18 | {{/if}} 19 | -------------------------------------------------------------------------------- /app/templates/components/dropdown-header.hbs: -------------------------------------------------------------------------------- 1 |
2 |
3 | 4 | {{yield}} 5 | 6 |
7 |
8 |
-------------------------------------------------------------------------------- /app/templates/components/guides-article.hbs: -------------------------------------------------------------------------------- 1 | {{#unless (eq version currentVersion)}} 2 |
3 | Old Guides - You are viewing the guides for Ember {{version}}. 4 | 5 | {{#link-to 'version.show' 'release' path class="btn"}} VIEW {{currentVersion}} {{/link-to}} 6 |
7 | {{/unless}} 8 | 9 | Edit Page 11 | 12 |

13 | {{page.currentPage.title}} 14 |

15 |
16 | 17 | {{markdown-to-html model.content extensions='showdown-section-groups'}} 18 | 19 | {{chapter-links pages=pages}} 20 | -------------------------------------------------------------------------------- /app/templates/components/search-input.hbs: -------------------------------------------------------------------------------- 1 | 11 | 12 | {{!-- Search results dropdown --}} 13 | {{#ember-tether 14 | target='#search-input' 15 | targetAttachment='bottom left' 16 | attachment='top left' 17 | constraints=_resultTetherConstraints 18 | class='ds-dropdown-results' 19 | }} 20 | {{#if showDropdown}} 21 | 22 | {{#dropdown-header}} 23 | Search Results 24 | {{/dropdown-header}} 25 | 26 | {{#each searchService.results as |result|}} 27 | {{search-result result=result}} 28 | {{else}} 29 |
30 |
31 |

No results found. Try searching the deprecations guide.

32 |
33 |
34 | {{/each}} 35 |
36 | 37 | Search Powered by Algolia 38 | 39 |
40 |
41 | {{/if}} 42 | {{/ember-tether}} 43 | -------------------------------------------------------------------------------- /app/templates/components/search-result.hbs: -------------------------------------------------------------------------------- 1 |
3 |
4 |
5 | {{sectionTitle}} 6 |
7 | 20 |
21 |
22 | -------------------------------------------------------------------------------- /app/templates/components/table-of-contents.hbs: -------------------------------------------------------------------------------- 1 | {{!-- #{build_toc_for(pages)} --}} 2 | {{#each data as |page|}} 3 | {{#unless (or page.skipToc (get page 'skip-toc'))}} 4 |
  • 5 | {{#if page.pages}} 6 | {{#cp-panel open=(eq currentSection.id page.id) as |p|}} 7 | {{#if fastboot.isFastBoot}} 8 | {{#link-to 'version.show' page.id activeClass="selected" class="cp-Panel-toggle" data-test-toc-link=page.title}} 9 | {{page.title}} 10 | {{/link-to}} 11 | {{else}} 12 | {{#p.toggle data-test-toc-title=page.title}} 13 | {{page.title}} 14 | {{/p.toggle}} 15 | {{/if}} 16 | 17 | {{#p.body}} 18 | {{table-of-contents data=page.pages currentPage=currentPage level=(inc level)}} 19 | {{/p.body}} 20 | {{/cp-panel}} 21 | {{else}} 22 | {{#link-to 'version.show' page.url activeClass="selected" data-test-toc-link=page.title}} 23 | {{page.title}} 24 | {{/link-to}} 25 | {{/if}} 26 |
  • 27 | {{/unless}} 28 | {{/each}} 29 | -------------------------------------------------------------------------------- /app/templates/error.hbs: -------------------------------------------------------------------------------- 1 |
    2 | {{#if (contains model.errors.0.status (array "404" "403"))}} 3 | 4 |

    Ack! 404 friend, you're in the wrong place

    5 | 6 |

    7 | This page wasn't found. If you were looking for documentation, please try 8 | the Guides section of the site. If you expected 9 | something else to be here, please file a ticket. 10 |

    11 | {{else}} 12 | 13 |

    Ack! An unknown error has occured!

    14 | 15 |

    16 | We're not quite sure what happened. If you were looking for documentation, please try 17 | the Guides section of the site. If you expected 18 | something else to be here, please file a ticket. 19 |

    20 | {{/if}} 21 |
    22 | -------------------------------------------------------------------------------- /app/templates/version.hbs: -------------------------------------------------------------------------------- 1 | 16 | 17 | {{outlet}} 18 | -------------------------------------------------------------------------------- /app/templates/version/index.hbs: -------------------------------------------------------------------------------- 1 | {{guides-article model=model.content pages=model.pages path='index' version=model.version currentVersion=model.currentVersion}} 2 | -------------------------------------------------------------------------------- /app/templates/version/show.hbs: -------------------------------------------------------------------------------- 1 | {{guides-article model=model.content pages=model.pages path=model.path version=model.version currentVersion=model.currentVersion}} 2 | -------------------------------------------------------------------------------- /appveyor.yml: -------------------------------------------------------------------------------- 1 | # Test against the latest version of this Node.js version 2 | environment: 3 | nodejs_version: "8" 4 | 5 | # Install scripts. (runs after repo cloning) 6 | install: 7 | # Get the latest stable version of Node.js or io.js 8 | - ps: Install-Product node $env:nodejs_version 9 | # install modules 10 | - npm install 11 | 12 | # Post-install test scripts. 13 | test_script: 14 | # Output useful info for debugging. 15 | - node --version 16 | - npm --version 17 | # run tests 18 | - npm test 19 | 20 | # Don't actually build. 21 | build: off 22 | -------------------------------------------------------------------------------- /config/deploy.js: -------------------------------------------------------------------------------- 1 | /* eslint-env node */ 2 | 'use strict'; 3 | 4 | let credentials; 5 | 6 | try { 7 | credentials = require('./credentials.json'); 8 | } catch (e) { 9 | credentials = {}; 10 | } 11 | 12 | module.exports = function(deployTarget) { 13 | let ENV = { 14 | build: {}, 15 | 'prember-algolia': { 16 | indexName: credentials.algoliaIndex || process.env.ALGOLIA_INDEX, 17 | applicationId: credentials.algoliaApplication || process.env.ALGOLIA_APPLICATION, 18 | apiKey: credentials.algoliaKey || process.env.ALGOLIA_KEY, 19 | tagsToExclude: '.old-version-warning,.edit-page,code,pre', 20 | cssSelector: 'section', 21 | versionsToIgnore: ['1.x', '2.x'], 22 | versionPattern: /^(v\d+\.\d+\.\d+)\//, 23 | pathPattern: /^v\d+\.\d+\.\d+\/(.*)\/index.html$/, 24 | } 25 | }; 26 | 27 | if (deployTarget === 'development') { 28 | ENV.build.environment = 'development'; 29 | // configure other plugins for development deploy target here 30 | } 31 | 32 | if (deployTarget === 'staging') { 33 | ENV.build.environment = 'production'; 34 | // configure other plugins for staging deploy target here 35 | } 36 | 37 | if (deployTarget === 'production') { 38 | ENV.build.environment = 'production'; 39 | // configure other plugins for production deploy target here 40 | } 41 | 42 | // Note: if you need to build some configuration asynchronously, you can return 43 | // a promise that resolves with the ENV object instead of returning the 44 | // ENV object synchronously. 45 | return ENV; 46 | }; 47 | -------------------------------------------------------------------------------- /config/environment.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | module.exports = function(environment) { 4 | let ENV = { 5 | modulePrefix: 'guides-app', 6 | environment, 7 | rootURL: '/', 8 | locationType: 'trailing-history', 9 | historySupportMiddleware: true, 10 | EmberENV: { 11 | FEATURES: { 12 | // Here you can enable experimental features on an ember canary build 13 | // e.g. 'with-controller': true 14 | }, 15 | EXTEND_PROTOTYPES: { 16 | // Prevent Ember Data from overriding Date.parse. 17 | Date: false 18 | } 19 | }, 20 | 21 | APP: { 22 | // Here you can pass flags/options to your application instance 23 | // when it is created 24 | }, 25 | 26 | fastboot: { 27 | hostWhitelist: [/^localhost:\d+$/, /^127\.0\.0\.1:\d+$/] 28 | }, 29 | 30 | showdown: { 31 | ghCompatibleHeaderId: true, 32 | prefixHeaderId: 'toc_' 33 | }, 34 | 35 | 'ember-body-class': { 36 | includeRouteName: false 37 | }, 38 | 39 | 'ember-algolia': { 40 | algoliaId: 'Y1OMR4C7MF', 41 | algoliaKey: '5d01c83734dc36754d9e94cbf6f8964d' 42 | }, 43 | 44 | metricsAdapters: [ 45 | { 46 | name: 'GoogleAnalytics', 47 | environments: ['production'], 48 | config: { 49 | id: 'UA-27675533-1', 50 | require: ['linkid'] 51 | } 52 | }, 53 | ], 54 | 55 | 'ember-meta': { 56 | description: 'Ember.js helps developers be more productive out of the box. Designed with developer ergonomics in mind, its friendly APIs help you get your job done—fast.', 57 | imgSrc: '/images/logos/ember.png', 58 | siteName: 'Ember', 59 | title: 'Ember Guides', 60 | twitterUsername: '@emberjs', 61 | url: 'https://guides.emberjs.com/' 62 | }, 63 | 64 | deprecationsGuideURL: 'https://www.emberjs.com/deprecations/' 65 | }; 66 | 67 | if (environment === 'development') { 68 | // ENV.APP.LOG_RESOLVER = true; 69 | // ENV.APP.LOG_ACTIVE_GENERATION = true; 70 | // ENV.APP.LOG_TRANSITIONS = true; 71 | // ENV.APP.LOG_TRANSITIONS_INTERNAL = true; 72 | // ENV.APP.LOG_VIEW_LOOKUPS = true; 73 | } 74 | 75 | if (environment === 'test') { 76 | // Testem prefers this... 77 | ENV.locationType = 'none'; 78 | 79 | // keep test console output quieter 80 | ENV.APP.LOG_ACTIVE_GENERATION = false; 81 | ENV.APP.LOG_VIEW_LOOKUPS = false; 82 | 83 | ENV.APP.rootElement = '#ember-testing'; 84 | ENV.APP.autoboot = false; 85 | 86 | ENV['ember-tether'] = { 87 | bodyElementId: 'ember-testing' 88 | }; 89 | 90 | } 91 | 92 | if (environment === 'production') { 93 | // here you can enable a production-specific feature 94 | } 95 | 96 | return ENV; 97 | }; 98 | -------------------------------------------------------------------------------- /config/targets.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const browsers = [ 4 | 'last 1 Chrome versions', 5 | 'last 1 Firefox versions', 6 | 'last 1 Safari versions' 7 | ]; 8 | 9 | const isCI = !!process.env.CI; 10 | const isProduction = process.env.EMBER_ENV === 'production'; 11 | 12 | if (isCI || isProduction) { 13 | browsers.push('ie 11'); 14 | } 15 | 16 | module.exports = { 17 | browsers 18 | }; 19 | -------------------------------------------------------------------------------- /ember-cli-build.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const EmberApp = require('ember-cli/lib/broccoli/ember-app'); 4 | 5 | module.exports = function(defaults) { 6 | let app = new EmberApp(defaults, { 7 | 'ember-prism': { 8 | 'theme': 'okaidia', 9 | 'components': [ 10 | 'apacheconf', 11 | 'bash', 12 | 'css', 13 | 'handlebars', 14 | 'http', 15 | 'javascript', 16 | 'json', 17 | 'markup-templating', 18 | 'ruby', 19 | 'scss', 20 | ], 21 | 'plugins': ['line-numbers', 'normalize-whitespace'] 22 | }, 23 | fingerprint: { 24 | extensions: ['js', 'css', 'map'], 25 | } 26 | }); 27 | 28 | app.import('node_modules/compare-versions/index.js'); 29 | 30 | return app.toTree(); 31 | }; 32 | -------------------------------------------------------------------------------- /lib/content-guides-generator/index.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const BroccoliMergeTrees = require('broccoli-merge-trees'); 4 | const Funnel = require('broccoli-funnel'); 5 | const StaticSiteJson = require('broccoli-static-site-json'); 6 | const walkSync = require('walk-sync'); 7 | const writeFile = require('broccoli-file-creator'); 8 | const yaml = require('js-yaml'); 9 | 10 | const { readFileSync } = require('fs'); 11 | const { Serializer } = require('jsonapi-serializer'); 12 | const { extname } = require('path'); 13 | 14 | const guidesSrcPkg = 'node_modules/@ember-learn/guides-source'; 15 | const guidesSourcePublic = new Funnel(`${guidesSrcPkg}/public`); 16 | 17 | const VersionsSerializer = new Serializer('version', { 18 | attributes: [ 19 | 'allVersions', 20 | 'currentVersion', 21 | ], 22 | }); 23 | 24 | const versions = yaml.safeLoad(readFileSync(`${guidesSrcPkg}/versions.yml`, 'utf8')); 25 | 26 | let premberVersions = [...versions.allVersions, 'release']; 27 | 28 | const urls = premberVersions.map(version => `/${version}`); 29 | 30 | premberVersions.forEach((premberVersion) => { 31 | const filesVersion = ['current', 'release'].includes(premberVersion) ? versions.currentVersion : premberVersion; 32 | const paths = walkSync(`${guidesSrcPkg}/guides/${filesVersion}`); 33 | 34 | const mdFiles = paths. 35 | filter(path => extname(path) === '.md') 36 | .map(path => path.replace(/\.md/, '')) 37 | .map(path => path.replace(/\/index$/, '')); 38 | 39 | mdFiles.forEach((file) => { 40 | urls.push(`/${premberVersion}/${file}`) 41 | }) 42 | }); 43 | 44 | // setting an ID so that it's not undefined 45 | versions.id = 'versions'; 46 | 47 | const jsonTrees = versions.allVersions.map((version) => new StaticSiteJson(`${guidesSrcPkg}/guides/${version}`, { 48 | contentFolder: `content/${version}`, 49 | contentTypes: ['content', 'description'], 50 | type: 'contents', 51 | attributes: ['canonical', 'redirect'], 52 | })); 53 | 54 | var versionsFile = writeFile('/content/versions.json', JSON.stringify(VersionsSerializer.serialize(versions))); 55 | 56 | module.exports = { 57 | name: 'content-docs-generator', 58 | 59 | isDevelopingAddon() { 60 | return true; 61 | }, 62 | 63 | urlsForPrember() { 64 | return urls; 65 | }, 66 | 67 | treeForPublic() { 68 | return new BroccoliMergeTrees([versionsFile, guidesSourcePublic, ...jsonTrees]); 69 | } 70 | }; 71 | -------------------------------------------------------------------------------- /lib/content-guides-generator/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "content-guides-generator", 3 | "keywords": [ 4 | "ember-addon", 5 | "prember-plugin" 6 | ] 7 | } 8 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "guides-app", 3 | "version": "0.1.0", 4 | "private": true, 5 | "description": "Ember app that powers https://guides.emberjs.com", 6 | "repository": "https://github.com/ember-learn/guides-app", 7 | "license": "MIT", 8 | "author": "Chris Manson ", 9 | "directories": { 10 | "doc": "doc", 11 | "test": "tests" 12 | }, 13 | "scripts": { 14 | "build": "ember build", 15 | "build-production": "ember build -e production", 16 | "lint:js": "eslint ./*.js app blueprints config lib server tests", 17 | "start": "ember serve", 18 | "test": "ember test" 19 | }, 20 | "devDependencies": { 21 | "@ember-learn/guides-source": "^3.7.0", 22 | "algoliasearch": "^3.32.0", 23 | "broccoli-asset-rev": "^3.0.0", 24 | "broccoli-merge-trees": "^3.0.2", 25 | "broccoli-static-site-json": "^3.0.0", 26 | "compare-versions": "3.4.0", 27 | "ember-ajax": "^4.0.2", 28 | "ember-auto-import": "^1.2.19", 29 | "ember-body-class": "^1.1.3", 30 | "ember-cli": "~3.7.1", 31 | "ember-cli-app-version": "^3.0.0", 32 | "ember-cli-babel": "^7.4.1", 33 | "ember-cli-dependency-checker": "^3.1.0", 34 | "ember-cli-deploy": "^1.0.2", 35 | "ember-cli-deploy-build": "^1.1.1", 36 | "ember-cli-deploy-prember-algolia": "^1.0.1", 37 | "ember-cli-eslint": "^4.2.1", 38 | "ember-cli-fastboot": "^2.0.4", 39 | "ember-cli-head": "^0.4.0", 40 | "ember-cli-htmlbars": "^3.0.1", 41 | "ember-cli-htmlbars-inline-precompile": "^2.1.0", 42 | "ember-cli-inject-live-reload": "^2.0.1", 43 | "ember-cli-netlify": "0.1.0", 44 | "ember-cli-qunit": "^4.4.0", 45 | "ember-cli-sass": "^8.0.1", 46 | "ember-cli-shims": "^1.2.0", 47 | "ember-cli-showdown": "^4.4.4", 48 | "ember-cli-sri": "^2.1.0", 49 | "ember-cli-uglify": "^2.0.0", 50 | "ember-collapsible-panel": "^3.2.1", 51 | "ember-composable-helpers": "^2.1.0", 52 | "ember-data": "~3.7.0", 53 | "ember-export-application-global": "^2.0.0", 54 | "ember-href-to": "^2.0.1", 55 | "ember-load-initializers": "^1.0.0", 56 | "ember-maybe-import-regenerator": "^0.1.6", 57 | "ember-meta": "^0.5.0", 58 | "ember-metrics": "^0.13.0", 59 | "ember-percy": "^1.5.0", 60 | "ember-power-select": "^2.2.1", 61 | "ember-prism": "^0.4.0", 62 | "ember-resolver": "^5.0.1", 63 | "ember-source": "~3.5.0", 64 | "ember-styleguide": "^2.5.0", 65 | "ember-test-selectors": "^2.0.0", 66 | "ember-tether": "^1.0.0", 67 | "eslint-plugin-ember": "^6.2.0", 68 | "loader.js": "^4.2.3", 69 | "normalize.css": "^8.0.1", 70 | "percy-client": "^3.0.3", 71 | "prember": "^1.0.2", 72 | "qunit-dom": "^0.8.4", 73 | "showdown-section-groups": "^0.3.0" 74 | }, 75 | "engines": { 76 | "node": "^4.5 || 6.* || >= 7.*" 77 | }, 78 | "ember-addon": { 79 | "paths": [ 80 | "lib/content-guides-generator" 81 | ] 82 | } 83 | } 84 | -------------------------------------------------------------------------------- /public/fonts/config.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "", 3 | "css_prefix_text": "icon-", 4 | "css_use_suffix": false, 5 | "hinting": true, 6 | "units_per_em": 1000, 7 | "ascent": 850, 8 | "glyphs": [ 9 | { 10 | "uid": "9dd9e835aebe1060ba7190ad2b2ed951", 11 | "css": "search", 12 | "code": 59392, 13 | "src": "fontawesome" 14 | }, 15 | { 16 | "uid": "5211af474d3a9848f67f945e2ccaf143", 17 | "css": "cancel", 18 | "code": 59393, 19 | "src": "fontawesome" 20 | }, 21 | { 22 | "uid": "0ddd3e8201ccc7d41f7b7c9d27eca6c1", 23 | "css": "link", 24 | "code": 59394, 25 | "src": "fontawesome" 26 | }, 27 | { 28 | "uid": "d35a1d35efeb784d1dc9ac18b9b6c2b6", 29 | "css": "pencil", 30 | "code": 59395, 31 | "src": "fontawesome" 32 | }, 33 | { 34 | "uid": "bc4b94dd7a9a1dd2e02f9e4648062596", 35 | "css": "fork", 36 | "code": 59396, 37 | "src": "fontawesome" 38 | }, 39 | { 40 | "uid": "5e0a374728ffa8d0ae1f331a8f648231", 41 | "css": "github", 42 | "code": 59397, 43 | "src": "fontawesome" 44 | }, 45 | { 46 | "uid": "627abcdb627cb1789e009c08e2678ef9", 47 | "css": "twitter", 48 | "code": 59398, 49 | "src": "fontawesome" 50 | }, 51 | { 52 | "uid": "0350d80455823796cce38d84b2b91ba6", 53 | "css": "gplus", 54 | "code": 59399, 55 | "src": "custom_icons", 56 | "selected": true, 57 | "svg": { 58 | "path": "M801.9 509.5Q801.9 625.6 753.3 716.2T615 858 409 909Q325.9 909 250 876.7T119.4 789.6 32.4 659 0 500 32.4 341 119.4 210.4 250 123.3 409 91Q568.6 91 683 198.1L572 304.7Q506.7 241.6 409 241.6 340.4 241.6 282.1 276.2T189.7 370.3 155.7 500 189.7 629.7 282.1 723.8 409 758.4Q455.4 758.4 494.1 745.5T558 713.4 601.8 669.6 629.2 623.3 641.2 582H409V441.4H795.2Q801.9 476.6 801.9 509.5ZM1285.7 441.4V558.6H1169.1V675.2H1051.9V558.6H935.3V441.4H1051.9V324.8H1169.1V441.4H1285.7Z", 59 | "width": 1285.7142857142858 60 | }, 61 | "search": [ 62 | "glyph" 63 | ] 64 | }, 65 | { 66 | "uid": "b035c28eba2b35c6ffe92aee8b0df507", 67 | "css": "attention-circled", 68 | "code": 59400, 69 | "src": "fontawesome" 70 | }, 71 | { 72 | "uid": "e15f0d620a7897e2035c18c80142f6d9", 73 | "css": "link-ext", 74 | "code": 61582, 75 | "src": "fontawesome" 76 | } 77 | ] 78 | } -------------------------------------------------------------------------------- /public/fonts/fontello.eot: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/public/fonts/fontello.eot -------------------------------------------------------------------------------- /public/fonts/fontello.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Copyright (C) 2016 by original authors @ fontello.com 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | -------------------------------------------------------------------------------- /public/fonts/fontello.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/public/fonts/fontello.ttf -------------------------------------------------------------------------------- /public/fonts/fontello.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/public/fonts/fontello.woff -------------------------------------------------------------------------------- /public/fonts/fontello.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/public/fonts/fontello.woff2 -------------------------------------------------------------------------------- /public/images/logos/ember.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/public/images/logos/ember.png -------------------------------------------------------------------------------- /public/images/logos/search-by-algolia.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /public/robots.txt: -------------------------------------------------------------------------------- 1 | # http://www.robotstxt.org 2 | User-agent: * 3 | Disallow: 4 | -------------------------------------------------------------------------------- /static.json: -------------------------------------------------------------------------------- 1 | { 2 | "https_only": true, 3 | "root": "dist", 4 | "error_page": "_empty.html", 5 | "force_redirect": true, 6 | "redirects": { 7 | "/": { 8 | "url": "/release/" 9 | }, 10 | "/current/(.*)": { 11 | "url": "/release/$1" 12 | } 13 | }, 14 | "headers": { 15 | "/**/*.html": { 16 | "Cache-Control": "public, max-age=3600" 17 | }, 18 | "/content/**": { 19 | "Cache-Control": "public, max-age=3600" 20 | }, 21 | "/assets/**": { 22 | "Cache-Control": "public, max-age=31536000" 23 | } 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /testem.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | test_page: 'tests/index.html?hidepassed', 3 | disable_watching: true, 4 | launch_in_ci: [ 5 | 'Chrome' 6 | ], 7 | launch_in_dev: [ 8 | 'Chrome' 9 | ], 10 | browser_args: { 11 | Chrome: { 12 | mode: 'ci', 13 | args: [ 14 | // --no-sandbox is needed when running Chrome inside a container 15 | process.env.TRAVIS ? '--no-sandbox' : null, 16 | 17 | '--disable-gpu', 18 | '--headless', 19 | '--remote-debugging-port=0', 20 | '--window-size=1440,900' 21 | ].filter(Boolean) 22 | } 23 | } 24 | }; 25 | -------------------------------------------------------------------------------- /tests/acceptance/cookbook-test.js: -------------------------------------------------------------------------------- 1 | import { click, currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | 5 | module('Acceptance | cookbook', function(hooks) { 6 | setupApplicationTest(hooks); 7 | 8 | /* 9 | Cookbook gets its own tests because it has one additional layer of nesting than 10 | the rest of the guides 11 | */ 12 | 13 | test('visiting /cookbook', async function(assert) { 14 | await visit('/v1.10.0/cookbook/'); 15 | assert.equal(currentURL(), '/v1.10.0/cookbook/'); 16 | await click('.next-guide') 17 | assert.equal(currentURL(), '/v1.10.0/cookbook/contributing'); 18 | await click('.next-guide') 19 | assert.equal(currentURL(), '/v1.10.0/cookbook/contributing/understanding_the_cookbook_format'); 20 | await click('.previous-guide') 21 | assert.equal(currentURL(), '/v1.10.0/cookbook/contributing'); 22 | }); 23 | }); 24 | -------------------------------------------------------------------------------- /tests/acceptance/current-url-test.js: -------------------------------------------------------------------------------- 1 | import { currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | 4 | import { setupApplicationTest } from 'ember-qunit'; 5 | 6 | module('Acceptance | current url', function(hooks) { 7 | setupApplicationTest(hooks); 8 | 9 | test('visiting /release-url', async function(assert) { 10 | await visit('/release'); 11 | let page = this.owner.lookup('service:page'); 12 | let currentVersion = page.get('currentVersion'); 13 | currentVersion = currentVersion.slice(currentVersion.indexOf('v') + 1 || 0, currentVersion.lastIndexOf('.') === currentVersion.indexOf('.') ? currentVersion.length : currentVersion.lastIndexOf('.')) 14 | assert.dom('.ember-basic-dropdown-trigger').hasText(currentVersion); 15 | }); 16 | 17 | test('visiting / redirects you to /release', async function(assert) { 18 | await visit('/'); 19 | let page = this.owner.lookup('service:page'); 20 | assert.equal(currentURL(), "/release"); 21 | 22 | let currentVersion = page.get('currentVersion'); 23 | currentVersion = currentVersion.slice(currentVersion.indexOf('v') + 1 || 0, currentVersion.lastIndexOf('.') === currentVersion.indexOf('.') ? currentVersion.length : currentVersion.lastIndexOf('.')) 24 | assert.dom('.ember-basic-dropdown-trigger').hasText(currentVersion); 25 | }); 26 | }); 27 | -------------------------------------------------------------------------------- /tests/acceptance/error-page-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { visit, currentURL } from '@ember/test-helpers'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | 5 | module('Acceptance | error page', function(hooks) { 6 | setupApplicationTest(hooks); 7 | 8 | test('visiting a non-existent page shows the 404 error-page', async function(assert) { 9 | await visit('/v1.12.0/nonsense/route/'); 10 | 11 | assert.equal(currentURL(), '/v1.12.0/nonsense/route/'); 12 | assert.dom('[data-test-error-page]').exists(); 13 | assert.dom('[data-test-error-message]').hasText(`Ack! 404 friend, you're in the wrong place`); 14 | }); 15 | }); 16 | -------------------------------------------------------------------------------- /tests/acceptance/meta-data-test.js: -------------------------------------------------------------------------------- 1 | import { click, currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | 5 | module('Acceptance | meta data', function(hooks) { 6 | setupApplicationTest(hooks); 7 | 8 | test('meta data title and description', async function(assert) { 9 | await visit('/release/'); 10 | await click('[data-test-toc-title="Routing"]') 11 | await click('[data-test-toc-link="Defining Your Routes"]') 12 | 13 | assert.equal(currentURL(), '/release/routing/defining-your-routes'); 14 | 15 | // lookup title from service because testem modifies title https://github.com/testem/testem/issues/195 16 | let headData = this.owner.lookup('service:head-data'); 17 | let title = headData.title; 18 | 19 | let description = document.head.querySelector('meta[name="description"]'); 20 | 21 | assert.ok(title); 22 | assert.ok(description); 23 | 24 | assert.equal(title, 25 | 'Defining Your Routes - Routing - Ember Guides'); 26 | assert.ok(description.content.startsWith('When your application starts, the router matches the current URL to the routes')); 27 | }); 28 | }); 29 | -------------------------------------------------------------------------------- /tests/acceptance/previous-next-links-test.js: -------------------------------------------------------------------------------- 1 | import { click, currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | 5 | module('Acceptance | previous next links', function(hooks) { 6 | setupApplicationTest(hooks); 7 | 8 | test('navigation by previous and next links', async function(assert) { 9 | await visit('/v2.17.0/models/'); 10 | assert.equal(currentURL(), '/v2.17.0/models/'); 11 | await click('.next-guide') 12 | assert.equal(currentURL(), '/v2.17.0/models/defining-models'); 13 | await click('.next-guide') 14 | assert.equal(currentURL(), '/v2.17.0/models/finding-records'); 15 | await click('.previous-guide') 16 | assert.equal(currentURL(), '/v2.17.0/models/defining-models'); 17 | }); 18 | }); 19 | -------------------------------------------------------------------------------- /tests/acceptance/table-of-contents-test.js: -------------------------------------------------------------------------------- 1 | import { click, currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | 5 | module('Acceptance | table of contents', function(hooks) { 6 | setupApplicationTest(hooks); 7 | 8 | test('navigation by TOC', async function(assert) { 9 | await visit('/v2.17.0/'); 10 | await click('[data-test-toc-title="Routing"]') 11 | await click('[data-test-toc-link="Defining Your Routes"]') 12 | assert.equal(currentURL(), '/v2.17.0/routing/defining-your-routes'); 13 | }); 14 | }); 15 | -------------------------------------------------------------------------------- /tests/acceptance/version-menu-test.js: -------------------------------------------------------------------------------- 1 | import { currentURL, visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | import { selectChoose } from 'ember-power-select/test-support'; 5 | 6 | module('Acceptance | version menu when changing versions', function(hooks) { 7 | setupApplicationTest(hooks); 8 | 9 | test('stays on index page', async function(assert) { 10 | await visit('/v2.17.0/'); 11 | await selectChoose('.ember-basic-dropdown-trigger', '2.10'); 12 | assert.equal(currentURL(), '/v2.10.0'); 13 | }); 14 | 15 | test('stays on same section', async function(assert) { 16 | await visit('/v1.13.0/getting-started/'); 17 | await selectChoose('.ember-basic-dropdown-trigger', '1.12'); 18 | assert.equal(currentURL(), '/v1.12.0/getting-started'); 19 | }); 20 | 21 | test('stays on same section and page', async function(assert) { 22 | await visit('/v3.0.0/object-model/classes-and-instances/'); 23 | await selectChoose('.ember-basic-dropdown-trigger', '3.1'); 24 | assert.equal(currentURL(), '/v3.1.0/object-model/classes-and-instances'); 25 | }); 26 | 27 | test("redirects to index page if current section/page doesn't exist in new version", async function(assert) { 28 | await visit('/v1.10.0/getting-started/using-fixtures/'); 29 | await selectChoose('.ember-basic-dropdown-trigger', '1.13'); 30 | assert.equal(currentURL(), '/v1.13.0'); 31 | }); 32 | }); 33 | -------------------------------------------------------------------------------- /tests/acceptance/visual-regression-test.js: -------------------------------------------------------------------------------- 1 | import { visit } from '@ember/test-helpers'; 2 | import { module, test } from 'qunit'; 3 | import { setupApplicationTest } from 'ember-qunit'; 4 | import { get } from '@ember/object'; 5 | import { percySnapshot } from 'ember-percy'; 6 | 7 | module('Acceptance | visual regression', function(hooks) { 8 | setupApplicationTest(hooks); 9 | 10 | test(`visiting visual regressions with Percy`, async function(assert) { 11 | assert.expect(0); 12 | await visit('/release'); 13 | 14 | let store = this.owner.lookup('service:store'); 15 | let pages = store.peekAll('page'); 16 | 17 | await pages.reduce(async (prev, section) => { 18 | await prev; 19 | 20 | return section.get('pages').reduce(async (prev, page) => { 21 | await prev; 22 | 23 | let url = get(page, 'url'); 24 | 25 | await visit(`/release/${url}`); 26 | 27 | let name = `/${page.url}/index.html`; 28 | 29 | if (page.url.endsWith('index')) { 30 | name = `/${page.url}.html`; 31 | } else if (page.url.endsWith('index/')) { 32 | name = '/index.html'; 33 | } 34 | 35 | await percySnapshot(name); 36 | }, Promise.resolve()); 37 | }, Promise.resolve()); 38 | }); 39 | }); 40 | -------------------------------------------------------------------------------- /tests/helpers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/tests/helpers/.gitkeep -------------------------------------------------------------------------------- /tests/helpers/start-app.js: -------------------------------------------------------------------------------- 1 | import Application from '../../app'; 2 | import config from '../../config/environment'; 3 | import { merge } from '@ember/polyfills'; 4 | import { run } from '@ember/runloop'; 5 | import './percy/register-helpers'; 6 | 7 | export default function startApp(attrs) { 8 | let attributes = merge({}, config.APP); 9 | attributes.autoboot = true; 10 | attributes = merge(attributes, attrs); // use defaults, but you can override; 11 | 12 | return run(() => { 13 | let application = Application.create(attributes); 14 | application.setupForTesting(); 15 | application.injectTestHelpers(); 16 | return application; 17 | }); 18 | } 19 | -------------------------------------------------------------------------------- /tests/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | {{content-for "head"}} 9 | {{content-for "test-head"}} 10 | 11 | 12 | 13 | 14 | 15 | {{content-for "head-footer"}} 16 | {{content-for "test-head-footer"}} 17 | 18 | 19 | {{content-for "body"}} 20 | {{content-for "test-body"}} 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | {{content-for "body-footer"}} 29 | {{content-for "test-body-footer"}} 30 | 31 | 32 | -------------------------------------------------------------------------------- /tests/integration/components/dropdown-header-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { setupRenderingTest } from 'ember-qunit'; 3 | import { render } from '@ember/test-helpers'; 4 | import hbs from 'htmlbars-inline-precompile'; 5 | 6 | module('Integration | Component | dropdown-header', function(hooks) { 7 | setupRenderingTest(hooks); 8 | 9 | test('it renders', async function(assert) { 10 | // Set any properties with this.set('myProperty', 'value'); 11 | // Handle any actions with this.set('myAction', function(val) { ... }); 12 | 13 | await render(hbs`{{dropdown-header}}`); 14 | 15 | assert.equal(this.element.textContent.trim(), ''); 16 | 17 | // Template block usage: 18 | await render(hbs` 19 | {{#dropdown-header}} 20 | template block text 21 | {{/dropdown-header}} 22 | `); 23 | 24 | assert.equal(this.element.textContent.trim(), 'template block text'); 25 | }); 26 | }); 27 | -------------------------------------------------------------------------------- /tests/integration/components/search-input-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { setupRenderingTest } from 'ember-qunit'; 3 | import { fillIn } from '@ember/test-helpers'; 4 | import hbs from 'htmlbars-inline-precompile'; 5 | import { set } from '@ember/object'; 6 | 7 | module('Integration | Component | search input', function(hooks) { 8 | setupRenderingTest(hooks); 9 | 10 | test('no search hits display no results', async function(assert) { 11 | 12 | let searchService = this.owner.lookup('service:search'); 13 | 14 | set(searchService, 'doSearch', () => { 15 | return []; 16 | }); 17 | 18 | await this.render(hbs`{{search-input}}`); 19 | 20 | await fillIn('#search-input', 'model'); 21 | 22 | assert.dom('.algolia-docsearch-suggestion--noresults').exists(); 23 | }); 24 | }); 25 | -------------------------------------------------------------------------------- /tests/integration/helpers/shorten-version-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { setupRenderingTest } from 'ember-qunit'; 3 | import { render } from '@ember/test-helpers'; 4 | import hbs from 'htmlbars-inline-precompile'; 5 | 6 | module('Integration | Helper | shorten-version', function(hooks) { 7 | setupRenderingTest(hooks); 8 | 9 | test('it shortens version v3.2.0 to 3.2', async function(assert) { 10 | this.set('inputValue', 'v3.2.0'); 11 | await render(hbs`{{shorten-version inputValue}}`); 12 | assert.dom(this.element).hasText('3.2'); 13 | }); 14 | 15 | test('it keeps version 3.2 without changes', async function(assert) { 16 | this.set('inputValue', '3.2'); 17 | await render(hbs`{{shorten-version inputValue}}`); 18 | assert.dom(this.element).hasText('3.2'); 19 | }); 20 | }); 21 | -------------------------------------------------------------------------------- /tests/test-helper.js: -------------------------------------------------------------------------------- 1 | import Application from '../app'; 2 | import config from '../config/environment'; 3 | import { setApplication } from '@ember/test-helpers'; 4 | import { start } from 'ember-qunit'; 5 | 6 | setApplication(Application.create(config.APP)); 7 | 8 | start(); 9 | -------------------------------------------------------------------------------- /tests/unit/services/head-data-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { setupTest } from 'ember-qunit'; 3 | 4 | module('Unit | Service | head data', function(hooks) { 5 | setupTest(hooks); 6 | 7 | test('title with currentPage and currentSelect details', function(assert) { 8 | 9 | let headDataService = this.owner.lookup('service:head-data'); 10 | let pageService = this.owner.lookup('service:page'); 11 | 12 | assert.equal(headDataService.title, 'Ember Guides', 'Default title without currentPage nor currentSelection'); 13 | 14 | let currentPage = { 15 | title: 'CurrentPage title' 16 | } 17 | pageService.set('currentPage', currentPage); 18 | 19 | assert.equal(headDataService.title, 'Ember Guides', 'Default title without currentSelection'); 20 | 21 | let currentSection = { 22 | title: 'CurrentSection title' 23 | } 24 | pageService.set('currentSection', currentSection); 25 | 26 | assert.equal(headDataService.title, 'CurrentPage title - CurrentSection title - Ember Guides', 'Title with currentPage and currentSelection'); 27 | }); 28 | }); 29 | -------------------------------------------------------------------------------- /tests/unit/services/search-test.js: -------------------------------------------------------------------------------- 1 | import { module, test } from 'qunit'; 2 | import { setupTest } from 'ember-qunit'; 3 | 4 | module('Unit | Service | search', function(hooks) { 5 | setupTest(hooks); 6 | 7 | // Replace this with your real tests. 8 | test('it exists', function(assert) { 9 | let service = this.owner.lookup('service:search'); 10 | assert.ok(service); 11 | }); 12 | }); 13 | -------------------------------------------------------------------------------- /vendor/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ember-learn/guides-app/2e3a25fd3b924d6ced52d15d65f19232cc3145b3/vendor/.gitkeep -------------------------------------------------------------------------------- /vendor/ember-cli-spinkit/styles/spinkit-cube-grid.css: -------------------------------------------------------------------------------- 1 | 2 | /* 3 | * Usage: 4 | * 5 |
    6 |
    7 |
    8 |
    9 |
    10 |
    11 |
    12 |
    13 |
    14 |
    15 |
    16 | * 17 | */ 18 | .sk-cube-grid { 19 | width: 40px; 20 | height: 40px; 21 | margin: 40px auto; 22 | /* 23 | * Spinner positions 24 | * 1 2 3 25 | * 4 5 6 26 | * 7 8 9 27 | */ } 28 | .sk-cube-grid .sk-cube { 29 | width: 33.33%; 30 | height: 33.33%; 31 | background-color: #f16f58; 32 | float: left; 33 | -webkit-animation: sk-cubeGridScaleDelay 1.3s infinite ease-in-out; 34 | animation: sk-cubeGridScaleDelay 1.3s infinite ease-in-out; } 35 | .sk-cube-grid .sk-cube1 { 36 | -webkit-animation-delay: 0.2s; 37 | animation-delay: 0.2s; } 38 | .sk-cube-grid .sk-cube2 { 39 | -webkit-animation-delay: 0.3s; 40 | animation-delay: 0.3s; } 41 | .sk-cube-grid .sk-cube3 { 42 | -webkit-animation-delay: 0.4s; 43 | animation-delay: 0.4s; } 44 | .sk-cube-grid .sk-cube4 { 45 | -webkit-animation-delay: 0.1s; 46 | animation-delay: 0.1s; } 47 | .sk-cube-grid .sk-cube5 { 48 | -webkit-animation-delay: 0.2s; 49 | animation-delay: 0.2s; } 50 | .sk-cube-grid .sk-cube6 { 51 | -webkit-animation-delay: 0.3s; 52 | animation-delay: 0.3s; } 53 | .sk-cube-grid .sk-cube7 { 54 | -webkit-animation-delay: 0.0s; 55 | animation-delay: 0.0s; } 56 | .sk-cube-grid .sk-cube8 { 57 | -webkit-animation-delay: 0.1s; 58 | animation-delay: 0.1s; } 59 | .sk-cube-grid .sk-cube9 { 60 | -webkit-animation-delay: 0.2s; 61 | animation-delay: 0.2s; } 62 | 63 | @-webkit-keyframes sk-cubeGridScaleDelay { 64 | 0%, 70%, 100% { 65 | -webkit-transform: scale3D(1, 1, 1); 66 | transform: scale3D(1, 1, 1); } 67 | 35% { 68 | -webkit-transform: scale3D(0, 0, 1); 69 | transform: scale3D(0, 0, 1); } } 70 | 71 | @keyframes sk-cubeGridScaleDelay { 72 | 0%, 70%, 100% { 73 | -webkit-transform: scale3D(1, 1, 1); 74 | transform: scale3D(1, 1, 1); } 75 | 35% { 76 | -webkit-transform: scale3D(0, 0, 1); 77 | transform: scale3D(0, 0, 1); } } 78 | -------------------------------------------------------------------------------- /vendor/ember-cli-spinkit/templates/components/spinkit-cube-grid.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 |
    5 |
    6 |
    7 |
    8 |
    9 |
    10 |
    11 |
    12 |
    13 |
    14 |
    15 | --------------------------------------------------------------------------------