Start implementing rounded rectangles in Shape.Rectangle.

This commit is contained in:
Jürg Lehni 2013-10-14 19:57:28 +02:00
parent d8f7799fc4
commit 2263afea59
2 changed files with 155 additions and 48 deletions

View 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>

View file

@ -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);
} }
}; };
} }