summaryrefslogtreecommitdiff
path: root/deps/node/deps/icu-small/source/i18n/numsys.cpp
diff options
context:
space:
mode:
authorFlorian Dold <florian.dold@gmail.com>2019-04-03 15:43:32 +0200
committerFlorian Dold <florian.dold@gmail.com>2019-04-03 15:45:57 +0200
commit71e285b94c7edaa43aa8115965cf5a36b8e0f80a (patch)
tree7d4aa9d0d5aff686b106cd5da72ba77960c4af43 /deps/node/deps/icu-small/source/i18n/numsys.cpp
parent7dadf9356b4f3f4137ce982ea5bb960283116e9a (diff)
downloadakono-71e285b94c7edaa43aa8115965cf5a36b8e0f80a.tar.gz
akono-71e285b94c7edaa43aa8115965cf5a36b8e0f80a.tar.bz2
akono-71e285b94c7edaa43aa8115965cf5a36b8e0f80a.zip
Node.js v11.13.0
Diffstat (limited to 'deps/node/deps/icu-small/source/i18n/numsys.cpp')
-rw-r--r--deps/node/deps/icu-small/source/i18n/numsys.cpp359
1 files changed, 359 insertions, 0 deletions
diff --git a/deps/node/deps/icu-small/source/i18n/numsys.cpp b/deps/node/deps/icu-small/source/i18n/numsys.cpp
new file mode 100644
index 00000000..514fe05e
--- /dev/null
+++ b/deps/node/deps/icu-small/source/i18n/numsys.cpp
@@ -0,0 +1,359 @@
+// © 2016 and later: Unicode, Inc. and others.
+// License & terms of use: http://www.unicode.org/copyright.html
+/*
+*******************************************************************************
+* Copyright (C) 2010-2015, International Business Machines Corporation and
+* others. All Rights Reserved.
+*******************************************************************************
+*
+*
+* File NUMSYS.CPP
+*
+* Modification History:*
+* Date Name Description
+*
+********************************************************************************
+*/
+
+#include "unicode/utypes.h"
+#include "unicode/localpointer.h"
+#include "unicode/uchar.h"
+#include "unicode/unistr.h"
+#include "unicode/ures.h"
+#include "unicode/ustring.h"
+#include "unicode/uloc.h"
+#include "unicode/schriter.h"
+#include "unicode/numsys.h"
+#include "cstring.h"
+#include "uassert.h"
+#include "uresimp.h"
+#include "numsys_impl.h"
+
+#if !UCONFIG_NO_FORMATTING
+
+U_NAMESPACE_BEGIN
+
+// Useful constants
+
+#define DEFAULT_DIGITS UNICODE_STRING_SIMPLE("0123456789");
+static const char gNumberingSystems[] = "numberingSystems";
+static const char gNumberElements[] = "NumberElements";
+static const char gDefault[] = "default";
+static const char gNative[] = "native";
+static const char gTraditional[] = "traditional";
+static const char gFinance[] = "finance";
+static const char gDesc[] = "desc";
+static const char gRadix[] = "radix";
+static const char gAlgorithmic[] = "algorithmic";
+static const char gLatn[] = "latn";
+
+
+UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NumberingSystem)
+UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NumsysNameEnumeration)
+
+ /**
+ * Default Constructor.
+ *
+ * @draft ICU 4.2
+ */
+
+NumberingSystem::NumberingSystem() {
+ radix = 10;
+ algorithmic = FALSE;
+ UnicodeString defaultDigits = DEFAULT_DIGITS;
+ desc.setTo(defaultDigits);
+ uprv_strcpy(name,gLatn);
+}
+
+ /**
+ * Copy constructor.
+ * @draft ICU 4.2
+ */
+
+NumberingSystem::NumberingSystem(const NumberingSystem& other)
+: UObject(other) {
+ *this=other;
+}
+
+NumberingSystem* U_EXPORT2
+NumberingSystem::createInstance(int32_t radix_in, UBool isAlgorithmic_in, const UnicodeString & desc_in, UErrorCode &status) {
+
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+
+ if ( radix_in < 2 ) {
+ status = U_ILLEGAL_ARGUMENT_ERROR;
+ return nullptr;
+ }
+
+ if ( !isAlgorithmic_in ) {
+ if ( desc_in.countChar32() != radix_in ) {
+ status = U_ILLEGAL_ARGUMENT_ERROR;
+ return nullptr;
+ }
+ }
+
+ LocalPointer<NumberingSystem> ns(new NumberingSystem(), status);
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+
+ ns->setRadix(radix_in);
+ ns->setDesc(desc_in);
+ ns->setAlgorithmic(isAlgorithmic_in);
+ ns->setName(nullptr);
+
+ return ns.orphan();
+}
+
+NumberingSystem* U_EXPORT2
+NumberingSystem::createInstance(const Locale & inLocale, UErrorCode& status) {
+
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+
+ UBool nsResolved = TRUE;
+ UBool usingFallback = FALSE;
+ char buffer[ULOC_KEYWORDS_CAPACITY];
+ int32_t count = inLocale.getKeywordValue("numbers", buffer, sizeof(buffer), status);
+ if (U_FAILURE(status) || status == U_STRING_NOT_TERMINATED_WARNING) {
+ // the "numbers" keyword exceeds ULOC_KEYWORDS_CAPACITY; ignore and use default.
+ count = 0;
+ status = U_ZERO_ERROR;
+ }
+ if ( count > 0 ) { // @numbers keyword was specified in the locale
+ U_ASSERT(count < ULOC_KEYWORDS_CAPACITY);
+ buffer[count] = '\0'; // Make sure it is null terminated.
+ if ( !uprv_strcmp(buffer,gDefault) || !uprv_strcmp(buffer,gNative) ||
+ !uprv_strcmp(buffer,gTraditional) || !uprv_strcmp(buffer,gFinance)) {
+ nsResolved = FALSE;
+ }
+ } else {
+ uprv_strcpy(buffer, gDefault);
+ nsResolved = FALSE;
+ }
+
+ if (!nsResolved) { // Resolve the numbering system ( default, native, traditional or finance ) into a "real" numbering system
+ UErrorCode localStatus = U_ZERO_ERROR;
+ LocalUResourceBundlePointer resource(ures_open(nullptr, inLocale.getName(), &localStatus));
+ LocalUResourceBundlePointer numberElementsRes(ures_getByKey(resource.getAlias(), gNumberElements, nullptr, &localStatus));
+ // Don't stomp on the catastrophic failure of OOM.
+ if (localStatus == U_MEMORY_ALLOCATION_ERROR) {
+ status = U_MEMORY_ALLOCATION_ERROR;
+ return nullptr;
+ }
+ while (!nsResolved) {
+ localStatus = U_ZERO_ERROR;
+ count = 0;
+ const UChar *nsName = ures_getStringByKeyWithFallback(numberElementsRes.getAlias(), buffer, &count, &localStatus);
+ // Don't stomp on the catastrophic failure of OOM.
+ if (localStatus == U_MEMORY_ALLOCATION_ERROR) {
+ status = U_MEMORY_ALLOCATION_ERROR;
+ return nullptr;
+ }
+ if ( count > 0 && count < ULOC_KEYWORDS_CAPACITY ) { // numbering system found
+ u_UCharsToChars(nsName, buffer, count);
+ buffer[count] = '\0'; // Make sure it is null terminated.
+ nsResolved = TRUE;
+ }
+
+ if (!nsResolved) { // Fallback behavior per TR35 - traditional falls back to native, finance and native fall back to default
+ if (!uprv_strcmp(buffer,gNative) || !uprv_strcmp(buffer,gFinance)) {
+ uprv_strcpy(buffer,gDefault);
+ } else if (!uprv_strcmp(buffer,gTraditional)) {
+ uprv_strcpy(buffer,gNative);
+ } else { // If we get here we couldn't find even the default numbering system
+ usingFallback = TRUE;
+ nsResolved = TRUE;
+ }
+ }
+ }
+ }
+
+ if (usingFallback) {
+ status = U_USING_FALLBACK_WARNING;
+ NumberingSystem *ns = new NumberingSystem();
+ if (ns == nullptr) {
+ status = U_MEMORY_ALLOCATION_ERROR;
+ }
+ return ns;
+ } else {
+ return NumberingSystem::createInstanceByName(buffer, status);
+ }
+ }
+
+NumberingSystem* U_EXPORT2
+NumberingSystem::createInstance(UErrorCode& status) {
+ return NumberingSystem::createInstance(Locale::getDefault(), status);
+}
+
+NumberingSystem* U_EXPORT2
+NumberingSystem::createInstanceByName(const char *name, UErrorCode& status) {
+ int32_t radix = 10;
+ int32_t algorithmic = 0;
+
+ LocalUResourceBundlePointer numberingSystemsInfo(ures_openDirect(nullptr, gNumberingSystems, &status));
+ LocalUResourceBundlePointer nsCurrent(ures_getByKey(numberingSystemsInfo.getAlias(), gNumberingSystems, nullptr, &status));
+ LocalUResourceBundlePointer nsTop(ures_getByKey(nsCurrent.getAlias(), name, nullptr, &status));
+
+ UnicodeString nsd = ures_getUnicodeStringByKey(nsTop.getAlias(), gDesc, &status);
+
+ ures_getByKey(nsTop.getAlias(), gRadix, nsCurrent.getAlias(), &status);
+ radix = ures_getInt(nsCurrent.getAlias(), &status);
+
+ ures_getByKey(nsTop.getAlias(), gAlgorithmic, nsCurrent.getAlias(), &status);
+ algorithmic = ures_getInt(nsCurrent.getAlias(), &status);
+
+ UBool isAlgorithmic = ( algorithmic == 1 );
+
+ if (U_FAILURE(status)) {
+ // Don't stomp on the catastrophic failure of OOM.
+ if (status != U_MEMORY_ALLOCATION_ERROR) {
+ status = U_UNSUPPORTED_ERROR;
+ }
+ return nullptr;
+ }
+
+ LocalPointer<NumberingSystem> ns(NumberingSystem::createInstance(radix, isAlgorithmic, nsd, status), status);
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+ ns->setName(name);
+ return ns.orphan();
+}
+
+ /**
+ * Destructor.
+ * @draft ICU 4.2
+ */
+NumberingSystem::~NumberingSystem() {
+}
+
+int32_t NumberingSystem::getRadix() const {
+ return radix;
+}
+
+UnicodeString NumberingSystem::getDescription() const {
+ return desc;
+}
+
+const char * NumberingSystem::getName() const {
+ return name;
+}
+
+void NumberingSystem::setRadix(int32_t r) {
+ radix = r;
+}
+
+void NumberingSystem::setAlgorithmic(UBool c) {
+ algorithmic = c;
+}
+
+void NumberingSystem::setDesc(const UnicodeString &d) {
+ desc.setTo(d);
+}
+void NumberingSystem::setName(const char *n) {
+ if ( n == nullptr ) {
+ name[0] = (char) 0;
+ } else {
+ uprv_strncpy(name,n,NUMSYS_NAME_CAPACITY);
+ name[NUMSYS_NAME_CAPACITY] = '\0'; // Make sure it is null terminated.
+ }
+}
+UBool NumberingSystem::isAlgorithmic() const {
+ return ( algorithmic );
+}
+
+StringEnumeration* NumberingSystem::getAvailableNames(UErrorCode &status) {
+ // TODO(ticket #11908): Init-once static cache, with u_cleanup() callback.
+ static StringEnumeration* availableNames = nullptr;
+
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+
+ if ( availableNames == nullptr ) {
+ // TODO: Simple array of UnicodeString objects, based on length of table resource?
+ LocalPointer<UVector> numsysNames(new UVector(uprv_deleteUObject, nullptr, status), status);
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+
+ UErrorCode rbstatus = U_ZERO_ERROR;
+ UResourceBundle *numberingSystemsInfo = ures_openDirect(nullptr, "numberingSystems", &rbstatus);
+ numberingSystemsInfo = ures_getByKey(numberingSystemsInfo, "numberingSystems", numberingSystemsInfo, &rbstatus);
+ if (U_FAILURE(rbstatus)) {
+ // Don't stomp on the catastrophic failure of OOM.
+ if (rbstatus == U_MEMORY_ALLOCATION_ERROR) {
+ status = rbstatus;
+ } else {
+ status = U_MISSING_RESOURCE_ERROR;
+ }
+ ures_close(numberingSystemsInfo);
+ return nullptr;
+ }
+
+ while ( ures_hasNext(numberingSystemsInfo) && U_SUCCESS(status) ) {
+ LocalUResourceBundlePointer nsCurrent(ures_getNextResource(numberingSystemsInfo, nullptr, &rbstatus));
+ if (rbstatus == U_MEMORY_ALLOCATION_ERROR) {
+ status = rbstatus; // we want to report OOM failure back to the caller.
+ break;
+ }
+ const char *nsName = ures_getKey(nsCurrent.getAlias());
+ LocalPointer<UnicodeString> newElem(new UnicodeString(nsName, -1, US_INV), status);
+ if (U_SUCCESS(status)) {
+ numsysNames->addElement(newElem.getAlias(), status);
+ if (U_SUCCESS(status)) {
+ newElem.orphan(); // on success, the numsysNames vector owns newElem.
+ }
+ }
+ }
+
+ ures_close(numberingSystemsInfo);
+ if (U_FAILURE(status)) {
+ return nullptr;
+ }
+ availableNames = new NumsysNameEnumeration(numsysNames.getAlias(), status);
+ if (availableNames == nullptr) {
+ status = U_MEMORY_ALLOCATION_ERROR;
+ return nullptr;
+ }
+ numsysNames.orphan(); // The names got adopted.
+ }
+
+ return availableNames;
+}
+
+NumsysNameEnumeration::NumsysNameEnumeration(UVector *numsysNames, UErrorCode& /*status*/) {
+ pos=0;
+ fNumsysNames = numsysNames;
+}
+
+const UnicodeString*
+NumsysNameEnumeration::snext(UErrorCode& status) {
+ if (U_SUCCESS(status) && (fNumsysNames != nullptr) && (pos < fNumsysNames->size())) {
+ return (const UnicodeString*)fNumsysNames->elementAt(pos++);
+ }
+ return nullptr;
+}
+
+void
+NumsysNameEnumeration::reset(UErrorCode& /*status*/) {
+ pos=0;
+}
+
+int32_t
+NumsysNameEnumeration::count(UErrorCode& /*status*/) const {
+ return (fNumsysNames==nullptr) ? 0 : fNumsysNames->size();
+}
+
+NumsysNameEnumeration::~NumsysNameEnumeration() {
+ delete fNumsysNames;
+}
+U_NAMESPACE_END
+
+#endif /* #if !UCONFIG_NO_FORMATTING */
+
+//eof