Subversion Repositories eFlore/Applications.cel

Rev

Details | 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;
60 jpm 4
import org.tela_botanica.client.interfaces.ListePaginable;
2 aperonnet 5
import org.tela_botanica.client.interfaces.Rafraichissable;
6
import org.tela_botanica.client.interfaces.VueListable;
195 david 7
import org.tela_botanica.client.vues.BarrePaginationVue;
2 aperonnet 8
 
9
import com.google.gwt.user.client.Element;
81 jpm 10
import com.google.gwt.user.client.ui.HTML;
134 aurelien 11
import com.gwtext.client.core.EventCallback;
2 aperonnet 12
import com.gwtext.client.core.EventObject;
134 aurelien 13
import com.gwtext.client.core.Ext;
14
import com.gwtext.client.core.ExtElement;
2 aperonnet 15
import com.gwtext.client.core.XTemplate;
16
import com.gwtext.client.data.FieldDef;
17
import com.gwtext.client.data.IntegerFieldDef;
18
import com.gwtext.client.data.Record;
19
import com.gwtext.client.data.RecordDef;
20
import com.gwtext.client.data.Store;
21
import com.gwtext.client.data.StringFieldDef;
5 aperonnet 22
import com.gwtext.client.dd.DragData;
23
import com.gwtext.client.dd.DragSource;
24
import com.gwtext.client.dd.DropTarget;
25
import com.gwtext.client.dd.DropTargetConfig;
2 aperonnet 26
import com.gwtext.client.util.Format;
27
import com.gwtext.client.widgets.Component;
134 aurelien 28
import com.gwtext.client.widgets.Container;
2 aperonnet 29
import com.gwtext.client.widgets.DataView;
30
import com.gwtext.client.widgets.Panel;
146 aurelien 31
import com.gwtext.client.widgets.event.ContainerListener;
2 aperonnet 32
import com.gwtext.client.widgets.event.ContainerListenerAdapter;
33
import com.gwtext.client.widgets.event.DataViewListenerAdapter;
146 aurelien 34
import com.gwtext.client.widgets.event.PanelListenerAdapter;
5 aperonnet 35
import com.gwtext.client.widgets.grid.GridDragData;
2 aperonnet 36
 
37
/**
5 aperonnet 38
 * Galerie d'images miniatures Avec barre de pagination
39
 *
2 aperonnet 40
 * @author aurelien
41
 */
5 aperonnet 42
public class GalerieImageVue extends Panel implements Rafraichissable,
60 jpm 43
		VueListable, ListePaginable {
2 aperonnet 44
 
45
	/**
46
	 * instance du médiateur
5 aperonnet 47
	 */
48
	private ImageMediateur iMediateur = null;
2 aperonnet 49
	/**
5 aperonnet 50
	 * Dataview, littéralement "vue de données" qui permet de définir la manière
51
	 * d'afficher les données
52
	 */
53
	private DataView dView = null;
2 aperonnet 54
	/**
5 aperonnet 55
	 * Dataview, littéralement "vue de données" qui permet de définir la manière
56
	 * d'afficher les données
57
	 */
58
	private Store st = null;
2 aperonnet 59
	/**
5 aperonnet 60
	 * Barre de pagination gérant l'affichage d'un nombre donné d'élements par
61
	 * page et la navigation entre eux
62
	 */
59 david 63
	private BarrePaginationVue pt = null;
2 aperonnet 64
	/**
65
	 * Booleen indiquant si la galerie est instanciée ou pas
5 aperonnet 66
	 */
67
	private boolean estInstancie = false;
140 aurelien 68
 
69
	private boolean garderRatio = false;
70
 
71
	private int tailleOr = 100 ;
146 aurelien 72
 
73
	boolean lienUploadInitialise = false ;
74
 
75
	HTML videPanel = null ;
2 aperonnet 76
 
146 aurelien 77
 
2 aperonnet 78
	/**
79
	 * Constructeur sans argument, privé car ne doit pas être utilisé
80
	 */
81
	@SuppressWarnings("unused")
5 aperonnet 82
	private GalerieImageVue() {
83
		super();
2 aperonnet 84
	}
5 aperonnet 85
 
2 aperonnet 86
	/**
87
	 * Constructeur avec argument
5 aperonnet 88
	 *
89
	 * @param im
90
	 *            le médiateur avec lequel la vue va communiquer
2 aperonnet 91
	 */
92
	public GalerieImageVue(ImageMediateur im) {
93
		super("Galerie");
94
		iMediateur = im;
146 aurelien 95
 
2 aperonnet 96
		// on ajoute des listeners au composant tout entier
97
		this.addListener(new ContainerListenerAdapter() {
98
 
5 aperonnet 99
			// pour gagner du temps on n'instancie la vue en elle même que lors
100
			// du premier affichage (lazy rendering)
101
 
2 aperonnet 102
			public void onShow(Component component) {
103
 
104
				if (!estInstancie) {
105
					initialiser();
106
				}
134 aurelien 107
 
108
			}
2 aperonnet 109
		});
146 aurelien 110
 
111
		AjouterListenersLiens();
5 aperonnet 112
 
2 aperonnet 113
		// et on ajoute la tool bar
60 jpm 114
		pt = new BarrePaginationVue(this);
59 david 115
		pt.setLabelElement("Images");
116
		pt.setTaillePageParDefaut(50);
5 aperonnet 117
		this.setBottomToolbar(pt);
2 aperonnet 118
 
119
	}
120
 
121
	/**
5 aperonnet 122
	 * Ajoute tous les listeners nécessaires à l'intercation utilisateur avec la
123
	 * vue de données
2 aperonnet 124
	 */
125
	public void ajouterListenersDataView() {
126
 
127
		// ajout de listeners pour la gestion de la selection
128
		// dans la galerie
146 aurelien 129
 
2 aperonnet 130
		dView.addListener(new DataViewListenerAdapter() {
5 aperonnet 131
 
2 aperonnet 132
			// gestion du clic sur une image
5 aperonnet 133
 
2 aperonnet 134
			public void onClick(DataView source, int index, Element node,
135
					EventObject e) {
146 aurelien 136
 
2 aperonnet 137
				// on en notifie le médiateur
138
				getIMediateur().clicGalerieImage(index, node, e);
139
 
140
			}
141
 
142
			// gestion du clic droit
5 aperonnet 143
 
2 aperonnet 144
			public void onContextMenu(DataView source, int index, Element node,
145
					EventObject e) {
146
 
5 aperonnet 147
				// on stoppe l'évenement
2 aperonnet 148
				e.stopEvent();
149
				// et on notifie le médiateur
150
				getIMediateur().montrerContextMenu(e);
151
 
152
			}
153
 
154
			// gestion du double clic
5 aperonnet 155
 
2 aperonnet 156
			public void onDblClick(DataView source, int index, Element node,
157
					EventObject e) {
158
 
159
				// on notife le mediateur
160
				getIMediateur().clicGalerieImage(index, node, e);
5 aperonnet 161
 
2 aperonnet 162
			}
163
 
164
			// gestion des actions en fonction de la selection
5 aperonnet 165
 
2 aperonnet 166
			public void onSelectionChange(DataView view, Element[] selections) {
167
 
168
				// s'il n'y a aucun élement sélectionné
169
				if (selections.length <= 0) {
170
					// on en notifie le médiateur
171
					getIMediateur().aucuneSelection();
172
				} else {
173
					// sinon on notifie le médiateur
174
					getIMediateur().selection();
5 aperonnet 175
					// et on lui demande de synchroniser la séléction avec les
176
					// autres vues
2 aperonnet 177
					getIMediateur().synchroniserSelection("galerie");
178
				}
179
			}
180
		});
181
	}
182
 
183
	/**
184
	 * Accesseur pour la dataview
5 aperonnet 185
	 *
186
	 * @return la dataview
2 aperonnet 187
	 */
188
	public DataView getDView() {
189
		return dView;
190
	}
191
 
192
	/**
193
	 * Renvoie les ids des images sélectionnées
5 aperonnet 194
	 *
195
	 * @return un tableau de String contenant les identifiants des images
196
	 *         sélectionnées
2 aperonnet 197
	 */
198
	public String[] getIdSelectionnees() {
199
		Record[] selection = getDView().getSelectedRecords();
200
		int taille = selection.length;
201
		String id_selection[] = new String[taille];
202
 
203
		for (int i = 0; i < selection.length; i++) {
204
 
205
			id_selection[i] = selection[i].getAsString("num_image");
206
		}
207
 
208
		return id_selection;
209
	}
210
 
211
	/**
212
	 * Accesseur pour le médiateur
5 aperonnet 213
	 *
2 aperonnet 214
	 * @return le médiateur associé à la vue
215
	 */
216
	public ImageMediateur getIMediateur() {
217
		return iMediateur;
218
	}
219
 
220
	/**
221
	 * Accesseur pour le store
5 aperonnet 222
	 *
2 aperonnet 223
	 * @return le store associé à la vue
224
	 */
225
	public Store getSt() {
226
		return st;
227
	}
5 aperonnet 228
 
2 aperonnet 229
	/**
5 aperonnet 230
	 * Accesseur pour la toolbar
231
	 *
2 aperonnet 232
	 * @return la toolbar associée à la vue
233
	 */
59 david 234
	public BarrePaginationVue getToolBarVue() {
5 aperonnet 235
		return pt;
2 aperonnet 236
	}
237
 
238
	/**
5 aperonnet 239
	 * Fonction d'initialisation du contenu (appelée lors du premier affichage
240
	 * de la liste)
2 aperonnet 241
	 */
242
	public void initialiser() {
5 aperonnet 243
 
2 aperonnet 244
		// Preparation de la dataview et du template
245
		// le template va créer une div contenant une image
246
		// pour chacune des photos
247
		final XTemplate template = new XTemplate(
248
				new String[] {
249
						"<tpl for='.'>",
250
						"<div class='thumb-wrap' id='{num_image}'>",
140 aurelien 251
						"<div class='thumb'><img src='{url_image_M}' width='{taille_x_s} px' height='{taille_y_s} px' title='{num_image}'></div>",
2 aperonnet 252
						"<span>{nom}</span></div>", "</tpl>",
253
						"<div class='x-clear'></div>" });
5 aperonnet 254
		// pour des raisons de performances on compile le template en une
255
		// fonction
2 aperonnet 256
		template.compile();
257
 
258
		// la dataview affichera les images en accord avec le template
259
		// cree precedemment
260
		dView = new DataView("div.thumb-wrap") {
261
 
262
			public void prepareData(Data data) {
263
				data.setProperty("shortName", Format.ellipsis(data
140 aurelien 264
						.getProperty("num_image"), 15));
265
 
266
				int[] XY = {data.getPropertyAsInt("taille_x") ,data.getPropertyAsInt("taille_y")} ;
267
				int[] XYresize ;
268
 
269
				if(garderRatio) {
270
					XYresize = calculerDimensions(XY);
271
				}
272
				else {
273
					XYresize = new int[2] ;
274
					XYresize[0] = XYresize[1] = tailleOr ;
275
 
276
				}
277
 
278
				data.setProperty("taille_x_s", XYresize[0]);
279
				data.setProperty("taille_y_s", XYresize[1]);
2 aperonnet 280
			}
281
		};
282
		dView.setTpl(template);
283
 
284
		// parametre d'affichage de la dataview
285
		this.setAutoScroll(true);
286
		dView.setAutoHeight(true);
287
		dView.setMultiSelect(true);
288
		dView.setOverCls("x-view-over");
146 aurelien 289
		dView.setEmptyText("");
2 aperonnet 290
 
291
		// creation du store
292
		FieldDef defNumImage = new IntegerFieldDef("num_image");
293
		FieldDef defDatImage = new StringFieldDef("dat_image");
294
		FieldDef defLieImage = new StringFieldDef("lie_image");
295
		FieldDef defAppImage = new StringFieldDef("app_image");
296
		FieldDef defUrlImageS = new StringFieldDef("url_image_S");
297
		FieldDef defUrlImageM = new StringFieldDef("url_image_M");
298
		FieldDef defUrlImage = new StringFieldDef("url_image");
140 aurelien 299
		FieldDef defTailleX = new IntegerFieldDef("taille_x");
300
		FieldDef defTailleY = new IntegerFieldDef("taille_y");
2 aperonnet 301
		FieldDef[] defTab = { defNumImage, defDatImage, defLieImage,
140 aurelien 302
				defAppImage, defUrlImageS, defUrlImageM, defUrlImage,defTailleX,defTailleY};
2 aperonnet 303
		RecordDef rd = new RecordDef(defTab);
304
		st = new Store(rd);
305
		dView.setStore(st);
306
 
307
		this.getDView().setLoadingText("chargement");
308
 
146 aurelien 309
		this.add(dView);
310
		dView.hide();
311
 
312
		videPanel = new HTML("<div class=\"avertissement\" >Aucune image à afficher. <br/> <a id=\"lienUploadMultiple\" href=\"#\" > Cliquez ici pour ajouter un dossier entier ou plusieurs fichiers </a> (nécessite Java) <br/> " +
313
				"			<a id=\"lienUploadSimple\" href=\"#\" > Cliquez ici pour ajouter un fichier à la fois </a> <br/> " +
314
				"			Pour ajouter des images plus tard, allez dans le menu Fichier -> Ajouter des images </div>");
315
		this.add(videPanel);
316
 
2 aperonnet 317
		// ajouts de la gestion des evenements pour la dataview
146 aurelien 318
		configDragAndDrop() ;
2 aperonnet 319
		ajouterListenersDataView();
146 aurelien 320
 
321
		estInstancie = true ;
322
 
2 aperonnet 323
	}
5 aperonnet 324
 
325
	public void configDragAndDrop()
326
	{
327
		// on fabrique la nouvelle configuration
328
		// les éléments sur lesquels on fait du drag 'n drop doivent tous avoir le même ddGroup
329
		DropTargetConfig dtc = new DropTargetConfig();
330
		dtc.setdDdGroup("DragGroupName");
2 aperonnet 331
 
5 aperonnet 332
		//La drop target permet de gérer l'évenement onDrop sur l'élement courant
333
		@SuppressWarnings("unused")
334
		DropTarget tg = new DropTarget(this, dtc)
335
		{
336
			public boolean notifyDrop(DragSource source, EventObject e, DragData data){
337
 
338
				// si les données proviennent d'une grille
339
				if(data instanceof GridDragData)
340
		    	{
341
					// on appelle le médiateur
342
		    		  	return iMediateur.lierObsDD(source, e, data,getId()) ;
343
		    	}
344
				return false ;
345
			}
346
 
347
			public String notifyOver(DragSource source, EventObject e, DragData data){
348
			    return "x-dd-drop-ok";
349
			}
350
		};
351
 
352
	}
353
 
2 aperonnet 354
	/**
355
	 * Méthode héritée de l'interface rafraichissable
356
	 */
357
	public void rafraichir(Object nouvelleDonnees,
358
			boolean repandreRafraichissement) {
359
 
360
		// si l'objet reçu est un store
361
		if (nouvelleDonnees instanceof Store) {
5 aperonnet 362
 
134 aurelien 363
				st = (Store) nouvelleDonnees;
364
 
365
			if(st.getCount() != 0) {
366
 
367
				// on le charge
368
				st.load();
369
 
146 aurelien 370
				if(videPanel.isVisible()) {
371
					videPanel.setVisible(false) ;
372
				}
373
 
374
				if(!dView.isVisible()) {
375
					dView.setVisible(true);
376
				}
134 aurelien 377
					// on l'affecte à la vue
378
					dView.setStore(st);
379
					// et on rafrachit la vue
380
					dView.refresh();
381
			}
140 aurelien 382
			else
383
			{
384
				st.removeAll();
385
				st.load();
386
				dView.setStore(st);
146 aurelien 387
 
388
				if(dView.isVisible()) {
389
					dView.hide() ;
390
				}
391
 
392
				if(!videPanel.isVisible()) {
393
					videPanel.setVisible(true);
394
					AjouterListenersLiens();
395
				}
396
 
140 aurelien 397
			}
2 aperonnet 398
		}
399
 
400
		// si le composant doit répandre le rafraichissement
401
		if (repandreRafraichissement) {
5 aperonnet 402
			// il en notifie le médiateur en lui donnant une copie des données
403
			// et en notifiant qu'il en est l'expéditeur
2 aperonnet 404
			getIMediateur().synchroniserDonneesZoomListeGalerie(
405
					nouvelleDonnees, this);
146 aurelien 406
		}
2 aperonnet 407
	}
408
 
409
	/**
5 aperonnet 410
	 * Méthode héritée de l'interface VueListable Sélectionne les images dans la
411
	 * galerie suivant les identifiants donnés en paramètres
412
	 *
413
	 * @param ids
414
	 *            les identifiants des images à sélectionner
2 aperonnet 415
	 */
416
	public void selectionnerImages(int[] ids) {
417
 
418
		getDView().select(ids);
419
 
420
	}
5 aperonnet 421
 
60 jpm 422
	public void changerNumeroPage(int pageCourante) {
423
 
424
		iMediateur.changerNumeroPage(pageCourante) ;
425
 
426
	}
427
 
428
	public void changerTaillePage(int nouvelleTaillePage) {
429
 
430
		iMediateur.changerTaillePage(nouvelleTaillePage) ;
431
	}
140 aurelien 432
 
433
	public int[] calculerDimensions(int[] tailleXY) {
434
 
435
		float[] tailleXYf = {new Float(tailleXY[0]),new Float(tailleXY[1])} ;
436
        float tailleOr = this.tailleOr ;
437
        float maxTaille = Math.max(tailleXYf[1],tailleXYf[0]) ;
438
        float[] XYresize = new float[2];
439
 
440
        if(maxTaille == tailleXY[0]) {
441
            float rapport = tailleXYf[1]/tailleXYf[0] ;
442
            XYresize[0] = tailleOr ;
443
            XYresize[1] = tailleOr*rapport ;
444
        }else {
445
            float rapport = tailleXYf[0]/tailleXYf[1] ;
446
            XYresize[1] = tailleOr ;
447
            XYresize[0] = tailleOr*rapport ;
448
        }
449
 
450
        int[] res = {Math.round(XYresize[0]),Math.round(XYresize[1])} ;
451
 
452
        return res;
453
    }
146 aurelien 454
 
455
	private void AjouterListenersLiens() {
60 jpm 456
 
146 aurelien 457
		addListener(new PanelListenerAdapter() {
458
 
459
			public void onAfterLayout(Container c) {
460
				ExtElement uploadS = Ext.get("lienUploadSimple");
461
				uploadS.removeAllListeners();
462
				uploadS.addListener("click", new EventCallback() {
463
					public void execute(EventObject e) {
464
						getIMediateur().uploaderImages(false);
465
					}
466
 
467
				}) ;
468
 
469
				ExtElement uploadM = Ext.get("lienUploadMultiple");
470
				uploadM.removeAllListeners();
471
				uploadM.addListener("click", new EventCallback() {
472
					public void execute(EventObject e) {
473
						getIMediateur().uploaderImages(true);
474
					}
475
				});
476
			}
477
 
478
		});
479
	}
2 aperonnet 480
}