Subversion Repositories eFlore/Applications.cel

Rev

Rev 195 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
195 david 1
package org.tela_botanica.client.vues.image;
2 aperonnet 2
 
3
import org.tela_botanica.client.image.ImageMediateur;
4
import org.tela_botanica.client.interfaces.Rafraichissable;
5
 
6
import com.google.gwt.user.client.Event;
207 aurelien 7
import com.google.gwt.user.client.Timer;
8
import com.google.gwt.user.client.Window;
2 aperonnet 9
import com.google.gwt.user.client.ui.ClickListener;
10
import com.google.gwt.user.client.ui.Image;
207 aurelien 11
import com.google.gwt.user.client.ui.LoadListener;
12
import com.google.gwt.user.client.ui.MouseWheelListener;
13
import com.google.gwt.user.client.ui.MouseWheelVelocity;
2 aperonnet 14
import com.google.gwt.user.client.ui.Widget;
207 aurelien 15
import com.gwtext.client.core.AnimationConfig;
5 aperonnet 16
import com.gwtext.client.core.EventObject;
207 aurelien 17
import com.gwtext.client.core.Ext;
18
import com.gwtext.client.core.ExtElement;
19
 
2 aperonnet 20
import com.gwtext.client.core.RegionPosition;
5 aperonnet 21
import com.gwtext.client.dd.DragData;
22
import com.gwtext.client.dd.DragSource;
23
import com.gwtext.client.dd.DropTarget;
24
import com.gwtext.client.dd.DropTargetConfig;
2 aperonnet 25
import com.gwtext.client.widgets.Container;
26
import com.gwtext.client.widgets.Panel;
99 jpm 27
import com.gwtext.client.widgets.ToolTip;
2 aperonnet 28
import com.gwtext.client.widgets.event.PanelListenerAdapter;
5 aperonnet 29
import com.gwtext.client.widgets.grid.GridDragData;
2 aperonnet 30
import com.gwtext.client.widgets.layout.BorderLayout;
31
import com.gwtext.client.widgets.layout.BorderLayoutData;
32
 
33
/**
34
 * Panneau d'affichage d'une image avec des boutons précdents et suivant
5 aperonnet 35
 *
2 aperonnet 36
 * @author aurelien
5 aperonnet 37
 *
2 aperonnet 38
 */
39
public class ZoomImageVue extends Panel implements Rafraichissable {
5 aperonnet 40
 
2 aperonnet 41
	/**
42
	 * Le médiateur associé à la vue
43
	 */
5 aperonnet 44
	private ImageMediateur iMediateur = null;
45
 
2 aperonnet 46
	/**
47
	 * Panneau conteneur pour l'image
48
	 */
5 aperonnet 49
	private Panel imageConteneur = null;
2 aperonnet 50
	/**
51
	 * l'image à afficher
52
	 */
5 aperonnet 53
	private Image image = new Image("vide.jpg");
2 aperonnet 54
	/**
55
	 * Bouton précédent
56
	 */
57
	private final com.google.gwt.user.client.ui.Button prev = new com.google.gwt.user.client.ui.Button();
58
	/**
59
	 * Bouton suivant
60
	 */
61
	private final com.google.gwt.user.client.ui.Button suiv = new com.google.gwt.user.client.ui.Button();
5 aperonnet 62
 
2 aperonnet 63
	/**
64
	 * Taille originale Y de l'image
65
	 */
5 aperonnet 66
	private int imageHeight = 0;
67
 
2 aperonnet 68
	/**
69
	 * Taille originale X de l'image
70
	 */
5 aperonnet 71
	private int imageWidth = 0;
2 aperonnet 72
 
73
	/**
5 aperonnet 74
	 * Identifiant de l'image
75
	 */
76
	private String idImage = "0" ;
77
 
78
	/**
2 aperonnet 79
	 * Booleen d'initalisation général
80
	 */
5 aperonnet 81
	private boolean initialise = false;
82
 
2 aperonnet 83
	/**
84
	 * Booleen d'initalisation du conteneur d'image
85
	 */
5 aperonnet 86
	private boolean conteneurInitialise = false;
99 jpm 87
 
88
	private ToolTip tp = new ToolTip("<div class=\"x-tooltip-help\"> Double cliquez pour revenir à la vue précédente, utilisez la roulette ou les boutons fléchés pour faire défiler les images </div>") ;
5 aperonnet 89
 
2 aperonnet 90
	/**
207 aurelien 91
	 * true pour animer les transitions des animations
92
	 */
93
	private boolean animerTransition = false;
94
 
95
	/**
96
	 * duree des animation
97
	 */
98
	private float dureeAnimation = (float) 0.15 ;
99
 
100
	/**.
101
	 * boolean de gestion du double clic
102
	 */
103
	private boolean enClic = false;
104
 
105
	/**
2 aperonnet 106
	 * Constructeur sans argument (privé car ne doit être utilisé)
107
	 */
108
	@SuppressWarnings("unused")
5 aperonnet 109
	private ZoomImageVue() {
110
		super();
2 aperonnet 111
	}
5 aperonnet 112
 
2 aperonnet 113
	/**
114
	 * Constructeur avec argument
5 aperonnet 115
	 *
116
	 * @param im
117
	 *            le médiateur à associer à la vue
2 aperonnet 118
	 */
5 aperonnet 119
	public ZoomImageVue(ImageMediateur im) {
2 aperonnet 120
		super("Zoom");
5 aperonnet 121
		setId("x-view-zoom-panel");
2 aperonnet 122
		// on associe le médiateur
5 aperonnet 123
		iMediateur = im;
2 aperonnet 124
 
5 aperonnet 125
		prev.setStylePrimaryName("x-view-zoom-button-p");
126
		suiv.setStylePrimaryName("x-view-zoom-button-s");
127
 
2 aperonnet 128
		// on crée une image qui gère le double clic et la roulette de la souris
207 aurelien 129
		image = new Image();
130
 
5 aperonnet 131
		this.setHeader(false);
132
 
207 aurelien 133
		imageConteneur = new Panel();
134
 
2 aperonnet 135
		imageConteneur.setBorder(false);
5 aperonnet 136
 
137
		imageConteneur.add(image);
138
		imageConteneur.setId("x-view-zoom-img");
139
 
140
		// il n'existe pas de méthode pour ajouter un listener pour le double
141
		// clic sur une image
2 aperonnet 142
		// alors on lui dit manuellement de capter l'évènement double clic
207 aurelien 143
		//imageConteneur.sinkEvents(Event.ONMOUSEWHEEL);
5 aperonnet 144
 
2 aperonnet 145
		// l'image de base est vide
146
		image.setUrl("vide.jpg");
5 aperonnet 147
 
2 aperonnet 148
		this.setLayout(new BorderLayout());
5 aperonnet 149
 
2 aperonnet 150
		prev.setWidth("60px");
151
		suiv.setWidth("60px");
5 aperonnet 152
 
153
		this.add(prev, new BorderLayoutData(RegionPosition.WEST));
154
		this.add(imageConteneur, new BorderLayoutData(RegionPosition.CENTER));
155
		this.add(suiv, new BorderLayoutData(RegionPosition.EAST));
156
 
157
		imageConteneur.setMaskDisabled(true);
2 aperonnet 158
		this.setBorder(false);
5 aperonnet 159
 
160
		conteneurInitialise = true;
161
		initialise = true;
2 aperonnet 162
 
5 aperonnet 163
		configDragAndDrop() ;
164
 
2 aperonnet 165
		// on ajoute les listeners
166
		ajouterListeners();
99 jpm 167
 
100 jpm 168
		tp.setDismissDelay(1750);
99 jpm 169
		tp.applyTo(image.getElement()) ;
2 aperonnet 170
 
171
	}
172
 
173
	/**
174
	 * Méthode héritée de l'interface rafraichissable
5 aperonnet 175
	 *
176
	 * @param nouvelleDonnees
177
	 *            les nouvelles données
178
	 * @param repandreRafraichissement
179
	 *            le booleen qui dit si on doit répnadre l'évènement
2 aperonnet 180
	 */
5 aperonnet 181
	public void rafraichir(Object nouvelleDonnees,
182
			boolean repandreRafraichissement) {
183
 
2 aperonnet 184
		// si on reçoit une string
5 aperonnet 185
		if (nouvelleDonnees instanceof String[] && initialise
186
				&& conteneurInitialise) {
187
			String[] infos = (String[]) nouvelleDonnees;
2 aperonnet 188
			// c'est l'url de l'image qu'on associe à la vue
5 aperonnet 189
			if (infos[0] != null && infos[1] != null && infos[2] != null && infos[3] != null) {
190
				getImage().setUrl(infos[0]);
191
				int x = Integer.parseInt(infos[1]);
192
				int y = Integer.parseInt(infos[2]);
193
 
194
				setTailleImage(x, y);
2 aperonnet 195
 
5 aperonnet 196
				setIdImage(infos[3]) ;
197
 
207 aurelien 198
				//verifierEtRetaillerImage();
2 aperonnet 199
			}
5 aperonnet 200
		} else {
2 aperonnet 201
			// sinon on met une image vide
40 david 202
			getImage().setUrl("vide.jpg");
2 aperonnet 203
		}
204
	}
5 aperonnet 205
 
2 aperonnet 206
	/**
207
	 * Desactive visuellement le panneau et les boutons
208
	 */
5 aperonnet 209
	public void desactiverPanneau() {
210
		getImage().setUrl("vide.jpeg");
211
		prev.setEnabled(false);
2 aperonnet 212
		suiv.setEnabled(false);
213
	}
5 aperonnet 214
 
2 aperonnet 215
	/**
216
	 * Active visuellement le panneau et les boutons
217
	 */
5 aperonnet 218
	public void activerPanneau() {
2 aperonnet 219
		prev.setEnabled(true);
220
		suiv.setEnabled(true);
221
	}
5 aperonnet 222
 
2 aperonnet 223
	/**
224
	 * Ajoute les listeners pour la gestions d'évènement
225
	 */
5 aperonnet 226
	public void ajouterListeners() {
227
 
207 aurelien 228
		image.addLoadListener(new LoadListener() {
229
 
230
			public void onError(Widget sender) {
231
				// TODO Auto-generated method stub
232
 
233
			}
234
 
235
			public void onLoad(Widget sender) {
236
 
237
				int max = Math.min(imageConteneur.getHeight(), imageConteneur.getWidth());
238
				int[] tailleImage = calculerDimensions(getTailleImage(), max, max);
239
				ExtElement imgElement = Ext.get(image.getElement());
240
				if(animerTransition) {
241
					AnimationConfig a = new AnimationConfig() ;
242
					a.setDuration((float) dureeAnimation);
243
					imgElement.setHeight(tailleImage[1], a);
244
					imgElement.setWidth(tailleImage[0], a);
245
				} else {
246
					imgElement.setHeight(tailleImage[1], false);
247
					imgElement.setWidth(tailleImage[0], false);
248
				}
249
 
250
			}
251
 
252
		});
253
 
254
		image.addClickListener(new ClickListener() {
255
 
256
			public void onClick(Widget sender) {
257
 
258
				Window.alert(""+Event.getCurrentEvent().getButton());
259
					if(enClic) {
260
						getIMediateur().doubleClicZoomImage();
261
					} else {
262
						Timer t = new Timer() {
263
							public void run() {
264
								enClic = false;
265
							}
266
 
267
						};
268
						enClic = true;
269
						t.schedule(800);
270
					}
271
			}
272
 
273
		});
274
 
275
		image.addMouseWheelListener(new MouseWheelListener() {
276
 
277
			public void onMouseWheel(Widget sender, MouseWheelVelocity velocity) {
278
 
279
				if(velocity.getDeltaY() > 0) {
280
					suiv.click();
281
				} else {
282
					prev.click();
283
				}
284
			}
285
 
286
		});
287
 
2 aperonnet 288
		// gestion du clic sur le bouton précedent
289
		prev.addClickListener(new ClickListener() {
290
 
291
			// en cas de clic
292
			public void onClick(Widget sender) {
293
				// on notifie le médiateur
294
				getIMediateur().clicBoutonZoomImage("prev");
295
			}
5 aperonnet 296
 
2 aperonnet 297
		});
5 aperonnet 298
 
2 aperonnet 299
		// gestion du clic sur le bouton suivant
300
		suiv.addClickListener(new ClickListener() {
301
 
302
			// en cas de clic
303
			public void onClick(Widget sender) {
5 aperonnet 304
				// on notifie le médiateur
2 aperonnet 305
				getIMediateur().clicBoutonZoomImage("suiv");
306
			}
5 aperonnet 307
 
2 aperonnet 308
		});
5 aperonnet 309
 
2 aperonnet 310
		// gestion du redimensionnement
311
		this.addListener(new PanelListenerAdapter() {
5 aperonnet 312
 
2 aperonnet 313
			// lors d'un redimensionnement de l'application
5 aperonnet 314
 
315
			public void onBodyResize(Panel panel, java.lang.String width,
316
					java.lang.String height) {
2 aperonnet 317
				// on vérifie et on retaille l'image
5 aperonnet 318
				verifierEtRetaillerImage();
2 aperonnet 319
			}
5 aperonnet 320
		});
321
 
2 aperonnet 322
		// gestion du redimensionnement lors de l'affichage du conteneur
323
		imageConteneur.addListener(new PanelListenerAdapter() {
5 aperonnet 324
 
2 aperonnet 325
			// avant de finir d'afficher
5 aperonnet 326
 
327
			public void onAfterLayout(Container self) {
2 aperonnet 328
				// on redimensionne
5 aperonnet 329
				verifierEtRetaillerImage();
2 aperonnet 330
			}
5 aperonnet 331
		});
2 aperonnet 332
	}
333
 
334
	/**
5 aperonnet 335
	 * Verifie si l'image est plus grande que le conteneur et la retaille le cas
336
	 * echeant
2 aperonnet 337
	 */
5 aperonnet 338
	public void verifierEtRetaillerImage() {
2 aperonnet 339
		// si l'image est nulle
5 aperonnet 340
		if (image == null) {
2 aperonnet 341
			// on ne fait rien
5 aperonnet 342
			return;
343
		}
344
 
2 aperonnet 345
		// on prend la taille originale de l'image
5 aperonnet 346
		int originalX = getTailleImage()[0];
347
		int originalY = getTailleImage()[1];
348
 
349
		// on la transforme en float (la division entre entier donne de curieux
350
		// résultats)
351
		float fOriginalX = (new Float(originalX)).floatValue();
352
		float fOriginalY = (new Float(originalY)).floatValue();
353
 
2 aperonnet 354
		// et on fait le rapport longueur sur hauteur (pour connaitre le ratio)
5 aperonnet 355
		float rapportTaille = fOriginalX / fOriginalY;
356
 
2 aperonnet 357
		// la nouvelle taille est pour le moment égale à l'ancienne
5 aperonnet 358
		int nouvelleTailleX = originalX;
359
		int nouvelleTailleY = originalY;
360
 
2 aperonnet 361
		// on prend la taille du conteneur
5 aperonnet 362
		int tailleConteneurX = imageConteneur.getWidth();
363
		int tailleConteneurY = imageConteneur.getHeight();
364
 
2 aperonnet 365
		// si celle-ci est égale à 0 (conteneur mal initialisé)
5 aperonnet 366
		/*
367
		 * if(imageConteneur.getHeight() == 0 && tailleConteneurX == 0) { // on
368
		 * essaie de la calculer en fonction de la taille du parent et des
369
		 * frères tailleConteneurY = this.getHeight() ; tailleConteneurX =
370
		 * this.getWidth() - prev.getOffsetWidth() * 2 ;
371
		 *  }
372
		 */
373
 
374
		// si l'image ne rentre pas telle quelle (longueur ou hauteur trop
375
		// grande)
376
		if (originalY > tailleConteneurY || originalX > tailleConteneurX) {
2 aperonnet 377
			// si la longueur de l'image est la plus grande des deux
5 aperonnet 378
			if (originalX > originalY) {
379
				// on prend la longueur comme taille de référence, qu'on met à
380
				// la longueur du conteneur
381
				nouvelleTailleX = tailleConteneurX;
382
				// et on recalcule la hauteur, par rapport à la nouvelle
383
				// longueur, en gardant le format de 'limage
384
				nouvelleTailleY = (int) Math.floor(nouvelleTailleX * 1
385
						/ rapportTaille);
386
			} else {
387
				// si la hauteur est la plus grande, on fait le même genre
388
				// d'opération en prenant la hauteur comme référence
389
				nouvelleTailleY = tailleConteneurY;
390
				nouvelleTailleX = (int) Math.floor(nouvelleTailleY
391
						* rapportTaille);
2 aperonnet 392
			}
393
		}
5 aperonnet 394
 
2 aperonnet 395
		// on modifie enfin la taille de l'image pour qu'elle soit affichée
5 aperonnet 396
		getImage().setSize("" + nouvelleTailleX + "px",
397
				"" + nouvelleTailleY + "px");
2 aperonnet 398
	}
5 aperonnet 399
 
207 aurelien 400
	public int[] calculerDimensions(int[] tailleXY, double tailleMax, double tailleConteneur) {
401
 
402
		float[] tailleXYf = {new Float(tailleXY[0]),new Float(tailleXY[1])} ;
403
        float tailleOr = Math.min(new Float(tailleMax),new Float(tailleConteneur)) ;
404
        float maxTaille = Math.max(tailleXYf[1],tailleXYf[0]) ;
405
        float[] XYresize = new float[2];
406
 
407
        if(maxTaille == tailleXY[0]) {
408
            float rapport = tailleXYf[1]/tailleXYf[0] ;
409
            XYresize[0] = tailleOr ;
410
            XYresize[1] = tailleOr*rapport ;
411
        }else {
412
            float rapport = tailleXYf[0]/tailleXYf[1] ;
413
            XYresize[1] = tailleOr ;
414
            XYresize[0] = tailleOr*rapport ;
415
        }
416
 
417
        int[] res = {Math.round(XYresize[0]),Math.round(XYresize[1])} ;
418
 
419
        return res;
420
    }
421
 
5 aperonnet 422
	public void configDragAndDrop()
423
	{
2 aperonnet 424
 
5 aperonnet 425
		// on fabrique la nouvelle configuration
426
		// les éléments sur lesquels on fait du drag 'n drop doivent tous avoir le même ddGroup
427
		DropTargetConfig dtc = new DropTargetConfig();
428
		dtc.setdDdGroup("DragGroupName");
429
 
430
		//La drop target permet de gérer l'évenement onDrop sur l'élement courant
431
		@SuppressWarnings("unused")
432
		DropTarget tg = new DropTarget(this, dtc)
433
		{
434
			public boolean notifyDrop(DragSource source, EventObject e, DragData data){
435
 
436
				// si on reçoit des données provenant d'une grille
437
				if(data instanceof GridDragData)
438
		    	  {
439
		    		  		// on appelle le médiateur
440
		    		  		return iMediateur.lierObsDD(source, e, data,getId()) ;
441
		    	  }
442
				return false ;
443
			}
444
 
445
			public String notifyOver(DragSource source, EventObject e, DragData data){
446
			    return "x-dd-drop-ok";
447
			}
448
		};
449
 
450
	}
451
 
2 aperonnet 452
	/**
453
	 * Accesseur pour le médiateur
5 aperonnet 454
	 *
2 aperonnet 455
	 * @return le médiateur associé à la vue
456
	 */
457
	public ImageMediateur getIMediateur() {
458
		return iMediateur;
459
	}
460
 
461
	/**
462
	 * Accesseur au conteneur de l'image
5 aperonnet 463
	 *
2 aperonnet 464
	 * @return le conteneur de l'image
465
	 */
466
	public Image getImage() {
467
		return image;
468
	}
5 aperonnet 469
 
470
	/**
471
	 * Acesseurs pour l'identifiant de l'image
472
	 * @return l'id de l'image
473
	 */
474
	public String getIdImage()
475
	{
476
		return idImage ;
477
	}
2 aperonnet 478
 
479
	/**
480
	 * Accesseur pour le bouton précédent
5 aperonnet 481
	 *
2 aperonnet 482
	 * @return le bouton précédent
483
	 */
484
	public com.google.gwt.user.client.ui.Button getPrev() {
485
		return prev;
486
	}
487
 
488
	/**
489
	 * Accesseur pour le bouton suivant
5 aperonnet 490
	 *
2 aperonnet 491
	 * @return le bouton suivant
492
	 */
493
	public com.google.gwt.user.client.ui.Button getSuiv() {
494
		return suiv;
495
	}
5 aperonnet 496
 
2 aperonnet 497
	/**
498
	 * Setter pour la taille de l'image
5 aperonnet 499
	 *
7 aperonnet 500
	 * @param x
5 aperonnet 501
	 *            la largeur en pixels
7 aperonnet 502
	 * @param y
5 aperonnet 503
	 *            la hauteur en pixels
2 aperonnet 504
	 */
5 aperonnet 505
	public void setTailleImage(int x, int y) {
506
		imageHeight = y;
507
		imageWidth = x;
2 aperonnet 508
	}
509
 
510
	/**
5 aperonnet 511
	 * Setteur pour l'identifiant de l'image
512
	 * @param id le nouvel identifiant
513
	 */
514
	public void setIdImage(String id)
515
	{
516
		idImage = id ;
517
	}
518
 
519
	/**
2 aperonnet 520
	 * renvoie la taille originale de l'image
5 aperonnet 521
	 *
2 aperonnet 522
	 * @return un tableau de deux entiers contenant la largeur puis la hauteur
523
	 */
5 aperonnet 524
	public int[] getTailleImage() {
525
		int[] taille = { imageHeight, imageWidth };
526
 
527
		return taille;
2 aperonnet 528
	}
5 aperonnet 529
 
530
	/**
531
	 * Accesseur pour le conteneur de l'image
532
	 * @return le conteur de l'image
533
	 */
534
	public Panel getImageConteneur() {
2 aperonnet 535
 
5 aperonnet 536
		return imageConteneur;
537
 
2 aperonnet 538
	}
539
}