1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <sal/config.h>
26 #include <com/sun/star/beans/NamedValue.hpp>
27 #include <com/sun/star/beans/PropertyValue.hpp>
28 #include <com/sun/star/configuration/theDefaultProvider.hpp>
29 #include <com/sun/star/lang/EventObject.hpp>
30 #include <com/sun/star/lang/Locale.hpp>
31 #include <com/sun/star/lang/XLocalizable.hpp>
32 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
33 #include <com/sun/star/lang/XServiceInfo.hpp>
34 #include <com/sun/star/lang/XSingleComponentFactory.hpp>
35 #include <com/sun/star/uno/Any.hxx>
36 #include <com/sun/star/uno/Exception.hpp>
37 #include <com/sun/star/uno/Reference.hxx>
38 #include <com/sun/star/uno/RuntimeException.hpp>
39 #include <com/sun/star/uno/Sequence.hxx>
40 #include <com/sun/star/uno/XComponentContext.hpp>
41 #include <com/sun/star/uno/XInterface.hpp>
42 #include <com/sun/star/util/XFlushListener.hpp>
43 #include <com/sun/star/util/XFlushable.hpp>
44 #include <com/sun/star/util/XRefreshListener.hpp>
45 #include <com/sun/star/util/XRefreshable.hpp>
46 #include <cppu/unotype.hxx>
47 #include <cppuhelper/basemutex.hxx>
48 #include <cppuhelper/compbase.hxx>
49 #include <cppuhelper/factory.hxx>
50 #include <cppuhelper/implbase.hxx>
51 #include <cppuhelper/interfacecontainer.hxx>
52 #include <cppuhelper/supportsservice.hxx>
53 #include <cppuhelper/weak.hxx>
54 #include <osl/mutex.hxx>
55 #include <sal/types.h>
56 #include <rtl/ref.hxx>
57 #include <rtl/ustring.hxx>
59 #include <i18nlangtag/languagetag.hxx>
61 #include "components.hxx"
62 #include "configurationprovider.hxx"
64 #include "defaultprovider.hxx"
65 #include "rootaccess.hxx"
67 namespace configmgr
{ namespace configuration_provider
{
71 char const accessServiceName
[] =
72 "com.sun.star.configuration.ConfigurationAccess";
73 char const updateAccessServiceName
[] =
74 "com.sun.star.configuration.ConfigurationUpdateAccess";
77 throw css::uno::Exception(
78 ("com.sun.star.configuration.ConfigurationProvider expects a single,"
79 " non-empty, string nodepath argument"),
84 cppu::WeakComponentImplHelper
<
85 css::lang::XServiceInfo
, css::lang::XMultiServiceFactory
,
86 css::util::XRefreshable
, css::util::XFlushable
,
87 css::lang::XLocalizable
>
91 private cppu::BaseMutex
, public ServiceBase
95 const css::uno::Reference
< css::uno::XComponentContext
>& context
):
96 ServiceBase(m_aMutex
), context_(context
), default_(true),
103 const css::uno::Reference
< css::uno::XComponentContext
>& context
,
104 OUString
const & locale
):
105 ServiceBase(m_aMutex
), context_(context
), locale_(locale
),
109 assert(context
.is());
113 Service(const Service
&) = delete;
114 Service
& operator=(const Service
&) = delete;
116 virtual ~Service() override
{}
118 virtual void SAL_CALL
disposing() override
{ flushModifications(); }
120 virtual OUString SAL_CALL
getImplementationName() override
123 ? default_provider::getImplementationName()
124 : configuration_provider::getImplementationName();
127 virtual sal_Bool SAL_CALL
supportsService(OUString
const & ServiceName
) override
128 { return cppu::supportsService(this, ServiceName
); }
130 virtual css::uno::Sequence
< OUString
> SAL_CALL
131 getSupportedServiceNames() override
134 ? default_provider::getSupportedServiceNames()
135 : configuration_provider::getSupportedServiceNames();
138 virtual css::uno::Reference
< css::uno::XInterface
> SAL_CALL
createInstance(
139 OUString
const & aServiceSpecifier
) override
;
141 virtual css::uno::Reference
< css::uno::XInterface
> SAL_CALL
142 createInstanceWithArguments(
143 OUString
const & ServiceSpecifier
,
144 css::uno::Sequence
< css::uno::Any
> const & Arguments
) override
;
146 virtual css::uno::Sequence
< OUString
> SAL_CALL
147 getAvailableServiceNames() override
;
149 virtual void SAL_CALL
refresh() override
;
151 virtual void SAL_CALL
addRefreshListener(
152 css::uno::Reference
< css::util::XRefreshListener
> const & l
) override
;
154 virtual void SAL_CALL
removeRefreshListener(
155 css::uno::Reference
< css::util::XRefreshListener
> const & l
) override
;
157 virtual void SAL_CALL
flush() override
;
159 virtual void SAL_CALL
addFlushListener(
160 css::uno::Reference
< css::util::XFlushListener
> const & l
) override
;
162 virtual void SAL_CALL
removeFlushListener(
163 css::uno::Reference
< css::util::XFlushListener
> const & l
) override
;
165 virtual void SAL_CALL
setLocale(css::lang::Locale
const & eLocale
) override
;
167 virtual css::lang::Locale SAL_CALL
getLocale() override
;
169 void flushModifications() const;
171 css::uno::Reference
< css::uno::XComponentContext
> context_
;
174 std::shared_ptr
<osl::Mutex
> lock_
;
177 css::uno::Reference
< css::uno::XInterface
> Service::createInstance(
178 OUString
const & aServiceSpecifier
)
180 return createInstanceWithArguments(
181 aServiceSpecifier
, css::uno::Sequence
< css::uno::Any
>());
184 css::uno::Reference
< css::uno::XInterface
>
185 Service::createInstanceWithArguments(
186 OUString
const & ServiceSpecifier
,
187 css::uno::Sequence
< css::uno::Any
> const & Arguments
)
191 for (sal_Int32 i
= 0; i
< Arguments
.getLength(); ++i
) {
192 css::beans::NamedValue v1
;
193 css::beans::PropertyValue v2
;
196 if (Arguments
[i
] >>= v1
) {
199 } else if (Arguments
[i
] >>= v2
) {
202 } else if (Arguments
.getLength() == 1 && (Arguments
[i
] >>= nodepath
)) {
203 // For backwards compatibility, allow a single string argument that
205 if (nodepath
.isEmpty()) {
210 throw css::uno::Exception(
211 ("com.sun.star.configuration.ConfigurationProvider expects"
212 " NamedValue or PropertyValue arguments"),
215 // For backwards compatibility, allow "nodepath" and "Locale" in any
217 if (name
.equalsIgnoreAsciiCase("nodepath")) {
218 if (!nodepath
.isEmpty() || !(value
>>= nodepath
) ||
223 } else if (name
.equalsIgnoreAsciiCase("locale")) {
224 if (!locale
.isEmpty() || !(value
>>= locale
) ||
227 throw css::uno::Exception(
228 ("com.sun.star.configuration.ConfigurationProvider expects"
229 " at most one, non-empty, string Locale argument"),
234 if (nodepath
.isEmpty()) {
237 // For backwards compatibility, allow a nodepath that misses the leading
239 if (nodepath
[0] != '/') {
240 nodepath
= "/" + nodepath
;
242 if (locale
.isEmpty()) {
243 //TODO: should the Access use the dynamically changing locale_ instead?
245 if (locale
.isEmpty()) {
250 if (ServiceSpecifier
== accessServiceName
) {
252 } else if (ServiceSpecifier
== updateAccessServiceName
) {
255 throw css::uno::Exception(
256 ("com.sun.star.configuration.ConfigurationProvider does not support"
257 " service " + ServiceSpecifier
),
258 static_cast< cppu::OWeakObject
* >(this));
260 osl::MutexGuard
guard(*lock_
);
261 Components
& components
= Components::getSingleton(context_
);
262 rtl::Reference
< RootAccess
> root(
263 new RootAccess(components
, nodepath
, locale
, update
));
264 if (root
->isValue()) {
265 throw css::uno::Exception(
266 ("com.sun.star.configuration.ConfigurationProvider: there is a leaf"
267 " value at nodepath " + nodepath
),
268 static_cast< cppu::OWeakObject
* >(this));
270 components
.addRootAccess(root
);
271 return static_cast< cppu::OWeakObject
* >(root
.get());
274 css::uno::Sequence
< OUString
> Service::getAvailableServiceNames()
276 css::uno::Sequence
< OUString
> names(2);
277 names
[0] = accessServiceName
;
278 names
[1] = updateAccessServiceName
;
282 void Service::refresh() {
284 cppu::OInterfaceContainerHelper
* cont
= rBHelper
.getContainer(
285 cppu::UnoType
< css::util::XRefreshListener
>::get());
286 if (cont
!= nullptr) {
287 css::lang::EventObject
ev(static_cast< cppu::OWeakObject
* >(this));
288 cont
->notifyEach(&css::util::XRefreshListener::refreshed
, ev
);
292 void Service::addRefreshListener(
293 css::uno::Reference
< css::util::XRefreshListener
> const & l
)
295 rBHelper
.addListener(
296 cppu::UnoType
< css::util::XRefreshListener
>::get(), l
);
299 void Service::removeRefreshListener(
300 css::uno::Reference
< css::util::XRefreshListener
> const & l
)
302 rBHelper
.removeListener(
303 cppu::UnoType
< css::util::XRefreshListener
>::get(), l
);
306 void Service::flush() {
307 flushModifications();
308 cppu::OInterfaceContainerHelper
* cont
= rBHelper
.getContainer(
309 cppu::UnoType
< css::util::XFlushListener
>::get());
310 if (cont
!= nullptr) {
311 css::lang::EventObject
ev(static_cast< cppu::OWeakObject
* >(this));
312 cont
->notifyEach(&css::util::XFlushListener::flushed
, ev
);
316 void Service::addFlushListener(
317 css::uno::Reference
< css::util::XFlushListener
> const & l
)
319 rBHelper
.addListener(cppu::UnoType
< css::util::XFlushListener
>::get(), l
);
322 void Service::removeFlushListener(
323 css::uno::Reference
< css::util::XFlushListener
> const & l
)
325 rBHelper
.removeListener(
326 cppu::UnoType
< css::util::XFlushListener
>::get(), l
);
329 void Service::setLocale(css::lang::Locale
const & eLocale
)
331 osl::MutexGuard
guard(*lock_
);
332 locale_
= LanguageTag::convertToBcp47( eLocale
, false);
335 css::lang::Locale
Service::getLocale() {
336 osl::MutexGuard
guard(*lock_
);
337 css::lang::Locale loc
;
338 if (! locale_
.isEmpty()) {
339 loc
= LanguageTag::convertToLocale( locale_
, false);
344 void Service::flushModifications() const {
345 Components
* components
;
347 osl::MutexGuard
guard(*lock_
);
348 components
= &Components::getSingleton(context_
);
350 components
->flushModifications();
354 public cppu::WeakImplHelper
<
355 css::lang::XSingleComponentFactory
, css::lang::XServiceInfo
>
361 Factory(const Factory
&) = delete;
362 Factory
& operator=(const Factory
&) = delete;
364 virtual ~Factory() override
{}
366 virtual css::uno::Reference
< css::uno::XInterface
> SAL_CALL
367 createInstanceWithContext(
368 css::uno::Reference
< css::uno::XComponentContext
> const & Context
) override
;
370 virtual css::uno::Reference
< css::uno::XInterface
> SAL_CALL
371 createInstanceWithArgumentsAndContext(
372 css::uno::Sequence
< css::uno::Any
> const & Arguments
,
373 css::uno::Reference
< css::uno::XComponentContext
> const & Context
) override
;
375 virtual OUString SAL_CALL
getImplementationName() override
376 { return configuration_provider::getImplementationName(); }
378 virtual sal_Bool SAL_CALL
supportsService(OUString
const & ServiceName
) override
379 { return cppu::supportsService(this, ServiceName
); }
381 virtual css::uno::Sequence
< OUString
> SAL_CALL
382 getSupportedServiceNames() override
383 { return configuration_provider::getSupportedServiceNames(); }
386 css::uno::Reference
< css::uno::XInterface
> Factory::createInstanceWithContext(
387 css::uno::Reference
< css::uno::XComponentContext
> const & Context
)
389 return createInstanceWithArgumentsAndContext(
390 css::uno::Sequence
< css::uno::Any
>(), Context
);
393 css::uno::Reference
< css::uno::XInterface
>
394 Factory::createInstanceWithArgumentsAndContext(
395 css::uno::Sequence
< css::uno::Any
> const & Arguments
,
396 css::uno::Reference
< css::uno::XComponentContext
> const & Context
)
398 if (Arguments
.getLength() == 0) {
399 return css::configuration::theDefaultProvider::get(Context
);
402 for (sal_Int32 i
= 0; i
< Arguments
.getLength(); ++i
) {
403 css::beans::NamedValue v1
;
404 css::beans::PropertyValue v2
;
407 if (Arguments
[i
] >>= v1
) {
410 } else if (Arguments
[i
] >>= v2
) {
414 throw css::uno::Exception(
415 ("com.sun.star.configuration.ConfigurationProvider factory"
416 " expects NamedValue or PropertyValue arguments"),
419 // For backwards compatibility, allow "Locale" and (ignored)
420 // "EnableAsync" in any case:
421 if (name
.equalsIgnoreAsciiCase("locale")) {
422 if (!locale
.isEmpty() || !(value
>>= locale
) ||
425 throw css::uno::Exception(
426 ("com.sun.star.configuration.ConfigurationProvider"
427 " factory expects at most one, non-empty, string"
431 } else if (!name
.equalsIgnoreAsciiCase("enableasync")) {
432 throw css::uno::Exception(
433 ("com.sun.star.configuration.ConfigurationProvider factory:"
434 " unknown argument " + name
),
438 return static_cast< cppu::OWeakObject
* >(new Service(Context
, locale
));
444 css::uno::Reference
< css::uno::XInterface
> createDefault(
445 css::uno::Reference
< css::uno::XComponentContext
> const & context
)
447 return static_cast< cppu::OWeakObject
* >(new Service(context
));
450 OUString
getImplementationName() {
451 return OUString("com.sun.star.comp.configuration.ConfigurationProvider");
454 css::uno::Sequence
< OUString
> getSupportedServiceNames() {
455 return css::uno::Sequence
< OUString
> { "com.sun.star.configuration.ConfigurationProvider" };
458 css::uno::Reference
< css::lang::XSingleComponentFactory
>
460 SAL_UNUSED_PARAMETER
cppu::ComponentFactoryFunc
,
461 SAL_UNUSED_PARAMETER OUString
const &,
462 SAL_UNUSED_PARAMETER
css::uno::Sequence
< OUString
> const &,
463 SAL_UNUSED_PARAMETER rtl_ModuleCount
*)
470 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */