#include "shortcutcreator.h" #include "win32.hpp" #include "converter.hpp" #include "ginvoke.hpp" #include "dyn_mod.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 { // 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 0x%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 0x%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; }; void ImplEnsureAumiddedShortcutExists( const std::string_view menu_rel_path, const std::string_view narrow_aumid) { if (menu_rel_path.empty()) throw std::runtime_error{"empty menu-relative shortcut path"}; const auto aumid = sview_to_wstr(narrow_aumid); 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)); } } #undef checked } // nameless namespace extern "C" { gboolean EnsureAumiddedShortcutExists(const gchar *const aumid) noexcept { return g_try_invoke( ImplEnsureAumiddedShortcutExists, R"(Programs\Dino)", aumid) .has_value(); } }