-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcustomdialog.h
405 lines (330 loc) · 13.9 KB
/
customdialog.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
#pragma once
#ifndef CUSTOMDIALOG_H
#define CUSTOMDIALOG_H
//############################################################
//
// This file contains a special class, "CustomDialog", for generating
// a modal dialog box with multiple form elements/inputs including:
// labels, checkboxes, line edit boxes, number edit boxes, combo boxes,
// radio buttons, spin boxes (for ints or floats), group boxes (to group elements)
// and even "pick color" buttons (letting you chose a color).
//
// NOTES:
//
// > Each form element is created by a single function only, includes an optional tooltip,
// and a pointer to the variable you wish to modify.
//
// > The value of the variable is updated ONLY if and when hits "Ok", and will change only
// if the user has changed it from the original/default/starting value in that variable.
//
// > This class if fantastic for making modal dialogs (i.e. dialogs where you can't click
// anything else until you click okay or cancel), but isn't designed for times when you
// want/need the parent dialog to dymaically update as the user changes values
// (i.e. interactive changes).
//
// > This method of generating dialog elements on-the-fly by using a vectors of structs is
// obviously less versatile than creating a seperate class which extend QDialog for EACH
// dialog you need. The huge advantage, however, is you can can greatly reduce your code
// and avoid the tedium of creating a seperate .h and .cpp file everytime you wants to
// prompt the user for multiple pieces of info in a single modal dialog!
//
// > This file also includes several "convinience" functions such as "MsgBoxYesNo" and
// "setBold" for quickly displaying dialogs/returning results and modifying strings.
// These convinience functions are at the bottom of this file (and defined in the .cpp).
//
// -----------------
//
// An example of how to use this dialog follows:
//
// void GetDetailsFromUser()
// {
// string name = "First Last";
// bool student = true;
// int age = 20;
// int sportIdx = 1;
//
// CustomDialog d("Registration", this);
// d.addLabel ("Please enter your details below ...");
// d.addLineEdit ("name: ", &name, "No middle name!");
// d.addCheckBox ("current student", &student);
// d.addSpinBox ("your age: ", 1, 120, &age, 1);
// d.addComboBox ("sport: ", "tennis|soccer|cricket", &sportIdx);
//
// d.exec(); // Execution stops here until user closes dialog
//
// if(d.wasCancelled()) {
// cout << "You have cancelled your registration" << endl;
// return;
// }
//
// cout << "Dear " << name << " thank you for registering..." << endl;
//
// ... CODE ...
// }
//
// -----------------
//
// And produces a dialog which looks like this:
//
// +----------------------------------------+
// | Registration ? X |
// +----------------------------------------+
// | Please enter your details below ... |
// | _____________________________ |
// | name: |__First_Last_________________| |
// | |
// | [X] current student |
// | _____ |
// | your age: [_22_^] |
// | ______________ |
// | sport: |_soccer_____[V] |
// | |
// | +-------------+ +-------------+ |
// | | Ok | | Cancel | |
// | +-------------+ +-------------+ |
// +----------------------------------------+
//
//
//
// > author: Andrew Noske
// > last updated: 6-June-2012
//
// http://www.andrewnoske.com/wiki/index.php?title=Code_-_qt_custom_input_dialog
//############################################################
#include <iostream>
#include <vector>
#include <string>
#include <qdialog.h>
#include <qobject.h>
#include <qvariant.h>
#include <qaction.h>
#include <qapplication.h>
#include <qpushbutton.h>
#include <qcheckbox.h>
#include <qlabel.h>
#include <qcombobox.h>
#include <qbuttongroup.h>
#include <qradiobutton.h>
#include <qdialog.h>
#include <qspinbox.h>
#include <qlayout.h>
#include <qgroupbox.h>
#include <qtextedit.h>
#include <qprogressbar.h>
#include <qtooltip.h>
#include <qstringlist.h>
#include <qmessagebox.h>
#include <qinputdialog.h>
#include <qcompleter.h>
#include <QHBoxLayout>
#include <QLabel>
#include <QVBoxLayout>
#include <QDesktopServices>
#include <QDir>
#include <QUrl>
//############################################################
//## CONSTANTS:
enum DlgType { DLG_LABEL, DLG_CHECKBOX, DLG_LINEEDIT, DLG_FLOATEDIT,
DLG_SPINBOX, DLG_DBLSPINBOX, DLG_MINMAXSPIN,
DLG_COMBOBOX, DLG_RADIOGRP, DLG_GRPBOX, DLG_COLOR, DLG_TEXTEDIT, DGL_ALL };
enum chkbehav { CB_NONE, CB_DISABLE, CB_ENABLE, CB_HIDE, CB_SHOW };
enum btnset { BS_CANCEL_OKAY, BS_OKAY_ONLY, BS_NO_YES, BS_CUSTOM };
enum btnbehav { BB_ACCEPT, BB_REJECT, BB_DISABLED, BB_POPUP, BB_OPENURL };
//############################################################
//## SMALL CLASSES:
class ColorButton : public QPushButton // Used to create a "pick color" button.
{
Q_OBJECT // NOTE: To compile on my version of IMOD I have to comment these out.
public:
QColor color;
ColorButton(QColor _color, QWidget *parent=0);
void setColor(QColor _color);
QColor getColor();
public slots:
void pickColor();
//public signals:
// void valueChanged() {};
};
//############################################################
//************************
//** DialogElement is used to store gui widgets in the array of
//** widgets displayed in GuiDialogCustomizable
struct DialogElement
{
DlgType type; // The "type" of dialog element displayed this row.
bool extraChkAdded; // Set true if a special extra checkbox is added.
// using CustomDialog.addCheckPrev()
//** POINTERS USE TO PASS BACK ANY CHANGED VALUES:
std::string *returnString; // For DLG_LINEEDIT.
int *returnInt; // For DLG_SPINBOX, DLG_COMBOBOX & DLG_RADIOGRP.
int *returnInt2; // For DLG_DBLSPINBOX.
bool *returnBool; // For DLG_CHECKBOX.
float *returnFloat; // For DLG_FLOATEDIT & DLG_DBLSPINBOX.
QColor *returnColor; // For DLG_COLOR.
bool *returnChkExtra; // Fsed if extraChkAdded is true.
bool readOnly; // if set to true, user cannot change the text.
//** FORM ELEMENTS TO DISPLAY (DEPENDING ON TYPE):
QWidget *wid;
QHBoxLayout *layout;
QLabel *label;
QLabel *label2;
QCheckBox *chkBox;
QLineEdit *lineEdit;
QSpinBox *spnBox;
QSpinBox *spnBox2;
QDoubleSpinBox *dblSpnBox;
QComboBox *cmbBox;
ColorButton *btnColor;
std::vector<QRadioButton*> radBtn;
QGroupBox *grpBox;
QTextEdit *textEdit;
QCheckBox *chkExtra;
};
//############################################################
//************************
//** GuiDialogCustomizable is used to present a customizable gui
//** dialog and retrieve user input with minimal code!
class CustomDialog : public QDialog
{
Q_OBJECT
public: //## METHODS:
CustomDialog(QString title, QWidget *parent = 0, btnset=BS_CANCEL_OKAY);
~CustomDialog() {};
bool setDialogElements();
bool wasCancelled();
bool addCustomButton(QString buttonStr, btnbehav buttonBehav=BB_ACCEPT, QString tooltip="");
DialogElement& addNewElement(DlgType _type, QString caption, QString tooltip, bool makeLabel);
int addLabel(QString caption, bool bold=false, QString tooltip="");
int addHtmlLabel(QString caption, QString tooltip="");
int addCheckBox(QString caption, bool *checked, QString tooltip="");
int addLineEdit(QString caption, std::string *stringValue, QString tooltip="", bool isPassword=false);
int addReadOnlyLineEdit(QString caption, QString text, QString tooltip="");
int addLineEditF(QString caption, float min, float max, float *value, float decimals, QString tooltip="", QString unitsStr="");
int addSpinBox(QString caption, int min, int max, int *value, int step, QString tooltip="");
int addDblSpinBoxF(QString caption, float min, float max, float *value, int decimals, float step=0.1, QString tooltip="");
int addComboBox(QString caption, QString barSepList, int *selIdx, QString tooltip="");
int addRadioGrp(QString caption, QString barSepList, int *selIdx, QString tooltip="", QString tooltipArr="", bool checkable=false, bool *checked=0);
int addColorSel(QString caption, QColor *color, QString tooltip="");
int addMinMaxSpinBoxPair(QString caption, QString middleCaption, int min, int max, int *minValue, int *maxValue, int step=1, QString tooltip="");
int addTextEdit(std::string *text, bool richText, bool readOnly, int minHeight=90, QString tooltip="");
int addReadOnlyTextEdit(QString text, bool richText, int minHeight=90, QString tooltip="");
int addProgressBar(QString caption, int percent, int width, bool showValue, QString tooltip="");
int addPercentBar(QString caption, QString valueLabel, float percent, int width, QColor colorBar, QString tooltip="", QFrame::Shape shape = QFrame::StyledPanel, QFrame::Shadow shadow = QFrame::Sunken);
int addVSpacer(int minHeight=0);
int beginGroupBox(QString caption, bool flat=false, QString tooltip="", bool checkable=false, bool *checked=0);
void endGroupBox();
int addCheckPrev(QString caption, bool *checked, chkbehav chkBeh, bool removeLabel, QString tooltip="");
int addAutoCompletePrev(QStringList wordList, bool caseSensitive=false);
bool setStyleElem(int idx, std::string styleStr, bool bold=false);
void setStylePrev(std::string styleStr, bool bold=false);
bool setEnabledElem(int idx, bool enabled);
void setEnabledPrev(bool enabled);
void setEnabledAll(bool enabled);
public: //## DATA:
std::vector<DialogElement> elements; // The vector of GUI elements used to display
// and change the values.
int customBtnClicked; // Set to the index of the button
// "customBtn" clicked.
private:
std::vector<QPushButton*> customBtn; // Vector of buttons down the button of the GUI.
QVBoxLayout *vboxLayout;
QHBoxLayout *hbtnLayout;
bool addToGroupBox;
QVBoxLayout *groupBoxLayout;
QVBoxLayout *layoutNextElement;
public slots: //## SLOTS:
void customBtnAccept();
void customBtnReject();
void customBtnMessage();
void customBtnOpenUrl();
void updateBtnClicked(QObject *btnClicked);
void resizeMe();
int exec();
};
//############################################################
//-------------------------------
//## SMALL MESSAGE BOX FUNCTIONS:
void MsgBox(std::string str);
void MsgBox(QWidget *parent, QString title, QString str);
bool MsgBoxYesNo(QWidget *parent, std::string str);
std::string InputBoxString(QWidget *parent, std::string title, std::string label, std::string defaultStr);
//-------------------------------
//## SMALL INLINE GUI FUNCTIONS:
inline QString QStr(int number );
inline QString QStr(long number );
inline QString QStr(float number);
inline QString QStr(double number);
inline std::string qStringToString(QString qstr);
inline QString nbsp(int numSpaces);
inline void setBold(QWidget *wid, bool bold);
inline void setTextColor(QWidget *wid, int r, int g, int b);
inline void setDefaultColorAndFont(QWidget *wid);
inline void openUrl(QString urlString, bool addFilePrefix=false);
//############################################################
//----------------------------------------------------------------------------
//
// SMALL INLINE GUI FUNCTIONS:
//
//----------------------------------------------------------------------------
//---------
//-- Short function name for converting numbers to a QString.
inline QString QStr(int number) { return QString::number(number); }
inline QString QStr(long number) { return QString::number(number); }
inline QString QStr(float number) { return QString::number(number); }
inline QString QStr(double number) { return QString::number(number); }
//---------
//-- Converts a QString to a standard string
inline std::string qStringToString(QString qstr)
{
std::string str = "";
for(int i=0; i<qstr.length(); i++)
str += qstr.at(i).toLatin1();
return str;
}
//---------
//-- Creates a qstring with the specified number
//-- of "non breaking space" HTML characters
inline QString nbsp(int numSpaces)
{
QString str = "";
for(int i=0; i<numSpaces; i++)
str += " ";
return str;
}
//---------
//-- Short function name for taking any widget and making the text in it bold.
inline void setBold(QWidget *wid, bool bold)
{
QFont font;
font.setBold(bold);
wid->setFont(font);
}
//---------
//-- Short function name for setting the text (forground) color of a widget.
inline void setTextColor(QWidget *wid, int r, int g, int b)
{
wid->setStyleSheet("color: rgb(" + QStr(r) + "," + QStr(g) + "," + QStr(b) + ");");
}
//---------
//-- Short function which sets the font to default, the foreground/text color to black
//-- and background to transparent. This function is useful for when you might apply a
//-- stylesheet to a container object, but you don't want those changes to apply to
//-- (heirarchially) to widgets within it.
inline void setDefaultColorAndFont(QWidget *wid)
{
wid->setFont(QFont());
wid->setStyleSheet("color: rgb(0, 0, 0); background-color: rgba(255, 255, 255, 0);");
}
//---------
//-- Opens the given URL ("urlString") in the default web browser.
//-- If ("addFilePrefix") is true, the string "file://" is added to the
//-- front and, in most operating systems (including OSX), this will
//-- mean the specified file on the computer should be opened
//-- in the default program (eg: a .xls file open in Excel) instead.
inline void openUrl(QString urlString, bool addFilePrefix)
{
if(addFilePrefix)
urlString = "file://" + urlString;
QDesktopServices::openUrl(QUrl(urlString));
}
#endif