blender/source/gameengine/BlenderRoutines/KX_BlenderCanvas.cpp

366 lines
7.7 KiB
C++
Raw Normal View History

/*
* ***** BEGIN GPL LICENSE BLOCK *****
2002-10-12 11:37:38 +00:00
*
* 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
* of the License, or (at your option) any later version.
2002-10-12 11:37:38 +00:00
*
* 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 this program; if not, write to the Free Software Foundation,
2010-02-12 13:34:04 +00:00
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
2002-10-12 11:37:38 +00:00
*
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
* All rights reserved.
*
* The Original Code is: all of this file.
*
* Contributor(s): none yet.
*
* ***** END GPL LICENSE BLOCK *****
2002-10-12 11:37:38 +00:00
*/
/** \file gameengine/BlenderRoutines/KX_BlenderCanvas.cpp
* \ingroup blroutines
*/
#include <GL/glew.h>
#include "MEM_guardedalloc.h"
2002-10-12 11:37:38 +00:00
#include "KX_BlenderCanvas.h"
#include "DNA_image_types.h"
#include "DNA_scene_types.h"
#include "DNA_screen_types.h"
#include "DNA_windowmanager_types.h"
#include "BKE_image.h"
#include "BKE_global.h"
#include "BKE_main.h"
#include "BLI_path_util.h"
#include "BLI_string.h"
#include <assert.h>
2002-10-12 11:37:38 +00:00
extern "C" {
#include "IMB_imbuf.h"
#include "IMB_imbuf_types.h"
#include "WM_api.h"
#include "wm_cursors.h"
#include "wm_window.h"
}
KX_BlenderCanvas::KX_BlenderCanvas(wmWindowManager *wm, wmWindow *win, RAS_Rect &rect, struct ARegion *ar) :
m_wm(wm),
m_win(win),
m_frame_rect(rect)
2002-10-12 11:37:38 +00:00
{
// initialize area so that it's available for game logic on frame 1 (ImageViewport)
m_area_rect = rect;
// area boundaries needed for mouse coordinates in Letterbox framing mode
m_area_left = ar->winrct.xmin;
m_area_top = ar->winrct.ymax;
2012-10-20 18:46:57 +00:00
glGetIntegerv(GL_VIEWPORT, (GLint *)m_viewport);
2002-10-12 11:37:38 +00:00
}
KX_BlenderCanvas::~KX_BlenderCanvas()
{
}
void KX_BlenderCanvas::Init()
{
glDepthFunc(GL_LEQUAL);
2012-09-16 04:58:18 +00:00
}
2002-10-12 11:37:38 +00:00
void KX_BlenderCanvas::SwapBuffers()
{
wm_window_swap_buffers(m_win);
2002-10-12 11:37:38 +00:00
}
void KX_BlenderCanvas::SetSwapInterval(int interval)
{
wm_window_set_swap_interval(m_win, interval);
}
int KX_BlenderCanvas::GetSwapInterval()
{
return wm_window_get_swap_interval(m_win);
}
void KX_BlenderCanvas::ResizeWindow(int width, int height)
{
// Not implemented for the embedded player
}
void KX_BlenderCanvas::SetFullScreen(bool enable)
{
// Not implemented for the embedded player
}
bool KX_BlenderCanvas::GetFullScreen()
{
// Not implemented for the embedded player
return false;
}
bool KX_BlenderCanvas::BeginDraw()
{
// in case of multi-window we need to ensure we are drawing to the correct
// window always, because it may change in window event handling
wm_window_make_drawable(m_wm, m_win);
return true;
}
void KX_BlenderCanvas::EndDraw()
{
// nothing needs to be done here
}
2002-10-12 11:37:38 +00:00
void KX_BlenderCanvas::BeginFrame()
{
glEnable(GL_DEPTH_TEST);
glDepthFunc(GL_LEQUAL);
}
void KX_BlenderCanvas::EndFrame()
{
glDisable(GL_FOG);
}
void KX_BlenderCanvas::ClearColor(float r,float g,float b,float a)
{
glClearColor(r,g,b,a);
}
void KX_BlenderCanvas::ClearBuffer(int type)
{
int ogltype = 0;
if (type & RAS_ICanvas::COLOR_BUFFER )
ogltype |= GL_COLOR_BUFFER_BIT;
if (type & RAS_ICanvas::DEPTH_BUFFER )
ogltype |= GL_DEPTH_BUFFER_BIT;
glClear(ogltype);
}
int KX_BlenderCanvas::GetWidth(
) const {
return m_frame_rect.GetWidth();
2002-10-12 11:37:38 +00:00
}
int KX_BlenderCanvas::GetHeight(
) const {
return m_frame_rect.GetHeight();
2002-10-12 11:37:38 +00:00
}
int KX_BlenderCanvas::GetMouseX(int x)
{
2012-10-14 08:49:01 +00:00
int left = GetWindowArea().GetLeft();
return x - (left - m_area_left);
}
int KX_BlenderCanvas::GetMouseY(int y)
{
2012-10-14 08:49:01 +00:00
int top = GetWindowArea().GetTop();
return y - (m_area_top - top);
}
float KX_BlenderCanvas::GetMouseNormalizedX(int x)
{
int can_x = GetMouseX(x);
return float(can_x)/this->GetWidth();
}
float KX_BlenderCanvas::GetMouseNormalizedY(int y)
{
int can_y = GetMouseY(y);
return float(can_y)/this->GetHeight();
}
RAS_Rect &
KX_BlenderCanvas::
GetWindowArea(
2012-03-28 05:03:24 +00:00
) {
return m_area_rect;
2012-09-16 04:58:18 +00:00
}
2002-10-12 11:37:38 +00:00
void
KX_BlenderCanvas::
SetViewPort(
int x1, int y1,
int x2, int y2
2012-03-28 05:03:24 +00:00
) {
2013-02-02 04:48:21 +00:00
/* x1 and y1 are the min pixel coordinate (e.g. 0)
* x2 and y2 are the max pixel coordinate
* the width,height is calculated including both pixels
* therefore: max - min + 1
*/
2002-10-12 11:37:38 +00:00
int vp_width = (x2 - x1) + 1;
int vp_height = (y2 - y1) + 1;
int minx = m_frame_rect.GetLeft();
int miny = m_frame_rect.GetBottom();
2002-10-12 11:37:38 +00:00
m_area_rect.SetLeft(minx + x1);
m_area_rect.SetBottom(miny + y1);
m_area_rect.SetRight(minx + x2);
m_area_rect.SetTop(miny + y2);
m_viewport[0] = minx+x1;
m_viewport[1] = miny+y1;
m_viewport[2] = vp_width;
m_viewport[3] = vp_height;
2002-10-12 11:37:38 +00:00
glViewport(minx + x1, miny + y1, vp_width, vp_height);
glScissor(minx + x1, miny + y1, vp_width, vp_height);
}
void
KX_BlenderCanvas::
UpdateViewPort(
int x1, int y1,
int x2, int y2
) {
m_viewport[0] = x1;
m_viewport[1] = y1;
m_viewport[2] = x2;
m_viewport[3] = y2;
}
const int*
KX_BlenderCanvas::
GetViewPort() {
#ifdef DEBUG
// If we're in a debug build, we might as well make sure our values don't differ
// from what the gpu thinks we have. This could lead to nasty, hard to find bugs.
int viewport[4];
glGetIntegerv(GL_VIEWPORT, viewport);
assert(viewport[0] == m_viewport[0]);
assert(viewport[1] == m_viewport[1]);
assert(viewport[2] == m_viewport[2]);
assert(viewport[3] == m_viewport[3]);
#endif
return m_viewport;
}
2002-10-12 11:37:38 +00:00
void KX_BlenderCanvas::SetMouseState(RAS_MouseState mousestate)
{
Patch #21789 - BGE Keyboard and Mouse Python types - by Mitchell Stokes(Moguri) The patch exposes mouse and keyboard read-only properties in the GameLogic module Also renames bge.keys to bge.events (* Note: name of bge submodules (logic, render, ...) may change before 2.5 final release [right Campbell?]). """ This patch adds two new types to the BGE: SCA_PythonKeyboard SCA_PythonMouse These two types allow users to make use of the keyboard and mouse without the need for a keyboard or mouse sensor. SCA_PythonKeyboard has an events property that acts just like SCA_KeyboardSensor.events. SCA_PythonMouse also has an events property to check for mouse events. Further more it supports getting and setting normalized cursor position (from 0.0 to 1.0) with SCA_PythonMouse.position. The cursor can be shown/hidden using SCA_PythonMouse.visible. """ Its use is similar with current mouse and keyboard controllers. With the exception of mouse position being normalized and writable as well (replacing Rasterizer.setMousePosition). Code Sample: ###### from bge import logic, events mouse = logic.mouse keyboard = logic.keyboard for key,status in keyboard.events: if status == logic.KX_INPUT_JUST_ACTIVATED: if key == events.WKEY: print(mouse.position) # move_forward() mouse.visible = True # turn cursor visible mouse.position = 0.5,0.5 # centralize mouse - use tuple ###### * Important Note: mouse.position still will not work properly for Letterbox mode. In order to fix letterboxing I may need to move the set x,y mouse function to inside the canvas code (to avoid duplicated code between mouse sensor and bge.logic.mouse). I'll leave this for another commit though. Thanks Mitchell for the work on that.
2010-04-17 06:52:14 +00:00
m_mousestate = mousestate;
2002-10-12 11:37:38 +00:00
switch (mousestate)
{
case MOUSE_INVISIBLE:
{
WM_cursor_set(m_win, CURSOR_NONE);
2002-10-12 11:37:38 +00:00
break;
}
case MOUSE_WAIT:
{
WM_cursor_set(m_win, CURSOR_WAIT);
2002-10-12 11:37:38 +00:00
break;
}
case MOUSE_NORMAL:
{
WM_cursor_set(m_win, CURSOR_STD);
2002-10-12 11:37:38 +00:00
break;
}
default:
{
}
}
}
// (0,0) is top left, (width,height) is bottom right
void KX_BlenderCanvas::SetMousePosition(int x,int y)
{
int winX = m_frame_rect.GetLeft();
int winY = m_frame_rect.GetBottom();
int winH = m_frame_rect.GetHeight();
2002-10-12 11:37:38 +00:00
WM_cursor_warp(m_win, winX + x, winY + (winH-y));
2002-10-12 11:37:38 +00:00
}
/* get shot from frontbuffer sort of a copy from screendump.c */
static unsigned int *screenshot(ScrArea *curarea, int *dumpsx, int *dumpsy)
{
int x=0, y=0;
unsigned int *dumprect= NULL;
x= curarea->totrct.xmin;
y= curarea->totrct.ymin;
*dumpsx= curarea->totrct.xmax-x;
*dumpsy= curarea->totrct.ymax-y;
if (*dumpsx && *dumpsy) {
dumprect= (unsigned int *)MEM_mallocN(sizeof(int) * (*dumpsx) * (*dumpsy), "dumprect");
glReadBuffer(GL_FRONT);
glReadPixels(x, y, *dumpsx, *dumpsy, GL_RGBA, GL_UNSIGNED_BYTE, dumprect);
glFinish();
glReadBuffer(GL_BACK);
}
return dumprect;
}
2002-10-12 11:37:38 +00:00
void KX_BlenderCanvas::MakeScreenShot(const char *filename)
2002-10-12 11:37:38 +00:00
{
ScrArea area_dummy= {0};
bScreen *screen = m_win->screen;
unsigned int *dumprect;
int dumpsx, dumpsy;
area_dummy.totrct.xmin = m_frame_rect.GetLeft();
area_dummy.totrct.xmax = m_frame_rect.GetRight();
area_dummy.totrct.ymin = m_frame_rect.GetBottom();
area_dummy.totrct.ymax = m_frame_rect.GetTop();
dumprect = screenshot(&area_dummy, &dumpsx, &dumpsy);
if (dumprect) {
/* initialize image file format data */
Scene *scene = (screen)? screen->scene: NULL;
ImageFormatData im_format;
if (scene)
im_format = scene->r.im_format;
else
BKE_imformat_defaults(&im_format);
/* create file path */
char path[FILE_MAX];
BLI_strncpy(path, filename, sizeof(path));
BLI_path_abs(path, G.main->name);
BKE_add_image_extension_from_type(path, im_format.imtype);
/* create and save imbuf */
ImBuf *ibuf = IMB_allocImBuf(dumpsx, dumpsy, 24, 0);
ibuf->rect = dumprect;
BKE_imbuf_write_as(ibuf, path, &im_format, false);
ibuf->rect = NULL;
IMB_freeImBuf(ibuf);
MEM_freeN(dumprect);
}
2002-10-12 11:37:38 +00:00
}