/* This file is part of Telegram Desktop, the official desktop version of Telegram messaging app, see https://telegram.org Telegram Desktop is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. In addition, as a special exception, the copyright holders give permission to link the code of portions of this program with the OpenSSL library. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once #include #include "lang_auto.h" #include "base/weak_unique_ptr.h" namespace Lang { constexpr auto kLegacyLanguageNone = -2; constexpr auto kLegacyCustomLanguage = -1; constexpr auto kLegacyDefaultLanguage = 0; constexpr str_const kLegacyLanguages[] = { "en", "it", "es", "de", "nl", "pt_BR", "ko", }; inline QString ConvertLegacyLanguageId(const QString &languageId) { return languageId.toLower().replace('_', '-'); } QString DefaultLanguageId(); class Instance; Instance &Current(); rpl::producer Viewer(LangKey key); class Instance { public: Instance() { fillDefaults(); } void switchToId(const QString &id); void switchToCustomFile(const QString &filePath); Instance(const Instance &other) = delete; Instance &operator=(const Instance &other) = delete; Instance(Instance &&other) = default; Instance &operator=(Instance &&other) = default; QString systemLangCode() const; QString cloudLangCode() const; QString id() const { return _id; } bool isCustom() const { return (_id == qstr("custom") || _id == qstr("TEST_X") || _id == qstr("TEST_0")); } int version() const { return _version; } QByteArray serialize() const; void fillFromSerialized(const QByteArray &data); void fillFromLegacy(int legacyId, const QString &legacyPath); void applyDifference(const MTPDlangPackDifference &difference); static std::map ParseStrings(const MTPVector &strings); base::Observable &updated() { return _updated; } QString getValue(LangKey key) const { Expects(key >= 0 && key < kLangKeysCount); Expects(_values.size() == kLangKeysCount); return _values[key]; } bool isNonDefaultPlural(LangKey key) const { Expects(key >= 0 && key < kLangKeysCount); Expects(_nonDefaultSet.size() == kLangKeysCount); return _nonDefaultSet[key] || _nonDefaultSet[key + 1] || _nonDefaultSet[key + 2] || _nonDefaultSet[key + 3] || _nonDefaultSet[key + 4] || _nonDefaultSet[key + 5]; } private: // SetCallback takes two QByteArrays: key, value. // It is called for all key-value pairs in string. // ResetCallback takes one QByteArray: key. template static void HandleString(const MTPLangPackString &mtpString, SetCallback setCallback, ResetCallback resetCallback); // Writes each key-value pair in the result container. template static LangKey ParseKeyValue(const QByteArray &key, const QByteArray &value, Result &result); void applyValue(const QByteArray &key, const QByteArray &value); void resetValue(const QByteArray &key); void reset(); void fillDefaults(); void fillFromCustomFile(const QString &filePath); void loadFromContent(const QByteArray &content); void loadFromCustomContent(const QString &absolutePath, const QString &relativePath, const QByteArray &content); void updatePluralRules(); QString _id; int _legacyId = kLegacyLanguageNone; QString _customFilePathAbsolute; QString _customFilePathRelative; QByteArray _customFileContent; int _version = 0; base::Observable _updated; mutable QString _systemLanguage; std::vector _values; std::vector _nonDefaultSet; std::map _nonDefaultValues; }; } // namespace Lang