diff --git a/domain/section/flowchart/flowchart.go b/domain/section/flowchart/flowchart.go new file mode 100644 index 00000000..e054d30a --- /dev/null +++ b/domain/section/flowchart/flowchart.go @@ -0,0 +1,54 @@ +// Copyright 2016 Documize Inc. . All rights reserved. +// +// This software (Documize Community Edition) is licensed under +// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html +// +// You can operate outside the AGPL restrictions by purchasing +// Documize Enterprise Edition and obtaining a commercial license +// by contacting . +// +// https://documize.com + +package flowchart + +import ( + "net/http" + + "github.com/documize/community/core/env" + "github.com/documize/community/domain" + "github.com/documize/community/domain/section/provider" +) + +// Provider represents Draw.io +type Provider struct { + Runtime *env.Runtime + Store *domain.Store +} + +// Meta describes us +func (*Provider) Meta() provider.TypeMeta { + section := provider.TypeMeta{} + + section.ID = "d46a18f6-49fb-11e8-842f-0ed5f89f718b" + section.Title = "Graphical Diagram" + section.Description = "Draw.io powered flowcharts and diagrams" + section.ContentType = "flowchart" + section.PageType = "tab" + section.Order = 9991 + + return section +} + +// Command stub. +func (p *Provider) Command(ctx *provider.Context, w http.ResponseWriter, r *http.Request) { +} + +// Render returns data as-is (HTML). +func (p *Provider) Render(ctx *provider.Context, config, data string) string { + return data +} + +// Refresh just sends back data as-is. +func (*Provider) Refresh(ctx *provider.Context, config, data string) string { + return data +} diff --git a/domain/section/plantuml/plantuml.go b/domain/section/plantuml/plantuml.go index 64c7f4da..cb0b4e2c 100644 --- a/domain/section/plantuml/plantuml.go +++ b/domain/section/plantuml/plantuml.go @@ -25,7 +25,7 @@ import ( "github.com/documize/community/domain/section/provider" ) -// Provider represents Mermaid Diagram +// Provider represents PlantUML Text Diagram type Provider struct { Runtime *env.Runtime Store *domain.Store diff --git a/domain/section/register.go b/domain/section/register.go index 7edfb685..4f766252 100644 --- a/domain/section/register.go +++ b/domain/section/register.go @@ -18,6 +18,7 @@ import ( "github.com/documize/community/domain" "github.com/documize/community/domain/section/airtable" "github.com/documize/community/domain/section/code" + "github.com/documize/community/domain/section/flowchart" "github.com/documize/community/domain/section/gemini" "github.com/documize/community/domain/section/markdown" "github.com/documize/community/domain/section/papertrail" @@ -41,6 +42,7 @@ func Register(rt *env.Runtime, s *domain.Store) { provider.Register("wysiwyg", &wysiwyg.Provider{Runtime: rt, Store: s}) provider.Register("airtable", &airtable.Provider{Runtime: rt, Store: s}) provider.Register("plantuml", &plantuml.Provider{Runtime: rt, Store: s}) + provider.Register("flowchart", &flowchart.Provider{Runtime: rt, Store: s}) p := provider.List() rt.Log.Info(fmt.Sprintf("Registered %d sections", len(p))) diff --git a/gui/app/components/section/flowchart/embed.pdf b/gui/app/components/section/flowchart/embed.pdf new file mode 100644 index 00000000..dcb7a500 Binary files /dev/null and b/gui/app/components/section/flowchart/embed.pdf differ diff --git a/gui/app/components/section/flowchart/type-editor.js b/gui/app/components/section/flowchart/type-editor.js new file mode 100644 index 00000000..c0b54769 --- /dev/null +++ b/gui/app/components/section/flowchart/type-editor.js @@ -0,0 +1,138 @@ +// Copyright 2016 Documize Inc. . All rights reserved. +// +// This software (Documize Community Edition) is licensed under +// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html +// +// You can operate outside the AGPL restrictions by purchasing +// Documize Enterprise Edition and obtaining a commercial license +// by contacting . +// +// https://documize.com + +import { schedule } from '@ember/runloop'; +import { inject as service } from '@ember/service'; +import { computed, observer } from '@ember/object'; +import Component from '@ember/component'; + +export default Component.extend({ + appMeta: service(), + sectionSvc: service('section'), + isDirty: false, + waiting: false, + diagram: '', + diagramXML: '', + title: '', + readyToSave: false, + previewButtonCaption: 'Preview', + flowCallback: null, + editorId: computed('page', function () { + let page = this.get('page'); + return `flowchart-editor-${page.id}`; + }), + + goSave: observer('readyToSave', function() { + if (this.get('readyToSave')) { + let page = this.get('page'); + let meta = this.get('meta'); + meta.set('rawBody', this.get('diagram')); + page.set('title', this.get('title')); + + this.set('waiting', false); + this.teardownEditor(); + this.get('onAction')(page, meta); + } + }), + + didReceiveAttrs() { + this._super(...arguments); + + this.set('waiting', false); + this.set('diagram', this.get('meta.rawBody')); + this.set('title', this.get('page.title')); + }, + + didInsertElement() { + this._super(...arguments); + schedule('afterRender', () => { + this.setupEditor(); + }); + }, + + willDestroyElement() { + this._super(...arguments); + this.teardownEditor(); + }, + + setupEditor() { + let self = this; + + let flowCallback = function(evt) { + if (self.get('isDestroyed') || self.get('isDestroying')) return; + if (evt.origin !== 'https://www.draw.io') return; + if (evt.data.length === 0) return; + + let editorFrame = document.getElementById(self.get('editorId')); + var msg = JSON.parse(evt.data); + + switch (msg.event) { + case 'init': + editorFrame.contentWindow.postMessage( + JSON.stringify({action: 'load', autosave: 1, xmlpng: self.get('diagram')}), '*'); + break; + + case 'save': + self.set('diagramXML', msg.xml); + // Trigger onAction() callback using sneaky trick. + Mousetrap.trigger('ctrl+s'); + break; + + case 'autosave': + self.set('diagramXML', msg.xml); + break; + + case 'load': + break; + + case 'exit': + self.sendAction('onCancel'); + break; + + case 'export': + self.set('diagram', msg.data); + self.set('readyToSave', true); + break; + } + }; + + window.addEventListener('message', flowCallback); + this.set('flowCallback', flowCallback); + }, + + teardownEditor() { + window.removeEventListener('message', this.get('flowCallback')); + }, + + actions: { + isDirty() { + return this.get('isDirty') || (this.get('diagram') !== this.get('meta.rawBody')); + }, + + onCancel() { + this.teardownEditor(); + this.get('onCancel')(); + }, + + onAction(title) { + this.set('waiting', true); + this.set('title', title); + + let editorFrame = document.getElementById(this.get('editorId')); + editorFrame.contentWindow.postMessage(JSON.stringify({action: 'export', format: 'xmlpng', xml: this.get('diagramXML'), spin: 'Updating'}), '*'); + } + } +}); + +// https://github.com/jgraph/drawio-html5/blob/master/localstorage.html +// https://desk.draw.io/support/solutions/articles/16000042546-what-url-parameters-are-supported- +// https://desk.draw.io/support/solutions/articles/16000042544-how-does-embed-mode-work- +// https://jgraph.github.io/mxgraph/docs/js-api/files/editor/mxEditor-js.html diff --git a/gui/app/components/section/flowchart/type-renderer.js b/gui/app/components/section/flowchart/type-renderer.js new file mode 100644 index 00000000..e790b207 --- /dev/null +++ b/gui/app/components/section/flowchart/type-renderer.js @@ -0,0 +1,15 @@ +// Copyright 2016 Documize Inc. . All rights reserved. +// +// This software (Documize Community Edition) is licensed under +// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html +// +// You can operate outside the AGPL restrictions by purchasing +// Documize Enterprise Edition and obtaining a commercial license +// by contacting . +// +// https://documize.com + +import Component from '@ember/component'; + +export default Component.extend({ +}); diff --git a/gui/app/initializers/application.js b/gui/app/initializers/application.js index ab571f21..97fb8e8e 100644 --- a/gui/app/initializers/application.js +++ b/gui/app/initializers/application.js @@ -26,7 +26,6 @@ export function initialize(application) { application.register('constants:main', cs); Dropzone.autoDiscover = false; - CodeMirror.modeURL = "/codemirror/mode/%N/%N.js"; } export default { diff --git a/gui/app/models/page-meta.js b/gui/app/models/page-meta.js index d584d70b..ca1e7ca0 100644 --- a/gui/app/models/page-meta.js +++ b/gui/app/models/page-meta.js @@ -9,14 +9,14 @@ // // https://documize.com -import Model from 'ember-data/model'; import attr from 'ember-data/attr'; -// import { belongsTo } from 'ember-data/relationships'; +import Model from 'ember-data/model'; export default Model.extend({ pageId: attr('string'), documentId: attr('string'), orgId: attr('string'), + userId: attr('string'), rawBody: attr(), config: attr(), externalSource: attr('boolean', { defaultValue: false }), diff --git a/gui/app/pods/document/section/route.js b/gui/app/pods/document/section/route.js index f23d7f98..2ea4351b 100644 --- a/gui/app/pods/document/section/route.js +++ b/gui/app/pods/document/section/route.js @@ -10,7 +10,6 @@ // https://documize.com import { hash } from 'rsvp'; - import { inject as service } from '@ember/service'; import Route from '@ember/routing/route'; import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin'; diff --git a/gui/app/styles/app.scss b/gui/app/styles/app.scss index e7bd3b50..13e9ac40 100644 --- a/gui/app/styles/app.scss +++ b/gui/app/styles/app.scss @@ -22,16 +22,8 @@ @import "vendor.scss"; @import "print.scss"; -@import "section/trello.scss"; -@import "section/gemini.scss"; -@import "section/github.scss"; -@import "section/markdown.scss"; -@import "section/table.scss"; -@import "section/code.scss"; -@import "section/plantuml.scss"; -@import "section/papertrail.scss"; -@import "section/wysiwyg.scss"; @import "news.scss"; +@import "section/all.scss"; @import "enterprise/all.scss"; // Bootstrap override that removes gutter space on smaller screens diff --git a/gui/app/styles/section/all.scss b/gui/app/styles/section/all.scss new file mode 100644 index 00000000..bcc2fdd0 --- /dev/null +++ b/gui/app/styles/section/all.scss @@ -0,0 +1,10 @@ +@import "trello.scss"; +@import "gemini.scss"; +@import "github.scss"; +@import "markdown.scss"; +@import "table.scss"; +@import "code.scss"; +@import "plantuml.scss"; +@import "papertrail.scss"; +@import "wysiwyg.scss"; +@import "flowchart.scss"; diff --git a/gui/app/styles/section/flowchart.scss b/gui/app/styles/section/flowchart.scss new file mode 100644 index 00000000..dd3e6586 --- /dev/null +++ b/gui/app/styles/section/flowchart.scss @@ -0,0 +1,715 @@ +.geEditor { + font-family:Helvetica Neue,Helvetica,Arial Unicode MS,Arial !important; + font-size:10pt; + border:none; + margin:0px; + color:#000000 !important; +} +.geEditor input[type=text]::-ms-clear { + display: none; +} +.geMenubarContainer .geItem, .geToolbar .geButton, .geToolbar .geLabel, .geSidebarContainer .geTitle { + cursor:pointer !important; +} +.geBackgroundPage { + -webkit-box-shadow:0px 0px 3px 0px #d9d9d9; + -moz-box-shadow:0px 0px 3px 0px #d9d9d9; + box-shadow:0px 0px 3px 0px #d9d9d9; +} +.geSidebarContainer a, .geMenubarContainer a, .geToolbar a { + color:#000000 !important; + text-decoration:none; +} +.geMenubarContainer, .geToolbarContainer, .geDiagramContainer, .geSidebarContainer, .geFooterContainer, .geHsplit, .geVsplit { + overflow:hidden; + position:absolute; + cursor:default; +} +.geFormatContainer { + background-color:whiteSmoke; + overflow-x: hidden !important; + overflow-y: auto !important; + font-size:12px; +} +.geDiagramContainer { + background-color:#ffffff; + outline:none; +} +.geMenubar, .geToolbar { + white-space:nowrap; + display:block; + width:100%; +} +.geMenubarContainer .geItem, .geToolbar .geButton, .geToolbar .geLabel, .geSidebar, .geSidebarContainer .geTitle, .geSidebar .geItem, .mxPopupMenuItem { + -webkit-transition: all 0.1s ease-in-out; + -moz-transition: all 0.1s ease-in-out; + -o-transition: all 0.1s ease-in-out; + -ms-transition: all 0.1s ease-in-out; + transition: all 0.1s ease-in-out; +} +.geHint { + background-color: #ffffff; + border: 1px solid gray; + padding: 4px 16px 4px 16px; + border-radius:3px; + -webkit-box-shadow: 1px 1px 2px 0px #ddd; + -moz-box-shadow: 1px 1px 2px 0px #ddd; + box-shadow: 1px 1px 2px 0px #ddd; + opacity:0.8; + filter:alpha(opacity=80); +} +.geStatusAlert { + white-space:nowrap; + margin-top:-5px; + font-size:12px; + padding:4px 6px 4px 6px; + background-color:#f2dede; + border:1px solid #ebccd1; + color:#a94442 !important; + border-radius:3px; +} +.geStatusAlert:hover { + background-color:#f1d8d8; + border-color:#d6b2b8; +} +.geStatusMessage { + white-space:nowrap; + margin-top:-5px; + padding:4px 6px 4px 6px; + font-size:12px; + background-image: -webkit-linear-gradient(top,#dff0d8 0,#c8e5bc 100%); + background-image: -o-linear-gradient(top,#dff0d8 0,#c8e5bc 100%); + background-image: -webkit-gradient(linear,left top,left bottom,from(#dff0d8),to(#c8e5bc)); + background-image: linear-gradient(to bottom,#dff0d8 0,#c8e5bc 100%); + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffc8e5bc', GradientType=0); + background-repeat: repeat-x; + border:1px solid #b2dba1; + border-radius:3px; + color:#3c763d !important; +} +.geStatusMessage:hover { + background:#c8e5bc; + border-color:#b2dba1; +} +.geAlert { + position:absolute; + white-space:nowrap; + padding:14px; + background-color:#f2dede; + border:1px solid #ebccd1; + color:#a94442; + border-radius:3px; + -webkit-box-shadow: 2px 2px 3px 0px #ddd; + -moz-box-shadow: 2px 2px 3px 0px #ddd; + box-shadow: 2px 2px 3px 0px #ddd; +} +.geBtn { + background-color: #f5f5f5; + border-radius: 2px; + border: 1px solid #d8d8d8; + color: #333; + cursor: default; + font-size: 11px; + font-weight: bold; + height: 29px; + line-height: 27px; + margin: 0 0 0 8px; + min-width: 72px; + outline: 0; + padding: 0 8px; + cursor: pointer; +} +.geBtn:hover, .geBtn:focus { + -webkit-box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + -moz-box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + border: 1px solid #c6c6c6; + background-color: #f8f8f8; + background-image: linear-gradient(#f8f8f8 0px,#f1f1f1 100%); + color: #111; +} +.geBtn:disabled { + opacity: .5; +} +.geBtnUp { + background-image: url(); + _background-image: url(up.gif); + background-position: center center; + background-repeat: no-repeat; +} +.geBtnUp:active { + background-color: #4d90fe; + background-image: linear-gradient(#4d90fe 0px,#357ae8 100%); +} +.geBtnDown { + background-image: url(); + _background-image: url(down.gif); + background-position: center center; + background-repeat: no-repeat; +} +.geBtnDown:active { + background-color: #4d90fe; + background-image: linear-gradient(#4d90fe 0px,#357ae8 100%); +} +.geColorBtn { + background-color: #f5f5f5; + background-image: linear-gradient(#f5f5f5 0px,#e1e1e1 100%); + border-radius: 4px; + border: 1px solid rgba(0,0,0,0.5); + color: #333; + cursor: default; + margin: 0px; + outline: 0; + padding: 0px; + cursor: pointer; +} +.geColorBtn:hover { + -webkit-box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + -moz-box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + box-shadow: 0px 1px 1px rgba(0,0,0,0.1); + border: 1px solid rgba(0,0,0,0.7); +} +.geColorBtn:active { + background-color: #4d90fe; + background-image: linear-gradient(#4d90fe 0px,#357ae8 100%); + border: 1px solid #2f5bb7; + color: #fff; +} +.geColorBtn:disabled { + opacity: .5; +} +.gePrimaryBtn { + background-color: #4d90fe; + background-image: linear-gradient(#4d90fe 0px,#4787ed 100%); + border: 1px solid #3079ed; + color: #fff; +} +.gePrimaryBtn:hover, .gePrimaryBtn:focus { + background-color: #357ae8; + background-image: linear-gradient(#4d90fe 0px,#357ae8 100%); + border: 1px solid #2f5bb7; + color: #fff; +} +.gePrimaryBtn:disabled { + opacity: .5; +} +.geAlertLink { + color:#843534; + font-weight:700; + text-decoration:none; +} +.geMenubarContainer { + background-color:#ffffff; +} +.geMenubar { + padding:0px 2px 0px 2px; + vertical-align:middle; +} +.geMenubarContainer .geItem, .geToolbar .geItem { + padding:6px 8px 6px 8px; + cursor:default; +} +.geMenubarContainer .geItem:hover { + background:#eeeeee; +} +.mxDisabled:hover { + background:inherit !important; +} +.geMenubar a.geStatus { + color:#b3b3b3; + padding-left:6px; + display:inline-block; + cursor:default !important; +} +.geMenubar a.geStatus:hover { + background:transparent; +} +.geMenubarMenu { + border:1px solid #d5d5d5 !important; +} +.geToolbarContainer { + background:whiteSmoke; + border-bottom:1px solid #e0e0e0; +} +.geSidebarContainer .geToolbarContainer { + background:transparent; + border-bottom:none; +} +.geSidebarContainer button { + text-overflow:ellipsis; + overflow:hidden; +} +.geToolbar { + padding:1px 0px 0px 6px; + border-top:1px solid #e0e0e0; + -webkit-box-shadow: inset 0 1px 0 0 #fff; + -moz-box-shadow: inset 0 1px 0 0 #fff; + box-shadow: inset 0 1px 0 0 #fff; +} +.geToolbarContainer .geSeparator { + float:left; + width:1px; + height:34px; + background:#e5e5e5; + margin-left:6px; + margin-right:6px; + margin-top:-2px; +} +.geToolbarContainer .geButton { + float:left; + width:20px; + height:20px; + padding:0px 2px 4px 2px; + margin:2px; + border:1px solid transparent; + cursor:pointer; + opacity:0.6; + filter:alpha(opacity=60); +} +.geToolbarContainer .geButton:hover { + border:1px solid gray; + border-radius:2px; + opacity:1; + filter:none !important; +} +.geToolbarContainer .geButton:active { + border:1px solid black; +} +div.mxWindow .geButton { + margin: -1px 2px 2px 2px; + padding: 1px 2px 2px 1px; +} +.geToolbarContainer .geLabel { + float:left; + margin:2px; + cursor:pointer; + padding:3px 5px 3px 5px; + border:1px solid transparent; + opacity:0.6; + filter:alpha(opacity=60); +} +.geToolbarContainer .geLabel:hover { + border:1px solid gray; + border-radius:2px; + opacity:0.9; + filter:alpha(opacity=90) !important; +} +.geToolbarContainer .geLabel:active { + border:1px solid black; + opacity:1; + filter:none !important; +} +.geToolbarContainer .mxDisabled:hover { + border:1px solid transparent !important; + opacity:0.2 !important; + filter:alpha(opacity=20) !important; +} +.geToolbarMenu { + border:3px solid #e0e0e0 !important; + -webkit-box-shadow:none !important; + -moz-box-shadow:none !important; + box-shadow:none !important; + filter:none !important; +} +.geDiagramBackdrop { + background-color: #ebebeb; + border: 1px solid #e5e5e5; +} +.geSidebarContainer { + background:#ffffff; + overflow:hidden; + position:absolute; + border-top:1px solid #e5e5e5; + overflow:auto; +} +.geSidebar { + background:whiteSmoke; + border-bottom:1px solid #e5e5e5; + padding:5px; + _padding:1px; + padding-bottom:12px; + overflow:hidden; +} +.geSidebarContainer .geTitle { + display:block; + font-size:9pt; + border-bottom:1px solid #e5e5e5; + font-weight:normal; + padding:6px 0px 6px 14px; + margin:0px; + cursor:default; + background:#eeeeee; + white-space:nowrap; + overflow:hidden; + text-overflow:ellipsis; + line-height:1.4em; +} +.geSidebarContainer .geTitle:hover { + background:#e5e5e5; +} +.geTitle img { + opacity:0.5; + _filter:alpha(opacity=50); +} +.geTitle img:hover { + opacity:1; + _filter:alpha(opacity=100); +} +.geTitle .geButton { + border:1px solid transparent; + padding:3px; + border-radius:2px; +} +.geTitle .geButton:hover { + border:1px solid gray; +} +.geSidebar .geItem { + display:inline-block; + background-repeat:no-repeat; + background-position:50% 50%; + border:1px solid transparent; + border-radius:2px; + cursor: move; +} +.geSidebar .geItem:hover { + border:1px solid gray !important; +} +.geItem { + vertical-align: top; + display: inline-block; +} +.geSidebarTooltip { + position:absolute; + background:white; + overflow:hidden; + border:1px solid gray; + border-radius:8px; + -webkit-box-shadow:0px 0px 2px 2px #d5d5d5; + -moz-box-shadow:0px 0px 2px 2px #d5d5d5; + box-shadow:0px 0px 2px 2px #d5d5d5; + _filter:progid:DXImageTransform.Microsoft.DropShadow(OffX=2, OffY=2, Color='#d5d5d5', Positive='true'); +} +.geFooterContainer { + background:#e5e5e5; + border-top:1px solid #c0c0c0; +} +.geFooterContainer a { + display:inline-block; + box-sizing:border-box; + width:100%; + white-space:nowrap; + font-size:14px; + color:#235695; + font-weight:bold; + text-decoration:none; +} +.geFooterContainer table { + border-collapse:collapse; + margin:0 auto; +} +.geFooterContainer td { + border-left:1px solid #c0c0c0; + border-right:1px solid #c0c0c0; +} +.geFooterContainer td:hover { + background-color: #b3b3b3; +} +.geHsplit { + cursor:col-resize; + background-color:#e5e5e5; + background-image:url(); + _background-image:url('thumb_vertical.png'); + background-repeat:no-repeat; + background-position:center center; +} +.geVsplit { + font-size:1pt; + cursor:row-resize; + background-color:#e5e5e5; + background-image:url(); + _background-image:url('thumb_horz.png'); + background-repeat:no-repeat; + background-position:center center; +} +.geHsplit:hover, .geVsplit:hover { + background-color:#d5d5d5; +} +.geDialog { + position:absolute; + background:white; + overflow:hidden; + padding:30px; + border:1px solid #acacac; + -webkit-box-shadow:0px 0px 2px 2px #d5d5d5; + -moz-box-shadow:0px 0px 2px 2px #d5d5d5; + box-shadow:0px 0px 2px 2px #d5d5d5; + _filter:progid:DXImageTransform.Microsoft.DropShadow(OffX=2, OffY=2, Color='#d5d5d5', Positive='true'); + z-index: 2; +} +.geDialogClose { + position:absolute; + width:9px; + height:9px; + opacity:0.5; + cursor:pointer; + _filter:alpha(opacity=50); +} +.geDialogClose:hover { + opacity:1; +} +.geDialogTitle { + box-sizing:border-box; + white-space:nowrap; + background:rgb(229, 229, 229); + border-bottom:1px solid rgb(192, 192, 192); + font-size:15px; + font-weight:bold; + text-align:center; + color:rgb(35, 86, 149); +} +.geDialogFooter { + background:whiteSmoke; + white-space:nowrap; + text-align:right; + box-sizing:border-box; + border-top:1px solid #e5e5e5; + color:darkGray; +} +.geSprite { + background:url('') no-repeat; + _background:url('sprites.png') no-repeat top left; + width:21px; + height:21px; +} +.geBaseButton { + padding:10px; + border-radius:6px; + border:1px solid #c0c0c0; + cursor:pointer; + background-color:#ececec; + background-image:linear-gradient(#ececec 0%, #fcfcfc 100%); +} +.geBaseButton:hover { + background:#ececec; +} +.geBigButton { + color:#ffffff; + border: none; + padding:10px; + font-size:14pt; + white-space: nowrap; + border-radius:6px; + text-shadow: rgb(41, 89, 137) 0px 1px 0px; + background-color:#428bca; + background-image:linear-gradient(rgb(70, 135, 206) 0px, rgb(48, 104, 162) 100%); + -webkit-box-shadow: rgba(255, 255, 255, 0.0980392) 0px 1px 0px 0px inset, rgba(0, 0, 0, 0.2) 0px 1px 1px 0px; + -moz-box-shadow: rgba(255, 255, 255, 0.0980392) 0px 1px 0px 0px inset, rgba(0, 0, 0, 0.2) 0px 1px 1px 0px; + box-shadow: rgba(255, 255, 255, 0.0980392) 0px 1px 0px 0px inset, rgba(0, 0, 0, 0.2) 0px 1px 1px 0px; +} +.geBigButton:hover { + background-color:#2d6ca2; + background-image: linear-gradient(rgb(90, 148, 211) 0px, rgb(54, 115, 181) 100%); +} +.geBigButton:active { + background-color: rgb(54, 115, 181); + background-image: none; +} +@media print { + div.geNoPrint { display: none !important; } +} +.geSprite-actualsize { background-position: 0 0; } +.geSprite-bold { background-position: 0 -46px; } +.geSprite-bottom { background-position: 0 -92px; } +.geSprite-center { background-position: 0 -138px; } +.geSprite-delete { background-position: 0 -184px; } +.geSprite-fillcolor { background-position: 0 -229px; } +.geSprite-fit { background-position: 0 -277px; } +.geSprite-fontcolor { background-position: 0 -322px; } +.geSprite-gradientcolor { background-position: 0 -368px; } +.geSprite-image { background-position: 0 -414px; } +.geSprite-italic { background-position: 0 -460px; } +.geSprite-left { background-position: 0 -505px; } +.geSprite-middle { background-position: 0 -552px; } +.geSprite-print { background-position: 0 -598px; } +.geSprite-redo { background-position: 0 -644px; } +.geSprite-right { background-position: 0 -689px; } +.geSprite-shadow { background-position: 0 -735px; } +.geSprite-strokecolor { background-position: 0 -782px; } +.geSprite-top { background-position: 0 -828px; } +.geSprite-underline { background-position: 0 -874px; } +.geSprite-undo { background-position: 0 -920px; } +.geSprite-zoomin { background-position: 0 -966px; } +.geSprite-zoomout { background-position: 0 -1012px; } +.geSprite-arrow { background-position: 0 -1059px; } +.geSprite-linkedge { background-position: 0 -1105px; } +.geSprite-straight { background-position: 0 -1150px; } +.geSprite-entity { background-position: 0 -1196px; } +.geSprite-orthogonal { background-position: 0 -1242px; } +.geSprite-curved { background-position: 0 -1288px; } +.geSprite-noarrow { background-position: 0 -1334px; } +.geSprite-endclassic { background-position: 0 -1380px; } +.geSprite-endopen { background-position: 0 -1426px; } +.geSprite-endblock { background-position: 0 -1472px; } +.geSprite-endoval { background-position: 0 -1518px; } +.geSprite-enddiamond { background-position: 0 -1564px; } +.geSprite-endthindiamond { background-position: 0 -1610px; } +.geSprite-endclassictrans { background-position: 0 -1656px; } +.geSprite-endblocktrans { background-position: 0 -1702px; } +.geSprite-endovaltrans { background-position: 0 -1748px; } +.geSprite-enddiamondtrans { background-position: 0 -1794px; } +.geSprite-endthindiamondtrans { background-position: 0 -1840px; } +.geSprite-startclassic { background-position: 0 -1886px; } +.geSprite-startopen { background-position: 0 -1932px; } +.geSprite-startblock { background-position: 0 -1978px; } +.geSprite-startoval { background-position: 0 -2024px; } +.geSprite-startdiamond { background-position: 0 -2070px; } +.geSprite-startthindiamond { background-position: 0 -2116px; } +.geSprite-startclassictrans { background-position: 0 -2162px; } +.geSprite-startblocktrans { background-position: 0 -2208px; } +.geSprite-startovaltrans { background-position: 0 -2254px; } +.geSprite-startdiamondtrans { background-position: 0 -2300px; } +.geSprite-startthindiamondtrans { background-position: 0 -2346px; } +.geSprite-globe { background-position: 0 -2392px; } +.geSprite-orderedlist { background-position: 0 -2438px; } +.geSprite-unorderedlist { background-position: 0 -2484px; } +.geSprite-horizontalrule { background-position: 0 -2530px; } +.geSprite-link { background-position: 0 -2576px; } +.geSprite-indent { background-position: 0 -2622px; } +.geSprite-outdent { background-position: 0 -2668px; } +.geSprite-code { background-position: 0 -2714px; } +.geSprite-fontbackground { background-position: 0 -2760px; } +.geSprite-removeformat { background-position: 0 -2806px; } +.geSprite-superscript { background-position: 0 -2852px; } +.geSprite-subscript { background-position: 0 -2898px; } +.geSprite-table { background-position: 0 -2944px; } +.geSprite-deletecolumn { background-position: 0 -2990px; } +.geSprite-deleterow { background-position: 0 -3036px; } +.geSprite-insertcolumnafter { background-position: 0 -3082px; } +.geSprite-insertcolumnbefore { background-position: 0 -3128px; } +.geSprite-insertrowafter { background-position: 0 -3174px; } +.geSprite-insertrowbefore { background-position: 0 -3220px; } +.geSprite-grid { background-position: 0 -3272px; } +.geSprite-guides { background-position: 0 -3324px; } +.geSprite-dots { background-position: 0 -3370px; } +.geSprite-alignleft { background-position: 0 -3416px; } +.geSprite-alignright { background-position: 0 -3462px; } +.geSprite-aligncenter { background-position: 0 -3508px; } +.geSprite-aligntop { background-position: 0 -3554px; } +.geSprite-alignbottom { background-position: 0 -3600px; } +.geSprite-alignmiddle { background-position: 0 -3646px; } +.geSprite-justifyfull { background-position: 0 -3692px; } +.geSprite-formatpanel { background-position: 0 -3738px; } +.geSprite-connection { background-position: 0 -3784px; } +.geSprite-vertical { background-position: 0 -3830px; } +.geSprite-simplearrow { background-position: 0 -3876px; } +.geSprite-plus { background-position: 0 -3922px; } +.geSprite-rounded { background-position: 0 -3968px; } +.geSprite-toback { background-position: 0 -4014px; } +.geSprite-tofront { background-position: 0 -4060px; } +.geSprite-duplicate { background-position: 0 -4106px; } +.geSprite-insert { background-position: 0 -4152px; } +.geSprite-endblockthin { background-position: 0 -4201px; } +.geSprite-endblockthintrans { background-position: 0 -4247px; } +.geSprite-enderone { background-position: 0 -4293px; } +.geSprite-enderonetoone { background-position: 0 -4339px; } +.geSprite-enderonetomany { background-position: 0 -4385px; } +.geSprite-endermany { background-position: 0 -4431px; } +.geSprite-enderoneopt { background-position: 0 -4477px; } +.geSprite-endermanyopt { background-position: 0 -4523px; } +.geSprite-endclassicthin { background-position: 0 -4938px; } +.geSprite-endclassicthintrans { background-position: 0 -4984px; } +.geSprite-enddash { background-position: 0 -5029px; } +.geSprite-endcircleplus { background-position: 0 -5075px; } +.geSprite-endcircle { background-position: 0 -5121px; } +.geSprite-endasync { background-position: 0 -5167px; } +.geSprite-endasynctrans { background-position: 0 -5213px; } +.geSprite-startblockthin { background-position: 0 -4569px; } +.geSprite-startblockthintrans { background-position: 0 -4615px; } +.geSprite-starterone { background-position: 0 -4661px; } +.geSprite-starteronetoone { background-position: 0 -4707px; } +.geSprite-starteronetomany { background-position: 0 -4753px; } +.geSprite-startermany { background-position: 0 -4799px; } +.geSprite-starteroneopt { background-position: 0 -4845px; } +.geSprite-startermanyopt { background-position: 0 -4891px; } +.geSprite-startclassicthin { background-position: 0 -5259px; } +.geSprite-startclassicthintrans { background-position: 0 -5305px; } +.geSprite-startdash { background-position: 0 -5351px; } +.geSprite-startcircleplus { background-position: 0 -5397px; } +.geSprite-startcircle { background-position: 0 -5443px; } +.geSprite-startasync { background-position: 0 -5489px; } +.geSprite-startasynctrans { background-position: 0 -5535px; } +.geSprite-startcross { background-position: 0 -5581px; } +.geSprite-startopenthin { background-position: 0 -5627px; } +.geSprite-startopenasync { background-position: 0 -5673px; } +.geSprite-endcross { background-position: 0 -5719px; } +.geSprite-endopenthin { background-position: 0 -5765px; } +.geSprite-endopenasync { background-position: 0 -5811px; } +.geSprite-verticalelbow { background-position: 0 -5857px; } +.geSprite-horizontalelbow { background-position: 0 -5903px; } +.geSprite-horizontalisometric { background-position: 0 -5949px; } +.geSprite-verticalisometric { background-position: 0 -5995px; } +html div.mxRubberband { + border-color:#0000DD; + background:#99ccff; +} +td.mxPopupMenuIcon div { + width:16px; + height:16px; +} +html div.mxPopupMenu { + -webkit-box-shadow:2px 2px 3px #d5d5d5; + -moz-box-shadow:2px 2px 3px #d5d5d5; + box-shadow:2px 2px 3px #d5d5d5; + _filter:progid:DXImageTransform.Microsoft.DropShadow(OffX=2, OffY=2, Color='#d0d0d0', Positive='true'); + background:white; + position:absolute; + border:3px solid #e7e7e7; + padding:3px; +} +html table.mxPopupMenu { + border-collapse:collapse; + margin:0px; +} +html td.mxPopupMenuItem { + padding:7px 30px 7px 30px; + font-family:Helvetica Neue,Helvetica,Arial Unicode MS,Arial; + font-size:10pt; +} +html td.mxPopupMenuIcon { + background-color:white; + padding:0px; +} +td.mxPopupMenuIcon .geIcon { + padding:2px; + padding-bottom:4px; + margin:2px; + border:1px solid transparent; + opacity:0.5; + _width:26px; + _height:26px; +} +td.mxPopupMenuIcon .geIcon:hover { + border:1px solid gray; + border-radius:2px; + opacity:1; +} +html tr.mxPopupMenuItemHover { + background-color: #eeeeee; + color: black; +} +table.mxPopupMenu hr { + color:#cccccc; + background-color:#cccccc; + border:none; + height:1px; +} +table.mxPopupMenu tr { + font-size:4pt; +} +html td.mxWindowTitle { + font-family:Helvetica Neue,Helvetica,Arial Unicode MS,Arial; + text-align:left; + font-size:12px; + color:rgb(112, 112, 112); + padding:4px; +} diff --git a/gui/app/templates/components/section/flowchart/type-editor.hbs b/gui/app/templates/components/section/flowchart/type-editor.hbs new file mode 100644 index 00000000..ddf192cd --- /dev/null +++ b/gui/app/templates/components/section/flowchart/type-editor.hbs @@ -0,0 +1,14 @@ +{{#section/base-editor document=document folder=folder page=page busy=waiting tip="Concise name that describes the diagram" isDirty=(action 'isDirty') onCancel=(action 'onCancel') onAction=(action 'onAction')}} +
+
+
+
+ +
+
+
+
+{{/section/base-editor}} diff --git a/gui/app/templates/components/section/flowchart/type-renderer.hbs b/gui/app/templates/components/section/flowchart/type-renderer.hbs new file mode 100644 index 00000000..18b62985 --- /dev/null +++ b/gui/app/templates/components/section/flowchart/type-renderer.hbs @@ -0,0 +1,3 @@ +
+ +
diff --git a/gui/public/sections/flowchart.png b/gui/public/sections/flowchart.png new file mode 100644 index 00000000..56ee0627 Binary files /dev/null and b/gui/public/sections/flowchart.png differ diff --git a/gui/public/sections/flowchart@2x.png b/gui/public/sections/flowchart@2x.png new file mode 100644 index 00000000..8ddb71b6 Binary files /dev/null and b/gui/public/sections/flowchart@2x.png differ