1 /*-*- c++ -*-********************************************************
2 * wxLwindow.h : a scrolled Window for displaying/entering rich text*
4 * (C) 1998, 1999 by Karsten Ballüder (Ballueder@usa.net) *
7 *******************************************************************/
10 # pragma implementation "wxlwindow.h"
18 # include "gui/wxMenuDefs.h"
19 # include "gui/wxMApp.h"
21 # include "gui/wxlwindow.h"
31 # include "wxlwindow.h"
32 # define TRACEMESSAGE(x)
34 # define WXL_VAR(x) { wxString s; s << #x " = " << x; wxLogDebug(s); }
36 BEGIN_EVENT_TABLE(wxLayoutWindow
,wxScrolledWindow
)
37 EVT_PAINT (wxLayoutWindow::OnPaint
)
38 EVT_CHAR (wxLayoutWindow::OnChar
)
40 EVT_LEFT_DOWN(wxLayoutWindow::OnLeftMouseClick
)
41 EVT_RIGHT_DOWN(wxLayoutWindow::OnRightMouseClick
)
42 EVT_LEFT_DCLICK(wxLayoutWindow::OnMouseDblClick
)
43 EVT_MENU(WXLOWIN_MENU_LARGER
, wxLayoutWindow::OnMenu
)
44 EVT_MENU(WXLOWIN_MENU_SMALLER
, wxLayoutWindow::OnMenu
)
45 EVT_MENU(WXLOWIN_MENU_UNDERLINE
, wxLayoutWindow::OnMenu
)
46 EVT_MENU(WXLOWIN_MENU_BOLD
, wxLayoutWindow::OnMenu
)
47 EVT_MENU(WXLOWIN_MENU_ITALICS
, wxLayoutWindow::OnMenu
)
48 EVT_MENU(WXLOWIN_MENU_ROMAN
, wxLayoutWindow::OnMenu
)
49 EVT_MENU(WXLOWIN_MENU_TYPEWRITER
, wxLayoutWindow::OnMenu
)
50 EVT_MENU(WXLOWIN_MENU_SANSSERIF
, wxLayoutWindow::OnMenu
)
51 EVT_SET_FOCUS(wxLayoutWindow::OnSetFocus
)
52 EVT_KILL_FOCUS(wxLayoutWindow::OnKillFocus
)
55 wxLayoutWindow::wxLayoutWindow(wxWindow
*parent
)
56 : wxScrolledWindow(parent
, -1, wxDefaultPosition
, wxDefaultSize
,
57 wxHSCROLL
| wxVSCROLL
| wxBORDER
)
60 m_doSendEvents
= false;
61 m_ViewStartX
= 0; m_ViewStartY
= 0;
64 m_memDC
= new wxMemoryDC
;
65 m_bitmap
= new wxBitmap(4,4);
66 m_bitmapSize
= wxPoint(4,4);
69 max_x
, max_y
, lineHeight
;
70 m_llist
.GetSize(&max_x
, &max_y
, &lineHeight
);
71 SetScrollbars(10, lineHeight
, max_x
/10+1, max_y
/lineHeight
+1);
72 EnableScrolling(true,true);
73 m_maxx
= max_x
; m_maxy
= max_y
; m_lineHeight
= lineHeight
;
76 wxLayoutWindow::~wxLayoutWindow()
78 delete m_memDC
; // deletes bitmap automatically (?)
80 if(m_PopupMenu
) delete m_PopupMenu
;
85 wxLayoutWindow::MSWGetDlgCode()
87 // if we don't return this, we won't get OnChar() events for TABs and ENTER
88 return DLGC_WANTCHARS
| DLGC_WANTARROWS
| DLGC_WANTMESSAGE
;
93 wxLayoutWindow::Update(void)
101 DoPaint(true); // only the cursor
106 wxLayoutWindow::OnMouse(int eventId
, wxMouseEvent
& event
)
108 if(!m_doSendEvents
) // nothing to do
111 wxPaintDC
dc( this );
116 findPos
.x
= dc
.DeviceToLogicalX(event
.GetX());
117 findPos
.y
= dc
.DeviceToLogicalY(event
.GetY());
119 TRACEMESSAGE(("wxLayoutWindow::OnMouse: (%d, %d) -> (%d, %d)",
120 event
.GetX(), event
.GetY(), findPos
.x
, findPos
.y
));
122 m_ClickPosition
= findPos
;
123 wxLayoutObjectBase
*obj
= m_llist
.Find(findPos
);
125 // only do the menu if activated, editable and not on a clickable object
126 if(eventId
== WXLOWIN_MENU_RCLICK
128 && m_llist
.IsEditable()
129 && obj
&& obj
->GetUserData() == NULL
)
131 // when does this menu get freed?
132 // how do we handle toggling? FIXME
133 PopupMenu(MakeFormatMenu(), event
.GetX(), event
.GetY());
136 // find the object at this position
139 wxCommandEvent
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, eventId
);
140 commandEvent
.SetEventObject( this );
141 commandEvent
.SetClientData((char *)obj
);
142 GetEventHandler()->ProcessEvent(commandEvent
);
147 wxLayoutWindow::DeleteToEndOfLine(void)
149 int help
= m_llist
.GetLineLength(
150 m_llist
.GetCurrentObject())
151 - m_llist
.GetCursor().x
;
152 m_llist
.Delete(help
>1 ? help
-1 : 1);
156 wxLayoutWindow::GotoEndOfLine(void)
158 wxPoint p
= m_llist
.GetCursor();
159 p
.x
= m_llist
.GetLineLength(m_llist
.GetCurrentObject());
160 if(p
.x
> 0) p
.x
--; // do not count the linebreak
161 m_llist
.SetCursor(p
);
165 wxLayoutWindow::GotoBeginOfLine(void)
167 wxPoint p
= m_llist
.GetCursor();
169 m_llist
.SetCursor(p
);
173 wxLayoutWindow::DeleteLine(void)
177 m_llist
.Delete(1); // newline
181 wxLayoutWindow::DeleteToBeginOfLine(void)
183 wxPoint p
= m_llist
.GetCursor();
188 m_llist
.SetCursor(p
);
189 m_llist
.Delete(count
);
195 wxLayoutWindow::ScrollToCursor(void)
197 /** Scroll so that cursor is visible! */
198 int x0
,y0
,x1
,y1
,ux
,uy
;
200 GetScrollPixelsPerUnit(&ux
,&uy
);
202 GetClientSize(&x1
,&y1
);
204 wxPoint cc
= m_llist
.GetCursorCoords();
206 if(cc
.x
< x0
|| cc
.y
< y0
207 || cc
.x
>= x0
+(9*x1
)/10 || cc
.y
>= y0
+(9*y1
/10)) // (9*x)/10 == 90%
210 nx
= cc
.x
- (8*x1
)/10; if(nx
< 0) nx
= 0;
211 ny
= cc
.y
- (8*y1
)/10; if(ny
< 0) ny
= 0;
217 * some simple keyboard handling
220 wxLayoutWindow::OnChar(wxKeyEvent
& event
)
222 if(! m_llist
.IsEditable()) // do nothing
228 long keyCode
= event
.KeyCode();
230 /* First, handle control keys */
231 if(event
.ControlDown())
233 switch(event
.KeyCode())
237 DeleteToEndOfLine(); break;
239 m_llist
.Delete(1); break;
242 case 'h': // like backspace
243 if(m_llist
.MoveCursor(-1))
247 DeleteToBeginOfLine(); break;
252 else // no control keys
254 switch(event
.KeyCode())
257 m_llist
.MoveCursor(1);
260 m_llist
.MoveCursor(-1);
263 m_llist
.MoveCursor(0,-1);
266 m_llist
.MoveCursor(0,1);
269 m_llist
.MoveCursor(0,-20);
272 m_llist
.MoveCursor(0,20);
281 if(event
.ControlDown()) // delete to end of line
286 case WXK_BACK
: // backspace
287 if(m_llist
.MoveCursor(-1)) {
295 #ifdef WXLAYOUT_DEBUG
305 if((!(event
.ControlDown() || event
.AltDown() || event
.MetaDown()))
306 && (keyCode
< 256 && keyCode
>= 32)
324 wxLayoutWindow::OnPaint( wxPaintEvent
&WXUNUSED(event
)) // or: OnDraw(wxDC& dc)
330 wxLayoutWindow::DoPaint(bool cursorOnly
) // or: OnDraw(wxDC& dc)
332 wxPaintDC
dc( this );
335 // wxGTK: wxMemoryDC broken? YES!!
336 int x0
,y0
,x1
,y1
, dx
, dy
;
338 GetClientSize(&x1
,&y1
); // this is the size of the visible window
341 GetScrollPixelsPerUnit(&dx
, &dy
);
343 //FIXME: trying an offset for small border:
344 wxPoint
offset(-x0
+4,-y0
+4);
346 //Blit() doesn't work on scrolled window!
347 // So we have to draw the cursor on the memdc.
350 if(x1
> m_bitmapSize
.x
|| y1
> m_bitmapSize
.y
)
352 wxASSERT(m_bitmapSize
.x
> 0);
353 wxASSERT(m_bitmapSize
.y
> 0);
355 m_memDC
->SelectObject(wxNullBitmap
);
357 m_bitmapSize
= wxPoint(x1
,y1
);
358 m_bitmap
= new wxBitmap(x1
,y1
);
359 m_memDC
->SelectObject(*m_bitmap
);
361 m_memDC
->SetDeviceOrigin(0,0);
363 if(IsDirty() || m_llist
.CursorMoved())
366 m_llist
.EraseAndDraw(*m_memDC
,
367 wxLayoutObjectList::iterator(NULL
),offset
);
368 m_llist
.DrawCursor(*m_memDC
,false,offset
);
369 dc
.Blit(x0
,y0
,x1
,y1
,m_memDC
,0,0,wxCOPY
,FALSE
);
372 //FIXME obsolete? ResetDirty();
376 // change the range and position of scroll bars
378 wxLayoutWindow::UpdateScrollbars(bool exact
)
381 max_x
, max_y
, lineHeight
;
383 m_llist
.GetSize(&max_x
, &max_y
, &lineHeight
);
385 if(max_x
> m_maxx
|| max_y
> m_maxy
|| exact
)
387 if(! exact
) // add an extra 50% to the sizes to avoid future updates
392 ViewStart(&m_ViewStartX
, &m_ViewStartY
);
393 SetScrollbars(10, 20, max_x
/10+1,max_y
/20+1,m_ViewStartX
,m_ViewStartY
,true);
394 m_maxx
= max_x
; m_maxy
= max_y
;
399 wxLayoutWindow::Print(wxDC
&dc
)
401 if (dc
.Ok() && dc
.StartDoc((char *)_("Printing message...")))
403 //dc.SetUserScale(1.0, 1.0);
410 wxLayoutWindow::MakeFormatMenu()
415 wxMenu
*m
= new wxMenu();
417 m
->Append(WXLOWIN_MENU_LARGER
,_("&Larger"),_("Switch to larger font."), false);
418 m
->Append(WXLOWIN_MENU_SMALLER
,_("&Smaller"),_("Switch to smaller font."), false);
419 m
->AppendSeparator();
420 m
->Append(WXLOWIN_MENU_UNDERLINE
,_("&Underline"),_("Toggle underline mode."), true);
421 m
->Append(WXLOWIN_MENU_BOLD
,_("&Bold"),_("Toggle bold mode."), true);
422 m
->Append(WXLOWIN_MENU_ITALICS
,_("&Italics"),_("Toggle italics mode."), true);
423 m
->AppendSeparator();
424 m
->Append(WXLOWIN_MENU_ROMAN
,_("&Roman"),_("Toggle underline mode."), false);
425 m
->Append(WXLOWIN_MENU_TYPEWRITER
,_("&Typewriter"),_("Toggle bold mode."), false);
426 m
->Append(WXLOWIN_MENU_SANSSERIF
,_("&Sans Serif"),_("Toggle italics mode."), false);
428 return m_PopupMenu
= m
;
431 void wxLayoutWindow::OnMenu(wxCommandEvent
& event
)
433 if(! m_llist
.IsEditable())
436 switch (event
.GetId())
438 case WXLOWIN_MENU_LARGER
:
439 m_llist
.SetFontLarger();
441 case WXLOWIN_MENU_SMALLER
:
442 m_llist
.SetFontSmaller();
444 case WXLOWIN_MENU_UNDERLINE
:
445 m_llist
.SetFontUnderline(
446 m_PopupMenu
->IsChecked(WXLOWIN_MENU_UNDERLINE
) ? false : true
449 case WXLOWIN_MENU_BOLD
:
450 m_llist
.SetFontWeight(
451 m_PopupMenu
->IsChecked(WXLOWIN_MENU_BOLD
) ? wxNORMAL
: wxBOLD
453 case WXLOWIN_MENU_ITALICS
:
454 m_llist
.SetFontStyle(
455 m_PopupMenu
->IsChecked(WXLOWIN_MENU_ITALICS
) ? wxNORMAL
: wxITALIC
458 case WXLOWIN_MENU_ROMAN
:
459 m_llist
.SetFontFamily(wxROMAN
); break;
460 case WXLOWIN_MENU_TYPEWRITER
:
461 m_llist
.SetFontFamily(wxFIXED
); break;
462 case WXLOWIN_MENU_SANSSERIF
:
463 m_llist
.SetFontFamily(wxSWISS
); break;
468 wxLayoutWindow::OnSetFocus(wxFocusEvent
&ev
)
470 m_llist
.SetBoldCursor(true);
475 wxLayoutWindow::OnKillFocus(wxFocusEvent
&ev
)
477 m_llist
.SetBoldCursor(false);