/** * Created by Alex on 26-Feb-15. */ if (typeof window !== 'undefined') { window.requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame; } class CanvasRenderer { constructor(body) { this.body = body; this.redrawRequested = false; this.renderTimer = false; this.requiresTimeout = true; this.continueRendering = true; this.renderRequests = 0; this.translation = {x: 0, y: 0}; this.scale = 1.0; this.canvasTopLeft = {x: 0, y: 0}; this.canvasBottomRight = {x: 0, y: 0}; this.body.emitter.on("_setScale", (scale) => this.scale = scale); this.body.emitter.on("_setTranslation", (translation) => {this.translation.x = translation.x; this.translation.y = translation.y;}); this.body.emitter.on("_redraw", this._redraw.bind(this)); this.body.emitter.on("_redrawHidden", this._redraw.bind(this, true)); this.body.emitter.on("_requestRedraw", this._requestRedraw.bind(this)); this.body.emitter.on("_startRendering", () => {this.renderRequests += 1; this.continueRendering = true; this.startRendering();}); this.body.emitter.on("_stopRendering", () => {this.renderRequests -= 1; this.continueRendering = this.renderRequests > 0;}); this._determineBrowserMethod(); } startRendering() { if (this.continueRendering === true) { if (!this.renderTimer) { if (this.requiresTimeout == true) { this.renderTimer = window.setTimeout(this.renderStep.bind(this), this.simulationInterval); // wait this.renderTimeStep milliseconds and perform the animation step function } else { this.renderTimer = window.requestAnimationFrame(this.renderStep.bind(this)); // wait this.renderTimeStep milliseconds and perform the animation step function } } } } renderStep() { // reset the renderTimer so a new scheduled animation step can be set this.renderTimer = undefined; if (this.requiresTimeout == true) { // this schedules a new simulation step this.startRendering(); } this._redraw(); if (this.requiresTimeout == false) { // this schedules a new simulation step this.startRendering(); } } setCanvas(canvas) { this.canvas = canvas; } /** * Redraw the network with the current data * chart will be resized too. */ redraw() { this.setSize(this.constants.width, this.constants.height); this._redraw(); } /** * Redraw the network with the current data * @param hidden | used to get the first estimate of the node sizes. only the nodes are drawn after which they are quickly drawn over. * @private */ _requestRedraw(hidden) { if (this.redrawRequested !== true) { this.redrawRequested = true; if (this.requiresTimeout === true) { window.setTimeout(this._redraw.bind(this, hidden),0); } else { window.requestAnimationFrame(this._redraw.bind(this, hidden, true)); } } } _redraw(hidden = false) { this.body.emitter.emit("_beforeRender"); this.redrawRequested = false; var ctx = this.canvas.frame.canvas.getContext('2d'); ctx.setTransform(this.pixelRatio, 0, 0, this.pixelRatio, 0, 0); // clear the canvas var w = this.canvas.frame.canvas.clientWidth; var h = this.canvas.frame.canvas.clientHeight; ctx.clearRect(0, 0, w, h); // set scaling and translation ctx.save(); ctx.translate(this.translation.x, this.translation.y); ctx.scale(this.scale, this.scale); this.canvasTopLeft = this.canvas.DOMtoCanvas({x:0,y:0}); this.canvasBottomRight = this.canvas.DOMtoCanvas({x:this.canvas.frame.canvas.clientWidth,y:this.canvas.frame.canvas.clientHeight}); if (hidden === false) { // todo: solve this //if (this.drag.dragging == false || this.drag.dragging === undefined || this.constants.hideEdgesOnDrag == false) { this._drawEdges(ctx); //} } // todo: solve this //if (this.drag.dragging == false || this.drag.dragging === undefined || this.constants.hideNodesOnDrag == false) { this._drawNodes(ctx, this.body.nodes, hidden); //} if (hidden === false) { if (this.controlNodesActive == true) { this._drawControlNodes(ctx); } } //this._drawNodes(ctx,this.body.supportNodes,true); // this.physics.nodesSolver._debug(ctx,"#F00F0F"); // restore original scaling and translation ctx.restore(); if (hidden === true) { ctx.clearRect(0, 0, w, h); } } /** * Redraw all nodes * The 2d context of a HTML canvas can be retrieved by canvas.getContext('2d'); * @param {CanvasRenderingContext2D} ctx * @param {Boolean} [alwaysShow] * @private */ _drawNodes(ctx,nodes,alwaysShow = false) { // first draw the unselected nodes var selected = []; for (var id in nodes) { if (nodes.hasOwnProperty(id)) { nodes[id].setScaleAndPos(this.scale,this.canvasTopLeft,this.canvasBottomRight); if (nodes[id].isSelected()) { selected.push(id); } else { if (alwaysShow === true) { nodes[id].draw(ctx); } else if (nodes[id].inArea() === true) { nodes[id].draw(ctx); } } } } // draw the selected nodes on top for (var s = 0, sMax = selected.length; s < sMax; s++) { if (nodes[selected[s]].inArea() || alwaysShow) { nodes[selected[s]].draw(ctx); } } } /** * Redraw all edges * The 2d context of a HTML canvas can be retrieved by canvas.getContext('2d'); * @param {CanvasRenderingContext2D} ctx * @private */ _drawEdges(ctx) { var edges = this.body.edges; for (var id in edges) { if (edges.hasOwnProperty(id)) { var edge = edges[id]; edge.setScale(this.scale); if (edge.connected === true) { edges[id].draw(ctx); } } } } /** * Redraw all edges * The 2d context of a HTML canvas can be retrieved by canvas.getContext('2d'); * @param {CanvasRenderingContext2D} ctx * @private */ _drawControlNodes(ctx) { var edges = this.body.edges; for (var id in edges) { if (edges.hasOwnProperty(id)) { edges[id]._drawControlNodes(ctx); } } } /** * Determine if the browser requires a setTimeout or a requestAnimationFrame. This was required because * some implementations (safari and IE9) did not support requestAnimationFrame * @private */ _determineBrowserMethod() { if (typeof window !== 'undefined') { var browserType = navigator.userAgent.toLowerCase(); this.requiresTimeout = false; if (browserType.indexOf('msie 9.0') != -1) { // IE 9 this.requiresTimeout = true; } else if (browserType.indexOf('safari') != -1) { // safari if (browserType.indexOf('chrome') <= -1) { this.requiresTimeout = true; } } } else { this.requiresTimeout = true; } } } export {CanvasRenderer};