// // OpenGLSprite.m /* * * Oolite * * Created by Giles Williams on Sat Apr 03 2004. * Copyright (c) 2004 for aegidian.org. All rights reserved. * Copyright (c) 2004, Giles C Williams All rights reserved. This work is licensed under the Creative Commons Attribution-NonCommercial-ShareAlike License. To view a copy of this license, visit http://creativecommons.org/licenses/by-nc-sa/2.0/ or send a letter to Creative Commons, 559 Nathan Abbott Way, Stanford, California 94305, USA. You are free: ¥ to copy, distribute, display, and perform the work ¥ to make derivative works Under the following conditions: ¥ Attribution. You must give the original author credit. ¥ Noncommercial. You may not use this work for commercial purposes. ¥ Share Alike. If you alter, transform, or build upon this work, you may distribute the resulting work only under a license identical to this one. For any reuse or distribution, you must make clear to others the license terms of this work. Any of these conditions can be waived if you get permission from the copyright holder. Your fair use and other rights are in no way affected by the above. */ #import "OpenGLSprite.h" @implementation OpenGLSprite - (id) init { self = [super init]; return self; } - (id) initWithImage:(NSImage *)textureImage cropRectangle:(NSRect)cropRect size:(NSSize) spriteSize { self = [super init]; [self makeTextureFromImage:textureImage cropRectangle:cropRect size:spriteSize]; return self; } - (id) initWithText:(NSString *)str { NSImage *image; NSSize strsize; NSMutableDictionary *stringAttributes = [NSMutableDictionary dictionaryWithObjectsAndKeys: [NSFont fontWithName:@"ArialNarrow-Bold" size:18], NSFontAttributeName, [NSColor blackColor], NSForegroundColorAttributeName, NULL]; strsize = [str sizeWithAttributes:stringAttributes]; strsize.width += 3; strsize.height += 1; image= [[NSImage alloc] initWithSize:strsize]; [image lockFocus]; [stringAttributes setObject:[NSColor blackColor] forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(2,0) withAttributes:stringAttributes]; [stringAttributes setObject:[NSColor yellowColor] forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(1,1) withAttributes:stringAttributes]; [image unlockFocus]; self = [super init]; [self makeTextureFromImage:image cropRectangle:NSMakeRect(0, 0, [image size].width, [image size].height) size:[image size]]; [image release]; return self; } - (id) initWithText:(NSString *)str ofColor:(NSColor *) textColor { NSImage *image; NSSize strsize; NSMutableDictionary *stringAttributes = [NSMutableDictionary dictionaryWithObjectsAndKeys: [NSFont fontWithName:@"ArialNarrow-Bold" size:18], NSFontAttributeName, [NSColor blackColor], NSForegroundColorAttributeName, NULL]; strsize = [str sizeWithAttributes:stringAttributes]; strsize.width += 3; strsize.height += 1; image= [[NSImage alloc] initWithSize:strsize]; [image lockFocus]; [stringAttributes setObject:[NSColor blackColor] forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(2,0) withAttributes:stringAttributes]; [stringAttributes setObject:textColor forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(1,1) withAttributes:stringAttributes]; [image unlockFocus]; self = [super init]; [self makeTextureFromImage:image cropRectangle:NSMakeRect(0, 0, [image size].width, [image size].height) size:[image size]]; [image release]; return self; } - (void) dealloc { const GLuint delTextures[1] = { texName }; glDeleteTextures(1, delTextures); // clean up the texture from the 3d card's memory if (textureData) [textureData release]; [super dealloc]; } - (NSSize) size { return size; } - (void)blitToX:(float)x Y:(float)y Z:(float)z Alpha:(float)a { if (a < 0.0) a = 0.0; // clamp the alpha value if (a > 1.0) a = 1.0; // clamp the alpha value glEnable(GL_TEXTURE_2D); glColor4f(1.0, 1.0, 1.0, a); // Note that the textured Quad is drawn ACW from the Top Left glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE); glBindTexture(GL_TEXTURE_2D, texName); glBegin(GL_QUADS); glTexCoord2f(0.0, 1.0-textureCropRect.size.height); glVertex3f(x, y+size.height, z); glTexCoord2f(0.0, 1.0); glVertex3f(x, y, z); glTexCoord2f(textureCropRect.size.width, 1.0); glVertex3f(x+size.width, y, z); glTexCoord2f(textureCropRect.size.width, 1.0-textureCropRect.size.height); glVertex3f(x+size.width, y+size.height, z); glEnd(); glDisable(GL_TEXTURE_2D); } - (void)blitCentredToX:(float)x Y:(float)y Z:(float)z Alpha:(float)a { float xs = x - size.width / 2.0; float ys = y - size.height / 2.0; [self blitToX:xs Y:ys Z:z Alpha:a]; } - (void) setText:(NSString *)str { NSImage *image; NSSize strsize; NSMutableDictionary *stringAttributes = [NSMutableDictionary dictionaryWithObjectsAndKeys: [NSFont fontWithName:@"ArialNarrow-Bold" size:18], NSFontAttributeName, [NSColor blackColor], NSForegroundColorAttributeName, NULL]; strsize = [str sizeWithAttributes:stringAttributes]; strsize.width += 3; strsize.height += 1; image= [[NSImage alloc] initWithSize:strsize]; [image lockFocus]; [stringAttributes setObject:[NSColor blackColor] forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(2,0) withAttributes:stringAttributes]; [stringAttributes setObject:[NSColor yellowColor] forKey:NSForegroundColorAttributeName]; [str drawAtPoint:NSMakePoint(1,1) withAttributes:stringAttributes]; [image unlockFocus]; self = [super init]; [self makeTextureFromImage:image cropRectangle:NSMakeRect(0, 0, [image size].width, [image size].height) size:[image size]]; [image release]; // //NSLog(@"%@ message sprite [ %f, %f ]", str, [image size].width, [image size].height); // } - (void)makeTextureFromImage:(NSImage *)texImage cropRectangle:(NSRect)cropRect size:(NSSize)spriteSize { NSBitmapImageRep* bitmapImageRep; NSRect textureRect = NSMakeRect(0.0,0.0,OPEN_GL_SPRITE_MIN_WIDTH,OPEN_GL_SPRITE_MIN_HEIGHT); NSImage* image; if (!texImage) return; size = spriteSize; textureCropRect = cropRect; while (textureRect.size.width < cropRect.size.width) textureRect.size.width *= 2; while (textureRect.size.height < cropRect.size.height) textureRect.size.height *= 2; textureRect.origin= NSMakePoint(0,0); textureCropRect.origin= NSMakePoint(0,0); textureSize = textureRect.size; image = [[NSImage alloc] initWithSize:textureRect.size]; // retained [image lockFocus]; [[NSColor clearColor] set]; NSRectFill(textureRect); [texImage drawInRect:textureCropRect fromRect:cropRect operation:NSCompositeSourceOver fraction:1.0]; bitmapImageRep = [[NSBitmapImageRep alloc] initWithFocusedViewRect:textureRect]; // retained [image unlockFocus]; [image release]; // released // normalise textureCropRect size to 0.0 -> 1.0 textureCropRect.size.width /= textureRect.size.width; textureCropRect.size.height /= textureRect.size.height; int n_bytes = 4 * textureRect.size.width * textureRect.size.height; if (textureData) [textureData release]; textureData = [[NSData dataWithBytes:[bitmapImageRep bitmapData] length:n_bytes] retain]; [bitmapImageRep release]; // released if (texName !=0) { const GLuint delTextures[1] = { texName }; glDeleteTextures(1, delTextures); texName = 0; // clean up the texture from the 3d card's memory } glPixelStorei(GL_UNPACK_ALIGNMENT, 1); glGenTextures(1, &texName); // get a new unique texture name glBindTexture(GL_TEXTURE_2D, texName); // initialise it glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, textureRect.size.width, textureRect.size.height, 0, GL_RGBA, GL_UNSIGNED_BYTE, [textureData bytes]); } - (void)replaceTextureFromImage:(NSImage *)texImage cropRectangle:(NSRect)cropRect { NSBitmapImageRep* bitmapImageRep; NSRect textureRect = NSMakeRect(0.0,0.0,OPEN_GL_SPRITE_MIN_WIDTH,OPEN_GL_SPRITE_MIN_HEIGHT); NSImage* image; if (!texImage) return; textureCropRect = cropRect; // correct size for texture to a power of two while (textureRect.size.width < cropRect.size.width) textureRect.size.width *= 2; while (textureRect.size.height < cropRect.size.height) textureRect.size.height *= 2; if ((textureRect.size.width != textureSize.width)||(textureRect.size.height != textureSize.height)) { NSLog(@"***** ERROR! replacement texture isn't the same size as original texture"); NSLog(@"***** cropRect %f x %f textureSize %f x %f",textureRect.size.width, textureRect.size.height, textureSize.width, textureSize.height); return; } textureRect.origin= NSMakePoint(0,0); textureCropRect.origin= NSMakePoint(0,0); image = [[NSImage alloc] initWithSize:textureRect.size]; [image lockFocus]; [[NSColor clearColor] set]; NSRectFill(textureRect); [texImage drawInRect:textureCropRect fromRect:cropRect operation:NSCompositeSourceOver fraction:1.0]; bitmapImageRep = [[NSBitmapImageRep alloc] initWithFocusedViewRect:textureRect]; [image unlockFocus]; [image release]; // normalise textureCropRect size to 0.0 -> 1.0 textureCropRect.size.width /= textureRect.size.width; textureCropRect.size.height /= textureRect.size.height; if (textureData) [textureData autorelease]; textureData = [[NSData dataWithBytes:[bitmapImageRep bitmapData] length:textureSize.width*textureSize.height*4] retain]; [bitmapImageRep release]; glBindTexture(GL_TEXTURE_2D, texName); glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, textureSize.width, textureSize.height, GL_RGBA, GL_UNSIGNED_BYTE, [textureData bytes]); } - (void)substituteTextureFromImage:(NSImage *)texImage { NSBitmapImageRep* bitmapImageRep; NSRect cropRect = NSMakeRect(0.0,0.0,[texImage size].width,[texImage size].height); NSRect textureRect = NSMakeRect(0.0,0.0,textureSize.width,textureSize.height); NSImage* image; if (!texImage) return; image = [[NSImage alloc] initWithSize:textureSize]; [image lockFocus]; [[NSColor clearColor] set]; NSRectFill(textureRect); [texImage drawInRect:textureRect fromRect:cropRect operation:NSCompositeSourceOver fraction:1.0]; bitmapImageRep = [[NSBitmapImageRep alloc] initWithFocusedViewRect:textureRect]; [image unlockFocus]; [image release]; // normalise textureCropRect size to 0.0 -> 1.0 textureCropRect = NSMakeRect(0.0,0.0,1.0,1.0); if ([bitmapImageRep bitsPerPixel]==32) { if (textureData) [textureData autorelease]; textureData = [[NSData dataWithBytes:[bitmapImageRep bitmapData] length:textureSize.width*textureSize.height*4] retain]; glBindTexture(GL_TEXTURE_2D, texName); glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, textureSize.width, textureSize.height, GL_RGBA, GL_UNSIGNED_BYTE, [textureData bytes]); } else if ([bitmapImageRep bitsPerPixel]==24) { if (textureData) [textureData autorelease]; textureData = [[NSData dataWithBytes:[bitmapImageRep bitmapData] length:textureSize.width*textureSize.height*3] retain]; glBindTexture(GL_TEXTURE_2D, texName); glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, textureSize.width, textureSize.height, GL_RGB, GL_UNSIGNED_BYTE, [textureData bytes]); } [bitmapImageRep release]; } @end