add partial support of Arabic reshaping on Linux

This commit is contained in:
Jinhao
2020-03-26 01:41:32 +08:00
parent b46966bf02
commit c9b9451443
5 changed files with 363 additions and 28 deletions

View File

@@ -146,6 +146,8 @@ IsWindows8OrGreater()
# include "posix/platform_spec.hpp"
# include <fontconfig/fontconfig.h>
# if defined(NANA_USE_XFT)
# include <nana/unicode_bidi.hpp>
# include "text_reshaping.hpp"
# include <X11/Xft/Xft.h>
# include <iconv.h>
# include <fstream>
@@ -211,16 +213,40 @@ namespace nana
int const init_x = x;
std::unique_ptr<FT_UInt[]> glyph_indexes(new FT_UInt[len]);
while(true)
//The RTL and shaping should be handled manually, because the libXft and X doesn't support these language features.
std::wstring rtl;
auto ents = unicode_reorder(str, len);
for(auto & e : ents)
{
auto preferred = _m_scan_fonts(xft, str, len, glyph_indexes.get());
x += _m_draw(xftdraw, xftcolor, preferred.first, x, y, str, preferred.second, glyph_indexes.get());
auto size = static_cast<std::size_t>(e.end - e.begin);
auto p = e.begin;
if(len == preferred.second)
break;
if(unicode_bidi::is_text_right(e))
{
std::wstring reverse_str;
for(auto i = static_cast<int>(e.end - e.begin) - 1; i > -1; --i)
reverse_str += e.begin[i];
len -= preferred.second;
str += preferred.second;
//Process reshaping.
rtl = nana::reshaping::arabic::reshape(reverse_str);
p = rtl.c_str();
size = rtl.size();
}
while(true)
{
//Scan the string until the character which font is not same with the font of the first character where the scan begins.
auto preferred = _m_scan_fonts(xft, p, size, glyph_indexes.get());
x += _m_draw(xftdraw, xftcolor, preferred.first, x, y, p, preferred.second, glyph_indexes.get());
if(size == preferred.second)
break;
size -= preferred.second;
p += preferred.second;
}
rtl.clear();
}
return x - init_x;
@@ -298,14 +324,14 @@ namespace nana
if(ptab == p)
{
++p;
//x += static_cast<int>(tab_pixels_);
continue;
}
auto const size = ptab - p;
::XftDrawGlyphs(xftdraw, xftcolor, xft, x, y, glyph_indexes + off, size);
::XftGlyphExtents(disp_, xft, glyph_indexes + off, size, &ext);
x += ext.xOff;
if(ptab == end)