JunkBox_Lib++ (for Windows) 1.10.1
Loading...
Searching...
No Matches
JpegTool.cpp
Go to the documentation of this file.
1
12#include "JpegTool.h"
13
14#ifdef ENABLE_JPEG
15
16
17using namespace jbxl;
18
19
25void JPEGImage::init(void)
26{
27 xs = 0;
28 ys = 0;
29 col = 0;
30 state = 0;
31 gp = NULL;
32 image = NULL;
33 return;
34}
35
36
42bool JPEGImage::isNull(void)
43{
44 if (gp==NULL || image==NULL) return true;
45 return false;
46}
47
48
52void JPEGImage::clear(void)
53{
54 if (gp!=NULL) memset(gp, 0, sizeof(JSAMPLE)*xs*ys*col);
55
56 return;
57}
58
59
63void JPEGImage::fill(JSAMPLE v)
64{
65 if (gp==NULL) return;
66 for (int i=0; i<xs*ys*col; i++) gp[i] = v;
67
68 return;
69}
70
71
77void JPEGImage::free(void)
78{
79 if (gp!=NULL) ::free(gp);
80 if (image!=NULL) ::free(image);
81 init();
82
83 return;
84}
85
86
90void JPEGImage::set(int x, int y, int c)
91{
92 if (c!=1) c = 3;
93
94 getm(x, y, c);
95 if (gp==NULL) return;
96
97 xs = x;
98 ys = y;
99 col = c;
100 state = 0;
101
102 memset(gp, 0, sizeof(JSAMPLE)*c*x*y);
103
104 return;
105}
106
107
111void JPEGImage::getm(int x, int y, int c)
112{
113 image = (JSAMPARRAY)malloc(sizeof(JSAMPROW)*y);
114 if (image==NULL) {
116 return;
117 }
118 memset(image, 0, sizeof(JSAMPROW)*y);
119
120 gp = (JSAMPLE*)malloc(sizeof(JSAMPLE)*c*x*y);
121 if (gp==NULL) {
122 freeNull(image);
124 return;
125 }
126 memset(gp, 0, sizeof(JSAMPLE)*c*x*y);
127
128 int j;
129 for (j=0; j<y; j++) image[j] = (JSAMPROW)&gp[j*c*x];
130
131 return;
132}
133
134
135
137
150JPEGImage jbxl::readJPEGFile(const char* fname)
151{
152 JPEGImage jp;
153 FILE* fp;
154
155 fp = fopen(fname, "rb");
156 if (fp==NULL) {
157 jp.gp = NULL;
158 jp.state = JBXL_GRAPH_OPFILE_ERROR;
159 return jp;
160 }
161
162 jp = readJPEGData(fp);
163 fclose(fp);
164
165 return jp;
166}
167
168
181JPEGImage jbxl::readJPEGData(FILE* fp)
182{
183 JPEGImage jp;
184 int xs, ys, col;
185 struct jpeg_decompress_struct jdat;
186 struct jpeg_error_mgr jerr;
187
188 fseek(fp, 0, 0);
189 jp.init();
190 jdat.err = jpeg_std_error(&jerr);
191 jpeg_create_decompress(&jdat);
192
193 // エラーハンドラー
194/* jdat.client_data = "Client Data";
195 jerr.error_exit = jpeg_error_exit;
196 jerr.output_message = ptint_message;
197*/
198 jpeg_stdio_src(&jdat, fp);
199 jpeg_read_header(&jdat, TRUE);
200 jpeg_start_decompress(&jdat);
201
202 xs = jdat.output_width;
203 ys = jdat.output_height;
204 col = jdat.output_components;
205 if (xs<=0 || ys<=0 || col<=0) {
206 jpeg_destroy_decompress(&jdat);
207 fclose(fp);
208 jp.state = JBXL_GRAPH_HEADER_ERROR;
209 return jp;
210 }
211
212 jp.set(xs, ys, col);
213 if (jp.state) return jp;
214
215 int rmn = jdat.output_height;
216 while(rmn>0) {
217 jpeg_read_scanlines(&jdat, jp.image+jdat.output_scanline, rmn);
218 rmn = jdat.output_height - jdat.output_scanline;
219 }
220 jpeg_finish_decompress (&jdat);
221 jpeg_destroy_decompress(&jdat);
222
223 return jp;
224}
225
226
244int jbxl::writeJPEGFile(const char* fname, JPEGImage* jp, int qulty)
245{
246 FILE* fp;
247 int ret;
248
249 if (fname==NULL) return JBXL_GRAPH_IVDARG_ERROR;
250 if (jp->col!=1 && jp->col!=3) return JBXL_GRAPH_IVDCOLOR_ERROR;
251 if (jp->gp==NULL || jp->image==NULL) return JBXL_GRAPH_NODATA_ERROR;
252
253 fp = fopen(fname, "wb");
254 if (fp==NULL) {
256 }
257
258 ret = writeJPEGData(fp, jp, qulty);
259 fclose(fp);
260
261 return ret;
262}
263
264
281int jbxl::writeJPEGData(FILE* fp, JPEGImage* jp, int qulty)
282{
283 struct jpeg_compress_struct jdat;
284 struct jpeg_error_mgr jerr;
285
286 if (fp==NULL) return JBXL_GRAPH_OPFILE_ERROR;
287 if (jp->col!=1 && jp->col!=3) return JBXL_GRAPH_IVDCOLOR_ERROR;
288 if (jp->gp==NULL || jp->image==NULL) return JBXL_GRAPH_NODATA_ERROR;
289
290 if (qulty>100) qulty = 100;
291 else if (qulty<0) qulty = 0;
292
293 jdat.err = jpeg_std_error(&jerr);
294 jpeg_create_compress(&jdat);
295
296 // エラーハンドラ
297/* jdat.client_data = "Client Data";
298 jerr.error_exit = jpeg_error_exit;
299 jerr.output_message = ptint_message;
300*/
301 fseek(fp, 0, 0);
302 jpeg_stdio_dest(&jdat, fp);
303
304 jdat.image_width = jp->xs;
305 jdat.image_height = jp->ys;
306 jdat.input_components = jp->col;
307 if (jp->col==1) jdat.in_color_space = JCS_GRAYSCALE;
308 else jdat.in_color_space = JCS_RGB;
309
310 jpeg_set_quality (&jdat, qulty, TRUE);
311 jpeg_set_defaults(&jdat);
312
313 jpeg_start_compress (&jdat, TRUE);
314 jpeg_write_scanlines(&jdat, jp->image, jp->ys);
315 jpeg_finish_compress(&jdat);
316
317 jpeg_destroy_compress(&jdat);
318
319 return 0;
320}
321
322
323
325
337CmnHead jbxl::JPEGImage2CmnHead(JPEGImage jp)
338{
339 CmnHead hd;
340 int i, j, k, yp, zp;
341
342 memset(&hd, 0, sizeof(CmnHead));
343
344 if (jp.isNull()) {
345 hd.kind = HEADER_NONE;
347 return hd;
348 }
349
350 // color
351 if (jp.col==1) {
352 hd.kind = JPEG_MONO_DATA;
353 }
354 else if (jp.col==3) {
355 hd.kind = JPEG_RGB_DATA;
356 }
357 else {
358 hd.kind = HEADER_NONE;
360 return hd;
361 }
362
363 hd.xsize = jp.xs;
364 hd.ysize = jp.ys;
365 hd.zsize = jp.col;
366 hd.depth = 8;
367 hd.bsize = 0;
368 hd.lsize = jp.xs*jp.ys*jp.col*((hd.depth+7)/8);
369 hd.buf = NULL;
370 hd.grptr = (uByte*)malloc(hd.lsize);
371 if (hd.grptr==NULL) {
372 hd.kind = HEADER_NONE;
374 return hd;
375 }
376 memset(hd.grptr, 0, hd.lsize);
377
378 for (k=0; k<jp.col; k++) {
379 zp = k*jp.xs*jp.ys;
380 for (j=0; j<jp.ys; j++) {
381 yp = zp + j*jp.xs;
382 for (i=0; i<jp.xs; i++) {
383 hd.grptr[yp + i] = (uByte)jp.point(i, j, k);
384 }
385 }
386 }
387
388 return hd;
389}
390
391
404JPEGImage jbxl::CmnHead2JPEGImage(CmnHead hd)
405{
406 JPEGImage jp;
407 int i, j, k, yp, zp, col;
408 int kind = hd.kind & 0x00ff;
409
410 jp.init();
411
412 if (hd.grptr==NULL) {
413 jp.state = JBXL_GRAPH_NODATA_ERROR;
414 return jp;
415 }
416 if (hd.depth!=8 && hd.depth!=16) { // 24bit, 32bit は未サポート
417 jp.state = JBXL_GRAPH_IVDARG_ERROR;
418 return jp;
419 }
420
421 // カラー
422 if (kind==JPEG_MONO_DATA && hd.zsize==1) col = 1;
423 else if (kind==JPEG_RGB_DATA && hd.zsize==3) col = 3;
424 else if ((kind==JPEG_ARGB_DATA || kind==JPEG_RGBA_DATA) && hd.zsize==4) col = 3;
425 else if ((kind==JPEG16_RGB_DATA || kind==JPEG16_ARGB_DATA || kind==JPEG16_RGBA_DATA) && hd.zsize==1 && hd.depth==16) col = 3;
426 else {
427 jp.state = JBXL_GRAPH_IVDARG_ERROR;
428 return jp;
429 }
430
431 jp.set(hd.xsize, hd.ysize, col);
432 if (jp.isNull()) return jp;
433
434 // JPEG16_RGB_DATA: R5G6B5 -> R8G8B8
435 if (kind==JPEG16_RGB_DATA && hd.depth==16) {
436 for (j=0; j<jp.ys; j++) {
437 yp = j*jp.xs;
438 for (i=0; i<jp.xs; i++) {
439 uWord* ptr = (uWord*)&hd.grptr[(yp+i)*2];
440 if (isLittleEndian) {
441 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0xf800)>>11)*255)/32);
442 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x07e0)>>5) *255)/64);
443 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0x001f)) *255)/32);
444 }
445 else {
446 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0x001f)) *255)/32);
447 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x07e0)>>5) *255)/64);
448 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0xf800)>>11)*255)/32);
449 }
450 }
451 }
452 }
453
454 // JPEG16_ARGB_DATA: A4R4G4B4 -> R8G8B8
455 else if (kind==JPEG16_ARGB_DATA && hd.depth==16) {
456 for (j=0; j<jp.ys; j++) {
457 yp = j*jp.xs;
458 for (i=0; i<jp.xs; i++) {
459 uWord* ptr = (uWord*)&hd.grptr[(yp+i)*2];
460 if (isLittleEndian) {
461 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0x0f00)>>8) *255)/16);
462 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x00f0)>>4) *255)/16);
463 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0x000f)) *255)/16);
464 }
465 else {
466 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0x00f0)>>4) *255)/16);
467 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x0f00)>>8) *255)/16);
468 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0xf000)>>12)*255)/16);
469 }
470 }
471 }
472 }
473
474 // JPEG16_ARGB_DATA: R4G4B4A4 -> R8G8B8
475 else if (kind==JPEG16_RGBA_DATA && hd.depth==16) {
476 for (j=0; j<jp.ys; j++) {
477 yp = j*jp.xs;
478 for (i=0; i<jp.xs; i++) {
479 uWord* ptr = (uWord*)&hd.grptr[(yp+i)*2];
480 if (isLittleEndian) {
481 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0xf000)>>12)*255)/16);
482 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x0f00)>>8) *255)/16);
483 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0x00f0)>>4) *255)/16);
484 }
485 else {
486 jp.point(i, j, 0) = (JSAMPLE)(((int)(((*ptr)&0x000f)) *255)/16);
487 jp.point(i, j, 1) = (JSAMPLE)(((int)(((*ptr)&0x00f0)>>4) *255)/16);
488 jp.point(i, j, 2) = (JSAMPLE)(((int)(((*ptr)&0x0f00)>>8) *255)/16);
489 }
490 }
491 }
492 }
493
494 // JPEG_ARGB_DATA (アルファチャンネルの削除): A8R8G8B8 -> R8G8B8
495 else if (kind==JPEG_ARGB_DATA) {
496 if (hd.depth==8) {
497 for (k=1; k<=3; k++) {
498 zp = k*jp.xs*jp.ys;
499 for (j=0; j<jp.ys; j++) {
500 yp = zp + j*jp.xs;
501 for (i=0; i<jp.xs; i++) {
502 jp.point(i, j, k-1) = (JSAMPLE)hd.grptr[yp+i];
503 }
504 }
505 }
506 }
507
508 // 16bit->8bit 変換
509 else if (hd.depth==16) {
510 int max = 0;
511 uWord* ptr = (uWord*)hd.grptr;
512 for (i=hd.xsize*hd.ysize; i<hd.xsize*hd.ysize*hd.zsize; i++) {
513 if (max<ptr[i]) max = ptr[i];
514 }
515
516 for (k=1; k<=3; k++) {
517 zp = k*jp.xs*jp.ys;
518 for (j=0; j<jp.ys; j++) {
519 yp = zp + j*jp.xs;
520 for (i=0; i<jp.xs; i++) {
521 if (max<=255) {
522 if (isLittleEndian) jp.point(i, j, k-1) = (JSAMPLE)hd.grptr[(yp+i)*2];
523 else jp.point(i, j, k-1) = (JSAMPLE)hd.grptr[(yp+i)*2 + 1];
524 }
525 else {
526 jp.point(i, j, k-1) = (JSAMPLE)((ptr[yp+i]*255)/max);
527 }
528 }
529 }
530 }
531 }
532 }
533
534 // Default: JPEG_RGB_DATA/JPEG_MONO_DATA/JPEG_RGBA_DATA
535 else {
536 if (hd.depth==8) {
537 for (k=0; k<jp.col; k++) {
538 zp = k*jp.xs*jp.ys;
539 for (j=0; j<jp.ys; j++) {
540 yp = zp + j*jp.xs;
541 for (i=0; i<jp.xs; i++) {
542 jp.point(i, j, k) = (JSAMPLE)hd.grptr[yp+i];
543 }
544 }
545 }
546 }
547
548 // 16bit->8bit 変換
549 else if (hd.depth==16) {
550 int max = 0;
551 uWord* ptr = (uWord*)hd.grptr;
552 for (i=0; i<hd.xsize*hd.ysize*hd.zsize; i++) {
553 if (max<ptr[i]) max = ptr[i];
554 }
555
556 for (k=0; k<jp.col; k++) {
557 zp = k*jp.xs*jp.ys;
558 for (j=0; j<jp.ys; j++) {
559 yp = zp + j*jp.xs;
560 for (i=0; i<jp.xs; i++) {
561 if (max<=255) {
562 if (isLittleEndian) jp.point(i, j, k) = (JSAMPLE)hd.grptr[(yp+i)*2];
563 else jp.point(i, j, k) = (JSAMPLE)hd.grptr[(yp+i)*2 + 1];
564 }
565 else {
566 jp.point(i, j, k) = (JSAMPLE)((ptr[yp+i]*255)/max);
567 }
568 }
569 }
570 }
571 }
572 }
573
574 return jp;
575}
576
577
578
580
587int jbxl::isJPEGHeader(Buffer buf)
588{
589 if (buf.vldsz<10) return FALSE;
590
591 unsigned char* head = buf.buf;
592
593 if (head[0]==0xff && head[1]==0xd8) {
594 if (head[2]==0xff) return TRUE;
595 /*
596 if (head[2]==0xff && head[3]==0xe0) {
597 if (!strncmp((const char*)&head[6], "JFIF", 4)) {
598 return TRUE;
599 }
600 }
601 if (head[2]==0xff && head[3]==0xe1) {
602 if (!strncmp((const char*)&head[6], "EXIF", 4)) {
603 return TRUE;
604 }
605 }
606 else if (head[2]==0xff && head[3]==0xdb) {
607 return TRUE;
608 }*/
609 }
610
611 return FALSE;
612}
613
614
615#endif // ENABLE_JPEGLIB
616
JPEGグラフィックデータ定義用ヘッダ
unsigned short uWord
2Byte
Definition common.h:334
#define TRUE
Definition common.h:226
#define FALSE
Definition common.h:223
unsigned char uByte
1Byte
Definition common.h:332
#define JPEG16_RGB_DATA
0x0022 // 書き込み用(16bit->32bit変換)
Definition gheader.h:186
#define JPEG_ARGB_DATA
0x0025 // 書き込み用(アルファチャンネルは削除される)
Definition gheader.h:189
#define HEADER_NONE
0x8000 // ヘッダ種別の指定なし
Definition gheader.h:212
#define JPEG_RGBA_DATA
0x0026 // 書き込み用(アルファチャンネルは削除される)
Definition gheader.h:190
#define JPEG_RGB_DATA
0x0020 // JPEG RGB
Definition gheader.h:184
#define JPEG16_ARGB_DATA
0x0023 // 書き込み用(16bit->32bit変換)
Definition gheader.h:187
#define JPEG_MONO_DATA
0x0021 // JPEG MONO
Definition gheader.h:185
#define JPEG16_RGBA_DATA
0x0024 // 書き込み用(16bit->32bit変換)
Definition gheader.h:188
#define JBXL_GRAPH_IVDARG_ERROR
無効な引数
Definition jbxl_state.h:178
#define JBXL_GRAPH_NODATA_ERROR
データが無い
Definition jbxl_state.h:171
#define JBXL_GRAPH_OPFILE_ERROR
ファイルのオープンエラー
Definition jbxl_state.h:173
#define JBXL_GRAPH_IVDCOLOR_ERROR
無効なカラー指定
Definition jbxl_state.h:183
#define JBXL_GRAPH_HEADER_ERROR
画像ヘッダーのエラー
Definition jbxl_state.h:169
#define JBXL_GRAPH_MEMORY_ERROR
メモリエラー
Definition jbxl_state.h:170
Definition Brep.h:29
void freeNull(T &p)
Definition common++.h:37
int vldsz
データの長さ.バイナリデータの場合も使用可能.文字列の場合は 0x00 を含まない.
Definition buffer.h:37
unsigned char * buf
バッファの先頭へのポインタ.str[bufsz]は必ず 0x00となる.
Definition buffer.h:39
unsigned int lsize
Size of Graphics Data (byte unit)
Definition gheader.h:133
uByte * buf
Ture Header buffer
Definition gheader.h:137
int zsize
For 3D Data (or Color)
Definition gheader.h:130
int kind
Kind of Graphics Format.
Definition gheader.h:127
unsigned int bsize
Fllowing buf size or Any Data (byte unit)
Definition gheader.h:132
uByte * grptr
Pointer to Data.
Definition gheader.h:138
int ysize
Height of Graphics.
Definition gheader.h:129
int depth
Color Depth of Graphics (bit unit)
Definition gheader.h:131
int xsize
Width of Graphics.
Definition gheader.h:128
#define isLittleEndian
Definition tools.h:423