/* * Copyright (c) 2008 Hypertriton, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE * USE OF THIS SOFTWARE EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* * Wire entity. Displays connections between ports. */ #include "core.h" #include #include ES_SchemWire * ES_SchemWireNew(void *pNode, void *p1, void *p2) { ES_SchemWire *sw; sw = AG_Malloc(sizeof(ES_SchemWire)); VG_NodeInit(sw, &esSchemWireOps); sw->p1 = VGNODE(p1); sw->p2 = VGNODE(p2); VG_NodeAttach(pNode, sw); VG_AddRef(sw, sw->p1); VG_AddRef(sw, sw->p2); return (sw); } static void Init(void *p) { ES_SchemWire *sw = p; sw->p1 = NULL; sw->p2 = NULL; sw->thickness = 1; sw->name[0] = '\0'; sw->wire = NULL; } static int Load(void *p, AG_DataSource *ds, const AG_Version *ver) { ES_SchemWire *sw = p; AG_CopyString(sw->name, ds, sizeof(sw->name)); if ((sw->p1 = VG_ReadRef(ds, sw, "SchemPort")) == NULL || (sw->p2 = VG_ReadRef(ds, sw, "SchemPort")) == NULL) { return (-1); } sw->thickness = (Uint)AG_ReadUint32(ds); return (0); } static void Save(void *p, AG_DataSource *ds) { ES_SchemWire *sw = p; AG_WriteString(ds, OBJECT(sw->wire)->name); VG_WriteRef(ds, sw->p1); VG_WriteRef(ds, sw->p2); AG_WriteUint32(ds, (Uint32)sw->thickness); } static void Draw(void *p, VG_View *vv) { ES_SchemWire *sw = p; VG_Vector v1 = VG_Pos(sw->p1); VG_Vector v2 = VG_Pos(sw->p2); AG_Color c; int x1, y1, x2, y2; VG_GetViewCoords(vv, v1, &x1, &y1); VG_GetViewCoords(vv, v2, &x2, &y2); c = VG_MapColorRGB(VGNODE(sw)->color); AG_DrawLine(vv, x1,y1, x2,y2, &c); } static void Extent(void *p, VG_View *vv, VG_Vector *a, VG_Vector *b) { ES_SchemWire *sw = p; VG_Vector p1, p2; p1 = VG_Pos(sw->p1); p2 = VG_Pos(sw->p2); a->x = MIN(p1.x, p2.x); a->y = MIN(p1.y, p2.y); b->x = MAX(p1.x, p2.x); b->y = MAX(p1.y, p2.y); } static float PointProximity(void *p, VG_View *vv, VG_Vector *vPt) { ES_SchemWire *sw = p; VG_Vector v1 = VG_Pos(sw->p1); VG_Vector v2 = VG_Pos(sw->p2); return VG_PointLineDistance(v1, v2, vPt); } static void Delete(void *p) { ES_SchemWire *sw = p; if (VG_DelRef(sw, sw->p1) == 0) { VG_Delete(sw->p1); } if (VG_DelRef(sw, sw->p2) == 0) { VG_Delete(sw->p2); } } static void Move(void *p, VG_Vector vCurs, VG_Vector vRel) { ES_SchemWire *sw = p; VG_Matrix T1, T2; T1 = sw->p1->T; T2 = sw->p2->T; VG_LoadIdentity(sw->p1); VG_LoadIdentity(sw->p2); VG_Translate(sw->p1, vRel); VG_Translate(sw->p2, vRel); VG_MultMatrix(&sw->p1->T, &T1); VG_MultMatrix(&sw->p2->T, &T2); } static void * Edit(void *p, VG_View *vv) { ES_SchemWire *sw = p; AG_Box *box = AG_BoxNewVert(NULL, AG_BOX_EXPAND); #if 0 AG_LabelNewPolled(box, 0, _("Component: %s"), &sw->name); AG_SeparatorNewHoriz(box); #endif AG_NumericalNewUint(box, 0, "px", _("Thickness: "), &sw->thickness); return (box); } VG_NodeOps esSchemWireOps = { N_("SchemWire"), &esIconInsertWire, sizeof(ES_SchemWire), Init, NULL, /* destroy */ Load, Save, Draw, Extent, PointProximity, NULL, /* lineProximity */ Delete, Move, Edit };