/*! \file GUITexture.h \brief */ #ifndef GUILIB_GUITEXTURE_H #define GUILIB_GUITEXTURE_H #pragma once /* * Copyright (C) 2005-2008 Team XBMC * http://www.xbmc.org * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with XBMC; see the file COPYING. If not, write to * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. * http://www.gnu.org/copyleft/gpl.html * */ #include "TextureManager.h" #include "Geometry.h" typedef uint32_t color_t; struct FRECT { float left; float top; float right; float bottom; }; // image alignment for keep, scale or center #define ASPECT_ALIGN_CENTER 0 #define ASPECT_ALIGN_LEFT 1 #define ASPECT_ALIGN_RIGHT 2 #define ASPECT_ALIGNY_CENTER 0 #define ASPECT_ALIGNY_TOP 4 #define ASPECT_ALIGNY_BOTTOM 8 #define ASPECT_ALIGN_MASK 3 #define ASPECT_ALIGNY_MASK ~3 class CAspectRatio { public: enum ASPECT_RATIO { AR_STRETCH = 0, AR_SCALE, AR_KEEP, AR_CENTER }; CAspectRatio(ASPECT_RATIO aspect = AR_STRETCH) { ratio = aspect; align = ASPECT_ALIGN_CENTER | ASPECT_ALIGNY_CENTER; scaleDiffuse = true; }; bool operator!=(const CAspectRatio &right) const { if (ratio != right.ratio) return true; if (align != right.align) return true; if (scaleDiffuse != right.scaleDiffuse) return true; return false; }; ASPECT_RATIO ratio; uint32_t align; bool scaleDiffuse; }; class CTextureInfo { public: CTextureInfo() { memset(&border, 0, sizeof(FRECT)); orientation = 0; useLarge = false; }; CTextureInfo(const CStdString &file) { memset(&border, 0, sizeof(FRECT)); orientation = 0; useLarge = false; filename = file; } void operator=(const CTextureInfo &right) { memcpy(&border, &right.border, sizeof(FRECT)); orientation = right.orientation; diffuse = right.diffuse; filename = right.filename; useLarge = right.useLarge; }; bool useLarge; FRECT border; // scaled - unneeded if we get rid of scale on load int orientation; // orientation of the texture (0 - 7 == EXIForientation - 1) CStdString diffuse; // diffuse overlay texture CStdString filename; // main texture file }; class CGUITextureBase { public: CGUITextureBase(float posX, float posY, float width, float height, const CTextureInfo& texture); CGUITextureBase(const CGUITextureBase &left); virtual ~CGUITextureBase(void); void Render(); void DynamicResourceAlloc(bool bOnOff); void AllocResources(); void FreeResources(bool immediately = false); void SetVisible(bool visible); void SetAlpha(unsigned char alpha); void SetDiffuseColor(color_t color); void SetPosition(float x, float y); void SetWidth(float width); void SetHeight(float height); void SetFileName(const CStdString &filename); void SetAspectRatio(const CAspectRatio &aspect); const CStdString& GetFileName() const { return m_info.filename; }; float GetTextureWidth() const { return m_frameWidth; }; float GetTextureHeight() const { return m_frameHeight; }; float GetWidth() const { return m_width; }; float GetHeight() const { return m_height; }; float GetXPosition() const { return m_posX; }; float GetYPosition() const { return m_posY; }; int GetOrientation() const; const CRect &GetRenderRect() const { return m_vertex; }; bool IsLazyLoaded() const { return m_info.useLarge; }; bool HitTest(const CPoint &point) const { return CRect(m_posX, m_posY, m_posX + m_width, m_posY + m_height).PtInRect(point); }; bool IsAllocated() const { return m_isAllocated != NO; }; bool FailedToAlloc() const { return m_isAllocated == NORMAL_FAILED || m_isAllocated == LARGE_FAILED; }; bool ReadyToRender() const; protected: void CalculateSize(); void LoadDiffuseImage(); void AllocateOnDemand(); void UpdateAnimFrame(); void Render(float left, float top, float bottom, float right, float u1, float v1, float u2, float v2, float u3, float v3); void OrientateTexture(CRect &rect, float width, float height, int orientation); // functions that our implementation classes handle virtual void Allocate() {}; ///< called after our textures have been allocated virtual void Free() {}; ///< called after our textures have been freed virtual void Begin() {}; virtual void Draw(float *x, float *y, float *z, const CRect &texture, const CRect &diffuse, color_t color, int orientation)=0; virtual void End() {}; bool m_visible; color_t m_diffuseColor; float m_posX; // size of the frame float m_posY; float m_width; float m_height; CRect m_vertex; // vertex coords to render bool m_invalid; // if true, we need to recalculate unsigned char m_alpha; float m_frameWidth, m_frameHeight; // size in pixels of the actual frame within the texture float m_texCoordsScaleU, m_texCoordsScaleV; // scale factor for pixel->texture coordinates // animations int m_currentLoop; unsigned int m_currentFrame; DWORD m_frameCounter; float m_diffuseU, m_diffuseV; // size of the diffuse frame (in tex coords) float m_diffuseScaleU, m_diffuseScaleV; // scale factor of the diffuse frame (from texture coords to diffuse tex coords) CPoint m_diffuseOffset; // offset into the diffuse frame (it's not always the origin) bool m_allocateDynamically; enum ALLOCATE_TYPE { NO = 0, NORMAL, LARGE, NORMAL_FAILED, LARGE_FAILED }; ALLOCATE_TYPE m_isAllocated; CTextureInfo m_info; CAspectRatio m_aspect; int m_largeOrientation; // orientation for large textures CTextureArray m_diffuse; CTextureArray m_texture; }; #if defined(HAS_SDL_2D) #include "GUITextureSDL.h" #elif defined(HAS_GL) #include "GUITextureGL.h" #define CGUITexture CGUITextureGL #elif defined(HAS_GLES) #include "GUITextureGLES.h" #define CGUITexture CGUITextureGLES #elif defined(HAS_DX) #include "GUITextureD3D.h" #define CGUITexture CGUITextureD3D #endif #endif