--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/WebKit/mac/Misc/WebKitNSStringExtras.mm Fri Sep 17 09:02:29 2010 +0300
@@ -0,0 +1,344 @@
+/*
+ * Copyright (C) 2005, 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#import "WebKitNSStringExtras.h"
+
+#import <WebCore/Font.h>
+#import <WebCore/GraphicsContext.h>
+#import <WebCore/WebCoreNSStringExtras.h>
+#import <WebKit/WebNSFileManagerExtras.h>
+#import <WebKit/WebNSObjectExtras.h>
+#import <unicode/uchar.h>
+#import <sys/param.h>
+
+NSString *WebKitLocalCacheDefaultsKey = @"WebKitLocalCache";
+
+static inline CGFloat webkit_CGCeiling(CGFloat value)
+{
+ if (sizeof(value) == sizeof(float))
+ return ceilf(value);
+ return ceil(value);
+}
+
+using namespace WebCore;
+
+@implementation NSString (WebKitExtras)
+
+static BOOL canUseFastRenderer(const UniChar *buffer, unsigned length)
+{
+ unsigned i;
+ for (i = 0; i < length; i++) {
+ UCharDirection direction = u_charDirection(buffer[i]);
+ if (direction == U_RIGHT_TO_LEFT || direction > U_OTHER_NEUTRAL)
+ return NO;
+ }
+ return YES;
+}
+
+- (void)_web_drawAtPoint:(NSPoint)point font:(NSFont *)font textColor:(NSColor *)textColor
+{
+ [self _web_drawAtPoint:point font:font textColor:textColor allowingFontSmoothing:YES];
+}
+
+- (void)_web_drawAtPoint:(NSPoint)point font:(NSFont *)font textColor:(NSColor *)textColor allowingFontSmoothing:(BOOL)fontSmoothingIsAllowed
+{
+ unsigned length = [self length];
+ Vector<UniChar, 2048> buffer(length);
+
+ [self getCharacters:buffer.data()];
+
+ if (canUseFastRenderer(buffer.data(), length)) {
+ // The following is a half-assed attempt to match AppKit's rounding rules for drawAtPoint.
+ // It's probably incorrect for high DPI.
+ // If you change this, be sure to test all the text drawn this way in Safari, including
+ // the status bar, bookmarks bar, tab bar, and activity window.
+ point.y = webkit_CGCeiling(point.y);
+
+ NSGraphicsContext *nsContext = [NSGraphicsContext currentContext];
+ CGContextRef cgContext = static_cast<CGContextRef>([nsContext graphicsPort]);
+ GraphicsContext graphicsContext(cgContext);
+
+ // Safari doesn't flip the NSGraphicsContext before calling WebKit, yet WebCore requires a flipped graphics context.
+ BOOL flipped = [nsContext isFlipped];
+ if (!flipped)
+ CGContextScaleCTM(cgContext, 1, -1);
+
+ Font webCoreFont(FontPlatformData(font), ![nsContext isDrawingToScreen], fontSmoothingIsAllowed ? AutoSmoothing : Antialiased);
+ TextRun run(buffer.data(), length);
+ run.disableRoundingHacks();
+
+ CGFloat red;
+ CGFloat green;
+ CGFloat blue;
+ CGFloat alpha;
+ [[textColor colorUsingColorSpaceName:NSDeviceRGBColorSpace] getRed:&red green:&green blue:&blue alpha:&alpha];
+ graphicsContext.setFillColor(makeRGBA(red * 255, green * 255, blue * 255, alpha * 255), DeviceColorSpace);
+
+ webCoreFont.drawText(&graphicsContext, run, FloatPoint(point.x, (flipped ? point.y : (-1 * point.y))));
+
+ if (!flipped)
+ CGContextScaleCTM(cgContext, 1, -1);
+ } else {
+ // The given point is on the baseline.
+ if ([[NSView focusView] isFlipped])
+ point.y -= [font ascender];
+ else
+ point.y += [font descender];
+
+ [self drawAtPoint:point withAttributes:[NSDictionary dictionaryWithObjectsAndKeys:font, NSFontAttributeName, textColor, NSForegroundColorAttributeName, nil]];
+ }
+}
+
+- (void)_web_drawDoubledAtPoint:(NSPoint)textPoint
+ withTopColor:(NSColor *)topColor
+ bottomColor:(NSColor *)bottomColor
+ font:(NSFont *)font
+{
+ // turn off font smoothing so translucent text draws correctly (Radar 3118455)
+ [self _web_drawAtPoint:textPoint font:font textColor:bottomColor allowingFontSmoothing:NO];
+
+ textPoint.y += 1;
+ [self _web_drawAtPoint:textPoint font:font textColor:topColor allowingFontSmoothing:NO];
+}
+
+- (float)_web_widthWithFont:(NSFont *)font
+{
+ unsigned length = [self length];
+ Vector<UniChar, 2048> buffer(length);
+
+ [self getCharacters:buffer.data()];
+
+ if (canUseFastRenderer(buffer.data(), length)) {
+ Font webCoreFont(FontPlatformData(font), ![[NSGraphicsContext currentContext] isDrawingToScreen]);
+ TextRun run(buffer.data(), length);
+ run.disableRoundingHacks();
+ return webCoreFont.floatWidth(run);
+ }
+
+ return [self sizeWithAttributes:[NSDictionary dictionaryWithObjectsAndKeys:font, NSFontAttributeName, nil]].width;
+}
+
+- (NSString *)_web_stringByAbbreviatingWithTildeInPath
+{
+ NSString *resolvedHomeDirectory = [NSHomeDirectory() stringByResolvingSymlinksInPath];
+ NSString *path;
+
+ if ([self hasPrefix:resolvedHomeDirectory]) {
+ NSString *relativePath = [self substringFromIndex:[resolvedHomeDirectory length]];
+ path = [NSHomeDirectory() stringByAppendingPathComponent:relativePath];
+ } else {
+ path = self;
+ }
+
+ return [path stringByAbbreviatingWithTildeInPath];
+}
+
+- (NSString *)_web_stringByStrippingReturnCharacters
+{
+ NSMutableString *newString = [[self mutableCopy] autorelease];
+ [newString replaceOccurrencesOfString:@"\r" withString:@"" options:NSLiteralSearch range:NSMakeRange(0, [newString length])];
+ [newString replaceOccurrencesOfString:@"\n" withString:@"" options:NSLiteralSearch range:NSMakeRange(0, [newString length])];
+ return newString;
+}
+
++ (NSStringEncoding)_web_encodingForResource:(Handle)resource
+{
+ return CFStringConvertEncodingToNSStringEncoding(stringEncodingForResource(resource));
+}
+
+- (BOOL)_webkit_isCaseInsensitiveEqualToString:(NSString *)string
+{
+ return stringIsCaseInsensitiveEqualToString(self, string);
+}
+
+-(BOOL)_webkit_hasCaseInsensitivePrefix:(NSString *)prefix
+{
+ return [self rangeOfString:prefix options:(NSCaseInsensitiveSearch | NSAnchoredSearch)].location != NSNotFound;
+}
+
+-(BOOL)_webkit_hasCaseInsensitiveSuffix:(NSString *)suffix
+{
+ return hasCaseInsensitiveSuffix(self, suffix);
+}
+
+-(BOOL)_webkit_hasCaseInsensitiveSubstring:(NSString *)substring
+{
+ return hasCaseInsensitiveSubstring(self, substring);
+}
+
+-(NSString *)_webkit_filenameByFixingIllegalCharacters
+{
+ return filenameByFixingIllegalCharacters(self);
+}
+
+-(NSString *)_webkit_stringByTrimmingWhitespace
+{
+ NSMutableString *trimmed = [[self mutableCopy] autorelease];
+ CFStringTrimWhitespace((CFMutableStringRef)trimmed);
+ return trimmed;
+}
+
+- (NSString *)_webkit_stringByCollapsingNonPrintingCharacters
+{
+ NSMutableString *result = [NSMutableString string];
+ static NSCharacterSet *charactersToTurnIntoSpaces = nil;
+ static NSCharacterSet *charactersToNotTurnIntoSpaces = nil;
+
+ if (charactersToTurnIntoSpaces == nil) {
+ NSMutableCharacterSet *set = [[NSMutableCharacterSet alloc] init];
+ [set addCharactersInRange:NSMakeRange(0x00, 0x21)];
+ [set addCharactersInRange:NSMakeRange(0x7F, 0x01)];
+ charactersToTurnIntoSpaces = [set copy];
+ [set release];
+ charactersToNotTurnIntoSpaces = [[charactersToTurnIntoSpaces invertedSet] retain];
+ }
+
+ unsigned length = [self length];
+ unsigned position = 0;
+ while (position != length) {
+ NSRange nonSpace = [self rangeOfCharacterFromSet:charactersToNotTurnIntoSpaces
+ options:0 range:NSMakeRange(position, length - position)];
+ if (nonSpace.location == NSNotFound) {
+ break;
+ }
+
+ NSRange space = [self rangeOfCharacterFromSet:charactersToTurnIntoSpaces
+ options:0 range:NSMakeRange(nonSpace.location, length - nonSpace.location)];
+ if (space.location == NSNotFound) {
+ space.location = length;
+ }
+
+ if (space.location > nonSpace.location) {
+ if (position != 0) {
+ [result appendString:@" "];
+ }
+ [result appendString:[self substringWithRange:
+ NSMakeRange(nonSpace.location, space.location - nonSpace.location)]];
+ }
+
+ position = space.location;
+ }
+
+ return result;
+}
+
+- (NSString *)_webkit_stringByCollapsingWhitespaceCharacters
+{
+ NSMutableString *result = [[NSMutableString alloc] initWithCapacity:[self length]];
+ NSCharacterSet *spaces = [NSCharacterSet whitespaceAndNewlineCharacterSet];
+ static NSCharacterSet *notSpaces = nil;
+
+ if (notSpaces == nil)
+ notSpaces = [[spaces invertedSet] retain];
+
+ unsigned length = [self length];
+ unsigned position = 0;
+ while (position != length) {
+ NSRange nonSpace = [self rangeOfCharacterFromSet:notSpaces options:0 range:NSMakeRange(position, length - position)];
+ if (nonSpace.location == NSNotFound)
+ break;
+
+ NSRange space = [self rangeOfCharacterFromSet:spaces options:0 range:NSMakeRange(nonSpace.location, length - nonSpace.location)];
+ if (space.location == NSNotFound)
+ space.location = length;
+
+ if (space.location > nonSpace.location) {
+ if (position != 0)
+ [result appendString:@" "];
+ [result appendString:[self substringWithRange:NSMakeRange(nonSpace.location, space.location - nonSpace.location)]];
+ }
+
+ position = space.location;
+ }
+
+ return [result autorelease];
+}
+
+-(NSString *)_webkit_fixedCarbonPOSIXPath
+{
+ NSFileManager *fileManager = [NSFileManager defaultManager];
+ if ([fileManager fileExistsAtPath:self]) {
+ // Files exists, no need to fix.
+ return self;
+ }
+
+ NSMutableArray *pathComponents = [[[self pathComponents] mutableCopy] autorelease];
+ NSString *volumeName = [pathComponents objectAtIndex:1];
+ if ([volumeName isEqualToString:@"Volumes"]) {
+ // Path starts with "/Volumes", so the volume name is the next path component.
+ volumeName = [pathComponents objectAtIndex:2];
+ // Remove "Volumes" from the path because it may incorrectly be part of the path (3163647).
+ // We'll add it back if we have to.
+ [pathComponents removeObjectAtIndex:1];
+ }
+
+ if (!volumeName) {
+ // Should only happen if self == "/", so this shouldn't happen because that always exists.
+ return self;
+ }
+
+ if ([[fileManager _webkit_startupVolumeName] isEqualToString:volumeName]) {
+ // Startup volume name is included in path, remove it.
+ [pathComponents removeObjectAtIndex:1];
+ } else if ([[fileManager contentsOfDirectoryAtPath:@"/Volumes" error:NULL] containsObject:volumeName]) {
+ // Path starts with other volume name, prepend "/Volumes".
+ [pathComponents insertObject:@"Volumes" atIndex:1];
+ } else
+ // It's valid.
+ return self;
+
+ NSString *path = [NSString pathWithComponents:pathComponents];
+
+ if (![fileManager fileExistsAtPath:path])
+ // File at canonicalized path doesn't exist, return original.
+ return self;
+
+ return path;
+}
+
++ (NSString *)_webkit_localCacheDirectoryWithBundleIdentifier:(NSString*)bundleIdentifier
+{
+ NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
+ NSString *cacheDir = [defaults objectForKey:WebKitLocalCacheDefaultsKey];
+
+ if (!cacheDir || ![cacheDir isKindOfClass:[NSString class]]) {
+#ifdef BUILDING_ON_TIGER
+ cacheDir = [NSHomeDirectory() stringByAppendingPathComponent:@"Library/Caches"];
+#else
+ char cacheDirectory[MAXPATHLEN];
+ size_t cacheDirectoryLen = confstr(_CS_DARWIN_USER_CACHE_DIR, cacheDirectory, MAXPATHLEN);
+
+ if (cacheDirectoryLen)
+ cacheDir = [[NSFileManager defaultManager] stringWithFileSystemRepresentation:cacheDirectory length:cacheDirectoryLen - 1];
+#endif
+ }
+
+ return [cacheDir stringByAppendingPathComponent:bundleIdentifier];
+}
+
+@end