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: 3734: 3735: 3736: 3737: 3738: 3739: 3740: 3741: 3742: 3743: 3744: 3745: 3746: 3747: 3748: 3749: 3750: 3751: 3752: 3753: 3754: 3755: 3756: 3757: 3758: 3759: 3760: 3761: 3762: 3763: 3764: 3765: 3766: 3767: 3768: 3769: 3770: 3771: 3772: 3773: 3774: 3775: 3776: 3777: 3778: 3779: 3780: 3781: 3782: 3783: 3784: 3785: 3786: 3787: 3788: 3789: 3790: 3791: 3792: 3793: 3794: 3795: 3796: 3797: 3798: 3799: 3800: 3801: 3802: 3803: 3804: 3805: 3806: 3807: 3808: 3809: 3810: 3811: 3812: 3813: 3814: 3815: 3816: 3817: 3818: 3819: 3820: 3821: 3822: 3823: 3824: 3825: 3826: 3827: 3828: 3829: 3830: 3831: 3832: 3833: 3834: 3835: 3836: 3837: 3838: 3839: 3840: 3841: 3842: 3843: 3844: 3845: 3846: 3847: 3848: 3849: 3850: 3851: 3852: 3853: 3854: 3855: 3856: 3857: 3858: 3859: 3860: 3861: 3862: 3863: 3864: 3865: 3866: 3867: 3868: 3869: 3870: 3871: 3872: 3873: 3874: 3875: 3876: 3877: 3878: 3879: 3880: 3881: 3882: 3883: 3884: 3885: 3886: 3887: 3888: 3889: 3890: 3891: 3892: 3893: 3894: 3895: 3896: 3897: 3898: 3899: 3900: 3901: 3902: 3903: 3904: 3905: 3906: 3907: 3908: 3909: 3910: 3911: 3912: 3913: 3914: 3915: 3916: 3917: 3918: 3919: 3920: 3921: 3922: 3923: 3924: 3925: 3926: 3927: 3928: 3929: 3930: 3931: 3932: 3933: 3934: 3935: 3936: 3937: 3938: 3939: 3940: 3941: 3942: 3943: 3944: 3945: 3946: 3947: 3948: 3949: 3950: 3951: 3952: 3953: 3954: 3955: 3956: 3957: 3958: 3959: 3960: 3961: 3962: 3963: 3964: 3965: 3966: 3967: 3968: 3969: 3970: 3971: 3972: 3973: 3974: 3975: 3976: 3977: 3978: 3979: 3980: 3981: 3982: 3983: 3984: 3985: 3986: 3987: 3988: 3989: 3990: 3991: 3992: 3993: 3994: 3995: 3996: 3997: 3998: 3999: 4000: 4001: 4002: 4003: 4004: 4005: 4006: 4007: 4008: 4009: 4010: 4011: 4012: 4013: 4014: 4015: 4016: 4017: 4018: 4019: 4020: 4021: 4022: 4023: 4024: 4025: 4026: 4027: 4028: 4029: 4030: 4031: 4032: 4033: 4034: 4035: 4036: 4037: 4038: 4039: 4040: 4041: 4042: 4043: 4044: 4045: 4046: 4047: 4048: 4049: 4050: 4051: 4052: 4053: 4054: 4055: 4056: 4057: 4058: 4059: 4060: 4061: 4062: 4063: 4064: 4065: 4066: 4067: 4068: 4069: 4070: 4071: 4072: 4073: 4074: 4075: 4076: 4077: 4078: 4079: 4080: 4081: 4082: 4083: 4084: 4085: 4086: 4087: 4088: 4089: 4090: 4091: 4092: 4093: 4094: 4095: 4096: 4097: 4098: 4099: 4100: 4101: 4102: 4103: 4104: 4105: 4106: 4107: 4108: 4109: 4110: 4111: 4112: 4113: 4114: 4115: 4116: 4117: 4118: 4119: 4120: 4121: 4122: 4123: 4124: 4125: 4126: 4127: 4128: 4129: 4130: 4131: 4132: 4133: 4134: 4135: 4136: 4137: 4138: 4139: 4140: 4141: 4142: 4143: 4144: 4145: 4146: 4147: 4148: 4149: 4150: 4151: 4152: 4153: 4154: 4155: 4156: 4157: 4158: 4159: 4160: 4161: 4162: 4163: 4164: 4165: 4166: 4167: 4168: 4169: 4170: 4171: 4172: 4173: 4174: 4175: 4176: 4177: 4178: 4179: 4180: 4181: 4182: 4183: 4184: 4185: 4186: 4187: 4188: 4189: 4190: 4191: 4192: 4193: 4194: 4195: 4196: 4197: 4198: 4199: 4200: 4201: 4202: 4203: 4204: 4205: 4206: 4207: 4208: 4209: 4210: 4211: 4212: 4213: 4214: 4215: 4216: 4217: 4218: 4219: 4220: 4221: 4222: 4223: 4224: 4225: 4226: 4227: 4228: 4229: 4230: 4231: 4232: 4233: 4234: 4235: 4236: 4237: 4238: 4239: 4240: 4241: 4242: 4243: 4244: 4245: 4246: 4247: 4248: 4249: 4250: 4251: 4252: 4253: 4254: 4255: 4256: 4257: 4258: 4259: 4260: 4261: 4262: 4263: 4264: 4265: 4266: 4267: 4268: 4269: 4270: 4271: 4272: 4273: 4274: 4275: 4276: 4277: 4278: 4279: 4280: 4281: 4282: 4283: 4284: 4285: 4286: 4287: 4288: 4289: 4290: 4291: 4292: 4293: 4294: 4295: 4296: 4297: 4298: 4299: 4300: 4301: 4302: 4303: 4304: 4305: 4306: 4307: 4308: 4309: 4310: 4311: 4312: 4313: 4314: 4315: 4316: 4317: 4318: 4319: 4320: 4321: 4322: 4323: 4324: 4325: 4326: 4327: 4328: 4329: 4330: 4331: 4332: 4333: 4334: 4335: 4336: 4337: 4338: 4339: 4340: 4341: 4342: 4343: 4344: 4345: 4346: 4347: 4348: 4349: 4350: 4351: 4352: 4353: 4354: 4355: 4356: 4357: 4358: 4359: 4360: 4361: 4362: 4363: 4364: 4365: 4366: 4367: 4368: 4369: 4370: 4371: 4372: 4373: 4374: 4375: 4376: 4377: 4378: 4379: 4380: 4381: 4382: 4383: 4384: 4385: 4386: 4387: 4388: 4389: 4390: 4391: 4392: 4393: 4394: 4395: 4396: 4397: 4398: 4399: 4400: 4401: 4402: 4403: 4404: 4405: 4406: 4407: 4408: 4409: 4410: 4411: 4412: 4413: 4414: 4415: 4416: 4417: 4418: 4419: 4420: 4421: 4422: 4423: 4424: 4425: 4426: 4427: 4428: 4429: 4430: 4431: 4432: 4433: 4434: 4435: 4436: 4437: 4438: 4439: 4440: 4441: 4442: 4443: 4444: 4445: 4446: 4447: 4448: 4449: 4450: 4451: 4452: 4453: 4454: 4455: 4456: 4457: 4458: 4459: 4460: 4461: 4462: 4463: 4464: 4465: 4466: 4467: 4468: 4469: 4470: 4471: 4472: 4473: 4474: 4475: 4476: 4477: 4478: 4479: 4480: 4481: 4482: 4483: 4484: 4485: 4486: 4487: 4488: 4489: 4490: 4491: 4492: 4493: 4494: 4495: 4496: 4497: 4498: 4499: 4500: 4501: 4502: 4503: 4504: 4505: 4506: 4507: 4508: 4509: 4510: 4511: 4512: 4513: 4514: 4515: 4516: 4517: 4518: 4519: 4520: 4521: 4522: 4523: 4524: 4525: 4526: 4527: 4528: 4529: 4530: 4531: 4532: 4533: 4534: 4535: 4536: 4537: 4538: 4539: 4540: 4541: 4542: 4543: 4544: 4545: 4546: 4547: 4548: 4549: 4550: 4551: 4552: 4553: 4554: 4555: 4556: 4557: 4558: 4559: 4560: 4561: 4562: 4563: 4564: 4565: 4566: 4567: 4568: 4569: 4570: 4571: 4572: 4573: 4574: 4575: 4576: 4577: 4578: 4579: 4580: 4581: 4582: 4583: 4584: 4585: 4586: 4587: 4588: 4589: 4590: 4591: 4592: 4593: 4594: 4595: 4596: 4597: 4598: 4599: 4600: 4601: 4602: 4603: 4604: 4605: 4606: 4607: 4608: 4609: 4610: 4611: 4612: 4613: 4614: 4615: 4616: 4617: 4618: 4619: 4620: 4621: 4622: 4623: 4624: 4625: 4626: 4627: 4628: 4629: 4630: 4631: 4632: 4633: 4634: 4635: 4636: 4637: 4638: 4639: 4640: 4641: 4642: 4643: 4644: 4645: 4646: 4647: 4648: 4649: 4650: 4651: 4652: 4653: 4654: 4655: 4656: 4657: 4658: 4659: 4660: 4661: 4662: 4663: 4664: 4665: 4666: 4667: 4668: 4669: 4670: 4671: 4672: 4673: 4674: 4675: 4676: 4677: 4678: 4679: 4680: 4681: 4682: 4683: 4684: 4685: 4686: 4687: 4688: 4689: 4690: 4691: 4692: 4693: 4694: 4695: 4696: 4697: 4698: 4699: 4700: 4701: 4702: 4703: 4704: 4705: 4706: 4707: 4708: 4709: 4710: 4711: 4712: 4713: 4714: 4715: 4716: 4717: 4718: 4719: 4720: 4721: 4722: 4723: 4724: 4725: 4726: 4727: 4728: 4729: 4730: 4731: 4732: 4733: 4734: 4735: 4736: 4737: 4738: 4739: 4740: 4741: 4742: 4743: 4744: 4745: 4746: 4747: 4748: 4749: 4750: 4751: 4752: 4753: 4754: 4755: 4756: 4757: 4758: 4759: 4760: 4761: 4762: 4763: 4764: 4765: 4766: 4767: 4768: 4769: 4770: 4771: 4772: 4773: 4774: 4775: 4776: 4777: 4778: 4779: 4780: 4781: 4782: 4783: 4784: 4785: 4786: 4787: 4788: 4789: 4790: 4791: 4792: 4793: 4794: 4795: 4796: 4797: 4798: 4799: 4800: 4801: 4802: 4803: 4804: 4805: 4806: 4807: 4808: 4809: 4810: 4811: 4812: 4813: 4814: 4815: 4816: 4817: 4818: 4819: 4820: 4821: 4822: 4823: 4824: 4825: 4826: 4827: 4828: 4829: 4830: 4831: 4832: 4833: 4834: 4835: 4836: 4837: 4838: 4839: 4840: 4841: 4842: 4843: 4844: 4845: 4846: 4847: 4848: 4849: 4850: 4851: 4852: 4853: 4854: 4855: 4856: 4857: 4858: 4859: 4860: 4861: 4862: 4863: 4864: 4865: 4866: 4867: 4868: 4869: 4870: 4871: 4872: 4873: 4874: 4875: 4876: 4877: 4878: 4879: 4880: 4881: 4882: 4883: 4884: 4885: 4886: 4887: 4888: 4889: 4890: 4891: 4892: 4893: 4894: 4895: 4896: 4897: 4898: 4899: 4900: 4901: 4902: 4903: 4904: 4905: 4906: 4907: 4908: 4909: 4910: 4911: 4912: 4913: 4914: 4915: 4916: 4917: 4918: 4919: 4920: 4921: 4922: 4923: 4924: 4925: 4926: 4927: 4928: 4929: 4930: 4931: 4932: 4933: 4934: 4935: 4936: 4937: 4938: 4939: 4940: 4941: 4942: 4943: 4944: 4945: 4946: 4947: 4948: 4949: 4950: 4951: 4952: 4953: 4954: 4955: 4956: 4957: 4958: 4959: 4960: 4961: 4962: 4963: 4964: 4965: 4966: 4967: 4968: 4969: 4970: 4971: 4972: 4973: 4974: 4975: 4976: 4977: 4978: 4979: 4980: 4981: 4982: 4983: 4984: 4985: 4986: 4987: 4988: 4989: 4990: 4991: 4992: 4993: 4994: 4995: 4996: 4997: 4998: 4999: 5000: 5001: 5002: 5003: 5004: 5005: 5006: 5007: 5008: 5009: 5010: 5011: 5012: 5013: 5014: 5015: 5016: 5017: 5018: 5019: 5020: 5021: 5022: 5023: 5024: 5025: 5026: 5027: 5028: 5029: 5030: 5031: 5032: 5033: 5034: 5035: 5036: 5037: 5038: 5039: 5040: 5041: 5042: 5043: 5044: 5045: 5046: 5047: 5048: 5049: 5050: 5051: 5052: 5053: 5054: 5055: 5056: 5057: 5058: 5059: 5060: 5061: 5062: 5063: 5064: 5065: 5066: 5067: 5068: 5069: 5070: 5071: 5072: 5073: 5074: 5075: 5076: 5077: 5078: 5079: 5080: 5081: 5082: 5083: 5084: 5085: 5086: 5087: 5088: 5089: 5090: 5091: 5092: 5093: 5094: 5095: 5096: 5097: 5098: 5099: 5100: 5101: 5102: 5103: 5104: 5105: 5106: 5107: 5108: 5109: 5110: 5111: 5112: 5113: 5114: 5115: 5116: 5117: 5118: 5119: 5120: 5121: 5122: 5123: 5124: 5125: 5126: 5127: 5128: 5129: 5130: 5131: 5132: 5133: 5134: 5135: 5136: 5137: 5138: 5139: 5140: 5141: 5142: 5143: 5144: 5145: 5146: 5147: 5148: 5149: 5150: 5151: 5152: 5153: 5154: 5155: 5156: 5157: 5158: 5159: 5160: 5161: 5162: 5163: 5164: 5165: 5166: 5167: 5168: 5169: 5170: 5171: 5172: 5173: 5174: 5175: 5176: 5177: 5178: 5179: 5180: 5181: 5182: 5183: 5184: 5185: 5186: 5187: 5188: 5189: 5190: 5191: 5192: 5193: 5194: 5195: 5196: 5197: 5198: 5199: 5200: 5201: 5202: 5203: 5204: 5205: 5206: 5207: 5208: 5209: 5210: 5211: 5212: 5213: 5214: 5215: 5216: 5217: 5218: 5219: 5220: 5221: 5222: 5223: 5224: 5225: 5226: 5227: 5228: 5229: 5230: 5231: 5232: 5233: 5234: 5235: 5236: 5237: 5238: 5239: 5240: 5241: 5242: 5243: 5244: 5245: 5246: 5247: 5248: 5249: 5250: 5251: 5252: 5253: 5254: 5255: 5256: 5257: 5258: 5259: 5260: 5261: 5262: 5263: 5264: 5265: 5266: 5267: 5268: 5269: 5270: 5271: 5272: 5273: 5274: 5275: 5276: 5277: 5278: 5279: 5280: 5281: 5282: 5283: 5284: 5285: 5286: 5287: 5288: 5289: 5290: 5291: 5292: 5293: 5294: 5295: 5296: 5297: 5298: 5299: 5300: 5301: 5302: 5303: 5304: 5305: 5306: 5307: 5308: 5309: 5310: 5311: 5312: 5313: 5314: 5315: 5316: 5317: 5318: 5319: 5320: 5321: 5322: 5323: 5324: 5325: 5326: 5327: 5328: 5329: 5330: 5331: 5332: 5333: 5334: 5335: 5336: 5337: 5338: 5339: 5340: 5341: 5342: 5343: 5344: 5345: 5346: 5347: 5348: 5349: 5350: 5351: 5352: 5353: 5354: 5355: 5356: 5357: 5358: 5359: 5360: 5361: 5362: 5363: 5364: 5365: 5366: 5367: 5368: 5369: 5370: 5371: 5372: 5373: 5374: 5375: 5376: 5377: 5378: 5379: 5380: 5381: 5382: 5383: 5384: 5385: 5386: 5387: 5388: 5389: 5390: 5391: 5392: 5393: 5394: 5395: 5396: 5397: 5398: 5399: 5400: 5401: 5402: 5403: 5404: 5405: 5406: 5407: 5408: 5409: 5410: 5411: 5412: 5413: 5414: 5415: 5416: 5417: 5418: 5419: 5420: 5421: 5422: 5423: 5424: 5425: 5426: 5427: 5428: 5429: 5430: 5431: 5432: 5433: 5434: 5435: 5436: 5437: 5438: 5439: 5440: 5441: 5442: 5443: 5444: 5445: 5446: 5447: 5448: 5449: 5450: 5451: 5452: 5453: 5454: 5455: 5456: 5457: 5458: 5459: 5460: 5461: 5462: 5463: 5464: 5465: 5466: 5467: 5468: 5469: 5470: 5471: 5472: 5473: 5474: 5475: 5476: 5477: 5478: 5479: 5480: 5481: 5482: 5483: 5484: 5485: 5486: 5487: 5488: 5489: 5490: 5491: 5492: 5493: 5494: 5495: 5496: 5497: 5498: 5499: 5500: 5501: 5502: 5503: 5504: 5505: 5506: 5507: 5508: 5509: 5510: 5511: 5512: 5513: 5514: 5515: 5516: 5517: 5518: 5519: 5520: 5521: 5522: 5523: 5524: 5525: 5526: 5527: 5528: 5529: 5530: 5531: 5532: 5533: 5534: 5535: 5536: 5537: 5538: 5539: 5540: 5541: 5542: 5543: 5544: 5545: 5546: 5547: 5548: 5549: 5550: 5551: 5552: 5553: 5554: 5555: 5556: 5557: 5558: 5559: 5560: 5561: 5562: 5563: 5564: 5565: 5566: 5567: 5568: 5569: 5570: 5571: 5572: 5573: 5574: 5575: 5576: 5577: 5578: 5579: 5580: 5581: 5582: 5583: 5584: 5585: 5586: 5587: 5588: 5589: 5590: 5591: 5592: 5593: 5594: 5595: 5596: 5597: 5598: 5599: 5600: 5601: 5602: 5603: 5604: 5605: 5606: 5607: 5608: 5609: 5610: 5611: 5612: 5613: 5614: 5615: 5616: 5617: 5618: 5619: 5620: 5621: 5622: 5623: 5624: 5625: 5626: 5627: 5628: 5629: 5630: 5631: 5632: 5633: 5634: 5635: 5636: 5637: 5638: 5639: 5640: 5641: 5642: 5643: 5644: 5645: 5646: 5647: 5648: 5649: 5650: 5651: 5652: 5653: 5654: 5655: 5656: 5657: 5658: 5659: 5660: 5661: 5662: 5663: 5664: 5665: 5666: 5667: 5668: 5669: 5670: 5671: 5672: 5673: 5674: 5675: 5676: 5677: 5678: 5679: 5680: 5681: 5682: 5683: 5684: 5685: 5686: 5687: 5688: 5689: 5690: 5691: 5692: 5693: 5694: 5695: 5696: 5697: 5698: 5699: 5700: 5701: 5702: 5703: 5704: 5705: 5706: 5707: 5708: 5709: 5710: 5711: 5712: 5713: 5714: 5715: 5716: 5717: 5718: 5719: 5720: 5721: 5722: 5723: 5724: 5725: 5726: 5727: 5728: 5729: 5730: 5731: 5732: 5733: 5734: 5735: 5736: 5737: 5738: 5739: 5740: 5741: 5742: 5743: 5744: 5745: 5746: 5747: 5748: 5749: 5750: 5751: 5752: 5753: 5754: 5755: 5756: 5757: 5758: 5759: 5760: 5761: 5762: 5763: 5764: 5765: 5766: 5767: 5768: 5769: 5770: 5771: 5772: 5773: 5774: 5775: 5776: 5777: 5778: 5779: 5780: 5781: 5782: 5783: 5784: 5785: 5786: 5787: 5788: 5789: 5790: 5791: 5792: 5793: 5794: 5795: 5796: 5797: 5798: 5799: 5800: 5801: 5802: 5803: 5804: 5805: 5806: 5807: 5808: 5809: 5810: 5811: 5812: 5813: 5814: 5815: 5816: 5817: 5818: 5819: 5820: 5821: 5822: 5823: 5824: 5825: 5826: 5827: 5828: 5829: 5830: 5831: 5832: 5833: 5834: 5835: 5836: 5837: 5838: 5839: 5840: 5841: 5842: 5843: 5844: 5845: 5846: 5847: 5848: 5849: 5850: 5851: 5852: 5853: 5854: 5855: 5856: 5857: 5858: 5859: 5860: 5861: 5862: 5863: 5864: 5865: 5866: 5867: 5868: 5869: 5870: 5871: 5872: 5873: 5874: 5875: 5876: 5877: 5878: 5879: 5880: 5881: 5882: 5883: 5884: 5885: 5886: 5887: 5888: 5889: 5890: 5891: 5892: 5893: 5894: 5895: 5896: 5897: 5898: 5899: 5900: 5901: 5902: 5903: 5904: 5905: 5906: 5907: 5908: 5909: 5910: 5911: 5912: 5913: 5914: 5915: 5916: 5917: 5918: 5919: 5920: 5921: 5922: 5923: 5924: 5925: 5926: 5927: 5928: 5929: 5930: 5931: 5932: 5933: 5934: 5935: 5936: 5937: 5938: 5939: 5940: 5941: 5942: 5943: 5944: 5945: 5946: 5947: 5948: 5949: 5950: 5951: 5952: 5953: 5954: 5955: 5956: 5957: 5958: 5959: 5960: 5961: 5962: 5963: 5964: 5965: 5966: 5967: 5968: 5969: 5970: 5971: 5972: 5973: 5974: 5975: 5976: 5977: 5978: 5979: 5980: 5981: 5982: 5983: 5984: 5985: 5986: 5987: 5988: 5989: 5990: 5991: 5992: 5993: 5994: 5995: 5996: 5997: 5998: 5999: 6000: 6001: 6002: 6003: 6004: 6005: 6006: 6007: 6008: 6009: 6010: 6011: 6012: 6013: 6014: 6015: 6016: 6017: 6018: 6019: 6020: 6021: 6022: 6023: 6024: 6025: 6026: 6027: 6028: 6029: 6030: 6031: 6032: 6033: 6034: 6035: 6036: 6037: 6038: 6039: 6040: 6041: 6042: 6043: 6044: 6045: 6046: 6047: 6048: 6049: 6050: 6051: 6052: 6053: 6054: 6055: 6056: 6057: 6058: 6059: 6060: 6061: 6062: 6063: 6064: 6065: 6066: 6067: 6068: 6069: 6070: 6071: 6072: 6073: 6074: 6075: 6076: 6077: 6078: 6079: 6080: 6081: 6082: 6083: 6084: 6085: 6086: 6087: 6088: 6089: 6090: 6091: 6092: 6093: 6094: 6095: 6096: 6097: 6098: 6099: 6100: 6101: 6102: 6103: 6104: 6105: 6106: 6107: 6108: 6109: 6110: 6111: 6112: 6113: 6114: 6115: 6116: 6117: 6118: 6119: 6120: 6121: 6122: 6123: 6124: 6125: 6126: 6127: 6128: 6129: 6130: 6131: 6132: 6133: 6134: 6135: 6136: 6137: 6138: 6139: 6140: 6141: 6142: 6143: 6144: 6145: 6146: 6147: 6148: 6149: 6150: 6151: 6152: 6153: 6154: 6155: 6156: 6157: 6158: 6159: 6160: 6161: 6162: 6163: 6164: 6165: 6166: 6167: 6168: 6169: 6170: 6171: 6172: 6173: 6174: 6175: 6176: 6177: 6178: 6179: 6180: 6181: 6182: 6183: 6184: 6185: 6186: 6187: 6188: 6189: 6190: 6191: 6192: 6193: 6194: 6195: 6196: 6197: 6198: 6199: 6200: 6201: 6202: 6203: 6204: 6205: 6206: 6207: 6208: 6209: 6210: 6211: 6212: 6213: 6214: 6215: 6216: 6217: 6218: 6219: 6220: 6221: 6222: 6223: 6224: 6225: 6226: 6227: 6228: 6229: 6230: 6231: 6232: 6233: 6234: 6235: 6236: 6237: 6238: 6239: 6240: 6241: 6242: 6243: 6244: 6245: 6246: 6247: 6248: 6249: 6250: 6251: 6252: 6253: 6254: 6255: 6256: 6257: 6258: 6259: 6260: 6261: 6262: 6263: 6264: 6265: 6266: 6267: 6268: 6269: 6270: 6271: 6272: 6273: 6274: 6275: 6276: 6277: 6278: 6279: 6280: 6281: 6282: 6283: 6284: 6285: 6286: 6287: 6288: 6289: 6290: 6291: 6292: 6293: 6294: 6295: 6296: 6297: 6298: 6299: 6300: 6301: 6302: 6303: 6304: 6305: 6306: 6307: 6308: 6309: 6310: 6311: 6312: 6313: 6314: 6315: 6316: 6317: 6318: 6319: 6320: 6321: 6322: 6323: 6324: 6325: 6326: 6327: 6328: 6329: 6330: 6331: 6332: 6333: 6334: 6335: 6336: 6337: 6338: 6339: 6340: 6341: 6342: 6343: 6344: 6345: 6346: 6347: 6348: 6349: 6350: 6351: 6352: 6353: 6354: 6355: 6356: 6357: 6358: 6359: 6360: 6361: 6362: 6363: 6364: 6365: 6366: 6367: 6368: 6369: 6370: 6371: 6372: 6373: 6374: 6375: 6376: 6377: 6378: 6379: 6380: 6381: 6382: 6383: 6384: 6385: 6386: 6387: 6388: 6389: 6390: 6391: 6392: 6393: 6394: 6395: 6396: 6397: 6398: 6399: 6400: 6401: 6402: 6403: 6404: 6405: 6406: 6407: 6408: 6409: 6410: 6411: 6412: 6413: 6414: 6415: 6416: 6417: 6418: 6419: 6420: 6421: 6422: 6423: 6424: 6425: 6426: 6427: 6428: 6429: 6430: 6431: 6432: 6433: 6434: 6435: 6436: 6437: 6438: 6439: 6440: 6441: 6442: 6443: 6444: 6445: 6446: 6447: 6448: 6449: 6450: 6451: 6452: 6453: 6454: 6455: 6456: 6457: 6458: 6459: 6460: 6461: 6462: 6463: 6464: 6465: 6466: 6467: 6468: 6469: 6470: 6471: 6472: 6473: 6474: 6475: 6476: 6477: 6478: 6479: 6480: 6481: 6482: 6483: 6484: 6485: 6486: 6487: 6488: 6489: 6490: 6491: 6492: 6493: 6494: 6495: 6496: 6497: 6498: 6499: 6500: 6501: 6502: 6503: 6504: 6505: 6506: 6507: 6508: 6509: 6510: 6511: 6512: 6513: 6514: 6515: 6516: 6517: 6518: 6519: 6520: 6521: 6522: 6523: 6524: 6525: 6526: 6527: 6528: 6529: 6530: 6531: 6532: 6533: 6534: 6535: 6536: 6537: 6538: 6539: 6540: 6541: 6542: 6543: 6544: 6545: 6546: 6547: 6548: 6549: 6550: 6551: 6552: 6553: 6554: 6555: 6556: 6557: 6558: 6559: 6560: 6561: 6562: 6563: 6564: 6565: 6566: 6567: 6568: 6569: 6570: 6571: 6572: 6573: 6574: 6575: 6576: 6577: 6578: 6579: 6580: 6581: 6582: 6583: 6584: 6585: 6586: 6587: 6588: 6589: 6590: 6591: 6592: 6593: 6594: 6595: 6596: 6597: 6598: 6599: 6600: 6601: 6602: 6603: 6604: 6605: 6606: 6607: 6608: 6609: 6610: 6611: 6612: 6613: 6614: 6615: 6616: 6617: 6618: 6619: 6620: 6621: 6622: 6623: 6624: 6625: 6626: 6627: 6628: 6629: 6630: 6631: 6632: 6633: 6634: 6635: 6636: 6637: 6638: 6639: 6640: 6641: 6642: 6643: 6644: 6645: 6646: 6647: 6648: 6649: 6650: 6651: 6652: 6653: 6654: 6655: 6656: 6657: 6658: 6659: 6660: 6661: 6662: 6663: 6664: 6665: 6666: 6667: 6668: 6669: 6670: 6671: 6672: 6673: 6674: 6675: 6676: 6677: 6678: 6679: 6680: 6681: 6682: 6683: 6684: 6685: 6686: 6687: 6688: 6689: 6690: 6691: 6692: 6693: 6694: 6695: 6696: 6697: 6698: 6699: 6700: 6701: 6702: 6703: 6704: 6705: 6706: 6707: 6708: 6709: 6710: 6711: 6712: 6713: 6714: 6715: 6716: 6717: 6718: 6719: 6720: 6721: 6722: 6723: 6724: 6725: 6726: 6727: 6728: 6729: 6730: 6731: 6732: 6733: 6734: 6735: 6736: 6737: 6738: 6739: 6740: 6741: 6742: 6743: 6744: 6745: 6746: 6747: 6748: 6749: 6750: 6751: 6752: 6753: 6754: 6755: 6756: 6757: 6758: 6759: 6760: 6761: 6762: 6763: 6764: 6765: 6766: 6767: 6768: 6769: 6770: 6771: 6772: 6773: 6774: 6775: 6776: 6777: 6778: 6779: 6780: 6781: 6782: 6783: 6784: 6785: 6786: 6787: 6788: 6789: 6790: 6791: 6792: 6793: 6794: 6795: 6796: 6797: 6798: 6799: 6800: 6801: 6802: 6803: 6804: 6805: 6806: 6807: 6808: 6809: 6810: 6811: 6812: 6813: 6814: 6815: 6816: 6817: 6818: 6819: 6820: 6821: 6822: 6823: 6824: 6825: 6826: 6827: 6828: 6829: 6830: 6831: 6832: 6833: 6834: 6835: 6836: 6837: 6838: 6839: 6840: 6841: 6842: 6843: 6844: 6845: 6846: 6847: 6848: 6849: 6850: 6851: 6852: 6853: 6854: 6855: 6856: 6857: 6858: 6859: 6860: 6861: 6862: 6863: 6864: 6865: 6866: 6867: 6868: 6869: 6870: 6871: 6872: 6873: 6874: 6875: 6876: 6877: 6878: 6879: 6880: 6881: 6882: 6883: 6884: 6885: 6886: 6887: 6888: 6889: 6890: 6891: 6892: 6893: 6894: 6895: 6896: 6897: 6898: 6899: 6900: 6901: 6902: 6903: 6904: 6905: 6906: 6907: 6908: 6909: 6910: 6911: 6912: 6913: 6914: 6915: 6916: 6917: 6918: 6919: 6920: 6921: 6922: 6923: 6924: 6925: 6926: 6927: 6928: 6929: 6930: 6931: 6932: 6933: 6934: 6935: 6936: 6937: 6938: 6939: 6940: 6941: 6942: 6943: 6944: 6945: 6946: 6947: 6948: 6949: 6950: 6951: 6952: 6953: 6954: 6955: 6956: 6957: 6958: 6959: 6960: 6961: 6962: 6963: 6964: 6965: 6966: 6967: 6968: 6969: 6970: 6971: 6972: 6973: 6974: 6975: 6976: 6977: 6978: 6979: 6980: 6981: 6982: 6983: 6984: 6985: 6986: 6987: 6988: 6989: 6990: 6991: 6992: 6993: 6994: 6995: 6996: 6997: 6998: 6999: 7000: 7001: 7002: 7003: 7004: 7005: 7006: 7007: 7008: 7009: 7010: 7011: 7012: 7013: 7014: 7015: 7016: 7017: 7018: 7019: 7020: 7021: 7022: 7023: 7024: 7025: 7026: 7027: 7028: 7029: 7030: 7031: 7032: 7033: 7034: 7035: 7036: 7037: 7038: 7039: 7040: 7041: 7042: 7043: 7044: 7045: 7046: 7047: 7048: 7049: 7050: 7051: 7052: 7053: 7054: 7055: 7056: 7057: 7058: 7059: 7060: 7061: 7062: 7063: 7064: 7065: 7066: 7067: 7068: 7069: 7070: 7071: 7072: 7073: 7074: 7075: 7076: 7077: 7078: 7079: 7080: 7081: 7082: 7083: 7084: 7085: 7086: 7087: 7088: 7089: 7090: 7091: 7092: 7093: 7094: 7095: 7096: 7097: 7098: 7099: 7100: 7101: 7102: 7103: 7104: 7105: 7106: 7107: 7108: 7109: 7110: 7111: 7112: 7113: 7114: 7115: 7116: 7117: 7118: 7119: 7120: 7121: 7122: 7123: 7124: 7125: 7126: 7127: 7128: 7129: 7130: 7131: 7132: 7133: 7134: 7135: 7136: 7137: 7138: 7139: 7140: 7141: 7142: 7143: 7144: 7145: 7146: 7147: 7148: 7149: 7150: 7151: 7152: 7153: 7154: 7155: 7156: 7157: 7158: 7159: 7160: 7161: 7162: 7163: 7164: 7165: 7166: 7167: 7168: 7169: 7170: 7171: 7172: 7173: 7174: 7175: 7176: 7177: 7178: 7179: 7180: 7181: 7182: 7183: 7184: 7185: 7186: 7187: 7188: 7189: 7190: 7191: 7192: 7193: 7194: 7195: 7196: 7197: 7198:
<?php
if (!defined('SMF'))
die('No direct access...');
function updateStats($type, $parameter1 = null, $parameter2 = null)
{
global $modSettings, $smcFunc;
switch ($type)
{
case 'member':
$changes = array(
'memberlist_updated' => time(),
);
if (is_numeric($parameter1))
{
$changes['latestMember'] = $parameter1;
$changes['latestRealName'] = $parameter2;
updateSettings(array('totalMembers' => true), true);
}
else
{
$result = $smcFunc['db_query']('', '
SELECT COUNT(*), MAX(id_member)
FROM {db_prefix}members
WHERE is_activated = {int:is_activated}',
array(
'is_activated' => 1,
)
);
list ($changes['totalMembers'], $changes['latestMember']) = $smcFunc['db_fetch_row']($result);
$smcFunc['db_free_result']($result);
$result = $smcFunc['db_query']('', '
SELECT real_name
FROM {db_prefix}members
WHERE id_member = {int:id_member}
LIMIT 1',
array(
'id_member' => (int) $changes['latestMember'],
)
);
list ($changes['latestRealName']) = $smcFunc['db_fetch_row']($result);
$smcFunc['db_free_result']($result);
$result = $smcFunc['db_query']('', '
SELECT COUNT(*)
FROM {db_prefix}members
WHERE is_activated IN ({array_int:activation_status})',
array(
'activation_status' => array(3, 4, 5),
)
);
list ($changes['unapprovedMembers']) = $smcFunc['db_fetch_row']($result);
$smcFunc['db_free_result']($result);
}
updateSettings($changes);
break;
case 'message':
if ($parameter1 === true && $parameter2 !== null)
updateSettings(array('totalMessages' => true, 'maxMsgID' => $parameter2), true);
else
{
$result = $smcFunc['db_query']('', '
SELECT SUM(num_posts + unapproved_posts) AS total_messages, MAX(id_last_msg) AS max_msg_id
FROM {db_prefix}boards
WHERE redirect = {string:blank_redirect}' . (!empty($modSettings['recycle_enable']) && $modSettings['recycle_board'] > 0 ? '
AND id_board != {int:recycle_board}' : ''),
array(
'recycle_board' => isset($modSettings['recycle_board']) ? $modSettings['recycle_board'] : 0,
'blank_redirect' => '',
)
);
$row = $smcFunc['db_fetch_assoc']($result);
$smcFunc['db_free_result']($result);
updateSettings(array(
'totalMessages' => $row['total_messages'] === null ? 0 : $row['total_messages'],
'maxMsgID' => $row['max_msg_id'] === null ? 0 : $row['max_msg_id']
));
}
break;
case 'subject':
$smcFunc['db_query']('', '
DELETE FROM {db_prefix}log_search_subjects
WHERE id_topic = {int:id_topic}',
array(
'id_topic' => (int) $parameter1,
)
);
if ($parameter2 !== null)
{
$parameter1 = (int) $parameter1;
$parameter2 = text2words($parameter2);
$inserts = array();
foreach ($parameter2 as $word)
$inserts[] = array($word, $parameter1);
if (!empty($inserts))
$smcFunc['db_insert']('ignore',
'{db_prefix}log_search_subjects',
array('word' => 'string', 'id_topic' => 'int'),
$inserts,
array('word', 'id_topic')
);
}
break;
case 'topic':
if ($parameter1 === true)
updateSettings(array('totalTopics' => true), true);
else
{
$result = $smcFunc['db_query']('', '
SELECT SUM(num_topics + unapproved_topics) AS total_topics
FROM {db_prefix}boards' . (!empty($modSettings['recycle_enable']) && $modSettings['recycle_board'] > 0 ? '
WHERE id_board != {int:recycle_board}' : ''),
array(
'recycle_board' => !empty($modSettings['recycle_board']) ? $modSettings['recycle_board'] : 0,
)
);
$row = $smcFunc['db_fetch_assoc']($result);
$smcFunc['db_free_result']($result);
updateSettings(array('totalTopics' => $row['total_topics'] === null ? 0 : $row['total_topics']));
}
break;
case 'postgroups':
if ($parameter2 !== null && !in_array('posts', $parameter2))
return;
$postgroups = cache_get_data('updateStats:postgroups', 360);
if ($postgroups == null || $parameter1 == null)
{
$request = $smcFunc['db_query']('', '
SELECT id_group, min_posts
FROM {db_prefix}membergroups
WHERE min_posts != {int:min_posts}',
array(
'min_posts' => -1,
)
);
$postgroups = array();
while ($row = $smcFunc['db_fetch_assoc']($request))
$postgroups[$row['id_group']] = $row['min_posts'];
$smcFunc['db_free_result']($request);
arsort($postgroups);
cache_put_data('updateStats:postgroups', $postgroups, 360);
}
if (empty($postgroups))
return;
$conditions = '';
$lastMin = 0;
foreach ($postgroups as $id => $min_posts)
{
$conditions .= '
WHEN posts >= ' . $min_posts . (!empty($lastMin) ? ' AND posts <= ' . $lastMin : '') . ' THEN ' . $id;
$lastMin = $min_posts;
}
$smcFunc['db_query']('', '
UPDATE {db_prefix}members
SET id_post_group = CASE ' . $conditions . '
ELSE 0
END' . ($parameter1 != null ? '
WHERE ' . (is_array($parameter1) ? 'id_member IN ({array_int:members})' : 'id_member = {int:members}') : ''),
array(
'members' => $parameter1,
)
);
break;
default:
trigger_error('updateStats(): Invalid statistic type \'' . $type . '\'', E_USER_NOTICE);
}
}
function updateMemberData($members, $data)
{
global $modSettings, $user_info, $smcFunc, $sourcedir, $cache_enable;
if ($members === array())
return;
$parameters = array();
if (is_array($members))
{
$condition = 'id_member IN ({array_int:members})';
$parameters['members'] = $members;
}
elseif ($members === null)
$condition = '1=1';
else
{
$condition = 'id_member = {int:member}';
$parameters['member'] = $members;
}
$knownInts = array(
'date_registered', 'posts', 'id_group', 'last_login', 'instant_messages', 'unread_messages',
'new_pm', 'pm_prefs', 'gender', 'show_online', 'pm_receive_from', 'alerts',
'id_theme', 'is_activated', 'id_msg_last_visit', 'id_post_group', 'total_time_logged_in', 'warning',
);
$knownFloats = array(
'time_offset',
);
if (!empty($modSettings['integrate_change_member_data']))
{
$integration_vars = array(
'member_name',
'real_name',
'email_address',
'id_group',
'gender',
'birthdate',
'website_title',
'website_url',
'location',
'time_format',
'time_offset',
'avatar',
'lngfile',
);
$vars_to_integrate = array_intersect($integration_vars, array_keys($data));
if (count($vars_to_integrate) != 0)
{
if ((!is_array($members) && $members === $user_info['id']) || (is_array($members) && count($members) == 1 && in_array($user_info['id'], $members)))
$member_names = array($user_info['username']);
else
{
$member_names = array();
$request = $smcFunc['db_query']('', '
SELECT member_name
FROM {db_prefix}members
WHERE ' . $condition,
$parameters
);
while ($row = $smcFunc['db_fetch_assoc']($request))
$member_names[] = $row['member_name'];
$smcFunc['db_free_result']($request);
}
if (!empty($member_names))
foreach ($vars_to_integrate as $var)
call_integration_hook('integrate_change_member_data', array($member_names, $var, &$data[$var], &$knownInts, &$knownFloats));
}
}
$setString = '';
foreach ($data as $var => $val)
{
switch ($var)
{
case 'birthdate':
$type = 'date';
break;
case 'member_ip':
case 'member_ip2':
$type = 'inet';
break;
default:
$type = 'string';
}
if (in_array($var, $knownInts))
$type = 'int';
elseif (in_array($var, $knownFloats))
$type = 'float';
if ($var == 'alerts' && ($val === '+' || $val === '-'))
{
include_once($sourcedir . '/Profile-Modify.php');
if (is_array($members))
{
$val = 'CASE ';
foreach ($members as $k => $v)
$val .= 'WHEN id_member = ' . $v . ' THEN '. alert_count($v, true) . ' ';
$val = $val . ' END';
$type = 'raw';
}
else
$val = alert_count($members, true);
}
elseif ($type == 'int' && ($val === '+' || $val === '-'))
{
$val = $var . ' ' . $val . ' 1';
$type = 'raw';
}
if (in_array($var, array('posts', 'instant_messages', 'unread_messages')))
{
if (preg_match('~^' . $var . ' (\+ |- |\+ -)([\d]+)~', $val, $match))
{
if ($match[1] != '+ ')
$val = 'CASE WHEN ' . $var . ' <= ' . abs($match[2]) . ' THEN 0 ELSE ' . $val . ' END';
$type = 'raw';
}
}
$setString .= ' ' . $var . ' = {' . $type . ':p_' . $var . '},';
$parameters['p_' . $var] = $val;
}
$smcFunc['db_query']('', '
UPDATE {db_prefix}members
SET' . substr($setString, 0, -1) . '
WHERE ' . $condition,
$parameters
);
updateStats('postgroups', $members, array_keys($data));
if (!empty($cache_enable) && $cache_enable >= 2 && !empty($members))
{
if (!is_array($members))
$members = array($members);
foreach ($members as $member)
{
if ($cache_enable >= 3)
{
cache_put_data('member_data-profile-' . $member, null, 120);
cache_put_data('member_data-normal-' . $member, null, 120);
cache_put_data('member_data-minimal-' . $member, null, 120);
}
cache_put_data('user_settings-' . $member, null, 60);
}
}
}
function updateSettings($changeArray, $update = false)
{
global $modSettings, $smcFunc;
if (empty($changeArray) || !is_array($changeArray))
return;
$toRemove = array();
foreach ($changeArray as $k => $v)
if ($v === null)
{
unset($changeArray[$k]);
$toRemove[] = $k;
}
if (!empty($toRemove))
$smcFunc['db_query']('', '
DELETE FROM {db_prefix}settings
WHERE variable IN ({array_string:remove})',
array(
'remove' => $toRemove,
)
);
if ($update)
{
foreach ($changeArray as $variable => $value)
{
$smcFunc['db_query']('', '
UPDATE {db_prefix}settings
SET value = {' . ($value === false || $value === true ? 'raw' : 'string') . ':value}
WHERE variable = {string:variable}',
array(
'value' => $value === true ? 'value + 1' : ($value === false ? 'value - 1' : $value),
'variable' => $variable,
)
);
$modSettings[$variable] = $value === true ? $modSettings[$variable] + 1 : ($value === false ? $modSettings[$variable] - 1 : $value);
}
cache_put_data('modSettings', null, 90);
return;
}
$replaceArray = array();
foreach ($changeArray as $variable => $value)
{
if (isset($modSettings[$variable]) && $modSettings[$variable] == $value)
continue;
elseif (!isset($modSettings[$variable]) && empty($value))
continue;
$replaceArray[] = array($variable, $value);
$modSettings[$variable] = $value;
}
if (empty($replaceArray))
return;
$smcFunc['db_insert']('replace',
'{db_prefix}settings',
array('variable' => 'string-255', 'value' => 'string-65534'),
$replaceArray,
array('variable')
);
cache_put_data('modSettings', null, 90);
}
function constructPageIndex($base_url, &$start, $max_value, $num_per_page, $flexible_start = false, $show_prevnext = true)
{
global $modSettings, $context, $smcFunc, $settings, $txt;
$start = (int) $start;
$start_invalid = $start < 0;
if ($start_invalid)
$start = 0;
elseif ($start >= $max_value)
$start = max(0, (int) $max_value - (((int) $max_value % (int) $num_per_page) == 0 ? $num_per_page : ((int) $max_value % (int) $num_per_page)));
else
$start = max(0, (int) $start - ((int) $start % (int) $num_per_page));
$context['current_page'] = $start / $num_per_page;
if (!isset($settings['page_index']))
{
$settings['page_index'] = array(
'extra_before' => '<span class="pages">' . $txt['pages'] . '</span>',
'previous_page' => '<span class="main_icons previous_page"></span>',
'current_page' => '<span class="current_page">%1$d</span> ',
'page' => '<a class="nav_page" href="{URL}">%2$s</a> ',
'expand_pages' => '<span class="expand_pages" onclick="expandPages(this, {LINK}, {FIRST_PAGE}, {LAST_PAGE}, {PER_PAGE});"> ... </span>',
'next_page' => '<span class="main_icons next_page"></span>',
'extra_after' => '',
);
}
$base_link = strtr($settings['page_index']['page'], array('{URL}' => $flexible_start ? $base_url : strtr($base_url, array('%' => '%%')) . ';start=%1$d'));
$pageindex = $settings['page_index']['extra_before'];
if (empty($modSettings['compactTopicPagesEnable']))
{
$pageindex .= $start == 0 ? ' ' : sprintf($base_link, $start - $num_per_page, $settings['page_index']['previous_page']);
$display_page = 1;
for ($counter = 0; $counter < $max_value; $counter += $num_per_page)
$pageindex .= $start == $counter && !$start_invalid ? sprintf($settings['page_index']['current_page'], $display_page++) : sprintf($base_link, $counter, $display_page++);
$display_page = ($start + $num_per_page) > $max_value ? $max_value : ($start + $num_per_page);
if ($start != $counter - $max_value && !$start_invalid)
$pageindex .= $display_page > $counter - $num_per_page ? ' ' : sprintf($base_link, $display_page, $settings['page_index']['next_page']);
}
else
{
$PageContiguous = (int) ($modSettings['compactTopicPagesContiguous'] - ($modSettings['compactTopicPagesContiguous'] % 2)) / 2;
if (!empty($start) && $show_prevnext)
$pageindex .= sprintf($base_link, $start - $num_per_page, $settings['page_index']['previous_page']);
else
$pageindex .= '';
if ($start > $num_per_page * $PageContiguous)
$pageindex .= sprintf($base_link, 0, '1');
if ($start > $num_per_page * ($PageContiguous + 1))
$pageindex .= strtr($settings['page_index']['expand_pages'], array(
'{LINK}' => JavaScriptEscape($smcFunc['htmlspecialchars']($base_link)),
'{FIRST_PAGE}' => $num_per_page,
'{LAST_PAGE}' => $start - $num_per_page * $PageContiguous,
'{PER_PAGE}' => $num_per_page,
));
for ($nCont = $PageContiguous; $nCont >= 1; $nCont--)
if ($start >= $num_per_page * $nCont)
{
$tmpStart = $start - $num_per_page * $nCont;
$pageindex .= sprintf($base_link, $tmpStart, $tmpStart / $num_per_page + 1);
}
if (!$start_invalid)
$pageindex .= sprintf($settings['page_index']['current_page'], $start / $num_per_page + 1);
else
$pageindex .= sprintf($base_link, $start, $start / $num_per_page + 1);
$tmpMaxPages = (int) (($max_value - 1) / $num_per_page) * $num_per_page;
for ($nCont = 1; $nCont <= $PageContiguous; $nCont++)
if ($start + $num_per_page * $nCont <= $tmpMaxPages)
{
$tmpStart = $start + $num_per_page * $nCont;
$pageindex .= sprintf($base_link, $tmpStart, $tmpStart / $num_per_page + 1);
}
if ($start + $num_per_page * ($PageContiguous + 1) < $tmpMaxPages)
$pageindex .= strtr($settings['page_index']['expand_pages'], array(
'{LINK}' => JavaScriptEscape($smcFunc['htmlspecialchars']($base_link)),
'{FIRST_PAGE}' => $start + $num_per_page * ($PageContiguous + 1),
'{LAST_PAGE}' => $tmpMaxPages,
'{PER_PAGE}' => $num_per_page,
));
if ($start + $num_per_page * $PageContiguous < $tmpMaxPages)
$pageindex .= sprintf($base_link, $tmpMaxPages, $tmpMaxPages / $num_per_page + 1);
if ($start != $tmpMaxPages && $show_prevnext)
$pageindex .= sprintf($base_link, $start + $num_per_page, $settings['page_index']['next_page']);
}
$pageindex .= $settings['page_index']['extra_after'];
return $pageindex;
}
function comma_format($number, $override_decimal_count = false)
{
global $txt;
static $thousands_separator = null, $decimal_separator = null, $decimal_count = null;
if ($decimal_separator === null)
{
if (empty($txt['number_format']) || preg_match('~^1([^\d]*)?234([^\d]*)(0*?)$~', $txt['number_format'], $matches) != 1)
return $number;
$thousands_separator = $matches[1];
$decimal_separator = $matches[2];
$decimal_count = strlen($matches[3]);
}
return number_format($number, (float) $number === $number ? ($override_decimal_count === false ? $decimal_count : $override_decimal_count) : 0, $decimal_separator, $thousands_separator);
}
function timeformat($log_time, $show_today = true, $offset_type = false, $process_safe = false)
{
global $context, $user_info, $txt, $modSettings;
static $non_twelve_hour, $locale, $now;
static $unsupportedFormats, $finalizedFormats;
$unsupportedFormatsWindows = array('z', 'Z');
$user_info['time_offset'] = !empty($user_info['time_offset']) ? $user_info['time_offset'] : 0;
$modSettings['time_offset'] = !empty($modSettings['time_offset']) ? $modSettings['time_offset'] : 0;
$user_info['time_format'] = !empty($user_info['time_format']) ? $user_info['time_format'] : (!empty($modSettings['time_format']) ? $modSettings['time_format'] : '%F %H:%M');
if (!$offset_type)
$log_time = $log_time + ($user_info['time_offset'] + $modSettings['time_offset']) * 3600;
elseif ($offset_type == 'forum')
$log_time = $log_time + $modSettings['time_offset'] * 3600;
if ($log_time < 0)
$log_time = 0;
$prefix = '';
if ($modSettings['todayMod'] >= 1 && $show_today === true)
{
$now_time = forum_time();
if ($now_time - $log_time < (86400 * $modSettings['todayMod']))
{
$then = @getdate($log_time);
$now = (!empty($now) ? $now : @getdate($now_time));
if ($then['yday'] == $now['yday'] && $then['year'] == $now['year'])
{
$prefix = $txt['today'];
}
elseif ($modSettings['todayMod'] == '2' && (($then['yday'] == $now['yday'] - 1 && $then['year'] == $now['year']) || ($now['yday'] == 0 && $then['year'] == $now['year'] - 1) && $then['mon'] == 12 && $then['mday'] == 31))
{
$prefix = $txt['yesterday'];
}
}
}
$str = !is_bool($show_today) ? $show_today : $user_info['time_format'];
if (is_null($finalizedFormats))
$finalizedFormats = (array) cache_get_data('timeformatstrings', 86400);
if (!isset($finalizedFormats[$str]) || !is_array($finalizedFormats[$str]))
$finalizedFormats[$str] = array();
$format_type = !empty($prefix) ? 'time_only' : 'normal';
if (empty($finalizedFormats[$str][$format_type]))
{
$timeformat = $format_type == 'time_only' ? get_date_or_time_format('time', $str) : $str;
$strftimeFormatSubstitutions = array(
'a' => '#txt_days_short_%w#', 'A' => '#txt_days_%w#', 'e' => '%d', 'd' => '%d', 'j' => '%j', 'u' => '%w', 'w' => '%w',
'U' => '%U', 'V' => '%U', 'W' => '%U',
'b' => '#txt_months_short_%m#', 'B' => '#txt_months_%m#', 'h' => '%b', 'm' => '%m',
'C' => '%C', 'g' => '%y', 'G' => '%Y', 'y' => '%y', 'Y' => '%Y',
'H' => '%H', 'k' => '%H', 'I' => '%H', 'l' => '%I', 'M' => '%M', 'p' => '%p', 'P' => '%p',
'r' => '%I:%M:%S %p', 'R' => '%H:%M', 'S' => '%S', 'T' => '%H:%M:%S', 'X' => '%T', 'z' => '%z', 'Z' => '%Z',
'c' => '%F %T', 'D' => '%m/%d/%y', 'F' => '%Y-%m-%d', 's' => '%s', 'x' => '%F',
'n' => "\n", 't' => "\t", '%' => '%',
);
if (is_null($unsupportedFormats))
$unsupportedFormats = (array) cache_get_data('unsupportedtimeformats', 86400);
if (empty($unsupportedFormats))
{
foreach ($strftimeFormatSubstitutions as $format => $substitution)
{
if ($context['server']['is_windows'] && in_array($format, $unsupportedFormatsWindows))
{
$unsupportedFormats[] = $format;
continue;
}
$value = @strftime('%' . $format);
if ($value === false || $value === $format)
$unsupportedFormats[] = $format;
}
cache_put_data('unsupportedtimeformats', $unsupportedFormats, 86400);
}
if (DIRECTORY_SEPARATOR === '\\')
$timeformat = preg_replace('~%(?!' . implode('|', array_keys($strftimeFormatSubstitutions)) . ')~', '%', $timeformat);
if (!empty($unsupportedFormats))
while (preg_match('~%(' . implode('|', $unsupportedFormats) . ')~', $timeformat, $matches))
$timeformat = str_replace($matches[0], $strftimeFormatSubstitutions[$matches[1]], $timeformat);
$finalizedFormats[$str][$format_type] = $timeformat;
cache_put_data('timeformatstrings', $finalizedFormats, 86400);
}
$timeformat = $finalizedFormats[$str][$format_type];
if (!isset($locale) || ($process_safe === true && setlocale(LC_TIME, '0') != $locale))
$locale = setlocale(LC_TIME, array($txt['lang_locale'] . '.' . $modSettings['global_character_set'], $txt['lang_locale'] . '.' . $txt['lang_character_set'], $txt['lang_locale']));
if ($locale === false)
{
$timeformat = strtr($timeformat, array(
'%a' => '#txt_days_short_%w#',
'%A' => '#txt_days_%w#',
'%b' => '#txt_months_short_%m#',
'%B' => '#txt_months_%m#',
'%p' => '%p',
'%P' => '%p'
));
}
else
{
if (!isset($non_twelve_hour) && strpos($timeformat, '%p') !== false)
$non_twelve_hour = trim(strftime('%p')) === '';
if (!empty($non_twelve_hour))
$timeformat = strtr($timeformat, array(
'%p' => '%p',
'%P' => '%p'
));
}
$timestring = strftime($timeformat, $log_time);
if (strpos($timestring, '%p') !== false)
$timestring = str_replace('%p', (strftime('%H', $log_time) < 12 ? $txt['time_am'] : $txt['time_pm']), $timestring);
if (strpos($timestring, '#txt_') !== false)
{
if (strpos($timestring, '#txt_days_short_') !== false)
$timestring = strtr($timestring, array(
'#txt_days_short_0#' => $txt['days_short'][0],
'#txt_days_short_1#' => $txt['days_short'][1],
'#txt_days_short_2#' => $txt['days_short'][2],
'#txt_days_short_3#' => $txt['days_short'][3],
'#txt_days_short_4#' => $txt['days_short'][4],
'#txt_days_short_5#' => $txt['days_short'][5],
'#txt_days_short_6#' => $txt['days_short'][6],
));
if (strpos($timestring, '#txt_days_') !== false)
$timestring = strtr($timestring, array(
'#txt_days_0#' => $txt['days'][0],
'#txt_days_1#' => $txt['days'][1],
'#txt_days_2#' => $txt['days'][2],
'#txt_days_3#' => $txt['days'][3],
'#txt_days_4#' => $txt['days'][4],
'#txt_days_5#' => $txt['days'][5],
'#txt_days_6#' => $txt['days'][6],
));
if (strpos($timestring, '#txt_months_short_') !== false)
$timestring = strtr($timestring, array(
'#txt_months_short_01#' => $txt['months_short'][1],
'#txt_months_short_02#' => $txt['months_short'][2],
'#txt_months_short_03#' => $txt['months_short'][3],
'#txt_months_short_04#' => $txt['months_short'][4],
'#txt_months_short_05#' => $txt['months_short'][5],
'#txt_months_short_06#' => $txt['months_short'][6],
'#txt_months_short_07#' => $txt['months_short'][7],
'#txt_months_short_08#' => $txt['months_short'][8],
'#txt_months_short_09#' => $txt['months_short'][9],
'#txt_months_short_10#' => $txt['months_short'][10],
'#txt_months_short_11#' => $txt['months_short'][11],
'#txt_months_short_12#' => $txt['months_short'][12],
));
if (strpos($timestring, '#txt_months_') !== false)
$timestring = strtr($timestring, array(
'#txt_months_01#' => $txt['months'][1],
'#txt_months_02#' => $txt['months'][2],
'#txt_months_03#' => $txt['months'][3],
'#txt_months_04#' => $txt['months'][4],
'#txt_months_05#' => $txt['months'][5],
'#txt_months_06#' => $txt['months'][6],
'#txt_months_07#' => $txt['months'][7],
'#txt_months_08#' => $txt['months'][8],
'#txt_months_09#' => $txt['months'][9],
'#txt_months_10#' => $txt['months'][10],
'#txt_months_11#' => $txt['months'][11],
'#txt_months_12#' => $txt['months'][12],
));
}
return $prefix . str_replace('%', '%', $timestring);
}
function get_date_or_time_format($type = '', $format = '')
{
global $user_info, $modSettings;
static $formats;
if (strpos($format, '%') === false)
$format = !empty($user_info['time_format']) ? $user_info['time_format'] : (!empty($modSettings['time_format']) ? $modSettings['time_format'] : '%F %k:%M');
$orig_format = $format;
if (isset($formats[$orig_format][$type]))
return $formats[$orig_format][$type];
if ($type === 'date')
{
$specifications = array(
'%a' => '%a', '%A' => '%A', '%e' => '%e', '%d' => '%d', '%j' => '%j', '%u' => '%u', '%w' => '%w',
'%U' => '%U', '%V' => '%V', '%W' => '%W',
'%b' => '%b', '%B' => '%B', '%h' => '%h', '%m' => '%m',
'%C' => '%C', '%g' => '%g', '%G' => '%G', '%y' => '%y', '%Y' => '%Y',
'%H' => '', '%k' => '', '%I' => '', '%l' => '', '%M' => '', '%p' => '', '%P' => '',
'%r' => '', '%R' => '', '%S' => '', '%T' => '', '%X' => '', '%z' => '', '%Z' => '',
'%c' => '%x', '%D' => '%D', '%F' => '%F', '%s' => '%s', '%x' => '%x',
'%n' => '', '%t' => '', '%%' => '%%',
);
$default_format = '%F';
}
elseif ($type === 'time')
{
$specifications = array(
'%a' => '', '%A' => '', '%e' => '', '%d' => '', '%j' => '', '%u' => '', '%w' => '',
'%U' => '', '%V' => '', '%W' => '',
'%b' => '', '%B' => '', '%h' => '', '%m' => '',
'%C' => '', '%g' => '', '%G' => '', '%y' => '', '%Y' => '',
'%H' => '%H', '%k' => '%k', '%I' => '%I', '%l' => '%l', '%M' => '%M', '%p' => '%p', '%P' => '%P',
'%r' => '%r', '%R' => '%R', '%S' => '%S', '%T' => '%T', '%X' => '%X', '%z' => '%z', '%Z' => '%Z',
'%c' => '%X', '%D' => '', '%F' => '', '%s' => '%s', '%x' => '',
'%n' => '', '%t' => '', '%%' => '%%',
);
$default_format = '%k:%M';
}
else
return $format;
$wanted = array_filter($specifications);
$unwanted = array_diff(array_keys($specifications), $wanted);
$format = strtr($format, $wanted);
$format_parts = preg_split('~%[' . (strtr(implode('', $unwanted), array('%' => ''))) . ']~u', $format);
foreach ($format_parts as $p => $f)
{
if (strpos($f, '%') === false)
unset($format_parts[$p]);
}
$format = implode('', $format_parts);
$format = preg_replace(
array(
'~(?<!%)\p{L}|[^\p{L}\p{P}\s]~u',
'~([^%\P{P}])(\s*)(?'.'>(\1|[^%\P{Po}])\s*(?!$))*~u',
'~(?'.'>([\p{Pd}\p{Ps}\p{Pi}\p{Pc}]|[^%\P{Po}])\s*)*$~u',
'~^\s*(?'.'>([\p{Pd}\p{Pe}\p{Pf}\p{Pc}]|[^%\P{Po}])\s*)*~u',
),
array(
'',
'$1$2',
'',
'',
),
$format
);
if (empty($format))
$format = $default_format;
$formats[$orig_format][$type] = trim($format);
return $formats[$orig_format][$type];
}
function un_htmlspecialchars($string)
{
global $context;
static $translation = array();
if (empty($context['character_set']))
$charset = 'UTF-8';
elseif (strpos($context['character_set'], 'ISO-8859-') !== false && !in_array($context['character_set'], array('ISO-8859-5', 'ISO-8859-15')))
$charset = 'ISO-8859-1';
else
$charset = $context['character_set'];
if (empty($translation))
$translation = array_flip(get_html_translation_table(HTML_SPECIALCHARS, ENT_QUOTES, $charset)) + array(''' => '\'', ''' => '\'', ' ' => ' ');
return strtr($string, $translation);
}
function shorten_subject($subject, $len)
{
global $smcFunc;
if ($smcFunc['strlen']($subject) <= $len)
return $subject;
return $smcFunc['substr']($subject, 0, $len) . '...';
}
function forum_time($use_user_offset = true, $timestamp = null)
{
global $user_info, $modSettings;
if ($timestamp === null)
$timestamp = time();
elseif ($timestamp == 0)
return 0;
return $timestamp + ($modSettings['time_offset'] + ($use_user_offset ? $user_info['time_offset'] : 0)) * 3600;
}
function permute($array)
{
$orders = array($array);
$n = count($array);
$p = range(0, $n);
for ($i = 1; $i < $n; null)
{
$p[$i]--;
$j = $i % 2 != 0 ? $p[$i] : 0;
$temp = $array[$i];
$array[$i] = $array[$j];
$array[$j] = $temp;
for ($i = 1; $p[$i] == 0; $i++)
$p[$i] = 1;
$orders[] = $array;
}
return $orders;
}
function parse_bbc($message, $smileys = true, $cache_id = '', $parse_tags = array())
{
global $smcFunc, $txt, $scripturl, $context, $modSettings, $user_info, $sourcedir, $cache_enable;
static $bbc_lang_locales = array(), $itemcodes = array(), $no_autolink_tags = array();
static $disabled, $alltags_regex = '', $param_regexes = array();
if ($message === '')
return '';
if (!isset($context['utf8']))
$context['utf8'] = (empty($modSettings['global_character_set']) ? $txt['lang_character_set'] : $modSettings['global_character_set']) === 'UTF-8';
$message = sanitizeMSCutPaste($message);
if (!empty($context['load_average']) && !empty($modSettings['bbc']) && $context['load_average'] >= $modSettings['bbc'])
{
$context['disabled_parse_bbc'] = true;
return $message;
}
if ($smileys !== null && ($smileys == '1' || $smileys == '0'))
$smileys = (bool) $smileys;
if (empty($modSettings['enableBBC']) && $message !== false)
{
if ($smileys === true)
parsesmileys($message);
return $message;
}
if (!empty($bbc_lang_locales[$txt['lang_locale']]))
$bbc_codes = $bbc_lang_locales[$txt['lang_locale']];
else
$bbc_codes = array();
if (!empty($parse_tags))
$bbc_codes = array();
if (!empty($modSettings['autoLinkUrls']))
set_tld_regex();
call_integration_hook('integrate_pre_parsebbc', array(&$message, &$smileys, &$cache_id, &$parse_tags));
if (empty($bbc_codes) || $message === false || !empty($parse_tags))
{
if (!empty($modSettings['disabledBBC']))
{
$disabled = array();
$temp = explode(',', strtolower($modSettings['disabledBBC']));
foreach ($temp as $tag)
$disabled[trim($tag)] = true;
if (in_array('color', $disabled))
$disabled = array_merge($disabled, array(
'black' => true,
'white' => true,
'red' => true,
'green' => true,
'blue' => true,
)
);
}
$scripturl_parts = parse_url($scripturl);
$hosturl = $scripturl_parts['scheme'] . '://' . $scripturl_parts['host'];
$codes = array(
array(
'tag' => 'abbr',
'type' => 'unparsed_equals',
'before' => '<abbr title="$1">',
'after' => '</abbr>',
'quoted' => 'optional',
'disabled_after' => ' ($1)',
),
array(
'tag' => 'acronym',
'type' => 'unparsed_equals',
'before' => '<abbr title="$1">',
'after' => '</abbr>',
'quoted' => 'optional',
'disabled_after' => ' ($1)',
),
array(
'tag' => 'anchor',
'type' => 'unparsed_equals',
'test' => '[#]?([A-Za-z][A-Za-z0-9_\-]*)\]',
'before' => '<span id="post_$1">',
'after' => '</span>',
),
array(
'tag' => 'attach',
'type' => 'unparsed_content',
'parameters' => array(
'id' => array('match' => '(\d+)'),
'alt' => array('optional' => true),
'width' => array('optional' => true, 'match' => '(\d+)'),
'height' => array('optional' => true, 'match' => '(\d+)'),
'display' => array('optional' => true, 'match' => '(link|embed)'),
),
'content' => '$1',
'validate' => function(&$tag, &$data, $disabled, $params) use ($modSettings, $context, $sourcedir, $txt, $smcFunc)
{
$returnContext = '';
if (empty($modSettings['attachmentEnable']) || !empty($disabled['attach']))
return $data;
$attachID = $params['{id}'];
require_once($sourcedir . '/Subs-Attachments.php');
$currentAttachment = parseAttachBBC($attachID);
if (is_string($currentAttachment))
return $data = !empty($txt[$currentAttachment]) ? $txt[$currentAttachment] : $currentAttachment;
if (empty($params['{display}']))
{
if (!empty($currentAttachment['is_image']) || strpos($currentAttachment['mime_type'], 'video/') === 0 || strpos($currentAttachment['mime_type'], 'audio/') === 0)
$params['{display}'] = 'embed';
else
$params['{display}'] = 'link';
}
if ($params['{display}'] == 'embed')
{
$alt = ' alt="' . (!empty($params['{alt}']) ? $params['{alt}'] : $currentAttachment['name']) . '"';
$title = !empty($data) ? ' title="' . $smcFunc['htmlspecialchars']($data) . '"' : '';
$width = !empty($params['{width}']) ? $params['{width}'] : (!empty($currentAttachment['width']) ? $currentAttachment['width'] : '');
$height = !empty($params['{height}']) ? $params['{height}'] : (!empty($currentAttachment['height']) ? $currentAttachment['height'] : '');
if (!empty($currentAttachment['is_image']))
{
$width = !empty($width) ? ' width="' . $width . '"' : '';
$height = !empty($height) ? ' height="' . $height . '"' : '';
if ($currentAttachment['thumbnail']['has_thumb'] && empty($params['{width}']) && empty($params['{height}']))
$returnContext .= '<a href="' . $currentAttachment['href'] . ';image" id="link_' . $currentAttachment['id'] . '" onclick="' . $currentAttachment['thumbnail']['javascript'] . '"><img src="' . $currentAttachment['thumbnail']['href'] . '"' . $alt . $title . ' id="thumb_' . $currentAttachment['id'] . '" class="atc_img"></a>';
else
$returnContext .= '<img src="' . $currentAttachment['href'] . ';image"' . $alt . $title . $width . $height . ' class="bbc_img"/>';
}
elseif (strpos($currentAttachment['mime_type'], 'video/') === 0)
{
$width = !empty($width) ? ' width="' . $width . '"' : '';
$height = !empty($height) ? ' height="' . $height . '"' : '';
$returnContext .= '<div class="videocontainer"><div><video controls preload="none" src="'. $currentAttachment['href'] . '" playsinline' . $width . $height . ' style="object-fit:contain;"><a href="' . $currentAttachment['href'] . '" class="bbc_link">' . $smcFunc['htmlspecialchars'](!empty($data) ? $data : $currentAttachment['name']) . '</a></video></div></div>' . (!empty($data) && $data != $currentAttachment['name'] ? '<div class="smalltext">' . $data . '</div>' : '');
}
elseif (strpos($currentAttachment['mime_type'], 'audio/') === 0)
{
$width = 'max-width:100%; width: ' . (!empty($width) ? $width : '400') . 'px;';
$height = !empty($height) ? 'height: ' . $height . 'px;' : '';
$returnContext .= (!empty($data) && $data != $currentAttachment['name'] ? $data . ' ' : '') . '<audio controls preload="none" src="'. $currentAttachment['href'] . '" class="bbc_audio" style="vertical-align:middle;' . $width . $height . '"><a href="' . $currentAttachment['href'] . '" class="bbc_link">' . $smcFunc['htmlspecialchars'](!empty($data) ? $data : $currentAttachment['name']) . '</a></audio>';
}
else
{
$width = !empty($width) ? ' width="' . $width . '"' : '';
$height = !empty($height) ? ' height="' . $height . '"' : '';
$returnContext .= '<object type="' . $currentAttachment['mime_type'] . '" data="' . $currentAttachment['href'] . '"' . $width . $height . ' typemustmatch><a href="' . $currentAttachment['href'] . '" class="bbc_link">' . $smcFunc['htmlspecialchars'](!empty($data) ? $data : $currentAttachment['name']) . '</a></object>';
}
}
else
$returnContext .= '<a href="' . $currentAttachment['href'] . '" class="bbc_link">' . $smcFunc['htmlspecialchars'](!empty($data) ? $data : $currentAttachment['name']) . '</a>';
$data = $returnContext;
},
),
array(
'tag' => 'b',
'before' => '<b>',
'after' => '</b>',
),
array(
'tag' => 'bdo',
'type' => 'unparsed_equals',
'before' => '<bdo dir="$1">',
'after' => '</bdo>',
'test' => '(rtl|ltr)\]',
'block_level' => true,
),
array(
'tag' => 'black',
'before' => '<span style="color: black;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'blue',
'before' => '<span style="color: blue;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'br',
'type' => 'closed',
'content' => '<br>',
),
array(
'tag' => 'center',
'before' => '<div class="centertext">',
'after' => '</div>',
'block_level' => true,
),
array(
'tag' => 'code',
'type' => 'unparsed_content',
'content' => '<div class="codeheader"><span class="code floatleft">' . $txt['code'] . '</span> <a class="codeoperation smf_select_text">' . $txt['code_select'] . '</a> <a class="codeoperation smf_expand_code hidden" data-shrink-txt="' . $txt['code_shrink'] . '" data-expand-txt="' . $txt['code_expand'] . '">' . $txt['code_expand'] . '</a></div><code class="bbc_code">$1</code>',
'validate' => isset($disabled['code']) ? null : function(&$tag, &$data, $disabled) use ($context)
{
if (!isset($disabled['code']))
{
$php_parts = preg_split('~(<\?php|\?>)~', $data, -1, PREG_SPLIT_DELIM_CAPTURE);
for ($php_i = 0, $php_n = count($php_parts); $php_i < $php_n; $php_i++)
{
if ($php_parts[$php_i] != '<?php')
continue;
$php_string = '';
while ($php_i + 1 < count($php_parts) && $php_parts[$php_i] != '?>')
{
$php_string .= $php_parts[$php_i];
$php_parts[$php_i++] = '';
}
$php_parts[$php_i] = highlight_php_code($php_string . $php_parts[$php_i]);
}
$data = str_replace("<pre style=\"display: inline;\">\t</pre>", "\t", implode('', $php_parts));
$data = str_replace("\t", "<span style=\"white-space: pre;\">\t</span>", $data);
if (!empty($context['browser']['is_opera']))
$data .= ' ';
}
},
'block_level' => true,
),
array(
'tag' => 'code',
'type' => 'unparsed_equals_content',
'content' => '<div class="codeheader"><span class="code floatleft">' . $txt['code'] . '</span> ($2) <a class="codeoperation smf_select_text">' . $txt['code_select'] . '</a> <a class="codeoperation smf_expand_code hidden" data-shrink-txt="' . $txt['code_shrink'] . '" data-expand-txt="' . $txt['code_expand'] . '">' . $txt['code_expand'] . '</a></div><code class="bbc_code">$1</code>',
'validate' => isset($disabled['code']) ? null : function(&$tag, &$data, $disabled) use ($context)
{
if (!isset($disabled['code']))
{
$php_parts = preg_split('~(<\?php|\?>)~', $data[0], -1, PREG_SPLIT_DELIM_CAPTURE);
for ($php_i = 0, $php_n = count($php_parts); $php_i < $php_n; $php_i++)
{
if ($php_parts[$php_i] != '<?php')
continue;
$php_string = '';
while ($php_i + 1 < count($php_parts) && $php_parts[$php_i] != '?>')
{
$php_string .= $php_parts[$php_i];
$php_parts[$php_i++] = '';
}
$php_parts[$php_i] = highlight_php_code($php_string . $php_parts[$php_i]);
}
$data[0] = str_replace("<pre style=\"display: inline;\">\t</pre>", "\t", implode('', $php_parts));
$data[0] = str_replace("\t", "<span style=\"white-space: pre;\">\t</span>", $data[0]);
if (!empty($context['browser']['is_opera']))
$data[0] .= ' ';
}
},
'block_level' => true,
),
array(
'tag' => 'color',
'type' => 'unparsed_equals',
'test' => '(#[\da-fA-F]{3}|#[\da-fA-F]{6}|[A-Za-z]{1,20}|rgb\((?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\s?,\s?){2}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\))\]',
'before' => '<span style="color: $1;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'email',
'type' => 'unparsed_content',
'content' => '<a href="mailto:$1" class="bbc_email">$1</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
},
),
array(
'tag' => 'email',
'type' => 'unparsed_equals',
'before' => '<a href="mailto:$1" class="bbc_email">',
'after' => '</a>',
'disallow_children' => array('email', 'ftp', 'url', 'iurl'),
'disabled_after' => ' ($1)',
),
array(
'tag' => 'flash',
'type' => 'unparsed_commas_content',
'test' => '\d+,\d+\]',
'content' => '<a href="$1" target="_blank" rel="noopener">$1</a>',
'validate' => function (&$tag, &$data, $disabled)
{
$scheme = parse_url($data[0], PHP_URL_SCHEME);
if (empty($scheme))
$data[0] = '//' . ltrim($data[0], ':/');
},
),
array(
'tag' => 'float',
'type' => 'unparsed_equals',
'test' => '(left|right)(\s+max=\d+(?:%|px|em|rem|ex|pt|pc|ch|vw|vh|vmin|vmax|cm|mm|in)?)?\]',
'before' => '<div $1>',
'after' => '</div>',
'validate' => function(&$tag, &$data, $disabled)
{
$class = 'class="bbc_float float' . (strpos($data, 'left') === 0 ? 'left' : 'right') . '"';
if (preg_match('~\bmax=(\d+(?:%|px|em|rem|ex|pt|pc|ch|vw|vh|vmin|vmax|cm|mm|in)?)~', $data, $matches))
$css = ' style="max-width:' . $matches[1] . (is_numeric($matches[1]) ? 'px' : '') . '"';
else
$css = '';
$data = $class . $css;
},
'trim' => 'outside',
'block_level' => true,
),
array(
'tag' => 'ftp',
'type' => 'unparsed_content',
'content' => '<a href="$1" class="bbc_link" target="_blank" rel="noopener">$1</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = 'ftp://' . ltrim($data, ':/');
},
),
array(
'tag' => 'ftp',
'type' => 'unparsed_equals',
'before' => '<a href="$1" class="bbc_link" target="_blank" rel="noopener">',
'after' => '</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = 'ftp://' . ltrim($data, ':/');
},
'disallow_children' => array('email', 'ftp', 'url', 'iurl'),
'disabled_after' => ' ($1)',
),
array(
'tag' => 'font',
'type' => 'unparsed_equals',
'test' => '[A-Za-z0-9_,\-\s]+?\]',
'before' => '<span style="font-family: $1;" class="bbc_font">',
'after' => '</span>',
),
array(
'tag' => 'glow',
'type' => 'unparsed_commas',
'test' => '[#0-9a-zA-Z\-]{3,12},([012]\d{1,2}|\d{1,2})(,[^]]+)?\]',
'before' => '<span style="text-shadow: $1 1px 1px 1px">',
'after' => '</span>',
),
array(
'tag' => 'green',
'before' => '<span style="color: green;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'html',
'type' => 'unparsed_content',
'content' => '<div>$1</div>',
'block_level' => true,
'disabled_content' => '$1',
),
array(
'tag' => 'hr',
'type' => 'closed',
'content' => '<hr>',
'block_level' => true,
),
array(
'tag' => 'i',
'before' => '<i>',
'after' => '</i>',
),
array(
'tag' => 'img',
'type' => 'unparsed_content',
'parameters' => array(
'alt' => array('optional' => true),
'title' => array('optional' => true),
'width' => array('optional' => true, 'value' => ' width="$1"', 'match' => '(\d+)'),
'height' => array('optional' => true, 'value' => ' height="$1"', 'match' => '(\d+)'),
),
'content' => '<img src="$1" alt="{alt}" title="{title}"{width}{height} class="bbc_img resized">',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
if (parse_url($data, PHP_URL_SCHEME) === null)
$data = '//' . ltrim($data, ':/');
else
$data = get_proxied_url($data);
},
'disabled_content' => '($1)',
),
array(
'tag' => 'img',
'type' => 'unparsed_content',
'content' => '<img src="$1" alt="" class="bbc_img">',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
if (parse_url($data, PHP_URL_SCHEME) === null)
$data = '//' . ltrim($data, ':/');
else
$data = get_proxied_url($data);
},
'disabled_content' => '($1)',
),
array(
'tag' => 'iurl',
'type' => 'unparsed_content',
'content' => '<a href="$1" class="bbc_link">$1</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = '//' . ltrim($data, ':/');
},
),
array(
'tag' => 'iurl',
'type' => 'unparsed_equals',
'quoted' => 'optional',
'before' => '<a href="$1" class="bbc_link">',
'after' => '</a>',
'validate' => function(&$tag, &$data, $disabled)
{
if (substr($data, 0, 1) == '#')
$data = '#post_' . substr($data, 1);
else
{
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = '//' . ltrim($data, ':/');
}
},
'disallow_children' => array('email', 'ftp', 'url', 'iurl'),
'disabled_after' => ' ($1)',
),
array(
'tag' => 'justify',
'before' => '<div style="text-align: justify;">',
'after' => '</div>',
'block_level' => true,
),
array(
'tag' => 'left',
'before' => '<div style="text-align: left;">',
'after' => '</div>',
'block_level' => true,
),
array(
'tag' => 'li',
'before' => '<li>',
'after' => '</li>',
'trim' => 'outside',
'require_parents' => array('list'),
'block_level' => true,
'disabled_before' => '',
'disabled_after' => '<br>',
),
array(
'tag' => 'list',
'before' => '<ul class="bbc_list">',
'after' => '</ul>',
'trim' => 'inside',
'require_children' => array('li', 'list'),
'block_level' => true,
),
array(
'tag' => 'list',
'parameters' => array(
'type' => array('match' => '(none|disc|circle|square|decimal|decimal-leading-zero|lower-roman|upper-roman|lower-alpha|upper-alpha|lower-greek|upper-greek|lower-latin|upper-latin|hebrew|armenian|georgian|cjk-ideographic|hiragana|katakana|hiragana-iroha|katakana-iroha)'),
),
'before' => '<ul class="bbc_list" style="list-style-type: {type};">',
'after' => '</ul>',
'trim' => 'inside',
'require_children' => array('li'),
'block_level' => true,
),
array(
'tag' => 'ltr',
'before' => '<bdo dir="ltr">',
'after' => '</bdo>',
'block_level' => true,
),
array(
'tag' => 'me',
'type' => 'unparsed_equals',
'before' => '<div class="meaction">* $1 ',
'after' => '</div>',
'quoted' => 'optional',
'block_level' => true,
'disabled_before' => '/me ',
'disabled_after' => '<br>',
),
array(
'tag' => 'member',
'type' => 'unparsed_equals',
'before' => '<a href="' . $scripturl . '?action=profile;u=$1" class="mention" data-mention="$1">@',
'after' => '</a>',
),
array(
'tag' => 'move',
'before' => '<marquee>',
'after' => '</marquee>',
'block_level' => true,
'disallow_children' => array('move'),
),
array(
'tag' => 'nobbc',
'type' => 'unparsed_content',
'content' => '$1',
),
array(
'tag' => 'php',
'type' => 'unparsed_content',
'content' => '<span class="phpcode">$1</span>',
'validate' => isset($disabled['php']) ? null : function(&$tag, &$data, $disabled)
{
if (!isset($disabled['php']))
{
$add_begin = substr(trim($data), 0, 5) != '<?';
$data = highlight_php_code($add_begin ? '<?php ' . $data . '?>' : $data);
if ($add_begin)
$data = preg_replace(array('~^(.+?)<\?.{0,40}?php(?: |\s)~', '~\?>((?:</(font|span)>)*)$~'), '$1', $data, 2);
}
},
'block_level' => false,
'disabled_content' => '$1',
),
array(
'tag' => 'pre',
'before' => '<pre>',
'after' => '</pre>',
),
array(
'tag' => 'quote',
'before' => '<blockquote><cite>' . $txt['quote'] . '</cite>',
'after' => '</blockquote>',
'trim' => 'both',
'block_level' => true,
),
array(
'tag' => 'quote',
'parameters' => array(
'author' => array('match' => '(.{1,192}?)', 'quoted' => true),
),
'before' => '<blockquote><cite>' . $txt['quote_from'] . ': {author}</cite>',
'after' => '</blockquote>',
'trim' => 'both',
'block_level' => true,
),
array(
'tag' => 'quote',
'type' => 'parsed_equals',
'before' => '<blockquote><cite>' . $txt['quote_from'] . ': $1</cite>',
'after' => '</blockquote>',
'trim' => 'both',
'quoted' => 'optional',
'parsed_tags_allowed' => array('url', 'iurl', 'ftp'),
'block_level' => true,
),
array(
'tag' => 'quote',
'parameters' => array(
'author' => array('match' => '([^<>]{1,192}?)'),
'link' => array('match' => '(?:board=\d+;)?((?:topic|threadid)=[\dmsg#\./]{1,40}(?:;start=[\dmsg#\./]{1,40})?|msg=\d+?|action=profile;u=\d+)'),
'date' => array('match' => '(\d+)', 'validate' => 'timeformat'),
),
'before' => '<blockquote><cite><a href="' . $scripturl . '?{link}">' . $txt['quote_from'] . ': {author} ' . $txt['search_on'] . ' {date}</a></cite>',
'after' => '</blockquote>',
'trim' => 'both',
'block_level' => true,
),
array(
'tag' => 'quote',
'parameters' => array(
'author' => array('match' => '(.{1,192}?)'),
),
'before' => '<blockquote><cite>' . $txt['quote_from'] . ': {author}</cite>',
'after' => '</blockquote>',
'trim' => 'both',
'block_level' => true,
),
array(
'tag' => 'red',
'before' => '<span style="color: red;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'right',
'before' => '<div style="text-align: right;">',
'after' => '</div>',
'block_level' => true,
),
array(
'tag' => 'rtl',
'before' => '<bdo dir="rtl">',
'after' => '</bdo>',
'block_level' => true,
),
array(
'tag' => 's',
'before' => '<s>',
'after' => '</s>',
),
array(
'tag' => 'shadow',
'type' => 'unparsed_commas',
'test' => '[#0-9a-zA-Z\-]{3,12},(left|right|top|bottom|[0123]\d{0,2})\]',
'before' => '<span style="text-shadow: $1 $2">',
'after' => '</span>',
'validate' => function(&$tag, &$data, $disabled)
{
if ($data[1] == 'top' || (is_numeric($data[1]) && $data[1] < 50))
$data[1] = '0 -2px 1px';
elseif ($data[1] == 'right' || (is_numeric($data[1]) && $data[1] < 100))
$data[1] = '2px 0 1px';
elseif ($data[1] == 'bottom' || (is_numeric($data[1]) && $data[1] < 190))
$data[1] = '0 2px 1px';
elseif ($data[1] == 'left' || (is_numeric($data[1]) && $data[1] < 280))
$data[1] = '-2px 0 1px';
else
$data[1] = '1px 1px 1px';
},
),
array(
'tag' => 'size',
'type' => 'unparsed_equals',
'test' => '([1-9][\d]?p[xt]|small(?:er)?|large[r]?|x[x]?-(?:small|large)|medium|(0\.[1-9]|[1-9](\.[\d][\d]?)?)?em)\]',
'before' => '<span style="font-size: $1;" class="bbc_size">',
'after' => '</span>',
),
array(
'tag' => 'size',
'type' => 'unparsed_equals',
'test' => '[1-7]\]',
'before' => '<span style="font-size: $1;" class="bbc_size">',
'after' => '</span>',
'validate' => function(&$tag, &$data, $disabled)
{
$sizes = array(1 => 0.7, 2 => 1.0, 3 => 1.35, 4 => 1.45, 5 => 2.0, 6 => 2.65, 7 => 3.95);
$data = $sizes[$data] . 'em';
},
),
array(
'tag' => 'sub',
'before' => '<sub>',
'after' => '</sub>',
),
array(
'tag' => 'sup',
'before' => '<sup>',
'after' => '</sup>',
),
array(
'tag' => 'table',
'before' => '<table class="bbc_table">',
'after' => '</table>',
'trim' => 'inside',
'require_children' => array('tr'),
'block_level' => true,
),
array(
'tag' => 'td',
'before' => '<td>',
'after' => '</td>',
'require_parents' => array('tr'),
'trim' => 'outside',
'block_level' => true,
'disabled_before' => '',
'disabled_after' => '',
),
array(
'tag' => 'time',
'type' => 'unparsed_content',
'content' => '$1',
'validate' => function(&$tag, &$data, $disabled)
{
if (is_numeric($data))
$data = timeformat($data);
else
$tag['content'] = '[time]$1[/time]';
},
),
array(
'tag' => 'tr',
'before' => '<tr>',
'after' => '</tr>',
'require_parents' => array('table'),
'require_children' => array('td'),
'trim' => 'both',
'block_level' => true,
'disabled_before' => '',
'disabled_after' => '',
),
array(
'tag' => 'tt',
'before' => '<span class="monospace">',
'after' => '</span>',
),
array(
'tag' => 'u',
'before' => '<u>',
'after' => '</u>',
),
array(
'tag' => 'url',
'type' => 'unparsed_content',
'content' => '<a href="$1" class="bbc_link" target="_blank" rel="noopener">$1</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$data = strtr($data, array('<br>' => ''));
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = '//' . ltrim($data, ':/');
},
),
array(
'tag' => 'url',
'type' => 'unparsed_equals',
'quoted' => 'optional',
'before' => '<a href="$1" class="bbc_link" target="_blank" rel="noopener">',
'after' => '</a>',
'validate' => function(&$tag, &$data, $disabled)
{
$scheme = parse_url($data, PHP_URL_SCHEME);
if (empty($scheme))
$data = '//' . ltrim($data, ':/');
},
'disallow_children' => array('email', 'ftp', 'url', 'iurl'),
'disabled_after' => ' ($1)',
),
array(
'tag' => 'white',
'before' => '<span style="color: white;" class="bbc_color">',
'after' => '</span>',
),
array(
'tag' => 'youtube',
'type' => 'unparsed_content',
'content' => '<div class="videocontainer"><div><iframe frameborder="0" src="https://www.youtube.com/embed/$1?origin=' . $hosturl . '&wmode=opaque" data-youtube-id="$1" allowfullscreen></iframe></div></div>',
'disabled_content' => '<a href="https://www.youtube.com/watch?v=$1" target="_blank" rel="noopener">https://www.youtube.com/watch?v=$1</a>',
'block_level' => true,
),
);
$no_autolink_tags = array(
'url',
'iurl',
'email',
'img',
'html',
);
call_integration_hook('integrate_bbc_codes', array(&$codes, &$no_autolink_tags));
if ($message === false)
{
usort($codes, function($a, $b)
{
return strcmp($a['tag'], $b['tag']);
});
return $codes;
}
$itemcodes = array(
'*' => 'disc',
'@' => 'disc',
'+' => 'square',
'x' => 'square',
'#' => 'square',
'o' => 'circle',
'O' => 'circle',
'0' => 'circle',
);
if (!isset($disabled['li']) && !isset($disabled['list']))
{
foreach ($itemcodes as $c => $dummy)
$bbc_codes[$c] = array();
}
if (!isset($disabled['color']))
{
$codes[] = array(
'tag' => 'chrissy',
'before' => '<span style="color: #cc0099;">',
'after' => ' :-*</span>',
);
$codes[] = array(
'tag' => 'kissy',
'before' => '<span style="color: #cc0099;">',
'after' => ' :-*</span>',
);
}
$codes[] = array(
'tag' => 'cowsay',
'parameters' => array(
'e' => array('optional' => true, 'quoted' => true, 'match' => '(.*?)', 'default' => 'oo', 'validate' => function ($eyes) use ($smcFunc)
{
static $css_added;
if (empty($css_added))
{
$css = base64_decode('cHJlW2RhdGEtZV1bZGF0YS10XXt3aGl0ZS1zcGFjZTpwcmUtd3JhcDtsaW5lLWhlaWdodDppbml0aWFsO31wcmVbZGF0YS1lXVtkYXRhLXRdID4gZGl2e2Rpc3BsYXk6dGFibGU7Ym9yZGVyOjFweCBzb2xpZDtib3JkZXItcmFkaXVzOjAuNWVtO3BhZGRpbmc6MWNoO21heC13aWR0aDo4MGNoO21pbi13aWR0aDoxMmNoO31wcmVbZGF0YS1lXVtkYXRhLXRdOjphZnRlcntkaXNwbGF5OmlubGluZS1ibG9jazttYXJnaW4tbGVmdDo4Y2g7bWluLXdpZHRoOjIwY2g7ZGlyZWN0aW9uOmx0cjtjb250ZW50OidcNUMgICBeX19eXEEgIFw1QyAgKCcgYXR0cihkYXRhLWUpICcpXDVDX19fX19fX1xBICAgIChfXylcNUMgICAgICAgIClcNUMvXDVDXEEgICAgICcgYXR0cihkYXRhLXQpICcgfHwtLS0tdyB8XEEgICAgICAgIHx8ICAgICB8fCc7fQ==');
addInlineJavaScript('
$("head").append("<style>" + ' . JavaScriptEscape($css) . ' + "</style>");', true);
$css_added = true;
}
return $smcFunc['substr']($eyes . 'oo', 0, 2);
},
),
't' => array('optional' => true, 'quoted' => true, 'match' => '(.*?)', 'default' => ' ', 'validate' => function ($tongue) use ($smcFunc)
{
return $smcFunc['substr']($tongue . ' ', 0, 2);
},
),
),
'before' => '<pre data-e="{e}" data-t="{t}"><div>',
'after' => '</div></pre>',
'block_level' => true,
);
foreach ($codes as $code)
{
if (!empty($code['parameters']))
ksort($code['parameters'], SORT_STRING);
if (empty($parse_tags) || in_array($code['tag'], $parse_tags))
$bbc_codes[substr($code['tag'], 0, 1)][] = $code;
}
$codes = null;
}
if ($cache_id != '' && !empty($cache_enable) && (($cache_enable >= 2 && isset($message[1000])) || isset($message[2400])) && empty($parse_tags))
{
$cache_key = 'parse:' . $cache_id . '-' . md5(md5($message) . '-' . $smileys . (empty($disabled) ? '' : implode(',', array_keys($disabled))) . $smcFunc['json_encode']($context['browser']) . $txt['lang_locale'] . $user_info['time_offset'] . $user_info['time_format']);
if (($temp = cache_get_data($cache_key, 240)) != null)
return $temp;
$cache_t = microtime(true);
}
if ($smileys === 'print')
{
$disabled['glow'] = true;
$disabled['shadow'] = true;
$disabled['move'] = true;
$disabled['color'] = true;
$disabled['black'] = true;
$disabled['blue'] = true;
$disabled['white'] = true;
$disabled['red'] = true;
$disabled['green'] = true;
$disabled['me'] = true;
$disabled['php'] = true;
$disabled['ftp'] = true;
$disabled['url'] = true;
$disabled['iurl'] = true;
$disabled['email'] = true;
$disabled['flash'] = true;
if (!isset($_GET['images']))
$disabled['img'] = true;
call_integration_hook('integrate_bbc_print', array(&$disabled));
}
$open_tags = array();
$message = strtr($message, array("\n" => '<br>'));
if (!empty($parse_tags))
{
$real_alltags_regex = $alltags_regex;
$alltags_regex = '';
}
if (empty($alltags_regex))
{
$alltags = array();
foreach ($bbc_codes as $section)
{
foreach ($section as $code)
$alltags[] = $code['tag'];
}
$alltags_regex = '(?' . '>\b' . build_regex(array_unique($alltags)) . '\b|' . build_regex(array_keys($itemcodes)) . ')';
}
$pos = -1;
while ($pos !== false)
{
$last_pos = isset($last_pos) ? max($pos, $last_pos) : $pos;
preg_match('~\[/?(?=' . $alltags_regex . ')~i', $message, $matches, PREG_OFFSET_CAPTURE, $pos + 1);
$pos = isset($matches[0][1]) ? $matches[0][1] : false;
if ($pos === false || $last_pos > $pos)
$pos = strlen($message) + 1;
if ($last_pos < $pos - 1)
{
$last_pos = max($last_pos, 0);
$data = substr($message, $last_pos, $pos - $last_pos);
$placeholders = array();
$placeholders_counter = 0;
if (!empty($modSettings['enablePostHTML']) && strpos($data, '<') !== false)
{
$data = preg_replace('~<a\s+href=((?:")?)((?:https?://|ftps?://|mailto:|tel:)\S+?)\\1>(.*?)</a>~i', '[url="$2"]$3[/url]', $data);
$empty_tags = array('br', 'hr');
foreach ($empty_tags as $tag)
$data = str_replace(array('<' . $tag . '>', '<' . $tag . '/>', '<' . $tag . ' />'), '<' . $tag . '>', $data);
$closable_tags = array('b', 'u', 'i', 's', 'em', 'ins', 'del', 'pre', 'blockquote', 'strong');
foreach ($closable_tags as $tag)
{
$diff = substr_count($data, '<' . $tag . '>') - substr_count($data, '</' . $tag . '>');
$data = strtr($data, array('<' . $tag . '>' => '<' . $tag . '>', '</' . $tag . '>' => '</' . $tag . '>'));
if ($diff > 0)
$data = substr($data, 0, -1) . str_repeat('</' . $tag . '>', $diff) . substr($data, -1);
}
preg_match_all('~<img\s+src=((?:")?)((?:https?://|ftps?://)\S+?)\\1(?:\s+alt=(".*?"|\S*?))?(?:\s?/)?>~i', $data, $matches, PREG_PATTERN_ORDER);
if (!empty($matches[0]))
{
$replaces = array();
foreach ($matches[2] as $match => $imgtag)
{
$alt = empty($matches[3][$match]) ? '' : ' alt=' . preg_replace('~^"|"$~', '', $matches[3][$match]);
if (preg_match('~action(=|%3d)(?!dlattach)~i', $imgtag) != 0)
$imgtag = preg_replace('~action(?:=|%3d)(?!dlattach)~i', 'action-', $imgtag);
$placeholder = '<placeholder ' . ++$placeholders_counter . '>';
$placeholders[$placeholder] = '[img' . $alt . ']' . $imgtag . '[/img]';
$replaces[$matches[0][$match]] = $placeholder;
}
$data = strtr($data, $replaces);
}
}
if (!empty($modSettings['autoLinkUrls']))
{
$no_autolink_area = false;
if (!empty($open_tags))
{
foreach ($open_tags as $open_tag)
if (in_array($open_tag['tag'], $no_autolink_tags))
$no_autolink_area = true;
}
$lastAutoPos = isset($lastAutoPos) ? $lastAutoPos : 0;
if ($pos < $lastAutoPos)
$no_autolink_area = true;
$lastAutoPos = $pos;
if (!$no_autolink_area)
{
if (strpos($data, ' ') !== false)
{
$placeholders['<placeholder non-breaking-space>'] = ' ';
$data = strtr($data, array(' ' => '<placeholder non-breaking-space>'));
}
if (!isset($disabled['url']) && strpos($data, '[url') === false)
{
$url_regex = '(?(DEFINE)(?<tlds>' . $modSettings['tld_regex'] . '))';
$url_regex .=
'(?:' .
'(?:' .
'(?:' .
'\b[a-z][\w\-]+:' .
'|' .
'(?<=^|\W)(?=//)' .
')' .
'(?:' .
'//' .
'(?:' .
'localhost' .
'|' .
'[\p{L}\p{M}\p{N}\-.:@]+\.(?P>tlds)' .
')' .
'(?=[^\p{L}\p{N}\-.]|$)' .
'|' .
'[\p{L}\p{N}][\p{L}\p{M}\p{N}\-.:@]+[\p{L}\p{M}\p{N}]' .
'\.[\p{L}\p{M}\p{N}\-]+' .
')' .
')' .
'|' .
'(?:' .
'(?<=^|[^\p{L}\p{M}\p{N}\-:@])' .
'[\p{L}\p{N}][\p{L}\p{M}\p{N}\-.]+[\p{L}\p{M}\p{N}]' .
'\.(?P>tlds)' .
'(?=' .
'$|[^\p{L}\p{N}\-]' .
'|' .
'\.(?=$|[^\p{L}\p{N}\-])' .
')' .
')' .
')' .
'(?:' .
'/' .
'(?:' .
'(?:' .
'[^\s()<>]+' .
'|' .
'\(([^\s()<>]+|(\([^\s()<>]+\)))*\)' .
')+' .
'(?:' .
'\(([^\s()<>]+|(\([^\s()<>]+\)))*\)' .
'|' .
'[^\s`!()\[\]{};:\'".,<>?«»“”‘’/]' .
'|' .
'(?<!/)/' .
')' .
')?' .
')?';
$data = preg_replace_callback('~' . $url_regex . '~i' . ($context['utf8'] ? 'u' : ''), function($matches)
{
$url = array_shift($matches);
if ($url != sanitize_iri($url))
return $url;
$scheme = parse_url($url, PHP_URL_SCHEME);
if ($scheme == 'mailto')
{
$email_address = str_replace('mailto:', '', $url);
if (!isset($disabled['email']) && filter_var($email_address, FILTER_VALIDATE_EMAIL) !== false)
return '[email=' . $email_address . ']' . $url . '[/email]';
else
return $url;
}
if (empty($scheme))
$fullUrl = '//' . ltrim($url, ':/');
else
$fullUrl = $url;
if (validate_iri((strpos($fullUrl, '//') === 0 ? 'http:' : '') . $fullUrl) === false)
return $url;
return '[url="' . str_replace(array('[', ']'), array('[', ']'), $fullUrl) . '"]' . $url . '[/url]';
}, $data);
}
if (!isset($disabled['email']) && strpos($data, '@') !== false && strpos($data, '[email') === false && stripos($data, 'mailto:') === false)
{
$email_regex = '
# Preceded by a non-domain character or start of line
(?<=^|[^\p{L}\p{M}\p{N}\-\.])
# An email address
[\p{L}\p{M}\p{N}_\-.]{1,80}
@
[\p{L}\p{M}\p{N}\-.]+
\.
' . $modSettings['tld_regex'] . '
# Followed by either:
(?=
# end of line or a non-domain character (excluding the dot)
$|[^\p{L}\p{M}\p{N}\-]
| # or
# a dot followed by end of line or a non-domain character
\.(?=$|[^\p{L}\p{M}\p{N}\-])
)';
$data = preg_replace('~' . $email_regex . '~xi' . ($context['utf8'] ? 'u' : ''), '[email]$0[/email]', $data);
}
}
}
$data = strtr($data, $placeholders);
$data = strtr($data, array("\t" => ' '));
if ($data != substr($message, $last_pos, $pos - $last_pos))
{
$message = substr($message, 0, $last_pos) . $data . substr($message, $pos);
$old_pos = strlen($data) + $last_pos;
$pos = strpos($message, '[', $last_pos);
$pos = $pos === false ? $old_pos : min($pos, $old_pos);
}
}
if ($pos >= strlen($message) - 1)
break;
$tag_character = strtolower($message[$pos + 1]);
if ($tag_character == '/' && !empty($open_tags))
{
$pos2 = strpos($message, ']', $pos + 1);
if ($pos2 == $pos + 2)
continue;
$look_for = strtolower(substr($message, $pos + 2, $pos2 - $pos - 2));
if (!in_array($look_for, array_map(function($code)
{
return $code['tag'];
}, $open_tags)))
continue;
$to_close = array();
$block_level = null;
do
{
$tag = array_pop($open_tags);
if (!$tag)
break;
if (!empty($tag['block_level']))
{
if ($block_level === false)
{
array_push($open_tags, $tag);
break;
}
if (strlen($look_for) > 0 && isset($bbc_codes[$look_for[0]]))
{
foreach ($bbc_codes[$look_for[0]] as $temp)
if ($temp['tag'] == $look_for)
{
$block_level = !empty($temp['block_level']);
break;
}
}
if ($block_level !== true)
{
$block_level = false;
array_push($open_tags, $tag);
break;
}
}
$to_close[] = $tag;
}
while ($tag['tag'] != $look_for);
if ((empty($open_tags) && (empty($tag) || $tag['tag'] != $look_for)))
{
$open_tags = $to_close;
continue;
}
elseif (!empty($to_close) && $tag['tag'] != $look_for)
{
if ($block_level === null && isset($look_for[0], $bbc_codes[$look_for[0]]))
{
foreach ($bbc_codes[$look_for[0]] as $temp)
if ($temp['tag'] == $look_for)
{
$block_level = !empty($temp['block_level']);
break;
}
}
if (!$block_level)
{
foreach ($to_close as $tag)
array_push($open_tags, $tag);
continue;
}
}
foreach ($to_close as $tag)
{
$message = substr($message, 0, $pos) . "\n" . $tag['after'] . "\n" . substr($message, $pos2 + 1);
$pos += strlen($tag['after']) + 2;
$pos2 = $pos - 1;
$whitespace_regex = '';
if (!empty($tag['block_level']))
$whitespace_regex .= '( |\s)*(<br>)?';
if (!empty($tag['trim']) && $tag['trim'] != 'inside')
$whitespace_regex .= empty($tag['require_parents']) ? '( |\s)*' : '(<br>| |\s)*';
if (!empty($whitespace_regex) && preg_match('~' . $whitespace_regex . '~', substr($message, $pos), $matches) != 0)
$message = substr($message, 0, $pos) . substr($message, $pos + strlen($matches[0]));
}
if (!empty($to_close))
{
$to_close = array();
$pos--;
}
continue;
}
if (!isset($bbc_codes[$tag_character]))
continue;
$inside = empty($open_tags) ? null : $open_tags[count($open_tags) - 1];
$tag = null;
foreach ($bbc_codes[$tag_character] as $possible)
{
$pt_strlen = strlen($possible['tag']);
if (strtolower(substr($message, $pos + 1, $pt_strlen)) != $possible['tag'])
continue;
$next_c = isset($message[$pos + 1 + $pt_strlen]) ? $message[$pos + 1 + $pt_strlen] : '';
if ($next_c == '')
break;
if (isset($possible['test']) && preg_match('~^' . $possible['test'] . '~', substr($message, $pos + 1 + $pt_strlen + 1)) === 0)
continue;
elseif (!empty($possible['parameters']))
{
$param_required = false;
foreach ($possible['parameters'] as $param)
{
if (empty($param['optional']))
{
$param_required = true;
break;
}
}
if ($param_required && $next_c != ' ')
continue;
}
elseif (isset($possible['type']))
{
if (in_array($possible['type'], array('unparsed_equals', 'unparsed_commas', 'unparsed_commas_content', 'unparsed_equals_content', 'parsed_equals')) && $next_c != '=')
continue;
if ($possible['type'] == 'closed' && $next_c != ']' && substr($message, $pos + 1 + $pt_strlen, 2) != '/]' && substr($message, $pos + 1 + $pt_strlen, 3) != ' /]')
continue;
if ($possible['type'] == 'unparsed_content' && $next_c != ']')
continue;
}
elseif ($next_c != ']')
continue;
if (isset($possible['require_parents']) && ($inside === null || !in_array($inside['tag'], $possible['require_parents'])))
continue;
elseif (isset($inside['require_children']) && !in_array($possible['tag'], $inside['require_children']))
continue;
elseif (isset($inside['disallow_children']) && in_array($possible['tag'], $inside['disallow_children']))
continue;
$pos1 = $pos + 1 + $pt_strlen + 1;
if ($possible['tag'] == 'quote')
{
$quote_alt = false;
foreach ($open_tags as $open_quote)
{
if ($open_quote['tag'] == 'quote')
$quote_alt = !$quote_alt;
}
$possible['before'] = strtr($possible['before'], array('<blockquote>' => '<blockquote class="bbc_' . ($quote_alt ? 'alternate' : 'standard') . '_quote">'));
}
if (!empty($possible['parameters']))
{
$regex_key = $smcFunc['json_encode']($possible['parameters']);
if (!isset($params_regexes[$regex_key]))
{
$params_regexes[$regex_key] = '';
foreach ($possible['parameters'] as $p => $info)
$params_regexes[$regex_key] .= '(\s+' . $p . '=' . (empty($info['quoted']) ? '' : '"') . (isset($info['match']) ? $info['match'] : '(.+?)') . (empty($info['quoted']) ? '' : '"') . '\s*)' . (empty($info['optional']) ? '' : '?');
}
$blob = preg_split('~\[/?(?:' . $alltags_regex . ')~i', substr($message, $pos));
$blobs = preg_split('~\]~i', $blob[1]);
$splitters = implode('=|', array_keys($possible['parameters'])) . '=';
$blob_counter = 1;
while ($blob_counter <= count($blobs))
{
$given_param_string = implode(']', array_slice($blobs, 0, $blob_counter++));
$given_params = preg_split('~\s(?=(' . $splitters . '))~i', $given_param_string);
sort($given_params, SORT_STRING);
$match = preg_match('~^' . $params_regexes[$regex_key] . '$~i', implode(' ', $given_params), $matches) !== 0;
if ($match)
break;
}
if (!$match)
continue;
$params = array();
for ($i = 1, $n = count($matches); $i < $n; $i += 2)
{
$key = strtok(ltrim($matches[$i]), '=');
if ($key === false)
continue;
elseif (isset($possible['parameters'][$key]['value']))
$params['{' . $key . '}'] = strtr($possible['parameters'][$key]['value'], array('$1' => $matches[$i + 1]));
elseif (isset($possible['parameters'][$key]['validate']))
$params['{' . $key . '}'] = $possible['parameters'][$key]['validate']($matches[$i + 1]);
else
$params['{' . $key . '}'] = $matches[$i + 1];
$params['{' . $key . '}'] = strtr($params['{' . $key . '}'], array('$' => '$', '{' => '{'));
}
foreach ($possible['parameters'] as $p => $info)
{
if (!isset($params['{' . $p . '}']))
{
if (!isset($info['default']))
$params['{' . $p . '}'] = '';
elseif (isset($possible['parameters'][$p]['value']))
$params['{' . $p . '}'] = strtr($possible['parameters'][$p]['value'], array('$1' => $info['default']));
elseif (isset($possible['parameters'][$p]['validate']))
$params['{' . $p . '}'] = $possible['parameters'][$p]['validate']($info['default']);
else
$params['{' . $p . '}'] = $info['default'];
}
}
$tag = $possible;
if (isset($tag['before']))
$tag['before'] = strtr($tag['before'], $params);
if (isset($tag['after']))
$tag['after'] = strtr($tag['after'], $params);
if (isset($tag['content']))
$tag['content'] = strtr($tag['content'], $params);
$pos1 += strlen($given_param_string);
}
else
{
$tag = $possible;
$params = array();
}
break;
}
if ($smileys !== false && $tag === null && isset($itemcodes[$message[$pos + 1]]) && $message[$pos + 2] == ']' && !isset($disabled['list']) && !isset($disabled['li']))
{
if ($message[$pos + 1] == '0' && !in_array($message[$pos - 1], array(';', ' ', "\t", "\n", '>')))
continue;
$tag = $itemcodes[$message[$pos + 1]];
if ($inside === null || ($inside['tag'] != 'list' && $inside['tag'] != 'li'))
{
$open_tags[] = array(
'tag' => 'list',
'after' => '</ul>',
'block_level' => true,
'require_children' => array('li'),
'disallow_children' => isset($inside['disallow_children']) ? $inside['disallow_children'] : null,
);
$code = '<ul class="bbc_list">';
}
elseif ($inside['tag'] == 'li')
{
array_pop($open_tags);
$code = '</li>';
}
else
$code = '';
$open_tags[] = array(
'tag' => 'li',
'after' => '</li>',
'trim' => 'outside',
'block_level' => true,
'disallow_children' => isset($inside['disallow_children']) ? $inside['disallow_children'] : null,
);
$code .= '<li' . ($tag == '' ? '' : ' type="' . $tag . '"') . '>';
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos + 3);
$pos += strlen($code) - 1 + 2;
$pos2 = strpos($message, '<br>', $pos);
$pos3 = strpos($message, '[/', $pos);
if ($pos2 !== false && ($pos2 <= $pos3 || $pos3 === false))
{
preg_match('~^(<br>| |\s|\[)+~', substr($message, $pos2 + 4), $matches);
$message = substr($message, 0, $pos2) . (!empty($matches[0]) && substr($matches[0], -1) == '[' ? '[/li]' : '[/li][/list]') . substr($message, $pos2);
$open_tags[count($open_tags) - 2]['after'] = '</ul>';
}
else
{
$open_tags[count($open_tags) - 1]['after'] = '';
$open_tags[count($open_tags) - 2]['after'] = '</li></ul>';
}
continue;
}
if ($tag === null && $inside !== null && !empty($inside['require_children']))
{
array_pop($open_tags);
$message = substr($message, 0, $pos) . "\n" . $inside['after'] . "\n" . substr($message, $pos);
$pos += strlen($inside['after']) - 1 + 2;
}
if ($tag === null)
continue;
if (isset($inside['disallow_children']))
$tag['disallow_children'] = isset($tag['disallow_children']) ? array_unique(array_merge($tag['disallow_children'], $inside['disallow_children'])) : $inside['disallow_children'];
if (isset($disabled[$tag['tag']]))
{
if (!isset($tag['disabled_before']) && !isset($tag['disabled_after']) && !isset($tag['disabled_content']))
{
$tag['before'] = !empty($tag['block_level']) ? '<div>' : '';
$tag['after'] = !empty($tag['block_level']) ? '</div>' : '';
$tag['content'] = isset($tag['type']) && $tag['type'] == 'closed' ? '' : (!empty($tag['block_level']) ? '<div>$1</div>' : '$1');
}
elseif (isset($tag['disabled_before']) || isset($tag['disabled_after']))
{
$tag['before'] = isset($tag['disabled_before']) ? $tag['disabled_before'] : (!empty($tag['block_level']) ? '<div>' : '');
$tag['after'] = isset($tag['disabled_after']) ? $tag['disabled_after'] : (!empty($tag['block_level']) ? '</div>' : '');
}
else
$tag['content'] = $tag['disabled_content'];
}
$tag_strlen = strlen($tag['tag']);
if (!empty($tag['block_level']) && $tag['tag'] != 'html' && empty($inside['block_level']))
{
$n = count($open_tags) - 1;
while (empty($open_tags[$n]['block_level']) && $n >= 0)
$n--;
for ($i = count($open_tags) - 1; $i > $n; $i--)
{
$message = substr($message, 0, $pos) . "\n" . $open_tags[$i]['after'] . "\n" . substr($message, $pos);
$ot_strlen = strlen($open_tags[$i]['after']);
$pos += $ot_strlen + 2;
$pos1 += $ot_strlen + 2;
$whitespace_regex = '';
if (!empty($tag['block_level']))
$whitespace_regex .= '( |\s)*(<br>)?';
if (!empty($tag['trim']) && $tag['trim'] != 'inside')
$whitespace_regex .= empty($tag['require_parents']) ? '( |\s)*' : '(<br>| |\s)*';
if (!empty($whitespace_regex) && preg_match('~' . $whitespace_regex . '~', substr($message, $pos), $matches) != 0)
$message = substr($message, 0, $pos) . substr($message, $pos + strlen($matches[0]));
array_pop($open_tags);
}
}
$pos1 = min(strlen($message), $pos1);
if (!isset($tag['type']))
{
$open_tags[] = $tag;
$message = substr($message, 0, $pos) . "\n" . $tag['before'] . "\n" . substr($message, $pos1);
$pos += strlen($tag['before']) - 1 + 2;
}
elseif ($tag['type'] == 'unparsed_content')
{
$pos2 = stripos($message, '[/' . substr($message, $pos + 1, $tag_strlen) . ']', $pos1);
if ($pos2 === false)
continue;
$data = substr($message, $pos1, $pos2 - $pos1);
if (!empty($tag['block_level']) && substr($data, 0, 4) == '<br>')
$data = substr($data, 4);
if (isset($tag['validate']))
$tag['validate']($tag, $data, $disabled, $params);
$code = strtr($tag['content'], array('$1' => $data));
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos2 + 3 + $tag_strlen);
$pos += strlen($code) - 1 + 2;
$last_pos = $pos + 1;
}
elseif ($tag['type'] == 'unparsed_equals_content')
{
if (isset($tag['quoted']))
{
$quoted = substr($message, $pos1, 6) == '"';
if ($tag['quoted'] != 'optional' && !$quoted)
continue;
if ($quoted)
$pos1 += 6;
}
else
$quoted = false;
$pos2 = strpos($message, $quoted == false ? ']' : '"]', $pos1);
if ($pos2 === false)
continue;
$pos3 = stripos($message, '[/' . substr($message, $pos + 1, $tag_strlen) . ']', $pos2);
if ($pos3 === false)
continue;
$data = array(
substr($message, $pos2 + ($quoted == false ? 1 : 7), $pos3 - ($pos2 + ($quoted == false ? 1 : 7))),
substr($message, $pos1, $pos2 - $pos1)
);
if (!empty($tag['block_level']) && substr($data[0], 0, 4) == '<br>')
$data[0] = substr($data[0], 4);
if (isset($tag['validate']))
$tag['validate']($tag, $data, $disabled, $params);
$code = strtr($tag['content'], array('$1' => $data[0], '$2' => $data[1]));
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos3 + 3 + $tag_strlen);
$pos += strlen($code) - 1 + 2;
}
elseif ($tag['type'] == 'closed')
{
$pos2 = strpos($message, ']', $pos);
$message = substr($message, 0, $pos) . "\n" . $tag['content'] . "\n" . substr($message, $pos2 + 1);
$pos += strlen($tag['content']) - 1 + 2;
}
elseif ($tag['type'] == 'unparsed_commas_content')
{
$pos2 = strpos($message, ']', $pos1);
if ($pos2 === false)
continue;
$pos3 = stripos($message, '[/' . substr($message, $pos + 1, $tag_strlen) . ']', $pos2);
if ($pos3 === false)
continue;
$data = explode(',', ',' . substr($message, $pos1, $pos2 - $pos1));
$data[0] = substr($message, $pos2 + 1, $pos3 - $pos2 - 1);
if (isset($tag['validate']))
$tag['validate']($tag, $data, $disabled, $params);
$code = $tag['content'];
foreach ($data as $k => $d)
$code = strtr($code, array('$' . ($k + 1) => trim($d)));
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos3 + 3 + $tag_strlen);
$pos += strlen($code) - 1 + 2;
}
elseif ($tag['type'] == 'unparsed_commas')
{
$pos2 = strpos($message, ']', $pos1);
if ($pos2 === false)
continue;
$data = explode(',', substr($message, $pos1, $pos2 - $pos1));
if (isset($tag['validate']))
$tag['validate']($tag, $data, $disabled, $params);
foreach ($data as $k => $d)
$tag['after'] = strtr($tag['after'], array('$' . ($k + 1) => trim($d)));
$open_tags[] = $tag;
$code = $tag['before'];
foreach ($data as $k => $d)
$code = strtr($code, array('$' . ($k + 1) => trim($d)));
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos2 + 1);
$pos += strlen($code) - 1 + 2;
}
elseif ($tag['type'] == 'unparsed_equals' || $tag['type'] == 'parsed_equals')
{
if (isset($tag['quoted']))
{
$quoted = substr($message, $pos1, 6) == '"';
if ($tag['quoted'] != 'optional' && !$quoted)
continue;
if ($quoted)
$pos1 += 6;
}
else
$quoted = false;
$pos2 = strpos($message, $quoted == false ? ']' : '"]', $pos1);
if ($pos2 === false)
continue;
$data = substr($message, $pos1, $pos2 - $pos1);
if (isset($tag['validate']))
$tag['validate']($tag, $data, $disabled, $params);
if ($tag['type'] != 'unparsed_equals')
$data = parse_bbc($data, !empty($tag['parsed_tags_allowed']) ? false : true, '', !empty($tag['parsed_tags_allowed']) ? $tag['parsed_tags_allowed'] : array());
$tag['after'] = strtr($tag['after'], array('$1' => $data));
$open_tags[] = $tag;
$code = strtr($tag['before'], array('$1' => $data));
$message = substr($message, 0, $pos) . "\n" . $code . "\n" . substr($message, $pos2 + ($quoted == false ? 1 : 7));
$pos += strlen($code) - 1 + 2;
}
if (!empty($tag['block_level']) && substr($message, $pos + 1, 4) == '<br>')
$message = substr($message, 0, $pos + 1) . substr($message, $pos + 5);
if (!empty($tag['trim']) && $tag['trim'] != 'outside' && preg_match('~(<br>| |\s)*~', substr($message, $pos + 1), $matches) != 0)
$message = substr($message, 0, $pos + 1) . substr($message, $pos + 1 + strlen($matches[0]));
}
while ($tag = array_pop($open_tags))
$message .= "\n" . $tag['after'] . "\n";
if ($smileys === true)
{
$message_parts = explode("\n", $message);
for ($i = 0, $n = count($message_parts); $i < $n; $i += 2)
parsesmileys($message_parts[$i]);
$message = implode('', $message_parts);
}
else
$message = strtr($message, array("\n" => ''));
if ($message !== '' && $message[0] === ' ')
$message = ' ' . substr($message, 1);
$message = strtr($message, array(' ' => ' ', "\r" => '', "\n" => '<br>', '<br> ' => '<br> ', ' ' => "\n"));
call_integration_hook('integrate_post_parsebbc', array(&$message, &$smileys, &$cache_id, &$parse_tags));
if (isset($cache_key, $cache_t) && microtime(true) - $cache_t > 0.05)
cache_put_data($cache_key, $message, 240);
if (!empty($parse_tags))
{
$alltags_regex = empty($real_alltags_regex) ? '' : $real_alltags_regex;
unset($real_alltags_regex);
}
elseif (!empty($bbc_codes))
$bbc_lang_locales[$txt['lang_locale']] = $bbc_codes;
return $message;
}
function parsesmileys(&$message)
{
global $modSettings, $txt, $user_info, $context, $smcFunc;
static $smileyPregSearch = null, $smileyPregReplacements = array();
if ($user_info['smiley_set'] == 'none' || trim($message) == '')
return;
call_integration_hook('integrate_smileys', array(&$smileyPregSearch, &$smileyPregReplacements));
if (empty($smileyPregSearch))
{
$cache_time = empty($modSettings['smiley_enable']) ? 7200 : 480;
if (($temp = cache_get_data('parsing_smileys_' . $user_info['smiley_set'], $cache_time)) == null)
{
$result = $smcFunc['db_query']('', '
SELECT s.code, f.filename, s.description
FROM {db_prefix}smileys AS s
JOIN {db_prefix}smiley_files AS f ON (s.id_smiley = f.id_smiley)
WHERE f.smiley_set = {string:smiley_set}' . (empty($modSettings['smiley_enable']) ? '
AND s.code IN ({array_string:default_codes})' : '') . '
ORDER BY LENGTH(s.code) DESC',
array(
'default_codes' => array('>:D', ':D', '::)', '>:(', ':))', ':)', ';)', ';D', ':(', ':o', '8)', ':P', '???', ':-[', ':-X', ':-*', ':\'(', ':-\\', '^-^', 'O0', 'C:-)', 'O:-)'),
'smiley_set' => $user_info['smiley_set'],
)
);
$smileysfrom = array();
$smileysto = array();
$smileysdescs = array();
while ($row = $smcFunc['db_fetch_assoc']($result))
{
$smileysfrom[] = $row['code'];
$smileysto[] = $smcFunc['htmlspecialchars']($row['filename']);
$smileysdescs[] = !empty($txt['icon_' . strtolower($row['description'])]) ? $txt['icon_' . strtolower($row['description'])] : $row['description'];
}
$smcFunc['db_free_result']($result);
cache_put_data('parsing_smileys_' . $user_info['smiley_set'], array($smileysfrom, $smileysto, $smileysdescs), $cache_time);
}
else
list ($smileysfrom, $smileysto, $smileysdescs) = $temp;
$non_breaking_space = $context['utf8'] ? '\x{A0}' : '\xA0';
$smileyPregReplacements = array();
$searchParts = array();
$smileys_path = $smcFunc['htmlspecialchars']($modSettings['smileys_url'] . '/' . $user_info['smiley_set'] . '/');
for ($i = 0, $n = count($smileysfrom); $i < $n; $i++)
{
$specialChars = $smcFunc['htmlspecialchars']($smileysfrom[$i], ENT_QUOTES);
$smileyCode = '<img src="' . $smileys_path . $smileysto[$i] . '" alt="' . strtr($specialChars, array(':' => ':', '(' => '(', ')' => ')', '$' => '$', '[' => '[')) . '" title="' . strtr($smcFunc['htmlspecialchars']($smileysdescs[$i]), array(':' => ':', '(' => '(', ')' => ')', '$' => '$', '[' => '[')) . '" class="smiley">';
$smileyPregReplacements[$smileysfrom[$i]] = $smileyCode;
$searchParts[] = $smileysfrom[$i];
if ($smileysfrom[$i] != $specialChars)
{
$smileyPregReplacements[$specialChars] = $smileyCode;
$searchParts[] = $specialChars;
$specialChars2 = preg_replace('/&#(\d{2});/', '�$1;', $specialChars);
if ($specialChars2 != $specialChars)
{
$smileyPregReplacements[$specialChars2] = $smileyCode;
$searchParts[] = $specialChars2;
}
}
}
$smileyPregSearch = '~(?<=[>:\?\.\s' . $non_breaking_space . '[\]()*\\\;]|(?<![a-zA-Z0-9])\(|^)(' . build_regex($searchParts, '~') . ')(?=[^[:alpha:]0-9]|$)~' . ($context['utf8'] ? 'u' : '');
}
$message = preg_replace_callback($smileyPregSearch, function($matches) use ($smileyPregReplacements)
{
return $smileyPregReplacements[$matches[1]];
}, $message);
}
function highlight_php_code($code)
{
$code = un_htmlspecialchars(strtr($code, array('<br />' => "\n", '<br>' => "\n", "\t" => 'SMF_TAB();', '[' => '[')));
$oldlevel = error_reporting(0);
$buffer = str_replace(array("\n", "\r"), '', @highlight_string($code, true));
error_reporting($oldlevel);
$buffer = preg_replace('~SMF_TAB(?:</(?:font|span)><(?:font color|span style)="[^"]*?">)?\\(\\);~', '<pre style="display: inline;">' . "\t" . '</pre>', $buffer);
return strtr($buffer, array('\'' => ''', '<code>' => '', '</code>' => ''));
}
function get_proxied_url($url)
{
global $boardurl, $image_proxy_enabled, $image_proxy_secret, $user_info;
if (empty($image_proxy_enabled) || !empty($user_info['possibly_robot']))
return $url;
$parsedurl = parse_url($url);
if (empty($parsedurl['scheme']) || empty($parsedurl['host']) || empty($parsedurl['path']) || $parsedurl['scheme'] === 'https')
return $url;
if ($parsedurl['host'] === parse_url($boardurl, PHP_URL_HOST))
return strtr($url, array('http://' => 'https://'));
$proxied_url = strtr($boardurl, array('http://' => 'https://')) . '/proxy.php?request=' . urlencode($url) . '&hash=' . hash_hmac('sha1', $url, $image_proxy_secret);
call_integration_hook('integrate_proxy', array($url, &$proxied_url));
return $proxied_url;
}
function redirectexit($setLocation = '', $refresh = false, $permanent = false)
{
global $scripturl, $context, $modSettings, $db_show_debug, $db_cache;
if (!empty($context['flush_mail']))
AddMailQueue(true);
$add = preg_match('~^(ftp|http)[s]?://~', $setLocation) == 0 && substr($setLocation, 0, 6) != 'about:';
if ($add)
$setLocation = $scripturl . ($setLocation != '' ? '?' . $setLocation : '');
if (defined('SID') && SID != '')
$setLocation = preg_replace('/^' . preg_quote($scripturl, '/') . '(?!\?' . preg_quote(SID, '/') . ')\\??/', $scripturl . '?' . SID . ';', $setLocation);
elseif (isset($_GET['debug']))
$setLocation = preg_replace('/^' . preg_quote($scripturl, '/') . '\\??/', $scripturl . '?debug;', $setLocation);
if (!empty($modSettings['queryless_urls']) && (empty($context['server']['is_cgi']) || ini_get('cgi.fix_pathinfo') == 1 || @get_cfg_var('cgi.fix_pathinfo') == 1) && (!empty($context['server']['is_apache']) || !empty($context['server']['is_lighttpd']) || !empty($context['server']['is_litespeed'])))
{
if (defined('SID') && SID != '')
$setLocation = preg_replace_callback('~^' . preg_quote($scripturl, '~') . '\?(?:' . SID . '(?:;|&|&))((?:board|topic)=[^#]+?)(#[^"]*?)?$~',
function($m) use ($scripturl)
{
return $scripturl . '/' . strtr("$m[1]", '&;=', '//,') . '.html?' . SID . (isset($m[2]) ? "$m[2]" : "");
}, $setLocation);
else
$setLocation = preg_replace_callback('~^' . preg_quote($scripturl, '~') . '\?((?:board|topic)=[^#"]+?)(#[^"]*?)?$~',
function($m) use ($scripturl)
{
return $scripturl . '/' . strtr("$m[1]", '&;=', '//,') . '.html' . (isset($m[2]) ? "$m[2]" : "");
}, $setLocation);
}
call_integration_hook('integrate_redirect', array(&$setLocation, &$refresh, &$permanent));
header('location: ' . str_replace(' ', '%20', $setLocation), true, $permanent ? 301 : 302);
if (isset($db_show_debug) && $db_show_debug === true)
$_SESSION['debug_redirect'] = $db_cache;
obExit(false);
}
function obExit($header = null, $do_footer = null, $from_index = false, $from_fatal_error = false)
{
global $context, $settings, $modSettings, $txt, $smcFunc, $should_log;
static $header_done = false, $footer_done = false, $level = 0, $has_fatal_error = false;
++$level;
if ($level > 1 && !$from_fatal_error && !$has_fatal_error)
exit;
if ($from_fatal_error)
$has_fatal_error = true;
if (function_exists('trackStats'))
trackStats();
if (function_exists('AddMailQueue') && !empty($context['flush_mail']))
AddMailQueue(true);
$do_header = $header === null ? !$header_done : $header;
if ($do_footer === null)
$do_footer = $do_header;
if ($do_header)
{
if (!empty($context['page_title']) && empty($context['page_title_html_safe']))
$context['page_title_html_safe'] = $smcFunc['htmlspecialchars'](un_htmlspecialchars($context['page_title'])) . (!empty($context['current_page']) ? ' - ' . $txt['page'] . ' ' . ($context['current_page'] + 1) : '');
ob_start('ob_sessrewrite');
if (!empty($settings['output_buffers']) && is_string($settings['output_buffers']))
$buffers = explode(',', $settings['output_buffers']);
elseif (!empty($settings['output_buffers']))
$buffers = $settings['output_buffers'];
else
$buffers = array();
if (isset($modSettings['integrate_buffer']))
$buffers = array_merge(explode(',', $modSettings['integrate_buffer']), $buffers);
if (!empty($buffers))
foreach ($buffers as $function)
{
$call = call_helper($function, true);
if (!empty($call))
ob_start($call);
}
template_header();
$header_done = true;
}
if ($do_footer)
{
loadSubTemplate(isset($context['sub_template']) ? $context['sub_template'] : 'main');
if (!empty($context['insert_after_template']) && !isset($_REQUEST['xml']))
echo $context['insert_after_template'];
if (!$footer_done)
{
$footer_done = true;
template_footer();
if (!isset($_REQUEST['xml']))
displayDebug();
}
}
if ($should_log)
$_SESSION['old_url'] = $_SERVER['REQUEST_URL'];
$_SESSION['USER_AGENT'] = empty($_SERVER['HTTP_USER_AGENT']) ? '' : $_SERVER['HTTP_USER_AGENT'];
call_integration_hook('integrate_exit', array($do_footer));
if (!$from_index)
exit;
}
function url_image_size($url)
{
global $sourcedir;
$url = str_replace(' ', '%20', $url);
if (($temp = cache_get_data('url_image_size-' . md5($url), 240)) !== null)
return $temp;
$t = microtime(true);
preg_match('~^\w+://(.+?)/(.*)$~', $url, $match);
if ($url == '' || $url == 'http://' || $url == 'https://')
{
return false;
}
elseif (!isset($match[1]))
{
$size = @getimagesize($url);
}
else
{
$temp = 0;
$fp = @fsockopen($match[1], 80, $temp, $temp, 0.5);
if ($fp != false)
{
fwrite($fp, 'HEAD /' . $match[2] . ' HTTP/1.1' . "\r\n" . 'Host: ' . $match[1] . "\r\n" . 'user-agent: '. SMF_USER_AGENT . "\r\n" . 'Connection: close' . "\r\n\r\n");
$test = substr(fgets($fp, 11), -1);
fclose($fp);
if ($test < 4)
{
$size = @getimagesize($url);
if ($size === false && function_exists('imagecreatefromstring'))
{
$image = @imagecreatefromstring(fetch_web_data($url));
if ($image !== false)
{
$size = array(imagesx($image), imagesy($image));
imagedestroy($image);
}
}
}
}
}
if (!isset($size))
$size = false;
if (microtime(true) - $t > 0.8)
cache_put_data('url_image_size-' . md5($url), $size, 240);
return $size;
}
function setupThemeContext($forceload = false)
{
global $modSettings, $user_info, $scripturl, $context, $settings, $options, $txt, $maintenance;
global $smcFunc;
static $loaded = false;
if ($loaded && !$forceload)
return;
$loaded = true;
$context['in_maintenance'] = !empty($maintenance);
$context['current_time'] = timeformat(time(), false);
$context['current_action'] = isset($_GET['action']) ? $smcFunc['htmlspecialchars']($_GET['action']) : '';
$context['random_news_line'] = array();
$context['news_lines'] = array_filter(explode("\n", str_replace("\r", '', trim(addslashes($modSettings['news'])))));
for ($i = 0, $n = count($context['news_lines']); $i < $n; $i++)
{
if (trim($context['news_lines'][$i]) == '')
continue;
$context['news_lines'][$i] = parse_bbc(stripslashes(trim($context['news_lines'][$i])), true, 'news' . $i);
}
if (!empty($context['news_lines']) && (!empty($modSettings['allow_guestAccess']) || $context['user']['is_logged']))
$context['random_news_line'] = $context['news_lines'][mt_rand(0, count($context['news_lines']) - 1)];
if (!$user_info['is_guest'])
{
$context['user']['messages'] = &$user_info['messages'];
$context['user']['unread_messages'] = &$user_info['unread_messages'];
$context['user']['alerts'] = &$user_info['alerts'];
if ($user_info['unread_messages'] > (isset($_SESSION['unread_messages']) ? $_SESSION['unread_messages'] : 0))
$context['user']['popup_messages'] = true;
else
$context['user']['popup_messages'] = false;
$_SESSION['unread_messages'] = $user_info['unread_messages'];
if (allowedTo('moderate_forum'))
$context['unapproved_members'] = !empty($modSettings['unapprovedMembers']) ? $modSettings['unapprovedMembers'] : 0;
$context['user']['avatar'] = array();
if (($modSettings['gravatarEnabled'] && substr($user_info['avatar']['url'], 0, 11) == 'gravatar://') || !empty($modSettings['gravatarOverride']))
{
if (!empty($modSettings['gravatarAllowExtraEmail']) && stristr($user_info['avatar']['url'], 'gravatar://') && strlen($user_info['avatar']['url']) > 11)
$context['user']['avatar']['href'] = get_gravatar_url($smcFunc['substr']($user_info['avatar']['url'], 11));
else
$context['user']['avatar']['href'] = get_gravatar_url($user_info['email']);
}
elseif ($user_info['avatar']['url'] == '' && !empty($user_info['avatar']['id_attach']))
$context['user']['avatar']['href'] = $user_info['avatar']['custom_dir'] ? $modSettings['custom_avatar_url'] . '/' . $user_info['avatar']['filename'] : $scripturl . '?action=dlattach;attach=' . $user_info['avatar']['id_attach'] . ';type=avatar';
elseif (strpos($user_info['avatar']['url'], 'http://') === 0 || strpos($user_info['avatar']['url'], 'https://') === 0)
$context['user']['avatar']['href'] = $user_info['avatar']['url'];
elseif ($user_info['avatar']['url'] != '')
$context['user']['avatar']['href'] = $modSettings['avatar_url'] . '/' . $smcFunc['htmlspecialchars']($user_info['avatar']['url']);
else
$context['user']['avatar']['href'] = $modSettings['avatar_url'] . '/default.png';
if (!empty($context['user']['avatar']))
$context['user']['avatar']['image'] = '<img src="' . $context['user']['avatar']['href'] . '" alt="" class="avatar">';
$context['user']['total_time_logged_in'] = array(
'days' => floor($user_info['total_time_logged_in'] / 86400),
'hours' => floor(($user_info['total_time_logged_in'] % 86400) / 3600),
'minutes' => floor(($user_info['total_time_logged_in'] % 3600) / 60)
);
}
else
{
$context['user']['messages'] = 0;
$context['user']['unread_messages'] = 0;
$context['user']['avatar'] = array();
$context['user']['total_time_logged_in'] = array('days' => 0, 'hours' => 0, 'minutes' => 0);
$context['user']['popup_messages'] = false;
if (!empty($modSettings['registration_method']) && $modSettings['registration_method'] == 1)
$txt['welcome_guest'] .= $txt['welcome_guest_activate'];
if (!empty($modSettings['disableHashTime']) && ($modSettings['disableHashTime'] == 1 || time() < $modSettings['disableHashTime']))
$context['disable_login_hashing'] = true;
}
setupMenuContext();
$context['show_news'] = !empty($settings['enable_news']);
$context['show_pm_popup'] = $context['user']['popup_messages'] && !empty($options['popup_messages']) && (!isset($_REQUEST['action']) || $_REQUEST['action'] != 'pm');
if ($context['show_pm_popup'])
addInlineJavaScript('
jQuery(document).ready(function($) {
new smc_Popup({
heading: ' . JavaScriptEscape($txt['show_personal_messages_heading']) . ',
content: ' . JavaScriptEscape(sprintf($txt['show_personal_messages'], $context['user']['unread_messages'], $scripturl . '?action=pm')) . ',
icon_class: \'main_icons mail_new\'
});
});');
addInlineJavaScript('
var smf_you_sure =' . JavaScriptEscape($txt['quickmod_confirm']) . ';');
if (!empty($modSettings['avatar_max_width_external']) && !empty($modSettings['avatar_max_height_external']) && !empty($modSettings['avatar_action_too_large']) && $modSettings['avatar_action_too_large'] == 'option_css_resize')
addInlineCss('
img.avatar { max-width: ' . $modSettings['avatar_max_width_external'] . 'px; max-height: ' . $modSettings['avatar_max_height_external'] . 'px; }');
if (!empty($modSettings['max_image_width']))
addInlineCss('
.postarea .bbc_img { max-width: ' . $modSettings['max_image_width'] . 'px; }');
if (!empty($modSettings['max_image_height']))
addInlineCss('
.postarea .bbc_img { max-height: ' . $modSettings['max_image_height'] . 'px; }');
$context['common_stats']['latest_member'] = array(
'id' => $modSettings['latestMember'],
'name' => $modSettings['latestRealName'],
'href' => $scripturl . '?action=profile;u=' . $modSettings['latestMember'],
'link' => '<a href="' . $scripturl . '?action=profile;u=' . $modSettings['latestMember'] . '">' . $modSettings['latestRealName'] . '</a>',
);
$context['common_stats'] = array(
'total_posts' => comma_format($modSettings['totalMessages']),
'total_topics' => comma_format($modSettings['totalTopics']),
'total_members' => comma_format($modSettings['totalMembers']),
'latest_member' => $context['common_stats']['latest_member'],
);
$context['common_stats']['boardindex_total_posts'] = sprintf($txt['boardindex_total_posts'], $context['common_stats']['total_posts'], $context['common_stats']['total_topics'], $context['common_stats']['total_members']);
if (empty($settings['theme_version']))
addJavaScriptVar('smf_scripturl', $scripturl);
if (!isset($context['page_title']))
$context['page_title'] = '';
$context['page_title_html_safe'] = $smcFunc['htmlspecialchars'](un_htmlspecialchars($context['page_title'])) . (!empty($context['current_page']) ? ' - ' . $txt['page'] . ' ' . ($context['current_page'] + 1) : '');
$context['meta_keywords'] = !empty($modSettings['meta_keywords']) ? $smcFunc['htmlspecialchars']($modSettings['meta_keywords']) : '';
$context['meta_tags'][] = array('property' => 'og:site_name', 'content' => $context['forum_name']);
$context['meta_tags'][] = array('property' => 'og:title', 'content' => $context['page_title_html_safe']);
if (!empty($context['meta_keywords']))
$context['meta_tags'][] = array('name' => 'keywords', 'content' => $context['meta_keywords']);
if (!empty($context['canonical_url']))
$context['meta_tags'][] = array('property' => 'og:url', 'content' => $context['canonical_url']);
if (!empty($settings['og_image']))
$context['meta_tags'][] = array('property' => 'og:image', 'content' => $settings['og_image']);
if (!empty($context['meta_description']))
{
$context['meta_tags'][] = array('property' => 'og:description', 'content' => $context['meta_description']);
$context['meta_tags'][] = array('name' => 'description', 'content' => $context['meta_description']);
}
else
{
$context['meta_tags'][] = array('property' => 'og:description', 'content' => $context['page_title_html_safe']);
$context['meta_tags'][] = array('name' => 'description', 'content' => $context['page_title_html_safe']);
}
call_integration_hook('integrate_theme_context');
}
function setMemoryLimit($needed, $in_use = false)
{
$memory_current = memoryReturnBytes(ini_get('memory_limit'));
$memory_needed = memoryReturnBytes($needed);
if ($in_use)
$memory_needed += function_exists('memory_get_usage') ? memory_get_usage() : (2 * 1048576);
if ($memory_current < $memory_needed)
{
@ini_set('memory_limit', ceil($memory_needed / 1048576) . 'M');
$memory_current = memoryReturnBytes(ini_get('memory_limit'));
}
$memory_current = max($memory_current, memoryReturnBytes(get_cfg_var('memory_limit')));
return (bool) ($memory_current >= $memory_needed);
}
function memoryReturnBytes($val)
{
if (is_integer($val))
return $val;
$val = trim($val);
$num = intval(substr($val, 0, strlen($val) - 1));
$last = strtolower(substr($val, -1));
switch ($last)
{
case 'g':
$num *= 1024;
case 'm':
$num *= 1024;
case 'k':
$num *= 1024;
}
return $num;
}
function template_header()
{
global $txt, $modSettings, $context, $user_info, $boarddir, $cachedir, $cache_enable;
setupThemeContext();
if (empty($context['no_last_modified']))
{
header('expires: Mon, 26 Jul 1997 05:00:00 GMT');
header('last-modified: ' . gmdate('D, d M Y H:i:s') . ' GMT');
if (!isset($_REQUEST['xml']) && isset($_GET['debug']) && !isBrowser('ie'))
header('content-type: application/xhtml+xml');
elseif (!isset($_REQUEST['xml']))
header('content-type: text/html; charset=' . (empty($context['character_set']) ? 'ISO-8859-1' : $context['character_set']));
}
header('content-type: text/' . (isset($_REQUEST['xml']) ? 'xml' : 'html') . '; charset=' . (empty($context['character_set']) ? 'ISO-8859-1' : $context['character_set']));
if ($context['in_maintenance'] && $context['user']['is_admin'])
{
$position = array_search('body', $context['template_layers']);
if ($position === false)
$position = array_search('main', $context['template_layers']);
if ($position !== false)
{
$before = array_slice($context['template_layers'], 0, $position + 1);
$after = array_slice($context['template_layers'], $position + 1);
$context['template_layers'] = array_merge($before, array('maint_warning'), $after);
}
}
$checked_securityFiles = false;
$showed_banned = false;
foreach ($context['template_layers'] as $layer)
{
loadSubTemplate($layer . '_above', true);
if (in_array($layer, array('body', 'main')) && allowedTo('admin_forum') && !$user_info['is_guest'] && !$checked_securityFiles)
{
$checked_securityFiles = true;
$securityFiles = array('install.php', 'upgrade.php', 'convert.php', 'repair_paths.php', 'repair_settings.php', 'Settings.php~', 'Settings_bak.php~');
call_integration_hook('integrate_security_files', array(&$securityFiles));
foreach ($securityFiles as $i => $securityFile)
{
if (!file_exists($boarddir . '/' . $securityFile))
unset($securityFiles[$i]);
}
if (!empty($modSettings['currentAttachmentUploadDir']))
$path = $modSettings['attachmentUploadDir'][$modSettings['currentAttachmentUploadDir']];
else
$path = $modSettings['attachmentUploadDir'];
secureDirectory($path, true);
secureDirectory($cachedir);
if ($modSettings['requireAgreement'])
$agreement = !file_exists($boarddir . '/agreement.txt');
if (!empty($securityFiles) || (!empty($cache_enable) && !is_writable($cachedir)) || !empty($agreement))
{
echo '
<div class="errorbox">
<p class="alert">!!</p>
<h3>', empty($securityFiles) ? $txt['generic_warning'] : $txt['security_risk'], '</h3>
<p>';
foreach ($securityFiles as $securityFile)
{
echo '
', $txt['not_removed'], '<strong>', $securityFile, '</strong>!<br>';
if ($securityFile == 'Settings.php~' || $securityFile == 'Settings_bak.php~')
echo '
', sprintf($txt['not_removed_extra'], $securityFile, substr($securityFile, 0, -1)), '<br>';
}
if (!empty($cache_enable) && !is_writable($cachedir))
echo '
<strong>', $txt['cache_writable'], '</strong><br>';
if (!empty($agreement))
echo '
<strong>', $txt['agreement_missing'], '</strong><br>';
echo '
</p>
</div>';
}
}
elseif (in_array($layer, array('main', 'body')) && isset($_SESSION['ban']['cannot_post']) && !$showed_banned)
{
$showed_banned = true;
echo '
<div class="windowbg alert" style="margin: 2ex; padding: 2ex; border: 2px dashed red;">
', sprintf($txt['you_are_post_banned'], $user_info['is_guest'] ? $txt['guest_title'] : $user_info['name']);
if (!empty($_SESSION['ban']['cannot_post']['reason']))
echo '
<div style="padding-left: 4ex; padding-top: 1ex;">', $_SESSION['ban']['cannot_post']['reason'], '</div>';
if (!empty($_SESSION['ban']['expire_time']))
echo '
<div>', sprintf($txt['your_ban_expires'], timeformat($_SESSION['ban']['expire_time'], false)), '</div>';
else
echo '
<div>', $txt['your_ban_expires_never'], '</div>';
echo '
</div>';
}
}
}
function theme_copyright()
{
global $forum_copyright;
if (SMF !== 1)
return;
printf($forum_copyright, SMF_FULL_VERSION, SMF_SOFTWARE_YEAR);
}
function template_footer()
{
global $context, $modSettings, $db_count;
$context['show_load_time'] = !empty($modSettings['timeLoadPageEnable']);
$context['load_time'] = round(microtime(true) - TIME_START, 3);
$context['load_queries'] = $db_count;
if (!empty($context['template_layers']) && is_array($context['template_layers']))
foreach (array_reverse($context['template_layers']) as $layer)
loadSubTemplate($layer . '_below', true);
}
function template_javascript($do_deferred = false)
{
global $context, $modSettings, $settings;
call_integration_hook('integrate_pre_javascript_output', array(&$do_deferred));
$toMinify = array(
'standard' => array(),
'defer' => array(),
'async' => array(),
);
if (!empty($context['javascript_vars']) && !$do_deferred)
{
echo '
<script>';
foreach ($context['javascript_vars'] as $key => $value)
{
if (empty($value))
{
echo '
var ', $key, ';';
}
else
{
echo '
var ', $key, ' = ', $value, ';';
}
}
echo '
</script>';
}
if (!$do_deferred)
{
foreach ($context['javascript_files'] as $id => $js_file)
{
if (!empty($settings['disable_files']) && in_array($id, $settings['disable_files']))
continue;
if (!empty($js_file['options']['minimize']) && !empty($modSettings['minimize_files']))
{
if (!empty($js_file['options']['async']))
$toMinify['async'][] = $js_file;
elseif (!empty($js_file['options']['defer']))
$toMinify['defer'][] = $js_file;
else
$toMinify['standard'][] = $js_file;
if (!isset($minSeed) && isset($js_file['options']['seed']))
$minSeed = $js_file['options']['seed'];
}
else
{
echo '
<script src="', $js_file['fileUrl'], isset($js_file['options']['seed']) ? $js_file['options']['seed'] : '', '"', !empty($js_file['options']['async']) ? ' async' : '', !empty($js_file['options']['defer']) ? ' defer' : '';
if (!empty($js_file['options']['attributes']))
foreach ($js_file['options']['attributes'] as $key => $value)
{
if (is_bool($value))
echo !empty($value) ? ' ' . $key : '';
else
echo ' ', $key, '="', $value, '"';
}
echo '></script>';
}
}
foreach ($toMinify as $js_files)
{
if (!empty($js_files))
{
$result = custMinify($js_files, 'js');
$minSuccessful = array_keys($result) === array('smf_minified');
foreach ($result as $minFile)
echo '
<script src="', $minFile['fileUrl'], $minSuccessful && isset($minSeed) ? $minSeed : '', '"', !empty($minFile['options']['async']) ? ' async' : '', !empty($minFile['options']['defer']) ? ' defer' : '', '></script>';
}
}
}
if (!empty($context['javascript_inline']))
{
if (!empty($context['javascript_inline']['defer']) && $do_deferred)
{
echo '
<script>
window.addEventListener("DOMContentLoaded", function() {';
foreach ($context['javascript_inline']['defer'] as $js_code)
echo $js_code;
echo '
});
</script>';
}
if (!empty($context['javascript_inline']['standard']) && !$do_deferred)
{
echo '
<script>';
foreach ($context['javascript_inline']['standard'] as $js_code)
echo $js_code;
echo '
</script>';
}
}
}
function template_css()
{
global $context, $db_show_debug, $boardurl, $settings, $modSettings;
call_integration_hook('integrate_pre_css_output');
$toMinify = array();
$normal = array();
usort($context['css_files'], function ($a, $b)
{
return $a['options']['order_pos'] < $b['options']['order_pos'] ? -1 : ($a['options']['order_pos'] > $b['options']['order_pos'] ? 1 : 0);
});
foreach ($context['css_files'] as $id => $file)
{
if (!empty($settings['disable_files']) && in_array($id, $settings['disable_files']))
continue;
if (!isset($file['options']['minimize']))
$file['options']['minimize'] = true;
if (!empty($file['options']['minimize']) && !empty($modSettings['minimize_files']) && !isset($_REQUEST['normalcss']))
{
$toMinify[] = $file;
if (!isset($minSeed) && isset($file['options']['seed']))
$minSeed = $file['options']['seed'];
}
else
$normal[] = array(
'url' => $file['fileUrl'] . (isset($file['options']['seed']) ? $file['options']['seed'] : ''),
'attributes' => !empty($file['options']['attributes']) ? $file['options']['attributes'] : array()
);
}
if (!empty($toMinify))
{
$result = custMinify($toMinify, 'css');
$minSuccessful = array_keys($result) === array('smf_minified');
foreach ($result as $minFile)
echo '
<link rel="stylesheet" href="', $minFile['fileUrl'], $minSuccessful && isset($minSeed) ? $minSeed : '', '">';
}
if (!empty($normal))
foreach ($normal as $nf)
{
echo '
<link rel="stylesheet" href="', $nf['url'], '"';
if (!empty($nf['attributes']))
foreach ($nf['attributes'] as $key => $value)
{
if (is_bool($value))
echo !empty($value) ? ' ' . $key : '';
else
echo ' ', $key, '="', $value, '"';
}
echo '>';
}
if ($db_show_debug === true)
{
$repl = array($boardurl . '/Themes/' => '', $boardurl . '/' => '');
foreach ($context['css_files'] as $file)
$context['debug']['sheets'][] = strtr($file['fileName'], $repl);
}
if (!empty($context['css_header']))
{
echo '
<style>';
foreach ($context['css_header'] as $css)
echo $css . '
';
echo '
</style>';
}
}
function custMinify($data, $type)
{
global $settings, $txt;
$types = array('css', 'js');
$type = !empty($type) && in_array($type, $types) ? $type : false;
$data = is_array($data) ? $data : array();
if (empty($type) || empty($data))
return $data;
$hash = md5(implode(' ', array_map(function($file)
{
return $file['filePath'] . '-' . $file['mtime'];
}, $data)));
$async = $type === 'js';
$defer = $type === 'js';
if ($type === 'js')
{
foreach ($data as $id => $file)
{
if (empty($file['options']['async']))
$async = false;
if (empty($file['options']['defer']))
$defer = false;
}
}
$minified_file = $settings['theme_dir'] . '/' . ($type == 'css' ? 'css' : 'scripts') . '/minified_' . $hash . '.' . $type;
$already_exists = file_exists($minified_file);
if ($already_exists)
{
return array('smf_minified' => array(
'fileUrl' => $settings['theme_url'] . '/' . ($type == 'css' ? 'css' : 'scripts') . '/' . basename($minified_file),
'filePath' => $minified_file,
'fileName' => basename($minified_file),
'options' => array('async' => !empty($async), 'defer' => !empty($defer)),
));
}
elseif (@fopen($minified_file, 'w') === false || !smf_chmod($minified_file))
{
loadLanguage('Errors');
log_error(sprintf($txt['file_not_created'], $minified_file), 'general');
return $data;
}
$classType = 'MatthiasMullie\\Minify\\' . strtoupper($type);
$minifier = new $classType();
foreach ($data as $id => $file)
{
$toAdd = !empty($file['filePath']) && file_exists($file['filePath']) ? $file['filePath'] : false;
if (empty($toAdd))
{
loadLanguage('Errors');
log_error(sprintf($txt['file_minimize_fail'], !empty($file['fileName']) ? $file['fileName'] : $id), 'general');
continue;
}
$minifier->add($toAdd);
}
$minifier->minify($minified_file);
unset($minifier);
clearstatcache();
if (!filesize($minified_file))
{
loadLanguage('Errors');
log_error(sprintf($txt['file_not_created'], $minified_file), 'general');
return $data;
}
return array('smf_minified' => array(
'fileUrl' => $settings['theme_url'] . '/' . ($type == 'css' ? 'css' : 'scripts') . '/' . basename($minified_file),
'filePath' => $minified_file,
'fileName' => basename($minified_file),
'options' => array('async' => $async, 'defer' => $defer),
));
}
function deleteAllMinified()
{
global $smcFunc, $txt, $modSettings;
$not_deleted = array();
$most_recent = 0;
$request = $smcFunc['db_query']('', '
SELECT id_theme AS id, value AS dir
FROM {db_prefix}themes
WHERE variable = {string:var}',
array(
'var' => 'theme_dir',
)
);
while ($theme = $smcFunc['db_fetch_assoc']($request))
{
foreach (array('css', 'js') as $type)
{
foreach (glob(rtrim($theme['dir'], '/') . '/' . ($type == 'css' ? 'css' : 'scripts') . '/*.' . $type) as $filename)
{
if (strpos(pathinfo($filename, PATHINFO_BASENAME), 'minified') === false)
$most_recent = max($modSettings['browser_cache'], (int) @filemtime($filename));
elseif (!@unlink($filename))
$not_deleted[] = $filename;
}
}
}
$smcFunc['db_free_result']($request);
if ($most_recent > $modSettings['browser_cache'])
updateSettings(array('browser_cache' => $most_recent));
if (!empty($not_deleted))
{
loadLanguage('Errors');
log_error(sprintf($txt['unlink_minimized_fail'], implode('<br>', $not_deleted)), 'general');
}
}
function getAttachmentFilename($filename, $attachment_id, $dir = null, $new = false, $file_hash = '')
{
global $modSettings, $smcFunc;
if ($new)
return sha1(md5($filename . time()) . mt_rand());
$attachment_id = (int) $attachment_id;
if ($file_hash === '')
{
$request = $smcFunc['db_query']('', '
SELECT file_hash
FROM {db_prefix}attachments
WHERE id_attach = {int:id_attach}',
array(
'id_attach' => $attachment_id,
)
);
if ($smcFunc['db_num_rows']($request) === 0)
return false;
list ($file_hash) = $smcFunc['db_fetch_row']($request);
$smcFunc['db_free_result']($request);
}
if (empty($file_hash))
$file_hash = sha1(md5($filename . time()) . mt_rand());
if (is_array($modSettings['attachmentUploadDir']))
$path = $modSettings['attachmentUploadDir'][$dir];
else
$path = $modSettings['attachmentUploadDir'];
return $path . '/' . $attachment_id . '_' . $file_hash . '.dat';
}
function ip2range($fullip)
{
if ($fullip == 'unknown')
$fullip = '255.255.255.255';
$ip_parts = explode('-', $fullip);
$ip_array = array();
if (count($ip_parts) == 1 && isValidIP($fullip))
{
$ip_array['low'] = $fullip;
$ip_array['high'] = $fullip;
return $ip_array;
}
elseif (count($ip_parts) == 1)
{
$ip_parts[0] = $fullip;
$ip_parts[1] = $fullip;
}
if (count($ip_parts) == 2 && isValidIP($ip_parts[0]) && isValidIP($ip_parts[1]))
{
$ip_array['low'] = $ip_parts[0];
$ip_array['high'] = $ip_parts[1];
return $ip_array;
}
elseif (count($ip_parts) == 2)
{
$valid_low = isValidIP($ip_parts[0]);
$valid_high = isValidIP($ip_parts[1]);
$count = 0;
$mode = (preg_match('/:/', $ip_parts[0]) > 0 ? ':' : '.');
$max = ($mode == ':' ? 'ffff' : '255');
$min = 0;
if (!$valid_low)
{
$ip_parts[0] = preg_replace('/\*/', '0', $ip_parts[0]);
$valid_low = isValidIP($ip_parts[0]);
while (!$valid_low)
{
$ip_parts[0] .= $mode . $min;
$valid_low = isValidIP($ip_parts[0]);
$count++;
if ($count > 9) break;
}
}
$count = 0;
if (!$valid_high)
{
$ip_parts[1] = preg_replace('/\*/', $max, $ip_parts[1]);
$valid_high = isValidIP($ip_parts[1]);
while (!$valid_high)
{
$ip_parts[1] .= $mode . $max;
$valid_high = isValidIP($ip_parts[1]);
$count++;
if ($count > 9) break;
}
}
if ($valid_high && $valid_low)
{
$ip_array['low'] = $ip_parts[0];
$ip_array['high'] = $ip_parts[1];
}
}
return $ip_array;
}
function host_from_ip($ip)
{
global $modSettings;
if (($host = cache_get_data('hostlookup-' . $ip, 600)) !== null)
return $host;
$t = microtime(true);
if (!isset($host) && (strpos(strtolower(PHP_OS), 'win') === false || strpos(strtolower(PHP_OS), 'darwin') !== false) && mt_rand(0, 1) == 1)
{
if (!isset($modSettings['host_to_dis']))
$test = @shell_exec('host -W 1 ' . @escapeshellarg($ip));
else
$test = @shell_exec('host ' . @escapeshellarg($ip));
if (strpos($test, 'not found') !== false)
$host = '';
elseif ((strpos($test, 'invalid option') || strpos($test, 'Invalid query name 1')) && !isset($modSettings['host_to_dis']))
updateSettings(array('host_to_dis' => 1));
elseif (preg_match('~\s([^\s]+?)\.\s~', $test, $match) == 1)
$host = $match[1];
}
if (!isset($host) && stripos(PHP_OS, 'win') !== false && strpos(strtolower(PHP_OS), 'darwin') === false && mt_rand(0, 1) == 1)
{
$test = @shell_exec('nslookup -timeout=1 ' . @escapeshellarg($ip));
if (strpos($test, 'Non-existent domain') !== false)
$host = '';
elseif (preg_match('~Name:\s+([^\s]+)~', $test, $match) == 1)
$host = $match[1];
}
if (!isset($host) || $host === false)
$host = @gethostbyaddr($ip);
if (microtime(true) - $t > 0.5)
cache_put_data('hostlookup-' . $ip, $host, 600);
return $host;
}
function text2words($text, $max_chars = 20, $encrypt = false)
{
global $smcFunc, $context;
$words = preg_replace('~(?:[\x0B\0' . ($context['utf8'] ? '\x{A0}' : '\xA0') . '\t\r\s\n(){}\\[\\]<>!@$%^*.,:+=`\~\?/\\\\]+|&(?:amp|lt|gt|quot);)+~' . ($context['utf8'] ? 'u' : ''), ' ', strtr($text, array('<br>' => ' ')));
$words = un_htmlspecialchars($smcFunc['strtolower']($words));
$words = explode(' ', $words);
if ($encrypt)
{
$possible_chars = array_flip(array_merge(range(46, 57), range(65, 90), range(97, 122)));
$returned_ints = array();
foreach ($words as $word)
{
if (($word = trim($word, '-_\'')) !== '')
{
$encrypted = substr(crypt($word, 'uk'), 2, $max_chars);
$total = 0;
for ($i = 0; $i < $max_chars; $i++)
$total += $possible_chars[ord($encrypted[$i])] * pow(63, $i);
$returned_ints[] = $max_chars == 4 ? min($total, 16777215) : $total;
}
}
return array_unique($returned_ints);
}
else
{
$returned_words = array();
foreach ($words as $word)
if (($word = trim($word, '-_\'')) !== '')
$returned_words[] = $max_chars === null ? $word : substr($word, 0, $max_chars);
return array_unique($returned_words);
}
}
function create_button($name, $alt, $label = '', $custom = '', $force_use = false)
{
global $settings, $txt;
if (function_exists('template_create_button') && !$force_use)
return template_create_button($name, $alt, $label = '', $custom = '');
if (!$settings['use_image_buttons'])
return $txt[$alt];
elseif (!empty($settings['use_buttons']))
return '<span class="main_icons ' . $name . '" alt="' . $txt[$alt] . '"></span>' . ($label != '' ? ' <strong>' . $txt[$label] . '</strong>' : '');
else
return '<img src="' . $settings['lang_images_url'] . '/' . $name . '" alt="' . $txt[$alt] . '" ' . $custom . '>';
}
function setupMenuContext()
{
global $context, $modSettings, $user_info, $txt, $scripturl, $sourcedir, $settings, $smcFunc, $cache_enable;
$context['allow_search'] = !empty($modSettings['allow_guestAccess']) ? allowedTo('search_posts') : (!$user_info['is_guest'] && allowedTo('search_posts'));
$context['allow_admin'] = allowedTo(array('admin_forum', 'manage_boards', 'manage_permissions', 'moderate_forum', 'manage_membergroups', 'manage_bans', 'send_mail', 'edit_news', 'manage_attachments', 'manage_smileys'));
$context['allow_memberlist'] = allowedTo('view_mlist');
$context['allow_calendar'] = allowedTo('calendar_view') && !empty($modSettings['cal_enabled']);
$context['allow_moderation_center'] = $context['user']['can_mod'];
$context['allow_pm'] = allowedTo('pm_read');
$cacheTime = $modSettings['lastActive'] * 60;
if (!isset($context['allow_calendar_event']))
{
$context['allow_calendar_event'] = $context['allow_calendar'] && allowedTo('calendar_post');
if ($context['allow_calendar'] && $context['allow_calendar_event'] && empty($modSettings['cal_allow_unlinked']) && !$user_info['is_admin'])
{
$boards_can_post = boardsAllowedTo('post_new');
$context['allow_calendar_event'] &= !empty($boards_can_post);
}
}
if (!$context['user']['is_guest'])
{
addInlineJavaScript('
var user_menus = new smc_PopupMenu();
user_menus.add("profile", "' . $scripturl . '?action=profile;area=popup");
user_menus.add("alerts", "' . $scripturl . '?action=profile;area=alerts_popup;u=' . $context['user']['id'] . '");', true);
if ($context['allow_pm'])
addInlineJavaScript('
user_menus.add("pm", "' . $scripturl . '?action=pm;sa=popup");', true);
if (!empty($modSettings['enable_ajax_alerts']))
{
require_once($sourcedir . '/Subs-Notify.php');
$timeout = getNotifyPrefs($context['user']['id'], 'alert_timeout', true);
$timeout = empty($timeout) ? 10000 : $timeout[$context['user']['id']]['alert_timeout'] * 1000;
addInlineJavaScript('
var new_alert_title = "' . $context['forum_name'] . '";
var alert_timeout = ' . $timeout . ';');
loadJavaScriptFile('alerts.js', array('minimize' => true), 'smf_alerts');
}
}
if (($menu_buttons = cache_get_data('menu_buttons-' . implode('_', $user_info['groups']) . '-' . $user_info['language'], $cacheTime)) === null || time() - $cacheTime <= $modSettings['settings_updated'])
{
$buttons = array(
'home' => array(
'title' => $txt['home'],
'href' => $scripturl,
'show' => true,
'sub_buttons' => array(
),
'is_last' => $context['right_to_left'],
),
'search' => array(
'title' => $txt['search'],
'href' => $scripturl . '?action=search',
'show' => $context['allow_search'],
'sub_buttons' => array(
),
),
'admin' => array(
'title' => $txt['admin'],
'href' => $scripturl . '?action=admin',
'show' => $context['allow_admin'],
'sub_buttons' => array(
'featuresettings' => array(
'title' => $txt['modSettings_title'],
'href' => $scripturl . '?action=admin;area=featuresettings',
'show' => allowedTo('admin_forum'),
),
'packages' => array(
'title' => $txt['package'],
'href' => $scripturl . '?action=admin;area=packages',
'show' => allowedTo('admin_forum'),
),
'errorlog' => array(
'title' => $txt['errorlog'],
'href' => $scripturl . '?action=admin;area=logs;sa=errorlog;desc',
'show' => allowedTo('admin_forum') && !empty($modSettings['enableErrorLogging']),
),
'permissions' => array(
'title' => $txt['edit_permissions'],
'href' => $scripturl . '?action=admin;area=permissions',
'show' => allowedTo('manage_permissions'),
),
'memberapprove' => array(
'title' => $txt['approve_members_waiting'],
'href' => $scripturl . '?action=admin;area=viewmembers;sa=browse;type=approve',
'show' => !empty($context['unapproved_members']),
'is_last' => true,
),
),
),
'moderate' => array(
'title' => $txt['moderate'],
'href' => $scripturl . '?action=moderate',
'show' => $context['allow_moderation_center'],
'sub_buttons' => array(
'modlog' => array(
'title' => $txt['modlog_view'],
'href' => $scripturl . '?action=moderate;area=modlog',
'show' => !empty($modSettings['modlog_enabled']) && !empty($user_info['mod_cache']) && $user_info['mod_cache']['bq'] != '0=1',
),
'poststopics' => array(
'title' => $txt['mc_unapproved_poststopics'],
'href' => $scripturl . '?action=moderate;area=postmod;sa=posts',
'show' => $modSettings['postmod_active'] && !empty($user_info['mod_cache']['ap']),
),
'attachments' => array(
'title' => $txt['mc_unapproved_attachments'],
'href' => $scripturl . '?action=moderate;area=attachmod;sa=attachments',
'show' => $modSettings['postmod_active'] && !empty($user_info['mod_cache']['ap']),
),
'reports' => array(
'title' => $txt['mc_reported_posts'],
'href' => $scripturl . '?action=moderate;area=reportedposts',
'show' => !empty($user_info['mod_cache']) && $user_info['mod_cache']['bq'] != '0=1',
),
'reported_members' => array(
'title' => $txt['mc_reported_members'],
'href' => $scripturl . '?action=moderate;area=reportedmembers',
'show' => allowedTo('moderate_forum'),
'is_last' => true,
)
),
),
'calendar' => array(
'title' => $txt['calendar'],
'href' => $scripturl . '?action=calendar',
'show' => $context['allow_calendar'],
'sub_buttons' => array(
'view' => array(
'title' => $txt['calendar_menu'],
'href' => $scripturl . '?action=calendar',
'show' => $context['allow_calendar_event'],
),
'post' => array(
'title' => $txt['calendar_post_event'],
'href' => $scripturl . '?action=calendar;sa=post',
'show' => $context['allow_calendar_event'],
'is_last' => true,
),
),
),
'mlist' => array(
'title' => $txt['members_title'],
'href' => $scripturl . '?action=mlist',
'show' => $context['allow_memberlist'],
'sub_buttons' => array(
'mlist_view' => array(
'title' => $txt['mlist_menu_view'],
'href' => $scripturl . '?action=mlist',
'show' => true,
),
'mlist_search' => array(
'title' => $txt['mlist_search'],
'href' => $scripturl . '?action=mlist;sa=search',
'show' => true,
'is_last' => true,
),
),
'is_last' => !$context['right_to_left'] && (!$user_info['is_guest'] || !$context['can_register']),
),
'signup' => array(
'title' => $txt['register'],
'href' => $scripturl . '?action=signup',
'show' => $user_info['is_guest'] && $context['can_register'],
'sub_buttons' => array(
),
'is_last' => !$context['right_to_left'],
),
);
call_integration_hook('integrate_menu_buttons', array(&$buttons));
$menu_buttons = array();
foreach ($buttons as $act => $button)
if (!empty($button['show']))
{
$button['active_button'] = false;
if (isset($button['action_hook']))
$needs_action_hook = true;
if (!empty($button['is_last']))
{
if (isset($last_button))
unset($menu_buttons[$last_button]['is_last']);
$last_button = $act;
}
if (!empty($button['sub_buttons']))
foreach ($button['sub_buttons'] as $key => $subbutton)
{
if (empty($subbutton['show']))
unset($button['sub_buttons'][$key]);
if (!empty($subbutton['sub_buttons']))
{
foreach ($subbutton['sub_buttons'] as $key2 => $sub_button2)
{
if (empty($sub_button2['show']))
unset($button['sub_buttons'][$key]['sub_buttons'][$key2]);
}
}
}
if (isset($button['icon']) && file_exists($settings['theme_dir'] . '/images/' . $button['icon']))
$button['icon'] = '<img src="' . $settings['images_url'] . '/' . $button['icon'] . '" alt="">';
elseif (isset($button['icon']) && file_exists($settings['default_theme_dir'] . '/images/' . $button['icon']))
$button['icon'] = '<img src="' . $settings['default_images_url'] . '/' . $button['icon'] . '" alt="">';
elseif (isset($button['icon']))
$button['icon'] = '<span class="main_icons ' . $button['icon'] . '"></span>';
else
$button['icon'] = '<span class="main_icons ' . $act . '"></span>';
$menu_buttons[$act] = $button;
}
if (!empty($cache_enable) && $cache_enable >= 2)
cache_put_data('menu_buttons-' . implode('_', $user_info['groups']) . '-' . $user_info['language'], $menu_buttons, $cacheTime);
}
$context['menu_buttons'] = $menu_buttons;
if (isset($context['menu_buttons']['logout']))
$context['menu_buttons']['logout']['href'] = sprintf($context['menu_buttons']['logout']['href'], $context['session_var'], $context['session_id']);
$current_action = 'home';
if (isset($context['menu_buttons'][$context['current_action']]))
$current_action = $context['current_action'];
elseif ($context['current_action'] == 'search2')
$current_action = 'search';
elseif ($context['current_action'] == 'theme')
$current_action = isset($_REQUEST['sa']) && $_REQUEST['sa'] == 'pick' ? 'profile' : 'admin';
elseif ($context['current_action'] == 'register2')
$current_action = 'register';
elseif ($context['current_action'] == 'login2' || ($user_info['is_guest'] && $context['current_action'] == 'reminder'))
$current_action = 'login';
elseif ($context['current_action'] == 'groups' && $context['allow_moderation_center'])
$current_action = 'moderate';
if ($context['current_action'] == 'profile' && !empty($context['user']['is_owner']))
{
$current_action = !empty($_GET['area']) && $_GET['area'] == 'showalerts' ? 'self_alerts' : 'self_profile';
$context[$current_action] = true;
}
elseif ($context['current_action'] == 'pm')
{
$current_action = 'self_pm';
$context['self_pm'] = true;
}
$context['total_mod_reports'] = 0;
$context['total_admin_reports'] = 0;
if (!empty($user_info['mod_cache']) && $user_info['mod_cache']['bq'] != '0=1' && !empty($context['open_mod_reports']) && !empty($context['menu_buttons']['moderate']['sub_buttons']['reports']))
{
$context['total_mod_reports'] = $context['open_mod_reports'];
$context['menu_buttons']['moderate']['sub_buttons']['reports']['amt'] = $context['open_mod_reports'];
}
if (!empty($context['menu_buttons']['admin']['sub_buttons']['errorlog']))
{
if (!isset($context['num_errors']))
{
$query = $smcFunc['db_query']('', '
SELECT COUNT(*)
FROM {db_prefix}log_errors',
array()
);
list($context['num_errors']) = $smcFunc['db_fetch_row']($query);
$smcFunc['db_free_result']($query);
}
if (!empty($context['num_errors']))
{
$context['total_admin_reports'] += $context['num_errors'];
$context['menu_buttons']['admin']['sub_buttons']['errorlog']['amt'] = $context['num_errors'];
}
}
if (!empty($context['open_member_reports']) && !empty($context['menu_buttons']['moderate']['sub_buttons']['reported_members']))
{
$context['total_mod_reports'] += $context['open_member_reports'];
$context['menu_buttons']['moderate']['sub_buttons']['reported_members']['amt'] = $context['open_member_reports'];
}
if (!empty($context['unapproved_members']) && !empty($context['menu_buttons']['admin']))
{
$context['menu_buttons']['admin']['sub_buttons']['memberapprove']['amt'] = $context['unapproved_members'];
$context['total_admin_reports'] += $context['unapproved_members'];
}
if ($context['total_admin_reports'] > 0 && !empty($context['menu_buttons']['admin']))
{
$context['menu_buttons']['admin']['amt'] = $context['total_admin_reports'];
}
if ($context['total_mod_reports'] > 0 && !empty($context['menu_buttons']['moderate']))
{
$context['menu_buttons']['moderate']['amt'] = $context['total_mod_reports'];
}
if (!empty($needs_action_hook))
call_integration_hook('integrate_current_action', array(&$current_action));
if (isset($context['menu_buttons'][$current_action]))
$context['menu_buttons'][$current_action]['active_button'] = true;
}
function smf_seed_generator()
{
updateSettings(array('rand_seed' => microtime(true)));
}
function call_integration_hook($hook, $parameters = array())
{
global $modSettings, $settings, $boarddir, $sourcedir, $db_show_debug;
global $context, $txt;
if ($db_show_debug === true)
$context['debug']['hooks'][] = $hook;
if (!isset($context['instances']))
$context['instances'] = array();
$results = array();
if (empty($modSettings[$hook]))
return $results;
$functions = explode(',', $modSettings[$hook]);
foreach ($functions as $function)
{
if (strpos($function, '!') !== false)
continue;
$call = call_helper($function, true);
if (!empty($call))
$results[$function] = call_user_func_array($call, $parameters);
elseif (!empty($function) && !empty($context['ignore_hook_errors']))
return $results;
elseif (!empty($function))
{
loadLanguage('Errors');
if (strpos($function, '|') !== false)
{
list ($file, $string) = explode('|', $function);
$absPath = empty($settings['theme_dir']) ? (strtr(trim($file), array('$boarddir' => $boarddir, '$sourcedir' => $sourcedir))) : (strtr(trim($file), array('$boarddir' => $boarddir, '$sourcedir' => $sourcedir, '$themedir' => $settings['theme_dir'])));
log_error(sprintf($txt['hook_fail_call_to'], $string, $absPath), 'general');
}
else
log_error(sprintf($txt['hook_fail_call_to'], $function, $boarddir), 'general');
}
}
return $results;
}
function add_integration_function($hook, $function, $permanent = true, $file = '', $object = false)
{
global $smcFunc, $modSettings;
if ($object)
$function = $function . '#';
if (!empty($file) && is_string($file))
$function = $file . (!empty($function) ? '|' . $function : '');
$integration_call = $function;
if ($permanent)
{
$request = $smcFunc['db_query']('', '
SELECT value
FROM {db_prefix}settings
WHERE variable = {string:variable}',
array(
'variable' => $hook,
)
);
list ($current_functions) = $smcFunc['db_fetch_row']($request);
$smcFunc['db_free_result']($request);
if (!empty($current_functions))
{
$current_functions = explode(',', $current_functions);
if (in_array($integration_call, $current_functions))
return;
$permanent_functions = array_merge($current_functions, array($integration_call));
}
else
$permanent_functions = array($integration_call);
updateSettings(array($hook => implode(',', $permanent_functions)));
}
$functions = empty($modSettings[$hook]) ? array() : explode(',', $modSettings[$hook]);
if (in_array($integration_call, $functions))
return;
$functions[] = $integration_call;
$modSettings[$hook] = implode(',', $functions);
}
function remove_integration_function($hook, $function, $permanent = true, $file = '', $object = false)
{
global $smcFunc, $modSettings;
if ($object)
$function = $function . '#';
if (!empty($file) && is_string($file))
$function = $file . '|' . $function;
$integration_call = $function;
$request = $smcFunc['db_query']('', '
SELECT value
FROM {db_prefix}settings
WHERE variable = {string:variable}',
array(
'variable' => $hook,
)
);
list ($current_functions) = $smcFunc['db_fetch_row']($request);
$smcFunc['db_free_result']($request);
if (!empty($current_functions))
{
$current_functions = explode(',', $current_functions);
if (in_array($integration_call, $current_functions))
updateSettings(array($hook => implode(',', array_diff($current_functions, array($integration_call)))));
}
$functions = empty($modSettings[$hook]) ? array() : explode(',', $modSettings[$hook]);
if (!in_array($integration_call, $functions))
return;
$functions = array_diff($functions, array($integration_call));
$modSettings[$hook] = implode(',', $functions);
}
function call_helper($string, $return = false)
{
global $context, $smcFunc, $txt, $db_show_debug;
if (empty($string))
return false;
if (is_array($string) || $string instanceof Closure)
return $return ? $string : (is_callable($string) ? call_user_func($string) : false);
if (is_object($string))
return false;
$string = $smcFunc['htmlspecialchars']($smcFunc['htmltrim']($string));
$string = load_file($string);
if (empty($string))
return false;
if (strpos($string, '::') !== false)
{
list ($class, $method) = explode('::', $string);
if (strpos($method, '#') !== false)
{
$method = str_replace('#', '', $method);
if (empty($context['instances'][$class]) || !($context['instances'][$class] instanceof $class))
{
$context['instances'][$class] = new $class;
if ($db_show_debug === true)
{
if (!isset($context['debug']['instances']))
$context['debug']['instances'] = array();
$context['debug']['instances'][$class] = $class;
}
}
$func = array($context['instances'][$class], $method);
}
else
$func = array($class, $method);
}
else
$func = $string;
if (!is_callable($func, false, $callable_name) && !empty($context['ignore_hook_errors']))
return false;
elseif (!is_callable($func, false, $callable_name))
{
loadLanguage('Errors');
log_error(sprintf($txt['sub_action_fail'], $callable_name), 'general');
return false;
}
else
{
if ($return)
return $func;
else
{
if (is_array($func))
call_user_func($func);
else
$func();
}
}
}
function load_file($string)
{
global $sourcedir, $txt, $boarddir, $settings, $context;
if (empty($string))
return false;
if (strpos($string, '|') !== false)
{
list ($file, $string) = explode('|', $string);
if (empty($settings['theme_dir']))
$absPath = strtr(trim($file), array('$boarddir' => $boarddir, '$sourcedir' => $sourcedir));
else
$absPath = strtr(trim($file), array('$boarddir' => $boarddir, '$sourcedir' => $sourcedir, '$themedir' => $settings['theme_dir']));
if (file_exists($absPath))
require_once($absPath);
else
{
$absPath = $sourcedir . '/' . $file;
if (file_exists($absPath))
require_once($absPath);
elseif (empty($context['uninstalling']))
{
loadLanguage('Errors');
log_error(sprintf($txt['hook_fail_loading_file'], $absPath), 'general');
return false;
}
}
}
return $string;
}
function fetch_web_data($url, $post_data = '', $keep_alive = false, $redirection_level = 0)
{
global $webmaster_email, $sourcedir;
static $keep_alive_dom = null, $keep_alive_fp = null;
preg_match('~^(http|ftp)(s)?://([^/:]+)(:(\d+))?(.+)$~', $url, $match);
if (empty($match[1]))
return false;
elseif ($match[1] == 'ftp')
{
require_once($sourcedir . '/Class-Package.php');
$ftp = new ftp_connection(($match[2] ? 'ssl://' : '') . $match[3], empty($match[5]) ? 21 : $match[5], 'anonymous', $webmaster_email);
if ($ftp->error !== false || !$ftp->passive())
return false;
fwrite($ftp->connection, 'RETR ' . $match[6] . "\r\n");
$fp = @fsockopen($ftp->pasv['ip'], $ftp->pasv['port'], $err, $err, 5);
if (!$fp)
return false;
$ftp->check_response(150);
$data = '';
while (!feof($fp))
$data .= fread($fp, 4096);
fclose($fp);
$ftp->check_response(226);
$ftp->close();
}
elseif (isset($match[1]) && $match[1] == 'http')
{
if ($keep_alive && $match[3] == $keep_alive_dom)
$fp = $keep_alive_fp;
if (empty($fp))
{
$fp = @fsockopen(($match[2] ? 'ssl://' : '') . $match[3], empty($match[5]) ? ($match[2] ? 443 : 80) : $match[5], $err, $err, 5);
}
if (!empty($fp))
{
if ($keep_alive)
{
$keep_alive_dom = $match[3];
$keep_alive_fp = $fp;
}
if (empty($post_data))
{
fwrite($fp, 'GET ' . ($match[6] !== '/' ? str_replace(' ', '%20', $match[6]) : '') . ' HTTP/1.0' . "\r\n");
fwrite($fp, 'Host: ' . $match[3] . (empty($match[5]) ? ($match[2] ? ':443' : '') : ':' . $match[5]) . "\r\n");
fwrite($fp, 'user-agent: '. SMF_USER_AGENT . "\r\n");
if ($keep_alive)
fwrite($fp, 'connection: Keep-Alive' . "\r\n\r\n");
else
fwrite($fp, 'connection: close' . "\r\n\r\n");
}
else
{
fwrite($fp, 'POST ' . ($match[6] !== '/' ? $match[6] : '') . ' HTTP/1.0' . "\r\n");
fwrite($fp, 'Host: ' . $match[3] . (empty($match[5]) ? ($match[2] ? ':443' : '') : ':' . $match[5]) . "\r\n");
fwrite($fp, 'user-agent: '. SMF_USER_AGENT . "\r\n");
if ($keep_alive)
fwrite($fp, 'connection: Keep-Alive' . "\r\n");
else
fwrite($fp, 'connection: close' . "\r\n");
fwrite($fp, 'content-type: application/x-www-form-urlencoded' . "\r\n");
fwrite($fp, 'content-length: ' . strlen($post_data) . "\r\n\r\n");
fwrite($fp, $post_data);
}
$response = fgets($fp, 768);
if ($redirection_level < 3 && preg_match('~^HTTP/\S+\s+30[127]~i', $response) === 1)
{
$header = '';
$location = '';
while (!feof($fp) && trim($header = fgets($fp, 4096)) != '')
if (stripos($header, 'location:') !== false)
$location = trim(substr($header, strpos($header, ':') + 1));
if (empty($location))
return false;
else
{
if (!$keep_alive)
fclose($fp);
return fetch_web_data($location, $post_data, $keep_alive, $redirection_level + 1);
}
}
elseif (preg_match('~^HTTP/\S+\s+20[01]~i', $response) === 0)
return false;
while (!feof($fp) && trim($header = fgets($fp, 4096)) != '')
{
if (preg_match('~content-length:\s*(\d+)~i', $header, $match) != 0)
$content_length = $match[1];
elseif (preg_match('~connection:\s*close~i', $header) != 0)
{
$keep_alive_dom = null;
$keep_alive = false;
}
continue;
}
$data = '';
if (isset($content_length))
{
while (!feof($fp) && strlen($data) < $content_length)
$data .= fread($fp, $content_length - strlen($data));
}
else
{
while (!feof($fp))
$data .= fread($fp, 4096);
}
if (!$keep_alive)
fclose($fp);
}
elseif (function_exists('curl_init'))
{
require_once($sourcedir . '/Class-CurlFetchWeb.php');
$fetch_data = new curl_fetch_web_data();
$fetch_data->get_url_data($url, $post_data);
if ($fetch_data->result('code') == 200 && !$fetch_data->result('error'))
$data = $fetch_data->result('body');
else
return false;
}
else
return false;
}
else
{
trigger_error('fetch_web_data(): Bad URL', E_USER_NOTICE);
$data = false;
}
return $data;
}
function prepareLikesContext($topic)
{
global $user_info, $smcFunc;
if (empty($topic))
return array();
$user = $user_info['id'];
$cache_key = 'likes_topic_' . $topic . '_' . $user;
$ttl = 180;
if (($temp = cache_get_data($cache_key, $ttl)) === null)
{
$temp = array();
$request = $smcFunc['db_query']('', '
SELECT content_id
FROM {db_prefix}user_likes AS l
INNER JOIN {db_prefix}messages AS m ON (l.content_id = m.id_msg)
WHERE l.id_member = {int:current_user}
AND l.content_type = {literal:msg}
AND m.id_topic = {int:topic}',
array(
'current_user' => $user,
'topic' => $topic,
)
);
while ($row = $smcFunc['db_fetch_assoc']($request))
$temp[] = (int) $row['content_id'];
cache_put_data($cache_key, $temp, $ttl);
}
return $temp;
}
function sanitizeMSCutPaste($string)
{
global $context;
if (empty($string))
return $string;
$findchars_utf8 = array(
"\xe2\x80\x9a",
"\xe2\x80\x9e",
"\xe2\x80\xa6",
"\xe2\x80\x98",
"\xe2\x80\x99",
"\xe2\x80\x9c",
"\xe2\x80\x9d",
);
$findchars_iso = array(
chr(130),
chr(132),
chr(133),
chr(145),
chr(146),
chr(147),
chr(148),
);
$replacechars = array(
',',
',,',
'...',
"'",
"'",
'"',
'"',
);
if ($context['utf8'])
$string = str_replace($findchars_utf8, $replacechars, $string);
else
$string = str_replace($findchars_iso, $replacechars, $string);
return $string;
}
function replaceEntities__callback($matches)
{
global $context;
if (!isset($matches[2]))
return '';
$num = $matches[2][0] === 'x' ? hexdec(substr($matches[2], 1)) : (int) $matches[2];
if ($num === 0x202D || $num === 0x202E)
return '';
if (in_array($num, array(0x22, 0x26, 0x27, 0x3C, 0x3E)))
return '&#' . $num . ';';
if (empty($context['utf8']))
{
if ($num < 0x20)
return '';
elseif ($num < 0x80)
return chr($num);
else
return '&#' . $matches[2] . ';';
}
else
{
if ($num < 0x20 || $num > 0x10FFFF || ($num >= 0xD800 && $num <= 0xDFFF))
return '';
elseif ($num < 0x80)
return chr($num);
elseif ($num < 0x800)
return chr(($num >> 6) + 192) . chr(($num & 63) + 128);
elseif ($num < 0x10000)
return chr(($num >> 12) + 224) . chr((($num >> 6) & 63) + 128) . chr(($num & 63) + 128);
else
return chr(($num >> 18) + 240) . chr((($num >> 12) & 63) + 128) . chr((($num >> 6) & 63) + 128) . chr(($num & 63) + 128);
}
}
function fixchar__callback($matches)
{
if (!isset($matches[1]))
return '';
$num = $matches[1][0] === 'x' ? hexdec(substr($matches[1], 1)) : (int) $matches[1];
if ($num < 0x20 || $num > 0x10FFFF || ($num >= 0xD800 && $num <= 0xDFFF) || $num === 0x202D || $num === 0x202E)
return '';
elseif ($num < 0x80)
return chr($num);
elseif ($num < 0x800)
return chr(($num >> 6) + 192) . chr(($num & 63) + 128);
elseif ($num < 0x10000)
return chr(($num >> 12) + 224) . chr((($num >> 6) & 63) + 128) . chr(($num & 63) + 128);
else
return chr(($num >> 18) + 240) . chr((($num >> 12) & 63) + 128) . chr((($num >> 6) & 63) + 128) . chr(($num & 63) + 128);
}
function entity_fix__callback($matches)
{
if (!isset($matches[2]))
return '';
$num = $matches[2][0] === 'x' ? hexdec(substr($matches[2], 1)) : (int) $matches[2];
if ($num < 0x20 || $num > 0x10FFFF || ($num >= 0xD800 && $num <= 0xDFFF) || $num == 0x202D || $num == 0x202E)
return '';
else
return '&#' . $num . ';';
}
function get_gravatar_url($email_address)
{
global $modSettings, $smcFunc;
static $url_params = null;
if ($url_params === null)
{
$ratings = array('G', 'PG', 'R', 'X');
$defaults = array('mm', 'identicon', 'monsterid', 'wavatar', 'retro', 'blank');
$url_params = array();
if (!empty($modSettings['gravatarMaxRating']) && in_array($modSettings['gravatarMaxRating'], $ratings))
$url_params[] = 'rating=' . $modSettings['gravatarMaxRating'];
if (!empty($modSettings['gravatarDefault']) && in_array($modSettings['gravatarDefault'], $defaults))
$url_params[] = 'default=' . $modSettings['gravatarDefault'];
if (!empty($modSettings['avatar_max_width_external']))
$size_string = (int) $modSettings['avatar_max_width_external'];
if (!empty($modSettings['avatar_max_height_external']) && !empty($size_string))
if ((int) $modSettings['avatar_max_height_external'] < $size_string)
$size_string = $modSettings['avatar_max_height_external'];
if (!empty($size_string))
$url_params[] = 's=' . $size_string;
}
$http_method = !empty($modSettings['force_ssl']) ? 'https://secure' : 'http://www';
return $http_method . '.gravatar.com/avatar/' . md5($smcFunc['strtolower']($email_address)) . '?' . implode('&', $url_params);
}
function smf_list_timezones($when = 'now')
{
global $smcFunc, $modSettings, $tztxt, $txt, $cur_profile;
static $timezones = null, $lastwhen = null;
if (!empty($timezones) && $when == $lastwhen)
return $timezones;
else
$lastwhen = $when;
if (is_int($timestamp = strtotime($when)))
$when = $timestamp;
elseif (is_numeric($when))
$when = intval($when);
else
$when = time();
$date_when = date_create('@' . $when);
$later = strtotime('@' . $when . ' + 1 year');
loadLanguage('Timezones');
$priority_countries = !empty($modSettings['timezone_priority_countries']) ? explode(',', $modSettings['timezone_priority_countries']) : array();
$priority_tzids = array();
foreach ($priority_countries as $country)
{
$country_tzids = @timezone_identifiers_list(DateTimeZone::PER_COUNTRY, strtoupper(trim($country)));
if (!empty($country_tzids))
$priority_tzids = array_merge($priority_tzids, $country_tzids);
}
$low_priority_tzids = !in_array('AQ', $priority_countries) ? timezone_identifiers_list(DateTimeZone::ANTARCTICA) : array();
$tzids = array_merge(array_keys($tztxt), array_diff(timezone_identifiers_list(), array_keys($tztxt), $low_priority_tzids), $low_priority_tzids);
foreach ($tzids as $tzid)
{
if ($tzid == 'UTC')
continue;
$tz = timezone_open($tzid);
$tzinfo = timezone_transitions_get($tz, $when, $later);
$tzkey = serialize($tzinfo);
$tzgeo = timezone_location_get($tz);
if (empty($zones[$tzkey]['tzid']))
{
$zones[$tzkey]['tzid'] = $tzid;
$zones[$tzkey]['abbr'] = $tzinfo[0]['abbr'];
}
if (in_array($tzid, $priority_tzids))
$priority_zones[$tzkey] = true;
if (!empty($txt[$tzid]))
$zones[$tzkey]['locations'][] = $txt[$tzid];
else
{
$tzid_parts = explode('/', $tzid);
$zones[$tzkey]['locations'][] = str_replace(array('St_', '_'), array('St. ', ' '), array_pop($tzid_parts));
}
$offsets[$tzkey] = $tzinfo[0]['offset'];
$longitudes[$tzkey] = empty($longitudes[$tzkey]) ? $tzgeo['longitude'] : $longitudes[$tzkey];
if (isset($cur_profile['timezone']) && $cur_profile['timezone'] == $tzid)
$member_tzkey = $tzkey;
}
array_multisort($offsets, SORT_ASC, SORT_NUMERIC, $longitudes, SORT_ASC, SORT_NUMERIC, $zones);
$priority_timezones = array();
$timezones = array();
foreach ($zones as $tzkey => $tzvalue)
{
date_timezone_set($date_when, timezone_open($tzvalue['tzid']));
if (!empty($tztxt[$tzvalue['tzid']]))
$desc = $tztxt[$tzvalue['tzid']];
else
$desc = implode(', ', array_slice(array_unique($tzvalue['locations']), 0, 5)) . (count($tzvalue['locations']) > 5 ? ', ' . $txt['etc'] : '');
$desc = '[UTC' . date_format($date_when, 'P') . '] - ' . (!strspn($tzvalue['abbr'], '+-') ? $tzvalue['abbr'] . ' - ' : '') . $desc;
if (isset($priority_zones[$tzkey]))
$priority_timezones[$tzvalue['tzid']] = $desc;
else
$timezones[$tzvalue['tzid']] = $desc;
if (isset($member_tzkey) && $member_tzkey == $tzkey)
$cur_profile['timezone'] = $tzvalue['tzid'];
}
if (!empty($priority_timezones))
$priority_timezones[] = '-----';
$timezones = array_merge(
$priority_timezones,
array('UTC' => 'UTC' . (!empty($tztxt['UTC']) ? ' - ' . $tztxt['UTC'] : ''), '-----'),
$timezones
);
return $timezones;
}
function inet_ptod($ip_address)
{
if (!isValidIP($ip_address))
return $ip_address;
$bin = inet_pton($ip_address);
return $bin;
}
function inet_dtop($bin)
{
global $db_type;
if (empty($bin))
return '';
elseif ($db_type == 'postgresql')
return $bin;
elseif (isValidIP($bin))
return $bin;
return inet_ntop($bin);
}
function _safe_serialize($value)
{
if (is_null($value))
return 'N;';
if (is_bool($value))
return 'b:' . (int) $value . ';';
if (is_int($value))
return 'i:' . $value . ';';
if (is_float($value))
return 'd:' . str_replace(',', '.', $value) . ';';
if (is_string($value))
return 's:' . strlen($value) . ':"' . $value . '";';
if (is_array($value))
{
$out = '';
foreach ($value as $k => $v)
$out .= _safe_serialize($k) . _safe_serialize($v);
return 'a:' . count($value) . ':{' . $out . '}';
}
return false;
}
function safe_serialize($value)
{
if (function_exists('mb_internal_encoding') &&
(((int) ini_get('mbstring.func_overload')) & 2))
{
$mbIntEnc = mb_internal_encoding();
mb_internal_encoding('ASCII');
}
$out = _safe_serialize($value);
if (isset($mbIntEnc))
mb_internal_encoding($mbIntEnc);
return $out;
}
function _safe_unserialize($str)
{
if (empty($str) || !is_string($str))
return false;
$stack = array();
$expected = array();
$state = 0;
while ($state != 1)
{
$type = isset($str[0]) ? $str[0] : '';
if ($type == '}')
$str = substr($str, 1);
elseif ($type == 'N' && $str[1] == ';')
{
$value = null;
$str = substr($str, 2);
}
elseif ($type == 'b' && preg_match('/^b:([01]);/', $str, $matches))
{
$value = $matches[1] == '1' ? true : false;
$str = substr($str, 4);
}
elseif ($type == 'i' && preg_match('/^i:(-?[0-9]+);(.*)/s', $str, $matches))
{
$value = (int) $matches[1];
$str = $matches[2];
}
elseif ($type == 'd' && preg_match('/^d:(-?[0-9]+\.?[0-9]*(E[+-][0-9]+)?);(.*)/s', $str, $matches))
{
$value = (float) $matches[1];
$str = $matches[3];
}
elseif ($type == 's' && preg_match('/^s:([0-9]+):"(.*)/s', $str, $matches) && substr($matches[2], (int) $matches[1], 2) == '";')
{
$value = substr($matches[2], 0, (int) $matches[1]);
$str = substr($matches[2], (int) $matches[1] + 2);
}
elseif ($type == 'a' && preg_match('/^a:([0-9]+):{(.*)/s', $str, $matches))
{
$expectedLength = (int) $matches[1];
$str = $matches[2];
}
else
return false;
switch ($state)
{
case 3:
if ($type == 'a')
{
$stack[] = &$list;
$list[$key] = array();
$list = &$list[$key];
$expected[] = $expectedLength;
$state = 2;
break;
}
if ($type != '}')
{
$list[$key] = $value;
$state = 2;
break;
}
return false;
case 2:
if ($type == '}')
{
if (count($list) < end($expected))
return false;
unset($list);
$list = &$stack[count($stack) - 1];
array_pop($stack);
array_pop($expected);
if (count($expected) == 0)
$state = 1;
break;
}
if ($type == 'i' || $type == 's')
{
if (count($list) >= end($expected))
return false;
$key = $value;
$state = 3;
break;
}
return false;
case 0:
if ($type == 'a')
{
$data = array();
$list = &$data;
$expected[] = $expectedLength;
$state = 2;
break;
}
if ($type != '}')
{
$data = $value;
$state = 1;
break;
}
return false;
}
}
if (!empty($str))
return false;
return $data;
}
function safe_unserialize($str)
{
if (function_exists('mb_internal_encoding') &&
(((int) ini_get('mbstring.func_overload')) & 0x02))
{
$mbIntEnc = mb_internal_encoding();
mb_internal_encoding('ASCII');
}
$out = _safe_unserialize($str);
if (isset($mbIntEnc))
mb_internal_encoding($mbIntEnc);
return $out;
}
function smf_chmod($file, $value = 0)
{
if (empty($file))
return false;
if (is_writable($file))
return true;
$isDir = is_dir($file);
$isWritable = false;
$chmodValues = $isDir ? array(0750, 0755, 0775, 0777) : array(0644, 0664, 0666);
foreach ($chmodValues as $val)
{
if (is_writable($file))
{
$isWritable = true;
break;
}
else
@chmod($file, $val);
}
return $isWritable;
}
function smf_json_decode($json, $returnAsArray = false, $logIt = true)
{
global $txt;
if (empty($json) || !is_string($json))
return array();
$returnArray = @json_decode($json, $returnAsArray);
switch (json_last_error())
{
case JSON_ERROR_NONE:
$jsonError = false;
break;
case JSON_ERROR_DEPTH:
$jsonError = 'JSON_ERROR_DEPTH';
break;
case JSON_ERROR_STATE_MISMATCH:
$jsonError = 'JSON_ERROR_STATE_MISMATCH';
break;
case JSON_ERROR_CTRL_CHAR:
$jsonError = 'JSON_ERROR_CTRL_CHAR';
break;
case JSON_ERROR_SYNTAX:
$jsonError = 'JSON_ERROR_SYNTAX';
break;
case JSON_ERROR_UTF8:
$jsonError = 'JSON_ERROR_UTF8';
break;
default:
$jsonError = 'unknown';
break;
}
if (!empty($jsonError) && $logIt)
{
$jsonDebug = debug_backtrace();
$jsonDebug = $jsonDebug[0];
loadLanguage('Errors');
if (!empty($jsonDebug))
log_error($txt['json_' . $jsonError], 'critical', $jsonDebug['file'], $jsonDebug['line']);
else
log_error($txt['json_' . $jsonError], 'critical');
return array();
}
return $returnArray;
}
function isValidIP($IPString)
{
return filter_var($IPString, FILTER_VALIDATE_IP) !== false;
}
function smf_serverResponse($data = '', $type = 'content-type: application/json')
{
global $db_show_debug, $modSettings;
if (empty($data))
return false;
$db_show_debug = false;
ob_end_clean();
if (!empty($modSettings['CompressedOutput']))
@ob_start('ob_gzhandler');
else
ob_start();
header($type);
echo $data;
obExit(false);
}
function set_tld_regex($update = false)
{
global $sourcedir, $smcFunc, $modSettings;
static $done = false;
if (!$update && $done)
return;
if ($update)
{
$tlds = fetch_web_data('https://data.iana.org/TLD/tlds-alpha-by-domain.txt');
$tlds_md5 = fetch_web_data('https://data.iana.org/TLD/tlds-alpha-by-domain.txt.md5');
if ($tlds === false || $tlds_md5 === false)
$postapocalypticNightmare = true;
if (md5($tlds) != substr($tlds_md5, 0, 32))
$tlds = array();
}
elseif (!empty($modSettings['tld_regex']) && @preg_match('~' . $modSettings['tld_regex'] . '~', null) !== false)
{
$done = true;
return;
}
if (!empty($tlds))
{
$tlds = array_filter(explode("\n", strtolower($tlds)), function($line)
{
$line = trim($line);
if (empty($line) || strlen($line) != strspn($line, 'abcdefghijklmnopqrstuvwxyz0123456789-'))
return false;
else
return true;
});
require_once($sourcedir . '/Class-Punycode.php');
$Punycode = new Punycode();
$tlds = array_map(function($input) use ($Punycode)
{
return $Punycode->decode($input);
}, $tlds);
}
else
{
$tlds = array('com', 'net', 'org', 'edu', 'gov', 'mil', 'aero', 'asia', 'biz',
'cat', 'coop', 'info', 'int', 'jobs', 'mobi', 'museum', 'name', 'post',
'pro', 'tel', 'travel', 'xxx', 'ac', 'ad', 'ae', 'af', 'ag', 'ai', 'al',
'am', 'ao', 'aq', 'ar', 'as', 'at', 'au', 'aw', 'ax', 'az', 'ba', 'bb', 'bd',
'be', 'bf', 'bg', 'bh', 'bi', 'bj', 'bm', 'bn', 'bo', 'br', 'bs', 'bt', 'bv',
'bw', 'by', 'bz', 'ca', 'cc', 'cd', 'cf', 'cg', 'ch', 'ci', 'ck', 'cl', 'cm',
'cn', 'co', 'cr', 'cu', 'cv', 'cx', 'cy', 'cz', 'de', 'dj', 'dk', 'dm', 'do',
'dz', 'ec', 'ee', 'eg', 'er', 'es', 'et', 'eu', 'fi', 'fj', 'fk', 'fm', 'fo',
'fr', 'ga', 'gb', 'gd', 'ge', 'gf', 'gg', 'gh', 'gi', 'gl', 'gm', 'gn', 'gp',
'gq', 'gr', 'gs', 'gt', 'gu', 'gw', 'gy', 'hk', 'hm', 'hn', 'hr', 'ht', 'hu',
'id', 'ie', 'il', 'im', 'in', 'io', 'iq', 'ir', 'is', 'it', 'je', 'jm', 'jo',
'jp', 'ke', 'kg', 'kh', 'ki', 'km', 'kn', 'kp', 'kr', 'kw', 'ky', 'kz', 'la',
'lb', 'lc', 'li', 'lk', 'lr', 'ls', 'lt', 'lu', 'lv', 'ly', 'ma', 'mc', 'md',
'me', 'mg', 'mh', 'mk', 'ml', 'mm', 'mn', 'mo', 'mp', 'mq', 'mr', 'ms', 'mt',
'mu', 'mv', 'mw', 'mx', 'my', 'mz', 'na', 'nc', 'ne', 'nf', 'ng', 'ni', 'nl',
'no', 'np', 'nr', 'nu', 'nz', 'om', 'pa', 'pe', 'pf', 'pg', 'ph', 'pk', 'pl',
'pm', 'pn', 'pr', 'ps', 'pt', 'pw', 'py', 'qa', 're', 'ro', 'rs', 'ru', 'rw',
'sa', 'sb', 'sc', 'sd', 'se', 'sg', 'sh', 'si', 'sj', 'sk', 'sl', 'sm', 'sn',
'so', 'sr', 'ss', 'st', 'su', 'sv', 'sx', 'sy', 'sz', 'tc', 'td', 'tf', 'tg',
'th', 'tj', 'tk', 'tl', 'tm', 'tn', 'to', 'tr', 'tt', 'tv', 'tw', 'tz', 'ua',
'ug', 'uk', 'us', 'uy', 'uz', 'va', 'vc', 've', 'vg', 'vi', 'vn', 'vu', 'wf',
'ws', 'ye', 'yt', 'za', 'zm', 'zw',
);
if (empty($postapocalypticNightmare))
{
$smcFunc['db_insert']('insert', '{db_prefix}background_tasks',
array('task_file' => 'string-255', 'task_class' => 'string-255', 'task_data' => 'string', 'claimed_time' => 'int'),
array('$sourcedir/tasks/UpdateTldRegex.php', 'Update_TLD_Regex', '', 0), array()
);
}
}
$tlds = array_merge($tlds, array('local', 'onion', 'test'));
$tld_regex = build_regex($tlds);
updateSettings(array('tld_regex' => $tld_regex));
$done = true;
}
function build_regex($strings, $delim = null, $returnArray = false)
{
global $smcFunc;
static $regexes = array();
if (!is_array($strings))
return preg_quote(@strval($strings), $delim);
$regex_key = md5(json_encode(array($strings, $delim, $returnArray)));
if (isset($regexes[$regex_key]))
return $regexes[$regex_key];
if (function_exists('mb_internal_encoding') && function_exists('mb_detect_encoding') && function_exists('mb_strlen') && function_exists('mb_substr'))
{
if (($string_encoding = mb_detect_encoding(implode(' ', $strings))) !== false)
{
$current_encoding = mb_internal_encoding();
mb_internal_encoding($string_encoding);
}
$strlen = 'mb_strlen';
$substr = 'mb_substr';
}
else
{
$strlen = $smcFunc['strlen'];
$substr = $smcFunc['substr'];
}
$add_string_to_index = function($string, $index) use (&$strlen, &$substr, &$add_string_to_index)
{
static $depth = 0;
$depth++;
$first = (string) @$substr($string, 0, 1);
if ($first === '')
{
if (is_array($string) && $depth < 20)
{
foreach ($string as $str)
$index = $add_string_to_index($str, $index);
}
$depth--;
return $index;
}
if (empty($index[$first]))
$index[$first] = array();
if ($strlen($string) > 1)
{
if ($depth > 99)
$index[$first][$substr($string, 1)] = '';
else
$index[$first] = $add_string_to_index($substr($string, 1), $index[$first]);
}
else
$index[$first][''] = '';
$depth--;
return $index;
};
$index_to_regex = function(&$index, $delim) use (&$strlen, &$index_to_regex)
{
static $depth = 0;
$depth++;
$max_length = 30000;
$regex = array();
$length = 0;
foreach ($index as $key => $value)
{
$key_regex = preg_quote($key, $delim);
$new_key = $key;
if (empty($value))
$sub_regex = '';
else
{
$sub_regex = $index_to_regex($value, $delim);
if (count(array_keys($value)) == 1)
{
$new_key_array = explode('(?' . '>', $sub_regex);
$new_key .= $new_key_array[0];
}
else
$sub_regex = '(?' . '>' . $sub_regex . ')';
}
if ($depth > 1)
$regex[$new_key] = $key_regex . $sub_regex;
else
{
if (($length += strlen($key_regex) + 1) < $max_length || empty($regex))
{
$regex[$new_key] = $key_regex . $sub_regex;
unset($index[$key]);
}
else
break;
}
}
uksort($regex, function($k1, $k2) use (&$strlen)
{
$l1 = $strlen($k1);
$l2 = $strlen($k2);
if ($l1 == $l2)
return strcmp($k1, $k2) > 0 ? 1 : -1;
else
return $l1 > $l2 ? -1 : 1;
});
$depth--;
return implode('|', $regex);
};
$index = array();
$regex = '';
foreach ($strings as $string)
$index = $add_string_to_index($string, $index);
if ($returnArray === true)
{
$regex = array();
while (!empty($index))
$regex[] = '(?' . '>' . $index_to_regex($index, $delim) . ')';
}
else
$regex = '(?' . '>' . $index_to_regex($index, $delim) . ')';
if (!empty($current_encoding))
mb_internal_encoding($current_encoding);
$regexes[$regex_key] = $regex;
return $regex;
}
function ssl_cert_found($url)
{
if (!extension_loaded('openssl'))
return true;
$parsedurl = parse_url($url);
$url = 'ssl://' . $parsedurl['host'] . ':443';
if (version_compare(PHP_VERSION, '5.6.0', '<'))
$ssloptions = array("capture_peer_cert" => true);
else
$ssloptions = array("capture_peer_cert" => true, "verify_peer" => true, "allow_self_signed" => true);
$result = false;
$context = stream_context_create(array("ssl" => $ssloptions));
$stream = @stream_socket_client($url, $errno, $errstr, 30, STREAM_CLIENT_CONNECT, $context);
if ($stream !== false)
{
$params = stream_context_get_params($stream);
$result = isset($params["options"]["ssl"]["peer_certificate"]) ? true : false;
}
return $result;
}
function https_redirect_active($url)
{
$url = str_ireplace('https://', 'http://', $url) . '/';
$headers = @get_headers($url);
if ($headers === false)
return false;
if (strstr($headers[0], '301') === false && strstr($headers[0], '302') === false && strstr($headers[0], '307') === false)
return false;
$result = false;
foreach ($headers as $header)
{
if (stristr($header, 'Location: https://') !== false)
{
$result = true;
break;
}
}
return $result;
}
function build_query_board($userid)
{
global $user_info, $modSettings, $smcFunc, $db_prefix;
$query_part = array();
if (isset($user_info['id']) && $user_info['id'] == $userid && SMF != 'BACKGROUND')
{
$groups = $user_info['groups'];
$can_see_all_boards = $user_info['is_admin'] || $user_info['can_manage_boards'];
$ignoreboards = !empty($user_info['ignoreboards']) ? $user_info['ignoreboards'] : null;
}
else
{
$request = $smcFunc['db_query']('', '
SELECT mem.ignore_boards, mem.id_group, mem.additional_groups, mem.id_post_group
FROM {db_prefix}members AS mem
WHERE mem.id_member = {int:id_member}
LIMIT 1',
array(
'id_member' => $userid,
)
);
$row = $smcFunc['db_fetch_assoc']($request);
if (empty($row['additional_groups']))
$groups = array($row['id_group'], $row['id_post_group']);
else
$groups = array_merge(
array($row['id_group'], $row['id_post_group']),
explode(',', $row['additional_groups'])
);
foreach ($groups as $k => $v)
$groups[$k] = (int) $v;
$can_see_all_boards = in_array(1, $groups) || (!empty($modSettings['board_manager_groups']) && count(array_intersect($groups, explode(',', $modSettings['board_manager_groups']))) > 0);
$ignoreboards = !empty($row['ignore_boards']) && !empty($modSettings['allow_ignore_boards']) ? explode(',', $row['ignore_boards']) : array();
}
if ($can_see_all_boards)
$query_part['query_see_board'] = '1=1';
else
{
$query_part['query_see_board'] = '
EXISTS (
SELECT bpv.id_board
FROM ' . $db_prefix . 'board_permissions_view AS bpv
WHERE bpv.id_group IN ('. implode(',', $groups) .')
AND bpv.deny = 0
AND bpv.id_board = b.id_board
)';
if (!empty($modSettings['deny_boards_access']))
$query_part['query_see_board'] .= '
AND NOT EXISTS (
SELECT bpv.id_board
FROM ' . $db_prefix . 'board_permissions_view AS bpv
WHERE bpv.id_group IN ( '. implode(',', $groups) .')
AND bpv.deny = 1
AND bpv.id_board = b.id_board
)';
}
$query_part['query_see_message_board'] = str_replace('b.', 'm.', $query_part['query_see_board']);
$query_part['query_see_topic_board'] = str_replace('b.', 't.', $query_part['query_see_board']);
if (empty($ignoreboards))
{
$query_part['query_wanna_see_board'] = $query_part['query_see_board'];
$query_part['query_wanna_see_message_board'] = $query_part['query_see_message_board'];
$query_part['query_wanna_see_topic_board'] = $query_part['query_see_topic_board'];
}
else
{
$query_part['query_wanna_see_board'] = '(' . $query_part['query_see_board'] . ' AND b.id_board NOT IN (' . implode(',', $ignoreboards) . '))';
$query_part['query_wanna_see_message_board'] = '(' . $query_part['query_see_message_board'] . ' AND m.id_board NOT IN (' . implode(',', $ignoreboards) . '))';
$query_part['query_wanna_see_topic_board'] = '(' . $query_part['query_see_topic_board'] . ' AND t.id_board NOT IN (' . implode(',', $ignoreboards) . '))';
}
return $query_part;
}
function httpsOn()
{
$secure = false;
if (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on')
$secure = true;
elseif (!empty($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] == 'https' || !empty($_SERVER['HTTP_X_FORWARDED_SSL']) && $_SERVER['HTTP_X_FORWARDED_SSL'] == 'on')
$secure = true;
return $secure;
}
function validate_iri($iri, $flags = null)
{
$url = iri_to_url($iri);
if (filter_var($url, FILTER_VALIDATE_URL, $flags) !== false)
return $iri;
else
return false;
}
function sanitize_iri($iri)
{
$iri = preg_replace_callback('~[^\x00-\x7F\pZ\pC]~u', function($matches)
{
return rawurlencode($matches[0]);
}, $iri);
$iri = filter_var($iri, FILTER_SANITIZE_URL);
$iri = rawurldecode($iri);
return $iri;
}
function iri_to_url($iri)
{
global $sourcedir;
$host = parse_url((strpos($iri, '://') === false ? 'http://' : '') . ltrim($iri, ':/'), PHP_URL_HOST);
if (empty($host))
return $iri;
require_once($sourcedir . '/Class-Punycode.php');
$Punycode = new Punycode();
$encoded_host = $Punycode->encode($host);
$pos = strpos($iri, $host);
$iri = substr_replace($iri, $encoded_host, $pos, strlen($host));
$unescaped = array(
'%21' => '!', '%23' => '#', '%24' => '$', '%26' => '&',
'%27' => "'", '%28' => '(', '%29' => ')', '%2A' => '*',
'%2B' => '+', '%2C' => ',', '%2F' => '/', '%3A' => ':',
'%3B' => ';', '%3D' => '=', '%3F' => '?', '%40' => '@',
'%25' => '%',
);
$iri = strtr(rawurlencode($iri), $unescaped);
return $iri;
}
function url_to_iri($url)
{
global $sourcedir;
$host = parse_url((strpos($url, '://') === false ? 'http://' : '') . ltrim($url, ':/'), PHP_URL_HOST);
if (empty($host))
return $url;
require_once($sourcedir . '/Class-Punycode.php');
$Punycode = new Punycode();
$decoded_host = $Punycode->decode($host);
$pos = strpos($url, $host);
$url = substr_replace($url, $decoded_host, $pos, strlen($host));
$url = rawurldecode($url);
return $url;
}
function check_cron()
{
global $modSettings, $smcFunc, $txt;
if (!empty($modSettings['cron_is_real_cron']) && time() - @intval($modSettings['cron_last_checked']) > 84600)
{
$request = $smcFunc['db_query']('', '
SELECT COUNT(*)
FROM {db_prefix}scheduled_tasks
WHERE disabled = {int:not_disabled}
AND next_time < {int:yesterday}',
array(
'not_disabled' => 0,
'yesterday' => time() - 84600,
)
);
list($overdue) = $smcFunc['db_fetch_row']($request);
$smcFunc['db_free_result']($request);
if (!empty($overdue))
{
loadLanguage('ManageScheduledTasks');
log_error($txt['cron_not_working']);
updateSettings(array('cron_is_real_cron' => 0));
}
else
updateSettings(array('cron_last_checked' => time()));
}
}
function send_http_status($code, $status = '')
{
$statuses = array(
206 => 'Partial Content',
304 => 'Not Modified',
400 => 'Bad Request',
403 => 'Forbidden',
404 => 'Not Found',
410 => 'Gone',
500 => 'Internal Server Error',
503 => 'Service Unavailable',
);
$protocol = preg_match('~^\s*(HTTP/[12]\.\d)\s*$~i', $_SERVER['SERVER_PROTOCOL'], $matches) ? $matches[1] : 'HTTP/1.0';
if (!isset($statuses[$code]) && empty($status))
header($protocol . ' 500 Internal Server Error');
else
header($protocol . ' ' . $code . ' ' . (!empty($status) ? $status : $statuses[$code]));
}
function sentence_list($list)
{
global $txt;
if (empty($txt['sentence_list_format']['n']))
$txt['sentence_list_format']['n'] = '{series}';
if (!isset($txt['sentence_list_separator']))
$txt['sentence_list_separator'] = ', ';
if (!isset($txt['sentence_list_separator_alt']))
$txt['sentence_list_separator_alt'] = '; ';
if (isset($txt['sentence_list_format'][count($list)]))
$format = $txt['sentence_list_format'][count($list)];
else
$format = $txt['sentence_list_format']['n'];
$separator = $txt['sentence_list_separator'];
foreach ($list as $item)
{
if (strpos($item, $separator) !== false)
{
$separator = $txt['sentence_list_separator_alt'];
$format = strtr($format, trim($txt['sentence_list_separator']), trim($separator));
break;
}
}
$replacements = array();
$i = 0;
while (empty($done))
{
if (strpos($format, '{'. --$i . '}') !== false)
$replacements['{'. $i . '}'] = array_pop($list);
else
$done = true;
}
unset($done);
$i = 0;
while (empty($done))
{
if (strpos($format, '{'. ++$i . '}') !== false)
$replacements['{'. $i . '}'] = array_shift($list);
else
$done = true;
}
unset($done);
$replacements['{series}'] = implode($separator, $list);
return strtr($format, $replacements);
}
function truncate_array($array, $max_length = 1900, $deep = 3)
{
$array = (array) $array;
$curr_length = array_length($array, $deep);
if ($curr_length <= $max_length)
return $array;
else
{
$param_max = floor($max_length / count($array));
$current_deep = $deep - 1;
foreach ($array as $key => &$value)
{
if (is_array($value))
if ($current_deep > 0)
$value = truncate_array($value, $current_deep);
else
$value = substr($value, 0, $param_max - strlen($key) - 5);
}
return $array;
}
}
function array_length($array, $deep = 3)
{
$array = (array) $array;
$length = 0;
$deep_count = $deep - 1;
foreach ($array as $value)
{
if (is_array($value))
{
if ($deep_count <= 0)
continue;
$length += array_length($value, $deep_count);
}
else
$length += strlen($value);
}
return $length;
}
?>