1 /* 2 * ieframe - Internet Explorer main frame window 3 * 4 * Copyright 2006 Mike McCormack (for CodeWeavers) 5 * Copyright 2006 Jacek Caban (for CodeWeavers) 6 * 7 * This library is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU Lesser General Public 9 * License as published by the Free Software Foundation; either 10 * version 2.1 of the License, or (at your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 * Lesser General Public License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public 18 * License along with this library; if not, write to the Free Software 19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 20 */ 21 22 #define COBJMACROS 23 24 #include <stdarg.h> 25 26 #include "ieframe.h" 27 #include "resource.h" 28 29 #include "winuser.h" 30 #include "wingdi.h" 31 #include "winnls.h" 32 #include "ole2.h" 33 #include "exdisp.h" 34 #include "oleidl.h" 35 36 #include "mshtmcid.h" 37 #include "shellapi.h" 38 #include "winreg.h" 39 #include "shlwapi.h" 40 #include "intshcut.h" 41 #include "ddeml.h" 42 #include "ieautomation.h" 43 44 #include "wine/debug.h" 45 46 WINE_DEFAULT_DEBUG_CHANNEL(ieframe); 47 48 #define IDI_APPICON 1 49 50 #define WM_UPDATEADDRBAR (WM_APP+1) 51 52 static const WCHAR szIEWinFrame[] = { 'I','E','F','r','a','m','e',0 }; 53 54 /* Windows uses "Microsoft Internet Explorer" */ 55 static const WCHAR wszWineInternetExplorer[] = 56 {'W','i','n','e',' ','I','n','t','e','r','n','e','t',' ','E','x','p','l','o','r','e','r',0}; 57 58 static LONG obj_cnt; 59 static DWORD dde_inst; 60 static HSZ ddestr_iexplore, ddestr_openurl; 61 static struct list ie_list = LIST_INIT(ie_list); 62 63 HRESULT update_ie_statustext(InternetExplorer* This, LPCWSTR text) 64 { 65 if(!SendMessageW(This->status_hwnd, SB_SETTEXTW, MAKEWORD(SB_SIMPLEID, 0), (LPARAM)text)) 66 return E_FAIL; 67 68 return S_OK; 69 } 70 71 static void adjust_ie_docobj_rect(HWND frame, RECT* rc) 72 { 73 HWND hwndRebar = GetDlgItem(frame, IDC_BROWSE_REBAR); 74 HWND hwndStatus = GetDlgItem(frame, IDC_BROWSE_STATUSBAR); 75 INT barHeight = SendMessageW(hwndRebar, RB_GETBARHEIGHT, 0, 0); 76 77 InflateRect(rc, 0, -barHeight); 78 79 if(IsWindowVisible(hwndStatus)) 80 { 81 RECT statusrc; 82 83 GetClientRect(hwndStatus, &statusrc); 84 rc->bottom -= statusrc.bottom - statusrc.top; 85 } 86 } 87 88 static HMENU get_tb_menu(HMENU menu) 89 { 90 HMENU menu_view = GetSubMenu(menu, 1); 91 92 return GetSubMenu(menu_view, 0); 93 } 94 95 static HMENU get_fav_menu(HMENU menu) 96 { 97 return GetSubMenu(menu, 2); 98 } 99 100 static LPWSTR get_fav_url_from_id(HMENU menu, UINT id) 101 { 102 MENUITEMINFOW item; 103 104 item.cbSize = sizeof(item); 105 item.fMask = MIIM_DATA; 106 107 if(!GetMenuItemInfoW(menu, id, FALSE, &item)) 108 return NULL; 109 110 return (LPWSTR)item.dwItemData; 111 } 112 113 static void free_fav_menu_data(HMENU menu) 114 { 115 LPWSTR url; 116 int i; 117 118 for(i = 0; (url = get_fav_url_from_id(menu, ID_BROWSE_GOTOFAV_FIRST + i)); i++) 119 heap_free( url ); 120 } 121 122 static int get_menu_item_count(HMENU menu) 123 { 124 MENUITEMINFOW item; 125 int count = 0; 126 int i; 127 128 item.cbSize = sizeof(item); 129 item.fMask = MIIM_DATA | MIIM_SUBMENU; 130 131 for(i = 0; GetMenuItemInfoW(menu, i, TRUE, &item); i++) 132 { 133 if(item.hSubMenu) 134 count += get_menu_item_count(item.hSubMenu); 135 else 136 count++; 137 } 138 139 return count; 140 } 141 142 static void add_fav_to_menu(HMENU favmenu, HMENU menu, LPWSTR title, LPCWSTR url) 143 { 144 MENUITEMINFOW item; 145 /* Subtract the number of standard elements in the Favorites menu */ 146 int favcount = get_menu_item_count(favmenu) - 2; 147 LPWSTR urlbuf; 148 149 if(favcount > (ID_BROWSE_GOTOFAV_MAX - ID_BROWSE_GOTOFAV_FIRST)) 150 { 151 FIXME("Add support for more than %d Favorites\n", favcount); 152 return; 153 } 154 155 urlbuf = heap_alloc((lstrlenW(url) + 1) * sizeof(WCHAR)); 156 157 if(!urlbuf) 158 return; 159 160 lstrcpyW(urlbuf, url); 161 162 item.cbSize = sizeof(item); 163 item.fMask = MIIM_FTYPE | MIIM_STRING | MIIM_DATA | MIIM_ID; 164 item.fType = MFT_STRING; 165 item.dwTypeData = title; 166 item.wID = ID_BROWSE_GOTOFAV_FIRST + favcount; 167 item.dwItemData = (ULONG_PTR)urlbuf; 168 InsertMenuItemW(menu, -1, TRUE, &item); 169 } 170 171 static void add_favs_to_menu(HMENU favmenu, HMENU menu, LPCWSTR dir) 172 { 173 WCHAR path[MAX_PATH*2]; 174 const WCHAR search[] = {'*',0}; 175 WCHAR* filename; 176 HANDLE findhandle; 177 WIN32_FIND_DATAW finddata; 178 IUniformResourceLocatorW* urlobj; 179 IPersistFile* urlfile = NULL; 180 HRESULT res; 181 182 lstrcpyW(path, dir); 183 PathAppendW(path, search); 184 185 findhandle = FindFirstFileW(path, &finddata); 186 187 if(findhandle == INVALID_HANDLE_VALUE) 188 return; 189 190 res = CoCreateInstance(&CLSID_InternetShortcut, NULL, CLSCTX_INPROC_SERVER, &IID_IUniformResourceLocatorW, (PVOID*)&urlobj); 191 192 if(SUCCEEDED(res)) 193 res = IUnknown_QueryInterface((IUnknown*)urlobj, &IID_IPersistFile, (PVOID*)&urlfile); 194 195 if(SUCCEEDED(res)) 196 { 197 filename = path + lstrlenW(path) - lstrlenW(search); 198 199 do 200 { 201 lstrcpyW(filename, finddata.cFileName); 202 203 if(finddata.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) 204 { 205 MENUITEMINFOW item; 206 const WCHAR ignore1[] = {'.','.',0}; 207 const WCHAR ignore2[] = {'.',0}; 208 209 if(!lstrcmpW(filename, ignore1) || !lstrcmpW(filename, ignore2)) 210 continue; 211 212 item.cbSize = sizeof(item); 213 item.fMask = MIIM_STRING | MIIM_SUBMENU; 214 item.dwTypeData = filename; 215 item.hSubMenu = CreatePopupMenu(); 216 InsertMenuItemW(menu, -1, TRUE, &item); 217 add_favs_to_menu(favmenu, item.hSubMenu, path); 218 } else 219 { 220 WCHAR* fileext; 221 WCHAR* url = NULL; 222 const WCHAR urlext[] = {'.','u','r','l',0}; 223 224 if(lstrcmpiW(PathFindExtensionW(filename), urlext)) 225 continue; 226 227 if(FAILED(IPersistFile_Load(urlfile, path, 0))) 228 continue; 229 230 urlobj->lpVtbl->GetURL(urlobj, &url); 231 232 if(!url) 233 continue; 234 235 fileext = filename + lstrlenW(filename) - lstrlenW(urlext); 236 *fileext = 0; 237 add_fav_to_menu(favmenu, menu, filename, url); 238 } 239 } while(FindNextFileW(findhandle, &finddata)); 240 } 241 242 if(urlfile) 243 IPersistFile_Release(urlfile); 244 245 if(urlobj) 246 IUnknown_Release((IUnknown*)urlobj); 247 248 FindClose(findhandle); 249 } 250 251 static void add_tbs_to_menu(HMENU menu) 252 { 253 HUSKEY toolbar_handle; 254 WCHAR toolbar_key[] = {'S','o','f','t','w','a','r','e','\\', 255 'M','i','c','r','o','s','o','f','t','\\', 256 'I','n','t','e','r','n','e','t',' ', 257 'E','x','p','l','o','r','e','r','\\', 258 'T','o','o','l','b','a','r',0}; 259 260 if(SHRegOpenUSKeyW(toolbar_key, KEY_READ, NULL, &toolbar_handle, TRUE) == ERROR_SUCCESS) 261 { 262 HUSKEY classes_handle; 263 WCHAR classes_key[] = {'S','o','f','t','w','a','r','e','\\', 264 'C','l','a','s','s','e','s','\\','C','L','S','I','D',0}; 265 WCHAR guid[39]; 266 DWORD value_len = sizeof(guid)/sizeof(guid[0]); 267 int i; 268 269 if(SHRegOpenUSKeyW(classes_key, KEY_READ, NULL, &classes_handle, TRUE) != ERROR_SUCCESS) 270 { 271 SHRegCloseUSKey(toolbar_handle); 272 ERR("Failed to open key %s\n", debugstr_w(classes_key)); 273 return; 274 } 275 276 for(i = 0; SHRegEnumUSValueW(toolbar_handle, i, guid, &value_len, NULL, NULL, NULL, SHREGENUM_HKLM) == ERROR_SUCCESS; i++) 277 { 278 WCHAR tb_name[100]; 279 DWORD tb_name_len = sizeof(tb_name)/sizeof(tb_name[0]); 280 HUSKEY tb_class_handle; 281 MENUITEMINFOW item; 282 LSTATUS ret; 283 value_len = sizeof(guid)/sizeof(guid[0]); 284 285 if(lstrlenW(guid) != 38) 286 { 287 TRACE("Found invalid IE toolbar entry: %s\n", debugstr_w(guid)); 288 continue; 289 } 290 291 if(SHRegOpenUSKeyW(guid, KEY_READ, classes_handle, &tb_class_handle, TRUE) != ERROR_SUCCESS) 292 { 293 ERR("Failed to get class info for %s\n", debugstr_w(guid)); 294 continue; 295 } 296 297 ret = SHRegQueryUSValueW(tb_class_handle, NULL, NULL, tb_name, &tb_name_len, TRUE, NULL, 0); 298 299 SHRegCloseUSKey(tb_class_handle); 300 301 if(ret != ERROR_SUCCESS) 302 { 303 ERR("Failed to get toolbar name for %s\n", debugstr_w(guid)); 304 continue; 305 } 306 307 item.cbSize = sizeof(item); 308 item.fMask = MIIM_STRING; 309 item.dwTypeData = tb_name; 310 InsertMenuItemW(menu, GetMenuItemCount(menu), TRUE, &item); 311 } 312 313 SHRegCloseUSKey(classes_handle); 314 SHRegCloseUSKey(toolbar_handle); 315 } 316 } 317 318 static HMENU create_ie_menu(void) 319 { 320 HMENU menu = LoadMenuW(ieframe_instance, MAKEINTRESOURCEW(IDR_BROWSE_MAIN_MENU)); 321 HMENU favmenu = get_fav_menu(menu); 322 WCHAR path[MAX_PATH]; 323 324 add_tbs_to_menu(get_tb_menu(menu)); 325 326 if(SHGetFolderPathW(NULL, CSIDL_COMMON_FAVORITES, NULL, SHGFP_TYPE_CURRENT, path) == S_OK) 327 add_favs_to_menu(favmenu, favmenu, path); 328 329 if(SHGetFolderPathW(NULL, CSIDL_FAVORITES, NULL, SHGFP_TYPE_CURRENT, path) == S_OK) 330 add_favs_to_menu(favmenu, favmenu, path); 331 332 return menu; 333 } 334 335 static void ie_navigate(InternetExplorer* This, LPCWSTR url) 336 { 337 VARIANT variant; 338 339 V_VT(&variant) = VT_BSTR; 340 V_BSTR(&variant) = SysAllocString(url); 341 342 IWebBrowser2_Navigate2(&This->IWebBrowser2_iface, &variant, NULL, NULL, NULL, NULL); 343 344 SysFreeString(V_BSTR(&variant)); 345 } 346 347 static INT_PTR CALLBACK ie_dialog_open_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) 348 { 349 static InternetExplorer* This; 350 351 switch(msg) 352 { 353 case WM_INITDIALOG: 354 This = (InternetExplorer*)lparam; 355 EnableWindow(GetDlgItem(hwnd, IDOK), FALSE); 356 return TRUE; 357 358 case WM_COMMAND: 359 switch(LOWORD(wparam)) 360 { 361 case IDC_BROWSE_OPEN_URL: 362 { 363 HWND hwndurl = GetDlgItem(hwnd, IDC_BROWSE_OPEN_URL); 364 int len = GetWindowTextLengthW(hwndurl); 365 366 EnableWindow(GetDlgItem(hwnd, IDOK), len != 0); 367 break; 368 } 369 case IDOK: 370 { 371 HWND hwndurl = GetDlgItem(hwnd, IDC_BROWSE_OPEN_URL); 372 int len = GetWindowTextLengthW(hwndurl); 373 374 if(len) 375 { 376 VARIANT url; 377 378 V_VT(&url) = VT_BSTR; 379 V_BSTR(&url) = SysAllocStringLen(NULL, len); 380 381 GetWindowTextW(hwndurl, V_BSTR(&url), len + 1); 382 IWebBrowser2_Navigate2(&This->IWebBrowser2_iface, &url, NULL, NULL, NULL, NULL); 383 384 SysFreeString(V_BSTR(&url)); 385 } 386 } 387 /* fall through */ 388 case IDCANCEL: 389 EndDialog(hwnd, wparam); 390 return TRUE; 391 } 392 } 393 return FALSE; 394 } 395 396 static void ie_dialog_about(HWND hwnd) 397 { 398 HICON icon = LoadImageW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDI_APPICON), IMAGE_ICON, 48, 48, LR_SHARED); 399 400 ShellAboutW(hwnd, wszWineInternetExplorer, NULL, icon); 401 402 DestroyIcon(icon); 403 } 404 405 #ifdef __REACTOS__ 406 static void ie_dialog_properties(HWND hwnd) 407 { 408 ShellExecuteW(hwnd, NULL, L"inetcpl.cpl", NULL, NULL, 0); 409 } 410 #endif 411 412 static void add_tb_separator(InternetExplorer *ie) 413 { 414 TBBUTTON btn; 415 416 ZeroMemory(&btn, sizeof(btn)); 417 418 btn.iBitmap = 3; 419 btn.fsStyle = BTNS_SEP; 420 SendMessageW(ie->toolbar_hwnd, TB_ADDBUTTONSW, 1, (LPARAM)&btn); 421 } 422 423 static void add_tb_button(InternetExplorer *ie, int bmp, int cmd, int strId) 424 { 425 TBBUTTON btn; 426 WCHAR buf[30]; 427 428 LoadStringW(ieframe_instance, strId, buf, sizeof(buf)/sizeof(buf[0])); 429 430 btn.iBitmap = bmp; 431 btn.idCommand = cmd; 432 btn.fsState = TBSTATE_ENABLED; 433 btn.fsStyle = BTNS_SHOWTEXT; 434 btn.dwData = 0; 435 btn.iString = (INT_PTR)buf; 436 437 SendMessageW(ie->toolbar_hwnd, TB_ADDBUTTONSW, 1, (LPARAM)&btn); 438 } 439 440 static void enable_toolbar_button(InternetExplorer *ie, int command, BOOL enable) 441 { 442 SendMessageW(ie->toolbar_hwnd, TB_ENABLEBUTTON, command, enable); 443 } 444 445 static void create_rebar(InternetExplorer *ie) 446 { 447 HWND hwndRebar; 448 HWND hwndAddress; 449 REBARINFO rebarinf; 450 REBARBANDINFOW bandinf; 451 WCHAR addr[40]; 452 HIMAGELIST imagelist; 453 SIZE toolbar_size; 454 455 LoadStringW(ieframe_instance, IDS_ADDRESS, addr, sizeof(addr)/sizeof(addr[0])); 456 457 hwndRebar = CreateWindowExW(WS_EX_TOOLWINDOW, REBARCLASSNAMEW, NULL, 458 WS_CHILD|WS_VISIBLE|WS_CLIPSIBLINGS|WS_CLIPCHILDREN|RBS_VARHEIGHT|CCS_TOP|CCS_NODIVIDER, 0, 0, 0, 0, 459 ie->frame_hwnd, (HMENU)IDC_BROWSE_REBAR, ieframe_instance, NULL); 460 461 rebarinf.cbSize = sizeof(rebarinf); 462 rebarinf.fMask = 0; 463 rebarinf.himl = NULL; 464 465 SendMessageW(hwndRebar, RB_SETBARINFO, 0, (LPARAM)&rebarinf); 466 467 ie->toolbar_hwnd = CreateWindowExW(TBSTYLE_EX_MIXEDBUTTONS, TOOLBARCLASSNAMEW, NULL, TBSTYLE_FLAT | WS_CHILD | WS_VISIBLE | CCS_NORESIZE, 468 0, 0, 0, 0, hwndRebar, (HMENU)IDC_BROWSE_TOOLBAR, ieframe_instance, NULL); 469 470 imagelist = ImageList_LoadImageW(ieframe_instance, MAKEINTRESOURCEW(IDB_IETOOLBAR), 32, 0, CLR_NONE, IMAGE_BITMAP, LR_CREATEDIBSECTION); 471 472 SendMessageW(ie->toolbar_hwnd, TB_SETIMAGELIST, 0, (LPARAM)imagelist); 473 SendMessageW(ie->toolbar_hwnd, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0); 474 add_tb_button(ie, 0, ID_BROWSE_BACK, IDS_TB_BACK); 475 add_tb_button(ie, 1, ID_BROWSE_FORWARD, IDS_TB_FORWARD); 476 add_tb_button(ie, 2, ID_BROWSE_STOP, IDS_TB_STOP); 477 add_tb_button(ie, 3, ID_BROWSE_REFRESH, IDS_TB_REFRESH); 478 add_tb_button(ie, 4, ID_BROWSE_HOME, IDS_TB_HOME); 479 add_tb_separator(ie); 480 add_tb_button(ie, 5, ID_BROWSE_PRINT, IDS_TB_PRINT); 481 SendMessageW(ie->toolbar_hwnd, TB_SETBUTTONSIZE, 0, MAKELPARAM(65,50)); 482 SendMessageW(ie->toolbar_hwnd, TB_GETMAXSIZE, 0, (LPARAM)&toolbar_size); 483 484 bandinf.cbSize = sizeof(bandinf); 485 bandinf.fMask = RBBIM_STYLE | RBBIM_CHILD | RBBIM_CHILDSIZE; 486 bandinf.fStyle = RBBS_CHILDEDGE; 487 bandinf.cxMinChild = toolbar_size.cx; 488 bandinf.cyMinChild = toolbar_size.cy+2; 489 bandinf.hwndChild = ie->toolbar_hwnd; 490 491 SendMessageW(hwndRebar, RB_INSERTBANDW, -1, (LPARAM)&bandinf); 492 493 hwndAddress = CreateWindowExW(0, WC_COMBOBOXEXW, NULL, WS_BORDER|WS_CHILD|WS_VISIBLE|CBS_DROPDOWN, 494 0, 0, 100,20,hwndRebar, (HMENU)IDC_BROWSE_ADDRESSBAR, ieframe_instance, NULL); 495 496 bandinf.fMask |= RBBIM_TEXT; 497 bandinf.fStyle = RBBS_CHILDEDGE | RBBS_BREAK; 498 bandinf.lpText = addr; 499 bandinf.cxMinChild = 100; 500 bandinf.cyMinChild = 20; 501 bandinf.hwndChild = hwndAddress; 502 503 SendMessageW(hwndRebar, RB_INSERTBANDW, -1, (LPARAM)&bandinf); 504 } 505 506 static LRESULT iewnd_OnCreate(HWND hwnd, LPCREATESTRUCTW lpcs) 507 { 508 InternetExplorer* This = (InternetExplorer*)lpcs->lpCreateParams; 509 SetWindowLongPtrW(hwnd, 0, (LONG_PTR) lpcs->lpCreateParams); 510 511 This->doc_host.frame_hwnd = This->frame_hwnd = hwnd; 512 513 This->menu = create_ie_menu(); 514 515 This->status_hwnd = CreateStatusWindowW(WS_VISIBLE|WS_CHILD|SBT_NOBORDERS|CCS_NODIVIDER, 516 NULL, hwnd, IDC_BROWSE_STATUSBAR); 517 SendMessageW(This->status_hwnd, SB_SIMPLE, TRUE, 0); 518 519 create_rebar(This); 520 521 return 0; 522 } 523 524 static LRESULT iewnd_OnSize(InternetExplorer *This, INT width, INT height) 525 { 526 HWND hwndRebar = GetDlgItem(This->frame_hwnd, IDC_BROWSE_REBAR); 527 INT barHeight = SendMessageW(hwndRebar, RB_GETBARHEIGHT, 0, 0); 528 RECT docarea = {0, 0, width, height}; 529 530 SendMessageW(This->status_hwnd, WM_SIZE, 0, 0); 531 532 adjust_ie_docobj_rect(This->frame_hwnd, &docarea); 533 534 if(This->doc_host.hwnd) 535 SetWindowPos(This->doc_host.hwnd, NULL, docarea.left, docarea.top, docarea.right, docarea.bottom, 536 SWP_NOZORDER | SWP_NOACTIVATE); 537 538 SetWindowPos(hwndRebar, NULL, 0, 0, width, barHeight, SWP_NOZORDER | SWP_NOACTIVATE); 539 540 return 0; 541 } 542 543 static LRESULT iewnd_OnNotify(InternetExplorer *This, WPARAM wparam, LPARAM lparam) 544 { 545 NMHDR* hdr = (NMHDR*)lparam; 546 547 if(hdr->idFrom == IDC_BROWSE_ADDRESSBAR && hdr->code == CBEN_ENDEDITW) 548 { 549 NMCBEENDEDITW* info = (NMCBEENDEDITW*)lparam; 550 551 if(info->fChanged && info->iWhy == CBENF_RETURN) 552 { 553 VARIANT vt; 554 555 V_VT(&vt) = VT_BSTR; 556 V_BSTR(&vt) = SysAllocString(info->szText); 557 558 IWebBrowser2_Navigate2(&This->IWebBrowser2_iface, &vt, NULL, NULL, NULL, NULL); 559 560 SysFreeString(V_BSTR(&vt)); 561 562 return 0; 563 } 564 } 565 566 if(hdr->idFrom == IDC_BROWSE_REBAR && hdr->code == RBN_HEIGHTCHANGE) 567 { 568 RECT docarea; 569 570 GetClientRect(This->frame_hwnd, &docarea); 571 adjust_ie_docobj_rect(This->frame_hwnd, &docarea); 572 573 if(This->doc_host.hwnd) 574 SetWindowPos(This->doc_host.hwnd, NULL, docarea.left, docarea.top, docarea.right, docarea.bottom, 575 SWP_NOZORDER | SWP_NOACTIVATE); 576 } 577 578 return 0; 579 } 580 581 static LRESULT iewnd_OnDestroy(InternetExplorer *This) 582 { 583 HIMAGELIST list = (HIMAGELIST)SendMessageW(This->toolbar_hwnd, TB_GETIMAGELIST, 0, 0); 584 585 TRACE("%p\n", This); 586 587 free_fav_menu_data(get_fav_menu(This->menu)); 588 ImageList_Destroy(list); 589 This->frame_hwnd = NULL; 590 591 return 0; 592 } 593 594 static LRESULT iewnd_OnCommand(InternetExplorer *This, HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) 595 { 596 switch(LOWORD(wparam)) 597 { 598 case ID_BROWSE_OPEN: 599 DialogBoxParamW(ieframe_instance, MAKEINTRESOURCEW(IDD_BROWSE_OPEN), hwnd, ie_dialog_open_proc, (LPARAM)This); 600 break; 601 602 case ID_BROWSE_PRINT: 603 if(This->doc_host.document) 604 { 605 IOleCommandTarget* target; 606 607 if(FAILED(IUnknown_QueryInterface(This->doc_host.document, &IID_IOleCommandTarget, (LPVOID*)&target))) 608 break; 609 610 IOleCommandTarget_Exec(target, &CGID_MSHTML, IDM_PRINT, OLECMDEXECOPT_DODEFAULT, NULL, NULL); 611 612 IOleCommandTarget_Release(target); 613 } 614 break; 615 616 case ID_BROWSE_HOME: 617 IWebBrowser2_GoHome(&This->IWebBrowser2_iface); 618 break; 619 620 case ID_BROWSE_BACK: 621 IWebBrowser2_GoBack(&This->IWebBrowser2_iface); 622 break; 623 624 case ID_BROWSE_FORWARD: 625 IWebBrowser2_GoForward(&This->IWebBrowser2_iface); 626 break; 627 628 case ID_BROWSE_STOP: 629 IWebBrowser2_Stop(&This->IWebBrowser2_iface); 630 break; 631 632 case ID_BROWSE_REFRESH: 633 IWebBrowser2_Refresh(&This->IWebBrowser2_iface); 634 break; 635 636 case ID_BROWSE_ABOUT: 637 ie_dialog_about(hwnd); 638 break; 639 640 #ifdef __REACTOS__ 641 case ID_BROWSE_PROPERTIES: 642 ie_dialog_properties(hwnd); 643 break; 644 #endif 645 646 case ID_BROWSE_QUIT: 647 ShowWindow(hwnd, SW_HIDE); 648 break; 649 650 default: 651 if(LOWORD(wparam) >= ID_BROWSE_GOTOFAV_FIRST && LOWORD(wparam) <= ID_BROWSE_GOTOFAV_MAX) 652 { 653 LPCWSTR url = get_fav_url_from_id(get_fav_menu(This->menu), LOWORD(wparam)); 654 655 if(url) 656 ie_navigate(This, url); 657 } 658 return DefWindowProcW(hwnd, msg, wparam, lparam); 659 } 660 return 0; 661 } 662 663 static LRESULT update_addrbar(InternetExplorer *This, LPARAM lparam) 664 { 665 HWND hwndRebar = GetDlgItem(This->frame_hwnd, IDC_BROWSE_REBAR); 666 HWND hwndAddress = GetDlgItem(hwndRebar, IDC_BROWSE_ADDRESSBAR); 667 HWND hwndEdit = (HWND)SendMessageW(hwndAddress, CBEM_GETEDITCONTROL, 0, 0); 668 LPCWSTR url = (LPCWSTR)lparam; 669 670 SendMessageW(hwndEdit, WM_SETTEXT, 0, (LPARAM)url); 671 672 return 0; 673 } 674 675 static LRESULT WINAPI ie_window_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) 676 { 677 InternetExplorer *This = (InternetExplorer*) GetWindowLongPtrW(hwnd, 0); 678 679 switch (msg) 680 { 681 case WM_CREATE: 682 return iewnd_OnCreate(hwnd, (LPCREATESTRUCTW)lparam); 683 case WM_CLOSE: 684 TRACE("WM_CLOSE\n"); 685 ShowWindow(hwnd, SW_HIDE); 686 return 0; 687 case WM_SHOWWINDOW: 688 TRACE("WM_SHOWWINDOW %lx\n", wparam); 689 if(wparam) { 690 IWebBrowser2_AddRef(&This->IWebBrowser2_iface); 691 InterlockedIncrement(&This->extern_ref); 692 }else { 693 release_extern_ref(This, TRUE); 694 IWebBrowser2_Release(&This->IWebBrowser2_iface); 695 } 696 break; 697 case WM_DESTROY: 698 return iewnd_OnDestroy(This); 699 case WM_SIZE: 700 return iewnd_OnSize(This, LOWORD(lparam), HIWORD(lparam)); 701 case WM_COMMAND: 702 return iewnd_OnCommand(This, hwnd, msg, wparam, lparam); 703 case WM_NOTIFY: 704 return iewnd_OnNotify(This, wparam, lparam); 705 case WM_DOCHOSTTASK: 706 return process_dochost_tasks(&This->doc_host); 707 case WM_UPDATEADDRBAR: 708 return update_addrbar(This, lparam); 709 } 710 return DefWindowProcW(hwnd, msg, wparam, lparam); 711 } 712 713 void register_iewindow_class(void) 714 { 715 WNDCLASSEXW wc; 716 717 memset(&wc, 0, sizeof wc); 718 wc.cbSize = sizeof(wc); 719 wc.style = 0; 720 wc.lpfnWndProc = ie_window_proc; 721 wc.cbClsExtra = 0; 722 wc.cbWndExtra = sizeof(InternetExplorer*); 723 wc.hInstance = ieframe_instance; 724 wc.hIcon = LoadIconW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDI_APPICON)); 725 wc.hIconSm = LoadImageW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDI_APPICON), IMAGE_ICON, 726 GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), LR_SHARED); 727 wc.hCursor = LoadCursorW(0, (LPWSTR)IDC_ARROW); 728 wc.hbrBackground = 0; 729 wc.lpszClassName = szIEWinFrame; 730 wc.lpszMenuName = NULL; 731 732 RegisterClassExW(&wc); 733 } 734 735 void unregister_iewindow_class(void) 736 { 737 UnregisterClassW(szIEWinFrame, ieframe_instance); 738 } 739 740 static void create_frame_hwnd(InternetExplorer *This) 741 { 742 CreateWindowExW( 743 WS_EX_WINDOWEDGE, 744 szIEWinFrame, wszWineInternetExplorer, 745 WS_CLIPCHILDREN | WS_CAPTION | WS_SYSMENU | WS_THICKFRAME 746 | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, 747 CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, 748 NULL, NULL /* FIXME */, ieframe_instance, This); 749 750 create_doc_view_hwnd(&This->doc_host); 751 } 752 753 static inline InternetExplorer *impl_from_DocHost(DocHost *iface) 754 { 755 return CONTAINING_RECORD(iface, InternetExplorer, doc_host); 756 } 757 758 static ULONG IEDocHost_addref(DocHost *iface) 759 { 760 InternetExplorer *This = impl_from_DocHost(iface); 761 return IWebBrowser2_AddRef(&This->IWebBrowser2_iface); 762 } 763 764 static ULONG IEDocHost_release(DocHost *iface) 765 { 766 InternetExplorer *This = impl_from_DocHost(iface); 767 return IWebBrowser2_Release(&This->IWebBrowser2_iface); 768 } 769 770 static void DocHostContainer_get_docobj_rect(DocHost *This, RECT *rc) 771 { 772 GetClientRect(This->frame_hwnd, rc); 773 adjust_ie_docobj_rect(This->frame_hwnd, rc); 774 } 775 776 static HRESULT DocHostContainer_set_status_text(DocHost *iface, const WCHAR *text) 777 { 778 InternetExplorer *This = impl_from_DocHost(iface); 779 return update_ie_statustext(This, text); 780 } 781 782 static void DocHostContainer_on_command_state_change(DocHost *iface, LONG command, BOOL enable) 783 { 784 InternetExplorer *This = impl_from_DocHost(iface); 785 786 switch(command) { 787 case CSC_NAVIGATEBACK: 788 enable_toolbar_button(This, ID_BROWSE_BACK, enable); 789 break; 790 case CSC_NAVIGATEFORWARD: 791 enable_toolbar_button(This, ID_BROWSE_FORWARD, enable); 792 break; 793 } 794 } 795 796 static void DocHostContainer_set_url(DocHost* iface, const WCHAR *url) 797 { 798 InternetExplorer *This = impl_from_DocHost(iface); 799 800 This->nohome = FALSE; 801 SendMessageW(This->frame_hwnd, WM_UPDATEADDRBAR, 0, (LPARAM)url); 802 } 803 804 static const IDocHostContainerVtbl DocHostContainerVtbl = { 805 IEDocHost_addref, 806 IEDocHost_release, 807 DocHostContainer_get_docobj_rect, 808 DocHostContainer_set_status_text, 809 DocHostContainer_on_command_state_change, 810 DocHostContainer_set_url 811 }; 812 813 static HRESULT create_ie(InternetExplorer **ret_obj) 814 { 815 InternetExplorer *ret; 816 817 ret = heap_alloc_zero(sizeof(InternetExplorer)); 818 if(!ret) 819 return E_OUTOFMEMORY; 820 821 ret->ref = 1; 822 823 DocHost_Init(&ret->doc_host, &ret->IWebBrowser2_iface, &DocHostContainerVtbl); 824 825 InternetExplorer_WebBrowser_Init(ret); 826 827 HlinkFrame_Init(&ret->hlink_frame, (IUnknown*)&ret->IWebBrowser2_iface, &ret->doc_host); 828 829 create_frame_hwnd(ret); 830 831 InterlockedIncrement(&obj_cnt); 832 list_add_tail(&ie_list, &ret->entry); 833 *ret_obj = ret; 834 return S_OK; 835 } 836 837 HRESULT WINAPI InternetExplorer_Create(IClassFactory *iface, IUnknown *pOuter, REFIID riid, void **ppv) 838 { 839 InternetExplorer *ret; 840 HRESULT hres; 841 842 TRACE("(%p %s %p)\n", pOuter, debugstr_guid(riid), ppv); 843 844 hres = create_ie(&ret); 845 if(FAILED(hres)) 846 return hres; 847 848 hres = IWebBrowser2_QueryInterface(&ret->IWebBrowser2_iface, riid, ppv); 849 IWebBrowser2_Release(&ret->IWebBrowser2_iface); 850 if(FAILED(hres)) 851 return hres; 852 853 return S_OK; 854 } 855 856 /****************************************************************** 857 * IInternetExplorerManager implementation 858 */ 859 struct InternetExplorerManager { 860 IInternetExplorerManager IInternetExplorerManager_iface; 861 LONG ref; 862 }; 863 864 static inline InternetExplorerManager *impl_from_IInternetExplorerManager(IInternetExplorerManager *iface) 865 { 866 return CONTAINING_RECORD(iface, InternetExplorerManager, IInternetExplorerManager_iface); 867 } 868 869 static HRESULT WINAPI InternetExplorerManager_QueryInterface(IInternetExplorerManager *iface, REFIID riid, void **out) 870 { 871 TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), out); 872 873 if (IsEqualGUID(riid, &IID_IInternetExplorerManager) || IsEqualGUID(riid, &IID_IUnknown)) 874 { 875 IInternetExplorerManager_AddRef(iface); 876 *out = iface; 877 return S_OK; 878 } 879 880 FIXME("interface %s not implemented\n", debugstr_guid(riid)); 881 *out = NULL; 882 return E_NOINTERFACE; 883 } 884 885 static ULONG WINAPI InternetExplorerManager_AddRef(IInternetExplorerManager *iface) 886 { 887 InternetExplorerManager *This = impl_from_IInternetExplorerManager(iface); 888 ULONG ref = InterlockedIncrement(&This->ref); 889 890 TRACE("(%p) increasing refcount to %u\n", iface, ref); 891 892 return ref; 893 } 894 895 static ULONG WINAPI InternetExplorerManager_Release(IInternetExplorerManager *iface) 896 { 897 InternetExplorerManager *This = impl_from_IInternetExplorerManager(iface); 898 ULONG ref = InterlockedDecrement(&This->ref); 899 900 TRACE("(%p) decreasing refcount to %u\n", iface, ref); 901 902 if (ref == 0) 903 HeapFree(GetProcessHeap(), 0, This); 904 905 return ref; 906 } 907 908 static HRESULT WINAPI InternetExplorerManager_CreateObject(IInternetExplorerManager *iface, DWORD config, LPCWSTR url, REFIID riid, void **ppv) 909 { 910 FIXME("(%p)->(0x%x, %s, %s, %p) stub!\n", iface, config, debugstr_w(url), debugstr_guid(riid), ppv); 911 912 return E_NOTIMPL; 913 } 914 915 static const IInternetExplorerManagerVtbl InternetExplorerManager_vtbl = 916 { 917 InternetExplorerManager_QueryInterface, 918 InternetExplorerManager_AddRef, 919 InternetExplorerManager_Release, 920 InternetExplorerManager_CreateObject, 921 }; 922 923 HRESULT WINAPI InternetExplorerManager_Create(IClassFactory *iface, IUnknown *pOuter, REFIID riid, void **ppv) 924 { 925 InternetExplorerManager *ret; 926 HRESULT hr; 927 928 TRACE("(%p %s %p)\n", pOuter, debugstr_guid(riid), ppv); 929 930 if (!(ret = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*ret)))) 931 return E_OUTOFMEMORY; 932 933 ret->IInternetExplorerManager_iface.lpVtbl = &InternetExplorerManager_vtbl; 934 ret->ref = 1; 935 936 hr = IInternetExplorerManager_QueryInterface(&ret->IInternetExplorerManager_iface, riid, ppv); 937 IInternetExplorerManager_Release(&ret->IInternetExplorerManager_iface); 938 939 return hr; 940 } 941 942 void released_obj(void) 943 { 944 if(!InterlockedDecrement(&obj_cnt)) 945 PostQuitMessage(0); 946 } 947 948 static BOOL create_ie_window(BOOL nohome, const WCHAR *cmdline) 949 { 950 InternetExplorer *ie; 951 HRESULT hres; 952 953 hres = create_ie(&ie); 954 if(FAILED(hres)) 955 return FALSE; 956 957 IWebBrowser2_put_Visible(&ie->IWebBrowser2_iface, VARIANT_TRUE); 958 IWebBrowser2_put_MenuBar(&ie->IWebBrowser2_iface, VARIANT_TRUE); 959 960 if(!*cmdline) { 961 if (nohome) 962 ie->nohome = TRUE; 963 else 964 IWebBrowser2_GoHome(&ie->IWebBrowser2_iface); 965 }else { 966 VARIANT var_url; 967 int cmdlen; 968 969 cmdlen = strlenW(cmdline); 970 if(cmdlen > 2 && cmdline[0] == '"' && cmdline[cmdlen-1] == '"') { 971 cmdline++; 972 cmdlen -= 2; 973 } 974 975 V_VT(&var_url) = VT_BSTR; 976 V_BSTR(&var_url) = SysAllocStringLen(cmdline, cmdlen); 977 978 /* navigate to the first page */ 979 IWebBrowser2_Navigate2(&ie->IWebBrowser2_iface, &var_url, NULL, NULL, NULL, NULL); 980 981 SysFreeString(V_BSTR(&var_url)); 982 } 983 984 IWebBrowser2_Release(&ie->IWebBrowser2_iface); 985 return TRUE; 986 } 987 988 static HDDEDATA open_dde_url(WCHAR *dde_url) 989 { 990 InternetExplorer *ie = NULL, *iter; 991 WCHAR *url, *url_end; 992 VARIANT urlv; 993 HRESULT hres; 994 995 TRACE("%s\n", debugstr_w(dde_url)); 996 997 url = dde_url; 998 if(*url == '"') { 999 url++; 1000 url_end = strchrW(url, '"'); 1001 if(!url_end) { 1002 FIXME("missing string terminator\n"); 1003 return 0; 1004 } 1005 *url_end = 0; 1006 }else { 1007 url_end = strchrW(url, ','); 1008 if(url_end) 1009 *url_end = 0; 1010 else 1011 url_end = url + strlenW(url); 1012 } 1013 1014 LIST_FOR_EACH_ENTRY(iter, &ie_list, InternetExplorer, entry) { 1015 if(iter->nohome) { 1016 IWebBrowser2_AddRef(&iter->IWebBrowser2_iface); 1017 ie = iter; 1018 break; 1019 } 1020 } 1021 1022 if(!ie) { 1023 hres = create_ie(&ie); 1024 if(FAILED(hres)) 1025 return 0; 1026 } 1027 1028 IWebBrowser2_put_Visible(&ie->IWebBrowser2_iface, VARIANT_TRUE); 1029 IWebBrowser2_put_MenuBar(&ie->IWebBrowser2_iface, VARIANT_TRUE); 1030 1031 V_VT(&urlv) = VT_BSTR; 1032 V_BSTR(&urlv) = SysAllocStringLen(url, url_end-url); 1033 if(!V_BSTR(&urlv)) { 1034 IWebBrowser2_Release(&ie->IWebBrowser2_iface); 1035 return 0; 1036 } 1037 1038 hres = IWebBrowser2_Navigate2(&ie->IWebBrowser2_iface, &urlv, NULL, NULL, NULL, NULL); 1039 if(FAILED(hres)) 1040 return 0; 1041 1042 IWebBrowser2_Release(&ie->IWebBrowser2_iface); 1043 return ULongToHandle(DDE_FACK); 1044 } 1045 1046 static HDDEDATA WINAPI dde_proc(UINT type, UINT uFmt, HCONV hConv, HSZ hsz1, HSZ hsz2, HDDEDATA data, 1047 ULONG_PTR dwData1, ULONG_PTR dwData2) 1048 { 1049 switch(type) { 1050 case XTYP_CONNECT: 1051 TRACE("XTYP_CONNECT %p\n", hsz1); 1052 return ULongToHandle(!DdeCmpStringHandles(hsz1, ddestr_openurl)); 1053 1054 case XTYP_EXECUTE: { 1055 WCHAR *url; 1056 DWORD size; 1057 HDDEDATA ret; 1058 1059 TRACE("XTYP_EXECUTE %p\n", data); 1060 1061 size = DdeGetData(data, NULL, 0, 0); 1062 if(!size) { 1063 WARN("size = 0\n"); 1064 break; 1065 } 1066 1067 url = heap_alloc(size); 1068 if(!url) 1069 break; 1070 1071 if(DdeGetData(data, (BYTE*)url, size, 0) != size) { 1072 ERR("error during read\n"); 1073 heap_free(url); 1074 break; 1075 } 1076 1077 ret = open_dde_url(url); 1078 1079 heap_free(url); 1080 return ret; 1081 } 1082 1083 case XTYP_REQUEST: 1084 FIXME("XTYP_REQUEST\n"); 1085 break; 1086 1087 default: 1088 TRACE("type %d\n", type); 1089 } 1090 1091 return NULL; 1092 } 1093 1094 static void init_dde(void) 1095 { 1096 UINT res; 1097 1098 static const WCHAR iexploreW[] = {'I','E','x','p','l','o','r','e',0}; 1099 static const WCHAR openurlW[] = {'W','W','W','_','O','p','e','n','U','R','L',0}; 1100 1101 res = DdeInitializeW(&dde_inst, dde_proc, CBF_SKIP_ALLNOTIFICATIONS | CBF_FAIL_ADVISES | CBF_FAIL_POKES, 0); 1102 if(res != DMLERR_NO_ERROR) { 1103 WARN("DdeInitialize failed: %u\n", res); 1104 return; 1105 } 1106 1107 ddestr_iexplore = DdeCreateStringHandleW(dde_inst, iexploreW, CP_WINUNICODE); 1108 if(!ddestr_iexplore) 1109 WARN("Failed to create string handle: %u\n", DdeGetLastError(dde_inst)); 1110 1111 ddestr_openurl = DdeCreateStringHandleW(dde_inst, openurlW, CP_WINUNICODE); 1112 if(!ddestr_openurl) 1113 WARN("Failed to create string handle: %u\n", DdeGetLastError(dde_inst)); 1114 1115 if(!DdeNameService(dde_inst, ddestr_iexplore, 0, DNS_REGISTER)) 1116 WARN("DdeNameService failed\n"); 1117 } 1118 1119 static void release_dde(void) 1120 { 1121 if(ddestr_iexplore) 1122 DdeNameService(dde_inst, ddestr_iexplore, 0, DNS_UNREGISTER); 1123 if(ddestr_openurl) 1124 DdeFreeStringHandle(dde_inst, ddestr_openurl); 1125 if(ddestr_iexplore) 1126 DdeFreeStringHandle(dde_inst, ddestr_iexplore); 1127 DdeUninitialize(dde_inst); 1128 } 1129 1130 /****************************************************************** 1131 * IEWinMain (ieframe.101) 1132 * 1133 * Only returns on error. 1134 */ 1135 DWORD WINAPI IEWinMain(const WCHAR *cmdline, int nShowWindow) 1136 { 1137 MSG msg; 1138 HRESULT hres; 1139 BOOL embedding = FALSE, nohome = FALSE, manager = FALSE; 1140 DWORD reg_cookie; 1141 1142 static const WCHAR embeddingW[] = {'-','e','m','b','e','d','d','i','n','g',0}; 1143 static const WCHAR nohomeW[] = {'-','n','o','h','o','m','e',0}; 1144 static const WCHAR startmanagerW[] = {'-','s','t','a','r','t','m','a','n','a','g','e','r',0}; 1145 1146 TRACE("%s %d\n", debugstr_w(cmdline), nShowWindow); 1147 1148 CoInitialize(NULL); 1149 1150 init_dde(); 1151 1152 while (*cmdline) 1153 { 1154 int length = 0; 1155 1156 while (*cmdline == ' ' || *cmdline == '\t') cmdline++; 1157 if (!*cmdline) break; 1158 1159 while (cmdline[length] && cmdline[length] != ' ' && cmdline[length] != '\t') length++; 1160 1161 if (!strncmpiW(cmdline, embeddingW, length)) 1162 embedding = TRUE; 1163 else if (!strncmpiW(cmdline, nohomeW, length)) 1164 nohome = TRUE; 1165 else if (!strncmpiW(cmdline, startmanagerW, length)) 1166 manager = TRUE; 1167 else 1168 break; 1169 1170 cmdline += length; 1171 } 1172 1173 if (manager) 1174 hres = CoRegisterClassObject(&CLSID_InternetExplorerManager, 1175 (IUnknown*)&InternetExplorerManagerFactory, CLSCTX_SERVER, 1176 REGCLS_SINGLEUSE, ®_cookie); 1177 else 1178 hres = CoRegisterClassObject(&CLSID_InternetExplorer, 1179 (IUnknown*)&InternetExplorerFactory, CLSCTX_SERVER, 1180 REGCLS_MULTIPLEUSE, ®_cookie); 1181 1182 if (FAILED(hres)) 1183 { 1184 ERR("failed to register CLSID_InternetExplorer%s: %08x\n", manager ? "Manager" : "", hres); 1185 CoUninitialize(); 1186 ExitProcess(1); 1187 } 1188 1189 if (!embedding) 1190 { 1191 if(!create_ie_window(nohome, cmdline)) 1192 { 1193 CoUninitialize(); 1194 ExitProcess(1); 1195 } 1196 } 1197 1198 /* run the message loop for this thread */ 1199 while (GetMessageW(&msg, 0, 0, 0)) 1200 { 1201 TranslateMessage(&msg); 1202 DispatchMessageW(&msg); 1203 } 1204 1205 CoRevokeClassObject(reg_cookie); 1206 release_dde(); 1207 1208 CoUninitialize(); 1209 1210 ExitProcess(0); 1211 return 0; 1212 } 1213