Coverage Report

Created: 2025-11-16 09:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libreoffice/framework/source/uielement/controlmenucontroller.cxx
Line
Count
Source
1
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2
/*
3
 * This file is part of the LibreOffice project.
4
 *
5
 * This Source Code Form is subject to the terms of the Mozilla Public
6
 * License, v. 2.0. If a copy of the MPL was not distributed with this
7
 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8
 *
9
 * This file incorporates work covered by the following license notice:
10
 *
11
 *   Licensed to the Apache Software Foundation (ASF) under one or more
12
 *   contributor license agreements. See the NOTICE file distributed
13
 *   with this work for additional information regarding copyright
14
 *   ownership. The ASF licenses this file to you under the Apache
15
 *   License, Version 2.0 (the "License"); you may not use this file
16
 *   except in compliance with the License. You may obtain a copy of
17
 *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
18
 */
19
20
#include <sal/config.h>
21
22
#include <com/sun/star/frame/XDispatch.hpp>
23
#include <com/sun/star/frame/XDispatchProvider.hpp>
24
#include <com/sun/star/frame/XFrame.hpp>
25
#include <com/sun/star/frame/XStatusListener.hpp>
26
#include <com/sun/star/util/XURLTransformer.hpp>
27
28
#include <cppuhelper/supportsservice.hxx>
29
#include <vcl/graph.hxx>
30
#include <vcl/svapp.hxx>
31
#include <vcl/settings.hxx>
32
#include <vcl/image.hxx>
33
#include <svtools/popupmenucontrollerbase.hxx>
34
#include <toolkit/awt/vclxmenu.hxx>
35
#include <osl/mutex.hxx>
36
#include <unordered_map>
37
38
#include <classes/fwkresid.hxx>
39
#include <bitmaps.hlst>
40
#include <strings.hrc>
41
42
constexpr OUString aCommands[]
43
{
44
    u".uno:ConvertToEdit"_ustr,
45
    u".uno:ConvertToButton"_ustr,
46
    u".uno:ConvertToFixed"_ustr,
47
    u".uno:ConvertToList"_ustr,
48
    u".uno:ConvertToCheckBox"_ustr,
49
    u".uno:ConvertToRadio"_ustr,
50
    u".uno:ConvertToGroup"_ustr,
51
    u".uno:ConvertToCombo"_ustr,
52
    u".uno:ConvertToImageBtn"_ustr,
53
    u".uno:ConvertToFileControl"_ustr,
54
    u".uno:ConvertToDate"_ustr,
55
    u".uno:ConvertToTime"_ustr,
56
    u".uno:ConvertToNumeric"_ustr,
57
    u".uno:ConvertToCurrency"_ustr,
58
    u".uno:ConvertToPattern"_ustr,
59
    u".uno:ConvertToImageControl"_ustr,
60
    u".uno:ConvertToFormatted"_ustr,
61
    u".uno:ConvertToScrollBar"_ustr,
62
    u".uno:ConvertToSpinButton"_ustr,
63
    u".uno:ConvertToNavigationBar"_ustr
64
};
65
66
const TranslateId aLabels[] =
67
{
68
    RID_STR_PROPTITLE_EDIT,
69
    RID_STR_PROPTITLE_PUSHBUTTON,
70
    RID_STR_PROPTITLE_FIXEDTEXT,
71
    RID_STR_PROPTITLE_LISTBOX,
72
    RID_STR_PROPTITLE_CHECKBOX,
73
    RID_STR_PROPTITLE_RADIOBUTTON,
74
    RID_STR_PROPTITLE_GROUPBOX,
75
    RID_STR_PROPTITLE_COMBOBOX,
76
    RID_STR_PROPTITLE_IMAGEBUTTON,
77
    RID_STR_PROPTITLE_FILECONTROL,
78
    RID_STR_PROPTITLE_DATEFIELD,
79
    RID_STR_PROPTITLE_TIMEFIELD,
80
    RID_STR_PROPTITLE_NUMERICFIELD,
81
    RID_STR_PROPTITLE_CURRENCYFIELD,
82
    RID_STR_PROPTITLE_PATTERNFIELD,
83
    RID_STR_PROPTITLE_IMAGECONTROL,
84
    RID_STR_PROPTITLE_FORMATTED,
85
    RID_STR_PROPTITLE_SCROLLBAR,
86
    RID_STR_PROPTITLE_SPINBUTTON,
87
    RID_STR_PROPTITLE_NAVBAR
88
};
89
90
constexpr OUString aImgIds[]
91
{
92
    RID_SVXBMP_EDITBOX,
93
    RID_SVXBMP_BUTTON,
94
    RID_SVXBMP_FIXEDTEXT,
95
    RID_SVXBMP_LISTBOX,
96
    RID_SVXBMP_CHECKBOX,
97
    RID_SVXBMP_RADIOBUTTON,
98
    RID_SVXBMP_GROUPBOX,
99
    RID_SVXBMP_COMBOBOX,
100
    RID_SVXBMP_IMAGEBUTTON,
101
    RID_SVXBMP_FILECONTROL,
102
    RID_SVXBMP_DATEFIELD,
103
    RID_SVXBMP_TIMEFIELD,
104
    RID_SVXBMP_NUMERICFIELD,
105
    RID_SVXBMP_CURRENCYFIELD,
106
    RID_SVXBMP_PATTERNFIELD,
107
    RID_SVXBMP_IMAGECONTROL,
108
    RID_SVXBMP_FORMATTEDFIELD,
109
    RID_SVXBMP_SCROLLBAR,
110
    RID_SVXBMP_SPINBUTTON,
111
    RID_SVXBMP_NAVIGATIONBAR
112
};
113
114
using namespace css;
115
using namespace css::uno;
116
using namespace css::lang;
117
using namespace css::frame;
118
119
namespace {
120
121
class ControlMenuController :  public svt::PopupMenuControllerBase
122
{
123
    using svt::PopupMenuControllerBase::disposing;
124
125
public:
126
    explicit ControlMenuController( const uno::Reference< uno::XComponentContext >& xContext );
127
128
    // XServiceInfo
129
    virtual OUString SAL_CALL getImplementationName() override
130
0
    {
131
0
        return u"com.sun.star.comp.framework.ControlMenuController"_ustr;
132
0
    }
133
134
    virtual sal_Bool SAL_CALL supportsService(OUString const & ServiceName) override
135
0
    {
136
0
        return cppu::supportsService(this, ServiceName);
137
0
    }
138
139
    virtual css::uno::Sequence<OUString> SAL_CALL getSupportedServiceNames() override
140
0
    {
141
0
        return {u"com.sun.star.frame.PopupMenuController"_ustr};
142
0
    }
143
144
    // XPopupMenuController
145
    virtual void SAL_CALL updatePopupMenu() override;
146
147
    // XStatusListener
148
    virtual void SAL_CALL statusChanged( const frame::FeatureStateEvent& Event ) override;
149
150
    // XMenuListener
151
    virtual void SAL_CALL itemActivated( const awt::MenuEvent& rEvent ) override;
152
153
    // XEventListener
154
    virtual void SAL_CALL disposing( const lang::EventObject& Source ) override;
155
156
private:
157
    // XInitialization
158
    virtual void initializeImpl( std::unique_lock<std::mutex>& rGuard, const uno::Sequence< uno::Any >& aArguments ) override;
159
160
    class UrlToDispatchMap : public std::unordered_map< OUString,
161
                                                        uno::Reference< frame::XDispatch > >
162
    {
163
        public:
164
            void free()
165
0
            {
166
0
                UrlToDispatchMap().swap( *this );// get rid of reserved capacity
167
0
            }
168
    };
169
170
    void updateImagesPopupMenu(Reference<awt::XPopupMenu> const& rPopupMenu);
171
    void fillPopupMenu(uno::Reference<awt::XPopupMenu> const& rPopupMenu);
172
173
    bool                m_bShowMenuImages : 1;
174
    UrlToDispatchMap    m_aURLToDispatchMap;
175
};
176
177
ControlMenuController::ControlMenuController(const css::uno::Reference< css::uno::XComponentContext >& xContext)
178
0
    : svt::PopupMenuControllerBase(xContext)
179
0
{
180
0
    const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
181
0
    m_bShowMenuImages   = rSettings.GetUseImagesInMenus();
182
183
0
}
184
185
// private function
186
void ControlMenuController::updateImagesPopupMenu(Reference<awt::XPopupMenu> const& rPopupMenu)
187
0
{
188
0
    if (!rPopupMenu)
189
0
        return;
190
0
    for (size_t i=0; i < std::size(aCommands); ++i)
191
0
    {
192
0
        sal_Int16 nItemId = i + 1;
193
0
        if (m_bShowMenuImages)
194
0
        {
195
0
            Image aImage(StockImage::Yes, aImgIds[i]);
196
0
            Graphic aGraphic(aImage);
197
0
            rPopupMenu->setItemImage(nItemId, aGraphic.GetXGraphic(), false);
198
0
        }
199
0
        else
200
0
            rPopupMenu->setItemImage(nItemId, nullptr, false);
201
0
    }
202
0
}
203
204
// private function
205
void ControlMenuController::fillPopupMenu( Reference< css::awt::XPopupMenu > const & rPopupMenu )
206
0
{
207
0
    resetPopupMenu( rPopupMenu );
208
209
0
    for (size_t i=0; i < std::size(aCommands); ++i)
210
0
    {
211
0
        sal_Int16 nItemId = i + 1;
212
0
        OUString sCommand(aCommands[i]);
213
0
        rPopupMenu->insertItem(nItemId, FwkResId(aLabels[i]), 0, i);
214
0
        rPopupMenu->setCommand(nItemId, sCommand);
215
0
        rPopupMenu->enableItem(nItemId, false);
216
0
    }
217
218
0
    updateImagesPopupMenu(rPopupMenu);
219
220
0
    rPopupMenu->hideDisabledEntries(true);
221
0
}
222
223
}
224
225
// XEventListener
226
void SAL_CALL ControlMenuController::disposing( const EventObject& )
227
0
{
228
0
    Reference< css::awt::XMenuListener > xHolder(this);
229
230
0
    std::unique_lock aLock( m_aMutex );
231
0
    m_xFrame.clear();
232
0
    m_xDispatch.clear();
233
234
0
    if ( m_xPopupMenu.is() )
235
0
        m_xPopupMenu->removeMenuListener( Reference< css::awt::XMenuListener >(this) );
236
0
    m_xPopupMenu.clear();
237
0
}
238
239
// XStatusListener
240
void SAL_CALL ControlMenuController::statusChanged( const FeatureStateEvent& Event )
241
0
{
242
0
    std::unique_lock aLock( m_aMutex );
243
244
0
    if (!m_xPopupMenu)
245
0
        return;
246
247
0
    sal_Int16 nItemId = 0;
248
0
    for (size_t i=0; i < std::size(aCommands); ++i)
249
0
    {
250
0
        if ( Event.FeatureURL.Complete == aCommands[i] )
251
0
        {
252
0
            nItemId = i + 1;
253
0
            break;
254
0
        }
255
0
    }
256
257
0
    if (!nItemId)
258
0
        return;
259
260
0
    m_xPopupMenu->enableItem(nItemId, Event.IsEnabled);
261
0
}
262
263
// XMenuListener
264
void SAL_CALL ControlMenuController::itemActivated( const css::awt::MenuEvent& )
265
0
{
266
0
    std::unique_lock aLock( m_aMutex );
267
268
0
    SolarMutexGuard aSolarMutexGuard;
269
270
    // Check if some modes have changed so we have to update our menu images
271
0
    const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
272
0
    bool bShowMenuImages = rSettings.GetUseImagesInMenus();
273
274
0
    if (bShowMenuImages != m_bShowMenuImages)
275
0
    {
276
0
        m_bShowMenuImages = bShowMenuImages;
277
0
        updateImagesPopupMenu(m_xPopupMenu);
278
0
    }
279
0
}
280
281
// XPopupMenuController
282
void SAL_CALL ControlMenuController::updatePopupMenu()
283
0
{
284
0
    std::unique_lock aLock( m_aMutex );
285
286
0
    throwIfDisposed(aLock);
287
288
0
    if ( !(m_xFrame.is() && m_xPopupMenu.is()) )
289
0
        return;
290
291
0
    css::util::URL aTargetURL;
292
0
    Reference< XDispatchProvider > xDispatchProvider( m_xFrame, UNO_QUERY );
293
0
    fillPopupMenu( m_xPopupMenu );
294
0
    m_aURLToDispatchMap.free();
295
296
0
    for (const OUString& aCommand : aCommands)
297
0
    {
298
0
        aTargetURL.Complete = aCommand;
299
0
        m_xURLTransformer->parseStrict( aTargetURL );
300
301
0
        Reference< XDispatch > xDispatch = xDispatchProvider->queryDispatch( aTargetURL, OUString(), 0 );
302
0
        if ( xDispatch.is() )
303
0
        {
304
0
            aLock.unlock(); // the addStatusListener will call back into ::statusChanged
305
0
            xDispatch->addStatusListener( static_cast< XStatusListener* >(this), aTargetURL );
306
0
            xDispatch->removeStatusListener( static_cast< XStatusListener* >(this), aTargetURL );
307
0
            aLock.lock();
308
0
            m_aURLToDispatchMap.emplace( aTargetURL.Complete, xDispatch );
309
0
        }
310
0
    }
311
0
}
312
313
// XInitialization
314
void ControlMenuController::initializeImpl( std::unique_lock<std::mutex>& rGuard, const Sequence< Any >& aArguments )
315
0
{
316
0
    svt::PopupMenuControllerBase::initializeImpl(rGuard, aArguments);
317
0
    m_aBaseURL.clear();
318
0
}
319
320
extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface *
321
com_sun_star_comp_framework_ControlMenuController_get_implementation(
322
    css::uno::XComponentContext *context,
323
    css::uno::Sequence<css::uno::Any> const &)
324
0
{
325
0
    return cppu::acquire(new ControlMenuController(context));
326
0
}
327
328
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */