Subversion Repositories eFlore/Applications.cel

Rev

Rev 1291 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
2 aperonnet 1
package org.tela_botanica.client.vues;
2
 
56 david 3
 
4
import org.tela_botanica.client.interfaces.ListePaginable;
2 aperonnet 5
import org.tela_botanica.client.interfaces.Rafraichissable;
6
 
7
import com.gwtext.client.core.EventCallback;
8
import com.gwtext.client.core.EventObject;
9
import com.gwtext.client.core.Template;
10
import com.gwtext.client.data.Record;
11
import com.gwtext.client.data.SimpleStore;
12
import com.gwtext.client.data.Store;
13
import com.gwtext.client.widgets.Button;
14
import com.gwtext.client.widgets.Toolbar;
15
import com.gwtext.client.widgets.ToolbarButton;
16
import com.gwtext.client.widgets.ToolbarTextItem;
17
import com.gwtext.client.widgets.event.ButtonListenerAdapter;
18
import com.gwtext.client.widgets.form.ComboBox;
19
import com.gwtext.client.widgets.form.Field;
20
import com.gwtext.client.widgets.form.TextField;
21
import com.gwtext.client.widgets.form.event.ComboBoxListenerAdapter;
22
import com.gwtext.client.widgets.form.event.TextFieldListenerAdapter;
23
 
24
/**
5 aperonnet 25
 * Barre de pagination asynchrone avec filtrage des touches et accès directs et
26
 * séquentiels à une page
27
 *
2 aperonnet 28
 * @author aurelien
5 aperonnet 29
 *
2 aperonnet 30
 */
59 david 31
public class BarrePaginationVue extends Toolbar implements Rafraichissable {
5 aperonnet 32
 
2 aperonnet 33
	/**
34
	 * Instance du médiateur à qui on notifie les évnènements
35
	 */
56 david 36
	private ListePaginable listePaginable = null;
5 aperonnet 37
 
2 aperonnet 38
	/**
39
	 * Bouton précédent
40
	 */
5 aperonnet 41
	private ToolbarButton prevPage = new ToolbarButton();
2 aperonnet 42
	/**
43
	 * Bouton suivant
44
	 */
5 aperonnet 45
	private ToolbarButton suivPage = new ToolbarButton();
2 aperonnet 46
	/**
5 aperonnet 47
	 * Numéro de la page courante (attention, commence à zéro pour des raisons
48
	 * pratiques)
2 aperonnet 49
	 */
5 aperonnet 50
	private int pageCourante = 0;
2 aperonnet 51
	/**
52
	 * Nombre de page total
53
	 */
5 aperonnet 54
	private int pageTotale = 1;
2 aperonnet 55
	/**
56
	 * Nombre d'élements total
57
	 */
5 aperonnet 58
	private int nbElement = 0;
2 aperonnet 59
	/**
60
	 * Nombre d'élément par page
61
	 */
5 aperonnet 62
	private int taillePage = 0;
2 aperonnet 63
	/**
64
	 * Texte statique de la toolbar 1
65
	 */
5 aperonnet 66
	private ToolbarTextItem page = new ToolbarTextItem("Page ");
2 aperonnet 67
	/**
68
	 * Affichage de la page courante
69
	 */
5 aperonnet 70
	private TextField champPage = new TextField("" + (pageCourante + 1));
2 aperonnet 71
	/**
72
	 * Affichage de "sur pageTotale "
73
	 */
5 aperonnet 74
	private ToolbarTextItem surTotalPage = new ToolbarTextItem(" sur "
75
			+ pageTotale);
2 aperonnet 76
	/**
77
	 * Texte statique de la toolbar 2
78
	 */
5 aperonnet 79
	private ToolbarTextItem afficherNbElem = new ToolbarTextItem("Afficher ");
2 aperonnet 80
	/**
5 aperonnet 81
	 * Combobox permettant de selectionner le nombre d'élements à afficher par
82
	 * page et donc de changer la variable taillePage
2 aperonnet 83
	 */
5 aperonnet 84
	private ComboBox selecteurTaillePage = new ComboBox();
59 david 85
 
60 jpm 86
	/**
87
	 * Les différents intervalles de page possibles (intialisé par défaut)
88
	 */
89
	String[] pages = { "200","100","50", "20", "10" };
59 david 90
 
60 jpm 91
	Store storeIntervalle = null ;
92
 
93
 
59 david 94
 
2 aperonnet 95
	/**
59 david 96
	 * Label indiquant le type d'element affiché
97
	 *
98
	 */
99
 
100
	private String labelElement="Elements";
101
 
102
	/**
103
	 * taille de la page par defaut
104
	 *
105
	 */
106
 
107
	private int taillePageDefaut = 50;
108
 
109
 
110
	/**
2 aperonnet 111
	 * Texte statique de la toolbar 3
112
	 */
5 aperonnet 113
	private ToolbarTextItem nbElemParPage = new ToolbarTextItem(
59 david 114
			labelElement + " par page ");
2 aperonnet 115
	/**
116
	 * Affiche l'intervalle des éléments contenus dans la page
117
	 */
59 david 118
	private ToolbarTextItem intervalleElements = new ToolbarTextItem(labelElement + " "
5 aperonnet 119
			+ pageCourante * taillePage + " sur " + nbElement);
2 aperonnet 120
 
59 david 121
 
2 aperonnet 122
	/**
56 david 123
	 * retourne la liste associé à la barre
2 aperonnet 124
	 */
56 david 125
 
126
	public ListePaginable getlistePaginable() {
127
		return listePaginable;
2 aperonnet 128
	}
5 aperonnet 129
 
130
	/***************************************************************************
2 aperonnet 131
	 * constructeur sans argument (privé car ne doit pas être utilisé)
132
	 */
133
	@SuppressWarnings("unused")
59 david 134
	private BarrePaginationVue() {
5 aperonnet 135
		super();
2 aperonnet 136
	}
5 aperonnet 137
 
2 aperonnet 138
	/**
139
	 * constructeur avec paramètres
5 aperonnet 140
	 *
141
	 * @param im
142
	 *            le médiateur à associer à la barre
2 aperonnet 143
	 */
59 david 144
	public BarrePaginationVue(ListePaginable lst) {
5 aperonnet 145
		super();
146
 
56 david 147
		listePaginable = lst;
5 aperonnet 148
 
128 aurelien 149
		// on remplit l'espace pour que l'intervalle d'élement se place à droite
150
		// de la barre
151
		addItem(intervalleElements);
152
		addSpacer();
153
		// on remplit l'espace pour que l'intervalle d'élement se place à droite
154
		// de la barre
155
		addFill();
156
 
5 aperonnet 157
		// on dispose un peu de texte et quelques espaces pour séparer les
158
		// éléments
159
		addButton(prevPage);
160
		addSpacer();
161
		addItem(page);
162
		addField(champPage);
163
		addItem(surTotalPage);
164
		addSpacer();
165
		addButton(suivPage);
166
 
167
		champPage.setWidth(30);
168
 
169
		addSpacer();
170
		addItem(afficherNbElem);
171
 
172
		prevPage.setIcon("page_prev.png");
173
		suivPage.setIcon("page_suiv.png");
174
 
60 jpm 175
		setIntervallesPages(pages) ;
5 aperonnet 176
 
177
		// le template definit ce que l'on affiche pour chaque element du store
178
		// dans la combobox
179
		final Template tp = new Template("<div class=\"x-combo-list-item\">"
180
				+ "{nb_page}" + "<div class=\"x-clear\"></div></div>");
2 aperonnet 181
		tp.compile();
5 aperonnet 182
 
183
		selecteurTaillePage.setTpl(tp);
60 jpm 184
		selecteurTaillePage.setStore(storeIntervalle);
5 aperonnet 185
		selecteurTaillePage.setEditable(false);
186
		addField(selecteurTaillePage);
59 david 187
		selecteurTaillePage.setValue(""+taillePageDefaut);
5 aperonnet 188
		selecteurTaillePage.setWidth(50);
189
		addItem(nbElemParPage);
59 david 190
 
5 aperonnet 191
 
2 aperonnet 192
		// on ajoute les différents listeners
5 aperonnet 193
		ajouterListeners();
2 aperonnet 194
	}
5 aperonnet 195
 
59 david 196
 
2 aperonnet 197
	/**
59 david 198
	 *  Texte nommant les elements pagines (Images, Observation, truc, machin etc...).
199
	 * 	@param label
200
	 */
201
 
202
	public void setLabelElement(String label) {
203
		this.labelElement = label;
204
		nbElemParPage.setText(labelElement + " par page ");
205
		intervalleElements.setText(labelElement + " " + pageCourante * taillePage + " sur " + nbElement);
206
	}
207
 
208
	public void setTaillePageParDefaut(int taille) {
209
		this.taillePageDefaut = taille;
210
		selecteurTaillePage.setValue(""+taillePageDefaut);
211
	}
212
 
60 jpm 213
	public void setIntervallesPages(String[] intervalle)
214
	{
215
		String[][] intervallesPages = new String[intervalle.length][1] ;
216
 
217
		for(int i = 0 ; i < intervalle.length ; i++)
218
		{
219
			intervallesPages[i][0] = intervalle[i] ;
220
		}
221
 
222
		storeIntervalle = new SimpleStore(new String[] { "nb_page" },
223
				intervallesPages );
224
		storeIntervalle.load();
225
		selecteurTaillePage.setStore(storeIntervalle);
226
	}
59 david 227
 
60 jpm 228
 
59 david 229
	/**
5 aperonnet 230
	 * ajoute les différents listeners nécessaires au bon fonctionnement des
231
	 * éléments de la barre de pagination
2 aperonnet 232
	 */
5 aperonnet 233
	private void ajouterListeners() {
59 david 234
 
235
 
236
 
5 aperonnet 237
		// boutons suivants et précédents
2 aperonnet 238
		prevPage.addListener(new ButtonListenerAdapter() {
239
 
1291 aurelien 240
			@Override
2 aperonnet 241
			public void onClick(Button button, EventObject e) {
5 aperonnet 242
 
2 aperonnet 243
				// si la page courante n'est pas la première
5 aperonnet 244
				if (pageCourante > 0) {
2 aperonnet 245
					// on décrémente la page courante de 1
5 aperonnet 246
					pageCourante--;
2 aperonnet 247
					// on rafraichit l'affichage
5 aperonnet 248
					rafraichirNumeroPage();
2 aperonnet 249
					// et on notifie le médiateur de l'évenement
56 david 250
					listePaginable.changerNumeroPage(pageCourante);
5 aperonnet 251
 
2 aperonnet 252
				}
253
			}
5 aperonnet 254
		});
255
 
2 aperonnet 256
		suivPage.addListener(new ButtonListenerAdapter() {
5 aperonnet 257
 
1291 aurelien 258
			@Override
2 aperonnet 259
			public void onClick(Button button, EventObject e) {
5 aperonnet 260
 
2 aperonnet 261
				// si la page courante n'est pas la dernière
5 aperonnet 262
				if (pageCourante < pageTotale - 1) {
2 aperonnet 263
					// on incrémente la page courante de 1
5 aperonnet 264
					pageCourante++;
2 aperonnet 265
					// on rafraichit l'affichage
5 aperonnet 266
					rafraichirNumeroPage();
2 aperonnet 267
					// et on notifie le médiateur de l'évenement
56 david 268
					listePaginable.changerNumeroPage(pageCourante);
5 aperonnet 269
 
270
				}
2 aperonnet 271
			}
5 aperonnet 272
		});
273
 
2 aperonnet 274
		champPage.addListener(new TextFieldListenerAdapter() {
275
 
1291 aurelien 276
			@Override
2 aperonnet 277
			public void onSpecialKey(Field field, EventObject e) {
5 aperonnet 278
 
2 aperonnet 279
				// on teste si la touche entrée a été pressée
5 aperonnet 280
				if (e.getKey() == EventObject.ENTER) {
281
					int nouvellePage = pageCourante;
2 aperonnet 282
					// on teste avec parseInt si la valeur entrée est un entier
5 aperonnet 283
					try {
284
						nouvellePage = Integer
285
								.parseInt(champPage.getRawValue());
286
					}
287
					// si ce n'est pas le cas alors on remet le numéro de page
288
					// correct
289
					catch (NumberFormatException nfe) {
290
						rafraichirNumeroPage();
291
						champPage.focus(true);
292
						return;
293
					}
2 aperonnet 294
 
5 aperonnet 295
					// si la conversion reussit on verifie s'il est nécessaire
296
					// de changer de page
297
					// et si la nouvelle est comprise dans l'intervalle des
298
					// pages existantes (0..pageTotale)
299
					if (nouvellePage != pageCourante + 1 && nouvellePage > 0
300
							&& nouvellePage <= pageTotale) {
301
						// le cas échéant, on charge la nouvelle page et on
302
						// notifie le médiateur
303
						changerPageCourante(nouvellePage - 1);
56 david 304
						listePaginable.changerNumeroPage(pageCourante);
5 aperonnet 305
 
306
					} else {
307
						// sinon on reaffiche l'ancien numero de page sans rien
308
						// changer
309
						rafraichirNumeroPage();
310
						champPage.focus(true);
2 aperonnet 311
					}
312
				}
313
			}
5 aperonnet 314
 
1291 aurelien 315
			@Override
2 aperonnet 316
			public void onFocus(Field field) {
5 aperonnet 317
 
318
				champPage.focus(true);
2 aperonnet 319
			}
5 aperonnet 320
 
2 aperonnet 321
		});
5 aperonnet 322
 
323
		// pour éviter de se compliquer la vie, on filtre tous les charactères
324
		// non numériques
2 aperonnet 325
		champPage.addKeyPressListener(new EventCallback() {
326
 
1291 aurelien 327
			@Override
2 aperonnet 328
			public void execute(EventObject e) {
5 aperonnet 329
 
2 aperonnet 330
				// si c'est un numerique
5 aperonnet 331
				if (Character.isDigit((char) e.getCharCode())) {
2 aperonnet 332
					// on laisse passer
5 aperonnet 333
					return;
2 aperonnet 334
				}
5 aperonnet 335
 
2 aperonnet 336
				// si c'est la touche entrée ou backspace (valider ou effacer)
5 aperonnet 337
				if (e.getKey() == EventObject.ENTER
338
						|| e.getKey() == EventObject.BACKSPACE) {
2 aperonnet 339
					// on laisse passer
5 aperonnet 340
					return;
341
				} else {
342
					// sinon on remet le numero de page correct et on annule
343
					// l'évenement
344
					rafraichirNumeroPage();
345
					e.stopEvent();
2 aperonnet 346
				}
347
			}
5 aperonnet 348
 
349
		});
350
 
351
		// listener pour la selection dans la combobox
2 aperonnet 352
		selecteurTaillePage.addListener(new ComboBoxListenerAdapter() {
5 aperonnet 353
 
1291 aurelien 354
			@Override
2 aperonnet 355
			public void onSelect(ComboBox comboBox, Record record, int index) {
5 aperonnet 356
 
357
				String nouvelleTaillePageString = comboBox.getStore()
358
						.getRecordAt(index).getAsString("nb_page");
359
				int nouvelleTaillePage = Integer
360
						.parseInt(nouvelleTaillePageString);
361
 
2 aperonnet 362
				// si la taille de page est différente de l'ancienne
5 aperonnet 363
				if (nouvelleTaillePage != taillePage) {
2 aperonnet 364
					// on la change
5 aperonnet 365
					changerTaillePage(nouvelleTaillePage);
2 aperonnet 366
				}
367
				// et on met la valeur à jour dans la combobox
5 aperonnet 368
				comboBox.setValue(nouvelleTaillePageString);
2 aperonnet 369
			}
5 aperonnet 370
 
371
		});
2 aperonnet 372
	}
5 aperonnet 373
 
2 aperonnet 374
	/**
5 aperonnet 375
	 * Met à jour les affichage sur les numéros de pages et d'intervalle
376
	 * d'éléments à partir des variables de classes
2 aperonnet 377
	 */
5 aperonnet 378
	public void rafraichirNumeroPage() {
379
		surTotalPage.setText(" sur " + pageTotale);
380
 
381
		if (nbElement == 0) {
382
			champPage.setValue("" + (0));
383
			// on met simplement à jour l'intervalle qui contient toujours le
384
			// même nombre d'éléments
59 david 385
			intervalleElements.setText(labelElement + " 0 - 0 sur 0");
5 aperonnet 386
		} else {
387
			champPage.setValue("" + (pageCourante + 1));
388
 
389
			// si la page n'est pas la dernière
390
			if (pageCourante + 1 != pageTotale) {
391
				// sauf pour la dernière page qui contient souvent moins
392
				// d'élements que le nombre d'élements par page
59 david 393
				intervalleElements.setText(labelElement + " " + pageCourante
5 aperonnet 394
						* taillePage + " - " + (pageCourante + 1) * taillePage
395
						+ " sur " + nbElement);
396
			} else {
397
				// on met simplement à jour l'intervalle qui contient toujours
398
				// le même nombre d'éléments
59 david 399
				intervalleElements.setText(labelElement + " " + pageCourante
5 aperonnet 400
						* taillePage + " - " + nbElement + " sur " + nbElement);
401
			}
2 aperonnet 402
		}
403
	}
5 aperonnet 404
 
2 aperonnet 405
	/**
5 aperonnet 406
	 * Met à jour la page en cours
407
	 *
408
	 * @param nouvellePageCourante
409
	 *            la nouvelle page en cours
2 aperonnet 410
	 */
5 aperonnet 411
	public void changerPageCourante(int nouvellePageCourante) {
412
		pageCourante = nouvellePageCourante;
2 aperonnet 413
	}
5 aperonnet 414
 
2 aperonnet 415
	/**
416
	 * Methode héritée de l'interface rafraichissable
417
	 */
1291 aurelien 418
	@Override
2 aperonnet 419
	public void rafraichir(Object nouvelleDonnees,
420
			boolean repandreRaffraichissement) {
5 aperonnet 421
 
2 aperonnet 422
		// si on reçoit un tableau de int
5 aperonnet 423
		if (nouvelleDonnees instanceof int[]) {
424
			int[] page = (int[]) nouvelleDonnees;
425
			// le premier élement est le nombre de pages totales
426
			pageTotale = page[0];
2 aperonnet 427
			// le second la page en cours
5 aperonnet 428
			pageCourante = page[1];
2 aperonnet 429
			// le troisième la taille de la page
5 aperonnet 430
			taillePage = page[2];
2 aperonnet 431
			// et le dernier le nombre total d'éléments
5 aperonnet 432
			nbElement = page[3];
433
 
434
			// si la page courante dépasse la page totale (cas normalement
435
			// improbable car géré en amont)
436
			// on met le numéro de page à la page courante -1 (car la page
437
			// courante est comptée à partir
2 aperonnet 438
			// de zéro)
5 aperonnet 439
			if (pageCourante >= pageTotale && pageCourante != 0) {
440
				pageCourante = pageTotale - 1;
2 aperonnet 441
				// le cas échéant on en notifie le médiateur
56 david 442
				listePaginable.changerNumeroPage(pageCourante);
2 aperonnet 443
			}
444
		}
5 aperonnet 445
 
446
		// enfin on rafraichit les informations affichées à partir des nouvelles
447
		// variables de classes mises à jour
448
		rafraichirNumeroPage();
2 aperonnet 449
	}
5 aperonnet 450
 
2 aperonnet 451
	/**
452
	 * Renvoie les différents intervalles de pages possibles
5 aperonnet 453
	 *
60 jpm 454
	 * @return un tableau de string qui contient les différentes
5 aperonnet 455
	 *         taille de pages
2 aperonnet 456
	 */
60 jpm 457
	public String[] getNbPages() {
5 aperonnet 458
 
459
		return pages;
2 aperonnet 460
	}
461
 
462
	/**
5 aperonnet 463
	 * Envoie au médiateur une demande pour modifier la taille de la page (qui
464
	 * va à son tour faire les modifications nécessaires)
465
	 *
466
	 * @param nouvelleTaillePage
467
	 *            la nouvelle taille de page (élement appartenant au tableau
468
	 *            renvoyé par getNbPages())
2 aperonnet 469
	 */
470
	public void changerTaillePage(int nouvelleTaillePage) {
5 aperonnet 471
 
56 david 472
		listePaginable.changerTaillePage(nouvelleTaillePage);
5 aperonnet 473
 
2 aperonnet 474
	}
5 aperonnet 475
 
2 aperonnet 476
	/**
5 aperonnet 477
	 * Selectionne la valeur correspond à celle passée en paramètre dans la
478
	 * combobox (si elle existe)
479
	 *
480
	 * @param nouvelleTaillePage
481
	 *            la nouvelle taille de page
2 aperonnet 482
	 */
483
	public void selectionnerTaillePage(int nouvelleTaillePage) {
5 aperonnet 484
 
485
		selecteurTaillePage.setValue("" + nouvelleTaillePage);
2 aperonnet 486
	}
5 aperonnet 487
 
2 aperonnet 488
}