]> git.saurik.com Git - wxWidgets.git/blame - contrib/src/stc/PlatWX.cpp
OS/2 Statusbar fixes
[wxWidgets.git] / contrib / src / stc / PlatWX.cpp
CommitLineData
9ce192d4
RD
1// Scintilla source code edit control
2// PlatWX.cxx - implementation of platform facilities on wxWindows
3// Copyright 1998-1999 by Neil Hodgson <neilh@scintilla.org>
4// Robin Dunn <robin@aldunn.com>
5// The License.txt file describes the conditions under which this software may be distributed.
6
f6bcfd97 7#include <ctype.h>
9ce192d4
RD
8
9#include "Platform.h"
10#include "wx/stc/stc.h"
11
f97d84a6
RD
12
13#ifdef __WXGTK__
14#include <gtk/gtk.h>
15#endif
16
9ce192d4 17Point Point::FromLong(long lpoint) {
f6bcfd97 18 return Point(lpoint & 0xFFFF, lpoint >> 16);
9ce192d4
RD
19}
20
21wxRect wxRectFromPRectangle(PRectangle prc) {
22 wxRect rc(prc.left, prc.top,
21156596 23 prc.right-prc.left, prc.bottom-prc.top);
9ce192d4
RD
24 return rc;
25}
26
27PRectangle PRectangleFromwxRect(wxRect rc) {
bec17edf
RD
28 return PRectangle(rc.GetLeft(), rc.GetTop(),
29 rc.GetRight()+1, rc.GetBottom()+1);
9ce192d4
RD
30}
31
32Colour::Colour(long lcol) {
33 co.Set(lcol & 0xff, (lcol >> 8) & 0xff, (lcol >> 16) & 0xff);
34}
35
36Colour::Colour(unsigned int red, unsigned int green, unsigned int blue) {
37 co.Set(red, green, blue);
38}
39
40bool Colour::operator==(const Colour &other) const {
41 return co == other.co;
42}
43
44long Colour::AsLong() const {
45 return (((long)co.Blue() << 16) |
46 ((long)co.Green() << 8) |
47 ((long)co.Red()));
48}
49
50unsigned int Colour::GetRed() {
51 return co.Red();
52}
53
54unsigned int Colour::GetGreen() {
55 return co.Green();
56}
57
58unsigned int Colour::GetBlue() {
59 return co.Blue();
60}
61
62Palette::Palette() {
63 used = 0;
64 allowRealization = false;
65}
66
67Palette::~Palette() {
68 Release();
69}
70
71void Palette::Release() {
72 used = 0;
73}
74
75// This method either adds a colour to the list of wanted colours (want==true)
76// or retrieves the allocated colour back to the ColourPair.
77// This is one method to make it easier to keep the code for wanting and retrieving in sync.
78void Palette::WantFind(ColourPair &cp, bool want) {
79 if (want) {
80 for (int i=0; i < used; i++) {
81 if (entries[i].desired == cp.desired)
82 return;
83 }
84
85 if (used < numEntries) {
86 entries[used].desired = cp.desired;
87 entries[used].allocated = cp.desired;
88 used++;
89 }
90 } else {
91 for (int i=0; i < used; i++) {
92 if (entries[i].desired == cp.desired) {
93 cp.allocated = entries[i].allocated;
94 return;
95 }
96 }
97 cp.allocated = cp.desired;
98 }
99}
100
101void Palette::Allocate(Window &) {
102 if (allowRealization) {
103 }
104}
105
106
107Font::Font() {
108 id = 0;
109 ascent = 0;
110}
111
112Font::~Font() {
113}
114
f6bcfd97 115void Font::Create(const char *faceName, int characterSet, int size, bool bold, bool italic) {
65ec6247
RD
116 // TODO: what to do about the characterSet?
117
9ce192d4
RD
118 Release();
119 id = new wxFont(size,
120 wxDEFAULT,
121 italic ? wxITALIC : wxNORMAL,
122 bold ? wxBOLD : wxNORMAL,
123 false,
f6bcfd97
BP
124 faceName,
125 wxFONTENCODING_DEFAULT);
9ce192d4
RD
126}
127
128
129void Font::Release() {
130 if (id)
131 delete id;
132 id = 0;
133}
134
135
136Surface::Surface() :
137 hdc(0), hdcOwned(0), bitmap(0),
138 x(0), y(0) {
139}
140
141Surface::~Surface() {
142 Release();
143}
144
145void Surface::Release() {
146 if (bitmap) {
147 ((wxMemoryDC*)hdc)->SelectObject(wxNullBitmap);
148 delete bitmap;
149 bitmap = 0;
150 }
151 if (hdcOwned) {
152 delete hdc;
153 hdc = 0;
154 hdcOwned = false;
155 }
156}
157
158
159bool Surface::Initialised() {
160 return hdc != 0;
161}
162
163void Surface::Init() {
164 Release();
165 hdc = new wxMemoryDC();
166 hdcOwned = true;
9ce192d4
RD
167}
168
169void Surface::Init(SurfaceID hdc_) {
170 Release();
171 hdc = hdc_;
9ce192d4
RD
172}
173
174void Surface::InitPixMap(int width, int height, Surface *surface_) {
175 Release();
176 hdc = new wxMemoryDC(surface_->hdc);
177 hdcOwned = true;
2beb01db
RD
178 if (width < 1) width = 1;
179 if (height < 1) height = 1;
21156596 180 bitmap = new wxBitmap(width, height);
9ce192d4 181 ((wxMemoryDC*)hdc)->SelectObject(*bitmap);
9ce192d4
RD
182}
183
184void Surface::PenColour(Colour fore) {
185 hdc->SetPen(wxPen(fore.co, 1, wxSOLID));
186}
187
188void Surface::BrushColor(Colour back) {
189 hdc->SetBrush(wxBrush(back.co, wxSOLID));
190}
191
192void Surface::SetFont(Font &font_) {
21156596 193 if (font_.GetID()) {
f6bcfd97
BP
194 hdc->SetFont(*font_.GetID());
195 }
9ce192d4
RD
196}
197
198int Surface::LogPixelsY() {
199 return hdc->GetPPI().y;
200}
201
f6bcfd97
BP
202
203int Surface::DeviceHeightFont(int points) {
9968ba85 204 return points;
f6bcfd97
BP
205}
206
207
9ce192d4
RD
208void Surface::MoveTo(int x_, int y_) {
209 x = x_;
210 y = y_;
211}
212
213void Surface::LineTo(int x_, int y_) {
214 hdc->DrawLine(x,y, x_,y_);
215 x = x_;
216 y = y_;
217}
218
219void Surface::Polygon(Point *pts, int npts, Colour fore,
220 Colour back) {
221 PenColour(fore);
222 BrushColor(back);
223 hdc->DrawPolygon(npts, (wxPoint*)pts);
224}
225
226void Surface::RectangleDraw(PRectangle rc, Colour fore, Colour back) {
227 PenColour(fore);
228 BrushColor(back);
229 hdc->DrawRectangle(wxRectFromPRectangle(rc));
230}
231
232void Surface::FillRectangle(PRectangle rc, Colour back) {
233 BrushColor(back);
234 hdc->SetPen(*wxTRANSPARENT_PEN);
235 hdc->DrawRectangle(wxRectFromPRectangle(rc));
236}
237
238void Surface::FillRectangle(PRectangle rc, Surface &surfacePattern) {
239 wxBrush br;
240 if (surfacePattern.bitmap)
241 br = wxBrush(*surfacePattern.bitmap);
242 else // Something is wrong so display in red
243 br = wxBrush(*wxRED, wxSOLID);
244 hdc->SetPen(*wxTRANSPARENT_PEN);
245 hdc->SetBrush(br);
246 hdc->DrawRectangle(wxRectFromPRectangle(rc));
247}
248
249void Surface::RoundedRectangle(PRectangle rc, Colour fore, Colour back) {
250 PenColour(fore);
251 BrushColor(back);
f6bcfd97 252 hdc->DrawRoundedRectangle(wxRectFromPRectangle(rc), 4);
9ce192d4
RD
253}
254
255void Surface::Ellipse(PRectangle rc, Colour fore, Colour back) {
256 PenColour(fore);
257 BrushColor(back);
258 hdc->DrawEllipse(wxRectFromPRectangle(rc));
259}
260
261void Surface::Copy(PRectangle rc, Point from, Surface &surfaceSource) {
f6bcfd97
BP
262 wxRect r = wxRectFromPRectangle(rc);
263 hdc->Blit(r.x, r.y, r.width, r.height,
9ce192d4
RD
264 surfaceSource.hdc, from.x, from.y, wxCOPY);
265}
266
267void Surface::DrawText(PRectangle rc, Font &font, int ybase,
268 const char *s, int len, Colour fore, Colour back) {
269 SetFont(font);
270 hdc->SetTextForeground(fore.co);
271 hdc->SetTextBackground(back.co);
272 FillRectangle(rc, back);
273
274 // ybase is where the baseline should be, but wxWin uses the upper left
275 // corner, so I need to calculate the real position for the text...
276 hdc->DrawText(wxString(s, len), rc.left, ybase - font.ascent);
277}
278
279void Surface::DrawTextClipped(PRectangle rc, Font &font, int ybase, const char *s, int len, Colour fore, Colour back) {
280 SetFont(font);
281 hdc->SetTextForeground(fore.co);
282 hdc->SetTextBackground(back.co);
283 FillRectangle(rc, back);
284 hdc->SetClippingRegion(wxRectFromPRectangle(rc));
285
286 // see comments above
287 hdc->DrawText(wxString(s, len), rc.left, ybase - font.ascent);
288 hdc->DestroyClippingRegion();
289}
290
291int Surface::WidthText(Font &font, const char *s, int len) {
292 SetFont(font);
293 int w;
294 int h;
295 hdc->GetTextExtent(wxString(s, len), &w, &h);
296 return w;
297}
298
299void Surface::MeasureWidths(Font &font, const char *s, int len, int *positions) {
300 SetFont(font);
301 int totalWidth = 0;
302 for (int i=0; i<len; i++) {
303 int w;
304 int h;
305 hdc->GetTextExtent(s[i], &w, &h);
306 totalWidth += w;
307 positions[i] = totalWidth;
308 }
309}
310
311int Surface::WidthChar(Font &font, char ch) {
312 SetFont(font);
313 int w;
314 int h;
315 hdc->GetTextExtent(ch, &w, &h);
316 return w;
317}
318
319#define EXTENT_TEST " `~!@#$%^&*()-_=+\\|[]{};:\"\'<,>.?/1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
320
321int Surface::Ascent(Font &font) {
322 SetFont(font);
323 int w, h, d, e;
324 hdc->GetTextExtent(EXTENT_TEST, &w, &h, &d, &e);
325 font.ascent = h - d;
326 return font.ascent;
327}
328
329int Surface::Descent(Font &font) {
330 SetFont(font);
331 int w, h, d, e;
332 hdc->GetTextExtent(EXTENT_TEST, &w, &h, &d, &e);
333 return d;
334}
335
336int Surface::InternalLeading(Font &font) {
337 return 0;
338}
339
340int Surface::ExternalLeading(Font &font) {
341 SetFont(font);
342 int w, h, d, e;
343 hdc->GetTextExtent(EXTENT_TEST, &w, &h, &d, &e);
344 return e;
345}
346
347int Surface::Height(Font &font) {
348 SetFont(font);
349 return hdc->GetCharHeight();
350}
351
352int Surface::AverageCharWidth(Font &font) {
353 SetFont(font);
354 return hdc->GetCharWidth();
355}
356
357int Surface::SetPalette(Palette *pal, bool inBackGround) {
65ec6247 358 return 0;
9ce192d4
RD
359}
360
361void Surface::SetClip(PRectangle rc) {
362 hdc->SetClippingRegion(wxRectFromPRectangle(rc));
363}
364
f6bcfd97 365void Surface::FlushCachedState() {
f6bcfd97 366}
9ce192d4
RD
367
368Window::~Window() {
369}
370
371void Window::Destroy() {
372 if (id)
373 id->Destroy();
374 id = 0;
375}
376
377bool Window::HasFocus() {
378 return wxWindow::FindFocus() == id;
379}
380
381PRectangle Window::GetPosition() {
382 wxRect rc(id->GetPosition(), id->GetSize());
383 return PRectangleFromwxRect(rc);
384}
385
386void Window::SetPosition(PRectangle rc) {
f6bcfd97
BP
387 wxRect r = wxRectFromPRectangle(rc);
388 id->SetSize(r);
9ce192d4
RD
389}
390
391void Window::SetPositionRelative(PRectangle rc, Window) {
392 SetPosition(rc); // ????
393}
394
395PRectangle Window::GetClientPosition() {
396 wxSize sz = id->GetClientSize();
21156596 397 return PRectangle(0, 0, sz.x, sz.y);
9ce192d4
RD
398}
399
400void Window::Show(bool show) {
401 id->Show(show);
402}
403
404void Window::InvalidateAll() {
405 id->Refresh(false);
406}
407
408void Window::InvalidateRectangle(PRectangle rc) {
f6bcfd97
BP
409 wxRect r = wxRectFromPRectangle(rc);
410 id->Refresh(false, &r);
9ce192d4
RD
411}
412
413void Window::SetFont(Font &font) {
414 id->SetFont(*font.GetID());
415}
416
417void Window::SetCursor(Cursor curs) {
418 int cursorId;
419
420 switch (curs) {
421 case cursorText:
422 cursorId = wxCURSOR_IBEAM;
423 break;
424 case cursorArrow:
425 cursorId = wxCURSOR_ARROW;
426 break;
427 case cursorUp:
428 cursorId = wxCURSOR_ARROW; // ** no up arrow... wxCURSOR_UPARROW;
429 break;
430 case cursorWait:
431 cursorId = wxCURSOR_WAIT;
432 break;
433 case cursorHoriz:
434 cursorId = wxCURSOR_SIZEWE;
435 break;
436 case cursorVert:
437 cursorId = wxCURSOR_SIZENS;
438 break;
439 case cursorReverseArrow:
440 cursorId = wxCURSOR_POINT_RIGHT;
441 break;
442 default:
443 cursorId = wxCURSOR_ARROW;
444 break;
445 }
446
447 id->SetCursor(wxCursor(cursorId));
448}
449
450
451void Window::SetTitle(const char *s) {
452 id->SetTitle(s);
453}
454
455
f97d84a6
RD
456class wxSTCListBox : public wxListBox {
457public:
458 wxSTCListBox(wxWindow* parent, wxWindowID id)
459 : wxListBox(parent, id, wxDefaultPosition, wxDefaultSize,
75cada8e 460 0, NULL, wxLB_SINGLE | wxSIMPLE_BORDER) // | wxLB_SORT )
f97d84a6
RD
461 {}
462
463 void OnFocus(wxFocusEvent& event) {
464 GetParent()->SetFocus();
465 event.Skip();
466 }
467
f97d84a6
RD
468private:
469 DECLARE_EVENT_TABLE()
470};
471
472BEGIN_EVENT_TABLE(wxSTCListBox, wxListBox)
473 EVT_SET_FOCUS(wxSTCListBox::OnFocus)
474END_EVENT_TABLE()
475
476
9ce192d4
RD
477ListBox::ListBox() {
478}
479
480ListBox::~ListBox() {
481}
482
483void ListBox::Create(Window &parent, int ctrlID) {
f97d84a6 484 id = new wxSTCListBox(parent.id, ctrlID);
9ce192d4
RD
485}
486
f3c2c221
RD
487void ListBox::SetVisibleRows(int rows) {
488 desiredVisibleRows = rows;
f3c2c221
RD
489}
490
d134f170
RD
491PRectangle ListBox::GetDesiredRect() {
492 wxSize sz = ((wxListBox*)id)->GetBestSize();
493 PRectangle rc;
494 rc.top = 0;
495 rc.left = 0;
f3c2c221
RD
496 if (sz.x > 400)
497 sz.x = 400;
9af175d2
RD
498 if (sz.y > 160) // TODO: Use desiredVisibleRows??
499 sz.y = 160;
d134f170
RD
500 rc.right = sz.x;
501 rc.bottom = sz.y;
d134f170
RD
502 return rc;
503}
504
505void ListBox::SetAverageCharWidth(int width) {
506 aveCharWidth = width;
507}
508
509void ListBox::SetFont(Font &font) {
510 Window::SetFont(font);
511}
512
9ce192d4
RD
513void ListBox::Clear() {
514 ((wxListBox*)id)->Clear();
515}
516
517void ListBox::Append(char *s) {
518 ((wxListBox*)id)->Append(s);
519}
520
521int ListBox::Length() {
257ed895 522 return ((wxListBox*)id)->GetCount();
9ce192d4
RD
523}
524
525void ListBox::Select(int n) {
526 ((wxListBox*)id)->SetSelection(n);
f97d84a6
RD
527#ifdef __WXGTK__
528 if (n > 4)
529 n = n - 4;
530 else
531 n = 1;
532 ((wxListBox*)id)->SetFirstItem(n);
533#endif
9ce192d4
RD
534}
535
536int ListBox::GetSelection() {
537 return ((wxListBox*)id)->GetSelection();
538}
539
540int ListBox::Find(const char *prefix) {
b8b0e402 541 // No longer used
f6bcfd97 542 return -1;
9ce192d4
RD
543}
544
545void ListBox::GetValue(int n, char *value, int len) {
546 wxString text = ((wxListBox*)id)->GetString(n);
547 strncpy(value, text.c_str(), len);
548 value[len-1] = '\0';
549}
550
551void ListBox::Sort() {
552 // wxWindows keeps sorted so no need to sort
553}
554
555
556Menu::Menu() : id(0) {
557}
558
559void Menu::CreatePopUp() {
560 Destroy();
561 id = new wxMenu();
562}
563
564void Menu::Destroy() {
565 if (id)
566 delete id;
567 id = 0;
568}
569
570void Menu::Show(Point pt, Window &w) {
571 w.GetID()->PopupMenu(id, pt.x - 4, pt.y);
572 Destroy();
573}
574
575
576Colour Platform::Chrome() {
577 wxColour c;
e1c6c6ae 578 c = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
9ce192d4
RD
579 return Colour(c.Red(), c.Green(), c.Blue());
580}
581
582Colour Platform::ChromeHighlight() {
583 wxColour c;
e1c6c6ae 584 c = wxSystemSettings::GetColour(wxSYS_COLOUR_3DHIGHLIGHT);
9ce192d4
RD
585 return Colour(c.Red(), c.Green(), c.Blue());
586}
587
588const char *Platform::DefaultFont() {
589 return wxNORMAL_FONT->GetFaceName();
590}
591
592int Platform::DefaultFontSize() {
593 return 8;
594}
595
596unsigned int Platform::DoubleClickTime() {
597 return 500; // **** ::GetDoubleClickTime();
598}
599
600void Platform::DebugDisplay(const char *s) {
601 wxLogDebug(s);
602}
603
604bool Platform::IsKeyDown(int key) {
605 return false; // I don't think we'll need this.
606}
607
608long Platform::SendScintilla(WindowID w,
609 unsigned int msg,
610 unsigned long wParam,
611 long lParam) {
612
613 wxStyledTextCtrl* stc = (wxStyledTextCtrl*)w;
614 return stc->SendMsg(msg, wParam, lParam);
615}
616
617
618// These are utility functions not really tied to a platform
619
620int Platform::Minimum(int a, int b) {
621 if (a < b)
622 return a;
623 else
624 return b;
625}
626
627int Platform::Maximum(int a, int b) {
628 if (a > b)
629 return a;
630 else
631 return b;
632}
633
634#define TRACE
635
636void Platform::DebugPrintf(const char *format, ...) {
637#ifdef TRACE
638 char buffer[2000];
639 va_list pArguments;
640 va_start(pArguments, format);
641 vsprintf(buffer,format,pArguments);
642 va_end(pArguments);
643 Platform::DebugDisplay(buffer);
644#endif
645}
646
65ec6247
RD
647
648static bool assertionPopUps = true;
649
650bool Platform::ShowAssertionPopUps(bool assertionPopUps_) {
651 bool ret = assertionPopUps;
652 assertionPopUps = assertionPopUps_;
653 return ret;
654}
655
656void Platform::Assert(const char *c, const char *file, int line) {
657 char buffer[2000];
658 sprintf(buffer, "Assertion [%s] failed at %s %d", c, file, line);
659 if (assertionPopUps) {
660 int idButton = wxMessageBox(buffer, "Assertion failure",
661 wxICON_HAND | wxOK);
662// if (idButton == IDRETRY) {
663// ::DebugBreak();
664// } else if (idButton == IDIGNORE) {
665// // all OK
666// } else {
667// abort();
668// }
669 } else {
670 strcat(buffer, "\r\n");
671 Platform::DebugDisplay(buffer);
672 abort();
673 }
674}
675
676
9ce192d4
RD
677int Platform::Clamp(int val, int minVal, int maxVal) {
678 if (val > maxVal)
679 val = maxVal;
680 if (val < minVal)
681 val = minVal;
682 return val;
683}
684
685
686
687
688
689