mirror of
https://github.com/modernw/App-Installer-For-Windows-8.x-Reset.git
synced 2026-06-14 03:16:38 +10:00
Update Shell
This commit is contained in:
+4
-4
@@ -10,13 +10,13 @@ BOOL APIENTRY DllMain( HMODULE hModule,
|
||||
{
|
||||
case DLL_PROCESS_ATTACH:
|
||||
case DLL_THREAD_ATTACH:
|
||||
CoInitializeEx (NULL, COINIT_MULTITHREADED);
|
||||
RoInitialize (RO_INIT_MULTITHREADED);
|
||||
// CoInitializeEx (NULL, COINIT_MULTITHREADED);
|
||||
// RoInitialize (RO_INIT_MULTITHREADED);
|
||||
break;
|
||||
case DLL_THREAD_DETACH:
|
||||
case DLL_PROCESS_DETACH:
|
||||
CoUninitialize ();
|
||||
RoUninitialize ();
|
||||
// CoUninitialize ();
|
||||
// RoUninitialize ();
|
||||
break;
|
||||
}
|
||||
return TRUE;
|
||||
|
||||
+101
-8
@@ -25,7 +25,31 @@ std::wstring GetFullPathName (const std::wstring &lpFileName)
|
||||
if (result == 0) return L"";
|
||||
return std::wstring (buffer.data (), result);
|
||||
}
|
||||
|
||||
LPWSTR AllocWideString (const std::wstring &str)
|
||||
{
|
||||
size_t size = (str.length () + 1) * sizeof (WCHAR);
|
||||
LPWSTR buf = (LPWSTR)CoTaskMemAlloc (size);
|
||||
if (!buf) return nullptr;
|
||||
ZeroMemory (buf, size);
|
||||
wcscpy (buf, str.c_str ());
|
||||
return buf;
|
||||
}
|
||||
// 测试用
|
||||
LPWSTR AllocWideString (LPCWSTR str)
|
||||
{
|
||||
if (!str) return nullptr;
|
||||
size_t size = (wcslen (str) + 1) * sizeof (WCHAR);
|
||||
LPWSTR buf = (LPWSTR)CoTaskMemAlloc (size);
|
||||
if (!buf) return nullptr;
|
||||
ZeroMemory (buf, size);
|
||||
wcscpy (buf, str);
|
||||
return buf;
|
||||
}
|
||||
#define _wcsdup AllocWideString
|
||||
#define free CoTaskMemFree
|
||||
#define malloc CoTaskMemAlloc
|
||||
#define realloc CoTaskMemRealloc
|
||||
#define calloc(_cnt_, _size_) CoTaskMemAlloc (_cnt_ * _size_)
|
||||
struct destruct
|
||||
{
|
||||
std::function <void ()> endtask = nullptr;
|
||||
@@ -243,6 +267,7 @@ Windows::Data::Xml::Dom::XmlDocument ^SimpleToastNoticeXml2 (const std::wstring
|
||||
{
|
||||
case 1: templatename = L"ToastText01"; break; // 仅正文
|
||||
case 3: templatename = L"ToastText02"; break; // 标题 + 正文
|
||||
case 6:
|
||||
case 5: templatename = L"ToastImageAndText01"; break; // 图 + 正文
|
||||
case 7: templatename = L"ToastImageAndText02"; break; // 图 + 标题 + 正文
|
||||
default: templatename = L"ToastText01"; break;
|
||||
@@ -342,12 +367,12 @@ HRESULT CreateToastNoticeFromXml (const std::wstring &lpIdName, Windows::Data::X
|
||||
else notifier = ToastMgr::CreateToastNotifier ();
|
||||
auto &xmldoc = pIXml;
|
||||
auto toast = ref new Toast (xmldoc);
|
||||
toast->Activated += ref new Windows::Foundation::TypedEventHandler <
|
||||
Windows::UI::Notifications::ToastNotification ^,
|
||||
Platform::Object ^
|
||||
> ([=] (Windows::UI::Notifications::ToastNotification ^sender, Platform::Object ^args) {
|
||||
if (pfCallback) pfCallback (pCustom);
|
||||
});
|
||||
//toast->Activated += ref new Windows::Foundation::TypedEventHandler <
|
||||
// Windows::UI::Notifications::ToastNotification ^,
|
||||
// Platform::Object ^
|
||||
//> ([=] (Windows::UI::Notifications::ToastNotification ^sender, Platform::Object ^args) {
|
||||
// if (pfCallback) pfCallback (pCustom);
|
||||
//});
|
||||
notifier->Show (toast);
|
||||
return hr = S_OK;
|
||||
}
|
||||
@@ -445,7 +470,7 @@ std::wstring IStreamToTempFile (IStream *p, const std::wstring &ext = L".tmp")
|
||||
if (FAILED (hr)) throw Platform::Exception::CreateException (hr);
|
||||
LARGE_INTEGER liZero = {};
|
||||
stream->Seek (liZero, STREAM_SEEK_SET, nullptr);
|
||||
HANDLE hFile = CreateFileW (outpath.c_str (), GENERIC_WRITE, 0, nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE, nullptr);
|
||||
HANDLE hFile = CreateFileW (outpath.c_str (), GENERIC_WRITE, 0, nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_TEMPORARY, nullptr);
|
||||
const size_t bufsize = 4096;
|
||||
BYTE buf [bufsize] = {0};
|
||||
ULONG bytesRead = 0;
|
||||
@@ -554,3 +579,71 @@ HRESULT CreateShortcutWithAppIdW (LPCWSTR pszShortcutPath, LPCWSTR pszTargetPath
|
||||
else pPersistFile = nullptr;
|
||||
return hr;
|
||||
}
|
||||
|
||||
void NoticeApiFreeString (LPWSTR lpstr)
|
||||
{
|
||||
if (!lpstr) return;
|
||||
CoTaskMemFree (lpstr);
|
||||
}
|
||||
|
||||
size_t Base64ToBytes (const std::wstring &base64OrDataUri, std::vector <BYTE> &retbytes)
|
||||
{
|
||||
retbytes.clear ();
|
||||
std::wstring base64 = base64OrDataUri;
|
||||
size_t commaPos = base64.find (L',');
|
||||
if (commaPos != std::wstring::npos) base64 = base64.substr (commaPos + 1);
|
||||
DWORD binLen = 0;
|
||||
if (!CryptStringToBinaryW (base64.c_str (), static_cast <DWORD> (base64.length ()), CRYPT_STRING_BASE64, nullptr, &binLen, nullptr, nullptr))
|
||||
return 0;
|
||||
retbytes.resize (binLen);
|
||||
if (!CryptStringToBinaryW (base64.c_str (), static_cast <DWORD> (base64.length ()), CRYPT_STRING_BASE64, retbytes.data (), &binLen, nullptr, nullptr))
|
||||
{
|
||||
retbytes.clear ();
|
||||
return 0; // 解码失败
|
||||
}
|
||||
return binLen;
|
||||
}
|
||||
HRESULT BytesToIStream (const std::vector <BYTE> &data, IStream **ppStream)
|
||||
{
|
||||
if (!ppStream) return E_POINTER;
|
||||
*ppStream = nullptr;
|
||||
HGLOBAL hMem = GlobalAlloc (GMEM_MOVEABLE, data.size ());
|
||||
if (!hMem) return E_OUTOFMEMORY;
|
||||
void *pMem = GlobalLock (hMem);
|
||||
if (!pMem)
|
||||
{
|
||||
GlobalFree (hMem);
|
||||
return E_FAIL;
|
||||
}
|
||||
memcpy (pMem, data.data (), data.size ());
|
||||
GlobalUnlock (hMem);
|
||||
HRESULT hr = CreateStreamOnHGlobal (hMem, TRUE, ppStream);
|
||||
if (FAILED (hr)) GlobalFree (hMem);
|
||||
return hr;
|
||||
}
|
||||
HRESULT CreateToastNoticeWithImgBase64 (LPCWSTR lpIdName, LPCWSTR lpText, LPCWSTR lpImgBase64, NOTICE_ACTIVECALLBACK pfCallback, void *pCustom, LPWSTR *lpExceptMsg)
|
||||
{
|
||||
IStream *ist = nullptr;
|
||||
destruct relt ([&ist] () {
|
||||
if (ist) ist->Release ();
|
||||
ist = nullptr;
|
||||
});
|
||||
std::vector <BYTE> bytes;
|
||||
Base64ToBytes (lpImgBase64 ? lpImgBase64 : L"", bytes);
|
||||
if (bytes.size ())
|
||||
{ if (FAILED (BytesToIStream (bytes, &ist))) ist = nullptr; }
|
||||
return CreateToastNoticeWithIStream (lpIdName, lpText, ist, pfCallback, pCustom, lpExceptMsg);
|
||||
}
|
||||
HRESULT CreateToastNotice2WithImgBase64 (LPCWSTR lpIdName, LPCWSTR lpTitle, LPCWSTR lpText, LPCWSTR lpImgBase64, NOTICE_ACTIVECALLBACK pfCallback, void *pCustom, LPWSTR *lpExceptMsg)
|
||||
{
|
||||
IStream *ist = nullptr;
|
||||
destruct relt ([&ist] () {
|
||||
if (ist) ist->Release ();
|
||||
ist = nullptr;
|
||||
});
|
||||
std::vector <BYTE> bytes;
|
||||
Base64ToBytes (lpImgBase64 ? lpImgBase64 : L"", bytes);
|
||||
if (bytes.size ())
|
||||
{ if (FAILED (BytesToIStream (bytes, &ist))) ist = nullptr; }
|
||||
return CreateToastNoticeWithIStream2 (lpIdName, lpTitle, lpText, ist, pfCallback, pCustom, lpExceptMsg);
|
||||
}
|
||||
+127
-4
@@ -31,16 +31,16 @@ extern "C"
|
||||
// 参考:https://learn.microsoft.com/zh-cn/previous-versions/windows/apps/hh761494(v=win.10)
|
||||
// 通过 Toast 通知名来获取 XML 模板。
|
||||
// 不符合会返回一个默认模板(只会有一个 text 节点。根据需要的话可以自己添加)
|
||||
// 注意:返回的指针要自己 free 释放
|
||||
// 注意:返回的指针要自己 NoticeApiFreeString 释放
|
||||
NOTICE_API LPWSTR GetToastNoticeXml (LPCWSTR lpTemplateName);
|
||||
// 获取一个简单的 Toast 通知 XML 文档。第一个参数是必须的。第二个参数为图片 URI(file:///)。
|
||||
// 第二个参数如果为 NULL 或去掉首尾空的长度为 0 的文本则不会使用带图片的模板。
|
||||
// 注意:返回的指针要自己通过 free 释放
|
||||
// 注意:返回的指针要自己通过 NoticeApiFreeString 释放
|
||||
NOTICE_API LPWSTR GenerateSimpleToastNoticeXml (LPCWSTR lpText, LPCWSTR lpImagePath);
|
||||
// 获取一个简单的 Toast 通知 XML 文档。第一个参数是必须的。第三个参数为图片 URI(file:///)。
|
||||
// 第三个参数如果为 NULL 或去掉首尾空的长度为 0 的文本则不会使用带图片的模板。
|
||||
// 第二个参数可以为 NULL 或空文本。当为空时不会使用相关模板
|
||||
// 注意:返回的指针要自己通过 free 释放
|
||||
// 注意:返回的指针要自己通过 NoticeApiFreeString 释放
|
||||
NOTICE_API LPWSTR GenerateSimpleToastNoticeXml2 (LPCWSTR lpTitle, LPCWSTR lpText, LPCWSTR lpImagePath);
|
||||
// 创建并显示一个 Toast 通知
|
||||
// 参数1 为非必须项,这意味着可以传入 NULL 或空文本。但是建议必须填。桌面应用
|
||||
@@ -48,7 +48,7 @@ extern "C"
|
||||
// 在 Windows 10 已经去除。
|
||||
// pfCallback 为点击 Toast 通知本体后触发的回调函数。注意:仅运行期才能用,且不一定会调用成功
|
||||
// pCustom 可以传入自定义内容并在回调中使用
|
||||
// lpExceptMsg 返回异常信息。获取到的指针必须由 free 释放。
|
||||
// lpExceptMsg 返回异常信息。获取到的指针必须由 NoticeApiFreeString 释放。
|
||||
NOTICE_API HRESULT CreateToastNoticeFromXmlDocument (LPCWSTR lpIdName, LPCWSTR lpXmlString, NOTICE_ACTIVECALLBACK pfCallback, void *pCustom, LPWSTR *lpExceptMsg);
|
||||
// 创建一个简单的 Toast 通知。仅支持一段文本和一张图片(图片若不需要则设置为 NULL 或空文本)
|
||||
// 一些参数作用与 CreateToastNoticeFromXmlDocument 中的同名参数作用一致。
|
||||
@@ -71,6 +71,15 @@ extern "C"
|
||||
// 创建快捷方式
|
||||
// (不用安装程序原生的创建,因为需要 AppUserID 才能使用 Toast 通知,当然这个限制只有 Windows 8.x 有,Windows 10 没有这个限制了)
|
||||
NOTICE_API HRESULT CreateShortcutWithAppIdW (LPCWSTR pszShortcutPath, LPCWSTR pszTargetPath, LPCWSTR pszAppId);
|
||||
// 由 notice.dll 获取到的动态字符串必须由此释放。
|
||||
NOTICE_API void NoticeApiFreeString (LPWSTR lpstr);
|
||||
// 创建一个简单的 Toast 通知。支持两段文本和一张图片(图片是 data uri 或者只是 Base64 编码后的字符串,如果不想设置则置 NULL)
|
||||
// 一些参数作用与 CreateToastNoticeFromXmlDocument 中的同名参数作用一致。
|
||||
NOTICE_API HRESULT CreateToastNoticeWithImgBase64 (LPCWSTR lpIdName, LPCWSTR lpText, LPCWSTR lpImgBase64, NOTICE_ACTIVECALLBACK pfCallback, void *pCustom, LPWSTR *lpExceptMsg);
|
||||
// 创建一个简单的 Toast 通知。支持两段文本和一张图片(图片是 data uri 或者只是 Base64 编码后的字符串,如果不想设置则置 NULL)
|
||||
// lpText 可以设置为 NULL 或空文本。此时函数的作用与 CreateToastNoticeWithIStream 一致。
|
||||
// 一些参数作用与 CreateToastNoticeFromXmlDocument 中的同名参数作用一致。
|
||||
NOTICE_API HRESULT CreateToastNotice2WithImgBase64 (LPCWSTR lpIdName, LPCWSTR lpTitle, LPCWSTR lpText, LPCWSTR lpImgBase64, NOTICE_ACTIVECALLBACK pfCallback, void *pCustom, LPWSTR *lpExceptMsg);
|
||||
#ifdef _DEFAULT_INIT_VALUE_
|
||||
#undef _DEFAULT_INIT_VALUE_
|
||||
#endif
|
||||
@@ -80,3 +89,117 @@ extern "C"
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
#include <string>
|
||||
#include <functional>
|
||||
namespace notice
|
||||
{
|
||||
// using LPWSTR = wchar_t *;
|
||||
struct destruct
|
||||
{
|
||||
using funcend = std::function <void ()>;
|
||||
funcend endtask = nullptr;
|
||||
destruct (funcend endtask): endtask (endtask) {}
|
||||
~destruct () { if (endtask) endtask (); }
|
||||
destruct (destruct &) = delete;
|
||||
};
|
||||
struct autostr
|
||||
{
|
||||
LPWSTR lpstr = nullptr;
|
||||
destruct reltask = [this] () {
|
||||
if (lpstr) NoticeApiFreeString (lpstr);
|
||||
lpstr = nullptr;
|
||||
};
|
||||
autostr (LPWSTR str = nullptr): lpstr (str) {}
|
||||
operator LPWSTR () const { return lpstr; }
|
||||
operator std::wstring () const { return lpstr ? lpstr : L""; }
|
||||
std::wstring get_string () { return lpstr ? lpstr : L""; }
|
||||
};
|
||||
using qwstring = std::wstring &;
|
||||
using qcwstring = const std::wstring &;
|
||||
struct hresult
|
||||
{
|
||||
HRESULT hr = S_OK;
|
||||
std::wstring message = L"";
|
||||
operator HRESULT () const { return hr; }
|
||||
operator std::wstring () const { return message; }
|
||||
hresult (HRESULT hr = S_OK, qcwstring msg = L""): hr (hr), message (msg) {}
|
||||
};
|
||||
}
|
||||
std::wstring GetToastNoticeXml (notice::qcwstring template_name) { return notice::autostr (GetToastNoticeXml (template_name.c_str ())).get_string (); }
|
||||
std::wstring GenerateSimpleToastNoticeXml (notice::qcwstring text, notice::qcwstring imgurl = L"") { return notice::autostr (GenerateSimpleToastNoticeXml (text.c_str (), imgurl.c_str ())).get_string (); }
|
||||
std::wstring GenerateSimpleToastNoticeXml (notice::qcwstring title, notice::qcwstring text, notice::qcwstring imgurl) { return notice::autostr (GenerateSimpleToastNoticeXml2 (title.c_str (), text.c_str (), imgurl.c_str ())).get_string (); }
|
||||
std::wstring GenerateSimpleToastNoticeXml2 (notice::qcwstring title, notice::qcwstring text = L"", notice::qcwstring imgurl = L"") { return GenerateSimpleToastNoticeXml2 (title, text, imgurl); }
|
||||
void ToastNoticeEventCallback (void *pCustom)
|
||||
{
|
||||
if (pCustom)
|
||||
{
|
||||
using cbfunc = std::function <void ()>;
|
||||
auto func = reinterpret_cast <cbfunc *> (pCustom);
|
||||
if (func) (*func)();
|
||||
}
|
||||
}
|
||||
notice::hresult CreateToastNoticeFromXmlDocument (notice::qcwstring idname, notice::qcwstring xmlstring, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNoticeFromXmlDocument (idname.c_str (), xmlstring.c_str (), &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
notice::hresult CreateToastNotice (notice::qcwstring idname, notice::qcwstring text, notice::qcwstring imgpath, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNotice (idname.c_str (), text.c_str (), imgpath.c_str (), &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
notice::hresult CreateToastNotice (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text, notice::qcwstring imgpath, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNotice2 (idname.c_str (), title.c_str (), text.c_str (), imgpath.c_str (), &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
notice::hresult CreateToastNotice2 (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text = L"", notice::qcwstring imgpath = L"", std::function <void ()> callback = nullptr) { return CreateToastNotice (idname, title, text, imgpath, callback); }
|
||||
notice::hresult CreateToastNoticeWithIStream (notice::qcwstring idname, notice::qcwstring text, IStream *imgstream, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNoticeWithIStream (idname.c_str (), text.c_str (), imgstream, &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
notice::hresult CreateToastNoticeWithIStream2 (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text = L"", IStream *imgstream = nullptr, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNoticeWithIStream2 (idname.c_str (), title.c_str (), text.c_str (), imgstream, &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
notice::hresult CreateToastNoticeWithIStream (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text, IStream *imgstream, std::function <void ()> callback = nullptr) { return CreateToastNoticeWithIStream2 (idname, title, text, imgstream, callback); }
|
||||
notice::hresult CreateToastNotice (notice::qcwstring idname, notice::qcwstring xmlstring, std::function <void ()> callback) { return CreateToastNoticeFromXmlDocument (idname, xmlstring, callback); }
|
||||
notice::hresult CreateToastNotice (notice::qcwstring idname, notice::qcwstring text, IStream *imgstream, std::function <void ()> callback = nullptr) { return CreateToastNoticeWithIStream (idname, text, imgstream, callback); }
|
||||
notice::hresult CreateToastNotice (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text, IStream *imgstream, std::function <void ()> callback = nullptr) { return CreateToastNoticeWithIStream2 (idname, title, text, imgstream, callback); }
|
||||
HRESULT CreateShortcutWithAppIdW (notice::qcwstring shortcut_path, notice::qcwstring targetpath, notice::qcwstring appid) { return CreateShortcutWithAppIdW (shortcut_path.c_str (), targetpath.c_str (), appid.c_str ()); }
|
||||
HRESULT CreateToastNoticeWithImgBase64 (notice::qcwstring idname, notice::qcwstring text, notice::qcwstring imgbase64, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNoticeWithImgBase64 (idname.c_str (), text.c_str (), imgbase64.c_str (), &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
HRESULT CreateToastNotice2WithImgBase64 (notice::qcwstring idname, notice::qcwstring title, notice::qcwstring text, notice::qcwstring imgbase64, std::function <void ()> callback = nullptr)
|
||||
{
|
||||
notice::autostr exp;
|
||||
notice::hresult hr;
|
||||
hr.hr = CreateToastNotice2WithImgBase64 (idname.c_str (), title.c_str (), text.c_str (), imgbase64.c_str (), &ToastNoticeEventCallback, &callback, &exp.lpstr);
|
||||
hr.message = exp.get_string ();
|
||||
return hr;
|
||||
}
|
||||
#endif
|
||||
@@ -61,7 +61,7 @@
|
||||
<SubSystem>Windows</SubSystem>
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
<UACExecutionLevel>RequireAdministrator</UACExecutionLevel>
|
||||
<AdditionalDependencies>shlwapi.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||
<AdditionalDependencies>shlwapi.lib;crypt32.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
@@ -80,7 +80,7 @@
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||
<OptimizeReferences>true</OptimizeReferences>
|
||||
<AdditionalDependencies>shlwapi.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||
<AdditionalDependencies>shlwapi.lib;crypt32.lib;%(AdditionalDependencies)</AdditionalDependencies>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemGroup>
|
||||
|
||||
@@ -32,8 +32,13 @@
|
||||
#include <Shlwapi.h>
|
||||
#include <algorithm>
|
||||
#include <random>
|
||||
#include <ppltasks.h>
|
||||
#include <thread>
|
||||
#include <vector>
|
||||
#include <combaseapi.h>
|
||||
#using <Windows.winmd>
|
||||
using namespace Microsoft::WRL;
|
||||
using namespace ABI::Windows::UI::Notifications;
|
||||
using namespace ABI::Windows::Data::Xml::Dom;
|
||||
using namespace Microsoft::WRL::Wrappers;
|
||||
#include <wincrypt.h>
|
||||
Reference in New Issue
Block a user