blob: fd55dd611d79056fbeaba7accf02eba5dbdc0e7d (
plain)
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
|
/*
* Copyright (c) 2021, Max Wipfli <mail@maxwipfli.ch>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <AK/Concepts.h>
#include <AK/Error.h>
#include <AK/Forward.h>
namespace AK::UnicodeUtils {
template<typename Callback>
[[nodiscard]] constexpr int code_point_to_utf8(u32 code_point, Callback callback)
{
if (code_point <= 0x7f) {
callback((char)code_point);
return 1;
} else if (code_point <= 0x07ff) {
callback((char)(((code_point >> 6) & 0x1f) | 0xc0));
callback((char)(((code_point >> 0) & 0x3f) | 0x80));
return 2;
} else if (code_point <= 0xffff) {
callback((char)(((code_point >> 12) & 0x0f) | 0xe0));
callback((char)(((code_point >> 6) & 0x3f) | 0x80));
callback((char)(((code_point >> 0) & 0x3f) | 0x80));
return 3;
} else if (code_point <= 0x10ffff) {
callback((char)(((code_point >> 18) & 0x07) | 0xf0));
callback((char)(((code_point >> 12) & 0x3f) | 0x80));
callback((char)(((code_point >> 6) & 0x3f) | 0x80));
callback((char)(((code_point >> 0) & 0x3f) | 0x80));
return 4;
}
return -1;
}
template<FallibleFunction<char> Callback>
[[nodiscard]] ErrorOr<int> try_code_point_to_utf8(u32 code_point, Callback&& callback)
{
if (code_point <= 0x7f) {
TRY(callback(static_cast<char>(code_point)));
return 1;
}
if (code_point <= 0x07ff) {
TRY(callback(static_cast<char>((((code_point >> 6) & 0x1f) | 0xc0))));
TRY(callback(static_cast<char>((((code_point >> 0) & 0x3f) | 0x80))));
return 2;
}
if (code_point <= 0xffff) {
TRY(callback(static_cast<char>((((code_point >> 12) & 0x0f) | 0xe0))));
TRY(callback(static_cast<char>((((code_point >> 6) & 0x3f) | 0x80))));
TRY(callback(static_cast<char>((((code_point >> 0) & 0x3f) | 0x80))));
return 3;
}
if (code_point <= 0x10ffff) {
TRY(callback(static_cast<char>((((code_point >> 18) & 0x07) | 0xf0))));
TRY(callback(static_cast<char>((((code_point >> 12) & 0x3f) | 0x80))));
TRY(callback(static_cast<char>((((code_point >> 6) & 0x3f) | 0x80))));
TRY(callback(static_cast<char>((((code_point >> 0) & 0x3f) | 0x80))));
return 4;
}
return -1;
}
}
|