Күрделі жобаларда бағдарлама интерфейсінде статикалық виджеттердің болуы жеткіліксіз болуы мүмкін, өйткені кіріс ақпарат секунд сайын өзгеруі мүмкін. Бұл Qt орналасуындағы түймелер сияқты виджеттерді динамикалық құру мәселесін тудырады.
Бұл сабақ **QPushButton түймелерін динамикалық түрде жасауды, осы түймелерден сигналдарды қабылдауды, содан кейін осы түймелерді Qt орналасуынан жоюды сипаттайды.
Код Qt 5.4.1 негізінде QtCreator 3.3.1-де жазылған.
Жоба құрылымы
Жоба құрылымының сипаттамасы:
- DynamicButtons.pro - профиль;
- mainwindow.h – қосымшаның негізгі терезесінің тақырып файлы;
- mainwindow.cpp – терезенің бастапқы коды;
- main.cpp – қолданба басталатын негізгі бастапқы файл;
- mainwindow.ui – қосымшаның негізгі терезесінің формасы;
- qdynamicbutton.h – осы сабақта динамикалық объектілермен жұмыс істеу процесін жеңілдететін орауыш класының тақырып файлы;
- qdynamicbutton.cpp - бұл оқулықтағы динамикалық нысандармен жұмыс істеу процесін жеңілдететін қаптама класының бастапқы файлы.
mainwindow.ui
Бұл оқулықта бізге екі макет қабаты қажет болады. Бір қабатта барлық динамикалық түрде жасалған түймелер болады, ал екінші қабатта динамикалық түймелерді жасау және жою процесіне жауап беретін түймелер, сондай-ақ жасалып жатқан түймелер санын көрсету үшін lineEdit өрісі болады.
Қолданбаның сыртқы түрі Тікелей жұмыс қолданба терезесі нысанының келесі объектілерімен жүзеге асырылады:
- addButton - динамикалық түймелерді қосу батырмасы;
- deleteButton - динамикалық түймелерді жою батырмасы;
- verticalLayout - тік орналасуы бар динамикалық түймелерді қосу қабаты;
- lineEdit - құрылған түймелердің сандарын көрсету өрісі.
qdynamicbutton.h
QPushButton сыныбынан мұраланған орауыш класының тақырып файлы. Бұл класс сыныптың барлық нысандарына ортақ болатын статикалық айнымалыны жариялайды және келесіде жасалатын барлық динамикалық түймелердің есептегіштері болады. бұл қолданба. Бұл әр түймеге сәйкес сандарды тағайындау үшін қажет.
#ifndef QDYNAMICBUTTON\_H #define QDYNAMICBUTTON\_H #include <QPushButton> class QDynamicButton : public QPushButton { Q\_OBJECT public: explicit QDynamicButton(QWidget *parent = 0); ~QDynamicButton(); static int ResID; // Статическая переменная, счетчик номеров кнопок int getID(); // Функция для возврата локального номера кнопки public slots: private: int buttonID = 0; // Локальная переменная, номер кнопки }; #endif // QDYNAMICBUTTON\_H
qdynamicbutton.cpp
Қаптама класының бастапқы код файлында түйме оның конструкторында инициализацияланады, статикалық айнымалы инициализацияланады, сонымен қатар динамикалық түйменің нөмірін қайтару әдісі бар.
#include "qdynamicbutton.h" QDynamicButton::QDynamicButton(QWidget *parent) : QPushButton(parent) { ResID++; // Увеличение счетчика на единицу buttonID = ResID; /* Присвоение кнопке номера, по которому будет производиться * дальнейшая работа с кнопок * */ } QDynamicButton::~QDynamicButton() { } /* Метод для возврата значения номера кнопки * */ int QDynamicButton::getID() { return buttonID; } /* Инициализация статической переменной класса. * Статическая переменная класса должна инициализироваться в обязательном порядке * */ int QDynamicButton::ResID = 0;
mainwindow.h
Тақырып файлында басқару түймелерін өңдеуге арналған SLOT және басылған динамикалық түйменің нөмірін алу үшін SLOT қосу керек.
#ifndef MAINWINDOW\_H #define MAINWINDOW\_H #include <QMainWindow> /* My Includes */ #include <qdynamicbutton.h> namespace Ui { class MainWindow; } class MainWindow : public QMainWindow { Q\_OBJECT public: explicit MainWindow(QWidget *parent = 0); ~MainWindow(); private slots: void on\_addButton\_clicked(); // СЛОТ-обработчик нажатия кнопки добавления void on\_deleteButton\_clicked(); // СЛОТ-обработчик нажатия кнопки удаления void slotGetNumber(); // СЛОТ для получения номера нажатой динамической кнопки private: Ui::MainWindow *ui; }; #endif // MAINWINDOW\_H mainwindow.cpp #include "mainwindow.h" #include "ui\_mainwindow.h" MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow) { ui->setupUi(this); /* Для удобства работы слои разделены QSplitter * */ ui->splitter->setStretchFactor(0,1); ui->splitter->setStretchFactor(1,0); } MainWindow::~MainWindow() { delete ui; } /* Метод для добавления динамической кнопки * */ void MainWindow::on\_addButton\_clicked() { QDynamicButton *button = new QDynamicButton(this); // Создаем объект динамической кнопки /* Устанавливаем текст с номером этой кнопки * */ button->setText("Кнопочка " + QString::number(button->getID())); /* Добавляем кнопку в слой с вертикальной компоновкой * */ ui->verticalLayout->addWidget(button); /* Подключаем сигнал нажатия кнопки к СЛОТ получения номера кнопки * */ connect(button, SIGNAL(clicked()), this, SLOT(slotGetNumber())); } /* Метод для удаления динамической кнопки по её номеру * */ void MainWindow::on\_deleteButton\_clicked() { /* Выполняем перебор всех элементов слоя, где располагаются динамические кнопки * */ for(int i = 0; i < ui->verticalLayout->count(); i++){ /* Производим каст элемента слоя в объект динамической кнопки * */ QDynamicButton *button = qobject\_cast<QDynamicButton*>(ui->verticalLayout->itemAt(i)->widget()); /* Если номер кнопки соответствует числу, которое установлено * в lineEdit, то производим удаление данной кнопки * */ if(button->getID() == ui->lineEdit->text().toInt()){ button->hide(); delete button; } } } /* СЛОТ для получения номера кнопки. * */ void MainWindow::slotGetNumber() { /* Определяем объект, который вызвал сигнал * */ QDynamicButton *button = (QDynamicButton*) sender(); /* После чего устанавливаем номер кнопки в lineEdit, * который содержится в данной динамической кнопке * */ ui->lineEdit->setText(QString::number(button->getID())); /* То есть номер кнопки устанавливается в поле lineEdit только тогда, * когда мы нажимаем одну из динамических кнопок, и этот номер соответствует * номеру нажатой кнопки * */ }
Барлығы
Нәтижесінде түймелерді динамикалық түрде жасайтын және жойатын қолданба болуы керек. Қолданба төмендегі бейнеде көрсетілген.
В Qt элементы интерфейса имеют базовый класс QWidget . QDynamycButton наследован от QPushButton , который в свою очередь наследован в конечном итоге от QWidget . Из Layout`а указатели на объекты забираются в качестве указателей на объекты класса QWidget .
Таким образом, чтобы получить доступ к методам класса QDynamicButton , объектами которого являются в данном случае объекты, которые находятся в данном Layout`е, необходимо скастовать указатель из класса QWidget в QDynamicButton
Такие возможности языка C++ достигаются за счёт парадигмы полиморфизма. Пример полиморфизма с некоторыми пояснениями дан в этой статье .
Хм, не знаю как у других, а у меня отсутствует метод ui->splitter . Может быть потому что я на линухе сижу
компонент был создан в графическом дизайнере, о чём говорит название объекта ui , в котором находится объект splitter.
А если я неким неизвестным образом добавил разношерстные элементы в ui: checkbox,radiobutton,textedit и label. Они лежат группами одинаковых (одного типа) объектах в vbox, лежащий в groupbox, который в свою очередь находится в vertical layout. Как бы теперь двойным циклом пройтись по каждому из этих объектов в каждом из этих groupbox?
Можно поискать нужные элементы через метод findChildren
или выбрать все групбоксы, а потом каждом использовать метод findChildren.
Метод шаблонный, поэтому можно попытаться найти например только QLineEdit, вместо QWidget, поскольку QWidget - это базовый класс для QLineEdit и остальных виджетов.
Т.е. я могу сделать что-то вроде такого?
Когда я пробую такой вариант, происходит ошибка:candidate function not viable: no known conversion from 'QList<QCheckBox *>' to 'const QList<QWidget *>' for 1st argument, т.е. получается, лист чекбоксов в лист виджетов я все-таки поместить не могу.
Вы с ошибкой написали
Надо так
Не, я в смысле думал, что могу использовать родительский класс для этого, т.е. я хотел бы сделать что-то вроде:
То, что вы так хотели сделать, не значит что это работает в С++. Вы не можете присвоить объект QList<QCheckBox*> к объекту QList<QWidget*> - это разные сущности контейнеров, хотя через методы append вы сможете добавить в контейнер другие классы, у которых родительский класс бцдет QWidget, но в данном случае не будет происходит копирование одного контейнера в объект другого.
И вообще, что вам мешает сделать так?
Ну да, сейчас я уже буду делать тогда так, просто придется создавать листы под каждый тип виджетов. Спасибо!
Скажите пожалуйста, как в VerticalLayout добавить VerticalSpaser так, чтобы при этом кнопки смещались вверх, а не в низ. Если это сделать в дизайнере, кнопки смещаются вниз, т.к. они добавляются после.
используйте метод insertWidget. В нём можно указать индекс, куда добалвять кнопку. Нужно будет найти индекс VerticalSpacer с помощью метода indexOf(verticalSpacer)
и на этот индекс добавлять кнопку.
но у VerticalSpacer нет метода indexOf.
insertWidget использовать вместо addWidget?
У него нет, а вот у verticalLayout есть. Если вы добавите через дизайнер лейаут и спейсер, то можете забрать индекс спейсера так
Да, использовать insertWidget
у меня выдает ошибку D:\QTProject\ReaderResume\mainwindow.cpp:811: ошибка: cannot initialize a parameter of type 'QWidget *' with an lvalue of type 'QSpacerItem *'
А вот про это я забыл... Он же не наследован от QWidget.
Тогда, если вы знаете, что спецсер всегда в конце, то тогда так можете это сделать.
но у verticalLayout нет метода insertWidget
verticalLayout - это, по-моему предположению, должен быть у вас объект класса QVBoxLayout, который наследован от QBoxLayout.
Поэтому открываете документацию на QVBoxLayout .
И ищете там строку List of all members, including inherited members это сразу под краткой анотацией класса. Кликаете по этой надписи и переходите на страницу с полным списком всех методов класса QVBoxLayout.
После этого нужно воспользоваться поиском по странице Ctrl+F, чтобы найти insertWidget. И вы увидите, что там есть этот метод.
Если вы всё-таки использовали QGridLayout, то измените класс QVBoxLayout, если вам нужен имен просто вертикальный тип размещения, а не сетка.
Спасибо. Похоже где то описку сделал, поэтому не работало.
Я добавил на verticalLayout много виджитов. А можно ли их как то быстро и просто удалить?
пройтись циклом по всем виджетам в обратном порядке
Здравствуйте! А не подскажите, как можно при удалении какой либо кнопки, у щётчика отнять значение? Дабы например четвёртой кнопке соответствовал ID 4, а не 5 скажем
Добрый день!
Отнимать значение общего счётчика можно в деструкторе класса кнопки
При этом я бы ещё переустанавливал значения всех ID у всех кнопкок, когда была удалена одна из кнопок, дял этого можно написать метод установки ID, который будет называть setButtonID
Вот его релизация
После чего модифицировать метод удаления кнопок следующим образом
Спасибо большое! Вскоре буду разбираться!
а можно посмотреть финальный проект? Что-то не пойму что я делаю не так
Добрый день! В какую сторону копать чтобы изменить на кнопке созданной динамически например Иконку основываяь на данных полученных из потока. Ну напрепер функция из потока возвращает true олна иконка, false - другая.
Нужно хранить указатели на кнопки или получать их, после чего работать с этим указателем, например задать иконку как то так: указательКнопки->setIcon(QIcon)
Хотелось бы по подробнее про сохранение.
Вот создаю QLabel в потоке проверяю есть с хостом связь или нет.
Срабатывает только на посдний созданный QLabel.
Т.е Основной поток только его идентифицирует.
Думаю упускаете важный момент, в Qt можно создавать виджеты и работать с ними только в главном потоке.
Я из потока получаю только true\false и на основе этого хочу в главном потоке установить ту или иную иконку.
Тогда сделайте массив, запихните в него labelStatus, потом уже обращайтесьь к нему как к члену массива и меняйте как хотите