Change-Id: Ib19175affd48610fc164a996d609b369256d4f72 Reviewed-on: https://gerrit.libreoffice.org/29865 Reviewed-by: Stephan Bergmann <sbergman@redhat.com> Tested-by: Stephan Bergmann <sbergman@redhat.com>
849 lines
26 KiB
C++
849 lines
26 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include "shared.hxx"
|
|
#include "WinFileOpenImpl.hxx"
|
|
#include <osl/diagnose.h>
|
|
#include <osl/file.hxx>
|
|
#include <com/sun/star/ui/dialogs/ExtendedFilePickerElementIds.hpp>
|
|
#include <com/sun/star/ui/dialogs/FilePickerEvent.hpp>
|
|
#include <com/sun/star/ui/dialogs/ExecutableDialogResults.hpp>
|
|
#include <com/sun/star/ui/dialogs/CommonFilePickerElementIds.hpp>
|
|
#include <com/sun/star/ui/dialogs/FilePreviewImageFormats.hpp>
|
|
#include <com/sun/star/ui/dialogs/ListboxControlActions.hpp>
|
|
#include "../misc/WinImplHelper.hxx"
|
|
|
|
#include "FilePicker.hxx"
|
|
#include "controlaccess.hxx"
|
|
#include "customcontrolfactory.hxx"
|
|
#include <rtl/ustrbuf.hxx>
|
|
#include <rtl/string.hxx>
|
|
#include <osl/thread.hxx>
|
|
#include "filepickerstate.hxx"
|
|
|
|
using namespace com::sun::star;
|
|
|
|
using com::sun::star::ui::dialogs::FilePickerEvent;
|
|
using com::sun::star::lang::IllegalArgumentException;
|
|
using com::sun::star::ui::dialogs::XFilePicker2;
|
|
|
|
using namespace ::com::sun::star::ui::dialogs::ExtendedFilePickerElementIds;
|
|
using namespace ::com::sun::star::ui::dialogs::CommonFilePickerElementIds;
|
|
using namespace ::com::sun::star::ui::dialogs::ListboxControlActions;
|
|
|
|
// to distinguish what to do in the enum child window callback function
|
|
|
|
enum ECW_ACTION_T
|
|
{
|
|
INIT_CUSTOM_CONTROLS,
|
|
CACHE_CONTROL_VALUES
|
|
};
|
|
|
|
struct EnumParam
|
|
{
|
|
ECW_ACTION_T m_action;
|
|
CWinFileOpenImpl* m_instance;
|
|
|
|
EnumParam( ECW_ACTION_T action, CWinFileOpenImpl* instance ):
|
|
m_action( action ),
|
|
m_instance( instance )
|
|
{}
|
|
};
|
|
|
|
CWinFileOpenImpl::CWinFileOpenImpl(
|
|
CFilePicker* aFilePicker,
|
|
bool bFileOpenDialog,
|
|
sal_uInt32 dwFlags,
|
|
sal_uInt32 dwTemplateId,
|
|
HINSTANCE hInstance) :
|
|
CFileOpenDialog(bFileOpenDialog, dwFlags, dwTemplateId, hInstance),
|
|
m_filterContainer(new CFilterContainer()),
|
|
m_Preview(new CPreviewAdapter(hInstance)),
|
|
m_CustomControls(CCustomControlFactory::CreateCustomControlContainer()),
|
|
m_FilePicker(aFilePicker),
|
|
m_bInitialSelChanged(true),
|
|
m_HelpPopupWindow(hInstance, m_hwndFileOpenDlg),
|
|
m_ExecuteFilePickerState(new CExecuteFilePickerState()),
|
|
m_NonExecuteFilePickerState(new CNonExecuteFilePickerState())
|
|
{
|
|
m_FilePickerState = m_NonExecuteFilePickerState;
|
|
}
|
|
|
|
CWinFileOpenImpl::~CWinFileOpenImpl()
|
|
{
|
|
delete m_ExecuteFilePickerState;
|
|
delete m_NonExecuteFilePickerState;
|
|
}
|
|
|
|
// we expect the directory in URL format
|
|
|
|
void CWinFileOpenImpl::setDisplayDirectory(const OUString& aDirectory)
|
|
throw( IllegalArgumentException, uno::RuntimeException )
|
|
{
|
|
OUString aSysDirectory;
|
|
if( aDirectory.getLength() > 0)
|
|
{
|
|
if ( ::osl::FileBase::E_None !=
|
|
::osl::FileBase::getSystemPathFromFileURL(aDirectory,aSysDirectory))
|
|
throw IllegalArgumentException(
|
|
"Invalid directory",
|
|
static_cast<XFilePicker2*>(m_FilePicker), 1);
|
|
|
|
// we ensure that there is a trailing '/' at the end of
|
|
// he given file url, because the windows functions only
|
|
// works correctly when providing "c:\" or an environment
|
|
// variable like "=c:=c:\.." etc. is set, else the
|
|
// FolderPicker would stand in the root of the shell
|
|
// hierarchy which is the desktop folder
|
|
if ( aSysDirectory.lastIndexOf(BACKSLASH) != (aSysDirectory.getLength() - 1))
|
|
aSysDirectory += BACKSLASH;
|
|
}
|
|
|
|
// call base class method
|
|
CFileOpenDialog::setDisplayDirectory(aSysDirectory);
|
|
}
|
|
|
|
// we return the directory in URL format
|
|
|
|
OUString CWinFileOpenImpl::getDisplayDirectory() throw(uno::RuntimeException)
|
|
{
|
|
return m_FilePickerState->getDisplayDirectory(this);
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::setDefaultName(const OUString& aName)
|
|
throw( IllegalArgumentException, uno::RuntimeException )
|
|
{
|
|
// we don't set the default name directly
|
|
// because this influences how the file open
|
|
// dialog sets the initial path when it is about
|
|
// to open (see MSDN: OPENFILENAME)
|
|
// so we save the default name which should
|
|
// appear in the file-name-box and set
|
|
// this name when processing onInitDone
|
|
m_defaultName = aName;
|
|
}
|
|
|
|
// return format: URL
|
|
// if multiselection is allowed there are two different cases
|
|
// 1. one file selected: the sequence contains one entry path\filename.ext
|
|
// 2. multiple files selected: the sequence contains multiple entries
|
|
// the first entry is the path url, all other entries are file names
|
|
|
|
uno::Sequence<OUString> SAL_CALL CWinFileOpenImpl::getFiles()
|
|
throw(uno::RuntimeException)
|
|
{
|
|
return m_FilePickerState->getFiles(this);
|
|
}
|
|
|
|
// shows the FileOpen/FileSave dialog
|
|
|
|
sal_Int16 SAL_CALL CWinFileOpenImpl::execute( ) throw(uno::RuntimeException)
|
|
{
|
|
sal_Int16 rc = CFileOpenDialog::doModal();
|
|
|
|
if (1 == rc)
|
|
rc = css::ui::dialogs::ExecutableDialogResults::OK;
|
|
else if (0 == rc)
|
|
rc = css::ui::dialogs::ExecutableDialogResults::CANCEL;
|
|
else
|
|
throw uno::RuntimeException(
|
|
"Error executing dialog",
|
|
static_cast<XFilePicker2*>(m_FilePicker));
|
|
|
|
return rc;
|
|
}
|
|
|
|
// appends a new filter
|
|
// returns false if the title (aTitle) was already added or the title or the filter are
|
|
// empty
|
|
|
|
void SAL_CALL CWinFileOpenImpl::appendFilter(const OUString& aTitle, const OUString& aFilter)
|
|
throw(IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
bool bRet = m_filterContainer->addFilter(aTitle, aFilter);
|
|
|
|
if (!bRet)
|
|
throw IllegalArgumentException(
|
|
"filter already exists",
|
|
static_cast<XFilePicker2*>(m_FilePicker), 1);
|
|
|
|
// #95345# see MSDN OPENFILENAME
|
|
// If nFilterIndex is zero and lpstrCustomFilter is NULL,
|
|
// the system uses the first filter in the lpstrFilter buffer.
|
|
// to reflect this we must set the filter index so that calls
|
|
// to getSelectedFilterIndex without explicitly calling
|
|
// setFilterIndex before does not return 0 which leads to a
|
|
// false state
|
|
if (0 == getSelectedFilterIndex())
|
|
CFileOpenDialog::setFilterIndex(1);
|
|
}
|
|
|
|
// sets a current filter
|
|
|
|
void SAL_CALL CWinFileOpenImpl::setCurrentFilter(const OUString& aTitle)
|
|
throw( IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
sal_Int32 filterPos = m_filterContainer->getFilterPos(aTitle);
|
|
|
|
if (filterPos < 0)
|
|
throw IllegalArgumentException(
|
|
"filter doesn't exist",
|
|
static_cast<XFilePicker2*>(m_FilePicker), 1);
|
|
|
|
// filter index of the base class starts with 1
|
|
CFileOpenDialog::setFilterIndex(filterPos + 1);
|
|
}
|
|
|
|
// returns the currently selected filter
|
|
|
|
OUString SAL_CALL CWinFileOpenImpl::getCurrentFilter() throw(uno::RuntimeException)
|
|
{
|
|
sal_uInt32 nIndex = getSelectedFilterIndex();
|
|
|
|
OUString currentFilter;
|
|
if (nIndex > 0)
|
|
{
|
|
// filter index of the base class starts with 1
|
|
if (!m_filterContainer->getFilter(nIndex - 1, currentFilter)) {
|
|
OSL_ASSERT(false);
|
|
}
|
|
}
|
|
|
|
return currentFilter;
|
|
}
|
|
|
|
inline void SAL_CALL CWinFileOpenImpl::appendFilterGroupSeparator()
|
|
{
|
|
m_filterContainer->addFilter(FILTER_SEPARATOR, ALL_FILES_WILDCARD, ALLOW_DUPLICATES);
|
|
}
|
|
|
|
// XFilterGroupManager
|
|
|
|
void SAL_CALL CWinFileOpenImpl::appendFilterGroup(const OUString& sGroupTitle, const uno::Sequence<beans::StringPair>& aFilters)
|
|
throw (IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
(void) sGroupTitle; // avoid warning
|
|
OSL_ENSURE(0 == sGroupTitle.getLength(), "appendFilterGroup: Parameter 'GroupTitle' currently ignored");
|
|
|
|
sal_Int32 nFilters = aFilters.getLength();
|
|
|
|
OSL_PRECOND(nFilters > 0, "Empty filter list");
|
|
|
|
if (nFilters > 0)
|
|
{
|
|
// append a separator before the next group if
|
|
// there is already a group of filters
|
|
if (m_filterContainer->numFilter() > 0)
|
|
appendFilterGroupSeparator();
|
|
|
|
for (int i = 0; i < nFilters; i++)
|
|
appendFilter(aFilters[i].First, aFilters[i].Second);
|
|
}
|
|
}
|
|
|
|
// XExtendedFilePicker
|
|
|
|
// #i90917: Due to a different feature set for the system-dependent file pickers
|
|
// it's possible that generic code (e.g. sfx2) provides control ids
|
|
// (see ExtendedFilePickerElementIds::LISTBOX_FILTER_SELECTOR) which are NOT
|
|
// available on all platforms. This filter function should filter out control ids
|
|
// which are only available on KDE/GTK file pickers.
|
|
static bool filterControlCommand( sal_Int16 nControlId )
|
|
{
|
|
if ( nControlId == LISTBOX_FILTER_SELECTOR )
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::setValue(sal_Int16 aControlId, sal_Int16 aControlAction, const uno::Any& aValue)
|
|
throw(uno::RuntimeException)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState);
|
|
if ( !filterControlCommand( aControlId ))
|
|
m_FilePickerState->setValue(aControlId, aControlAction, aValue);
|
|
}
|
|
|
|
// returns the value of an custom template element
|
|
// we assume that there are only checkboxes or comboboxes
|
|
|
|
uno::Any SAL_CALL CWinFileOpenImpl::getValue(sal_Int16 aControlId, sal_Int16 aControlAction)
|
|
throw(uno::RuntimeException)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState);
|
|
if ( !filterControlCommand( aControlId ))
|
|
return m_FilePickerState->getValue(aControlId, aControlAction);
|
|
else
|
|
return uno::Any();
|
|
}
|
|
|
|
// enables a custom template element
|
|
|
|
void SAL_CALL CWinFileOpenImpl::enableControl(sal_Int16 ControlID, bool bEnable)
|
|
throw(uno::RuntimeException)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState);
|
|
if ( !filterControlCommand( ControlID ))
|
|
m_FilePickerState->enableControl(ControlID, bEnable);
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::setLabel( sal_Int16 aControlId, const OUString& aLabel )
|
|
throw (uno::RuntimeException)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState);
|
|
if ( !filterControlCommand( aControlId ))
|
|
m_FilePickerState->setLabel(aControlId, aLabel);
|
|
}
|
|
|
|
OUString SAL_CALL CWinFileOpenImpl::getLabel( sal_Int16 aControlId )
|
|
throw (uno::RuntimeException)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState);
|
|
if ( !filterControlCommand( aControlId ))
|
|
return m_FilePickerState->getLabel(aControlId);
|
|
else
|
|
return OUString();
|
|
}
|
|
|
|
uno::Sequence<sal_Int16> SAL_CALL CWinFileOpenImpl::getSupportedImageFormats()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return CPreviewAdapter::getSupportedImageFormats();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL CWinFileOpenImpl::getTargetColorDepth()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return m_Preview->getTargetColorDepth();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL CWinFileOpenImpl::getAvailableWidth()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return m_Preview->getAvailableWidth();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL CWinFileOpenImpl::getAvailableHeight()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return m_Preview->getAvailableHeight();
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::setImage(sal_Int16 aImageFormat, const uno::Any& aImage)
|
|
throw (IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
m_Preview->setImage(aImageFormat,aImage);
|
|
}
|
|
|
|
bool SAL_CALL CWinFileOpenImpl::setShowState(bool bShowState)
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return m_Preview->setShowState(bShowState);
|
|
}
|
|
|
|
bool SAL_CALL CWinFileOpenImpl::getShowState()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return m_Preview->getShowState();
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::cancel()
|
|
{
|
|
if (IsWindow(m_hwndFileOpenDlg))
|
|
{
|
|
// simulate a mouse click to the
|
|
// cancel button
|
|
BOOL const ret = PostMessage(
|
|
m_hwndFileOpenDlg,
|
|
WM_COMMAND,
|
|
MAKEWPARAM(IDCANCEL,BN_CLICKED),
|
|
reinterpret_cast<LPARAM>(GetDlgItem(m_hwndFileOpenDlg, IDCANCEL)));
|
|
SAL_WARN_IF(0 == ret, "fpicker", "ERROR: PostMessage() failed!");
|
|
}
|
|
}
|
|
|
|
// returns the id of a custom template element
|
|
|
|
sal_Int16 SAL_CALL CWinFileOpenImpl::getFocused()
|
|
{
|
|
int nID = GetDlgCtrlID(GetFocus());
|
|
|
|
// we don't forward id's of standard file open
|
|
// dialog elements (ctlFirst is defined in dlgs.h
|
|
// in MS Platform SDK)
|
|
if (nID >= ctlFirst)
|
|
nID = 0;
|
|
|
|
return sal::static_int_cast< sal_Int16 >(nID);
|
|
}
|
|
|
|
inline bool SAL_CALL CWinFileOpenImpl::IsCustomControlHelpRequested(LPHELPINFO lphi)
|
|
{
|
|
return ((lphi->iCtrlId != IDOK) && (lphi->iCtrlId != IDCANCEL) && (lphi->iCtrlId < ctlFirst));
|
|
}
|
|
|
|
// our own DlgProc because we do subclass the dialog
|
|
// we catch the WM_NCDESTROY message in order to erase an entry in our static map
|
|
// if one instance dies
|
|
|
|
LRESULT CALLBACK CWinFileOpenImpl::SubClassFunc(
|
|
HWND hWnd, UINT wMessage, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
unsigned int lResult = 0;
|
|
|
|
CWinFileOpenImpl* pImpl = dynamic_cast<CWinFileOpenImpl*>(getCurrentInstance(hWnd));
|
|
|
|
switch(wMessage)
|
|
{
|
|
case WM_HELP:
|
|
{
|
|
LPHELPINFO lphi = reinterpret_cast<LPHELPINFO>(lParam);
|
|
|
|
if (CWinFileOpenImpl::IsCustomControlHelpRequested(lphi))
|
|
pImpl->onCustomControlHelpRequest(lphi);
|
|
else
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
}
|
|
break;
|
|
|
|
case WM_SIZE:
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
|
|
pImpl->onWMSize();
|
|
break;
|
|
|
|
case WM_WINDOWPOSCHANGED:
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
|
|
pImpl->onWMWindowPosChanged();
|
|
break;
|
|
|
|
case WM_SHOWWINDOW:
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
|
|
pImpl->onWMShow(bool(wParam));
|
|
break;
|
|
|
|
case WM_NCDESTROY:
|
|
// restore the old window proc
|
|
SetWindowLongPtr(hWnd, GWLP_WNDPROC,
|
|
reinterpret_cast<LONG_PTR>(pImpl->m_pfnOldDlgProc));
|
|
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
break;
|
|
|
|
default:
|
|
lResult = CallWindowProc(
|
|
reinterpret_cast<WNDPROC>(pImpl->m_pfnOldDlgProc),
|
|
hWnd,wMessage,wParam,lParam);
|
|
break;
|
|
|
|
} // switch
|
|
|
|
return lResult;
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::InitControlLabel(HWND hWnd)
|
|
{
|
|
|
|
// set the labels for all extended controls
|
|
|
|
|
|
sal_Int16 aCtrlId = sal::static_int_cast< sal_Int16 >(GetDlgCtrlID(hWnd));
|
|
OUString aLabel = m_ResProvider.getResString(aCtrlId);
|
|
if (aLabel.getLength())
|
|
setLabel(aCtrlId, aLabel);
|
|
}
|
|
|
|
// There may be problems with the layout of our custom controls,
|
|
// so that they are not aligned with the standard controls of the
|
|
// FileOpen dialog.
|
|
// We use a simple algorithm to move the custom controls to their
|
|
// proper position and resize them.
|
|
// Our approach is to align all static text controls with the
|
|
// static text control "File name" of the FileOpen dialog,
|
|
// all checkboxes and all list/comboboxes will be left aligned with
|
|
// the standard combobox edt1/cmb13 (defined in MS platform sdk dlgs.h)
|
|
// and all push buttons will be left aligned with the standard
|
|
// "OK" button
|
|
|
|
void SAL_CALL CWinFileOpenImpl::InitCustomControlContainer(HWND hCustomControl)
|
|
{
|
|
m_CustomControls->AddControl(
|
|
CCustomControlFactory::CreateCustomControl(hCustomControl,m_hwndFileOpenDlg));
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::CacheControlState(HWND hWnd)
|
|
{
|
|
OSL_ASSERT(m_FilePickerState && m_NonExecuteFilePickerState);
|
|
m_ExecuteFilePickerState->cacheControlState(hWnd, m_NonExecuteFilePickerState);
|
|
}
|
|
|
|
BOOL CALLBACK CWinFileOpenImpl::EnumChildWndProc(HWND hWnd, LPARAM lParam)
|
|
{
|
|
EnumParam* enumParam = reinterpret_cast<EnumParam*>(lParam);
|
|
CWinFileOpenImpl* pImpl = enumParam->m_instance;
|
|
|
|
OSL_ASSERT(pImpl);
|
|
|
|
bool bRet = true;
|
|
|
|
switch(enumParam->m_action)
|
|
{
|
|
case INIT_CUSTOM_CONTROLS:
|
|
pImpl->InitControlLabel(hWnd);
|
|
pImpl->InitCustomControlContainer(hWnd);
|
|
break;
|
|
|
|
case CACHE_CONTROL_VALUES:
|
|
pImpl->CacheControlState(hWnd);
|
|
break;
|
|
|
|
default:
|
|
// should not end here
|
|
OSL_ASSERT(false);
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
sal_uInt32 SAL_CALL CWinFileOpenImpl::onFileOk()
|
|
{
|
|
m_NonExecuteFilePickerState->reset();
|
|
|
|
EnumParam enumParam(CACHE_CONTROL_VALUES,this);
|
|
|
|
EnumChildWindows(
|
|
m_hwndFileOpenDlgChild,
|
|
CWinFileOpenImpl::EnumChildWndProc,
|
|
reinterpret_cast<LPARAM>(&enumParam));
|
|
|
|
return 0;
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::onSelChanged(HWND)
|
|
{
|
|
// the windows file open dialog sends an initial
|
|
// SelChanged message after the InitDone message
|
|
// when the dialog is about to be opened
|
|
// if the lpstrFile buffer of the OPENFILENAME is
|
|
// empty (zero length string) the windows file open
|
|
// dialog sends a WM_SETTEXT message with an empty
|
|
// string to the file name edit line
|
|
// this would overwritte our text when we would set
|
|
// the default name in onInitDone, so we have to
|
|
// remember that this is the first SelChanged message
|
|
// and set the default name here to overwrite the
|
|
// windows setting
|
|
InitialSetDefaultName();
|
|
|
|
FilePickerEvent evt;
|
|
m_FilePicker->fileSelectionChanged(evt);
|
|
}
|
|
|
|
// #i40865# The size of the standard labels 'File name'
|
|
// and 'File type' is to short in some cases when the
|
|
// label will be changed (e.g. in the Brazil version).
|
|
// We just make sure that the labels are using the maximum
|
|
// available space.
|
|
void CWinFileOpenImpl::EnlargeStdControlLabels() const
|
|
{
|
|
HWND hFilterBoxLabel = GetDlgItem(m_hwndFileOpenDlg, stc2);
|
|
HWND hFileNameBoxLabel = GetDlgItem(m_hwndFileOpenDlg, stc3);
|
|
HWND hFileNameBox = GetDlgItem(m_hwndFileOpenDlg, cmb13);
|
|
if (!hFileNameBox)
|
|
hFileNameBox = GetDlgItem(m_hwndFileOpenDlg, edt1); // since Win2k it is cmb13 or edt1
|
|
|
|
HWND hFilterBox = GetDlgItem(m_hwndFileOpenDlg, cmb1);
|
|
HWND hOkButton = GetDlgItem(m_hwndFileOpenDlg, IDOK);
|
|
|
|
// Move filter and file name box nearer to OK and Cancel button
|
|
RECT rcOkButton;
|
|
GetWindowRect(hOkButton, &rcOkButton);
|
|
|
|
const int MAX_GAP = 10;
|
|
const int OFFSET = 0;
|
|
|
|
RECT rcFileNameBox;
|
|
GetWindowRect(hFileNameBox, &rcFileNameBox);
|
|
int w = rcFileNameBox.right - rcFileNameBox.left;
|
|
int h = rcFileNameBox.bottom - rcFileNameBox.top;
|
|
|
|
int gap = rcOkButton.left - rcFileNameBox.right;
|
|
gap = (gap > MAX_GAP) ? gap - MAX_GAP : gap;
|
|
|
|
ScreenToClient(m_hwndFileOpenDlg, reinterpret_cast<LPPOINT>(&rcFileNameBox));
|
|
MoveWindow(hFileNameBox, rcFileNameBox.left + gap + OFFSET, rcFileNameBox.top, w - OFFSET, h, true);
|
|
|
|
RECT rcFilterBox;
|
|
GetWindowRect(hFilterBox, &rcFilterBox);
|
|
w = rcFilterBox.right - rcFilterBox.left;
|
|
h = rcFilterBox.bottom - rcFilterBox.top;
|
|
ScreenToClient(m_hwndFileOpenDlg, reinterpret_cast<LPPOINT>(&rcFilterBox));
|
|
MoveWindow(hFilterBox, rcFilterBox.left + gap + OFFSET, rcFilterBox.top, w - OFFSET, h, true);
|
|
|
|
// get the new window rect
|
|
GetWindowRect(hFileNameBox, &rcFileNameBox);
|
|
|
|
RECT rcFilterBoxLabel;
|
|
GetWindowRect(hFilterBoxLabel, &rcFilterBoxLabel);
|
|
int offset = rcFileNameBox.left - rcFilterBoxLabel.right - 1;
|
|
|
|
w = rcFilterBoxLabel.right - rcFilterBoxLabel.left + offset;
|
|
h = rcFilterBoxLabel.bottom - rcFilterBoxLabel.top;
|
|
ScreenToClient(m_hwndFileOpenDlg, reinterpret_cast<LPPOINT>(&rcFilterBoxLabel));
|
|
MoveWindow(hFilterBoxLabel, rcFilterBoxLabel.left, rcFilterBoxLabel.top, w, h, true);
|
|
|
|
RECT rcFileNameBoxLabel;
|
|
GetWindowRect(hFileNameBoxLabel, &rcFileNameBoxLabel);
|
|
w = rcFileNameBoxLabel.right - rcFileNameBoxLabel.left + offset;
|
|
h = rcFileNameBoxLabel.bottom - rcFileNameBoxLabel.top;
|
|
ScreenToClient(m_hwndFileOpenDlg, reinterpret_cast<LPPOINT>(&rcFileNameBoxLabel));
|
|
MoveWindow(hFileNameBoxLabel, rcFileNameBoxLabel.left, rcFileNameBoxLabel.top, w, h, true);
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::onInitDone()
|
|
{
|
|
m_Preview->setParent(m_hwndFileOpenDlg);
|
|
|
|
// but now we have a valid parent handle
|
|
m_HelpPopupWindow.setParent(m_hwndFileOpenDlg);
|
|
|
|
EnlargeStdControlLabels();
|
|
|
|
// #99826
|
|
// Set the online filepicker state before initializing
|
|
// the control labels from the resource else we are
|
|
// overriding the offline settings
|
|
m_ExecuteFilePickerState->setHwnd(m_hwndFileOpenDlgChild);
|
|
|
|
m_FilePickerState = m_ExecuteFilePickerState;
|
|
|
|
// initialize controls from cache
|
|
|
|
EnumParam enumParam(INIT_CUSTOM_CONTROLS,this);
|
|
|
|
EnumChildWindows(
|
|
m_hwndFileOpenDlgChild,
|
|
CWinFileOpenImpl::EnumChildWndProc,
|
|
reinterpret_cast<LPARAM>(&enumParam));
|
|
|
|
m_ExecuteFilePickerState->initFilePickerControls(
|
|
m_NonExecuteFilePickerState->getControlCommand());
|
|
|
|
SetDefaultExtension();
|
|
|
|
m_CustomControls->Align();
|
|
|
|
m_CustomControls->SetFont(
|
|
reinterpret_cast<HFONT>(SendMessageW(m_hwndFileOpenDlg, WM_GETFONT, 0, 0)));
|
|
|
|
// resume event notification that was
|
|
// defered in onInitDialog
|
|
m_FilePicker->resumeEventNotification();
|
|
|
|
//#105996 let vcl know that now a system window is active
|
|
BOOL const ret = PostMessage(
|
|
HWND_BROADCAST,
|
|
RegisterWindowMessageW(L"SYSTEM_WINDOW_ACTIVATED"),
|
|
0,
|
|
0);
|
|
SAL_WARN_IF(0 == ret, "fpicker", "ERROR: PostMessage() failed!");
|
|
|
|
// call the parent function to center the
|
|
// dialog to its parent
|
|
CFileOpenDialog::onInitDone();
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::onFolderChanged()
|
|
{
|
|
FilePickerEvent evt;
|
|
m_FilePicker->directoryChanged(evt);
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::onTypeChanged(sal_uInt32)
|
|
{
|
|
SetDefaultExtension();
|
|
|
|
FilePickerEvent evt;
|
|
evt.ElementId = LISTBOX_FILTER;
|
|
m_FilePicker->controlStateChanged(evt);
|
|
}
|
|
|
|
// onMessageCommand handler
|
|
|
|
sal_uInt32 SAL_CALL CWinFileOpenImpl::onCtrlCommand(
|
|
HWND, sal_uInt16 ctrlId, sal_uInt16)
|
|
{
|
|
SetDefaultExtension();
|
|
|
|
if (ctrlId < ctlFirst)
|
|
{
|
|
FilePickerEvent evt;
|
|
evt.ElementId = ctrlId;
|
|
m_FilePicker->controlStateChanged(evt);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void CWinFileOpenImpl::onWMSize()
|
|
{
|
|
m_Preview->notifyParentSizeChanged();
|
|
m_CustomControls->Align();
|
|
m_FilePicker->dialogSizeChanged();
|
|
}
|
|
|
|
void CWinFileOpenImpl::onWMShow(bool bShow)
|
|
{
|
|
m_Preview->notifyParentShow(bShow);
|
|
}
|
|
|
|
void CWinFileOpenImpl::onWMWindowPosChanged()
|
|
{
|
|
m_Preview->notifyParentWindowPosChanged();
|
|
}
|
|
|
|
void CWinFileOpenImpl::onCustomControlHelpRequest(LPHELPINFO lphi)
|
|
{
|
|
FilePickerEvent evt;
|
|
evt.ElementId = sal::static_int_cast< sal_Int16 >(lphi->iCtrlId);
|
|
|
|
OUString aPopupHelpText = m_FilePicker->helpRequested(evt);
|
|
|
|
if (aPopupHelpText.getLength())
|
|
{
|
|
m_HelpPopupWindow.setText(aPopupHelpText);
|
|
|
|
DWORD dwMsgPos = GetMessagePos();
|
|
m_HelpPopupWindow.show(LOWORD(dwMsgPos),HIWORD(dwMsgPos));
|
|
}
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::onInitDialog(HWND hwndDlg)
|
|
{
|
|
// subclass the dialog window
|
|
m_pfnOldDlgProc =
|
|
reinterpret_cast<WNDPROC>(
|
|
SetWindowLongPtr( hwndDlg, GWLP_WNDPROC,
|
|
reinterpret_cast<LONG_PTR>(SubClassFunc)));
|
|
}
|
|
|
|
// processing before showing the dialog
|
|
|
|
bool SAL_CALL CWinFileOpenImpl::preModal()
|
|
{
|
|
CFileOpenDialog::setFilter(
|
|
makeWinFilterBuffer(*m_filterContainer.get()));
|
|
|
|
return true;
|
|
}
|
|
|
|
// processing after showing the dialog
|
|
|
|
void CWinFileOpenImpl::postModal(sal_Int16 nDialogResult)
|
|
{
|
|
CFileOpenDialog::postModal(nDialogResult);
|
|
|
|
// empty the container in order to get rid off
|
|
// invalid controls in case someone calls execute
|
|
// twice in sequence with the same instance
|
|
m_CustomControls->RemoveAllControls();
|
|
|
|
m_FilePickerState = m_NonExecuteFilePickerState;
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::SetDefaultExtension()
|
|
{
|
|
HWND hwndChkSaveWithExt = GetDlgItem(m_hwndFileOpenDlgChild, 100);
|
|
|
|
if (hwndChkSaveWithExt)
|
|
{
|
|
uno::Any aAny = CheckboxGetState(hwndChkSaveWithExt);
|
|
bool bChecked = *static_cast<const sal_Bool*>(aAny.getValue());
|
|
|
|
if (bChecked)
|
|
{
|
|
sal_uInt32 nIndex = getSelectedFilterIndex();
|
|
|
|
OUString currentFilter;
|
|
if (nIndex > 0)
|
|
{
|
|
// filter index of the base class starts with 1
|
|
m_filterContainer->getFilter(nIndex - 1, currentFilter);
|
|
|
|
if (currentFilter.getLength())
|
|
{
|
|
OUString FilterExt;
|
|
m_filterContainer->getFilter(currentFilter, FilterExt);
|
|
|
|
sal_Int32 posOfPoint = FilterExt.indexOf(L'.');
|
|
const sal_Unicode* pFirstExtStart = FilterExt.getStr() + posOfPoint + 1;
|
|
|
|
sal_Int32 posOfSemiColon = FilterExt.indexOf(L';') - 1;
|
|
if (posOfSemiColon < 0)
|
|
posOfSemiColon = FilterExt.getLength() - 1;
|
|
|
|
FilterExt = OUString(pFirstExtStart, posOfSemiColon - posOfPoint);
|
|
|
|
SendMessageW(m_hwndFileOpenDlg, CDM_SETDEFEXT, 0, reinterpret_cast<LPARAM>(FilterExt.getStr()));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SendMessageW(m_hwndFileOpenDlg, CDM_SETDEFEXT, 0, reinterpret_cast<LPARAM>(L""));
|
|
}
|
|
}
|
|
|
|
// !!! HACK !!!
|
|
}
|
|
|
|
void SAL_CALL CWinFileOpenImpl::InitialSetDefaultName()
|
|
{
|
|
// manually setting the file name that appears
|
|
// initially in the file-name-box of the file
|
|
// open dialog (reason: see above setDefaultName)
|
|
if (m_bInitialSelChanged && m_defaultName.getLength())
|
|
{
|
|
// under W2k by default there is a combobox instead
|
|
// of an edit field for the file name edit field
|
|
// the control id of this box is cmb13 and not
|
|
// edt1 as before.
|
|
// However, edt1 is still possible. See fdo#74295
|
|
HWND hFileNameBox = GetDlgItem(m_hwndFileOpenDlg, cmb13);
|
|
if (!hFileNameBox)
|
|
hFileNameBox = GetDlgItem(m_hwndFileOpenDlg, edt1);
|
|
SetWindowTextW(hFileNameBox, reinterpret_cast<PCWSTR>(m_defaultName.getStr()));
|
|
}
|
|
|
|
m_bInitialSelChanged = false;
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|