Add menu items to toggle multiple selections and multipaste support in wxSTC. See #16221. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@76461 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
		
			
				
	
	
		
			965 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			965 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //////////////////////////////////////////////////////////////////////////////
 | |
| // File:        contrib/samples/stc/edit.cpp
 | |
| // Purpose:     STC test module
 | |
| // Maintainer:  Wyo
 | |
| // Created:     2003-09-01
 | |
| // Copyright:   (c) wxGuide
 | |
| // Licence:     wxWindows licence
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // informations
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // headers
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| // For compilers that support precompilation, includes "wx/wx.h".
 | |
| #include "wx/wxprec.h"
 | |
| 
 | |
| #ifdef __BORLANDC__
 | |
|     #pragma hdrstop
 | |
| #endif
 | |
| 
 | |
| // for all others, include the necessary headers (this file is usually all you
 | |
| // need because it includes almost all 'standard' wxWidgets headers)
 | |
| #ifndef WX_PRECOMP
 | |
|     #include "wx/wx.h"
 | |
|     #include "wx/textdlg.h"
 | |
| #endif
 | |
| 
 | |
| //! wxWidgets headers
 | |
| #include "wx/file.h"     // raw file io support
 | |
| #include "wx/filename.h" // filename support
 | |
| 
 | |
| //! application headers
 | |
| #include "defsext.h"     // additional definitions
 | |
| 
 | |
| #include "edit.h"        // edit module
 | |
| 
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // resources
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| 
 | |
| //============================================================================
 | |
| // declarations
 | |
| //============================================================================
 | |
| 
 | |
| // The (uniform) style used for the annotations.
 | |
| const int ANNOTATION_STYLE = wxSTC_STYLE_LASTPREDEFINED + 1;
 | |
| 
 | |
| //============================================================================
 | |
| // implementation
 | |
| //============================================================================
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // Edit
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| wxBEGIN_EVENT_TABLE (Edit, wxStyledTextCtrl)
 | |
|     // common
 | |
|     EVT_SIZE (                         Edit::OnSize)
 | |
|     // edit
 | |
|     EVT_MENU (wxID_CLEAR,              Edit::OnEditClear)
 | |
|     EVT_MENU (wxID_CUT,                Edit::OnEditCut)
 | |
|     EVT_MENU (wxID_COPY,               Edit::OnEditCopy)
 | |
|     EVT_MENU (wxID_PASTE,              Edit::OnEditPaste)
 | |
|     EVT_MENU (myID_INDENTINC,          Edit::OnEditIndentInc)
 | |
|     EVT_MENU (myID_INDENTRED,          Edit::OnEditIndentRed)
 | |
|     EVT_MENU (wxID_SELECTALL,          Edit::OnEditSelectAll)
 | |
|     EVT_MENU (myID_SELECTLINE,         Edit::OnEditSelectLine)
 | |
|     EVT_MENU (wxID_REDO,               Edit::OnEditRedo)
 | |
|     EVT_MENU (wxID_UNDO,               Edit::OnEditUndo)
 | |
|     // find
 | |
|     EVT_MENU (wxID_FIND,               Edit::OnFind)
 | |
|     EVT_MENU (myID_FINDNEXT,           Edit::OnFindNext)
 | |
|     EVT_MENU (myID_REPLACE,            Edit::OnReplace)
 | |
|     EVT_MENU (myID_REPLACENEXT,        Edit::OnReplaceNext)
 | |
|     EVT_MENU (myID_BRACEMATCH,         Edit::OnBraceMatch)
 | |
|     EVT_MENU (myID_GOTO,               Edit::OnGoto)
 | |
|     // view
 | |
|     EVT_MENU_RANGE (myID_HILIGHTFIRST, myID_HILIGHTLAST,
 | |
|                                        Edit::OnHilightLang)
 | |
|     EVT_MENU (myID_DISPLAYEOL,         Edit::OnDisplayEOL)
 | |
|     EVT_MENU (myID_INDENTGUIDE,        Edit::OnIndentGuide)
 | |
|     EVT_MENU (myID_LINENUMBER,         Edit::OnLineNumber)
 | |
|     EVT_MENU (myID_LONGLINEON,         Edit::OnLongLineOn)
 | |
|     EVT_MENU (myID_WHITESPACE,         Edit::OnWhiteSpace)
 | |
|     EVT_MENU (myID_FOLDTOGGLE,         Edit::OnFoldToggle)
 | |
|     EVT_MENU (myID_OVERTYPE,           Edit::OnSetOverType)
 | |
|     EVT_MENU (myID_READONLY,           Edit::OnSetReadOnly)
 | |
|     EVT_MENU (myID_WRAPMODEON,         Edit::OnWrapmodeOn)
 | |
|     EVT_MENU (myID_CHARSETANSI,        Edit::OnUseCharset)
 | |
|     EVT_MENU (myID_CHARSETMAC,         Edit::OnUseCharset)
 | |
|     // annotations
 | |
|     EVT_MENU (myID_ANNOTATION_ADD,     Edit::OnAnnotationAdd)
 | |
|     EVT_MENU (myID_ANNOTATION_REMOVE,  Edit::OnAnnotationRemove)
 | |
|     EVT_MENU (myID_ANNOTATION_CLEAR,   Edit::OnAnnotationClear)
 | |
|     EVT_MENU (myID_ANNOTATION_STYLE_HIDDEN,   Edit::OnAnnotationStyle)
 | |
|     EVT_MENU (myID_ANNOTATION_STYLE_STANDARD, Edit::OnAnnotationStyle)
 | |
|     EVT_MENU (myID_ANNOTATION_STYLE_BOXED,    Edit::OnAnnotationStyle)
 | |
|     // extra
 | |
|     EVT_MENU (myID_CHANGELOWER,                 Edit::OnChangeCase)
 | |
|     EVT_MENU (myID_CHANGEUPPER,                 Edit::OnChangeCase)
 | |
|     EVT_MENU (myID_CONVERTCR,                   Edit::OnConvertEOL)
 | |
|     EVT_MENU (myID_CONVERTCRLF,                 Edit::OnConvertEOL)
 | |
|     EVT_MENU (myID_CONVERTLF,                   Edit::OnConvertEOL)
 | |
|     EVT_MENU(myID_MULTIPLE_SELECTIONS,          Edit::OnMultipleSelections)
 | |
|     EVT_MENU(myID_MULTI_PASTE,                  Edit::OnMultiPaste)
 | |
|     EVT_MENU(myID_MULTIPLE_SELECTIONS_TYPING,   Edit::OnMultipleSelectionsTyping)
 | |
|     // stc
 | |
|     EVT_STC_MARGINCLICK (wxID_ANY,     Edit::OnMarginClick)
 | |
|     EVT_STC_CHARADDED (wxID_ANY,       Edit::OnCharAdded)
 | |
|     EVT_STC_KEY( wxID_ANY , Edit::OnKey )
 | |
| 
 | |
|     EVT_KEY_DOWN( Edit::OnKeyDown )
 | |
| wxEND_EVENT_TABLE()
 | |
| 
 | |
| Edit::Edit (wxWindow *parent, wxWindowID id,
 | |
|             const wxPoint &pos,
 | |
|             const wxSize &size,
 | |
|             long style)
 | |
|     : wxStyledTextCtrl (parent, id, pos, size, style) {
 | |
| 
 | |
|     m_filename = wxEmptyString;
 | |
| 
 | |
|     m_LineNrID = 0;
 | |
|     m_DividerID = 1;
 | |
|     m_FoldingID = 2;
 | |
| 
 | |
|     // initialize language
 | |
|     m_language = NULL;
 | |
| 
 | |
|     // Use all the bits in the style byte as styles, not indicators.
 | |
|     SetStyleBits(8);
 | |
|     
 | |
|     // default font for all styles
 | |
|     SetViewEOL (g_CommonPrefs.displayEOLEnable);
 | |
|     SetIndentationGuides (g_CommonPrefs.indentGuideEnable);
 | |
|     SetEdgeMode (g_CommonPrefs.longLineOnEnable?
 | |
|                  wxSTC_EDGE_LINE: wxSTC_EDGE_NONE);
 | |
|     SetViewWhiteSpace (g_CommonPrefs.whiteSpaceEnable?
 | |
|                        wxSTC_WS_VISIBLEALWAYS: wxSTC_WS_INVISIBLE);
 | |
|     SetOvertype (g_CommonPrefs.overTypeInitial);
 | |
|     SetReadOnly (g_CommonPrefs.readOnlyInitial);
 | |
|     SetWrapMode (g_CommonPrefs.wrapModeInitial?
 | |
|                  wxSTC_WRAP_WORD: wxSTC_WRAP_NONE);
 | |
|     wxFont font(wxFontInfo(10).Family(wxFONTFAMILY_MODERN));
 | |
|     StyleSetFont (wxSTC_STYLE_DEFAULT, font);
 | |
|     StyleSetForeground (wxSTC_STYLE_DEFAULT, *wxBLACK);
 | |
|     StyleSetBackground (wxSTC_STYLE_DEFAULT, *wxWHITE);
 | |
|     StyleSetForeground (wxSTC_STYLE_LINENUMBER, wxColour (wxT("DARK GREY")));
 | |
|     StyleSetBackground (wxSTC_STYLE_LINENUMBER, *wxWHITE);
 | |
|     StyleSetForeground(wxSTC_STYLE_INDENTGUIDE, wxColour (wxT("DARK GREY")));
 | |
|     InitializePrefs (DEFAULT_LANGUAGE);
 | |
| 
 | |
|     // set visibility
 | |
|     SetVisiblePolicy (wxSTC_VISIBLE_STRICT|wxSTC_VISIBLE_SLOP, 1);
 | |
|     SetXCaretPolicy (wxSTC_CARET_EVEN|wxSTC_VISIBLE_STRICT|wxSTC_CARET_SLOP, 1);
 | |
|     SetYCaretPolicy (wxSTC_CARET_EVEN|wxSTC_VISIBLE_STRICT|wxSTC_CARET_SLOP, 1);
 | |
| 
 | |
|     // markers
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDER,        wxSTC_MARK_DOTDOTDOT, wxT("BLACK"), wxT("BLACK"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDEROPEN,    wxSTC_MARK_ARROWDOWN, wxT("BLACK"), wxT("BLACK"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDERSUB,     wxSTC_MARK_EMPTY,     wxT("BLACK"), wxT("BLACK"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDEREND,     wxSTC_MARK_DOTDOTDOT, wxT("BLACK"), wxT("WHITE"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDEROPENMID, wxSTC_MARK_ARROWDOWN, wxT("BLACK"), wxT("WHITE"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDERMIDTAIL, wxSTC_MARK_EMPTY,     wxT("BLACK"), wxT("BLACK"));
 | |
|     MarkerDefine (wxSTC_MARKNUM_FOLDERTAIL,    wxSTC_MARK_EMPTY,     wxT("BLACK"), wxT("BLACK"));
 | |
| 
 | |
|     // annotations
 | |
|     AnnotationSetVisible(wxSTC_ANNOTATION_BOXED);
 | |
| 
 | |
|     // miscellaneous
 | |
|     m_LineNrMargin = TextWidth (wxSTC_STYLE_LINENUMBER, wxT("_999999"));
 | |
|     m_FoldingMargin = 16;
 | |
|     CmdKeyClear (wxSTC_KEY_TAB, 0); // this is done by the menu accelerator key
 | |
|     SetLayoutCache (wxSTC_CACHE_PAGE);
 | |
| 
 | |
| }
 | |
| 
 | |
| Edit::~Edit () {}
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // common event handlers
 | |
| void Edit::OnSize( wxSizeEvent& event ) {
 | |
|     int x = GetClientSize().x +
 | |
|             (g_CommonPrefs.lineNumberEnable? m_LineNrMargin: 0) +
 | |
|             (g_CommonPrefs.foldEnable? m_FoldingMargin: 0);
 | |
|     if (x > 0) SetScrollWidth (x);
 | |
|     event.Skip();
 | |
| }
 | |
| 
 | |
| // edit event handlers
 | |
| void Edit::OnEditRedo (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (!CanRedo()) return;
 | |
|     Redo ();
 | |
| }
 | |
| 
 | |
| void Edit::OnEditUndo (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (!CanUndo()) return;
 | |
|     Undo ();
 | |
| }
 | |
| 
 | |
| void Edit::OnEditClear (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (GetReadOnly()) return;
 | |
|     Clear ();
 | |
| }
 | |
| 
 | |
| void Edit::OnKey (wxStyledTextEvent &WXUNUSED(event))
 | |
| {
 | |
|     wxMessageBox("OnKey");
 | |
| }
 | |
| 
 | |
| void Edit::OnKeyDown (wxKeyEvent &event)
 | |
| {
 | |
|     if (CallTipActive())
 | |
|         CallTipCancel();
 | |
|     if (event.GetKeyCode() == WXK_SPACE && event.ControlDown() && event.ShiftDown())
 | |
|     {
 | |
|         int pos = GetCurrentPos();
 | |
|         CallTipSetBackground(*wxYELLOW);
 | |
|         CallTipShow(pos,
 | |
|                     "This is a CallTip with multiple lines.\n"
 | |
|                     "It is meant to be a context sensitive popup helper for the user.");
 | |
|         return;
 | |
|     }
 | |
|     event.Skip();
 | |
| }
 | |
| 
 | |
| void Edit::OnEditCut (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (GetReadOnly() || (GetSelectionEnd()-GetSelectionStart() <= 0)) return;
 | |
|     Cut ();
 | |
| }
 | |
| 
 | |
| void Edit::OnEditCopy (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (GetSelectionEnd()-GetSelectionStart() <= 0) return;
 | |
|     Copy ();
 | |
| }
 | |
| 
 | |
| void Edit::OnEditPaste (wxCommandEvent &WXUNUSED(event)) {
 | |
|     if (!CanPaste()) return;
 | |
|     Paste ();
 | |
| }
 | |
| 
 | |
| void Edit::OnFind (wxCommandEvent &WXUNUSED(event)) {
 | |
| }
 | |
| 
 | |
| void Edit::OnFindNext (wxCommandEvent &WXUNUSED(event)) {
 | |
| }
 | |
| 
 | |
| void Edit::OnReplace (wxCommandEvent &WXUNUSED(event)) {
 | |
| }
 | |
| 
 | |
| void Edit::OnReplaceNext (wxCommandEvent &WXUNUSED(event)) {
 | |
| }
 | |
| 
 | |
| void Edit::OnBraceMatch (wxCommandEvent &WXUNUSED(event)) {
 | |
|     int min = GetCurrentPos ();
 | |
|     int max = BraceMatch (min);
 | |
|     if (max > (min+1)) {
 | |
|         BraceHighlight (min+1, max);
 | |
|         SetSelection (min+1, max);
 | |
|     }else{
 | |
|         BraceBadLight (min);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Edit::OnGoto (wxCommandEvent &WXUNUSED(event)) {
 | |
| }
 | |
| 
 | |
| void Edit::OnEditIndentInc (wxCommandEvent &WXUNUSED(event)) {
 | |
|     CmdKeyExecute (wxSTC_CMD_TAB);
 | |
| }
 | |
| 
 | |
| void Edit::OnEditIndentRed (wxCommandEvent &WXUNUSED(event)) {
 | |
|     CmdKeyExecute (wxSTC_CMD_DELETEBACK);
 | |
| }
 | |
| 
 | |
| void Edit::OnEditSelectAll (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetSelection (0, GetTextLength ());
 | |
| }
 | |
| 
 | |
| void Edit::OnEditSelectLine (wxCommandEvent &WXUNUSED(event)) {
 | |
|     int lineStart = PositionFromLine (GetCurrentLine());
 | |
|     int lineEnd = PositionFromLine (GetCurrentLine() + 1);
 | |
|     SetSelection (lineStart, lineEnd);
 | |
| }
 | |
| 
 | |
| void Edit::OnHilightLang (wxCommandEvent &event) {
 | |
|     InitializePrefs (g_LanguagePrefs [event.GetId() - myID_HILIGHTFIRST].name);
 | |
| }
 | |
| 
 | |
| void Edit::OnDisplayEOL (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetViewEOL (!GetViewEOL());
 | |
| }
 | |
| 
 | |
| void Edit::OnIndentGuide (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetIndentationGuides (!GetIndentationGuides());
 | |
| }
 | |
| 
 | |
| void Edit::OnLineNumber (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetMarginWidth (m_LineNrID,
 | |
|                     GetMarginWidth (m_LineNrID) == 0? m_LineNrMargin: 0);
 | |
| }
 | |
| 
 | |
| void Edit::OnLongLineOn (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetEdgeMode (GetEdgeMode() == 0? wxSTC_EDGE_LINE: wxSTC_EDGE_NONE);
 | |
| }
 | |
| 
 | |
| void Edit::OnWhiteSpace (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetViewWhiteSpace (GetViewWhiteSpace() == 0?
 | |
|                        wxSTC_WS_VISIBLEALWAYS: wxSTC_WS_INVISIBLE);
 | |
| }
 | |
| 
 | |
| void Edit::OnFoldToggle (wxCommandEvent &WXUNUSED(event)) {
 | |
|     ToggleFold (GetFoldParent(GetCurrentLine()));
 | |
| }
 | |
| 
 | |
| void Edit::OnSetOverType (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetOvertype (!GetOvertype());
 | |
| }
 | |
| 
 | |
| void Edit::OnSetReadOnly (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetReadOnly (!GetReadOnly());
 | |
| }
 | |
| 
 | |
| void Edit::OnWrapmodeOn (wxCommandEvent &WXUNUSED(event)) {
 | |
|     SetWrapMode (GetWrapMode() == 0? wxSTC_WRAP_WORD: wxSTC_WRAP_NONE);
 | |
| }
 | |
| 
 | |
| void Edit::OnUseCharset (wxCommandEvent &event) {
 | |
|     int Nr;
 | |
|     int charset = GetCodePage();
 | |
|     switch (event.GetId()) {
 | |
|         case myID_CHARSETANSI: {charset = wxSTC_CHARSET_ANSI; break;}
 | |
|         case myID_CHARSETMAC: {charset = wxSTC_CHARSET_ANSI; break;}
 | |
|     }
 | |
|     for (Nr = 0; Nr < wxSTC_STYLE_LASTPREDEFINED; Nr++) {
 | |
|         StyleSetCharacterSet (Nr, charset);
 | |
|     }
 | |
|     SetCodePage (charset);
 | |
| }
 | |
| 
 | |
| void Edit::OnAnnotationAdd(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     const int line = GetCurrentLine();
 | |
| 
 | |
|     wxString ann = AnnotationGetText(line);
 | |
|     ann = wxGetTextFromUser
 | |
|           (
 | |
|             wxString::Format("Enter annotation for the line %d", line),
 | |
|             "Edit annotation",
 | |
|             ann,
 | |
|             this
 | |
|           );
 | |
|     if ( ann.empty() )
 | |
|         return;
 | |
| 
 | |
|     AnnotationSetText(line, ann);
 | |
|     AnnotationSetStyle(line, ANNOTATION_STYLE);
 | |
| 
 | |
|     // Scintilla doesn't update the scroll width for annotations, even with
 | |
|     // scroll width tracking on, so do it manually.
 | |
|     const int width = GetScrollWidth();
 | |
| 
 | |
|     // NB: The following adjustments are only needed when using
 | |
|     //     wxSTC_ANNOTATION_BOXED annotations style, but we apply them always
 | |
|     //     in order to make things simpler and not have to redo the width
 | |
|     //     calculations when the annotations visibility changes. In a real
 | |
|     //     program you'd either just stick to a fixed annotations visibility or
 | |
|     //     update the width when it changes.
 | |
| 
 | |
|     // Take into account the fact that the annotation is shown indented, with
 | |
|     // the same indent as the line it's attached to.
 | |
|     int indent = GetLineIndentation(line);
 | |
| 
 | |
|     // This is just a hack to account for the width of the box, there doesn't
 | |
|     // seem to be any way to get it directly from Scintilla.
 | |
|     indent += 3;
 | |
| 
 | |
|     const int widthAnn = TextWidth(ANNOTATION_STYLE, ann + wxString(indent, ' '));
 | |
| 
 | |
|     if (widthAnn > width)
 | |
|         SetScrollWidth(widthAnn);
 | |
| }
 | |
| 
 | |
| void Edit::OnAnnotationRemove(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     AnnotationSetText(GetCurrentLine(), wxString());
 | |
| }
 | |
| 
 | |
| void Edit::OnAnnotationClear(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     AnnotationClearAll();
 | |
| }
 | |
| 
 | |
| void Edit::OnAnnotationStyle(wxCommandEvent& event)
 | |
| {
 | |
|     int style = 0;
 | |
|     switch (event.GetId()) {
 | |
|         case myID_ANNOTATION_STYLE_HIDDEN:
 | |
|             style = wxSTC_ANNOTATION_HIDDEN;
 | |
|             break;
 | |
| 
 | |
|         case myID_ANNOTATION_STYLE_STANDARD:
 | |
|             style = wxSTC_ANNOTATION_STANDARD;
 | |
|             break;
 | |
| 
 | |
|         case myID_ANNOTATION_STYLE_BOXED:
 | |
|             style = wxSTC_ANNOTATION_BOXED;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     AnnotationSetVisible(style);
 | |
| }
 | |
| 
 | |
| void Edit::OnChangeCase (wxCommandEvent &event) {
 | |
|     switch (event.GetId()) {
 | |
|         case myID_CHANGELOWER: {
 | |
|             CmdKeyExecute (wxSTC_CMD_LOWERCASE);
 | |
|             break;
 | |
|         }
 | |
|         case myID_CHANGEUPPER: {
 | |
|             CmdKeyExecute (wxSTC_CMD_UPPERCASE);
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Edit::OnConvertEOL (wxCommandEvent &event) {
 | |
|     int eolMode = GetEOLMode();
 | |
|     switch (event.GetId()) {
 | |
|         case myID_CONVERTCR: { eolMode = wxSTC_EOL_CR; break;}
 | |
|         case myID_CONVERTCRLF: { eolMode = wxSTC_EOL_CRLF; break;}
 | |
|         case myID_CONVERTLF: { eolMode = wxSTC_EOL_LF; break;}
 | |
|     }
 | |
|     ConvertEOLs (eolMode);
 | |
|     SetEOLMode (eolMode);
 | |
| }
 | |
| 
 | |
| void Edit::OnMultipleSelections(wxCommandEvent& WXUNUSED(event)) {
 | |
|     bool isSet = GetMultipleSelection();
 | |
|     SetMultipleSelection(!isSet);
 | |
| }
 | |
| 
 | |
| void Edit::OnMultiPaste(wxCommandEvent& WXUNUSED(event)) {
 | |
|     int pasteMode = GetMultiPaste();
 | |
|     if (wxSTC_MULTIPASTE_EACH == pasteMode) {
 | |
|         SetMultiPaste(wxSTC_MULTIPASTE_ONCE);
 | |
|     }
 | |
|     else {
 | |
|         SetMultiPaste(wxSTC_MULTIPASTE_EACH);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Edit::OnMultipleSelectionsTyping(wxCommandEvent& WXUNUSED(event)) {
 | |
|     bool isSet = GetAdditionalSelectionTyping();
 | |
|     SetAdditionalSelectionTyping(!isSet);
 | |
| }
 | |
| 
 | |
| //! misc
 | |
| void Edit::OnMarginClick (wxStyledTextEvent &event) {
 | |
|     if (event.GetMargin() == 2) {
 | |
|         int lineClick = LineFromPosition (event.GetPosition());
 | |
|         int levelClick = GetFoldLevel (lineClick);
 | |
|         if ((levelClick & wxSTC_FOLDLEVELHEADERFLAG) > 0) {
 | |
|             ToggleFold (lineClick);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Edit::OnCharAdded (wxStyledTextEvent &event) {
 | |
|     char chr = (char)event.GetKey();
 | |
|     int currentLine = GetCurrentLine();
 | |
|     // Change this if support for mac files with \r is needed
 | |
|     if (chr == '\n') {
 | |
|         int lineInd = 0;
 | |
|         if (currentLine > 0) {
 | |
|             lineInd = GetLineIndentation(currentLine - 1);
 | |
|         }
 | |
|         if (lineInd == 0) return;
 | |
|         SetLineIndentation (currentLine, lineInd);
 | |
|         GotoPos(PositionFromLine (currentLine) + lineInd);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // private functions
 | |
| wxString Edit::DeterminePrefs (const wxString &filename) {
 | |
| 
 | |
|     LanguageInfo const* curInfo;
 | |
| 
 | |
|     // determine language from filepatterns
 | |
|     int languageNr;
 | |
|     for (languageNr = 0; languageNr < g_LanguagePrefsSize; languageNr++) {
 | |
|         curInfo = &g_LanguagePrefs [languageNr];
 | |
|         wxString filepattern = curInfo->filepattern;
 | |
|         filepattern.Lower();
 | |
|         while (!filepattern.empty()) {
 | |
|             wxString cur = filepattern.BeforeFirst (';');
 | |
|             if ((cur == filename) ||
 | |
|                 (cur == (filename.BeforeLast ('.') + wxT(".*"))) ||
 | |
|                 (cur == (wxT("*.") + filename.AfterLast ('.')))) {
 | |
|                 return curInfo->name;
 | |
|             }
 | |
|             filepattern = filepattern.AfterFirst (';');
 | |
|         }
 | |
|     }
 | |
|     return wxEmptyString;
 | |
| 
 | |
| }
 | |
| 
 | |
| bool Edit::InitializePrefs (const wxString &name) {
 | |
| 
 | |
|     // initialize styles
 | |
|     StyleClearAll();
 | |
|     LanguageInfo const* curInfo = NULL;
 | |
| 
 | |
|     // determine language
 | |
|     bool found = false;
 | |
|     int languageNr;
 | |
|     for (languageNr = 0; languageNr < g_LanguagePrefsSize; languageNr++) {
 | |
|         curInfo = &g_LanguagePrefs [languageNr];
 | |
|         if (curInfo->name == name) {
 | |
|             found = true;
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     if (!found) return false;
 | |
| 
 | |
|     // set lexer and language
 | |
|     SetLexer (curInfo->lexer);
 | |
|     m_language = curInfo;
 | |
| 
 | |
|     // set margin for line numbers
 | |
|     SetMarginType (m_LineNrID, wxSTC_MARGIN_NUMBER);
 | |
|     StyleSetForeground (wxSTC_STYLE_LINENUMBER, wxColour (wxT("DARK GREY")));
 | |
|     StyleSetBackground (wxSTC_STYLE_LINENUMBER, *wxWHITE);
 | |
|     SetMarginWidth (m_LineNrID, 0); // start out not visible
 | |
| 
 | |
|     // annotations style
 | |
|     StyleSetBackground(ANNOTATION_STYLE, wxColour(244, 220, 220));
 | |
|     StyleSetForeground(ANNOTATION_STYLE, *wxBLACK);
 | |
|     StyleSetSizeFractional(ANNOTATION_STYLE,
 | |
|             (StyleGetSizeFractional(wxSTC_STYLE_DEFAULT)*4)/5);
 | |
| 
 | |
|     // default fonts for all styles!
 | |
|     int Nr;
 | |
|     for (Nr = 0; Nr < wxSTC_STYLE_LASTPREDEFINED; Nr++) {
 | |
|         wxFont font(wxFontInfo(10).Family(wxFONTFAMILY_MODERN));
 | |
|         StyleSetFont (Nr, font);
 | |
|     }
 | |
| 
 | |
|     // set common styles
 | |
|     StyleSetForeground (wxSTC_STYLE_DEFAULT, wxColour (wxT("DARK GREY")));
 | |
|     StyleSetForeground (wxSTC_STYLE_INDENTGUIDE, wxColour (wxT("DARK GREY")));
 | |
| 
 | |
|     // initialize settings
 | |
|     if (g_CommonPrefs.syntaxEnable) {
 | |
|         int keywordnr = 0;
 | |
|         for (Nr = 0; Nr < STYLE_TYPES_COUNT; Nr++) {
 | |
|             if (curInfo->styles[Nr].type == -1) continue;
 | |
|             const StyleInfo &curType = g_StylePrefs [curInfo->styles[Nr].type];
 | |
|             wxFont font(wxFontInfo(curType.fontsize)
 | |
|                             .Family(wxFONTFAMILY_MODERN)
 | |
|                             .FaceName(curType.fontname));
 | |
|             StyleSetFont (Nr, font);
 | |
|             if (curType.foreground) {
 | |
|                 StyleSetForeground (Nr, wxColour (curType.foreground));
 | |
|             }
 | |
|             if (curType.background) {
 | |
|                 StyleSetBackground (Nr, wxColour (curType.background));
 | |
|             }
 | |
|             StyleSetBold (Nr, (curType.fontstyle & mySTC_STYLE_BOLD) > 0);
 | |
|             StyleSetItalic (Nr, (curType.fontstyle & mySTC_STYLE_ITALIC) > 0);
 | |
|             StyleSetUnderline (Nr, (curType.fontstyle & mySTC_STYLE_UNDERL) > 0);
 | |
|             StyleSetVisible (Nr, (curType.fontstyle & mySTC_STYLE_HIDDEN) == 0);
 | |
|             StyleSetCase (Nr, curType.lettercase);
 | |
|             const char *pwords = curInfo->styles[Nr].words;
 | |
|             if (pwords) {
 | |
|                 SetKeyWords (keywordnr, pwords);
 | |
|                 keywordnr += 1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // set margin as unused
 | |
|     SetMarginType (m_DividerID, wxSTC_MARGIN_SYMBOL);
 | |
|     SetMarginWidth (m_DividerID, 0);
 | |
|     SetMarginSensitive (m_DividerID, false);
 | |
| 
 | |
|     // folding
 | |
|     SetMarginType (m_FoldingID, wxSTC_MARGIN_SYMBOL);
 | |
|     SetMarginMask (m_FoldingID, wxSTC_MASK_FOLDERS);
 | |
|     StyleSetBackground (m_FoldingID, *wxWHITE);
 | |
|     SetMarginWidth (m_FoldingID, 0);
 | |
|     SetMarginSensitive (m_FoldingID, false);
 | |
|     if (g_CommonPrefs.foldEnable) {
 | |
|         SetMarginWidth (m_FoldingID, curInfo->folds != 0? m_FoldingMargin: 0);
 | |
|         SetMarginSensitive (m_FoldingID, curInfo->folds != 0);
 | |
|         SetProperty (wxT("fold"), curInfo->folds != 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.comment"),
 | |
|                      (curInfo->folds & mySTC_FOLD_COMMENT) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.compact"),
 | |
|                      (curInfo->folds & mySTC_FOLD_COMPACT) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.preprocessor"),
 | |
|                      (curInfo->folds & mySTC_FOLD_PREPROC) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.html"),
 | |
|                      (curInfo->folds & mySTC_FOLD_HTML) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.html.preprocessor"),
 | |
|                      (curInfo->folds & mySTC_FOLD_HTMLPREP) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.comment.python"),
 | |
|                      (curInfo->folds & mySTC_FOLD_COMMENTPY) > 0? wxT("1"): wxT("0"));
 | |
|         SetProperty (wxT("fold.quotes.python"),
 | |
|                      (curInfo->folds & mySTC_FOLD_QUOTESPY) > 0? wxT("1"): wxT("0"));
 | |
|     }
 | |
|     SetFoldFlags (wxSTC_FOLDFLAG_LINEBEFORE_CONTRACTED |
 | |
|                   wxSTC_FOLDFLAG_LINEAFTER_CONTRACTED);
 | |
| 
 | |
|     // set spaces and indention
 | |
|     SetTabWidth (4);
 | |
|     SetUseTabs (false);
 | |
|     SetTabIndents (true);
 | |
|     SetBackSpaceUnIndents (true);
 | |
|     SetIndent (g_CommonPrefs.indentEnable? 4: 0);
 | |
| 
 | |
|     // others
 | |
|     SetViewEOL (g_CommonPrefs.displayEOLEnable);
 | |
|     SetIndentationGuides (g_CommonPrefs.indentGuideEnable);
 | |
|     SetEdgeColumn (80);
 | |
|     SetEdgeMode (g_CommonPrefs.longLineOnEnable? wxSTC_EDGE_LINE: wxSTC_EDGE_NONE);
 | |
|     SetViewWhiteSpace (g_CommonPrefs.whiteSpaceEnable?
 | |
|                        wxSTC_WS_VISIBLEALWAYS: wxSTC_WS_INVISIBLE);
 | |
|     SetOvertype (g_CommonPrefs.overTypeInitial);
 | |
|     SetReadOnly (g_CommonPrefs.readOnlyInitial);
 | |
|     SetWrapMode (g_CommonPrefs.wrapModeInitial?
 | |
|                  wxSTC_WRAP_WORD: wxSTC_WRAP_NONE);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool Edit::LoadFile ()
 | |
| {
 | |
| #if wxUSE_FILEDLG
 | |
|     // get filname
 | |
|     if (!m_filename) {
 | |
|         wxFileDialog dlg (this, wxT("Open file"), wxEmptyString, wxEmptyString,
 | |
|                           wxT("Any file (*)|*"), wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_CHANGE_DIR);
 | |
|         if (dlg.ShowModal() != wxID_OK) return false;
 | |
|         m_filename = dlg.GetPath();
 | |
|     }
 | |
| 
 | |
|     // load file
 | |
|     return LoadFile (m_filename);
 | |
| #else
 | |
|     return false;
 | |
| #endif // wxUSE_FILEDLG
 | |
| }
 | |
| 
 | |
| bool Edit::LoadFile (const wxString &filename) {
 | |
| 
 | |
|     // load file in edit and clear undo
 | |
|     if (!filename.empty()) m_filename = filename;
 | |
| 
 | |
|     wxStyledTextCtrl::LoadFile(m_filename);
 | |
| 
 | |
|     EmptyUndoBuffer();
 | |
| 
 | |
|     // determine lexer language
 | |
|     wxFileName fname (m_filename);
 | |
|     InitializePrefs (DeterminePrefs (fname.GetFullName()));
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool Edit::SaveFile ()
 | |
| {
 | |
| #if wxUSE_FILEDLG
 | |
|     // return if no change
 | |
|     if (!Modified()) return true;
 | |
| 
 | |
|     // get filname
 | |
|     if (!m_filename) {
 | |
|         wxFileDialog dlg (this, wxT("Save file"), wxEmptyString, wxEmptyString, wxT("Any file (*)|*"),
 | |
|                           wxFD_SAVE | wxFD_OVERWRITE_PROMPT);
 | |
|         if (dlg.ShowModal() != wxID_OK) return false;
 | |
|         m_filename = dlg.GetPath();
 | |
|     }
 | |
| 
 | |
|     // save file
 | |
|     return SaveFile (m_filename);
 | |
| #else
 | |
|     return false;
 | |
| #endif // wxUSE_FILEDLG
 | |
| }
 | |
| 
 | |
| bool Edit::SaveFile (const wxString &filename) {
 | |
| 
 | |
|     // return if no change
 | |
|     if (!Modified()) return true;
 | |
| 
 | |
| //     // save edit in file and clear undo
 | |
| //     if (!filename.empty()) m_filename = filename;
 | |
| //     wxFile file (m_filename, wxFile::write);
 | |
| //     if (!file.IsOpened()) return false;
 | |
| //     wxString buf = GetText();
 | |
| //     bool okay = file.Write (buf);
 | |
| //     file.Close();
 | |
| //     if (!okay) return false;
 | |
| //     EmptyUndoBuffer();
 | |
| //     SetSavePoint();
 | |
| 
 | |
| //     return true;
 | |
| 
 | |
|     return wxStyledTextCtrl::SaveFile(filename);
 | |
| 
 | |
| }
 | |
| 
 | |
| bool Edit::Modified () {
 | |
| 
 | |
|     // return modified state
 | |
|     return (GetModify() && !GetReadOnly());
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // EditProperties
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| EditProperties::EditProperties (Edit *edit,
 | |
|                                 long style)
 | |
|         : wxDialog (edit, wxID_ANY, wxEmptyString,
 | |
|                     wxDefaultPosition, wxDefaultSize,
 | |
|                     style | wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER) {
 | |
| 
 | |
|     // sets the application title
 | |
|     SetTitle (_("Properties"));
 | |
|     wxString text;
 | |
| 
 | |
|     // fullname
 | |
|     wxBoxSizer *fullname = new wxBoxSizer (wxHORIZONTAL);
 | |
|     fullname->Add (10, 0);
 | |
|     fullname->Add (new wxStaticText (this, wxID_ANY, _("Full filename"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL);
 | |
|     fullname->Add (new wxStaticText (this, wxID_ANY, edit->GetFilename()),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL);
 | |
| 
 | |
|     // text info
 | |
|     wxGridSizer *textinfo = new wxGridSizer (4, 0, 2);
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, _("Language"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, edit->m_language->name),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, _("Lexer-ID: "),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     text = wxString::Format (wxT("%d"), edit->GetLexer());
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, text),
 | |
|                    0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
|     wxString EOLtype = wxEmptyString;
 | |
|     switch (edit->GetEOLMode()) {
 | |
|         case wxSTC_EOL_CR: {EOLtype = wxT("CR (Unix)"); break; }
 | |
|         case wxSTC_EOL_CRLF: {EOLtype = wxT("CRLF (Windows)"); break; }
 | |
|         case wxSTC_EOL_LF: {EOLtype = wxT("CR (Macintosh)"); break; }
 | |
|     }
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, _("Line endings"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     textinfo->Add (new wxStaticText (this, wxID_ANY, EOLtype),
 | |
|                    0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
| 
 | |
|     // text info box
 | |
|     wxStaticBoxSizer *textinfos = new wxStaticBoxSizer (
 | |
|                      new wxStaticBox (this, wxID_ANY, _("Informations")),
 | |
|                      wxVERTICAL);
 | |
|     textinfos->Add (textinfo, 0, wxEXPAND);
 | |
|     textinfos->Add (0, 6);
 | |
| 
 | |
|     // statistic
 | |
|     wxGridSizer *statistic = new wxGridSizer (4, 0, 2);
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, _("Total lines"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                     0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     text = wxString::Format (wxT("%d"), edit->GetLineCount());
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, text),
 | |
|                     0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, _("Total chars"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                     0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     text = wxString::Format (wxT("%d"), edit->GetTextLength());
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, text),
 | |
|                     0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, _("Current line"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                     0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     text = wxString::Format (wxT("%d"), edit->GetCurrentLine());
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, text),
 | |
|                     0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, _("Current pos"),
 | |
|                                      wxDefaultPosition, wxSize(80, wxDefaultCoord)),
 | |
|                     0, wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL|wxLEFT, 4);
 | |
|     text = wxString::Format (wxT("%d"), edit->GetCurrentPos());
 | |
|     statistic->Add (new wxStaticText (this, wxID_ANY, text),
 | |
|                     0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxRIGHT, 4);
 | |
| 
 | |
|     // char/line statistics
 | |
|     wxStaticBoxSizer *statistics = new wxStaticBoxSizer (
 | |
|                      new wxStaticBox (this, wxID_ANY, _("Statistics")),
 | |
|                      wxVERTICAL);
 | |
|     statistics->Add (statistic, 0, wxEXPAND);
 | |
|     statistics->Add (0, 6);
 | |
| 
 | |
|     // total pane
 | |
|     wxBoxSizer *totalpane = new wxBoxSizer (wxVERTICAL);
 | |
|     totalpane->Add (fullname, 0, wxEXPAND | wxLEFT | wxRIGHT | wxTOP, 10);
 | |
|     totalpane->Add (0, 6);
 | |
|     totalpane->Add (textinfos, 0, wxEXPAND | wxLEFT | wxRIGHT, 10);
 | |
|     totalpane->Add (0, 10);
 | |
|     totalpane->Add (statistics, 0, wxEXPAND | wxLEFT | wxRIGHT, 10);
 | |
|     totalpane->Add (0, 6);
 | |
|     wxButton *okButton = new wxButton (this, wxID_OK, _("OK"));
 | |
|     okButton->SetDefault();
 | |
|     totalpane->Add (okButton, 0, wxALIGN_CENTER | wxALL, 10);
 | |
| 
 | |
|     SetSizerAndFit (totalpane);
 | |
| 
 | |
|     ShowModal();
 | |
| }
 | |
| 
 | |
| #if wxUSE_PRINTING_ARCHITECTURE
 | |
| 
 | |
| //----------------------------------------------------------------------------
 | |
| // EditPrint
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| EditPrint::EditPrint (Edit *edit, const wxChar *title)
 | |
|               : wxPrintout(title) {
 | |
|     m_edit = edit;
 | |
|     m_printed = 0;
 | |
| 
 | |
| }
 | |
| 
 | |
| bool EditPrint::OnPrintPage (int page) {
 | |
| 
 | |
|     wxDC *dc = GetDC();
 | |
|     if (!dc) return false;
 | |
| 
 | |
|     // scale DC
 | |
|     PrintScaling (dc);
 | |
| 
 | |
|     // print page
 | |
|     if (page == 1) m_printed = 0;
 | |
|     m_printed = m_edit->FormatRange (1, m_printed, m_edit->GetLength(),
 | |
|                                      dc, dc, m_printRect, m_pageRect);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool EditPrint::OnBeginDocument (int startPage, int endPage) {
 | |
| 
 | |
|     if (!wxPrintout::OnBeginDocument (startPage, endPage)) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void EditPrint::GetPageInfo (int *minPage, int *maxPage, int *selPageFrom, int *selPageTo) {
 | |
| 
 | |
|     // initialize values
 | |
|     *minPage = 0;
 | |
|     *maxPage = 0;
 | |
|     *selPageFrom = 0;
 | |
|     *selPageTo = 0;
 | |
| 
 | |
|     // scale DC if possible
 | |
|     wxDC *dc = GetDC();
 | |
|     if (!dc) return;
 | |
|     PrintScaling (dc);
 | |
| 
 | |
|     // get print page informations and convert to printer pixels
 | |
|     wxSize ppiScr;
 | |
|     GetPPIScreen (&ppiScr.x, &ppiScr.y);
 | |
|     wxSize page = g_pageSetupData->GetPaperSize();
 | |
|     page.x = static_cast<int> (page.x * ppiScr.x / 25.4);
 | |
|     page.y = static_cast<int> (page.y * ppiScr.y / 25.4);
 | |
|     m_pageRect = wxRect (0,
 | |
|                          0,
 | |
|                          page.x,
 | |
|                          page.y);
 | |
| 
 | |
|     // get margins informations and convert to printer pixels
 | |
|     wxPoint pt = g_pageSetupData->GetMarginTopLeft();
 | |
|     int left = pt.x;
 | |
|     int top = pt.y;
 | |
|     pt = g_pageSetupData->GetMarginBottomRight();
 | |
|     int right = pt.x;
 | |
|     int bottom = pt.y;
 | |
| 
 | |
|     top = static_cast<int> (top * ppiScr.y / 25.4);
 | |
|     bottom = static_cast<int> (bottom * ppiScr.y / 25.4);
 | |
|     left = static_cast<int> (left * ppiScr.x / 25.4);
 | |
|     right = static_cast<int> (right * ppiScr.x / 25.4);
 | |
| 
 | |
|     m_printRect = wxRect (left,
 | |
|                           top,
 | |
|                           page.x - (left + right),
 | |
|                           page.y - (top + bottom));
 | |
| 
 | |
|     // count pages
 | |
|     while (HasPage (*maxPage)) {
 | |
|         m_printed = m_edit->FormatRange (0, m_printed, m_edit->GetLength(),
 | |
|                                        dc, dc, m_printRect, m_pageRect);
 | |
|         *maxPage += 1;
 | |
|     }
 | |
|     if (*maxPage > 0) *minPage = 1;
 | |
|     *selPageFrom = *minPage;
 | |
|     *selPageTo = *maxPage;
 | |
| }
 | |
| 
 | |
| bool EditPrint::HasPage (int WXUNUSED(page)) {
 | |
| 
 | |
|     return (m_printed < m_edit->GetLength());
 | |
| }
 | |
| 
 | |
| bool EditPrint::PrintScaling (wxDC *dc){
 | |
| 
 | |
|     // check for dc, return if none
 | |
|     if (!dc) return false;
 | |
| 
 | |
|     // get printer and screen sizing values
 | |
|     wxSize ppiScr;
 | |
|     GetPPIScreen (&ppiScr.x, &ppiScr.y);
 | |
|     if (ppiScr.x == 0) { // most possible guess 96 dpi
 | |
|         ppiScr.x = 96;
 | |
|         ppiScr.y = 96;
 | |
|     }
 | |
|     wxSize ppiPrt;
 | |
|     GetPPIPrinter (&ppiPrt.x, &ppiPrt.y);
 | |
|     if (ppiPrt.x == 0) { // scaling factor to 1
 | |
|         ppiPrt.x = ppiScr.x;
 | |
|         ppiPrt.y = ppiScr.y;
 | |
|     }
 | |
|     wxSize dcSize = dc->GetSize();
 | |
|     wxSize pageSize;
 | |
|     GetPageSizePixels (&pageSize.x, &pageSize.y);
 | |
| 
 | |
|     // set user scale
 | |
|     float scale_x = (float)(ppiPrt.x * dcSize.x) /
 | |
|                     (float)(ppiScr.x * pageSize.x);
 | |
|     float scale_y = (float)(ppiPrt.y * dcSize.y) /
 | |
|                     (float)(ppiScr.y * pageSize.y);
 | |
|     dc->SetUserScale (scale_x, scale_y);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| #endif // wxUSE_PRINTING_ARCHITECTURE
 |