1 // Written in the D programming language. 2 /++ 3 + Authors: KanzakiKino 4 + Copyright: KanzakiKino 2018 5 + License: LGPL-3.0 6 ++/ 7 module g4d.util.bitmap; 8 import gl3n.linalg; 9 import std.algorithm, 10 std.conv; 11 import core.stdc.stdlib, 12 core.stdc.string; 13 14 /// A class of bitmap. 15 class Bitmap ( _Type = ubyte, size_t _LengthPerPixel = 4 ) 16 { 17 /// Type of the number that shows color. 18 alias Type = _Type; 19 20 /// Length to show the color of pixel. 21 enum LengthPerPixel = _LengthPerPixel; 22 23 protected Type* _data; 24 /// Readonly pointer to data. 25 const @property data () { return _data; } 26 27 protected size_t _width, _rows; 28 /// Width of the bitmap. 29 const @property width () { return _width; } 30 /// Height of the bitmap. 31 const @property rows () { return _rows; } 32 33 /// Size of the bitmap. 34 const @property size () 35 { 36 return vec2i( width.to!int, rows.to!int ); 37 } 38 39 /// Length of the bitmap data. 40 const @property dataLength () 41 { 42 return _width*_rows*LengthPerPixel; 43 } 44 /// Size of the bitmap data in byte. 45 const @property dataByteSize () 46 { 47 return dataLength * Type.sizeof; 48 } 49 50 /// 51 this ( vec2i sz ) 52 { 53 _data = null; 54 resize( sz ); 55 clear(); 56 } 57 /// 58 this ( vec2i sz, Type* src ) 59 { 60 this( sz ); 61 memcpy( _data, src, dataByteSize ); 62 } 63 /// 64 this ( vec2i sz, Type[] src ) 65 { 66 this( sz ); 67 memcpy( _data, src.ptr, dataByteSize ); 68 } 69 70 /// 71 ~this () 72 { 73 dispose(); 74 } 75 /// Releases all memories. 76 void dispose () 77 { 78 if ( _data ) { 79 free( _data ); 80 _data = null; 81 } 82 _width = 0; 83 _rows = 0; 84 } 85 86 protected void resize ( vec2i sz ) 87 { 88 dispose(); 89 _width = sz.x.to!uint; 90 _rows = sz.y.to!uint; 91 _data = cast(Type*)malloc( dataByteSize ); 92 } 93 94 /// Creates resized bitmap with the same format. 95 const auto conservativeResize ( vec2i sz ) 96 { 97 enum lpp = LengthPerPixel; 98 99 const srcw = width, srch = rows; 100 const src = data; 101 102 auto result = new Bitmap!(Type,lpp)( sz ); 103 const dstw = result.width, dsth = result.rows; 104 auto dst = result._data; 105 106 size_t dstx = 0, dsty = 0, 107 srci = 0, dsti = 0; 108 109 for ( dsty = 0; dsty < dsth; dsty++ ) { 110 for ( dstx = 0; dstx < dstw; dstx++ ) { 111 if ( dstx >= srcw || dsty >= srch ) { 112 static foreach ( j; 0..lpp ) { 113 dst[dsti++] = 0; 114 } 115 } else { 116 static foreach ( j; 0..lpp ) { 117 dst[dsti++] = src[srci++]; 118 } 119 } 120 } 121 } 122 return result; 123 } 124 125 /// Modifies a part of the bitmap. 126 void overwrite ( vec2i offset, typeof(this) bmp ) 127 { 128 enum lpp = LengthPerPixel; 129 130 const srcl = offset.x, srct = offset.y; 131 const srcr = srcl+bmp.width, srcb = srct+bmp.rows; 132 const src = bmp.data; 133 134 const dstw = width, dsth = rows; 135 auto dst = _data; 136 137 assert( srcl >= 0 && srct >= 0 ); 138 assert( srcr <= dstw && srcb <= dsth ); 139 140 size_t dstx = srcl, dsty = srct, 141 srci = 0, dsti = 0; 142 143 for ( dsty = srct; dsty < srcb; dsty++ ) { 144 dsti = (dsty*dstw + srcl)*lpp; 145 for ( dstx = srcl; dstx < srcr; dstx++ ) { 146 static foreach ( j; 0..lpp ) { 147 dst[dsti++] = src[srci++]; 148 } 149 } 150 } 151 } 152 153 /// Fills the bitmap data with 0. 154 void clear () 155 { 156 memset( _data, 0, dataByteSize ); 157 } 158 159 /// To prove this is bitmap. 160 enum this_is_a_bitmap_class_of_g4d = true; 161 } 162 163 /// 164 alias BitmapA = Bitmap!(ubyte,1); 165 /// 166 alias BitmapRGB = Bitmap!(ubyte,3); 167 /// 168 alias BitmapRGBA = Bitmap!(ubyte,4); 169 170 /// 171 alias BitmapAf = Bitmap!(float,1); 172 /// 173 alias BitmapRGBf = Bitmap!(float,3); 174 /// 175 alias BitmapRGBAf = Bitmap!(float,4); 176 177 /// Checks if T is bitmap. 178 enum isBitmap(T) = 179 __traits(hasMember,T,"this_is_a_bitmap_class_of_g4d");