Mercurial > hg > svgui
comparison widgets/SubdividingMenu.cpp @ 1347:edfc38ade098
Use locale-aware comparators for sorting user-visible strings
author | Chris Cannam |
---|---|
date | Mon, 01 Oct 2018 14:37:58 +0100 |
parents | 4a578a360011 |
children | 53fe33b00770 |
comparison
equal
deleted
inserted
replaced
1323:8068a0bee550 | 1347:edfc38ade098 |
---|---|
20 #include "base/Debug.h" | 20 #include "base/Debug.h" |
21 | 21 |
22 using std::set; | 22 using std::set; |
23 using std::map; | 23 using std::map; |
24 | 24 |
25 //#define DEBUG_SUBDIVIDING_MENU 1 | |
26 | |
25 SubdividingMenu::SubdividingMenu(int lowerLimit, int upperLimit, | 27 SubdividingMenu::SubdividingMenu(int lowerLimit, int upperLimit, |
26 QWidget *parent) : | 28 QWidget *parent) : |
27 QMenu(parent), | 29 QMenu(parent), |
28 m_lowerLimit(lowerLimit ? lowerLimit : 14), | 30 m_lowerLimit(lowerLimit ? lowerLimit : 14), |
29 m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), | 31 m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), |
30 m_entriesSet(false) | 32 m_entriesSet(false) |
31 { | 33 { |
34 #ifdef DEBUG_SUBDIVIDING_MENU | |
35 cerr << "SubdividingMenu: constructed without title" << endl; | |
36 #endif | |
32 } | 37 } |
33 | 38 |
34 SubdividingMenu::SubdividingMenu(const QString &title, int lowerLimit, | 39 SubdividingMenu::SubdividingMenu(const QString &title, int lowerLimit, |
35 int upperLimit, QWidget *parent) : | 40 int upperLimit, QWidget *parent) : |
36 QMenu(title, parent), | 41 QMenu(title, parent), |
37 m_lowerLimit(lowerLimit ? lowerLimit : 14), | 42 m_lowerLimit(lowerLimit ? lowerLimit : 14), |
38 m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), | 43 m_upperLimit(upperLimit ? upperLimit : (m_lowerLimit * 5) / 2), |
39 m_entriesSet(false) | 44 m_entriesSet(false) |
40 { | 45 { |
46 #ifdef DEBUG_SUBDIVIDING_MENU | |
47 cerr << "SubdividingMenu: constructed with title \"" | |
48 << title << "\"" << endl; | |
49 #endif | |
41 } | 50 } |
42 | 51 |
43 SubdividingMenu::~SubdividingMenu() | 52 SubdividingMenu::~SubdividingMenu() |
44 { | 53 { |
45 for (map<QString, QObject *>::iterator i = m_pendingEntries.begin(); | 54 for (map<QString, QObject *>::iterator i = m_pendingEntries.begin(); |
51 void | 60 void |
52 SubdividingMenu::setEntries(const std::set<QString> &entries) | 61 SubdividingMenu::setEntries(const std::set<QString> &entries) |
53 { | 62 { |
54 m_entriesSet = true; | 63 m_entriesSet = true; |
55 | 64 |
65 #ifdef DEBUG_SUBDIVIDING_MENU | |
66 cerr << "SubdividingMenu::setEntries(" << title() << "): " | |
67 << entries.size() << " entries" << endl; | |
68 #endif | |
69 | |
56 int total = int(entries.size()); | 70 int total = int(entries.size()); |
57 | 71 |
58 if (total < m_upperLimit) return; | 72 if (total < m_upperLimit) return; |
59 | 73 |
60 int count = 0; | 74 int count = 0; |
63 | 77 |
64 QString firstNameInChunk; | 78 QString firstNameInChunk; |
65 QChar firstInitialInChunk; | 79 QChar firstInitialInChunk; |
66 bool discriminateStartInitial = false; | 80 bool discriminateStartInitial = false; |
67 | 81 |
68 for (set<QString>::const_iterator j = entries.begin(); | 82 // Re-sort using locale-aware comparator |
69 j != entries.end(); | 83 |
70 ++j) { | 84 auto comparator = [](QString s1, QString s2) -> bool { |
71 | 85 return QString::localeAwareCompare(s1, s2) < 0; |
72 // SVDEBUG << "SubdividingMenu::setEntries: j -> " << j->toStdString() << endl; | 86 }; |
87 | |
88 set<QString, typeof(comparator)> sortedEntries(comparator); | |
89 sortedEntries.insert(entries.begin(), entries.end()); | |
90 | |
91 for (auto j = sortedEntries.begin(); j != sortedEntries.end(); ++j) { | |
92 | |
93 #ifdef DEBUG_SUBDIVIDING_MENU | |
94 cerr << "SubdividingMenu::setEntries: entry is: " << j->toStdString() << endl; | |
95 #endif | |
73 | 96 |
74 m_nameToChunkMenuMap[*j] = chunkMenu; | 97 m_nameToChunkMenuMap[*j] = chunkMenu; |
75 | 98 |
76 set<QString>::iterator k = j; | 99 auto k = j; |
77 ++k; | 100 ++k; |
78 | 101 |
79 QChar initial = (*j)[0]; | 102 QChar initial = (*j)[0].toUpper(); |
80 | 103 |
81 if (count == 0) { | 104 if (count == 0) { |
82 firstNameInChunk = *j; | 105 firstNameInChunk = *j; |
83 firstInitialInChunk = initial; | 106 firstInitialInChunk = initial; |
107 #ifdef DEBUG_SUBDIVIDING_MENU | |
108 cerr << "starting new chunk at initial " << initial << endl; | |
109 #endif | |
84 } | 110 } |
85 | 111 |
86 // cerr << "count = "<< count << ", upper limit = " << m_upperLimit << endl; | 112 #ifdef DEBUG_SUBDIVIDING_MENU |
87 | 113 cerr << "count = "<< count << ", upper limit = " << m_upperLimit << endl; |
88 bool lastInChunk = (k == entries.end() || | 114 #endif |
115 | |
116 bool lastInChunk = (k == sortedEntries.end() || | |
89 (count >= m_lowerLimit-1 && | 117 (count >= m_lowerLimit-1 && |
90 (count == m_upperLimit || | 118 (count == m_upperLimit || |
91 (*k)[0] != initial))); | 119 (*k)[0].toUpper() != initial))); |
92 | 120 |
93 ++count; | 121 ++count; |
94 | 122 |
95 if (lastInChunk) { | 123 if (lastInChunk) { |
96 | 124 |
97 bool discriminateEndInitial = (k != entries.end() && | 125 bool discriminateEndInitial = (k != sortedEntries.end() && |
98 (*k)[0] == initial); | 126 (*k)[0].toUpper() == initial); |
99 | 127 |
100 bool initialsEqual = (firstInitialInChunk == initial); | 128 bool initialsEqual = (firstInitialInChunk == initial); |
101 | 129 |
102 QString from = QString("%1").arg(firstInitialInChunk); | 130 QString from = QString("%1").arg(firstInitialInChunk); |
103 if (discriminateStartInitial || | 131 if (discriminateStartInitial || |
134 | 162 |
135 void | 163 void |
136 SubdividingMenu::entriesAdded() | 164 SubdividingMenu::entriesAdded() |
137 { | 165 { |
138 if (m_entriesSet) { | 166 if (m_entriesSet) { |
139 cerr << "ERROR: SubdividingMenu::entriesAdded: setEntries was also called -- should use one mechanism or the other, but not both" << endl; | 167 SVCERR << "ERROR: SubdividingMenu::entriesAdded: setEntries was also called -- should use one mechanism or the other, but not both" << endl; |
140 return; | 168 return; |
141 } | 169 } |
142 | 170 |
143 set<QString> entries; | 171 set<QString> entries; |
144 for (map<QString, QObject *>::const_iterator i = m_pendingEntries.begin(); | 172 for (auto i: m_pendingEntries) { |
145 i != m_pendingEntries.end(); ++i) { | 173 entries.insert(i.first); |
146 entries.insert(i->first); | 174 } |
147 } | |
148 | |
149 setEntries(entries); | 175 setEntries(entries); |
150 | 176 |
151 for (map<QString, QObject *>::iterator i = m_pendingEntries.begin(); | 177 // Re-sort using locale-aware comparator (setEntries will do this |
152 i != m_pendingEntries.end(); ++i) { | 178 // again, for the set passed to it, but we need the same sorting |
153 | 179 // for the subsequent loop in this function as well) |
154 QMenu *menu = dynamic_cast<QMenu *>(i->second); | 180 auto comparator = [](QString s1, QString s2) -> bool { |
181 return QString::localeAwareCompare(s1, s2) < 0; | |
182 }; | |
183 set<QString, typeof(comparator)> sortedEntries(comparator); | |
184 for (auto i: m_pendingEntries) { | |
185 sortedEntries.insert(i.first); | |
186 } | |
187 | |
188 for (QString entry: sortedEntries) { | |
189 | |
190 QObject *obj = m_pendingEntries[entry]; | |
191 | |
192 QMenu *menu = dynamic_cast<QMenu *>(obj); | |
155 if (menu) { | 193 if (menu) { |
156 addMenu(i->first, menu); | 194 addMenu(entry, menu); |
157 continue; | 195 continue; |
158 } | 196 } |
159 | 197 |
160 QAction *action = dynamic_cast<QAction *>(i->second); | 198 QAction *action = dynamic_cast<QAction *>(obj); |
161 if (action) { | 199 if (action) { |
162 addAction(i->first, action); | 200 addAction(entry, action); |
163 continue; | 201 continue; |
164 } | 202 } |
165 } | 203 } |
166 | 204 |
167 m_pendingEntries.clear(); | 205 m_pendingEntries.clear(); |
176 m_pendingEntries[name] = action; | 214 m_pendingEntries[name] = action; |
177 return; | 215 return; |
178 } | 216 } |
179 | 217 |
180 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 218 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
181 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 219 #ifdef DEBUG_SUBDIVIDING_MENU |
220 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
221 #endif | |
182 QMenu::addAction(action); | 222 QMenu::addAction(action); |
183 return; | 223 return; |
184 } | 224 } |
185 | 225 |
186 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 226 #ifdef DEBUG_SUBDIVIDING_MENU |
227 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
228 #endif | |
187 m_nameToChunkMenuMap[name]->addAction(action); | 229 m_nameToChunkMenuMap[name]->addAction(action); |
188 } | 230 } |
189 | 231 |
190 QAction * | 232 QAction * |
191 SubdividingMenu::addAction(const QString &name) | 233 SubdividingMenu::addAction(const QString &name) |
195 m_pendingEntries[name] = action; | 237 m_pendingEntries[name] = action; |
196 return action; | 238 return action; |
197 } | 239 } |
198 | 240 |
199 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 241 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
200 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 242 #ifdef DEBUG_SUBDIVIDING_MENU |
243 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
244 #endif | |
201 return QMenu::addAction(name); | 245 return QMenu::addAction(name); |
202 } | 246 } |
203 | 247 |
204 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 248 #ifdef DEBUG_SUBDIVIDING_MENU |
249 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
250 #endif | |
205 return m_nameToChunkMenuMap[name]->addAction(name); | 251 return m_nameToChunkMenuMap[name]->addAction(name); |
206 } | 252 } |
207 | 253 |
208 void | 254 void |
209 SubdividingMenu::addAction(const QString &name, QAction *action) | 255 SubdividingMenu::addAction(const QString &name, QAction *action) |
212 m_pendingEntries[name] = action; | 258 m_pendingEntries[name] = action; |
213 return; | 259 return; |
214 } | 260 } |
215 | 261 |
216 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 262 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
217 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 263 #ifdef DEBUG_SUBDIVIDING_MENU |
264 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
265 #endif | |
218 QMenu::addAction(action); | 266 QMenu::addAction(action); |
219 return; | 267 return; |
220 } | 268 } |
221 | 269 |
222 // SVDEBUG << "SubdividingMenu::addAction(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 270 #ifdef DEBUG_SUBDIVIDING_MENU |
271 cerr << "SubdividingMenu::addAction(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
272 #endif | |
223 m_nameToChunkMenuMap[name]->addAction(action); | 273 m_nameToChunkMenuMap[name]->addAction(action); |
224 } | 274 } |
225 | 275 |
226 void | 276 void |
227 SubdividingMenu::addMenu(QMenu *menu) | 277 SubdividingMenu::addMenu(QMenu *menu) |
232 m_pendingEntries[name] = menu; | 282 m_pendingEntries[name] = menu; |
233 return; | 283 return; |
234 } | 284 } |
235 | 285 |
236 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 286 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
237 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 287 #ifdef DEBUG_SUBDIVIDING_MENU |
288 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
289 #endif | |
238 QMenu::addMenu(menu); | 290 QMenu::addMenu(menu); |
239 return; | 291 return; |
240 } | 292 } |
241 | 293 |
242 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 294 #ifdef DEBUG_SUBDIVIDING_MENU |
295 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
296 #endif | |
243 m_nameToChunkMenuMap[name]->addMenu(menu); | 297 m_nameToChunkMenuMap[name]->addMenu(menu); |
244 } | 298 } |
245 | 299 |
246 QMenu * | 300 QMenu * |
247 SubdividingMenu::addMenu(const QString &name) | 301 SubdividingMenu::addMenu(const QString &name) |
252 m_pendingEntries[name] = menu; | 306 m_pendingEntries[name] = menu; |
253 return menu; | 307 return menu; |
254 } | 308 } |
255 | 309 |
256 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 310 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
257 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 311 #ifdef DEBUG_SUBDIVIDING_MENU |
312 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
313 #endif | |
258 return QMenu::addMenu(name); | 314 return QMenu::addMenu(name); |
259 } | 315 } |
260 | 316 |
261 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 317 #ifdef DEBUG_SUBDIVIDING_MENU |
318 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
319 #endif | |
262 return m_nameToChunkMenuMap[name]->addMenu(name); | 320 return m_nameToChunkMenuMap[name]->addMenu(name); |
263 } | 321 } |
264 | 322 |
265 void | 323 void |
266 SubdividingMenu::addMenu(const QString &name, QMenu *menu) | 324 SubdividingMenu::addMenu(const QString &name, QMenu *menu) |
269 m_pendingEntries[name] = menu; | 327 m_pendingEntries[name] = menu; |
270 return; | 328 return; |
271 } | 329 } |
272 | 330 |
273 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { | 331 if (m_nameToChunkMenuMap.find(name) == m_nameToChunkMenuMap.end()) { |
274 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): not found in name-to-chunk map, adding to main menu" << endl; | 332 #ifdef DEBUG_SUBDIVIDING_MENU |
333 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): not found in name-to-chunk map, adding to main menu" << endl; | |
334 #endif | |
275 QMenu::addMenu(menu); | 335 QMenu::addMenu(menu); |
276 return; | 336 return; |
277 } | 337 } |
278 | 338 |
279 // SVDEBUG << "SubdividingMenu::addMenu(" << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | 339 #ifdef DEBUG_SUBDIVIDING_MENU |
340 cerr << "SubdividingMenu::addMenu(" << title() << " | " << name << "): found in name-to-chunk map for menu " << m_nameToChunkMenuMap[name]->title() << endl; | |
341 #endif | |
280 m_nameToChunkMenuMap[name]->addMenu(menu); | 342 m_nameToChunkMenuMap[name]->addMenu(menu); |
281 } | 343 } |
282 | 344 |