idlehomescreen/xmluicontroller/src/xmluicontroller.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 12 Mar 2010 15:41:49 +0200
branchRCL_3
changeset 11 ff572dfe6d86
parent 9 f966699dea19
child 26 1b758917cafc
permissions -rw-r--r--
Revision: 201007 Kit: 201008

/*
* Copyright (c) 2005-2008 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "Eclipse Public License v1.0"
* which accompanies this distribution, and is available
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
*
* Initial Contributors:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:  XML UI Controller main class
*
*/

// System includes
#include <ecom/implementationproxy.h>
#include <AknQueryDialog.h>
#include <eikstart.h>
#include <apgtask.h>
#include <centralrepository.h>
#include <e32property.h>

// User includes
#include <activeidle2domainpskeys.h>
#include <activeidle2domaincrkeys.h>
#include <hscontentpublisher.h>
#include <hspublisherinfo.h>
#include <aisystemuids.hrh>
#include <ai3xmlui.rsg>

#include "xmluicontroller.h"

#include "xnuiengineappif.h"
#include "xnnodeappif.h"
#include "xnproperty.h"
#include "xntype.h"
#include "aistrcnv.h"

#include "application.h"
#include "appui.h"
#include "xmlnodeidgenerator.h"
#include "aixuikoneventhandler.h"
#include "aixmluiutils.h"
#include "aiutility.h"
#include "aipluginsettings.h"
#include "aixmluiconstants.h"
#include "aifweventhandler.h"
#include "debug.h"
#include "aiconsts.h"
#include "contentrenderer.h"
#include "xmluicontrollerpanic.h"
#include "pmodtiterator.h"
#include "contentpublisher.h"
#include "xnplugindefs.h"
#include "ainativeuiplugins.h"

// Constants
const TInt KOneSecondInMicroS = 1000*1000;
_LIT8( KData, "data" );

using namespace AiXmlUiController;

// ======== LOCAL FUNCTIONS ========
// ----------------------------------------------------------------------------
// ResolveEventParameters()
// ----------------------------------------------------------------------------
//
static void ResolveEventParameters( const TDesC8& aEventString, 
    TPtrC8& aParam1, TPtrC8& aParam2 )
    {
    TInt  paramPos( aEventString.Locate( KEventParameterSeparator ) );
    
    TLex8 lex( aEventString.Mid( ++paramPos ) );
    
    aParam1.Set( lex.NextToken() );   // first ui element id
    
    TInt commaPos( aParam1.LocateReverse( KComma ) );
    
    if( commaPos != KErrNotFound )
        {
        // Strip comma
        aParam1.Set( aParam1.Left( commaPos ) );
        
        aParam2.Set( lex.NextToken() );   // second ui element id
        
        TInt rightParenthesis( aParam2.LocateReverse( KRightParenthesis ) );
        
        if( rightParenthesis != KErrNotFound )
            {
            aParam2.Set( aParam2.Left( rightParenthesis ) );
            }        
        }
    else
        {
        // Strip right parenthesis
        aParam1.Set( aParam1.Left( aParam1.Length() -1 ) );
        }
    }

// ======== MEMBER FUNCTIONS ========
// ----------------------------------------------------------------------------
// CXmlUiController::CXmlUiController()
// ----------------------------------------------------------------------------
//
CXmlUiController::CXmlUiController()
    {
    }

// ----------------------------------------------------------------------------
// CXmlUiController::ConstructL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::ConstructL()
    {       
    iNodeIdGenerator = CXmlNodeIdGenerator::NewL();  
    
    iAISettingsRepository = 
        CRepository::NewL( TUid::Uid( KCRUidActiveIdleLV ) ); 
        
    // Indicate to wserv plugin if it launches logs on send key or not              
    RProperty::Set( KPSUidAiInformation, KActiveIdleActOnSendKey, ETrue );                     
    }

// ----------------------------------------------------------------------------
// CXmlUiController::NewL()
// ----------------------------------------------------------------------------
//
CXmlUiController* CXmlUiController::NewL()
    {
    CXmlUiController* self = new ( ELeave ) CXmlUiController;

    CleanupStack::PushL( self );
    self->ConstructL();
    
    // Set XML UI Controller object to TLS for access in static
    // CXmlUiController::NewApplication
    User::LeaveIfError( Dll::SetTls( self ) );
    
    CleanupStack::Pop( self ); 

    return self;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::~CXmlUiController()
// ----------------------------------------------------------------------------
//
CXmlUiController::~CXmlUiController()
    {
    delete iNodeIdGenerator;
    delete iExitTimer;
    delete iAISettingsRepository;
    delete iCPSpublisher;
    
    // Reset TLS pointer to this set in NewL
    Dll::SetTls( NULL );
    }

// ----------------------------------------------------------------------------
// CXmlUiController::Exit()
// ----------------------------------------------------------------------------
//
void CXmlUiController::Exit()
    {
    if( iExitTimer )
        {
        iExitTimer->Cancel();
        
        iExitTimer->Start( 0, KOneSecondInMicroS,           
           TCallBack( ExitTimerCallBack, this ) );
        }
    }

// ----------------------------------------------------------------------------
// CXmlUiController::ExitTimerCallBack()
// ----------------------------------------------------------------------------
//
TInt CXmlUiController::ExitTimerCallBack(TAny *aSelf)
    {
     CXmlUiController* self =
         static_cast< CXmlUiController* >( aSelf );
    
    if( self )
        {
        self->iExitTimer->Cancel();
        
        self->iAppUi->Exit();        
        }
    
    return KErrNone;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SetObserver()
// ----------------------------------------------------------------------------
//
void CXmlUiController::SetObserver( MAiContentObserver& aObserver )
    {
    iObserver = &aObserver;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SetAppUi()
// ----------------------------------------------------------------------------
//
void CXmlUiController::SetAppUi( CAppUi& aAppUi )
    {
    iAppUi = &aAppUi;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::AppUi()
// ----------------------------------------------------------------------------
//
CAppUi* CXmlUiController::AppUi() const
    {
    return iAppUi;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::UiEngineL()
// ----------------------------------------------------------------------------
//
TXnUiEngineAppIf* CXmlUiController::UiEngineL() const
    {
    return iAppUi ? iAppUi->UiEngineL() : NULL;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SettingsRepository()
// ----------------------------------------------------------------------------
//
CRepository& CXmlUiController::SettingsRepository() const
    {
    return *iAISettingsRepository;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::CreateXuikonEventHandlerL()
// ----------------------------------------------------------------------------
//
CAIXuikonEventHandler* CXmlUiController::CreateXuikonEventHandlerL(
    CContentRenderer& aRenderer )
    {
    CAIXuikonEventHandler* eventHandler = 
        CAIXuikonEventHandler::NewL( *this, aRenderer );
    
    eventHandler->SetFwEventHandler( iFwEventHandler );
    eventHandler->SetUiEventHandler( this );
    
    return eventHandler;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::NotifyAppEnvReadyL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::NotifyAppEnvReadyL()
    {    
    iCPSpublisher = CContentPublisher::NewL( *this );
    
    iExitTimer = CPeriodic::NewL( CActive::EPriorityStandard );    

    if( iFwEventHandler )
        {
        iFwEventHandler->AppEnvReadyL();        
        }          
    }

// ----------------------------------------------------------------------------
// CXmlUiController::LoadUIDefinition()
// ----------------------------------------------------------------------------
//
void CXmlUiController::LoadUIDefinitionL()
    {          
    }

// ----------------------------------------------------------------------------
// CXmlUiController::PublisherInfoL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::PublisherInfoL( CXnNodeAppIf& aSource,
    THsPublisherInfo& aInfo )
    {
    // Fetch uid from XML and convert it to TUid
    const TDesC8* uid( PropertyValue( aSource, 
        AiUiDef::xml::property::KValue ) ); 

    if ( !uid )
        {
        return;
        }
       
    _LIT8( KPrefix, "0x" );
    
    const TInt pos( uid->FindF( KPrefix ) );

    TInt32 uidValue( 0 );
    
    if( pos != KErrNotFound )
         {
         TLex8 lex( uid->Mid( pos + KPrefix().Length() ) );
          
         // Hex parsing needs unsigned int
         TUint32 value( 0 );
         TInt err( lex.Val( value, EHex ) );
         
         if ( err == KErrNone )
             {
             uidValue = value;
             }
         }
     
    // Fetch plug-in name from XML
    HBufC* name( PropertyValueL( aSource, AiUiDef::xml::property::KName ) ); 
                
    TUid pluginuid( TUid::Uid( uidValue ) );
           
    aInfo = THsPublisherInfo( pluginuid, *name, aSource.Namespace() );
    
    delete name;
    name = NULL;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::GetSettingsL()
//
// ----------------------------------------------------------------------------
//
void CXmlUiController::GetSettingsL( const THsPublisherInfo& aPublisherInfo,     
    RAiSettingsItemArray& aSettings )
    {
    if ( aPublisherInfo.Namespace() == KNativeUiNamespace )
        {
        // This controller doesn't provide settings for native ui namespace
        return;
        }
    
    const TDesC& classId( 
        iNodeIdGenerator->SettingsNodeIdL( aPublisherInfo ) ); 
               
    TBuf< KHsPublisherNamespaceMaxLength > ns;
    
    ns.Copy( aPublisherInfo.Namespace() );
    
    // Find settings nodes
    RPointerArray< CXnNodeAppIf > nodes( 
        UiEngineL()->FindNodeByClassL( classId, ns ) );
    
    CleanupClosePushL( nodes );
    
    // Reserve space for settings
    aSettings.ReserveL( aSettings.Count() + nodes.Count() );
    
    // Collect settings
    TInt count( nodes.Count() );
    
    for( TInt i = 0; i < count; ++i )
        {
        CXnNodeAppIf* node( nodes[i] );
        
        const TDesC8* name( 
            PropertyValue( *node, AiUiDef::xml::property::KName ) );
        
        HBufC* value( 
            PropertyValueL( *node, AiUiDef::xml::property::KValue ) );        
        
        CleanupStack::PushL( value );
        
        if( name && value )
            {
            MAiPluginSettings* settings( AiUtility::CreatePluginSettingsL() );
            CleanupDeletePushL( settings );
            
            MAiPluginSettingsItem& item( settings->AiPluginSettingsItem() ); 
            item.SetPublisherId( aPublisherInfo.Uid() );
            
            TInt32 key( 0 );
            
            User::LeaveIfError( AiUtility::ParseInt( key, *name ) );
            
            item.SetKey( key );            
            item.SetValueL( *value, EFalse );
            
            // Append settings into array.
            // This can not fail because space has been reserved.
            aSettings.Append( settings );            
            CleanupStack::Pop( settings );
            }
            
        CleanupStack::PopAndDestroy( value );
        }
    
    CleanupStack::PopAndDestroy( &nodes );
    
    GetContentModelL( aPublisherInfo, aSettings );
    }

// ----------------------------------------------------------------------------
// CXmlUiController::GetContentModelL()
// ----------------------------------------------------------------------------
//
// ContentModelL()
void CXmlUiController::GetContentModelL( const THsPublisherInfo& aPublisherInfo,         
    RAiSettingsItemArray& aSettings )
    {
    if ( aPublisherInfo.Namespace() == KNativeUiNamespace )
        {
        // This controller doesn't provide content model for native ui namespace
        return;
        }
    
    // Find the node for the publisher
	// TODO Does not work if widget is in view  
    RPointerArray<CXnNodeAppIf> list( 
        UiEngineL()->FindContentSourceNodesL( aPublisherInfo.Namespace() ) );
    
    CleanupClosePushL( list );
    
    CXnNodeAppIf* publisherNode( NULL );
    
    if ( list.Count() > 0 )
        {
        publisherNode = list[0];
        }
         
    CleanupStack::PopAndDestroy( &list );
    
    if( !publisherNode )
        {
        return;
        }
    
    // Find ui plugin node for the data plugin
    CXnNodeAppIf* parentNode( publisherNode->ParentL() );
    
    // parent must be correct type
    if( parentNode->InternalDomNodeType() == _L8("widget") )
        {
        // Get plugin configurations
        GetConfigurationsL( *parentNode, aSettings, AiUiDef::xml::element::K16Plugin());
        CPmODTIterator* iter = CPmODTIterator::NewL( *parentNode );
        CleanupStack::PushL( iter );
        CXnNodeAppIf* node = iter->First();
        while( node )
            {
            const TDesC8& nodeType = node->Type()->Type();
            if( nodeType == XnPropertyNames::action::KActions)
                {
                node = iter->SkipBranchL();
                }
            // Get the content model and configuration for the supported elements
            else if( nodeType == AiUiDef::xml::element::KImage ||
                     nodeType == AiUiDef::xml::element::KText ||
                     nodeType == AiUiDef::xml::element::KNewsTicker ||
                     nodeType == AiUiDef::xml::element::KAnimation ||
					 nodeType == AiUiDef::xml::element::KTextEditor || 
                     nodeType == KData )
                {
                // Is created in GetContenItem and used in GetConfigurationsL
                HBufC* confOwner( NULL );
                // get content item for the element, confOwner is filled
                GetContentItemL( *node, aSettings, confOwner );
                if( confOwner )
                    {
                    CleanupStack::PushL( confOwner );
                    // get configurations for the element
                    GetConfigurationsL( *node, aSettings, *confOwner );
                    CleanupStack::PopAndDestroy( confOwner );
                    }
                node = iter->SkipBranchL();
                }
            else
                {
                node = iter->NextL();
                }
            }
        CleanupStack::PopAndDestroy( iter );
        }
    }

// ----------------------------------------------------------------------------
// CXmlUiController::GetContentItemL()
// ----------------------------------------------------------------------------
//
// ContentItemL()
void CXmlUiController::GetContentItemL( CXnNodeAppIf& aNode, 
    RAiSettingsItemArray& aSettings, HBufC*& aItemName )
    {
    // Find property node
    RPointerArray<CXnNodeAppIf> childNodes( aNode.ChildrenL() );
    CleanupClosePushL( childNodes );
    
    TInt count( childNodes.Count() );
    
    for ( TInt i = 0; i < count; i++ )
        {
        CXnNodeAppIf* node( childNodes[i] );
        
        if( node->Type()->Type() == XnPropertyNames::action::KProperty )
            {
            HBufC* name = PropertyValueL( *node, AiUiDef::xml::property::KClass );
            CleanupStack::PushL( name );
        
            if( name )
                {
                // Content model found for the element, create content item
                MAiPluginSettings* settings = AiUtility::CreatePluginSettingsL();
                CleanupDeletePushL( settings );
                
                MAiPluginContentItem& item = settings->AiPluginContentItem();
                
                // Type of the element is needed in content model 
                HBufC* type( NULL ); 
                type = AiUtility::CopyToBufferL( type, aNode.Type()->Type());
                CleanupStack::PushL( type );
                item.SetTypeL( *type );
                CleanupStack::PopAndDestroy( type );
                
                item.SetNameL( *name );
                
                aItemName = name;
                aSettings.AppendL( settings );
                CleanupStack::Pop( settings );
                }
            
            // Ownership is given to aItemName
            CleanupStack::Pop( name );
            
            // First property element, which has class attribute is selected
            break;
            }
        }
    
    CleanupStack::PopAndDestroy( &childNodes );
    }

// ----------------------------------------------------------------------------
// CXmlUiController::GetConfigurationsL()
// ----------------------------------------------------------------------------
//
// ConfigurationItemsL()
void CXmlUiController::GetConfigurationsL( CXnNodeAppIf& aNode, 
    RAiSettingsItemArray& aSettings, const TDesC& aConfOwner  )
    {    
    // Find configuration nodes
    RPointerArray< CXnNodeAppIf > nodes( aNode.ChildrenL() );
    CleanupClosePushL( nodes );

    // Collect settings
    TInt count( nodes.Count() );
    
    for( TInt j = 0; j < count; j++ )
        {
        CXnNodeAppIf* node( nodes[j] );
        
        if( node->Type()->Type() == AiUiDef::xml::element::KConfiguration )
            {
            HBufC* name( PropertyValueL( *node, AiUiDef::xml::property::KName ) );
            CleanupStack::PushL( name );

            HBufC* value( PropertyValueL( *node, AiUiDef::xml::property::KValue ) );
            CleanupStack::PushL( value );

            MAiPluginSettings* settings = AiUtility::CreatePluginSettingsL();
            CleanupDeletePushL( settings );
               
            MAiPluginConfigurationItem& item( settings->AiPluginConfigurationItem() );
     
            item.SetOwnerL( aConfOwner );
            item.SetNameL( *name );
            item.SetValueL( *value );
            
            // Append settings into array.
            aSettings.AppendL( settings );
        
            CleanupStack::Pop( settings );
            CleanupStack::PopAndDestroy( value );    
            CleanupStack::PopAndDestroy( name );        
            }
        }
    
    CleanupStack::PopAndDestroy( &nodes );
    }

// ----------------------------------------------------------------------------
// CXmlUiController::ActivateUI()
// ----------------------------------------------------------------------------
//
void CXmlUiController::ActivateUI()
    {
    if ( iRunningAsMain && iAppUi )
        {
        iAppUi->ActivateUi();
        }
    }

// ----------------------------------------------------------------------------
// CXmlUiController::GetContentObserver()
// ----------------------------------------------------------------------------
//
MAiContentObserver& CXmlUiController::GetContentObserver()
    {
    return *iObserver;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SetEventHandler()
// ----------------------------------------------------------------------------
//
void CXmlUiController::SetEventHandler( MAiFwEventHandler& aFwEventHandler )
    {
    iFwEventHandler = &aFwEventHandler;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SetStateHandler()
// ----------------------------------------------------------------------------
//
void CXmlUiController::SetStateHandler( MAiFwStateHandler& aFwStateHandler )
    {
    iFwStateHandler = &aFwStateHandler;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::FwEventHandler()
// ----------------------------------------------------------------------------
//
MAiFwEventHandler* CXmlUiController::FwEventHandler() const
    {
    return iFwEventHandler;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::FwStateHandler()
// ----------------------------------------------------------------------------
//
MAiFwStateHandler* CXmlUiController::FwStateHandler() const
    {
    return iFwStateHandler;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::MainInterface()
// ----------------------------------------------------------------------------
//
MAiMainUiController* CXmlUiController::MainInterface()
    {
    return this;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::SecondaryInterface()
// ----------------------------------------------------------------------------
//
MAiSecondaryUiController* CXmlUiController::SecondaryInterface()
    {
    return NULL;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::RunApplicationL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::RunApplicationL()
    {
    __PRINTS( "*** CXmlUiController::RunApplicationL" );
    
    iRunningAsMain = ETrue;
                                                    
    User::LeaveIfError( 
        EikStart::RunApplication( &CXmlUiController::NewApplication ) );       
    }

// ----------------------------------------------------------------------------
// CXmlUiController::CoeEnv()
// ----------------------------------------------------------------------------
//
CCoeEnv& CXmlUiController::CoeEnv()
    {
    __ASSERT_ALWAYS( iAppUi, Panic(ECriticalPointerNull) );
    return *iAppUi->CoeEnv();
    }

// ----------------------------------------------------------------------------
// CXmlUiController::IsMenuOpen()
// ----------------------------------------------------------------------------
//
TBool CXmlUiController::IsMenuOpen()
    {
    return iAppUi->IsDisplayingMenuOrDialog();
    }

// ----------------------------------------------------------------------------
// CXmlUiController::HandleUiEvent()
// ----------------------------------------------------------------------------
//
TBool CXmlUiController::HandleUiEvent( TAny* aEvent, const TDesC8& aParam )
    {
    CXnNodeAppIf* event( static_cast<CXnNodeAppIf*>( aEvent ) );
    
    TBool retval( EFalse );
    
    if( event )
        {
        TPtrC8 param1;
        TPtrC8 param2;
        const TDesC8& eventString(
                aParam.Mid( AiUiDef::xml::event::KUiEventPrefix().Length() ) );
        
        // Extract event name
        TInt  paramPos( eventString.Locate( KEventParameterSeparator ) );
        
        const TDesC8& eventName( eventString.Left( Max( 0, paramPos ) ) );
    
        if( eventName == AiUiDef::xml::event::KFocusGained )
            {
            // Resolve event parameters
            ResolveEventParameters( eventString, param1, param2 );
            
            TRAP_IGNORE( HandleFocusGainedL( param1, param2, *event ) );    
            
            retval = ETrue;
            }
        else if( eventName == AiUiDef::xml::event::KSetElementSizeToCPS )
            {
            // Resolve event parameters
            ResolveEventParameters( eventString, param1, param2 );
            
            TRAP_IGNORE( HandleSetElementSizeL( param1, *event ));
            
            retval = ETrue;
            }
        }
    
    return retval;
    }

// ----------------------------------------------------------------------------
// CXmlUiController::HandleSetElementSizeL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::HandleSetElementSizeL( const TDesC8& aElementName, 
    CXnNodeAppIf& aOrigin )
    {
    __UHEAP_MARK;
    // Resolve ui elements
    CXnNodeAppIf* element( 
            UiEngineL()->FindNodeByIdL( aElementName, aOrigin.Namespace() ) );
    
    LeaveIfNull( element, KErrNotFound );

    TRect rect( element->Rect() );
    
    HBufC* elementName( NULL );
    elementName = AiUtility::CopyToBufferL( elementName, aElementName );
    CleanupStack::PushL( elementName );
    
    iCPSpublisher->PublishSizeL( *elementName, rect.Width(), rect.Height() );
    
    CleanupStack::PopAndDestroy( elementName );
    
    __UHEAP_MARKEND;    
    }

// ----------------------------------------------------------------------------
// CXmlUiController::HandleFocusGainedL()
// ----------------------------------------------------------------------------
//
void CXmlUiController::HandleFocusGainedL( const TDesC8& aUiElement1,
    const TDesC8& aUiElement2, CXnNodeAppIf& aOrigin )
    {
    // Resolve ui elements
    CXnNodeAppIf* element1( 
        UiEngineL()->FindNodeByIdL( aUiElement1, aOrigin.Namespace() ) );
    
    LeaveIfNull( element1, KErrNotFound );
    
    CXnNodeAppIf* element2( 
        UiEngineL()->FindNodeByIdL( aUiElement2, aOrigin.Namespace() ) );
    
    LeaveIfNull( element2, KErrNotFound );
    
    // swap display properties between ui elements
    CXnProperty* display1( 
        element1->GetPropertyL( XnPropertyNames::style::common::KDisplay ) );
    
    LeaveIfNull( display1, KErrNotFound );
    
    CXnProperty* display2(
        element2->GetPropertyL( XnPropertyNames::style::common::KDisplay ) );
    
    LeaveIfNull( display2, KErrNotFound );
    
    display1 = display1->CloneL();
    CleanupStack::PushL( display1 );
    
    display2 = display2->CloneL();
    CleanupStack::PushL( display2 );
    
    element1->SetPropertyL( display2 );
    CleanupStack::Pop( display2 );
    
    element2->SetPropertyL( display1 );
    CleanupStack::Pop( display1 );
    }

// ----------------------------------------------------------------------------
// CXmlUiController::NewApplication()
// ----------------------------------------------------------------------------
//
CApaApplication* CXmlUiController::NewApplication()
    {
    __TICK( "CXmlUiController::NewApplication" );
    __TIME_MARK( time );
    
    CXmlUiController* self = static_cast< CXmlUiController* >( Dll::Tls() );
    
    __ASSERT_ALWAYS( self, Panic( ECriticalPointerNull ) );
    
    CApplication* app = CApplication::New( *self );    
    
    __TIME_ENDMARK( "CXmlUiController::NewApplication, done", time );
    
    return app;
    }

const TImplementationProxy KImplementationTable[] =
    {
    IMPLEMENTATION_PROXY_ENTRY(KImplementationUidXmlUiController, CXmlUiController::NewL)
    };

EXPORT_C const TImplementationProxy* ImplementationGroupProxy(TInt& aTableCount)
    {
    aTableCount = sizeof(KImplementationTable) / sizeof(TImplementationProxy);
    return KImplementationTable;
    }