2 * PROJECT: ReactOS Search Shell Extension
3 * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later)
5 * COPYRIGHT: Copyright 2019 Brock Mammen
8 #include "CSearchBar.h"
9 #include <psdk/wingdi.h>
10 #include <commoncontrols.h>
11 #include <../browseui.h>
15 WINE_DEFAULT_DEBUG_CHANNEL(shellfind
);
20 #define UNIMPLEMENTED DbgPrint("%s is UNIMPLEMENTED!\n", __FUNCTION__)
23 CSearchBar::CSearchBar() :
29 CSearchBar::~CSearchBar()
33 LRESULT
CSearchBar::OnInitDialog(UINT uMsg
, WPARAM wParam
, LPARAM lParam
, BOOL
&bHandled
)
35 SetSearchInProgress(FALSE
);
37 HWND hCombobox
= GetDlgItem(IDC_SEARCH_COMBOBOX
);
38 CComPtr
<IImageList
> pImageList
;
39 HRESULT hResult
= SHGetImageList(SHIL_SMALL
, IID_PPV_ARG(IImageList
, &pImageList
));
40 SendMessage(hCombobox
, CBEM_SETIMAGELIST
, 0, FAILED_UNEXPECTEDLY(hResult
) ? 0 : reinterpret_cast<LPARAM
>(pImageList
.p
));
42 SendMessage(hCombobox
, CBEM_SETEXTENDEDSTYLE
,
43 CBES_EX_CASESENSITIVE
| CBES_EX_NOSIZELIMIT
, CBES_EX_CASESENSITIVE
| CBES_EX_NOSIZELIMIT
);
44 HWND hEditControl
= reinterpret_cast<HWND
>(SendMessage(hCombobox
, CBEM_GETEDITCONTROL
, 0, 0));
45 hResult
= CAddressEditBox_CreateInstance(IID_PPV_ARG(IAddressEditBox
, &m_AddressEditBox
));
46 if (FAILED_UNEXPECTEDLY(hResult
))
49 hResult
= m_AddressEditBox
->Init(hCombobox
, hEditControl
, 0, m_pSite
);
50 if (FAILED_UNEXPECTEDLY(hResult
))
53 // Subscribe to navigation events
54 CComPtr
<IShellBrowser
> pShellBrowser
;
55 hResult
= IUnknown_QueryService(m_pSite
, SID_SShellBrowser
, IID_PPV_ARG(IShellBrowser
, &pShellBrowser
));
57 if (SUCCEEDED(hResult
))
58 AtlAdvise(pShellBrowser
, static_cast<IDispatch
*>(this), DIID_DWebBrowserEvents
, &dwAdviseCookie
);
60 // Invoke the navigate event in case a search results folder is already open
61 DISPPARAMS params
= {0};
62 Invoke(DISPID_NAVIGATECOMPLETE2
, GUID_NULL
, 0, DISPATCH_METHOD
, ¶ms
, NULL
, NULL
, NULL
);
68 // *** ATL event handlers ***
69 LRESULT
CSearchBar::OnSetFocus(UINT uMsg
, WPARAM wParam
, LPARAM lParam
, BOOL
&bHandled
)
71 IUnknown_OnFocusChangeIS(m_pSite
, static_cast<IDeskBand
*>(this), TRUE
);
76 HRESULT
CSearchBar::GetSearchResultsFolder(IShellBrowser
**ppShellBrowser
, HWND
*pHwnd
, IShellFolder
**ppShellFolder
)
79 CComPtr
<IShellBrowser
> pShellBrowser
;
81 ppShellBrowser
= &pShellBrowser
;
84 hr
= IUnknown_QueryService(m_pSite
, SID_SShellBrowser
, IID_PPV_ARG(IShellBrowser
, ppShellBrowser
));
85 if (FAILED_UNEXPECTEDLY(hr
))
89 CComPtr
<IShellView
> pShellView
;
90 hr
= (*ppShellBrowser
)->QueryActiveShellView(&pShellView
);
91 if (FAILED(hr
) || !pShellView
)
94 CComPtr
<IFolderView
> pFolderView
;
95 hr
= pShellView
->QueryInterface(IID_PPV_ARG(IFolderView
, &pFolderView
));
96 if (FAILED(hr
) || !pFolderView
)
99 CComPtr
<IShellFolder
> pShellFolder
;
101 ppShellFolder
= &pShellFolder
;
102 hr
= pFolderView
->GetFolder(IID_PPV_ARG(IShellFolder
, ppShellFolder
));
103 if (FAILED(hr
) || !pShellFolder
)
107 hr
= IUnknown_GetClassID(*ppShellFolder
, &clsid
);
110 if (clsid
!= CLSID_FindFolder
)
115 hr
= pShellView
->GetWindow(pHwnd
);
116 if (FAILED_UNEXPECTEDLY(hr
))
123 LRESULT
CSearchBar::OnSearchButtonClicked(WORD wNotifyCode
, WORD wID
, HWND hWndCtl
, BOOL
& bHandled
)
125 CComHeapPtr
<SearchStart
> pSearchStart(static_cast<SearchStart
*>(CoTaskMemAlloc(sizeof(SearchStart
))));
126 GetDlgItemText(IDC_SEARCH_FILENAME
, pSearchStart
->szFileName
, _countof(pSearchStart
->szFileName
));
127 GetDlgItemText(IDC_SEARCH_QUERY
, pSearchStart
->szQuery
, _countof(pSearchStart
->szQuery
));
128 if (!GetAddressEditBoxPath(pSearchStart
->szPath
))
130 ShellMessageBoxW(_AtlBaseModule
.GetResourceInstance(), m_hWnd
, MAKEINTRESOURCEW(IDS_SEARCHINVALID
), MAKEINTRESOURCEW(IDS_SEARCHLABEL
), MB_OK
| MB_ICONERROR
, pSearchStart
->szPath
);
134 CComPtr
<IShellBrowser
> pShellBrowser
;
135 HRESULT hr
= IUnknown_QueryService(m_pSite
, SID_SShellBrowser
, IID_PPV_ARG(IShellBrowser
, &pShellBrowser
));
136 if (FAILED_UNEXPECTEDLY(hr
))
140 if (FAILED(GetSearchResultsFolder(&pShellBrowser
, &hwnd
, NULL
)))
142 // Open a new search results folder
143 WCHAR szShellGuid
[MAX_PATH
];
144 const WCHAR shellGuidPrefix
[] = L
"shell:::";
145 memcpy(szShellGuid
, shellGuidPrefix
, sizeof(shellGuidPrefix
));
146 hr
= StringFromGUID2(CLSID_FindFolder
, szShellGuid
+ _countof(shellGuidPrefix
) - 1,
147 _countof(szShellGuid
) - _countof(shellGuidPrefix
));
148 if (FAILED_UNEXPECTEDLY(hr
))
151 CComHeapPtr
<ITEMIDLIST
> findFolderPidl
;
152 hr
= SHParseDisplayName(szShellGuid
, NULL
, &findFolderPidl
, 0, NULL
);
153 if (FAILED_UNEXPECTEDLY(hr
))
156 hr
= pShellBrowser
->BrowseObject(findFolderPidl
, 0);
157 if (FAILED_UNEXPECTEDLY(hr
))
160 hr
= GetSearchResultsFolder(&pShellBrowser
, &hwnd
, NULL
);
161 if (FAILED_UNEXPECTEDLY(hr
))
165 ::PostMessageW(hwnd
, WM_SEARCH_START
, 0, (LPARAM
) pSearchStart
.Detach());
167 SetSearchInProgress(TRUE
);
172 LRESULT
CSearchBar::OnStopButtonClicked(WORD wNotifyCode
, WORD wID
, HWND hWndCtl
, BOOL
& bHandled
)
175 HRESULT hr
= GetSearchResultsFolder(NULL
, &hwnd
, NULL
);
177 ::PostMessageW(hwnd
, WM_SEARCH_STOP
, 0, 0);
182 BOOL
CSearchBar::GetAddressEditBoxPath(WCHAR
*szPath
)
184 HWND hComboboxEx
= GetDlgItem(IDC_SEARCH_COMBOBOX
);
185 ::GetWindowTextW(hComboboxEx
, szPath
, MAX_PATH
);
186 INT iSelectedIndex
= SendMessageW(hComboboxEx
, CB_GETCURSEL
, 0, 0);
187 if (iSelectedIndex
!= CB_ERR
)
189 WCHAR szItemText
[MAX_PATH
];
190 COMBOBOXEXITEMW item
= {0};
191 item
.mask
= CBEIF_LPARAM
| CBEIF_TEXT
;
192 item
.iItem
= iSelectedIndex
;
193 item
.pszText
= szItemText
;
194 item
.cchTextMax
= _countof(szItemText
);
195 SendMessageW(hComboboxEx
, CBEM_GETITEMW
, 0, (LPARAM
)&item
);
197 if (!wcscmp(szItemText
, szPath
) && SHGetPathFromIDListW((LPCITEMIDLIST
)item
.lParam
, szItemText
))
199 StringCbCopyW(szPath
, MAX_PATH
* sizeof(WCHAR
), szItemText
);
204 DWORD dwAttributes
= GetFileAttributesW(szPath
);
205 return dwAttributes
!= INVALID_FILE_ATTRIBUTES
206 && (dwAttributes
& FILE_ATTRIBUTE_DIRECTORY
);
209 LRESULT
CSearchBar::OnSize(UINT uMsg
, WPARAM wParam
, LPARAM lParam
, BOOL
&bHandled
)
211 INT iWidth
= LOWORD(lParam
);
214 ((CWindow
)GetDlgItem(IDC_SEARCH_LABEL
)).SetWindowPos(NULL
, 0, 0, iWidth
- iPadding
, 40, SWP_NOACTIVATE
| SWP_NOCOPYBITS
| SWP_NOMOVE
| SWP_NOOWNERZORDER
| SWP_NOZORDER
);
216 int inputs
[] = { IDC_SEARCH_FILENAME
, IDC_SEARCH_QUERY
, IDC_SEARCH_COMBOBOX
, IDC_SEARCH_BUTTON
, IDC_SEARCH_STOP_BUTTON
, IDC_PROGRESS_BAR
};
217 HDWP hdwp
= BeginDeferWindowPos(_countof(inputs
));
218 for (SIZE_T i
= 0; i
< _countof(inputs
); i
++)
220 CWindow wnd
= (CWindow
) GetDlgItem(inputs
[i
]);
222 wnd
.GetWindowRect(&rect
);
223 POINT pt
= { rect
.left
, rect
.top
};
225 hdwp
= wnd
.DeferWindowPos(hdwp
,
229 iWidth
- iPadding
* 2,
230 rect
.bottom
- rect
.top
,
231 SWP_NOZORDER
| SWP_NOACTIVATE
);
233 EndDeferWindowPos(hdwp
);
239 // *** IOleWindow methods ***
240 HRESULT STDMETHODCALLTYPE
CSearchBar::GetWindow(HWND
*lphwnd
)
248 HRESULT STDMETHODCALLTYPE
CSearchBar::ContextSensitiveHelp(BOOL fEnterMode
)
255 // *** IDockingWindow methods ***
256 HRESULT STDMETHODCALLTYPE
CSearchBar::CloseDW(DWORD dwReserved
)
258 // We do nothing, we don't have anything to save yet
259 TRACE("CloseDW called\n");
263 HRESULT STDMETHODCALLTYPE
CSearchBar::ResizeBorderDW(const RECT
*prcBorder
, IUnknown
*punkToolbarSite
, BOOL fReserved
)
265 /* Must return E_NOTIMPL according to MSDN */
269 HRESULT STDMETHODCALLTYPE
CSearchBar::ShowDW(BOOL fShow
)
277 // *** IDeskBand methods ***
278 HRESULT STDMETHODCALLTYPE
CSearchBar::GetBandInfo(DWORD dwBandID
, DWORD dwViewMode
, DESKBANDINFO
*pdbi
)
285 if (pdbi
->dwMask
& DBIM_MINSIZE
)
287 pdbi
->ptMinSize
.x
= 200;
288 pdbi
->ptMinSize
.y
= 30;
291 if (pdbi
->dwMask
& DBIM_MAXSIZE
)
293 pdbi
->ptMaxSize
.y
= -1;
296 if (pdbi
->dwMask
& DBIM_INTEGRAL
)
298 pdbi
->ptIntegral
.y
= 1;
301 if (pdbi
->dwMask
& DBIM_ACTUAL
)
303 pdbi
->ptActual
.x
= 200;
304 pdbi
->ptActual
.y
= 30;
307 if (pdbi
->dwMask
& DBIM_TITLE
)
309 if (!LoadStringW(_AtlBaseModule
.GetResourceInstance(), IDS_SEARCHLABEL
, pdbi
->wszTitle
, _countof(pdbi
->wszTitle
)))
310 return HRESULT_FROM_WIN32(GetLastError());
313 if (pdbi
->dwMask
& DBIM_MODEFLAGS
)
315 pdbi
->dwModeFlags
= DBIMF_NORMAL
| DBIMF_VARIABLEHEIGHT
;
318 if (pdbi
->dwMask
& DBIM_BKCOLOR
)
320 pdbi
->dwMask
&= ~DBIM_BKCOLOR
;
326 // *** IObjectWithSite methods ***
327 HRESULT STDMETHODCALLTYPE
CSearchBar::SetSite(IUnknown
*pUnkSite
)
332 if (pUnkSite
== m_pSite
)
335 TRACE("SetSite called \n");
343 hr
= IUnknown_GetWindow(pUnkSite
, &parentWnd
);
346 ERR("Could not get parent's window ! Status: %08lx\n", hr
);
355 SetParent(parentWnd
);
359 CDialogImpl::Create(parentWnd
);
365 HRESULT STDMETHODCALLTYPE
CSearchBar::GetSite(REFIID riid
, void **ppvSite
)
374 // *** IInputObject methods ***
375 HRESULT STDMETHODCALLTYPE
CSearchBar::UIActivateIO(BOOL fActivate
, LPMSG lpMsg
)
382 // TODO: handle message
385 TranslateMessage(lpMsg
);
386 DispatchMessage(lpMsg
);
391 HRESULT STDMETHODCALLTYPE
CSearchBar::HasFocusIO()
396 HRESULT STDMETHODCALLTYPE
CSearchBar::TranslateAcceleratorIO(LPMSG lpMsg
)
398 if (IsDialogMessage(lpMsg
))
401 if ((lpMsg
->hwnd
== m_hWnd
|| IsChild(lpMsg
->hwnd
)))
403 TranslateMessage(lpMsg
);
404 DispatchMessage(lpMsg
);
411 // *** IPersist methods ***
412 HRESULT STDMETHODCALLTYPE
CSearchBar::GetClassID(CLSID
*pClassID
)
416 *pClassID
= CLSID_FileSearchBand
;
421 // *** IPersistStream methods ***
422 HRESULT STDMETHODCALLTYPE
CSearchBar::IsDirty()
428 HRESULT STDMETHODCALLTYPE
CSearchBar::Load(IStream
*pStm
)
434 HRESULT STDMETHODCALLTYPE
CSearchBar::Save(IStream
*pStm
, BOOL fClearDirty
)
440 HRESULT STDMETHODCALLTYPE
CSearchBar::GetSizeMax(ULARGE_INTEGER
*pcbSize
)
442 // TODO: calculate max size
448 // *** IDispatch methods ***
449 HRESULT STDMETHODCALLTYPE
CSearchBar::GetTypeInfoCount(UINT
*pctinfo
)
455 HRESULT STDMETHODCALLTYPE
CSearchBar::GetTypeInfo(UINT iTInfo
, LCID lcid
, ITypeInfo
**ppTInfo
)
461 HRESULT STDMETHODCALLTYPE
CSearchBar::GetIDsOfNames(REFIID riid
, LPOLESTR
*rgszNames
, UINT cNames
, LCID lcid
, DISPID
*rgDispId
)
467 void CSearchBar::SetSearchInProgress(BOOL bInProgress
)
469 ::ShowWindow(GetDlgItem(IDC_SEARCH_BUTTON
), bInProgress
? SW_HIDE
: SW_SHOW
);
470 ::ShowWindow(GetDlgItem(IDC_SEARCH_STOP_BUTTON
), bInProgress
? SW_SHOW
: SW_HIDE
);
471 HWND hProgressBar
= GetDlgItem(IDC_PROGRESS_BAR
);
472 ::ShowWindow(hProgressBar
, bInProgress
? SW_SHOW
: SW_HIDE
);
473 ::PostMessage(hProgressBar
, PBM_SETMARQUEE
, bInProgress
, 0);
476 HRESULT
CSearchBar::TrySubscribeToSearchEvents()
478 CComPtr
<IShellFolder
> pShellFolder
;
479 HRESULT hr
= GetSearchResultsFolder(NULL
, NULL
, &pShellFolder
);
484 hr
= AtlAdvise(pShellFolder
, static_cast<IDispatch
*>(this), DIID_DSearchCommandEvents
, &fAdviseCookie
);
485 if (FAILED_UNEXPECTEDLY(hr
))
491 HRESULT STDMETHODCALLTYPE
CSearchBar::Invoke(DISPID dispIdMember
, REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
, VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
493 switch (dispIdMember
)
495 case DISPID_NAVIGATECOMPLETE2
:
496 case DISPID_DOCUMENTCOMPLETE
:
498 TrySubscribeToSearchEvents();
500 // Remove the search results folder from the address box
501 CComPtr
<IDispatch
> pDispatch
;
502 HRESULT hResult
= m_AddressEditBox
->QueryInterface(IID_PPV_ARG(IDispatch
, &pDispatch
));
503 if (FAILED_UNEXPECTEDLY(hResult
))
505 pDispatch
->Invoke(dispIdMember
, riid
, lcid
, wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
506 CComPtr
<IShellService
> pShellService
;
507 hResult
= m_AddressEditBox
->QueryInterface(IID_PPV_ARG(IShellService
, &pShellService
));
508 if (FAILED_UNEXPECTEDLY(hResult
))
510 hResult
= pShellService
->SetOwner(NULL
);
511 if (FAILED_UNEXPECTEDLY(hResult
))
513 HWND hComboboxEx
= GetDlgItem(IDC_SEARCH_COMBOBOX
);
514 int index
= SendMessageW(hComboboxEx
, CB_GETCOUNT
, 0, 0);
517 COMBOBOXEXITEMW item
= {0};
518 item
.mask
= CBEIF_LPARAM
;
519 item
.iItem
= index
- 1;
520 SendMessageW(hComboboxEx
, CBEM_GETITEMW
, 0, (LPARAM
)&item
);
523 CComPtr
<IShellFolder
> pDesktopFolder
;
524 hResult
= SHGetDesktopFolder(&pDesktopFolder
);
525 if (FAILED_UNEXPECTEDLY(hResult
))
527 CComPtr
<IShellFolder
> pShellFolder
;
528 hResult
= pDesktopFolder
->BindToObject((LPCITEMIDLIST
)item
.lParam
, NULL
, IID_PPV_ARG(IShellFolder
, &pShellFolder
));
532 hResult
= IUnknown_GetClassID(pShellFolder
, &clsid
);
533 if (SUCCEEDED(hResult
) && clsid
== CLSID_FindFolder
)
535 SendMessageW(hComboboxEx
, CBEM_DELETEITEM
, item
.iItem
, 0);
536 SendMessageW(hComboboxEx
, CB_SETCURSEL
, 0, 0);
540 case DISPID_SEARCHCOMPLETE
:
541 case DISPID_SEARCHABORT
:
542 SetSearchInProgress(FALSE
);