diff options
author | wout <wout@impinc.co.uk> | 2014-07-28 22:16:35 +0200 |
---|---|---|
committer | wout <wout@impinc.co.uk> | 2014-07-28 22:16:35 +0200 |
commit | 5e9ab52bd68670208ac25cfcf95cca178d285545 (patch) | |
tree | 0e83007452b8195886798889d2c7eeabb229d226 /dist/svg.js | |
parent | 0c336f2f4bc931c13779ed6ea3edf62f05f1f18f (diff) | |
download | svg.js-5e9ab52bd68670208ac25cfcf95cca178d285545.tar.gz svg.js-5e9ab52bd68670208ac25cfcf95cca178d285545.zip |
Added more control to SVG.Matrix
Diffstat (limited to 'dist/svg.js')
-rwxr-xr-x | dist/svg.js | 934 |
1 files changed, 473 insertions, 461 deletions
diff --git a/dist/svg.js b/dist/svg.js index a7cbb72..0742d27 100755 --- a/dist/svg.js +++ b/dist/svg.js @@ -6,7 +6,7 @@ * @copyright Wout Fierens <wout@impinc.co.uk> * @license MIT * -* BUILT: Sun Jul 27 2014 13:36:08 GMT+0200 (CEST) +* BUILT: Mon Jul 28 2014 19:09:36 GMT+0200 (CEST) */ ;(function() { // The main wrapping element @@ -1080,6 +1080,437 @@ SVG.Element = SVG.invent({ } }) +SVG.FX = SVG.invent({ + // Initialize FX object + create: function(element) { + // store target element + this.target = element + } + + // Add class methods +, extend: { + // Add animation parameters and start animation + animate: function(d, ease, delay) { + var akeys, skeys, key + , element = this.target + , fx = this + + // dissect object if one is passed + if (typeof d == 'object') { + delay = d.delay + ease = d.ease + d = d.duration + } + + // ensure default duration and easing + d = d == '=' ? d : d == null ? 1000 : new SVG.Number(d).valueOf() + ease = ease || '<>' + + // process values + fx.at = function(pos) { + var i + + // normalise pos + pos = pos < 0 ? 0 : pos > 1 ? 1 : pos + + // collect attribute keys + if (akeys == null) { + akeys = [] + for (key in fx.attrs) + akeys.push(key) + + // make sure morphable elements are scaled, translated and morphed all together + if (element.morphArray && (fx._plot || akeys.indexOf('points') > -1)) { + // get destination + var box + , p = new element.morphArray(fx._plot || fx.attrs.points || element.array) + + // add size + if (fx._size) p.size(fx._size.width.to, fx._size.height.to) + + // add movement + box = p.bbox() + if (fx._x) p.move(fx._x.to, box.y) + else if (fx._cx) p.move(fx._cx.to - box.width / 2, box.y) + + box = p.bbox() + if (fx._y) p.move(box.x, fx._y.to) + else if (fx._cy) p.move(box.x, fx._cy.to - box.height / 2) + + // delete element oriented changes + delete fx._x + delete fx._y + delete fx._cx + delete fx._cy + delete fx._size + + fx._plot = element.array.morph(p) + } + } + + // collect style keys + if (skeys == null) { + skeys = [] + for (key in fx.styles) + skeys.push(key) + } + + // apply easing + pos = ease == '<>' ? + (-Math.cos(pos * Math.PI) / 2) + 0.5 : + ease == '>' ? + Math.sin(pos * Math.PI / 2) : + ease == '<' ? + -Math.cos(pos * Math.PI / 2) + 1 : + ease == '-' ? + pos : + typeof ease == 'function' ? + ease(pos) : + pos + + // run plot function + if (fx._plot) { + element.plot(fx._plot.at(pos)) + + } else { + // run all x-position properties + if (fx._x) + element.x(fx._x.at(pos)) + else if (fx._cx) + element.cx(fx._cx.at(pos)) + + // run all y-position properties + if (fx._y) + element.y(fx._y.at(pos)) + else if (fx._cy) + element.cy(fx._cy.at(pos)) + + // run all size properties + if (fx._size) + element.size(fx._size.width.at(pos), fx._size.height.at(pos)) + } + + // run all viewbox properties + if (fx._viewbox) + element.viewbox( + fx._viewbox.x.at(pos) + , fx._viewbox.y.at(pos) + , fx._viewbox.width.at(pos) + , fx._viewbox.height.at(pos) + ) + + // run leading property + if (fx._leading) + element.leading(fx._leading.at(pos)) + + // animate attributes + for (i = akeys.length - 1; i >= 0; i--) + element.attr(akeys[i], at(fx.attrs[akeys[i]], pos)) + + // animate styles + for (i = skeys.length - 1; i >= 0; i--) + element.style(skeys[i], at(fx.styles[skeys[i]], pos)) + + // callback for each keyframe + if (fx._during) + fx._during.call(element, pos, function(from, to) { + return at({ from: from, to: to }, pos) + }) + } + + if (typeof d === 'number') { + // delay animation + this.timeout = setTimeout(function() { + var start = new Date().getTime() + + // initialize situation object + fx.situation = { + interval: 1000 / 60 + , start: start + , play: true + , finish: start + d + , duration: d + } + + // render function + fx.render = function() { + + if (fx.situation.play === true) { + // calculate pos + var time = new Date().getTime() + , pos = time > fx.situation.finish ? 1 : (time - fx.situation.start) / d + + // process values + fx.at(pos) + + // finish off animation + if (time > fx.situation.finish) { + if (fx._plot) + element.plot(new SVG.PointArray(fx._plot.destination).settle()) + + if (fx._loop === true || (typeof fx._loop == 'number' && fx._loop > 1)) { + if (typeof fx._loop == 'number') + --fx._loop + fx.animate(d, ease, delay) + } else { + fx._after ? fx._after.apply(element, [fx]) : fx.stop() + } + + } else { + requestAnimFrame(fx.render) + } + } else { + requestAnimFrame(fx.render) + } + + } + + // start animation + fx.render() + + }, new SVG.Number(delay).valueOf()) + } + + return this + } + // Get bounding box of target element + , bbox: function() { + return this.target.bbox() + } + // Add animatable attributes + , attr: function(a, v) { + // apply attributes individually + if (typeof a == 'object') { + for (var key in a) + this.attr(key, a[key]) + + } else { + // get the current state + var from = this.target.attr(a) + + // detect format + this.attrs[a] = a == 'transform' ? + this.target.ctm().morph(v) : + SVG.Color.isColor(from) ? + new SVG.Color(from).morph(v) : + SVG.regex.unit.test(from) ? + new SVG.Number(from).morph(v) : + { from: from, to: v } + } + + return this + } + // Add animatable styles + , style: function(s, v) { + if (typeof s == 'object') + for (var key in s) + this.style(key, s[key]) + + else + this.styles[s] = { from: this.target.style(s), to: v } + + return this + } + // Animatable x-axis + , x: function(x) { + this._x = new SVG.Number(this.target.x()).morph(x) + + return this + } + // Animatable y-axis + , y: function(y) { + this._y = new SVG.Number(this.target.y()).morph(y) + + return this + } + // Animatable center x-axis + , cx: function(x) { + this._cx = new SVG.Number(this.target.cx()).morph(x) + + return this + } + // Animatable center y-axis + , cy: function(y) { + this._cy = new SVG.Number(this.target.cy()).morph(y) + + return this + } + // Add animatable move + , move: function(x, y) { + return this.x(x).y(y) + } + // Add animatable center + , center: function(x, y) { + return this.cx(x).cy(y) + } + // Add animatable size + , size: function(width, height) { + if (this.target instanceof SVG.Text) { + // animate font size for Text elements + this.attr('font-size', width) + + } else { + // animate bbox based size for all other elements + var box = this.target.bbox() + + this._size = { + width: new SVG.Number(box.width).morph(width) + , height: new SVG.Number(box.height).morph(height) + } + } + + return this + } + // Add animatable plot + , plot: function(p) { + this._plot = p + + return this + } + // Add leading method + , leading: function(value) { + if (this.target._leading) + this._leading = new SVG.Number(this.target._leading).morph(value) + + return this + } + // Add animatable viewbox + , viewbox: function(x, y, width, height) { + if (this.target instanceof SVG.Container) { + var box = this.target.viewbox() + + this._viewbox = { + x: new SVG.Number(box.x).morph(x) + , y: new SVG.Number(box.y).morph(y) + , width: new SVG.Number(box.width).morph(width) + , height: new SVG.Number(box.height).morph(height) + } + } + + return this + } + // Add animateable gradient update + , update: function(o) { + if (this.target instanceof SVG.Stop) { + if (o.opacity != null) this.attr('stop-opacity', o.opacity) + if (o.color != null) this.attr('stop-color', o.color) + if (o.offset != null) this.attr('offset', new SVG.Number(o.offset)) + } + + return this + } + // Add callback for each keyframe + , during: function(during) { + this._during = during + + return this + } + // Callback after animation + , after: function(after) { + this._after = after + + return this + } + // Make loopable + , loop: function(times) { + this._loop = times || true + + return this + } + // Stop running animation + , stop: function(fulfill) { + // fulfill animation + if (fulfill === true) { + + this.animate(0) + + if (this._after) + this._after.apply(this.target, [this]) + + } else { + // stop current animation + clearTimeout(this.timeout) + + // reset storage for properties that need animation + this.attrs = {} + this.trans = {} + this.styles = {} + this.situation = {} + + // delete destinations + delete this._x + delete this._y + delete this._cx + delete this._cy + delete this._size + delete this._plot + delete this._loop + delete this._after + delete this._during + delete this._leading + delete this._viewbox + } + + return this + } + // Pause running animation + , pause: function() { + if (this.situation.play === true) { + this.situation.play = false + this.situation.pause = new Date().getTime() + } + + return this + } + // Play running animation + , play: function() { + if (this.situation.play === false) { + var pause = new Date().getTime() - this.situation.pause + + this.situation.finish += pause + this.situation.start += pause + this.situation.play = true + } + + return this + } + + } + + // Define parent class +, parent: SVG.Element + + // Add method to parent elements +, construct: { + // Get fx module or create a new one, then animate with given duration and ease + animate: function(d, ease, delay) { + return (this.fx || (this.fx = new SVG.FX(this))).stop().animate(d, ease, delay) + } + // Stop current animation; this is an alias to the fx instance + , stop: function(fulfill) { + if (this.fx) + this.fx.stop(fulfill) + + return this + } + // Pause current animation + , pause: function() { + if (this.fx) + this.fx.pause() + + return this + } + // Play paused current animation + , play: function() { + if (this.fx) + this.fx.play() + + return this + } + + } +}) + SVG.BBox = SVG.invent({ // Initialize create: function(element) { @@ -1330,37 +1761,37 @@ SVG.Matrix = SVG.invent({ cx = y } - return this - .multiply(new SVG.Matrix(1, 0, 0, 1, cx || 0, cy || 0)) - .multiply(new SVG.Matrix(x, 0, 0, y, 0, 0)) - .multiply(new SVG.Matrix(1, 0, 0, 1, -cx || 0, -cy || 0)) + return this.around(cx, cy, new SVG.Matrix(x, 0, 0, y, 0, 0)) } // Rotate matrix - , rotate: function(d, cx, cy) { + , rotate: function(r, cx, cy) { // convert degrees to radians - d = SVG.utils.radians(d) + r = SVG.utils.radians(r) - return this - .multiply(new SVG.Matrix(1, 0, 0, 1, cx || 0, cy || 0)) - .multiply(new SVG.Matrix(Math.cos(d), Math.sin(d), -Math.sin(d), Math.cos(d), 0, 0)) - .multiply(new SVG.Matrix(1, 0, 0, 1, -cx || 0, -cy || 0)) + return this.around(cx, cy, new SVG.Matrix(Math.cos(r), Math.sin(r), -Math.sin(r), Math.cos(r), 0, 0)) } - // Flip matrix on x or y - , flip: function(a) { - return new SVG.Matrix(this.native()['flip' + a.toUpperCase()]()) + // Flip matrix on x or y, at a given offset + , flip: function(a, o) { + return a == 'x' ? this.scale(-1, 1, o, 0) : this.scale(1, -1, 0, o) } // Skew , skew: function(x, y, cx, cy) { - // IMPLEMENT SKEW CENTER POINT - return new SVG.Matrix(this.native().skewX(x || 0).skewY(y || 0)) + return this.around(cx, cy, this.native().skewX(x || 0).skewY(y || 0)) + } + // Transform around a center point + , around: function(cx, cy, matrix) { + return this + .multiply(new SVG.Matrix(1, 0, 0, 1, cx || 0, cy || 0)) + .multiply(matrix) + .multiply(new SVG.Matrix(1, 0, 0, 1, -cx || 0, -cy || 0)) } // Convert to native SVGMatrix , native: function() { // create new matrix - var i, matrix = SVG.parser.draw.node.createSVGMatrix() + var matrix = SVG.parser.draw.node.createSVGMatrix() // update with current values - for (i = abcdef.length - 1; i >= 0; i--) + for (var i = abcdef.length - 1; i >= 0; i--) matrix[abcdef[i]] = this[abcdef[i]] return matrix @@ -1476,10 +1907,10 @@ SVG.extend(SVG.Element, SVG.FX, { // singular getter else if (typeof o === 'string') return target.ctm().extract()[o] - + // get current matrix var matrix = new SVG.Matrix(target) - + // act on matrix if (o.a != null) matrix = matrix.multiply(new SVG.Matrix(o)) @@ -1491,19 +1922,31 @@ SVG.extend(SVG.Element, SVG.FX, { , o.cx == null ? target.bbox().cx : o.cx , o.cy == null ? target.bbox().cy : o.cy ) - + // act on scale else if (o.scale != null || o.scaleX != null || o.scaleY != null) matrix = matrix.scale( o.scale != null ? o.scale : o.scaleX != null ? o.scaleX : 1 , o.scale != null ? o.scale : o.scaleY != null ? o.scaleY : 1 - , o.cx != null ? o.cx : target.bbox().x - , o.cy != null ? o.cy : target.bbox().y + , o.cx == null ? target.bbox().x : o.cx + , o.cy == null ? target.bbox().y : o.cy ) // act on skew else if (o.skewX || o.skewY) - matrix = matrix.skew(o.skewX, o.skewY) + matrix = matrix.skew( + o.skewX + , o.skewY + , o.cx == null ? target.bbox().cx : o.cx + , o.cy == null ? target.bbox().cy : o.cy + ) + + // act on flip + else if (o.flip) + matrix = matrix.flip( + o.flip + , o.offset == null ? target.bbox()['c' + o.flip] : o.offset + ) // act on translate else if (o.x || o.y) @@ -1706,441 +2149,6 @@ SVG.extend(SVG.Parent, SVG.Text, { } } }) -SVG.FX = SVG.invent({ - // Initialize FX object - create: function(element) { - // store target element - this.target = element - } - - // Add class methods -, extend: { - // Add animation parameters and start animation - animate: function(d, ease, delay) { - var akeys, skeys, key - , element = this.target - , fx = this - - // dissect object if one is passed - if (typeof d == 'object') { - delay = d.delay - ease = d.ease - d = d.duration - } - - // ensure default duration and easing - d = d == '=' ? d : d == null ? 1000 : new SVG.Number(d).valueOf() - ease = ease || '<>' - - // process values - fx.at = function(pos) { - var i - - // normalise pos - pos = pos < 0 ? 0 : pos > 1 ? 1 : pos - - // collect attribute keys - if (akeys == null) { - akeys = [] - for (key in fx.attrs) - akeys.push(key) - - // make sure morphable elements are scaled, translated and morphed all together - if (element.morphArray && (fx._plot || akeys.indexOf('points') > -1)) { - // get destination - var box - , p = new element.morphArray(fx._plot || fx.attrs.points || element.array) - - // add size - if (fx._size) p.size(fx._size.width.to, fx._size.height.to) - - // add movement - box = p.bbox() - if (fx._x) p.move(fx._x.to, box.y) - else if (fx._cx) p.move(fx._cx.to - box.width / 2, box.y) - - box = p.bbox() - if (fx._y) p.move(box.x, fx._y.to) - else if (fx._cy) p.move(box.x, fx._cy.to - box.height / 2) - - // delete element oriented changes - delete fx._x - delete fx._y - delete fx._cx - delete fx._cy - delete fx._size - - fx._plot = element.array.morph(p) - } - } - - // collect style keys - if (skeys == null) { - skeys = [] - for (key in fx.styles) - skeys.push(key) - } - - // apply easing - pos = ease == '<>' ? - (-Math.cos(pos * Math.PI) / 2) + 0.5 : - ease == '>' ? - Math.sin(pos * Math.PI / 2) : - ease == '<' ? - -Math.cos(pos * Math.PI / 2) + 1 : - ease == '-' ? - pos : - typeof ease == 'function' ? - ease(pos) : - pos - - // run plot function - if (fx._plot) { - element.plot(fx._plot.at(pos)) - - } else { - // run all x-position properties - if (fx._x) - element.x(fx._x.at(pos)) - else if (fx._cx) - element.cx(fx._cx.at(pos)) - - // run all y-position properties - if (fx._y) - element.y(fx._y.at(pos)) - else if (fx._cy) - element.cy(fx._cy.at(pos)) - - // run all size properties - if (fx._size) - element.size(fx._size.width.at(pos), fx._size.height.at(pos)) - } - - // run all viewbox properties - if (fx._viewbox) - element.viewbox( - fx._viewbox.x.at(pos) - , fx._viewbox.y.at(pos) - , fx._viewbox.width.at(pos) - , fx._viewbox.height.at(pos) - ) - - // run leading property - if (fx._leading) - element.leading(fx._leading.at(pos)) - - // animate attributes - for (i = akeys.length - 1; i >= 0; i--) - element.attr(akeys[i], at(fx.attrs[akeys[i]], pos)) - - // animate styles - for (i = skeys.length - 1; i >= 0; i--) - element.style(skeys[i], at(fx.styles[skeys[i]], pos)) - - // callback for each keyframe - if (fx._during) - fx._during.call(element, pos, function(from, to) { - return at({ from: from, to: to }, pos) - }) - } - - if (typeof d === 'number') { - // delay animation - this.timeout = setTimeout(function() { - var start = new Date().getTime() - - // initialize situation object - fx.situation = { - interval: 1000 / 60 - , start: start - , play: true - , finish: start + d - , duration: d - } - - // render function - fx.render = function() { - - if (fx.situation.play === true) { - // calculate pos - var time = new Date().getTime() - , pos = time > fx.situation.finish ? 1 : (time - fx.situation.start) / d - - // process values - fx.at(pos) - - // finish off animation - if (time > fx.situation.finish) { - if (fx._plot) - element.plot(new SVG.PointArray(fx._plot.destination).settle()) - - if (fx._loop === true || (typeof fx._loop == 'number' && fx._loop > 1)) { - if (typeof fx._loop == 'number') - --fx._loop - fx.animate(d, ease, delay) - } else { - fx._after ? fx._after.apply(element, [fx]) : fx.stop() - } - - } else { - requestAnimFrame(fx.render) - } - } else { - requestAnimFrame(fx.render) - } - - } - - // start animation - fx.render() - - }, new SVG.Number(delay).valueOf()) - } - - return this - } - // Get bounding box of target element - , bbox: function() { - return this.target.bbox() - } - // Add animatable attributes - , attr: function(a, v) { - // apply attributes individually - if (typeof a == 'object') { - for (var key in a) - this.attr(key, a[key]) - - } else { - // get the current state - var from = this.target.attr(a) - - // detect format - this.attrs[a] = a == 'transform' ? - this.target.ctm().morph(v) : - SVG.Color.isColor(from) ? - new SVG.Color(from).morph(v) : - SVG.regex.unit.test(from) ? - new SVG.Number(from).morph(v) : - { from: from, to: v } - } - - return this - } - // Add animatable transformations - , transform: function(o) { - return this.attr('transform', o) - } - // Add animatable styles - , style: function(s, v) { - if (typeof s == 'object') - for (var key in s) - this.style(key, s[key]) - - else - this.styles[s] = { from: this.target.style(s), to: v } - - return this - } - // Animatable x-axis - , x: function(x) { - this._x = new SVG.Number(this.target.x()).morph(x) - - return this - } - // Animatable y-axis - , y: function(y) { - this._y = new SVG.Number(this.target.y()).morph(y) - - return this - } - // Animatable center x-axis - , cx: function(x) { - this._cx = new SVG.Number(this.target.cx()).morph(x) - - return this - } - // Animatable center y-axis - , cy: function(y) { - this._cy = new SVG.Number(this.target.cy()).morph(y) - - return this - } - // Add animatable move - , move: function(x, y) { - return this.x(x).y(y) - } - // Add animatable center - , center: function(x, y) { - return this.cx(x).cy(y) - } - // Add animatable size - , size: function(width, height) { - if (this.target instanceof SVG.Text) { - // animate font size for Text elements - this.attr('font-size', width) - - } else { - // animate bbox based size for all other elements - var box = this.target.bbox() - - this._size = { - width: new SVG.Number(box.width).morph(width) - , height: new SVG.Number(box.height).morph(height) - } - } - - return this - } - // Add animatable plot - , plot: function(p) { - this._plot = p - - return this - } - // Add leading method - , leading: function(value) { - if (this.target._leading) - this._leading = new SVG.Number(this.target._leading).morph(value) - - return this - } - // Add animatable viewbox - , viewbox: function(x, y, width, height) { - if (this.target instanceof SVG.Container) { - var box = this.target.viewbox() - - this._viewbox = { - x: new SVG.Number(box.x).morph(x) - , y: new SVG.Number(box.y).morph(y) - , width: new SVG.Number(box.width).morph(width) - , height: new SVG.Number(box.height).morph(height) - } - } - - return this - } - // Add animateable gradient update - , update: function(o) { - if (this.target instanceof SVG.Stop) { - if (o.opacity != null) this.attr('stop-opacity', o.opacity) - if (o.color != null) this.attr('stop-color', o.color) - if (o.offset != null) this.attr('offset', new SVG.Number(o.offset)) - } - - return this - } - // Add callback for each keyframe - , during: function(during) { - this._during = during - - return this - } - // Callback after animation - , after: function(after) { - this._after = after - - return this - } - // Make loopable - , loop: function(times) { - this._loop = times || true - - return this - } - // Stop running animation - , stop: function(fulfill) { - // fulfill animation - if (fulfill === true) { - - this.animate(0) - - if (this._after) - this._after.apply(this.target, [this]) - - } else { - // stop current animation - clearTimeout(this.timeout) - - // reset storage for properties that need animation - this.attrs = {} - this.trans = {} - this.styles = {} - this.situation = {} - - // delete destinations - delete this._x - delete this._y - delete this._cx - delete this._cy - delete this._size - delete this._plot - delete this._loop - delete this._after - delete this._during - delete this._leading - delete this._viewbox - } - - return this - } - // Pause running animation - , pause: function() { - if (this.situation.play === true) { - this.situation.play = false - this.situation.pause = new Date().getTime() - } - - return this - } - // Play running animation - , play: function() { - if (this.situation.play === false) { - var pause = new Date().getTime() - this.situation.pause - - this.situation.finish += pause - this.situation.start += pause - this.situation.play = true - } - - return this - } - - } - - // Define parent class -, parent: SVG.Element - - // Add method to parent elements -, construct: { - // Get fx module or create a new one, then animate with given duration and ease - animate: function(d, ease, delay) { - return (this.fx || (this.fx = new SVG.FX(this))).stop().animate(d, ease, delay) - } - // Stop current animation; this is an alias to the fx instance - , stop: function(fulfill) { - if (this.fx) - this.fx.stop(fulfill) - - return this - } - // Pause current animation - , pause: function() { - if (this.fx) - this.fx.pause() - - return this - } - // Play paused current animation - , play: function() { - if (this.fx) - this.fx.play() - - return this - } - - } -}) - // SVG.extend(SVG.Element, SVG.FX, { // Relative move over x axis @@ -3572,8 +3580,8 @@ SVG.extend(SVG.Element, SVG.FX, { return this.transform({ rotation: d, cx: cx, cy: cy }) } // Map skew to transform -, skew: function(x, y) { - return this.transform({ skewX: x, skewY: y }) +, skew: function(x, y, cx, cy) { + return this.transform({ skewX: x, skewY: y, cx: cx, cy: cy }) } // Map scale to transform , scale: function(x, y, cx, cy) { @@ -3585,6 +3593,10 @@ SVG.extend(SVG.Element, SVG.FX, { , translate: function(x, y) { return this.transform({ x: x, y: y }) } + // Map flip to transform +, flip: function(a, o) { + return this.transform({ flip: a, offset: o }) + } // Map matrix to transform , matrix: function(m) { return this.attr('transform', new SVG.Matrix(m)) |