2011-12-07 01:31:55 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2010-2011 OTClient <https://github.com/edubart/otclient>
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "painter.h"
|
|
|
|
#include "texture.h"
|
|
|
|
#include "paintershadersources.h"
|
|
|
|
#include "paintershaderprogram.h"
|
|
|
|
#include "shaderprogram.h"
|
|
|
|
#include "graphics.h"
|
2011-12-07 19:49:20 +01:00
|
|
|
#include "vertexarray.h"
|
2011-12-07 01:31:55 +01:00
|
|
|
|
|
|
|
Painter g_painter;
|
|
|
|
|
|
|
|
void Painter::init()
|
|
|
|
{
|
|
|
|
setColor(Fw::white);
|
|
|
|
setOpacity(255);
|
|
|
|
setCompositionMode(CompositionMode_SourceOver);
|
|
|
|
|
|
|
|
PainterShaderProgramPtr program = PainterShaderProgramPtr(new PainterShaderProgram);
|
2011-12-07 02:48:39 +01:00
|
|
|
program->addShaderFromSourceCode(Shader::Vertex, glslMainWithTexCoordsVertexShader + glslPositionOnlyVertexShader);
|
|
|
|
program->addShaderFromSourceCode(Shader::Fragment, glslMainFragmentShader + glslTextureSrcFragmentShader);
|
2011-12-07 01:31:55 +01:00
|
|
|
program->bindAttributeLocation(VERTEX_COORDS_ATTR, "vertexCoord");
|
|
|
|
program->bindAttributeLocation(TEXTURE_COORDS_ATTR, "textureCoord");
|
|
|
|
assert(program->link());
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::PROJECTION_MATRIX_UNIFORM, "projectionMatrix");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::TEXTURE_TRANSFORM_MATRIX_UNIFORM, "textureTransformMatrix");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::COLOR_UNIFORM, "color");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::OPACITY_UNIFORM, "opacity");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::TEXTURE_UNIFORM, "texture");
|
|
|
|
m_drawTexturedProgram = program;
|
|
|
|
|
|
|
|
program = PainterShaderProgramPtr(new PainterShaderProgram);
|
2011-12-07 02:48:39 +01:00
|
|
|
program->addShaderFromSourceCode(Shader::Vertex, glslMainVertexShader + glslPositionOnlyVertexShader);
|
|
|
|
program->addShaderFromSourceCode(Shader::Fragment, glslMainFragmentShader + glslSolidColorFragmentShader);
|
2011-12-07 01:31:55 +01:00
|
|
|
program->bindAttributeLocation(VERTEX_COORDS_ATTR, "vertexCoord");
|
|
|
|
assert(program->link());
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::PROJECTION_MATRIX_UNIFORM, "projectionMatrix");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::COLOR_UNIFORM, "color");
|
|
|
|
program->bindUniformLocation(PainterShaderProgram::OPACITY_UNIFORM, "opacity");
|
|
|
|
m_drawSolidColorProgram = program;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::terminate()
|
|
|
|
{
|
|
|
|
m_drawTexturedProgram.reset();
|
|
|
|
m_drawSolidColorProgram.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::updateProjectionMatrix(const Size& viewportSize, bool inverseYAxis)
|
|
|
|
{
|
|
|
|
// The projection matrix converts from Painter's coordinate system to GL's coordinate system
|
|
|
|
// * GL's viewport is 2x2, Painter's is width x height
|
|
|
|
// * GL has +y -> -y going from bottom -> top, Painter is the other way round
|
|
|
|
// * GL has [0,0] in the center, Painter has it in the top-left
|
|
|
|
//
|
|
|
|
// This results in the Projection matrix below, which is multiplied by the painter's
|
|
|
|
// transformation matrix, as shown below:
|
|
|
|
//
|
|
|
|
// Projection Matrix Painter Coord GL Coord
|
|
|
|
// ------------------------------------------------ --------- ---------
|
|
|
|
// | 2.0 / width | 0.0 | -1.0 | | x | | y' |
|
|
|
|
// | 0.0 | -2.0 / height | 1.0 | * | y | = | x' |
|
|
|
|
// | 0.0 | 0.0 | 0.0 | | 1 | | 0 |
|
|
|
|
// ------------------------------------------------ --------- ---------
|
|
|
|
float w = viewportSize.width();
|
|
|
|
float h = viewportSize.height();
|
|
|
|
if(inverseYAxis) {
|
|
|
|
m_projectionMatrix[0][0] = 2.0f/w; m_projectionMatrix[0][1] = 0.0f; m_projectionMatrix[0][2] =-1.0f;
|
|
|
|
m_projectionMatrix[1][0] = 0.0f; m_projectionMatrix[1][1] = 2.0f/h; m_projectionMatrix[1][2] =-1.0f;
|
|
|
|
m_projectionMatrix[2][0] = 0.0f; m_projectionMatrix[2][1] = 0.0f; m_projectionMatrix[2][2] = 0.0f;
|
|
|
|
} else {
|
|
|
|
m_projectionMatrix[0][0] = 2.0f/w; m_projectionMatrix[0][1] = 0.0f; m_projectionMatrix[0][2] =-1.0f;
|
|
|
|
m_projectionMatrix[1][0] = 0.0f; m_projectionMatrix[1][1] =-2.0f/h; m_projectionMatrix[1][2] = 1.0f;
|
|
|
|
m_projectionMatrix[2][0] = 0.0f; m_projectionMatrix[2][1] = 0.0f; m_projectionMatrix[2][2] = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
void Painter::drawCoords(CoordsBuffer& coordsBuffer)
|
2011-12-07 01:31:55 +01:00
|
|
|
{
|
2011-12-07 19:49:20 +01:00
|
|
|
coordsBuffer.cacheVertexArrays();
|
2011-12-07 01:31:55 +01:00
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
if(coordsBuffer.getVertexCount() < 3)
|
2011-12-07 01:31:55 +01:00
|
|
|
return;
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
m_drawSolidColorProgram->prepareForDraw();
|
|
|
|
m_drawSolidColorProgram->setProjectionMatrix(m_projectionMatrix);
|
|
|
|
m_drawSolidColorProgram->setOpacity(m_currentOpacity);
|
|
|
|
m_drawSolidColorProgram->setColor(m_currentColor);
|
|
|
|
m_drawSolidColorProgram->setVertexCoords(coordsBuffer.getVertexCoords());
|
|
|
|
m_drawSolidColorProgram->drawTriangles(coordsBuffer.getVertexCount());
|
|
|
|
m_drawSolidColorProgram->releaseFromDraw();
|
|
|
|
}
|
2011-12-07 01:31:55 +01:00
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
void Painter::drawTextureCoords(CoordsBuffer& coordsBuffer, const TexturePtr& texture)
|
|
|
|
{
|
|
|
|
coordsBuffer.cacheVertexArrays();
|
|
|
|
|
|
|
|
if(coordsBuffer.getVertexCount() < 3)
|
|
|
|
return;
|
2011-12-07 01:31:55 +01:00
|
|
|
|
|
|
|
m_drawTexturedProgram->prepareForDraw();
|
|
|
|
m_drawTexturedProgram->setProjectionMatrix(m_projectionMatrix);
|
|
|
|
m_drawTexturedProgram->setOpacity(m_currentOpacity);
|
|
|
|
m_drawTexturedProgram->setColor(m_currentColor);
|
|
|
|
m_drawTexturedProgram->setTexture(texture);
|
2011-12-07 19:49:20 +01:00
|
|
|
m_drawTexturedProgram->setVertexCoords(coordsBuffer.getVertexCoords());
|
|
|
|
m_drawTexturedProgram->setTextureCoords(coordsBuffer.getTextureCoords());
|
|
|
|
m_drawTexturedProgram->drawTriangles(coordsBuffer.getVertexCount());
|
2011-12-07 01:31:55 +01:00
|
|
|
m_drawTexturedProgram->releaseFromDraw();
|
|
|
|
}
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
void Painter::drawTexturedRect(const Rect& dest, const TexturePtr& texture)
|
|
|
|
{
|
|
|
|
drawTexturedRect(dest, texture, Rect(Point(0,0), texture->getSize()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::drawTexturedRect(const Rect& dest, const TexturePtr& texture, const Rect& src)
|
|
|
|
{
|
|
|
|
if(dest.isEmpty() || src.isEmpty() || !texture->getId())
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_coordsBuffer.clear();
|
|
|
|
m_coordsBuffer.addRect(dest, src);
|
|
|
|
drawTextureCoords(m_coordsBuffer, texture);
|
|
|
|
}
|
|
|
|
|
2011-12-07 01:31:55 +01:00
|
|
|
void Painter::drawRepeatedTexturedRect(const Rect& dest, const TexturePtr& texture, const Rect& src)
|
|
|
|
{
|
|
|
|
if(dest.isEmpty() || src.isEmpty() || !texture->getId())
|
|
|
|
return;
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
m_coordsBuffer.clear();
|
|
|
|
m_coordsBuffer.addRepeatedRects(dest, src);
|
|
|
|
drawTextureCoords(m_coordsBuffer, texture);
|
2011-12-07 01:31:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::drawFilledRect(const Rect& dest)
|
|
|
|
{
|
|
|
|
if(dest.isEmpty())
|
|
|
|
return;
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
m_coordsBuffer.clear();
|
|
|
|
m_coordsBuffer.addRect(dest);
|
|
|
|
drawCoords(m_coordsBuffer);
|
2011-12-07 01:31:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::drawBoundingRect(const Rect& dest, int innerLineWidth)
|
|
|
|
{
|
|
|
|
if(dest.isEmpty() || innerLineWidth == 0)
|
|
|
|
return;
|
|
|
|
|
2011-12-07 19:49:20 +01:00
|
|
|
m_coordsBuffer.clear();
|
|
|
|
m_coordsBuffer.addBoudingRect(dest, innerLineWidth);
|
|
|
|
drawCoords(m_coordsBuffer);
|
2011-12-07 01:31:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Painter::setCompositionMode(Painter::CompositionMode compositionMode)
|
|
|
|
{
|
|
|
|
switch(compositionMode) {
|
|
|
|
case CompositionMode_SourceOver:
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
break;
|
|
|
|
case CompositionMode_ColorizeDest:
|
|
|
|
glBlendFunc(GL_DST_COLOR, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|