static int action_flip (int argc, const char *argv[]) { if (ot.postpone_callback (1, action_flip, argc, argv)) return 0; ot.read (); ImageRecRef A = ot.pop(); ot.push (new ImageRec (*A, ot.allsubimages ? -1 : 0, ot.allsubimages ? -1 : 0, true, false)); int subimages = ot.curimg->subimages(); for (int s = 0; s < subimages; ++s) { int miplevels = ot.curimg->miplevels(s); for (int m = 0; m < miplevels; ++m) { const ImageBuf &Aib ((*A)(s,m)); ImageBuf &Rib ((*ot.curimg)(s,m)); ImageBuf::ConstIterator<float> a (Aib); ImageBuf::Iterator<float> r (Rib); int nchans = Rib.nchannels(); int firstscanline = Rib.ymin(); int lastscanline = Rib.ymax(); for ( ; ! r.done(); ++r) { a.pos (r.x(), lastscanline - (r.y() - firstscanline)); for (int c = 0; c < nchans; ++c) r[c] = a[c]; } } } return 0; }
static int action_add (int argc, const char *argv[]) { if (ot.postpone_callback (2, action_add, argc, argv)) return 0; ImageRecRef B (ot.pop()); ImageRecRef A (ot.pop()); ot.read (A); ot.read (B); ot.push (new ImageRec (*A, ot.allsubimages ? -1 : 0, ot.allsubimages ? -1 : 0, true, false)); int subimages = ot.curimg->subimages(); for (int s = 0; s < subimages; ++s) { int miplevels = ot.curimg->miplevels(s); for (int m = 0; m < miplevels; ++m) { const ImageBuf &Aib ((*A)(s,m)); const ImageBuf &Bib ((*B)(s,m)); if (! same_size (Aib, Bib)) { // FIXME: some day, there should be options of combining // differing images somehow. std::cerr << "oiiotool: " << argv[0] << " could not combine images of differing sizes\n"; continue; } ImageBuf &Rib ((*ot.curimg)(s,m)); ImageBufAlgo::add (Rib, Aib, Bib); } } return 0; }
static int action_diff (int argc, const char *argv[]) { if (ot.postpone_callback (2, action_diff, argc, argv)) return 0; int ret = do_action_diff (*ot.image_stack.back(), *ot.curimg, ot); if (ret != DiffErrOK && ret != DiffErrWarn) ot.return_value = EXIT_FAILURE; return 0; }
static int action_colorconvert (int argc, const char *argv[]) { ASSERT (argc == 3); if (ot.postpone_callback (1, action_colorconvert, argc, argv)) return 0; std::string fromspace = argv[1]; std::string tospace = argv[2]; ot.read (); bool need_transform = false; ImageRecRef A = ot.curimg; ot.read (A); for (int s = 0, send = A->subimages(); s < send; ++s) { for (int m = 0, mend = A->miplevels(s); m < mend; ++m) { const ImageSpec *spec = A->spec(s,m); need_transform |= spec->get_string_attribute("oiio:ColorSpace") != tospace; } } if (! need_transform) return 1; // no need to do anything ot.pop (); ot.push (new ImageRec (*A, ot.allsubimages ? -1 : 0, ot.allsubimages ? -1 : 0, true, false)); if (fromspace == "current") fromspace = A->spec(0,0)->get_string_attribute ("oiio:Colorspace", "Linear"); ColorProcessor *processor = ot.colorconfig.createColorProcessor (fromspace.c_str(), tospace.c_str()); if (! processor) return 1; for (int s = 0, send = A->subimages(); s < send; ++s) { for (int m = 0, mend = A->miplevels(s); m < mend; ++m) { ImageBufAlgo::colorconvert ((*ot.curimg)(s,m), (*A)(s,m), processor, false); ot.curimg->spec(s,m)->attribute ("oiio::Colorspace", tospace); } } ot.colorconfig.deleteColorProcessor (processor); return 1; }
static int action_select_subimage (int argc, const char *argv[]) { if (ot.postpone_callback (1, action_select_subimage, argc, argv)) return 0; ot.read (); if (ot.curimg->subimages() == 1) return 0; // --subimage on a single-image file is a no-op int subimage = std::min (atoi(argv[1]), ot.curimg->subimages()); ImageRecRef A = ot.pop(); ot.push (new ImageRec (*A, subimage)); return 0; }
static int set_full_to_pixels (int argc, const char *argv[]) { if (ot.postpone_callback (1, set_full_to_pixels, argc, argv)) return 0; ot.read (); ImageRecRef A = ot.curimg; ImageSpec &spec (*A->spec(0,0)); spec.full_x = spec.x; spec.full_y = spec.y; spec.full_width = spec.width; spec.full_height = spec.height; A->metadata_modified (true); return 0; }
static int action_selectmip (int argc, const char *argv[]) { if (ot.postpone_callback (1, action_unmip, argc, argv)) return 0; ot.read (); bool mipmapped = false; for (int s = 0, send = ot.curimg->subimages(); s < send; ++s) mipmapped |= (ot.curimg->miplevels(s) > 1); if (! mipmapped) { return 0; // --selectmip on an unmipped image is a no-op } ImageRecRef newimg (new ImageRec (*ot.curimg, -1, atoi(argv[1]), true, true)); ot.curimg = newimg; return 0; }
static int action_sub (int argc, const char *argv[]) { if (ot.postpone_callback (2, action_sub, argc, argv)) return 0; ImageRecRef B (ot.pop()); ImageRecRef A (ot.pop()); ot.read (A); ot.read (B); ot.push (new ImageRec (*A, ot.allsubimages ? -1 : 0, ot.allsubimages ? -1 : 0, true, false)); int subimages = ot.curimg->subimages(); for (int s = 0; s < subimages; ++s) { int miplevels = ot.curimg->miplevels(s); for (int m = 0; m < miplevels; ++m) { const ImageBuf &Aib ((*A)(s,m)); const ImageBuf &Bib ((*B)(s,m)); if (! same_size (Aib, Bib)) { // FIXME: some day, there should be options of combining // differing images somehow. std::cerr << "oiiotool: " << argv[0] << " could not combine images of differing sizes\n"; continue; } ImageBuf &Rib ((*ot.curimg)(s,m)); ImageBuf::ConstIterator<float> a (Aib); ImageBuf::ConstIterator<float> b (Bib); ImageBuf::Iterator<float> r (Rib); int nchans = Rib.nchannels(); for ( ; ! r.done(); ++r) { a.pos (r.x(), r.y()); b.pos (r.x(), r.y()); for (int c = 0; c < nchans; ++c) r[c] = a[c] - b[c]; } } } return 0; }
static int set_origin (int argc, const char *argv[]) { if (ot.postpone_callback (1, set_origin, argc, argv)) return 0; ot.read (); ImageRecRef A = ot.curimg; ImageSpec &spec (*A->spec(0,0)); int x = spec.x, y = spec.y; int w = spec.width, h = spec.height; adjust_geometry (w, h, x, y, argv[1]); if (spec.width != w || spec.height != h) std::cerr << argv[0] << " can't be used to change the size, only the origin\n"; if (spec.x != x || spec.y != y) { spec.x = x; spec.y = y; A->metadata_modified (true); } return 0; }
static int set_fullsize (int argc, const char *argv[]) { if (ot.postpone_callback (1, set_fullsize, argc, argv)) return 0; ot.read (); ImageRecRef A = ot.curimg; ImageSpec &spec (*A->spec(0,0)); int x = spec.full_x, y = spec.full_y; int w = spec.full_width, h = spec.full_height; adjust_geometry (w, h, x, y, argv[1]); if (spec.full_x != x || spec.full_y != y || spec.full_width != w || spec.full_height != h) { spec.full_x = x; spec.full_y = y; spec.full_width = w; spec.full_height = h; A->metadata_modified (true); } return 0; }