Update to v068r19 release.

byuu says:

This adds proper manifest files to get the nice XP/Vista controls. Need
to find a way to auto-detect MinGW 32 vs 64 since I can't use $shell or
`` on gcc -v on Windows. For now you have to edit the
ui-phoenix/Makefile by hand.

I've implemented the video settings window. I am going to be using
separate windows this time. As nice as having everything in one place
was, I didn't like being forced to stretch things to fill out the
one-size-fits-all tab window I was using before. That and I don't feel
like implementing tab support with phoenix anyway.

The menu gets a load cartridge command, and bsnes writes save RAM files
now. Loading by file dialog crashes on 64-bit. Something's fucked up
there, but I don't know what. Again, help would be great here :)
This commit is contained in:
Tim Allen 2010-10-20 22:48:46 +11:00
parent b671e49644
commit 697f23d45c
31 changed files with 399 additions and 46 deletions

View File

@ -1,7 +1,7 @@
bool Font::create(const char *name, unsigned size, Font::Style style) { bool Font::create(const char *name, unsigned size, Font::Style style) {
font->font = pango_font_description_new(); font->font = pango_font_description_new();
pango_font_description_set_family(font->font, name); pango_font_description_set_family(font->font, name);
pango_font_description_set_absolute_size(font->font, size * PANGO_SCALE); pango_font_description_set_size(font->font, size * PANGO_SCALE);
pango_font_description_set_style(font->font, (style & Style::Italic) == Style::Italic ? PANGO_STYLE_OBLIQUE : PANGO_STYLE_NORMAL); pango_font_description_set_style(font->font, (style & Style::Italic) == Style::Italic ? PANGO_STYLE_OBLIQUE : PANGO_STYLE_NORMAL);
pango_font_description_set_weight(font->font, (style & Style::Bold) == Style::Bold ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL); pango_font_description_set_weight(font->font, (style & Style::Bold) == Style::Bold ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL);
return true; return true;

View File

@ -83,6 +83,7 @@ struct Window : Widget {
nall::function<bool ()> onClose; nall::function<bool ()> onClose;
void create(unsigned x, unsigned y, unsigned width, unsigned height, const char *text = ""); void create(unsigned x, unsigned y, unsigned width, unsigned height, const char *text = "");
void setGeometry(unsigned x, unsigned y, unsigned width, unsigned height); void setGeometry(unsigned x, unsigned y, unsigned width, unsigned height);
void setDefaultFont(Font &font);
void setFont(Font &font); void setFont(Font &font);
void setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue); void setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue);
void setTitle(const char *text); void setTitle(const char *text);
@ -148,6 +149,7 @@ struct HorizontalSlider : Widget {
struct Label : Widget { struct Label : Widget {
void create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text = ""); void create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text = "");
void setText(const char *text);
}; };
struct ListBox : Widget { struct ListBox : Widget {

View File

@ -1,7 +1,12 @@
void Label::create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text) { void Label::create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text) {
object->widget = gtk_label_new(text); object->widget = gtk_label_new(text);
gtk_misc_set_alignment(GTK_MISC(object->widget), 0.0, 0.0); gtk_misc_set_alignment(GTK_MISC(object->widget), 0.0, 0.5);
gtk_widget_set_size_request(object->widget, width, height); gtk_widget_set_size_request(object->widget, width, height);
if(parent.window->defaultFont) setFont(*parent.window->defaultFont);
gtk_fixed_put(GTK_FIXED(parent.object->formContainer), object->widget, x, y); gtk_fixed_put(GTK_FIXED(parent.object->formContainer), object->widget, x, y);
gtk_widget_show(object->widget); gtk_widget_show(object->widget);
} }
void Label::setText(const char *text) {
gtk_label_set_text(GTK_LABEL(object->widget), text);
}

View File

@ -40,11 +40,14 @@ void Window::setGeometry(unsigned x, unsigned y, unsigned width, unsigned height
gtk_widget_set_size_request(object->formContainer, width, height); gtk_widget_set_size_request(object->formContainer, width, height);
} }
void Window::setFont(Font &font) { void Window::setDefaultFont(Font &font) {
Widget::setFont(font);
window->defaultFont = &font; window->defaultFont = &font;
} }
void Window::setFont(Font &font) {
Widget_setFont(object->status, font.font->font);
}
void Window::setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue) { void Window::setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue) {
GdkColor color; GdkColor color;
color.pixel = (red << 16) | (green << 8) | (blue << 0); color.pixel = (red << 16) | (green << 8) | (blue << 0);

View File

@ -1,6 +1,6 @@
void Label::create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text) { void Label::create(Window &parent, unsigned x, unsigned y, unsigned width, unsigned height, const char *text) {
widget->window = CreateWindow( widget->window = CreateWindow(
L"STATIC", L"", L"phoenix_label", L"",
WS_CHILD | WS_VISIBLE, WS_CHILD | WS_VISIBLE,
x, y, width, height, x, y, width, height,
parent.widget->window, (HMENU)object->id, GetModuleHandle(0), 0 parent.widget->window, (HMENU)object->id, GetModuleHandle(0), 0
@ -13,3 +13,53 @@ void Label::create(Window &parent, unsigned x, unsigned y, unsigned width, unsig
void Label::setText(const char *text) { void Label::setText(const char *text) {
SetWindowText(widget->window, utf16_t(text)); SetWindowText(widget->window, utf16_t(text));
} }
//all of this for want of a STATIC SS_VCENTER flag ...
LRESULT CALLBACK Label_WindowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) {
Window *window_ptr = (Window*)GetWindowLongPtr(GetParent(hwnd), GWLP_USERDATA);
if(!window_ptr) return DefWindowProc(hwnd, msg, wparam, lparam);
Label *label_ptr = (Label*)GetWindowLongPtr(hwnd, GWLP_USERDATA);
if(!label_ptr) return DefWindowProc(hwnd, msg, wparam, lparam);
Window &window = *window_ptr;
Label &label = *label_ptr;
switch(msg) {
case WM_ERASEBKGND: {
if(window.window->brush == 0) break;
RECT rc;
GetClientRect(window.widget->window, &rc);
PAINTSTRUCT ps;
BeginPaint(window.widget->window, &ps);
FillRect(ps.hdc, &rc, window.window->brush);
EndPaint(window.widget->window, &ps);
return TRUE;
}
case WM_PAINT: {
PAINTSTRUCT ps;
BeginPaint(hwnd, &ps);
SelectObject(ps.hdc, label.widget->font);
if(window.window->brush) {
SetBkColor(ps.hdc, window.window->brushColor);
} else {
SetBkColor(ps.hdc, GetSysColor(COLOR_3DFACE));
}
RECT rc;
GetClientRect(hwnd, &rc);
unsigned length = GetWindowTextLength(hwnd) + 1;
wchar_t text[length];
GetWindowText(hwnd, text, length + 1);
text[length] = 0;
DrawText(ps.hdc, text, -1, &rc, DT_CALCRECT | DT_END_ELLIPSIS);
unsigned height = rc.bottom;
GetClientRect(hwnd, &rc);
rc.top = (rc.bottom - height) / 2;
rc.bottom = rc.top + height;
DrawText(ps.hdc, text, -1, &rc, DT_LEFT | DT_END_ELLIPSIS);
EndPaint(hwnd, &ps);
InvalidateRect(hwnd, 0, false);
}
}
return DefWindowProc(hwnd, msg, wparam, lparam);
}

View File

@ -16,6 +16,7 @@ struct Action::Data {
struct Widget::Data { struct Widget::Data {
HWND window; HWND window;
HFONT font;
}; };
struct Window::Data { struct Window::Data {

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
<assemblyIdentity type="win32" name="phoenix" version="1.0.0.0" processorArchitecture="AMD64"/>
<dependency>
<dependentAssembly>
<assemblyIdentity type="win32" name="Microsoft.Windows.Common-Controls" version="6.0.0.0" processorArchitecture="AMD64" publicKeyToken="6595b64144ccf1df" language="*"/>
</dependentAssembly>
</dependency>
</assembly>

View File

@ -1 +1,7 @@
1 24 "phoenix.Manifest" #ifdef ARCH_X86
1 24 "phoenix-x86.Manifest"
#endif
#ifdef ARCH_AMD64
1 24 "phoenix-amd64.Manifest"
#endif

View File

@ -1,4 +1,5 @@
void Widget::setFont(Font &font) { void Widget::setFont(Font &font) {
widget->font = font.font->font;
SendMessage(widget->window, WM_SETFONT, (WPARAM)font.font->font, 0); SendMessage(widget->window, WM_SETFONT, (WPARAM)font.font->font, 0);
} }
@ -31,4 +32,5 @@ Widget::Widget() {
os.objects.append(this); os.objects.append(this);
widget = new Widget::Data; widget = new Widget::Data;
widget->window = 0; widget->window = 0;
widget->font = os.os->proportionalFont;
} }

View File

@ -18,8 +18,11 @@ void Window::create(unsigned x, unsigned y, unsigned width, unsigned height, con
SetWindowLongPtr(widget->window, GWLP_USERDATA, (LONG_PTR)this); SetWindowLongPtr(widget->window, GWLP_USERDATA, (LONG_PTR)this);
} }
void Window::setFont(Font &font) { void Window::setDefaultFont(Font &font) {
window->defaultFont = font.font->font; window->defaultFont = font.font->font;
}
void Window::setFont(Font &font) {
SendMessage(window->status, WM_SETFONT, (WPARAM)window->defaultFont, 0); SendMessage(window->status, WM_SETFONT, (WPARAM)window->defaultFont, 0);
} }

View File

@ -214,7 +214,7 @@ static LRESULT CALLBACK OS_windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM
FillRect(ps.hdc, &rc, window.window->brush); FillRect(ps.hdc, &rc, window.window->brush);
EndPaint(window.widget->window, &ps); EndPaint(window.widget->window, &ps);
return TRUE; return TRUE;
}; }
case WM_CTLCOLORBTN: case WM_CTLCOLORBTN:
case WM_CTLCOLORSTATIC: { case WM_CTLCOLORSTATIC: {
@ -378,13 +378,25 @@ OS::OS() {
wc.cbWndExtra = 0; wc.cbWndExtra = 0;
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1); wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
wc.hCursor = LoadCursor(0, IDC_ARROW); wc.hCursor = LoadCursor(0, IDC_ARROW);
wc.hIcon = LoadIcon(0, IDI_APPLICATION); wc.hIcon = LoadIcon(GetModuleHandle(0), MAKEINTRESOURCE(2));
wc.hInstance = GetModuleHandle(0); wc.hInstance = GetModuleHandle(0);
wc.lpfnWndProc = OS_windowProc; wc.lpfnWndProc = OS_windowProc;
wc.lpszClassName = L"phoenix_window"; wc.lpszClassName = L"phoenix_window";
wc.lpszMenuName = 0; wc.lpszMenuName = 0;
wc.style = CS_HREDRAW | CS_VREDRAW; wc.style = CS_HREDRAW | CS_VREDRAW;
RegisterClass(&wc); RegisterClass(&wc);
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
wc.hCursor = LoadCursor(0, IDC_ARROW);
wc.hIcon = LoadIcon(0, IDI_APPLICATION);
wc.hInstance = GetModuleHandle(0);
wc.lpfnWndProc = Label_WindowProc;
wc.lpszClassName = L"phoenix_label";
wc.lpszMenuName = 0;
wc.style = CS_HREDRAW | CS_VREDRAW;
RegisterClass(&wc);
} }
} }

View File

@ -94,6 +94,7 @@ struct Window : Widget {
static Window None; static Window None;
nall::function<bool ()> onClose; nall::function<bool ()> onClose;
void create(unsigned x, unsigned y, unsigned width, unsigned height, const char *text = ""); void create(unsigned x, unsigned y, unsigned width, unsigned height, const char *text = "");
void setDefaultFont(Font &font);
void setFont(Font &font); void setFont(Font &font);
void setGeometry(unsigned x, unsigned y, unsigned width, unsigned height); void setGeometry(unsigned x, unsigned y, unsigned width, unsigned height);
void setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue); void setBackgroundColor(uint8_t red, uint8_t green, uint8_t blue);

View File

@ -1,7 +1,7 @@
namespace SNES { namespace SNES {
namespace Info { namespace Info {
static const char Name[] = "bsnes"; static const char Name[] = "bsnes";
static const char Version[] = "068.18"; static const char Version[] = "068.19";
static const unsigned SerializerVersion = 13; static const unsigned SerializerVersion = 13;
} }
} }

View File

@ -1,5 +1,6 @@
ui_objects := ui-main ui-general ui-cartridge ui_objects := ui-main ui-general ui-settings ui-utility ui-cartridge
ui_objects += ruby phoenix ui_objects += ruby phoenix
ui_objects += $(if $(call streq,$(platform),win),resource)
# platform # platform
ifeq ($(platform),x) ifeq ($(platform),x)
@ -18,6 +19,7 @@ else ifeq ($(platform),osx)
link += $(if $(findstring audio.openal,$(ruby)),-framework OpenAL) link += $(if $(findstring audio.openal,$(ruby)),-framework OpenAL)
else ifeq ($(platform),win) else ifeq ($(platform),win)
arch := -DARCH_X86
flags += -DPHOENIX_WINDOWS flags += -DPHOENIX_WINDOWS
link += link +=
@ -51,13 +53,20 @@ rubydef := $(foreach c,$(subst .,_,$(call strupper,$(ruby))),-D$c)
# rules # rules
objects := $(ui_objects) $(objects) objects := $(ui_objects) $(objects)
obj/ui-main.o: $(ui)/main.cpp $(call rwildcard,$(ui)/*) obj/ui-main.o: $(ui)/main.cpp $(call rwildcard,$(ui)/*.hpp) $(call rwildcard,$(ui)/*)
obj/ui-general.o: $(ui)/general/general.cpp $(call rwildcard,$(ui)/general/*) obj/ui-general.o: $(ui)/general/general.cpp $(call rwildcard,$(ui)/*.hpp) $(call rwildcard,$(ui)/general/*)
obj/ui-cartridge.o: $(ui)/cartridge/cartridge.cpp $(call rwildcard,$(ui)/cartridge/*) obj/ui-settings.o: $(ui)/settings/settings.cpp $(call rwildcard,$(ui)/*.hpp) $(call rwildcard,$(ui)/settings/*)
obj/ui-utility.o: $(ui)/utility/utility.cpp $(call rwildcard,$(ui)/*.hpp) $(call rwildcard,$(ui)/utility/*)
obj/ui-cartridge.o: $(ui)/cartridge/cartridge.cpp $(call rwildcard,$(ui)/*.hpp) $(call rwildcard,$(ui)/cartridge/*)
obj/ruby.o: ruby/ruby.cpp $(call rwildcard,ruby/*) obj/ruby.o: ruby/ruby.cpp $(call rwildcard,ruby/*)
$(call compile,$(rubydef) $(rubyflags)) $(call compile,$(rubydef) $(rubyflags))
obj/phoenix.o: phoenix/phoenix.cpp $(call rwildcard,phoenix/*) obj/phoenix.o: phoenix/phoenix.cpp $(call rwildcard,phoenix/*)
obj/resource.o: $(ui)/resource.rc
windres $(ui)/resource.rc obj/resource.o $(arch)
# targets # targets
ui_build:; ui_build:;

View File

@ -15,11 +15,26 @@ using namespace phoenix;
#include "interface.hpp" #include "interface.hpp"
#include "general/general.hpp" #include "general/general.hpp"
#include "settings/settings.hpp"
#include "utility/utility.hpp"
#include "cartridge/cartridge.hpp" #include "cartridge/cartridge.hpp"
struct Application { struct Application {
Font font;
bool quit; bool quit;
void main(int argc, char **argv); void main(int argc, char **argv);
}; };
extern Application application; extern Application application;
struct Style {
enum : unsigned {
#if defined(PHOENIX_WINDOWS)
SliderHeight = 25,
#elif defined(PHOENIX_GTK)
SliderHeight = 22,
#elif defined(PHOENIX_QT)
SliderHeight = 22,
#endif
};
};

View File

@ -1,9 +1,25 @@
#include "../base.hpp" #include "../base.hpp"
Cartridge cartridge; Cartridge cartridge;
bool Cartridge::loadNormal(string filename) { bool Cartridge::loadNormal(const char *filename) {
if(file::exists(filename) == false) return false; unload();
if(loadCartridge(SNES::memory::cartrom, baseXML, baseName = filename) == false) return false;
SNES::cartridge.load(SNES::Cartridge::Mode::Normal, lstring() << baseXML);
loadMemory(SNES::memory::cartram, baseName, ".srm");
loadMemory(SNES::memory::cartrtc, baseName, ".rtc");
utility.setTitle(notdir(nall::basename(baseName)));
return true;
}
void Cartridge::unload() {
if(SNES::cartridge.loaded() == false) return;
saveMemory(SNES::memory::cartram, baseName, ".srm");
saveMemory(SNES::memory::cartrtc, baseName, ".rtc");
SNES::cartridge.unload();
}
bool Cartridge::loadCartridge(SNES::MappedRAM &memory, string &XML, const char *filename) {
if(file::exists(filename) == false) return false;
file fp; file fp;
if(fp.open(filename, file::mode_read) == false) return false; if(fp.open(filename, file::mode_read) == false) return false;
@ -16,19 +32,30 @@ bool Cartridge::loadNormal(string filename) {
fp.read(data, size); fp.read(data, size);
fp.close(); fp.close();
SNES::memory::cartrom.copy(data, size); memory.copy(data, size);
XML = snes_information(data, size).xml_memory_map;
string baseXML = snes_information(data, size).xml_memory_map;
SNES::cartridge.load(SNES::Cartridge::Mode::Normal, lstring() << baseXML);
delete[] data; delete[] data;
return true;
filename = string(nall::basename(filename), ".srm"); }
if(SNES::memory::cartram.size() && file::exists(filename)) {
if(fp.open(filename, file::mode_read)) { bool Cartridge::loadMemory(SNES::MappedRAM &memory, string filename, const char *extension) {
fp.read(SNES::memory::cartram.data(), min(fp.size(), SNES::memory::cartram.size())); if(memory.size() == 0 || memory.size() == ~0) return true;
fp.close(); filename = string(nall::basename(filename), extension);
} if(file::exists(filename) == false) return false;
} file fp;
if(fp.open(filename, file::mode_read)) {
fp.read(memory.data(), min(memory.size(), fp.size()));
fp.close();
}
return true;
}
bool Cartridge::saveMemory(SNES::MappedRAM &memory, string filename, const char *extension) {
if(memory.size() == 0 || memory.size() == ~0) return true;
filename = string(nall::basename(filename), extension);
file fp;
if(fp.open(filename, file::mode_write) == false) return false;
fp.write(memory.data(), memory.size());
fp.close();
return true; return true;
} }

View File

@ -1,5 +1,14 @@
struct Cartridge { struct Cartridge {
bool loadNormal(string filename); bool loadNormal(const char *filename);
void unload();
private:
string baseName, slotAName, slotBName;
string baseXML, slotAXML, slotBXML;
bool loadCartridge(SNES::MappedRAM &memory, string &XML, const char *filename);
bool loadMemory(SNES::MappedRAM &memory, string filename, const char *extension);
bool saveMemory(SNES::MappedRAM &memory, string filename, const char *extension);
}; };
extern Cartridge cartridge; extern Cartridge cartridge;

BIN
bsnes/ui-phoenix/data/bsnes.ico Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

View File

@ -1,19 +1,39 @@
MainWindow mainWindow; MainWindow mainWindow;
void MainWindow::create() { void MainWindow::create() {
Window::create(256, 256, 595, 448, string(SNES::Info::Name, " v", SNES::Info::Version)); statusFont.create("Sans", 8, Font::Style::Bold);
//setBackgroundColor(0, 0, 0); Window::create(120, 120, 595, 448, string(SNES::Info::Name, " v", SNES::Info::Version));
setDefaultFont(application.font);
setFont(statusFont);
setBackgroundColor(0, 0, 0);
system.create(*this, "System"); system.create(*this, "System");
systemLoadCartridge.create(system, "Load Cartridge ...");
systemSeparator.create(system);
systemQuit.create(system, "Quit"); systemQuit.create(system, "Quit");
systemQuit.onTick = []() { application.quit = true; };
setMenuVisible(true); setMenuVisible(true);
settings.create(*this, "Settings"); settings.create(*this, "Settings");
settingsVideo.create(settings, "Video Settings ...");
tools.create(*this, "Tools"); tools.create(*this, "Tools");
help.create(*this, "Help"); help.create(*this, "Help");
viewport.create(*this, 0, 0, 595, 448); viewport.create(*this, 0, 0, 595, 448);
setStatusVisible(true); setStatusVisible(true);
setVisible(true);
onClose = []() { application.quit = true; return false; }; systemLoadCartridge.onTick = []() {
utility.loadCartridgeNormal();
};
systemQuit.onTick = []() {
application.quit = true;
};
settingsVideo.onTick = []() {
videoSettingsWindow.setVisible();
};
onClose = []() {
application.quit = true;
return false;
};
} }

View File

@ -1,10 +1,13 @@
struct MainWindow : Window { struct MainWindow : Window {
Font statusFont;
Menu system; Menu system;
MenuItem systemLoadCartridge;
MenuSeparator systemSeparator;
MenuItem systemQuit; MenuItem systemQuit;
Menu settings; Menu settings;
MenuItem settingsVideo;
Menu tools; Menu tools;
Menu help; Menu help;
Viewport viewport; Viewport viewport;
void create(); void create();

View File

@ -8,19 +8,66 @@ const uint8_t Palette::gammaRamp[32] = {
0xc8, 0xd0, 0xd8, 0xe0, 0xe8, 0xf0, 0xf8, 0xff, 0xc8, 0xd0, 0xd8, 0xe0, 0xe8, 0xf0, 0xf8, 0xff,
}; };
uint8_t Palette::contrastAdjust(uint8_t input) {
signed contrast = Palette::contrast - 100;
signed result = input - contrast + (2 * contrast * input + 127) / 255;
return max(0, min(255, result));
}
uint8_t Palette::brightnessAdjust(uint8_t input) {
signed brightness = Palette::brightness - 100;
signed result = input + brightness;
return max(0, min(255, result));
}
uint8_t Palette::gammaAdjust(uint8_t input) {
signed result = (signed)(pow(((double)input / 255.0), (double)gamma / 100.0) * 255.0 + 0.5);
return max(0, min(255, result));
}
void Palette::update() { void Palette::update() {
for(unsigned i = 0; i < 32768; i++) { for(unsigned i = 0; i < 32768; i++) {
unsigned r = gammaRamp[(i >> 10) & 31]; unsigned r = (i >> 10) & 31;
unsigned g = gammaRamp[(i >> 5) & 31]; unsigned g = (i >> 5) & 31;
unsigned b = gammaRamp[(i >> 0) & 31]; unsigned b = (i >> 0) & 31;
//r = (r << 3) | (r >> 2);
//g = (g << 3) | (g >> 2); r = (r << 3) | (r >> 2);
//b = (b << 3) | (b >> 2); g = (g << 3) | (g >> 2);
b = (b << 3) | (b >> 2);
if(useGammaRamp) {
r = gammaRamp[r >> 3];
g = gammaRamp[g >> 3];
b = gammaRamp[b >> 3];
}
if(contrast != 100) {
r = contrastAdjust(r);
g = contrastAdjust(g);
b = contrastAdjust(b);
}
if(brightness != 100) {
r = brightnessAdjust(r);
g = brightnessAdjust(g);
b = brightnessAdjust(b);
}
if(gamma != 100) {
r = gammaAdjust(r);
g = gammaAdjust(g);
b = gammaAdjust(b);
}
color[i] = (r << 16) | (g << 8) | (b << 0); color[i] = (r << 16) | (g << 8) | (b << 0);
} }
} }
Palette::Palette() { Palette::Palette() {
contrast = 100;
brightness = 100;
gamma = 100;
useGammaRamp = true;
update(); update();
} }

View File

@ -1,6 +1,13 @@
struct Palette { struct Palette {
static const uint8_t gammaRamp[32]; static const uint8_t gammaRamp[32];
uint32_t color[32768]; uint32_t color[32768];
unsigned contrast;
unsigned brightness;
unsigned gamma;
bool useGammaRamp;
uint8_t contrastAdjust(uint8_t);
uint8_t brightnessAdjust(uint8_t);
uint8_t gammaAdjust(uint8_t);
void update(); void update();
Palette(); Palette();
}; };

View File

@ -2,11 +2,29 @@
#include "interface.cpp" #include "interface.cpp"
Application application; Application application;
#if defined(PLATFORM_WIN)
static string VideoDriver = "Direct3D";
static string AudioDriver = "XAudio2";
static string InputDriver = "RawInput";
#elif defined(PLATFORM_X)
static string VideoDriver = "OpenGL";
static string AudioDriver = "ALSA";
static string InputDriver = "SDL";
#endif
void Application::main(int argc, char **argv) { void Application::main(int argc, char **argv) {
#if defined(PHOENIX_WINDOWS)
font.create("Tahoma", 8);
#else
font.create("Sans", 8);
#endif
mainWindow.create(); mainWindow.create();
videoSettingsWindow.create();
mainWindow.setVisible();
while(os.pending()) os.run(); while(os.pending()) os.run();
video.driver("OpenGL"); video.driver(VideoDriver);
video.set(Video::Handle, mainWindow.viewport.handle()); video.set(Video::Handle, mainWindow.viewport.handle());
video.set(Video::Synchronize, false); video.set(Video::Synchronize, false);
video.set(Video::Filter, (unsigned)Video::FilterLinear); video.set(Video::Filter, (unsigned)Video::FilterLinear);
@ -16,7 +34,7 @@ void Application::main(int argc, char **argv) {
video.init(); video.init();
} }
audio.driver("ALSA"); audio.driver(AudioDriver);
audio.set(Audio::Handle, mainWindow.viewport.handle()); audio.set(Audio::Handle, mainWindow.viewport.handle());
audio.set(Audio::Synchronize, false); audio.set(Audio::Synchronize, false);
audio.set(Audio::Frequency, (unsigned)32000); audio.set(Audio::Frequency, (unsigned)32000);
@ -26,7 +44,7 @@ void Application::main(int argc, char **argv) {
audio.init(); audio.init();
} }
input.driver("SDL"); input.driver(InputDriver);
input.set(Input::Handle, mainWindow.viewport.handle()); input.set(Input::Handle, mainWindow.viewport.handle());
if(input.init() == false) { if(input.init() == false) {
MessageWindow::critical(mainWindow, "Failed to initialize input."); MessageWindow::critical(mainWindow, "Failed to initialize input.");
@ -41,7 +59,7 @@ void Application::main(int argc, char **argv) {
} }
while(quit == false) { while(quit == false) {
if(os.pending()) os.run(); while(os.pending()) os.run();
if(SNES::cartridge.loaded()) { if(SNES::cartridge.loaded()) {
SNES::system.run(); SNES::system.run();
@ -50,6 +68,7 @@ void Application::main(int argc, char **argv) {
} }
} }
cartridge.unload();
mainWindow.setVisible(false); mainWindow.setVisible(false);
while(os.pending()) os.run(); while(os.pending()) os.run();
SNES::system.term(); SNES::system.term();

9
bsnes/ui-phoenix/resource.rc Executable file
View File

@ -0,0 +1,9 @@
#ifdef ARCH_X86
1 24 "../phoenix/windows/phoenix-x86.Manifest"
#endif
#ifdef ARCH_AMD64
1 24 "../phoenix/windows/phoenix-amd64.Manifest"
#endif
2 ICON DISCARDABLE "data/bsnes.ico"

View File

@ -0,0 +1,2 @@
#include "../base.hpp"
#include "video.cpp"

View File

@ -0,0 +1 @@
#include "video.hpp"

View File

@ -0,0 +1,49 @@
VideoSettingsWindow videoSettingsWindow;
void VideoSettingsWindow::create() {
Window::create(0, 0, 256, 256, "Video Settings");
setDefaultFont(application.font);
unsigned x = 5, y = 5;
contrastLabel.create (*this, x, y, 80, Style::SliderHeight, "Contrast:");
contrastValue.create (*this, x + 80, y, 50, Style::SliderHeight, "100%");
contrastSlider.create (*this, x + 130, y, 300, Style::SliderHeight, 201); y += Style::SliderHeight;
brightnessLabel.create (*this, x, y, 80, Style::SliderHeight, "Brightness:");
brightnessValue.create (*this, x + 80, y, 40, Style::SliderHeight, "100%");
brightnessSlider.create(*this, x + 130, y, 300, Style::SliderHeight, 201); y += Style::SliderHeight;
gammaLabel.create (*this, x, y, 80, Style::SliderHeight, "Gamma:");
gammaValue.create (*this, x + 80, y, 50, Style::SliderHeight, "100%");
gammaSlider.create (*this, x + 130, y, 300, Style::SliderHeight, 201); y += Style::SliderHeight + 5;
gammaRampCheck.create (*this, x, y, 430, 15, "Enable NTSC gamma ramp simulation"); y += 15;
setGeometry(0, 0, 440, y + 5);
contrastSlider.setPosition(100);
brightnessSlider.setPosition(100);
gammaSlider.setPosition(100);
gammaRampCheck.setChecked();
contrastSlider.onChange = brightnessSlider.onChange = gammaSlider.onChange = gammaRampCheck.onTick =
{ &VideoSettingsWindow::adjust, this };
onClose = []() {
videoSettingsWindow.setVisible(false);
return false;
};
}
void VideoSettingsWindow::adjust() {
contrastValue.setText(string(contrastSlider.position(), "%"));
brightnessValue.setText(string(brightnessSlider.position(), "%"));
gammaValue.setText(string(gammaSlider.position(), "%"));
palette.contrast = contrastSlider.position();
palette.brightness = brightnessSlider.position();
palette.gamma = gammaSlider.position();
palette.useGammaRamp = gammaRampCheck.checked();
palette.update();
}

View File

@ -0,0 +1,17 @@
struct VideoSettingsWindow : Window {
Label contrastLabel;
Label contrastValue;
HorizontalSlider contrastSlider;
Label brightnessLabel;
Label brightnessValue;
HorizontalSlider brightnessSlider;
Label gammaLabel;
Label gammaValue;
HorizontalSlider gammaSlider;
CheckBox gammaRampCheck;
void create();
void adjust();
};
extern VideoSettingsWindow videoSettingsWindow;

View File

@ -0,0 +1,18 @@
#include "../base.hpp"
Utility utility;
void Utility::setTitle(const char *text) {
if(*text) {
mainWindow.setTitle(string(text, " - ", SNES::Info::Name, " v", SNES::Info::Version));
} else {
mainWindow.setTitle(string(SNES::Info::Name, " v", SNES::Info::Version));
}
}
void Utility::loadCartridgeNormal() {
string filename = os.fileOpen(mainWindow, "SNES cartridges\t*.sfc\nAll files\t*", "/media/sdb1/root/snes_roms");
if(filename != "") {
cartridge.loadNormal(filename);
SNES::system.power();
}
}

View File

@ -0,0 +1,7 @@
struct Utility {
void setTitle(const char *text);
void loadCartridgeNormal();
};
extern Utility utility;