/*************************************************************************** CPicture.cpp (c) 2004-2006 - Daniel Campos Fernández GTK+ component Realizado para la Junta de Extremadura. Consejería de Educación Ciencia y Tecnología. Proyecto gnuLinEx 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 1, 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., 675 Mass Ave, Cambridge, MA 02139, USA. ***************************************************************************/ #define __CPICTURE_CPP #include "main.h" #include "gambas.h" #include "widgets.h" #include "ggambastag.h" #include "CDraw.h" #include "CScreen.h" #include "CPicture.h" CIMAGE *CIMAGE_create(gPicture *picture) { CIMAGE *pic; GB.New((void **)POINTER(&pic), GB.FindClass("Image"), 0, 0); if (picture) { pic->picture->unref(); pic->picture = picture; picture->setTag(new gGambasTag((void *)pic)); } return pic; } CPICTURE *CPICTURE_create(gPicture *picture) { CPICTURE *pic; GB.New((void **)POINTER(&pic), GB.FindClass("Picture"), 0, 0); if (picture) { pic->picture->unref(); pic->picture = picture; picture->setTag(new gGambasTag((void *)pic)); } return pic; } void* GTK_GetImage(GdkPixbuf *buf) { CIMAGE *pic = CIMAGE_create(new gPicture(buf)); g_object_ref(buf); return (void*)pic; } void* GTK_GetPicture(GdkPixbuf *buf) { CPICTURE *pic = CPICTURE_create(new gPicture(buf)); g_object_ref(buf); return (void*)pic; } /******************************************************************************* Image *******************************************************************************/ BEGIN_METHOD(CIMAGE_new, GB_INTEGER w; GB_INTEGER h; GB_BOOLEAN trans) int w = VARGOPT(w, 0); int h = VARGOPT(h, 0); bool trans = VARGOPT(trans, false); IMAGE = new gPicture(gPicture::MEMORY, w, h, trans); IMAGE->setTag(new gGambasTag((void *)THIS)); END_METHOD BEGIN_METHOD_VOID(CIMAGE_free) IMAGE->unref(); END_METHOD BEGIN_PROPERTY(CIMAGE_picture) CPICTURE *pic = CPICTURE_create(IMAGE->copy()); pic->picture->getPixmap(); GB.ReturnObject((void*)pic); END_PROPERTY BEGIN_METHOD(CIMAGE_resize, GB_INTEGER width; GB_INTEGER height) IMAGE->resize(VARG(width),VARG(height)); END_METHOD BEGIN_PROPERTY(CIMAGE_width) GB.ReturnInteger(IMAGE->width()); END_PROPERTY BEGIN_PROPERTY(CIMAGE_height) GB.ReturnInteger(IMAGE->height()); END_PROPERTY BEGIN_PROPERTY(CIMAGE_depth) GB.ReturnInteger(IMAGE->depth()); END_PROPERTY BEGIN_PROPERTY(CIMAGE_transparent) if (READ_PROPERTY) GB.ReturnBoolean(IMAGE->isTransparent()); else IMAGE->setTransparent(VPROP(GB_BOOLEAN)); END_PROPERTY BEGIN_METHOD(CIMAGE_load, GB_STRING path) CIMAGE *image; char *addr; int len; if (!GB.LoadFile(STRING(path), LENGTH(path), &addr, &len)) { gPicture *pic = gPicture::fromMemory(addr, len); GB.ReleaseFile(&addr, len); if (pic) { image = CIMAGE_create(pic); image->picture->getPixbuf(); GB.ReturnObject(image); return; } } GB.Error("Unable to load image"); END_METHOD BEGIN_METHOD(CIMAGE_save, GB_STRING path; GB_INTEGER quality) switch (IMAGE->save(GB.ToZeroString(ARG(path)), VARGOPT(quality, -1))) { case 0: break; case -1: GB.Error("Unknown format"); break; case -2: GB.Error("Unable to save picture"); break; } END_METHOD BEGIN_METHOD_VOID(CIMAGE_clear) IMAGE->clear(); END_METHOD BEGIN_METHOD(CIMAGE_fill, GB_INTEGER col) IMAGE->fill(VARG(col)); END_METHOD BEGIN_METHOD(CIMAGE_copy, GB_INTEGER x; GB_INTEGER y; GB_INTEGER w; GB_INTEGER h) CIMAGE *img; int x=0,y=0,w=IMAGE->width(),h=IMAGE->height(); if (!MISSING(x)) x=VARG(x); if (!MISSING(y)) y=VARG(y); if (!MISSING(w)) w=VARG(w); if (!MISSING(h)) h=VARG(h); img = CIMAGE_create(IMAGE->copy(x, y, w, h)); GB.ReturnObject((void*)img); END_METHOD BEGIN_METHOD(CIMAGE_stretch, GB_INTEGER width; GB_INTEGER height; GB_BOOLEAN smooth) CIMAGE *img; bool smooth = VARGOPT(smooth, true); img = CIMAGE_create(IMAGE->stretch(VARG(width), VARG(height), smooth)); GB.ReturnObject((void*)img); END_METHOD BEGIN_METHOD_VOID(CIMAGE_flip) CIMAGE *img; img = CIMAGE_create(IMAGE->flip()); GB.ReturnObject((void*)img); END_METHOD BEGIN_METHOD_VOID(CIMAGE_mirror) CIMAGE *img; img = CIMAGE_create(IMAGE->mirror()); GB.ReturnObject((void*)img); END_METHOD BEGIN_METHOD(CIMAGE_rotate, GB_FLOAT angle) CIMAGE *img; img = CIMAGE_create(IMAGE->rotate(VARG(angle))); GB.ReturnObject((void*)img); END_METHOD BEGIN_METHOD(CIMAGE_replace, GB_INTEGER src; GB_INTEGER dst; GB_BOOLEAN noteq) IMAGE->replace(VARG(src), VARG(dst), VARGOPT(noteq, false)); END_METHOD /******************************************************************************* .ImagePixels *******************************************************************************/ BEGIN_METHOD(CIMAGE_pixels_get, GB_INTEGER x; GB_INTEGER y) GB.ReturnInteger( IMAGE->getPixel(VARG(x),VARG(y)) ); END_METHOD BEGIN_METHOD(CIMAGE_pixels_put, GB_INTEGER col; GB_INTEGER x; GB_INTEGER y) IMAGE->putPixel(VARG(x),VARG(y),VARG(col)); END_METHOD BEGIN_PROPERTY(CIMAGE_data) GB.ReturnInteger((int)IMAGE->data()); END_PROPERTY BEGIN_METHOD(CIMAGE_draw, GB_OBJECT img; GB_INTEGER x; GB_INTEGER y; GB_INTEGER w; GB_INTEGER h; GB_INTEGER sx; GB_INTEGER sy; GB_INTEGER sw; GB_INTEGER sh) int x, y, w, h, sx, sy, sw, sh; CIMAGE *image = (CIMAGE *)VARG(img); gPicture *pic; if (GB.CheckObject(image)) return; pic = ((CIMAGE *)image)->picture; x = VARGOPT(x, 0); y = VARGOPT(y, 0); w = VARGOPT(w, -1); h = VARGOPT(h, -1); sx = VARGOPT(sx, 0); sy = VARGOPT(sy, 0); sw = VARGOPT(sw, -1); sh = VARGOPT(sh, -1); //DRAW_NORMALIZE(x, y, w, h, sx, sy, sw, sh, pic->width(), pic->height()); IMAGE->draw(pic, x, y, w, h, sx, sy, sw, sh); END_METHOD GB_DESC CImageDesc[] = { GB_DECLARE("Image", sizeof(CIMAGE)), //GB_STATIC_METHOD("FromString","Image",CIMAGE_fromMemory,"(Data)s"), GB_METHOD("_new", NULL, CIMAGE_new, "[(Width)i(Height)i(Transparent)b]"), GB_METHOD("_free", NULL, CIMAGE_free, NULL), GB_METHOD("_get", "i", CIMAGE_pixels_get, "(X)i(Y)i"), GB_METHOD("_put", NULL, CIMAGE_pixels_put, "(Color)i(X)i(Y)i"), GB_PROPERTY_READ("Width", "i", CIMAGE_width), GB_PROPERTY_READ("Height", "i", CIMAGE_height), GB_PROPERTY_READ("W", "i", CIMAGE_width), GB_PROPERTY_READ("H", "i", CIMAGE_height), GB_PROPERTY_READ("Depth", "i", CIMAGE_depth), GB_PROPERTY("Transparent", "b", CIMAGE_transparent), GB_PROPERTY_READ("Data", "p", CIMAGE_data), GB_STATIC_METHOD("Load", "Image", CIMAGE_load, "(Path)s"), GB_METHOD("Save", NULL, CIMAGE_save, "(Path)s[(Quality)i]"), GB_METHOD("Resize", NULL, CIMAGE_resize, "(Width)i(Height)i"), GB_METHOD("Clear", NULL, CIMAGE_clear, NULL), GB_METHOD("Fill", NULL, CIMAGE_fill, "(Color)i"), GB_METHOD("Replace", NULL, CIMAGE_replace, "(OldColor)i(NewColor)i[(NotEqual)b]"), GB_METHOD("Copy", "Image", CIMAGE_copy, "[(X)i(Y)i(Width)i(Height)i]"), GB_METHOD("Stretch", "Image", CIMAGE_stretch, "(Width)i(Height)i[(Smooth)b]"), GB_METHOD("Flip", "Image", CIMAGE_flip, NULL), GB_METHOD("Mirror", "Image", CIMAGE_mirror, NULL), GB_METHOD("Rotate", "Image", CIMAGE_rotate, "(Angle)f"), GB_METHOD("Draw", NULL, CIMAGE_draw, "(Image)Image;(X)i(Y)i[(Width)i(Height)i(SrcX)i(SrcY)i(SrcWidth)i(SrcHeight)i]"), GB_PROPERTY_READ("Picture", "Picture", CIMAGE_picture), GB_END_DECLARE }; /******************************************************************************* Picture Cache ********************************************************************************/ static GHashTable *_cache = NULL; static void destroy_key(char *key) { //fprintf(stderr, "destroy_key: '%s'\n", key); g_free(key); } static void destroy_value(CPICTURE *pic) { //fprintf(stderr, "destroy_value: %p\n", pic); GB.Unref((void **)POINTER(&pic)); } static CPICTURE *cache_get(char *key) { if (!_cache) return NULL; return (CPICTURE *)g_hash_table_lookup(_cache, (gconstpointer)key); } static void cache_add(char *key, CPICTURE *pic) { if (!_cache) _cache = g_hash_table_new_full(g_str_hash, g_str_equal, (GDestroyNotify)destroy_key, (GDestroyNotify)destroy_value); if (!key || !*key) return; GB.Ref((void *)pic); //fprintf(stderr, "cache_add: '%s' %p\n", key, pic); g_hash_table_replace(_cache, (gpointer)g_strdup(key), (gpointer)pic); } static void cache_flush() { if (!_cache) return; g_hash_table_destroy(_cache); _cache = NULL; } /******************************************************************************* class Picture *******************************************************************************/ #define LOAD_IMAGE_FUNC CPICTURE_load_image #define IMAGE_TYPE gPicture #define CREATE_IMAGE_FROM_MEMORY(_image, _addr, _len, _ok) \ _image = gPicture::fromMemory(_addr, _len); #define CREATE_PICTURE_FROM_IMAGE(_cpicture, _image) \ _cpicture = CPICTURE_create(_image); #define GET_FROM_CACHE(_key) (cache_get(_key)) #define GET_FROM_STOCK(_name, _len) gPicture::fromNamedIcon(_name, _len) #define INSERT_INTO_CACHE(_key, _cpicture) cache_add((_key), (_cpicture)) #define APPLICATION_THEME CAPPLICATION_Theme #include "gb.form.picture.h" // BEGIN_METHOD(CPICTURE_get, GB_STRING path) // // char *addr; // int len; // CPICTURE *pic=NULL; // gPicture *buf; // GB_VALUE *value; // char *key; // // key = g_strdup_printf("%s\n%.*s", GB.CurrentComponent(), (int)LENGTH(path), STRING(path)); // // pic = cache_get(key); // if (pic) // { // g_free(key); // GB.ReturnObject(pic); // return; // } // // if (GB.LoadFile(STRING(path),LENGTH(path),&addr,&len)) // { // buf=gPicture::fromNamedIcon(GB.ToZeroString(ARG(path))); // if (buf) // { // pic = CPICTURE_create(buf); // goto __OK; // } // // if (LENGTH(path) >= STOCK_PREFIX_LEN && strncmp(STRING(path), STOCK_PREFIX, STOCK_PREFIX_LEN) == 0) // { // if (LENGTH(path) == STOCK_PREFIX_LEN) { GB.Error("Unable to load image");return; } // if (init_stock()) { GB.Error("Unable to load image");return; } // // GB.Push(1, GB_T_STRING, &STRING(path)[STOCK_PREFIX_LEN], LENGTH(path) - STOCK_PREFIX_LEN); // value = GB.Call(&_stock_get_func, 1, false); // if (value->type >= GB_T_OBJECT) // pic = (CPICTURE *)((GB_OBJECT *)value)->value; // if (!pic) { GB.Error("Unable to load image");return; } // GB.ReturnObject(pic); // return; // // } // // //GB.Error("Unable to load image: &1", GB.ToZeroString(ARG(path))); // //return; // pic = NULL; // } // else // { // pic = CPICTURE_create(gPicture::fromMemory(addr, len)); // GB.ReleaseFile(&addr,len); // } // // __OK: // // if (pic) // cache_add(key, pic); // // g_free(key); // GB.ReturnObject(pic); // // END_METHOD /*BEGIN_METHOD(CPICTURE_put, GB_OBJECT picture; GB_STRING path) CPICTURE *picture=(CPICTURE*)VARG(picture); char *key; if (picture) { key = g_strdup_printf("%s\n%.*s", GB.CurrentComponent(), (int)LENGTH(path), STRING(path)); cache_add(key, picture); g_free(key); } END_METHOD */ ///////////// BEGIN_METHOD(CPICTURE_get, GB_STRING path) GB.ReturnObject(get_picture(STRING(path), LENGTH(path))); END_METHOD BEGIN_METHOD(CPICTURE_put, GB_OBJECT picture; GB_STRING path) set_picture(STRING(path), LENGTH(path), (CPICTURE *)VARG(picture)); END_METHOD BEGIN_METHOD_VOID(CPICTURE_flush) cache_flush(); END_METHOD BEGIN_METHOD(CPICTURE_new, GB_INTEGER w; GB_INTEGER h; GB_BOOLEAN trans) int w = VARGOPT(w, 0); int h = VARGOPT(h, 0); bool trans = VARGOPT(trans, false); PICTURE = new gPicture(gPicture::SERVER, w, h, trans); PICTURE->setTag(new gGambasTag(THIS)); END_METHOD BEGIN_METHOD_VOID(CPICTURE_free) PICTURE->unref(); END_METHOD BEGIN_METHOD(CPICTURE_resize, GB_INTEGER width; GB_INTEGER height) PICTURE->resize(VARG(width),VARG(height)); END_METHOD BEGIN_PROPERTY(CPICTURE_width) GB.ReturnInteger(PICTURE->width()); END_PROPERTY BEGIN_PROPERTY(CPICTURE_height) GB.ReturnInteger(PICTURE->height()); END_PROPERTY BEGIN_PROPERTY(CPICTURE_depth) GB.ReturnInteger(PICTURE->depth()); END_PROPERTY BEGIN_METHOD(CPICTURE_load, GB_STRING path) CPICTURE *picture; char *addr; int len; if (!GB.LoadFile(STRING(path), LENGTH(path), &addr, &len)) { gPicture *pic = gPicture::fromMemory(addr, len); GB.ReleaseFile(&addr, len); if (pic) { picture = CPICTURE_create(pic); GB.ReturnObject(picture); return; } } GB.Error("Unable to load picture"); END_METHOD /* BEGIN_METHOD(CPICTURE_fromMemory,GB_STRING data;) CPICTURE *pic=NULL; if (!LENGTH(data)) return; GB.New((void **)&pic, GB.FindClass("Picture"), 0, 0); pic->picture->fromMemory(STRING(data),LENGTH(data)); GB.ReturnObject(pic); END_METHOD */ BEGIN_METHOD(CPICTURE_save, GB_STRING path; GB_INTEGER quality) switch (PICTURE->save(GB.ToZeroString(ARG(path))), VARGOPT(quality, -1)) { case 0: break; case -1: GB.Error("Unknown format"); break; case -2: GB.Error("Unable to save picture"); break; } END_METHOD BEGIN_METHOD_VOID(CPICTURE_clear) PICTURE->clear(); END_METHOD BEGIN_METHOD(CPICTURE_fill, GB_INTEGER col) PICTURE->fill(VARG(col)); END_METHOD BEGIN_METHOD(CPICTURE_copy, GB_INTEGER x; GB_INTEGER y; GB_INTEGER w; GB_INTEGER h) CPICTURE *pic=NULL; int x=0; int y=0; int w=PICTURE->width(); int h=PICTURE->height(); if (!MISSING(x)) x=VARG(x); if (!MISSING(y)) y=VARG(y); if (!MISSING(w)) w=VARG(w); if (!MISSING(h)) h=VARG(h); pic = CPICTURE_create(PICTURE->copy(x, y, w, h)); GB.ReturnObject(pic); END_METHOD BEGIN_PROPERTY(CPICTURE_image) CIMAGE *img = CIMAGE_create(IMAGE->copy()); img->picture->getPixbuf(); GB.ReturnObject((void*)img); END_PROPERTY BEGIN_PROPERTY(CPICTURE_transparent) if (READ_PROPERTY) { GB.ReturnBoolean(PICTURE->isTransparent()); return; } PICTURE->setTransparent(VPROP(GB_BOOLEAN)); END_PROPERTY GB_DESC CPictureDesc[] = { GB_DECLARE("Picture", sizeof(CPICTURE)), //GB_STATIC_METHOD("_exit", NULL, CPICTURE_flush, NULL), GB_METHOD("_new", NULL, CPICTURE_new, "[(Width)i(Height)i(Transparent)b]"), GB_METHOD("_free", NULL, CPICTURE_free, NULL), GB_STATIC_METHOD("_exit",NULL,CPICTURE_flush,NULL), GB_STATIC_METHOD("_get", "Picture", CPICTURE_get, "(Path)s"), GB_STATIC_METHOD("_put", NULL, CPICTURE_put, "(Picture)Picture;(Path)s"), GB_STATIC_METHOD("Flush", NULL, CPICTURE_flush, NULL), GB_PROPERTY_READ("Width", "i", CPICTURE_width), GB_PROPERTY_READ("Height", "i", CPICTURE_height), GB_PROPERTY_READ("W", "i", CPICTURE_width), GB_PROPERTY_READ("H", "i", CPICTURE_height), GB_PROPERTY_READ("Depth", "i", CPICTURE_depth), GB_PROPERTY("Transparent", "b", CPICTURE_transparent), GB_STATIC_METHOD("Load", "Picture", CPICTURE_load, "(Path)s"), GB_METHOD("Save", NULL, CPICTURE_save, "(Path)s[(Quality)i]"), GB_METHOD("Resize", NULL, CPICTURE_resize, "(Width)i(Height)i"), GB_METHOD("Clear", NULL, CPICTURE_clear, NULL), GB_METHOD("Fill", NULL, CPICTURE_fill, "(Color)i"), GB_METHOD("Copy", "Picture", CPICTURE_copy, "[(X)i(Y)i(Width)i(Height)i]"), GB_PROPERTY_READ("Image", "Image", CPICTURE_image), GB_INTERFACE("Draw", &DRAW_Interface), GB_END_DECLARE };