mirror of
https://github.com/ruby/ruby.git
synced 2025-09-18 18:13:58 +02:00

* ext/tk/lib/tkextlib/tile.rb: [incompatible] remove TileWidgets' instate/state/identify method to avoid the conflict with standard widget options. Those methods are renamed to ttk_instate/ttk_state/ ttk_identify (tile_instate/tile_state/tile_identify are available too). Although I don't recommend, if you realy need old methods, please define "Tk::USE_OBSOLETE_TILE_STATE_METHOD = true" before "require 'tkextlib/tile'". * ext/tk/lib/tkextlib/tile.rb: "Tk::Tile::__Import_Tile_Widgets__!" is obsolete. It outputs warning. To control default widget set, use "Tk.default_widget_set = :Ttk". * ext/tk/lib/tk.rb: __IGNORE_UNKNOWN_CONFIGURE_OPTION__ method and __set_IGNORE_UNKNOWN_CONFIGURE_OPTION__!(mode) method are defind as module methods of TkConfigMethod. It may help users to wrap old Ruby/Tk scripts (use standard widgets) to force to use Ttk widgets. Ttk widgets don't have some options of standard widgets which are control the view of widgets. When set ignore-mode true, configure method tries to ignoure such unknown options with no exception. Of course, it may raise other troubles on the GUI design. So, those are a little danger methods. * ext/tk/lib/tk/itemconfig.rb: __IGNORE_UNKNOWN_CONFIGURE_OPTION__ method and __set_IGNORE_UNKNOWN_CONFIGURE_OPTION__!(mode) method are defind as module methods of TkItemConfigMethod as the same purpose as TkConfigMethod's ones. * ext/tk/sample/ttk_wrapper.rb: A new example. This is a tool for wrapping old Ruby/Tk scripts (which use standard widgets) to use Ttk (Tile) widgets as default. * ext/tk/sample/tkextlib/tile/demo.rb: use ttk_instate/ttk_state method instead of instate/state method. * ext/tk/lib/tk/root, ext/tk/lib/tk/namespace.rb, ext/tk/lib/tk/text.rb, ext/tk/lib/tkextlib/*: some 'instance_eval's are replaced to "instance_exec(self)". * ext/tk/lib/tk/event.rb: bug fix on KEY_TBL and PROC_TBL (?x is not a character code on Ruby1.9). * ext/tk/lib/tk/variable.rb: support new style of operation argument on Tcl/Tk's 'trace' command for variables. * ext/tk/sample/demos-jp/widget, ext/tk/sample/demos-en/widget: bug fix * ext/tk/sammple/demos-jp/textpeer.rb, ext/tk/sammple/demos-en/textpeer.rb: new widget demo. * ext/tk/tcltklib.c: decrase SEGV troubles (probably) * ext/tk/lib/tk.rb: remove Thread.critical access if Ruby1.9 * ext/tk/lib/tk/multi-tk.rb: support Ruby1.9 (probably) * ext/tk/lib/tkextlib/tile.rb: add method to define Tcl/Tk command to make Tcl/Tk theme sources (based on different version of Tile extension) available. (Tk::Tile::__define_LoadImages_proc_for_comaptibility__) * ext/tk/lib/tk.rb, ext/tk/lib/tk/wm.rb: support dockable frames (Tcl/Tk8.5 feature). 'wm' command can treat many kinds of widgets as toplevel widgets. * ext/tk/lib/tkextlib/tile/style.rb: ditto. (Tk::Tile::Style.__define_wrapper_proc_for_compatibility__) * ext/tk/lib/tk/font.rb: add actual_hash and metrics_hash to get properties as a hash. metrics_hash method returns a boolean value for 'fixed' option. But metrics method returns numeric value (0 or 1) for 'fixed' option, because of backward compatibility. * ext/tk/lib/tk/timer.rb: somtimes fail to set callback procedure. * ext/tk/lib/tk.rb: add Tk.sleep and Tk.wakeup method. Tk.sleep doesn't block the eventloop. It will be better to use the method in event callbacks. * ext/tk/sample/tksleep_sample.rb: sample script about Tk.sleep. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_1_8@15849 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
205 lines
7 KiB
Ruby
205 lines
7 KiB
Ruby
# -*- coding: euc-jp -*-
|
||
# entry3.rb --
|
||
#
|
||
# This demonstration script creates several entry widgets whose
|
||
# permitted input is constrained in some way. It also shows off a
|
||
# password entry.
|
||
#
|
||
# based on Tcl/Tk8.4.4 widget demos
|
||
|
||
if defined?($entry3_demo) && $entry3_demo
|
||
$entry3_demo.destroy
|
||
$entry3_demo = nil
|
||
end
|
||
|
||
$entry3_demo = TkToplevel.new {|w|
|
||
title("Constrained Entry Demonstration")
|
||
iconname("entry3")
|
||
positionWindow(w)
|
||
}
|
||
|
||
TkLabel.new($entry3_demo,
|
||
:font=>$font, :wraplength=>'5i', :justify=>:left,
|
||
:text=><<EOL).pack(:side=>:top)
|
||
以下には4種類のエントリボックスが表示されています.各エントリボックスは,\
|
||
マウスクリックで選択し文字を打ち込むことが可能ですが,それぞれがどのような\
|
||
入力を受け付けることができるかには制約が設けられています.\
|
||
一つめのエントリボックスは整数と見なされる文字列か入力文字がない空の状態か\
|
||
の場合だけを受け付け,問題がある場合はエントリボックスが点滅します\
|
||
(フォーカスが去る時にチェックされます).\
|
||
二つめのエントリボックスは,入力された文字列の長さが\
|
||
10文字未満の場合だけを受け付け,制限を越えて書き込もうとしたときには\
|
||
ベルを鳴らして知らせます.\
|
||
三つめは米国の電話番号を受け付けるエントリボックスです.\
|
||
アルファベットは,電話機のダイヤル上で対応づけられている数字に変換されます.\
|
||
不適切な文字が入力されたり数字以外の文字の位置に数字を入力しようとしたり\
|
||
した場合には警告のベルが鳴ります.\
|
||
四つめのエントリボックスは,8文字までの入力を受け付ける\
|
||
パスワードフィールドです(8文字以上は特に警告を出すことなく無視されます).\
|
||
入力された文字はアスタリスク記号に置き換えて表示されます.
|
||
EOL
|
||
|
||
TkFrame.new($entry3_demo){|f|
|
||
pack(:side=>:bottom, :fill=>:x, :pady=>'2m')
|
||
|
||
TkButton.new(f, :text=>'閉じる', :width=>15, :command=>proc{
|
||
$entry3_demo.destroy
|
||
$entry3_demo = nil
|
||
}).pack(:side=>:left, :expand=>true)
|
||
|
||
TkButton.new(f, :text=>'コード参照', :width=>15, :command=>proc{
|
||
showCode 'entry3'
|
||
}).pack(:side=>:left, :expand=>true)
|
||
}
|
||
|
||
# focusAndFlash --
|
||
# Error handler for entry widgets that forces the focus onto the
|
||
# widget and makes the widget flash by exchanging the foreground and
|
||
# background colours at intervals of 200ms (i.e. at approximately
|
||
# 2.5Hz).
|
||
#
|
||
# Arguments:
|
||
# widget - entry widget to flash
|
||
# fg - Initial foreground colour
|
||
# bg - Initial background colour
|
||
# count - Counter to control the number of times flashed
|
||
def focusAndFlash(widget, fg, bg, count=5)
|
||
return if count <= 0
|
||
TkTimer.new(100, count,
|
||
proc{widget.configure(:foreground=>bg, :background=>fg)},
|
||
proc{widget.configure(:foreground=>fg, :background=>bg)}
|
||
).start
|
||
widget.focus(true)
|
||
end
|
||
|
||
l1 = TkLabelFrame.new($entry3_demo, :text=>"整数エントリ")
|
||
TkEntry.new(l1, :validate=>:focus,
|
||
:vcmd=>[
|
||
proc{|s| s == '' || /^[+-]?\d+$/ =~ s }, '%P'
|
||
]) {|e|
|
||
invalidcommand [proc{|w| focusAndFlash(w, e.fg, e.bg)}, '%W']
|
||
pack(:fill=>:x, :expand=>true, :padx=>'1m', :pady=>'1m')
|
||
}
|
||
|
||
l2 = TkLabelFrame.new($entry3_demo, :text=>"長さ制約付きエントリ")
|
||
TkEntry.new(l2, :validate=>:key, :invcmd=>proc{Tk.bell},
|
||
:vcmd=>[proc{|s| s.length < 10}, '%P']
|
||
).pack(:fill=>:x, :expand=>true, :padx=>'1m', :pady=>'1m')
|
||
|
||
### PHONE NUMBER ENTRY ###
|
||
# Note that the source to this is quite a bit longer as the behaviour
|
||
# demonstrated is a lot more ambitious than with the others.
|
||
|
||
# Initial content for the third entry widget
|
||
entry3content = TkVariable.new("1-(000)-000-0000")
|
||
|
||
# Mapping from alphabetic characters to numbers.
|
||
$phoneNumberMap = {}
|
||
Hash[*(%w(abc 2 def 3 ghi 4 jkl 5 mno 6 pqrs 7 tuv 8 wxyz 9))].each{|chars, n|
|
||
chars.split('').each{|c|
|
||
$phoneNumberMap[c] = n
|
||
$phoneNumberMap[c.upcase] = n
|
||
}
|
||
}
|
||
|
||
# phoneSkipLeft --
|
||
# Skip over fixed characters in a phone-number string when moving left.
|
||
#
|
||
# Arguments:
|
||
# widget - The entry widget containing the phone-number.
|
||
def phoneSkipLeft(widget)
|
||
idx = widget.index('insert')
|
||
if idx == 8
|
||
# Skip back two extra characters
|
||
widget.cursor = idx - 2
|
||
elsif idx == 7 || idx == 12
|
||
# Skip back one extra character
|
||
widget.cursor = idx - 1
|
||
elsif idx <= 3
|
||
# Can't move any further
|
||
Tk.bell
|
||
Tk.callback_break
|
||
end
|
||
end
|
||
|
||
# phoneSkipRight --
|
||
# Skip over fixed characters in a phone-number string when moving right.
|
||
#
|
||
# Arguments:
|
||
# widget - The entry widget containing the phone-number.
|
||
# add - Offset to add to index before calculation (used by validation.)
|
||
def phoneSkipRight(widget, add = 0)
|
||
idx = widget.index('insert')
|
||
if (idx + add == 5)
|
||
# Skip forward two extra characters
|
||
widget.cursor = idx + 2
|
||
elsif (idx + add == 6 || idx + add == 10)
|
||
# Skip forward one extra character
|
||
widget.cursor = idx + 1
|
||
elsif (idx + add == 15 && add == 0)
|
||
# Can't move any further
|
||
Tk.bell
|
||
Tk.callback_break
|
||
end
|
||
end
|
||
|
||
# validatePhoneChange --
|
||
# Checks that the replacement (mapped to a digit) of the given
|
||
# character in an entry widget at the given position will leave a
|
||
# valid phone number in the widget.
|
||
#
|
||
# widget - entry widget to validate
|
||
# vmode - The widget's validation mode
|
||
# idx - The index where replacement is to occur
|
||
# char - The character (or string, though that will always be
|
||
# refused) to be overwritten at that point.
|
||
|
||
def validatePhoneChange(widget, vmode, idx, char)
|
||
return true if idx == nil
|
||
Tk.after_idle(proc{widget.configure(:validate=>vmode,
|
||
:invcmd=>proc{Tk.bell})})
|
||
if !(idx<3 || idx==6 || idx==7 || idx==11 || idx>15) && char =~ /[0-9A-Za-z]/
|
||
widget.delete(idx)
|
||
widget.insert(idx, $phoneNumberMap[char] || char)
|
||
Tk.after_idle(proc{phoneSkipRight(widget, -1)})
|
||
# Tk.update(true) # Don't work 'update' inter validation callback.
|
||
# It depends on Tcl/Tk side (tested on Tcl/Tk8.5a1).
|
||
return true
|
||
end
|
||
return false
|
||
end
|
||
|
||
|
||
l3 = TkLabelFrame.new($entry3_demo, :text=>"米国電話番号エントリ")
|
||
TkEntry.new(l3, :validate=>:key, :invcmd=>proc{Tk.bell},
|
||
:textvariable=>entry3content,
|
||
:vcmd=>[
|
||
proc{|w,v,i,s| validatePhoneChange(w,v,i,s)},
|
||
"%W %v %i %S"
|
||
]){|e|
|
||
# Click to focus goes to the first editable character...
|
||
bind('FocusIn', proc{|d,w|
|
||
if d != "NotifyAncestor"
|
||
w.cursor = 3
|
||
Tk.after_idle(proc{w.selection_clear})
|
||
end
|
||
}, '%d %W')
|
||
bind('Left', proc{|w| phoneSkipLeft(w)}, '%W')
|
||
bind('Right', proc{|w| phoneSkipRight(w)}, '%W')
|
||
pack(:fill=>:x, :expand=>true, :padx=>'1m', :pady=>'1m')
|
||
}
|
||
|
||
l4 = TkLabelFrame.new($entry3_demo, :text=>"パスワードエントリ")
|
||
TkEntry.new(l4, :validate=>:key, :show=>'*',
|
||
:vcmd=>[
|
||
proc{|s| s.length <= 8},
|
||
'%P'
|
||
]).pack(:fill=>:x, :expand=>true, :padx=>'1m', :pady=>'1m')
|
||
|
||
TkFrame.new($entry3_demo){|f|
|
||
lower
|
||
TkGrid.configure(l1, l2, :in=>f, :padx=>'3m', :pady=>'1m', :sticky=>:ew)
|
||
TkGrid.configure(l3, l4, :in=>f, :padx=>'3m', :pady=>'1m', :sticky=>:ew)
|
||
TkGrid.columnconfigure(f, [0,1], :uniform=>1)
|
||
pack(:fill=>:both, :expand=>true)
|
||
}
|