javauis/lcdui_akn/javalcdui/src/CMIDEnv.cpp
author Pat Downey <patd@symbian.org>
Wed, 01 Sep 2010 12:33:18 +0100
branchRCL_3
changeset 26 2455ef1f5bbc
permissions -rw-r--r--
Revert incorrect RCL_3 drop: Revision: v2.2.11 Kit: 201035

/*
* Copyright (c) 2002 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:
*
*/

#include <coemain.h>
#include <eikenv.h>
#include <hal.h>
#include "CMIDEnv.h"
#include "CMIDToolkit.h"
#include "CMIDNotify.h"
#include "CMIDKeyTranslator.h"
#include "CMIDToLcduiObserver.h"

#ifdef RD_JAVA_NGA_ENABLED
#include <EGL/egl.h>
#include <GLES/gl.h>
#include <libc/string.h>
#endif // RD_JAVA_NGA_ENABLED

const TUint16 KCharComma = ',';

CMIDEnv::CMIDEnv(CMIDToolkit* aToolkit, const TSize& aSize)
        : iToolkit(aToolkit)
        , iObservers(EGranularity)
        , iCanvasAssumedSize(aSize)
        , iToLcduiObserver(NULL)
#ifdef RD_JAVA_NGA_ENABLED
        , iHardwareStatus(0)
#endif
{
    iCanvasData.iLock.CreateLocal();
}

void CMIDEnv::ConstructL()
{
    iKeyTranslator = new(ELeave) CMIDKeyTranslator(*this);
    iKeyTranslator->ConstructL();
    iToLcduiObserver = new(ELeave) CMIDToLcduiObserver();

    // On a non-full-screen (normal) mode Canvas the background image must be initially shown,
    // if the value of the Nokia-UI-Enhancement attribute is "CanvasHasBackground". The
    // attribute may be placed in the JAD or the manifest.
    iCanvasData.iHasBackground = MidletAttributeContainsVal(
                                     LcduiMidletAttributes::KAttribUIEnhancement,
                                     LcduiMidletAttributeValues::KUIEnhCanvasBackground);
#ifdef RD_JAVA_NGA_ENABLED
    InitHardwareStatusL();
#endif // RD_JAVA_NGA_ENABLED
}

CMIDEnv::~CMIDEnv()
{
    if (iToLcduiObserver)
    {
        delete iToLcduiObserver;
        iToLcduiObserver = NULL;
    }
    delete iKeyTranslator;
    iObservers.Close();
    iCanvasData.iLock.Close();
}

void CMIDEnv::SetUtils(MMIDUtils* aUtils)
{
    iKeyTranslator->SetUtils(aUtils);
}

TPtrC CMIDEnv::MidletName() const
{
    return iToolkit->MidletName();
}

TUid CMIDEnv::MidletUid() const
{
    return iToolkit->MidletUid();
}

TPtrC CMIDEnv::MidletHome() const
{
    return iToolkit->MidletHome();
}

#ifdef RD_SCALABLE_UI_V2
TUid CMIDEnv::MidletSuiteUid()
{
    if (!iMidletSuiteUid.iUid)
    {
        //Midlet suite uid is located inside brackets in the midlet home path.
        const TChar KUidStart = '[';
        const TChar KUidEnd = ']';

        const TPtrC midletHome = MidletHome();

        //Finding start and end positions
        TInt startPos = midletHome.Locate(KUidStart) + 1;
        TInt endPos = midletHome.LocateReverse(KUidEnd);

        if (startPos != KErrNotFound && endPos != KErrNotFound &&
                startPos < endPos && endPos <= midletHome.Length())
        {
            //Creating new pointer descriptor representing the midlet suite uid
            const TPtrC uidDesc = midletHome.Mid(startPos, endPos - startPos);

            if (uidDesc.Length())
            {
                //Creating unsigned integer out of the hex string.
                TUint uint;
                TLex lexer(uidDesc);
                TInt err = lexer.Val(uint, EHex);

                if (err == KErrNone)
                {
                    iMidletSuiteUid = TUid::Uid(uint);
                }
                else
                {
                    iMidletSuiteUid = TUid::Uid(KNullUidValue);
                }
            }
            else
            {
                iMidletSuiteUid = TUid::Uid(KNullUidValue);
            }
        }
        else
        {
            iMidletSuiteUid = TUid::Uid(KNullUidValue);
        }
    }
    return iMidletSuiteUid;
}
#endif // RD_SCALABLE_UI_V2


TBool CMIDEnv::PostJavaEvent
(
    MMIDComponent& aSource,
    TSourceType aSourceType,
    TEventType  aEventType,
    TInt        aEventData,
    TInt        aEventData1,
    TInt        aEventData2
)
{
#ifdef TRACE_EVENTS
    RDebug::Print(_L("CMIDEnv::PostJavaEvent(%x,%d,%d,%d,%d,%d)"),
                  aSource,
                  aSourceType,
                  aEventType,
                  aEventData,aEventData1, aEventData2);
#endif
    switch (aSourceType)
    {
    case EItem:
        return iToolkit->PostItemEvent(aSource, aEventType, aEventData, aEventData1, aEventData2);
    case EDisplayable:
        return iToolkit->PostDisplayableEvent(aSource, aEventType, aEventData, aEventData1);
    case ECanvasGraphicsItemPainterEvent:
        return iToolkit->PostCanvasGraphicsItemPainterEvent(aSource, aEventType, aEventData, aEventData1);
    default:
        ASSERT(EFalse);
    }
    return EFalse;
}

TBool CMIDEnv::PostJavaEvent(MMIDComponent& aSource,TSourceType aSourceType,TEventType aEventType,TInt aEventData)
{
    return PostJavaEvent(aSource, aSourceType, aEventType, aEventData, 0, 0);
}

TBool CMIDEnv::PostJavaEvent(MMIDComponent& aSource,TSourceType aSourceType,TInt aEventData)
{
    return PostJavaEvent(aSource, aSourceType, ENoType, aEventData, 0, 0);
}

TBool CMIDEnv::PostMidletEvent(TEventType aEventType)
{
    return iToolkit->PostDisplayEvent(aEventType);
}

MMIDNotifyEvent* CMIDEnv::NewNotifyL(MMIDComponent& aSource)
{
    return new(ELeave) CMIDNotifyEvent(iToolkit->ComponentPeer(aSource));
}

TBool   CMIDEnv::PostJavaNotify(MMIDNotifyEvent* aNotifyEvent)
{
    CMIDNotifyEvent* notify = static_cast<CMIDNotifyEvent*>(aNotifyEvent);
    return iToolkit->PostEvent(notify, CJavaEventBase::ENotifyPriority);
}

TBool CMIDEnv::TranslateKeyL(TMIDKeyEvent& aEvent, const TKeyEvent& aWsEvent, TEventCode aType)
{
    return iKeyTranslator->TranslateL(aEvent, aWsEvent, aType);
}

TBool CMIDEnv::PostKeyEvent(MMIDComponent& aSource, TMIDKeyEvent& aEvent)
{
    return iKeyTranslator->PostKeyEvent(aSource, aEvent);
}

void CMIDEnv::ResetKeys()
{
    iKeyTranslator->Reset();
}

MMIDDisplayable* CMIDEnv::Current()
{
    return iToolkit->Current();
}

/**
 * Set the zoom size which may be queried by Canvas in the
 * plugin. This is a relic from when Canvas was implemented
 * in the framework. Should be deprecated.
 */
void CMIDEnv::SetCanvasZoomSize(const TSize& aSize)
{
    iCanvasZoomSize=aSize;
}

/**
 * Get the zoom size - used by the Canvas in the plugin.
 * Relic from when Canvas was implementd in the framework.
 * Should be deprecated.
 */
TSize CMIDEnv::CanvasZoomSize()
{
    return iCanvasZoomSize;
}

TSize CMIDEnv::CanvasAssumedSize()
{
    return iCanvasAssumedSize;
}

TInt CMIDEnv::NumColors()
{
    TDisplayMode mode = DisplayMode();
    return TDisplayModeUtils::NumDisplayModeColors(mode);
}

TDisplayMode CMIDEnv::DisplayMode()
{
    MMIDGraphicsFactory& factory = iToolkit->GraphicsFactory();
    return factory.DisplayMode();
}

void CMIDEnv::AddObserverL(MMIDEnvObserver& aObserver)
{
    ASSERT(iObservers.Find(&aObserver) == KErrNotFound);
    //
    User::LeaveIfError(iObservers.Append(&aObserver));
}

void CMIDEnv::RemoveObserver(MMIDEnvObserver& aObserver)
{
    TInt index = iObservers.Find(&aObserver);
    if (index != KErrNotFound)
    {
        iObservers.Remove(index);
    }
}

void CMIDEnv::HandleSwitchOnL(TBool aSwitchOn)
{
    for (TInt i = iObservers.Count(); i--;)
        iObservers[i]->HandleSwitchOnL(aSwitchOn);
}

void CMIDEnv::HandleForegroundL(TBool aForeground)
{
#ifdef RD_JAVA_NGA_ENABLED
    if (aForeground)
    {
        iFullOrPartialFg = ETrue;
    }
#endif // RD_JAVA_NGA_ENABLED    

    for (TInt i = iObservers.Count(); i--;)
    {
        iObservers[i]->HandleForegroundL(aForeground);
    }
    if (aForeground && iToolkit && iToolkit->Utils())
    {
        iToolkit->Utils()->HandleForegroundL(aForeground);
    }
}

void CMIDEnv::HandleResourceChangeL(TInt aType)
{
    for (TInt i = iObservers.Count(); i--;)
    {
        iObservers[i]->HandleResourceChangeL(aType);
    }
    if (iToolkit && iToolkit->Utils())
    {
        iToolkit->Utils()->HandleResourceChangedL();
    }
}

#ifdef RD_JAVA_NGA_ENABLED
void CMIDEnv::HandleFullOrPartialForegroundL(TBool aForeground)
{
    iFullOrPartialFg = aForeground;
    for (TInt i = iObservers.Count(); i--;)
    {
        iObservers[i]->HandleFullOrPartialForegroundL(aForeground);
    }
}

void CMIDEnv::HandleFreeGraphicsMemory()
{
    for (TInt i = iObservers.Count(); i--;)
    {
        iObservers[i]->HandleFreeGraphicsMemory();
    }
}

TBool CMIDEnv::HasFullOrPartialForeground() const
{
    return iFullOrPartialFg;
}
#endif //RD_JAVA_NGA_ENABLED

TInt CMIDEnv::MidletAttribute(const TDesC& aAttributeName, TPtrC& aAttributeValue)
{
    return iToolkit->MidletAttribute(aAttributeName,aAttributeValue);
}

TBool CMIDEnv::MidletAttributeIsSetToVal(const TDesC& aAttributeName,
        const TDesC& aAttributeValue)
{
    TPtrC attributeValue;

    if (MidletAttribute(aAttributeName, attributeValue) == KErrNone &&
            attributeValue.CompareF(aAttributeValue) == 0)
    {
        return ETrue;
    }
    else
    {
        return EFalse;
    }
}

TBool CMIDEnv::MidletAttributeContainsVal(const TDesC& aAttributeName,
        const TDesC& aAttributeValue)
{
    TPtrC attributeValue;

    if (MidletAttribute(aAttributeName, attributeValue) == KErrNone)
    {
        TInt startIdx = attributeValue.FindF(aAttributeValue);
        if (startIdx != KErrNotFound)
        {
            if (aAttributeName.FindF(LcduiMidletAttributes::KAttribUIEnhancement) == KErrNotFound)
            {
                // If there is a preceding character before the value substring, it must be a comma
                if (startIdx == 0 || attributeValue[ startIdx - 1 ] == KCharComma)
                {
                    // Check what comes after the found value substring.
                    TInt endIdx = startIdx + aAttributeValue.Length();
                    if (endIdx < attributeValue.Length())
                    {
                        // only comma is ok
                        if (attributeValue[ endIdx ] == KCharComma)
                        {
                            return ETrue;
                        }
                    }
                }
                else // the found substring was right at the end of attributeValue string
                {
                    return ETrue;
                }
            }
            else // it is "Nokia-UI-Enhancement" - it is enough just to identify the substring in list of values
            {
                return ETrue;
            }
        }
    }

    return EFalse;
}

void CMIDEnv::SetCanvasAssumedSize(const TSize& aSize)
{
    iCanvasAssumedSize = aSize;
}
CFbsBitmap* CMIDEnv::ReserveCanvasFrameBufferL(MMIDCanvas& aCanvas, const TSize& aSz)
{
    CMIDEnv::TMutexAutoLock lock(&iCanvasData.iLock);
    if (!aCanvas.IsGameCanvas())
    {
        if (!iCanvasData.iFrameBuffer)
        {
            iCanvasData.iFrameBuffer = CreateCanvasFrameBufferL(aCanvas, ETrue, aSz);
        }
        iCanvasData.iRefCount++;
        return iCanvasData.iFrameBuffer;
    }
    return CreateCanvasFrameBufferL(aCanvas, ETrue, aSz);
}

void CMIDEnv::ReleaseCanvasFrameBuffer(MMIDCanvas& aCanvas, CFbsBitmap*& aFrameBuffer)
{
    CMIDEnv::TMutexAutoLock lock(&iCanvasData.iLock);
    if (!aCanvas.IsGameCanvas())
    {
        if (iCanvasData.iFrameBuffer)
        {
            iCanvasData.iRefCount--;
            if (iCanvasData.iRefCount <= 0)
            {
                iCanvasData.Reset();
            }
        }
    }
    else
    {
        delete aFrameBuffer;
        aFrameBuffer = NULL;
    }
}

CFbsBitmap* CMIDEnv::CreateCanvasFrameBufferL(
    MMIDCanvas& /* aCanvas */, TBool aWsBitmap, const TSize& aSz)
{
    CFbsBitmap* bitmap = new(ELeave) CFbsBitmap;
    CleanupStack::PushL(bitmap);
    TInt err = KErrGeneral;
    err = bitmap->CreateHardwareBitmap(aSz, DisplayMode(), MidletUid());
    if (KErrNone != err)
    {
        // Failed to create hardware bitmap. Fall back to normal software only bitmaps
        User::LeaveIfError(bitmap->Create(aSz, DisplayMode()));
    }
    if (!aWsBitmap)
    {
        CleanupStack::Pop();
        return bitmap;
    }
    CWsBitmap* wsBitmap =
        new(ELeave) CWsBitmap(CCoeEnv::Static()->WsSession());
    err = wsBitmap->Duplicate(bitmap->Handle());
    if (KErrNone != err)
    {
        delete wsBitmap;
        User::Leave(err);
    }
    CleanupStack::PopAndDestroy(bitmap);
    return wsBitmap;
}

TBool CMIDEnv::CanvasHasBackground(const MMIDCanvas& /* aCanvas */) const
{
    return iCanvasData.iHasBackground;
}

#ifdef RD_JAVA_NGA_ENABLED
// ---------------------------------------------------------------------------
//
// ---------------------------------------------------------------------------
//
TBool CMIDEnv::IsHardwareAcceleratedL(MMIDEnv::THardwareType aHardwareType)
{
    ASSERT(iHardwareStatus);
    return (iHardwareStatus & aHardwareType);
}

// ---------------------------------------------------------------------------
// Checks from OpenGL ES API, if it is HW accelerated
// ---------------------------------------------------------------------------
//
TInt CMIDEnv::InitHardwareStatusL()
{
    iHardwareStatus = 1;

    EGLDisplay eglDisplay = eglGetDisplay(EGL_DEFAULT_DISPLAY);
    eglInitialize(eglDisplay, NULL, NULL);

    EGLConfig config;
    EGLint numOfConfigs = 0;
    // Choose suitable display config
    const EGLint attribs[] =
    {
        EGL_SURFACE_TYPE, EGL_PBUFFER_BIT,
        EGL_BUFFER_SIZE, 32,
        EGL_DEPTH_SIZE, 16,
        EGL_NONE
    };
    eglChooseConfig(eglDisplay, attribs, &config, 1, &numOfConfigs);
    // Create a surface
    const EGLint pBuffAttribs[] =
    {
        EGL_WIDTH, 1,
        EGL_HEIGHT, 1,
        EGL_NONE
    };
    EGLSurface surface = eglCreatePbufferSurface(eglDisplay, config, pBuffAttribs);
    // Create a context into this thread.
    EGLContext context = eglCreateContext(eglDisplay, config, EGL_NO_CONTEXT, NULL);
    // Try to bind surface to context
    if (context != EGL_NO_CONTEXT && surface != EGL_NO_SURFACE &&
            eglMakeCurrent(eglDisplay, surface, surface, context))
    {
        if (strstr((const char *)glGetString(GL_RENDERER), "HW"))
        {
            iHardwareStatus |= MMIDEnv::EHardware3D;
        }
        eglMakeCurrent(eglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
    }

    if (context != EGL_NO_CONTEXT)
    {
        eglDestroyContext(eglDisplay, context);
    }

    if (surface != EGL_NO_SURFACE)
    {
        eglDestroySurface(eglDisplay, surface);
    }

    eglTerminate(eglDisplay);

    return iHardwareStatus;
}

#endif // RD_JAVA_NGA_ENABLED

void CMIDEnv::MappingDataForKey(TKeyEvent& aEvent, TEventCode aType)
{
    iToolkit->Utils()->MappingDataForKey(aEvent, aType);
}

void CMIDEnv::SetLastKeyEvent(const TKeyEvent& aEvent)
{
    iToolkit->Utils()->SetLastKeyEvent(aEvent);
}


// ---------------------------------------------------------------------------
// CMIDEnv::ToLcduiObserver
// Gets an instance of ToLcduiObserver.
// ---------------------------------------------------------------------------
//
MMIDToLcduiObserver& CMIDEnv::ToLcduiObserver()
{
    return *iToLcduiObserver;
}

void CMIDEnv::DisplayableIsDestructed(const MMIDDisplayable* aDisplayable)
{
    // Inform CMIDToolkit about deleting of displayble. This is needed
    // for prevereting of panic during changing to new Displayable
    iToolkit->DisplayableIsDestructed(aDisplayable);
}

const MMIDDisplayable* CMIDEnv::LastFullscreenDisplayable() const
{
    return iToolkit->LastFullscreenDisplayable();
}