2015-11-19 07:23:41 +01:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2015-11-19 07:56:54 +01:00
|
|
|
* Copyright (C) 2015 - Ali Bouhlel
|
2015-11-19 07:23:41 +01:00
|
|
|
* Copyright (C) 2011-2015 - Daniel De Matteis
|
|
|
|
*
|
|
|
|
* RetroArch 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 Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* RetroArch 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 RetroArch.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <boolean.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma comment( lib, "comctl32" )
|
|
|
|
#endif
|
|
|
|
|
2015-11-19 07:53:00 +01:00
|
|
|
#define IDI_ICON 1
|
2015-11-19 07:48:56 +01:00
|
|
|
|
2015-11-19 07:37:22 +01:00
|
|
|
#ifndef _WIN32_WINNT
|
|
|
|
#define _WIN32_WINNT 0x0500 //_WIN32_WINNT_WIN2K
|
|
|
|
#endif
|
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
#ifndef _WIN32_IE
|
|
|
|
#define _WIN32_IE 0x0300
|
|
|
|
#endif
|
|
|
|
|
2015-11-19 07:37:22 +01:00
|
|
|
#include <windows.h>
|
|
|
|
#include <commdlg.h>
|
2015-11-19 07:56:54 +01:00
|
|
|
#include <commctrl.h>
|
2015-11-19 07:37:22 +01:00
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
#include <retro_inline.h>
|
2015-11-19 07:23:41 +01:00
|
|
|
#include <file/file_path.h>
|
2015-11-19 07:56:54 +01:00
|
|
|
|
2015-11-19 08:59:45 +01:00
|
|
|
#ifdef HAVE_MENU
|
2015-11-19 09:00:32 +01:00
|
|
|
#include "../../menu/menu_hash.h"
|
2015-11-19 08:59:45 +01:00
|
|
|
#endif
|
|
|
|
|
2015-11-19 07:23:41 +01:00
|
|
|
#include "../ui_companion_driver.h"
|
2015-11-19 07:56:54 +01:00
|
|
|
#include "../../driver.h"
|
|
|
|
#include "../../runloop.h"
|
|
|
|
#include "../../gfx/video_context_driver.h"
|
|
|
|
|
|
|
|
#include "../../gfx/common/gl_common.h"
|
|
|
|
#include "../../gfx/common/win32_common.h"
|
|
|
|
|
|
|
|
#define SHADER_DLG_WIDTH 220
|
|
|
|
#define SHADER_DLG_MIN_HEIGHT 200
|
|
|
|
#define SHADER_DLG_MAX_HEIGHT 800
|
|
|
|
#define SHADER_DLG_CTRL_MARGIN 8
|
|
|
|
#define SHADER_DLG_CTRL_X 10
|
|
|
|
#define SHADER_DLG_CHECKBOX_HEIGHT 15
|
|
|
|
#define SHADER_DLG_SEPARATOR_HEIGHT 10
|
|
|
|
#define SHADER_DLG_LABEL_HEIGHT 14
|
|
|
|
#define SHADER_DLG_TRACKBAR_HEIGHT 22
|
|
|
|
#define SHADER_DLG_TRACKBAR_LABEL_WIDTH 30
|
2015-11-19 07:23:41 +01:00
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
#define SHADER_DLG_CTRL_WIDTH (SHADER_DLG_WIDTH - 2 * SHADER_DLG_CTRL_X)
|
|
|
|
#define SHADER_DLG_TRACKBAR_WIDTH (SHADER_DLG_CTRL_WIDTH - SHADER_DLG_TRACKBAR_LABEL_WIDTH)
|
2015-11-19 07:37:22 +01:00
|
|
|
|
2015-11-19 07:23:41 +01:00
|
|
|
typedef struct ui_companion_win32
|
|
|
|
{
|
|
|
|
void *empty;
|
|
|
|
} ui_companion_win32_t;
|
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
enum shader_param_ctrl_type
|
|
|
|
{
|
|
|
|
SHADER_PARAM_CTRL_NONE = 0,
|
|
|
|
SHADER_PARAM_CTRL_CHECKBOX,
|
|
|
|
SHADER_PARAM_CTRL_TRACKBAR
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
SHADER_DLG_CHECKBOX_ONTOP_ID = GFX_MAX_PARAMETERS,
|
|
|
|
SHADER_DLG_CHECKBOX_BUTTON1_ID,
|
|
|
|
SHADER_DLG_CHECKBOX_BUTTON2_ID
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
enum shader_param_ctrl_type type;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
HWND hwnd;
|
|
|
|
} checkbox;
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
HWND hwnd;
|
|
|
|
HWND label_title;
|
|
|
|
HWND label_val;
|
|
|
|
} trackbar;
|
|
|
|
};
|
|
|
|
} shader_param_ctrl_t;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
HWND hwnd;
|
|
|
|
HWND on_top_checkbox;
|
|
|
|
HWND separator;
|
|
|
|
shader_param_ctrl_t controls[GFX_MAX_PARAMETERS];
|
|
|
|
int parameters_start_y;
|
|
|
|
} shader_dlg_t;
|
|
|
|
|
|
|
|
static shader_dlg_t g_shader_dlg = {0};
|
|
|
|
|
2015-11-30 05:13:12 +01:00
|
|
|
static void shader_dlg_refresh_trackbar_label(int index)
|
2015-11-19 07:56:54 +01:00
|
|
|
{
|
|
|
|
char val_buffer[32] = {0};
|
|
|
|
struct video_shader* shader = video_shader_driver_get_current_shader();
|
|
|
|
|
|
|
|
if (floorf(shader->parameters[index].current) == shader->parameters[index].current)
|
|
|
|
snprintf(val_buffer, sizeof(val_buffer), "%.0f", shader->parameters[index].current);
|
|
|
|
else
|
|
|
|
snprintf(val_buffer, sizeof(val_buffer), "%.2f", shader->parameters[index].current);
|
|
|
|
|
|
|
|
SendMessage(g_shader_dlg.controls[index].trackbar.label_val, WM_SETTEXT, 0, (LPARAM)val_buffer);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void shader_dlg_params_refresh(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct video_shader* shader = video_shader_driver_get_current_shader();
|
|
|
|
|
|
|
|
for (i = 0; i < GFX_MAX_PARAMETERS; i++)
|
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
shader_param_ctrl_t*control = &g_shader_dlg.controls[i];
|
|
|
|
|
|
|
|
if (control->type == SHADER_PARAM_CTRL_NONE)
|
2015-11-19 07:56:54 +01:00
|
|
|
break;
|
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
switch (control->type)
|
2015-11-19 07:56:54 +01:00
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
case SHADER_PARAM_CTRL_CHECKBOX:
|
|
|
|
{
|
|
|
|
bool checked = (shader->parameters[i].current == shader->parameters[i].maximum);
|
|
|
|
SendMessage(control->checkbox.hwnd, BM_SETCHECK, checked, 0);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SHADER_PARAM_CTRL_TRACKBAR:
|
|
|
|
shader_dlg_refresh_trackbar_label(i);
|
|
|
|
|
|
|
|
SendMessage(control->trackbar.hwnd, TBM_SETRANGEMIN, (WPARAM)TRUE, (LPARAM)0);
|
|
|
|
SendMessage(control->trackbar.hwnd, TBM_SETRANGEMAX, (WPARAM)TRUE,
|
|
|
|
(LPARAM)((shader->parameters[i].maximum - shader->parameters[i].minimum) / shader->parameters[i].step));
|
|
|
|
SendMessage(control->trackbar.hwnd, TBM_SETPOS, (WPARAM)TRUE,
|
|
|
|
(LPARAM)((shader->parameters[i].current - shader->parameters[i].minimum) / shader->parameters[i].step));
|
|
|
|
break;
|
2015-11-22 22:25:01 +01:00
|
|
|
case SHADER_PARAM_CTRL_NONE:
|
|
|
|
default:
|
|
|
|
break;
|
2015-11-19 07:56:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void shader_dlg_params_clear(void)
|
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
unsigned i;
|
2015-11-19 07:56:54 +01:00
|
|
|
|
|
|
|
for (i = 0; i < GFX_MAX_PARAMETERS; i++)
|
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
shader_param_ctrl_t*control = &g_shader_dlg.controls[i];
|
|
|
|
|
|
|
|
if (control->type == SHADER_PARAM_CTRL_NONE)
|
2015-11-19 07:56:54 +01:00
|
|
|
break;
|
2015-11-19 08:52:26 +01:00
|
|
|
|
|
|
|
switch (control->type)
|
2015-11-19 07:56:54 +01:00
|
|
|
{
|
2015-11-23 10:51:03 +01:00
|
|
|
case SHADER_PARAM_CTRL_NONE:
|
|
|
|
break;
|
2015-11-19 08:52:26 +01:00
|
|
|
case SHADER_PARAM_CTRL_CHECKBOX:
|
|
|
|
DestroyWindow(control->checkbox.hwnd);
|
|
|
|
break;
|
|
|
|
case SHADER_PARAM_CTRL_TRACKBAR:
|
|
|
|
DestroyWindow(control->trackbar.label_title);
|
|
|
|
DestroyWindow(control->trackbar.label_val);
|
|
|
|
DestroyWindow(control->trackbar.hwnd);
|
|
|
|
break;
|
2015-11-19 07:56:54 +01:00
|
|
|
}
|
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
control->type = SHADER_PARAM_CTRL_NONE;
|
2015-11-19 07:56:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void shader_dlg_params_reload(void)
|
|
|
|
{
|
|
|
|
HFONT hFont;
|
|
|
|
RECT parent_rect;
|
|
|
|
int i, pos_x, pos_y;
|
|
|
|
struct video_shader* shader = video_shader_driver_get_current_shader();
|
|
|
|
|
|
|
|
shader_dlg_params_clear();
|
|
|
|
|
|
|
|
if (!shader)
|
|
|
|
return;
|
|
|
|
if (shader->num_parameters > GFX_MAX_PARAMETERS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
|
|
|
|
pos_y = g_shader_dlg.parameters_start_y;
|
|
|
|
pos_x = SHADER_DLG_CTRL_X;
|
|
|
|
|
|
|
|
for (i = 0; i < (int)shader->num_parameters; i++)
|
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
shader_param_ctrl_t*control = &g_shader_dlg.controls[i];
|
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
if ((shader->parameters[i].minimum == 0.0)
|
|
|
|
&& (shader->parameters[i].maximum == (shader->parameters[i].minimum + shader->parameters[i].step)))
|
|
|
|
{
|
|
|
|
if ((pos_y + SHADER_DLG_CHECKBOX_HEIGHT + SHADER_DLG_CTRL_MARGIN + 20) > SHADER_DLG_MAX_HEIGHT)
|
|
|
|
{
|
2015-11-19 08:52:26 +01:00
|
|
|
pos_y = g_shader_dlg.parameters_start_y;
|
2015-11-19 07:56:54 +01:00
|
|
|
pos_x += SHADER_DLG_WIDTH;
|
|
|
|
}
|
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
control->type = SHADER_PARAM_CTRL_CHECKBOX;
|
|
|
|
control->checkbox.hwnd = CreateWindowEx(0, "BUTTON", shader->parameters[i].desc,
|
2015-11-19 07:56:54 +01:00
|
|
|
WS_CHILD | WS_VISIBLE | BS_AUTOCHECKBOX, pos_x, pos_y, SHADER_DLG_CTRL_WIDTH, SHADER_DLG_CHECKBOX_HEIGHT,
|
|
|
|
g_shader_dlg.hwnd, (HMENU)(size_t)i, NULL, NULL);
|
2015-11-19 08:52:26 +01:00
|
|
|
SendMessage(control->checkbox.hwnd, WM_SETFONT, (WPARAM)hFont, MAKELPARAM(TRUE, 0));
|
2015-11-19 07:56:54 +01:00
|
|
|
pos_y += SHADER_DLG_CHECKBOX_HEIGHT + SHADER_DLG_CTRL_MARGIN;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((pos_y + SHADER_DLG_LABEL_HEIGHT + SHADER_DLG_TRACKBAR_HEIGHT +
|
|
|
|
SHADER_DLG_CTRL_MARGIN + 20) > SHADER_DLG_MAX_HEIGHT)
|
|
|
|
{
|
|
|
|
pos_y = g_shader_dlg.parameters_start_y;
|
|
|
|
pos_x += SHADER_DLG_WIDTH;
|
|
|
|
}
|
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
control->type = SHADER_PARAM_CTRL_TRACKBAR;
|
|
|
|
control->trackbar.label_title = CreateWindowEx(0, "STATIC", shader->parameters[i].desc,
|
2015-11-19 07:56:54 +01:00
|
|
|
WS_CHILD | WS_VISIBLE | SS_LEFT, pos_x, pos_y, SHADER_DLG_CTRL_WIDTH, SHADER_DLG_LABEL_HEIGHT, g_shader_dlg.hwnd,
|
|
|
|
(HMENU)(size_t)i, NULL, NULL);
|
2015-11-19 08:52:26 +01:00
|
|
|
SendMessage(control->trackbar.label_title, WM_SETFONT, (WPARAM)hFont, MAKELPARAM(TRUE, 0));
|
2015-11-19 07:56:54 +01:00
|
|
|
|
|
|
|
pos_y += SHADER_DLG_LABEL_HEIGHT;
|
2015-11-19 08:52:26 +01:00
|
|
|
control->trackbar.hwnd = CreateWindowEx(0, TRACKBAR_CLASS, "",
|
2015-11-19 07:56:54 +01:00
|
|
|
WS_CHILD | WS_VISIBLE | TBS_HORZ | TBS_NOTICKS, pos_x + SHADER_DLG_TRACKBAR_LABEL_WIDTH, pos_y,
|
|
|
|
SHADER_DLG_TRACKBAR_WIDTH, SHADER_DLG_TRACKBAR_HEIGHT, g_shader_dlg.hwnd, (HMENU)(size_t)i, NULL, NULL);
|
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
control->trackbar.label_val = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE | SS_LEFT, pos_x,
|
2015-11-19 07:56:54 +01:00
|
|
|
pos_y, SHADER_DLG_TRACKBAR_LABEL_WIDTH, SHADER_DLG_LABEL_HEIGHT, g_shader_dlg.hwnd, (HMENU)(size_t)i, NULL, NULL);
|
2015-11-19 08:52:26 +01:00
|
|
|
SendMessage(control->trackbar.label_val, WM_SETFONT, (WPARAM)hFont, MAKELPARAM(TRUE, 0));
|
2015-11-19 07:56:54 +01:00
|
|
|
|
2015-11-19 08:52:26 +01:00
|
|
|
SendMessage(control->trackbar.hwnd, TBM_SETBUDDY, (WPARAM)TRUE,
|
|
|
|
(LPARAM)control->trackbar.label_val);
|
2015-11-19 07:56:54 +01:00
|
|
|
|
|
|
|
pos_y += SHADER_DLG_TRACKBAR_HEIGHT + SHADER_DLG_CTRL_MARGIN;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_shader_dlg.separator)
|
|
|
|
DestroyWindow(g_shader_dlg.separator);
|
|
|
|
|
|
|
|
g_shader_dlg.separator = CreateWindowEx(0, "STATIC", "", SS_ETCHEDHORZ | WS_VISIBLE | WS_CHILD, SHADER_DLG_CTRL_X,
|
|
|
|
g_shader_dlg.parameters_start_y - SHADER_DLG_CTRL_MARGIN - SHADER_DLG_SEPARATOR_HEIGHT / 2,
|
|
|
|
(pos_x - SHADER_DLG_CTRL_X) + SHADER_DLG_CTRL_WIDTH, SHADER_DLG_SEPARATOR_HEIGHT / 2, g_shader_dlg.hwnd, NULL, NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
shader_dlg_params_refresh();
|
|
|
|
|
|
|
|
GetWindowRect(g_shader_dlg.hwnd, &parent_rect);
|
|
|
|
SetWindowPos(g_shader_dlg.hwnd, NULL, 0, 0,
|
|
|
|
(pos_x - SHADER_DLG_CTRL_X) + SHADER_DLG_WIDTH,
|
|
|
|
(pos_x == SHADER_DLG_CTRL_X) ? pos_y + 30 : SHADER_DLG_MAX_HEIGHT,
|
|
|
|
SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void shader_dlg_update_on_top_state(void)
|
|
|
|
{
|
|
|
|
bool on_top = SendMessage(g_shader_dlg.on_top_checkbox, BM_GETCHECK, 0, 0) == BST_CHECKED;
|
|
|
|
SetWindowPos(g_shader_dlg.hwnd, on_top ? HWND_TOPMOST : HWND_NOTOPMOST , 0, 0, 0, 0,
|
|
|
|
SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void shader_dlg_show(HWND parent_hwnd)
|
|
|
|
{
|
|
|
|
if (!IsWindowVisible(g_shader_dlg.hwnd))
|
|
|
|
{
|
|
|
|
if (parent_hwnd)
|
|
|
|
{
|
|
|
|
RECT parent_rect;
|
|
|
|
GetWindowRect(parent_hwnd, &parent_rect);
|
|
|
|
SetWindowPos(g_shader_dlg.hwnd, HWND_TOP, parent_rect.right, parent_rect.top,
|
|
|
|
0, 0, SWP_NOSIZE | SWP_SHOWWINDOW);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ShowWindow(g_shader_dlg.hwnd, SW_SHOW);
|
|
|
|
|
|
|
|
shader_dlg_update_on_top_state();
|
|
|
|
|
|
|
|
shader_dlg_params_reload();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
SetFocus(g_shader_dlg.hwnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
static LRESULT CALLBACK ShaderDlgWndProc(HWND hwnd, UINT message,
|
|
|
|
WPARAM wparam, LPARAM lparam)
|
|
|
|
{
|
|
|
|
int i, pos;
|
|
|
|
struct video_shader* shader = video_shader_driver_get_current_shader();
|
|
|
|
|
|
|
|
switch (message)
|
|
|
|
{
|
|
|
|
case WM_CREATE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_CLOSE:
|
|
|
|
case WM_DESTROY:
|
|
|
|
case WM_QUIT:
|
|
|
|
ShowWindow(g_shader_dlg.hwnd, 0);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case WM_COMMAND:
|
|
|
|
i = LOWORD(wparam);
|
|
|
|
|
|
|
|
if (i == SHADER_DLG_CHECKBOX_ONTOP_ID)
|
|
|
|
{
|
|
|
|
shader_dlg_update_on_top_state();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i >= GFX_MAX_PARAMETERS)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (g_shader_dlg.controls[i].type != SHADER_PARAM_CTRL_CHECKBOX)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (SendMessage(g_shader_dlg.controls[i].checkbox.hwnd, BM_GETCHECK, 0, 0) == BST_CHECKED)
|
|
|
|
shader->parameters[i].current = shader->parameters[i].maximum;
|
|
|
|
else
|
|
|
|
shader->parameters[i].current = shader->parameters[i].minimum;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_HSCROLL:
|
|
|
|
i = GetWindowLong((HWND)lparam, GWL_ID);
|
|
|
|
|
|
|
|
if (i >= GFX_MAX_PARAMETERS)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (g_shader_dlg.controls[i].type != SHADER_PARAM_CTRL_TRACKBAR)
|
|
|
|
break;
|
|
|
|
|
|
|
|
pos = (int)SendMessage(g_shader_dlg.controls[i].trackbar.hwnd, TBM_GETPOS, 0, 0);
|
|
|
|
shader->parameters[i].current = shader->parameters[i].minimum + pos * shader->parameters[i].step;
|
|
|
|
|
|
|
|
shader_dlg_refresh_trackbar_label(i);
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return DefWindowProc(hwnd, message, wparam, lparam);
|
|
|
|
}
|
|
|
|
|
2015-11-19 08:37:25 +01:00
|
|
|
bool win32_window_init(WNDCLASSEX *wndclass, bool fullscreen, const char *class_name)
|
|
|
|
{
|
|
|
|
wndclass->cbSize = sizeof(WNDCLASSEX);
|
|
|
|
wndclass->style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC;
|
|
|
|
wndclass->hInstance = GetModuleHandle(NULL);
|
|
|
|
wndclass->hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
|
|
wndclass->lpszClassName = (class_name == NULL) ? "RetroArch" : class_name;
|
|
|
|
wndclass->hIcon = LoadIcon(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON));
|
|
|
|
wndclass->hIconSm = (HICON)LoadImage(GetModuleHandle(NULL),
|
|
|
|
MAKEINTRESOURCE(IDI_ICON), IMAGE_ICON, 16, 16, 0);
|
|
|
|
if (!fullscreen)
|
|
|
|
wndclass->hbrBackground = (HBRUSH)COLOR_WINDOW;
|
|
|
|
|
2015-11-28 21:50:28 +01:00
|
|
|
if (class_name != NULL)
|
2015-11-19 08:37:25 +01:00
|
|
|
wndclass->style |= CS_CLASSDC;
|
|
|
|
|
|
|
|
if (!RegisterClassEx(wndclass))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* This is non-NULL when we want a window for shader dialogs,
|
|
|
|
* therefore early return here */
|
|
|
|
/* TODO/FIXME - this is ugly. Find a better way */
|
|
|
|
if (class_name != NULL)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!win32_shader_dlg_init())
|
|
|
|
RARCH_ERR("[WGL]: wgl_shader_dlg_init() failed.\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-11-19 07:56:54 +01:00
|
|
|
bool win32_shader_dlg_init(void)
|
|
|
|
{
|
|
|
|
static bool inited = false;
|
|
|
|
int pos_y;
|
|
|
|
HFONT hFont;
|
|
|
|
|
|
|
|
if (g_shader_dlg.hwnd)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!inited)
|
|
|
|
{
|
|
|
|
WNDCLASSEX wc_shader_dlg = {0};
|
|
|
|
INITCOMMONCONTROLSEX comm_ctrl_init = {0};
|
|
|
|
|
|
|
|
comm_ctrl_init.dwSize = sizeof(comm_ctrl_init);
|
|
|
|
comm_ctrl_init.dwICC = ICC_BAR_CLASSES;
|
|
|
|
|
|
|
|
if (!InitCommonControlsEx(&comm_ctrl_init))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
wc_shader_dlg.lpfnWndProc = ShaderDlgWndProc;
|
|
|
|
wc_shader_dlg.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1);
|
|
|
|
|
2015-11-19 08:34:59 +01:00
|
|
|
if (!win32_window_init(&wc_shader_dlg, true, "Shader Dialog"))
|
2015-11-19 07:56:54 +01:00
|
|
|
return false;
|
|
|
|
|
|
|
|
inited = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
|
|
|
|
|
|
|
|
g_shader_dlg.hwnd = CreateWindowEx(0, "Shader Dialog", "Shader Parameters", WS_POPUPWINDOW | WS_CAPTION, 100, 100,
|
|
|
|
SHADER_DLG_WIDTH, SHADER_DLG_MIN_HEIGHT, NULL, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
pos_y = SHADER_DLG_CTRL_MARGIN;
|
|
|
|
g_shader_dlg.on_top_checkbox = CreateWindowEx(0, "BUTTON", "Always on Top", BS_AUTOCHECKBOX | WS_VISIBLE | WS_CHILD,
|
|
|
|
SHADER_DLG_CTRL_X, pos_y, SHADER_DLG_CTRL_WIDTH, SHADER_DLG_CHECKBOX_HEIGHT, g_shader_dlg.hwnd,
|
|
|
|
(HMENU)SHADER_DLG_CHECKBOX_ONTOP_ID, NULL, NULL);
|
|
|
|
pos_y += SHADER_DLG_CHECKBOX_HEIGHT + SHADER_DLG_CTRL_MARGIN;
|
|
|
|
|
|
|
|
SendMessage(g_shader_dlg.on_top_checkbox, WM_SETFONT, (WPARAM)hFont, MAKELPARAM(TRUE, 0));
|
|
|
|
|
|
|
|
pos_y += SHADER_DLG_SEPARATOR_HEIGHT + SHADER_DLG_CTRL_MARGIN;
|
|
|
|
|
|
|
|
g_shader_dlg.parameters_start_y = pos_y;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-11-19 08:17:59 +01:00
|
|
|
bool win32_browser(
|
|
|
|
HWND owner,
|
|
|
|
char *filename,
|
|
|
|
const char *extensions,
|
|
|
|
const char *title,
|
|
|
|
const char *initial_dir)
|
|
|
|
{
|
|
|
|
OPENFILENAME ofn;
|
|
|
|
|
|
|
|
memset((void*)&ofn, 0, sizeof(OPENFILENAME));
|
|
|
|
|
|
|
|
ofn.lStructSize = sizeof(OPENFILENAME);
|
|
|
|
ofn.hwndOwner = owner;
|
|
|
|
ofn.lpstrFilter = extensions;
|
|
|
|
ofn.lpstrFile = filename;
|
|
|
|
ofn.lpstrTitle = title;
|
|
|
|
ofn.lpstrInitialDir = TEXT(initial_dir);
|
|
|
|
ofn.lpstrDefExt = "";
|
|
|
|
ofn.nMaxFile = PATH_MAX;
|
|
|
|
ofn.Flags = OFN_FILEMUSTEXIST | OFN_HIDEREADONLY;
|
|
|
|
|
|
|
|
if (!GetOpenFileName(&ofn))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-11-19 08:16:31 +01:00
|
|
|
LRESULT win32_menu_loop(HWND owner, WPARAM wparam)
|
|
|
|
{
|
|
|
|
WPARAM mode = wparam & 0xffff;
|
|
|
|
enum event_command cmd = EVENT_CMD_NONE;
|
|
|
|
bool do_wm_close = false;
|
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case ID_M_LOAD_CORE:
|
|
|
|
case ID_M_LOAD_CONTENT:
|
|
|
|
{
|
|
|
|
char win32_file[PATH_MAX_LENGTH] = {0};
|
|
|
|
const char *extensions = NULL;
|
|
|
|
const char *title = NULL;
|
|
|
|
const char *initial_dir = NULL;
|
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case ID_M_LOAD_CORE:
|
|
|
|
extensions = "All Files\0*.*\0 Libretro core(.dll)\0*.dll\0";
|
2015-11-19 08:59:45 +01:00
|
|
|
#ifdef HAVE_MENU
|
|
|
|
title = menu_hash_to_str(MENU_LABEL_VALUE_CORE_LIST);
|
|
|
|
#else
|
2015-11-19 08:16:31 +01:00
|
|
|
title = "Load Core";
|
2015-11-19 08:59:45 +01:00
|
|
|
#endif
|
2015-11-19 08:16:31 +01:00
|
|
|
initial_dir = settings->libretro_directory;
|
|
|
|
cmd = EVENT_CMD_LOAD_CORE;
|
|
|
|
break;
|
|
|
|
case ID_M_LOAD_CONTENT:
|
|
|
|
extensions = "All Files\0*.*\0\0";
|
2015-11-19 08:59:45 +01:00
|
|
|
#ifdef HAVE_MENU
|
|
|
|
title = menu_hash_to_str(MENU_LABEL_VALUE_LOAD_CONTENT_LIST);
|
|
|
|
#else
|
2015-11-19 08:16:31 +01:00
|
|
|
title = "Load Content";
|
2015-11-19 08:59:45 +01:00
|
|
|
#endif
|
2015-11-19 08:16:31 +01:00
|
|
|
initial_dir = settings->menu_content_directory;
|
|
|
|
cmd = EVENT_CMD_LOAD_CONTENT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (win32_browser(owner, win32_file, extensions, title, initial_dir))
|
|
|
|
{
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case ID_M_LOAD_CORE:
|
2015-11-30 21:35:50 +01:00
|
|
|
rarch_main_ctl(RUNLOOP_CTL_SET_LIBRETRO_PATH, win32_file);
|
2015-11-19 08:16:31 +01:00
|
|
|
break;
|
|
|
|
case ID_M_LOAD_CONTENT:
|
2015-11-30 21:35:50 +01:00
|
|
|
rarch_main_ctl(RUNLOOP_CTL_SET_CONTENT_PATH, win32_file);
|
2015-11-19 08:16:31 +01:00
|
|
|
do_wm_close = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ID_M_RESET:
|
|
|
|
cmd = EVENT_CMD_RESET;
|
|
|
|
break;
|
|
|
|
case ID_M_MUTE_TOGGLE:
|
|
|
|
cmd = EVENT_CMD_AUDIO_MUTE_TOGGLE;
|
|
|
|
break;
|
|
|
|
case ID_M_MENU_TOGGLE:
|
|
|
|
cmd = EVENT_CMD_MENU_TOGGLE;
|
|
|
|
break;
|
|
|
|
case ID_M_PAUSE_TOGGLE:
|
|
|
|
cmd = EVENT_CMD_PAUSE_TOGGLE;
|
|
|
|
break;
|
|
|
|
case ID_M_LOAD_STATE:
|
|
|
|
cmd = EVENT_CMD_LOAD_STATE;
|
|
|
|
break;
|
|
|
|
case ID_M_SAVE_STATE:
|
|
|
|
cmd = EVENT_CMD_SAVE_STATE;
|
|
|
|
break;
|
|
|
|
case ID_M_DISK_CYCLE:
|
|
|
|
cmd = EVENT_CMD_DISK_EJECT_TOGGLE;
|
|
|
|
break;
|
|
|
|
case ID_M_DISK_NEXT:
|
|
|
|
cmd = EVENT_CMD_DISK_NEXT;
|
|
|
|
break;
|
|
|
|
case ID_M_DISK_PREV:
|
|
|
|
cmd = EVENT_CMD_DISK_PREV;
|
|
|
|
break;
|
|
|
|
case ID_M_FULL_SCREEN:
|
|
|
|
cmd = EVENT_CMD_FULLSCREEN_TOGGLE;
|
|
|
|
break;
|
|
|
|
#ifndef _XBOX
|
|
|
|
case ID_M_SHADER_PARAMETERS:
|
|
|
|
shader_dlg_show(owner);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case ID_M_MOUSE_GRAB:
|
|
|
|
cmd = EVENT_CMD_GRAB_MOUSE_TOGGLE;
|
|
|
|
break;
|
|
|
|
case ID_M_TAKE_SCREENSHOT:
|
|
|
|
cmd = EVENT_CMD_TAKE_SCREENSHOT;
|
|
|
|
break;
|
|
|
|
case ID_M_QUIT:
|
|
|
|
do_wm_close = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (mode >= ID_M_WINDOW_SCALE_1X && mode <= ID_M_WINDOW_SCALE_10X)
|
|
|
|
{
|
|
|
|
unsigned idx = (mode - (ID_M_WINDOW_SCALE_1X-1));
|
2015-11-30 21:35:50 +01:00
|
|
|
rarch_main_ctl(RUNLOOP_CTL_SET_WINDOWED_SCALE, &idx);
|
2015-11-19 08:16:31 +01:00
|
|
|
cmd = EVENT_CMD_RESIZE_WINDOWED_SCALE;
|
|
|
|
}
|
|
|
|
else if (mode == ID_M_STATE_INDEX_AUTO)
|
|
|
|
{
|
|
|
|
signed idx = -1;
|
|
|
|
settings->state_slot = idx;
|
|
|
|
}
|
|
|
|
else if (mode >= (ID_M_STATE_INDEX_AUTO+1) && mode <= (ID_M_STATE_INDEX_AUTO+10))
|
|
|
|
{
|
|
|
|
signed idx = (mode - (ID_M_STATE_INDEX_AUTO+1));
|
|
|
|
settings->state_slot = idx;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cmd != EVENT_CMD_NONE)
|
|
|
|
event_command(cmd);
|
|
|
|
|
|
|
|
if (do_wm_close)
|
|
|
|
PostMessage(owner, WM_CLOSE, 0, 0);
|
|
|
|
|
|
|
|
return 0L;
|
|
|
|
}
|
|
|
|
|
2015-11-19 07:23:41 +01:00
|
|
|
static void ui_companion_win32_deinit(void *data)
|
|
|
|
{
|
|
|
|
ui_companion_win32_t *handle = (ui_companion_win32_t*)data;
|
|
|
|
|
|
|
|
if (handle)
|
|
|
|
free(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *ui_companion_win32_init(void)
|
|
|
|
{
|
|
|
|
ui_companion_win32_t *handle = (ui_companion_win32_t*)calloc(1, sizeof(*handle));
|
|
|
|
|
|
|
|
if (!handle)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ui_companion_win32_iterate(void *data, unsigned action)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)action;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ui_companion_win32_notify_content_loaded(void *data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ui_companion_win32_toggle(void *data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ui_companion_win32_event_command(void *data, enum event_command cmd)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ui_companion_win32_notify_list_pushed(void *data,
|
|
|
|
file_list_t *list, file_list_t *menu_list)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)list;
|
|
|
|
(void)menu_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ui_companion_driver_t ui_companion_win32 = {
|
|
|
|
ui_companion_win32_init,
|
|
|
|
ui_companion_win32_deinit,
|
|
|
|
ui_companion_win32_iterate,
|
|
|
|
ui_companion_win32_toggle,
|
|
|
|
ui_companion_win32_event_command,
|
|
|
|
ui_companion_win32_notify_content_loaded,
|
|
|
|
ui_companion_win32_notify_list_pushed,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
"win32",
|
|
|
|
};
|