#include "shortcutcreator.h" #include "win32.hpp" #include "converter.hpp" #include "ginvoke.hpp" #include // COM stuff #include // IShellLink #include // InitPropVariantFromString #include // PKEY_AppUserModel_ID #include // At least one COM header must have been previously // included, for `winrt::create_instance` to work with the `GUID` type. #include namespace dyn { inline auto load_module(const wchar_t *const path, const char *const dbgnym) { const auto mod = ::LoadLibraryW(path); if (mod) return mod; const win32_error e{}; g_warning("failed to load %s", dbgnym); throw e; } template inline T &load_symbol( const wchar_t *const mod_path, const char *const mod_dbgnym, const char *const symbol) { const auto p = reinterpret_cast( ::GetProcAddress(load_module(mod_path, mod_dbgnym), symbol)); if (p) return *p; const win32_error e{}; g_warning("couldn't find %s in %s", symbol, mod_dbgnym); throw e; } #define dyn_load_symbol(mod_name, symbol) \ ::dyn::load_symbol(L ## mod_name, mod_name, #symbol) // PropVariantToString is a pain to use, and // MinGW 6.0.0 doesn't have libpropsys.a in the first place; // MinGW 9.0.0 doesn't have PropVariantToStringAlloc in its libpropsys.a. // So... constexpr auto PropVariantToStringAlloc = [](const auto &... arg) { static const auto &f = dyn_load_symbol("propsys.dll", PropVariantToStringAlloc); return f(arg...); }; } namespace { #define checked(func, args) \ if (const auto hr = ((func)args); FAILED(hr)) \ { \ g_warning("%s%s failed: hresult %#08" PRIX32, \ #func, #args, static_cast(hr)); \ winrt::throw_hresult(hr); \ } struct property { property() noexcept : var{} {} explicit property(const std::wstring &value) { checked(::InitPropVariantFromString,(value.c_str(), &var)); } ~property() { if (const auto hr = ::PropVariantClear(&var); FAILED(hr)) g_critical("PropVariantClear failed: hresult %#08" PRIX32, static_cast(hr)); } auto str() const { wchar_t *str; checked(dyn::PropVariantToStringAlloc,(var, &str)); return std::unique_ptr { str, &::CoTaskMemFree }; } operator const PROPVARIANT &() const noexcept { return var; } operator PROPVARIANT *() noexcept { return &var; } private: PROPVARIANT var; }; bool ImplEnsureAumiddedShortcutExists( const std::string_view menu_rel_path, const std::string_view narrow_aumid) { const auto aumid = sview_to_wstr(narrow_aumid); if (aumid.empty()) { return false; } const auto exe_path = GetExePath(); const auto shortcut_path = GetEnv(L"APPDATA") + LR"(\Microsoft\Windows\Start Menu\)" + sview_to_wstr(menu_rel_path) + L".lnk"; const auto lnk = winrt::create_instance(CLSID_ShellLink); const auto file = lnk.as(); const auto store = lnk.as(); if (SUCCEEDED(file->Load(shortcut_path.c_str(), STGM_READWRITE))) { property aumid_prop; checked(store->GetValue,(PKEY_AppUserModel_ID, aumid_prop)); if (aumid_prop.str().get() != aumid) checked(store->SetValue,(PKEY_AppUserModel_ID, property{aumid})); std::array targ_path; checked(lnk->GetPath,(targ_path.data(), targ_path.size(), nullptr, 0)); if (targ_path.data() != exe_path) checked(lnk->SetPath,(exe_path.c_str())); } else { checked(store->SetValue,(PKEY_AppUserModel_ID, property{aumid})); checked(lnk->SetPath,(exe_path.c_str())); } checked(store->Commit,()); if (file->IsDirty() != S_FALSE) // not the same as `== S_OK` { constexpr auto set_file_as_current = TRUE; checked(file->Save,(shortcut_path.c_str(), set_file_as_current)); } return true; } #undef checked } // nameless namespace extern "C" { gboolean EnsureAumiddedShortcutExists(const gchar *const aumid) noexcept { return g_try_invoke( ImplEnsureAumiddedShortcutExists, R"(Programs\Dino)", aumid); } }