CnC_Remastered_Collection/TIBERIANDAWN/SLIDER.CPP

406 lines
21 KiB
C++

//
// Copyright 2020 Electronic Arts Inc.
//
// TiberianDawn.DLL and RedAlert.dll and corresponding source code is free
// software: you can redistribute it and/or modify it under the terms of
// the GNU General Public License as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version.
// TiberianDawn.DLL and RedAlert.dll and corresponding source code is distributed
// in the hope that it will be useful, but with permitted additional restrictions
// under Section 7 of the GPL. See the GNU General Public License in LICENSE.TXT
// distributed with this program. You should have received a copy of the
// GNU General Public License along with permitted additional restrictions
// with this program. If not, see https://github.com/electronicarts/CnC_Remastered_Collection
/* $Header: F:\projects\c&c\vcs\code\slider.cpv 2.17 16 Oct 1995 16:51:54 JOE_BOSTIC $ */
/***********************************************************************************************
*** C O N F I D E N T I A L --- W E S T W O O D S T U D I O S ***
***********************************************************************************************
* *
* Project Name : Command & Conquer *
* *
* File Name : SLIDER.CPP *
* *
* Programmer : Joe L. Bostic *
* *
* Start Date : 01/15/95 *
* *
* Last Update : January 16, 1995 [JLB] *
* *
*---------------------------------------------------------------------------------------------*
* Functions: *
* SliderClass::Action -- Handles input processing for the slider. *
* SliderClass::Bump -- Bumps the slider one "thumb size" up or down. *
* SliderClass::Recalc_Thumb -- Recalculates the thumb pixel size and starting offset. *
* SliderClass::Set_Maximum -- Sets the maximum value for this slider. *
* SliderClass::Set_Thumb_Size -- Sets the size fo the thumb in "slider units". *
* SliderClass::Set_Value -- Sets the current thumb position for the slider. *
* SliderClass::SliderClass -- Normal constructor for a slider (with thumb) gadget. *
* SliderClass::Step -- Steps the slider one value up or down. *
* SliderClass::Draw_Thumb -- Draws the "thumb" for this slider. *
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
#include "function.h"
#include "slider.h"
/***********************************************************************************************
* SliderClass::SliderClass -- Normal constructor for a slider (with thumb) gadget. *
* *
* This is the normal constructor for the slider gadget. *
* *
* INPUT: id -- The ID number to assign to this gadget. *
* x,y -- The pixel coordinate of the upper left corner for this gadget. *
* w,h -- The width and height of the slider gadget. The slider automatically *
* adapts for horizontal or vertical operation depending on which of these *
* dimensions is greater. *
* OUTPUT: none *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
SliderClass::SliderClass(unsigned id, int x, int y, int w, int h, int belong_to_list)
: GaugeClass(id, x, y, w, h)
{
BelongToList = belong_to_list ? true : false;
PlusGadget = 0;
MinusGadget = 0;
if (!BelongToList) {
PlusGadget = new ShapeButtonClass(id, MixFileClass::Retrieve("BTN-PLUS.SHP"), X+Width+2, Y);
MinusGadget = new ShapeButtonClass(id, MixFileClass::Retrieve("BTN-MINS.SHP"), X-6, Y);
if (PlusGadget) {
PlusGadget->Make_Peer(*this);
PlusGadget->Add(*this);
PlusGadget->Flag_To_Redraw();
}
if (MinusGadget) {
MinusGadget->Make_Peer(*this);
MinusGadget->Add(*this);
MinusGadget->Flag_To_Redraw();
}
}
Set_Thumb_Size(1);
Recalc_Thumb();
/*
** Gauges have at least 2 colors, but sliders should only have one.
*/
IsColorized = 0;
}
SliderClass::~SliderClass(void)
{
if (PlusGadget) {
delete PlusGadget;
PlusGadget = 0;
}
if (MinusGadget) {
delete MinusGadget;
MinusGadget = 0;
}
}
/***********************************************************************************************
* SliderClass::Set_Maximum -- Sets the maximum value for this slider. *
* *
* This sets the maximum value that the slider can be set at. The maximum value controls *
* the size of the thumb and the resolution of the thumb's movement. *
* *
* INPUT: value -- The value to set for the slider's maximum. *
* OUTPUT: bool; Was the maximum value changed? A false indicates a set to the value it *
* is currently set to already. *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Set_Maximum(int value)
{
if (GaugeClass::Set_Maximum(value)) {
Recalc_Thumb();
return(true);
}
return(false);
}
/***********************************************************************************************
* SliderClass::Set_Thumb_Size -- Sets the size fo the thumb in "slider units". *
* *
* This routine will set the size of the thumb as it relates to the maximum value the *
* slider can achieve. This serves to display a proportionally sized thumb as well as *
* control how the slider "bumps" up or down. *
* *
* INPUT: value -- The new value of the thumb. It should never be larger than the slider *
* maximum. *
* OUTPUT: none *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
void SliderClass::Set_Thumb_Size(int value)
{
Thumb = MIN(value, MaxValue);
Thumb = MAX(Thumb, 1);
Flag_To_Redraw();
Recalc_Thumb();
}
/***********************************************************************************************
* SliderClass::Set_Value -- Sets the current thumb position for the slider. *
* *
* This routine will set the thumb position for the slider. *
* *
* INPUT: value -- The position to set the slider. This position is relative to the maximum *
* value for the slider. *
* *
* OUTPUT: bool; Was the slider thumb position changed at all? *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Set_Value(int value)
{
value = MIN(value, MaxValue-Thumb);
if (GaugeClass::Set_Value(value)) {
Recalc_Thumb();
return(true);
}
return(false);
}
/***********************************************************************************************
* SliderClass::Recalc_Thumb -- Recalculates the thumb pixel size and starting offset. *
* *
* This takes the current thumb logical size and starting value and calculates the pixel *
* size and starting offset accordingly. This function should be called whenever one of *
* these elements has changed. *
* *
* INPUT: none *
* OUTPUT: none *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
void SliderClass::Recalc_Thumb(void)
{
int length = IsHorizontal ? Width : Height;
int size = Fixed_To_Cardinal(length, Cardinal_To_Fixed(MaxValue, Thumb));
ThumbSize = MAX(size, 4);
int start = Fixed_To_Cardinal(length, Cardinal_To_Fixed(MaxValue, CurValue));
ThumbStart = MIN(start, length-ThumbSize);
}
/***********************************************************************************************
* SliderClass::Action -- Handles input processing for the slider. *
* *
* This routine is called when a qualifying input event has occured. This routine will *
* process that event and make any adjustments to the slider as necessary. *
* *
* INPUT: flags -- Flag bits that tell the input event that caused this function to *
* be called. *
* key -- Reference to the key that caused the input event. *
* OUTPUT: bool; Was the event consumed and further processing of the gadget list should be *
* aborted? *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Action(unsigned flags, KeyNumType &key)
{
/*
** Handle the mouse click in a special way. If the click was not on the thumb, then
** jump the thumb position one "step" in the appropriate direction. Otherwise, let normal
** processing take place -- the slider then "sticks" and the thumb moves according to
** mouse position.
*/
if (flags & LEFTPRESS) {
int mouse; // Mouse pixel position.
int edge; // Edge of slider.
if (IsHorizontal) {
mouse = Get_Mouse_X();
edge = X;
} else {
mouse = Get_Mouse_Y();
edge = Y;
}
edge += 1;
/*
** Clicking outside the thumb: invoke parent's Action to process flags etc,
** but turn off the event & return true so processing stops at this button.
*/
if (mouse < edge+ThumbStart) {
Bump(true);
GaugeClass::Action(0, key);
key = KN_NONE;
return(true);
} else {
if (mouse > edge+ThumbStart+ThumbSize) {
Bump(false);
GaugeClass::Action(0, key);
key = KN_NONE;
return(true);
} else {
GaugeClass::Action(flags, key);
key = KN_NONE;
return(true);
}
}
}
/*
** CHANGE GAUGECLASS::ACTION -- REMOVE (LEFTRELEASE) FROM IF STMT
*/
return(GaugeClass::Action(flags, key));
}
/***********************************************************************************************
* SliderClass::Bump -- Bumps the slider one "thumb size" up or down. *
* *
* This support function will bump the slider one "step" or the size of the thumb up or *
* down as specified. It is typically called when the slider is clicked outside of the *
* thumb region but still inside of the slider. *
* *
* INPUT: up -- Should the bump be to increase the current position? *
* OUTPUT: bool; Was the slider changed at all? A false indicates that the slider is already *
* at one end or the other. *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Bump(int up)
{
if (up) {
return(Set_Value(CurValue - Thumb));
}
return(Set_Value(CurValue + Thumb));
}
/***********************************************************************************************
* SliderClass::Step -- Steps the slider one value up or down. *
* *
* This routine will move the slider thumb one step in the direction specified. *
* *
* INPUT: up -- Should the step be up (i.e., forward)? *
* OUTPUT: bool; Was the slider changed at all? A false indicates that the slider is already *
* at one end or the other. *
* WARNINGS: none *
* HISTORY: 01/15/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Step(int up)
{
if (up) {
return(Set_Value(CurValue - 1));
}
return(Set_Value(CurValue + 1));
}
/***********************************************************************************************
* SliderClass::Draw_Thumb -- Draws the "thumb" for this slider. *
* *
* This will draw the thumb graphic for this slider. Sometimes the thumb requires special *
* drawing, thus the need for this function separate from the normal Draw_Me function. *
* *
* INPUT: none *
* OUTPUT: none *
* WARNINGS: The mouse is guaranteed to be hidden when this routine is called. *
* HISTORY: 01/16/1995 JLB : Created. *
*=============================================================================================*/
void SliderClass::Draw_Thumb(void)
{
if (IsHorizontal) {
Draw_Box(X+ThumbStart, Y, ThumbSize, Height, BOXSTYLE_GREEN_RAISED, true);
} else {
Draw_Box(X, Y+ThumbStart, Width, ThumbSize, BOXSTYLE_GREEN_RAISED, true);
}
}
/***********************************************************************************************
* SliderClass::Draw_Me -- Draws the body of the gauge. *
* *
* This routine will draw the body of the gauge if necessary. *
* *
* INPUT: forced -- Should the gauge be redrawn regardless of the current redraw flag? *
* OUTPUT: bool; Was the gauge redrawn? *
* WARNINGS: none *
* HISTORY: 01/16/1995 JLB : Created. *
*=============================================================================================*/
int SliderClass::Draw_Me(int forced)
{
if (BelongToList) {
if (ControlClass::Draw_Me(forced)) {
/*
===================== Hide the mouse =====================
*/
if (LogicPage == &SeenBuff) {
Conditional_Hide_Mouse(X, Y, X+Width, Y+Height);
}
/*
=========== Draw the body & set text color ===============
*/
Draw_Box (X, Y, Width, Height, BOXSTYLE_GREEN_DOWN, true);
// if (IsHorizontal) {
// LogicPage->Fill_Rect(X, Y+1, X+Width-1, Y+Height-2, 141);
// LogicPage->Draw_Line(X, Y, X+Width-1, Y, 140); // top
// LogicPage->Draw_Line(X, Y+Height, X+Width, Y+Height, 159); // bottom
// } else {
// LogicPage->Fill_Rect(X+1, Y, X+Width-2, Y+Height-1, 141);
// LogicPage->Draw_Line(X, Y, X, Y+Height, 140); // left
// LogicPage->Draw_Line(X+Width-1, Y, X+Width-1, Y+Height, 159); // right
// }
Draw_Thumb();
/*
=================== Display the mouse ===================
*/
if (LogicPage == &SeenBuff) {
Conditional_Show_Mouse();
}
return(true);
}
}
/*
** If it does not belong to a listbox...
*/
return(GaugeClass::Draw_Me(forced));
}
/***********************************************************************************************
* SliderClass::Peer_To_Peer -- A peer gadget was touched -- make adjustments. *
* *
* This routine is called when one of the peer gadgets (the scroll arrows or the slider) *
* was touched in some fashion. This routine will sort out whom and why and then make *
* any necessary adjustments to the list box. *
* *
* INPUT: flags -- The event flags that affected the peer gadget. *
* key -- The key value at the time of the event. *
* whom -- Which gadget is being touched. *
* OUTPUT: none *
* WARNINGS: none *
* HISTORY: 01/16/1995 JLB : Created. *
*=============================================================================================*/
void SliderClass::Peer_To_Peer(unsigned flags, KeyNumType & , ControlClass & whom)
{
if (flags & LEFTRELEASE) {
if (&whom == PlusGadget) {
Step(false);
}
if (&whom == MinusGadget) {
Step(true);
}
}
}