#include "emojis.hpp" #include EmojiResource::EmojiResource(std::string filepath) : m_filepath(filepath) {} bool EmojiResource::Load() { m_fp = std::fopen(m_filepath.c_str(), "rb"); if (m_fp == nullptr) return false; int index_pos; std::fread(&index_pos, 4, 1, m_fp); std::fseek(m_fp, index_pos, SEEK_SET); int num_entries; std::fread(&num_entries, 4, 1, m_fp); for (int i = 0; i < num_entries; i++) { static int pattern_strlen, shortcode_strlen, len, pos; std::fread(&pattern_strlen, 4, 1, m_fp); std::string pattern(pattern_strlen, '\0'); std::fread(pattern.data(), pattern_strlen, 1, m_fp); const auto pattern_hex = HexToPattern(pattern); std::fread(&shortcode_strlen, 4, 1, m_fp); std::string shortcode(shortcode_strlen, '\0'); if (shortcode_strlen > 0) { std::fread(shortcode.data(), shortcode_strlen, 1, m_fp); m_shortcode_index[shortcode] = pattern_hex; m_pattern_shortcode_index[pattern_hex] = shortcode; } std::fread(&len, 4, 1, m_fp); std::fread(&pos, 4, 1, m_fp); m_index[pattern] = std::make_pair(pos, len); m_patterns.push_back(pattern_hex); } std::sort(m_patterns.begin(), m_patterns.end(), [](const Glib::ustring &a, const Glib::ustring &b) { return a.size() > b.size(); }); return true; } Glib::RefPtr EmojiResource::GetPixBuf(const Glib::ustring &pattern) { Glib::ustring key = PatternToHex(pattern); const auto it = m_index.find(key); if (it == m_index.end()) return Glib::RefPtr(); const int pos = it->second.first; const int len = it->second.second; std::fseek(m_fp, pos, SEEK_SET); std::vector data(len); std::fread(data.data(), len, 1, m_fp); auto loader = Gdk::PixbufLoader::create(); loader->write(static_cast(data.data()), data.size()); loader->close(); return loader->get_pixbuf(); } Glib::ustring EmojiResource::PatternToHex(const Glib::ustring &pattern) { Glib::ustring ret; for (int i = 0; i < pattern.size(); i++) { auto c = pattern.at(i); ret += Glib::ustring::format(std::hex, c); ret += "-"; } return ret.substr(0, ret.size() - 1); } Glib::ustring EmojiResource::HexToPattern(Glib::ustring hex) { std::vector parts; Glib::ustring::size_type pos = 0; Glib::ustring token; while ((pos = hex.find("-")) != Glib::ustring::npos) { token = hex.substr(0, pos); if (token.length() % 2 == 1) token = "0" + token; parts.push_back(token); hex.erase(0, pos + 1); } parts.push_back(hex); Glib::ustring ret; for (const auto &part : parts) { auto x = static_cast(std::stoul(part.c_str(), nullptr, 16)); ret += x; } return ret; } void EmojiResource::ReplaceEmojis(Glib::RefPtr buf, int size) { auto get_text = [&]() -> auto { Gtk::TextBuffer::iterator a, b; buf->get_bounds(a, b); return buf->get_slice(a, b, true); }; auto text = get_text(); int searchpos; for (const auto &pattern : m_patterns) { searchpos = 0; Glib::RefPtr pixbuf; while (true) { size_t r = text.find(pattern, searchpos); if (r == Glib::ustring::npos) break; if (!pixbuf) { pixbuf = GetPixBuf(pattern); if (pixbuf) pixbuf = pixbuf->scale_simple(size, size, Gdk::INTERP_BILINEAR); else break; } searchpos = r + pattern.size(); const auto start_it = buf->get_iter_at_offset(r); const auto end_it = buf->get_iter_at_offset(r + pattern.size()); auto it = buf->erase(start_it, end_it); buf->insert_pixbuf(it, pixbuf); int alen = text.size(); text = get_text(); int blen = text.size(); searchpos -= (alen - blen); } } } std::string EmojiResource::GetShortCodeForPattern(const Glib::ustring &pattern) { auto it = m_pattern_shortcode_index.find(pattern); if (it != m_pattern_shortcode_index.end()) return it->second; return ""; } const std::vector &EmojiResource::GetPatterns() const { return m_patterns; } const std::map &EmojiResource::GetShortCodes() const { return m_shortcode_index; }