Skip to content
OE_display.cpp 39.1 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_display.h"
#include "OE_utils.h"

#include "OE_controller.h"

raoul's avatar
raoul committed
#include <qevent.h>

#include <GL/gl.h>
#include <cstdlib>
#include <math.h>
#include <cstdio>
#include <cstring>
#include <algorithm>
raoul's avatar
raoul committed
#include <QMimeData>
#include <QTabWidget>
#include <iostream>
#include <sstream>
#include <string>
raoul's avatar
raoul committed

raoul's avatar
raoul committed
OE_display::OE_display(QWidget* parent, Qt::WindowFlags f) : QOpenGLWidget(parent, f), zoom(1.0), pan(false), width(0), height(0)
{
}
raoul's avatar
raoul committed
OE_display::~OE_display()
raoul's avatar
raoul committed
void OE_display::mouseMoveEvent(QMouseEvent* event)
raoul's avatar
raoul committed
	//DEBUG printf("MouseEvent pos %d, %d\n", event->pos().x(), event->pos().y()); fflush(stdout);
	event->accept();
	bool redraw = mouse_Pos(event->pos().x(), event->pos().y());
	if (redraw)
		update();
	//TODO ? repaint();
raoul's avatar
raoul committed
void OE_display::mousePressEvent(QMouseEvent* event)
{
	event->accept();
	bool redraw = mouse_Button(event);
	if (redraw)
		update();
}

void OE_display::mouseReleaseEvent(QMouseEvent* event)
{
	event->accept();
	bool redraw = mouse_Button(event);
	if (redraw)
		update();
}

void OE_display::wheelEvent(QWheelEvent* event)
{
	event->accept();
	OE_display::scroll(0, event->delta());
	update();
}

raoul's avatar
raoul committed

static bool endsWith(const std::string& str, const std::string& suffix)
{
	return str.size() >= suffix.size() && 0 == str.compare(str.size()-suffix.size(), suffix.size(), suffix);
}

static std::vector<std::string> getAllowedFiles(QDropEvent *event)
{
	std::vector<std::string> uris;
	if (event->mimeData()->hasFormat("text/uri-list"))
	{
		for (const QUrl& qurl : event->mimeData()->urls())
raoul's avatar
raoul committed
		{
raoul's avatar
raoul committed
				continue;
			std::string url = qurl.toLocalFile().toStdString();
			std::string urlLower = url;
			std::transform(urlLower.begin(), urlLower.end(), urlLower.begin(), ::tolower);
			if (!(endsWith(urlLower, ".oe") || endsWith(urlLower, ".svg") || endsWith(urlLower, ".pes")))
raoul's avatar
raoul committed
				continue;
raoul's avatar
raoul committed
		}
	}
	return uris;
}

void OE_display::dragEnterEvent(QDragEnterEvent *event)
{
	std::vector<std::string> files = getAllowedFiles(event);
	if (files.size() != 1)
	{
		event->ignore();
		return;
	}
	event->setDropAction(Qt::MoveAction);
	event->accept();
}

void OE_display::dropEvent(QDropEvent *event)
{
	event->acceptProposedAction();
	loadFile(getAllowedFiles(event).front());
}
void OE_display::loadFile(std::string file)
{
raoul's avatar
raoul committed
	std::string fileLower = file;
	std::transform(fileLower.begin(), fileLower.end(), fileLower.begin(), ::tolower);

	OE_document* newDoc = nullptr;
	printf("Importing file from %s\n", file.c_str());
	newDoc = new OE_document();
	if (!newDoc->loadFromFile(file))
raoul's avatar
raoul committed
	{
		delete newDoc;
		newDoc = nullptr;
raoul's avatar
raoul committed
	}

	if (newDoc)
	{
		OE_document* oldDoc = document;
		document = newDoc;
		if (oldDoc)
			delete oldDoc;

		if (controller)
		{
			controller->setDocument(newDoc);
			controller->initNewDocument();
		}

		setDocument(newDoc);
		showAll();
		QTabWidget* tabWidget = dynamic_cast<QTabWidget*>(this->parent()->parent()->parent());
		int idx = tabWidget->indexOf(dynamic_cast<QWidget*>(this->parent()));
		tabWidget->setTabText(idx, QString::fromStdString(file.substr(file.rfind('/')+1)));
		tabWidget->setTabToolTip(idx, QString::fromStdString(file));
raoul's avatar
raoul committed
void OE_display::initializeGL()
raoul's avatar
raoul committed
	initializeOpenGLFunctions();
	glEnable(GL_POINT_SMOOTH);
	glEnable(GL_LINE_SMOOTH);
}

void OE_display::resizeGL(int w, int h)
{
	resize(w, h);
	update();
}

void OE_display::paintGL()
{
	draw();
}

/** \brief draw the document on screen
 *
 * \return true if all is ok
 *
 */

raoul's avatar
raoul committed
bool OE_display::setDocument(OE_document* document)
raoul's avatar
raoul committed
/*	if (document)
raoul's avatar
raoul committed
		std::list<OE_pointcurve*>::iterator curve = document->curves.begin();

		while (curve != document->curves.end())
			(*curve)->refresh(zoom/2);
			curve++;
bool OE_display::setController(OE_controller* controller)
	this->controller = controller;
	return true;
}

bool OE_display::setDisplayStyle(OE_displayStyle* style)
{
	this->style = style;
	return true;
}

bool OE_display::refreshAll()
raoul's avatar
raoul committed
	if (document)
	{
		std::list<OE_pointcurve*>::iterator curve = document->curves.begin();

		while (curve != document->curves.end())
		{
			(*curve)->refresh(zoom/2, !changeDpi);
		std::list<OE_stitchs*>::iterator stitch = document->stitchs.begin();

		while (stitch != document->stitchs.end())
		{
			(*stitch)->refresh(zoom/2, !changeDpi);

		if (controller) controller->generateInstructions();

		changeDpi = false;
		return true;
	}
	return false;
}

/** \brief draw the move gizmo
 *
 * \return true if all is ok
 *
 */
bool OE_display::drawSelectionTools(bool select)
{
	vector_2d min = selectionBounds.getMin();
	vector_2d max = selectionBounds.getMax();

    float crossSize = zoom*OE_preferences::moveCrossSize;
    float boxOffset = zoom*OE_preferences::moveCrossSize;

	min.x -= boxOffset;
	min.y -= boxOffset;
	max.x += boxOffset;
	max.y += boxOffset;

	if (selectionBounds.init)
	{
		if (select)
		{
raoul's avatar
raoul committed
			glTranslatef(min.x, min.y, 0);
3dsman's avatar
3dsman committed
            glPushName(MOVETOOLID); //id of the move tool
                glColor4fv(OE_preferences::selectionBoundColor.rgba);
raoul's avatar
raoul committed
				//glColor4d(1, 0, 0, 1);
				glVertex2f(crossSize*1.0f, 0);
				glVertex2f(0, crossSize*1.0f);
				glVertex2f(-crossSize*1.0f, 0);
				glVertex2f(0, -crossSize*1.0f);
3dsman's avatar
3dsman committed
            glPushName(SCALETOOLID); //id of the scale tool
raoul's avatar
raoul committed
			glVertex2f(-crossSize*0.5f, -crossSize*0.5f);
			glVertex2f(-crossSize*0.5f, -crossSize*1.5f);
			glVertex2f(-crossSize*1.5f, -crossSize*1.5f);
			glVertex2f(-crossSize*1.5f, -crossSize*0.5f);
raoul's avatar
raoul committed
			glTranslatef(-min.x, -min.y, 0);
		}
		else
		{
			//draw selection bound
			glLineWidth(1.5);
			glBegin(GL_LINE_LOOP);
                glColor4fv(OE_preferences::selectionBoundColor.rgba);
raoul's avatar
raoul committed
				glVertex2f(min.x, min.y);
				glVertex2f(min.x, max.y);
				glVertex2f(max.x, max.y);
				glVertex2f(max.x, min.y);
raoul's avatar
raoul committed
			glTranslatef(min.x, min.y, 0);

			//draw move cross
			glBegin(GL_TRIANGLES);
                glColor4fv(OE_preferences::transformGizmoColor.rgba);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*0.6f, -crossSize*0.1f);
				glVertex2f(-crossSize*0.6f, crossSize*0.1f);
				glVertex2f(crossSize*0.6f, crossSize*0.1f);
raoul's avatar
raoul committed
				glVertex2f(crossSize*0.6f, crossSize*0.1f);
				glVertex2f(-crossSize*0.6f, -crossSize*0.1f);
				glVertex2f(crossSize*0.6f, -crossSize*0.1f);
raoul's avatar
raoul committed
				glVertex2f(crossSize*0.6f, crossSize*0.3f);
				glVertex2f(crossSize*1.0f, 0);
				glVertex2f(crossSize*0.6f, -crossSize*0.3f);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*0.6f, crossSize*0.3f);
				glVertex2f(-crossSize*1.0f, 0);
				glVertex2f(-crossSize*0.6f, -crossSize*0.3f);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*0.1f, -crossSize*0.6f);
				glVertex2f(crossSize*0.1f, -crossSize*0.6f);
				glVertex2f(crossSize*0.1f, crossSize*0.6f);
raoul's avatar
raoul committed
				glVertex2f(crossSize*0.1f, crossSize*0.6f);
				glVertex2f(-crossSize*0.1f, -crossSize*0.6f);
				glVertex2f(-crossSize*0.1f, crossSize*0.6f);
raoul's avatar
raoul committed
				glVertex2f(crossSize*0.3f, crossSize*0.6f);
				glVertex2f(0, crossSize*1.0f);
				glVertex2f(-crossSize*0.3f, crossSize*0.6f);
raoul's avatar
raoul committed
				glVertex2f(crossSize*0.3f, -crossSize*0.6f);
				glVertex2f(0, -crossSize*1.0f);
				glVertex2f(-crossSize*0.3f, -crossSize*0.6f);
			glEnd();
			//draw scale icon
			glBegin(GL_TRIANGLES);

                glColor4fv(OE_preferences::transformGizmoColor.rgba);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*1.5f, -crossSize*1.2f);
				glVertex2f(-crossSize*0.6f, -crossSize*0.55f);
				glVertex2f(-crossSize*0.55f, -crossSize*0.6f);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*1.5f, -crossSize*1.2f);
				glVertex2f(-crossSize*0.5f, -crossSize*0.55f);
				glVertex2f(-crossSize*1.2f, -crossSize*1.5f);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*0.5f, -crossSize*0.5f);
				glVertex2f(-crossSize*0.85f, -crossSize*0.5f);
				glVertex2f(-crossSize*0.5f, -crossSize*0.85f);
raoul's avatar
raoul committed
				glVertex2f(-crossSize*1.5f, -crossSize*1.5f);
				glVertex2f(-crossSize*1.5f, -crossSize*0.9f);
				glVertex2f(-crossSize*0.9f, -crossSize*1.5f);
raoul's avatar
raoul committed
			glTranslatef(-min.x, -min.y, 0);
		}
		return true;
	}
	return false;
}

/** \brief draw a triangle */
void OE_display::drawTriangle(vector_2d point1, vector_2d point2, float size)
{
raoul's avatar
raoul committed
	glBegin(GL_TRIANGLE_FAN);
	vector_2d dir  = point2 - point1;
	dir.normalize();
	dir = dir*zoom*size;

	vector_2d lat  = dir;
	lat.turnRight();

	point1 = point1 - dir;
	vector_2d p1 = dir*2+point1;
	glVertex2f(p1.x, p1.y);
	p1 = point1+lat;
	glVertex2f(p1.x, p1.y);
	p1 = point1+dir/2;
	glVertex2f(p1.x, p1.y);
	p1 = point1-lat;
	glVertex2f(p1.x, p1.y);
	glEnd();
}


/** \brief draw a stitch on screen
 *
 * \return true if all is ok
 *
 */
raoul's avatar
raoul committed
bool OE_display::drawStitch(OE_stitchs* stitch, int curpoint)
raoul's avatar
raoul committed
	if (!stitch->check() || !style->drawStitches || stitch->getNpts()==0)
	{
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	const std::vector<vector_2d>& points = stitch->getPoints();
    glPointSize(OE_preferences::stitchPointSize);
raoul's avatar
raoul committed
	{
        glColor4fv(OE_preferences::stitchPointBeforeCurrentColor.rgba);
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	{
        glColor4fv(OE_preferences::stitchPointAfterCurrentColor.rgba);
raoul's avatar
raoul committed
	}
	for (i = 1; i < points.size(); i++)
	{
raoul's avatar
raoul committed
		{
            glColor4fv(OE_preferences::stitchPointAfterCurrentColor.rgba);
raoul's avatar
raoul committed
		}
raoul's avatar
raoul committed
		glVertex2f(points[i].x, points[i].y);
    glLineWidth(OE_preferences::stitchLineWidth);
raoul's avatar
raoul committed
	OE_thread* thread = stitch->getThread();
raoul's avatar
raoul committed
	{
		glColor4fv(thread->getColor().rgba);
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	{
        glColor4fv(OE_preferences::stitchLineBeforeCurrentColor.rgba);
raoul's avatar
raoul committed
	}
	for (i = 0; i < points.size(); i++)
	{
		glVertex2f(points.at(i).x, points.at(i).y);
	}
    glColor4fv(OE_preferences::stitchStartPointColor.rgba);
    glPointSize(OE_preferences::stitchStartPointSize);
raoul's avatar
raoul committed
	glVertex2f(points.at(0).x, points.at(0).y);
	glEnd();

	return true;
}

/** \brief draw a curve on screen
 *
 * \return true if all is ok
 *
 */
raoul's avatar
raoul committed
bool OE_display::drawCurve(OE_curve* curve)
raoul's avatar
raoul committed
	if (!curve->check() || !style->drawCurves)
	{
raoul's avatar
raoul committed
	}
	if (curve->getNeedRefresh())
raoul's avatar
raoul committed
	{
		curve->refresh(zoom/2);
    glLineWidth(OE_preferences::curveWidth);
        glColor4fv(OE_preferences::curveColor.rgba);
raoul's avatar
raoul committed
		for (i = 0; i < curve->discPts.size(); i++)
		{
			glVertex2f(curve->discPts.at(i).v.x, curve->discPts.at(i).v.y);
raoul's avatar
raoul committed
		if (curve->getClosed())
		{
			glVertex2f(curve->discPts.at(0).v.x, curve->discPts.at(0).v.y);
		}
	glEnd();
	if (!curve->getClosed())
	{
        glPointSize(OE_preferences::curveStartPointSize);
		glBegin(GL_POINTS);
		glVertex2f(curve->discPts.at(0).v.x, curve->discPts.at(0).v.y);
raoul's avatar
raoul committed
		glVertex2f(curve->discPts.at(curve->discPts.size()-1).v.x, curve->discPts.at(curve->discPts.size()-1).v.y);
raoul's avatar
raoul committed
	glColor3d(1.0f, 0.0f, 0.0f);
	glBegin(GL_POINTS);
	vector_2dt point = curve->closestPoint(vector_2d (absMouseX, absMouseY));
	glVertex2f(point.v.x, point.v.y);
	glEnd();
*/
	//draw curve points
raoul's avatar
raoul committed
	if (style->drawCurvePoints)
        glPointSize(OE_preferences::curvePointSize);
		glBegin(GL_POINTS);
		glVertex2f(curve->discPts[0].v.x, curve->discPts[0].v.y);
raoul's avatar
raoul committed
		for (i = 0; i < curve->discPts.size(); i++)
		{
			glColor3d(curve->discPts.at(i).t/((double)curve->pts.size()/3.0f), 0.0f, 0.0f);
				glVertex2f(curve->discPts.at(i).v.x, curve->discPts.at(i).v.y);
		}
raoul's avatar
raoul committed
		if (curve->getClosed())
		{
			glVertex2f(curve->discPts[0].v.x, curve->discPts[0].v.y);
		}
		glEnd();
	}
	return true;
}


/** \brief draw the curve controls on screen
 *
 * \return true if all is ok
 *
 */

raoul's avatar
raoul committed
bool OE_display::drawCurveControl(OE_curve* curve, bool select)
raoul's avatar
raoul committed
	if (!curve->check() || !style->drawCurves || !style->drawCurveControls)
	{
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	OE_pointcurve* tmpcurve = dynamic_cast<OE_pointcurve*> (curve);
	if (tmpcurve)
raoul's avatar
raoul committed
		if (select)
		{
			// Points
			glPointSize(4.0f);

			glPushName(0); //init the controller type index
			glPushName(0); //init the controller points index

raoul's avatar
raoul committed
			for (unsigned i = 0; i < tmpcurve->pts.size(); i++)
			{
				glLoadName(i);
				//vector_2d* p = &tmpcurve->pts[i];
				glBegin(GL_POINTS);
raoul's avatar
raoul committed
				glVertex2f(tmpcurve->pts[i].x, tmpcurve->pts[i].y);
raoul's avatar
raoul committed
		}
		else
            glColor4fv(OE_preferences::curveTangeantColor.rgba);
raoul's avatar
raoul committed
			for (unsigned i = 0; i < tmpcurve->pts.size()-3; i += 3)
			{
				vector_2d* p = &tmpcurve->pts[i];
raoul's avatar
raoul committed
				glVertex2f(p[0].x, p[0].y);
				glVertex2f(p[1].x, p[1].y);
				glVertex2f(p[2].x, p[2].y);
				glVertex2f(p[3].x, p[3].y);
            glPointSize(OE_preferences::curvePointSize);
			//glColor4ubv(controlEndPointColor);

			glBegin(GL_POINTS);

            glColor4fv(OE_preferences::curvePointColor.rgba);
raoul's avatar
raoul committed
			for (unsigned i = 0; i < tmpcurve->pts.size()-3; i += 3)
			{
				vector_2d* p = &tmpcurve->pts[i];
raoul's avatar
raoul committed
				glVertex2f(p[0].x, p[0].y);
            glPointSize(OE_preferences::curvePointSize);
            glColor4fv(OE_preferences::curvePointColor.rgba);
raoul's avatar
raoul committed
			for (unsigned i = 0; i < curve->pts.size()-3; i += 3)
			{
raoul's avatar
raoul committed
				glVertex2f(p[1].x, p[1].y);
				glVertex2f(p[2].x, p[2].y);
raoul's avatar
raoul committed
			/*glColor4f(1.0, 0.0, 0.0, 1.0);
			glPointSize(2.0f);
			glBegin(GL_POINTS);
raoul's avatar
raoul committed
			glVertex2f(curve->pts[0].x, curve->pts[0].y);
            glColor4fv(OE_preferences::curveControlsColor.rgba);
			drawTriangle(curve->pts[0], curve->pts[1], 13);

			glLineWidth(2);
			glBegin(GL_LINES);
			vector_2d p  = curve->pts[1] - curve->pts[0];
			p.normalize();
			p = p+curve->pts[0];
raoul's avatar
raoul committed
			glVertex2f(curve->pts[0].x, curve->pts[0].y);
			//glVertex2f(curve->pts[1].x, curve->pts[1].y);
			glVertex2f(p.x, p.y);
3dsman's avatar
3dsman committed
/*
raoul's avatar
raoul committed
	OE_joincurve* joincurve = dynamic_cast<OE_joincurve*> (curve);
	if (joincurve)
	{
		std::list<OE_subcurve*>::iterator curve = joincurve->curves.begin();
raoul's avatar
raoul committed
		if (!select)
		{
			while (curve != joincurve->curves.end())
			{
raoul's avatar
raoul committed
				OE_curve* tmpcurve = (*curve);
				glLineWidth(3);
				glBegin(GL_LINE_STRIP);
				style->curveTangeantColor.gl();
raoul's avatar
raoul committed
				for (unsigned i = 0; i < tmpcurve->discPts.size(); i++)
				{
					glVertex2f(tmpcurve->discPts.at(i).v.x, tmpcurve->discPts.at(i).v.y);
				}
				glEnd();
				if (tmpcurve->discPts.size()>1)
				{
3dsman's avatar
3dsman committed
					style->curveControlsColor.gl();
					drawTriangle(tmpcurve->pts[0], tmpcurve->pts[1], 13);
				}
				//draw the add subcurve controler
				if (tmpcurve->discPts.size()>1)
				{
					style->curveControlsColor.gl();
					drawTriangle(tmpcurve->pts[tmpcurve->getNpts()-1]+vector_2d(10,10), tmpcurve->pts[tmpcurve->getNpts()-2]+vector_2d(10,10), 13);
				}
				curve++;
			}
		}
		return true;
	}

	//if this curve is a subcurve
raoul's avatar
raoul committed
	OE_subcurve* subcurve = dynamic_cast<OE_subcurve*> (curve);
	if (subcurve)
raoul's avatar
raoul committed
		if (select)
		{
			// Points
			glPointSize(4.0f);

			glPushName(0); //init the controller type index
			glPushName(0); //init the controller points index

			glBegin(GL_POINTS);
raoul's avatar
raoul committed
			glVertex2f(subcurve->pts[0].x, subcurve->pts[0].y);
raoul's avatar
raoul committed
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x, subcurve->pts[subcurve->getNpts()-1].y);
raoul's avatar
raoul committed
		}
		else
			style->curvePointColor.gl();
			glPointSize(5.0f);
			glBegin(GL_POINTS);
raoul's avatar
raoul committed
			glVertex2f(subcurve->pts[0].x, subcurve->pts[0].y);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x, subcurve->pts[subcurve->getNpts()-1].y);
3dsman's avatar
3dsman committed
			style->curveControlsColor.gl();
			drawTriangle(subcurve->pts[0], subcurve->pts[1], 13);
		}
		return true;
3dsman's avatar
3dsman committed
	}*/
	return false;
void OE_display::drawJoincurveControl(OE_joincurve* curve, bool select, int index)
	const float uiSize = zoom * 10;
3dsman's avatar
3dsman committed
	const float uiDistance = zoom * 17;

		glLoadName(index);//set the controller type index
		glPushName(0); //init the controller points index

raoul's avatar
raoul committed
		for (unsigned i = 0; i < curve->curves.size(); i++)
		{
			OE_subcurve* subcurve = curve->getCurve(i);
raoul's avatar
raoul committed
			glLoadName(i*2+1);
			glBegin(GL_POINTS);
			glVertex2f(subcurve->pts[0].x, subcurve->pts[0].y);
			glEnd();
raoul's avatar
raoul committed
			glLoadName(i*2);
			glBegin(GL_POINTS);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x, subcurve->pts[subcurve->getNpts()-1].y);
			glEnd();
		}
		glPopName();
		glLoadName(index + 1);//set the controller type index
		glPushName(0); //init the controller points index

		// draw the insert subcurve control picking zone
		OE_subcurve* subcurve = curve->getCurve(0);
		glPointSize(2*uiSize);
		glLoadName(0);
		glBegin(GL_POINTS);
		glVertex2f(subcurve->pts[0].x+uiDistance, subcurve->pts[0].y-uiDistance);
		glEnd();
		for (unsigned i = 0; i < curve->curves.size(); i++)
		{
			subcurve = curve->getCurve(i);

			glLoadName(i+1);
			glBegin(GL_POINTS);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x+uiDistance, subcurve->pts[subcurve->getNpts()-1].y-uiDistance);
			glEnd();
		}
		glPopName();
raoul's avatar
raoul committed
	else
	{
		for (unsigned i = 0; i < curve->curves.size(); i++)
		{
			OE_subcurve* subcurve = curve->getCurve(i);

			// draw the subcurves
			glLineWidth(2);
			glBegin(GL_LINE_STRIP);
            glColor4fv(OE_preferences::curveTangeantColor.rgba);
raoul's avatar
raoul committed
			for (unsigned i = 0; i < subcurve->discPts.size(); i++)
			{
				glVertex2f(subcurve->discPts.at(i).v.x, subcurve->discPts.at(i).v.y);
			// draw the insert subcurve control
            glColor4fv(OE_preferences::curveControlsColor.rgba);
			if (i == 0)
			{
				glBegin(GL_LINES);
				glVertex2f(subcurve->pts[0].x-uiSize+uiDistance, subcurve->pts[0].y-uiDistance);
				glVertex2f(subcurve->pts[0].x+uiSize+uiDistance, subcurve->pts[0].y-uiDistance);
				glVertex2f(subcurve->pts[0].x+uiDistance, subcurve->pts[0].y-uiSize-uiDistance);
				glVertex2f(subcurve->pts[0].x+uiDistance, subcurve->pts[0].y+uiSize-uiDistance);
				glEnd();
			}
3dsman's avatar
3dsman committed
			glBegin(GL_LINES);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x-uiSize+uiDistance, subcurve->pts[subcurve->getNpts()-1].y-uiDistance);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x+uiSize+uiDistance, subcurve->pts[subcurve->getNpts()-1].y-uiDistance);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x+uiDistance, subcurve->pts[subcurve->getNpts()-1].y-uiSize-uiDistance);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x+uiDistance, subcurve->pts[subcurve->getNpts()-1].y+uiSize-uiDistance);
			glEnd();

            glColor4fv(OE_preferences::curveControlsColor.rgba);
			glPointSize(6.0f);
			glBegin(GL_POINTS);
raoul's avatar
raoul committed
			glVertex2f(subcurve->pts[0].x, subcurve->pts[0].y);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x, subcurve->pts[subcurve->getNpts()-1].y);
            glColor4fv(OE_preferences::curveControlsColor.rgba);
			drawTriangle(subcurve->pts[0], subcurve->pts[1], 13);
		}
	}
}

void OE_display::drawFillstitchControl(OE_fillstitch* fillstitch, bool select)
{
	std::vector<vector_2d> grid = fillstitch->getGridControl();
	std::vector<OE_pointcurve> curves = fillstitch->getCurveControl();

	if (select)
	{
		for (unsigned i = 0; i < grid.size(); i++)
		{
			glLoadName(i);

			glBegin(GL_POINTS);
			glVertex2f(grid[i].x, grid[i].y);
			glEnd();
		}
	}
	else
	{
        glColor4fv(OE_preferences::curveControlsColor.rgba);
		for (unsigned i = 0; i < curves.size(); i++)
		{
			drawCurve(&curves[i]);
		}
        glColor4fv(OE_preferences::curveControlsColor.rgba);
		for (unsigned i = 0; i < grid.size(); i++)
		{
			glBegin(GL_POINTS);
			glVertex2f(grid[i].x, grid[i].y);
			glEnd();
		}
		/*
		for (unsigned i = 0; i < curve->curves.size(); i++)
		{
			OE_subcurve* subcurve = curve->getCurve(i);

			// draw the subcurves
			glLineWidth(2);
			glBegin(GL_LINE_STRIP);
			style->curveTangeantColor.gl();

			for (unsigned i = 0; i < subcurve->discPts.size(); i++)
			{
				glVertex2f(subcurve->discPts.at(i).v.x, subcurve->discPts.at(i).v.y);
			}
			glEnd();

			// draw the subcurves controls
3dsman's avatar
3dsman committed
			style->curveControlsColor.gl();
			glPointSize(6.0f);
			glBegin(GL_POINTS);
			glVertex2f(subcurve->pts[0].x, subcurve->pts[0].y);
			glVertex2f(subcurve->pts[subcurve->getNpts()-1].x, subcurve->pts[subcurve->getNpts()-1].y);

			glEnd();
			// start point & direction
3dsman's avatar
3dsman committed
			style->curveControlsColor.gl();
			drawTriangle(subcurve->pts[0], subcurve->pts[1], 13);
		}*/
	}
}


/** \brief draw the stitchs controls on screen
 *
 * \return true if all is ok
 *
 */
raoul's avatar
raoul committed
bool OE_display::drawStitchControl(OE_stitchs* stitch, bool select)
raoul's avatar
raoul committed
	if (!stitch->check() || !style->drawStitches || !style->drawStitchControl)
	{
raoul's avatar
raoul committed
	}
raoul's avatar
raoul committed
	OE_linestitch* linestitch = dynamic_cast<OE_linestitch*> (stitch);
	if (linestitch)
	{
		OE_joincurve* joinCurve  = linestitch->getJoincurve();
		vector_2d dir = joinCurve->pts[1] - joinCurve->pts[0];
		dir.normalize();
		dir.turnRight();
		vector_2d scaleDir = dir*zoom*10;
		dir = scaleDir * 10+ joinCurve->pts[0];

raoul's avatar
raoul committed
		if (select)
		{

			// Points
			glPointSize(4.0f);

			glPushName(0); //init the controller type index
			if (linestitch->getJoincurve())
raoul's avatar
raoul committed
			{
				drawJoincurveControl(linestitch->getJoincurve(), true, 0);
raoul's avatar
raoul committed
			}
			glLoadName(2); //set the controller type index
			glPushName(0); //init the controller points index
3dsman's avatar
3dsman committed

				//draw stitch width controller
raoul's avatar
raoul committed
				glVertex2f(dir.x-scaleDir.y+scaleDir.x/3, dir.y+scaleDir.x+scaleDir.y/3);
				glVertex2f(dir.x+scaleDir.x*2, dir.y+scaleDir.y*2);
				glVertex2f(dir.x+scaleDir.y+scaleDir.x/3, dir.y-scaleDir.x+scaleDir.y/3);
raoul's avatar
raoul committed
				glVertex2f(dir.x+scaleDir.y-scaleDir.x/3, dir.y-scaleDir.x-scaleDir.y/3);
				glVertex2f(dir.x-scaleDir.x*2, dir.y-scaleDir.y*2);
				glVertex2f(dir.x-scaleDir.y-scaleDir.x/3, dir.y+scaleDir.x-scaleDir.y/3);
raoul's avatar
raoul committed
		}
		else
		{
			glLineWidth(2);
			glBegin(GL_LINE_STRIP);
            glColor4fv(OE_preferences::stitchLineBeforeCurrentColor.rgba);
raoul's avatar
raoul committed
				glVertex2f(joinCurve->pts[0].x, joinCurve->pts[0].y);
				glVertex2f(dir.x, dir.y);
raoul's avatar
raoul committed
			drawJoincurveControl(joinCurve, false);
3dsman's avatar
3dsman committed
			//draw stitch width controller
            glColor4fv(OE_preferences::stitchControlsColor.rgba);
raoul's avatar
raoul committed
			glVertex2f(dir.x-scaleDir.y+scaleDir.x/3, dir.y+scaleDir.x+scaleDir.y/3);
			glVertex2f(dir.x+scaleDir.x*2, dir.y+scaleDir.y*2);
			glVertex2f(dir.x+scaleDir.y+scaleDir.x/3, dir.y-scaleDir.x+scaleDir.y/3);
raoul's avatar
raoul committed
			glVertex2f(dir.x+scaleDir.y-scaleDir.x/3, dir.y-scaleDir.x-scaleDir.y/3);
			glVertex2f(dir.x-scaleDir.x*2, dir.y-scaleDir.y*2);
			glVertex2f(dir.x-scaleDir.y-scaleDir.x/3, dir.y+scaleDir.x-scaleDir.y/3);
raoul's avatar
raoul committed
	OE_birailstitch* birailstitch = dynamic_cast<OE_birailstitch*> (stitch);
	if (birailstitch)
raoul's avatar
raoul committed
		if (select)
		{

			// Points
			glPointSize(4.0f);

			glPushName(0); //init the controller type index
			drawJoincurveControl(birailstitch->getJoincurve1(), true, 0);
			drawJoincurveControl(birailstitch->getJoincurve2(), true, 2);
raoul's avatar
raoul committed
		}
		else
raoul's avatar
raoul committed
			drawJoincurveControl(birailstitch->getJoincurve1(), false);
			drawJoincurveControl(birailstitch->getJoincurve2(), false);
	OE_fillstitch* fillstitch = dynamic_cast<OE_fillstitch*> (stitch);
	if (fillstitch)
	{
		if (select)
		{

			// Points
			glPointSize(4.0f);

			glPushName(0); //init the controller type index
3dsman's avatar
3dsman committed
			drawJoincurveControl(fillstitch->getJoincurve(), true, 0);
			glLoadName(2);
			glPushName(0);
			drawFillstitchControl(fillstitch, true);
			glPopName();
			glPopName();
		}
		else
		{
			drawJoincurveControl(fillstitch->getJoincurve(), false);
			drawFillstitchControl(fillstitch, false);
raoul's avatar
raoul committed
	OE_linkstitch* linkstitch = dynamic_cast<OE_linkstitch*> (stitch);
	if (linkstitch)