SettingsView.js 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import React, { Component } from 'react'
  2. import { connect } from 'react-redux'
  3. import PropTypes from 'prop-types'
  4. import { setInstallationDirectory, setInstallationType, setAutoLoadMore, setOfflineMode, setTheme, setFolderStructure, setUpdateChannel, setLatestReleaseNotes } from '../actions/settingsActions'
  5. import { checkDownloadedSongs } from '../actions/queueActions'
  6. import { checkInstalledMods } from '../actions/modActions'
  7. import '../css/SettingsView.scss'
  8. import Button from './Button'
  9. import Toggle from './Toggle';
  10. const { ipcRenderer } = window.require('electron')
  11. class SettingsView extends Component {
  12. constructor(props) {
  13. super(props)
  14. this.state = {
  15. updateStatus: ''
  16. }
  17. }
  18. componentDidMount() {
  19. ipcRenderer.on('electron-updater', (_, event, message) => {
  20. if(event === 'checking-for-update' || event === 'update-not-available' || event === 'update-available' || event === 'error') {
  21. this.setState({ updateStatus: event })
  22. }
  23. })
  24. }
  25. updateValue() {
  26. switch(this.state.updateStatus) {
  27. case 'checking-for-update':
  28. return 'Checking...'
  29. case 'update-not-available':
  30. return 'Up to Date!'
  31. case 'update-available':
  32. return 'Update Available!'
  33. case 'error':
  34. return 'Update Error!'
  35. default:
  36. return 'Check for Updates'
  37. }
  38. }
  39. render() {
  40. return (
  41. <div id='settings-view'>
  42. <h1>Settings</h1>
  43. <h2>Beat Saber Installation</h2>
  44. <label htmlFor="dl-location">Installation Directory</label><br /><br />
  45. <span id="installation-directory-display"><input className="text-box" type="text" id="dl-loc-box" value={ this.props.settings.installationDirectory } disabled /></span>
  46. <input type="file" id="dl-location" webkitdirectory="" onChange={ (e) => {this.props.setInstallationDirectory(e.target.files[0].path || this.props.settings.installationDirectory)} } /><br />
  47. <label htmlFor="dl-location"><Button type="primary" onClick={ () => {} }>Choose Folder</Button></label><Button onClick={ () => {window.require('child_process').exec('start "" "' + this.props.settings.installationDirectory + '"')} }>Open Folder</Button><br /><br />
  48. <label htmlFor="installation-type">Installation Type</label><br /><br />
  49. <select id="installation-type-select" name="installation-type-select" value={ this.props.settings.installationType } onChange={ (e) => { this.props.setInstallationType(e.target.value) } }>
  50. <option value="steam">Steam</option>
  51. <option value="oculus">Oculus</option>
  52. </select>
  53. { this.props.settings.installationType === 'steam' && this.props.settings.installationDirectory.includes('Oculus') ? <><br /><br /><span style={ { fontWeight: 'bold', color: 'salmon' } }>Warning: BeatDrop has detected that you may be using the Oculus version of BeatSaber. If this is the case, please set "Installation Type" to "Oculus". Otherwise, you can ignore this message.</span><br /><br /></> : null }
  54. { this.props.settings.installationType === 'oculus' && this.props.settings.installationDirectory.includes('Steam') ? <><br /><br /><span style={ { fontWeight: 'bold', color: 'salmon' } }>Warning: BeatDrop has detected that you may be using the Steam version of BeatSaber. If this is the case, please set "Installation Type" to "Steam". Otherwise, you can ignore this message.</span><br /><br /></> : null }
  55. <hr />
  56. <h2>Song List</h2>
  57. <Toggle toggled={ this.props.settings.autoLoadMore } onToggle={ () => this.props.setAutoLoadMore(!this.props.settings.autoLoadMore) } /><label htmlFor="auto-load-more">Auto Load More</label><br /><br />
  58. <Toggle toggled={ this.props.settings.offlineMode } onToggle={ () => this.props.setOfflineMode(!this.props.settings.offlineMode) } /><label htmlFor="offline-mode">Offline Mode</label><br />
  59. <hr />
  60. <h2>Downloads</h2>
  61. <Button onClick={ this.props.checkDownloadedSongs }>Scan for Songs</Button><Button onClick={ this.props.checkInstalledMods }>{ this.props.scanningForMods ? 'Scanning...' : 'Scan for Mods' }</Button><br /><br />
  62. <label htmlFor="folder-structure-select">Folder Structure</label><br /><br />
  63. <select id="folder-structure-select" name="folder-structure-select" value={ this.props.settings.folderStructure } onChange={ (e) => { this.props.setFolderStructure(e.target.value) } }>
  64. <option value="idKey">ID/Key</option>
  65. <option value="songName">Song Name</option>
  66. </select>
  67. <hr />
  68. <h2>Theme</h2>
  69. <select id="theme-select" name="theme-select" value={ this.props.settings.theme } onChange={ (e) => { this.props.setTheme(e.target.value) } }>
  70. <option value="light">Light</option>
  71. <option value="dark">Dark</option>
  72. <option value="hc">High Contrast</option>
  73. </select>
  74. <hr />
  75. <h2>Updates</h2>
  76. <b>Current Version: </b>{ require('../../package.json').version }<br /><br />
  77. <Button onClick={ () => { this.props.setLatestReleaseNotes('0.0.0') } }>View Release Notes</Button><br /><br />
  78. <label htmlFor="update-channel-select">Update Channel</label><br /><br />
  79. <select id="update-channel-select" name="update-channel-select" value={ this.props.settings.updateChannel } onChange={ (e) => { this.props.setUpdateChannel(e.target.value) } }>
  80. <option value="latest">Stable</option>
  81. <option value="beta">Beta</option>
  82. </select><br /><br />
  83. {this.props.settings.updateChannel === 'beta' ? <><span style={ { fontWeight: 'bold', color: 'salmon' } }>Warning: Beta builds are unstable, untested and may result in unexpected crashes, loss of files and other adverse effects! By updating to a beta build, you understand and accept these risks.</span><br /><br /></> : null}
  84. <Button type={ this.state.updateStatus === 'error' ? 'destructive' : null } onClick={ () => { ipcRenderer.send('electron-updater', 'check-for-updates') } }>{this.updateValue()}</Button>
  85. <br /><br />
  86. <hr />
  87. <h2>Credits</h2>
  88. <b>BeatDrop Developer</b><br />
  89. <ul>
  90. <li>StarGazer1258</li>
  91. </ul>
  92. <br />
  93. <b>Icon and Animation Designer, BeastSaber Developer</b><br />
  94. <ul>
  95. <li>Elliotttate</li>
  96. </ul>
  97. <br />
  98. <b>BeatMods Developer</b><br />
  99. <ul>
  100. <li>vanZeben</li>
  101. </ul>
  102. <br />
  103. <b>Additional Icons Provided by</b><br />
  104. <ul>
  105. <li><a href="https://icons8.com/" onClick={ (e) => { e.preventDefault(); e.stopPropagation(); window.require('electron').shell.openExternal(e.target.href) } }>Icons8</a></li>
  106. </ul>
  107. <br />
  108. <h3>Patreon Supporters</h3>
  109. <ul>
  110. <li>
  111. <b>Wave Tier</b>
  112. <ul>
  113. <li>Shane R. Monroe</li>
  114. </ul>
  115. </li>
  116. </ul>
  117. <br /><br />
  118. </div>
  119. )
  120. }
  121. }
  122. SettingsView.propTypes = {
  123. settings: PropTypes.object.isRequired,
  124. scanningForMods: PropTypes.bool.isRequired,
  125. setInstallationDirectory: PropTypes.func.isRequired,
  126. setInstallationType: PropTypes.func.isRequired,
  127. setAutoLoadMore: PropTypes.func.isRequired,
  128. setTheme: PropTypes.func.isRequired,
  129. setFolderStructure: PropTypes.func.isRequired,
  130. setUpdateChannel: PropTypes.func.isRequired,
  131. checkDownloadedSongs: PropTypes.func.isRequired
  132. }
  133. const mapStateToProps = state => ({
  134. settings: state.settings,
  135. scanningForMods: state.mods.scanning
  136. })
  137. export default connect(mapStateToProps, { setInstallationDirectory, setInstallationType, setAutoLoadMore, setOfflineMode, setTheme, setFolderStructure, setUpdateChannel, setLatestReleaseNotes, checkDownloadedSongs, checkInstalledMods })(SettingsView)
  138. //<input type="checkbox" name="auto-refresh" id="auto-refresh" checked={this.props.settings.autoRefresh} onClick={() => this.props.setAutoLoadMore(!this.props.settings.autoLoadMore)} /><label htmlFor="auto-refresh">Refresh feed every </label><input type="number" name="auto-refresh-interval" id="auto-refresh-interval"/><label htmlFor="auto-refresh-interval"> seconds</label>