/* ImageThumbnailer.m * * Copyright (C) 2003 Free Software Foundation, Inc. * * Author: Enrico Sersale * Date: August 2001 * * This file is part of the GNUstep GWorkspace application * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111 USA. */ #include #include #include #include "ImageThumbnailer.h" #define TMBMAX (48.0) #define RESZLIM 4 #define MIX_LIM 16 @implementation ImageThumbnailer - (void)dealloc { [super dealloc]; } - (BOOL)canProvideThumbnailForPath:(NSString *)path { NSString *ext = [[path pathExtension] lowercaseString]; return (ext && [[NSImage imageFileTypes] containsObject: ext]); } - (NSData *)makeThumbnailForPath:(NSString *)path { CREATE_AUTORELEASE_POOL(arp); NSImage *image = [[NSImage alloc] initWithContentsOfFile: path]; if (image && [image isValid]) { NSData *tiffdata = [image TIFFRepresentation]; NSBitmapImageRep *rep = [NSBitmapImageRep imageRepWithData: tiffdata]; int spp = [rep samplesPerPixel]; int bitsPerPixel = [rep bitsPerPixel]; int bpp = bitsPerPixel / 8; if (((spp == 3) && (bitsPerPixel == 24)) || ((spp == 4) && (bitsPerPixel == 32)) || ((spp == 1) && (bitsPerPixel == 8)) || ((spp == 2) && (bitsPerPixel == 16))) { NSSize imsize = [image size]; if ((imsize.width <= TMBMAX) && (imsize.height <= TMBMAX) && (imsize.width >= (TMBMAX - RESZLIM)) && (imsize.height >= (TMBMAX - RESZLIM))) { RETAIN (tiffdata); RELEASE (image); RELEASE (arp); return AUTORELEASE (tiffdata); } else { float fact = (imsize.width >= imsize.height) ? (imsize.width / TMBMAX) : (imsize.height / TMBMAX); NSSize newsize = NSMakeSize(floor(imsize.width / fact + 0.5), floor(imsize.height / fact + 0.5)); float xratio = imsize.width / newsize.width; float yratio = imsize.height / newsize.height; BOOL hasAlpha = [rep hasAlpha]; BOOL isColor = hasAlpha ? (spp > 2) : (spp > 1); NSString *colorSpaceName = isColor ? NSCalibratedRGBColorSpace : NSCalibratedWhiteColorSpace; NSBitmapImageRep *newrep; unsigned char *srcData; unsigned char *dstData; unsigned x, y; NSData *data = nil; newrep = [[NSBitmapImageRep alloc] initWithBitmapDataPlanes: NULL pixelsWide: (int)newsize.width pixelsHigh: (int)newsize.height bitsPerSample: 8 samplesPerPixel: (isColor ? 4 : 2) hasAlpha: YES isPlanar: NO colorSpaceName: colorSpaceName bytesPerRow: 0 bitsPerPixel: 0]; srcData = [rep bitmapData]; dstData = [newrep bitmapData]; for (y = 0; y < (int)(newsize.height); y++) { int px[2], py[2]; py[0] = floor(y * yratio); py[1] = ceil((y + 1) * yratio); py[1] = ((py[1] > imsize.height) ? (int)(imsize.height) : py[1]); for (x = 0; x < (int)(newsize.width); x++) { int expos = (int)(bpp * (floor(y * yratio) * imsize.width + floor(x * xratio))); unsigned expix[4] = { 0, 0, 0, 0 }; unsigned pix[4] = { 0, 0, 0, 0 }; int count = 0; unsigned char c; int i, j; expix[0] = srcData[expos]; if (isColor) { expix[1] = srcData[expos + 1]; expix[2] = srcData[expos + 2]; expix[3] = (hasAlpha ? srcData[expos + 3] : 255); } else { expix[1] = (hasAlpha ? srcData[expos + 1] : 255); } px[0] = floor(x * xratio); px[1] = ceil((x + 1) * xratio); px[1] = ((px[1] > imsize.width) ? (int)(imsize.width) : px[1]); for (i = px[0]; i < px[1]; i++) { for (j = py[0]; j < py[1]; j++) { int pos = (int)(bpp * (j * imsize.width + i)); pix[0] += srcData[pos]; if (isColor) { pix[1] += srcData[pos + 1]; pix[2] += srcData[pos + 2]; pix[3] += (hasAlpha ? srcData[pos + 3] : 255); } else { pix[1] += (hasAlpha ? srcData[pos + 1] : 255); } count++; } } c = (unsigned char)(pix[0] / count); *dstData++ = ((abs(c - expix[0]) < MIX_LIM) ? (unsigned char)expix[0] : c); if (isColor) { c = (unsigned char)(pix[1] / count); *dstData++ = ((abs(c - expix[1]) < MIX_LIM) ? (unsigned char)expix[1] : c); c = (unsigned char)(pix[2] / count); *dstData++ = ((abs(c - expix[2]) < MIX_LIM) ? (unsigned char)expix[2] : c); c = (unsigned char)(pix[3] / count); *dstData++ = ((abs(c - expix[3]) < MIX_LIM) ? (unsigned char)expix[3] : c); } else { c = (unsigned char)(pix[1] / count); *dstData++ = ((abs(c - expix[1]) < MIX_LIM) ? (unsigned char)expix[1] : c); } } } data = [newrep TIFFRepresentation]; RETAIN (data); RELEASE (image); RELEASE (newrep); RELEASE (arp); return AUTORELEASE (data); } } } RELEASE (image); RELEASE (arp); return nil; } /* - (NSData *)makeThumbnailForPath:(NSString *)path { CREATE_AUTORELEASE_POOL(arp); NSImage *image = [[NSImage alloc] initWithContentsOfFile: path]; if (image && [image isValid]) { NSSize size = [image size]; NSRect srcr = NSMakeRect(0, 0, size.width, size.height); NSRect dstr = NSZeroRect; NSImageRep *rep = [image bestRepresentationForDevice: nil]; NSImage *newimage = nil; NSBitmapImageRep *newBitmapImageRep = nil; NSData *data = nil; if ((size.width <= TMBMAX) && (size.height <= TMBMAX) && (size.width >= (TMBMAX - RESZLIM)) && (size.height >= (TMBMAX - RESZLIM))) { if ([rep isKindOfClass: [NSBitmapImageRep class]]) { data = [(NSBitmapImageRep *)rep TIFFRepresentation]; if (data) { RELEASE (image); RETAIN (data); RELEASE (arp); return [data autorelease]; } } } if (size.width >= size.height) { dstr.size.width = TMBMAX; dstr.size.height = TMBMAX * size.height / size.width; } else { dstr.size.height = TMBMAX; dstr.size.width = TMBMAX * size.width / size.height; } newimage = [[NSImage alloc] initWithSize: dstr.size]; [newimage lockFocus]; [image drawInRect: dstr fromRect: srcr operation: NSCompositeSourceOver fraction: 1.0]; newBitmapImageRep = [[NSBitmapImageRep alloc] initWithFocusedViewRect: dstr]; [newimage unlockFocus]; data = [newBitmapImageRep TIFFRepresentation]; RETAIN (data); RELEASE (image); RELEASE (newimage); RELEASE (newBitmapImageRep); RELEASE (arp); return [data autorelease]; } RELEASE (arp); return nil; } */ - (NSString *)fileNameExtension { return @"tiff"; } - (NSString *)description { return @"Images Thumbnailer"; } @end