This code was not tested and breaks in Release builds, reverting to restore
functionality of the nightly. All in-game menus do not work and generating
a world crashes.
This reverts commit a9be52c41a.
329 lines
8.1 KiB
C++
329 lines
8.1 KiB
C++
#include "stdafx.h"
|
|
#include "Screen.h"
|
|
#include "Button.h"
|
|
#include "ChatScreen.h"
|
|
#include "GuiParticles.h"
|
|
#include "Tesselator.h"
|
|
#include "Textures.h"
|
|
#include "..\Minecraft.World\SoundTypes.h"
|
|
#ifdef _WINDOWS64
|
|
#include "Windows64\KeyboardMouseInput.h"
|
|
#endif
|
|
|
|
|
|
|
|
Screen::Screen() // 4J added
|
|
{
|
|
minecraft = NULL;
|
|
width = 0;
|
|
height = 0;
|
|
passEvents = false;
|
|
font = NULL;
|
|
particles = NULL;
|
|
clickedButton = NULL;
|
|
}
|
|
|
|
void Screen::render(int xm, int ym, float a)
|
|
{
|
|
for (Button* button : buttons)
|
|
{
|
|
if ( button )
|
|
button->render(minecraft, xm, ym);
|
|
}
|
|
}
|
|
|
|
void Screen::keyPressed(wchar_t eventCharacter, int eventKey)
|
|
{
|
|
if (eventKey == Keyboard::KEY_ESCAPE)
|
|
{
|
|
minecraft->setScreen(NULL);
|
|
// minecraft->grabMouse(); // 4J - removed
|
|
}
|
|
}
|
|
|
|
wstring Screen::getClipboard()
|
|
{
|
|
#ifdef _WINDOWS64
|
|
if (!OpenClipboard(NULL)) return wstring();
|
|
HANDLE h = GetClipboardData(CF_UNICODETEXT);
|
|
wstring out;
|
|
if (h)
|
|
{
|
|
const wchar_t *p = reinterpret_cast<const wchar_t*>(GlobalLock(h));
|
|
if (p) { out = p; GlobalUnlock(h); }
|
|
}
|
|
CloseClipboard();
|
|
return out;
|
|
#else
|
|
return wstring();
|
|
#endif
|
|
}
|
|
|
|
void Screen::setClipboard(const wstring& str)
|
|
{
|
|
#ifdef _WINDOWS64
|
|
if (!OpenClipboard(NULL)) return;
|
|
EmptyClipboard();
|
|
size_t len = (str.length() + 1) * sizeof(wchar_t);
|
|
HGLOBAL h = GlobalAlloc(GMEM_MOVEABLE, len);
|
|
if (h) { memcpy(GlobalLock(h), str.c_str(), len); GlobalUnlock(h); SetClipboardData(CF_UNICODETEXT, h); }
|
|
CloseClipboard();
|
|
#endif
|
|
}
|
|
|
|
void Screen::mouseClicked(int x, int y, int buttonNum)
|
|
{
|
|
if (buttonNum == 0)
|
|
{
|
|
for (Button* button : buttons)
|
|
{
|
|
if ( button && button->clicked(minecraft, x, y) )
|
|
{
|
|
clickedButton = button;
|
|
minecraft->soundEngine->playUI(eSoundType_RANDOM_CLICK, 1, 1);
|
|
buttonClicked(button);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Screen::mouseReleased(int x, int y, int buttonNum)
|
|
{
|
|
if (clickedButton!=NULL && buttonNum==0)
|
|
{
|
|
clickedButton->released(x, y);
|
|
clickedButton = NULL;
|
|
}
|
|
}
|
|
|
|
void Screen::buttonClicked(Button *button)
|
|
{
|
|
}
|
|
|
|
void Screen::init(Minecraft *minecraft, int width, int height)
|
|
{
|
|
particles = new GuiParticles(minecraft);
|
|
this->minecraft = minecraft;
|
|
this->font = minecraft->font;
|
|
this->width = width;
|
|
this->height = height;
|
|
buttons.clear();
|
|
init();
|
|
}
|
|
|
|
void Screen::setSize(int width, int height)
|
|
{
|
|
this->width = width;
|
|
this->height = height;
|
|
}
|
|
|
|
void Screen::init()
|
|
{
|
|
}
|
|
|
|
void Screen::updateEvents()
|
|
{
|
|
#ifdef _WINDOWS64
|
|
// Poll mouse button state and dispatch click/release events
|
|
for (int btn = 0; btn < 3; btn++)
|
|
{
|
|
if (g_KBMInput.IsMouseButtonPressed(btn))
|
|
{
|
|
int xm = Mouse::getX() * width / minecraft->width;
|
|
int ym = height - Mouse::getY() * height / minecraft->height - 1;
|
|
mouseClicked(xm, ym, btn);
|
|
}
|
|
if (g_KBMInput.IsMouseButtonReleased(btn))
|
|
{
|
|
int xm = Mouse::getX() * width / minecraft->width;
|
|
int ym = height - Mouse::getY() * height / minecraft->height - 1;
|
|
mouseReleased(xm, ym, btn);
|
|
}
|
|
}
|
|
|
|
// Only drain WM_CHAR when this screen wants text input (e.g. ChatScreen); otherwise we'd steal keys from the game
|
|
if (dynamic_cast<ChatScreen*>(this) != nullptr)
|
|
{
|
|
wchar_t ch;
|
|
while (g_KBMInput.ConsumeChar(ch))
|
|
{
|
|
if (ch >= 0x20)
|
|
keyPressed(ch, -1);
|
|
else if (ch == 0x08)
|
|
keyPressed(0, Keyboard::KEY_BACK);
|
|
else if (ch == 0x0D)
|
|
keyPressed(0, Keyboard::KEY_RETURN);
|
|
}
|
|
}
|
|
|
|
// Arrow key repeat: deliver on first press (when key down and last==0) and while held (throttled)
|
|
static DWORD s_arrowLastTime[2] = { 0, 0 };
|
|
static bool s_arrowFirstRepeat[2] = { false, false };
|
|
const DWORD ARROW_REPEAT_DELAY_MS = 250;
|
|
const DWORD ARROW_REPEAT_INTERVAL_MS = 50;
|
|
DWORD now = GetTickCount();
|
|
|
|
// Poll keyboard events (special keys that may not come through WM_CHAR, e.g. Escape, arrows)
|
|
for (int vk = 0; vk < 256; vk++)
|
|
{
|
|
bool deliver = g_KBMInput.IsKeyPressed(vk);
|
|
if (vk == VK_LEFT || vk == VK_RIGHT)
|
|
{
|
|
int idx = (vk == VK_LEFT) ? 0 : 1;
|
|
if (!g_KBMInput.IsKeyDown(vk))
|
|
{
|
|
s_arrowLastTime[idx] = 0;
|
|
s_arrowFirstRepeat[idx] = false;
|
|
}
|
|
else
|
|
{
|
|
DWORD last = s_arrowLastTime[idx];
|
|
if (last == 0)
|
|
deliver = true;
|
|
else if (!deliver)
|
|
{
|
|
DWORD interval = s_arrowFirstRepeat[idx] ? ARROW_REPEAT_INTERVAL_MS : ARROW_REPEAT_DELAY_MS;
|
|
if ((now - last) >= interval)
|
|
{
|
|
deliver = true;
|
|
s_arrowFirstRepeat[idx] = true;
|
|
}
|
|
}
|
|
if (deliver)
|
|
s_arrowLastTime[idx] = now;
|
|
}
|
|
}
|
|
// Escape: deliver when key is down so we don't miss it (IsKeyPressed can be one frame late)
|
|
if (vk == VK_ESCAPE && g_KBMInput.IsKeyDown(VK_ESCAPE))
|
|
deliver = true;
|
|
if (!deliver) continue;
|
|
|
|
if (dynamic_cast<ChatScreen*>(this) != nullptr &&
|
|
(vk >= 'A' && vk <= 'Z' || vk >= '0' && vk <= '9' || vk == VK_SPACE || vk == VK_RETURN || vk == VK_BACK))
|
|
continue;
|
|
// Map to Screen::keyPressed
|
|
int mappedKey = -1;
|
|
wchar_t ch = 0;
|
|
if (vk == VK_ESCAPE) mappedKey = Keyboard::KEY_ESCAPE;
|
|
else if (vk == VK_RETURN) mappedKey = Keyboard::KEY_RETURN;
|
|
else if (vk == VK_BACK) mappedKey = Keyboard::KEY_BACK;
|
|
else if (vk == VK_UP) mappedKey = Keyboard::KEY_UP;
|
|
else if (vk == VK_DOWN) mappedKey = Keyboard::KEY_DOWN;
|
|
else if (vk == VK_LEFT) mappedKey = Keyboard::KEY_LEFT;
|
|
else if (vk == VK_RIGHT) mappedKey = Keyboard::KEY_RIGHT;
|
|
else if (vk == VK_LSHIFT || vk == VK_RSHIFT) mappedKey = Keyboard::KEY_LSHIFT;
|
|
else if (vk == VK_TAB) mappedKey = Keyboard::KEY_TAB;
|
|
else if (vk >= 'A' && vk <= 'Z')
|
|
{
|
|
ch = static_cast<wchar_t>(vk - 'A' + L'a');
|
|
if (g_KBMInput.IsKeyDown(VK_LSHIFT) || g_KBMInput.IsKeyDown(VK_RSHIFT)) ch = static_cast<wchar_t>(vk);
|
|
}
|
|
else if (vk >= '0' && vk <= '9') ch = static_cast<wchar_t>(vk);
|
|
else if (vk == VK_SPACE) ch = L' ';
|
|
|
|
if (mappedKey != -1) keyPressed(ch, mappedKey);
|
|
else if (ch != 0) keyPressed(ch, -1);
|
|
}
|
|
#else
|
|
/* 4J - TODO
|
|
while (Mouse.next()) {
|
|
mouseEvent();
|
|
}
|
|
|
|
while (Keyboard.next()) {
|
|
keyboardEvent();
|
|
}
|
|
*/
|
|
#endif
|
|
}
|
|
|
|
void Screen::mouseEvent()
|
|
{
|
|
#ifdef _WINDOWS64
|
|
// Mouse event dispatching is handled directly in updateEvents() for Windows
|
|
#else
|
|
/* 4J - TODO
|
|
if (Mouse.getEventButtonState()) {
|
|
int xm = Mouse.getEventX() * width / minecraft.width;
|
|
int ym = height - Mouse.getEventY() * height / minecraft.height - 1;
|
|
mouseClicked(xm, ym, Mouse.getEventButton());
|
|
} else {
|
|
int xm = Mouse.getEventX() * width / minecraft.width;
|
|
int ym = height - Mouse.getEventY() * height / minecraft.height - 1;
|
|
mouseReleased(xm, ym, Mouse.getEventButton());
|
|
}
|
|
*/
|
|
#endif
|
|
}
|
|
|
|
void Screen::keyboardEvent()
|
|
{
|
|
/* 4J - TODO
|
|
if (Keyboard.getEventKeyState()) {
|
|
if (Keyboard.getEventKey() == Keyboard.KEY_F11) {
|
|
minecraft.toggleFullScreen();
|
|
return;
|
|
}
|
|
keyPressed(Keyboard.getEventCharacter(), Keyboard.getEventKey());
|
|
}
|
|
*/
|
|
}
|
|
|
|
void Screen::tick()
|
|
{
|
|
}
|
|
|
|
void Screen::removed()
|
|
{
|
|
}
|
|
|
|
void Screen::renderBackground()
|
|
{
|
|
renderBackground(0);
|
|
}
|
|
|
|
void Screen::renderBackground(int vo)
|
|
{
|
|
if (minecraft->level != NULL)
|
|
{
|
|
fillGradient(0, 0, width, height, 0xc0101010, 0xd0101010);
|
|
}
|
|
else
|
|
{
|
|
renderDirtBackground(vo);
|
|
}
|
|
}
|
|
|
|
void Screen::renderDirtBackground(int vo)
|
|
{
|
|
// 4J Unused
|
|
#if 0
|
|
glDisable(GL_LIGHTING);
|
|
glDisable(GL_FOG);
|
|
Tesselator *t = Tesselator::getInstance();
|
|
glBindTexture(GL_TEXTURE_2D, minecraft->textures->loadTexture(L"/gui/background.png"));
|
|
glColor4f(1, 1, 1, 1);
|
|
float s = 32;
|
|
t->begin();
|
|
t->color(0x404040);
|
|
t->vertexUV(static_cast<float>(0), static_cast<float>(height), static_cast<float>(0), static_cast<float>(0), static_cast<float>(height / s + vo));
|
|
t->vertexUV(static_cast<float>(width), static_cast<float>(height), static_cast<float>(0), static_cast<float>(width / s), static_cast<float>(height / s + vo));
|
|
t->vertexUV(static_cast<float>(width), static_cast<float>(0), static_cast<float>(0), static_cast<float>(width / s), static_cast<float>(0 + vo));
|
|
t->vertexUV(static_cast<float>(0), static_cast<float>(0), static_cast<float>(0), static_cast<float>(0), static_cast<float>(0 + vo));
|
|
t->end();
|
|
#endif
|
|
}
|
|
|
|
bool Screen::isPauseScreen()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
void Screen::confirmResult(bool result, int id)
|
|
{
|
|
}
|
|
|
|
void Screen::tabPressed()
|
|
{
|
|
}
|