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