2 * \file GuiThesaurus.cpp
3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
7 * \author Jürgen Spitzmüller
9 * Full author contact details are available in file CREDITS.
14 #include "GuiThesaurus.h"
15 #include "GuiApplication.h"
17 #include "qt_helpers.h"
20 #include "BufferParams.h"
21 #include "BufferView.h"
22 #include "FuncRequest.h"
25 #include "WordLangTuple.h"
27 #include "support/debug.h"
28 #include "support/gettext.h"
29 #include "support/lstrings.h"
31 #include <QAbstractItemModel>
33 #include <QDialogButtonBox>
34 #include <QHeaderView>
36 #include <QPushButton>
37 #include <QTreeWidget>
38 #include <QTreeWidgetItem>
41 using namespace lyx::support;
47 GuiThesaurus::GuiThesaurus(GuiView & lv)
48 : GuiDialog(lv, "thesaurus", qt_("Thesaurus"))
52 meaningsTV->setColumnCount(1);
53 meaningsTV->header()->hide();
55 connect(buttonBox, SIGNAL(clicked(QAbstractButton *)),
56 this, SLOT(slotButtonBox(QAbstractButton *)));
57 connect(replaceED, SIGNAL(returnPressed()),
58 this, SLOT(replaceClicked()));
59 connect(replaceED, SIGNAL(textChanged(QString)),
60 this, SLOT(change_adaptor()));
61 connect(entryCO, SIGNAL(editTextChanged(const QString &)),
62 this, SLOT(entryChanged()));
63 connect(entryCO, SIGNAL(activated(int)),
64 this, SLOT(entryChanged()));
65 connect(lookupPB, SIGNAL(clicked()),
66 this, SLOT(entryChanged()));
67 connect(replacePB, SIGNAL(clicked()),
68 this, SLOT(replaceClicked()));
69 connect(languageCO, SIGNAL(activated(int)),
70 this, SLOT(entryChanged()));
71 connect(meaningsTV, SIGNAL(itemClicked(QTreeWidgetItem *, int)),
72 this, SLOT(itemClicked(QTreeWidgetItem *, int)));
73 connect(meaningsTV, SIGNAL(itemSelectionChanged()),
74 this, SLOT(selectionChanged()));
75 connect(meaningsTV, SIGNAL(itemDoubleClicked(QTreeWidgetItem *, int)),
76 this, SLOT(selectionClicked(QTreeWidgetItem *, int)));
79 QAbstractItemModel * language_model = guiApp->languageModel();
80 // FIXME: it would be nice if sorting was enabled/disabled via a checkbox.
81 language_model->sort(0);
82 languageCO->setModel(language_model);
83 languageCO->setModelColumn(2);
86 if (entryCO->completer())
87 entryCO->completer()->setCompletionMode(QCompleter::PopupCompletion);
89 bc().setCancel(buttonBox->button(QDialogButtonBox::Close));
90 bc().setApply(replacePB, true);
91 bc().setPolicy(ButtonPolicy::OkApplyCancelReadOnlyPolicy);
93 setFocusProxy(entryCO);
96 void GuiThesaurus::checkStatus()
98 if (!isBufferAvailable()) {
99 // deactivate the thesaurus if we have no buffer
106 void GuiThesaurus::change_adaptor()
112 void GuiThesaurus::entryChanged()
118 void GuiThesaurus::selectionChanged()
120 int const col = meaningsTV->currentColumn();
121 if (col < 0 || isBufferReadonly())
124 QString item = meaningsTV->currentItem()->text(col);
125 // cut out the classification in brackets:
126 // "hominid (generic term)" -> "hominid"
127 #if QT_VERSION < 0x060000
128 QRegExp re("^([^\\(\\)]+)\\b\\(?.*\\)?.*$");
130 QRegularExpression re("^([^\\(\\)]+)\\b\\(?.*\\)?.*$");
132 // This is for items with classifications at the beginning:
133 // "(noun) man" -> "man"; "(noun) male (generic term)" -> "male"
134 #if QT_VERSION < 0x060000
135 QRegExp rex("^(\\(.+\\))\\s*([^\\(\\)]+)\\s*\\(?.*\\)?.*$");
136 int pos = re.indexIn(item);
138 item = re.cap(1).trimmed();
139 pos = rex.indexIn(item);
141 item = rex.cap(2).trimmed();
143 QRegularExpression rex("^(\\(.+\\))\\s*([^\\(\\)]+)\\s*\\(?.*\\)?.*$");
144 QRegularExpressionMatch match = re.match(item);
145 if (match.hasMatch())
146 item = match.captured(1).trimmed();
147 match = rex.match(item);
148 if (match.hasMatch())
149 item = match.captured(2).trimmed();
151 replaceED->setText(item);
152 replacePB->setEnabled(!isBufferReadonly());
157 void GuiThesaurus::itemClicked(QTreeWidgetItem * /*item*/, int /*col*/)
163 void GuiThesaurus::selectionClicked(QTreeWidgetItem * item, int col)
165 QString str = item->text(col);
166 // cut out the classification in brackets:
167 // "hominid (generic term)" -> "hominid"
168 #if QT_VERSION < 0x060000
169 QRegExp re("^([^\\(\\)]+)\\b\\(?.*\\)?.*$");
171 QRegularExpression re("^([^\\(\\)]+)\\b\\(?.*\\)?.*$");
173 // This is for items with classifications at the beginning:
174 // "(noun) man" -> "man"; "(noun) male (generic term)" -> "male"
175 #if QT_VERSION < 0x060000
176 QRegExp rex("^(\\(.+\\))\\s*([^\\(\\)]+)\\s*\\(?.*\\)?.*$");
177 int pos = re.indexIn(str);
179 str = re.cap(1).trimmed();
180 pos = rex.indexIn(str);
182 str = rex.cap(2).trimmed();
184 QRegularExpression rex("^(\\(.+\\))\\s*([^\\(\\)]+)\\s*\\(?.*\\)?.*$");
185 QRegularExpressionMatch match = re.match(str);
186 if (match.hasMatch())
187 str = match.captured(1).trimmed();
188 match = rex.match(str);
189 if (match.hasMatch())
190 str = match.captured(2).trimmed();
192 entryCO->insertItem(0, str);
193 entryCO->setCurrentIndex(0);
200 void GuiThesaurus::updateLists()
204 if (entryCO->currentText().isEmpty())
207 meaningsTV->setUpdatesEnabled(false);
209 QString const lang = languageCO->itemData(
210 languageCO->currentIndex()).toString();
211 Language * language = const_cast<Language*>(lyx::languages.getLanguage(fromqstr(lang)));
212 docstring const lang_code = from_ascii(language->code());
214 Thesaurus::Meanings meanings =
215 getMeanings(WordLangTuple(qstring_to_ucs4(entryCO->currentText()), language));
217 for (auto const & meaning_p : meanings) {
218 QTreeWidgetItem * i = new QTreeWidgetItem(meaningsTV);
219 i->setText(0, toqstr(meaning_p.first));
220 meaningsTV->expandItem(i);
221 for (docstring const & word : meaning_p.second) {
222 QTreeWidgetItem * i2 = new QTreeWidgetItem(i);
223 i2->setText(0, toqstr(word));
227 meaningsTV->setEnabled(!meanings.empty());
228 lookupPB->setEnabled(!meanings.empty());
229 selectionLA->setEnabled(!meanings.empty() && !isBufferReadonly());
230 replaceED->setEnabled(!meanings.empty() && !isBufferReadonly());
231 replacePB->setEnabled(!meanings.empty() && !isBufferReadonly());
233 if (meanings.empty() && !thesaurus.thesaurusAvailable(lang_code)) {
234 QTreeWidgetItem * i = new QTreeWidgetItem(meaningsTV);
235 i->setText(0, qt_("No thesaurus available for this language!"));
238 meaningsTV->setUpdatesEnabled(true);
239 meaningsTV->update();
243 void GuiThesaurus::updateContents()
246 entryCO->addItem(toqstr(text_));
247 entryCO->setCurrentIndex(0);
248 replaceED->setText("");
249 int const pos = languageCO->findData(toqstr(lang_));
251 languageCO->setCurrentIndex(pos);
256 void GuiThesaurus::replaceClicked()
258 replace(qstring_to_ucs4(replaceED->text()));
262 bool GuiThesaurus::initialiseParams(string const & sdata)
265 string const lang = rsplit(sdata, arg, ' ');
266 if (prefixIs(lang, "lang=")) {
267 lang_ = from_utf8(split(lang, '='));
268 text_ = from_utf8(arg);
270 text_ = from_utf8(sdata);
273 bufferview()->buffer().params().language->lang());
279 void GuiThesaurus::clearParams()
286 void GuiThesaurus::replace(docstring const & newstr)
288 /* FIXME: this is not suitable ! We need to have a "lock"
289 * on a particular charpos in a paragraph that is broken on
292 docstring const sdata =
293 replace2string(newstr, text_,
294 true, // case sensitive
300 false); // only selection
301 dispatch(FuncRequest(LFUN_WORD_REPLACE, sdata));
305 Thesaurus::Meanings const & GuiThesaurus::getMeanings(WordLangTuple const & wl)
307 if (wl.word() != laststr_)
308 meanings_ = thesaurus.lookup(wl);
313 } // namespace frontend
317 #include "moc_GuiThesaurus.cpp"