mirror of https://github.com/bsnes-emu/bsnes.git
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:
parent
b671e49644
commit
697f23d45c
|
@ -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;
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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>
|
|
@ -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
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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:;
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Binary file not shown.
After Width: | Height: | Size: 22 KiB |
|
@ -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;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
};
|
};
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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"
|
|
@ -0,0 +1,2 @@
|
||||||
|
#include "../base.hpp"
|
||||||
|
#include "video.cpp"
|
|
@ -0,0 +1 @@
|
||||||
|
#include "video.hpp"
|
|
@ -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();
|
||||||
|
}
|
|
@ -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;
|
|
@ -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();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
struct Utility {
|
||||||
|
void setTitle(const char *text);
|
||||||
|
|
||||||
|
void loadCartridgeNormal();
|
||||||
|
};
|
||||||
|
|
||||||
|
extern Utility utility;
|
Loading…
Reference in New Issue