# HG changeset patch # User Chris Cannam # Date 1159183272 0 # Node ID 8f51db2434dcc6412523f2d5c9f265fd2d6bcb06 # Parent b1a3a9400284431343097870374b7fb3f85bd333 * Pull alphabetical categorisation code out into a SubdividingMenu class diff -r b1a3a9400284 -r 8f51db2434dc widgets/SubdividingMenu.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/widgets/SubdividingMenu.cpp Mon Sep 25 11:21:12 2006 +0000 @@ -0,0 +1,194 @@ +/* -*- 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 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 + +using std::set; +using std::map; + +SubdividingMenu::SubdividingMenu(QWidget *parent) : + QMenu(parent) +{ +} + +SubdividingMenu::SubdividingMenu(const QString &title, QWidget *parent) : + QMenu(title, parent) +{ +} + +SubdividingMenu::~SubdividingMenu() +{ +} + +void +SubdividingMenu::setEntries(const std::set &entries) +{ + size_t total = entries.size(); + size_t chunk = 14; + + if (total < (chunk * 3) / 2) return; + + size_t count = 0; + QMenu *chunkMenu = new QMenu(); + + QString firstNameInChunk; + QChar firstInitialInChunk; + bool discriminateStartInitial = false; + + for (set::const_iterator j = entries.begin(); + j != entries.end(); + ++j) { + + std::cerr << "SubdividingMenu::setEntries: j -> " << j->toStdString() << std::endl; + + m_nameToChunkMenuMap[*j] = chunkMenu; + + set::iterator k = j; + ++k; + + QChar initial = (*j)[0]; + + if (count == 0) { + firstNameInChunk = *j; + firstInitialInChunk = initial; + } + + bool lastInChunk = (k == entries.end() || + (count >= chunk-1 && + (count == (5*chunk) / 2 || + (*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(); + + count = 0; + } + } + + if (count == 0) delete chunkMenu; +} + +void +SubdividingMenu::addAction(QAction *action) +{ + QString name = action->text(); + + 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_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_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_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_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_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); +} + diff -r b1a3a9400284 -r 8f51db2434dc widgets/SubdividingMenu.h --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/widgets/SubdividingMenu.h Mon Sep 25 11:21:12 2006 +0000 @@ -0,0 +1,63 @@ +/* -*- 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 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. +*/ + +#ifndef _SUBDIVIDING_MENU_H_ +#define _SUBDIVIDING_MENU_H_ + +#include + +#include +#include +#include + +/** + * A menu that divides its entries into submenus, alphabetically. For + * menus that may contain a very large or small number of named items + * (e.g. plugins). + * + * The menu needs to be told, before any of the actions are added, + * what the set of entry strings will be, so it can determine a + * reasonable categorisation. Do this by calling the setEntries() + * method. + */ + +class SubdividingMenu : public QMenu +{ +public: + SubdividingMenu(QWidget *parent = 0); + SubdividingMenu(const QString &title, QWidget *parent = 0); + virtual ~SubdividingMenu(); + + void setEntries(const std::set &entries); + + // Action names and strings passed to addAction and addMenu must + // appear in the set previously given to setEntries. If you want + // to use a different string, use the two-argument method and pass + // the entry string (used to determine which submenu the action + // ends up on) as the first argument. + + virtual void addAction(QAction *); + virtual QAction *addAction(const QString &); + virtual void addAction(const QString &entry, QAction *); + + virtual void addMenu(QMenu *); + virtual QMenu *addMenu(const QString &); + virtual void addMenu(const QString &entry, QMenu *); + +protected: + std::map m_nameToChunkMenuMap; +}; + +#endif + diff -r b1a3a9400284 -r 8f51db2434dc widgets/widgets.pro --- a/widgets/widgets.pro Fri Sep 22 16:46:10 2006 +0000 +++ b/widgets/widgets.pro Mon Sep 25 11:21:12 2006 +0000 @@ -24,6 +24,7 @@ PluginParameterDialog.h \ PropertyBox.h \ PropertyStack.h \ + SubdividingMenu.h \ Thumbwheel.h \ WindowShapePreview.h \ WindowTypeSelector.h @@ -37,6 +38,7 @@ PluginParameterDialog.cpp \ PropertyBox.cpp \ PropertyStack.cpp \ + SubdividingMenu.cpp \ Thumbwheel.cpp \ WindowShapePreview.cpp \ WindowTypeSelector.cpp