mirror of
https://github.com/euphy/polargraphcontroller
synced 2025-01-10 04:05:15 +01:00
Spirograph generator added, Raspberry Pi GPIO rotary encoders added
Draw Spirograph-like shapes within the polargraphcontroller by adjusting some parameters. Using a Raspberry Pi to run polargraphcontroller rotary encoders can be used to adjust the spirograph parameters.
This commit is contained in:
parent
936d918402
commit
dcd28fe4f9
@ -374,6 +374,84 @@ class DisplayMachine extends Machine
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void drawForSpiro()
|
||||||
|
{
|
||||||
|
// work out the scaling factor.
|
||||||
|
noStroke();
|
||||||
|
// draw machine outline
|
||||||
|
|
||||||
|
// drop shadow
|
||||||
|
fill(80);
|
||||||
|
rect(getOutline().getLeft()+DROP_SHADOW_DISTANCE, getOutline().getTop()+DROP_SHADOW_DISTANCE, getOutline().getWidth(), getOutline().getHeight());
|
||||||
|
|
||||||
|
fill(getMachineColour());
|
||||||
|
rect(getOutline().getLeft(), getOutline().getTop(), getOutline().getWidth(), getOutline().getHeight());
|
||||||
|
//text("machine " + getDimensionsAsText(getSize()) + " " + getZoomText(), getOutline().getLeft(), getOutline().getTop());
|
||||||
|
|
||||||
|
if (displayingGuides)
|
||||||
|
{
|
||||||
|
// draw some guides
|
||||||
|
stroke(getGuideColour());
|
||||||
|
strokeWeight(1);
|
||||||
|
// centre line
|
||||||
|
line(getOutline().getLeft()+(getOutline().getWidth()/2), getOutline().getTop(),
|
||||||
|
getOutline().getLeft()+(getOutline().getWidth()/2), getOutline().getBottom());
|
||||||
|
|
||||||
|
// page top line
|
||||||
|
line(getOutline().getLeft(), getOutline().getTop()+sc(getHomePoint().y),
|
||||||
|
getOutline().getRight(), getOutline().getTop()+sc(getHomePoint().y));
|
||||||
|
}
|
||||||
|
|
||||||
|
// draw page
|
||||||
|
fill(getPageColour());
|
||||||
|
rect(getOutline().getLeft()+sc(getPage().getLeft()),
|
||||||
|
getOutline().getTop()+sc(getPage().getTop()),
|
||||||
|
sc(getPage().getWidth()),
|
||||||
|
sc(getPage().getHeight()));
|
||||||
|
//text("page " + getDimensionsAsText(getPage()), getOutline().getLeft()+sc(getPage().getLeft()),
|
||||||
|
//getOutline().getTop()+sc(getPage().getTop()));
|
||||||
|
fill(0);
|
||||||
|
//text("offset " + getDimensionsAsText(getPage().getPosition()),
|
||||||
|
//getOutline().getLeft()+sc(getPage().getLeft()),
|
||||||
|
//getOutline().getTop()+sc(getPage().getTop())+10);
|
||||||
|
noFill();
|
||||||
|
|
||||||
|
// draw home point
|
||||||
|
noFill();
|
||||||
|
strokeWeight(5);
|
||||||
|
stroke(0, 128);
|
||||||
|
PVector onScreen = scaleToScreen(inMM(getHomePoint()));
|
||||||
|
ellipse(onScreen.x, onScreen.y, 15, 15);
|
||||||
|
strokeWeight(2);
|
||||||
|
stroke(255);
|
||||||
|
ellipse(onScreen.x, onScreen.y, 15, 15);
|
||||||
|
|
||||||
|
text("Home point", onScreen.x+ 15, onScreen.y-5);
|
||||||
|
//text(int(inMM(getHomePoint().x)+0.5) + ", " + int(inMM(getHomePoint().y)+0.5), onScreen.x+ 15, onScreen.y+15);
|
||||||
|
|
||||||
|
|
||||||
|
if (displayingGuides
|
||||||
|
&& getOutline().surrounds(getMouseVector())
|
||||||
|
&& currentMode != MODE_MOVE_IMAGE
|
||||||
|
&& mouseOverControls().isEmpty()
|
||||||
|
)
|
||||||
|
{
|
||||||
|
drawHangingStrings();
|
||||||
|
//drawLineLengthTexts();
|
||||||
|
cursor(CROSS);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cursor(ARROW);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (displayingVector && getVectorShape() != null)
|
||||||
|
{
|
||||||
|
displayVectorImage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// public void displayLiveVideo()
|
// public void displayLiveVideo()
|
||||||
// {
|
// {
|
||||||
// // draw actual image, maximum size
|
// // draw actual image, maximum size
|
||||||
|
@ -15,3 +15,11 @@ copies of all the libraries that I use, as well as all the source, and compiled
|
|||||||
|
|
||||||
sandy.noble@gmail.com
|
sandy.noble@gmail.com
|
||||||
http://www.polargraph.co.uk/
|
http://www.polargraph.co.uk/
|
||||||
|
|
||||||
|
|
||||||
|
Spirograph functions Copyright Mark Craig 2017
|
||||||
|
|
||||||
|
GPIO access on Raspberry Pi requires PiJ4 library available from http://pi4j.com/index.html.
|
||||||
|
|
||||||
|
@markcra
|
||||||
|
http://www.markcra.com/
|
||||||
|
@ -153,6 +153,59 @@ void numberbox_mode_vectorPathLengthHighPassCutoff(int value)
|
|||||||
pathLengthHighPassCutoff = value;
|
pathLengthHighPassCutoff = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void numberbox_mode_liveBigRadiusValue(int value)
|
||||||
|
{
|
||||||
|
if (value != Radius)
|
||||||
|
{
|
||||||
|
Radius = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void numberbox_mode_liveSmallRadiusValue(int value)
|
||||||
|
{
|
||||||
|
if (value != radius)
|
||||||
|
{
|
||||||
|
radius = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void numberbox_mode_liveRhoRadiusValue(int value)
|
||||||
|
{
|
||||||
|
if (value != rho)
|
||||||
|
{
|
||||||
|
rho = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void numberbox_mode_liveIntervalValue(int value)
|
||||||
|
{
|
||||||
|
if (value != Interval)
|
||||||
|
{
|
||||||
|
Interval = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void numberbox_mode_liveMultiplierValue(int value)
|
||||||
|
{
|
||||||
|
if (value != multiplier)
|
||||||
|
{
|
||||||
|
multiplier = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void numberbox_mode_liveModeValue(int value)
|
||||||
|
{
|
||||||
|
if (value != mode)
|
||||||
|
{
|
||||||
|
mode = value;
|
||||||
|
retraceShape = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void button_mode_exportSpirograph()
|
||||||
|
{
|
||||||
|
Export();
|
||||||
|
}
|
||||||
|
|
||||||
void button_mode_liveConfirmDraw()
|
void button_mode_liveConfirmDraw()
|
||||||
{
|
{
|
||||||
if (captureShape != null)
|
if (captureShape != null)
|
||||||
|
@ -131,6 +131,18 @@ Map<String, Panel> buildPanels() {
|
|||||||
tracePanel.setControlSizes(buildControlSizesForPanel(tracePanel));
|
tracePanel.setControlSizes(buildControlSizesForPanel(tracePanel));
|
||||||
panels.put(PANEL_NAME_TRACE, tracePanel);
|
panels.put(PANEL_NAME_TRACE, tracePanel);
|
||||||
|
|
||||||
|
Rectangle panelOutlineSpirograph = new Rectangle(getMainPanelPosition(),
|
||||||
|
new PVector((DEFAULT_CONTROL_SIZE.x+CONTROL_SPACING.x)*2, panelHeight));
|
||||||
|
Panel spirographPanel = new Panel(PANEL_NAME_SPIROGRAPH, panelOutlineSpirograph);
|
||||||
|
spirographPanel.setOutlineColour(color(200,255,200));
|
||||||
|
// get controls
|
||||||
|
spirographPanel.setResizable(true);
|
||||||
|
spirographPanel.setControls(getControlsForPanels().get(PANEL_NAME_SPIROGRAPH));
|
||||||
|
// get control positions
|
||||||
|
spirographPanel.setControlPositions(buildControlPositionsForPanel(spirographPanel));
|
||||||
|
spirographPanel.setControlSizes(buildControlSizesForPanel(spirographPanel));
|
||||||
|
panels.put(PANEL_NAME_SPIROGRAPH, spirographPanel);
|
||||||
|
|
||||||
Rectangle panelOutlineDetails = new Rectangle(getMainPanelPosition(),
|
Rectangle panelOutlineDetails = new Rectangle(getMainPanelPosition(),
|
||||||
new PVector((DEFAULT_CONTROL_SIZE.x+CONTROL_SPACING.x)*2, panelHeight));
|
new PVector((DEFAULT_CONTROL_SIZE.x+CONTROL_SPACING.x)*2, panelHeight));
|
||||||
Panel detailsPanel = new Panel(PANEL_NAME_DETAILS, panelOutlineDetails);
|
Panel detailsPanel = new Panel(PANEL_NAME_DETAILS, panelOutlineDetails);
|
||||||
@ -510,6 +522,54 @@ Map<String, Controller> initialiseNumberboxValues(Map<String, Controller> map)
|
|||||||
n.setMax(10);
|
n.setMax(10);
|
||||||
n.setMultiplier(0.1);
|
n.setMultiplier(0.1);
|
||||||
}
|
}
|
||||||
|
else if (MODE_LIVE_BIG_RADIUS_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(Radius);
|
||||||
|
n.setMin(1);
|
||||||
|
n.setMax(255);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
|
else if (MODE_LIVE_SMALL_RADIUS_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(radius);
|
||||||
|
n.setMin(1);
|
||||||
|
n.setMax(255);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
|
else if (MODE_LIVE_RHO_RADIUS_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(rho);
|
||||||
|
n.setMin(1);
|
||||||
|
n.setMax(255);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
|
else if (MODE_LIVE_INTERVAL_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(Interval);
|
||||||
|
n.setMin(100);
|
||||||
|
n.setMax(5000);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
|
else if (MODE_LIVE_MULTIPLIER_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(multiplier);
|
||||||
|
n.setMin(0);
|
||||||
|
n.setMax(60);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
|
else if (MODE_LIVE_MODE_VALUE.equals(key))
|
||||||
|
{
|
||||||
|
n.setDecimalPrecision(1);
|
||||||
|
n.setValue(mode);
|
||||||
|
n.setMin(0);
|
||||||
|
n.setMax(3);
|
||||||
|
n.setMultiplier(1);
|
||||||
|
}
|
||||||
else if (MODE_LIVE_SIMPLIFICATION_VALUE.equals(key))
|
else if (MODE_LIVE_SIMPLIFICATION_VALUE.equals(key))
|
||||||
{
|
{
|
||||||
n.setDecimalPrecision(1);
|
n.setDecimalPrecision(1);
|
||||||
@ -693,6 +753,7 @@ Map<String, List<Controller>> buildControlsForPanels()
|
|||||||
map.put(PANEL_NAME_QUEUE, getControllersForControllerNames(getControlNamesForQueuePanel()));
|
map.put(PANEL_NAME_QUEUE, getControllersForControllerNames(getControlNamesForQueuePanel()));
|
||||||
map.put(PANEL_NAME_GENERAL, getControllersForControllerNames(getControlNamesForGeneralPanel()));
|
map.put(PANEL_NAME_GENERAL, getControllersForControllerNames(getControlNamesForGeneralPanel()));
|
||||||
map.put(PANEL_NAME_TRACE, getControllersForControllerNames(getControlNamesForTracePanel()));
|
map.put(PANEL_NAME_TRACE, getControllersForControllerNames(getControlNamesForTracePanel()));
|
||||||
|
map.put(PANEL_NAME_SPIROGRAPH, getControllersForControllerNames(getControlNamesForSpirographPanel()));
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -806,6 +867,31 @@ List<String> getControlNamesForTracePanel()
|
|||||||
return controlNames;
|
return controlNames;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<String> getControlNamesForSpirographPanel() // now spirograph panel
|
||||||
|
{
|
||||||
|
List<String> controlNames = new ArrayList<String>();
|
||||||
|
controlNames.add(MODE_LIVE_BIG_RADIUS_VALUE);
|
||||||
|
controlNames.add(MODE_LIVE_SMALL_RADIUS_VALUE);
|
||||||
|
controlNames.add(MODE_LIVE_INTERVAL_VALUE);
|
||||||
|
controlNames.add(MODE_LIVE_RHO_RADIUS_VALUE);
|
||||||
|
controlNames.add(MODE_LIVE_MULTIPLIER_VALUE);
|
||||||
|
controlNames.add(MODE_LIVE_MODE_VALUE);
|
||||||
|
controlNames.add(MODE_EXPORT_SPIROGRAPH);
|
||||||
|
|
||||||
|
//copied from INPUT tab:
|
||||||
|
//controlNames.add(MODE_LOAD_VECTOR_FILE);
|
||||||
|
controlNames.add(MODE_RESIZE_VECTOR);
|
||||||
|
controlNames.add(MODE_MOVE_VECTOR);
|
||||||
|
controlNames.add(MODE_PEN_LIFT_UP);
|
||||||
|
controlNames.add(MODE_PEN_LIFT_DOWN);
|
||||||
|
controlNames.add(MODE_SET_POSITION_HOME);
|
||||||
|
controlNames.add(MODE_RETURN_TO_HOME);
|
||||||
|
controlNames.add(MODE_CLEAR_QUEUE);
|
||||||
|
controlNames.add(MODE_RENDER_VECTORS);
|
||||||
|
|
||||||
|
return controlNames;
|
||||||
|
}
|
||||||
|
|
||||||
List<String> getControlNamesForDetailPanel()
|
List<String> getControlNamesForDetailPanel()
|
||||||
{
|
{
|
||||||
List<String> controlNames = new ArrayList<String>();
|
List<String> controlNames = new ArrayList<String>();
|
||||||
@ -1016,6 +1102,14 @@ Map<String, String> buildControlLabels()
|
|||||||
result.put(MODE_LIVE_CANCEL_CAPTURE, "Cancel capture");
|
result.put(MODE_LIVE_CANCEL_CAPTURE, "Cancel capture");
|
||||||
result.put(MODE_LIVE_ADD_CAPTION, "Add caption");
|
result.put(MODE_LIVE_ADD_CAPTION, "Add caption");
|
||||||
|
|
||||||
|
result.put(MODE_LIVE_BIG_RADIUS_VALUE, "Major Radius");
|
||||||
|
result.put(MODE_LIVE_SMALL_RADIUS_VALUE, "Minor radius");
|
||||||
|
result.put(MODE_LIVE_RHO_RADIUS_VALUE, "Rho");
|
||||||
|
result.put(MODE_LIVE_INTERVAL_VALUE, "Interval");
|
||||||
|
result.put(MODE_LIVE_MULTIPLIER_VALUE, "Multiplier");
|
||||||
|
result.put(MODE_LIVE_MODE_VALUE, "Mode");
|
||||||
|
result.put(MODE_EXPORT_SPIROGRAPH, "Export Spirograph");
|
||||||
|
|
||||||
result.put(MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF, "Path length cutoff");
|
result.put(MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF, "Path length cutoff");
|
||||||
result.put(MODE_SHOW_WEBCAM_RAW_VIDEO, "Show video");
|
result.put(MODE_SHOW_WEBCAM_RAW_VIDEO, "Show video");
|
||||||
result.put(MODE_FLIP_WEBCAM_INPUT, "Flip video");
|
result.put(MODE_FLIP_WEBCAM_INPUT, "Flip video");
|
||||||
@ -1169,6 +1263,14 @@ Set<String> buildControlNames()
|
|||||||
result.add(MODE_LIVE_ADD_CAPTION);
|
result.add(MODE_LIVE_ADD_CAPTION);
|
||||||
result.add(MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF);
|
result.add(MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF);
|
||||||
|
|
||||||
|
result.add(MODE_LIVE_BIG_RADIUS_VALUE);
|
||||||
|
result.add(MODE_LIVE_SMALL_RADIUS_VALUE);
|
||||||
|
result.add(MODE_LIVE_RHO_RADIUS_VALUE);
|
||||||
|
result.add(MODE_LIVE_INTERVAL_VALUE);
|
||||||
|
result.add(MODE_LIVE_MULTIPLIER_VALUE);
|
||||||
|
result.add(MODE_LIVE_MODE_VALUE);
|
||||||
|
result.add(MODE_EXPORT_SPIROGRAPH);
|
||||||
|
|
||||||
result.add(MODE_SHOW_WEBCAM_RAW_VIDEO);
|
result.add(MODE_SHOW_WEBCAM_RAW_VIDEO);
|
||||||
result.add(MODE_FLIP_WEBCAM_INPUT);
|
result.add(MODE_FLIP_WEBCAM_INPUT);
|
||||||
result.add(MODE_ROTATE_WEBCAM_INPUT);
|
result.add(MODE_ROTATE_WEBCAM_INPUT);
|
||||||
|
@ -27,6 +27,14 @@
|
|||||||
http://www.polargraph.co.uk
|
http://www.polargraph.co.uk
|
||||||
https://github.com/euphy/polargraphcontroller
|
https://github.com/euphy/polargraphcontroller
|
||||||
|
|
||||||
|
===================
|
||||||
|
Modified by Mark Craig to include Spirograph (replaced Trace tab and butchered some of it's code)
|
||||||
|
|
||||||
|
Spirograph functions Copyright Mark Craig 2017
|
||||||
|
GPIO access on Raspberry Pi requires PiJ4 library available from http://pi4j.com/index.html.
|
||||||
|
|
||||||
|
@markcra
|
||||||
|
http://www.markcra.com/
|
||||||
*/
|
*/
|
||||||
|
|
||||||
//import processing.video.*;
|
//import processing.video.*;
|
||||||
@ -56,6 +64,24 @@ import java.awt.BorderLayout;
|
|||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
|
|
||||||
|
/* // Comment block the following out for use on non-Raspberry Pi hardware.
|
||||||
|
// for GPIO rotary encoders & buttons
|
||||||
|
import com.pi4j.io.gpio.*;
|
||||||
|
import com.pi4j.io.gpio.event.GpioPinDigitalStateChangeEvent;
|
||||||
|
import com.pi4j.io.gpio.event.GpioPinListenerDigital;
|
||||||
|
|
||||||
|
import com.pi4j.io.gpio.GpioController;
|
||||||
|
import com.pi4j.io.gpio.GpioFactory;
|
||||||
|
import com.pi4j.io.gpio.GpioPinDigitalInput;
|
||||||
|
import com.pi4j.io.gpio.PinPullResistance;
|
||||||
|
import com.pi4j.io.gpio.RaspiPin;
|
||||||
|
|
||||||
|
GpioController gpio;
|
||||||
|
GpioPinDigitalInput button1, input1A, input1B, button2, input2A, input2B, button3, input3A, input3B, button4, input4A, input4B;
|
||||||
|
GpioPinDigitalInput button5, input5A, input5B, button6, input6A, input6B, button7, input7A, input7B;
|
||||||
|
// end of comment block for RaspPi
|
||||||
|
*/
|
||||||
|
|
||||||
int majorVersionNo = 2;
|
int majorVersionNo = 2;
|
||||||
int minorVersionNo = 4;
|
int minorVersionNo = 4;
|
||||||
int buildNo = 2;
|
int buildNo = 2;
|
||||||
@ -330,6 +356,14 @@ static final String MODE_LIVE_CANCEL_CAPTURE = "button_mode_liveClearCapture";
|
|||||||
static final String MODE_LIVE_ADD_CAPTION = "button_mode_liveAddCaption";
|
static final String MODE_LIVE_ADD_CAPTION = "button_mode_liveAddCaption";
|
||||||
static final String MODE_LIVE_CONFIRM_DRAW = "button_mode_liveConfirmDraw";
|
static final String MODE_LIVE_CONFIRM_DRAW = "button_mode_liveConfirmDraw";
|
||||||
|
|
||||||
|
static final String MODE_LIVE_BIG_RADIUS_VALUE = "numberbox_mode_liveBigRadiusValue";
|
||||||
|
static final String MODE_LIVE_SMALL_RADIUS_VALUE = "numberbox_mode_liveSmallRadiusValue";
|
||||||
|
static final String MODE_LIVE_RHO_RADIUS_VALUE = "numberbox_mode_liveRhoRadiusValue";
|
||||||
|
static final String MODE_LIVE_INTERVAL_VALUE = "numberbox_mode_liveIntervalValue";
|
||||||
|
static final String MODE_LIVE_MULTIPLIER_VALUE = "numberbox_mode_liveMultiplierValue";
|
||||||
|
static final String MODE_LIVE_MODE_VALUE = "numberbox_mode_liveModeValue";
|
||||||
|
static final String MODE_EXPORT_SPIROGRAPH = "button_mode_exportSpirograph";
|
||||||
|
|
||||||
static final String MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF = "numberbox_mode_vectorPathLengthHighPassCutoff";
|
static final String MODE_VECTOR_PATH_LENGTH_HIGHPASS_CUTOFF = "numberbox_mode_vectorPathLengthHighPassCutoff";
|
||||||
static final String MODE_SHOW_WEBCAM_RAW_VIDEO = "toggle_mode_showWebcamRawVideo";
|
static final String MODE_SHOW_WEBCAM_RAW_VIDEO = "toggle_mode_showWebcamRawVideo";
|
||||||
static final String MODE_FLIP_WEBCAM_INPUT = "toggle_mode_flipWebcam";
|
static final String MODE_FLIP_WEBCAM_INPUT = "toggle_mode_flipWebcam";
|
||||||
@ -452,6 +486,8 @@ public static final String TAB_NAME_QUEUE = "tab_queue";
|
|||||||
public static final String TAB_LABEL_QUEUE = "Queue";
|
public static final String TAB_LABEL_QUEUE = "Queue";
|
||||||
public static final String TAB_NAME_TRACE = "tab_trace";
|
public static final String TAB_NAME_TRACE = "tab_trace";
|
||||||
public static final String TAB_LABEL_TRACE = "Trace";
|
public static final String TAB_LABEL_TRACE = "Trace";
|
||||||
|
public static final String TAB_NAME_SPIROGRAPH = "tab_spirograph";
|
||||||
|
public static final String TAB_LABEL_SPIROGRAPH = "Spirograph";
|
||||||
|
|
||||||
// Page states
|
// Page states
|
||||||
public String currentTab = TAB_NAME_INPUT;
|
public String currentTab = TAB_NAME_INPUT;
|
||||||
@ -462,6 +498,7 @@ public static final String PANEL_NAME_ROVING = "panel_roving";
|
|||||||
public static final String PANEL_NAME_DETAILS = "panel_details";
|
public static final String PANEL_NAME_DETAILS = "panel_details";
|
||||||
public static final String PANEL_NAME_QUEUE = "panel_queue";
|
public static final String PANEL_NAME_QUEUE = "panel_queue";
|
||||||
public static final String PANEL_NAME_TRACE = "panel_trace";
|
public static final String PANEL_NAME_TRACE = "panel_trace";
|
||||||
|
public static final String PANEL_NAME_SPIROGRAPH = "panel_spirograph";
|
||||||
|
|
||||||
public static final String PANEL_NAME_GENERAL = "panel_general";
|
public static final String PANEL_NAME_GENERAL = "panel_general";
|
||||||
|
|
||||||
@ -545,6 +582,17 @@ boolean rescaleDisplayMachine = true;
|
|||||||
int polygonizer = 0;
|
int polygonizer = 0;
|
||||||
float polygonizerLength = 0.0;
|
float polygonizerLength = 0.0;
|
||||||
|
|
||||||
|
// Global variables for Spirograph.
|
||||||
|
int mode = 0; // different spirograph drawing methods
|
||||||
|
int pos_x = 320; // position of spirograph preview on screen
|
||||||
|
int pos_y = 320;
|
||||||
|
int radius = 1; // minor radius
|
||||||
|
int Radius = 171; // major radius
|
||||||
|
int rho = 53; // pen position on minor radius
|
||||||
|
int Interval = 516; // how many lines/segments are used
|
||||||
|
int multiplier = 0; // used in mode 2
|
||||||
|
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
println("Running polargraph controller");
|
println("Running polargraph controller");
|
||||||
@ -608,6 +656,8 @@ void setup()
|
|||||||
useSerialPortConnection = false;
|
useSerialPortConnection = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setupGPIO();
|
||||||
|
|
||||||
currentMode = MODE_BEGIN;
|
currentMode = MODE_BEGIN;
|
||||||
preLoadCommandQueue();
|
preLoadCommandQueue();
|
||||||
changeTab(TAB_NAME_INPUT, TAB_NAME_INPUT);
|
changeTab(TAB_NAME_INPUT, TAB_NAME_INPUT);
|
||||||
@ -669,6 +719,230 @@ void addEventListeners()
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setupGPIO() {
|
||||||
|
/* // Comment block the following out for use on non-Raspberry Pi hardware.
|
||||||
|
gpio = GpioFactory.getInstance();
|
||||||
|
button1 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_00, "Pin1", PinPullResistance.PULL_UP);
|
||||||
|
input1A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_03, "Pin1A", PinPullResistance.PULL_UP);
|
||||||
|
input1B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_02, "Pin1B", PinPullResistance.PULL_UP);
|
||||||
|
button2 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_08, "Pin2", PinPullResistance.PULL_UP);
|
||||||
|
input2A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_09, "Pin2A", PinPullResistance.PULL_UP);
|
||||||
|
input2B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_07, "Pin2B", PinPullResistance.PULL_UP);
|
||||||
|
button3 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_13, "Pin3", PinPullResistance.PULL_UP);
|
||||||
|
input3A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_12, "Pin3A", PinPullResistance.PULL_UP);
|
||||||
|
input3B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_14, "Pin3B", PinPullResistance.PULL_UP);
|
||||||
|
button4 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_23, "Pin4", PinPullResistance.PULL_UP);
|
||||||
|
input4A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_22, "Pin4A", PinPullResistance.PULL_UP);
|
||||||
|
input4B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_21, "Pin4B", PinPullResistance.PULL_UP);
|
||||||
|
button5 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_15, "Pin5", PinPullResistance.PULL_UP);
|
||||||
|
input5A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_16, "Pin5A", PinPullResistance.PULL_UP);
|
||||||
|
input5B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_01, "Pin5B", PinPullResistance.PULL_UP);
|
||||||
|
button6 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_04, "Pin6", PinPullResistance.PULL_UP);
|
||||||
|
input6A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_06, "Pin6A", PinPullResistance.PULL_UP);
|
||||||
|
input6B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_05, "Pin6B", PinPullResistance.PULL_UP);
|
||||||
|
button7 = gpio.provisionDigitalInputPin(RaspiPin.GPIO_29, "Pin7", PinPullResistance.PULL_UP);
|
||||||
|
input7A = gpio.provisionDigitalInputPin(RaspiPin.GPIO_28, "Pin7A", PinPullResistance.PULL_UP);
|
||||||
|
input7B = gpio.provisionDigitalInputPin(RaspiPin.GPIO_27, "Pin7B", PinPullResistance.PULL_UP);
|
||||||
|
|
||||||
|
button1.addListener(new GpioPinListenerDigital() { // Cancel button: clears the queue, lifts the pen and returns to home position.
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button1.getState().getValue();
|
||||||
|
print("button1.getState().getValue() returned ");
|
||||||
|
println(a);
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
resetQueue();
|
||||||
|
//addToCommandQueue(CMD_PENUP + penLiftUpPosition +",END");
|
||||||
|
button_mode_returnToHome();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button2.addListener(new GpioPinListenerDigital() { // Export button: Use these spirgraph settings (move it across to polargraph).
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button2.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
button_mode_exportSpirograph();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button3.addListener(new GpioPinListenerDigital() { // start button
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button3.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
sendVectorShapes();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button4.addListener(new GpioPinListenerDigital() { // pause/resume queue button
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button4.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
commandQueueRunning = !commandQueueRunning;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button5.addListener(new GpioPinListenerDigital() { // set Home
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button5.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
sendSetHomePosition();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button6.addListener(new GpioPinListenerDigital() { // reset position to defualt
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button6.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
PVector newVecPosition = new PVector(0, 0);
|
||||||
|
vectorPosition = newVecPosition;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
button7.addListener(new GpioPinListenerDigital() { // reset scale to default
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = button7.getState().getValue();
|
||||||
|
if (a == 0) { // rising edge
|
||||||
|
cp5.getController(MODE_RESIZE_VECTOR).setValue( 38.6 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input1A.addListener(new GpioPinListenerDigital() {
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input1A.getState().getValue();
|
||||||
|
int b = input1B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
if (Radius + n > 0 && Radius + n < 1000) {
|
||||||
|
cp5.getController(MODE_LIVE_BIG_RADIUS_VALUE).setValue( Radius + n );
|
||||||
|
}
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input2A.addListener(new GpioPinListenerDigital() {
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input2A.getState().getValue();
|
||||||
|
int b = input2B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
if (radius + n > 0 && radius + n < 1000) {
|
||||||
|
cp5.getController(MODE_LIVE_SMALL_RADIUS_VALUE).setValue( radius + n );
|
||||||
|
}
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
input3A.addListener(new GpioPinListenerDigital() {
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input3A.getState().getValue();
|
||||||
|
int b = input3B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
if (rho + n > 0 && rho + n < 1000) {
|
||||||
|
cp5.getController(MODE_LIVE_RHO_RADIUS_VALUE).setValue( rho + n );
|
||||||
|
}
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input4A.addListener(new GpioPinListenerDigital() {
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input4A.getState().getValue();
|
||||||
|
int b = input4B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
if (Interval + n > 0 && Interval + n < 1000) {
|
||||||
|
cp5.getController(MODE_LIVE_INTERVAL_VALUE).setValue( Interval + n );
|
||||||
|
}
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input5A.addListener(new GpioPinListenerDigital() { // X-position of Vectors
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input5A.getState().getValue();
|
||||||
|
int b = input5B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
PVector push = new PVector(n, 0);
|
||||||
|
vectorPosition = PVector.add(vectorPosition, push);
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input6A.addListener(new GpioPinListenerDigital() { // Y-position of Vectors
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input6A.getState().getValue();
|
||||||
|
int b = input6B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
PVector push = new PVector(0, n);
|
||||||
|
vectorPosition = PVector.add(vectorPosition, push);
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
input7A.addListener(new GpioPinListenerDigital() { // Scale of vectors
|
||||||
|
int lastA;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void handleGpioPinDigitalStateChangeEvent(GpioPinDigitalStateChangeEvent arg0) {
|
||||||
|
int a = input7A.getState().getValue();
|
||||||
|
int b = input7B.getState().getValue();
|
||||||
|
if (lastA != a) {
|
||||||
|
int n = (b == a ? -1 : 1);
|
||||||
|
float scale = vectorScaling;
|
||||||
|
if (scale + n > 0 && scale + n < 100) {
|
||||||
|
cp5.getController(MODE_RESIZE_VECTOR).setValue( scale + n );
|
||||||
|
}
|
||||||
|
lastA = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
*/ // end of comment block for RaspPi
|
||||||
|
}
|
||||||
|
|
||||||
void preLoadCommandQueue()
|
void preLoadCommandQueue()
|
||||||
{
|
{
|
||||||
@ -723,6 +997,9 @@ void draw()
|
|||||||
else if (getCurrentTab() == TAB_NAME_TRACE) {
|
else if (getCurrentTab() == TAB_NAME_TRACE) {
|
||||||
drawTracePage();
|
drawTracePage();
|
||||||
}
|
}
|
||||||
|
else if (getCurrentTab() == TAB_NAME_SPIROGRAPH) {
|
||||||
|
drawSpiroPage();
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
drawDetailsPage();
|
drawDetailsPage();
|
||||||
}
|
}
|
||||||
@ -936,6 +1213,281 @@ void drawTracePage()
|
|||||||
// displayGamepadOverlay();
|
// displayGamepadOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void drawSpiroPage() {
|
||||||
|
strokeWeight(1);
|
||||||
|
background(100);
|
||||||
|
noFill();
|
||||||
|
stroke(255, 150, 255, 100);
|
||||||
|
strokeWeight(3);
|
||||||
|
stroke(150);
|
||||||
|
noFill();
|
||||||
|
|
||||||
|
getDisplayMachine().drawForSpiro();//Trace();
|
||||||
|
if (true || retraceShape) // displayingImage && getDisplayMachine().imageIsReady() &&
|
||||||
|
{
|
||||||
|
spirograph();
|
||||||
|
//processedLiveImage = trace_processImageForTrace(getDisplayMachine().getImage());
|
||||||
|
//colourSeparations = trace_buildSeps(processedLiveImage, sepKeyColour);
|
||||||
|
//traceShape = trace_traceImage(colourSeparations);
|
||||||
|
//drawingTraceShape = true;
|
||||||
|
retraceShape = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
stroke(255, 0, 0);
|
||||||
|
|
||||||
|
for (Panel panel : getPanelsForTab(TAB_NAME_SPIROGRAPH))
|
||||||
|
{
|
||||||
|
panel.draw();
|
||||||
|
}
|
||||||
|
text(propertiesFilename, getPanel(PANEL_NAME_GENERAL).getOutline().getLeft(), getPanel(PANEL_NAME_GENERAL).getOutline().getTop()-7);
|
||||||
|
|
||||||
|
|
||||||
|
if (displayingInfoTextOnInputPage)
|
||||||
|
showText(250,45);
|
||||||
|
drawStatusText((int)statusTextPosition.x, (int)statusTextPosition.y);
|
||||||
|
showCommandQueue((int) width-200, 20);
|
||||||
|
|
||||||
|
|
||||||
|
// processGamepadInput();
|
||||||
|
//
|
||||||
|
// if (displayGamepadOverlay)
|
||||||
|
// displayGamepadOverlay();
|
||||||
|
}
|
||||||
|
|
||||||
|
void spirograph() {
|
||||||
|
float theta;
|
||||||
|
int last_x = (Radius-radius)*1+rho*1;
|
||||||
|
if (mode != 0)
|
||||||
|
last_x = (Radius+radius)*1-rho*1;
|
||||||
|
int last_y = 0;
|
||||||
|
int x;
|
||||||
|
int y;
|
||||||
|
if (radius <= 0) radius = 1;
|
||||||
|
if (Radius <= 0) Radius = 1;
|
||||||
|
if (rho <= 0) rho = 1;
|
||||||
|
if (mode == 0) { // hypotrochoid
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius-radius)*cos(theta)+rho*cos((Radius-radius)/radius*theta));
|
||||||
|
y = int((Radius-radius)*sin(theta)+rho*sin((Radius-radius)/radius*theta));
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+x, pos_y+y);
|
||||||
|
last_x = x;
|
||||||
|
last_y = y;
|
||||||
|
}
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+(Radius-radius)*1+rho*1, pos_y);
|
||||||
|
}
|
||||||
|
else if (mode == 1) { // epitrochoid
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius+radius)*cos(theta)-rho*cos((Radius+radius)/radius*theta));
|
||||||
|
y = int((Radius+radius)*sin(theta)-rho*sin((Radius+radius)/radius*theta));
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+x, pos_y+y);
|
||||||
|
last_x = x;
|
||||||
|
last_y = y;
|
||||||
|
}
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+(Radius+radius)*1-rho*1, pos_y);
|
||||||
|
}
|
||||||
|
else if (mode == 2) {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius)*cos(theta)*cos(multiplier*theta)-rho*cos((Radius+radius)/radius*theta));
|
||||||
|
y = int((Radius)*sin(theta)*sin(multiplier*theta)-rho*sin((Radius+radius)/radius*theta));
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+x, pos_y+y);
|
||||||
|
last_x = x;
|
||||||
|
last_y = y;
|
||||||
|
}
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+(Radius+(0 + radius/2 * 1))*1-rho*1, pos_y);
|
||||||
|
}
|
||||||
|
else if (mode == 3){
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius)*cos(theta)*cos(multiplier*theta));
|
||||||
|
y = int((Radius)*sin(theta)*sin(multiplier*theta));
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+x, pos_y+y);
|
||||||
|
last_x = x;
|
||||||
|
last_y = y;
|
||||||
|
}
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+(Radius+(0 + radius/2 * 1))*1-rho*1, pos_y);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius)*(sin(53*theta)*sin(multiplier*theta+PI/4)+sin(53*theta+PI/4)));
|
||||||
|
y = int((Radius)*(sin(53*theta+PI/4)*(sin(53*theta)+sin(53*theta+PI/4))));
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+x, pos_y+y);
|
||||||
|
last_x = x;
|
||||||
|
last_y = y;
|
||||||
|
}
|
||||||
|
line(pos_x+last_x, pos_y+last_y, pos_x+(Radius+(0 + radius/2 * 1))*1-rho*1, pos_y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Export() {
|
||||||
|
// handle the EXPORT button click
|
||||||
|
|
||||||
|
// Open a dialog to choose the location and name for the file
|
||||||
|
File file = new File("output.svg");
|
||||||
|
exportToSVG(file);
|
||||||
|
//exportToSVG("output.svg"); // doesn't work as it's a string, not a File.
|
||||||
|
loadVectorFromSpirograph();
|
||||||
|
//selectOutput("Save the file as a .svg", "exportToSVG");
|
||||||
|
|
||||||
|
// turn off polygonizer if it's on
|
||||||
|
if (polygonizer == 1) {
|
||||||
|
button_mode_cyclePolygonizer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void exportToSVG(File filepath) {
|
||||||
|
// write an SVG file
|
||||||
|
String data[] = new String[7];
|
||||||
|
int lineNo = 0;
|
||||||
|
|
||||||
|
// int Radius = int(RadiusNumbox.getValue());
|
||||||
|
// int radius = int(radiusNumbox.getValue());
|
||||||
|
// int rho = int(rhoNumbox.getValue());
|
||||||
|
// int Interval = int(IntervalNumbox.getValue());
|
||||||
|
float theta;
|
||||||
|
int last_x = (Radius-radius)*1+rho*1;
|
||||||
|
int last_y = 0;
|
||||||
|
float x;
|
||||||
|
float y;
|
||||||
|
if (radius <= 0) radius = 1;
|
||||||
|
if (Radius <= 0) Radius = 1;
|
||||||
|
if (rho <= 0) rho = 1;
|
||||||
|
|
||||||
|
int SVGwidth = 2*(Radius-radius+rho);
|
||||||
|
int SVGheight = 2*(Radius-radius+rho);
|
||||||
|
int SVGx = -SVGwidth / 2;
|
||||||
|
int SVGy = -SVGwidth / 2;
|
||||||
|
|
||||||
|
if (mode == 3) {
|
||||||
|
SVGwidth = 2*Radius;
|
||||||
|
SVGheight = SVGwidth;
|
||||||
|
SVGx = -SVGwidth / 2;
|
||||||
|
SVGy = -SVGwidth / 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Header
|
||||||
|
data[lineNo] = "<?xml version=\"1.0\" standalone=\"no\"?>";
|
||||||
|
lineNo++;
|
||||||
|
data[lineNo] = "<!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\" \"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\">";
|
||||||
|
lineNo++;
|
||||||
|
data[lineNo] = "<!-- Created by Mark Craig (http://www.markcra.com/) -->";
|
||||||
|
lineNo++;
|
||||||
|
// Start of XML
|
||||||
|
data[lineNo] = "<svg xmlns:svg=\"http://www.w3.org/2000/svg\" xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" ";
|
||||||
|
data[lineNo] += "width=\"" + str(SVGwidth);
|
||||||
|
data[lineNo] += "\" height=\"" + str(SVGheight);
|
||||||
|
data[lineNo] += "\" viewBox=\"" + str(SVGx) + " " + str(SVGy) + " " + str(SVGwidth) + " " + str(SVGheight);
|
||||||
|
data[lineNo] += "\" id=\"Layer_1\" xml:space=\"preserve\"><defs id=\"defs8928\" />";
|
||||||
|
lineNo++;
|
||||||
|
|
||||||
|
// hide the settings in plain sight so anyone can recreate the same spirograph from the SVG
|
||||||
|
data[lineNo] = "<!-- Settings Radius: " + str(Radius) + " radius: " + str(radius) + " rho: " + str(rho) + " Interval: " + str(Interval) + " mode: " + str(mode) + "-->";
|
||||||
|
lineNo++;
|
||||||
|
|
||||||
|
// Drawing lines
|
||||||
|
if (mode == 0) {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = (Radius-radius)*cos(theta)+rho*cos((Radius-radius)/radius*theta);
|
||||||
|
y = (Radius-radius)*sin(theta)+rho*sin((Radius-radius)/radius*theta);
|
||||||
|
|
||||||
|
if (i>0) {
|
||||||
|
data[lineNo] += "L "+str(x) + " " + str(y) + " ";
|
||||||
|
}
|
||||||
|
else if (i==0) { // first time through we start the path
|
||||||
|
data[lineNo] = "<path d=\"M "+str(x) + " " + str(y) + " ";
|
||||||
|
//lineNo++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// End Line
|
||||||
|
data[lineNo] += "L " + str((Radius-radius)*1+rho*1) + " " + str(0);
|
||||||
|
}
|
||||||
|
else if (mode == 1) {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = (Radius+radius)*cos(theta)-rho*cos((Radius+radius)/radius*theta);
|
||||||
|
y = (Radius+radius)*sin(theta)-rho*sin((Radius+radius)/radius*theta);
|
||||||
|
|
||||||
|
if (i>0) {
|
||||||
|
data[lineNo] += "L "+str(x) + " " + str(y) + " ";
|
||||||
|
}
|
||||||
|
else if (i==0) { // first time through we start the path
|
||||||
|
data[lineNo] = "<path d=\"M "+str(x) + " " + str(y) + " ";
|
||||||
|
//lineNo++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// End Line
|
||||||
|
data[lineNo] += "L " + str((Radius+radius)*1-rho*1) + " " + str(0);
|
||||||
|
}
|
||||||
|
else if (mode == 2) {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = (Radius)*cos(theta)*cos(multiplier*theta)-rho*cos((Radius+radius)/radius*theta);
|
||||||
|
y = (Radius)*sin(theta)*sin(multiplier*theta)-rho*sin((Radius+radius)/radius*theta);
|
||||||
|
|
||||||
|
if (i>0) {
|
||||||
|
data[lineNo] += "L "+str(x) + " " + str(y) + " ";
|
||||||
|
}
|
||||||
|
else if (i==0) { // first time through we start the path
|
||||||
|
data[lineNo] = "<path d=\"M "+str(x) + " " + str(y) + " ";
|
||||||
|
//lineNo++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// End Line
|
||||||
|
data[lineNo] += "L " + str((Radius+radius)*1-rho*1) + " " + str(0);
|
||||||
|
}
|
||||||
|
else if (mode == 3) {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = (Radius)*cos(theta)*cos(multiplier*theta);
|
||||||
|
y = (Radius)*sin(theta)*sin(multiplier*theta);
|
||||||
|
|
||||||
|
if (i>0) {
|
||||||
|
data[lineNo] += "L "+str(x) + " " + str(y) + " ";
|
||||||
|
}
|
||||||
|
else if (i==0) { // first time through we start the path
|
||||||
|
data[lineNo] = "<path d=\"M "+str(x) + " " + str(y) + " ";
|
||||||
|
//lineNo++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// End Line
|
||||||
|
data[lineNo] += "L " + str((Radius+radius)*1) + " " + str(0);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
for(int i=0; i<Interval; i++) {
|
||||||
|
theta = float(i) / (Interval/2) * PI;
|
||||||
|
x = int((Radius)*(sin(53*theta)*sin(multiplier*theta+PI/4)+sin(53*theta+PI/4)));
|
||||||
|
y = int((Radius)*(sin(53*theta+PI/4)*(sin(53*theta)+sin(53*theta+PI/4))));
|
||||||
|
|
||||||
|
if (i>0) {
|
||||||
|
data[lineNo] += "L "+str(x) + " " + str(y) + " ";
|
||||||
|
}
|
||||||
|
else if (i==0) { // first time through we start the path
|
||||||
|
data[lineNo] = "<path d=\"M "+str(x) + " " + str(y) + " ";
|
||||||
|
//lineNo++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// End Line
|
||||||
|
data[lineNo] += "L " + str((Radius+radius)*1) + " " + str(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
data[lineNo] += "\" id=\"path1\" fill=\"none\" stroke=\"black\" stroke-width=\"0.1\"/>";
|
||||||
|
lineNo++;
|
||||||
|
|
||||||
|
// Wrap up the file
|
||||||
|
data[lineNo] = "</svg>";
|
||||||
|
|
||||||
|
// Save to File
|
||||||
|
// The same file is overwritten by adding the data folder path to saveStrings().
|
||||||
|
saveStrings(filepath, data);
|
||||||
|
println("Export complete.");
|
||||||
|
}
|
||||||
|
|
||||||
void drawCommandQueuePage()
|
void drawCommandQueuePage()
|
||||||
{
|
{
|
||||||
@ -1166,6 +1718,21 @@ void loadVectorWithFileChooser()
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void loadVectorFromSpirograph() {
|
||||||
|
// need a hard coded filename and location to load the spirograph.
|
||||||
|
File file = new File("output.svg"); // doesn't look right to me but can't find any better function.
|
||||||
|
RShape shape = loadShapeFromFile(file.getPath());
|
||||||
|
if (shape != null)
|
||||||
|
{
|
||||||
|
setVectorFilename(file.getPath());
|
||||||
|
setVectorShape(shape);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
println("File not found (" + file.getPath() + ")");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class VectorFileFilter extends javax.swing.filechooser.FileFilter
|
class VectorFileFilter extends javax.swing.filechooser.FileFilter
|
||||||
{
|
{
|
||||||
public boolean accept(File file) {
|
public boolean accept(File file) {
|
||||||
|
12
tabSetup.pde
12
tabSetup.pde
@ -61,11 +61,16 @@ Map<String, Set<Panel>> buildPanelsForTabs()
|
|||||||
queuePanels.add(getPanel(PANEL_NAME_QUEUE));
|
queuePanels.add(getPanel(PANEL_NAME_QUEUE));
|
||||||
queuePanels.add(getPanel(PANEL_NAME_GENERAL));
|
queuePanels.add(getPanel(PANEL_NAME_GENERAL));
|
||||||
|
|
||||||
|
Set<Panel> spirographPanels = new HashSet<Panel>(2);
|
||||||
|
spirographPanels.add(getPanel(PANEL_NAME_SPIROGRAPH));
|
||||||
|
spirographPanels.add(getPanel(PANEL_NAME_GENERAL));
|
||||||
|
|
||||||
map.put(TAB_NAME_INPUT, inputPanels);
|
map.put(TAB_NAME_INPUT, inputPanels);
|
||||||
map.put(TAB_NAME_ROVING, rovingPanels);
|
map.put(TAB_NAME_ROVING, rovingPanels);
|
||||||
map.put(TAB_NAME_TRACE, tracePanels);
|
map.put(TAB_NAME_TRACE, tracePanels);
|
||||||
map.put(TAB_NAME_DETAILS, detailsPanels);
|
map.put(TAB_NAME_DETAILS, detailsPanels);
|
||||||
map.put(TAB_NAME_QUEUE, queuePanels);
|
map.put(TAB_NAME_QUEUE, queuePanels);
|
||||||
|
map.put(TAB_NAME_SPIROGRAPH, spirographPanels);
|
||||||
|
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
@ -78,6 +83,7 @@ List<String> buildTabNames()
|
|||||||
list.add(TAB_NAME_TRACE);
|
list.add(TAB_NAME_TRACE);
|
||||||
list.add(TAB_NAME_DETAILS);
|
list.add(TAB_NAME_DETAILS);
|
||||||
list.add(TAB_NAME_QUEUE);
|
list.add(TAB_NAME_QUEUE);
|
||||||
|
list.add(TAB_NAME_SPIROGRAPH);
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -112,6 +118,11 @@ void initTabs()
|
|||||||
queue.setLabel(TAB_LABEL_QUEUE);
|
queue.setLabel(TAB_LABEL_QUEUE);
|
||||||
queue.activateEvent(true);
|
queue.activateEvent(true);
|
||||||
queue.setId(5);
|
queue.setId(5);
|
||||||
|
|
||||||
|
Tab spirograph = cp5.getTab(TAB_NAME_SPIROGRAPH);
|
||||||
|
spirograph.setLabel(TAB_LABEL_SPIROGRAPH);
|
||||||
|
spirograph.activateEvent(true);
|
||||||
|
spirograph.setId(6);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Set<String> buildPanelNames()
|
public Set<String> buildPanelNames()
|
||||||
@ -123,6 +134,7 @@ public Set<String> buildPanelNames()
|
|||||||
set.add(PANEL_NAME_DETAILS);
|
set.add(PANEL_NAME_DETAILS);
|
||||||
set.add(PANEL_NAME_QUEUE);
|
set.add(PANEL_NAME_QUEUE);
|
||||||
set.add(PANEL_NAME_GENERAL);
|
set.add(PANEL_NAME_GENERAL);
|
||||||
|
set.add(PANEL_NAME_SPIROGRAPH);
|
||||||
return set;
|
return set;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user