]> git.saurik.com Git - wxWidgets.git/blame - src/stc/stc.cpp.in
Fix another line somehow missing from the wxGraphicsBitmap forward port.
[wxWidgets.git] / src / stc / stc.cpp.in
CommitLineData
f97d84a6
RD
1////////////////////////////////////////////////////////////////////////////
2// Name: stc.cpp
be5a51fb 3// Purpose: A wxWidgets implementation of Scintilla. This class is the
f97d84a6
RD
4// one meant to be used directly by wx applications. It does not
5// derive directly from the Scintilla classes, but instead
6// delegates most things to the real Scintilla class.
7// This allows the use of Scintilla without polluting the
8// namespace with all the classes and identifiers from Scintilla.
9//
10// Author: Robin Dunn
11//
12// Created: 13-Jan-2000
13// RCS-ID: $Id$
14// Copyright: (c) 2000 by Total Control Software
15// Licence: wxWindows license
16/////////////////////////////////////////////////////////////////////////////
17
a5c2ccf2
VZ
18/*
19 IMPORTANT: src/stc/stc.cpp is generated by src/stc/gen_iface.py from
20 src/stc/stc.cpp.in, don't edit stc.cpp file as your changes will be
21 lost after the next regeneration, edit stc.cpp.in and rerun the
22 gen_iface.py script instead!
23
24 Parts of this file generated by the script are found in between
25 the special "{{{" and "}}}" markers, the rest of it is copied
26 verbatim from src.h.in.
27 */
28
54429bb3
RD
29// For compilers that support precompilation, includes "wx.h".
30#include "wx/wxprec.h"
31
32#ifdef __BORLANDC__
33 #pragma hdrstop
34#endif
35
29825f5f
PC
36#if wxUSE_STC
37
38#include "wx/stc/stc.h"
4c784823 39#include "wx/stc/private.h"
54429bb3
RD
40
41#ifndef WX_PRECOMP
29825f5f 42 #include "wx/wx.h"
54429bb3
RD
43#endif // WX_PRECOMP
44
f97d84a6
RD
45#include <ctype.h>
46
d6655166
WS
47#include "wx/tokenzr.h"
48#include "wx/mstream.h"
49#include "wx/image.h"
3396739d 50#include "wx/ffile.h"
f97d84a6 51
f9ee2e27 52#include "ScintillaWX.h"
f97d84a6
RD
53
54//----------------------------------------------------------------------
55
23318a53 56const char wxSTCNameStr[] = "stcwindow";
f97d84a6 57
451c5cc7
RD
58#ifdef MAKELONG
59#undef MAKELONG
60#endif
61
62#define MAKELONG(a, b) ((a) | ((b) << 16))
63
64
65static long wxColourAsLong(const wxColour& co) {
66 return (((long)co.Blue() << 16) |
67 ((long)co.Green() << 8) |
68 ((long)co.Red()));
69}
70
71static wxColour wxColourFromLong(long c) {
72 wxColour clr;
a5b274d7
WS
73 clr.Set((unsigned char)(c & 0xff),
74 (unsigned char)((c >> 8) & 0xff),
75 (unsigned char)((c >> 16) & 0xff));
451c5cc7
RD
76 return clr;
77}
78
79
80static wxColour wxColourFromSpec(const wxString& spec) {
5ee1d760
RD
81 // spec should be a colour name or "#RRGGBB"
82 if (spec.GetChar(0) == wxT('#')) {
dc8005e2 83
5ee1d760
RD
84 long red, green, blue;
85 red = green = blue = 0;
86 spec.Mid(1,2).ToLong(&red, 16);
87 spec.Mid(3,2).ToLong(&green, 16);
88 spec.Mid(5,2).ToLong(&blue, 16);
a5b274d7
WS
89 return wxColour((unsigned char)red,
90 (unsigned char)green,
91 (unsigned char)blue);
5ee1d760
RD
92 }
93 else
94 return wxColour(spec);
451c5cc7
RD
95}
96
97//----------------------------------------------------------------------
98
9b11752c
VZ
99wxDEFINE_EVENT( wxEVT_STC_CHANGE, wxStyledTextEvent );
100wxDEFINE_EVENT( wxEVT_STC_STYLENEEDED, wxStyledTextEvent );
101wxDEFINE_EVENT( wxEVT_STC_CHARADDED, wxStyledTextEvent );
102wxDEFINE_EVENT( wxEVT_STC_SAVEPOINTREACHED, wxStyledTextEvent );
103wxDEFINE_EVENT( wxEVT_STC_SAVEPOINTLEFT, wxStyledTextEvent );
104wxDEFINE_EVENT( wxEVT_STC_ROMODIFYATTEMPT, wxStyledTextEvent );
105wxDEFINE_EVENT( wxEVT_STC_KEY, wxStyledTextEvent );
106wxDEFINE_EVENT( wxEVT_STC_DOUBLECLICK, wxStyledTextEvent );
107wxDEFINE_EVENT( wxEVT_STC_UPDATEUI, wxStyledTextEvent );
108wxDEFINE_EVENT( wxEVT_STC_MODIFIED, wxStyledTextEvent );
109wxDEFINE_EVENT( wxEVT_STC_MACRORECORD, wxStyledTextEvent );
110wxDEFINE_EVENT( wxEVT_STC_MARGINCLICK, wxStyledTextEvent );
111wxDEFINE_EVENT( wxEVT_STC_NEEDSHOWN, wxStyledTextEvent );
112wxDEFINE_EVENT( wxEVT_STC_PAINTED, wxStyledTextEvent );
113wxDEFINE_EVENT( wxEVT_STC_USERLISTSELECTION, wxStyledTextEvent );
114wxDEFINE_EVENT( wxEVT_STC_URIDROPPED, wxStyledTextEvent );
115wxDEFINE_EVENT( wxEVT_STC_DWELLSTART, wxStyledTextEvent );
116wxDEFINE_EVENT( wxEVT_STC_DWELLEND, wxStyledTextEvent );
117wxDEFINE_EVENT( wxEVT_STC_START_DRAG, wxStyledTextEvent );
118wxDEFINE_EVENT( wxEVT_STC_DRAG_OVER, wxStyledTextEvent );
119wxDEFINE_EVENT( wxEVT_STC_DO_DROP, wxStyledTextEvent );
120wxDEFINE_EVENT( wxEVT_STC_ZOOM, wxStyledTextEvent );
121wxDEFINE_EVENT( wxEVT_STC_HOTSPOT_CLICK, wxStyledTextEvent );
122wxDEFINE_EVENT( wxEVT_STC_HOTSPOT_DCLICK, wxStyledTextEvent );
123wxDEFINE_EVENT( wxEVT_STC_CALLTIP_CLICK, wxStyledTextEvent );
124wxDEFINE_EVENT( wxEVT_STC_AUTOCOMP_SELECTION, wxStyledTextEvent );
125wxDEFINE_EVENT( wxEVT_STC_INDICATOR_CLICK, wxStyledTextEvent );
126wxDEFINE_EVENT( wxEVT_STC_INDICATOR_RELEASE, wxStyledTextEvent );
9e730a78 127
d25f5fbb
RD
128
129
f97d84a6
RD
130BEGIN_EVENT_TABLE(wxStyledTextCtrl, wxControl)
131 EVT_PAINT (wxStyledTextCtrl::OnPaint)
132 EVT_SCROLLWIN (wxStyledTextCtrl::OnScrollWin)
5fa4613c 133 EVT_SCROLL (wxStyledTextCtrl::OnScroll)
f97d84a6
RD
134 EVT_SIZE (wxStyledTextCtrl::OnSize)
135 EVT_LEFT_DOWN (wxStyledTextCtrl::OnMouseLeftDown)
4ceb1196
RD
136 // Let Scintilla see the double click as a second click
137 EVT_LEFT_DCLICK (wxStyledTextCtrl::OnMouseLeftDown)
f97d84a6
RD
138 EVT_MOTION (wxStyledTextCtrl::OnMouseMove)
139 EVT_LEFT_UP (wxStyledTextCtrl::OnMouseLeftUp)
451c5cc7 140#if defined(__WXGTK__) || defined(__WXMAC__)
ddf2da08
RD
141 EVT_RIGHT_UP (wxStyledTextCtrl::OnMouseRightUp)
142#else
65ec6247 143 EVT_CONTEXT_MENU (wxStyledTextCtrl::OnContextMenu)
ddf2da08 144#endif
37d62433 145 EVT_MOUSEWHEEL (wxStyledTextCtrl::OnMouseWheel)
2b5f62a0 146 EVT_MIDDLE_UP (wxStyledTextCtrl::OnMouseMiddleUp)
f97d84a6
RD
147 EVT_CHAR (wxStyledTextCtrl::OnChar)
148 EVT_KEY_DOWN (wxStyledTextCtrl::OnKeyDown)
149 EVT_KILL_FOCUS (wxStyledTextCtrl::OnLoseFocus)
150 EVT_SET_FOCUS (wxStyledTextCtrl::OnGainFocus)
151 EVT_SYS_COLOUR_CHANGED (wxStyledTextCtrl::OnSysColourChanged)
152 EVT_ERASE_BACKGROUND (wxStyledTextCtrl::OnEraseBackground)
dd4aa550 153 EVT_MENU_RANGE (10, 16, wxStyledTextCtrl::OnMenu)
dc8005e2 154 EVT_LISTBOX_DCLICK (wxID_ANY, wxStyledTextCtrl::OnListBox)
f97d84a6
RD
155END_EVENT_TABLE()
156
157
158IMPLEMENT_CLASS(wxStyledTextCtrl, wxControl)
159IMPLEMENT_DYNAMIC_CLASS(wxStyledTextEvent, wxCommandEvent)
160
40716a51 161#ifdef LINK_LEXERS
1a2fb4cd 162// forces the linking of the lexer modules
a834585d 163int Scintilla_LinkLexers();
40716a51 164#endif
1a2fb4cd 165
f97d84a6
RD
166//----------------------------------------------------------------------
167// Constructor and Destructor
168
169wxStyledTextCtrl::wxStyledTextCtrl(wxWindow *parent,
170 wxWindowID id,
171 const wxPoint& pos,
172 const wxSize& size,
173 long style,
39c0acb6
RD
174 const wxString& name)
175{
176 m_swx = NULL;
177 Create(parent, id, pos, size, style, name);
178}
179
180
a48cb415
RD
181bool wxStyledTextCtrl::Create(wxWindow *parent,
182 wxWindowID id,
183 const wxPoint& pos,
184 const wxSize& size,
185 long style,
186 const wxString& name)
f97d84a6 187{
2659dad3 188 style |= wxVSCROLL | wxHSCROLL;
a48cb415
RD
189 if (!wxControl::Create(parent, id, pos, size,
190 style | wxWANTS_CHARS | wxCLIP_CHILDREN,
191 wxDefaultValidator, name))
192 return false;
39c0acb6 193
40716a51 194#ifdef LINK_LEXERS
a834585d 195 Scintilla_LinkLexers();
40716a51 196#endif
f97d84a6
RD
197 m_swx = new ScintillaWX(this);
198 m_stopWatch.Start();
dc8005e2 199 m_lastKeyDownConsumed = false;
60957703 200 m_lastWheelTimestamp = 0;
5fa4613c
RD
201 m_vScrollBar = NULL;
202 m_hScrollBar = NULL;
10ef30eb
RD
203#if wxUSE_UNICODE
204 // Put Scintilla into unicode (UTF-8) mode
205 SetCodePage(wxSTC_CP_UTF8);
206#endif
8ae4f086 207
170acdc9 208 SetInitialSize(size);
f9ee2e27
RD
209
210 // Reduces flicker on GTK+/X11
211 SetBackgroundStyle(wxBG_STYLE_CUSTOM);
5e328f38
RD
212
213 // Make sure it can take the focus
214 SetCanFocus(true);
215
a48cb415 216 return true;
f97d84a6
RD
217}
218
219
220wxStyledTextCtrl::~wxStyledTextCtrl() {
221 delete m_swx;
222}
223
224
225//----------------------------------------------------------------------
226
fafd43c5 227wxIntPtr wxStyledTextCtrl::SendMsg(int msg, wxUIntPtr wp, wxIntPtr lp) const
8e0945da 228{
f97d84a6
RD
229 return m_swx->WndProc(msg, wp, lp);
230}
231
ccfc3219
RD
232//----------------------------------------------------------------------
233
234// Set the vertical scrollbar to use instead of the ont that's built-in.
235void wxStyledTextCtrl::SetVScrollBar(wxScrollBar* bar) {
236 m_vScrollBar = bar;
237 if (bar != NULL) {
238 // ensure that the built-in scrollbar is not visible
239 SetScrollbar(wxVERTICAL, 0, 0, 0);
240 }
241}
f97d84a6 242
f97d84a6 243
ccfc3219
RD
244// Set the horizontal scrollbar to use instead of the ont that's built-in.
245void wxStyledTextCtrl::SetHScrollBar(wxScrollBar* bar) {
246 m_hScrollBar = bar;
247 if (bar != NULL) {
248 // ensure that the built-in scrollbar is not visible
249 SetScrollbar(wxHORIZONTAL, 0, 0, 0);
250 }
251}
252
f97d84a6 253//----------------------------------------------------------------------
a5c2ccf2 254// Generated methods implementation section {{{
f97d84a6
RD
255
256%(METHOD_IMPS)s
257
a5c2ccf2 258//}}}
f97d84a6
RD
259//----------------------------------------------------------------------
260
261
262// Returns the line number of the line with the caret.
263int wxStyledTextCtrl::GetCurrentLine() {
264 int line = LineFromPosition(GetCurrentPos());
265 return line;
266}
267
268
269// Extract style settings from a spec-string which is composed of one or
270// more of the following comma separated elements:
271//
272// bold turns on bold
273// italic turns on italics
5ee1d760
RD
274// fore:[name or #RRGGBB] sets the foreground colour
275// back:[name or #RRGGBB] sets the background colour
f97d84a6
RD
276// face:[facename] sets the font face name to use
277// size:[num] sets the font size in points
278// eol turns on eol filling
279// underline turns on underlining
280//
281void wxStyledTextCtrl::StyleSetSpec(int styleNum, const wxString& spec) {
282
451c5cc7 283 wxStringTokenizer tkz(spec, wxT(","));
f97d84a6
RD
284 while (tkz.HasMoreTokens()) {
285 wxString token = tkz.GetNextToken();
286
287 wxString option = token.BeforeFirst(':');
288 wxString val = token.AfterFirst(':');
289
451c5cc7 290 if (option == wxT("bold"))
f97d84a6
RD
291 StyleSetBold(styleNum, true);
292
451c5cc7 293 else if (option == wxT("italic"))
f97d84a6
RD
294 StyleSetItalic(styleNum, true);
295
451c5cc7 296 else if (option == wxT("underline"))
f97d84a6
RD
297 StyleSetUnderline(styleNum, true);
298
451c5cc7 299 else if (option == wxT("eol"))
f97d84a6
RD
300 StyleSetEOLFilled(styleNum, true);
301
451c5cc7 302 else if (option == wxT("size")) {
f97d84a6
RD
303 long points;
304 if (val.ToLong(&points))
305 StyleSetSize(styleNum, points);
306 }
307
451c5cc7 308 else if (option == wxT("face"))
f97d84a6
RD
309 StyleSetFaceName(styleNum, val);
310
451c5cc7 311 else if (option == wxT("fore"))
f97d84a6
RD
312 StyleSetForeground(styleNum, wxColourFromSpec(val));
313
451c5cc7 314 else if (option == wxT("back"))
f97d84a6
RD
315 StyleSetBackground(styleNum, wxColourFromSpec(val));
316 }
317}
318
319
7e0c58e9
RD
320// Get the font of a style
321wxFont wxStyledTextCtrl::StyleGetFont(int style) {
322 wxFont font;
323 font.SetPointSize(StyleGetSize(style));
324 font.SetFaceName(StyleGetFaceName(style));
325 if( StyleGetBold(style) )
326 font.SetWeight(wxFONTWEIGHT_BOLD);
327 else
328 font.SetWeight(wxFONTWEIGHT_NORMAL);
329
330 if( StyleGetItalic(style) )
331 font.SetStyle(wxFONTSTYLE_ITALIC);
332 else
333 font.SetStyle(wxFONTSTYLE_NORMAL);
334
335 return font;
336}
337
338
f97d84a6
RD
339// Set style size, face, bold, italic, and underline attributes from
340// a wxFont's attributes.
341void wxStyledTextCtrl::StyleSetFont(int styleNum, wxFont& font) {
7475e814
RD
342#ifdef __WXGTK__
343 // Ensure that the native font is initialized
344 int x, y;
345 GetTextExtent(wxT("X"), &x, &y, NULL, NULL, &font);
346#endif
af0531a5
RD
347 int size = font.GetPointSize();
348 wxString faceName = font.GetFaceName();
349 bool bold = font.GetWeight() == wxBOLD;
350 bool italic = font.GetStyle() != wxNORMAL;
351 bool under = font.GetUnderlined();
c5bd09bf 352 wxFontEncoding encoding = font.GetEncoding();
7e0c58e9 353
af0531a5 354 StyleSetFontAttr(styleNum, size, faceName, bold, italic, under, encoding);
f97d84a6
RD
355}
356
357// Set all font style attributes at once.
358void wxStyledTextCtrl::StyleSetFontAttr(int styleNum, int size,
359 const wxString& faceName,
360 bool bold, bool italic,
3727c043
RD
361 bool underline,
362 wxFontEncoding encoding) {
f97d84a6
RD
363 StyleSetSize(styleNum, size);
364 StyleSetFaceName(styleNum, faceName);
365 StyleSetBold(styleNum, bold);
366 StyleSetItalic(styleNum, italic);
367 StyleSetUnderline(styleNum, underline);
3727c043
RD
368 StyleSetFontEncoding(styleNum, encoding);
369}
f97d84a6 370
3727c043
RD
371
372// Set the character set of the font in a style. Converts the Scintilla
373// character set values to a wxFontEncoding.
374void wxStyledTextCtrl::StyleSetCharacterSet(int style, int characterSet)
375{
376 wxFontEncoding encoding;
377
378 // Translate the Scintilla characterSet to a wxFontEncoding
379 switch (characterSet) {
380 default:
381 case wxSTC_CHARSET_ANSI:
382 case wxSTC_CHARSET_DEFAULT:
383 encoding = wxFONTENCODING_DEFAULT;
384 break;
385
386 case wxSTC_CHARSET_BALTIC:
387 encoding = wxFONTENCODING_ISO8859_13;
388 break;
389
390 case wxSTC_CHARSET_CHINESEBIG5:
391 encoding = wxFONTENCODING_CP950;
392 break;
393
394 case wxSTC_CHARSET_EASTEUROPE:
395 encoding = wxFONTENCODING_ISO8859_2;
396 break;
397
398 case wxSTC_CHARSET_GB2312:
399 encoding = wxFONTENCODING_CP936;
400 break;
401
402 case wxSTC_CHARSET_GREEK:
403 encoding = wxFONTENCODING_ISO8859_7;
404 break;
405
406 case wxSTC_CHARSET_HANGUL:
407 encoding = wxFONTENCODING_CP949;
408 break;
409
410 case wxSTC_CHARSET_MAC:
411 encoding = wxFONTENCODING_DEFAULT;
412 break;
413
414 case wxSTC_CHARSET_OEM:
415 encoding = wxFONTENCODING_DEFAULT;
416 break;
417
418 case wxSTC_CHARSET_RUSSIAN:
419 encoding = wxFONTENCODING_KOI8;
420 break;
421
422 case wxSTC_CHARSET_SHIFTJIS:
423 encoding = wxFONTENCODING_CP932;
424 break;
425
426 case wxSTC_CHARSET_SYMBOL:
427 encoding = wxFONTENCODING_DEFAULT;
428 break;
429
430 case wxSTC_CHARSET_TURKISH:
431 encoding = wxFONTENCODING_ISO8859_9;
432 break;
433
434 case wxSTC_CHARSET_JOHAB:
435 encoding = wxFONTENCODING_DEFAULT;
436 break;
437
438 case wxSTC_CHARSET_HEBREW:
439 encoding = wxFONTENCODING_ISO8859_8;
440 break;
441
442 case wxSTC_CHARSET_ARABIC:
443 encoding = wxFONTENCODING_ISO8859_6;
444 break;
445
446 case wxSTC_CHARSET_VIETNAMESE:
447 encoding = wxFONTENCODING_DEFAULT;
448 break;
449
450 case wxSTC_CHARSET_THAI:
451 encoding = wxFONTENCODING_ISO8859_11;
452 break;
1e9bafca
RD
453
454 case wxSTC_CHARSET_CYRILLIC:
455 encoding = wxFONTENCODING_ISO8859_5;
456 break;
7e0c58e9 457
1e9bafca
RD
458 case wxSTC_CHARSET_8859_15:
459 encoding = wxFONTENCODING_ISO8859_15;;
460 break;
3727c043
RD
461 }
462
463 // We just have Scintilla track the wxFontEncoding for us. It gets used
464 // in Font::Create in PlatWX.cpp. We add one to the value so that the
465 // effective wxFONENCODING_DEFAULT == SC_SHARSET_DEFAULT and so when
466 // Scintilla internally uses SC_CHARSET_DEFAULT we will translate it back
467 // to wxFONENCODING_DEFAULT in Font::Create.
468 SendMsg(SCI_STYLESETCHARACTERSET, style, encoding+1);
469}
470
471
472// Set the font encoding to be used by a style.
473void wxStyledTextCtrl::StyleSetFontEncoding(int style, wxFontEncoding encoding)
474{
475 SendMsg(SCI_STYLESETCHARACTERSET, style, encoding+1);
f97d84a6
RD
476}
477
478
479// Perform one of the operations defined by the wxSTC_CMD_* constants.
480void wxStyledTextCtrl::CmdKeyExecute(int cmd) {
481 SendMsg(cmd);
482}
483
484
485// Set the left and right margin in the edit area, measured in pixels.
486void wxStyledTextCtrl::SetMargins(int left, int right) {
487 SetMarginLeft(left);
488 SetMarginRight(right);
489}
490
491
f97d84a6
RD
492// Retrieve the point in the window where a position is displayed.
493wxPoint wxStyledTextCtrl::PointFromPosition(int pos) {
494 int x = SendMsg(SCI_POINTXFROMPOSITION, 0, pos);
495 int y = SendMsg(SCI_POINTYFROMPOSITION, 0, pos);
496 return wxPoint(x, y);
497}
498
499// Scroll enough to make the given line visible
500void wxStyledTextCtrl::ScrollToLine(int line) {
501 m_swx->DoScrollToLine(line);
502}
503
504
505// Scroll enough to make the given column visible
506void wxStyledTextCtrl::ScrollToColumn(int column) {
507 m_swx->DoScrollToColumn(column);
508}
509
510
2bfca191 511#if wxUSE_TEXTCTRL
3396739d
VZ
512bool wxStyledTextCtrl::DoSaveFile(const wxString& filename, int fileType)
513{
514 bool ok = wxTextAreaBase::DoSaveFile(filename, fileType);
2bfca191 515#else
51566b0b
RD
516bool wxStyledTextCtrl::SaveFile(const wxString& filename)
517{
3396739d
VZ
518#if wxUSE_FFILE
519 wxFFile file(filename, wxT("w"));
520 bool ok = file.IsOpened() && file.Write(GetValue(), *wxConvCurrent);
521#else
522 bool ok = false;
523#endif // wxUSE_FFILE
524#endif
525 if (ok)
526 {
51566b0b 527 SetSavePoint();
3396739d
VZ
528 }
529 return ok;
51566b0b
RD
530}
531
2bfca191 532#if wxUSE_TEXTCTRL
3396739d
VZ
533bool wxStyledTextCtrl::DoLoadFile(const wxString& filename, int fileType)
534{
535 bool ok = wxTextAreaBase::DoLoadFile(filename, fileType);
2bfca191 536#else
51566b0b
RD
537bool wxStyledTextCtrl::LoadFile(const wxString& filename)
538{
3396739d
VZ
539#if wxUSE_FFILE
540 wxFFile file(filename);
541 bool ok = file.IsOpened();
542 if (ok)
51566b0b 543 {
3396739d
VZ
544 wxString text;
545 ok = file.ReadAll(&text, *wxConvCurrent);
546 if (ok)
041973c5 547 {
3396739d 548 SetValue(text);
041973c5 549 }
51566b0b 550 }
3396739d
VZ
551#else
552 bool ok = false;
553#endif // wxUSE_FFILE
554#endif
555 if (ok)
556 {
557 EmptyUndoBuffer();
558 SetSavePoint();
559 }
560 return ok;
51566b0b
RD
561}
562
2fcce896 563#if wxUSE_DRAG_AND_DROP
dc8005e2
RD
564wxDragResult wxStyledTextCtrl::DoDragOver(wxCoord x, wxCoord y, wxDragResult def) {
565 return m_swx->DoDragOver(x, y, def);
566}
4a65f2c8
RD
567
568
dc8005e2 569bool wxStyledTextCtrl::DoDropText(long x, long y, const wxString& data) {
4a65f2c8
RD
570 return m_swx->DoDropText(x, y, data);
571}
2fcce896 572#endif
4a65f2c8
RD
573
574
d1558f3d
RD
575void wxStyledTextCtrl::SetUseAntiAliasing(bool useAA) {
576 m_swx->SetUseAntiAliasing(useAA);
577}
578
579bool wxStyledTextCtrl::GetUseAntiAliasing() {
580 return m_swx->GetUseAntiAliasing();
581}
582
41a499cd
RD
583
584
585
586
587void wxStyledTextCtrl::AddTextRaw(const char* text)
588{
b796ba39 589 SendMsg(SCI_ADDTEXT, strlen(text), (sptr_t)text);
41a499cd
RD
590}
591
592void wxStyledTextCtrl::InsertTextRaw(int pos, const char* text)
593{
b796ba39 594 SendMsg(SCI_INSERTTEXT, pos, (sptr_t)text);
41a499cd
RD
595}
596
597wxCharBuffer wxStyledTextCtrl::GetCurLineRaw(int* linePos)
598{
599 int len = LineLength(GetCurrentLine());
600 if (!len) {
601 if (linePos) *linePos = 0;
602 wxCharBuffer empty;
603 return empty;
604 }
605
606 wxCharBuffer buf(len);
b796ba39 607 int pos = SendMsg(SCI_GETCURLINE, len, (sptr_t)buf.data());
41a499cd
RD
608 if (linePos) *linePos = pos;
609 return buf;
610}
611
612wxCharBuffer wxStyledTextCtrl::GetLineRaw(int line)
613{
614 int len = LineLength(line);
615 if (!len) {
616 wxCharBuffer empty;
617 return empty;
618 }
619
620 wxCharBuffer buf(len);
b796ba39 621 SendMsg(SCI_GETLINE, line, (sptr_t)buf.data());
41a499cd
RD
622 return buf;
623}
624
625wxCharBuffer wxStyledTextCtrl::GetSelectedTextRaw()
626{
fa70ec2b
RD
627 long start;
628 long end;
41a499cd
RD
629
630 GetSelection(&start, &end);
631 int len = end - start;
632 if (!len) {
633 wxCharBuffer empty;
634 return empty;
7e0c58e9 635 }
41a499cd
RD
636
637 wxCharBuffer buf(len);
b796ba39 638 SendMsg(SCI_GETSELTEXT, 0, (sptr_t)buf.data());
41a499cd
RD
639 return buf;
640}
641
642wxCharBuffer wxStyledTextCtrl::GetTextRangeRaw(int startPos, int endPos)
643{
644 if (endPos < startPos) {
645 int temp = startPos;
646 startPos = endPos;
647 endPos = temp;
648 }
649 int len = endPos - startPos;
650 if (!len) {
651 wxCharBuffer empty;
652 return empty;
7e0c58e9 653 }
41a499cd
RD
654
655 wxCharBuffer buf(len);
656 TextRange tr;
657 tr.lpstrText = buf.data();
658 tr.chrg.cpMin = startPos;
659 tr.chrg.cpMax = endPos;
b796ba39 660 SendMsg(SCI_GETTEXTRANGE, 0, (sptr_t)&tr);
41a499cd
RD
661 return buf;
662}
663
664void wxStyledTextCtrl::SetTextRaw(const char* text)
665{
b796ba39 666 SendMsg(SCI_SETTEXT, 0, (sptr_t)text);
41a499cd
RD
667}
668
669wxCharBuffer wxStyledTextCtrl::GetTextRaw()
670{
949750de
VZ
671 int len = GetTextLength();
672 wxCharBuffer buf(len); // adds 1 for NUL automatically
b796ba39 673 SendMsg(SCI_GETTEXT, len + 1, (sptr_t)buf.data());
41a499cd
RD
674 return buf;
675}
676
677void wxStyledTextCtrl::AppendTextRaw(const char* text)
678{
b796ba39 679 SendMsg(SCI_APPENDTEXT, strlen(text), (sptr_t)text);
41a499cd
RD
680}
681
682
683
684
685
f97d84a6
RD
686//----------------------------------------------------------------------
687// Event handlers
688
88a8b04e 689void wxStyledTextCtrl::OnPaint(wxPaintEvent& WXUNUSED(evt)) {
f97d84a6 690 wxPaintDC dc(this);
9e730a78 691 m_swx->DoPaint(&dc, GetUpdateRegion().GetBox());
f97d84a6
RD
692}
693
694void wxStyledTextCtrl::OnScrollWin(wxScrollWinEvent& evt) {
695 if (evt.GetOrientation() == wxHORIZONTAL)
696 m_swx->DoHScroll(evt.GetEventType(), evt.GetPosition());
697 else
698 m_swx->DoVScroll(evt.GetEventType(), evt.GetPosition());
699}
700
5fa4613c
RD
701void wxStyledTextCtrl::OnScroll(wxScrollEvent& evt) {
702 wxScrollBar* sb = wxDynamicCast(evt.GetEventObject(), wxScrollBar);
703 if (sb) {
704 if (sb->IsVertical())
705 m_swx->DoVScroll(evt.GetEventType(), evt.GetPosition());
706 else
707 m_swx->DoHScroll(evt.GetEventType(), evt.GetPosition());
708 }
709}
710
88a8b04e 711void wxStyledTextCtrl::OnSize(wxSizeEvent& WXUNUSED(evt)) {
39c0acb6
RD
712 if (m_swx) {
713 wxSize sz = GetClientSize();
714 m_swx->DoSize(sz.x, sz.y);
715 }
f97d84a6
RD
716}
717
718void wxStyledTextCtrl::OnMouseLeftDown(wxMouseEvent& evt) {
cb1871ca 719 SetFocus();
f97d84a6 720 wxPoint pt = evt.GetPosition();
2b5f62a0 721 m_swx->DoLeftButtonDown(Point(pt.x, pt.y), m_stopWatch.Time(),
f97d84a6
RD
722 evt.ShiftDown(), evt.ControlDown(), evt.AltDown());
723}
724
725void wxStyledTextCtrl::OnMouseMove(wxMouseEvent& evt) {
726 wxPoint pt = evt.GetPosition();
2b5f62a0 727 m_swx->DoLeftButtonMove(Point(pt.x, pt.y));
f97d84a6
RD
728}
729
730void wxStyledTextCtrl::OnMouseLeftUp(wxMouseEvent& evt) {
731 wxPoint pt = evt.GetPosition();
2b5f62a0 732 m_swx->DoLeftButtonUp(Point(pt.x, pt.y), m_stopWatch.Time(),
f97d84a6
RD
733 evt.ControlDown());
734}
735
736
ddf2da08
RD
737void wxStyledTextCtrl::OnMouseRightUp(wxMouseEvent& evt) {
738 wxPoint pt = evt.GetPosition();
739 m_swx->DoContextMenu(Point(pt.x, pt.y));
740}
741
742
2b5f62a0
VZ
743void wxStyledTextCtrl::OnMouseMiddleUp(wxMouseEvent& evt) {
744 wxPoint pt = evt.GetPosition();
745 m_swx->DoMiddleButtonUp(Point(pt.x, pt.y));
746}
747
65ec6247 748void wxStyledTextCtrl::OnContextMenu(wxContextMenuEvent& evt) {
f97d84a6 749 wxPoint pt = evt.GetPosition();
65ec6247 750 ScreenToClient(&pt.x, &pt.y);
25484746
RD
751 /*
752 Show context menu at event point if it's within the window,
753 or at caret location if not
754 */
755 wxHitTest ht = this->HitTest(pt);
756 if (ht != wxHT_WINDOW_INSIDE) {
757 pt = this->PointFromPosition(this->GetCurrentPos());
758 }
f97d84a6
RD
759 m_swx->DoContextMenu(Point(pt.x, pt.y));
760}
761
37d62433 762
60957703
VZ
763void wxStyledTextCtrl::OnMouseWheel(wxMouseEvent& evt)
764{
765 // prevent having an event queue with wheel events that cannot be processed
766 // reasonably fast (see ticket #9057)
767 if ( m_lastWheelTimestamp <= evt.GetTimestamp() )
768 {
769 m_lastWheelTimestamp = m_stopWatch.Time();
770 m_swx->DoMouseWheel(evt.GetWheelRotation(),
771 evt.GetWheelDelta(),
772 evt.GetLinesPerAction(),
773 evt.ControlDown(),
774 evt.IsPageScroll());
775 m_lastWheelTimestamp = m_stopWatch.Time() - m_lastWheelTimestamp;
776 m_lastWheelTimestamp += evt.GetTimestamp();
777 }
37d62433
RD
778}
779
780
f97d84a6 781void wxStyledTextCtrl::OnChar(wxKeyEvent& evt) {
5fd656d5 782 // On (some?) non-US PC keyboards the AltGr key is required to enter some
f3c2c221
RD
783 // common characters. It comes to us as both Alt and Ctrl down so we need
784 // to let the char through in that case, otherwise if only ctrl or only
785 // alt let's skip it.
786 bool ctrl = evt.ControlDown();
28e0c28e
RD
787#ifdef __WXMAC__
788 // On the Mac the Alt key is just a modifier key (like Shift) so we need
789 // to allow the char events to be processed when Alt is pressed.
790 // TODO: Should we check MetaDown instead in this case?
791 bool alt = false;
792#else
f3c2c221 793 bool alt = evt.AltDown();
28e0c28e 794#endif
00c64037 795 bool skip = ((ctrl || alt) && ! (ctrl && alt));
f3c2c221 796
4358b585
VZ
797#if wxUSE_UNICODE
798 // apparently if we don't do this, Unicode keys pressed after non-char
799 // ASCII ones (e.g. Enter, Tab) are not taken into account (patch 1615989)
800 if (m_lastKeyDownConsumed && evt.GetUnicodeKey() > 255)
801 m_lastKeyDownConsumed = false;
802#endif
803
5fd656d5
RD
804 if (!m_lastKeyDownConsumed && !skip) {
805#if wxUSE_UNICODE
806 int key = evt.GetUnicodeKey();
807 bool keyOk = true;
808
809 // if the unicode key code is not really a unicode character (it may
810 // be a function key or etc., the platforms appear to always give us a
811 // small value in this case) then fallback to the ascii key code but
812 // don't do anything for function keys or etc.
1b14227e 813 if (key <= 127) {
5fd656d5 814 key = evt.GetKeyCode();
1b14227e 815 keyOk = (key <= 127);
5fd656d5
RD
816 }
817 if (keyOk) {
818 m_swx->DoAddChar(key);
819 return;
820 }
821#else
822 int key = evt.GetKeyCode();
823 if (key <= WXK_START || key > WXK_COMMAND) {
824 m_swx->DoAddChar(key);
825 return;
826 }
827#endif
f97d84a6 828 }
7e0c58e9 829
f3c2c221 830 evt.Skip();
f97d84a6
RD
831}
832
d6582821 833
f97d84a6 834void wxStyledTextCtrl::OnKeyDown(wxKeyEvent& evt) {
5fd656d5 835 int processed = m_swx->DoKeyDown(evt, &m_lastKeyDownConsumed);
d6582821 836 if (!processed && !m_lastKeyDownConsumed)
f97d84a6
RD
837 evt.Skip();
838}
839
d6582821 840
b6bfd8e8 841void wxStyledTextCtrl::OnLoseFocus(wxFocusEvent& evt) {
ec830416 842 m_swx->DoLoseFocus();
b6bfd8e8 843 evt.Skip();
f97d84a6
RD
844}
845
d6582821 846
b6bfd8e8 847void wxStyledTextCtrl::OnGainFocus(wxFocusEvent& evt) {
f97d84a6 848 m_swx->DoGainFocus();
b6bfd8e8 849 evt.Skip();
f97d84a6
RD
850}
851
d6582821 852
88a8b04e 853void wxStyledTextCtrl::OnSysColourChanged(wxSysColourChangedEvent& WXUNUSED(evt)) {
f97d84a6
RD
854 m_swx->DoSysColourChange();
855}
856
d6582821 857
88a8b04e 858void wxStyledTextCtrl::OnEraseBackground(wxEraseEvent& WXUNUSED(evt)) {
f97d84a6
RD
859 // do nothing to help avoid flashing
860}
861
862
863
864void wxStyledTextCtrl::OnMenu(wxCommandEvent& evt) {
865 m_swx->DoCommand(evt.GetId());
866}
867
868
88a8b04e 869void wxStyledTextCtrl::OnListBox(wxCommandEvent& WXUNUSED(evt)) {
f97d84a6
RD
870 m_swx->DoOnListBox();
871}
872
873
8e54aaed
RD
874void wxStyledTextCtrl::OnIdle(wxIdleEvent& evt) {
875 m_swx->DoOnIdle(evt);
876}
877
878
8ae4f086
RD
879wxSize wxStyledTextCtrl::DoGetBestSize() const
880{
881 // What would be the best size for a wxSTC?
882 // Just give a reasonable minimum until something else can be figured out.
883 return wxSize(200,100);
884}
885
886
f97d84a6
RD
887//----------------------------------------------------------------------
888// Turn notifications from Scintilla into events
889
890
891void wxStyledTextCtrl::NotifyChange() {
892 wxStyledTextEvent evt(wxEVT_STC_CHANGE, GetId());
a29a241f 893 evt.SetEventObject(this);
f97d84a6
RD
894 GetEventHandler()->ProcessEvent(evt);
895}
896
2b5f62a0
VZ
897
898static void SetEventText(wxStyledTextEvent& evt, const char* text,
899 size_t length) {
900 if(!text) return;
901
1c930beb 902 evt.SetText(stc2wx(text, length));
2b5f62a0
VZ
903}
904
905
f97d84a6
RD
906void wxStyledTextCtrl::NotifyParent(SCNotification* _scn) {
907 SCNotification& scn = *_scn;
65ec6247
RD
908 wxStyledTextEvent evt(0, GetId());
909
a29a241f 910 evt.SetEventObject(this);
65ec6247
RD
911 evt.SetPosition(scn.position);
912 evt.SetKey(scn.ch);
913 evt.SetModifiers(scn.modifiers);
914
f97d84a6
RD
915 switch (scn.nmhdr.code) {
916 case SCN_STYLENEEDED:
65ec6247 917 evt.SetEventType(wxEVT_STC_STYLENEEDED);
f97d84a6 918 break;
65ec6247 919
f97d84a6 920 case SCN_CHARADDED:
65ec6247 921 evt.SetEventType(wxEVT_STC_CHARADDED);
f97d84a6 922 break;
65ec6247 923
f97d84a6 924 case SCN_SAVEPOINTREACHED:
65ec6247 925 evt.SetEventType(wxEVT_STC_SAVEPOINTREACHED);
f97d84a6 926 break;
65ec6247 927
f97d84a6 928 case SCN_SAVEPOINTLEFT:
65ec6247 929 evt.SetEventType(wxEVT_STC_SAVEPOINTLEFT);
f97d84a6 930 break;
65ec6247 931
f97d84a6 932 case SCN_MODIFYATTEMPTRO:
65ec6247
RD
933 evt.SetEventType(wxEVT_STC_ROMODIFYATTEMPT);
934 break;
935
936 case SCN_KEY:
937 evt.SetEventType(wxEVT_STC_KEY);
f97d84a6 938 break;
65ec6247 939
f97d84a6 940 case SCN_DOUBLECLICK:
65ec6247 941 evt.SetEventType(wxEVT_STC_DOUBLECLICK);
f97d84a6 942 break;
65ec6247
RD
943
944 case SCN_UPDATEUI:
945 evt.SetEventType(wxEVT_STC_UPDATEUI);
f97d84a6 946 break;
65ec6247
RD
947
948 case SCN_MODIFIED:
949 evt.SetEventType(wxEVT_STC_MODIFIED);
950 evt.SetModificationType(scn.modificationType);
2b5f62a0 951 SetEventText(evt, scn.text, scn.length);
65ec6247
RD
952 evt.SetLength(scn.length);
953 evt.SetLinesAdded(scn.linesAdded);
954 evt.SetLine(scn.line);
955 evt.SetFoldLevelNow(scn.foldLevelNow);
956 evt.SetFoldLevelPrev(scn.foldLevelPrev);
f97d84a6 957 break;
65ec6247 958
f97d84a6 959 case SCN_MACRORECORD:
65ec6247
RD
960 evt.SetEventType(wxEVT_STC_MACRORECORD);
961 evt.SetMessage(scn.message);
962 evt.SetWParam(scn.wParam);
963 evt.SetLParam(scn.lParam);
f97d84a6 964 break;
65ec6247 965
f97d84a6 966 case SCN_MARGINCLICK:
65ec6247
RD
967 evt.SetEventType(wxEVT_STC_MARGINCLICK);
968 evt.SetMargin(scn.margin);
f97d84a6 969 break;
65ec6247 970
f97d84a6 971 case SCN_NEEDSHOWN:
65ec6247
RD
972 evt.SetEventType(wxEVT_STC_NEEDSHOWN);
973 evt.SetLength(scn.length);
f97d84a6 974 break;
65ec6247 975
65ec6247
RD
976 case SCN_PAINTED:
977 evt.SetEventType(wxEVT_STC_PAINTED);
978 break;
979
0daf5e6b
RD
980 case SCN_AUTOCSELECTION:
981 evt.SetEventType(wxEVT_STC_AUTOCOMP_SELECTION);
982 evt.SetListType(scn.listType);
983 SetEventText(evt, scn.text, strlen(scn.text));
984 evt.SetPosition(scn.lParam);
985 break;
7e0c58e9 986
65ec6247
RD
987 case SCN_USERLISTSELECTION:
988 evt.SetEventType(wxEVT_STC_USERLISTSELECTION);
989 evt.SetListType(scn.listType);
2b5f62a0 990 SetEventText(evt, scn.text, strlen(scn.text));
0daf5e6b 991 evt.SetPosition(scn.lParam);
f97d84a6 992 break;
f97d84a6 993
65ec6247
RD
994 case SCN_URIDROPPED:
995 evt.SetEventType(wxEVT_STC_URIDROPPED);
2b5f62a0 996 SetEventText(evt, scn.text, strlen(scn.text));
65ec6247
RD
997 break;
998
999 case SCN_DWELLSTART:
1000 evt.SetEventType(wxEVT_STC_DWELLSTART);
1001 evt.SetX(scn.x);
1002 evt.SetY(scn.y);
1003 break;
1004
1005 case SCN_DWELLEND:
1006 evt.SetEventType(wxEVT_STC_DWELLEND);
1007 evt.SetX(scn.x);
1008 evt.SetY(scn.y);
1009 break;
1010
a834585d
RD
1011 case SCN_ZOOM:
1012 evt.SetEventType(wxEVT_STC_ZOOM);
1013 break;
1014
9e730a78
RD
1015 case SCN_HOTSPOTCLICK:
1016 evt.SetEventType(wxEVT_STC_HOTSPOT_CLICK);
1017 break;
1018
1019 case SCN_HOTSPOTDOUBLECLICK:
1020 evt.SetEventType(wxEVT_STC_HOTSPOT_DCLICK);
1021 break;
1022
1023 case SCN_CALLTIPCLICK:
1024 evt.SetEventType(wxEVT_STC_CALLTIP_CLICK);
1025 break;
7e0c58e9
RD
1026
1027 case SCN_INDICATORCLICK:
1028 evt.SetEventType(wxEVT_STC_INDICATOR_CLICK);
1029 break;
1030
1031 case SCN_INDICATORRELEASE:
1032 evt.SetEventType(wxEVT_STC_INDICATOR_RELEASE);
1033 break;
1034
65ec6247
RD
1035 default:
1036 return;
f97d84a6 1037 }
65ec6247
RD
1038
1039 GetEventHandler()->ProcessEvent(evt);
f97d84a6
RD
1040}
1041
1042
f97d84a6
RD
1043//----------------------------------------------------------------------
1044//----------------------------------------------------------------------
1045//----------------------------------------------------------------------
1046
1047wxStyledTextEvent::wxStyledTextEvent(wxEventType commandType, int id)
1048 : wxCommandEvent(commandType, id)
1049{
1050 m_position = 0;
1051 m_key = 0;
1052 m_modifiers = 0;
1053 m_modificationType = 0;
1054 m_length = 0;
1055 m_linesAdded = 0;
1056 m_line = 0;
1057 m_foldLevelNow = 0;
1058 m_foldLevelPrev = 0;
1059 m_margin = 0;
1060 m_message = 0;
1061 m_wParam = 0;
1062 m_lParam = 0;
65ec6247
RD
1063 m_listType = 0;
1064 m_x = 0;
1065 m_y = 0;
dc8005e2 1066 m_dragAllowMove = false;
92bbd64f 1067#if wxUSE_DRAG_AND_DROP
a29a241f 1068 m_dragResult = wxDragNone;
92bbd64f 1069#endif
f97d84a6
RD
1070}
1071
1072bool wxStyledTextEvent::GetShift() const { return (m_modifiers & SCI_SHIFT) != 0; }
1073bool wxStyledTextEvent::GetControl() const { return (m_modifiers & SCI_CTRL) != 0; }
1074bool wxStyledTextEvent::GetAlt() const { return (m_modifiers & SCI_ALT) != 0; }
1075
f97d84a6 1076
5fa4613c
RD
1077wxStyledTextEvent::wxStyledTextEvent(const wxStyledTextEvent& event):
1078 wxCommandEvent(event)
1079{
1080 m_position = event.m_position;
1081 m_key = event.m_key;
1082 m_modifiers = event.m_modifiers;
1083 m_modificationType = event.m_modificationType;
1084 m_text = event.m_text;
1085 m_length = event.m_length;
1086 m_linesAdded = event.m_linesAdded;
1087 m_line = event.m_line;
1088 m_foldLevelNow = event.m_foldLevelNow;
1089 m_foldLevelPrev = event.m_foldLevelPrev;
1090
1091 m_margin = event.m_margin;
1092
1093 m_message = event.m_message;
1094 m_wParam = event.m_wParam;
1095 m_lParam = event.m_lParam;
1096
1097 m_listType = event.m_listType;
1098 m_x = event.m_x;
1099 m_y = event.m_y;
f97d84a6 1100
5fa4613c
RD
1101 m_dragText = event.m_dragText;
1102 m_dragAllowMove =event.m_dragAllowMove;
92bbd64f 1103#if wxUSE_DRAG_AND_DROP
5fa4613c 1104 m_dragResult = event.m_dragResult;
92bbd64f 1105#endif
f97d84a6
RD
1106}
1107
1108//----------------------------------------------------------------------
1109//----------------------------------------------------------------------
1110
29825f5f 1111#endif // wxUSE_STC