making_a_web_site2.anubis 137 KB
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917 1918 1919 1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982 1983 1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035 2036 2037 2038 2039 2040 2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054 2055 2056 2057 2058 2059 2060 2061 2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077 2078 2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114 2115 2116 2117 2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178 2179 2180 2181 2182 2183 2184 2185 2186 2187 2188 2189 2190 2191 2192 2193 2194 2195 2196 2197 2198 2199 2200 2201 2202 2203 2204 2205 2206 2207 2208 2209 2210 2211 2212 2213 2214 2215 2216 2217 2218 2219 2220 2221 2222 2223 2224 2225 2226 2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288 2289 2290 2291 2292 2293 2294 2295 2296 2297 2298 2299 2300 2301 2302 2303 2304 2305 2306 2307 2308 2309 2310 2311 2312 2313 2314 2315 2316 2317 2318 2319 2320 2321 2322 2323 2324 2325 2326 2327 2328 2329 2330 2331 2332 2333 2334 2335 2336 2337 2338 2339 2340 2341 2342 2343 2344 2345 2346 2347 2348 2349 2350 2351 2352 2353 2354 2355 2356 2357 2358 2359 2360 2361 2362 2363 2364 2365 2366 2367 2368 2369 2370 2371 2372 2373 2374 2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392 2393 2394 2395 2396 2397 2398 2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425 2426 2427 2428 2429 2430 2431 2432 2433 2434 2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448 2449 2450 2451 2452 2453 2454 2455 2456 2457 2458 2459 2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478 2479 2480 2481 2482 2483 2484 2485 2486 2487 2488 2489 2490 2491 2492 2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505 2506 2507 2508 2509 2510 2511 2512 2513 2514 2515 2516 2517 2518 2519 2520 2521 2522 2523 2524 2525 2526 2527 2528 2529 2530 2531 2532 2533 2534 2535 2536 2537 2538 2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586 2587 2588 2589 2590 2591 2592 2593 2594 2595 2596 2597 2598 2599 2600 2601 2602 2603 2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625 2626 2627 2628 2629 2630 2631 2632 2633 2634 2635 2636 2637 2638 2639 2640 2641 2642 2643 2644 2645 2646 2647 2648 2649 2650 2651 2652 2653 2654 2655 2656 2657 2658 2659 2660 2661 2662 2663 2664 2665 2666 2667 2668 2669 2670 2671 2672 2673 2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694 2695 2696 2697 2698 2699 2700 2701 2702 2703 2704 2705 2706 2707 2708 2709 2710 2711 2712 2713 2714 2715 2716 2717 2718 2719 2720 2721 2722 2723 2724 2725 2726 2727 2728 2729 2730 2731 2732 2733 2734 2735 2736 2737 2738 2739 2740 2741 2742 2743 2744 2745 2746 2747 2748 2749 2750 2751 2752 2753 2754 2755 2756 2757 2758 2759 2760 2761 2762 2763 2764 2765 2766 2767 2768 2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786 2787 2788 2789 2790 2791 2792 2793 2794 2795 2796 2797 2798 2799 2800 2801 2802 2803 2804 2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831 2832 2833 2834 2835 2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912 2913 2914 2915 2916 2917 2918 2919 2920 2921 2922 2923 2924 2925 2926 2927 2928 2929 2930 2931 2932 2933 2934 2935 2936 2937 2938 2939 2940 2941 2942 2943 2944 2945 2946 2947 2948 2949 2950 2951 2952 2953 2954 2955 2956 2957 2958 2959 2960 2961 2962 2963 2964 2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985 2986 2987 2988 2989 2990 2991 2992 2993 2994 2995 2996 2997 2998 2999 3000 3001 3002 3003 3004 3005 3006 3007 3008 3009 3010 3011 3012 3013 3014 3015 3016 3017 3018 3019 3020 3021 3022 3023 3024 3025 3026 3027 3028 3029 3030 3031 3032 3033 3034 3035 3036 3037 3038 3039 3040 3041 3042 3043 3044 3045 3046 3047 3048 3049 3050 3051 3052 3053 3054 3055 3056 3057 3058 3059 3060 3061 3062 3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091 3092 3093 3094 3095 3096 3097 3098 3099 3100 3101 3102 3103 3104 3105 3106 3107 3108 3109 3110 3111 3112 3113 3114 3115 3116 3117 3118 3119 3120 3121 3122 3123 3124 3125 3126 3127 3128 3129 3130 3131 3132 3133 3134 3135 3136 3137 3138 3139 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 3150 3151 3152 3153 3154 3155 3156 3157 3158 3159 3160 3161 3162 3163 3164 3165 3166 3167 3168 3169 3170 3171 3172 3173 3174 3175 3176 3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202 3203 3204 3205 3206 3207 3208 3209 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241 3242 3243 3244 3245 3246 3247 3248 3249 3250 3251 3252 3253 3254 3255 3256 3257 3258 3259 3260 3261 3262 3263 3264 3265 3266 3267 3268 3269 3270 3271 3272 3273 3274 3275 3276 3277 3278 3279 3280 3281 3282 3283 3284 3285 3286 3287 3288 3289 3290 3291 3292 3293 3294 3295 3296 3297 3298 3299 3300 3301 3302 3303 3304 3305 3306 3307 3308 3309 3310 3311 3312 3313 3314 3315 3316 3317 3318 3319 3320 3321 3322 3323 3324 3325 3326 3327 3328 3329 3330 3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342 3343 3344 3345 3346 3347 3348 3349 3350 3351 3352 3353 3354 3355 3356 3357 3358 3359 3360 3361 3362 3363 3364 3365 3366 3367 3368 3369 3370 3371 3372 3373 3374 3375 3376 3377 3378 3379 3380 3381 3382 3383 3384 3385 3386 3387 3388 3389 3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404 3405 3406 3407 3408 3409 3410 3411 3412 3413 3414 3415 3416 3417 3418 3419 3420 3421 3422 3423 3424 3425 3426 3427 3428 3429 3430 3431 3432 3433 3434 3435 3436 3437 3438 3439 3440 3441 3442 3443 3444 3445 3446 3447 3448 3449 3450 3451 3452 3453 3454 3455 3456 3457 3458 3459 3460 3461 3462 3463 3464 3465 3466 3467 3468 3469 3470 3471 3472 3473 3474 3475 3476 3477 3478 3479 3480 3481 3482 3483 3484 3485 3486 3487 3488 3489 3490 3491 3492 3493 3494 3495 3496 3497 3498 3499 3500 3501 3502 3503 3504 3505 3506 3507 3508 3509 3510 3511 3512 3513 3514 3515 3516 3517 3518 3519 3520 3521 3522 3523 3524 3525 3526 3527 3528 3529 3530 3531 3532 3533 3534 3535 3536 3537 3538 3539 3540 3541 3542 3543 3544 3545 3546 3547 3548 3549 3550 3551 3552 3553 3554 3555 3556 3557 3558 3559 3560 3561 3562 3563 3564 3565 3566 3567 3568 3569 3570 3571 3572 3573 3574 3575 3576 3577 3578 3579 3580 3581 3582 3583 3584 3585 3586 3587 3588 3589 3590 3591 3592 3593 3594 3595 3596 3597 3598 3599 3600 3601 3602 3603 3604 3605 3606 3607 3608 3609 3610 3611 3612 3613 3614 3615 3616 3617 3618 3619 3620 3621 3622 3623 3624 3625 3626 3627 3628 3629 3630 3631 3632 3633 3634 3635 3636 3637 3638 3639 3640 3641 3642 3643 3644 3645 3646 3647 3648 3649 3650 3651 3652 3653 3654 3655 3656 3657 3658 3659 3660 3661 3662 3663 3664 3665 3666 3667 3668 3669 3670 3671 3672 3673 3674 3675 3676 3677 3678 3679 3680 3681 3682 3683 3684 3685 3686 3687 3688 3689 3690 3691 3692 3693 3694 3695 3696 3697 3698 3699 3700 3701 3702 3703 3704 3705 3706 3707 3708 3709 3710 3711 3712 3713 3714 3715 3716 3717 3718 3719 3720 3721 3722 3723 3724 3725 3726 3727 3728 3729 3730 3731 3732 3733

   
 *Project*                                Anubis
   
 *Title*                        Making interactive Web sites.
   
 *Copyright*                Copyright (c) Alain Prouté 2004-2005.
   
   
 *Author* Alain Prouté

 *Revised* January 2005. 
   
   
  *Overview*
   
   In this file we propose simple tools for making well structured interactive and secured
   web sites.
   
   
   ----------------------------------- Table of Contents ---------------------------------

   * (1) Structure of a web site. 
      ** (1.1) Three sorts of data. 
      ** (1.2) How requests are handled. 
      ** (1.3) What web pages are made of. 
      ** (1.4) Actions.
      ** (1.5) States. 
   
   * (2) Carrying on. 
      ** (2.1) Describing your web sites. 
      ** (2.2) Directories on the server's disk.
      ** (2.3) Starting your web sites. 
   
   * (3) The HTML interface. 
      ** (3.1) Types used by the HTML interface. 
      ** (3.2) ``in form'' versus ``off form''. 
      ** (3.3) Defining your own style. 
      ** (3.4) Actioners and forms. 
      ** (3.5) Local popup. 
   
   ---------------------------------------------------------------------------------------
   
   
read tools/basis.anubis   
read tools/printable_tree.anubis   
read common.anubis   
read multihost_http_server.anubis   
read mime.anubis   

   
   
   * (1) Structure of a web site. 
   
   First of  all we  need to  explain what  a web site  should be  made of.   Ideally, the
   visitor  (also called  the  'client') should  see the  web  site working  as any  other
   interactive computer software.  So, it should  be clear that a 'session' (i.e.  a visit
   to  the  web site,  including  the  consultation of  several  pages)  is  some kind  of
   conversation  between the  visitor  and the  web site,  and  that the  web site  should
   maintain a 'current state' of this  conversation.  At each new request (click) from the
   visitor, this state must be updated.  This whole conversation is called a 'session' and
   should not be confused with a single request.

   
   
      ** (1.1) Three sorts of data. 
   
   All  the data  needed for  putting a  web site  at work  may be  dispatched  into three
   categories:
   
     1. Constant data  (data that  never change). These  data may  be hard coded  into the
        Anubis source files of the web site.
   
     2. Permanent data  (data which always exist  independantly of the  users connected to
        the web site). These data are normally recorded into data bases. 
   
     3. Session  data (data  which depend  on a  particular visitor  and which  exist only
        during the  time he  visits the web  site). These  data are stored  into so-called
        'states'.
   
   
   It is important to determine which data belongs to which category. This is part of your
   design decisions.
   
   
   
      ** (1.2) How requests are handled. 
   
   We want to  separate the following two functionalities (which are  used at each request
   (click) during a single session):
   
      - computing the new  state from the previous state and from  the client request, and
        updating the data base,
   
      - computing the page  to be sent to the  client from the new current  state and from
        the informations in the data base.
   
   
   The next picture shows the structure we have in mind:
   
   
                request            +---------+         HTML page (with a hidden state name)
               .-------------------| client  |<--------------.
               | .-----------------|         |               |
               | | previous state  +---------+               |
               | | name (if any)                             |         
               | |                                           |          client side
       ............................................................................
               | |                                           |          server side
               | |                                           |
               | | .-------------------.                     |
               | | | previous state    |                     |
               V V V                   |                     |
       +---------------+             +---------------+     +--------------+
       | compute state |             | server's disk |     | compute page |
       +---------------+             +---------------+     +--------------+
           ^     | |                    ^ ^                  ^ ^  ^
           |     | |                    | |                  | |  |
           |     | `--------------------+--------------------' |  |
           |     |   new state            |                    |  |
      read |     `------------------------+--------------------'  |
     write |         new state name                               |
    update V                                                      |
         +-----------+                                            |
         | data base |--------------------------------------------'
         +-----------+              read only 
   
   
   When the client begins  a session, there is no previous state.  In this case, a default
   'initial state' is used instead.

   The data base  may be updated by 'compute  state' box, but should not be  update by the
   'compute page'  box. The 'compute  page' box  should be allowed  only to read  the data
   base.
   
   In this file, all  the above stuff is defined, except the  'compute state' and 'compute
   page' boxes. You just  have to provide the function for computing  a new state (compute
   state) and the function for computing the  page (compute page) from the new state.  You
   don't have to worry about state names, saving and retrieving states and the like.

   
   
   
   
      ** (1.3) What web pages are made of. 
   
   What the client can see in his browser's  window may be called a 'page'. Within a page,
   we have several sorts of components:
   
      - 'local'  components, i.e.  all  components which  do not  open a  connection, like
        texts, images, etc...  possibly using JavaScript programmation, 
   
      - 'actioners', which, when  clicked upon, open a connection with  our web site; they
        may appear as links or buttons, etc...
   
      - 'foreign links', which when clicked upon,  open a connection with another web site
        (or ours eventually).
      
   Of course, what an actioner does is just ask our web site to perform an action. To that
   end, the actioner essentially sends the name of the action to be performed. However, it
   may be necessary to provide additional  informations which may be seen as 'operands' of
   the action.   In order to  attach operands  to an action,  HTML provides the  notion of
   'form'. Indeed, a form contains essentially a set of input fields into which the client
   may put values for  the required operands of the action, and  a submit button, which is
   the actioner  itself. Notice  that a  single form may  contain several  submit buttons,
   which simply  means that  there are several  distincts actions  taking the same  set of
   operands.
   
   Restrictions  must be  put on  the use  of all  theses gadgets.   Indeed,  for example,
   putting a form within another form  is officially meaningless in HTML, and the client's
   browser may be  seriously disturbed by this.  In this file, we propose  an interface to
   the HTML language,  which forbids such meaningless things, simply  by imposing a strict
   typing of HTML concepts.

   Each web site may be accessible through two communication channels: 
   
      - a non secured channel (HTTP), 
      - a secured channel (HTTPS). 
   
   Nevertheless, the whole  thing should be considered as a single  web site. For example,
   you may have a secured page,  obtained through HTTPS, containing public images obtained
   through HTTP.   An actioner in a  non secured page  may open a secured  connection, and
   conversely.
   
   Summarizing,  a web  page  is made  of  local elements,  foreign  links and  actioners.
   Actioners receive operands from forms, and  they also choose to communicate through the
   non secured or through the secured channel.
   
   
   
       +-------------------+
       | page              |
       |                   |                        +---------------+
       | +--------------+  |                        | next page     | 
       | | form         |  |                        | (non secured) |
       | | +----------+ |  |    HTTP                |               |
       | | | actioner |---------------------------->|               |
       | | +----------+ |  |                        +---------------+
       | |              |  |
       | | +----------+ |  |                        +---------------+
       | | | actioner |---------------------------->| next page     |
       | | +----------+ |  |    HTTPS               | (secured)     |
       | |              |  |                        |               |
       | +--------------+  |                        |               |
       |                   |                        +---------------+
       |                   |
       +-------------------+
   

   Notice that actioners need no be necessarily put into forms. In that case, they work as
   ordinary links, but they still may receive operands as we shall see.
            
   
   
   
      ** (1.4) Actions.
   
   The  client  opens a  new  connection  with  our web  site  whenever  he clicks  on  an
   actioner. The  result is that  a request is  sent, essentially made  of a list  of 'web
   arguments'.  Each web argument is a  pair (name,value). One of these web arguments, the
   'action' web argument (whose name is  "a"), determines the action to be performed.  The
   other web  arguments (not including "s", used  to identify the state)  are the operands
   for this action.
      
   Hence, the  'compute state'  box in the  picture above,  splits naturally into  as many
   sub-boxes  as there are  actions. For  this reason,  we define  the following  type for
   representing actions (where '$State' is the type representing session informations):
      
public type Web_Action($SessionTicket, $State):
   http_action  (String                                     name,  // name of action
                 (Maybe($State)) -> Bool                    allow, // true if action allowed
                 (HTTP_Info          http_info, 
                  List(Web_arg)      web_args,      // actually only 'operands' web arguments
                  Maybe($State)      state) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))       do_it),
   https_action (String                                     name,  // name of action
                 (Maybe($State)) -> Bool                    allow, // true if action allowed
                 (HTTP_Info          http_info, 
                  List(Web_arg)      web_args,      // actually only 'operands' web arguments
                  Maybe($State)      state) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))       do_it),
   http_https_action (String                                name, 
                 (Maybe($State)) -> Bool                    allow, // true if action allowed
                 (HTTP_Info          http_info, 
                  List(Web_arg)      web_args,    
                  Maybe($State)      state) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))       do_it).
                  
   'http_action's  are executed  only under  HTTP, and  'https_action's are  executed only
   under HTTPS. 'http_https_action's may be executed under both types of connections.
   
   Each action has  a name, which is used  to identify the action. Each action  also has a
   function 'allow'  whose job  is to  verify that the  action is  allowed in  the current
   state, and a function 'do_it' for performing the action.  The function 'do_it' receives
   a lot of informations:
   
       - 'HTTP informations':
           - the IP address of the client, 
           - the URI requested by the client (after redirection), 
           - the list of HTTP headers generated by the client's browser, 
       - the list  of web arguments sent  by the client  (except "s" and "a"),
       - the previous state (or the 'initial' or 'ticket expired' state if no previous 
           state can be found).
   
   In most cases, HTTP informations are not used. This is the reason why they are gathered
   for simplicity into a unique datum of type 'HTTP_Info'.
   
//   For your convenience, we introduce the following simpler variants:
//   
//public define Web_Action($State)   
//   http_action
//     (
//       String                                name, 
//       $State -> Bool                        allow, 
//       (List(Web_arg),$State) -> $State      do_it
//     ) =
//   http_action(name,
//               (Maybe($State) ms) |-> if ms is 
//               {
//                  failure then    true,
//                  success(s) then allow(s)
//               },
//               (HTTP_Info h, List(Web_arg) l, Maybe($State) s) |-> if s is 
//                  {
//                    failure then      (failure, []),
//                    success(s2) then  (success(do_it(l,s2)), [])
//                  }).
//   
//public define Web_Action($State)   
//   https_action
//     (
//       String                                name, 
//       $State -> Bool                        allow, 
//       (List(Web_arg),$State) -> $State      do_it
//     ) =
//   https_action(name,
//               (Maybe($State) ms) |-> if ms is 
//               {
//                  failure then    true,
//                  success(s) then allow(s)
//               },
//               (HTTP_Info h, List(Web_arg) l, Maybe($State) s) |-> if s is 
//                  {
//                    failure then      (failure, []),
//                    success(s2) then  (success(do_it(l,s2)), [])
//                  }).
//   
//public define Web_Action($State)   
//   http_https_action
//     (
//       String                                name, 
//       $State -> Bool                        allow, 
//       (List(Web_arg),$State) -> $State      do_it
//     ) =
//   http_https_action(name,
//                     (Maybe($State) ms) |-> if ms is 
//                     {
//                        failure then    true,
//                        success(s) then allow(s)
//                     },
//                     (HTTP_Info h, List(Web_arg) l, Maybe($State) s) |-> if s is 
//                        {
//                          failure then      (failure, []),
//                          success(s2) then  (success(do_it(l,s2)), [])
//                        }).
   
   
   
   When you  define your web  site, you must  provide the list of  all the actions  of the
   site.  When  a new state  has been computed,  a graphical representation of  this state
   must be  sent to the  client.  To that  end, you must  provide a function  (named below
   'compute_page') of type:
   
                                   $State -> HTML_Page

   where  the type 'HTML_Page'  (defined below  in this  file) abstractly  represents HTML
   pages.
   
public type HTML_Page:...   
   
   It should be clear that states and  pages are deeply linked together. Indeed, we really
   understand the page shown to the client as a representation of the current state of the
   conversation  between the client  and the  web site,  but also  containing informations
   taken from the data bases.
   
   
   

   
      ** (1.5) States. 

   Now, we explain how you can define the  type (say 'State') to be used as an instance of
   the type parameter '$State'. The following is just a suggestion. 
   
   Each  state   determines  a  page  (since   'compute_page'  computes  a   page  from  a
   state). However, some components of the state may be independant of the page. It may be
   the  case  for  example  for  the  indication  of the  natural  language  used  by  the
   client. Hence, a state should be made of (at least) two parts:
   
      - informations which are the same for all pages, 
      - informations which are particular to each page. 
   
   For example, you could define:
   
   type Page:                // one alternative per page, with particular informations
     login(...),             // in the components
     main_page(...),
     ...etc...
   
   Now, the type 'State' could be defined as follows:
   
   type State:
     state(Language,         // informations valid for all pages
           ...,
           Page).            // informations particular to a page

   However, if you are making a secured web site within which clients should be identified
   (by id and  password), it may be a good  idea to have two sorts of  states, one for non
   identified  clients and  one for  identified  clients. In  this case,  define the  type
   'State' as follows (this is just a suggestion):
   
   type State:
     non_identified(Language), 
     identified(String id, 
                Language, 
                Page). 
   
   When  a  request  arrives,  check  if  the  previous  state  is  'identified(...)'   or
   'non_identified(...)',  and don't  provide access  to certain  pages to  non identified
   clients. This is required for security.

   Some  more words  on security.   If your  site needs  to identify  clients,  define the
   initial state as 'non_identified(...)'.  Construct a 'login' page, and check the id and
   password of the  client. If the id and  password are correct, then change  the state of
   the client to  'identified(...)'. No other action  should be able to do  that.  Now, be
   confident that clients cannot forge states.  The only information they have is the name
   of a state, not the state itself which  is never sent over the network, but only stored
   on the server's disk. The name of the state is constructed using strong cryptographical
   methods (sha1).  If everything (since the  'login' page) is performed under HTTPS, even
   state names  cannot be seen by  a third party. So,  if the system  retrieves a previous
   state of  the form  'identified(...)', you can  be confident  that your client  is well
   identified, and you can send him confidential informations.

   States have a  limited life time. It may happen  that a client clicks on  a button at a
   time its state is  out of date. In this case, this system  considers that the new state
   is a special  state named 'ticket expired'. You must provide  a function producing this
   state when you describe  your web site. The page corresponding to  this state must just
   inform the client that  he/she waited a too long time before  clicking on a button, and
   has to restart (a new conversation) from the begining.
   
   
   
   * (2) Carrying on. 
   
      ** (2.1) Describing your web sites. 
   
   Before you may start your web site, you  must describe it, i.e.  produce a datum of the
   opaque type 'Web_Site'.
   
public type Web_Site:...
   
   Producing such a datum may be performed by:
   
public define Web_Site
   make_web_site_description
     (
       List(String)                common_names,          // for example: ["www.our-business.com",
                                                          //               "192.168.0.1"]
                                                          // the second one is just for testing
       String                      site_directory,        // where 'public' and other directories are
                                                          // located (should NOT end with '/')
       One -> One                  init, 
       (HTTP_Info) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))        initial_state,
       ($State expired,
        HTTP_Info,
        List(Web_arg),
        Bool is_https) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))    ticket_expired_state, 
       (HTTP_Info,
        List(Web_arg),
        Bool is_https) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))    ticket_lost_state, 
       List(Web_Action($SessionTicket, $State))    actions, 
       (Maybe($SessionTicket), Maybe($State)) -> HTML_Page                             compute_page,
       Int                         timeout,               // seconds (todo: minutes)
       List(Redirection)           redirections, 
       String                      charset, 
       List(String)                journal_extensions,
       List(String)                journal_headers, 
       String                      authorization_secret, 
       List(MIME)                  known_mime_types,
       (String        action_name, 
        List(Web_arg) args)-> One  before_send_file
     ). 

   
   Explanations:
   
   'common_names' is the list of names of the  site (the name the browser must send as the
   value of the 'Host' HTTP header in order to access the site must be in that list). Such
   a name generally looks like this:
   
                                       www.somewhere.com
   
   If you are using  HTTPS, you also have an 'X.509 SSL  server certificate'.  The name of
   the site must  be exactly the same as  the name on the certificate  (which is precisely
   called the 'common name' in the X.509 jargon).  If the two names do not match, the site
   will still work, but the transaction will not be transparent to the client. His browser
   will complain that the name of the certificate does not match the name of the site, and
   he will have to accept the certificate manually.

   'site_directory' is  the absolute path to the  directory where the files  needed by the
   site are located. Usually this directory looks like:
   
                            my_anubis/web_sites/www.somewhere.com
   
   However, this  information is not computed  from 'common_name', so that  you can change
   the  common name  (for example  temporarily,  for networking  reasons) without  loosing
   access to the files.
   
   'ticket_expired_state(expired_state,http_info,lwa,is_https)'  must  produce  the  state
   whose graphical representation  is a page explaining to the user  that its 'ticket' (or
   'session information')  has expired, and that  he/she must close all  popup windows and
   start a  new session.   The arguments  of the function  contain the  previous (expired)
   state and all  current informations concerning the user.  This  arguments may be useful
   for example  for producing the expiration message  in the language chosen  by the user.
   You  can  also (and  this  may  be much  smarter)  send  a  'ticket prolongation  page'
   (including a new  login for example), and resume the same  conversation, since you have
   all the pertinent informations at hand. In  the case the ticket is definitely lost, the
   second fonction 'ticket_lost_state' is used. 
   
   Notice that despite the fact that the  parameter $State is involved in the arguments of
   the above function, the type 'Web_Site' does not depend on this parameter.  This allows
   to produce  lists of web site  descriptions, where each description  may be constructed
   with a different instance of $State.  This is required because distinct sites must have
   distinct types  of session informations.   This is made  possible by the fact  that the
   type  is  obscure,  and  the   constructor  replaced  by  a  function  which  assembles
   'ticket_expired_state', ticket_lost_state', 'actions'  and 'compute_page' into a single
   entity not depending on $State. You should have a look to the private part of this file
   if you want more precisions about this programming technique.
   
   'charset' is  a string which will  determine the character  encoding to be used  by the
   browser.  Typically, this  string  is one  of:  "UTF-8", "ISO-8859-1",  "Windows-1252",
   etc...
   
   'before_send_file' is a function which is  executed just before the HTTP server sends a
   file. It gets an  action name and the web arguments received  with the request for that
   file.   Notice that  this  action  name and  these  web arguments  may  be  put into  a
   'private_download'  element, and  will  come back  to the  server  at the  time of  the
   download.
   
   
   
      ** (2.2) Directories on the server's disk.

   The  description of  you  site contains  the name  of  the directory  within which  the
   required files are located. This may be for example:
   
                          my_anubis/web_sites/www.our-business.com/ 
   
   This is  called the 'site directory' (for  the given site). Within  the site directory,
   the following directories are created by this program:
   
          states
          public
          journal
          private_download
          upload_temporary

   The directory 'states'  is used for storing states (session  informations). Out of date
   states are automatically removed after some time.
   
   The tree rooted  at 'public' contains files that  the server is allowed to  send to the
   clients. For security  reasons, the server never  sends a file which is  not within the
   tree whose root is this 'public' directory (except for the 'private download' mecanism;
   see 'web/multihost_http_server.anubis').   Also, the MIME  type (see 'web/mime.anubis')
   must have been recognized before the file may be sent.
   
   The  directory  'journal'  contains the  jounal  files.   The  roles of  the  remaining
   directories    'private_download'    and    'upload_temporary'    is    explained    in
   'multihost_http_server.anubis',  where  you  will  also find  further  informations  on
   'public' and 'journal'.
   
   

   
      ** (2.3) Web servers parameters. 
   
   The web servers  have several parameters useful for  administration. They are described
   as follows:
   
 public type WebServersParameters:
   wsparms(Var(Bool)      shutdown_required, 
   
   
   
 
      ** (2.4) Starting your web sites. 
   
   When you have described all your web sites (you may want to have several web sites, and
   they are distinguished  by their 'common name'), you may start  them all together using
   'start_web_sites' below.  This function returns a result of the following type:
   
public type Start_Web_Sites_Result:
   cannot_bind_to_port(Word32), 
   cannot_bind_to_port(Word32,Word32), 
   ok(Server http_server, 
      Server https_server).
      
   Indeed, it may happen  that the system cannot bind (begin to listen)  to one of the two
   ports (or to both). The main reason is that another server is already listening on that
   port. Another reason may be that  'anbexec' has not been correctly installed, i.e. that
   the 's' bit  has not been set for  'user' and 'group' (there is not  such problem under
   Windows). Also notice that the Linux kernel  may need a rather long time (up to several
   minutes) before liberating  a listening port.  Now,  if the system can bind  to the two
   ports, the pair of  the two servers is returned. Two tools  are useful for manipulating
   servers:
   
      shutdown        of type   Server -> One
      is_down         of type   Server -> Bool
   
   They are defined in 'predefined.anubis' (together with the type 'Server'). 
   
   
public define Start_Web_Sites_Result
   start_web_sites
     (
       Word32                       ip_address,    // the IP address shared by the web sites
       Word32                       http_port,     // usually: 80
       Word32                       https_port,    // usually: 443
       String                       ssl_certificate_common_name, 
       List(Web_Site)               web_sites,     // web sites to be started
       Var(Bool)                    shutdown_required
     ). 
   
   'ip_address' is  the IP  address on which  the two  servers listen. If  you put  0, the
   servers  listen on all  the IP  addresses of  the machine.  This may  be useful  if the
   machine has several network interfaces. 

   'ssl_certificate_common_name' is the common name  of the SSL certificate that 'anbexec'
   loads when it starts. One instance of  'anbexec' cannot handle more than one SSL server
   certificate. This is  due to a problem of  conception of SSL itself. See  the book 'SSL
   and TLS' by Eric Rescorla (at Addison Wesley) for more explanations.
   
   Notice that the  number of servers is always  2, regardless of the number  of web sites
   you are starting.  
   
   The dynamic variable 'shutdown_required' may be used to control the shutdown of the two
   servers from within the web site  (typically the administration part). The servers will
   shutdown as  soon as  this variable contains  'true'.  So  you must provide  a variable
   containing 'false' otherwise your servers will  not run. You may also use the primitive
   'must_restart' (see 'predefined.anubis') to control the restarting of your servers. 
   
   
   
   
   
   
   * (3) The HTML interface. 
   
   We propose an interface to dynamic  HTML. Dynamic HTML includes HTML, and a combination
   of CSS (Cascading Style Sheet) and  JavaScript techniques for making HTML elements more
   reactive and attractive on the client side.
      
   
      ** (3.1) Types used by the HTML interface. 
   
   For easy reference, we  gather below the definitions of all the  types used by the HTML
   interface, and we comment them immediately. 

   
public type HTML_Size:
   absolute(Word32),        // in pixels
   percentage(Word32). 
   
   
public type Text_Option:   
   size(Word32),            // size of character font to use
   font(String),           // name of character font to use (such as "helvetica",...)
   color(RGB),             // color to be used for characters
   italic, 
   oblique, 
   small_capitals, 
   bold, 
   underlined, 
   left_justified,
   right_justified,
   justified,              // justified on both sides
   line_through,
   nowrap,
   css_class(String).      //CSS class 
   
   A list of 'Text_Option' must be given with each text you want to put in your page.

  This indicate the way of reading text. 
public type Reading_Way:
  ltr,  //the text is readable from "Left To Right" like english
  rtl.  //the text is readable from "Right To Left" like arabic
  
public type DIV_Option:
  class (String),
  dir   (Reading_Way),
  id    (String),
  title (String),
  lang  (String).
  
  A list of 'DIV_Option' must be given with each DIV you want to put in your page.

  
public type Table_Option:
   background_color(RGB),                    // applied to all cells in the table
   background_image(String url), 
   border(Word32   width_of_outer_edge,       // if not present, all values are 0
          Word32   width_of_top_of_relief,
          Word32   width_of_inner_edge,
          RGB     border_color),
   width(Word32),               // sets a minimal width for the table
   percentage_width(Word32). 
   
   
public define Table_Option nude = border(0,0,0,rgb(0,0,0)).
   
   
   A list of 'Table_Option' must be given with each table. 
   
   
public type BackgroundOption:
   repeat,              // repeat the background in both directions
   repeat_horizontal,   // repeat the background only horizontally
   repeat_vertical,     // repeat the background only verticall
   no_repeat,           // don't repeat the background
   center.              
  
   
public type Cell_Option:
   left,                    // put the content of the cell on the left
   h_center,                // center the content of the cell horizontally
   right,                   // put the content of the cell on the right
   top,                     // put the content of the cell upwards
   v_center,                // center the content of tye cell vertically, 
   bottom,                  // put the content of the cell downwards
   base_line,               // align the content vertically according to base lines
   background_color(RGB), 
   background_image(String url, BackgroundOption), 
   width(Word32),            // sets a minimal width for the cell
   percentage_width(Word32), 
   height(Word32),           // sets a minimal height for the cell
   columns(Word32),          // lets the cell span over several columns
   rows(Word32),             // lets the cell span over several rows
   nowrap.                  // do not allow text wrapping within the cell
   
   A list of 'Cell_Option'  must be given with each cell and each  row in a table. Options
   given with a row apply to all the cells in the row, but are superseded by options given
   with cells, which apply only to the cell they are given with.
   
   
public type HTML_Cell($T):
   cell(List(Cell_Option) options, $T content). 
   
   The parameter $T is later  instantiated either to 'HTML_In_Form' or to 'HTML_Off_Form',
   depending on where  you put your table (within  a form or not within a  form). For your
   convenience, we define the following particular case:
   
public define HTML_Cell($T)   
   cell
     (
       $T content
     ) =
   cell([],content). 
   
   
   
public type HTML_Row($T):
   row(List(Cell_Option) options, List(HTML_Cell($T)) cells). 
   
   Same remark as for 'HTML_Cell($T)'. We define several convenience functions:
   
public define HTML_Row($T)
   row
     (
       List(HTML_Cell($T)) cells
     ) =
   row([],cells).
   
public define HTML_Row($T)   
   row
     (
       HTML_Cell($T) cell
     ) =
   row([],[cell]). 
   
public type Actioner_Connection:
   same,                            // use same type of connection as current page
   http,                            // use non secured connection
   https.                           // use secured connection
   
public type Other_Window_Option:
   resizable,                       // the new window may be resized by the client
   scrollbars,                      // the new window has scrollbars
   width(Word32),                    // the new window has the specified width
   height(Word32).                   // the new window has the specified height
   
public type Actioner_Target:
   same,
   same (String label), 
   other(String window_name, List(Other_Window_Option)). 

public type Actioner_Aspect:
   link                (List(Text_Option),String text),                    // hypertext link
   push_button         (List(Text_Option),String text),
   button              (String url_off, String url_on),                    // rollover button
   button              (String url_off, String url_on, Word32 w, Word32 h),  // idem with size
   button              (String text, Word32 width, RGB color),              // rollover button
   immediate_selector  (String name, Word32 size, List(String) choices).

   
public type Actioner_Local_Action:
   close_window. 
   
   
public define Actioner_Aspect
   link
     (
       String text
     ) =
   link([],text). 
   
   
public define Actioner_Aspect
   link
     (
       List(Text_Option) options,
       Word32              i
     ) =
   link(options,to_decimal(i)). 
   
public define Actioner_Aspect
   link
     (
       Word32 i
     ) =
   link([],i).
   
public define Actioner_Aspect
   button
     (
       String  url_img
     ) =
   button(url_img,url_img).
   
   
   
   Actioners are explained in details below.    
   
   
public type TextAreaOption:
   disabled,
   read_only, 
   wrap_lines. 
   
public type HTML_In_Form:
   literal_pt            (Printable_tree), 
   literal               (String), 
   sequence              (List(HTML_In_Form) items), 
   text                  (List(Text_Option), String the_text), 
   preformated           (List(Text_Option), String), 
   paragraph             (List(Text_Option), String the_text), 
   image                 (String url), 
   image                 (String url, Word32 width, Word32 height), 
   table                 (List(Table_Option), List(HTML_Row(HTML_In_Form))), 
   center                (HTML_In_Form), 
   mail_to               (String email, HTML_In_Form element), 
   scroller              (Word32 width, Word32 height, 
                          Word32 content_width, Word32 content_height,
                          HTML_In_Form content), 
   actioner              (Actioner_Connection, Actioner_Target, Actioner_Aspect, 
                          String action_name, List((String,String)) extra_ops, 
                          List(Actioner_Local_Action)), 
   local_popup           (Actioner_Aspect, HTML_In_Form content, 
                          Word32 x, Word32 y, String title, RGB color, Word32 width), 
   foreign_link          (List(Text_Option), String url, String name), 
   private_download      (String abs_path, String name, String extra_ext, 
                            Maybe((String,List((String,String)))) action),    
   text_input            (String name, String init, Word32 width), 
   password_input        (String name, Word32 width), 
   text_area             (List(TextAreaOption), String name, String init, Word32 width, Word32 height), 
   file_upload           (String name, Word32 width), 
   selector              (String name, Word32 size, List(String) choices), 
   selector              (String name, Word32 size, List(String) choices, String selected), 
   // List((String,String)) = List((code,name)) where : 
   //    name appears in selector
   //    code is the web-arg value
   selector_c            (String name, Word32 size, List((String,String)) choices), 
   selector_c            (String name, Word32 size, List((String,String)) choices, String selected),
   radio_button          (String name, String value, Bool checked), 
   check_box             (String name, Bool checked). 
   
   'HTML_In_Form'  defines all  the  elements  you may  put  within a  form.  We define  a
   convenience function:

public define HTML_In_Form literal(Printable_tree t) = literal_pt(t). 
   
public define HTML_In_Form
   foreign_link
     (
       Word32 tsize,
       String url, 
       String name
     ) =
   foreign_link([size(tsize)],url,name). 
   
public define HTML_In_Form
   actioner
     (
       Actioner_Connection       conn, 
       Actioner_Target           targ, 
       Actioner_Aspect           asp, 
       String                    action_name, 
       List((String,String))     extra_ops
     ) =
   actioner(conn,targ,asp,action_name,extra_ops,[]). 
   
   
   
public define HTML_In_Form
   text_area
     (
       String name,
       String init,
       Word32  width, 
       Word32  height
     ) =
   text_area([],name,init,width,height). 
   
public define HTML_In_Form
   table
     (
       List(HTML_Row(HTML_In_Form)) rows
     ) =
   table([],rows). 
   
   
public define HTML_In_Form   
   private_download
     (
       String        abs_path, 
       String        name, 
       String        extra_ext
     ) =
   private_download(abs_path,name,extra_ext,failure). 

public define HTML_In_Form   
   private_download
     (
       String        abs_path, 
       String        name, 
       String        extra_ext,
       String        action_name,
       List((String,String))   args
     ) =
   private_download(abs_path,name,extra_ext,success((action_name,args))). 
   
public define HTML_In_Form
   text
     (
       String s
     ) =
   text([],s).
 
 
   
   
public type HTML_Off_Form:
   literal_pt            (Printable_tree),
   literal               (String),
   sequence              (List(HTML_Off_Form) items), 
   text                  (List(Text_Option), String the_text), 
   preformated           (List(Text_Option), String), 
   paragraph             (List(Text_Option), String the_text), 
   image                 (String url), 
   image                 (String url, Word32 width, Word32 height), 
   table                 (List(Table_Option), List(HTML_Row(HTML_Off_Form))), 
   center                (HTML_Off_Form),
   mail_to               (String email, HTML_Off_Form element), 
   scroller              (Word32 width, Word32 height, 
                          Word32 content_width, Word32 content_height,
                          HTML_Off_Form content), 
   fixed_size            (HTML_Size width, HTML_Size height, HTML_Off_Form content), 
   fixed_size_2          (HTML_Size width, HTML_Size height, String name_of_HTML_file), 
   actioner              (Actioner_Connection, Actioner_Target, Actioner_Aspect, 
                          String action_name, List((String,String)) extra_ops, 
                          List(Actioner_Local_Action)), 
   actioner              (Actioner_Connection, Actioner_Target, Actioner_Aspect, 
                          String action_name, List((String,String)) extra_ops, 
                          List(Actioner_Local_Action), String form_name), 
   local_popup           (Actioner_Aspect, HTML_Off_Form content, 
                          Word32 x, Word32 y, String title, RGB color, Word32 width), 
   foreign_link          (List(Text_Option), String url, String name), 
   private_download      (String abs_path, String name, String extra_ext,
                            Maybe((String,List((String,String)))) action),    
   label                 (String name),
   form                  (String form_name, HTML_In_Form content),
   div                   (List(DIV_Option), HTML_Off_Form content),
   div_empty             (List(DIV_Option)).
   
   'HTML_Off_Form' defines all the elements you may put outside any form. 

   
public define HTML_Off_Form literal(Printable_tree t) = literal_pt(t). 
public define HTML_Off_Form fixed_size(HTML_Size width, HTML_Size height, String name_of_HTML_file)
   = fixed_size_2(width,height,name_of_HTML_file). 

   
public define HTML_Off_Form
   foreign_link
     (
       Word32 tsize,
       String url, 
       String name
     ) =
   foreign_link([size(tsize)],url,name). 
   
   
public define HTML_Off_Form
   actioner
     (
       Actioner_Connection       conn, 
       Actioner_Target           targ, 
       Actioner_Aspect           asp, 
       String                    action_name, 
       List((String,String))     extra_ops
     ) =
   actioner(conn,targ,asp,action_name,extra_ops,[]). 
   
public define HTML_Off_Form
   table
     (
       List(HTML_Row(HTML_Off_Form)) rows
     ) =
   table([],rows). 
   
   
   
   We add two  convenience functions for 'row'.  The reason why we add  two functions, one
   for  'HTML_In_Form' and  one  for 'HTML_Off_Form',  is  that adding  a  schema with  an
   arbitrary '$T' creates too many ambiguities. This is due to the fact that, if we do so,
   the arguments of the function do not refer to any of the types defined here.
   
public define HTML_Row(HTML_In_Form)
   row
     (
       HTML_In_Form content
     ) =
   row([],[cell([],content)]).
   
public define HTML_Row(HTML_Off_Form)
   row
     (
       HTML_Off_Form content
     ) =
   row([],[cell([],content)]).
   
   
public define HTML_Off_Form   
   private_download
     (
       String        abs_path, 
       String        name, 
       String        extra_ext
     ) =
   private_download(abs_path,name,extra_ext,failure). 
   
   
public define HTML_Off_Form   
   private_download
     (
       String        abs_path, 
       String        name, 
       String        extra_ext,
       String        action_name,
       List((String,String))   args
     ) =
   private_download(abs_path,name,extra_ext,success((action_name,args))). 
   
public define HTML_Off_Form
   text
     (
       List(Text_Option) lto,
       Word32                i
     ) =
   text(lto,to_decimal(i)). 
   
   
public define HTML_Off_Form
   text
     (
       Word32                i
     ) =
   text([],i). 
   
public define HTML_Off_Form
   text
     (
       String s
     ) =
   text([],s).
   
   - Cell a gap between two other cells : 
   
public define HTML_Cell(HTML_Off_Form)
   width_gap
     (
       Word32 w
     ) =
   cell([width(w)],text([],"")).
   
public define HTML_Cell(HTML_In_Form)
   width_gap
     (
       Word32 w
     ) =
   cell([width(w)],text([],"")).
   
   
   - Row a gap between two other rows : 
   
public define HTML_Row(HTML_Off_Form)
   height_gap
     (
       Word32 h
     ) =
   row([],[cell([height(h)],text([],""))]). 
   
public define HTML_Row(HTML_In_Form)
   height_gap
     (
       Word32 h
     ) =
   row([],[cell([height(h)],text([],""))]). 
   
   
   Notice that the two types have alternatives in common (same name, same arguments types,
   up to  the value of the  parameter $T), which correspond  to elements which  may be put
   anywhere in the page.
   
   
   
public type CSS_Style:
   text_options(List(Text_Option)). 

public type CSS_File:
   css_file(String file_name).
   
define String
   format
     (
       List(Text_Option) l
     ).    
   
   
   
   
define Printable_tree
   format_css_styles
     (
       List(CSS_Style)   l
     ) =
   if l is 
     {
       [ ] then [ ], 
       [h . t] then 
         [ if h is 
             {
               text_options(tos) then 
                 ["   body, span, p { ", format(tos), " }\n" ]
             }
         . format_css_styles(t)]
     }.
   
   
   
public type HTML_Meta:
   keywords      (List(String)), 
   refresh       (Actioner_Connection connection, 
                  Actioner_Target target,
                  String action_name, 
                  Word32 delay), // in seconds
   meta          (String name, String content),
   http_equiv    (String name, String content),
   generic_meta  (List((String,String))),
   literal       (String). 
   
   Meta tags are put in the 'head' of the HTML page. 
   
   
public type Body_Option:
   background_color   (RGB),
   background_image   (String url),
   background_image   (String url, List(BackgroundOption)).
   
public type HTML_Body:  
   body(List(Body_Option) options, HTML_Off_Form content).        
   
   
public type HTML_Page:
   html_page(String title, 
             List(HTML_Meta) meta_tags, 
             List(CSS_Style) styles,
             List(CSS_File)  css_files,
             HTML_Body body). 
   
public define HTML_Page
   html_page
     (
       String             title, 
       List(HTML_Meta)    metas,
       HTML_Body          body
     ) =
   html_page(title,metas,[],[],body).
   
public define HTML_Page
   html_page
     (
       String             title, 
       List(HTML_Meta)    metas,
       List(CSS_Style)    styles,
       HTML_Body          body
     ) =
   html_page(title, metas, styles, [], body).
   
   'HTML_Page' represents the final product of the construction of a web page. 
   
   
   
   
      *** (3.2) ``in form'' versus ``off form''. 
   
   There is a variety of HTML  elements: texts, buttons, links, forms, inputs, etc... Some
   of them may have  a content, which is yet another HTML  element (or several). Hence, it
   is meaningful to say that an element  is 'within' another one. Now, putting any element
   within any  other one may be  meaningless.  For example,  an input element must  be put
   within a form (otherwise, it is useless), and a from within another form has no precise
   meaning (and is forbidden by the HTML specification).
   
   Actually, the  main criterium  is ``within  a form or  not within  a form''.   So, HTML
   elements in  a given page  are separated  into two categories:  those who are  within a
   form,  and the  others.  Nevertheless,  there  are elements  which may  belong to  both
   categories, like images and texts. We want to make use of the strong typing mecanism of
   Anubis in order to forbid non meaningful placement of elements.
   
   The  type  'HTML_In_Form'  defines  elements   to  be  put  within  forms.   Similarly,
   'HTML_Off_Form' defines elements not to be  put within forms. Both types are recursive,
   and 'HTML_Off_Form' refers  to 'HTML_In_Form' (via the 'form'  alternative, of course),
   but the two types  are not cross recursive. This is the reason  why it is impossible to
   put a  form within a form.  In order to construct  a web page, you  essentially have to
   produce a datum of type 'HTML_Off_Form' (maybe containing data of type 'HTML_In_Form').
   
   In practice, you  don't have to worry  so much about these two  types, because elements
   which may  be put anywhere are  constructed for both  types by functions with  the same
   name and the same arguments. Hence, for  both types, you just write the same thing. You
   are warned by  the compiler only when  you try to put an  element at a place  it is not
   allowed.

   
   
      *** (3.3) Defining your own style. 
   
   We provide generic tools for constructing  HTML elements.  However, your web site needs
   to have a ``style''.
   
   To that  end, you need to  write down a set  of ``styling functions'',  using the tools
   defined here. These  styling functions allow the introduction of  your colors and other
   visual characteristics into the constructed elements once and for all. For example, you
   may want all your texts to be rendered  in the ``Helvetica'' font, in size 14 and using
   some 'text_color'. You may write something like this:
   
   define RGB    text_color = rgb(10,40,40). 
   
   define HTML_Off_Form
     text
       (
         String the_text
       ) =
     text([font("helvetica"),size(14),color(text_color)],
          the_text).
   
   (and the same one for type 'HTML_In_Form') so that in order to put a piece of text in a
   page, you just write:
   
                 text("... some text ...")
   
   and you don't have  to provide the font, size and color for each  text.  If you want to
   have  several  styles  of text  presentation,  you  just  write  several sets  of  such
   convenience functions. This also suggests a  trick.  You may want for example different
   colors for 'in form' texts and 'off form' texts.  This may be achieved automatically by
   defining  two functions  as  above, with  the same  name  and same  argument type,  but
   returning either a 'HTML_In_Form' or a 'HTML_Off_Form'.
   
   If this  preliminary work is  well done, you  will not waste  your time later  when you
   concentrate on the actual informational content of your pages.
   
   This general  principle should be applied  to all sorts  of elements. This is  the best
   thing to  do in  order to  separate the functions  defining the  visual style  from the
   functions defining the informational content itself, so that changing the style without
   changing the  content becomes easy. This  is also the best  way for having  a clean and
   easily readable source for your web site.
   
   
   
   
   
      *** (3.4) Actioners and forms. 
  
   We have gathered several  notions from HTML into that of an  'actioner'. An actioner is
   an HTML element which opens a connection to our server when clicked upon. Actioners may
   have different  visual aspects.   They may  look like hypertext  links or  like buttons
   (rollovers),  or even  like  selectors (with  immediate  action).  In  any case,  their
   behavior is  the same: they  open a connection  to our server, and  send a set  of 'web
   arguments', i.e.  pairs  'name=value'.  Among these web arguments,  one of them denotes
   the action to  be performed, and the  others should be considered as  operands for this
   action.  Actually, the precise behavior of the actioner has several variants.
    
   The connection with  the server may be  secured (HTTPS) or non secured  (HTTP). See the
   type 'Actioner_Connection' above.
   
   You must also choose where the answer must be rendered.  This may be in the same window
   or in another window  (or frame).  If it is in another window,  the name of that window
   must be given. If  the window does not exist, the browser  will create it.  Optionally,
   you may give the  dimensions of the new window and other  characteristics. See the type
   'Actioner_Target' above.
   
   The actioner  also has a visual aspect.  See the type 'Actioner_Aspect'  above.  In the
   case  of a rollover  button, you  provide the  URLs of  two images  (of the  same size)
   representing the button:
   
      url_off:    to be used when the mouse is not over the button,
      url_on:     to be used when the mouse is over the button. 

   You  can also  create rollover  buttons without  creating images.  Just use  the second
   alternative named 'button'. The server creates the images automatically. 
   
   The purpose of forms is just to  give operands to actioners.  If the actioner is placed
   within a form,  all the input elements  which are within this form  provide operands to
   the actioner (except sometimes  when they are not set by the client).  If it is not put
   within a form, the actioner gets no operand, except if the name of a form is explicitly
   given,  in   which  case  the  actioner  gets   all  the  inputs  from   that  form  as
   operands. Furthermore, you may want to give extra operands to the actioner. This may be
   useful for separating  families of actioners with the same  action name. Extra operands
   'name=value' must be given in the form of pairs '(name,value)'.
            
   Notice that the name of a form may be  used by an actioner which is off the form, so as
   to get the operands provided by this form. Also notice that several actioners may refer
   to  the same  form,  being either  in  the form,  or  referring to  the  form from  the
   outside. These actioners  simply get the same set of operands,  even if they correspond
   to distinct actions.

   Input elements may be put only within a form. 

   
   
      *** (3.5) Local popup. 
   
   This element looks like a link or  a rollover button. When this button is clicked upon,
   a 'popup window' appears.  Actually, this popup window is just a layer in the same HTML
   page,  which becomes suddenly  visible.  It  is realized  with a  '<div>' HTML  tag. In
   particular, clicking  on the button does  not open any  connection.  This is why  it is
   called 'local'. The arguments have the following roles:
   
      Actioner_Aspect         aspect of the button (same semantics as for actioners)
      content                 content of the popup window
      x, y,                   position of the popup window on the HTML page (not relative 
                                to the button but to the page itself)
      title                   title of the popup window
      color                   color of the title bar and close button in the popup window.
                                A lightened version of this color is used for the background
                                of the popup window.
      width                   width of the title bar
   
   
   
   
   
   --- That's all for the public part ! --------------------------------------------------

   
   
   
   
   ----------------------------------- Table of Contents ---------------------------------

   *** [1] States. 
      *** [1.1] Saving and retrieving states. 
      *** [1.2] Deleting out of date states.
   
   *** [2] Tools. 
      *** [2.1] Directories. 
      *** [2.2] Secondary documents. 
   
   *** [3] Managing web arguments. 
      *** [3.1] Prefixing web arguments names.
      *** [3.2] Separating web arguments. 
      *** [3.3] Applying an action. 
   
   *** [4] Web site descriptions and the 'awp handlers'. 
      *** [4.1] The type 'Web_Site'.
      *** [4.2] Making a web site description. 
      *** [4.3] Starting the servers. 
   
   *** [5] HTML Formating. 
      *** [5.1] The type 'HTML_Any($T)'. 
      *** [5.2] Formating a color. 
      *** [5.3] Creating buttons. 
      *** [5.4] Formating an actioner. 
      *** [5.5] Formating a private download link. 
      *** [5.6] Formating rows and cells in a table. 
      *** [5.7] Formating elements which may be put anywhere. 
      *** [5.8] Formating 'in form' elements. 
      *** [5.9] Formating 'off form' elements. 
      *** [5.10] Formating meta-tags. 
   
   ---------------------------------------------------------------------------------------   
   

   
   
public define String
   doctype_w3c_header
     =
   "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\" "+
      "\"http://www.w3c.org/TR/html4/loose.dtd\">\n".
   

   
   
   *** [1] States. 
   
   We have  to define functions for  saving a state,  retrieving a state, deleting  out of
   date  states. We  need one  such function  per web  site. The  types of  the  first two
   functions depend on  the parameter $State. This is  not the case of the  third one. The
   fact that the  instance of $State is variable  from one web sites to  the other implies
   rather subtle manipulations using full functionality.
   
   
      *** [1.1] Saving and retrieving states. 
   
   Each state is saved  into a file on the server's disk  (in the directory represented by
   the symbol 'state_directory',  which is 'my_anubis/web_sites/common_name/states').  The
   state is saved together with a time stamp whose value is obtained by adding the current
   time to  the given timeout for  states. The state  receives a name obtained  by hashing
   (using  sha1)  the  content of  the  file  itself,  and  then  encoding the  hash  with
   'web_arg_encode'.  The  name  of  the  file  into  which the  state  is  saved  is  the
   concatenation of "s" and the name of the state.
      
read web/web_arg_encode.anubis   
   
   The tool below  constructs the function which is  able to save a state  on the server's
   disk.
   
define (Maybe($State) s) -> String    // the function constructed returns the name of the state
   make_save_state_function
     (
       Int     timeout, 
       String    state_directory
     ) =
   (Maybe($State) mbs) |->
   if mbs is 
   {
      failure then "",
      success(s) then
         with    time_stamp = now+timeout, 
                to_be_saved = (time_stamp,s), 
                 state_name = web_arg_encode(sha1(s)), 
           if save(to_be_saved,state_directory+"/s"+state_name) is ok 
           then state_name
           else (print("Cannot create state file in '"+state_directory+"'.\n"); "")
   }.
   
   
   When  a request  arrives, we  need to  retrieve the  previous state  from  the server's
   disk. We receive the name of that state. If the state is out of date, the state file is
   kept 3 days, and then deleted. 
   
type PreviousState($State):
   not_found,                    // cannot retrieve the previous state
   out_of_date($State),          // the previous state is out of date
   still_valid($State).          // the previous state is still valid
   
define (String state_name) -> PreviousState($State)
   make_retrieve_state_function
     (
       String   state_directory
     ) =
   (String state_name) |->
   with file_path = state_directory+"/s"+state_name,
   if (RetrieveResult((Int,$State)))retrieve(file_path) is ok(d)
   then (
        if d is (time_stamp,s) then 
        if time_stamp < now
        then (
             forget(remove(file_path));
             out_of_date(s)
             )
        else still_valid(s)        // state has been successfully retrieved
        )
   else not_found. 
   
   
   
      *** [1.2] Deleting out of date states.
   
   We also need to delete states which are  out of date and which will never be deleted by
   the above method. This may be performed  by a machine doing this periodically (say once
   per states life time period).
   
define (List(String) file_names) -> One
   make_delete_out_of_date_states_function
     (
       Maybe($State)    dummy, 
       String           state_directory
     ) =
   (List(String) file_names) |-df->
   if file_names is 
     {
       [ ] then unique, 
       [h . t] then
         with file_path = state_directory+"/"+h, 
         if (RetrieveResult((Int,$State)))retrieve(file_path) is ok(d)
         then (
                if d is (time_stamp,data) then
                if time_stamp < now
                then (forget(remove(file_path)); df(t))
                else df(t)
              )
         else (forget(remove(file_path)); df(t))
     }. 
   
   
   The 'labelled arrow' |-df-> is documented in 'documentation/en/anubis_doc.txt'.

   Note: The  argument 'dummy'  (of type  Maybe($State)) is not  used in  the body  of the
   function (hence  its name). Nevertheless, it  is required. Indeed,  the Anubis compiler
   does not accept a  parameter in the body of a function  (here the parameter is required
   by  the use  of 'retrieve')  if  this parameter  does not  appear  in the  type of  the
   function. This  is because this  would create ambiguities  that no explicit  typing may
   ever resolve. If you  put a double slash in front of  the declaration of 'dummy' above,
   and if you compile this file, you will get a message like this one:
     
    Error in 'making_a_web_site.anubis', line 1300, column 7:
       A definition may not contain parameters which are not present
       in the declaration part (hidden parameters):
          $State 
   
   The type  of the  function constructed by  'make_delete_out_of_date_states_function' is
   independant of  the parameter $State. This  is important because this  allows to create
   the list of  such functions for all web  sites. From this list, it is  possible to call
   the  functions one  after  the  other, so  deleting  out of  date  states  for all  web
   sites. Actually, the next function receives a list of pairs (state_directory,function),
   one for each web site.
   
define One
   delete_out_of_date_states          // for all web sites
     (
       List((String, List(String) -> One))   directories_and_functions     
     ) =
   if directories_and_functions is 
     {
       [ ] then unique, 
       [h . t] then if h is (state_directory,function) then
         function(directory_list(state_directory,"s*")); 
         delete_out_of_date_states(t)
     }. 
      

   The  above function  must be  called periodically  in a  separate virtual  machine. The
   period we have choosen  is (rather logically) the life time of  states itself. This may
   be achieved by an 'infinite' loop, using a 'sleep(timeout)'. However, the loop must not
   be really  infinite, because the  servers may be  shutdown.  Hence, our loop  must test
   (rather frequently; say  every second) if the  servers are down. If they  are, the loop
   must be exited.
   
define One
   delete_states_loop
     (
       List((String,List(String) -> One))    directories_and_functions,
       Int                                   timeout, 
       Int                                   next_time, 
       Server                                http_server,
       Server                                https_server,
       Var(Bool)                             shutdown_required
     ) =
   if *shutdown_required 
     then (shutdown(http_server); shutdown(https_server))
     else unique; 
   if (is_down(http_server) & is_down(https_server))
   then unique
   else if now > next_time
        then
          (
            delete_out_of_date_states(directories_and_functions); 
            delete_states_loop(directories_and_functions,
                               timeout, 
                               now+timeout,
                               http_server,
                               https_server,
                               shutdown_required)
          )
        else
          (
            sleep(1000);    // sleep just one second and try again
            delete_states_loop(directories_and_functions,
                               timeout, 
                               next_time,
                               http_server,
                               https_server,
                               shutdown_required)
          ). 
   
   The above loop must be run in a  separate virtual machine. This will be done just after
   the two servers are started.
     
   

   
   
   *** [2] Tools. 
   
      *** [2.1] Directories. 
   
   We need a tool for creating directories (if needed). 

   (This tool has been moved to 'tools/basis.anubis'). 
   
   
   
   
      *** [2.2] Secondary documents. 
   
   Some HTML elements  (like '<object>', '<frame>') cannot receive  their content directly
   from the current  document, but only through  an URL.  For this reason,  we implement a
   mecanism for creating secondary  documents on the fly. To that end  we use the 'private
   download' mecanism. 
   
   A secondary document is formated by the same functions as the main document itself. The
   next function  takes an 'off form'  element, creates the file  containing the secondary
   document in HTML format, and returns the URL at which the document will be available.
   
define String
   create_secondary_document
     (
       String                  sd,            // site directory
       String                  as,            // authorization_secret
       String                  sn,            // state name
       $T -> Printable_tree    format_element, 
       $T                      content,
       HTML_Size               width
     ) =
   with  private_download_directory = sd+"/private_download", 
                               hash = web_arg_encode(sha1(content)), 
                       file_content = (Printable_tree)
         [doctype_w3c_header, 
          "<html><body><table ",
                  if width is 
                    {
                      absolute(w)      then ["width=\"",w-25],
                      percentage(w)    then ["width=\"95\""]
                    },"\"><tr><td align=right>",
             format_element(content),
          "</td></tr></table></body></html>"
         ], 
                          file_name = "sd"+hash+".html", 
                          file_path = private_download_directory+"/"+file_name, 
      if write_to_file(file_path,file_content) is 
        {
          cannot_open_file then print("Cannot open file '"+file_path+"'.\n"); "", 
          write_error(n)   then print("Error writing file '"+file_path+"'.\n"); "", 
          ok               then file_name+"?zauth="+
                             make_authorization(sd,as,private_download_directory+"/"+file_name)
        }. 
     
   
   
   
      *** [2.3] Generating unique ids. 
   
   In order to uniquely name object for JavaScript we generate unique ids from a counter. 
   
define Word32
   new_idnum
     (
       Var(Word32)  ic_v      // 'idnum' counter variable
     ) =
   protect
   with result = *ic_v+1, 
   ic_v <- result; 
   result. 
   

   
   
   
   
   *** [3] Managing web arguments. 
   
   Web  arguments are  those pairs  'name=value' which  are transmitted  through  the HTTP
   protocol. We need precise naming conventions for these web arguments. 
   
   
   
      *** [3.1] Prefixing web arguments names.
   
   We want  to assign different  roles to web  arguments, and we also  want to be  able to
   recognize its role directly from the name  of a web argument.  The name "s" is reserved
   for the  web argument whose  value is the  name of the current  state. The name  "a" is
   reserved  for  the  web  argument  whose  value  is  the  name  of  the  action  to  be
   performed. Other web arguments receive arbitrary  names, and in order to avoid clashes,
   these names are prefixed by:
   
      "p" for names of password inputs,
      "o" for other web arguments

   The reason why password input names have a distinct prefix is that this allows the HTTP
   server to hide the passwords on the console of the server and in the journal.
   

     
   
      *** [3.2] Separating web arguments. 
   
   When a new request arrives, we need to separate the web arguments, that is to say:
   
     - find the value of "s", and recover the corresponding state, 
     - find the value of "a", which is the name of the action to be performed, 
     - get the list of all the remaining web arguments (operands of the action). 

   We must also  determine if the previous state  may be recovered. If it is  not the case
   (either because  the previous state name  is invalid, or  the previous state is  out of
   date), we must check if there is an action name. Indeed, the presence of an action name
   indicates that the user has clicked on one  of our buttons or links. If on the contrary
   there is no action  name the user has just entered our address  in his browser. In this
   last case, we must send the first page of our site (maybe a 'login' page), but if there
   is an action, we  must send a page just saying that the  session ticket has expired. If
   the previous state  is recovered and there is  no action, the new state is  the same as
   the previous state.
   
   The result of the separation of the web arguments is of type:
   
type Separated_Web_Args($State):   
   swa(Maybe(PreviousState($State))     previous_state,
       Maybe(String)                    action_name,
       List(Web_arg)                    operands). 
   
   
   
   The next function constructs the function which separates the web arguments. 
   
define (List(Web_arg) lwa) -> Separated_Web_Args($State)
   make_separate_web_args_function
     (
       String                             state_directory, 
       String -> PreviousState($State)    retrieve_state
     ) =
   (List(Web_arg) lwa) |-swaf->
   if lwa is 
     {
       [ ] then 
         //
         // no web arg found => no previous state and no action
         //
         swa(failure,failure,[]), 
   
       [wa_1 . wa_others] then 
         //
         // at least one web arg =>
         //    separate other web args, and insert the first one as needed
         //
         if (Separated_Web_Args($State))swaf(wa_others) is 
           {
              swa(ps1, // possible previous state
                  an1, // maybe an action name
                  op1) // operands so far
              then 
         if wa_1 is
           {
             web_arg(n,v)    then 
               with prefix = if substr(n,0,4) = "amp;" then substr(n,4,1) else substr(n,0,1), 
                name_start = (Int)(if substr(n,0,4) = "amp;" then 5 else 1), 
                 if prefix = "s" then 
                   swa(success(retrieve_state(v)),an1,op1) else
                 if prefix = "a" then 
                   swa(ps1,success(v),op1) else
                 if prefix = "t" then 
                   swa(ps1,an1,[web_arg("target",v) . op1]) else
                 if prefix = "p" then 
                   swa(ps1,an1,[web_arg(substr(n,name_start,length(n)-name_start),v) . op1]) else
                 if prefix = "o" then 
                   swa(ps1,an1,[web_arg(substr(n,name_start,length(n)-name_start),v) . op1]) else
                 swa(ps1,an1,op1), 
   
             upload(n,v,t)   then 
               swa(ps1,an1,[upload(substr(n,1,length(n)-1),v,t) . op1])
           }}
     }.


   

   
   
      *** [3.3] Applying an action. 
   
   When the web arguments are separated (and their names cleaned up from prefixes), we may
   apply the action to the operands and the  current state. We search for the action to be
   applied in the  list of actions.  If no action  is found, the new state  is the same as
   the previous  state. Also, we  deny the  application of an  HTTP action if  the request
   arrives through the HTTPS channel and conversely.

   
define (Maybe($State)     previous,
        String            action_name, 
        HTTP_Info         http_info, 
        List(Web_arg)     lwa,
        Bool              is_https) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))
   make_apply_action_function
     (
       List(Web_Action($SessionTicket, $State))   actions
     ) = 
   with f = 
     (Maybe($State)                previous, 
      String                       action_name, 
      HTTP_Info                    http_info, 
      List(Web_arg)                lwa,
      Bool                         is_https,
      List(Web_Action($SessionTicket, $State))     actions1) |-f->   
        if actions1 is 
          {
            [ ] then (print("action '"+action_name+
                              "' not found.\n"); (failure, previous, [])),
            [ac1 . others] then if ac1 is 
              {
                http_action(an,allow,do_it) then 
                  if an = action_name 
                  then if is_https
                       then (print("HTTP action '"+an+
                                     "' called through HTTPS (denied).\n"); 
                             (failure, previous, []))
                       else if allow(previous) 
                            then do_it(http_info,lwa,previous)
                            else (failure, previous, [])
                  else f(previous,action_name,http_info,lwa,is_https,others),
   
                https_action(an,allow,do_it) then
                  if an = action_name 
                  then if is_https
                       then if allow(previous)
                            then do_it(http_info,lwa,previous)
                            else (failure, previous, [])
                       else (print("HTTPS action '"+an+
                                     "' called through HTTP (denied).\n"); 
                             (failure, previous, []))
                  else f(previous,action_name,http_info,lwa,is_https,others),
   
                http_https_action(an,allow,do_it) then
                  if an = action_name 
                  then if allow(previous)
                            then do_it(http_info,lwa,previous)
                            else (failure, previous, [])
                  else f(previous,action_name,http_info,lwa,is_https,others),
   
              }
          },
   (Maybe($State)                previous, 
    String                       action_name, 
    HTTP_Info                    http_info, 
    List(Web_arg)                lwa,
    Bool                         is_https) |->
      f(previous,action_name,http_info,lwa,is_https,actions). 
   
   
     

   
   
   
   
   *** [4] Web site descriptions and the 'awp handlers'. 
   
      *** [4.1] The type 'Web_Site'.
   
   The  type 'Web_Site_Description' is  defined in  'web/multihost_http_server.anubis'. We
   need another one, because, we have some extra informations to record for each site.
   
public type Web_Site:
   web_site((Word32,Word32) -> Web_Site_Description     description,
            List(String) -> One                       delete_out_of_date). 
   
   
   
   
      *** [4.2] Making a web site description. 
   
   Below  is the  function which  creates a  web site  description. It  first  creates (if
   needed) the directories for the site,  then constructs the tool functions for the site,
   and the site handler. Finally, it constructs the web site description. 
   
   We gather common (constant) informations in the following type:
   
type CommonInfo:
   info(String      common_name, 
        Word32       http_port, 
        Word32       https_port, 
        String      site_directory, 
        String      authorization_secret, 
        SystemFont  font). 
   
   We need a forward declaration. 
   
public define Printable_tree 
   format
     (
       CommonInfo   cinfo, 
       String       state_name, 
       HTML_Page    page,
       Bool         is_https,
       String       charset
     ).    
   
   
define Printable_tree
   format
     (
       HTML_Size   s
     ) = 
   if s is 
     {
       absolute(x)     then ["\"",x,"\""],
       percentage(x)   then ["\"",x,"\""]
     }. 
   
define Printable_tree
   top_redirection_page
     (
       String common_name
     ) =
   [ doctype_w3c_header, 
     "<html><head>",
       "<meta http-equiv=\"Refresh\" content=\"0; URL=javascript:void window.open('/?t=','_top')\">",
     "</head><body></body></html>"
   ].  
   
public define Web_Site
   make_web_site_description
     (
       List(String)                common_names,   // for example: ["www.our-business.com"]
       String                      site_directory,     
       One -> One         init, 
       (HTTP_Info) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))        initial_state, 
       ($State expired,
        HTTP_Info,
        List(Web_arg),
        Bool is_https) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))    ticket_expired_state, 
       (HTTP_Info,
        List(Web_arg),
        Bool is_https) -> (Maybe($SessionTicket), Maybe($State), List(HTTP_header))    ticket_lost_state, 
       List(Web_Action($SessionTicket, $State))    actions, 
       (Maybe($SessionTicket), Maybe($State)) -> HTML_Page                             compute_page,
       Int                         timeout,
       List(Redirection)           redirections, 
       String                      charset, 
       List(String)                journal_extensions,
       List(String)                journal_headers, 
       String                      secret, 
       List(MIME)                  known_mime_types,
       (String action_name,
        List(Web_arg) args) -> One before_send_file
     ) = 
      init(unique); 
      //
      // load a font for button texts
      //
      if load_system_font("times_medium_r_12") is 
        {
          failure then print("Cannot load system font 'times_medium_r_12'.\n"); alert, 
          success(font) then with
      //
      // make required directories (if needed)
      //
              web_sites_directory = (String)make_directory(my_anubis_directory+"/web_sites"),  
                   base_directory = (String)make_directory(site_directory),
                  state_directory = (String)make_directory(site_directory+"/states"), 
      //
      // manage automatic buttons: create the buttons directory (if needed), 
      // and remove old buttons.
      //
                buttons_directory = site_directory+"/public/buttons",
                             forget((String)make_directory(site_directory+"/public")); 
                             forget((String)make_directory(buttons_directory)); 
      with button_files = directory_list(buttons_directory,"b*.jpg"), 
        forget(map((String fn) |-> remove(buttons_directory+"/"+fn), button_files));
      //
      // construct tool functions
      //
      with
        save_state = make_save_state_function(timeout,state_directory),    
    retrieve_state = make_retrieve_state_function(state_directory), 
 separate_web_args = make_separate_web_args_function(state_directory,retrieve_state), 
      apply_action = make_apply_action_function(actions),
      //
      // construct the site handler
      //
      site_handler = (Word32 http_port, Word32 https_port) |-> 
                     ((String         host_name, 
                       HTTP_Info      http_info, 
                       List(Web_arg)  lwa,
                       Bool           is_https) |-> 
        ((List(HTTP_header),Printable_tree))
        if separate_web_args(lwa) is 
          {
            swa(mb_previous_state,mb_action_name,operands) then 
              with state_and_headers = if mb_previous_state is
                {
                  failure then 
                      if mb_action_name is 
                        {
                          failure              then initial_state(http_info),
                          success(action_name) then 
                             apply_action(failure,action_name,http_info,operands,is_https)   
                        },
                  success(previous_state) then if previous_state is
                  {
                    not_found               then 
                      if mb_action_name is 
                        {
                          failure then initial_state(http_info), 
                          success(_) then 
                            ticket_lost_state(http_info,lwa,is_https)
                        }, 
     
                    out_of_date(state)      then 
                      ticket_expired_state(state,http_info,lwa,is_https), 
     
                    still_valid(state)      then 
                      if mb_action_name is 
                        {
                          failure              then (failure, success(state), []),
                          success(action_name) then 
                             apply_action(success(state),action_name,http_info,operands,is_https)   
                        }
                  }
                },
              if state_and_headers is (session_ticket, mb_new_state, headers) then
                with state_name = save_state(mb_new_state),
                (headers,
                 format(info(host_name,http_port,https_port,site_directory,secret,font),
                        state_name,compute_page(session_ticket, mb_new_state),is_https,charset))
          }),
       //
       // make the delete_out_of_date function
       //
       delete_out_of_date = 
         make_delete_out_of_date_states_function((Maybe($State))failure,
                                                 site_directory+"/states"), 
       //
       // construct the web site description
       //
    web_site((Word32 http_port, Word32 https_port) |-> 
             web_site_description(common_names,
                                  site_directory,
                                  redirections,
                                  charset, 
                                  journal_extensions, 
                                  journal_headers,
                                  secret,
                                  known_mime_types,
                                  site_handler(http_port,https_port),
                                  (List(Web_arg) lwa) |-> if separate_web_args(lwa) is 
                                     swa(mb_previous_state,mb_action_name,operands) then 
                                       if mb_action_name is 
                                         {
                                           failure     then unique
                                           success(an) then before_send_file(an,operands)
                                         }),
             delete_out_of_date)}. 
   
      
   
   
      *** [4.3] Starting the servers. 
   
public define Start_Web_Sites_Result
   start_web_sites
     (
       Word32                        ip_address,    // the IP address shared by the web sites
       Word32                        http_port,     // usually: 80
       Word32                        https_port,    // usually: 443
       String                       ssl_certificate_common_name, 
       List(Web_Site)               web_sites,     // web sites to be started
       Var(Bool)                    shutdown_required
     ) =
   with get_description = (Web_Site ws) |-> description(ws)(http_port,https_port), 
   with http_server_r  = 
     start_http_server(ip_address,http_port, 
                       map(get_description,web_sites),
                       load_denial_of_service_info), 
   with https_server_r = 
     start_https_server(ip_address,https_port,
                        ssl_certificate_common_name,
                        map(get_description,web_sites),
                        load_denial_of_service_info), 
   if http_server_r is ok(http_server)
   then
     (
     if https_server_r is ok(https_server)
     then 
       (
         start_http_servers_tasks(map(get_description,web_sites),
                                  [http_server,https_server],
                                  600);  // period of 10 minutes
         delegate
         delete_states_loop(
                            map((Web_Site ws) |-> 
                                  (site_directory(description(ws)(http_port,https_port))+
                                     "/states",delete_out_of_date(ws)),
                                web_sites),
                            3600*24*3,   // keep out of date states 3 days
                            now,
                            http_server,
                            https_server,
                            shutdown_required),
         ok(http_server,https_server)
       )
     else cannot_bind_to_port(https_port)
     )
   else
     (
     if https_server_r is ok(https_server)
     then cannot_bind_to_port(http_port) 
     else cannot_bind_to_port(http_port,https_port)
     ). 
   
   
public define One
   start_web_sites
     (
       Word32                        ip_address,    // the IP address shared by the web sites
       Word32                        http_port,     // usually: 80
       Word32                        https_port,    // usually: 443
       String                       ssl_certificate_common_name, 
       List(Web_Site)               web_sites,     // web sites to be started
       Var(Bool)                    shutdown_required
     ) =
   if (Start_Web_Sites_Result)start_web_sites(ip_address,
                                              http_port,
                                              https_port,
                                              ssl_certificate_common_name,
                                              web_sites,
                                              shutdown_required) is 
     {
       cannot_bind_to_port(n)     then print("Cannot bind to port: "+n+"\n"),
       cannot_bind_to_port(n,m)   then print("Cannot bind to ports: "+n+", "+m+"\n"),
       ok(s1,s2)                  then print("Servers started.\n")
     }.
   
   
   
   
   
   *** [5] HTML Formating. 

   We need to translate HTML elements as defined above into actual HTML text.
   
   Actioners require  special informations, which must  be transmitted when  needed by the
   'format' functions:
   
       - the 'common name', which is used for URLs, 
       - the HTTP/HTTPS port number, 
       - the 'state name', which must be transmitted when the actioner is clicked upon, 
       - the 'form name' (if any) to which the actioner refers. 
   
   If the  actioner is off  form, and if  it refers to  a form, the  name of that  form is
   already known by the actioner. On the contrary, if the actioner is 'in form', it refers
   implicitly to  the form containing  it.  The  name of that  form is transmitted  to the
   'format' functions called from within the formating of that form.

   
   
   
      *** [5.1] The type 'HTML_Any($T)'. 
   
   The type  'HTML_Any($T)' gathers elements  which may be  put anywhere in the  page. The
   parameter $T becomes either 'HTML_Off_Form' or 'HTML_In_Form'.
   
type HTML_Any($T):
   any_text               (List(Text_Option), String the_text), 
   any_preformated        (List(Text_Option), String), 
   any_paragraph          (List(Text_Option), String the_text), 
   any_image              (String url), 
   any_image              (String url, Word32 width, Word32 height), 
   any_table              (List(Table_Option), List(HTML_Row($T))), 
   any_center             ($T), 
   any_mail_to            (String email, $T element), 
   any_scroller           (Word32 width, Word32 height, 
                           Word32 content_width, Word32 content_height,
                           $T content), 
   any_fixed_size         (HTML_Size width, HTML_Size height, $T content), 
   any_fixed_size_2       (HTML_Size width, HTML_Size height, String name_of_HTML_file), 
   any_actioner           (Actioner_Connection, 
                           Actioner_Target, 
                           Actioner_Aspect, 
                           String action_name, 
                           List((String,String)) extra_ops, 
                           List(Actioner_Local_Action),
                           Maybe(String) form_name),
   any_local_popup        (Actioner_Aspect, $T content, Word32 x, Word32 y, 
                           String title, RGB color, Word32 width), 
   any_foreign_link       (List(Text_Option), String url, String name), 
   any_private_download   (String abs_path, String name, String extra_ext,
                             Maybe((String,List((String,String))))),
   any_div                (List(DIV_Option), $T element),
   any_div_empty          (List(DIV_Option)).
   
      
   
      *** [5.2] Formating a color. 
   
   RGB  colors  are  formatted as  '#rrggbb'  where  rr,  gg  and  bb are  two  characters
   hexadecimal values.
   
define String
  html_format
    (
      RGB color
    ) =
  if color is rgb(r,g,b) then 
     "#" + to_hexa(r) + to_hexa(g) + to_hexa(b).  

   
   The following is a very arbitrary definition  of the opposite color. The thing which is
   important is that it  is far from the original, so that  characters in 'opposite' color
   are clearly visible over the original.
      
define RGB
   opposite
     (
       RGB color
     ) =
   if color is rgb(r,g,b) then rgb(255-r,255-g,255-b).

   with r1 = word8_to_Word32(r), 
   with g1 = word8_to_Word32(g), 
   with b1 = word8_to_Word32(b), 
   rgb(truncate_to_word8(255-r1),
       truncate_to_word8(255-g1),
       truncate_to_word8(255-b1)).
   

   
   
      *** [5.3] Creating buttons. 
   
   We want  to be  able to  create buttons  in the form  of a  pair of  images (rollovers)
   automatically. We  use the  JPEG interface,  because for the  time being  Anubis cannot
   handle other kinds of images.
   
   
    Computing printed text length.
   
define Int
   printed_text_width
    (
     Word8 -> Int  char_size,
     List(Word8)     l
    ) =
   if l is 
    {
     []       then (Int) 0,
     [h . t]  then char_size(h) + 1+ printed_text_width(char_size,t)
    }.
   
define Int
   printed_text_width
    (
     SystemFont       font,
     String              s
    ) =
     printed_text_width((Word8 c) |-> to_Int([width(get_char_info(font,c))]),
                         explode(s)).
   
   
   
   Converting RGB to RGBA. 
   
define RGBA
   to_rgba
     (
       RGB color
     ) =
   if color is rgb(r,g,b) then rgba(r,g,b,255). 
   
   
   Drawing a 'relief'. 
   
define One
   draw_relief
     (
       RGBAImage   dest, 
       RGBA        color,
       Word16       contrast, 
       Int       x, 
       Int       y,
       Int       width, 
       Int       height
     ) =
   with l = lighten(color,contrast), 
        d = darken(color,contrast), 
   draw_rectangle(dest,rect(x,y,x+width,y+1),l); 
   draw_rectangle(dest,rect(x,y+1,x+1,y+height),l);
   draw_rectangle(dest,rect(x+width-1,y+1,x+width,y+height),d); 
   draw_rectangle(dest,rect(x+1,y+height-1,x+width-1,y+height),d). 
   
   
   Creating a button background. 
   
define RGBAImage
   create_button_background
     (
       RGBA     color, 
       Int      width,
       Int      height
     ) = 
   with result = create_rgba_image(width,height,color), 
   draw_relief(result,color,100,0,0,width,height); 
   draw_relief(result,color,70,1,1,width-2,height-2); 
   draw_relief(result,color,55,2,2,width-4,height-4); 
   draw_relief(result,color,35,3,3,width-6,height-6); 
   draw_relief(result,color,20,4,4,width-8,height-8); 
   draw_relief(result,color,10,5,5,width-10,height-10); 
   draw_relief(result,color,5,6,6,width-12,height-12); 
   result. 
   

   Drawing the text over the background. 
   
define One 
   draw_button_text
     (
       RGBAImage    image, 
       String       text, 
       Int          text_index,
       Int          pixel_x,
       Int          y,
       Rectangle    clip,
       RGBA         color,
       SystemFont   font,
     ) =
   if nth(text_index,text) is 
     {
       failure then unique, 
       success(c) then 
         with cw = draw_system_character(image,clip,
                     truncate_to_Word32(pixel_x),truncate_to_Word32(y),
                     font,to_Word32(c),color), 
         draw_button_text(image,text,text_index+1,pixel_x+to_Int[cw]+1,y,clip,color,font)
     }. 
   
define One
   draw_button_text
     (
       RGBAImage    image,
       String       text,
       Int          text_width,
       RGBA         light_color, 
       RGBA         dark_color,
       SystemFont   font
     ) = 
   with image_width = width(image), 
       image_height = height(image), 
              x_pos = (image_width-truncate_to_Word32(text_width))>>1,
               clip = rect(0,0,image_width,image_height),
    new_light_color = lighten(light_color,150),
     new_dark_color = darken(dark_color,40), 
   draw_button_text(image, text, 0, to_Int[x_pos]+2, 16, clip, new_dark_color, font);
   draw_button_text(image, text, 0, to_Int[x_pos], 14, clip, new_light_color, font).
   
   
   The next  function creates the two  images for a  button. The information given  is the
   main color of the  button, the text of the button and the  minimal width (in pixels) of
   the button. The  function does not create  the button if the images  already exist. The
   two images are stored in the directory 'site_directory/buttons'. The names of the files
   are of the form:
   
         bxxxx_off.jpg
         bxxxx_on.jpg
   
   where the  prefix 'b' is  to avoid  leading '-' which  may perturb UNIX  commands (like
   'rm'), and where 'xxxx' is created from the given informations by the formula:
   
      xxxx = web_arg_encode(sha1((color,text,width)))
   
   Hence, distinct informations give distinct file names. 
   
   
define String      // returns xxxx
   create_button_images
     (
       String       site_directory, 
       RGBA         color, 
       String       text,
       Int          width,
       SystemFont   font
     ) =
   with       xxxx = web_arg_encode(sha1((color,text,width))), 
       buttons_dir = site_directory+"/public/buttons", 
      off_filepath = buttons_dir+"/b"+xxxx+"_off.jpg",
       on_filepath = buttons_dir+"/b"+xxxx+"_on.jpg",
     if file_exists(on_filepath)
     then xxxx
     else with 
            text_width = printed_text_width(font,text),
          button_width = max(width,text_width+12), 
         button_height = (Int)20, 
           light_color = lighten(color,60), 
      very_light_color = lighten(light_color,30), 
            dark_color = darken(color,40), 
        background_off = 
          create_button_background(color,button_width,button_height),
         background_on = 
          create_button_background(light_color,button_width,button_height),
   
       draw_button_text(background_off,text,text_width,very_light_color,dark_color,font);
       draw_button_text(background_on, text,text_width,very_light_color,dark_color,font); 
       forget(write_image_to_JPEG_file(to_JPEG(background_off),
                                       off_filepath,
                                       100)); 
       forget(write_image_to_JPEG_file(to_JPEG(background_on),
                                       on_filepath,
                                       100));
       xxxx. 
   
   

   
   
      *** [5.4] Formating an actioner. 
   
   An actioner works as follows. Assume first that it refers to a form. When it is clicked
   upon, the actioner puts (via 'onMouseDown')  the URL into the 'action' attribute of the
   form, and submits  the form, using the JavaScript  command 'form_name.submit()'. If the
   actioner does  not refer to a  form, it fires the  URL directly via  'href', because in
   that case, the actioner is always an <a> tag. 
   
   The URL itself is composed using the  connection sort (same, http or https), the common
   name  and port  number (if  needed), the  state name,  the action  name, and  the extra
   operands, which are put into a query string. It may look like this:
   
      http://common_name:port/?s=state_name&a=action_name&oname=value...
         
   Each extra operand is a pair of strings: (name,value). It is formated as:
   
                 &oname=value
   
   
define String 
   format_extra_operands
     (
       List((String,String)) l
     ) =
   if l is 
     {
       [ ] then "", 
       [h . t] then if h is (n,v) then  
         "&amp;o"+n+"="+v+format_extra_operands(t)
     }.

   
   It seams that the standard requires "&amp;" instead of "&" ! 
   
   In case the target is another window, we need to format the options for this window. 
   
define String 
   format
     (
       List(Other_Window_Option) l 
     ) =
   if l is 
     {
       [ ] then "", 
       [h . t] then if h is 
         {
           resizable      then "resizable",
           scrollbars     then "scrollbars",
           width(w)       then "width="+w,
           height(h1)     then "height="+h1
         } + if t is [ ] then "" else (","+format(t))
     }. 
   
   
   
   Formating choices for a <select> tag. 
   
public define Printable_tree
  format_choices
    (
      List(String) l
    ) =
  if l is 
    {
      [ ] then [ ], 
      [h . t] then ["<option>",h . format_choices(t)]
    }.


public define Printable_tree
  format_choices
    (
      List(String) l,
      String selected
    ) =
  if l is 
    {
      [ ] then [ ], 
      [h . t] then if h = selected 
        then ["<option selected>",h . format_choices(t)]
        else ["<option>",h . format_choices(t,selected)]
    }.

   
public define Printable_tree
  format_choices
    (
      List((String,String)) l
    ) =
  if l is 
    {
      [ ] then [ ], 
      [h . t] then
       if h is (val,item) 
        then ["<option value=\""+val+"\">",item . format_choices(t)]
    }.
   
public define Printable_tree
  format_choices
    (
      List((String,String)) l,
      String                selected
    ) =
  if l is 
    {
      [ ] then [ ], 
      [h . t] then 
       if h is (val,item) then 
       if item = selected 
         then ["<option value=\""+val+"\" selected>",item . format_choices(t)]
         else ["<option value=\""+val+"\">",item . format_choices(t,selected)]
    }.
   
   
   
variable Word32 count = 0.
   
   Note: this  counter is private to  the virtual machine,  hence there is one  counter by
   client.
   
define Word32
   new_count
     = 
   count <- *count+1; 
   *count. 
   
   
   The next function composes  the URL. It is a JavaScript URL  when the target is another
   window.
   
define String 
   make_actioner_url
     (
       CommonInfo                cinfo, 
       Actioner_Connection       connection, 
       Actioner_Target           target, 
       String                    state_name, 
       String                    action_name, 
       List((String,String))     extra_ops,
       Bool                      is_https
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with strict_url = 
     if connection is 
       {
         same      then "/", 
         /*
         same      then if is_https
                        then "https://"+common_name+":"+https_port+"/" 
                        else "http://"+common_name+":"+https_port+"/", 
         */
   
         http      then "http://"+common_name+":"+http_port+"/", 
         https     then "https://"+common_name+":"+https_port+"/", 
       } +
     "?s=" + state_name + "&amp;a=" + action_name + 
     format_extra_operands(extra_ops), 
   if target is 
     {
       same            then strict_url,
       same(label)     then strict_url+"#"+label, 
       other(wn,ops)   then 
         "javascript:void window.open('"+strict_url+"&amp;t="+wn+"','"+
                "w"+to_ascii(sha1(wn))+"','"+format(ops)+"')"
     }. 
   
   
   Depending on the fact that the actioner refers to a form or not, the URL is used in two
   different ways.  If  the actioner does not refer  to a form, it is realized  by a '<a>'
   tag, with a 'href'  attribute. If it refers to a form, it is  still realized by a '<a>'
   tag, but with no href attribute.  In  this case, we use the 'onMouseDown' or 'onChange'
   event handler. The handler calls a JavaScript  function which puts the URL as the value
   of the 'action' attribute of the form, and submits the form.

type URL_or_JavaScript:
   url          (String),
   javascript   (Printable_tree script, Printable_tree handler). 
   
 define URL_or_JavaScript
   format_action
     (
       String                 the_url, 
       Maybe(String)          mb_form_name
     ) = 
   url(the_url).
   
   
   It was as  shown below, in order to  allow submission of a form from  outside the form,
   but this makes problems:
   
define URL_or_JavaScript
   format_action
     (
       String                 the_url, 
       Maybe(String)          mb_form_name
     ) = 
   if mb_form_name is 
     {
       failure then 
         url(the_url), 
       success(form_name) then 
       with n = new_count, 
         javascript(
           [
             "<script type=\"text/javascript\" language =\"JavaScript\">\n", 
               "function pfu",form_name,n,"() {\n",
               "  var fa = document.forms.f",form_name,";\n", 
               "  var u = \"",the_url,"\";\n",
               "  fa.action = u;\n",
               "  fa.submit(); }",
             "</script>"
           ],
           ["pfu",form_name,n,"();"]
         )
     }.
   
   
   
   Now, we format the actioner according to its aspect. 
   
define List(Text_Option)
  get_text_options
  (
     List(Text_Option) l
  )
  =
  if l is
  {
    []        then [],
    [h . t ]  then
      if h is css_class(class) then
        get_text_options(t)
      else
      [  h .  get_text_options(t) ]
   
  }.
  
  /**
   * Extract the CSS class list from the list of Text_Option
   */
define List(Text_Option)
  get_css_class
  (
     List(Text_Option) l
  )
  =
  if l is
  {
    []        then [],
    [h . t ]  then
    
    
      if h is css_class(class) then
       [ h .  get_css_class(t) ]
      else
        get_css_class(t)
  }
  .

define String 
  format_text_options
  (
     List(Text_Option) l
  )
  =
  with text_options = get_text_options(l),
        css_classes = get_css_class(l),
  if text_options is 
  {
    []    then "",
    [_ . _] then " style=\"" + format(text_options) + "\" "
  }
  +
  if css_classes is 
  {
    []    then "",
    [_ . _] then format(css_classes)
  }.
   
define Printable_tree
   format
     (
       List(Actioner_Local_Action)   l
     ) =
   if l is 
     {
       [ ] then [ ], 
       [h . t] then [if h is 
         {
           close_window then [" window.close(); "]
         } 
         . format(t)]
     }. 
     
define String
  _format
  (
    List(DIV_Option)  opt
  )=
  if opt is
  {
    []      then "",
    [h . t]  then 
      with current = if h is
      {
         class(class_name) then 
           " class=\"" + class_name + "\"",
         dir(reading_Way)  then  
          if reading_Way is
          {
            ltr then  " dir=ltr",
            rtl then  " dir=rtl"
          }
         id(id_name)       then
           " id=\"" + id_name + "\"",         
         title(title_string) then  
           " title=\"" + title_string + "\"",                  
         lang(lang) then  
           " xml:lang=" + lang
      },
      current + _format(t)
  }
  .
  
define Printable_tree
  format_div_option
  (
    List(DIV_Option) options
  )=
  ["<DIV" + _format(options) + ">"] .
  
  
define Printable_tree
   format_actioner
     (
       CommonInfo                     cinfo, 
       String                         state_name, 
       Actioner_Connection            connection, 
       Actioner_Target                target, 
       Actioner_Aspect                aspect, 
       String                         action_name, 
       List((String,String))          extra_ops, 
       List(Actioner_Local_Action)    local_actions,
       Maybe(String)                  mb_form_name,
       Bool                           is_https,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_dir,secret,font) then 
   with url = make_actioner_url(cinfo,connection,target,
                                state_name,action_name,extra_ops,is_https), 
   with action = format_action(url,mb_form_name),
   if aspect is 
     {
   
       link(opt, text) then [
                        if action is 
                          {
                            url(u)          then 
                            ["<a href=\"",u,"\"", format_text_options(opt), ">"],
//                          ["<a href=\"",u,"\" style=\"",format(reverse(opt)),"\">"],
                            javascript(s,h) then 
                              [s,"<a href=\"javascript: ",h,"\">"]
                          },
                          text,
                        "</a>"
                       ], 
                       
       push_button(options, text) then 
         [
           if action is 
           {
              url(u)          then    ["<a href=\"",u,"\"", format_text_options(options), ">",
                                        text, "</a>"
                                      ],
              javascript(s,h) then 
                              [s,"<INPUT TYPE =\"button\" Value=\"",text,"\"", format_text_options(options), "
onClick=\"",h,"\">"]
           }
         ], 
   
       button(url_off,url_on) then 
         [ if action is 
             {
               url(u)           then ["<a href=\"",u],
               javascript(s,h)  then [s,"<a onMouseDown=\"",h]
             },
             "\" style=\"text-decoration:none\">",
             "<img alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",
           "</a>"
         ],
   
       button(url_off,url_on,w,h) then 
         [ if action is 
             {
               url(u)           then ["<a href=\"",u],
               javascript(s,h1) then [s,"<a onMouseDown=\"",h1]
             },
             "\" style=\"text-decoration:none\">",
             "<img width=",w," height=",h," alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",
           "</a>"
         ],
   
       button(text,width,color) then 
         with xxxx = create_button_images(site_dir,
                                          to_rgba(color),text,to_Int[width],font),
         url_on = "buttons/b"+xxxx+"_on.jpg",
         url_off = "buttons/b"+xxxx+"_off.jpg",
         [ if action is 
             {
               url(u)           then ["<a href=\"",u],
               javascript(s,h)  then [s,"<a onMouseDown=\"",h]   
             },
             "\" style=\"text-decoration:none\">",
             "<img alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",
           "</a>"
         ],
   
       immediate_selector(name,size,choices) then          
         [ if action is 
             {
               url(u)           then ["<select href=\"",u]
               javascript(s,h)  then [s,"<select onchange=\"",h]
             },
             "\" name=o",name," size=",size,">",
              format_choices(choices),"</select>"
         ]
     }. 
    

define Printable_tree
   format_local_popup_button
     (
       CommonInfo      cinfo, 
       Actioner_Aspect aspect, 
       Word32           n,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_dir,secret,font) then 
   [ "<script type=\"text/javascript\" language=\"JavaScript\">",
     "  var lpust_",n," = new Array(); ", 
     "  lpust_",n,"[0] = 0; ",
     "</script>",
     "<a href=\"javascript:show_local_popup('lpu_",n,"','lpust_",n,"');\">",
     if aspect is 
       {
         link(opt,text)          then [text], 
         push_button(opt, text)  then [text],  
         button(url_off,url_on)  then 
           [
             "<img alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",             
           ],
   
         button(url_off,url_on,w,h)  then 
           [
             "<img width=",w," height=",h," alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",             
           ],
   
         button(text,width,color)   then
           with xxxx = create_button_images(site_dir,
                                            to_rgba(color),text,to_Int[width],font),
           url_on = "buttons/b"+xxxx+"_on.jpg",
           url_off = "buttons/b"+xxxx+"_off.jpg",
           [
             "<img alt=\"",url_off,"\" src=\"",url_off,"\" border=0",
               " onMouseOver=\"this.src='",url_on,"'\" ",
                " onMouseOut=\"this.src='",url_off,"'\">",             
           ],
   
         immediate_selector(name,size,choices) then should_not_happen([]),

       },
     "</a>"]. 
   
   
   
   
   
      *** [5.5] Formating a private download link. 
   
   We get  the absolute path  of the file  to be downloaded, and  the name under  which it
   should  appear  to  the  client.   The function  'format_private_download'  creates  an
   hypertext link  for downloading the file.  The secured mecanism of  private download is
   used. This function is called by the function which formats HTML_Any($T) elements.
   
   
define Printable_tree 
   format_private_download
     (
       CommonInfo               cinfo, 
       String                   sn,          // state name
       String                   abs_path,    // absolute file path on server
       String                   name,        // name of file as it appears in the browser
       String                   extra,       // extra extension
       Maybe((String,List((String,String))))   action
   
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with private_download_directory = site_directory+"/private_download",                   
   with auth = make_authorization(site_directory,secret,abs_path), 
     [ 
       "<a href=\"",name,extra,"?zauth=",auth,
           if action is 
             {
               failure then [ ]
               success(a) then if a is (an,args) then 
                 ["&amp;a=",an,format_extra_operands(args)]
             },
            "\">",
         name,
       "</a>"
     ].
   
   
      
   
      *** [5.6] Formating rows and cells in a table. 

define Word32
   percent
     (
       Word32 p
     ) =
   if p -< 0 then 0 else if p >- 100 then 100 else p. 
   
   
   
   
   Formating cell options. 

   
   
define String 
   format
     (
       BackgroundOption o
     ) =
   if o is 
     {
       repeat             then "",
       repeat_horizontal  then "; background-repeat: repeat-x",
       repeat_vertical    then "; background-repeat: repeat-y",
       no_repeat          then "; background-repeat: no-repeat",
       center             then "; background-position: center top"      
     }. 
   
define String
   format     
     (
        List(BackgroundOption) l
     ) =
   if l is 
     {
       [ ] then "", 
       [h . t] then format(h)+format(t)
     }.
     
   
define String 
   format
     (
       List(Cell_Option)   options
     ) =
   if options is 
     {
       [ ] then "", 
       [h . t] then 
         if h is
            {
              left                          then " align=left",
              h_center                      then " align=center",
              right                         then " align=right",
              top                           then " valign=top",
              v_center                      then " valign=middle",
              bottom                        then " valign=bottom",
              base_line                     then " valign=baseline",
              background_color(c)           then " bgcolor=\""+html_format(c)+"\"",
              background_image(n,o)         then " style=\"background: url("+n+")"+format(o)+"\"",
              width(w)                      then " width=\""+w+"\"",
              percentage_width(n)           then " width=\""+percent(n)+"\"", 
              height(h1)                    then " height="+h1,
              columns(n)                    then " colspan="+n,
              rows(n)                       then " rowspan="+n,
              nowrap                        then " nowrap"          
            }
         + format(t)
     }.
   
   
   Normalizing a list of cell options  (horizontal position must be specified; the default
   is 'left').
   
define List(Cell_Option)
   normalize
     (
       List(Cell_Option)   l
     ) =
   if member(l,left)      then l else
   if member(l,h_center)  then l else
   if member(l,right)     then l else
   [left . l]. 
   
   
   Formating cells in a row. 
   
define Printable_tree
   format
     (
       List(HTML_Cell($T))               cells,
       $T -> Printable_tree     format_element
     ) =
   if cells is 
     {
       [ ] then [ ], 
       [h . t] then if h is cell(options,element) then 
         ["<td ",format(reverse(normalize(options))),">",
            format_element(element), 
          "</td>" 
          . format(t,format_element)]
     }. 
   
   
   Formating the rows in a table. 
   
define Printable_tree
   format
     (
       List(HTML_Row($T))                rows,
       $T -> Printable_tree     format_element,    
     ) =
   if rows is 
     {
       [ ] then [ ], 
       [h . t] then if h is row(options,cells) then 
         ["<tr ",format(reverse(options)),">",
          format(cells,format_element),
          "</tr>"
          . format(t,format_element)]
     }. 
   
   

define Printable_tree
   format1
     (
       List(TextAreaOption) l
     ) =
   if l is 
     {
       [] then [], 
       [h . t] then if h is 
         {
           disabled    then [" disabled " . format1(t)]
           read_only   then [" readonly " . format1(t)]
           wrap_lines  then [" wrap " . format1(t)]
         }
     }.
   
define Printable_tree
   format
     (
       List(TextAreaOption) l
     ) =
   if member(l,wrap_lines)
   then format1(l)
   else [" wrap=off " . format1(l)]. 
   
   
      *** [5.7] Formating elements which may be put anywhere. 
   
   The  function  below  involves  the   parameter  $T  which  is  later  instantiated  as
   'HTML_In_Form' or as 'HTML_Off_Form'. Now,  since there are dictinct 'format' functions
   for these two  types, and because formating of tables requires  recursive calls of such
   functions, it is necessary to provide the 'format' function to be called recursively as
   an argument.   Putting naively a call to  'format' will not work,  because the compiler
   will look for a function able to format data of type $T (which is at that time distinct
   from any other type,  including our two types).  Such a function  does not exist. Hence
   the  function to  be  called for  formating elements  must  be passed  as a  functional
   argument (called 'format_element' below).  Actually,  what we pass is a function taking
   a unique  argument of type  $T.  Other  informations (like the  name of the  state) are
   already in the function by way of full functionality.
   
   
   Formating  text  options.  They are  formated  in  CSS  syntax,  to  be used  within  a
   'style=...'.
   
define String
   format
     (
       List(Text_Option) l
     ) =
   if l is 
     {
       [ ] then "",
       [h . t] then if h is 
         {
         size(n)            then "font-size:"+n+"pt",
         font(fn)           then "font-family:"+fn,
         color(c)           then if c is rgb(r,g,b) then 
         "color:rgb("+to_decimal(r)+","+to_decimal(g)+","+to_decimal(b)+")",
         italic           then "font-style:italic",
           oblique          then "font-style:oblique",
           small_capitals   then "font-variant:small-caps",
           bold             then "font-weight:bold",
           underlined       then "text-decoration:underline",
           left_justified   then "text-align:left",
           right_justified  then "text-align:right",
           justified        then "text-align:justify",
           line_through     then "text-decoration:line-through",
           nowrap           then "white-space:nowrap",
           css_class(class_name)then " class=\"" +class_name +"\""
         } + if t is [ ] then "" else ("; "+format(t))
     }.
   
   
   
   Formating table options. 
   
define String 
   format
     (
       List(Table_Option)    l,
       Bool                  border_seen
     ) =
   if l is 
     {
       [ ] then if border_seen then "" else " border=0 cellspacing=0 cellpadding=0",
       [h . t] then if h is 
         {
           background_color(c)    then " bgcolor=\""+html_format(c)+"\""+format(t,border_seen), 
           background_image(url)  then " background="+url+format(t,border_seen),
           border(o,top,i,c)      then " border="+o+" cellspacing="+top+" cellpadding="+i+
                                         //" bordercolor="+format(c)+
                                         format(t,true), 
           width(w)               then " width=\""+w+"\""+format(t,border_seen),
           percentage_width(p)    then " width=\""+percent(p)+"\""+format(t,border_seen),
         }
     }.
   

   
   
   
define Printable_tree
   format_scroller
     (
       String                       sn,
       Word32                        width,
       Word32                        height,
       Word32                        content_width,
       Word32                        content_height,
       Word32                        idnum,       // identifying the scroller
       $T                           content,
       $T -> Printable_tree         format_element
     ) =
   [
     "<script type = \"text/javascript\" language=\"JavaScript\">",
       "function doscroll_",idnum,"(dx,dy) {\n", 
       "  if (document.layers) { var c_",idnum," = eval(document.cs_",idnum,"); } else\n",
       "  if (document.getElementById) {var c_",idnum," = eval(\"document.getElementById('cs_",
                                                            idnum,"').style\"); } else\n", 
       "  if (document.all) { var c_",idnum," = eval(document.all.cs_",idnum,".style); };\n",
       "  var x_",idnum," = parseInt(c_",idnum,".left);\n",
       "  var y_",idnum," = parseInt(c_",idnum,".top);\n",
       "  if ((x_",idnum,"+dx <= 0) && (x_",idnum,"+dx > ",width-content_width,"))\n",
       "    { x_",idnum," += dx; }\n", 
       "  if ((y_",idnum,"+dy <= 0) && (y_",idnum,"+dy > ",height-content_height,"))\n",
       "    { y_",idnum," += dy; }\n",
       "  c_",idnum,".left = x_",idnum,";\n",
       "  c_",idnum,".top = y_",idnum,";\n",
       "   }\n", 
     "</script>",
     "<table>",
       "<tr>",
         "<td align=left valign=top",
            " width=",width,
            " height=",height,
            ">",
            "<div id=\"ws_",idnum,"\" style=\"position:absolute; width:",width,"px; height:",height,"px;",
                                       " clip:rect(0px ",width,"px ",height,"px 0px)\">",
              "<div id=\"cs_",idnum,"\" style=\"position:absolute; left:0px; top:0px\">", 
                 format_element(content), 
              "</div>", 
            "</div>",
         "</td>",
         "<td valign=bottom>",
           "<table>",
             "<tr><td><img alt=\"sroll up\" src=\"scrollup.gif\"   onMouseDown=\"doscroll_",
                     idnum,"(0,20);\"></td></tr>",
             "<tr><td><img alt=\"scroll down\" src=\"scrolldown.gif\" onMouseDown=\"doscroll_",
                     idnum,"(0,-20);\"></td></tr>",
           "</table>",
         "</td>",
       "</tr>",
       (if content_width >- width then 
        [
        "<tr>",
          "<td align=right>",
             "<table>",
               "<tr>",
                 "<td><img alt=\"scroll left\" src=\"scrollleft.gif\" onMouseDown=\"doscroll_",
                           idnum,"(20,0);\"></td>",
                 "<td><img alt=\"scroll right\" src=\"scrollright.gif\" onMouseDown=\"doscroll_",
                           idnum,"(-20,0);\"></td>",
               "</tr>",
             "</table>",
           "</td>",
         "</tr>",
        ] else [ ]), 
     "</table>"
   ]. 

   
   
define Printable_tree
   popup_topbar
     (
       CommonInfo     cinfo, 
       String         title,
       RGB            color,
       Word32          width,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with xxxx = web_arg_encode(sha1((title,color,width))), 
        path = site_directory+"/public/buttons/t"+xxxx+".jpg",
      result = (Printable_tree)["<img alt=\"button\" src=\"buttons/t"+xxxx+".jpg\">"],
     if file_exists(path) then result else
           with col = to_rgba(color), 
                 bg = create_button_background(col,to_Int[width],20), 
   very_light_color = lighten(col,70),
         dark_color = darken(col,40), 
        title_width = printed_text_width(font,title), 
       draw_button_text(bg,title,title_width,very_light_color,dark_color,font); 
       forget(write_image_to_JPEG_file(to_JPEG(bg),path,100)); 
       result. 
   
      
define Printable_tree
   popup_close_button
     (
       CommonInfo       cinfo, 
       RGB              color,
       String           div_name,
       String           state_var_name,
     ) = 
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with xxxx = web_arg_encode(sha1(color)), 
     path_on = site_directory+"/public/buttons/c"+xxxx+"_on.jpg",
    path_off = site_directory+"/public/buttons/c"+xxxx+"_off.jpg",
      result = (Printable_tree)["<img alt=\"button\" src=\"buttons/c"+xxxx+"_off.jpg\"",
        " onMouseOver=\"this.src='buttons/c"+xxxx+"_on.jpg'\"",
        "  onMouseOut=\"this.src='buttons/c"+xxxx+"_off.jpg'\"",
        "  onMouseDown=\"show_local_popup('",div_name,"','",state_var_name,"')\">"],
     if file_exists(path_on) then result else
           with col = to_rgba(color), 
              title = "x", 
        title_width = printed_text_width(font,title), 
              bg_on = create_button_background(lighten(col,30),20,20), 
             bg_off = create_button_background(col,20,20), 
   very_light_color = lighten(col,70),
         dark_color = darken(col,40), 
       draw_button_text(bg_on,title,title_width,very_light_color,dark_color,font); 
       draw_button_text(bg_off,title,title_width,very_light_color,dark_color,font); 
       forget(write_image_to_JPEG_file(to_JPEG(bg_on),path_on,100)); 
       forget(write_image_to_JPEG_file(to_JPEG(bg_off),path_off,100)); 
       result. 
   
   
   
   
   
   The function below formats a datum of type 'HTML_Any($T)'. 
   
define Printable_tree
   format
     ( 
       CommonInfo                 cinfo, 
       String                     sn,              // state_name 
       Var(Word32)                 ic_v, 
       HTML_Any($T)               element,
       $T -> Printable_tree       format_element,  // able to format a datum of type $T
       Bool                       is_https,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   if element is 
     {
       any_text(opts,t)              then 
         ["<span ", format_text_options(opts), ">",t,"</span>"], 
       any_preformated(opts,s)       then 
         ["<span ", format_text_options(opts), "><pre>",s,"</pre></span>"], 
         //["<pre>",s,"</pre>"], 
       any_paragraph(opts,t)         then 
         ["<p ", format_text_options(opts), ">",t,"</p>"], 
       any_image(url)                then 
         ["<img alt=\"",url,"\" src=\"",url,"\">"],
       any_image(url,w,h)            then 
         ["<img alt=\"",url,"\" src=\"",url,"\" width=",w," height=",h,">"],
       any_table(opts,rows)          then 
         ["<table ",format(reverse(opts),false),">",format(rows,format_element),"</table>"],
       any_center(e)                 then 
         ["<center>",format_element(e),"</center>"], 
       any_mail_to(email,elem)       then
         ["<a href=\"mailto:",email,"\">",format_element(elem),"</a>"], 
       any_scroller(w,h,cw,ch,c)     then 
         format_scroller(sn,w,h,cw,ch,new_idnum(ic_v),c,format_element), 
       any_fixed_size(w,h,c)         then 
         with url = create_secondary_document(site_directory,secret,sn,format_element,c,w), 
         ["<object data=\"",url,"\" type=\"text/html\" width=",format(w)," height=",format(h)," >",
             "secondary document",
          "</object>"], 
       any_fixed_size_2(w,h,fn)        then
         with url = fn+"?zauth="+make_authorization(site_directory,secret,
                 fn),
         ["<object data=\"",url,"\" type=\"text/html\" width=",format(w)," height=",format(h)," >",
             "secondary document",
          "</object>"],          
       any_actioner(c,t,a,an,eo,ja,fn)  then 
         format_actioner(cinfo,sn,c,t,a,an,eo,ja,fn,is_https),
       any_local_popup(a,c,x,y,t,clr,w)  then 
         with n = new_idnum(ic_v), 
              b = format_local_popup_button(cinfo,a,n), 
              f = ["<div id=\"lpu_",n,"\" style=\"position:absolute; left:",x,"px; top:",y,"px; ",
                        "visibility:hidden; \">", 
             "<table width=\"",w+20,"\" border=0 cellspacing=0 cellpadding=0 bgcolor=\"",
                  html_format(lighten(clr,100)),"\"><tr><td align=center width=\"",w,"\">",
                 popup_topbar(cinfo,t,clr,w),
                   "</td><td width=\"20\">",
                 popup_close_button(cinfo,clr,"lpu_"+n,"lpust_"+n),
                   "</td></tr>",
                "<tr><td align=center valign=middle colspan=2>",
                  format_element(c), 
             "</td></tr></table>",
           "</div>"],
         [ "<table><tr><td>",
           b,
           "</td><td>",
           f,
           "</td></tr></table>"
         ],
       any_foreign_link(options,url,name)    then
         ["<a href=\"",url,"\"><span ", format_text_options(options), ">",name,"</span></a>"], 
       any_private_download(url,name,extra_ext,action)  then 
         format_private_download(cinfo,sn,url,name,extra_ext,action),
       any_div(options, e)                   then
         [format_div_option(options), format_element(e),"</DIV>"],
       any_div_empty(options)                then
         [format_div_option(options), "</DIV>"],
     }. 
   
   

   
      *** [5.8] Formating 'in form' elements. 
   
   

   
   
define Printable_tree
   format
     (
       CommonInfo               cinfo, 
       String                   fn,      // form_name
       String                   sn,      // state_name
       Var(Word32)               ic_v,    // idnum counter variable
       HTML_In_Form             element,
       Bool                     is_https,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with format_element = (HTML_In_Form e) |-> format(cinfo,fn,sn,ic_v,e,is_https), 
   if element is 
     {
       literal_pt(t)               then t, 
       literal(t)                  then [t], 
       sequence(l)                 then flat(map(format_element,l))
       text(opts,t)                then 
         format(cinfo,sn,ic_v,any_text(opts,t),format_element,is_https), 
       preformated(o,s)            then 
         format(cinfo,sn,ic_v,any_preformated(o,s),format_element,is_https), 
       paragraph(opts,t)                then 
         format(cinfo,sn,ic_v,any_paragraph(opts,t),format_element,is_https), 
       image(url)                  then 
         format(cinfo,sn,ic_v,any_image(url),format_element,is_https),
       image(url,w,h)              then 
         format(cinfo,sn,ic_v,any_image(url,w,h),format_element,is_https),
       table(opts,rows)            then 
         format(cinfo,sn,ic_v,any_table(opts,rows),format_element,is_https),
       center(e)                   then 
         format(cinfo,sn,ic_v,any_center(e),format_element,is_https), 
       mail_to(a,e)                then 
         format(cinfo,sn,ic_v,any_mail_to(a,e),format_element,is_https), 
       scroller(w,h,cw,ch,c)       then 
         format(cinfo,sn,ic_v,any_scroller(w,h,cw,ch,c),format_element,is_https), 
       actioner(c,t,a,an,eo,ja)    then 
         format(cinfo,sn,ic_v,any_actioner(c,t,a,an,eo,ja,success(fn)),format_element,is_https), 
       local_popup(a,c,x,y,t,clr,w)  then 
         format(cinfo,sn,ic_v,any_local_popup(a,c,x,y,t,clr,w),format_element,is_https), 
       foreign_link(options,url,name)  then
         format(cinfo,sn,ic_v,any_foreign_link(options,url,name),format_element,is_https), 
       private_download(url,name,extra,action)    then 
         format(cinfo,sn,ic_v,any_private_download(url,name,extra,action),format_element,is_https),    
       text_input(n,i,w)           then 
         ["<input type=text name=o",n," size=",w," value=\"",i,"\">"], 
         //["&nbsp; <input type=text name=o",n," size=",w," value=\"",i,"\">"], 
       password_input(n,w)         then 
         ["<input type=password name=p",n," size=",w,">"],
         //["&nbsp; <input type=password name=p",n," size=",w,">"],
       text_area(opts,n,i,w,h)          then
         ["<textarea ",format(opts)," name=o",n," cols=",w," rows=",h,">",i,"</textarea>"],
       file_upload(n,w)            then 
         ["<input type=file size=",w," name=o",n,">"],
       selector(n,s,cs)            then 
         ["<select name=o",n," size=",s,">",format_choices(cs),"</select>"],
       selector(n,s,cs,sd)         then 
         ["<select name=o",n," size=",s,">",format_choices(cs,sd),"</select>"],
       selector_c(n,s,cs)            then 
         ["<select name=o",n," size=",s,">",format_choices(cs),"</select>"],
       selector_c(n,s,cs,sd)         then 
         ["<select name=o",n," size=",s,">",format_choices(cs,sd),"</select>"],
   
       radio_button(n,v,c)         then  
         ["<input type=radio name=o",n," value=\"",v,"\"",(if c then " checked" else ""),">"],
       check_box(n,c)              then 
         ["<input type=checkbox name=o",n,(if c then " checked " else ""),">"]
     }. 

   
   
   
   
   
      *** [5.9] Formating 'off form' elements. 

   The encryption type 'multipart/form-data' is required for a form containing an upload. 
   
   
define Bool
   contains_an_upload
    (
      HTML_In_Form form_content
    ).   
   
define Bool
   contains_an_upload
     (
       HTML_Row(HTML_In_Form) row
     ) =
   mapor(contains_an_upload,
         map(content,cells(row))). 
   
   
define Bool
   contains_an_upload
    (
      HTML_In_Form form_content
    ) =
   if form_content is
     {
   literal_pt(t)              then false,
   literal(t)                 then false,
   sequence(l)                then mapor(contains_an_upload,l)
   text(o,t)                  then false,  
   preformated(o,s)           then false, 
   paragraph(o,t)             then false,  
   image(u)                   then false, 
   image(u,w,h)               then false, 
   table(o,rows)              then mapor(contains_an_upload,rows),
   center(e)                  then contains_an_upload(e), 
   mail_to(m,e)               then false, // 'e' may but should not contain an upload 
   scroller(w,h,cw,ch,e)      then contains_an_upload(e), 
   actioner(c,t,a,an,eo,ja)   then false, 
   local_popup(a,c,x,y,t,clr,w) then contains_an_upload(c), 
   foreign_link(o,u,n)        then false,
   private_download(p,n,e,a)  then false, 
   text_input(n,i,w)          then false,
   password_input(n,w)        then false,
   text_area(o,n,i,w,h)       then false, 
   file_upload(n,w)           then true, 
   selector(n,s,c)            then false, 
   selector(n,s,c,p)          then false,  
   selector_c(n,s,c)            then false, 
   selector_c(n,s,c,p)          then false,  
   radio_button(n,v,c)        then false,
   check_box(n,c)             then false    
     }. 
   
define String
  enctype
    (
      HTML_In_Form form_content
    ) =
  if contains_an_upload(form_content)
  then " enctype=multipart/form-data"
  else "". 
 
   
   
define Printable_tree
   format
     (
       CommonInfo               cinfo, 
       String                   sn,               // state_name
       Var(Word32)               ic_v,             // 'idnum' counter variable
       HTML_Off_Form            element,
       Bool                     is_https,
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with format_element = (HTML_Off_Form e) |-> format(cinfo,sn,ic_v,e,is_https), 
   if element is 
     {
       literal_pt(t)                then t, 
       literal(t)                   then [t], 
       sequence(l)                  then flat(map(format_element,l)),
       text(opts,t)                 then 
         format(cinfo,sn,ic_v,any_text(opts,t),format_element,is_https), 
       preformated(o,s)             then 
         format(cinfo,sn,ic_v,any_preformated(o,s),format_element,is_https), 
       paragraph(opts,t)            then 
         format(cinfo,sn,ic_v,any_paragraph(opts,t),format_element,is_https), 
       image(url)                   then 
         format(cinfo,sn,ic_v,any_image(url),format_element,is_https), 
       image(url,w,h)               then 
         format(cinfo,sn,ic_v,any_image(url,w,h),format_element,is_https), 
       table(opts,rows)             then 
         format(cinfo,sn,ic_v,any_table(opts,rows),format_element,is_https), 
       center(e)                    then 
         format(cinfo,sn,ic_v,any_center(e),format_element,is_https), 
       mail_to(a,e)                 then
         format(cinfo,sn,ic_v,any_mail_to(a,e),format_element,is_https), 
       scroller(w,h,cw,ch,c)        then 
         format(cinfo,sn,ic_v,any_scroller(w,h,cw,ch,c),format_element,is_https), 
       fixed_size(w,h,c)            then 
         format(cinfo,sn,ic_v,any_fixed_size(w,h,c),format_element,is_https), 
       fixed_size_2(w,h,fn)           then 
         format(cinfo,sn,ic_v,any_fixed_size_2(w,h,fn),format_element,is_https), 
       actioner(c,t,a,an,eo,ja)     then 
         format(cinfo,sn,ic_v,any_actioner(c,t,a,an,eo,ja,failure),format_element,is_https), 
       actioner(c,t,a,an,eo,ja,fn)  then 
         format(cinfo,sn,ic_v,any_actioner(c,t,a,an,eo,ja,success(fn)),format_element,is_https), 
       local_popup(a,c,x,y,t,clr,w)   then 
         format(cinfo,sn,ic_v,any_local_popup(a,c,x,y,t,clr,w),format_element,is_https),    
       foreign_link(options,url,name)       then
         format(cinfo,sn,ic_v,any_foreign_link(options,url,name),format_element,is_https), 
       private_download(url,name,extra,action)     then 
         format(cinfo,sn,ic_v,any_private_download(url,name,extra,action),format_element,is_https),    
       label(n)                     then ["<a name=\"",n,"\">"],
       form(fn,c)                   then 
         [
           "<form name=\"f",fn,"\" method=POST",
              enctype(c),
              " action=\"http",
                         if is_https then "s" else "",
                         "://",common_name,":",http_port,"/\">",      
                                      // action is set dynamically by
                                      // the actioner using JavaScript
              format(cinfo,fn,sn,ic_v,c,is_https),               
           "</form>"
         ]
       div(options, e)              then
        format(cinfo,sn,ic_v,any_div(options, e),format_element,is_https),    
       div_empty(options)           then
        format(cinfo,sn,ic_v,any_div_empty(options),format_element,is_https),    

     }. 
   
   
   
   
      *** [5.10] Formating meta-tags. 
   
define Printable_tree
   format_keywords
     (
       List(String) l
     ) =
   if l is 
     {
       []      then [ ], 
       [h . t] then if t is []
         then [h]
         else [h , ", " . format_keywords(t)]
     }. 
   
   
define Printable_tree
   format
     (
       CommonInfo        cinfo, 
       String            state_name,
       HTML_Meta         m,
       Bool              is_https
     ) =
   if m is 
     {
       keywords(l)        then ["<meta name=\"keywords\" content=\"",format_keywords(l),"\">"],
       refresh(co,ta,an,delay)  then 
                               ["<meta http-equiv=\"Refresh\" content=\"",delay,"; URL=",
                                    make_actioner_url(cinfo,co,ta,state_name,an,[],is_https),"\">"], 
       meta(n,c)          then ["<meta name=\"",n,"\" content=\"",c,"\">"], 
       http_equiv(n,c)    then ["<meta http-equiv=\"",n,"\" content=\"",c,"\">"],
       generic_meta(l)    then ["<meta ",
                                  flat(map(((String,String) p) |-> if p is (n,v) then [n,"=\"",v,"\" "],
                                      l)),
                                 ">"],
       literal(s)         then [s]
     }.
   
   
define Printable_tree
   format
     (
       CommonInfo         cinfo, 
       String             state_name, 
       List(HTML_Meta)    metas,
       Bool               is_https,
       String             charset
     ) =
   if metas is 
     {
       []      then [format(cinfo,state_name,http_equiv("content-type",
                                                        "text/html; charset="+charset),is_https)],
       [h . t] then [format(cinfo,state_name,h,is_https) 
                     . format(cinfo,state_name,t,is_https,charset)]
     }. 

     
define Printable_tree
  format
    (
      Body_Option o
    ) =
  if o is
    {
      background_color(c)                then [" bgcolor=\"" , (String)html_format(c), "\""],
      background_image(n)                then [" background=", n],
      background_image(n,o1)              then [" style=\"background: url(",n,")",format(o1),"\""]
      
    }.
   
     
          
define Printable_tree
   format
     (
       List(Body_Option) l
     ) =
   if l is 
     {
       [ ]     then [ ], 
       [h . t] then [format(h) . format(t)]
     }.
     
define Printable_tree
   add_css_files
     (
       List(CSS_File)   l
     ) =
   if l is 
     {
       [ ] then [ ], 
       [h . t] then 
       [ ["<LINK rel=\"stylesheet\" type=\"text/css\" href=" + file_name(h) + ">\n" ]
         . add_css_files(t)]
     }.

define Printable_tree
  add_css_styles
  (
    List(CSS_Style) css_styles
  ) =
  
  if css_styles is
  {
    [] then [],
    [_ . _] then [ "<style type=\"text/css\"><!--\n",
                  format_css_styles(css_styles), 
                 "  --></style>"
               ]
  }.
    
define Printable_tree
   format
     (
       CommonInfo               cinfo, 
       String                   state_name, 
       HTML_Page                page,
       Bool                     is_https,
       String                   charset
     ) =
   if cinfo is info(common_name,http_port,https_port,site_directory,secret,font) then 
   with ic_v = var((Word32)0), 
   if page is 
     {
       html_page(title,metas,css_styles, css_files, body) then 
         if body is body(options,element) then
         [ doctype_w3c_header, 
           "<html>",
             "<head>",
             add_css_styles(css_styles),
             add_css_files(css_files),
             "<link rel=\"shortcut icon\" href=\"favicon.png\">", 
             "<script type = \"text/javascript\" language=\"JavaScript\">",
    "  function show_local_popup(divname,stvname) {",
    "  if (document.layers) { var d = eval(document.divname); } else\n",
    "  if (document.getElementById) { var d = eval(\"document.getElementById(divname)\"); } else\n",
    "  if (document.all) { var d = eval(document.all.divname.style)};\n",
    // " alert(typeof(eval(stvname))); ", 
    " var s = eval(stvname); ", 
    "  if (s[0]==0) ",
    "   { s[0]=1; d.style.visibility = 'visible'; d.zIndex = 100; } else\n",
    "   { s[0]=0; d.style.visibility = 'hidden'; }; }", 
               "</script>",
               "<title>",title,"</title>",                         // put title
               format(cinfo,state_name,metas,is_https,charset),      // format the metas
             "</head>",
             "<body ", format(options), ">",                       // format body options
               //"<center>",
               format(cinfo,state_name,ic_v,element,is_https), 
               //"</center>", 
             "</body>",
           "</html>"
         ]
     }.