/******************************************************************************
* Copyright (c) 2018(-2023) STMicroelectronics.
* All rights reserved.
*
* This file is part of the TouchGFX 4.22.0 distribution.
*
* This software is licensed under terms that can be found in the LICENSE file in
* the root directory of this software component.
* If no LICENSE file comes with this software, it is provided AS-IS.
*
*******************************************************************************/

/**
 * @file touchgfx/Unicode.hpp
 *
 * Declares the touchgfx::Unicode class.
 */
#ifndef TOUCHGFX_UNICODE_HPP
#define TOUCHGFX_UNICODE_HPP

#include <stdarg.h>
#include <touchgfx/hal/Types.hpp>

namespace touchgfx
{
/**
 * This class provides simple helper functions for working with strings which are stored as a
 * null-terminated array of 16-bit characters.
 */
class Unicode
{
public:
    /** Use the UnicodeChar typename when referring to characters in a string. */
    typedef uint16_t UnicodeChar;

    /**
     * Gets the length of a null-terminated Unicode string.
     *
     * @param  str The string in question.
     *
     * @return Length of string.
     */
    static uint16_t strlen(const UnicodeChar* str);

    /**
     * Gets the length of a null-terminated string.
     *
     * @param  str The string.
     *
     * @return Length of string.
     */
    static uint16_t strlen(const char* str);

    /**
     * Copy a string to a destination buffer, UnicodeChar to UnicodeChar version. Stops
     * after copying maxchars Unicode characters or after copying the ending zero
     * termination UnicodeChar.
     *
     * @param [out] dst      The destination buffer. Must have a size of at least maxchars.
     * @param [in]  src      The source string.
     * @param       maxchars Maximum number of UnicodeChars to copy.
     *
     * @return The number of characters copied (excluding null-termination if encountered)
     *
     * @note If there is no null-termination among the first n UnicodeChars of src, the string
     *       placed in destination will NOT be null-terminated!
     */
    static uint16_t strncpy(UnicodeChar* RESTRICT dst, const UnicodeChar* RESTRICT src, uint16_t maxchars);

    /**
     * Copy a string to a destination buffer, char to UnicodeChar version. Stops after
     * copying maxchars Unicode characters or after copying the ending null-termination
     * UnicodeChar.
     *
     * @param [out] dst      The destination buffer. Must have a size of at least maxchars.
     * @param [in]  src      The source string.
     * @param       maxchars Maximum number of chars to copy.
     *
     * @return The number of characters copied (excluding null-termination if encountered)
     *
     * @note If there is no null-termination among the first n UnicodeChars of src, the string
     *       placed in destination will NOT be null-terminated!
     */
    static uint16_t strncpy(UnicodeChar* RESTRICT dst, const char* RESTRICT src, uint16_t maxchars);

    /**
     * Integer to ASCII conversion. Supports radix 2 to radix 36.
     *
     * @param       value      to convert.
     * @param [out] buffer     to place result in.
     * @param       bufferSize Size of buffer (number of UnicodeChar's).
     * @param       radix      to use (8 for octal, 10 for decimal, 16 for hex)
     */
    static void itoa(int32_t value, UnicodeChar* buffer, uint16_t bufferSize, int radix);

    /**
     * Integer to ASCII conversion. Supports radix 2 to radix 36.
     *
     * @param       value      to convert.
     * @param [out] buffer     to place result in.
     * @param       bufferSize Size of buffer (number of UnicodeChar's).
     * @param       radix      to use (8 for octal, 10 for decimal, 16 for hex)
     */
    static void utoa(uint32_t value, UnicodeChar* buffer, uint16_t bufferSize, int radix);

    /**
     * String to integer conversion. Starts conversion at the start of the string. Running
     * digits from here are converted. Only radix 10 supported.
     *
     * @param  s Radix 10, null-terminated Unicode string to convert.
     *
     * @return The converted integer value of the string, 0 if the string does not start
     *         with a digit.
     */
    static int atoi(const UnicodeChar* s);

    /**
     * Formats a string and adds null termination. The string is formatted like when the
     * standard printf is used.
     *
     * Support formats: \%c (element type: char), \%s (element type: null-terminated
     * UnicodeChar list), \%u, \%i, \%d, \%o, \%x (all these are integers formatted in radix
     * 10, 10, 10, 8, 16 respectively).
     *
     * The number formats (\%u, \%i, \%d, \%o and \%x) all support \%[0][length]X to specify
     * the size of the generated field (length) and whether the number should be prefixed
     * with zeros (or blanks).
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string.
     * @param       ...     The values to insert in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintfFloat, snprintfFloats
     *
     * @note \%f is not supported by this function because floats are converted to doubles when
     *       given as parameters in a variable argument list (va_list). Use snprintfFloat
     *       or snprintfFloats instead.
     */
    static UnicodeChar* snprintf(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* format, ...);

    /**
     * Variant of snprintf.
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string.
     * @param       pArg    The values to insert in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf
     */
    static UnicodeChar* vsnprintf(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* format, va_list pArg);

    /**
     * Formats a string and adds null termination. The string is formatted like when the
     * standard printf is used.
     *
     * Support formats: \%c (element type: char), \%s (element type: null-terminated
     * UnicodeChar list), \%u, \%i, \%d, \%o, \%x (all these are integers formatted in radix
     * 10, 10, 10, 8, 16 respectively).
     *
     * The number formats (\%u, \%i, \%d, \%o and \%x) all support
     *
     *     \%[flags][width][.precision]X
     *
     * Where flags can be:
     * - '-': left justify the field (see width).
     * - '+': force sign.
     * - ' ': insert space if value is positive.
     * - '0': left pad with zeros instead of spaces (see width).
     *
     * Where width is the desired width of the output. If the value is larger, more
     * characters may be generated, but not more than the parameter dstSize. If width is '*'
     * the actual width is read from the parameters passed to this function.
     *
     * Where precision is the number of number of digits after the decimal point, default is
     * 3. Use "\%.f" to not generate any numbers after the decimal point. If precision is '*'
     * the actual precision is read from the parameters passed to this function.
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string.
     * @param       ...     The values to insert in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintfFloat, snprintfFloats
     *
     * @note \%f is not supported by this function because floats are converted to doubles when
     *       given as parameters in a variable argument list (va_list). Use snprintfFloat or
     *       snprintfFloats instead.
     *
     * @see snprintfFloat, snprintfFloats
     */
    static UnicodeChar* snprintf(UnicodeChar* dst, uint16_t dstSize, const char* format, ...);

    /**
     * Variant of snprintf.
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string.
     * @param       pArg    The values to insert in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf
     */
    static UnicodeChar* vsnprintf(UnicodeChar* dst, uint16_t dstSize, const char* format, va_list pArg);

    /**
     * Variant of snprintf for floats only. The format supports several \%f with
     * flags/modifiers. The following is supported:
     *
     *     \%[flags][width][.precision]f
     *
     * Where flags can be:
     * - '-': left justify the field (see width).
     * - '+': force sign.
     * - ' ': insert space if value is positive
     * - '\#': insert decimal point even if there are not decimals
     * - '0': left pad with zeros instead of spaces (see width)
     *
     * Where width is the desired width of the output. If the value is larger, more
     * characters may be generated, but not more than the parameter dstSize. If width is '*'
     * the actual width is read from the list of values passed to this function.
     *
     * Where precision is the number of number of digits after the decimal point, default is
     * 3. Use "\%.f" to not generate any numbers after the decimal point. If precision is '*'
     * the actual precision is read from the list of values passed to this function.
     * @code{.cpp}
     *            float param1[3] = { 6.0f, 4.0f, 3.14159f };
     *            Unicode::snprintfFloats(buffer, 20, "%*.*f", param1);
     *            // buffer="3.1416" float param2[2] = { 3.14159f, -123.4f };
     *            Unicode::snprintfFloats(buffer, 20, "%f %f", param2);
     *            // buffer="3.142 -123.400"
     * @endcode
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string containing %f's.
     * @param [in]  values  The floating point values to insert for %f. The number of elements in
     *                      the array must match the number of %f's in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf, snprintfFloat
     */
    static UnicodeChar* snprintfFloats(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* format, const float* values);

    /**
     * Variant of snprintfFloats() for exactly one float only.
     *
     * The number format supports only one \%f with flags/modifiers. The following is
     * supported:
     *
     *     \%[flags][width][.precision]f
     *
     * Where flags can be:
     * - '-': left justify the field (see width).
     * - '+': force sign.
     * - ' ': insert space if value is positive.
     * - '\#': insert decimal point even if there are not decimals.
     * - '0': left pad with zeros instead of spaces (see width).
     *
     * Where width is the desired width of the output. If the value is larger, more
     * characters may be generated, but not more than the parameter dstSize.
     *
     * Where precision is the number of number of digits after the decimal point, default is
     * "3". Use "\%.f" to not generate any numbers after the decimal point.
     * @code{.cpp}
     *            Unicode::UnicodeChar buffer[20];
     *            Unicode::snprintfFloat(buffer, 20, "%6.4f", 3.14159f);
     *            // buffer="3.1416"
     *            Unicode::snprintfFloat(buffer, 20, "%#6.f", 3.14159f);
     *            // buffer="    3."
     *            Unicode::snprintfFloat(buffer, 20, "%6f", 3.14159f);
     *            // buffer=" 3.142"
     *            Unicode::snprintfFloat(buffer, 20, "%+06.f", 3.14159f);
     *            // buffer="+00003"
     * @endcode
     *
     * If more control over the output is needed, see snprintfFloats which can have more
     * than a single "\%f" in the string and also supports "*" in place of a number.
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string containing exactly on occurrence of %f.
     * @param       value   The floating point value to insert for %f.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf, snprintfFloats
     */
    static UnicodeChar* snprintfFloat(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* format, const float value)
    {
        return snprintfFloats(dst, dstSize, format, &value);
    }

    /**
     * Variant of snprintf for floats only. The format supports several \%f with
     * flags/modifiers. The following is supported:
     *
     *     \%[flags][width][.precision]f
     *
     * Where flags can be:
     * - '-': left justify the field (see width).
     * - '+': force sign.
     * - ' ': insert space if value is positive
     * - '\#': insert decimal point even if there are not decimals
     * - '0': left pad with zeros instead of spaces (see width)
     *
     * Where width is the desired width of the output. If the value is larger, more
     * characters may be generated, but not more than the parameter dstSize. If width is '*'
     * the actual width is read from the list of values passed to this function.
     *
     * Where precision is the number of number of digits after the decimal point, default is
     * 3. Use "\%.f" to not generate any numbers after the decimal point. If precision is '*'
     * the actual precision is read from the list of values passed to this function.
     * @code{.cpp}
     *            float param1[3] = { 6.0f, 4.0f, 3.14159f };
     *            Unicode::snprintfFloats(buffer, 20, "%*.*f", param1);
     *            // buffer="3.1416" float param2[2] = { 3.14159f, -123.4f };
     *            Unicode::snprintfFloats(buffer, 20, "%f %f", param2);
     *            // buffer="3.142 -123.400"
     * @endcode
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string containing %f's.
     * @param [in]  values  The floating point values to insert for %f. The number of elements in
     *                      the array must match the number of %f's in the format string.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf, snprintfFloat
     */
    static UnicodeChar* snprintfFloats(UnicodeChar* dst, uint16_t dstSize, const char* format, const float* values);

    /**
     * Variant of snprintfFloats() for exactly one float only.
     *
     * The number format supports only one \%f with flags/modifiers. The following is
     * supported:
     *
     *     \%[flags][width][.precision]f
     *
     * Where flags can be:
     * - '-': left justify the field (see width).
     * - '+': force sign.
     * - ' ': insert space if value is positive.
     * - '\#': insert decimal point even if there are not decimals.
     * - '0': left pad with zeros instead of spaces (see width).
     *
     * Where width is the desired width of the output. If the value is larger, more
     * characters may be generated, but not more than the parameter dstSize.
     *
     * Where precision is the number of number of digits after the decimal point, default is
     * "3". Use "\%.f" to not generate any numbers after the decimal point.
     * @code{.cpp}
     *            Unicode::UnicodeChar buffer[20];
     *            Unicode::snprintfFloat(buffer, 20, "%6.4f", 3.14159f);
     *            // buffer="3.1416"
     *            Unicode::snprintfFloat(buffer, 20, "%#6.f", 3.14159f);
     *            // buffer="    3."
     *            Unicode::snprintfFloat(buffer, 20, "%6f", 3.14159f);
     *            // buffer=" 3.142"
     *            Unicode::snprintfFloat(buffer, 20, "%+06.f", 3.14159f);
     *            // buffer="+00003"
     * @endcode
     *
     * If more control over the output is needed, see snprintfFloats which can have more
     * than a single "\%f" in the string and also supports "*" in place of a number.
     *
     * @param [out] dst     Buffer for the formatted string.
     * @param       dstSize Size of the dst buffer measured by number of UnicodeChars the buffer
     *                      can hold.
     * @param [in]  format  The format string containing exactly on occurrence of %f.
     * @param       value   The floating point value to insert for %f.
     *
     * @return pointer to the first element in the buffer where the formatted string is
     *         placed.
     *
     * @see snprintf, snprintfFloats
     */
    static UnicodeChar* snprintfFloat(UnicodeChar* dst, uint16_t dstSize, const char* format, const float value)
    {
        return snprintfFloats(dst, dstSize, format, &value);
    }

    /**
     * Compares up to maxchars characters in two strings. One character from each buffer is
     * compared, one at a time until the characters differ, until a terminating null-
     * character is reached, or until maxchars characters match in both strings, whichever
     * happens first.
     *
     * @param  str1     The first string.
     * @param  str2     The second string.
     * @param  maxchars The maximum number of chars to compare.
     *
     * @return Returns an integral value indicating the relationship between the strings: A
     *         zero value indicates that the characters compared in both strings are all
     *         equal. A value greater than zero indicates that the first character that does
     *         not match has a greater value in str1 than in str2; And a value less than
     *         zero indicates the opposite.
     */
    static int strncmp(const UnicodeChar* RESTRICT str1, const UnicodeChar* RESTRICT str2, uint16_t maxchars);

    /**
     * Like strncmp except that ignore any whitespaces in the two strings.
     *
     * @param  str1     The first string.
     * @param  str2     The second string.
     * @param  maxchars The maximum number of chars to compare.
     *
     * @return Returns an integral value indicating the relationship between the strings: A
     *         zero value indicates that the characters compared in both strings are all
     *         equal. A value greater than zero indicates that the first character that does
     *         not match has a greater value in str1 than in str2; And a value less than
     *         zero indicates the opposite.
     */
    static int strncmp_ignore_whitespace(const UnicodeChar* RESTRICT str1, const UnicodeChar* RESTRICT str2, uint16_t maxchars);

    /**
     * Convert a string from UTF8 to Unicode. The conversion stops if there is no more room
     * in the destination or if the terminating zero character has been converted.
     *
     * @param       utf8     The UTF8 string.
     * @param [out] dst      The destination buffer for the converted string.
     * @param       maxchars The maximum number of chars that the dst array can hold.
     *
     * @return The number of characters successfully converted from UTF8 to Unicode
     *         including the terminating zero.
     */
    static uint16_t fromUTF8(const uint8_t* utf8, UnicodeChar* dst, uint16_t maxchars);

    /**
     * Converts a string from Unicode to UTF8. The conversion stops if there is no more room
     * in the destination or if the terminating zero character has been converted. U+10000
     * through U+10FFFF are skipped.
     *
     * @param       unicode  The Unicode string.
     * @param [out] utf8     The destination buffer for the converted string.
     * @param       maxbytes The maximum number of bytes that the UTF8 array can hold.
     *
     * @return The number of characters successfully converted from Unicode to UTF8
     *         including the terminating zero.
     */
    static uint16_t toUTF8(const UnicodeChar* unicode, uint8_t* utf8, uint16_t maxbytes);

private:
    static void composeString(const UnicodeChar*& bufptr, UnicodeChar numberSign, UnicodeChar formatChar, bool hasPrecision, bool zeroPrefix, int precision, bool hasWidth, int width, bool alignLeft, int& charNumber, uint16_t dstSize, UnicodeChar* dst);

    static void parseFlagsAndPrecision(const UnicodeChar*& ucFormat, const char*& cFormat, UnicodeChar& numberSign, bool& alignLeft, bool& forceDecimalPoint, bool& zeroPrefix, bool& hasWidth, int& width, bool& hasPrecision, int& precision);

    static const UnicodeChar* skip_spaces(const UnicodeChar* str);
    static const UnicodeChar* skip_whitespace(const UnicodeChar* str);

    FORCE_INLINE_FUNCTION static UnicodeChar peekChar(const UnicodeChar* ucFormat, const char* cFormat);
    FORCE_INLINE_FUNCTION static void nextChar(const UnicodeChar*& ucFormat, const char*& cFormat);
    static UnicodeChar* vsnprintf(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* ucFormat, const char* cFormat, va_list pArg);
    static UnicodeChar* snprintfFloats(UnicodeChar* dst, uint16_t dstSize, const UnicodeChar* ucFormat, const char* cFormat, const float* values);
};

} // namespace touchgfx

#endif // TOUCHGFX_UNICODE_HPP