mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-03 17:56:44 +02:00
<!-- Enter a brief description/summary of your PR here. What does it fix/what does it change/how was it tested (even manually, if necessary)? --> ## Summary of the Pull Request 1. Introduce WIC for power rename and add new class WICMetadataExtractor to use WIC to extract metadata. 2. Add some patterns for metadata extract. 3. Support XMP and EXIF metadata extract. 4. Add test data for xmp and exif extractor 5. Add attribution for the test data uploader. UI: <img width="2052" height="1415" alt="image" src="https://github.com/user-attachments/assets/9051b12e-4e66-4fdc-a4d4-3bada661c235" /> <img width="284" height="170" alt="image" src="https://github.com/user-attachments/assets/2fd67193-77a7-48f0-a5ac-08a69fe64e55" /> <img width="715" height="1160" alt="image" src="https://github.com/user-attachments/assets/5fa68a8c-d129-44dd-b747-099dfbcded12" /> demo: https://github.com/user-attachments/assets/e90bc206-62e5-4101-ada2-3187ee7e2039 <!-- Please review the items on the PR checklist before submitting--> ## PR Checklist - [x] Closes: #5612 - [x] **Communication:** I've discussed this with core contributors already. If the work hasn't been agreed, this work might be rejected - [x] **Tests:** Added/updated and all pass - [x] **Localization:** All end-user-facing strings can be localized - [ ] **Dev docs:** Added/updated - [ ] **New binaries:** Added on the required places - [ ] [JSON for signing](https://github.com/microsoft/PowerToys/blob/main/.pipelines/ESRPSigning_core.json) for new binaries - [ ] [WXS for installer](https://github.com/microsoft/PowerToys/blob/main/installer/PowerToysSetup/Product.wxs) for new binaries and localization folder - [ ] [YML for CI pipeline](https://github.com/microsoft/PowerToys/blob/main/.pipelines/ci/templates/build-powertoys-steps.yml) for new test projects - [ ] [YML for signed pipeline](https://github.com/microsoft/PowerToys/blob/main/.pipelines/release.yml) - [ ] **Documentation updated:** If checked, please file a pull request on [our docs repo](https://github.com/MicrosoftDocs/windows-uwp/tree/docs/hub/powertoys) and link it here: #xxx <!-- Provide a more detailed description of the PR, other things fixed, or any additional comments/features here --> ## Detailed Description of the Pull Request / Additional comments <!-- Describe how you validated the behavior. Add automated tests wherever possible, but list manual validation steps taken as well --> ## Validation Steps Performed --------- Co-authored-by: Yu Leng <yuleng@microsoft.com>
288 lines
9.8 KiB
C++
288 lines
9.8 KiB
C++
#include "pch.h"
|
|
#include <winrt/base.h>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <optional>
|
|
|
|
#include "Renaming.h"
|
|
#include <Helpers.h>
|
|
#include "MetadataPatternExtractor.h"
|
|
#include "PowerRenameRegEx.h"
|
|
namespace fs = std::filesystem;
|
|
|
|
bool DoRename(CComPtr<IPowerRenameRegEx>& spRenameRegEx, unsigned long& itemEnumIndex, CComPtr<IPowerRenameItem>& spItem)
|
|
{
|
|
bool wouldRename = false;
|
|
DWORD flags = 0;
|
|
winrt::check_hresult(spRenameRegEx->GetFlags(&flags));
|
|
|
|
PWSTR replaceTerm = nullptr;
|
|
bool useFileTime = false;
|
|
bool useMetadata = false;
|
|
|
|
winrt::check_hresult(spRenameRegEx->GetReplaceTerm(&replaceTerm));
|
|
|
|
if (isFileTimeUsed(replaceTerm))
|
|
{
|
|
useFileTime = true;
|
|
}
|
|
|
|
int id = -1;
|
|
winrt::check_hresult(spItem->GetId(&id));
|
|
|
|
bool isFolder = false;
|
|
bool isSubFolderContent = false;
|
|
winrt::check_hresult(spItem->GetIsFolder(&isFolder));
|
|
winrt::check_hresult(spItem->GetIsSubFolderContent(&isSubFolderContent));
|
|
|
|
// Get metadata type to check if metadata patterns are used
|
|
PowerRenameLib::MetadataType metadataType;
|
|
HRESULT hr = spRenameRegEx->GetMetadataType(&metadataType);
|
|
if (FAILED(hr))
|
|
{
|
|
// Fallback to default metadata type if call fails
|
|
metadataType = PowerRenameLib::MetadataType::EXIF;
|
|
}
|
|
|
|
// Check if metadata is used AND if this file type supports metadata
|
|
// Get file path early for metadata type checking and reuse later
|
|
PWSTR filePath = nullptr;
|
|
winrt::check_hresult(spItem->GetPath(&filePath));
|
|
std::wstring filePathStr(filePath); // Copy once for reuse
|
|
CoTaskMemFree(filePath); // Free immediately after copying
|
|
|
|
if (isMetadataUsed(replaceTerm, metadataType, filePathStr.c_str(), isFolder))
|
|
{
|
|
useMetadata = true;
|
|
}
|
|
|
|
CoTaskMemFree(replaceTerm);
|
|
if ((isFolder && (flags & PowerRenameFlags::ExcludeFolders)) ||
|
|
(!isFolder && (flags & PowerRenameFlags::ExcludeFiles)) ||
|
|
(isSubFolderContent && (flags & PowerRenameFlags::ExcludeSubfolders)) ||
|
|
(isFolder && (flags & PowerRenameFlags::ExtensionOnly)))
|
|
{
|
|
// Exclude this item from renaming. Ensure new name is cleared.
|
|
winrt::check_hresult(spItem->PutNewName(nullptr));
|
|
|
|
return wouldRename;
|
|
}
|
|
|
|
PWSTR originalName = nullptr;
|
|
winrt::check_hresult(spItem->GetOriginalName(&originalName));
|
|
|
|
PWSTR currentNewName = nullptr;
|
|
winrt::check_hresult(spItem->GetNewName(¤tNewName));
|
|
|
|
wchar_t sourceName[MAX_PATH] = { 0 };
|
|
|
|
if (isFolder)
|
|
{
|
|
StringCchCopy(sourceName, ARRAYSIZE(sourceName), originalName);
|
|
}
|
|
else
|
|
{
|
|
if (flags & NameOnly)
|
|
{
|
|
StringCchCopy(sourceName, ARRAYSIZE(sourceName), fs::path(originalName).stem().c_str());
|
|
}
|
|
else if (flags & ExtensionOnly)
|
|
{
|
|
std::wstring extension = fs::path(originalName).extension().wstring();
|
|
if (!extension.empty() && extension.front() == '.')
|
|
{
|
|
extension = extension.erase(0, 1);
|
|
}
|
|
StringCchCopy(sourceName, ARRAYSIZE(sourceName), extension.c_str());
|
|
}
|
|
else
|
|
{
|
|
StringCchCopy(sourceName, ARRAYSIZE(sourceName), originalName);
|
|
}
|
|
}
|
|
|
|
SYSTEMTIME fileTime = { 0 };
|
|
|
|
if (useFileTime)
|
|
{
|
|
winrt::check_hresult(spItem->GetTime(flags, &fileTime));
|
|
winrt::check_hresult(spRenameRegEx->PutFileTime(fileTime));
|
|
}
|
|
|
|
if (useMetadata)
|
|
{
|
|
// Extract metadata patterns from the file
|
|
// Note: filePathStr was already obtained and saved earlier for reuse
|
|
|
|
// Get metadata type using the interface method
|
|
PowerRenameLib::MetadataType metadataType;
|
|
HRESULT hr = spRenameRegEx->GetMetadataType(&metadataType);
|
|
if (FAILED(hr))
|
|
{
|
|
// Fallback to default metadata type if call fails
|
|
metadataType = PowerRenameLib::MetadataType::EXIF;
|
|
}
|
|
// Extract all patterns for the selected metadata type
|
|
// At this point we know the file is a supported image format (jpg/jpeg/png/tif/tiff)
|
|
static std::mutex s_metadataMutex; // Mutex to protect static variables
|
|
static std::once_flag s_metadataExtractorInitFlag;
|
|
static std::shared_ptr<PowerRenameLib::MetadataPatternExtractor> s_metadataExtractor;
|
|
static std::optional<PowerRenameLib::MetadataType> s_activeMetadataType;
|
|
|
|
// Initialize the extractor only once
|
|
std::call_once(s_metadataExtractorInitFlag, []() {
|
|
s_metadataExtractor = std::make_shared<PowerRenameLib::MetadataPatternExtractor>();
|
|
});
|
|
|
|
// Protect access to shared state
|
|
{
|
|
std::lock_guard<std::mutex> lock(s_metadataMutex);
|
|
|
|
// Clear cache if metadata type has changed
|
|
if (s_activeMetadataType.has_value() && s_activeMetadataType.value() != metadataType)
|
|
{
|
|
s_metadataExtractor->ClearCache();
|
|
}
|
|
|
|
// Update the active metadata type
|
|
s_activeMetadataType = metadataType;
|
|
}
|
|
|
|
// Extract patterns (this can be done outside the lock if ExtractPatterns is thread-safe)
|
|
PowerRenameLib::MetadataPatternMap patterns = s_metadataExtractor->ExtractPatterns(filePathStr, metadataType);
|
|
|
|
// Always call PutMetadataPatterns to ensure all patterns get replaced
|
|
// Even if empty, this keeps metadata placeholders consistent when no values are extracted
|
|
winrt::check_hresult(spRenameRegEx->PutMetadataPatterns(patterns));
|
|
}
|
|
|
|
PWSTR newName = nullptr;
|
|
|
|
// Failure here means we didn't match anything or had nothing to match
|
|
// Call put_newName with null in that case to reset it
|
|
winrt::check_hresult(spRenameRegEx->Replace(sourceName, &newName, itemEnumIndex));
|
|
|
|
if (useFileTime)
|
|
{
|
|
winrt::check_hresult(spRenameRegEx->ResetFileTime());
|
|
}
|
|
|
|
if (useMetadata)
|
|
{
|
|
winrt::check_hresult(spRenameRegEx->ResetMetadata());
|
|
}
|
|
wchar_t resultName[MAX_PATH] = { 0 };
|
|
|
|
PWSTR newNameToUse = nullptr;
|
|
|
|
// newName == nullptr likely means we have an empty search string. We should leave newNameToUse
|
|
// as nullptr so we clear the renamed column
|
|
// Except string transformation is selected.
|
|
|
|
if (newName == nullptr && (flags & Uppercase || flags & Lowercase || flags & Titlecase || flags & Capitalized))
|
|
{
|
|
SHStrDup(sourceName, &newName);
|
|
}
|
|
|
|
if (newName != nullptr)
|
|
{
|
|
newNameToUse = resultName;
|
|
|
|
if (isFolder)
|
|
{
|
|
StringCchCopy(resultName, ARRAYSIZE(resultName), newName);
|
|
}
|
|
else
|
|
{
|
|
if (flags & NameOnly)
|
|
{
|
|
StringCchPrintf(resultName, ARRAYSIZE(resultName), L"%s%s", newName, fs::path(originalName).extension().c_str());
|
|
}
|
|
else if (flags & ExtensionOnly)
|
|
{
|
|
std::wstring extension = fs::path(originalName).extension().wstring();
|
|
if (!extension.empty())
|
|
{
|
|
StringCchPrintf(resultName, ARRAYSIZE(resultName), L"%s.%s", fs::path(originalName).stem().c_str(), newName);
|
|
}
|
|
else
|
|
{
|
|
StringCchCopy(resultName, ARRAYSIZE(resultName), originalName);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
StringCchCopy(resultName, ARRAYSIZE(resultName), newName);
|
|
}
|
|
}
|
|
}
|
|
|
|
wchar_t trimmedName[MAX_PATH] = { 0 };
|
|
if (newNameToUse != nullptr)
|
|
{
|
|
winrt::check_hresult(GetTrimmedFileName(trimmedName, ARRAYSIZE(trimmedName), newNameToUse));
|
|
newNameToUse = trimmedName;
|
|
}
|
|
|
|
wchar_t transformedName[MAX_PATH] = { 0 };
|
|
if (newNameToUse != nullptr && (flags & Uppercase || flags & Lowercase || flags & Titlecase || flags & Capitalized))
|
|
{
|
|
try
|
|
{
|
|
winrt::check_hresult(GetTransformedFileName(transformedName, ARRAYSIZE(transformedName), newNameToUse, flags, isFolder));
|
|
}
|
|
catch (...)
|
|
{
|
|
}
|
|
newNameToUse = transformedName;
|
|
}
|
|
|
|
// No change from originalName so set newName to
|
|
// null so we clear it from our UI as well.
|
|
if (lstrcmp(originalName, newNameToUse) == 0)
|
|
{
|
|
newNameToUse = nullptr;
|
|
}
|
|
|
|
spItem->PutStatus(PowerRenameItemRenameStatus::ShouldRename);
|
|
if (newNameToUse != nullptr)
|
|
{
|
|
wouldRename = true;
|
|
std::wstring newNameToUseWstr{ newNameToUse };
|
|
PWSTR path = nullptr;
|
|
spItem->GetPath(&path);
|
|
|
|
// Following characters cannot be used for file names.
|
|
// Ref https://learn.microsoft.com/windows/win32/fileio/naming-a-file#naming-conventions
|
|
if (newNameToUseWstr.contains('<') ||
|
|
newNameToUseWstr.contains('>') ||
|
|
newNameToUseWstr.contains(':') ||
|
|
newNameToUseWstr.contains('"') ||
|
|
newNameToUseWstr.contains('\\') ||
|
|
newNameToUseWstr.contains('/') ||
|
|
newNameToUseWstr.contains('|') ||
|
|
newNameToUseWstr.contains('?') ||
|
|
newNameToUseWstr.contains('*'))
|
|
{
|
|
spItem->PutStatus(PowerRenameItemRenameStatus::ItemNameInvalidChar);
|
|
wouldRename = false;
|
|
}
|
|
// Max file path is 260 and max folder path is 247.
|
|
// Ref https://learn.microsoft.com/windows/win32/fileio/maximum-file-path-limitation?tabs=registry
|
|
else if ((isFolder && lstrlen(path) + (lstrlen(newNameToUse) - lstrlen(originalName)) > 247) ||
|
|
lstrlen(path) + (lstrlen(newNameToUse) - lstrlen(originalName)) > 260)
|
|
{
|
|
spItem->PutStatus(PowerRenameItemRenameStatus::ItemNameTooLong);
|
|
wouldRename = false;
|
|
}
|
|
}
|
|
|
|
winrt::check_hresult(spItem->PutNewName(newNameToUse));
|
|
|
|
CoTaskMemFree(newName);
|
|
CoTaskMemFree(currentNewName);
|
|
CoTaskMemFree(originalName);
|
|
|
|
return wouldRename;
|
|
}
|