blob: 6d252cf6baac27d7d1b2c4fe0290c45c952b378a [file] [log] [blame] [edit]
/*
* Copyright 2025 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkSpan.h"
#include "include/core/SkString.h"
#include "include/core/SkTypes.h"
#include "include/private/base/SkAssert.h"
#include "include/private/base/SkTArray.h"
#include "include/private/base/SkTo.h"
#include "modules/skunicode/include/SkUnicode.h"
#include "modules/skunicode/include/SkUnicode_bidi.h"
#include "modules/skunicode/src/SkBidiFactory_icu_subset.h"
#include "modules/skunicode/src/SkUnicode_hardcoded.h"
#include "modules/skunicode/src/SkUnicode_icu_bidi.h"
#include "src/base/SkBitmaskEnum.h"
#include "src/base/SkUTF.h"
#include <algorithm>
#include <cstdint>
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include <array>
#include <unicode/ubidi.h>
#include <unicode/ubrk.h>
#include <unicode/uchar.h>
#include <unicode/uloc.h>
#include <unicode/uscript.h>
#include <unicode/ustring.h>
#include <unicode/utext.h>
#include <unicode/utypes.h>
using namespace skia_private;
class SkUnicode_bidi : public SkUnicodeHardCodedCharProperties {
public:
SkUnicode_bidi() {}
~SkUnicode_bidi() override = default;
std::unique_ptr<SkBidiIterator> makeBidiIterator(const uint16_t text[], int count,
SkBidiIterator::Direction dir) override {
return fBidiFact->MakeIterator(text, count, dir);
}
std::unique_ptr<SkBidiIterator> makeBidiIterator(const char text[],
int count,
SkBidiIterator::Direction dir) override {
SkDEBUGF("Method 'makeBidiIterator' is not implemented\n");
return nullptr;
}
std::unique_ptr<SkBreakIterator> makeBreakIterator(const char locale[],
BreakType breakType) override {
SkDEBUGF("Method 'makeBreakIterator' is not implemented\n");
return nullptr;
}
std::unique_ptr<SkBreakIterator> makeBreakIterator(BreakType breakType) override {
SkDEBUGF("Method 'makeBreakIterator' is not implemented\n");
return nullptr;
}
bool getBidiRegions(const char utf8[],
int utf8Units,
TextDirection dir,
std::vector<BidiRegion>* results) override {
return fBidiFact->ExtractBidi(utf8, utf8Units, dir, results);
}
bool getUtf8Words(const char utf8[],
int utf8Units,
const char* locale,
std::vector<Position>* results) override {
SkDEBUGF("Method 'getUtf8Words' is not implemented\n");
return false;
}
bool getSentences(const char utf8[],
int utf8Units,
const char* locale,
std::vector<SkUnicode::Position>* results) override {
SkDEBUGF("Method 'getSentences' is not implemented\n");
return false;
}
bool computeCodeUnitFlags(char utf8[],
int utf8Units,
bool replaceTabs,
TArray<SkUnicode::CodeUnitFlags, true>* results) override {
SkDEBUGF("Method 'computeCodeUnitFlags' is not implemented\n");
return false;
}
bool computeCodeUnitFlags(char16_t utf16[], int utf16Units, bool replaceTabs,
TArray<SkUnicode::CodeUnitFlags, true>* results) override {
results->clear();
results->push_back_n(utf16Units + 1, CodeUnitFlags::kNoCodeUnitFlag);
for (auto i = 0; i < utf16Units; ++i) {
auto unichar = utf16[i];
if (this->isSpace(unichar)) {
results->at(i) |= SkUnicode::kPartOfIntraWordBreak;
}
if (this->isWhitespace(unichar)) {
results->at(i) |= SkUnicode::kPartOfWhiteSpaceBreak;
}
if (this->isControl(unichar)) {
results->at(i) |= SkUnicode::kControl;
}
if (this->isIdeographic(unichar)) {
results->at(i) |= SkUnicode::kIdeographic;
}
}
return true;
}
bool getWords(const char utf8[], int utf8Units, const char* locale, std::vector<Position>* results) override {
SkDEBUGF("Method 'getWords' is not implemented\n");
return false;
}
SkString toUpper(const SkString& str) override {
SkDEBUGF("Method 'toUpper' is not implemented\n");
return SkString();
}
SkString toUpper(const SkString& str, const char* locale) override {
SkDEBUGF("Method 'toUpper' is not implemented\n");
return SkString();
}
void reorderVisual(const BidiLevel runLevels[],
int levelsCount,
int32_t logicalFromVisual[]) override {
if (levelsCount == 0) {
// To avoid an assert in unicode
return;
}
SkASSERT(runLevels != nullptr);
fBidiFact->bidi_reorderVisual(runLevels, levelsCount, logicalFromVisual);
}
private:
sk_sp<SkBidiFactory> fBidiFact = sk_make_sp<SkBidiSubsetFactory>();
};
namespace SkUnicodes::Bidi {
sk_sp<SkUnicode> Make() {
return sk_make_sp<SkUnicode_bidi>();
}
}