cs_ShamSS
Messages postés5Date d'inscriptionvendredi 19 décembre 2008StatutMembreDernière intervention22 janvier 2009
-
9 janv. 2009 à 14:32
cs_ShamSS
Messages postés5Date d'inscriptionvendredi 19 décembre 2008StatutMembreDernière intervention22 janvier 2009
-
22 janv. 2009 à 12:10
Bonjour.
je suis en train de developper une petite application d'auto-completion et je n'arrive pas a recuperer la position du caret sur les application exterieur (genre firefox, notepad...) afin de positionner ma petite fenetre de choix a coté de ce caret.
Si quelqu'un aurrais une piste pour m'aider ce serais cool.
En tout cas, cordialement et merci d'avance.
IntPtr ptrE = FindWindowEx(ptrW,
IntPtr.Zero,
"RICHEDIT50W",
null);
int start 0, end 0;
SendMessage(ptrE, EM_GETSEL,
ref start,
ref end);
Console.WriteLine(
"Start {0}, End {1}", start, end);
}
}
}
Lutinore> Un truc qui m'est pas clair: je vois tout le temps des déclarations différentes pour SendMessage, (paramètres, type de retour, etc...) comment ça se fait?
Lutinore
Messages postés3246Date d'inscriptionlundi 25 avril 2005StatutMembreDernière intervention27 octobre 201241 11 janv. 2009 à 00:48
Les IntPtr doivent être initialisés avec l'adresse des Int32 et comme ça n'a pas bcp de sens en code managé il faut mieux utiliser soit des "ref int" soit du code unsafe comme ça :
cs_ShamSS
Messages postés5Date d'inscriptionvendredi 19 décembre 2008StatutMembreDernière intervention22 janvier 2009 9 janv. 2009 à 15:45
Merci Lutinor je vais essayé ça.
mais l'utilisation de EM_GETSEL est quand meme assez abstraite pour moi.
C'est un type opaque et il y a pas de fonction precise pour l'utiliser mais je vais faire mes recherche merci beaucoups.
Lutinore
Messages postés3246Date d'inscriptionlundi 25 avril 2005StatutMembreDernière intervention27 octobre 201241 10 janv. 2009 à 01:52
Bidou > dans les déclarations P/Invoke on peut plus ou moins changer les types du moment qu'on respecte leur taille pour ne pas déborder de la pile à l'exécution, il n'y pas de vérification de type effectué puisque le compilo ne connait rien de la fonction non managée.
En général j'essaye de rester le plus fidèle à la déclaration du header C/C++. C'est pas ce que j'ai fait dans le lien de mon exemple précedent et c'est vrai que retourner un bool c'est vilain et ça peut jouer des tours, je sais pas ou j'ai copie/collé cette défintion. ^^
La défintion la plus proche du header ( et la plus sûre ) est celle-ci :
cs_Bidou
Messages postés5487Date d'inscriptiondimanche 4 août 2002StatutMembreDernière intervention20 juin 201360 10 janv. 2009 à 13:09
Ha ok, voilà qui est plus clair....
Mais alors dans le cas où j'envoie EM_GELSEL (comme dans l'exemple plus haut), comment je fais pour récupérer le start et le end avec des IntPtr pour wParam et lParam? J'ai essayé avec Marshal.ToInt mais ça donnait de rien de bien...
J'ai fini par passer des ref int et là c'était bon, mais alors je n'utilise plus la "bonne" signature.
Lutinore
Messages postés3246Date d'inscriptionlundi 25 avril 2005StatutMembreDernière intervention27 octobre 201241 10 janv. 2009 à 15:36
Tu utilises une bonne signature et à mon avis la plus simple en mode non-unsafe puisque on est sûre que EM_GELSEL renvoie des pointeurs sur des valeurs 32 bits. Les IntPtr ça ne marchait pas car tu as du oublier de les initialiser, ce qui revient à envoyer des pointeurs invalides ou NULL.
en unsafe ça donne ça :
int start = 0;
int end = 0;
// Unsafe
SendMessage( hWnd, EM_GETSEL, ( IntPtr )( &start ), ( IntPtr )( &end ) );
cs_ShamSS
Messages postés5Date d'inscriptionvendredi 19 décembre 2008StatutMembreDernière intervention22 janvier 2009 22 janv. 2009 à 08:55
merci pour ces propositions de code seulement il faut déterminer en dur dans le code le programme de saisie de texte. Or dans mon projet, il doit être détermine en dynamique.
Est-ce qu'il y a une solution avec les identifiant de processus, ou autre avec la classe du programme associé ?
cs_ShamSS
Messages postés5Date d'inscriptionvendredi 19 décembre 2008StatutMembreDernière intervention22 janvier 2009 22 janv. 2009 à 12:10
Pardon pour le spam mais apres avoir fais d'autres test, je me suis rendu compte que c'est une position en nombre de caractere.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Text;
using System.Runtime.InteropServices;
private const int EM_GETSEL = 0xB0;
static unsafe void Main(string[] args)
{
IntPtr hWnd = FindWindowEx( IntPtr.Zero, IntPtr.Zero, "WordPadClass", null );
Console.WriteLine(hWnd+"\n");
if (hWnd != IntPtr.Zero)
{
hWnd = FindWindowEx(hWnd, IntPtr.Zero, "RICHEDIT50W", null);
if (hWnd != IntPtr.Zero)
{
int start = 0;
int end = 0;
SendMessage(hWnd, EM_GETSEL, (IntPtr)(&start), (IntPtr)(&end));
Console.WriteLine(hWnd + "\n");
Console.WriteLine("Point : " + start + " ; " + end + "\n");
}
else
Console.WriteLine(" La fonction FindWindowEx(hWnd, IntPtr.Zero, RICHEDIT50W, null) a raté.");
}
else
{
Console.WriteLine(" La fonction FindWindowEx( IntPtr.Zero, IntPtr.Zero, WordPadClass, null )a raté.");
}
System.Console.ReadKey();
}
}
}
Ceci est le code de test que j'ai utilisé. j'ai repris la code de Lutinore quasiment mot pour mot mais j'ai mis de l'affichage a la place des retours. J'ai donc remarqué que les variables start et end sont egales et me donnent un emplacement selon le nombre de carracter. Or, il me faut des coordonnées en pixel (ou autre) pour pouvoir placer ma fenetre correctement.
En esperant que mes explications soient claires
Cordialement et merci.