]> gitweb.factorcode.org Git - factor.git/blob - basis/roman/roman.factor
Merge branch 'master' of git://factorcode.org/git/factor
[factor.git] / basis / roman / roman.factor
1 ! Copyright (C) 2007 Doug Coleman.
2 ! See http://factorcode.org/license.txt for BSD license.
3 USING: arrays assocs kernel math math.order math.vectors
4 namespaces make quotations sequences splitting.monotonic
5 sequences.private strings unicode.case lexer parser
6 grouping ;
7 IN: roman
8
9 <PRIVATE
10
11 : roman-digits ( -- seq )
12     { "m" "cm" "d" "cd" "c" "xc" "l" "xl" "x" "ix" "v" "iv" "i" } ;
13
14 : roman-values ( -- seq )
15     { 1000 900 500 400 100 90 50 40 10 9 5 4 1 } ;
16
17 ERROR: roman-range-error n ;
18
19 : roman-range-check ( n -- )
20     dup 1 3999 between? [ drop ] [ roman-range-error ] if ;
21
22 : roman<= ( ch1 ch2 -- ? )
23     [ 1string roman-digits index ] bi@ >= ;
24
25 : roman>n ( ch -- n )
26     1string roman-digits index roman-values nth ;
27
28 : (>roman) ( n -- )
29     roman-values roman-digits [
30         [ /mod swap ] dip <repetition> concat %
31     ] 2each drop ;
32
33 : (roman>) ( seq -- n )
34     [ [ roman>n ] map ] [ all-eq? ] bi [
35         sum
36     ] [
37         first2 swap -
38     ] if ;
39
40 PRIVATE>
41
42 : >roman ( n -- str )
43     dup roman-range-check
44     [ (>roman) ] "" make ;
45
46 : >ROMAN ( n -- str ) >roman >upper ;
47
48 : roman> ( str -- n )
49     >lower [ roman<= ] monotonic-split
50     [ (roman>) ] sigma ;
51
52 <PRIVATE
53
54 : 2roman> ( str1 str2 -- m n )
55     [ roman> ] bi@ ;
56
57 : binary-roman-op ( str1 str2 quot -- str3 )
58     [ 2roman> ] dip call >roman ; inline
59
60 PRIVATE>
61
62 : roman+ ( str1 str2 -- str3 )
63     [ + ] binary-roman-op ;
64
65 : roman- ( str1 str2 -- str3 )
66     [ - ] binary-roman-op ;
67
68 : roman* ( str1 str2 -- str3 )
69     [ * ] binary-roman-op ;
70
71 : roman/i ( str1 str2 -- str3 )
72     [ /i ] binary-roman-op ;
73
74 : roman/mod ( str1 str2 -- str3 str4 )
75     [ /mod ] binary-roman-op [ >roman ] dip ;
76
77 SYNTAX: ROMAN: scan roman> parsed ;