From 11652d015df7421806553300bc6f693a28aa99ec Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Thu, 26 May 2022 01:44:46 +0200 Subject: [PATCH] SDK launcher progress * Fixed compiler error when trying to compile SDK Launcher in debug (static lib was not compiled with static link runtime). * Documented most stuff in basepanel.cpp * Split some logic into dedicated functions in basepanel.cpp * Implemented VDF parser from Matthias Moeller. * Parse playlist file and load all playlists into the combo box. --- license/thirdpartylegalnotices.txt | 27 + r5dev/core/stdafx.h | 3 + r5dev/public/include/vdf_parser.h | 712 ++++++++++++++++++++++++++ r5dev/sdklauncher/basepanel.cpp | 388 ++++++++------ r5dev/sdklauncher/basepanel.h | 12 +- r5dev/sdklauncher/sdklauncher.cpp | 4 +- r5dev/sdklauncher/sdklauncher.h | 6 +- r5dev/sdklauncher/sdklauncher_const.h | 2 +- r5dev/vproj/libcppkore.vcxproj | 2 +- 9 files changed, 989 insertions(+), 167 deletions(-) create mode 100644 r5dev/public/include/vdf_parser.h diff --git a/license/thirdpartylegalnotices.txt b/license/thirdpartylegalnotices.txt index e04fc2d8..fd1985b5 100644 --- a/license/thirdpartylegalnotices.txt +++ b/license/thirdpartylegalnotices.txt @@ -231,6 +231,33 @@ Format {fmt} // without including the above copyright and permission notices. //////////////////////////////////////////////////////////////////////////// +************************************************************************************ +VDF Parser +************************************************************************************ + + // The MIT License (MIT) + // + // Copyright (c) Matthias Moeller 2016 m_moeller@live.de + // + // Permission is hereby granted, free of charge, to any person obtaining a copy + // of this software and associated documentation files (the "Software"), to deal + // in the Software without restriction, including without limitation the rights + // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + // copies of the Software, and to permit persons to whom the Software is + // furnished to do so, subject to the following conditions: + // + // The above copyright notice and this permission notice shall be included in all + // copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + // SOFTWARE. + //////////////////////////////////////////////////////////////////////////// + ************************************************************************************ Nlohmann JSON ************************************************************************************ diff --git a/r5dev/core/stdafx.h b/r5dev/core/stdafx.h index 3a4ec341..53d61f5a 100644 --- a/r5dev/core/stdafx.h +++ b/r5dev/core/stdafx.h @@ -81,12 +81,15 @@ #include "public/include/memaddr.h" #include "public/include/module.h" #include "public/include/httplib.h" +#include "public/include/vdf_parser.h" #include "core/assert.h" #include "core/termutil.h" #include "tier0/basetypes.h" #include "tier0/platform.h" +#if !defined(SDKLAUNCHER) && !defined (NETCONSOLE) #include "tier0/dbg.h" +#endif // !SDKLAUNCHER && !NETCONSOLE #if !defined(SDKLAUNCHER) && !defined (NETCONSOLE) #if !defined (DEDICATED) diff --git a/r5dev/public/include/vdf_parser.h b/r5dev/public/include/vdf_parser.h new file mode 100644 index 00000000..66fcc1c2 --- /dev/null +++ b/r5dev/public/include/vdf_parser.h @@ -0,0 +1,712 @@ +//MIT License +// +//Copyright(c) 2016 Matthias Moeller +// +//Permission is hereby granted, free of charge, to any person obtaining a copy +//of this software and associated documentation files(the "Software"), to deal +//in the Software without restriction, including without limitation the rights +//to use, copy, modify, merge, publish, distribute, sublicense, and / or sell +//copies of the Software, and to permit persons to whom the Software is +//furnished to do so, subject to the following conditions : +// +//The above copyright notice and this permission notice shall be included in all +//copies or substantial portions of the Software. +// +//THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +//IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +//FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE +//AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +//LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +//OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +//SOFTWARE. + +#ifndef __TYTI_STEAM_VDF_PARSER_H__ +#define __TYTI_STEAM_VDF_PARSER_H__ +#pragma warning( disable : 4996 ) + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +//for wstring support +#include +#include + +// internal +#include + +//VS < 2015 has only partial C++11 support +#if defined(_MSC_VER) && _MSC_VER < 1900 +#ifndef CONSTEXPR +#define CONSTEXPR +#endif + +#ifndef NOEXCEPT +#define NOEXCEPT +#endif +#else +#ifndef CONSTEXPR +#define CONSTEXPR constexpr +#define TYTI_UNDEF_CONSTEXPR +#endif + +#ifndef NOEXCEPT +#define NOEXCEPT noexcept +#define TYTI_UNDEF_NOEXCEPT +#endif + +#endif +namespace vdf +{ + namespace detail + { + /////////////////////////////////////////////////////////////////////////// + // Helper functions selecting the right encoding (char/wchar_T) + /////////////////////////////////////////////////////////////////////////// + + template + struct literal_macro_help + { + static CONSTEXPR const char* result(const char* c, const wchar_t*) NOEXCEPT + { + return c; + } + static CONSTEXPR const char result(const char c, const wchar_t) NOEXCEPT + { + return c; + } + }; + + template <> + struct literal_macro_help + { + static CONSTEXPR const wchar_t* result(const char*, const wchar_t* wc) NOEXCEPT + { + return wc; + } + static CONSTEXPR const wchar_t result(const char, const wchar_t wc) NOEXCEPT + { + return wc; + } + }; +#define TYTI_L(type, text) vdf::detail::literal_macro_help::result(text, L##text) + + inline std::string string_converter(const std::string& w) NOEXCEPT + { + return w; + } + + // utility wrapper to adapt locale-bound facets for wstring/wbuffer convert + // from cppreference + template + struct deletable_facet : Facet + { + template + deletable_facet(Args &&... args) : Facet(std::forward(args)...) {} + ~deletable_facet() {} + }; + + inline std::string string_converter(const std::wstring& w) //todo: use us-locale + { + std::wstring_convert>> conv1; + return conv1.to_bytes(w); + } + + /////////////////////////////////////////////////////////////////////////// + // Writer helper functions + /////////////////////////////////////////////////////////////////////////// + + template + class tabs + { + const size_t t; + + public: + explicit CONSTEXPR tabs(size_t i) NOEXCEPT : t(i) {} + std::basic_string print() const { return std::basic_string(t, TYTI_L(charT, '\t')); } + inline CONSTEXPR tabs operator+(size_t i) const NOEXCEPT + { + return tabs(t + i); + } + }; + + template + oStreamT& operator<<(oStreamT& s, const tabs t) + { + s << t.print(); + return s; + } + } // end namespace detail + + /////////////////////////////////////////////////////////////////////////// + // Interface + /////////////////////////////////////////////////////////////////////////// + + /// custom objects and their corresponding write functions + + /// basic object node. Every object has a name and can contains attributes saved as key_value pairs or childrens + template + struct basic_object + { + typedef CharT char_type; + std::basic_string name; + std::unordered_map, std::basic_string> attribs; + std::unordered_map, std::shared_ptr>> childs; + + void add_attribute(std::basic_string key, std::basic_string value) + { + attribs.emplace(std::move(key), std::move(value)); + } + void add_child(std::unique_ptr> child) + { + std::shared_ptr> obj{ child.release() }; + childs.emplace(obj->name, obj); + } + void set_name(std::basic_string n) + { + name = std::move(n); + } + }; + + template + struct basic_multikey_object + { + typedef CharT char_type; + std::basic_string name; + std::unordered_multimap, std::basic_string> attribs; + std::unordered_multimap, std::shared_ptr>> childs; + + void add_attribute(std::basic_string key, std::basic_string value) + { + attribs.emplace(std::move(key), std::move(value)); + } + void add_child(std::unique_ptr> child) + { + std::shared_ptr> obj{ child.release() }; + childs.emplace(obj->name, obj); + } + void set_name(std::basic_string n) + { + name = std::move(n); + } + }; + + typedef basic_object object; + typedef basic_object wobject; + typedef basic_multikey_object multikey_object; + typedef basic_multikey_object wmultikey_object; + + struct Options + { + bool strip_escape_symbols; + bool ignore_all_platform_conditionals; + bool ignore_includes; + + Options() : strip_escape_symbols(true), ignore_all_platform_conditionals(false), ignore_includes(false) {} + }; + + //forward decls + //forward decl + template + OutputT read(iStreamT& inStream, const Options& opt = Options{}); + + /** \brief writes given object tree in vdf format to given stream. + Output is prettyfied, using tabs + */ + template + void write(oStreamT& s, const T& r, + const detail::tabs tab = detail::tabs(0)) + { + typedef typename oStreamT::char_type charT; + using namespace detail; + s << tab << TYTI_L(charT, '"') << r.name << TYTI_L(charT, "\"\n") << tab << TYTI_L(charT, "{\n"); + for (const auto& i : r.attribs) + s << tab + 1 << TYTI_L(charT, '"') << i.first << TYTI_L(charT, "\"\t\t\"") << i.second << TYTI_L(charT, "\"\n"); + for (const auto& i : r.childs) + if (i.second) + write(s, *i.second, tab + 1); + s << tab << TYTI_L(charT, "}\n"); + } + + namespace detail + { + template + std::basic_string read_file(iStreamT& inStream) + { + // cache the file + typedef typename iStreamT::char_type charT; + std::basic_string str; + inStream.seekg(0, std::ios::end); + str.resize(static_cast(inStream.tellg())); + if (str.empty()) + return str; + + inStream.seekg(0, std::ios::beg); + inStream.read(&str[0], str.size()); + return str; + } + + /** \brief Read VDF formatted sequences defined by the range [first, last). + If the file is mailformatted, parser will try to read it until it can. + @param first begin iterator + @param end end iterator + @param exclude_files list of files which cant be included anymore. + prevents circular includes + + can thow: + - "std::runtime_error" if a parsing error occured + - "std::bad_alloc" if not enough memory coup be allocated + */ + template + std::vector> read_internal(IterT first, const IterT last, + std::unordered_set::value_type>>& exclude_files, + const Options& opt) + { + static_assert(std::is_default_constructible::value, + "Output Type must be default constructible (provide constructor without arguments)"); + static_assert(std::is_move_constructible::value, + "Output Type must be move constructible"); + + typedef typename std::iterator_traits::value_type charT; + + const std::basic_string comment_end_str = TYTI_L(charT, "*/"); + const std::basic_string whitespaces = TYTI_L(charT, " \n\v\f\r\t"); + +#ifdef WIN32 + std::function&)> is_platform_str = [](const std::basic_string& in) { + return in == TYTI_L(charT, "$WIN32") || in == TYTI_L(charT, "$WINDOWS"); + }; +#elif __APPLE__ + // WIN32 stands for pc in general + std::function&)> is_platform_str = [](const std::basic_string& in) { + return in == TYTI_L(charT, "$WIN32") || in == TYTI_L(charT, "$POSIX") || in == TYTI_L(charT, "$OSX"); + }; + +#elif __linux__ + // WIN32 stands for pc in general + std::function&)> is_platform_str = [](const std::basic_string& in) { + return in == TYTI_L(charT, "$WIN32") || in == TYTI_L(charT, "$POSIX") || in == TYTI_L(charT, "$LINUX"); + }; +#else + std::function&)> is_platform_str = [](const std::basic_string& in) { + return false; + }; +#endif + + if (opt.ignore_all_platform_conditionals) + is_platform_str = [](const std::basic_string&) { + return false; + }; + + // function for skipping a comment block + // iter: iterator poition to the position after a '/' + auto skip_comments = [&comment_end_str](IterT iter, const IterT& last) -> IterT { + ++iter; + if (iter != last) + { + if (*iter == TYTI_L(charT, '/')) + { + // line comment, skip whole line + iter = std::find(iter + 1, last, TYTI_L(charT, '\n')); + } + + if (*iter == '*') + { + // block comment, skip until next occurance of "*\" + iter = std::search(iter + 1, last, std::begin(comment_end_str), std::end(comment_end_str)); + iter += 2; + } + } + return iter; + }; + + auto end_quote = [](IterT iter, const IterT& last) -> IterT { + const auto begin = iter; + auto last_esc = iter; + do + { + ++iter; + iter = std::find(iter, last, TYTI_L(charT, '\"')); + if (iter == last) + break; + + last_esc = std::prev(iter); + while (last_esc != begin && *last_esc == '\\') + --last_esc; + } while (!(std::distance(last_esc, iter) % 2)); + if (iter == last) + throw std::runtime_error{ "quote was opened but not closed." }; + return iter; + }; + + auto end_word = [&whitespaces](IterT iter, const IterT& last) -> IterT { + const auto begin = iter; + auto last_esc = iter; + do + { + ++iter; + iter = std::find_first_of(iter, last, std::begin(whitespaces), std::end(whitespaces)); + if (iter == last) + break; + + last_esc = std::prev(iter); + while (last_esc != begin && *last_esc == '\\') + --last_esc; + } while (!(std::distance(last_esc, iter) % 2)); + //if (iter == last) + // throw std::runtime_error{ "word wasnt properly ended" }; + return iter; + }; + + auto skip_whitespaces = [&whitespaces](IterT iter, const IterT& last) -> IterT { + iter = std::find_if_not(iter, last, [&whitespaces](charT c) { + // return true if whitespace + return std::any_of(std::begin(whitespaces), std::end(whitespaces), [c](charT pc) { return pc == c; }); + }); + return iter; + }; + + std::function&)> strip_escape_symbols = [](std::basic_string& s) { + auto quote_searcher = [&s](size_t pos) { return s.find(TYTI_L(charT, "\\\""), pos); }; + auto p = quote_searcher(0); + while (p != s.npos) + { + s.replace(p, 2, TYTI_L(charT, "\"")); + p = quote_searcher(p); + } + auto searcher = [&s](size_t pos) { return s.find(TYTI_L(charT, "\\\\"), pos); }; + p = searcher(0); + while (p != s.npos) + { + s.replace(p, 2, TYTI_L(charT, "\\")); + p = searcher(p); + } + }; + + if (!opt.strip_escape_symbols) + strip_escape_symbols = [](std::basic_string&) {}; + + auto conditional_fullfilled = [&skip_whitespaces, &is_platform_str](IterT& iter, const IterT& last) { + iter = skip_whitespaces(iter, last); + if (*iter == '[') + { + ++iter; + const auto end = std::find(iter, last, ']'); + const bool negate = *iter == '!'; + if (negate) + ++iter; + auto conditional = std::basic_string(iter, end); + + const bool is_platform = is_platform_str(conditional); + iter = end + 1; + + return static_cast(is_platform ^ negate); + } + return true; + }; + + //read header + // first, quoted name + std::unique_ptr curObj = nullptr; + std::vector> roots; + std::stack> lvls; + auto curIter = first; + + while (curIter != last && *curIter != '\0') + { + //find first starting attrib/child, or ending + curIter = skip_whitespaces(curIter, last); + if (curIter == last || *curIter == '\0') + break; + if (*curIter == TYTI_L(charT, '/')) + { + curIter = skip_comments(curIter, last); + } + else if (*curIter != TYTI_L(charT, '}')) + { + + // get key + const auto keyEnd = (*curIter == TYTI_L(charT, '\"')) ? end_quote(curIter, last) : end_word(curIter, last); + if (*curIter == TYTI_L(charT, '\"')) + ++curIter; + std::basic_string key(curIter, keyEnd); + strip_escape_symbols(key); + curIter = keyEnd + ((*keyEnd == TYTI_L(charT, '\"')) ? 1 : 0); + + curIter = skip_whitespaces(curIter, last); + + auto conditional = conditional_fullfilled(curIter, last); + if (!conditional) + continue; + + while (*curIter == TYTI_L(charT, '/')) + { + + curIter = skip_comments(curIter, last); + if (curIter == last || *curIter == '}') + throw std::runtime_error{ "key declared, but no value" }; + curIter = skip_whitespaces(curIter, last); + if (curIter == last || *curIter == '}') + throw std::runtime_error{ "key declared, but no value" }; + } + // get value + if (*curIter != '{') + { + const auto valueEnd = (*curIter == TYTI_L(charT, '\"')) ? end_quote(curIter, last) : end_word(curIter, last); + if (*curIter == TYTI_L(charT, '\"')) + ++curIter; + + auto value = std::basic_string(curIter, valueEnd); + strip_escape_symbols(value); + curIter = valueEnd + ((*valueEnd == TYTI_L(charT, '\"')) ? 1 : 0); + + auto conditional = conditional_fullfilled(curIter, last); + if (!conditional) + continue; + + // process value + if (key != TYTI_L(charT, "#include") && key != TYTI_L(charT, "#base")) + { + curObj->add_attribute(std::move(key), std::move(value)); + } + else + { + if (!opt.ignore_includes && exclude_files.find(value) == exclude_files.end()) + { + exclude_files.insert(value); + std::basic_ifstream i(detail::string_converter(value)); + auto str = read_file(i); + auto file_objs = read_internal(str.begin(), str.end(), exclude_files, opt); + for (auto& n : file_objs) + { + if (curObj) + curObj->add_child(std::move(n)); + else + roots.push_back(std::move(n)); + } + exclude_files.erase(value); + } + } + } + else if (*curIter == '{') + { + if (curObj) + lvls.push(std::move(curObj)); + curObj = std::make_unique(); + curObj->set_name(std::move(key)); + ++curIter; + } + } + //end of new object + else if (*curIter == TYTI_L(charT, '}')) + { + if (!lvls.empty()) + { + //get object before + std::unique_ptr prev{ std::move(lvls.top()) }; + lvls.pop(); + + // add finished obj to obj before and release it from processing + prev->add_child(std::move(curObj)); + curObj = std::move(prev); + } + else + { + roots.push_back(std::move(curObj)); + curObj.reset(); + } + ++curIter; + } + } + return roots; + } + + } // namespace detail + + /** \brief Read VDF formatted sequences defined by the range [first, last). + If the file is mailformatted, parser will try to read it until it can. + @param first begin iterator + @param end end iterator + + can thow: + - "std::runtime_error" if a parsing error occured + - "std::bad_alloc" if not enough memory coup be allocated + */ + template + OutputT read(IterT first, const IterT last, const Options& opt = Options{}) + { + auto exclude_files = std::unordered_set::value_type>>{}; + auto roots = detail::read_internal(first, last, exclude_files, opt); + + OutputT result; + if (roots.size() > 1) + { + for (auto& i : roots) + result.add_child(std::move(i)); + } + else if (roots.size() == 1) + result = std::move(*roots[0]); + + return result; + } + + /** \brief Read VDF formatted sequences defined by the range [first, last). + If the file is mailformatted, parser will try to read it until it can. + @param first begin iterator + @param end end iterator + @param ec output bool. 0 if ok, otherwise, holds an system error code + + Possible error codes: + std::errc::protocol_error: file is mailformatted + std::errc::not_enough_memory: not enough space + std::errc::invalid_argument: iterators throws e.g. out of range + */ + template + OutputT read(IterT first, IterT last, std::error_code& ec, const Options& opt = Options{}) NOEXCEPT + + { + ec.clear(); + OutputT r{}; + try + { + r = read(first, last, opt); + } + catch (std::runtime_error&) + { + ec = std::make_error_code(std::errc::protocol_error); + } + catch (std::bad_alloc&) + { + ec = std::make_error_code(std::errc::not_enough_memory); + } + catch (...) + { + ec = std::make_error_code(std::errc::invalid_argument); + } + return r; + } + + /** \brief Read VDF formatted sequences defined by the range [first, last). + If the file is mailformatted, parser will try to read it until it can. + @param first begin iterator + @param end end iterator + @param ok output bool. true, if parser successed, false, if parser failed + */ + template + OutputT read(IterT first, const IterT last, bool* ok, const Options& opt = Options{}) NOEXCEPT + { + std::error_code ec; + auto r = read(first, last, ec, opt); + if (ok) + *ok = !ec; + return r; + } + + template + inline auto read(IterT first, const IterT last, bool* ok, const Options& opt = Options{}) NOEXCEPT -> basic_object::value_type> + { + return read::value_type>>(first, last, ok, opt); + } + + template + inline auto read(IterT first, IterT last, std::error_code& ec, const Options& opt = Options{}) NOEXCEPT + -> basic_object::value_type> + { + return read::value_type>>(first, last, ec, opt); + } + + template + inline auto read(IterT first, const IterT last, const Options& opt = Options{}) + -> basic_object::value_type> + { + return read::value_type>>(first, last, opt); + } + + /** \brief Loads a stream (e.g. filestream) into the memory and parses the vdf formatted data. + throws "std::bad_alloc" if file buffer could not be allocated + */ + template + OutputT read(iStreamT& inStream, std::error_code& ec, const Options& opt = Options{}) + { + // cache the file + typedef typename iStreamT::char_type charT; + std::basic_string str = detail::read_file(inStream); + + // parse it + return read(str.begin(), str.end(), ec, opt); + } + + template + inline basic_object read(iStreamT& inStream, std::error_code& ec, const Options& opt = Options{}) + { + return read>(inStream, ec, opt); + } + + /** \brief Loads a stream (e.g. filestream) into the memory and parses the vdf formatted data. + throws "std::bad_alloc" if file buffer could not be allocated + ok == false, if a parsing error occured + */ + template + OutputT read(iStreamT& inStream, bool* ok, const Options& opt = Options{}) + { + std::error_code ec; + const auto r = read(inStream, ec, opt); + if (ok) + *ok = !ec; + return r; + } + + template + inline basic_object read(iStreamT& inStream, bool* ok, const Options& opt = Options{}) + { + return read>(inStream, ok, opt); + } + + /** \brief Loads a stream (e.g. filestream) into the memory and parses the vdf formatted data. + throws "std::bad_alloc" if file buffer could not be allocated + throws "std::runtime_error" if a parsing error occured + */ + template + OutputT read(iStreamT& inStream, const Options& opt) + { + + // cache the file + typedef typename iStreamT::char_type charT; + std::basic_string str = detail::read_file(inStream); + // parse it + return read(str.begin(), str.end(), opt); + } + + template + inline basic_object read(iStreamT& inStream, const Options& opt = Options{}) + { + return read>(inStream, opt); + } + +} // namespace vdf + +#ifndef TYTI_NO_L_UNDEF +#undef TYTI_L +#endif + +#ifdef TYTI_UNDEF_CONSTEXPR +#undef CONSTEXPR +#undef TYTI_NO_L_UNDEF +#endif + +#ifdef TYTI_UNDEF_NOTHROW +#undef NOTHROW +#undef TYTI_UNDEF_NOTHROW +#endif + +#endif //__TYTI_STEAM_VDF_PARSER_H__ diff --git a/r5dev/sdklauncher/basepanel.cpp b/r5dev/sdklauncher/basepanel.cpp index 47aef8e6..64eda47a 100644 --- a/r5dev/sdklauncher/basepanel.cpp +++ b/r5dev/sdklauncher/basepanel.cpp @@ -1,9 +1,16 @@ - +//=============================================================================// +// +// Purpose: +// +//=============================================================================// #include "core/stdafx.h" #include "sdklauncher.h" #include "basepanel.h" -void CUIBasePanel::Init() +//----------------------------------------------------------------------------- +// Purpose: creates the surface layout +//----------------------------------------------------------------------------- +void CUIBaseSurface::Init() { // START DESIGNER CODE const INT WindowX = 800; @@ -18,10 +25,11 @@ void CUIBasePanel::Init() this->SetStartPosition(Forms::FormStartPosition::CenterParent); this->SetMinimizeBox(false); this->SetMaximizeBox(false); + this->SetBackColor(Drawing::Color(47, 54, 61)); - // ################################################################################################# - // - // ################################################################################################# + // ######################################################################## + // GAME + // ######################################################################## this->m_GameGroup = new UIX::UIXGroupBox(); this->m_GameGroup->SetSize({ 458, 84 }); this->m_GameGroup->SetLocation({ 12, 10 }); @@ -54,29 +62,6 @@ void CUIBasePanel::Init() this->m_MapCombo->SetSelectedIndex(0); this->m_MapCombo->SetAnchor(Forms::AnchorStyles::Top | Forms::AnchorStyles::Left); this->m_MapCombo->SetDropDownStyle(Forms::ComboBoxStyle::DropDownList); - std::regex rgArchiveRegex{ R"([^_]*_(.*)(.bsp.pak000_dir).*)" }; - std::smatch smRegexMatches; - - for (const auto& dEntry : fs::directory_iterator("vpk")) - { - std::string svFileName = dEntry.path().string(); - std::regex_search(svFileName, smRegexMatches, rgArchiveRegex); - - if (smRegexMatches.size() > 0) - { - if (strcmp(smRegexMatches[1].str().c_str(), "frontend") == 0) - { - continue; - } - else if (strcmp(smRegexMatches[1].str().c_str(), "mp_common") == 0) - { - this->m_MapCombo->Items.Add("mp_lobby"); - continue; - } - - this->m_MapCombo->Items.Add(smRegexMatches[1].str().c_str()); - } - } this->m_GameGroup->AddControl(this->m_MapCombo); this->m_PlaylistLabel = new UIX::UIXLabel(); @@ -136,6 +121,7 @@ void CUIBasePanel::Init() this->m_PlaylistFileTextBox->SetTabIndex(0); this->m_PlaylistFileTextBox->SetText("playlists_r5_patch.txt"); this->m_PlaylistFileTextBox->SetAnchor(Forms::AnchorStyles::Top | Forms::AnchorStyles::Left); + this->m_PlaylistFileTextBox->LostFocus += &ReloadPlaylists; this->m_GameGroupExt->AddControl(this->m_PlaylistFileTextBox); this->m_PlaylistFileLabel = new UIX::UIXLabel(); @@ -146,9 +132,9 @@ void CUIBasePanel::Init() this->m_PlaylistFileLabel->SetAnchor(Forms::AnchorStyles::Bottom | Forms::AnchorStyles::Left); this->m_GameGroupExt->AddControl(this->m_PlaylistFileLabel); - // ################################################################################################# - // - // ################################################################################################# + // ######################################################################## + // MAIN + // ######################################################################## this->m_MainGroup = new UIX::UIXGroupBox(); this->m_MainGroup->SetSize({ 308, 84 }); this->m_MainGroup->SetLocation({ 480, 10 }); @@ -172,9 +158,6 @@ void CUIBasePanel::Init() this->m_ModeCombo->SetSelectedIndex(0); this->m_ModeCombo->SetAnchor(Forms::AnchorStyles::Top | Forms::AnchorStyles::Left); this->m_ModeCombo->SetDropDownStyle(Forms::ComboBoxStyle::DropDownList); - this->m_ModeCombo->Items.Add("Host"); - this->m_ModeCombo->Items.Add("Server"); - this->m_ModeCombo->Items.Add("Client"); this->m_MainGroup->AddControl(this->m_ModeCombo); this->m_ModeLabel = new UIX::UIXLabel(); @@ -209,9 +192,6 @@ void CUIBasePanel::Init() this->m_VisibilityCombo->SetSelectedIndex(0); this->m_VisibilityCombo->SetAnchor(Forms::AnchorStyles::Top | Forms::AnchorStyles::Left); this->m_VisibilityCombo->SetDropDownStyle(Forms::ComboBoxStyle::DropDownList); - this->m_VisibilityCombo->Items.Add("Public"); - this->m_VisibilityCombo->Items.Add("Hidden"); - this->m_VisibilityCombo->Items.Add("Offline"); this->m_MainGroup->AddControl(this->m_VisibilityCombo); this->m_VisibilityLabel = new UIX::UIXLabel(); @@ -265,9 +245,9 @@ void CUIBasePanel::Init() this->m_LaunchSDK->Click += &LaunchGame; this->m_MainGroupExt->AddControl(this->m_LaunchSDK); - // ################################################################################################# - // - // ################################################################################################# + // ######################################################################## + // ENGINE + // ######################################################################## this->m_EngineBaseGroup = new UIX::UIXGroupBox(); this->m_EngineBaseGroup->SetSize({ 337, 73 }); this->m_EngineBaseGroup->SetLocation({ 12, 158 }); @@ -443,9 +423,9 @@ void CUIBasePanel::Init() this->m_ResolutionLabel->SetTextAlign(Drawing::ContentAlignment::TopLeft); this->m_EngineVideoGroup->AddControl(this->m_ResolutionLabel); - // ################################################################################################# - // - // ################################################################################################# + // ######################################################################## + // CONSOLE + // ######################################################################## this->m_ConsoleGroup = new UIX::UIXGroupBox(); this->m_ConsoleGroup->SetSize({ 429, 181 }); this->m_ConsoleGroup->SetLocation({ 359, 158 }); @@ -466,322 +446,414 @@ void CUIBasePanel::Init() this->ResumeLayout(false); this->PerformLayout(); // END DESIGNER CODE - - this->SetBackColor(Drawing::Color(47, 54, 61)); } -void CUIBasePanel::LaunchGame(Forms::Control* pSender) +//----------------------------------------------------------------------------- +// Purpose: post-init surface setup +//----------------------------------------------------------------------------- +void CUIBaseSurface::Setup() +{ + this->ParseMaps(); + this->ParsePlaylists(); + + this->m_ModeCombo->Items.Add("Host"); + this->m_ModeCombo->Items.Add("Server"); + this->m_ModeCombo->Items.Add("Client"); + + this->m_VisibilityCombo->Items.Add("Public"); + this->m_VisibilityCombo->Items.Add("Hidden"); + this->m_VisibilityCombo->Items.Add("Offline"); +} + +//----------------------------------------------------------------------------- +// Purpose: parses all available maps from the main vpk directory +//----------------------------------------------------------------------------- +void CUIBaseSurface::ParseMaps() +{ + std::regex rgArchiveRegex{ R"([^_]*_(.*)(.bsp.pak000_dir).*)" }; + std::smatch smRegexMatches; + for (const auto& dEntry : fs::directory_iterator("vpk")) + { + std::string svFileName = dEntry.path().string(); + std::regex_search(svFileName, smRegexMatches, rgArchiveRegex); + + if (smRegexMatches.size() > 0) + { + if (strcmp(smRegexMatches[1].str().c_str(), "frontend") == 0) + { + continue; + } + else if (strcmp(smRegexMatches[1].str().c_str(), "mp_common") == 0) + { + this->m_MapCombo->Items.Add("mp_lobby"); + continue; + } + + this->m_MapCombo->Items.Add(smRegexMatches[1].str().c_str()); + } + } +} + +//----------------------------------------------------------------------------- +// Purpose: parses all playlists from user selected playlist file +//----------------------------------------------------------------------------- +void CUIBaseSurface::ParsePlaylists() +{ + const string svBaseDir = "platform\\"; + fs::path fsPlaylistPath(svBaseDir + this->m_PlaylistFileTextBox->Text().ToCString()); + + if (fs::exists(fsPlaylistPath)) + { + bool bOk{ }; + std::ifstream iFile(fsPlaylistPath); + vdf::object vRoot = vdf::read(iFile, &bOk); + + if (bOk) + { + const auto& vcPlaylists = vRoot.childs.at("Playlists"); + for (auto [id, it] = std::tuple{ 1, vcPlaylists->childs.begin()}; it != vcPlaylists->childs.end(); id++, it++) + { + this->m_PlaylistCombo->Items.Add(it->first.c_str()); + if (strcmp(it->first.c_str(), "dev_default")) + { + this->m_PlaylistCombo->SetSelectedIndex(id); + } + } + } + } +} + +//----------------------------------------------------------------------------- +// Purpose: launches the game with the SDK +// Input : *pSender - +//----------------------------------------------------------------------------- +void CUIBaseSurface::LaunchGame(Forms::Control* pSender) { string svParameter = "-launcher -dev "; - eLaunchMode launchMode = eLaunchMode::LM_NULL; + eLaunchMode launchMode = eLaunchMode::LM_NONE; launchMode = g_pLauncher->GetMainSurface()->BuildParameter(svParameter); - printf("%s\n", svParameter.c_str()); - printf("launchMode %d\n", launchMode); - g_pLauncher->Setup(launchMode, svParameter); g_pLauncher->Launch(); } -eLaunchMode CUIBasePanel::BuildParameter(string& svParameter) +//----------------------------------------------------------------------------- +// Purpose: clears the form and reloads the playlist +// Input : *pSender - +//----------------------------------------------------------------------------- +void CUIBaseSurface::ReloadPlaylists(Forms::Control* pSender) { - eLaunchMode results = eLaunchMode::LM_NULL; + CUIBaseSurface* pSurface = reinterpret_cast(pSender->FindForm()); + + pSurface->m_PlaylistCombo->Items.Clear(); + pSurface->m_PlaylistCombo->OnSizeChanged(); + pSurface->ParsePlaylists(); +} + +//----------------------------------------------------------------------------- +// Purpose: clears the form and reloads the playlist +// Input : &svParameters - +// Output : eLaunchMode [HOST - SERVER - CLIENT - NONE] +//----------------------------------------------------------------------------- +eLaunchMode CUIBaseSurface::BuildParameter(string& svParameters) +{ + eLaunchMode results = eLaunchMode::LM_NONE; switch (static_cast(this->m_ModeCombo->SelectedIndex())) { case eMode::HOST: { - // GAME ############################################################################################ + // GAME ############################################################### if (!String::IsNullOrEmpty(this->m_MapCombo->Text())) { - svParameter.append("+map \"" + this->m_MapCombo->Text() + "\" "); + svParameters.append("+map \"" + this->m_MapCombo->Text() + "\" "); } if (!String::IsNullOrEmpty(this->m_PlaylistCombo->Text())) { - svParameter.append("+launchplaylist \"" + this->m_PlaylistCombo->Text() + "\" "); + svParameters.append("+launchplaylist \"" + this->m_PlaylistCombo->Text() + "\" "); } if (this->m_DevelopmentToggle->Checked()) { - svParameter.append("-devsdk "); + svParameters.append("-devsdk "); results = eLaunchMode::LM_HOST_DEBUG; } else results = eLaunchMode::LM_HOST; if (this->m_CheatsToggle->Checked()) - svParameter.append("+sv_cheats \"1\" "); + svParameters.append("+sv_cheats \"1\" "); if (this->m_ConsoleToggle->Checked()) - svParameter.append("-wconsole "); + svParameters.append("-wconsole "); if (this->m_ColorConsoleToggle->Checked()) - svParameter.append("-ansiclr "); + svParameters.append("-ansiclr "); if (!String::IsNullOrEmpty(this->m_PlaylistFileTextBox->Text())) - svParameter.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); + svParameters.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); - // ENGINE ########################################################################################## + // ENGINE ############################################################### if (StringIsDigit(this->m_ReservedCoresTextBox->Text().ToCString())) - svParameter.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); + svParameters.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); //else error; if (StringIsDigit(this->m_WorkerThreadsTextBox->Text().ToCString())) - svParameter.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); + svParameters.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); //else error; if (this->m_SingleCoreDediToggle->Checked()) - svParameter.append("+sv_single_core_dedi \"1\" "); + svParameters.append("+sv_single_core_dedi \"1\" "); if (this->m_NoAsyncJobsToggle->Checked()) { - svParameter.append("-noasync "); - svParameter.append("+async_serialize \"0\" "); - svParameter.append("+buildcubemaps_async \"0\" "); - svParameter.append("+sv_asyncAIInit \"0\" "); - svParameter.append("+sv_asyncSendSnapshot \"0\" "); - svParameter.append("+sv_scriptCompileAsync \"0\" "); - svParameter.append("+cl_scriptCompileAsync \"0\" "); - svParameter.append("+cl_async_bone_setup \"0\" "); - svParameter.append("+cl_updatedirty_async \"0\" "); - svParameter.append("+mat_syncGPU \"1\" "); - svParameter.append("+mat_sync_rt \"1\" "); - svParameter.append("+mat_sync_rt_flushes_gpu \"1\" "); - svParameter.append("+net_async_sendto \"0\" "); - svParameter.append("+physics_async_sv \"0\" "); - svParameter.append("+physics_async_cl \"0\" "); + svParameters.append("-noasync "); + svParameters.append("+async_serialize \"0\" "); + svParameters.append("+buildcubemaps_async \"0\" "); + svParameters.append("+sv_asyncAIInit \"0\" "); + svParameters.append("+sv_asyncSendSnapshot \"0\" "); + svParameters.append("+sv_scriptCompileAsync \"0\" "); + svParameters.append("+cl_scriptCompileAsync \"0\" "); + svParameters.append("+cl_async_bone_setup \"0\" "); + svParameters.append("+cl_updatedirty_async \"0\" "); + svParameters.append("+mat_syncGPU \"1\" "); + svParameters.append("+mat_sync_rt \"1\" "); + svParameters.append("+mat_sync_rt_flushes_gpu \"1\" "); + svParameters.append("+net_async_sendto \"0\" "); + svParameters.append("+physics_async_sv \"0\" "); + svParameters.append("+physics_async_cl \"0\" "); } if (this->m_NetEncryptionToggle->Checked()) - svParameter.append("+net_encryptionEnable \"1\" "); + svParameters.append("+net_encryptionEnable \"1\" "); if (this->m_NetRandomKeyToggle->Checked()) - svParameter.append("+net_useRandomKey \"1\" "); + svParameters.append("+net_useRandomKey \"1\" "); if (this->m_NoQueuedPacketThread->Checked()) - svParameter.append("+net_queued_packet_thread \"0\" "); + svParameters.append("+net_queued_packet_thread \"0\" "); if (this->m_NoTimeOutToggle->Checked()) - svParameter.append("-notimeout "); + svParameters.append("-notimeout "); if (this->m_WindowedToggle->Checked()) - svParameter.append("-windowed "); + svParameters.append("-windowed "); if (this->m_NoBorderToggle->Checked()) - svParameter.append("-noborder "); + svParameters.append("-noborder "); if (StringIsDigit(this->m_FpsTextBox->Text().ToCString())) - svParameter.append("+fps_max \"" + this->m_FpsTextBox->Text() + "\" "); + svParameters.append("+fps_max \"" + this->m_FpsTextBox->Text() + "\" "); if (!String::IsNullOrEmpty(this->m_WidthTextBox->Text())) - svParameter.append("-w \"" + this->m_WidthTextBox->Text() + "\" "); + svParameters.append("-w \"" + this->m_WidthTextBox->Text() + "\" "); if (!String::IsNullOrEmpty(this->m_HeightTextBox->Text())) - svParameter.append("-h \"" + this->m_HeightTextBox->Text() + "\" "); + svParameters.append("-h \"" + this->m_HeightTextBox->Text() + "\" "); - // MAIN ############################################################################################ + // MAIN ############################################################### if (!String::IsNullOrEmpty(this->m_HostNameTextBox->Text())) { - svParameter.append("+sv_pylonHostName \"" + this->m_HostNameTextBox->Text() + "\" "); + svParameters.append("+sv_pylonHostName \"" + this->m_HostNameTextBox->Text() + "\" "); switch (static_cast(this->m_VisibilityCombo->SelectedIndex())) { case eVisibility::PUBLIC: { - svParameter.append("+sv_pylonVisibility \"2\" "); + svParameters.append("+sv_pylonVisibility \"2\" "); break; } case eVisibility::HIDDEN: { - svParameter.append("+sv_pylonVisibility \"1\" "); + svParameters.append("+sv_pylonVisibility \"1\" "); break; } default: { - svParameter.append("+sv_pylonVisibility \"0\" "); + svParameters.append("+sv_pylonVisibility \"0\" "); break; } } } if (!String::IsNullOrEmpty(this->m_LaunchArgsTextBox->Text())) - svParameter.append(this->m_LaunchArgsTextBox->Text()); + svParameters.append(this->m_LaunchArgsTextBox->Text()); return results; } case eMode::SERVER: { - // GAME ############################################################################################ + // GAME ############################################################### if (!String::IsNullOrEmpty(this->m_MapCombo->Text())) { - svParameter.append("+map \"" + this->m_MapCombo->Text() + "\" "); + svParameters.append("+map \"" + this->m_MapCombo->Text() + "\" "); } if (!String::IsNullOrEmpty(this->m_PlaylistCombo->Text())) { - svParameter.append("+launchplaylist \"" + this->m_PlaylistCombo->Text() + "\" "); + svParameters.append("+launchplaylist \"" + this->m_PlaylistCombo->Text() + "\" "); } if (this->m_DevelopmentToggle->Checked()) { - svParameter.append("-devsdk "); + svParameters.append("-devsdk "); results = eLaunchMode::LM_SERVER_DEBUG; } else results = eLaunchMode::LM_SERVER; if (this->m_CheatsToggle->Checked()) - svParameter.append("+sv_cheats \"1\" "); + svParameters.append("+sv_cheats \"1\" "); if (this->m_ConsoleToggle->Checked()) - svParameter.append("-wconsole "); + svParameters.append("-wconsole "); if (this->m_ColorConsoleToggle->Checked()) - svParameter.append("-ansiclr "); + svParameters.append("-ansiclr "); if (!String::IsNullOrEmpty(this->m_PlaylistFileTextBox->Text())) - svParameter.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); + svParameters.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); - // ENGINE ########################################################################################## + // ENGINE ############################################################### if (StringIsDigit(this->m_ReservedCoresTextBox->Text().ToCString())) - svParameter.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); + svParameters.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); //else error; if (StringIsDigit(this->m_WorkerThreadsTextBox->Text().ToCString())) - svParameter.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); + svParameters.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); //else error; if (this->m_SingleCoreDediToggle->Checked()) - svParameter.append("+sv_single_core_dedi \"1\" "); + svParameters.append("+sv_single_core_dedi \"1\" "); if (this->m_NoAsyncJobsToggle->Checked()) { - svParameter.append("-noasync "); - svParameter.append("+async_serialize \"0\" "); - svParameter.append("+sv_asyncAIInit \"0\" "); - svParameter.append("+sv_asyncSendSnapshot \"0\" "); - svParameter.append("+sv_scriptCompileAsync \"0\" "); - svParameter.append("+physics_async_sv \"0\" "); + svParameters.append("-noasync "); + svParameters.append("+async_serialize \"0\" "); + svParameters.append("+sv_asyncAIInit \"0\" "); + svParameters.append("+sv_asyncSendSnapshot \"0\" "); + svParameters.append("+sv_scriptCompileAsync \"0\" "); + svParameters.append("+physics_async_sv \"0\" "); } if (this->m_NetEncryptionToggle->Checked()) - svParameter.append("+net_encryptionEnable \"1\" "); + svParameters.append("+net_encryptionEnable \"1\" "); if (this->m_NetRandomKeyToggle->Checked()) - svParameter.append("+net_useRandomKey \"1\" "); + svParameters.append("+net_useRandomKey \"1\" "); if (this->m_NoQueuedPacketThread->Checked()) - svParameter.append("+net_queued_packet_thread \"0\" "); + svParameters.append("+net_queued_packet_thread \"0\" "); if (this->m_NoTimeOutToggle->Checked()) - svParameter.append("-notimeout "); + svParameters.append("-notimeout "); - // MAIN ############################################################################################ + // MAIN ############################################################### if (!String::IsNullOrEmpty(this->m_HostNameTextBox->Text())) { - svParameter.append("+sv_pylonHostName \"" + this->m_HostNameTextBox->Text() + "\" "); + svParameters.append("+sv_pylonHostName \"" + this->m_HostNameTextBox->Text() + "\" "); switch (static_cast(this->m_VisibilityCombo->SelectedIndex())) { case eVisibility::PUBLIC: { - svParameter.append("+sv_pylonVisibility \"2\" "); + svParameters.append("+sv_pylonVisibility \"2\" "); break; } case eVisibility::HIDDEN: { - svParameter.append("+sv_pylonVisibility \"1\" "); + svParameters.append("+sv_pylonVisibility \"1\" "); break; } default: { - svParameter.append("+sv_pylonVisibility \"0\" "); + svParameters.append("+sv_pylonVisibility \"0\" "); break; } } } if (!String::IsNullOrEmpty(this->m_LaunchArgsTextBox->Text())) - svParameter.append(this->m_LaunchArgsTextBox->Text()); + svParameters.append(this->m_LaunchArgsTextBox->Text()); return results; } case eMode::CLIENT: { - // GAME ############################################################################################ + // GAME ############################################################### if (this->m_DevelopmentToggle->Checked()) { - svParameter.append("-devsdk "); + svParameters.append("-devsdk "); results = eLaunchMode::LM_CLIENT_DEBUG; } else results = eLaunchMode::LM_CLIENT; if (this->m_CheatsToggle->Checked()) - svParameter.append("+sv_cheats \"1\" "); + svParameters.append("+sv_cheats \"1\" "); if (this->m_ConsoleToggle->Checked()) - svParameter.append("-wconsole "); + svParameters.append("-wconsole "); if (this->m_ColorConsoleToggle->Checked()) - svParameter.append("-ansiclr "); + svParameters.append("-ansiclr "); if (!String::IsNullOrEmpty(this->m_PlaylistFileTextBox->Text())) - svParameter.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); + svParameters.append("-playlistfile \"" + this->m_PlaylistFileTextBox->Text() + "\" "); - // ENGINE ########################################################################################## + // ENGINE ############################################################### if (StringIsDigit(this->m_ReservedCoresTextBox->Text().ToCString())) - svParameter.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); + svParameters.append("-numreservedcores \"" + this->m_ReservedCoresTextBox->Text() + "\" "); //else error; if (StringIsDigit(this->m_WorkerThreadsTextBox->Text().ToCString())) - svParameter.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); + svParameters.append("-numworkerthreads \"" + this->m_WorkerThreadsTextBox->Text() + "\" "); //else error; if (this->m_SingleCoreDediToggle->Checked()) - svParameter.append("+sv_single_core_dedi \"1\" "); + svParameters.append("+sv_single_core_dedi \"1\" "); if (this->m_NoAsyncJobsToggle->Checked()) { - svParameter.append("-noasync "); - svParameter.append("+async_serialize \"0\" "); - svParameter.append("+buildcubemaps_async \"0\" "); - svParameter.append("+cl_scriptCompileAsync \"0\" "); - svParameter.append("+cl_async_bone_setup \"0\" "); - svParameter.append("+cl_updatedirty_async \"0\" "); - svParameter.append("+mat_syncGPU \"1\" "); - svParameter.append("+mat_sync_rt \"1\" "); - svParameter.append("+mat_sync_rt_flushes_gpu \"1\" "); - svParameter.append("+net_async_sendto \"0\" "); - svParameter.append("+physics_async_cl \"0\" "); + svParameters.append("-noasync "); + svParameters.append("+async_serialize \"0\" "); + svParameters.append("+buildcubemaps_async \"0\" "); + svParameters.append("+cl_scriptCompileAsync \"0\" "); + svParameters.append("+cl_async_bone_setup \"0\" "); + svParameters.append("+cl_updatedirty_async \"0\" "); + svParameters.append("+mat_syncGPU \"1\" "); + svParameters.append("+mat_sync_rt \"1\" "); + svParameters.append("+mat_sync_rt_flushes_gpu \"1\" "); + svParameters.append("+net_async_sendto \"0\" "); + svParameters.append("+physics_async_cl \"0\" "); } if (this->m_NetEncryptionToggle->Checked()) - svParameter.append("+net_encryptionEnable \"1\" "); + svParameters.append("+net_encryptionEnable \"1\" "); if (this->m_NetRandomKeyToggle->Checked()) - svParameter.append("+net_useRandomKey \"1\" "); + svParameters.append("+net_useRandomKey \"1\" "); if (this->m_NoQueuedPacketThread->Checked()) - svParameter.append("+net_queued_packet_thread \"0\" "); + svParameters.append("+net_queued_packet_thread \"0\" "); if (this->m_NoTimeOutToggle->Checked()) - svParameter.append("-notimeout "); + svParameters.append("-notimeout "); if (this->m_WindowedToggle->Checked()) - svParameter.append("-windowed "); + svParameters.append("-windowed "); if (this->m_NoBorderToggle->Checked()) - svParameter.append("-noborder "); + svParameters.append("-noborder "); if (StringIsDigit(this->m_FpsTextBox->Text().ToCString())) - svParameter.append("+fps_max \"" + this->m_FpsTextBox->Text() + "\" "); + svParameters.append("+fps_max \"" + this->m_FpsTextBox->Text() + "\" "); if (!String::IsNullOrEmpty(this->m_WidthTextBox->Text())) - svParameter.append("-w \"" + this->m_WidthTextBox->Text() + "\" "); + svParameters.append("-w \"" + this->m_WidthTextBox->Text() + "\" "); if (!String::IsNullOrEmpty(this->m_HeightTextBox->Text())) - svParameter.append("-h \"" + this->m_HeightTextBox->Text() + "\" "); + svParameters.append("-h \"" + this->m_HeightTextBox->Text() + "\" "); - // MAIN ############################################################################################ + // MAIN ############################################################### if (!String::IsNullOrEmpty(this->m_LaunchArgsTextBox->Text())) - svParameter.append(this->m_LaunchArgsTextBox->Text()); + svParameters.append(this->m_LaunchArgsTextBox->Text()); return results; } @@ -790,8 +862,12 @@ eLaunchMode CUIBasePanel::BuildParameter(string& svParameter) } } -CUIBasePanel::CUIBasePanel() : Forms::Form() +//----------------------------------------------------------------------------- +// Purpose: +//----------------------------------------------------------------------------- +CUIBaseSurface::CUIBaseSurface() : Forms::Form() { this->Init(); + this->Setup(); } -CUIBasePanel* g_pMainUI; \ No newline at end of file +CUIBaseSurface* g_pMainUI; \ No newline at end of file diff --git a/r5dev/sdklauncher/basepanel.h b/r5dev/sdklauncher/basepanel.h index 86eb2bdf..21335a75 100644 --- a/r5dev/sdklauncher/basepanel.h +++ b/r5dev/sdklauncher/basepanel.h @@ -1,17 +1,21 @@ #pragma once #include "sdklauncher_const.h" -class CUIBasePanel : public Forms::Form +class CUIBaseSurface : public Forms::Form { public: - CUIBasePanel(); - virtual ~CUIBasePanel() = default; + CUIBaseSurface(); + virtual ~CUIBaseSurface() = default; private: - void Init(); + void Init(); + void Setup(); + void ParseMaps(); + void ParsePlaylists(); static void LaunchGame(Forms::Control* pSender); + static void ReloadPlaylists(Forms::Control* pSender); eLaunchMode BuildParameter(string& svParameter); diff --git a/r5dev/sdklauncher/sdklauncher.cpp b/r5dev/sdklauncher/sdklauncher.cpp index 6bbb68d5..92cdbc4b 100644 --- a/r5dev/sdklauncher/sdklauncher.cpp +++ b/r5dev/sdklauncher/sdklauncher.cpp @@ -277,8 +277,8 @@ int main(int argc, char* argv[], char* envp[]) Forms::Application::EnableVisualStyles(); UIX::UIXTheme::InitializeRenderer(new Themes::KoreTheme()); - g_pLauncher->m_pMainUI = new CUIBasePanel(); - Forms::Application::Run(g_pLauncher->m_pMainUI); + g_pLauncher->m_pSurface = new CUIBaseSurface(); + Forms::Application::Run(g_pLauncher->m_pSurface); UIX::UIXTheme::ShutdownRenderer(); } else diff --git a/r5dev/sdklauncher/sdklauncher.h b/r5dev/sdklauncher/sdklauncher.h index e2503a11..17928be9 100644 --- a/r5dev/sdklauncher/sdklauncher.h +++ b/r5dev/sdklauncher/sdklauncher.h @@ -11,15 +11,15 @@ public: } ~CLauncher() { - delete[] m_pMainUI; + delete[] m_pSurface; } bool Setup(eLaunchMode lMode, eLaunchState lState); bool Setup(eLaunchMode lMode, const string& svCommandLine); bool Launch(); - CUIBasePanel* GetMainSurface() const { return m_pMainUI; } + CUIBaseSurface* GetMainSurface() const { return m_pSurface; } - CUIBasePanel* m_pMainUI; + CUIBaseSurface* m_pSurface; private: diff --git a/r5dev/sdklauncher/sdklauncher_const.h b/r5dev/sdklauncher/sdklauncher_const.h index cd8bff09..cdd2206e 100644 --- a/r5dev/sdklauncher/sdklauncher_const.h +++ b/r5dev/sdklauncher/sdklauncher_const.h @@ -15,7 +15,7 @@ enum class eLaunchMode : int { - LM_NULL = -1, + LM_NONE = -1, LM_HOST_DEBUG, LM_HOST, LM_SERVER_DEBUG, diff --git a/r5dev/vproj/libcppkore.vcxproj b/r5dev/vproj/libcppkore.vcxproj index d6e73055..3c0b9236 100644 --- a/r5dev/vproj/libcppkore.vcxproj +++ b/r5dev/vproj/libcppkore.vcxproj @@ -131,7 +131,7 @@ false %(AdditionalIncludeDirectories) stdcpp17 - MultiThreadedDebugDLL + MultiThreadedDebug Windows