The Frame Animator API allows MIDlets to trigger linear drag and kinetic scroll animations in response to drag and flick gestures. The Frame Animator API does not draw the UI updates for an animation, but rather calculates the motion interpolation for the animation. To draw the animation, use the standard LCDUI methods for rendering graphics.
The Frame Animator API is supported since Series 40 6th Edition, Feature Pack 1. The Frame Animator API is packaged as part of the Nokia UI API and consists of the following classes:
FrameAnimator
FrameAnimatorListener
Use the Frame Animator API together with the Gesture API.
Note: The following instructions
and code snippets only focus on the Frame Animator API. For instructions
on creating a full MIDlet, see sections Getting started and Implementing MIDlet lifecycle requirements. For instructions
on using the Canvas
class in a MIDlet, see sections Canvas and Example: Creating a simple test MIDlet.
To use drag and scroll animations in your MIDlet:
Create a UI component that receives drag and flick events, and define handlers for these events. For instructions, see sections Receiving gesture events and Handling gesture events.
Register a FrameAnimatorListener
for the UI component
using the FrameAnimator.register
method. The FrameAnimatorListener
notifies the MIDlet of each frame
update in the triggered animation.
The following
code snippet registers a MyFrameAnimatorListener
(see
step 4 of this example) for a custom Canvas
class
called GestureCanvas
.
// import the necessary classes import com.nokia.mid.ui.frameanimator.FrameAnimator; import com.nokia.mid.ui.gestures.GestureInteractiveZone; import com.nokia.mid.ui.gestures.GestureRegistrationManager; import javax.microedition.lcdui.Canvas; public class GestureCanvas extends Canvas { // FrameAnimator for registering a FrameAnimatorListener private FrameAnimator frameAnimator; // variables for keeping track of the animation coordinates private int currentX; private int currentY; public GestureCanvas() { // register a GestureListener and GestureInteractiveZone // ... // specify int referenceX, int referenceY, short maxFps, short maxPps // ... this.currentX = referenceX; this.currentY = referenceY; // create a MyFrameAnimatorListener instance MyFrameAnimatorListener myFrameAnimatorListener = new MyFrameAnimatorListener(this); // create a FrameAnimator instance and // use it to register the MyFrameAnimatorListener instance frameAnimator = new FrameAnimator(); frameAnimator.register(referenceX, referenceY, maxFps, maxPps, myFrameAnimatorListener); } // end GestureCanvas() // return and update the animation coordinates public int getCurrentX() { return this.currentX; } public void setCurrentX(int newX) { this.currentX = newX; } public int getCurrentY() { return this.currentY; } public void setCurrentY(int newY) { this.currentY = newY; } // return the FrameAnimator instance public FrameAnimator getFrameAnimator() { return this.frameAnimator; } // create the Canvas UI by implementing the paint method and // other necessary methods (see the LCDUI instructions) } // end class GestureCanvas
Registering a FrameAnimatorListener
requires that you specify the following values:
referenceX
specifies the starting x-coordinate
for the animation. The x-coordinates passed to the FrameAnimatorListener.animate
method are initially calculated from referenceX
.
referenceY
specifies the starting y-coordinate
for the animation. The y-coordinates passed to the FrameAnimatorListener.animate
method are initially calculated from referenceY
.
maxFps
specifies the maximum number of frames
per second (fps) for the animation. This controls how often the FrameAnimatorListener.animate
method is called per second
at maximum. The value is a percentage of the default platform value
for maximum fps. To retrieve the default platform value, use the com.nokia.mid.ui.frameanimator.fps
system
property:
defaultMaxFps = (Short)Integer.parseInt(System.getProperty("com.nokia.mid.ui.frameanimator.fps"));
The supported maxFps
values are 0
-200
. 0
and 100
indicate that the default platform value is used.
For example,
if your MIDlet can only handle 10 fps, but the native platform rate
is 20 fps, ensure that maxFps
is set to 50
or less. A value of 50
indicates 50% of the native
platform rate of 20 fps, amounting to 10 fps, the MIDlet's maximum
fps.
maxPps
specifies the maximum number of pixels
per second (pps) for a kinetic scroll animation. This controls the
distance in pixels between consecutive animation frame updates. The
value is a percentage of the default platform value for maximum pps.
To retrieve the default platform value, use the com.nokia.mid.ui.frameanimator.pps
system
property.
defaultMaxPps = (Short)Integer.parseInt(System.getProperty("com.nokia.mid.ui.frameanimator.pps"));
The supported maxPps
values are 0
-200
. 0
and 100
indicate that the default platform values is used.
For example,
if your MIDlet can only handle 10 pps, but the native platform rate
is 20 pps, ensure that maxPps
is set to 50
or less. A value of 50
indicates 50% of the native
platform rate of 20 pps, amounting to 10 pps, the MIDlet's maximum
pps.
Trigger the animations from the corresponding gesture event handlers:
To trigger a linear drag animation, use the FrameAnimator.drag
method.
To trigger a kinetic scroll animation, use the FrameAnimator.kineticScroll
method.
Calling the methods results in one or more calls to the FrameAnimatorListener.animate
method, which describes the
animation for the platform to draw (see step 4 of this example).
The following code snippet triggers the animations directly from
a GestureListener
implementation called MyGestureListener
.
// import the necessary classes import com.nokia.mid.ui.frameanimator.FrameAnimator; import com.nokia.mid.ui.gestures.GestureEvent; import com.nokia.mid.ui.gestures.GestureInteractiveZone; import com.nokia.mid.ui.gestures.GestureListener; public class MyGestureListener implements GestureListener { public void gestureAction(Object container, GestureInteractiveZone gestureInteractiveZone, GestureEvent gestureEvent) { // cast the container as a GestureCanvas for easy access GestureCanvas gestureCanvas = (GestureCanvas)container; // retrieve the FrameAnimator instance from the UI component FrameAnimator frameAnimator = gestureCanvas.getFrameAnimator(); // trigger the correct animation for the received event switch (gestureEvent.getType()) { case GestureInteractiveZone.GESTURE_DRAG: { // set the target coordinates for the animation by // retrieving the current animation coordinates and // adding the drag distance to them int newX = gestureCanvas.getCurrentX() + gestureEvent.getDragDistanceX(); int newY = gestureCanvas.getCurrentY() + gestureEvent.getDragDistanceY(); frameAnimator.drag(newX, newY); }; break; case GestureInteractiveZone.GESTURE_FLICK: { frameAnimator.kineticScroll(gestureEvent.getFlickSpeed(), FrameAnimator.FRAME_ANIMATOR_FREE_ANGLE, FrameAnimator.FRAME_ANIMATOR_FRICTION_MEDIUM, gestureEvent.getFlickDirection()); }; break; } } // end gestureAction() } // end class MyGestureListener
For more information
about implementing the GestureListener
interface,
see section Handling gesture events.
Tip: If you find that a drag animation
works too slowly and does not match the physical drag action, consider
implementing counters for keeping track of the distance moved after
each animate
call, and then using these counters
to align the animation with the corresponding drag events. For instructions
on how to create the counters, see Getting started with the Gesture API & FrameAnimator
API on Forum Nokia.
Implement the FrameAnimatorListener
registered in step 2. Use the FrameAnimatorListener.animate
method to define
the UI updates for the animation. The animate
method
is called every time the MIDlet calls either FrameAnimator.drag
or FrameAnimator.kineticScroll
. For a kinetic scroll
animation, the animate
method is called at regular
intervals with the updated position and other information describing
the current state of the animation, that is, its current frame.
The following code snippet creates the framework for
the MyFrameAnimatorListener
custom class.
// import the necessary classes import com.nokia.mid.ui.frameanimator.FrameAnimator; import com.nokia.mid.ui.frameanimator.FrameAnimatorListener; public class MyFrameAnimatorListener implements FrameAnimatorListener { private GestureCanvas gestureCanvas; public MyFrameAnimatorListener(GestureCanvas gestureCanvas) { this.gestureCanvas = gestureCanvas; } // implement the animate method public void animate(FrameAnimator frameAnimator, int x, int y, short delta, short deltaX, short deltaY, boolean lastFrame) { // update the current animation coordinates this.gestureCanvas.setCurrentX(x); this.gestureCanvas.setCurrentY(y); // refresh the UI based on received animation events // ... } // animate() } // end class MyFrameAnimatorListener
Each animate
call returns the following parameters:
frameAnimator
specifies the FrameAnimator
instance that triggered the animation.
x
specifies the current x-coordinate of the
animation. For a kinetic scroll animation, this value is initially
calculated from referenceX
used in registering the FrameAnimatorListener
.
y
specifies the current y-coordinate of the
animation. For a kinetic scroll animation, this value is initially
calculated from referenceY
used in registering the FrameAnimatorListener
.
delta
specifies the distance in pixels from
the previous position. This indicates the distance moved since the
previous animate
call.
deltaX
specifies the distance in pixels between
the current x-coordinate and the x-coordinate from the previous animate
call. This indicates the horizontal distance moved
since the previous animate
call.
deltaY
specifies the distance in pixels between
the current y-coordinate and the y-coordinate from the previous animate
call. This indicates the vertical distance moved
since the previous animate
call.
lastFrame
specifies whether this is the end
of the current animation. true
indicates that this
is the last frame of the animation, while false
indicates
that this is not the last frame.
After the animation has ended, the last coordinates are stored
in memory for as long as the FrameAnimatorListener
remains registered. Therefore a new animation using the same FrameAnimatorListener
instance begins where the last one
ended.