mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-16 21:10:26 +00:00
c293be56bd
In particular, the directory frontends/qt4 is renamed to frontends/qt. Many configurations file have to be updated. All mentions of qt4 in the source have been audited, and changed to qt if necessary. The only part that has not been updated is the CMake build system.
630 lines
15 KiB
C++
630 lines
15 KiB
C++
/**
|
|
* \file GuiSpellchecker.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author John Levon
|
|
* \author Edwin Leuven
|
|
* \author Abdelrazak Younes
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "GuiSpellchecker.h"
|
|
#include "GuiApplication.h"
|
|
|
|
#include "qt_helpers.h"
|
|
|
|
#include "ui_SpellcheckerUi.h"
|
|
|
|
#include "Buffer.h"
|
|
#include "BufferParams.h"
|
|
#include "BufferView.h"
|
|
#include "buffer_funcs.h"
|
|
#include "Cursor.h"
|
|
#include "Text.h"
|
|
#include "CutAndPaste.h"
|
|
#include "FuncRequest.h"
|
|
#include "Language.h"
|
|
#include "LyX.h"
|
|
#include "LyXRC.h"
|
|
#include "lyxfind.h"
|
|
#include "Paragraph.h"
|
|
#include "WordLangTuple.h"
|
|
|
|
#include "support/debug.h"
|
|
#include "support/docstring.h"
|
|
#include "support/docstring_list.h"
|
|
#include "support/ExceptionMessage.h"
|
|
#include "support/gettext.h"
|
|
#include "support/lstrings.h"
|
|
#include "support/textutils.h"
|
|
|
|
#include <QKeyEvent>
|
|
#include <QListWidgetItem>
|
|
#include <QMessageBox>
|
|
|
|
#include "SpellChecker.h"
|
|
|
|
#include "frontends/alert.h"
|
|
|
|
using namespace std;
|
|
using namespace lyx::support;
|
|
|
|
namespace lyx {
|
|
namespace frontend {
|
|
|
|
|
|
struct SpellcheckerWidget::Private
|
|
{
|
|
Private(SpellcheckerWidget * parent, DockView * dv, GuiView * gv)
|
|
: p(parent), dv_(dv), gv_(gv), incheck_(false), wrap_around_(false) {}
|
|
/// update from controller
|
|
void updateSuggestions(docstring_list & words);
|
|
/// move to next position after current word
|
|
void forward();
|
|
/// check text until next misspelled/unknown word
|
|
void check();
|
|
/// close the spell checker dialog
|
|
void hide() const;
|
|
/// make/restore a selection between from and to
|
|
void setSelection(DocIterator const & from, DocIterator const & to) const;
|
|
/// if no selection was checked:
|
|
/// ask the user if the check should start over
|
|
bool continueFromBeginning();
|
|
/// set the given language in language chooser
|
|
void setLanguage(Language const * lang);
|
|
/// test and set guard flag
|
|
bool inCheck() {
|
|
if (incheck_)
|
|
return true;
|
|
incheck_ = true;
|
|
return false;
|
|
}
|
|
void canCheck() { incheck_ = false; }
|
|
/// check for wrap around
|
|
void wrapAround(bool flag) {
|
|
wrap_around_ = flag;
|
|
if (flag) {
|
|
end_ = start_;
|
|
}
|
|
}
|
|
/// test for existing association with a document buffer
|
|
/// and test for already active check
|
|
bool disabled() {
|
|
return gv_->documentBufferView() == 0 || inCheck();
|
|
}
|
|
/// the cursor position of the buffer view
|
|
DocIterator const cursor() const;
|
|
/// status checks
|
|
bool isCurrentBuffer(DocIterator const & cursor) const;
|
|
bool isWrapAround(DocIterator const & cursor) const;
|
|
bool isWrapAround() const { return wrap_around_; }
|
|
bool atLastPos(DocIterator const & cursor) const;
|
|
/// validate the cached doc iterators
|
|
/// The spell checker dialog is not modal.
|
|
/// The user may change the buffer being checked and break the iterators.
|
|
void fixPositionsIfBroken();
|
|
///
|
|
Ui::SpellcheckerUi ui;
|
|
///
|
|
SpellcheckerWidget * p;
|
|
///
|
|
DockView * dv_;
|
|
///
|
|
GuiView * gv_;
|
|
/// current word being checked and lang code
|
|
WordLangTuple word_;
|
|
/// cursor position where spell checking starts
|
|
DocIterator start_;
|
|
/// range to spell check
|
|
/// for selection both are non-empty
|
|
/// after wrap around the start position becomes the end
|
|
DocIterator begin_;
|
|
DocIterator end_;
|
|
///
|
|
bool incheck_;
|
|
///
|
|
bool wrap_around_;
|
|
};
|
|
|
|
|
|
SpellcheckerWidget::SpellcheckerWidget(GuiView * gv, DockView * dv, QWidget * parent)
|
|
: QTabWidget(parent), d(new Private(this, dv, gv))
|
|
{
|
|
d->ui.setupUi(this);
|
|
|
|
connect(d->ui.suggestionsLW, SIGNAL(itemDoubleClicked(QListWidgetItem*)),
|
|
this, SLOT(on_replacePB_clicked()));
|
|
|
|
// language
|
|
QAbstractItemModel * language_model = guiApp->languageModel();
|
|
// FIXME: it would be nice if sorting was enabled/disabled via a checkbox.
|
|
language_model->sort(0);
|
|
d->ui.languageCO->setModel(language_model);
|
|
d->ui.languageCO->setModelColumn(1);
|
|
|
|
d->ui.wordED->setReadOnly(true);
|
|
|
|
d->ui.suggestionsLW->installEventFilter(this);
|
|
}
|
|
|
|
|
|
SpellcheckerWidget::~SpellcheckerWidget()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
|
|
bool SpellcheckerWidget::eventFilter(QObject *obj, QEvent *event)
|
|
{
|
|
if (obj == d->ui.suggestionsLW && event->type() == QEvent::KeyPress) {
|
|
QKeyEvent *e = static_cast<QKeyEvent *> (event);
|
|
if (e->key() == Qt::Key_Enter || e->key() == Qt::Key_Return) {
|
|
if (d->ui.suggestionsLW->currentItem()) {
|
|
on_suggestionsLW_itemClicked(d->ui.suggestionsLW->currentItem());
|
|
on_replacePB_clicked();
|
|
}
|
|
return true;
|
|
} else if (e->key() == Qt::Key_Right) {
|
|
if (d->ui.suggestionsLW->currentItem())
|
|
on_suggestionsLW_itemClicked(d->ui.suggestionsLW->currentItem());
|
|
return true;
|
|
}
|
|
}
|
|
// standard event processing
|
|
return QWidget::eventFilter(obj, event);
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_suggestionsLW_itemClicked(QListWidgetItem * item)
|
|
{
|
|
if (d->ui.replaceCO->count() != 0)
|
|
d->ui.replaceCO->setItemText(0, item->text());
|
|
else
|
|
d->ui.replaceCO->addItem(item->text());
|
|
|
|
d->ui.replaceCO->setCurrentIndex(0);
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_replaceCO_highlighted(const QString & str)
|
|
{
|
|
QListWidget * lw = d->ui.suggestionsLW;
|
|
if (lw->currentItem() && lw->currentItem()->text() == str)
|
|
return;
|
|
|
|
for (int i = 0; i != lw->count(); ++i) {
|
|
if (lw->item(i)->text() == str) {
|
|
lw->setCurrentRow(i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::updateView()
|
|
{
|
|
BufferView * bv = d->gv_->documentBufferView();
|
|
// we need a buffer view and the buffer has to be writable
|
|
bool const enabled = bv != 0 && !bv->buffer().isReadonly();
|
|
setEnabled(enabled);
|
|
if (enabled && hasFocus()) {
|
|
Cursor const & cursor = bv->cursor();
|
|
if (d->start_.empty() || !d->isCurrentBuffer(cursor)) {
|
|
if (cursor.selection()) {
|
|
d->begin_ = cursor.selectionBegin();
|
|
d->end_ = cursor.selectionEnd();
|
|
d->start_ = d->begin_;
|
|
bv->cursor().setCursor(d->start_);
|
|
} else {
|
|
d->begin_ = DocIterator();
|
|
d->end_ = DocIterator();
|
|
d->start_ = cursor;
|
|
}
|
|
d->wrapAround(false);
|
|
d->check();
|
|
}
|
|
}
|
|
}
|
|
|
|
DocIterator const SpellcheckerWidget::Private::cursor() const
|
|
{
|
|
BufferView * bv = gv_->documentBufferView();
|
|
return bv ? bv->cursor() : DocIterator();
|
|
}
|
|
|
|
bool SpellcheckerWidget::Private::continueFromBeginning()
|
|
{
|
|
DocIterator const current_ = cursor();
|
|
if (isCurrentBuffer(current_) && !begin_.empty()) {
|
|
// selection was checked
|
|
// start over from beginning makes no sense
|
|
fixPositionsIfBroken();
|
|
hide();
|
|
if (current_ == start_) {
|
|
// no errors found... tell the user the good news
|
|
// so there is some feedback
|
|
QMessageBox::information(p,
|
|
qt_("Spell Checker"),
|
|
qt_("Spell check of the selection done, "
|
|
"did not find any errors."));
|
|
}
|
|
return false;
|
|
}
|
|
QMessageBox::StandardButton const answer = QMessageBox::question(p,
|
|
qt_("Spell Checker"),
|
|
qt_("We reached the end of the document, would you like to "
|
|
"continue from the beginning?"),
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
|
|
if (answer == QMessageBox::No) {
|
|
fixPositionsIfBroken();
|
|
hide();
|
|
return false;
|
|
}
|
|
// there is no selection, start over from the beginning now
|
|
wrapAround(true);
|
|
dispatch(FuncRequest(LFUN_BUFFER_BEGIN));
|
|
return true;
|
|
}
|
|
|
|
bool SpellcheckerWidget::Private::isCurrentBuffer(DocIterator const & cursor) const
|
|
{
|
|
return start_.buffer() == cursor.buffer();
|
|
}
|
|
|
|
bool SpellcheckerWidget::Private::atLastPos(DocIterator const & cursor) const
|
|
{
|
|
bool const valid_end = !end_.empty();
|
|
return cursor.depth() <= 1 && (
|
|
cursor.atEnd() ||
|
|
(valid_end && isCurrentBuffer(cursor) && cursor >= end_));
|
|
}
|
|
|
|
bool SpellcheckerWidget::Private::isWrapAround(DocIterator const & cursor) const
|
|
{
|
|
return wrap_around_ && isCurrentBuffer(cursor) && start_ < cursor;
|
|
}
|
|
|
|
void SpellcheckerWidget::Private::fixPositionsIfBroken()
|
|
{
|
|
DocIterator const current_ = cursor();
|
|
if (!isCurrentBuffer(current_)) {
|
|
LYXERR(Debug::GUI, "wrong document of current cursor position " << start_);
|
|
start_ = current_;
|
|
begin_ = DocIterator();
|
|
end_ = DocIterator();
|
|
}
|
|
if (start_.fixIfBroken())
|
|
LYXERR(Debug::GUI, "broken start position fixed " << start_);
|
|
if (begin_.fixIfBroken()) {
|
|
LYXERR(Debug::GUI, "broken selection begin position fixed " << begin_);
|
|
begin_ = DocIterator();
|
|
end_ = DocIterator();
|
|
}
|
|
if (end_.fixIfBroken())
|
|
LYXERR(Debug::GUI, "broken selection end position fixed " << end_);
|
|
}
|
|
|
|
void SpellcheckerWidget::Private::hide() const
|
|
{
|
|
BufferView * bv = gv_->documentBufferView();
|
|
Cursor & bvcur = bv->cursor();
|
|
dv_->hide();
|
|
if (isCurrentBuffer(bvcur)) {
|
|
if (!begin_.empty() && !end_.empty()) {
|
|
// restore previous selection
|
|
setSelection(begin_, end_);
|
|
} else {
|
|
// restore cursor position
|
|
bvcur.setCursor(start_);
|
|
bvcur.clearSelection();
|
|
bv->processUpdateFlags(Update::Force | Update::FitCursor);
|
|
}
|
|
}
|
|
}
|
|
|
|
void SpellcheckerWidget::Private::setSelection(
|
|
DocIterator const & from, DocIterator const & to) const
|
|
{
|
|
BufferView * bv = gv_->documentBufferView();
|
|
DocIterator end = to;
|
|
|
|
if (from.pit() != end.pit()) {
|
|
// there are multiple paragraphs in selection
|
|
Cursor & bvcur = bv->cursor();
|
|
bvcur.setCursor(from);
|
|
bvcur.clearSelection();
|
|
bvcur.selection(true);
|
|
bvcur.setCursor(end);
|
|
bvcur.selection(true);
|
|
} else {
|
|
// FIXME LFUN
|
|
// If we used a LFUN, dispatch would do all of this for us
|
|
int const size = end.pos() - from.pos();
|
|
bv->putSelectionAt(from, size, false);
|
|
}
|
|
bv->processUpdateFlags(Update::Force | Update::FitCursor);
|
|
}
|
|
|
|
void SpellcheckerWidget::Private::forward()
|
|
{
|
|
DocIterator const from = cursor();
|
|
|
|
dispatch(FuncRequest(LFUN_ESCAPE));
|
|
fixPositionsIfBroken();
|
|
if (!atLastPos(cursor())) {
|
|
dispatch(FuncRequest(LFUN_CHAR_FORWARD));
|
|
}
|
|
if (atLastPos(cursor())) {
|
|
return;
|
|
}
|
|
if (from == cursor()) {
|
|
//FIXME we must be at the end of a cell
|
|
dispatch(FuncRequest(LFUN_CHAR_FORWARD));
|
|
}
|
|
if (isWrapAround(cursor())) {
|
|
hide();
|
|
}
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_languageCO_activated(int index)
|
|
{
|
|
string const lang =
|
|
fromqstr(d->ui.languageCO->itemData(index).toString());
|
|
if (!d->word_.lang() || d->word_.lang()->lang() == lang)
|
|
// nothing changed
|
|
return;
|
|
dispatch(FuncRequest(LFUN_LANGUAGE, lang));
|
|
d->check();
|
|
}
|
|
|
|
|
|
bool SpellcheckerWidget::initialiseParams(std::string const &)
|
|
{
|
|
BufferView * bv = d->gv_->documentBufferView();
|
|
if (bv == 0)
|
|
return false;
|
|
std::set<Language const *> langs =
|
|
bv->buffer().masterBuffer()->getLanguages();
|
|
if (!langs.empty())
|
|
d->setLanguage(*langs.begin());
|
|
d->start_ = DocIterator();
|
|
d->wrapAround(false);
|
|
d->canCheck();
|
|
return true;
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_ignoreAllPB_clicked()
|
|
{
|
|
/// ignore all occurrences of word
|
|
if (d->disabled())
|
|
return;
|
|
LYXERR(Debug::GUI, "Spellchecker: ignore all button");
|
|
if (d->word_.lang() && !d->word_.word().empty())
|
|
theSpellChecker()->accept(d->word_);
|
|
d->forward();
|
|
d->check();
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_addPB_clicked()
|
|
{
|
|
/// insert word in personal dictionary
|
|
if (d->disabled())
|
|
return;
|
|
LYXERR(Debug::GUI, "Spellchecker: add word button");
|
|
theSpellChecker()->insert(d->word_);
|
|
d->forward();
|
|
d->check();
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_ignorePB_clicked()
|
|
{
|
|
/// ignore this occurrence of word
|
|
if (d->disabled())
|
|
return;
|
|
LYXERR(Debug::GUI, "Spellchecker: ignore button");
|
|
d->forward();
|
|
d->check();
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_findNextPB_clicked()
|
|
{
|
|
if (d->disabled())
|
|
return;
|
|
docstring const textfield = qstring_to_ucs4(d->ui.wordED->text());
|
|
docstring const datastring = find2string(textfield,
|
|
true, true, true);
|
|
LYXERR(Debug::GUI, "Spellchecker: find next (" << textfield << ")");
|
|
dispatch(FuncRequest(LFUN_WORD_FIND, datastring));
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_replacePB_clicked()
|
|
{
|
|
if (d->disabled())
|
|
return;
|
|
docstring const textfield = qstring_to_ucs4(d->ui.wordED->text());
|
|
docstring const replacement = qstring_to_ucs4(d->ui.replaceCO->currentText());
|
|
docstring const datastring =
|
|
replace2string(replacement, textfield,
|
|
true, // case sensitive
|
|
true, // match word
|
|
false, // all words
|
|
true, // forward
|
|
false); // find next
|
|
|
|
LYXERR(Debug::GUI, "Replace (" << replacement << ")");
|
|
dispatch(FuncRequest(LFUN_WORD_REPLACE, datastring));
|
|
d->forward();
|
|
d->check();
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::on_replaceAllPB_clicked()
|
|
{
|
|
if (d->disabled())
|
|
return;
|
|
docstring const textfield = qstring_to_ucs4(d->ui.wordED->text());
|
|
docstring const replacement = qstring_to_ucs4(d->ui.replaceCO->currentText());
|
|
docstring const datastring =
|
|
replace2string(replacement, textfield,
|
|
true, // case sensitive
|
|
true, // match word
|
|
true, // all words
|
|
true, // forward
|
|
false); // find next
|
|
|
|
LYXERR(Debug::GUI, "Replace all (" << replacement << ")");
|
|
dispatch(FuncRequest(LFUN_WORD_REPLACE, datastring));
|
|
d->forward();
|
|
// replace all wraps around
|
|
d->wrapAround(true);
|
|
d->check(); // continue spellchecking
|
|
d->canCheck();
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::Private::updateSuggestions(docstring_list & words)
|
|
{
|
|
QString const suggestion = toqstr(word_.word());
|
|
ui.wordED->setText(suggestion);
|
|
QListWidget * lw = ui.suggestionsLW;
|
|
lw->clear();
|
|
|
|
if (words.empty()) {
|
|
p->on_suggestionsLW_itemClicked(new QListWidgetItem(suggestion));
|
|
return;
|
|
}
|
|
for (size_t i = 0; i != words.size(); ++i)
|
|
lw->addItem(toqstr(words[i]));
|
|
|
|
p->on_suggestionsLW_itemClicked(lw->item(0));
|
|
lw->setCurrentRow(0);
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::Private::setLanguage(Language const * lang)
|
|
{
|
|
int const pos = ui.languageCO->findData(toqstr(lang->lang()));
|
|
if (pos != -1)
|
|
ui.languageCO->setCurrentIndex(pos);
|
|
}
|
|
|
|
|
|
void SpellcheckerWidget::Private::check()
|
|
{
|
|
BufferView * bv = gv_->documentBufferView();
|
|
if (!bv || bv->buffer().text().empty())
|
|
return;
|
|
|
|
fixPositionsIfBroken();
|
|
|
|
SpellChecker * speller = theSpellChecker();
|
|
if (speller && !speller->hasDictionary(bv->buffer().language())) {
|
|
int dsize = speller->numDictionaries();
|
|
if (0 == dsize) {
|
|
hide();
|
|
QMessageBox::information(p,
|
|
qt_("Spell Checker"),
|
|
qt_("Spell checker has no dictionaries."));
|
|
return;
|
|
}
|
|
}
|
|
|
|
DocIterator from = bv->cursor();
|
|
DocIterator to = isCurrentBuffer(from) ? end_ : doc_iterator_end(&bv->buffer());
|
|
WordLangTuple word_lang;
|
|
docstring_list suggestions;
|
|
|
|
LYXERR(Debug::GUI, "Spellchecker: start check at " << from);
|
|
try {
|
|
bv->buffer().spellCheck(from, to, word_lang, suggestions);
|
|
} catch (ExceptionMessage const & message) {
|
|
if (message.type_ == WarningException) {
|
|
Alert::warning(message.title_, message.details_);
|
|
return;
|
|
}
|
|
throw message;
|
|
}
|
|
|
|
// end of document or selection?
|
|
if (atLastPos(from)) {
|
|
if (isWrapAround()) {
|
|
hide();
|
|
return;
|
|
}
|
|
if (continueFromBeginning())
|
|
check();
|
|
return;
|
|
}
|
|
|
|
if (isWrapAround(from)) {
|
|
hide();
|
|
return;
|
|
}
|
|
|
|
word_ = word_lang;
|
|
|
|
// set suggestions
|
|
updateSuggestions(suggestions);
|
|
// set language
|
|
if (!word_lang.lang())
|
|
return;
|
|
setLanguage(word_lang.lang());
|
|
// mark misspelled word
|
|
setSelection(from, to);
|
|
}
|
|
|
|
|
|
GuiSpellchecker::GuiSpellchecker(GuiView & parent,
|
|
Qt::DockWidgetArea area, Qt::WindowFlags flags)
|
|
: DockView(parent, "spellchecker", qt_("Spellchecker"),
|
|
area, flags)
|
|
{
|
|
widget_ = new SpellcheckerWidget(&parent, this);
|
|
setWidget(widget_);
|
|
setFocusProxy(widget_);
|
|
}
|
|
|
|
|
|
GuiSpellchecker::~GuiSpellchecker()
|
|
{
|
|
setFocusProxy(0);
|
|
delete widget_;
|
|
}
|
|
|
|
|
|
void GuiSpellchecker::updateView()
|
|
{
|
|
widget_->updateView();
|
|
}
|
|
|
|
|
|
Dialog * createGuiSpellchecker(GuiView & lv)
|
|
{
|
|
GuiSpellchecker * gui = new GuiSpellchecker(lv, Qt::RightDockWidgetArea);
|
|
#ifdef Q_OS_MAC
|
|
gui->setFloating(true);
|
|
#endif
|
|
return gui;
|
|
}
|
|
|
|
|
|
} // namespace frontend
|
|
} // namespace lyx
|
|
|
|
#include "moc_GuiSpellchecker.cpp"
|