diff --git a/ext/fox16_c/FXRuby.cpp b/ext/fox16_c/FXRuby.cpp index 260d74d572626b7b1ed299889db54464244ad531..11f2ab13c670cb6f5f1630f69ab1bd74f1b1cf36 100644 --- a/ext/fox16_c/FXRuby.cpp +++ b/ext/fox16_c/FXRuby.cpp @@ -466,17 +466,22 @@ FXuint FXRbNumberOfFXColors(VALUE string_or_ary){ return len; } -FXColor *FXRbConvertToFXColors(VALUE string_or_ary){ +FXColor *FXRbConvertToFXColors(VALUE string_or_ary, FXuint *opts){ FXColor* pix=0; if(TYPE(string_or_ary) == T_ARRAY){ if(FXMALLOC(&pix,FXColor,RARRAY_LEN(string_or_ary))){ + *opts |= IMAGE_OWNED; for(long i=0; i<RARRAY_LEN(string_or_ary); i++){ pix[i]=static_cast<FXColor>(NUM2UINT(rb_ary_entry(string_or_ary,i))); } } }else{ - if(FXMALLOC(&pix,FXColor,RSTRING_LEN(string_or_ary)/sizeof(FXColor))){ - memcpy(pix, RSTRING_PTR(string_or_ary), RSTRING_LEN(string_or_ary)); + if( *opts & IMAGE_OWNED ){ + if(FXMALLOC(&pix,FXColor,RSTRING_LEN(string_or_ary)/sizeof(FXColor))){ + memcpy(pix, RSTRING_PTR(string_or_ary), RSTRING_LEN(string_or_ary)); + } + }else{ + pix = (FXColor*)(RSTRING_PTR(string_or_ary)); } } return pix; diff --git a/ext/fox16_c/include/FXRbImage.h b/ext/fox16_c/include/FXRbImage.h index c071d8d6de853ae81e572f69b277f6ca5a78918c..f2d658265cce296e36c894463ef6e1081110a4a3 100644 --- a/ext/fox16_c/include/FXRbImage.h +++ b/ext/fox16_c/include/FXRbImage.h @@ -143,9 +143,11 @@ protected: #include "FXRbIdVirtuals.h" #include "FXRbDrawableVirtuals.h" #include "FXRbImageVirtuals.h" + + VALUE data_string; public: /// Create an image - FXRbImage(FXApp* a,const FXColor* pix=NULL,FXuint opts=0,FXint w=1,FXint h=1):FXImage(a,pix,opts,w,h){ + FXRbImage(FXApp* a,const FXColor* pix=NULL,FXuint opts=0,FXint w=1,FXint h=1):FXImage(a,pix,opts,w,h),data_string(Qnil){ FXRbRegisterAppSensitiveObject(this); } diff --git a/ext/fox16_c/include/FXRuby.h b/ext/fox16_c/include/FXRuby.h index 6f6b2335fc82b77d69b16835740c340d9050eab1..d54a3e7a094d3d8a9e9f1fba4a2e8dd5b2cc1cc4 100644 --- a/ext/fox16_c/include/FXRuby.h +++ b/ext/fox16_c/include/FXRuby.h @@ -163,7 +163,7 @@ extern VALUE FXRbMakeColorArray(const FXColor* colors,FXint w,FXint h); extern FXuint FXRbNumberOfFXColors(VALUE string_or_ary); // Allocate a FXColor buffer and populate with data -extern FXColor *FXRbConvertToFXColors(VALUE string_or_ary); +extern FXColor *FXRbConvertToFXColors(VALUE string_or_ary, FXuint *opts); extern void* FXRbGetExpectedData(VALUE recv,FXSelector key,VALUE data); diff --git a/ext/fox16_c/markfuncs.cpp b/ext/fox16_c/markfuncs.cpp index e001bbb3da2c5df972a1a0dccc2069e1ecd43ef6..65d2422c80cac15891e49ad717a669b93a774e67 100644 --- a/ext/fox16_c/markfuncs.cpp +++ b/ext/fox16_c/markfuncs.cpp @@ -121,6 +121,9 @@ void FXRbIcon::markfunc(FXIcon* icon){ void FXRbImage::markfunc(FXImage* image){ FXRbDrawable::markfunc(image); + if( image ){ + rb_gc_mark(dynamic_cast<FXRbImage*>(image)->data_string); + } } diff --git a/rdoc-sources/FXImage.rb b/rdoc-sources/FXImage.rb index 1bdb698ad097db8a72e3d381fb66fd058103a3da..46292c162d3196497093215c16029a7ff98e18b9 100755 --- a/rdoc-sources/FXImage.rb +++ b/rdoc-sources/FXImage.rb @@ -17,7 +17,11 @@ module Fox # if you intend to do repeated re-rendering of the image after it has been # created. # +IMAGE_OWNED+:: - # Does nothing - for backward compatibility only. + # If +IMAGE_OWNED+ is set, the image pixel data is copied into the image. + # If +IMAGE_OWNED+ is not set, a image pixel string is directly used as + # memory buffer of the image, without copying the data. + # If pixel data is given as an Array instead of a string, data is copied in + # any case. # +IMAGE_DITHER+:: # Dither image to look better # +IMAGE_NEAREST+:: @@ -91,7 +95,9 @@ module Fox # # Image pixels are stored bytewise as [RGBA] values. # - def pixel_string ; end + # Optional offset and size can be used to retrieve a part of + # the image data. Both are counted in bytes. + def pixel_string(offset=nil, size=nil) ; end # # Return the color of the pixel at (_x_, _y_). diff --git a/swig-interfaces/FXImage.i b/swig-interfaces/FXImage.i index b912a0bcd6f7c6cb26c9bbeffa8eb290a8f8a2af..d01dc08e43fdc696a0926029aa46f3c00d5e61ae 100644 --- a/swig-interfaces/FXImage.i +++ b/swig-interfaces/FXImage.i @@ -62,10 +62,11 @@ public: if(w*h != len){ rb_raise( rb_eArgError, "Array size does not match image size" ); } - pix=FXRbConvertToFXColors(string_or_ary); - opts|=IMAGE_OWNED; + pix=FXRbConvertToFXColors(string_or_ary, &opts); } - return new FXRbImage(a,pix,opts,w,h); + FXRbImage *img = new FXRbImage(a,pix,opts,w,h); + img->data_string = (opts & IMAGE_OWNED) ? Qnil : string_or_ary; + return img; } /// To get to the pixel data @@ -99,8 +100,8 @@ public: rb_raise( rb_eArgError, "Array size does not match image size" ); } - FXColor* pix=FXRbConvertToFXColors(string_or_ary); - opts|=IMAGE_OWNED; + FXColor* pix=FXRbConvertToFXColors(string_or_ary, &opts); + (dynamic_cast<FXRbImage*>(self))->data_string = (opts & IMAGE_OWNED) ? Qnil : string_or_ary; if( NIL_P(w) || NIL_P(h) ){ self->setData(pix,opts); }else{ @@ -129,6 +130,25 @@ public: return Qnil; } } + + VALUE pixel_string(FXlong offset, FXlong size){ + FXColor* data = self->getData(); + if (data) { + FXlong maxsize = self->getWidth()*self->getHeight()*sizeof(FXColor); + if( offset > maxsize || offset < 0 ){ + return Qnil; + } + if( offset+size > maxsize ){ + size = maxsize-offset; + }else if( size < 0 ){ + return Qnil; + } + + return rb_str_new((char*)data + offset, size); + } else { + return Qnil; + } + } } /// Get pixel at x,y diff --git a/swig-interfaces/ruby-typemaps.i b/swig-interfaces/ruby-typemaps.i index 929a054157e31cc82659da248ff5abbe72b9a1e7..25c1e936e618a6e346fbdef9550c0a45bcb50dff 100644 --- a/swig-interfaces/ruby-typemaps.i +++ b/swig-interfaces/ruby-typemaps.i @@ -43,6 +43,10 @@ %typemap(in) FXlong "$1 = NUM2LONG($input);"; %typemap(in) FXulong "$1 = NUM2ULONG($input);"; +%typecheck(SWIG_TYPECHECK_INTEGER) FXchar, FXuchar, FXshort, FXushort, FXint, FXuint, FXlong, FXulong { + $1 = (TYPE($input) == T_FIXNUM || TYPE($input) == T_BIGNUM) ? 1 : 0; +} + /* Type-checking rules */ %typecheck(SWIG_TYPECHECK_STRING) const FXString&, FXuchar *data { $1 = (NIL_P($input) || TYPE($input) == T_STRING) ? 1 : 0; diff --git a/test/TC_FXImage.rb b/test/TC_FXImage.rb index 36ff1d694eb5ef947e5bd5c0ca759245bc487c40..e683c46795997e3909f96d929f533c22b869ee83 100755 --- a/test/TC_FXImage.rb +++ b/test/TC_FXImage.rb @@ -76,7 +76,7 @@ class TC_FXImage < Fox::TestCase def test_setPixels_string img = FXImage.new(app, nil, 0, 2, 1) img.pixels = "rgbaRGBA" - assert_equal(IMAGE_OWNED, img.options) + assert_equal(0, img.options) assert_equal("rgbaRGBA", img.pixel_string) end @@ -101,7 +101,7 @@ class TC_FXImage < Fox::TestCase end def test_create_with_data - img = FXImage.new(app, "rgbaRGBA", 0, 1, 2) + img = FXImage.new(app, "rgbaRGBA", IMAGE_OWNED, 1, 2) assert_equal("rgbaRGBA", img.pixel_string) img.create assert_nil(img.pixels) @@ -112,6 +112,44 @@ class TC_FXImage < Fox::TestCase assert_not_nil(img.pixels) end + def image_with_non_owned_data + FXImage.new(app, "rgbaRGBA", 0, 1, 2) + end + + def test_create_with_non_owned_data + GC.stress = true + img = image_with_non_owned_data + " " * 10000 + GC.stress = false + assert_equal("rgbaRGBA", img.pixel_string) + assert_equal(0, img.options) + img.create + end + + def set_non_owned_data(img) + img.setPixels("rgbaRGBA", 0, 2, 1) + end + + def test_set_pixel_with_non_owned_data + img = FXImage.new(app, nil, 0, 1, 2) + GC.stress = true + set_non_owned_data(img) + " " * 10000 + GC.stress = false + assert_equal("rgbaRGBA", img.pixel_string) + assert_equal(0, img.options) + img.create + end + + def test_partial_pixel_string + img = FXImage.new(app, "rgbaRGBA", IMAGE_OWNED, 1, 2) + assert_equal("baR", img.pixel_string(2,3)) + assert_equal(nil, img.pixel_string(-2,3)) + assert_equal(nil, img.pixel_string(2,-1)) + assert_equal(nil, img.pixel_string(10,3)) + assert_equal("baRGBA", img.pixel_string(2,10)) + end + # # Restore client-side pixel buffer from image. #