xref: /trunk/main/fpicker/source/win32/filepicker/getfilenamewrapper.cxx (revision 772835b774ba4c34b8de932fc3929f72593722f7)
1b557fc96SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3b557fc96SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4b557fc96SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5b557fc96SAndrew Rist  * distributed with this work for additional information
6b557fc96SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7b557fc96SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8b557fc96SAndrew Rist  * "License"); you may not use this file except in compliance
9b557fc96SAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
11b557fc96SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
13b557fc96SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14b557fc96SAndrew Rist  * software distributed under the License is distributed on an
15b557fc96SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16b557fc96SAndrew Rist  * KIND, either express or implied.  See the License for the
17b557fc96SAndrew Rist  * specific language governing permissions and limitations
18b557fc96SAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
20b557fc96SAndrew Rist  *************************************************************/
21b557fc96SAndrew Rist 
22b557fc96SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_fpicker.hxx"
26cdf0e10cSrcweir 
27cdf0e10cSrcweir //------------------------------------------------------------------------
28cdf0e10cSrcweir // includes
29cdf0e10cSrcweir //------------------------------------------------------------------------
30cdf0e10cSrcweir 
31cdf0e10cSrcweir #include <stdio.h>
32cdf0e10cSrcweir #include <osl/diagnose.h>
33cdf0e10cSrcweir #include "getfilenamewrapper.hxx"
34cdf0e10cSrcweir 
35cdf0e10cSrcweir #if defined _MSC_VER
36cdf0e10cSrcweir #pragma warning(push, 1)
37cdf0e10cSrcweir #endif
38cdf0e10cSrcweir #include <objbase.h>
39cdf0e10cSrcweir #include <process.h>
40cdf0e10cSrcweir #if defined _MSC_VER
41cdf0e10cSrcweir #pragma warning(pop)
42cdf0e10cSrcweir #endif
43cdf0e10cSrcweir 
44cdf0e10cSrcweir namespace /* private */
45cdf0e10cSrcweir {
46cdf0e10cSrcweir 
47cdf0e10cSrcweir     //-----------------------------------------------
48cdf0e10cSrcweir     // This class prevents changing of the working
49cdf0e10cSrcweir     // directory.
50cdf0e10cSrcweir     //-----------------------------------------------
51cdf0e10cSrcweir     class CurDirGuard
52cdf0e10cSrcweir     {
53cdf0e10cSrcweir         sal_Bool m_bValid;
54cdf0e10cSrcweir         wchar_t* m_pBuffer;
55cdf0e10cSrcweir         DWORD m_nBufLen;
56cdf0e10cSrcweir 
57cdf0e10cSrcweir     public:
CurDirGuard()58cdf0e10cSrcweir         CurDirGuard()
59cdf0e10cSrcweir         : m_bValid( sal_False )
60cdf0e10cSrcweir         , m_pBuffer( NULL )
61cdf0e10cSrcweir         , m_nBufLen( 0 )
62cdf0e10cSrcweir         {
63cdf0e10cSrcweir             m_nBufLen = GetCurrentDirectoryW( 0, NULL );
64cdf0e10cSrcweir             if ( m_nBufLen )
65cdf0e10cSrcweir             {
66cdf0e10cSrcweir                 m_pBuffer = new wchar_t[m_nBufLen];
67cdf0e10cSrcweir                 m_bValid = ( GetCurrentDirectoryW( m_nBufLen, m_pBuffer ) == ( m_nBufLen - 1 ) );
68cdf0e10cSrcweir             }
69cdf0e10cSrcweir         }
70cdf0e10cSrcweir 
~CurDirGuard()71cdf0e10cSrcweir         ~CurDirGuard()
72cdf0e10cSrcweir         {
73cdf0e10cSrcweir             bool bDirSet = false;
74cdf0e10cSrcweir 
75cdf0e10cSrcweir             if ( m_pBuffer )
76cdf0e10cSrcweir             {
77cdf0e10cSrcweir                 if ( m_bValid )
78cdf0e10cSrcweir                 {
79cdf0e10cSrcweir                     if ( m_nBufLen - 1 > MAX_PATH )
80cdf0e10cSrcweir                     {
81cdf0e10cSrcweir                         if ( (LONG32)GetVersion() < 0 )
82cdf0e10cSrcweir                         {
83cdf0e10cSrcweir                             // this is Win 98/ME branch, such a long path can not be set
84cdf0e10cSrcweir                             // use the system path as fallback later
85cdf0e10cSrcweir                         }
86cdf0e10cSrcweir                         else
87cdf0e10cSrcweir                         {
88cdf0e10cSrcweir                             DWORD nNewLen = m_nBufLen + 8;
89cdf0e10cSrcweir                             wchar_t* pNewBuffer = new wchar_t[nNewLen];
90cdf0e10cSrcweir                             if ( m_nBufLen > 3 && m_pBuffer[0] == (wchar_t)'\\' && m_pBuffer[1] == (wchar_t)'\\' )
91cdf0e10cSrcweir                             {
92cdf0e10cSrcweir                                 if ( m_pBuffer[2] == (wchar_t)'?' )
93cdf0e10cSrcweir                                     _snwprintf( pNewBuffer, nNewLen, L"%s", m_pBuffer );
94cdf0e10cSrcweir                                 else
95cdf0e10cSrcweir                                     _snwprintf( pNewBuffer, nNewLen, L"\\\\?\\UNC\\%s", m_pBuffer+2 );
96cdf0e10cSrcweir                             }
97cdf0e10cSrcweir                             else
98cdf0e10cSrcweir                                 _snwprintf( pNewBuffer, nNewLen, L"\\\\?\\%s", m_pBuffer );
99cdf0e10cSrcweir                             bDirSet = SetCurrentDirectoryW( pNewBuffer );
100cdf0e10cSrcweir 
101cdf0e10cSrcweir                             delete [] pNewBuffer;
102cdf0e10cSrcweir                         }
103cdf0e10cSrcweir                     }
104cdf0e10cSrcweir                     else
105cdf0e10cSrcweir                         bDirSet = SetCurrentDirectoryW( m_pBuffer );
106cdf0e10cSrcweir                 }
107cdf0e10cSrcweir 
108cdf0e10cSrcweir                 delete [] m_pBuffer;
109cdf0e10cSrcweir                 m_pBuffer = NULL;
110cdf0e10cSrcweir             }
111cdf0e10cSrcweir 
112cdf0e10cSrcweir             if ( !bDirSet )
113cdf0e10cSrcweir             {
114cdf0e10cSrcweir                 // the fallback solution
115cdf0e10cSrcweir                 wchar_t pPath[MAX_PATH+1];
116cdf0e10cSrcweir                 if ( GetWindowsDirectoryW( pPath, MAX_PATH+1 ) <= MAX_PATH )
117cdf0e10cSrcweir                 {
118cdf0e10cSrcweir                     SetCurrentDirectoryW( pPath );
119cdf0e10cSrcweir                 }
120cdf0e10cSrcweir                 else
121cdf0e10cSrcweir                 {
122cdf0e10cSrcweir                     // the system path is also too long?!!
123cdf0e10cSrcweir                 }
124cdf0e10cSrcweir             }
125cdf0e10cSrcweir         }
126cdf0e10cSrcweir     };
127cdf0e10cSrcweir 
128cdf0e10cSrcweir     //-----------------------------------------------
129cdf0e10cSrcweir     //
130cdf0e10cSrcweir     //-----------------------------------------------
131cdf0e10cSrcweir 
132cdf0e10cSrcweir     struct GetFileNameParam
133cdf0e10cSrcweir     {
GetFileNameParam__anon9817a6fb0111::GetFileNameParam134cdf0e10cSrcweir         GetFileNameParam(bool bOpen, LPOPENFILENAME lpofn) :
135cdf0e10cSrcweir             m_bOpen(bOpen),
136cdf0e10cSrcweir             m_lpofn(lpofn),
137cdf0e10cSrcweir             m_bRet(false),
138cdf0e10cSrcweir             m_ExtErr(0)
139cdf0e10cSrcweir         {}
140cdf0e10cSrcweir 
141cdf0e10cSrcweir         bool            m_bOpen;
142cdf0e10cSrcweir         LPOPENFILENAME  m_lpofn;
143cdf0e10cSrcweir         bool            m_bRet;
144cdf0e10cSrcweir         int             m_ExtErr;
145cdf0e10cSrcweir     };
146cdf0e10cSrcweir 
147cdf0e10cSrcweir     //-----------------------------------------------
148cdf0e10cSrcweir     //
149cdf0e10cSrcweir     //-----------------------------------------------
150cdf0e10cSrcweir 
ThreadProc(void * pParam)151cdf0e10cSrcweir     unsigned __stdcall ThreadProc(void* pParam)
152cdf0e10cSrcweir     {
153cdf0e10cSrcweir         CurDirGuard aGuard;
154cdf0e10cSrcweir 
155cdf0e10cSrcweir         GetFileNameParam* lpgfnp =
156cdf0e10cSrcweir             reinterpret_cast<GetFileNameParam*>(pParam);
157cdf0e10cSrcweir 
158cdf0e10cSrcweir         HRESULT hr = OleInitialize( NULL );
159cdf0e10cSrcweir 
160cdf0e10cSrcweir         if (lpgfnp->m_bOpen)
161cdf0e10cSrcweir             lpgfnp->m_bRet = GetOpenFileName(lpgfnp->m_lpofn);
162cdf0e10cSrcweir         else
163cdf0e10cSrcweir             lpgfnp->m_bRet = GetSaveFileName(lpgfnp->m_lpofn);
164cdf0e10cSrcweir 
165cdf0e10cSrcweir         lpgfnp->m_ExtErr = CommDlgExtendedError();
166cdf0e10cSrcweir 
167cdf0e10cSrcweir         if ( SUCCEEDED( hr ) )
168cdf0e10cSrcweir             OleUninitialize();
169cdf0e10cSrcweir 
170cdf0e10cSrcweir         return 0;
171cdf0e10cSrcweir     }
172cdf0e10cSrcweir 
173cdf0e10cSrcweir     //-----------------------------------------------
174*772835b7SJohn Bampton     // executes GetOpenFileName/GetSaveFileName in
175*772835b7SJohn Bampton     // a separate thread
176cdf0e10cSrcweir     //-----------------------------------------------
177cdf0e10cSrcweir 
ThreadExecGetFileName(LPOPENFILENAME lpofn,bool bOpen,int & ExtErr)178cdf0e10cSrcweir     bool ThreadExecGetFileName(LPOPENFILENAME lpofn, bool bOpen, /*out*/ int& ExtErr)
179cdf0e10cSrcweir     {
180cdf0e10cSrcweir         GetFileNameParam gfnp(bOpen,lpofn);
181cdf0e10cSrcweir         unsigned         id;
182cdf0e10cSrcweir 
183cdf0e10cSrcweir         HANDLE hThread = reinterpret_cast<HANDLE>(
184cdf0e10cSrcweir             _beginthreadex(0, 0, ThreadProc, &gfnp, 0, &id));
185cdf0e10cSrcweir 
186cdf0e10cSrcweir         OSL_POSTCOND(hThread, "could not create STA thread");
187cdf0e10cSrcweir 
188cdf0e10cSrcweir         WaitForSingleObject(hThread, INFINITE);
189cdf0e10cSrcweir         CloseHandle(hThread);
190cdf0e10cSrcweir 
191cdf0e10cSrcweir         ExtErr = gfnp.m_ExtErr;
192cdf0e10cSrcweir 
193cdf0e10cSrcweir         return gfnp.m_bRet;
194cdf0e10cSrcweir     }
195cdf0e10cSrcweir 
196cdf0e10cSrcweir     //-----------------------------------------------
197cdf0e10cSrcweir     // This function returns true if the calling
198cdf0e10cSrcweir     // thread belongs to a Multithreaded Appartment
199cdf0e10cSrcweir     // (MTA)
200cdf0e10cSrcweir     //-----------------------------------------------
201cdf0e10cSrcweir 
IsMTA()202cdf0e10cSrcweir     bool IsMTA()
203cdf0e10cSrcweir     {
204cdf0e10cSrcweir         HRESULT hr = CoInitialize(NULL);
205cdf0e10cSrcweir 
206cdf0e10cSrcweir         if (RPC_E_CHANGED_MODE == hr)
207cdf0e10cSrcweir             return true;
208cdf0e10cSrcweir 
209cdf0e10cSrcweir         if(SUCCEEDED(hr))
210cdf0e10cSrcweir             CoUninitialize();
211cdf0e10cSrcweir 
212cdf0e10cSrcweir         return false;
213cdf0e10cSrcweir     }
214cdf0e10cSrcweir 
215cdf0e10cSrcweir } // namespace private
216cdf0e10cSrcweir 
217cdf0e10cSrcweir 
218cdf0e10cSrcweir //-----------------------------------------------
219cdf0e10cSrcweir //
220cdf0e10cSrcweir //-----------------------------------------------
221cdf0e10cSrcweir 
CGetFileNameWrapper()222cdf0e10cSrcweir CGetFileNameWrapper::CGetFileNameWrapper() :
223cdf0e10cSrcweir     m_ExtendedDialogError(0)
224cdf0e10cSrcweir {
225cdf0e10cSrcweir }
226cdf0e10cSrcweir 
227cdf0e10cSrcweir //-----------------------------------------------
228cdf0e10cSrcweir //
229cdf0e10cSrcweir //-----------------------------------------------
230cdf0e10cSrcweir 
getOpenFileName(LPOPENFILENAME lpofn)231cdf0e10cSrcweir bool CGetFileNameWrapper::getOpenFileName(LPOPENFILENAME lpofn)
232cdf0e10cSrcweir {
233cdf0e10cSrcweir     OSL_PRECOND(lpofn,"invalid parameter");
234cdf0e10cSrcweir 
235cdf0e10cSrcweir     bool bRet = false;
236cdf0e10cSrcweir 
237cdf0e10cSrcweir     if (IsMTA())
238cdf0e10cSrcweir     {
239cdf0e10cSrcweir         bRet = ThreadExecGetFileName(
240cdf0e10cSrcweir             lpofn, true, m_ExtendedDialogError);
241cdf0e10cSrcweir     }
242cdf0e10cSrcweir     else
243cdf0e10cSrcweir     {
244cdf0e10cSrcweir         CurDirGuard aGuard;
245cdf0e10cSrcweir 
246cdf0e10cSrcweir         HRESULT hr = OleInitialize( NULL );
247cdf0e10cSrcweir 
248cdf0e10cSrcweir         bRet = GetOpenFileName(lpofn);
249cdf0e10cSrcweir         m_ExtendedDialogError = CommDlgExtendedError();
250cdf0e10cSrcweir 
251cdf0e10cSrcweir         if ( SUCCEEDED( hr ) )
252cdf0e10cSrcweir             OleUninitialize();
253cdf0e10cSrcweir     }
254cdf0e10cSrcweir 
255cdf0e10cSrcweir     return bRet;
256cdf0e10cSrcweir }
257cdf0e10cSrcweir 
258cdf0e10cSrcweir //-----------------------------------------------
259cdf0e10cSrcweir //
260cdf0e10cSrcweir //-----------------------------------------------
261cdf0e10cSrcweir 
getSaveFileName(LPOPENFILENAME lpofn)262cdf0e10cSrcweir bool CGetFileNameWrapper::getSaveFileName(LPOPENFILENAME lpofn)
263cdf0e10cSrcweir {
264cdf0e10cSrcweir     OSL_PRECOND(lpofn,"invalid parameter");
265cdf0e10cSrcweir 
266cdf0e10cSrcweir     bool bRet = false;
267cdf0e10cSrcweir 
268cdf0e10cSrcweir     if (IsMTA())
269cdf0e10cSrcweir     {
270cdf0e10cSrcweir         bRet = ThreadExecGetFileName(
271cdf0e10cSrcweir             lpofn, false, m_ExtendedDialogError);
272cdf0e10cSrcweir     }
273cdf0e10cSrcweir     else
274cdf0e10cSrcweir     {
275cdf0e10cSrcweir         CurDirGuard aGuard;
276cdf0e10cSrcweir 
277cdf0e10cSrcweir         bRet = GetSaveFileName(lpofn);
278cdf0e10cSrcweir         m_ExtendedDialogError = CommDlgExtendedError();
279cdf0e10cSrcweir     }
280cdf0e10cSrcweir 
281cdf0e10cSrcweir     return bRet;
282cdf0e10cSrcweir }
283cdf0e10cSrcweir 
284cdf0e10cSrcweir //-----------------------------------------------
285cdf0e10cSrcweir //
286cdf0e10cSrcweir //-----------------------------------------------
287cdf0e10cSrcweir 
commDlgExtendedError()288cdf0e10cSrcweir int CGetFileNameWrapper::commDlgExtendedError( )
289cdf0e10cSrcweir {
290cdf0e10cSrcweir     return m_ExtendedDialogError;
291cdf0e10cSrcweir }
292