2013-08-29 11:45:22 +09:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 yvt
|
|
|
|
|
|
|
|
This file is part of OpenSpades.
|
|
|
|
|
|
|
|
OpenSpades 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 Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
OpenSpades 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 OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-08-29 00:21:27 +02:00
|
|
|
#include <OpenSpades.h>
|
2013-08-18 16:18:06 +09:00
|
|
|
#include "MainWindow.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "../Core/Debug.h"
|
|
|
|
#include "SDLRunner.h"
|
|
|
|
#include <FL/fl_ask.H>
|
|
|
|
#include "../Core/Settings.h"
|
|
|
|
#include "../Imports/SDL.h"
|
|
|
|
#include <FL/Fl_PNG_Image.H>
|
|
|
|
#include "../Core/FileManager.h"
|
|
|
|
#include "../Core/IStream.h"
|
|
|
|
#include "SDLAsyncRunner.h"
|
|
|
|
#include "DetailConfigWindow.h"
|
|
|
|
#include "../Core/Math.h"
|
2013-08-31 18:06:46 +02:00
|
|
|
#include "Serverbrowser.h"
|
2013-09-02 01:19:14 +09:00
|
|
|
#include "ErrorDialog.h"
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-08-27 16:35:12 +02:00
|
|
|
#include "../Imports/OpenGL.h" //for gpu info
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
using namespace spades::gui;
|
|
|
|
|
|
|
|
SPADES_SETTING(cg_smp, "0");
|
|
|
|
SPADES_SETTING(cg_blood, "1");
|
|
|
|
SPADES_SETTING(cg_lastQuickConnectHost, "127.0.0.1");
|
|
|
|
SPADES_SETTING(cg_playerName, "Deuce");
|
|
|
|
SPADES_SETTING(r_bloom, "1");
|
|
|
|
SPADES_SETTING(r_lens, "1");
|
|
|
|
SPADES_SETTING(r_cameraBlur, "1");
|
|
|
|
SPADES_SETTING(r_softParticles, "1");
|
|
|
|
SPADES_SETTING(r_mapSoftShadow, "0");
|
|
|
|
SPADES_SETTING(r_modelShadows, "1");
|
|
|
|
SPADES_SETTING(r_radiosity, "0");
|
|
|
|
SPADES_SETTING(r_dlights, "1");
|
2013-09-03 21:52:32 +09:00
|
|
|
SPADES_SETTING(r_water, "2");
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_SETTING(r_multisamples, "0");
|
2013-08-22 22:39:39 +09:00
|
|
|
SPADES_SETTING(r_fxaa, "1");
|
2013-09-06 10:48:32 +09:00
|
|
|
SPADES_SETTING(r_depthBits, "");
|
2013-09-01 00:34:41 +02:00
|
|
|
SPADES_SETTING(r_colorBits, "");
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_SETTING(r_videoWidth, "1024");
|
|
|
|
SPADES_SETTING(r_videoHeight, "640");
|
|
|
|
SPADES_SETTING(r_fullscreen, "0");
|
2013-08-22 21:21:52 +09:00
|
|
|
SPADES_SETTING(r_fogShadow, "0");
|
|
|
|
SPADES_SETTING(r_lensFlare, "1");
|
2013-08-29 23:54:23 +09:00
|
|
|
SPADES_SETTING(r_blitFramebuffer, "1");
|
2013-09-04 03:44:38 +09:00
|
|
|
SPADES_SETTING(r_srgb, "");
|
2013-09-06 10:48:32 +09:00
|
|
|
SPADES_SETTING(r_shadowMapSize, "");
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_SETTING(s_maxPolyphonics, "96");
|
|
|
|
SPADES_SETTING(s_eax, "1");
|
2013-09-02 19:35:14 +09:00
|
|
|
SPADES_SETTING(r_maxAnisotropy, "8");
|
2013-09-04 15:26:38 +09:00
|
|
|
SPADES_SETTING(r_colorCorrection, "1");
|
2013-09-04 15:00:04 +09:00
|
|
|
SPADES_SETTING(r_physicalLighting, "");
|
2013-09-08 16:44:50 +09:00
|
|
|
SPADES_SETTING(r_occlusionQuery, "");
|
2013-09-12 14:39:42 +09:00
|
|
|
SPADES_SETTING(r_depthOfField, "");
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-08-28 02:20:26 +09:00
|
|
|
static std::vector<spades::IntVector3> g_modes;
|
|
|
|
|
2013-09-01 00:35:56 +02:00
|
|
|
MainWindow::~MainWindow()
|
|
|
|
{
|
|
|
|
if( browser ) {
|
|
|
|
if( browser->IsAlive() ) {
|
|
|
|
browser->stopReading();
|
|
|
|
browser->Join();
|
|
|
|
}
|
|
|
|
delete browser;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
void MainWindow::StartGame(const std::string &host) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
|
|
|
hide();
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
SDLRunner r(host);
|
|
|
|
r.Run();
|
|
|
|
#else
|
2013-09-02 01:19:14 +09:00
|
|
|
std::string err;
|
2013-08-18 16:18:06 +09:00
|
|
|
try{
|
|
|
|
if(cg_smp){
|
|
|
|
SDLAsyncRunner r(host, cg_playerName);
|
|
|
|
r.Run();
|
|
|
|
}else{
|
|
|
|
SDLRunner r(host, cg_playerName);
|
|
|
|
r.Run();
|
|
|
|
}
|
2013-08-26 20:41:29 +09:00
|
|
|
}catch(const spades::Exception& ex){
|
2013-09-02 01:19:14 +09:00
|
|
|
err = ex.GetShortMessage();
|
2013-08-26 20:41:29 +09:00
|
|
|
SPLog("Unhandled exception in SDLRunner:\n%s", ex.what());
|
2013-08-18 16:18:06 +09:00
|
|
|
}catch(const std::exception& ex){
|
2013-09-02 01:19:14 +09:00
|
|
|
err = ex.what();
|
2013-08-26 01:27:44 +09:00
|
|
|
SPLog("Unhandled exception in SDLRunner:\n%s", ex.what());
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2013-09-02 01:19:14 +09:00
|
|
|
if(!err.empty()){
|
|
|
|
ErrorDialog dlg;
|
|
|
|
dlg.set_modal();
|
|
|
|
dlg.result = 0;
|
|
|
|
|
2013-09-02 19:35:14 +09:00
|
|
|
// TODO: free this buffer (just leaking)
|
|
|
|
Fl_Text_Buffer *buf = new Fl_Text_Buffer;
|
|
|
|
buf->append(err.c_str());
|
2013-09-02 01:19:14 +09:00
|
|
|
dlg.infoView->wrap_mode(Fl_Text_Display::WRAP_AT_BOUNDS, 0);
|
|
|
|
dlg.infoView->buffer(buf);
|
|
|
|
dlg.helpView->value("See SystemMessages.log for more details.");
|
|
|
|
dlg.show();
|
|
|
|
while(dlg.visible()){
|
|
|
|
Fl::wait();
|
|
|
|
}
|
|
|
|
if( dlg.result == 1 ){
|
2013-09-02 02:34:52 +09:00
|
|
|
//show();
|
2013-09-02 01:19:14 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::QuickConnectPressed() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
|
|
|
StartGame(quickHostInput->value());
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - Setup
|
|
|
|
|
|
|
|
void MainWindow::LoadPrefs() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
2013-08-26 01:27:44 +09:00
|
|
|
SPLog("Loading Preferences to MainWindow");
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
// --- video
|
|
|
|
// add modes
|
|
|
|
char buf[64];
|
2013-08-28 02:20:26 +09:00
|
|
|
modeSelect->clear();
|
|
|
|
for(size_t i = 0; i < g_modes.size(); i++){
|
|
|
|
spades::IntVector3 mode = g_modes[i];
|
|
|
|
sprintf(buf, "%dx%d", mode.x, mode.y);
|
|
|
|
modeSelect->add(buf);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
sprintf(buf, "%dx%d", (int)r_videoWidth, (int)r_videoHeight);
|
|
|
|
modeSelect->value(buf);
|
|
|
|
|
|
|
|
msaaSelect->clear();
|
2013-08-29 23:54:23 +09:00
|
|
|
if(r_blitFramebuffer) {
|
|
|
|
msaaSelect->add("Off");
|
|
|
|
msaaSelect->add("MSAA 2x");
|
|
|
|
msaaSelect->add("MSAA 4x");
|
|
|
|
msaaSelect->add("FXAA");
|
|
|
|
msaaSelect->add("Custom");
|
|
|
|
if(r_fxaa) {
|
|
|
|
if(r_multisamples){
|
|
|
|
msaaSelect->value(4);
|
|
|
|
}else{
|
|
|
|
msaaSelect->value(3);
|
|
|
|
}
|
2013-08-22 22:39:39 +09:00
|
|
|
}else{
|
2013-08-29 23:54:23 +09:00
|
|
|
switch((int)r_multisamples){
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
default:
|
|
|
|
msaaSelect->value(0);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
msaaSelect->value(1);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
msaaSelect->value(2);
|
|
|
|
break;
|
|
|
|
}
|
2013-08-22 22:39:39 +09:00
|
|
|
}
|
2013-08-29 23:54:23 +09:00
|
|
|
}else {
|
|
|
|
// MSAA is not supported with r_blitFramebuffer = 0
|
|
|
|
msaaSelect->add("Off");
|
|
|
|
msaaSelect->add("FXAA");
|
|
|
|
msaaSelect->add("Custom");
|
|
|
|
if(r_fxaa) {
|
|
|
|
if(r_multisamples){
|
2013-08-22 22:39:39 +09:00
|
|
|
msaaSelect->value(2);
|
2013-08-29 23:54:23 +09:00
|
|
|
}else{
|
|
|
|
msaaSelect->value(1);
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
switch((int)r_multisamples){
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
default:
|
|
|
|
msaaSelect->value(0);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
case 4:
|
|
|
|
msaaSelect->value(2);
|
|
|
|
break;
|
|
|
|
}
|
2013-08-22 22:39:39 +09:00
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
quickHostInput->value(cg_lastQuickConnectHost.CString());
|
|
|
|
fullscreenCheck->value(r_fullscreen ? 1 : 0);
|
|
|
|
|
|
|
|
// --- graphics
|
2013-09-12 14:39:42 +09:00
|
|
|
postFilterSelect->clear();
|
|
|
|
postFilterSelect->add("Low");
|
|
|
|
postFilterSelect->add("Medium");
|
|
|
|
if(postFilterHighCapable){
|
|
|
|
postFilterSelect->add("High");
|
|
|
|
}
|
|
|
|
postFilterSelect->add("Custom");
|
|
|
|
if(postFilterHighCapable){
|
|
|
|
postFilterSelect->value(3);
|
|
|
|
}else{
|
|
|
|
postFilterSelect->value(2);
|
|
|
|
}
|
|
|
|
|
2013-09-04 15:26:38 +09:00
|
|
|
if(r_cameraBlur && r_bloom && r_lens && r_lensFlare &&
|
2013-09-12 14:39:42 +09:00
|
|
|
r_colorCorrection && r_depthOfField
|
|
|
|
&& postFilterHighCapable) {
|
|
|
|
postFilterSelect->value(2);
|
|
|
|
}else if(r_cameraBlur && r_bloom && r_lens && r_lensFlare &&
|
|
|
|
r_colorCorrection && (!r_depthOfField)) {
|
|
|
|
postFilterSelect->value(1);
|
2013-08-18 16:18:06 +09:00
|
|
|
}else{
|
2013-09-12 14:39:42 +09:00
|
|
|
postFilterSelect->value(0);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
softParticleCheck->value(r_softParticles ? 1 : 0);
|
|
|
|
radiosityCheck->value(r_radiosity ? 1 : 0);
|
|
|
|
bloodCheck->value(cg_blood ? 1 : 0);
|
|
|
|
|
|
|
|
directLightSelect->clear();
|
|
|
|
directLightSelect->add("Low");
|
|
|
|
directLightSelect->add("Medium");
|
|
|
|
directLightSelect->add("High");
|
2013-09-13 14:16:52 +09:00
|
|
|
directLightSelect->add("Ultra");
|
2013-08-18 16:18:06 +09:00
|
|
|
directLightSelect->add("Custom");
|
|
|
|
|
2013-09-04 15:00:04 +09:00
|
|
|
if((!r_mapSoftShadow) && (r_dlights) && (!r_modelShadows) && (!r_fogShadow) && (!r_physicalLighting)){
|
2013-08-18 16:18:06 +09:00
|
|
|
directLightSelect->value(0);
|
2013-09-04 15:00:04 +09:00
|
|
|
}else if((!r_mapSoftShadow) && (r_dlights) && (r_modelShadows) && (!r_fogShadow) && (!r_physicalLighting)){
|
2013-08-18 16:18:06 +09:00
|
|
|
directLightSelect->value(1);
|
2013-09-13 14:16:52 +09:00
|
|
|
}else if((r_mapSoftShadow) && (r_dlights) && (r_modelShadows) && (!r_fogShadow) && (r_physicalLighting)){
|
2013-08-18 16:18:06 +09:00
|
|
|
directLightSelect->value(2);
|
2013-09-13 14:16:52 +09:00
|
|
|
}else if((r_mapSoftShadow) && (r_dlights) && (r_modelShadows) && (r_fogShadow) && (r_physicalLighting)){
|
2013-08-18 16:18:06 +09:00
|
|
|
directLightSelect->value(3);
|
2013-09-13 14:16:52 +09:00
|
|
|
}else{
|
|
|
|
directLightSelect->value(4);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
shaderSelect->clear();
|
|
|
|
shaderSelect->add("Low");
|
2013-09-02 19:35:14 +09:00
|
|
|
shaderSelect->add("Medium");
|
|
|
|
if(shaderHighCapable){
|
|
|
|
shaderSelect->add("High");
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
shaderSelect->add("Custom");
|
|
|
|
|
2013-09-02 19:35:14 +09:00
|
|
|
if(shaderHighCapable){
|
|
|
|
if((!r_water)){
|
|
|
|
shaderSelect->value(0);
|
|
|
|
}else if((int)r_water == 1){
|
|
|
|
shaderSelect->value(1);
|
|
|
|
}else if((int)r_water == 2){
|
|
|
|
shaderSelect->value(2);
|
|
|
|
}else{
|
|
|
|
shaderSelect->value(3);
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}else{
|
2013-09-02 19:35:14 +09:00
|
|
|
if((!r_water)){
|
|
|
|
shaderSelect->value(0);
|
|
|
|
}else if((int)r_water == 1){
|
|
|
|
shaderSelect->value(1);
|
|
|
|
}else{
|
|
|
|
shaderSelect->value(2);
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// --- audio
|
|
|
|
polyInput->step(16.);
|
|
|
|
polyInput->range(32., 256.);
|
|
|
|
polyInput->value((int)s_maxPolyphonics);
|
|
|
|
|
|
|
|
eaxCheck->value(s_eax ? 1 : 0);
|
|
|
|
|
|
|
|
// --- game
|
|
|
|
playerNameInput->value(cg_playerName.CString());
|
|
|
|
playerNameInput->maximum_size(15);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::Init() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
|
|
|
|
|
|
|
// banner
|
|
|
|
std::string data = spades::FileManager::ReadAllBytes("Gfx/Banner.png");
|
|
|
|
Fl_PNG_Image *img = new Fl_PNG_Image("Gfx/Banner.png", (const unsigned char *)data.data(), data.size());
|
|
|
|
bannerBox->image(img);
|
|
|
|
|
2013-08-27 16:35:12 +02:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
// --- about
|
|
|
|
std::string text, pkg;
|
|
|
|
pkg = PACKAGE_STRING;
|
|
|
|
text = std::string((const char *)aboutText, sizeof(aboutText));
|
|
|
|
text = spades::Replace(text, "${PACKAGE_STRING}",
|
|
|
|
pkg);
|
|
|
|
|
|
|
|
aboutView->value(text.c_str());
|
|
|
|
|
2013-08-31 18:06:46 +02:00
|
|
|
browser = new spades::Serverbrowser( serverListbox );
|
2013-09-01 00:35:56 +02:00
|
|
|
spades::ServerFilter::Flags flags = browser->Filter();
|
|
|
|
checkFilterEmpty->value( flags & spades::ServerFilter::flt_Empty );
|
|
|
|
checkFilterFull->value( flags & spades::ServerFilter::flt_Full );
|
|
|
|
checkFilterV75->value( flags & spades::ServerFilter::flt_Ver075 );
|
|
|
|
checkFilterV76->value( flags & spades::ServerFilter::flt_Ver076 );
|
|
|
|
checkFilterVOther->value( flags & spades::ServerFilter::flt_VerOther );
|
2013-08-31 18:06:46 +02:00
|
|
|
browser->Start();
|
2013-08-28 02:20:26 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/** This function is called after showing window.
|
|
|
|
* Creating SDL window before showing MainWindow results in
|
|
|
|
* internal error on Mac OS X. */
|
|
|
|
void MainWindow::CheckGLCapability() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
|
|
|
// check GL capabilities
|
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
SPLog("Initializing SDL for capability query");
|
2013-08-28 02:20:26 +09:00
|
|
|
SDL_Init(SDL_INIT_VIDEO);
|
|
|
|
|
|
|
|
SDL_Rect **modes = SDL_ListModes(NULL, SDL_OPENGL | SDL_FULLSCREEN |
|
|
|
|
SDL_DOUBLEBUF);
|
|
|
|
if(modes && modes != (SDL_Rect **)-1){
|
|
|
|
g_modes.clear();
|
|
|
|
for(size_t i = 0; modes[i]; i++){
|
|
|
|
SDL_Rect mode = *(modes[i]);
|
|
|
|
if(mode.w < 800 || mode.h < 600)
|
|
|
|
continue;
|
|
|
|
g_modes.push_back(spades::IntVector3::Make(mode.w, mode.h, 0));
|
|
|
|
SPLog("Video Mode Found: %dx%d", mode.w, mode.h);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
bool capable = true;
|
|
|
|
std::string msg;
|
2013-08-29 00:46:54 +09:00
|
|
|
if(!SDL_SetVideoMode(1,1, 32, SDL_OPENGL|SDL_NOFRAME)){
|
2013-08-28 03:59:26 +09:00
|
|
|
// OpenGL initialization failed!
|
|
|
|
outputGLRenderer->value("N/A");
|
|
|
|
outputGLVersion->value("N/A");
|
|
|
|
outputGLSLVersion->value("N/A");
|
|
|
|
|
|
|
|
std::string err = SDL_GetError();
|
|
|
|
SPLog("SDL_SetVideoMode failed: %s", err.c_str());
|
|
|
|
msg = "<b>OpenGL/SDL couldn't be initialized. "
|
|
|
|
"You cannot play OpenSpades.</b><br><br>"
|
|
|
|
"<b>Message from SDL</b><br>"
|
|
|
|
+ err;
|
|
|
|
capable = false;
|
2013-09-02 19:35:14 +09:00
|
|
|
shaderHighCapable = false;
|
2013-09-12 14:39:42 +09:00
|
|
|
postFilterHighCapable = false;
|
2013-08-28 03:59:26 +09:00
|
|
|
}else{
|
|
|
|
|
2013-09-02 19:35:14 +09:00
|
|
|
shaderHighCapable = true;
|
2013-09-12 14:39:42 +09:00
|
|
|
postFilterHighCapable = true;
|
2013-09-02 19:35:14 +09:00
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
const char *str;
|
2013-09-01 17:47:16 +09:00
|
|
|
GLint maxTextureSize;
|
|
|
|
GLint max3DTextureSize;
|
2013-09-02 19:35:14 +09:00
|
|
|
GLint maxCombinedTextureUnits;
|
|
|
|
GLint maxVertexTextureUnits;
|
2013-08-28 03:59:26 +09:00
|
|
|
SPLog("--- OpenGL Renderer Info ---");
|
|
|
|
if((str = (const char*)glGetString(GL_VENDOR)) != NULL) {
|
|
|
|
SPLog("Vendor: %s", str);
|
|
|
|
}
|
|
|
|
if((str = (const char *)glGetString(GL_RENDERER)) != NULL) {
|
|
|
|
outputGLRenderer->value(str);
|
|
|
|
SPLog("Name: %s", str);
|
|
|
|
}else{
|
|
|
|
outputGLRenderer->value("(unknown)");
|
|
|
|
}
|
|
|
|
if((str = (const char *)glGetString(GL_VERSION)) != NULL) {
|
|
|
|
outputGLVersion->value(str);
|
|
|
|
SPLog("Version: %s", str);
|
|
|
|
}else{
|
|
|
|
outputGLVersion->value("(unknown)");
|
|
|
|
}
|
|
|
|
if((str = (const char *)glGetString(GL_SHADING_LANGUAGE_VERSION)) != NULL) {
|
|
|
|
outputGLSLVersion->value(str);
|
|
|
|
SPLog("Shading Language Version: %s", str);
|
|
|
|
}else{
|
|
|
|
outputGLSLVersion->value("(unknown)");
|
|
|
|
}
|
2013-09-02 19:35:14 +09:00
|
|
|
|
2013-09-01 17:47:16 +09:00
|
|
|
maxTextureSize = 0;
|
|
|
|
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &maxTextureSize);
|
|
|
|
if(maxTextureSize > 0) {
|
|
|
|
SPLog("Max Texture Size: %d", (int)maxTextureSize);
|
|
|
|
}
|
|
|
|
max3DTextureSize = 0;
|
|
|
|
glGetIntegerv(GL_MAX_3D_TEXTURE_SIZE, &max3DTextureSize);
|
|
|
|
if(max3DTextureSize > 0) {
|
|
|
|
SPLog("Max 3D Texture Size: %d", (int)max3DTextureSize);
|
|
|
|
}
|
2013-08-28 03:59:26 +09:00
|
|
|
|
2013-09-02 19:35:14 +09:00
|
|
|
maxCombinedTextureUnits = 0;
|
|
|
|
glGetIntegerv(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS, &maxCombinedTextureUnits);
|
|
|
|
if(maxCombinedTextureUnits > 0) {
|
|
|
|
SPLog("Max Combined Texture Image Units: %d", (int)maxCombinedTextureUnits);
|
|
|
|
}
|
|
|
|
|
|
|
|
maxVertexTextureUnits = 0;
|
|
|
|
glGetIntegerv(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, &maxVertexTextureUnits);
|
|
|
|
if(maxVertexTextureUnits > 0) {
|
|
|
|
SPLog("Max Vertex Texture Image Units: %d", (int)maxVertexTextureUnits);
|
|
|
|
}
|
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
str = (const char*)glGetString(GL_EXTENSIONS);
|
|
|
|
std::string extensions;
|
|
|
|
if(str)
|
|
|
|
extensions = str;
|
|
|
|
const char * const requiredExtensions[] = {
|
|
|
|
"GL_ARB_multitexture",
|
|
|
|
"GL_ARB_shader_objects",
|
|
|
|
"GL_ARB_shading_language_100",
|
2013-09-01 17:36:17 +09:00
|
|
|
"GL_ARB_texture_non_power_of_two",
|
2013-08-28 03:59:26 +09:00
|
|
|
"GL_ARB_vertex_buffer_object",
|
|
|
|
"GL_EXT_framebuffer_object",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
SPLog("--- Extensions ---");
|
|
|
|
std::vector<std::string> strs = spades::Split(str, " ");
|
|
|
|
for(size_t i = 0; i < strs.size(); i++)
|
|
|
|
SPLog("%s", strs[i].c_str());
|
|
|
|
SPLog("------------------");
|
|
|
|
|
|
|
|
for(size_t i = 0; requiredExtensions[i]; i++) {
|
|
|
|
const char *ex = requiredExtensions[i];
|
|
|
|
if(extensions.find(ex) == std::string::npos) {
|
|
|
|
// extension not found
|
|
|
|
msg += "<font color=#ff0000>";
|
|
|
|
msg += ex;
|
|
|
|
msg += " NOT SUPPORTED!";
|
|
|
|
msg += "</font>";
|
|
|
|
capable = false;
|
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += ex;
|
|
|
|
msg += " is supported";
|
|
|
|
msg += "</font>";
|
|
|
|
}
|
|
|
|
msg += "<br>";
|
|
|
|
}
|
|
|
|
|
2013-09-01 23:15:37 +09:00
|
|
|
msg += "<br> <br>";
|
2013-09-01 17:47:16 +09:00
|
|
|
msg += "<b>Other Extensions:</b><br>";
|
|
|
|
|
2013-08-28 14:55:16 +09:00
|
|
|
// non-requred extensions
|
|
|
|
if(extensions.find("GL_ARB_framebuffer_sRGB") ==
|
|
|
|
std::string::npos) {
|
|
|
|
if(r_srgb) {
|
|
|
|
r_srgb = 0;
|
|
|
|
SPLog("Disabling r_srgb: no GL_ARB_framebuffer_sRGB");
|
|
|
|
}
|
2013-09-01 17:47:16 +09:00
|
|
|
msg += "GL_ARB_framebuffer_sRGB is NOT SUPPORTED<br>";
|
|
|
|
msg += " r_srgb is disabled.<br>";
|
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += "GL_ARB_framebuffer_sRGB is supported";
|
|
|
|
msg += "</font><br>";
|
2013-08-28 14:55:16 +09:00
|
|
|
}
|
2013-08-29 23:54:23 +09:00
|
|
|
if(extensions.find("GL_EXT_framebuffer_blit") ==
|
|
|
|
std::string::npos) {
|
|
|
|
if(r_blitFramebuffer) {
|
|
|
|
r_blitFramebuffer = 0;
|
|
|
|
SPLog("Disabling r_blitFramebuffer: no GL_EXT_framebuffer_blit");
|
|
|
|
}
|
|
|
|
if(r_multisamples) {
|
|
|
|
r_multisamples = 0;
|
|
|
|
SPLog("Disabling r_multisamples: no GL_EXT_framebuffer_blit");
|
|
|
|
}
|
2013-09-01 17:47:16 +09:00
|
|
|
msg += "GL_EXT_framebuffer_blit is NOT SUPPORTED<br>";
|
|
|
|
msg += " MSAA is disabled.<br>";
|
|
|
|
msg += " r_blitFramebuffer is disabled.<br>";
|
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += "GL_EXT_framebuffer_blit is supported";
|
|
|
|
msg += "</font><br>";
|
2013-08-29 23:54:23 +09:00
|
|
|
}
|
2013-09-02 19:35:14 +09:00
|
|
|
if(extensions.find("GL_EXT_texture_filter_anisotropic") ==
|
|
|
|
std::string::npos) {
|
|
|
|
if((float)r_maxAnisotropy > 1.1f) {
|
|
|
|
r_maxAnisotropy = 1;
|
|
|
|
SPLog("Setting r_maxAnisotropy to 1: no GL_EXT_texture_filter_anisotropic");
|
|
|
|
}
|
|
|
|
msg += "GL_EXT_texture_filter_anisotropic is NOT SUPPORTED<br>";
|
|
|
|
msg += " r_maxAnisotropy is disabled.<br>";
|
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += "GL_EXT_texture_filter_anisotropic is supported";
|
|
|
|
msg += "</font><br>";
|
|
|
|
}
|
2013-08-28 14:55:16 +09:00
|
|
|
|
2013-09-07 01:36:53 +09:00
|
|
|
if(extensions.find("GL_ARB_occlusion_query") ==
|
|
|
|
std::string::npos) {
|
2013-09-08 16:44:50 +09:00
|
|
|
if(r_occlusionQuery) {
|
|
|
|
r_occlusionQuery = 0;
|
|
|
|
SPLog("Disabling r_occlusionQuery: no GL_ARB_occlusion_query");
|
2013-09-07 01:36:53 +09:00
|
|
|
}
|
|
|
|
msg += "GL_ARB_occlusion_query is NOT SUPPORTED<br>";
|
2013-09-08 16:44:50 +09:00
|
|
|
msg += " r_occlusionQuery is disabled<br>";
|
2013-09-07 01:36:53 +09:00
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += "GL_ARB_occlusion_query is supported";
|
|
|
|
msg += "</font><br>";
|
|
|
|
}
|
|
|
|
|
|
|
|
if(extensions.find("GL_NV_conditional_render") ==
|
|
|
|
std::string::npos) {
|
2013-09-08 16:44:50 +09:00
|
|
|
if(r_occlusionQuery) {
|
|
|
|
r_occlusionQuery = 0;
|
|
|
|
SPLog("Disabling r_occlusionQuery: no GL_NV_conditional_render");
|
2013-09-07 01:36:53 +09:00
|
|
|
}
|
|
|
|
msg += "GL_NV_conditional_render is NOT SUPPORTED<br>";
|
2013-09-08 16:44:50 +09:00
|
|
|
msg += " r_occlusionQuery is disabled<br>";
|
2013-09-07 01:36:53 +09:00
|
|
|
}else{
|
|
|
|
msg += "<font color=#007f00>";
|
|
|
|
msg += "GL_NV_conditional_render is supported";
|
|
|
|
msg += "</font><br>";
|
|
|
|
}
|
|
|
|
|
2013-09-01 23:15:37 +09:00
|
|
|
msg += "<br> <br>";
|
|
|
|
msg += "<b>Miscellaneous:</b><br>";
|
|
|
|
char buf[256];
|
|
|
|
sprintf(buf, "Max Texture Size: %d<br>", (int)maxTextureSize);
|
|
|
|
msg += buf;
|
|
|
|
if(maxTextureSize < 1024) {
|
|
|
|
capable = false;
|
|
|
|
msg += "<font color=#ff0000>";
|
|
|
|
msg += " TOO SMALL (1024 required)";
|
|
|
|
msg += "</font><br>";
|
|
|
|
}
|
|
|
|
if((int)r_shadowMapSize > maxTextureSize) {
|
|
|
|
SPLog("Changed r_shadowMapSize from %d to %d: too small GL_MAX_TEXTURE_SIZE", (int)r_shadowMapSize, maxTextureSize);
|
|
|
|
|
|
|
|
r_shadowMapSize = maxTextureSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(buf, "Max 3D Texture Size: %d<br>", (int)max3DTextureSize);
|
|
|
|
msg += buf;
|
|
|
|
if(max3DTextureSize < 512) {
|
2013-09-02 19:35:14 +09:00
|
|
|
msg += " Global Illumation is disabled (512 required)<br>";
|
2013-09-01 23:15:37 +09:00
|
|
|
|
|
|
|
if(r_radiosity) {
|
|
|
|
r_radiosity = 0;
|
|
|
|
SPLog("Disabling r_radiosity: too small GL_MAX_3D_TEXTURE_SIZE");
|
|
|
|
|
|
|
|
radiosityCheck->deactivate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-02 19:35:14 +09:00
|
|
|
|
|
|
|
sprintf(buf, "Max Combined Texture Image Units: %d<br>", (int)maxCombinedTextureUnits);
|
|
|
|
msg += buf;
|
|
|
|
if(maxCombinedTextureUnits < 12) {
|
|
|
|
msg += " Global Illumation is disabled (12 required)<br>";
|
|
|
|
|
|
|
|
if(r_radiosity) {
|
|
|
|
r_radiosity = 0;
|
|
|
|
SPLog("Disabling r_radiosity: too small GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS");
|
|
|
|
|
|
|
|
radiosityCheck->deactivate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(buf, "Max Vertex Texture Image Units: %d<br>", (int)maxVertexTextureUnits);
|
|
|
|
msg += buf;
|
|
|
|
if(maxVertexTextureUnits < 3) {
|
|
|
|
msg += " Water 2 is disabled (3 required)<br>";
|
|
|
|
msg += " (Shader Effects is limited to Medium)<br>";
|
|
|
|
shaderHighCapable = false;
|
|
|
|
|
|
|
|
if((int)r_water >= 2) {
|
|
|
|
r_water = 1;
|
|
|
|
SPLog("Disabling Water 2: too small GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
if(capable){
|
|
|
|
msg = "Your video card supports all "
|
2013-09-01 23:15:37 +09:00
|
|
|
"required OpenGL extensions/features.<br> <br>" + msg;
|
2013-08-28 03:59:26 +09:00
|
|
|
}else{
|
|
|
|
msg = "<b>Your video card/driver doesn't support "
|
2013-09-01 23:15:37 +09:00
|
|
|
"at least one of required OpenGL extensions/features."
|
|
|
|
" You cannot play OpenSpades.</b><br> <br>" + msg;
|
2013-08-28 03:59:26 +09:00
|
|
|
}
|
|
|
|
|
2013-09-01 23:15:37 +09:00
|
|
|
|
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
}
|
|
|
|
msg = "<font face=Helvetica>" + msg + "</font><a name=last></a>";
|
2013-08-28 02:20:26 +09:00
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
glInfoView->value(msg.c_str());
|
2013-08-28 02:20:26 +09:00
|
|
|
SDL_QuitSubSystem(SDL_INIT_VIDEO);
|
2013-08-28 03:59:26 +09:00
|
|
|
SPLog("SDL video subsystem finalized");
|
2013-08-28 02:20:26 +09:00
|
|
|
|
2013-08-28 03:59:26 +09:00
|
|
|
SPLog("System is OpenSpades capable: %s",
|
|
|
|
capable ? "YES": "NO");
|
|
|
|
if(!capable) {
|
|
|
|
mainTab->value(groupReport);
|
|
|
|
connectButton->deactivate();
|
|
|
|
}
|
2013-08-28 02:20:26 +09:00
|
|
|
|
|
|
|
LoadPrefs();
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
inited = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::SavePrefs() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
if(!inited)
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::string modeStr = modeSelect->value();
|
|
|
|
size_t pos = modeStr.find('x');
|
|
|
|
if(pos != std::string::npos){
|
|
|
|
int w = atoi(modeStr.substr(0, pos).c_str());
|
|
|
|
int h = atoi(modeStr.substr(pos + 1).c_str());
|
|
|
|
if(w >= 256 && h >= 256){
|
|
|
|
r_videoWidth = w;
|
|
|
|
r_videoHeight = h;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cg_lastQuickConnectHost = quickHostInput->value();
|
|
|
|
r_fullscreen = fullscreenCheck->value() ? 1 : 0;
|
|
|
|
switch(msaaSelect->value()){
|
2013-08-22 22:39:39 +09:00
|
|
|
case 0: r_multisamples = 0; r_fxaa = 0; break;
|
|
|
|
case 1: r_multisamples = 2; r_fxaa = 0; break;
|
|
|
|
case 2: r_multisamples = 4; r_fxaa = 0; break;
|
|
|
|
case 3: r_multisamples = 0; r_fxaa = 1; break;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// --- graphics
|
|
|
|
cg_blood = bloodCheck->value() ? 1 : 0;
|
2013-09-12 14:39:42 +09:00
|
|
|
switch(postFilterSelect->value()){
|
|
|
|
case 0:
|
|
|
|
r_bloom = 0;
|
|
|
|
r_lens = 0;
|
|
|
|
r_lensFlare = 0;
|
|
|
|
r_cameraBlur = 0;
|
|
|
|
r_colorCorrection = 0;
|
|
|
|
r_depthOfField = 0;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
r_bloom = 1;
|
|
|
|
r_lens = 1;
|
|
|
|
r_lensFlare = 1;
|
|
|
|
r_cameraBlur = 1;
|
|
|
|
r_colorCorrection = 1;
|
|
|
|
r_depthOfField = 0;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
if(postFilterHighCapable){
|
|
|
|
r_bloom = 1;
|
|
|
|
r_lens = 1;
|
|
|
|
r_lensFlare = 1;
|
|
|
|
r_cameraBlur = 1;
|
|
|
|
r_colorCorrection = 1;
|
|
|
|
r_depthOfField = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
r_softParticles = softParticleCheck->value() ? 1 : 0;
|
|
|
|
r_radiosity = radiosityCheck->value() ? 1 : 0;
|
|
|
|
switch(directLightSelect->value()){
|
|
|
|
case 0:
|
|
|
|
r_modelShadows = 0;
|
2013-09-03 01:25:58 +09:00
|
|
|
r_dlights = 1;
|
2013-08-18 16:18:06 +09:00
|
|
|
r_mapSoftShadow = 0;
|
2013-08-22 21:21:52 +09:00
|
|
|
r_fogShadow = 0;
|
2013-09-04 15:00:04 +09:00
|
|
|
r_physicalLighting = 0;
|
2013-08-18 16:18:06 +09:00
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
r_modelShadows = 1;
|
|
|
|
r_dlights = 1;
|
|
|
|
r_mapSoftShadow = 0;
|
2013-08-22 21:21:52 +09:00
|
|
|
r_fogShadow = 0;
|
2013-09-04 15:00:04 +09:00
|
|
|
r_physicalLighting = 0;
|
2013-08-18 16:18:06 +09:00
|
|
|
break;
|
|
|
|
case 2:
|
2013-09-13 14:16:52 +09:00
|
|
|
r_modelShadows = 1;
|
|
|
|
r_dlights = 1;
|
|
|
|
r_mapSoftShadow = 1;
|
|
|
|
r_fogShadow = 0;
|
|
|
|
r_physicalLighting = 1;
|
|
|
|
break;
|
|
|
|
case 3:
|
2013-08-18 16:18:06 +09:00
|
|
|
r_modelShadows = 1;
|
|
|
|
r_dlights = 1;
|
|
|
|
r_mapSoftShadow = 1;
|
2013-08-22 21:21:52 +09:00
|
|
|
r_fogShadow = 1;
|
2013-09-04 15:00:04 +09:00
|
|
|
r_physicalLighting = 1;
|
2013-08-18 16:18:06 +09:00
|
|
|
break;
|
|
|
|
}
|
2013-09-02 19:35:14 +09:00
|
|
|
if(shaderHighCapable){
|
|
|
|
switch(shaderSelect->value()){
|
|
|
|
case 0:
|
|
|
|
r_water = 0;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
r_water = 1;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
r_water = 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
switch(shaderSelect->value()){
|
|
|
|
case 0:
|
|
|
|
r_water = 0;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
r_water = 1;
|
|
|
|
break;
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// --- audio
|
|
|
|
s_maxPolyphonics = (int)polyInput->value();
|
|
|
|
s_eax = eaxCheck->value() ? 1 : 0;
|
|
|
|
|
|
|
|
// --- game
|
|
|
|
cg_playerName = playerNameInput->value();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-08-22 22:39:39 +09:00
|
|
|
void MainWindow::DisableMSAA() {
|
2013-08-29 23:54:23 +09:00
|
|
|
if(r_blitFramebuffer){
|
|
|
|
if(msaaSelect->value() >= 1 && msaaSelect->value() <= 2)
|
|
|
|
msaaSelect->value(3);
|
|
|
|
}
|
2013-08-22 22:39:39 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::MSAAEnabled() {
|
2013-08-29 23:54:23 +09:00
|
|
|
if(msaaSelect->value() >= 1 &&
|
|
|
|
msaaSelect->value() <= 2 &&
|
|
|
|
r_blitFramebuffer){
|
|
|
|
if(shaderSelect->value() == 1)
|
|
|
|
shaderSelect->value(0);
|
2013-09-13 14:16:52 +09:00
|
|
|
if(directLightSelect->value() == 3)
|
|
|
|
directLightSelect->value(2);
|
2013-08-29 23:54:23 +09:00
|
|
|
}
|
2013-08-22 22:39:39 +09:00
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
void MainWindow::OpenDetailConfig() {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
|
|
|
|
DetailConfigWindow cfg;
|
|
|
|
cfg.set_modal();
|
|
|
|
cfg.Init();
|
|
|
|
cfg.show();
|
|
|
|
while(cfg.visible()){
|
|
|
|
Fl::wait();
|
|
|
|
}
|
|
|
|
LoadPrefs();
|
|
|
|
}
|
|
|
|
|
2013-08-31 18:06:46 +02:00
|
|
|
void MainWindow::ServerSelectionChanged()
|
|
|
|
{
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
if( browser ) {
|
|
|
|
int item = serverListbox->value();
|
2013-09-01 00:35:56 +02:00
|
|
|
if( item > 1 ) {
|
2013-08-31 18:06:46 +02:00
|
|
|
browser->onSelection( serverListbox->data( item ), quickHostInput );
|
2013-09-01 00:35:56 +02:00
|
|
|
} else if( item == 1 ) {
|
|
|
|
browser->onHeaderClick( Fl::event_x() - serverListbox->x() );
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-08-31 18:06:46 +02:00
|
|
|
void MainWindow::updateFilters()
|
|
|
|
{
|
|
|
|
if( browser ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
spades::ServerFilter::Flags flags = spades::ServerFilter::flt_None;
|
2013-08-31 18:06:46 +02:00
|
|
|
if( checkFilterEmpty->value() ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
flags |= spades::ServerFilter::flt_Empty;
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
if( checkFilterFull->value() ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
flags |= spades::ServerFilter::flt_Full;
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
if( checkFilterV75->value() ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
flags |= spades::ServerFilter::flt_Ver075;
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
if( checkFilterV76->value() ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
flags |= spades::ServerFilter::flt_Ver076;
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
if( checkFilterVOther->value() ) {
|
2013-08-31 19:29:58 +02:00
|
|
|
flags |= spades::ServerFilter::flt_VerOther;
|
2013-08-31 18:06:46 +02:00
|
|
|
}
|
|
|
|
browser->setFilter( flags );
|
|
|
|
browser->refreshList();
|
|
|
|
}
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
|