mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-16 13:02:49 +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.
409 lines
9.9 KiB
C++
409 lines
9.9 KiB
C++
/**
|
|
* \file GuiParagraph.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Edwin Leuven
|
|
* \author Richard Heck
|
|
* \author Abdelrazak Younes
|
|
* \author Angus Leeming
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "GuiParagraph.h"
|
|
|
|
#include "qt_helpers.h"
|
|
|
|
#include "Buffer.h"
|
|
#include "BufferParams.h"
|
|
#include "BufferView.h"
|
|
#include "Cursor.h"
|
|
#include "FuncRequest.h"
|
|
#include "GuiView.h"
|
|
#include "Lexer.h"
|
|
#include "Paragraph.h"
|
|
#include "ParagraphParameters.h"
|
|
#include "Spacing.h"
|
|
|
|
#include "support/debug.h"
|
|
#include "support/gettext.h"
|
|
|
|
#include <QCheckBox>
|
|
#include <QDialogButtonBox>
|
|
#include <QLineEdit>
|
|
#include <QPushButton>
|
|
#include <QSettings>
|
|
#include <QVariant>
|
|
|
|
#include <sstream>
|
|
|
|
using namespace std;
|
|
|
|
namespace lyx {
|
|
namespace frontend {
|
|
|
|
GuiParagraph::GuiParagraph(GuiView & lv)
|
|
: DialogView(lv, "paragraph", qt_("Paragraph Settings"))
|
|
{
|
|
setupUi(this);
|
|
|
|
// fix height to minimum
|
|
setFixedHeight(sizeHint().height());
|
|
|
|
connect(alignDefaultRB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(alignJustRB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(alignLeftRB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(alignRightRB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(alignCenterRB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(linespacing, SIGNAL(activated(int)), this, SLOT(changed()));
|
|
connect(linespacingValue, SIGNAL(textChanged(QString)),
|
|
this, SLOT(changed()));
|
|
connect(indentCB, SIGNAL(clicked()), this, SLOT(changed()));
|
|
connect(labelWidth, SIGNAL(textChanged(QString)),
|
|
this, SLOT(changed()));
|
|
|
|
#ifdef Q_OS_MAC
|
|
// On Mac it's common to have tool windows which are always in the
|
|
// foreground and are hidden when the main window is not focused.
|
|
setWindowFlags(Qt::Tool);
|
|
synchronizedViewCB->setChecked(true);
|
|
#else
|
|
synchronizedViewCB->setChecked(false);
|
|
#endif
|
|
|
|
on_synchronizedViewCB_toggled();
|
|
QDoubleValidator * val = new QDoubleValidator(linespacingValue);
|
|
val->setNotation(QDoubleValidator::StandardNotation);
|
|
linespacingValue->setValidator(val);
|
|
|
|
labelWidth->setWhatsThis(qt_(
|
|
"As described in the User Guide, the width of"
|
|
" this text determines the width of the label part"
|
|
" of each item in environments like List and"
|
|
" Description.\n"
|
|
"\n"
|
|
" Normally, you won't need to set this,"
|
|
" since the largest label width of all the"
|
|
" items is used."
|
|
));
|
|
|
|
radioMap_[LYX_ALIGN_LAYOUT] = alignDefaultRB;
|
|
radioMap_[LYX_ALIGN_BLOCK] = alignJustRB;
|
|
radioMap_[LYX_ALIGN_LEFT] = alignLeftRB;
|
|
radioMap_[LYX_ALIGN_RIGHT] = alignRightRB;
|
|
radioMap_[LYX_ALIGN_CENTER] = alignCenterRB;
|
|
|
|
alignDefaultLabel_ = alignDefaultRB->text();
|
|
}
|
|
|
|
|
|
void GuiParagraph::on_linespacing_activated(int index)
|
|
{
|
|
linespacingValue->setEnabled(index == 4);
|
|
}
|
|
|
|
|
|
void GuiParagraph::checkAlignmentRadioButtons()
|
|
{
|
|
static std::map<LyXAlignment, QString> labelMap_;
|
|
if (labelMap_.empty()) {
|
|
labelMap_[LYX_ALIGN_BLOCK] = qt_("Justified");
|
|
labelMap_[LYX_ALIGN_LEFT] = qt_("Left");
|
|
labelMap_[LYX_ALIGN_RIGHT] = qt_("Right");
|
|
labelMap_[LYX_ALIGN_CENTER] = qt_("Center");
|
|
}
|
|
|
|
RadioMap::iterator it = radioMap_.begin();
|
|
for (; it != radioMap_.end(); ++it) {
|
|
LyXAlignment const align = it->first;
|
|
it->second->setEnabled(align & alignPossible());
|
|
}
|
|
if (haveMultiParSelection())
|
|
alignDefaultRB->setText(alignDefaultLabel_);
|
|
else
|
|
alignDefaultRB->setText(alignDefaultLabel_ + " ("
|
|
+ labelMap_[bufferview()->cursor().innerParagraph().getDefaultAlign(buffer().params())] + ")");
|
|
}
|
|
|
|
|
|
void GuiParagraph::alignmentToRadioButtons(LyXAlignment align)
|
|
{
|
|
RadioMap::const_iterator it = radioMap_.begin();
|
|
for (;it != radioMap_.end(); ++it) {
|
|
it->second->blockSignals(true);
|
|
it->second->setChecked(align == it->first);
|
|
it->second->blockSignals(false);
|
|
}
|
|
}
|
|
|
|
|
|
LyXAlignment GuiParagraph::getAlignmentFromDialog() const
|
|
{
|
|
LyXAlignment alignment = LYX_ALIGN_NONE;
|
|
RadioMap::const_iterator it = radioMap_.begin();
|
|
for (; it != radioMap_.end(); ++it) {
|
|
if (it->second->isChecked()) {
|
|
alignment = it->first;
|
|
break;
|
|
}
|
|
}
|
|
return alignment;
|
|
}
|
|
|
|
|
|
void GuiParagraph::on_synchronizedViewCB_toggled()
|
|
{
|
|
bool in_sync = synchronizedViewCB->isChecked();
|
|
buttonBox->button(QDialogButtonBox::Reset)->setEnabled(!in_sync);
|
|
buttonBox->button(QDialogButtonBox::Apply)->setEnabled(!in_sync);
|
|
buttonBox->button(QDialogButtonBox::Ok)->setEnabled(!in_sync);
|
|
if (!in_sync)
|
|
buttonBox->button(QDialogButtonBox::Cancel)->setText(qt_("&Cancel"));
|
|
else
|
|
buttonBox->button(QDialogButtonBox::Cancel)->setText(qt_("&Close"));
|
|
}
|
|
|
|
|
|
void GuiParagraph::changed()
|
|
{
|
|
QLocale loc;
|
|
// We apply immediately, except if we have custom line spacing
|
|
// with an intermediate result (trailing decimal separator) or
|
|
// an invalid value (which might as well be intermediate)
|
|
if (synchronizedViewCB->isChecked()
|
|
&& (linespacing->currentIndex() != 4
|
|
|| (!linespacingValue->text().endsWith(loc.decimalPoint())
|
|
&& linespacingValue->hasAcceptableInput())))
|
|
applyView();
|
|
}
|
|
|
|
|
|
void GuiParagraph::on_buttonBox_clicked(QAbstractButton * button)
|
|
{
|
|
switch (buttonBox->standardButton(button)) {
|
|
case QDialogButtonBox::Ok:
|
|
applyView();
|
|
hide();
|
|
break;
|
|
case QDialogButtonBox::Apply:
|
|
applyView();
|
|
break;
|
|
case QDialogButtonBox::Cancel:
|
|
hide();
|
|
break;
|
|
case QDialogButtonBox::Reset:
|
|
updateView();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void GuiParagraph::applyView()
|
|
{
|
|
params_ = params();
|
|
|
|
params_.align(getAlignmentFromDialog());
|
|
|
|
// get spacing
|
|
Spacing::Space ls = Spacing::Default;
|
|
string other;
|
|
switch (linespacing->currentIndex()) {
|
|
case 0:
|
|
ls = Spacing::Default;
|
|
break;
|
|
case 1:
|
|
ls = Spacing::Single;
|
|
break;
|
|
case 2:
|
|
ls = Spacing::Onehalf;
|
|
break;
|
|
case 3:
|
|
ls = Spacing::Double;
|
|
break;
|
|
case 4:
|
|
ls = Spacing::Other;
|
|
other = widgetToDoubleStr(linespacingValue);
|
|
break;
|
|
}
|
|
|
|
Spacing const spacing(ls, other);
|
|
params_.spacing(spacing);
|
|
|
|
// label width
|
|
params_.labelWidthString(qstring_to_ucs4(labelWidth->text()));
|
|
// indentation
|
|
params_.noindent(!indentCB->isChecked());
|
|
|
|
dispatchParams();
|
|
}
|
|
|
|
|
|
void GuiParagraph::updateView()
|
|
{
|
|
on_synchronizedViewCB_toggled();
|
|
|
|
ParagraphParameters const & pp = params();
|
|
|
|
// label width
|
|
docstring const & labelwidth = pp.labelWidthString();
|
|
if (hasLabelwidth()) {
|
|
labelwidthGB->setEnabled(true);
|
|
labelWidth->setText(toqstr(labelwidth));
|
|
} else {
|
|
labelwidthGB->setEnabled(false);
|
|
labelWidth->setText(QString());
|
|
}
|
|
|
|
// alignment
|
|
checkAlignmentRadioButtons();
|
|
alignmentToRadioButtons(pp.align());
|
|
|
|
//indentation
|
|
bool const canindent = canIndent();
|
|
indentCB->setEnabled(canindent);
|
|
indentCB->setChecked(canindent && !pp.noindent());
|
|
|
|
// linespacing
|
|
int ls;
|
|
bool pending_input = false;
|
|
Spacing const & space = pp.spacing();
|
|
if (synchronizedViewCB->isChecked() && linespacingValue->hasFocus()) {
|
|
// The user is about to enter custom spacing.
|
|
// We thus stay in Custom mode.
|
|
// This prevents the combo from e.g. immediately
|
|
// switching to single if a user enters "1" in the
|
|
// linespacingValue widget while aiming at e.g. "1.3"
|
|
ls = 4;
|
|
pending_input = true;
|
|
} else {
|
|
switch (space.getSpace()) {
|
|
case Spacing::Single:
|
|
ls = 1;
|
|
break;
|
|
case Spacing::Onehalf:
|
|
ls = 2;
|
|
break;
|
|
case Spacing::Double:
|
|
ls = 3;
|
|
break;
|
|
case Spacing::Other:
|
|
ls = 4;
|
|
break;
|
|
default:
|
|
ls = 0;
|
|
break;
|
|
}
|
|
}
|
|
linespacing->setCurrentIndex(ls);
|
|
if (space.getSpace() == Spacing::Other || pending_input) {
|
|
doubleToWidget(linespacingValue, space.getValue());
|
|
linespacingValue->setEnabled(true);
|
|
} else {
|
|
linespacingValue->setText(QString());
|
|
linespacingValue->setEnabled(false);
|
|
}
|
|
// Somewhere in the chain this can lose default status (#11417)
|
|
buttonBox->button(QDialogButtonBox::Ok)->setDefault(true);
|
|
}
|
|
|
|
|
|
void GuiParagraph::enableView(bool enable)
|
|
{
|
|
indentCB->setEnabled(enable);
|
|
linespacing->setEnabled(enable);
|
|
labelWidth->setEnabled(enable);
|
|
synchronizedViewCB->setEnabled(enable);
|
|
buttonBox->button(QDialogButtonBox::Apply)->setEnabled(enable);
|
|
buttonBox->button(QDialogButtonBox::Reset)->setEnabled(enable);
|
|
if (!enable)
|
|
synchronizedViewCB->setChecked(true);
|
|
RadioMap::const_iterator it = radioMap_.begin();
|
|
for (; it != radioMap_.end(); ++it)
|
|
it->second->setEnabled(enable);
|
|
}
|
|
|
|
|
|
ParagraphParameters const & GuiParagraph::params() const
|
|
{
|
|
if (haveMultiParSelection()) {
|
|
// FIXME: in case of multi-paragraph selection, it would be nice to
|
|
// initialise the parameters that are common to all paragraphs.
|
|
static ParagraphParameters const empty;
|
|
return empty;
|
|
}
|
|
return bufferview()->cursor().innerParagraph().params();
|
|
}
|
|
|
|
|
|
void GuiParagraph::dispatchParams()
|
|
{
|
|
ostringstream os;
|
|
params_.write(os);
|
|
FuncRequest const fr(getLfun(), os.str());
|
|
dispatch(fr);
|
|
}
|
|
|
|
|
|
bool GuiParagraph::haveMultiParSelection() const
|
|
{
|
|
Cursor const & cur = bufferview()->cursor();
|
|
return cur.selection() && cur.selBegin().pit() != cur.selEnd().pit();
|
|
}
|
|
|
|
|
|
bool GuiParagraph::canIndent() const
|
|
{
|
|
Layout const lay = bufferview()->cursor().innerParagraph().layout();
|
|
if (buffer().params().paragraph_separation
|
|
== BufferParams::ParagraphIndentSeparation)
|
|
return (lay.toggle_indent != ITOGGLE_NEVER);
|
|
return (lay.toggle_indent == ITOGGLE_ALWAYS);
|
|
}
|
|
|
|
|
|
LyXAlignment GuiParagraph::alignPossible() const
|
|
{
|
|
return bufferview()->cursor().innerParagraph().layout().alignpossible;
|
|
}
|
|
|
|
|
|
bool GuiParagraph::hasLabelwidth() const
|
|
{
|
|
Layout layout = bufferview()->cursor().innerParagraph().layout();
|
|
return (layout.margintype == MARGIN_MANUAL
|
|
|| layout.latextype == LATEX_BIB_ENVIRONMENT);
|
|
}
|
|
|
|
|
|
void GuiParagraph::saveSession(QSettings & settings) const
|
|
{
|
|
Dialog::saveSession(settings);
|
|
settings.setValue(sessionKey() + "/autoapply", synchronizedViewCB->isChecked());
|
|
}
|
|
|
|
|
|
void GuiParagraph::restoreSession()
|
|
{
|
|
Dialog::restoreSession();
|
|
QSettings settings;
|
|
synchronizedViewCB->setChecked(
|
|
settings.value(sessionKey() + "/autoapply").toBool());
|
|
}
|
|
|
|
|
|
Dialog * createGuiParagraph(GuiView & lv)
|
|
{
|
|
return new GuiParagraph(lv);
|
|
}
|
|
|
|
|
|
} // namespace frontend
|
|
} // namespace lyx
|
|
|
|
#include "moc_GuiParagraph.cpp"
|