diff options
author | Rob Mensching <rob@firegiant.com> | 2021-04-22 17:57:31 -0700 |
---|---|---|
committer | Rob Mensching <rob@firegiant.com> | 2021-04-29 16:49:19 -0700 |
commit | 03d5c46cbbb94f73ac468709345fc6a0e50def8d (patch) | |
tree | 9460f73cc308ead863e7a28a68fdeb6d85a2e69e /src/samples/thmviewer/load.cpp | |
parent | 35606d2cd04a7b1bec1d669f9619501dff2bf9dc (diff) | |
download | wix-03d5c46cbbb94f73ac468709345fc6a0e50def8d.tar.gz wix-03d5c46cbbb94f73ac468709345fc6a0e50def8d.tar.bz2 wix-03d5c46cbbb94f73ac468709345fc6a0e50def8d.zip |
Move Tools into wix
Diffstat (limited to 'src/samples/thmviewer/load.cpp')
-rw-r--r-- | src/samples/thmviewer/load.cpp | 221 |
1 files changed, 221 insertions, 0 deletions
diff --git a/src/samples/thmviewer/load.cpp b/src/samples/thmviewer/load.cpp new file mode 100644 index 00000000..0267402a --- /dev/null +++ b/src/samples/thmviewer/load.cpp | |||
@@ -0,0 +1,221 @@ | |||
1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
2 | |||
3 | #include "precomp.h" | ||
4 | |||
5 | struct LOAD_THREAD_CONTEXT | ||
6 | { | ||
7 | HWND hWnd; | ||
8 | LPCWSTR wzThemePath; | ||
9 | LPCWSTR wzWxlPath; | ||
10 | |||
11 | HANDLE hInit; | ||
12 | }; | ||
13 | |||
14 | static DWORD WINAPI LoadThreadProc( | ||
15 | __in LPVOID pvContext | ||
16 | ); | ||
17 | |||
18 | |||
19 | extern "C" HRESULT LoadStart( | ||
20 | __in_z LPCWSTR wzThemePath, | ||
21 | __in_z_opt LPCWSTR wzWxlPath, | ||
22 | __in HWND hWnd, | ||
23 | __out HANDLE* phThread | ||
24 | ) | ||
25 | { | ||
26 | HRESULT hr = S_OK; | ||
27 | HANDLE rgHandles[2] = { }; | ||
28 | LOAD_THREAD_CONTEXT context = { }; | ||
29 | |||
30 | rgHandles[0] = ::CreateEventW(NULL, TRUE, FALSE, NULL); | ||
31 | ExitOnNullWithLastError(rgHandles[0], hr, "Failed to create load init event."); | ||
32 | |||
33 | context.hWnd = hWnd; | ||
34 | context.wzThemePath = wzThemePath; | ||
35 | context.wzWxlPath = wzWxlPath; | ||
36 | context.hInit = rgHandles[0]; | ||
37 | |||
38 | rgHandles[1] = ::CreateThread(NULL, 0, LoadThreadProc, &context, 0, NULL); | ||
39 | ExitOnNullWithLastError(rgHandles[1], hr, "Failed to create load thread."); | ||
40 | |||
41 | ::WaitForMultipleObjects(countof(rgHandles), rgHandles, FALSE, INFINITE); | ||
42 | |||
43 | *phThread = rgHandles[1]; | ||
44 | rgHandles[1] = NULL; | ||
45 | |||
46 | LExit: | ||
47 | ReleaseHandle(rgHandles[1]); | ||
48 | ReleaseHandle(rgHandles[0]); | ||
49 | return hr; | ||
50 | } | ||
51 | |||
52 | |||
53 | static DWORD WINAPI LoadThreadProc( | ||
54 | __in LPVOID pvContext | ||
55 | ) | ||
56 | { | ||
57 | HRESULT hr = S_OK; | ||
58 | WIX_LOCALIZATION* pWixLoc = NULL; | ||
59 | LPWSTR sczThemePath = NULL; | ||
60 | LPWSTR sczWxlPath = NULL; | ||
61 | BOOL fComInitialized = FALSE; | ||
62 | HANDLE hDirectory = INVALID_HANDLE_VALUE; | ||
63 | LPWSTR sczDirectory = NULL; | ||
64 | LPWSTR wzFileName = NULL; | ||
65 | |||
66 | THEME* pTheme = NULL; | ||
67 | HANDLE_THEME* pHandle = NULL; | ||
68 | |||
69 | LOAD_THREAD_CONTEXT* pContext = static_cast<LOAD_THREAD_CONTEXT*>(pvContext); | ||
70 | HWND hWnd = pContext->hWnd; | ||
71 | |||
72 | hr = StrAllocString(&sczThemePath, pContext->wzThemePath, 0); | ||
73 | ExitOnFailure(hr, "Failed to copy path to initial theme file."); | ||
74 | |||
75 | if (pContext->wzWxlPath) | ||
76 | { | ||
77 | hr = StrAllocString(&sczWxlPath, pContext->wzWxlPath, 0); | ||
78 | ExitOnFailure(hr, "Failed to copy .wxl path to initial file."); | ||
79 | } | ||
80 | |||
81 | // We can signal the initialization event as soon as we have copied the context | ||
82 | // values into local variables. | ||
83 | ::SetEvent(pContext->hInit); | ||
84 | |||
85 | hr = ::CoInitialize(NULL); | ||
86 | ExitOnFailure(hr, "Failed to initialize COM on load thread."); | ||
87 | fComInitialized = TRUE; | ||
88 | |||
89 | // Open a handle to the directory so we can put a notification on it. | ||
90 | hr = PathGetDirectory(sczThemePath, &sczDirectory); | ||
91 | ExitOnFailure(hr, "Failed to get path directory."); | ||
92 | |||
93 | wzFileName = PathFile(sczThemePath); | ||
94 | |||
95 | hDirectory = ::CreateFileW(sczDirectory, FILE_LIST_DIRECTORY, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); | ||
96 | if (INVALID_HANDLE_VALUE == hDirectory) | ||
97 | { | ||
98 | ExitWithLastError(hr, "Failed to open directory: %ls", sczDirectory); | ||
99 | } | ||
100 | |||
101 | BOOL fUpdated = FALSE; | ||
102 | do | ||
103 | { | ||
104 | // Get the last modified time on the file we're loading for verification that the | ||
105 | // file actually gets changed down below. | ||
106 | FILETIME ftModified = { }; | ||
107 | FileGetTime(sczThemePath, NULL, NULL, &ftModified); | ||
108 | |||
109 | ::SendMessageW(hWnd, WM_THMVWR_THEME_LOAD_BEGIN, 0, 0); | ||
110 | |||
111 | // Try to load the theme file. | ||
112 | hr = ThemeLoadFromFile(sczThemePath, &pTheme); | ||
113 | if (FAILED(hr)) | ||
114 | { | ||
115 | ::SendMessageW(hWnd, WM_THMVWR_THEME_LOAD_ERROR, 0, hr); | ||
116 | } | ||
117 | else | ||
118 | { | ||
119 | if (sczWxlPath) | ||
120 | { | ||
121 | hr = LocLoadFromFile(sczWxlPath, &pWixLoc); | ||
122 | ExitOnFailure(hr, "Failed to load loc file from path: %ls", sczWxlPath); | ||
123 | |||
124 | hr = ThemeLocalize(pTheme, pWixLoc); | ||
125 | ExitOnFailure(hr, "Failed to localize theme: %ls", sczWxlPath); | ||
126 | } | ||
127 | |||
128 | hr = AllocHandleTheme(pTheme, &pHandle); | ||
129 | ExitOnFailure(hr, "Failed to allocate handle to theme"); | ||
130 | |||
131 | ::SendMessageW(hWnd, WM_THMVWR_NEW_THEME, 0, reinterpret_cast<LPARAM>(pHandle)); | ||
132 | pHandle = NULL; | ||
133 | } | ||
134 | |||
135 | fUpdated = FALSE; | ||
136 | do | ||
137 | { | ||
138 | DWORD rgbNotifications[1024]; | ||
139 | DWORD cbRead = 0; | ||
140 | if (!::ReadDirectoryChangesW(hDirectory, rgbNotifications, sizeof(rgbNotifications), FALSE, FILE_NOTIFY_CHANGE_LAST_WRITE, &cbRead, NULL, NULL)) | ||
141 | { | ||
142 | ExitWithLastError(hr, "Failed while watching directory: %ls", sczDirectory); | ||
143 | } | ||
144 | |||
145 | // Wait for half a second to let all the file handles get closed to minimize access | ||
146 | // denied errors. | ||
147 | ::Sleep(500); | ||
148 | |||
149 | FILE_NOTIFY_INFORMATION* pNotification = reinterpret_cast<FILE_NOTIFY_INFORMATION*>(rgbNotifications); | ||
150 | while (pNotification) | ||
151 | { | ||
152 | // If our file was updated, check to see if the modified time really changed. The notifications | ||
153 | // are often trigger happy thinking the file changed two or three times in a row. Maybe it's AV | ||
154 | // software creating the problems but actually checking the modified date works well. | ||
155 | if (CSTR_EQUAL == ::CompareStringW(LOCALE_NEUTRAL, NORM_IGNORECASE, pNotification->FileName, pNotification->FileNameLength / sizeof(WCHAR), wzFileName, -1)) | ||
156 | { | ||
157 | FILETIME ft = { }; | ||
158 | FileGetTime(sczThemePath, NULL, NULL, &ft); | ||
159 | |||
160 | fUpdated = (ftModified.dwHighDateTime < ft.dwHighDateTime) || (ftModified.dwHighDateTime == ft.dwHighDateTime && ftModified.dwLowDateTime < ft.dwLowDateTime); | ||
161 | break; | ||
162 | } | ||
163 | |||
164 | pNotification = pNotification->NextEntryOffset ? reinterpret_cast<FILE_NOTIFY_INFORMATION*>(reinterpret_cast<BYTE*>(pNotification) + pNotification->NextEntryOffset) : NULL; | ||
165 | } | ||
166 | } while (!fUpdated); | ||
167 | } while(fUpdated); | ||
168 | |||
169 | LExit: | ||
170 | if (fComInitialized) | ||
171 | { | ||
172 | ::CoUninitialize(); | ||
173 | } | ||
174 | |||
175 | LocFree(pWixLoc); | ||
176 | ReleaseFileHandle(hDirectory); | ||
177 | ReleaseStr(sczDirectory); | ||
178 | ReleaseStr(sczThemePath); | ||
179 | ReleaseStr(sczWxlPath); | ||
180 | return hr; | ||
181 | } | ||
182 | |||
183 | extern "C" HRESULT AllocHandleTheme( | ||
184 | __in THEME* pTheme, | ||
185 | __out HANDLE_THEME** ppHandle | ||
186 | ) | ||
187 | { | ||
188 | HRESULT hr = S_OK; | ||
189 | HANDLE_THEME* pHandle = NULL; | ||
190 | |||
191 | pHandle = static_cast<HANDLE_THEME*>(MemAlloc(sizeof(HANDLE_THEME), TRUE)); | ||
192 | ExitOnNull(pHandle, hr, E_OUTOFMEMORY, "Failed to allocate theme handle."); | ||
193 | |||
194 | pHandle->cReferences = 1; | ||
195 | pHandle->pTheme = pTheme; | ||
196 | |||
197 | *ppHandle = pHandle; | ||
198 | pHandle = NULL; | ||
199 | |||
200 | LExit: | ||
201 | ReleaseMem(pHandle); | ||
202 | return hr; | ||
203 | } | ||
204 | |||
205 | extern "C" void IncrementHandleTheme( | ||
206 | __in HANDLE_THEME* pHandle | ||
207 | ) | ||
208 | { | ||
209 | ::InterlockedIncrement(reinterpret_cast<LONG*>(&pHandle->cReferences)); | ||
210 | } | ||
211 | |||
212 | extern "C" void DecrementHandleTheme( | ||
213 | __in HANDLE_THEME* pHandle | ||
214 | ) | ||
215 | { | ||
216 | if (pHandle && 0 == ::InterlockedDecrement(reinterpret_cast<LONG*>(&pHandle->cReferences))) | ||
217 | { | ||
218 | ThemeFree(pHandle->pTheme); | ||
219 | MemFree(pHandle); | ||
220 | } | ||
221 | } | ||