// myTimer.cpp : Defines the entry point for the application. // #include "stdafx.h" #include "myTimer.h" #include <math.h> #define MAX_LOADSTRING 100 // Global Variables: HINSTANCE hInst; // current instance TCHAR szTitle[MAX_LOADSTRING]; // The title bar text TCHAR szWindowClass[MAX_LOADSTRING]; // the main window class name // Forward declarations of functions included in this code module: ATOM MyRegisterClass(HINSTANCE hInstance); BOOL InitInstance(HINSTANCE, int); LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM); void mySetIsotropic(HDC hdc, int cx, int cy); void myDrawClockFace(HDC hdc); void myOnPaint(HWND hWnd); void myDrawHand(HDC hdc, int nLength, int nWidth, int nDegrees, COLORREF clrColor); static int s_nPreHour = 0; static int s_nPreMin = 0; static int s_nPreSec = 0; const int IDTIME = 100; void myDrawHand(HDC hdc, int nLength, int nWidth, int nDegrees, COLORREF clrColor) { double nRadians = (double)nDegrees * 0.0174533; POINT pt[2]; pt[0].x = (int)nLength * sin(nRadians); pt[0].y = (int)nLength * cos(nRadians); pt[1].x = -pt[0].x / 5; pt[1].y = -pt[0].y / 5; HPEN hPen = ::CreatePen(PS_SOLID, nWidth, clrColor); HPEN hOldPen = (HPEN)::SelectObject(hdc, hPen); ::MoveToEx(hdc, pt[0].x, pt[0].y, NULL); ::LineTo(hdc, pt[1].x, pt[1].y); ::SelectObject(hdc, hOldPen); ::DeleteObject(hPen); } void mySetIsotropic(HDC hdc, int cx, int cy) { ::SetMapMode(hdc, MM_ISOTROPIC); ::SetWindowExtEx(hdc, 1000, 1000, NULL); ::SetViewportExtEx(hdc, cx, -cy, NULL); ::SetViewportOrgEx(hdc, cx / 2, cy / 2, NULL); } void myDrawClockFace(HDC hdc) { const int SQUARESIZE = 20; static POINT pt[] = { 0, 450, 225, 390, 390, 225, 450, 0, 390, -225, 225, -390, 0, -450, -225, -390, -390, -225, -450, 0, -390, 225, -225, 390 }; HBRUSH hOldBrush = (HBRUSH)::SelectObject(hdc, ::GetStockObject(BLACK_BRUSH)); for (int i = 0; i < 12; i++) { ::Ellipse(hdc, pt[i].x - SQUARESIZE, pt[i].y + SQUARESIZE, pt[i].x + SQUARESIZE, pt[i].y - SQUARESIZE); } ::SelectObject(hdc, hOldBrush); } void myOnPaint(HWND hWnd) { PAINTSTRUCT ps; RECT rt; HDC hdc = ::BeginPaint(hWnd, &ps); ::GetClientRect(hWnd, &rt); mySetIsotropic(hdc, rt.right - rt.left, rt.bottom - rt.top); myDrawClockFace(hdc); ::EndPaint(hWnd, &ps); } int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) { UNREFERENCED_PARAMETER(hPrevInstance); UNREFERENCED_PARAMETER(lpCmdLine); // TODO: Place code here. MSG msg; HACCEL hAccelTable; // Initialize global strings LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING); LoadString(hInstance, IDC_MYTIMER, szWindowClass, MAX_LOADSTRING); MyRegisterClass(hInstance); // Perform application initialization: if (!InitInstance (hInstance, nCmdShow)) { return FALSE; } hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_MYTIMER)); // Main message loop: while (GetMessage(&msg, NULL, 0, 0)) { if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } } return (int) msg.wParam; } // // FUNCTION: MyRegisterClass() // // PURPOSE: Registers the window class. // // COMMENTS: // // This function and its usage are only necessary if you want this code // to be compatible with Win32 systems prior to the 'RegisterClassEx' // function that was added to Windows 95. It is important to call this function // so that the application will get 'well formed' small icons associated // with it. // ATOM MyRegisterClass(HINSTANCE hInstance) { WNDCLASSEX wcex; wcex.cbSize = sizeof(WNDCLASSEX); wcex.style = CS_HREDRAW | CS_VREDRAW; wcex.lpfnWndProc = WndProc; wcex.cbClsExtra = 0; wcex.cbWndExtra = 0; wcex.hInstance = hInstance; wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_MYTIMER)); wcex.hCursor = LoadCursor(NULL, IDC_ARROW); wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1); wcex.lpszMenuName = NULL;//MAKEINTRESOURCE(IDR_MENU_EXIT); wcex.lpszClassName = szWindowClass; wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL)); return RegisterClassEx(&wcex); } // // FUNCTION: InitInstance(HINSTANCE, int) // // PURPOSE: Saves instance handle and creates main window // // COMMENTS: // // In this function, we save the instance handle in a global variable and // create and display the main program window. // BOOL InitInstance(HINSTANCE hInstance, int nCmdShow) { HWND hWnd; hInst = hInstance; // Store instance handle in our global variable hWnd = CreateWindow(szWindowClass, szTitle, WS_POPUP | WS_SYSMENU | WS_SIZEBOX, 100, 100, 500, 500, NULL, NULL, hInstance, NULL); if (!hWnd) { return FALSE; } ShowWindow(hWnd, nCmdShow); UpdateWindow(hWnd); return TRUE; } // // FUNCTION: WndProc(HWND, UINT, WPARAM, LPARAM) // // PURPOSE: Processes messages for the main window. // // WM_COMMAND - process the application menu // WM_PAINT - Paint the main window // WM_DESTROY - post a quit message and return // // void myTest(HWND hWnd); void myTest(HWND hWnd) { PAINTSTRUCT ps; HDC hdc = ::BeginPaint(hWnd, &ps); ::TextOut(hdc, 0, 0, _T("Hello World"), _tcslen(_T("Hello World"))); ::EndPaint(hWnd, &ps); } LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { int wmId, wmEvent; PAINTSTRUCT ps; HDC hdc; RECT rt; HMENU hSysMenu; POINT pt; SYSTEMTIME st; int nID = 0; UINT nHitTest; switch (message) { case WM_CREATE: ::GetLocalTime(&st); s_nPreHour = st.wHour % 12; s_nPreMin = st.wMinute; s_nPreSec = st.wSecond; ::SetTimer(hWnd, IDTIME, 1000, NULL); hSysMenu = ::GetSystemMenu(hWnd, FALSE); ::AppendMenu(hSysMenu, MF_SEPARATOR, 0, NULL); ::AppendMenu(hSysMenu, MF_STRING, ID_MY_EXIT, _T("退出")); break; case WM_TIMER: if (!::IsIconic(hWnd)) { ::GetClientRect(hWnd, &rt); ::GetLocalTime(&st); hdc = ::GetDC(hWnd); mySetIsotropic(hdc, rt.right - rt.left, rt.bottom - rt.top); COLORREF crfColor = ::GetSysColor(COLOR_WINDOW); if (st.wMinute != s_nPreMin) { myDrawHand(hdc, 200, 8, s_nPreHour * 30 + s_nPreMin / 2, crfColor); myDrawHand(hdc, 400, 6, s_nPreMin * 6, crfColor); s_nPreHour = st.wHour; s_nPreMin = st.wMinute; } if (st.wSecond != s_nPreSec) { myDrawHand(hdc, 400, 1, s_nPreSec * 6, crfColor); myDrawHand(hdc, 200, 8, s_nPreHour * 30 + s_nPreMin / 2, RGB(0, 0, 0)); myDrawHand(hdc, 400, 6, s_nPreMin * 6, RGB(0, 0, 0)); s_nPreSec = st.wSecond; myDrawHand(hdc, 400, 1, s_nPreSec * 6, RGB(0, 0, 0)); } } break; case WM_COMMAND: wmId = LOWORD(wParam); wmEvent = HIWORD(wParam); // Parse the menu selections: switch (wmId) { case IDM_ABOUT: DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About); break; case IDM_EXIT: case ID_MY_EXIT: DestroyWindow(hWnd); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } break; case WM_PAINT: hdc = BeginPaint(hWnd, &ps); // TODO: Add any drawing code here... ::GetClientRect(hWnd, &rt); mySetIsotropic(hdc, rt.right - rt.left, rt.bottom - rt.top); myDrawClockFace(hdc); myDrawHand(hdc, 200, 8, s_nPreHour * 30 + s_nPreMin / 2, RGB(0, 0, 0)); myDrawHand(hdc, 400, 6, s_nPreMin * 6, RGB(0, 0, 0)); myDrawHand(hdc, 400, 1, s_nPreSec * 6, RGB(0, 0, 0)); EndPaint(hWnd, &ps); //myOnPaint(hWnd); break; case WM_DESTROY: PostQuitMessage(0); break; case WM_CONTEXTMENU: pt.x = LOWORD(lParam); pt.y = HIWORD(lParam); { hSysMenu = ::GetSystemMenu(hWnd, FALSE); int nID = ::TrackPopupMenu(hSysMenu, TPM_LEFTALIGN | TPM_RETURNCMD, pt.x, pt.y, 0, hWnd, NULL); if (nID > 0) { ::SendMessage(hWnd, WM_SYSCOMMAND, nID, 0); } } break; case WM_SYSCOMMAND: nID = wParam; { if (nID == ID_MY_EXIT) { DestroyWindow(hWnd); break; } } case WM_NCHITTEST: nHitTest = ::DefWindowProc(hWnd, message, wParam, lParam); if (nHitTest == HTCLIENT && ::GetAsyncKeyState(MK_LBUTTON) < 0) { nHitTest = HTCAPTION; } return nHitTest; default: return DefWindowProc(hWnd, message, wParam, lParam); } return 0; } // Message handler for about box. INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam) { UNREFERENCED_PARAMETER(lParam); switch (message) { case WM_INITDIALOG: return (INT_PTR)TRUE; case WM_COMMAND: if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) { EndDialog(hDlg, LOWORD(wParam)); return (INT_PTR)TRUE; } break; } return (INT_PTR)FALSE; }