├── .circleci └── config.yml ├── .editorconfig ├── .ember-cli ├── .eslintignore ├── .eslintrc.js ├── .gitignore ├── .travis.yml ├── .watchmanconfig ├── README.md ├── app ├── app.js ├── components │ ├── .gitkeep │ ├── muuri-grid-component.js │ └── muuri-item-component.js ├── controllers │ ├── .gitkeep │ ├── index.js │ └── schema.js ├── helpers │ ├── .gitkeep │ └── is-image-type.js ├── index.html ├── lib │ ├── generateUUID.js │ └── sanctu.js ├── models │ └── .gitkeep ├── resolver.js ├── router.js ├── routes │ ├── .gitkeep │ ├── application.js │ ├── index.js │ └── schema.js ├── services │ └── extension.js ├── styles │ └── app.css └── templates │ ├── application.hbs │ ├── components │ ├── .gitkeep │ ├── check-icon.hbs │ ├── cross-icon.hbs │ ├── muuri-grid-component.hbs │ ├── muuri-item-component.hbs │ ├── pencil-icon.hbs │ └── plus-icon.hbs │ ├── index.hbs │ └── schema.hbs ├── config ├── deploy.js ├── dotenv.js ├── environment.js └── targets.js ├── dist ├── assets │ ├── auto-import-fastboot-d41d8cd98f00b204e9800998ecf8427e.js │ ├── contentful-fragment-c7a594e5271f5bf6e7069c108e0b2552.js │ ├── contentful-fragment-c8008061df33bffd6a340d2f150eca77.css │ ├── vendor-4a1d15267e4f3f19c160f2f0fcdeba97.js │ └── vendor-e785a04175e54de5481bdb16263f496d.css ├── draggable.svg ├── extension.json ├── index.html └── robots.txt ├── ember-cli-build.js ├── package.json ├── public ├── draggable.svg ├── extension.json └── robots.txt ├── testem.js ├── tests ├── helpers │ └── .gitkeep ├── index.html ├── integration │ ├── .gitkeep │ ├── components │ │ ├── muuri-grid-component-test.js │ │ └── muuri-item-component-test.js │ └── helpers │ │ └── image-type-test.js ├── test-helper.js └── unit │ ├── .gitkeep │ ├── controllers │ ├── index-test.js │ └── schema-test.js │ ├── routes │ ├── application-test.js │ ├── index-test.js │ └── schema-test.js │ └── services │ └── extension-test.js ├── vendor └── .gitkeep └── yarn.lock /.circleci/config.yml: -------------------------------------------------------------------------------- 1 | defaults: &defaults 2 | working_directory: ~/contentful-fragment 3 | environment: 4 | TZ: "/usr/share/zoneinfo/America/New_York" 5 | docker: 6 | - image: circleci/node:8-browsers 7 | 8 | 9 | version: 2 10 | jobs: 11 | install-dependencies: 12 | <<: *defaults 13 | steps: 14 | - checkout 15 | - attach_workspace: 16 | at: ~/contentful-fragment 17 | - restore_cache: 18 | keys: 19 | - v1-dependencies-{{ checksum "yarn.lock" }} 20 | - v1-dependencies- 21 | - run: yarn install 22 | - save_cache: 23 | paths: 24 | - node_modules 25 | key: v1-dependencies-{{ checksum "yarn.lock" }} 26 | - persist_to_workspace: 27 | root: . 28 | paths: 29 | - . 30 | 31 | run-tests: 32 | <<: *defaults 33 | steps: 34 | - attach_workspace: 35 | at: ~/contentful-fragment 36 | - run: yarn test 37 | 38 | workflows: 39 | version: 2 40 | build-n-test: 41 | jobs: 42 | - install-dependencies 43 | - run-tests: 44 | requires: 45 | - install-dependencies 46 | -------------------------------------------------------------------------------- /.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 | # unconventional js 2 | /blueprints/*/files/ 3 | /vendor/ 4 | 5 | # compiled output 6 | /dist/ 7 | /tmp/ 8 | 9 | # dependencies 10 | /bower_components/ 11 | 12 | # misc 13 | /coverage/ 14 | 15 | # ember-try 16 | /.node_modules.ember-try/ 17 | -------------------------------------------------------------------------------- /.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 | overrides: [ 20 | // node files 21 | { 22 | files: [ 23 | 'ember-cli-build.js', 24 | 'testem.js', 25 | 'blueprints/*/index.js', 26 | 'config/**/*.js', 27 | 'lib/*/index.js' 28 | ], 29 | parserOptions: { 30 | sourceType: 'script', 31 | ecmaVersion: 2015, 32 | ecmaFeatures: { 33 | experimentalObjectRestSpread: true 34 | } 35 | }, 36 | env: { 37 | browser: false, 38 | node: true 39 | } 40 | } 41 | ] 42 | }; 43 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/ignore-files/ for more about ignoring files. 2 | 3 | # compiled output 4 | /tmp/ 5 | 6 | # dependencies 7 | /bower_components/ 8 | /node_modules/ 9 | 10 | # misc 11 | /.sass-cache 12 | /connect.lock 13 | /coverage/ 14 | /libpeerconnection.log 15 | /npm-debug.log* 16 | /testem.log 17 | /yarn-error.log 18 | 19 | # ember-try 20 | /.node_modules.ember-try/ 21 | /bower.json.ember-try 22 | /package.json.ember-try 23 | 24 | .env 25 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | --- 2 | language: node_js 3 | node_js: 4 | - "6" 5 | 6 | sudo: false 7 | dist: trusty 8 | 9 | addons: 10 | chrome: stable 11 | 12 | cache: 13 | directories: 14 | - $HOME/.npm 15 | 16 | env: 17 | global: 18 | # See https://git.io/vdao3 for details. 19 | - JOBS=1 20 | 21 | before_install: 22 | - npm config set spin false 23 | 24 | script: 25 | - npm run lint:js 26 | - npm test 27 | -------------------------------------------------------------------------------- /.watchmanconfig: -------------------------------------------------------------------------------- 1 | { 2 | "ignore_dirs": ["tmp", "dist"] 3 | } 4 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Contentful Fragment 2 | 3 | Contentful Fragment is a [Contentful UI Extension](https://www.contentful.com/developers/docs/concepts/uiextensions/) that allows content 4 | modellers to add a "mini" model fragment inside of their content types. 5 | It's intended for small, repeatable pieces of content that don't necessarily 6 | warrant their own model. 7 | 8 | ## Installation on Contentful Space 9 | 10 | 1. Navigate to your Contentful Space 11 | 2. Select the "Settings" dropdown, and click "Extensions" 12 | 3. Click the "Add Extension" button, and select "Install from Github" 13 | 4. To get auto-updates on the master channel, enter `https://github.com/sanctuarycomputer/contentful-fragment/blob/master/dist/extension.json` 14 | 15 | Finally, click "Install". 16 | 17 | Now, when you're adding fields to your Content Model, you'll be able to 18 | use Contentful Fragment with any `JSON object` field type. Just navigate 19 | to the "Appearance" settings for your field, and select "Contentful Fragment"! 20 | 21 | ## Using a predefined schema for all instances 22 | 23 | In order to strictly dictate the schema for a Contentful Fragment so that 24 | your content editor can not change it (and potentially break the site), you'll 25 | need to use a predefined schema. 26 | 27 | A predefined schema is simply a string, formatted like so: 28 | ``` 29 | [Schema Key]:[Schema Type],[Schema Key]:[Schema Type] 30 | ``` 31 | 32 | For example: 33 | ``` 34 | Event Name:Symbol,Event Date:Date,Company Logo:Blob 35 | ``` 36 | 37 | To configure that schema, as a Content Modeller, navigate to the "Appearance" 38 | tab of your Contentful field, and under "Predefined Schema", enter your schema 39 | string. 40 | 41 | This will disable the schema editor from the Fragment UI, and preload that 42 | schema for your fragments to use in each instance of that model. 43 | 44 | ## Prerequisites 45 | 46 | You will need the following things properly installed on your computer. 47 | 48 | * [Git](https://git-scm.com/) 49 | * [Node.js](https://nodejs.org/) (with npm) 50 | * [Ember CLI](https://ember-cli.com/) 51 | * [Google Chrome](https://google.com/chrome/) 52 | 53 | ## Installation 54 | 55 | * `git clone ` this repository 56 | * `cd contentful-fragment` 57 | * `npm install` 58 | 59 | ## Running / Development 60 | 61 | * `yarn start` 62 | * Visit your app at [http://localhost:4200](http://localhost:4200). 63 | * Visit your tests at [http://localhost:4200/tests](http://localhost:4200/tests). 64 | 65 | Don't forget to change **line 5** of **application.js** back to `const DEV = false;` before you commit your changes! 66 | 67 | ### Code Generators 68 | 69 | Make use of the many generators for code, try `ember help generate` for more details 70 | 71 | ### Running Tests 72 | 73 | * `ember test` 74 | * `ember test --server` 75 | 76 | ### Linting 77 | 78 | * `npm run lint:js` 79 | * `npm run lint:js -- --fix` 80 | 81 | ### Building 82 | 83 | * `ember build` (development) 84 | * `ember build --environment production` (production) 85 | 86 | ### Deploying 87 | 88 | Specify what it takes to deploy your app. 89 | 90 | ## Further Reading / Useful Links 91 | 92 | * [ember.js](https://emberjs.com/) 93 | * [ember-cli](https://ember-cli.com/) 94 | * Development Browser Extensions 95 | * [ember inspector for chrome](https://chrome.google.com/webstore/detail/ember-inspector/bmdblncegkenkacieihfhpjfppoconhi) 96 | * [ember inspector for firefox](https://addons.mozilla.org/en-US/firefox/addon/ember-inspector/) 97 | -------------------------------------------------------------------------------- /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/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/components/.gitkeep -------------------------------------------------------------------------------- /app/components/muuri-grid-component.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import Muuri from 'muuri'; 3 | import { get } from '@ember/object'; 4 | 5 | export default Component.extend({ 6 | classNames: ['muuri-grid-component'], 7 | didInsertElement() { 8 | this._super(...arguments); 9 | this.grid = new Muuri(this.element, { 10 | dragEnabled: true, 11 | dragSortInterval: 100, 12 | dragReleaseDuration: 300, 13 | dragReleaseEasing: 'ease', 14 | dragStartPredicate: { 15 | handle: '.cf-card-icon' 16 | } 17 | }); 18 | this.grid.on('layoutEnd', (items) => { 19 | const newOrder = items.map(item => item.getElement().dataset.id); 20 | return get(this, 'updateSort')(newOrder); 21 | }); 22 | }, 23 | didUpdate() { 24 | const grid = get(this, 'grid'); 25 | const prevItems = get(this, 'grid._items'); 26 | const prevItemIds = prevItems.map(item => item._element.getAttribute('data-id')); 27 | 28 | const currentItemIds = []; 29 | const currentGridElements = document.getElementsByClassName('muuri-item-component'); 30 | 31 | for (var i = 0; i < currentGridElements.length; i++) { 32 | const itemEl = currentGridElements[i]; 33 | const uuid = itemEl.getAttribute('data-id'); 34 | 35 | if (!!prevItems[i] && itemEl.clientHeight !== prevItems[i]._height) { 36 | grid.refreshItems().layout() 37 | } 38 | 39 | if (!prevItemIds.includes(uuid)) { 40 | grid.add(itemEl); 41 | } 42 | 43 | currentItemIds.push(uuid); 44 | } 45 | 46 | prevItems.forEach(item => { 47 | const itemEl = item._element 48 | if (!currentItemIds.includes(itemEl.getAttribute('data-id'))) { 49 | grid.remove(itemEl); 50 | } 51 | }); 52 | 53 | grid.refreshItems(); 54 | }, 55 | willRemoveElement() { 56 | delete this.grid; 57 | } 58 | }); -------------------------------------------------------------------------------- /app/components/muuri-item-component.js: -------------------------------------------------------------------------------- 1 | import Component from '@ember/component'; 2 | import { computed } from '@ember/object'; 3 | 4 | export default Component.extend({ 5 | classNames: ['muuri-item-component', 'mb1'], 6 | classNameBindings: ['isEditing:muuri-item-component--editing'], 7 | attributeBindings: ['uuid:data-id'], 8 | modelId: computed('uuid', function () { 9 | return this.get('uuid'); 10 | }) 11 | }); -------------------------------------------------------------------------------- /app/controllers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/controllers/.gitkeep -------------------------------------------------------------------------------- /app/controllers/index.js: -------------------------------------------------------------------------------- 1 | import Controller from '@ember/controller'; 2 | import { get, set } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | 5 | export default Controller.extend({ 6 | extension: service(), 7 | fileQueue: service(), 8 | 9 | editingUUID: "", 10 | showPreview: false, 11 | 12 | actions: { 13 | addFragment() { 14 | const newFragment = get(this, 'extension').addFragment(); 15 | if (!newFragment) return; 16 | const uuid = get(newFragment.findBy('key', 'uuid'), 'value'); 17 | // TODO: re-enable this after muuri rerender on "iEditing" 18 | // bug is fixed 19 | // set(this, 'editingUUID', uuid); 20 | return uuid; 21 | }, 22 | 23 | editFragment(uuid) { 24 | set(this, 'editingUUID', uuid); 25 | }, 26 | 27 | cancelEditing() { 28 | set(this, 'editingUUID', ""); 29 | }, 30 | 31 | removeFragment(fragment) { 32 | get(this, 'extension').removeFragment(fragment); 33 | set(this, 'editingUUID', ""); 34 | }, 35 | 36 | // Savers 37 | save() { 38 | get(this, 'extension').persist(); 39 | }, 40 | 41 | saveDate(fragmentField, date) { 42 | if (date) { 43 | set(fragmentField, 'value', date.toISOString()); 44 | get(this, 'extension').persist(); 45 | } 46 | }, 47 | 48 | saveBlob(fragmentField, file) { 49 | if (!file) return; 50 | 51 | const { name, size, type } = file; 52 | return file.readAsDataURL().then(data => { 53 | set(fragmentField, 'value', { 54 | data, name, size, type 55 | }); 56 | return get(this, 'extension').persist(); 57 | }).finally(() => { 58 | const queue = get(this, 'fileQueue.queues').find(queue => get(queue, 'files').includes(file)); 59 | if (queue) queue.remove(file); 60 | }); 61 | }, 62 | 63 | showPreview() { 64 | set(this, 'showPreview', true); 65 | }, 66 | 67 | hidePreview() { 68 | set(this, 'showPreview', false); 69 | }, 70 | 71 | updateSort(order) { 72 | get(this, 'extension').updateSort(order); 73 | } 74 | } 75 | }); 76 | -------------------------------------------------------------------------------- /app/controllers/schema.js: -------------------------------------------------------------------------------- 1 | import Controller from '@ember/controller'; 2 | import { get, set } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | 5 | export default Controller.extend({ 6 | extension: service(), 7 | 8 | deletingUUID: "", 9 | 10 | actions: { 11 | stageFieldDeletion(field) { 12 | set(this, 'deletingUUID', field.uuid); 13 | }, 14 | unstageFieldDeletion() { 15 | set(this, 'deletingUUID', ""); 16 | }, 17 | addEmptySchemaField() { 18 | get(this, 'extension').addSchemaField(); 19 | }, 20 | keyDidChange() { 21 | get(this, 'extension').validateSchema(); 22 | }, 23 | setFieldType(value) { 24 | const [uuid, type] = value.split('-'); 25 | const field = get(this, 'extension.data._schema').findBy('uuid', uuid); 26 | set(field, 'type', type); 27 | get(this, 'extension').validateSchema(); 28 | }, 29 | cancel() { 30 | this.transitionToRoute('index'); 31 | }, 32 | save() { 33 | get(this, 'extension').persist().then(() => { 34 | get(this, 'extension').syncFragmentsToSchema(); 35 | this.transitionToRoute('index'); 36 | }); 37 | }, 38 | removeFragmentField(field) { 39 | get(this, 'extension').removeSchemaField(field); 40 | } 41 | } 42 | 43 | }); 44 | -------------------------------------------------------------------------------- /app/helpers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/helpers/.gitkeep -------------------------------------------------------------------------------- /app/helpers/is-image-type.js: -------------------------------------------------------------------------------- 1 | import { helper } from '@ember/component/helper'; 2 | 3 | export function imageType(params/*, hash*/) { 4 | const type = params[0] || ""; 5 | return type.startsWith('image/'); 6 | } 7 | 8 | export default helper(imageType); 9 | -------------------------------------------------------------------------------- /app/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | ContentfulFragment 7 | 8 | 9 | 10 | {{content-for "head"}} 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | {{content-for "head-footer"}} 20 | 21 | 22 | {{content-for "body"}} 23 | 24 | 48 | 49 | 50 | 51 | 52 | {{content-for "body-footer"}} 53 | 54 | 55 | -------------------------------------------------------------------------------- /app/lib/generateUUID.js: -------------------------------------------------------------------------------- 1 | export default function generateUUID() { 2 | let d = new Date().getTime(); 3 | if (window.performance && typeof window.performance.now === 'function') { 4 | d += performance.now(); // use high-precision timer if available 5 | } 6 | const uuid = 'xxxxxx'.replace(/[xy]/g, (c) => { 7 | const r = (d + Math.random() * 16) % 16 | 0; 8 | d = Math.floor(d / 16); 9 | return (c === 'x' ? r : (r&0x3|0x8)).toString(16); 10 | }); 11 | return uuid; 12 | } 13 | -------------------------------------------------------------------------------- /app/lib/sanctu.js: -------------------------------------------------------------------------------- 1 | export default ""; 2 | -------------------------------------------------------------------------------- /app/models/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/models/.gitkeep -------------------------------------------------------------------------------- /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 | import config from './config/environment'; 3 | 4 | const Router = EmberRouter.extend({ 5 | location: config.locationType, 6 | rootURL: config.rootRouterURL 7 | }); 8 | 9 | Router.map(function() { 10 | this.route('schema'); 11 | }); 12 | 13 | export default Router; 14 | -------------------------------------------------------------------------------- /app/routes/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/routes/.gitkeep -------------------------------------------------------------------------------- /app/routes/application.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | import { get } from '@ember/object'; 3 | import { inject as service } from '@ember/service'; 4 | import sanctuLogo from 'contentful-fragment/lib/sanctu'; 5 | 6 | const DUMMY_DATA = { 7 | "_schema": [ 8 | { 9 | "key": "Event Location", 10 | "type": "Symbol", 11 | "uuid": "6a92b9" 12 | }, 13 | { 14 | "key": "Event Url", 15 | "type": "Symbol", 16 | "uuid": "6a27d9" 17 | }, 18 | { 19 | "key": "Event Date", 20 | "type": "Date", 21 | "uuid": "a592h9" 22 | }, 23 | { 24 | "key": "Logo", 25 | "type": "Blob", 26 | "uuid": "a592f3" 27 | } 28 | ], 29 | "fragments": [ 30 | [ 31 | { 32 | "key": "uuid", 33 | "value": "c36d6d" 34 | }, 35 | { 36 | "key": "Event Location", 37 | "value": "Sanctuary Computer Inc", 38 | "type": "Symbol", 39 | "_schemaRef": "6a92b9" 40 | }, 41 | { 42 | "key": "Event Url", 43 | "value": "https://www.google.com/maps/place/Sanctuary+Computer/@40.71811,-73.997507,17z/data=!3m1!4b1!4m5!3m4!1s0x89c259880e5637e3:0xcdc06390643521f5!8m2!3d40.71811!4d-73.995313", 44 | "type": "Symbol", 45 | "_schemaRef": "6a27d9" 46 | }, 47 | { 48 | "key": "Event Date", 49 | "value": new Date().setDate(new Date().getDate() + 1), 50 | "type": "Date", 51 | "_schemaRef": "a592h9" 52 | }, 53 | { 54 | "key": "Logo", 55 | "value": { 56 | "data": sanctuLogo, 57 | "name": "default logo.jpg", 58 | "size": 1859, 59 | "type": "image/png" 60 | }, 61 | "type": "Blob", 62 | "_schemaRef": "a592f3" 63 | } 64 | ] 65 | ] 66 | }; 67 | 68 | const DUMMY_SCHEMA_SHORTHAND = null; //"Event Location:Symbol,Event Url: Symbol,Event Date:Date,Logo:Blob" 69 | 70 | const DummyExtension = { 71 | parameters: { 72 | instance: { 73 | schemaShorthand: DUMMY_SCHEMA_SHORTHAND 74 | } 75 | }, 76 | field: { 77 | _value: DUMMY_DATA, 78 | getValue: function() { 79 | return { ...DummyExtension.field._value }; 80 | }, 81 | setValue: function(newValue) { 82 | DummyExtension.field._value = newValue; 83 | window.newValue = newValue; 84 | return new Promise(resolve => resolve(newValue)); 85 | } 86 | } 87 | }; 88 | 89 | export default Route.extend({ 90 | extension: service(), 91 | 92 | model() { 93 | const isDummy = ['localhost', 'contentful-fragment.io'].includes(window.location.hostname) || window.location.pathname === "/dummy"; 94 | if (isDummy) { 95 | document.body.style.overflow = "auto"; 96 | document.body.style.padding = "10px 10px 0px 10px"; 97 | } 98 | if (isDummy) return DummyExtension; 99 | 100 | return new Promise(window.contentfulFragment.getExtension, DummyExtension); 101 | }, 102 | 103 | afterModel(contentfulExtension) { 104 | get(this, 'extension').setup(contentfulExtension); 105 | } 106 | }); 107 | -------------------------------------------------------------------------------- /app/routes/index.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | 3 | export default Route.extend({ 4 | }); 5 | -------------------------------------------------------------------------------- /app/routes/schema.js: -------------------------------------------------------------------------------- 1 | import Route from '@ember/routing/route'; 2 | 3 | export default Route.extend({ 4 | }); 5 | -------------------------------------------------------------------------------- /app/services/extension.js: -------------------------------------------------------------------------------- 1 | import Service from '@ember/service'; 2 | import { get, set } from '@ember/object'; 3 | 4 | const TYPES = [ 5 | 'Symbol', 6 | 'Date', 7 | 'Blob' 8 | ]; 9 | 10 | const emptyForType = (/*type*/) => { 11 | // TODO 12 | return null; 13 | }; 14 | 15 | const coerce = (type, value) => { 16 | if (!value) return emptyForType(type); 17 | return value; 18 | }; 19 | 20 | const generateUUID = () => { 21 | let d = new Date().getTime(); 22 | if (window.performance && typeof window.performance.now === 'function') { 23 | d += performance.now(); // use high-precision timer if available 24 | } 25 | const uuid = 'xxxxxx'.replace(/[xy]/g, (c) => { 26 | const r = (d + Math.random() * 16) % 16 | 0; 27 | d = Math.floor(d / 16); 28 | return (c === 'x' ? r : (r&0x3|0x8)).toString(16); 29 | }); 30 | return uuid; 31 | }; 32 | 33 | const validateSchemaField = field => { 34 | let validation = ''; 35 | if (!field.key || field.key.length === 0) validation += 'Enter a key for this field'; 36 | if (!field.type) { 37 | if (validation.length) validation += ', '; 38 | validation += 'Select a field type'; 39 | } 40 | return validation; 41 | }; 42 | 43 | const newFragmentFromSchema = schema => { 44 | return schema.reduce((acc, field) => { 45 | return [ 46 | ...acc, 47 | { 48 | key: field.key, 49 | value: emptyForType(field.type), 50 | type: field.type, 51 | _schemaRef: field.uuid 52 | } 53 | ]; 54 | }, [{ 55 | key: 'uuid', 56 | value: generateUUID() 57 | }]); 58 | }; 59 | 60 | export default Service.extend({ 61 | data: null, 62 | extension: null, 63 | preview: null, 64 | 65 | setup(extension) { 66 | set(this, 'data', extension.field.getValue() || {}); 67 | set(this, 'extension', extension); 68 | 69 | this.loadSchemaFromShorthand(); 70 | this.syncFragmentsToSchema(); 71 | this.makeSimpleFragments(); 72 | }, 73 | 74 | setSetting(key, value) { 75 | set(this, 'data._settings', get(this, 'data._settings') || {}); 76 | set(this, `data._settings.${key}`, value); 77 | }, 78 | 79 | loadSchemaFromShorthand() { 80 | const shorthand = (get(this, 'extension.parameters.instance.schemaShorthand') || ""); 81 | if (!shorthand.length) return; 82 | 83 | const parsedSchemaFields = 84 | shorthand.split(",").map(tuple => tuple.split(":").map(t => t.trim())); 85 | const existingSchema = get(this, 'data._schema') || []; 86 | 87 | const loadedSchema = parsedSchemaFields.reduce((acc, fieldTuple) => { 88 | const [key, type] = fieldTuple; 89 | if (key.length === 0) { 90 | // eslint-disable-next-line 91 | console.warn( 92 | `Contentful Fragment: Your Predefined Schema included a blank schema key! Please refer to the documentation.` 93 | ); 94 | return acc; 95 | } 96 | if (!TYPES.includes(type)) { 97 | // eslint-disable-next-line 98 | console.warn( 99 | `Contentful Fragment: Your Predefined Schema included unknown type: ${type}. Must be one of <${TYPES.join(' ')}>` 100 | ); 101 | return acc; 102 | } 103 | const match = existingSchema.findBy('key', fieldTuple[0]); 104 | return [...acc, { 105 | uuid: (match ? match.uuid : generateUUID()), key, type 106 | }]; 107 | }, []).filter(schemaField => { 108 | return parsedSchemaFields.find(schemaTuple => schemaTuple[0] === schemaField.key); 109 | }); 110 | 111 | this.setSetting('usesPredefinedSchema', true); 112 | set(this, 'data._schema', loadedSchema); 113 | this.validateSchema(); 114 | }, 115 | 116 | /* Main Editor */ 117 | addFragment() { 118 | if (get(this, 'data._schema')) { 119 | set(this, 'data.fragments', get(this, 'data.fragments') || []); 120 | const newFragment = newFragmentFromSchema(get(this, 'data._schema')); 121 | get(this, 'data.fragments').pushObject(newFragment); 122 | this.persist(); 123 | return newFragment; 124 | } 125 | }, 126 | 127 | removeFragment(fragment) { 128 | if (get(this, 'data.fragments')) { 129 | const uuid = get(fragment.findBy('key', 'uuid'), 'value'); 130 | const newFragments = this.data.fragments.reject(fragment => { 131 | return get(fragment.findBy('key', 'uuid'), 'value') === uuid; 132 | }); 133 | set(this, 'data.fragments', newFragments); 134 | this.persist(); 135 | } 136 | }, 137 | 138 | /* Schema Editor */ 139 | validateSchema() { 140 | const allValid = get(this, 'data._schema').map(schemaField => { 141 | const validation = validateSchemaField(schemaField); 142 | set(schemaField, 'validation', validation); 143 | 144 | return (validation.length === 0) 145 | }).every(Boolean); 146 | 147 | set(this, 'data.valid', allValid); 148 | }, 149 | 150 | syncFragmentsToSchema() { 151 | const { fragments, _schema = [] } = get(this, 'data'); 152 | const syncedFragments = (fragments || []).map(fragment => { 153 | const syncedFragment = _schema.map(schemaField => { 154 | const dataForSchemaField = fragment.findBy('_schemaRef', schemaField.uuid); 155 | return { 156 | key: schemaField.key, 157 | type: schemaField.type, 158 | value: coerce(schemaField.type, dataForSchemaField ? dataForSchemaField.value : null), 159 | _schemaRef: schemaField.uuid, 160 | }; 161 | }); 162 | return [fragment.findBy('key', 'uuid'), ...syncedFragment]; 163 | }); 164 | 165 | set(this, 'data.fragments', syncedFragments); 166 | this.persist(); 167 | }, 168 | 169 | addSchemaField() { 170 | const newField = { key: '', type: null, uuid: generateUUID(), validation: '' }; 171 | set(newField, 'validation', validateSchemaField(newField)); 172 | if (get(this, 'data._schema')) { 173 | get(this, 'data._schema').pushObject(newField); 174 | } else { 175 | set(this, 'data._schema', [newField]); 176 | } 177 | 178 | return newField; 179 | }, 180 | 181 | removeSchemaField(field) { 182 | const newFields = get(this, 'data._schema').reject(existingField => { 183 | return existingField.uuid === field.uuid; 184 | }); 185 | set(this, 'data._schema', newFields); 186 | }, 187 | 188 | makeSimpleFragments() { 189 | const simpleFragments = (get(this, 'data.fragments') || []).reduce((simpleFragments, fragment, index) => { 190 | const uuid = get(fragment.findBy('key', 'uuid'), 'value'); 191 | 192 | simpleFragments[uuid] = fragment.reduce((simpleFragment, fragmentField) => { 193 | if (fragmentField.key === "uuid") return simpleFragment; 194 | if (fragmentField.key) { 195 | simpleFragment[fragmentField.key.camelize()] = fragmentField.value; 196 | } 197 | 198 | return { 199 | index, 200 | uuid, 201 | ...simpleFragment 202 | }; 203 | }, {}); 204 | 205 | return simpleFragments; 206 | }, {}); 207 | 208 | set(this, 'data.simpleFragments', simpleFragments); 209 | }, 210 | 211 | generateJSONPreview() { 212 | try { 213 | const jsonPreview = JSON.stringify(this.data, null, 2); 214 | set(this, 'preview', jsonPreview); 215 | } catch (e) { 216 | set(this, 'preview', 'Could not generate preview'); 217 | } 218 | }, 219 | 220 | updateSort(order) { 221 | const fragments = get(this, 'data.fragments'); 222 | const resorted = order.map(uuid => fragments.find(fragment => { 223 | const fragmentUuid = get(fragment.findBy('key', 'uuid'), 'value'); 224 | return uuid === fragmentUuid 225 | })); 226 | 227 | set(this, 'data.fragments', resorted); 228 | this.persist(); 229 | }, 230 | 231 | persist() { 232 | this.makeSimpleFragments(); 233 | this.generateJSONPreview(); 234 | return this.extension.field.setValue(get(this, 'data')); 235 | }, 236 | }); 237 | -------------------------------------------------------------------------------- /app/styles/app.css: -------------------------------------------------------------------------------- 1 | body { 2 | overflow: hidden; 3 | margin: 0; 4 | } 5 | 6 | .green { 7 | color: #14d997; 8 | } 9 | 10 | .red { 11 | color: #fe5c60; 12 | } 13 | 14 | .transition { 15 | transition: all .1s ease-in-out; 16 | } 17 | 18 | .pointer:hover { 19 | cursor: pointer; 20 | } 21 | 22 | .events-none { 23 | pointer-events: none; 24 | } 25 | 26 | .events-all { 27 | pointer-events: all; 28 | } 29 | 30 | .icon-wrapper { 31 | display: inline-block; 32 | } 33 | 34 | a:hover .icon-small { 35 | fill: #2a3039; 36 | } 37 | 38 | .mb0 { 39 | margin-bottom: 0 !important; 40 | } 41 | 42 | .mb1 { 43 | margin-bottom: 0.642857142857143em; 44 | } 45 | 46 | .mb2 { 47 | margin-bottom: 1.928571428571429em; 48 | } 49 | 50 | .icon-small { 51 | display: inline-block; 52 | fill: #3c80cf; 53 | vertical-align: middle; 54 | transition: fill .1s ease-in-out; 55 | position: relative; 56 | bottom: 1px; 57 | } 58 | 59 | .icon-small.red { 60 | fill: #fe5c60; 61 | } 62 | 63 | .plus-icon { 64 | width: 1.125rem; 65 | height: 1.125rem; 66 | } 67 | 68 | .cross-icon { 69 | width: 0.75rem; 70 | height: 0.75rem; 71 | } 72 | 73 | .check-icon { 74 | width: 0.875rem; 75 | height: 0.875rem; 76 | } 77 | 78 | .pencil-icon { 79 | width: 1.0625rem; 80 | height: 1rem; 81 | } 82 | 83 | .red { 84 | color: #fe5c60; 85 | } 86 | 87 | hr { 88 | border: 0; 89 | height: 0; 90 | border-bottom: 1px solid #d3dce0; 91 | } 92 | 93 | .cf-card { 94 | border: 1px solid #d3dce0; 95 | -webkit-border-radius: 2px; 96 | border-radius: 2px; 97 | display: -webkit-box; 98 | display: -moz-box; 99 | display: -webkit-flex; 100 | display: -ms-flexbox; 101 | display: box; 102 | display: flex; 103 | -webkit-box-shadow: 0 1px 3px rgba(0,0,0,0.08); 104 | box-shadow: 0 1px 3px rgba(0,0,0,0.08); 105 | background: #fff; 106 | -webkit-transition: all 200ms ease-in-out; 107 | -moz-transition: all 200ms ease-in-out; 108 | -o-transition: all 200ms ease-in-out; 109 | -ms-transition: all 200ms ease-in-out; 110 | transition: all 200ms ease-in-out; 111 | -webkit-transform: translate3d(0, 0, 0); 112 | -moz-transform: translate3d(0, 0, 0); 113 | -o-transform: translate3d(0, 0, 0); 114 | -ms-transform: translate3d(0, 0, 0); 115 | transform: translate3d(0, 0, 0); 116 | } 117 | 118 | .cf-card-inner { 119 | -webkit-box-flex: 1; 120 | -moz-box-flex: 1; 121 | -o-box-flex: 1; 122 | box-flex: 1; 123 | -webkit-flex: 1 0 0; 124 | -ms-flex: 1 0 0; 125 | flex: 1 0 0; 126 | padding: 14px; 127 | max-width: calc(100% - 19px); 128 | } 129 | 130 | .cf-card-status { 131 | font-size: 12px; 132 | text-transform: uppercase; 133 | font-weight: 600; 134 | letter-spacing: 0.1em; 135 | } 136 | 137 | .cf-card-field { 138 | text-overflow: ellipsis; 139 | max-width: 20rem; 140 | margin-bottom: .5rem; 141 | } 142 | 143 | .cf-card-field--title { 144 | margin-bottom: 0rem; 145 | color: #8091a5; 146 | font-size: 0.875rem; 147 | max-width: 50rem; 148 | } 149 | 150 | .cf-card-field--content { 151 | margin-top: 0rem; 152 | font-size: 1rem; 153 | color: black; 154 | text-overflow: ellipsis; 155 | white-space: nowrap; 156 | overflow: hidden; 157 | } 158 | 159 | .pika-single { 160 | font-family: "Avenir Next W01", Helvetica, sans-serif; 161 | text-rendering: optimizeLegibility; 162 | font-size: 14px; 163 | line-height: 18px; 164 | color: #2d2f31; 165 | } 166 | 167 | .blob-preview img { 168 | max-width: 100px; 169 | } 170 | 171 | .blob-preview .blob-meta { 172 | display: inline-block; 173 | } 174 | 175 | .blob-preview .blob-no-preview { 176 | height: 100%; 177 | background-color: #d3dce0; 178 | } 179 | 180 | .json-preview { 181 | height: 200px; 182 | border: 1px solid #c3d0d5; 183 | -webkit-border-radius: 2px; 184 | border-radius: 2px; 185 | background-color: #f7f9fa; 186 | overflow: scroll; 187 | width: 100%; 188 | font-family: monospace; 189 | font-size: .75rem; 190 | line-height: 1rem; 191 | } 192 | 193 | .cf-card { 194 | display: flex; 195 | transition: all 200ms ease-in-out; 196 | } 197 | .cf-card-icon { 198 | display: flex; 199 | align-items: center; 200 | justify-content: center; 201 | width: 1.25rem; 202 | border-right: 1px solid #d3dce0; 203 | background-color: #f7f9fa; 204 | cursor: grab; 205 | transition: background-color 200ms ease-in-out; 206 | } 207 | .cf-card-icon:hover { 208 | background-color: #e5ebed; 209 | } 210 | 211 | .cf-form-field { 212 | margin-bottom: 1rem !important; 213 | } 214 | 215 | .cf-form-field:last-of-type { 216 | margin-bottom: 2rem !important; 217 | } 218 | 219 | .muuri-grid-component { 220 | position: relative; 221 | display: flex; 222 | flex-wrap: wrap; 223 | } 224 | .muuri-item-component { 225 | display: block; 226 | position: absolute; 227 | width: 100%; 228 | z-index: 1; 229 | } 230 | .muuri-item-component--editing { 231 | z-index: 4; 232 | } 233 | .item-content > .cf-card:hover { 234 | border: solid #3c80cf 1px; 235 | } 236 | .muuri-item-component--editing .item-content > .cf-card { 237 | box-shadow: 0px 6px 12px -3px rgba(0,0,0,0.3); 238 | } 239 | .muuri-item-dragging, .muuri-item-releasing { 240 | z-index: 3; 241 | } 242 | .item-content { 243 | position: relative; 244 | } 245 | 246 | .border { 247 | border: solid red; 248 | } 249 | -------------------------------------------------------------------------------- /app/templates/application.hbs: -------------------------------------------------------------------------------- 1 | {{outlet}} 2 | 3 |
4 |

5 | Contentful Fragment was built by Sanctuary Computer in NYC. 6 |

7 |
8 | -------------------------------------------------------------------------------- /app/templates/components/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sanctuarycomputer/contentful-fragment/c61fa2a8e15cbec06acc7d5ecb8acafd09af9c8e/app/templates/components/.gitkeep -------------------------------------------------------------------------------- /app/templates/components/check-icon.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 8 | 9 | -------------------------------------------------------------------------------- /app/templates/components/cross-icon.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /app/templates/components/muuri-grid-component.hbs: -------------------------------------------------------------------------------- 1 | {{yield}} -------------------------------------------------------------------------------- /app/templates/components/muuri-item-component.hbs: -------------------------------------------------------------------------------- 1 | 2 |
3 |
4 |
5 | 6 | 7 | 8 | 9 |
10 |
11 |
12 | {{yield}} 13 |
14 |
15 |
16 |
-------------------------------------------------------------------------------- /app/templates/components/pencil-icon.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /app/templates/components/plus-icon.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /app/templates/index.hbs: -------------------------------------------------------------------------------- 1 | {{#if extension.data._schema}} 2 | {{#muuri-grid-component 3 | updateSort=(action 'updateSort') 4 | editingUUID=editingUUID 5 | }} 6 | {{#each extension.data.fragments as |fragment index|}} 7 | {{#with (get (find-by 'key' 'uuid' fragment) 'value') as |uuid|}} 8 | {{#muuri-item-component uuid=uuid isEditing=(eq editingUUID uuid)}} 9 | {{!-- card content --}} 10 |
11 | {{#each fragment as |fragmentField|}} 12 | {{#if (not-eq fragmentField.key 'uuid')}} 13 |
14 | {{#if (eq editingUUID uuid)}} 15 | 64 | {{else}} 65 | {{#if (eq fragmentField.type 'Symbol')}} 66 |

67 | {{fragmentField.key}} 68 |

69 |

70 | {{fragmentField.value}} 71 |

72 | {{/if}} 73 | 74 | {{#if (eq fragmentField.type 'Date')}} 75 |

76 | {{fragmentField.key}} 77 |

78 |

79 | {{if fragmentField.value (moment-format (utc fragmentField.value) 'dddd, Do MMMM YYYY')}} 80 |

81 | {{/if}} 82 | 83 | {{#if (eq fragmentField.type 'Blob')}} 84 |

85 | {{fragmentField.key}} 86 |

87 | {{#if fragmentField.value}} 88 |
89 |
90 | {{#if (is-image-type fragmentField.value.type)}} 91 | 92 | {{else}} 93 |
94 |

No Preview Available

95 |
96 | {{/if}} 97 |
98 |

File Name: {{fragmentField.value.name}}

99 |

File Size: {{fragmentField.value.size}}

100 |

File Type: {{fragmentField.value.type}}

101 |
102 |
103 |
104 | {{else}} 105 |
106 | {{/if}} 107 | {{/if}} 108 | {{/if}} 109 |
110 | {{/if}} 111 | {{/each}} 112 | 113 |
114 | {{#cross-icon}} 115 | {{/cross-icon}} 116 |
117 | Delete Entry 118 |
119 |
120 | 121 | {{!-- edit or cancel button --}} 122 |
123 | {{#if (eq editingUUID uuid)}} 124 |
125 | {{#check-icon}} 126 | {{/check-icon}} 127 |
128 | {{else}} 129 |
130 | {{#pencil-icon}} 131 | {{/pencil-icon}} 132 |
133 | {{/if}} 134 |
135 | {{/muuri-item-component}} 136 | {{/with}} 137 | {{else}} 138 |

139 | Hey! You haven't added any entries yet. 140 | Add one now! 141 |

142 | {{/each}} 143 | {{/muuri-grid-component}} 144 | {{!-- Add / Edit buttons --}} 145 |
146 | 147 |
148 | {{#plus-icon}} 149 | {{/plus-icon}} 150 |
151 | Add Entry 152 |
153 | {{#unless extension.data._settings.usesPredefinedSchema}} 154 | {{#link-to 'schema' class='ml2 transition'}} 155 |
156 | {{#plus-icon}} 157 | {{/plus-icon}} 158 |
159 | Edit Fragment 160 | {{/link-to}} 161 | {{/unless}} 162 | 163 | {{!-- JSON Preview --}} 164 | 183 |
184 | {{#if showPreview}} 185 |