+my $menumin;
+my %set = (
+ base => 10,
+ numb => 0, # fixed scientific engineering
+ card => 1, # degrees radians grades
+ coord => 0, # cartesian polar spherical
+ complex => 0, # real complex
+ menushow => 12,
+); # %set
+
+@menu = @{$menus[0]};
+$menumin = 0;
+
+my %falias = (
+ "\033" => 0, # esc
+ "\033\117\120" => 1, # f1
+ "\033\133\061\061\176" => 1, # f1
+ "\033\133\061\062\176" => 2, # f2
+ "\033\133\061\063\176" => 3, # f3
+ "\033\133\061\064\176" => 4, # f4
+ "\033\117\121" => 2, # f2
+ "\033\117\122" => 3, # f3
+ "\033\117\123" => 4, # f4
+ "\033\133\061\065\176" => 5, # f5
+ "\033\133\061\067\176" => 6, # f6
+ "\033\133\061\070\176" => 7, # f7
+ "\033\133\061\071\176" => 8, # f8
+ "\033\133\062\060\176" => 9, # f9
+ "\033\133\062\061\176" => 10, # f10
+ "\033\133\062\063\176" => 11, # f11/F1
+ "\033\133\062\064\176" => 12, # f12/F2
+ "\033\133\062\065\176" => 13, # F3
+ "\033\133\062\066\176" => 14, # F4
+ "\033\133\062\070\176" => 15, # F5
+ "\033\133\062\071\176" => 16, # F6
+ "\033\133\063\061\176" => 17, # F7
+ "\033\133\063\062\176" => 18, # F8
+ "\033\133\063\063\176" => 19, # F9
+ "\033\133\063\064\176" => 20, # F10
+ "\033\133\062\063\073\062\176" => 21, # F11
+ "\033\133\062\064\073\062\176" => 22, # F12
+); # %falias
+
+my %alias = (
+ chr 4 => 'quit', # ^D
+ chr 9 => 'more', # tab
+ '_' => 'chs', # change sign; 48: y
+ 'e' => 'eex', # exponent; 48: z
+ "\033\133\062\176" => 'eex', # ins
+ "\033\133\063\176" => "clear", # del
+ chr 127 => 'drop', # backspace
+ chr 8 => 'drop', # backspace
+ chr 13 => 'enter', # enter
+ ' ' => 'enter', # space
+ "\014" => 'refresh', # ^L
+# "\033\133\110" => 'refresh', # home
+
+# "\033\133\101" => '', # up; 48: k (stack)
+ "\033\133\104" => 'undo', # left; 48: p (picture)
+# "\033\133\102" => '', # down; 48: q (view)
+ "\033\133\103" => 'swap', # right; 48: r (swap)
+
+ '&' => 'and',
+ '|' => 'or',
+ '#' => 'xor',
+ '~' => 'not',
+ '(' => 'sl',
+ ')' => 'sr',
+
+ "s" => "sin",
+ "\033s" => "asin",
+ "c" => "cos",
+ "\033c" => "acos",
+ "t" => "tan",
+ "\033t" => "atan",
+ "l" => "log",
+ "\033l" => "alog",
+ "n" => "ln",
+ "\033n" => "exp",
+ "q" => "sq",
+ "\033q" => "sqrt",
+ "x" => "^",
+ "\033x" => "xroot",
+ "\033^" => "xroot",
+ "v" => "inv",
+); # %alias
+
+=cut
+HP48 keys:
+ S T U V W X
+ - sin cos tan sqrt ^ 1/x
+ < asin acos atan sq alog exp
+ > [a] ∫ ∑ xroot log ln
+=cut
+
+my %action = (
+ 'more' => [-1, sub {
+ $menumin += $set{menushow};
+ $menumin = 0 if $menumin>=$#menu;
+ showmenu();
+ }], # tab
+ 'chs' => [0, sub {$stack[0] = -$stack[0]}], # negative
+
+ 'drop' => [0, sub {shift @stack}], # backspace
+ 'clear'=> [0, sub {@stack = (); %val = (i=>undef, frac=>0) }], # clear all #todo: if (val{i}) delete char after cursor
+
+ 'enter'=> [0, sub {
+ unshift @stack, defined $val{i} ? $val{i} : $stack[0];
+ %val = (i=>undef, frac=>0);
+ }], # duplication
+
+ 'swap' => [1, sub {@stack[0, 1] = @stack[1, 0]}], # swap x<->y
+
+ '=' => [1, sub {$var{a} = $stack[0]}], # copy
+ '?' => [1, sub {$var{a} = shift @stack}], # assign
+
+ '+' => [2, sub {$stack[1] += shift @stack}], # addition
+ '-' => [2, sub {$stack[1] -= shift @stack}], # substraction
+ '*' => [2, sub {$stack[1] *= shift @stack}], # multiplication
+ '/' => [2, sub {$stack[1] /= shift @stack}], # division
+ 'mod' => [2, sub {$stack[1] %= shift @stack}], # modulo
+
+ 'inv' => [1, sub {$stack[0] = 1 / $stack[0]}], # 1/x
+ 'sqrt' => [1, sub {$stack[0] = sqrt $stack[0]}], # square root
+ 'sq' => [1, sub {$stack[0] *= $stack[0]}], # squared
+ '^' => [2, sub {$stack[1] **= shift @stack}], # exponentiation
+ 'xroot'=> [2, sub {$stack[1] **= 1 / shift @stack}], # x-root of y
+
+ 'log' => [1, sub {$stack[0] = log($stack[0]) / log(10)}], # logarithm
+ 'alog' => [1, sub {$stack[0] = 10 ** $stack[0]}], # 10^x
+ 'ln' => [1, sub {$stack[0] = log $stack[0]}], # natural logaritm
+ 'lnp1' => [1, sub {$stack[0] = log($stack[0]+1)}], # ln(x+1)
+ 'exp' => [1, sub {$stack[0] = exp($stack[0])}], # e^x
+ 'expm' => [1, sub {$stack[0] = exp($stack[0])-1}], # exp(x)-1
+
+ 'sin' => [1, sub {$stack[0] = sin $stack[0]}], # sine
+ 'asin' => [1, sub {$stack[0] = atan2($stack[0], sqrt(1 - $stack[0]*$stack[0]))}], # inverse sine
+ 'cos' => [1, sub {$stack[0] = cos $stack[0]}], # cosine
+ 'acos' => [1, sub {$stack[0] = atan2(sqrt(1 - $stack[0]*$stack[0]), $stack[0])}], # inverse cosine
+ 'tan' => [1, sub {$stack[0] = sin($stack[0]) / cos($stack[0])}], # tangent
+# 'atan' => [1, sub {}], # arctangent
+
+ 'sinh' => [1, sub {$stack[0] = ( exp($stack[0]) - exp(-$stack[0]) )/2}], # hyperbolic sine
+ 'cosh' => [1, sub {$stack[0] = ( exp($stack[0]) + exp(-$stack[0]) )/2}], # hyperbolic cosine
+ 'tanh' => [1, sub {$stack[0] = ( exp($stack[0]) - exp(-$stack[0]) )/( exp($stack[0]) + exp(-$stack[0]) )}], # hyperbolic tangent (sinh/cosh)
+ 'asinh'=> [1, sub {$stack[0] = log( sqrt($stack[0]**2+1)+$stack[0] )}], # inverse hyperbolic sine
+ 'acosh'=> [1, sub {$stack[0] = log( sqrt($stack[0]**2-1)+$stack[0] )}], # inverse hyperbolic cosine
+ 'atanh'=> [1, sub {$stack[0] = log( (1+$stack[0]) / (1-$stack[0]) )/2}], # inverse hyperbolic tangent
+
+ '%' => [2, sub {$stack[0] /= shift @stack}], # percentage
+ '%ch' => [2, sub {$val{i} = 100*(shift(@stack)-$val{i})/$val{i}}], # percentage change
+ '%t' => [2, sub {$val{i} = 100*$val{i}/shift(@stack)}], # percentage total
+
+ 'and' => [2, sub {$stack[1] &= shift @stack}], # bitwise and
+ 'or' => [2, sub {$stack[1] |= shift @stack}], # bitwise or
+ 'xor' => [2, sub {$stack[1] ^= shift @stack}], # bitwise xor
+ 'not' => [2, sub {$stack[0] = ~$stack[0]}], # bitwise not
+ 'sl' => [1, sub {$stack[0] *= 2}], # shift left
+ 'sr' => [1, sub {$stack[0] /= 2}], # shift right
+
+ 'abs' => [1, sub {$stack[0] = abs $stack[0]}], # absolute #todo
+ 'sign' => [1, sub {$stack[0] = $stack[0] <=> 0}], # sign
+ 'ip' => [1, sub {$stack[0] = int $stack[0]}], # integer part
+ 'fp' => [1, sub {$stack[0] -= int $stack[0]}], # fractional part
+
+ 'rnd' => [1, sub {local $_ = 10**shift @stack; $val{i} = int(($val{i}+.5)*$_)/$_}], # round
+ 'trnc' => [1, sub {local $_ = 10**shift @stack; $val{i} = int($val{i}*$_)/$_}], # truncate
+ 'floor'=> [1, sub {$stack[0] = int $stack[0]}], # floor
+ 'ceil' => [1, sub {$stack[0] = int $stack[0]+.9999}], # ceil
+
+ 'min' => [2, sub {local $_ = shift @stack; $stack[0] = $_ if $_<$stack[0] }], # minimum
+ 'max' => [2, sub {local $_ = shift @stack; $stack[0] = $_ if $_>$stack[0] }], # maximum
+
+ 'dec' => [0, sub {$set{base} = 10}], # decimal
+ 'bin' => [0, sub {$set{base} = 2}], # binary
+ 'oct' => [0, sub {$set{base} = 8}], # octal
+ 'hex' => [0, sub {$set{base} = 16}], # hexadecimal
+ 'base' => [1, sub {$set{base} = shift @stack}], # alphanumerical
+
+ '!' => [1, sub {local $_ = $stack[0]; $stack[0] *= $_ while --$_>1}], # factor
+ 'rand' => [0, sub {unshift @stack, rand}], # random value <1
+
+ 'undo' => [0, sub {@stack = @{ $var{undo} }}], # undo
+); # %action