template<class T = int> struct slider_point
{
slider_point(){}
slider_point(T x, T y): X(x), Y(y){}
~slider_point(){}
T X;
T Y;
};
template<class T = int> struct slider_values
{
slider_values(){}
slider_values(slider_point<T> min_values, slider_point<T> max_values, slider_point<T> default_values)
:minimum(min_values),maximum(max_values),default(default_values),current(default_values)
{}
~slider_values(){}
slider_point<T> minimum;
slider_point<T> maximum;
slider_point<T> default;
slider_point<T> current;
};
struct slider_box
{
slider_box(){}
~slider_box(){}
slider_box(slider_point<float> position, slider_point<float> size, DWORD color_inner, DWORD color_outline):
pos(position), size(size), color_inner(color_inner), color_outline(color_outline)
{}
slider_point<float> pos;
slider_point<float> size;
DWORD color_inner;
DWORD color_outline;
void draw()
{
return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,color_outline,color_inner);
}
};
struct slider_button
{
slider_button(){}
slider_button(slider_point<float> size, slider_point<float> pos, button_color _appearance = default_appearance, button_color _highlight = default_highlighted, button_color _mousedown = default_mousedown)
: size(size), pos(pos), appearance(_appearance), highlight(_highlight), mousedown(_mousedown)
{}
~slider_button(){}
button_color appearance;
button_color highlight;
button_color mousedown;
slider_point<float> size;
slider_point<float> pos;
bool IsPointOnThis(slider_point<float> point)
{
return IsPointInArea(point.X,point.Y,pos.X,pos.Y,size.X,size.Y);
}
bool IsMouseOnThis()
{
return IsPointOnThis(slider_point<float>((float)cursorPos.x,(float)cursorPos.y));
}
void SetPos(slider_point<float> point)
{
pos = point;
}
void SetCenter(slider_point<float> point)
{
pos.X = (point.X - (size.X/2.0f));
pos.Y = (point.Y - (size.Y/2.0f));
}
void SetTopLeft(slider_point<float> point)
{
pos = point;
}
void SetTopRight(slider_point<float> point)
{
pos.Y = point.Y;
pos.X = point.X-size.X;
}
void SetBottomLeft(slider_point<float> point)
{
pos.X = point.X;
pos.Y = point.Y-size.Y;
}
void SetBottomRight(slider_point<float> point)
{
pos.X = point.X-size.X;
pos.Y = point.Y-size.Y;
}
float GetLeft()
{
return pos.X;
}
float GetRight()
{
return pos.X+size.X;
}
float GetTop()
{
return pos.Y;
}
float GetBottom()
{
return pos.Y+size.Y;
}
slider_point<float> GetCenter()
{
return slider_point<float>(pos.X+(size.X/2.0f),pos.Y+(size.Y/2.0f));
}
void draw()
{
if(!IsPointInArea(cursorPos.x,cursorPos.y,pos.X,pos.Y,size.X,size.Y))
return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,appearance.border,appearance.button);
if(Keys(VK_LBUTTON).Down)
return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,mousedown.border,mousedown.button);
return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,highlight.border,highlight.button);
}
};
//not only left/right or up/down, but also both at once!
template<class T = int> class slider
{
private:
slider_values<T> data;
bool Dragging;
slider_point<float> DragMouseStart;
slider_point<float> DragSliderStart;
public:
bool Show;
bool LockXMovement;
bool LockYMovement;
slider_box box;
slider_button sliding;
slider(){Show = true;Dragging=false;}
slider(slider_point<float> pos, slider_point<float> size, slider_point<T> minimum, slider_point<T> maximum, slider_point<T> default, slider_point<float> slider_size, DWORD _color_inner, DWORD _color_outline, bool LockX = false, bool LockY = false, button_color _appearance = default_appearance, button_color _highlighted = default_highlighted, button_color _mousedown = default_mousedown)
: data(minimum,maximum,default),LockXMovement(LockX),LockYMovement(LockY),box(pos,size,_color_inner,_color_outline), sliding(slider_size,pos,_appearance,_highlighted,_mousedown)
{
Dragging = false;
if(sliding.size.Y > box.size.Y)
{
sliding.size.Y = box.size.Y;
LockYMovement = true;
}
if(sliding.size.X > box.size.X)
{
sliding.size.X = box.size.X;
LockXMovement = true;
}
sliding.SetCenter( slider_point<float>( pos.X+( (float)(((float)(default.X-minimum.X))/((float)(maximum.X-minimum.X))) *size.X), pos.Y+( (float)(((float)(default.Y-minimum.Y))/((float)(maximum.Y-minimum.Y))) *size.Y)));
if(sliding.pos.X < box.pos.X)
sliding.pos = slider_point<float>( box.pos.X , sliding.pos.Y );
if(sliding.GetRight() > (box.pos.X + box.size.X))
sliding.SetTopRight(slider_point<float>( box.pos.X+box.size.X , sliding.pos.Y ));
if(sliding.pos.Y < box.pos.Y)
sliding.pos = slider_point<float>( sliding.pos.X , box.pos.Y );
if(sliding.GetBottom() > (box.pos.Y+box.size.Y))
sliding.SetBottomLeft(slider_point<float>( sliding.pos.X , box.pos.Y+box.size.Y ));
Show = true;
}
slider_values<T> GetValue()
{
return data.current;
}
void Process()
{
if(Dragging)
{
if(Keys(VK_LBUTTON).Up)
{
Dragging = false;
}
if(!LockXMovement)
{
float mousepos = (float)cursorPos.x;
sliding.pos = slider_point<float>( DragSliderStart.X+(mousepos-DragMouseStart.X) , sliding.pos.Y );
if(sliding.pos.X < box.pos.X)
sliding.pos = slider_point<float>( box.pos.X , sliding.pos.Y );
if(sliding.GetRight() > (box.pos.X+box.size.X))
sliding.SetTopRight(slider_point<float>( box.pos.X+box.size.X , sliding.pos.Y ));
data.current.X = data.minimum.X + (T)((float)(data.maximum.X-data.minimum.X) * ((sliding.pos.X-box.pos.X)/(box.size.X-sliding.size.X)));
}
if(!LockYMovement)
{
float mousepos = (float)cursorPos.y;
sliding.pos = slider_point<float>( sliding.pos.X , DragSliderStart.Y+(mousepos-DragMouseStart.Y) );
if(sliding.pos.Y < box.pos.Y)
sliding.pos = slider_point<float>( sliding.pos.X , box.pos.Y );
if(sliding.GetBottom() > (box.pos.Y+box.size.Y))
sliding.SetBottomLeft(slider_point<float>( sliding.pos.X , box.pos.Y+box.size.Y ));
data.current.Y = data.minimum.Y + (T)((float)(data.maximum.Y-data.minimum.Y) * ((sliding.pos.Y-box.pos.Y)/(box.size.Y-sliding.size.Y)));
}
DirectXFont::Access(4)->Print(0.0f,50.0f,0xFF00FF00,string_format("[%04d,%04d]",(int)data.current.X,(int)data.current.Y).c_str());
}else
if(Keys(VK_LBUTTON).Pressed)
{
if(sliding.IsMouseOnThis())
{
Dragging = true;
DragMouseStart.X = (float)cursorPos.x;
DragMouseStart.Y = (float)cursorPos.y;
DragSliderStart = sliding.pos;
}
}
box.draw();
sliding.draw();
}
~slider()
{
}
};
template<class T = int> struct slider_point
{
    slider_point(){}
	slider_point(T x, T y): X(x), Y(y){}
	~slider_point(){}
	T X;
	T Y;
};

template<class T = int> struct slider_values
{
	slider_values(){}
	slider_values(slider_point<T> min_values, slider_point<T> max_values, slider_point<T> default_values)
		:minimum(min_values),maximum(max_values),default(default_values),current(default_values)
	{}
	~slider_values(){}
	slider_point<T> minimum;
	slider_point<T> maximum;
	slider_point<T> default;
	slider_point<T> current;
};

struct slider_box
{
	slider_box(){}
	~slider_box(){}
	slider_box(slider_point<float> position, slider_point<float> size, DWORD color_inner, DWORD color_outline):
		pos(position), size(size), color_inner(color_inner), color_outline(color_outline)
	{}
	slider_point<float> pos;
	slider_point<float> size;
	DWORD color_inner;
	DWORD color_outline;
	void draw()
	{
		return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,color_outline,color_inner);
	}
};

struct slider_button
{
	slider_button(){}
	slider_button(slider_point<float> size, slider_point<float> pos, button_color _appearance = default_appearance, button_color _highlight = default_highlighted, button_color _mousedown = default_mousedown)
		: size(size), pos(pos), appearance(_appearance), highlight(_highlight), mousedown(_mousedown)
	{}
	~slider_button(){}
	button_color appearance;
	button_color highlight;
	button_color mousedown;
	slider_point<float> size;
	slider_point<float> pos;
	bool IsPointOnThis(slider_point<float> point)
	{
		return IsPointInArea(point.X,point.Y,pos.X,pos.Y,size.X,size.Y);
	}
	bool IsMouseOnThis()
	{
		return IsPointOnThis(slider_point<float>((float)cursorPos.x,(float)cursorPos.y));
	}
	void SetPos(slider_point<float> point)
	{
		pos = point;
	}
	void SetCenter(slider_point<float> point)
	{
		pos.X = (point.X - (size.X/2.0f));
		pos.Y = (point.Y - (size.Y/2.0f));
	}
	void SetTopLeft(slider_point<float> point)
	{
		pos = point;
	}
	void SetTopRight(slider_point<float> point)
	{
		pos.Y = point.Y;
		pos.X = point.X-size.X;
	}
	void SetBottomLeft(slider_point<float> point)
	{
		pos.X = point.X;
		pos.Y = point.Y-size.Y;
	}
	void SetBottomRight(slider_point<float> point)
	{
		pos.X = point.X-size.X;
		pos.Y = point.Y-size.Y;
	}
	float GetLeft()
	{
		return pos.X;
	}
	float GetRight()
	{
		return pos.X+size.X;
	}
	float GetTop()
	{
		return pos.Y;
	}
	float GetBottom()
	{
		return pos.Y+size.Y;
	}
	slider_point<float> GetCenter()
	{
		return slider_point<float>(pos.X+(size.X/2.0f),pos.Y+(size.Y/2.0f));
	}
	void draw()
	{
		if(!IsPointInArea(cursorPos.x,cursorPos.y,pos.X,pos.Y,size.X,size.Y))
			return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,appearance.border,appearance.button);
		if(Keys(VK_LBUTTON).Down)
			return render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,mousedown.border,mousedown.button);
		return  render->D3DBoxBorder(pos.X,pos.Y,size.X,size.Y,highlight.border,highlight.button);
	}
};

//not only left/right or up/down, but also both at once!
template<class T = int> class slider
{
private:
	slider_values<T> data;
	bool Dragging;	
	slider_point<float> DragMouseStart;
	slider_point<float> DragSliderStart;
public:
	bool Show;
	bool LockXMovement;
	bool LockYMovement;
	slider_box box;
	slider_button sliding;
	slider(){Show = true;Dragging=false;}
	slider(slider_point<float> pos, slider_point<float> size, slider_point<T> minimum, slider_point<T> maximum, slider_point<T> default, slider_point<float> slider_size, DWORD _color_inner, DWORD _color_outline, bool LockX = false, bool LockY = false, button_color _appearance = default_appearance, button_color _highlighted = default_highlighted, button_color _mousedown = default_mousedown)
		: data(minimum,maximum,default),LockXMovement(LockX),LockYMovement(LockY),box(pos,size,_color_inner,_color_outline), sliding(slider_size,pos,_appearance,_highlighted,_mousedown)
	{
		Dragging = false;
		if(sliding.size.Y > box.size.Y)
		{
			sliding.size.Y = box.size.Y;
			LockYMovement = true;
		}
		if(sliding.size.X > box.size.X)
		{
			sliding.size.X = box.size.X;
			LockXMovement = true;
		}
		
			
		sliding.SetCenter( slider_point<float>( pos.X+( (float)(((float)(default.X-minimum.X))/((float)(maximum.X-minimum.X))) *size.X), pos.Y+( (float)(((float)(default.Y-minimum.Y))/((float)(maximum.Y-minimum.Y))) *size.Y)));
		if(sliding.pos.X < box.pos.X)
			sliding.pos = slider_point<float>( box.pos.X , sliding.pos.Y );
		if(sliding.GetRight() > (box.pos.X + box.size.X))
			sliding.SetTopRight(slider_point<float>( box.pos.X+box.size.X , sliding.pos.Y ));
		if(sliding.pos.Y < box.pos.Y)
			sliding.pos = slider_point<float>( sliding.pos.X , box.pos.Y );
		if(sliding.GetBottom() > (box.pos.Y+box.size.Y))
			sliding.SetBottomLeft(slider_point<float>( sliding.pos.X , box.pos.Y+box.size.Y ));
		Show = true;
	}
	slider_values<T> GetValue()
	{
		return data.current;
	}
	void Process()
	{
		if(Dragging)
		{
			if(Keys(VK_LBUTTON).Up)
			{
				Dragging = false;
			}
			if(!LockXMovement)
			{
				float mousepos = (float)cursorPos.x;
				sliding.pos = slider_point<float>( DragSliderStart.X+(mousepos-DragMouseStart.X) , sliding.pos.Y );
				if(sliding.pos.X < box.pos.X)
					sliding.pos = slider_point<float>( box.pos.X , sliding.pos.Y );
				if(sliding.GetRight() > (box.pos.X+box.size.X))
					sliding.SetTopRight(slider_point<float>( box.pos.X+box.size.X , sliding.pos.Y ));
				data.current.X = data.minimum.X + (T)((float)(data.maximum.X-data.minimum.X) * ((sliding.pos.X-box.pos.X)/(box.size.X-sliding.size.X)));
			}
			if(!LockYMovement)
			{
				float mousepos = (float)cursorPos.y;
				sliding.pos = slider_point<float>( sliding.pos.X , DragSliderStart.Y+(mousepos-DragMouseStart.Y) );
				if(sliding.pos.Y < box.pos.Y)
					sliding.pos = slider_point<float>( sliding.pos.X , box.pos.Y );
				if(sliding.GetBottom() > (box.pos.Y+box.size.Y))
					sliding.SetBottomLeft(slider_point<float>( sliding.pos.X , box.pos.Y+box.size.Y ));
				data.current.Y = data.minimum.Y + (T)((float)(data.maximum.Y-data.minimum.Y) * ((sliding.pos.Y-box.pos.Y)/(box.size.Y-sliding.size.Y)));
			}
			DirectXFont::Access(4)->Print(0.0f,50.0f,0xFF00FF00,string_format("[%04d,%04d]",(int)data.current.X,(int)data.current.Y).c_str());
		}else
		if(Keys(VK_LBUTTON).Pressed)
		{
			if(sliding.IsMouseOnThis())
			{
				Dragging = true;
				DragMouseStart.X = (float)cursorPos.x;
				DragMouseStart.Y = (float)cursorPos.y;
				DragSliderStart = sliding.pos;
			}
		}
		box.draw();
		sliding.draw();
	}
	~slider()
	{

	}
};
