Club Delphi  
    FTP   CCD     Buscar   Trucos   Trabajo   Foros

Retroceder   Foros Club Delphi > Otros entornos y lenguajes > C++ Builder
Registrarse FAQ Miembros Calendario Guía de estilo Temas de Hoy

Respuesta
 
Herramientas Buscar en Tema Desplegado
  #1  
Antiguo 01-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Question identificador de un proceso

Hola a todos, estoy de nuevo por aquí y me gustaría preguntarles si alguno sabe como cambiar el identificador de un proceso.

Mi problema concretamente es que tengo 2 aplicaciones hechas por mi (digamos apli1 y apli2), el apli1 tiene una subclase llamada TForm2 la cual por medio de ::SetParent le doy como padre apli2.

Bien hasta ahí todo va bien el problema es que cuando cierro apli1 como era de esperarse se cierra también TForm2, me imagino que es porque TForm2 nunca camia su PID. Así que me gustaría saber como puedo mantener abierto TForm2 y cerrar apli1 (o al menos saber si es posible hacer algo así).

Estoy en Builder 2007, pero como siempre cualquier tipo de orientación es bien recibida, de antemano gracias.
Responder Con Cita
  #2  
Antiguo 01-12-2008
Avatar de escafandra
[escafandra] escafandra is offline
Miembro Premium
 
Registrado: nov 2007
Posts: 2.197
Poder: 20
escafandra Tiene un aura espectacularescafandra Tiene un aura espectacular
Pues en OnClose del Form1 haz ::SetParent(Apli1, 0);
Así la liberas antes de terminar Apli1

Saludos.
Responder Con Cita
  #3  
Antiguo 01-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Pues después de leer tu respuesta escafandra, lo intente y no logre hacerlo sigue cerrando TForm2 cuando cierro apli1.

Si tienen otras sugerencias por favor díganme.
Responder Con Cita
  #4  
Antiguo 01-12-2008
Avatar de escafandra
[escafandra] escafandra is offline
Miembro Premium
 
Registrado: nov 2007
Posts: 2.197
Poder: 20
escafandra Tiene un aura espectacularescafandra Tiene un aura espectacular
Pues a mi me funciona perfectamente, no se como es tu código. Debes guardar el Handle de la ventana principal de tu Form2 en una variable global o miembro de Form1.

Fíjate en este código. Simplemente creo un Form que va a contener a "Mi PC", por comodidad de manejar un sólo programa:
Código:
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm1 *Form1;
HANDLE M_Handle;

//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
}
//---------------------------------------------------------------------------

void __fastcall TForm1::FormCreate(TObject *Sender)
{
    M_Handle = ::FindWindow(0, "Mi PC");
    ::SetParent(M_Handle, Handle);
}
//---------------------------------------------------------------------------

void __fastcall TForm1::FormClose(TObject *Sender, TCloseAction &Action)
{
     ::SetParent(M_Handle, 0);
}
//---------------------------------------------------------------------------
Saludos.
Responder Con Cita
  #5  
Antiguo 01-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
La cuestión es que lo que emparento en apli2 es una subclase de apli1, dejo algo de código:

Aquí lo que hago es decirle a ff2 de aplicacion1 que su padre será ff1 de aplicacion1.

Aplicacion1
Código:
//---------------------------------------------------------------------------
void __fastcall Tff1::sButton1Click(TObject *Sender)
{
/**/
  Application->CreateForm(__classid(Tff2), &ff2);
  ff2->Show();
  ::SetParent(ff2->Handle, Handle);
}
//---------------------------------------------------------------------------
Y ahora desde una aplicación ajena jalo a ff2

Aplicacion1
Código:
//---------------------------------------------------------------------------
void __fastcall Tff1::sButton1Click(TObject *Sender)
{
/**/
  HWND BWnd = FindWindow("Tff1",0);
  HWND TWnd = FindWindowEx(BWnd, 0, "Tff2", 0);
  ::SetParent(TWnd, Handle);
}
//---------------------------------------------------------------------------
Y si cierro ff1 de aplicacion1 se cerrara ff2 que ahora está en aplicacion2, ya que ff2 nunca deja de pertenecer a ff1.

Así que lo que yo quiero saber es si hay alguna forma de que ff2 se quede abierto y ff1 cerrado (gracias).
Responder Con Cita
  #6  
Antiguo 01-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Wink

mmm creo que enrede un poco las cosas así que pondré este ejemplillo.

Código:
HWND BARRA = FindWindow("Shell_TrayWnd", 0);
HWND TRAYY = FindWindowEx(BWnd, 0, "TrayNotifyWnd", 0);
::SetParent(TRAYY, Handle);
Con este código pongo el systray en mi formulario, pero si cierro explorer.exe se cerrara también el systray.

Entonces la pregunta es puedo dejar el systray en mi formulario aun cerrando explorer.exe (esto es lo que intento hacer con mis dos aplicaciones donde ff1=explorer.exe y ff2=systry).
Responder Con Cita
  #7  
Antiguo 01-12-2008
Avatar de escafandra
[escafandra] escafandra is offline
Miembro Premium
 
Registrado: nov 2007
Posts: 2.197
Poder: 20
escafandra Tiene un aura espectacularescafandra Tiene un aura espectacular
Claro, ya entendí lo que pretendes, justo lo contrario de lo que en un principio creía. No puedes dejas abierto parte de un proceso mientras otra parte de cierra, a no ser que esa parte sea creada desde tu proceso o... que realices una inyección en tu propio código de esa parte de otro proceso, previamente colocado en tu espacio de memoria. Cosa que me parece muy complicada.

No se si alguien en el foro puede darte una respuesta mejor.

Saludos.
Responder Con Cita
  #8  
Antiguo 01-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Unhappy

mmm pues creo que tienes toda la razón escafandra, sería sumamente complicado hacer lo que pretendo, creo que tendré que buscar algo alternativo como ocultar y paralizar el proceso, solo que tenía una leve esperanza de que no me partiera la cabeza con esto.

Aunque creo que cambiando el id del proceso puedo mantener la clase abierta ya que no abría nada que le vinculara con la aplicación original.

Si alguien sabe cambiar id de procesos me dicen por favor.
Responder Con Cita
  #9  
Antiguo 02-12-2008
Avatar de cHackAll
[cHackAll] cHackAll is offline
Baneado?
 
Registrado: oct 2006
Posts: 2.159
Poder: 20
cHackAll Va por buen camino
Cita:
Empezado por _cero_ Ver Mensaje
...Aunque creo que cambiando el id del proceso puedo mantener la clase abierta ya que no abría nada que le vinculara con la aplicación original...


Cita:
Empezado por _cero_ Ver Mensaje
...Si alguien sabe cambiar id de procesos me dicen por favor.
Considero totalmente inutil tu intento de cambiar el PID de un proceso, el premio adjunto de dicha asaña muy probablemente sea un BSOD

Creo que es claro que deseas usar un formulario (o parte) de un programa ajeno en el tuyo; para ello se inventaron los objetos OLE-COM, y si la aplicacion de la cual pretendes posecionar "algo" no tiene dicha funcionalidad, lo más probable es que consigas "regaños" del autor

Aunque le falta madurar, te dejo una idea de la linea "no finalizar aplicacion" (aunque te aconsejo buscar otras), pero esta mas relacionada a tu caso. Para concluirla te aconseo analizar la API TerminateProcess y sus efectos.

Código Delphi [-]
var si: TStartupInfo = (cb: SizeOf(si));
procedure TForm1.Button1Click(Sender: TObject);
var
 pi: TProcessInformation;
 de: TDebugEvent;
begin
 if CreateProcess(nil, 'c:\windows\notepad.exe', nil, nil, False, DEBUG_ONLY_THIS_PROCESS, nil, nil, si, pi) then
  while WaitForDebugEvent(de, INFINITE) do
   if (de.dwDebugEventCode <> EXIT_PROCESS_DEBUG_EVENT) then
    ContinueDebugEvent(de.dwProcessId, de.dwThreadId, DBG_CONTINUE);
end;
Código:
void __fastcall TForm1::Button1Click(TObject *Sender)
{
 STARTUPINFOA si = {sizeof si};
 PROCESS_INFORMATION pi;
 DEBUG_EVENT de;
 if (CreateProcess(NULL, "c:/windows/notepad.exe", NULL, NULL, false, DEBUG_ONLY_THIS_PROCESS, NULL, NULL, &si, &pi))
  while (WaitForDebugEvent(&de, INFINITE))
   if (de.dwDebugEventCode != EXIT_PROCESS_DEBUG_EVENT)
    ContinueDebugEvent(de.dwProcessId, de.dwThreadId, DBG_CONTINUE);
}
PD; talvez si nos presentas el escenario real y completo podamos darte mejores alternativas.

Saludos
__________________
RTFM > STFW > Foro > Truco > Post > cHackAll > KeBugCheckEx
Responder Con Cita
  #10  
Antiguo 02-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Question Que are

Pues si [cHackAll] todo en tu post anterior es cierto.

Mi proyecto real es un remplazo de Shell para win xp y vista (con unas cosas mejores), hasta ahora ya llevo echo el manejador de archivos, el conmutador de tareas(alt+tab), un menú inicio y como el 80% de la nueva barra de tareas, la cuestión es que no he podido averiguar cuales son los procesos que corren en el systray(aunque lo supiera sería difícil obtener sus popup y otras funciones que se obtienen con los click encima de los iconos), así que pensé en solo quedarme con el systray y cerrar el explorer.exe para no consumir recursos de mas pero esto se me esta haciendo mas difícil que todo lo demás así que ustedes que harían?.

(Gracias por el código lo estoy analizando).
Responder Con Cita
  #11  
Antiguo 07-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Unhappy

Pues solo les escribo para comentarles que por fin logre cambiar el pid de una aplicación, y por simple curiosidad lo implemente junto con mi aplicación, y si lo único que me gane fue un lindo BSOD tal y como dijo [cHackAll], pero bueno al menos aprendí algo muy útil (programación de módulos del kernel, algo medio complicado por eso no les dejo código).

Así que opte por esconder la barra, el menú inicio, y todo lo referente al Explorer dentro de un form invisible, el problema ahora es que cada vez que se presiona la tecla de inicio se mueve el foco de la aplicación activa hacia el menú inicio (aunque esta invisible).

Como podría evitar esto?, ya intente hookear el teclado, matar el proceso del menú inicio, volver a ocultar la ventana cuando se presione la tecla, regresar el foco con SetForegroundWindow y no queda (juro que esto me va a llevar a la tumba y todo por un systray).
Responder Con Cita
  #12  
Antiguo 07-12-2008
_cero_ _cero_ is offline
Miembro
 
Registrado: abr 2007
Posts: 147
Poder: 18
_cero_ Va por buen camino
Talking

Acabo de solucionar lo del menú inicio, simulando que se vuelve a presionar la tecla inicio.

Código:
if (IsWindowVisible(menu)) {
  keybd_event(91,0,0,0);
  keybd_event(91,0,KEYEVENTF_KEYUP,0);
}
Pero si de casualidad alguien sabe o tiene código que me ayude a crear mi propio systray se lo agradecería mucho.
Responder Con Cita
Respuesta



Normas de Publicación
no Puedes crear nuevos temas
no Puedes responder a temas
no Puedes adjuntar archivos
no Puedes editar tus mensajes

El código vB está habilitado
Las caritas están habilitado
Código [IMG] está habilitado
Código HTML está deshabilitado
Saltar a Foro

Temas Similares
Tema Autor Foro Respuestas Último mensaje
Identificador de llamadas... Sinaloense Varios 4 07-07-2007 21:31:51
Identificador de llamadas majaco Varios 3 14-02-2007 11:58:08
Identificador de llamadas en .NET edalmasso Varios 1 18-05-2006 17:02:35
Identificador no declarado Enan0 Varios 2 16-02-2005 18:58:51
Identificador de llamadas.. dmariscal Varios 2 10-11-2003 21:39:45


La franja horaria es GMT +2. Ahora son las 14:20:09.


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
Copyright 1996-2007 Club Delphi