All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.codename1.ui.animations.Timeline Maven / Gradle / Ivy

There is a newer version: 7.0.167
Show newest version
/*
 * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores
 * CA 94065 USA or visit www.oracle.com if you need additional information or
 * have any questions.
 */

package com.codename1.ui.animations;

import com.codename1.ui.Graphics;
import com.codename1.ui.Image;
import com.codename1.ui.Painter;
import com.codename1.ui.geom.Dimension;
import com.codename1.ui.geom.Rectangle;

/**
 * A timeline represents the motions of animation objects
 *
 * @author Shai Almog
 */
public final class Timeline extends Image implements Animation, Painter {
    private int time;
    private int duration;
    AnimationObject[] animations;
    private Dimension size;
    private Dimension scaledTo;
    private long currentTime = -1;

    /**
     * Inidicates the minimal delay between animation frames allowing the CPU to rest.
     * Increase this number to increase general device performance, decrease it to speed
     * the animation.
     */
    private int animationDelay = 100;

    private boolean pause;

    private boolean loop = true;

    private Timeline() {
        super(null);
    }

    /**
     * {@inheritDoc}
     */
    public void lock() {
        if(animations != null) {
            int alen = animations.length;
            for(int iter = 0 ; iter < alen ; iter++) {
                animations[iter].lock();
            }
        }
    }


    /**
     * {@inheritDoc}
     */
    public void unlock() {
        if(animations != null) {
            int alen = animations.length;
            for(int iter = 0 ; iter < alen ; iter++) {
                animations[iter].unlock();
            }
        }
    }

    /**
     * {@inheritDoc}
     */
    public int[] getRGB() {
        Image i = Image.createImage(getWidth(), getHeight());
        paint(i.getGraphics(), new Rectangle(0, 0, getWidth(), getHeight()));
        return i.getRGB();
    }

    /**
     * {@inheritDoc}
     */
    public int[] getRGBCached() {
        return getRGB();
    }

    /**
     * Create a new timeline animation
     *
     * @param duration the duration of the animation in milliseconds
     * @param animations the animation objects that are part of this timeline
     * @param size size of the animation in virtual pixels, if the size differs the animation would be
     * scaled on the fly
     * @return the new timeline instance
     */
    public static Timeline createTimeline(int duration, AnimationObject[] animations, Dimension size) {
        if(duration <= 0) {
            throw new IllegalArgumentException("Illegal duration " + duration);
        }
        Timeline t = new Timeline();
        t.duration = duration;
        t.animations = animations;
        t.size = size;
        return t;
    }

    /**
     * Adds an animation object to show using this timeline
     * 
     * @param o animation object featured in this timeline
     */
    public void addAnimation(AnimationObject o) {
        AnimationObject[] n = new AnimationObject[animations.length + 1];
        System.arraycopy(animations, 0, n, 0, animations.length);
        n[animations.length] = o;
        animations = n;
    }

    /**
     * Set the time of the timeline
     *
     * @param time the time of the timeline in ms starting from 0
     */
    public void setTime(int time) {
        if(!pause) {
            if(time >= 0 && time <= duration) {
                this.time = time;
                currentTime = System.currentTimeMillis();
            }
        }
    }

    /**
     * Returns the time of the timeline
     *
     * @return the time of the timeline in ms starting from 0
     */
    public int getTime() {
        return time;
    }

    /**
     * {@inheritDoc}
     */
    public boolean isAnimation() {
        return true;
    }

    /**
     * {@inheritDoc}
     */
    public boolean animate() {
        if(!pause) {
            if(currentTime < 0) {
                currentTime = System.currentTimeMillis();
                setTime(0);
                return true;
            } else {
                long newCurrentTime = System.currentTimeMillis();
                if(newCurrentTime - currentTime >= animationDelay) {
                    int newTime = (int)(time + (newCurrentTime - currentTime));
                    currentTime = newCurrentTime;
                    if(newTime > duration) {
                        if(!loop) {
                            return false;
                        }
                        newTime = 0;
                    }
                    setTime(newTime);
                    return true;
                }
            }
        }
        return false;
    }

    /**
     * {@inheritDoc}
     */
    public void paint(Graphics g) {
        paint(g, null);
    }

    /**
     * {@inheritDoc}
     */
    public void paint(Graphics g, Rectangle rect) {
        float scaleX = 1;
        float scaleY = 1;
        if(rect != null) {
            scaleX = ((float)rect.getSize().getWidth()) / ((float)size.getWidth());
            scaleY = ((float)rect.getSize().getHeight()) / ((float)size.getHeight());
        }
        paintScaled(g, scaleX, scaleY);
    }

    private void paintScaled(Graphics g, float scaleX, float scaleY) {
        int alen = animations.length;
        for(int iter = 0 ; iter < alen ; iter++) {
            int s = animations[iter].getStartTime();
            if(s > -1 && s > time) {
                continue;
            }
            int e = animations[iter].getEndTime();
            if(e > -1 && e < time) {
                continue;
            }
            animations[iter].setTime(time);
            animations[iter].draw(g, scaleX, scaleY);
        }
    }

    /**
     * Inidicates the minimal delay between animation frames allowing the CPU to rest.
     * Increase this number to increase general device performance, decrease it to speed
     * the animation.
     *
     * @return the animationDelay
     */
    public int getAnimationDelay() {
        return animationDelay;
    }

    /**
     * Inidicates the minimal delay between animation frames allowing the CPU to rest.
     * Increase this number to increase general device performance, decrease it to speed
     * the animation.
     *
     * @param animationDelay the animationDelay to set
     */
    public void setAnimationDelay(int animationDelay) {
        this.animationDelay = animationDelay;
    }

    /**
     * {@inheritDoc}
     */
    protected void drawImage(Graphics g, Object nativeGraphics, int x, int y) {
        g.translate(x, y);
        if(scaledTo != null) {
            float scaleX = ((float)scaledTo.getWidth()) / ((float)size.getWidth());
            float scaleY = ((float)scaledTo.getHeight()) / ((float)size.getHeight());
            paintScaled(g, scaleX, scaleY);
        } else {
            paint(g);
        }
        g.translate(-x, -y);
    }

    /**
     * {@inheritDoc}
     */
    protected void drawImage(Graphics g, Object nativeGraphics, int x, int y, int w, int h) {
        g.translate(x, y);
        float scaleX = ((float)w) / ((float)size.getWidth());
        float scaleY = ((float)h) / ((float)size.getHeight());
        paintScaled(g, scaleX, scaleY);
        g.translate(-x, -y);
    }
    
    /**
     * {@inheritDoc}
     */
    public int getWidth() {
        if(scaledTo != null) {
            return scaledTo.getWidth();
        }
        return size.getWidth();
    }

    /**
     * {@inheritDoc}
     */
    public int getHeight() {
        if(scaledTo != null) {
            return scaledTo.getHeight();
        }
        return size.getHeight();
    }

    /**
     * {@inheritDoc}
     */
    public Image scaled(int width, int height) {
        Timeline t = new Timeline();
        t.animationDelay = animationDelay;
        t.animations = animations;
        t.currentTime = currentTime;
        t.duration = duration;
        t.size = size;
        t.time = time;
        t.scaledTo = new Dimension(width, height);
        return t;
    }

    /**
     * Returns true when the timeline is paused
     *
     * @return the pause state
     */
    public boolean isPause() {
        return pause;
    }

    /**
     * Indicate that the application is paused
     *
     * @param pause true to pause the application
     */
    public void setPause(boolean pause) {
        this.pause = pause;
    }

    /**
     * Returns the duration of the entire timeline in milliseconds
     *
     * @return the duration
     */
    public int getDuration() {
        return duration;
    }

    /**
     * Returns the pixel based unscaled dimentions of this timeline
     *
     * @return the size
     */
    public Dimension getSize() {
        return size;
    }

    /**
     * Returns the number of animation objects in this timeline
     *
     * @return the number of animations
     */
    public int getAnimationCount() {
        return animations.length;
    }

    /**
     * Returns the animation object in the given offset
     *
     * @param i the offset of the animation
     * @return the animation object
     */
    public AnimationObject getAnimation(int i) {
        return animations[i];
    }

    /**
     * Returns the animation object at the given X/Y coordinate in the timeline
     * for the current frame. This allows functionality such as responding to pointer
     * events on the resource editor. Notice that this method is not efficient since it tests
     * the pixel opacity which is a pretty expensive operation...
     *
     * @param x the x location in the timeline
     * @param y the y location in the timeline
     * @return an animation object or null if no animation object is at that position.
     */
    public AnimationObject getAnimationAt(int x, int y) {
        int alen = animations.length;
        for(int iter = 0 ; iter < alen ; iter++) {
            float scaleX = 1;
            float scaleY = 1;
            if(scaledTo != null) {
                scaleX = ((float)scaledTo.getWidth()) / ((float)size.getWidth());
                scaleY = ((float)scaledTo.getHeight()) / ((float)size.getHeight());
            }
            int w = (int)(animations[iter].getWidth() * scaleX);
            int h = (int)(animations[iter].getHeight() * scaleY);
            int ax = (int)(animations[iter].getX() * scaleX);
            int ay = (int)(animations[iter].getY() * scaleY);
            if(Rectangle.intersects(ax, ay, w, h, x, y, 1, 1)) {
                // we now need to check if the pixel at that position is not transparent
                int[] rgb = animations[iter].getImage().scaled(w, h).getRGB();
                int relativeX = x - ax;
                int relativeY = y - ay;
                int offset = relativeX + (relativeY * h);
                if(offset >= 0 && offset < rgb.length && (rgb[offset] & 0xff000000) != 0) {
                    return animations[iter];
                }
            }
        }
        return null;
    }

    /**
     * Indicates if the image should loop
     *
     * @return the loop
     */
    public boolean isLoop() {
        return loop;
    }

    /**
     * Indicates if the image should loop
     *
     * @param loop the loop to set
     */
    public void setLoop(boolean loop) {
        this.loop = loop;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean requiresDrawImage() {
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy