mirror of
https://github.com/php/php-src.git
synced 2025-08-15 13:38:49 +02:00
243 lines
6.6 KiB
C++
243 lines
6.6 KiB
C++
/*
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| https://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Authors: Gustavo Lopes <cataphract@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "../intl_cppshims.h"
|
|
|
|
#include <unicode/timezone.h>
|
|
#include <unicode/calendar.h>
|
|
#include <unicode/datefmt.h>
|
|
|
|
extern "C" {
|
|
#include "../php_intl.h"
|
|
#include "dateformat_class.h"
|
|
#define USE_TIMEZONE_POINTER 1
|
|
#include "../timezone/timezone_class.h"
|
|
#define USE_CALENDAR_POINTER 1
|
|
#include "../calendar/calendar_class.h"
|
|
}
|
|
|
|
#include "../intl_convertcpp.h"
|
|
#include "dateformat_helpers.h"
|
|
|
|
static inline DateFormat *fetch_datefmt(IntlDateFormatter_object *dfo) {
|
|
return (DateFormat *)dfo->datef_data.udatf;
|
|
}
|
|
|
|
/* {{{ Get formatter timezone_id. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_get_timezone_id)
|
|
{
|
|
zend_string *u8str;
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "O",
|
|
&object, IntlDateFormatter_ce_ptr ) == FAILURE) {
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
UnicodeString res = UnicodeString();
|
|
fetch_datefmt(dfo)->getTimeZone().getID(res);
|
|
u8str = intl_charFromString(res, &INTL_DATA_ERROR_CODE(dfo));
|
|
INTL_METHOD_CHECK_STATUS(dfo, "Could not convert time zone id to UTF-8");
|
|
|
|
RETVAL_STR(u8str);
|
|
}
|
|
|
|
/* {{{ Get formatter timezone. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_get_timezone)
|
|
{
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "O",
|
|
&object, IntlDateFormatter_ce_ptr ) == FAILURE) {
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
const TimeZone& tz = fetch_datefmt(dfo)->getTimeZone();
|
|
TimeZone *tz_clone = tz.clone();
|
|
if (UNEXPECTED(tz_clone == NULL)) {
|
|
intl_errors_set(INTL_DATA_ERROR_P(dfo), U_MEMORY_ALLOCATION_ERROR,
|
|
"Out of memory when cloning time zone");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
timezone_object_construct(tz_clone, return_value, 1);
|
|
}
|
|
|
|
/* {{{ Set formatter's timezone. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_set_timezone)
|
|
{
|
|
zend_object *timezone_object = nullptr;
|
|
zend_string *timezone_string = nullptr;
|
|
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(2, 2)
|
|
Z_PARAM_OBJECT_OF_CLASS(object, IntlDateFormatter_ce_ptr)
|
|
Z_PARAM_OBJ_OR_STR_OR_NULL(timezone_object, timezone_string)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
TimeZone *timezone = timezone_process_timezone_argument(
|
|
timezone_object, timezone_string, INTL_DATA_ERROR_P(dfo));
|
|
if (timezone == nullptr) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
fetch_datefmt(dfo)->adoptTimeZone(timezone);
|
|
|
|
RETURN_TRUE;
|
|
}
|
|
|
|
U_CFUNC PHP_METHOD(IntlDateFormatter, setTimeZone)
|
|
{
|
|
zend_object *timezone_object = nullptr;
|
|
zend_string *timezone_string = nullptr;
|
|
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_OBJ_OR_STR_OR_NULL(timezone_object, timezone_string)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
object = ZEND_THIS;
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
TimeZone *timezone = timezone_process_timezone_argument(
|
|
timezone_object, timezone_string, INTL_DATA_ERROR_P(dfo));
|
|
if (timezone == nullptr) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
fetch_datefmt(dfo)->adoptTimeZone(timezone);
|
|
|
|
RETURN_TRUE;
|
|
}
|
|
|
|
/* {{{ Get formatter calendar type. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_get_calendar)
|
|
{
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "O",
|
|
&object, IntlDateFormatter_ce_ptr ) == FAILURE) {
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
if (dfo->calendar == -1) {
|
|
/* an IntlCalendar was provided to the constructor */
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
RETURN_LONG(dfo->calendar);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ Get formatter calendar. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_get_calendar_object)
|
|
{
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "O",
|
|
&object, IntlDateFormatter_ce_ptr ) == FAILURE) {
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
const Calendar *cal = fetch_datefmt(dfo)->getCalendar();
|
|
if (cal == NULL) {
|
|
RETURN_NULL();
|
|
}
|
|
|
|
Calendar *cal_clone = cal->clone();
|
|
if (UNEXPECTED(cal_clone == NULL)) {
|
|
intl_errors_set(INTL_DATA_ERROR_P(dfo), U_MEMORY_ALLOCATION_ERROR,
|
|
"Out of memory when cloning calendar");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
calendar_object_create(return_value, cal_clone);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ Set formatter's calendar. */
|
|
U_CFUNC PHP_FUNCTION(datefmt_set_calendar)
|
|
{
|
|
zend_object *calendar_obj = NULL;
|
|
zend_long calendar_long = 0;
|
|
bool calendar_is_null;
|
|
DATE_FORMAT_METHOD_INIT_VARS;
|
|
|
|
object = getThis();
|
|
|
|
if (object) {
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_OBJ_OF_CLASS_OR_LONG_OR_NULL(calendar_obj, Calendar_ce_ptr, calendar_long, calendar_is_null)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
} else {
|
|
ZEND_PARSE_PARAMETERS_START(2, 2)
|
|
Z_PARAM_OBJECT_OF_CLASS(object, IntlDateFormatter_ce_ptr)
|
|
Z_PARAM_OBJ_OF_CLASS_OR_LONG_OR_NULL(calendar_obj, Calendar_ce_ptr, calendar_long, calendar_is_null)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
}
|
|
|
|
DATE_FORMAT_METHOD_FETCH_OBJECT;
|
|
|
|
Calendar *cal;
|
|
zend_long cal_type;
|
|
bool cal_owned;
|
|
Locale locale = Locale::createFromName(dfo->requested_locale);
|
|
// getting the actual locale from the DateFormat is not enough
|
|
// because we would have lost modifiers such as @calendar. We
|
|
// must store the requested locale on object creation
|
|
|
|
if (datefmt_process_calendar_arg(calendar_obj, calendar_long, calendar_is_null, locale,
|
|
INTL_DATA_ERROR_P(dfo), cal, cal_type, cal_owned) == FAILURE
|
|
) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (cal_owned) {
|
|
/* a non IntlCalendar was specified, we want to keep the timezone */
|
|
TimeZone *old_timezone = fetch_datefmt(dfo)->getTimeZone().clone();
|
|
if (UNEXPECTED(old_timezone == NULL)) {
|
|
intl_errors_set(INTL_DATA_ERROR_P(dfo), U_MEMORY_ALLOCATION_ERROR,
|
|
"Out of memory when cloning calendar");
|
|
delete cal;
|
|
RETURN_FALSE;
|
|
}
|
|
cal->adoptTimeZone(old_timezone);
|
|
} else {
|
|
cal = cal->clone();
|
|
if (UNEXPECTED(cal == NULL)) {
|
|
intl_errors_set(INTL_DATA_ERROR_P(dfo), U_MEMORY_ALLOCATION_ERROR,
|
|
"Out of memory when cloning calendar");
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
fetch_datefmt(dfo)->adoptCalendar(cal);
|
|
|
|
dfo->calendar = cal_type;
|
|
|
|
RETURN_TRUE;
|
|
}
|
|
/* }}} */
|