/* * Copyright (C) 2013 - 2015 Hong Jen Yee (PCMan) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * */ #include "pathedit.h" #include "pathedit_p.h" #include #include #include #include #include #include #include #include namespace Fm { void PathEditJob::runJob() { GError* err = nullptr; GFileEnumerator* enu = g_file_enumerate_children(dirName, // G_FILE_ATTRIBUTE_STANDARD_NAME"," G_FILE_ATTRIBUTE_STANDARD_DISPLAY_NAME"," G_FILE_ATTRIBUTE_STANDARD_TYPE, G_FILE_QUERY_INFO_NONE, cancellable, &err); if(enu) { while(!g_cancellable_is_cancelled(cancellable)) { GFileInfo* inf = g_file_enumerator_next_file(enu, cancellable, &err); if(inf) { GFileType type = g_file_info_get_file_type(inf); if(type == G_FILE_TYPE_DIRECTORY) { const char* name = g_file_info_get_display_name(inf); // FIXME: encoding conversion here? subDirs.append(QString::fromUtf8(name)); } g_object_unref(inf); } else { if(err) { g_error_free(err); err = nullptr; } else { /* EOF */ break; } } } g_file_enumerator_close(enu, cancellable, nullptr); g_object_unref(enu); } // finished! let's update the UI in the main thread Q_EMIT finished(); QThread::currentThread()->quit(); } PathEdit::PathEdit(QWidget* parent): QLineEdit(parent), completer_(new QCompleter()), model_(new QStringListModel()), cancellable_(nullptr) { setCompleter(completer_); completer_->setModel(model_); connect(this, &PathEdit::textChanged, this, &PathEdit::onTextChanged); connect(this, &PathEdit::textEdited, this, &PathEdit::onTextEdited); } PathEdit::~PathEdit() { delete completer_; if(model_) { delete model_; } if(cancellable_) { g_cancellable_cancel(cancellable_); g_object_unref(cancellable_); } } void PathEdit::focusInEvent(QFocusEvent* e) { QLineEdit::focusInEvent(e); // build the completion list only when we have the keyboard focus reloadCompleter(true); } void PathEdit::focusOutEvent(QFocusEvent* e) { QLineEdit::focusOutEvent(e); // free the completion list since we don't need it anymore freeCompleter(); } bool PathEdit::event(QEvent* e) { // Stop Qt from moving the keyboard focus to the next widget when "Tab" is pressed. // Instead, we need to do auto-completion in this case. if(e->type() == QEvent::KeyPress) { QKeyEvent* keyEvent = static_cast(e); if(keyEvent->key() == Qt::Key_Tab && keyEvent->modifiers() == Qt::NoModifier) { // Tab key is pressed e->accept(); // do auto-completion when the user press the Tab key. // This fixes #201: https://github.com/lxde/pcmanfm-qt/issues/201 autoComplete(); return true; } } return QLineEdit::event(e); } void PathEdit::onTextEdited(const QString& text) { // just replace start tilde with home path if text is changed by user if(text == QLatin1String("~") || text.startsWith(QLatin1String("~/"))) { QString txt(text); txt.replace(0, 1, QDir::homePath()); setText(txt); // emits textChanged() return; } } void PathEdit::onTextChanged(const QString& text) { if(text == QLatin1String("~") || text.startsWith(QLatin1String("~/"))) { // do nothing with a start tilde because neither Fm::FilePath nor autocompletion // understands it; instead, wait until textChanged() is emitted again without it // WARNING: replacing tilde may not be safe here return; } int pos = text.lastIndexOf('/'); if(pos >= 0) { ++pos; } else { pos = text.length(); } QString newPrefix = text.left(pos); if(currentPrefix_ != newPrefix) { currentPrefix_ = newPrefix; // only build the completion list if we have the keyboard focus // if we don't have the focus now, then we'll rebuild the completion list // when focusInEvent happens. this avoid unnecessary dir loading. if(hasFocus()) { reloadCompleter(false); } } } void PathEdit::autoComplete() { // find longest common prefix of the strings currently shown in the candidate list QAbstractItemModel* model = completer_->completionModel(); if(model->rowCount() > 0) { int minLen = text().length(); QString commonPrefix = model->data(model->index(0, 0)).toString(); for(int row = 1; row < model->rowCount() && commonPrefix.length() > minLen; ++row) { QModelIndex index = model->index(row, 0); QString rowText = model->data(index).toString(); int prefixLen = 0; while(prefixLen < rowText.length() && prefixLen < commonPrefix.length() && rowText[prefixLen] == commonPrefix[prefixLen]) { ++prefixLen; } commonPrefix.truncate(prefixLen); } if(commonPrefix.length() > minLen) { setText(commonPrefix); } } } void PathEdit::reloadCompleter(bool triggeredByFocusInEvent) { // parent dir has been changed, reload dir list // if(currentPrefix_[0] == "~") { // special case for home dir // cancel running dir-listing jobs, if there's any if(cancellable_) { g_cancellable_cancel(cancellable_); g_object_unref(cancellable_); } // create a new job to do dir listing PathEditJob* job = new PathEditJob(); job->edit = this; job->triggeredByFocusInEvent = triggeredByFocusInEvent; // need to use fm_file_new_for_commandline_arg() rather than g_file_new_for_commandline_arg(). // otherwise, our own vfs, such as menu://, won't be loaded. job->dirName = fm_file_new_for_commandline_arg(currentPrefix_.toLocal8Bit().constData()); // qDebug("load: %s", g_file_get_uri(data->dirName)); cancellable_ = g_cancellable_new(); job->cancellable = (GCancellable*)g_object_ref(cancellable_); // launch a new worker thread to handle the job QThread* thread = new QThread(); job->moveToThread(thread); connect(job, &PathEditJob::finished, this, &PathEdit::onJobFinished, Qt::BlockingQueuedConnection); // connect(job, &PathEditJob::finished, thread, &QThread::quit); connect(thread, &QThread::started, job, &PathEditJob::runJob); connect(thread, &QThread::finished, thread, &QObject::deleteLater); connect(thread, &QThread::finished, job, &QObject::deleteLater); thread->start(QThread::LowPriority); } void PathEdit::freeCompleter() { if(cancellable_) { g_cancellable_cancel(cancellable_); g_object_unref(cancellable_); cancellable_ = nullptr; } model_->setStringList(QStringList()); } // This slot is called from main thread so it's safe to access the GUI void PathEdit::onJobFinished() { PathEditJob* data = static_cast(sender()); if(!g_cancellable_is_cancelled(data->cancellable)) { // update the completer only if the job is not cancelled QStringList::iterator it; for(it = data->subDirs.begin(); it != data->subDirs.end(); ++it) { // qDebug("%s", it->toUtf8().constData()); *it = (currentPrefix_ % *it); } model_->setStringList(data->subDirs); // trigger completion manually if(hasFocus() && !data->triggeredByFocusInEvent) { completer_->complete(); } } else { model_->setStringList(QStringList()); } if(cancellable_) { g_object_unref(cancellable_); cancellable_ = nullptr; } } } // namespace Fm