/* * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "BidiUtils.h" #include #include "minikin/Range.h" #include "UnicodeUtils.h" namespace minikin { const char LTR_1[] = "Hello, World"; const char RTL_1[] = "\u0627\u0644\u0633\u0644\u0627\u0645\u0020\u0639\u0644\u064A\u0643\u0645"; const char LTR_2[] = "Hello, Android"; const char RTL_2[] = "\u0639\u0644\u064A\u0643\u0645\u0020\u0627\u0644\u0633\u0644\u0627\u0645"; TEST(BidiUtilsTest, AllLTRCharText) { auto text = utf8ToUtf16(LTR_1); uint32_t ltrLength = text.size(); { BidiText bidiText(text, Range(0, ltrLength), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, ltrLength), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, ltrLength), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, ltrLength), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, ltrLength), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, ltrLength), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } TEST(BidiUtilsTest, AllRTLCharText) { auto text = utf8ToUtf16(RTL_1); uint32_t rtlLength = text.size(); { BidiText bidiText(text, Range(0, rtlLength), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, rtlLength), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, rtlLength), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, rtlLength), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, rtlLength), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { BidiText bidiText(text, Range(0, rtlLength), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } TEST(BidiUtilsTest, LTR_RTL_CharText) { auto text = utf8ToUtf16(std::string(LTR_1) + RTL_1); uint32_t ltrLength = utf8ToUtf16(LTR_1).size(); uint32_t rtlLength = utf8ToUtf16(RTL_1).size(); { // Logical Run: L1 L2 R1 R2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltrLength, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 // Visual Run : R2 R1 L1 L2 BidiText bidiText(text, Range(0, text.size()), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltrLength, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltrLength, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltrLength, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 // Visual Run : L1 L2 R1 R2 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 // Visual Run : R2 R1 L2 L1 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } TEST(BidiUtilsTest, RTL_LTR_CharText) { auto text = utf8ToUtf16(std::string(RTL_1) + LTR_1); uint32_t ltrLength = utf8ToUtf16(LTR_1).size(); uint32_t rtlLength = utf8ToUtf16(RTL_1).size(); { // Logical Run: R1 R2 L1 L2 // Visual Run : R2 R1 L1 L2 BidiText bidiText(text, Range(0, text.size()), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtlLength, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtlLength, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtlLength, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 // Visual Run : L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtlLength, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 // Visual Run : R1 R2 L1 L2 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength + rtlLength), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 // Visual Run : L2 L1 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltrLength + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } TEST(BidiUtilsTest, LTR_RTL_LTR_CharText) { auto text = utf8ToUtf16(std::string(LTR_1) + RTL_1 + LTR_2); uint32_t ltr1Length = utf8ToUtf16(LTR_1).size(); uint32_t ltr2Length = utf8ToUtf16(LTR_2).size(); uint32_t rtlLength = utf8ToUtf16(RTL_1).size(); { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L1 L2 R2 R1 L3 L4 BidiText bidiText(text, Range(0, text.size()), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length, ltr1Length + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length + rtlLength, ltr1Length + rtlLength + ltr2Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L3 L4 R2 R1 L1 2L BidiText bidiText(text, Range(0, text.size()), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length + rtlLength, text.size()), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length, ltr1Length + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L1 L2 R2 R1 L3 L4 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length, ltr1Length + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length + rtlLength, ltr1Length + rtlLength + ltr2Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L1 L2 R2 R1 L3 L4 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length, ltr1Length + rtlLength), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(ltr1Length + rtlLength, ltr1Length + rtlLength + ltr2Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L1 L2 R2 R1 L3 L4 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length + rtlLength + ltr2Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: L1 L2 R1 R2 L3 L4 // Visual Run : L1 L2 R2 R1 L3 L4 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, ltr1Length + rtlLength + ltr2Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } TEST(BidiUtilsTest, RTL_LTR_RTL_CharText) { auto text = utf8ToUtf16(std::string(RTL_1) + LTR_1 + RTL_2); uint32_t ltrLength = utf8ToUtf16(LTR_1).size(); uint32_t rtl1Length = utf8ToUtf16(RTL_1).size(); uint32_t rtl2Length = utf8ToUtf16(RTL_2).size(); { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R2 R1 L1 L2 R4 R3 BidiText bidiText(text, Range(0, text.size()), Bidi::LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length, ltrLength + rtl1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length + ltrLength, text.size()), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R4 R3 L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length + ltrLength, text.size()), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length, ltrLength + rtl1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R4 R3 L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length + ltrLength, text.size()), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length, ltrLength + rtl1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R4 R3 L1 L2 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::DEFAULT_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length + ltrLength, text.size()), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(rtl1Length, ltrLength + rtl1Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R1 R2 L1 L2 R3 R4 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_LTR); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length + ltrLength + rtl2Length), (*it).range); EXPECT_FALSE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } { // Logical Run: R1 R2 L1 L2 R3 R4 // Visual Run : R4 R3 L2 L1 R2 R1 BidiText bidiText(text, Range(0, text.size()), Bidi::FORCE_RTL); auto it = bidiText.begin(); EXPECT_NE(bidiText.end(), it); EXPECT_EQ(Range(0, rtl1Length + ltrLength + rtl2Length), (*it).range); EXPECT_TRUE((*it).isRtl); ++it; EXPECT_EQ(bidiText.end(), it); } } } // namespace minikin