Mercurial > hg > easaier-soundaccess
diff widgets/SubdividingMenu.cpp @ 0:fc9323a41f5a
start base : Sonic Visualiser sv1-1.0rc1
author | lbajardsilogic |
---|---|
date | Fri, 11 May 2007 09:08:14 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/widgets/SubdividingMenu.cpp Fri May 11 09:08:14 2007 +0000 @@ -0,0 +1,280 @@ +/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ + +/* + Sonic Visualiser + An audio file viewer and annotation editor. + Centre for Digital Music, Queen Mary, University of London. + This file copyright 2006 QMUL. + + 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 "SubdividingMenu.h" + +#include <iostream> + +using std::set; +using std::map; + +SubdividingMenu::SubdividingMenu(size_t lowerLimit, size_t upperLimit, + QWidget *parent) : + QMenu(parent), + m_lowerLimit(lowerLimit ? lowerLimit : 14), + m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), + m_entriesSet(false) +{ +} + +SubdividingMenu::SubdividingMenu(const QString &title, size_t lowerLimit, + size_t upperLimit, QWidget *parent) : + QMenu(title, parent), + m_lowerLimit(lowerLimit ? lowerLimit : 14), + m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), + m_entriesSet(false) +{ +} + +SubdividingMenu::~SubdividingMenu() +{ + for (map<QString, QObject *>::iterator i = m_pendingEntries.begin(); + i != m_pendingEntries.end(); ++i) { + delete i->second; + } +} + +void +SubdividingMenu::setEntries(const std::set<QString> &entries) +{ + m_entriesSet = true; + + size_t total = entries.size(); + + if (total < m_upperLimit) return; + + size_t count = 0; + QMenu *chunkMenu = new QMenu(); + chunkMenu->setTearOffEnabled(isTearOffEnabled()); + + QString firstNameInChunk; + QChar firstInitialInChunk; + bool discriminateStartInitial = false; + + for (set<QString>::const_iterator j = entries.begin(); + j != entries.end(); + ++j) { + +// std::cerr << "SubdividingMenu::setEntries: j -> " << j->toStdString() << std::endl; + + m_nameToChunkMenuMap[*j] = chunkMenu; + + set<QString>::const_iterator k = j; + ++k; + + QChar initial = (*j)[0]; + + if (count == 0) { + firstNameInChunk = *j; + firstInitialInChunk = initial; + } + +// std::cerr << "count = "<< count << ", upper limit = " << m_upperLimit << std::endl; + + bool lastInChunk = (k == entries.end() || + (count >= m_lowerLimit-1 && + (count == m_upperLimit || + (*k)[0] != initial))); + + ++count; + + if (lastInChunk) { + + bool discriminateEndInitial = (k != entries.end() && + (*k)[0] == initial); + + bool initialsEqual = (firstInitialInChunk == initial); + + QString from = QString("%1").arg(firstInitialInChunk); + if (discriminateStartInitial || + (discriminateEndInitial && initialsEqual)) { + from = firstNameInChunk.left(3); + } + + QString to = QString("%1").arg(initial); + if (discriminateEndInitial || + (discriminateStartInitial && initialsEqual)) { + to = j->left(3); + } + + QString menuText; + + if (from == to) menuText = from; + else menuText = tr("%1 - %2").arg(from).arg(to); + + discriminateStartInitial = discriminateEndInitial; + + chunkMenu->setTitle(menuText); + + QMenu::addMenu(chunkMenu); + + chunkMenu = new QMenu(); + chunkMenu->setTearOffEnabled(isTearOffEnabled()); + + count = 0; + } + } + + if (count == 0) delete chunkMenu; +} + +void +SubdividingMenu::entriesAdded() +{ + if (m_entriesSet) { + std::cerr << "ERROR: SubdividingMenu::entriesAdded: setEntries was also called -- should use one mechanism or the other, but not both" << std::endl; + return; + } + + set<QString> entries; + for (map<QString, QObject *>::const_iterator i = m_pendingEntries.begin(); + i != m_pendingEntries.end(); ++i) { + entries.insert(i->first); + } + + setEntries(entries); + + for (map<QString, QObject *>::iterator i = m_pendingEntries.begin(); + i != m_pendingEntries.end(); ++i) { + + QMenu *menu = dynamic_cast<QMenu *>(i->second); + if (menu) { + addMenu(i->first, menu); + continue; + } + + QAction *action = dynamic_cast<QAction *>(i->second); + if (action) { + addAction(i->first, action); + continue; + } + } + + m_pendingEntries.clear(); +} + +void +SubdividingMenu::addAction(QAction *action) +{ + QString name = action->text(); + + if (!m_entriesSet) { + m_pendingEntries[name] = action; + return; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + QMenu::addAction(action); + return; + } + +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + m_nameToChunkMenuMap[name]->addAction(action); +} + +QAction * +SubdividingMenu::addAction(const QString &name) +{ + if (!m_entriesSet) { + QAction *action = new QAction(name, this); + m_pendingEntries[name] = action; + return action; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + return QMenu::addAction(name); + } + +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + return m_nameToChunkMenuMap[name]->addAction(name); +} + +void +SubdividingMenu::addAction(const QString &name, QAction *action) +{ + if (!m_entriesSet) { + m_pendingEntries[name] = action; + return; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + QMenu::addAction(action); + return; + } + +// std::cerr << "SubdividingMenu::addAction(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + m_nameToChunkMenuMap[name]->addAction(action); +} + +void +SubdividingMenu::addMenu(QMenu *menu) +{ + QString name = menu->title(); + + if (!m_entriesSet) { + m_pendingEntries[name] = menu; + return; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + QMenu::addMenu(menu); + return; + } + +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + m_nameToChunkMenuMap[name]->addMenu(menu); +} + +QMenu * +SubdividingMenu::addMenu(const QString &name) +{ + if (!m_entriesSet) { + QMenu *menu = new QMenu(name, this); + menu->setTearOffEnabled(isTearOffEnabled()); + m_pendingEntries[name] = menu; + return menu; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + return QMenu::addMenu(name); + } + +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + return m_nameToChunkMenuMap[name]->addMenu(name); +} + +void +SubdividingMenu::addMenu(const QString &name, QMenu *menu) +{ + if (!m_entriesSet) { + m_pendingEntries[name] = menu; + return; + } + + if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): not found in name-to-chunk map, adding to main menu" << std::endl; + QMenu::addMenu(menu); + return; + } + +// std::cerr << "SubdividingMenu::addMenu(" << name.toStdString() << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title().toStdString() << std::endl; + m_nameToChunkMenuMap[name]->addMenu(menu); +} +