| |
| |
|
|
| #include "tpixelutils.h" |
| #include "tpalette.h" |
| #include "tcolorstyles.h" |
| #include "timage_io.h" |
| #include "tropcm.h" |
| #include "ttile.h" |
| #include "toonz/toonzscene.h" |
| #include "toonz/tcamera.h" |
| #include "autoadjust.h" |
| #include "autopos.h" |
| #include "cleanuppalette.h" |
| #include "cleanupcommon.h" |
| #include "tmsgcore.h" |
| #include "toonz/cleanupparameters.h" |
| |
| #include "toonz/tcleanupper.h" |
| |
| using namespace CleanupTypes; |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| namespace |
| { |
| |
| |
| |
| inline double affMV1(const TAffine &aff, double v1, double v2) |
| { |
| return aff.a11 * v1 + aff.a12 * v2 + aff.a13; |
| } |
| |
| |
| |
| inline double affMV2(const TAffine &aff, double v1, double v2) |
| { |
| return aff.a21 * v1 + aff.a22 * v2 + aff.a23; |
| } |
| |
| |
| |
| |
| struct HSVColor { |
| double m_h; |
| double m_s; |
| double m_v; |
| |
| public: |
| HSVColor(double h = 0, double s = 0, double v = 0) |
| : m_h(h), m_s(s), m_v(v) {} |
| |
| static HSVColor fromRGB(double r, double g, double b); |
| }; |
| |
| |
| |
| HSVColor HSVColor::fromRGB(double r, double g, double b) |
| { |
| double h, s, v; |
| double max, min, delta; |
| |
| max = std::max({r, g, b}); |
| min = std::min({r, g, b}); |
| |
| v = max; |
| |
| if (max != 0) |
| s = (max - min) / max; |
| else |
| s = 0; |
| |
| if (s == 0) |
| h = 0; |
| else { |
| delta = max - min; |
| |
| if (r == max) |
| h = (g - b) / delta; |
| else if (g == max) |
| h = 2.0 + (b - r) / delta; |
| else if (b == max) |
| h = 4.0 + (r - g) / delta; |
| h = h * 60.0; |
| if (h < 0) |
| h += 360.0; |
| } |
| |
| return HSVColor(h, s, v); |
| } |
| |
| |
| |
| |
| struct TargetColorData { |
| int m_idx; |
| HSVColor m_hsv; |
| double m_saturationLower; |
| double m_hueLower, m_hueUpper; |
| |
| public: |
| TargetColorData(const TargetColor &color) |
| : m_idx(-1), m_hsv(HSVColor::fromRGB(color.m_color.r / 255.0, color.m_color.g / 255.0, color.m_color.b / 255.0)), m_saturationLower(1.0 - color.m_threshold / 100.0), m_hueLower(m_hsv.m_h - color.m_hRange * 0.5), m_hueUpper(m_hsv.m_h + color.m_hRange * 0.5) |
| { |
| if (m_hueLower < 0.0) |
| m_hueLower += 360.0; |
| if (m_hueUpper > 360.0) |
| m_hueUpper -= 360.0; |
| } |
| }; |
| |
| |
| |
| |
| |
| #define MAX_N_PENCILS 8 |
| |
| |
| int N_pencils = 4; |
| TPixelRGBM32 Pencil[MAX_N_PENCILS + 1]; |
| int Pencil_index[MAX_N_PENCILS + 1]; |
| int Pencil_id[MAX_N_PENCILS + 1]; |
| TPixelRGBM32 Paper = TPixel32::White; |
| |
| |
| |
| |
| class TransfFunction |
| { |
| |
| USHORT TransfFun[(MAX_N_PENCILS + 1) << 8]; |
| |
| void setTransfFun(int pencil, int b1, int c1) |
| { |
| int i, p1, p2, brig, cont, max; |
| |
| cont = 255 - c1; |
| brig = 255 - b1; |
| max = 255; |
| notLessThan(1, cont); |
| p2 = brig; |
| p1 = p2 - cont; |
| |
| for (i = 0; i <= p1; i++) |
| TransfFun[pencil << 8 | i] = 0; |
| for (; i < p2; i++) |
| TransfFun[pencil << 8 | i] = std::min(max, max * (i - p1) / cont); |
| for (; i < 256; i++) |
| TransfFun[pencil << 8 | i] = max; |
| } |
| |
| public: |
| TransfFunction(const TargetColors &colors) |
| { |
| memset(TransfFun, 0, sizeof TransfFun); |
| int count = std::min(colors.getColorCount(), MAX_N_PENCILS); |
| for (int p = 0; p < count; p++) { |
| int brightness = troundp(2.55 * colors.getColor(p).m_brightness); |
| int contrast = troundp(2.55 * colors.getColor(p).m_contrast); |
| setTransfFun(p, brightness, contrast); |
| } |
| } |
| |
| USHORT *getTransfFun() { return TransfFun; } |
| }; |
| |
| |
| |
| |
| |
| void brightnessContrast(const TRasterCM32P &cm, const TargetColors &colors) |
| { |
| TransfFunction transform(colors); |
| USHORT *transf_fun = transform.getTransfFun(); |
| |
| int ink, tone; |
| int newTone, newInk; |
| |
| for (int y = 0; y < cm->getLy(); ++y) { |
| TPixelCM32 *pix = cm->pixels(y); |
| TPixelCM32 *endPix = pix + cm->getLx(); |
| |
| for (; pix < endPix; ++pix) { |
| tone = pix->getTone(); |
| if (tone < 255) { |
| ink = pix->getInk(); |
| newTone = transf_fun[ink << 8 | tone]; |
| newInk = (newTone == 255) ? 0 : colors.getColor(ink).m_index; |
| |
| *pix = TPixelCM32(newInk, 0, newTone); |
| } |
| } |
| } |
| } |
| |
| |
| |
| void brightnessContrastGR8(const TRasterCM32P &cm, const TargetColors &colors) |
| { |
| TransfFunction transform(colors); |
| USHORT *transf_fun = transform.getTransfFun(); |
| |
| int val, black = colors.getColor(1).m_index; |
| |
| for (int y = 0; y < cm->getLy(); ++y) { |
| TPixelCM32 *pix = cm->pixels(y); |
| TPixelCM32 *endPix = pix + cm->getLx(); |
| |
| for (; pix < endPix; ++pix) { |
| val = transf_fun[pix->getValue() + 256]; |
| *pix = (val < 255) ? TPixelCM32(black, 0, val) : TPixelCM32(); |
| } |
| } |
| } |
| |
| |
| |
| |
| |
| void transparencyCheck(const TRasterCM32P &cmin, const TRaster32P &rasout) |
| { |
| for (int y = 0; y < cmin->getLy(); ++y) { |
| TPixelCM32 *pix = cmin->pixels(y); |
| TPixelCM32 *endPix = pix + cmin->getLx(); |
| |
| TPixel32 *outPix = rasout->pixels(y); |
| for (; pix < endPix; ++pix, ++outPix) { |
| int ink = pix->getInk(); |
| int tone = pix->getTone(); |
| |
| if (ink == 4095) |
| *outPix = TPixel32::Green; |
| else |
| *outPix = (tone == 0) ? TPixel32::Black : (tone == 255) ? TPixel32::White : TPixel32::Red; |
| } |
| } |
| } |
| |
| } |
| |
| |
| |
| |
| |
| TCleanupper *TCleanupper::instance() |
| { |
| static TCleanupper theCleanupper; |
| return &theCleanupper; |
| } |
| |
| |
| |
| void TCleanupper::setParameters(CleanupParameters *parameters) |
| { |
| m_parameters = parameters; |
| } |
| |
| |
| |
| TPalette *TCleanupper::createToonzPaletteFromCleanupPalette() |
| { |
| TPalette *cleanupPalette = m_parameters->m_cleanupPalette.getPointer(); |
| return createToonzPalette(cleanupPalette, 1); |
| } |
| |
| |
| |
| |
| |
| TToonzImageP CleanupPreprocessedImage::getImg() const |
| { |
| return (TToonzImageP)(TImageCache::instance()->get(m_imgId, true)); |
| } |
| |
| |
| |
| CleanupPreprocessedImage::CleanupPreprocessedImage( |
| CleanupParameters *parameters, |
| TToonzImageP processed, |
| bool fromGr8) |
| : m_wasFromGR8(fromGr8), m_autocentered(false), m_size(processed->getSize()) |
| { |
| if (!processed) |
| m_imgId = ""; |
| else { |
| m_imgId = TImageCache::instance()->getUniqueId(); |
| assert(!processed->getRaster()->getParent()); |
| TImageCache::instance()->add(m_imgId, (TImageP)processed); |
| } |
| |
| if (!m_wasFromGR8) { |
| const TPixel32 white(255, 255, 255, 0); |
| for (int i = 0; i < parameters->m_colors.getColorCount(); ++i) { |
| TPixel32 cc = parameters->m_colors.getColor(i).m_color; |
| for (int tone = 0; tone < 256; tone++) { |
| m_pixelsLut.push_back( |
| blend(parameters->m_colors.getColor(i).m_color, |
| white, |
| tone, |
| TPixelCM32::getMaxTone())); |
| } |
| } |
| } |
| } |
| |
| |
| |
| CleanupPreprocessedImage::~CleanupPreprocessedImage() |
| { |
| TImageCache::instance()->remove(m_imgId); |
| } |
| |
| |
| |
| TRasterImageP CleanupPreprocessedImage::getPreviewImage() const |
| { |
| TRaster32P ras(getSize()); |
| TRasterImageP ri(ras); |
| double xdpi = 0, ydpi = 0; |
| getImg()->getDpi(xdpi, ydpi); |
| ri->setDpi(xdpi, ydpi); |
| return ri; |
| } |
| |
| |
| |
| |
| |
| bool TCleanupper::getResampleValues(const TRasterImageP &image, TAffine &aff, double &blur, |
| TDimension &outDim, TPointD &outDpi, bool isCameraTest, bool &isSameDpi) |
| { |
| double outlp, outlq; |
| double scalex, scaley; |
| double cxin, cyin, cpout, cqout; |
| double max_blur; |
| TPointD dpi; |
| |
| |
| image->getRaster()->lock(); |
| |
| |
| int rasterLx = image->getRaster()->getLx(); |
| int rasterLy = image->getRaster()->getLy(); |
| |
| |
| TRect saveBox = image->getSavebox(); |
| bool raster_is_savebox = true; |
| if (saveBox == TRect() && (saveBox.getLx() > 0 && saveBox.getLx() < rasterLx || |
| saveBox.getLy() > 0 && saveBox.getLy() < rasterLy)) |
| raster_is_savebox = false; |
| |
| image->getDpi(dpi.x, dpi.y); |
| if (dpi == TPointD()) { |
| dpi = getCustomDpi(); |
| if (dpi == TPointD()) |
| dpi.x = dpi.y = 65.0; |
| } else if (!dpi.x) |
| dpi.x = dpi.y; |
| else if (!dpi.y) |
| dpi.y = dpi.x; |
| |
| |
| int rotate = m_parameters->m_rotate; |
| |
| |
| { |
| m_parameters->getOutputImageInfo(outDim, outDpi.x, outDpi.y); |
| |
| |
| scalex = outDpi.x / dpi.x; |
| scaley = outDpi.y / dpi.y; |
| |
| outlp = outDim.lx; |
| outlq = outDim.ly; |
| } |
| |
| |
| isSameDpi = areAlmostEqual(outDpi.x, dpi.x, 0.1) && areAlmostEqual(outDpi.y, dpi.y, 0.1); |
| |
| |
| if (raster_is_savebox) { |
| cxin = -saveBox.getP00().x + (saveBox.getLx() - 1) / 2.0; |
| cyin = -saveBox.getP00().y + (saveBox.getLy() - 1) / 2.0; |
| } else { |
| cxin = (rasterLx - 1) / 2.0; |
| cyin = (rasterLy - 1) / 2.0; |
| } |
| |
| |
| cpout = (outlp - 1) / 2.0; |
| cqout = (outlq - 1) / 2.0; |
| |
| double angle = 0.0; |
| double skew = 0.0; |
| TAffine pre_aff; |
| image->getRaster()->lock(); |
| |
| bool autocentered = |
| doAutocenter( |
| angle, skew, |
| cxin, cyin, cqout, cpout, |
| dpi.x, dpi.y, |
| raster_is_savebox, saveBox, |
| image, scalex); |
| image->getRaster()->unlock(); |
| |
| |
| if (m_parameters->m_autocenterType == AUTOCENTER_CTR && skew) { |
| pre_aff.a11 = cos(skew * M_PI_180); |
| pre_aff.a21 = sin(skew * M_PI_180); |
| } |
| |
| aff = (TScale(scalex, scaley) * pre_aff) * TRotation(angle); |
| aff = aff.place(cxin, cyin, cpout, cqout); |
| |
| |
| TPointD pout = TPointD((outlp - 1) / 2.0, (outlq - 1) / 2.0); |
| |
| if (m_parameters->m_rotate != 0) |
| aff = TRotation(-(double)m_parameters->m_rotate).place(pout, pout) * aff; |
| |
| if (m_parameters->m_flipx || m_parameters->m_flipy) |
| aff = TScale(m_parameters->m_flipx ? -1 : 1, m_parameters->m_flipy ? -1 : 1).place(pout, pout) * aff; |
| |
| if (!isCameraTest) |
| aff = TTranslation(m_parameters->m_offx * outDpi.x / 2, m_parameters->m_offy * outDpi.y / 2) * aff; |
| |
| max_blur = 20.0 * sqrt(fabs(scalex )); |
| blur = pow(max_blur, (100 - m_parameters->m_sharpness) / (100 - 1)); |
| return autocentered; |
| } |
| |
| |
| |
| |
| TRasterP TCleanupper::processColors(const TRasterP &rin) |
| { |
| if (m_parameters->m_lineProcessingMode == lpNone) |
| return rin; |
| |
| TRasterCM32P rcm = TRasterCM32P(rin->getSize()); |
| if (!rcm) { |
| assert(!"failed finalRas allocation!"); |
| return TRasterCM32P(); |
| } |
| |
| |
| m_parameters->m_colors.update(m_parameters->m_cleanupPalette.getPointer(), m_parameters->m_noAntialias); |
| |
| bool toGr8 = (m_parameters->m_lineProcessingMode == lpGrey); |
| if (toGr8) { |
| |
| |
| |
| |
| |
| |
| rin->lock(); |
| rcm->lock(); |
| |
| if (TRasterGR8P(rin)) { |
| UCHAR *rowin = rin->getRawData(); |
| TUINT32 *rowout = reinterpret_cast<TUINT32 *>(rcm->getRawData()); |
| for (int i = 0; i < rin->getLy(); i++) { |
| for (int j = 0; j < rin->getLx(); j++) |
| *rowout++ = *rowin++; |
| rowin += rin->getWrap() - rin->getLx(); |
| rowout += rcm->getWrap() - rcm->getLx(); |
| } |
| } else { |
| TPixel32 *rowin = reinterpret_cast<TPixel32 *>(rin->getRawData()); |
| TUINT32 *rowout = reinterpret_cast<TUINT32 *>(rcm->getRawData()); |
| for (int i = 0; i < rin->getLy(); i++) { |
| for (int j = 0; j < rin->getLx(); j++) |
| *rowout++ = TPixelGR8::from(*rowin++).value; |
| rowin += rin->getWrap() - rin->getLx(); |
| rowout += rcm->getWrap() - rcm->getLx(); |
| } |
| } |
| |
| rin->unlock(); |
| rcm->unlock(); |
| } else { |
| assert(TRaster32P(rin)); |
| preprocessColors(rcm, rin, m_parameters->m_colors); |
| } |
| |
| |
| CleanupPreprocessedImage cpi(m_parameters, TToonzImageP(rcm, rcm->getBounds()), toGr8); |
| cpi.m_autocentered = true; |
| |
| TRaster32P rout = TRaster32P(rin->getSize()); |
| finalize(rout, &cpi); |
| return rout; |
| } |
| |
| |
| |
| CleanupPreprocessedImage *TCleanupper::process( |
| TRasterImageP &image, bool first_image, TRasterImageP &onlyResampledImage, |
| bool isCameraTest, bool returnResampled, bool onlyForSwatch, TAffine *resampleAff) |
| { |
| TAffine aff; |
| double blur; |
| TDimension outDim(0, 0); |
| TPointD outDpi; |
| |
| bool isSameDpi = false; |
| bool autocentered = getResampleValues(image, aff, blur, outDim, outDpi, isCameraTest, isSameDpi); |
| if (m_parameters->m_autocenterType != AUTOCENTER_NONE && !autocentered) |
| DVGui::warning(QObject::tr("The autocentering failed on the current drawing.")); |
| |
| bool fromGr8 = (bool)TRasterGR8P(image->getRaster()); |
| bool toGr8 = (m_parameters->m_lineProcessingMode == lpGrey); |
| |
| |
| if (!isCameraTest && m_parameters->m_lineProcessingMode != lpNone && toGr8 && m_parameters->m_autoAdjustMode != AUTO_ADJ_NONE && |
| !onlyForSwatch) { |
| static int ref_cum[256]; |
| UCHAR lut[256]; |
| int cum[256]; |
| double x0_src_f, y0_src_f, x1_src_f, y1_src_f; |
| int x0_src, y0_src, x1_src, y1_src; |
| |
| |
| |
| TAffine inv = aff.inv(); |
| |
| x0_src_f = affMV1(inv, 0, 0); |
| y0_src_f = affMV2(inv, 0, 0); |
| x1_src_f = affMV1(inv, outDim.lx - 1, outDim.ly - 1); |
| y1_src_f = affMV2(inv, outDim.lx - 1, outDim.ly - 1); |
| |
| x0_src = tround(x0_src_f); |
| y0_src = tround(y0_src_f); |
| x1_src = tround(x1_src_f); |
| y1_src = tround(y1_src_f); |
| |
| set_autoadjust_window(x0_src, y0_src, x1_src, y1_src); |
| |
| if (!TRasterGR8P(image->getRaster())) { |
| |
| TRaster32P ras32(image->getRaster()->clone()); |
| TRop::addBackground(ras32, TPixel32::White); |
| image = TRasterImageP(ras32); |
| ras32 = TRaster32P(); |
| |
| TRasterGR8P rgr(image->getRaster()->getSize()); |
| TRop::copy(rgr, image->getRaster()); |
| |
| |
| image->setRaster(rgr); |
| } |
| switch (m_parameters->m_autoAdjustMode) { |
| case AUTO_ADJ_HISTOGRAM: |
| if (first_image) { |
| build_gr_cum(image, ref_cum); |
| } else { |
| build_gr_cum(image, cum); |
| build_gr_lut(ref_cum, cum, lut); |
| apply_lut(image, lut); |
| } |
| break; |
| |
| case AUTO_ADJ_HISTO_L: |
| histo_l_algo(image, first_image); |
| break; |
| |
| case AUTO_ADJ_BLACK_EQ: |
| black_eq_algo(image); |
| break; |
| |
| case AUTO_ADJ_NONE: |
| default: |
| assert(false); |
| break; |
| } |
| } |
| |
| fromGr8 = (bool)TRasterGR8P(image->getRaster()); |
| |
| assert(returnResampled || !onlyForSwatch); |
| |
| |
| TRasterCM32P finalRas; |
| if (!onlyForSwatch) { |
| finalRas = TRasterCM32P(outDim); |
| if (!finalRas) { |
| TImageCache::instance()->outputMap(outDim.lx * outDim.ly * 4, "C:\\cachelog"); |
| assert(!"failed finalRas allocation!"); |
| return 0; |
| } |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| TRasterP tmp_ras; |
| |
| if (returnResampled || (fromGr8 && toGr8)) { |
| if (fromGr8 && toGr8) |
| tmp_ras = TRasterGR8P(outDim); |
| else |
| tmp_ras = TRaster32P(outDim); |
| |
| if (!tmp_ras) { |
| TImageCache::instance()->outputMap(outDim.lx * outDim.ly * 4, "C:\\cachelog"); |
| assert(!"failed tmp_ras allocation!"); |
| return 0; |
| } |
| } else |
| |
| |
| tmp_ras = TRaster32P(outDim.lx, outDim.ly, outDim.lx, (TPixel32 *)finalRas->getRawData()); |
| |
| TRop::ResampleFilterType flt_type; |
| if (isSameDpi) |
| flt_type = TRop::ClosestPixel; |
| else if (isCameraTest) |
| flt_type = TRop::Triangle; |
| else |
| flt_type = TRop::Hann2; |
| TRop::resample(tmp_ras, image->getRaster(), aff, flt_type, blur); |
| |
| if ((TRaster32P)tmp_ras) |
| |
| TRop::addBackground(tmp_ras, TPixel32::White); |
| |
| if (resampleAff) |
| *resampleAff = aff; |
| |
| image->getRaster()->unlock(); |
| image = TRasterImageP(); |
| |
| if (returnResampled) { |
| onlyResampledImage = TRasterImageP(tmp_ras); |
| onlyResampledImage->setDpi(outDpi.x, outDpi.y); |
| } |
| |
| if (onlyForSwatch) |
| return 0; |
| |
| assert(finalRas); |
| |
| |
| m_parameters->m_colors.update(m_parameters->m_cleanupPalette.getPointer(), m_parameters->m_noAntialias); |
| |
| if (toGr8) { |
| |
| |
| |
| |
| |
| tmp_ras->lock(); |
| finalRas->lock(); |
| assert(tmp_ras->getSize() == finalRas->getSize()); |
| assert(tmp_ras->getLx() == tmp_ras->getWrap()); |
| assert(finalRas->getLx() == finalRas->getWrap()); |
| |
| int pixCount = outDim.lx * outDim.ly; |
| |
| if (fromGr8) { |
| UCHAR *rowin = tmp_ras->getRawData(); |
| TUINT32 *rowout = reinterpret_cast<TUINT32 *>(finalRas->getRawData()); |
| for (int i = 0; i < pixCount; i++) |
| *rowout++ = *rowin++; |
| } else { |
| TPixel32 *rowin = reinterpret_cast<TPixel32 *>(tmp_ras->getRawData()); |
| TUINT32 *rowout = reinterpret_cast<TUINT32 *>(finalRas->getRawData()); |
| for (int i = 0; i < pixCount; i++) |
| *rowout++ = TPixelGR8::from(*rowin++).value; |
| } |
| |
| tmp_ras->unlock(); |
| finalRas->unlock(); |
| } else { |
| |
| assert(TRaster32P(tmp_ras)); |
| preprocessColors(finalRas, tmp_ras, m_parameters->m_colors); |
| } |
| |
| TToonzImageP final; |
| final = TToonzImageP(finalRas, finalRas->getBounds()); |
| final->setDpi(outDpi.x, outDpi.y); |
| |
| CleanupPreprocessedImage *cpi = new CleanupPreprocessedImage(m_parameters, final, toGr8); |
| cpi->m_autocentered = autocentered; |
| cpi->m_appliedAff = aff; |
| return cpi; |
| } |
| |
| |
| |
| TRasterImageP TCleanupper::autocenterOnly( |
| const TRasterImageP &image, bool isCameraTest, bool &autocentered) |
| { |
| double xDpi, yDpi; |
| |
| double skew = 0, angle = 0, dist = 0 ; |
| double cxin, cyin, cpout, cqout; |
| int rasterIsSavebox = true; |
| TAffine aff, preAff, inv; |
| int rasterLx, finalLx, rasterLy, finalLy; |
| |
| rasterLx = finalLx = image->getRaster()->getLx(); |
| rasterLy = finalLy = image->getRaster()->getLy(); |
| |
| TRect saveBox = image->getSavebox(); |
| if ((saveBox == TRect()) && ((saveBox.getLx() > 0 && saveBox.getLx() < rasterLx) || |
| (saveBox.getLy() > 0 && saveBox.getLy() < rasterLy))) |
| rasterIsSavebox = false; |
| |
| int rotate = m_parameters->m_rotate; |
| |
| image->getDpi(xDpi, yDpi); |
| |
| if (!xDpi) |
| xDpi = (yDpi ? yDpi : 65); |
| |
| if (!yDpi) |
| yDpi = (xDpi ? xDpi : 65); |
| |
| if (rasterIsSavebox) { |
| cxin = -saveBox.getP00().x + (saveBox.getLx() - 1) / 2.0; |
| cyin = -saveBox.getP00().y + (saveBox.getLy() - 1) / 2.0; |
| } else { |
| cxin = (rasterLx - 1) / 2.0; |
| cyin = (rasterLy - 1) / 2.0; |
| } |
| |
| cpout = (rasterLx - 1) / 2.0; |
| cqout = (rasterLy - 1) / 2.0; |
| |
| if (m_parameters->m_autocenterType != AUTOCENTER_NONE) |
| autocentered = doAutocenter(angle, skew, cxin, cyin, cqout, cpout, |
| xDpi, yDpi, rasterIsSavebox, saveBox, image, 1.0); |
| else |
| autocentered = true; |
| |
| if (m_parameters->m_autocenterType == AUTOCENTER_CTR && skew) { |
| aff.a11 = cos(skew * M_PI_180); |
| aff.a21 = sin(skew * M_PI_180); |
| } |
| |
| aff = aff * TRotation(angle); |
| |
| aff = aff.place(cxin, cyin, cpout, cqout); |
| |
| if (rotate != 0 && rotate != 180) |
| tswap(finalLx, finalLy); |
| |
| TPointD pin = TPointD((rasterLx - 1) / 2.0, (rasterLy - 1) / 2.0); |
| TPointD pout = TPointD((finalLx - 1) / 2.0, (finalLy - 1) / 2.0); |
| |
| if (rotate != 0) |
| aff = TRotation(-(double)rotate).place(pin, pout) * aff; |
| |
| if (m_parameters->m_flipx || m_parameters->m_flipy) |
| aff = TScale(m_parameters->m_flipx ? -1 : 1, m_parameters->m_flipy ? -1 : 1).place(pout, pout) * aff; |
| |
| if (!isCameraTest) |
| aff = TTranslation(m_parameters->m_offx * xDpi / 2, |
| m_parameters->m_offy * yDpi / 2) * |
| aff; |
| |
| TRasterP tmpRas; |
| TPoint dp; |
| if (isCameraTest) |
| |
| |
| { |
| |
| |
| TPointD dstActualCenter = TPointD(finalLx / 2.0, finalLy / 2.0); |
| TPointD dstCenter = aff * image->getRaster()->getCenterD(); |
| |
| dp = convert(dstCenter - dstActualCenter); |
| |
| TRect r = convert(aff * convert(image->getRaster()->getBounds())); |
| aff = (TTranslation(convert(-r.getP00())) * aff); |
| |
| tmpRas = image->getRaster()->create(r.getLx(), r.getLy()); |
| |
| } else |
| tmpRas = image->getRaster()->create(finalLx, finalLy); |
| |
| TRop::resample(tmpRas, image->getRaster(), aff); |
| |
| |
| |
| TRasterImageP final(tmpRas); |
| final->setOffset(dp); |
| |
| final->setDpi(xDpi, yDpi); |
| |
| |
| return final; |
| } |
| |
| |
| |
| |
| |
| bool TCleanupper::doAutocenter( |
| |
| double &angle, double &skew, |
| double &cxin, double &cyin, |
| double &cqout, double &cpout, |
| |
| const double xdpi, const double ydpi, |
| const int raster_is_savebox, |
| const TRect saveBox, |
| const TRasterImageP &image, |
| const double scalex) |
| { |
| double sigma = 0, theta = 0; |
| FDG_INFO fdg_info = m_parameters->getFdgInfo(); |
| |
| switch (m_parameters->m_autocenterType) { |
| case AUTOCENTER_CTR: |
| angle = fdg_info.ctr_angle; |
| skew = fdg_info.ctr_skew; |
| cxin = mmToPixel(fdg_info.ctr_x, xdpi); |
| cyin = mmToPixel(fdg_info.ctr_y, ydpi); |
| if (raster_is_savebox) { |
| cxin -= saveBox.getP00().x; |
| cyin -= saveBox.getP00().y; |
| } |
| |
| break; |
| |
| case AUTOCENTER_FDG: { |
| |
| |
| int strip_width = compute_strip_pixel(&fdg_info, xdpi) + 1; |
| |
| switch (m_parameters->m_pegSide) { |
| case PEGS_BOTTOM: |
| sigma = 0.0; |
| break; |
| case PEGS_RIGHT: |
| sigma = 90.0; |
| break; |
| case PEGS_TOP: |
| sigma = 180.0; |
| break; |
| case PEGS_LEFT: |
| sigma = -90.0; |
| break; |
| default: |
| sigma = 0.0; |
| break; |
| } |
| |
| theta = sigma; |
| if (theta > 180.0) |
| theta -= 360.0; |
| else if (theta <= -180.0) |
| theta += 360.0; |
| |
| PEGS_SIDE pegs_ras_side; |
| if (theta == 0.0) |
| pegs_ras_side = PEGS_BOTTOM; |
| else if (theta == 90.0) |
| pegs_ras_side = PEGS_RIGHT; |
| else if (theta == 180.0) |
| pegs_ras_side = PEGS_TOP; |
| else if (theta == -90.0) |
| pegs_ras_side = PEGS_LEFT; |
| else |
| pegs_ras_side = PEGS_BOTTOM; |
| |
| switch (pegs_ras_side) { |
| case PEGS_LEFT: |
| case PEGS_RIGHT: |
| notMoreThan(image->getRaster()->getLx(), strip_width); |
| break; |
| default: |
| notMoreThan(image->getRaster()->getLy(), strip_width); |
| break; |
| } |
| |
| convert_dots_mm_to_pixel( |
| &fdg_info.dots[0], |
| fdg_info.dots.size(), |
| xdpi, ydpi); |
| |
| double cx, cy; |
| if (!get_image_rotation_and_center( |
| image->getRaster(), strip_width, pegs_ras_side, |
| &angle, &cx, &cy, &fdg_info.dots[0], fdg_info.dots.size())) { |
| return false; |
| } else { |
| angle *= M_180_PI; |
| cxin = cx; |
| cyin = cy; |
| double dist = (double)mmToPixel(fdg_info.dist_ctr_to_ctr_hole, xdpi * scalex); |
| switch (m_parameters->m_pegSide) { |
| case PEGS_BOTTOM: |
| cqout -= dist; |
| break; |
| case PEGS_TOP: |
| cqout += dist; |
| break; |
| case PEGS_LEFT: |
| cpout -= dist; |
| break; |
| case PEGS_RIGHT: |
| cpout += dist; |
| break; |
| default: |
| |
| return false; |
| } |
| } |
| fdg_info.dots.clear(); |
| |
| break; |
| } |
| |
| default: |
| return false; |
| } |
| |
| return true; |
| } |
| |
| |
| |
| |
| |
| inline void preprocessColor(const TPixel32 &pix, const TargetColorData &blackColor, |
| const std::vector<TargetColorData> &featureColors, int nFeatures, |
| TPixelCM32 &outpix) |
| { |
| |
| HSVColor pixHSV(HSVColor::fromRGB(pix.r / 255.0, pix.g / 255.0, pix.b / 255.0)); |
| |
| |
| |
| int idx = -1, tone = 255; |
| double hDist = (std::numeric_limits<double>::max)(), newHDist; |
| |
| for (int i = 0; i < nFeatures; ++i) { |
| const TargetColorData &fColor = featureColors[i]; |
| |
| |
| |
| |
| |
| newHDist = (pixHSV.m_h > fColor.m_hsv.m_h) ? std::min(pixHSV.m_h - fColor.m_hsv.m_h, fColor.m_hsv.m_h - pixHSV.m_h + 360.0) : std::min(fColor.m_hsv.m_h - pixHSV.m_h, pixHSV.m_h - fColor.m_hsv.m_h + 360.0); |
| if (newHDist < hDist) { |
| hDist = newHDist; |
| idx = i; |
| } |
| } |
| |
| if (idx >= 0) { |
| const TargetColorData &fColor = featureColors[idx]; |
| |
| |
| bool saturationOk = (pixHSV.m_s > fColor.m_saturationLower) && |
| ((fColor.m_hueLower <= fColor.m_hueUpper) ? (pixHSV.m_h >= fColor.m_hueLower) && (pixHSV.m_h <= fColor.m_hueUpper) : (pixHSV.m_h >= fColor.m_hueLower) || (pixHSV.m_h <= fColor.m_hueUpper)); |
| |
| if (saturationOk) { |
| tone = 255.0 * (1.0 - pixHSV.m_s) / (1.0 - fColor.m_saturationLower); |
| idx = fColor.m_idx; |
| } else |
| idx = -1; |
| } |
| |
| |
| if (pixHSV.m_v < blackColor.m_hsv.m_v) { |
| |
| tone = (tone * pixHSV.m_v / blackColor.m_hsv.m_v); |
| |
| |
| |
| if ((idx < 0) || (pixHSV.m_s * pixHSV.m_v) < blackColor.m_saturationLower) |
| |
| idx = 1; |
| } |
| |
| outpix = (idx > 0 && tone < 255) ? TPixelCM32(idx, 0, tone) : TPixelCM32(); |
| } |
| |
| |
| |
| void TCleanupper::preprocessColors( |
| const TRasterCM32P &outRas, |
| const TRaster32P &raster32, |
| const TargetColors &colors) |
| { |
| assert(outRas && outRas->getSize() == raster32->getSize()); |
| |
| |
| std::vector<TargetColorData> pencilsHSV; |
| for (int i = 2; i < colors.getColorCount(); ++i) { |
| TargetColorData cdata(colors.getColor(i)); |
| cdata.m_idx = i; |
| pencilsHSV.push_back(cdata); |
| } |
| |
| |
| TargetColor black = colors.getColor(1); |
| TargetColorData blackData(black); |
| blackData.m_hsv.m_v += (1.0 - black.m_threshold / 100.0); |
| blackData.m_saturationLower = sq(1.0 - black.m_hRange / 100.0); |
| |
| raster32->lock(); |
| outRas->lock(); |
| |
| |
| for (int j = 0; j < raster32->getLy(); j++) { |
| TPixel32 *pix = raster32->pixels(j); |
| TPixel32 *endPix = pix + raster32->getLx(); |
| TPixelCM32 *outPix = outRas->pixels(j); |
| |
| while (pix < endPix) { |
| if (*pix == TPixel32::White || pix->m < 255) |
| |
| |
| *outPix = TPixelCM32(); |
| else |
| preprocessColor(*pix, blackData, pencilsHSV, pencilsHSV.size(), *outPix); |
| |
| pix++; |
| outPix++; |
| } |
| } |
| |
| raster32->unlock(); |
| outRas->unlock(); |
| } |
| |
| |
| |
| |
| |
| void TCleanupper::finalize( |
| const TRaster32P &outRas, |
| CleanupPreprocessedImage *srcImg) |
| { |
| if (!outRas) |
| return; |
| |
| if (srcImg->m_wasFromGR8) |
| doPostProcessingGR8(outRas, srcImg); |
| else |
| doPostProcessingColor(outRas, srcImg); |
| } |
| |
| |
| |
| TToonzImageP TCleanupper::finalize(CleanupPreprocessedImage *src, bool isCleanupper) |
| { |
| if (src->m_wasFromGR8) |
| return doPostProcessingGR8(src); |
| else |
| return doPostProcessingColor(src->getImg(), isCleanupper); |
| } |
| |
| |
| |
| void TCleanupper::doPostProcessingGR8( |
| const TRaster32P &outRas, |
| CleanupPreprocessedImage *srcImg) |
| { |
| TToonzImageP image = srcImg->getImg(); |
| TRasterCM32P rasCM32 = image->getRaster(); |
| |
| rasCM32->lock(); |
| outRas->lock(); |
| |
| TRasterCM32P cmout(outRas->getLx(), outRas->getLy(), outRas->getWrap(), (TPixelCM32 *)outRas->getRawData()); |
| TRop::copy(cmout, rasCM32); |
| |
| rasCM32->unlock(); |
| |
| |
| brightnessContrastGR8(cmout, m_parameters->m_colors); |
| |
| |
| if (m_parameters->m_despeckling) |
| TRop::despeckle(cmout, m_parameters->m_despeckling, m_parameters->m_transparencyCheckEnabled); |
| |
| |
| if (m_parameters->m_postAntialias) { |
| TRasterCM32P newRas(cmout->getLx(), cmout->getLy()); |
| TRop::antialias(cmout, newRas, 10, m_parameters->m_aaValue); |
| |
| cmout->unlock(); |
| cmout = newRas; |
| cmout->lock(); |
| } |
| |
| |
| if (m_parameters->m_transparencyCheckEnabled) |
| transparencyCheck(cmout, outRas); |
| else |
| |
| TRop::convert(outRas, cmout, createToonzPaletteFromCleanupPalette()); |
| |
| outRas->unlock(); |
| } |
| |
| |
| |
| TToonzImageP TCleanupper::doPostProcessingGR8(const CleanupPreprocessedImage *img) |
| { |
| TToonzImageP image = img->getImg(); |
| |
| TRasterCM32P rasCM32 = image->getRaster(); |
| TRasterCM32P cmout(rasCM32->clone()); |
| |
| cmout->lock(); |
| |
| |
| brightnessContrastGR8(cmout, m_parameters->m_colors); |
| |
| |
| if (m_parameters->m_despeckling) |
| TRop::despeckle(cmout, m_parameters->m_despeckling, false); |
| |
| |
| if (m_parameters->m_postAntialias) { |
| TRasterCM32P newRas(cmout->getLx(), cmout->getLy()); |
| TRop::antialias(cmout, newRas, 10, m_parameters->m_aaValue); |
| |
| cmout->unlock(); |
| cmout = newRas; |
| cmout->lock(); |
| } |
| |
| cmout->unlock(); |
| |
| |
| TRect bbox; |
| TRop::computeBBox(cmout, bbox); |
| |
| |
| TToonzImageP outImg(cmout, bbox); |
| double dpix, dpiy; |
| image->getDpi(dpix, dpiy); |
| outImg->setDpi(dpix, dpiy); |
| |
| return outImg; |
| } |
| |
| |
| |
| void TCleanupper::doPostProcessingColor( |
| const TRaster32P &outRas, |
| CleanupPreprocessedImage *srcImg) |
| { |
| assert(srcImg); |
| assert(outRas->getSize() == srcImg->getSize()); |
| |
| TToonzImageP imgToProcess = srcImg->getImg(); |
| TRasterCM32P rasCM32 = imgToProcess->getRaster(); |
| |
| rasCM32->lock(); |
| outRas->lock(); |
| |
| TRasterCM32P cmout(outRas->getLx(), outRas->getLy(), outRas->getWrap(), (TPixelCM32 *)outRas->getRawData()); |
| TRop::copy(cmout, rasCM32); |
| |
| rasCM32->unlock(); |
| |
| |
| brightnessContrast(cmout, m_parameters->m_colors); |
| |
| |
| if (m_parameters->m_despeckling) |
| TRop::despeckle(cmout, m_parameters->m_despeckling, m_parameters->m_transparencyCheckEnabled); |
| |
| |
| if (m_parameters->m_postAntialias) { |
| TRasterCM32P newRas(cmout->getLx(), cmout->getLy()); |
| TRop::antialias(cmout, newRas, 10, m_parameters->m_aaValue); |
| |
| cmout->unlock(); |
| cmout = newRas; |
| cmout->lock(); |
| } |
| |
| |
| if (m_parameters->m_transparencyCheckEnabled) |
| transparencyCheck(cmout, outRas); |
| else |
| |
| TRop::convert(outRas, cmout, createToonzPaletteFromCleanupPalette()); |
| |
| outRas->unlock(); |
| } |
| |
| |
| |
| TToonzImageP TCleanupper::doPostProcessingColor(const TToonzImageP &imgToProcess, bool isCleanupper) |
| { |
| |
| TToonzImageP outImage; |
| if (isCleanupper) |
| outImage = imgToProcess; |
| else |
| outImage = TToonzImageP(imgToProcess->cloneImage()); |
| |
| assert(outImage); |
| assert(m_parameters->m_colors.getColorCount() < 9); |
| |
| |
| TRasterCM32P outRasCM32 = outImage->getRaster(); |
| outRasCM32->lock(); |
| |
| |
| brightnessContrast(outRasCM32, m_parameters->m_colors); |
| |
| |
| if (m_parameters->m_despeckling) |
| TRop::despeckle(outRasCM32, m_parameters->m_despeckling, false); |
| |
| |
| if (m_parameters->m_postAntialias) { |
| TRasterCM32P newRas(outRasCM32->getLx(), outRasCM32->getLy()); |
| TRop::antialias(outRasCM32, newRas, 10, m_parameters->m_aaValue); |
| |
| outRasCM32->unlock(); |
| outRasCM32 = newRas; |
| outImage->setCMapped(outRasCM32); |
| outRasCM32->lock(); |
| } |
| |
| TRect bbox; |
| TRop::computeBBox(outRasCM32, bbox); |
| outImage->setSavebox(bbox); |
| |
| outRasCM32->unlock(); |
| return outImage; |
| } |
| |