/********************************************************/ /* Module to handle libraries (second part - drawing ). */ /********************************************************/ #include "fctsys.h" #include "gr_basic.h" #include "common.h" #include "program.h" #include "libcmp.h" #include "component_class.h" #include "general.h" #include "trigo.h" #include "protos.h" #define UNVISIBLE_COLOR DARKGRAY //#define DRAW_ARC_WITH_ANGLE // pour trace arc avec routine utilsant les angles /* Fonctions locales */ /* Description du composant utilisé lorsque des composants non trouvés en librarie doivent etre dessinés */ /* ce composant est un rectangle de 400 mils de coté avec le texte ?? au centre DEF DUMMY U 0 40 Y Y 1 0 N F0 "U" 0 -350 60 H V F1 "DUMMY" 0 350 60 H V DRAW T 0 0 0 150 0 0 0 ?? S -200 200 200 -200 0 1 0 ENDDRAW ENDDEF */ static int s_ItemSelectColor = BROWN; static EDA_LibComponentStruct * DummyCmp; static int * Buf_Poly_Drawings, Buf_Poly_Size; // Used fo polyline drawings static void DrawLibPartAux(WinEDA_DrawPanel * panel, wxDC * DC, EDA_SchComponentStruct *Component, EDA_LibComponentStruct *Entry, const wxPoint & Pos, int TransMat[2][2], int Multi, int convert, int DrawMode, int Color = -1, bool DrawPinText = TRUE); /******************************/ static void CreateDummyCmp(void) /******************************/ { DummyCmp = new EDA_LibComponentStruct( NULL); LibDrawSquare * Square = new LibDrawSquare(); Square->m_Start = wxPoint(- 200,200); Square->m_End = wxPoint(200, - 200); Square->m_Width = 4; LibDrawText * Text = new LibDrawText(); Text->m_Size.x = Text->m_Size.y = 150; Text->m_Text = wxT("??"); DummyCmp->m_Drawings = Square; Square->Pnext = Text; } /*************************************************************/ void DrawLibEntry(WinEDA_DrawPanel * panel,wxDC * DC, EDA_LibComponentStruct *LibEntry, int posX, int posY, int Multi, int convert, int DrawMode, int Color) /**************************************************************/ /* Routine de dessin d'un composant d'une librairie LibEntry = pointeur sur la description en librairie posX, posY = position du composant DrawMode = GrOR .. Color = 0 : dessin en vraies couleurs, sinon couleur = Color Une croix symbolise le point d'accrochage (ref position) du composant Le composant est toujours trace avec orientation 0 */ { int color; int TransMat[2][2]; wxString Prefix; LibDrawField * Field; wxPoint text_pos; /* Orientation normale */ TransMat[0][0] = 1; TransMat[1][1] = -1; TransMat[1][0] = TransMat[0][1] = 0; DrawLibPartAux(panel, DC, NULL, LibEntry, wxPoint(posX, posY), TransMat, Multi, convert, DrawMode, Color); /* Trace des 2 champs ref et value (Attention aux coord: la matrice de transformation change de signe les coord Y */ GRSetDrawMode(DC, DrawMode); if( LibEntry->m_Prefix.m_Attributs & TEXT_NO_VISIBLE ) { if( Color >= 0 ) color = Color; else color = UNVISIBLE_COLOR; } else { if( Color >= 0) color = Color; else color = ReturnLayerColor(LAYER_REFERENCEPART); } if (LibEntry->m_UnitCount > 1) Prefix.Printf( wxT("%s?%c"),LibEntry->m_Prefix.m_Text.GetData(),Multi + 'A' - 1); else Prefix = LibEntry->m_Prefix.m_Text + wxT("?"); text_pos.x = LibEntry->m_Prefix.m_Pos.x + posX; text_pos.y = posY - LibEntry->m_Prefix.m_Pos.y; DrawGraphicText(panel, DC, text_pos, color,LibEntry->m_Prefix.m_Text.GetData(), LibEntry->m_Prefix.m_Orient ? TEXT_ORIENT_VERT : TEXT_ORIENT_HORIZ, LibEntry->m_Prefix.m_Size, LibEntry->m_Prefix.m_HJustify, LibEntry->m_Prefix.m_VJustify); if( LibEntry->m_Name.m_Attributs & TEXT_NO_VISIBLE ) { if( Color >= 0) color = Color; else color = UNVISIBLE_COLOR; } else { if( Color >= 0 ) color = Color; else color = ReturnLayerColor(LAYER_VALUEPART); } text_pos.x = LibEntry->m_Name.m_Pos.x + posX; text_pos.y = posY - LibEntry->m_Name.m_Pos.y; DrawGraphicText(panel, DC, text_pos, color, LibEntry->m_Name.m_Text.GetData(), LibEntry->m_Name.m_Orient ? TEXT_ORIENT_VERT : TEXT_ORIENT_HORIZ, LibEntry->m_Name.m_Size, LibEntry->m_Name.m_HJustify, LibEntry->m_Name.m_VJustify); for( Field = LibEntry->Fields; Field != NULL; Field = (LibDrawField *)Field->Pnext ) { if( Field->m_Text.IsEmpty() ) return; if( Field->m_Attributs & TEXT_NO_VISIBLE ) { if( Color >= 0) color = Color; else color = UNVISIBLE_COLOR; } else { if( Color >= 0) color = Color; else color = ReturnLayerColor(LAYER_FIELDS); } text_pos.x = Field->m_Pos.x + posX; text_pos.y = posY - Field->m_Pos.y; DrawGraphicText(panel, DC, text_pos, color, Field->m_Text, Field->m_Orient ? TEXT_ORIENT_VERT : TEXT_ORIENT_HORIZ, Field->m_Size, Field->m_HJustify, Field->m_VJustify); } // Tracé de l'ancre int len = 3 * panel->GetZoom(); GRLine(&panel->m_ClipBox, DC, posX, posY - len, posX, posY + len, color); GRLine(&panel->m_ClipBox, DC, posX - len, posY, posX + len, posY, color); } /***************************************************************************** * Routine to draw the given part at given position, transformed/mirror as * * specified, and in the given drawing mode. Only this one is visible... * *****************************************************************************/ void EDA_SchComponentStruct::Draw(WinEDA_DrawPanel * panel,wxDC * DC, const wxPoint & offset, int DrawMode, int Color) { EDA_LibComponentStruct *Entry; int ii; bool dummy = FALSE; if( (Entry = FindLibPart(m_ChipName.GetData(),wxEmptyString,FIND_ROOT)) == NULL) { /* composant non trouvé, on affiche un composant "dummy" */ dummy = TRUE; if( DummyCmp == NULL ) CreateDummyCmp(); Entry = DummyCmp; } DrawLibPartAux(panel, DC, this, Entry, m_Pos + offset, m_Transform, dummy ? 0 : m_Multi, dummy ? 0 : m_Convert, DrawMode); /* Trace des champs, avec placement et orientation selon orient. du composant */ if( ((m_Field[REFERENCE].m_Attributs & TEXT_NO_VISIBLE) == 0) && ! (m_Field[REFERENCE].m_Flags & IS_MOVED) ) { if ( Entry->m_UnitCount > 1 ) DrawTextField(panel, DC, &m_Field[REFERENCE],1,DrawMode); else DrawTextField(panel, DC, &m_Field[REFERENCE],0,DrawMode); } for( ii = VALUE; ii < NUMBER_OF_FIELDS; ii++ ) { if (m_Field[ii].m_Flags & IS_MOVED) continue; DrawTextField(panel, DC, &m_Field[ii],0,DrawMode); } } /***********************************************************/ void DrawTextField(WinEDA_DrawPanel * panel,wxDC * DC, PartTextStruct * Field, int IsMulti, int DrawMode) /***********************************************************/ /* Routine de trace des textes type Field du composant. entree: IsMulti: flag Non Null si il y a plusieurs parts par boitier. n'est utile que pour le champ reference pour ajouter a celui ci l'identification de la part ( A, B ... ) DrawMode: mode de trace */ { int orient, color; wxPoint pos; /* Position des textes */ EDA_SchComponentStruct *DrawLibItem = (EDA_SchComponentStruct *) Field->m_Parent; int hjustify, vjustify; if( Field->m_Attributs & TEXT_NO_VISIBLE ) return; if( Field->IsVoid() ) return; GRSetDrawMode(DC, DrawMode); /* Calcul de la position des textes, selon orientation du composant */ orient = Field->m_Orient; hjustify = Field->m_HJustify; vjustify = Field->m_VJustify; pos.x = Field->m_Pos.x - DrawLibItem->m_Pos.x; pos.y = Field->m_Pos.y - DrawLibItem->m_Pos.y; pos = DrawLibItem->GetScreenCoord(pos); pos.x += DrawLibItem->m_Pos.x; pos.y += DrawLibItem->m_Pos.y; /* Y a t-il rotation (pour l'orientation, la justification)*/ if(DrawLibItem->m_Transform[0][1]) // Rotation du composant de 90deg { if ( orient == TEXT_ORIENT_HORIZ) orient = TEXT_ORIENT_VERT; else orient = TEXT_ORIENT_HORIZ; /* Y a t-il rotation, miroir (pour les justifications)*/ EXCHG(hjustify, vjustify); if (DrawLibItem->m_Transform[1][0] < 0 ) vjustify = - vjustify; if (DrawLibItem->m_Transform[1][0] > 0 ) hjustify = - hjustify; } else { /* Texte horizontal: Y a t-il miroir (pour les justifications)*/ if (DrawLibItem->m_Transform[0][0] < 0 ) hjustify = - hjustify; if (DrawLibItem->m_Transform[1][1] > 0 ) vjustify = - vjustify; } if( Field->m_FieldId == REFERENCE ) color = ReturnLayerColor(LAYER_REFERENCEPART); else if( Field->m_FieldId == VALUE ) color = ReturnLayerColor(LAYER_VALUEPART); else color = ReturnLayerColor(LAYER_FIELDS); if( !IsMulti || (Field->m_FieldId != REFERENCE) ) { DrawGraphicText(panel, DC, pos, color, Field->m_Text.GetData(), orient ? TEXT_ORIENT_VERT : TEXT_ORIENT_HORIZ, Field->m_Size, hjustify, vjustify); } else /* Le champ est la reference, et il y a plusieurs parts par boitier */ {/* On ajoute alors A ou B ... a la reference */ wxString fulltext = Field->m_Text; fulltext.Append('A' - 1 + DrawLibItem->m_Multi); DrawGraphicText(panel, DC, pos, color, fulltext.GetData(), orient ? TEXT_ORIENT_VERT : TEXT_ORIENT_HORIZ, Field->m_Size, hjustify, vjustify); } } /********************************************************************************/ EDA_LibComponentStruct *FindLibPart(const wxChar *Name, const wxString & LibName, int Alias) /********************************************************************************/ /* Routine to find a part in one of the libraries given its name. Name = Name of part. LibName = Name of Lib; if "": seach in all libs Alias = Flag: si flag != 0, retourne un pointeur sur une part ou un alias si flag = 0, retourne un pointeur sur une part meme si le nom correspond a un alias Alias = FIND_ROOT, ou Alias = FIND_ALIAS */ { EDA_LibComponentStruct *Entry; static EDA_LibComponentStruct DummyEntry(wxEmptyString); /* Used only to call PQFind. */ LibraryStruct *Lib = g_LibraryList; DummyEntry.m_Drawings = NULL; /* Used only to call PQFind. */ DummyEntry.m_Name.m_Text = Name; PQCompFunc((PQCompFuncType) LibraryEntryCompare); Entry = NULL; FindLibName.Empty(); while (Lib) { if( ! LibName.IsEmpty() ) { if( Lib->m_Name != LibName ) { Lib = Lib->m_Pnext; continue ; } } if( Lib == NULL ) break; Entry = (EDA_LibComponentStruct*)PQFind(Lib->m_Entries, &DummyEntry); if( Entry != NULL) { FindLibName = Lib->m_Name; break; } Lib = Lib->m_Pnext; } /* Si le nom est un alias, recherche du vrai composant */ if( Entry ) { if( (Entry->Type != ROOT ) && (Alias == FIND_ROOT) ) Entry = FindLibPart( ((EDA_LibCmpAliasStruct*)Entry)->m_RootName.GetData() , Lib->m_Name, FIND_ROOT); } return (Entry); } /***************************************************************************** * Routine to draw the given part at given position, transformed/mirror as * specified, and in the given drawing mode. * if Color < 0: Draw in normal color * else draw in color = Color *****************************************************************************/ /* DrawMode = GrXOR, GrOR ..*/ void DrawLibPartAux(WinEDA_DrawPanel * panel,wxDC * DC, EDA_SchComponentStruct *Component, EDA_LibComponentStruct *Entry, const wxPoint & Pos, int TransMat[2][2], int Multi, int convert, int DrawMode, int Color, bool DrawPinText) { int i, x1, y1, x2, y2, t1, t2, orient; LibEDA_BaseStruct *DEntry = NULL; int CharColor; int fill_option; int SetHightColor; //#define GETCOLOR(l) Color < 0 ? (ReturnLayerColor(l)| SetHightColor) : Color; #define GETCOLOR(l) Color < 0 ? SetHightColor ? s_ItemSelectColor : (ReturnLayerColor(l)| SetHightColor) : Color; if (Entry->m_Drawings == NULL) return; GRSetDrawMode(DC, DrawMode); for( DEntry = Entry->m_Drawings; DEntry != NULL;DEntry = DEntry->Next()) { /* Elimination des elements non relatifs a l'unite */ if( Multi && DEntry->m_Unit && (DEntry->m_Unit != Multi) ) continue; if( convert && DEntry->m_Convert && (DEntry->m_Convert != convert) ) continue; if ( DEntry->m_Flags & IS_MOVED ) continue; // Element en deplacement non trace SetHightColor = (DEntry->m_Selected & IS_SELECTED) ? HIGHT_LIGHT_FLAG : 0; switch (DEntry->m_StructType) { case COMPONENT_ARC_DRAW_TYPE: { int xc,yc, x2, y2; LibDrawArc * Arc = (LibDrawArc *) DEntry; CharColor = GETCOLOR(LAYER_DEVICE); xc = Pos.x + TransMat[0][0] * Arc->m_Pos.x + TransMat[0][1] * Arc->m_Pos.y; yc = Pos.y + TransMat[1][0] * Arc->m_Pos.x + TransMat[1][1] * Arc->m_Pos.y; x2 = Pos.x + TransMat[0][0] * Arc->m_Start.x + TransMat[0][1] * Arc->m_Start.y;; y2 = Pos.y + TransMat[1][0] * Arc->m_Start.x + TransMat[1][1] * Arc->m_Start.y; x1 = Pos.x + TransMat[0][0] * Arc->m_End.x + TransMat[0][1] * Arc->m_End.y;; y1 = Pos.y + TransMat[1][0] * Arc->m_End.x + TransMat[1][1] * Arc->m_End.y; t1 = Arc->t1; t2 = Arc->t2; bool swap = MapAngles(&t1, &t2, TransMat); if ( swap ) { EXCHG(x1,x2); EXCHG(y1, y2) } fill_option = Arc->m_Fill & (~g_PrintFillMask); if ( Color < 0 ) // Normal Color Layer { if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRFilledArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor, CharColor); #ifdef DRAW_ARC_WITH_ANGLE else GRArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor); #else else GRArc1(&panel->m_ClipBox, DC, x1, y1, x2, y2, xc, yc , CharColor); #endif } #ifdef DRAW_ARC_WITH_ANGLE else GRArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor); #else else GRArc1(&panel->m_ClipBox, DC, x1, y1, x2, y2, xc, yc, CharColor); #endif } break; case COMPONENT_CIRCLE_DRAW_TYPE: { LibDrawCircle * Circle = (LibDrawCircle *) DEntry; CharColor = GETCOLOR(LAYER_DEVICE); x1 = Pos.x + TransMat[0][0] * Circle->m_Pos.x + TransMat[0][1] * Circle->m_Pos.y; y1 = Pos.y + TransMat[1][0] * Circle->m_Pos.x + TransMat[1][1] * Circle->m_Pos.y; fill_option = Circle->m_Fill & (~g_PrintFillMask); if ( Color < 0 ) { if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRFilledCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor, CharColor); else GRCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor); } else GRCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor); } break; case COMPONENT_GRAPHIC_TEXT_DRAW_TYPE: { LibDrawText * Text = (LibDrawText *) DEntry; CharColor = GETCOLOR(LAYER_DEVICE); /* The text orientation may need to be flipped if the transformation matrix cuases xy axes to be flipped. */ t1 = (TransMat[0][0] != 0) ^ (Text->m_Horiz != 0); x1 = Pos.x + TransMat[0][0] * Text->m_Pos.x + TransMat[0][1] * Text->m_Pos.y; y1 = Pos.y + TransMat[1][0] * Text->m_Pos.x + TransMat[1][1] * Text->m_Pos.y; PutTextInfo(panel, DC, t1 ? TEXT_ORIENT_HORIZ : TEXT_ORIENT_VERT, wxPoint(x1, y1), Text->m_Size, Text->m_Text, DrawMode,CharColor); } break; case COMPONENT_RECT_DRAW_TYPE: { LibDrawSquare * Square = (LibDrawSquare *) DEntry; CharColor = GETCOLOR(LAYER_DEVICE); x1 = Pos.x + TransMat[0][0] * Square->m_Start.x + TransMat[0][1] * Square->m_Start.y; y1 = Pos.y + TransMat[1][0] * Square->m_Start.x + TransMat[1][1] * Square->m_Start.y; x2 = Pos.x + TransMat[0][0] * Square->m_End.x + TransMat[0][1] * Square->m_End.y; y2 = Pos.y + TransMat[1][0] * Square->m_End.x + TransMat[1][1] * Square->m_End.y; fill_option = Square->m_Fill & (~g_PrintFillMask); if ( Color < 0 ) { if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRFilledRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor, CharColor); else GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor); } else GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor); } break; case COMPONENT_PIN_DRAW_TYPE: /* Trace des Pins */ { LibDrawPin * Pin = (LibDrawPin *) DEntry; if(Pin->m_Attributs & PINNOTDRAW) { if( (ActiveScreen->m_Type == SCHEMATIC_FRAME) && !g_ShowAllPins ) break; } /* Calcul de l'orientation reelle de la Pin */ orient = Pin->ReturnPinDrawOrient(TransMat); /* Calcul de la position du point de reference */ x2 = Pos.x + (TransMat[0][0] * Pin->m_Pos.x) + (TransMat[0][1] * Pin->m_Pos.y); y2 = Pos.y + (TransMat[1][0] * Pin->m_Pos.x) + (TransMat[1][1] * Pin->m_Pos.y); /* Dessin de la pin et du symbole special associe */ CharColor = GETCOLOR(LAYER_PIN); DrawPinSymbol(panel, DC, x2, y2, Pin->m_PinLen, orient, Pin->m_PinShape, DrawMode, CharColor ); if ( DrawPinText ) { wxPoint pinpos(x2,y2); CharColor = SetHightColor ? s_ItemSelectColor : Color; Pin->DrawPinTexts(panel, DC, pinpos, orient, Entry->m_TextInside, Entry->m_DrawPinNum,Entry->m_DrawPinName, CharColor, DrawMode); } } break; case COMPONENT_POLYLINE_DRAW_TYPE: { LibDrawPolyline * polyline = (LibDrawPolyline *) DEntry; CharColor = GETCOLOR(LAYER_DEVICE); if ( Buf_Poly_Drawings == NULL ) { Buf_Poly_Size = polyline->n; Buf_Poly_Drawings = (int *) MyMalloc(sizeof(int) * 2 * Buf_Poly_Size); } else if ( Buf_Poly_Size < polyline->n ) { Buf_Poly_Size = polyline->n; Buf_Poly_Drawings = (int *) realloc(Buf_Poly_Drawings, sizeof(int) * 2 * Buf_Poly_Size); } for (i = 0; i < polyline->n; i++) { Buf_Poly_Drawings[i * 2] = Pos.x + TransMat[0][0] * polyline->PolyList[i * 2] + TransMat[0][1] * polyline->PolyList[i * 2 + 1]; Buf_Poly_Drawings[i * 2 + 1] = Pos.y + TransMat[1][0] * polyline->PolyList[i * 2] + TransMat[1][1] * polyline->PolyList[i * 2 + 1]; } fill_option = polyline->m_Fill & (~g_PrintFillMask); if ( Color < 0 ) { if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 1, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 1, CharColor, CharColor); else GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 0, CharColor, CharColor); } else GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 0, CharColor, CharColor); } break; default: wxBell(); break; } /* Fin Switch */ } /* Fin Boucle de dessin */ if ( g_DebugLevel > 4 ) /* Draw the component boundary box */ { EDA_Rect BoundaryBox; if ( Component ) BoundaryBox = Component->GetBoundaryBox(); else BoundaryBox = Entry->GetBoundaryBox(Multi, convert); x1 = BoundaryBox.GetX(); y1 = BoundaryBox.GetY(); x2 = BoundaryBox.GetRight(); y2 = BoundaryBox.GetBottom(); GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, BROWN); BoundaryBox = Component->m_Field[REFERENCE].GetBoundaryBox(); x1 = BoundaryBox.GetX(); y1 = BoundaryBox.GetY(); x2 = BoundaryBox.GetRight(); y2 = BoundaryBox.GetBottom(); GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, BROWN); BoundaryBox = Component->m_Field[VALUE].GetBoundaryBox(); x1 = BoundaryBox.GetX(); y1 = BoundaryBox.GetY(); x2 = BoundaryBox.GetRight(); y2 = BoundaryBox.GetBottom(); GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, BROWN); } } /*******************************************************/ void DrawPinSymbol(WinEDA_DrawPanel * panel, wxDC * DC, int posX, int posY, int len, int orient, int Shape, int DrawMode, int Color) /*******************************************************/ /* Dessine la pin du symbole en cours de trace si Color != 0 dessin en couleur Color, sinon en couleurs standard. */ { int MapX1, MapY1, x1, y1; int color; if( Color >= 0) color = Color; else color = ReturnLayerColor(LAYER_PIN); GRSetDrawMode(DC, DrawMode); MapX1 = MapY1 = 0; x1 = posX; y1 = posY; switch ( orient ) { case PIN_UP: y1 = posY - len; MapY1 = 1; break; case PIN_DOWN: y1 = posY + len; MapY1 = -1; break; case PIN_LEFT: x1 = posX - len, MapX1 = 1; break; case PIN_RIGHT: x1 = posX + len; MapX1 = -1; break; } if( Shape & INVERT) { GRCircle(&panel->m_ClipBox, DC, MapX1 * INVERT_PIN_RADIUS + x1, MapY1 * INVERT_PIN_RADIUS + y1, INVERT_PIN_RADIUS, color); GRMoveTo(MapX1 * INVERT_PIN_RADIUS * 2 + x1, MapY1 * INVERT_PIN_RADIUS * 2 + y1); GRLineTo(&panel->m_ClipBox, DC, posX, posY, color); } else { GRMoveTo(x1, y1); GRLineTo(&panel->m_ClipBox, DC, posX, posY, color); } if(Shape & CLOCK) { if(MapY1 == 0 ) /* MapX1 = +- 1 */ { GRMoveTo(x1, y1 + CLOCK_PIN_DIM); GRLineTo(&panel->m_ClipBox, DC, x1 - MapX1 * CLOCK_PIN_DIM, y1, color); GRLineTo(&panel->m_ClipBox, DC, x1, y1 - CLOCK_PIN_DIM, color); } else /* MapX1 = 0 */ { GRMoveTo(x1 + CLOCK_PIN_DIM, y1 ); GRLineTo(&panel->m_ClipBox, DC, x1, y1 - MapY1 * CLOCK_PIN_DIM, color); GRLineTo(&panel->m_ClipBox, DC, x1 - CLOCK_PIN_DIM, y1, color); } } if(Shape & LOWLEVEL_IN) /* IEEE symbol "Active Low Input" */ { if(MapY1 == 0 ) /* MapX1 = +- 1 */ { GRMoveTo(x1 + MapX1 * IEEE_SYMBOL_PIN_DIM*2, y1); GRLineTo(&panel->m_ClipBox, DC, x1 + MapX1 * IEEE_SYMBOL_PIN_DIM*2, y1 - IEEE_SYMBOL_PIN_DIM, color); GRLineTo(&panel->m_ClipBox, DC, x1, y1, color); } else /* MapX1 = 0 */ { GRMoveTo(x1, y1 + MapY1 * IEEE_SYMBOL_PIN_DIM*2); GRLineTo(&panel->m_ClipBox, DC, x1 - IEEE_SYMBOL_PIN_DIM, y1 + MapY1 * IEEE_SYMBOL_PIN_DIM*2, color); GRLineTo(&panel->m_ClipBox, DC, x1 , y1, color); } } if(Shape & LOWLEVEL_OUT) /* IEEE symbol "Active Low Output" */ { if(MapY1 == 0 ) /* MapX1 = +- 1 */ { GRMoveTo(x1, y1 - IEEE_SYMBOL_PIN_DIM); GRLineTo(&panel->m_ClipBox, DC, x1 + MapX1 * IEEE_SYMBOL_PIN_DIM*2, y1, color); } else /* MapX1 = 0 */ { GRMoveTo(x1 - IEEE_SYMBOL_PIN_DIM, y1); GRLineTo(&panel->m_ClipBox, DC, x1 , y1 + MapY1 * IEEE_SYMBOL_PIN_DIM*2, color); } } /* Draw the pin end target (active end of the pin) */ if ( ! g_IsPrinting ) // Draw but do not print the pin end target */ GRCircle(&panel->m_ClipBox, DC, posX,posY,TARGET_PIN_DIAM, color); } /***************************************************************************** * Routine to rotate the given angular direction by the given Transformation. * * Input (and output) angles must be as follows: * * Unit is 0.1 degre * * Angle1 in [0..3600], Angle2 > Angle1 in [0..7200]. Arc is assumed to be less * * than 180.0 degrees. * * Algorithm: * * Map the angles to a point on the unit circle which is mapped using the * * transform (only mirror and rotate so it remains on the unit circle) to * * a new point which is used to detect new angle. * *****************************************************************************/ bool MapAngles(int *Angle1, int *Angle2, int TransMat[2][2]) { int Angle, Delta; double x, y, t; bool swap = FALSE; Delta = *Angle2 - *Angle1; if ( Delta >= 1800 ) { *Angle1 -=1; *Angle2 +=1; } x = cos(*Angle1 * M_PI / 1800.0); y = sin(*Angle1 * M_PI / 1800.0); t = x * TransMat[0][0] + y * TransMat[0][1]; y = x * TransMat[1][0] + y * TransMat[1][1]; x = t; *Angle1 = (int) (atan2(y, x) * 1800.0 / M_PI + 0.5); x = cos(*Angle2 * M_PI / 1800.0); y = sin(*Angle2 * M_PI / 1800.0); t = x * TransMat[0][0] + y * TransMat[0][1]; y = x * TransMat[1][0] + y * TransMat[1][1]; x = t; *Angle2 = (int) (atan2(y, x) * 1800.0 / M_PI + 0.5); NORMALIZE_ANGLE(*Angle1); NORMALIZE_ANGLE(*Angle2); if (*Angle2 < *Angle1) *Angle2 += 3600; if (*Angle2 - *Angle1 > 1800) { /* Need to swap the two angles. */ Angle = (*Angle1); *Angle1 = (*Angle2); *Angle2 = Angle; NORMALIZE_ANGLE(*Angle1); NORMALIZE_ANGLE(*Angle2); if (*Angle2 < *Angle1) *Angle2 += 3600; swap = TRUE; } if ( Delta >= 1800 ) { *Angle1 +=1; *Angle2 -=1; } return swap; } /***************************************************************************** * Routine to display an outline version of given library entry. * * This routine is applied by the PlaceLibItem routine above. * *****************************************************************************/ void DrawingLibInGhost(WinEDA_DrawPanel * panel, wxDC * DC, EDA_LibComponentStruct *LibEntry, EDA_SchComponentStruct *DrawLibItem, int PartX, int PartY, int multi, int convert, int Color, bool DrawPinText) { int DrawMode = g_XorMode; DrawLibPartAux(panel, DC, DrawLibItem, LibEntry, wxPoint(PartX, PartY), DrawLibItem->m_Transform, multi, convert, DrawMode, Color, DrawPinText); } /************************************************************/ /* Routine to draw One LibraryDrawStruct at given position, */ /* matrice de transformation 1 0 0 -1 (normale) */ /* DrawMode = GrXOR, GrOR .. */ /************************************************************/ /* Utilise en LibEdit et Lib Browse */ void DrawLibraryDrawStruct(WinEDA_DrawPanel * panel, wxDC * DC, EDA_LibComponentStruct *LibEntry, int PartX, int PartY, LibEDA_BaseStruct *DrawItem, int Multi, int DrawMode, int Color) { int i, x1, y1, x2, y2, t1, t2, orient; int CharColor; int TransMat[2][2]; int fill_option; #undef GETCOLOR #define GETCOLOR(l) Color < 0 ? ReturnLayerColor(l) : Color; Multi = 0; /* unused */ /* Trace de la structure */ CharColor = GETCOLOR(LAYER_DEVICE); GRSetDrawMode(DC, DrawMode); TransMat[0][0] = 1; TransMat[0][1] = TransMat[1][0] = 0; TransMat[1][1] = -1; switch (DrawItem->m_StructType) { case COMPONENT_ARC_DRAW_TYPE: { int xc,yc, x2,y2; LibDrawArc * Arc = (LibDrawArc *) DrawItem; t1 = Arc->t1; t2 = Arc->t2; bool swap = MapAngles(&t1, &t2, TransMat); xc = PartX + Arc->m_Pos.x; yc = PartY - Arc->m_Pos.y; x2 = PartX + Arc->m_Start.x; y2 = PartY - Arc->m_Start.y; x1 = PartX + Arc->m_End.x; y1 = PartY - Arc->m_End.y; if ( swap ) { EXCHG(x1,x2); EXCHG(y1, y2)} fill_option = Arc->m_Fill & (~g_PrintFillMask); if ( (Arc->m_Fill == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( Arc->m_Fill == FILLED_SHAPE) GRFilledArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor, CharColor); #ifdef DRAW_ARC_WITH_ANGLE else GRArc(&panel->m_ClipBox, DC, xc, yc, t1, t2, Arc->m_Rayon, CharColor); #else else GRArc1(&panel->m_ClipBox, DC, x1, y1, x2, y2, xc, yc, CharColor); #endif } break; case COMPONENT_CIRCLE_DRAW_TYPE: { LibDrawCircle * Circle = (LibDrawCircle *) DrawItem; x1 = PartX + Circle->m_Pos.x; y1 = PartY - Circle->m_Pos.y; fill_option = Circle->m_Fill & (~g_PrintFillMask); if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRFilledCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor, CharColor); else GRCircle(&panel->m_ClipBox, DC, x1, y1, Circle->m_Rayon, CharColor); } break; case COMPONENT_GRAPHIC_TEXT_DRAW_TYPE: { LibDrawText * Text = (LibDrawText *) DrawItem; x1 = PartX + Text->m_Pos.x; y1 = PartY - Text->m_Pos.y; PutTextInfo(panel, DC, Text->m_Horiz, wxPoint(x1, y1), Text->m_Size, Text->m_Text, DrawMode,CharColor); } break; case COMPONENT_RECT_DRAW_TYPE: { LibDrawSquare * Square = (LibDrawSquare *) DrawItem; x1 = PartX + Square->m_Start.x; y1 = PartY - Square->m_Start.y; x2 = PartX + Square->m_End.x; y2 = PartY - Square->m_End.y; fill_option = Square->m_Fill & (~g_PrintFillMask); if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRFilledRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRFilledRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor, CharColor); else GRRect(&panel->m_ClipBox, DC, x1, y1, x2, y2, CharColor); } break; case COMPONENT_PIN_DRAW_TYPE: /* Trace des Pins */ { LibDrawPin * Pin = (LibDrawPin *) DrawItem; x2 = PartX + Pin->m_Pos.x; y2 = PartY - Pin->m_Pos.y; /* Compute the real pin orientation, i.e. pin orient + component orient */ orient = Pin->ReturnPinDrawOrient(TransMat); /* Dessin de la pin et du symbole special associe */ if( Pin->m_Attributs & PINNOTDRAW) CharColor = DARKGRAY; else CharColor = -1; DrawPinSymbol(panel, DC, x2, y2, Pin->m_PinLen, orient, Pin->m_PinShape, DrawMode); wxPoint pinpos(x2,y2); Pin->DrawPinTexts(panel, DC, pinpos, orient, LibEntry->m_TextInside, LibEntry->m_DrawPinNum,LibEntry->m_DrawPinName, CharColor, DrawMode); } break; case COMPONENT_POLYLINE_DRAW_TYPE: { LibDrawPolyline * polyline = (LibDrawPolyline *) DrawItem; if ( Buf_Poly_Drawings == NULL ) { Buf_Poly_Size = polyline->n; Buf_Poly_Drawings = (int *) MyMalloc(sizeof(int) * 2 * Buf_Poly_Size); } else if ( Buf_Poly_Size < polyline->n ) { Buf_Poly_Size = polyline->n; Buf_Poly_Drawings = (int *) realloc(Buf_Poly_Drawings, sizeof(int) * 2 * Buf_Poly_Size); } for (i = 0; i < polyline->n; i++) { Buf_Poly_Drawings[i * 2] = PartX + polyline->PolyList[i * 2]; Buf_Poly_Drawings[i * 2 + 1] = PartY - polyline->PolyList[i * 2 + 1]; } fill_option = polyline->m_Fill & (~g_PrintFillMask); if ( (fill_option == FILLED_WITH_BG_BODYCOLOR) && ! g_IsPrinting ) GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 1, CharColor, ReturnLayerColor(LAYER_DEVICE_BACKGROUND)); else if ( fill_option == FILLED_SHAPE) GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 1, CharColor, CharColor); else GRPoly(&panel->m_ClipBox, DC, polyline->n, Buf_Poly_Drawings, 0, CharColor, CharColor); break; } } }