2
1
mirror of https://github.com/qpdf/qpdf.git synced 2025-01-11 10:38:13 +00:00
qpdf/libqpdf/QPDF_String.cc

174 lines
4.6 KiB
C++
Raw Normal View History

#include <qpdf/QPDF_String.hh>
#include <qpdf/JSON_writer.hh>
#include <qpdf/QUtil.hh>
// DO NOT USE ctype -- it is locale dependent for some things, and it's not worth the risk of
// including it in case it may accidentally be used.
static bool
is_iso_latin1_printable(char ch)
{
2023-05-21 17:35:09 +00:00
return (((ch >= 32) && (ch <= 126)) || (static_cast<unsigned char>(ch) >= 160));
}
QPDF_String::QPDF_String(std::string const& val) :
2024-08-12 16:52:42 +00:00
QPDFValue(::ot_string),
val(val)
{
}
std::shared_ptr<QPDFObject>
QPDF_String::create(std::string const& val)
{
return do_create(new QPDF_String(val));
}
std::shared_ptr<QPDFObject>
QPDF_String::create_utf16(std::string const& utf8_val)
{
std::string result;
if (!QUtil::utf8_to_pdf_doc(utf8_val, result, '?')) {
result = QUtil::utf8_to_utf16(utf8_val);
}
return do_create(new QPDF_String(result));
}
std::shared_ptr<QPDFObject>
QPDF_String::copy(bool shallow)
{
return create(val);
}
std::string
QPDF_String::unparse()
{
return unparse(false);
}
void
QPDF_String::writeJSON(int json_version, JSON::Writer& p)
{
auto candidate = getUTF8Val();
if (json_version == 1) {
p << "\"" << JSON::Writer::encode_string(candidate) << "\"";
} else {
// See if we can unambiguously represent as Unicode.
if (QUtil::is_utf16(this->val) || QUtil::is_explicit_utf8(this->val)) {
2024-02-24 15:29:41 +00:00
p << "\"u:" << JSON::Writer::encode_string(candidate) << "\"";
return;
} else if (!useHexString()) {
std::string test;
if (QUtil::utf8_to_pdf_doc(candidate, test, '?') && (test == this->val)) {
// This is a PDF-doc string that can be losslessly encoded as Unicode.
2024-02-24 15:29:41 +00:00
p << "\"u:" << JSON::Writer::encode_string(candidate) << "\"";
return;
}
}
2024-02-24 15:29:41 +00:00
p << "\"b:" << QUtil::hex_encode(val) << "\"";
}
}
bool
QPDF_String::useHexString() const
{
// Heuristic: use the hexadecimal representation of a string if there are any non-printable (in
// PDF Doc encoding) characters or if too large of a proportion of the string consists of
// non-ASCII characters.
unsigned int non_ascii = 0;
2022-09-23 17:56:07 +00:00
for (auto const ch: this->val) {
if (ch > 126) {
++non_ascii;
} else if (ch >= 32) {
continue;
} else if (ch < 0 || ch >= 24) {
++non_ascii;
2023-05-21 17:35:09 +00:00
} else if (!(ch == '\n' || ch == '\r' || ch == '\t' || ch == '\b' || ch == '\f')) {
2022-09-23 17:56:07 +00:00
return true;
}
}
2022-09-23 17:56:07 +00:00
return 5 * non_ascii > val.length();
}
std::string
QPDF_String::unparse(bool force_binary)
{
bool use_hexstring = force_binary || useHexString();
std::string result;
if (use_hexstring) {
static auto constexpr hexchars = "0123456789abcdef";
result.reserve(2 * this->val.length() + 2);
result += '<';
for (const char c: this->val) {
result += hexchars[static_cast<unsigned char>(c) >> 4];
result += hexchars[c & 0x0f];
}
result += '>';
} else {
result += "(";
for (unsigned int i = 0; i < this->val.length(); ++i) {
char ch = this->val.at(i);
switch (ch) {
case '\n':
result += "\\n";
break;
case '\r':
result += "\\r";
break;
case '\t':
result += "\\t";
break;
case '\b':
result += "\\b";
break;
case '\f':
result += "\\f";
break;
case '(':
result += "\\(";
break;
case ')':
result += "\\)";
break;
case '\\':
result += "\\\\";
break;
default:
if (is_iso_latin1_printable(ch)) {
result += this->val.at(i);
} else {
2023-05-21 17:35:09 +00:00
result += "\\" +
QUtil::int_to_string_base(
2023-05-21 17:35:09 +00:00
static_cast<int>(static_cast<unsigned char>(ch)), 8, 3);
}
break;
}
}
result += ")";
}
return result;
}
std::string
QPDF_String::getUTF8Val() const
{
if (QUtil::is_utf16(this->val)) {
return QUtil::utf16_to_utf8(this->val);
2022-04-23 20:39:27 +00:00
} else if (QUtil::is_explicit_utf8(this->val)) {
// PDF 2.0 allows UTF-8 strings when explicitly prefixed with the three-byte representation
// of U+FEFF.
return this->val.substr(3);
} else {
return QUtil::pdf_doc_to_utf8(this->val);
}
}