/* Weather Protocol plugin for Miranda IM Copyright (c) 2012 Miranda NG Team Copyright (c) 2005-2011 Boris Krasnovskiy All Rights Reserved Copyright (c) 2002-2005 Calvin Che This program 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; version 2 of the License. This program 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 this program. If not, see . */ /* This file contain the source that is related to display contact information, including the one shows in user detail and the brief information */ #include "weather.h" //============ CONTACT INFORMATION ============ // initialize user info // lParam = current contact int UserInfoInit(WPARAM wParam, LPARAM lParam) { OPTIONSDIALOGPAGE odp = { sizeof(odp) }; odp.hInstance = hInst; odp.position = 100000000; odp.ptszTitle = _T(WEATHERPROTONAME); if (lParam == 0) { odp.pszTemplate = MAKEINTRESOURCEA(IDD_INFO); odp.pfnDlgProc = DlgProcINIPage; odp.flags = ODPF_TCHAR; UserInfo_AddPage(wParam, &odp); } else { // check if it is a weather contact if (IsMyContact(lParam)) { // register the contact info page odp.pszTemplate = MAKEINTRESOURCEA(IDD_USERINFO); odp.pfnDlgProc = DlgProcUIPage; odp.flags = ODPF_BOLDGROUPS|ODPF_TCHAR; UserInfo_AddPage(wParam, &odp); } } return 0; } // dialog process for the weather tab under user info // lParam = current contact INT_PTR CALLBACK DlgProcUIPage(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam) { WEATHERINFO w; TCHAR str[MAX_TEXT_SIZE]; MCONTACT hContact = (MCONTACT)GetWindowLongPtr(hwndDlg, GWLP_USERDATA); switch (msg) { case WM_INITDIALOG: TranslateDialogDefault(hwndDlg); SendMessage(GetDlgItem(hwndDlg,IDC_MOREDETAIL), BUTTONSETASFLATBTN, TRUE, 0); // save the contact handle for later use hContact = lParam; SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)hContact); // load weather info for the contact w = LoadWeatherInfo(lParam); SetDlgItemText(hwndDlg, IDC_INFO1, GetDisplay(&w, TranslateT("Current condition for %n"), str)); SendDlgItemMessage(hwndDlg, IDC_INFOICON, STM_SETICON, (WPARAM)LoadSkinnedProtoIcon(WEATHERPROTONAME, db_get_w(hContact, WEATHERPROTONAME, "StatusIcon",0)), 0); { // bold and enlarge the current condition LOGFONT lf; HFONT hNormalFont = (HFONT)SendDlgItemMessage(hwndDlg,IDC_INFO2,WM_GETFONT,0,0); GetObject(hNormalFont,sizeof(lf),&lf); lf.lfWeight = FW_BOLD; lf.lfWidth = 7; lf.lfHeight = 15; SendDlgItemMessage(hwndDlg, IDC_INFO2, WM_SETFONT, (WPARAM)CreateFontIndirect(&lf), 0); } // set the text for displaying other current weather conditions data GetDisplay(&w, _T("%c %t"), str); SetDlgItemText(hwndDlg, IDC_INFO2, str); SetDlgItemText(hwndDlg, IDC_INFO3, w.feel); SetDlgItemText(hwndDlg, IDC_INFO4, w.pressure); GetDisplay(&w, _T("%i %w"), str); SetDlgItemText(hwndDlg, IDC_INFO5, str); SetDlgItemText(hwndDlg, IDC_INFO6, w.dewpoint); SetDlgItemText(hwndDlg, IDC_INFO7, w.sunrise); SetDlgItemText(hwndDlg, IDC_INFO8, w.sunset); SetDlgItemText(hwndDlg, IDC_INFO9, w.high); SetDlgItemText(hwndDlg, IDC_INFO10, w.low); GetDisplay(&w, TranslateT("Last update on: %u"), str); SetDlgItemText(hwndDlg, IDC_INFO11, str); SetDlgItemText(hwndDlg, IDC_INFO12, w.humid); SetDlgItemText(hwndDlg, IDC_INFO13, w.vis); break; case WM_DESTROY: Skin_ReleaseIcon((HICON)SendDlgItemMessage(hwndDlg, IDC_INFOICON, STM_SETICON, 0, 0)); DeleteObject((HFONT)SendDlgItemMessage(hwndDlg, IDC_INFO2, WM_GETFONT, 0, 0)); break; case WM_COMMAND: switch(LOWORD(wParam)) { case IDC_MOREDETAIL: HWND hMoreDataDlg = WindowList_Find(hDataWindowList, hContact); if (hMoreDataDlg == NULL) hMoreDataDlg = CreateDialogParam(hInst, MAKEINTRESOURCE(IDD_BRIEF), NULL, DlgProcMoreData, hContact); else { SetForegroundWindow(hMoreDataDlg); SetFocus(hMoreDataDlg); } ShowWindow(GetDlgItem(hMoreDataDlg, IDC_MTEXT), 0); ShowWindow(GetDlgItem(hMoreDataDlg, IDC_DATALIST), 1); } break; } return 0; } //============ BRIEF INFORMATION ============ static int BriefDlgResizer(HWND hwnd, LPARAM lParam, UTILRESIZECONTROL *urc) { switch (urc->wId) { case IDC_HEADERBAR: return RD_ANCHORX_LEFT | RD_ANCHORY_TOP | RD_ANCHORX_WIDTH; case IDC_MTEXT: case IDC_DATALIST: return RD_ANCHORX_LEFT | RD_ANCHORY_TOP | RD_ANCHORX_WIDTH | RD_ANCHORY_HEIGHT; case IDC_MUPDATE: return RD_ANCHORX_LEFT | RD_ANCHORY_BOTTOM; case IDC_MTOGGLE: case IDC_MWEBPAGE: case IDCANCEL: return RD_ANCHORX_RIGHT | RD_ANCHORY_BOTTOM; } return RD_ANCHORX_LEFT | RD_ANCHORY_TOP; } // dialog process for more data in the user info window // lParam = contact handle INT_PTR CALLBACK DlgProcMoreData(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam) { static const unsigned tabstops = 48; MCONTACT hContact = (MCONTACT)GetWindowLongPtr(hwndDlg, GWLP_USERDATA); switch (msg) { case WM_INITDIALOG: // save the contact handle for later use hContact = lParam; SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)hContact); SendDlgItemMessage(hwndDlg, IDC_MTEXT, EM_AUTOURLDETECT, (WPARAM) TRUE, 0); SendDlgItemMessage(hwndDlg, IDC_MTEXT, EM_SETEVENTMASK, 0, ENM_LINK); SendDlgItemMessage(hwndDlg, IDC_MTEXT, EM_SETMARGINS, EC_LEFTMARGIN, 5); SendDlgItemMessage(hwndDlg, IDC_MTEXT, EM_SETTABSTOPS, 1, (LPARAM)&tabstops); // get the list to display { LV_COLUMN lvc = { 0 }; HWND hList = GetDlgItem(hwndDlg, IDC_DATALIST); RECT aRect = { 0 }; GetClientRect(hList, &aRect); // managing styles lvc.mask = LVCF_WIDTH | LVCF_TEXT; ListView_SetExtendedListViewStyleEx(hList, LVS_EX_FULLROWSELECT | LVS_EX_INFOTIP, LVS_EX_FULLROWSELECT | LVS_EX_INFOTIP); // inserting columns lvc.cx = LIST_COLUMN; lvc.pszText = TranslateT("Variable"); ListView_InsertColumn(hList, 0, &lvc); lvc.cx = aRect.right - LIST_COLUMN - GetSystemMetrics(SM_CXVSCROLL) - 3; lvc.pszText = TranslateT("Information"); ListView_InsertColumn(hList, 1, &lvc); // inserting data SendMessage(hwndDlg, WM_UPDATEDATA, 0, 0); } TranslateDialogDefault(hwndDlg); // prevent dups of the window WindowList_Add(hDataWindowList, hwndDlg, hContact); // restore window position Utils_RestoreWindowPositionNoMove(hwndDlg, NULL, WEATHERPROTONAME, "BriefInfo_"); return TRUE; case WM_UPDATEDATA: ListView_DeleteAllItems(GetDlgItem(hwndDlg, IDC_DATALIST)); LoadBriefInfoText(hwndDlg, hContact); DBDataManage(hContact, WDBM_DETAILDISPLAY, (WPARAM)hwndDlg, 0); // set icons { WORD statusIcon = db_get_w(hContact, WEATHERPROTONAME, "StatusIcon", 0); ReleaseIconEx((HICON)SendMessage(hwndDlg, WM_SETICON, ICON_BIG, (LPARAM)LoadSkinnedProtoIconBig(WEATHERPROTONAME, statusIcon))); ReleaseIconEx((HICON)SendMessage(hwndDlg, WM_SETICON, ICON_SMALL, (LPARAM)LoadSkinnedProtoIcon(WEATHERPROTONAME, statusIcon))); } RedrawWindow(GetDlgItem(hwndDlg, IDC_HEADERBAR), NULL, NULL, RDW_INVALIDATE | RDW_UPDATENOW); break; case WM_SIZE: { RECT rc; HWND hList = GetDlgItem(hwndDlg, IDC_DATALIST); GetWindowRect(hList, &rc); ListView_SetColumnWidth(hList, 1, ListView_GetColumnWidth(hList, 1) + (int)LOWORD(lParam) - (rc.right - rc.left)); UTILRESIZEDIALOG urd = { sizeof(urd) }; urd.hwndDlg = hwndDlg; urd.hInstance = hInst; urd.lpTemplate = MAKEINTRESOURCEA(IDD_BRIEF); urd.pfnResizer = BriefDlgResizer; CallService(MS_UTILS_RESIZEDIALOG, 0, (LPARAM)&urd); } break; case WM_GETMINMAXINFO: { LPMINMAXINFO mmi = (LPMINMAXINFO)lParam; // The minimum width in points mmi->ptMinTrackSize.x = 350; // The minimum height in points mmi->ptMinTrackSize.y = 300; } break; case WM_COMMAND: switch(LOWORD(wParam)) { case IDCANCEL: // close the info window DestroyWindow(hwndDlg); break; case IDC_MUPDATE: { HWND hList = GetDlgItem(hwndDlg, IDC_DATALIST); // update current data // set the text to "updating" SetDlgItemText(hwndDlg, IDC_MTEXT, TranslateT("Retrieving new data, please wait...")); ListView_DeleteAllItems(hList); LV_ITEM lvi = { 0 }; lvi.mask = LVIF_TEXT | LVIF_PARAM; lvi.lParam = 1; lvi.pszText = _T(""); lvi.iItem = ListView_InsertItem(hList, &lvi); lvi.pszText = TranslateT("Retrieving new data, please wait..."); ListView_SetItemText(hList, lvi.iItem, 1, lvi.pszText); UpdateSingleStation(hContact, 0); break; } case IDC_MWEBPAGE: LoadForecast(hContact, 0); // read complete forecast break; case IDC_MTOGGLE: if (IsWindowVisible(GetDlgItem(hwndDlg,IDC_DATALIST))) SetDlgItemText(hwndDlg, IDC_MTOGGLE, TranslateT("More Info")); else SetDlgItemText(hwndDlg, IDC_MTOGGLE, TranslateT("Brief Info")); ShowWindow(GetDlgItem(hwndDlg,IDC_DATALIST), (int)!IsWindowVisible( GetDlgItem(hwndDlg,IDC_DATALIST))); ShowWindow(GetDlgItem(hwndDlg,IDC_MTEXT), (int)!IsWindowVisible(GetDlgItem(hwndDlg,IDC_MTEXT))); break; } break; case WM_NOTIFY: { LPNMHDR pNmhdr = (LPNMHDR)lParam; if (pNmhdr->idFrom == IDC_MTEXT && pNmhdr->code == EN_LINK) { ENLINK *enlink = (ENLINK *) lParam; switch (enlink->msg) { case WM_LBUTTONUP: TEXTRANGE tr; tr.chrg = enlink->chrg; tr.lpstrText = ( LPTSTR )mir_alloc( sizeof(TCHAR)*(tr.chrg.cpMax - tr.chrg.cpMin + 8)); SendMessage(pNmhdr->hwndFrom, EM_GETTEXTRANGE, 0, (LPARAM)&tr); CallService(MS_UTILS_OPENURL, OUF_NEWWINDOW | OUF_TCHAR, (LPARAM) tr.lpstrText); mir_free(tr.lpstrText); break; } } } break; case WM_CLOSE: DestroyWindow(hwndDlg); break; case WM_DESTROY: ReleaseIconEx((HICON)SendMessage(hwndDlg, WM_SETICON, ICON_BIG, 0)); ReleaseIconEx((HICON)SendMessage(hwndDlg, WM_SETICON, ICON_SMALL, 0)); Utils_SaveWindowPosition(hwndDlg, NULL, WEATHERPROTONAME, "BriefInfo_"); WindowList_Remove(hDataWindowList, hwndDlg); break; } return FALSE; } // set the title of the dialog and on the which rectangle // also load brief info into message box void LoadBriefInfoText(HWND hwndDlg, MCONTACT hContact) { WEATHERINFO winfo; TCHAR str[4096], str2[4096]; // load weather information from the contact into the WEATHERINFO struct winfo = LoadWeatherInfo(hContact); // check if data exist. If not, display error message box if ( !(BOOL)db_get_b(hContact, WEATHERPROTONAME, "IsUpdated", FALSE)) _tcscpy(str, TranslateT("No information available.\r\nPlease update weather condition first.")); else // set the display text and show the message box GetDisplay(&winfo, opt.bText, str); SetDlgItemText(hwndDlg, IDC_MTEXT, str); GetDisplay(&winfo, opt.bTitle, str); SetWindowText(hwndDlg, str); GetDisplay(&winfo, _T("%c, %t"), str); mir_sntprintf(str2, SIZEOF(str2), _T("%s\n%s"), winfo.city, str); SetDlgItemText(hwndDlg, IDC_HEADERBAR, str2); } // show brief information dialog // wParam = current contact int BriefInfo(WPARAM wParam, LPARAM lParam) { // make sure that the contact is actually a weather one if (IsMyContact(wParam)) { HWND hMoreDataDlg = WindowList_Find(hDataWindowList, wParam); if (hMoreDataDlg != NULL) { SetForegroundWindow(hMoreDataDlg); SetFocus(hMoreDataDlg); } else hMoreDataDlg = CreateDialogParam(hInst, MAKEINTRESOURCE(IDD_BRIEF), NULL, DlgProcMoreData, (LPARAM)wParam); ShowWindow(GetDlgItem(hMoreDataDlg, IDC_DATALIST), 0); ShowWindow(GetDlgItem(hMoreDataDlg, IDC_MTEXT), 1); SetDlgItemText(hMoreDataDlg, IDC_MTOGGLE, TranslateT("More Info")); return 1; } return 0; } INT_PTR BriefInfoSvc(WPARAM wParam, LPARAM lParam) { return BriefInfo(wParam, lParam); }