You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
rosegarden/src/gui/editors/tempo/TempoView.cpp

840 lines
23 KiB

/* -*- 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.
This program is Copyright 2000-2008
Guillaume Laurent <glaurent@telegraph-road.org>,
Chris Cannam <cannam@all-day-breakfast.com>,
Richard Bown <richard.bown@ferventsoftware.com>
The moral rights of Guillaume Laurent, Chris Cannam, and Richard
Bown to claim authorship of this work have been asserted.
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 "TempoView.h"
#include <klocale.h>
#include <kstddirs.h>
#include "misc/Debug.h"
#include "base/Composition.h"
#include "base/NotationTypes.h"
#include "base/RealTime.h"
#include "base/Segment.h"
#include "commands/segment/AddTempoChangeCommand.h"
#include "commands/segment/AddTimeSignatureAndNormalizeCommand.h"
#include "commands/segment/AddTimeSignatureCommand.h"
#include "commands/segment/RemoveTempoChangeCommand.h"
#include "commands/segment/RemoveTimeSignatureCommand.h"
#include "document/RosegardenGUIDoc.h"
#include "document/ConfigGroups.h"
#include "gui/dialogs/TempoDialog.h"
#include "gui/dialogs/TimeSignatureDialog.h"
#include "gui/general/EditViewBase.h"
#include "gui/kdeext/KTmpStatusMsg.h"
#include "TempoListItem.h"
#include <tdeaction.h>
#include <kglobal.h>
#include <tdeconfig.h>
#include <tdelistview.h>
#include <kxmlguiclient.h>
#include <tqbuttongroup.h>
#include <tqcheckbox.h>
#include <tqdialog.h>
#include <tqiconset.h>
#include <tqlistview.h>
#include <tqpixmap.h>
#include <tqptrlist.h>
#include <tqsize.h>
#include <tqstring.h>
#include <tqlayout.h>
#include <tqcanvas.h>
#include <kstatusbar.h>
namespace Rosegarden
{
int
TempoView::m_lastSetFilter = -1;
TempoView::TempoView(RosegardenGUIDoc *doc, TQWidget *parent, timeT openTime):
EditViewBase(doc, std::vector<Segment *>(), 2, parent, "tempoview"),
m_filter(Tempo | TimeSignature),
m_ignoreUpdates(true)
{
if (m_lastSetFilter < 0)
m_lastSetFilter = m_filter;
else
m_filter = m_lastSetFilter;
initStatusBar();
setupActions();
// define some note filtering buttons in a group
//
m_filterGroup =
new TQButtonGroup(1, Qt::Horizontal, i18n("Filter"), getCentralWidget());
m_tempoCheckBox = new TQCheckBox(i18n("Tempo"), m_filterGroup);
m_timeSigCheckBox = new TQCheckBox(i18n("Time Signature"), m_filterGroup);
m_grid->addWidget(m_filterGroup, 2, 0);
// Connect up
//
connect(m_filterGroup, TQT_SIGNAL(released(int)),
TQT_SLOT(slotModifyFilter(int)));
m_list = new TDEListView(getCentralWidget());
m_list->setItemsRenameable(true);
m_grid->addWidget(m_list, 2, 1);
updateViewCaption();
doc->getComposition().addObserver(this);
// Connect double clicker
//
connect(m_list, TQT_SIGNAL(doubleClicked(TQListViewItem*)),
TQT_SLOT(slotPopupEditor(TQListViewItem*)));
m_list->setAllColumnsShowFocus(true);
m_list->setSelectionMode(TQListView::Extended);
m_list->addColumn(i18n("Time "));
m_list->addColumn(i18n("Type "));
m_list->addColumn(i18n("Value "));
m_list->addColumn(i18n("Properties "));
for (int col = 0; col < m_list->columns(); ++col)
m_list->setRenameable(col, true);
readOptions();
setButtonsToFilter();
applyLayout();
makeInitialSelection(openTime);
m_ignoreUpdates = false;
setOutOfCtor();
}
TempoView::~TempoView()
{
if (!getDocument()->isBeingDestroyed() && !isCompositionDeleted()) {
getDocument()->getComposition().removeObserver(this);
}
}
void
TempoView::closeEvent(TQCloseEvent *e)
{
emit closing();
EditViewBase::closeEvent(e);
}
void
TempoView::tempoChanged(const Composition *comp)
{
if (m_ignoreUpdates)
return ;
if (comp == &getDocument()->getComposition()) {
applyLayout();
}
}
void
TempoView::timeSignatureChanged(const Composition *comp)
{
if (m_ignoreUpdates)
return ;
if (comp == &getDocument()->getComposition()) {
applyLayout();
}
}
bool
TempoView::applyLayout(int /*staffNo*/)
{
// If no selection has already been set then we copy what's
// already set and try to replicate this after the rebuild
// of the view. This code borrowed from EventView.
//
if (m_listSelection.size() == 0) {
TQPtrList<TQListViewItem> selection = m_list->selectedItems();
if (selection.count()) {
TQPtrListIterator<TQListViewItem> it(selection);
TQListViewItem *listItem;
while ((listItem = it.current()) != 0) {
m_listSelection.push_back(m_list->itemIndex(*it));
++it;
}
}
}
// Ok, recreate list
//
m_list->clear();
Composition *comp = &getDocument()->getComposition();
m_config->setGroup(TempoViewConfigGroup);
int timeMode = m_config->readNumEntry("timemode", 0);
if (m_filter & TimeSignature) {
for (int i = 0; i < comp->getTimeSignatureCount(); ++i) {
std::pair<timeT, Rosegarden::TimeSignature> sig =
comp->getTimeSignatureChange(i);
TQString properties;
if (sig.second.isHidden()) {
if (sig.second.isCommon())
properties = i18n("Common, hidden");
else
properties = i18n("Hidden");
} else {
if (sig.second.isCommon())
properties = i18n("Common");
}
TQString timeString = makeTimeString(sig.first, timeMode);
new TempoListItem(comp, TempoListItem::TimeSignature,
sig.first, i, m_list, timeString,
i18n("Time Signature "),
TQString("%1/%2 ").arg(sig.second.getNumerator()).
arg(sig.second.getDenominator()),
properties);
}
}
if (m_filter & Tempo) {
for (int i = 0; i < comp->getTempoChangeCount(); ++i) {
std::pair<timeT, tempoT> tempo =
comp->getTempoChange(i);
TQString desc;
//!!! imprecise -- better to work from tempoT directly
float qpm = comp->getTempoQpm(tempo.second);
int qpmUnits = int(qpm + 0.001);
int qpmTenths = int((qpm - qpmUnits) * 10 + 0.001);
int qpmHundredths = int((qpm - qpmUnits - qpmTenths / 10.0) * 100 + 0.001);
Rosegarden::TimeSignature sig = comp->getTimeSignatureAt(tempo.first);
if (sig.getBeatDuration() ==
Note(Note::Crotchet).getDuration()) {
desc = i18n("%1.%2%3").
arg(qpmUnits).arg(qpmTenths).arg(qpmHundredths);
} else {
float bpm = (qpm *
Note(Note::Crotchet).getDuration()) /
sig.getBeatDuration();
int bpmUnits = int(bpm + 0.001);
int bpmTenths = int((bpm - bpmUnits) * 10 + 0.001);
int bpmHundredths = int((bpm - bpmUnits - bpmTenths / 10.0) * 100 + 0.001);
desc = i18n("%1.%2%3 qpm (%4.%5%6 bpm) ").
arg(qpmUnits).arg(qpmTenths).arg(qpmHundredths).
arg(bpmUnits).arg(bpmTenths).arg(bpmHundredths);
}
TQString timeString = makeTimeString(tempo.first, timeMode);
new TempoListItem(comp, TempoListItem::Tempo,
tempo.first, i, m_list, timeString,
i18n("Tempo "),
desc);
}
}
if (m_list->childCount() == 0) {
new TQListViewItem(m_list,
i18n("<nothing at this filter level>"));
m_list->setSelectionMode(TQListView::NoSelection);
stateChanged("have_selection", KXMLGUIClient::StateReverse);
} else {
m_list->setSelectionMode(TQListView::Extended);
// If no selection then select the first event
if (m_listSelection.size() == 0)
m_listSelection.push_back(0);
stateChanged("have_selection", KXMLGUIClient::StateNoReverse);
}
// Set a selection from a range of indexes
//
std::vector<int>::iterator sIt = m_listSelection.begin();
int index = 0;
for (; sIt != m_listSelection.end(); ++sIt) {
index = *sIt;
while (index > 0 && !m_list->itemAtIndex(index))
index--;
m_list->setSelected(m_list->itemAtIndex(index), true);
m_list->setCurrentItem(m_list->itemAtIndex(index));
// ensure visible
m_list->ensureItemVisible(m_list->itemAtIndex(index));
}
m_listSelection.clear();
return true;
}
void
TempoView::makeInitialSelection(timeT time)
{
m_listSelection.clear();
TempoListItem *goodItem = 0;
int goodItemNo = 0;
for (int i = 0; m_list->itemAtIndex(i); ++i) {
TempoListItem *item = dynamic_cast<TempoListItem *>
(m_list->itemAtIndex(i));
m_list->setSelected(item, false);
if (item) {
if (item->getTime() > time)
break;
goodItem = item;
goodItemNo = i;
}
}
if (goodItem) {
m_listSelection.push_back(goodItemNo);
m_list->setSelected(goodItem, true);
m_list->ensureItemVisible(goodItem);
}
}
Segment *
TempoView::getCurrentSegment()
{
if (m_segments.empty())
return 0;
else
return *m_segments.begin();
}
TQString
TempoView::makeTimeString(timeT time, int timeMode)
{
switch (timeMode) {
case 0: // musical time
{
int bar, beat, fraction, remainder;
getDocument()->getComposition().getMusicalTimeForAbsoluteTime
(time, bar, beat, fraction, remainder);
++bar;
return TQString("%1%2%3-%4%5-%6%7-%8%9 ")
.arg(bar / 100)
.arg((bar % 100) / 10)
.arg(bar % 10)
.arg(beat / 10)
.arg(beat % 10)
.arg(fraction / 10)
.arg(fraction % 10)
.arg(remainder / 10)
.arg(remainder % 10);
}
case 1: // real time
{
RealTime rt =
getDocument()->getComposition().getElapsedRealTime(time);
// return TQString("%1 ").arg(rt.toString().c_str());
return TQString("%1 ").arg(rt.toText().c_str());
}
default:
return TQString("%1 ").arg(time);
}
}
void
TempoView::refreshSegment(Segment * /*segment*/,
timeT /*startTime*/,
timeT /*endTime*/)
{
RG_DEBUG << "TempoView::refreshSegment" << endl;
applyLayout(0);
}
void
TempoView::updateView()
{
m_list->update();
}
void
TempoView::slotEditCut()
{
// not implemented yet -- can't use traditional clipboard (which
// only holds events from segments, or segments)
}
void
TempoView::slotEditCopy()
{
// likewise
}
void
TempoView::slotEditPaste()
{
// likewise
}
void
TempoView::slotEditDelete()
{
TQPtrList<TQListViewItem> selection = m_list->selectedItems();
if (selection.count() == 0)
return ;
RG_DEBUG << "TempoView::slotEditDelete - deleting "
<< selection.count() << " items" << endl;
TQPtrListIterator<TQListViewItem> it(selection);
TQListViewItem *listItem;
TempoListItem *item;
int itemIndex = -1;
m_ignoreUpdates = true;
bool haveSomething = false;
// We want the Remove commands to be in reverse order, because
// removing one item by index will affect the indices of
// subsequent items. So we'll stack them onto here and then pull
// them off again.
std::vector<KCommand *> commands;
while ((listItem = it.current()) != 0) {
item = dynamic_cast<TempoListItem*>((*it));
if (itemIndex == -1)
itemIndex = m_list->itemIndex(*it);
if (item) {
if (item->getType() == TempoListItem::TimeSignature) {
commands.push_back(new RemoveTimeSignatureCommand
(item->getComposition(),
item->getIndex()));
haveSomething = true;
} else {
commands.push_back(new RemoveTempoChangeCommand
(item->getComposition(),
item->getIndex()));
haveSomething = true;
}
}
++it;
}
if (haveSomething) {
KMacroCommand *command = new KMacroCommand
(i18n("Delete Tempo or Time Signature"));
for (std::vector<KCommand *>::iterator i = commands.end();
i != commands.begin();) {
command->addCommand(*--i);
}
addCommandToHistory(command);
}
applyLayout();
m_ignoreUpdates = false;
}
void
TempoView::slotEditInsertTempo()
{
timeT insertTime = 0;
TQPtrList<TQListViewItem> selection = m_list->selectedItems();
if (selection.count() > 0) {
TempoListItem *item =
dynamic_cast<TempoListItem*>(selection.getFirst());
if (item)
insertTime = item->getTime();
}
TempoDialog dialog(this, getDocument(), true);
dialog.setTempoPosition(insertTime);
connect(&dialog,
TQT_SIGNAL(changeTempo(timeT,
tempoT,
tempoT,
TempoDialog::TempoDialogAction)),
this,
TQT_SIGNAL(changeTempo(timeT,
tempoT,
tempoT,
TempoDialog::TempoDialogAction)));
dialog.exec();
}
void
TempoView::slotEditInsertTimeSignature()
{
timeT insertTime = 0;
TQPtrList<TQListViewItem> selection = m_list->selectedItems();
if (selection.count() > 0) {
TempoListItem *item =
dynamic_cast<TempoListItem*>(selection.getFirst());
if (item)
insertTime = item->getTime();
}
Composition &composition(m_doc->getComposition());
Rosegarden::TimeSignature sig = composition.getTimeSignatureAt(insertTime);
TimeSignatureDialog dialog(this, &composition, insertTime, sig, true);
if (dialog.exec() == TQDialog::Accepted) {
insertTime = dialog.getTime();
if (dialog.shouldNormalizeRests()) {
addCommandToHistory
(new AddTimeSignatureAndNormalizeCommand
(&composition, insertTime, dialog.getTimeSignature()));
} else {
addCommandToHistory
(new AddTimeSignatureCommand
(&composition, insertTime, dialog.getTimeSignature()));
}
}
}
void
TempoView::slotEdit()
{
RG_DEBUG << "TempoView::slotEdit" << endl;
TQPtrList<TQListViewItem> selection = m_list->selectedItems();
if (selection.count() > 0) {
TempoListItem *item =
dynamic_cast<TempoListItem*>(selection.getFirst());
if (item)
slotPopupEditor(item);
}
}
void
TempoView::slotSelectAll()
{
m_listSelection.clear();
for (int i = 0; m_list->itemAtIndex(i); ++i) {
m_listSelection.push_back(i);
m_list->setSelected(m_list->itemAtIndex(i), true);
}
}
void
TempoView::slotClearSelection()
{
m_listSelection.clear();
for (int i = 0; m_list->itemAtIndex(i); ++i) {
m_list->setSelected(m_list->itemAtIndex(i), false);
}
}
void
TempoView::setupActions()
{
EditViewBase::setupActions("tempoview.rc", false);
TQString pixmapDir = TDEGlobal::dirs()->findResource("appdata", "pixmaps/");
TQIconSet icon(TQPixmap(pixmapDir + "/toolbar/event-insert-tempo.png"));
new TDEAction(AddTempoChangeCommand::getGlobalName(), icon, Key_I, TQT_TQOBJECT(this),
TQT_SLOT(slotEditInsertTempo()), actionCollection(),
"insert_tempo");
TQCanvasPixmap pixmap(pixmapDir + "/toolbar/event-insert-timesig.png");
icon = TQIconSet(pixmap);
new TDEAction(AddTimeSignatureCommand::getGlobalName(), icon, Key_G, TQT_TQOBJECT(this),
TQT_SLOT(slotEditInsertTimeSignature()), actionCollection(),
"insert_timesig");
pixmap.load(pixmapDir + "/toolbar/event-delete.png");
icon = TQIconSet(pixmap);
new TDEAction(i18n("&Delete"), icon, Key_Delete, TQT_TQOBJECT(this),
TQT_SLOT(slotEditDelete()), actionCollection(),
"delete");
pixmap.load(pixmapDir + "/toolbar/event-edit.png");
icon = TQIconSet(pixmap);
new TDEAction(i18n("&Edit Item"), icon, Key_E, TQT_TQOBJECT(this),
TQT_SLOT(slotEdit()), actionCollection(),
"edit");
new TDEAction(i18n("Select &All"), 0, TQT_TQOBJECT(this),
TQT_SLOT(slotSelectAll()), actionCollection(),
"select_all");
new TDEAction(i18n("Clear Selection"), Key_Escape, TQT_TQOBJECT(this),
TQT_SLOT(slotClearSelection()), actionCollection(),
"clear_selection");
m_config->setGroup(TempoViewConfigGroup);
int timeMode = m_config->readNumEntry("timemode", 0);
TDERadioAction *action;
pixmap.load(pixmapDir + "/toolbar/time-musical.png");
icon = TQIconSet(pixmap);
action = new TDERadioAction(i18n("&Musical Times"), icon, 0, TQT_TQOBJECT(this),
TQT_SLOT(slotMusicalTime()),
actionCollection(), "time_musical");
action->setExclusiveGroup("timeMode");
if (timeMode == 0)
action->setChecked(true);
pixmap.load(pixmapDir + "/toolbar/time-real.png");
icon = TQIconSet(pixmap);
action = new TDERadioAction(i18n("&Real Times"), icon, 0, TQT_TQOBJECT(this),
TQT_SLOT(slotRealTime()),
actionCollection(), "time_real");
action->setExclusiveGroup("timeMode");
if (timeMode == 1)
action->setChecked(true);
pixmap.load(pixmapDir + "/toolbar/time-raw.png");
icon = TQIconSet(pixmap);
action = new TDERadioAction(i18n("Ra&w Times"), icon, 0, TQT_TQOBJECT(this),
TQT_SLOT(slotRawTime()),
actionCollection(), "time_raw");
action->setExclusiveGroup("timeMode");
if (timeMode == 2)
action->setChecked(true);
createGUI(getRCFileName());
}
void
TempoView::initStatusBar()
{
KStatusBar* sb = statusBar();
sb->insertItem(KTmpStatusMsg::getDefaultMsg(),
KTmpStatusMsg::getDefaultId(), 1);
sb->setItemAlignment(KTmpStatusMsg::getDefaultId(),
AlignLeft | AlignVCenter);
}
TQSize
TempoView::getViewSize()
{
return m_list->size();
}
void
TempoView::setViewSize(TQSize s)
{
m_list->setFixedSize(s);
}
void
TempoView::readOptions()
{
m_config->setGroup(TempoViewConfigGroup);
EditViewBase::readOptions();
m_filter = m_config->readNumEntry("filter", m_filter);
m_list->restoreLayout(m_config, TempoViewLayoutConfigGroupName);
}
void
TempoView::slotSaveOptions()
{
m_config->setGroup(TempoViewConfigGroup);
m_config->writeEntry("filter", m_filter);
m_list->saveLayout(m_config, TempoViewLayoutConfigGroupName);
}
void
TempoView::slotModifyFilter(int button)
{
TQCheckBox *checkBox = dynamic_cast<TQCheckBox*>(m_filterGroup->find(button));
if (checkBox == 0)
return ;
if (checkBox->isChecked()) {
switch (button) {
case 0:
m_filter |= Tempo;
break;
case 1:
m_filter |= TimeSignature;
break;
default:
break;
}
} else {
switch (button) {
case 0:
m_filter ^= Tempo;
break;
case 1:
m_filter ^= TimeSignature;
break;
default:
break;
}
}
m_lastSetFilter = m_filter;
applyLayout(0);
}
void
TempoView::setButtonsToFilter()
{
if (m_filter & Tempo)
m_tempoCheckBox->setChecked(true);
else
m_tempoCheckBox->setChecked(false);
if (m_filter & TimeSignature)
m_timeSigCheckBox->setChecked(true);
else
m_timeSigCheckBox->setChecked(false);
}
void
TempoView::slotMusicalTime()
{
m_config->setGroup(TempoViewConfigGroup);
m_config->writeEntry("timemode", 0);
applyLayout();
}
void
TempoView::slotRealTime()
{
m_config->setGroup(TempoViewConfigGroup);
m_config->writeEntry("timemode", 1);
applyLayout();
}
void
TempoView::slotRawTime()
{
m_config->setGroup(TempoViewConfigGroup);
m_config->writeEntry("timemode", 2);
applyLayout();
}
void
TempoView::slotPopupEditor(TQListViewItem *qitem)
{
TempoListItem *item = dynamic_cast<TempoListItem *>(qitem);
if (!item)
return ;
timeT time = item->getTime();
switch (item->getType()) {
case TempoListItem::Tempo:
{
TempoDialog dialog(this, getDocument(), true);
dialog.setTempoPosition(time);
connect(&dialog,
TQT_SIGNAL(changeTempo(timeT,
tempoT,
tempoT,
TempoDialog::TempoDialogAction)),
this,
TQT_SIGNAL(changeTempo(timeT,
tempoT,
tempoT,
TempoDialog::TempoDialogAction)));
dialog.exec();
break;
}
case TempoListItem::TimeSignature:
{
Composition &composition(getDocument()->getComposition());
Rosegarden::TimeSignature sig = composition.getTimeSignatureAt(time);
TimeSignatureDialog dialog(this, &composition, time, sig, true);
if (dialog.exec() == TQDialog::Accepted) {
time = dialog.getTime();
if (dialog.shouldNormalizeRests()) {
addCommandToHistory
(new AddTimeSignatureAndNormalizeCommand
(&composition, time, dialog.getTimeSignature()));
} else {
addCommandToHistory
(new AddTimeSignatureCommand
(&composition, time, dialog.getTimeSignature()));
}
}
}
default:
break;
}
}
void
TempoView::updateViewCaption()
{
setCaption(i18n("%1 - Tempo and Time Signature Editor")
.arg(getDocument()->getTitle()));
}
}
#include "TempoView.moc"