/** * Abstract class for D3 Components: {@link Ext.d3.canvas.Canvas} and {@link Ext.d3.svg.Svg}. * * Notes: * * Unlike the Charts package with its Draw layer, the D3 package does not provide * an abstraction layer and the user is expected to deal with the SVG and Canvas * directly. * * D3 package supports IE starting from version 9, as neither Canvas nor SVG * are supported by prior IE versions. */Ext.define('Ext.d3.Component', { extend: 'Ext.d3.ComponentBase', requires: [ 'Ext.d3.lib.d3' ], config: { /** * The store with data to render. * @cfg {Ext.data.Store} store */ store: null, /** * The CSS class used by a subclass of the D3 Component. * Normally, the lower-cased name of a subclass. * @cfg {String} componentCls */ componentCls: '', /** * The list of interaction configs for this D3 component. * D3 package interactions are very simular to native D3 behaviors. * However, D3 behaviors, as well as event system, is incompatible * with ExtJS event system. D3 package interactions may also support * certain features that D3 behaviors lack, like like kinetic scrolling, * elastic borders and scroll indicators (see the {@link Ext.d3.interaction.PanZoom panzoom} * interaction for more information). * */ interactions: [], /** * A map of transition configs. For example: * * transitions: { * select: { * duration: 500, * ease: 'cubicInOut' * }, * zoom: { * name: 'zoom', * duration: 1000 * }, * ... * } * * A class would define the defaults for its transitions, and a user only needs * to set the `transitions` config of an instance to disable a transition, e.g.: * * transitions: { * // transitions are enabled by default, `true` should never be used here * select: false * } * * or alter its config: * * transitions: { * select: { * // the `duration` stays the same, * // only the easing function is altered * ease: 'bounceOut' * } * } * * The transitions defined this way are merely configs. To create an actual transition * from one of these configs, use the {@link #createTransition} method. For example: * * this.createTransition('select') * * A transition object can optionally specify a name, if it's different from * the key in the `transitions` config. For example: * * transitions: { * layout: { * name: 'foo', * duration: 500 * } * } * * Otherwise the name will be set automatically, for example: * * transition.name = this.getId() + '-' + key * * Transition names (whether explicitly given or not) are prefixed by component ID * to prevent transitions with the same name but on a different component from * cancelling each other out. * * However, transitions with the same name on the same component will still cancel * each other out, if created via {@link #createTransition} on the same selection * or with no selection provided. * * `duration`, `ease` and `name` properties of transition objects in this config * are reserved, and will be used to configure a `d3.transition` instance. * However, transition objects may also have other properties that are related to * this transition. For example: * * transitions: { * select: { * duration: 500, * ease: 'cubicInOut', * targetScale: 1.1 * } * } * * The `targetScale` property here won't be consumed by a `d3.transition` instance; * instead a component can make use of it in whichever way it sees fit to animate * the selected element. */ transitions: { none: { name: undefined, duration: 0, ease: Ext.identityFn } }, touchAction: { panX: false, panY: false, pinchZoom: false, doubleTapZoom: false } }, baseCls: Ext.baseCSSPrefix + 'd3', /** * @private * Some configs in the D3 package are saved as properties on the class instance for faster access. * Prefixed properties are not used, as prefix can in theory change. * `$configPrefixed: false` is not used, as there's really no need to change the default for all configs. * All instance properties that are accessed bypassing getters/setters are listed on the prototype * to keep things explicit. */ defaultBindProperty: 'store', isInitializing: true, resizeDelay: 250, // in milliseconds resizeTimerId: 0, size: null, // cached size d3Components: null, constructor: function (config) { var me = this; me.d3Components = {}; me.callParent(arguments); me.isInitializing = false; me.on('resize', 'onElementResize', me); }, destroy: function () { var me = this; if (me.resizeTimerId) { clearTimeout(me.resizeTimerId); } me.un('resize', 'onElementResize', me); me.setInteractions(); me.callParent(); }, updateComponentCls: function (componentCls, oldComponentCls) { var baseCls = this.baseCls, el = this.element; if (componentCls && Ext.isString(componentCls)) { el.addCls(componentCls, baseCls); if (oldComponentCls) { el.removeCls(oldComponentCls, baseCls); } } }, register: function (component) { var map = this.d3Components, id = component.getId(); //<debug> if (id === undefined) { Ext.raise('Attempting to register a component with no ID.'); } if (id in map) { Ext.raise('Component with ID "' + id + '" is already registered.'); } //</debug> map[id] = component; }, unregister: function (component) { var map = this.d3Components, id = component.getId(); delete map[id]; }, applyInteractions: function (interactions, oldInteractions) { if (!oldInteractions) { oldInteractions = []; oldInteractions.map = {}; } var me = this, result = [], oldMap = oldInteractions.map, i, ln, interaction, id; result.map = {}; interactions = Ext.Array.from(interactions, true); // `true` to clone for (i = 0, ln = interactions.length; i < ln; i++) { interaction = interactions[i]; if (!interaction) { continue; } if (interaction.isInteraction) { id = interaction.getId(); } else { id = interaction.id; interaction.element = me.element; } // Create a new interaction by alias or reconfigure the old one. interaction = Ext.factory(interaction, null, oldMap[id], 'd3.interaction'); if (interaction) { interaction.setComponent(me); me.addInteraction(interaction); result.push(interaction); result.map[interaction.getId()] = interaction; } } // Destroy old interactions that were not reused. for (i in oldMap) { if (!result.map[i]) { interaction = oldMap[i]; me.removeInteraction(interaction); interaction.destroy(); } } return result; }, applyTransitions: function (transitions) { var name, transition, ease, ret = {}; for (name in transitions) { transition = transitions[name]; if (transition === true) { Ext.raise("'true' is not a valid transition value (should be an object or 'false')."); } ret[name] = transition = Ext.apply({}, transition || transitions.none); ease = transition.ease; if (typeof ease === 'string') { transition.ease = d3['ease' + ease.charAt(0).toUpperCase() + ease.substr(1)]; } if (!('name' in transition)) { transition.name = this.getId() + '-' + name; } } return ret; }, /** * Creates a `d3.transition` instance from a named object in the {@link #transitions} config. * @param {String} name The name of the transition in the {@link #transitions} config. * @param {d3.selection} [selection] The selection to create this transition on. * @return {d3.transition} */ createTransition: function (name, selection) { var transition = this.getTransitions()[name]; //<debug> if (!transition) { Ext.raise('No transition named "' + name + '"'); } //</debug> if (!selection) { selection = d3; } return selection.transition(transition.name) .ease(transition.ease) .duration(transition.duration); }, /** * @protected * If `panzoom` interaction has been added, save a reference to it on component instance * for quick access. */ panZoom: null, /** * @protected * When {@link Ext.d3.interaction.PanZoom `panzoom`} interaction is added to the component, * this method is used as a listener for the interaction's `panzoom` event. * This method should be implemented by subclasses what wish to be affected by the interaction. * @param {Ext.d3.interaction.PanZoom} interaction * @param {Number[]} scaling * @param {Number[]} translation */ onPanZoom: Ext.emptyFn, /** * @protected * Returns the bounding box of the content before transformations. * This method should be implemented by subclasses that wish to support constrained panning * via {@link Ext.d3.interaction.PanZoom `panzoom`} interaction. * @return {Object} rect * @return {Number} rect.x * @return {Number} rect.y * @return {Number} rect.width * @return {Number} rect.height */ getContentRect: Ext.emptyFn, /** * @protected * Returns the position and size of the viewport in component's coordinates. * This method should be implemented by subclasses that wish to support constrained panning * via {@link Ext.d3.interaction.PanZoom `panzoom`} interaction. * @return {Object} rect * @return {Number} rect.x * @return {Number} rect.y * @return {Number} rect.width * @return {Number} rect.height */ getViewportRect: Ext.emptyFn, addInteraction: function (interaction) { var me = this; if (interaction.isPanZoom) { interaction.setContentRect(me.getContentRect.bind(me)); interaction.setViewportRect(me.getViewportRect.bind(me)); me.panZoom = interaction; interaction.on('panzoom', me.onPanZoom, me); } }, removeInteraction: function (interaction) { if (interaction.isPanZoom) { interaction.setContentRect(null); interaction.setViewportRect(null); this.panZoom = null; interaction.un('panzoom', this.onPanZoom, this); } }, applyStore: function (store) { return store && Ext.StoreManager.lookup(store); }, updateStore: function (store, oldStore) { var me = this, storeEvents = { datachanged: 'onDataChange', update: 'onDataUpdate', load: 'onDataLoad', scope: me, order: 'after' }; if (oldStore) { oldStore.un(storeEvents); if (oldStore.getAutoDestroy()) { oldStore.destroy(); } } if (store) { store.on(storeEvents); me.onDataChange(store); } }, onDataChange: function (store) { var me = this; if (me.isInitializing) { return; } me.processDataChange(store); }, onDataUpdate: function (store, record, operation, modifiedFieldNames, details) { var me = this; if (me.isInitializing) { return; } me.processDataUpdate(store, record, operation, modifiedFieldNames, details); }, onDataLoad: function (store, records, successful, operation) { this.processDataLoad(store, records, successful, operation); }, processDataChange: Ext.emptyFn, processDataUpdate: Ext.emptyFn, processDataLoad: Ext.emptyFn, onElementResize: function (element, size) { this.handleResize(size); }, maskEl: null, /** * @private */ showMask: function (msg) { var me = this; if (me.maskEl) { me.maskEl.dom.firstChild.textContent = msg; me.maskEl.setStyle('display', 'flex'); } else { me.maskEl = this.element.createChild({ style: { position: 'absolute', display: 'flex', alignItems: 'center', justifyContent: 'center', left: 0, top: 0, bottom: 0, right: 0, background: 'rgba(0,0,0,0.5)' }, children: [{ html: msg, style: { background: 'rgba(0,0,0,0.8)', color: 'white', fontWeight: 'bold', borderRadius: '10px', padding: '10px' } }] }); } }, /** * @private */ hideMask: function () { this.maskEl && this.maskEl.setStyle('display', 'none'); }, handleResize: function (size, instantly) { var me = this, el = me.element; if (!(el && (size = size || el.getSize()) && size.width && size.height)) { return; } clearTimeout(me.resizeTimerId); if (instantly) { me.resizeTimerId = 0; } else { me.resizeTimerId = Ext.defer(me.handleResize, me.resizeDelay, me, [size, true]); return; } me.resizeHandler(size); if (me.panZoom && me.panZoom.updateIndicator) { me.panZoom.updateIndicator(); } me.size = size; }, resizeHandler: Ext.emptyFn, /** * Converts event coordinates from page coordinates to view coordinates. * @param {Ext.event.Event} event * @param {Ext.dom.Element} [view] If omitted, the component's element will be used. * @return {Number[]} */ toLocalXY: function (event, view) { var pageXY = event.getXY(), viewXY = view ? view.getXY() : this.element.getXY(); return [ pageXY[0] - viewXY[0], pageXY[1] - viewXY[1] ]; } });