libs-gui/Source/NSBitmapImageRep+PNM.m

139 lines
3.8 KiB
Mathematica
Raw Normal View History

/* NSBitmapImageRep+PNM
Methods for reading PNM images
Copyright (C) 2003 Free Software Foundation, Inc.
Written by: Adam Fedor <fedor@gnu.org>
Date: Oct 2003
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 3 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 <http://www.gnu.org/licenses/> or write to the
Free Software Foundation, 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "NSBitmapImageRep+PNM.h"
#include <Foundation/NSData.h>
#include "AppKit/NSGraphics.h"
@implementation NSBitmapImageRep (PNM)
/* Return YES if this is a pgm or ppm file (raw only for now) */
+ (BOOL) _bitmapIsPNM: (NSData *)imageData
{
const char *bytes = [imageData bytes];
/*
3/4 bytes header line, 4/5 bytes size, 2/3 bytes depth. Thus, the image
must have a least 9 bytes.
*/
if ([imageData length]<9)
return NO;
if (bytes[0] == 'P' && (bytes[2]=='\n' || bytes[2]=='\r'))
{
if (bytes[1] == '5' || bytes[1] == '6')
return YES;
}
return NO;
}
#define GET_LINE() \
do \
{ \
char *p = buffer; \
while (*ptr != '\n' && *ptr != '\r' && (ptr-bytes) < length) \
{ \
*p++ = *ptr++; \
if (p == &buffer[sizeof(buffer)]) \
{ \
ERRMSG(@"PNM header line too long"); \
} \
} \
ptr++; \
*p = '\0'; \
} while (0)
#define ERRMSG(msg) \
do { NSLog(@"Error loading PNM: %@", msg); if (error) *error = msg; RELEASE(self); return nil; } while (0)
/* Read the ppm image. Assume it is a ppm file and imageData is not nil */
-(id) _initBitmapFromPNM: (NSData *)imageData errorMessage: (NSString **)error
{
int num, xsize, ysize, levels;
char ptype;
2005-01-21 21:43 Alexander Malmberg <alexander@malmberg.org> Various whitespace cleanups, comment type fixes, and changes to avoid warnings from recent versions of gcc. * Headers/Additions/GNUstepGUI/GSToolbar.h (-_toolbars): Declare. * Source/NSWindow+Toolbar.m: Remove conflicting declaration of [NSToolbar -_toolbars]. * Headers/Additions/GNUstepGUI/GSServicesManager.h, Source/GSServicesMananger.m (-item2title:, -validateMenuItem:): Adjust argument types. * Headers/AppKit/NSMenu.h (-validateMenuItem:): Adjust argument type. * Source/NSTextView.m (-sizeToFit): Don't use size uninitialized if neither resizable flags is set. (-insertText:): Adjust argument type. * Headers/AppKit/NSResponder.h, Source/NSResponder.m (-insertText:): Adjust argument type. Document. * Headers/AppKit/NSView.h: Change type of ivar _window to NSWindow *. * Source/GSTitleView.m (-mouseDown:): Always initialize startWindowOrigin. * Source/NSApplication.m (-setApplicationIconImage:): Add casts to avoid warnings. * Source/NSCell.m (-cellSize): Add default: case. * Source/NSPasteboard.m ([GSFiltered -pasteboard:provideDataForType:]): Detect and warn if we can't find a filter that will get us the desired type. * Source/NSProgressIndicator.m: Comment out unused variable 'images'. * Source/NSBezierPath.m: Declare GSBezierPath fully before using it. (-bezierPathByFlatteningPath, -bezierPathByReversingPath): Make sure variables are always initialized. * Source/NSMenuView.m, * Source/NSPrintOperation.m, * Source/NSSplitView.m, * Source/NSTableHeaderView.m: Make sure variables are always initialized. * Source/NSBox.m, * Source/NSImageview.m, * Source/NSText.m, * Source/NSTextStorage.m: Add missing includes. * Source/GSKeyBindingTable.m, * Source/GSLayoutManager.m, * Source/NSBitmapImageRep+PNM.m, * Source/NSBundleAdditions.m, * Source/NSLayoutManager.m, * Source/nsimage-tiff.h, * Source/tiff.m, * Headers/Additions/GNUstepGUI/GSDisplayServer.h, * Source/GSDisplayServer.m: Change signedness of various variables. * Source/NSPanel.m (-sendEvent:): Remove. * Source/NSWindow.m (-becomesKeyOnlyIfNeeded): New method. (-_sendEvent:becomesKeyOnlyIfNeeded:): Remove. Move code ... (-sendEvent:): ... here. Use -becomesKeyOnlyIfNeeded instead of the argument. git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@20590 72102866-910b-0410-8b05-ffd578937521
2005-01-21 20:39:18 +00:00
char buffer[256];
unsigned char *pchar;
unsigned length;
id colorspace;
const char *ptr;
const char *bytes = [imageData bytes];
/* magic number */
ptr = bytes;
length = [imageData length];
GET_LINE();
if (bytes[0] != 'P')
ERRMSG(@"Invalid PNM header (magic number)");
ptype = bytes[1];
if (ptype != '5' && ptype != '6')
ERRMSG(@"Unsupported PNM type");
do
{
GET_LINE();
} while (buffer[0] == '#');
num = sscanf(buffer, "%d %d", &xsize, &ysize);
if (num != 2 || xsize <= 0 || ysize <= 0)
ERRMSG(@"Invalid PNM header (xsize/ysize)");
if (xsize * ysize > (1 << 31))
ERRMSG(@"Invalid PNM header (image size:xsize*ysize too large)");
GET_LINE();
num = sscanf(buffer, "%d", &levels);
if (num != 1)
ERRMSG(@"Invalid PNM header (levels)");
colorspace = (ptype == '5') ? NSDeviceBlackColorSpace : NSDeviceRGBColorSpace;
self = [self initWithBitmapDataPlanes: NULL
pixelsWide: xsize
pixelsHigh: ysize
bitsPerSample: 8
samplesPerPixel: (ptype == '5') ? 1 : 3
hasAlpha: NO
isPlanar: NO
colorSpaceName: colorspace
bytesPerRow: 0
bitsPerPixel: 0];
if ([self bytesPerRow] * ysize > (length - (ptr - bytes)))
ERRMSG(@"Invalid PNM file (short data)");
pchar = [self bitmapData];
if (levels < 256)
{
memcpy(pchar, ptr, [self bytesPerRow] * ysize);
}
else
{
ERRMSG(@"Cannot handle > 256 level PNM files");
}
return self;
}
@end