File: utf8_support.c

package info (click to toggle)
latex2rtf 2.3.18a-5
  • links: PTS
  • area: main
  • in suites: forky, sid, trixie
  • size: 9,392 kB
  • sloc: ansic: 20,424; makefile: 660; sh: 478; perl: 22
file content (102 lines) | stat: -rwxr-xr-x 3,226 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/* utf8_support.c - LaTeX to RTF conversion program

This file contains a function used to convert verbatim sections containing
Utf-8 characters properly.

Authors:
    2018 Alex Itkes
*/

#include "main.h"
#include "chars.h"
#include "fonts.h"
#include "encodings.h"

/* Initializes a unicode character represented by the Utf-8 sequencd
   based on the first byte of the sequence. The returned value must
   then be updated with updateUf8Sequence a number of times (returned
   by getUtf8SequenceLength using the following bytes of the input
   stream.
*/
uint16_t getUtf8SequenceInitialValue (uint8_t byte)
{
    if (byte >= 0xF0) {
        return byte & ~0xF0;
    } else if (byte >= 0xE0) {
        return byte & ~0xE0;
    } else if (byte >= 0xC0) {
        return byte & ~0xC0;
    } else {
        return 0;
    }
}

/* Determines the length of a Utf-8 sequence based on its first byte.
   Actually returns the length decreased by 1, i.e. the number of
   bytes to be read later.
*/
uint16_t getUtf8SequenceLength (uint8_t byte)
{
    if (byte >= 0xF0) {
        return 3;
    } else if (byte >= 0xE0) {
        return 2;
    } else if (byte >= 0xC0) {
        return 1;
    } else {
        return 0;
    }
}

/* Adds a following byte of Utf-8 sequence to the unicode
   character, return the resulting code.
*/
uint16_t updateUtf8Sequence (uint16_t value, uint8_t byte)
{
    return ((value << 6) + (byte & ~0xC0));
}

/* Prints a string to RTF with escaped characters if needed.
   unlike putRtfStrEscaped it prints Utf8 characters properly.
*/
void putRtfUtf8StrEscaped(const char * string)
{
    char *s = (char *) string;
    if (string == NULL) return;
    while (*s) {
        /* Much of code actually copied from Convert () function, but some
           code later moved to additional functions (getUtf8SequenceLength,
           getUtf8SequenceInitialValue and updateUtf8Sequence).
           TODO: Could it be a good idea to call them also from Convert?
        */
        if ((uint8_t)(*s) >= 0x80 && (CurrentFontEncoding() == ENCODING_UTF8) && *(s + 1)) {
            /* Handle a Utf-8 byte sequence. Try to convert it to a Unicode
               character in same way with the Convert function does and then
               output it with CmdUnicodeChar.
            */
            uint8_t byte = *s;

            uint16_t len = getUtf8SequenceLength (byte);
            uint16_t value = getUtf8SequenceInitialValue (byte);
            uint16_t i;

            s++;
            for (i=0; i<len; i++) {
                if (*s) {
                    value = updateUtf8Sequence (value, *s);
                    s++;
                } else {
                    /* If the sequnce is shorted then it should be, display warning and output nothing. */
                    diagnostics(1, "An incorrect Utf-8 sequence encountered at end of string '%s', continuing anyway...", string);
                    return;
                }
            }

            diagnostics(4,"(flag = 0x%X) char value = 0X%04X or %u (%u bytes)", (unsigned char) byte, value, value, len);
            CmdUnicodeChar(value);
        } else {
            /* Not a Utf8 character. */
            putRtfCharEscaped(*s++);
        }
    }
}