#include "solvespace.h" const hEntity Entity::FREE_IN_3D = { 0 }; const hGroup Group::HGROUP_REFERENCES = { 1 }; const hRequest Request::HREQUEST_REFERENCE_XY = { 1 }; const hRequest Request::HREQUEST_REFERENCE_YZ = { 2 }; const hRequest Request::HREQUEST_REFERENCE_ZX = { 3 }; void Group::AddParam(IdList *param, hParam hp, double v) { Param pa; memset(&pa, 0, sizeof(pa)); pa.h = hp; pa.val = v; param->Add(&pa); } void Group::MenuGroup(int id) { Group g; memset(&g, 0, sizeof(g)); g.visible = true; switch(id) { case GraphicsWindow::MNU_GROUP_DRAWING: g.type = DRAWING; g.name.strcpy("drawing"); break; case GraphicsWindow::MNU_GROUP_EXTRUDE: g.type = EXTRUDE; g.opA.v = 2; g.name.strcpy("extrude"); break; default: oops(); } SS.group.AddAndAssignId(&g); SS.GenerateAll(SS.GW.solving == GraphicsWindow::SOLVE_ALWAYS); SS.GW.activeGroup = g.h; SS.TW.Show(); } char *Group::DescriptionString(void) { static char ret[100]; if(name.str[0]) { sprintf(ret, "g%04x-%s", h.v, name.str); } else { sprintf(ret, "g%04x-(unnamed)", h.v); } return ret; } void Group::Generate(IdList *entity, IdList *param) { int i; switch(type) { case DRAWING: return; case EXTRUDE: AddParam(param, h.param(0), 50); AddParam(param, h.param(1), 50); AddParam(param, h.param(2), 50); for(i = 0; i < entity->n; i++) { Entity *e = &(entity->elem[i]); if(e->group.v != opA.v) continue; CopyEntity(e->h, 0, h.param(0), h.param(1), h.param(2)); } break; default: oops(); } } hEntity Group::Remap(hEntity in, int copyNumber) { int i; for(i = 0; i < remap.n; i++) { EntityMap *em = &(remap.elem[i]); if(em->input.v == in.v && em->copyNumber == copyNumber) { // We already have a mapping for this entity. return h.entity(em->h.v); } } // We don't have a mapping yet, so create one. EntityMap em; em.input = in; em.copyNumber = copyNumber; remap.AddAndAssignId(&em); return h.entity(em.h.v); } void Group::CopyEntity(hEntity in, int a, hParam dx, hParam dy, hParam dz) { Entity *ep = SS.GetEntity(in); Entity en; memset(&en, 0, sizeof(en)); en.type = ep->type; en.h = Remap(ep->h, a); en.group = h; switch(ep->type) { case Entity::WORKPLANE: // Don't copy these. return; case Entity::LINE_SEGMENT: en.point[0] = Remap(ep->point[0], a); en.point[1] = Remap(ep->point[1], a); break; case Entity::CUBIC: en.point[0] = Remap(ep->point[0], a); en.point[1] = Remap(ep->point[1], a); en.point[2] = Remap(ep->point[2], a); en.point[3] = Remap(ep->point[3], a); break; case Entity::POINT_IN_3D: case Entity::POINT_IN_2D: en.type = Entity::POINT_XFRMD; en.point[0] = ep->h; en.param[0] = dx; en.param[1] = dy; en.param[2] = dz; break; default: oops(); } SS.entity.Add(&en); } void Group::Draw(void) { edges.l.Clear(); int i; for(i = 0; i < SS.entity.n; i++) { Entity *e = &(SS.entity.elem[i]); if(e->group.v != h.v) continue; e->GenerateEdges(&edges); } SPolygon poly; memset(&poly, 0, sizeof(poly)); SEdge error; if(edges.AssemblePolygon(&poly, &error)) { glxColor4d(0, 0, 1, 0.1); glxFillPolygon(&poly); } else { glxColor4d(1, 0, 0, 0.2); glLineWidth(10); glBegin(GL_LINES); glxVertex3v(error.a); glxVertex3v(error.b); glEnd(); glLineWidth(1); glxColor3d(1, 0, 0); glPushMatrix(); glxTranslatev(error.b); glxOntoWorkplane(SS.GW.projRight, SS.GW.projUp); glxWriteText("not closed contour!"); glPopMatrix(); } poly.Clear(); } hParam Request::AddParam(IdList *param, hParam hp) { Param pa; memset(&pa, 0, sizeof(pa)); pa.h = hp; param->Add(&pa); return hp; } void Request::Generate(IdList *entity, IdList *param) { int points = 0; int params = 0; int et = 0; int i; Group *g = SS.group.FindById(group); Entity e; memset(&e, 0, sizeof(e)); switch(type) { case Request::WORKPLANE: et = Entity::WORKPLANE; points = 1; params = 4; goto c; case Request::DATUM_POINT: et = 0; points = 1; params = 0; goto c; case Request::LINE_SEGMENT: et = Entity::LINE_SEGMENT; points = 2; params = 0; goto c; case Request::CUBIC: et = Entity::CUBIC; points = 4; params = 0; goto c; c: { // Generate the entity that's specific to this request. e.type = et; e.group = group; e.h = h.entity(0); // And generate entities for the points for(i = 0; i < points; i++) { Entity p; memset(&p, 0, sizeof(p)); p.workplane = workplane; // points start from entity 1, except for datum point case p.h = h.entity(i+(et ? 1 : 0)); p.group = group; if(workplane.v == Entity::FREE_IN_3D.v) { p.type = Entity::POINT_IN_3D; // params for x y z p.param[0] = AddParam(param, h.param(16 + 3*i + 0)); p.param[1] = AddParam(param, h.param(16 + 3*i + 1)); p.param[2] = AddParam(param, h.param(16 + 3*i + 2)); } else { p.type = Entity::POINT_IN_2D; // params for u v p.param[0] = AddParam(param, h.param(16 + 3*i + 0)); p.param[1] = AddParam(param, h.param(16 + 3*i + 1)); } entity->Add(&p); e.point[i] = p.h; } // And generate any params not associated with the point that // we happen to need. for(i = 0; i < params; i++) { e.param[i] = AddParam(param, h.param(i)); } if(et) entity->Add(&e); break; } default: oops(); } } char *Request::DescriptionString(void) { static char ret[100]; if(name.str[0]) { sprintf(ret, "r%03x-%s", h.v, name.str); } else { sprintf(ret, "r%03x-(unnamed)", h.v); } return ret; }