]> git.lyx.org Git - features.git/blob - src/frontends/qt4/GuiSymbols.cpp
ef1d27d03814cf1b65139201ad8157f021864cc7
[features.git] / src / frontends / qt4 / GuiSymbols.cpp
1 /**
2  * \file GuiSymbols.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Jürgen Spitzmüller
7  *
8  * Full author contact details are available in file CREDITS.
9  */
10
11 #include <config.h>
12
13 #include "GuiSymbols.h"
14
15 #include "GuiApplication.h"
16 #include "GuiView.h"
17 #include "qt_helpers.h"
18
19 #include "Buffer.h"
20 #include "BufferView.h"
21 #include "Encoding.h"
22
23 #include "support/gettext.h"
24
25 #include <QChar>
26 #include <QPixmap>
27 #include <QListWidgetItem>
28 #include <QString>
29
30 using namespace std;
31
32 namespace lyx {
33 namespace frontend {
34
35
36 namespace {
37
38 /// name of unicode block, start and end code point
39 struct UnicodeBlocks {
40         char const * name;
41         char_type start;
42         char_type end;
43 };
44
45
46 /// all unicode blocks with start and end code point
47 UnicodeBlocks unicode_blocks[] = {
48         { N_("Basic Latin"), 0x0000, 0x007f },
49         { N_("Latin-1 Supplement"), 0x0080, 0x00ff },
50         { N_("Latin Extended-A"), 0x0100, 0x017f },
51         { N_("Latin Extended-B"), 0x0180, 0x024f },
52         { N_("IPA Extensions"), 0x0250, 0x02af },
53         { N_("Spacing Modifier Letters"), 0x02b0, 0x02ff },
54         { N_("Combining Diacritical Marks"), 0x0300, 0x036f },
55         { N_("Greek"), 0x0370, 0x03ff },
56         { N_("Cyrillic"), 0x0400, 0x04ff },
57         { N_("Armenian"), 0x0530, 0x058f },
58         { N_("Hebrew"), 0x0590, 0x05ff },
59         { N_("Arabic"), 0x0600, 0x06ff },
60         { N_("Devanagari"), 0x0900, 0x097f },
61         { N_("Bengali"), 0x0980, 0x09ff },
62         { N_("Gurmukhi"), 0x0a00, 0x0a7f },
63         { N_("Gujarati"), 0x0a80, 0x0aff },
64         { N_("Oriya"), 0x0b00, 0x0b7f },
65         { N_("Tamil"), 0x0b80, 0x0bff },
66         { N_("Telugu"), 0x0c00, 0x0c7f },
67         { N_("Kannada"), 0x0c80, 0x0cff },
68         { N_("Malayalam"), 0x0d00, 0x0d7f },
69         { N_("Thai"), 0x0e00, 0x0e7f },
70         { N_("Lao"), 0x0e80, 0x0eff },
71         { N_("Tibetan"), 0x0f00, 0x0fbf },
72         { N_("Georgian"), 0x10a0, 0x10ff },
73         { N_("Hangul Jamo"), 0x1100, 0x11ff },
74         { N_("Phonetic Extensions"), 0x1d00, 0x1d7f },
75         { N_("Latin Extended Additional"), 0x1e00, 0x1eff },
76         { N_("Greek Extended"), 0x1f00, 0x1fff },
77         { N_("General Punctuation"), 0x2000, 0x206f },
78         { N_("Superscripts and Subscripts"), 0x2070, 0x209f },
79         { N_("Currency Symbols"), 0x20a0, 0x20cf },
80         { N_("Combining Diacritical Marks for Symbols"), 0x20d0, 0x20ff },
81         { N_("Letterlike Symbols"), 0x2100, 0x214f },
82         { N_("Number Forms"), 0x2150, 0x218f },
83         { N_("Arrows"), 0x2190, 0x21ff },
84         { N_("Mathematical Operators"), 0x2200, 0x22ff },
85         { N_("Miscellaneous Technical"), 0x2300, 0x23ff },
86         { N_("Control Pictures"), 0x2400, 0x243f },
87         { N_("Optical Character Recognition"), 0x2440, 0x245f },
88         { N_("Enclosed Alphanumerics"), 0x2460, 0x24ff },
89         { N_("Box Drawing"), 0x2500, 0x257f },
90         { N_("Block Elements"), 0x2580, 0x259f },
91         { N_("Geometric Shapes"), 0x25a0, 0x25ff },
92         { N_("Miscellaneous Symbols"), 0x2600, 0x26ff },
93         { N_("Dingbats"), 0x2700, 0x27bf },
94         { N_("Miscellaneous Mathematical Symbols-A"), 0x27c0, 0x27ef },
95         { N_("CJK Symbols and Punctuation"), 0x3000, 0x303f },
96         { N_("Hiragana"), 0x3040, 0x309f },
97         { N_("Katakana"), 0x30a0, 0x30ff },
98         { N_("Bopomofo"), 0x3100, 0x312f },
99         { N_("Hangul Compatibility Jamo"), 0x3130, 0x318f },
100         { N_("Kanbun"), 0x3190, 0x319f },
101         { N_("Enclosed CJK Letters and Months"), 0x3200, 0x32ff },
102         { N_("CJK Compatibility"), 0x3300, 0x33ff },
103         { N_("CJK Unified Ideographs"), 0x4e00, 0x9fa5 },
104         { N_("Hangul Syllables"), 0xac00, 0xd7a3 },
105         { N_("High Surrogates"), 0xd800, 0xdb7f },
106         { N_("Private Use High Surrogates"), 0xdb80, 0xdbff },
107         { N_("Low Surrogates"), 0xdc00, 0xdfff },
108         { N_("Private Use Area"), 0xe000, 0xf8ff },
109         { N_("CJK Compatibility Ideographs"), 0xf900, 0xfaff },
110         { N_("Alphabetic Presentation Forms"), 0xfb00, 0xfb4f },
111         { N_("Arabic Presentation Forms-A"), 0xfb50, 0xfdff },
112         { N_("Combining Half Marks"), 0xfe20, 0xfe2f },
113         { N_("CJK Compatibility Forms"), 0xfe30, 0xfe4f },
114         { N_("Small Form Variants"), 0xfe50, 0xfe6f },
115         { N_("Arabic Presentation Forms-B"), 0xfe70, 0xfeff },
116         { N_("Halfwidth and Fullwidth Forms"), 0xff00, 0xffef },
117         { N_("Specials"), 0xfff0, 0xffff },
118         { N_("Linear B Syllabary"), 0x10000, 0x1007f },
119         { N_("Linear B Ideograms"), 0x10080, 0x100ff },
120         { N_("Aegean Numbers"), 0x10100, 0x1013f },
121         { N_("Ancient Greek Numbers"), 0x10140, 0x1018f },
122         { N_("Old Italic"), 0x10300, 0x1032f },
123         { N_("Gothic"), 0x10330, 0x1034f },
124         { N_("Ugaritic"), 0x10380, 0x1039f },
125         { N_("Old Persian"), 0x103a0, 0x103df },
126         { N_("Deseret"), 0x10400, 0x1044f },
127         { N_("Shavian"), 0x10450, 0x1047f },
128         { N_("Osmanya"), 0x10480, 0x104af },
129         { N_("Cypriot Syllabary"), 0x10800, 0x1083f },
130         { N_("Kharoshthi"), 0x10a00, 0x10a5f },
131         { N_("Byzantine Musical Symbols"), 0x1d000, 0x1d0ff },
132         { N_("Musical Symbols"), 0x1d100, 0x1d1ff },
133         { N_("Ancient Greek Musical Notation"), 0x1d200, 0x1d24f },
134         { N_("Tai Xuan Jing Symbols"), 0x1d300, 0x1d35f },
135         { N_("Mathematical Alphanumeric Symbols"), 0x1d400, 0x1d7ff },
136         { N_("CJK Unified Ideographs Extension B"), 0x20000, 0x2a6d6 },
137         { N_("CJK Compatibility Ideographs Supplement"), 0x2f800, 0x2fa1f },
138         { N_("Tags"), 0xe0000, 0xe007f },
139         { N_("Variation Selectors Supplement"), 0xe0100, 0xe01ef },
140         { N_("Supplementary Private Use Area-A"), 0xf0000, 0xe01ef },
141         { N_("Supplementary Private Use Area-B"), 0x100000, 0x10ffff }
142 };
143
144 const int no_blocks = sizeof(unicode_blocks) / sizeof(UnicodeBlocks);
145
146 } // namespace anon
147
148
149 GuiSymbols::GuiSymbols(GuiView & lv)
150         : DialogView(lv, "symbols", qt_("Symbols")), encoding_("ascii")
151 {
152         setupUi(this);
153
154         setFocusProxy(symbolsLW);
155
156         symbolsLW->setViewMode(QListView::IconMode);
157         symbolsLW->setUniformItemSizes(true);
158         // increase the display size of the symbols a bit
159         QFont font= symbolsLW->font();
160         int size = font.pointSize() + 3;
161         font.setPointSize(size);
162         symbolsLW->setFont(font);
163 }
164
165
166 void GuiSymbols::updateView()
167 {
168         chosenLE->clear();
169
170         string const & new_encoding = bufferview()->cursor().getEncoding()->name();
171         if (new_encoding == encoding_)
172                 // everything up to date
173                 return;
174         if (!new_encoding.empty())
175                 encoding_ = new_encoding;
176         updateSymbolList();
177 }
178
179
180 void GuiSymbols::enableView(bool enable)
181 {
182         chosenLE->setEnabled(enable);
183         okPB->setEnabled(enable);
184         applyPB->setEnabled(enable);
185 }
186
187
188 void GuiSymbols::on_applyPB_clicked()
189 {
190         dispatchParams();
191 }
192
193
194 void GuiSymbols::on_okPB_clicked()
195 {
196         dispatchParams();
197         hide();
198 }
199
200
201 void GuiSymbols::on_closePB_clicked()
202 {
203         hide();
204 }
205
206
207 void GuiSymbols::on_symbolsLW_itemActivated(QListWidgetItem *)
208 {
209         on_okPB_clicked();
210 }
211
212
213 void GuiSymbols::on_chosenLE_textChanged(QString const & text)
214 {
215         bool const empty_sel = text.isEmpty();
216         okPB->setEnabled(!empty_sel);
217         applyPB->setEnabled(!empty_sel);
218 }
219
220
221 void GuiSymbols::on_chosenLE_returnPressed()
222 {
223         on_okPB_clicked();
224 }
225
226
227 void GuiSymbols::on_symbolsLW_itemClicked(QListWidgetItem * item)
228 {
229         QString const text = item->text();
230         if (text.isEmpty())
231                 return;
232         if (chosenLE->isEnabled())
233                 chosenLE->insert(text);
234         QString const category = getBlock(text.data()->unicode());
235         categoryCO->setCurrentIndex(categoryCO->findText(category));
236 }
237
238
239 void GuiSymbols::on_categoryCO_activated(QString const & text)
240 {
241         if (!categoryFilterCB->isChecked())
242                 updateSymbolList();
243         if (used_blocks.find(text) != used_blocks.end())
244                 symbolsLW->scrollToItem(used_blocks[text],
245                         QAbstractItemView::PositionAtTop);
246 }
247
248
249 void GuiSymbols::on_categoryFilterCB_toggled(bool on)
250 {
251         updateSymbolList();
252         if (on) {
253                 QString const category = categoryCO->currentText();
254                 if (used_blocks.find(category) != used_blocks.end())
255                         symbolsLW->scrollToItem(used_blocks[category],
256                                 QAbstractItemView::PositionAtTop);
257         }
258 }
259
260
261 void GuiSymbols::updateSymbolList()
262 {
263         QString category = categoryCO->currentText();
264         bool const nocategory = category.isEmpty();
265         char_type range_start = 0x0000;
266         char_type range_end = 0x110000;
267         symbolsLW->clear();
268         used_blocks.clear();
269         categoryCO->clear();
270         bool const show_all = categoryFilterCB->isChecked();
271
272         typedef set<char_type> SymbolsList;
273         Encoding enc = *(encodings.getFromLyXName(encoding_));
274         SymbolsList symbols = enc.getSymbolsList();
275
276         if (!show_all) {
277                 for (int i = 0 ; i < no_blocks; ++i)
278                         if (unicode_blocks[i].name == fromqstr(category)) {
279                                 range_start = unicode_blocks[i].start;
280                                 range_end = unicode_blocks[i].end;
281                                 break;
282                         }
283         }
284
285         SymbolsList::const_iterator const end = symbols.end();
286         for (SymbolsList::const_iterator it = symbols.begin(); it != end; ++it) {
287                 char_type c = *it;
288 #if QT_VERSION >= 0x040300
289                 QChar::Category const cat = QChar::category(uint(c));
290 #else
291                 QChar const qc = uint(c);
292                 QChar::Category const cat = qc.category();
293 #endif
294                 // we do not want control or space characters
295                 if (cat == QChar::Other_Control || cat == QChar::Separator_Space)
296                         continue;
297                 QListWidgetItem * lwi = new QListWidgetItem(
298                         QString::fromUcs4((uint const *) &c, 1));
299                 if (show_all || c >= range_start && c <= range_end) {
300                         lwi->setTextAlignment(Qt::AlignCenter);
301                         symbolsLW->addItem(lwi);
302                 }
303                 QString block = getBlock(c);
304                 if (category.isEmpty())
305                         category = block;
306                 if (used_blocks.find(block) == used_blocks.end())
307                         used_blocks[block] = lwi;
308         }
309
310         // update category combo
311         for (UsedBlocks::iterator it = used_blocks.begin(); it != used_blocks.end(); ++it) {
312                 categoryCO->addItem(it->first);
313         }
314         int old = categoryCO->findText(category);
315         if (old != -1)
316                 categoryCO->setCurrentIndex(old);
317         // update again in case the combo has not yet been filled
318         // on first cycle (at dialog initialization)
319         if (nocategory && !category.isEmpty())
320                 updateSymbolList();
321 }
322
323
324 QString const GuiSymbols::getBlock(char_type c) const
325 {
326         int i = 0;
327         while (c > unicode_blocks[i].end && i < no_blocks)
328                 ++i;
329         if (unicode_blocks[i].name)
330                 return toqstr(unicode_blocks[i].name);
331         return QString();
332 }
333
334
335 void GuiSymbols::dispatchParams()
336 {
337         dispatch(FuncRequest(getLfun(), fromqstr(chosenLE->text())));
338 }
339
340
341 Dialog * createGuiSymbols(GuiView & lv)
342 {
343         return new GuiSymbols(lv);
344 }
345
346
347 } // namespace frontend
348 } // namespace lyx
349
350 #include "GuiSymbols_moc.cpp"