615 lines
16 KiB
C++
615 lines
16 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "lang/lang_cloud_manager.h"
|
|
|
|
#include "lang/lang_instance.h"
|
|
#include "mtproto/mtp_instance.h"
|
|
#include "storage/localstorage.h"
|
|
#include "core/application.h"
|
|
#include "apiwrap.h"
|
|
#include "auth_session.h"
|
|
#include "boxes/confirm_box.h"
|
|
#include "ui/wrap/padding_wrap.h"
|
|
#include "ui/widgets/labels.h"
|
|
#include "lang/lang_file_parser.h"
|
|
#include "core/file_utilities.h"
|
|
#include "core/click_handler_types.h"
|
|
#include "styles/style_boxes.h"
|
|
|
|
namespace Lang {
|
|
namespace {
|
|
|
|
class ConfirmSwitchBox : public BoxContent {
|
|
public:
|
|
ConfirmSwitchBox(
|
|
QWidget*,
|
|
const MTPDlangPackLanguage &data,
|
|
Fn<void()> apply);
|
|
|
|
protected:
|
|
void prepare() override;
|
|
|
|
private:
|
|
QString _name;
|
|
int _percent = 0;
|
|
bool _official = false;
|
|
QString _editLink;
|
|
Fn<void()> _apply;
|
|
|
|
};
|
|
|
|
class NotReadyBox : public BoxContent {
|
|
public:
|
|
NotReadyBox(
|
|
QWidget*,
|
|
const MTPDlangPackLanguage &data);
|
|
|
|
protected:
|
|
void prepare() override;
|
|
|
|
private:
|
|
QString _name;
|
|
QString _editLink;
|
|
|
|
};
|
|
|
|
ConfirmSwitchBox::ConfirmSwitchBox(
|
|
QWidget*,
|
|
const MTPDlangPackLanguage &data,
|
|
Fn<void()> apply)
|
|
: _name(qs(data.vnative_name))
|
|
, _percent(data.vtranslated_count.v * 100 / data.vstrings_count.v)
|
|
, _official(data.is_official())
|
|
, _editLink(qs(data.vtranslations_url))
|
|
, _apply(std::move(apply)) {
|
|
}
|
|
|
|
void ConfirmSwitchBox::prepare() {
|
|
setTitle(langFactory(lng_language_switch_title));
|
|
|
|
auto link = TextWithEntities{ lang(lng_language_switch_link) };
|
|
link.entities.push_back(EntityInText(
|
|
EntityInTextCustomUrl,
|
|
0,
|
|
link.text.size(),
|
|
QString("internal:go_to_translations")));
|
|
auto name = TextWithEntities{ _name };
|
|
name.entities.push_back(EntityInText(
|
|
EntityInTextBold,
|
|
0,
|
|
name.text.size()));
|
|
auto percent = TextWithEntities{ QString::number(_percent) };
|
|
percent.entities.push_back(EntityInText(
|
|
EntityInTextBold,
|
|
0,
|
|
percent.text.size()));
|
|
const auto text = (_official
|
|
? lng_language_switch_about_official__generic<TextWithEntities>
|
|
: lng_language_switch_about_unofficial__generic<TextWithEntities>)(
|
|
lt_lang_name,
|
|
name,
|
|
lt_percent,
|
|
percent,
|
|
lt_link,
|
|
link);
|
|
auto content = Ui::CreateChild<Ui::PaddingWrap<Ui::FlatLabel>>(
|
|
this,
|
|
object_ptr<Ui::FlatLabel>(
|
|
this,
|
|
rpl::single(text),
|
|
st::boxLabel),
|
|
QMargins{ st::boxPadding.left(), 0, st::boxPadding.right(), 0 });
|
|
content->entity()->setClickHandlerFilter([=](auto&&...) {
|
|
UrlClickHandler::Open(_editLink);
|
|
return false;
|
|
});
|
|
|
|
addButton(langFactory(lng_language_switch_apply), [=] {
|
|
const auto apply = _apply;
|
|
closeBox();
|
|
apply();
|
|
});
|
|
addButton(langFactory(lng_cancel), [=] { closeBox(); });
|
|
|
|
content->resizeToWidth(st::boxWideWidth);
|
|
content->heightValue(
|
|
) | rpl::start_with_next([=](int height) {
|
|
setDimensions(st::boxWideWidth, height);
|
|
}, lifetime());
|
|
}
|
|
|
|
NotReadyBox::NotReadyBox(
|
|
QWidget*,
|
|
const MTPDlangPackLanguage &data)
|
|
: _name(qs(data.vnative_name))
|
|
, _editLink(qs(data.vtranslations_url)) {
|
|
}
|
|
|
|
void NotReadyBox::prepare() {
|
|
setTitle(langFactory(lng_language_not_ready_title));
|
|
|
|
auto link = TextWithEntities{ lang(lng_language_not_ready_link) };
|
|
link.entities.push_back(EntityInText(
|
|
EntityInTextCustomUrl,
|
|
0,
|
|
link.text.size(),
|
|
QString("internal:go_to_translations")));
|
|
auto name = TextWithEntities{ _name };
|
|
const auto text = lng_language_not_ready_about__generic(
|
|
lt_lang_name,
|
|
name,
|
|
lt_link,
|
|
link);
|
|
auto content = Ui::CreateChild<Ui::PaddingWrap<Ui::FlatLabel>>(
|
|
this,
|
|
object_ptr<Ui::FlatLabel>(
|
|
this,
|
|
rpl::single(text),
|
|
st::boxLabel),
|
|
QMargins{ st::boxPadding.left(), 0, st::boxPadding.right(), 0 });
|
|
content->entity()->setClickHandlerFilter([=](auto&&...) {
|
|
UrlClickHandler::Open(_editLink);
|
|
return false;
|
|
});
|
|
|
|
addButton(langFactory(lng_box_ok), [=] { closeBox(); });
|
|
|
|
content->resizeToWidth(st::boxWidth);
|
|
content->heightValue(
|
|
) | rpl::start_with_next([=](int height) {
|
|
setDimensions(st::boxWidth, height);
|
|
}, lifetime());
|
|
}
|
|
|
|
} // namespace
|
|
|
|
Language ParseLanguage(const MTPLangPackLanguage &data) {
|
|
return data.match([](const MTPDlangPackLanguage &data) {
|
|
return Language{
|
|
qs(data.vlang_code),
|
|
qs(data.vplural_code),
|
|
(data.has_base_lang_code()
|
|
? qs(data.vbase_lang_code)
|
|
: QString()),
|
|
qs(data.vname),
|
|
qs(data.vnative_name)
|
|
};
|
|
});
|
|
}
|
|
|
|
CloudManager::CloudManager(
|
|
Instance &langpack,
|
|
not_null<MTP::Instance*> mtproto)
|
|
: MTP::Sender()
|
|
, _langpack(langpack) {
|
|
}
|
|
|
|
Pack CloudManager::packTypeFromId(const QString &id) const {
|
|
if (id == LanguageIdOrDefault(_langpack.id())) {
|
|
return Pack::Current;
|
|
} else if (id == _langpack.baseId()) {
|
|
return Pack::Base;
|
|
}
|
|
return Pack::None;
|
|
}
|
|
|
|
void CloudManager::requestLangPackDifference(const QString &langId) {
|
|
Expects(!langId.isEmpty());
|
|
|
|
if (langId == LanguageIdOrDefault(_langpack.id())) {
|
|
requestLangPackDifference(Pack::Current);
|
|
} else {
|
|
requestLangPackDifference(Pack::Base);
|
|
}
|
|
}
|
|
|
|
mtpRequestId &CloudManager::packRequestId(Pack pack) {
|
|
return (pack != Pack::Base)
|
|
? _langPackRequestId
|
|
: _langPackBaseRequestId;
|
|
}
|
|
|
|
mtpRequestId CloudManager::packRequestId(Pack pack) const {
|
|
return (pack != Pack::Base)
|
|
? _langPackRequestId
|
|
: _langPackBaseRequestId;
|
|
}
|
|
|
|
void CloudManager::requestLangPackDifference(Pack pack) {
|
|
request(base::take(packRequestId(pack))).cancel();
|
|
if (_langpack.isCustom()) {
|
|
return;
|
|
}
|
|
|
|
const auto version = _langpack.version(pack);
|
|
const auto code = _langpack.cloudLangCode(pack);
|
|
if (code.isEmpty()) {
|
|
return;
|
|
}
|
|
if (version > 0) {
|
|
packRequestId(pack) = request(MTPlangpack_GetDifference(
|
|
MTP_string(CloudLangPackName()),
|
|
MTP_string(code),
|
|
MTP_int(version)
|
|
)).done([=](const MTPLangPackDifference &result) {
|
|
packRequestId(pack) = 0;
|
|
applyLangPackDifference(result);
|
|
}).fail([=](const RPCError &error) {
|
|
packRequestId(pack) = 0;
|
|
}).send();
|
|
} else {
|
|
packRequestId(pack) = request(MTPlangpack_GetLangPack(
|
|
MTP_string(CloudLangPackName()),
|
|
MTP_string(code)
|
|
)).done([=](const MTPLangPackDifference &result) {
|
|
packRequestId(pack) = 0;
|
|
applyLangPackDifference(result);
|
|
}).fail([=](const RPCError &error) {
|
|
packRequestId(pack) = 0;
|
|
}).send();
|
|
}
|
|
}
|
|
|
|
void CloudManager::setSuggestedLanguage(const QString &langCode) {
|
|
if (Lang::LanguageIdOrDefault(langCode) != Lang::DefaultLanguageId()) {
|
|
_suggestedLanguage = langCode;
|
|
} else {
|
|
_suggestedLanguage = QString();
|
|
}
|
|
|
|
if (!_languageWasSuggested) {
|
|
_languageWasSuggested = true;
|
|
_firstLanguageSuggestion.notify();
|
|
|
|
if (AuthSession::Exists() && _langpack.id().isEmpty() && !_suggestedLanguage.isEmpty()) {
|
|
auto isLegacy = [](const QString &languageId) {
|
|
for (auto &legacyString : kLegacyLanguages) {
|
|
auto legacyId = str_const_toString(legacyString);
|
|
if (ConvertLegacyLanguageId(legacyId) == languageId) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
// The old available languages (de/it/nl/ko/es/pt_BR) won't be
|
|
// suggested anyway, because everyone saw the suggestion in intro.
|
|
if (!isLegacy(_suggestedLanguage)) {
|
|
_offerSwitchToId = _suggestedLanguage;
|
|
offerSwitchLangPack();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CloudManager::setCurrentVersions(int version, int baseVersion) {
|
|
const auto check = [&](Pack pack, int version) {
|
|
if (version > _langpack.version(pack) && !packRequestId(pack)) {
|
|
requestLangPackDifference(pack);
|
|
}
|
|
};
|
|
check(Pack::Current, version);
|
|
check(Pack::Base, baseVersion);
|
|
}
|
|
|
|
void CloudManager::applyLangPackDifference(
|
|
const MTPLangPackDifference &difference) {
|
|
Expects(difference.type() == mtpc_langPackDifference);
|
|
|
|
if (_langpack.isCustom()) {
|
|
return;
|
|
}
|
|
|
|
const auto &langpack = difference.c_langPackDifference();
|
|
const auto langpackId = qs(langpack.vlang_code);
|
|
const auto pack = packTypeFromId(langpackId);
|
|
if (pack != Pack::None) {
|
|
applyLangPackData(pack, langpack);
|
|
if (_restartAfterSwitch) {
|
|
restartAfterSwitch();
|
|
}
|
|
} else {
|
|
LOG(("Lang Warning: "
|
|
"Ignoring update for '%1' because our language is '%2'"
|
|
).arg(langpackId
|
|
).arg(_langpack.id()));
|
|
}
|
|
}
|
|
|
|
void CloudManager::requestLanguageList() {
|
|
_languagesRequestId = request(MTPlangpack_GetLanguages(
|
|
MTP_string(CloudLangPackName())
|
|
)).done([=](const MTPVector<MTPLangPackLanguage> &result) {
|
|
auto languages = Languages();
|
|
for (const auto &language : result.v) {
|
|
languages.push_back(ParseLanguage(language));
|
|
}
|
|
if (_languages != languages) {
|
|
_languages = languages;
|
|
_languagesChanged.notify();
|
|
}
|
|
_languagesRequestId = 0;
|
|
}).fail([=](const RPCError &error) {
|
|
_languagesRequestId = 0;
|
|
}).send();
|
|
}
|
|
|
|
void CloudManager::offerSwitchLangPack() {
|
|
Expects(!_offerSwitchToId.isEmpty());
|
|
Expects(_offerSwitchToId != DefaultLanguageId());
|
|
|
|
if (!showOfferSwitchBox()) {
|
|
subscribe(languageListChanged(), [this] {
|
|
showOfferSwitchBox();
|
|
});
|
|
requestLanguageList();
|
|
}
|
|
}
|
|
|
|
Language CloudManager::findOfferedLanguage() const {
|
|
for (const auto &language : _languages) {
|
|
if (language.id == _offerSwitchToId) {
|
|
return language;
|
|
}
|
|
}
|
|
return {};
|
|
}
|
|
|
|
bool CloudManager::showOfferSwitchBox() {
|
|
const auto language = findOfferedLanguage();
|
|
if (language.id.isEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
const auto confirm = [=] {
|
|
Ui::hideLayer();
|
|
if (_offerSwitchToId.isEmpty()) {
|
|
return;
|
|
}
|
|
performSwitchAndRestart(language);
|
|
};
|
|
const auto cancel = [=] {
|
|
Ui::hideLayer();
|
|
changeIdAndReInitConnection(DefaultLanguage());
|
|
Local::writeLangPack();
|
|
};
|
|
Ui::show(
|
|
Box<ConfirmBox>(
|
|
"Do you want to switch your language to "
|
|
+ language.nativeName
|
|
+ "? You can always change your language in Settings.",
|
|
"Change",
|
|
lang(lng_cancel),
|
|
confirm,
|
|
cancel),
|
|
LayerOption::KeepOther);
|
|
return true;
|
|
}
|
|
|
|
void CloudManager::applyLangPackData(
|
|
Pack pack,
|
|
const MTPDlangPackDifference &data) {
|
|
if (_langpack.version(pack) < data.vfrom_version.v) {
|
|
requestLangPackDifference(pack);
|
|
} else if (!data.vstrings.v.isEmpty()) {
|
|
_langpack.applyDifference(pack, data);
|
|
Local::writeLangPack();
|
|
} else if (_restartAfterSwitch) {
|
|
Local::writeLangPack();
|
|
} else {
|
|
LOG(("Lang Info: Up to date."));
|
|
}
|
|
}
|
|
|
|
bool CloudManager::canApplyWithoutRestart(const QString &id) const {
|
|
if (id == qstr("#TEST_X") || id == qstr("#TEST_0")) {
|
|
return true;
|
|
}
|
|
|
|
// We don't support instant language switch if the auth session exists :(
|
|
return !AuthSession::Exists();
|
|
}
|
|
|
|
void CloudManager::resetToDefault() {
|
|
performSwitch(DefaultLanguage());
|
|
}
|
|
|
|
void CloudManager::switchToLanguage(const QString &id) {
|
|
requestLanguageAndSwitch(id, false);
|
|
}
|
|
|
|
void CloudManager::switchWithWarning(const QString &id) {
|
|
requestLanguageAndSwitch(id, true);
|
|
}
|
|
|
|
void CloudManager::requestLanguageAndSwitch(
|
|
const QString &id,
|
|
bool warning) {
|
|
Expects(!id.isEmpty());
|
|
|
|
if (LanguageIdOrDefault(_langpack.id()) == id) {
|
|
Ui::show(Box<InformBox>(lang(lng_language_already)));
|
|
return;
|
|
} else if (id == qstr("#custom")) {
|
|
performSwitchToCustom();
|
|
return;
|
|
}
|
|
|
|
request(_switchingToLanguageRequest).cancel();
|
|
_switchingToLanguageRequest = request(MTPlangpack_GetLanguage(
|
|
MTP_string(Lang::CloudLangPackName()),
|
|
MTP_string(id)
|
|
)).done([=](const MTPLangPackLanguage &result) {
|
|
_switchingToLanguageRequest = 0;
|
|
const auto language = Lang::ParseLanguage(result);
|
|
const auto finalize = [=] {
|
|
performSwitchAndRestart(language);
|
|
};
|
|
if (!warning) {
|
|
finalize();
|
|
return;
|
|
}
|
|
result.match([=](const MTPDlangPackLanguage &data) {
|
|
if (data.vstrings_count.v > 0) {
|
|
Ui::show(Box<ConfirmSwitchBox>(data, finalize));
|
|
} else {
|
|
Ui::show(Box<NotReadyBox>(data));
|
|
}
|
|
});
|
|
}).fail([=](const RPCError &error) {
|
|
_switchingToLanguageRequest = 0;
|
|
if (error.type() == "LANG_CODE_NOT_SUPPORTED") {
|
|
Ui::show(Box<InformBox>(lang(lng_language_not_found)));
|
|
}
|
|
}).send();
|
|
}
|
|
|
|
void CloudManager::switchToLanguage(const Language &data) {
|
|
if (_langpack.id() == data.id && data.id != qstr("#custom")) {
|
|
return;
|
|
}
|
|
|
|
request(_switchingToLanguageRequest).cancel();
|
|
if (data.id == qstr("#custom")) {
|
|
performSwitchToCustom();
|
|
} else if (canApplyWithoutRestart(data.id)) {
|
|
performSwitch(data);
|
|
} else {
|
|
QVector<MTPstring> keys;
|
|
keys.reserve(3);
|
|
keys.push_back(MTP_string("lng_sure_save_language"));
|
|
_switchingToLanguageRequest = request(MTPlangpack_GetStrings(
|
|
MTP_string(Lang::CloudLangPackName()),
|
|
MTP_string(data.id),
|
|
MTP_vector<MTPstring>(std::move(keys))
|
|
)).done([=](const MTPVector<MTPLangPackString> &result) {
|
|
_switchingToLanguageRequest = 0;
|
|
const auto values = Instance::ParseStrings(result);
|
|
const auto getValue = [&](LangKey key) {
|
|
auto it = values.find(key);
|
|
return (it == values.cend())
|
|
? GetOriginalValue(key)
|
|
: it->second;
|
|
};
|
|
const auto text = lang(lng_sure_save_language)
|
|
+ "\n\n"
|
|
+ getValue(lng_sure_save_language);
|
|
Ui::show(
|
|
Box<ConfirmBox>(
|
|
text,
|
|
lang(lng_box_ok),
|
|
lang(lng_cancel),
|
|
[=] { performSwitchAndRestart(data); }),
|
|
LayerOption::KeepOther);
|
|
}).fail([=](const RPCError &error) {
|
|
_switchingToLanguageRequest = 0;
|
|
}).send();
|
|
}
|
|
}
|
|
|
|
void CloudManager::performSwitchToCustom() {
|
|
auto filter = qsl("Language files (*.strings)");
|
|
auto title = qsl("Choose language .strings file");
|
|
FileDialog::GetOpenPath(Core::App().getFileDialogParent(), title, filter, [weak = base::make_weak(this)](const FileDialog::OpenResult &result) {
|
|
if (!weak || result.paths.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
auto filePath = result.paths.front();
|
|
Lang::FileParser loader(filePath, { lng_sure_save_language });
|
|
if (loader.errors().isEmpty()) {
|
|
weak->request(weak->_switchingToLanguageRequest).cancel();
|
|
if (weak->canApplyWithoutRestart(qsl("#custom"))) {
|
|
weak->_langpack.switchToCustomFile(filePath);
|
|
} else {
|
|
const auto values = loader.found();
|
|
const auto getValue = [&](LangKey key) {
|
|
const auto it = values.find(key);
|
|
return (it == values.cend())
|
|
? GetOriginalValue(key)
|
|
: it.value();
|
|
};
|
|
const auto text = lang(lng_sure_save_language)
|
|
+ "\n\n"
|
|
+ getValue(lng_sure_save_language);
|
|
const auto change = [=] {
|
|
weak->_langpack.switchToCustomFile(filePath);
|
|
App::restart();
|
|
};
|
|
Ui::show(
|
|
Box<ConfirmBox>(
|
|
text,
|
|
lang(lng_box_ok),
|
|
lang(lng_cancel),
|
|
change),
|
|
LayerOption::KeepOther);
|
|
}
|
|
} else {
|
|
Ui::show(
|
|
Box<InformBox>("Custom lang failed :(\n\nError: " + loader.errors()),
|
|
LayerOption::KeepOther);
|
|
}
|
|
});
|
|
}
|
|
|
|
void CloudManager::switchToTestLanguage() {
|
|
const auto testLanguageId = (_langpack.id() == qstr("#TEST_X"))
|
|
? qsl("#TEST_0")
|
|
: qsl("#TEST_X");
|
|
performSwitch({ testLanguageId });
|
|
}
|
|
|
|
void CloudManager::performSwitch(const Language &data) {
|
|
_restartAfterSwitch = false;
|
|
switchLangPackId(data);
|
|
requestLangPackDifference(Pack::Current);
|
|
requestLangPackDifference(Pack::Base);
|
|
}
|
|
|
|
void CloudManager::performSwitchAndRestart(const Language &data) {
|
|
Local::pushRecentLanguage(data);
|
|
performSwitch(data);
|
|
restartAfterSwitch();
|
|
}
|
|
|
|
void CloudManager::restartAfterSwitch() {
|
|
if (_langPackRequestId || _langPackBaseRequestId) {
|
|
_restartAfterSwitch = true;
|
|
} else {
|
|
App::restart();
|
|
}
|
|
}
|
|
|
|
void CloudManager::switchLangPackId(const Language &data) {
|
|
const auto currentId = _langpack.id();
|
|
const auto currentBaseId = _langpack.baseId();
|
|
const auto notChanged = (currentId == data.id
|
|
&& currentBaseId == data.baseId)
|
|
|| (currentId.isEmpty()
|
|
&& currentBaseId.isEmpty()
|
|
&& data.id == DefaultLanguageId());
|
|
if (!notChanged) {
|
|
changeIdAndReInitConnection(data);
|
|
}
|
|
}
|
|
|
|
void CloudManager::changeIdAndReInitConnection(const Language &data) {
|
|
_langpack.switchToId(data);
|
|
|
|
auto mtproto = requestMTP();
|
|
mtproto->reInitConnection(mtproto->mainDcId());
|
|
}
|
|
|
|
CloudManager &CurrentCloudManager() {
|
|
auto result = Core::App().langCloudManager();
|
|
Assert(result != nullptr);
|
|
return *result;
|
|
}
|
|
|
|
} // namespace Lang
|