The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
# -*- perl -*-
#
# tkfbox.tcl --
#
#       Implements the "TK" standard file selection dialog box. This
#       dialog box is used on the Unix platforms whenever the tk_strictMotif
#       flag is not set.
#
#       The "TK" standard file selection dialog box is similar to the
#       file selection dialog box on Win95(TM). The user can navigate
#       the directories by clicking on the folder icons or by
#       selectinf the "Directory" option menu. The user can select
#       files by clicking on the file icons or by entering a filename
#       in the "Filename:" entry.
#
# Copyright (c) 1994-1996 Sun Microsystems, Inc.
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
#
# Translated to perk/Tk and modified by Slaven Rezic <slaven@rezic.de>.
#

#----------------------------------------------------------------------
#
#                     I C O N   L I S T
#
# This is a pseudo-widget that implements the icon list inside the
# tkFDialog dialog box.
#
#----------------------------------------------------------------------
# tkIconList --
#
#       Creates an IconList widget.
#

package Tk::IconList;
require Tk::Frame;

use vars qw($VERSION);
$VERSION = '4.007'; # $Id: //depot/Tkutf8/Tk/IconList.pm#7 $

use Tk qw(Ev);
use strict;
use Carp;

use base 'Tk::Frame';

Construct Tk::Widget 'IconList';

# tkIconList_Create --
#
#       Creates an IconList widget by assembling a canvas widget and a
#       scrollbar widget. Sets all the bindings necessary for the IconList's
#       operations.
#
sub Populate {
    my($w, $args) = @_;
    $w->SUPER::Populate($args);

    my $sbar = $w->Component('Scrollbar' => 'sbar',
			     -orient => 'horizontal',
			     -highlightthickness => 0,
			     -takefocus => 0,
			    );
    # make sure that the size does not exceed handhelds' dimensions
    my($sw,$sh) = ($w->screenwidth, $w->screenheight);
    my $canvas = $w->Component('Canvas' => 'canvas',
			       -bd => 2,
			       -relief => 'sunken',
			       -width  => ($sw > 420 ? 400 : $sw-20),
			       -height => ($sh > 160 ? 120 : $sh-40),
			       -takefocus => 1,
			      );
    $sbar->pack(-side => 'bottom', -fill => 'x', -padx => 2);
    $canvas->pack(-expand => 'yes', -fill => 'both');
    $sbar->configure(-command => ['xview', $canvas]);
    $canvas->configure(-xscrollcommand => ['set', $sbar]);

    # Initializes the max icon/text width and height and other variables
    $w->{'maxIW'} = 1;
    $w->{'maxIH'} = 1;
    $w->{'maxTW'} = 1;
    $w->{'maxTH'} = 1;
    $w->{'numItems'} = 0;
#XXX curItem never used    delete $w->{'curItem'};
    $w->{'noScroll'} = 1;
    $w->{'selection'} = [];
    $w->{'index,anchor'} = '';

    # Creates the event bindings.
    $canvas->Tk::bind('<Configure>', sub { $w->Arrange } );
    $canvas->Tk::bind('<1>', [$w,'Btn1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<B1-Motion>', [$w,'Motion1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<Control-B1-Motion>', 'NoOp');
    $canvas->Tk::bind('<Shift-B1-Motion>', 'NoOp');
    $canvas->Tk::bind('<Control-1>', [$w,'CtrlBtn1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<Shift-1>', [$w,'ShiftBtn1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<Double-ButtonRelease-1>', [$w,'Double1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<Control-Double-ButtonRelease-1>', 'NoOp');
    $canvas->Tk::bind('<Shift-Double-ButtonRelease-1>', 'NoOp');
    $canvas->Tk::bind('<ButtonRelease-1>', [$w,'CancelRepeat']);
    $canvas->Tk::bind('<B1-Leave>', [$w,'Leave1',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<B1-Enter>', [$w,'CancelRepeat']);
    $canvas->Tk::bind('<Up>',       [$w,'UpDown',   -1]);
    $canvas->Tk::bind('<Down>',     [$w,'UpDown',    1]);
    $canvas->Tk::bind('<Left>',     [$w,'LeftRight',-1]);
    $canvas->Tk::bind('<Right>',    [$w,'LeftRight', 1]);
    $canvas->Tk::bind('<Return>',   [$w,'ReturnKey']);
    $canvas->Tk::bind('<KeyPress>', [$w,'KeyPress',Ev('A')]);
    $canvas->Tk::bind('<Control-KeyPress>', 'NoOp');
    $canvas->Tk::bind('<Alt-KeyPress>', 'NoOp');
    $canvas->Tk::bind('<Meta-KeyPress>', 'NoOp');
#XXX bad....
#    $canvas->Tk::bind('<FocusIn>', sub { $w->FocusIn });
#    $canvas->Tk::bind('<FocusOut>', sub { $w->FocusOut });

    # additional bindings not in tkfbox.tcl
    $canvas->Tk::bind('<2>',['scan','mark',Ev('x'),Ev('y')]);
    $canvas->Tk::bind('<B2-Motion>',['scan','dragto',Ev('x'),Ev('y')]);
    # Remove the standard Canvas bindings
    $canvas->bindtags([$canvas, $canvas->toplevel, 'all']);
    # ... and define some again
    $canvas->Tk::bind('<Home>', ['xview','moveto',0]);
    $canvas->Tk::bind('<End>',  ['xview','moveto',1]);

    $w->ConfigSpecs(-browsecmd =>
		    ['METHOD', 'browseCommand', 'BrowseCommand', undef],
		    -command =>
		    ['CALLBACK', 'command', 'Command', undef],
		    -font =>
		    ['PASSIVE', 'font', 'Font', undef],
		    -foreground =>
		    ['PASSIVE', 'foreground', 'Foreground', undef],
		    -fg => '-foreground',
		    -multiple =>
		    ['PASSIVE', 'multiple', 'Multiple', 0],
		    -selectmode =>
		    ['PASSIVE', 'selectMode', 'SelectMode', 'browse'],
		    -selectbackground =>
		    ['PASSIVE', 'selectBackground', 'Foreground', '#a0a0ff'],
		   );

    $w;
}

# compatibility for old -browsecmd options
sub browsecmd {
    my $w = shift;
    if (@_) {
	$w->{Configure}{'-browsecmd'} = $_[0];
	$w->bind('<<ListboxSelect>>' => $_[0]);
    }
    $w->{Configure}{'-browsecmd'};
}

sub Index {
    my($w, $i) = @_;
    if (!$w->{'list'}) { $w->{'list'} = [] }
    if ($i =~ /^-?[0-9]+$/) {
	if ($i < 0) {
	    $i = 0;
	}
	if ($i > @{ $w->{'list'} }) {
	    $i = @{ $w->{'list'} } - 1;
	}
	return $i;
    } elsif ($i eq 'active') {
	return $w->{'index,active'};
    } elsif ($i eq 'anchor') {
	return $w->{'index,anchor'};
    } elsif ($i eq 'end') {
	return @{ $w->{'list'} };
    } elsif ($i =~ /@(-?[0-9]+),(-?[0-9]+)/) {
	my($x, $y) = ($1, $2);
	my $canvas = $w->Subwidget('canvas');
	my $item = $canvas->find('closest', $x, $y);
	if (defined $item) {
	    return $canvas->itemcget($item, '-tags')->[1];
	} else {
	    return "";
	}
    } else {
	croak "Unrecognized Index parameter `$i', use active, anchor, end, \@x,y, or x";
    }
}

sub Selection {
    my($w, $op, @args) = @_;
    if ($op eq 'anchor') {
	if (@args == 1) {
	    $w->{'index,anchor'} = $w->Index($args[0]);
	} else {
	    return $w->{'index,anchor'};
	}
    } elsif ($op eq 'clear') {
	my($first, $last);
	if (@args == 2) {
	    ($first, $last) = @args;
	} elsif (@args == 1) {
	    $first = $last = $args[0];
	} else {
	    croak "wrong # args: should be Selection('clear', first, ?last?)"
	}
	$first = $w->Index($first);
	$last  = $w->Index($last);
	if ($first > $last) {
	    ($first, $last) = ($last, $first);
	}
	my $ind = 0;
	for my $item (@{ $w->{'selection'} }) {
	    if ($item >= $first) {
		$first = $ind;
		last;
	    }
	    $ind++; # XXX seems to be missing in the Tcl version
	}
	$ind = @{ $w->{'selection'} } - 1;
	for(; $ind >= 0; $ind--) {
	    my $item = $w->{'selection'}->[$ind];
	    if ($item <= $last) {
		$last = $ind;
		last;
	    }
	}
	if ($first > $last) {
	    return;
	}
	splice @{ $w->{'selection'} }, $first, $last-$first+1;
	$w->event('generate', '<<ListboxSelect>>');
	$w->DrawSelection;
    } elsif ($op eq 'includes') {
	my $index;
	for (@{ $w->{'selection'} }) {
	    if ($args[0] eq $_) {
		return 1;
	    }
	}
	return 0;
    } elsif ($op eq 'set') {
	my($first, $last);
	if (@args == 2) {
	    ($first, $last) = @args;
	} elsif (@args == 1) {
	    $first = $last = $args[0];
	} else {
	    croak "wrong # args: should be Selection('set', first, ?last?)";
	}

	$first = $w->Index($first);
	$last  = $w->Index($last);
	if ($first > $last) {
	    ($first, $last) = ($last, $first);
	}
	for(my $i = $first; $i <= $last; $i++) {
	    push @{ $w->{'selection'} }, $i;
	}
	# lsort -integer -unique
	my %sel = map { ($_ => 1) } @{ $w->{'selection'} };
	@{ $w->{'selection'} } = sort { $a <=> $b } keys %sel;
	$w->event('generate', '<<ListboxSelect>>');
	$w->DrawSelection;
    } else {
	croak "Unrecognized Selection parameter `$op', use anchor, clear, includes, or set";
    }
}

# XXX why lower case 's' here and upper in DrawSelection?
sub Curselection {
    my $w = shift;
    @{ $w->{'selection'} };
}

sub DrawSelection {
    my $w = shift;
    my $canvas = $w->Subwidget('canvas');
    $canvas->delete('selection');
    my $selBg = $w->cget('-selectbackground');
    for my $item (@{ $w->{'selection'} }) {
	my $rTag = $w->{'list'}->[$item][2];
	my($iTag, $tTag, $text, $serial) = @{ $w->{'itemList'}{$rTag} };
	my @bbox = $canvas->bbox($tTag);
	# XXX don't hardcode colors
	$canvas->createRectangle
	    (@bbox, -fill => $selBg, -outline => $selBg, -tags => 'selection');
    }
    $canvas->lower('selection');
}

# Returns the selected item
#
sub Get {
    my($w, $item) = @_;
    my $rTag = $w->{'list'}->[$item][2];
    my($iTag, $tTag, $text, $serial) = @{ $w->{'itemList'}{$rTag} };
    $text;
}


# tkIconList_AutoScan --
#
# This procedure is invoked when the mouse leaves an entry window
# with button 1 down.  It scrolls the window up, down, left, or
# right, depending on where the mouse left the window, and reschedules
# itself as an "after" command so that the window continues to scroll until
# the mouse moves back into the window or the mouse button is released.
#
# Arguments:
# w -           The IconList window.
#
sub AutoScan {
    my $w = shift;
    return unless ($w->exists);
    return if ($w->{'noScroll'});
    my($x, $y);
    $x = $Tk::x;
    $y = $Tk::y;
    my $canvas = $w->Subwidget('canvas');
    if ($x >= $canvas->width) {
	$canvas->xview('scroll', 1, 'units');
    } elsif ($x < 0) {
	$canvas->xview('scroll', -1, 'units');
    } elsif ($y >= $canvas->height) {
	# do nothing
    } elsif ($y < 0) {
	# do nothing
    } else {
	return;
    }
    $w->Motion1($x, $y);
    $w->RepeatId($w->after(50, ['AutoScan', $w]));
}

# Deletes all the items inside the canvas subwidget and reset the IconList's
# state.
#
sub DeleteAll {
    my $w = shift;
    my $canvas = $w->Subwidget('canvas');
    $canvas->delete('all');
    delete $w->{'selected'};
    delete $w->{'rect'};
    delete $w->{'list'};
    delete $w->{'itemList'};
    $w->{'maxIW'} = 1;
    $w->{'maxIH'} = 1;
    $w->{'maxTW'} = 1;
    $w->{'maxTH'} = 1;
    $w->{'numItems'} = 0;
#XXX curItem never used    delete $w->{'curItem'};
    $w->{'noScroll'} = 1;
    $w->{'selection'} = [];
    $w->{'index,anchor'} = '';
    $w->Subwidget('sbar')->set(0.0, 1.0);
    $canvas->xview('moveto', 0);
}

# Adds an icon into the IconList with the designated image and items
#
sub Add {
    my($w, $image, @items) = @_;
    my $canvas = $w->Subwidget('canvas');
    my $font = $w->cget(-font);
    my $fg   = $w->cget(-foreground);
    foreach my $text (@items) {
	my $iTag = $canvas->createImage
	    (0, 0, -image => $image, -anchor => 'nw',
	     -tags => ['icon', $w->{numItems}, 'item'.$w->{numItems}],
	    );
	my $tTag = $canvas->createText
	    (0, 0, -text => $text, -anchor => 'nw',
	     (defined $fg   ? (-fill => $fg)   : ()),
	     (defined $font ? (-font => $font) : ()),
	     -tags => ['text', $w->{numItems}, 'item'.$w->{numItems}],
	    );
	my $rTag = $canvas->createRectangle
	    (0, 0, 0, 0,
	     -fill => undef,
	     -outline => undef,
	     -tags => ['rect', $w->{numItems}, 'item'.$w->{numItems}],
	    );
	my(@b) = $canvas->bbox($iTag);
	my $iW = $b[2] - $b[0];
	my $iH = $b[3] - $b[1];
	$w->{'maxIW'} = $iW if ($w->{'maxIW'} < $iW);
	$w->{'maxIH'} = $iH if ($w->{'maxIH'} < $iH);
	@b = $canvas->bbox($tTag);
	my $tW = $b[2] - $b[0];
	my $tH = $b[3] - $b[1];
	$w->{'maxTW'} = $tW if ($w->{'maxTW'} < $tW);
	$w->{'maxTH'} = $tH if ($w->{'maxTH'} < $tH);
	push @{ $w->{'list'} }, [$iTag, $tTag, $rTag, $iW, $iH, $tW, $tH,
				 $w->{'numItems'}];
	$w->{'itemList'}{$rTag} = [$iTag, $tTag, $text, $w->{'numItems'}];
	$w->{'textList'}{$w->{'numItems'}} = lc($text);
	++$w->{'numItems'};
    }
}

# Places the icons in a column-major arrangement.
#
sub Arrange {
    my $w = shift;
    my $canvas = $w->Subwidget('canvas');
    my $sbar   = $w->Subwidget('sbar');
    unless (exists $w->{'list'}) {
	if (defined $canvas && Tk::Exists($canvas)) {
	    $w->{'noScroll'} = 1;
	    $sbar->configure(-command => sub { });
	}
	return;
    }

    my $W = $canvas->width;
    my $H = $canvas->height;
    my $pad = $canvas->cget(-highlightthickness) + $canvas->cget(-bd);
    $pad = 2 if ($pad < 2);
    $W -= $pad*2;
    $H -= $pad*2;
    my $dx = $w->{'maxIW'} + $w->{'maxTW'} + 8;
    my $dy;
    if ($w->{'maxTH'} > $w->{'maxIH'}) {
	$dy = $w->{'maxTH'};
    } else {
	$dy = $w->{'maxIH'};
    }
    $dy += 2;
    my $shift = $w->{'maxIW'} + 4;
    my $x = $pad * 2;
    my $y = $pad;
    my $usedColumn = 0;
    foreach my $sublist (@{ $w->{'list'} }) {
	$usedColumn = 1;
	my($iTag, $tTag, $rTag, $iW, $iH, $tW, $tH) = @$sublist;
	my $i_dy = ($dy - $iH) / 2;
	my $t_dy = ($dy - $tH) / 2;
	$canvas->coords($iTag, $x, $y + $i_dy);
	$canvas->coords($tTag, $x + $shift, $y + $t_dy);
	$canvas->coords($rTag, $x, $y, $x + $dx, $y + $dy);
	$y += $dy;
	if ($y + $dy > $H) {
	    $y = $pad;
	    $x += $dx;
	    $usedColumn = 0;
	}
    }
    my $sW;
    if ($usedColumn) {
	$sW = $x + $dx;
    } else {
	$sW = $x;
    }
    if ($sW < $W) {
	$canvas->configure(-scrollregion => [$pad, $pad, $sW, $H]);
	$sbar->configure(-command => sub { });
	$canvas->xview(moveto => 0);
	$w->{'noScroll'} = 1;
    } else {
	$canvas->configure(-scrollregion => [$pad, $pad, $sW, $H]);
	$sbar->configure(-command => ['xview', $canvas]);
	$w->{'noScroll'} = 0;
    }
    $w->{'itemsPerColumn'} = int(($H - $pad) / $dy);
    $w->{'itemsPerColumn'} = 1 if ($w->{'itemsPerColumn'} < 1);
#XXX    $w->Select($w->{'list'}[$w->{'curItem'}][2], 0)
#      if (exists $w->{'curItem'});
    $w->DrawSelection; # missing in Tcl XXX
}

# Gets called when the user invokes the IconList (usually by double-clicking
# or pressing the Return key).
#
sub Invoke {
    my $w = shift;
    $w->Callback(-command => $w->{'selected'}) if (@{ $w->{'selection'} });
}

# tkIconList_See --
#
#       If the item is not (completely) visible, scroll the canvas so that
#       it becomes visible.
sub See {
    my($w, $rTag) = @_;
    return if ($w->{'noScroll'});
    return if ($rTag < 0 || $rTag >= @{ $w->{'list'} });
    my $canvas = $w->Subwidget('canvas');
    my(@sRegion) = @{ $canvas->cget('-scrollregion') };
    return unless (@sRegion);
    my(@bbox) = $canvas->bbox('item'.$rTag);
    my $pad = $canvas->cget(-highlightthickness) + $canvas->cget(-bd);
    my $x1 = $bbox[0];
    my $x2 = $bbox[2];
    $x1 -= $pad * 2;
    $x2 -= $pad;
    my $cW = $canvas->width - $pad * 2;
    my $scrollW = $sRegion[2] - $sRegion[0] + 1;
    my $dispX = int(($canvas->xview)[0] * $scrollW);
    my $oldDispX = $dispX;
    # check if out of the right edge
    $dispX = $x2 - $cW if ($x2 - $dispX >= $cW);
    # check if out of the left edge
    $dispX = $x1 if ($x1 - $dispX < 0);
    if ($oldDispX != $dispX) {
	my $fraction = $dispX / $scrollW;
	$canvas->xview('moveto', $fraction);
    }
}

sub Btn1 {
    my($w, $x, $y) = @_;

    my $canvas = $w->Subwidget('canvas');
    $canvas->CanvasFocus;
    $x = int($canvas->canvasx($x));
    $y = int($canvas->canvasy($y));
    my $i = $w->Index('@'.$x.','.$y);
    return if ($i eq '');
    $w->Selection('clear', 0, 'end');
    $w->Selection('set', $i);
    $w->Selection('anchor', $i);
}

sub CtrlBtn1 {
    my($w, $x, $y) = @_;

    if ($w->cget(-multiple)) {
	my $canvas = $w->Subwidget('canvas');
	$canvas->CanvasFocus;
	my $x = int($canvas->canvasx($x));
	my $y = int($canvas->canvasy($y));
	my $i = $w->Index('@'.$x.','.$y);
	return if ($i eq '');
	if ($w->Selection('includes', $i)) {
	    $w->Selection('clear', $i);
	} else {
	    $w->Selection('set', $i);
	    $w->Selection('anchor', $i);
	}
    }
}

sub ShiftBtn1 {
    my($w, $x, $y) = @_;

    if ($w->cget(-multiple)) {
    my $canvas = $w->Subwidget('canvas');
	$canvas->CanvasFocus;
	my $x = int($canvas->canvasx($x));
	my $y = int($canvas->canvasy($y));
	my $i = $w->Index('@'.$x.','.$y);
	return if ($i eq '');
	my $a = $w->Index('anchor');
	if ($a eq '') {
	    $a = $i;
	}
	$w->Selection('clear', 0, 'end');
	$w->Selection('set', $a, $i);
    }
}

# Gets called on button-1 motions
#
sub Motion1 {
    my($w, $x, $y) = @_;
    $Tk::x = $x;
    $Tk::y = $y;
    my $canvas = $w->Subwidget('canvas');
    $canvas->CanvasFocus;
    $x = int($canvas->canvasx($x));
    $y = int($canvas->canvasy($y));
    my $i = $w->Index('@'.$x.','.$y);
    return if ($i eq '');
    $w->Selection('clear', 0, 'end');
    $w->Selection('set', $i);
}

sub Double1 {
    my($w, $x, $y) = @_;
    $w->Invoke if (@{ $w->{'selection'} });
}

sub ReturnKey {
    my $w = shift;
    $w->Invoke;
}

sub Leave1 {
    my($w, $x, $y) = @_;
    $Tk::x = $x;
    $Tk::y = $y;
    $w->AutoScan;
}

sub FocusIn {
    my $w = shift;
    return unless (exists $w->{'list'});
    if (@{ $w->{'selection'} }) {
	$w->DrawSelection;
    }
}

sub FocusOut {
    my $w = shift;
    $w->Selection('clear', 0, 'end');
}

# tkIconList_UpDown --
#
# Moves the active element up or down by one element
#
# Arguments:
# w -           The IconList widget.
# amount -      +1 to move down one item, -1 to move back one item.
#
sub UpDown {
    my($w, $amount) = @_;
    return unless (exists $w->{'list'});
    my $i;
    my(@curr) = $w->Curselection;
    if (!@curr) {
	$i = 0;
    } else {
	$i = $w->Index('anchor');
	return if ($i eq '');
	$i += $amount;
    }
    $w->Selection('clear', 0, 'end');
    $w->Selection('set', $i);
    $w->Selection('anchor', $i);
    $w->See($i);
}

# tkIconList_LeftRight --
#
# Moves the active element left or right by one column
#
# Arguments:
# w -           The IconList widget.
# amount -      +1 to move right one column, -1 to move left one column.
#
sub LeftRight {
    my($w, $amount) = @_;
    return unless (exists $w->{'list'});
    my $i;
    my(@curr) = $w->Curselection;
    if (!@curr) {
	$i = 0;
    } else {
	$i = $w->Index('anchor');
	return if ($i eq '');
	$i += $amount*$w->{'itemsPerColumn'};
    }
    $w->Selection('clear', 0, 'end');
    $w->Selection('set', $i);
    $w->Selection('anchor', $i);
    $w->See($i);
}

#----------------------------------------------------------------------
#               Accelerator key bindings
#----------------------------------------------------------------------
# tkIconList_KeyPress --
#
#       Gets called when user enters an arbitrary key in the listbox.
#
sub KeyPress {
    my($w, $key) = @_;
    $w->{'_ILAccel'} .= $key;
    $w->Goto($w->{'_ILAccel'});
    eval {
	$w->afterCancel($w->{'_ILAccel_afterid'});
    };
    $w->{'_ILAccel_afterid'} = $w->after(500, ['Reset', $w]);
}

sub Goto {
    my($w, $text) = @_;
    return unless (exists $w->{'list'});
    return if (not defined $text or $text eq '');
#XXX curItem never used    my $start = (!exists $w->{'curItem'} ? 0 : $w->{'curItem'});
    my $start = 0;
    $text = lc($text);
    my $theIndex = -1;
    my $less = 0;
    my $len = length($text);
    my $i = $start;
    # Search forward until we find a filename whose prefix is an exact match
    # with $text
    while (1) {
	my $sub = substr($w->{'textList'}{$i}, 0, $len);
	if ($text eq $sub) {
	    $theIndex = $i;
	    last;
	}
	++$i;
	$i = 0 if ($i == $w->{'numItems'});
	last if ($i == $start);
    }
    if ($theIndex > -1) {
	$w->Selection(qw(clear 0 end));
	$w->Selection('set', $theIndex);
	$w->Selection('anchor', $theIndex);
	$w->See($theIndex);
    }
}

sub Reset {
    my $w = shift;
    undef $w->{'_ILAccel'};
}

1;