/**************************************************************
 * 
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 * 
 *************************************************************/



// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_sc.hxx"


#include "AccessibleContextBase.hxx"
#include "unoguard.hxx"
#include <com/sun/star/accessibility/AccessibleRole.hpp>
#include <com/sun/star/accessibility/AccessibleEventId.hpp>
#ifndef _COM_SUN_STAR_ACCESSIBILITY_XACCESSIBLESTATETYPE_HPP_
#include <com/sun/star/accessibility/AccessibleStateType.hpp>
#endif
#include <com/sun/star/beans/PropertyChangeEvent.hpp>
#include <rtl/uuid.h>
#include <tools/debug.hxx>
#include <tools/gen.hxx>
#ifndef _UTL_ACCESSIBLESTATESETHELPER_HXX
#include <unotools/accessiblestatesethelper.hxx>
#endif
#include <toolkit/helper/convert.hxx>
#include <svl/smplhint.hxx>
#include <comphelper/sequence.hxx>
#include <unotools/accessiblerelationsethelper.hxx>
#include <vcl/unohelp.hxx>
#include <tools/color.hxx>
#include <comphelper/accessibleeventnotifier.hxx>

using namespace	::rtl;
using namespace	::com::sun::star;
using namespace	::com::sun::star::accessibility;

//=====  internal  ============================================================

DBG_NAME(ScAccessibleContextBase)

ScAccessibleContextBase::ScAccessibleContextBase(
												 const uno::Reference<XAccessible>& rxParent,
												 const sal_Int16 aRole)
												 :
	ScAccessibleContextBaseWeakImpl(m_aMutex),
	mxParent(rxParent),
	mnClientId(0),
	maRole(aRole)
{
	DBG_CTOR(ScAccessibleContextBase, NULL);
}


ScAccessibleContextBase::~ScAccessibleContextBase(void)
{
	DBG_DTOR(ScAccessibleContextBase, NULL);

	if (!IsDefunc() && !rBHelper.bInDispose)
	{
		// increment refcount to prevent double call off dtor
		osl_incrementInterlockedCount( &m_refCount );
		// call dispose to inform object wich have a weak reference to this object
		dispose();
	}
}

void ScAccessibleContextBase::Init()
{
	// hold reference to make sure that the destructor is not called
	uno::Reference< XAccessibleContext > xOwnContext(this);

	if (mxParent.is())
	{
		uno::Reference< XAccessibleEventBroadcaster > xBroadcaster (mxParent->getAccessibleContext(), uno::UNO_QUERY);
		if (xBroadcaster.is())
			xBroadcaster->addEventListener(this);
	}
	msName = createAccessibleName();
	msDescription = createAccessibleDescription();
}

void SAL_CALL ScAccessibleContextBase::disposing()
{
    ScUnoGuard aGuard;
//	CommitDefunc(); not necessary and should not be send, because it cost a lot of time

	// hold reference to make sure that the destructor is not called
	uno::Reference< XAccessibleContext > xOwnContext(this);

	if ( mnClientId )
	{
	    sal_Int32 nTemClientId(mnClientId);
		mnClientId =  0;
        comphelper::AccessibleEventNotifier::revokeClientNotifyDisposing( nTemClientId, *this );
	}

    if (mxParent.is())
	{
		uno::Reference< XAccessibleEventBroadcaster > xBroadcaster (mxParent->getAccessibleContext(), uno::UNO_QUERY);
		if (xBroadcaster.is())
			xBroadcaster->removeEventListener(this);
		mxParent = NULL;
	}

	ScAccessibleContextBaseWeakImpl::disposing();
}

//=====  XInterface  =====================================================

uno::Any SAL_CALL ScAccessibleContextBase::queryInterface( uno::Type const & rType )
	throw (uno::RuntimeException)
{
	uno::Any aAny (ScAccessibleContextBaseWeakImpl::queryInterface(rType));
	return aAny.hasValue() ? aAny : ScAccessibleContextBaseImplEvent::queryInterface(rType);
}

void SAL_CALL ScAccessibleContextBase::acquire()
	throw ()
{
	ScAccessibleContextBaseWeakImpl::acquire();
}

void SAL_CALL ScAccessibleContextBase::release()
	throw ()
{
	ScAccessibleContextBaseWeakImpl::release();
}

//=====  SfxListener  =====================================================

void ScAccessibleContextBase::Notify( SfxBroadcaster&, const SfxHint& rHint )
{
	if (rHint.ISA( SfxSimpleHint ) )
	{
		const SfxSimpleHint& rRef = (const SfxSimpleHint&)rHint;
		if (rRef.GetId() == SFX_HINT_DYING)
		{
			// it seems the Broadcaster is dying, since the view is dying
			dispose();
		}
	}
}

//=====  XAccessible  =========================================================

uno::Reference< XAccessibleContext> SAL_CALL
    ScAccessibleContextBase::getAccessibleContext(void)
    throw (uno::RuntimeException)
{
	return this;
}

//=====  XAccessibleComponent  ================================================

sal_Bool SAL_CALL ScAccessibleContextBase::containsPoint(const awt::Point& rPoint )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	return Rectangle (Point(), GetBoundingBox().GetSize()).IsInside(VCLPoint(rPoint));
}

uno::Reference< XAccessible > SAL_CALL ScAccessibleContextBase::getAccessibleAtPoint(
        const awt::Point& /* rPoint */ )
		throw (uno::RuntimeException)
{
	DBG_ERROR("not implemented");
	return uno::Reference<XAccessible>();
}

awt::Rectangle SAL_CALL ScAccessibleContextBase::getBounds(  )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	return AWTRectangle(GetBoundingBox());
}

awt::Point SAL_CALL ScAccessibleContextBase::getLocation(  )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	return AWTPoint(GetBoundingBox().TopLeft());
}

awt::Point SAL_CALL ScAccessibleContextBase::getLocationOnScreen(  )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	return AWTPoint(GetBoundingBoxOnScreen().TopLeft());
}

awt::Size SAL_CALL ScAccessibleContextBase::getSize(  )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	return AWTSize(GetBoundingBox().GetSize());
}

sal_Bool SAL_CALL ScAccessibleContextBase::isShowing(  )
		throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	sal_Bool bShowing(sal_False);
    if (mxParent.is())
    {
	    uno::Reference<XAccessibleComponent> xParentComponent (mxParent->getAccessibleContext(), uno::UNO_QUERY);
	    if (xParentComponent.is())
	    {
		    Rectangle aParentBounds(VCLRectangle(xParentComponent->getBounds()));
		    Rectangle aBounds(VCLRectangle(getBounds()));
		    bShowing = aBounds.IsOver(aParentBounds);
	    }
    }
	return bShowing;
}

sal_Bool SAL_CALL ScAccessibleContextBase::isVisible(  )
		throw (uno::RuntimeException)
{
	return sal_True;
}

void SAL_CALL ScAccessibleContextBase::grabFocus(  )
		throw (uno::RuntimeException)
{
	DBG_ERROR("not implemented");
}

sal_Int32 SAL_CALL ScAccessibleContextBase::getForeground(  )
        throw (uno::RuntimeException)
{
    return COL_BLACK;
}

sal_Int32 SAL_CALL ScAccessibleContextBase::getBackground(  )
        throw (uno::RuntimeException)
{
    return COL_WHITE;
}

//=====  XAccessibleContext  ==================================================

sal_Int32 SAL_CALL
   	ScAccessibleContextBase::getAccessibleChildCount(void)
    throw (uno::RuntimeException)
{
	DBG_ERROR("should be implemented in the abrevated class");
	return 0;
}

uno::Reference<XAccessible> SAL_CALL
    ScAccessibleContextBase::getAccessibleChild(sal_Int32 /* nIndex */)
        throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
{
	DBG_ERROR("should be implemented in the abrevated class");
    return uno::Reference<XAccessible>();
}

uno::Reference<XAccessible> SAL_CALL
   	ScAccessibleContextBase::getAccessibleParent(void)
    throw (uno::RuntimeException)
{
	return mxParent;
}

sal_Int32 SAL_CALL
   	ScAccessibleContextBase::getAccessibleIndexInParent(void)
    throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	//	Use a simple but slow solution for now.  Optimize later.
   //	Return -1 to indicate that this object's parent does not know about the
   //	object.
	sal_Int32 nIndex(-1);

    //	Iterate over all the parent's children and search for this object.
    if (mxParent.is())
    {
    	uno::Reference<XAccessibleContext> xParentContext (
        	mxParent->getAccessibleContext());
        if (xParentContext.is())
        {
        	sal_Int32 nChildCount = xParentContext->getAccessibleChildCount();
            for (sal_Int32 i=0; i<nChildCount; ++i)
            {
            	uno::Reference<XAccessible> xChild (xParentContext->getAccessibleChild (i));
                if (xChild.is())
                {
	            	if (xChild.get() == this)
                    	nIndex = i;
                }
            }
        }
   }

   return nIndex;
}

sal_Int16 SAL_CALL
	ScAccessibleContextBase::getAccessibleRole(void)
    throw (uno::RuntimeException)
{
	return maRole;
}

::rtl::OUString SAL_CALL
   	ScAccessibleContextBase::getAccessibleDescription(void)
    throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	if (!msDescription.getLength())
	{
		OUString sDescription(createAccessibleDescription());
//		DBG_ASSERT(sDescription.getLength(), "We should give always a descripition.");

        if (msDescription != sDescription)
        {
		    AccessibleEventObject aEvent;
		    aEvent.EventId = AccessibleEventId::DESCRIPTION_CHANGED;
		    aEvent.Source = uno::Reference< XAccessibleContext >(this);
		    aEvent.OldValue <<= msDescription;
		    aEvent.NewValue <<= sDescription;

		    msDescription = sDescription;

		    CommitChange(aEvent);
        }
	}
	return msDescription;
}

OUString SAL_CALL
   	ScAccessibleContextBase::getAccessibleName(void)
    throw (uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	if (!msName.getLength())
	{
		OUString sName(createAccessibleName());
		DBG_ASSERT(sName.getLength(), "We should give always a name.");

        if (msName != sName)
        {
		    AccessibleEventObject aEvent;
		    aEvent.EventId = AccessibleEventId::NAME_CHANGED;
		    aEvent.Source = uno::Reference< XAccessibleContext >(this);
		    aEvent.OldValue <<= msName;
		    aEvent.NewValue <<= sName;

		    msName = sName;

		    CommitChange(aEvent);
        }
	}
	return msName;
}

uno::Reference<XAccessibleRelationSet> SAL_CALL
   	ScAccessibleContextBase::getAccessibleRelationSet(void)
    throw (uno::RuntimeException)
{
	return new utl::AccessibleRelationSetHelper();
}

uno::Reference<XAccessibleStateSet> SAL_CALL
    	ScAccessibleContextBase::getAccessibleStateSet(void)
    throw (uno::RuntimeException)
{
	return uno::Reference<XAccessibleStateSet>();
}

lang::Locale SAL_CALL
   	ScAccessibleContextBase::getLocale(void)
	throw (IllegalAccessibleComponentStateException,
		uno::RuntimeException)
{
	ScUnoGuard aGuard;
    IsObjectValid();
	if (mxParent.is())
    {
    	uno::Reference<XAccessibleContext> xParentContext (
        	mxParent->getAccessibleContext());
        if (xParentContext.is())
	    	return xParentContext->getLocale ();
    }

    //	No locale and no parent.  Therefore throw exception to indicate this
    //	cluelessness.
    throw IllegalAccessibleComponentStateException ();
}

	//=====  XAccessibleEventBroadcaster  =====================================

void SAL_CALL
   	ScAccessibleContextBase::addEventListener(
       	const uno::Reference<XAccessibleEventListener>& xListener)
    throw (uno::RuntimeException)
{
	if (xListener.is())
    {
		ScUnoGuard aGuard;
        IsObjectValid();
		if (!IsDefunc())
		{
			if (!mnClientId)
                mnClientId = comphelper::AccessibleEventNotifier::registerClient( );
			comphelper::AccessibleEventNotifier::addEventListener( mnClientId, xListener );
		}
    }
}

void SAL_CALL
   	ScAccessibleContextBase::removeEventListener(
		const uno::Reference<XAccessibleEventListener>& xListener)
    throw (uno::RuntimeException)
{
	if (xListener.is())
	{
		ScUnoGuard aGuard;
		if (!IsDefunc() && mnClientId)
        {
		    sal_Int32 nListenerCount = comphelper::AccessibleEventNotifier::removeEventListener( mnClientId, xListener );
		    if ( !nListenerCount )
		    {
			    // no listeners anymore
			    // -> revoke ourself. This may lead to the notifier thread dying (if we were the last client),
			    // and at least to us not firing any events anymore, in case somebody calls
			    // NotifyAccessibleEvent, again
			    comphelper::AccessibleEventNotifier::revokeClient( mnClientId );
			    mnClientId = 0;
		    }
        }
	}
}

	//=====  XAccessibleEventListener  ========================================

void SAL_CALL ScAccessibleContextBase::disposing(
	const lang::EventObject& rSource )
		throw (uno::RuntimeException)
{
    ScUnoGuard aGuard;
	if (rSource.Source == mxParent)
		dispose();
}

void SAL_CALL ScAccessibleContextBase::notifyEvent(
        const AccessibleEventObject& /* aEvent */ )
		throw (uno::RuntimeException)
{
}

//=====  XServiceInfo  ========================================================

::rtl::OUString SAL_CALL
   	ScAccessibleContextBase::getImplementationName(void)
    throw (uno::RuntimeException)
{
	return OUString(RTL_CONSTASCII_USTRINGPARAM ("ScAccessibleContextBase"));
}

sal_Bool SAL_CALL
 	ScAccessibleContextBase::supportsService(const OUString& sServiceName)
    throw (uno::RuntimeException)
{
    //  Iterate over all supported service names and return true if on of them
    //  matches the given name.
    uno::Sequence< ::rtl::OUString> aSupportedServices (
        getSupportedServiceNames ());
	sal_Int32 nLength(aSupportedServices.getLength());
	const OUString* pServiceNames = aSupportedServices.getConstArray();
    for (int i=0; i<nLength; ++i, ++pServiceNames)
        if (sServiceName == *pServiceNames)
            return sal_True;
    return sal_False;
}

uno::Sequence< ::rtl::OUString> SAL_CALL
   	ScAccessibleContextBase::getSupportedServiceNames(void)
    throw (uno::RuntimeException)
{
	uno::Sequence<OUString> aServiceNames(2);
	OUString* pServiceNames = aServiceNames.getArray();
	if (pServiceNames)
	{
		pServiceNames[0] = OUString(RTL_CONSTASCII_USTRINGPARAM ("com.sun.star.accessibility.Accessible"));
		pServiceNames[1] = OUString(RTL_CONSTASCII_USTRINGPARAM ("com.sun.star.accessibility.AccessibleContext"));
	}

	return aServiceNames;
}

//=====  XTypeProvider  =======================================================

uno::Sequence< uno::Type > SAL_CALL ScAccessibleContextBase::getTypes()
		throw (uno::RuntimeException)
{
	return comphelper::concatSequences(ScAccessibleContextBaseWeakImpl::getTypes(), ScAccessibleContextBaseImplEvent::getTypes());
}

uno::Sequence<sal_Int8> SAL_CALL
	ScAccessibleContextBase::getImplementationId(void)
    throw (uno::RuntimeException)
{
    ScUnoGuard aGuard;
    IsObjectValid();
	static uno::Sequence<sal_Int8> aId;
	if (aId.getLength() == 0)
	{
		aId.realloc (16);
		rtl_createUuid (reinterpret_cast<sal_uInt8 *>(aId.getArray()), 0, sal_True);
	}
	return aId;
}

//=====  internal  ============================================================

::rtl::OUString SAL_CALL
    ScAccessibleContextBase::createAccessibleDescription(void)
    throw (uno::RuntimeException)
{
	DBG_ERROR("should be implemented in the abrevated class");
	return rtl::OUString();
}

::rtl::OUString SAL_CALL
    ScAccessibleContextBase::createAccessibleName(void)
    throw (uno::RuntimeException)
{
	DBG_ERROR("should be implemented in the abrevated class");
	return rtl::OUString();
}

void ScAccessibleContextBase::CommitChange(const AccessibleEventObject& rEvent) const
{
	if (mnClientId)
		comphelper::AccessibleEventNotifier::addEvent( mnClientId, rEvent );
}

void ScAccessibleContextBase::ChangeName()
{
	AccessibleEventObject aEvent;
	aEvent.EventId = AccessibleEventId::NAME_CHANGED;
	aEvent.Source = uno::Reference< XAccessibleContext >(const_cast<ScAccessibleContextBase*>(this));
    aEvent.OldValue <<= msName;

    msName = rtl::OUString(); // reset the name so it will be hold again
    getAccessibleName(); // create the new name

	aEvent.NewValue <<= msName;

	CommitChange(aEvent);
}

void ScAccessibleContextBase::CommitFocusGained() const
{
	AccessibleEventObject aEvent;
	aEvent.EventId = AccessibleEventId::STATE_CHANGED;
	aEvent.Source = uno::Reference< XAccessibleContext >(const_cast<ScAccessibleContextBase*>(this));
	aEvent.NewValue <<= AccessibleStateType::FOCUSED;

	CommitChange(aEvent);

    ::vcl::unohelper::NotifyAccessibleStateEventGlobally(aEvent);
}

void ScAccessibleContextBase::CommitFocusLost() const
{
	AccessibleEventObject aEvent;
	aEvent.EventId = AccessibleEventId::STATE_CHANGED;
	aEvent.Source = uno::Reference< XAccessibleContext >(const_cast<ScAccessibleContextBase*>(this));
	aEvent.OldValue <<= AccessibleStateType::FOCUSED;

	CommitChange(aEvent);

    vcl::unohelper::NotifyAccessibleStateEventGlobally(aEvent);
}

Rectangle ScAccessibleContextBase::GetBoundingBoxOnScreen(void) const
		throw (uno::RuntimeException)
{
	DBG_ERROR("not implemented");
	return Rectangle();
}

Rectangle ScAccessibleContextBase::GetBoundingBox(void) const
		throw (uno::RuntimeException)
{
	DBG_ERROR("not implemented");
	return Rectangle();
}

void ScAccessibleContextBase::IsObjectValid() const
        throw (lang::DisposedException)
{
    if (rBHelper.bDisposed || rBHelper.bInDispose)
        throw lang::DisposedException();
}

