2010-11-19 16:55:55 +01:00
|
|
|
/* The MIT License
|
|
|
|
*
|
|
|
|
* Copyright (c) 2010 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.
|
|
|
|
*/
|
|
|
|
|
2010-11-21 22:48:58 +01:00
|
|
|
|
2011-04-17 21:14:24 +02:00
|
|
|
#include <prerequisites.h>
|
|
|
|
#include <graphics/graphics.h>
|
2010-11-18 23:13:35 +01:00
|
|
|
|
2011-04-02 23:02:39 +02:00
|
|
|
#include <GL/gl.h>
|
|
|
|
#include <GL/glu.h>
|
|
|
|
#include <GL/glext.h>
|
|
|
|
|
2010-11-18 23:13:35 +01:00
|
|
|
Graphics g_graphics;
|
|
|
|
|
|
|
|
void Graphics::init()
|
|
|
|
{
|
|
|
|
// setup opengl
|
2010-11-23 22:56:28 +01:00
|
|
|
glEnable(GL_ALPHA_TEST); // enable alpha by default
|
2010-11-18 23:13:35 +01:00
|
|
|
glAlphaFunc(GL_GREATER, 0.0f); // default alpha mode
|
|
|
|
glDisable(GL_DEPTH_TEST); // we are rendering 2D only, we don't need it
|
2010-11-23 22:56:28 +01:00
|
|
|
glEnable(GL_TEXTURE_2D); // enable textures by default
|
2011-02-08 23:48:26 +01:00
|
|
|
glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);
|
|
|
|
glShadeModel(GL_SMOOTH);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
2011-04-17 22:39:03 +02:00
|
|
|
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
2010-11-19 13:37:02 +01:00
|
|
|
|
2011-04-22 20:48:02 +02:00
|
|
|
flogInfo("GPU %s", (const char*)glGetString(GL_RENDERER));
|
|
|
|
flogInfo("OpenGL %s", (const char*)glGetString(GL_VERSION));
|
2010-11-18 23:13:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::terminate()
|
|
|
|
{
|
2011-05-01 20:47:35 +02:00
|
|
|
m_bindedTexture.reset();
|
2010-11-18 23:13:35 +01:00
|
|
|
}
|
|
|
|
|
2011-03-19 01:22:29 +01:00
|
|
|
bool Graphics::isExtensionSupported(const char *extension)
|
|
|
|
{
|
|
|
|
const GLubyte *extensions = NULL;
|
|
|
|
const GLubyte *start;
|
|
|
|
GLubyte *where, *terminator;
|
|
|
|
where = (GLubyte *)strchr(extension, ' ');
|
|
|
|
|
|
|
|
if(where || *extension == '\0')
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
extensions = glGetString(GL_EXTENSIONS);
|
|
|
|
|
|
|
|
start = extensions;
|
|
|
|
while(true) {
|
|
|
|
where = (GLubyte *) strstr((const char *)start, extension);
|
|
|
|
if(!where)
|
|
|
|
break;
|
|
|
|
|
|
|
|
terminator = where + strlen(extension);
|
|
|
|
|
|
|
|
if(where == start || *(where - 1) == ' ')
|
|
|
|
if(*terminator == ' ' || *terminator == '\0')
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
start = terminator;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-04-08 07:10:00 +02:00
|
|
|
void Graphics::resize(const Size& size)
|
2010-11-23 16:30:43 +01:00
|
|
|
{
|
2011-04-08 07:10:00 +02:00
|
|
|
m_screenSize = size;
|
2010-11-23 16:30:43 +01:00
|
|
|
restoreViewport();
|
|
|
|
}
|
|
|
|
|
2010-11-23 22:56:28 +01:00
|
|
|
void Graphics::restoreViewport()
|
2010-11-18 23:13:35 +01:00
|
|
|
{
|
2011-04-16 18:08:55 +02:00
|
|
|
disableDrawing();
|
|
|
|
|
2010-11-25 00:36:15 +01:00
|
|
|
const int& width = m_screenSize.width();
|
|
|
|
const int& height = m_screenSize.height();
|
|
|
|
|
2010-11-18 23:13:35 +01:00
|
|
|
// resize gl viewport
|
2010-11-25 00:36:15 +01:00
|
|
|
glViewport(0, 0, width, height);
|
2010-11-18 23:13:35 +01:00
|
|
|
|
|
|
|
/*
|
2010-11-23 22:56:28 +01:00
|
|
|
0,0---------0,w
|
2010-11-23 16:30:43 +01:00
|
|
|
| |
|
|
|
|
| |
|
|
|
|
| |
|
|
|
|
h,0---------h,w
|
|
|
|
*/
|
2010-11-18 23:13:35 +01:00
|
|
|
// setup view region like above
|
|
|
|
glMatrixMode(GL_PROJECTION);
|
|
|
|
glLoadIdentity();
|
2010-11-25 00:36:15 +01:00
|
|
|
gluOrtho2D(0.0f, width, height, 0.0f);
|
2010-11-18 23:13:35 +01:00
|
|
|
|
|
|
|
// back to model view
|
|
|
|
glMatrixMode(GL_MODELVIEW);
|
|
|
|
glLoadIdentity();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::beginRender()
|
|
|
|
{
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
glLoadIdentity();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::endRender()
|
|
|
|
{
|
2011-04-17 23:15:24 +02:00
|
|
|
disableDrawing();
|
2011-04-16 18:08:55 +02:00
|
|
|
}
|
2010-11-18 23:13:35 +01:00
|
|
|
|
2011-04-16 18:08:55 +02:00
|
|
|
void Graphics::disableDrawing()
|
|
|
|
{
|
|
|
|
glEnd();
|
|
|
|
m_drawMode = DRAW_NONE;
|
2010-11-18 23:13:35 +01:00
|
|
|
}
|
2010-11-25 00:36:15 +01:00
|
|
|
|
2011-04-10 22:40:44 +02:00
|
|
|
void Graphics::drawTexturedRect(const Rect& screenCoords, const TexturePtr& texture, const Rect& textureCoords, const Color& color)
|
2011-04-07 09:30:32 +02:00
|
|
|
{
|
2011-04-17 02:36:58 +02:00
|
|
|
if(screenCoords.isEmpty() || textureCoords.isEmpty())
|
2011-04-10 00:55:58 +02:00
|
|
|
return;
|
|
|
|
|
2010-11-25 00:36:15 +01:00
|
|
|
// rect correction for opengl
|
2011-04-05 02:39:57 +02:00
|
|
|
int right = screenCoords.right() + 1;
|
|
|
|
int bottom = screenCoords.bottom() + 1;
|
2010-11-25 00:36:15 +01:00
|
|
|
int top = screenCoords.top();
|
|
|
|
int left = screenCoords.left();
|
2011-04-10 22:40:44 +02:00
|
|
|
const Size& textureSize = texture->getSize();
|
2010-11-25 00:36:15 +01:00
|
|
|
|
2011-04-05 02:39:57 +02:00
|
|
|
float textureRight = 0.0f;
|
|
|
|
float textureBottom = 1.0f;
|
|
|
|
float textureTop = 0.0f;
|
|
|
|
float textureLeft = 1.0f;
|
2010-11-25 00:36:15 +01:00
|
|
|
|
2011-04-05 02:39:57 +02:00
|
|
|
if(!textureCoords.isEmpty()) {
|
|
|
|
textureRight = (float)(textureCoords.right() + 1) / textureSize.width();
|
|
|
|
textureBottom = (float)(textureCoords.bottom() + 1) / textureSize.height();
|
|
|
|
textureTop = (float)textureCoords.top() / textureSize.height();
|
|
|
|
textureLeft = (float)textureCoords.left() / textureSize.width();
|
2010-11-25 00:36:15 +01:00
|
|
|
}
|
2011-04-07 02:58:36 +02:00
|
|
|
|
2011-04-16 18:08:55 +02:00
|
|
|
bindTexture(texture, color);
|
2011-04-07 09:30:32 +02:00
|
|
|
glTexCoord2f(textureLeft, textureTop); glVertex2i(left, top);
|
|
|
|
glTexCoord2f(textureLeft, textureBottom); glVertex2i(left, bottom);
|
|
|
|
glTexCoord2f(textureRight, textureBottom); glVertex2i(right, bottom);
|
|
|
|
glTexCoord2f(textureRight, textureTop); glVertex2i(right, top);
|
2011-04-08 11:50:26 +02:00
|
|
|
}
|
|
|
|
|
2011-04-10 22:40:44 +02:00
|
|
|
void Graphics::drawRepeatedTexturedRect(const Rect& screenCoords, const TexturePtr& texture, const Rect& textureCoords, const Color& color)
|
2011-04-08 11:50:26 +02:00
|
|
|
{
|
2011-04-17 02:36:58 +02:00
|
|
|
if(screenCoords.isEmpty() || textureCoords.isEmpty())
|
2011-04-10 00:55:58 +02:00
|
|
|
return;
|
|
|
|
|
2011-04-08 11:50:26 +02:00
|
|
|
// render many repeated texture rects
|
|
|
|
Rect virtualScreenCoords(0,0,screenCoords.size());
|
|
|
|
for(int y = 0; y <= virtualScreenCoords.height(); y += textureCoords.height()) {
|
|
|
|
for(int x = 0; x <= virtualScreenCoords.width(); x += textureCoords.width()) {
|
|
|
|
Rect partialCoords(x, y, textureCoords.size());
|
|
|
|
Rect partialTextureCoords = textureCoords;
|
|
|
|
|
|
|
|
// partialCoords to screenCoords bottomRight
|
|
|
|
if(partialCoords.bottom() > virtualScreenCoords.bottom()) {
|
|
|
|
partialTextureCoords.setBottom(partialTextureCoords.bottom() + (virtualScreenCoords.bottom() - partialCoords.bottom()));
|
|
|
|
partialCoords.setBottom(virtualScreenCoords.bottom());
|
|
|
|
}
|
|
|
|
if(partialCoords.right() > virtualScreenCoords.right()) {
|
|
|
|
partialTextureCoords.setRight(partialTextureCoords.right() + (virtualScreenCoords.right() - partialCoords.right()));
|
|
|
|
partialCoords.setRight(virtualScreenCoords.right());
|
|
|
|
}
|
|
|
|
|
|
|
|
partialCoords.translate(screenCoords.topLeft());
|
2011-04-10 22:40:44 +02:00
|
|
|
drawTexturedRect(partialCoords, texture, partialTextureCoords, color);
|
2011-04-08 11:50:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-10 22:40:44 +02:00
|
|
|
void Graphics::drawFilledRect(const Rect& screenCoords, const Color& color)
|
2010-11-25 00:36:15 +01:00
|
|
|
{
|
2011-04-15 17:46:30 +02:00
|
|
|
if(screenCoords.isEmpty())
|
2011-04-10 00:55:58 +02:00
|
|
|
return;
|
|
|
|
|
2010-11-25 00:36:15 +01:00
|
|
|
// rect correction for opengl
|
2011-04-05 02:39:57 +02:00
|
|
|
int right = screenCoords.right() + 1;
|
|
|
|
int bottom = screenCoords.bottom() + 1;
|
2010-11-25 00:36:15 +01:00
|
|
|
int top = screenCoords.top();
|
|
|
|
int left = screenCoords.left();
|
|
|
|
|
2011-04-16 18:08:55 +02:00
|
|
|
bindColor(color);
|
2010-11-25 00:36:15 +01:00
|
|
|
glVertex2i(left, top);
|
|
|
|
glVertex2i(left, bottom);
|
|
|
|
glVertex2i(right, bottom);
|
|
|
|
glVertex2i(right, top);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-08 23:48:26 +01:00
|
|
|
void Graphics::drawBoundingRect(const Rect& screenCoords, const Color& color, int innerLineWidth)
|
2010-11-25 00:36:15 +01:00
|
|
|
{
|
2011-04-15 17:46:30 +02:00
|
|
|
if(2 * innerLineWidth > screenCoords.height() || screenCoords.isEmpty())
|
2010-11-25 00:36:15 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
// rect correction for opengl
|
|
|
|
int right = screenCoords.right()+1;
|
|
|
|
int bottom = screenCoords.bottom()+1;
|
|
|
|
int top = screenCoords.top();
|
|
|
|
int left = screenCoords.left();
|
|
|
|
|
2011-04-16 18:08:55 +02:00
|
|
|
bindColor(color);
|
|
|
|
|
|
|
|
// top line
|
|
|
|
glVertex2i(left, top);
|
|
|
|
glVertex2i(left, top + innerLineWidth);
|
|
|
|
glVertex2i(right, top + innerLineWidth);
|
|
|
|
glVertex2i(right, top);
|
|
|
|
|
|
|
|
// left
|
|
|
|
glVertex2i(left, screenCoords.top() + innerLineWidth);
|
|
|
|
glVertex2i(left, bottom - innerLineWidth);
|
|
|
|
glVertex2i(left + innerLineWidth, bottom - innerLineWidth);
|
|
|
|
glVertex2i(left + innerLineWidth, screenCoords.top() + innerLineWidth);
|
|
|
|
|
|
|
|
// bottom line
|
|
|
|
glVertex2i(left, bottom);
|
|
|
|
glVertex2i(left, bottom - innerLineWidth);
|
|
|
|
glVertex2i(right, bottom - innerLineWidth);
|
|
|
|
glVertex2i(right, bottom);
|
|
|
|
|
|
|
|
// right line
|
|
|
|
glVertex2i(right , top + innerLineWidth);
|
|
|
|
glVertex2i(right , bottom - innerLineWidth);
|
|
|
|
glVertex2i(right - innerLineWidth, bottom - innerLineWidth);
|
|
|
|
glVertex2i(right - innerLineWidth, top + innerLineWidth);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::bindColor(const Color& color)
|
|
|
|
{
|
|
|
|
// switch drawing to colored quads
|
|
|
|
if(m_drawMode != DRAW_COLOR_QUADS || m_bindedColor != color) {
|
|
|
|
if(m_drawMode != DRAW_NONE)
|
|
|
|
glEnd();
|
|
|
|
glDisable(GL_TEXTURE_2D);
|
|
|
|
if(m_bindedColor != color) {
|
|
|
|
glColor4ubv(color.rgbaPtr());
|
|
|
|
m_bindedColor = color;
|
|
|
|
}
|
|
|
|
glBegin(GL_QUADS);
|
|
|
|
m_drawMode = DRAW_COLOR_QUADS;
|
|
|
|
}
|
|
|
|
}
|
2010-11-25 00:36:15 +01:00
|
|
|
|
2011-04-16 18:08:55 +02:00
|
|
|
void Graphics::bindTexture(const TexturePtr& texture, const Color& color)
|
|
|
|
{
|
|
|
|
// switch drawing to textured quads
|
|
|
|
if(m_drawMode != DRAW_TEXTURE_QUADS || m_bindedTexture != texture) {
|
|
|
|
if(m_drawMode != DRAW_NONE)
|
|
|
|
glEnd();
|
|
|
|
glEnable(GL_TEXTURE_2D);
|
|
|
|
if(m_bindedTexture != texture) {
|
|
|
|
glBindTexture(GL_TEXTURE_2D, texture->getTextureId());
|
|
|
|
m_bindedTexture = texture;
|
|
|
|
}
|
|
|
|
if(m_bindedColor != color) {
|
|
|
|
glColor4ubv(color.rgbaPtr());
|
|
|
|
m_bindedColor = color;
|
|
|
|
}
|
|
|
|
glBegin(GL_QUADS);
|
|
|
|
m_drawMode = DRAW_TEXTURE_QUADS;
|
|
|
|
}
|
2011-04-08 11:50:26 +02:00
|
|
|
}
|