2019-04-25 23:29:34 +03:00
|
|
|
/*
|
2024-01-14 12:56:36 +03:00
|
|
|
* Copyright 2010-2024 Branimir Karadzic. All rights reserved.
|
2022-01-15 22:59:06 +03:00
|
|
|
* License: https://github.com/bkaradzic/bgfx/blob/master/LICENSE
|
2019-04-25 23:29:34 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <bx/allocator.h>
|
|
|
|
#include <bx/filepath.h>
|
|
|
|
#include <bx/string.h>
|
|
|
|
#include <bx/readerwriter.h>
|
|
|
|
#include <bx/process.h>
|
|
|
|
|
|
|
|
#include "dialog.h"
|
|
|
|
|
|
|
|
#if BX_PLATFORM_WINDOWS
|
|
|
|
typedef uintptr_t (__stdcall *LPOFNHOOKPROC)(void*, uint32_t, uintptr_t, uint64_t);
|
|
|
|
|
|
|
|
struct OPENFILENAMEA
|
|
|
|
{
|
|
|
|
uint32_t structSize;
|
|
|
|
void* hwndOwner;
|
|
|
|
void* hinstance;
|
|
|
|
const char* filter;
|
|
|
|
const char* customFilter;
|
|
|
|
uint32_t maxCustomFilter;
|
|
|
|
uint32_t filterIndex;
|
|
|
|
const char* file;
|
|
|
|
uint32_t maxFile;
|
|
|
|
const char* fileTitle;
|
|
|
|
uint32_t maxFileTitle;
|
|
|
|
const char* initialDir;
|
|
|
|
const char* title;
|
|
|
|
uint32_t flags;
|
|
|
|
uint16_t fileOffset;
|
|
|
|
uint16_t fileExtension;
|
|
|
|
const char* defExt;
|
|
|
|
uintptr_t customData;
|
|
|
|
LPOFNHOOKPROC hook;
|
|
|
|
const char* templateName;
|
|
|
|
void* reserved0;
|
|
|
|
uint32_t reserved1;
|
|
|
|
uint32_t flagsEx;
|
|
|
|
};
|
|
|
|
|
2019-08-15 07:19:03 +03:00
|
|
|
extern "C" bool __stdcall GetOpenFileNameA(OPENFILENAMEA* _ofn);
|
2022-12-24 10:00:26 +03:00
|
|
|
extern "C" bool __stdcall GetSaveFileNameA(OPENFILENAMEA * _ofn);
|
2019-08-15 07:19:03 +03:00
|
|
|
extern "C" void* __stdcall GetModuleHandleA(const char* _moduleName);
|
|
|
|
extern "C" uint32_t __stdcall GetModuleFileNameA(void* _module, char* _outFilePath, uint32_t _size);
|
|
|
|
extern "C" void* __stdcall ShellExecuteA(void* _hwnd, void* _operation, void* _file, void* _parameters, void* _directory, int32_t _showCmd);
|
2019-04-25 23:29:34 +03:00
|
|
|
|
|
|
|
#endif // BX_PLATFORM_WINDOWS
|
|
|
|
|
2019-08-15 07:19:03 +03:00
|
|
|
void openUrl(const bx::StringView& _url)
|
|
|
|
{
|
2019-08-15 07:46:50 +03:00
|
|
|
char tmp[4096];
|
|
|
|
|
2019-08-15 07:19:03 +03:00
|
|
|
#if BX_PLATFORM_WINDOWS
|
2019-08-15 07:46:50 +03:00
|
|
|
# define OPEN ""
|
|
|
|
#elif BX_PLATFORM_OSX
|
|
|
|
# define OPEN "open "
|
|
|
|
#else
|
|
|
|
# define OPEN "xdg-open "
|
|
|
|
#endif // BX_PLATFORM_OSX
|
|
|
|
|
|
|
|
bx::snprintf(tmp, BX_COUNTOF(tmp), OPEN "%.*s", _url.getLength(), _url.getPtr() );
|
|
|
|
|
|
|
|
#undef OPEN
|
|
|
|
|
|
|
|
#if BX_PLATFORM_WINDOWS
|
2019-08-15 08:31:32 +03:00
|
|
|
void* result = ShellExecuteA(NULL, NULL, tmp, NULL, NULL, false);
|
|
|
|
BX_UNUSED(result);
|
2019-08-16 17:05:01 +03:00
|
|
|
#elif !BX_PLATFORM_IOS
|
2019-08-15 08:31:32 +03:00
|
|
|
int32_t result = system(tmp);
|
|
|
|
BX_UNUSED(result);
|
2019-08-15 07:19:03 +03:00
|
|
|
#endif // BX_PLATFORM_*
|
|
|
|
}
|
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
class Split
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Split(const bx::StringView& _str, char _ch)
|
|
|
|
: m_str(_str)
|
|
|
|
, m_token(_str.getPtr(), bx::strFind(_str, _ch).getPtr() )
|
|
|
|
, m_ch(_ch)
|
|
|
|
{
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::StringView next()
|
|
|
|
{
|
|
|
|
bx::StringView result = m_token;
|
|
|
|
m_token = bx::strTrim(
|
2019-04-30 06:52:39 +03:00
|
|
|
bx::StringView(m_token.getTerm()+1, bx::strFind(bx::StringView(m_token.getTerm()+1, m_str.getTerm() ), m_ch).getPtr() )
|
|
|
|
, " \t\n"
|
|
|
|
);
|
2019-04-25 23:29:34 +03:00
|
|
|
return result;
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bool isDone() const
|
|
|
|
{
|
|
|
|
return m_token.isEmpty();
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
private:
|
|
|
|
const bx::StringView& m_str;
|
|
|
|
bx::StringView m_token;
|
|
|
|
char m_ch;
|
|
|
|
};
|
|
|
|
|
2022-12-24 10:00:26 +03:00
|
|
|
#if BX_PLATFORM_WINDOWS
|
|
|
|
extern "C" typedef bool(__stdcall* OPENFILENAMEFUNCTION)(OPENFILENAMEA* _ofn);
|
|
|
|
static const struct { OPENFILENAMEFUNCTION m_function; uint32_t m_flags; }
|
|
|
|
s_getFileNameA[] =
|
|
|
|
{
|
|
|
|
{ GetOpenFileNameA, /* OFN_EXPLORER */ 0x00080000 | /* OFN_DONTADDTORECENT */ 0x02000000 | /* OFN_FILEMUSTEXIST */ 0x00001000 },
|
|
|
|
{ GetSaveFileNameA, /* OFN_EXPLORER */ 0x00080000 | /* OFN_DONTADDTORECENT */ 0x02000000 },
|
|
|
|
};
|
|
|
|
BX_STATIC_ASSERT(BX_COUNTOF(s_getFileNameA) == FileSelectionDialogType::Count);
|
|
|
|
#endif
|
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
#if !BX_PLATFORM_OSX
|
|
|
|
bool openFileSelectionDialog(
|
2019-04-30 06:52:39 +03:00
|
|
|
bx::FilePath& _inOutFilePath
|
|
|
|
, FileSelectionDialogType::Enum _type
|
|
|
|
, const bx::StringView& _title
|
|
|
|
, const bx::StringView& _filter
|
|
|
|
)
|
2019-04-25 23:29:34 +03:00
|
|
|
{
|
|
|
|
#if BX_PLATFORM_LINUX
|
|
|
|
char tmp[4096];
|
|
|
|
bx::StaticMemoryBlockWriter writer(tmp, sizeof(tmp) );
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::Error err;
|
|
|
|
bx::write(&writer, &err
|
2019-04-30 06:52:39 +03:00
|
|
|
, "--file-selection%s --title \"%.*s\" --filename \"%s\""
|
|
|
|
, FileSelectionDialogType::Save == _type ? " --save" : ""
|
|
|
|
, _title.getLength()
|
|
|
|
, _title.getPtr()
|
|
|
|
, _inOutFilePath.getCPtr()
|
|
|
|
);
|
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
for (bx::LineReader lr(_filter); !lr.isDone();)
|
|
|
|
{
|
|
|
|
const bx::StringView line = lr.next();
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::write(&writer, &err
|
2019-04-30 06:52:39 +03:00
|
|
|
, " --file-filter \"%.*s\""
|
|
|
|
, line.getLength()
|
|
|
|
, line.getPtr()
|
|
|
|
);
|
2019-04-25 23:29:34 +03:00
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-07-28 05:26:48 +03:00
|
|
|
bx::write(&writer, '\0', &err);
|
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
if (err.isOk() )
|
|
|
|
{
|
|
|
|
bx::ProcessReader pr;
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
if (bx::open(&pr, "zenity", tmp, &err) )
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
int32_t total = bx::read(&pr, buffer, sizeof(buffer), &err);
|
|
|
|
bx::close(&pr);
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
if (0 == pr.getExitCode() )
|
|
|
|
{
|
|
|
|
_inOutFilePath.set(bx::strRTrim(bx::StringView(buffer, total), "\n\r") );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#elif BX_PLATFORM_WINDOWS
|
2022-12-24 10:00:26 +03:00
|
|
|
if (_type < 0 || _type >= BX_COUNTOF(s_getFileNameA))
|
|
|
|
return false;
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
char out[bx::kMaxFilePath] = { '\0' };
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
OPENFILENAMEA ofn;
|
|
|
|
bx::memSet(&ofn, 0, sizeof(ofn) );
|
|
|
|
ofn.structSize = sizeof(OPENFILENAMEA);
|
|
|
|
ofn.initialDir = _inOutFilePath.getCPtr();
|
|
|
|
ofn.file = out;
|
|
|
|
ofn.maxFile = sizeof(out);
|
2022-12-24 10:00:26 +03:00
|
|
|
ofn.flags = s_getFileNameA[_type].m_flags;
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
char tmp[4096];
|
|
|
|
bx::StaticMemoryBlockWriter writer(tmp, sizeof(tmp) );
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::Error err;
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
ofn.title = tmp;
|
|
|
|
bx::write(&writer, &err, "%.*s", _title.getLength(), _title.getPtr() );
|
|
|
|
bx::write(&writer, '\0', &err);
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
ofn.filter = tmp + uint32_t(bx::seek(&writer) );
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
for (bx::LineReader lr(_filter); !lr.isDone() && err.isOk();)
|
|
|
|
{
|
|
|
|
const bx::StringView line = lr.next();
|
|
|
|
const bx::StringView sep = bx::strFind(line, '|');
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
if (!sep.isEmpty() )
|
|
|
|
{
|
|
|
|
bx::write(&writer, bx::strTrim(bx::StringView(line.getPtr(), sep.getPtr() ), " "), &err);
|
|
|
|
bx::write(&writer, '\0', &err);
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bool first = true;
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
for (Split split(bx::strTrim(bx::StringView(sep.getPtr()+1, line.getTerm() ), " "), ' '); !split.isDone() && err.isOk();)
|
|
|
|
{
|
|
|
|
const bx::StringView token = split.next();
|
|
|
|
if (!first)
|
|
|
|
{
|
|
|
|
bx::write(&writer, ';', &err);
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
first = false;
|
|
|
|
bx::write(&writer, token, &err);
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::write(&writer, '\0', &err);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bx::write(&writer, line, &err);
|
|
|
|
bx::write(&writer, '\0', &err);
|
|
|
|
bx::write(&writer, '\0', &err);
|
|
|
|
}
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
bx::write(&writer, '\0', &err);
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
if (err.isOk()
|
2022-12-24 10:00:26 +03:00
|
|
|
&& s_getFileNameA[_type].m_function(&ofn))
|
2019-04-25 23:29:34 +03:00
|
|
|
{
|
|
|
|
_inOutFilePath.set(ofn.file);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
BX_UNUSED(_inOutFilePath, _type, _title, _filter);
|
|
|
|
#endif // BX_PLATFORM_LINUX
|
2019-04-30 06:52:39 +03:00
|
|
|
|
2019-04-25 23:29:34 +03:00
|
|
|
return false;
|
|
|
|
}
|
2019-04-30 06:52:39 +03:00
|
|
|
#endif // !BX_PLATFORM_OSX
|