Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   Varios (https://www.clubdelphi.com/foros/forumdisplay.php?f=11)
-   -   bring to top de la aplicacion (https://www.clubdelphi.com/foros/showthread.php?t=37568)

Ricardo Alfredo 16-11-2006 18:08:11

bring to top de la aplicacion
 
Hola he busca sin éxito por el momento de como hacer que mi aplicación no se ejecute dos veces , si no que cuando ya se esta ejecutando "lanzarla" hacia a delante.

Lo que sucede que tengo varias aplicaciones y se pueden llamar entre ellas , pero sucede que si la invocas mas de una vez , te la abrirá y lo que quiero es que te la muestre para que sigas trabajando en lo que estabas.

Por ejemplo si uds. abren varias veces el Excel, este lo que hace es ejecutar varias veces las aplicación yo quiero lo contrario que solo se ejecute una vez hasta que le des cerrar.


gracias por la ayuda

seoane 16-11-2006 20:43:29

Permiteme que te recomiende esta excelente unit del compañero roman

http://romansg.net/index.php?pg=uiapp

Ricardo Alfredo 16-11-2006 21:18:10

gracias , la prove y funciona en forma estupenda.

Ahora otra preguntilla, es como puedo terminar un grupo de aplicaciones , estas por ejemplo yo tengo mi aplicacion principal que me llama a todas las demas , la idea es si el usuario se sale de esta yo termine con todas las que tenga abierta y esten relacionadas con la principal.



gracias

seoane 16-11-2006 22:04:32

Bueno, en la unit de roman tienes un buen ejemplo de como usar la funciones RegisterWindowMessage y SendMessage para enviar mensajes de tipo BroadCast. En tu caso, solo tendrías que enviar un mensaje a todas las ventanas (usando HWND_BROADCAST), de esta manera cuando una de las otras aplicaciones reciba el mensaje sabrán que tienen que cerrarse y actuaran en consecuencia.

Si echándole un vistazo al código de roman, no te aclaras con el funcionamiento de RegisterWindowmessage y SendMessage, pasate otra vez por aquí e intentaremos echarte una mano.

Ricardo Alfredo 17-11-2006 15:57:42

sabes , tuve un pequeño inconveniente con la unit de roman, sucede que yo tengo un sistema el cual contiene 6 aplicaciones, pues bien, debo cambiar los parametros sMutex y sActivar para generar la compilacion de la aplicacion, esto lo debo hacer tantas veces como apliaciones tenga, pero no habra alguna otra forma , por ejemplo dejar esas constantes en el dpr de cada aplicacion, lo trate de hacer de varias formas :eek: y no he logrado solucionar ese pequeño inconveniente, si tienen alguna idea me la pueden decir por favor.

seoane 17-11-2006 16:30:09

:confused: No entiendo el problema. Es verdad que para cada aplicación tienes que utilizar un valor de sMutex y sActivar diferente porque precisamente se trata de diferencia unas aplicaciones de otras. Pero en el caso que mencionabas después, cerrar todas tus aplicaciones a la vez, debes de utilizar la misma cadena en todas de esta forma la orden de cerrar llegara a todas tus aplicaciones.

Lo dicho, puede que no este entendiendo cual es problema.

Ricardo Alfredo 17-11-2006 16:46:52

si es cierto lo que dices, pero lo que ahora esta preguntando era de que modo puede cambiar los valors mutez y activar sin tener que crear una copia distinta para cada aplicacion , porque digo esto porque sabemos que esas dos constantes deben ir con valor distinto para tantas aplicaciones tenga, entoces la unica manera que he encontrado para es tener una copia de la unit de roman por tantas apliacaciones tenga, esto debido que no he encontrado la forma de pasar esas constantes a la unit de roman con valores distintos desde el dpr

lo otro que hablas tu estoy haciendo las pruebas y cambios necesario para ver si funciona, cuando lo logre te cuento

seoane 17-11-2006 17:08:09

Se me ocurre una solución, modificamos la unit de roman de esta manera:
Código Delphi [-]
unit UIApp;

interface

uses
  Windows, SysUtils, Forms;

procedure Registrar(sActivar, sMutex: string);
procedure Activar;

implementation

var
  mActivar    : Cardinal; { Mensaje para activar la instancia anterior }
  Mutex       : Cardinal; { Mutex                                      }
  PrevWndProc : TFarProc; { Procedimiento de ventana original          }


function AppWndProc(Handle: HWND; Msg: UINT; wParam: WPARAM; lParam: LPARAM): LongInt; stdcall;
var
  FgThreadId  : DWORD; { Hilo de la app. que tenga el foco }
  AppThreadId : DWORD; { Hilo de nuestra aplicación        }

begin
  if Msg = mActivar then
  begin
    { Si está minimizada basta restaurarla }
    if IsIconic(Handle) then
      ShowWindow(Handle, SW_RESTORE)
    else
    begin
      { Obtener los hilos }
      FgThreadId  := GetWindowThreadProcessId(GetForegroundWindow, nil);
      AppThreadId := GetWindowThreadProcessId(Handle, nil);

      { Anexar el hilo de nuestra app. al de la  que tenga el foco }
      AttachThreadInput(AppThreadId, FgThreadId, true);

      { Ahora sí, activar la applicación }
      SetForegroundWindow(Handle);

      { Separar el hilo de nuestra app de la otra }
      AttachThreadInput(AppThreadId, FgThreadId, false);
    end;

    Result := 0;
  end
  else
    { Dejar que el procedimiento original se encargue de los otros mensajes }
    Result := CallWindowProc(PrevWndProc, Handle, Msg, wParam, lParam);
end;

procedure Activar;
begin
  { Mandamos el mensaje a todas las ventanas }
  SendMessage(HWND_BROADCAST, mActivar, 0, 0);
end;

procedure Registrar(sActivar, sMutex: string);
begin
  mActivar := RegisterWindowMessage(sActivar);
  Mutex    := CreateMutex(nil, true, sMutex);

  { Si ya existe el mutex lanzamos una excepción silenciosa }
  if GetLastError = ERROR_ALREADY_EXISTS then
  begin
   Mutex := 0;
   abort;
  end
  else
  begin
    { Sustituimos el procedimiento de ventana }
    PrevWndProc := TFarProc(GetWindowLong(Application.Handle, GWL_WNDPROC));
    SetWindowLong(Application.Handle, GWL_WNDPROC, LongWord(@AppWndProc));
  end;
end;

initialization 

finalization
  if Mutex <> 0 then ReleaseMutex(Mutex);
end.

Luego en el dpr de nuestra aplicación, colocamos esto al principio:
Código Delphi [-]
 try
    Registrar(sActivar, sMutex); // SActivar, sMutex son constantes del dpr
  except
    Activar;
    Halt;
  end;

:confused: ¿que te parece?

roman 17-11-2006 17:20:33

Está bien la modificación. Yo estaba pensando algo por la vía de

Código Delphi [-]
var
  {$include contantes.inc}
  // mActivar    : Cardinal; { Mensaje para activar la instancia anterior }
  // Mutex       : Cardinal; { Mutex                                      }
  PrevWndProc : TFarProc; { Procedimiento de ventana original          }

pero constantes.inc tendría que ser específico de la aplicación y no sé como inidicarle al $include una ruta que dependa de la aplicación. ¿En Delphi no hay 'magic constants'?

// Saludos

seoane 17-11-2006 17:29:35

:D Roman, tanto hablar de ti ya se me hacia raro que no intervinieras.

Ricardo Alfredo 17-11-2006 17:32:03

basicamente cambie lo que dice seoane, con un pequeño cambio que fue
procedure Registrar_APP(sActivar, sMutex: PAnsiChar);
ya que si lo dejo como string este es incompatible con cardinal.

y funciona correctamente por lo menos lo probe con 6 aplicaciones, ahora terminara con el resto con son 23 y vero como hacer el otro tema de si me cierran el principal cierre todos los "hijos"

roman 17-11-2006 17:58:26

Cita:

Empezado por seoane
Luego en el dpr de nuestra aplicación, colocamos esto al principio:

Todo esto está muy bien, pero se pierde la idea original de no tener que codificar nada, sólo incluir un archivo. Dándole vueltas un poco creo que he encontrado una solución.

Donde dice:

Código Delphi [-]
const
  { Cadenas para registrar el mutex y el mensaje }
  sMutex   = '10D73234-C9F7-4C2D-BC7E-39B5820AF456';
  sActivar = '3F154732-CCDE-4BC7-9439-AFCD3BCFA84D';

var
  mActivar    : Cardinal; { Mensaje para activar la instancia anterior }
  Mutex       : Cardinal; { Mutex                                      }
  PrevWndProc : TFarProc; { Procedimiento de ventana original          }

hay que cambiar a:

Código Delphi [-]
var
  { Cadenas para registrar el mutex y el mensaje }
  sMutex: String = '10D73234-C9F7-4C2D-BC7E-39B5820AF456';
  sActivar: String = '3F154732-CCDE-4BC7-9439-AFCD3BCFA84D';

  mActivar    : Cardinal; { Mensaje para activar la instancia anterior }
  Mutex       : Cardinal; { Mutex                                      }
  PrevWndProc : TFarProc; { Procedimiento de ventana original          }

es decir, promovemos las constantes a variables inicializadas.

Luego agregamos un procedimiento:

Código Delphi [-]
procedure LeerConstantes(var sActivar, sMutex: String);
var
  Archivo: TIniFile;
  sArchivo: String;

begin
  sArchivo := ExtractFilePath(Application.ExeName) + 'uiapp.ini';

  if FileExists(sArchivo) then
  begin
    Archivo := TIniFile.Create(sArchivo);

    try
      sActivar := Archivo.ReadString('constantes', 'activar', sActivar);
      sMutex := Archivo.ReadString('constantes', 'mutex', sMutex);
    finally
      Archivo.Free;
    end;
  end;
end;

Este procedimiento busca el archivo uiapp.ini en el directorio de la aplicación y sí existe, cambia los valores que se le pasan por los que tenga ese archivo.

Finalmente se introduce la llamada a LeerConstantes al principio:

Código Delphi [-]
initialization
  LeerConstantes(sActivar, sMutex);
  
  try
    Registrar;
  except
    Activar;
    Halt;
  end;

Así, el programador final sólo tiene que agregar el uiapp.pas en el uses del dpr e incluir un ini en su directorio:

Código:

[constantes]
activar=0E69F5F8-5AB5-4A31-B2DF-6A2467DEAC63
mutex=24B85122-DEEB-4939-8C34-5DF11B89D0B9

// Saludos

seoane 17-11-2006 18:03:52

Pues a mi me se ocurre otra.

Dejar la unit tal cual esta en tu web. Y solo cambair estas dos lineas
Código Delphi [-]
  mActivar := RegisterWindowMessage(sActivar);
  Mutex    := CreateMutex(nil, true, sMutex);

Cambiarlas por:
Código Delphi [-]
  mActivar := RegisterWindowMessage(PChar(sActivar+ParamStr(0)));
  Mutex    := CreateMutex(nil, true, Pchar(sMutex+ParamStr(0)));

roman 17-11-2006 18:09:37

Sí, esta está mejor :)

// Saludos

roman 17-11-2006 22:21:04

Cita:

Empezado por seoane
Dejar la unit tal cual esta en tu web

Bueno sí, por el momento la unidad se queda tal cual en la web, pero eso no quita que podamos mencionar el cambio. :)

// Saludos

roman 30-11-2006 22:38:06

Hola,

Gracias a dec que me avisó de un pequeño problema con la modificación propuesta por seoane. Extrañamente no funciona y como no le encontraba ninguna lógica, lo mejor era irse a la documentación de CreateMutex:

Cita:

Código:

HANDLE CreateMutex(
  LPSECURITY_ATTRIBUTES lpMutexAttributes,
  BOOL bInitialOwner,
  LPCTSTR lpName
);

[...]

lpName

Points to a null-terminated string specifying the name of the mutex object. The name is limited to MAX_PATH characters and can contain any character except the backslash path-separator character (\).
y claro, ParamStr(0) contiene de esos caracteres. Al parecer se soluciona fácilmente poniendo algo como:

Código Delphi [-]
StringReplace(ParamStr(0), '\', '', [rfReplaceAll]);

// Saludos

seoane 30-11-2006 22:46:42

Cita:

Empezado por roman
Hola,
Gracias a dec que me avisó de un pequeño problema con la modificación propuesta por seoane. Extrañamente no funciona y como no le encontraba ninguna lógica ...

:cool: Si, es MUY extraño que no funcione un código mio. :p Es broma, la verdad es que cuando lo dije se me ocurrió de pronto y no me paré a probarlo. De todas formas yo sustituiría '\' por '/', o por cualquier otro carácter que no pueda formar parte del nombre de un fichero para asegurarnos de que sigue siendo único.

roman 30-11-2006 22:52:50

Je, je, tienes razón, al colapsar puede perderse la unicidad:

ab\cd.exe => abcd.exe
a\bcd.exe => abcd.exe

Entonces habrá que sustituirlo por un caracter que no pueda formar parte de un nombre de archivo, como ? ó *.

// Saludos

German 25-02-2008 06:02:39

Por "actualizar" un poco el hilo, hago un pequeño comentario, después de utilizar la unidad UIApp en D2007, y bajo Vista 64...

La función IsIconic(Handle) siempre devuelve False, y ShowWindow(Handle, SW_RESTORE) no restaura la aplicación. Debería utilizarse:
Código Delphi [-]
if IsIconic(Application.MainForm.Handle) then ShowWindow( Application.MainForm.Handle, SW_RESTORE)

Además, en el proc. "Registrar", la excepción silenciosa utilizada cuando ya existe el mutex, siempre es lanzada (o sea que se convierte en escandalosa), apareciendo el mensaje de diálogo, con lo que el try de initialization deja de tener sentido. Se podría solucionar de este manera...

Código Delphi [-]
(...)

procedure Registrar;
begin
  mActivar := RegisterWindowMessage(sActivar);
  Mutex    := CreateMutex(nil, true, sMutex);

  { Si ya existe , ponemos el mutex a cero }
  if GetLastError = ERROR_ALREADY_EXISTS then Mutex := 0
  else begin
    { Si no existe, sustituimos el procedimiento de ventana }
    PrevWndProc := TFarProc(GetWindowLong(Application.Handle, GWL_WNDPROC));
    SetWindowLong(Application.Handle, GWL_WNDPROC, LongWord(@AppWndProc));
  end;
end;

initialization
  Registrar;
  if Mutex = 0 then
  begin
    Activar;
    Halt;
  end;

(...)

El ejecutable resultante funciona tanto en XP como en Vista de forma correcta.

Salu2.


La franja horaria es GMT +2. Ahora son las 04:30:04.

Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi