372 lines
17 KiB
C#
Raw Normal View History

using System;
using System.Drawing;
using static System.Math;
using static DS4Windows.Global;
using System.Diagnostics;
namespace DS4Windows
{
public class DS4LightBar
{
private readonly static byte[/* Light On duration */, /* Light Off duration */] BatteryIndicatorDurations =
{
{ 28, 252 }, // on 10% of the time at 0
{ 28, 252 },
{ 56, 224 },
{ 84, 196 },
{ 112, 168 },
{ 140, 140 },
{ 168, 112 },
{ 196, 84 },
2017-07-18 15:37:01 -05:00
{ 224, 56 }, // on 80% of the time at 80, etc.
{ 252, 28 }, // on 90% of the time at 90
{ 0, 0 } // use on 100%. 0 is for "charging" OR anything sufficiently-"charged"
};
2017-07-18 15:37:01 -05:00
static double[] counters = new double[4] { 0, 0, 0, 0 };
2017-07-18 15:37:01 -05:00
public static Stopwatch[] fadewatches = new Stopwatch[4]
{ new Stopwatch(), new Stopwatch(), new Stopwatch(), new Stopwatch() };
static bool[] fadedirection = new bool[4] { false, false, false, false };
2017-07-18 15:37:01 -05:00
static DateTime[] oldnow = new DateTime[4]
{ DateTime.UtcNow, DateTime.UtcNow, DateTime.UtcNow, DateTime.UtcNow };
public static bool[] forcelight = new bool[4] { false, false, false, false };
public static DS4Color[] forcedColor = new DS4Color[4];
public static byte[] forcedFlash = new byte[4];
internal const int PULSE_FLASH_DURATION = 2000;
2017-09-19 22:08:03 -05:00
internal const double PULSE_FLASH_SEGMENTS = PULSE_FLASH_DURATION / 40;
internal const int PULSE_CHARGING_DURATION = 4000;
internal const double PULSE_CHARGING_SEGMENTS = (PULSE_CHARGING_DURATION / 40) - 2;
2017-04-21 19:58:27 -07:00
public static void updateLightBar(DS4Device device, int deviceNum)
{
DS4Color color = new DS4Color();
2020-02-22 12:32:19 -06:00
bool useForceLight = forcelight[deviceNum];
LightbarSettingInfo lightbarSettingInfo = getLightbarSettingsInfo(deviceNum);
LightbarDS4WinInfo lightModeInfo = lightbarSettingInfo.ds4winSettings;
2020-02-22 16:09:33 -06:00
bool useLightRoutine = lightbarSettingInfo.mode == LightbarMode.DS4Win;
//bool useLightRoutine = false;
2020-02-22 16:09:33 -06:00
if (!defaultLight && !useForceLight && useLightRoutine)
{
2020-02-22 12:32:19 -06:00
if (lightModeInfo.useCustomLed)
{
2020-02-22 12:32:19 -06:00
if (lightModeInfo.ledAsBattery)
{
2020-02-22 12:32:19 -06:00
ref DS4Color fullColor = ref lightModeInfo.m_CustomLed; // ref getCustomColor(deviceNum);
ref DS4Color lowColor = ref lightModeInfo.m_LowLed; //ref getLowColor(deviceNum);
color = getTransitionedColor(ref lowColor, ref fullColor, device.getBattery());
}
else
2020-02-22 12:32:19 -06:00
color = lightModeInfo.m_CustomLed; //getCustomColor(deviceNum);
}
else
{
2020-02-22 12:32:19 -06:00
double rainbow = lightModeInfo.rainbow;// getRainbow(deviceNum);
2017-04-15 23:22:04 -07:00
if (rainbow > 0)
2017-04-21 19:58:27 -07:00
{
// Display rainbow
DateTime now = DateTime.UtcNow;
if (now >= oldnow[deviceNum] + TimeSpan.FromMilliseconds(10)) //update by the millisecond that way it's a smooth transtion
{
oldnow[deviceNum] = now;
2017-04-15 23:22:04 -07:00
if (device.isCharging())
counters[deviceNum] -= 1.5 * 3 / rainbow;
else
2017-04-15 23:22:04 -07:00
counters[deviceNum] += 1.5 * 3 / rainbow;
}
if (counters[deviceNum] < 0)
counters[deviceNum] = 180000;
2017-04-15 23:22:04 -07:00
else if (counters[deviceNum] > 180000)
counters[deviceNum] = 0;
2017-04-15 23:22:04 -07:00
2020-02-22 12:32:19 -06:00
double maxSat = lightModeInfo.maxRainbowSat; // GetMaxSatRainbow(deviceNum);
if (lightModeInfo.ledAsBattery)
{
byte useSat = (byte)(maxSat == 1.0 ?
device.getBattery() * 2.55 :
device.getBattery() * 2.55 * maxSat);
color = HuetoRGB((float)counters[deviceNum] % 360, useSat);
}
else
color = HuetoRGB((float)counters[deviceNum] % 360,
(byte)(maxSat == 1.0 ? 255 : 255 * maxSat));
}
2020-02-22 12:32:19 -06:00
else if (lightModeInfo.ledAsBattery)
{
2020-02-22 12:32:19 -06:00
ref DS4Color fullColor = ref lightModeInfo.m_Led; //ref getMainColor(deviceNum);
ref DS4Color lowColor = ref lightModeInfo.m_LowLed; //ref getLowColor(deviceNum);
color = getTransitionedColor(ref lowColor, ref fullColor, device.getBattery());
}
else
{
2017-04-21 19:58:27 -07:00
color = getMainColor(deviceNum);
}
}
2020-02-22 12:32:19 -06:00
if (device.getBattery() <= lightModeInfo.flashAt && !defaultLight && !device.isCharging())
{
2020-02-22 12:32:19 -06:00
ref DS4Color flashColor = ref lightModeInfo.m_FlashLed; //ref getFlashColor(deviceNum);
2017-04-15 23:22:04 -07:00
if (!(flashColor.red == 0 &&
flashColor.green == 0 &&
flashColor.blue == 0))
color = flashColor;
2020-02-22 12:32:19 -06:00
if (lightModeInfo.flashType == 1)
{
double ratio = 0.0;
2017-04-15 23:22:04 -07:00
if (!fadewatches[deviceNum].IsRunning)
{
bool temp = fadedirection[deviceNum];
fadedirection[deviceNum] = !temp;
fadewatches[deviceNum].Restart();
ratio = temp ? 100.0 : 0.0;
}
else
{
long elapsed = fadewatches[deviceNum].ElapsedMilliseconds;
if (fadedirection[deviceNum])
{
if (elapsed < PULSE_FLASH_DURATION)
{
elapsed = elapsed / 40;
2017-09-19 21:36:33 -05:00
ratio = 100.0 * (elapsed / PULSE_FLASH_SEGMENTS);
}
else
{
ratio = 100.0;
fadewatches[deviceNum].Stop();
}
}
else
{
if (elapsed < PULSE_FLASH_DURATION)
{
elapsed = elapsed / 40;
2017-09-19 21:36:33 -05:00
ratio = (0 - 100.0) * (elapsed / PULSE_FLASH_SEGMENTS) + 100.0;
}
else
{
ratio = 0.0;
fadewatches[deviceNum].Stop();
}
}
}
2017-04-15 23:22:04 -07:00
DS4Color tempCol = new DS4Color(0, 0, 0);
color = getTransitionedColor(ref color, ref tempCol, ratio);
}
}
2017-04-15 23:22:04 -07:00
int idleDisconnectTimeout = getIdleDisconnectTimeout(deviceNum);
2020-02-22 12:32:19 -06:00
if (idleDisconnectTimeout > 0 && lightModeInfo.ledAsBattery &&
2017-04-21 19:58:27 -07:00
(!device.isCharging() || device.getBattery() >= 100))
{
2020-02-22 12:32:19 -06:00
// Fade lightbar by idle time
TimeSpan timeratio = new TimeSpan(DateTime.UtcNow.Ticks - device.lastActive.Ticks);
double botratio = timeratio.TotalMilliseconds;
2017-04-15 23:22:04 -07:00
double topratio = TimeSpan.FromSeconds(idleDisconnectTimeout).TotalMilliseconds;
double ratio = 100.0 * (botratio / topratio), elapsed = ratio;
2017-05-27 07:05:46 -07:00
if (ratio >= 50.0 && ratio < 100.0)
{
DS4Color emptyCol = new DS4Color(0, 0, 0);
color = getTransitionedColor(ref color, ref emptyCol,
(uint)(-100.0 * (elapsed = 0.02 * (ratio - 50.0)) * (elapsed - 2.0)));
}
else if (ratio >= 100.0)
{
DS4Color emptyCol = new DS4Color(0, 0, 0);
color = getTransitionedColor(ref color, ref emptyCol, 100.0);
}
}
2017-04-15 23:22:04 -07:00
if (device.isCharging() && device.getBattery() < 100)
{
2020-02-22 12:32:19 -06:00
switch (lightModeInfo.chargingType)
{
case 1:
2017-04-15 23:22:04 -07:00
{
double ratio = 0.0;
2017-04-15 23:22:04 -07:00
if (!fadewatches[deviceNum].IsRunning)
{
bool temp = fadedirection[deviceNum];
fadedirection[deviceNum] = !temp;
fadewatches[deviceNum].Restart();
ratio = temp ? 100.0 : 0.0;
}
else
{
long elapsed = fadewatches[deviceNum].ElapsedMilliseconds;
if (fadedirection[deviceNum])
{
if (elapsed < PULSE_CHARGING_DURATION)
{
2017-09-19 22:08:03 -05:00
elapsed = elapsed / 40;
if (elapsed > PULSE_CHARGING_SEGMENTS)
elapsed = (long)PULSE_CHARGING_SEGMENTS;
2017-09-19 22:08:03 -05:00
ratio = 100.0 * (elapsed / PULSE_CHARGING_SEGMENTS);
}
else
{
ratio = 100.0;
fadewatches[deviceNum].Stop();
}
}
else
{
if (elapsed < PULSE_CHARGING_DURATION)
{
2017-09-19 22:08:03 -05:00
elapsed = elapsed / 40;
if (elapsed > PULSE_CHARGING_SEGMENTS)
elapsed = (long)PULSE_CHARGING_SEGMENTS;
2017-09-19 22:08:03 -05:00
ratio = (0 - 100.0) * (elapsed / PULSE_CHARGING_SEGMENTS) + 100.0;
}
else
{
ratio = 0.0;
fadewatches[deviceNum].Stop();
}
}
}
2017-04-15 23:22:04 -07:00
DS4Color emptyCol = new DS4Color(0, 0, 0);
color = getTransitionedColor(ref color, ref emptyCol, ratio);
break;
2017-04-15 23:22:04 -07:00
}
case 2:
2017-04-15 23:22:04 -07:00
{
2017-04-21 19:58:27 -07:00
counters[deviceNum] += 0.167;
color = HuetoRGB((float)counters[deviceNum] % 360, 255);
break;
2017-04-15 23:22:04 -07:00
}
case 3:
2017-04-15 23:22:04 -07:00
{
2020-02-22 12:32:19 -06:00
color = lightModeInfo.m_ChargingLed; //getChargingColor(deviceNum);
break;
2017-04-15 23:22:04 -07:00
}
default: break;
}
2017-04-15 23:22:04 -07:00
}
}
2020-02-22 12:32:19 -06:00
else if (useForceLight)
{
color = forcedColor[deviceNum];
2020-02-22 16:09:33 -06:00
useLightRoutine = true;
}
else if (shuttingdown)
2020-02-22 16:09:33 -06:00
{
color = new DS4Color(0, 0, 0);
2020-02-22 16:09:33 -06:00
useLightRoutine = true;
}
else if (useLightRoutine)
{
2017-04-15 23:22:04 -07:00
if (device.getConnectionType() == ConnectionType.BT)
color = new DS4Color(32, 64, 64);
else
color = new DS4Color(0, 0, 0);
}
2017-04-15 23:22:04 -07:00
2020-02-22 16:09:33 -06:00
if (useLightRoutine)
{
2020-02-22 16:09:33 -06:00
bool distanceprofile = DistanceProfiles[deviceNum] || tempprofileDistance[deviceNum];
//distanceprofile = (ProfilePath[deviceNum].ToLower().Contains("distance") || tempprofilename[deviceNum].ToLower().Contains("distance"));
if (distanceprofile && !defaultLight)
{
2020-02-22 16:09:33 -06:00
// Thing I did for Distance
float rumble = device.getLeftHeavySlowRumble() / 2.55f;
byte max = Max(color.red, Max(color.green, color.blue));
if (device.getLeftHeavySlowRumble() > 100)
{
DS4Color maxCol = new DS4Color(max, max, 0);
DS4Color redCol = new DS4Color(255, 0, 0);
color = getTransitionedColor(ref maxCol, ref redCol, rumble);
}
else
{
DS4Color maxCol = new DS4Color(max, max, 0);
DS4Color redCol = new DS4Color(255, 0, 0);
DS4Color tempCol = getTransitionedColor(ref maxCol,
ref redCol, 39.6078f);
color = getTransitionedColor(ref color, ref tempCol,
device.getLeftHeavySlowRumble());
}
}
2020-02-22 16:09:33 -06:00
DS4HapticState haptics = new DS4HapticState
{
2020-02-22 16:09:33 -06:00
LightBarColor = color
};
2017-04-22 08:03:24 -07:00
2020-02-22 16:09:33 -06:00
if (haptics.IsLightBarSet())
{
2020-02-22 16:09:33 -06:00
if (useForceLight && forcedFlash[deviceNum] > 0)
{
haptics.LightBarFlashDurationOff = haptics.LightBarFlashDurationOn = (byte)(25 - forcedFlash[deviceNum]);
haptics.LightBarExplicitlyOff = true;
}
else if (device.getBattery() <= lightModeInfo.flashAt && lightModeInfo.flashType == 0 && !defaultLight && !device.isCharging())
{
int level = device.getBattery() / 10;
if (level >= 10)
level = 10; // all values of >~100% are rendered the same
haptics.LightBarFlashDurationOn = BatteryIndicatorDurations[level, 0];
haptics.LightBarFlashDurationOff = BatteryIndicatorDurations[level, 1];
}
else if (distanceprofile && device.getLeftHeavySlowRumble() > 155) //also part of Distance
{
haptics.LightBarFlashDurationOff = haptics.LightBarFlashDurationOn = (byte)((-device.getLeftHeavySlowRumble() + 265));
haptics.LightBarExplicitlyOff = true;
}
else
{
//haptics.LightBarFlashDurationOff = haptics.LightBarFlashDurationOn = 1;
haptics.LightBarFlashDurationOff = haptics.LightBarFlashDurationOn = 0;
haptics.LightBarExplicitlyOff = true;
}
}
else
{
haptics.LightBarExplicitlyOff = true;
}
2020-02-22 16:09:33 -06:00
byte tempLightBarOnDuration = device.getLightBarOnDuration();
if (tempLightBarOnDuration != haptics.LightBarFlashDurationOn && tempLightBarOnDuration != 1 && haptics.LightBarFlashDurationOn == 0)
haptics.LightBarFlashDurationOff = haptics.LightBarFlashDurationOn = 1;
2020-02-22 16:09:33 -06:00
device.SetHapticState(ref haptics);
//device.pushHapticState(ref haptics);
}
}
2017-04-21 20:01:20 -07:00
public static bool defaultLight = false, shuttingdown = false;
public static DS4Color HuetoRGB(float hue, byte sat)
{
byte C = sat;
int X = (int)((C * (float)(1 - Math.Abs((hue / 60) % 2 - 1))));
if (0 <= hue && hue < 60)
return new DS4Color(C, (byte)X, 0);
else if (60 <= hue && hue < 120)
return new DS4Color((byte)X, C, 0);
else if (120 <= hue && hue < 180)
return new DS4Color(0, C, (byte)X);
else if (180 <= hue && hue < 240)
return new DS4Color(0, (byte)X, C);
else if (240 <= hue && hue < 300)
return new DS4Color((byte)X, 0, C);
else if (300 <= hue && hue < 360)
return new DS4Color(C, 0, (byte)X);
else
return new DS4Color(Color.Red);
}
}
}