vieuxpere
Messages postés195Date d'inscriptionsamedi 3 janvier 2004StatutMembreDernière intervention17 décembre 2014
-
3 mai 2009 à 22:29
vieuxpere
Messages postés195Date d'inscriptionsamedi 3 janvier 2004StatutMembreDernière intervention17 décembre 2014
-
7 mai 2009 à 15:31
Bonjour,
Mon application, qui se lance au démarrage de Windows, reste en arriere plan en permanence.
Jusqu'à maintenant mon application bloquée la fermeture de Windows quans l'utilisateur demandé l'arret.
Depuis le code ci-dessous, mon appli se ferme mais l'arret de windows s'interrompt, il faut donc le refaire arreter Windows pour que ca s'arrete enfin.
Je doit donc fermer en deux fois !
Si quelqu'un sait m'améliorer le code ci-dessous je lui en serait bien reconnaissant.
Cordialement,
// Detecte la fermeture de Windows (demandée par l'utilisateur)
procedure TForm4.WMQueryEndSession(var Message: TWMQueryEndSession);
begin
Message.Result := 1;
end;
// Execute des operations pendant la fermeture de Windows
procedure TForm4.WM_ENDSESSION(var Message: TWMQueryEndSession);
Begin
Application.Terminate; // On ferme l'application
End;
Bacterius
Messages postés3792Date d'inscriptionsamedi 22 décembre 2007StatutMembreDernière intervention 3 juin 201610 5 mai 2009 à 21:46
Ben un flag de façon générale c'est un indicateur en fait.
Par exemple, tu déclares en variable globale :
var
MyFlag: Integer=0;
Quand cette valeur est à 0, tout va bien.
Quand elle est à 1, il faut arrêter toutes les opérations en cours pour fermer l'application.
Au moment de la réception du WM_QUERYENDSESSION tu vas définir ce flag à 1, et toutes tes opérations stopperont, et Windows pourra fermer ton application et s'éteindre normalement.
Dans tes boucles de travail tu fais :
for I := 0 to ... do
begin
...
if MyFlag = 1 then Break;
end;
while ... do
begin
...
if MyFlag = 1 then Break;
end;
C'est une façon efficace de stopper toute opération rapidement.
ni69
Messages postés1418Date d'inscriptionsamedi 12 juin 2004StatutMembreDernière intervention 5 juillet 201012 3 mai 2009 à 23:02
Bonsoir,
Tu as oublié l'instruction inherited
Voilà le code corrigé (remarque: il est inutile d'utiliser les deux messages Windows WMQueryEndSession et WMEndSession, un seul suffit pour réaliser ce que tu recherches) :
vieuxpere
Messages postés195Date d'inscriptionsamedi 3 janvier 2004StatutMembreDernière intervention17 décembre 2014 4 mai 2009 à 14:38
Merci de ta réponse, je vais tester mais j'ai quand même une question comment "Msg.Result := 1;"va s'executer si avant je fais un application.terminate ?
vieuxpere
Messages postés195Date d'inscriptionsamedi 3 janvier 2004StatutMembreDernière intervention17 décembre 2014 4 mai 2009 à 14:40
A j'oubliez aussi, est-ce que ce code doit-être mis sur toutes les fenetres potentiellement ouvertes au moment de la fermeture de Windows ou juste sur la fenetre principale ?
Vous n’avez pas trouvé la réponse que vous recherchez ?
ni69
Messages postés1418Date d'inscriptionsamedi 12 juin 2004StatutMembreDernière intervention 5 juillet 201012 4 mai 2009 à 20:41
@vieuxpere:
Désolé, une petite inversion de lignes... J'ai pas trop réfléchi à vrai dire il était tard ! Il faut bien sûr lire :
inherited;
Msg.Result := 1;
Application.Terminate;
Mais en pratique, il est vrai que
Msg.Result := 1; est facultatif, vu qu'il est géré par l'appel à inherited; Cette procédure mise sur la form principale est amplement suffisante. Cela marche très bien pour moi je viens de tester, donc regarde peut-être au niveau des autres applications qui tournent en même temps (cf taskmgr). Peut-être que le problème ne vient pas de la tienne!
@Bacterius: Tout dépend de si l'application est dans une phase bloquante ou non (procédure longue en cours, boucle qui n'est pas encore finie...)
Dans ce genre de cas, l'utilisation d'un flag associé à la boucle en cours peut être utile (on modifie le flag dans la procédure liée au WMQES, et cela est répercuté dans la boucle...)
Bacterius
Messages postés3792Date d'inscriptionsamedi 22 décembre 2007StatutMembreDernière intervention 3 juin 201610 4 mai 2009 à 20:59
En général il faut tout faire pour respecter les choix de l'utilisateur.
Si l'utilisateur veut fermer, tu ne dois pas être en boucle de travail.
M'enfin des fois evidemment il faut attendre quelques minutes par exemple pendant une sauvegarde de base de données de dernière minute, mais dans ce cas-là la moindre des choses est d'afficher une messagebox avec marqué "patientez svp" en gros. Ou bien sur Vista, appeller ShutdownReasonBlockCreate.
Mais effectivement ça peut être utile d'utiliser un flag pour savoir de quoi on parle ...
ni69
Messages postés1418Date d'inscriptionsamedi 12 juin 2004StatutMembreDernière intervention 5 juillet 201012 4 mai 2009 à 21:14
"Si l'utilisateur veut fermer, tu ne dois pas être en boucle de travail."
Je ne suis pas d'accord avec cela. Par exemple, si tu es en train de parcourir tous les fichiers du disque pour une recherche particulière qui dure longtemps, et qu'en plein milieu, l'utilisateur décide inopinément de quitter windows, il faut stopper la boucle. Là est l'utilité d'un flag. Mais c'est loin d'être le seul cas...
Bacterius
Messages postés3792Date d'inscriptionsamedi 22 décembre 2007StatutMembreDernière intervention 3 juin 201610 4 mai 2009 à 21:34
Je n'ai pas dit le contraire, je dis qu'il faut éviter les longs traitements qui empêchent Windows de fermer. Mais je me suis mal exprimé je le reconnais. En fait on va mettre à True une variable booléenne globale quand on veut stopper la boucle, et la boucle, à chaque itération teste la variable et se casse si c'est à True (en gros, après on peut utiliser une variable de type Integer pour spécifier plusieurs types de réactions, etc ...) ?
vieuxpere
Messages postés195Date d'inscriptionsamedi 3 janvier 2004StatutMembreDernière intervention17 décembre 2014 5 mai 2009 à 11:37
Bonjour, je suis désolé ça ne fonctionne pas !
A la demande de fermeture Windows, mon application se ferme, mais windows s'interrompt de fermer, je dois donc continuer à le faire en deux fois.
Voici mon code complet :
public
{ Déclarations publiques }
// Déclaration de la procédure qui va permettre de detecter la fermeture de Windows
procedure WMQueryEndSession(var Msg: TWMQueryEndSession); message WM_QUERYENDSESSION;
// Detecte la fermeture de Windows (demandée par l'utilisateur)
procedure TForm17.WMQueryEndSession (var Msg: TWMQueryEndSession);
begin
Inherited;
Msg.Result:= 1;
Application.Terminate; // On ferme l'application
end;
Merci par avance pour votre aide, car là je séche !
Bacterius
Messages postés3792Date d'inscriptionsamedi 22 décembre 2007StatutMembreDernière intervention 3 juin 201610 5 mai 2009 à 18:15
Bon sang ! Enleve ce Application.Terminate ! Il n'a rien à faire là ! On s'occupera de fermer ton app, ne t'inquiète pas (en revanche, si tu as une boucle, utilise les flags).
Et euh je vois Form17. Il ne faut mettre ce code que dans (et DANS) ta fiche principale, et non pas dans une autre fiche.
ni69
Messages postés1418Date d'inscriptionsamedi 12 juin 2004StatutMembreDernière intervention 5 juillet 201012 5 mai 2009 à 21:54
Bonsoir,
Je me permets un petit complément à ce que tu viens d'écrire :
Mieux vaut parfois utiliser exit plutôt que break dans les boucles, pour sortir directement de la procédure (si une boucle n'est pas finie et qu'on passe à la suite, il y a risque de plantage). Mais cela dépend bien évidemment du contexte ! On peut trouver des cas contraires où c'est exit qui génère un bug (mais c'est quand même plus rare).
Quoiqu'il en soit ces flags ne doivent être employés qu'en cas de réelle nécessité, sinon il finit par y en avoir de partout et ça alourdit le code en diminuant sa lisibilité.
A savoir aussi que généralement les flags sont des booléens, car on n'a dans la plupart des cas pas besoin d'une information dépassant la taille du bit.
Bacterius
Messages postés3792Date d'inscriptionsamedi 22 décembre 2007StatutMembreDernière intervention 3 juin 201610 5 mai 2009 à 22:11
En effet on aurait pu utiliser un Boolean. Mais parfois on a besoin de savoir exactement ce que veut l'utilisateur (par exemple 0 pour tout va bien, 1 pour l'utilisateur a choisi de fermer l'application, 2 pour le système va s'éteindre). Mais c'est également rare ...