lunes, 25 de julio de 2011
Añadir elementos a una QToolBar
Además si queremos que los QActions luzcan con icono, debemos crearnos un fichero de recursos qrc, allí iremos importando los iconos necesarios y posteriormente se los asignamos a los QActions.
Acabo de crear los botones de la barra de herramientas de la aplicación y de asignarle sus iconos correspondientes. He obtenido los iconos de freeiconsdownload ya que tienen una licencia que permite usarlos en proyectos no comercial.
Dejo una captura de pantalla del estado actual de la aplicación. En principio está desarrollada como MDI.
El código de esta versión se puede descargar desde aquí:
svn checkout http://infobasic.googlecode.com/svn/tags/infobasic-0.0.4 infobasic-read-only
Servidor simple de echo UDP
echoserver.h
#ifndef ECHOSERVER_H #define ECHOSERVER_H #include "IForm.h" #include#include namespace Ui { class EchoServer; } class EchoServer : public QWidget { Q_OBJECT public: explicit EchoServer(QWidget *parent = 0); virtual ~EchoServer(); static const quint16 PORT_MIN; static const quint16 PORT_MAX; static const quint16 PORT_DEFAULT; protected: QUdpSocket *udpServer; virtual void putTexts(); virtual void setuid(); static const QString START; static const QString STOP; public slots: void onClickStart(); void readPendingDatagrams(); private: Ui::EchoServer *ui; bool started; }; #endif // ECHOSERVER_H
echoserver.cpp
#include "echoserver.h" #include "ui_echoserver.h" #includeconst quint16 EchoServer::PORT_MIN = 1; const quint16 EchoServer::PORT_MAX = 65535; const quint16 EchoServer::PORT_DEFAULT = 8888; const QString EchoServer::START = tr("Start"); const QString EchoServer::STOP = tr("Stop"); EchoServer::EchoServer(QWidget *parent) : QWidget(parent), ui(new Ui::EchoServer) { ui->setupUi(this); this->udpServer = new QUdpSocket(this); this->started = false; putTexts(); setuid(); } EchoServer::~EchoServer() { delete ui; delete this->udpServer; qDebug("Delete EchoServer"); } void EchoServer::putTexts() { this->setWindowTitle(tr("Servidor echo UDP")); ui->lbPort->setText(tr("Port")); ui->pbStart->setText(tr("Start")); } void EchoServer::setuid() { setAttribute(Qt::WA_DeleteOnClose); connect(ui->pbStart, SIGNAL(clicked()),SLOT(onClickStart())); connect(ui->sldPort, SIGNAL(valueChanged(int)), ui->lcdPort, SLOT(display(int))); ui->sldPort->setMinimum(EchoServer::PORT_MIN); ui->sldPort->setMaximum(EchoServer::PORT_MAX); ui->sldPort->setValue(EchoServer::PORT_DEFAULT); } void EchoServer::onClickStart() { if(!this->started) { qDebug("Server started"); quint16 port = ui->lcdPort->value(); udpServer->bind(port, QUdpSocket::ShareAddress); connect(udpServer, SIGNAL(readyRead()), this, SLOT(readPendingDatagrams())); ui->pbStart->setText(STOP); } else { qDebug("Server stoped"); udpServer->close(); ui->pbStart->setText(START); } this->started = !this->started; } void EchoServer::readPendingDatagrams() { qDebug("readPendingDatagrams"); while (udpServer->hasPendingDatagrams()) { QByteArray datagram; datagram.resize(udpServer->pendingDatagramSize()); QHostAddress sender; quint16 senderPort; udpServer->readDatagram(datagram.data(), datagram.size(), &sender, &senderPort); QString str(datagram); str.append(" from " + sender.toString() + " on " + QDateTime::currentDateTime().toString()+ "\n"); ui->textBrowser->insertPlainText(str); udpServer->writeDatagram(datagram, sender, senderPort); } }
Para iniciar el servidor, simplemente hay que indicar el puerto por el que escuchará y el modo. Además hay que conectar la llegada de paquetes a una función, en este caso la función es readPendingDatagrams() y se ejecutará cada vez que llegue un paquete UDP.
quint16 port = ui->lcdPort->value(); udpServer->bind(port, QUdpSocket::ShareAddress); connect(udpServer, SIGNAL(readyRead()), this, SLOT(readPendingDatagrams()));
En la función readPendingDatagrams() es donde se realiza el procesamiento de datos y se monta la respuesta para el cliente. En este caso el procesamiento de datos consiste en sacar los datos recibidos por pantalla y la respuesta contiene exactamente los mismos datos que se han recibido.
while (udpServer->hasPendingDatagrams()) { QByteArray datagram; datagram.resize(udpServer->pendingDatagramSize()); QHostAddress sender; quint16 senderPort; udpServer->readDatagram(datagram.data(), datagram.size(),&sender, &senderPort); QString str(datagram); str.append(" from " + sender.toString() + " on " + QDateTime::currentDateTime().toString()+ "\n"); ui->textBrowser->insertPlainText(str); udpServer->writeDatagram(datagram, sender, senderPort); }
Como siempre podeis descargar el código desde el svn, para esta versión en concreto desde aquí:
svn checkout http://infobasic.googlecode.com/svn/tags/infobasic-0.0.3 infobasic-read-only
QByteArray to QString
QByteArray datagram; ... //Suponemos que el QByteArray tiene datos. QString str(datagram);
domingo, 24 de julio de 2011
Cliente simple echo. UDP. Añadida selección de puerto.
Conectamos el valor producido por el QSlider al valor mostrado en el QLCDNumber y configuramos los valore máximos y mínimos del QSlider a los valores de puerto máximo y mínimo respectivamente.
const quint16 EchoClient::PORT_MIN = 1; const quint16 EchoClient::PORT_MAX = 65535; const quint16 EchoClient::PORT_DEFAULT = 8888; ... void EchoClient::setuid() { setAttribute(Qt::WA_DeleteOnClose); connect(ui->pbSend, SIGNAL(clicked()),SLOT(onClickSend())); connect(ui->sldPort, SIGNAL(valueChanged(int)), ui->lcdPort, SLOT(display(int))); ui->sldPort->setMinimum(EchoClient::PORT_MIN); ui->sldPort->setMaximum(EchoClient::PORT_MAX); ui->sldPort->setValue(EchoClient::PORT_DEFAULT); }
Por último en la función de enviar cogemos el valor del puerto del QLCDNumber y lo enviamos al host especificado en el QLineEdit.
void EchoClient::onClickSend() { qDebug("onClickSend"); QHostInfo info = QHostInfo::fromName(ui->leServerName->text()); if (!info.addresses().isEmpty()) { QHostAddress address = info.addresses().first(); QByteArray ba = ui->leText->text().toLocal8Bit(); const char *txt = ba.data(); qint16 len = ui->leText->text().length(); qint16 port = ui->sldPort->value(); QUdpSocket *udp =new QUdpSocket(this); udp->writeDatagram(txt, len, address, port); delete udp; } else { QMessageBox msgBox; msgBox.setWindowTitle(tr("Error")); msgBox.setText(tr("Error host not found")); msgBox.setIcon(QMessageBox::Critical); msgBox.setStandardButtons(QMessageBox::Ok); msgBox.exec(); } } Para descargar esta versión:svn checkout http://infobasic.googlecode.com/svn/tags/infobasic-0.0.2 infobasic-read-only
sábado, 23 de julio de 2011
Nuevas tareas en google code.
Voy a ir definiendo las próximas acciones de desarrollo en google code. No se muy bien en que va a acabar la aplicación pero la idea es que poco a poco se vayan definiendo nuevas tareas y objetivos para seguir practicando con las herramientas que nos proveen las librerias qt. Las nuevas tareas se irán definiendo en la siguiente página. http://code.google.com/p/infobasic/issues/list
Analisis de datos enviados por el cliente udp.
Al capturar los datos enviado por nuestro cliente de echo UDP obteniamos la siguiente secuencia de bytes:
0x0000: 4500 0028 0000 4000 4011 3cc3 7f00 0001 E..(..@.@.<..... 0x0010: 7f00 0001 df38 22b8 0014 fe27 6865 6c6c .....8"....'hell 0x0020: 6f20 7365 7276 6572 o.server
Teniendo en cuenta que el encabezado de un paquete UDP es el siguiente:
(16 bits) | (16 bits) |
(16 bits) | (16 bits) |
(longitud variable). |
Tenemos que:
Puerto origen: df38 (2 bytes)
Puerto destino: 22b8 (2 bytes. 0x22b8 es 8888 en decimal)
Longitud total: 0014 (2 bytes. La longitud total es de 20 bytes, es decir 0x14 en hexadecimal)
Checksum: fe27 (2 bytes)
Datos enviados: 6865 6c6c 6f20 7365 7276 6572 (12 bytes)
Los datos anteriores corresponde al protocolo IP en el que va montada la trama UDP.
0x0000: 4500 0028 0000 4000 4011 3cc3 7f00 0001 E..(..@.@.<..... 0x0010: 7f00 0001
Y su significado es el siguiente:
Versión: 45 (4 bits 1/2 byte, 0100 significa que se está usando la versión IPV4)Longitud del encabezado: 00 (4 bits, 1/2 byte, 0101 es decir 5 en decimal)
Tipo de servicio: 00 (1 bytes)
Longitud total: 0028 (2 bytes, corresponde a una longitude de 40 bytes)
Identificación: 0000 (2 bytes)
Fragment offset: 4000 (2 bytes)
Tiempo de vida: 40 (1 byte)
Protocolo: 11 (1 byte, corresponde al protocolo UDP 17 en decimal)
Checksum de la cabecera: 3cc3 (2 bytes)
Dirección origen: 7f00 0001 (4 bytes, 127.0.0.1)
Dirección destino: 7f00 0001 (4 bytes, 127.0.0.1)
El formato de un paquete IP es el siguiente:
(4 bits) | (4 bits) | (8 bits) | (16 bits) | |
(16 bits) | (3 bits) | (13 bits) | ||
(8 bits) | (8 bits) | (16 bits) | ||
En los posteriores post intentaré analizar cuales son las diferencias al utilizar diferentes protocolos como TCP y diferentes puertos para el cliente de echo.
Infobasic svn
Acabo de subir el código del programa ejemplo infobasic al un repositorio en google code. La dirección para descargar la rama principal es la siguiente:
svn checkout http://infobasic.googlecode.com/svn/trunk/ infobasic-read-only
Para descargar la versión que tiene como último elemento añadido la el diálogo de about:
svn checkout http://infobasic.googlecode.com/svn/tags/infobasic-0.0.1 infobasic-read-only
miércoles, 20 de julio de 2011
Convertir un label en un hipervinculo.
aboutdialog.h
#ifndef ABOUTDIALOG_H #define ABOUTDIALOG_H #include "IForm.h" #includenamespace Ui { class AboutDialog; } class AboutDialog : public QDialog, public IForm { Q_OBJECT public: static const char* HREF; static const char* HREF2; explicit AboutDialog(QWidget *parent = 0); ~AboutDialog(); virtual void putTexts(); virtual void setuid(); public slots: void onClickLink(); private: Ui::AboutDialog *ui; }; #endif // ABOUTDIALOG_H
aboutdialog.cpp
#include "aboutdialog.h" #include "ui_aboutdialog.h" #include#include const char* AboutDialog::HREF ="Orgía de objetos"; const char* AboutDialog::HREF2 ="Thank you for visiting us."; AboutDialog::AboutDialog(QWidget *parent) : QDialog(parent), ui(new Ui::AboutDialog) { ui->setupUi(this); putTexts(); setuid(); } AboutDialog::~AboutDialog() { delete ui; } void AboutDialog::onClickLink() { qDebug("onClickLink"); QString str = ui->labelLink->text(); int startIndex = str.indexOf(QString("'")); int endIndex = str.indexOf(QString("'"), ++startIndex); qDebug(str.mid(startIndex, endIndex).toLocal8Bit().data()); QDesktopServices::openUrl(QUrl(str.mid(startIndex, endIndex - startIndex))); ui->labelLink->setText(HREF2); } void AboutDialog::putTexts() { ui->labelLink->setText(HREF); ui->labelLink->adjustSize(); } void AboutDialog::setuid() { connect(ui->labelLink, SIGNAL(linkActivated(QString)), this, SLOT(onClickLink())); }
Este código abre la dirección del QLabel en el explorador por defecto.
En los próximos días voy a intentar sacar tiempo para subir el código fuente.
lunes, 18 de julio de 2011
Cliente simple echo. UDP.
echoclient.h
#ifndef ECHOCLIENT_H #define ECHOCLIENT_H #include < QWidget> #include "IForm.h" namespace Ui { class EchoClient; } class EchoClient : public QWidget, public IForm { Q_OBJECT public: explicit EchoClient(QWidget *parent = 0); virtual ~EchoClient(); virtual void putTexts(); virtual void setuid(); public slots: void onClickSend(); private: Ui::EchoClient *ui; }; #endif // ECHOCLIENT_H
echoclient.cpp
#include "echoclient.h" #include "ui_echoclient.h" #include "qtudpclient.h" #include < QUdpSocket> #include < QHostAddress> #include < iostream> EchoClient::EchoClient(QWidget *parent) : QWidget(parent), ui(new Ui::EchoClient) { ui->setupUi(this); putTexts(); setuid(); } EchoClient::~EchoClient() { delete ui; qDebug("Delete EchoClient"); } void EchoClient::putTexts() { ui->lbServerName->setText(tr("Server Name")); ui->lbText->setText(tr("Text")); ui->leServerName->setText("localhost"); ui->pbSend->setText(tr("Send")); } void EchoClient::setuid() { setAttribute(Qt::WA_DeleteOnClose); connect(ui->pbSend, SIGNAL(clicked()),SLOT(onClickSend())); } void EchoClient::onClickSend() { qDebug("onClickSend"); QUdpSocket *udp =new QUdpSocket(this); const QHostAddress *host = new QHostAddress(ui->leServerName->text()); //const char *txt = (const char*) ui->leText->text().toLocal8Bit().data(); QByteArray ba = ui->leText->text().toLocal8Bit(); const char *txt = ba.data(); qint16 len = ui->leText->text().length(); qint16 port = 8888; udp->writeDatagram(txt, len, QHostAddress::LocalHost, port); qDebug(ui->leServerName->text().toStdString().c_str()); delete host; delete udp; }Y aquí podemos ver la información enviada al servidor:
sudo tcpdump -xX -i lo 0x0000: 4500 0028 0000 4000 4011 3cc3 7f00 0001 E..(..@.@.<..... 0x0010: 7f00 0001 c30e 22b8 0014 fe27 6865 6c6c ......"....'hell 0x0020: 6f20 7365 7276 6572 o.serverEn los próximos envíos veremos la interpretación de esta información y apliaremos el cliente para que sea capaz de recibir e interpretar las respuestas del servidor.
Patrones de diseño.
Actualmente estoy leyendo un sobre patrones de diseño, es mi intención comentar e implementar los ejemplos propuestos. Aunque el libro está orientado a programación en java intentaré desarrollar todos los ejercicios usando c++ con las librerías qt y bajo una ubuntu 11.04.
Además pienso ir publicando mis pequeños programas que realizarán acciones básicas con las qt.
Hello blogger!
My first post from Blogilo!