package org.simantics.diagram.export;
import java.awt.Point;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.io.File;
import java.util.concurrent.Semaphore;
import java.util.concurrent.atomic.AtomicReference;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.SubMonitor;
import org.simantics.Simantics;
import org.simantics.db.ReadGraph;
import org.simantics.db.Resource;
import org.simantics.db.common.ResourceArray;
import org.simantics.db.common.request.PossibleIndexRoot;
import org.simantics.db.common.request.UniqueRead;
import org.simantics.db.common.utils.NameUtils;
import org.simantics.db.exception.DatabaseException;
import org.simantics.db.exception.ValidationException;
import org.simantics.db.management.ISessionContext;
import org.simantics.diagram.elements.DiagramNodeUtil;
import org.simantics.g2d.canvas.ICanvasContext;
import org.simantics.g2d.canvas.impl.CanvasContext;
import org.simantics.g2d.chassis.ICanvasChassis;
import org.simantics.g2d.chassis.SWTChassis;
import org.simantics.g2d.participant.TransformUtil;
import org.simantics.g2d.scenegraph.ICanvasSceneGraphProvider;
import org.simantics.g2d.scenegraph.SceneGraphConstants;
import org.simantics.modeling.ModelingResources;
import org.simantics.scenegraph.g2d.nodes.NavigationNode;
import org.simantics.scenegraph.utils.NodeUtil;
import org.simantics.structural2.StructuralVariables;
import org.simantics.utils.DataContainer;
import org.simantics.utils.datastructures.Pair;
import org.simantics.utils.threads.AWTThread;
import org.simantics.utils.threads.IThreadWorkQueue;
import org.simantics.utils.threads.ThreadUtils;
import org.simantics.utils.threads.WorkerThread;
public class ImagePrinter {
public static class ImageExportPlan {
public File exportLocation;
public String name;
public Resource diagram;
// use dpi or size, not both.
public Double dpi;
public Point size;
public double margin;
}
/**
* @param monitor
* the progress monitor to use for reporting progress to the
* user. It is the caller's responsibility to call done() on the
* given monitor. Accepts null
, indicating that no
* progress should be reported and that the operation cannot be
* cancelled.
*
* @param exportPath
* @throws Exception
*/
public static BufferedImage printToImage(IProgressMonitor monitor, ImageExportPlan exportPlan) throws Exception {
SubMonitor progress = SubMonitor.convert(monitor, "Export to Image", 1);
WorkerThread workerThread = new WorkerThread("Diagram Image Painter");
workerThread.start();
String loc;
if (exportPlan.exportLocation == null)
loc = "clipboard";
else
loc = exportPlan.exportLocation.getAbsolutePath();
progress.beginTask("Writing " + exportPlan.name + " to " + loc, 1);
return render(workerThread, Simantics.getSessionContext(), exportPlan);
}
/**
* Renders diagram to BufferedImage
*
* @param thread
* @param sessionContext
* @param exportPlan
* @return
* @throws Exception
*/
public static BufferedImage render(
final IThreadWorkQueue thread,
final ISessionContext sessionContext,
final ImageExportPlan exportPlan)
throws Exception
{
final DataContainer result = new DataContainer(null);
final DataContainer exception = new DataContainer();
final CanvasContext ctx = new CanvasContext(thread);
final AtomicReference sgProvider = new AtomicReference();
try {
final Semaphore done = new Semaphore(0);
// IMPORTANT: Load diagram in a different thread than the canvas context thread!
ThreadUtils.getBlockingWorkExecutor().execute(new Runnable() {
@Override
public void run() {
try {
Pair modelAndRVI = sessionContext.getSession().syncRequest(new UniqueRead>() {
@Override
public Pair perform(ReadGraph graph) throws DatabaseException {
return new Pair( resolveModel(graph, exportPlan.diagram ), resolveRVI(graph, exportPlan.diagram) );
}
});
ICanvasSceneGraphProvider provider = DiagramNodeUtil.loadSceneGraphProvider(ctx, modelAndRVI.first, exportPlan.diagram, modelAndRVI.second);
sgProvider.set( provider );
ThreadUtils.asyncExec(thread, new Runnable() {
@Override
public void run() {
try {
ImageBuilder chassis = new ImageBuilder(exportPlan.exportLocation,exportPlan.dpi,exportPlan.size,exportPlan.margin);
result.set(chassis.paint(ctx));
} catch (Exception e) {
exception.set(e);
} finally {
done.release();
}
}
});
} catch (DatabaseException e) {
done.release();
exception.set(e);
} catch (Throwable e) {
done.release();
exception.set(new DatabaseException(e));
} finally {
done.release();
}
}
});
done.acquire(2);
if (exception.get() != null)
throw exception.get();
return result.get();
} finally {
if (sgProvider.get() != null)
sgProvider.get().dispose();
ctx.dispose();
}
}
/**
* Renders diagram to BufferedImage using existing CanvasContext.
*
* Note: While the method tries to restore setting as they were, there may be visible side effects in the diagram.
*
* @param context
* @param chassis
* @param exportPlan
* @return
* @throws Exception
*/
public static BufferedImage renderLocal(final ICanvasContext context, final ICanvasChassis chassis, final ImageExportPlan exportPlan) throws Exception {
final Semaphore done = new Semaphore(0);
final DataContainer result = new DataContainer(null);
final DataContainer exception = new DataContainer(null);
ThreadUtils.asyncExec(AWTThread.getThreadAccess(), new Runnable() {
@Override
public void run() {
try {
result.set(renderLocal(context, exportPlan, chassis));
} catch (Exception e) {
exception.set(e);
} finally {
done.release();
}
}
});
done.acquire();
if (exception.get() != null)
throw exception.get();
return result.get();
}
private static BufferedImage renderLocal(ICanvasContext context, ImagePrinter.ImageExportPlan exportPlan, ICanvasChassis chassis) throws Exception {
TransformUtil util = context.getSingleItem(TransformUtil.class);
AffineTransform at = util.getTransform();
final NavigationNode node = NodeUtil.findNodeById(context.getSceneGraph(), SceneGraphConstants.NAVIGATION_NODE_PATH);
boolean adapt = true;
if (node != null) {
adapt = node.getAdaptViewportToResizedControl();
// prevent NavigationNode from re-scaling when it detects canvas size change.
node.setAdaptViewportToResizedControl(false);
}
ImageBuilder builder = new ImageBuilder(exportPlan.exportLocation,exportPlan.dpi,exportPlan.size,exportPlan.margin);
BufferedImage image = builder.paint(context);
util.setTransform(at);
if (node != null && adapt != false) {
if (chassis instanceof SWTChassis) {
((SWTChassis) chassis).getAWTComponent().repaint();
}
// restore re-scaling setting (need to be scheduled, so that Diagram canvas is re-painted once.
final boolean b = adapt;
ThreadUtils.asyncExec(AWTThread.getThreadAccess(), new Runnable() {
@Override
public void run() {
node.setAdaptViewportToResizedControl(b);
}
});
}
return image;
}
private static Resource resolveModel(ReadGraph graph, Resource diagram) throws DatabaseException {
ModelingResources mod = ModelingResources.getInstance(graph);
Resource composite = graph.getSingleObject(diagram, mod.DiagramToComposite);
Resource model = graph.syncRequest(new PossibleIndexRoot(composite));
if (model == null)
throw new ValidationException("no model found for composite " + NameUtils.getSafeName(graph, composite));
return model;
}
private static String resolveRVI(ReadGraph graph, Resource diagram) throws DatabaseException {
ModelingResources mod = ModelingResources.getInstance(graph);
Resource composite = graph.getSingleObject(diagram, mod.DiagramToComposite);
final ResourceArray compositePath = StructuralVariables.getCompositeArray(graph, composite);
final ResourceArray variablePath = compositePath.removeFromBeginning(1);
return StructuralVariables.getRVI(graph, variablePath);
}
}