/** NSBezierPath.m The NSBezierPath class Copyright (C) 1999, 2005 Free Software Foundation, Inc. Author: Enrico Sersale Date: Dec 1999 Modified: Fred Kiefer Date: January 2001 This file is part of the GNUstep GUI Library. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; see the file COPYING.LIB. If not, see or write to the Free Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #import #import "AppKit/NSAffineTransform.h" #import "AppKit/NSBezierPath.h" #import "AppKit/NSFont.h" #import "AppKit/NSImage.h" #import "AppKit/PSOperators.h" #import "GNUstepGUI/GSFontInfo.h" #import "GSGuiPrivate.h" #include #ifndef PI #define PI 3.1415926535897932384626434 #endif // This magic number is 4 *(sqrt(2) -1)/3 #define KAPPA 0.5522847498 #define INVALIDATE_CACHE() [self _invalidateCache] static void flatten(NSPoint coeff[], float flatness, NSBezierPath *path); static Class NSBezierPath_concrete_class = nil; static NSWindingRule default_winding_rule = NSNonZeroWindingRule; static float default_line_width = 1.0; static float default_flatness = 0.6; static NSLineJoinStyle default_line_join_style = NSMiterLineJoinStyle; static NSLineCapStyle default_line_cap_style = NSButtLineCapStyle; static float default_miter_limit = 10.0; @interface NSBezierPath (PrivateMethods) - (void)_invalidateCache; - (void)_recalculateBounds; @end typedef struct _PathElement { NSBezierPathElement type; NSPoint points[3]; } PathElement; //#define GSUNION_TYPES GSUNION_OBJ #define GSI_ARRAY_TYPES 0 #define GSI_ARRAY_TYPE PathElement #define GSI_ARRAY_NO_RETAIN #define GSI_ARRAY_NO_RELEASE #ifdef GSIArray #undef GSIArray #endif #include @interface GSBezierPath : NSBezierPath { GSIArray pathElements; BOOL flat; } @end @implementation NSBezierPath + (void)initialize { if (self == [NSBezierPath class]) NSBezierPath_concrete_class = [GSBezierPath class]; } + (void)_setConcreteClass:(Class)c { NSBezierPath_concrete_class = c; } + (Class)_concreteClass { return NSBezierPath_concrete_class; } // // Creating common paths // + (id) allocWithZone: (NSZone*)z { if (self != NSBezierPath_concrete_class) { return [NSBezierPath_concrete_class allocWithZone: z]; } else { return NSAllocateObject (self, 0, z); } } + (NSBezierPath *)bezierPath { return AUTORELEASE ([[NSBezierPath_concrete_class alloc] init]); } + (NSBezierPath *)bezierPathWithRect: (NSRect)aRect { NSBezierPath *path; path = [self bezierPath]; [path appendBezierPathWithRect: aRect]; return path; } + (NSBezierPath *)bezierPathWithOvalInRect: (NSRect)aRect { NSBezierPath *path; path = [self bezierPath]; [path appendBezierPathWithOvalInRect: aRect]; return path; } + (NSBezierPath *)bezierPathWithRoundedRect: (NSRect)aRect xRadius: (CGFloat)xRadius yRadius: (CGFloat)yRadius { NSBezierPath *path; path = [self bezierPath]; [path appendBezierPathWithRoundedRect: aRect xRadius: xRadius yRadius: yRadius]; return path; } // // Immediate mode drawing of common paths // + (void)fillRect: (NSRect)aRect { PSrectfill(NSMinX(aRect), NSMinY(aRect), NSWidth(aRect), NSHeight(aRect)); } + (void)strokeRect: (NSRect)aRect { PSrectstroke(NSMinX(aRect), NSMinY(aRect), NSWidth(aRect), NSHeight(aRect)); } + (void)clipRect: (NSRect)aRect { PSrectclip(NSMinX(aRect), NSMinY(aRect), NSWidth(aRect), NSHeight(aRect)); } + (void)strokeLineFromPoint: (NSPoint)point1 toPoint: (NSPoint)point2 { NSBezierPath *path = [[self alloc] init]; [path moveToPoint: point1]; [path lineToPoint: point2]; [path stroke]; RELEASE(path); } + (void)drawPackedGlyphs: (const char *)packedGlyphs atPoint: (NSPoint)aPoint { NSBezierPath *path = [[self alloc] init]; [path moveToPoint: aPoint]; [path appendBezierPathWithPackedGlyphs: packedGlyphs]; [path fill]; RELEASE(path); } // // Default path rendering parameters // + (void)setDefaultMiterLimit:(float)limit { default_miter_limit = limit; // Do we need this? PSsetmiterlimit(limit); } + (float)defaultMiterLimit { return default_miter_limit; } + (void)setDefaultFlatness:(float)flatness { default_flatness = flatness; PSsetflat(flatness); } + (float)defaultFlatness { return default_flatness; } + (void)setDefaultWindingRule:(NSWindingRule)windingRule { default_winding_rule = windingRule; } + (NSWindingRule)defaultWindingRule { return default_winding_rule; } + (void)setDefaultLineCapStyle:(NSLineCapStyle)lineCapStyle { default_line_cap_style = lineCapStyle; PSsetlinecap(lineCapStyle); } + (NSLineCapStyle)defaultLineCapStyle { return default_line_cap_style; } + (void)setDefaultLineJoinStyle:(NSLineJoinStyle)lineJoinStyle { default_line_join_style = lineJoinStyle; PSsetlinejoin(lineJoinStyle); } + (NSLineJoinStyle)defaultLineJoinStyle { return default_line_join_style; } + (void)setDefaultLineWidth:(float)lineWidth { default_line_width = lineWidth; PSsetlinewidth(lineWidth); } + (float)defaultLineWidth { return default_line_width; } - (id) init { [super init]; // Those values come from the default. [self setLineWidth: default_line_width]; [self setFlatness: default_flatness]; [self setLineCapStyle: default_line_cap_style]; [self setLineJoinStyle: default_line_join_style]; [self setMiterLimit: default_miter_limit]; [self setWindingRule: default_winding_rule]; // Set by allocation //_bounds = NSZeroRect; //_controlPointBounds = NSZeroRect; //_cachesBezierPath = NO; //_cacheImage = nil; //_dash_count = 0; //_dash_phase = 0; //_dash_pattern = NULL; return self; } - (void) dealloc { if (_cacheImage != nil) RELEASE(_cacheImage); if (_dash_pattern != NULL) NSZoneFree([self zone], _dash_pattern); [super dealloc]; } // // Path construction // - (void)moveToPoint:(NSPoint)aPoint { [self subclassResponsibility:_cmd]; } - (void)lineToPoint:(NSPoint)aPoint { [self subclassResponsibility:_cmd]; } - (void)curveToPoint:(NSPoint)aPoint controlPoint1:(NSPoint)controlPoint1 controlPoint2:(NSPoint)controlPoint2 { [self subclassResponsibility:_cmd]; } - (void)closePath { [self subclassResponsibility:_cmd]; } - (void)removeAllPoints { [self subclassResponsibility:_cmd]; } // // Relative path construction // - (void)relativeMoveToPoint:(NSPoint)aPoint { NSPoint p = [self currentPoint]; p.x = p.x + aPoint.x; p.y = p.y + aPoint.y; [self moveToPoint: p]; } - (void)relativeLineToPoint:(NSPoint)aPoint { NSPoint p = [self currentPoint]; p.x = p.x + aPoint.x; p.y = p.y + aPoint.y; [self lineToPoint: p]; } - (void)relativeCurveToPoint:(NSPoint)aPoint controlPoint1:(NSPoint)controlPoint1 controlPoint2:(NSPoint)controlPoint2 { NSPoint p = [self currentPoint]; aPoint.x = p.x + aPoint.x; aPoint.y = p.y + aPoint.y; controlPoint1.x = p.x + controlPoint1.x; controlPoint1.y = p.y + controlPoint1.y; controlPoint2.x = p.x + controlPoint2.x; controlPoint2.y = p.y + controlPoint2.y; [self curveToPoint: aPoint controlPoint1: controlPoint1 controlPoint2: controlPoint2]; } // // Path rendering parameters // - (float)lineWidth { return _lineWidth; } - (void)setLineWidth:(float)lineWidth { _lineWidth = lineWidth; } - (NSLineCapStyle)lineCapStyle { return _lineCapStyle; } - (void)setLineCapStyle:(NSLineCapStyle)lineCapStyle { _lineCapStyle = lineCapStyle; } - (NSLineJoinStyle)lineJoinStyle { return _lineJoinStyle; } - (void)setLineJoinStyle:(NSLineJoinStyle)lineJoinStyle { _lineJoinStyle = lineJoinStyle; } - (NSWindingRule)windingRule { return _windingRule; } - (void)setWindingRule:(NSWindingRule)windingRule { _windingRule = windingRule; } - (void)setFlatness:(float)flatness { _flatness = flatness; } - (float)flatness { return _flatness; } - (void)setMiterLimit:(float)limit { _miterLimit = limit; } - (float)miterLimit { return _miterLimit; } - (void)getLineDash:(float *)pattern count:(int *)count phase:(float *)phase { // FIXME: How big is the pattern array? // We assume that this value is in count! if (count != NULL) { if (*count < _dash_count) { *count = _dash_count; return; } *count = _dash_count; } if (phase != NULL) *phase = _dash_phase; memcpy(pattern, _dash_pattern, _dash_count * sizeof(float)); } - (void)setLineDash:(const float *)pattern count:(int)count phase:(float)phase { NSZone *myZone = [self zone]; if ((pattern == NULL) || (count == 0)) { if (_dash_pattern != NULL) { NSZoneFree(myZone, _dash_pattern); _dash_pattern = NULL; } _dash_count = 0; _dash_phase = 0.0; return; } if (_dash_pattern == NULL) _dash_pattern = NSZoneMalloc(myZone, count * sizeof(float)); else NSZoneRealloc(myZone, _dash_pattern, count * sizeof(float)); _dash_count = count; _dash_phase = phase; memcpy(_dash_pattern, pattern, _dash_count * sizeof(float)); } // // Path operations // - (void)stroke { NSGraphicsContext *ctxt = GSCurrentContext(); if (_cachesBezierPath) { NSRect bounds = [self bounds]; NSPoint origin = bounds.origin; // FIXME: I don't see how this should work with color changes if (_cacheImage == nil) { _cacheImage = [[NSImage alloc] initWithSize: bounds.size]; [_cacheImage lockFocus]; DPStranslate(ctxt, -origin.x, -origin.y); [ctxt GSSendBezierPath: self]; DPSstroke(ctxt); [_cacheImage unlockFocus]; } [_cacheImage compositeToPoint: origin operation: NSCompositeCopy]; } else { [ctxt GSSendBezierPath: self]; DPSstroke(ctxt); } } - (void)fill { NSGraphicsContext *ctxt = GSCurrentContext(); if (_cachesBezierPath) { NSRect bounds = [self bounds]; NSPoint origin = bounds.origin; // FIXME: I don't see how this should work with color changes if (_cacheImage == nil) { _cacheImage = [[NSImage alloc] initWithSize: bounds.size]; [_cacheImage lockFocus]; DPStranslate(ctxt, -origin.x, -origin.y); [ctxt GSSendBezierPath: self]; if ([self windingRule] == NSNonZeroWindingRule) DPSfill(ctxt); else DPSeofill(ctxt); [_cacheImage unlockFocus]; } [_cacheImage compositeToPoint: origin operation: NSCompositeCopy]; } else { [ctxt GSSendBezierPath: self]; if ([self windingRule] == NSNonZeroWindingRule) DPSfill(ctxt); else DPSeofill(ctxt); } } - (void)addClip { NSGraphicsContext *ctxt = GSCurrentContext(); [ctxt GSSendBezierPath: self]; if ([self windingRule] == NSNonZeroWindingRule) DPSclip(ctxt); else DPSeoclip(ctxt); } - (void)setClip { NSGraphicsContext *ctxt = GSCurrentContext(); DPSinitclip(ctxt); [ctxt GSSendBezierPath: self]; if ([self windingRule] == NSNonZeroWindingRule) DPSclip(ctxt); else DPSeoclip(ctxt); } // // Path modifications. // - (NSBezierPath *)bezierPathByFlatteningPath { NSBezierPath *path = [object_getClass(self) bezierPath]; NSBezierPathElement type; NSPoint pts[3]; NSPoint coeff[4]; NSPoint p, last_p; int i, count; BOOL first = YES; /* Silence compiler warnings. */ p = NSZeroPoint; last_p = NSZeroPoint; count = [self elementCount]; for (i = 0; i < count; i++) { type = [self elementAtIndex: i associatedPoints: pts]; switch(type) { case NSMoveToBezierPathElement: [path moveToPoint: pts[0]]; last_p = p = pts[0]; first = NO; break; case NSLineToBezierPathElement: [path lineToPoint: pts[0]]; p = pts[0]; if (first) { last_p = pts[0]; first = NO; } break; case NSCurveToBezierPathElement: coeff[0] = p; coeff[1] = pts[0]; coeff[2] = pts[1]; coeff[3] = pts[2]; flatten(coeff, [self flatness], path); p = pts[2]; if (first) { last_p = pts[2]; first = NO; } break; case NSClosePathBezierPathElement: [path closePath]; p = last_p; break; default: break; } } return path; } - (NSBezierPath *) bezierPathByReversingPath { NSBezierPath *path = [object_getClass(self) bezierPath]; NSBezierPathElement type, last_type; NSPoint pts[3]; NSPoint p, cp1, cp2; int i, j, count; BOOL closed = NO; /* Silence compiler warnings. */ p = NSZeroPoint; last_type = NSMoveToBezierPathElement; count = [self elementCount]; for (i = count - 1; i >= 0; i--) { type = [self elementAtIndex: i associatedPoints: pts]; switch(type) { case NSMoveToBezierPathElement: p = pts[0]; break; case NSLineToBezierPathElement: p = pts[0]; break; case NSCurveToBezierPathElement: cp1 = pts[0]; cp2 = pts[1]; p = pts[2]; break; case NSClosePathBezierPathElement: // find the first point of segment for (j = i - 1; j >= 0; j--) { type = [self elementAtIndex: i associatedPoints: pts]; if (type == NSMoveToBezierPathElement) { p = pts[0]; break; } } // FIXME: What to do if we don't find a move element? break; default: break; } switch(last_type) { case NSMoveToBezierPathElement: if (closed) { [path closePath]; closed = NO; } [path moveToPoint: p]; break; case NSLineToBezierPathElement: [path lineToPoint: p]; break; case NSCurveToBezierPathElement: [path curveToPoint: p controlPoint1: cp2 controlPoint2: cp1]; break; case NSClosePathBezierPathElement: closed = YES; break; default: break; } last_type = type; } if (closed) [path closePath]; return path; } // // Applying transformations. // - (void) transformUsingAffineTransform: (NSAffineTransform *)transform { NSBezierPathElement type; NSPoint pts[3]; int i, count; count = [self elementCount]; for (i = 0; i < count; i++) { type = [self elementAtIndex: i associatedPoints: pts]; switch(type) { case NSMoveToBezierPathElement: case NSLineToBezierPathElement: pts[0] = [transform transformPoint: pts[0]]; [self setAssociatedPoints: pts atIndex: i]; break; case NSCurveToBezierPathElement: pts[0] = [transform transformPoint: pts[0]]; pts[1] = [transform transformPoint: pts[1]]; pts[2] = [transform transformPoint: pts[2]]; [self setAssociatedPoints: pts atIndex: i]; break; case NSClosePathBezierPathElement: break; default: break; } } INVALIDATE_CACHE(); } // // Path info // - (BOOL) isEmpty { return ([self elementCount] == 0); } - (NSPoint) currentPoint { NSBezierPathElement type; NSPoint points[3]; int i, count; count = [self elementCount]; if (!count) [NSException raise: NSGenericException format: @"No current Point in NSBezierPath"]; type = [self elementAtIndex: count - 1 associatedPoints: points]; switch(type) { case NSMoveToBezierPathElement: case NSLineToBezierPathElement: return points[0]; break; case NSCurveToBezierPathElement: return points[2]; break; case NSClosePathBezierPathElement: // We have to find the last move element and take its point for (i = count - 2; i >= 0; i--) { type = [self elementAtIndex: i associatedPoints: points]; if (type == NSMoveToBezierPathElement) return points[0]; } break; default: break; } return NSZeroPoint; } - (NSRect) controlPointBounds { if (_shouldRecalculateBounds) [self _recalculateBounds]; return _controlPointBounds; } - (NSRect) bounds { if (_shouldRecalculateBounds) [self _recalculateBounds]; return _bounds; } // // Elements // - (int) elementCount { [self subclassResponsibility:_cmd]; return 0; } - (NSBezierPathElement) elementAtIndex: (int)index associatedPoints: (NSPoint *)points { [self subclassResponsibility:_cmd]; return 0; } - (NSBezierPathElement) elementAtIndex: (int)index { return [self elementAtIndex: index associatedPoints: NULL]; } - (void)setAssociatedPoints:(NSPoint *)points atIndex:(int)index { [self subclassResponsibility:_cmd]; } // // Appending common paths // - (void) appendBezierPath: (NSBezierPath *)aPath { NSBezierPathElement type; NSPoint points[3]; int i, count; count = [aPath elementCount]; for (i = 0; i < count; i++) { type = [aPath elementAtIndex: i associatedPoints: points]; switch(type) { case NSMoveToBezierPathElement: [self moveToPoint: points[0]]; break; case NSLineToBezierPathElement: [self lineToPoint: points[0]]; break; case NSCurveToBezierPathElement: [self curveToPoint: points[2] controlPoint1: points[0] controlPoint2: points[1]]; break; case NSClosePathBezierPathElement: [self closePath]; break; default: break; } } } - (void)appendBezierPathWithRect:(NSRect)aRect { NSPoint p; [self moveToPoint: aRect.origin]; p.x = aRect.origin.x + aRect.size.width; p.y = aRect.origin.y; [self lineToPoint: p]; p.x = aRect.origin.x + aRect.size.width; p.y = aRect.origin.y + aRect.size.height; [self lineToPoint: p]; p.x = aRect.origin.x; p.y = aRect.origin.y + aRect.size.height; [self lineToPoint: p]; [self closePath]; } - (void)appendBezierPathWithPoints:(NSPoint *)points count:(int)count { int i; if (!count) return; if ([self isEmpty]) { [self moveToPoint: points[0]]; } else { [self lineToPoint: points[0]]; } for (i = 1; i < count; i++) { [self lineToPoint: points[i]]; } } - (void) appendBezierPathWithOvalInRect: (NSRect)aRect { NSPoint p, p1, p2; double originx = aRect.origin.x; double originy = aRect.origin.y; double width = aRect.size.width; double height = aRect.size.height; double hdiff = width / 2 * KAPPA; double vdiff = height / 2 * KAPPA; p = NSMakePoint(originx + width / 2, originy + height); [self moveToPoint: p]; p = NSMakePoint(originx, originy + height / 2); p1 = NSMakePoint(originx + width / 2 - hdiff, originy + height); p2 = NSMakePoint(originx, originy + height / 2 + vdiff); [self curveToPoint: p controlPoint1: p1 controlPoint2: p2]; p = NSMakePoint(originx + width / 2, originy); p1 = NSMakePoint(originx, originy + height / 2 - vdiff); p2 = NSMakePoint(originx + width / 2 - hdiff, originy); [self curveToPoint: p controlPoint1: p1 controlPoint2: p2]; p = NSMakePoint(originx + width, originy + height / 2); p1 = NSMakePoint(originx + width / 2 + hdiff, originy); p2 = NSMakePoint(originx + width, originy + height / 2 - vdiff); [self curveToPoint: p controlPoint1: p1 controlPoint2: p2]; p = NSMakePoint(originx + width / 2, originy + height); p1 = NSMakePoint(originx + width, originy + height / 2 + vdiff); p2 = NSMakePoint(originx + width / 2 + hdiff, originy + height); [self curveToPoint: p controlPoint1: p1 controlPoint2: p2]; } /* startAngle and endAngle are in degrees, counterclockwise, from the x axis */ - (void) appendBezierPathWithArcWithCenter: (NSPoint)center radius: (float)radius startAngle: (float)startAngle endAngle: (float)endAngle clockwise: (BOOL)clockwise { float startAngle_rad, endAngle_rad, diff; NSPoint p0, p1, p2, p3; /* We use the Postscript prescription for managing the angles and drawing the arc. See the documentation for `arc' and `arcn' in the Postscript Reference. */ if (clockwise) { /* This modification of the angles is the postscript prescription. */ while (startAngle < endAngle) endAngle -= 360; /* This is used when we draw a clockwise quarter of circumference. By adding diff at the starting angle of the quarter, we get the ending angle. diff is negative because we draw clockwise. */ diff = - PI / 2; } else { /* This modification of the angles is the postscript prescription. */ while (endAngle < startAngle) endAngle += 360; /* This is used when we draw a counterclockwise quarter of circumference. By adding diff at the starting angle of the quarter, we get the ending angle. diff is positive because we draw counterclockwise. */ diff = PI / 2; } /* Convert the angles to radians */ startAngle_rad = PI * startAngle / 180; endAngle_rad = PI * endAngle / 180; /* Start point */ p0 = NSMakePoint (center.x + radius * cos (startAngle_rad), center.y + radius * sin (startAngle_rad)); if ([self elementCount] == 0) { [self moveToPoint: p0]; } else { NSPoint ps = [self currentPoint]; if (p0.x != ps.x || p0.y != ps.y) { [self lineToPoint: p0]; } } while ((clockwise) ? (startAngle_rad > endAngle_rad) : (startAngle_rad < endAngle_rad)) { /* Add a quarter circle */ if ((clockwise) ? (startAngle_rad + diff >= endAngle_rad) : (startAngle_rad + diff <= endAngle_rad)) { float sin_start = sin (startAngle_rad); float cos_start = cos (startAngle_rad); float sign = (clockwise) ? -1.0 : 1.0; p1 = NSMakePoint (center.x + radius * (cos_start - KAPPA * sin_start * sign), center.y + radius * (sin_start + KAPPA * cos_start * sign)); p2 = NSMakePoint (center.x + radius * (-sin_start * sign + KAPPA * cos_start), center.y + radius * (cos_start * sign + KAPPA * sin_start)); p3 = NSMakePoint (center.x + radius * (-sin_start * sign), center.y + radius * cos_start * sign); [self curveToPoint: p3 controlPoint1: p1 controlPoint2: p2]; startAngle_rad += diff; } else { /* Add the missing bit * We require that the arc be less than a semicircle. * The arc may go either clockwise or counterclockwise. * The approximation is a very simple one: a single curve * whose middle two control points are a fraction F of the way * to the intersection of the tangents, where * F = (4/3) / (1 + sqrt (1 + (d / r)^2)) * where r is the radius and d is the distance from either tangent * point to the intersection of the tangents. This produces * a curve whose center point, as well as its ends, lies on * the desired arc. */ NSPoint ps = [self currentPoint]; /* tangent is the tangent of half the angle */ float tangent = tan ((endAngle_rad - startAngle_rad) / 2); /* trad is the distance from either tangent point to the intersection of the tangents */ float trad = radius * tangent; /* pt is the intersection of the tangents */ NSPoint pt = NSMakePoint (ps.x - trad * sin (startAngle_rad), ps.y + trad * cos (startAngle_rad)); /* This is F - in this expression we need to compute (trad/radius)^2, which is simply tangent^2 */ float f = (4.0 / 3.0) / (1.0 + sqrt (1.0 + (tangent * tangent))); p1 = NSMakePoint (ps.x + (pt.x - ps.x) * f, ps.y + (pt.y - ps.y) * f); p3 = NSMakePoint(center.x + radius * cos (endAngle_rad), center.y + radius * sin (endAngle_rad)); p2 = NSMakePoint (p3.x + (pt.x - p3.x) * f, p3.y + (pt.y - p3.y) * f); [self curveToPoint: p3 controlPoint1: p1 controlPoint2: p2]; break; } } } - (void) appendBezierPathWithArcWithCenter: (NSPoint)center radius: (float)radius startAngle: (float)startAngle endAngle: (float)endAngle { [self appendBezierPathWithArcWithCenter: center radius: radius startAngle: startAngle endAngle: endAngle clockwise: NO]; } - (void) appendBezierPathWithArcFromPoint: (NSPoint)point1 toPoint: (NSPoint)point2 radius: (float)radius { float x1, y1; float dx1, dy1, dx2, dy2; float l, a1, a2; NSPoint p; p = [self currentPoint]; x1 = point1.x; y1 = point1.y; dx1 = p.x - x1; dy1 = p.y - y1; l= dx1*dx1 + dy1*dy1; if (l <= 0) { [self lineToPoint: point1]; return; } l = 1/sqrt(l); dx1 *= l; dy1 *= l; dx2 = point2.x - x1; dy2 = point2.y - y1; l = dx2*dx2 + dy2*dy2; if (l <= 0) { [self lineToPoint: point1]; return; } l = 1/sqrt(l); dx2 *= l; dy2 *= l; l = dx1*dx2 + dy1*dy2; if (l < -0.999) { [self lineToPoint: point1]; return; } l = radius/sin(acos(l)); p.x = x1 + (dx1 + dx2)*l; p.y = y1 + (dy1 + dy2)*l; if (dx1 < -1) a1 = 180; else if (dx1 > 1) a1 = 0; else a1 = acos(dx1)/PI*180; if (dy1 < 0) { a1 = -a1; } if (dx2 < -1) a2 = 180; else if (dx2 > 1) a2 = 0; else a2 = acos(dx2)/PI*180; if (dy2 < 0) { a2 = -a2; } l = dx1*dy2 - dx2*dy1; if (l < 0) { a2 = a2 - 90; a1 = a1 + 90; [self appendBezierPathWithArcWithCenter: p radius: radius startAngle: a1 endAngle: a2 clockwise: NO]; } else { a2 = a2 + 90; a1 = a1 - 90; [self appendBezierPathWithArcWithCenter: p radius: radius startAngle: a1 endAngle: a2 clockwise: YES]; } } - (void)appendBezierPathWithGlyph:(NSGlyph)glyph inFont:(NSFont *)font { [[font fontInfo] appendBezierPathWithGlyphs: &glyph count: 1 toBezierPath: self]; } - (void)appendBezierPathWithGlyphs:(NSGlyph *)glyphs count:(int)count inFont:(NSFont *)font { [[font fontInfo] appendBezierPathWithGlyphs: glyphs count: count toBezierPath: self]; } - (void)appendBezierPathWithPackedGlyphs:(const char *)packedGlyphs { [GSCurrentContext() appendBezierPathWithPackedGlyphs: packedGlyphs path: self]; } - (void) appendBezierPathWithRoundedRect: (NSRect)aRect xRadius: (CGFloat)xRadius yRadius: (CGFloat)yRadius { NSPoint startp, endp, controlp1, controlp2, topLeft, topRight, bottomRight; xRadius = MIN(xRadius, aRect.size.width / 2.0); yRadius = MIN(yRadius, aRect.size.height / 2.0); if (xRadius == 0.0 || yRadius == 0.0) { [self appendBezierPathWithRect: aRect]; return; } topLeft = NSMakePoint(NSMinX(aRect), NSMaxY(aRect)); topRight = NSMakePoint(NSMaxX(aRect), NSMaxY(aRect)); bottomRight = NSMakePoint(NSMaxX(aRect), NSMinY(aRect)); startp = NSMakePoint(topLeft.x + xRadius, topLeft.y); endp = NSMakePoint(topLeft.x, topLeft.y - yRadius); controlp1 = NSMakePoint(startp.x - (KAPPA * xRadius), startp.y); controlp2 = NSMakePoint(endp.x, endp.y + (KAPPA * yRadius)); [self moveToPoint: startp]; [self curveToPoint: endp controlPoint1: controlp1 controlPoint2: controlp2]; startp = NSMakePoint(aRect.origin.x, aRect.origin.y + yRadius); endp = NSMakePoint(aRect.origin.x + xRadius, aRect.origin.y); controlp1 = NSMakePoint(startp.x, startp.y - (KAPPA * yRadius)); controlp2 = NSMakePoint(endp.x - (KAPPA * xRadius), endp.y); [self lineToPoint: startp]; [self curveToPoint: endp controlPoint1: controlp1 controlPoint2: controlp2]; startp = NSMakePoint(bottomRight.x - xRadius, bottomRight.y); endp = NSMakePoint(bottomRight.x, bottomRight.y + yRadius); controlp1 = NSMakePoint(startp.x + (KAPPA * xRadius), startp.y); controlp2 = NSMakePoint(endp.x, endp.y - (KAPPA * yRadius)); [self lineToPoint: startp]; [self curveToPoint: endp controlPoint1: controlp1 controlPoint2: controlp2]; startp = NSMakePoint(topRight.x, topRight.y - yRadius); endp = NSMakePoint(topRight.x - xRadius, topRight.y); controlp1 = NSMakePoint(startp.x, startp.y + (KAPPA * yRadius)); controlp2 = NSMakePoint(endp.x + (KAPPA * xRadius), endp.y); [self lineToPoint: startp]; [self curveToPoint: endp controlPoint1: controlp1 controlPoint2: controlp2]; [self closePath]; } /* We use our own point structure with double elements while recursing to avoid losing accuracy at really fine subdivisions of curves. */ typedef struct { double x, y; } double_point; static int winding_line(double_point from, double_point to, double_point p) { int y_dir; double k, x; if (from.y == to.y) return 0; if (to.y < from.y) { y_dir = -2; if (p.y < to.y) return 0; if (p.y > from.y) return 0; } else { y_dir = 2; if (p.y < from.y) return 0; if (p.y > to.y) return 0; } if (p.y == from.y || p.y == to.y) y_dir /= 2; /* The line is intersected. Check if the intersection is outside the line's bounding box. */ if (to.x < from.x) { if (p.x < to.x) return 0; if (p.x > from.x) return y_dir; } else { if (p.x < from.x) return 0; if (p.x > to.x) return y_dir; } /* Determine the exact x coordinate of the intersection. */ k = (double)(from.x - to.x) / (double)(from.y - to.y); x = to.x + k * (double)(p.y - to.y); if (x < p.x) return y_dir; return 0; } static int winding_curve(double_point from, double_point to, double_point c1, double_point c2, double_point p, int depth) { double x0, x1; double y0, y1; double scale; /* Get the vertical extents of the convex hull. */ y0 = y1 = from.y; if (to.y < y0) y0 = to.y; else if (to.y > y1) y1 = to.y; if (c1.y < y0) y0 = c1.y; else if (c1.y > y1) y1 = c1.y; if (c2.y < y0) y0 = c2.y; else if (c2.y > y1) y1 = c2.y; /* If the point is outside the convex hull, the line can't intersect the curve. */ if (p.y < y0 || p.y > y1) return 0; /* Get the horizontal convex hull. */ x0 = x1 = from.x; if (to.x < x0) x0 = to.x; else if (to.x > x1) x1 = to.x; if (c1.x < x0) x0 = c1.x; else if (c1.x > x1) x1 = c1.x; if (c2.x < x0) x0 = c2.x; else if (c2.x > x1) x1 = c2.x; /* If the point is left of the convex hull, the line doesn't intersect the curve. */ if (p.x < x0) return 0; /* If the point is right of the convex hull, the net winding count is 0, 1, or -1, and it depends only on how the end-points are placed in relation to the point. Essentially, it's equivalent to a line. */ if (p.x > x1) return winding_line(from, to, p); /* Limit the recursion, just to be safe. */ if (depth >= 40) return winding_line(from, to, p); /* The line possibly intersects the curve in some interesting way. If the curve is flat enough, we can pretend it's a line. Otherwise, we subdivide and recurse. First, calculate a suitable scale based on the coordinates of the convex hull. This is used to get a good cutoff for the subdivision. Since it's based on the coordinates in the curve, scaling the curve up or down won't affect relative accuracy. Note that if the scale is zero, the convex hull, and thus the curve, has no extent. */ scale = fabs(x0) + fabs(x1) + fabs(y0) + fabs(y1); if (!scale) return 0; scale /= 40000000.0; /* Deal with the degenerate case to == from. */ if (to.x == from.x && to.y == from.y) { if (x1 - x0 < scale && y1 - y0 < scale) return winding_line(from, to, p); } else { double dx, dy; double nx, ny; double d0, d1, d2, d3; /* Get the direction vector and the normal vector. */ dx = to.x - from.x; dy = to.y - from.y; d0 = sqrt(dx * dx + dy * dy); dx /= d0; dy /= d0; nx = dy; ny = -dx; /* Check that the distances along the direction vector are monotone. */ d0 = from.x * dx + from.y * dy; d1 = c1.x * dx + c1.y * dy; d2 = c2.x * dx + c2.y * dy; d3 = to.x * dx + to.y * dy; if ((d3 > d2 && d2 > d1 && d1 > d0) || (d3 < d2 && d2 < d1 && d1 < d0)) { /* Check that the control points are close to the straigt line between from and to. */ d0 = to.x * nx + to.y * ny; d1 = c1.x * nx + c1.y * ny; d2 = c2.x * nx + c2.y * ny; if (fabs(d0 - d1) < scale && fabs(d0 - d2) < scale) { /* It's flat enough. */ return winding_line(from, to, p); } } } { /* Subdivide. */ double_point m, l1, l2, r1, r2; m.x = (from.x + to.x + 3 * (c1.x + c2.x)) / 8; m.y = (from.y + to.y + 3 * (c1.y + c2.y)) / 8; l1.x = (from.x + c1.x) / 2; l1.y = (from.y + c1.y) / 2; l2.x = (from.x + 2 * c1.x + c2.x) / 4; l2.y = (from.y + 2 * c1.y + c2.y) / 4; r2.x = (to.x + c2.x) / 2; r2.y = (to.y + c2.y) / 2; r1.x = (to.x + 2 * c2.x + c1.x) / 4; r1.y = (to.y + 2 * c2.y + c1.y) / 4; return winding_curve(from, m, l1, l2, p, depth + 1) + winding_curve(m, to, r1, r2, p, depth + 1); } } - (int) windingCountAtPoint: (NSPoint)point { int total; NSBezierPathElement type; int count; BOOL first; NSPoint pts[3]; NSPoint first_p, last_p; int i; /* We trace a line from (-INF, point.y) to (point) and count the intersections. Simple, really. ;) Lines are trivially checked with a few complications: a. Tangent lines, i.e. horizontal lines. These can be ignored since the winding count is undefined on edges. b. Lines whose endpoints are touched by our infinite line. To get these right, we return half a winding for such intersections. Except for intermediate horizontal lines, which are ignored, the next line will also be intersected in one endpoint. If it's going in the same direction as the first line, the two half intersections will add up to one real intersection. If it isn't, the two half intersections with opposite signs will cancel out. Either way, we get the right results. (If this happens for the first element, s/next/previous/ in the explanation. In practice, we double the winding counts while working and divide by 2 just before returning.) Curves are checked first using the convex hull, and if necessary, by subdividing until they are flat enough to check as lines. We use a very fine subdivision, and thus get good accuracy. This is possible because only the parts of the curve that might intersect the line are subdivided (due to the convex hull checks). */ total = 0; count = [self elementCount]; if (count == 0) return 0; /* 'Unroll' the first element to avoid compiler warnings. It has to be a MoveTo, anyway. */ type = [self elementAtIndex: 0 associatedPoints: pts]; if (type != NSMoveToBezierPathElement) { NSWarnLog(@"Invalid path, first element isn't MoveTo."); return 0; } last_p = first_p = pts[0]; first = NO; #define D(a) (double_point){a.x,a.y} for (i = 1; i < count; i++) { type = [self elementAtIndex: i associatedPoints: pts]; switch(type) { case NSMoveToBezierPathElement: if (!first) { total += winding_line(D(last_p), D(first_p), D(point)); } last_p = first_p = pts[0]; first = NO; break; case NSLineToBezierPathElement: if (first) { NSWarnLog(@"Invalid path, LineTo without MoveTo."); return 0; } total += winding_line(D(last_p), D(pts[0]), D(point)); last_p = pts[0]; break; case NSCurveToBezierPathElement: if (first) { NSWarnLog(@"Invalid path, CurveTo without MoveTo."); return 0; } total += winding_curve(D(last_p), D(pts[2]), D(pts[0]), D(pts[1]), D(point), 0); last_p = pts[2]; break; case NSClosePathBezierPathElement: if (first) { NSWarnLog(@"Invalid path, ClosePath with no open subpath."); return 0; } first = YES; total += winding_line(D(last_p), D(first_p), D(point)); break; default: NSWarnLog(@"Invalid element in path."); return 0; } } if (!first) total += winding_line(D(last_p), D(first_p), D(point)); #undef D if (total & 1) { /* This should only happen for points on edges, and the winding count is undefined at such points. */ NSDebugLLog(@"NSBezierPath", @"winding count total is odd"); } return total / 2; } - (BOOL) containsPoint: (NSPoint)point { int sum; if (![self elementCount]) return NO; if (!NSPointInRect(point, [self bounds])) return NO; sum = [self windingCountAtPoint: point]; if ([self windingRule] == NSNonZeroWindingRule) { if (sum == 0) return NO; else return YES; } else { if ((sum % 2) == 0) return NO; else return YES; } } // // Caching paths // // // Caching // - (BOOL)cachesBezierPath { return _cachesBezierPath; } - (void)setCachesBezierPath:(BOOL)flag { _cachesBezierPath = flag; if (!flag) INVALIDATE_CACHE(); } // // NSCoding protocol // - (void)encodeWithCoder:(NSCoder *)aCoder { NSBezierPathElement type; NSPoint pts[3]; int i, count; float f; f = [self lineWidth]; [aCoder encodeValueOfObjCType: @encode(float) at: &f]; i = [self lineCapStyle]; [aCoder encodeValueOfObjCType: @encode(int) at: &i]; i = [self lineJoinStyle]; [aCoder encodeValueOfObjCType: @encode(int) at: &i]; i = [self windingRule]; [aCoder encodeValueOfObjCType: @encode(int) at: &i]; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &_cachesBezierPath]; count = [self elementCount]; [aCoder encodeValueOfObjCType: @encode(int) at: &count]; for (i = 0; i < count; i++) { type = [self elementAtIndex: i associatedPoints: pts]; [aCoder encodeValueOfObjCType: @encode(int) at: &type]; switch(type) { case NSMoveToBezierPathElement: case NSLineToBezierPathElement: [aCoder encodePoint: pts[0]]; break; case NSCurveToBezierPathElement: [aCoder encodePoint: pts[0]]; [aCoder encodePoint: pts[1]]; [aCoder encodePoint: pts[2]]; break; case NSClosePathBezierPathElement: break; default: break; } } } - (id)initWithCoder:(NSCoder *)aCoder { NSBezierPathElement type; NSPoint pts[3]; int i, count; float f; // We have to init the place to store the elements [self init]; [aCoder decodeValueOfObjCType: @encode(float) at: &f]; [self setLineWidth: f]; [aCoder decodeValueOfObjCType: @encode(int) at: &i]; [self setLineCapStyle: i]; [aCoder decodeValueOfObjCType: @encode(int) at: &i]; [self setLineJoinStyle: i]; [aCoder decodeValueOfObjCType: @encode(int) at: &i]; [self setWindingRule: i]; [aCoder decodeValueOfObjCType: @encode(BOOL) at: &_cachesBezierPath]; _cacheImage = nil; _shouldRecalculateBounds = YES; [aCoder decodeValueOfObjCType: @encode(int) at: &count]; for (i = 0; i < count; i++) { [aCoder decodeValueOfObjCType: @encode(int) at: &type]; switch(type) { case NSMoveToBezierPathElement: pts[0] = [aCoder decodePoint]; [self moveToPoint: pts[0]]; break; case NSLineToBezierPathElement: pts[0] = [aCoder decodePoint]; [self lineToPoint: pts[0]]; break; case NSCurveToBezierPathElement: pts[0] = [aCoder decodePoint]; pts[1] = [aCoder decodePoint]; pts[2] = [aCoder decodePoint]; [self curveToPoint: pts[2] controlPoint1: pts[0] controlPoint2: pts[1]]; break; case NSClosePathBezierPathElement: [self closePath]; break; default: break; } } return self; } // // NSCopying Protocol // - (id)copyWithZone:(NSZone *)zone { NSBezierPath *path = (NSBezierPath*)NSCopyObject (self, 0, zone); if (_cachesBezierPath && _cacheImage) path->_cacheImage = [_cacheImage copy]; if (_dash_pattern != NULL) { float *pattern = NSZoneMalloc(zone, _dash_count * sizeof(float)); memcpy(pattern, _dash_pattern, _dash_count * sizeof(float)); _dash_pattern = pattern; } return path; } @end @implementation NSBezierPath (PrivateMethods) - (void) _invalidateCache { _shouldRecalculateBounds = YES; DESTROY(_cacheImage); } /* Helper for -_recalculateBounds. */ static NSPoint point_on_curve(double t, NSPoint a, NSPoint b, NSPoint c, NSPoint d) { double ti = 1.0 - t; return NSMakePoint(ti * ti * ti * a.x + 3 * ti * ti * t * b.x + 3 * ti * t * t * c.x + t * t * t * d.x, ti * ti * ti * a.y + 3 * ti * ti * t * b.y + 3 * ti * t * t * c.y + t * t * t * d.y); } - (void)_recalculateBounds { NSBezierPathElement type; NSPoint p; /* Current point. */ NSPoint last_move; NSPoint pts[3]; NSPoint min, max; /* Path bounding box. */ NSPoint cmin, cmax; /* Control-point bounding box. */ int i, count, num_curves; count = [self elementCount]; if (!count) { _bounds = NSZeroRect; _controlPointBounds = NSZeroRect; _shouldRecalculateBounds = NO; return; } last_move = p = min = max = cmin = cmax = NSMakePoint(0, 0); #define CHECK_MAX(max, p) \ if (p.x > max.x) max.x = p.x; \ if (p.y > max.y) max.y = p.y; #define CHECK_MIN(min, p) \ if (p.x < min.x) min.x = p.x; \ if (p.y < min.y) min.y = p.y; num_curves = 0; for (i = 0; i < count; i++) { type = [self elementAtIndex: i associatedPoints: pts]; if (!i) { cmin = cmax = min = max = p = last_move = pts[0]; } switch (type) { case NSClosePathBezierPathElement: p = last_move; continue; case NSMoveToBezierPathElement: last_move = pts[0]; /* Fall-through. */ case NSLineToBezierPathElement: CHECK_MAX(max, pts[0]) CHECK_MIN(min, pts[0]) p = pts[0]; break; case NSCurveToBezierPathElement: { double t0, t1, t; NSPoint q; num_curves++; CHECK_MAX(cmax, pts[0]) CHECK_MIN(cmin, pts[0]) CHECK_MAX(cmax, pts[1]) CHECK_MIN(cmin, pts[1]) CHECK_MAX(max, pts[2]) CHECK_MIN(min, pts[2]) #define CHECK_CURVE_EXTREMES(x) \ t = (p.x * (pts[2].x - pts[1].x) \ + pts[0].x * (-pts[2].x - pts[1].x) \ + pts[1].x * pts[1].x + pts[0].x * pts[0].x); \ if (t >= 0.0) \ { \ t = sqrt(t); \ t0 = (pts[1].x - 2 * pts[0].x + p.x + t) \ / (-pts[2].x + 3 * pts[1].x - 3 * pts[0].x + p.x); \ t1 = (pts[1].x - 2 * pts[0].x + p.x - t) \ / (-pts[2].x + 3 * pts[1].x - 3 * pts[0].x + p.x); \ \ if (t0 > 0.0 && t0 < 1.0) \ { \ q = point_on_curve(t0, p, pts[0], pts[1], pts[2]); \ CHECK_MAX(max, q) \ CHECK_MIN(min, q) \ } \ if (t1 > 0.0 && t1 < 1.0) \ { \ q = point_on_curve(t1, p, pts[0], pts[1], pts[2]); \ CHECK_MAX(max, q) \ CHECK_MIN(min, q) \ } \ } CHECK_CURVE_EXTREMES(x) CHECK_CURVE_EXTREMES(y) #undef CHECK_CURVE_EXTREMES p = pts[2]; break; } } } /* If there were no curve elements, the control-point bounding box is the same as the path bounding box. Otherwise, the control-point bounding box is the union of the path bounding box and the bounding box of the curve control points. */ if (num_curves) { CHECK_MAX(cmax, max) CHECK_MIN(cmin, min) } else { cmin = min; cmax = max; } _bounds = NSMakeRect(min.x, min.y, max.x - min.x, max.y - min.y); _controlPointBounds = NSMakeRect(cmin.x, cmin.y, cmax.x - cmin.x, cmax.y - cmin.y); _shouldRecalculateBounds = NO; #undef CHECK_MAX #undef CHECK_MIN } @end @implementation GSBezierPath - (id)init { NSZone *zone; self = [super init]; zone = [self zone]; pathElements = NSZoneMalloc(zone, sizeof(GSIArray_t)); GSIArrayInitWithZoneAndCapacity(pathElements, zone, 8); flat = YES; return self; } - (void)dealloc { GSIArrayEmpty(pathElements); NSZoneFree([self zone], pathElements); [super dealloc]; } // // Path construction // - (void)moveToPoint:(NSPoint)aPoint { PathElement elem; elem.type = NSMoveToBezierPathElement; elem.points[0] = aPoint; elem.points[1] = NSZeroPoint; elem.points[2] = NSZeroPoint; GSIArrayAddItem(pathElements, (GSIArrayItem)elem); INVALIDATE_CACHE(); } - (void)lineToPoint:(NSPoint)aPoint { PathElement elem; elem.type = NSLineToBezierPathElement; elem.points[0] = aPoint; elem.points[1] = NSZeroPoint; elem.points[2] = NSZeroPoint; GSIArrayAddItem(pathElements, (GSIArrayItem)elem); INVALIDATE_CACHE(); } - (void) curveToPoint: (NSPoint)aPoint controlPoint1: (NSPoint)controlPoint1 controlPoint2: (NSPoint)controlPoint2 { PathElement elem; elem.type = NSCurveToBezierPathElement; elem.points[0] = controlPoint1; elem.points[1] = controlPoint2; elem.points[2] = aPoint; GSIArrayAddItem(pathElements, (GSIArrayItem)elem); flat = NO; INVALIDATE_CACHE(); } - (void)closePath { PathElement elem; elem.type = NSClosePathBezierPathElement; elem.points[0] = NSZeroPoint; elem.points[1] = NSZeroPoint; elem.points[2] = NSZeroPoint; GSIArrayAddItem(pathElements, (GSIArrayItem)elem); INVALIDATE_CACHE(); } - (void)removeAllPoints { GSIArrayRemoveAllItems(pathElements); INVALIDATE_CACHE(); } // // Elements // - (int)elementCount { return GSIArrayCount(pathElements); } - (NSBezierPathElement)elementAtIndex:(int)index associatedPoints:(NSPoint *)points { PathElement elm = GSIArrayItemAtIndex(pathElements, index).ext; NSBezierPathElement type = elm.type; if (points != NULL) { if (type == NSMoveToBezierPathElement || type == NSLineToBezierPathElement) { points[0] = elm.points[0]; } else if (type == NSCurveToBezierPathElement) { points[0] = elm.points[0]; points[1] = elm.points[1]; points[2] = elm.points[2]; } } return type; } - (void)setAssociatedPoints:(NSPoint *)points atIndex:(int)index { PathElement elm = GSIArrayItemAtIndex(pathElements, index).ext; NSBezierPathElement type = elm.type; switch(type) { case NSMoveToBezierPathElement: case NSLineToBezierPathElement: elm.points[0] = points[0]; break; case NSCurveToBezierPathElement: elm.points[0] = points[0]; elm.points[1] = points[1]; elm.points[2] = points[2]; break; case NSClosePathBezierPathElement: break; default: break; } GSIArraySetItemAtIndex(pathElements, (GSIArrayItem)elm, index); INVALIDATE_CACHE(); } // // Path modifications. // - (NSBezierPath *)bezierPathByFlatteningPath { if (flat) return self; return [super bezierPathByFlatteningPath]; } - (void) transformUsingAffineTransform: (NSAffineTransform *)transform { NSBezierPathElement type; int i, count; GSIArrayItem *elments = GSIArrayItems(pathElements); SEL transformPointSel = @selector(transformPoint:); NSPoint (*transformPointImp)(NSAffineTransform*, SEL, NSPoint); transformPointImp = (NSPoint (*)(NSAffineTransform*, SEL, NSPoint)) [transform methodForSelector: transformPointSel]; count = GSIArrayCount(pathElements); for (i = 0; i < count; i++) { type = elments[i].ext.type; switch(type) { case NSMoveToBezierPathElement: case NSLineToBezierPathElement: elments[i].ext.points[0] = (*transformPointImp)(transform, transformPointSel, elments[i].ext.points[0]); break; case NSCurveToBezierPathElement: elments[i].ext.points[0] = (*transformPointImp)(transform, transformPointSel, elments[i].ext.points[0]); elments[i].ext.points[1] = (*transformPointImp)(transform, transformPointSel, elments[i].ext.points[1]); elments[i].ext.points[2] = (*transformPointImp)(transform, transformPointSel, elments[i].ext.points[2]); break; case NSClosePathBezierPathElement: break; default: break; } } INVALIDATE_CACHE(); } - (void) appendBezierPath: (NSBezierPath *)aPath { PathElement elem; int i, count; if (![aPath isKindOfClass: object_getClass(self)]) { [super appendBezierPath: aPath]; return; } flat = flat && ((GSBezierPath*)aPath)->flat; count = [aPath elementCount]; for (i = 0; i < count; i++) { elem = GSIArrayItemAtIndex(((GSBezierPath*)aPath)->pathElements, i).ext; GSIArrayAddItem(pathElements, (GSIArrayItem)elem); } INVALIDATE_CACHE(); } // // NSCopying Protocol // - (id)copyWithZone:(NSZone *)zone { GSBezierPath *path = [super copyWithZone: zone]; path->pathElements = GSIArrayCopyWithZone(pathElements, zone); return path; } @end // GSBezierPath static void flatten(NSPoint coeff[], float flatness, NSBezierPath *path) { // Check if the Bezier path defined by the four points has the given flatness. // If not split it up in the middle and recurse. // Otherwise add the end point to the path. BOOL flat = YES; // This criteria for flatness is based on code from Libart which has the // following copyright: /* Libart_LGPL - library of basic graphic primitives * Copyright (C) 1998 Raph Levien * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ double x1_0, y1_0; double x3_2, y3_2; double x3_0, y3_0; double z3_0_dot; double z1_dot, z2_dot; double z1_perp, z2_perp; double max_perp_sq; x3_0 = coeff[3].x - coeff[0].x; y3_0 = coeff[3].y - coeff[0].y; x3_2 = coeff[3].x - coeff[2].x; y3_2 = coeff[3].y - coeff[2].y; x1_0 = coeff[1].x - coeff[0].x; y1_0 = coeff[1].y - coeff[0].y; z3_0_dot = x3_0 * x3_0 + y3_0 * y3_0; if (z3_0_dot < 0.001) flat = YES; else { max_perp_sq = flatness * flatness * z3_0_dot; z1_perp = y1_0 * x3_0 - x1_0 * y3_0; if (z1_perp * z1_perp > max_perp_sq) flat = NO; else { z2_perp = y3_2 * x3_0 - x3_2 * y3_0; if (z2_perp * z2_perp > max_perp_sq) flat = NO; else { z1_dot = x1_0 * x3_0 + y1_0 * y3_0; if (z1_dot < 0 && z1_dot * z1_dot > max_perp_sq) flat = NO; else { z2_dot = x3_2 * x3_0 + y3_2 * y3_0; if (z2_dot < 0 && z2_dot * z2_dot > max_perp_sq) flat = NO; else { if ((z1_dot + z1_dot > z3_0_dot) || (z2_dot + z2_dot > z3_0_dot)) flat = NO; } } } } } if (!flat) { NSPoint bleft[4], bright[4]; bleft[0] = coeff[0]; bleft[1].x = (coeff[0].x + coeff[1].x) / 2; bleft[1].y = (coeff[0].y + coeff[1].y) / 2; bleft[2].x = (coeff[0].x + 2*coeff[1].x + coeff[2].x) / 4; bleft[2].y = (coeff[0].y + 2*coeff[1].y + coeff[2].y) / 4; bleft[3].x = (coeff[0].x + 3*(coeff[1].x + coeff[2].x) + coeff[3].x) / 8; bleft[3].y = (coeff[0].y + 3*(coeff[1].y + coeff[2].y) + coeff[3].y) / 8; bright[0].x = bleft[3].x; bright[0].y = bleft[3].y; bright[1].x = (coeff[3].x + 2*coeff[2].x + coeff[1].x) / 4; bright[1].y = (coeff[3].y + 2*coeff[2].y + coeff[1].y) / 4; bright[2].x = (coeff[3].x + coeff[2].x) / 2; bright[2].y = (coeff[3].y + coeff[2].y) / 2; bright[3] = coeff[3]; flatten(bleft, flatness, path); flatten(bright, flatness, path); } else { //[path lineToPoint: coeff[1]]; //[path lineToPoint: coeff[2]]; [path lineToPoint: coeff[3]]; } }