|
@@ -9,18 +9,17 @@ using System.Threading.Tasks;
|
|
|
using System.Windows.Forms;
|
|
|
using System.Text.RegularExpressions;
|
|
|
|
|
|
-
|
|
|
// This is the code for your desktop app.
|
|
|
// Press Ctrl+F5 (or go to Debug > Start Without Debugging) to run your app.
|
|
|
|
|
|
namespace SketchAssistant
|
|
|
{
|
|
|
- public partial class Form1 : Form
|
|
|
+ public partial class Form1 : Form, MVP_View
|
|
|
{
|
|
|
public Form1()
|
|
|
{
|
|
|
InitializeComponent();
|
|
|
- fileImporter = new FileImporter(this);
|
|
|
+ ProgramPresenter = new MVP_Presenter(this);
|
|
|
}
|
|
|
|
|
|
/**********************************/
|
|
@@ -29,6 +28,13 @@ namespace SketchAssistant
|
|
|
|
|
|
//important: add new variables only at the end of the list to keep the order of definition consistent with the order in which they are returned by GetAllVariables()
|
|
|
|
|
|
+ public enum ButtonState
|
|
|
+ {
|
|
|
+ Enabled,
|
|
|
+ Disabled,
|
|
|
+ Active
|
|
|
+ }
|
|
|
+
|
|
|
/// <summary>
|
|
|
/// Different Program States
|
|
|
/// </summary>
|
|
@@ -39,85 +45,21 @@ namespace SketchAssistant
|
|
|
Delete
|
|
|
}
|
|
|
/// <summary>
|
|
|
- /// Current Program State
|
|
|
- /// </summary>
|
|
|
- private ProgramState currentState;
|
|
|
- /// <summary>
|
|
|
- /// instance of FileImporter to handle drawing imports
|
|
|
- /// </summary>
|
|
|
- private FileImporter fileImporter;
|
|
|
- /// <summary>
|
|
|
/// Dialog to select a file.
|
|
|
/// </summary>
|
|
|
OpenFileDialog openFileDialog = new OpenFileDialog();
|
|
|
/// <summary>
|
|
|
- /// Image loaded on the left
|
|
|
- /// </summary>
|
|
|
- private Image leftImage = null;
|
|
|
- /// <summary>
|
|
|
- /// the graphic shown in the left window, represented as a list of polylines
|
|
|
- /// </summary>
|
|
|
- private List<Line> leftLineList;
|
|
|
- /// <summary>
|
|
|
- /// Image on the right
|
|
|
- /// </summary>
|
|
|
- Image rightImage = null;
|
|
|
- /// <summary>
|
|
|
- /// Current Line being Drawn
|
|
|
- /// </summary>
|
|
|
- List<Point> currentLine;
|
|
|
- /// <summary>
|
|
|
/// All Lines in the current session
|
|
|
/// </summary>
|
|
|
List<Tuple<bool,Line>> rightLineList = new List<Tuple<bool, Line>>();
|
|
|
/// <summary>
|
|
|
- /// Whether the Mouse is currently pressed in the rightPictureBox
|
|
|
- /// </summary>
|
|
|
- bool mousePressed = false;
|
|
|
- /// <summary>
|
|
|
- /// The Position of the Cursor in the right picture box
|
|
|
- /// </summary>
|
|
|
- Point currentCursorPosition;
|
|
|
- /// <summary>
|
|
|
- /// The Previous Cursor Position in the right picture box
|
|
|
- /// </summary>
|
|
|
- Point previousCursorPosition;
|
|
|
- /// <summary>
|
|
|
/// Queue for the cursorPositions
|
|
|
/// </summary>
|
|
|
Queue<Point> cursorPositions = new Queue<Point>();
|
|
|
/// <summary>
|
|
|
- /// The graphic representation of the right image
|
|
|
- /// </summary>
|
|
|
- Graphics rightGraph = null;
|
|
|
- /// <summary>
|
|
|
- /// Lookup Matrix for checking postions of lines in the image
|
|
|
- /// </summary>
|
|
|
- bool[,] isFilledMatrix;
|
|
|
- /// <summary>
|
|
|
- /// Lookup Matrix for getting line ids at a certain postions of the image
|
|
|
- /// </summary>
|
|
|
- HashSet<int>[,] linesMatrix;
|
|
|
- /// <summary>
|
|
|
- /// Size of deletion area
|
|
|
- /// </summary>
|
|
|
- int deletionRadius = 2;
|
|
|
- /// <summary>
|
|
|
- /// History of Actions
|
|
|
- /// </summary>
|
|
|
- ActionHistory historyOfActions;
|
|
|
- /// <summary>
|
|
|
- /// List of items which will be overlayed over the right canvas.
|
|
|
+ /// The Presenter Component of the MVP-Model
|
|
|
/// </summary>
|
|
|
- List<HashSet<Point>> overlayItems;
|
|
|
- /// <summary>
|
|
|
- /// The assistant responsible for the redraw mode
|
|
|
- /// </summary>
|
|
|
- RedrawAssistant redrawAss;
|
|
|
- /// <summary>
|
|
|
- /// Size of areas marking endpoints of lines in the redraw mode.
|
|
|
- /// </summary>
|
|
|
- int markerRadius = 10;
|
|
|
+ MVP_Presenter ProgramPresenter;
|
|
|
|
|
|
/******************************************/
|
|
|
/*** FORM SPECIFIC FUNCTIONS START HERE ***/
|
|
@@ -125,11 +67,7 @@ namespace SketchAssistant
|
|
|
|
|
|
private void Form1_Load(object sender, EventArgs e)
|
|
|
{
|
|
|
- currentState = ProgramState.Idle;
|
|
|
this.DoubleBuffered = true;
|
|
|
- historyOfActions = new ActionHistory(null);
|
|
|
- redrawAss = new RedrawAssistant();
|
|
|
- UpdateButtonStatus();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -137,23 +75,9 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void Form1_Resize(object sender, System.EventArgs e)
|
|
|
{
|
|
|
+ ProgramPresenter.Resize(new Tuple<int, int>(pictureBoxLeft.Width, pictureBoxLeft.Height),
|
|
|
+ new Tuple<int, int>(pictureBoxRight.Width, pictureBoxRight.Height));
|
|
|
this.Refresh();
|
|
|
- UpdateSizes();
|
|
|
- }
|
|
|
-
|
|
|
- //Load button, will open an OpenFileDialog
|
|
|
- private void loadToolStripMenuItem_Click(object sender, EventArgs e)
|
|
|
- {
|
|
|
- openFileDialog.Filter = "Image|*.jpg;*.png;*.jpeg";
|
|
|
- if(openFileDialog.ShowDialog() == DialogResult.OK)
|
|
|
- {
|
|
|
- toolStripLoadStatus.Text = openFileDialog.SafeFileName;
|
|
|
- leftImage = Image.FromFile(openFileDialog.FileName);
|
|
|
- pictureBoxLeft.Image = leftImage;
|
|
|
- //Refresh the left image box when the content is changed
|
|
|
- this.Refresh();
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -161,38 +85,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void examplePictureToolStripMenuItem_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if (CheckSavedStatus())
|
|
|
- {
|
|
|
- openFileDialog.Filter = "Interactive Sketch-Assistant Drawing|*.isad";
|
|
|
- if (openFileDialog.ShowDialog() == DialogResult.OK)
|
|
|
- {
|
|
|
- toolStripLoadStatus.Text = openFileDialog.SafeFileName;
|
|
|
- try
|
|
|
- {
|
|
|
- (int, int, List<Line>) values = fileImporter.ParseISADInputFile(openFileDialog.FileName);
|
|
|
- DrawEmptyCanvasLeft(values.Item1, values.Item2);
|
|
|
- BindAndDrawLeftImage(values.Item3);
|
|
|
-
|
|
|
- //Match The right canvas to the left
|
|
|
- historyOfActions = new ActionHistory(lastActionTakenLabel);
|
|
|
- DrawEmptyCanvasRight();
|
|
|
- isFilledMatrix = new bool[rightImage.Width, rightImage.Height];
|
|
|
- linesMatrix = new HashSet<int>[rightImage.Width, rightImage.Height];
|
|
|
- rightLineList = new List<Tuple<bool, Line>>();
|
|
|
- //Start the redraw mode
|
|
|
- redrawAss = new RedrawAssistant(leftLineList);
|
|
|
- UpdateSizes();
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, -1, false);
|
|
|
- RedrawRightImage();
|
|
|
- this.Refresh();
|
|
|
- }
|
|
|
- catch (FileImporterException ex)
|
|
|
- {
|
|
|
- ShowInfoMessage(ex.ToString());
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.ExamplePictureToolStripMenuItemClick();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -200,18 +93,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void drawButton_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if(rightImage != null)
|
|
|
- {
|
|
|
- if (currentState.Equals(ProgramState.Draw))
|
|
|
- {
|
|
|
- ChangeState(ProgramState.Idle);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- ChangeState(ProgramState.Draw);
|
|
|
- }
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.ChangeState(true);
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -219,18 +101,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void deleteButton_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if (rightImage != null)
|
|
|
- {
|
|
|
- if (currentState.Equals(ProgramState.Delete))
|
|
|
- {
|
|
|
- ChangeState(ProgramState.Idle);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- ChangeState(ProgramState.Delete);
|
|
|
- }
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.ChangeState(false);
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -238,28 +109,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void undoButton_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if (historyOfActions.CanUndo())
|
|
|
- {
|
|
|
- HashSet<int> affectedLines = historyOfActions.GetCurrentAction().GetLineIDs();
|
|
|
- SketchAction.ActionType undoAction = historyOfActions.GetCurrentAction().GetActionType();
|
|
|
- switch (undoAction)
|
|
|
- {
|
|
|
- case SketchAction.ActionType.Delete:
|
|
|
- //Deleted Lines need to be shown
|
|
|
- ChangeLines(affectedLines, true);
|
|
|
- break;
|
|
|
- case SketchAction.ActionType.Draw:
|
|
|
- //Drawn lines need to be hidden
|
|
|
- ChangeLines(affectedLines, false);
|
|
|
- break;
|
|
|
- default:
|
|
|
- break;
|
|
|
- }
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, -1, false);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- historyOfActions.MoveAction(true);
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.Undo();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -267,28 +117,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void redoButton_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if (historyOfActions.CanRedo())
|
|
|
- {
|
|
|
- historyOfActions.MoveAction(false);
|
|
|
- HashSet<int> affectedLines = historyOfActions.GetCurrentAction().GetLineIDs();
|
|
|
- SketchAction.ActionType redoAction = historyOfActions.GetCurrentAction().GetActionType();
|
|
|
- switch (redoAction)
|
|
|
- {
|
|
|
- case SketchAction.ActionType.Delete:
|
|
|
- //Deleted Lines need to be redeleted
|
|
|
- ChangeLines(affectedLines, false);
|
|
|
- break;
|
|
|
- case SketchAction.ActionType.Draw:
|
|
|
- //Drawn lines need to be redrawn
|
|
|
- ChangeLines(affectedLines, true);
|
|
|
- break;
|
|
|
- default:
|
|
|
- break;
|
|
|
- }
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, -1, false);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.Redo();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -298,20 +127,28 @@ namespace SketchAssistant
|
|
|
{
|
|
|
if (e.Modifiers == Keys.Control && e.KeyCode == Keys.Z)
|
|
|
{
|
|
|
- undoButton_Click(sender, e);
|
|
|
+ ProgramPresenter.Undo();
|
|
|
}
|
|
|
if (e.Modifiers == Keys.Control && e.KeyCode == Keys.Y)
|
|
|
{
|
|
|
- redoButton_Click(sender, e);
|
|
|
+ ProgramPresenter.Redo();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// <summary>
|
|
|
+ /// The Picture box is clicked.
|
|
|
+ /// </summary>
|
|
|
+ private void pictureBoxRight_Click(object sender, EventArgs e)
|
|
|
+ {
|
|
|
+ ProgramPresenter.MouseEvent(MVP_Presenter.MouseAction.Click);
|
|
|
+ }
|
|
|
+
|
|
|
/// <summary>
|
|
|
/// Get current Mouse positon within the right picture box.
|
|
|
/// </summary>
|
|
|
private void pictureBoxRight_MouseMove(object sender, MouseEventArgs e)
|
|
|
{
|
|
|
- currentCursorPosition = ConvertCoordinates(new Point(e.X, e.Y));
|
|
|
+ ProgramPresenter.MouseEvent(MVP_Presenter.MouseAction.Move, e);
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -319,11 +156,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void pictureBoxRight_MouseDown(object sender, MouseEventArgs e)
|
|
|
{
|
|
|
- mousePressed = true;
|
|
|
- if (currentState.Equals(ProgramState.Draw))
|
|
|
- {
|
|
|
- currentLine = new List<Point>();
|
|
|
- }
|
|
|
+ ProgramPresenter.MouseEvent(MVP_Presenter.MouseAction.Down);
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -331,18 +164,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void pictureBoxRight_MouseUp(object sender, MouseEventArgs e)
|
|
|
{
|
|
|
- mousePressed = false;
|
|
|
- if (currentState.Equals(ProgramState.Draw) && currentLine.Count > 0)
|
|
|
- {
|
|
|
- Line newLine = new Line(currentLine, rightLineList.Count);
|
|
|
- rightLineList.Add(new Tuple<bool, Line>(true, newLine));
|
|
|
- newLine.PopulateMatrixes(isFilledMatrix, linesMatrix);
|
|
|
- historyOfActions.AddNewAction(new SketchAction(SketchAction.ActionType.Draw, newLine.GetID()));
|
|
|
- //Execute a RedrawAssistant tick with the currently finished Line
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, newLine.GetID(), true);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
+ ProgramPresenter.MouseEvent(MVP_Presenter.MouseAction.Up);
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -352,25 +174,7 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void canvasButton_Click(object sender, EventArgs e)
|
|
|
{
|
|
|
- if (CheckSavedStatus())
|
|
|
- {
|
|
|
- historyOfActions = new ActionHistory(lastActionTakenLabel);
|
|
|
- DrawEmptyCanvasRight();
|
|
|
- //The following lines cannot be in DrawEmptyCanvas()
|
|
|
- isFilledMatrix = new bool[rightImage.Width, rightImage.Height];
|
|
|
- linesMatrix = new HashSet<int>[rightImage.Width, rightImage.Height];
|
|
|
- rightLineList = new List<Tuple<bool, Line>>();
|
|
|
- //Reinitialise the Redraw Assistant.
|
|
|
- if(leftLineList != null)
|
|
|
- {
|
|
|
- redrawAss = new RedrawAssistant(leftLineList);
|
|
|
- UpdateSizes();
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, -1, false);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- }
|
|
|
- UpdateButtonStatus();
|
|
|
- UpdateSizes();
|
|
|
+ ProgramPresenter.NewCanvas();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
@@ -379,424 +183,141 @@ namespace SketchAssistant
|
|
|
/// </summary>
|
|
|
private void mouseTimer_Tick(object sender, EventArgs e)
|
|
|
{
|
|
|
- if(cursorPositions.Count > 0) { previousCursorPosition = cursorPositions.Dequeue(); }
|
|
|
- else { previousCursorPosition = currentCursorPosition; }
|
|
|
- cursorPositions.Enqueue(currentCursorPosition);
|
|
|
- //Drawing
|
|
|
- if (currentState.Equals(ProgramState.Draw) && mousePressed)
|
|
|
- {
|
|
|
- rightGraph = Graphics.FromImage(rightImage);
|
|
|
- currentLine.Add(currentCursorPosition);
|
|
|
- Line drawline = new Line(currentLine);
|
|
|
- drawline.DrawLine(rightGraph);
|
|
|
- pictureBoxRight.Image = rightImage;
|
|
|
- //Redraw overlay gets ticked
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, rightLineList.Count, false);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- //Deleting
|
|
|
- if (currentState.Equals(ProgramState.Delete) && mousePressed)
|
|
|
- {
|
|
|
- List<Point> uncheckedPoints = GeometryCalculator.BresenhamLineAlgorithm(previousCursorPosition, currentCursorPosition);
|
|
|
- foreach (Point currPoint in uncheckedPoints)
|
|
|
- {
|
|
|
- HashSet<int> linesToDelete = CheckDeletionMatrixesAroundPoint(currPoint, deletionRadius);
|
|
|
- if (linesToDelete.Count > 0)
|
|
|
- {
|
|
|
- historyOfActions.AddNewAction(new SketchAction(SketchAction.ActionType.Delete, linesToDelete));
|
|
|
- foreach (int lineID in linesToDelete)
|
|
|
- {
|
|
|
- rightLineList[lineID] = new Tuple<bool, Line>(false, rightLineList[lineID].Item2);
|
|
|
- }
|
|
|
- RepopulateDeletionMatrixes();
|
|
|
- //Redraw overlay gets ticked
|
|
|
- overlayItems = redrawAss.Tick(currentCursorPosition, rightLineList, -1, false);
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
+ ProgramPresenter.Tick();
|
|
|
}
|
|
|
|
|
|
- /***********************************/
|
|
|
- /*** HELPER FUNCTIONS START HERE ***/
|
|
|
- /***********************************/
|
|
|
+ /*************************/
|
|
|
+ /*** PRESENTER -> VIEW ***/
|
|
|
+ /*************************/
|
|
|
|
|
|
/// <summary>
|
|
|
- /// A function that returns a white canvas for a given width and height.
|
|
|
+ /// Enables the timer of the View, which will tick the Presenter.
|
|
|
/// </summary>
|
|
|
- /// <param name="width">The width of the canvas in pixels</param>
|
|
|
- /// <param name="height">The height of the canvas in pixels</param>
|
|
|
- /// <returns>The new canvas</returns>
|
|
|
- private Image GetEmptyCanvas(int width, int height)
|
|
|
+ public void EnableTimer()
|
|
|
{
|
|
|
- Image image;
|
|
|
- try
|
|
|
- {
|
|
|
- image = new Bitmap(width, height);
|
|
|
- }
|
|
|
- catch(ArgumentException e)
|
|
|
- {
|
|
|
- ShowInfoMessage("The requested canvas size caused an error: \n" + e.ToString() + "\n The Canvas will be set to match your window.");
|
|
|
- image = new Bitmap(pictureBoxLeft.Width, pictureBoxLeft.Height);
|
|
|
- }
|
|
|
- Graphics graph = Graphics.FromImage(image);
|
|
|
- graph.FillRectangle(Brushes.White, 0, 0, width + 10, height + 10);
|
|
|
- return image;
|
|
|
+ mouseTimer.Enabled = true;
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// Creates an empty Canvas
|
|
|
+ /// A function that opens a file dialog and returns the filename.
|
|
|
/// </summary>
|
|
|
- private void DrawEmptyCanvasRight()
|
|
|
+ /// <param name="Filter">The filter that should be applied to the new Dialog.</param>
|
|
|
+ /// <returns>Returns the FileName and the SafeFileName if the user correctly selects a file,
|
|
|
+ /// else returns a tuple with empty strigns</returns>
|
|
|
+ public Tuple<String, String> openNewDialog(String Filter)
|
|
|
{
|
|
|
- if (leftImage == null)
|
|
|
+ openFileDialog.Filter = Filter;
|
|
|
+ if (openFileDialog.ShowDialog() == DialogResult.OK)
|
|
|
{
|
|
|
- SetAndRefreshRightImage(GetEmptyCanvas(pictureBoxRight.Width, pictureBoxRight.Height));
|
|
|
+ return new Tuple<string, string>(openFileDialog.FileName, openFileDialog.SafeFileName);
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
- SetAndRefreshRightImage(GetEmptyCanvas(leftImage.Width, leftImage.Height));
|
|
|
+ return new Tuple<string, string>("", "");
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// Creates an empty Canvas on the left
|
|
|
+ /// Sets the contents of the load status indicator label.
|
|
|
/// </summary>
|
|
|
- /// <param name="width"> width of the new canvas in pixels </param>
|
|
|
- /// <param name="height"> height of the new canvas in pixels </param>
|
|
|
- private void DrawEmptyCanvasLeft(int width, int height)
|
|
|
+ /// <param name="message">The new contents</param>
|
|
|
+ public void SetToolStripLoadStatus(String message)
|
|
|
{
|
|
|
- if (width == 0)
|
|
|
- {
|
|
|
- SetAndRefreshLeftImage(GetEmptyCanvas(pictureBoxLeft.Width, pictureBoxLeft.Height));
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- SetAndRefreshLeftImage(GetEmptyCanvas(width, height));
|
|
|
- }
|
|
|
+ toolStripLoadStatus.Text = message;
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// Redraws all lines in lineList, for which their associated boolean value equals true.
|
|
|
+ /// Sets the contents of the last action taken indicator label.
|
|
|
/// </summary>
|
|
|
- private void RedrawRightImage()
|
|
|
+ /// <param name="message">The new contents</param>
|
|
|
+ public void SetLastActionTakenText(String message)
|
|
|
{
|
|
|
- var workingCanvas = GetEmptyCanvas(rightImage.Width, rightImage.Height);
|
|
|
- var workingGraph = Graphics.FromImage(workingCanvas);
|
|
|
- //Lines
|
|
|
- foreach (Tuple<bool, Line> lineBoolTuple in rightLineList)
|
|
|
- {
|
|
|
- if (lineBoolTuple.Item1)
|
|
|
- {
|
|
|
- lineBoolTuple.Item2.DrawLine(workingGraph);
|
|
|
- }
|
|
|
- }
|
|
|
- //The Line being currently drawn
|
|
|
- if(currentLine != null && currentLine.Count > 0 && currentState.Equals(ProgramState.Draw) && mousePressed)
|
|
|
- {
|
|
|
- var currLine = new Line(currentLine);
|
|
|
- currLine.DrawLine(workingGraph);
|
|
|
- }
|
|
|
- //Overlay Items
|
|
|
- foreach (HashSet<Point> item in overlayItems)
|
|
|
- {
|
|
|
- foreach(Point p in item)
|
|
|
- {
|
|
|
- workingGraph.FillRectangle(Brushes.Green, p.X, p.Y, 1, 1);
|
|
|
- }
|
|
|
- }
|
|
|
- SetAndRefreshRightImage(workingCanvas);
|
|
|
+ lastActionTakenLabel.Text = message;
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// A function to set rightImage and to refresh the respective PictureBox with it.
|
|
|
+ /// Changes the states of a tool strip button.
|
|
|
/// </summary>
|
|
|
- /// <param name="image">The new Image</param>
|
|
|
- private void SetAndRefreshRightImage(Image image)
|
|
|
+ /// <param name="buttonName">The name of the button.</param>
|
|
|
+ /// <param name="state">The new state of the button.</param>
|
|
|
+ public void SetToolStripButtonStatus(String buttonName, ButtonState state)
|
|
|
{
|
|
|
- rightImage = image;
|
|
|
- pictureBoxRight.Image = rightImage;
|
|
|
- pictureBoxRight.Refresh();
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// A function to set leftImage and to refresh the respective PictureBox with it.
|
|
|
- /// </summary>
|
|
|
- /// <param name="image">The new Image</param>
|
|
|
- private void SetAndRefreshLeftImage(Image image)
|
|
|
- {
|
|
|
- leftImage = image;
|
|
|
- pictureBoxLeft.Image = leftImage;
|
|
|
- pictureBoxLeft.Refresh();
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// Change the status of whether or not the lines are shown.
|
|
|
- /// </summary>
|
|
|
- /// <param name="lines">The HashSet containing the affected Line IDs.</param>
|
|
|
- /// <param name="shown">True if the lines should be shown, false if they should be hidden.</param>
|
|
|
- private void ChangeLines(HashSet<int> lines, bool shown)
|
|
|
- {
|
|
|
- foreach (int lineId in lines)
|
|
|
- {
|
|
|
- if (lineId <= rightLineList.Count - 1 && lineId >= 0)
|
|
|
- {
|
|
|
- rightLineList[lineId] = new Tuple<bool, Line>(shown, rightLineList[lineId].Item2);
|
|
|
- }
|
|
|
- }
|
|
|
- RedrawRightImage();
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// Updates the active status of buttons. Currently draw, delete, undo and redo button.
|
|
|
- /// </summary>
|
|
|
- private void UpdateButtonStatus()
|
|
|
- {
|
|
|
- undoButton.Enabled = historyOfActions.CanUndo();
|
|
|
- redoButton.Enabled = historyOfActions.CanRedo();
|
|
|
- drawButton.Enabled = (rightImage != null);
|
|
|
- deleteButton.Enabled = (rightImage != null);
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// A helper function which handles tasks associated witch changing states,
|
|
|
- /// such as checking and unchecking buttons and changing the state.
|
|
|
- /// </summary>
|
|
|
- /// <param name="newState">The new state of the program</param>
|
|
|
- private void ChangeState(ProgramState newState)
|
|
|
- {
|
|
|
- switch (currentState)
|
|
|
+ ToolStripButton buttonToChange;
|
|
|
+ switch (buttonName)
|
|
|
{
|
|
|
- case ProgramState.Draw:
|
|
|
- drawButton.CheckState = CheckState.Unchecked;
|
|
|
- mouseTimer.Enabled = false;
|
|
|
+ case "canvasButton":
|
|
|
+ buttonToChange = canvasButton;
|
|
|
break;
|
|
|
- case ProgramState.Delete:
|
|
|
- deleteButton.CheckState = CheckState.Unchecked;
|
|
|
- mouseTimer.Enabled = false;
|
|
|
+ case "drawButton":
|
|
|
+ buttonToChange = drawButton;
|
|
|
break;
|
|
|
- default:
|
|
|
+ case "deleteButton":
|
|
|
+ buttonToChange = deleteButton;
|
|
|
break;
|
|
|
- }
|
|
|
- switch (newState)
|
|
|
- {
|
|
|
- case ProgramState.Draw:
|
|
|
- drawButton.CheckState = CheckState.Checked;
|
|
|
- mouseTimer.Enabled = true;
|
|
|
+ case "undoButton":
|
|
|
+ buttonToChange = undoButton;
|
|
|
break;
|
|
|
- case ProgramState.Delete:
|
|
|
- deleteButton.CheckState = CheckState.Checked;
|
|
|
- mouseTimer.Enabled = true;
|
|
|
+ case "redoButton":
|
|
|
+ buttonToChange = redoButton;
|
|
|
break;
|
|
|
default:
|
|
|
- break;
|
|
|
+ Console.WriteLine("Invalid Button was given to SetToolStripButton. \nMaybe you forgot to add a case?");
|
|
|
+ return;
|
|
|
}
|
|
|
- currentState = newState;
|
|
|
- pictureBoxRight.Refresh();
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// A function that calculates the coordinates of a point on a zoomed in image.
|
|
|
- /// </summary>
|
|
|
- /// <param name="">The position of the mouse cursor</param>
|
|
|
- /// <returns>The real coordinates of the mouse cursor on the image</returns>
|
|
|
- private Point ConvertCoordinates(Point cursorPosition)
|
|
|
- {
|
|
|
- Point realCoordinates = new Point(5,3);
|
|
|
- if(pictureBoxRight.Image == null)
|
|
|
+ switch (state)
|
|
|
{
|
|
|
- return cursorPosition;
|
|
|
- }
|
|
|
-
|
|
|
- int widthImage = pictureBoxRight.Image.Width;
|
|
|
- int heightImage = pictureBoxRight.Image.Height;
|
|
|
- int widthBox = pictureBoxRight.Width;
|
|
|
- int heightBox = pictureBoxRight.Height;
|
|
|
-
|
|
|
- float imageRatio = (float)widthImage / (float)heightImage;
|
|
|
- float containerRatio = (float)widthBox / (float)heightBox;
|
|
|
-
|
|
|
- if (imageRatio >= containerRatio)
|
|
|
- {
|
|
|
- //Image is wider than it is high
|
|
|
- float zoomFactor = (float)widthImage / (float)widthBox;
|
|
|
- float scaledHeight = heightImage / zoomFactor;
|
|
|
- float filler = (heightBox - scaledHeight) / 2;
|
|
|
- realCoordinates.X = (int)(cursorPosition.X * zoomFactor);
|
|
|
- realCoordinates.Y = (int)((cursorPosition.Y - filler) * zoomFactor);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- //Image is higher than it is wide
|
|
|
- float zoomFactor = (float)heightImage / (float)heightBox;
|
|
|
- float scaledWidth = widthImage / zoomFactor;
|
|
|
- float filler = (widthBox - scaledWidth) / 2;
|
|
|
- realCoordinates.X = (int)((cursorPosition.X - filler) * zoomFactor);
|
|
|
- realCoordinates.Y = (int)(cursorPosition.Y * zoomFactor);
|
|
|
- }
|
|
|
- return realCoordinates;
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// A function that populates the matrixes needed for deletion detection with line data.
|
|
|
- /// </summary>
|
|
|
- private void RepopulateDeletionMatrixes()
|
|
|
- {
|
|
|
- if(rightImage != null)
|
|
|
- {
|
|
|
- isFilledMatrix = new bool[rightImage.Width,rightImage.Height];
|
|
|
- linesMatrix = new HashSet<int>[rightImage.Width, rightImage.Height];
|
|
|
- foreach(Tuple<bool,Line> lineTuple in rightLineList)
|
|
|
- {
|
|
|
- if (lineTuple.Item1)
|
|
|
- {
|
|
|
- lineTuple.Item2.PopulateMatrixes(isFilledMatrix, linesMatrix);
|
|
|
- }
|
|
|
- }
|
|
|
+ case ButtonState.Active:
|
|
|
+ buttonToChange.Checked = true;
|
|
|
+ break;
|
|
|
+ case ButtonState.Disabled:
|
|
|
+ buttonToChange.Checked = false;
|
|
|
+ buttonToChange.Enabled = false;
|
|
|
+ break;
|
|
|
+ case ButtonState.Enabled:
|
|
|
+ buttonToChange.Checked = false;
|
|
|
+ buttonToChange.Enabled = true;
|
|
|
+ break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// A function that checks the deletion matrixes at a certain point
|
|
|
- /// and returns all Line ids at that point and in a square around it in a certain range.
|
|
|
+ /// Displays an image in the left Picture box.
|
|
|
/// </summary>
|
|
|
- /// <param name="p">The point around which to check.</param>
|
|
|
- /// <param name="range">The range around the point. If range is 0, only the point is checked.</param>
|
|
|
- /// <returns>A List of all lines.</returns>
|
|
|
- private HashSet<int> CheckDeletionMatrixesAroundPoint(Point p, int range)
|
|
|
+ /// <param name="img">The new image.</param>
|
|
|
+ public void DisplayInLeftPictureBox(Image img)
|
|
|
{
|
|
|
- HashSet<int> returnSet = new HashSet<int>();
|
|
|
-
|
|
|
- foreach(Point pnt in GeometryCalculator.FilledCircleAlgorithm(p, (int)range))
|
|
|
- {
|
|
|
- if(pnt.X >= 0 && pnt.Y >= 0 && pnt.X < rightImage.Width && pnt.Y < rightImage.Height)
|
|
|
- {
|
|
|
- if (isFilledMatrix[pnt.X, pnt.Y])
|
|
|
- {
|
|
|
- returnSet.UnionWith(linesMatrix[pnt.X, pnt.Y]);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- return returnSet;
|
|
|
+ pictureBoxLeft.Image = img;
|
|
|
+ pictureBoxLeft.Refresh();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// binds the given picture to templatePicture and draws it
|
|
|
+ /// Displays an image in the right Picture box.
|
|
|
/// </summary>
|
|
|
- /// <param name="newTemplatePicture"> the new template picture, represented as a list of polylines </param>
|
|
|
- /// <returns></returns>
|
|
|
- private void BindAndDrawLeftImage(List<Line> newTemplatePicture)
|
|
|
+ /// <param name="img">The new image.</param>
|
|
|
+ public void DisplayInRightPictureBox(Image img)
|
|
|
{
|
|
|
- leftLineList = newTemplatePicture;
|
|
|
- foreach(Line l in leftLineList)
|
|
|
- {
|
|
|
- l.DrawLine(Graphics.FromImage(leftImage));
|
|
|
- }
|
|
|
+ pictureBoxRight.Image = img;
|
|
|
+ pictureBoxRight.Refresh();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// shows the given info message in a popup and asks the user to aknowledge it
|
|
|
/// </summary>
|
|
|
/// <param name="message">the message to show</param>
|
|
|
- private void ShowInfoMessage(String message)
|
|
|
+ public void ShowInfoMessage(String message)
|
|
|
{
|
|
|
MessageBox.Show(message);
|
|
|
}
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// Will calculate the start and endpoints of the given line on the right canvas.
|
|
|
- /// </summary>
|
|
|
- /// <param name="line">The line.</param>
|
|
|
- /// <param name="size">The size of the circle with which the endpoints of the line are marked.</param>
|
|
|
- private Tuple<HashSet<Point>, HashSet<Point>> CalculateStartAndEnd(Line line, int size)
|
|
|
- {
|
|
|
- var circle0 = GeometryCalculator.FilledCircleAlgorithm(line.GetStartPoint(), size);
|
|
|
- var circle1 = GeometryCalculator.FilledCircleAlgorithm(line.GetEndPoint(), size);
|
|
|
- var currentLineEndings = new Tuple<HashSet<Point>, HashSet<Point>>(circle0, circle1);
|
|
|
- return currentLineEndings;
|
|
|
- }
|
|
|
|
|
|
/// <summary>
|
|
|
- /// A helper Function that updates the markerRadius & deletionRadius, considering the size of the canvas.
|
|
|
+ /// Shows a warning box with the given message (Yes/No Buttons)and returns true if the user aknowledges it.
|
|
|
/// </summary>
|
|
|
- private void UpdateSizes()
|
|
|
+ /// <param name="message">The message of the warning.</param>
|
|
|
+ /// <returns>True if the user confirms (Yes), negative if he doesn't (No)</returns>
|
|
|
+ public bool ShowWarning(String message)
|
|
|
{
|
|
|
- if (rightImage != null)
|
|
|
- {
|
|
|
- int widthImage = pictureBoxRight.Image.Width;
|
|
|
- int heightImage = pictureBoxRight.Image.Height;
|
|
|
- int widthBox = pictureBoxRight.Width;
|
|
|
- int heightBox = pictureBoxRight.Height;
|
|
|
-
|
|
|
- float imageRatio = (float)widthImage / (float)heightImage;
|
|
|
- float containerRatio = (float)widthBox / (float)heightBox;
|
|
|
- float zoomFactor = 0;
|
|
|
- if (imageRatio >= containerRatio)
|
|
|
- {
|
|
|
- //Image is wider than it is high
|
|
|
- zoomFactor = (float)widthImage / (float)widthBox;
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- //Image is higher than it is wide
|
|
|
- zoomFactor = (float)heightImage / (float)heightBox;
|
|
|
- }
|
|
|
- markerRadius = (int)(10 * zoomFactor);
|
|
|
- redrawAss.SetMarkerRadius(markerRadius);
|
|
|
- deletionRadius = (int)(5 * zoomFactor);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// Checks if there is unsaved progess, and warns the user. Returns True if it safe to continue.
|
|
|
- /// </summary>
|
|
|
- /// <returns>true if there is none, or the user wishes to continue without saving.
|
|
|
- /// false if there is progress, and the user doesn't wish to continue.</returns>
|
|
|
- private bool CheckSavedStatus()
|
|
|
- {
|
|
|
- if (!historyOfActions.IsEmpty())
|
|
|
- {
|
|
|
- return (MessageBox.Show("You have unsaved changes, do you wish to continue?",
|
|
|
- "Attention", MessageBoxButtons.YesNo, MessageBoxIcon.Warning) == DialogResult.Yes);
|
|
|
- }
|
|
|
- return true;
|
|
|
- }
|
|
|
-
|
|
|
- /********************************************/
|
|
|
- /*** TESTING RELATED FUNCTIONS START HERE ***/
|
|
|
- /********************************************/
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// returns all instance variables in the order of their definition for testing
|
|
|
- /// </summary>
|
|
|
- /// <returns>A list of tuples containing names of variables and the variable themselves.
|
|
|
- /// Cast according to the Type definitions in the class variable section.</returns>
|
|
|
- public List<Tuple<String, Object>>GetAllVariables()
|
|
|
- {
|
|
|
- var objArr = new (String, object)[] { ("currentState", currentState), ("fileImporter", fileImporter), ("openFileDialog", openFileDialog),
|
|
|
- ("leftImage", leftImage), ("leftLineList", leftLineList), ("rightImage", rightImage), ("currentLine", currentLine),
|
|
|
- ("rightLineList", rightLineList), ("mousePressed", mousePressed), ("currentCursorPosition", currentCursorPosition),
|
|
|
- ("previousCursorPosition", previousCursorPosition), ("cursorPositions", cursorPositions), ("rightGraph", rightGraph),
|
|
|
- ("isFilledMatrix", isFilledMatrix), ("linesMatrix", linesMatrix), ("deletionRadius", deletionRadius),
|
|
|
- ("historyOfActions", historyOfActions), ("overlayItems", overlayItems), ("redrawAss", redrawAss), ("markerRadius", markerRadius) };
|
|
|
- var varArr = new List<Tuple<String, Object>>();
|
|
|
- foreach((String, object) obj in objArr)
|
|
|
- {
|
|
|
- varArr.Add(new Tuple<string, object>(obj.Item1, obj.Item2));
|
|
|
- }
|
|
|
- return varArr;
|
|
|
+ return (MessageBox.Show(message, "Warning", MessageBoxButtons.YesNo, MessageBoxIcon.Warning) == DialogResult.Yes);
|
|
|
}
|
|
|
|
|
|
- /// <summary>
|
|
|
- /// public method wrapper for testing purposes, invoking DrawEmptyCanvas(...) and BindAndDrawLeftImage(...)
|
|
|
- /// </summary>
|
|
|
- /// <param name="width">width of the parsed image</param>
|
|
|
- /// <param name="height">height of the parsed image</param>
|
|
|
- /// <param name="newImage">the parsed image</param>
|
|
|
- public void CreateCanvasAndSetPictureForTesting(int width, int height, List<Line> newImage)
|
|
|
- {
|
|
|
- DrawEmptyCanvasLeft(width, height);
|
|
|
- BindAndDrawLeftImage(newImage);
|
|
|
- }
|
|
|
}
|
|
|
}
|