Thomas Lebrun
Développer une visionneuse d'images avec WPF et WCF
Au travers de cet article, nous allons découvrir comment mettre en place une visionneuse d'images, grâce aux technologies WPF et WCF.
Par Thomas Lebrun publié le 22/04/2008 à 22:46
 
La première chose qu'il convient de faire est de définir les interfaces de notre service WCF. Au niveau de notre serveur, voici la définition de l'interface IServerContract :

[ServiceContract(CallbackContract = typeof(IClientContract))]

    public interface IServerContract

    {

        [OperationContract(IsOneWay = false)]

        void StartWatcher(string path);

 

        [OperationContract(IsOneWay = false)]

        void GetImagesFromPathAtFirstLoad(string path);

    }

Comme vous pouvez le voir, nous avons assigné la propriété CallbackContract sur notre interface. En effet, étant donné que notre application devra réagir à certaines actions déclenchées par notre serveur, nous devons passer par cette propriété et indiquer le type de l'interface qui sera utilisée, coté client, pour définir les actions devant s'exécuter suite aux appels du serveur. Si l'on regarde cette deuxième interface, nous remarquons qu'il n'y a rien de bien particulier, hormis la propriété IsOneWay, positionnée au dessus des méthodes, et qui permet d'indiquer que les méthodes sont à sens unique (pas d'aller-retour avec le serveur) :

[ServiceContract]

    public interface IClientContract

    {

        [OperationContract(IsOneWay = true)]

        void GetImagesFromServer(List<CustomImage> img);

 

        [OperationContract(IsOneWay = true)]

        void GetUpdateFromFileShare(CustomImage img);

    }

Vous avez sans doute aussi pu remarquer que l'on utilise une classe particulière, CustomImage, pour communiquer entre notre client et notre serveur. Voici le code, très simple, de cette classe (notez au passage l'utilisation d'un tableau de byte, pour représenter l'image, car le type Image n'est pas sérializable et ne peut donc pas être échangé entre notre client et notre serveur) :

[DataContract]

    public class CustomImage

    {

        public enum ChangeType

        {

            Add,

            Delete

        }

 

        [DataMember]

        public string Name { get; set; }

 

        [DataMember]

        public string FullPath { get; set; }

 

        [DataMember]

        public Byte[] Img { get; set; }

 

        [DataMember]

        public ChangeType Modification { get; set; }

    }

Maintenant que nous avons définit les différentes opérations possibles pour notre serveur, il nous faut le développer. Nous allons donc écrire le code du service WCF qui sera hébergé par le serveur. Ce service WCF implémente l'interface affichée plus haut, nous devons donc créer une classe qui implémente cette interface :

[ServiceBehavior(ConcurrencyMode = ConcurrencyMode.Multiple, InstanceContextMode = InstanceContextMode.Single)]

    public class ServiceImplementation : IServerContract

    {

        #region Member Fields

 

        private System.IO.FileSystemWatcher fsw = null;

        private OperationContext context = null;

        private IClientContract client = null;

 

        #endregion

 

        #region IServerContract Members

 

        /// <summary>

        /// Start the FileSystemWatcher on the server.

        /// </summary>

        /// <param name="path">The path to the directory to watch.</param>

        public void StartWatcher(string path)

        {

            try

            {

                context = OperationContext.Current;

                client = context.GetCallbackChannel<IClientContract>();

 

                fsw = new System.IO.FileSystemWatcher();

                fsw.Path = path;

                fsw.IncludeSubdirectories = true;

                fsw.Filter = "*.*";

                fsw.Created += new System.IO.FileSystemEventHandler(GetUpdateFileFromFSW);

                fsw.Deleted += new System.IO.FileSystemEventHandler(GetUpdateFileFromFSW);

 

                fsw.EnableRaisingEvents = true;

            }

            catch (Exception ex)

            {

                throw new FaultException<ServerException>(new ServerException(

                    string.Format("An error occured !{0}{1}", Environment.NewLine, ex.Message)));

            }

        }

 

        /// <summary>

        /// Retrieve the images which are in the specified folder during the startup.

        /// </summary>

        /// <param name="path">The folder to get the images.</param>

        public void GetImagesFromPathAtFirstLoad(string path)

        {

            var imgList = new List<CustomImage>();

            CustomImage image = null;

 

            foreach (var file in Directory.GetFiles(path))

            {

                try

                {

                    if (Path.GetExtension(file).ToLower() == ".png" || Path.GetExtension(file).ToLower() == ".jpg" ||

                        Path.GetExtension(file).ToLower() == ".jpeg" || Path.GetExtension(file).ToLower() == ".bmp" ||

                        Path.GetExtension(file).ToLower() == ".gif")

                    {

                        using (var stream = File.OpenRead(file))

                        {

                            if (stream.Length > 0)

                            {

                                image = new CustomImage();

 

                                image.Name = Path.GetFileName(file);

                                image.FullPath = Path.GetFullPath(file);

                                image.Modification = CustomImage.ChangeType.Add;

                            }

                        }

 

                        byte[] array = GetByteFromImage(file);

                        image.Img = array;

 

                        imgList.Add(image);

                    }

                }

                catch (Exception ex)

                {

                    throw new FaultException<ServerException>(new ServerException(

                        string.Format("An error occured !{0}{1}", Environment.NewLine, ex.Message)));

                }

            }

 

            if (client != null)

            {

                client.GetImagesFromServer(imgList);

            }

        }

 

        #endregion

 

        /// <summary>

        /// Convert a file on a byte array.

        /// </summary>

        /// <param name="file">The file to convert.</param>

        /// <returns>An array of byte corresponding to the file.</returns>

        private static byte[] GetByteFromImage(string file)

        {

            byte[] array;

 

            try

            {

                using (FileStream fs = new FileStream(file, FileMode.Open, FileAccess.Read))

                {

                    BinaryReader reader = new BinaryReader(fs);

 

                    array = reader.ReadBytes((int)fs.Length);

 

                    reader.Close();

                }

            }

            catch (Exception ex)

            {

                throw new FaultException<ServerException>(new ServerException(

                    string.Format("An error occured !{0}{1}", Environment.NewLine, ex.Message)));

            }

 

            return array;

        }

 

        /// <summary>

        /// Method called when the FileSystemWatcher detects that a file has been added/deleted.

        /// </summary>

        private void GetUpdateFileFromFSW(object sender, System.IO.FileSystemEventArgs e)

        {

            CustomImage image = null;

 

            if (e.ChangeType == System.IO.WatcherChangeTypes.Created)

            {

                if (File.Exists(e.FullPath) && (Path.GetExtension(e.FullPath).ToLower() == ".png" ||

                    Path.GetExtension(e.FullPath).ToLower() == ".jpg"

                    || Path.GetExtension(e.FullPath).ToLower() == ".jpeg" 

                    || Path.GetExtension(e.FullPath).ToLower() == ".bmp" 

                    || Path.GetExtension(e.FullPath).ToLower() == ".gif"))

                {

                    try

                    {

                        using (var stream = File.OpenRead(e.FullPath))

                        {

                            if (stream.Length > 0)

                            {

                                image = new CustomImage();

 

                                image.Name = e.Name;

                                image.FullPath = e.FullPath;

                                image.Modification = CustomImage.ChangeType.Add;

                            }

                        }

 

                        byte[] array = GetByteFromImage(e.FullPath);

                        image.Img = array;

                    }

                    catch (Exception ex)

                    {

                        throw new FaultException<ServerException>(new ServerException(

                            string.Format("An error occured !{0}{1}", Environment.NewLine, ex.Message)));

                    }

                }

            }

            else if (e.ChangeType == System.IO.WatcherChangeTypes.Deleted)

            {

                image = new CustomImage();

 

                image.Name = e.Name;

                image.FullPath = e.FullPath;

                image.Modification = CustomImage.ChangeType.Delete;

            }

 

            if (image != null)

            {

                SendUpdateToClient(image);

            }

        }

 

        /// <summary>

        /// Inforl the client that an update is available.

        /// </summary>

        /// <param name="image">The image that have been added/deleted.</param>

        private void SendUpdateToClient(CustomImage image)

        {

            if (client != null)

            {

                client.GetUpdateFromFileShare(image);

            }

        }

    }

Il y a plusieurs choses à observer sur cette classe. Tout d'abord, la méthode StartWatcher, qui sera appelée par notre client, se contente de démarrer le FileSystemWatcher qui va scanner le répertoire contenant les images et nous informer des mises à jour disponibles. La méthode GetImagesFromPathAtFirstLoad, quand à elle, est utilisée pour récupérer les images lors du premier lancement de l'application cliente. Cette méthode, ainsi que la méthode utilisée par le FileSystemWatcher, font toutes les deux appels aux méthodes proposées par l'interface « cliente ».
Ensuite, vous pouvez remarquez que la propriété InstanceContexteMode a été assignée à Single. Cela est obligatoire car l'application cliente, celle qui va consommer le service WCF, est une application WPF (cela aurait été la même chose dans le cas d'une simple application WindowsForms). En ce qui concerne le reste de la classe, vous pouvez constater qu'il est tout ce qu'il y a de plus simple et ne nécessite pas d'être commenté.

 Commentaires (2) - Développer une visionneuse d'images avec WPF et WCF 

Discussion démarée par Laurent Duveau le 24/04/2008 à 22:56 , 2 commentaire(s).

 Dernières Publications      

Utilisation de jQuery avec ASP.NET MVC
  Développer une IHM à page unique avec ASP.NET MVC et jQuery
par Nicolas Moyère posté le 30/06/2008 à 10:28, lu 824 fois, #0
Tags: ASP.NET MVC, Ajax
Windows Media Center et WCF : développez votre maison intelligente
  Le développement d'applications pour Windows Media Center est facilité avec l'arrivée du SDK 5.3. Même si l'on sent un modèle objet bien lourd derrière, il devient plus facile d'exposer les fonctionnalités de WMC sous la forme de services WCF.
par Frédéric Colin posté le 23/06/2008 à 08:04, lu 891 fois, #0
Notions avancées avec Biztalk Server 2006 R2
  Utilisation des notions d'interchange, corrélation et convoi avec BizTalk Server 2006 R2
par Kader Yildirim posté le 09/06/2008 à 08:04, lu 705 fois, #0
Lucene Persistence Engine pour Evaluant Universal Storage Services
  Suite à l'article de Laurent Kempé, voici un moteur de stockage pour EUSS permettant l'indexation d'entités métier avec Lucene.
par Nicolas Penin posté le 01/06/2008 à 23:38, lu 1091 fois, #1
Tags: C#, Linq
XMLA Trivia : Découverte du XMLA
  Le XMLA (XML for Analysis) est un langage normalisé par plusieurs éditeurs BI pour simplifier l'accès aux données aux cubes et aux métadonnées des bases multidimensionnelles.
par Renaud Harduin posté le 25/05/2008 à 11:57, lu 1008 fois, #1
Exploiter les données CSV via Linq en toute simplicité
  A partir du requêteur dynamique fourni en exemple avec Visual Studio 2008, nous allons essayer de remplir les propriétés d'un ensemble d'objets à partir des données d'un fichier CSV. Nous enrichirons aussi le parseur de nos propres fonctions.
par Frédéric Mélantois posté le 17/05/2008 à 11:41, lu 2785 fois, #0
Comment manipuler simplement le contenu d'un fichier WordML ?
  Manipulations autour du format WordML
par Fabien Reinle posté le 14/05/2008 à 23:55, lu 1405 fois, #0
Polymorphisme et contrats de données WCF
  WCF aborde les types polymorphes du point de vue de la sérialisation. En effet, la connaissance du type réel potentiel est rendue nécessaire dès la description du contrat de données. Une fois n'est pas coutume, j'ai réalisé l'exemple en VB.NET.
par Frédéric Colin posté le 14/05/2008 à 08:40, lu 2931 fois, #2

 Dernières Actualités      

Reprise du projet Reflector par RedGate
  La nouvelle était connue depuis quelques jours par les développeurs de plugins, mais c’est désormais officiel : Lutz Roeder, le responsable de Reflector confie à la société RedGate le futur du projet....
Microsoft publie Visual Studio 2008 Service Pack 1
  Il est recommandé d’utiliser l’outil Visual Studio 2008 Service Pack preparation Tool avant de faire l’installation du Service Pack si vous avez installé des versions béta sur votre machine. Une fois que...
Tags: Framework .NET, Visual Studio 2008
Evaluant dévoile ses sources
  L'ensemble des projets R&D réalisés par les consultants de la SSII Evaluant sont en cours de publication sur CodePlex . L'objectif est de les centraliser et surtout d'augmenter leur visibilité. L'avantage...
Le Silverlight Tour en français!
  Le Silverlight Tour passe maintenant dans les pays francophones! En effet RunAtServer Consulting est partenaire du Silverlight Tour pour la gestion de cette formation Silverlight en français à commencer...
Microsoft publie ASP.NET AJAX 4.0 CodePlex Preview 1
  Cette pré-version contient les améliorations suivantes: Client-side template rendering Declarative instantiation of behaviors and controls DataView control Markup extensions Bindings Vous pouvez en lire...
Tags: Ajax
Deep Earth – Une belle utilisation de Virtual Earth et de Silverlight Deep Zoom
  Ce projet très intéressant est disponible sur Codeplex et vous pouvez voir une démo sur la page suivante . Bien entendu comme touts les projets sur Codeplex vous avez accès aux sources....
Tags: Silverlight