From 9e052a79b186547efd8eebc13cd96d8257352e78 Mon Sep 17 00:00:00 2001 From: Xander Frangos <33106561+xanderfrangos@users.noreply.github.com> Date: Wed, 8 Nov 2023 12:25:46 -0500 Subject: [PATCH] Improved Settings window error handling, fixed minor errors --- src/components/SafeRender.jsx | 29 ++ src/components/SettingsWindow.jsx | 752 +++++++++++++++--------------- src/css/page.scss | 6 +- src/html/settings.html | 10 + 4 files changed, 418 insertions(+), 379 deletions(-) create mode 100644 src/components/SafeRender.jsx diff --git a/src/components/SafeRender.jsx b/src/components/SafeRender.jsx new file mode 100644 index 00000000..05a1f10e --- /dev/null +++ b/src/components/SafeRender.jsx @@ -0,0 +1,29 @@ +import React from "react"; + +export default class SafeRender extends React.Component { + constructor(props) { + super(props); + this.state = { hasError: false, error: "" }; + } + + static getDerivedStateFromError(error) { + return { hasError: true, error: error }; + } + + componentDidCatch(error, errorInfo) { + console.log(error) + this.setState({ + hasError: true, + error: error, + errorInfo: errorInfo?.componentStack + }) + } + + render() { + if (this.state.hasError) { + return (
Error: {JSON.stringify(this.state.error)}
{this.state.errorInfo}
); + } + + return <>{this.props?.children}; + } + } \ No newline at end of file diff --git a/src/components/SettingsWindow.jsx b/src/components/SettingsWindow.jsx index ebd960e7..8b76c077 100644 --- a/src/components/SettingsWindow.jsx +++ b/src/components/SettingsWindow.jsx @@ -17,6 +17,7 @@ import TranslateReact from "../TranslateReact" import MonitorInfo from "./MonitorInfo" import MonitorFeatures from "./MonitorFeatures" import { SettingsOption, SettingsChild } from "./SettingsOption"; +import SafeRender from "./SafeRender"; import DefaultIcon from "../assets/tray-icons/dark/icon@4x.png" import MDL2Icon from "../assets/tray-icons/dark/mdl2@4x.png" @@ -167,7 +168,7 @@ export default class SettingsWindow extends PureComponent { componentDidMount() { window.addEventListener("monitorsUpdated", this.recievedMonitors) window.addEventListener("settingsUpdated", this.recievedSettings) - window.addEventListener("localizationUpdated", (e) => { this.setState({ languages: e.detail.languages }); console.log(e.detail); T.setLocalizationData(e.detail.desired, e.detail.default) }) + window.addEventListener("localizationUpdated", (e) => { this.setState({ languages: e.detail.languages }); T.setLocalizationData(e.detail.desired, e.detail.default)}); window.addEventListener("windowHistory", e => this.setState({ windowHistory: e.detail })) if (window.isAppX === false) { @@ -197,6 +198,7 @@ export default class SettingsWindow extends PureComponent { } + onDragEnd(result) { // dropped outside the list if (!result.destination) { @@ -242,7 +244,6 @@ export default class SettingsWindow extends PureComponent { minMaxChanged = (value, slider) => { - console.log(value, slider, this.state.remaps) const name = slider.props.monitorID let remaps = Object.assign({}, this.state.remaps) @@ -544,7 +545,6 @@ export default class SettingsWindow extends PureComponent { return this.state.adjustmentTimes.map((time, index) => { let timeElem = ( { - console.log("OUTVAL", e.target.value) this.setAdjustmentTimeValue(index, e.target.value) }} value={time.time}> ) @@ -621,7 +621,6 @@ export default class SettingsWindow extends PureComponent { this.state.adjustmentTimes[index].monitors = {} } this.state.adjustmentTimes[index].monitors[monitor.id] = value - console.log(this.state.adjustmentTimes[index].monitors) this.forceUpdate(); this.adjustmentTimesUpdated() } @@ -629,7 +628,6 @@ export default class SettingsWindow extends PureComponent { setAdjustmentTimeValue = (index, arr) => { for (let i in arr) { - console.log(arr[i]) if (i < 2 && isNaN(arr[i])) return false; } this.state.adjustmentTimes[index].time = arr @@ -657,7 +655,6 @@ export default class SettingsWindow extends PureComponent { { e.preventDefault() - console.log(e.keyCode) let key = cleanUpKeyboardKeys(e.key, e.keyCode) if (this.downKeys[key] === undefined) { this.downKeys[key] = true; @@ -679,8 +676,8 @@ export default class SettingsWindow extends PureComponent { }> { hotkey.actions?.map((action, actionIdx) => { return ( - - this.updateHotkeyAction(updatedAction, idx, actionIdx)} onDelete={() => { deleteHotkeyAction(idx, actionIdx) }} monitors={this.state.monitors} monitorNames={this.state.names} /> + + this.updateHotkeyAction(updatedAction, idx, actionIdx)} onDelete={() => { deleteHotkeyAction(idx, actionIdx) }} monitors={this.state.monitors} monitorNames={this.state.names} /> ) }) } @@ -961,436 +958,435 @@ export default class SettingsWindow extends PureComponent { } renderToggle = (setting, showText = true, textSide = "right") => { - return (
+ return (
{ this.setSetting(setting, e.target.checked) }} checked={(this.state.rawSettings?.[setting] ? true : false)} data-checked={(this.state.rawSettings?.[setting] ? true : false)} type="checkbox" />
{(this.state.rawSettings?.[setting] ? T.t("GENERIC_ON") : T.t("GENERIC_OFF"))}
) } render() { - const { rawSettings } = this.state return ( -
- -
- -
+ +
+ +
+ +
+ + +
+ +
{T.t("SETTINGS_GENERAL_TITLE")}
+ + + + + { + this.setState({ language: e.target.value }) + window.sendSettings({ language: e.target.value }) + }}> + + {this.getLanguages()} + + )} /> + + + + + + + )} /> + + this.setSetting("windowsStyle", e.target.value)}> + + + + + )} /> + +
+ +
- -
+
+ +
-
{T.t("SETTINGS_GENERAL_TITLE")}
- + +
window.sendSettings({ icon: "icon" })}> + +
+
window.sendSettings({ icon: "mdl2" })}> + +
+
window.sendSettings({ icon: "fluent" })}> + +
+
+ )} /> - + ' + T.t("SETTINGS_GENERAL_ANALYTICS_LINK") + '')} input={this.renderToggle("analytics")} /> - { - this.setState({ language: e.target.value }) - window.sendSettings({ language: e.target.value }) - }}> - - {this.getLanguages()} - - )} /> +
- - - - - - )} /> +
+
{T.t("SETTINGS_GENERAL_TROUBLESHOOTING")}
- this.setSetting("windowsStyle", e.target.value)}> - - - - - )} /> + ' + T.t("SETTINGS_GENERAL_ANALYTICS_LINK") + '')} expandable={true}> + + + + -
- -
+ this.setSetting("defaultOverlayType", e.target.value)}> + + + + + } expandable={true}> + +

+ Disabled: Do not show overlay.
+ Safe: The overlay will display over most windows, but will not force itself above apps that are marked as "always on top".
+ Aggressive: Always try to show the overlay on top of other windows. This can cause issues with exclusive fullscreen games and other fullscreen apps. It can also trigger anti-cheat in some games. +

+
+
+ + {T.t("SETTINGS_GENERAL_RESET_BUTTON")}} /> -
-
- - -
window.sendSettings({ icon: "icon" })}> - -
-
window.sendSettings({ icon: "mdl2" })}> - -
-
window.sendSettings({ icon: "fluent" })}> - -
+ + + +
+
{T.t("SETTINGS_TIME_TITLE")}
+

{T.t("SETTINGS_TIME_DESC")}

+
+ {this.getAdjustmentTimes()}
- )} /> - - ' + T.t("SETTINGS_GENERAL_ANALYTICS_LINK") + '')} input={this.renderToggle("analytics")} /> - -
- -
-
{T.t("SETTINGS_GENERAL_TROUBLESHOOTING")}
- - ' + T.t("SETTINGS_GENERAL_ANALYTICS_LINK") + '')} expandable={true}> - - - - - - this.setSetting("defaultOverlayType", e.target.value)}> - - - - - } expandable={true}> - -

- Disabled: Do not show overlay.
- Safe: The overlay will display over most windows, but will not force itself above apps that are marked as "always on top".
- Aggressive: Always try to show the overlay on top of other windows. This can cause issues with exclusive fullscreen games and other fullscreen apps. It can also trigger anti-cheat in some games. -

-
-
- - {T.t("SETTINGS_GENERAL_RESET_BUTTON")}} /> - -
-
- - -
-
{T.t("SETTINGS_TIME_TITLE")}
-

{T.t("SETTINGS_TIME_DESC")}

-
- {this.getAdjustmentTimes()} +

+ {T.t("SETTINGS_TIME_ADD")}

-

+ {T.t("SETTINGS_TIME_ADD")}

-
-
- - -
-
- - this.setSetting("adjustmentTimeLatitude", e.target.value)} style={{width: "100%", boxSizing: "border-box"}} /> -
-
- - this.setSetting("adjustmentTimeLongitude", e.target.value)} style={{width: "100%", boxSizing: "border-box"}} /> -
-
-
-
- - - this.setSetting("adjustmentTimeSpeed", e.target.value)}> - - - - - - - - } /> - -
-
-
{T.t("SETTINGS_TIME_IDLE_TITLE")}
- - + +
-
- - this.setSetting("detectIdleTimeMinutes", e.target.value)} /> +
+ + this.setSetting("adjustmentTimeLatitude", e.target.value)} style={{width: "100%", boxSizing: "border-box"}} />
-
- - this.setSetting("detectIdleTimeSeconds", e.target.value)} /> +
+ + this.setSetting("adjustmentTimeLongitude", e.target.value)} style={{width: "100%", boxSizing: "border-box"}} />
- } /> - -
- + + + + + this.setSetting("adjustmentTimeSpeed", e.target.value)}> + + + + + + + + } /> + +
+
+
{T.t("SETTINGS_TIME_IDLE_TITLE")}
+ + +
+ + this.setSetting("detectIdleTimeMinutes", e.target.value)} /> +
+
+ + this.setSetting("detectIdleTimeSeconds", e.target.value)} /> +
+
+ } /> +
+
+
- + -
-
{T.t("GENERIC_ALL_DISPLAYS")}
-
- {this.getInfoMonitors()} +
+
{T.t("GENERIC_ALL_DISPLAYS")}
+
+ {this.getInfoMonitors()} +
-
-
- - - - - - - - )} /> - - {this.getHideMonitors()} - - - - {this.getRenameMonitors()} - - - - {this.getReorderMonitors()} -
- } /> - -
+
+ + + + + + + + )} /> + + {this.getHideMonitors()} + + + + {this.getRenameMonitors()} + + + + {this.getReorderMonitors()} +
+ } /> +
+
-
-
{T.t("SETTINGS_MONITORS_NORMALIZE_TITLE")}
-

{T.t("SETTINGS_MONITORS_NORMALIZE_DESC")}

- {this.getMinMaxMonitors()} -
+
+
{T.t("SETTINGS_MONITORS_NORMALIZE_TITLE")}
+

{T.t("SETTINGS_MONITORS_NORMALIZE_DESC")}

+ {this.getMinMaxMonitors()} +
-
+ - -
-
{T.t("SETTINGS_SIDEBAR_FEATURES")}
-

{T.t("SETTINGS_FEATURES_DESCRIPTION")}

- {this.getFeaturesMonitors()} -
-
- - { - this.setState({ ddcPowerOffValue: parseInt(e.target.value) }) - window.sendSettings({ ddcPowerOffValue: parseInt(e.target.value) }) - }}> - - - - - }> - ⚠️ The "Standby" option is more likely to allow toggling the monitor on/off from Twinkle Tray. However, many monitors do not respond correctly to changing power state. Use at your own risk.} /> - -
-
+ +
+
{T.t("SETTINGS_SIDEBAR_FEATURES")}
+

{T.t("SETTINGS_FEATURES_DESCRIPTION")}

+ {this.getFeaturesMonitors()} +
+
+ + { + this.setState({ ddcPowerOffValue: parseInt(e.target.value) }) + window.sendSettings({ ddcPowerOffValue: parseInt(e.target.value) }) + }}> + + + + + }> + ⚠️ The "Standby" option is more likely to allow toggling the monitor on/off from Twinkle Tray. However, many monitors do not respond correctly to changing power state. Use at your own risk.} /> + +
+
+ + +
+
{T.t("SETTINGS_HOTKEYS_TITLE")}
+

{T.t("SETTINGS_HOTKEYS_DESC")}

+ - - +
+ +
-
- -
+
+
{T.t("SETTINGS_GENERAL_SCROLL_TITLE")}
+ + { + this.state.rawSettings.scrollShortcutAmount = parseInt(e.target.value) + window.sendSettings({ scrollShortcutAmount: parseInt(e.target.value) }) + this.forceUpdate() + }} /> + } /> + +
-
-
{T.t("SETTINGS_GENERAL_SCROLL_TITLE")}
- - { - this.state.rawSettings.scrollShortcutAmount = parseInt(e.target.value) - window.sendSettings({ scrollShortcutAmount: parseInt(e.target.value) }) - this.forceUpdate() - }} /> - } /> - -
+
+
{T.t("SETTINGS_HOTKEYS_TOD_TITLE")}
+ + + + + + + }> + + {T.t("SETTINGS_HOTKEYS_TOD_NOTE")} + { (this.state.rawSettings?.sleepAction === "ddcci" || this.state.rawSettings?.sleepAction === "ps_ddcci" ? (

⚠️ {T.t("GENERIC_DDC_WARNING")}
) : null) } +
+ } /> + +

-
-
{T.t("SETTINGS_HOTKEYS_TOD_TITLE")}
- - - - - - - }> - - {T.t("SETTINGS_HOTKEYS_TOD_NOTE")} - { (this.state.rawSettings?.sleepAction === "ddcci" || this.state.rawSettings?.sleepAction === "ps_ddcci" ? (

⚠️ {T.t("GENERIC_DDC_WARNING")}
) : null) } -
- } /> - -

+ +
- -
+
+
Profiles
+

Automatically adjust the brightness or shortcut overlay behavior depending on the focused app. You can also add profiles to the right-click menu in the system tray to quickly change the brightness to pre-defined profiles.

+
+ {this.state.rawSettings?.profiles?.map((profile, idx) => { + profile[key] = value + sendSettings({ profiles: this.state.rawSettings?.profiles }) + this.forceUpdate() + }} + onDelete={ + () => { + this.state.rawSettings?.profiles.splice(idx, 1) + sendSettingsImmediate({ profiles: this.state.rawSettings?.profiles }) + this.forceUpdate() + } + } />)} +

addNewProfile(this.state)}>+ New Profile

+
-
-
Profiles
-

Automatically adjust the brightness or shortcut overlay behavior depending on the focused app. You can also add profiles to the right-click menu in the system tray to quickly change the brightness to pre-defined profiles.

-
- {this.state.rawSettings?.profiles?.map((profile, idx) => { - profile[key] = value - sendSettings({ profiles: this.state.rawSettings?.profiles }) - this.forceUpdate() - }} - onDelete={ - () => { - this.state.rawSettings?.profiles.splice(idx, 1) - sendSettingsImmediate({ profiles: this.state.rawSettings?.profiles }) - this.forceUpdate() - } - } />)} -

addNewProfile(this.state)}>+ New Profile

- -
- + - -
-
{T.t("SETTINGS_UPDATES_TITLE")}
-

{T.h("SETTINGS_UPDATES_VERSION", '' + (window.version || "not available") + '')}

- {this.getUpdate()} -
-
- - { window.sendSettings({ branch: e.target.value }) }}> - - - - } /> -
-
- - - - -
- { window.requestMonitors(true) }}>Refresh Monitors}> - - {this.getDebugMonitors()} - - - - window.ipc.send('open-settings-file')}>Open Settings} expandable={true}> - -

{JSON.stringify(this.state.rawSettings, undefined, 2)}

-
-
- - - -
{JSON.stringify(window.allMonitors, undefined, 2)}
-
-
-
+ +
+
{T.t("SETTINGS_UPDATES_TITLE")}
+

{T.h("SETTINGS_UPDATES_VERSION", '' + (window.version || "not available") + '')}

+ {this.getUpdate()} +
+
+ + { window.sendSettings({ branch: e.target.value }) }}> + + + + } /> +
+
+ + + + +
+ { window.requestMonitors(true) }}>Refresh Monitors}> + + {this.getDebugMonitors()} + + + + window.ipc.send('open-settings-file')}>Open Settings} expandable={true}> + +

{JSON.stringify(this.state.rawSettings, undefined, 2)}

+
+
+ + + +
{JSON.stringify(window.allMonitors, undefined, 2)}
+
+
+
-
-
Other
- - - - - this.setSetting("udpPortStart", e.target.value)} />} /> - - - - - - - - +
+
Other
+ + + + + this.setSetting("udpPortStart", e.target.value)} />} /> + + + + + + + + -
- +
+
+
-
-
-
-
-
-

Enter the VCP code for the feature you would like to add to your display. Please note that Twinkle Tray does not validate if your display actually supports this VCP code. Use at your own risk.

- - this.setState({ addFeatureValue: e.target.value })} onKeyUp={e => { - console.log(e) - if (e.which === 13 && this.state.addFeatureValue) { - // Enter - this.addFeatureOKRef.current.click() - } else if (e.which === 27) { - // Escape - this.addFeatureCancelRef.current.click() +
+
+
+
+

Enter the VCP code for the feature you would like to add to your display. Please note that Twinkle Tray does not validate if your display actually supports this VCP code. Use at your own risk.

+ + this.setState({ addFeatureValue: e.target.value })} onKeyUp={e => { + if (e.which === 13 && this.state.addFeatureValue) { + // Enter + this.addFeatureOKRef.current.click() + } else if (e.which === 27) { + // Escape + this.addFeatureCancelRef.current.click() + } + }} /> +
+
+
+

This feature is already active.

+
+
+ this.setState({ showAddFeatureOverlay: false })} /> + { + let isActive = false + const vcp = this.state.addFeatureValue.toUpperCase() + try { + isActive = this.state.rawSettings.monitorFeatures[this.state.addFeatureMonitor][vcp]; + } catch (e) { } + if (isActive) { + this.setState({ addFeatureError: true }) + } else { + this.setState({ showAddFeatureOverlay: false }) + this.toggleFeature(this.state.addFeatureMonitor, vcp) } + }} />
-
-

This feature is already active.

-
-
- this.setState({ showAddFeatureOverlay: false })} /> - { - let isActive = false - const vcp = this.state.addFeatureValue.toUpperCase() - try { - isActive = this.state.rawSettings.monitorFeatures[this.state.addFeatureMonitor][vcp]; - } catch (e) { } - if (isActive) { - this.setState({ addFeatureError: true }) - } else { - this.setState({ showAddFeatureOverlay: false }) - this.toggleFeature(this.state.addFeatureMonitor, vcp) - } - - }} /> -
-
- -
+
+ ); } } @@ -1482,7 +1478,7 @@ function AppProfile(props) { function SettingsPage(props) { if (props.current === props.id) { return ( -
{props.children}
+
{props.children}
) } return null @@ -1567,7 +1563,7 @@ function ActionItem(props) { {action.values?.map((value, idx2) => { return ( -
+
{ const value = e.target.value diff --git a/src/css/page.scss b/src/css/page.scss index 5b7f0c65..ece92fb9 100644 --- a/src/css/page.scss +++ b/src/css/page.scss @@ -429,7 +429,7 @@ select { margin-left: 8px; } - &[data-textSide="left"] { + &[data-textside="left"] { input { order: 2; } @@ -1139,4 +1139,8 @@ body[data-is-win11="true"] { .win10only { display: none; } +} + +.try-catch-failure { + user-select: auto; } \ No newline at end of file diff --git a/src/html/settings.html b/src/html/settings.html index e45d6c76..552ae5dc 100644 --- a/src/html/settings.html +++ b/src/html/settings.html @@ -11,6 +11,16 @@ +