1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/textentry.cpp 
   3 // Purpose:     wxTextEntry implementation for wxMSW 
   4 // Author:      Vadim Zeitlin 
   7 // Copyright:   (c) 2007 Vadim Zeitlin <vadim@wxwindows.org> 
   8 // Licence:     wxWindows licence 
   9 /////////////////////////////////////////////////////////////////////////////// 
  11 // ============================================================================ 
  13 // ============================================================================ 
  15 // ---------------------------------------------------------------------------- 
  17 // ---------------------------------------------------------------------------- 
  19 // for compilers that support precompilation, includes "wx.h". 
  20 #include "wx/wxprec.h" 
  27     #include "wx/arrstr.h" 
  28     #include "wx/string.h" 
  31 #if wxUSE_TEXTCTRL || wxUSE_COMBOBOX 
  33 #include "wx/textentry.h" 
  34 #include "wx/dynlib.h" 
  36 #include "wx/msw/private.h" 
  38 #define GetEditHwnd() ((HWND)(GetEditHWND())) 
  40 // ---------------------------------------------------------------------------- 
  41 // wxIEnumString implements IEnumString interface 
  42 // ---------------------------------------------------------------------------- 
  44 // standard VC6 SDK (WINVER == 0x0400) does not know about IAutoComplete 
  45 #if wxUSE_OLE && (WINVER >= 0x0500) 
  46     #define HAS_AUTOCOMPLETE 
  49 #ifdef HAS_AUTOCOMPLETE 
  51 #include "wx/msw/ole/oleutils.h" 
  54 #if defined(__MINGW32__) || defined (__WATCOMC__) 
  55     // needed for IID_IAutoComplete, IID_IAutoComplete2 and ACO_AUTOSUGGEST 
  59 #ifndef ACO_UPDOWNKEYDROPSLIST 
  60     #define ACO_UPDOWNKEYDROPSLIST 0x20 
  63 #ifndef SHACF_FILESYS_ONLY 
  64     #define SHACF_FILESYS_ONLY 0x00000010 
  67 DEFINE_GUID(CLSID_AutoComplete
, 
  68     0x00bb2763, 0x6a77, 0x11d0, 0xa5, 0x35, 0x00, 0xc0, 0x4f, 0xd7, 0xd0, 0x62); 
  70 class wxIEnumString 
: public IEnumString
 
  73     wxIEnumString(const wxArrayString
& strings
) : m_strings(strings
) 
  78     DECLARE_IUNKNOWN_METHODS
; 
  80     virtual HRESULT STDMETHODCALLTYPE 
Next(ULONG celt
, 
  84         if ( !rgelt 
|| (!pceltFetched 
&& celt 
> 1) ) 
  87         ULONG pceltFetchedDummy
; 
  89             pceltFetched 
= &pceltFetchedDummy
; 
  93         for ( const unsigned count 
= m_strings
.size(); celt
--; ++m_index 
) 
  95             if ( m_index 
== count 
) 
  98             const wxWX2WCbuf wcbuf 
= m_strings
[m_index
].wc_str(); 
  99             const size_t size 
= (wcslen(wcbuf
) + 1)*sizeof(wchar_t); 
 100             void *olestr 
= CoTaskMemAlloc(size
); 
 102                 return E_OUTOFMEMORY
; 
 104             memcpy(olestr
, wcbuf
, size
); 
 106             *rgelt
++ = wx_static_cast(LPOLESTR
, olestr
); 
 114     virtual HRESULT STDMETHODCALLTYPE 
Skip(ULONG celt
) 
 117         if ( m_index 
> m_strings
.size() ) 
 119             m_index 
= m_strings
.size(); 
 126     virtual HRESULT STDMETHODCALLTYPE 
Reset() 
 133     virtual HRESULT STDMETHODCALLTYPE 
Clone(IEnumString 
**ppEnum
) 
 138         wxIEnumString 
*e 
= new wxIEnumString(m_strings
); 
 139         e
->m_index 
= m_index
; 
 148     // dtor doesn't have to be virtual as we're only ever deleted from our own 
 149     // Release() and are not meant to be derived form anyhow, but making it 
 150     // virtual silences gcc warnings; making it private makes it impossible to 
 151     // (mistakenly) delete us directly instead of calling Release() 
 152     virtual ~wxIEnumString() { } 
 155     const wxArrayString m_strings
; 
 158     DECLARE_NO_COPY_CLASS(wxIEnumString
) 
 161 BEGIN_IID_TABLE(wxIEnumString
) 
 166 IMPLEMENT_IUNKNOWN_METHODS(wxIEnumString
) 
 168 #endif // HAS_AUTOCOMPLETE 
 170 // ============================================================================ 
 171 // wxTextEntry implementation 
 172 // ============================================================================ 
 174 // ---------------------------------------------------------------------------- 
 175 // operations on text 
 176 // ---------------------------------------------------------------------------- 
 178 void wxTextEntry::WriteText(const wxString
& text
) 
 180     ::SendMessage(GetEditHwnd(), EM_REPLACESEL
, 0, (LPARAM
)text
.wx_str()); 
 183 wxString 
wxTextEntry::GetValue() const 
 185     return wxGetWindowText(GetEditHWND()); 
 188 void wxTextEntry::Remove(long from
, long to
) 
 190     DoSetSelection(from
, to
, SetSel_NoScroll
); 
 191     WriteText(wxString()); 
 194 // ---------------------------------------------------------------------------- 
 195 // clipboard operations 
 196 // ---------------------------------------------------------------------------- 
 198 void wxTextEntry::Copy() 
 200     ::SendMessage(GetEditHwnd(), WM_COPY
, 0, 0); 
 203 void wxTextEntry::Cut() 
 205     ::SendMessage(GetEditHwnd(), WM_CUT
, 0, 0); 
 208 void wxTextEntry::Paste() 
 210     ::SendMessage(GetEditHwnd(), WM_PASTE
, 0, 0); 
 213 // ---------------------------------------------------------------------------- 
 215 // ---------------------------------------------------------------------------- 
 217 void wxTextEntry::Undo() 
 219     ::SendMessage(GetEditHwnd(), EM_UNDO
, 0, 0); 
 222 void wxTextEntry::Redo() 
 224     // same as Undo, since Undo undoes the undo 
 229 bool wxTextEntry::CanUndo() const 
 231     return ::SendMessage(GetEditHwnd(), EM_CANUNDO
, 0, 0) != 0; 
 234 bool wxTextEntry::CanRedo() const 
 236     // see comment in Redo() 
 240 // ---------------------------------------------------------------------------- 
 241 // insertion point and selection 
 242 // ---------------------------------------------------------------------------- 
 244 void wxTextEntry::SetInsertionPoint(long pos
) 
 246     // calling DoSetSelection(-1, -1) would select everything which is not what 
 249         pos 
= GetLastPosition(); 
 251     // be careful to call DoSetSelection() which is overridden in wxTextCtrl 
 252     // and not just SetSelection() here 
 253     DoSetSelection(pos
, pos
); 
 256 long wxTextEntry::GetInsertionPoint() const 
 259     GetSelection(&from
, NULL
); 
 263 long wxTextEntry::GetLastPosition() const 
 265     return ::SendMessage(GetEditHwnd(), EM_LINELENGTH
, 0, 0); 
 268 void wxTextEntry::DoSetSelection(long from
, long to
, int WXUNUSED(flags
)) 
 270     // if from and to are both -1, it means (in wxWidgets) that all text should 
 271     // be selected, translate this into Windows convention 
 272     if ( (from 
== -1) && (to 
== -1) ) 
 277     ::SendMessage(GetEditHwnd(), EM_SETSEL
, from
, to
); 
 280 void wxTextEntry::GetSelection(long *from
, long *to
) const 
 282     DWORD dwStart
, dwEnd
; 
 283     ::SendMessage(GetEditHwnd(), EM_GETSEL
, (WPARAM
)&dwStart
, (LPARAM
)&dwEnd
); 
 291 // ---------------------------------------------------------------------------- 
 293 // ---------------------------------------------------------------------------- 
 296 bool wxTextEntry::AutoCompleteFileNames() 
 298 #ifdef HAS_AUTOCOMPLETE 
 299     typedef HRESULT (WINAPI 
*SHAutoComplete_t
)(HWND
, DWORD
); 
 300     static SHAutoComplete_t s_pfnSHAutoComplete 
= (SHAutoComplete_t
)-1; 
 301     static wxDynamicLibrary s_dllShlwapi
; 
 302     if ( s_pfnSHAutoComplete 
== (SHAutoComplete_t
)-1 ) 
 304         if ( !s_dllShlwapi
.Load(_T("shlwapi.dll"), wxDL_VERBATIM 
| wxDL_QUIET
) ) 
 306             s_pfnSHAutoComplete 
= NULL
; 
 310             wxDL_INIT_FUNC(s_pfn
, SHAutoComplete
, s_dllShlwapi
); 
 314     if ( !s_pfnSHAutoComplete 
) 
 317     HRESULT hr 
= (*s_pfnSHAutoComplete
)(GetEditHwnd(), SHACF_FILESYS_ONLY
); 
 320         wxLogApiError(_T("SHAutoComplete()"), hr
); 
 325 #else // !HAS_AUTOCOMPLETE 
 327 #endif // HAS_AUTOCOMPLETE/!HAS_AUTOCOMPLETE 
 330 bool wxTextEntry::AutoComplete(const wxArrayString
& choices
) 
 332 #ifdef HAS_AUTOCOMPLETE 
 333     // create an object exposing IAutoComplete interface (don't go for 
 334     // IAutoComplete2 immediately as, presumably, it might be not available on 
 335     // older systems as otherwise why do we have both -- although in practice I 
 336     // don't know when can this happen) 
 337     IAutoComplete 
*pAutoComplete 
= NULL
; 
 338     HRESULT hr 
= CoCreateInstance
 
 342                     CLSCTX_INPROC_SERVER
, 
 344                     wx_reinterpret_cast(void **, &pAutoComplete
) 
 348         wxLogApiError(_T("CoCreateInstance(CLSID_AutoComplete)"), hr
); 
 352     // associate it with our strings 
 353     wxIEnumString 
*pEnumString 
= new wxIEnumString(choices
); 
 354     pEnumString
->AddRef(); 
 355     hr 
= pAutoComplete
->Init(GetEditHwnd(), pEnumString
, NULL
, NULL
); 
 356     pEnumString
->Release(); 
 359         wxLogApiError(_T("IAutoComplete::Init"), hr
); 
 363     // if IAutoComplete2 is available, set more user-friendly options 
 364     IAutoComplete2 
*pAutoComplete2 
= NULL
; 
 365     hr 
= pAutoComplete
->QueryInterface
 
 368                            wx_reinterpret_cast(void **, &pAutoComplete2
) 
 372         pAutoComplete2
->SetOptions(ACO_AUTOSUGGEST 
| ACO_UPDOWNKEYDROPSLIST
); 
 373         pAutoComplete2
->Release(); 
 376     // the docs are unclear about when can we release it but it seems safe to 
 377     // do it immediately, presumably the edit control itself keeps a reference 
 378     // to the auto completer object 
 379     pAutoComplete
->Release(); 
 381 #else // !HAS_AUTOCOMPLETE 
 382     wxUnusedVar(choices
); 
 385 #endif // HAS_AUTOCOMPLETE/!HAS_AUTOCOMPLETE 
 389 // ---------------------------------------------------------------------------- 
 391 // ---------------------------------------------------------------------------- 
 393 bool wxTextEntry::IsEditable() const 
 395     return !(::GetWindowLong(GetEditHwnd(), GWL_STYLE
) & ES_READONLY
); 
 398 void wxTextEntry::SetEditable(bool editable
) 
 400     ::SendMessage(GetEditHwnd(), EM_SETREADONLY
, !editable
, 0); 
 403 // ---------------------------------------------------------------------------- 
 405 // ---------------------------------------------------------------------------- 
 407 void wxTextEntry::SetMaxLength(unsigned long len
) 
 411         // this will set it to a platform-dependent maximum (much more 
 412         // than 64Kb under NT) 
 416     ::SendMessage(GetEditHwnd(), EM_LIMITTEXT
, len
, 0); 
 419 #endif // wxUSE_TEXTCTRL || wxUSE_COMBOBOX