//----------------------------------------------------------------------------- // Routines for the user interface when doing interactive things with prox // cards; this is basically a command line thing, in one window, and then // another window to do the graphs. // Jonathan Westhues, Sept 2005 //----------------------------------------------------------------------------- #include #include #include #include #include #include #include "prox.h" #define oops() do { \ char line[100]; \ sprintf(line, "Internal error at line %d file '%s'", __LINE__, \ __FILE__); \ MessageBox(NULL, line, "Error", MB_ICONERROR); \ exit(-1); \ } while(0) void dbp(char *str, ...) { va_list f; char buf[1024]; va_start(f, str); vsprintf(buf, str, f); OutputDebugString(buf); OutputDebugString("\n"); } int GraphBuffer[MAX_GRAPH_TRACE_LEN]; int GraphTraceLen; HPEN GreyPen, GreenPen, WhitePen, YellowPen; HBRUSH GreenBrush, YellowBrush; static int GraphStart = 0; static double GraphPixelsPerPoint = 1; static int CursorAPos; static int CursorBPos; double CursorScaleFactor = 1.0; static HPEN CursorAPen; static HPEN CursorBPen; static HWND CommandWindow; static HWND GraphWindow; static HWND ScrollbackEdit; static HWND CommandEdit; #define COMMAND_HISTORY_MAX 16 static char CommandHistory[COMMAND_HISTORY_MAX][256]; static int CommandHistoryPos = -1; static int CommandHistoryNext; static HFONT MyFixedFont; #define FixedFont(x) SendMessage((x), WM_SETFONT, (WPARAM)MyFixedFont, TRUE) void ExecCmd(char *cmd) { } int CommandFinished; int offset = 64; static void ResizeCommandWindow(void) { int w, h; RECT r; GetClientRect(CommandWindow, &r); w = r.right - r.left; h = r.bottom - r.top; MoveWindow(ScrollbackEdit, 10, 10, w - 20, h - 50, TRUE); MoveWindow(CommandEdit, 10, h - 29, w - 20, 22, TRUE); } void RepaintGraphWindow(void) { InvalidateRect(GraphWindow, NULL, TRUE); } static LRESULT CALLBACK CommandWindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch (msg) { case WM_DESTROY: case WM_QUIT: exit(0); return 0; case WM_SIZE: ResizeCommandWindow(); return 0; case WM_SETFOCUS: SetFocus(CommandEdit); break; default: return DefWindowProc(hwnd, msg, wParam, lParam); } return 1; } static void PaintGraph(HDC hdc) { HBRUSH brush; HPEN pen; brush = GreenBrush; pen = GreenPen; if(GraphStart < 0) { GraphStart = 0; } RECT r; GetClientRect(GraphWindow, &r); SelectObject(hdc, WhitePen); MoveToEx(hdc, r.left + offset, r.top, NULL); LineTo(hdc, r.left + offset, r.bottom); int zeroHeight = r.top + (r.bottom - r.top) / 2; SelectObject(hdc, GreyPen); MoveToEx(hdc, r.left, zeroHeight, NULL); LineTo(hdc, r.right, zeroHeight); int startMax = (GraphTraceLen - (int)((r.right - r.left - offset) / GraphPixelsPerPoint)); if(startMax < 0) { startMax = 0; } if(GraphStart > startMax) { GraphStart = startMax; } int absYMax = 1; SelectObject(hdc, pen); int i; for(i = GraphStart; ; i++) { if(i >= GraphTraceLen) { break; } if(fabs((double)GraphBuffer[i]) > absYMax) { absYMax = (int)fabs((double)GraphBuffer[i]); } int x = offset + (int)((i - GraphStart)*GraphPixelsPerPoint); if(x > r.right) { break; } } absYMax = (int)(absYMax*1.2 + 1); SelectObject(hdc, MyFixedFont); SetTextColor(hdc, RGB(255, 255, 255)); SetBkColor(hdc, RGB(0, 0, 0)); // number of points that will be plotted double span = (int)((r.right - r.left) / GraphPixelsPerPoint); // one label every offset pixels, let us say int labels = (r.right - r.left - offset) / offset; if(labels <= 0) labels = 1; // round to nearest power of 2 int pointsPerLabel = (int)(log(span / labels)/log(2.0)); if(pointsPerLabel <= 0) pointsPerLabel = 1; pointsPerLabel = (int)pow(2.0,pointsPerLabel); int yMin = INT_MAX; int yMax = INT_MIN; int yMean = 0; int n = 0; for(i = GraphStart; ; i++) { if(i >= GraphTraceLen) { break; } int x = offset + (int)((i - GraphStart)*GraphPixelsPerPoint); if(x > r.right + GraphPixelsPerPoint) { break; } int y = GraphBuffer[i]; if(y < yMin) { yMin = y; } if(y > yMax) { yMax = y; } yMean += y; n++; y = (y * (r.top - r.bottom) / (2*absYMax)) + zeroHeight; if(i == GraphStart) { MoveToEx(hdc, x, y, NULL); } else { LineTo(hdc, x, y); } if(GraphPixelsPerPoint > 10) { RECT f; f.left = x - 3; f.top = y - 3; f.right = x + 3; f.bottom = y + 3; FillRect(hdc, &f, brush); } if(((i - GraphStart) % pointsPerLabel == 0) && i != GraphStart) { SelectObject(hdc, WhitePen); MoveToEx(hdc, x, zeroHeight - 8, NULL); LineTo(hdc, x, zeroHeight + 8); char str[100]; sprintf(str, "+%d", (i - GraphStart)); SIZE size; GetTextExtentPoint32(hdc, str, strlen(str), &size); TextOut(hdc, x - size.cx, zeroHeight + 8, str, strlen(str)); SelectObject(hdc, pen); MoveToEx(hdc, x, y, NULL); } if(i == CursorAPos || i == CursorBPos) { if(i == CursorAPos) { SelectObject(hdc, CursorAPen); } else { SelectObject(hdc, CursorBPen); } MoveToEx(hdc, x, r.top, NULL); LineTo(hdc, x, r.bottom); SelectObject(hdc, pen); MoveToEx(hdc, x, y, NULL); } } if(n != 0) { yMean /= n; } char str[200]; sprintf(str, "@%d max=%d min=%d mean=%d n=%d/%d dt=%d [%.3f] zoom=%.3f CursorA=%d [%d] CursorB=%d [%d]", GraphStart, yMax, yMin, yMean, n, GraphTraceLen, CursorBPos - CursorAPos, (CursorBPos - CursorAPos)/CursorScaleFactor, GraphPixelsPerPoint, CursorAPos, GraphBuffer[CursorAPos], CursorBPos, GraphBuffer[CursorBPos]); TextOut(hdc, 50, r.bottom - 20, str, strlen(str)); } static LRESULT CALLBACK GraphWindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch (msg) { case WM_DESTROY: case WM_QUIT: GraphWindow = NULL; return DefWindowProc(hwnd, msg, wParam, lParam); case WM_SIZE: RepaintGraphWindow(); return 0; case WM_PAINT: { PAINTSTRUCT ps; HDC hdc = BeginPaint(hwnd, &ps); if(GraphStart < 0) { GraphStart = 0; } // This draws the trace. PaintGraph(hdc); EndPaint(hwnd, &ps); break; } case WM_KEYDOWN: switch(wParam) { case VK_DOWN: if(GraphPixelsPerPoint <= 8) { GraphPixelsPerPoint *= 2; } break; case VK_UP: if(GraphPixelsPerPoint >= 0.01) { GraphPixelsPerPoint /= 2; } break; case VK_RIGHT: if(GraphPixelsPerPoint < 16) { GraphStart += (int)(16 / GraphPixelsPerPoint); } else { GraphStart++; } break; case VK_LEFT: if(GraphPixelsPerPoint < 16) { GraphStart -= (int)(16 / GraphPixelsPerPoint); } else { GraphStart--; } break; default: goto nopaint; } RepaintGraphWindow(); nopaint: break; case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: { int x = LOWORD(lParam); x -= offset; x = (int)(x / GraphPixelsPerPoint); x += GraphStart; if(msg == WM_LBUTTONDOWN) { CursorAPos = x; } else { CursorBPos = x; } RepaintGraphWindow(); break; } default: return DefWindowProc(hwnd, msg, wParam, lParam); } return 1; } void PrintToScrollback(char *fmt, ...) { va_list f; char str[1024]; strcpy(str, "\r\n"); va_start(f, fmt); vsprintf(str+2, fmt, f); static char TextBuf[1024*32]; SendMessage(ScrollbackEdit, WM_GETTEXT, (WPARAM)sizeof(TextBuf), (LPARAM)TextBuf); if(strlen(TextBuf) + strlen(str) + 1 <= sizeof(TextBuf)) { strcat(TextBuf, str); } else { lstrcpyn(TextBuf, str, sizeof(TextBuf)); } SendMessage(ScrollbackEdit, WM_SETTEXT, 0, (LPARAM)TextBuf); SendMessage(ScrollbackEdit, EM_LINESCROLL, 0, (LPARAM)INT_MAX); } void ShowGraphWindow(void) { if(GraphWindow) return; GraphWindow = CreateWindowEx(0, "Graph", "graphed", WS_OVERLAPPED | WS_BORDER | WS_MINIMIZEBOX | WS_SYSMENU | WS_SIZEBOX | WS_VISIBLE, 200, 150, 600, 500, NULL, NULL, NULL, NULL); if(!GraphWindow) oops(); } void HideGraphWindow(void) { if(GraphWindow) { DestroyWindow(GraphWindow); GraphWindow = NULL; } } static void SetCommandEditTo(char *str) { SendMessage(CommandEdit, WM_SETTEXT, 0, (LPARAM)str); SendMessage(CommandEdit, EM_SETSEL, strlen(str), strlen(str)); } void ShowGui() { WNDCLASSEX wc; memset(&wc, 0, sizeof(wc)); wc.cbSize = sizeof(wc); wc.style = CS_BYTEALIGNCLIENT | CS_BYTEALIGNWINDOW | CS_OWNDC; wc.lpfnWndProc = (WNDPROC)CommandWindowProc; wc.hInstance = NULL; wc.hbrBackground = (HBRUSH)(COLOR_BTNSHADOW); wc.lpszClassName = "Command"; wc.lpszMenuName = NULL; wc.hCursor = LoadCursor(NULL, IDC_ARROW); if(!RegisterClassEx(&wc)) oops(); wc.lpszClassName = "Graph"; wc.lpfnWndProc = (WNDPROC)GraphWindowProc; wc.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); if(!RegisterClassEx(&wc)) oops(); CommandWindow = CreateWindowEx(0, "Command", "prox", WS_OVERLAPPED | WS_BORDER | WS_MINIMIZEBOX | WS_SYSMENU | WS_SIZEBOX | WS_VISIBLE, 20, 20, 500, 400, NULL, NULL, NULL, NULL); if(!CommandWindow) oops(); ScrollbackEdit = CreateWindowEx(WS_EX_CLIENTEDGE, "edit", "", WS_CHILD | WS_CLIPSIBLINGS | WS_VISIBLE | ES_MULTILINE | ES_AUTOVSCROLL | WS_VSCROLL, 0, 0, 0, 0, CommandWindow, NULL, NULL, NULL); CommandEdit = CreateWindowEx(WS_EX_CLIENTEDGE, "edit", "", WS_CHILD | WS_CLIPSIBLINGS | WS_TABSTOP | WS_VISIBLE | ES_AUTOHSCROLL, 0, 0, 0, 0, CommandWindow, NULL, NULL, NULL); MyFixedFont = CreateFont(14, 0, 0, 0, FW_REGULAR, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, FF_DONTCARE, "Lucida Console"); if(!MyFixedFont) MyFixedFont = (HFONT)GetStockObject(SYSTEM_FONT); FixedFont(ScrollbackEdit); FixedFont(CommandEdit); ResizeCommandWindow(); SetFocus(CommandEdit); PrintToScrollback(">> Started prox, built " __DATE__ " " __TIME__); PrintToScrollback(">> Connected to device"); GreyPen = CreatePen(PS_SOLID, 1, RGB(100, 100, 100)); GreenPen = CreatePen(PS_SOLID, 1, RGB(100, 255, 100)); YellowPen = CreatePen(PS_SOLID, 1, RGB(255, 255, 0)); GreenBrush = CreateSolidBrush(RGB(100, 255, 100)); YellowBrush = CreateSolidBrush(RGB(255, 255, 0)); WhitePen = CreatePen(PS_SOLID, 1, RGB(255, 255, 255)); CursorAPen = CreatePen(PS_DASH, 1, RGB(255, 255, 0)); CursorBPen = CreatePen(PS_DASH, 1, RGB(255, 0, 255)); MSG msg; for(;;) { if(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { if(msg.message == WM_KEYDOWN && msg.wParam == VK_RETURN) { char got[1024]; SendMessage(CommandEdit, WM_GETTEXT, (WPARAM)sizeof(got), (LPARAM)got); if(strcmp(got, "cls")==0) { SendMessage(ScrollbackEdit, WM_SETTEXT, 0, (LPARAM)""); } else { CommandReceived(got); } SendMessage(CommandEdit, WM_SETTEXT, 0, (LPARAM)""); // Insert it into the command history, unless it is // identical to the previous command in the history. int prev = CommandHistoryNext - 1; if(prev < 0) prev += COMMAND_HISTORY_MAX; if(strcmp(CommandHistory[prev], got) != 0) { strcpy(CommandHistory[CommandHistoryNext], got); CommandHistoryNext++; if(CommandHistoryNext == COMMAND_HISTORY_MAX) { CommandHistoryNext = 0; } } CommandHistoryPos = -1; } else if(msg.message == WM_KEYDOWN && msg.wParam == VK_UP && msg.hwnd == CommandEdit) { if(CommandHistoryPos == -1) { CommandHistoryPos = CommandHistoryNext; } CommandHistoryPos--; if(CommandHistoryPos < 0) { CommandHistoryPos = COMMAND_HISTORY_MAX-1; } SetCommandEditTo(CommandHistory[CommandHistoryPos]); } else if(msg.message == WM_KEYDOWN && msg.wParam == VK_DOWN && msg.hwnd == CommandEdit) { CommandHistoryPos++; if(CommandHistoryPos >= COMMAND_HISTORY_MAX) { CommandHistoryPos = 0; } SetCommandEditTo(CommandHistory[CommandHistoryPos]); } else if(msg.message == WM_KEYDOWN && msg.wParam == VK_ESCAPE && msg.hwnd == CommandEdit) { SendMessage(CommandEdit, WM_SETTEXT, 0, (LPARAM)""); } else { if(msg.message == WM_KEYDOWN) { CommandHistoryPos = -1; } TranslateMessage(&msg); DispatchMessage(&msg); } } if (!offline) { UsbCommand c; if(ReceiveCommandPoll(&c)) UsbCommandReceived(&c); } Sleep(10); } }