Ver Mensaje Individual
  #2 (permalink)  
Antiguo 08/05/2015, 04:09
eferion
 
Fecha de Ingreso: octubre-2014
Ubicación: Madrid
Mensajes: 1.212
Antigüedad: 10 años, 3 meses
Puntos: 204
Respuesta: QMenu, abrir multiples ventanas

Pues mira, yo te propongo una solución algo diferente.

Pasos a seguir:

1. Crea una función miembro en la ventana principal tal que:

Código C++:
Ver original
  1. class MainWindow : public QMainWindow
  2. {
  3.   private:
  4.     template< typename Class >
  5.     QDialog* NewWindow( )
  6.     {
  7.       return new Class( this );
  8.     }
  9. };

Este template nos va a permitir crear cualquier ventana que herede de QDialog. Ahora veremos cómo usarla.

2. Creamos un mapa que nos permita asociar los QAction. En este caso, cada QAction irá asociado a la llamada correcta a la función que hemos creado en el paso anterior. Esto lo conseguimos así:

Código C++:
Ver original
  1. #include <functional>
  2. #include <map>
  3.  
  4. class MainWindow
  5. {
  6.   private:
  7.     std::map< QAction*, std::function< QDialog*( MainWindow* ) > > _newWindows;
  8. };

Usamos std::function por legibilidad más que nada. El mapa anterior tiene como clave un QAction. El segundo parámetro es un puntero a una función miembro de MainWindow, con el siguiente prototipo: "QDialog* MainWindow::X( )".

3. Rellenamos el mapa. Esto lo suyo sería hacerlo en una función que se llame desde el constructor, para mantener la limpieza del código:

Código C++:
Ver original
  1. QAction* newAction = new QAction( "Test", this );
  2. connect( newAction, SIGNAL( triggered( ) ), this, SLOT( MenuTriggered( ) ) );
  3.  
  4. QMenu* menu = menuBar( )->addMenu( "File" );
  5. menu->addAction( newAction );
  6.  
  7. _newWindows[ newAction ] = &MainWindow::NewWindow< TestDialog >;

En el ejemplo estamos creando primero el QAction, por defecto, todos los QAction llamarán al slot MenuTriggered( ), ahí es donde terminaremos de obrar la magia. A continuación añadimos el QAction al menu. Finalmente añadimos una entrada en el mapa creado anteriormente. En este caso "TestDialog" es el nombre de un diálogo que he creado para la ocasión. Lo que estamos haciendo en esta última línea es devolver el puntero a una versión especializada de "NewWindow" que nos devolverá un objeto de tipo "TestDialog".

4. Para terminar, el código del slot. Este código después se puede ampliar para hacer comprobaciones o para añadir otro tipo de opciones:

Código C++:
Ver original
  1. void MainWindow::MenuTriggered( )
  2. {
  3.   auto it = _newWindows.find( dynamic_cast< QAction* >( sender( ) ) );
  4.   if( it != _newWindows.end( ) )
  5.   {
  6.     QDialog* dlg = it->second( this );
  7.     dlg->exec( );
  8.   }
  9. }

La función "sender" está presente en todos los QObject y almacena el objeto que ha generado la última señal... en nuestro caso el objeto en cuestión será el QAction sobre el que hayamos clickado. Una vez tenemos el objeto en cuestión, buscamos en el mapa para conocer cual es la función a la que debemos llamar... la llamamos y lanzamos la ventana que corresponda. Magia!!! jejejeje

Espero que te sirva esta solución.

Un saludo