/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static std::map& GetLOKPopupsMap() { // Map to remember the LOKWindowId <-> vcl popup binding. static std::map s_aLOKPopupsMap; return s_aLOKPopupsMap; } namespace { void response_help(vcl::Window* pWindow) { ::Dialog* pDialog = dynamic_cast<::Dialog*>(pWindow); if (!pDialog) return; vcl::Window* pButtonWindow = pDialog->get_widget_for_response(RET_HELP); ::Button* pButton = dynamic_cast<::Button*>(pButtonWindow); if (!pButton) { // Is it a wizard dialog? vcl::RoadmapWizard* pWizard = dynamic_cast(pWindow); if (!pWizard || !pWizard->m_pHelp) return; pWizard->m_pHelp->Click(); return; } pButton->Click(); } } JSDialogNotifyIdle::JSDialogNotifyIdle(VclPtr aNotifierWindow, VclPtr aContentWindow, const OUString& sTypeOfJSON) : Idle("JSDialog notify") , m_aNotifierWindow(std::move(aNotifierWindow)) , m_aContentWindow(std::move(aContentWindow)) , m_sTypeOfJSON(sTypeOfJSON) , m_bForce(false) { SetPriority(TaskPriority::POST_PAINT); } void JSDialogNotifyIdle::forceUpdate() { m_bForce = true; } void JSDialogNotifyIdle::send(const OString& sMsg) { if (!m_aNotifierWindow) { return; } const vcl::ILibreOfficeKitNotifier* pNotifier = m_aNotifierWindow->GetLOKNotifier(); if (pNotifier) { if (m_bForce || sMsg != m_LastNotificationMessage) { m_bForce = false; m_LastNotificationMessage = sMsg; pNotifier->libreOfficeKitViewCallback(LOK_CALLBACK_JSDIALOG, m_LastNotificationMessage); } } } void JSDialogNotifyIdle::sendMessage(jsdialog::MessageType eType, const VclPtr& pWindow, std::unique_ptr pData) { std::scoped_lock aGuard(m_aQueueMutex); // we want only the latest update of same type // TODO: also if we met full update - previous updates are not valid auto it = m_aMessageQueue.begin(); while (it != m_aMessageQueue.end()) { if (it->m_eType == eType && it->m_pWindow == pWindow) { // actions should be always sent, eg. rendering of custom entries in combobox if (eType == jsdialog::MessageType::Action) { it++; continue; } it = m_aMessageQueue.erase(it); } else it++; } JSDialogMessageInfo aMessage(eType, pWindow, std::move(pData)); m_aMessageQueue.push_back(aMessage); } OString JSDialogNotifyIdle::generateFullUpdate() const { if (!m_aContentWindow || !m_aNotifierWindow) return OString(); tools::JsonWriter aJsonWriter; m_aContentWindow->DumpAsPropertyTree(aJsonWriter); if (m_aNotifierWindow) aJsonWriter.put("id", m_aNotifierWindow->GetLOKWindowId()); aJsonWriter.put("jsontype", m_sTypeOfJSON); return aJsonWriter.finishAndGetAsOString(); } OString JSDialogNotifyIdle::generateWidgetUpdate(VclPtr pWindow) const { if (!pWindow || !m_aNotifierWindow) return OString(); tools::JsonWriter aJsonWriter; aJsonWriter.put("jsontype", m_sTypeOfJSON); aJsonWriter.put("action", "update"); if (m_aNotifierWindow) aJsonWriter.put("id", m_aNotifierWindow->GetLOKWindowId()); { auto aEntries = aJsonWriter.startNode("control"); pWindow->DumpAsPropertyTree(aJsonWriter); } return aJsonWriter.finishAndGetAsOString(); } OString JSDialogNotifyIdle::generateCloseMessage() const { tools::JsonWriter aJsonWriter; if (m_aNotifierWindow) aJsonWriter.put("id", m_aNotifierWindow->GetLOKWindowId()); aJsonWriter.put("jsontype", m_sTypeOfJSON); aJsonWriter.put("action", "close"); return aJsonWriter.finishAndGetAsOString(); } OString JSDialogNotifyIdle::generateActionMessage(VclPtr pWindow, std::unique_ptr pData) const { tools::JsonWriter aJsonWriter; aJsonWriter.put("jsontype", m_sTypeOfJSON); aJsonWriter.put("action", "action"); if (m_aNotifierWindow) aJsonWriter.put("id", m_aNotifierWindow->GetLOKWindowId()); { auto aDataNode = aJsonWriter.startNode("data"); aJsonWriter.put("control_id", pWindow->get_id()); for (auto it = pData->begin(); it != pData->end(); it++) aJsonWriter.put(it->first, it->second); } return aJsonWriter.finishAndGetAsOString(); } OString JSDialogNotifyIdle::generatePopupMessage(VclPtr pWindow, OUString sParentId, OUString sCloseId) const { if (!pWindow || !m_aNotifierWindow) return OString(); if (!pWindow->GetParentWithLOKNotifier()) return OString(); tools::JsonWriter aJsonWriter; { auto aChildren = aJsonWriter.startArray("children"); { auto aStruct = aJsonWriter.startStruct(); pWindow->DumpAsPropertyTree(aJsonWriter); } } // try to get the position eg. for the autofilter { vcl::Window* pVclWindow = pWindow.get(); DockingWindow* pDockingWindow = dynamic_cast(pVclWindow); while (pVclWindow && !pDockingWindow) { pVclWindow = pVclWindow->GetParent(); pDockingWindow = dynamic_cast(pVclWindow); } if (pDockingWindow) { Point aPos = pDockingWindow->GetFloatingPos(); aJsonWriter.put("posx", aPos.getX()); aJsonWriter.put("posy", aPos.getY()); if (!pDockingWindow->IsVisible()) aJsonWriter.put("visible", "false"); } } aJsonWriter.put("jsontype", "dialog"); aJsonWriter.put("type", "modalpopup"); aJsonWriter.put("cancellable", true); aJsonWriter.put("popupParent", sParentId); aJsonWriter.put("clickToClose", sCloseId); aJsonWriter.put("id", pWindow->GetParentWithLOKNotifier()->GetLOKWindowId()); return aJsonWriter.finishAndGetAsOString(); } OString JSDialogNotifyIdle::generateClosePopupMessage(OUString sWindowId) const { if (!m_aNotifierWindow) return OString(); tools::JsonWriter aJsonWriter; aJsonWriter.put("jsontype", "dialog"); aJsonWriter.put("type", "modalpopup"); aJsonWriter.put("action", "close"); aJsonWriter.put("id", sWindowId); return aJsonWriter.finishAndGetAsOString(); } void JSDialogNotifyIdle::Invoke() { std::deque aMessageQueue; { std::scoped_lock aGuard(m_aQueueMutex); std::swap(aMessageQueue, m_aMessageQueue); } for (auto& rMessage : aMessageQueue) { jsdialog::MessageType eType = rMessage.m_eType; if (m_sTypeOfJSON == "formulabar" && eType != jsdialog::MessageType::Action) continue; switch (eType) { case jsdialog::MessageType::FullUpdate: send(generateFullUpdate()); break; case jsdialog::MessageType::WidgetUpdate: send(generateWidgetUpdate(rMessage.m_pWindow)); break; case jsdialog::MessageType::Close: send(generateCloseMessage()); break; case jsdialog::MessageType::Action: send(generateActionMessage(rMessage.m_pWindow, std::move(rMessage.m_pData))); break; case jsdialog::MessageType::Popup: send(generatePopupMessage(rMessage.m_pWindow, (*rMessage.m_pData)[PARENT_ID ""_ostr], (*rMessage.m_pData)[CLOSE_ID ""_ostr])); break; case jsdialog::MessageType::PopupClose: send(generateClosePopupMessage((*rMessage.m_pData)[WINDOW_ID ""_ostr])); break; } } } void JSDialogNotifyIdle::clearQueue() { m_aMessageQueue.clear(); } JSDialogSender::~JSDialogSender() COVERITY_NOEXCEPT_FALSE { sendClose(); if (mpIdleNotify) mpIdleNotify->Stop(); } void JSDialogSender::sendFullUpdate(bool bForce) { if (!mpIdleNotify) return; if (bForce) mpIdleNotify->forceUpdate(); mpIdleNotify->sendMessage(jsdialog::MessageType::FullUpdate, nullptr); mpIdleNotify->Start(); } void JSDialogSender::sendClose() { if (!mpIdleNotify || !m_bCanClose) return; mpIdleNotify->clearQueue(); mpIdleNotify->sendMessage(jsdialog::MessageType::Close, nullptr); flush(); } void JSDialogSender::sendUpdate(VclPtr pWindow, bool bForce) { if (!mpIdleNotify) return; if (bForce) mpIdleNotify->forceUpdate(); mpIdleNotify->sendMessage(jsdialog::MessageType::WidgetUpdate, pWindow); mpIdleNotify->Start(); } void JSDialogSender::sendAction(const VclPtr& pWindow, std::unique_ptr pData) { if (!mpIdleNotify) return; mpIdleNotify->sendMessage(jsdialog::MessageType::Action, pWindow, std::move(pData)); mpIdleNotify->Start(); } void JSDialogSender::sendPopup(const VclPtr& pWindow, const OUString& rParentId, const OUString& rCloseId) { if (!mpIdleNotify) return; std::unique_ptr pData = std::make_unique(); (*pData)[PARENT_ID ""_ostr] = rParentId; (*pData)[CLOSE_ID ""_ostr] = rCloseId; mpIdleNotify->sendMessage(jsdialog::MessageType::Popup, pWindow, std::move(pData)); mpIdleNotify->Start(); } void JSDialogSender::sendClosePopup(vcl::LOKWindowId nWindowId) { if (!mpIdleNotify) return; std::unique_ptr pData = std::make_unique(); (*pData)[WINDOW_ID ""_ostr] = OUString::number(nWindowId); mpIdleNotify->sendMessage(jsdialog::MessageType::PopupClose, nullptr, std::move(pData)); flush(); } namespace { vcl::Window* extract_sal_widget(weld::Widget* pParent) { SalInstanceWidget* pInstanceWidget = dynamic_cast(pParent); return pInstanceWidget ? pInstanceWidget->getWidget() : nullptr; } } // Drag and drop namespace { class JSDropTargetDropContext : public cppu::WeakImplHelper { public: JSDropTargetDropContext() {} // XDropTargetDropContext virtual void SAL_CALL acceptDrop(sal_Int8 /*dragOperation*/) override {} virtual void SAL_CALL rejectDrop() override {} virtual void SAL_CALL dropComplete(sal_Bool /*bSuccess*/) override {} }; } static JSTreeView* g_DragSource; JSDropTarget::JSDropTarget() {} void JSDropTarget::initialize(const css::uno::Sequence& /*rArgs*/) {} void JSDropTarget::addDropTargetListener( const css::uno::Reference& xListener) { std::unique_lock aGuard(m_aMutex); m_aListeners.push_back(xListener); } void JSDropTarget::removeDropTargetListener( const css::uno::Reference& xListener) { std::unique_lock aGuard(m_aMutex); std::erase(m_aListeners, xListener); } sal_Bool JSDropTarget::isActive() { return false; } void JSDropTarget::setActive(sal_Bool /*active*/) {} sal_Int8 JSDropTarget::getDefaultActions() { return 0; } void JSDropTarget::setDefaultActions(sal_Int8 /*actions*/) {} OUString JSDropTarget::getImplementationName() { return "com.sun.star.datatransfer.dnd.JSDropTarget"; } sal_Bool JSDropTarget::supportsService(OUString const& ServiceName) { return cppu::supportsService(this, ServiceName); } css::uno::Sequence JSDropTarget::getSupportedServiceNames() { css::uno::Sequence aRet{ "com.sun.star.datatransfer.dnd.JSDropTarget" }; return aRet; } void JSDropTarget::fire_drop(const css::datatransfer::dnd::DropTargetDropEvent& dtde) { std::unique_lock aGuard(m_aMutex); std::vector> aListeners( m_aListeners); aGuard.unlock(); for (auto const& listener : aListeners) { listener->drop(dtde); } } void JSDropTarget::fire_dragEnter(const css::datatransfer::dnd::DropTargetDragEnterEvent& dtde) { std::unique_lock aGuard(m_aMutex); std::vector> aListeners( m_aListeners); aGuard.unlock(); for (auto const& listener : aListeners) { listener->dragEnter(dtde); } } OUString JSInstanceBuilder::getMapIdFromWindowId() const { if (m_sTypeOfJSON == "sidebar" || m_sTypeOfJSON == "notebookbar" || m_sTypeOfJSON == "formulabar") return OUString::number(m_nWindowId) + m_sTypeOfJSON; else return OUString::number(m_nWindowId); } // used for dialogs JSInstanceBuilder::JSInstanceBuilder(weld::Widget* pParent, const OUString& rUIRoot, const OUString& rUIFile, bool bPopup) : SalInstanceBuilder(extract_sal_widget(pParent), rUIRoot, rUIFile) , m_nWindowId(0) , m_aParentDialog(nullptr) , m_aContentWindow(nullptr) , m_sTypeOfJSON("dialog") , m_bHasTopLevelDialog(false) , m_bIsNotebookbar(false) , m_bSentInitialUpdate(false) , m_bIsNestedBuilder(false) , m_aWindowToRelease(nullptr) { // when it is a popup we initialize sender in weld_popover if (bPopup) { m_sTypeOfJSON = "popup"; return; } vcl::Window* pRoot = m_xBuilder->get_widget_root(); if (pRoot && pRoot->GetParent()) { m_aParentDialog = pRoot->GetParent()->GetParentWithLOKNotifier(); if (m_aParentDialog) m_nWindowId = m_aParentDialog->GetLOKWindowId(); InsertWindowToMap(getMapIdFromWindowId()); } initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); } // used for sidebar panels JSInstanceBuilder::JSInstanceBuilder(weld::Widget* pParent, const OUString& rUIRoot, const OUString& rUIFile, sal_uInt64 nLOKWindowId) : SalInstanceBuilder(extract_sal_widget(pParent), rUIRoot, rUIFile) , m_nWindowId(nLOKWindowId) , m_aParentDialog(nullptr) , m_aContentWindow(nullptr) , m_sTypeOfJSON("sidebar") , m_bHasTopLevelDialog(false) , m_bIsNotebookbar(false) , m_bSentInitialUpdate(false) , m_bIsNestedBuilder(false) , m_aWindowToRelease(nullptr) { vcl::Window* pRoot = m_xBuilder->get_widget_root(); m_aParentDialog = pRoot->GetParentWithLOKNotifier(); if (rUIFile == "sfx/ui/panel.ui") { // builder for Panel, get SidebarDockingWindow as m_aContentWindow m_aContentWindow = pRoot; for (int i = 0; i < 7 && m_aContentWindow; i++) m_aContentWindow = m_aContentWindow->GetParent(); } else { // embedded fragments cannot send close message for whole sidebar if (rUIFile == "modules/simpress/ui/customanimationfragment.ui") m_bCanClose = false; // builder for PanelLayout, get SidebarDockingWindow as m_aContentWindow m_aContentWindow = pRoot; for (int i = 0; i < 9 && m_aContentWindow; i++) m_aContentWindow = m_aContentWindow->GetParent(); } InsertWindowToMap(getMapIdFromWindowId()); initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); } // used for notebookbar JSInstanceBuilder::JSInstanceBuilder(vcl::Window* pParent, const OUString& rUIRoot, const OUString& rUIFile, const css::uno::Reference& rFrame, sal_uInt64 nWindowId) : SalInstanceBuilder(pParent, rUIRoot, rUIFile, rFrame) , m_nWindowId(0) , m_aParentDialog(nullptr) , m_aContentWindow(nullptr) , m_sTypeOfJSON("notebookbar") , m_bHasTopLevelDialog(false) , m_bIsNotebookbar(false) , m_bSentInitialUpdate(false) , m_bIsNestedBuilder(false) , m_aWindowToRelease(nullptr) { vcl::Window* pRoot = m_xBuilder->get_widget_root(); if (pRoot && pRoot->GetParent()) { m_aParentDialog = pRoot->GetParent()->GetParentWithLOKNotifier(); if (m_aParentDialog) m_nWindowId = m_aParentDialog->GetLOKWindowId(); if (!m_nWindowId && nWindowId) { m_nWindowId = nWindowId; m_bIsNotebookbar = true; } InsertWindowToMap(getMapIdFromWindowId()); } initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); } // used for formulabar JSInstanceBuilder::JSInstanceBuilder(vcl::Window* pParent, const OUString& rUIRoot, const OUString& rUIFile, sal_uInt64 nLOKWindowId) : SalInstanceBuilder(pParent, rUIRoot, rUIFile) , m_nWindowId(nLOKWindowId) , m_aParentDialog(nullptr) , m_aContentWindow(nullptr) , m_sTypeOfJSON("formulabar") , m_bHasTopLevelDialog(false) , m_bIsNotebookbar(false) , m_bSentInitialUpdate(false) , m_bIsNestedBuilder(false) , m_aWindowToRelease(nullptr) { vcl::Window* pRoot = m_xBuilder->get_widget_root(); m_aContentWindow = pParent; if (pRoot && pRoot->GetParent()) { m_aParentDialog = pRoot->GetParent()->GetParentWithLOKNotifier(); InsertWindowToMap(getMapIdFromWindowId()); } initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); } std::unique_ptr JSInstanceBuilder::CreateDialogBuilder(weld::Widget* pParent, const OUString& rUIRoot, const OUString& rUIFile) { return std::make_unique(pParent, rUIRoot, rUIFile); } std::unique_ptr JSInstanceBuilder::CreateNotebookbarBuilder( vcl::Window* pParent, const OUString& rUIRoot, const OUString& rUIFile, const css::uno::Reference& rFrame, sal_uInt64 nWindowId) { return std::make_unique(pParent, rUIRoot, rUIFile, rFrame, nWindowId); } std::unique_ptr JSInstanceBuilder::CreateSidebarBuilder(weld::Widget* pParent, const OUString& rUIRoot, const OUString& rUIFile, sal_uInt64 nLOKWindowId) { return std::make_unique(pParent, rUIRoot, rUIFile, nLOKWindowId); } std::unique_ptr JSInstanceBuilder::CreatePopupBuilder(weld::Widget* pParent, const OUString& rUIRoot, const OUString& rUIFile) { return std::make_unique(pParent, rUIRoot, rUIFile, true); } std::unique_ptr JSInstanceBuilder::CreateFormulabarBuilder(vcl::Window* pParent, const OUString& rUIRoot, const OUString& rUIFile, sal_uInt64 nLOKWindowId) { return std::make_unique(pParent, rUIRoot, rUIFile, nLOKWindowId); } JSInstanceBuilder::~JSInstanceBuilder() { // tab page closed -> refresh parent window if (m_bIsNestedBuilder && m_sTypeOfJSON == "dialog") { jsdialog::SendFullUpdate(OUString::number(m_nWindowId), "__DIALOG__"); } if (m_sTypeOfJSON == "popup") sendClosePopup(m_nWindowId); if (m_aWindowToRelease) { m_aWindowToRelease->ReleaseLOKNotifier(); m_aWindowToRelease.clear(); } if (m_nWindowId && (m_bHasTopLevelDialog || m_bIsNotebookbar)) { GetLOKWeldWidgetsMap().erase(getMapIdFromWindowId()); } else { auto it = GetLOKWeldWidgetsMap().find(getMapIdFromWindowId()); if (it != GetLOKWeldWidgetsMap().end()) { std::for_each(m_aRememberedWidgets.begin(), m_aRememberedWidgets.end(), [it](const OUString& sId) { it->second.erase(sId); }); } } GetLOKPopupsMap().erase(OUString::number(m_nWindowId)); } std::map& JSInstanceBuilder::GetLOKWeldWidgetsMap() { // Map to remember the LOKWindowId <-> weld widgets binding. static std::map s_aLOKWeldBuildersMap; return s_aLOKWeldBuildersMap; } weld::Widget* JSInstanceBuilder::FindWeldWidgetsMap(const OUString& nWindowId, const OUString& rWidget) { const auto it = GetLOKWeldWidgetsMap().find(nWindowId); if (it != GetLOKWeldWidgetsMap().end()) { auto widgetIt = it->second.find(rWidget); if (widgetIt != it->second.end()) return widgetIt->second; } return nullptr; } void JSInstanceBuilder::InsertWindowToMap(const OUString& nWindowId) { WidgetMap map; auto it = GetLOKWeldWidgetsMap().find(nWindowId); if (it == GetLOKWeldWidgetsMap().end()) GetLOKWeldWidgetsMap().insert({ nWindowId, map }); } void JSInstanceBuilder::RememberWidget(OUString sId, weld::Widget* pWidget) { // do not use the same id for two widgets inside one builder // exception is sidebar where we base our full invalidation on that "Panel" id sharing if (m_sTypeOfJSON != "sidebar") { static std::atomic nNotRepeatIndex = 0; auto aWindowIt = GetLOKWeldWidgetsMap().find(getMapIdFromWindowId()); if (aWindowIt != GetLOKWeldWidgetsMap().end()) { auto aWidgetIt = aWindowIt->second.find(sId); if (aWidgetIt != aWindowIt->second.end()) { unsigned long long int nIndex = nNotRepeatIndex++; // found duplicated it -> add some number to the id and apply to the widget sId = sId + OUString::number(nIndex); SalInstanceWidget* pSalWidget = dynamic_cast(pWidget); assert(pSalWidget && "can only be a SalInstanceWidget"); vcl::Window* pVclWidget = pSalWidget->getWidget(); pVclWidget->set_id(pVclWidget->get_id() + OUString::number(nIndex)); } } } RememberWidget(getMapIdFromWindowId(), sId, pWidget); m_aRememberedWidgets.push_back(sId); } void JSInstanceBuilder::RememberWidget(const OUString& nWindowId, const OUString& id, weld::Widget* pWidget) { auto it = GetLOKWeldWidgetsMap().find(nWindowId); if (it != GetLOKWeldWidgetsMap().end()) { it->second.erase(id); it->second.insert(WidgetMap::value_type(id, pWidget)); } } void JSInstanceBuilder::AddChildWidget(const OUString& nWindowId, const OUString& id, weld::Widget* pWidget) { auto it = GetLOKWeldWidgetsMap().find(nWindowId); if (it != GetLOKWeldWidgetsMap().end()) { it->second.erase(id); it->second.insert(WidgetMap::value_type(id, pWidget)); } } void JSInstanceBuilder::RemoveWindowWidget(const OUString& nWindowId) { auto it = JSInstanceBuilder::GetLOKWeldWidgetsMap().find(nWindowId); if (it != JSInstanceBuilder::GetLOKWeldWidgetsMap().end()) { JSInstanceBuilder::GetLOKWeldWidgetsMap().erase(it); } } void JSInstanceBuilder::RememberPopup(const OUString& nWindowId, VclPtr pWidget) { GetLOKPopupsMap()[nWindowId] = pWidget; } void JSInstanceBuilder::ForgetPopup(const OUString& nWindowId) { auto it = GetLOKPopupsMap().find(nWindowId); if (it != GetLOKPopupsMap().end()) GetLOKPopupsMap().erase(it); } vcl::Window* JSInstanceBuilder::FindPopup(const OUString& nWindowId) { const auto it = GetLOKPopupsMap().find(nWindowId); if (it != GetLOKPopupsMap().end()) return it->second; return nullptr; } const OUString& JSInstanceBuilder::GetTypeOfJSON() const { return m_sTypeOfJSON; } VclPtr& JSInstanceBuilder::GetContentWindow() { if (m_aContentWindow) return m_aContentWindow; else return m_bHasTopLevelDialog ? m_aOwnedToplevel : m_aParentDialog; } VclPtr& JSInstanceBuilder::GetNotifierWindow() { return m_bHasTopLevelDialog ? m_aOwnedToplevel : m_aParentDialog; } std::unique_ptr JSInstanceBuilder::weld_dialog(const OUString& id) { std::unique_ptr pRet; ::Dialog* pDialog = m_xBuilder->get<::Dialog>(id); if (pDialog) { m_nWindowId = pDialog->GetLOKWindowId(); pDialog->SetLOKTunnelingState(false); InsertWindowToMap(getMapIdFromWindowId()); assert(!m_aOwnedToplevel && "only one toplevel per .ui allowed"); m_aOwnedToplevel.set(pDialog); m_xBuilder->drop_ownership(pDialog); m_bHasTopLevelDialog = true; pRet.reset(new JSDialog(this, pDialog, this, false)); RememberWidget("__DIALOG__", pRet.get()); initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); m_bSentInitialUpdate = true; } return pRet; } std::unique_ptr JSInstanceBuilder::weld_assistant(const OUString& id) { vcl::RoadmapWizard* pDialog = m_xBuilder->get(id); std::unique_ptr pRet(pDialog ? new JSAssistant(this, pDialog, this, false) : nullptr); if (pDialog) { m_nWindowId = pDialog->GetLOKWindowId(); pDialog->SetLOKTunnelingState(false); InsertWindowToMap(getMapIdFromWindowId()); assert(!m_aOwnedToplevel && "only one toplevel per .ui allowed"); m_aOwnedToplevel.set(pDialog); m_xBuilder->drop_ownership(pDialog); m_bHasTopLevelDialog = true; pRet.reset(new JSAssistant(this, pDialog, this, false)); RememberWidget("__DIALOG__", pRet.get()); initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); m_bSentInitialUpdate = true; } return pRet; } std::unique_ptr JSInstanceBuilder::weld_message_dialog(const OUString& id) { std::unique_ptr pRet; ::MessageDialog* pMessageDialog = m_xBuilder->get<::MessageDialog>(id); if (pMessageDialog) { m_nWindowId = pMessageDialog->GetLOKWindowId(); pMessageDialog->SetLOKTunnelingState(false); InsertWindowToMap(getMapIdFromWindowId()); assert(!m_aOwnedToplevel && "only one toplevel per .ui allowed"); m_aOwnedToplevel.set(pMessageDialog); m_xBuilder->drop_ownership(pMessageDialog); m_bHasTopLevelDialog = true; initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); m_bSentInitialUpdate = true; } pRet.reset(pMessageDialog ? new JSMessageDialog(this, pMessageDialog, this, false) : nullptr); if (pRet) RememberWidget("__DIALOG__", pRet.get()); return pRet; } std::unique_ptr JSInstanceBuilder::weld_container(const OUString& id) { vcl::Window* pContainer = m_xBuilder->get(id); auto pWeldWidget = pContainer ? std::make_unique(this, pContainer, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); if (!m_bSentInitialUpdate && pContainer) { m_bSentInitialUpdate = true; // use parent builder to send update - avoid multiple calls from many builders vcl::Window* pParent = pContainer->GetParent(); OUString sId = OUString::number(m_nWindowId); while (pParent && !FindWeldWidgetsMap(sId, pParent->get_id())) pParent = pParent->GetParent(); if (pParent) jsdialog::SendFullUpdate(sId, pParent->get_id()); // this is nested builder, don't close parent dialog on destroy (eg. single tab page is closed) m_bCanClose = false; m_bIsNestedBuilder = true; } return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_scrolled_window(const OUString& id, bool bUserManagedScrolling) { VclScrolledWindow* pScrolledWindow = m_xBuilder->get(id); auto pWeldWidget = pScrolledWindow ? std::make_unique(this, pScrolledWindow, this, false, bUserManagedScrolling) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_label(const OUString& id) { Control* pLabel = m_xBuilder->get(id); auto pWeldWidget = std::make_unique(this, pLabel, this, false); if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_button(const OUString& id) { ::Button* pButton = m_xBuilder->get<::Button>(id); auto pWeldWidget = pButton ? std::make_unique(this, pButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_link_button(const OUString& id) { ::FixedHyperlink* pButton = m_xBuilder->get<::FixedHyperlink>(id); auto pWeldWidget = pButton ? std::make_unique(this, pButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_toggle_button(const OUString& id) { ::PushButton* pButton = m_xBuilder->get<::PushButton>(id); auto pWeldWidget = pButton ? std::make_unique(this, pButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_entry(const OUString& id) { Edit* pEntry = m_xBuilder->get(id); auto pWeldWidget = pEntry ? std::make_unique(this, pEntry, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_combo_box(const OUString& id) { vcl::Window* pWidget = m_xBuilder->get(id); ::ComboBox* pComboBox = dynamic_cast<::ComboBox*>(pWidget); std::unique_ptr pWeldWidget; if (pComboBox) { pWeldWidget = std::make_unique(this, pComboBox, this, false); } else { ListBox* pListBox = dynamic_cast(pWidget); pWeldWidget = pListBox ? std::make_unique(this, pListBox, this, false) : nullptr; } if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_notebook(const OUString& id) { std::unique_ptr pWeldWidget; vcl::Window* pNotebook = m_xBuilder->get(id); if (pNotebook && pNotebook->GetType() == WindowType::TABCONTROL) pWeldWidget = std::make_unique(this, static_cast(pNotebook), this, false); else if (pNotebook->GetType() == WindowType::VERTICALTABCONTROL) pWeldWidget = std::make_unique( this, static_cast(pNotebook), this, false); if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_spin_button(const OUString& id) { FormattedField* pSpinButton = m_xBuilder->get(id); auto pWeldWidget = pSpinButton ? std::make_unique(this, pSpinButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_formatted_spin_button(const OUString& id) { FormattedField* pSpinButton = m_xBuilder->get(id); auto pWeldWidget = pSpinButton ? std::make_unique(this, pSpinButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_check_button(const OUString& id) { CheckBox* pCheckButton = m_xBuilder->get(id); auto pWeldWidget = pCheckButton ? std::make_unique(this, pCheckButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_drawing_area(const OUString& id, const a11yref& rA11yImpl, FactoryFunction pUITestFactoryFunction, void* pUserData) { VclDrawingArea* pArea = m_xBuilder->get(id); auto pWeldWidget = pArea ? std::make_unique(this, pArea, this, rA11yImpl, pUITestFactoryFunction, pUserData) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_toolbar(const OUString& id) { ToolBox* pToolBox = m_xBuilder->get(id); auto pWeldWidget = pToolBox ? std::make_unique(this, pToolBox, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_text_view(const OUString& id) { VclMultiLineEdit* pTextView = m_xBuilder->get(id); auto pWeldWidget = pTextView ? std::make_unique(this, pTextView, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_tree_view(const OUString& id) { SvTabListBox* pTreeView = m_xBuilder->get(id); auto pWeldWidget = pTreeView ? std::make_unique(this, pTreeView, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_expander(const OUString& id) { VclExpander* pExpander = m_xBuilder->get(id); auto pWeldWidget = pExpander ? std::make_unique(this, pExpander, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_icon_view(const OUString& id) { ::IconView* pIconView = m_xBuilder->get<::IconView>(id); auto pWeldWidget = pIconView ? std::make_unique(this, pIconView, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_radio_button(const OUString& id) { ::RadioButton* pRadioButton = m_xBuilder->get<::RadioButton>(id); auto pWeldWidget = pRadioButton ? std::make_unique(this, pRadioButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_frame(const OUString& id) { ::VclFrame* pFrame = m_xBuilder->get<::VclFrame>(id); auto pWeldWidget = pFrame ? std::make_unique(this, pFrame, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_menu_button(const OUString& id) { ::MenuButton* pMenuButton = m_xBuilder->get<::MenuButton>(id); auto pWeldWidget = pMenuButton ? std::make_unique(this, pMenuButton, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_popover(const OUString& id) { DockingWindow* pDockingWindow = m_xBuilder->get(id); JSPopover* pPopover = pDockingWindow ? new JSPopover(this, pDockingWindow, this, false) : nullptr; std::unique_ptr pWeldWidget(pPopover); if (pDockingWindow) { assert(!m_aOwnedToplevel && "only one toplevel per .ui allowed"); m_aOwnedToplevel.set(pDockingWindow); m_xBuilder->drop_ownership(pDockingWindow); if (VclPtr pWin = pDockingWindow->GetParentWithLOKNotifier()) { vcl::Window* pPopupRoot = pDockingWindow->GetChild(0); pPopupRoot->SetLOKNotifier(pWin->GetLOKNotifier()); m_aParentDialog = pPopupRoot; m_aWindowToRelease = pPopupRoot; m_nWindowId = m_aParentDialog->GetLOKWindowId(); pPopover->set_window_id(m_nWindowId); JSInstanceBuilder::RememberPopup(OUString::number(m_nWindowId), pDockingWindow); InsertWindowToMap(getMapIdFromWindowId()); initializeSender(GetNotifierWindow(), GetContentWindow(), GetTypeOfJSON()); } } if (pWeldWidget) RememberWidget("__POPOVER__", pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_box(const OUString& id) { VclBox* pContainer = m_xBuilder->get(id); auto pWeldWidget = pContainer ? std::make_unique(this, pContainer, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_widget(const OUString& id) { vcl::Window* pWidget = m_xBuilder->get(id); auto pWeldWidget = pWidget ? std::make_unique(this, pWidget, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_image(const OUString& id) { FixedImage* pImage = m_xBuilder->get(id); auto pWeldWidget = pImage ? std::make_unique(this, pImage, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_level_bar(const OUString& id) { ::ProgressBar* pLevelBar = m_xBuilder->get<::ProgressBar>(id); auto pWeldWidget = pLevelBar ? std::make_unique(this, pLevelBar, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } std::unique_ptr JSInstanceBuilder::weld_calendar(const OUString& id) { ::Calendar* pCalendar = m_xBuilder->get<::Calendar>(id); auto pWeldWidget = pCalendar ? std::make_unique(this, pCalendar, this, false) : nullptr; if (pWeldWidget) RememberWidget(id, pWeldWidget.get()); return pWeldWidget; } weld::MessageDialog* JSInstanceBuilder::CreateMessageDialog(weld::Widget* pParent, VclMessageType eMessageType, VclButtonsType eButtonType, const OUString& rPrimaryMessage, const vcl::ILibreOfficeKitNotifier* pNotifier) { SalInstanceWidget* pParentInstance = dynamic_cast(pParent); SystemWindow* pParentWidget = pParentInstance ? pParentInstance->getSystemWindow() : nullptr; VclPtrInstance<::MessageDialog> xMessageDialog(pParentWidget, rPrimaryMessage, eMessageType, eButtonType); if (pNotifier) xMessageDialog->SetLOKNotifier(pNotifier); pNotifier = xMessageDialog->GetLOKNotifier(); if (pNotifier) { OUString sWindowId = OUString::number(xMessageDialog->GetLOKWindowId()); InsertWindowToMap(sWindowId); xMessageDialog->SetLOKTunnelingState(false); return new JSMessageDialog(xMessageDialog, nullptr, true); } else SAL_WARN("vcl", "No notifier in JSInstanceBuilder::CreateMessageDialog"); return new JSMessageDialog(xMessageDialog, nullptr, true); } JSDialog::JSDialog(JSDialogSender* pSender, ::Dialog* pDialog, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pDialog, pBuilder, bTakeOwnership) { } void JSDialog::collapse(weld::Widget* pEdit, weld::Widget* pButton) { SalInstanceDialog::collapse(pEdit, pButton); sendFullUpdate(); } void JSDialog::undo_collapse() { SalInstanceDialog::undo_collapse(); sendFullUpdate(); } void JSDialog::response(int response) { if (response == RET_HELP) { response_help(m_xWidget.get()); return; } sendClose(); SalInstanceDialog::response(response); } void JSAssistant::response(int response) { if (response == RET_HELP) { response_help(m_xWidget.get()); return; } sendClose(); SalInstanceAssistant::response(response); } int JSDialog::run() { sendFullUpdate(true); int ret = SalInstanceDialog::run(); return ret; } bool JSDialog::runAsync(std::shared_ptr const& rxOwner, const std::function& rEndDialogFn) { bool ret = SalInstanceDialog::runAsync(rxOwner, rEndDialogFn); sendFullUpdate(); return ret; } bool JSDialog::runAsync(std::shared_ptr const& rxSelf, const std::function& func) { bool ret = SalInstanceDialog::runAsync(rxSelf, func); sendFullUpdate(); return ret; } int JSAssistant::run() { sendFullUpdate(true); int ret = SalInstanceDialog::run(); return ret; } bool JSAssistant::runAsync(std::shared_ptr const& rxOwner, const std::function& rEndDialogFn) { bool ret = SalInstanceDialog::runAsync(rxOwner, rEndDialogFn); sendFullUpdate(); return ret; } bool JSAssistant::runAsync(std::shared_ptr const& rxSelf, const std::function& func) { bool ret = SalInstanceDialog::runAsync(rxSelf, func); sendFullUpdate(); return ret; } weld::Button* JSDialog::weld_widget_for_response(int nResponse) { PushButton* pButton = dynamic_cast<::PushButton*>(m_xDialog->get_widget_for_response(nResponse)); auto pWeldWidget = pButton ? new JSButton(m_pSender, pButton, nullptr, false) : nullptr; if (pWeldWidget) { auto pParentDialog = m_xDialog->GetParentWithLOKNotifier(); if (pParentDialog) JSInstanceBuilder::RememberWidget(OUString::number(pParentDialog->GetLOKWindowId()), pButton->get_id(), pWeldWidget); } return pWeldWidget; } weld::Button* JSAssistant::weld_widget_for_response(int nResponse) { ::PushButton* pButton = nullptr; JSButton* pWeldWidget = nullptr; if (nResponse == RET_YES) pButton = m_xWizard->m_pNextPage; else if (nResponse == RET_NO) pButton = m_xWizard->m_pPrevPage; else if (nResponse == RET_OK) pButton = m_xWizard->m_pFinish; else if (nResponse == RET_CANCEL) pButton = m_xWizard->m_pCancel; else if (nResponse == RET_HELP) pButton = m_xWizard->m_pHelp; if (pButton) pWeldWidget = new JSButton(m_pSender, pButton, nullptr, false); if (pWeldWidget) { auto pParentDialog = m_xWizard->GetParentWithLOKNotifier(); if (pParentDialog) JSInstanceBuilder::RememberWidget(OUString::number(pParentDialog->GetLOKWindowId()), pButton->get_id(), pWeldWidget); } return pWeldWidget; } JSAssistant::JSAssistant(JSDialogSender* pSender, vcl::RoadmapWizard* pDialog, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pDialog, pBuilder, bTakeOwnership) { } void JSAssistant::set_current_page(int nPage) { SalInstanceAssistant::set_current_page(nPage); sendFullUpdate(); } void JSAssistant::set_current_page(const OUString& rIdent) { SalInstanceAssistant::set_current_page(rIdent); sendFullUpdate(); } JSContainer::JSContainer(JSDialogSender* pSender, vcl::Window* pContainer, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pContainer, pBuilder, bTakeOwnership) { } JSScrolledWindow::JSScrolledWindow(JSDialogSender* pSender, ::VclScrolledWindow* pContainer, SalInstanceBuilder* pBuilder, bool bTakeOwnership, bool bUserManagedScrolling) : JSWidget( pSender, pContainer, pBuilder, bTakeOwnership, bUserManagedScrolling) { } void JSScrolledWindow::vadjustment_configure(int value, int lower, int upper, int step_increment, int page_increment, int page_size) { SalInstanceScrolledWindow::vadjustment_configure(value, lower, upper, step_increment, page_increment, page_size); sendUpdate(); } void JSScrolledWindow::vadjustment_set_value(int value) { SalInstanceScrolledWindow::vadjustment_set_value(value); sendUpdate(); } void JSScrolledWindow::vadjustment_set_value_no_notification(int value) { SalInstanceScrolledWindow::vadjustment_set_value(value); } void JSScrolledWindow::vadjustment_set_page_size(int size) { SalInstanceScrolledWindow::vadjustment_set_page_size(size); sendUpdate(); } void JSScrolledWindow::set_vpolicy(VclPolicyType eVPolicy) { SalInstanceScrolledWindow::set_vpolicy(eVPolicy); sendUpdate(); } void JSScrolledWindow::hadjustment_configure(int value, int lower, int upper, int step_increment, int page_increment, int page_size) { SalInstanceScrolledWindow::hadjustment_configure(value, lower, upper, step_increment, page_increment, page_size); sendUpdate(); } void JSScrolledWindow::hadjustment_set_value(int value) { SalInstanceScrolledWindow::hadjustment_set_value(value); sendUpdate(); } void JSScrolledWindow::hadjustment_set_value_no_notification(int value) { SalInstanceScrolledWindow::hadjustment_set_value(value); } void JSScrolledWindow::hadjustment_set_page_size(int size) { SalInstanceScrolledWindow::hadjustment_set_page_size(size); sendUpdate(); } void JSScrolledWindow::set_hpolicy(VclPolicyType eVPolicy) { SalInstanceScrolledWindow::set_hpolicy(eVPolicy); sendUpdate(); } JSLabel::JSLabel(JSDialogSender* pSender, Control* pLabel, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pLabel, pBuilder, bTakeOwnership) { } void JSLabel::set_label(const OUString& rText) { SalInstanceLabel::set_label(rText); sendUpdate(); }; JSButton::JSButton(JSDialogSender* pSender, ::Button* pButton, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pButton, pBuilder, bTakeOwnership) { } JSLinkButton::JSLinkButton(JSDialogSender* pSender, ::FixedHyperlink* pButton, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pButton, pBuilder, bTakeOwnership) { } JSToggleButton::JSToggleButton(JSDialogSender* pSender, ::PushButton* pButton, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pButton, pBuilder, bTakeOwnership) { } JSEntry::JSEntry(JSDialogSender* pSender, ::Edit* pEntry, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pEntry, pBuilder, bTakeOwnership) { } void JSEntry::set_text(const OUString& rText) { SalInstanceEntry::set_text(rText); sendUpdate(); } void JSEntry::set_text_without_notify(const OUString& rText) { SalInstanceEntry::set_text(rText); } void JSEntry::replace_selection(const OUString& rText) { SalInstanceEntry::replace_selection(rText); sendUpdate(); } JSListBox::JSListBox(JSDialogSender* pSender, ::ListBox* pListBox, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pListBox, pBuilder, bTakeOwnership) { } void JSListBox::insert(int pos, const OUString& rStr, const OUString* pId, const OUString* pIconName, VirtualDevice* pImageSurface) { SalInstanceComboBoxWithoutEdit::insert(pos, rStr, pId, pIconName, pImageSurface); sendUpdate(); } void JSListBox::remove(int pos) { SalInstanceComboBoxWithoutEdit::remove(pos); sendUpdate(); } void JSListBox::set_active(int pos) { SalInstanceComboBoxWithoutEdit::set_active(pos); sendUpdate(); } JSComboBox::JSComboBox(JSDialogSender* pSender, ::ComboBox* pComboBox, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pComboBox, pBuilder, bTakeOwnership) { } void JSComboBox::insert(int pos, const OUString& rStr, const OUString* pId, const OUString* pIconName, VirtualDevice* pImageSurface) { SalInstanceComboBoxWithEdit::insert(pos, rStr, pId, pIconName, pImageSurface); sendUpdate(); } void JSComboBox::remove(int pos) { SalInstanceComboBoxWithEdit::remove(pos); sendUpdate(); } void JSComboBox::set_entry_text_without_notify(const OUString& rText) { SalInstanceComboBoxWithEdit::set_entry_text(rText); } void JSComboBox::set_entry_text(const OUString& rText) { SalInstanceComboBoxWithEdit::set_entry_text(rText); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "setText"; (*pMap)["text"_ostr] = rText; sendAction(std::move(pMap)); } void JSComboBox::set_active(int pos) { if (pos == get_active()) return; SalInstanceComboBoxWithEdit::set_active(pos); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "select"; (*pMap)["position"_ostr] = OUString::number(pos); sendAction(std::move(pMap)); } void JSComboBox::set_active_id(const OUString& rStr) { sal_uInt16 nPos = find_id(rStr); set_active(nPos); } bool JSComboBox::changed_by_direct_pick() const { return true; } void JSComboBox::render_entry(int pos, int dpix, int dpiy) { ScopedVclPtrInstance pDevice(DeviceFormat::WITH_ALPHA); pDevice->SetDPIX(96.0 * dpix / 100); pDevice->SetDPIY(96.0 * dpiy / 100); Size aRenderSize = signal_custom_get_size(*pDevice); pDevice->SetOutputSize(aRenderSize); signal_custom_render(*pDevice, tools::Rectangle(Point(0, 0), aRenderSize), false, get_id(pos)); BitmapEx aImage = pDevice->GetBitmapEx(Point(0, 0), aRenderSize); SvMemoryStream aOStm(65535, 65535); if (GraphicConverter::Export(aOStm, aImage, ConvertDataFormat::PNG) == ERRCODE_NONE) { css::uno::Sequence aSeq(static_cast(aOStm.GetData()), aOStm.Tell()); OUStringBuffer aBuffer("data:image/png;base64,"); ::comphelper::Base64::encode(aBuffer, aSeq); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "rendered_combobox_entry"; (*pMap)["pos"_ostr] = OUString::number(pos); (*pMap)["image"_ostr] = aBuffer; sendAction(std::move(pMap)); } } JSNotebook::JSNotebook(JSDialogSender* pSender, ::TabControl* pControl, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pControl, pBuilder, bTakeOwnership) { } void JSNotebook::remove_page(const OUString& rIdent) { SalInstanceNotebook::remove_page(rIdent); sendFullUpdate(); } void JSNotebook::insert_page(const OUString& rIdent, const OUString& rLabel, int nPos) { SalInstanceNotebook::insert_page(rIdent, rLabel, nPos); sendFullUpdate(); } JSVerticalNotebook::JSVerticalNotebook(JSDialogSender* pSender, ::VerticalTabControl* pControl, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pControl, pBuilder, bTakeOwnership) { } void JSVerticalNotebook::remove_page(const OUString& rIdent) { SalInstanceVerticalNotebook::remove_page(rIdent); sendFullUpdate(); } void JSVerticalNotebook::insert_page(const OUString& rIdent, const OUString& rLabel, int nPos) { SalInstanceVerticalNotebook::insert_page(rIdent, rLabel, nPos); sendFullUpdate(); } JSSpinButton::JSSpinButton(JSDialogSender* pSender, ::FormattedField* pSpin, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pSpin, pBuilder, bTakeOwnership) { } void JSSpinButton::set_value(sal_Int64 value) { SalInstanceSpinButton::set_value(value); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "setText"; (*pMap)["text"_ostr] = OUString::number(m_rFormatter.GetValue()); sendAction(std::move(pMap)); } void JSSpinButton::set_range(sal_Int64 min, sal_Int64 max) { SalInstanceSpinButton::set_range(min, max); sendUpdate(); } JSFormattedSpinButton::JSFormattedSpinButton(JSDialogSender* pSender, ::FormattedField* pSpin, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pSpin, pBuilder, bTakeOwnership) { } void JSFormattedSpinButton::set_text(const OUString& rText) { SalInstanceFormattedSpinButton::set_text(rText); sendUpdate(); } void JSFormattedSpinButton::set_text_without_notify(const OUString& rText) { SalInstanceFormattedSpinButton::set_text(rText); } JSMessageDialog::JSMessageDialog(JSDialogSender* pSender, ::MessageDialog* pDialog, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pDialog, pBuilder, bTakeOwnership) { } JSMessageDialog::JSMessageDialog(::MessageDialog* pDialog, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(nullptr, pDialog, pBuilder, bTakeOwnership) , m_pOwnedSender(new JSDialogSender(pDialog, pDialog, "dialog")) { m_pSender = m_pOwnedSender.get(); if (pBuilder) return; m_sWindowId = OUString::number(m_xMessageDialog->GetLOKWindowId()); if (::OKButton* pOKBtn = dynamic_cast<::OKButton*>(m_xMessageDialog->get_widget_for_response(RET_OK))) { m_pOK.reset(new JSButton(m_pSender, pOKBtn, nullptr, false)); JSInstanceBuilder::AddChildWidget(m_sWindowId, pOKBtn->get_id(), m_pOK.get()); m_pOK->connect_clicked(LINK(this, JSMessageDialog, OKHdl)); } if (::CancelButton* pCancelBtn = dynamic_cast<::CancelButton*>(m_xMessageDialog->get_widget_for_response(RET_CANCEL))) { m_pCancel.reset(new JSButton(m_pSender, pCancelBtn, nullptr, false)); JSInstanceBuilder::AddChildWidget(m_sWindowId, pCancelBtn->get_id(), m_pCancel.get()); m_pCancel->connect_clicked(LINK(this, JSMessageDialog, CancelHdl)); } } JSMessageDialog::~JSMessageDialog() { if (!m_pBuilder) { // For Message Dialogs created from Application::CreateMessageDialog // (where there is no builder to take care of this for us) explicitly // remove this window id on tear down JSInstanceBuilder::RemoveWindowWidget(m_sWindowId); } } void JSMessageDialog::RememberMessageDialog() { static constexpr OUString sWidgetName = u"__DIALOG__"_ustr; OUString sWindowId = OUString::number(m_xMessageDialog->GetLOKWindowId()); if (JSInstanceBuilder::FindWeldWidgetsMap(sWindowId, sWidgetName) != nullptr) return; JSInstanceBuilder::InsertWindowToMap(sWindowId); JSInstanceBuilder::RememberWidget(sWindowId, sWidgetName, this); } int JSMessageDialog::run() { if (GetLOKNotifier()) { RememberMessageDialog(); sendFullUpdate(); } int bRet = SalInstanceMessageDialog::run(); return bRet; } bool JSMessageDialog::runAsync(std::shared_ptr const& rxOwner, const std::function& rEndDialogFn) { bool bRet = SalInstanceMessageDialog::runAsync(rxOwner, rEndDialogFn); RememberMessageDialog(); sendFullUpdate(); return bRet; } bool JSMessageDialog::runAsync(std::shared_ptr const& rxSelf, const std::function& rEndDialogFn) { bool bRet = SalInstanceMessageDialog::runAsync(rxSelf, rEndDialogFn); RememberMessageDialog(); sendFullUpdate(); return bRet; } IMPL_LINK_NOARG(JSMessageDialog, OKHdl, weld::Button&, void) { response(RET_OK); } IMPL_LINK_NOARG(JSMessageDialog, CancelHdl, weld::Button&, void) { response(RET_CANCEL); } void JSMessageDialog::set_primary_text(const OUString& rText) { SalInstanceMessageDialog::set_primary_text(rText); sendFullUpdate(); } void JSMessageDialog::set_secondary_text(const OUString& rText) { SalInstanceMessageDialog::set_secondary_text(rText); sendFullUpdate(); } void JSMessageDialog::response(int response) { if (response == RET_HELP) { response_help(m_xWidget.get()); return; } sendClose(); SalInstanceMessageDialog::response(response); } JSCheckButton::JSCheckButton(JSDialogSender* pSender, ::CheckBox* pCheckBox, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pCheckBox, pBuilder, bTakeOwnership) { } void JSCheckButton::set_active(bool active) { bool bWasActive = get_active(); SalInstanceCheckButton::set_active(active); if (bWasActive != active) sendUpdate(); } JSDrawingArea::JSDrawingArea(JSDialogSender* pSender, VclDrawingArea* pDrawingArea, SalInstanceBuilder* pBuilder, const a11yref& rAlly, FactoryFunction pUITestFactoryFunction, void* pUserData) : JSWidget(pSender, pDrawingArea, pBuilder, rAlly, std::move(pUITestFactoryFunction), pUserData, false) { } void JSDrawingArea::queue_draw() { SalInstanceDrawingArea::queue_draw(); sendUpdate(); } void JSDrawingArea::queue_draw_area(int x, int y, int width, int height) { SalInstanceDrawingArea::queue_draw_area(x, y, width, height); sendUpdate(); } JSToolbar::JSToolbar(JSDialogSender* pSender, ::ToolBox* pToolbox, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pToolbox, pBuilder, bTakeOwnership) { } void JSToolbar::set_menu_item_active(const OUString& rIdent, bool bActive) { bool bWasActive = get_menu_item_active(rIdent); SalInstanceToolbar::set_menu_item_active(rIdent, bActive); ToolBoxItemId nItemId = m_xToolBox->GetItemId(rIdent); VclPtr pFloat = m_aFloats[nItemId]; if (!pFloat) return; // See WeldToolbarPopup : include/svtools/toolbarmenu.hxx // TopLevel (Popover) -> Container -> main container of the popup vcl::Window* pPopupRoot = pFloat->GetChild(0); if (pPopupRoot) pPopupRoot = pPopupRoot->GetChild(0); if (pPopupRoot) { if (bActive) { JSInstanceBuilder::RememberPopup(OUString::number(pPopupRoot->GetLOKWindowId()), pFloat); sendPopup(pPopupRoot, m_xToolBox->get_id(), rIdent); } else if (bWasActive) { JSInstanceBuilder::ForgetPopup(OUString::number(pPopupRoot->GetLOKWindowId())); sendClosePopup(pPopupRoot->GetLOKWindowId()); } } } void JSToolbar::set_item_sensitive(const OUString& rIdent, bool bSensitive) { bool bWasSensitive = get_item_sensitive(rIdent); SalInstanceToolbar::set_item_sensitive(rIdent, bSensitive); if (bWasSensitive != bSensitive) sendUpdate(); } void JSToolbar::set_item_icon_name(const OUString& rIdent, const OUString& rIconName) { SalInstanceToolbar::set_item_icon_name(rIdent, rIconName); sendUpdate(); } JSTextView::JSTextView(JSDialogSender* pSender, ::VclMultiLineEdit* pTextView, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pTextView, pBuilder, bTakeOwnership) { } void JSTextView::set_text(const OUString& rText) { SalInstanceTextView::set_text(rText); sendUpdate(); } void JSTextView::set_text_without_notify(const OUString& rText) { SalInstanceTextView::set_text(rText); } void JSTextView::replace_selection(const OUString& rText) { SalInstanceTextView::replace_selection(rText); sendUpdate(); } JSTreeView::JSTreeView(JSDialogSender* pSender, ::SvTabListBox* pTreeView, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pTreeView, pBuilder, bTakeOwnership) { } void JSTreeView::set_toggle(int pos, TriState eState, int col) { SvTreeListEntry* pEntry = m_xTreeView->GetEntry(nullptr, 0); while (pEntry && pos--) pEntry = m_xTreeView->Next(pEntry); if (pEntry) { SalInstanceTreeView::set_toggle(pEntry, eState, col); signal_toggled(iter_col(SalInstanceTreeIter(pEntry), col)); sendUpdate(); } } void JSTreeView::set_toggle(const weld::TreeIter& rIter, TriState bOn, int col) { SalInstanceTreeView::set_toggle(rIter, bOn, col); sendUpdate(); } void JSTreeView::select(int pos) { assert(m_xTreeView->IsUpdateMode() && "don't select when frozen"); disable_notify_events(); if (pos == -1 || (pos == 0 && n_children() == 0)) m_xTreeView->SelectAll(false); else { SvTreeListEntry* pEntry = m_xTreeView->GetEntry(nullptr, 0); while (pEntry && pos--) pEntry = m_xTreeView->Next(pEntry); if (pEntry) { m_xTreeView->Select(pEntry, true); m_xTreeView->MakeVisible(pEntry); } } enable_notify_events(); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "select"; (*pMap)["position"_ostr] = OUString::number(pos); sendAction(std::move(pMap)); } weld::TreeView* JSTreeView::get_drag_source() const { return g_DragSource; } void JSTreeView::drag_start() { g_DragSource = this; } void JSTreeView::drag_end() { css::datatransfer::dnd::XDropTarget* xDropTarget = m_xDropTarget.get(); if (xDropTarget) { css::datatransfer::dnd::DropTargetDropEvent aEvent; aEvent.Source = xDropTarget; aEvent.Context = new JSDropTargetDropContext(); // dummy values aEvent.LocationX = 50; aEvent.LocationY = 50; aEvent.DropAction = css::datatransfer::dnd::DNDConstants::ACTION_DEFAULT; aEvent.SourceActions = css::datatransfer::dnd::DNDConstants::ACTION_DEFAULT; m_xDropTarget->fire_drop(aEvent); sendUpdate(); if (g_DragSource) g_DragSource->sendUpdate(); } g_DragSource = nullptr; } void JSTreeView::insert(const weld::TreeIter* pParent, int pos, const OUString* pStr, const OUString* pId, const OUString* pIconName, VirtualDevice* pImageSurface, bool bChildrenOnDemand, weld::TreeIter* pRet) { SalInstanceTreeView::insert(pParent, pos, pStr, pId, pIconName, pImageSurface, bChildrenOnDemand, pRet); sendUpdate(); } void JSTreeView::set_text(int row, const OUString& rText, int col) { SalInstanceTreeView::set_text(row, rText, col); sendUpdate(); } void JSTreeView::set_text(const weld::TreeIter& rIter, const OUString& rStr, int col) { SalInstanceTreeView::set_text(rIter, rStr, col); sendUpdate(); } void JSTreeView::remove(int pos) { SalInstanceTreeView::remove(pos); sendUpdate(); } void JSTreeView::remove(const weld::TreeIter& rIter) { SalInstanceTreeView::remove(rIter); sendUpdate(); } void JSTreeView::clear() { SalInstanceTreeView::clear(); sendUpdate(); } void JSTreeView::set_cursor_without_notify(const weld::TreeIter& rIter) { SalInstanceTreeView::set_cursor(rIter); } void JSTreeView::set_cursor(const weld::TreeIter& rIter) { SalInstanceTreeView::set_cursor(rIter); sendUpdate(); } void JSTreeView::set_cursor(int pos) { SalInstanceTreeView::set_cursor(pos); sendUpdate(); } void JSTreeView::expand_row(const weld::TreeIter& rIter) { bool bNotify = false; const SalInstanceTreeIter& rVclIter = static_cast(rIter); if (!m_xTreeView->IsExpanded(rVclIter.iter)) bNotify = true; SalInstanceTreeView::expand_row(rIter); if (bNotify) sendUpdate(); } void JSTreeView::collapse_row(const weld::TreeIter& rIter) { bool bNotify = false; const SalInstanceTreeIter& rVclIter = static_cast(rIter); if (m_xTreeView->IsExpanded(rVclIter.iter)) bNotify = true; SalInstanceTreeView::collapse_row(rIter); if (bNotify) sendUpdate(); } JSExpander::JSExpander(JSDialogSender* pSender, ::VclExpander* pExpander, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pExpander, pBuilder, bTakeOwnership) { } void JSExpander::set_expanded(bool bExpand) { SalInstanceExpander::set_expanded(bExpand); sendUpdate(); } JSIconView::JSIconView(JSDialogSender* pSender, ::IconView* pIconView, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pIconView, pBuilder, bTakeOwnership) { } void JSIconView::insert(int pos, const OUString* pStr, const OUString* pId, const OUString* pIconName, weld::TreeIter* pRet) { SalInstanceIconView::insert(pos, pStr, pId, pIconName, pRet); sendUpdate(); } void JSIconView::insert(int pos, const OUString* pStr, const OUString* pId, const VirtualDevice* pIcon, weld::TreeIter* pRet) { SalInstanceIconView::insert(pos, pStr, pId, pIcon, pRet); sendUpdate(); } void JSIconView::insert_separator(int pos, const OUString* pId) { SalInstanceIconView::insert_separator(pos, pId); sendUpdate(); } void JSIconView::clear() { SalInstanceIconView::clear(); sendUpdate(); } void JSIconView::select(int pos) { SalInstanceIconView::select(pos); std::unique_ptr pMap = std::make_unique(); (*pMap)[ACTION_TYPE ""_ostr] = "select"; (*pMap)["position"_ostr] = OUString::number(pos); sendAction(std::move(pMap)); } void JSIconView::unselect(int pos) { SalInstanceIconView::unselect(pos); sendUpdate(); } JSRadioButton::JSRadioButton(JSDialogSender* pSender, ::RadioButton* pRadioButton, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pRadioButton, pBuilder, bTakeOwnership) { } void JSRadioButton::set_active(bool active) { SalInstanceRadioButton::set_active(active); sendUpdate(); } JSFrame::JSFrame(JSDialogSender* pSender, ::VclFrame* pFrame, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pFrame, pBuilder, bTakeOwnership) { } JSMenuButton::JSMenuButton(JSDialogSender* pSender, ::MenuButton* pMenuButton, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pMenuButton, pBuilder, bTakeOwnership) { } void JSMenuButton::set_label(const OUString& rText) { OUString aPreviousLabel = get_label(); SalInstanceMenuButton::set_label(rText); if (aPreviousLabel != rText) sendUpdate(); } void JSMenuButton::set_image(VirtualDevice* pDevice) { SalInstanceMenuButton::set_image(pDevice); sendUpdate(); } void JSMenuButton::set_image(const css::uno::Reference& rImage) { SalInstanceMenuButton::set_image(rImage); sendUpdate(); } void JSMenuButton::set_active(bool bActive) { SalInstanceMenuButton::set_active(bActive); VclPtr pPopup = m_xMenuButton->GetPopover(); if (pPopup) { if (bActive) sendPopup(pPopup->GetChild(0), m_xMenuButton->get_id(), m_xMenuButton->get_id()); else sendClosePopup(pPopup->GetChild(0)->GetLOKWindowId()); } } JSPopover::JSPopover(JSDialogSender* pSender, DockingWindow* pDockingWindow, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pDockingWindow, pBuilder, bTakeOwnership) , mnWindowId(0) { } void JSPopover::popup_at_rect(weld::Widget* pParent, const tools::Rectangle& rRect, weld::Placement ePlace) { SalInstancePopover::popup_at_rect(pParent, rRect, ePlace); sendPopup(getWidget()->GetChild(0), "_POPOVER_", "_POPOVER_"); } void JSPopover::popdown() { vcl::Window* pPopup = JSInstanceBuilder::FindPopup(OUString::number(mnWindowId)); if (pPopup) { sendClosePopup(mnWindowId); vcl::Window::GetDockingManager()->EndPopupMode(pPopup); } if (getWidget() && getWidget()->GetChild(0)) sendClosePopup(getWidget()->GetChild(0)->GetLOKWindowId()); SalInstancePopover::popdown(); } JSBox::JSBox(JSDialogSender* pSender, VclBox* pBox, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pBox, pBuilder, bTakeOwnership) { } void JSBox::reorder_child(weld::Widget* pWidget, int nNewPosition) { SalInstanceBox::reorder_child(pWidget, nNewPosition); sendUpdate(); } JSImage::JSImage(JSDialogSender* pSender, FixedImage* pImage, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pImage, pBuilder, bTakeOwnership) { } void JSImage::set_image(VirtualDevice* pDevice) { SalInstanceImage::set_image(pDevice); sendUpdate(); } void JSImage::set_image(const css::uno::Reference& rImage) { SalInstanceImage::set_image(rImage); sendUpdate(); } JSLevelBar::JSLevelBar(JSDialogSender* pSender, ::ProgressBar* pProgressBar, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pProgressBar, pBuilder, bTakeOwnership) { } void JSLevelBar::set_percentage(double fPercentage) { SalInstanceLevelBar::set_percentage(fPercentage); sendUpdate(); } JSCalendar::JSCalendar(JSDialogSender* pSender, ::Calendar* pCalendar, SalInstanceBuilder* pBuilder, bool bTakeOwnership) : JSWidget(pSender, pCalendar, pBuilder, bTakeOwnership) { } /* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s cinkeys+=0=break: */