1 // Copyright 2014 PDFium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com 6 7 #include "core/fxcrt/fx_bidi.h" 8 #include "core/fxcrt/fx_ucd.h" 9 10 #include <algorithm> 11 12 CFX_BidiChar::CFX_BidiChar() 13 : m_CurrentSegment({0, 0, NEUTRAL}), m_LastSegment({0, 0, NEUTRAL}) {} 14 15 bool CFX_BidiChar::AppendChar(FX_WCHAR wch) { 16 uint32_t dwProps = FX_GetUnicodeProperties(wch); 17 int32_t iBidiCls = (dwProps & FX_BIDICLASSBITSMASK) >> FX_BIDICLASSBITS; 18 Direction direction = NEUTRAL; 19 switch (iBidiCls) { 20 case FX_BIDICLASS_L: 21 case FX_BIDICLASS_AN: 22 case FX_BIDICLASS_EN: 23 direction = LEFT; 24 break; 25 case FX_BIDICLASS_R: 26 case FX_BIDICLASS_AL: 27 direction = RIGHT; 28 break; 29 } 30 31 bool bChangeDirection = (direction != m_CurrentSegment.direction); 32 if (bChangeDirection) 33 StartNewSegment(direction); 34 35 m_CurrentSegment.count++; 36 return bChangeDirection; 37 } 38 39 bool CFX_BidiChar::EndChar() { 40 StartNewSegment(NEUTRAL); 41 return m_LastSegment.count > 0; 42 } 43 44 void CFX_BidiChar::StartNewSegment(CFX_BidiChar::Direction direction) { 45 m_LastSegment = m_CurrentSegment; 46 m_CurrentSegment.start += m_CurrentSegment.count; 47 m_CurrentSegment.count = 0; 48 m_CurrentSegment.direction = direction; 49 } 50 51 CFX_BidiString::CFX_BidiString(const CFX_WideString& str) 52 : m_Str(str), 53 m_pBidiChar(new CFX_BidiChar), 54 m_eOverallDirection(CFX_BidiChar::LEFT) { 55 for (int i = 0; i < m_Str.GetLength(); ++i) { 56 if (m_pBidiChar->AppendChar(m_Str.GetAt(i))) 57 m_Order.push_back(m_pBidiChar->GetSegmentInfo()); 58 } 59 if (m_pBidiChar->EndChar()) 60 m_Order.push_back(m_pBidiChar->GetSegmentInfo()); 61 62 size_t nR2L = std::count_if(m_Order.begin(), m_Order.end(), 63 [](const CFX_BidiChar::Segment& seg) { 64 return seg.direction == CFX_BidiChar::RIGHT; 65 }); 66 67 size_t nL2R = std::count_if(m_Order.begin(), m_Order.end(), 68 [](const CFX_BidiChar::Segment& seg) { 69 return seg.direction == CFX_BidiChar::LEFT; 70 }); 71 72 if (nR2L > 0 && nR2L >= nL2R) 73 SetOverallDirectionRight(); 74 } 75 76 CFX_BidiString::~CFX_BidiString() {} 77 78 void CFX_BidiString::SetOverallDirectionRight() { 79 if (m_eOverallDirection != CFX_BidiChar::RIGHT) { 80 std::reverse(m_Order.begin(), m_Order.end()); 81 m_eOverallDirection = CFX_BidiChar::RIGHT; 82 } 83 } 84