summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorwout <wout@impinc.co.uk>2014-08-29 12:45:21 +0200
committerwout <wout@impinc.co.uk>2014-08-29 12:45:21 +0200
commitdd4be62e397e4db78444881eed32865f98e52582 (patch)
treec2300f62c756fde22c80f30aba66dca04602297b /src
parent8789391f036ffded3ac919d62ee652e7cae15c5c (diff)
downloadsvg.js-dd4be62e397e4db78444881eed32865f98e52582.tar.gz
svg.js-dd4be62e397e4db78444881eed32865f98e52582.zip
Added SVG.TBox for transformed bounding boxes
Diffstat (limited to 'src')
-rwxr-xr-xsrc/boxes.js98
-rwxr-xr-xsrc/fx.js19
-rwxr-xr-xsrc/group.js4
-rw-r--r--src/helpers.js213
-rw-r--r--src/matrix.js19
-rw-r--r--src/transform.js19
6 files changed, 212 insertions, 160 deletions
diff --git a/src/boxes.js b/src/boxes.js
index 96c4246..2c5cd8b 100755
--- a/src/boxes.js
+++ b/src/boxes.js
@@ -1,23 +1,14 @@
SVG.BBox = SVG.invent({
// Initialize
create: function(element) {
- var box
-
- // Initialize zero box
- this.x = 0
- this.y = 0
- this.width = 0
- this.height = 0
-
- // Get values if element is given
+ // get values if element is given
if (element) {
- // Get current extracted transformations
- var t = new SVG.Matrix(element).extract()
-
- // Find native bbox
+ var box
+
+ // find native bbox
if (element.node.getBBox)
box = element.node.getBBox()
- // Mimic bbox
+ // mimic bbox
else
box = {
x: element.node.clientLeft
@@ -26,55 +17,80 @@ SVG.BBox = SVG.invent({
, height: element.node.clientHeight
}
- // Include translations on x an y
+ // plain x and y
+ this.x = box.x
+ this.y = box.y
+
+ // plain width and height
+ this.width = box.width
+ this.height = box.height
+ }
+
+ // add center, right and bottom
+ fullBox(this)
+ }
+
+ // Define Parent
+, parent: SVG.Element
+
+ // Constructor
+, construct: {
+ // Get bounding box
+ bbox: function() {
+ return new SVG.BBox(this)
+ }
+ }
+
+})
+
+SVG.TBox = SVG.invent({
+ // Initialize
+ create: function(element) {
+ // get values if element is given
+ if (element) {
+ var t = element.ctm().extract()
+ , box = element.bbox()
+
+ // x and y including transformations
this.x = box.x + t.x
this.y = box.y + t.y
- // Plain width and height
+ // width and height including transformations
this.width = box.width * t.scaleX
this.height = box.height * t.scaleY
}
- // Add center, right and bottom
+ // add center, right and bottom
fullBox(this)
}
- // define Parent
+ // Define Parent
, parent: SVG.Element
// Constructor
, construct: {
- // Get bounding box
- bbox: function() {
- return new SVG.BBox(this)
+ // Get transformed bounding box
+ tbox: function() {
+ return new SVG.TBox(this)
}
}
})
+
SVG.RBox = SVG.invent({
// Initialize
create: function(element) {
- var box = {}
-
- // Initialize zero box
- this.x = 0
- this.y = 0
- this.width = 0
- this.height = 0
-
if (element) {
- var e = element.doc().parent()
+ var e = element.doc().parent()
+ , box = element.node.getBoundingClientRect()
, zoom = 1
- // Actual, native bounding box
- box = element.node.getBoundingClientRect()
-
- // Get screen offset
+ // get screen offset
this.x = box.left
this.y = box.top
- // Subtract parent offset
+ // subtract parent offset
this.x -= e.offsetLeft
this.y -= e.offsetTop
@@ -83,7 +99,7 @@ SVG.RBox = SVG.invent({
this.y -= e.offsetTop
}
- // Calculate cumulative zoom from svg documents
+ // calculate cumulative zoom from svg documents
e = element
while (e.parent && (e = e.parent())) {
if (e.viewbox) {
@@ -94,15 +110,15 @@ SVG.RBox = SVG.invent({
}
}
- // Recalculate viewbox distortion
+ // recalculate viewbox distortion
this.width = box.width /= zoom
this.height = box.height /= zoom
- // Offset by window scroll position, because getBoundingClientRect changes when window is scrolled
+ // offset by window scroll position, because getBoundingClientRect changes when window is scrolled
this.x += window.scrollX
this.y += window.scrollY
- // Add center, right and bottom
+ // add center, right and bottom
fullBox(this)
}
@@ -120,14 +136,14 @@ SVG.RBox = SVG.invent({
})
// Add universal merge method
-;[SVG.BBox, SVG.RBox].forEach(function(c) {
+;[SVG.BBox, SVG.TBox, SVG.RBox].forEach(function(c) {
SVG.extend(c, {
// Merge rect box with another, return a new instance
merge: function(box) {
var b = new c()
- // Merge box
+ // merge boxes
b.x = Math.min(this.x, box.x)
b.y = Math.min(this.y, box.y)
b.width = Math.max(this.x + this.width, box.x + box.width) - b.x
diff --git a/src/fx.js b/src/fx.js
index 0ba21a2..03e9317 100755
--- a/src/fx.js
+++ b/src/fx.js
@@ -222,13 +222,18 @@ SVG.FX = SVG.invent({
var from = this.target.attr(a)
// detect format
- this.attrs[a] = a == 'transform' ?
- this.target.ctm().morph(v) :
- SVG.Color.isColor(v) ?
- new SVG.Color(from).morph(v) :
- SVG.regex.unit.test(v) ?
- new SVG.Number(from).morph(v) :
- { from: from, to: v }
+ if (a == 'transform') {
+ if (this.attrs[a])
+ v = this.attrs[a].destination.multiply(v)
+
+ this.attrs[a] = this.target.ctm().morph(v)
+ } else {
+ this.attrs[a] = SVG.Color.isColor(v) ?
+ new SVG.Color(from).morph(v) :
+ SVG.regex.unit.test(v) ?
+ new SVG.Number(from).morph(v) :
+ { from: from, to: v }
+ }
}
return this
diff --git a/src/group.js b/src/group.js
index 66a8437..1b61371 100755
--- a/src/group.js
+++ b/src/group.js
@@ -17,11 +17,11 @@ SVG.G = SVG.invent({
}
// Move by center over x-axis
, cx: function(x) {
- return x == null ? this.bbox().cx : this.x(x - this.bbox().width / 2)
+ return x == null ? this.tbox().cx : this.x(x - this.tbox().width / 2)
}
// Move by center over y-axis
, cy: function(y) {
- return y == null ? this.bbox().cy : this.y(y - this.bbox().height / 2)
+ return y == null ? this.tbox().cy : this.y(y - this.tbox().height / 2)
}
}
diff --git a/src/helpers.js b/src/helpers.js
index 8b11a00..70f189f 100644
--- a/src/helpers.js
+++ b/src/helpers.js
@@ -1,156 +1,165 @@
// Convert dash-separated-string to camelCase
function camelCase(s) {
- return s.toLowerCase().replace(/-(.)/g, function(m, g) {
- return g.toUpperCase()
- })
+ return s.toLowerCase().replace(/-(.)/g, function(m, g) {
+ return g.toUpperCase()
+ })
}
// Capitalize first letter of a string
function capitalize(s) {
- return s.charAt(0).toUpperCase() + s.slice(1)
+ return s.charAt(0).toUpperCase() + s.slice(1)
}
// Ensure to six-based hex
function fullHex(hex) {
- return hex.length == 4 ?
- [ '#',
- hex.substring(1, 2), hex.substring(1, 2)
- , hex.substring(2, 3), hex.substring(2, 3)
- , hex.substring(3, 4), hex.substring(3, 4)
- ].join('') : hex
+ return hex.length == 4 ?
+ [ '#',
+ hex.substring(1, 2), hex.substring(1, 2)
+ , hex.substring(2, 3), hex.substring(2, 3)
+ , hex.substring(3, 4), hex.substring(3, 4)
+ ].join('') : hex
}
// Component to hex value
function compToHex(comp) {
- var hex = comp.toString(16)
- return hex.length == 1 ? '0' + hex : hex
+ var hex = comp.toString(16)
+ return hex.length == 1 ? '0' + hex : hex
}
// Calculate proportional width and height values when necessary
function proportionalSize(box, width, height) {
- if (height == null)
- height = box.height / box.width * width
- else if (width == null)
- width = box.width / box.height * height
-
- return {
- width: width
- , height: height
- }
+ if (height == null)
+ height = box.height / box.width * width
+ else if (width == null)
+ width = box.width / box.height * height
+
+ return {
+ width: width
+ , height: height
+ }
}
// Delta transform point
function deltaTransformPoint(matrix, x, y) {
- return {
- x: x * matrix.a + y * matrix.c + 0
- , y: x * matrix.b + y * matrix.d + 0
- }
+ return {
+ x: x * matrix.a + y * matrix.c + 0
+ , y: x * matrix.b + y * matrix.d + 0
+ }
}
// Map matrix array to object
function arrayToMatrix(a) {
- return { a: a[0], b: a[1], c: a[2], d: a[3], e: a[4], f: a[5] }
+ return { a: a[0], b: a[1], c: a[2], d: a[3], e: a[4], f: a[5] }
}
// Parse matrix if required
function parseMatrix(matrix) {
- if (!(matrix instanceof SVG.Matrix))
- matrix = new SVG.Matrix(matrix)
-
- return matrix
+ if (!(matrix instanceof SVG.Matrix))
+ matrix = new SVG.Matrix(matrix)
+
+ return matrix
}
// Convert string to matrix
function stringToMatrix(source) {
- // remove matrix wrapper and split to individual numbers
- source = source
- .replace(SVG.regex.whitespace, '')
- .replace(SVG.regex.matrix, '')
- .split(',')
-
- // convert string values to floats and convert to a matrix-formatted object
- return arrayToMatrix(
- SVG.utils.map(source, function(n) {
- return parseFloat(n)
- })
- )
+ // remove matrix wrapper and split to individual numbers
+ source = source
+ .replace(SVG.regex.whitespace, '')
+ .replace(SVG.regex.matrix, '')
+ .split(',')
+
+ // convert string values to floats and convert to a matrix-formatted object
+ return arrayToMatrix(
+ SVG.utils.map(source, function(n) {
+ return parseFloat(n)
+ })
+ )
}
// Calculate position according to from and to
function at(o, pos) {
- // number recalculation (don't bother converting to SVG.Number for performance reasons)
- return typeof o.from == 'number' ?
- o.from + (o.to - o.from) * pos :
-
- // instance recalculation
- o instanceof SVG.Color || o instanceof SVG.Number || o instanceof SVG.Matrix ? o.at(pos) :
-
- // for all other values wait until pos has reached 1 to return the final value
- pos < 1 ? o.from : o.to
+ // number recalculation (don't bother converting to SVG.Number for performance reasons)
+ return typeof o.from == 'number' ?
+ o.from + (o.to - o.from) * pos :
+
+ // instance recalculation
+ o instanceof SVG.Color || o instanceof SVG.Number || o instanceof SVG.Matrix ? o.at(pos) :
+
+ // for all other values wait until pos has reached 1 to return the final value
+ pos < 1 ? o.from : o.to
}
// PathArray Helpers
function arrayToString(a) {
- for (var i = 0, il = a.length, s = ''; i < il; i++) {
- s += a[i][0]
-
- if (a[i][1] != null) {
- s += a[i][1]
-
- if (a[i][2] != null) {
- s += ' '
- s += a[i][2]
-
- if (a[i][3] != null) {
- s += ' '
- s += a[i][3]
- s += ' '
- s += a[i][4]
-
- if (a[i][5] != null) {
- s += ' '
- s += a[i][5]
- s += ' '
- s += a[i][6]
-
- if (a[i][7] != null) {
- s += ' '
- s += a[i][7]
- }
- }
- }
- }
- }
- }
-
- return s + ' '
+ for (var i = 0, il = a.length, s = ''; i < il; i++) {
+ s += a[i][0]
+
+ if (a[i][1] != null) {
+ s += a[i][1]
+
+ if (a[i][2] != null) {
+ s += ' '
+ s += a[i][2]
+
+ if (a[i][3] != null) {
+ s += ' '
+ s += a[i][3]
+ s += ' '
+ s += a[i][4]
+
+ if (a[i][5] != null) {
+ s += ' '
+ s += a[i][5]
+ s += ' '
+ s += a[i][6]
+
+ if (a[i][7] != null) {
+ s += ' '
+ s += a[i][7]
+ }
+ }
+ }
+ }
+ }
+ }
+
+ return s + ' '
}
// Deep new id assignment
function assignNewId(node) {
- // Do the same for SVG child nodes as well
- for (var i = node.childNodes.length - 1; i >= 0; i--)
- if (node.childNodes[i] instanceof SVGElement)
- assignNewId(node.childNodes[i])
+ // do the same for SVG child nodes as well
+ for (var i = node.childNodes.length - 1; i >= 0; i--)
+ if (node.childNodes[i] instanceof SVGElement)
+ assignNewId(node.childNodes[i])
- return SVG.adopt(node).id(SVG.eid(node.nodeName))
+ return SVG.adopt(node).id(SVG.eid(node.nodeName))
}
// Add more bounding box properties
function fullBox(b) {
- b.x2 = b.x + b.width
- b.y2 = b.y + b.height
- b.cx = b.x + b.width / 2
- b.cy = b.y + b.height / 2
-
- return b
+ if (b.x == null) {
+ b.x = 0
+ b.y = 0
+ b.width = 0
+ b.height = 0
+ }
+
+ b.w = b.width
+ b.h = b.height
+ b.x2 = b.x + b.width
+ b.y2 = b.y + b.height
+ b.cx = b.x + b.width / 2
+ b.cy = b.y + b.height / 2
+
+ return b
}
// Get id from reference string
function idFromReference(url) {
- var m = url.toString().match(SVG.regex.reference)
+ var m = url.toString().match(SVG.regex.reference)
- if (m) return m[1]
+ if (m) return m[1]
}
// Create matrix array for looping
@@ -158,9 +167,9 @@ var abcdef = 'abcdef'.split('')
// Shim layer with setTimeout fallback by Paul Irish
window.requestAnimFrame = (function(){
- return window.requestAnimationFrame ||
- window.webkitRequestAnimationFrame ||
- window.mozRequestAnimationFrame ||
- window.msRequestAnimationFrame ||
- function (c) { window.setTimeout(c, 1000 / 60) }
+ return window.requestAnimationFrame ||
+ window.webkitRequestAnimationFrame ||
+ window.mozRequestAnimationFrame ||
+ window.msRequestAnimationFrame ||
+ function (c) { window.setTimeout(c, 1000 / 60) }
})() \ No newline at end of file
diff --git a/src/matrix.js b/src/matrix.js
index 7ee2874..26b7613 100644
--- a/src/matrix.js
+++ b/src/matrix.js
@@ -18,6 +18,9 @@ SVG.Matrix = SVG.invent({
this[abcdef[i]] = typeof source[abcdef[i]] === 'number' ?
source[abcdef[i]] : base[abcdef[i]]
+ // merge polymertic values
+ if (source._r)
+ this._r = source._r
}
// Add methods
@@ -52,6 +55,10 @@ SVG.Matrix = SVG.invent({
// store new destination
this.destination = new SVG.Matrix(matrix)
+ // detect polymetric rotation
+ if (this.destination._r)
+ this._r = this.extract()
+
return this
}
// Get morphed matrix at a given position
@@ -60,7 +67,7 @@ SVG.Matrix = SVG.invent({
if (!this.destination) return this
// calculate morphed matrix at a given position
- return new SVG.Matrix({
+ var matrix = new SVG.Matrix({
a: this.a + (this.destination.a - this.a) * pos
, b: this.b + (this.destination.b - this.b) * pos
, c: this.c + (this.destination.c - this.c) * pos
@@ -68,6 +75,16 @@ SVG.Matrix = SVG.invent({
, e: this.e + (this.destination.e - this.e) * pos
, f: this.f + (this.destination.f - this.f) * pos
})
+
+ // add polymetric rotation if required
+ if (this._r && this.destination._r)
+ matrix = matrix.rotate(
+ this._r.rotation + (this.destination._r.rotation - this._r.rotation) * pos
+ , this.destination._r.cx
+ , this.destination._r.cy
+ )
+
+ return matrix
}
// Multiplies by given matrix
, multiply: function(matrix) {
diff --git a/src/transform.js b/src/transform.js
index da2bf37..c4f8f70 100644
--- a/src/transform.js
+++ b/src/transform.js
@@ -26,16 +26,21 @@ SVG.extend(SVG.Element, SVG.FX, {
// absolute
new SVG.Matrix(o)
- // act on rotate
+ // act on rotation
} else if (o.rotation != null) {
o.cx = o.cx == null ? target.bbox().cx : o.cx
o.cy = o.cy == null ? target.bbox().cy : o.cy
- matrix = relative ?
- // relative
- target.attr('transform', matrix + ' rotate(' + [o.rotation, o.cx, o.cy].join() + ')').ctm() :
- // absolute
- matrix.rotate(o.rotation - matrix.extract().rotation, o.cx, o.cy)
+ if (this instanceof SVG.FX) {
+ o.rotation -= (relative ? 0 : matrix.extract().rotation)
+ matrix._r = o
+ } else {
+ matrix = relative ?
+ // relative
+ target.attr('transform', matrix + ' rotate(' + [o.rotation, o.cx, o.cy].join() + ')').ctm() :
+ // absolute
+ matrix.rotate(o.rotation - matrix.extract().rotation, o.cx, o.cy)
+ }
// act on scale
} else if (o.scale != null || o.scaleX != null || o.scaleY != null) {
@@ -86,7 +91,7 @@ SVG.extend(SVG.Element, SVG.FX, {
if (o.y != null) matrix.f = o.y
}
}
-
+
return this.attr('transform', matrix)
}
})