mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
Start implementing rounded rectangles in Shape.Rectangle.
This commit is contained in:
parent
d8f7799fc4
commit
2263afea59
2 changed files with 155 additions and 48 deletions
57
examples/SVG Export/Shapes.html
Normal file
57
examples/SVG Export/Shapes.html
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>Shapes</title>
|
||||||
|
<link rel="stylesheet" href="../css/style.css">
|
||||||
|
<script type="text/javascript" src="../../dist/paper.js"></script>
|
||||||
|
<script type="text/paperscript" canvas="canvas">
|
||||||
|
var rect = new Shape.Rectangle({
|
||||||
|
point: [200, 100],
|
||||||
|
size: [200, 300],
|
||||||
|
fillColor: 'red'
|
||||||
|
});
|
||||||
|
rect.rotate(40);
|
||||||
|
|
||||||
|
var circle = new Shape.Circle({
|
||||||
|
center: [200, 300],
|
||||||
|
radius: 100,
|
||||||
|
fillColor: 'green'
|
||||||
|
});
|
||||||
|
circle.scale(0.5, 1);
|
||||||
|
circle.rotate(40);
|
||||||
|
|
||||||
|
var ellipse = new Shape.Ellipse({
|
||||||
|
point: [300, 300],
|
||||||
|
size: [100, 200],
|
||||||
|
fillColor: 'blue'
|
||||||
|
});
|
||||||
|
ellipse.rotate(-40);
|
||||||
|
|
||||||
|
var rect = new Shape.Rectangle({
|
||||||
|
point: [250, 20],
|
||||||
|
size: [200, 300],
|
||||||
|
radius: [40, 20],
|
||||||
|
fillColor: 'yellow'
|
||||||
|
});
|
||||||
|
rect.rotate(-20);
|
||||||
|
document.getElementById('svg').appendChild(project.exportSVG());
|
||||||
|
|
||||||
|
var prev = null;
|
||||||
|
function onMouseMove(event) {
|
||||||
|
if (prev)
|
||||||
|
prev.selected = false;
|
||||||
|
var result = project.hitTest(event.point);
|
||||||
|
if (result) {
|
||||||
|
var item = result.item;
|
||||||
|
item.selected = true;
|
||||||
|
prev = item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<canvas id="canvas" width="500" height="500"></canvas>
|
||||||
|
<svg id="svg" width="500" height="500"></svg>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -20,11 +20,15 @@
|
||||||
var Shape = Item.extend(/** @lends Shape# */{
|
var Shape = Item.extend(/** @lends Shape# */{
|
||||||
_class: 'Shape',
|
_class: 'Shape',
|
||||||
_transformContent: false,
|
_transformContent: false,
|
||||||
|
_boundsSelected: true,
|
||||||
|
|
||||||
initialize: function Shape(type, point, size, props) {
|
// TODO: SVG, serialization
|
||||||
this._initialize(props, point);
|
|
||||||
|
initialize: function Shape(type, point, size, radius, props) {
|
||||||
this._type = type;
|
this._type = type;
|
||||||
this._size = size;
|
this._size = size;
|
||||||
|
this._radius = radius;
|
||||||
|
this._initialize(props, point);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -39,28 +43,55 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
setSize: function(/* size */) {
|
setSize: function(/* size */) {
|
||||||
var size = Size.read(arguments);
|
var type = this._type,
|
||||||
|
size = Size.read(arguments);
|
||||||
if (!this._size.equals(size)) {
|
if (!this._size.equals(size)) {
|
||||||
this._size.set(size.width, size.height);
|
var width = size.width,
|
||||||
|
height = size.height;
|
||||||
|
if (type === 'circle') {
|
||||||
|
// Use average of width and height as new size, then calculate
|
||||||
|
// radius as a number from that:
|
||||||
|
width = height = (width + height) / 2;
|
||||||
|
this._radius = width / 2;
|
||||||
|
} else if (type === 'ellipse') {
|
||||||
|
// The radius is a size.
|
||||||
|
this._radius.set(width / 2, height / 2);
|
||||||
|
}
|
||||||
|
this._size.set(width, height);
|
||||||
this._changed(/*#=*/ Change.GEOMETRY);
|
this._changed(/*#=*/ Change.GEOMETRY);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The radius of the shape if it is a circle.
|
* The radius of the shape, as a number if it is a circle, or a size object
|
||||||
|
* for ellipses and rounded rectangles.
|
||||||
*
|
*
|
||||||
* @type Size
|
* @type Number|Size
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getRadius: function() {
|
getRadius: function() {
|
||||||
var size = this._size;
|
var rad = this._radius;
|
||||||
// Average half of width & height for radius...
|
return this._type === 'circle'
|
||||||
return (size.width + size.height) / 4;
|
? rad
|
||||||
|
: new LinkedSize(rad.width, rad.height, this, 'setRadius');
|
||||||
},
|
},
|
||||||
|
|
||||||
setRadius: function(radius) {
|
setRadius: function(radius) {
|
||||||
|
var type = this._type;
|
||||||
|
if (type === 'circle') {
|
||||||
|
if (radius === this._radius)
|
||||||
|
return;
|
||||||
var size = radius * 2;
|
var size = radius * 2;
|
||||||
this.setSize(size, size);
|
this._size.set(size, size);
|
||||||
|
} else {
|
||||||
|
radius = Size.read(arguments);
|
||||||
|
if (this._radius.equals(radius))
|
||||||
|
return;
|
||||||
|
this._radius.set(radius.width, radius.height);
|
||||||
|
if (type === 'ellipse')
|
||||||
|
this._size.set(radius.width * 2, radius.height * 2);
|
||||||
|
}
|
||||||
|
this._changed(/*#=*/ Change.GEOMETRY);
|
||||||
},
|
},
|
||||||
|
|
||||||
isEmpty: function() {
|
isEmpty: function() {
|
||||||
|
@ -72,34 +103,52 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
|
|
||||||
_draw: function(ctx, param) {
|
_draw: function(ctx, param) {
|
||||||
var style = this._style,
|
var style = this._style,
|
||||||
size = this._size,
|
|
||||||
width = size.width,
|
|
||||||
height = size.height,
|
|
||||||
fillColor = style.getFillColor(),
|
fillColor = style.getFillColor(),
|
||||||
strokeColor = style.getStrokeColor();
|
strokeColor = style.getStrokeColor();
|
||||||
if (fillColor || strokeColor || param.clip) {
|
if (fillColor || strokeColor || param.clip) {
|
||||||
|
var radius = this._radius,
|
||||||
|
type = this._type;
|
||||||
ctx.beginPath();
|
ctx.beginPath();
|
||||||
switch (this._type) {
|
if (type === 'circle') {
|
||||||
case 'rect':
|
ctx.arc(0, 0, radius, 0, Math.PI * 2, true);
|
||||||
ctx.rect(-width / 2, -height / 2, width, height);
|
} else {
|
||||||
break;
|
var rx = radius.width,
|
||||||
case 'circle':
|
ry = radius.height,
|
||||||
// Average half of width & height for radius...
|
kappa = Numerical.KAPPA;
|
||||||
ctx.arc(0, 0, (width + height) / 4, 0, Math.PI * 2, true);
|
if (type === 'ellipse') {
|
||||||
break;
|
|
||||||
case 'ellipse':
|
|
||||||
// Use four bezier curves and KAPPA value to aproximate ellipse
|
// Use four bezier curves and KAPPA value to aproximate ellipse
|
||||||
var mx = width / 2,
|
var cx = rx * kappa,
|
||||||
my = height / 2,
|
cy = ry * kappa;
|
||||||
kappa = Numerical.KAPPA,
|
ctx.moveTo(-rx, 0);
|
||||||
cx = mx * kappa,
|
ctx.bezierCurveTo(-rx, -cy, -cx, -ry, 0, -ry);
|
||||||
cy = my * kappa;
|
ctx.bezierCurveTo(cx, -ry, rx, -cy, rx, 0);
|
||||||
ctx.moveTo(-mx, 0);
|
ctx.bezierCurveTo(rx, cy, cx, ry, 0, ry);
|
||||||
ctx.bezierCurveTo(-mx, -cy, -cx, -my, 0, -my);
|
ctx.bezierCurveTo(-cx, ry, -rx, cy, -rx, 0);
|
||||||
ctx.bezierCurveTo(cx, -my, mx, -cy, mx, 0);
|
} else { // rect
|
||||||
ctx.bezierCurveTo(mx, cy, cx, my, 0, my);
|
var size = this._size,
|
||||||
ctx.bezierCurveTo(-cx, my, -mx, cy, -mx, 0);
|
width = size.width,
|
||||||
break;
|
height = size.height;
|
||||||
|
if (rx === 0 && ry === 0) {
|
||||||
|
// straight rect
|
||||||
|
ctx.rect(-width / 2, -height / 2, width, height);
|
||||||
|
} else {
|
||||||
|
// rounded rect. Use inverse kappa to calculate position
|
||||||
|
// of control points from the corners inwards.
|
||||||
|
kappa = 1 - kappa;
|
||||||
|
var x = width / 2,
|
||||||
|
y = height / 2,
|
||||||
|
cx = rx * kappa,
|
||||||
|
cy = ry * kappa;
|
||||||
|
ctx.moveTo(-x, -y + ry);
|
||||||
|
ctx.bezierCurveTo(-x, -y + cy, -x + cx, -y, -x + rx, -y);
|
||||||
|
ctx.lineTo(x - rx, -y);
|
||||||
|
ctx.bezierCurveTo(x - cx, -y, x, -y + cy, x, -y + ry);
|
||||||
|
ctx.lineTo(x, y - ry);
|
||||||
|
ctx.bezierCurveTo(x, y - cy, x - cx, y, x - rx, y);
|
||||||
|
ctx.lineTo(-x + rx, y);
|
||||||
|
ctx.bezierCurveTo(-x + cx, y, -x, y - cy, -x, y - ry);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!param.clip && (fillColor || strokeColor)) {
|
if (!param.clip && (fillColor || strokeColor)) {
|
||||||
|
@ -148,19 +197,18 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
break;
|
break;
|
||||||
case 'circle':
|
case 'circle':
|
||||||
case 'ellipse':
|
case 'ellipse':
|
||||||
var size = this._size,
|
var radius;
|
||||||
width = size.width,
|
|
||||||
height = size.height,
|
|
||||||
radius;
|
|
||||||
if (type === 'ellipse') {
|
if (type === 'ellipse') {
|
||||||
// Calculate ellipse radius at angle
|
// Calculate ellipse radius at angle
|
||||||
var angle = point.getAngleInRadians(),
|
var angle = point.getAngleInRadians(),
|
||||||
|
size = this._size,
|
||||||
|
width = size.width,
|
||||||
|
height = size.height,
|
||||||
x = width * Math.sin(angle),
|
x = width * Math.sin(angle),
|
||||||
y = height * Math.cos(angle);
|
y = height * Math.cos(angle);
|
||||||
radius = width * height / (2 * Math.sqrt(x * x + y * y));
|
radius = width * height / (2 * Math.sqrt(x * x + y * y));
|
||||||
} else {
|
} else {
|
||||||
// Average half of width & height for radius...
|
radius = this._radius;
|
||||||
radius = (width + height) / 4;
|
|
||||||
}
|
}
|
||||||
if (2 * Math.abs(point.getLength() - radius) <= strokeWidth)
|
if (2 * Math.abs(point.getLength() - radius) <= strokeWidth)
|
||||||
return new HitResult('stroke', this);
|
return new HitResult('stroke', this);
|
||||||
|
@ -171,8 +219,8 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
statics: new function() {
|
statics: new function() {
|
||||||
function createShape(type, point, size, args) {
|
function createShape(type, point, size, radius, args) {
|
||||||
return new Shape(type, point, size, Base.getNamed(args));
|
return new Shape(type, point, size, radius, Base.getNamed(args));
|
||||||
}
|
}
|
||||||
|
|
||||||
return /** @lends Shape */{
|
return /** @lends Shape */{
|
||||||
|
@ -198,7 +246,7 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
var center = Point.readNamed(arguments, 'center'),
|
var center = Point.readNamed(arguments, 'center'),
|
||||||
radius = Base.readNamed(arguments, 'radius');
|
radius = Base.readNamed(arguments, 'radius');
|
||||||
return createShape('circle', center, new Size(radius * 2),
|
return createShape('circle', center, new Size(radius * 2),
|
||||||
arguments);
|
radius, arguments);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -276,7 +324,8 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
Rectangle: function(/* rectangle */) {
|
Rectangle: function(/* rectangle */) {
|
||||||
var rect = Rectangle.readNamed(arguments, 'rectangle');
|
var rect = Rectangle.readNamed(arguments, 'rectangle');
|
||||||
return createShape('rect', rect.getCenter(true),
|
return createShape('rect', rect.getCenter(true),
|
||||||
rect.getSize(true), arguments);
|
rect.getSize(true), Size.readNamed(arguments, 'radius'),
|
||||||
|
arguments);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -301,9 +350,10 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
* });
|
* });
|
||||||
*/
|
*/
|
||||||
Ellipse: function(/* rectangle */) {
|
Ellipse: function(/* rectangle */) {
|
||||||
var rect = Rectangle.readNamed(arguments, 'rectangle');
|
var rect = Rectangle.readNamed(arguments, 'rectangle'),
|
||||||
return createShape('ellipse', rect.getCenter(true),
|
size = rect.getSize(true);
|
||||||
rect.getSize(true), arguments);
|
return createShape('ellipse', rect.getCenter(true), size,
|
||||||
|
new Size(size.width / 2, size.height / 2), arguments);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue