mini-cards/app/src/main/java/com/codigoparallevar/minicards/CanvasView.java

482 lines
16 KiB
Java
Raw Normal View History

2017-07-03 18:43:22 +00:00
package com.codigoparallevar.minicards;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
2017-07-03 23:09:37 +00:00
import android.support.annotation.NonNull;
2017-07-03 22:53:12 +00:00
import android.support.annotation.Nullable;
import android.util.AttributeSet;
2017-07-03 21:25:25 +00:00
import android.util.Log;
2017-07-03 21:55:32 +00:00
import android.view.MotionEvent;
2017-07-03 18:43:22 +00:00
import android.view.View;
2017-07-03 23:26:40 +00:00
import com.codigoparallevar.minicards.motion.MotionMode;
2017-07-22 22:35:31 +00:00
import com.codigoparallevar.minicards.types.PartConnection;
2017-07-03 21:25:25 +00:00
import com.codigoparallevar.minicards.parts.buttons.RoundButton;
2017-07-22 20:31:21 +00:00
import com.codigoparallevar.minicards.parts.samples.ColorBox;
import com.codigoparallevar.minicards.parts.samples.Placeholder;
import com.codigoparallevar.minicards.types.connectors.input.InputConnector;
import com.codigoparallevar.minicards.types.connectors.input.SignalInputConnector;
import com.codigoparallevar.minicards.types.connectors.output.OutputConnector;
import com.codigoparallevar.minicards.types.Part;
2017-07-22 20:31:21 +00:00
import com.codigoparallevar.minicards.types.PartGrid;
import com.codigoparallevar.minicards.types.Position;
import com.codigoparallevar.minicards.types.Selectable;
2017-07-22 22:35:31 +00:00
import com.codigoparallevar.minicards.types.Tuple2;
import com.codigoparallevar.minicards.types.Tuple4;
2017-07-03 21:25:25 +00:00
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.ObjectStreamException;
2017-07-03 21:55:32 +00:00
import java.util.ArrayList;
2017-07-22 22:35:31 +00:00
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
2017-07-03 21:25:25 +00:00
2017-07-22 20:31:21 +00:00
class CanvasView extends View implements PartGrid {
2017-07-03 18:43:22 +00:00
2017-07-03 23:09:37 +00:00
@NonNull
2017-07-03 21:55:32 +00:00
ArrayList<Part> parts = new ArrayList<>();
2017-07-03 18:43:22 +00:00
2017-07-03 23:09:37 +00:00
@Nullable
Selectable selectedPart;
2017-07-03 23:09:37 +00:00
2017-07-03 23:26:40 +00:00
@Nullable
final Position lastTouchedPosition = new Position();
@Nullable
long lastTouchedTime;
@Nullable
private MotionMode.Type motionMode;
@NonNull
private String name = "default";
2017-07-03 23:26:40 +00:00
private final static float touchTimeForLongTouchInMillis = 500;
private boolean _isDragging = false;
private MainActivity parentActivity = null;
private Tuple4<Integer, Integer, Integer, Integer> _dropToRemoveZone = new Tuple4<>(0, 0, 0, 0);
private boolean _devMode = false;
2017-07-03 23:26:40 +00:00
2018-01-18 21:15:54 +00:00
@NonNull
private Tuple2<Integer, Integer> _viewOrigin = new Tuple2<>(0, 0);
@Nullable
private Tuple2<Integer, Integer> _mouseDownPoint = null;
public CanvasView(Context context) {
2017-07-03 18:43:22 +00:00
super(context);
init();
}
public CanvasView(Context context, AttributeSet attr){
super(context, attr);
init();
}
private void init() {
2017-07-03 22:21:44 +00:00
this.setBackgroundColor(Color.rgb(4, 69, 99));
if (!loadState()){
// Sprinkle some elements around if no state is found
2017-07-22 20:31:21 +00:00
parts.add(new Placeholder(this, 50, 50, 750, 500));
2017-07-22 22:42:55 +00:00
parts.add(new ColorBox(this, 600, 1000, 700, 1100));
parts.add(new RoundButton(this, 300, 1200, 80, 100));
}
}
private boolean loadState(){
File filesDir = getContext().getFilesDir();
File file = new File(filesDir + "/" + name);
FileReader fileIn = null;
2017-07-22 22:35:31 +00:00
List<PartConnection> connections = new LinkedList<>();
try {
fileIn = new FileReader(file);
char[] data = new char[(int) file.length()];
fileIn.read(data);
JSONObject state = new JSONObject(new String(data));
JSONObject metadata = state.getJSONObject("metadata");
int version = metadata.getInt("version");
if (version != 0) {
return false;
}
// If version 0 -> known structure
JSONArray jsonParts = state.getJSONArray("parts");
for (int i = 0; i < jsonParts.length(); i++){
2017-07-22 22:35:31 +00:00
connections.addAll(deserializeObject(jsonParts.getJSONObject(i)));
}
} catch (IOException e) {
parts.clear();
2017-07-22 22:35:31 +00:00
Log.w("CanvasView", e.getMessage(), e);
return false;
} catch (JSONException e) {
parts.clear();
2017-07-22 22:35:31 +00:00
Log.w("CanvasView", e.getMessage(), e);
return false;
}
2017-07-22 22:35:31 +00:00
resolveConnections(connections);
try {
fileIn.close();
} catch (IOException e) {
Log.w("PartCanvasView", e.getMessage());
return false;
}
2017-07-22 22:35:31 +00:00
return true;
}
2017-07-22 22:35:31 +00:00
private List<PartConnection> deserializeObject(JSONObject jsonObject) throws JSONException {
String type = jsonObject.getString("_type");
2017-07-22 22:35:31 +00:00
if(type.equals(RoundButton.class.getName())) {
2017-07-22 22:35:31 +00:00
Tuple2<Part, List<PartConnection>> buttonInfo = RoundButton.deserialize(
this,
jsonObject.getJSONObject("_data"));
parts.add(buttonInfo.item1);
return buttonInfo.item2;
}
else if (type.equals(Placeholder.class.getName())) {
2017-07-22 20:31:21 +00:00
parts.add(Placeholder.deserialize(this, jsonObject.getJSONObject("_data")));
2017-07-22 22:35:31 +00:00
return Collections.emptyList();
2017-07-22 20:31:21 +00:00
}
else if (type.equals(ColorBox.class.getName())){
parts.add(ColorBox.deserialize(this, jsonObject.getJSONObject("_data")));
2017-07-22 22:35:31 +00:00
return Collections.emptyList();
}
else {
throw new JSONException("Expected known class, found " + type);
}
2017-07-03 18:43:22 +00:00
}
2017-07-22 22:35:31 +00:00
private void resolveConnections(List<PartConnection> connections) {
Map<String, Part> partsById = buildPartsById();
for (PartConnection connection : connections){
if (!partsById.containsKey(connection.inputPartId)){
Log.e("Canvas view", "Key '" + connection.inputPartId
+ "' not found on deserialization");
continue;
}
Part inputPart = partsById.get(connection.inputPartId);
InputConnector inputConnector = inputPart.getConnectorWithId(connection.inputConnectorId);
if (inputConnector == null){
Log.e("Canvas view", "Connector ID '" + connection.inputConnectorId
+ "' not found on deserialization");
continue;
}
OutputConnector outputConnector = connection.outputConnector;
outputConnector.connectTo(inputConnector);
}
}
private Map<String, Part> buildPartsById() {
Map<String, Part> partsById = new HashMap<>(parts.size());
for (Part part : parts) {
partsById.put(part.getId(), part);
Log.w("CanvasView", "Added part ID: " + part.getId() + " - " + part);
}
return partsById;
}
2017-07-03 18:43:22 +00:00
@Override
public void onDraw(Canvas canvas){
2017-07-03 23:09:37 +00:00
final long renderStartTime = System.currentTimeMillis();
2018-01-18 21:15:54 +00:00
ScrolledCanvas scrolledCanvas = new ScrolledCanvas(canvas, _viewOrigin);
drawBackground(scrolledCanvas);
2017-07-03 22:21:44 +00:00
2017-07-03 21:25:25 +00:00
for (Part part : parts){
2018-01-18 21:15:54 +00:00
part.draw(scrolledCanvas, _devMode);
2017-07-03 21:25:25 +00:00
}
2017-07-03 23:09:37 +00:00
Log.d("Render time", System.currentTimeMillis() - renderStartTime + "ms");
2017-07-03 18:43:22 +00:00
}
2017-07-03 21:55:32 +00:00
2018-01-18 21:15:54 +00:00
private void drawBackground(ScrolledCanvas canvas) {
if (!_devMode){
return;
}
2017-07-03 22:21:44 +00:00
// Blueprint background
final int width = getWidth() + getLeft();
final int height = getHeight() + getTop();
final int cellSize = 50;
Paint blueprintLines = new Paint(Paint.ANTI_ALIAS_FLAG);
blueprintLines.setColor(Color.argb(100, 255, 255, 255));
// Vertical lines
for (int x = cellSize; x < width; x += cellSize){
canvas.drawLine(x, 0, x, height, blueprintLines);
}
// Horizontal lines
for (int y = cellSize; y < height; y += cellSize){
canvas.drawLine(0, y, width, y, blueprintLines);
}
}
2017-07-03 21:55:32 +00:00
@Override
public boolean onTouchEvent(MotionEvent event){
2018-01-18 21:15:54 +00:00
final int xInScreen = (int) event.getX() - this.getLeft();
final int yInScreen = (int) event.getY() - this.getTop();
final int xInCanvas = xInScreen + _viewOrigin.item1;
final int yInCanvas = yInScreen + _viewOrigin.item2;
2017-07-03 23:09:37 +00:00
switch (event.getAction()){
case MotionEvent.ACTION_DOWN:
{
2018-01-18 21:15:54 +00:00
_mouseDownPoint = new Tuple2<>(xInScreen, yInScreen);
selectedPart = getPartOn(xInCanvas, yInCanvas);
lastTouchedPosition.to(xInScreen, yInScreen);
2017-07-03 23:26:40 +00:00
lastTouchedTime = System.currentTimeMillis();
2017-07-03 23:09:37 +00:00
if (selectedPart == null) {
Log.d("Touched part", "not found");
return false;
}
Log.d("Touched part", "Part: " + selectedPart);
}
break;
case MotionEvent.ACTION_UP:
{
2017-07-03 23:26:40 +00:00
if (motionMode == null){
motionMode = getMotionMode(false);
}
if (motionMode != MotionMode.Type.LongTouch) {
if (selectedPart != null){
if (selectedPart instanceof Part){
((Part) selectedPart).touched();
}
2017-07-03 23:26:40 +00:00
}
}
2018-01-18 21:30:12 +00:00
else if (motionMode == MotionMode.Type.LongTouch && _devMode) {
if (selectedPart != null) {
2018-01-18 21:15:54 +00:00
selectedPart.getMoveable().drop(xInCanvas, yInCanvas);
2018-01-18 21:15:54 +00:00
if (inDropZone(xInScreen, yInScreen)) {
Log.d("Canvas", "Deleting element" + selectedPart);
parts.remove(selectedPart);
selectedPart.unlink();
}
invalidate();
}
}
2017-07-03 23:26:40 +00:00
_isDragging = false;
2017-07-03 23:26:40 +00:00
motionMode = null;
2017-07-03 23:09:37 +00:00
selectedPart = null;
}
try {
saveState();
} catch (IOException e) {
Log.w("PartCanvasView", e.getMessage());
}
2017-07-03 23:09:37 +00:00
break;
case MotionEvent.ACTION_MOVE:
{
2018-01-18 21:30:12 +00:00
if (!_devMode) {
break;
}
2018-01-18 21:15:54 +00:00
if (selectedPart == null){
int xMovement = _mouseDownPoint.item1 - xInScreen;
int yMovement = _mouseDownPoint.item2 - yInScreen;
_viewOrigin = new Tuple2(
Math.max(_viewOrigin.item1 + xMovement, 0),
Math.max(_viewOrigin.item2 + yMovement, 0));
_mouseDownPoint = new Tuple2(xInScreen, yInScreen);
}
2018-01-18 21:15:54 +00:00
else {
Log.i("Canvas", "X: " + xInScreen + " Y: " + yInScreen
+ " in drop zone " + _dropToRemoveZone + " : "
+ inDropZone(xInScreen, yInScreen));
if (motionMode == null) {
final Selectable nowSelectedPart = getPartOn(xInCanvas, yInCanvas);
final boolean canWait = selectedPart == nowSelectedPart;
motionMode = getMotionMode(canWait);
}
if (motionMode == MotionMode.Type.LongTouch) {
if (selectedPart != null) {
_isDragging = true;
selectedPart.getMoveable().moveEnd(xInCanvas, yInCanvas);
invalidate();
}
2017-07-03 23:26:40 +00:00
}
2017-07-03 23:09:37 +00:00
}
}
break;
2017-07-03 21:55:32 +00:00
2017-07-03 23:09:37 +00:00
default:
{
Log.d("PartCanvasView", "Unhandled action: " + event.getAction());
}
2017-07-03 21:55:32 +00:00
}
invalidate();
if (parentActivity != null) {
parentActivity.invalidate();
}
2017-07-03 21:55:32 +00:00
return true;
}
private boolean inDropZone(int x, int y) {
2017-07-22 20:31:21 +00:00
return (x >= _dropToRemoveZone._x1) && (x <= _dropToRemoveZone._x2)
&& (y >= _dropToRemoveZone._y1) && (y <= _dropToRemoveZone._y2);
}
private void saveState() throws IOException {
File filesDir = getContext().getFilesDir();
FileOutputStream fileOut = new FileOutputStream(filesDir + "/" + name);
byte[] serialized = serializeState();
fileOut.write(serialized);
fileOut.close();
}
private byte[] serializeState() throws IOException {
final JSONArray partArray = new JSONArray();
for (Part part : parts) {
JSONObject serializedPart = new JSONObject();
try {
serializedPart.put("_type", part.getClass().getName());
serializedPart.put("_data", part.serialize());
} catch (JSONException e) {
throw new IOException(e.getMessage(), e.getCause());
}
partArray.put(serializedPart);
}
final JSONObject metadataObject = new JSONObject(new HashMap<String, Object>(){{
put("version", 0);
}});
final JSONObject state = new JSONObject(new HashMap<String, Object>(){{
put("metadata", metadataObject);
put("parts", partArray);
}});
return state.toString().getBytes("UTF-8");
}
2017-07-03 23:26:40 +00:00
@Nullable
private MotionMode.Type getMotionMode(boolean canWait) {
if (selectedPart == null){
return MotionMode.Type.Displacement;
}
if ((System.currentTimeMillis() - lastTouchedTime) >= touchTimeForLongTouchInMillis){
return MotionMode.Type.LongTouch;
}
if (canWait) {
return null;
}
return MotionMode.Type.ShortTouch;
}
2017-07-03 22:53:12 +00:00
@Nullable
2017-07-22 20:31:21 +00:00
public Selectable getPartOn(int x, int y) {
// Look in the list of parts, in reverse so top-most elements are checked before
2017-07-03 21:55:32 +00:00
for (int i = parts.size() - 1; i >= 0; i--){
2017-07-03 23:09:37 +00:00
final Part part = parts.get(i);
if (part.containsPoint(x, y)){
2017-07-03 21:55:32 +00:00
return part;
}
}
// If no part was found, do the same for connectors
for (int i = parts.size() - 1; i >= 0; i--){
final Part part = parts.get(i);
// First try with output connectors
for (OutputConnector outputConnector : part.getOutputConnectors()){
if (outputConnector.containsPoint(x, y)){
return outputConnector;
}
}
// Then with input ones
for (InputConnector inputConnector : part.getSignalInputConnectors()){
if (inputConnector.containsPoint(x, y)){
return inputConnector;
}
}
}
2017-07-03 21:55:32 +00:00
return null;
}
2017-07-09 19:49:52 +00:00
2017-07-22 20:31:21 +00:00
@Override
@Nullable
public SignalInputConnector getSignalInputConnectorOn(int x, int y) {
2017-07-22 20:31:21 +00:00
// If no part was found, do the same for connectors
for (int i = parts.size() - 1; i >= 0; i--){
final Part part = parts.get(i);
// Then with input ones
for (SignalInputConnector inputConnector : part.getSignalInputConnectors()){
2017-07-22 20:31:21 +00:00
if (inputConnector.containsPoint(x, y)){
return inputConnector;
}
}
}
return null;
}
2017-07-09 19:49:52 +00:00
public void addPart(Part part) {
parts.add(part);
invalidate();
}
public boolean isDragging() {
return _isDragging;
}
public void setParentActivity(MainActivity parentActivity) {
this.parentActivity = parentActivity;
}
public void setDropZone(float x1, float x2, float y1, float y2) {
_dropToRemoveZone = new Tuple4<>((int) x1, (int) x2, (int) y1, (int) y2);
}
public void setDevMode(boolean devMode) {
_devMode = devMode;
2018-01-18 21:30:12 +00:00
if (!_devMode) {
2018-01-20 21:08:08 +00:00
CenterView();
2018-01-18 21:30:12 +00:00
}
this.invalidate();
}
2018-01-20 21:08:08 +00:00
public void CenterView() {
this._viewOrigin = new Tuple2<>(0, 0);
this.invalidate();
}
2017-07-03 18:43:22 +00:00
}