1 /*-*- c++ -*-******************************************************** 
   2  * wxLwindow.h : a scrolled Window for displaying/entering rich text* 
   4  * (C) 1998-2000 by Karsten Ballüder (ballueder@gmx.net)            * 
   7  *******************************************************************/ 
   9 // ============================================================================ 
  11 // ============================================================================ 
  13 // ---------------------------------------------------------------------------- 
  15 // ---------------------------------------------------------------------------- 
  18 #   pragma implementation "wxlwindow.h" 
  21 #include "wx/wxprec.h" 
  32 #     include "gui/wxMenuDefs.h" 
  33 #     include "gui/wxMApp.h" 
  35 #   include "gui/wxlwindow.h" 
  36 #   include "gui/wxlparser.h" 
  38 #   include "MDialogs.h" 
  42 #       include <wx/msw/private.h> 
  45 #   include "wxlwindow.h" 
  46 #   include "wxlparser.h" 
  49 #include <wx/clipbrd.h> 
  50 #include <wx/textctrl.h> 
  51 #include <wx/dataobj.h> 
  53 #ifdef WXLAYOUT_USE_CARET 
  54 #   include <wx/caret.h> 
  55 #endif // WXLAYOUT_USE_CARET 
  60 // ---------------------------------------------------------------------------- 
  62 // ---------------------------------------------------------------------------- 
  65 #  define   WXLO_DEBUG(x)      wxLogDebug x 
  67 #  define WXLO_DEBUG(x) 
  70 // for profiling in debug mode: 
  71 WXLO_TIMER_DEFINE(UpdateTimer
); 
  72 WXLO_TIMER_DEFINE(BlitTimer
); 
  73 WXLO_TIMER_DEFINE(LayoutTimer
); 
  74 WXLO_TIMER_DEFINE(TmpTimer
); 
  75 WXLO_TIMER_DEFINE(DrawTimer
); 
  77 // ---------------------------------------------------------------------------- 
  79 // ---------------------------------------------------------------------------- 
  81 /// offsets to put a nice frame around text 
  82 #define WXLO_XOFFSET   4 
  83 #define WXLO_YOFFSET   4 
  85 /// offset to the right and bottom for when to redraw scrollbars 
  86 #define   WXLO_ROFFSET   20 
  87 #define   WXLO_BOFFSET   20 
  89 /// scroll margins when selecting with the mouse 
  90 #define WXLO_SCROLLMARGIN_X   10 
  91 #define WXLO_SCROLLMARGIN_Y   10 
  93 /// the size of one scrollbar page in pixels 
  94 static const int X_SCROLL_PAGE 
= 10; 
  95 static const int Y_SCROLL_PAGE 
= 20; 
  99 // ---------------------------------------------------------------------------- 
 101 // ---------------------------------------------------------------------------- 
 103 BEGIN_EVENT_TABLE(wxLayoutWindow
,wxScrolledWindow
) 
 104    EVT_SIZE    (wxLayoutWindow::OnSize
) 
 106    EVT_PAINT    (wxLayoutWindow::OnPaint
) 
 108    EVT_CHAR     (wxLayoutWindow::OnChar
) 
 109    EVT_KEY_UP   (wxLayoutWindow::OnKeyUp
) 
 111    EVT_LEFT_DOWN(wxLayoutWindow::OnLeftMouseDown
) 
 112    EVT_LEFT_UP(wxLayoutWindow::OnLeftMouseUp
) 
 113    EVT_RIGHT_DOWN(wxLayoutWindow::OnRightMouseClick
) 
 114    EVT_LEFT_DCLICK(wxLayoutWindow::OnMouseDblClick
) 
 115    EVT_MIDDLE_DOWN(wxLayoutWindow::OnMiddleMouseDown
) 
 116    EVT_MOTION    (wxLayoutWindow::OnMouseMove
) 
 118    EVT_UPDATE_UI(WXLOWIN_MENU_UNDERLINE
, wxLayoutWindow::OnUpdateMenuUnderline
) 
 119    EVT_UPDATE_UI(WXLOWIN_MENU_BOLD
, wxLayoutWindow::OnUpdateMenuBold
) 
 120    EVT_UPDATE_UI(WXLOWIN_MENU_ITALICS
, wxLayoutWindow::OnUpdateMenuItalic
) 
 121    EVT_MENU_RANGE(WXLOWIN_MENU_FIRST
, WXLOWIN_MENU_LAST
, wxLayoutWindow::OnMenu
) 
 123    EVT_SET_FOCUS(wxLayoutWindow::OnSetFocus
) 
 124    EVT_KILL_FOCUS(wxLayoutWindow::OnKillFocus
) 
 126 //   EVT_IDLE(wxLayoutWindow::ResizeScrollbars) 
 129 // ---------------------------------------------------------------------------- 
 130 // function prototypes 
 131 // ---------------------------------------------------------------------------- 
 133 /// returns TRUE if keyCode is one of arrows/home/end/page{up|down} keys 
 134 static bool IsDirectionKey(long keyCode
); 
 136 // ============================================================================ 
 138 // ============================================================================ 
 140 #ifndef wxWANTS_CHARS 
 141 #   define wxWANTS_CHARS 0 
 144 // ---------------------------------------------------------------------------- 
 146 // ---------------------------------------------------------------------------- 
 148 wxLayoutWindow::wxLayoutWindow(wxWindow 
*parent
) 
 149               : wxScrolledWindow(parent
, -1, 
 150                                  wxDefaultPosition
, wxDefaultSize
, 
 151                                  wxHSCROLL 
| wxVSCROLL 
| 
 156    SetStatusBar(NULL
); // don't use statusbar 
 158    m_doSendEvents 
= false; 
 159    m_ViewStartX 
= 0; m_ViewStartY 
= 0; 
 160    m_DoPopupMenu 
= true; 
 161    m_PopupMenu 
= MakeFormatMenu(); 
 162    m_memDC 
= new wxMemoryDC
; 
 163    m_bitmap 
= new wxBitmap(4,4); 
 164    m_bitmapSize 
= wxPoint(4,4); 
 165    m_llist 
= new wxLayoutList(); 
 167    m_ScrollToCursor 
= false; 
 169    m_FocusFollowMode 
= false; 
 174    // no scrollbars initially 
 176    m_hasVScrollbar 
= false; 
 180 #ifdef WXLAYOUT_USE_CARET 
 181    // FIXME cursor size shouldn't be hardcoded 
 182    wxCaret 
*caret 
= new wxCaret(this, 2, 20); 
 184    m_llist
->SetCaret(caret
); 
 185 #endif // WXLAYOUT_USE_CARET 
 188    m_HandCursor 
= FALSE
; 
 189    m_CursorVisibility 
= -1; 
 190    SetCursor(wxCURSOR_IBEAM
); 
 193    // at least under Windows, this should be the default behaviour 
 194    m_AutoDeleteSelection 
= TRUE
; 
 197 wxLayoutWindow::~wxLayoutWindow() 
 199    delete m_memDC
; // deletes bitmap automatically (?) 
 203    SetBackgroundBitmap(NULL
); 
 207 wxLayoutWindow::Clear(int family
, 
 215    GetLayoutList()->Clear(family
,size
,style
,weight
,underline
,fg
,bg
); 
 216    SetBackgroundColour(GetLayoutList()->GetDefaultStyleInfo().GetBGColour()); 
 217    //wxScrolledWindow::Clear(); 
 218    ResizeScrollbars(true); 
 222       m_CursorVisibility 
= 1; 
 224 #ifdef WXLAYOUT_USE_CARET 
 225    if ( m_CursorVisibility 
== 1 ) 
 227 #endif // WXLAYOUT_USE_CARET 
 229    RequestUpdate((wxRect 
*)NULL
); 
 232 void wxLayoutWindow::Refresh(bool eraseBackground
, const wxRect 
*rect
) 
 234    wxScrolledWindow::Refresh(eraseBackground
, rect
); 
 238 wxLayoutWindow::OnMouse(int eventId
, wxMouseEvent
& event
) 
 240    wxClientDC 
dc( this ); 
 242    if ( (eventId 
!= WXLOWIN_MENU_MOUSEMOVE
 
 246         ) && (wxWindow::FindFocus() != this) 
 251    findPos
.x 
= dc
.DeviceToLogicalX(event
.GetX()); 
 252    findPos
.y 
= dc
.DeviceToLogicalY(event
.GetY()); 
 254    findPos
.x 
-= WXLO_XOFFSET
; 
 255    findPos
.y 
-= WXLO_YOFFSET
; 
 262    m_ClickPosition 
= wxPoint(event
.GetX(), event
.GetY()); 
 264    // Scroll the window if the mouse is at the end of it: 
 265    if(m_Selecting 
&& eventId 
== WXLOWIN_MENU_MOUSEMOVE
) 
 267       //WXLO_DEBUG(("selecting at : %d/%d", (int) event.GetX(), (int)event.GetY())); 
 269       GetViewStart(&left
, &top
); 
 270       wxSize size 
= GetClientSize(); 
 273       if(event
.GetX() < WXLO_SCROLLMARGIN_X
) 
 274          xdelta 
= -(WXLO_SCROLLMARGIN_X
-event
.GetX()); 
 275       else if(event
.GetX() > size
.x
-WXLO_SCROLLMARGIN_X
) 
 276          xdelta 
= event
.GetX()-size
.x
+WXLO_SCROLLMARGIN_X
; 
 279       if(event
.GetY() < WXLO_SCROLLMARGIN_Y
) 
 280          ydelta 
= -(WXLO_SCROLLMARGIN_Y
-event
.GetY()); 
 281       else if(event
.GetY() > size
.y
-WXLO_SCROLLMARGIN_Y
) 
 282          ydelta 
= event
.GetY()-size
.y
+WXLO_SCROLLMARGIN_Y
; 
 286       //WXLO_DEBUG(("xdelta: %d", (int) xdelta)); 
 287       if(xdelta 
!= 0 || ydelta 
!= 0) 
 289          top  
+= ydelta
; if(top 
< 0) top 
= 0; 
 290          left 
+= xdelta
; if(left 
< 0) left 
= 0; 
 297    wxLayoutObject 
*obj 
= m_llist
->FindObjectScreen(dc
, findPos
, 
 299    wxLayoutObject::UserData 
*u 
= obj 
? obj
->GetUserData() : NULL
; 
 301    // has the mouse only been moved? 
 304    case WXLOWIN_MENU_MOUSEMOVE
: 
 306       // this variables is used to only erase the message in the status 
 307       // bar if we had put it there previously - otherwise empting status 
 308       // bar might be undesirable 
 309       static bool s_hasPutMessageInStatusBar 
= false; 
 311       // found is only true if we are really over an object, not just 
 313       if(found 
&& u 
&& ! m_Selecting
) 
 316             SetCursor(wxCURSOR_HAND
); 
 318          if(m_StatusBar 
&& m_StatusFieldLabel 
!= -1) 
 320             const wxString 
&label 
= u
->GetLabel(); 
 323                m_StatusBar
->SetStatusText(label
, 
 325                s_hasPutMessageInStatusBar 
= true; 
 332             SetCursor(wxCURSOR_IBEAM
); 
 333          m_HandCursor 
= FALSE
; 
 334          if( m_StatusBar 
&& m_StatusFieldLabel 
!= -1 && 
 335              s_hasPutMessageInStatusBar 
) 
 337             m_StatusBar
->SetStatusText(wxEmptyString
, m_StatusFieldLabel
); 
 343    if ( event
.LeftIsDown() ) 
 345       // m_Selecting might not be set if the button got pressed 
 346       // outside this window, so check for it: 
 349          m_llist
->ContinueSelection(cursorPos
, m_ClickPosition
); 
 350          RequestUpdate();  // TODO: we don't have to redraw everything! 
 361    case WXLOWIN_MENU_LDOWN
: 
 363       // always move cursor to mouse click: 
 364       m_llist
->MoveCursorTo(cursorPos
); 
 366       // clicking a mouse removes the selection 
 367       if ( m_llist
->HasSelection() ) 
 369          m_llist
->DiscardSelection(); 
 371          RequestUpdate();     // TODO: we don't have to redraw everything! 
 374       // Calculate where the top of the visible area is: 
 376       GetViewStart(&x0
,&y0
); 
 378       GetScrollPixelsPerUnit(&dx
, &dy
); 
 381       wxPoint 
offset(-x0
+WXLO_XOFFSET
, -y0
+WXLO_YOFFSET
); 
 383       if(m_CursorVisibility 
== -1) 
 384          m_CursorVisibility 
= 1; 
 385 #ifdef WXLAYOUT_USE_CARET 
 386       if ( m_CursorVisibility 
== 1 ) 
 388 #endif // WXLAYOUT_USE_CARET 
 390       if(m_CursorVisibility
) 
 392          // draw a thick cursor for editable windows with focus 
 393          m_llist
->DrawCursor(dc
, m_HaveFocus 
&& IsEditable(), offset
); 
 397       RequestUpdate(); // RequestUpdate suppresses flicker under GTK 
 401       m_llist
->StartSelection(wxPoint(-1, -1), m_ClickPosition
); 
 406    case WXLOWIN_MENU_LUP
: 
 409          // end selection at the cursor position corresponding to the 
 410          // current mouse position, but don´t move cursor there. 
 411          m_llist
->EndSelection(cursorPos
,m_ClickPosition
); 
 414          RequestUpdate();     // TODO: we don't have to redraw everything! 
 418    case WXLOWIN_MENU_MDOWN
: 
 422    case WXLOWIN_MENU_DBLCLICK
: 
 423       // select a word under cursor 
 424       m_llist
->MoveCursorTo(cursorPos
); 
 425       m_llist
->MoveCursorWord(-1); 
 426       m_llist
->StartSelection(); 
 427       m_llist
->MoveCursorWord(1, false); 
 428       m_llist
->EndSelection(); 
 430       RequestUpdate();     // TODO: we don't have to redraw everything! 
 434    // notify about mouse events? 
 437       // only do the menu if activated, editable and not on a clickable object 
 438       if(eventId 
== WXLOWIN_MENU_RCLICK
 
 440          && (! obj 
|| u 
== NULL
)) 
 442          PopupMenu(m_PopupMenu
, m_ClickPosition
.x
, m_ClickPosition
.y
); 
 447       // find the object at this position 
 450          wxCommandEvent 
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, eventId
); 
 451          commandEvent
.SetEventObject( this ); 
 452          commandEvent
.SetClientData((char *)obj
); 
 453          GetEventHandler()->ProcessEvent(commandEvent
); 
 460 // ---------------------------------------------------------------------------- 
 461 // keyboard handling. 
 462 // ---------------------------------------------------------------------------- 
 465 wxLayoutWindow::OnChar(wxKeyEvent
& event
) 
 467    int keyCode 
= event
.KeyCode(); 
 468    bool ctrlDown 
= event
.ControlDown(); 
 470 #ifdef WXLAYOUT_DEBUG 
 471    if(keyCode 
== WXK_F1
) 
 478    // Force m_Selecting to be false if shift is no longer 
 479    // pressed. OnKeyUp() cannot catch all Shift-Up events. 
 480    if(m_Selecting 
&& !event
.ShiftDown()) 
 483       m_llist
->EndSelection(); 
 484       m_llist
->DiscardSelection(); //FIXME: correct? 
 487    // If we deleted the selection here, we must not execute the 
 488    // deletion in Delete/Backspace handling. 
 489    bool deletedSelection 
= false; 
 490    // pressing any non-arrow key optionally replaces the selection: 
 491    if(m_AutoDeleteSelection
 
 494       && m_llist
->HasSelection()  
 495       && ! IsDirectionKey(keyCode
) 
 496       && ! (event
.AltDown() || ctrlDown
) 
 499       m_llist
->DeleteSelection(); 
 500       deletedSelection 
= true; 
 504    // <Shift>+<arrow> starts selection 
 505    if ( IsDirectionKey(keyCode
) ) 
 507       // just continue the old selection 
 508       if ( m_Selecting 
&& event
.ShiftDown() ) 
 509             m_llist
->ContinueSelection(); 
 512          m_llist
->DiscardSelection(); 
 514          if( event
.ShiftDown() ) 
 517             m_llist
->StartSelection(); 
 522    // If needed, make cursor visible: 
 523    if(m_CursorVisibility 
== -1) 
 524       m_CursorVisibility 
= 1; 
 526    /* These two nested switches work like this: 
 527       The first one processes all non-editing keycodes, to move the 
 528       cursor, etc. It's default will process all keycodes causing 
 529       modifications to the buffer, but only if editing is allowed. 
 535          m_llist
->MoveCursorWord(1); 
 537          m_llist
->MoveCursorHorizontally(1); 
 541          m_llist
->MoveCursorWord(-1); 
 543          m_llist
->MoveCursorHorizontally(-1); 
 546       m_llist
->MoveCursorVertically(-1); 
 549       m_llist
->MoveCursorVertically(1); 
 552       m_llist
->MoveCursorVertically(-Y_SCROLL_PAGE
); 
 555       m_llist
->MoveCursorVertically(Y_SCROLL_PAGE
); 
 559          m_llist
->MoveCursorTo(wxPoint(0, 0)); 
 561          m_llist
->MoveCursorToBeginOfLine(); 
 565          m_llist
->MoveCursorToEnd(); 
 567          m_llist
->MoveCursorToEndOfLine(); 
 571       if(ctrlDown 
&& ! IsEditable()) 
 575             // this should work even in read-only mode 
 581          case 't': // search again 
 585             // we don't handle it, maybe an accelerator? 
 589       else if( IsEditable() ) 
 591          /* First, handle control keys */ 
 592          if(ctrlDown 
&& ! event
.AltDown()) 
 594             if(keyCode 
>= 'A' && keyCode 
<= 'Z') 
 595                keyCode 
= tolower(keyCode
); 
 602                if(! deletedSelection
) 
 604                   m_llist
->DeleteWord(); 
 609                if(! deletedSelection
) // already done 
 616                m_llist
->DeleteLines(1); 
 619             case 'h': // like backspace 
 620                if(m_llist
->MoveCursorHorizontally(-1)) 
 629             case 't': // search again 
 633                m_llist
->DeleteToBeginOfLine(); 
 637                m_llist
->DeleteToEndOfLine(); 
 651                   m_llist
->WrapLine(m_WrapMargin
); 
 655                   m_llist
->WrapAll(m_WrapMargin
); 
 657 #ifdef WXLAYOUT_DEBUG 
 659                m_llist
->SetFont(-1,-1,-1,-1,true);  // underlined 
 666             // we don't handle it, maybe an accelerator? 
 671          else if( event
.AltDown() && ! event
.ControlDown() ) 
 677                m_llist
->DeleteWord(); 
 681                // we don't handle it, maybe an accelerator? 
 686          else if ( ! event
.AltDown() && ! event
.ControlDown()) 
 691                if(event
.ShiftDown()) 
 695                if(event
.ShiftDown()) 
 698                   if(! deletedSelection
) 
 704             case WXK_BACK
: // backspace 
 705                if(! deletedSelection
) 
 706                   if(m_llist
->MoveCursorHorizontally(-1)) 
 715                   && m_llist
->GetCursorPos().x 
> m_WrapMargin
) 
 716                   m_llist
->WrapLine(m_WrapMargin
); 
 717                m_llist
->LineBreak(); 
 722                if ( !event
.ShiftDown() ) 
 724                   // TODO should be configurable 
 725                   static const int tabSize 
= 8; 
 727                   CoordType x 
= m_llist
->GetCursorPos().x
; 
 728                   size_t numSpaces 
= tabSize 
- x 
% tabSize
; 
 729                   m_llist
->Insert(wxString(' ', numSpaces
)); 
 735                if((!(event
.ControlDown() || event
.AltDown() 
 737                   && (keyCode 
< 256 && keyCode 
>= 32) 
 742                      && m_llist
->GetCursorPos().x 
> m_WrapMargin
 
 745                      m_llist
->WrapLine(m_WrapMargin
); 
 748                   m_llist
->Insert((wxChar
)keyCode
); 
 752                   // we don't handle it, maybe an accelerator? 
 759          // we don't handle it, maybe an accelerator? 
 765       // continue selection to the current (new) cursor position 
 766       m_llist
->ContinueSelection(); 
 769    // refresh the screen 
 770    RequestUpdate(m_llist
->GetUpdateRect()); 
 774 wxLayoutWindow::OnKeyUp(wxKeyEvent
& event
) 
 776    if ( event
.KeyCode() == WXK_SHIFT 
&& m_Selecting 
) 
 778       m_llist
->EndSelection(); 
 787 wxLayoutWindow::ScrollToCursor(void) 
 789    //is always needed to make sure we know where the cursor is 
 791    //RequestUpdate(m_llist->GetUpdateRect()); 
 796    int x0
,y0
,x1
,y1
, dx
, dy
; 
 798    // Calculate where the top of the visible area is: 
 799    GetViewStart(&x0
,&y0
); 
 800    GetScrollPixelsPerUnit(&dx
, &dy
); 
 803    WXLO_DEBUG(("ScrollToCursor: GetViewStart is %d/%d", x0
, y0
)); 
 805    // Get the size of the visible window: 
 806    GetClientSize(&x1
, &y1
); 
 808    // Make sure that the scrollbars are at a position so that the cursor is 
 809    // visible if we are editing 
 810    WXLO_DEBUG(("m_ScrollToCursor = %d", (int) m_ScrollToCursor
)); 
 811    wxPoint cc 
= m_llist
->GetCursorScreenPos(); 
 813    // the cursor should be completely visible in both directions 
 814    wxPoint 
cs(m_llist
->GetCursorSize()); 
 817    if ( cc
.x 
< x0 
|| cc
.x 
>= x0 
+ x1 
- cs
.x 
) 
 824    if ( cc
.y 
< y0 
|| cc
.y 
>= y0 
+ y1 
- cs
.y 
) 
 831    if( nx 
!= -1 || ny 
!= -1 ) 
 833       // set new view start 
 834       Scroll(nx 
== -1 ? -1 : (nx
+dx
-1)/dx
, ny 
== -1 ? -1 : (ny
+dy
-1)/dy
); 
 836       m_ScrollToCursor 
= false; 
 842 wxLayoutWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
)) 
 844    wxRect region 
= GetUpdateRegion().GetBox(); 
 845    InternalPaint(®ion
); 
 849 wxLayoutWindow::RequestUpdate(const wxRect 
* 
 858    // Calling Refresh() causes bad flicker under wxGTK!!! 
 859    InternalPaint(updateRect
); 
 861    // shouldn't specify the update rectangle if it doesn't include all the 
 862    // changed locations - otherwise, they won't be repainted at all because 
 863    // the system clips the display to the update rect 
 864    Refresh(FALSE
); //, updateRect); 
 869 wxLayoutWindow::InternalPaint(const wxRect 
*updateRect
) 
 872    wxPaintDC 
dc( this ); 
 875 #ifdef WXLAYOUT_USE_CARET 
 876    // hide the caret before drawing anything 
 878 #endif // WXLAYOUT_USE_CARET 
 880    int x0
,y0
,x1
,y1
, dx
, dy
; 
 882    // Calculate where the top of the visible area is: 
 883    GetViewStart(&x0
,&y0
); 
 884    GetScrollPixelsPerUnit(&dx
, &dy
); 
 887    // Get the size of the visible window: 
 888    GetClientSize(&x1
,&y1
); 
 894       WXLO_DEBUG(("Update rect: %ld,%ld / %ld,%ld", 
 895                   updateRect
->x
, updateRect
->y
, 
 896                   updateRect
->x
+updateRect
->width
, 
 897                   updateRect
->y
+updateRect
->height
)); 
 900    ResizeScrollbars(true); 
 902    WXLO_TIMER_START(TmpTimer
); 
 903    /* Check whether the window has grown, if so, we need to reallocate 
 904       the bitmap to be larger. */ 
 905    if(x1 
> m_bitmapSize
.x 
|| y1 
> m_bitmapSize
.y
) 
 907       wxASSERT(m_bitmapSize
.x 
> 0); 
 908       wxASSERT(m_bitmapSize
.y 
> 0); 
 910       m_memDC
->SelectObject(wxNullBitmap
); 
 912       m_bitmapSize 
= wxPoint(x1
,y1
); 
 913       m_bitmap 
= new wxBitmap(x1
,y1
); 
 914       m_memDC
->SelectObject(*m_bitmap
); 
 917    m_memDC
->SetDeviceOrigin(0,0); 
 918    m_memDC
->SetBackground(wxBrush(m_llist
->GetDefaultStyleInfo().GetBGColour(),wxSOLID
)); 
 919    m_memDC
->SetPen(wxPen(m_llist
->GetDefaultStyleInfo().GetBGColour(), 
 921    m_memDC
->SetLogicalFunction(wxCOPY
); 
 923    WXLO_TIMER_STOP(TmpTimer
); 
 925    // fill the background with the background bitmap 
 930          w 
= m_BGbitmap
->GetWidth(), 
 931          h 
= m_BGbitmap
->GetHeight(); 
 932       for(y 
= 0; y 
< y1
; y
+=h
) 
 933          for(x 
= 0; x 
< x1
; x
+=w
) 
 934             m_memDC
->DrawBitmap(*m_BGbitmap
, x
, y
); 
 935       m_memDC
->SetBackgroundMode(wxTRANSPARENT
); 
 938    // This is the important bit: we tell the list to draw itself 
 942       WXLO_DEBUG(("Update rect: %ld,%ld / %ld,%ld", 
 943                   updateRect
->x
, updateRect
->y
, 
 944                   updateRect
->x
+updateRect
->width
, 
 945                   updateRect
->y
+updateRect
->height
)); 
 949    // Device origins on the memDC are suspect, we translate manually 
 950    // with the translate parameter of Draw(). 
 951    wxPoint 
offset(-x0
+WXLO_XOFFSET
,-y0
+WXLO_YOFFSET
); 
 952    m_llist
->Draw(*m_memDC
,offset
, y0
, y0
+y1
); 
 954    // We start calculating a new update rect before drawing the 
 955    // cursor, so that the cursor coordinates get included in the next 
 956    // update rectangle (although they are drawn on the memDC, this is 
 957    // needed to erase it): 
 958    m_llist
->InvalidateUpdateRect(); 
 959    if(m_CursorVisibility 
== 1) 
 961       // draw a thick cursor for editable windows with focus 
 962       m_llist
->DrawCursor(*m_memDC
, 
 963                           m_HaveFocus 
&& IsEditable(), 
 967    WXLO_TIMER_START(BlitTimer
); 
 968 // Now copy everything to the screen: 
 970    // This somehow doesn't work, but even the following bit with the 
 971    // whole rect at once is still a bit broken I think. 
 972    wxRegionIterator 
ri ( GetUpdateRegion() ); 
 976          WXLO_DEBUG(("UpdateRegion: %ld,%ld, %ld,%ld", 
 977                      ri
.GetX(),ri
.GetY(),ri
.GetW(),ri
.GetH())); 
 978          dc
.Blit(x0
+ri
.GetX(),y0
+ri
.GetY(),ri
.GetW(),ri
.GetH(), 
 979                  m_memDC
,ri
.GetX(),ri
.GetY(),wxCOPY
,FALSE
); 
 985       // FIXME: Trying to copy only the changed parts, but it does not seem 
 987 //      x0 = updateRect->x; y0 = updateRect->y; 
 988 //      if(updateRect->height < y1) 
 989 //         y1 = updateRect->height; 
 990 //      y1 += WXLO_YOFFSET; //FIXME might not be needed 
 991       dc
.Blit(x0
,y0
,x1
,y1
,m_memDC
,0,0,wxCOPY
,FALSE
); 
 993    WXLO_TIMER_STOP(BlitTimer
); 
 996 #ifdef WXLAYOUT_USE_CARET 
 997    // show the caret back after everything is redrawn 
 999 #endif // WXLAYOUT_USE_CARET 
1003    if ( m_StatusBar 
&& m_StatusFieldCursor 
!= -1 ) 
1005       static wxPoint 
s_oldCursorPos(-1, -1); 
1007       wxPoint 
pos(m_llist
->GetCursorPos()); 
1009       // avoid unnecessary status bar refreshes 
1010       if ( pos 
!= s_oldCursorPos 
) 
1012          s_oldCursorPos 
= pos
; 
1015          label
.Printf(_("Ln:%d Col:%d"), pos
.y 
+ 1, pos
.x 
+ 1); 
1016          m_StatusBar
->SetStatusText(label
, m_StatusFieldCursor
); 
1020    WXLO_TIMER_PRINT(LayoutTimer
); 
1021    WXLO_TIMER_PRINT(BlitTimer
); 
1022    WXLO_TIMER_PRINT(TmpTimer
); 
1026 wxLayoutWindow::OnSize(wxSizeEvent 
&event
) 
1035 Change the range and position of scrollbars. Has evolved into a 
1036 generic Update function which will at some time later cause a repaint 
1041 wxLayoutWindow::ResizeScrollbars(bool exact
) 
1043    wxClientDC 
dc( this ); 
1045 //   m_llist->ForceTotalLayout(); 
1049       // we are laying out just the minimum, but always up to the 
1050       // cursor line, so the cursor position is updated. 
1051       m_llist
->Layout(dc
, 0); 
1054    WXLO_TIMER_START(LayoutTimer
); 
1055    m_llist
->Layout(dc
, -1); 
1056    WXLO_TIMER_STOP(LayoutTimer
); 
1059    wxPoint max 
= m_llist
->GetSize(); 
1060    wxSize size 
= GetClientSize(); 
1062    WXLO_DEBUG(("ResizeScrollbars: max size = (%ld, %ld)", 
1063                (long int)max
.x
, (long int) max
.y
)); 
1065    // in the absence of scrollbars we should compare with the client size 
1066    if ( !m_hasHScrollbar 
) 
1067       m_maxx 
= size
.x
;// - WXLO_ROFFSET; 
1068    if ( !m_hasVScrollbar 
) 
1069       m_maxy 
= size
.y
;// - WXLO_BOFFSET; 
1071    // check if the text hasn't become too big 
1072    // TODO why do we set both at once? they're independent... 
1073    if( max
.x 
> m_maxx 
- WXLO_ROFFSET
 
1074        || max
.y 
> m_maxy 
- WXLO_BOFFSET
 
1075        || (max
.x 
< m_maxx 
- X_SCROLL_PAGE
) 
1076        || (max
.y 
< m_maxy 
- Y_SCROLL_PAGE
) 
1079       // text became too large 
1082          // add an extra bit to the sizes to avoid future updates 
1083          max
.x 
+= WXLO_ROFFSET
; 
1084          max
.y 
+= WXLO_BOFFSET
; 
1088       if(max
.x 
< X_SCROLL_PAGE 
&& m_hasHScrollbar
) 
1090          SetScrollbars(0,-1,0,-1,0,-1,true); 
1091          m_hasHScrollbar 
= FALSE
; 
1094       if(max
.y 
< Y_SCROLL_PAGE 
&& m_hasVScrollbar
) 
1096          SetScrollbars(-1,0,-1,0,-1,0,true); 
1097          m_hasVScrollbar 
= FALSE
; 
1101 //         (max.x > X_SCROLL_PAGE || max.y > Y_SCROLL_PAGE) 
1102          (max
.x 
> size
.x 
- X_SCROLL_PAGE
|| max
.y 
> size
.y 
- Y_SCROLL_PAGE
) 
1105          GetViewStart(&m_ViewStartX
, &m_ViewStartY
); 
1106          SetScrollbars(X_SCROLL_PAGE
, 
1108                        max
.x 
/ X_SCROLL_PAGE 
+ 2, 
1109                        max
.y 
/ Y_SCROLL_PAGE 
+ 2, 
1114             m_hasVScrollbar 
= true; 
1115 //         ScrollToCursor(); 
1118       m_maxx 
= max
.x 
+ X_SCROLL_PAGE
; 
1119       m_maxy 
= max
.y 
+ Y_SCROLL_PAGE
; 
1123 // ---------------------------------------------------------------------------- 
1125 // clipboard operations 
1127 // ---------------------------------------------------------------------------- 
1130 wxLayoutWindow::Paste(bool usePrivate
, bool primary
) 
1132    // this only has an effect under X11: 
1133    wxTheClipboard
->UsePrimarySelection(primary
); 
1135    if (wxTheClipboard
->Open()) 
1139          wxLayoutDataObject wxldo
; 
1140          if (wxTheClipboard
->IsSupported( wxldo
.GetFormat() )) 
1142             if(wxTheClipboard
->GetData(wxldo
)) 
1144                wxTheClipboard
->Close(); 
1145                wxString str 
= wxldo
.GetLayoutData(); 
1153       wxTextDataObject data
; 
1154       if (wxTheClipboard
->IsSupported( data
.GetFormat() ) 
1155           && wxTheClipboard
->GetData(data
) ) 
1157          wxTheClipboard
->Close(); 
1158          wxString text 
= data
.GetText(); 
1159          wxLayoutImportText( m_llist
, text
); 
1165    // if everything failed we can still try the primary: 
1166    wxTheClipboard
->Close(); 
1167    if(! primary
) // not tried before 
1169       wxTheClipboard
->UsePrimarySelection(); 
1170       if (wxTheClipboard
->Open()) 
1172          wxTextDataObject data
; 
1173          if (wxTheClipboard
->IsSupported( data
.GetFormat() ) 
1174              && wxTheClipboard
->GetData(data
) ) 
1176             wxString text 
= data
.GetText(); 
1177             wxLayoutImportText( m_llist
, text
); 
1181          wxTheClipboard
->Close(); 
1187 wxLayoutWindow::Copy(bool invalidate
, bool privateFormat
, bool primary
) 
1189    // Calling GetSelection() will automatically do an EndSelection() 
1190    // on the list, but we need to take a note of it, too: 
1193       m_Selecting 
= false; 
1194       m_llist
->EndSelection(); 
1197    wxLayoutDataObject 
*wldo 
= new wxLayoutDataObject
; 
1198    wxLayoutList 
*llist 
= m_llist
->GetSelection(wldo
, invalidate
); 
1201    // Export selection as text: 
1203    wxLayoutExportObject 
*exp
; 
1204    wxLayoutExportStatus 
status(llist
); 
1205    while((exp 
= wxLayoutExport( &status
, WXLO_EXPORT_AS_TEXT
)) != NULL
) 
1207       if(exp
->type 
== WXLO_EXPORT_TEXT
) 
1208          text 
<< *(exp
->content
.text
); 
1213    // The exporter always appends a newline, so we chop it off if it 
1216       size_t len 
= text
.Length(); 
1217       if(len 
> 2 && text
[len
-2] ==  '\r') // Windows 
1218          text 
= text
.Mid(0,len
-2); 
1219       else if(len 
> 1 && text
[len
-1] == '\n') 
1220          text 
= text
.Mid(0,len
-1); 
1224 if(! primary
) // always copy as text-only to primary selection 
1226       wxTheClipboard
->UsePrimarySelection(); 
1227       if (wxTheClipboard
->Open()) 
1229          wxTextDataObject 
*data 
= new wxTextDataObject( text 
); 
1230          wxTheClipboard
->SetData( data 
); 
1231          wxTheClipboard
->Close(); 
1236    wxTheClipboard
->UsePrimarySelection(primary
); 
1237    if (wxTheClipboard
->Open()) 
1239       wxTextDataObject 
*data 
= new wxTextDataObject( text 
); 
1242          rc 
= wxTheClipboard
->SetData( data 
); 
1244          rc 
|= wxTheClipboard
->SetData( wldo 
); 
1245       wxTheClipboard
->Close(); 
1255 wxLayoutWindow::Cut(bool privateFormat
, bool usePrimary
) 
1257    if(Copy(false, privateFormat
, usePrimary
)) // do not invalidate selection after copy 
1259       m_llist
->DeleteSelection(); 
1267 // ---------------------------------------------------------------------------- 
1269 // ---------------------------------------------------------------------------- 
1272 wxLayoutWindow::Find(const wxString 
& 
1290                                      WXUNUSED(configPath
) 
1297    if(needle
.Length() == 0) 
1299       if( ! MInputBox(&m_FindString
, 
1304           || strutil_isempty(m_FindString
)) 
1308       m_FindString 
= needle
; 
1310    if(fromWhere 
== NULL
) 
1311       found 
= m_llist
->FindText(m_FindString
, m_llist
->GetCursorPos()); 
1313       found 
= m_llist
->FindText(m_FindString
, *fromWhere
); 
1321       m_llist
->MoveCursorTo(found
); 
1332 wxLayoutWindow::FindAgain(void) 
1334    bool rc 
= Find(m_FindString
); 
1338 // ---------------------------------------------------------------------------- 
1340 // ---------------------------------------------------------------------------- 
1343 wxLayoutWindow::MakeFormatMenu() 
1345    wxMenu 
*m 
= new wxMenu(_("Layout Menu")); 
1347    m
->Append(WXLOWIN_MENU_LARGER   
,_("&Larger"),_("Switch to larger font."), false); 
1348    m
->Append(WXLOWIN_MENU_SMALLER  
,_("&Smaller"),_("Switch to smaller font."), false); 
1349    m
->AppendSeparator(); 
1350    m
->Append(WXLOWIN_MENU_UNDERLINE
, _("&Underline"),_("Underline mode."), true); 
1351    m
->Append(WXLOWIN_MENU_BOLD
, _("&Bold"),_("Bold mode."), true); 
1352    m
->Append(WXLOWIN_MENU_ITALICS
, _("&Italics"),_("Italics mode."), true); 
1353    m
->AppendSeparator(); 
1354    m
->Append(WXLOWIN_MENU_ROMAN     
,_("&Roman"),_("Switch to roman font."), false); 
1355    m
->Append(WXLOWIN_MENU_TYPEWRITER
,_("&Typewriter"),_("Switch to typewriter font."), false); 
1356    m
->Append(WXLOWIN_MENU_SANSSERIF 
,_("&Sans Serif"),_("Switch to sans serif font."), false); 
1361 void wxLayoutWindow::OnUpdateMenuUnderline(wxUpdateUIEvent
& event
) 
1363    event
.Check(m_llist
->IsFontUnderlined()); 
1366 void wxLayoutWindow::OnUpdateMenuBold(wxUpdateUIEvent
& event
) 
1368    event
.Check(m_llist
->IsFontBold()); 
1371 void wxLayoutWindow::OnUpdateMenuItalic(wxUpdateUIEvent
& event
) 
1373    event
.Check(m_llist
->IsFontItalic()); 
1376 void wxLayoutWindow::OnMenu(wxCommandEvent
& event
) 
1378    switch (event
.GetId()) 
1380    case WXLOWIN_MENU_LARGER
: 
1381       m_llist
->SetFontLarger(); RequestUpdate(); break; 
1382    case WXLOWIN_MENU_SMALLER
: 
1383       m_llist
->SetFontSmaller(); RequestUpdate(); break; 
1384    case WXLOWIN_MENU_UNDERLINE
: 
1385       m_llist
->ToggleFontUnderline(); RequestUpdate(); break; 
1386    case WXLOWIN_MENU_BOLD
: 
1387       m_llist
->ToggleFontWeight(); RequestUpdate(); break; 
1388    case WXLOWIN_MENU_ITALICS
: 
1389       m_llist
->ToggleFontItalics(); RequestUpdate(); break; 
1390    case WXLOWIN_MENU_ROMAN
: 
1391       m_llist
->SetFontFamily(wxROMAN
); RequestUpdate(); break; 
1392    case WXLOWIN_MENU_TYPEWRITER
: 
1393       m_llist
->SetFontFamily(wxFIXED
); RequestUpdate(); break; 
1394    case WXLOWIN_MENU_SANSSERIF
: 
1395       m_llist
->SetFontFamily(wxSWISS
); RequestUpdate(); break; 
1399 // ---------------------------------------------------------------------------- 
1401 // ---------------------------------------------------------------------------- 
1404 wxLayoutWindow::OnSetFocus(wxFocusEvent 
&ev
) 
1408    RequestUpdate(); // cursor must change 
1412 wxLayoutWindow::OnKillFocus(wxFocusEvent 
&ev
) 
1414    m_HaveFocus 
= false; 
1416    RequestUpdate();// cursor must change 
1419 // ---------------------------------------------------------------------------- 
1420 // private functions 
1421 // ---------------------------------------------------------------------------- 
1423 static bool IsDirectionKey(long keyCode
)