#include "TocModel.h"
#include "Buffer.h"
+#include "BufferView.h"
#include "CutAndPaste.h"
#include "FuncRequest.h"
-#include "LyXFunc.h"
+#include "FuncStatus.h"
+#include "LyX.h"
#include "Menus.h"
#include "TocBackend.h"
// Only one item selected at a time.
tocTV->setSelectionMode(QAbstractItemView::SingleSelection);
+ setFocusProxy(tocTV);
// The toc types combo won't change its model.
typeCO->setModel(gui_view_.tocModels().nameModel());
this, SLOT(showContextMenu(const QPoint &)));
connect(tocTV, SIGNAL(customContextMenuRequested(const QPoint &)),
this, SLOT(showContextMenu(const QPoint &)));
+ connect(filterLE, SIGNAL(textEdited(QString)),
+ this, SLOT(filterContents()));
init(QString());
}
}
-void TocWidget::doDispatch(Cursor & cur, FuncRequest const & cmd)
+Inset * TocWidget::itemInset() const
{
QModelIndex const & index = tocTV->currentIndex();
TocItem const & item =
DocIterator const & dit = item.dit();
Inset * inset = 0;
- if (current_type_ == "label")
+ if (current_type_ == "label"
+ || current_type_ == "graphics"
+ || current_type_ == "citation"
+ || current_type_ == "child")
inset = dit.nextInset();
+ else if (current_type_ == "branch"
+ || current_type_ == "index"
+ || current_type_ == "change")
+ inset = &dit.inset();
+
+ else if (current_type_ == "table"
+ || current_type_ == "listing"
+ || current_type_ == "figure") {
+ DocIterator tmp_dit(dit);
+ tmp_dit.pop_back();
+ inset = &tmp_dit.inset();
+ }
+ return inset;
+}
+
+
+bool TocWidget::getStatus(Cursor & cur, FuncRequest const & cmd,
+ FuncStatus & status) const
+{
+ Inset * inset = itemInset();
FuncRequest tmpcmd(cmd);
- if (inset)
- inset->dispatch(cur, tmpcmd);
+
+ QModelIndex const & index = tocTV->currentIndex();
+ TocItem const & item =
+ gui_view_.tocModels().currentItem(current_type_, index);
+
+ switch (cmd.action())
+ {
+ case LFUN_CHANGE_ACCEPT:
+ case LFUN_CHANGE_REJECT:
+ case LFUN_OUTLINE_UP:
+ case LFUN_OUTLINE_DOWN:
+ case LFUN_OUTLINE_IN:
+ case LFUN_OUTLINE_OUT:
+ case LFUN_SECTION_SELECT:
+ status.setEnabled(true);
+ return true;
+
+ case LFUN_LABEL_COPY_AS_REF: {
+ // For labels in math, we need to supply the label as a string
+ FuncRequest label_copy(LFUN_LABEL_COPY_AS_REF, item.asString());
+ if (inset)
+ return inset->getStatus(cur, label_copy, status);
+ }
+
+ default:
+ if (inset)
+ return inset->getStatus(cur, tmpcmd, status);
+ }
+
+ return false;
+}
+
+
+void TocWidget::doDispatch(Cursor & cur, FuncRequest const & cmd)
+{
+ Inset * inset = itemInset();
+ FuncRequest tmpcmd(cmd);
+
+ QModelIndex const & index = tocTV->currentIndex();
+ TocItem const & item =
+ gui_view_.tocModels().currentItem(current_type_, index);
+
+ // Start an undo group.
+ cur.beginUndoGroup();
+
+ switch (cmd.action())
+ {
+ case LFUN_CHANGE_ACCEPT:
+ case LFUN_CHANGE_REJECT:
+ case LFUN_SECTION_SELECT:
+ dispatch(item.action());
+ cur.dispatch(tmpcmd);
+ break;
+
+ case LFUN_LABEL_COPY_AS_REF: {
+ // For labels in math, we need to supply the label as a string
+ FuncRequest label_copy(LFUN_LABEL_COPY_AS_REF, item.asString());
+ if (inset)
+ inset->dispatch(cur, label_copy);
+ break;
+ }
+
+ case LFUN_OUTLINE_UP:
+ case LFUN_OUTLINE_DOWN:
+ case LFUN_OUTLINE_IN:
+ case LFUN_OUTLINE_OUT:
+ outline(cmd.action());
+ break;
+
+ default:
+ if (inset)
+ inset->dispatch(cur, tmpcmd);
+ }
+ cur.endUndoGroup();
}
updateView();
}
+
void TocWidget::on_persistentCB_stateChanged(int state)
{
persistent_ = state == Qt::Checked;
{
current_type_ = typeCO->itemData(index).toString();
updateView();
- gui_view_.setFocus();
+ if (typeCO->hasFocus())
+ gui_view_.setFocus();
}
-void TocWidget::outline(int func_code)
+void TocWidget::outline(FuncCode func_code)
{
- enableControls(false);
QModelIndexList const & list = tocTV->selectionModel()->selectedIndexes();
if (list.isEmpty())
return;
enableControls(false);
goTo(list[0]);
- dispatch(FuncRequest(static_cast<FuncCode>(func_code)));
+ dispatch(FuncRequest(func_code));
enableControls(true);
gui_view_.setFocus();
}
}
-/// Test if outlining operation is possible
-static bool canOutline(QString const & type)
-{
- return type == "tableofcontents";
-}
-
-
void TocWidget::enableControls(bool enable)
{
updateTB->setEnabled(enable);
- if (!canOutline(current_type_))
+ if (!canOutline())
enable = false;
moveUpTB->setEnabled(enable);
moveDownTB->setEnabled(enable);
moveInTB->setEnabled(enable);
moveOutTB->setEnabled(enable);
- if (!enable) {
- depthSL->setMaximum(0);
- depthSL->setValue(0);
- }
-}
-
-
-/// Test if synchronized navigation is possible
-static bool canNavigate(QString const & type)
-{
- // It is not possible to have synchronous navigation in a correctl
- // and efficient way with the label type because Toc::item() do a linear
- // seatch. Even if fixed, it might even not be desirable to do so if we
- // want to support drag&drop of labels and references.
- return type != "label" && type != "change";
}
void TocWidget::updateView()
{
- if (!gui_view_.view()) {
- enableControls(false);
- typeCO->setEnabled(false);
+ if (!gui_view_.documentBufferView()) {
tocTV->setModel(0);
- tocTV->setEnabled(false);
+ depthSL->setMaximum(0);
+ depthSL->setValue(0);
+ setEnabled(false);
return;
}
- typeCO->setEnabled(true);
+ setEnabled(true);
+ bool const is_sortable = isSortable();
+ sortCB->setEnabled(is_sortable);
+ bool focus_ = tocTV->hasFocus();
tocTV->setEnabled(false);
tocTV->setUpdatesEnabled(false);
- QAbstractItemModel * toc_model = gui_view_.tocModels().model(current_type_);
+ QAbstractItemModel * toc_model =
+ gui_view_.tocModels().model(current_type_);
if (tocTV->model() != toc_model) {
tocTV->setModel(toc_model);
tocTV->setEditTriggers(QAbstractItemView::NoEditTriggers);
}
sortCB->blockSignals(true);
- sortCB->setChecked(gui_view_.tocModels().isSorted(current_type_));
+ sortCB->setChecked(is_sortable
+ && gui_view_.tocModels().isSorted(current_type_));
sortCB->blockSignals(false);
- persistentCB->setChecked(persistent_);
+ bool const can_navigate_ = canNavigate();
+ persistentCB->setEnabled(can_navigate_);
bool controls_enabled = toc_model && toc_model->rowCount() > 0
- && !gui_view_.buffer()->isReadonly();
+ && !gui_view_.documentBufferView()->buffer().isReadonly();
enableControls(controls_enabled);
depthSL->setMaximum(gui_view_.tocModels().depth(current_type_));
depthSL->setValue(depth_);
- if (!persistent_)
+ if (!persistent_ && can_navigate_)
setTreeDepth(depth_);
- if (canNavigate(current_type_))
+ if (can_navigate_) {
+ persistentCB->setChecked(persistent_);
select(gui_view_.tocModels().currentIndex(current_type_));
+ }
+ filterContents();
tocTV->setEnabled(true);
tocTV->setUpdatesEnabled(true);
+ if (focus_)
+ tocTV->setFocus();
+}
+
+
+void TocWidget::filterContents()
+{
+ if (!tocTV->model())
+ return;
+
+ QModelIndexList indices = tocTV->model()->match(
+ tocTV->model()->index(0, 0),
+ Qt::DisplayRole, "*", -1,
+ Qt::MatchFlags(Qt::MatchWildcard|Qt::MatchRecursive));
+
+ int size = indices.size();
+ for (int i = 0; i < size; i++) {
+ QModelIndex index = indices[i];
+ bool const matches =
+ index.data().toString().contains(
+ filterLE->text(), Qt::CaseInsensitive);
+ tocTV->setRowHidden(index.row(), index.parent(), !matches);
+ }
+ // recursively unhide parents of unhidden children
+ for (int i = size - 1; i >= 0; i--) {
+ QModelIndex index = indices[i];
+ if (!tocTV->isRowHidden(index.row(), index.parent())
+ && index.parent() != QModelIndex())
+ tocTV->setRowHidden(index.parent().row(),
+ index.parent().parent(), false);
+ }
}