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>
#include <GL/gl.h>
#include <cstdlib>
#include <math.h>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include "actions/OE_actionsCurves.h"
#include "actions/OE_actionsStitchs.h"
#include "actions/OE_actionsLineStitchs.h"
#include "actions/OE_actionsBirailStitchs.h"
#include "actions/OE_actionsSelection.h"
#include "actions/OE_actionsThreads.h"
OE_interfaceDisplay::OE_interfaceDisplay(OE_document* document)
{
editDisplay = new OE_editDisplay();
commandDisplay = new OE_commandDisplay();
curDisplay = editDisplay;
setDocument(document);
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
}
OE_interfaceDisplay::~OE_interfaceDisplay()
{
delete editDisplay;
delete commandDisplay;
}
bool OE_interfaceDisplay::setDocument(OE_document * document)
{
this->document = document;
if(document)
{
editDisplay->setDocument(document);
commandDisplay->setDocument(document);
}
return true;
}
bool OE_interfaceDisplay::setController(OE_controller* controller)
{
this->controller = controller;
if(controller)
{
if (editDisplay) editDisplay->setController(controller);
if (commandDisplay) commandDisplay->setController(controller);
}
return true;
}
void OE_interfaceDisplay::showAll()
{
if (curDisplay) curDisplay->showAll();
}
void OE_interfaceDisplay::mouse_Pos(double x, double y)
{
if (curDisplay) curDisplay->mouse_Pos(x, y);
if (editSubcurve)
{
editClosestPoint = controller->getClosestPoint(editSubcurve->getCurve(),absMouse);
}
closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
if ((selControlCurve)&&(selControlType == 0))
controller->editLastAction(vector_2d((mouse.x - clicOldMouse.x)*zoom*2,(mouse.y - clicOldMouse.y)*zoom*2));
if ((selControlStitch)&&(selControlType == 0))
{
OE_linestitch* tmpLineStitch = dynamic_cast<OE_linestitch*>(selControlStitch);
if (tmpLineStitch)
controller->editLastAction(editClosestPoint.t);
OE_birailstitch* tmpBirailStitch = dynamic_cast<OE_birailstitch*>(selControlStitch);
if (tmpBirailStitch)
controller->editLastAction(editClosestPoint.t);
}
void OE_interfaceDisplay::mouse_Button(int button, int action, int mods)
{
modState = mods;
if (curDisplay) curDisplay->mouse_Button(button, action, mods);
if((curDisplay == editDisplay)&&(button == GLFW_MOUSE_BUTTON_LEFT)&&(action==GLFW_PRESS))
{
else
{
controller->addAction(new OE_actionJoincurveAddSubCurve(curJoincurve,-1,closestCurve,editClosestPoint.t,editClosestPoint.t,false));
selControlStitch = controller->getStitch(-1);
selControlType = 0;
editSubcurve = curJoincurve->getCurve(-1);
}
else
{
clicOldMouse = mouse;
selection = true;
if((curDisplay == editDisplay)&&(button == GLFW_MOUSE_BUTTON_LEFT)&&(action==GLFW_RELEASE))
{
selectBox();
selectApply(selectedCurves, selectedStitches);
selection = false;
}
if(selControlCurve)selControlCurve = 0;
if(selControlStitch)selControlStitch = 0;
}
void OE_interfaceDisplay::scroll(double xoffset, double yoffset)
{
if (curDisplay) curDisplay->scroll(xoffset, yoffset);
}
void OE_interfaceDisplay::resize(int width, int height)
{
if (curDisplay) curDisplay->resize(width, height);
}
void OE_interfaceDisplay::key(int key, int scancode, int action, int mods)
{
modState = mods;
if((action == GLFW_PRESS)&&(curDisplay == editDisplay))
{
if ((key == GLFW_KEY_Z)&&(mods == GLFW_MOD_CONTROL))
{
if (controller) controller->undoAction();
}
if ((key == GLFW_KEY_Z)&&(mods == GLFW_MOD_CONTROL+GLFW_MOD_SHIFT))
{
if (controller) controller->redoAction();
}
if (mods == 0)
{
if (key == GLFW_KEY_C)
{
if (controller) controller->toggleCloseSelectedCurve();
}
{
{
if(curDisplay == editDisplay)
{
if (controller) controller->generateInstructions();
curDisplay = commandDisplay;
}
else
curDisplay = editDisplay;
}
if(key == GLFW_KEY_S)
{
if (controller)
{
document->selectedStitchs.push_back(controller->getLineStitch(-1));
curJoincurve = controller->getLineStitch(-1)->getJoincurve();
closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
creationState = NewSubcurve;
joincurve_id = 0;
controller->newBirailStitch(false, false, 0,0,1);
document->selectedStitchs.push_back(controller->getBirailStitch(-1));
curJoincurve = controller->getBirailStitch(-1)->getJoincurve1();
closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
creationState = NewSubcurve;
joincurve_id = 1;
}
}
if(key == GLFW_KEY_ENTER)
{
if (joincurve_id == 0)
{
creationState = None;
curJoincurve = 0;
}else
{
curJoincurve = controller->getBirailStitch(-1)->getJoincurve2();
closestCurve = controller->getClosestCurve(absMouse, editClosestPoint);
joincurve_id -= 1;
}
}
}
curDisplay->key( key, scancode, action, mods);
}
selectedCurves.clear();
selectedStitches.clear();
vector_2d clicAbsMouse = viewPos+(clicOldMouse-vector_2d(width,height)/2)*zoom*2;
float tmpXMin, tmpYMin, tmpXMax, tmpYMax;
std::list<OE_curve*>::iterator curve = document->curves.begin();
while (curve != document->curves.end())
{
(*curve)->getBound(&tmpXMin, &tmpYMin, &tmpXMax, &tmpYMax);
if ((tmpXMin>minf(clicAbsMouse.x,absMouse.x))
&&(tmpXMax<maxf(clicAbsMouse.x,absMouse.x))
&&(tmpYMin>minf(clicAbsMouse.y,absMouse.y))
&&(tmpYMax<maxf(clicAbsMouse.y,absMouse.y)))
selectedCurves.push_back(*curve);
curve++;
}
std::list<OE_stitchs*>::iterator stitch = document->stitchs.begin();
while (stitch != document->stitchs.end())
{
(*stitch)->getBound(&tmpXMin, &tmpYMin, &tmpXMax, &tmpYMax);
if ((tmpXMin>minf(clicAbsMouse.x,absMouse.x))
&&(tmpXMax<maxf(clicAbsMouse.x,absMouse.x))
&&(tmpYMin>minf(clicAbsMouse.y,absMouse.y))
&&(tmpYMax<maxf(clicAbsMouse.y,absMouse.y)))
selectedStitches.push_back(*stitch);
stitch++;
}
bool OE_interfaceDisplay::checkPicking()
{
if(selControlStitch)
{
OE_linestitch* tmpLineStitch = dynamic_cast<OE_linestitch*>(selControlStitch);
editSubcurve = tmpLineStitch->getJoincurve()->getCurve(selControlIndex/2);
vector_2dt pos = editSubcurve->getCurve()->closestPoint(absMouse);
// and add an action to move the point
controller->addAction(new OE_actionSetSubcurvePos( editSubcurve, selControlIndex%2, pos.t));
return true;
}
selControlStitch = 0;
}
if(selControlCurve)
{
OE_pointcurve* tmpPointCurve = dynamic_cast<OE_pointcurve*>(selControlCurve);
selection = false;
if ((tmpPointCurve)&&(selControlType == 0))
{
return controller->addAction(new OE_actionMovePointCurve( tmpPointCurve, selControlIndex + 1, vector_2d(0,0)));
}
selControlCurve = 0;
}
}
/** \brief set selection
*
* \return true if all is ok
*
*/
bool OE_interfaceDisplay::select()
{
GLuint selectBuf[64];
GLint hits;
glSelectBuffer(64, selectBuf);
draw(true);
hits = glRenderMode(GL_RENDER);
selectedCurves.clear();
selectedStitches.clear();
GLuint i, nb_names, name, *ptr;
std::cout<<"hits : "<<std::endl;
ptr = (GLuint *) selectBuf;
for (GLuint i = 0; i < hits; i++) // for each hit
{
if (names==4) //if name count correspond to a sub element (like control point)
ptr+=3; //jump near and far info and goto first index
if (*ptr == 1) //stitches index
{
ptr++;
if(*ptr < document->selectedStitchs.size())
{
std::list<OE_stitchs*>::iterator it = document->selectedStitchs.begin();
std::advance(it, *ptr++);
selControlStitch = *it;
selControlType = *ptr++;
selControlIndex = *ptr++;
return true;
}else
ptr+=4;
}else if (*ptr == 2) //curves index
{
ptr++;
if(*ptr < document->selectedCurves.size())
{
std::list<OE_curve*>::iterator it = document->selectedCurves.begin();
std::advance(it, *ptr++);
selControlCurve = *it;
selControlType = *ptr++;
selControlIndex = *ptr++;
return true;
}else
ptr+=4;
}
}else if (names==2) //if name count correspond to a simple element
{
ptr+=3; //jump near and far info and goto first index
if (*ptr == 0) //tools index
{
ptr++;
if(*ptr == 0)
std::cout<<" zero ";
ptr++;
}else if (*ptr == 1) //stitches index
{
ptr++;
if (document->stitchs.size() > *ptr)
{
std::list<OE_stitchs*>::iterator it = document->stitchs.begin();
std::advance(it, *ptr);
selectedStitches.push_back(*it);
}
}else if (*ptr == 2) //curves index
{
ptr++;
std::cout<<" curve "<<*ptr;
if (document->curves.size() > *ptr)
{
std::list<OE_curve*>::iterator it = document->curves.begin();
std::advance(it, *ptr);
selectedCurves.push_back(*it);
}
ptr++;
}else
{
ptr++;
std::cout<<" unrecognized item "<<*ptr;ptr++;
}
std::cout<<std::endl;
}
else
{
std::cout<<" strange hit! probably a bug : "<< names<<std::endl; ptr+=3;
std::cout<<" names are ";
for (GLuint j = 0; j < names; j++) { /* for each name */
std::cout<< *ptr<<" ";
ptr++;
}
std::cout<<std::endl;
}
}
return true;
}
bool OE_interfaceDisplay::selectApply(std::list<OE_curve*> selectedCurves, std::list<OE_stitchs*> selectedStitches)
{
if(selectedCurves.size())
if(modState&GLFW_MOD_CONTROL)
controller->unselectCurves(selectedCurves);
else
controller->selectCurves(selectedCurves, !(modState&GLFW_MOD_SHIFT));
if(selectedStitches.size())
if(modState&GLFW_MOD_CONTROL)
controller->unselectStitches(selectedStitches);
else
controller->selectStitches(selectedStitches, !(modState&GLFW_MOD_SHIFT));
3dsman
committed
if(!(selectedStitches.size())&&!(selectedCurves.size()))
{
controller->clearSelection();
}
/** \brief draw the document on screen
*
* \return true if all is ok
*
*/
bool OE_interfaceDisplay::draw()
{
draw(false);
}
/** \brief draw the document on screen
*
* \return true if all is ok
*
*/
bool OE_interfaceDisplay::draw(bool select)
{
if (select)
{
glRenderMode(GL_SELECT);
glInitNames();
glPushName(0);
}
float wzoom, hzoom;
glViewport(0, 0, width, height);
glClearColor(220.0f/255.0f, 220.0f/255.0f, 220.0f/255.0f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glDisable(GL_TEXTURE_2D);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
wzoom = (float)width*zoom;
hzoom = (float)height*zoom;
glOrtho(absMouse.x-7*zoom, absMouse.x+7*zoom, absMouse.y+7*zoom, absMouse.y-7*zoom, -1, 1);
glOrtho(viewPos.x-wzoom, viewPos.x+wzoom, viewPos.y+hzoom, viewPos.y-hzoom, -1, 1);
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
glDisable(GL_DEPTH_TEST);
glColor4ub(255,255,255,255);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA,GL_ONE_MINUS_SRC_ALPHA);
if (document && curDisplay)
{
if(select && (curDisplay == editDisplay))
editDisplay->select(document->selectedCurves, document->selectedStitchs);
else
curDisplay->draw();
}
if(!select)
{
{
// Points
glPointSize(5.0f);
glBegin(GL_POINTS);
glColor4d(1.0f, 0.0f, 0.0f, 1.0f);
glVertex2f(editClosestPoint.v.x,editClosestPoint.v.y);
glVertex2f(absMouse.x,-10000);
glVertex2f(absMouse.x,10000);
glVertex2f(-10000,absMouse.y);
glVertex2f(10000,absMouse.y);
//cursor select zone
glLineWidth(2);
glBegin(GL_LINE_STRIP);
glColor4d(0.0f, 0.0f, 0.0f, 0.3f);
glVertex2f(absMouse.x-5*zoom,absMouse.y+5*zoom);
glVertex2f(absMouse.x-5*zoom,absMouse.y-5*zoom);
glVertex2f(absMouse.x+5*zoom,absMouse.y-5*zoom);
glVertex2f(absMouse.x+5*zoom,absMouse.y+5*zoom);
glVertex2f(absMouse.x-5*zoom,absMouse.y+5*zoom);
vector_2d clicAbsMouse = viewPos+(clicOldMouse-vector_2d(width,height)/2)*zoom*2;
//float clicAbsMouseX = viewPos.x+(clicOldMouse.x-width/2)*zoom*2;
//float clicAbsMouseY = viewPos.y+(clicOldMouse.y-height/2)*zoom*2;
glLineWidth(2);
glBegin(GL_LINE_STRIP);
glColor4d(0.0f, 0.0f, 0.0f, 0.3f);
glVertex2f(absMouse.x,clicAbsMouse.y);
glVertex2f(absMouse.x,absMouse.y);
glVertex2f(clicAbsMouse.x,absMouse.y);
glVertex2f(clicAbsMouse.x,clicAbsMouse.y);
glVertex2f(absMouse.x,clicAbsMouse.y);