Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1632 lines
44 KiB

#include "shellprv.h"
#include "ids.h"
#include "help.h"
#include "apithk.h"
#include "ascstr.h"
#include "filetype.h"
#include "ftdlg.h"
#include "ftadv.h"
#include "ftaction.h"
const static DWORD cs_rgdwHelpIDsArray[] =
{ // Context Help IDs
IDC_NO_HELP_1, NO_HELP,
IDC_FT_EDIT_DOCICON, IDH_FCAB_FT_EDIT_DOCICON,
IDC_FT_EDIT_DESC, IDH_FCAB_FT_EDIT_DESC,
IDC_FT_EDIT_CHANGEICON, IDH_FCAB_FT_EDIT_CHANGEICON,
IDC_FT_EDIT_LV_CMDSTEXT, IDH_FCAB_FT_EDIT_LV_CMDS,
IDC_FT_EDIT_LV_CMDS, IDH_FCAB_FT_EDIT_LV_CMDS,
IDC_FT_EDIT_NEW, IDH_FCAB_FT_EDIT_NEW,
IDC_FT_EDIT_EDIT, IDH_FCAB_FT_EDIT_EDIT,
IDC_FT_EDIT_REMOVE, IDH_FCAB_FT_EDIT_REMOVE,
IDC_FT_EDIT_DEFAULT, IDH_FCAB_FT_EDIT_DEFAULT,
IDC_FT_EDIT_CONFIRM_OPEN, IDH_CONFIRM_OPEN,
IDC_FT_EDIT_SHOWEXT, IDH_FCAB_FT_EDIT_SHOWEXT,
IDC_FT_EDIT_BROWSEINPLACE, IDH_SAME_WINDOW,
0, 0
};
struct LV_ADDDATA
{
BOOL fDefaultAction;
TCHAR szActionReg[MAX_ACTION];
};
#define ADDDATA_ACTIONREG(plvItem) (((LV_ADDDATA*)((plvItem)->lParam))->szActionReg)
#define ADDDATA_DEFAULTACTION(plvItem) (((LV_ADDDATA*)((plvItem)->lParam))->fDefaultAction)
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
CFTAdvDlg::CFTAdvDlg(LPTSTR pszProgID, LPTSTR pszExt) :
CFTDlg((ULONG_PTR)cs_rgdwHelpIDsArray),
_iDefaultAction(-1), _iLVSel(-1)
{
_szProgID[0] = NULL;
if (pszProgID)
StringCchCopy(_szProgID, ARRAYSIZE(_szProgID), pszProgID);
_szExt[0] = NULL;
if (pszExt && (*pszExt != NULL))
{
StringCchPrintf(_szExt, ARRAYSIZE(_szExt), TEXT(".%s"), pszExt);
}
_hdpaActions = DPA_Create(4);
_hdpaRemovedActions = DPA_Create(1);
}
static int _DeleteLocalAllocCB(void *pItem, void *pData)
{
LocalFree((HLOCAL)pItem);
return 1;
}
CFTAdvDlg::~CFTAdvDlg()
{
if (_hIcon)
DeleteObject(_hIcon);
if (_hfontReg)
DeleteObject(_hfontReg);
if (_hfontBold)
DeleteObject(_hfontBold);
if (_hdpaActions)
DPA_DestroyCallback(_hdpaActions, _DeleteLocalAllocCB, NULL);
if (_hdpaRemovedActions)
DPA_DestroyCallback(_hdpaRemovedActions, _DeleteLocalAllocCB, NULL);
}
///////////////////////////////////////////////////////////////////////////////
// Logic specific to our problem
LRESULT CFTAdvDlg::OnInitDialog(WPARAM wParam, LPARAM lParam)
{
HRESULT hres = _InitAssocStore();
DECLAREWAITCURSOR;
SetWaitCursor();
if (SUCCEEDED(hres))
{
_InitListView();
_InitDefaultActionFont();
// FTEdit_AreDefaultViewersInstalled ????
if (*_szProgID)
{
_SetDocIcon();
_InitDescription();
_FillListView();
_InitDefaultAction();
_SelectListViewItem(0);
_InitChangeIconButton();
_UpdateCheckBoxes();
}
}
else
EndDialog(_hwnd, -1);
ResetWaitCursor();
// Return TRUE so that system set focus
return TRUE;
}
int CFTAdvDlg::_GetIconIndex()
{
// check under the file progid
int iImageIndex = -1;
IAssocInfo* pAI = NULL;
HRESULT hr = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hr))
{
hr = pAI->GetDWORD(AIDWORD_DOCLARGEICON, (DWORD*)&iImageIndex);
pAI->Release();
}
return iImageIndex;
}
HRESULT CFTAdvDlg::_SetDocIcon(int iIndex)
{
HRESULT hres = E_FAIL;
if (-1 == iIndex)
{
iIndex = _GetIconIndex();
}
if (-1 != iIndex)
{
HIMAGELIST hIL = NULL;
Shell_GetImageLists(&hIL, NULL);
if (_hIcon)
{
DeleteObject(_hIcon);
_hIcon = NULL;
}
if (hIL)
{
_hIcon = ImageList_ExtractIcon(g_hinst, hIL, iIndex);
_hIcon = (HICON)CopyImage(_hIcon, IMAGE_ICON, 32, 32, LR_COPYDELETEORG);
HICON hiOld = (HICON)SendDlgItemMessage(_hwnd, IDC_FT_EDIT_DOCICON, STM_SETIMAGE, IMAGE_ICON,
(LPARAM)_hIcon);
if (hiOld)
DestroyIcon(hiOld);
}
}
return hres;
}
LRESULT CFTAdvDlg::OnListViewSelItem(int iItem, LPARAM lParam)
{
_UpdateActionButtons();
return TRUE;
}
LRESULT CFTAdvDlg::OnMeasureItem(WPARAM wParam, LPARAM lParam)
{
TEXTMETRIC tm = {0};
RECT rect;
LPMEASUREITEMSTRUCT lpmis = (LPMEASUREITEMSTRUCT)lParam;
HDC hdc = GetDC(NULL);
HFONT hfontOld = (HFONT)SelectObject(hdc, _hfontBold);
GetTextMetrics(hdc, &tm);
GetClientRect(_GetLVHWND(), &rect);
lpmis->itemWidth = rect.right;
lpmis->itemHeight = tm.tmHeight;
SelectObject(hdc, hfontOld);
ReleaseDC(NULL, hdc);
return TRUE;
}
LRESULT CFTAdvDlg::OnDrawItem(WPARAM wParam, LPARAM lParam)
{
LPDRAWITEMSTRUCT lpDIS = (LPDRAWITEMSTRUCT)lParam;
LRESULT lRet = FALSE;
if (ODT_LISTVIEW == lpDIS->CtlType)
{
HWND hwndLV = _GetLVHWND();
LVITEM lvItem = {0};
HFONT hfontOld = NULL;
BOOL fSel = FALSE;
BOOL fListFocus = FALSE;
TCHAR szAction[MAX_ACTION];
COLORREF crBkgd = 0;
COLORREF crOldText = 0;
lvItem.mask = LVIF_PARAM | LVIF_TEXT | LVIF_STATE;
lvItem.iItem = lpDIS->itemID;
lvItem.stateMask = LVIS_SELECTED|LVIS_FOCUSED;
lvItem.pszText = szAction;
lvItem.cchTextMax = ARRAYSIZE(szAction);
ListView_GetItem(hwndLV, &lvItem);
fSel = (lvItem.state & LVIS_SELECTED);
fListFocus = (GetFocus() == hwndLV);
crBkgd = (fSel ? (fListFocus ? COLOR_HIGHLIGHT : COLOR_3DFACE) : COLOR_WINDOW);
SetBkColor(lpDIS->hDC, GetSysColor(crBkgd));
FillRect(lpDIS->hDC, &lpDIS->rcItem, (HBRUSH)IntToPtr(crBkgd + 1));
crOldText = SetTextColor(lpDIS->hDC,
GetSysColor(fSel ? (fListFocus ? COLOR_HIGHLIGHTTEXT : COLOR_WINDOWTEXT) :
COLOR_WINDOWTEXT));
// Use Bold font for default action
hfontOld = (HFONT)SelectObject(lpDIS->hDC,
_IsDefaultAction(ADDDATA_ACTIONREG(&lvItem)) ? _hfontBold : _hfontReg);
int iOldBkMode = SetBkMode(lpDIS->hDC, OPAQUE);
DrawText(lpDIS->hDC, szAction, lstrlen(szAction), &lpDIS->rcItem, 0);
SetBkMode(lpDIS->hDC, iOldBkMode);
SetTextColor(lpDIS->hDC, crOldText);
SelectObject(lpDIS->hDC, hfontOld);
if(fListFocus && (lvItem.state & LVIS_FOCUSED))
DrawFocusRect(lpDIS->hDC, &lpDIS->rcItem);
lRet = TRUE;
}
return lRet;
}
HRESULT CFTAdvDlg::_InitDefaultActionFont()
{
HFONT hfontDlg = GetWindowFont(_hwnd);
LOGFONT lf = {0};
LOGFONT lfDlg = {0};
GetObject(hfontDlg, sizeof(LOGFONT), &lfDlg);
SystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(lf), &lf, FALSE);
// Normal font
lf.lfWeight = FW_NORMAL;
lf.lfHeight = lfDlg.lfHeight;
_hfontReg = CreateFontIndirect(&lf);
// Bold font
lf.lfWeight = FW_BOLD;
_hfontBold = CreateFontIndirect(&lf);
return (_hfontReg && _hfontBold) ? S_OK : E_FAIL;
}
HRESULT CFTAdvDlg::_SelectListViewItem(int i)
{
LVITEM lvItem = {0};
lvItem.iItem = i;
lvItem.mask = LVIF_STATE;
lvItem.stateMask = LVIS_SELECTED | LVIS_FOCUSED;
lvItem.state = LVIS_SELECTED | LVIS_FOCUSED;
ListView_SetItem(_GetLVHWND(), &lvItem);
return S_OK;
}
// pszText and cchTextMax needs to be set
BOOL CFTAdvDlg::_FindActionLVITEM(LPTSTR pszActionReg, LVITEM* plvItem)
{
HWND hwndLV = _GetLVHWND();
int iCount = ListView_GetItemCount(hwndLV);
BOOL fRet = FALSE;
plvItem->mask = LVIF_TEXT | LVIF_PARAM;
for (int i = 0; i < iCount; ++i)
{
plvItem->iItem = i;
if (ListView_GetItem(hwndLV, plvItem))
{
if (!lstrcmpi(pszActionReg, ADDDATA_ACTIONREG(plvItem)))
{
fRet = TRUE;
break;
}
}
}
return fRet;
}
HRESULT CFTAdvDlg::_InitDefaultAction()
{
// Get it from the classstore
IAssocInfo* pAI = NULL;
HRESULT hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
TCHAR szActionReg[MAX_ACTION];
DWORD cchActionReg = ARRAYSIZE(szActionReg);
HWND hwndLV = _GetLVHWND();
int iIndex = -1;
hres = pAI->GetString(AISTR_PROGIDDEFAULTACTION, szActionReg, &cchActionReg);
if (SUCCEEDED(hres))
{
TCHAR szActionLVI[MAX_ACTION];
LVITEM lvItem = {0};
lvItem.pszText = szActionLVI;
lvItem.cchTextMax = ARRAYSIZE(szActionLVI);
if (_FindActionLVITEM(szActionReg, &lvItem))
hres = _SetDefaultAction(lvItem.iItem);
else
hres = S_OK;
}
pAI->Release();
}
return hres;
}
BOOL CFTAdvDlg::_GetDefaultAction(LPTSTR pszActionReg, DWORD cchActionReg)
{
BOOL fRet = FALSE;
HWND hwndLV = _GetLVHWND();
LVITEM lvItem = {0};
int iCount = ListView_GetItemCount(hwndLV);
TCHAR szActionRegLocal[MAX_ACTION];
lvItem.mask = LVIF_TEXT | LVIF_PARAM;
lvItem.pszText = szActionRegLocal;
lvItem.cchTextMax = ARRAYSIZE(szActionRegLocal);
for (int i = 0; i < iCount; ++i)
{
lvItem.iItem = i;
if (ListView_GetItem(hwndLV, &lvItem))
{
if (ADDDATA_DEFAULTACTION(&lvItem))
{
if(SUCCEEDED(StringCchCopy(pszActionReg, cchActionReg, ADDDATA_ACTIONREG(&lvItem))))
fRet = TRUE;
else
fRet = FALSE;
break;
}
}
}
return fRet;
}
BOOL CFTAdvDlg::_IsDefaultAction(LPTSTR pszActionReg)
{
BOOL fRet = FALSE;
TCHAR szActionReg[MAX_ACTION];
if (_GetDefaultAction(szActionReg, ARRAYSIZE(szActionReg)))
{
if (!lstrcmpi(szActionReg, pszActionReg))
fRet = TRUE;
}
return fRet;
}
void CFTAdvDlg::_CheckDefaultAction()
{
HWND hwndLV = _GetLVHWND();
// Is there only one elem?
if (1 == ListView_GetItemCount(hwndLV))
{
_SetDefaultActionHelper(0, TRUE);
}
}
HRESULT CFTAdvDlg::_SetDefaultAction(int iIndex)
{
HWND hwndLV = _GetLVHWND();
// Remove previous default if any
if (-1 != _iDefaultAction)
{
_SetDefaultActionHelper(_iDefaultAction, FALSE);
ListView_RedrawItems(hwndLV, _iDefaultAction, _iDefaultAction);
}
// Set new
_iDefaultAction = iIndex;
// iIndex == -1 means no default
if (iIndex >= 0)
{
_SetDefaultActionHelper(_iDefaultAction, TRUE);
ListView_RedrawItems(hwndLV, _iDefaultAction, _iDefaultAction);
}
return S_OK;
}
void CFTAdvDlg::_SetDefaultActionHelper(int iIndex, BOOL fDefault)
{
HWND hwndLV = _GetLVHWND();
LVITEM lvItem = {0};
lvItem.mask = LVIF_PARAM;
lvItem.iItem = iIndex;
_iDefaultAction = -1;
if (ListView_GetItem(hwndLV, &lvItem))
{
ADDDATA_DEFAULTACTION(&lvItem) = fDefault;
_iDefaultAction = iIndex;
}
}
HRESULT CFTAdvDlg::_InitListView()
{
LVCOLUMN lvColumn = {0};
HWND hwndLV = _GetLVHWND();
RECT rc = {0};
{
// What's this?
// We need to handle the WM_MEASUREITEM message from the listview. This msg
// is sent before we receive the WM_INITDIALOG and thus before we connect the
// this C++ obj to the HWND. By changing the style here we receive the msg
// after the C++ obj and the HWND are connected.
LONG lStyle = GetWindowLong(hwndLV, GWL_STYLE);
lStyle &= ~LVS_LIST;
SetWindowLong(hwndLV, GWL_STYLE, lStyle | LVS_REPORT);
}
//
// Set the columns
//
GetClientRect(hwndLV, &rc);
lvColumn.mask = LVCF_SUBITEM|LVCF_WIDTH;
lvColumn.cx = rc.right - GetSystemMetrics(SM_CXBORDER);
lvColumn.iSubItem = 0;
ListView_InsertColumn(hwndLV, 0, &lvColumn);
return S_OK;
}
HRESULT CFTAdvDlg::_UpdateActionButtons()
{
HRESULT hres = E_FAIL;
TCHAR szAction[MAX_ACTION];
BOOL bRet = FALSE;
LVITEM lvItem = {0};
lvItem.pszText = szAction;
lvItem.cchTextMax = ARRAYSIZE(szAction);
bRet = _GetListViewSelectedItem(LVIF_TEXT | LVIF_PARAM, 0, &lvItem);
// If we don't have a selected item Or we don't have any text for that item.
if (!bRet || !(*(lvItem.pszText)))
{
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_EDIT), FALSE);
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_REMOVE), FALSE);
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_DEFAULT), TRUE);
hres = S_OK;
}
else
{
if (_IsNewPROGIDACTION(lvItem.pszText))
{
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_EDIT), TRUE);
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_REMOVE), TRUE);
hres = S_OK;
}
else
{
IAssocInfo* pAI = NULL;
hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
DWORD dwAttributes;
HWND hwndLV = _GetLVHWND();
// REARCHITECT: This code should be in ftassoc.cpp, and we should have
// more AIBOOL_ flags for this
hres = pAI->GetDWORD(AIDWORD_PROGIDEDITFLAGS, &dwAttributes);
if (FAILED(hres))
{
// It failed, probably there is no EditFlags value for this progID, let's
// set some default value for dwAttributes
dwAttributes = 0;
}
// REARCHITECT (end)
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_EDIT),
!((dwAttributes & FTA_NoEditVerb) &&
!(dwAttributes & FTAV_UserDefVerb)));
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_REMOVE),
!((dwAttributes & FTA_NoRemoveVerb) &&
!(dwAttributes & FTAV_UserDefVerb)));
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_DEFAULT),
!(dwAttributes & FTA_NoEditDflt));
// Enable the default button only if the action is not already
// the default action
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_DEFAULT),
!_IsDefaultAction(ADDDATA_ACTIONREG(&lvItem)));
pAI->Release();
}
}
}
return hres;
}
HRESULT CFTAdvDlg::_UpdateCheckBoxes()
{
BOOL fBool;
IAssocInfo* pAI = NULL;
HRESULT hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
hres = pAI->GetBOOL(AIBOOL_CONFIRMOPEN, &fBool);
if (SUCCEEDED(hres))
CheckDlgButton(_hwnd, IDC_FT_EDIT_CONFIRM_OPEN, !fBool);
hres = pAI->GetBOOL(AIBOOL_ALWAYSSHOWEXT, &fBool);
if (SUCCEEDED(hres))
CheckDlgButton(_hwnd, IDC_FT_EDIT_SHOWEXT, fBool);
hres = pAI->GetBOOL(AIBOOL_BROWSEINPLACEENABLED, &fBool);
if (SUCCEEDED(hres))
{
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_BROWSEINPLACE), fBool);
if (fBool)
{
hres = pAI->GetBOOL(AIBOOL_BROWSEINPLACE, &fBool);
if (SUCCEEDED(hres))
CheckDlgButton(_hwnd, IDC_FT_EDIT_BROWSEINPLACE, fBool);
}
else
CheckDlgButton(_hwnd, IDC_FT_EDIT_BROWSEINPLACE, FALSE);
}
pAI->Release();
}
return hres;
}
HRESULT CFTAdvDlg::_InitChangeIconButton()
{
HRESULT hres = E_FAIL;
BOOL fChangeIcon = TRUE;
IAssocInfo* pAI = NULL;
hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
hres = pAI->GetBOOL(AIBOOL_EDITDOCICON, &fChangeIcon);
if (SUCCEEDED(hres))
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_CHANGEICON), fChangeIcon);
pAI->Release();
}
return hres;
}
HRESULT CFTAdvDlg::_InitDescription()
{
HRESULT hres = E_FAIL;
BOOL fEditDescr = TRUE;
IAssocInfo* pAI = NULL;
hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
TCHAR szProgIDDescr[MAX_PROGIDDESCR];
DWORD cchProgIDDescr = ARRAYSIZE(szProgIDDescr);
hres = pAI->GetString(AISTR_PROGIDDESCR, szProgIDDescr, &cchProgIDDescr);
if (SUCCEEDED(hres))
SetDlgItemText(_hwnd, IDC_FT_EDIT_DESC, szProgIDDescr);
hres = pAI->GetBOOL(AIBOOL_EDITDESCR, &fEditDescr);
EnableWindow(GetDlgItem(_hwnd, IDC_FT_EDIT_DESC), fEditDescr);
pAI->Release();
Edit_LimitText(GetDlgItem(_hwnd, IDC_FT_EDIT_DESC), MAX_PROGIDDESCR - 1);
}
return hres;
}
HRESULT CFTAdvDlg::_FillListView()
{
HRESULT hres = E_FAIL;
IEnumAssocInfo* pEnum = NULL;
hres = _pAssocStore->EnumAssocInfo(ASENUM_ACTION, _szProgID, AIINIT_PROGID, &pEnum);
if (SUCCEEDED(hres))
{
int iItem = 0;
IAssocInfo* pAI = NULL;
while (S_OK == pEnum->Next(&pAI))
{
TCHAR szActionReg[MAX_ACTION];
DWORD cchActionReg = ARRAYSIZE(szActionReg);
hres = pAI->GetString(AISTR_ACTION, szActionReg, &cchActionReg);
if (SUCCEEDED(hres))
{
TCHAR szActionFN[MAX_ACTION];
DWORD cchActionFN = ARRAYSIZE(szActionFN);
hres = pAI->GetString(AISTR_ACTIONFRIENDLY, szActionFN, &cchActionFN);
if (SUCCEEDED(hres))
{
if (S_FALSE == hres)
{
hres = StringCchCopy(szActionFN, ARRAYSIZE(szActionFN), szActionReg);
}
if (SUCCEEDED(hres) && -1 != _InsertListViewItem(iItem, szActionReg, szActionFN))
{
++iItem;
}
}
}
pAI->Release();
}
pEnum->Release();
}
return hres;
}
LRESULT CFTAdvDlg::OnChangeIcon(WORD wNotif)
{
IAssocInfo* pAI;
HRESULT hr = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hr))
{
DWORD cchIconLoc = ARRAYSIZE(_szOldIconLoc);
hr = pAI->GetString(AISTR_ICONLOCATION, _szOldIconLoc, &cchIconLoc);
pAI->Release();
_iOldIcon = PathParseIconLocation(_szOldIconLoc);
}
if (FAILED(hr))
{
hr = StringCchCopy(_szOldIconLoc, ARRAYSIZE(_szOldIconLoc), TEXT("shell32.dll"));
if(SUCCEEDED(hr))
{
_iOldIcon = -(IDI_SYSFILE);
}
}
if (SUCCEEDED(hr))
{
// setup the in params
int iIcon = _iOldIcon;
hr = StringCchCopy(_szIconLoc, ARRAYSIZE(_szIconLoc), _szOldIconLoc);
if (SUCCEEDED(hr) && PickIconDlg(_hwnd, _szIconLoc, ARRAYSIZE(_szIconLoc), &iIcon))
{
_SetDocIcon(Shell_GetCachedImageIndex(_szIconLoc, iIcon, 0));
// Format the _szIconLoc
int iLen = lstrlen(_szIconLoc);
hr = StringCchPrintf(_szIconLoc + iLen, ARRAYSIZE(_szIconLoc) - iLen, TEXT(",%d"), iIcon);
}
else
{
_szIconLoc[0] = 0;
}
}
return FALSE;
}
// Return value:
// TRUE: Check succeeded, everything is OK
// FALSE: Check failed
BOOL CFTAdvDlg::_CheckForDuplicateNewAction(LPTSTR pszActionReg, LPTSTR pszActionFN)
{
// we just go through the listview content
HWND hwndLV = _GetLVHWND();
int cItem = ListView_GetItemCount(hwndLV);
BOOL fRet = TRUE;
for (int i = 0; (i < cItem) && fRet; ++i)
{
TCHAR szActionFN[MAX_ACTION];
LVITEM lvItem = {0};
lvItem.mask = LVIF_PARAM | LVIF_TEXT;
lvItem.iItem = i;
lvItem.pszText = szActionFN;
lvItem.cchTextMax = ARRAYSIZE(szActionFN);
ListView_GetItem(hwndLV, &lvItem);
if (!lstrcmpi(lvItem.pszText, pszActionFN))
{
fRet = FALSE;
}
else
{
if (!lstrcmpi(ADDDATA_ACTIONREG(&lvItem), pszActionReg))
{
fRet = FALSE;
}
}
}
return fRet;
}
// Return value:
// TRUE: Check succeeded, everything is OK
// FALSE: Check failed
BOOL CFTAdvDlg::_CheckForDuplicateEditAction(LPTSTR pszActionRegOriginal, LPTSTR pszActionReg,
LPTSTR pszActionFNOriginal, LPTSTR pszActionFN)
{
// we just go through the listview content
HWND hwndLV = _GetLVHWND();
int cItem = ListView_GetItemCount(hwndLV);
BOOL fRet = TRUE;
for (int i = 0; (i < cItem) && fRet; ++i)
{
TCHAR szActionFN[MAX_ACTION];
LVITEM lvItem = {0};
lvItem.mask = LVIF_PARAM | LVIF_TEXT;
lvItem.iItem = i;
lvItem.pszText = szActionFN;
lvItem.cchTextMax = ARRAYSIZE(szActionFN);
ListView_GetItem(hwndLV, &lvItem);
if (!lstrcmpi(lvItem.pszText, pszActionFN))
{
// they are the same, this can happen if this is the Action we were editing
// and we did not change the action name
// Is this the original one we were editing?
if (lstrcmpi(szActionFN, pszActionFNOriginal))
{
// No, it's not the original, we have a dup
fRet = FALSE;
}
}
else
{
if (!lstrcmpi(ADDDATA_ACTIONREG(&lvItem), pszActionReg))
{
// they are the same, this can happen if this is the Action we were editing
// and we did not change the action name
// Is this the original one we were editing?
if (lstrcmpi(ADDDATA_ACTIONREG(&lvItem), pszActionRegOriginal))
{
// No, it's not the original, we have a dup
fRet = FALSE;
}
}
}
}
return fRet;
}
LRESULT CFTAdvDlg::OnNewButton(WORD wNotif)
{
TCHAR szProgIDDescr[MAX_PROGIDDESCR];
PROGIDACTION pida = {0};
CFTActionDlg* pActionDlg = NULL;
pida.fNew = TRUE;
GetDlgItemText(_hwnd, IDC_FT_EDIT_DESC, szProgIDDescr, ARRAYSIZE(szProgIDDescr));
// FALSE: New (not-Edit)
pActionDlg = new CFTActionDlg(&pida, szProgIDDescr, FALSE);
if (pActionDlg)
{
BOOL fShowAgain;
do
{
fShowAgain = FALSE;
if (IDOK == pActionDlg->DoModal(g_hinst, MAKEINTRESOURCE(DLG_FILETYPEOPTIONSCMD), _hwnd))
{
// Do we have duplicate actions?
if (_CheckForDuplicateNewAction(pida.szActionReg, pida.szAction))
{
// No
HRESULT hres = _AppendPROGIDACTION(&pida);
if (SUCCEEDED(hres))
{
int iItem = _InsertListViewItem(0, pida.szActionReg, pida.szAction);
hres = S_OK;
if (-1 != iItem)
_SelectListViewItem(iItem);
}
}
else
{
// Yes
fShowAgain = TRUE;
pActionDlg->SetShowAgain();
}
}
if (fShowAgain)
{
ShellMessageBox(g_hinst, _hwnd, MAKEINTRESOURCE(IDS_FT_MB_EXISTINGACTION),
MAKEINTRESOURCE(IDS_FT), MB_OK | MB_ICONSTOP, pida.szAction);
}
} while (fShowAgain);
pActionDlg->Release();
}
_CheckDefaultAction();
return FALSE;
}
LRESULT CFTAdvDlg::OnEditButton(WORD wNotif)
{
TCHAR szAction[MAX_ACTION];
HRESULT hres = E_FAIL;
LONG lRes = 0;
LVITEM lvItem = {0};
// lvItem.iSubItem = 0;
lvItem.pszText = szAction;
lvItem.cchTextMax = ARRAYSIZE(szAction);
if (_GetListViewSelectedItem(LVIF_TEXT, 0, &lvItem))
{
TCHAR szProgIDDescr[MAX_PROGIDDESCR];
PROGIDACTION* pPIDA = NULL;
PROGIDACTION pida = {0};
GetDlgItemText(_hwnd, IDC_FT_EDIT_DESC, szProgIDDescr, ARRAYSIZE(szProgIDDescr));
BOOL fNewOrEdit = SUCCEEDED(_GetPROGIDACTION(lvItem.pszText, &pPIDA));
if (!fNewOrEdit)
{
hres = _FillPROGIDACTION(&pida, ADDDATA_ACTIONREG(&lvItem), szAction);
pPIDA = &pida;
}
else
{
hres = S_OK;
}
if (SUCCEEDED(hres))
{
// TRUE: Edit
CFTActionDlg* pActionDlg = new CFTActionDlg(pPIDA, szProgIDDescr, TRUE);
if (pActionDlg)
{
BOOL fShowAgain;
do
{
fShowAgain = FALSE;
if (IDOK == pActionDlg->DoModal(g_hinst, MAKEINTRESOURCE(DLG_FILETYPEOPTIONSCMD), _hwnd))
{
// Do we have duplicate actions?
if (_CheckForDuplicateEditAction(ADDDATA_ACTIONREG(&lvItem), pPIDA->szActionReg,
lvItem.pszText, pPIDA->szAction))
{
// No
if (!fNewOrEdit)
{
hres = _AppendPROGIDACTION(pPIDA);
}
else
{
hres = S_OK;
}
if (SUCCEEDED(hres))
{
// Replace the current item text
hres = StringCchCopy(lvItem.pszText, lvItem.cchTextMax, pPIDA->szAction);
if(SUCCEEDED(hres))
{
ListView_SetItem(_GetLVHWND(), &lvItem);
}
}
}
else
{
// Yes
fShowAgain = TRUE;
pActionDlg->SetShowAgain();
}
}
if (fShowAgain)
{
ShellMessageBox(g_hinst, _hwnd, MAKEINTRESOURCE(IDS_FT_MB_EXISTINGACTION),
MAKEINTRESOURCE(IDS_FT), MB_OK | MB_ICONSTOP, pPIDA->szAction);
}
} while (fShowAgain);
pActionDlg->Release();
}
}
}
return FALSE;
}
LRESULT CFTAdvDlg::OnSetDefault(WORD wNotif)
{
BOOL bRet;
LVITEM lvItem = {0};
// lvItem.iSubItem = 0;
bRet = _GetListViewSelectedItem(0, 0, &lvItem);
if (bRet)
_SetDefaultAction(lvItem.iItem);
else
_SetDefaultAction(-1);
return FALSE;
}
LRESULT CFTAdvDlg::OnRemoveButton(WORD wNotif)
{
TCHAR szExt[MAX_EXT];
HRESULT hres = E_FAIL;
LONG lRes = 0;
LVITEM lvItem = {0};
// lvItem.iSubItem = 0;
lvItem.pszText = szExt;
lvItem.cchTextMax = ARRAYSIZE(szExt);
if (_GetListViewSelectedItem(LVIF_TEXT, 0, &lvItem))
{
if (IDYES == ShellMessageBox(g_hinst, _hwnd, MAKEINTRESOURCE(IDS_FT_MB_REMOVEACTION),
MAKEINTRESOURCE(IDS_FT), MB_YESNO | MB_ICONQUESTION))
{
//
// First take care of data side
//
// Yes. Is this a new Action?
PROGIDACTION* pPIDA = NULL;
if (SUCCEEDED(_GetPROGIDACTION(lvItem.pszText, &pPIDA)) && pPIDA->fNew)
{
// Yes, we'll just remove it from the DPA
hres = _RemovePROGIDACTION(pPIDA);
}
else
{
// No, add its name to the list to delete if user press OK
DWORD cchSize = ARRAYSIZE(ADDDATA_ACTIONREG(&lvItem));
LPTSTR pszActionToRemove = (LPTSTR)LocalAlloc(LPTR,
cchSize * sizeof(TCHAR));
hres = E_OUTOFMEMORY;
if (pszActionToRemove)
{
hres = StringCchCopy(pszActionToRemove, cchSize, ADDDATA_ACTIONREG(&lvItem));
if (SUCCEEDED(hres) && -1 != DPA_AppendPtr(_hdpaRemovedActions, pszActionToRemove))
hres = S_OK;
else
LocalFree((HLOCAL)pszActionToRemove);
}
if (E_OUTOFMEMORY == hres)
{
//Out of memory
ShellMessageBox(g_hinst, _hwnd, MAKEINTRESOURCE(IDS_ERROR +
ERROR_NOT_ENOUGH_MEMORY), MAKEINTRESOURCE(IDS_FT),
MB_OK | MB_ICONSTOP);
}
}
//
// Then update UI, I/A
//
if (SUCCEEDED(hres))
{
HWND hwndLV = _GetLVHWND();
int iCount = ListView_GetItemCount(hwndLV);
int iNextSel = -1;
ListView_DeleteItem(hwndLV, lvItem.iItem);
if (iCount > lvItem.iItem)
iNextSel = lvItem.iItem;
else
if (lvItem.iItem > 0)
iNextSel = lvItem.iItem - 1;
if (-1 != iNextSel)
_SelectListViewItem(iNextSel);
}
}
else
hres = S_FALSE;
}
_CheckDefaultAction();
return FALSE;
}
LRESULT CFTAdvDlg::OnOK(WORD wNotif)
{
BOOL fChecksPassed = FALSE;
// Yes, we need to:
// - remove "removed" items, modify "edited" ones,
// and add "New" ones
// - update checkboxes related stuff
// - set the default action
// - set the icon
// - set the description
{
int n = DPA_GetPtrCount(_hdpaRemovedActions);
if (n)
{
IAssocInfo* pAI;
HRESULT hres = E_FAIL;
for (int i = 0; i < n; ++i)
{
LPTSTR pszActionToRemove = (LPTSTR)DPA_GetPtr(_hdpaRemovedActions, i);
if (pszActionToRemove && *pszActionToRemove)
{
hres = _pAssocStore->GetComplexAssocInfo(_szProgID, AIINIT_PROGID,
pszActionToRemove, AIINIT_ACTION, &pAI);
if (SUCCEEDED(hres))
{
pAI->Delete(AIALL_NONE);
pAI->Release();
}
LocalFree((HLOCAL)pszActionToRemove);
DPA_DeletePtr(_hdpaRemovedActions, i);
}
}
}
}
{
int n = DPA_GetPtrCount(_hdpaActions);
if (n)
{
IAssocInfo* pAI = NULL;
HRESULT hres = E_FAIL;
for (int i = n - 1; i >= 0; --i)
{
PROGIDACTION* pPIDAFromList = (PROGIDACTION*)DPA_GetPtr(_hdpaActions, i);
if (pPIDAFromList)
{
// Is it an Edited one?
if (!pPIDAFromList->fNew)
{
// Yes, remove the old one first
hres = _pAssocStore->GetComplexAssocInfo(_szProgID, AIINIT_PROGID,
pPIDAFromList->szOldActionReg, AIINIT_ACTION, &pAI);
if (SUCCEEDED(hres))
{
pAI->Delete(AIALL_NONE);
pAI->Release();
}
}
// Add new data
hres = _pAssocStore->GetComplexAssocInfo(_szProgID, AIINIT_PROGID,
pPIDAFromList->szActionReg, AIINIT_ACTION, &pAI);
if (SUCCEEDED(hres))
{
hres = pAI->SetData(AIDATA_PROGIDACTION, (PBYTE)pPIDAFromList,
sizeof(*pPIDAFromList));
pAI->Release();
}
// Clean up DPA
_DeletePROGIDACTION(pPIDAFromList);
DPA_DeletePtr(_hdpaActions, i);
}
}
}
}
{
IAssocInfo* pAI = NULL;
HWND hwndLV = _GetLVHWND();
LVFINDINFO lvFindInfo = {0};
int iIndex = -1;
HRESULT hres = _pAssocStore->GetAssocInfo(_szProgID, AIINIT_PROGID, &pAI);
if (SUCCEEDED(hres))
{
TCHAR szActionReg[MAX_ACTION];
hres = pAI->SetBOOL(AIBOOL_CONFIRMOPEN,
!IsDlgButtonChecked(_hwnd, IDC_FT_EDIT_CONFIRM_OPEN));
hres = pAI->SetBOOL(AIBOOL_ALWAYSSHOWEXT,
IsDlgButtonChecked(_hwnd, IDC_FT_EDIT_SHOWEXT));
hres = pAI->SetBOOL(AIBOOL_BROWSEINPLACE,
IsDlgButtonChecked(_hwnd, IDC_FT_EDIT_BROWSEINPLACE));
// Set the default action, if any
if (_GetDefaultAction(szActionReg, ARRAYSIZE(szActionReg)))
{
hres = pAI->SetString(AISTR_PROGIDDEFAULTACTION, szActionReg);
}
else
{
hres = pAI->SetString(AISTR_PROGIDDEFAULTACTION, TEXT(""));
}
// Set the icon, if changed
if (_szIconLoc[0])
{
// Set it in the registry
hres = pAI->SetString(AISTR_ICONLOCATION, _szIconLoc);
if (_szOldIconLoc[0])
{
int iIconIndex = Shell_GetCachedImageIndex(_szOldIconLoc, _iOldIcon, 0);
SHUpdateImage(_szOldIconLoc, _iOldIcon, 0, iIconIndex);
}
}
// Set the description
{
TCHAR szProgIDDescr[MAX_PROGIDDESCR];
GetDlgItemText(_hwnd, IDC_FT_EDIT_DESC, szProgIDDescr,
ARRAYSIZE(szProgIDDescr));
hres = pAI->SetString(AISTR_PROGIDDESCR, szProgIDDescr);
}
pAI->Release();
}
}
EndDialog(_hwnd, IDOK);
return FALSE;
}
LRESULT CFTAdvDlg::OnNotifyListView(UINT uCode, LPNMHDR pNMHDR)
{
HWND hwndLV = _GetLVHWND();
LRESULT lres = FALSE;
switch(uCode)
{
case NM_DBLCLK:
if (IsWindowEnabled(GetDlgItem(_hwnd, IDC_FT_EDIT_EDIT)))
PostMessage(_hwnd, WM_COMMAND, (WPARAM)IDC_FT_EDIT_EDIT, 0);
break;
//review: do I really need to do this?
case NM_SETFOCUS:
case NM_KILLFOCUS:
// update list view
ListView_RedrawItems(hwndLV, 0, ListView_GetItemCount(hwndLV));
UpdateWindow(hwndLV);
break;
case LVN_DELETEITEM:
{
NMLISTVIEW* pNMLV = (NMLISTVIEW*)pNMHDR;
if (pNMLV->lParam)
{
LocalFree((HLOCAL)(pNMLV->lParam));
}
break;
}
case LVN_ITEMCHANGED:
{
NMLISTVIEW* pNMLV = (NMLISTVIEW*)pNMHDR;
// Is a new item being selected/unselected?
if (pNMLV->uChanged & LVIF_STATE)
{
// Yes
OnListViewSelItem(pNMLV->iItem, NULL);
}
break;
}
}
return lres;
}
LRESULT CFTAdvDlg::OnCancel(WORD wNotif)
{
EndDialog(_hwnd, IDCANCEL);
return FALSE;
}
LRESULT CFTAdvDlg::OnDestroy(WPARAM wParam, LPARAM lParam)
{
CFTDlg::OnDestroy(wParam, lParam);
return FALSE;
}
BOOL CFTAdvDlg::_GetListViewSelectedItem(UINT uMask, UINT uStateMask, LVITEM* plvItem)
{
BOOL fSel = FALSE;
HWND hwndLV = _GetLVHWND();
plvItem->mask = uMask | LVIF_STATE | LVIF_PARAM;
plvItem->stateMask = uStateMask | LVIS_SELECTED;
// Do we have the selection cached?
if (-1 != _iLVSel)
{
// Yes, make sure it's valid
plvItem->iItem = _iLVSel;
ListView_GetItem(hwndLV, plvItem);
if (plvItem->state & LVIS_SELECTED)
fSel = TRUE;
}
// Cache was wrong
if (!fSel)
{
int iCount = ListView_GetItemCount(hwndLV);
for (int i=0; (i < iCount) && !fSel; ++i)
{
plvItem->iItem = i;
ListView_GetItem(hwndLV, plvItem);
if (plvItem->state & LVIS_SELECTED)
fSel = TRUE;
}
if (fSel)
_iLVSel = i;
}
return fSel;
}
int CFTAdvDlg::_InsertListViewItem(int iItem, LPTSTR pszActionReg, LPTSTR pszActionFN)
{
int iRet = -1;
HWND hwndLV = _GetLVHWND();
LVITEM lvItem = {0};
lvItem.mask = LVIF_TEXT | LVIF_PARAM;
// Extension
lvItem.iItem = iItem;
lvItem.pszText = pszActionFN;
lvItem.cchTextMax = lstrlen(pszActionFN);
LV_ADDDATA* plvadddata = (LV_ADDDATA*)LocalAlloc(LPTR, sizeof(LV_ADDDATA));
if (plvadddata)
{
lvItem.lParam = (LPARAM)plvadddata;
if(SUCCEEDED(StringCchCopy(ADDDATA_ACTIONREG(&lvItem), ARRAYSIZE(ADDDATA_ACTIONREG(&lvItem)), pszActionReg)))
{
ADDDATA_DEFAULTACTION(&lvItem) = 0;
iRet = ListView_InsertItem(hwndLV, &lvItem);
}
else
{
LocalFree(plvadddata);
}
}
return iRet;
}
HWND CFTAdvDlg::_GetLVHWND()
{
return GetDlgItem(_hwnd, IDC_FT_EDIT_LV_CMDS);
}
void CFTAdvDlg::_DeletePROGIDACTION(PROGIDACTION* pPIDA)
{
if (pPIDA)
LocalFree((HLOCAL)pPIDA);
}
HRESULT CFTAdvDlg::_RemovePROGIDACTION(PROGIDACTION* pPIDA)
{
HRESULT hres = E_FAIL;
int n = DPA_GetPtrCount(_hdpaActions);
for (int i = 0; (i < n) && FAILED(hres); ++i)
{
PROGIDACTION* pPIDAFromList = (PROGIDACTION*)DPA_GetPtr(_hdpaActions, i);
if (pPIDAFromList == pPIDA)
{
_DeletePROGIDACTION(pPIDAFromList);
DPA_DeletePtr(_hdpaActions, i);
hres = S_OK;
}
}
return hres;
}
HRESULT CFTAdvDlg::_CreatePROGIDACTION(PROGIDACTION** ppPIDA)
{
HRESULT hres = E_OUTOFMEMORY;
*ppPIDA = (PROGIDACTION*)LocalAlloc(LPTR, sizeof(PROGIDACTION));
if (*ppPIDA)
hres = S_OK;
return hres;
}
HRESULT CFTAdvDlg::_CopyPROGIDACTION(PROGIDACTION* pPIDADest, PROGIDACTION* pPIDASrc)
{
memcpy(pPIDADest, pPIDASrc, sizeof(PROGIDACTION));
return S_OK;
}
HRESULT CFTAdvDlg::_GetPROGIDACTION(LPTSTR pszActionFN, PROGIDACTION** ppPIDA)
{
HRESULT hres = E_FAIL;
*ppPIDA = NULL;
if (pszActionFN && *pszActionFN)
{
int n = DPA_GetPtrCount(_hdpaActions);
for (int i = 0; (i < n) && FAILED(hres); ++i)
{
*ppPIDA = (PROGIDACTION*)DPA_GetPtr(_hdpaActions, i);
if (!StrCmpN((*ppPIDA)->szAction, pszActionFN, ARRAYSIZE((*ppPIDA)->szAction)))
hres = S_OK;
}
}
if (FAILED(hres))
*ppPIDA = NULL;
return hres;
}
HRESULT CFTAdvDlg::_AppendPROGIDACTION(PROGIDACTION* pPIDA)
{
PROGIDACTION* pPIDANew = NULL;
HRESULT hres = _CreatePROGIDACTION(&pPIDANew);
if (SUCCEEDED(hres))
{
_CopyPROGIDACTION(pPIDANew, pPIDA);
if (-1 != DPA_AppendPtr(_hdpaActions, pPIDANew))
{
hres = S_OK;
}
else
{
_DeletePROGIDACTION(pPIDANew);
hres = E_OUTOFMEMORY;
}
}
if (E_OUTOFMEMORY == hres)
{
//Out of memory
ShellMessageBox(g_hinst, _hwnd, MAKEINTRESOURCE(IDS_ERROR +
ERROR_NOT_ENOUGH_MEMORY), MAKEINTRESOURCE(IDS_FT),
MB_OK | MB_ICONSTOP);
}
return hres;
}
BOOL CFTAdvDlg::_IsNewPROGIDACTION(LPTSTR pszActionFN)
{
BOOL fRet = FALSE;
PROGIDACTION* pPIDA = NULL;
HRESULT hres = _GetPROGIDACTION(pszActionFN, &pPIDA);
if (SUCCEEDED(hres))
if (pPIDA->fNew)
fRet = TRUE;
return fRet;
}
HRESULT CFTAdvDlg::_FillPROGIDACTION(PROGIDACTION* pPIDA, LPTSTR pszActionReg,
LPTSTR pszActionFN)
{
PROGIDACTION* pPIDAList = NULL;
HRESULT hres = _GetPROGIDACTION(pszActionFN, &pPIDAList);
if (SUCCEEDED(hres))
{
_CopyPROGIDACTION(pPIDA, pPIDAList);
}
else
{
IAssocInfo* pAI = NULL;
hres = _pAssocStore->GetComplexAssocInfo(_szProgID, AIINIT_PROGID,
pszActionReg, AIINIT_ACTION, &pAI);
if (SUCCEEDED(hres))
{
DWORD cbPIDA = sizeof(*pPIDA);
hres = pAI->GetData(AIDATA_PROGIDACTION, (PBYTE)pPIDA, &cbPIDA);
pAI->Release();
}
}
return hres;
}
///////////////////////////////////////////////////////////////////////////////
// Windows boiler plate code
LRESULT CFTAdvDlg::OnCommand(WPARAM wParam, LPARAM lParam)
{
LRESULT lRes = FALSE;
switch(GET_WM_COMMAND_ID(wParam, lParam))
{
case IDC_FT_EDIT_NEW:
lRes = OnNewButton(GET_WM_COMMAND_CMD(wParam, lParam));
break;
case IDC_FT_EDIT_REMOVE:
lRes = OnRemoveButton(GET_WM_COMMAND_CMD(wParam, lParam));
break;
case IDC_FT_EDIT_EDIT:
lRes = OnEditButton(GET_WM_COMMAND_CMD(wParam, lParam));
break;
case IDC_FT_EDIT_CHANGEICON:
lRes = OnChangeIcon(GET_WM_COMMAND_CMD(wParam, lParam));
break;
case IDC_FT_EDIT_DEFAULT:
lRes = OnSetDefault(GET_WM_COMMAND_CMD(wParam, lParam));
break;
default:
lRes = CFTDlg::OnCommand(wParam, lParam);
break;
}
return lRes;
}
LRESULT CFTAdvDlg::OnNotify(WPARAM wParam, LPARAM lParam)
{
LRESULT lRes = FALSE;
LPNMHDR pNMHDR = (LPNMHDR)lParam;
UINT_PTR idFrom = pNMHDR->idFrom;
UINT uCode = pNMHDR->code;
//GET_WM_COMMAND_CMD
switch(idFrom)
{
case IDC_FT_EDIT_LV_CMDS:
OnNotifyListView(uCode, pNMHDR);
lRes = CFTDlg::OnNotify(wParam, lParam);
break;
default:
lRes = CFTDlg::OnNotify(wParam, lParam);
break;
}
return lRes;
}
LRESULT CFTAdvDlg::WndProc(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
LRESULT lRes = FALSE;
switch(uMsg)
{
case WM_DRAWITEM:
lRes = OnDrawItem(wParam, lParam);
break;
case WM_MEASUREITEM:
lRes = OnMeasureItem(wParam, lParam);
break;
default:
lRes = CFTDlg::WndProc(uMsg, wParam, lParam);
break;
}
return lRes;
}