/* * Copyright (C) 2004, 2005, 2006, 2007 Nikolas Zimmermann * Copyright (C) 2004, 2005 Rob Buis * Copyright (C) 2005 Eric Seidel * Copyright (C) 2009 Dirk Schulze * Copyright (C) Research In Motion Limited 2010. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include "config.h" #if ENABLE(FILTERS) #include "FEMorphology.h" #include "Filter.h" #include "RenderTreeAsText.h" #include "TextStream.h" #include #include using std::min; using std::max; namespace WebCore { FEMorphology::FEMorphology(Filter* filter, MorphologyOperatorType type, float radiusX, float radiusY) : FilterEffect(filter) , m_type(type) , m_radiusX(radiusX) , m_radiusY(radiusY) { } PassRefPtr FEMorphology::create(Filter* filter, MorphologyOperatorType type, float radiusX, float radiusY) { return adoptRef(new FEMorphology(filter, type, radiusX, radiusY)); } MorphologyOperatorType FEMorphology::morphologyOperator() const { return m_type; } bool FEMorphology::setMorphologyOperator(MorphologyOperatorType type) { if (m_type == type) return false; m_type = type; return true; } float FEMorphology::radiusX() const { return m_radiusX; } bool FEMorphology::setRadiusX(float radiusX) { if (m_radiusX == radiusX) return false; m_radiusX = radiusX; return true; } float FEMorphology::radiusY() const { return m_radiusY; } void FEMorphology::determineAbsolutePaintRect() { FloatRect paintRect = inputEffect(0)->absolutePaintRect(); Filter* filter = this->filter(); paintRect.inflateX(filter->applyHorizontalScale(m_radiusX)); paintRect.inflateY(filter->applyVerticalScale(m_radiusY)); paintRect.intersect(maxEffectRect()); setAbsolutePaintRect(enclosingIntRect(paintRect)); } bool FEMorphology::setRadiusY(float radiusY) { if (m_radiusY == radiusY) return false; m_radiusY = radiusY; return true; } void FEMorphology::apply() { if (hasResult()) return; FilterEffect* in = inputEffect(0); in->apply(); if (!in->hasResult()) return; ByteArray* dstPixelArray = createPremultipliedImageResult(); if (!dstPixelArray) return; setIsAlphaImage(in->isAlphaImage()); if (m_radiusX <= 0 || m_radiusY <= 0) return; Filter* filter = this->filter(); int radiusX = static_cast(floorf(filter->applyHorizontalScale(m_radiusX))); int radiusY = static_cast(floorf(filter->applyVerticalScale(m_radiusY))); IntRect effectDrawingRect = requestedRegionOfInputImageData(in->absolutePaintRect()); RefPtr srcPixelArray = in->asPremultipliedImage(effectDrawingRect); int effectWidth = effectDrawingRect.width() * 4; // Limit the radius size to effect dimensions radiusX = min(effectDrawingRect.width() - 1, radiusX); radiusY = min(effectDrawingRect.height() - 1, radiusY); Vector extrema; for (int y = 0; y < effectDrawingRect.height(); ++y) { int startY = max(0, y - radiusY); int endY = min(effectDrawingRect.height() - 1, y + radiusY); for (unsigned channel = 0; channel < 4; ++channel) { // Fill the kernel extrema.clear(); for (int j = 0; j <= radiusX; ++j) { unsigned char columnExtrema = srcPixelArray->get(startY * effectWidth + 4 * j + channel); for (int i = startY; i <= endY; ++i) { unsigned char pixel = srcPixelArray->get(i * effectWidth + 4 * j + channel); if ((m_type == FEMORPHOLOGY_OPERATOR_ERODE && pixel <= columnExtrema) || (m_type == FEMORPHOLOGY_OPERATOR_DILATE && pixel >= columnExtrema)) columnExtrema = pixel; } extrema.append(columnExtrema); } // Kernel is filled, get extrema of next column for (int x = 0; x < effectDrawingRect.width(); ++x) { unsigned endX = min(x + radiusX, effectDrawingRect.width() - 1); unsigned char columnExtrema = srcPixelArray->get(startY * effectWidth + endX * 4 + channel); for (int i = startY; i <= endY; ++i) { unsigned char pixel = srcPixelArray->get(i * effectWidth + endX * 4 + channel); if ((m_type == FEMORPHOLOGY_OPERATOR_ERODE && pixel <= columnExtrema) || (m_type == FEMORPHOLOGY_OPERATOR_DILATE && pixel >= columnExtrema)) columnExtrema = pixel; } if (x - radiusX >= 0) extrema.remove(0); if (x + radiusX <= effectDrawingRect.width()) extrema.append(columnExtrema); unsigned char entireExtrema = extrema[0]; for (unsigned kernelIndex = 0; kernelIndex < extrema.size(); ++kernelIndex) { if ((m_type == FEMORPHOLOGY_OPERATOR_ERODE && extrema[kernelIndex] <= entireExtrema) || (m_type == FEMORPHOLOGY_OPERATOR_DILATE && extrema[kernelIndex] >= entireExtrema)) entireExtrema = extrema[kernelIndex]; } dstPixelArray->set(y * effectWidth + 4 * x + channel, entireExtrema); } } } } void FEMorphology::dump() { } static TextStream& operator<<(TextStream& ts, const MorphologyOperatorType& type) { switch (type) { case FEMORPHOLOGY_OPERATOR_UNKNOWN: ts << "UNKNOWN"; break; case FEMORPHOLOGY_OPERATOR_ERODE: ts << "ERODE"; break; case FEMORPHOLOGY_OPERATOR_DILATE: ts << "DILATE"; break; } return ts; } TextStream& FEMorphology::externalRepresentation(TextStream& ts, int indent) const { writeIndent(ts, indent); ts << "[feMorphology"; FilterEffect::externalRepresentation(ts); ts << " operator=\"" << morphologyOperator() << "\" " << "radius=\"" << radiusX() << ", " << radiusY() << "\"]\n"; inputEffect(0)->externalRepresentation(ts, indent + 1); return ts; } } // namespace WebCore #endif // ENABLE(FILTERS)