Tłumaczenie i adaptacja materiałów: dr Tomasz Xięski. Na podstawie prezentacji udostępnionych przez Digia Plc. na licencji CC.

Podobne dokumenty
Tłumaczenie i adaptacja materiałów: dr Tomasz Xięski. Na podstawie prezentacji udostępnionych przez Digia Plc. na licencji CC.

Tłumaczenie i adaptacja materiałów: dr Tomasz Xięski. Na podstawie prezentacji udostępnionych przez Digia Plc. na licencji CC.

Podstawy programowania w Qt4

Tłumaczenie i adaptacja materiałów: dr Tomasz Xięski. Na podstawie prezentacji udostępnionych przez Digia Plc. na licencji CC.

Oprogramowanie i wykorzystanie stacji roboczych. Wykład 4

Qt in Education. Sieć I drukowanie

Qt - dialogi. Bogdan Kreczmer. ZPCiR ICT PWR pokój 307 budynek C3

Backend Administratora

Programowanie w środowiskach RAD QtCreator, Qt i C++

Programowanie obiektowe

The Graphics View Framework. Oprogramowanie i wykorzystanie stacji roboczych. Wykład 5. he Graphics View Framework - architektura

Tabela wewnętrzna - definicja

Qt - podstawowe cechy

ASP.NET MVC. Podstawy. Zaawansowane programowanie internetowe Instrukcja nr 3

Architektura interfejsu użytkownika

Kompleksowe tworzenie aplikacji klasy Desktop z wykorzystaniem SWT i

Programowanie obiektowe i zdarzeniowe wykład 4 Kompozycja, kolekcje, wiązanie danych

Informatyka Edytor tekstów Word 2010 dla WINDOWS cz.3

Programowanie w środowiskach RAD Qt i C++

Qt sygnały i designer

I. Program II. Opis głównych funkcji programu... 19

Informatyka Edytor tekstów Word 2010 dla WINDOWS cz.3

Zaawansowane programowanie w C++ (PCP)

ECDL/ICDL Użytkowanie baz danych Moduł S1 Sylabus - wersja 6.0

MS Excell 2007 Kurs podstawowy Filtrowanie raportu tabeli przestawnej

5. Bazy danych Base Okno bazy danych

Miniaplikacja Kredyty zapewnia dostęp do produktów kredytowych, do których uprawniony jest użytkownik.

1. INFORMACJE O DOKUMENCIE 2. WPROWADZENIE

Szablony klas, zastosowanie szablonów w programach

Laboratorium 7 Blog: dodawanie i edycja wpisów

Techniki programowania INP001002Wl rok akademicki 2018/19 semestr letni. Wykład 3. Karol Tarnowski A-1 p.

Zaawansowane programowanie w języku C++ Klasy w C++

Szablony funkcji i szablony klas

Programowanie obiektowe

Aplikacje w środowisku Java

ACCESS ćwiczenia (zestaw 1)

Instrukcja obsługi funkcji specjalnych szablonu C01 v.1.0

Wykład II. dr Artur Bartoszewski Wydział Nauczycielski, Kierunek Pedagogika Wprowadzenie do baz danych

Informatyka I. Klasy i obiekty. Podstawy programowania obiektowego. dr inż. Andrzej Czerepicki. Politechnika Warszawska Wydział Transportu 2018

Temat: Organizacja skoroszytów i arkuszy

Omówienie wzorców wykorzystywanych w Prism 5.0. Dominika Różycka

JAVA W SUPER EXPRESOWEJ PIGUŁCE

Funkcje standardowe. Filtrowanie

O b s ł u g a p r o g r a m u d o t w o r z e n i a d o k u m e n t ó w z a m ó w i e ń z a k u p u z z a m ó w i e ń k l i e n t ó w

Formularze w programie Word

Wzorce Strukturalne. Adapter: opis. Tomasz Borzyszkowski

Instrukcja użytkownika

Systemy operacyjne na platformach mobilnych

Lokalizacja jest to położenie geograficzne zajmowane przez aparat. Miejsce, w którym zainstalowane jest to urządzenie.

1. Dodawanie integracji

ECDL/ICDL Użytkowanie baz danych Moduł S1 Sylabus - wersja 5.0

W tej sekcji znajdują się foldery gromadzące wszystkie pliki znajdujące się w Twoim projekcie. Nazwa folderu odpowiada rodzaj owi załącznika.

5.3. Tabele. Tworzenie tabeli. Tworzenie tabeli z widoku projektu. Rozdział III Tworzenie i modyfikacja tabel

Tworzenie bazy danych na przykładzie Access

Wykład 4: Klasy i Metody

Systemy baz danych Prowadzący: Adam Czyszczoń. Systemy baz danych. 1. Import bazy z MS Access do MS SQL Server 2012:

Listy powiązane zorientowane obiektowo

Architektura Systemu. Architektura systemu umożliwia kontrolowanie iteracyjnego i przyrostowego procesu tworzenia systemu.

Instrukcja zmiany stawki VAT oraz innych informacji dodatkowych dotyczących kartoteki asortymentowej oraz cenników w systemie MAAT

W powyższym kodzie utworzono wyliczenie dni tygodnia.

Sylabus Moduł 2: Przetwarzanie tekstów

Zwróćmy uwagę w jakiej lokalizacji i pod jaką nazwą zostanie zapisana baza (plik z rozszerzeniem *.accdb). Nazywamy

Dane słowa oraz wyrażenia są tłumaczone przy pomocy polecenia Przetwarzanie > Tłumaczenie

Aktywności są związane z ekranem i definiują jego wygląd. Dzieje się to poprzez podpięcie do aktywności odpowiedniego widoku.

Podstawowe funkcjonalności interfejsu. - co warto wiedzieć o interfejsie Mozart-a

Wykład III. dr Artur Bartoszewski Wydział Nauczycielski, Kierunek Pedagogika Wprowadzenie do baz danych

Projekt Hurtownia, realizacja skojarzeń dostawców i produktów

Tworzenie prezentacji w MS PowerPoint

MS Excel 2007 Kurs zaawansowany Obsługa baz danych. prowadzi: Dr inż. Tomasz Bartuś. Kraków:

Sylabus Moduł 4: Grafika menedżerska i prezentacyjna

1. Zaczynamy! (9) 2. Edycja dokumentów (33)

Część I Rozpoczęcie pracy z usługami Reporting Services

Implementacja aplikacji sieciowych z wykorzystaniem środowiska Qt

Kurs ECDL Moduł 3. Nagłówek i stopka Microsoft Office Word Autor: Piotr Dębowski. piotr.debowski@konto.pl

1. Które składowe klasa posiada zawsze, niezależnie od tego czy je zdefiniujemy, czy nie?

Sieciowe Technologie Mobilne. Laboratorium 2

TP1 - TABELE PRZESTAWNE od A do Z

UONET+ - moduł Sekretariat. Jak wykorzystać wydruki list w formacie XLS do analizy danych uczniów?

Klasy Obiekty Dziedziczenie i zaawansowane cechy Objective-C

WinSkład / WinUcz 15.00

Widoki zagnieżdżone, layout. 1. Wprowadzenie Repozytoria danych

Wprowadzenie (17) Część I. Makra w Excelu - podstawy (23)

Przedszkolaki Przygotowanie organizacyjne

UML a kod w C++ i Javie. Przypadki użycia. Diagramy klas. Klasy użytkowników i wykorzystywane funkcje. Związki pomiędzy przypadkami.

Wstęp 7 Rozdział 1. OpenOffice.ux.pl Writer środowisko pracy 9

Instrukcja obsługi Zaplecza epk w zakresie zarządzania tłumaczeniami opisów procedur, publikacji oraz poradników przedsiębiorcy

Kurs programowania 2 - listy

Wykład 3 Składnia języka C# (cz. 2)

private - oznacza, że wszystkie elementy klasy bazowej zmieniają się w prywatne.

Algorytmy i Struktury Danych. Anna Paszyńska

Technologie i usługi internetowe cz. 2

INSTRUKCJA OBSŁUGI DO PROGRAMU I-ANALITYK GT WERSJA i-sys Integracja Systemów Spółka z o.o. ul. Zwoleńska 127 lok. 32, Warszawa

Arkusz kalkulacyjny. R. Robert Gajewski omklnx.il.pw.edu.pl/~rgajewski

OBIEKTY TECHNICZNE OBIEKTY TECHNICZNE

Java - tablice, konstruktory, dziedziczenie i hermetyzacja

Przewodnik Szybki start

Builder (budowniczy) Cel: Przykład:

Temat 3. Projektowanie interfejsu użytkonwnika Kalkulator pierwszy program dla IOS

Dynamiczne i wydajne tworzenie interfejsu. Piotr Michałkiewicz

Transkrypt:

Model MVC w Qt

Tłumaczenie i adaptacja materiałów: dr Tomasz Xięski. Na podstawie prezentacji udostępnionych przez Digia Plc. na licencji CC. 2012 Digia Plc. The enclosed Qt Materials are provided under the Creative Commons Attribution-Share Alike 2.5 License Agreement. The full license text is available here: http://creativecommons.org/licenses/by-sa/2.5/legalcode. Digia, Qt and the Digia and Qt logos are the registered trademarks of Digia Plc. in Finland and other countries worldwide.

Celem wzorca MVC jest separacja danych (model) wizualizacji (widok) modyfikacji (kontroler) View Model Controller Każdy z elementów wchodzących w jego skład ma ściśle określony zestaw obowiązków i nie powinno się ich mieszać dokonując jak najczęściej tzw. refactoring kodu.

Oddziela dane od ich graficznej reprezentacji Unika się powielania danych Może pokazywać te same dane w wielu widokach Może wykorzystywać ten sam widok dla wielu różnych danych Oddziela warstwę wizualną od modyfikacji Jeden widok może być wykorzystywany przez wiele kontrolerów (wyświetlanie, edycja, usuwanie) Ten sam widok może być wykorzystywany w wielu różnych zastosowaniach

Implementacja modelu MVC w Qt jest nieco inna niż klasyczna, ale działa w ten sam sposób: Warstwa modelu i widoku jest obecna, ale kontroler nazywany jest delegatem i ma większe możliwości Delegat odpowiada również za zmianę widoku podczas edytowania wartości elementu Występuje dodatkowo tzw. selection model, który synchronizuje zaznaczone elementy danych dla wielu widoków. Model Delegate Selection Model View

Wszystkie modele oparte są o klasę QAbstractItemModel, która wspiera: Listy dane w jednej kolumnie, wiele wierszy Tabele dane w wielu wierszach i kolumnach Drzewa zagnieżdżone tabele Dane nie muszą być trzymane w modelu mogą pochodzić np. z pliku lub bazy danych.

Model ten przechowuje wiele elementów w formie jednej kolumny. Każdy element jest adresowany przez QModelIndex uzyskuje się dzięki temu separację między reprezentacją danych a sposobem dostępu do nich.

Model oparty o tabele rozmieszcza elementy w siatce składającej się z wierszy i kolumn Dostęp do danych: QModelIndex indexa = model->index(0, 0, QModelIndex()); QModelIndex indexb = model->index(1, 1, QModelIndex()); QModelIndex indexc = model->index(2, 1, QModelIndex());

Analizowany model może być postrzegany jako agregacja zagnieżdżonych tabel. Dostęp do każdego elementu odbywa się przez podanie rodzica, numeru wiersza i kolumny. Dostęp do elementów: QModelIndex indexa = model->index(0, 0, QModelIndex()); QModelIndex indexc = model->index(2, 1, QModelIndex()); QModelIndex indexb = model->index(1, 0, indexa);

Każdy model ma metodę data() używaną do odczytu danych QVariant QAbstractItemModel::data( const QModelIndex &index, int role) const Drugi argument określa używaną aktualnie rolę (domyślnie Qt::DisplayRole): DecorationRole dla ikon, kolorów itp EditRole dane przystosowane do edycji FontRole używana domyślnie czcionka CheckStateRole rola do przechowywania zaznaczenia elementów

Używany jest do adresacji indywidualnych elementów w danym modelu QModelIndex zapewnia następujące metody: data(role) dostęp do danych child(row, column) dostęp do potomka parent() dostęp do rodzica Do tworzenia nowych modeli zaleca się wykorzystanie klasy QAbstractItemModel, która zapewnia m.in. metody: index(row, column, parent=qmodelindex()) zwraca indeks elementu dla zadanego modelu rowcount(parent=qmodelindex()) liczba wierszy (dla zadanego rodzica) columncount(parent=qmodelindex()) liczba kolumn (dla zadanego rodzica) parent(index) rodzic zadanego elementu

Qt dostarcza szeregu gotowych modeli, wśród których najpopularniejsze to: QStringListModel model do przechowywania listy ciągów znaków QFileSystemModel model umożliwiający dostęp do systemu plików QStandardItemModel model przechowujący obiekty typu QStandardItem. Wykorzystywany do tworzenia list, tablic, drzew. QSqlQueryModel, QSqlTableModel, QSqlRelationalTableModel modele wykorzystywane do interakcji z bazą danych

Wszystkie widoki dziedziczą po klasie QAbstractItemView Najważniejsze dostępne widoki to: QListView wyświetla tekst i/lub ikony QTableView QTreeView QColumnView

Wyświetla pojedynczą kolumnę zestawu danych Właściwość modelcolumn określa którą kolumnę należy wyświetlić Dwa tryby: IconMode oraz ListMode

Pokazuje siatkę elementów: Metod hiderow oraz hidecolumn służą do ukrywania elementów Za ponowne wyświetlanie elementów odpowiadają metody showrow oraz showcolumn

Umożliwia dostosowanie się do zawartości wykorzystując metody resizecolumnstocontents oraz resizerowstocontents Dostęp do nagłówków gwarantują metody verticalheader lub horizontalheader Właściwość stretchlastsection powoduje rozciągnięcie zawartości do rozmiarów komponentu inaczej może zostać puste miejsce Nagłówki mogą być widoczne lub nie Dostęp do pasków przewijania umożliwiają właściwości horizontalscrollbarpolicy oraz verticalscrollbarpolicy

Struktura drzewiasta Użyj setrowhidden oraz setcolumnhidden by pokazać bądź ukryć zawartość Użyj expandall, expandtodepth oraz collapseall by kontrolować jak wiele danych pokazać naraz

Hierarchia w poziomie Na ostatnim poziomie może agregować komponent do graficznego pokazania danych np. miniatury obrazka

Wykorzystując QDataWidgetMapper możliwe jest powiązanie danych z poszczególnymi widgetami QDataWidgetMapper *mapper = new QDataWidgetMapper; mapper->setmodel(model); mapper->addmapping(cityedit, 0); mapper->addmapping(populationedit, 1); mapper->tofirst(); connect(nextbutton, SIGNAL(clicked()), mapper, SLOT(toNext())); connect(prevbutton, SIGNAL(clicked()), mapper, SLOT(toPrevious()));

W niektórych zastosowaniach separacja modelu od widoku nie jest wskazana: Gdy dane nie są powielane Gdy model musi indywidualnie przetworzyć dane (implikując wykonanie ich kopii) Dla takich zastosowań wykorzystuje się klasy QListWidget, QTableWidget oraz QTreeWidget Przechowują one elementy kolejno typu QListWidgetItem, QTableWidgetItem oraz QTreeWidgetItem.

Delegat odpowiada za edytowanie i wizualną reprezentację elementu (w tym stanie) Widok wykorzystuje i wchodzi w interakcję z delegatem Wszystkie delegaty implementują interfejs QAbstractItemDelegate Standardowo wykorzystywany jest delegat QStyledItemDelegate Model Delegate Selection Model View

Delegat QStyledItemDelegate akceptuje następujące typy Role Types danych CheckStateRole Qt::CheckState DecorationStyle DisplayRole EditRole QIcon, QPixmap, QImage and QColor QString (QVariant::toString()) Klasa QItemEditorFactor określa jaki widget zostanie wykorzystany dla danego typu Type bool double int / unsigned int QDate QDateTime QPixmap QString QTime Widget QComboBox QDoubleSpinBox QSpinBox QDateEdit QDateTimeEdit QLabel QLineEdit QTimeEdit

Możliwa jest implementacja własnych delegatów do obsługi rysowania i/lub edycji Jeżeli interesuje nas tylko edycja, najprościej dokonać redefinicji klasy QItemEditorCreatorBase Delegaty przyporządkowywane są do całego widoku, bądź poszczególnych kolumn i wierszy

Należy dokonać redefinicji metod paint oraz sizehint class BarDelegate : public QStyledItemDelegate { Q_OBJECT public: explicit BarDelegate(int maxrange, QObject *parent = 0); void paint(qpainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const; QSize sizehint(const QStyleOptionViewItem &option, const QModelIndex &index) const; private: int m_maxrange; };

BarDelegate::BarDelegate(int maxrange, QObject *parent) : QStyledItemDelegate(parent), m_maxrange(maxrange) { } QSize BarDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const { return QSize(100, 1); } Implementacja konstruktora jest trywialna przypisywana jest wartość pola m_maxrange. Implementacja sizehint ustawia maksymalny rozmiar.

void BarDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { if(index.data().canconvert<int>()) { QRect barrect = QRect(option.rect.topLeft(), QSize(option.rect.width()*((qreal)index.data().toInt()/(qreal)m_maxRange), option.rect.height())); barrect.adjust(0, 2, 0, -2); if(option.state & QStyle::State_Selected) { painter->fillrect(option.rect, option.palette.highlight()); painter->fillrect(barrect, option.palette.highlightedtext()); } else painter->fillrect(barrect, option.palette.text()); } else QStyledItemDelegate::paint(painter, option, index); } Zakłada się, że dane można zrzutować na typ integer w przeciwnym razie wywoływane jest domyślne zachowanie

tableview->setmodel(model); tableview->setitemdelegateforcolumn(1, new BarDelegate(3000000, this));

Należy dokonać redefinicji metod: createeditor, seteditordata, setmodeldata oraz updateeditorgeometry class BarDelegate : public QStyledItemDelegate { Q_OBJECT public:... QWidget *createeditor(qwidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index ) const; void seteditordata(qwidget *editor, const QModelIndex &index) const; void updateeditorgeometry(qwidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const; void setmodeldata(qwidget *editor, QAbstractItemModel *model, const QModelIndex &index) const;... Należy wykorzystywać rolę EditRole zamiast DisplayRole przy edycji

QWidget *BarDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index ) const { QSlider *slider = new QSlider(parent); slider->setrange(0, m_maxrange); slider->setorientation(qt::horizontal); slider->setautofillbackground(true); } return slider; void BarDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const { QSlider *slider = qobject_cast<qslider*>(editor); if(slider) slider->setgeometry(option.rect); }

void BarDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const { QSlider *slider = qobject_cast<qslider*>(editor); if(slider) slider->setvalue(index.data(qt::editrole).toint()); } void BarDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const { QSlider *slider = qobject_cast<qslider*>(editor); if(slider) model->setdata(index, slider->value(), Qt::EditRole); }

tableview->setmodel(model); tableview->setitemdelegateforcolumn(1, new BarDelegate(3000000, this));

Tworząc swoje delegaty często zachodzi potrzeba przekazania dodatkowych danych między modelem a delegatem Umożliwia to mechanizm tworzenia własnych rul Użyj Qt::UserRole jako pierwszą class CustomRoleModel : public QAbstractListModel { Q_OBJECT public: enum MyTypes { FooRole = Qt::UserRole, BarRole, BazRole };...

Możliwe jest sortowanie i filtrowanie modeli wykorzystując warstwę pośrednią tzw. proxy QAbstractProxyModel dostarcza Połączenie między modelami Połączenie miedzy zaznaczonymi elementami (selection model) Klasa QSortFilterProxyModel uproszcza zadanie dostarczając metod do sortowania i filtrowania danych Właściwość dynamicsortfilter określa czy rezultat ma być buforowany czy generowany dynamicznie

Jeżeli ustawiono właściwość sortingenabled kliknięcie na nagłówek sortuje wartość w danej kolumnie Dotyczy to komponentów QTableView oraz QTreeView Wykorzystując QSortFilterProxyModel możliwe jest posortowanie określonej kolumny i określonej roli sortrole która rola jest brana pod uwagę (domyślnie DisplayRole) sortcasesensitivity czy wielkość liter ma znaczenie

QSortFilterProxyModel *sortingmodel = new QSortFilterProxyModel(this); sortingmodel->sort(0, Qt::AscendingOrder); sortingmodel->setdynamicsortfilter(true); sortingmodel->setsourcemodel(model); nonsortedview->setmodel(model); sortedview->setmodel(sortingmodel);

By zaimplementować bardziej złożony algorytm sortowania należy zredefiniować metodę lessthan bool MySortProxyModel::lessThan(const QModelIndex &left, const QModelIndex &right) const { if(left.data().tostring().length() == right.data().tostring().length()) return left.data().tostring() < right.data().tostring(); else return (left.data().tostring().length() < right.data().tostring().length()); } MySortProxyModel *customsortmodel = new MySortProxyModel(this); customsortmodel->sort(0, Qt::DescendingOrder); customsortmodel->setdynamicsortfilter(true); customsortmodel->setsourcemodel(model); customsortedview->setmodel(customsortmodel);

Filtrowanie umożliwia redukcję liczby wierszy i kolumn w modelu: filterregexp / filterwildcard / filterfixedstring możliwe jest używanie wyrażeń regularnych bądź określonego ciągu do dopasowania filtercasesensitivity wielkość liter może być brana pod uwagę filterrole filtrowana jest określona rola filterkeycolumn klucz na podstawie którego filtrowane są rekordy (jeżeli nie podano, klucz tworzą wszystkie kolumny)

QSortFilterProxyModel *filteringmodel = new QSortFilterProxyModel(this); filteringmodel->setfilterwildcard("*stad*"); filteringmodel->setfilterkeycolumn(0); filteringmodel->setdynamicsortfilter(true); filteringmodel->setsourcemodel(model); nonfilteredview->setmodel(model); filteredview->setmodel(filteringmodel);

Implementacja bardziej złożonych filtrów wymaga redefinicji metod filteracceptrow oraz filteracceptcolumn bool filteracceptsrow(int sourcerow, const QModelIndex &sourceparent) const { const QModelIndex &index = sourcemodel()->index(sourcerow, filterkeycolumn(), sourceparent); return index.data().tostring().contains("berg") index.data().tostring().contains("stad"); } MyFilterProxyModel *customfiltermodel = new MyFilterProxyModel(this); customfiltermodel->setfilterkeycolumn(0); customfiltermodel->setdynamicsortfilter(true); customfiltermodel->setsourcemodel(model); customfilteredview->setmodel(customfiltermodel);

Wybór danych obsługiwany jest przez dedykowany selection model Możliwym jest wprowadzenie ograniczeń zaznaczania: Pojedyncze elementy / wiersze / kolumny Pojedyncze zaznaczenie/ ciągłe / rozszerzone / wielokrotne / brak Model wybranych elementów może być współdzielony między widokami Model Delegate Selection Model View

view->setselectionbehavior( QAbstractItemView::SelectItems); view->setselectionmode( QAbstractItemView::SingleSelection); view->setselectionbehavior( QAbstractItemView::SelectRows); view->setselectionmode( QAbstractItemView::ContiguousSelection); view->setselectionbehavior( view->setselectionmode( QAbstractItemView::SelectColumns); QAbstractItemView::ExtendedSelection);

Współdzielenie zaznaczenia między różnymi widokami: listview->setmodel(model); tableview->setmodel(model); listview->setselectionmodel( tableview->selectionmodel());

Połączenia należy zestawiać z modelem zaznaczenia a nie konkretnym widokiem connect(view->selectionmodel(), SIGNAL(selectionChanged(QItemSelection,QItemSelection)), this, SLOT(updateSelectionStats())); void Widget::updateSelectionStats() { indexeslabel->settext(qstring::number(view->selectionmodel()->selectedindexes().count())); rowslabel->settext(qstring::number(view->selectionmodel()->selectedrows().count())); columnslabel->settext(qstring::number(view->selectionmodel()->selectedcolumns().count())); } removebutton->setenabled(view->selectionmodel()->selectedindexes().count() > 0);