Skip to content
OE_interfaceDisplay.cpp 27.5 KiB
Newer Older
/*
 * Copyright (c) 2015 Tricoire Sebastien 3dsman@free.fr
 *
 * This software is provided 'as-is', without any express or implied
 * warranty.  In no event will the authors be held liable for any damages
 * arising from the use of this software.
 *
 * Permission is granted to anyone to use this software for any purpose,
 * including commercial applications, and to alter it and redistribute it
 * freely, subject to the following restrictions:
 *
 * 1. The origin of this software must not be misrepresented; you must not
 * claim that you wrote the original software. If you use this software
 * in a product, an acknowledgment in the product documentation would be
 * appreciated but is not required.
 * 2. Altered source versions must be plainly marked as such, and must not be
 * misrepresented as being the original software.
 * 3. This notice may not be removed or altered from any source distribution.
 *
 */

#include "OE_interfaceDisplay.h"
#include "OE_utils.h"

#include "OE_controller.h"

#include <iostream>
raoul's avatar
raoul committed
#include <qevent.h>
#include <GL/gl.h>
#include <cstdlib>
#include <math.h>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <QFileDialog>
#include "actions/OE_actionsCurves.h"
#include "actions/OE_actionsStitchs.h"
#include "actions/OE_actionsLineStitchs.h"
#include "actions/OE_actionsBirailStitchs.h"
3dsman's avatar
3dsman committed
#include "actions/OE_actionsFillStitchs.h"
#include "actions/OE_actionsSelection.h"
#include "actions/OE_actionsThreads.h"

raoul's avatar
raoul committed
static const GLuint TOOLSSELID = 0;
static const GLuint STITCHSSELID  = 1;
static const GLuint CURVESSELID = 2;
raoul's avatar
raoul committed
static const GLuint ZEROID  = 0;
static const GLuint TRANSTOOLID = 1;
static const GLuint MOVETOOLID = 0;
static const GLuint SCALETOOLID = 1;
raoul's avatar
raoul committed
OE_interfaceDisplay::OE_interfaceDisplay(QWidget *parent, Qt::WindowFlags f) : OE_display(parent, f)
	editStyle = new OE_display::OE_displayStyle();
	commandStyle = new OE_display::OE_displayStyle();
	commandStyle->drawGrid = false;
	commandStyle->drawCurves = false;
	commandStyle->drawStitches = false;
	commandStyle->drawCommands = true;
raoul's avatar
raoul committed
	setDocument(nullptr);
	modState = Qt::KeyboardModifier::NoModifier;
}

OE_interfaceDisplay::~OE_interfaceDisplay()
{
raoul's avatar
raoul committed
bool OE_interfaceDisplay::setDocument(OE_document* document)
raoul's avatar
raoul committed
	return true;
OE_document* OE_interfaceDisplay::getDocument()
{
    return document;
}

OE_controller* OE_interfaceDisplay::getController()
{
    return controller;
}


bool OE_interfaceDisplay::setController(OE_controller* controller)
{
raoul's avatar
raoul committed
	return true;
raoul's avatar
raoul committed
bool OE_interfaceDisplay::mouse_Pos(double x, double y)
raoul's avatar
raoul committed
	bool redraw = false;
	redraw |= OE_display::mouse_Pos(x, y);
	vector_2dt oldClosestPoint = editClosestPoint;
raoul's avatar
raoul committed
	vector_2d absMove = vector_2d((mouse.x - clicOldMouse.x)*zoom*2, (mouse.y - clicOldMouse.y)*zoom*2);
raoul's avatar
raoul committed
	if (editionState == MoveSelection) //if the move action is enabled
raoul's avatar
raoul committed
		controller->editActionMoveSelection(vector_2d((mouse.x - clicOldMouse.x)*zoom*2,
		                                              (mouse.y - clicOldMouse.y)*zoom*2));
raoul's avatar
raoul committed
		return true;
raoul's avatar
raoul committed
	if (editionState == ScaleSelection) //if the scale action is enabled
	{
		vector_2d bbCenter = controller->getSelectionBoundingBox().getCenter();
		float scale = (absMouse-bbCenter).len()/(screenToDocument(clicOldMouse)-bbCenter).len();
        controller->editActionScaleSelection(vector_2d(scale,scale));
        emit refreshStitchList();
raoul's avatar
raoul committed
		return true;
	if (editionState == NewSubcurve) //if we must find the closest subcurve (get the closest point on it too)
	{
		closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
raoul's avatar
raoul committed
		redraw = true;
	}
	else if (editSubcurve) //if we are working on a defined subcurve get the closest point on it
raoul's avatar
raoul committed
	{
		editClosestPoint = controller->getClosestPoint(editSubcurve->getCurve(), absMouse);
raoul's avatar
raoul committed
		redraw = true;
raoul's avatar
raoul committed
	}
	if (editionState == MovePointcurve && selControlCurve && selControlType==0) //if we're moving a curve control point
raoul's avatar
raoul committed
		controller->editActionMovePointCurve(vector_2d((mouse.x - clicOldMouse.x)*zoom*2,
		                                               (mouse.y - clicOldMouse.y)*zoom*2));
raoul's avatar
raoul committed
		return true;
	if (editionState == MovePointstitch) //if we're moving a linkstitch control point
	{
		controller->editActionMovePointLinkStitch(absMove);
        controller->editActionMoveGridPointFillStitch(absMove);
        emit refreshStitchList();
        return true;

	//if we're editing the scaleWidth of a linestitch
	if (editionState == OffsetWidthLinestitch)
raoul's avatar
raoul committed
	{
		controller->editActionSetLinestitchWidth((mouse.x - clicOldMouse.x)/100.0f, true);
raoul's avatar
raoul committed
		return true;
	//if we're define a new subcurve and the reference curve is already picked
	if ((editionState == NewSubcurve)&&(selControlStitch))
	{
		editionState = TraceSubcurve;
		if (oldClosestPoint.t>editClosestPoint.t) controller->editActionAddSubcurvePosSwitchDir();
        redraw = true;
        emit refreshStitchList();
	}

	if (editionState == TraceSubcurve)
	{
        controller->editActionAddSubcurvePosEnd(editClosestPoint.t);
        redraw = true;
        emit refreshStitchList();
	}
	else if (editionState == MoveSubcurve)
	{
		controller->editActionSetSubcurvePos(editClosestPoint.t);
        redraw = true;
        emit refreshStitchList();
    }

raoul's avatar
raoul committed
	return redraw;
raoul's avatar
raoul committed
bool OE_interfaceDisplay::mouse_Button(QMouseEvent* event)
raoul's avatar
raoul committed
	bool redraw = false;
	redraw |= OE_display::mouse_Button(event);
	if (curScreen == edit && event->button() == Qt::LeftButton && event->type() == QEvent::MouseButtonPress)
raoul's avatar
raoul committed
		redraw = true;
raoul's avatar
raoul committed
		if (editionState == NewSubcurve)
		{
			if (!closestCurve)
raoul's avatar
raoul committed
			else
			{
				controller->addAction(new OE_actionJoincurveAddSubCurve(document, curJoincurve, subcurve_id, closestCurve,
raoul's avatar
raoul committed
				                                                        editClosestPoint.t, editClosestPoint.t,
				                                                        false));
				//subcurve_id = -1;
raoul's avatar
raoul committed
				selControlStitch = controller->getStitch(-1);
				selControlType = 0;
				//editSubcurve = curJoincurve->getCurve(-1);
				editSubcurve = curJoincurve->getCurve(subcurve_id);
raoul's avatar
raoul committed
			}
		}
		else
		{
			clicOldMouse = mouse;
raoul's avatar
raoul committed
			if (editionState == None)
			{
raoul's avatar
raoul committed
			}
			if (editionState == None)
			{
raoul's avatar
raoul committed
			}
		}
raoul's avatar
raoul committed
		return redraw;
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	if (curScreen == edit && event->button() == Qt::LeftButton && event->type() == QEvent::MouseButtonRelease)
raoul's avatar
raoul committed
		redraw = true;
		selControlCurve = nullptr;
		selControlStitch = nullptr;
		if (editionState == AddSubcurve)
		{
			//curJoincurve = controller->getLineStitch(-1)->getJoincurve();
			//closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
			editionState = NewSubcurve;
			joincurve_id = 0;
raoul's avatar
raoul committed
			return true;
raoul's avatar
raoul committed
		if (editionState == Selection)
		{
			if ((mouse-clicOldMouse).len()>5)
raoul's avatar
raoul committed
			{
raoul's avatar
raoul committed
			}
			selectApply(selectedCurves, selectedStitches);
			editionState = None;
		}
		if ((editionState == NewSubcurve)||(editionState == TraceSubcurve))
raoul's avatar
raoul committed
		{
			if (joincurve_id == 0)
			{
				editionState = None;
				editSubcurve = nullptr;
			}
			else
			{
				editionState = NewSubcurve;
				curJoincurve = controller->getBirailStitch(-1)->getJoincurve2();
				closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
				joincurve_id -= 1;
			}
raoul's avatar
raoul committed
			return true;
raoul's avatar
raoul committed
		return redraw;
	return redraw;
}

void OE_interfaceDisplay::scroll(double xoffset, double yoffset)
{
	OE_display::scroll(xoffset, yoffset);
}

void OE_interfaceDisplay::resize(int width, int height)
{
	OE_display::resize(width, height);
raoul's avatar
raoul committed
void OE_interfaceDisplay::key(QKeyEvent* event)
	if (event->type() == QEvent::KeyPress)
		modState = event->modifiers();
	else
		modState = Qt::KeyboardModifier::NoModifier;
raoul's avatar
raoul committed
	if (event->type() == QEvent::KeyPress && curScreen == edit)
raoul's avatar
raoul committed
	{
		if (event->key() == Qt::Key_R && event->modifiers() == Qt::NoModifier)
raoul's avatar
raoul committed
		{
			if (editionState == TraceSubcurve)
raoul's avatar
raoul committed
			{
				controller->editActionAddSubcurvePosSwitchDir();
                emit refreshStitchList();
raoul's avatar
raoul committed
			}
			else if (editionState == MoveSubcurve)
raoul's avatar
raoul committed
			{
				controller->editActionSetSubcurvePosSwitchDir();
                update();
                emit refreshStitchList();
raoul's avatar
raoul committed
			}
		}
raoul's avatar
raoul committed
		{
			if (event->key() == Qt::Key_Z && event->modifiers() == Qt::ControlModifier)
raoul's avatar
raoul committed
			{
				if (controller)
				{
                    emit refreshStitchList();
raoul's avatar
raoul committed
				}
			}
			else if (event->key() == Qt::Key_S && event->modifiers() == Qt::ControlModifier)
raoul's avatar
raoul committed
			{
					QString fileName = QFileDialog::getSaveFileName(this, "Save design", "", "OE designer (*.oe)");
					if (!fileName.isEmpty())
					{
						document->saveToFile(fileName.toStdString());
					}
raoul's avatar
raoul committed
				}
			else if (event->key() == Qt::Key_O && event->modifiers() == Qt::ControlModifier)
				QString type;
				QString fileName = QFileDialog::getOpenFileName(this, "Open design", "", "All supported (*.oe *.svg *.pes);;OE designer (*.oe);;SVG drawing (*.svg);;PES (*.pes)", &type);
				if (!fileName.isEmpty())
					loadFile(fileName.toStdString());
					update();
                    emit refreshStitchList();
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_Z && event->modifiers() == (Qt::ControlModifier|Qt::ShiftModifier))
				if (controller)
				{
					controller->redoAction();
                    emit refreshStitchList();
raoul's avatar
raoul committed
			}
			else if (event->modifiers() == Qt::NoModifier)
				if (event->key() == Qt::Key_C)
						controller->toggleCloseSelectedCurve();
                        emit refreshStitchList();
raoul's avatar
raoul committed
				else if (event->key() == Qt::Key_Delete)
				{
					//TODO create an action to delete all selected stitches
                    std::list<OE_stitchs*>::iterator it=document->selectedStitchs.begin();
					controller->addAction(new OE_actionDelStitch(document, *it));
                    emit refreshStitchList();
raoul's avatar
raoul committed
				else if (event->key() == Qt::Key_Right)
				{
					std::list<OE_stitchs*>::iterator it=selectedStitches.begin();
					if (it != selectedStitches.end())
					{
raoul's avatar
raoul committed
						it = std::find (document->stitchs.begin(), document->stitchs.end(), *it);
						it++;
						if (it != document->stitchs.end())
						{
							selectedStitches.clear();
							selectedStitches.push_back(*it);
							controller->clearSelection();
                            controller->selectStitches(selectedStitches,true);
                            emit stitchSelectionChange();
raoul's avatar
raoul committed
				else if (event->key() == Qt::Key_Left)
					std::list<OE_stitchs*>::iterator it=selectedStitches.begin();
					if (it != selectedStitches.end())
raoul's avatar
raoul committed
						it = std::find (document->stitchs.begin(), document->stitchs.end(), *it);
						if (it != document->stitchs.begin())
						{
							it--;
							selectedStitches.clear();
							selectedStitches.push_back(*it);
							controller->clearSelection();
                            controller->selectStitches(selectedStitches,true);
                            emit stitchSelectionChange();
raoul's avatar
raoul committed
	if (event->type() == QEvent::KeyRelease)
		if (event->key() == Qt::Key_R)
				{
					if (editionState == TraceSubcurve)
					{
						controller->editActionAddSubcurvePosSwitchDir();
                        update();
                        emit refreshStitchList();
					}
					else if (editionState == MoveSubcurve)
					{
						controller->editActionSetSubcurvePosSwitchDir();
                        emit refreshStitchList();
					}
				}
		if(editionState==None)
raoul's avatar
raoul committed
		{
raoul's avatar
raoul committed
			if (curScreen == command && event->key() == Qt::Key_Space)
				run=true;
raoul's avatar
raoul committed
			}
raoul's avatar
raoul committed
			if (event->key() == Qt::Key_F12)
				if (curScreen == edit)
				{
					curScreen = command;
					setDisplayStyle(commandStyle);
				}
				else
				{
					curScreen = edit;
					setDisplayStyle(editStyle);
			else if (event->key() == Qt::Key_L)
raoul's avatar
raoul committed
			{
				if (controller)
				{
					controller->newLineStitch(2.5, 2.5, controller->getPattern(0));
					document->selectedStitchs.push_back(controller->getLineStitch(-1));
					curJoincurve = controller->getLineStitch(-1)->getJoincurve();
					closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
raoul's avatar
raoul committed

					editionState = NewSubcurve;
					joincurve_id = 0;
					subcurve_id = -1;
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_B)
raoul's avatar
raoul committed
			{
				if (controller)
				{
					controller->newBirailStitch(false, false, 0, 0, 0.7f);
					document->selectedStitchs.push_back(controller->getBirailStitch(-1));
					curJoincurve = controller->getBirailStitch(-1)->getJoincurve1();
					closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
raoul's avatar
raoul committed

					editionState = NewSubcurve;
					joincurve_id = 1;
					subcurve_id = -1;
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_F)
3dsman's avatar
3dsman committed
			{
				if (controller)
				{
					controller->newFillStitch(1.5f, 0.3f, controller->getPattern(0));
					document->selectedStitchs.push_back(controller->getFillStitch(-1));
					curJoincurve = controller->getFillStitch(-1)->getJoincurve();
					closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
3dsman's avatar
3dsman committed

					editionState = NewSubcurve;
					joincurve_id = 0;
					subcurve_id = -1;
3dsman's avatar
3dsman committed
			}
raoul's avatar
raoul committed
			/*else if (event->key() == Qt::Key_Enter)
			{
				if (editionState == NewSubcurve)
				{
					if (joincurve_id == 0)
					{
						editionState = None;
						curJoincurve = 0;
					}
					else
					{
						curJoincurve = controller->getBirailStitch(-1)->getJoincurve2();
						closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
						joincurve_id -= 1;
					}
				}
			}*/

			else if (event->key() == Qt::Key_Plus && event->modifiers() & Qt::KeypadModifier)
raoul's avatar
raoul committed
			{
				if (event->modifiers() & Qt::ControlModifier)
raoul's avatar
raoul committed
				{
					controller->addAction(new OE_actionScaleSelectedStitchLen(document, 1.1f));
                    emit refreshStitchList();
raoul's avatar
raoul committed
				}
				else
				{
					controller->addAction(new OE_actionScaleSelectedStitchWidth(document, 1.1f));
raoul's avatar
raoul committed
				}
			}
			else if (event->key() == Qt::Key_Minus && event->modifiers() & Qt::KeypadModifier)
raoul's avatar
raoul committed
			{
				if (event->modifiers() & Qt::ControlModifier)
					controller->addAction(new OE_actionScaleSelectedStitchLen(document, 1.f/1.1f));
                    emit refreshStitchList();
					controller->addAction(new OE_actionScaleSelectedStitchWidth(document, 1.f/1.1f));
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_1 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(0);
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_2 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(1);
			else if (event->key() == Qt::Key_3 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(2);
raoul's avatar
raoul committed
			}
			else if (event->key() == Qt::Key_4 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(3);
			else if (event->key() == Qt::Key_5 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(4);
			else if (event->key() == Qt::Key_6 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(5);
			else if (event->key() == Qt::Key_7 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(6);
			else if (event->key() == Qt::Key_8 && event->modifiers() & Qt::KeypadModifier)
            {
                SetSelectedStitchPattern(7);
			else if (event->key() == Qt::Key_9 && event->modifiers() & Qt::KeypadModifier)
                SetSelectedStitchPattern(8);
raoul's avatar
raoul committed
			}
raoul's avatar
raoul committed
			if (event->key() == Qt::Key_Home)
			{
				zoomSelection();
raoul's avatar
raoul committed
	}
3dsman's avatar
3dsman committed

void OE_interfaceDisplay::SetSelectedStitchPattern(int index)
{
    if (!controller->getPattern(index))
        return;
    controller->addAction(new OE_actionSetSelectedStitchPattern(document, controller->getPattern(index)));
    update();
    emit refreshStitchList();
}

void OE_interfaceDisplay::selectBox()
3dsman's avatar
3dsman committed
{
raoul's avatar
raoul committed
	selectedCurves.clear();
	selectedStitches.clear();
	vector_2d clicAbsMouse = viewPos+(clicOldMouse-vector_2d(width,height)/2)*zoom*2;
3dsman's avatar
3dsman committed

raoul's avatar
raoul committed
	std::list<OE_pointcurve*>::iterator curve = document->curves.begin();
	while (curve != document->curves.end())
	{
		BoundingBox box = (*curve)->getBound();
		if ((box.getMin().x > std::min(clicAbsMouse.x, absMouse.x))
		     &&(box.getMax().x < std::max(clicAbsMouse.x, absMouse.x))
		     &&(box.getMin().y > std::min(clicAbsMouse.y, absMouse.y))
		     &&(box.getMax().y < std::max(clicAbsMouse.y, absMouse.y)))
raoul's avatar
raoul committed
		{
			selectedCurves.push_back(*curve);
		}
		curve++;
	}
3dsman's avatar
3dsman committed

raoul's avatar
raoul committed
	std::list<OE_stitchs*>::iterator stitch = document->stitchs.begin();
	while (stitch != document->stitchs.end())
	{
raoul's avatar
raoul committed
		BoundingBox box = (*stitch)->getBound();
		if ((box.getMin().x > std::min(clicAbsMouse.x, absMouse.x))
		     &&(box.getMax().x < std::max(clicAbsMouse.x, absMouse.x))
		     &&(box.getMin().y > std::min(clicAbsMouse.y, absMouse.y))
		     &&(box.getMax().y < std::max(clicAbsMouse.y, absMouse.y)))
raoul's avatar
raoul committed
		{
			selectedStitches.push_back(*stitch);
		}
		stitch++;
	}
    emit stitchSelectionChange();
bool OE_interfaceDisplay::checkJoinCurveControl(OE_joincurve* joinCurve, int &index)
{
	if (selControlType == index)
	{
		// get the corresponding subcurve
		editSubcurve = joinCurve->getCurve(selControlIndex/2);
		if (editSubcurve)
		{
			editionState = MoveSubcurve;
			// get the closest point t value
			vector_2dt pos = editSubcurve->getCurve()->closestPoint(absMouse);
			// and add an action to move the point
			controller->addAction(new OE_actionSetSubcurvePos(document, editSubcurve, selControlIndex%2, pos.t));
			return true;
		}
		return false;
	}
	index++;
	if (selControlType == index)
	{
		editionState = AddSubcurve;
		subcurve_id = selControlIndex;
		curJoincurve = joinCurve;
		return true;
	}
	index++;
	return false;
}

bool OE_interfaceDisplay::checkPicking()
{
	int ControlTypeIndex = 0;
raoul's avatar
raoul committed
	if (toolAction)
raoul's avatar
raoul committed
		if (toolAction == 1)
			controller->addAction(new OE_actionMoveSelection(document, vector_2d(0,0)));
			return true;
raoul's avatar
raoul committed
		if (toolAction == 2)
			controller->addAction(new OE_actionScaleSelection(document, vector_2d(1,1), controller->getSelectionBoundingBox().getCenter()));
			return true;
	// if we picked a stitch control
raoul's avatar
raoul committed
	if (selControlStitch)
raoul's avatar
raoul committed
		OE_linestitch* tmpLineStitch = dynamic_cast<OE_linestitch*>(selControlStitch);
			// check generic joinCurve controls
            if (checkJoinCurveControl(tmpLineStitch->getJoincurve(), ControlTypeIndex)) return true;
			//check picking for linestitch width controller
			if (selControlType == ControlTypeIndex && selControlIndex == 0)
				editionState = OffsetWidthLinestitch;
				controller->addAction(new OE_actionSetLineStitchWidth(document, tmpLineStitch, tmpLineStitch->getWidth()));
raoul's avatar
raoul committed
		OE_birailstitch* tmpBirailStitch = dynamic_cast<OE_birailstitch*>(selControlStitch);
		if (tmpBirailStitch)
raoul's avatar
raoul committed
		{
			// check generic joinCurve controls for each birail's joincurve
			if (checkJoinCurveControl(tmpBirailStitch->getJoincurve1(), ControlTypeIndex)) return true;
			if (checkJoinCurveControl(tmpBirailStitch->getJoincurve2(), ControlTypeIndex)) return true;
raoul's avatar
raoul committed
			return false;
		}
		OE_fillstitch* tmpFillStitch = dynamic_cast<OE_fillstitch*>(selControlStitch);

		if (tmpFillStitch)
			// check generic joinCurve controls
			if (checkJoinCurveControl(tmpFillStitch->getJoincurve(), ControlTypeIndex)) return true;

			// check picking for grid's controller points
			else if (selControlType == ControlTypeIndex)
3dsman's avatar
3dsman committed
				editionState = MovePointstitch;
				//edit gridpoint
				controller->addAction(new OE_actionMovePointFillStitch(document, tmpFillStitch, selControlIndex, vector_2d(0,0)));
		OE_linkstitch* tmpLinkStitch = dynamic_cast<OE_linkstitch*>(selControlStitch);

		if (tmpLinkStitch)
			//check picking of control point, add or delete
			if (selControlType == 0)
			{
				editionState = MovePointstitch;
				controller->addAction(new OE_actionMovePointLinkStitch(document, tmpLinkStitch, selControlIndex, vector_2d()));
raoul's avatar
raoul committed
			}
			else if (selControlType == 1)
				controller->addAction(new OE_actionAddPointLinkStitch(document, tmpLinkStitch, selControlIndex, absMouse));
raoul's avatar
raoul committed
			}
			else if (selControlType == 2)
			{
				editionState = DelPointstitch;
				controller->addAction(new OE_actionDelPointLinkStitch(document, tmpLinkStitch, selControlIndex));
		selControlStitch = nullptr;
raoul's avatar
raoul committed
	if (selControlCurve)
raoul's avatar
raoul committed
		OE_pointcurve* tmpPointCurve = dynamic_cast<OE_pointcurve*>(selControlCurve);
raoul's avatar
raoul committed
		if (tmpPointCurve && selControlType == 0)
		{
			editionState = MovePointcurve;
			controller->addAction(new OE_actionMovePointCurve(document, tmpPointCurve,
			                                                  selControlIndex+1,
			                                                  vector_2d(0,0)));
			return true;
raoul's avatar
raoul committed
		}
		selControlCurve = nullptr;
	return false;
/** \brief set selection
 *
 * \return true if all is ok
 *
 */
bool OE_interfaceDisplay::drawPicking()
	GLuint selectBuf[64];
	glSelectBuffer(64, selectBuf);
	OE_display::pick(document->selectedCurves, document->selectedStitchs);

	//get the hits of the picking
	//hits are represented by an array of GLuint*
	// first value give us the number of items on the name stack at the time of the hit
	// next 2 values are deep value of the picked object (we don't use it)
	// then the items we need arrived
	// we format it like that:
	// - type of the picked object (tool, stitch or curve)
	// - index of the picked object (list index for stitch or curve, index of the tool for the tools)
	// - if we picked a sub element of a curve, stich or tool here come his type (handle, startpoint,...)
	// - then his index
	//
	// so a hit can be defined by
	// 2 values in case a curve of a stitch was picked directly
	// 3 values in case of picking a transform tool like move
	// or 4 values if we picked a sub element of a curve or stitch

	GLint hits = glRenderMode(GL_RENDER);
	selectedCurves.clear();
	selectedStitches.clear();
	GLuint *ptr;
	//std::cout<<"hits : "<<"\r\n";
	ptr = static_cast<GLuint*>(selectBuf);
	for (GLint i=0; i<hits; i++)  //  for each hit
		ptr+=3; //skip nearZ and farZ info (we don't use it) and goto first index
		if (*ptr == TOOLSSELID) //tools picking
		{
			ptr++;
raoul's avatar
raoul committed
			if (*ptr == ZEROID) //zero
raoul's avatar
raoul committed
			if (*ptr == TRANSTOOLID) //one of the transform selection tools
raoul's avatar
raoul committed
				if (*ptr == MOVETOOLID) //move selection tools
				{
					toolAction = 1;
					checkPicking();
					return true;
				}
raoul's avatar
raoul committed
				if (*ptr == SCALETOOLID) //scale selection tools
				{
					toolAction = 2;
					checkPicking();
					return true;
				}
raoul's avatar
raoul committed
		}
		else if (*ptr == STITCHSSELID) //stitch picking
		{
			ptr++;
			if (names==2) //if name count correspond to an object (we picked a stitch, not one of his sub element)
raoul's avatar
raoul committed
				if (*ptr >= document->stitchs.size())
				{
					return false;
				}
				std::list<OE_stitchs*>::iterator it = document->stitchs.begin();
				std::advance(it, *ptr);
                selectedStitches.push_back(*it);
raoul's avatar
raoul committed
			}
			else if (names==4) //if name count correspond to a stitch sub element (like control point)
raoul's avatar
raoul committed
				if (*ptr >= document->selectedStitchs.size())
				{
					return false;
				}

				std::list<OE_stitchs*>::iterator it = document->selectedStitchs.begin();
				std::advance(it, *ptr++);
				selControlStitch = *it;
				selControlType = *ptr++;
				selControlIndex = *ptr++;

				checkPicking();
				return true;
raoul's avatar
raoul committed
		}
		else if (*ptr == CURVESSELID) //curves picking
		{
			ptr++;
			if (names==2) //if name count correspond to an object (we picked a curve, not one of his sub element)
			{
raoul's avatar
raoul committed
				if (*ptr >= document->curves.size())
				{
					return false;
				}

				std::list<OE_pointcurve*>::iterator it = document->curves.begin();
				std::advance(it, *ptr);
				selectedCurves.push_back(*it);
				return true;
raoul's avatar
raoul committed
			}
			else if (names==4) //if name count correspond to a sub element of curve (like control point)
raoul's avatar
raoul committed
				if (*ptr >= document->selectedCurves.size())
				{
					return false;
				}
				//in *ptr we got "curve id" then "control type" and at last "control id"
				std::list<OE_pointcurve*>::iterator it = document->selectedCurves.begin();
				std::advance(it, *ptr++);
				selControlCurve = *it;
				selControlType = *ptr++;
				selControlIndex = *ptr++;

				checkPicking();

				return true;
			}
bool OE_interfaceDisplay::selectApply(std::list<OE_pointcurve*> selectedCurves, std::list<OE_stitchs*> selectedStitches)
raoul's avatar
raoul committed
	if (selectedCurves.size())
	{
raoul's avatar
raoul committed
		if (modState.testFlag(Qt::ControlModifier))
raoul's avatar
raoul committed
		{
			controller->unselectCurves(selectedCurves);
		}
		else
		{
raoul's avatar
raoul committed
			controller->selectCurves(selectedCurves, !modState.testFlag(Qt::ShiftModifier));
raoul's avatar
raoul committed
		}
	}

	if (selectedStitches.size())
	{
raoul's avatar
raoul committed
		if (modState.testFlag(Qt::ControlModifier))
raoul's avatar
raoul committed
		{
			controller->unselectStitches(selectedStitches);
		}
		else
		{
            controller->selectStitches(selectedStitches, !modState.testFlag(Qt::ShiftModifier));
            emit stitchSelectionChange();
raoul's avatar
raoul committed
		}
	}
raoul's avatar
raoul committed
	if (!selectedStitches.size() && !selectedCurves.size())
	{
        controller->clearSelection();
        emit stitchSelectionChange();
raoul's avatar
raoul committed
	}

	return true;

/** \brief draw the document on screen
 *
 * \return true if all is ok
 *
 */
bool OE_interfaceDisplay::draw()
{
raoul's avatar
raoul committed
	if (run)
	{
		//TODO remove controller->sendInstPoint();
raoul's avatar
raoul committed
	}

	//draw the closest point
	if (editionState==NewSubcurve)
raoul's avatar
raoul committed
	{
		// Points
		glPointSize(5.0f);

		glBegin(GL_POINTS);
		if (joincurve_id == 0)
raoul's avatar
raoul committed
		{
			glColor4f(0.8f, 0.0f, 0.0f, 1.0f);
raoul's avatar
raoul committed
		}
		else
		{
			glColor4f(0.0f, 0.8f, 0.0f, 1.0f);
raoul's avatar
raoul committed
		}
		glVertex2f(editClosestPoint.v.x,editClosestPoint.v.y);
		glEnd();
raoul's avatar
raoul committed
	}
	// draw the cursor cross
	glLineWidth(1.5);
	glBegin(GL_LINES);
		if (editionState)
raoul's avatar
raoul committed
		{
			if (joincurve_id == 0)
			{
				glColor4f(0.8f, 0.0f, 0.0f, 0.5f);
raoul's avatar
raoul committed
			}
			else
			{
				glColor4f(0.0f, 0.8f, 0.0f, 0.5f);
raoul's avatar
raoul committed
			}
		}
			glColor4fv(style->cursorColor.rgba);