//----------------------------------------------------------------------------- // Copyright 2007 Jonathan Westhues // // This file is part of LDmicro. // // LDmicro is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // LDmicro is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with LDmicro. If not, see . //------ // // The two `output devices' for the drawing code: either the export as text // stuff to write to a file, or all the routines concerned with drawing to // the screen. // Jonathan Westhues, Dec 2004 // // Ported to linus by: R Ramana, 2018 //----------------------------------------------------------------------------- #include "linuxUI.h" #include #include #include "ldmicro.h" void (*DrawChars)(int, int, char *); // After an undo all the memory addresses change but make an effort to put // the cursor roughly where it should be. int SelectedGxAfterNextPaint = -1; int SelectedGyAfterNextPaint = -1; // After pushing a rung up or down the position of that rung in the table // changes, but the cursor should stay where it was. BOOL ScrollSelectedIntoViewAfterNextPaint; // Buffer that we write to when exporting (drawing) diagram to a text file. // Dynamically allocated so that we're at least slightly efficient. static char **ExportBuffer; // The fonts that we will use to draw the ladder diagram: fixed width, one // normal-weight, one bold. HFONT FixedWidthFont; HFONT FixedWidthFontBold; // Different colour brushes for right and left buses in simulation, but same // colour for both in edit mode; also for the backgrounds in simulation and // edit modes. static HBRUSH BusRightBus; static HBRUSH BusLeftBrush; static HBRUSH BusBrush; static HBRUSH BgBrush; static HBRUSH SimBgBrush; // Parameters that determine our offset if we are scrolled int ScrollXOffset; int ScrollYOffset; int ScrollXOffsetMax; int ScrollYOffsetMax; // Is the cursor currently drawn? We XOR it so this gets toggled. static BOOL CursorDrawn; // Colours with which to do syntax highlighting, configurable SyntaxHighlightingColours HighlightColours; #define X_RIGHT_PADDING 30 //----------------------------------------------------------------------------- // Blink the cursor on the schematic; called by a Windows timer. We XOR // draw it so just draw the same rectangle every time to show/erase the // cursor. Cursor may be in one of four places in the selected leaf (top, // bottom, left, right) but we don't care; just go from the coordinates // computed when we drew the schematic in the paint procedure. //----------------------------------------------------------------------------- void CALLBACK BlinkCursor(HWND hwnd, UINT msg, UINT_PTR id, DWORD time) { // if(!isFocus(MainWindow) && !CursorDrawn) return; // if(Cursor.left == 0) return; // PlcCursor c; // memcpy(&c, &Cursor, sizeof(c)); // c.top -= ScrollYOffset*POS_HEIGHT*FONT_HEIGHT; // c.left -= ScrollXOffset; // if(c.top >= IoListTop) return; // if(c.top + c.height >= IoListTop) { // c.height = IoListTop - c.top - 3; // } // Hdc = GetDC(MainWindow); // SelectObject(Hdc, GetStockObject(WHITE_BRUSH)); // PatBlt(Hdc, c.left, c.top, c.width, c.height, PATINVERT); // CursorDrawn = !CursorDrawn; // ReleaseDC(MainWindow, Hdc); } //----------------------------------------------------------------------------- // Output a string to the screen at a particular location, in character- // sized units. //----------------------------------------------------------------------------- static void DrawCharsToScreen(int cx, int cy, char *str) { // cy -= ScrollYOffset*POS_HEIGHT; // if(cy < -2) return; // if(cy*FONT_HEIGHT + Y_PADDING > IoListTop) return; // COLORREF prev; // BOOL firstTime = TRUE; // BOOL inNumber = FALSE; // BOOL inComment = FALSE; // int inBrace = 0; // for(; *str; str++, cx++) { // int x = cx*FONT_WIDTH + X_PADDING; // int y = cy*FONT_HEIGHT + Y_PADDING; // BOOL hiOk = !(InSimulationMode || ThisHighlighted); // if(strchr("{}[]", *str) && hiOk && !inComment) { // if(*str == '{' || *str == '[') inBrace++; // if(inBrace == 1) { // prev = GetTextColor(Hdc); // SetTextColor(Hdc, HighlightColours.punct); // TextOut(Hdc, x, y, str, 1); // SetTextColor(Hdc, prev); // } else { // TextOut(Hdc, x, y, str, 1); // } // if(*str == ']' || *str == '}') inBrace--; // } else if(( // (isdigit(*str) && (firstTime || isspace(str[-1]) // || str[-1] == ':' || str[-1] == '[')) || // (*str == '-' && isdigit(str[1]))) && hiOk && !inComment) // { // prev = GetTextColor(Hdc); // SetTextColor(Hdc, HighlightColours.lit); // TextOut(Hdc, x, y, str, 1); // SetTextColor(Hdc, prev); // inNumber = TRUE; // } else if(*str == '\x01') { // cx--; // if(hiOk) { // prev = GetTextColor(Hdc); // SetTextColor(Hdc, HighlightColours.op); // } // } else if(*str == '\x02') { // cx--; // if(hiOk) { // SetTextColor(Hdc, prev); // inComment = FALSE; // } // } else if(*str == '\x03') { // cx--; // if(hiOk) { // prev = GetTextColor(Hdc); // SetTextColor(Hdc, HighlightColours.comment); // inComment = TRUE; // } // } else if(inNumber) { // if(isdigit(*str) || *str == '.') { // prev = GetTextColor(Hdc); // SetTextColor(Hdc, HighlightColours.lit); // TextOut(Hdc, x, y, str, 1); // SetTextColor(Hdc, prev); // } else { // TextOut(Hdc, x, y, str, 1); // inNumber = FALSE; // } // } else { // TextOut(Hdc, x, y, str, 1); // } // firstTime = FALSE; // } } //----------------------------------------------------------------------------- // Total number of columns that we can display in the given amount of // window area. Need to leave some slop on the right for the scrollbar, of // course. //----------------------------------------------------------------------------- int ScreenColsAvailable(void) { // RECT r; // GetClientRect(MainWindow, &r); // return (r.right - (X_PADDING + X_RIGHT_PADDING)) / (POS_WIDTH*FONT_WIDTH); } //----------------------------------------------------------------------------- // Total number of columns that we can display in the given amount of // window area. Need to leave some slop on the right for the scrollbar, of // course, and extra slop at the bottom for the horiz scrollbar if it is // shown. //----------------------------------------------------------------------------- int ScreenRowsAvailable(void) { // int adj; // if(ScrollXOffsetMax == 0) { // adj = 0; // } else { // adj = 18; // } // return (IoListTop - Y_PADDING - adj) / (POS_HEIGHT*FONT_HEIGHT); } //----------------------------------------------------------------------------- // Paint the ladder logic program to the screen. Also figure out where the // cursor should go and fill in coordinates for BlinkCursor. Not allowed to // draw deeper than IoListTop, as we would run in to the I/O listbox. //----------------------------------------------------------------------------- void PaintWindow(void) { // static HBITMAP BackBitmap; // static HDC BackDc; // static int BitmapWidth; // ok(); // RECT r; // GetClientRect(MainWindow, &r); // int bw = r.right; // int bh = IoListTop; // HDC paintDc; // if(!BackDc) { // HWND desktop = GetDesktopWindow(); // RECT dk; // GetClientRect(desktop, &dk); // BitmapWidth = max(2000, dk.right + 300); // BackBitmap = CreateCompatibleBitmap(Hdc, BitmapWidth, dk.bottom + 300); // BackDc = CreateCompatibleDC(Hdc); // SelectObject(BackDc, BackBitmap); // } // paintDc = Hdc; // Hdc = BackDc; // RECT fi; // fi.left = 0; fi.top = 0; // fi.right = BitmapWidth; fi.bottom = bh; // FillRect(Hdc, &fi, InSimulationMode ? SimBgBrush : BgBrush); // // now figure out how we should draw the ladder logic // ColsAvailable = ProgCountWidestRow(); // if(ColsAvailable < ScreenColsAvailable()) { // ColsAvailable = ScreenColsAvailable(); // } // memset(DisplayMatrix, 0, sizeof(DisplayMatrix)); // SelectionActive = FALSE; // memset(&Cursor, 0, sizeof(Cursor)); // DrawChars = DrawCharsToScreen; // int i; // int cy = 0; // int rowsAvailable = ScreenRowsAvailable(); // for(i = 0; i < Prog.numRungs; i++) { // int thisHeight = POS_HEIGHT*CountHeightOfElement(ELEM_SERIES_SUBCKT, // Prog.rungs[i]); // // For speed, there is no need to draw everything all the time, but // // we still must draw a bit above and below so that the DisplayMatrix // // is filled in enough to make it possible to reselect using the // // cursor keys. // if(((cy + thisHeight) >= (ScrollYOffset - 8)*POS_HEIGHT) && // (cy < (ScrollYOffset + rowsAvailable + 8)*POS_HEIGHT)) // { // SetBkColor(Hdc, InSimulationMode ? HighlightColours.simBg : // HighlightColours.bg); // SetTextColor(Hdc, InSimulationMode ? HighlightColours.simRungNum : // HighlightColours.rungNum); // SelectObject(Hdc, FixedWidthFont); // int rung = i + 1; // int y = Y_PADDING + FONT_HEIGHT*cy; // int yp = y + FONT_HEIGHT*(POS_HEIGHT/2) - // POS_HEIGHT*FONT_HEIGHT*ScrollYOffset; // if(rung < 10) { // char r[1] = { rung + '0' }; // TextOut(Hdc, 8 + FONT_WIDTH, yp, r, 1); // } else { // char r[2] = { (rung / 10) + '0', (rung % 10) + '0' }; // TextOut(Hdc, 8, yp, r, 2); // } // int cx = 0; // DrawElement(ELEM_SERIES_SUBCKT, Prog.rungs[i], &cx, &cy, // Prog.rungPowered[i]); // } // cy += thisHeight; // cy += POS_HEIGHT; // } // cy -= 2; // DrawEndRung(0, cy); // if(SelectedGxAfterNextPaint >= 0) { // MoveCursorNear(SelectedGxAfterNextPaint, SelectedGyAfterNextPaint); // InvalidateRect(MainWindow, NULL, FALSE); // SelectedGxAfterNextPaint = -1; // SelectedGyAfterNextPaint = -1; // } else if(ScrollSelectedIntoViewAfterNextPaint && Selected) { // SelectElement(-1, -1, Selected->selectedState); // ScrollSelectedIntoViewAfterNextPaint = FALSE; // InvalidateRect(MainWindow, NULL, FALSE); // } else { // if(!SelectionActive) { // if(Prog.numRungs > 0) { // if(MoveCursorTopLeft()) { // InvalidateRect(MainWindow, NULL, FALSE); // } // } // } // } // // draw the `buses' at either side of the screen // r.left = X_PADDING - FONT_WIDTH; // r.top = 0; // r.right = r.left + 4; // r.bottom = IoListTop; // FillRect(Hdc, &r, InSimulationMode ? BusLeftBrush : BusBrush); // r.left += POS_WIDTH*FONT_WIDTH*ColsAvailable + 2; // r.right += POS_WIDTH*FONT_WIDTH*ColsAvailable + 2; // FillRect(Hdc, &r, InSimulationMode ? BusRightBus : BusBrush); // CursorDrawn = FALSE; // BitBlt(paintDc, 0, 0, bw, bh, BackDc, ScrollXOffset, 0, SRCCOPY); // if(InSimulationMode) { // KillTimer(MainWindow, TIMER_BLINK_CURSOR); // } else { // KillTimer(MainWindow, TIMER_BLINK_CURSOR); // BlinkCursor(NULL, 0, NULL, 0); // SetTimer(MainWindow, TIMER_BLINK_CURSOR, 800, BlinkCursor); // } // Hdc = paintDc; // ok(); } //----------------------------------------------------------------------------- // Set up the syntax highlighting colours, according to the currently selected // scheme. //----------------------------------------------------------------------------- static void SetSyntaxHighlightingColours(void) { static const SyntaxHighlightingColours Schemes[] = { { RGB(0, 0, 0), // bg RGB(255, 255, 225), // def RGB(255, 110, 90), // selected RGB(255, 150, 90), // op RGB(255, 255, 100), // punct RGB(255, 160, 160), // lit RGB(120, 255, 130), // name RGB(130, 130, 130), // rungNum RGB(130, 130, 245), // comment RGB(255, 255, 255), // bus RGB(0, 0, 0), // simBg RGB(130, 130, 130), // simRungNum RGB(100, 130, 130), // simOff RGB(255, 150, 150), // simOn RGB(255, 150, 150), // simBusLeft RGB(150, 150, 255), // simBusRight }, }; memcpy(&HighlightColours, &Schemes[0], sizeof(Schemes[0])); } //----------------------------------------------------------------------------- // Set up the stuff we'll need to draw our schematic diagram. Fonts, brushes, // pens, etc. //----------------------------------------------------------------------------- void InitForDrawing(void) { // SetSyntaxHighlightingColours(); // FixedWidthFont = CreateFont( // FONT_HEIGHT, FONT_WIDTH, // 0, 0, // FW_REGULAR, // FALSE, // FALSE, // FALSE, // ANSI_CHARSET, // OUT_DEFAULT_PRECIS, // CLIP_DEFAULT_PRECIS, // DEFAULT_QUALITY, // FF_DONTCARE, // "Lucida Console"); // FixedWidthFontBold = CreateFont( // FONT_HEIGHT, FONT_WIDTH, // 0, 0, // FW_REGULAR, // the bold text renders funny under Vista // FALSE, // FALSE, // FALSE, // ANSI_CHARSET, // OUT_DEFAULT_PRECIS, // CLIP_DEFAULT_PRECIS, // DEFAULT_QUALITY, // FF_DONTCARE, // "Lucida Console"); // LOGBRUSH lb; // lb.lbStyle = BS_SOLID; // lb.lbColor = HighlightColours.simBusRight; // BusRightBus = CreateBrushIndirect(&lb); // lb.lbColor = HighlightColours.simBusLeft; // BusLeftBrush = CreateBrushIndirect(&lb); // lb.lbColor = HighlightColours.bus; // BusBrush = CreateBrushIndirect(&lb); // lb.lbColor = HighlightColours.bg; // BgBrush = CreateBrushIndirect(&lb); // lb.lbColor = HighlightColours.simBg; // SimBgBrush = CreateBrushIndirect(&lb); } //----------------------------------------------------------------------------- // DrawChars function, for drawing to the export buffer instead of to the // screen. //----------------------------------------------------------------------------- static void DrawCharsToExportBuffer(int cx, int cy, char *str) { while(*str) { if(*str >= 10) { ExportBuffer[cy][cx] = *str; cx++; } str++; } } //----------------------------------------------------------------------------- // Export a text drawing of the ladder logic program to a file. //----------------------------------------------------------------------------- void ExportDrawingAsText(char *file) { // int maxWidth = ProgCountWidestRow(); // ColsAvailable = maxWidth; // int totalHeight = 0; // int i; // for(i = 0; i < Prog.numRungs; i++) { // totalHeight += CountHeightOfElement(ELEM_SERIES_SUBCKT, Prog.rungs[i]); // totalHeight += 1; // } // totalHeight *= POS_HEIGHT; // totalHeight += 3; // ExportBuffer = (char **)CheckMalloc(totalHeight * sizeof(char *)); // int l = maxWidth*POS_WIDTH + 8; // for(i = 0; i < totalHeight; i++) { // ExportBuffer[i] = (char *)CheckMalloc(l); // memset(ExportBuffer[i], ' ', l-1); // ExportBuffer[i][4] = '|'; // ExportBuffer[i][3] = '|'; // ExportBuffer[i][l-3] = '|'; // ExportBuffer[i][l-2] = '|'; // ExportBuffer[i][l-1] = '\0'; // } // DrawChars = DrawCharsToExportBuffer; // int cy = 1; // for(i = 0; i < Prog.numRungs; i++) { // int cx = 5; // DrawElement(ELEM_SERIES_SUBCKT, Prog.rungs[i], &cx, &cy, // Prog.rungPowered[i]); // if((i + 1) < 10) { // ExportBuffer[cy+1][1] = '0' + (i + 1); // } else { // ExportBuffer[cy+1][1] = '0' + ((i + 1) % 10); // ExportBuffer[cy+1][0] = '0' + ((i + 1) / 10); // } // cy += POS_HEIGHT*CountHeightOfElement(ELEM_SERIES_SUBCKT, // Prog.rungs[i]); // cy += POS_HEIGHT; // } // cy -= 2; // DrawEndRung(5, cy); // FILE *f = fopen(file, "w"); // if(!f) { // Error(_("Couldn't open '%s'\n"), f); // return; // } // fprintf(f, "LDmicro export text\n"); // if(Prog.mcu) { // fprintf(f, "for '%s', %.6f MHz crystal, %.1f ms cycle time\n\n", // Prog.mcu->mcuName, Prog.mcuClock/1e6, Prog.cycleTime/1e3); // } else { // fprintf(f, "no MCU assigned, %.6f MHz crystal, %.1f ms cycle time\n\n", // Prog.mcuClock/1e6, Prog.cycleTime/1e3); // } // fprintf(f, "\nLADDER DIAGRAM:\n\n"); // for(i = 0; i < totalHeight; i++) { // ExportBuffer[i][4] = '|'; // fprintf(f, "%s\n", ExportBuffer[i]); // CheckFree(ExportBuffer[i]); // } // CheckFree(ExportBuffer); // ExportBuffer = NULL; // fprintf(f, _("\n\nI/O ASSIGNMENT:\n\n")); // fprintf(f, _(" Name | Type | Pin\n")); // fprintf(f, " ----------------------------+--------------------+------\n"); // for(i = 0; i < Prog.io.count; i++) { // char b[1024]; // memset(b, '\0', sizeof(b)); // PlcProgramSingleIo *io = &Prog.io.assignment[i]; // char *type = IoTypeToString(io->type); // char pin[MAX_NAME_LEN]; // PinNumberForIo(pin, io); // sprintf(b, " | | %s\n", // pin); // memcpy(b+2, io->name, strlen(io->name)); // memcpy(b+31, type, strlen(type)); // fprintf(f, "%s", b); // } // fclose(f); // // we may have trashed the grid tables a bit; a repaint will fix that // InvalidateRect(MainWindow, NULL, FALSE); } //----------------------------------------------------------------------------- // Determine the settings of the vertical and (if needed) horizontal // scrollbars used to scroll our view of the program. //----------------------------------------------------------------------------- void SetUpScrollbars(BOOL *horizShown, SCROLLINFO *horiz, SCROLLINFO *vert) { // int totalHeight = 0; // int i; // for(i = 0; i < Prog.numRungs; i++) { // totalHeight += CountHeightOfElement(ELEM_SERIES_SUBCKT, Prog.rungs[i]); // totalHeight++; // } // totalHeight += 1; // for the end rung // int totalWidth = ProgCountWidestRow(); // if(totalWidth <= ScreenColsAvailable()) { // *horizShown = FALSE; // ScrollXOffset = 0; // ScrollXOffsetMax = 0; // } else { // *horizShown = TRUE; // memset(horiz, 0, sizeof(*horiz)); // horiz->cbSize = sizeof(*horiz); // horiz->fMask = SIF_DISABLENOSCROLL | SIF_ALL; // horiz->nMin = 0; // horiz->nMax = X_PADDING + totalWidth*POS_WIDTH*FONT_WIDTH; // RECT r; // GetClientRect(MainWindow, &r); // horiz->nPage = r.right - X_PADDING; // horiz->nPos = ScrollXOffset; // ScrollXOffsetMax = horiz->nMax - horiz->nPage + 1; // if(ScrollXOffset > ScrollXOffsetMax) ScrollXOffset = ScrollXOffsetMax; // if(ScrollXOffset < 0) ScrollXOffset = 0; // } // vert->cbSize = sizeof(*vert); // vert->fMask = SIF_DISABLENOSCROLL | SIF_ALL; // vert->nMin = 0; // vert->nMax = totalHeight - 1; // vert->nPos = ScrollYOffset; // vert->nPage = ScreenRowsAvailable(); // ScrollYOffsetMax = vert->nMax - vert->nPage + 1; // if(ScrollYOffset > ScrollYOffsetMax) ScrollYOffset = ScrollYOffsetMax; // if(ScrollYOffset < 0) ScrollYOffset = 0; }