Initial semi-working windows implementation
This commit is contained in:
parent
5dd4d6a961
commit
7c38633d1e
9 changed files with 833 additions and 18 deletions
|
@ -33,6 +33,9 @@ Notification::Notification(v8::Isolate* isolate, v8::Local<v8::Object> wrapper,
|
||||||
opts.Get("title", &title_);
|
opts.Get("title", &title_);
|
||||||
opts.Get("body", &body_);
|
opts.Get("body", &body_);
|
||||||
has_icon_ = opts.Get("icon", &icon_);
|
has_icon_ = opts.Get("icon", &icon_);
|
||||||
|
if (has_icon_) {
|
||||||
|
opts.Get("icon", &icon_path_);
|
||||||
|
}
|
||||||
opts.Get("silent", &silent_);
|
opts.Get("silent", &silent_);
|
||||||
opts.Get("replyPlaceholder", &reply_placeholder_);
|
opts.Get("replyPlaceholder", &reply_placeholder_);
|
||||||
opts.Get("hasReply", &has_reply_);
|
opts.Get("hasReply", &has_reply_);
|
||||||
|
@ -63,17 +66,17 @@ Notification* Notification::FromID(int id) {
|
||||||
|
|
||||||
// Getters
|
// Getters
|
||||||
int Notification::GetID() { return id_; }
|
int Notification::GetID() { return id_; }
|
||||||
std::string Notification::GetTitle() { return title_; }
|
base::string16 Notification::GetTitle() { return title_; }
|
||||||
std::string Notification::GetBody() { return body_; }
|
base::string16 Notification::GetBody() { return body_; }
|
||||||
bool Notification::GetSilent() { return silent_; }
|
bool Notification::GetSilent() { return silent_; }
|
||||||
std::string Notification::GetReplyPlaceholder() { return reply_placeholder_; }
|
base::string16 Notification::GetReplyPlaceholder() { return reply_placeholder_; }
|
||||||
bool Notification::GetHasReply() { return has_reply_; }
|
bool Notification::GetHasReply() { return has_reply_; }
|
||||||
|
|
||||||
// Setters
|
// Setters
|
||||||
void Notification::SetTitle(std::string new_title) { title_ = new_title; NotifyPropsUpdated(); }
|
void Notification::SetTitle(base::string16 new_title) { title_ = new_title; NotifyPropsUpdated(); }
|
||||||
void Notification::SetBody(std::string new_body) { body_ = new_body; NotifyPropsUpdated(); }
|
void Notification::SetBody(base::string16 new_body) { body_ = new_body; NotifyPropsUpdated(); }
|
||||||
void Notification::SetSilent(bool new_silent) { silent_ = new_silent; NotifyPropsUpdated(); }
|
void Notification::SetSilent(bool new_silent) { silent_ = new_silent; NotifyPropsUpdated(); }
|
||||||
void Notification::SetReplyPlaceholder(std::string new_reply_placeholder) { reply_placeholder_ = new_reply_placeholder; NotifyPropsUpdated(); }
|
void Notification::SetReplyPlaceholder(base::string16 new_reply_placeholder) { reply_placeholder_ = new_reply_placeholder; NotifyPropsUpdated(); }
|
||||||
void Notification::SetHasReply(bool new_has_reply) { has_reply_ = new_has_reply; NotifyPropsUpdated(); }
|
void Notification::SetHasReply(bool new_has_reply) { has_reply_ = new_has_reply; NotifyPropsUpdated(); }
|
||||||
|
|
||||||
void Notification::OnClicked() {
|
void Notification::OnClicked() {
|
||||||
|
|
|
@ -42,28 +42,29 @@ class Notification : public mate::TrackableObject<Notification>,
|
||||||
|
|
||||||
// Prop Getters
|
// Prop Getters
|
||||||
int GetID();
|
int GetID();
|
||||||
std::string GetTitle();
|
base::string16 GetTitle();
|
||||||
std::string GetBody();
|
base::string16 GetBody();
|
||||||
bool GetSilent();
|
bool GetSilent();
|
||||||
std::string GetReplyPlaceholder();
|
base::string16 GetReplyPlaceholder();
|
||||||
bool GetHasReply();
|
bool GetHasReply();
|
||||||
|
|
||||||
// Prop Setters
|
// Prop Setters
|
||||||
void SetTitle(std::string new_title);
|
void SetTitle(base::string16 new_title);
|
||||||
void SetBody(std::string new_body);
|
void SetBody(base::string16 new_body);
|
||||||
void SetSilent(bool new_silent);
|
void SetSilent(bool new_silent);
|
||||||
void SetReplyPlaceholder(std::string new_reply_placeholder);
|
void SetReplyPlaceholder(base::string16 new_reply_placeholder);
|
||||||
void SetHasReply(bool new_has_reply);
|
void SetHasReply(bool new_has_reply);
|
||||||
|
|
||||||
void NotifyPropsUpdated();
|
void NotifyPropsUpdated();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::string title_ = "";
|
base::string16 title_ = L"";
|
||||||
std::string body_ = "";
|
base::string16 body_ = L"";
|
||||||
gfx::Image icon_;
|
gfx::Image icon_;
|
||||||
|
base::string16 icon_path_ = L"";
|
||||||
bool has_icon_ = false;
|
bool has_icon_ = false;
|
||||||
bool silent_ = false;
|
bool silent_ = false;
|
||||||
std::string reply_placeholder_ = "";
|
base::string16 reply_placeholder_ = L"";
|
||||||
bool has_reply_ = false;
|
bool has_reply_ = false;
|
||||||
|
|
||||||
int id_;
|
int id_;
|
||||||
|
|
|
@ -69,8 +69,8 @@ void Notification::OnInitialProps() {
|
||||||
void Notification::NotifyPropsUpdated() {
|
void Notification::NotifyPropsUpdated() {
|
||||||
base::scoped_nsobject<NSUserNotification> notification_ = native_notifications_[id_];
|
base::scoped_nsobject<NSUserNotification> notification_ = native_notifications_[id_];
|
||||||
|
|
||||||
[notification_ setTitle:base::SysUTF8ToNSString(title_)];
|
[notification_ setTitle:base::SysUTF16ToNSString(title_)];
|
||||||
[notification_ setInformativeText:base::SysUTF8ToNSString(body_)];
|
[notification_ setInformativeText:base::SysUTF16ToNSString(body_)];
|
||||||
|
|
||||||
NSDictionary * userInfo = [NSMutableDictionary dictionary];
|
NSDictionary * userInfo = [NSMutableDictionary dictionary];
|
||||||
[userInfo setValue:[NSNumber numberWithInt:id_] forKey:@"id"];
|
[userInfo setValue:[NSNumber numberWithInt:id_] forKey:@"id"];
|
||||||
|
@ -81,7 +81,7 @@ void Notification::NotifyPropsUpdated() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (has_reply_) {
|
if (has_reply_) {
|
||||||
[notification_ setResponsePlaceholder:base::SysUTF8ToNSString(reply_placeholder_)];
|
[notification_ setResponsePlaceholder:base::SysUTF16ToNSString(reply_placeholder_)];
|
||||||
[notification_ setHasReplyButton:true];
|
[notification_ setHasReplyButton:true];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
48
atom/browser/api/atom_api_notification_win.cc
Normal file
48
atom/browser/api/atom_api_notification_win.cc
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
// Copyright (c) 2014 GitHub, Inc.
|
||||||
|
// Use of this source code is governed by the MIT license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#include "atom/browser/api/atom_api_notification.h"
|
||||||
|
|
||||||
|
#include "atom/browser/ui/win/toast_handler.h"
|
||||||
|
#include "atom/browser/ui/win/toast_lib.h"
|
||||||
|
#include "base/strings/utf_string_conversions.h"
|
||||||
|
#include "common/string_conversion.h"
|
||||||
|
|
||||||
|
namespace atom {
|
||||||
|
|
||||||
|
namespace api {
|
||||||
|
|
||||||
|
bool can_toast_ = true;
|
||||||
|
bool initialized_ = false;
|
||||||
|
|
||||||
|
void Notification::Show() {
|
||||||
|
atom::AtomToastHandler* handler = new atom::AtomToastHandler(this);
|
||||||
|
WinToastLib::WinToastTemplate toast = WinToastLib::WinToastTemplate(WinToastLib::WinToastTemplate::TextTwoLines);
|
||||||
|
// toast.setImagePath(L"C:\example.png");
|
||||||
|
toast.setTextField(title_, WinToastLib::WinToastTemplate::TextField::FirstLine);
|
||||||
|
toast.setTextField(body_, WinToastLib::WinToastTemplate::TextField::SecondLine);
|
||||||
|
toast.setSilent(silent_);
|
||||||
|
|
||||||
|
WinToastLib::WinToast::instance()->showToast(toast, handler);
|
||||||
|
|
||||||
|
OnShown();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Notification::OnInitialProps() {
|
||||||
|
if (!initialized_) {
|
||||||
|
WinToastLib::WinToast::instance()->setAppName(L"WinToastExample");
|
||||||
|
WinToastLib::WinToast::instance()->setAppUserModelId(
|
||||||
|
WinToastLib::WinToast::configureAUMI(L"mohabouje", L"wintoast", L"wintoastexample", L"20161006")
|
||||||
|
);
|
||||||
|
can_toast_ = WinToastLib::WinToast::instance()->initialize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Notification::NotifyPropsUpdated() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
28
atom/browser/ui/win/toast_handler.cc
Normal file
28
atom/browser/ui/win/toast_handler.cc
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
// Copyright (c) 2014 GitHub, Inc.
|
||||||
|
// Use of this source code is governed by the MIT license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#include "atom/browser/ui/win/toast_handler.h"
|
||||||
|
|
||||||
|
#include "atom/browser/ui/win/toast_lib.h"
|
||||||
|
#include "atom/browser/api/atom_api_notification.h"
|
||||||
|
|
||||||
|
namespace atom {
|
||||||
|
|
||||||
|
AtomToastHandler::AtomToastHandler(atom::api::Notification* target) {
|
||||||
|
observer_ = target;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AtomToastHandler::toastActivated() {
|
||||||
|
observer_->OnClicked();
|
||||||
|
};
|
||||||
|
|
||||||
|
void AtomToastHandler::toastDismissed(WinToastLib::WinToastHandler::WinToastDismissalReason state) {
|
||||||
|
// observer_->OnDismissed();
|
||||||
|
};
|
||||||
|
|
||||||
|
void AtomToastHandler::toastFailed() {
|
||||||
|
// observer_->OnErrored();
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
25
atom/browser/ui/win/toast_handler.h
Normal file
25
atom/browser/ui/win/toast_handler.h
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
// Copyright (c) 2014 GitHub, Inc.
|
||||||
|
// Use of this source code is governed by the MIT license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#include "atom/browser/ui/win/toast_lib.h"
|
||||||
|
#include "atom/browser/api/atom_api_notification.h"
|
||||||
|
|
||||||
|
#ifndef ATOM_BROWSER_UI_TOAST_HANDLER_H_
|
||||||
|
#define ATOM_BROWSER_UI_TOAST_HANDLER_H_
|
||||||
|
|
||||||
|
namespace atom {
|
||||||
|
|
||||||
|
class AtomToastHandler : public WinToastLib::WinToastHandler {
|
||||||
|
public:
|
||||||
|
atom::api::Notification* observer_;
|
||||||
|
AtomToastHandler(atom::api::Notification* target);
|
||||||
|
|
||||||
|
void toastActivated() override;
|
||||||
|
void toastDismissed(WinToastLib::WinToastHandler::WinToastDismissalReason state);
|
||||||
|
void toastFailed();
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // ATOM_BROWSER_UI_TOAST_HANDLER_H_
|
537
atom/browser/ui/win/toast_lib.cc
Normal file
537
atom/browser/ui/win/toast_lib.cc
Normal file
|
@ -0,0 +1,537 @@
|
||||||
|
// MIT License
|
||||||
|
|
||||||
|
// Copyright (c) 2016 Mohammed Boujemaoui Boulaghmoudi
|
||||||
|
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
#include "toast_lib.h"
|
||||||
|
#include "browser/win/scoped_hstring.h"
|
||||||
|
#pragma comment(lib,"shlwapi")
|
||||||
|
#pragma comment(lib,"user32")
|
||||||
|
|
||||||
|
using namespace WinToastLib;
|
||||||
|
namespace DllImporter {
|
||||||
|
|
||||||
|
// Function load a function from library
|
||||||
|
template <typename Function>
|
||||||
|
HRESULT loadFunctionFromLibrary(HINSTANCE library, LPCSTR name, Function &func) {
|
||||||
|
if (!library)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
func = reinterpret_cast<Function>(GetProcAddress(library, name));
|
||||||
|
return (func != nullptr) ? S_OK : E_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef HRESULT(FAR STDAPICALLTYPE *f_SetCurrentProcessExplicitAppUserModelID)(__in PCWSTR AppID);
|
||||||
|
typedef HRESULT(FAR STDAPICALLTYPE *f_PropVariantToString)(_In_ REFPROPVARIANT propvar, _Out_writes_(cch) PWSTR psz, _In_ UINT cch);
|
||||||
|
typedef HRESULT(FAR STDAPICALLTYPE *f_RoGetActivationFactory)(_In_ HSTRING activatableClassId, _In_ REFIID iid, _COM_Outptr_ void ** factory);
|
||||||
|
typedef HRESULT(FAR STDAPICALLTYPE *f_WindowsCreateStringReference)(_In_reads_opt_(length + 1) PCWSTR sourceString, UINT32 length, _Out_ HSTRING_HEADER * hstringHeader, _Outptr_result_maybenull_ _Result_nullonfailure_ HSTRING * string);
|
||||||
|
typedef HRESULT(FAR STDAPICALLTYPE *f_WindowsDeleteString)(_In_opt_ HSTRING string);
|
||||||
|
|
||||||
|
f_SetCurrentProcessExplicitAppUserModelID SetCurrentProcessExplicitAppUserModelID;
|
||||||
|
f_PropVariantToString PropVariantToString;
|
||||||
|
f_RoGetActivationFactory RoGetActivationFactory;
|
||||||
|
f_WindowsCreateStringReference WindowsCreateStringReference;
|
||||||
|
f_WindowsDeleteString WindowsDeleteString;
|
||||||
|
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
_Check_return_ __inline HRESULT _1_GetActivationFactory(_In_ HSTRING activatableClassId, _COM_Outptr_ T** factory) {
|
||||||
|
return RoGetActivationFactory(activatableClassId, IID_INS_ARGS(factory));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
inline HRESULT Wrap_GetActivationFactory(_In_ HSTRING activatableClassId, _Inout_ Details::ComPtrRef<T> factory) throw() {
|
||||||
|
return _1_GetActivationFactory(activatableClassId, factory.ReleaseAndGetAddressOf());
|
||||||
|
}
|
||||||
|
|
||||||
|
inline HRESULT initialize() {
|
||||||
|
HINSTANCE LibShell32 = LoadLibrary(L"SHELL32.DLL");
|
||||||
|
HRESULT hr = loadFunctionFromLibrary(LibShell32, "SetCurrentProcessExplicitAppUserModelID", SetCurrentProcessExplicitAppUserModelID);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
HINSTANCE LibPropSys = LoadLibrary(L"PROPSYS.DLL");
|
||||||
|
hr = loadFunctionFromLibrary(LibPropSys, "PropVariantToString", PropVariantToString);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
HINSTANCE LibComBase = LoadLibrary(L"COMBASE.DLL");
|
||||||
|
return SUCCEEDED(loadFunctionFromLibrary(LibComBase, "RoGetActivationFactory", RoGetActivationFactory))
|
||||||
|
&& SUCCEEDED(loadFunctionFromLibrary(LibComBase, "WindowsCreateStringReference", WindowsCreateStringReference))
|
||||||
|
&& SUCCEEDED(loadFunctionFromLibrary(LibComBase, "WindowsDeleteString", WindowsDeleteString));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace Util {
|
||||||
|
inline HRESULT defaultExecutablePath(_In_ WCHAR* path, _In_ DWORD nSize = MAX_PATH) {
|
||||||
|
DWORD written = GetModuleFileNameEx(GetCurrentProcess(), nullptr, path, nSize);
|
||||||
|
return (written > 0) ? S_OK : E_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
inline HRESULT defaultShellLinksDirectory(_In_ WCHAR* path, _In_ DWORD nSize = MAX_PATH) {
|
||||||
|
DWORD written = GetEnvironmentVariable(L"APPDATA", path, nSize);
|
||||||
|
HRESULT hr = written > 0 ? S_OK : E_INVALIDARG;
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
errno_t result = wcscat_s(path, nSize, DEFAULT_SHELL_LINKS_PATH);
|
||||||
|
hr = (result == 0) ? S_OK : E_INVALIDARG;
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline HRESULT defaultShellLinkPath(const std::wstring& appname, _In_ WCHAR* path, _In_ DWORD nSize = MAX_PATH) {
|
||||||
|
HRESULT hr = defaultShellLinksDirectory(path, nSize);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
const std::wstring appLink(appname + DEFAULT_LINK_FORMAT);
|
||||||
|
errno_t result = wcscat_s(path, nSize, appLink.c_str());
|
||||||
|
hr = (result == 0) ? S_OK : E_INVALIDARG;
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
inline HRESULT setNodeStringValue(const std::wstring& string, IXmlNode *node, IXmlDocument *xml) {
|
||||||
|
ComPtr<IXmlText> textNode;
|
||||||
|
HRESULT hr = xml->CreateTextNode( WinToastStringWrapper(string).Get(), &textNode);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNode> stringNode;
|
||||||
|
hr = textNode.As(&stringNode);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNode> appendedChild;
|
||||||
|
hr = node->AppendChild(stringNode.Get(), &appendedChild);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline HRESULT setEventHandlers(_In_ IToastNotification* notification, _In_ WinToastHandler* eventHandler) {
|
||||||
|
EventRegistrationToken activatedToken, dismissedToken, failedToken;
|
||||||
|
HRESULT hr = notification->add_Activated(
|
||||||
|
Callback < Implements < RuntimeClassFlags<ClassicCom>,
|
||||||
|
ITypedEventHandler<ToastNotification*, IInspectable* >> >(
|
||||||
|
[eventHandler](IToastNotification*, IInspectable*)
|
||||||
|
{
|
||||||
|
eventHandler->toastActivated();
|
||||||
|
return S_OK;
|
||||||
|
}).Get(), &activatedToken);
|
||||||
|
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = notification->add_Dismissed(Callback < Implements < RuntimeClassFlags<ClassicCom>,
|
||||||
|
ITypedEventHandler<ToastNotification*, ToastDismissedEventArgs* >> >(
|
||||||
|
[eventHandler](IToastNotification*, IToastDismissedEventArgs* e)
|
||||||
|
{
|
||||||
|
ToastDismissalReason reason;
|
||||||
|
if (SUCCEEDED(e->get_Reason(&reason)))
|
||||||
|
{
|
||||||
|
eventHandler->toastDismissed(static_cast<WinToastHandler::WinToastDismissalReason>(reason));
|
||||||
|
}
|
||||||
|
return S_OK;
|
||||||
|
}).Get(), &dismissedToken);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = notification->add_Failed(Callback < Implements < RuntimeClassFlags<ClassicCom>,
|
||||||
|
ITypedEventHandler<ToastNotification*, ToastFailedEventArgs* >> >(
|
||||||
|
[eventHandler](IToastNotification*, IToastFailedEventArgs*)
|
||||||
|
{
|
||||||
|
eventHandler->toastFailed();
|
||||||
|
return S_OK;
|
||||||
|
}).Get(), &failedToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WinToast* WinToast::_instance = nullptr;
|
||||||
|
WinToast* WinToast::instance() {
|
||||||
|
if (_instance == nullptr) {
|
||||||
|
_instance = new WinToast();
|
||||||
|
}
|
||||||
|
return _instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
WinToast::WinToast() : _isInitialized(false)
|
||||||
|
{
|
||||||
|
DllImporter::initialize();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WinToast::setAppName(_In_ const std::wstring& appName) {
|
||||||
|
_appName = appName;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::wstring WinToast::appName() const {
|
||||||
|
return _appName;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::wstring WinToast::appUserModelId() const {
|
||||||
|
return _aumi;
|
||||||
|
}
|
||||||
|
void WinToast::setAppUserModelId(_In_ const std::wstring& aumi) {
|
||||||
|
_aumi = aumi;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WinToast::isCompatible() {
|
||||||
|
return !((DllImporter::SetCurrentProcessExplicitAppUserModelID == nullptr)
|
||||||
|
|| (DllImporter::PropVariantToString == nullptr)
|
||||||
|
|| (DllImporter::RoGetActivationFactory == nullptr)
|
||||||
|
|| (DllImporter::WindowsCreateStringReference == nullptr)
|
||||||
|
|| (DllImporter::WindowsDeleteString == nullptr));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::wstring WinToast::configureAUMI(_In_ const std::wstring &company,
|
||||||
|
_In_ const std::wstring &name,
|
||||||
|
_In_ const std::wstring &surname,
|
||||||
|
_In_ const std::wstring &versionInfo)
|
||||||
|
{
|
||||||
|
std::wstring aumi = company;
|
||||||
|
aumi += L"." + name;
|
||||||
|
aumi += L"." + surname;
|
||||||
|
aumi += L"." + versionInfo;
|
||||||
|
|
||||||
|
return aumi;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WinToast::initialize() {
|
||||||
|
if (_aumi.empty() || _appName.empty()) {
|
||||||
|
return _isInitialized = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isCompatible()) {
|
||||||
|
return _isInitialized = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (FAILED(DllImporter::SetCurrentProcessExplicitAppUserModelID(_aumi.c_str()))) {
|
||||||
|
return _isInitialized = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
HRESULT hr = validateShellLink();
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
hr = createShellLink();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = DllImporter::Wrap_GetActivationFactory(WinToastStringWrapper(RuntimeClass_Windows_UI_Notifications_ToastNotificationManager).Get(), &_notificationManager);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = notificationManager()->CreateToastNotifierWithId(WinToastStringWrapper(_aumi).Get(), &_notifier);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = DllImporter::Wrap_GetActivationFactory(WinToastStringWrapper(RuntimeClass_Windows_UI_Notifications_ToastNotification).Get(), &_notificationFactory);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return _isInitialized = SUCCEEDED(hr);
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WinToast::validateShellLink() {
|
||||||
|
|
||||||
|
WCHAR _path[MAX_PATH];
|
||||||
|
Util::defaultShellLinkPath(_appName, _path);
|
||||||
|
// Check if the file exist
|
||||||
|
DWORD attr = GetFileAttributes(_path);
|
||||||
|
if (attr >= 0xFFFFFFF) {
|
||||||
|
std::wcout << "Error, shell link not found. Try to create a new one in: " << _path << std::endl;
|
||||||
|
return E_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Let's load the file as shell link to validate.
|
||||||
|
// - Create a shell link
|
||||||
|
// - Create a persistant file
|
||||||
|
// - Load the path as data for the persistant file
|
||||||
|
// - Read the property AUMI and validate with the current
|
||||||
|
// - Review if AUMI is equal.
|
||||||
|
ComPtr<IShellLink> shellLink;
|
||||||
|
HRESULT hr = CoCreateInstance(CLSID_ShellLink, nullptr, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&shellLink));
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IPersistFile> persistFile;
|
||||||
|
hr = shellLink.As(&persistFile);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = persistFile->Load(_path, STGM_READWRITE);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IPropertyStore> propertyStore;
|
||||||
|
hr = shellLink.As(&propertyStore);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
PROPVARIANT appIdPropVar;
|
||||||
|
hr = propertyStore->GetValue(PKEY_AppUserModel_ID, &appIdPropVar);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
WCHAR AUMI[MAX_PATH];
|
||||||
|
hr = DllImporter::PropVariantToString(appIdPropVar, AUMI, MAX_PATH);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = (_aumi == AUMI) ? S_OK : E_FAIL;
|
||||||
|
} else { // AUMI Changed for the same app, let's update the current value! =)
|
||||||
|
PropVariantClear(&appIdPropVar);
|
||||||
|
hr = InitPropVariantFromString(_aumi.c_str(), &appIdPropVar);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = propertyStore->SetValue(PKEY_AppUserModel_ID, appIdPropVar);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = propertyStore->Commit();
|
||||||
|
if (SUCCEEDED(hr) && SUCCEEDED(persistFile->IsDirty())) {
|
||||||
|
hr = persistFile->Save(_path, TRUE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PropVariantClear(&appIdPropVar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
HRESULT WinToast::createShellLink() {
|
||||||
|
WCHAR exePath[MAX_PATH];
|
||||||
|
WCHAR slPath[MAX_PATH];
|
||||||
|
Util::defaultShellLinkPath(_appName, slPath);
|
||||||
|
Util::defaultExecutablePath(exePath);
|
||||||
|
ComPtr<IShellLink> shellLink;
|
||||||
|
HRESULT hr = CoCreateInstance(CLSID_ShellLink, nullptr, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&shellLink));
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = shellLink->SetPath(exePath);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = shellLink->SetArguments(L"");
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = shellLink->SetWorkingDirectory(exePath);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IPropertyStore> propertyStore;
|
||||||
|
hr = shellLink.As(&propertyStore);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
PROPVARIANT appIdPropVar;
|
||||||
|
hr = InitPropVariantFromString(_aumi.c_str(), &appIdPropVar);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = propertyStore->SetValue(PKEY_AppUserModel_ID, appIdPropVar);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = propertyStore->Commit();
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IPersistFile> persistFile;
|
||||||
|
hr = shellLink.As(&persistFile);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = persistFile->Save(slPath, TRUE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PropVariantClear(&appIdPropVar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
CoTaskMemFree(exePath);
|
||||||
|
CoTaskMemFree(slPath);
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
bool WinToast::showToast(_In_ const WinToastTemplate& toast, _In_ WinToastHandler* handler) {
|
||||||
|
if (!isInitialized()) {
|
||||||
|
return _isInitialized;
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT hr = _notificationManager->GetTemplateContent(ToastTemplateType(toast.type()), &_xmlDocument);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
const int fieldsCount = toast.textFieldsCount();
|
||||||
|
for (int i = 0; i < fieldsCount && SUCCEEDED(hr); i++) {
|
||||||
|
hr = setTextField(toast.textField(WinToastTemplate::TextField(i)), i);
|
||||||
|
}
|
||||||
|
if (makeSilent(toast.isSilent())) {
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = toast.hasImage() ? setImageField(toast.imagePath()) : hr;
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = _notificationFactory->CreateToastNotification(xmlDocument(), &_notification);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = Util::setEventHandlers(notification(), handler);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = _notifier->Show(notification());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return SUCCEEDED(hr);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
HRESULT WinToast::setTextField(_In_ const std::wstring& text, _In_ int pos) {
|
||||||
|
ComPtr<IXmlNodeList> nodeList;
|
||||||
|
HRESULT hr = _xmlDocument->GetElementsByTagName(WinToastStringWrapper(L"text").Get(), &nodeList);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNode> node;
|
||||||
|
hr = nodeList->Item(pos, &node);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
hr = Util::setNodeStringValue(text, node.Get(), xmlDocument());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WinToast::makeSilent(bool is_silent) {
|
||||||
|
if (!is_silent) return true;
|
||||||
|
ScopedHString tag(L"toast");
|
||||||
|
if (!tag.success())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNodeList> node_list;
|
||||||
|
if (FAILED(xmlDocument()->GetElementsByTagName(tag, &node_list)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> root;
|
||||||
|
if (FAILED(node_list->Item(0, &root)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlElement> audio_element;
|
||||||
|
ScopedHString audio_str(L"audio");
|
||||||
|
if (FAILED(xmlDocument()->CreateElement(audio_str, &audio_element)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> audio_node_tmp;
|
||||||
|
if (FAILED(audio_element.As(&audio_node_tmp)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Append audio node to toast xml
|
||||||
|
ComPtr<IXmlNode> audio_node;
|
||||||
|
if (FAILED(root->AppendChild(audio_node_tmp.Get(), &audio_node)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Create silent attribute
|
||||||
|
ComPtr<IXmlNamedNodeMap> attributes;
|
||||||
|
if (FAILED(audio_node->get_Attributes(&attributes)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlAttribute> silent_attribute;
|
||||||
|
ScopedHString silent_str(L"silent");
|
||||||
|
if (FAILED(xmlDocument()->CreateAttribute(silent_str, &silent_attribute)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> silent_attribute_node;
|
||||||
|
if (FAILED(silent_attribute.As(&silent_attribute_node)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Set silent attribute to true
|
||||||
|
ScopedHString silent_value(L"true");
|
||||||
|
if (!silent_value.success())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlText> silent_text;
|
||||||
|
if (FAILED(xmlDocument()->CreateTextNode(silent_value, &silent_text)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> silent_node;
|
||||||
|
if (FAILED(silent_text.As(&silent_node)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> child_node;
|
||||||
|
if (FAILED(
|
||||||
|
silent_attribute_node->AppendChild(silent_node.Get(), &child_node)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ComPtr<IXmlNode> silent_attribute_pnode;
|
||||||
|
return SUCCEEDED(attributes.Get()->SetNamedItem(silent_attribute_node.Get(),
|
||||||
|
&silent_attribute_pnode));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
HRESULT WinToast::setImageField(_In_ const std::wstring& path) {
|
||||||
|
wchar_t imagePath[MAX_PATH] = L"file:///";
|
||||||
|
HRESULT hr = StringCchCat(imagePath, MAX_PATH, path.c_str());
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNodeList> nodeList;
|
||||||
|
hr = _xmlDocument->GetElementsByTagName(WinToastStringWrapper(L"image").Get(), &nodeList);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNode> node;
|
||||||
|
hr = nodeList->Item(0, &node);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNamedNodeMap> attributes;
|
||||||
|
hr = node->get_Attributes(&attributes);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
ComPtr<IXmlNode> editedNode;
|
||||||
|
hr = attributes->GetNamedItem(WinToastStringWrapper(L"src").Get(), &editedNode);
|
||||||
|
if (SUCCEEDED(hr)) {
|
||||||
|
Util::setNodeStringValue(imagePath, editedNode.Get(), xmlDocument());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
WinToastTemplate::WinToastTemplate(const WinToastTemplateType& type) :
|
||||||
|
_type(type)
|
||||||
|
{
|
||||||
|
initComponentsFromType();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WinToastTemplate::~WinToastTemplate()
|
||||||
|
{
|
||||||
|
_textFields.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WinToastTemplate::setTextField(_In_ const std::wstring& txt, _In_ const WinToastTemplate::TextField& pos) {
|
||||||
|
_textFields[pos] = txt;
|
||||||
|
}
|
||||||
|
void WinToastTemplate::setImagePath(_In_ const std::wstring& imgPath) {
|
||||||
|
if (!_hasImage)
|
||||||
|
return;
|
||||||
|
_imagePath = imgPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WinToastTemplate::setSilent(bool is_silent) {
|
||||||
|
_silent = is_silent;
|
||||||
|
}
|
||||||
|
|
||||||
|
int WinToastTemplate::TextFieldsCount[WinToastTemplateTypeCount] = { 1, 2, 2, 3, 1, 2, 2, 3};
|
||||||
|
void WinToastTemplate::initComponentsFromType() {
|
||||||
|
_hasImage = _type < ToastTemplateType_ToastText01;
|
||||||
|
_textFields = std::vector<std::wstring>(TextFieldsCount[_type], L"");
|
||||||
|
}
|
||||||
|
|
||||||
|
WinToastStringWrapper::WinToastStringWrapper(_In_reads_(length) PCWSTR stringRef, _In_ UINT32 length) throw() {
|
||||||
|
HRESULT hr = DllImporter::WindowsCreateStringReference(stringRef, length, &_header, &_hstring);
|
||||||
|
if (!SUCCEEDED(hr)) {
|
||||||
|
RaiseException(static_cast<DWORD>(STATUS_INVALID_PARAMETER), EXCEPTION_NONCONTINUABLE, 0, nullptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WinToastStringWrapper::WinToastStringWrapper(const std::wstring &stringRef)
|
||||||
|
{
|
||||||
|
HRESULT hr = DllImporter::WindowsCreateStringReference(stringRef.c_str(), static_cast<UINT32>(stringRef.length()), &_header, &_hstring);
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
RaiseException(static_cast<DWORD>(STATUS_INVALID_PARAMETER), EXCEPTION_NONCONTINUABLE, 0, nullptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WinToastStringWrapper::~WinToastStringWrapper() {
|
||||||
|
DllImporter::WindowsDeleteString(_hstring);
|
||||||
|
}
|
||||||
|
|
||||||
|
HSTRING WinToastStringWrapper::Get() const {
|
||||||
|
return _hstring;
|
||||||
|
}
|
168
atom/browser/ui/win/toast_lib.h
Normal file
168
atom/browser/ui/win/toast_lib.h
Normal file
|
@ -0,0 +1,168 @@
|
||||||
|
// MIT License
|
||||||
|
|
||||||
|
// Copyright (c) 2016 Mohammed Boujemaoui Boulaghmoudi
|
||||||
|
|
||||||
|
// 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 WINTOASTLIB_H
|
||||||
|
#define WINTOASTLIB_H
|
||||||
|
#include <windows.h>
|
||||||
|
#include <SDKDDKVer.h>
|
||||||
|
#include <WinUser.h>
|
||||||
|
#include <Shobjidl.h>
|
||||||
|
#include <wrl/implements.h>
|
||||||
|
#include <wrl/event.h>
|
||||||
|
#include <windows.ui.notifications.h>
|
||||||
|
#include <strsafe.h>
|
||||||
|
#include <Psapi.h>
|
||||||
|
#include <shlobj.h>
|
||||||
|
#include <roapi.h>
|
||||||
|
#include <propvarutil.h>
|
||||||
|
#include <functiondiscoverykeys.h>
|
||||||
|
#include <iostream>
|
||||||
|
#include <winstring.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <vector>
|
||||||
|
using namespace Microsoft::WRL;
|
||||||
|
using namespace ABI::Windows::Data::Xml::Dom;
|
||||||
|
using namespace ABI::Windows::Foundation;
|
||||||
|
using namespace ABI::Windows::UI::Notifications;
|
||||||
|
using namespace Windows::Foundation;
|
||||||
|
|
||||||
|
#define DEFAULT_SHELL_LINKS_PATH L"\\Microsoft\\Windows\\Start Menu\\Programs\\"
|
||||||
|
#define DEFAULT_LINK_FORMAT L".lnk"
|
||||||
|
|
||||||
|
namespace WinToastLib {
|
||||||
|
class WinToastStringWrapper {
|
||||||
|
public:
|
||||||
|
WinToastStringWrapper(_In_reads_(length) PCWSTR stringRef, _In_ UINT32 length) throw();
|
||||||
|
WinToastStringWrapper(_In_ const std::wstring &stringRef) throw();
|
||||||
|
~WinToastStringWrapper();
|
||||||
|
HSTRING Get() const throw();
|
||||||
|
private:
|
||||||
|
HSTRING _hstring;
|
||||||
|
HSTRING_HEADER _header;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class WinToastHandler {
|
||||||
|
public:
|
||||||
|
enum WinToastDismissalReason {
|
||||||
|
UserCanceled = ToastDismissalReason::ToastDismissalReason_UserCanceled,
|
||||||
|
ApplicationHidden = ToastDismissalReason::ToastDismissalReason_ApplicationHidden,
|
||||||
|
TimedOut = ToastDismissalReason::ToastDismissalReason_TimedOut
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void toastActivated() {};
|
||||||
|
virtual void toastDismissed(WinToastDismissalReason state) {};
|
||||||
|
virtual void toastFailed() {};
|
||||||
|
};
|
||||||
|
|
||||||
|
class WinToastTemplate
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum TextField {
|
||||||
|
FirstLine = 0,
|
||||||
|
SecondLine,
|
||||||
|
ThirdLine,
|
||||||
|
LineCount
|
||||||
|
};
|
||||||
|
|
||||||
|
enum WinToastTemplateType {
|
||||||
|
ImageWithOneLine = ToastTemplateType::ToastTemplateType_ToastImageAndText01,
|
||||||
|
ImageWithTwoLines = ToastTemplateType::ToastTemplateType_ToastImageAndText02,
|
||||||
|
ImageWithThreeLines = ToastTemplateType::ToastTemplateType_ToastImageAndText03,
|
||||||
|
ImageWithFourLines = ToastTemplateType::ToastTemplateType_ToastImageAndText04,
|
||||||
|
TextOneLine = ToastTemplateType::ToastTemplateType_ToastText01,
|
||||||
|
TextTwoLines = ToastTemplateType::ToastTemplateType_ToastText02,
|
||||||
|
TextThreeLines = ToastTemplateType::ToastTemplateType_ToastText03,
|
||||||
|
TextFourLines = ToastTemplateType::ToastTemplateType_ToastText04,
|
||||||
|
WinToastTemplateTypeCount
|
||||||
|
};
|
||||||
|
|
||||||
|
WinToastTemplate(_In_ const WinToastTemplateType& type = ImageWithTwoLines);
|
||||||
|
~WinToastTemplate();
|
||||||
|
|
||||||
|
int textFieldsCount() const { return _textFields.size(); }
|
||||||
|
bool hasImage() const { return _hasImage; }
|
||||||
|
std::vector<std::wstring> textFields() const { return _textFields; }
|
||||||
|
std::wstring textField(_In_ const TextField& pos) const { return _textFields[pos]; }
|
||||||
|
std::wstring imagePath() const { return _imagePath; }
|
||||||
|
WinToastTemplateType type() const { return _type; }
|
||||||
|
void setTextField(_In_ const std::wstring& txt, _In_ const TextField& pos);
|
||||||
|
void setImagePath(_In_ const std::wstring& imgPath);
|
||||||
|
void setSilent(bool is_silent);
|
||||||
|
bool isSilent() const { return _silent; }
|
||||||
|
private:
|
||||||
|
static int TextFieldsCount[WinToastTemplateTypeCount];
|
||||||
|
bool _hasImage;
|
||||||
|
bool _silent = false;
|
||||||
|
std::vector<std::wstring> _textFields;
|
||||||
|
std::wstring _imagePath;
|
||||||
|
WinToastTemplateType _type;
|
||||||
|
void initComponentsFromType();
|
||||||
|
};
|
||||||
|
|
||||||
|
class WinToast {
|
||||||
|
public:
|
||||||
|
static WinToast* instance();
|
||||||
|
static bool isCompatible();
|
||||||
|
static std::wstring configureAUMI(_In_ const std::wstring& company,
|
||||||
|
_In_ const std::wstring& name,
|
||||||
|
_In_ const std::wstring& surname,
|
||||||
|
_In_ const std::wstring& versionInfo
|
||||||
|
);
|
||||||
|
bool initialize();
|
||||||
|
bool isInitialized() const { return _isInitialized; }
|
||||||
|
bool showToast(_In_ const WinToastTemplate& toast, _In_ WinToastHandler* handler);
|
||||||
|
std::wstring appName() const;
|
||||||
|
std::wstring appUserModelId() const;
|
||||||
|
void setAppUserModelId(_In_ const std::wstring& appName);
|
||||||
|
void setAppName(_In_ const std::wstring& appName);
|
||||||
|
private:
|
||||||
|
bool _isInitialized;
|
||||||
|
std::wstring _appName;
|
||||||
|
std::wstring _aumi;
|
||||||
|
ComPtr<IXmlDocument> _xmlDocument;
|
||||||
|
ComPtr<IToastNotificationManagerStatics> _notificationManager;
|
||||||
|
ComPtr<IToastNotifier> _notifier;
|
||||||
|
ComPtr<IToastNotificationFactory> _notificationFactory;
|
||||||
|
ComPtr<IToastNotification> _notification;
|
||||||
|
static WinToast* _instance;
|
||||||
|
|
||||||
|
WinToast(void);
|
||||||
|
IXmlDocument* xmlDocument() const { return _xmlDocument.Get(); }
|
||||||
|
IToastNotifier* notifier() const { return _notifier.Get(); }
|
||||||
|
IToastNotificationFactory* notificationFactory() const { return _notificationFactory.Get(); }
|
||||||
|
IToastNotificationManagerStatics* notificationManager() const { return _notificationManager.Get(); }
|
||||||
|
IToastNotification* notification() const { return _notification.Get(); }
|
||||||
|
|
||||||
|
HRESULT validateShellLink();
|
||||||
|
HRESULT createShellLink();
|
||||||
|
HRESULT setImageField(_In_ const std::wstring& path);
|
||||||
|
HRESULT setTextField(_In_ const std::wstring& text, _In_ int pos);
|
||||||
|
bool makeSilent(bool is_silent);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // WINTOASTLIB_H
|
|
@ -131,6 +131,7 @@
|
||||||
'atom/browser/api/atom_api_notification.cc',
|
'atom/browser/api/atom_api_notification.cc',
|
||||||
'atom/browser/api/atom_api_notification.h',
|
'atom/browser/api/atom_api_notification.h',
|
||||||
'atom/browser/api/atom_api_notification_mac.mm',
|
'atom/browser/api/atom_api_notification_mac.mm',
|
||||||
|
'atom/browser/api/atom_api_notification_win.cc',
|
||||||
'atom/browser/api/atom_api_power_monitor.cc',
|
'atom/browser/api/atom_api_power_monitor.cc',
|
||||||
'atom/browser/api/atom_api_power_monitor.h',
|
'atom/browser/api/atom_api_power_monitor.h',
|
||||||
'atom/browser/api/atom_api_power_save_blocker.cc',
|
'atom/browser/api/atom_api_power_save_blocker.cc',
|
||||||
|
@ -357,6 +358,10 @@
|
||||||
'atom/browser/ui/win/notify_icon.h',
|
'atom/browser/ui/win/notify_icon.h',
|
||||||
'atom/browser/ui/win/taskbar_host.cc',
|
'atom/browser/ui/win/taskbar_host.cc',
|
||||||
'atom/browser/ui/win/taskbar_host.h',
|
'atom/browser/ui/win/taskbar_host.h',
|
||||||
|
'atom/browser/ui/win/toast_handler.cc',
|
||||||
|
'atom/browser/ui/win/toast_handler.h',
|
||||||
|
'atom/browser/ui/win/toast_lib.cc',
|
||||||
|
'atom/browser/ui/win/toast_lib.h',
|
||||||
'atom/browser/ui/x/event_disabler.cc',
|
'atom/browser/ui/x/event_disabler.cc',
|
||||||
'atom/browser/ui/x/event_disabler.h',
|
'atom/browser/ui/x/event_disabler.h',
|
||||||
'atom/browser/ui/x/window_state_watcher.cc',
|
'atom/browser/ui/x/window_state_watcher.cc',
|
||||||
|
|
Loading…
Reference in a new issue