//----------------------------------------------------------------------------- // The screens when an entity is selected, that show some description of it-- // endpoints of the lines, diameter of the circle, etc. // // Copyright 2008-2013 Jonathan Westhues. //----------------------------------------------------------------------------- #include "solvespace.h" void TextWindow::ScreenUnselectAll(int link, uint32_t v) { GraphicsWindow::MenuEdit(Command::UNSELECT_ALL); } void TextWindow::ScreenEditTtfText(int link, uint32_t v) { hRequest hr = { v }; Request *r = SK.GetRequest(hr); SS.TW.ShowEditControl(10, r->str); SS.TW.edit.meaning = Edit::TTF_TEXT; SS.TW.edit.request = hr; } void TextWindow::ScreenSetTtfFont(int link, uint32_t v) { int i = (int)v; if(i < 0) return; if(i >= SS.fonts.l.n) return; SS.GW.GroupSelection(); auto const &gs = SS.GW.gs; if(gs.entities != 1 || gs.n != 1) return; Entity *e = SK.entity.FindByIdNoOops(gs.entity[0]); if(!e || e->type != Entity::Type::TTF_TEXT || !e->h.isFromRequest()) return; Request *r = SK.request.FindByIdNoOops(e->h.request()); if(!r) return; SS.UndoRemember(); r->font = SS.fonts.l.elem[i].FontFileBaseName(); SS.MarkGroupDirty(r->group); SS.ScheduleShowTW(); } void TextWindow::ScreenConstraintShowAsRadius(int link, uint32_t v) { hConstraint hc = { v }; Constraint *c = SK.GetConstraint(hc); SS.UndoRemember(); c->other = !c->other; SS.ScheduleShowTW(); } void TextWindow::DescribeSelection() { Printf(false, ""); auto const &gs = SS.GW.gs; if(gs.n == 1 && (gs.points == 1 || gs.entities == 1)) { Entity *e = SK.GetEntity(gs.points == 1 ? gs.point[0] : gs.entity[0]); Vector p; #define COSTR(p) \ SS.MmToString((p).x).c_str(), \ SS.MmToString((p).y).c_str(), \ SS.MmToString((p).z).c_str() #define PT_AS_STR "(%Fi%s%E, %Fi%s%E, %Fi%s%E)" #define PT_AS_NUM "(%Fi%3%E, %Fi%3%E, %Fi%3%E)" switch(e->type) { case Entity::Type::POINT_IN_3D: case Entity::Type::POINT_IN_2D: case Entity::Type::POINT_N_TRANS: case Entity::Type::POINT_N_ROT_TRANS: case Entity::Type::POINT_N_COPY: case Entity::Type::POINT_N_ROT_AA: p = e->PointGetNum(); Printf(false, "%FtPOINT%E at " PT_AS_STR, COSTR(p)); break; case Entity::Type::NORMAL_IN_3D: case Entity::Type::NORMAL_IN_2D: case Entity::Type::NORMAL_N_COPY: case Entity::Type::NORMAL_N_ROT: case Entity::Type::NORMAL_N_ROT_AA: { Quaternion q = e->NormalGetNum(); p = q.RotationN(); Printf(false, "%FtNORMAL / COORDINATE SYSTEM%E"); Printf(true, " basis n = " PT_AS_NUM, CO(p)); p = q.RotationU(); Printf(false, " u = " PT_AS_NUM, CO(p)); p = q.RotationV(); Printf(false, " v = " PT_AS_NUM, CO(p)); break; } case Entity::Type::WORKPLANE: { p = SK.GetEntity(e->point[0])->PointGetNum(); Printf(false, "%FtWORKPLANE%E"); Printf(true, " origin = " PT_AS_STR, COSTR(p)); Quaternion q = e->Normal()->NormalGetNum(); p = q.RotationN(); Printf(true, " normal = " PT_AS_NUM, CO(p)); break; } case Entity::Type::LINE_SEGMENT: { Vector p0 = SK.GetEntity(e->point[0])->PointGetNum(); p = p0; Printf(false, "%FtLINE SEGMENT%E"); Printf(true, " thru " PT_AS_STR, COSTR(p)); Vector p1 = SK.GetEntity(e->point[1])->PointGetNum(); p = p1; Printf(false, " " PT_AS_STR, COSTR(p)); Printf(true, " len = %Fi%s%E", SS.MmToString((p1.Minus(p0).Magnitude())).c_str()); break; } case Entity::Type::CUBIC_PERIODIC: case Entity::Type::CUBIC: int pts; if(e->type == Entity::Type::CUBIC_PERIODIC) { Printf(false, "%FtPERIODIC C2 CUBIC SPLINE%E"); pts = (3 + e->extraPoints); } else if(e->extraPoints > 0) { Printf(false, "%FtINTERPOLATING C2 CUBIC SPLINE%E"); pts = (4 + e->extraPoints); } else { Printf(false, "%FtCUBIC BEZIER CURVE%E"); pts = 4; } for(int i = 0; i < pts; i++) { p = SK.GetEntity(e->point[i])->PointGetNum(); Printf((i==0), " p%d = " PT_AS_STR, i, COSTR(p)); } break; case Entity::Type::ARC_OF_CIRCLE: { Printf(false, "%FtARC OF A CIRCLE%E"); p = SK.GetEntity(e->point[0])->PointGetNum(); Printf(true, " center = " PT_AS_STR, COSTR(p)); p = SK.GetEntity(e->point[1])->PointGetNum(); Printf(true, " endpoints = " PT_AS_STR, COSTR(p)); p = SK.GetEntity(e->point[2])->PointGetNum(); Printf(false, " " PT_AS_STR, COSTR(p)); double r = e->CircleGetRadiusNum(); Printf(true, " diameter = %Fi%s", SS.MmToString(r*2).c_str()); Printf(false, " radius = %Fi%s", SS.MmToString(r).c_str()); double thetas, thetaf, dtheta; e->ArcGetAngles(&thetas, &thetaf, &dtheta); Printf(false, " arc len = %Fi%s", SS.MmToString(dtheta*r).c_str()); break; } case Entity::Type::CIRCLE: { Printf(false, "%FtCIRCLE%E"); p = SK.GetEntity(e->point[0])->PointGetNum(); Printf(true, " center = " PT_AS_STR, COSTR(p)); double r = e->CircleGetRadiusNum(); Printf(true, " diameter = %Fi%s", SS.MmToString(r*2).c_str()); Printf(false, " radius = %Fi%s", SS.MmToString(r).c_str()); break; } case Entity::Type::FACE_NORMAL_PT: case Entity::Type::FACE_XPROD: case Entity::Type::FACE_N_ROT_TRANS: case Entity::Type::FACE_N_ROT_AA: case Entity::Type::FACE_N_TRANS: Printf(false, "%FtPLANE FACE%E"); p = e->FaceGetNormalNum(); Printf(true, " normal = " PT_AS_NUM, CO(p)); p = e->FaceGetPointNum(); Printf(false, " thru = " PT_AS_STR, COSTR(p)); break; case Entity::Type::TTF_TEXT: { Printf(false, "%FtTRUETYPE FONT TEXT%E"); Printf(true, " font = '%Fi%s%E'", e->font.c_str()); if(e->h.isFromRequest()) { Printf(false, " text = '%Fi%s%E' %Fl%Ll%f%D[change]%E", e->str.c_str(), &ScreenEditTtfText, e->h.request().v); Printf(true, " select new font"); SS.fonts.LoadAll(); int i; for(i = 0; i < SS.fonts.l.n; i++) { TtfFont *tf = &(SS.fonts.l.elem[i]); if(e->font == tf->FontFileBaseName()) { Printf(false, "%Bp %s", (i & 1) ? 'd' : 'a', tf->name.c_str()); } else { Printf(false, "%Bp %f%D%Fl%Ll%s%E%Bp", (i & 1) ? 'd' : 'a', &ScreenSetTtfFont, i, tf->name.c_str(), (i & 1) ? 'd' : 'a'); } } } else { Printf(false, " text = '%Fi%s%E'", e->str.c_str()); } break; } case Entity::Type::IMAGE: { Printf(false, "%FtIMAGE%E"); Platform::Path relativePath = e->file.RelativeTo(SS.saveFile.Parent()); if(relativePath.IsEmpty()) { Printf(true, " file = '%Fi%s%E'", e->file.raw.c_str()); } else { Printf(true, " file = '%Fi%s%E'", relativePath.raw.c_str()); } break; } default: Printf(true, "%Ft?? ENTITY%E"); break; } Group *g = SK.GetGroup(e->group); Printf(false, ""); Printf(false, "%FtIN GROUP%E %s", g->DescriptionString().c_str()); if(e->workplane.v == Entity::FREE_IN_3D.v) { Printf(false, "%FtNOT LOCKED IN WORKPLANE%E"); } else { Entity *w = SK.GetEntity(e->workplane); Printf(false, "%FtIN WORKPLANE%E %s", w->DescriptionString().c_str()); } if(e->style.v) { Style *s = Style::Get(e->style); Printf(false, "%FtIN STYLE%E %s", s->DescriptionString().c_str()); } else { Printf(false, "%FtIN STYLE%E none"); } if(e->construction) { Printf(false, "%FtCONSTRUCTION"); } std::vector lhc = {}; for(const Constraint &c : SK.constraint) { if(!(c.ptA.v == e->h.v || c.ptB.v == e->h.v || c.entityA.v == e->h.v || c.entityB.v == e->h.v || c.entityC.v == e->h.v || c.entityD.v == e->h.v)) continue; lhc.push_back(c.h); } if(!lhc.empty()) { Printf(true, "%FtCONSTRAINED BY:%E"); int a = 0; for(hConstraint hc : lhc) { Constraint *c = SK.GetConstraint(hc); std::string s = c->DescriptionString(); Printf(false, "%Bp %Fl%Ll%D%f%h%s%E %s", (a & 1) ? 'd' : 'a', c->h.v, (&TextWindow::ScreenSelectConstraint), (&TextWindow::ScreenHoverConstraint), s.c_str(), c->reference ? "(ref)" : ""); a++; } } } else if(gs.n == 2 && gs.points == 2) { Printf(false, "%FtTWO POINTS"); Vector p0 = SK.GetEntity(gs.point[0])->PointGetNum(); Printf(true, " at " PT_AS_STR, COSTR(p0)); Vector p1 = SK.GetEntity(gs.point[1])->PointGetNum(); Printf(false, " " PT_AS_STR, COSTR(p1)); double d = (p1.Minus(p0)).Magnitude(); Printf(true, " d = %Fi%s", SS.MmToString(d).c_str()); } else if(gs.n == 2 && gs.points == 1 && gs.circlesOrArcs == 1) { Entity *ec = SK.GetEntity(gs.entity[0]); if(ec->type == Entity::Type::CIRCLE) { Printf(false, "%FtPOINT AND A CIRCLE"); } else if(ec->type == Entity::Type::ARC_OF_CIRCLE) { Printf(false, "%FtPOINT AND AN ARC"); } else ssassert(false, "Unexpected entity type"); Vector p = SK.GetEntity(gs.point[0])->PointGetNum(); Printf(true, " pt at " PT_AS_STR, COSTR(p)); Vector c = SK.GetEntity(ec->point[0])->PointGetNum(); Printf(true, " center = " PT_AS_STR, COSTR(c)); double r = ec->CircleGetRadiusNum(); Printf(false, " diameter = %Fi%s", SS.MmToString(r*2).c_str()); Printf(false, " radius = %Fi%s", SS.MmToString(r).c_str()); double d = (p.Minus(c)).Magnitude() - r; Printf(true, " distance = %Fi%s", SS.MmToString(d).c_str()); } else if(gs.n == 2 && gs.faces == 1 && gs.points == 1) { Printf(false, "%FtA POINT AND A PLANE FACE"); Vector pt = SK.GetEntity(gs.point[0])->PointGetNum(); Printf(true, " point = " PT_AS_STR, COSTR(pt)); Vector n = SK.GetEntity(gs.face[0])->FaceGetNormalNum(); Printf(true, " plane normal = " PT_AS_NUM, CO(n)); Vector pl = SK.GetEntity(gs.face[0])->FaceGetPointNum(); Printf(false, " plane thru = " PT_AS_STR, COSTR(pl)); double dd = n.Dot(pl) - n.Dot(pt); Printf(true, " distance = %Fi%s", SS.MmToString(dd).c_str()); } else if(gs.n == 3 && gs.points == 2 && gs.vectors == 1) { Printf(false, "%FtTWO POINTS AND A VECTOR"); Vector p0 = SK.GetEntity(gs.point[0])->PointGetNum(); Printf(true, " pointA = " PT_AS_STR, COSTR(p0)); Vector p1 = SK.GetEntity(gs.point[1])->PointGetNum(); Printf(false, " pointB = " PT_AS_STR, COSTR(p1)); Vector v = SK.GetEntity(gs.vector[0])->VectorGetNum(); v = v.WithMagnitude(1); Printf(true, " vector = " PT_AS_NUM, CO(v)); double d = (p1.Minus(p0)).Dot(v); Printf(true, " proj_d = %Fi%s", SS.MmToString(d).c_str()); } else if(gs.n == 2 && gs.lineSegments == 1 && gs.points == 1) { Entity *ln = SK.GetEntity(gs.entity[0]); Vector lp0 = SK.GetEntity(ln->point[0])->PointGetNum(), lp1 = SK.GetEntity(ln->point[1])->PointGetNum(); Printf(false, "%FtLINE SEGMENT AND POINT%E"); Printf(true, " ln thru " PT_AS_STR, COSTR(lp0)); Printf(false, " " PT_AS_STR, COSTR(lp1)); Entity *p = SK.GetEntity(gs.point[0]); Vector pp = p->PointGetNum(); Printf(true, " point " PT_AS_STR, COSTR(pp)); Printf(true, " pt-ln distance = %Fi%s%E", SS.MmToString(pp.DistanceToLine(lp0, lp1.Minus(lp0))).c_str()); hEntity wrkpl = SS.GW.ActiveWorkplane(); if(wrkpl.v != Entity::FREE_IN_3D.v && !(p->workplane.v == wrkpl.v && ln->workplane.v == wrkpl.v)) { Vector ppw = pp.ProjectInto(wrkpl); Vector lp0w = lp0.ProjectInto(wrkpl); Vector lp1w = lp1.ProjectInto(wrkpl); Printf(false, " or distance = %Fi%s%E (in workplane)", SS.MmToString(ppw.DistanceToLine(lp0w, lp1w.Minus(lp0w))).c_str()); } } else if(gs.n == 2 && gs.vectors == 2) { Printf(false, "%FtTWO VECTORS"); Vector v0 = SK.GetEntity(gs.entity[0])->VectorGetNum(), v1 = SK.GetEntity(gs.entity[1])->VectorGetNum(); v0 = v0.WithMagnitude(1); v1 = v1.WithMagnitude(1); Printf(true, " vectorA = " PT_AS_NUM, CO(v0)); Printf(false, " vectorB = " PT_AS_NUM, CO(v1)); double theta = acos(v0.Dot(v1)); Printf(true, " angle = %Fi%2%E degrees", theta*180/PI); while(theta < PI/2) theta += PI; while(theta > PI/2) theta -= PI; Printf(false, " or angle = %Fi%2%E (mod 180)", theta*180/PI); } else if(gs.n == 2 && gs.faces == 2) { Printf(false, "%FtTWO PLANE FACES"); Vector n0 = SK.GetEntity(gs.face[0])->FaceGetNormalNum(); Printf(true, " planeA normal = " PT_AS_NUM, CO(n0)); Vector p0 = SK.GetEntity(gs.face[0])->FaceGetPointNum(); Printf(false, " planeA thru = " PT_AS_STR, COSTR(p0)); Vector n1 = SK.GetEntity(gs.face[1])->FaceGetNormalNum(); Printf(true, " planeB normal = " PT_AS_NUM, CO(n1)); Vector p1 = SK.GetEntity(gs.face[1])->FaceGetPointNum(); Printf(false, " planeB thru = " PT_AS_STR, COSTR(p1)); double theta = acos(n0.Dot(n1)); Printf(true, " angle = %Fi%2%E degrees", theta*180/PI); while(theta < PI/2) theta += PI; while(theta > PI/2) theta -= PI; Printf(false, " or angle = %Fi%2%E (mod 180)", theta*180/PI); if(fabs(theta) < 0.01) { double d = (p1.Minus(p0)).Dot(n0); Printf(true, " distance = %Fi%s", SS.MmToString(d).c_str()); } } else if(gs.n == 0 && gs.stylables > 0) { Printf(false, "%FtSELECTED:%E comment text"); } else if(gs.n == 0 && gs.constraints == 1) { Constraint *c = SK.GetConstraint(gs.constraint[0]); if(c->type == Constraint::Type::DIAMETER) { Printf(false, "%FtDIAMETER CONSTRAINT"); Printf(true, " %Fd%f%D%Ll%s show as radius", &ScreenConstraintShowAsRadius, gs.constraint[0].v, c->other ? CHECK_TRUE : CHECK_FALSE); } else { Printf(false, "%FtSELECTED:%E %s", c->DescriptionString().c_str()); } std::vector lhe = {}; lhe.push_back(c->ptA); lhe.push_back(c->ptB); lhe.push_back(c->entityA); lhe.push_back(c->entityB); lhe.push_back(c->entityC); lhe.push_back(c->entityD); auto it = std::remove_if(lhe.begin(), lhe.end(), [](hEntity he) { return he.v == Entity::NO_ENTITY.v || !he.isFromRequest(); }); lhe.erase(it, lhe.end()); if(!lhe.empty()) { Printf(true, "%FtCONSTRAINS:%E"); int a = 0; for(hEntity he : lhe) { Request *r = SK.GetRequest(he.request()); std::string s = r->DescriptionString(); Printf(false, "%Bp %Fl%Ll%D%f%h%s%E", (a & 1) ? 'd' : 'a', r->h.v, (&TextWindow::ScreenSelectRequest), &(TextWindow::ScreenHoverRequest), s.c_str()); a++; } } } else { int n = SS.GW.selection.n; Printf(false, "%FtSELECTED:%E %d item%s", n, n == 1 ? "" : "s"); } if(shown.screen == Screen::STYLE_INFO && shown.style.v >= Style::FIRST_CUSTOM && gs.stylables > 0) { // If we are showing a screen for a particular style, then offer the // option to assign our selected entities to that style. Style *s = Style::Get(shown.style); Printf(true, "%Fl%D%f%Ll(assign to style %s)%E", shown.style.v, &ScreenAssignSelectionToStyle, s->DescriptionString().c_str()); } // If any of the selected entities have an assigned style, then offer // the option to remove that style. bool styleAssigned = false; for(int i = 0; i < gs.entities; i++) { Entity *e = SK.GetEntity(gs.entity[i]); if(e->style.v != 0) { styleAssigned = true; } } for(int i = 0; i < gs.constraints; i++) { Constraint *c = SK.GetConstraint(gs.constraint[i]); if(c->type == Constraint::Type::COMMENT && c->disp.style.v != 0) { styleAssigned = true; } } if(styleAssigned) { Printf(true, "%Fl%D%f%Ll(remove assigned style)%E", 0, &ScreenAssignSelectionToStyle); } Printf(true, "%Fl%f%Ll(unselect all)%E", &TextWindow::ScreenUnselectAll); } void TextWindow::GoToScreen(Screen screen) { shown.screen = screen; }