dialog.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /*
  2. * Copyright 2010-2019 Branimir Karadzic. All rights reserved.
  3. * License: https://github.com/bkaradzic/bgfx#license-bsd-2-clause
  4. */
  5. #include <bx/allocator.h>
  6. #include <bx/filepath.h>
  7. #include <bx/string.h>
  8. #include <bx/readerwriter.h>
  9. #include <bx/process.h>
  10. #include "dialog.h"
  11. #if BX_PLATFORM_WINDOWS
  12. extern "C" void* __stdcall GetModuleHandleA(const char* _moduleName);
  13. extern "C" uint32_t __stdcall GetModuleFileNameA(void* _module, char* _outFilePath, uint32_t _size);
  14. typedef uintptr_t (__stdcall *LPOFNHOOKPROC)(void*, uint32_t, uintptr_t, uint64_t);
  15. struct OPENFILENAMEA
  16. {
  17. uint32_t structSize;
  18. void* hwndOwner;
  19. void* hinstance;
  20. const char* filter;
  21. const char* customFilter;
  22. uint32_t maxCustomFilter;
  23. uint32_t filterIndex;
  24. const char* file;
  25. uint32_t maxFile;
  26. const char* fileTitle;
  27. uint32_t maxFileTitle;
  28. const char* initialDir;
  29. const char* title;
  30. uint32_t flags;
  31. uint16_t fileOffset;
  32. uint16_t fileExtension;
  33. const char* defExt;
  34. uintptr_t customData;
  35. LPOFNHOOKPROC hook;
  36. const char* templateName;
  37. void* reserved0;
  38. uint32_t reserved1;
  39. uint32_t flagsEx;
  40. };
  41. extern "C" bool __stdcall GetOpenFileNameA(OPENFILENAMEA* _ofn);
  42. #endif // BX_PLATFORM_WINDOWS
  43. class Split
  44. {
  45. public:
  46. Split(const bx::StringView& _str, char _ch)
  47. : m_str(_str)
  48. , m_token(_str.getPtr(), bx::strFind(_str, _ch).getPtr() )
  49. , m_ch(_ch)
  50. {
  51. }
  52. bx::StringView next()
  53. {
  54. bx::StringView result = m_token;
  55. m_token = bx::strTrim(
  56. bx::StringView(m_token.getTerm()+1, bx::strFind(bx::StringView(m_token.getTerm()+1, m_str.getTerm() ), m_ch).getPtr() )
  57. , " \t\n"
  58. );
  59. return result;
  60. }
  61. bool isDone() const
  62. {
  63. return m_token.isEmpty();
  64. }
  65. private:
  66. const bx::StringView& m_str;
  67. bx::StringView m_token;
  68. char m_ch;
  69. };
  70. #if !BX_PLATFORM_OSX
  71. bool openFileSelectionDialog(
  72. bx::FilePath& _inOutFilePath
  73. , FileSelectionDialogType::Enum _type
  74. , const bx::StringView& _title
  75. , const bx::StringView& _filter
  76. )
  77. {
  78. #if BX_PLATFORM_LINUX
  79. char tmp[4096];
  80. bx::StaticMemoryBlockWriter writer(tmp, sizeof(tmp) );
  81. bx::Error err;
  82. bx::write(&writer, &err
  83. , "--file-selection%s --title \"%.*s\" --filename \"%s\""
  84. , FileSelectionDialogType::Save == _type ? " --save" : ""
  85. , _title.getLength()
  86. , _title.getPtr()
  87. , _inOutFilePath.getCPtr()
  88. );
  89. for (bx::LineReader lr(_filter); !lr.isDone();)
  90. {
  91. const bx::StringView line = lr.next();
  92. bx::write(&writer, &err
  93. , " --file-filter \"%.*s\""
  94. , line.getLength()
  95. , line.getPtr()
  96. );
  97. }
  98. if (err.isOk() )
  99. {
  100. bx::ProcessReader pr;
  101. if (bx::open(&pr, "zenity", tmp, &err) )
  102. {
  103. char buffer[1024];
  104. int32_t total = bx::read(&pr, buffer, sizeof(buffer), &err);
  105. bx::close(&pr);
  106. if (0 == pr.getExitCode() )
  107. {
  108. _inOutFilePath.set(bx::strRTrim(bx::StringView(buffer, total), "\n\r") );
  109. return true;
  110. }
  111. }
  112. }
  113. #elif BX_PLATFORM_WINDOWS
  114. BX_UNUSED(_type);
  115. char out[bx::kMaxFilePath] = { '\0' };
  116. OPENFILENAMEA ofn;
  117. bx::memSet(&ofn, 0, sizeof(ofn) );
  118. ofn.structSize = sizeof(OPENFILENAMEA);
  119. ofn.initialDir = _inOutFilePath.getCPtr();
  120. ofn.file = out;
  121. ofn.maxFile = sizeof(out);
  122. ofn.flags = 0
  123. | /* OFN_EXPLORER */ 0x00080000
  124. | /* OFN_FILEMUSTEXIST */ 0x00001000
  125. | /* OFN_DONTADDTORECENT */ 0x02000000
  126. ;
  127. char tmp[4096];
  128. bx::StaticMemoryBlockWriter writer(tmp, sizeof(tmp) );
  129. bx::Error err;
  130. ofn.title = tmp;
  131. bx::write(&writer, &err, "%.*s", _title.getLength(), _title.getPtr() );
  132. bx::write(&writer, '\0', &err);
  133. ofn.filter = tmp + uint32_t(bx::seek(&writer) );
  134. for (bx::LineReader lr(_filter); !lr.isDone() && err.isOk();)
  135. {
  136. const bx::StringView line = lr.next();
  137. const bx::StringView sep = bx::strFind(line, '|');
  138. if (!sep.isEmpty() )
  139. {
  140. bx::write(&writer, bx::strTrim(bx::StringView(line.getPtr(), sep.getPtr() ), " "), &err);
  141. bx::write(&writer, '\0', &err);
  142. bool first = true;
  143. for (Split split(bx::strTrim(bx::StringView(sep.getPtr()+1, line.getTerm() ), " "), ' '); !split.isDone() && err.isOk();)
  144. {
  145. const bx::StringView token = split.next();
  146. if (!first)
  147. {
  148. bx::write(&writer, ';', &err);
  149. }
  150. first = false;
  151. bx::write(&writer, token, &err);
  152. }
  153. bx::write(&writer, '\0', &err);
  154. }
  155. else
  156. {
  157. bx::write(&writer, line, &err);
  158. bx::write(&writer, '\0', &err);
  159. bx::write(&writer, '\0', &err);
  160. }
  161. }
  162. bx::write(&writer, '\0', &err);
  163. if (err.isOk()
  164. && GetOpenFileNameA(&ofn) )
  165. {
  166. _inOutFilePath.set(ofn.file);
  167. return true;
  168. }
  169. #else
  170. BX_UNUSED(_inOutFilePath, _type, _title, _filter);
  171. #endif // BX_PLATFORM_LINUX
  172. return false;
  173. }
  174. #endif // !BX_PLATFORM_OSX