|
- /*
- ==============================================================================
-
- This file is part of the JUCE library.
- Copyright (c) 2022 - Raw Material Software Limited
-
- JUCE is an open source library subject to commercial or open-source
- licensing.
-
- By using JUCE, you agree to the terms of both the JUCE 7 End-User License
- Agreement and JUCE Privacy Policy.
-
- End User License Agreement: www.juce.com/juce-7-licence
- Privacy Policy: www.juce.com/juce-privacy-policy
-
- Or: You may also use this code under the terms of the GPL v3 (see
- www.gnu.org/licenses).
-
- JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
- EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
- DISCLAIMED.
-
- ==============================================================================
- */
-
- namespace juce::detail
- {
-
- #if JUCE_MSVC
- // required to enable the newer dialog box on vista and above
- #pragma comment(linker, \
- "\"/MANIFESTDEPENDENCY:type='Win32' " \
- "name='Microsoft.Windows.Common-Controls' " \
- "version='6.0.0.0' " \
- "processorArchitecture='*' " \
- "publicKeyToken='6595b64144ccf1df' " \
- "language='*'\"" \
- )
- #endif
-
- std::unique_ptr<ScopedMessageBoxInterface> ScopedMessageBoxInterface::create (const MessageBoxOptions& options)
- {
- class WindowsMessageBoxBase : public ScopedMessageBoxInterface
- {
- public:
- explicit WindowsMessageBoxBase (Component* comp)
- : associatedComponent (comp) {}
-
- void runAsync (std::function<void (int)> recipient) override
- {
- future = std::async (std::launch::async, [showMessageBox = getShowMessageBox(), recipient]
- {
- const auto initComResult = CoInitializeEx (nullptr, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);
-
- if (initComResult != S_OK)
- return;
-
- const ScopeGuard scope { [] { CoUninitialize(); } };
-
- const auto messageResult = showMessageBox != nullptr ? showMessageBox() : 0;
- NullCheckedInvocation::invoke (recipient, messageResult);
- });
- }
-
- int runSync() override
- {
- if (auto showMessageBox = getShowMessageBox())
- return showMessageBox();
-
- return 0;
- }
-
- void close() override
- {
- if (auto* toClose = windowHandle.exchange (nullptr))
- EndDialog (toClose, 0);
- }
-
- void setDialogWindowHandle (HWND dialogHandle)
- {
- windowHandle = dialogHandle;
- }
-
- private:
- std::function<int()> getShowMessageBox()
- {
- const auto parent = associatedComponent != nullptr ? (HWND) associatedComponent->getWindowHandle() : nullptr;
- return getShowMessageBoxForParent (parent);
- }
-
- /* Returns a function that should display a message box and return the result.
-
- getShowMessageBoxForParent() will be called on the message thread.
-
- The returned function will be called on a separate thread, in order to avoid blocking the
- message thread.
-
- 'this' is guaranteed to be alive when the returned function is called.
- */
- virtual std::function<int()> getShowMessageBoxForParent (HWND parent) = 0;
-
- Component::SafePointer<Component> associatedComponent;
- std::atomic<HWND> windowHandle { nullptr };
- std::future<void> future;
-
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (WindowsMessageBoxBase)
- };
-
- class PreVistaMessageBox final : public WindowsMessageBoxBase
- {
- public:
- PreVistaMessageBox (const MessageBoxOptions& opts, UINT extraFlags)
- : WindowsMessageBoxBase (opts.getAssociatedComponent()),
- flags (extraFlags | getMessageBoxFlags (opts.getIconType())),
- title (opts.getTitle()), message (opts.getMessage()) {}
-
- private:
- std::function<int()> getShowMessageBoxForParent (const HWND parent) override
- {
- JUCE_ASSERT_MESSAGE_THREAD
-
- static std::map<DWORD, PreVistaMessageBox*> map;
- static std::mutex mapMutex;
-
- return [this, parent]
- {
- const auto threadId = GetCurrentThreadId();
-
- {
- const std::scoped_lock scope { mapMutex };
- map.emplace (threadId, this);
- }
-
- const ScopeGuard eraseFromMap { [threadId]
- {
- const std::scoped_lock scope { mapMutex };
- map.erase (threadId);
- } };
-
- const auto hookCallback = [] (int nCode, const WPARAM wParam, const LPARAM lParam)
- {
- auto* params = reinterpret_cast<CWPSTRUCT*> (lParam);
-
- if (nCode >= 0
- && params != nullptr
- && (params->message == WM_INITDIALOG || params->message == WM_DESTROY))
- {
- const auto callbackThreadId = GetCurrentThreadId();
-
- const std::scoped_lock scope { mapMutex };
-
- if (const auto iter = map.find (callbackThreadId); iter != map.cend())
- iter->second->setDialogWindowHandle (params->message == WM_INITDIALOG ? params->hwnd : nullptr);
- }
-
- return CallNextHookEx ({}, nCode, wParam, lParam);
- };
-
- const auto hook = SetWindowsHookEx (WH_CALLWNDPROC,
- hookCallback,
- (HINSTANCE) juce::Process::getCurrentModuleInstanceHandle(),
- threadId);
- const ScopeGuard removeHook { [hook] { UnhookWindowsHookEx (hook); } };
-
- const auto result = MessageBox (parent, message.toWideCharPointer(), title.toWideCharPointer(), flags);
-
- if (result == IDYES || result == IDOK) return 0;
- if (result == IDNO && ((flags & 1) != 0)) return 1;
-
- return 2;
- };
- }
-
- static UINT getMessageBoxFlags (MessageBoxIconType iconType) noexcept
- {
- // this window can get lost behind JUCE windows which are set to be alwaysOnTop
- // so if there are any set it to be topmost
- const auto topmostFlag = WindowUtils::areThereAnyAlwaysOnTopWindows() ? MB_TOPMOST : 0;
-
- const auto iconFlags = [&]() -> decltype (topmostFlag)
- {
- switch (iconType)
- {
- case MessageBoxIconType::QuestionIcon: return MB_ICONQUESTION;
- case MessageBoxIconType::WarningIcon: return MB_ICONWARNING;
- case MessageBoxIconType::InfoIcon: return MB_ICONINFORMATION;
- case MessageBoxIconType::NoIcon: break;
- }
-
- return 0;
- }();
-
- return static_cast<UINT> (MB_TASKMODAL | MB_SETFOREGROUND | topmostFlag | iconFlags);
- }
-
- const UINT flags;
- const String title, message;
-
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (PreVistaMessageBox)
- };
-
- class WindowsTaskDialog final : public WindowsMessageBoxBase
- {
- static auto getTaskDialogFunc()
- {
- using TaskDialogIndirectFunc = HRESULT (WINAPI*) (const TASKDIALOGCONFIG*, INT*, INT*, BOOL*);
-
- static const auto result = [&]() -> TaskDialogIndirectFunc
- {
- if (SystemStats::getOperatingSystemType() < SystemStats::WinVista)
- return nullptr;
-
- const auto comctl = "Comctl32.dll";
- LoadLibraryA (comctl);
- const auto comctlModule = GetModuleHandleA (comctl);
-
- JUCE_BEGIN_IGNORE_WARNINGS_GCC_LIKE ("-Wcast-function-type")
- if (comctlModule != nullptr)
- return (TaskDialogIndirectFunc) GetProcAddress (comctlModule, "TaskDialogIndirect");
- JUCE_END_IGNORE_WARNINGS_GCC_LIKE
-
- return nullptr;
- }();
-
- return result;
- }
-
- public:
- explicit WindowsTaskDialog (const MessageBoxOptions& opts)
- : WindowsMessageBoxBase (opts.getAssociatedComponent()),
- iconType (opts.getIconType()),
- title (opts.getTitle()), message (opts.getMessage()),
- buttons { opts.getButtonText (0), opts.getButtonText (1), opts.getButtonText (2) } {}
-
- static bool isAvailable()
- {
- return getTaskDialogFunc() != nullptr;
- }
-
- private:
- std::function<int()> getShowMessageBoxForParent (const HWND parent) override
- {
- JUCE_ASSERT_MESSAGE_THREAD
-
- return [this, parent]
- {
- TASKDIALOGCONFIG config{};
-
- config.cbSize = sizeof (config);
- config.hwndParent = parent;
- config.pszWindowTitle = title.toWideCharPointer();
- config.pszContent = message.toWideCharPointer();
- config.hInstance = (HINSTANCE) Process::getCurrentModuleInstanceHandle();
- config.lpCallbackData = reinterpret_cast<LONG_PTR> (this);
- config.pfCallback = [] (HWND hwnd, UINT msg, WPARAM, LPARAM, LONG_PTR lpRefData)
- {
- if (auto* t = reinterpret_cast<WindowsTaskDialog*> (lpRefData))
- {
- switch (msg)
- {
- case TDN_CREATED:
- case TDN_DIALOG_CONSTRUCTED:
- t->setDialogWindowHandle (hwnd);
- break;
-
- case TDN_DESTROYED:
- t->setDialogWindowHandle (nullptr);
- break;
- }
- }
-
- return S_OK;
- };
-
- if (iconType == MessageBoxIconType::QuestionIcon)
- {
- if (auto* questionIcon = LoadIcon (nullptr, IDI_QUESTION))
- {
- config.hMainIcon = questionIcon;
- config.dwFlags |= TDF_USE_HICON_MAIN;
- }
- }
- else
- {
- config.pszMainIcon = [&]() -> LPWSTR
- {
- switch (iconType)
- {
- case MessageBoxIconType::WarningIcon: return TD_WARNING_ICON;
- case MessageBoxIconType::InfoIcon: return TD_INFORMATION_ICON;
-
- case MessageBoxIconType::QuestionIcon: JUCE_FALLTHROUGH
- case MessageBoxIconType::NoIcon:
- break;
- }
-
- return nullptr;
- }();
- }
-
- std::vector<TASKDIALOG_BUTTON> buttonLabels;
-
- for (const auto& buttonText : buttons)
- if (buttonText.isNotEmpty())
- buttonLabels.push_back ({ (int) buttonLabels.size(), buttonText.toWideCharPointer() });
-
- config.pButtons = buttonLabels.data();
- config.cButtons = (UINT) buttonLabels.size();
-
- int buttonIndex = 0;
-
- if (auto* func = getTaskDialogFunc())
- func (&config, &buttonIndex, nullptr, nullptr);
- else
- jassertfalse;
-
- return buttonIndex;
- };
- }
-
- const MessageBoxIconType iconType;
- const String title, message;
- const std::array<String, 3> buttons;
-
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (WindowsTaskDialog)
- };
-
- if (WindowsTaskDialog::isAvailable())
- return std::make_unique<WindowsTaskDialog> (options);
-
- const auto extraFlags = [&options]
- {
- const auto numButtons = options.getNumButtons();
-
- if (numButtons == 3)
- return MB_YESNOCANCEL;
-
- if (numButtons == 2)
- return options.getButtonText (0) == "OK" ? MB_OKCANCEL
- : MB_YESNO;
-
- return MB_OK;
- }();
-
- return std::make_unique<PreVistaMessageBox> (options, (UINT) extraFlags);
- }
-
- } // namespace juce::detail
|