#include "stdafx.h" #include "PowerRenameExt.h" #include #include #include #include #include #include #include "resource.h" extern HINSTANCE g_hInst; struct InvokeStruct { HWND hwndParent; IStream* pstrm; }; CPowerRenameMenu::CPowerRenameMenu() { DllAddRef(); } CPowerRenameMenu::~CPowerRenameMenu() { m_spdo = nullptr; DllRelease(); } HRESULT CPowerRenameMenu::s_CreateInstance(_In_opt_ IUnknown*, _In_ REFIID riid, _Outptr_ void **ppv) { *ppv = nullptr; HRESULT hr = E_OUTOFMEMORY; CPowerRenameMenu *pprm = new CPowerRenameMenu(); if (pprm) { hr = pprm->QueryInterface(riid, ppv); pprm->Release(); } return hr; } // IShellExtInit HRESULT CPowerRenameMenu::Initialize(_In_opt_ PCIDLIST_ABSOLUTE, _In_ IDataObject *pdtobj, HKEY) { // Check if we have disabled ourselves if (!CSettings::GetEnabled()) return E_FAIL; // Cache the data object to be used later m_spdo = pdtobj; return S_OK; } // IContextMenu HRESULT CPowerRenameMenu::QueryContextMenu(HMENU hMenu, UINT index, UINT uIDFirst, UINT, UINT uFlags) { // Check if we have disabled ourselves if (!CSettings::GetEnabled()) return E_FAIL; // Check if we should only be on the extended context menu if (CSettings::GetExtendedContextMenuOnly() && (!(uFlags & CMF_EXTENDEDVERBS))) return E_FAIL; HRESULT hr = E_UNEXPECTED; if (m_spdo && !(uFlags & (CMF_DEFAULTONLY | CMF_VERBSONLY | CMF_OPTIMIZEFORINVOKE))) { wchar_t menuName[64] = { 0 }; LoadString(g_hInst, IDS_POWERRENAME, menuName, ARRAYSIZE(menuName)); MENUITEMINFO mii; mii.cbSize = sizeof(MENUITEMINFO); mii.fMask = MIIM_STRING | MIIM_FTYPE | MIIM_ID | MIIM_STATE; mii.wID = uIDFirst++; mii.fType = MFT_STRING; mii.dwTypeData = (PWSTR)menuName; mii.fState = MFS_ENABLED; if (CSettings::GetShowIconOnMenu()) { HICON hIcon = (HICON)LoadImage(g_hInst, MAKEINTRESOURCE(IDI_RENAME), IMAGE_ICON, 16, 16, 0); if (hIcon) { mii.fMask |= MIIM_BITMAP; mii.hbmpItem = CreateBitmapFromIcon(hIcon); DestroyIcon(hIcon); } } if (!InsertMenuItem(hMenu, index, TRUE, &mii)) { hr = HRESULT_FROM_WIN32(GetLastError()); } else { hr = MAKE_HRESULT(SEVERITY_SUCCESS, FACILITY_NULL, 1); } } return hr; } HRESULT CPowerRenameMenu::InvokeCommand(_In_ LPCMINVOKECOMMANDINFO pici) { HRESULT hr = E_FAIL; if (CSettings::GetEnabled() && (IS_INTRESOURCE(pici->lpVerb)) && (LOWORD(pici->lpVerb) == 0)) { Trace::Invoked(); InvokeStruct* pInvokeData = new InvokeStruct; hr = pInvokeData ? S_OK : E_OUTOFMEMORY; if (SUCCEEDED(hr)) { pInvokeData->hwndParent = pici->hwnd; hr = CoMarshalInterThreadInterfaceInStream(__uuidof(m_spdo), m_spdo, &(pInvokeData->pstrm)); if (SUCCEEDED(hr)) { hr = SHCreateThread(s_PowerRenameUIThreadProc, pInvokeData, CTF_COINIT | CTF_PROCESS_REF, nullptr) ? S_OK : E_FAIL; if (FAILED(hr)) { pInvokeData->pstrm->Release(); // if we failed to create the thread, then we must release the stream } } if (FAILED(hr)) { delete pInvokeData; } } Trace::InvokedRet(hr); } return hr; } DWORD WINAPI CPowerRenameMenu::s_PowerRenameUIThreadProc(_In_ void* pData) { InvokeStruct* pInvokeData = static_cast(pData); CComPtr spdo; HRESULT hr = CoGetInterfaceAndReleaseStream(pInvokeData->pstrm, IID_PPV_ARGS(&spdo)); if (SUCCEEDED(hr)) { // Create the rename manager CComPtr spsrm; hr = CPowerRenameManager::s_CreateInstance(&spsrm); if (SUCCEEDED(hr)) { // Create the factory for our items CComPtr spsrif; hr = CPowerRenameItem::s_CreateInstance(nullptr, IID_PPV_ARGS(&spsrif)); if (SUCCEEDED(hr)) { // Pass the factory to the manager hr = spsrm->put_renameItemFactory(spsrif); if (SUCCEEDED(hr)) { // Create the rename UI instance and pass the rename manager CComPtr spsrui; hr = CPowerRenameUI::s_CreateInstance(spsrm, spdo, false, &spsrui); if (SUCCEEDED(hr)) { // Call blocks until we are done spsrui->Show(pInvokeData->hwndParent); spsrui->Close(); } } } // Need to call shutdown to break circular dependencies spsrm->Shutdown(); } } delete pInvokeData; Trace::UIShownRet(hr); return 0; }