factor/basis/roman/roman.factor

78 lines
1.7 KiB
Factor
Raw Normal View History

2007-09-20 18:09:08 -04:00
! Copyright (C) 2007 Doug Coleman.
! See http://factorcode.org/license.txt for BSD license.
2008-09-10 23:11:40 -04:00
USING: arrays assocs kernel math math.order math.vectors
namespaces make quotations sequences splitting.monotonic
sequences.private strings unicode.case lexer parser
grouping ;
2007-09-20 18:09:08 -04:00
IN: roman
<PRIVATE
2008-10-21 21:30:26 -04:00
2007-09-20 18:09:08 -04:00
: roman-digits ( -- seq )
{ "m" "cm" "d" "cd" "c" "xc" "l" "xl" "x" "ix" "v" "iv" "i" } ;
: roman-values ( -- seq )
{ 1000 900 500 400 100 90 50 40 10 9 5 4 1 } ;
2008-09-09 15:57:23 -04:00
ERROR: roman-range-error n ;
2007-09-20 18:09:08 -04:00
: roman-range-check ( n -- )
2008-09-09 15:57:23 -04:00
dup 1 3999 between? [ drop ] [ roman-range-error ] if ;
2007-09-20 18:09:08 -04:00
: roman<= ( ch1 ch2 -- ? )
2008-03-29 21:36:58 -04:00
[ 1string roman-digits index ] bi@ >= ;
2007-09-20 18:09:08 -04:00
: roman>n ( ch -- n )
1string roman-digits index roman-values nth ;
: (>roman) ( n -- )
roman-values roman-digits [
2008-12-14 01:37:37 -05:00
[ /mod swap ] dip <repetition> concat %
2007-09-20 18:09:08 -04:00
] 2each drop ;
: (roman>) ( seq -- n )
2009-02-05 10:09:24 -05:00
[ [ roman>n ] map ] [ all-eq? ] bi [
2007-09-20 18:09:08 -04:00
sum
] [
first2 swap -
] if ;
2008-10-21 21:30:26 -04:00
2007-09-20 18:09:08 -04:00
PRIVATE>
: >roman ( n -- str )
2009-01-15 18:56:09 -05:00
dup roman-range-check
[ (>roman) ] "" make ;
2007-09-20 18:09:08 -04:00
: >ROMAN ( n -- str ) >roman >upper ;
: roman> ( str -- n )
2009-01-15 18:56:09 -05:00
>lower [ roman<= ] monotonic-split
[ (roman>) ] sigma ;
2007-09-20 18:09:08 -04:00
<PRIVATE
2008-10-21 21:30:26 -04:00
2007-09-20 18:09:08 -04:00
: 2roman> ( str1 str2 -- m n )
2008-03-29 21:36:58 -04:00
[ roman> ] bi@ ;
2007-09-20 18:09:08 -04:00
: binary-roman-op ( str1 str2 quot -- str3 )
2008-12-14 01:37:37 -05:00
[ 2roman> ] dip call >roman ; inline
2008-10-21 21:30:26 -04:00
2007-09-20 18:09:08 -04:00
PRIVATE>
: roman+ ( str1 str2 -- str3 )
[ + ] binary-roman-op ;
: roman- ( str1 str2 -- str3 )
[ - ] binary-roman-op ;
: roman* ( str1 str2 -- str3 )
[ * ] binary-roman-op ;
: roman/i ( str1 str2 -- str3 )
[ /i ] binary-roman-op ;
: roman/mod ( str1 str2 -- str3 str4 )
2008-12-14 01:37:37 -05:00
[ /mod ] binary-roman-op [ >roman ] dip ;
2008-10-21 21:30:26 -04:00
: ROMAN: scan roman> parsed ; parsing