KTextAddons

translatorwidget.cpp
1/*
2
3 SPDX-FileCopyrightText: 2012-2025 Laurent Montel <montel@kde.org>
4
5 SPDX-License-Identifier: GPL-2.0-or-later
6*/
7
8#include "translatorwidget.h"
9
10#include "texttranslator_debug.h"
11#include "translator/misc/translatorutil.h"
12#include "translator/networkmanager.h"
13#include "translator/translatorengineclient.h"
14#include "translator/translatorengineloader.h"
15#include "translator/translatorengineplugin.h"
16#include "translatorconfiguredialog.h"
17#include "translatordebugdialog.h"
18#include <KBusyIndicatorWidget>
19
20#include <KConfigGroup>
21#include <KLocalizedString>
22#include <KMessageBox>
23#include <KSeparator>
24#include <QPushButton>
25
26#include <KSharedConfig>
27#include <QComboBox>
28#include <QHBoxLayout>
29#include <QIcon>
30#include <QKeyEvent>
31#include <QLabel>
32#include <QMimeData>
33#include <QShortcut>
34#include <QSplitter>
35#include <QToolButton>
36#include <QVBoxLayout>
37
38using namespace Qt::Literals::StringLiterals;
39using namespace TextTranslator;
40namespace
41{
42static const char myTranslatorWidgetConfigGroupName[] = "TranslatorWidget";
43}
44class Q_DECL_HIDDEN TranslatorWidget::TranslatorWidgetPrivate
45{
46public:
47 TranslatorWidgetPrivate() = default;
48
49 ~TranslatorWidgetPrivate()
50 {
51 delete translatorPlugin;
52 }
53
54 void initLanguage();
55 void fillToCombobox(const QString &lang);
56
57 QByteArray data;
58 TranslatorTextEdit *inputText = nullptr;
59 QPlainTextEdit *translatorResultTextEdit = nullptr;
60 QComboBox *fromCombobox = nullptr;
61 QComboBox *toCombobox = nullptr;
62 QPushButton *translate = nullptr;
63 QPushButton *clear = nullptr;
64 QToolButton *closeBtn = nullptr;
65 QLabel *engineNameLabel = nullptr;
66 TextTranslator::TranslatorEngineClient *translatorClient = nullptr;
67 TextTranslator::TranslatorEnginePlugin *translatorPlugin = nullptr;
68 KBusyIndicatorWidget *progressIndicator = nullptr;
69 QPushButton *invert = nullptr;
70 QSplitter *splitter = nullptr;
71 QString engineName;
72 bool languageSettingsChanged = false;
73 bool standalone = true;
74};
75
76void TranslatorWidget::TranslatorWidgetPrivate::fillToCombobox(const QString &lang)
77{
78 toCombobox->clear();
79
80 TranslatorUtil translatorUtil;
81 const QMapIterator<TranslatorUtil::Language, QString> listToLanguage = translatorClient->supportedToLanguages();
82 QMapIterator<TranslatorUtil::Language, QString> i(listToLanguage);
83 while (i.hasNext()) {
84 i.next();
85 const QString languageCode = TranslatorUtil::languageCode(i.key());
86 if ((i.key() != TranslatorUtil::automatic) && languageCode != lang) {
87 translatorUtil.addItemToFromComboBox(toCombobox, languageCode, i.value());
88 }
89 }
90}
91
92void TranslatorWidget::TranslatorWidgetPrivate::initLanguage()
93{
94 if (!translatorClient) {
95 return;
96 }
97 toCombobox->clear();
98 fromCombobox->clear();
99 const QMapIterator<TranslatorUtil::Language, QString> listFromLanguage = translatorClient->supportedFromLanguages();
100
101 QMapIterator<TranslatorUtil::Language, QString> i(listFromLanguage);
102 TranslatorUtil translatorUtil;
103 while (i.hasNext()) {
104 i.next();
105 const QString languageCode = TranslatorUtil::languageCode(i.key());
106 translatorUtil.addItemToFromComboBox(fromCombobox, languageCode, i.value());
107 }
108}
109
110TranslatorTextEdit::TranslatorTextEdit(QWidget *parent)
111 : QPlainTextEdit(parent)
112{
113}
114
115void TranslatorTextEdit::dropEvent(QDropEvent *event)
116{
117 if (event->source() != this) {
118 if (event->mimeData()->hasText()) {
119 QTextCursor cursor = textCursor();
120 cursor.beginEditBlock();
121 cursor.insertText(event->mimeData()->text());
122 cursor.endEditBlock();
123 event->setDropAction(Qt::CopyAction);
124 event->accept();
125 Q_EMIT translateText();
126 return;
127 }
128 }
130}
131
132TranslatorWidget::TranslatorWidget(QWidget *parent)
133 : QWidget(parent)
134 , d(new TranslatorWidgetPrivate)
135{
136 init();
137}
138
139TranslatorWidget::TranslatorWidget(const QString &text, QWidget *parent)
140 : QWidget(parent)
141 , d(new TranslatorWidgetPrivate)
142{
143 init();
144 d->inputText->setPlainText(text);
145}
146
147TranslatorWidget::~TranslatorWidget()
148{
149 disconnect(d->inputText, &TranslatorTextEdit::textChanged, this, &TranslatorWidget::slotTextChanged);
150 disconnect(d->inputText, &TranslatorTextEdit::translateText, this, &TranslatorWidget::slotTranslate);
151 writeConfig();
152}
153
154void TranslatorWidget::writeConfig()
155{
156 if (d->languageSettingsChanged) {
157 KConfigGroup myGroup(KSharedConfig::openConfig(), QStringLiteral("General"));
158 myGroup.writeEntry(QStringLiteral("FromLanguage"), d->fromCombobox->itemData(d->fromCombobox->currentIndex()).toString());
159 myGroup.writeEntry("ToLanguage", d->toCombobox->itemData(d->toCombobox->currentIndex()).toString());
160 myGroup.sync();
161 }
162 KConfigGroup myGroupUi(KSharedConfig::openStateConfig(), QLatin1StringView(myTranslatorWidgetConfigGroupName));
163 myGroupUi.writeEntry("mainSplitter", d->splitter->sizes());
164 myGroupUi.sync();
165}
166
167void TranslatorWidget::readConfig()
168{
169 KConfigGroup myGroupUi(KSharedConfig::openStateConfig(), QLatin1StringView(myTranslatorWidgetConfigGroupName));
170 const QList<int> size = {100, 100};
171 d->splitter->setSizes(myGroupUi.readEntry("mainSplitter", size));
172
173 KConfigGroup myGroup(KSharedConfig::openConfig(), QStringLiteral("General"));
174 const QString from = myGroup.readEntry(QStringLiteral("FromLanguage"));
175 if (from.isEmpty()) {
176 return;
177 }
178 const int indexFrom = d->fromCombobox->findData(from);
179 if (indexFrom != -1) {
180 d->fromCombobox->setCurrentIndex(indexFrom);
181 }
182 d->translatorClient->generateToListFromCurrentToLanguage(from);
183 // Update "to" combobox
184 d->toCombobox->blockSignals(true);
185 d->fillToCombobox(from);
186 d->toCombobox->blockSignals(false);
187
188 const QString to = myGroup.readEntry(QStringLiteral("ToLanguage"));
189 const int indexTo = d->toCombobox->findData(to);
190 if (indexTo != -1) {
191 d->toCombobox->setCurrentIndex(indexTo);
192 }
193 d->invert->setEnabled(from != "auto"_L1);
194}
195
196void TranslatorWidget::loadEngineSettings()
197{
198 d->engineName = TranslatorUtil::loadEngine();
199 // TODO fallback if name is empty ?
200 switchEngine();
201}
202
203void TranslatorWidget::init()
204{
205 auto layout = new QVBoxLayout(this);
206 layout->setSpacing(0);
207 layout->setContentsMargins({});
208
209 auto hboxLayout = new QHBoxLayout;
210 hboxLayout->setSpacing(style()->pixelMetric(QStyle::PM_LayoutHorizontalSpacing));
211 hboxLayout->setContentsMargins(style()->pixelMetric(QStyle::PM_LayoutLeftMargin),
212 style()->pixelMetric(QStyle::PM_LayoutTopMargin),
213 style()->pixelMetric(QStyle::PM_LayoutRightMargin),
214 style()->pixelMetric(QStyle::PM_LayoutBottomMargin));
215 d->closeBtn = new QToolButton(this);
216 d->closeBtn->setObjectName(QStringLiteral("close-button"));
217 d->closeBtn->setIcon(QIcon::fromTheme(QStringLiteral("dialog-close")));
218 d->closeBtn->setIconSize(QSize(16, 16));
219 d->closeBtn->setToolTip(i18nc("@info:tooltip", "Close"));
220
221#ifndef QT_NO_ACCESSIBILITY
222 d->closeBtn->setAccessibleName(i18n("Close"));
223#endif
224 d->closeBtn->setAutoRaise(true);
225 hboxLayout->addWidget(d->closeBtn);
226 connect(d->closeBtn, &QToolButton::clicked, this, &TranslatorWidget::slotCloseWidget);
227
228 auto label = new QLabel(i18nc("Translate from language", "From:"), this);
229 label->setTextFormat(Qt::PlainText);
230 hboxLayout->addWidget(label);
231 d->fromCombobox = new QComboBox(this);
232 d->fromCombobox->setMinimumWidth(50);
233 d->fromCombobox->setObjectName(QStringLiteral("from"));
234 hboxLayout->addWidget(d->fromCombobox);
235
236 label = new QLabel(i18nc("Translate to language", "To:"), this);
237 label->setTextFormat(Qt::PlainText);
238 hboxLayout->addWidget(label);
239 d->toCombobox = new QComboBox(this);
240 d->toCombobox->setMinimumWidth(50);
241 d->toCombobox->setObjectName(QStringLiteral("to"));
242
243 hboxLayout->addWidget(d->toCombobox);
244
245 auto separator = new KSeparator(this);
246 separator->setOrientation(Qt::Vertical);
247 hboxLayout->addWidget(separator);
248
249 d->invert = new QPushButton(i18nc("Invert language choices so that from becomes to and to becomes from", "Invert"), this);
250 d->invert->setObjectName(QStringLiteral("invert-button"));
251 connect(d->invert, &QPushButton::clicked, this, &TranslatorWidget::slotInvertLanguage);
252 hboxLayout->addWidget(d->invert);
253
254 d->clear = new QPushButton(i18nc("@action:button", "Clear"), this);
255 d->clear->setObjectName(QStringLiteral("clear-button"));
256#ifndef QT_NO_ACCESSIBILITY
257 d->clear->setAccessibleName(i18n("Clear"));
258#endif
259 connect(d->clear, &QPushButton::clicked, this, &TranslatorWidget::slotClear);
260 hboxLayout->addWidget(d->clear);
261
262 d->translate = new QPushButton(i18nc("@action:button", "Translate"), this);
263 d->translate->setObjectName(QStringLiteral("translate-button"));
264#ifndef QT_NO_ACCESSIBILITY
265 d->translate->setAccessibleName(i18n("Translate"));
266#endif
267
268 hboxLayout->addWidget(d->translate);
269 connect(d->translate, &QPushButton::clicked, this, &TranslatorWidget::slotTranslate);
270
271 if (!qEnvironmentVariableIsEmpty("TRANSLATING_DEBUGGING")) {
272 auto debugButton = new QPushButton(i18nc("@action:button", "Debug"), this);
273 hboxLayout->addWidget(debugButton);
274 connect(debugButton, &QPushButton::clicked, this, &TranslatorWidget::slotDebug);
275 }
276
277 d->progressIndicator = new KBusyIndicatorWidget(this);
278 hboxLayout->addWidget(d->progressIndicator);
279 d->progressIndicator->setFixedHeight(d->toCombobox->height());
280
281 hboxLayout->addStretch();
282
283 d->engineNameLabel = new QLabel(this);
284 hboxLayout->addWidget(d->engineNameLabel);
285
286 auto configureButton = new QToolButton(this);
287 configureButton->setObjectName(QStringLiteral("configure_button"));
288 configureButton->setIcon(QIcon::fromTheme(QStringLiteral("configure")));
289 configureButton->setIconSize(QSize(16, 16));
290 configureButton->setToolTip(i18nc("@info:tooltip", "Configure"));
291 connect(configureButton, &QToolButton::clicked, this, [this]() {
292 TranslatorConfigureDialog dlg(this);
293 if (dlg.exec()) {
294 loadEngineSettings();
295 }
296 });
297 hboxLayout->addWidget(configureButton);
298
299 layout->addLayout(hboxLayout);
300
301 separator = new KSeparator(this);
302 separator->setOrientation(Qt::Horizontal);
303 layout->addWidget(separator);
304
305 d->splitter = new QSplitter;
306 d->splitter->setChildrenCollapsible(false);
307 d->inputText = new TranslatorTextEdit(this);
308 d->inputText->setObjectName(QStringLiteral("inputtext"));
309
310 connect(d->inputText, &TranslatorTextEdit::textChanged, this, &TranslatorWidget::slotTextChanged);
311 connect(d->inputText, &TranslatorTextEdit::translateText, this, &TranslatorWidget::slotTranslate);
312
313 d->splitter->addWidget(d->inputText);
314 d->translatorResultTextEdit = new QPlainTextEdit(this);
315 d->translatorResultTextEdit->setObjectName(QStringLiteral("translatedtext"));
316 d->translatorResultTextEdit->setReadOnly(true);
317 d->splitter->addWidget(d->translatorResultTextEdit);
318
319 layout->addWidget(d->splitter);
320
321 d->fromCombobox->setCurrentIndex(0); // Fill "to" combobox
322 loadEngineSettings();
323 switchEngine();
324 slotFromLanguageChanged(0, true);
325 slotTextChanged();
326 readConfig();
327 connect(d->fromCombobox, &QComboBox::currentIndexChanged, this, [this](int val) {
328 slotFromLanguageChanged(val, false);
329 slotConfigChanged();
330 });
331 connect(d->toCombobox, &QComboBox::currentIndexChanged, this, [this]() {
332 slotConfigChanged();
333 slotTranslate();
334 });
335
336 hide();
338 d->languageSettingsChanged = false;
339}
340
341void TranslatorWidget::switchEngine()
342{
343 if (d->translatorPlugin) {
344 disconnect(d->translatorPlugin);
345 delete d->translatorPlugin;
346 d->translatorPlugin = nullptr;
347 }
348 d->translatorClient = TextTranslator::TranslatorEngineLoader::self()->createTranslatorClient(d->engineName);
349 if (!d->translatorClient) {
350 const QString fallBackEngineName = TextTranslator::TranslatorEngineLoader::self()->fallbackFirstEngine();
351 if (!fallBackEngineName.isEmpty()) {
352 d->translatorClient = TextTranslator::TranslatorEngineLoader::self()->createTranslatorClient(fallBackEngineName);
353 }
354 }
355 if (d->translatorClient) {
356 d->translatorPlugin = d->translatorClient->createTranslator();
357 connect(d->translatorPlugin, &TextTranslator::TranslatorEnginePlugin::translateDone, this, &TranslatorWidget::slotTranslateDone);
358 connect(d->translatorPlugin, &TextTranslator::TranslatorEnginePlugin::translateFailed, this, &TranslatorWidget::slotTranslateFailed);
359 d->initLanguage();
360 d->engineNameLabel->setText(QStringLiteral("[%1]").arg(d->translatorClient->translatedName()));
361 d->invert->setVisible(d->translatorClient->hasInvertSupport());
362 updatePlaceHolder();
363 }
364}
365
366void TranslatorWidget::updatePlaceHolder()
367{
368 if (d->translatorClient->engineType() == TextTranslator::TranslatorEngineClient::Network) {
369 d->inputText->setPlaceholderText(
370 i18nc("@info:placeholder", "Drag text that you want to translate. (Be careful text will be send to external Server)."));
371 } else {
372 d->inputText->setPlaceholderText(i18nc("@info:placeholder", "Drag text that you want to translate."));
373 }
374}
375
376void TranslatorWidget::slotConfigChanged()
377{
378 d->languageSettingsChanged = true;
379}
380
381void TranslatorWidget::slotTextChanged()
382{
383 d->translate->setEnabled(!d->inputText->document()->isEmpty());
384 d->clear->setEnabled(!d->inputText->document()->isEmpty());
385}
386
387void TranslatorWidget::slotFromLanguageChanged(int index, bool initialize)
388{
389 const QString lang = d->fromCombobox->itemData(index).toString();
390 d->invert->setEnabled(lang != "auto"_L1);
391 const QString to = d->toCombobox->itemData(d->toCombobox->currentIndex()).toString();
392
393 // Get "from" language code for generating "to" language list
394 // qDebug() << " d->fromCombobox->currentIndex() " << lang;
395 d->translatorClient->generateToListFromCurrentToLanguage(lang);
396 d->toCombobox->blockSignals(true);
397 d->fillToCombobox(lang);
398 d->toCombobox->blockSignals(false);
399 const int indexTo = d->toCombobox->findData(to);
400 if (indexTo != -1) {
401 d->toCombobox->setCurrentIndex(indexTo);
402 }
403 if (!initialize) {
404 slotTranslate();
405 }
406}
407
408void TranslatorWidget::setTextToTranslate(const QString &text)
409{
410 d->inputText->setPlainText(text);
411 slotTranslate();
412}
413
414void TranslatorWidget::slotTranslate()
415{
416 if (!d->translatorPlugin) {
417 qCWarning(TEXTTRANSLATOR_LOG) << " Translator plugin invalid";
418 return;
419 }
420 if (!TextTranslator::NetworkManager::self()->isOnline()) {
421 KMessageBox::information(this, i18n("No network connection detected, we cannot translate text."), i18nc("@title:window", "No network"));
422 return;
423 }
424 const QString textToTranslate = d->inputText->toPlainText();
425 if (textToTranslate.trimmed().isEmpty()) {
426 return;
427 }
428
429 d->translatorResultTextEdit->clear();
430
431 const QString from = d->fromCombobox->itemData(d->fromCombobox->currentIndex()).toString();
432 const QString to = d->toCombobox->itemData(d->toCombobox->currentIndex()).toString();
433 d->translate->setEnabled(false);
434 d->progressIndicator->show();
435
436 const QString inputText{d->inputText->toPlainText()};
437 if (!inputText.isEmpty() && !from.isEmpty() && !to.isEmpty()) {
438 d->translatorPlugin->setFrom(from);
439 d->translatorPlugin->setTo(to);
440 d->translatorPlugin->setInputText(inputText);
441 d->translatorPlugin->translate();
442 }
443}
444
445void TranslatorWidget::slotTranslateDone()
446{
447 d->translate->setEnabled(true);
448 d->progressIndicator->hide();
449 d->translatorResultTextEdit->setPlainText(d->translatorPlugin->resultTranslate());
450}
451
452void TranslatorWidget::slotTranslateFailed(const QString &message)
453{
454 d->translate->setEnabled(true);
455 d->progressIndicator->hide();
456 d->translatorResultTextEdit->clear();
457 if (!message.isEmpty()) {
458 KMessageBox::error(this, message, i18nc("@title:window", "Translate error"));
459 }
460}
461
462void TranslatorWidget::slotInvertLanguage()
463{
464 const QString fromLanguage = d->fromCombobox->itemData(d->fromCombobox->currentIndex()).toString();
465 // don't invert when fromLanguage == auto
466 if (fromLanguage == "auto"_L1) {
467 return;
468 }
469
470 const QString toLanguage = d->toCombobox->itemData(d->toCombobox->currentIndex()).toString();
471 const int indexFrom = d->fromCombobox->findData(toLanguage);
472 if (indexFrom != -1) {
473 d->fromCombobox->setCurrentIndex(indexFrom);
474 }
475 const int indexTo = d->toCombobox->findData(fromLanguage);
476 if (indexTo != -1) {
477 d->toCombobox->setCurrentIndex(indexTo);
478 }
479 slotTranslate();
480}
481
482void TranslatorWidget::setStandalone(bool b)
483{
484 d->standalone = b;
485 d->closeBtn->setVisible(b);
486}
487
488void TranslatorWidget::slotCloseWidget()
489{
490 if (isHidden()) {
491 return;
492 }
493 d->inputText->clear();
494 d->translatorResultTextEdit->clear();
495 d->progressIndicator->hide();
496 if (d->standalone) {
497 hide();
498 }
499 Q_EMIT toolsWasClosed();
500}
501
502bool TranslatorWidget::event(QEvent *e)
503{
504 // Close the bar when pressing Escape.
505 // Not using a QShortcut for this because it could conflict with
506 // window-global actions (e.g. Emil Sedgh binds Esc to "close tab").
507 // With a shortcut override we can catch this before it gets to kactions.
508 if (e->type() == QEvent::ShortcutOverride || e->type() == QEvent::KeyPress) {
509 auto kev = static_cast<QKeyEvent *>(e);
510 if (kev->key() == Qt::Key_Escape) {
511 e->accept();
512 slotCloseWidget();
513 return true;
514 }
515 }
516 return QWidget::event(e);
517}
518
519void TranslatorWidget::slotClear()
520{
521 d->inputText->clear();
522 d->translatorResultTextEdit->clear();
523 d->translate->setEnabled(false);
524 if (d->translatorPlugin) {
525 d->translatorPlugin->clear();
526 }
527}
528
529void TranslatorWidget::slotDebug()
530{
531 if (d->translatorPlugin) {
532 TranslatorDebugDialog dlg(this);
533 dlg.setDebug(d->translatorPlugin->jsonDebug());
534 dlg.exec();
535 } else {
536 qCWarning(TEXTTRANSLATOR_LOG) << " Translator plugin invalid";
537 }
538}
539
540#include "moc_translatorwidget.cpp"
static KSharedConfig::Ptr openConfig(const QString &fileName=QString(), OpenFlags mode=FullConfig, QStandardPaths::StandardLocation type=QStandardPaths::GenericConfigLocation)
static KSharedConfig::Ptr openStateConfig(const QString &fileName=QString())
The TranslatorWidget class.
QString i18nc(const char *context, const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
void information(QWidget *parent, const QString &text, const QString &title=QString(), const QString &dontShowAgainName=QString(), Options options=Notify)
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
QAction * clear(const QObject *recvr, const char *slot, QObject *parent)
QString label(StandardShortcut id)
QCA_EXPORT void init()
void clicked(bool checked)
virtual bool event(QEvent *event) override
void currentIndexChanged(int index)
ShortcutOverride
void accept()
Type type() const const
QIcon fromTheme(const QString &name)
Q_EMITQ_EMIT
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
bool disconnect(const QMetaObject::Connection &connection)
QObject * parent() const const
virtual void dropEvent(QDropEvent *e) override
void textChanged()
QTextCursor textCursor() const const
void clear()
bool isEmpty() const const
QString trimmed() const const
PM_LayoutHorizontalSpacing
CopyAction
Key_Escape
Vertical
PlainText
QWidget(QWidget *parent, Qt::WindowFlags f)
virtual bool event(QEvent *event) override
void hide()
bool isHidden() const const
QLayout * layout() const const
void setSizePolicy(QSizePolicy)
QStyle * style() const const
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 11:55:35 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.