formatter_attr.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | http://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Stanislav Malyshev <stas@zend.com> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include "php_intl.h"
  20. #include "formatter_class.h"
  21. #include "formatter_attr.h"
  22. #include "intl_convert.h"
  23. #include <unicode/ustring.h>
  24. /* {{{ proto mixed NumberFormatter::getAttribute( int $attr )
  25. * Get formatter attribute value. }}} */
  26. /* {{{ proto mixed numfmt_get_attribute( NumberFormatter $nf, int $attr )
  27. * Get formatter attribute value.
  28. */
  29. PHP_FUNCTION( numfmt_get_attribute )
  30. {
  31. long attribute, value;
  32. FORMATTER_METHOD_INIT_VARS;
  33. /* Parse parameters. */
  34. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ol",
  35. &object, NumberFormatter_ce_ptr, &attribute ) == FAILURE )
  36. {
  37. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  38. "numfmt_get_attribute: unable to parse input params", 0 TSRMLS_CC );
  39. RETURN_FALSE;
  40. }
  41. /* Fetch the object. */
  42. FORMATTER_METHOD_FETCH_OBJECT;
  43. switch(attribute) {
  44. case UNUM_PARSE_INT_ONLY:
  45. case UNUM_GROUPING_USED:
  46. case UNUM_DECIMAL_ALWAYS_SHOWN:
  47. case UNUM_MAX_INTEGER_DIGITS:
  48. case UNUM_MIN_INTEGER_DIGITS:
  49. case UNUM_INTEGER_DIGITS:
  50. case UNUM_MAX_FRACTION_DIGITS:
  51. case UNUM_MIN_FRACTION_DIGITS:
  52. case UNUM_FRACTION_DIGITS:
  53. case UNUM_MULTIPLIER:
  54. case UNUM_GROUPING_SIZE:
  55. case UNUM_ROUNDING_MODE:
  56. case UNUM_FORMAT_WIDTH:
  57. case UNUM_PADDING_POSITION:
  58. case UNUM_SECONDARY_GROUPING_SIZE:
  59. case UNUM_SIGNIFICANT_DIGITS_USED:
  60. case UNUM_MIN_SIGNIFICANT_DIGITS:
  61. case UNUM_MAX_SIGNIFICANT_DIGITS:
  62. case UNUM_LENIENT_PARSE:
  63. value = unum_getAttribute(FORMATTER_OBJECT(nfo), attribute);
  64. if(value == -1) {
  65. INTL_DATA_ERROR_CODE(nfo) = U_UNSUPPORTED_ERROR;
  66. } else {
  67. RETVAL_LONG(value);
  68. }
  69. break;
  70. case UNUM_ROUNDING_INCREMENT:
  71. {
  72. double value_double = unum_getDoubleAttribute(FORMATTER_OBJECT(nfo), attribute);
  73. if(value_double == -1) {
  74. INTL_DATA_ERROR_CODE(nfo) = U_UNSUPPORTED_ERROR;
  75. } else {
  76. RETVAL_DOUBLE(value_double);
  77. }
  78. }
  79. break;
  80. default:
  81. INTL_DATA_ERROR_CODE(nfo) = U_UNSUPPORTED_ERROR;
  82. break;
  83. }
  84. INTL_METHOD_CHECK_STATUS( nfo, "Error getting attribute value" );
  85. }
  86. /* }}} */
  87. /* {{{ proto string NumberFormatter::getTextAttribute( int $attr )
  88. * Get formatter attribute value. }}} */
  89. /* {{{ proto string numfmt_get_text_attribute( NumberFormatter $nf, int $attr )
  90. * Get formatter attribute value.
  91. */
  92. PHP_FUNCTION( numfmt_get_text_attribute )
  93. {
  94. long attribute;
  95. UChar value_buf[64];
  96. int value_buf_size = USIZE( value_buf );
  97. UChar* value = value_buf;
  98. int length = 0;
  99. FORMATTER_METHOD_INIT_VARS;
  100. /* Parse parameters. */
  101. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ol",
  102. &object, NumberFormatter_ce_ptr, &attribute ) == FAILURE )
  103. {
  104. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  105. "numfmt_get_text_attribute: unable to parse input params", 0 TSRMLS_CC );
  106. RETURN_FALSE;
  107. }
  108. /* Fetch the object. */
  109. FORMATTER_METHOD_FETCH_OBJECT;
  110. length = unum_getTextAttribute( FORMATTER_OBJECT(nfo), attribute, value, value_buf_size, &INTL_DATA_ERROR_CODE(nfo) );
  111. if(INTL_DATA_ERROR_CODE(nfo) == U_BUFFER_OVERFLOW_ERROR && length >= value_buf_size) {
  112. ++length; /* to avoid U_STRING_NOT_TERMINATED_WARNING */
  113. INTL_DATA_ERROR_CODE(nfo) = U_ZERO_ERROR;
  114. value = eumalloc(length);
  115. length = unum_getTextAttribute( FORMATTER_OBJECT(nfo), attribute, value, length, &INTL_DATA_ERROR_CODE(nfo) );
  116. if(U_FAILURE(INTL_DATA_ERROR_CODE(nfo))) {
  117. efree(value);
  118. value = value_buf;
  119. }
  120. }
  121. INTL_METHOD_CHECK_STATUS( nfo, "Error getting attribute value" );
  122. INTL_METHOD_RETVAL_UTF8( nfo, value, length, ( value != value_buf ) );
  123. }
  124. /* }}} */
  125. /* {{{ proto bool NumberFormatter::setAttribute( int $attr, mixed $value )
  126. * Get formatter attribute value. }}} */
  127. /* {{{ proto bool numfmt_set_attribute( NumberFormatter $nf, int $attr, mixed $value )
  128. * Get formatter attribute value.
  129. */
  130. PHP_FUNCTION( numfmt_set_attribute )
  131. {
  132. long attribute;
  133. zval **value;
  134. FORMATTER_METHOD_INIT_VARS;
  135. /* Parse parameters. */
  136. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OlZ",
  137. &object, NumberFormatter_ce_ptr, &attribute, &value ) == FAILURE)
  138. {
  139. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  140. "numfmt_set_attribute: unable to parse input params", 0 TSRMLS_CC );
  141. RETURN_FALSE;
  142. }
  143. /* Fetch the object. */
  144. FORMATTER_METHOD_FETCH_OBJECT;
  145. switch(attribute) {
  146. case UNUM_PARSE_INT_ONLY:
  147. case UNUM_GROUPING_USED:
  148. case UNUM_DECIMAL_ALWAYS_SHOWN:
  149. case UNUM_MAX_INTEGER_DIGITS:
  150. case UNUM_MIN_INTEGER_DIGITS:
  151. case UNUM_INTEGER_DIGITS:
  152. case UNUM_MAX_FRACTION_DIGITS:
  153. case UNUM_MIN_FRACTION_DIGITS:
  154. case UNUM_FRACTION_DIGITS:
  155. case UNUM_MULTIPLIER:
  156. case UNUM_GROUPING_SIZE:
  157. case UNUM_ROUNDING_MODE:
  158. case UNUM_FORMAT_WIDTH:
  159. case UNUM_PADDING_POSITION:
  160. case UNUM_SECONDARY_GROUPING_SIZE:
  161. case UNUM_SIGNIFICANT_DIGITS_USED:
  162. case UNUM_MIN_SIGNIFICANT_DIGITS:
  163. case UNUM_MAX_SIGNIFICANT_DIGITS:
  164. case UNUM_LENIENT_PARSE:
  165. convert_to_long_ex(value);
  166. unum_setAttribute(FORMATTER_OBJECT(nfo), attribute, Z_LVAL_PP(value));
  167. break;
  168. case UNUM_ROUNDING_INCREMENT:
  169. convert_to_double_ex(value);
  170. unum_setDoubleAttribute(FORMATTER_OBJECT(nfo), attribute, Z_DVAL_PP(value));
  171. break;
  172. default:
  173. INTL_DATA_ERROR_CODE(nfo) = U_UNSUPPORTED_ERROR;
  174. break;
  175. }
  176. INTL_METHOD_CHECK_STATUS( nfo, "Error setting attribute value" );
  177. RETURN_TRUE;
  178. }
  179. /* }}} */
  180. /* {{{ proto bool NumberFormatter::setTextAttribute( int $attr, string $value )
  181. * Get formatter attribute value. }}} */
  182. /* {{{ proto bool numfmt_set_text_attribute( NumberFormatter $nf, int $attr, string $value )
  183. * Get formatter attribute value.
  184. */
  185. PHP_FUNCTION( numfmt_set_text_attribute )
  186. {
  187. int slength = 0;
  188. UChar *svalue = NULL;
  189. long attribute;
  190. char *value;
  191. int len;
  192. FORMATTER_METHOD_INIT_VARS;
  193. /* Parse parameters. */
  194. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ols",
  195. &object, NumberFormatter_ce_ptr, &attribute, &value, &len ) == FAILURE)
  196. {
  197. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  198. "numfmt_set_text_attribute: unable to parse input params", 0 TSRMLS_CC );
  199. RETURN_FALSE;
  200. }
  201. /* Fetch the object. */
  202. FORMATTER_METHOD_FETCH_OBJECT;
  203. /* Convert given attribute value to UTF-16. */
  204. intl_convert_utf8_to_utf16(&svalue, &slength, value, len, &INTL_DATA_ERROR_CODE(nfo));
  205. INTL_METHOD_CHECK_STATUS( nfo, "Error converting attribute value to UTF-16" );
  206. /* Actually set new attribute value. */
  207. unum_setTextAttribute(FORMATTER_OBJECT(nfo), attribute, svalue, slength, &INTL_DATA_ERROR_CODE(nfo));
  208. if (svalue) {
  209. efree(svalue);
  210. }
  211. INTL_METHOD_CHECK_STATUS( nfo, "Error setting text attribute" );
  212. RETURN_TRUE;
  213. }
  214. /* }}} */
  215. /* {{{ proto string NumberFormatter::getSymbol( int $attr )
  216. * Get formatter symbol value. }}} */
  217. /* {{{ proto string numfmt_get_symbol( NumberFormatter $nf, int $attr )
  218. * Get formatter symbol value.
  219. */
  220. PHP_FUNCTION( numfmt_get_symbol )
  221. {
  222. long symbol;
  223. UChar value_buf[4];
  224. UChar *value = value_buf;
  225. int length = USIZE(value_buf);
  226. FORMATTER_METHOD_INIT_VARS;
  227. /* Parse parameters. */
  228. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ol",
  229. &object, NumberFormatter_ce_ptr, &symbol ) == FAILURE )
  230. {
  231. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  232. "numfmt_get_symbol: unable to parse input params", 0 TSRMLS_CC );
  233. RETURN_FALSE;
  234. }
  235. if(symbol >= UNUM_FORMAT_SYMBOL_COUNT || symbol < 0) {
  236. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR, "numfmt_get_symbol: invalid symbol value", 0 TSRMLS_CC );
  237. RETURN_FALSE;
  238. }
  239. /* Fetch the object. */
  240. FORMATTER_METHOD_FETCH_OBJECT;
  241. length = unum_getSymbol(FORMATTER_OBJECT(nfo), symbol, value_buf, length, &INTL_DATA_ERROR_CODE(nfo));
  242. if(INTL_DATA_ERROR_CODE(nfo) == U_BUFFER_OVERFLOW_ERROR && length >= USIZE( value_buf )) {
  243. ++length; /* to avoid U_STRING_NOT_TERMINATED_WARNING */
  244. INTL_DATA_ERROR_CODE(nfo) = U_ZERO_ERROR;
  245. value = eumalloc(length);
  246. length = unum_getSymbol(FORMATTER_OBJECT(nfo), symbol, value, length, &INTL_DATA_ERROR_CODE(nfo));
  247. if(U_FAILURE(INTL_DATA_ERROR_CODE(nfo))) {
  248. efree(value);
  249. value = value_buf;
  250. }
  251. }
  252. INTL_METHOD_CHECK_STATUS( nfo, "Error getting symbol value" );
  253. INTL_METHOD_RETVAL_UTF8( nfo, value, length, ( value_buf != value ) );
  254. }
  255. /* }}} */
  256. /* {{{ proto bool NumberFormatter::setSymbol( int $attr, string $symbol )
  257. * Set formatter symbol value. }}} */
  258. /* {{{ proto bool numfmt_set_symbol( NumberFormatter $nf, int $attr, string $symbol )
  259. * Set formatter symbol value.
  260. */
  261. PHP_FUNCTION( numfmt_set_symbol )
  262. {
  263. long symbol;
  264. char* value = NULL;
  265. int value_len = 0;
  266. UChar* svalue = 0;
  267. int slength = 0;
  268. FORMATTER_METHOD_INIT_VARS;
  269. /* Parse parameters. */
  270. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ols",
  271. &object, NumberFormatter_ce_ptr, &symbol, &value, &value_len ) == FAILURE )
  272. {
  273. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  274. "numfmt_set_symbol: unable to parse input params", 0 TSRMLS_CC );
  275. RETURN_FALSE;
  276. }
  277. if (symbol >= UNUM_FORMAT_SYMBOL_COUNT || symbol < 0) {
  278. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR, "numfmt_set_symbol: invalid symbol value", 0 TSRMLS_CC );
  279. RETURN_FALSE;
  280. }
  281. /* Fetch the object. */
  282. FORMATTER_METHOD_FETCH_OBJECT;
  283. /* Convert given symbol to UTF-16. */
  284. intl_convert_utf8_to_utf16(&svalue, &slength, value, value_len, &INTL_DATA_ERROR_CODE(nfo));
  285. INTL_METHOD_CHECK_STATUS( nfo, "Error converting symbol value to UTF-16" );
  286. /* Actually set the symbol. */
  287. unum_setSymbol(FORMATTER_OBJECT(nfo), symbol, svalue, slength, &INTL_DATA_ERROR_CODE(nfo));
  288. if (svalue) {
  289. efree(svalue);
  290. }
  291. INTL_METHOD_CHECK_STATUS( nfo, "Error setting symbol value" );
  292. RETURN_TRUE;
  293. }
  294. /* }}} */
  295. /* {{{ proto string NumberFormatter::getPattern( )
  296. * Get formatter pattern. }}} */
  297. /* {{{ proto string numfmt_get_pattern( NumberFormatter $nf )
  298. * Get formatter pattern.
  299. */
  300. PHP_FUNCTION( numfmt_get_pattern )
  301. {
  302. UChar value_buf[64];
  303. int length = USIZE( value_buf );
  304. UChar* value = value_buf;
  305. FORMATTER_METHOD_INIT_VARS;
  306. /* Parse parameters. */
  307. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O",
  308. &object, NumberFormatter_ce_ptr ) == FAILURE )
  309. {
  310. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  311. "numfmt_get_pattern: unable to parse input params", 0 TSRMLS_CC );
  312. RETURN_FALSE;
  313. }
  314. /* Fetch the object. */
  315. FORMATTER_METHOD_FETCH_OBJECT;
  316. length = unum_toPattern(FORMATTER_OBJECT(nfo), 0, value, length, &INTL_DATA_ERROR_CODE(nfo));
  317. if(INTL_DATA_ERROR_CODE(nfo) == U_BUFFER_OVERFLOW_ERROR && length >= USIZE( value_buf )) {
  318. ++length; /* to avoid U_STRING_NOT_TERMINATED_WARNING */
  319. INTL_DATA_ERROR_CODE(nfo) = U_ZERO_ERROR;
  320. value = eumalloc(length);
  321. length = unum_toPattern( FORMATTER_OBJECT(nfo), 0, value, length, &INTL_DATA_ERROR_CODE(nfo) );
  322. if(U_FAILURE(INTL_DATA_ERROR_CODE(nfo))) {
  323. efree(value);
  324. value = value_buf;
  325. }
  326. }
  327. INTL_METHOD_CHECK_STATUS( nfo, "Error getting formatter pattern" );
  328. INTL_METHOD_RETVAL_UTF8( nfo, value, length, ( value != value_buf ) );
  329. }
  330. /* }}} */
  331. /* {{{ proto bool NumberFormatter::setPattern( string $pattern )
  332. * Set formatter pattern. }}} */
  333. /* {{{ proto bool numfmt_set_pattern( NumberFormatter $nf, string $pattern )
  334. * Set formatter pattern.
  335. */
  336. PHP_FUNCTION( numfmt_set_pattern )
  337. {
  338. char* value = NULL;
  339. int value_len = 0;
  340. int slength = 0;
  341. UChar* svalue = NULL;
  342. FORMATTER_METHOD_INIT_VARS;
  343. /* Parse parameters. */
  344. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os",
  345. &object, NumberFormatter_ce_ptr, &value, &value_len ) == FAILURE )
  346. {
  347. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  348. "numfmt_set_pattern: unable to parse input params", 0 TSRMLS_CC );
  349. RETURN_FALSE;
  350. }
  351. FORMATTER_METHOD_FETCH_OBJECT;
  352. /* Convert given pattern to UTF-16. */
  353. intl_convert_utf8_to_utf16(&svalue, &slength, value, value_len, &INTL_DATA_ERROR_CODE(nfo));
  354. INTL_METHOD_CHECK_STATUS( nfo, "Error converting pattern to UTF-16" );
  355. /* TODO: add parse error information */
  356. unum_applyPattern(FORMATTER_OBJECT(nfo), 0, svalue, slength, NULL, &INTL_DATA_ERROR_CODE(nfo));
  357. if (svalue) {
  358. efree(svalue);
  359. }
  360. INTL_METHOD_CHECK_STATUS( nfo, "Error setting pattern value" );
  361. RETURN_TRUE;
  362. }
  363. /* }}} */
  364. /* {{{ proto string NumberFormatter::getLocale([int type])
  365. * Get formatter locale. }}} */
  366. /* {{{ proto string numfmt_get_locale( NumberFormatter $nf[, int type] )
  367. * Get formatter locale.
  368. */
  369. PHP_FUNCTION( numfmt_get_locale )
  370. {
  371. long type = ULOC_ACTUAL_LOCALE;
  372. char* loc;
  373. FORMATTER_METHOD_INIT_VARS;
  374. /* Parse parameters. */
  375. if( zend_parse_method_parameters( ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O|l",
  376. &object, NumberFormatter_ce_ptr, &type ) == FAILURE )
  377. {
  378. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  379. "numfmt_get_locale: unable to parse input params", 0 TSRMLS_CC );
  380. RETURN_FALSE;
  381. }
  382. /* Fetch the object. */
  383. FORMATTER_METHOD_FETCH_OBJECT;
  384. loc = (char *)unum_getLocaleByType(FORMATTER_OBJECT(nfo), type, &INTL_DATA_ERROR_CODE(nfo));
  385. INTL_METHOD_CHECK_STATUS( nfo, "Error getting locale" );
  386. RETURN_STRING(loc, 1);
  387. }
  388. /* }}} */
  389. /*
  390. * Local variables:
  391. * tab-width: 4
  392. * c-basic-offset: 4
  393. * End:
  394. * vim600: noet sw=4 ts=4 fdm=marker
  395. * vim<600: noet sw=4 ts=4
  396. */