Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 // | 4 // |
| 5 // This file defines functions that integrate Chrome in Windows shell. These | 5 // This file defines functions that integrate Chrome in Windows shell. These |
| 6 // functions can be used by Chrome as well as Chrome installer. All of the | 6 // functions can be used by Chrome as well as Chrome installer. All of the |
| 7 // work is done by the local functions defined in anonymous namespace in | 7 // work is done by the local functions defined in anonymous namespace in |
| 8 // this class. | 8 // this class. |
| 9 | 9 |
| 10 #include "chrome/installer/util/shell_util.h" | 10 #include "chrome/installer/util/shell_util.h" |
| 11 | 11 |
| 12 #include <windows.h> | 12 #include <windows.h> |
| 13 #include <shlobj.h> | 13 #include <shlobj.h> |
| 14 | 14 |
| 15 #include <limits> | 15 #include <limits> |
| 16 #include <string> | 16 #include <string> |
| 17 | 17 |
| 18 #include "base/bind.h" | |
| 18 #include "base/command_line.h" | 19 #include "base/command_line.h" |
| 19 #include "base/file_util.h" | 20 #include "base/file_util.h" |
| 20 #include "base/files/file_path.h" | 21 #include "base/files/file_path.h" |
| 21 #include "base/lazy_instance.h" | 22 #include "base/lazy_instance.h" |
| 22 #include "base/logging.h" | 23 #include "base/logging.h" |
| 23 #include "base/md5.h" | 24 #include "base/md5.h" |
| 24 #include "base/memory/scoped_ptr.h" | 25 #include "base/memory/scoped_ptr.h" |
| 25 #include "base/memory/scoped_vector.h" | 26 #include "base/memory/scoped_vector.h" |
| 26 #include "base/path_service.h" | 27 #include "base/path_service.h" |
| 27 #include "base/string16.h" | 28 #include "base/string16.h" |
| (...skipping 1124 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1152 const base::win::Version windows_version = base::win::GetVersion(); | 1153 const base::win::Version windows_version = base::win::GetVersion(); |
| 1153 | 1154 |
| 1154 if (windows_version >= base::win::VERSION_WIN8) | 1155 if (windows_version >= base::win::VERSION_WIN8) |
| 1155 return ProbeCurrentDefaultHandlers(protocols, num_protocols); | 1156 return ProbeCurrentDefaultHandlers(protocols, num_protocols); |
| 1156 else if (windows_version >= base::win::VERSION_VISTA) | 1157 else if (windows_version >= base::win::VERSION_VISTA) |
| 1157 return ProbeAppIsDefaultHandlers(protocols, num_protocols); | 1158 return ProbeAppIsDefaultHandlers(protocols, num_protocols); |
| 1158 | 1159 |
| 1159 return ProbeOpenCommandHandlers(protocols, num_protocols); | 1160 return ProbeOpenCommandHandlers(protocols, num_protocols); |
| 1160 } | 1161 } |
| 1161 | 1162 |
| 1162 // Removes shortcut at |shortcut_path| if it is a shortcut that points to | 1163 // (Windows 8+) Finds the folder for app shortcuts, store it in *|path|. |
| 1163 // |target_exe|. If |delete_folder| is true, deletes the parent folder of | 1164 // Returns true on success. |
| 1164 // the shortcut completely. Returns true if either the shortcut was deleted | 1165 bool GetAppShortcutsFolder(BrowserDistribution* dist, |
| 1165 // successfully or if the shortcut did not point to |target_exe|. | 1166 bool is_user_level, |
|
grt (UTC plus 2)
2013/04/30 14:38:07
indentation
grt (UTC plus 2)
2013/04/30 14:38:07
bool -> ShellChange for consistency
huangs
2013/04/30 15:58:38
Done.
huangs
2013/04/30 15:58:38
Done.
| |
| 1166 bool MaybeRemoveShortcutAtPath(const base::FilePath& shortcut_path, | 1167 base::FilePath *path) { |
| 1167 const base::FilePath& target_exe, | 1168 DCHECK(path); |
| 1168 bool delete_folder) { | 1169 if (base::win::GetVersion() < base::win::VERSION_WIN8) |
| 1169 base::FilePath target_path; | |
| 1170 if (!base::win::ResolveShortcut(shortcut_path, &target_path, NULL)) | |
| 1171 return false; | 1170 return false; |
| 1172 | 1171 |
| 1173 if (InstallUtil::ProgramCompare(target_exe).EvaluatePath(target_path)) { | 1172 base::FilePath folder; |
| 1174 // Unpin the shortcut if it was ever pinned by the user or the installer. | 1173 if (!PathService::Get(base::DIR_APP_SHORTCUTS, &folder)) { |
| 1175 VLOG(1) << "Trying to unpin " << shortcut_path.value(); | 1174 LOG(ERROR) << "Could not get application shortcuts location."; |
| 1176 if (!base::win::TaskbarUnpinShortcutLink(shortcut_path.value().c_str())) { | 1175 return false; |
| 1177 VLOG(1) << shortcut_path.value() | |
| 1178 << " wasn't pinned (or the unpin failed)."; | |
| 1179 } | |
| 1180 if (delete_folder) | |
| 1181 return file_util::Delete(shortcut_path.DirName(), true); | |
| 1182 else | |
| 1183 return file_util::Delete(shortcut_path, false); | |
| 1184 } | 1176 } |
| 1185 | 1177 |
| 1186 // The shortcut at |shortcut_path| doesn't point to |target_exe|, act as if | 1178 folder = folder.Append(ShellUtil::GetBrowserModelId(dist, is_user_level)); |
| 1187 // our shortcut had been deleted. | 1179 if (!file_util::DirectoryExists(folder)) { |
| 1180 VLOG(1) << "No start screen shortcuts."; | |
| 1181 return false; | |
| 1182 } | |
| 1183 | |
| 1184 *path = folder; | |
| 1188 return true; | 1185 return true; |
| 1189 } | 1186 } |
| 1190 | 1187 |
| 1188 const wchar_t kAllShortcuts[] = L"*.lnk"; | |
| 1189 | |
| 1190 typedef base::Callback<bool(const base::FilePath&)> FileOperationCallback; | |
| 1191 | |
| 1192 FileOperationCallback kNullFileOperation; | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
remove this (globals are banned).
huangs
2013/04/30 15:58:38
Done. Replacing uses with FileOperationCallback().
| |
| 1193 | |
| 1194 // Shortcut operations for BatchShortcutAction(). | |
| 1195 | |
| 1196 bool ShortcutOpUnpin(const base::FilePath& shortcut_path) { | |
| 1197 VLOG(1) << "Trying to unpin " << shortcut_path.value(); | |
| 1198 if (!base::win::TaskbarUnpinShortcutLink(shortcut_path.value().c_str())) { | |
| 1199 VLOG(1) << shortcut_path.value() << " wasn't pinned (or the unpin failed)."; | |
| 1200 // No error, since shortcut might not be pinned. | |
| 1201 } | |
| 1202 return true; | |
| 1203 } | |
| 1204 | |
| 1205 bool ShortcutOpDelete(const base::FilePath& shortcut_path) { | |
| 1206 return file_util::Delete(shortcut_path, false); | |
| 1207 } | |
| 1208 | |
| 1209 bool ShortcutOpUnpinAndDelete(const base::FilePath& shortcut_path) { | |
| 1210 // Want to try both, without short-circuiting. | |
| 1211 bool ret1 = ShortcutOpUnpin(shortcut_path); | |
| 1212 bool ret2 = ShortcutOpDelete(shortcut_path); | |
| 1213 return ret1 && ret2; | |
| 1214 } | |
| 1215 | |
| 1216 bool ShortcutOpSetTarget(const base::FilePath& target_path, | |
| 1217 const base::FilePath& shortcut_path) { | |
| 1218 base::win::ShortcutProperties shortcut_properties; | |
| 1219 shortcut_properties.set_target(target_path); | |
| 1220 shortcut_properties.set_working_dir(target_path.DirName()); | |
| 1221 return base::win::CreateOrUpdateShortcutLink( | |
| 1222 shortcut_path, shortcut_properties, base::win::SHORTCUT_REPLACE_EXISTING); | |
| 1223 } | |
| 1224 | |
| 1225 // Folder operations for BatchShortcutAction(). | |
| 1226 | |
| 1227 bool FolderOpDelete(const base::FilePath& folder_path) { | |
| 1228 return file_util::Delete(folder_path, true); | |
| 1229 } | |
| 1230 | |
| 1231 // {|location|, |dist|, |level|} determine |shortcut_folder| to operate in. | |
| 1232 // In |shortcut_folder|, iterate over shortcuts with names matching | |
| 1233 // |name_filter| and targets pointing to |target_exe|. | |
| 1234 // Applies |shortcut_operation| on each matching shortcuts and then applies | |
| 1235 // |folder_operation| on |shortcut_folder|. | |
| 1236 // If |name_filter| is empty, then ignores |shortcut_operation|; otherwise | |
| 1237 // iterate with implicitly ".lnk" suffix. |kAllShortcuts| is used to specify | |
| 1238 // all shortcuts. | |
| 1239 // Returns true iff all operations are successful. | |
| 1240 bool BatchShortcutAction(const FileOperationCallback& shortcut_operation, | |
| 1241 const FileOperationCallback& folder_operation, | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
folder_operation seems overly abstract. it's only
huangs
2013/04/30 15:58:38
Decided to extract RemoveShortcutFolder(). Before
| |
| 1242 ShellUtil::ShortcutLocation location, | |
| 1243 BrowserDistribution* dist, | |
| 1244 ShellUtil::ShellChange level, | |
| 1245 const base::FilePath& target_exe, | |
| 1246 const string16& name_filter) { | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
simplify by removing name_filter and kAllShortcuts
huangs
2013/04/30 15:58:38
Done.
| |
| 1247 base::FilePath shortcut_folder; | |
| 1248 if (!ShellUtil::GetShortcutPath(location, dist, level, &shortcut_folder) || | |
| 1249 !file_util::PathExists(shortcut_folder)) { | |
|
gab
2013/04/30 19:20:33
Why did you remove this PathExists() check? This i
| |
| 1250 LOG(WARNING) << "Cannot find path at location " << location; | |
| 1251 return false; | |
| 1252 } | |
| 1253 | |
| 1254 // Operate on specified shortcuts in the folder. | |
| 1255 InstallUtil::ProgramCompare target_compare(target_exe); | |
| 1256 bool success = true; | |
| 1257 if (!shortcut_operation.is_null() && !name_filter.empty()) { | |
| 1258 // Ensure ".lnk" extension. | |
| 1259 string16 pattern = name_filter; | |
| 1260 if (!EndsWith (pattern, installer::kLnkExt, false)) | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
remove space before open paren
huangs
2013/04/30 15:58:38
Moot; deleted code.
| |
| 1261 pattern.append(installer::kLnkExt); | |
| 1262 file_util::FileEnumerator enumerator( | |
| 1263 shortcut_folder, false, file_util::FileEnumerator::FILES, pattern); | |
| 1264 for (base::FilePath shortcut_path = enumerator.Next(); | |
| 1265 !shortcut_path.empty(); | |
| 1266 shortcut_path = enumerator.Next()) { | |
| 1267 base::FilePath target_path; | |
| 1268 if (base::win::ResolveShortcut(shortcut_path, &target_path, NULL)) { | |
| 1269 if (target_compare.EvaluatePath(target_path) && | |
| 1270 !shortcut_operation.Run(shortcut_path)) { | |
| 1271 LOG(ERROR) << "Failed shortcut operation on " | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
suggestion: move the logging into the operations s
huangs
2013/04/30 15:58:38
Done.
| |
| 1272 << shortcut_path.value(); | |
| 1273 success = false; | |
| 1274 } | |
| 1275 } else { | |
| 1276 LOG(ERROR) << "Cannot resolve shortcut at " << shortcut_path.value(); | |
| 1277 success = false; | |
| 1278 } | |
| 1279 } | |
| 1280 } | |
| 1281 | |
| 1282 // Operate on shortcut folder itself. | |
| 1283 if (!folder_operation.is_null() && !folder_operation.Run(shortcut_folder)) { | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
document that folder_operation (or delete_folder b
huangs
2013/04/30 15:58:38
Moved to RemoveShortcutFolder(). Right thing to do
| |
| 1284 LOG(ERROR) << "Failed folder operation on " << shortcut_folder.value(); | |
| 1285 success = false; | |
| 1286 } | |
| 1287 return success; | |
| 1288 } | |
| 1289 | |
| 1191 } // namespace | 1290 } // namespace |
| 1192 | 1291 |
| 1193 const wchar_t* ShellUtil::kRegDefaultIcon = L"\\DefaultIcon"; | 1292 const wchar_t* ShellUtil::kRegDefaultIcon = L"\\DefaultIcon"; |
| 1194 const wchar_t* ShellUtil::kRegShellPath = L"\\shell"; | 1293 const wchar_t* ShellUtil::kRegShellPath = L"\\shell"; |
| 1195 const wchar_t* ShellUtil::kRegShellOpen = L"\\shell\\open\\command"; | 1294 const wchar_t* ShellUtil::kRegShellOpen = L"\\shell\\open\\command"; |
| 1196 const wchar_t* ShellUtil::kRegStartMenuInternet = | 1295 const wchar_t* ShellUtil::kRegStartMenuInternet = |
| 1197 L"Software\\Clients\\StartMenuInternet"; | 1296 L"Software\\Clients\\StartMenuInternet"; |
| 1198 const wchar_t* ShellUtil::kRegClasses = L"Software\\Classes"; | 1297 const wchar_t* ShellUtil::kRegClasses = L"Software\\Classes"; |
| 1199 const wchar_t* ShellUtil::kRegRegisteredApplications = | 1298 const wchar_t* ShellUtil::kRegRegisteredApplications = |
| 1200 L"Software\\RegisteredApplications"; | 1299 L"Software\\RegisteredApplications"; |
| (...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1247 const string16& chrome_exe, | 1346 const string16& chrome_exe, |
| 1248 const string16& suffix) { | 1347 const string16& suffix) { |
| 1249 return QuickIsChromeRegistered(dist, chrome_exe, suffix, | 1348 return QuickIsChromeRegistered(dist, chrome_exe, suffix, |
| 1250 CONFIRM_SHELL_REGISTRATION_IN_HKLM); | 1349 CONFIRM_SHELL_REGISTRATION_IN_HKLM); |
| 1251 } | 1350 } |
| 1252 | 1351 |
| 1253 bool ShellUtil::GetShortcutPath(ShellUtil::ShortcutLocation location, | 1352 bool ShellUtil::GetShortcutPath(ShellUtil::ShortcutLocation location, |
| 1254 BrowserDistribution* dist, | 1353 BrowserDistribution* dist, |
| 1255 ShellChange level, | 1354 ShellChange level, |
| 1256 base::FilePath* path) { | 1355 base::FilePath* path) { |
| 1356 DCHECK(path); | |
| 1257 int dir_key = -1; | 1357 int dir_key = -1; |
| 1258 bool add_folder_for_dist = false; | 1358 bool add_folder_for_dist = false; |
| 1259 switch (location) { | 1359 switch (location) { |
| 1260 case SHORTCUT_LOCATION_DESKTOP: | 1360 case SHORTCUT_LOCATION_DESKTOP: |
| 1261 dir_key = (level == CURRENT_USER) ? base::DIR_USER_DESKTOP : | 1361 dir_key = (level == CURRENT_USER) ? base::DIR_USER_DESKTOP : |
| 1262 base::DIR_COMMON_DESKTOP; | 1362 base::DIR_COMMON_DESKTOP; |
| 1263 break; | 1363 break; |
| 1264 case SHORTCUT_LOCATION_QUICK_LAUNCH: | 1364 case SHORTCUT_LOCATION_QUICK_LAUNCH: |
| 1265 dir_key = (level == CURRENT_USER) ? base::DIR_USER_QUICK_LAUNCH : | 1365 dir_key = (level == CURRENT_USER) ? base::DIR_USER_QUICK_LAUNCH : |
| 1266 base::DIR_DEFAULT_USER_QUICK_LAUNCH; | 1366 base::DIR_DEFAULT_USER_QUICK_LAUNCH; |
| 1267 break; | 1367 break; |
| 1268 case SHORTCUT_LOCATION_START_MENU: | 1368 case SHORTCUT_LOCATION_START_MENU: |
| 1269 dir_key = (level == CURRENT_USER) ? base::DIR_START_MENU : | 1369 dir_key = (level == CURRENT_USER) ? base::DIR_START_MENU : |
| 1270 base::DIR_COMMON_START_MENU; | 1370 base::DIR_COMMON_START_MENU; |
| 1271 add_folder_for_dist = true; | 1371 add_folder_for_dist = true; |
| 1272 break; | 1372 break; |
| 1373 case SHORTCUT_LOCATION_TASKBAR_PINS: | |
| 1374 if (base::win::GetVersion() < base::win::VERSION_WIN7) | |
| 1375 return false; | |
| 1376 | |
| 1377 dir_key = base::DIR_TASKBAR_PINS; | |
| 1378 break; | |
| 1379 case SHORTCUT_LOCATION_APP_SHORTCUTS: | |
| 1380 return GetAppShortcutsFolder(dist, level == CURRENT_USER, path); | |
| 1381 | |
| 1273 default: | 1382 default: |
| 1274 NOTREACHED(); | 1383 NOTREACHED(); |
| 1275 return false; | 1384 return false; |
| 1276 } | 1385 } |
| 1277 | 1386 |
| 1278 if (!PathService::Get(dir_key, path) || path->empty()) { | 1387 if (!PathService::Get(dir_key, path) || path->empty()) { |
| 1279 NOTREACHED() << dir_key; | 1388 NOTREACHED() << dir_key; |
| 1280 return false; | 1389 return false; |
| 1281 } | 1390 } |
| 1282 | 1391 |
| 1283 if (add_folder_for_dist) | 1392 if (add_folder_for_dist) |
| 1284 *path = path->Append(dist->GetAppShortCutName()); | 1393 *path = path->Append(dist->GetAppShortCutName()); |
| 1285 | 1394 |
| 1286 return true; | 1395 return true; |
| 1287 } | 1396 } |
| 1288 | 1397 |
| 1289 bool ShellUtil::CreateOrUpdateShortcut( | 1398 bool ShellUtil::CreateOrUpdateShortcut( |
| 1290 ShellUtil::ShortcutLocation location, | 1399 ShellUtil::ShortcutLocation location, |
| 1291 BrowserDistribution* dist, | 1400 BrowserDistribution* dist, |
| 1292 const ShellUtil::ShortcutProperties& properties, | 1401 const ShellUtil::ShortcutProperties& properties, |
| 1293 ShellUtil::ShortcutOperation operation) { | 1402 ShellUtil::ShortcutOperation operation) { |
| 1294 DCHECK(dist); | 1403 DCHECK(dist); |
| 1295 // |pin_to_taskbar| is only acknowledged when first creating the shortcut. | 1404 // |pin_to_taskbar| is only acknowledged when first creating the shortcut. |
| 1296 DCHECK(!properties.pin_to_taskbar || | 1405 DCHECK(!properties.pin_to_taskbar || |
| 1297 operation == SHELL_SHORTCUT_CREATE_ALWAYS || | 1406 operation == SHELL_SHORTCUT_CREATE_ALWAYS || |
| 1298 operation == SHELL_SHORTCUT_CREATE_IF_NO_SYSTEM_LEVEL); | 1407 operation == SHELL_SHORTCUT_CREATE_IF_NO_SYSTEM_LEVEL); |
| 1299 | 1408 |
| 1300 base::FilePath user_shortcut_path; | 1409 base::FilePath user_shortcut_path; |
| 1301 base::FilePath system_shortcut_path; | 1410 base::FilePath system_shortcut_path; |
| 1302 if (!GetShortcutPath(location, dist, SYSTEM_LEVEL, &system_shortcut_path) || | 1411 if (!GetShortcutPath(location, dist, SYSTEM_LEVEL, &system_shortcut_path)) { |
| 1303 system_shortcut_path.empty()) { | |
|
grt (UTC plus 2)
2013/04/30 14:38:07
why remove this? it seems like a reasonable safegu
huangs
2013/04/30 15:58:38
GetShortcutPath() already checks for this.
Remove
| |
| 1304 NOTREACHED(); | 1412 NOTREACHED(); |
| 1305 return false; | 1413 return false; |
| 1306 } | 1414 } |
| 1307 | 1415 |
| 1308 string16 shortcut_name(ExtractShortcutNameFromProperties(dist, properties)); | 1416 string16 shortcut_name(ExtractShortcutNameFromProperties(dist, properties)); |
| 1309 system_shortcut_path = system_shortcut_path.Append(shortcut_name); | 1417 system_shortcut_path = system_shortcut_path.Append(shortcut_name); |
| 1310 | 1418 |
| 1311 base::FilePath* chosen_path; | 1419 base::FilePath* chosen_path; |
| 1312 bool should_install_shortcut = true; | 1420 bool should_install_shortcut = true; |
| 1313 if (properties.level == SYSTEM_LEVEL) { | 1421 if (properties.level == SYSTEM_LEVEL) { |
| 1314 // Install the system-level shortcut if requested. | 1422 // Install the system-level shortcut if requested. |
| 1315 chosen_path = &system_shortcut_path; | 1423 chosen_path = &system_shortcut_path; |
| 1316 } else if (operation != SHELL_SHORTCUT_CREATE_IF_NO_SYSTEM_LEVEL || | 1424 } else if (operation != SHELL_SHORTCUT_CREATE_IF_NO_SYSTEM_LEVEL || |
| 1317 !file_util::PathExists(system_shortcut_path)) { | 1425 !file_util::PathExists(system_shortcut_path)) { |
| 1318 // Otherwise install the user-level shortcut, unless the system-level | 1426 // Otherwise install the user-level shortcut, unless the system-level |
| 1319 // variant of this shortcut is present on the machine and |operation| states | 1427 // variant of this shortcut is present on the machine and |operation| states |
| 1320 // not to create a user-level shortcut in that case. | 1428 // not to create a user-level shortcut in that case. |
| 1321 if (!GetShortcutPath(location, dist, CURRENT_USER, &user_shortcut_path) || | 1429 if (!GetShortcutPath(location, dist, CURRENT_USER, &user_shortcut_path)) { |
| 1322 user_shortcut_path.empty()) { | |
| 1323 NOTREACHED(); | 1430 NOTREACHED(); |
| 1324 return false; | 1431 return false; |
| 1325 } | 1432 } |
| 1326 user_shortcut_path = user_shortcut_path.Append(shortcut_name); | 1433 user_shortcut_path = user_shortcut_path.Append(shortcut_name); |
| 1327 chosen_path = &user_shortcut_path; | 1434 chosen_path = &user_shortcut_path; |
| 1328 } else { | 1435 } else { |
| 1329 // Do not install any shortcut if we are told to install a user-level | 1436 // Do not install any shortcut if we are told to install a user-level |
| 1330 // shortcut, but the system-level variant of that shortcut is present. | 1437 // shortcut, but the system-level variant of that shortcut is present. |
| 1331 // Other actions (e.g., pinning) can still happen with respect to the | 1438 // Other actions (e.g., pinning) can still happen with respect to the |
| 1332 // existing system-level shortcut however. | 1439 // existing system-level shortcut however. |
| (...skipping 518 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1851 } else if (elevate_if_not_admin && | 1958 } else if (elevate_if_not_admin && |
| 1852 base::win::GetVersion() >= base::win::VERSION_VISTA) { | 1959 base::win::GetVersion() >= base::win::VERSION_VISTA) { |
| 1853 // Elevate to do the whole job | 1960 // Elevate to do the whole job |
| 1854 return ElevateAndRegisterChrome(dist, chrome_exe, suffix, protocol); | 1961 return ElevateAndRegisterChrome(dist, chrome_exe, suffix, protocol); |
| 1855 } else { | 1962 } else { |
| 1856 // Admin rights are required to register capabilities before Windows 8. | 1963 // Admin rights are required to register capabilities before Windows 8. |
| 1857 return false; | 1964 return false; |
| 1858 } | 1965 } |
| 1859 } | 1966 } |
| 1860 | 1967 |
| 1968 // static | |
| 1861 bool ShellUtil::RemoveShortcut(ShellUtil::ShortcutLocation location, | 1969 bool ShellUtil::RemoveShortcut(ShellUtil::ShortcutLocation location, |
| 1862 BrowserDistribution* dist, | 1970 BrowserDistribution* dist, |
| 1863 const base::FilePath& target_exe, | |
| 1864 ShellChange level, | 1971 ShellChange level, |
| 1865 const string16* shortcut_name) { | 1972 const base::FilePath& target_exe) { |
| 1866 const bool delete_folder = (location == SHORTCUT_LOCATION_START_MENU); | 1973 bool delete_file = (location != SHORTCUT_LOCATION_TASKBAR_PINS); |
| 1867 | 1974 bool delete_folder = (location == SHORTCUT_LOCATION_START_MENU); |
| 1868 base::FilePath shortcut_folder; | 1975 return BatchShortcutAction( |
| 1869 if (!GetShortcutPath(location, dist, level, &shortcut_folder) || | 1976 base::Bind(delete_file ? &ShortcutOpUnpinAndDelete : &ShortcutOpUnpin), |
| 1870 shortcut_folder.empty()) { | 1977 delete_folder ? base::Bind(&FolderOpDelete) : kNullFileOperation, |
| 1871 NOTREACHED(); | 1978 location, dist, level, target_exe, kAllShortcuts); |
| 1872 return false; | |
| 1873 } | |
| 1874 | |
| 1875 if (!delete_folder && !shortcut_name) { | |
| 1876 file_util::FileEnumerator enumerator(shortcut_folder, false, | |
| 1877 file_util::FileEnumerator::FILES); | |
| 1878 bool had_failures = false; | |
| 1879 for (base::FilePath path = enumerator.Next(); !path.empty(); | |
| 1880 path = enumerator.Next()) { | |
| 1881 if (path.Extension() != installer::kLnkExt) | |
| 1882 continue; | |
| 1883 | |
| 1884 if (!MaybeRemoveShortcutAtPath(path, target_exe, delete_folder)) | |
| 1885 had_failures = true; | |
| 1886 } | |
| 1887 return !had_failures; | |
| 1888 } | |
| 1889 | |
| 1890 const string16 shortcut_base_name( | |
| 1891 (shortcut_name ? *shortcut_name : dist->GetAppShortCutName()) + | |
| 1892 installer::kLnkExt); | |
| 1893 const base::FilePath shortcut_path( | |
| 1894 shortcut_folder.Append(shortcut_base_name)); | |
| 1895 if (!file_util::PathExists(shortcut_path)) | |
| 1896 return true; | |
| 1897 | |
| 1898 return MaybeRemoveShortcutAtPath(shortcut_path, target_exe, delete_folder); | |
| 1899 } | 1979 } |
| 1900 | 1980 |
| 1901 void ShellUtil::RemoveTaskbarShortcuts(const string16& target_exe) { | 1981 // static |
| 1902 if (base::win::GetVersion() < base::win::VERSION_WIN7) | 1982 void ShellUtil::RemoveStartScreenShortcuts(BrowserDistribution* dist, |
| 1983 const base::FilePath& target_exe) { | |
| 1984 if (base::win::GetVersion() < base::win::VERSION_WIN8) | |
| 1903 return; | 1985 return; |
| 1904 | 1986 |
| 1905 base::FilePath taskbar_pins_path; | 1987 ShortcutLocation location = SHORTCUT_LOCATION_APP_SHORTCUTS; |
|
grt (UTC plus 2)
2013/04/30 14:38:07
nit: remove this local var and put the constant in
huangs
2013/04/30 15:58:38
Done.
| |
| 1906 if (!PathService::Get(base::DIR_TASKBAR_PINS, &taskbar_pins_path) || | 1988 ShellChange level = |
| 1907 !file_util::PathExists(taskbar_pins_path)) { | 1989 InstallUtil::IsPerUserInstall(target_exe.value().c_str()) ? |
| 1908 LOG(ERROR) << "Couldn't find path to taskbar pins."; | 1990 ShellUtil::CURRENT_USER : ShellUtil::SYSTEM_LEVEL; |
| 1909 return; | 1991 VLOG(1) << "Removing start screen shortcuts."; |
| 1910 } | 1992 if (!BatchShortcutAction(kNullFileOperation, |
| 1911 | 1993 base::Bind(&FolderOpDelete), |
| 1912 file_util::FileEnumerator shortcuts_enum( | 1994 location, dist, level, base::FilePath(target_exe), |
|
grt (UTC plus 2)
2013/04/30 14:38:07
base::FilePath(target_exe) -> target_exe
huangs
2013/04/30 15:58:38
Ah, right. But moot: Removed code.
| |
| 1913 taskbar_pins_path, false, | 1995 L"")) { |
|
grt (UTC plus 2)
2013/04/30 14:38:07
L"" -> string16()
huangs
2013/04/30 15:58:38
Moot; removed code.
| |
| 1914 file_util::FileEnumerator::FILES, FILE_PATH_LITERAL("*.lnk")); | 1996 LOG(ERROR) << "Failed to remove start screen shortcuts."; |
| 1915 | |
| 1916 base::FilePath target_path(target_exe); | |
| 1917 InstallUtil::ProgramCompare target_compare(target_path); | |
| 1918 for (base::FilePath shortcut_path = shortcuts_enum.Next(); | |
| 1919 !shortcut_path.empty(); | |
| 1920 shortcut_path = shortcuts_enum.Next()) { | |
| 1921 base::FilePath read_target; | |
| 1922 if (!base::win::ResolveShortcut(shortcut_path, &read_target, NULL)) { | |
| 1923 LOG(ERROR) << "Couldn't resolve shortcut at " << shortcut_path.value(); | |
| 1924 continue; | |
| 1925 } | |
| 1926 if (target_compare.EvaluatePath(read_target)) { | |
| 1927 // Unpin this shortcut if it points to |target_exe|. | |
| 1928 base::win::TaskbarUnpinShortcutLink(shortcut_path.value().c_str()); | |
| 1929 } | |
| 1930 } | 1997 } |
| 1931 } | 1998 } |
| 1932 | 1999 |
| 1933 void ShellUtil::RemoveStartScreenShortcuts(BrowserDistribution* dist, | 2000 // static |
| 1934 const string16& target_exe) { | 2001 bool ShellUtil::MigrateShortcut(ShellUtil::ShortcutLocation location, |
| 2002 BrowserDistribution* dist, | |
| 2003 ShellChange level, | |
| 2004 const base::FilePath& old_target_exe, | |
| 2005 const base::FilePath& new_target_exe) { | |
| 2006 return BatchShortcutAction(base::Bind(&ShortcutOpSetTarget, new_target_exe), | |
| 2007 kNullFileOperation, location, dist, level, | |
| 2008 old_target_exe, kAllShortcuts); | |
| 2009 } | |
| 2010 | |
| 2011 // static | |
| 2012 void ShellUtil::MigrateStartScreenShortcuts( | |
| 2013 BrowserDistribution* dist, | |
| 2014 const base::FilePath& old_target_exe, | |
| 2015 const base::FilePath& new_target_exe) { | |
| 1935 if (base::win::GetVersion() < base::win::VERSION_WIN8) | 2016 if (base::win::GetVersion() < base::win::VERSION_WIN8) |
| 1936 return; | 2017 return; |
| 1937 | 2018 |
| 1938 base::FilePath app_shortcuts_path; | 2019 ShortcutLocation location = SHORTCUT_LOCATION_APP_SHORTCUTS; |
| 1939 if (!PathService::Get(base::DIR_APP_SHORTCUTS, &app_shortcuts_path)) { | 2020 ShellChange level = |
| 1940 LOG(ERROR) << "Could not get application shortcuts location to delete" | 2021 InstallUtil::IsPerUserInstall(old_target_exe.value().c_str()) ? |
| 1941 << " start screen shortcuts."; | 2022 ShellUtil::CURRENT_USER : ShellUtil::SYSTEM_LEVEL; |
| 1942 return; | 2023 VLOG(1) << "Migrating start screen shortcuts."; |
| 2024 if (!BatchShortcutAction(base::Bind(&ShortcutOpSetTarget, new_target_exe), | |
| 2025 kNullFileOperation, location, dist, level, | |
| 2026 old_target_exe, kAllShortcuts)) { | |
| 2027 LOG(ERROR) << "Failed to migrate start screen shortcuts."; | |
| 1943 } | 2028 } |
| 1944 | 2029 |
| 1945 app_shortcuts_path = app_shortcuts_path.Append( | |
| 1946 GetBrowserModelId(dist, | |
| 1947 InstallUtil::IsPerUserInstall(target_exe.c_str()))); | |
| 1948 if (!file_util::DirectoryExists(app_shortcuts_path)) { | |
| 1949 VLOG(1) << "No start screen shortcuts to delete."; | |
| 1950 return; | |
| 1951 } | |
| 1952 | |
| 1953 VLOG(1) << "Removing start screen shortcuts from " | |
| 1954 << app_shortcuts_path.value(); | |
| 1955 if (!file_util::Delete(app_shortcuts_path, true)) { | |
| 1956 LOG(ERROR) << "Failed to remove start screen shortcuts from " | |
| 1957 << app_shortcuts_path.value(); | |
| 1958 } | |
| 1959 } | 2030 } |
| 1960 | 2031 |
| 1961 bool ShellUtil::GetUserSpecificRegistrySuffix(string16* suffix) { | 2032 bool ShellUtil::GetUserSpecificRegistrySuffix(string16* suffix) { |
| 1962 // Use a thread-safe cache for the user's suffix. | 2033 // Use a thread-safe cache for the user's suffix. |
| 1963 static base::LazyInstance<UserSpecificRegistrySuffix>::Leaky suffix_instance = | 2034 static base::LazyInstance<UserSpecificRegistrySuffix>::Leaky suffix_instance = |
| 1964 LAZY_INSTANCE_INITIALIZER; | 2035 LAZY_INSTANCE_INITIALIZER; |
| 1965 return suffix_instance.Get().GetSuffix(suffix); | 2036 return suffix_instance.Get().GetSuffix(suffix); |
| 1966 } | 2037 } |
| 1967 | 2038 |
| 1968 bool ShellUtil::GetOldUserSpecificRegistrySuffix(string16* suffix) { | 2039 bool ShellUtil::GetOldUserSpecificRegistrySuffix(string16* suffix) { |
| (...skipping 49 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2018 // are any left...). | 2089 // are any left...). |
| 2019 if (free_bits >= 8 && next_byte_index < size) { | 2090 if (free_bits >= 8 && next_byte_index < size) { |
| 2020 free_bits -= 8; | 2091 free_bits -= 8; |
| 2021 bit_stream += bytes[next_byte_index++] << free_bits; | 2092 bit_stream += bytes[next_byte_index++] << free_bits; |
| 2022 } | 2093 } |
| 2023 } | 2094 } |
| 2024 | 2095 |
| 2025 DCHECK_EQ(ret.length(), encoded_length); | 2096 DCHECK_EQ(ret.length(), encoded_length); |
| 2026 return ret; | 2097 return ret; |
| 2027 } | 2098 } |
| OLD | NEW |