Files
libreoffice/configmgr2/source/childaccess.cxx

264 lines
7.8 KiB
C++
Raw Normal View History

/*************************************************************************
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright 2009 by Sun Microsystems, Inc.
*
* OpenOffice.org - a multi-platform office productivity suite
*
* $RCSfile: code,v $
*
* $Revision: 1.4 $
*
* This file is part of OpenOffice.org.
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
*
* OpenOffice.org is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
* <http://www.openoffice.org/license.html>
* for a copy of the LGPLv3 License.
************************************************************************/
#include "precompiled_configmgr.hxx"
#include "sal/config.h"
#include <vector>
#include "com/sun/star/lang/NoSupportException.hpp"
#include "com/sun/star/uno/Any.hxx"
#include "com/sun/star/uno/Reference.hxx"
#include "com/sun/star/uno/RuntimeException.hpp"
#include "com/sun/star/uno/Sequence.hxx"
#include "com/sun/star/uno/XInterface.hpp"
#include "com/sun/star/util/ElementChange.hpp"
#include "cppuhelper/implbase1.hxx"
#include "cppuhelper/weak.hxx"
#include "osl/diagnose.h"
#include "osl/interlck.h"
#include "osl/mutex.hxx"
#include "rtl/ref.hxx"
#include "rtl/ustring.h"
#include "rtl/ustring.hxx"
#include "rtl/uuid.h"
#include "sal/types.h"
#include "access.hxx"
#include "childaccess.hxx"
#include "components.hxx"
#include "localizedpropertynode.hxx"
#include "localizedpropertyvaluenode.hxx"
#include "lock.hxx"
#include "node.hxx"
#include "propertynode.hxx"
#include "rootaccess.hxx"
namespace configmgr {
namespace {
namespace css = com::sun::star;
}
css::uno::Sequence< sal_Int8 > ChildAccess::getTunnelId() {
static css::uno::Sequence< sal_Int8 > id;
if (id.getLength() == 0) {
css::uno::Sequence< sal_Int8 > uuid(16);
rtl_createUuid(
reinterpret_cast< sal_uInt8 * >(uuid.getArray()), 0, false);
id = uuid;
}
return id;
}
ChildAccess::ChildAccess(
rtl::Reference< RootAccess > const & root,
rtl::Reference< Access > const & parent,
rtl::Reference< Node > const & node):
root_(root), parent_(parent), node_(node), status_(STATUS_UNMODIFIED)
{
OSL_ASSERT(root.is() && parent.is() && node.is());
}
ChildAccess::ChildAccess(
rtl::Reference< RootAccess > const & root,
rtl::Reference< Node > const & node):
root_(root), node_(node), status_(STATUS_UNMODIFIED)
{
OSL_ASSERT(root.is() && node.is());
}
rtl::Reference< Node > ChildAccess::getNode() {
return node_;
}
rtl::Reference< RootAccess > ChildAccess::getRoot() {
return root_;
}
oslInterlockedCount ChildAccess::acquireCounting() {
return osl_incrementInterlockedCount(&m_refCount);
}
void ChildAccess::releaseNondeleting() {
osl_decrementInterlockedCount(&m_refCount);
}
void ChildAccess::bind(
rtl::Reference< RootAccess > const & root,
rtl::Reference< Access > const & parent) throw ()
{
OSL_ASSERT(!parent_.is() && root.is() && parent.is());
root_ = root;
parent_ = parent;
}
void ChildAccess::unbind() throw () {
OSL_ASSERT(parent_.is());
parent_.clear();
}
void ChildAccess::reportChanges(
std::vector< css::util::ElementChange > * changes) const
{
OSL_ASSERT(changes != 0);
switch (status_) {
case STATUS_CHANGED:
changes->push_back(css::util::ElementChange()); //TODO
break;
case STATUS_ADDED:
changes->push_back(css::util::ElementChange()); //TODO
// fall through
case STATUS_UNMODIFIED:
for (HardChildMap::const_iterator i(modifiedChildren_.begin());
i != modifiedChildren_.end(); ++i)
{
i->second->reportChanges(changes);
}
break;
case STATUS_REMOVED:
changes->push_back(css::util::ElementChange()); //TODO
break;
}
}
void ChildAccess::commitChanges() {
/*
switch (status_) {
case STATUS_UNMODIFIED:
case STATUS_CHANGED:
case STATUS_ADDED:
{
rtl::Reference< Node > p(parent_->getNode());
if (GroupNode * group = dynamic_cast< GroupNode * >(p.get())) {
group->getMembers()[getNode()->getName()] = getNode();
} else if (SetNode * set = dynamic_cast< SetNode * >(p.get())) {
set->getMembers()[getNode()->getName()] = getNode();
} else {
OSL_ASSERT(false);
throw css::uno::RuntimeException(
rtl::OUString(
RTL_CONSTASCII_USTRINGPARAM("this cannot happen")),
static_cast< cppu::OWeakObject * >(this));
}
}
break;
case STATUS_REMOVED:
{
rtl::Reference< Node > p(parent_->getNode());
if (GroupNode * group = dynamic_cast< GroupNode * >(p.get())) {
rtl::OUString name;
changeData_ >>= name;
group->getMembers().erase(name);
} else if (SetNode * set = dynamic_cast< SetNode * >(p.get())) {
set->getMembers()[getNode()->getName()] = getNode();
} else {
OSL_ASSERT(false);
throw css::uno::RuntimeException(
rtl::OUString(
RTL_CONSTASCII_USTRINGPARAM("this cannot happen")),
static_cast< cppu::OWeakObject * >(this));
}
}
break;
}
*/
}
void ChildAccess::setStatus(Status status, css::uno::Any const & changeData) {
//TODO: parent hierarchy
status_ = status;
changeData_ = changeData;
}
css::uno::Any ChildAccess::asValue() {
if (status_ == STATUS_CHANGED) {
return changeData_;
}
rtl::Reference< Node > p(getNode());
if (PropertyNode * prop = dynamic_cast< PropertyNode * >(p.get())) {
return prop->getValue();
}
if (LocalizedPropertyValueNode * locval =
dynamic_cast< LocalizedPropertyValueNode * >(p.get()))
{
return locval->getValue();
}
if (LocalizedPropertyNode * locprop =
dynamic_cast< LocalizedPropertyNode * >(p.get()))
{
rtl::OUString locale(getRoot()->getLocale());
if (!Components::allLocales(locale)) {
rtl::Reference< LocalizedPropertyValueNode > value(
locprop->getValue(locale));
return value.is() ? value->getValue() : css::uno::Any();
}
}
return css::uno::makeAny(
css::uno::Reference< css::uno::XInterface >(
static_cast< cppu::OWeakObject * >(this)));
}
ChildAccess::~ChildAccess() {
osl::MutexGuard g(lock);
if (parent_.is()) {
parent_->releaseChild(getNode()->getName());
}
}
css::uno::Reference< css::uno::XInterface > ChildAccess::getParent()
throw (css::uno::RuntimeException)
{
OSL_ASSERT(thisIs(IS_ANY));
osl::MutexGuard g(lock);
return static_cast< cppu::OWeakObject * >(parent_.get());
}
void ChildAccess::setParent(css::uno::Reference< css::uno::XInterface > const &)
throw (css::lang::NoSupportException, css::uno::RuntimeException)
{
OSL_ASSERT(thisIs(IS_ANY));
throw css::lang::NoSupportException(
rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("setParent")),
static_cast< cppu::OWeakObject * >(this));
}
sal_Int64 ChildAccess::getSomething(
css::uno::Sequence< sal_Int8 > const & aIdentifier)
throw (css::uno::RuntimeException)
{
return aIdentifier == getTunnelId()
? reinterpret_cast< sal_Int64 >(this) : 0;
}
}