1 /*************************************************************************
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * Copyright 2008 by Sun Microsystems, Inc.
7 * OpenOffice.org - a multi-platform office productivity suite
9 * $RCSfile: msokill.cpp,v $
11 * This file is part of OpenOffice.org.
13 * OpenOffice.org is free software: you can redistribute it and/or modify
14 * it under the terms of the GNU Lesser General Public License version 3
15 * only, as published by the Free Software Foundation.
17 * OpenOffice.org is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU Lesser General Public License version 3 for more details
21 * (a copy is included in the LICENSE file that accompanied this code).
23 * You should have received a copy of the GNU Lesser General Public License
24 * version 3 along with OpenOffice.org. If not, see
25 * <http://www.openoffice.org/license.html>
26 * for a copy of the LGPLv3 License.
28 ************************************************************************/
31 * Description: Put MSO in a state where it can be closed using
32 * automation or kill it completely
40 BOOL
KillAppFromWindow(HWND hWnd
, char *appName
);
41 BOOL
CloseActiveDialogs();
44 //Callbacks used in closing
45 BOOL CALLBACK
CloseOfficeDlgProc(HWND hwndChild
, LPARAM lParam
);
46 BOOL CALLBACK
CountOfficeDlgProc(HWND hwndChild
, LPARAM lParam
);
48 //Global counters for number of windows found
53 //Dialog window class names for excel, powerpoint and word
54 //These are "Best guess" dialog names
55 const char *pWordDlg2k
= "bosa_sdm_Microsoft Word 9.0";
56 const char *pWordDlg2k3
= "bosa_sdm_Microsoft Office Word";
57 const char *pXLDlg2k
= "bosa_sdm_XL9";
58 const char *pPPDlg2k
= "#32770";
59 const char *pXLDlg2k3
= "bosa_sdm_XL9";
60 const char *pPPDlg2k3
= "#32770";
61 const char *pGenMSODlg
= "bosa_sdm_Mso96";
62 //consider adding - bosa_sdm_Mso96
64 //Command Line Argument constants
65 const char *ARG_HELP
= "--help";
66 const char *ARG_KILL
= "--kill";
67 const char *ARG_CLOSE
= "--close";
69 //Window class names for MSO apps - if we need to look at other office instances
70 //then this list would need to be expanded
71 #define NUM_WINDOWCLASSNAMES 4
72 char *wndClassName
[NUM_WINDOWCLASSNAMES
] = {"OpusApp", "XLMAIN", "PP9FrameClass", "PP10FrameClass"};
74 int main(int argc
, char* argv
[])
81 if (strcmpi(argv
[1], ARG_HELP
) == 0) {
86 if (strcmpi(argv
[1], ARG_KILL
) == 0) {
91 if (strcmpi(argv
[1], ARG_CLOSE
) == 0) {
99 /*--------------------------------------------------------------
100 Find the MSO window if it is available and explictly kill it
101 MSO apps in this case are Excel, Word and PP
102 Use FindWindow Win32 API to detect if they are available
104 -------------------------------------------------------------*/
108 for (int i
=0;i
<NUM_WINDOWCLASSNAMES
;i
++) {
110 while (((hWnd
= FindWindow(wndClassName
[i
], NULL
)) != NULL
) && (j
< 10)) {
111 KillAppFromWindow(hWnd
, wndClassName
[i
]);
117 /*--------------------------------------------------------------
118 Using window handle, get process handle and try to kill the
119 app. This may not be successful if you do not have enough
120 privileges to kill the app.
122 --------------------------------------------------------------*/
123 BOOL
KillAppFromWindow(
134 //The app doesn't appear to be running
136 printf("App %s: window not found.\n,", appName
);
140 DWORD pid
; // Variable to hold the process ID.
141 DWORD dThread
; // Variable to hold (unused) thread ID.
142 dThread
= GetWindowThreadProcessId(hWnd
, &pid
);
143 HANDLE hProcess
; // Handle to existing process
145 hProcess
= OpenProcess(SYNCHRONIZE
| PROCESS_ALL_ACCESS
, TRUE
, pid
);
146 if (hProcess
== NULL
) {
148 printf("App %s : Failed to get process handle",appName
);
152 if (!TerminateProcess(hProcess
, 0)) {
155 FORMAT_MESSAGE_ALLOCATE_BUFFER
| FORMAT_MESSAGE_FROM_SYSTEM
| FORMAT_MESSAGE_IGNORE_INSERTS
,
156 NULL
, GetLastError(), MAKELANGID(LANG_NEUTRAL
, SUBLANG_DEFAULT
),
157 (LPTSTR
) &lpMsgBuf
, 0, NULL
);
158 printf("%s\n", lpMsgBuf
);
159 LocalFree( lpMsgBuf
);
164 printf("Kill %s appears to be successful.\n", appName
);
172 /*--------------------------------------------------------------
173 Close the dialogs if possible based on their window class
174 Use the EnumChildWindows win32 api for this
175 --------------------------------------------------------------*/
176 BOOL
CloseActiveDialogs() {
183 EnumChildWindows(GetDesktopWindow(), CloseOfficeDlgProc
, (LPARAM
) 0);
184 sprintf(buff
, "Word: %d\tExcel: %d\tPP: %d", gWDDlgCount
, gXLDlgCount
, gPPDlgCount
);
188 /*--------------------------------------------------------------
189 Callback for EnumChildWindows that sends close message to
190 any dialogs that match window class of MSO dialogs
192 --------------------------------------------------------------*/
193 BOOL CALLBACK
CloseOfficeDlgProc(HWND hwndChild
, LPARAM
)
195 //bosa_sdm_Microsoft Word 9.0
200 if (GetClassName(hwndChild
, szBuff
, 4096) == 0) {
203 if ((strcmpi(szBuff
, pWordDlg2k
) == 0) || (strcmpi(szBuff
, pWordDlg2k3
) == 0)) {
205 SendMessage(hwndChild
, WM_CLOSE
, 0, 0);
207 if (strcmpi(szBuff
, pXLDlg2k
) == 0) {
209 SendMessage(hwndChild
, WM_CLOSE
, 0, 0);
211 if (strcmpi(szBuff
, pPPDlg2k
) == 0) {
213 SendMessage(hwndChild
, WM_CLOSE
, 0, 0);
215 if (strcmpi(szBuff
, pGenMSODlg
) == 0) {
216 SendMessage(hwndChild
, WM_CLOSE
, 0, 0);
224 /*--------------------------------------------------------------
225 Callback for EnumChildWindows that counts numnnber of
226 dialogs that match window class of MSO dialogs
228 --------------------------------------------------------------*/
229 BOOL CALLBACK
CountOfficeDlgProc(HWND hwndChild
, LPARAM
)
232 if (GetClassName(hwndChild
, szBuff
, 4096) == 0) {
235 if ((strcmpi(szBuff
, pWordDlg2k
) == 0) || (strcmpi(szBuff
, pWordDlg2k3
) == 0)) {
238 if (strcmpi(szBuff
, pXLDlg2k
) == 0) {
241 if (strcmpi(szBuff
, pPPDlg2k
) == 0) {
249 /*--------------------------------------------------------------
250 Simple usage message...
252 -------------------------------------------------------------*/
254 printf("Recovery Assistant Utility - try and put MSO apps in a recoverable state\n");
255 printf("Copyright Sun Microsystems 2008\n");
256 printf("Options:\n");
257 printf(" --help : This message\n");
258 printf(" --close: Attempt to close any open dialogs owned by \n");
259 printf(" MSO apps so Application.Quit() can succeed\n");
260 printf(" --kill : Kill any open MSO apps. Use with caution and only as a last resort\n\n");