青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

zhonghua

C++博客 首頁 新隨筆 聯(lián)系 聚合 管理
  72 Posts :: 1 Stories :: 4 Comments :: 0 Trackbacks

#

該類為單張的數(shù)據(jù)庫表提供了一個(gè)可編輯的數(shù)據(jù)模型,它支持外鍵,除此之外和QSqlTableModel沒有什么不同
  1. model = new QSqlRelationalTableModel(this);  
  2. model->setEditStrategy(QSqlTableModel::OnFieldChange); //屬性變化時(shí)寫入數(shù)據(jù)庫  
  3. model->setTable("student");  
  4. model->setRelation(2,QSqlRelation("course","id","name"));//將student表的第三個(gè)屬性設(shè)為course表的id屬性的外鍵,并將其顯示為course表的name屬性的值  
  5. model->setHeaderData(0, Qt::Horizontal, QObject::tr("ID"));  
  6. model->setHeaderData(1, Qt::Horizontal, QObject::tr("Name"));  
  7. model->setHeaderData(2, Qt::Horizontal, QObject::tr("Course"));  
  8. model->select();  
  9. tableView->setModel(model);  

 

如果我們希望用戶更改課程屬性時(shí),只能在課程表中已有的課程中進(jìn)行選擇,而不能隨意填寫課程,那么Qt中的QSqlRelationalDelegate委托類就能實(shí)現(xiàn)這個(gè)功能

  1. tableView->setItemDelegate(new QSqlRelationalDelegate(tableView)); 

例子:

QSqlRelationalTableModel,該類為單張的數(shù)據(jù)庫表提供了一個(gè)可編輯的數(shù)據(jù)模型,它支持外鍵。
我 們還是新建Qt4 Gui Application工程,我這里工程名為relationalTableModel ,然后選中QtSql模塊,Base class選QWidget。工程建好后,添加C++ Header File ,命名為database.h,更改其內(nèi)容如下:
#ifndef DATABASE_H
#define DATABASE_H

#include <QSqlDatabase>
#include <QSqlQuery>

static bool createConnection()
{
    QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
    db.setDatabaseName("database.db");
    if(!db.open()) return false;
    QSqlQuery query;
    query.exec("create table student (id int primary key, name vchar,course int)");
    query.exec("insert into student values (1,'yafei0',1)");
    query.exec("insert into student values (2,'yafei1',1)");
    query.exec("insert into student values (3,'yafei2',2)");

    query.exec("create table course (id int primary key, name vchar, teacher vchar)");
    query.exec("insert into course values (1,'Math','yafeilinux1')");
    query.exec("insert into course values (2,'English','yafeilinux2')");
    query.exec("insert into course values (3,'Computer','yafeilinux3')");
    return true;
}

#endif // DATABASE_H
我們在這里建立了兩個(gè)表,student表中有一項(xiàng)是course,它是int型的,而course表的主鍵也是int型的。如果要將course項(xiàng)和course表進(jìn)行關(guān)聯(lián),它們的類型就必須相同,一定要注意這一點(diǎn)。
然后將main.cpp中的內(nèi)容更改如下:
#include <QtGui/QApplication>
#include "widget.h"
#include "database.h"
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    if(!createConnection()) return 1;
    Widget w;
    w.show();
    return a.exec();
}


我們在widget.h中添加頭文件: #include <QSqlRelationalTableModel>
然后在private中聲明對(duì)象:    QSqlRelationalTableModel *model;
我們在widget.ui中添加一個(gè)Table View部件到窗體上,然后到widget.cpp中的構(gòu)造函數(shù)里添加如下代碼:
    model = new QSqlRelationalTableModel(this);
    model->setEditStrategy(QSqlTableModel::OnFieldChange); //屬性變化時(shí)寫入數(shù)據(jù)庫
    model->setTable("student");
    model->setRelation(2,QSqlRelation("course","id","name"));
    //將student表的第三個(gè)屬性設(shè)為course表的id屬性的外鍵,并將其顯示為course表的name屬性的值
    model->setHeaderData(0, Qt::Horizontal, QObject::tr("ID"));
    model->setHeaderData(1, Qt::Horizontal, QObject::tr("Name"));
    model->setHeaderData(2, Qt::Horizontal, QObject::tr("Course"));
    model->select();
    ui->tableView->setModel(model);

我們修改了model的提交策略,OnFieldChange表示只要屬性被改動(dòng)就馬上寫入數(shù)據(jù)庫,這樣就不需要我們再執(zhí)行提交函數(shù)了。setRelation()函數(shù)實(shí)現(xiàn)了創(chuàng)建外鍵,注意它的格式就行了。
運(yùn)行效果如下:


可以看到Course屬性已經(jīng)不再是編號(hào),而是具體的課程了。關(guān)于外鍵,你也應(yīng)該有一定的認(rèn)識(shí)了吧,說簡單點(diǎn)就是將兩個(gè)相關(guān)的表建立一個(gè)橋梁,讓它們關(guān)聯(lián)起來。
那么我們也希望,如果用戶更改課程屬性,那么他只能在課程表中有的課程中進(jìn)行選擇,而不能隨意填寫課程。在Qt中的QSqlRelationalDelegate委托類就能實(shí)現(xiàn)這個(gè)功能。我們只需在上面的構(gòu)造函數(shù)的最后添加一行代碼:
ui->tableView->setItemDelegate(new QSqlRelationalDelegate(ui->tableView));
添加代理(委托),在我這里不知為什么會(huì)出現(xiàn)SqlRelationalDelegate is not a type name的提示,不過可以編譯通過。
我們需要在widget.cpp中添加頭文件: #include <QSqlRelationalDelegate>
運(yùn)行效果如下:


可以看到這時(shí)修改Course屬性時(shí),就會(huì)出現(xiàn)一個(gè)下拉框,只能選擇course表中的幾個(gè)值。
而利用這個(gè)類來操作數(shù)據(jù)庫,與前面講到的QSqlTableModel沒有區(qū)別,這里就不再重復(fù)。這幾篇文章一共講了好幾種操作數(shù)據(jù)庫的方法,到底應(yīng)該使用哪個(gè)呢?那就看你的需求了,根據(jù)這幾種方法的特點(diǎn)進(jìn)行選擇吧。


posted @ 2012-04-24 14:39 米米 閱讀(550) | 評(píng)論 (0)編輯 收藏

  QSqlTableModel類繼承至QSqlQueryModel類,該類提供了一個(gè)可讀寫單張SQL表的可編輯數(shù)據(jù)模型,功能:修改,插入,刪除,查詢,和排序

常用函數(shù)

QVariant headerData ( int section,Qt::Orientation orientation, int role = Qt::DisplayRole ) const  獲取水平頭或垂直頭標(biāo)題

bool setHeaderData ( int section,Qt::Orientation orientation, const QVariant & value, int role = Qt::EditRole ) 設(shè)置水平頭或垂直頭標(biāo)題

int rowCount ( const QModelIndex & parent= QModelIndex() ) const // 返回行數(shù)

int columnCount ( const QModelIndex &index = QModelIndex() ) const // 返回列數(shù)

virtual bool removeColumns ( int column, int count, const QModelIndex & parent = QModelIndex() ) //model->removeColumns (0)刪除第一列

bool QSqlTableModel::submitAll (),//提交所有被修改的數(shù)據(jù),然后修改的數(shù)據(jù)被保存在數(shù)據(jù)庫中

void QSqlTableModel::revertAll ()  //撤銷所有的修改,如果數(shù)據(jù)庫已經(jīng)被提交了修改,就不能通過撤銷修改改回來了

virtual void revertRow ( int row ) //恢復(fù)指定行的改變

void QSqlTableModel::setFilter ( const QString & filter )  //篩選,按照字符串filter對(duì)數(shù)據(jù)庫進(jìn)行篩選,相當(dāng)于SQL中的WHERE語句

bool QSqlTableModel::select ()   //在篩選和排序的條件下,將數(shù)據(jù)庫中符合要求的在mode表格中顯示出來

void QSqlTableModel::setSort ( int column, Qt::SortOrder order ) //排序操作。按照列和Qt::SortOrder排序。Qt::SortOrder有升序和降序

bool insertRow ( int row, const QModelIndex & parent = QModelIndex() )  //插入行

bool insertColumn ( int column, constQModelIndex & parent = QModelIndex() ) // 插入列

model->setEditStrategy(QSqlTableModel::OnManualSubmit);   //設(shè)置保存策略為手動(dòng)提交

 

一、在QTableView中顯示數(shù)據(jù)庫中表的數(shù)據(jù)

  1. QSqlTableModel *model = new QSqlTableModel(parentObject, database); // 摘抄自幫助文檔  
  2. model->setTable("employee");  
  3. model->setEditStrategy(QSqlTableModel::OnManualSubmit);  
  4. model->select();  
  5. model->removeColumn(0); // don't show the ID  
  6. model->setHeaderData(0, Qt::Horizontal, tr("Name"));  
  7. model->setHeaderData(1, Qt::Horizontal, tr("Salary"));  
  8.   
  9. QTableView *view = new QTableView;  
  10. view->setModel(model);  
  11. view->show();  


二、修改QTableView中數(shù)據(jù)后的提交,加入事務(wù)處理

  1. model->database().transaction(); //開始事務(wù)操作  
  2. if (model->submitAll()) // 提交所有被修改的數(shù)據(jù)到數(shù)據(jù)庫中  
  3. {  
  4.     model->database().commit(); //提交成功,事務(wù)將真正修改數(shù)據(jù)庫數(shù)據(jù)  
  5. else {  
  6.     model->database().rollback(); //提交失敗,事務(wù)回滾  
  7.     QMessageBox::warning(this, tr(“tableModel”),tr(“數(shù)據(jù)庫錯(cuò)誤: %1″).arg(model->lastError().text()));  
  8. }  
  9.     model->revertAll(); //撤銷修改  

 

三、查詢操作

相當(dāng)于SQL語句:SELECT * FROM 表名 WHERE name = "name變量"

  1. model->setFilter(QObject::tr(“name = ‘%1′”).arg(name)); //根據(jù)姓名進(jìn)行篩選  
  2. model->select(); //顯示結(jié)果  
  3. for (int i = 0; i < model.rowCount(); ++i)  
  4. {  
  5.     QString name = model.record(i).value("name").toString();  
  6.     // ... 在此處理每一條的記錄  
  7. }  
  8.   
  9. // 在操作大數(shù)據(jù)集時(shí),建議通過索引指定字段  
  10. int primaryKeyIndex  = model.record().indexOf("id");  
  11. for (int i = 0; i < model.rowCount(); ++i)  
  12. {  
  13.     QSqlRecord record = model.record(i);  
  14.     QString name = record.value("name").toString();  
  15.     // ... 在此處理每一條的記錄  
  16. }  


四、排序操作

  1. model->setSort(0,Qt::AscendingOrder); //id屬性,即第0列,升序排列,Qt::DescendingOrder為降序排序  
  2. model->select();  


五、插入操作

  1. int rowNum = model->rowCount(); //獲得表的行數(shù)  
  2. int id = 最后一個(gè)ID+1;  
  3. model->insertRow(rowNum); //添加一行,或者用insertRows(0,1),在0行添加1條記錄,根據(jù)表的排序規(guī)則,可能移到與指定行不同的行位置上  
  4. model->setData(model->index(rowNum,0),id);  //因?yàn)檫@里設(shè)置了ID為主鍵,所以必須給新行添加id屬性值,id字段在第0列上  
  5. model->submitAll(); //可以直接提交  

 

六、刪除一條記錄

首先要定位到待刪除的行上

  1. model.setFilter("id = 10");  
  2. model.select();  
  3. if (model.rowCount() == 1)  
  4. {  
  5.     model.removeRows(0,1) // 如果要?jiǎng)h除所有滿足條件的記錄則把1改成model.rowCount()  
  6.     model.submitAll();  
  7. }  

 

在QTableView中刪除選中的一行

  1. int curRow = tableView->currentIndex().row();  
  2. model->removeRow(curRow);   //刪除一行  


在QTableView中刪除選中的多行

QAbstractItemView::SelectionModeselectionMode()const // 原型

QModelIndexList QItemSelectionModel::selectedIndexes()const  //原型

  1. QItemSelectionModel *selections = tableView->selectionModel(); //返回當(dāng)前的選擇模式  
  2. QModelIndexList selecteds = selections->selectedIndexes(); //返回所有選定的模型項(xiàng)目索引列表  
  3. foreach (QModelIndex index, selecteds)  
  4. {  
  5.     int curRow = index.row(); //刪除所有被選中的行  
  6.     model->removeRow(curRow);  
  7. }  
  8.   
  9. int ok = QMessageBox::warning(this,tr("刪除選中的行!"),tr("你確定刪除當(dāng)前選取中的行嗎?"),QMessageBox::Yes,QMessageBox::No);  
  10. if(ok == QMessageBox::Yes)  
  11. {  
  12.     model->submitAll(); //提交,在數(shù)據(jù)庫中刪除該行  
  13. else {  
  14.     model->revertAll(); //如果不刪除,則撤銷  
  15. }  



七、更新記錄

必須先定位記錄

  1. model.setFilter("id = 10");  
  2. model.select();  
  3. if (model.rowCount() == 1)  
  4. {  
  5.     model.setData(model.index(0,1),QObject::tr("小王"));  
  6.     model.submitAll();  
  7. }  


    可以看到這個(gè)模型很強(qiáng)大,而且完全脫離了SQL語句,就算你不怎么懂?dāng)?shù)據(jù)庫,也可以利用它進(jìn)行大部分常用的操作。這個(gè)模型提供了緩沖區(qū),可以將所有修改先 保存到model中,只有當(dāng)我們執(zhí)行提交修改后,才會(huì)真正寫入數(shù)據(jù)庫。當(dāng)然這也是因?yàn)槲覀冊谧铋_始設(shè)置了它的保存策略:

model->setEditStrategy(QSqlTableModel::OnManualSubmit);

OnManualSubmit表明我們要提交修改才能使其生效。可以先將修改保存起來,當(dāng)我們執(zhí)行提交函數(shù)時(shí),再去真正地修改數(shù)據(jù)庫。當(dāng)然,這個(gè)模型比前面的模型更高級(jí),前面講的所有操作,在這里都能執(zhí)行。

posted @ 2012-04-23 11:22 米米 閱讀(553) | 評(píng)論 (0)編輯 收藏

    QSqlQueryModel類為SQL的結(jié)果集提供了一個(gè)只讀的數(shù)據(jù)模型,下面我們先利用這個(gè)類進(jìn)行一個(gè)最簡單的操作.

常用函數(shù)

void QSqlQueryModel::setQuery ("SQL語句") // 執(zhí)行SQL語句,此處還可以傳入QSqlQuery對(duì)象,此時(shí)可以利用QSqlQuery類的某些特性,如預(yù)操作等.

setHeaderData()     //設(shè)置水平頭標(biāo)題

columnCount(); //獲得列數(shù)

columnCount(); //獲得列數(shù)

QSqlRecord QSqlQueryModel::record ( int row ) const //返回row行包含的信息,可訪問單條的記錄

QModelIndex QAbstractItemModel::index ( int row, int column, const QModelIndex & parent = QModelIndex() )  //返回指定的行和列的索引(index)

index.data()  //返回index索引的值

query() //返回與QSqlQuery相關(guān)的模型

  1. QSqlQueryModel *model = new QSqlQueryModel;  
  2. model->setQuery(“select * from student”);  
  3. model->setHeaderData(0, Qt::Horizontal, tr(“id”));  
  4. model->setHeaderData(1, Qt::Horizontal, tr(“name”));  
  5. QTableView *view = new QTableView;  
  6. view->setModel(model);  
  7. view->show();  

 

利用query執(zhí)行SQL語句

  1. QSqlQuery query = model->query();  
  2. query.exec("select name from student where id = 1");  
  3. query.next();  
  4. qDebug() << query.value(0).toString(); // 如果上面的select改成insert語句,而且是顯示在QTableView中的話,需再查詢一次model->setQuery("select ...")才能顯示剛插入的語句  


       因?yàn)镼SqlQueryMode模型默認(rèn)是只讀的,所以我們在窗口上并不能對(duì)表格中的內(nèi)容進(jìn)行修改。但是我們可以創(chuàng)建自己的模型,然后按照我們自己的需要來顯示數(shù)據(jù)和修改數(shù)據(jù)。如果要想使其可讀寫,需要自己的類繼承自QSqlQueryModel,并且重寫setData() 和 flags() 兩個(gè)函數(shù)如果我們要改變數(shù)據(jù)的顯示,就要重寫data() 函數(shù)

  1. Qt::ItemFlags MySqlQueryModel::flags(const QModelIndex &index) const //返回表格是否可更改的標(biāo)志  
  2. {  
  3.      Qt::ItemFlags flags = QSqlQueryModel::flags(index);  
  4.      if (index.column() == 1) //第二個(gè)字段可更改,即學(xué)生的名字字段  
  5.         flags |= Qt::ItemIsEditable;  
  6.      return flags;  
  7.  }  
  8.   
  9. bool MySqlQueryModel::setData(const QModelIndex &index, const QVariant &value, int /* role */) //表格添加數(shù)據(jù)  
  10. {  
  11.     QModelIndex primaryKeyIndex = QSqlQueryModel::index(index.row(), 0);  
  12.     int id = data(primaryKeyIndex).toInt(); //獲取id號(hào) 也可以這樣用,primaryKeyIndex.data();  
  13.   
  14.     clear();  
  15.     bool isOk;  
  16.     if (index.column() == 1) //第二個(gè)屬性可更改  
  17.     {  
  18.         QSqlQuery query;  
  19.         query.prepare("UPDATE STUDENT SET NAME = :name WHERE id = :id");  
  20.         query.bindValue(":name","小五");  
  21.         query.bindValue(":id",id);  
  22.         isOk = query.exec();  
  23.   
  24.         refresh();// 此處最好添加代碼刷新結(jié)果,或在此調(diào)用刷新函數(shù)  
  25.         return isOK;  
  26.     }  
  27.     return false;  
  28. }  
  29.   
  30. void MySqlQueryModel::refresh() //更新顯示  
  31. {  
  32.     setQuery("select * from student");  
  33.     setHeaderData(0, Qt::Horizontal, QObject::tr("學(xué)號(hào)ID"));  
  34.     setHeaderData(1, Qt::Horizontal, QObject::tr("名字"));  
  35. //  
  1. //tata()改寫某一個(gè)列的顯示樣式,對(duì)齊方式  
  2. QVariant MySqlQueryModel::data(const QModelIndex &index, int role) const  
  3. {  
  4.     QVariant value = QSqlQueryModel::data(index, role);  
  5.     if (role == Qt::TextColorRole && index.column() == 0)  
  6.         return qVariantFromValue(QColor(Qt::red)); //第一個(gè)屬性的字體顏色為紅色  
  7.   
  8.     if (role == Qt::TextAlignmentRole && index.column() == 1)  
  9.     {  
  10.         value = (Qt::AlignVCenter + Qt::AlignRight); // 靠右垂直居中  
  11.     }  
  12.       
  13.     return value;  

posted @ 2012-04-23 11:21 米米 閱讀(551) | 評(píng)論 (0)編輯 收藏

 #include <QtSql>
QT += sql
QSqlDatabase類實(shí)現(xiàn)了數(shù)據(jù)庫連接的操作
QSqlQuery類執(zhí)行SQL語句
QSqlRecord類封裝數(shù)據(jù)庫所有記錄

QSqlDatabase類

  1. QSqlDatabase db = QSqlDatabase::addDatabase("QOCI");  
  2. db.setHostName("localhost");    //數(shù)據(jù)庫主機(jī)名   
  3. db.setDatabaseName("scott");    //數(shù)據(jù)庫名   
  4. db.setUserName("stott");        //數(shù)據(jù)庫用戶名   
  5. db.setPassword("tiger");        //數(shù)據(jù)庫密碼   
  6. db.open();          //打開數(shù)據(jù)庫連接  
  7. db.close();         //釋放數(shù)據(jù)庫連接  

建立數(shù)據(jù)庫文件

  1. QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");  
  2. db.setDatabaseName("database.db");  
  3. if (!db.open())   
  4. {  
  5.     qDebug("數(shù)據(jù)庫不能打開");  
  6. }  
  7. return false;  
  1. 建立數(shù)據(jù)庫文件后創(chuàng)建表并插入兩條數(shù)據(jù)  
  1. QSqlQuery query;  
  2. query.exec("create table student(id INTEGER PRIMARY KEY autoincrement,  
  3.             name nvarchar(20), age int)"); //id自動(dòng)增加  
  4. query.exec("insert into student values(1,'小明', 14)");  
  5. query.exec("insert into student values(2,'小王',15)");  


QSqlQuery類

插入值到數(shù)據(jù)庫操作

一、直接用SQL語句插入(參照上面)

二、利用預(yù)處理方式插入(ORACLE語法和ODBC語法)

適合插入多條記錄,或者避免將值轉(zhuǎn)換成字符串(即正確地轉(zhuǎn)義),調(diào)用prepare()函數(shù)指定一個(gè)包含占位符的query,然后綁定要插入的值

ORACLE語法

  1. QSqlQuery query;   
  2. query.prepare("INSERT INTO T_STUDENT (name, age) VALUES (:name, :age)"); //準(zhǔn)備執(zhí)行SQL查詢  
  3. query.bindValue(":name", "小王");   //在綁定要插入的值  
  4. query.bindValue(":age", 11);   
  5. query.exec();  


ODBC語法

  1. QSqlQuery query;  
  2. query.prepare("INSERT INTO T_STUDENT (name,age) VALUES (?,?)"); //準(zhǔn)備執(zhí)行SQL查詢  
  3. query.addBindValue("小王");   //在綁定要插入的值  
  4. query.bindValue(11);  
  5. query.exec();  

三、批量插入到數(shù)據(jù)庫中

  1. QSqlQuery query;  
  2. query.prepare(“insert into student values (?, ?)”);  
  3. QVariantList names;  
  4. names << "小王" << "小明" << "小張" << "小新"; // 如果要提交空串,用QVariant(QVariant::String)代替名字  
  5. query.addBindValue(names);  
  6. QVariantList ages;  
  7. ages << 11 << 13 << 12 << 11;  
  8. query.addBindValue(ages);  
  9. if (!q.execBatch()) //進(jìn)行批處理,如果出錯(cuò)就輸出錯(cuò)誤  
  10.     qDebug() << q.lastError();  


查詢數(shù)據(jù)庫操作

  1. QSqlQuery query;  
  2. query.exec("SELECT * FROM t_STUDENT"); // 查詢的結(jié)果可能不止一條記錄,所以我們稱之為結(jié)果集  
  3. while (query.next())  
  4. {  
  5.     QString name = query.value(0).toString(); //取第i條記錄第1個(gè)字段(從0開始計(jì)數(shù))的結(jié)果  
  6.     int age = query.value(0).toInt(); //取第i條記錄第2個(gè)字段的結(jié)果  
  7.     // ... 處理name,age變量數(shù)據(jù)   
  8. }  


seek(int n) :query指向結(jié)果集的第n條記錄。指定當(dāng)前的位置

first() :query指向結(jié)果集的第一條記錄。

last() :query指向結(jié)果集的最后一條記錄。

next() :query指向下一條記錄,每執(zhí)行一次該函數(shù),便指向相鄰的下一條記錄。

previous() :query指向上一條記錄,每執(zhí)行一次該函數(shù),便指向相鄰的上一條記錄。

record() :獲得現(xiàn)在指向的記錄。

value(int n) :獲得屬性的值。其中n表示你查詢的第n個(gè)屬性

int rowNum = query.at(); //獲取query所指向的記錄在結(jié)果集中的編號(hào)

int fieldNo = query.record().indexOf(“name”); //返回"name"的列號(hào)

int columnNum = query.record().count(); //獲取每條記錄中屬性(即列)的個(gè)數(shù)

 

事務(wù)操作

操作函數(shù):transaction(),commit()提交,rollback()回滾
操作事務(wù)前,先判斷該數(shù)據(jù)庫是否支持事務(wù)操作。hasFeature是QSQLDriver類函數(shù)

  1. if (QSqlDatabase::database().driver()->hasFeature(QSqlDriver::Transactions)){ ... } //   

 插入一條記錄,然后提交事務(wù)

  1. QSqlDatabase::database().transaction();  
  2. QSqlQuery query;  
  3. query.exec("SELECT id FROM T_STUDENT WHERE class=1");  
  4. if (query.next())  
  5. {  
  6.     query.exec("INSERT INTO T_STUDENT (id,name,age) VALUES (3,'小李',13)");  
  7. }  
  8. QSqlDatabase::database().commit();  
posted @ 2012-04-23 11:16 米米 閱讀(630) | 評(píng)論 (0)編輯 收藏

設(shè)計(jì)軟件中各個(gè)部件的位置排列,有兩種方法:

1.設(shè)置widget的在父窗體中的坐標(biāo)和大小

widget->move(x,y);

widget->resize(width,height);

//widget->setGeometry(QRect(x,y,width,height));

什么時(shí)候用:不想被布局約束,對(duì)這種控件可控制時(shí)

例如下面的可活動(dòng)按鈕

缺點(diǎn):在遇到很多個(gè)部件時(shí),一個(gè)個(gè)算坐標(biāo)會(huì)很麻煩

部件不自動(dòng)隨父窗體的變化而變化(這里要重寫resizeEvent事件算變化比例建立父子窗體之間的聯(lián)系,而布局很容易做到)

 

2.布局管理Layout(可控制的部件不要使用布局中,一旦改變會(huì)影響到整體布局)

經(jīng)常要用到的三個(gè)布局部件

QHBoxLayout水平布局

QVBoxLayout垂直布局

QGridLayout表格布局

 

————————————————————————————————

QHBoxLayout和QVBoxLayout用法相同

基本:

vlayout->addWidget(w1);

vlayout->addWidget(w2);

……

——————————————————————————————————

QGridLayout用法

layout->addWidget(widget,row,column,rowSpan,columnSpan);

layout->addLayout(layout,row,column,rowSpan,columnSpan);

(row,column):控件占據(jù)左上角的單元格位置

rowSpan:控件占據(jù)的行數(shù)

columnSpan:控件占據(jù)的列數(shù)(不寫默認(rèn)為1)

——————————————————————————————

控件隨窗體伸縮

將布局與主窗體建立聯(lián)系

layout=new QHBoxLayout(this);

或者this->setLayout(layout);

——————————————————————————————

layout->addStretch();可伸縮的空間

layout->setAlignment(Qt::AlignHCenter);//布局內(nèi)控件水平居中 

layout->setSpacing(10);//控件間隔為10

layout->setMargin(10);//頁邊距為10

Qt設(shè)計(jì)器中spacer控件

以上在對(duì)控件位置調(diào)整時(shí)常用

 

——————————————————————————————————

結(jié)合SizePolicy屬性控制布局管理中的控件的尺寸自適應(yīng)方式

也就是控件隨主窗體伸縮方式(可自行增大或縮小)

QSizePolicy sizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);//可自行增大和收縮

Fixed:不能放大或縮小

Minimum:不能小于sizeHint尺寸,可放大(若設(shè)置了最大尺寸)

Maximum:不能放大,可縮小到允許的最小尺寸(若setMinimumSize(10,10)只能縮小到(10,10))

Preferred:控件的sizeHint是他的Hint,可以放大或縮小

 

————————————————————————————————————————————

比例因子的使用

例:布局中有左右兩控件,右邊比左邊水平伸縮2倍

sizePolicy.setHorizontalStretch(1);
sizePolicy.setVerticalStretch(0);

left->setSizePolicy(sizePolicy);

sizePolicy.setHorizontalStretch(2);
sizePolicy.setVerticalStretch(0);

left->setSizePolicy(sizePolicy);

 

水平伸展比例(1,2)

垂直伸展比例(0,0)

——————————————————————————————————————————————————

布局顧名思義是對(duì)控件的位置的整體規(guī)劃,每個(gè)控件之間都存在著位置關(guān)系,所以改變控件(移動(dòng)、關(guān)閉)會(huì)破壞布局

——————————————————————————————————————————————————

結(jié)論:兩種方法結(jié)合使用,應(yīng)該可以應(yīng)對(duì)大部分界面布局

本人是菜鳥,實(shí)踐不多,如有錯(cuò)誤歡迎指正

—————————————————————————————————————————————————

上述代碼是為了理解布局使用,Qt設(shè)計(jì)器可實(shí)現(xiàn)

posted @ 2012-04-09 17:42 米米 閱讀(925) | 評(píng)論 (0)編輯 收藏

看到一些人經(jīng)常會(huì)問QTreeWidget的列寬怎么設(shè)啊,怎么固定列寬哦,行寬怎么設(shè)啊什么的.其實(shí)慢慢看manual就會(huì)找到了,下面就列下常用的方法:


1.基本用法:

初始化用基本用到的:

 //this->setMouseTracking(true);
this->setRootIsDecorated(false);
this->setSelectionMode(QAbstractItemView::ExtendedSelection);
this->setStyleSheet("QTreeWidget::item{height:25px}");  //設(shè)置行寬,     我也找不到提供的接口  setStyleSheet很強(qiáng)大建議看

this->setColumnWidth(0, 100);  //設(shè)置列寬
this->setColumnCount(2);
QStringList lists;
lists << "NO" << "name";
this->setHeaderLabels(lists);


for (int i=0;i<10;i++)
{

QStringList contentList;
contentList << QString("00%1").arg(i) << "boy";
  QTreeWidgetItem *pNewItem = new QTreeWidgetItem(this, contentList);  //增加
pNewItem->setBackgroundColor(0, QColor(150,0,0));
pNewItem->setFont(0, QFont());
}

//pNewItemRoot->setExpanded(true);  //展開子項(xiàng)


增加刪除頂層用:

addTopLevelItem(QTreeWidgetItem*);

takeTopLeveltem(int);

topLevelItem(int); //返回

topLevelItemCount();

增加刪除子層:

addChild(QTreeWidgetItem*);

addChildren(const QList<QTreeWidgetItem*>&);

takeChild(int);

takeChildren();

child(int)  //返回

childCount();

2.進(jìn)價(jià):

//加checkbox, icon

pNewItem->setFlags(Qt::ItemIsSelectable | Qt::ItemIsUserCheckable |Qt::ItemIsEnabled);
pNewItem->setCheckState(0, Qt::Unchecked);

//pNewItem->setIcon(int , QIcon);


//向ITEM里放一些QT的控件,如QSpinBox,QComboBox,QPushButton等

this->setItemWidget(pNewItem, 0, new QSpinBox());


//獲取某項(xiàng)的QRect, 有時(shí)候會(huì)很有用,如顯縮略圖:

QRect rect = this->visualItemRect(QTreeWidgetItem*);


//右鍵菜單重寫

contextMenuEvent( QContextMenuEvent * event );或者使用信號(hào)itemPressed(QTreeWidgetItem*, int);


//對(duì)頭的操作,可以用QLabel或者QTreeWidgetItem對(duì)頭進(jìn)行初始化,

//隱藏頭

header()->hide();

 m_pHeaderView->setClickable(true);  //能夠發(fā)射sectionClicked(int)信號(hào),可做菜單,顯示隱藏列:setSectionHidden(int, bool);isSectionHidden(int);
m_pHeaderView->setMovable(true);
 m_pHeaderView->setResizeMode(0, QHeaderView::Fixed);  //固定列寬
 m_pHeaderView->setDefaultSectionSize(100);  //默認(rèn)


//排序

treeWidget->setSortingEnabled(true);

treeWidget->header()->setSortIndicatorShown(true);
treeWidget->header()->setSortIndicator(0, Qt::AscendingOrder);

//要自定義就用信號(hào)

connect( treeWidget->header(), SIGNAL( sectionClicked(int) ), this, SLOT( sectionClickedSlot(int) ) );

在小項(xiàng)目里,使用qtreewidget完全足夠,

如果想更多靈活用法請(qǐng)使用QTreeView等 MODEL/VIEW控件

posted @ 2012-03-31 10:49 米米 閱讀(2591) | 評(píng)論 (0)編輯 收藏

今天簡單說一下Qt的樹形控件,在Qt中樹形控件的名稱叫做QTreeWidget,而控件里的樹節(jié)點(diǎn)的名稱叫做QTreeWidgetItem。今天這 里講的是如何創(chuàng)建具有復(fù)選框的樹形控件,這種控件其實(shí)有時(shí)挺有用的,如飛信群發(fā)短信時(shí)的選擇聯(lián)系人的界面中就使用了有復(fù)選框的樹形控件。先來看一下長什么 樣的:

當(dāng)選中頂層的樹形節(jié)點(diǎn)時(shí),子節(jié)點(diǎn)全部被選中,當(dāng)取消頂層樹形節(jié)點(diǎn)時(shí),子節(jié)點(diǎn)全部被取消選中狀態(tài),而當(dāng)選中子節(jié)點(diǎn)時(shí),父節(jié)點(diǎn)顯示部分選中的狀態(tài)。

      要實(shí)現(xiàn)這種界面其實(shí)很簡單的。在Qt的設(shè)計(jì)器中,拖出一個(gè)QTreeWidget,然后在主窗口中寫一個(gè)函數(shù)init初始化界面,連接樹形控件的節(jié)點(diǎn)改變 信號(hào)itemChanged(QTreeWidgetItem* item, int column),實(shí)現(xiàn)這個(gè)信號(hào)的即可。下面是具體代碼:

      ui.setupUi(this);

       //初始化
        init();

     //連接信號(hào)和槽

      connect(ui.treeWidget, SIGNAL(itemChanged(QTreeWidgetItem*, int)), this, SLOT(treeItemChanged(QTreeWidgetItem*, int)));

void TreeWidgetTest::init()
{
    ui.treeWidget->clear();

   //第一個(gè)分組
   QTreeWidgetItem *group1 = new QTreeWidgetItem(ui.treeWidget);
   group1->setText(0, "group1");
   group1->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
   group1->setCheckState(0, Qt::Unchecked);

   QTreeWidgetItem *subItem11 = new QTreeWidgetItem(group1);
   subItem11->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
   subItem11->setText(0, "subItem11");
   subItem11->setCheckState(0, Qt::Unchecked);

   QTreeWidgetItem *subItem12 = new QTreeWidgetItem(group1);
   subItem12->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
   subItem12->setText(0, "subItem12");
   subItem12->setCheckState(0, Qt::Unchecked);

  QTreeWidgetItem *subItem13 = new QTreeWidgetItem(group1);
  subItem13->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  subItem13->setText(0, "subItem13");
  subItem13->setCheckState(0, Qt::Unchecked);

  QTreeWidgetItem *subItem14 = new QTreeWidgetItem(group1);
  subItem14->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  subItem14->setText(0, "subItem14");
  subItem14->setCheckState(0, Qt::Unchecked);

//第二個(gè)分組
  QTreeWidgetItem *group2 = new QTreeWidgetItem(ui.treeWidget);
  group2->setText(0, "group2");
  group2->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  group2->setCheckState(0, Qt::Unchecked);

  QTreeWidgetItem *subItem21 = new QTreeWidgetItem(group2);
  subItem21->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  subItem21->setText(0, "subItem21");
  subItem21->setCheckState(0, Qt::Unchecked);

  QTreeWidgetItem *subItem22 = new QTreeWidgetItem(group2);
  subItem22->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  subItem22->setText(0, "subItem22");
  subItem22->setCheckState(0, Qt::Unchecked);

  QTreeWidgetItem *subItem23 = new QTreeWidgetItem(group2);
  subItem23->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable);
  subItem23->setText(0, "subItem23");
  subItem23->setCheckState(0, Qt::Unchecked);

}

void TreeWidgetTest::treeItemChanged(QTreeWidgetItem* item, int column)
{
  QString itemText = item->text(0);
  //選中時(shí)
if (Qt::Checked == item->checkState(0))
{
   QTreeWidgetItem* parent = item->parent();
   int count = item->childCount();
   if (count > 0)
   {
    for (int i = 0; i < count; i++)
    {
     //子節(jié)點(diǎn)也選中
     item->child(i)->setCheckState(0, Qt::Checked);
    }   
   }
   else
   {
    //是子節(jié)點(diǎn)
    updateParentItem(item);
   }
}
else if (Qt::Unchecked == item->checkState(0))
{
   int count = item->childCount();
   if (count > 0)
   {
    for (int i = 0; i < count; i++)
    {
     item->child(i)->setCheckState(0, Qt::Unchecked);
    }
   }
   else
   {
    updateParentItem(item);
   }
}

}

void TreeWidgetTest::updateParentItem(QTreeWidgetItem* item)
{
QTreeWidgetItem *parent = item->parent();
if (parent == NULL)
{
   return;
}

//選中的子節(jié)點(diǎn)個(gè)數(shù)
int selectedCount = 0;
int childCount = parent->childCount();
for (int i = 0; i < childCount; i++)
{
   QTreeWidgetItem *childItem = parent->child(i);
   if (childItem->checkState(0) == Qt::Checked)
   {
    selectedCount++;
   }
}

if (selectedCount <= 0)
{
   //選中狀態(tài)
   parent->setCheckState(0, Qt::Unchecked);
}
else if (selectedCount > 0 && selectedCount < childCount)
{
   //部分選中狀態(tài)
   parent->setCheckState(0, Qt::PartiallyChecked);
}
else if (selectedCount == childCount)
{
  //未選中狀態(tài)
   parent->setCheckState(0, Qt::Checked);
}

//changeFromUser = true;

}

      上面需要注意的是,當(dāng)用代碼setCheckState后,會(huì)觸發(fā)itemChanged事件,有時(shí)這個(gè)會(huì)帶來麻煩,這時(shí)需要加一些標(biāo)志來判斷是由用戶操作產(chǎn)生的還是程序本身代碼產(chǎn)生的。

posted @ 2012-03-31 10:49 米米 閱讀(3276) | 評(píng)論 (0)編輯 收藏

因?yàn)?nbsp;Q_DECLARE_METATYPE 是一個(gè)宏。借用一句話:“宏是萬惡之源”。什么惡呢?就是,如果它的參數(shù)當(dāng)中有逗號(hào)",",會(huì)被當(dāng)成參數(shù)分隔符。


所以,當(dāng)你試圖這么寫:

  1. Q_DECLARE_METATYPE(QMap<QString, bool>);  

你就會(huì)發(fā)現(xiàn)編譯不過,并且抓狂了。


解決方法嘛,當(dāng)然很簡單:參照Q_DECLARE_METATYPE的宏定義,自己把它展開,編譯器就不會(huì)再犯傻了。


  1. QT_BEGIN_NAMESPACE  
  2. template <>  
  3. struct QMetaTypeId< QMap<QString, bool> >  
  4. {  
  5.     enum { Defined = 1 };  
  6.     static int qt_metatype_id()  
  7.     {  
  8.         static QBasicAtomicInt metatype_id = Q_BASIC_ATOMIC_INITIALIZER(0);  
  9.         if (!metatype_id)  
  10.             metatype_id = qRegisterMetaType< QMap<QString, bool> >("QMap<QString, bool>",  
  11.                 reinterpret_cast< QMap<QString, bool> *>(quintptr(-1)));  
  12.         return metatype_id;  
  13.     }  
  14. };  
  15. QT_END_NAMESPACE 
posted @ 2012-03-30 13:13 米米 閱讀(1444) | 評(píng)論 (0)編輯 收藏

例:新建一個(gè)項(xiàng)目,用QT Designer設(shè)計(jì)界面,然后在VS中編譯。

 

過程:

1. 用QT Designer進(jìn)行設(shè)計(jì),另存成ui文件(如Demo.ui)。

2. 在project目錄下運(yùn)行“uic Demo.ui -o ui_Demo.h”生成ui_Demo.h文件。

3. 繼續(xù)編寫代碼,自定義對(duì)話框類繼承自QDialog和ui_Demo.h中定義的類。

4. 保存所有文件,在project目錄下運(yùn)行“qmake -project”,生成pro文件(如DemoProject.pro)。

5. 在project目錄下運(yùn)行“qmake -tp vc DemoProject.pro”生成新的vcproj文件。

6. 回到VS,此時(shí)VS會(huì)提示重新加載project,同意,然后編譯運(yùn)行。

 

注意:

1. 上述第2步中,推薦對(duì)uic生成的目標(biāo)頭文件,就用默認(rèn)的“ui_[原文件名].h”格式。

2. 第2步理論上可以不要,在第4、5步,qmake會(huì)自動(dòng)調(diào)用uic再做一次,第2步的意義主要是提前把h文件生成,這樣可以利用intelligence對(duì)其的解析輔助代碼編寫。

3. 第3步當(dāng)中,自定義對(duì)話框類的父類中,QDialog一定要是第一個(gè),否則會(huì)編譯出錯(cuò)。

posted @ 2012-03-30 13:11 米米 閱讀(600) | 評(píng)論 (0)編輯 收藏

要實(shí)現(xiàn)對(duì)話框不在任務(wù)欄顯示,而且無規(guī)則邊框(用半透明png在paintevent里畫),需要把屬性設(shè)為:

 

Qt::Popup | Qt::Dialog | Qt::FramelessWindowHint

 

不過發(fā)現(xiàn)這樣對(duì)話框show()的時(shí)候不激活。

 

解決方式是在showEvent里調(diào)用this->activateWindow()

 

特此備忘。

posted @ 2012-03-30 12:20 米米 閱讀(914) | 評(píng)論 (0)編輯 收藏

僅列出標(biāo)題
共8頁: 1 2 3 4 5 6 7 8 
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久亚洲综合色| 久久久久久久91| 欧美日韩综合一区| 99国产一区| 午夜综合激情| 一区二区在线不卡| 欧美高清视频在线播放| 91久久久久久久久久久久久| 亚洲韩国日本中文字幕| 欧美日韩国产小视频在线观看| 99在线|亚洲一区二区| 欧美一站二站| 亚洲欧洲日韩综合二区| 欧美精品一区在线| 亚洲影视九九影院在线观看| 久久综合伊人77777| 日韩视频不卡| 国产日韩亚洲欧美精品| 免费在线观看日韩欧美| 99在线|亚洲一区二区| 久久婷婷激情| 制服丝袜激情欧洲亚洲| 国产日韩欧美一二三区| 欧美大片网址| 亚洲欧美在线高清| 亚洲电影在线播放| 亚洲第一精品夜夜躁人人爽 | 欧美黄色成人网| 一本一本久久| 狠狠色狠狠色综合日日小说 | 欧美体内谢she精2性欧美| 亚洲男同1069视频| 亚洲国产精品尤物yw在线观看 | 欧美午夜一区二区| 久久久蜜臀国产一区二区| 亚洲乱码国产乱码精品精98午夜| 久久精品一本久久99精品| 亚洲精品系列| 尤物九九久久国产精品的特点| 欧美日韩中文| 欧美成人精品| 久久久精品久久久久| 亚洲一区bb| 日韩视频免费观看高清完整版| 久久夜色精品国产亚洲aⅴ| 亚洲一区网站| 99re6热只有精品免费观看| 好吊妞这里只有精品| 国产欧美短视频| 欧美三级网址| 欧美人妖在线观看| 久久在线观看视频| 欧美自拍偷拍午夜视频| 亚洲影院色在线观看免费| 亚洲欧洲午夜| 最新日韩在线| 亚洲高清av在线| 裸体歌舞表演一区二区 | 久久这里有精品15一区二区三区| 亚洲综合日韩中文字幕v在线| 亚洲精品国产视频| 亚洲电影在线播放| 激情六月婷婷综合| 国产欧美日韩精品在线| 国产精品久久久久999| 欧美性色综合| 国产精品护士白丝一区av| 欧美日韩天堂| 国产精品久久77777| 欧美视频中文字幕| 国产精品白丝av嫩草影院| 欧美午夜电影网| 国产精品欧美在线| 国产欧美日韩综合精品二区| 国产精品永久| 国产在线精品成人一区二区三区| 国产日韩亚洲欧美精品| 国产日韩欧美中文在线播放| 国产日韩精品一区| 国语自产精品视频在线看抢先版结局 | 亚洲国产视频一区| 亚洲国产精品欧美一二99| 亚洲国产成人91精品| 亚洲福利在线视频| 亚洲国产另类久久精品| 日韩天天综合| 亚洲一区二区欧美| 欧美中文在线视频| 久久婷婷麻豆| 亚洲福利视频网| 日韩午夜激情av| 亚洲一区亚洲| 久久女同精品一区二区| 免费成人高清在线视频| 欧美日韩视频免费播放| 国产精品捆绑调教| 国产综合香蕉五月婷在线| 在线观看91精品国产入口| 亚洲黄色在线观看| 一区二区免费在线视频| 欧美在线免费一级片| 免费成人av| 亚洲免费精彩视频| 欧美在线高清视频| 欧美激情精品久久久久| 国产精品―色哟哟| 一区三区视频| 宅男噜噜噜66一区二区| 午夜宅男欧美| 欧美高清你懂得| 亚洲已满18点击进入久久| 久久亚洲一区二区| 欧美午夜激情小视频| 国产综合网站| 亚洲视频第一页| 久久色在线观看| 日韩视频永久免费| 久久久国产一区二区| 欧美日韩网站| 亚洲国产三级| 久久精品视频在线看| 欧美激情久久久久久| 亚洲一区二区三区精品动漫| 免费看的黄色欧美网站| 国产精品一区二区久久久| 亚洲国产小视频| 久久狠狠一本精品综合网| 亚洲精品在线一区二区| 久久爱另类一区二区小说| 欧美日韩一级视频| 亚洲第一在线| 久久精品国产免费| 一本久久综合亚洲鲁鲁| 久久夜色精品一区| 国产一区久久| 欧美亚洲在线| 一区二区三区国产精品| 欧美高清自拍一区| 1000部精品久久久久久久久 | 久久综合九色99| 亚洲一级特黄| 欧美色网在线| 一区二区三区欧美激情| 欧美激情国产日韩| 玖玖玖国产精品| 一区二区在线不卡| 久久午夜视频| 欧美一区二区三区四区高清| 国产精品久久精品日日| 一区二区三区视频免费在线观看| 亚洲国产另类久久精品| 久久综合一区二区三区| 国产亚洲一区二区三区在线观看| 亚洲香蕉伊综合在人在线视看| 91久久久国产精品| 欧美激情第8页| 亚洲美女黄色片| 亚洲精品欧美日韩| 欧美精品系列| 一本到高清视频免费精品| 亚洲激情在线视频| 欧美精品亚洲| 中文亚洲字幕| 亚洲视频综合在线| 国产精品久久网站| 小黄鸭精品aⅴ导航网站入口| 亚洲专区在线| 国产日韩欧美精品综合| 久久精品视频免费| 久久久美女艺术照精彩视频福利播放| 激情一区二区三区| 免费观看国产成人| 欧美成人性网| 亚洲视频一区在线| 亚洲天堂偷拍| 韩国欧美一区| 亚洲国产福利在线| 欧美连裤袜在线视频| 亚洲素人一区二区| 亚洲欧美日韩在线高清直播| 国产一区999| 欧美成人精品不卡视频在线观看 | 欧美国产日韩在线| 亚洲视频精选| 午夜精品久久久久久久蜜桃app| 国产在线精品成人一区二区三区| 美女视频网站黄色亚洲| 蜜桃av噜噜一区| 亚洲专区一二三| 久久久久久国产精品mv| 亚洲久久视频| 亚洲欧美韩国| 亚洲激情视频在线观看| 日韩亚洲欧美高清| 国产亚洲欧美一区二区| 欧美大片在线看| 国产精品久久久久aaaa樱花| 久久久久久噜噜噜久久久精品 | 亚洲大胆在线| 日韩一区二区电影网|