blob: 235f1973b523464b35303a417b0c7fc3428d95b5 [file] [log] [blame]
/*
* Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies)
* Copyright (C) 2012 Adobe Systems Incorporated
* Copyright (C) 2012, 2016 Igalia S.L.
*
* 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"
#include "ClipStack.h"
#include "TextureMapperGLHeaders.h"
namespace WebCore {
void ClipStack::push()
{
clipStack.append(clipState);
clipStateDirty = true;
}
void ClipStack::pop()
{
if (clipStack.isEmpty())
return;
clipState = clipStack.last();
clipStack.removeLast();
clipStateDirty = true;
}
void ClipStack::reset(const IntRect& rect, ClipStack::YAxisMode mode)
{
clipStack.clear();
size = rect.size();
yAxisMode = mode;
clipState = State(rect);
clipStateDirty = true;
}
void ClipStack::intersect(const IntRect& rect)
{
clipState.scissorBox.intersect(rect);
clipStateDirty = true;
}
void ClipStack::setStencilIndex(int stencilIndex)
{
clipState.stencilIndex = stencilIndex;
clipStateDirty = true;
}
void ClipStack::addRoundedRect(const FloatRoundedRect& roundedRect, const TransformationMatrix& matrix)
{
if (clipState.roundedRectCount >= s_roundedRectMaxClips)
return;
// Ensure that the vectors holding the components have the required size.
m_roundedRectComponents.grow(s_roundedRectComponentsArraySize);
m_roundedRectInverseTransformComponents.grow(s_roundedRectInverseTransformComponentsArraySize);
// Copy the RoundedRect components to the appropriate position in the array.
int basePosition = clipState.roundedRectCount * s_roundedRectComponentsPerRect;
m_roundedRectComponents[basePosition] = roundedRect.rect().x();
m_roundedRectComponents[basePosition + 1] = roundedRect.rect().y();
m_roundedRectComponents[basePosition + 2] = roundedRect.rect().width();
m_roundedRectComponents[basePosition + 3] = roundedRect.rect().height();
m_roundedRectComponents[basePosition + 4] = roundedRect.radii().topLeft().width();
m_roundedRectComponents[basePosition + 5] = roundedRect.radii().topLeft().height();
m_roundedRectComponents[basePosition + 6] = roundedRect.radii().topRight().width();
m_roundedRectComponents[basePosition + 7] = roundedRect.radii().topRight().height();
m_roundedRectComponents[basePosition + 8] = roundedRect.radii().bottomLeft().width();
m_roundedRectComponents[basePosition + 9] = roundedRect.radii().bottomLeft().height();
m_roundedRectComponents[basePosition + 10] = roundedRect.radii().bottomRight().width();
m_roundedRectComponents[basePosition + 11] = roundedRect.radii().bottomRight().height();
// Copy the TransformationMatrix components to the appropriate position in the array.
basePosition = clipState.roundedRectCount * s_roundedRectInverseTransformComponentsPerRect;
memcpy(m_roundedRectInverseTransformComponents.data() + basePosition, matrix.toColumnMajorFloatArray().data(), s_roundedRectInverseTransformComponentsPerRect * sizeof(float));
clipState.roundedRectCount++;
}
void ClipStack::apply()
{
if (clipState.scissorBox.isEmpty())
return;
glScissor(clipState.scissorBox.x(),
(yAxisMode == YAxisMode::Inverted) ? size.height() - clipState.scissorBox.maxY() : clipState.scissorBox.y(),
clipState.scissorBox.width(), clipState.scissorBox.height());
glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP);
glStencilFunc(GL_EQUAL, clipState.stencilIndex - 1, clipState.stencilIndex - 1);
if (clipState.stencilIndex == 1)
glDisable(GL_STENCIL_TEST);
else
glEnable(GL_STENCIL_TEST);
}
void ClipStack::applyIfNeeded()
{
if (!clipStateDirty)
return;
clipStateDirty = false;
apply();
}
} // namespace WebCore