/*
* Copyright (C) 2010-2021 JPEXS
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package com.jpexs.decompiler.flash.gui;
import com.jpexs.decompiler.flash.SWF;
import com.jpexs.decompiler.flash.SWFInputStream;
import com.jpexs.decompiler.flash.action.Action;
import com.jpexs.decompiler.flash.action.LocalDataArea;
import com.jpexs.decompiler.flash.action.Stage;
import com.jpexs.decompiler.flash.configuration.Configuration;
import com.jpexs.decompiler.flash.ecma.Undefined;
import com.jpexs.decompiler.flash.exporters.commonshape.Matrix;
import com.jpexs.decompiler.flash.gui.player.MediaDisplay;
import com.jpexs.decompiler.flash.gui.player.MediaDisplayListener;
import com.jpexs.decompiler.flash.gui.player.Zoom;
import com.jpexs.decompiler.flash.tags.DefineButtonSoundTag;
import com.jpexs.decompiler.flash.tags.DoActionTag;
import com.jpexs.decompiler.flash.tags.base.BoundedTag;
import com.jpexs.decompiler.flash.tags.base.ButtonTag;
import com.jpexs.decompiler.flash.tags.base.CharacterTag;
import com.jpexs.decompiler.flash.tags.base.DrawableTag;
import com.jpexs.decompiler.flash.tags.base.PlaceObjectTypeTag;
import com.jpexs.decompiler.flash.tags.base.RenderContext;
import com.jpexs.decompiler.flash.tags.base.SoundTag;
import com.jpexs.decompiler.flash.tags.base.TextTag;
import com.jpexs.decompiler.flash.timeline.DepthState;
import com.jpexs.decompiler.flash.timeline.Frame;
import com.jpexs.decompiler.flash.timeline.Timeline;
import com.jpexs.decompiler.flash.timeline.Timelined;
import com.jpexs.decompiler.flash.types.ConstantColorColorTransform;
import com.jpexs.decompiler.flash.types.MATRIX;
import com.jpexs.decompiler.flash.types.RECT;
import com.jpexs.decompiler.flash.types.SOUNDINFO;
import com.jpexs.decompiler.flash.exporters.commonshape.ExportRectangle;
import com.jpexs.decompiler.flash.tags.DefineButton2Tag;
import com.jpexs.decompiler.flash.tags.DefineButtonTag;
import com.jpexs.decompiler.flash.tags.base.DisplayObjectCacheKey;
import com.jpexs.decompiler.flash.types.BUTTONCONDACTION;
import com.jpexs.helpers.ByteArrayRange;
import com.jpexs.helpers.Cache;
import com.jpexs.helpers.Reference;
import com.jpexs.helpers.SerializableImage;
import com.jpexs.helpers.Stopwatch;
import java.awt.AlphaComposite;
import java.awt.BasicStroke;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Cursor;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.KeyEventDispatcher;
import java.awt.KeyboardFocusManager;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.Shape;
import java.awt.Stroke;
import java.awt.Toolkit;
import java.awt.Transparency;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.event.MouseMotionAdapter;
import java.awt.event.MouseMotionListener;
import java.awt.event.MouseWheelEvent;
import java.awt.geom.AffineTransform;
import java.awt.geom.Ellipse2D;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.awt.image.VolatileImage;
import java.io.IOException;
import java.text.DecimalFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;
import java.util.Map;
import java.util.Timer;
import java.util.TimerTask;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.UnsupportedAudioFileException;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.event.MouseInputAdapter;
/**
*
* @author JPEXS
*/
public final class ImagePanel extends JPanel implements MediaDisplay {
private static final Logger logger = Logger.getLogger(ImagePanel.class.getName());
private final List listeners = new ArrayList<>();
private Timelined timelined;
private boolean stillFrame = false;
private volatile Timer timer;
private int frame = -1;
private int prevFrame = -1;
private boolean loop;
private LocalDataArea lda;
private boolean zoomAvailable = false;
private SWF swf;
private boolean loaded;
private int mouseButton;
private final JLabel debugLabel = new JLabel("-");
private Point cursorPosition = null;
private MouseEvent lastMouseEvent = null;
private final List soundPlayers = new ArrayList<>();
private final Cache displayObjectCache = Cache.getInstance(false, false, "displayObject");
private final IconPanel iconPanel;
private int time = 0;
private int selectedDepth = -1;
private int freeTransformDepth = -1;
private Zoom zoom = new Zoom();
private final Object delayObject = new Object();
private boolean drawReady;
private final int drawWaitLimit = 50; // ms
private TextTag textTag;
private TextTag newTextTag;
private int msPerFrame;
private final boolean lowQuality = false;
private Object lock = new Object();
private Point2D registrationPoint = null;
private Point2D registrationPointUpdated = null;
private int mode = Cursor.DEFAULT_CURSOR;
private Rectangle2D bounds;
private Matrix transform;
private AffineTransform transformUpdated;
private final double LQ_FACTOR = 2;
private static final int TOLERANCE_SCALESHEAR = 8;
private static final int TOLERANCE_ROTATE = 30;
private static final int REGISTRATION_TOLERANCE = 8;
private static final double CENTER_POINT_SIZE = 8;
private static final double HANDLES_WIDTH = 5;
private static final int HANDLES_STROKE_WIDTH = 2;
private static final int MODE_ROTATE_NE = -1;
private static final int MODE_ROTATE_SE = -2;
private static final int MODE_ROTATE_NW = -3;
private static final int MODE_ROTATE_SW = -4;
private static final int MODE_SHEAR_S = -5;
private static final int MODE_SHEAR_E = -6;
private static final int MODE_SHEAR_N = -7;
private static final int MODE_SHEAR_W = -8;
private static Cursor moveCursor;
private static Cursor moveRegPointCursor;
private static Cursor resizeNWSECursor;
private static Cursor resizeSWNECursor;
private static Cursor resizeXCursor;
private static Cursor resizeYCursor;
private static Cursor rotateCursor;
private static Cursor selectCursor;
private static Cursor shearXCursor;
private static Cursor shearYCursor;
private Point offsetPoint = new Point(0, 0);
private Rectangle _rect = null;
private ExportRectangle _viewRect = new ExportRectangle(0, 0, 1, 1);
private boolean playing = false;
private static Cursor loadCursor(String name, int x, int y) throws IOException {
Toolkit toolkit = Toolkit.getDefaultToolkit();
Image image = ImageIO.read(MainPanel.class.getResource("/com/jpexs/decompiler/flash/gui/graphics/cursors/" + name + ".png"));
return toolkit.createCustomCursor(image, new Point(x, y), name);
}
static {
try {
moveCursor = loadCursor("move", 0, 0);
moveRegPointCursor = loadCursor("move_regpoint", 0, 0);
resizeNWSECursor = loadCursor("resize_nw_se", 5, 5);
resizeSWNECursor = loadCursor("resize_sw_ne", 5, 5);
resizeXCursor = loadCursor("resize_x", 7, 4);
resizeYCursor = loadCursor("resize_y", 4, 7);
rotateCursor = loadCursor("rotate", 10, 7);
selectCursor = loadCursor("select", 0, 0);
shearXCursor = loadCursor("shear_x", 9, 5);
shearYCursor = loadCursor("shear_y", 5, 9);
} catch (IOException ex) {
Logger.getLogger(MainPanel.class.getName()).log(Level.SEVERE, null, ex);
}
}
public MATRIX getNewMatrix() {
synchronized (lock) {
DepthState ds = null;
Timeline timeline = timelined.getTimeline();
if (freeTransformDepth > -1 && timeline.getFrameCount() > frame) {
ds = timeline.getFrame(frame).layers.get(freeTransformDepth);
}
if (ds != null) {
CharacterTag cht = swf.getCharacter(ds.characterId);
if (cht != null) {
if (cht instanceof DrawableTag) {
RECT rect = timelined.getRect();
Matrix m = new Matrix();
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
if (lowQuality) {
zoomDouble /= LQ_FACTOR;
}
double zoom = zoomDouble;
m.translate(-rect.Xmin * zoom, -rect.Ymin * zoom);
m.scale(zoom);
Matrix eMatrix = Matrix.getScaleInstance(1 / SWF.unitDivisor).concatenate(m).inverse();
return transform.preConcatenate(eMatrix).toMATRIX();
}
}
}
return null;
}
}
public synchronized void selectDepth(int depth) {
if (depth != selectedDepth) {
this.selectedDepth = depth;
freeTransformDepth = -1;
}
hideMouseSelection();
redraw();
}
private void calculateFreeTransform() {
DepthState ds = null;
Timeline timeline = timelined.getTimeline();
if (freeTransformDepth > -1 && timeline.getFrameCount() > frame) {
ds = timeline.getFrame(frame).layers.get(freeTransformDepth);
}
if (ds != null) {
CharacterTag cht = swf.getCharacter(ds.characterId);
if (cht != null) {
if (cht instanceof DrawableTag) {
DrawableTag dt = (DrawableTag) cht;
int drawableFrameCount = dt.getNumFrames();
if (drawableFrameCount == 0) {
drawableFrameCount = 1;
}
Matrix m = new Matrix();
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
if (lowQuality) {
zoomDouble /= LQ_FACTOR;
}
double zoom = zoomDouble;
m.translate(-_viewRect.xMin * zoom, -_viewRect.yMin * zoom);
m.scale(zoom);
transform = Matrix.getScaleInstance(1 / SWF.unitDivisor).concatenate(m).concatenate(new Matrix(ds.matrix));
}
}
}
}
public synchronized void freeTransformDepth(int depth) {
if (depth != freeTransformDepth) {
this.freeTransformDepth = depth;
}
registrationPoint = null;
calculateFreeTransform();
hideMouseSelection();
}
public void fireMediaDisplayStateChanged() {
for (MediaDisplayListener l : listeners) {
l.mediaDisplayStateChanged(this);
}
}
@Override
public void addEventListener(MediaDisplayListener listener) {
listeners.add(listener);
}
@Override
public void removeEventListener(MediaDisplayListener listener) {
listeners.remove(listener);
}
@Override
public Color getBackgroundColor() {
if (swf != null && swf.getBackgroundColor() != null) {
return swf.getBackgroundColor().backgroundColor.toColor();
}
return Color.white;
}
private class IconPanel extends JPanel {
private SerializableImage _img;
private ButtonTag mouseOverButton = null;
private boolean autoFit = false;
private boolean allowMove = true;
private Point dragStart = null;
private synchronized SerializableImage getImg() {
return _img;
}
public synchronized Rectangle getRect() {
return _rect;
}
public boolean hasAllowMove() {
return allowMove;
}
VolatileImage renderImage;
public void render() {
SerializableImage img = getImg();
Rectangle rect = getRect();
/*if (img == null) {
return;
}*/
Graphics2D g2 = null;
VolatileImage ri;
do {
ri = this.renderImage;
if (ri == null) {
return;
}
int valid = ri.validate(View.getDefaultConfiguration());
if (valid == VolatileImage.IMAGE_INCOMPATIBLE) {
ri = View.createRenderImage(getWidth(), getHeight(), Transparency.TRANSLUCENT);
}
try {
g2 = ri.createGraphics();
g2.setPaint(View.transparentPaint);
g2.fill(new Rectangle(0, 0, getWidth(), getHeight()));
g2.setComposite(AlphaComposite.SrcOver);
g2.setPaint(View.getSwfBackgroundColor());
g2.fill(new Rectangle(0, 0, getWidth(), getHeight()));
g2.setComposite(AlphaComposite.SrcOver);
if (rect != null && img != null) {
int x = rect.x < 0 ? 0 : rect.x;
int y = rect.y < 0 ? 0 : rect.y;
g2.drawImage(img.getBufferedImage(), x, y, x + img.getWidth(), y + img.getHeight(), 0, 0, img.getWidth(), img.getHeight(), null);
/*g2.setColor(Color.red);
g2.drawRect(x, y, img.getWidth(), img.getHeight());*/
}
} finally {
if (g2 != null) {
g2.dispose();
}
}
} while (ri.contentsLost());
}
private boolean ctrlDown = false;
public IconPanel() {
KeyboardFocusManager manager = KeyboardFocusManager.getCurrentKeyboardFocusManager();
manager.addKeyEventDispatcher(new KeyEventDispatcher() {
@Override
public boolean dispatchKeyEvent(KeyEvent e) {
if ((e.getID() == KeyEvent.KEY_PRESSED) || (e.getID() == KeyEvent.KEY_RELEASED)) {
ctrlDown = ((e.getModifiersEx() & InputEvent.CTRL_DOWN_MASK) == InputEvent.CTRL_DOWN_MASK);
}
return false;
}
});
addComponentListener(new ComponentAdapter() {
@Override
public void componentResized(ComponentEvent e) {
int width = getWidth();
int height = getHeight();
if (width > 0 && height > 0) {
renderImage = View.createRenderImage(width, height, Transparency.TRANSLUCENT);
} else {
renderImage = null;
}
calcRect();
render();
repaint();
}
});
MouseInputAdapter mouseInputAdapter = new MouseInputAdapter() {
@Override
public void mousePressed(MouseEvent e) {
if (e.getButton() == MouseEvent.BUTTON1) {
dragStart = e.getPoint();
}
requestFocusInWindow();
}
@Override
public void mouseReleased(MouseEvent e) {
if (e.getButton() == MouseEvent.BUTTON1) {
dragStart = null;
if (freeTransformDepth > -1 && mode != Cursor.DEFAULT_CURSOR && registrationPointUpdated != null && transformUpdated != null) {
synchronized (lock) {
registrationPoint = new Point2D.Double(registrationPointUpdated.getX(), registrationPointUpdated.getY());
transform = new Matrix(transformUpdated);
transformUpdated = null;
}
repaint();
}
mode = Cursor.DEFAULT_CURSOR;
}
}
@Override
public void mouseDragged(MouseEvent e) {
if (dragStart != null && allowMove && mode == Cursor.DEFAULT_CURSOR) {
Point dragEnd = e.getPoint();
Point delta = new Point(dragEnd.x - dragStart.x, dragEnd.y - dragStart.y);
offsetPoint.x += delta.x;
offsetPoint.y += delta.y;
ExportRectangle oldViewRect = new ExportRectangle(_viewRect);
dragStart = dragEnd;
iconPanel.calcRect();
_viewRect = getViewRect();
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
synchronized (lock) {
if (transform != null) {
Matrix prevTransform = transform.clone();
Matrix m = new Matrix();
m.scale(1 / SWF.unitDivisor);
m.translate(-oldViewRect.xMin * zoomDouble, -oldViewRect.yMin * zoomDouble);
//m.scale(zoomDouble);
Matrix mi = m.inverse();
transform = transform.preConcatenate(mi);
Matrix m2 = new Matrix();
m2.scale(1 / SWF.unitDivisor);
m2.translate(-_viewRect.xMin * zoomDouble, -_viewRect.yMin * zoomDouble);
//m2.scale(zoomDouble);
transform = transform.preConcatenate(m2);
if (registrationPoint != null) {
double deltaX = ((oldViewRect.xMin - _viewRect.xMin) * zoomDouble / SWF.unitDivisor);
double deltaY = ((oldViewRect.yMin - _viewRect.yMin) * zoomDouble / SWF.unitDivisor);
registrationPoint = new Point2D.Double(registrationPoint.getX() + deltaX, registrationPoint.getY() + deltaY);
}
}
}
repaint();
return;
}
if (dragStart != null && freeTransformDepth > -1) {
if (transform == null) {
return;
}
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
int ex = e.getX() - _rect.x - (int) (_viewRect.xMin * zoomDouble / SWF.unitDivisor);
int ey = e.getY() - _rect.y - (int) (_viewRect.yMin * zoomDouble / SWF.unitDivisor);
int dsx = dragStart.x - _rect.x - (int) (_viewRect.xMin * zoomDouble / SWF.unitDivisor);
int dsy = dragStart.y - _rect.y - (int) (_viewRect.yMin * zoomDouble / SWF.unitDivisor);
if (mode == MODE_SHEAR_N) {
double shearX = -(ex - dsx) / (bounds.getHeight());
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(bounds.getX(), bounds.getY());
t.shear(shearX, 0);
t.translate(-bounds.getX(), -bounds.getY());
t.translate(ex - dsx, 0);
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_SHEAR_S) {
double shearX = (ex - dsx) / (bounds.getHeight());
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(bounds.getX(), bounds.getY());
t.shear(shearX, 0);
t.translate(-bounds.getX(), -bounds.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_SHEAR_W) {
double shearY = -(ey - dsy) / (bounds.getWidth());
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(bounds.getX(), bounds.getY());
t.shear(0, shearY);
t.translate(-bounds.getX(), -bounds.getY());
t.translate(0, ey - dsy);
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_SHEAR_E) {
double shearY = (ey - dsy) / (bounds.getWidth());
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(bounds.getX(), bounds.getY());
t.shear(0, shearY);
t.translate(-bounds.getX(), -bounds.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_ROTATE_SE) {
double deltaStartX = Math.abs(dsx - registrationPoint.getX());
double deltaStartY = Math.abs(dsy - registrationPoint.getY());
double deltaEndX = Math.abs(ex - registrationPoint.getX());
double deltaEndY = Math.abs(ey - registrationPoint.getY());
double deltaTheta = 0;
if (ex >= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//same
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaEnd - thetaStart;
} else if (ex >= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//anti clockwise
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = -(thetaStart + thetaEnd);
} else if (ex <= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//clock wise
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = thetaStart + thetaEnd;
} else if (ex <= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//opposite
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart + Math.toRadians(90) + thetaEnd;
}
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.rotate(deltaTheta, registrationPoint.getX(), registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_ROTATE_NW) {
double deltaStartX = Math.abs(dsx - registrationPoint.getX());
double deltaStartY = Math.abs(dsy - registrationPoint.getY());
double deltaEndX = Math.abs(ex - registrationPoint.getX());
double deltaEndY = Math.abs(ey - registrationPoint.getY());
double deltaTheta = 0;
if (ex >= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//opposite
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart + Math.toRadians(90) + thetaEnd;
} else if (ex >= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//clock wise
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = thetaStart + thetaEnd;
} else if (ex <= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//anti clockwise
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = -(thetaStart + thetaEnd);
} else if (ex <= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//same
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaEnd - thetaStart;
}
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.rotate(deltaTheta, registrationPoint.getX(), registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_ROTATE_NE) {
double deltaStartX = Math.abs(dsx - registrationPoint.getX());
double deltaStartY = Math.abs(dsy - registrationPoint.getY());
double deltaEndX = Math.abs(ex - registrationPoint.getX());
double deltaEndY = Math.abs(ey - registrationPoint.getY());
double deltaTheta = 0;
if (ex >= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//clock wise
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart + thetaEnd;
} else if (ex >= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//same
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart - thetaEnd;
} else if (ex <= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//opposite
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = thetaStart + Math.toRadians(90) + thetaEnd;
} else if (ex <= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//anti clockwise
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = -(thetaStart + thetaEnd);
}
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.rotate(deltaTheta, registrationPoint.getX(), registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == MODE_ROTATE_SW) {
double deltaStartX = Math.abs(dsx - registrationPoint.getX());
double deltaStartY = Math.abs(dsy - registrationPoint.getY());
double deltaEndX = Math.abs(ex - registrationPoint.getX());
double deltaEndY = Math.abs(ey - registrationPoint.getY());
double deltaTheta = 0;
if (ex >= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//anti clockwise
double thetaStart = Math.atan(deltaStartX / deltaStartY);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = -(thetaStart + thetaEnd);
} else if (ex >= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//opposite
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndX / deltaEndY);
deltaTheta = thetaStart + Math.toRadians(90) + thetaEnd;
} else if (ex <= registrationPoint.getX() && ey >= registrationPoint.getY()) {
//same
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart - thetaEnd;
} else if (ex <= registrationPoint.getX() && ey <= registrationPoint.getY()) {
//clock wise
double thetaStart = Math.atan(deltaStartY / deltaStartX);
double thetaEnd = Math.atan(deltaEndY / deltaEndX);
deltaTheta = thetaStart + thetaEnd;
}
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.rotate(deltaTheta, registrationPoint.getX(), registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.HAND_CURSOR) {
transformUpdated = new AffineTransform(transform.toTransform());
registrationPointUpdated = new Point2D.Double(ex, ey);
repaint();
}
if (mode == Cursor.MOVE_CURSOR) {
int deltaX = ex - dsx;
int deltaY = ey - dsy;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(deltaX, deltaY);
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.E_RESIZE_CURSOR) {
double deltaBefore = bounds.getX() + bounds.getWidth() - registrationPoint.getX();
double deltaX = ex - registrationPoint.getX();
double scaleX = deltaX / deltaBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), 0);
t.scale(scaleX, 1);
t.translate(-registrationPoint.getX(), 0);
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.W_RESIZE_CURSOR) {
double deltaBefore = registrationPoint.getX() - bounds.getX();
double deltaX = registrationPoint.getX() - ex;
double scaleX = deltaX / deltaBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), 0);
t.scale(scaleX, 1);
t.translate(-registrationPoint.getX(), 0);
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.S_RESIZE_CURSOR) {
double deltaBefore = bounds.getY() + bounds.getHeight() - registrationPoint.getY();
double deltaY = ey - registrationPoint.getY();
double scaleY = deltaY / deltaBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(0, registrationPoint.getY());
t.scale(1, scaleY);
t.translate(0, -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.N_RESIZE_CURSOR) {
double deltaBefore = registrationPoint.getY() - bounds.getY();
double deltaY = registrationPoint.getY() - ey;
double scaleY = deltaY / deltaBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(0, registrationPoint.getY());
t.scale(1, scaleY);
t.translate(0, -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.SE_RESIZE_CURSOR) {
double deltaXBefore = bounds.getX() + bounds.getWidth() - registrationPoint.getX();
double deltaYBefore = bounds.getY() + bounds.getHeight() - registrationPoint.getY();
double deltaX = ex - registrationPoint.getX();
double deltaY = ey - registrationPoint.getY();
double scaleX = deltaX / deltaXBefore;
double scaleY = deltaY / deltaYBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), registrationPoint.getY());
t.scale(scaleX, scaleY);
t.translate(-registrationPoint.getX(), -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.NE_RESIZE_CURSOR) {
double deltaXBefore = bounds.getX() + bounds.getWidth() - registrationPoint.getX();
double deltaYBefore = registrationPoint.getY() - bounds.getY();
double deltaX = ex - registrationPoint.getX();
double deltaY = registrationPoint.getY() - ey;
double scaleX = deltaX / deltaXBefore;
double scaleY = deltaY / deltaYBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), registrationPoint.getY());
t.scale(scaleX, scaleY);
t.translate(-registrationPoint.getX(), -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.SW_RESIZE_CURSOR) {
double deltaXBefore = registrationPoint.getX() - bounds.getX();
double deltaYBefore = bounds.getY() + bounds.getHeight() - registrationPoint.getY();
double deltaX = registrationPoint.getX() - ex;
double deltaY = ey - registrationPoint.getY();
double scaleX = deltaX / deltaXBefore;
double scaleY = deltaY / deltaYBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), registrationPoint.getY());
t.scale(scaleX, scaleY);
t.translate(-registrationPoint.getX(), -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
if (mode == Cursor.NW_RESIZE_CURSOR) {
double deltaXBefore = registrationPoint.getX() - bounds.getX();
double deltaYBefore = registrationPoint.getY() - bounds.getY();
double deltaX = registrationPoint.getX() - ex;
double deltaY = registrationPoint.getY() - ey;
double scaleX = deltaX / deltaXBefore;
double scaleY = deltaY / deltaYBefore;
AffineTransform newTransform = new AffineTransform(transform.toTransform());
AffineTransform t = new AffineTransform();
t.translate(registrationPoint.getX(), registrationPoint.getY());
t.scale(scaleX, scaleY);
t.translate(-registrationPoint.getX(), -registrationPoint.getY());
newTransform.preConcatenate(t);
Point2D newRegistrationPoint = new Point2D.Double();
t.transform(registrationPoint, newRegistrationPoint);
transformUpdated = newTransform;
registrationPointUpdated = newRegistrationPoint;
repaint();
}
}
}
@Override
public void mouseMoved(MouseEvent e) {
if (freeTransformDepth > -1) {
if (bounds == null) {
return;
}
if (registrationPoint == null) {
return;
}
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
int ex = e.getX() - _rect.x - (int) (_viewRect.xMin * zoomDouble / SWF.unitDivisor);
int ey = e.getY() - _rect.y - (int) (_viewRect.yMin * zoomDouble / SWF.unitDivisor);
boolean left = ex >= bounds.getX() - TOLERANCE_SCALESHEAR && ex <= bounds.getX() + TOLERANCE_SCALESHEAR;
boolean right = ex >= bounds.getX() + bounds.getWidth() - TOLERANCE_SCALESHEAR && ex <= bounds.getX() + bounds.getWidth() + TOLERANCE_SCALESHEAR;
boolean top = ey >= bounds.getY() - TOLERANCE_SCALESHEAR && ey <= bounds.getY() + TOLERANCE_SCALESHEAR;
boolean bottom = ey >= bounds.getY() + bounds.getHeight() - TOLERANCE_SCALESHEAR && ey <= bounds.getY() + bounds.getHeight() + TOLERANCE_SCALESHEAR;
boolean xcenter = ex >= bounds.getCenterX() - TOLERANCE_SCALESHEAR && ex <= bounds.getCenterX() + TOLERANCE_SCALESHEAR;
boolean ycenter = ey >= bounds.getCenterY() - TOLERANCE_SCALESHEAR && ey <= bounds.getCenterY() + TOLERANCE_SCALESHEAR;
boolean registration = ex >= registrationPoint.getX() - REGISTRATION_TOLERANCE
&& ex <= registrationPoint.getX() + REGISTRATION_TOLERANCE
&& ey >= registrationPoint.getY() - REGISTRATION_TOLERANCE
&& ey <= registrationPoint.getY() + REGISTRATION_TOLERANCE;
boolean rightRotate = ex > bounds.getX() + bounds.getWidth() - TOLERANCE_ROTATE && ex
<= bounds.getX() + bounds.getWidth() + TOLERANCE_ROTATE;
boolean bottomRotate = ey > bounds.getY() + bounds.getHeight() - TOLERANCE_ROTATE && ey
<= bounds.getY() + bounds.getHeight() + TOLERANCE_ROTATE;
boolean leftRotate = ex < bounds.getX() + TOLERANCE_ROTATE
&& ex >= bounds.getX() - TOLERANCE_ROTATE;
boolean topRotate = ey < bounds.getY() + TOLERANCE_ROTATE
&& ey >= bounds.getY() - TOLERANCE_ROTATE;
boolean inBounds = bounds.contains(ex, ey);
boolean shearX = ex > bounds.getX() && ex < bounds.getX() + bounds.getWidth();
boolean shearY = ey > bounds.getY() && ey < bounds.getY() + bounds.getHeight();
Cursor cursor;
if (top && left) {
mode = Cursor.NW_RESIZE_CURSOR;
cursor = resizeNWSECursor;
} else if (bottom && left) {
mode = Cursor.SW_RESIZE_CURSOR;
cursor = resizeSWNECursor;
} else if (top && right) {
mode = Cursor.NE_RESIZE_CURSOR;
cursor = resizeSWNECursor;
} else if (bottom && right) {
mode = Cursor.SE_RESIZE_CURSOR;
cursor = resizeNWSECursor;
} else if (top && xcenter) {
mode = Cursor.N_RESIZE_CURSOR;
cursor = resizeYCursor;
} else if (bottom && xcenter) {
mode = Cursor.S_RESIZE_CURSOR;
cursor = resizeYCursor;
} else if (left && ycenter) {
mode = Cursor.W_RESIZE_CURSOR;
cursor = resizeXCursor;
} else if (right && ycenter) {
mode = Cursor.E_RESIZE_CURSOR;
cursor = resizeXCursor;
} else if (registration) {
mode = Cursor.HAND_CURSOR;
cursor = moveRegPointCursor;
} else if (!inBounds && rightRotate && topRotate) {
mode = MODE_ROTATE_NE;
cursor = rotateCursor;
} else if (!inBounds && rightRotate && bottomRotate) {
mode = MODE_ROTATE_SE;
cursor = rotateCursor;
} else if (!inBounds && leftRotate && topRotate) {
mode = MODE_ROTATE_NW;
cursor = rotateCursor;
} else if (!inBounds && leftRotate && bottomRotate) {
mode = MODE_ROTATE_SW;
cursor = rotateCursor;
} else if (shearY && (left || right)) {
if (left) {
mode = MODE_SHEAR_W;
} else {
mode = MODE_SHEAR_E;
}
cursor = shearYCursor;
} else if (shearX && (top || bottom)) {
if (top) {
mode = MODE_SHEAR_N;
} else {
mode = MODE_SHEAR_S;
}
cursor = shearXCursor;
} else if (inBounds) {
mode = Cursor.MOVE_CURSOR;
cursor = moveCursor;
} else {
mode = Cursor.DEFAULT_CURSOR;
cursor = selectCursor;
}
setCursor(cursor);
}
}
@Override
public void mouseWheelMoved(MouseWheelEvent e) {
if (ctrlDown) {
if (e.getScrollType() == MouseWheelEvent.WHEEL_UNIT_SCROLL) {
int rotation = e.getWheelRotation();
if (rotation < 0) {
zoomIn();
} else {
zoomOut();
}
}
}
}
};
addMouseListener(mouseInputAdapter);
addMouseMotionListener(mouseInputAdapter);
addMouseWheelListener(mouseInputAdapter);
}
public void setAutoFit(boolean autoFit) {
this.autoFit = autoFit;
repaint();
}
public synchronized BufferedImage getLastImage() {
if (_img == null) {
return null;
}
return _img.getBufferedImage();
}
public synchronized void setImg(SerializableImage img) {
this._img = img;
calcRect();
render();
repaint();
}
public synchronized Point toImagePoint(Point p) {
if (_img == null) {
return null;
}
if (timelined != null) {
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
int w1 = (int) (timelined.getRect().Xmin * zoomDouble / SWF.unitDivisor);
int h1 = (int) (timelined.getRect().Ymin * zoomDouble / SWF.unitDivisor);
return new Point((p.x - _rect.x + w1), (p.y - _rect.y + h1));
}
return new Point((p.x - _rect.x), (p.y - _rect.y));
}
private void setAllowMove(boolean allowMove) {
this.allowMove = allowMove;
if (!allowMove) {
offsetPoint = new Point();
}
}
private synchronized void calcRect() {
_rect = calcRect(zoom);
}
private synchronized Rectangle calcRect(Zoom z) {
if (_img != null || timelined != null) {
//int w1 = (int) (_img.getWidth() * (lowQuality ? LQ_FACTOR : 1));
//int h1 = (int) (_img.getHeight() * (lowQuality ? LQ_FACTOR : 1));
double zoomDouble = z.fit ? getZoomToFit() : z.value;
int w1;
int h1;
if (timelined != null) {
w1 = (int) (timelined.getRectWithStrokes().getWidth() * zoomDouble / SWF.unitDivisor);
h1 = (int) (timelined.getRectWithStrokes().getHeight() * zoomDouble / SWF.unitDivisor);
} else {
w1 = (int) (_img.getWidth() * (lowQuality ? LQ_FACTOR : 1));
h1 = (int) (_img.getHeight() * (lowQuality ? LQ_FACTOR : 1));
}
int w2 = getWidth();
int h2 = getHeight();
int w;
int h;
if (autoFit) {
if (w1 <= w2 && h1 <= h2) {
w = w1;
h = h1;
} else {
h = h1 * w2 / w1;
if (h > h2) {
w = w1 * h2 / h1;
h = h2;
} else {
w = w2;
}
}
} else {
w = w1;
h = h1;
}
setAllowMove(h > h2 || w > w2);
Rectangle r2 = new Rectangle(getWidth() / 2 - w / 2 + offsetPoint.x, getHeight() / 2 - h / 2 + offsetPoint.y, w, h);
return r2;
}
return null;
}
@Override
protected void paintComponent(Graphics g) {
Graphics2D g2d = (Graphics2D) g;
VolatileImage ri = this.renderImage;
if (ri != null) {
calcRect();
if (ri.validate(View.getDefaultConfiguration()) != VolatileImage.IMAGE_OK) {
ri = View.createRenderImage(getWidth(), getHeight(), Transparency.TRANSLUCENT);
render();
}
if (ri != null) {
g2d.drawImage(ri, 0, 0, null);
}
}
g2d.setColor(Color.red);
DecimalFormat df = new DecimalFormat();
df.setMaximumFractionDigits(2);
df.setMinimumFractionDigits(0);
df.setGroupingUsed(false);
float frameLoss = 100 - (getFpsIs() / fpsShouldBe * 100);
if (Configuration._debugMode.get()) {
g2d.drawString("frameLoss:" + df.format(frameLoss) + "%", 20, 20);
}
}
}
@Override
public void setBackground(Color bg) {
if (iconPanel != null) {
iconPanel.setBackground(bg);
}
super.setBackground(bg);
}
@Override
public synchronized void addMouseListener(MouseListener l) {
iconPanel.addMouseListener(l);
}
@Override
public synchronized void removeMouseListener(MouseListener l) {
iconPanel.removeMouseListener(l);
}
@Override
public synchronized void addMouseMotionListener(MouseMotionListener l) {
iconPanel.addMouseMotionListener(l);
}
@Override
public synchronized void removeMouseMotionListener(MouseMotionListener l) {
iconPanel.removeMouseMotionListener(l);
}
private void updatePos(Timelined timelined, MouseEvent lastMouseEvent, Timer thisTimer) {
if (timelined != null) {
BoundedTag bounded = (BoundedTag) timelined;
RECT rect = bounded.getRect();
int width = rect.getWidth();
double scale = 1.0;
/*if (width > swf.displayRect.getWidth()) {
scale = (double) swf.displayRect.getWidth() / (double) width;
}*/
Matrix m = Matrix.getTranslateInstance(-rect.Xmin, -rect.Ymin);
m.scale(scale);
Point p = lastMouseEvent == null ? null : lastMouseEvent.getPoint();
synchronized (ImagePanel.class) {
if (timer == thisTimer) {
cursorPosition = p;
}
}
}
}
private void showSelectedName() {
if (selectedDepth > -1 && frame > -1 && timelined != null) {
DepthState ds = timelined.getTimeline().getFrame(frame).layers.get(selectedDepth);
if (ds != null) {
CharacterTag cht = timelined.getTimeline().swf.getCharacter(ds.characterId);
if (cht != null) {
debugLabel.setText(cht.getName());
}
}
}
}
public void hideMouseSelection() {
if (selectedDepth > -1) {
showSelectedName();
} else {
debugLabel.setText(" - ");
}
}
public ImagePanel() {
super(new BorderLayout());
//iconPanel.setHorizontalAlignment(JLabel.CENTER);
setOpaque(true);
setBackground(View.getDefaultBackgroundColor());
loop = true;
iconPanel = new IconPanel();
//labelPan.add(label, new GridBagConstraints());
add(iconPanel, BorderLayout.CENTER);
add(debugLabel, BorderLayout.NORTH);
iconPanel.addMouseListener(new MouseAdapter() {
@Override
public void mouseEntered(MouseEvent e) {
synchronized (ImagePanel.class) {
lastMouseEvent = e;
redraw();
}
}
@Override
public void mouseExited(MouseEvent e) {
synchronized (ImagePanel.class) {
lastMouseEvent = null;
hideMouseSelection();
redraw();
}
}
@Override
public void mousePressed(MouseEvent e) {
synchronized (ImagePanel.class) {
mouseButton = e.getButton();
lastMouseEvent = e;
redraw();
ButtonTag button = iconPanel.mouseOverButton;
if (button != null && freeTransformDepth == -1) {
DefineButtonSoundTag sounds = button.getSounds();
if (sounds != null && sounds.buttonSoundChar2 != 0) { // OverUpToOverDown
playSound((SoundTag) swf.getCharacter(sounds.buttonSoundChar2), sounds.buttonSoundInfo2, timer);
}
List actions = new ArrayList<>();
if (button instanceof DefineButton2Tag) {
DefineButton2Tag button2 = (DefineButton2Tag) button;
for (BUTTONCONDACTION ca : button2.actions) {
if (ca.condOverUpToOverDown) { //press
actions.add(ca.actionBytes);
}
}
}
if (button instanceof DefineButtonTag) {
DefineButtonTag button1 = (DefineButtonTag) button;
actions.add(button1.actionBytes);
}
for (ByteArrayRange actionBytes : actions) {
try {
int prevLength = actionBytes.getPos();
SWFInputStream rri = new SWFInputStream(swf, actionBytes.getArray(), 0, prevLength + actionBytes.getLength());
if (prevLength != 0) {
rri.seek(prevLength);
}
execute(rri);
} catch (IOException ex) {
Logger.getLogger(ImagePanel.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}
}
@Override
public void mouseReleased(MouseEvent e) {
synchronized (ImagePanel.class) {
mouseButton = 0;
lastMouseEvent = e;
redraw();
ButtonTag button = iconPanel.mouseOverButton;
if (button != null && freeTransformDepth == -1) {
DefineButtonSoundTag sounds = button.getSounds();
if (sounds != null && sounds.buttonSoundChar3 != 0) { // OverDownToOverUp
playSound((SoundTag) swf.getCharacter(sounds.buttonSoundChar3), sounds.buttonSoundInfo3, timer);
}
}
}
}
});
iconPanel.addMouseMotionListener(new MouseMotionAdapter() {
@Override
public void mouseMoved(MouseEvent e) {
synchronized (ImagePanel.class) {
lastMouseEvent = e;
redraw();
}
}
@Override
public void mouseDragged(MouseEvent e) {
synchronized (ImagePanel.class) {
lastMouseEvent = e;
redraw();
}
}
});
}
private synchronized void redraw() {
if (timer == null && timelined != null) {
startTimer(timelined.getTimeline(), false);
}
}
@Override
public synchronized void zoom(Zoom zoom) {
double zoomDoubleBefore = this.zoom.fit ? getZoomToFit() : this.zoom.value;
boolean modified = this.zoom.value != zoom.value || this.zoom.fit != zoom.fit;
if (modified) {
ExportRectangle oldViewRect = new ExportRectangle(_viewRect);
Rectangle oldRect = new Rectangle(_rect);
this.zoom = zoom;
displayObjectCache.clear();
double zoomDouble = zoom.fit ? getZoomToFit() : zoom.value;
offsetPoint.x = (int) (offsetPoint.x * zoomDouble / zoomDoubleBefore);
offsetPoint.y = (int) (offsetPoint.y * zoomDouble / zoomDoubleBefore);
iconPanel.calcRect();
_viewRect = getViewRect();
synchronized (lock) {
if (transform != null) {
Matrix m = new Matrix();
m.scale(1 / SWF.unitDivisor);
m.translate(-oldViewRect.xMin * zoomDoubleBefore, -oldViewRect.yMin * zoomDoubleBefore);
m.scale(zoomDoubleBefore);
Matrix mi = m.inverse();
transform = transform.preConcatenate(mi);
Matrix m2 = new Matrix();
m2.scale(1 / SWF.unitDivisor);
m2.translate(-_viewRect.xMin * zoomDouble, -_viewRect.yMin * zoomDouble);
m2.scale(zoomDouble);
transform = transform.preConcatenate(m2);
if (registrationPoint != null) {
double actualRegPointX = (oldViewRect.xMin * zoomDoubleBefore / SWF.unitDivisor + registrationPoint.getX());
double actualRegPointY = (oldViewRect.yMin * zoomDoubleBefore / SWF.unitDivisor + registrationPoint.getY());
double sizedRegPointX = (actualRegPointX * zoomDouble / zoomDoubleBefore);
double sizedRegPointY = (actualRegPointY * zoomDouble / zoomDoubleBefore);
double regPointX = (sizedRegPointX - _viewRect.xMin * zoomDouble / SWF.unitDivisor);
double regPointY = (sizedRegPointY - _viewRect.yMin * zoomDouble / SWF.unitDivisor);
registrationPoint = new Point2D.Double(regPointX, regPointY);
}
}
}
redraw();
if (textTag != null) {
setText(textTag, newTextTag);
}
fireMediaDisplayStateChanged();
}
}
@Override
public synchronized BufferedImage printScreen() {
return iconPanel.getLastImage();
}
@Override
public synchronized double getZoomToFit() {
if (timelined != null) {
RECT bounds = timelined.getRectWithStrokes();
double w1 = bounds.getWidth() / SWF.unitDivisor;
double h1 = bounds.getHeight() / SWF.unitDivisor;
double w2 = iconPanel.getWidth();
double h2 = iconPanel.getHeight();
double w;
double h;
h = h1 * w2 / w1;
if (h > h2) {
w = w1 * h2 / h1;
} else {
w = w2;
}
if (w1 <= Double.MIN_NORMAL) {
return 1.0;
}
return (double) w / (double) w1;
}
return 1;
}
@Override
public synchronized boolean zoomAvailable() {
return zoomAvailable;
}
public void setTimelined(final Timelined drawable, final SWF swf, int frame) {
Stage stage = new Stage(drawable) {
@Override
public void callFrame(int frame) {
executeFrame(frame);
}
@Override
public Object callFunction(long functionAddress, long functionLength, List