|  | /* | 
|  | * Copyright (C) 2017 The Android Open Source Project | 
|  | * All rights reserved. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions | 
|  | * are met: | 
|  | *  * Redistributions of source code must retain the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer. | 
|  | *  * Redistributions in binary form must reproduce the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer in | 
|  | *    the documentation and/or other materials provided with the | 
|  | *    distribution. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
|  | * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|  | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
|  | * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
|  | * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
|  | * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
|  | * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
|  | * SUCH DAMAGE. | 
|  | */ | 
|  |  | 
|  | #include <wchar.h> | 
|  |  | 
|  | #include "private/icu.h" | 
|  |  | 
|  | int wcwidth(wchar_t wc) { | 
|  | // Fast-path ASCII. | 
|  | if (wc >= 0x20 && wc < 0x7f) return 1; | 
|  |  | 
|  | // ASCII NUL is a special case. | 
|  | if (wc == 0) return 0; | 
|  |  | 
|  | // C0. | 
|  | if (wc < ' ' || (wc >= 0x7f && wc <= 0xa0)) return -1; | 
|  |  | 
|  | // Now for the i18n part. This isn't defined or standardized, so a lot of the choices are | 
|  | // pretty arbitrary. See https://www.cl.cam.ac.uk/~mgk25/ucs/wcwidth.c for more details. | 
|  |  | 
|  | // Fancy unicode control characters? | 
|  | switch (__icu_charType(wc)) { | 
|  | case -1: | 
|  | // No icu4c available; give up. | 
|  | return -1; | 
|  | case U_CONTROL_CHAR: | 
|  | return -1; | 
|  | case U_NON_SPACING_MARK: | 
|  | case U_ENCLOSING_MARK: | 
|  | case U_FORMAT_CHAR: | 
|  | return 0; | 
|  | } | 
|  | if (__icu_hasBinaryProperty(wc, UCHAR_DEFAULT_IGNORABLE_CODE_POINT, nullptr)) return 0; | 
|  |  | 
|  | // Medial and final jamo render as zero width when used correctly. | 
|  | switch (__icu_getIntPropertyValue(wc, UCHAR_HANGUL_SYLLABLE_TYPE)) { | 
|  | case U_HST_VOWEL_JAMO: | 
|  | case U_HST_TRAILING_JAMO: | 
|  | return 0; | 
|  | case U_HST_LEADING_JAMO: | 
|  | case U_HST_LV_SYLLABLE: | 
|  | case U_HST_LVT_SYLLABLE: | 
|  | return 2; | 
|  | } | 
|  |  | 
|  | if (wc >= 0x3248 && wc <= 0x4dff) { | 
|  | // Circled two-digit CJK "speed sign" numbers. EastAsianWidth is ambiguous, | 
|  | // but wide makes more sense. | 
|  | if (wc <= 0x324f) return 2; | 
|  | // Hexagrams. EastAsianWidth is neutral, but wide seems better. | 
|  | if (wc >= 0x4dc0) return 2; | 
|  | } | 
|  |  | 
|  | // The EastAsianWidth property is at least defined by the Unicode standard! | 
|  | switch (__icu_getIntPropertyValue(wc, UCHAR_EAST_ASIAN_WIDTH)) { | 
|  | case U_EA_AMBIGUOUS: | 
|  | case U_EA_HALFWIDTH: | 
|  | case U_EA_NARROW: | 
|  | case U_EA_NEUTRAL: | 
|  | return 1; | 
|  | case U_EA_FULLWIDTH: | 
|  | case U_EA_WIDE: | 
|  | return 2; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |