/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ /* Rosegarden A MIDI and audio sequencer and musical notation editor. Copyright 2000-2011 the Rosegarden development team. Other copyrights also apply to some parts of this work. Please see the AUTHORS file and individual file headers for details. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. See the file COPYING included with this distribution for more information. */ #include "HeadersConfigurationPage.h" #include "misc/ConfigGroups.h" #include "document/RosegardenDocument.h" #include "document/io/LilyPondExporter.h" #include "gui/widgets/CollapsingFrame.h" #include "misc/Strings.h" #include "misc/Debug.h" #include "gui/widgets/LineEdit.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Rosegarden { HeadersConfigurationPage::HeadersConfigurationPage(QWidget *parent, RosegardenDocument *doc) : QWidget(parent), m_doc(doc) { QVBoxLayout *layout = new QVBoxLayout; // // LilyPond export: Printable headers // QGroupBox *headersBox = new QGroupBox(tr("Printable headers"), this); QHBoxLayout *headersBoxLayout = new QHBoxLayout; layout->addWidget(headersBox); QFrame *frameHeaders = new QFrame(headersBox); frameHeaders->setContentsMargins(10, 10, 10, 10); QGridLayout *layoutHeaders = new QGridLayout(frameHeaders); layoutHeaders->setSpacing(5); headersBoxLayout->addWidget(frameHeaders); headersBox->setLayout(headersBoxLayout); // grab user headers from metadata Configuration metadata = (&m_doc->getComposition())->getMetadata(); std::vector propertyNames = metadata.getPropertyNames(); std::vector fixedKeys = CompositionMetadataKeys::getFixedKeys(); std::set shown; for (unsigned int index = 0; index < fixedKeys.size(); index++) { std::string key = fixedKeys[index].getName(); std::string header = ""; QString headerStr(""); for (unsigned int i = 0; i < propertyNames.size(); ++i) { std::string property = propertyNames [i]; if (property == key) { // get the std::string from metadata header = metadata.get(property); // use that variable in a QObject::tr().arg() call, which will // return the English in a QString QString text(QObject::tr("%1").arg(strtoqstr(header))); // now we can QObject::tr() the QString in step 3 of this convoluted // process headerStr = QObject::tr(text); } } unsigned int row = 0, col = 0, width = 1; LineEdit *editHeader = new LineEdit(headerStr, frameHeaders); QString trName; if (key == headerDedication) { m_editDedication = editHeader; row = 0; col = 2; width = 2; trName = tr("Dedication"); } else if (key == headerTitle) { m_editTitle = editHeader; row = 1; col = 1; width = 4; trName = tr("Title"); } else if (key == headerSubtitle) { m_editSubtitle = editHeader; row = 2; col = 1; width = 4; trName = tr("Subtitle"); } else if (key == headerSubsubtitle) { m_editSubsubtitle = editHeader; row = 3; col = 2; width = 2; trName = tr("Subsubtitle"); } else if (key == headerPoet) { m_editPoet = editHeader; row = 4; col = 0; width = 2; trName = tr("Poet"); } else if (key == headerInstrument) { m_editInstrument = editHeader; row = 4; col = 2; width = 2; trName = tr("Instrument"); } else if (key == headerComposer) { m_editComposer = editHeader; row = 4; col = 4; width = 2; trName = tr("Composer"); } else if (key == headerMeter) { m_editMeter = editHeader; row = 5; col = 0; width = 3; trName = tr("Meter"); } else if (key == headerArranger) { m_editArranger = editHeader; row = 5; col = 3; width = 3; trName = tr("Arranger"); } else if (key == headerPiece) { m_editPiece = editHeader; row = 6; col = 0; width = 3; trName = tr("Piece"); } else if (key == headerOpus) { m_editOpus = editHeader; row = 6; col = 3; width = 3; trName = tr("Opus"); } else if (key == headerCopyright) { m_editCopyright = editHeader; row = 8; col = 1; width = 4; trName = tr("Copyright"); } else if (key == headerTagline) { m_editTagline = editHeader; row = 9; col = 1; width = 4; trName = tr("Tagline"); } // editHeader->setReadOnly(true); editHeader->setAlignment((col == 0 ? Qt::AlignLeft : (col >= 3 ? Qt::AlignRight : Qt::AlignCenter))); layoutHeaders->addWidget(editHeader, row, col, 1, width); // // ToolTips // // (These particular LineEdit objects were still taking a mixture of styles // from the external sheet and the internal hard coded stylesheet in the // subclass LineEdit. I guess I should have re-implemented QLineEdit // instead of subclassing it. QString localStyle("QToolTip {background-color: #FFFBD4; color: #000000;} QLineEdit {background-color: #FFFFFF; color: #000000;}"); editHeader->setStyleSheet(localStyle); editHeader->setToolTip(trName); shown.insert(key); } QLabel *separator = new QLabel(tr("The composition comes here."), frameHeaders); separator->setAlignment(Qt::AlignCenter); layoutHeaders->addWidget(separator, 7, 1, 1, 4 - 1+1); frameHeaders->setLayout(layoutHeaders); // // LilyPond export: Non-printable headers // // set default expansion to false for this group -- what a faff // (see note in TrackParameterBox.cpp for an explanation of this baffling // but necessary code) QSettings settings; settings.beginGroup(CollapsingFrameConfigGroup); bool expanded = qStrToBool(settings.value("nonprintableheaders", "false")) ; settings.setValue("nonprintableheaders", expanded); settings.endGroup(); CollapsingFrame *otherHeadersBox = new CollapsingFrame (tr("Additional headers"), this, "nonprintableheaders"); layout->addWidget(otherHeadersBox); setLayout(layout); QFrame *frameOtherHeaders = new QFrame(otherHeadersBox); otherHeadersBox->setWidgetFill(true); QFont font(otherHeadersBox->font()); font.setBold(false); otherHeadersBox->setFont(font); otherHeadersBox->setWidget(frameOtherHeaders); frameOtherHeaders->setContentsMargins(10, 10, 10, 10); QGridLayout *layoutOtherHeaders = new QGridLayout(frameOtherHeaders); layoutOtherHeaders->setSpacing(5); m_metadata = new QTableWidget( 2, 2, frameOtherHeaders ); // rows, columns m_metadata->setObjectName("StyledTable"); m_metadata->setAlternatingRowColors(true); m_metadata->setHorizontalHeaderItem( 0, new QTableWidgetItem(tr("Name")) ); // column, item m_metadata->setHorizontalHeaderItem( 1, new QTableWidgetItem(tr("Value")) ); // column, item // m_metadata->setFullWidth(true); m_metadata->setMinimumSize(40, 40); // width, height // m_metadata->setItemsRenameable(true); //&&& disabled item renaming ||use: openPersistentEditor (QTableWidgetItem * item) // m_metadata->setRenameable(0); // m_metadata->setRenameable(1); // m_metadata->setItemMargin(5); // m_metadata->setDefaultRenameAction(QListWidget::Accept); // m_metadata->setShowSortIndicator(true); std::vector names(metadata.getPropertyNames()); QTableWidgetItem* tabItem; int row = m_metadata->rowCount(); int col = 0; m_metadata->setRowCount(row + 1); for (unsigned int i = 0; i < names.size(); ++i) { if (shown.find(names[i]) != shown.end()) continue; QString name(strtoqstr(names[i])); // property names stored in lower case name = name.left(1).toUpper() + name.right(name.length() - 1); // new QListWidgetItem(m_metadata, name, strtoqstr(metadata.get(names[i]))); // qt4: icon, text, type tabItem = new QTableWidgetItem(strtoqstr(metadata.get(names[i]))); m_metadata->setItem(row, col, tabItem); shown.insert(names[i]); } layoutOtherHeaders->addWidget(m_metadata, 0, 0, 0- 0+1, 1-0+ 1); QPushButton* addPropButton = new QPushButton(tr("Add New Property"), frameOtherHeaders); layoutOtherHeaders->addWidget(addPropButton, 1, 0, Qt::AlignHCenter); QPushButton* deletePropButton = new QPushButton(tr("Delete Property"), frameOtherHeaders); layoutOtherHeaders->addWidget(deletePropButton, 1, 1, Qt::AlignHCenter); frameOtherHeaders->setLayout(layoutOtherHeaders); connect(addPropButton, SIGNAL(clicked()), this, SLOT(slotAddNewProperty())); connect(deletePropButton, SIGNAL(clicked()), this, SLOT(slotDeleteProperty())); } void HeadersConfigurationPage::slotAddNewProperty() { QString propertyName; int i = 0; QList foundItems; while (1) { propertyName = (i > 0 ? tr("{new property %1}").arg(i) : tr("{new property}")); // foundItems = m_metadata->findItems(propertyName, Qt::MatchContains | Qt::MatchCaseSensitive); if (!m_doc->getComposition().getMetadata().has(qstrtostr(propertyName)) && foundItems.isEmpty()){ break; } ++i; } //new QTableWidgetItem(m_metadata, propertyName, tr("{undefined}")); int rc = m_metadata->rowCount(); int col = 0; m_metadata->setRowCount(rc + 1); m_metadata->setItem(rc, col, new QTableWidgetItem(propertyName)); } void HeadersConfigurationPage::slotDeleteProperty() { //delete m_metadata->currentIndex(); m_metadata->removeRow(m_metadata->currentRow()); } void HeadersConfigurationPage::apply() { //@@@ Next two lines not needed in this function. Commented out //@@@ QSettings settings; //@@@ settings.beginGroup(NotationViewConfigGroup); // If one of the items still has focus, it won't remember edits. // Switch between two fields in order to lose the current focus. m_editTitle->setFocus(); m_metadata->setFocus(); // // Update header fields // Configuration &metadata = (&m_doc->getComposition())->getMetadata(); metadata.clear(); metadata.set(CompositionMetadataKeys::Dedication, qstrtostr(m_editDedication->text())); metadata.set(CompositionMetadataKeys::Title, qstrtostr(m_editTitle->text())); metadata.set(CompositionMetadataKeys::Subtitle, qstrtostr(m_editSubtitle->text())); metadata.set(CompositionMetadataKeys::Subsubtitle, qstrtostr(m_editSubsubtitle->text())); metadata.set(CompositionMetadataKeys::Poet, qstrtostr(m_editPoet->text())); metadata.set(CompositionMetadataKeys::Composer, qstrtostr(m_editComposer->text())); metadata.set(CompositionMetadataKeys::Meter, qstrtostr(m_editMeter->text())); metadata.set(CompositionMetadataKeys::Opus, qstrtostr(m_editOpus->text())); metadata.set(CompositionMetadataKeys::Arranger, qstrtostr(m_editArranger->text())); metadata.set(CompositionMetadataKeys::Instrument, qstrtostr(m_editInstrument->text())); metadata.set(CompositionMetadataKeys::Piece, qstrtostr(m_editPiece->text())); metadata.set(CompositionMetadataKeys::Copyright, qstrtostr(m_editCopyright->text())); metadata.set(CompositionMetadataKeys::Tagline, qstrtostr(m_editTagline->text())); // for (QTableWidgetItem *item = m_metadata->firstChild(); // item != 0; item = item->nextSibling()) { QTableWidgetItem* tabItem; QTableWidgetItem* tabItem2; int c = 0; for(int r=0; r < m_metadata->rowCount(); r++){ //for(int c=0; c < m_metadata->columnCount(); c++){ tabItem = m_metadata->item(r, c); tabItem2 = m_metadata->item(r, c+1); if((!tabItem) || (!tabItem2)){ RG_DEBUG << "ERROR: Any TableWidgetItem is NULL in HeadersConfigurationPage::apply() " << endl; continue; } metadata.set(qstrtostr(tabItem->text().toLower()), qstrtostr(tabItem2->text())); // metadata.set(qstrtostr(item->text(0).toLower()), // qstrtostr(item->text(1))); //}//end for(c) }//end for(r) m_doc->slotDocumentModified(); } } #include "HeadersConfigurationPage.moc"