Modern Bindings for the Text Widget

KJN: See ntext for the completed version of this package.

  • The tasks in the "To Do" list below have been completed (except writing the TIP; feedback from users would be helpful before the TIP is written).
  • The package has been renamed from "modernText" to "ntext" to avoid confusion with the Computer Modern fonts (my thanks to the person who pointed this out).
  • The old page on "modernText" is below.

KJN: This page defines a binding tag, modernText, as an alternative to the text widget's default binding tag, Text.

The purpose of modernText is to make the text widget behave more like a "modern" (Year 2005 CE) text editor. It makes the text widget more useful for implementing a text editor, and makes it behave in a way that is more familiar to many users.

modernText is implemented as a binding tag, and apart from these bindings its code is contained entirely in the ::modernText namespace, with no exports to the global or other namespaces, and no new widget commands. It uses modified copies of the Tk code, leaving the original code, and the Text binding tag, unchanged.

'Vices' of the default bindings (that are 'cured' in modernText):

  • clicking near the end of a line moves the cursor to the start of the next line. Double-clicking or dragging may highlight and select the space at the end of the line (notably the region where there are no characters). It may select the first word of the next line. It may make a nonsense selection.
  • the Home and End keys could be improved - these have been changed in recent versions of Tcl/Tk to move to the beginning or end of a display line, rather than of a logical line; further improvement is possible - successive keypresses can implement "Smart Home" and "Smart End".
  • When a selection exists, a <<Paste>> operation (e.g. <Control-v>) deletes the selection (as modern editors do) - unless you are using X11 (the windowing system for Linux etc).
  • The <Escape> key does not clear the selection.
  • Selecting with <Shift-Button1> may select from the position of the previous mouse click, not the previous position of the insertion cursor.
  • <Shift-Button1> operations can move the selection anchor.

The last two 'vices' are often useful features, so modernText gives you the option of retaining them, by setting variables defined in the ::modernText namespace to 1 (instead of their default 0). Explaining these vices/features in more detail:

  • If the mouse is clicked at position A, then the keyboard is used to move the cursor to B, then shift is held down, and the mouse is clicked at C: the Text binding tag gives a selection from A to C; the modernText gives a selection from B to C. If you want modernText to behave like Text in this respect, set ::modernText::mouseSelectIgnoresKbd to 1.
  • The Text binding tag allows successive Shift-Button-1 events to change both ends of the selection, by moving the selection anchor to the end of the selection furthest from the mouse click. Instead, the modernText binding tag fixes the anchor, and multiple Shift-Button-1 events can only move the non-anchored end of the selection. If you want modernText to behave like Text in this respect, set ::modernText::variableAnchor to 1. In both Text and modernText, keyboard navigation with the Shift key held down alters the selection and keeps the selection anchor fixed.

I have worked on the bindings that I personally use a lot, but there are many others that I do not use - so I would appreciate feedback (on this Wiki page) on any features in modernText, particularly bindings or other features that may have been broken by these changes.

modernText is mainly intended for text widgets whose -wrap mode is "none" or "word". For a text widget that is wrapped in "char" mode, display line ends are much less meaningful, and so, for "-wrap char", modernText's rules on crossing line ends are relaxed for display lines (but not for logical lines).

Requirements

The code requires Tk 8.5; the latest development release (ActiveTcl 8.5 beta 4, 8.5a4) has a bug in the text widget ([L1 ], now fixed in CVS - thanks Vincent). So I am using an earlier version, ActiveTcl 8.5 beta 3 (8.5a2). I recommend using this version for text widget work until a bugfixed version of 8.5 is released; or, alternatively, use Tcl/Tk from CVS, but make sure you have version 1.61 of tk/generic/tkText.c, dated Mon Oct 31 23:21:08 2005 UTC, which has the bug fix; and delete the test of $::tk_patchLevel from the start of modernText: CVS still uses patchLevel 8.5a4.

To use modernText, source the modernText code in the first box below, and then use the bindtags command to specify the modernText binding tag (instead of the default Text) for your text widget, e.g.

    bindtags .t {.t modernText . all}

The second box below has a simple demo program.

To Do

  • improvement of word selection functions in lib/tcl8.5/word.tcl
  • indentation of wrapped lines using tags and -lmargin2
  • write a TIP to change the text widget's core bindings (Vince's suggestion - see discussion at end of page)

The modernText code:

  namespace eval ::modernText {

  # Some parts of this code are copied with modifications from lib/tk8.5/text.tcl which is
  # Copyright (c) 1992-1994 The Regents of the University of California.
  # Copyright (c) 1994-1997 Sun Microsystems, Inc.
  # Copyright (c) 1998 by Scriptics Corporation.

  # Any part of this file that is not copied from lib/tk8.5/text.tcl is Copyright (c) 2005 K. J. Nash and 
  # other contributors to https://wiki.tcl-lang.org/14918
  # You are hereby granted a permanent and irrevocable license to use modify and redistribute this file subject
  # to the terms of the TCL LICENSE AGREEMENT - see https://www.tcl-lang.org/software/tcltk/license.html for
  # further information and note in particular the DISCLAIMER OF ALL WARRANTIES.
  # All other rights reserved.

  if {[catch {package require Tk 8.5}]} {
     puts "Package modernText requires Tk 8.5 or above."
     puts "When obtaining 8.5, note that Tk Version 8.5a4 has a bug (#1333951) in its text widget."
     puts "Please do not use Tcl/Tk 8.5a4 unless you are sure the bug has been fixed."
     exit 1
  }

  if {$::tk_patchLevel eq "8.5a4"} {
   puts "Tk Version 8.5a4 has a bug (#1333951) in its text widget."
   puts "Please do not use Tcl/Tk 8.5a4 unless you are sure the bug has been fixed."
   exit 1
  }

                                     #  Set variables to 1 for "Tk text default" style, 0 for "modern" style
  variable mouseSelectIgnoresKbd 0  ;#  Whether Shift-Button-1 ignores changes made by the kbd to the insert mark
  variable variableAnchor        0  ;#  Whether Shift-Button-1 has a variable or fixed anchor

  variable bCount 0

  ### Two new functions, homeIndex and nameIndex, that can be used for "smart" Home and End operations

  proc homeIndex {w index} {
     # Return the index to jump to (from $index) as "Smart Home"
     # Some corner cases (e.g. lots of leading whitespace, wrapped around) probably have a better solution; but
     # there's no consensus on how a text editor should behave in such cases.

     set index   [$w index $index]
     set dls     [$w index "$index display linestart"]

     # Set firstNonSpace to the index of the first non-space character on the logical line.
     set dlsList [split $dls .]
     set dlsLine [lindex $dlsList 0]
     set lls     $dlsLine.0
     set firstNonSpace [$w search -regexp -- {[^[:space:]]} $dlsLine.0 [expr {$dlsLine + 1}].0]

     # Now massage $firstNonSpace so it contains the "usual" home position on the first display line
     if {$firstNonSpace eq {}} {
        # No non-whitespace characters on the line
        set firstNonSpace $dlsLine.0
     } elseif {[$w count -displaylines $lls $firstNonSpace] != 0} {
        # Either lots of whitespace, or whitespace with character wrap forces $firstNonSpace onto the next
        # display line
        set firstNonSpace $dlsLine.0
     } else {
        # The usual case: the first non-whitespace $firstNonSpace is on the first display line
     }

     if {$dls eq $lls} {
        # We're on the first display line
        if {$index eq $firstNonSpace} {
           # we're at the first non-whitespace of the first display line
           set home $lls
        } else {
           # we're on the first display line, but not at the first non-whitespace
           set home $firstNonSpace
        }
     } else {
        if {$dls eq $index} {
           # we're at the start of a display line other than the first
           set home $firstNonSpace
        } else {
           # we're not on the first display line, and we're not at our display line's start
           set home $dls
        }
     }
     return $home
  }

  proc endIndex {w index} {
     # Return the index to jump to (from $index) as "Smart End"
     set index    [$w index $index]
     set dle      [$w index "$index display lineend"]

     if {$dle eq $index} {
         # we're at the end of a display line: return the logical line end
         return [$w index "$index lineend"]
     } else {
         # return the display line end
         return $dle
     }
  }

  # Make sure that each function we want to copy and modify is loaded - probably unnecessary
  foreach function {
      ::tk_textPaste
      ::tk::TextPasteSelection
      ::tk::TextButton1
      ::tk::TextSelectTo
      ::tk::TextAutoScan
  } {catch $function}

  ### A new function, ::modernText::modernPaste, to replace ::tk_textPaste in bindings - to switch off the
  ### code that makes it behave differently in X11 from other windowing systems.  X11 desktops such as KDE
  ### and GNOME made this change years ago - making the Tk defaults appear anachronistic.

  proc modernPaste w [string map {x11 x11TheOldFashionedWay} [info body ::tk_textPaste]]

  ### Two procs that are copied from ::tk with modifications:

  ### Modify TextClosestGap to fix the jump-to-next-line issue
  ### Modify TextSelectTo to prevent word selection from crossing a line end

  proc TextClosestGap {w x y} {
      # Modified from function ::tk::TextClosestGap
      set pos [$w index @$x,$y]
      set bbox [$w bbox $pos]
      if {$bbox eq ""} {
          return $pos
      }
      if {($x - [lindex $bbox 0]) < ([lindex $bbox 2]/2)} {
          return $pos
      }
      # Never return a position that will place the cursor on the next display line.
      # This used to happen if $x is closer to the end of the display line than to its last character.
      if {[$w cget -wrap] eq "word"} {
          set lineType displaylines
      } else {
          set lineType lines
      }
      if {[$w count -$lineType $pos "$pos + 1 char"] != 0} {
      return $pos
      } else {
      }
      $w index "$pos + 1 char"
  }


  proc TextSelectTo {w x y {extend 0}} {
      # Modified from function ::tk::TextSelectTo
      global tcl_platform
      variable ::tk::Priv

      set cur [TextClosestGap $w $x $y]
      if {[catch {$w index tk::anchor$w}]} {
          $w mark set tk::anchor$w $cur
      }
      set anchor [$w index tk::anchor$w]
      if {[$w compare $cur != $anchor] || (abs($Priv(pressX) - $x) >= 3)} {
          set Priv(mouseMoved) 1
      }
      switch -- $Priv(selectMode) {
          char {
              if {[$w compare $cur < tk::anchor$w]} {
                  set first $cur
                  set last tk::anchor$w
              } else {
                  set first tk::anchor$w
                  set last $cur
              }
          }
          word {
              # Set initial range based only on the anchor (1 char min width - MOD - unless this straddles a
              # display line end)
          if {[$w cget -wrap] eq "word"} {
              set lineType displaylines
          } else {
              set lineType lines
          }
              if {[$w mark gravity tk::anchor$w] eq "right"} {
                  set first "tk::anchor$w"
                  set last "tk::anchor$w + 1c"
                  if {[$w count -$lineType $first $last] != 0} {
                          set last $first
                  } else {
                  }
              } else {
                  set first "tk::anchor$w - 1c"
                  set last "tk::anchor$w"
                  if {[$w count -$lineType $first $last] != 0} {
                          set first $last
                  } else {
                  }
              }
          if {$last eq $first && [$w index $first] eq $cur} {
              # Use $first and $last as above; further extension will straddle a display line.
              # Better to have no selection than a bad one.
          } else {
              # Extend range (if necessary) based on the current point
              if {[$w compare $cur < $first]} {
                  set first $cur
              } elseif {[$w compare $cur > $last]} {
                  set last $cur
              }

              # Now find word boundaries
              set first1 [$w index "$first + 1c"]
              set last1  [$w index "$last - 1c"]
                  if {[$w count -$lineType $first $first1] != 0} {
                          set first1 [$w index $first]
                  } else {
                  }
                  if {[$w count -$lineType $last $last1] != 0} {
                          set last1 [$w index $last]
                  } else {
                  }
              set first2 [::tk::TextPrevPos $w "$first1" tcl_wordBreakBefore]
              set last2  [::tk::TextNextPos $w "$last1"  tcl_wordBreakAfter]
              # Don't allow a "word" to straddle a display line boundary (or, in -wrap char mode, a logical line
              # boundary). Not the right result if -wrap word has been forced into -wrap char because a word is
              # too long.
              # tcl_wordBreakBefore and tcl_wordBreakAfter need fixing too.
              if {[$w count -$lineType $first2 $first] != 0} {
                  set first [$w index "$first display linestart"]
              } else {
                  set first $first2
              }
              if {[$w count -$lineType $last2 $last] != 0} {
                  set last [$w index "$last display lineend"]
              } else {
                  set last $last2
              }
          }
          }
          line {
              # Set initial range based only on the anchor
              set first "tk::anchor$w linestart"
              set last "tk::anchor$w lineend"

              # Extend range (if necessary) based on the current point
              if {[$w compare $cur < $first]} {
                  set first "$cur linestart"
              } elseif {[$w compare $cur > $last]} {
                  set last "$cur lineend"
              }
              set first [$w index $first]
              set last [$w index "$last + 1c"]
          }
      }
      if {$Priv(mouseMoved) || ($Priv(selectMode) ne "char")} {
          $w tag remove sel 0.0 end
          $w mark set insert $cur
          $w tag add sel $first $last
          $w tag remove sel $last end
          update idletasks
      }
  }

  ### (a) The procs in ::tk that we have copied to ::modernText and modified (above) are called directly or 
  ###     indirectly by several procs in ::tk.  Copy and modify these procs too, so that a widget of class
  ###     modernText always uses the ::modernText procs defined above, even if they are called from other procs.

  ### The copy-and-modify code below will likely break when the ::tk code is revised - but this technique makes
  ### the ::modernText code short, and shows exactly what has changed.

  proc TextPasteSelection {w x y} [info body ::tk::TextPasteSelection]

  proc TextButton1 {w x y} [info body ::tk::TextButton1]

  proc TextAutoScan {w}  [string map {tk::TextAutoScan modernText::TextAutoScan} [info body ::tk::TextAutoScan]]


  ### (b) Now make sure that widgets of class modernText always bind to the ::modernText procs defined above
  ### The ::tk namespace, ::tk_textPaste, and the Text binding tag remain in their pristine state.

  ### modernText procs replace these functions - all except ::tk::TextClosestGap occur in bindings
  #    ::tk_textPaste
  #    ::tk::TextClosestGap
  #    ::tk::TextSelectTo
  #    ::tk::TextPasteSelection
  #    ::tk::TextButton1
  #    ::tk::TextAutoScan


  proc copyBindingClass {class newClass {mapping {}}} {
      # call this proc to make $newClass inherit the bindings of $class, but with some substitutions
      # Derived from https://wiki.tcl-lang.org/2944 by George Peter Staplin
      set bindingList [bind $class]
      foreach binding $bindingList {
          bind $newClass $binding [string map $mapping [bind $class $binding]]
      }
  }

  copyBindingClass Text modernText  {
     tk_textPaste            modernText::modernPaste
     tk::TextSelectTo        modernText::TextSelectTo
     tk::TextPasteSelection  modernText::TextPasteSelection
     tk::TextButton1         modernText::TextButton1
     tk::TextAutoScan        modernText::TextAutoScan
  }


  # Now alter some of the bindings

  # Keyboard bindings to implement "Smart Home/End" and Escape (to clear the selection)
  bind modernText <Home>  {
      tk::TextSetCursor %W  [::modernText::homeIndex %W insert]
  }

  bind modernText <End>  {
      tk::TextSetCursor %W  [::modernText::endIndex %W insert]
  }

  bind modernText <Shift-Home> {
      tk::TextKeySelect %W [::modernText::homeIndex %W insert]
  }

  bind modernText <Shift-End> {
      tk::TextKeySelect %W [::modernText::endIndex %W insert]
  }

  bind modernText <Escape>  {
      %W tag remove sel 0.0 end
  }

  # Mouse bindings: when the modernText bindings are copied from the Text bindings by copyBindingClass (above),
  # they are modified so that they use the modernText functions.  The further modifications below:
  # (1) Use ::modernText::bCount to deal with out-of-order multiple clicks
  # (2) (With Shift modifier only) Use ::modernText::mouseSelectIgnoresKbd and ::modernText::variableAnchor to
  #     change the text selection algorithm.

  bind modernText <1> {
      set ::modernText::bCount 1
      modernText::TextButton1 %W %x %y
      %W tag remove sel 0.0 end
  }

  bind modernText <Double-1> {
      if {$::modernText::bCount != 1} {
          # The previous Button-1 event was not a single-click, but a double, triple, or quadruple.
          # We can simplify the bindings if we ensure that a double-click is *always* preceded by a single-click.
          # So inject a <1> handler before doing <Double-1> ...
          set ::modernText::bCount 1
          modernText::TextButton1 %W %x %y
          %W tag remove sel 0.0 end
          # ... end of copied <1> handler ...
      }
      #     ... now process the <Double-1> event.
      set ::modernText::bCount 2
      set tk::Priv(selectMode) word
      modernText::TextSelectTo %W %x %y
      catch {%W mark set insert sel.first}
  }

  bind modernText <Triple-1> {
      if {$::modernText::bCount != 2} {
          # ignore an out-of-order triple click.  This has no adverse consequences.
          continue
      }
      set ::modernText::bCount 3
      set tk::Priv(selectMode) line
      modernText::TextSelectTo %W %x %y
      catch {%W mark set insert sel.first}
  }

  bind modernText <Quadruple-1> {
      # don't care if a quadruple click is out-of-order (i.e. follows a quadruple click, not a triple click).
      # the binding does nothing except set bCount.
      set ::modernText::bCount 4
  }


  bind modernText <Shift-1> {
      set ::modernText::bCount 1
      if {!$::modernText::mouseSelectIgnoresKbd && [%W tag ranges sel] eq ""} {
          # Move the selection anchor mark to the old insert mark
          # Should the mark's gravity be set?
          %W mark set tk::anchor%W insert
      }

      if {$::modernText::variableAnchor} {
          tk::TextResetAnchor %W @%x,%y
          # if sel exists, sets anchor to end furthest from x,y
          # changes anchor only, not insert
      }

      set tk::Priv(selectMode) char
      modernText::TextSelectTo %W %x %y
  }

  bind modernText <Double-Shift-1>        {
      if {$::modernText::bCount != 1} {
          # The previous Button-1 event was not a single-click, but a double, triple, or quadruple.
          # We can simplify the bindings if we ensure that a double-click is *always* preceded by a single-click.
          # So inject a <Shift-1> handler before doing <Double-Shift-1> ...
          set ::modernText::bCount 1
          if {!$::modernText::mouseSelectIgnoresKbd && [%W tag ranges sel] eq ""} {
              # Move the selection anchor mark to the old insert mark
              # Should the mark's gravity be set?
              %W mark set tk::anchor%W insert
          }

          if {$::modernText::variableAnchor} {
              tk::TextResetAnchor %W @%x,%y
              # if sel exists, sets anchor to end furthest from x,y
              # changes anchor only, not insert
          }

          set tk::Priv(selectMode) char
          modernText::TextSelectTo %W %x %y
          # ... end of copied <Shift-1> handler ...

      }
      #     ... now process the <Double-Shift-1> event.
      set ::modernText::bCount 2
      set tk::Priv(selectMode) word
      modernText::TextSelectTo %W %x %y 1
  }

  bind modernText <Triple-Shift-1>        {
      if {$::modernText::bCount != 2} {
          # ignore an out-of-order triple click.  This has no adverse consequences.
          continue
      }
      set ::modernText::bCount 3
      set tk::Priv(selectMode) line
      modernText::TextSelectTo %W %x %y
  }

  bind modernText <Quadruple-Shift-1> {
      # don't care if a quadruple click is out-of-order (i.e. follows a quadruple click, not a triple click).
      # the binding does nothing except set bCount.
      set ::modernText::bCount 4
  }

  }

Demo Program for modernText

  append message \
  "\n" \
  "QOTW:  \"C/C++, which is used by 16% of users, is the most popular programming language, " \
  "but Tcl, used by 0%, seems to be the language of choice for the highest scoring users.\"\n" \
  "(new line)\n" \
  "Some Tcl Source Code:\n" \
  "\n" \
  "bind modernText <1> {\n" \
  "    set ::modernText::bCount 1\n" \
  "    modernText::TextButton1 %W %x %y\n" \
  "    %W tag remove sel 0.0 end\n" \
  "}"

  # save the modernText file from https://wiki.tcl-lang.org/14918 to modernText.tcl
  source modernText.tcl

  #  Whether Shift-Button-1 ignores changes made by the kbd to the insert mark:
  set ::modernText::mouseSelectIgnoresKbd 0

  #  Whether Shift-Button-1 has a variable or fixed anchor:
  set ::modernText::variableAnchor        0


  pack [text .t ] -side right
  .t configure -width 37 -height 20 -wrap word -font {{Courier New} -15} -bg white
  .t insert end "  I use the modernText bindings.\n$message"

  bindtags .t {.t modernText . all}

  pack [text .u ] -side right
  .u configure -width 37 -height 20 -wrap word -font {{Courier New} -15} -bg #FFFFEE
  .u insert end "  I use the (default) Text bindings.\n$message"

modernText should not be confused with text set in Computer Modern typeface [L2 ].


DKF: What are the license terms on this code? Is it worthwhile trying to adopt this sort of thing into the Tk core text widget's bindings?


KJN: Tcl License. You are welcome to use it in the core, or Tklib. I would be very pleased if the text default bindings could be changed, but I suppose that depends on how many people prefer the existing bindings. It is clear from the code that some aspects of the default bindings are deliberate decisions - e.g.

  • pasting when a selection exists - the default code specifically checks whether it is using X11 so that it does not delete the selection in this case
  • leaving <Escape> unbound

so it depends really what people prefer.


Vince: Why not write a TIP to change the text widget's core bindings?

KJN: Thanks for the suggestion. I've added it to the To-Do list.


[Category Example|Category Widget]