-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathdllmain.cpp
175 lines (144 loc) · 4.97 KB
/
dllmain.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
/****************************** Module Header ******************************\
Module Name: dllmain.cpp
Project: CppShellExtContextMenuHandler
Copyright (c) Microsoft Corporation.
The file implements DllMain, and the DllGetClassObject, DllCanUnloadNow,
DllRegisterServer, DllUnregisterServer functions that are necessary for a COM
DLL.
DllGetClassObject invokes the class factory defined in ClassFactory.h/cpp and
queries to the specific interface.
DllCanUnloadNow checks if we can unload the component from the memory.
DllRegisterServer registers the COM server and the context menu handler in
the registry by invoking the helper functions defined in Reg.h/cpp. The
context menu handler is associated with the .cpp file class.
DllUnregisterServer unregisters the COM server and the context menu handler.
This source is subject to the Microsoft Public License.
See http://www.microsoft.com/opensource/licenses.mspx#Ms-PL.
All other rights reserved.
THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND,
EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE.
\***************************************************************************/
#include <windows.h>
#include <Guiddef.h>
#include "ClassFactory.h" // For the class factory
#include "Reg.h"
#include "common.h"
// {BFD98515-CD74-48A4-98E2-13D209E3EE4F}
// When you write your own handler, you must create a new CLSID by using the
// "Create GUID" tool in the Tools menu, and specify the CLSID value here.
const CLSID CLSID_FileContextMenuExt =
{ 0xBFD98515, 0xCD74, 0x48A4, { 0x98, 0xE2, 0x13, 0xD2, 0x09, 0xE3, 0xEE, 0x4F } };
HINSTANCE g_hInst = NULL;
long g_cDllRef = 0;
BOOL APIENTRY DllMain(HMODULE hModule, DWORD dwReason, LPVOID lpReserved)
{
switch (dwReason)
{
case DLL_PROCESS_ATTACH:
// Hold the instance of this DLL module, we will use it to get the
// path of the DLL to register the component.
g_hInst = hModule;
DisableThreadLibraryCalls(hModule);
break;
case DLL_THREAD_ATTACH:
case DLL_THREAD_DETACH:
case DLL_PROCESS_DETACH:
break;
}
return TRUE;
}
//
// FUNCTION: DllGetClassObject
//
// PURPOSE: Create the class factory and query to the specific interface.
//
// PARAMETERS:
// * rclsid - The CLSID that will associate the correct data and code.
// * riid - A reference to the identifier of the interface that the caller
// is to use to communicate with the class object.
// * ppv - The address of a pointer variable that receives the interface
// pointer requested in riid. Upon successful return, *ppv contains the
// requested interface pointer. If an error occurs, the interface pointer
// is NULL.
//
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppv)
{
HRESULT hr = CLASS_E_CLASSNOTAVAILABLE;
if (IsEqualCLSID(CLSID_FileContextMenuExt, rclsid))
{
hr = E_OUTOFMEMORY;
ClassFactory *pClassFactory = new ClassFactory();
if (pClassFactory)
{
hr = pClassFactory->QueryInterface(riid, ppv);
pClassFactory->Release();
}
}
return hr;
}
//
// FUNCTION: DllCanUnloadNow
//
// PURPOSE: Check if we can unload the component from the memory.
//
// NOTE: The component can be unloaded from the memory when its reference
// count is zero (i.e. nobody is still using the component).
//
STDAPI DllCanUnloadNow(void)
{
return g_cDllRef > 0 ? S_FALSE : S_OK;
}
//
// FUNCTION: DllRegisterServer
//
// PURPOSE: Register the COM server and the context menu handler.
//
STDAPI DllRegisterServer(void)
{
HRESULT hr;
wchar_t szModule[MAX_PATH];
if (GetModuleFileName(g_hInst, szModule, ARRAYSIZE(szModule)) == 0)
{
hr = HRESULT_FROM_WIN32(GetLastError());
return hr;
}
// Register the component.
hr = RegisterInprocServer(szModule, CLSID_FileContextMenuExt,
L_Friendly_Class_Name,
L"Apartment");
if (SUCCEEDED(hr))
{
// Register the context menu handler. The context menu handler is
// associated with the any file class.
// Control the visibility in QueryContextMenu
hr = RegisterShellExtContextMenuHandler(L"*",
CLSID_FileContextMenuExt,
L_Friendly_Menu_Name);
}
return hr;
}
//
// FUNCTION: DllUnregisterServer
//
// PURPOSE: Unregister the COM server and the context menu handler.
//
STDAPI DllUnregisterServer(void)
{
HRESULT hr = S_OK;
wchar_t szModule[MAX_PATH];
if (GetModuleFileName(g_hInst, szModule, ARRAYSIZE(szModule)) == 0)
{
hr = HRESULT_FROM_WIN32(GetLastError());
return hr;
}
// Unregister the component.
hr = UnregisterInprocServer(CLSID_FileContextMenuExt);
if (SUCCEEDED(hr))
{
// Unregister the context menu handler.
hr = UnregisterShellExtContextMenuHandler(L"*",
CLSID_FileContextMenuExt);
}
return hr;
}