rpl.rb/lib/rpl/words/operations-reals-complexes.rb
2022-08-31 10:10:42 +02:00

202 lines
9.6 KiB
Ruby
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# frozen_string_literal: true
module RplLang
module Words
module OperationsRealsAndComplexes
include Types
def populate_dictionary
super
category = 'Usual operations on reals and complexes'
# Usual operations on reals and complexes
@dictionary.add_word( ['+'],
category,
'( a b -- c ) addition',
proc do
addable = [RplNumeric, RplString, RplName, RplList]
args = stack_extract( [addable, addable] )
# | + | 1 numeric | 1 string | 1 name |v1 list |
# |-----------+-----------+----------+--------+--------|
# | 0 numeric | numeric | string | name |vlist |
# |v0 string |vstring |vstring |vstring |vlist |
# |v0 name |vstring |vstring |vname |vlist |
# |v0 list |vlist |vlist |vlist |vlist |
args.reverse!
result = if args[0].instance_of?( RplList )
new_list = if args[1].instance_of?( RplList )
RplList.new( args[0].to_s ).value.concat( args[1].value )
else
RplList.new( args[0].to_s ).value.concat( [args[1]] )
end
RplList.new( "{ #{new_list.join(' ')} }" )
elsif args[1].instance_of?( RplList )
new_list = if args[0].instance_of?( RplList )
RplList.new( args[0].to_s ).value.concat( args[1].value )
else
RplList.new( "{ #{args[0]} }" ).value.concat( args[1].value )
end
RplList.new( "{ #{new_list.join(' ')} }" )
elsif args[0].instance_of?( RplString )
RplString.new( if args[1].instance_of?( RplString ) ||
args[1].instance_of?( RplName )
"\"#{args[0].value}#{args[1].value}\""
else
"\"#{args[0].value}#{args[1]}\""
end )
elsif args[0].instance_of?( RplName )
if args[1].instance_of?( RplName )
RplName.new( "'#{args[0].value}#{args[1].value}'" )
elsif args[1].instance_of?( RplString )
Types.new_object( RplString, "\"#{args[0].value}#{args[1].value}\"" )
elsif args[1].instance_of?( RplNumeric )
RplName.new( "'#{args[0].value}#{args[1]}'" )
else
Types.new_object( RplString, "\"#{args[0]}#{args[1]}\"" )
end
elsif args[0].instance_of?( RplNumeric )
if args[1].instance_of?( RplNumeric )
RplNumeric.new( args[0].value + args[1].value, args[0].base )
elsif args[1].instance_of?( RplString )
RplString.new( "\"#{args[0]}#{args[1].value}\"" )
elsif args[1].instance_of?( RplName )
RplName.new( "'#{args[0]}#{args[1].value}'" )
end
end
@stack << result
end )
@dictionary.add_word( ['-'],
category,
'( a b -- c ) subtraction',
proc do
args = stack_extract( [[RplNumeric], [RplNumeric]] )
@stack << RplNumeric.new( args[1].value - args[0].value, args[1].base )
end )
@dictionary.add_word( ['chs'],
category,
'( a -- b ) negate',
proc do
run( '-1 *' )
end )
@dictionary.add_word( ['×', '*'],
category,
'( a b -- c ) multiplication',
proc do
args = stack_extract( [[RplNumeric], [RplNumeric]] )
@stack << RplNumeric.new( args[1].value * args[0].value, args[1].base )
end )
@dictionary.add_word( ['÷', '/'],
category,
'( a b -- c ) division',
proc do
args = stack_extract( [[RplNumeric], [RplNumeric]] )
@stack << RplNumeric.new( args[1].value / args[0].value, args[1].base )
end )
@dictionary.add_word( ['inv'],
category,
'( a -- b ) invert numeric',
proc do
run( '1.0 swap /' )
end )
@dictionary.add_word( ['^'],
category,
'( a b -- c ) a to the power of b',
proc do
args = stack_extract( [[RplNumeric], [RplNumeric]] )
@stack << RplNumeric.new( args[1].value**args[0].value, args[1].base )
end )
@dictionary.add_word( ['√', 'sqrt'],
category,
'( a -- b ) square root',
proc do
args = stack_extract( [[RplNumeric]] )
@stack << RplNumeric.new( BigMath.sqrt( args[0].value, RplNumeric.precision ), args[0].base )
end )
@dictionary.add_word( ['²', 'sq'],
category,
'( a -- b ) square',
proc do
run( 'dup ×')
end )
@dictionary.add_word( ['abs'],
category,
'( a -- b ) absolute value',
proc do
args = stack_extract( [[RplNumeric]] )
@stack << RplNumeric.new( args[0].value.abs, args[0].base )
end )
@dictionary.add_word( ['dec'],
category,
'( a -- a ) set numeric\'s base to 10',
proc do
run( '10 base' )
end )
@dictionary.add_word( ['hex'],
category,
'( a -- a ) set numeric\'s base to 16',
proc do
run( '16 base' )
end )
@dictionary.add_word( ['bin'],
category,
'( a -- a ) set numeric\'s base to 2',
proc do
run( '2 base' )
end )
@dictionary.add_word( ['base'],
category,
'( a b -- a ) set numeric\'s base to b',
proc do
args = stack_extract( [[RplNumeric], [RplNumeric]] )
@stack << RplNumeric.new( args[1].value, args[0].value )
end )
@dictionary.add_word( ['sign'],
category,
'( a -- b ) sign of element',
proc do
args = stack_extract( [[RplNumeric]] )
@stack << RplNumeric.new( if args[0].value.positive?
1
elsif args[0].value.negative?
-1
else
0
end )
end )
end
end
end
end