//#17.0.1.0 WDZRHorizontale.js
//VersionVI: 30F170061k
// Le seul support technique disponible pour cette librairie est
// accessible a travers le service "Assistance Directe".

// Manipulation des champs zone repetee galerie

// Manipulation d'une zone repetee galerie
function WDZRHorizontale(sAliasChamp)
{
	// Si on est pas dans l'init d'un protoype
	if (sAliasChamp)
	{
		// Appel le constructeur de la classe de base
		WDChamp.prototype.constructor.apply(this, [sAliasChamp, undefined, undefined]);

		// Rien pour l'instant
	}
};

// Declare l'heritage
WDZRHorizontale.prototype = new WDChamp();
// Surcharge le constructeur qui a ete efface
WDZRHorizontale.prototype.constructor = WDZRHorizontale;

// Pas de l'augmentation de la vitesse de defilement
WDZRHorizontale.prototype.ms_nPasDefilement = 8;

// Initialisation
WDZRHorizontale.prototype.Init = function Init()
{
	// Appel de la methode de la classe de base
	WDChamp.prototype.Init.apply(this, arguments);

//	// Fixe le contenu du div de positionnement
//	if (bIEQuirks)
//	{
//		this.m_oDivPos.style.width = Math.min(this.m_oDivPosParent.offsetWidth, this.m_oDivPosTable.offsetWidth) + "px";
//		this.m_oDivScrollBar.parentNode.style.overflow = "hidden";
//	}

	// Initialisation du defilement
	this._InitDefilement();

	// Hook le resizing pour deplacer les elements en cas de changement de geometrie
	// => Pas de hook on est deja automatiquement appele sur la methode OnResize

	// Positionne le champ
	this._InitDeplace();
};

// Methode d'initialisation generale de la classe
// Filtre et ne s'execute que lors de l'init du PREMIER champ de ce type
WDZRHorizontale.prototype._vInitInitiale = function _vInitInitiale()
{
	// Appel de la methode de la classe de base
	WDChamp.prototype._vInitInitiale.apply(this, arguments);

	// Il faut intercepter cet evenement sur le document sinon on ne recoit pas le relachement du bouton hors de la barre
	HookOnXXX(document, 'onmouseup', 'mouseup', WDZRHorizontale.prototype._s_CallOnMouseUpFondSB);
	HookOnXXX(document, 'onmousemove', 'mousemove', WDZRHorizontale.prototype._s_CallOnMouseMoveFondSB);

	// S'auto efface pour ne plus etre appele
	WDZRHorizontale.prototype._vInitInitiale = clWDUtil.m_pfVide;
};

// Trouve les divers elements : liaison avec le HTML
WDZRHorizontale.prototype._vLiaisonHTML = function _vLiaisonHTML(sSuffixeHote)
{
	// Appel de la methode de la classe de base
	WDChamp.prototype._vLiaisonHTML.apply(this, arguments);

	// Recupere les differents champs caches
	this.m_oChampDeb = this.oGetElementByName(document, "_DEB");
	this.m_oChampOcc = this.oGetElementByName(document, "_OCC");
	this.m_oChampPos = this.oGetElementByName(document, "_POSV");

	// Recupere la partie defilante de la ZR
	this.m_oDivPos = this.oGetIDElement("POS");
	this.m_oDivPosTable = this.m_oDivPos.getElementsByTagName("table")[0];
	this.m_oDivPosParent = this.m_oDivPos.parentNode;
	this.m_oDivScrollBar = this.oGetIDElement("SB");
};

// Pas de fonction pour la gestion des Pcodes il faut utiliser directement DeclarePCode et RecuperePCode

// Initialisation du defilement
WDZRHorizontale.prototype._InitDefilement = function _InitDefilement()
{
	// Le gestionnaire de glissement de la barre en base
	this.m_oDragScrollBar = new WDDragScrollBar(this);

	// La gestion du defilement sur les boutons gauche et droite
	this._InitBoutons();
	// La gestion du defilement sur le fond de la barre de defilement
	this._InitDefilementFondSB()
};

// La gestion du defilement sur les boutons gauche et droite
WDZRHorizontale.prototype._InitBoutons = function _InitBouton()
{
	this._InitUnBouton("BG", false);
	this._InitUnBouton("BD", true);
};

// La gestion du defilement sur un des boutons
WDZRHorizontale.prototype._InitUnBouton = function _InitUnBouton(sSuffixe, bDroite)
{
	// Intercepte le bouton down, le bouton up
	// On en gere pas le clic car on veux un defilement immediat
	var oBtn = this.oGetIDElement(sSuffixe);
	var oThis = this;
	HookOnXXX(oBtn, 'onmousedown', 'mousedown', function(oEvent) { oThis.OnMouseDownBtn.apply(oThis, [oEvent || event, bDroite]); return clWDUtil.bStopPropagation(oEvent); });
	HookOnXXX(oBtn, 'onmouseup', 'mouseup', function(oEvent) { oThis.OnMouseUpBtn.apply(oThis, [oEvent || event, bDroite, true]); return clWDUtil.bStopPropagation(oEvent); });
};

// La gestion du defilement sur le fond de la barre de defilement
WDZRHorizontale.prototype._InitDefilementFondSB = function _InitDefilementFondSB()
{
	var oThis = this;
	HookOnXXX(this.m_oDivScrollBar.parentNode, 'onmousedown', 'mousedown', function(oEvent) { oThis.OnMouseDownFondSB.apply(oThis, [oEvent || event]); return clWDUtil.bStopPropagation(oEvent); });
	// Il faut intercepter cet evenement sur le document sinon on ne recoit pas le relachement du bouton hors de la barre
	// => Deja fait de maniere globale avec un seul evenement
};

// Positionne le champ en fonction de la valeur initiale du positionnement
WDZRHorizontale.prototype._InitDeplace = function _InitDeplace()
{
	// Recupere la valeur initiale
	var nPosition = parseInt(this.m_oChampPos.value, 10);
	// Filtre les valeurs invalides
	if (isNaN(nPosition) || (nPosition < 0))
	{
		nPosition = 0;
	}

	// Positionne le champ
	// Appele la methode interne commune qui reecrit dans m_oChampPos car on peut avoir modifie la valeur
	this._Deplace(nPosition);
};

// Positionne le champ en fonction d'une position et la memorise
WDZRHorizontale.prototype._Deplace = function _Deplace(nPosition)
{
	// Uniquement si la position change
	if (this.m_nPosition != nPosition)
	{
		// Memorise la nouvelle position
		this.m_nPosition = nPosition;
		this.m_oChampPos.value = nPosition;

		// Deplace les elements graphiques
		this._DeplaceElements();
	}
};

// - Tous les champs : notifie que une ZRs a ete MAJ en AJAX (= ZR non AJAX mais avec les ZR horizontales)
// Attention ne surcharge pas _vOnZRAfficheAJAXInterne mais _vOnZRAfficheAJAX pour etre avant le test de this.m_sAliasZR == sAliasZR
WDZRHorizontale.prototype._vOnZRAfficheAJAX = function _vOnZRAfficheAJAX(sAliasZR)
{
	// Appel de la methode de la classe de base (qui normalement ne fait rien car on est une ZR donc pas dans une autre ZR)
	WDChamp.prototype._vOnZRAfficheAJAX.apply(this, arguments);

	// Si c'est nous meme
	if (this.m_sAliasChamp == sAliasZR)
	{
		// Reinit de la liaison HTML
		this._vLiaisonHTML();
		// Deplace les elements graphiques
		this._DeplaceElements();
	}
};

// Recupere les differentes largeurs
WDZRHorizontale.prototype.nGetLargeurTotale = function nGetLargeurTotale()
{
//	return this.m_oDivPos.offsetWidth;
//	return this.m_oDivPos.scrollWidth;
	return this.m_oDivPosTable.offsetWidth;
};
WDZRHorizontale.prototype.nGetLargeurCliente = function nGetLargeurCliente()
{
	return this.m_oDivPosParent.clientWidth;
};
WDZRHorizontale.prototype.nGetLargeurClienteSB = function nGetLargeurClienteSB()
{
	return this.m_oDivScrollBar.parentNode.clientWidth;
};

// Positionne les elements graphique du champ en fonction de la position courante
WDZRHorizontale.prototype._DeplaceElements = function _DeplaceElements()
{
	// Recupere la taille totale de la zone a defiler
	var nLargeurTotale = this.nGetLargeurTotale();
	var nLargeurCliente = this.nGetLargeurCliente();
	var nLargeurClienteSB = this.nGetLargeurClienteSB();

	// Valeur qui seront calculees
	// - Deplacement horizontal de la zone au pixel
	// - Position de l'ascenseur
	// - Largeur de l'ascenseur
	var nScrollLeft = this.m_nPosition;
	var nLeft = 0;
	var nWidth = 0;

	// Si on a trop de place
	if (nLargeurTotale <= nLargeurCliente)
	{
		// Positionne la zone au pixel au centre avec une valeur negative
		nScrollLeft = parseInt(-(nLargeurCliente - nLargeurTotale) / 2, 10);

		// Positionne la barre de defilement pour prendre toute la place
		nLeft = 0;
		nWidth = nLargeurClienteSB;
	}
	else
	{
		// Il y a un defilement

		// Positionne la zone au pixel mais pas trop loin
		// En revanche on ne corrige pas la position
		if ((nLargeurTotale - nScrollLeft) < nLargeurCliente)
		{
			nScrollLeft = nLargeurTotale - nLargeurCliente;
		}

		// Positionne la barre de defilement
		nWidth = parseInt(nLargeurCliente * nLargeurClienteSB / nLargeurTotale, 10);
		// Si on arrive a un ascenseur trop petit (34 = 2 * 17)
		if (nWidth < 34)
		{
			nLargeurClienteSB -= (34 - nWidth);
		}
		nLeft = parseInt(nScrollLeft * nLargeurClienteSB / nLargeurTotale, 10);
	}

	this.m_oDivPosParent.scrollLeft = nScrollLeft;
	this.m_oDivScrollBar.style.left = nLeft + "px";
	this.m_oDivScrollBar.style.width = nWidth + "px";
};

// Fonction appele en cas de redimensionnement de la fenetre
// Evenement sur le rechargement du champ
WDZRHorizontale.prototype.OnResize = function OnResize(oEvent)
{
	// Appel de la classe de base
	WDChamp.prototype.OnResize.apply(this, arguments);

	// Force le redessin de la barre
	this._DeplaceElements();
};

// Debut du clic sur un des boutons de defilement
WDZRHorizontale.prototype.OnMouseDownBtn = function OnMouseDownBtn(oEvent, bDroite)
{
	// Normalement il n'y a plus de timer mais on le supprime au besoin
	this.OnMouseUpBtn(oEvent, bDroite, false);

	// Effectue un premier deplacement
	this.m_nDefilementV = this.ms_nPasDefilement;
	this.OnDefilementBouton(bDroite);

	// Lance le timer du deplacement
	var oThis = this;
	var bDroiteLocal = bDroite;
	this.SetInterval("OnDefilementBouton", function () { oThis.OnDefilementBouton(bDroiteLocal); }, 50);
};

// Fin du clic sur un des boutons de defilement
WDZRHorizontale.prototype.OnMouseUpBtn = function OnMouseUpBtn(oEvent, bDroite, bDepuisUp)
{
	// Probleme avec IE, sur une serie de clic rapide on recoit Down, Up, Up et pas Down, Up, Down, Up
	// Donc si on recoit un Up seul, on force un defilement
	if (bIE && bDepuisUp && !this.bGetTimeXXXExiste("OnDefilementBouton"))
	{
		// Effectue un premier deplacement
		this.m_nDefilementV = this.ms_nPasDefilement;
		this.OnDefilementBouton(bDroite);
	}
	// Normalement il y a un timer mais on le supprime uniquement s'il existe
	this.AnnuleTimeXXX("OnDefilementBouton", true);
};

// Effectue un deplacement du contenu par appuis continu sur le bouton
WDZRHorizontale.prototype.OnDefilementBouton = function OnDefilementBouton(bDroite)
{
	// Effectue le defilement avec le pas courant
	this.OnDefilement(bDroite, this.m_nDefilementV);

	// Augmente la vitesse de defilement
	if (this.m_nDefilementV <= this.ms_nPasDefilement * 10)
	{
		this.m_nDefilementV += this.ms_nPasDefilement;
	}
};

// Debut du clic sur le fond de l'ascenseur
WDZRHorizontale.prototype.OnMouseDownFondSB = function OnMouseDownFondSB(oEvent)
{
	// Normalement il n'y a plus de timer mais on le supprime au besoin
	this.bOnMouseUpFondSB(oEvent);

	// Effectue un premier deplacement
	// En memorisant la position de clic car on n'aura pas l'evenement dans le timer
	this.bOnMouseMoveFondSB(oEvent);
	this.OnDefilementFondSB();

	// Lance le timer du deplacement, la repetition est 'lente' (300-500ms)
	// Normalement le comportement est : une repetition 'lente' puis des repetitions 'rapides' (50-100ms)
	var oThis = this;
	this.SetInterval("OnDefilementFondSB", function () { oThis.OnDefilementFondSB(); }, 300);
};

// Fin du clic sur un des boutons de defilement
WDZRHorizontale.prototype.bOnMouseUpFondSB = function bOnMouseUpFondSB(oEvent)
{
	// Si on etait vraiment dans un defilement
	var bRes = this.bGetTimeXXXExiste("OnDefilementFondSB");

	// Normalement il y a un timer mais on le supprime uniquement s'il existe
	this.AnnuleTimeXXX("OnDefilementFondSB", true);

	return bRes;
};

// MAJ des coordonnees de la souris si le bouton est enfonce
WDZRHorizontale.prototype.bOnMouseMoveFondSB = function bOnMouseMoveFondSB(oEvent)
{
	this.m_nClientX = oEvent.clientX;
	// Si on est vraiment dans un defilement
	return this.bGetTimeXXXExiste("OnDefilementFondSB");
};

// Appel bOnMouseUpFondSB sur tous les champs ZRs horizontales
// Attention le this n'existe pas dans la fonction (pseudo fonction statique)
WDZRHorizontale.prototype._s_CallOnMouseUpFondSB = function _s_CallOnMouseUpFondSB(oEvent)
{
	WDZRHorizontale.prototype._s_CallOnMouseXxxFondSB(oEvent, "bOnMouseUpFondSB");
};

// Appel bOnMouseMoveFondSB sur tous les champs ZRs horizontales
// Attention le this n'existe pas dans la fonction (pseudo fonction statique)
WDZRHorizontale.prototype._s_CallOnMouseMoveFondSB = function _s_CallOnMouseMoveFondSB(oEvent)
{
	WDZRHorizontale.prototype._s_CallOnMouseXxxFondSB(oEvent, "bOnMouseMoveFondSB");
};

// Combine les appels a une methode sur tous les champs ZRs horizontales
WDZRHorizontale.prototype._s_CallOnMouseXxxFondSB = function _s_CallOnMouseMoveFondSB(oEvent, sFonction)
{
	// Appel de la fonction sur tous les champs
	var bStopPropagationFond = oAppelFonction(sFonction, [oEvent || event], function(bOldRes, bNewRes) { return (bOldRes || bNewRes); }, false);
	return clWDUtil.bStopPropagationCond(oEvent, bStopPropagationFond);
};

// Effectue un deplacement du contenu par appuis continu sur le fond de l'ascenseur
WDZRHorizontale.prototype.OnDefilementFondSB = function OnDefilementFondSB()
{
	// Verifie que le clic n'est pas sur l'ascenseur et en profite pour calculer si on est a droite ou a gauche
	var bDroite;
	var nClientXLocal = _JCCP(this.m_nClientX, this.m_oDivScrollBar, true, false);
	if (nClientXLocal < this.m_oDivScrollBar.offsetLeft)
	{
		bDroite = false;
	}
	else if (nClientXLocal > (this.m_oDivScrollBar.offsetLeft + this.m_oDivScrollBar.offsetWidth))
	{
		bDroite = true;
	}
	else
	{
		return;
	}

	// Calcule le pas : 90% de l'ecran
	var nPas = Math.ceil(this.nGetLargeurCliente() * 0.9);

	// Effectue le defilement avec le pas courant
	this.OnDefilement(bDroite, nPas);
};

// Effectue un deplacement du contenu d'une quantite fixe
WDZRHorizontale.prototype.OnDefilement = function OnDefilement(bDroite, nPas)
{
	// Effectue le defilement
	var nPosition;
	if (bDroite)
	{
		nPosition = this.m_nPosition + nPas;
	}
	else
	{
		nPosition = this.m_nPosition - nPas;
	}

	// Limite le deplacement
	nPosition = Math.min(nPosition, this.nGetLargeurTotale() - this.nGetLargeurCliente());
	nPosition = Math.max(nPosition, 0);

	this._Deplace(nPosition);
};

// Le gestionnaire de glissement de la barre en base
function WDDragScrollBar (oObjetZRHorizontale)
{
	// Si on est pas dans l'init d'un protoype
	if (oObjetZRHorizontale)
	{
		// Appel le constructeur de la classe de base
		WDDrag.prototype.constructor.apply(this, [0, 50]);

		// Sauve l'objet attache
		this.m_oObjetZRHorizontale = oObjetZRHorizontale;

		// Intercepte le click sur la barre
		HookOnXXX(oObjetZRHorizontale.m_oDivScrollBar, 'onmousedown', 'mousedown', this.m_fMouseDown);
	}
};

// Declare l'heritage
WDDragScrollBar.prototype = new WDDrag();
// Surcharge le constructeur qui a ete efface
WDDragScrollBar.prototype.constructor = WDDragScrollBar;

// Appel lors du debut d'un click pour le deplacement
// Pose les hooks
WDDragScrollBar.prototype._vbOnMouseDown = function _vbOnMouseDown(oEvent)
{
	// Appel de la classe de base : sauve la position de la souris et place les hooks
	if (!WDDrag.prototype._vbOnMouseDown.apply(this, arguments))
	{
		return false;
	}

	// Sauve la position originale
	this.m_nPosition = this.m_oObjetZRHorizontale.m_nPosition;

	return true;
};

// Appel lors du deplacement de la souris
WDDragScrollBar.prototype._vOnMouseMove = function _vOnMouseMove(oEvent)
{
	// Appel de la classe de base
	WDDrag.prototype._vOnMouseMove.apply(this, arguments);

	// Recupere la taille totale de la zone a defiler
	var nLargeurTotale = this.m_oObjetZRHorizontale.nGetLargeurTotale();
	var nLargeurCliente = this.m_oObjetZRHorizontale.nGetLargeurCliente();
	var nLargeurClienteSB = this.m_oObjetZRHorizontale.nGetLargeurClienteSB();

	// Calcule la nouvelle position
	// ATTENTION : Si on est en affichage de droite a gauche il faut inverser le deplacement de la souris car
	// les coordonnees X sont toujours de gauche a droite
	var nOffset = this.nGetOffsetPosX(oEvent);
	// Modifie en fonction de la largueur relative de la zone de defilement
	nOffset = parseInt(nOffset * nLargeurTotale / nLargeurClienteSB, 10);

	var nNouvellePosition = nOffset + this.m_nPosition;

	// Pas trop au debut
	if (nNouvellePosition < 0)
	{
		nNouvellePosition = 0;
	}

	// Pas trop a la fin
	// Si on a trop de place
	if (nLargeurTotale <= nLargeurCliente)
	{
		nNouvellePosition = 0;
	}
	else if (nNouvellePosition + nLargeurCliente > nLargeurTotale)
	{
		// Si on est trop loin
		nNouvellePosition = nLargeurTotale - nLargeurCliente;
	}

	// Fixe la nouvelle position
	this.m_oObjetZRHorizontale._Deplace(nNouvellePosition);
};

// Appel lors du relachement de la souris
WDDragScrollBar.prototype._vOnMouseUp = function _vOnMouseUp(oEvent)
{
	// Vire la position originale
	delete this.m_nPosition;

	// Appel de la classe de base
	WDDrag.prototype._vOnMouseUp.apply(this, arguments);
};

