From f76c4de7f369efd109a1aba398e01deca089ff34 Mon Sep 17 00:00:00 2001 From: karantalapalli Date: Thu, 10 Mar 2022 19:01:47 +0530 Subject: [PATCH 1/9] feat: add content type visualizer extension [ECO-656] --- README.md | 3 + content-type-visualizer/.eslintrc.json | 18 + content-type-visualizer/.gitignore | 4 + content-type-visualizer/LICENSE | 21 + content-type-visualizer/README.md | 30 ++ content-type-visualizer/gulpfile.js | 50 ++ content-type-visualizer/index.html | 160 +++++++ content-type-visualizer/package.json | 27 ++ content-type-visualizer/source/index.html | 161 +++++++ content-type-visualizer/source/index.js | 510 ++++++++++++++++++++ content-type-visualizer/source/style.css | 559 ++++++++++++++++++++++ 11 files changed, 1543 insertions(+) create mode 100644 content-type-visualizer/.eslintrc.json create mode 100644 content-type-visualizer/.gitignore create mode 100644 content-type-visualizer/LICENSE create mode 100644 content-type-visualizer/README.md create mode 100644 content-type-visualizer/gulpfile.js create mode 100644 content-type-visualizer/index.html create mode 100644 content-type-visualizer/package.json create mode 100644 content-type-visualizer/source/index.html create mode 100644 content-type-visualizer/source/index.js create mode 100644 content-type-visualizer/source/style.css diff --git a/README.md b/README.md index 000fa639..29760be0 100644 --- a/README.md +++ b/README.md @@ -50,6 +50,9 @@ This extension lets you fetch and display Youtube videos into a field of your co [External API Lookup](./external-api-lookup-template): This extension lets you fetch data from an external API and display the data as possible values for the field on an entry page in Contentstack. +[Conten Type Visualizer](./content-type-visualizer) +Content Type Visualizer Dashboard Widget offers a graphical representation of all content types, along with their fields, in a particular stack. + ### Examples of custom widgets created using Extensions Here are some examples/use cases of custom widgets that can be created using Extensions. These examples come with readme files that explain how to install and get started with these widgets. diff --git a/content-type-visualizer/.eslintrc.json b/content-type-visualizer/.eslintrc.json new file mode 100644 index 00000000..5a437149 --- /dev/null +++ b/content-type-visualizer/.eslintrc.json @@ -0,0 +1,18 @@ +{ + "env": { + "browser": true, + "es6": true, + "jquery": true + }, + "extends": "airbnb-base/legacy", + "parserOptions": { + "ecmaVersion": 2016, + "sourceType": "module" + }, + "globals": { + "ContentstackUIExtension": false + }, + "rules": { + "no-restricted-syntax": ["error", "LabeledStatement", "WithStatement"] + } +} \ No newline at end of file diff --git a/content-type-visualizer/.gitignore b/content-type-visualizer/.gitignore new file mode 100644 index 00000000..92a6367f --- /dev/null +++ b/content-type-visualizer/.gitignore @@ -0,0 +1,4 @@ +*.log +.cache +package-lock.json +node_modules diff --git a/content-type-visualizer/LICENSE b/content-type-visualizer/LICENSE new file mode 100644 index 00000000..0302523a --- /dev/null +++ b/content-type-visualizer/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2020 Contentstack + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. \ No newline at end of file diff --git a/content-type-visualizer/README.md b/content-type-visualizer/README.md new file mode 100644 index 00000000..04db54d5 --- /dev/null +++ b/content-type-visualizer/README.md @@ -0,0 +1,30 @@ +# Content Type Visualizer – Contentstack Extension + +#### About this extension. +Content Type Visualizer Dashboard Widget offers a graphical representation of all content types, along with their fields, in a particular stack. + +![Content Type Visualizer Screenshot](https://images.contentstack.io/v3/assets/blt83726f918894d893/bltae909a9786341bbf/5f394a3ddb5c28785b6f048d/content-type-visualizer.png) + + +#### How to use this extension +We have created a step-by-step guide on how to create a Content Type Visualizer extension for your content types. You can refer the [Content Type Visualizer extension guide]() on our documentation site for more info. + + +#### Other Documentation +- [Extensions guide](https://www.contentstack.com/docs/guide/extensions) +- [Common questions about extensions](https://www.contentstack.com/docs/faqs#extensions) + + +#### Modifying Extension + +To modify the extension, first clone this repo and install the dependencies. Then, edit the HTML, CSS and JS files from the source folder, and create a build using gulp task. + +To install dependencies, run the following command in the root folder +``` +npm install gulp-cli -g +npm install +``` +To create new build for the extension, run the following command (index.html): + + gulp build + diff --git a/content-type-visualizer/gulpfile.js b/content-type-visualizer/gulpfile.js new file mode 100644 index 00000000..b274fd76 --- /dev/null +++ b/content-type-visualizer/gulpfile.js @@ -0,0 +1,50 @@ +const gulp = require('gulp'); +const inline = require('gulp-inline'); +const uglify = require('gulp-uglify'); +const gulpStylelint = require('gulp-stylelint'); +const minifyCss = require('gulp-clean-css'); +const babel = require('gulp-babel'); +const eslint = require('gulp-eslint'); + + +gulp.task('lint-js', () => { + return gulp.src('source/*.js') + .pipe(eslint()) + .pipe(eslint.format()) + .pipe(eslint.failAfterError()); +}); + +gulp.task('lint-css', () => { + return gulp + .src('source/*.css') + .pipe(gulpStylelint({ + config: { + extends: 'stylelint-config-standard' + }, + reporters: [{ + formatter: 'string', + console: true + }], + failAfterError: false + })); +}); + +gulp.task('inline', () => { + return gulp.src('./source/index.html') + .pipe(inline({ + js: [babel({ + presets: ["@babel/preset-env"] + }), uglify], + css: [minifyCss], + disabledTypes: ['svg', 'img'] + })) + .pipe(gulp.dest('./')); +}); + +gulp.task('build', gulp.series('lint-js', 'lint-css', 'inline')); + +gulp.task('watch', () => { + gulp.watch('source/*', gulp.series('build')); +}); + +gulp.task('default', gulp.series('build')); diff --git a/content-type-visualizer/index.html b/content-type-visualizer/index.html new file mode 100644 index 00000000..0480dbbc --- /dev/null +++ b/content-type-visualizer/index.html @@ -0,0 +1,160 @@ + + + + + + External API UI Extension Sample + + + + + + + + + + + + + + + + + +
+
+
+
+
+
+
+ _ Content Types +
+
+ +
+
+
+ +
+ +
+
+ + + + +
+
+
+
+
+
+
+
+
+ + + + \ No newline at end of file diff --git a/content-type-visualizer/package.json b/content-type-visualizer/package.json new file mode 100644 index 00000000..db2bb5c6 --- /dev/null +++ b/content-type-visualizer/package.json @@ -0,0 +1,27 @@ +{ + "name": "@contentstack/extension-custom-dashboard-content-type-visualizer", + "version": "1.0.0", + "description": "A diagrammatic representation of all content types in a stack.", + "main": "gulpfile.js", + "scripts": { + "build": "npx gulp build" + }, + "author": "Contentstack", + "license": "MIT", + "devDependencies": { + "@babel/cli": "7.5.0", + "@babel/core": "7.5.4", + "@babel/preset-env": "7.5.4", + "eslint": "5.6.0", + "eslint-config-airbnb-base": "13.1.0", + "gulp": "4.0.0", + "gulp-babel": "8.0.0", + "gulp-clean-css": "4.2.0", + "gulp-eslint": "6.0.0", + "gulp-inline": "0.1.3", + "gulp-stylelint": "7.0.0", + "gulp-uglify": "3.0.0", + "stylelint": "13.2.0", + "stylelint-config-standard": "20.0.0" + } +} diff --git a/content-type-visualizer/source/index.html b/content-type-visualizer/source/index.html new file mode 100644 index 00000000..313aa067 --- /dev/null +++ b/content-type-visualizer/source/index.html @@ -0,0 +1,161 @@ + + + + + + External API UI Extension Sample + + + + + + + + + + + + + + + + + +
+
+
+
+
+
+
+ _ Content Types +
+
+ +
+
+
+ +
+ +
+
+ + + + +
+
+
+
+
+
+
+
+
+ + + + \ No newline at end of file diff --git a/content-type-visualizer/source/index.js b/content-type-visualizer/source/index.js new file mode 100644 index 00000000..aa73b604 --- /dev/null +++ b/content-type-visualizer/source/index.js @@ -0,0 +1,510 @@ +/* global Handlebars,jsPlumb,dagre,_,ClipboardJS */ +/* eslint no-undef: "error" */ +/* eslint-disable radix */ +/* eslint-disable no-param-reassign */ +/* eslint-disable default-case */ +/* eslint-disable no-underscore-dangle */ +/* eslint-disable one-var */ +/* eslint-disable no-plusplus */ +/* eslint-disable no-else-return */ +/* eslint-disable no-prototype-builtins */ + +let extensionField; +let referenceConnections = {}; +let labels = {}; +let json = {}; +let count = {}; +let entryCount = {}; +let env = []; +let loader = $('.reference-loading'); +let envArray; +let apiKey; +let appHost; +let blockLength = 0; + +// Counts number of fields in a content type +function countFields(schema, uid) { + if (count.hasOwnProperty(uid)) { + count[uid] += schema.length + blockLength; + blockLength = 0; + } else { + Object.defineProperty(count, uid, { + value: schema.length, + writable: true + }); + } +} + +// Reads the content type schema +function readSchema(schema, uid) { + schema.forEach(field => { + if (field.data_type === 'group' || field.data_type === 'global_field') { + readSchema(field.schema, uid); + } + + if (field.data_type === 'blocks') { + blockLength += field.blocks.length; + field.blocks.forEach(block => { + readSchema(block.schema, uid); + }); + } + + if (field.data_type === 'reference') { + if (referenceConnections.hasOwnProperty(uid)) { + referenceConnections[uid] = referenceConnections[uid].concat(field.reference_to); + } else { + let references = [field.reference_to]; + Object.defineProperty(referenceConnections, uid, { + value: references.flat(1), + writable: true + }); + } + } + }); + + countFields(schema, uid); +} + +// To add labels on the connections +function connectorLabel(source, target) { + let sourceLabel = (labels[source] === true) ? '1' : 'M'; + let targetLabel = (labels[target] === true) ? '1' : 'M'; + return sourceLabel + ':' + targetLabel; +} + +function removeDuplicates(originalArray, prop) { + /* eslint-disable guard-for-in */ + /* eslint-disable vars-on-top */ + var newArray = []; + var lookupObject = {}; + /* eslint-disable block-scoped-var */ + for (var i in originalArray) { + lookupObject[originalArray[i][prop]] = originalArray[i]; + } + for (i in lookupObject) { + newArray.push(lookupObject[i]); + } + return newArray; +} + +function updateSidebarDetails(id) { + envArray.forEach(envData => { + extensionField.stack.ContentType(id).Entry + .Query() + .environment(envData.env) + .includeCount() + .find() + .then(result => { + if (entryCount.hasOwnProperty(id)) { + entryCount[id][envData.env] = result.count; + } else { + entryCount[id] = { + [envData.env]: result.count + }; + } + }); + }); + + $('.copy-btn').attr('data-clipboard-text', JSON.stringify(json[id], null, 2)); + $('#json-collapsed').JSONView(json[id], { + collapsed: false + }); + $('.title').text(json[id].title); + $('.edit-icon').attr('href', `https://${appHost}/#!/stack/${apiKey}/content-type/${id}/content-type-builder`); + $('.count').text('Field count: ' + count[id]); + $('#environments').val('default').change(); + $('#entry-count').text('Entry count: _'); + $('#environments').click(function () { + let selectedEnv = $(this).children('option:selected').val(); + if (selectedEnv === 'default') { + $('#entry-count').text('Entry count: _'); + } else { + $('#entry-count').text('Entry count: ' + entryCount[id][selectedEnv]); + } + }); +} + +function init() { + let diagramSource = $('#diagram-template').html(); + let diagramTemplate = Handlebars.compile(diagramSource); + + // Helper that handles if else conditions + Handlebars.registerHelper('if_eq', function (a, b, opts) { + if (a === b) { + return opts.fn(this); + } else { + return opts.inverse(this); + } + }); + + // Helper that returns field icon image url for each field in the content type + Handlebars.registerHelper('fieldIcon', function (type, metaData, displayType) { + let iconUrl = ''; + + if (type === 'text') { + switch (true) { + case (metaData.multiline === true): + iconUrl = ''; + break; + case (metaData.allow_rich_text === true): + iconUrl = ''; + break; + case (metaData.markdown === true): + iconUrl = ''; + break; + case (metaData.extension === true): + iconUrl = ''; + break; + case (displayType === 'dropdown'): + iconUrl = ''; + break; + default: + iconUrl = ''; + } + } else { + switch (true) { + case (type === 'number'): + iconUrl = ''; + break; + case (type === 'boolean'): + iconUrl = ''; + break; + case (type === 'isodate'): + iconUrl = ''; + break; + case (type === 'file' && metaData.rich_text_type === 'standard'): + iconUrl = ''; + break; + case (type === 'link'): + iconUrl = ''; + break; + case (type === 'reference'): + iconUrl = ''; + break; + case (type === 'group'): + iconUrl = ''; + break; + case (type === 'global_field'): + iconUrl = ''; + break; + case (type === 'blocks'): + iconUrl = ''; + break; + case (metaData.extension === true): + iconUrl = ''; + break; + } + } + return iconUrl; + }); + + // Helper that returns the url which redirects to the particular content type builder + Handlebars.registerHelper('contentTypeBuilderUrl', function (uid) { + let url; + if (uid) { + url = `https://${appHost}/#!/stack/${apiKey}/content-type/${uid}/content-type-builder`; + } + return url; + }); + + // Retrieves all the content types from the stack + extensionField.stack.getContentTypes('', { include_count: true, include_global_field_schema: true }) + .then(res => { + res.content_types.forEach((contentType) => { + labels[contentType.uid] = contentType.options.singleton; + json[contentType.uid] = contentType; + readSchema(contentType.schema, contentType.uid); + }); + + // Using handlebars to pass content type data to render in HTML template + Handlebars.registerPartial('list', $('#partial').html()); + $('#diagram-placeholder').html(diagramTemplate({ diagramData: res.content_types })); + + $('.content-type-count span').text(res.count); + + // Sidebar toggle + const sidebar = document.querySelector('.sidebar'); + const mainContainer = document.querySelector('.container'); + const button = document.querySelector('.toggle'); + + var defaultId = Object.keys(json)[0]; + updateSidebarDetails(defaultId); + + document.querySelector('.toggle').onclick = function (e) { + e.preventDefault(); + var defaultJsonId = Object.keys(json)[0]; + updateSidebarDetails(defaultJsonId); + sidebar.classList.toggle('toggle-sidebar'); + mainContainer.classList.toggle('toggle-container'); + button.classList.toggle('toggle-button'); + }; + + // Event listener for jsonview tab + $('.icon-eye-open').on('click', function () { + var id = $(this).data('id'); + updateSidebarDetails(id); + sidebar.classList.remove('toggle-sidebar'); + mainContainer.classList.remove('toggle-container'); + button.classList.remove('toggle-button'); + }); + + // Copy button text change + let clipboard = new ClipboardJS('.btn'); + + $('.copy-btn').on('click', function () { + let copyBtn = $(this); + clipboard.on('success', function (e) { + copyBtn.text('Copied'); + setTimeout(function () { + copyBtn.text('Copy'); + }, 1000); + e.clearSelection(); + }); + }); + + // Building diagram using jsplumb + jsPlumb.ready(function () { + let $container = $('.container'); + let $panzoom = null; + let sourceAnchors = [ + [0.2, 0, 0, -1, 0, 0], + [1, 0.2, 1, 0, 0, 0], + [0.8, 1, 0, 1, 0, 0], + [0, 0.8, -1, 0, 0, 0] + ], + targetAnchors = [ + [0.6, 0, 0, -1], + [1, 0.6, 1, 0], + [0.4, 1, 0, 1], + [0, 0.4, -1, 0] + ], + connector = ['Flowchart', + { + cornerRadius: 5, + stub: 16 + } + ], + connectorStyle = { + gradient: { + stops: [ + [0, '#00f'], + [0.5, '#09098e'], + [1, '#00f'] + ] + }, + strokeWidth: 3, + stroke: '#00f', + outlineStroke: 'white', + outlineWidth: 4 + }, + endpoint = ['Dot', { cssClass: 'endpointClass', radius: 10, hoverClass: 'endpointHoverClass' }], + anEndpoint = { + endpoint: endpoint, + paintStyle: { fill: '#00f' }, + isSource: true, + isTarget: true, + maxConnections: -1, + connector: connector, + connectorStyle: connectorStyle + }; + + let plumb = jsPlumb.getInstance({ + DragOptions: { cursor: 'pointer', zIndex: 2000 }, + Anchors: [['Left', 'Right', 'Bottom'], ['Top', 'Bottom']], + Container: $container.find('.diagram') + }); + + plumb.batch(function () { + let endpoints = {}, + divsWithWindowClass = jsPlumb.getSelector('.container .window'); + + // Adding endpoints for each content type element + for (let i = 0; i < divsWithWindowClass.length; i++) { + let id = plumb.getId(divsWithWindowClass[i]); + endpoints[id] = [ + plumb.addEndpoint(id, anEndpoint, { anchor: sourceAnchors }), + plumb.addEndpoint(id, anEndpoint, { anchor: targetAnchors }) + ]; + } + + // Adding connections between the content type elements + for (let e in endpoints) { + if (referenceConnections[e]) { + for (let j = 0; j < referenceConnections[e].length; j++) { + plumb.connect({ + source: endpoints[e][0], + target: endpoints[referenceConnections[e][j]][1], + overlays: [ + ['Label', { label: connectorLabel((endpoints[e][0]).elementId, (endpoints[referenceConnections[e][j]][1]).elementId), cssClass: 'aLabel', location: 0.5 }] + ] + }); + } + } + } + + plumb.draggable(divsWithWindowClass); + + // Positioning and laying out all the content type elements on the canvas + let dg = new dagre.graphlib.Graph(); + dg.setGraph({ + nodesep: 30, ranksep: 30, marginx: 50, marginy: 50 + }); + dg.setDefaultEdgeLabel(function () { return {}; }); + $container.find('.window').each( + function (idx, node) { + let $n = $(node); + let box = { + width: Math.round($n.outerWidth()), + height: Math.round($n.outerHeight()) + }; + dg.setNode($n.attr('id'), box); + } + ); + plumb.getAllConnections() + .forEach(function (edge) { dg.setEdge(edge.source.id, edge.target.id); }); + dagre.layout(dg); + dg.nodes().forEach( + function (n) { + let node = dg.node(n); + let top = Math.round(node.y - node.height / 2) + 100 + 'px'; + let left = Math.round(node.x - node.width / 2) - 400 + 'px'; + $('#' + n).css({ left: left, top: top }); + } + ); + plumb.repaintEverything(); + + // Zooming and paning the diagram + _.defer(function () { + $panzoom = $container.find('.panzoom').panzoom({ + $zoomIn: $('.zoom-in'), + $zoomOut: $('.zoom-out'), + minScale: 0.2, + maxScale: 1, + increment: 0.1, + cursor: '', + startTransform: 'scale(0.5)', + ignoreChildrensEvents: true + }).on('panzoomstart', function () { + $panzoom.css('cursor', 'move'); + }) + .on('panzoomend', function () { + $panzoom.css('cursor', ''); + }); + + $panzoom.parent() + .on('mousewheel.focal', function (e) { + if (e.ctrlKey || e.originalEvent.ctrlKey) { + e.preventDefault(); + let delta = e.delta || e.originalEvent.wheelDelta; + let zoomOut = delta ? delta < 0 : e.originalEvent.deltaY > 0; + $panzoom.panzoom('zoom', zoomOut, { + animate: true, + exponential: false + }); + } else { + e.preventDefault(); + let deltaY = e.deltaY || e.originalEvent.wheelDeltaY || (-e.originalEvent.deltaY); + let deltaX = e.deltaX || e.originalEvent.wheelDeltaX || (-e.originalEvent.deltaX); + $panzoom.panzoom('pan', deltaX / 2, deltaY / 2, { + animate: true, + relative: true + }); + } + }) + .on('mousedown touchstart', function (ev) { + let matrix = $container.find('.panzoom').panzoom('getMatrix'); + let offsetX = matrix[4]; + let offsetY = matrix[5]; + let dragstart = { + x: ev.pageX, y: ev.pageY, dx: offsetX, dy: offsetY + }; + $(ev.target).css('cursor', 'move'); + $(this).data('dragstart', dragstart); + }) + .on('mousemove touchmove', function (ev) { + let dragstart = $(this).data('dragstart'); + if (dragstart) { + let deltaX = dragstart.x - ev.pageX; + let deltaY = dragstart.y - ev.pageY; + let matrix = $container.find('.panzoom').panzoom('getMatrix'); + matrix[4] = parseInt(dragstart.dx) - deltaX; + matrix[5] = parseInt(dragstart.dy) - deltaY; + $container.find('.panzoom').panzoom('setMatrix', matrix); + } + }) + .on('mouseup touchend touchcancel', function (ev) { + $(this).data('dragstart', null); + $(ev.target).css('cursor', ''); + }); + }); + + // Dragging all the content type elements + let currentScale = 1; + $container.find('.diagram .window').draggable({ + /* eslint-disable no-unused-vars */ + start: function (e) { + let pz = $container.find('.panzoom'); + currentScale = pz.panzoom('getMatrix')[0]; + $(this).css('cursor', 'move'); + pz.panzoom('disable'); + }, + drag: function (e, ui) { + ui.position.left /= currentScale; + ui.position.top /= currentScale; + if ($(this).hasClass('jsplumb-connected')) { + plumb.repaint($(this).attr('id'), ui.position); + } + }, + stop: function (e, ui) { + let nodeId = $(this).attr('id'); + if ($(this).hasClass('jsplumb-connected')) { + plumb.repaint(nodeId, ui.position); + } + $(this).css('cursor', ''); + $container.find('.panzoom').panzoom('enable'); + } + }); + }); + }); + + loader.hide(); + $('.sidebar').show(); + $('.parent-container').css('opacity', '1'); + }); +} + +function refresh() { + loader.show(); + count = {}; + $('.parent-container').css('opacity', '0.5'); + init(); +} + +$(document).ready(() => { + ContentstackUIExtension.init().then(extension => { + extensionField = extension; + Object.assign(extensionField.config, extensionField.fieldConfig); + apiKey = extensionField.stack._data.api_key; + appHost = extensionField.config.appHost; + + extensionField.stack.getEnvironments().then(res => { + res.environments.forEach(environment => { + env.push({ + env: environment.name + }); + }); + envArray = removeDuplicates(env, 'env'); + $.each(envArray, function (i, item) { + $('#environments').append($('