sl@0: // Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // Interface for Window Server Scene sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @publishedPartner sl@0: @prototype sl@0: */ sl@0: sl@0: #ifndef WSSCENE_H sl@0: #define WSSCENE_H sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: class MWsElement; sl@0: sl@0: /** sl@0: * @brief Window Server Scene Interface Definition. sl@0: * sl@0: * MWsScene is a CWsRenderStage interface extension. An understanding sl@0: * of the Render Stage Plug-in architecture, and the CWsRenderStage API sl@0: * is required in order to use and understand this interface. sl@0: * sl@0: * A MWsScene manages a collection of MWsElements that define sl@0: * what is shown to a User on a particular target display, or off-screen sl@0: * buffer. sl@0: * sl@0: * A MWsScene constitute of MWsElements. Each MWsElement represents sl@0: * a rectangular region of the screen. MWsElements have meta-data which sl@0: * describe the region, such as the image orientation, extent, and sl@0: * alpha-blending characteristics. MWsElements also have surface-data, sl@0: * which provides the pixel data to be shown in the given area of the sl@0: * screen. sl@0: * sl@0: * The purpose of an MWsScene is to manage the relative ordinal positions sl@0: * of scene elements. Typically elements representing the camera view- sl@0: * finder, or video frames, is placed at the bottom, and the user interface sl@0: * element is placed above. Semi-transparency is often used to blend sl@0: * such elements into a visually appealing scene, and transparent holes sl@0: * in the user interface element allow elements below to poke through. sl@0: * sl@0: * The goal of this interface is to allow suitable platforms to implement sl@0: * this API using composition hardware. This is to reduce power and sl@0: * CPU load, so a long movie might be played, for example. sl@0: * sl@0: * A secondary goal of this interface is to enable transition effects sl@0: * to be efficiently implemented. This is because MWsScenes can be sl@0: * chained together in a Render Stage pipeline. Element meta-data sl@0: * can be manipulated by transition effects render stages before actual sl@0: * rendering. sl@0: * sl@0: *

Examples of Elements

sl@0: * @li Camera view-finder sl@0: * @li Video frame sl@0: * @li User interface sl@0: * sl@0: * There can be two types of user interface element: directly rendered sl@0: * and indirectly rendered. sl@0: * sl@0: * Indirect rendering is the normal case. Here the client provides sl@0: * CWindowGc commands, which Window Server passes as MWsGraphicsContext sl@0: * API calls to the Render Stage. sl@0: * sl@0: * Direct rendering is where Window Server does not see the graphics sl@0: * commands, as is the case for Direct Screen Access clients. sl@0: * sl@0: *

Use Cases

sl@0: * sl@0: *

Instantiation

sl@0: * sl@0: * During system-startup Window Server needs to obtain an instance of sl@0: * the MWsScene interface in order to place MWsElements in the scene sl@0: * for rendering. sl@0: * sl@0: * MWsScene is a CWsRenderStage interface extension, accessed by sl@0: * @code sl@0: * CWsRenderStage::ResolveObjectInterface(KMWsScene); sl@0: * @endcode sl@0: * sl@0: * The initial values are: sl@0: * @li No rotation is applied sl@0: * @li No Elements are in the Scene sl@0: * @li Scene is connected to same display as its CWsRenderStage sl@0: * sl@0: * Note Elements may be added/inserted/removed subject to implementation- sl@0: * defined constraints; the connected display is immutable. sl@0: * sl@0: *

Rendering

sl@0: * sl@0: * When the Scene changes, Window Server needs to communicate the changes sl@0: * and then render the new Scene. This can happen when a video player sl@0: * application is launched, whereby a new User Interface control is placed sl@0: * on the screen, and a new Element representing the Video Surface is added. sl@0: * sl@0: * The Window Server must use the CWsRenderStage interface to provide sl@0: * updates to the Scene. sl@0: * @code sl@0: * CWsRenderStage::Begin(regionOfScreenWithVideoPlayer); sl@0: * drawing ops sent to MWsGraphicsContext for UI sl@0: * drawing ops to setup a transparent hole for video surface to poke sl@0: * through sl@0: * . sl@0: * . sl@0: * sl@0: * sceneElement = CreateSceneElementL(); sl@0: * sceneElement->ConnectSurface(videoSurfaceAlreadySetup); sl@0: * InsertSceneElement(sceneElement, NULL); sl@0: * CWsRenderStage::End(asyncCallbackNotification); sl@0: * @endcode sl@0: * sl@0: * When Scene updates, such as in the example above, are done there is sl@0: * a concept of the "Committed Scene". All Scene updates are considered sl@0: * to be "Pending" until a CWsRenderStage::End() has been called. Then sl@0: * the pending changes are applied atomically. Either the previously sl@0: * Committed Scene is re-instated or all the Pending changes are accepted sl@0: * to form the newly Committed Scene. This depends on platform-specific sl@0: * limitations, and system-wide resource availability. sl@0: * sl@0: *

Orientation Changes to Elements

sl@0: * sl@0: * Suppose we have an MWsElement representing people as seen through a video sl@0: * camera view finder in landscape orientation, with the handset also in sl@0: * landscape orientation. The handset is rotated into portrait mode. sl@0: * sl@0: * We desire the view finder to be rotated in the opposite sl@0: * sense to the rest of the User Interface. This is to allow people sl@0: * previously vertical in the view finder to remain vertical. sl@0: * sl@0: * For this use case, a MWsElement::SetSourceRotation() must be called. sl@0: * sl@0: * The changes are visible after the next @c CWsRenderStage::End() call. sl@0: * sl@0: *

Orientation Changes to Scenes

sl@0: * sl@0: * Suppose the entire Scene comprises a single User Interface element. sl@0: * The handset is then rotated from landscape to portrait mode. sl@0: * The whole scene must be rotated to compensate. Here sl@0: * MWsScene::SetSceneRotation() must be called. sl@0: * sl@0: * The changes are visible after the next @c CWsRenderStage::End() call. sl@0: * sl@0: *

Leveraging external APIs for Transition Effects

sl@0: * sl@0: * Suppose we want to manipulate the Scene to achieve a transition sl@0: * effect. The Scene as presented by Window Server comprises one sl@0: * UI Element, and one External Surface Element. We have an external sl@0: * APIs, such as OpenGL or OpenWF, available to deliver a transition effect sl@0: * into a different External Surface. sl@0: * sl@0: * This task is accomplished by using the output Surface of OpenGL as sl@0: * the connected Image Source for the one Element which comprises the Scene sl@0: * presented on the Screen. sl@0: * The Elements inserted into the Scene by Window Server are instead given sl@0: * to OpenGL as textures. When the Screen needs to be rendered, OpenGL sl@0: * is first used to render to its output Surface, and this is taken sl@0: * as-is for the Scene Element that appears on the Screen. sl@0: * sl@0: * A combination of Rendering APIs may be employed if they are available sl@0: * to the Transition Effects Render Stage -- to combine the power of sl@0: * OpenVG, OpenGL, and OpenWF, as an example. sl@0: * sl@0: *

Managing the life-cycle of Surfaces from Window Server

sl@0: * sl@0: * Suppose we want to maintain the life-cycle of a given Surface outside sl@0: * that needed by the Scene. For example, suppose we have a video player sl@0: * application whose picture is frozen to a particular frame. Then the sl@0: * application switches to some other view, no longer showing the video sl@0: * frame. Finally, the application switches back to the original video sl@0: * frame so that the video clip can be resumed. sl@0: * sl@0: * In such a scenario, we would want the contents of the Surface to sl@0: * remain constant event though at one point the Surface is not part sl@0: * of the Scene. This is so that the user does not see anything sl@0: * inconsistent appear with the video frame between switching from and sl@0: * to the video frame. sl@0: * sl@0: * Another scenario where the Surface needs to have a life-cycle beyond sl@0: * that of the Scene is when there is a large start-up cost to populating sl@0: * the Surface with initial data. For example, a computer game application sl@0: * might take some seconds to construct the textures for an initial game sl@0: * position. The game would want to construct the Surface, and populate sl@0: * its contents before supplying the Surface as a Background Surface for sl@0: * a given window. This is so that the user does not see the game building sl@0: * up the Surface contents during game start-up. sl@0: * sl@0: * To facilitate the above use case, a call to RegisterSurface and a sl@0: * corresponding call to UnregisterSurface must be made. Note, Window sl@0: * Server should provide an UnregisterSurface call if the client program sl@0: * dies or exits abruptly before it could call UnregisterSurface. sl@0: */ sl@0: class MWsScene : public MWsObjectProvider sl@0: { sl@0: public: sl@0: DECLARE_WS_TYPE_ID(KMWsScene) sl@0: sl@0: public: sl@0: sl@0: /** sl@0: * Amount of rotation of pixel data in the target display or sl@0: * target surface for the entire scene. sl@0: */ sl@0: enum TSceneRotation sl@0: { sl@0: ESceneAntiClockwise0 = 0, /** No rotation */ sl@0: ESceneAntiClockwise90 = 1, /** 90 Degrees Anti-clockwise in target */ sl@0: ESceneAntiClockwise180 = 2, /** 180 Degrees Anti-clockwise in target */ sl@0: ESceneAntiClockwise270 = 3 /** 270 Degrees Anti-clockwise in target */ sl@0: }; sl@0: sl@0: /** sl@0: * @brief Update the rotation setting for the entire scene. sl@0: * @param aSceneRotation The new rotation setting sl@0: * @note The changes are visible after the next @c CWsRenderStage::End() sl@0: * or @c ScreenSnapshot() call. sl@0: * @return One of the system-wide error codes. KErrArgument means sl@0: * aSceneRotation is out-of-range, KErrNotSupported means the sl@0: * setting is not supported. KErrNone is returned upon success. sl@0: */ sl@0: virtual TInt SetSceneRotation(const TSceneRotation aSceneRotation) = 0; sl@0: sl@0: /** sl@0: * @brief Obtain the current rotation setting for the entire scene sl@0: * @return The currently committed scene rotation setting. sl@0: */ sl@0: virtual TSceneRotation SceneRotation() const = 0; sl@0: sl@0: /** sl@0: * @brief Create a Scene Element for this scene. sl@0: * sl@0: * Create a new Scene Element for use in the Scene. The Scene Element sl@0: * is immediately available to the caller, but is not in the Scene. sl@0: * It must be inserted into the Scene, at which time is is considered sl@0: * a Pending Scene change. Once @c CWsRenderStage::End() has been called sl@0: * the Element is rendered in the target and becomes part of the Committed sl@0: * Scene. sl@0: * sl@0: * @return The Scene Element sl@0: * @leave KErrNoMemory There are insufficient memory resources sl@0: */ sl@0: virtual MWsElement* CreateSceneElementL() = 0; sl@0: sl@0: /** sl@0: * @brief Destroy Scene Element sl@0: * sl@0: * Remove the specified element from the scene, and then destroy sl@0: * resources associated with it. sl@0: * sl@0: * @note The update is seen after the next @c CWsRenderStage::End() sl@0: * call. sl@0: * @param aElement The element to destroy. If NULL, the scene is sl@0: * unmodified. sl@0: */ sl@0: virtual void DestroySceneElement(MWsElement* aElement) = 0; sl@0: sl@0: /** sl@0: * @brief Insert or Re-insert the Scene Element into the Scene sl@0: * sl@0: * Insert the Scene Element into the Scene above the aSubordinateElement, sl@0: * or re-insert the existing Element but into a new position in the Scene. sl@0: * sl@0: * @note The same element cannot be in the scene at more than one sl@0: * z-order position. sl@0: * @pre The element must have previously been created @c sl@0: * CreateSceneElementL() sl@0: * @return One of the system-wide error codes. KErrArgument means sl@0: * aElement or aSubordinateElement is not a valid element for this sl@0: * Scene. KErrNone is returned upon success. sl@0: * @param aInsertElement The element to add; cannot be NULL sl@0: * @param aSubordinateElement The element which will be below sl@0: * aInsertElement when the scene is committed, sl@0: * or NULL to place at it the bottom of sl@0: * the scene. sl@0: */ sl@0: virtual TInt InsertSceneElement(MWsElement* aInsertElement, sl@0: MWsElement* aSubordinateElement) = 0; sl@0: sl@0: /** sl@0: * @brief Remove an Element from the Scene sl@0: * sl@0: * Remove the specified Element from the Scene, without destroying sl@0: * it. sl@0: * sl@0: * @note The update is seen after the next @c CWsRenderStage::End() sl@0: * call. sl@0: * @param aRemoveElement The element to be removed from the scene. sl@0: * If NULL, no action is taken. sl@0: * @pre The supplied element, if non-NULL, must be in the Scene; sl@0: * see @c InsertSceneElement() sl@0: * @post The removed element is no longer in the Scene, but is still sl@0: * valid object; it may receive further method calls. sl@0: * @return One of the system-wide error codes. KErrArgument means sl@0: * aRemoveElement was not in this Scene. KErrNone is returned sl@0: * upon success. sl@0: */ sl@0: virtual TInt RemoveSceneElement(MWsElement* aRemoveElement) = 0; sl@0: sl@0: /** sl@0: * @brief Compose pending Scene into an off-screen target sl@0: * sl@0: * Compose the pending Scene into an off-screen target. This sl@0: * method is an asynchronous call, and signals the optionally provided sl@0: * TRequestStatus, either immediately from some error cases, or later sl@0: * when the supplied off-screen target has been populated. sl@0: * sl@0: * However, the pending scene changes for the target screen are not sl@0: * committed by this function call. sl@0: * sl@0: * The TRequestStatus status value is updated by this function to one of sl@0: * the system-wide error codes. KErrArgument is set if the supplied surface sl@0: * or TRequestStatus is in error. KErrNoMemory is set for memory or other sl@0: * resource failures. KErrInUse is set if the surface is in use or sl@0: * otherwise inaccessible. KErrNotSupported is set if the supplied surface sl@0: * is incompatible with the screen. sl@0: * sl@0: * @param aOffScreenTarget Surface to be updated with pending sl@0: * Scene sl@0: * @param aCompleted When non-NULL, is used to signal the sl@0: * client when completed sl@0: * @post Either the supplied surface is updated with new contents sl@0: * and the aCompleted status is set to KErrNone, or an error value sl@0: * is set in the aCompleted status. sl@0: * @post Its guaranteed that the supplied surface is not accessed sl@0: * by MWsScene after aCompleted has been signaled. sl@0: */ sl@0: virtual void ComposePendingScene(TSurfaceId& aOffScreenTarget, sl@0: TRequestStatus* aCompleted) = 0; sl@0: sl@0: /** sl@0: * @brief Register the Window Server use of a Surface sl@0: * sl@0: * Register the Window Server use of a Surface for this Scene. sl@0: * The Scene abstraction already manages its use of Surfaces without sl@0: * this method call. The purpose of this method is to allow Window sl@0: * Server, on behalf of its clients, to mark the given Surface as sl@0: * being used by such clients. This allows the Surface to maintain sl@0: * a lifecycle beyond that which would normally be needed by the sl@0: * Scene, and allows early indication of incompatibility between sl@0: * the Surface and the Scene. sl@0: * sl@0: * @param aSurface Surface to be marked as in use by the caller sl@0: * @post Either the supplied surface accepted as potentially useable sl@0: * by this Scene, or the local error value is set. sl@0: * @return One of the system-wide error codes. KErrNotSupported means the sl@0: * Surface is incompatible with this Scene. KErrNoMemory means sl@0: * there was a memory problem. KErrArgument means the supplied sl@0: * Surface is not valid. KErrServerBusy means the Surface is sl@0: * currently used by another Scene. KErrNone is returned upon sl@0: * success. sl@0: */ sl@0: virtual TInt RegisterSurface(const TSurfaceId& aSurface) = 0; sl@0: sl@0: /** sl@0: * @brief Unregister the Window Server use of a Surface sl@0: * sl@0: * Unregister the Window Server use of a Surface with this Scene. sl@0: * The Scene abstraction already manages its use of Surfaces without sl@0: * this method call. The purpose of the method is to allow Window sl@0: * Server, on behalf of its clients, to mark the given Surface as sl@0: * no longer in use by such clients. This ends the lifecycle of sl@0: * the Surface from the point of view of such clients. sl@0: * sl@0: * @param aSurface Surface to be marked as no longer in use by the sl@0: * caller sl@0: * @return One of the system-wide error codes. KErrBadHandle means the sl@0: * supplied Surface is not known. KErrNone is returned upon sl@0: * success. sl@0: */ sl@0: virtual TInt UnregisterSurface(const TSurfaceId& aSurface) = 0; sl@0: }; sl@0: #endif // WSSCENE_H