Created
          July 1, 2016 03:26 
        
      - 
      
- 
        Save elquimista/677af350749de3dd631596494ba8a01d to your computer and use it in GitHub Desktop. 
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
  | SVGAnimatedTransformList.prototype.translate = function (x, y) { | |
| var xForms = this.baseVal, | |
| xFormTranslate = undefined, | |
| transform = undefined; | |
| for (var i = 0, n = xForms.numberOfItems; i < n; i ++) { | |
| transform = xForms.getItem(i); | |
| if (transform.type === SVGTransform.SVG_TRANSFORM_TRANSLATE) { | |
| xFormTranslate = transform; | |
| break; | |
| } | |
| if (transform.type === SVGTransform.SVG_TRANSFORM_MATRIX) { | |
| xFormTranslate = transform; | |
| break; | |
| } | |
| } | |
| if (xFormTranslate) { | |
| if (x) { | |
| xFormTranslate.matrix.e = x; | |
| } | |
| if (y) { | |
| xFormTranslate.matrix.f = y; | |
| } | |
| } | |
| }; | |
| var RAPPID = function() { | |
| var $ = window.jQuery, | |
| graph, paper, paperScroller, snaplines, stencil, nav, inspector, halo, gOptions, | |
| shapeSize = [50, 50], | |
| shapeMargin = [20, 10], | |
| shapeTextMarginTop = 5, | |
| iconSize = [32, 32], | |
| shapeBigSize = [150, 150], | |
| zoomMax = 1, | |
| zoomMin = 0.4, | |
| zoomStep = 0.2, | |
| defaultPaperSize = [$('#paper').width() || 1500, $('#paper').height() || 900], | |
| wfAutoSaveInterval = 1000, // in milliseconds | |
| chartPieRadius = 60, | |
| chartSerieFillColor = ['#8bce5d', '#53abdd', '#c377b1', '#ffe891', '#888888', '#53abdd', '#c377b1', '#ffe891', '#888888'], | |
| logTimerId, | |
| chart = new joint.shapes.chart.Pie({ | |
| attrs: { | |
| '.slice-inner-label': { fill: '#000' } | |
| }, | |
| position: { x: 0, y: 0 }, | |
| size: { width: chartPieRadius * 2, height: chartPieRadius * 2 }, | |
| serieDefaults: { | |
| degree: 360, | |
| startAngle: 0, | |
| showLegend: false | |
| }, | |
| sliceDefaults: { | |
| innerLabel: '{label}' | |
| }, | |
| series: [] | |
| }), | |
| $statusBarEl = $('#status_bar'), | |
| configErrors = {}, | |
| linkDownFlag = false, | |
| graphHasChanged = false, | |
| agentDownFlag = false, | |
| $goalMarkerEl = V('<path d="M 0.000 10.000 L 11.756 16.180 L 9.511 3.090 L 19.021 -6.180 L 5.878 -8.090 L 0.000 -20.000 L -5.878 -8.090 L -19.021 -6.180 L -9.511 3.090 L -11.756 16.180 L 0.000 10.000" fill="#f8b600" stroke="#f8b600" transform="translate(75 145)"></path>').node, | |
| // Used for showing sticky config option icons for agents that are not configured yet. | |
| halo2 = {}, | |
| configIcon = '', | |
| configIconGreen = '', | |
| // Customized Halo - Halo2 | |
| Halo2 = function(options) { | |
| // possible options: { cellView } | |
| this.options = options; | |
| this.$el = undefined; | |
| } | |
| ; | |
| Halo2.prototype = { | |
| _options: { | |
| tinyThreshold: 40, | |
| smallThreshold: 80 | |
| }, | |
| render: function() { | |
| var cellView = this.options.cellView, | |
| $cell = cellView.$el, | |
| cellPos = $cell.offset(), | |
| paperPos = $('#paper .paper').offset(), | |
| $haloEl = $('<div class="halo surrounding halo2"></div>'), | |
| cellSize = cellView.model.attributes.size, | |
| width = cellSize.width * paperScroller._sx, | |
| height = cellSize.height * paperScroller._sy, | |
| _options = this._options, | |
| $handleEl = $('<div class="handle config"></div>'); | |
| $haloEl.css('width', width + 'px') | |
| .css('height', height + 'px') | |
| .css('left', (cellPos.left - paperPos.left) + 'px') | |
| .css('top', (cellPos.top - paperPos.top) + 'px') | |
| .attr('data-model-id', cellView.model.id) | |
| .toggleClass('tiny', width < _options.tinyThreshold && height < _options.tinyThreshold) | |
| .toggleClass('small', !$haloEl.hasClass('tiny') && width < _options.smallThreshold && height < _options.smallThreshold); | |
| $handleEl.css('background-image', 'url("' + configIcon + '")') | |
| .appendTo($haloEl); | |
| $haloEl.appendTo(cellView.paper.$el); | |
| this.$el = $haloEl; | |
| }, | |
| remove: function() { | |
| if (this.$el) { | |
| this.$el.remove(); | |
| this.$el = undefined; | |
| } | |
| }, | |
| update: function() { | |
| try { | |
| var cellView = this.options.cellView, | |
| $cell = cellView.$el, | |
| cellPos = $cell.offset(), | |
| paperPos = $('#paper .paper').offset(), | |
| cellSize = cellView.model.attributes.size, | |
| width = cellSize.width * paperScroller._sx, | |
| height = cellSize.height * paperScroller._sy, | |
| _options = this._options, | |
| $el = this.$el; | |
| $el.css('width', width + 'px') | |
| .css('height', height + 'px') | |
| .css('left', (cellPos.left - paperPos.left) + 'px') | |
| .css('top', (cellPos.top - paperPos.top) + 'px') | |
| .toggleClass('tiny', width < _options.tinyThreshold && height < _options.tinyThreshold) | |
| .toggleClass('small', !$el.hasClass('tiny') && width < _options.smallThreshold && height < _options.smallThreshold); | |
| } catch (error) { | |
| console.log(error); | |
| } | |
| } | |
| }; | |
| /** | |
| * getWordWrapText() | |
| * | |
| * @param String text | |
| * @param Object options | |
| */ | |
| function getWordWrapText(text, options) { | |
| var lines = joint.util.breakText(text, options).split("\n"), | |
| textContent = ''; | |
| for (var i = 0, n = lines.length; i < n; i ++) { | |
| textContent += '<tspan dy="' + (i > 0 ? 1 : 0) + 'em" x="0" class="v-line">' + lines[i] + '</tspan>'; | |
| } | |
| return textContent; | |
| } | |
| /** | |
| * wrapText(): Wrap texts in SVG using joint.util.breakText | |
| * | |
| * @param Array size | |
| * @param Bool isOnPaper | |
| */ | |
| function wrapText(size, isOnPaper) { | |
| var xForms, xFormTranslate; | |
| if (isOnPaper) { | |
| // Yes, it's on the paper | |
| $('.wrap').each(function () { | |
| var $this = $(this), | |
| klass = $this.attr('class'), | |
| model = graph.getCell( $this.parents('[model-id]').attr('model-id') ), | |
| modelText = model && model.attributes.textRaw; | |
| if (!model) return; | |
| model.attributes.attrs.text.text = modelText; | |
| $this.html(getWordWrapText(modelText, { width: size[0] })) | |
| .attr('fill', '#000') | |
| .attr('font-size', '14'); | |
| this.transform.translate(null, size[1] / 2); | |
| $this.attr('class', klass.replace(/\bwrap\b/, '')); | |
| }); | |
| } else { | |
| // No, it's not on the paper. Then it must be on the STENCIL. No doubt!! | |
| $('.wrap').each(function () { | |
| var $this = $(this), | |
| klass = $this.attr('class'); | |
| this.transform.translate(null, size[1] + shapeTextMarginTop); | |
| $this.attr('class', klass.replace(/\bwrap\b/, '')); | |
| }); | |
| } | |
| } | |
| /** | |
| * logStatus() | |
| * | |
| * @param Any data | |
| * @param Number duration milliseconds | |
| */ | |
| function logStatus(data, duration) { | |
| clearTimeout(logTimerId); | |
| $statusBarEl.html(data); | |
| if (duration) { | |
| logTimerId = setTimeout(function () { $statusBarEl.html(''); }, duration); | |
| } | |
| } | |
| /** | |
| * saveGraphAsync() | |
| * | |
| */ | |
| function saveGraphAsync() { | |
| var workflow_draft = graph.toJSON(); | |
| workflow_draft.options = { | |
| paperWidth: defaultPaperSize[0], | |
| paperHeight: defaultPaperSize[1], | |
| paperOrigin: {x: 0, y: 0} | |
| }; | |
| if (paper && paperScroller) { | |
| workflow_draft.options = { | |
| paperWidth: paper.options.width / paperScroller._sx, | |
| paperHeight: paper.options.height / paperScroller._sy, | |
| paperOrigin: { | |
| x: paper.options.origin.x / paperScroller._sx, | |
| y: paper.options.origin.y / paperScroller._sy | |
| } | |
| } | |
| } | |
| return $.ajax({ | |
| url: gOptions.save_to, | |
| type: 'post', | |
| dataType: 'json', | |
| data: { | |
| id: gOptions.workflow_id, | |
| workflow_draft: JSON.stringify(workflow_draft) | |
| } | |
| }); | |
| } | |
| /** | |
| * saveWorkflow() | |
| * | |
| * @param Function doneCallback | |
| */ | |
| function saveWorkflow(doneCallback) { | |
| logStatus('<div class="alert alert-info">Saving...</div>'); | |
| saveGraphAsync().then(function(data) { | |
| configErrors = data.errors; | |
| if (data.isValid) { | |
| logStatus('<div class="alert alert-success">The workflow has been saved!</div>', 5000); | |
| } else { | |
| logStatus('<div class="alert alert-warning">The workflow has been saved, BUT we have invalid config.</div>'); | |
| } | |
| if (doneCallback) { | |
| doneCallback(data); | |
| } | |
| }, function(jqXHR, textStatus, errorThrown) { | |
| logStatus('<div class="alert alert-danger">Uh oh, seems we lost connection. The workflow could not be saved.</div>'); | |
| }); | |
| } | |
| /** | |
| * cronSaveWorkflow() | |
| */ | |
| function cronSaveWorkflow() { | |
| if (graphHasChanged) { | |
| saveWorkflow(); | |
| graphHasChanged = false; | |
| } | |
| } | |
| /** | |
| * isAgent() | |
| * | |
| * @param joint.dia.ElementView|joint.dia.Element cell | |
| * @return Bool | |
| */ | |
| function isAgent(cell) { | |
| if (cell.model) { | |
| return (cell.model.attributes.config != undefined); | |
| } else { | |
| return (cell.attributes.config != undefined); | |
| } | |
| } | |
| /** | |
| * validateConnection() | |
| * | |
| * @param joint.dia.ElementView cellViewS, | |
| * @param magnetS | |
| * @param joint.dia.ElementView cellViewT, | |
| * @param magnetT | |
| * @param String end | |
| * @param joint.dia.LinkView linkView | |
| */ | |
| function validateConnection(cellViewS, magnetS, cellViewT, magnetT, end, linkView) { | |
| if (isAgent(cellViewT) && cellViewS.model.id != cellViewT.model.id && | |
| cellViewT.model.attributes.cannot_receive_events === false) { | |
| return true; | |
| } | |
| return false; | |
| } | |
| /** | |
| * initializePaper(): Create a graph, paper and wrap the paper in a PaperScroller. | |
| */ | |
| function initializePaper() { | |
| var workflow_draft = gOptions.workflow_draft, | |
| wfdOptions = workflow_draft.options, | |
| $paperContainer = $('#paper'); | |
| graph = new joint.dia.Graph; | |
| paper = new joint.dia.Paper({ | |
| width: defaultPaperSize[0], | |
| height: defaultPaperSize[1], | |
| gridSize: 10, | |
| model: graph, | |
| async: { batchSize: 1 }, | |
| defaultLink: new joint.dia.Link({ | |
| attrs: { | |
| '.connection': { stroke: '#8d8d94', 'stroke-width': 3 }, | |
| '.marker-target': { stroke: '#8d8d94', fill: '#272634', d: 'M5.5,15.499,15.8,21.447,15.8,9.552z' }, | |
| '.marker-source': { stroke: '#8d8d94', fill: '#272634' } | |
| } | |
| }), | |
| linkConnectionPoint: joint.util.shapePerimeterConnectionPoint, | |
| validateConnection: validateConnection | |
| }); | |
| paperScroller = new joint.ui.PaperScroller({ | |
| paper: paper, | |
| autoResizePaper: true, | |
| padding: 50, | |
| baseWidth: 200, | |
| baseHeight: 200 | |
| }); | |
| $paperContainer.append(paperScroller.$el); | |
| paper.on('render:done', function() { | |
| var cells = graph.getElements(), | |
| cellView, attributes, | |
| agentsConfigByType = gOptions.agentsConfigByType, | |
| updatedAttrs; | |
| if (wfdOptions) { | |
| if (wfdOptions.paperOrigin) { | |
| paper.setOrigin(workflow_draft.options.paperOrigin.x, workflow_draft.options.paperOrigin.y); | |
| nav.updatePaper(paper.options.width, paper.options.height); | |
| } | |
| if (wfdOptions.paperWidth && wfdOptions.paperHeight) { | |
| paper.setDimensions(wfdOptions.paperWidth, wfdOptions.paperHeight); | |
| } | |
| } | |
| wrapText(shapeBigSize, true); | |
| for (var i = 0, n = cells.length; i < n; i ++) { | |
| if (!isAgent(cells[i])) continue; | |
| attributes = cells[i].attributes; | |
| cellView = paper.findViewByModel(cells[i]); | |
| checkAgentIfConfiguredAndDoIt(cellView); | |
| initAgentSubtitle(cellView); | |
| if (attributes.is_goal) { | |
| showGoalMarker(cellView); | |
| } | |
| updatedAttrs = agentsConfigByType[attributes.config.type] && agentsConfigByType[attributes.config.type].attrs; | |
| if (updatedAttrs) { | |
| attributes.config = agentsConfigByType[attributes.config.type].config; | |
| if (updatedAttrs.rect) { | |
| cells[i].attr('rect/fill', updatedAttrs.rect.fill); | |
| cells[i].attr('rect/stroke', updatedAttrs.rect.stroke); | |
| } else if (updatedAttrs.circle) { | |
| cells[i].attr('circle/fill', updatedAttrs.circle.fill); | |
| cells[i].attr('circle/stroke', updatedAttrs.circle.stroke); | |
| } else if (updatedAttrs.path) { | |
| cells[i].attr('path/fill', updatedAttrs.path.fill); | |
| cells[i].attr('path/stroke', updatedAttrs.path.stroke); | |
| } | |
| cells[i].attr('image/xlink:href', updatedAttrs.image['xlink:href']); | |
| } | |
| } | |
| if (!gOptions.wf_valid) { | |
| logStatus('<div class="alert alert-warning">Some agents have invalid configuration.</div>'); | |
| } | |
| configErrors = gOptions.wf_errors; | |
| paperScroller.zoomToFit({ | |
| padding: 100, | |
| maxScaleX: 1, | |
| maxScaleY: 1 | |
| }); | |
| setInterval(cronSaveWorkflow, wfAutoSaveInterval); | |
| }); | |
| paper.on('cell:pointerdown', function(cellView) { | |
| if (isAgent(cellView)) { | |
| agentDownFlag = true; | |
| } | |
| }); | |
| paper.on('cell:pointerup', function() { | |
| agentDownFlag = false; | |
| }); | |
| paper.on('blank:pointerdown', paperScroller.startPanning); | |
| $paperContainer.on('contextmenu', function(e) { e.preventDefault(); }); | |
| snaplines = new joint.ui.Snaplines({ paper: paper }); | |
| snaplines.startListening(); | |
| graph.on('add', function(cell) { | |
| if (isAgent(cell)) { | |
| var $image = $('image', '[model-id=' + cell.id + ']'), | |
| image = cell.attr('image'), | |
| newX = (shapeBigSize[0] - iconSize[0]) / 2, | |
| newY = shapeBigSize[1] / 2 - iconSize[1] - 10; | |
| cell.set('size', { width: shapeBigSize[0], height: shapeBigSize[1] }); | |
| $image.attr('x', newX).attr('y', newY); | |
| image.x = newX; | |
| image.y = newY; | |
| wrapText(shapeBigSize, true); | |
| initializeDefaultConfigValues(cell); | |
| } | |
| }); | |
| if (workflow_draft.cells && workflow_draft.cells.length > 0) { | |
| graph.fromJSON(workflow_draft); | |
| } else { | |
| setInterval(cronSaveWorkflow, wfAutoSaveInterval); | |
| } | |
| graph.on('change', function(cell) { | |
| if (cell.id != chart.id && linkDownFlag == false) { | |
| graphHasChanged = true; | |
| } | |
| }); | |
| graph.on('remove', function(cell) { | |
| if (cell instanceof joint.dia.Link && cell.get('target').id) { | |
| var link = cell, | |
| source = graph.getCell(link.get('source').id), | |
| sourceAttr = source.attributes, | |
| groupKey = link.attributes.groupKey; | |
| if (groupKey) { | |
| sourceAttr.targetIds[groupKey] = _.without(sourceAttr.targetIds[groupKey], link.get('target').id); | |
| } else { | |
| sourceAttr.targetIds = _.without(sourceAttr.targetIds, link.get('target').id); | |
| } | |
| } | |
| if (cell.id != chart.id && linkDownFlag == false) { | |
| graphHasChanged = true; | |
| } | |
| }); | |
| paper.on('scale resize', function() { | |
| for (var modelId in halo2) { | |
| halo2[modelId].update(); | |
| } | |
| }); | |
| } | |
| /** | |
| * createCustomShapes(): Create custom elements. | |
| */ | |
| function createCustomShapes() { | |
| joint.shapes.custom = {}; | |
| joint.shapes.custom.CircleEx = joint.shapes.basic.Circle.extend({ | |
| markup: '<g class="scalable"><circle/></g><image/><text/>', | |
| defaults: joint.util.deepSupplement({ | |
| type: 'custom.CircleEx' | |
| }, joint.shapes.basic.Circle.prototype.defaults) | |
| }); | |
| joint.shapes.custom.RectEx = joint.shapes.basic.Rect.extend({ | |
| markup: '<g class="scalable"><rect/></g><image/><text/>', | |
| defaults: joint.util.deepSupplement({ | |
| type: 'custom.RectEx' | |
| }, joint.shapes.basic.Rect.prototype.defaults) | |
| }); | |
| joint.shapes.custom.PathEx = joint.shapes.basic.Path.extend({ | |
| markup: '<g class="scalable"><path/></g><image/><text/>', | |
| defaults: joint.util.deepSupplement({ | |
| type: 'custom.PathEx' | |
| }, joint.shapes.basic.Path.prototype.defaults) | |
| }); | |
| } | |
| /** | |
| * initializeStencil(): Create and populate stencil. | |
| * | |
| * @param Object stencilData | |
| */ | |
| function initializeStencil(stencilData) { | |
| stencil = new joint.ui.Stencil({ | |
| paper: paperScroller, | |
| width: (shapeSize[0] + shapeMargin[0]) * 3, | |
| groups: stencilData.groups, | |
| dropAnimation: { duration: 200, easing: 'swing' } | |
| }); | |
| $('#stencil').append(stencil.render().el) | |
| .find('.content').mCustomScrollbar({ theme: 'minimal-dark' }); | |
| stencil.$el.on('contextmenu', function(e) { e.preventDefault(); }); | |
| $('.stencil-paper-drag').on('contextmenu', function(e) { e.preventDefault(); }); | |
| var layoutOptions = { | |
| columns: 3, | |
| columnWidth: shapeSize[0] + shapeMargin[0], | |
| rowHeight: shapeSize[1] * 2 + shapeTextMarginTop | |
| }, | |
| shapes = stencilData.shapes; | |
| _.each(stencilData.groups, function(group, name) { | |
| stencil.load(shapes[name], name); | |
| joint.layout.GridLayout.layout(stencil.getGraph(name), layoutOptions); | |
| stencil.getPaper(name).fitToContent(1, 1, 30); | |
| }); | |
| wrapText(shapeSize); | |
| } | |
| /** | |
| * initializeHaloAndInspector() | |
| */ | |
| function initializeHaloAndInspector() { | |
| paper.on('cell:mouseover', function(cellView) { | |
| if (linkDownFlag || agentDownFlag) return; | |
| // We don't want a Halo for links. | |
| if (cellView.model instanceof joint.dia.Link || !isAgent(cellView)) return; | |
| var model = cellView.model, | |
| modelId = model.id, | |
| attributes = model.attributes, | |
| configHandleOptions; | |
| showAgentSubtitle(cellView, false); | |
| if (halo && halo.options.cellView === cellView) return; | |
| // Removes the sticky halo | |
| if (halo2[modelId]) { | |
| halo2[modelId].remove(); | |
| delete halo2[modelId]; | |
| } | |
| halo = new joint.ui.Halo({ cellView: cellView }); | |
| halo.removeHandle('clone'); | |
| halo.removeHandle('resize'); | |
| halo.removeHandle('rotate'); | |
| halo.removeHandle('fork'); | |
| if (attributes.cannot_create_events) { | |
| halo.removeHandle('link'); | |
| } | |
| configHandleOptions = { | |
| name: 'config', | |
| position: 'n' | |
| }; | |
| if (isAgentConfigured(cellView)) { | |
| configHandleOptions.icon = configIconGreen; | |
| } else { | |
| configHandleOptions.icon = configIcon; | |
| configHandleOptions.attrs = { | |
| '.config': { | |
| 'data-toggle': 'tooltip', | |
| 'data-placement': 'top', | |
| 'title': configErrors[modelId] ? configErrors[modelId].join('\n') : 'Invalid config' | |
| } | |
| }; | |
| } | |
| halo.addHandle(configHandleOptions); | |
| halo.render(); | |
| $('.handle.config[data-toggle="tooltip"]').tooltip(); | |
| halo.on('action:link:add', function(link) { | |
| var sourceId = link.get('source').id, | |
| targetId = link.get('target').id, | |
| source = graph.getCell(sourceId), | |
| sourceAttr = source.attributes, | |
| sourceType = sourceAttr.config.type, | |
| activeSliceIndex = chart.prop('active-slice'); | |
| linkDownFlag = false; | |
| chart.remove(); | |
| if ( !sourceId || !targetId || | |
| (sourceType == 'Split' && activeSliceIndex === undefined) || | |
| (sourceType == 'Balance' && activeSliceIndex === undefined) || | |
| (sourceType == 'Send Email' && activeSliceIndex === undefined) || | |
| (sourceType == 'Twitter Status' && activeSliceIndex === undefined)) { | |
| link.remove(); | |
| } else { | |
| if (sourceType == 'Split' || sourceType == 'Balance' || sourceType == 'Send Email' || | |
| sourceType == 'Twitter Status' || sourceType == 'Twitter Message') { | |
| var chartData = chart.get('series')[0].data[activeSliceIndex], | |
| groupKey = chartData.groupKey; | |
| link.label(0, { position: 0.5, attrs: { text: { text: chartData.label } } }); | |
| link.attributes.groupKey = groupKey; | |
| if (!sourceAttr.targetIds) { | |
| sourceAttr.targetIds = {}; | |
| } | |
| if (!sourceAttr.targetIds[groupKey]) { | |
| sourceAttr.targetIds[groupKey] = []; | |
| } | |
| sourceAttr.targetIds[groupKey].push(targetId); | |
| } | |
| else { | |
| if (!sourceAttr.targetIds) { | |
| sourceAttr.targetIds = []; | |
| } | |
| sourceAttr.targetIds.push(targetId); | |
| } | |
| graph.trigger('change', link); | |
| } | |
| chart.prop('active-slice', undefined); | |
| if (halo2[sourceId] === undefined) { | |
| halo && (halo.remove(), halo = undefined); | |
| checkAgentIfConfiguredAndDoIt(paper.findViewByModel(source)); | |
| } | |
| if (targetId) { | |
| paper.trigger('cell:mouseover', paper.findViewByModel(graph.getCell(targetId))); | |
| } | |
| }); | |
| halo.on('action:config:pointerdown', function(e) { | |
| e.stopPropagation(); | |
| if (model instanceof joint.dia.Link) return; | |
| createInspector(cellView); | |
| }); | |
| halo.on('action:link:pointerdown', function(e) { | |
| var bbox = model.getBBox(), | |
| pieValue, chartView, $chartViewDataEl; | |
| linkDownFlag = true; | |
| // Display pie chart for Split and Balance agents. | |
| if (attributes.config.type === 'Split') { | |
| var criteria = attributes.criteriaUI, | |
| serieData = []; | |
| if (criteria && criteria.length > 0) { | |
| chart.prop('position', { x: bbox.x + bbox.width - chartPieRadius, y: bbox.y + bbox.height / 2 - chartPieRadius }); | |
| for (var i = 0, n = criteria.length, v = parseFloat((100 / n).toFixed(2)); i < n; i ++) { | |
| serieData.push({ | |
| value: v, | |
| label: criteria[i].weight_label + ' (' + criteria[i].weight + ')', | |
| fill: chartSerieFillColor[i], | |
| groupKey: criteria[i].weight_label.toString() | |
| }); | |
| } | |
| chart.prop('series', [{ data: serieData }]); | |
| chart.prop('active-slice', undefined); | |
| graph.addCell(chart); | |
| chart.toFront(); | |
| chartView = paper.findViewByModel(chart); | |
| $chartViewDataEl = chartView.$('.data'); | |
| chartView.$('.slice-inner-label').each(function () { | |
| var $textEl = $(this); | |
| $textEl.attr('data-slice', $textEl.parents('.slice').attr('data-slice')); | |
| $textEl.appendTo($chartViewDataEl); | |
| }); | |
| chartView.on('mouseover', function(slice, e) { | |
| var $slice = $('.slice[data-serie="' + slice.serieIndex + '"][data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl), | |
| elSlice = V($slice[0]); | |
| // _.each(this.$('.slice'), function(slice) { V(slice).scale(1); $('.slice-inner-label', $(slice)).css('display', 'none'); }); | |
| _.each($('.slice', $chartViewDataEl), function(slice) { V(slice).scale(1); }); | |
| $('.slice-inner-label', $chartViewDataEl).css('display', 'none'); | |
| elSlice.scale(1.2); | |
| chart.prop('active-slice', slice.sliceIndex); | |
| // $('.slice-inner-label', $slice).css('display', 'block'); | |
| $('.slice-inner-label[data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl).css('display', 'block'); | |
| }); | |
| } | |
| } | |
| else if (attributes.config.type === 'Balance') { | |
| chart.prop('position', { x: bbox.x + bbox.width - chartPieRadius, y: bbox.y + bbox.height / 2 - chartPieRadius }); | |
| chart.prop('series', [{ data: [ | |
| { value: 50, label: '50', fill: chartSerieFillColor[0], groupKey: '50' }, | |
| { value: 50, label: '50', fill: chartSerieFillColor[1], groupKey: '50' } | |
| ] }]); | |
| chart.prop('active-slice', undefined); | |
| graph.addCell(chart); | |
| chart.toFront(); | |
| chartView = paper.findViewByModel(chart); | |
| $chartViewDataEl = chartView.$('.data'); | |
| chartView.$('.slice-inner-label').each(function () { | |
| var $textEl = $(this); | |
| $textEl.attr('data-slice', $textEl.parents('.slice').attr('data-slice')); | |
| $textEl.appendTo($chartViewDataEl); | |
| }); | |
| chartView.on('mouseover', function(slice, e) { | |
| var $slice = $('.slice[data-serie="' + slice.serieIndex + '"][data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl), | |
| elSlice = V($slice[0]); | |
| // _.each(this.$('.slice'), function(slice) { V(slice).scale(1); $('.slice-inner-label', $(slice)).css('display', 'none'); }); | |
| _.each($('.slice', $chartViewDataEl), function(slice) { V(slice).scale(1); }); | |
| $('.slice-inner-label', $chartViewDataEl).css('display', 'none'); | |
| elSlice.scale(1.2); | |
| chart.prop('active-slice', slice.sliceIndex); | |
| // $('.slice-inner-label', $slice).css('display', 'block'); | |
| $('.slice-inner-label[data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl).css('display', 'block'); | |
| }); | |
| } | |
| else if (attributes.config.type === 'Send Email') { | |
| chart.prop('position', { x: bbox.x + bbox.width - chartPieRadius, y: bbox.y + bbox.height / 2 - chartPieRadius }); | |
| pieValue = 100 / 7; | |
| chart.prop('series', [{ data: [ | |
| { value: pieValue, label: 'On Bounce', fill: chartSerieFillColor[0], groupKey: 'on_bounce' }, | |
| { value: pieValue, label: 'On Click', fill: chartSerieFillColor[1], groupKey: 'on_click' }, | |
| { value: pieValue, label: 'On Open', fill: chartSerieFillColor[2], groupKey: 'on_open' }, | |
| { value: pieValue, label: 'On Unsubscribe', fill: chartSerieFillColor[3], groupKey: 'on_unsubscribe' }, | |
| { value: pieValue, label: 'On Reply', fill: chartSerieFillColor[4], groupKey: 'on_reply' }, | |
| { value: pieValue, label: 'On Send', fill: chartSerieFillColor[5], groupKey: 'on_send' }, | |
| { value: pieValue, label: 'On No Response', fill: chartSerieFillColor[6], groupKey: 'on_no_response' } | |
| ] }]); | |
| chart.prop('active-slice', undefined); | |
| graph.addCell(chart); | |
| chart.toFront(); | |
| chartView = paper.findViewByModel(chart); | |
| $chartViewDataEl = chartView.$('.data'); | |
| chartView.$('.slice-inner-label').each(function () { | |
| var $textEl = $(this); | |
| $textEl.attr('data-slice', $textEl.parents('.slice').attr('data-slice')); | |
| $textEl.appendTo($chartViewDataEl); | |
| }); | |
| chartView.on('mouseover', function(slice, e) { | |
| var $slice = $('.slice[data-serie="' + slice.serieIndex + '"][data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl), | |
| elSlice = V($slice[0]); | |
| // _.each(this.$('.slice'), function(slice) { V(slice).scale(1); $('.slice-inner-label', $(slice)).css('display', 'none'); }); | |
| _.each($('.slice', $chartViewDataEl), function(slice) { V(slice).scale(1); }); | |
| $('.slice-inner-label', $chartViewDataEl).css('display', 'none'); | |
| elSlice.scale(1.2); | |
| chart.prop('active-slice', slice.sliceIndex); | |
| // $('.slice-inner-label', $slice).css('display', 'block'); | |
| $('.slice-inner-label[data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl).css('display', 'block'); | |
| }); | |
| } | |
| else if (attributes.config.type === 'Twitter Status') { | |
| chart.prop('position', { x: bbox.x + bbox.width - chartPieRadius, y: bbox.y + bbox.height / 2 - chartPieRadius }); | |
| pieValue = 100 / 4; | |
| chart.prop('series', [{ data: [ | |
| { value: pieValue, label: 'Following Us', fill: chartSerieFillColor[0], groupKey: 'following_us' }, | |
| { value: pieValue, label: 'Not Following Us', fill: chartSerieFillColor[1], groupKey: 'not_following_us' }, | |
| { value: pieValue, label: 'Blocking Us', fill: chartSerieFillColor[2], groupKey: 'blocking_us' }, | |
| { value: pieValue, label: 'Muted By Us', fill: chartSerieFillColor[3], groupKey: 'muted_by_us' } | |
| ] }]); | |
| chart.prop('active-slice', undefined); | |
| graph.addCell(chart); | |
| chart.toFront(); | |
| chartView = paper.findViewByModel(chart); | |
| $chartViewDataEl = chartView.$('.data'); | |
| chartView.$('.slice-inner-label').each(function () { | |
| var $textEl = $(this); | |
| $textEl.attr('data-slice', $textEl.parents('.slice').attr('data-slice')); | |
| $textEl.appendTo($chartViewDataEl); | |
| }); | |
| chartView.on('mouseover', function(slice, e) { | |
| var $slice = $('.slice[data-serie="' + slice.serieIndex + '"][data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl), | |
| elSlice = V($slice[0]); | |
| // _.each(this.$('.slice'), function(slice) { V(slice).scale(1); $('.slice-inner-label', $(slice)).css('display', 'none'); }); | |
| _.each($('.slice', $chartViewDataEl), function(slice) { V(slice).scale(1); }); | |
| $('.slice-inner-label', $chartViewDataEl).css('display', 'none'); | |
| elSlice.scale(1.2); | |
| chart.prop('active-slice', slice.sliceIndex); | |
| // $('.slice-inner-label', $slice).css('display', 'block'); | |
| $('.slice-inner-label[data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl).css('display', 'block'); | |
| }); | |
| } | |
| else if (attributes.config.type === 'Twitter Message') { | |
| chart.prop('position', { x: bbox.x + bbox.width - chartPieRadius, y: bbox.y + bbox.height / 2 - chartPieRadius }); | |
| pieValue = 100 / 5; | |
| chart.prop('series', [{ data: [ | |
| { value: pieValue, label: '@Mentions Us', fill: chartSerieFillColor[0], groupKey: 'mentions_us' }, | |
| { value: pieValue, label: 'DMs Us', fill: chartSerieFillColor[1], groupKey: 'dms_us' }, | |
| { value: pieValue, label: 'Blocking Us', fill: chartSerieFillColor[2], groupKey: 'blocking_us' }, | |
| { value: pieValue, label: 'On Send', fill: chartSerieFillColor[3], groupKey: 'on_send' }, | |
| { value: pieValue, label: 'On No Response', fill: chartSerieFillColor[4], groupKey: 'on_no_response' } | |
| ] }]); | |
| chart.prop('active-slice', undefined); | |
| graph.addCell(chart); | |
| chart.toFront(); | |
| chartView = paper.findViewByModel(chart); | |
| $chartViewDataEl = chartView.$('.data'); | |
| chartView.$('.slice-inner-label').each(function () { | |
| var $textEl = $(this); | |
| $textEl.attr('data-slice', $textEl.parents('.slice').attr('data-slice')); | |
| $textEl.appendTo($chartViewDataEl); | |
| }); | |
| chartView.on('mouseover', function(slice, e) { | |
| var $slice = $('.slice[data-serie="' + slice.serieIndex + '"][data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl), | |
| elSlice = V($slice[0]); | |
| // _.each(this.$('.slice'), function(slice) { V(slice).scale(1); $('.slice-inner-label', $(slice)).css('display', 'none'); }); | |
| _.each($('.slice', $chartViewDataEl), function(slice) { V(slice).scale(1); }); | |
| $('.slice-inner-label', $chartViewDataEl).css('display', 'none'); | |
| elSlice.scale(1.2); | |
| chart.prop('active-slice', slice.sliceIndex); | |
| // $('.slice-inner-label', $slice).css('display', 'block'); | |
| $('.slice-inner-label[data-slice="' + slice.sliceIndex + '"]', $chartViewDataEl).css('display', 'block'); | |
| }); | |
| } | |
| }); | |
| }); | |
| paper.on('cell:mouseout', function(cellView, e) { | |
| if (linkDownFlag || agentDownFlag || !isAgent(cellView)) return; | |
| var $te = $(e.relatedTarget); | |
| if (($te.hasClass('handle') && ($te.parents('.halo2').length == 0)) || | |
| $te.parents('[model-id=' + cellView.model.id + ']').length > 0) return; | |
| if (halo) { | |
| halo.remove(); | |
| halo = undefined; | |
| checkAgentIfConfiguredAndDoIt(cellView); | |
| } | |
| showAgentSubtitle(cellView); | |
| }); | |
| paper.$el.on('mouseenter', '.halo2 .handle.config', function() { | |
| var modelId = $(this).parents('.halo2').attr('data-model-id'); | |
| paper.trigger('cell:mouseover', paper.findViewByModel(paper.getModelById(modelId))); | |
| }); | |
| paper.$el.on('mouseleave', '.halo .handle', function(e) { | |
| if (!halo || linkDownFlag || agentDownFlag) return; | |
| var $te = $(e.relatedTarget), | |
| cellView = halo.options.cellView; | |
| if ($te.parents('[model-id=' + cellView.model.id + ']').length > 0 || | |
| ($te.hasClass('handle') && ($te.parents('.halo2').length == 0))) return; | |
| checkAgentIfConfiguredAndDoIt(cellView); | |
| halo.remove(); | |
| halo = undefined; | |
| showAgentSubtitle(cellView); | |
| }); | |
| } | |
| /** | |
| * createInspector() | |
| * | |
| * @param joint.dia.ElementView cellView | |
| */ | |
| function createInspector(cellView) { | |
| // No need to re-render inspector if the cellView didn't change. | |
| // ------------------------------------------------------------- | |
| if (!inspector || inspector.options.cellView !== cellView) { | |
| if (inspector) { | |
| // Set unsaved changes to the model and clean up the old inspector if there was one. | |
| inspector.updateCell(); | |
| inspector.remove(); | |
| } | |
| var $dlgEl = $('#inspector_dialog'), | |
| attributes = cellView.model.attributes, | |
| config = attributes.config, | |
| agentType = config.type, | |
| addServiceURL = '', | |
| fields = config.fields, | |
| inputs = { | |
| is_goal: { type: 'toggle', label: 'Set this as goal' } | |
| }, | |
| $isGoalDivEl; | |
| for (var i = 0, n = fields.length; i < n; i ++) { | |
| inputs[fields[i]] = {}; | |
| } | |
| initializeInspectorInputs(inputs, attributes); | |
| inspector = new joint.ui.Inspector({ | |
| cellView: cellView, | |
| inputs: inputs, | |
| live: false | |
| }); | |
| $('[data-field="schedule"]', '#inspector_dialog_header').remove(); | |
| $dlgEl.find('.inspector-container').html(inspector.render().el); | |
| $dlgEl.find('.modal-title') | |
| .html(attributes.attrs.text.text) | |
| .after($('[data-field="schedule"]', '.inspector').addClass('pull-right')); | |
| $dlgEl.find('.description').html( config.description || '' ); | |
| $dlgEl.find('.btn-list-add').addClass('btn btn-xs'); | |
| $dlgEl.find('.btn-list-del').addClass('btn btn-xs pull-right').html('×'); | |
| if (agentHasField(cellView, 'segment_id')) { | |
| $dlgEl.find('[data-field="segment_id"]').append('<a href="' + gOptions.add_segment_url + '" class="btn btn-xs btn-primary">Add New Segment</a>'); | |
| } | |
| if (agentHasField(cellView, 'service_id')) { | |
| switch (agentType) { | |
| case 'Facebook Update': | |
| case 'Twitter Action': | |
| case 'Twitter Message': | |
| case 'Twitter Status': | |
| addServiceURL = gOptions.add_twitter_service_url; break; | |
| case 'Send Email': | |
| addServiceURL = gOptions.add_email_service_url; break; | |
| } | |
| $dlgEl.find('[data-field="service_id"]').append('<a href="' + addServiceURL + '" class="btn btn-xs btn-primary">Add New Service</a>'); | |
| } | |
| if (agentHasField(cellView, 'message_template_id')) { | |
| $dlgEl.find('[data-field="message_template_id"]').append('<a href="' + gOptions.add_message_template_url + '" class="btn btn-xs btn-primary">Add New Message Template</a>'); | |
| } | |
| if (agentHasField(cellView, 'campaign_list_id')) { | |
| $dlgEl.find('[data-field="campaign_list_id"]').append('<a href="' + gOptions.add_campaign_list_url + '" class="btn btn-xs btn-primary">Add New Campaign List</a>'); | |
| } | |
| if (agentType == 'Twitter Action') { | |
| var actionValue = $dlgEl.find('[data-attribute="criteria/action"]').val(); | |
| $dlgEl.find('[data-field="criteria/list_name"]').toggle(actionValue == 'list' || actionValue == 'unlist'); | |
| } | |
| $dlgEl.find('select > option[value=""]').prop('disabled', true); | |
| $dlgEl.find('.modal-footer').find('[data-field="is_goal"]').remove(); | |
| var dataFieldId, $dataFieldDivEl, $inputEl; | |
| $dlgEl.find('[data-type="toggle"]').each(function () { | |
| $inputEl = $(this); | |
| $dataFieldDivEl = $inputEl.parents('[data-field]'); | |
| dataFieldId = $dataFieldDivEl.attr('data-field'); | |
| if (dataFieldId === 'is_goal') { | |
| $dataFieldDivEl.prependTo($dlgEl.find('.modal-footer')); | |
| } | |
| $dataFieldDivEl | |
| .addClass('checkbox') | |
| .css('float', 'left') | |
| .css('margin-top', '5px'); | |
| $inputEl.attr('id', dataFieldId).insertBefore($('label', $dataFieldDivEl)); | |
| $('label', $dataFieldDivEl).attr('for', dataFieldId); | |
| $('div.toggle', $dataFieldDivEl).remove(); | |
| }); | |
| // $('.toggle', $isGoalDivEl).remove(); | |
| $dlgEl.modal('show'); | |
| } | |
| } | |
| /** | |
| * readableScheduleText() | |
| * | |
| * @param String text | |
| * @return String | |
| */ | |
| function readableScheduleText(text) { | |
| text = text.toString(); | |
| if (!text || text.trim().length === 0) return ''; | |
| var matches, | |
| timeUnitsSingular = { 'm': 'minute', 'h': 'hour', 'd': 'day' }, | |
| timeUnitsPlural = { 'm': 'minutes', 'h': 'hours', 'd': 'days' }; | |
| if (matches = text.match(/^every_(\d+)(.+)/)) { | |
| text = 'Every ' + ( parseInt(matches[1], 10) > 1 ? matches[1] + ' ' + timeUnitsPlural[matches[2]] : timeUnitsSingular[matches[2]] ); | |
| } | |
| else if (matches = text.match(/^(\d+)(.+)/)) { | |
| text = matches[1] + ' ' + matches[2].toUpperCase(); | |
| } | |
| else { | |
| text = text.charAt(0).toUpperCase() + text.slice(1); | |
| } | |
| return text; | |
| } | |
| /** | |
| * initializeDefaultConfigValues() | |
| * | |
| * @param joint.dia.Cell cell | |
| */ | |
| function initializeDefaultConfigValues(cell) { | |
| var attributes = cell.attributes, | |
| agentType = attributes.config.type; | |
| if (agentHasField(cell, 'workflow_id')) { | |
| attributes.workflow_id = gOptions.workflow_id; | |
| } | |
| if (agentType === 'Has Facebook' || | |
| agentType === 'Has Email' || | |
| agentType === 'Has Twitter') { | |
| if (agentHasField(cell, 'criteria')) { | |
| attributes.criteria = attributes.config.criteria; | |
| } | |
| } | |
| attributes.schedule = attributes.default_schedule; | |
| } | |
| /** | |
| * initializeInspectorInputs() | |
| * | |
| * @param Object inputs | |
| * @param Object attributes | |
| */ | |
| function initializeInspectorInputs(inputs, attributes) { | |
| var config = attributes.config, | |
| criteria = config.criteria, | |
| tmp, options, | |
| type = config.type; | |
| if (type.match(/Segment$/)) { | |
| type = type.match(/(.*) Segment$/)[1].toLowerCase(); | |
| } | |
| for (var field in inputs) { | |
| switch (field) { | |
| case 'segment_id': | |
| tmp = config.dropdowns.segments; | |
| options = [{ value: '', content: 'Choose segment..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i].value, content: tmp[i].name }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'Segments', options: options, index: 1 | |
| }; | |
| // Assume these agents are triggers and let's add another trigger-specific field | |
| inputs.only_new = { | |
| type: 'toggle', label: ['Only new ', type, ' that matches this segment'].join('') | |
| }; | |
| break; | |
| case 'schedule': | |
| tmp = config.dropdowns.schedule; | |
| options = []; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i], content: readableScheduleText(tmp[i]) }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'Schedule', options: options | |
| }; | |
| break; | |
| case 'service_id': | |
| tmp = config.dropdowns.services; | |
| if (config.type === 'Twitter Action' || config.type === 'Twitter Status' || config.type === 'Twitter Message') { | |
| tmp = config.dropdowns.twitter_services; | |
| } | |
| else if (config.type === 'Facebook Update') { | |
| tmp = config.dropdowns.facebook_services; | |
| } | |
| else if (config.type === 'Send Email') { | |
| tmp = config.dropdowns.email_services; | |
| } | |
| options = [{ value: '', content: 'Choose service..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i].value, content: tmp[i].name }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'Services', options: options, index: 1 | |
| }; | |
| break; | |
| case 'campaign_list_id': | |
| tmp = config.dropdowns.campaigns_list; | |
| options = [{ value: '', content: 'Choose campaign..' }]; | |
| for (var i in tmp) { | |
| options.push({ value: tmp[i].id, content: tmp[i].name }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'Campaigns', options: options, index: 1 | |
| }; | |
| options = [{ value: '', content: 'Choose stage..' }]; | |
| if (attributes.campaign_list_id) { | |
| tmp = config.dropdowns.campaigns_list[attributes.campaign_list_id].stages; | |
| for (var i in tmp) { | |
| options.push({ value: tmp[i].id, content: tmp[i].name }); | |
| } | |
| } | |
| inputs['stage_id'] = { | |
| type: 'select', label: 'Stages', options: options, index: 2 | |
| }; | |
| // Assume this is a campaign list agent and let's add another campaign-specific field | |
| inputs.only_new = { | |
| type: 'toggle', label: 'Only new list that matches this segment' | |
| }; | |
| break; | |
| case 'message_template_id': | |
| tmp = config.dropdowns.message_templates; | |
| options = [{ value: '', content: 'Choose message template..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i].message_template.id, content: tmp[i].message_template.name }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'Message Template', options: options | |
| }; | |
| break; | |
| case 'user_ids': | |
| tmp = config.dropdowns.users; | |
| options = [{ value: '', content: 'Choose a user..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i].user.id, content: tmp[i].user.name + ' (' + tmp[i].user.login + ')' }); | |
| } | |
| inputs[field] = { | |
| type: 'select', label: 'User', options: options | |
| }; | |
| break; | |
| case 'criteria': | |
| if (config.type == "Add Field" || config.type == "Modify Field") { | |
| inputs.criteria = { | |
| type: 'list', | |
| item: { | |
| type: 'object', | |
| properties: { | |
| field_name: { type: 'text', label: 'Field Name' }, | |
| field_value: { type: 'text', label: 'Field Value' } | |
| } | |
| }, | |
| label: 'Click the {+} button to add a field and value.', | |
| attrs: { | |
| label: { class: 'custom-label' } | |
| }, | |
| index: 99 | |
| }; | |
| } | |
| else if (config.type == 'Filter') { | |
| inputs.criteria = { type: 'text', label: 'Criteria', index: 99 }; | |
| } | |
| else if (config.type == 'Twitter Action') { | |
| inputs.criteria = { | |
| action: { | |
| type: 'select', | |
| label: 'Action', | |
| options: [ | |
| { value: '', content: 'Choose action..' }, | |
| { value: 'follow', content: 'Follow' }, | |
| { value: 'unfollow', content: 'Unfollow' }, | |
| { value: 'list', content: 'List' }, | |
| { value: 'unlist', content: 'Unlist' } | |
| ], | |
| index: 98 | |
| }, | |
| list_name: { type: 'text', label: 'List Name', index: 99 } | |
| }; | |
| } | |
| else if (config.type == "Split") { | |
| inputs.criteriaUI = { | |
| type: 'list', | |
| item: { | |
| type: 'object', | |
| properties: { | |
| weight_label: { type: 'text', label: 'Label' }, | |
| weight: { type: 'text', label: 'Weight' } | |
| } | |
| }, | |
| label: 'Click the {+} button to add a weight.', | |
| attrs: { | |
| label: { class: 'custom-label' } | |
| }, | |
| index: 99 | |
| }; | |
| } | |
| else if (config.type == 'Delay') { | |
| inputs.criteria = { | |
| length: { type: 'text', label: 'Length', attrs: { type: 'number' }, index: 98 }, | |
| unit: { | |
| type: 'select', | |
| label: 'Type', | |
| options: [ | |
| { value: '', content: 'Choose..' }, | |
| { value: 'minutes', content: 'Minutes' }, | |
| { value: 'hours', content: 'Hours' }, | |
| { value: 'days', content: 'Days' }, | |
| { value: 'weeks', content: 'Weeks' }, | |
| { value: 'months', content: 'Months' } | |
| ], | |
| index: 99 | |
| } | |
| }; | |
| } | |
| else if (config.type == 'Twitter Message') { | |
| tmp = config.dropdowns.message_type; | |
| options = [{ value: '', content: 'Choose message type..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i], content: tmp[i].replace(/\w/, function(match, capture) { return match.toUpperCase(); }) }); | |
| } | |
| inputs.criteria = { | |
| type: 'select', label: 'Message Type', options: options, index: 99 | |
| }; | |
| } | |
| else if (config.type == 'Limit') { | |
| inputs.criteria = { type: 'text', label: 'Limit', index: 99 }; | |
| } | |
| else if (config.type == "Update Field") { | |
| inputs.criteria = { | |
| type: 'list', | |
| item: { | |
| type: 'object', | |
| properties: { | |
| field_name: { type: 'text', label: 'Field Name' }, | |
| field_value: { type: 'text', label: 'Field Value' } | |
| } | |
| }, | |
| label: 'Click the {+} button to add a field and value.', | |
| attrs: { | |
| label: { class: 'custom-label' } | |
| }, | |
| index: 99 | |
| }; | |
| } | |
| else if (config.type == 'Has Profile') { | |
| tmp = config.dropdowns.services; | |
| options = [{ value: '', content: 'Choose..' }]; | |
| for (var i = 0, n = tmp.length; i < n; i ++) { | |
| options.push({ value: tmp[i], content: tmp[i].replace(/\w/, function(match, capture) { return match.toUpperCase(); }) }); | |
| } | |
| inputs.criteria = { type: 'select', label: 'Has Profile', options: options, index: 99 }; | |
| } | |
| else if (config.type == 'Modify Campaign List') { | |
| inputs.criteria = { | |
| type: 'select', | |
| label: 'Action', | |
| options: [ | |
| { value: '', content: 'Choose action..' }, | |
| { value: 'add', content: 'Add' }, | |
| { value: 'remove', content: 'Remove' } | |
| ], | |
| index: 99 | |
| }; | |
| } | |
| else { | |
| delete inputs.criteria; | |
| } | |
| break; | |
| default: | |
| } | |
| } | |
| } | |
| /** | |
| * agentHasField() | |
| * | |
| * @param joint.dia.ElementView|joint.dia.Cell cell | |
| * @param String fieldName | |
| * @return Bool | |
| */ | |
| function agentHasField(cell, fieldName) { | |
| var model = cell.model || cell, | |
| fields = model.attributes.config.fields; | |
| return (fields.indexOf(fieldName) >= 0); | |
| } | |
| /** | |
| * isAgentConfigured() | |
| * | |
| * @param joint.dia.ElementView cellView | |
| * @return Bool | |
| */ | |
| function isAgentConfigured(cellView) { | |
| if (!cellView || !cellView.model) return false; | |
| var attributes = cellView.model.attributes, | |
| agentType = attributes.config.type, | |
| data, k; | |
| if (agentHasField(cellView, 'criteria')) { | |
| data = attributes.criteria; | |
| if (!data) return false; | |
| if (data instanceof Array && data.length === 0) return false; | |
| if (agentType == 'Add Field' || agentType == 'Update Field') { | |
| for (var i = 0, n = data.length; i < n; i ++) { | |
| if (data[i].field_name.toString().match(/^\s*$/) || data[i].field_value.toString().match(/^\s*$/)) { | |
| return false; | |
| } | |
| } | |
| } else if (agentType == 'Filter') { | |
| if (data.toString().match(/^\s*$/)) return false; | |
| } else if (agentType == 'Twitter Action') { | |
| if (!data.action || data.action.toString().match(/^\s*$/)) return false; | |
| if (data.action.toString().match(/^(list|unlist)$/) && data.list_name.toString().match(/^\s*$/)) return false; | |
| } else if (agentType == 'Split') { | |
| data = attributes.criteriaUI; | |
| for (var i = 0, n = data.length; i < n; i ++) { | |
| if (data[i].weight_label.toString().match(/^\s*$/) || data[i].weight.toString().match(/^\s*$/)) { | |
| return false; | |
| } | |
| } | |
| } else if (agentType == 'Twitter Message') { | |
| if (data.toString().match(/^\s*$/)) return false; | |
| } else if (agentType == 'Delay') { | |
| if (data.length.toString().match(/^\s*$/) || !data.unit || data.unit.toString().match(/^\s*$/)) return false; | |
| } else if (agentType == 'Limit') { | |
| if (data.toString().match(/^\s*$/)) return false; | |
| } else if (agentType == 'Has Profile') { | |
| if (data.toString().match(/^\s*$/)) return false; | |
| } else { | |
| for (k in data) { | |
| if (data[k].toString().match(/^\s*$/)) { | |
| return false; | |
| } | |
| } | |
| } | |
| } | |
| if (agentHasField(cellView, 'segment_id')) { | |
| data = attributes.segment_id; | |
| if (!data) return false; | |
| } | |
| if (agentHasField(cellView, 'service_id')) { | |
| data = attributes.service_id; | |
| if (!data) return false; | |
| } | |
| if (agentHasField(cellView, 'message_template_id')) { | |
| data = attributes.message_template_id; | |
| if (!data) return false; | |
| } | |
| if (agentHasField(cellView, 'campaign_list_id')) { | |
| data = attributes.campaign_list_id; | |
| if (!data || !attributes.stage_id) return false; | |
| } | |
| if (agentHasField(cellView, 'user_ids')) { | |
| data = attributes.user_ids; | |
| if (!data) return false; | |
| } | |
| return true; | |
| } | |
| /** | |
| * checkAgentIfConfiguredAndDoIt() | |
| * | |
| * @param joint.dia.ElementView cellView | |
| */ | |
| function checkAgentIfConfiguredAndDoIt(cellView) { | |
| var modelId = cellView.model.id; | |
| if (isAgentConfigured(cellView)) | |
| { | |
| // In case agent is configured, remove config link. | |
| halo2[modelId] && halo2[modelId].remove(); | |
| delete halo2[modelId]; | |
| } | |
| else | |
| { | |
| // This case implies agent is not configured. | |
| if (halo2[modelId] === undefined) { | |
| var h = new Halo2({ cellView: cellView }); | |
| h.render(); | |
| halo2[modelId] = h; | |
| } | |
| } | |
| } | |
| /** | |
| * initializeNavigator() | |
| */ | |
| function initializeNavigator() { | |
| nav = new joint.ui.Navigator({ | |
| paperScroller: paperScroller, | |
| width: 200, | |
| height: 100, | |
| padding: 10, | |
| zoomOptions: { max: zoomMax, min: zoomMin } | |
| }); | |
| nav.$el.appendTo('#navigator'); | |
| nav.render(); | |
| $('#zoom-in').on('click', function() { | |
| paperScroller.zoom(zoomStep, { max: zoomMax }); | |
| }); | |
| $('#zoom-out').on('click', function() { | |
| paperScroller.zoom(-zoomStep, { min: zoomMin }); | |
| }); | |
| $('#toggle-panning').on('click', function() { | |
| }); | |
| } | |
| /** | |
| * clearGraph() | |
| */ | |
| function clearGraph() { | |
| for (var k in halo2) { | |
| halo2[k].remove(); | |
| delete halo2[k]; | |
| } | |
| graph.clear(); | |
| } | |
| /** | |
| * initAgentSubtitle() | |
| * | |
| * @param joint.dia.ElementView cellView | |
| */ | |
| function initAgentSubtitle(cellView) { | |
| var subtitle = cellView.model.attributes.subtitle, | |
| subtitleAlt = cellView.model.attributes.subtitleAlt, | |
| $cellEl = cellView.$el, | |
| subtitleTextNode = $cellEl.find('.agent-subtitle'); | |
| if (subtitle || subtitleAlt) { | |
| if (subtitleTextNode.length == 0) { | |
| subtitleTextNode = V('<text class="agent-subtitle" display="null" xml:space="preserve" font-size="14" text-anchor="middle" fill="#fff" y="0.8em" transform="matrix(1 0 0 1 0 0)"></text>').node; | |
| cellView.el.appendChild(subtitleTextNode); | |
| subtitleTextNode = $(subtitleTextNode); | |
| } | |
| subtitleTextNode.attr('data-normal-text', subtitle).attr('data-hover-text', subtitleAlt); | |
| if (subtitle) { | |
| showAgentSubtitle(cellView); | |
| } | |
| } | |
| else { | |
| if (subtitleTextNode.length > 0) { | |
| subtitleTextNode.remove(); | |
| } | |
| } | |
| } | |
| function showAgentSubtitle(cellView, showNormalText) { | |
| var $cellEl = cellView.$el, | |
| subtitleTextNode = $cellEl.find('.agent-subtitle'), | |
| subtitleNormalText = subtitleTextNode.attr('data-normal-text'), | |
| subtitleHoverText = subtitleTextNode.attr('data-hover-text'), | |
| $textTitleEl = $cellEl.find('text.title'), | |
| x, y; | |
| if (subtitleTextNode.length == 0) return; | |
| if (showNormalText === undefined) { | |
| showNormalText = true; | |
| } | |
| if (showNormalText) { | |
| $textTitleEl.show(); | |
| subtitleNormalText = subtitleNormalText || ""; | |
| x = shapeBigSize[0] / 2; | |
| y = shapeBigSize[1] / 2 + $textTitleEl[0].getBBox().height + 5; | |
| subtitleTextNode.html(getWordWrapText(subtitleNormalText, { width: shapeBigSize[0] }))[0].transform.translate(x, y); | |
| } | |
| else if (!showNormalText && subtitleHoverText !== undefined) { | |
| $textTitleEl.hide(); | |
| x = shapeBigSize[0] / 2; | |
| y = shapeBigSize[1] / 2; | |
| subtitleTextNode.html(getWordWrapText(subtitleHoverText, { width: shapeBigSize[0] }))[0].transform.translate(x, y); | |
| } | |
| } | |
| /** | |
| * showGoalMarker() | |
| * | |
| * @param joint.dia.ElementView cellView | |
| */ | |
| function showGoalMarker(cellView) { | |
| cellView.el.appendChild($goalMarkerEl); | |
| } | |
| /** | |
| * initializeMisc() | |
| */ | |
| function initializeMisc() { | |
| $('#btn_clear').on('click', function() { | |
| clearGraph(); | |
| }); | |
| $('#btn_save').on('click', function() { | |
| saveWorkflow(); | |
| }); | |
| $('#btn_save_close').on('click', function() { | |
| saveWorkflow(function (data) { | |
| window.location.replace(gOptions.workflows_path); | |
| }); | |
| }); | |
| $('#btn_open_png').on('click', function() { | |
| paper.toPNG(function (dataURL) { | |
| window.open(dataURL); | |
| }); | |
| }); | |
| $('#inspector_dialog').on('hide.bs.modal', function() { | |
| if (inspector) { | |
| inspector.remove(); | |
| inspector = null; | |
| } | |
| }); | |
| $('#inspector_btn_save').on('click', function() { | |
| if (inspector) { | |
| inspector.updateCell(); | |
| var cellView = inspector.options.cellView, | |
| attributes = cellView.model.attributes, | |
| agentType = attributes.config.type, | |
| tmp; | |
| // Special process for Split agent | |
| if (agentType === 'Split') { | |
| attributes.criteria = {}; | |
| for (var i = 0, criteriaUI = attributes.criteriaUI, n = criteriaUI.length; i < n; i ++) { | |
| attributes.criteria[criteriaUI[i].weight_label] = criteriaUI[i].weight; | |
| } | |
| } | |
| checkAgentIfConfiguredAndDoIt(cellView); | |
| if (isAgentConfigured(cellView)) { | |
| if (agentHasField(cellView, 'segment_id')) { | |
| attributes.subtitle = $('select[data-attribute="segment_id"] option:selected', '#inspector_dialog').html(); | |
| } | |
| else { | |
| switch (agentType) { | |
| case 'Facebook Update': | |
| case 'Twitter Message': | |
| case 'Twitter Status': | |
| attributes.subtitle = $('select[data-attribute="service_id"] option:selected', '#inspector_dialog').html(); | |
| break; | |
| case 'Send Email': | |
| attributes.subtitle = $('select[data-attribute="message_template_id"] option:selected', '#inspector_dialog').html(); | |
| break; | |
| case 'Twitter Action': | |
| attributes.subtitle = $('select[data-attribute="criteria/action"] option:selected', '#inspector_dialog').html(); | |
| break; | |
| case 'Delay': | |
| attributes.subtitle = [attributes.criteria.length, $('select[data-attribute="criteria/unit"] option:selected', '#inspector_dialog').html()].join(' '); | |
| break; | |
| case 'Limit': | |
| attributes.subtitle = attributes.criteria.toString(); | |
| break; | |
| case 'Has Profile': | |
| attributes.subtitle = $('select[data-attribute="criteria"] option:selected', '#inspector_dialog').html(); | |
| break; | |
| case 'Split': | |
| attributes.subtitle = $('[data-attribute="criteriaUI"] .list-item', '#inspector_dialog').length.toString(); | |
| break; | |
| case 'Campaign List': | |
| attributes.subtitleAlt = [$('[data-attribute="campaign_list_id"] option:selected', '#inspector_dialog').html(), $('[data-attribute="stage_id"] option:selected', '#inspector_dialog').html()].join(': '); | |
| break; | |
| case 'Modify Campaign List': | |
| attributes.subtitle = $('select[data-attribute="criteria"] option:selected', '#inspector_dialog').html(); | |
| attributes.subtitleAlt = [$('[data-attribute="campaign_list_id"] option:selected', '#inspector_dialog').html(), $('[data-attribute="stage_id"] option:selected', '#inspector_dialog').html()].join(': '); | |
| break; | |
| default: | |
| } | |
| } | |
| } | |
| else { | |
| delete attributes.subtitle; | |
| } | |
| initAgentSubtitle(cellView); | |
| if (attributes.is_goal) { | |
| for (var i = 0, cells = graph.getCells(), n = cells.length; i < n; i ++) { | |
| if (cells[i] === cellView.model) continue; | |
| delete cells[i].attributes.is_goal; | |
| } | |
| showGoalMarker(cellView); | |
| } | |
| $('#inspector_dialog').modal('hide'); | |
| } | |
| }); | |
| $('#inspector_dialog').on('click', '.btn-list-add', function () { | |
| $('.btn-list-del', '#inspector_dialog').addClass('btn btn-xs pull-right').html('×'); | |
| if (inspector.options.cellView.model.attributes.config.type == 'Split' && $('.list-item', '#inspector_dialog').length >= 5) { | |
| $(this).prop('disabled', true); | |
| } | |
| }) | |
| .on('click', '[data-attribute="criteria/action"]', function() { | |
| var value = $(this).val(); | |
| $('[data-field="criteria/list_name"]', '#inspector_dialog').toggle(value == 'list' || value == 'unlist'); | |
| }) | |
| .on('change', '[data-attribute="campaign_list_id"]', function () { | |
| var options = '<option value="" selected disabled>Choose stage..</option>', | |
| campaign_list_id = $(this).val(); | |
| $('[data-field="stage_id"]', '#inspector_dialog').toggle(campaign_list_id != ''); | |
| if (campaign_list_id) { | |
| for (var i = 0, | |
| stages = inspector.options.cellView.model.attributes.config.dropdowns.campaigns_list[campaign_list_id].stages, | |
| n = stages.length; | |
| i < n; i ++) { | |
| options += '<option value="' + stages[i].id + '">' + stages[i].name + '</option>'; | |
| } | |
| } | |
| $('[data-attribute="stage_id"]', '#inspector_dialog').html(options); | |
| }); | |
| $(document).on('click.btnListDel', '.btn-list-del', function () { | |
| if (inspector.options.cellView.model.attributes.config.type == 'Split') { | |
| $('.btn-list-add', '#inspector_dialog').prop('disabled', false); | |
| } | |
| }); | |
| } | |
| return { | |
| shapeSize: shapeSize, | |
| iconSize: iconSize, | |
| init: function(stencilData, options) { | |
| gOptions = options; | |
| initializePaper(); | |
| initializeStencil(stencilData); | |
| initializeHaloAndInspector(); | |
| initializeNavigator(); | |
| initializeMisc(); | |
| }, | |
| createCustomShapes: function() { | |
| createCustomShapes(); | |
| }, | |
| /* for debugging */ | |
| exposed: function() { | |
| return { | |
| paper: paper, | |
| graph: graph, | |
| paperScroller: paperScroller, | |
| nav: nav, | |
| stencil: stencil, | |
| snaplines: snaplines, | |
| halo: halo, | |
| inspector: inspector, | |
| chart: chart, | |
| gOptions: gOptions | |
| }; | |
| } | |
| }; | |
| }(); | 
  
    Sign up for free
    to join this conversation on GitHub.
    Already have an account?
    Sign in to comment