1
0
forked from 0ad/0ad

Moved all cursor code into cursor.cpp

This was SVN commit r844.
This commit is contained in:
Ykkrosh 2004-07-29 16:14:22 +00:00
parent 836d34660c
commit f31bc328f2
2 changed files with 174 additions and 193 deletions

View File

@ -1,24 +1,189 @@
#include "precompiled.h" #include "precompiled.h"
#include <string.h> #include <string.h>
#include <sstream>
#ifdef _WIN32 #ifdef _WIN32
// From win.cpp (so we don't need all the Windows headers in here) #include "lib/sysdep/win/win_internal.h"
void cursor_set(const char* name);
#include "lib/res/tex.h"
#include "lib/res/ogl_tex.h"
#include "lib/res/vfs.h"
#include "lib/ogl.h"
#include "lib/res/CVFSFile.h"
// Slightly hacky resource-loading code, but it doesn't
// need to do anything particularly clever.
// Pass name==NULL to unset the cursor. Pass name=="xyz"
// to load "art/textures/cursors/xyz.txt" (containing e.g. "10 10"
// for hotspot position) and "art/textures/cursors/xyz.png".
void cursor_draw(const char* name) void cursor_draw(const char* name)
{ {
cursor_set(name); static HICON last_cursor = NULL;
static char* last_name = NULL;
if (name == NULL)
{
// Clean up and go back to the standard Windows cursor
SetCursor(LoadCursor(NULL, IDC_ARROW));
if (last_cursor)
DestroyIcon(last_cursor);
last_cursor = NULL;
delete[] last_name;
last_name = NULL;
return;
}
// Don't do much if it's the same as last time
if (last_name && !strcmp(name, last_name))
{
if (GetCursor() != last_cursor)
SetCursor(last_cursor);
return;
}
// Store the name, so that the code can tell when it's
// being called several times with the same cursor
delete[] last_name;
last_name = new char[strlen(name)+1];
strcpy(last_name, name);
char filename[VFS_MAX_PATH];
// Load the .txt file containing the pixel offset of
// the cursor's hotspot (the bit of it that's
// drawn at (mouse_x,mouse_y) )
sprintf(filename, "art/textures/cursors/%s.txt", name);
int hotspotx, hotspoty;
{
CVFSFile file;
if (file.Load(filename) != PSRETURN_OK)
{
assert(! "Error loading cursor hotspot .txt file");
return;
}
std::stringstream s;
s << file.GetAsString();
s >> hotspotx >> hotspoty;
}
sprintf(filename, "art/textures/cursors/%s.png", name);
Handle tex = tex_load(filename);
if (tex <= 0)
{
// TODO: Handle errors
assert(! "Error loading cursor texture");
return;
}
// Convert the image data into a DIB
int w, h, fmt, bpp;
u32* imgdata;
tex_info(tex, &w, &h, &fmt, &bpp, (void**)&imgdata);
u32* imgdata_bgra;
if (fmt == GL_BGRA)
{
// No conversion needed
imgdata_bgra = imgdata;
}
else if (fmt == GL_RGBA)
{
// Convert ABGR -> ARGB (little-endian)
imgdata_bgra = new u32[w*h];
for (int i=0; i<w*h; ++i)
{
imgdata_bgra[i] = (
(imgdata[i] & 0xff00ff00) // G and A
| ( (imgdata[i] << 16) & 0x00ff0000) // R
| ( (imgdata[i] >> 16) & 0x000000ff) // B
);
}
}
else
{
// TODO: Handle errors
assert(! "Cursor texture not 32-bit RGBA/BGRA");
tex_free(tex);
return;
}
BITMAPINFOHEADER dibheader = {
sizeof(BITMAPINFOHEADER), // biSize
w, // biWidth
-h, // biHeight (negative for top-down)
1, // biPlanes
32, // biBitCount
BI_RGB, // biCompression
0, // biSizeImage (not needed for BI_RGB)
0, // biXPelsPerMeter (I really don't care how many pixels are in a meter)
0, // biYPelsPerMeter
0, // biClrUser (0 == maximum for this biBitCount. I hope we're not going to run in paletted display modes.)
0 // biClrImportant
};
BITMAPINFO dibinfo = {
dibheader, // bmiHeader
NULL // bmiColors[]
};
HDC hDC = wglGetCurrentDC();
HBITMAP iconbitmap = CreateDIBitmap(hDC, &dibheader, CBM_INIT, imgdata_bgra, &dibinfo, 0);
if (! iconbitmap)
{
// TODO: Handle errors
assert(! "Error creating icon DIB");
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
return;
}
ICONINFO info = { FALSE, hotspotx, hotspoty, iconbitmap, iconbitmap };
HICON cursor = CreateIconIndirect(&info);
DeleteObject(iconbitmap);
if (! cursor)
{
// TODO: Handle errors
assert(! "Error creating cursor");
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
return;
}
if (last_cursor)
DestroyIcon(last_cursor);
SetCursor(cursor);
last_cursor = cursor;
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
} }
/****************************************************************/
#else // #ifdef _WIN32 #else // #ifdef _WIN32
#include "lib/res/tex.h" #include "lib/res/tex.h"
#include "lib/res/ogl_tex.h" #include "lib/res/ogl_tex.h"
#include "lib/res/vfs.h" #include "lib/res/vfs.h"
#include "lib/ogl.h" #include "lib/ogl.h"
#include "lib/res/CVFSFile.h"
extern int mouse_x, mouse_y; extern int mouse_x, mouse_y;
@ -90,24 +255,17 @@ void cursor_draw(const char* name)
sprintf(filename, "art/textures/cursors/%s.txt", name); sprintf(filename, "art/textures/cursors/%s.txt", name);
int hotspotx, hotspoty; int hotspotx, hotspoty;
{ {
void* data; CVFSFile file;
size_t size; if (file.Load(filename) != PSRETURN_OK)
Handle h = vfs_load(filename, data, size);
if (h <= 0)
{ {
// TODO: Handle errors
assert(! "Error loading cursor hotspot .txt file"); assert(! "Error loading cursor hotspot .txt file");
return; return;
} }
if (sscanf((const char*)data, "%d %d", &hotspotx, &hotspoty) != 2) std::stringstream s;
{ s << file.GetAsString();
// TODO: Handle errors s >> hotspotx >> hotspoty;
assert(! "Invalid contents of cursor hotspot .txt file (should be like \"123 456\")");
return;
}
// TODO: Unload the file
} }
sprintf(filename, "art/textures/cursors/%s.png", name); sprintf(filename, "art/textures/cursors/%s.png", name);

View File

@ -27,10 +27,6 @@
#include <stdlib.h> // __argc #include <stdlib.h> // __argc
#include <malloc.h> #include <malloc.h>
#include "lib/res/vfs.h"
#include "lib/res/ogl_tex.h"
#include "lib/ogl.h"
void sle(int x) void sle(int x)
{ {
SetLastError((DWORD)x); SetLastError((DWORD)x);
@ -189,179 +185,6 @@ int clipboard_free(wchar_t* copy)
} }
// Slightly hacky resource-loading code, but it doesn't
// need to do anything particularly clever.
// Pass name==NULL to unset the cursor. Pass name=="xyz"
// to load "art/textures/cursors/xyz.txt" (containing e.g. "10 10"
// for hotspot position) and "art/textures/cursors/xyz.png".
void cursor_set(const char* name)
{
static HICON last_cursor = NULL;
static char* last_name = NULL;
if (name == NULL)
{
// Clean up and go back to the standard Windows cursor
SetCursor(LoadCursor(NULL, IDC_ARROW));
if (last_cursor)
DestroyIcon(last_cursor);
last_cursor = NULL;
delete[] last_name;
last_name = NULL;
return;
}
// Don't do much if it's the same as last time
if (last_name && !strcmp(name, last_name))
{
if (GetCursor() != last_cursor)
SetCursor(last_cursor);
return;
}
// Store the name, so that the code can tell when it's
// being called several times with the same cursor
delete[] last_name;
last_name = new char[strlen(name)+1];
strcpy(last_name, name);
char filename[VFS_MAX_PATH];
// Load the .txt file containing the pixel offset of
// the cursor's hotspot (the bit of it that's
// drawn at (mouse_x,mouse_y) )
sprintf(filename, "art/textures/cursors/%s.txt", name);
int hotspotx, hotspoty;
{
void* data;
size_t size;
Handle h = vfs_load(filename, data, size);
if (h <= 0)
{
// TODO: Handle errors
assert(! "Error loading cursor hotspot .txt file");
return;
}
if (sscanf((const char*)data, "%d %d", &hotspotx, &hotspoty) != 2)
{
// TODO: Handle errors
assert(! "Invalid contents of cursor hotspot .txt file (should be like \"123 456\")");
return;
}
// TODO: Unload the file
}
sprintf(filename, "art/textures/cursors/%s.png", name);
Handle tex = tex_load(filename);
if (tex <= 0)
{
// TODO: Handle errors
assert(! "Error loading cursor texture");
return;
}
// Convert the image data into a DIB
int w, h, fmt, bpp;
u32* imgdata;
tex_info(tex, &w, &h, &fmt, &bpp, (void**)&imgdata);
u32* imgdata_bgra;
if (fmt == GL_BGRA)
{
// No conversion needed
imgdata_bgra = imgdata;
}
else if (fmt == GL_RGBA)
{
// Convert ABGR -> ARGB (little-endian)
imgdata_bgra = new u32[w*h];
for (int i=0; i<w*h; ++i)
{
imgdata_bgra[i] = (
(imgdata[i] & 0xff00ff00) // G and A
| ( (imgdata[i] << 16) & 0x00ff0000) // R
| ( (imgdata[i] >> 16) & 0x000000ff) // B
);
}
}
else
{
// TODO: Handle errors
assert(! "Cursor texture not 32-bit RGBA/BGRA");
tex_free(tex);
return;
}
BITMAPINFOHEADER dibheader = {
sizeof(BITMAPINFOHEADER), // biSize
w, // biWidth
-h, // biHeight (negative for top-down)
1, // biPlanes
32, // biBitCount
BI_RGB, // biCompression
0, // biSizeImage (not needed for BI_RGB)
0, // biXPelsPerMeter (I really don't care how many pixels are in a meter)
0, // biYPelsPerMeter
0, // biClrUser (0 == maximum for this biBitCount. I hope we're not going to run in paletted display modes.)
0 // biClrImportant
};
BITMAPINFO dibinfo = {
dibheader, // bmiHeader
NULL // bmiColors[]
};
HDC hDC = wglGetCurrentDC();
HBITMAP iconbitmap = CreateDIBitmap(hDC, &dibheader, CBM_INIT, imgdata_bgra, &dibinfo, 0);
if (! iconbitmap)
{
// TODO: Handle errors
assert(! "Error creating icon DIB");
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
return;
}
ICONINFO info = { FALSE, hotspotx, hotspoty, iconbitmap, iconbitmap };
HICON cursor = CreateIconIndirect(&info);
DeleteObject(iconbitmap);
if (! cursor)
{
// TODO: Handle errors
assert(! "Error creating cursor");
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
return;
}
if (last_cursor)
DestroyIcon(last_cursor);
SetCursor(cursor);
last_cursor = cursor;
if (imgdata_bgra != imgdata) delete[] imgdata_bgra;
tex_free(tex);
}
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
// //
// init and shutdown mechanism // init and shutdown mechanism