/* * Copyright (C) 2016 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 "ResourceTable.h" #include "ResourceValues.h" #include "ValueVisitor.h" #include "compile/PseudolocaleGenerator.h" #include "compile/Pseudolocalizer.h" #include <algorithm> namespace aapt { std::unique_ptr<StyledString> pseudolocalizeStyledString(StyledString* string, Pseudolocalizer::Method method, StringPool* pool) { Pseudolocalizer localizer(method); const StringPiece16 originalText = *string->value->str; StyleString localized; // Copy the spans. We will update their offsets when we localize. localized.spans.reserve(string->value->spans.size()); for (const StringPool::Span& span : string->value->spans) { localized.spans.push_back(Span{ *span.name, span.firstChar, span.lastChar }); } // The ranges are all represented with a single value. This is the start of one range and // end of another. struct Range { size_t start; // Once the new string is localized, these are the pointers to the spans to adjust. // Since this struct represents the start of one range and end of another, we have // the two pointers respectively. uint32_t* updateStart; uint32_t* updateEnd; }; auto cmp = [](const Range& r, size_t index) -> bool { return r.start < index; }; // Construct the ranges. The ranges are represented like so: [0, 2, 5, 7] // The ranges are the spaces in between. In this example, with a total string length of 9, // the vector represents: (0,1], (2,4], (5,6], (7,9] // std::vector<Range> ranges; ranges.push_back(Range{ 0 }); ranges.push_back(Range{ originalText.size() - 1 }); for (size_t i = 0; i < string->value->spans.size(); i++) { const StringPool::Span& span = string->value->spans[i]; // Insert or update the Range marker for the start of this span. auto iter = std::lower_bound(ranges.begin(), ranges.end(), span.firstChar, cmp); if (iter != ranges.end() && iter->start == span.firstChar) { iter->updateStart = &localized.spans[i].firstChar; } else { ranges.insert(iter, Range{ span.firstChar, &localized.spans[i].firstChar, nullptr }); } // Insert or update the Range marker for the end of this span. iter = std::lower_bound(ranges.begin(), ranges.end(), span.lastChar, cmp); if (iter != ranges.end() && iter->start == span.lastChar) { iter->updateEnd = &localized.spans[i].lastChar; } else { ranges.insert(iter, Range{ span.lastChar, nullptr, &localized.spans[i].lastChar }); } } localized.str += localizer.start(); // Iterate over the ranges and localize each section. for (size_t i = 0; i < ranges.size(); i++) { const size_t start = ranges[i].start; size_t len = originalText.size() - start; if (i + 1 < ranges.size()) { len = ranges[i + 1].start - start; } if (ranges[i].updateStart) { *ranges[i].updateStart = localized.str.size(); } if (ranges[i].updateEnd) { *ranges[i].updateEnd = localized.str.size(); } localized.str += localizer.text(originalText.substr(start, len)); } localized.str += localizer.end(); std::unique_ptr<StyledString> localizedString = util::make_unique<StyledString>( pool->makeRef(localized)); localizedString->setSource(string->getSource()); return localizedString; } namespace { struct Visitor : public RawValueVisitor { StringPool* mPool; Pseudolocalizer::Method mMethod; Pseudolocalizer mLocalizer; // Either value or item will be populated upon visiting the value. std::unique_ptr<Value> mValue; std::unique_ptr<Item> mItem; Visitor(StringPool* pool, Pseudolocalizer::Method method) : mPool(pool), mMethod(method), mLocalizer(method) { } void visit(Plural* plural) override { std::unique_ptr<Plural> localized = util::make_unique<Plural>(); for (size_t i = 0; i < plural->values.size(); i++) { Visitor subVisitor(mPool, mMethod); if (plural->values[i]) { plural->values[i]->accept(&subVisitor); if (subVisitor.mValue) { localized->values[i] = std::move(subVisitor.mItem); } else { localized->values[i] = std::unique_ptr<Item>(plural->values[i]->clone(mPool)); } } } localized->setSource(plural->getSource()); localized->setWeak(true); mValue = std::move(localized); } void visit(String* string) override { std::u16string result = mLocalizer.start() + mLocalizer.text(*string->value) + mLocalizer.end(); std::unique_ptr<String> localized = util::make_unique<String>(mPool->makeRef(result)); localized->setSource(string->getSource()); localized->setWeak(true); mItem = std::move(localized); } void visit(StyledString* string) override { mItem = pseudolocalizeStyledString(string, mMethod, mPool); mItem->setWeak(true); } }; ConfigDescription modifyConfigForPseudoLocale(const ConfigDescription& base, Pseudolocalizer::Method m) { ConfigDescription modified = base; switch (m) { case Pseudolocalizer::Method::kAccent: modified.language[0] = 'e'; modified.language[1] = 'n'; modified.country[0] = 'X'; modified.country[1] = 'A'; break; case Pseudolocalizer::Method::kBidi: modified.language[0] = 'a'; modified.language[1] = 'r'; modified.country[0] = 'X'; modified.country[1] = 'B'; break; default: break; } return modified; } void pseudolocalizeIfNeeded(const Pseudolocalizer::Method method, ResourceConfigValue* originalValue, StringPool* pool, ResourceEntry* entry) { Visitor visitor(pool, method); originalValue->value->accept(&visitor); std::unique_ptr<Value> localizedValue; if (visitor.mValue) { localizedValue = std::move(visitor.mValue); } else if (visitor.mItem) { localizedValue = std::move(visitor.mItem); } if (!localizedValue) { return; } ConfigDescription configWithAccent = modifyConfigForPseudoLocale( originalValue->config, method); ResourceConfigValue* newConfigValue = entry->findOrCreateValue( configWithAccent, originalValue->product); if (!newConfigValue->value) { // Only use auto-generated pseudo-localization if none is defined. newConfigValue->value = std::move(localizedValue); } } /** * A value is pseudolocalizable if it does not define a locale (or is the default locale) * and is translateable. */ static bool isPseudolocalizable(ResourceConfigValue* configValue) { const int diff = configValue->config.diff(ConfigDescription::defaultConfig()); if (diff & ConfigDescription::CONFIG_LOCALE) { return false; } return configValue->value->isTranslateable(); } } // namespace bool PseudolocaleGenerator::consume(IAaptContext* context, ResourceTable* table) { for (auto& package : table->packages) { for (auto& type : package->types) { for (auto& entry : type->entries) { std::vector<ResourceConfigValue*> values = entry->findValuesIf(isPseudolocalizable); for (ResourceConfigValue* value : values) { pseudolocalizeIfNeeded(Pseudolocalizer::Method::kAccent, value, &table->stringPool, entry.get()); pseudolocalizeIfNeeded(Pseudolocalizer::Method::kBidi, value, &table->stringPool, entry.get()); } } } } return true; } } // namespace aapt