settingsModalView.html 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. <modal id="settings" status="default" icon="fas fa-cog" heading="{{'Settings' | translate}}" large="yes" closeable="yes">
  2. <div class="modal-body">
  3. <form role="form" name="settingsEditor">
  4. <ul class="nav nav-tabs">
  5. <li class="active"><a data-toggle="tab" href="#settings-general"><span class="fas fa-cog"></span> <span translate>General</span></a></li>
  6. <li><a data-toggle="tab" href="#settings-gui"><span class="fas fa-desktop"></span> <span translate>GUI</span></a></li>
  7. <li><a data-toggle="tab" href="#settings-connections"><span class="fas fa-sitemap"></span> <span translate>Connections</span></a></li>
  8. <li>
  9. <a data-toggle="tab" href="#settings-ignored-devices">
  10. <span class="fas fa-laptop"></span>
  11. &nbsp;
  12. <span translate>Ignored Devices</span>
  13. &nbsp;
  14. <span class="badge">{{ tmpRemoteIgnoredDevices.length }}</span>
  15. </a>
  16. </li>
  17. <li>
  18. <a data-toggle="tab" href="#settings-ignored-folders">
  19. <span class="fas fa-folder"></span>
  20. &nbsp;
  21. <span translate>Ignored Folders</span>
  22. &nbsp;
  23. <span class="badge">{{ ignoredFoldersCountTmpConfig() }}</span>
  24. </a>
  25. </li>
  26. </ul>
  27. <div class="tab-content">
  28. <div id="settings-general" class="tab-pane in active">
  29. <div class="form-group">
  30. <label translate for="DeviceName">Device Name</label>
  31. <input id="DeviceName" class="form-control" type="text" ng-model="tmpOptions.deviceName" />
  32. </div>
  33. <div class="row">
  34. <div class="col-md-6">
  35. <div class="form-horizontal">
  36. <div class="form-group" ng-class="{'has-error': settingsEditor.minHomeDiskFree.$invalid && settingsEditor.minHomeDiskFree.$dirty}">
  37. <label class="col-xs-12" for="minHomeDiskFree"><span translate>Minimum Free Disk Space</span></label><br />
  38. <div class="col-xs-9"><input name="minHomeDiskFree" id="minHomeDiskFree" class="form-control" type="number" ng-model="tmpOptions.minHomeDiskFree.value" required="" aria-required="true" min="0" step="0.01" /></div>
  39. <div class="col-xs-3"><select class="col-sm-3 form-control" ng-model="tmpOptions.minHomeDiskFree.unit">
  40. <option value="%">%</option>
  41. <option value="kB">kB</option>
  42. <option value="MB">MB</option>
  43. <option value="GB">GB</option>
  44. <option value="TB">TB</option>
  45. </select></div>
  46. <p class="col-xs-12 help-block">
  47. <span translate ng-show="settingsEditor.minHomeDiskFree.$invalid">Enter a non-negative number (e.g., "2.35") and select a unit. Percentages are as part of the total disk size.</span>
  48. <span translate ng-hide="settingsEditor.minHomeDiskFree.$invalid">This setting controls the free space required on the home (i.e., index database) disk.</span>
  49. </p>
  50. </div>
  51. </div>
  52. </div>
  53. <div class="col-md-6">
  54. <div class="form-group">
  55. <label translate>API Key</label>
  56. <div class="input-group">
  57. <input type="text" readonly class="text-monospace form-control" value="{{tmpGUI.apiKey || '-'}}" />
  58. <span class="input-group-btn">
  59. <button type="button" class="btn btn-default btn-secondary" ng-click="setAPIKey(tmpGUI)">
  60. <span class="fas fa-redo"></span>&nbsp;<span translate>Generate</span>
  61. </button>
  62. </span>
  63. </div>
  64. </div>
  65. </div>
  66. </div>
  67. <div class="row">
  68. <div class="col-md-6">
  69. <div class="form-group">
  70. <label translate for="urVersion">Anonymous Usage Reporting</label> (<a href="" translate data-toggle="modal" data-target="#urPreview">Preview</a>)
  71. <div ng-if="tmpOptions.upgrades != 'candidate' && !version.isCandidate">
  72. <select class="form-control" id="urVersion" ng-model="tmpOptions._urAcceptedStr">
  73. <option ng-repeat="n in urVersions()" value="{{n}}">{{'Version' | translate}} {{n}}</option>
  74. <!-- 1 does not exist, as we did not support incremental formats back then. -->
  75. <option value="0" translate>Undecided (will prompt)</option>
  76. <option value="-1" translate>Disabled</option>
  77. </select>
  78. </div>
  79. <p class="help-block" ng-if="tmpOptions.upgrades == 'candidate' || version.isCandidate">
  80. <span translate>Usage reporting is always enabled for candidate releases.</span>
  81. </p>
  82. </div>
  83. </div>
  84. <div class="col-md-6">
  85. <div class="form-group">
  86. <label translate>Automatic upgrades</label>&emsp;<a href="{{docsURL('users/releases')}}" target="_blank"><span class="fas fa-question-circle"></span>&nbsp;<span translate>Help</span></a>
  87. <select class="form-control" ng-model="tmpOptions.upgrades" ng-if="upgradeInfo">
  88. <option ng-if="!version.isCandidate" value="none" translate>No upgrades</option>
  89. <option value="stable" translate>Stable releases only</option>
  90. <option value="candidate" translate>Stable releases and release candidates</option>
  91. </select>
  92. <p class="help-block" ng-if="!upgradeInfo">
  93. <span translate>Unavailable/Disabled by administrator or maintainer</span>
  94. </p>
  95. <p class="help-block" ng-if="version.isCandidate && upgradeInfo">
  96. <span translate>Automatic upgrades are always enabled for candidate releases.</span>
  97. </p>
  98. </div>
  99. </div>
  100. </div>
  101. <div>
  102. <label translate>Default Configuration</label>
  103. <p>
  104. <button type="button" class="btn btn-default btn-secondary" ng-click="editFolderDefaults()">
  105. <span translate>Edit Folder Defaults</span>
  106. </button>
  107. <button type="button" class="btn btn-default btn-secondary" ng-click="editDeviceDefaults()">
  108. <span translate>Edit Device Defaults</span>
  109. </button>
  110. </p>
  111. </div>
  112. </div>
  113. <div id="settings-gui" class="tab-pane">
  114. <div class="form-group" ng-class="{'has-error': settingsEditor.Address.$invalid && settingsEditor.Address.$dirty}">
  115. <label translate for="Address">GUI Listen Address</label>&emsp;<a href="{{docsURL('users/guilisten')}}" target="_blank"><span class="fas fa-question-circle"></span>&nbsp;<span translate>Help</span></a>
  116. <p class="text-warning" ng-show="system.guiAddressOverridden">
  117. <span class="fas fa-exclamation-triangle"></span>
  118. <span translate>The GUI address is overridden by startup options. Changes here will not take effect while the override is in place.</span>
  119. </p>
  120. <input id="Address" name="Address" class="form-control" type="text" ng-model="tmpGUI.address" ng-pattern="/^(/.*)|(.*:0*((102[4-9])|(10[3-9][0-9])|(1[1-9][0-9][0-9])|([2-9][0-9][0-9][0-9])|([1-6]\d{4})))$/" />
  121. <p class="help-block" ng-show="settingsEditor.Address.$invalid" translate>
  122. Enter a non-privileged port number (1024 - 65535).
  123. </p>
  124. </div>
  125. <div class="row">
  126. <div class="col-md-6">
  127. <div class="form-group">
  128. <label translate for="User">GUI Authentication User</label>
  129. <input id="User" class="form-control" type="text" ng-model="tmpGUI.user" />
  130. </div>
  131. </div>
  132. <div class="col-md-6">
  133. <div class="form-group">
  134. <label translate for="Password">GUI Authentication Password</label>
  135. <input id="Password" class="form-control" type="password" ng-model="tmpGUI.password" ng-trim="false" />
  136. </div>
  137. </div>
  138. </div>
  139. <div class="row">
  140. <div class="col-md-6">
  141. <div class="form-group">
  142. <div class="checkbox">
  143. <label>
  144. <input id="UseTLS" type="checkbox" ng-model="tmpGUI.useTLS" /> <span translate>Use HTTPS for GUI</span>
  145. </label>
  146. </div>
  147. </div>
  148. </div>
  149. <div class="col-md-6">
  150. <div class="form-group">
  151. <div class="checkbox">
  152. <label>
  153. <input id="StartBrowser" type="checkbox" ng-model="tmpOptions.startBrowser" /> <span translate>Start Browser</span>
  154. </label>
  155. </div>
  156. </div>
  157. </div>
  158. </div>
  159. <div class="row">
  160. <div class="col-md-6">
  161. <div class="form-group">
  162. <label translate>GUI Theme</label>
  163. <select class="form-control" ng-model="tmpGUI.theme" ng-if="themes.length > 1">
  164. <option ng-repeat="theme in themes.sort()" value="{{ theme }}">
  165. {{ themeName(theme) }}
  166. </option>
  167. </select>
  168. <p class="help-block" ng-if="themes.length < 2">
  169. <span translate>Unavailable</span>
  170. </p>
  171. </div>
  172. </div>
  173. <div class="col-md-6">
  174. <div ng-if="isUnixAddress(tmpGUI.address)" class="form-group" ng-class="{'has-error': settingsEditor.UnixSocketPermissions.$invalid && settingsEditor.UnixSocketPermissions.$dirty}">
  175. <label translate>UNIX Permissions</label>
  176. <input id="UnixSocketPermissions" name="UnixSocketPermissions" class="form-control" type="text" ng-model="tmpGUI.unixSocketPermissions" ng-pattern="/^0?[0-7]{0,3}$/" />
  177. <p class="help-block" ng-show="settingsEditor.UnixSocketPermissions.$invalid" translate>
  178. Enter up to three octal digits.
  179. </p>
  180. </div>
  181. </div>
  182. </div>
  183. </div>
  184. <div id="settings-connections" class="tab-pane">
  185. <div class="form-group">
  186. <label translate for="ListenAddressesStr">Sync Protocol Listen Addresses</label>&emsp;<a href="{{docsURL('users/config#listen-addresses')}}" target="_blank"><span class="fas fa-question-circle"></span>&nbsp;<span translate>Help</span></a>
  187. <input id="ListenAddressesStr" class="form-control" type="text" ng-model="tmpOptions._listenAddressesStr" />
  188. </div>
  189. <div class="row">
  190. <div class="col-md-6">
  191. <div class="form-group" ng-class="{'has-error': settingsEditor.MaxRecvKbps.$invalid && settingsEditor.MaxRecvKbps.$dirty}">
  192. <label translate for="MaxRecvKbps">Incoming Rate Limit (KiB/s)</label>
  193. <input id="MaxRecvKbps" name="MaxRecvKbps" class="form-control" type="number" ng-model="tmpOptions.maxRecvKbps" min="0" />
  194. <p class="help-block">
  195. <span translate ng-if="settingsEditor.MaxRecvKbps.$error.min && settingsEditor.MaxRecvKbps.$dirty">The rate limit must be a non-negative number (0: no limit)</span>
  196. </p>
  197. </div>
  198. </div>
  199. <div class="col-md-6">
  200. <div class="form-group" ng-class="{'has-error': settingsEditor.MaxSendKbps.$invalid && settingsEditor.MaxSendKbps.$dirty}">
  201. <label translate for="MaxSendKbps">Outgoing Rate Limit (KiB/s)</label>
  202. <input id="MaxSendKbps" name="MaxSendKbps" class="form-control" type="number" ng-model="tmpOptions.maxSendKbps" min="0" />
  203. <p class="help-block">
  204. <span translate ng-if="settingsEditor.MaxSendKbps.$error.min && settingsEditor.MaxSendKbps.$dirty">The rate limit must be a non-negative number (0: no limit)</span>
  205. </p>
  206. </div>
  207. </div>
  208. </div>
  209. <div class="row">
  210. <div class="col-md-6">
  211. <div class="form-group">
  212. <div class="checkbox">
  213. <label>
  214. <input id="NATEnabled" type="checkbox" ng-model="tmpOptions.natEnabled" /> <span translate>Enable NAT traversal</span>
  215. </label>
  216. </div>
  217. </div>
  218. </div>
  219. <div class="col-md-6">
  220. <div class="form-group">
  221. <div class="checkbox">
  222. <label>
  223. <input id="LocalAnnEnabled" type="checkbox" ng-model="tmpOptions.localAnnounceEnabled" /> <span translate>Local Discovery</span>
  224. </label>
  225. </div>
  226. </div>
  227. </div>
  228. </div>
  229. <div class="row">
  230. <div class="col-md-6">
  231. <div class="form-group">
  232. <div class="checkbox">
  233. <label>
  234. <input id="GlobalAnnEnabled" type="checkbox" ng-model="tmpOptions.globalAnnounceEnabled" /> <span translate>Global Discovery</span>
  235. </label>
  236. </div>
  237. </div>
  238. </div>
  239. <div class="col-md-6">
  240. <div class="form-group">
  241. <div class="checkbox">
  242. <label>
  243. <input id="RelaysEnabled" type="checkbox" ng-model="tmpOptions.relaysEnabled" /> <span translate>Enable Relaying</span>
  244. </label>
  245. </div>
  246. </div>
  247. </div>
  248. </div>
  249. <div class="row">
  250. <div class="col-md-6">
  251. <div class="form-group">
  252. <label translate for="GlobalAnnServersStr">Global Discovery Servers</label>
  253. <input ng-disabled="!tmpOptions.globalAnnounceEnabled" id="GlobalAnnServersStr" class="form-control" type="text" ng-model="tmpOptions._globalAnnounceServersStr" />
  254. </div>
  255. </div>
  256. <div class="col-md-6">
  257. </div>
  258. </div>
  259. </div>
  260. <div id="settings-ignored-devices" class="tab-pane">
  261. <div class="form-group">
  262. <span ng-if="tmpRemoteIgnoredDevices.length === 0" translate>You have no ignored devices.</span>
  263. <div class="table-responsive" ng-if="tmpRemoteIgnoredDevices.length > 0">
  264. <table class="table-condensed table-striped table" style="table-layout: auto;">
  265. <thead>
  266. <tr>
  267. <th translate>Ignored at</th>
  268. <th translate>Device</th>
  269. <th translate>Address</th>
  270. <th></th>
  271. </tr>
  272. </thead>
  273. <tbody>
  274. <tr ng-repeat="ignoredDevice in tmpRemoteIgnoredDevices">
  275. <td class="no-overflow-ellipse">{{ ignoredDevice.time | date:"yyyy-MM-dd HH:mm:ss" }}</td>
  276. <td>
  277. <span ng-if="!ignoredDevice.name">{{ ignoredDevice.deviceID }}</span>
  278. <span tooltip data-original-title="{{ ignoredDevice.deviceID }}" ng-if="ignoredDevice.name">{{ ignoredDevice.name }}</span>
  279. </td>
  280. <td class="no-overflow-ellipse">{{ ignoredDevice.address }}</td>
  281. <td>
  282. <a href="" ng-click="unignoreDeviceFromTemporaryConfig(ignoredDevice)">
  283. <span class="fas fa-times"></span>&nbsp;<span translate>Unignore</span>
  284. </a>
  285. </td>
  286. </tr>
  287. </tbody>
  288. </table>
  289. </div>
  290. </div>
  291. </div>
  292. <div id="settings-ignored-folders" class="tab-pane">
  293. <div class="form-group">
  294. <span ng-if="ignoredFoldersCountTmpConfig() === 0" translate>You have no ignored folders.</span>
  295. <div class="table-responsive" ng-if="ignoredFoldersCountTmpConfig() > 0">
  296. <table class="table-condensed table-striped table" style="table-layout: auto;">
  297. <thead>
  298. <tr>
  299. <th translate>Ignored at</th>
  300. <th translate>Folder</th>
  301. <th translate>Device</th>
  302. <th></th>
  303. </tr>
  304. </thead>
  305. <tbody ng-repeat-start="device in tmpDevices">
  306. <tr ng-repeat="ignoredFolder in device.ignoredFolders">
  307. <td class="no-overflow-ellipse">{{ ignoredFolder.time | date:"yyyy-MM-dd HH:mm:ss" }}</td>
  308. <td>{{ folderLabel(ignoredFolder.id) }}</td>
  309. <td>
  310. <span tooltip data-original-title="{{ device.deviceID }}">{{ friendlyNameFromID(device.deviceID) }}</span>
  311. </td>
  312. <td>
  313. <a href="" ng-click="unignoreFolderFromTemporaryConfig(device.deviceID, ignoredFolder.id)">
  314. <span class="fas fa-times"></span>&nbsp;<span translate>Unignore</span>
  315. </a>
  316. </td>
  317. </tr>
  318. </tbody>
  319. <tfoot ng-repeat-end></tfoot>
  320. </table>
  321. </div>
  322. </div>
  323. </div>
  324. </div>
  325. </form>
  326. </div>
  327. <div class="modal-footer">
  328. <button type="button" class="btn btn-primary btn-sm" ng-click="saveSettings()">
  329. <span class="fas fa-check"></span>&nbsp;<span translate>Save</span>
  330. </button>
  331. <button type="button" class="btn btn-default btn-sm" data-dismiss="modal">
  332. <span class="fas fa-times"></span>&nbsp;<span translate>Close</span>
  333. </button>
  334. </div>
  335. </modal>