mirror of
https://github.com/octoleo/syncthing.git
synced 2024-11-17 18:45:13 +00:00
5baf5fedb5
Currently, a custom JS script is used to select the whole device ID on click. However, the current script isn't compatible with all browsers (and in IE in particular), making it impossible to select the ID in them at all. Additionally, the same functionality is already available in CSS with no such drawbacks, as the whole selection process is handled by the Web browser natively, which is lightweight and does not require custom code. Thus, remove the currently used JS script completely, replacing it with a new CSS class that can be added to an element when required. If the browser does not support the CSS, the user can still select the element manually, which makes it safer than the current behaviour that can block the user from being able to select the element at all. Signed-off-by: Tomasz Wilczyński <twilczynski@naver.com>
1010 lines
71 KiB
HTML
1010 lines
71 KiB
HTML
<!DOCTYPE html>
|
|
<!--
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
-->
|
|
<html lang="en" ng-app="syncthing" ng-controller="SyncthingController">
|
|
<head>
|
|
<meta charset="utf-8"/>
|
|
<meta http-equiv="X-UA-Compatible" content="IE=edge"/>
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
|
<link rel="shortcut icon" href="assets/img/favicon-{{syncthingStatus()}}.png"/>
|
|
<link rel="mask-icon" href="assets/img/safari-pinned-tab.svg" color="#0882c8"/>
|
|
|
|
<title ng-bind="thisDeviceName() + ' | Syncthing'"></title>
|
|
<link href="vendor/bootstrap/css/bootstrap.css" rel="stylesheet"/>
|
|
<link href="vendor/daterangepicker/daterangepicker.css" rel="stylesheet"/>
|
|
<link href="assets/font/raleway.css" rel="stylesheet"/>
|
|
<link href="vendor/fork-awesome/css/fork-awesome.css" rel="stylesheet"/>
|
|
<link href="vendor/fork-awesome/css/v5-compat.css" rel="stylesheet"/>
|
|
<link href="assets/css/tree.css" rel="stylesheet"/>
|
|
<link href="assets/css/overrides.css" rel="stylesheet"/>
|
|
<link href="assets/css/theme.css" rel="stylesheet"/>
|
|
</head>
|
|
|
|
<body>
|
|
<noscript>
|
|
<nav class="navbar navbar-top navbar-default" role="navigation">
|
|
<div class="container">
|
|
<span class="navbar-brand" aria-hidden="true">
|
|
<img class="logo hidden-xs" src="assets/img/logo-horizontal.svg" height="32" width="117" alt=""/>
|
|
<img class="logo hidden visible-xs" src="assets/img/favicon-default.png" height="32" alt=""/>
|
|
</span>
|
|
</div>
|
|
</nav>
|
|
|
|
<div class="container content">
|
|
<div class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-danger">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-exclamation-circle"></span>
|
|
</div>
|
|
Warning!
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p>
|
|
The Syncthing admin interface requires JavaScript. Please enable JavaScript in your web browser and try again.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</noscript>
|
|
|
|
<div class="ng-cloak">
|
|
<script type="text/javascript" src="syncthing/development/logbar.js"></script>
|
|
<div ng-if="version.isBeta" ng-include="'syncthing/development/logbar.html'"></div>
|
|
<!-- Top bar -->
|
|
|
|
<nav class="navbar navbar-top navbar-default" role="navigation">
|
|
<div class="container">
|
|
<span class="navbar-brand" aria-hidden="true">
|
|
<img class="logo hidden-xs" src="assets/img/logo-horizontal.svg" height="32" width="117" alt=""/>
|
|
<img class="logo hidden visible-xs" src="assets/img/favicon-default.png" height="32" alt=""/>
|
|
</span>
|
|
<p class="navbar-text hidden-xs" ng-class="{'hidden-sm':upgradeInfo && upgradeInfo.newer}">{{thisDeviceName()}}</p>
|
|
<ul class="nav navbar-nav navbar-right">
|
|
<li ng-if="upgradeInfo && upgradeInfo.newer" class="upgrade-newer">
|
|
<button type="button" class="btn navbar-btn btn-primary btn-sm" data-toggle="modal" data-target="#upgrade">
|
|
<span class="fas fa-arrow-circle-up"></span>
|
|
<span class="hidden-xs" translate translate-value-version="{{upgradeInfo.latest}}">Upgrade To {%version%}</span>
|
|
</button>
|
|
</li>
|
|
<li ng-if="upgradeInfo && upgradeInfo.majorNewer" class="upgrade-newer-major">
|
|
<button type="button" class="btn navbar-btn btn-danger btn-sm" data-toggle="modal" data-target="#majorUpgrade">
|
|
<span class="fas fa-arrow-circle-up"></span>
|
|
<span class="hidden-xs" translate translate-value-version="{{upgradeInfo.latest}}">Upgrade To {%version%}</span>
|
|
</button>
|
|
</li>
|
|
<li class="dropdown" language-select></li>
|
|
<li>
|
|
<a class="navbar-link" href="{{docsURL('intro/gui')}}" target="_blank">
|
|
<span class="fas fa-question-circle"></span>
|
|
<span class="hidden-xs" translate>Help</span>
|
|
</a>
|
|
</li>
|
|
<li class="dropdown action-menu">
|
|
<a href="#" class="dropdown-toggle" data-toggle="dropdown" aria-expanded="false">
|
|
<span class="fas fa-cog"></span>
|
|
<span class="hidden-xs" translate>Actions</span>
|
|
<span class="caret"></span>
|
|
</a>
|
|
<ul class="dropdown-menu">
|
|
<li><a href="" ng-click="showSettings()"><span class="fas fa-fw fa-cog"></span> <span translate>Settings</span></a></li>
|
|
<li><a href="" ng-click="showDeviceIdentification(thisDevice())"><span class="fas fa-fw fa-qrcode"></span> <span translate>Show ID</span></a></li>
|
|
<li class="divider" aria-hidden="true"></li>
|
|
<li><a href="" ng-click="shutdown()"><span class="fas fa-fw fa-power-off"></span> <span translate>Shutdown</span></a></li>
|
|
<li><a href="" ng-click="restart()"><span class="fas fa-fw fa-refresh"></span> <span translate>Restart</span></a></li>
|
|
<li class="divider" aria-hidden="true"></li>
|
|
<li class="visible-xs">
|
|
<a href="{{docsURL('intro/gui')}}" target="_blank">
|
|
<span class="fas fa-fw fa-question-circle"></span> <span translate>Help</span>
|
|
</a>
|
|
</li>
|
|
<li><a href="" ng-click="about.show()"><span class="far fa-fw fa-heart"></span> <span translate>About</span></a></li>
|
|
<li class="divider" aria-hidden="true"></li>
|
|
<li><a href="" ng-click="advanced()"><span class="fas fa-fw fa-cogs"></span> <span translate>Advanced</span></a></li>
|
|
<li><a href="" ng-click="logging.show()"><span class="far fa-fw fa-file-alt"></span> <span translate>Logs</span></a></li>
|
|
<li class="divider" aria-hidden="true" ng-if="config.gui.debugging"></li>
|
|
<li><a href="/rest/debug/support" target="_blank" ng-if="config.gui.debugging"><span class="fa fa-user-md"></span> <span translate>Support Bundle</span></a></li>
|
|
</ul>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</nav>
|
|
|
|
<div class="container content">
|
|
|
|
<!-- Panel: Open, no auth -->
|
|
|
|
<div ng-if="openNoAuth" class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-danger">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-exclamation-circle"></span>
|
|
</div>
|
|
<span translate>Danger!</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p>
|
|
<span translate>The Syncthing admin interface is configured to allow remote access without a password.</span>
|
|
<b><span translate>This can easily give hackers access to read and change any files on your computer.</span></b>
|
|
<span translate>Please set a GUI Authentication User and Password in the Settings dialog.</span>
|
|
</p>
|
|
</div>
|
|
<div class="panel-footer">
|
|
<button type="button" class="btn btn-sm btn-default pull-right" ng-click="showSettings()">
|
|
<span class="fas fa-cog"></span> <span translate>Settings</span>
|
|
</button>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Panel: Restart Needed -->
|
|
|
|
<div ng-if="!configInSync" class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-warning">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-exclamation-circle"></span>
|
|
</div>
|
|
<span translate>Restart Needed</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p translate>The configuration has been saved but not activated. Syncthing must restart to activate the new configuration.</p>
|
|
</div>
|
|
<div class="panel-footer">
|
|
<button type="button" class="btn btn-sm btn-default pull-right" ng-click="restart()">
|
|
<span class="fas fa-refresh"></span> <span translate>Restart</span>
|
|
</button>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div ng-if="config">
|
|
|
|
<!-- Panel: Notifications -->
|
|
|
|
<div ng-if="config.options && config.options.unackedNotificationIDs" ng-include="'syncthing/core/notifications.html'"></div>
|
|
|
|
<!-- Panel: New Device -->
|
|
|
|
<div ng-repeat="(deviceID, pendingDevice) in pendingDevices" class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-warning">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<identicon class="panel-icon" data-value="device"></identicon>
|
|
<span translate>New Device</span>
|
|
<span class="pull-right">{{ pendingDevice.time | date:"yyyy-MM-dd HH:mm:ss" }}</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p>
|
|
<span translate translate-value-device="{{ deviceID }}" translate-value-address="{{ pendingDevice.address }}" translate-value-name="{{ pendingDevice.name }}">
|
|
Device "{%name%}" ({%device%} at {%address%}) wants to connect. Add new device?
|
|
</span>
|
|
</p>
|
|
</div>
|
|
<div class="panel-footer clearfix">
|
|
<div class="pull-right">
|
|
<button type="button" class="btn btn-sm btn-success" ng-click="addDevice(deviceID, pendingDevice.name)">
|
|
<span class="fas fa-plus"></span> <span translate>Add Device</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-danger" ng-click="ignoreDevice(deviceID, pendingDevice)" tooltip data-original-title="{{'Permanently add it to the ignore list, suppressing further notifications.' | translate}}">
|
|
<span class="fas fa-times"></span> <span translate>Ignore</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="dismissPendingDevice(deviceID)" tooltip data-original-title="{{'Do not add it to the ignore list, so this notification may recur.' | translate}}">
|
|
<span class="far fa-clock"></span> <span translate>Dismiss</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Panel: New Folder -->
|
|
<div ng-repeat="(folderID, pendingFolder) in pendingFolders">
|
|
<div ng-repeat="(deviceID, offeringDevice) in pendingFolder.offeredBy" class="row reject">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-warning">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-folder"></span>
|
|
</div>
|
|
<span translate ng-if="!folders[folderID]">New Folder</span>
|
|
<span translate ng-if="folders[folderID]">Share Folder</span>
|
|
<span class="pull-right">{{ offeringDevice.time | date:"yyyy-MM-dd HH:mm:ss" }}</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p>
|
|
<span ng-if="offeringDevice.label.length == 0" translate translate-value-device="{{ deviceName(devices[deviceID]) }}" translate-value-folder="{{ folderID }}">
|
|
{%device%} wants to share folder "{%folder%}".
|
|
</span>
|
|
<span ng-if="offeringDevice.label.length != 0" translate translate-value-device="{{ deviceName(devices[deviceID]) }}" translate-value-folder="{{ folderID }}" translate-value-folderlabel="{{ offeringDevice.label }}">
|
|
{%device%} wants to share folder "{%folderlabel%}" ({%folder%}).
|
|
</span>
|
|
<span translate ng-if="folders[folderID]">Share this folder?</span>
|
|
<span translate ng-if="!folders[folderID]">Add new folder?</span>
|
|
</p>
|
|
</div>
|
|
<div class="panel-footer clearfix">
|
|
<div class="pull-right">
|
|
<button type="button" class="btn btn-sm btn-success" ng-click="addFolderAndShare(folderID, pendingFolder, deviceID)" ng-if="!folders[folderID]">
|
|
<span class="fas fa-check"></span> <span translate>Add</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-success" ng-click="shareFolderWithDevice(folderID, deviceID)" ng-if="folders[folderID]">
|
|
<span class="fas fa-check"></span> <span translate>Share</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-danger" ng-click="ignoreFolder(deviceID, folderID, offeringDevice)" tooltip data-original-title="{{'Permanently add it to the ignore list, suppressing further notifications.' | translate}}">
|
|
<span class="fas fa-times"></span> <span translate>Ignore</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="dismissPendingFolder(folderID, deviceID)" tooltip data-original-title="{{'Do not add it to the ignore list, so this notification may recur.' | translate}}">
|
|
<span class="far fa-clock"></span> <span translate>Dismiss</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<!-- Panel: Notice -->
|
|
|
|
<div ng-if="errorList().length > 0" class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-warning">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-exclamation-circle"></span>
|
|
</div>
|
|
<span translate>Notice</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p ng-repeat="err in errorList()">
|
|
<small>{{err.when | date:"yyyy-MM-dd HH:mm:ss"}}:</small>
|
|
<span ng-bind-html="friendlyDevices(err.message) | linky: '_blank'"></span>
|
|
</p>
|
|
</div>
|
|
<div class="panel-footer">
|
|
<button type="button" class="btn btn-sm btn-default pull-right" ng-click="clearErrors()">
|
|
<span class="fas fa-check"></span> <span translate>OK</span>
|
|
</button>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Panel: FS watcher errors -->
|
|
|
|
<div ng-if="sizeOf(fsWatcherErrorMap()) > 0" class="row">
|
|
<div class="col-md-12">
|
|
<div class="panel panel-warning">
|
|
<div class="panel-heading">
|
|
<h3 class="panel-title">
|
|
<div class="panel-icon">
|
|
<span class="fas fa-exclamation-circle"></span>
|
|
</div>
|
|
<span translate>Filesystem Watcher Errors</span>
|
|
</h3>
|
|
</div>
|
|
<div class="panel-body">
|
|
<p>
|
|
<span translate>For the following folders an error occurred while starting to watch for changes. It will be retried every minute, so the errors might go away soon. If they persist, try to fix the underlying issue and ask for help if you can't.</span> <a href="https://forum.syncthing.net" target="_blank"><span class="fas fa-question-circle"></span> <span translate>Support</span></a>
|
|
</p>
|
|
<table>
|
|
<tr ng-repeat="(id, err) in fsWatcherErrorMap()">
|
|
<td>{{folderLabel(id)}}</td><td>{{err}}</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- First regular row -->
|
|
|
|
<div class="row">
|
|
|
|
<!-- Folder list (top left) -->
|
|
|
|
<div class="col-md-6" aria-labelledby="folder_list" role="region" >
|
|
<h3 id="folder_list" translate>Folders</h3>
|
|
<div class="panel-group" id="folders">
|
|
<div class="panel panel-default" ng-repeat="folder in folderList()">
|
|
<button class="btn panel-heading" data-toggle="collapse" data-parent="#folders" data-target="#folder-{{$index}}" aria-expanded="false">
|
|
<div class="panel-progress" ng-show="folderStatus(folder) == 'syncing'" ng-attr-style="width: {{syncPercentage(folder.id) | percent}}"></div>
|
|
<div class="panel-progress" ng-show="folderStatus(folder) == 'scanning' && scanProgress[folder.id] != undefined" ng-attr-style="width: {{scanPercentage(folder.id) | percent}}"></div>
|
|
<h4 class="panel-title">
|
|
<div class="panel-icon hidden-xs">
|
|
<span ng-if="folder.type == 'sendreceive'" class="fas fa-fw fa-folder"></span>
|
|
<span ng-if="folder.type == 'sendonly'" class="fas fa-fw fa-upload"></span>
|
|
<span ng-if="folder.type == 'receiveonly'" class="fas fa-fw fa-download"></span>
|
|
<span ng-if="folder.type == 'receiveencrypted'" class="fas fa-fw fa-lock"></span>
|
|
</div>
|
|
<div class="panel-status pull-right text-{{folderClass(folder)}}" ng-switch="folderStatus(folder)">
|
|
<span ng-switch-when="paused"><span class="hidden-xs" translate>Paused</span><span class="visible-xs" aria-label="{{'Paused' | translate}}"><i class="fas fa-fw fa-pause"></i></span></span>
|
|
<span ng-switch-when="unknown"><span class="hidden-xs" translate>Unknown</span><span class="visible-xs" aria-label="{{'Unknown' | translate}}"><i class="fas fa-fw fa-question-circle"></i></span></span>
|
|
<span ng-switch-when="unshared"><span class="hidden-xs" translate>Unshared</span><span class="visible-xs" aria-label="{{'Unshared' | translate}}"><i class="fas fa-fw fa-unlink"></i></span></span>
|
|
<span ng-switch-when="scan-waiting"><span class="hidden-xs" translate>Waiting to Scan</span><span class="visible-xs" aria-label="{{'Waiting to Scan' | translate}}"><i class="fas fa-fw fa-hourglass-half"></i></span></span>
|
|
<span ng-switch-when="cleaning"><span class="hidden-xs" translate>Cleaning Versions</span><span class="visible-xs" aria-label="{{'Cleaning Versions' | translate}}"><i class="fas fa-fw fa-recycle"></i></span></span>
|
|
<span ng-switch-when="clean-waiting"><span class="hidden-xs" translate>Waiting to Clean</span><span class="visible-xs" aria-label="{{'Waiting to Clean' | translate}}"><i class="fas fa-fw fa-hourglass-half"></i></span></span>
|
|
<span ng-switch-when="stopped"><span class="hidden-xs" translate>Stopped</span><span class="visible-xs" aria-label="{{'Stopped' | translate}}"><i class="fas fa-fw fa-stop"></i></span></span>
|
|
<span ng-switch-when="scanning">
|
|
<span class="hidden-xs" translate>Scanning</span>
|
|
<span class="hidden-xs" ng-if="scanPercentage(folder.id) != undefined">
|
|
({{scanPercentage(folder.id) | percent}})
|
|
</span>
|
|
<span class="visible-xs" aria-label="{{'Scanning' | translate}}"><i class="fas fa-fw fa-search"></i></span>
|
|
</span>
|
|
<span ng-switch-when="idle"><span class="hidden-xs" translate>Up to Date</span><span class="visible-xs" aria-label="{{'Up to Date' | translate}}"><i class="fas fa-fw fa-check"></i></span></span>
|
|
<span ng-switch-when="localadditions"><span class="hidden-xs" translate>Local Additions</span><span class="visible-xs" aria-label="{{'Local Additions' | translate}}"><i class="fas fa-fw fa-check"></i></span></span>
|
|
<span ng-switch-when="sync-waiting">
|
|
<span class="hidden-xs" translate>Waiting to Sync</span>
|
|
<span class="visible-xs" aria-label="{{'Waiting to Sync' | translate}}"><i class="fas fa-fw fa-hourglass-half"></i></span>
|
|
</span>
|
|
<span ng-switch-when="sync-preparing">
|
|
<span class="hidden-xs" translate>Preparing to Sync</span>
|
|
<span class="visible-xs" aria-label="{{'Preparing to Sync' | translate}}"><i class="fas fa-fw fa-hourglass-half"></i></span>
|
|
</span>
|
|
<span ng-switch-when="syncing">
|
|
<span class="hidden-xs" translate>Syncing</span>
|
|
<span>({{syncPercentage(folder.id) | percent}}, {{model[folder.id].needBytes | binary}}B)</span>
|
|
</span>
|
|
<span ng-switch-when="outofsync"><span class="hidden-xs" translate>Out of Sync</span><span class="visible-xs" aria-label="{{'Out of Sync' | translate}}"><i class="fas fa-fw fa-exclamation-circle"></i></span></span>
|
|
<span ng-switch-when="faileditems"><span class="hidden-xs" translate>Failed Items</span><span class="visible-xs" aria-label="{{'Failed Items' | translate}}"><i class="fas fa-fw fa-exclamation-circle"></i></span></span>
|
|
<span ng-switch-when="localunencrypted"><span class="hidden-xs">{{'Unexpected Items' | translate}}</span><span class="visible-xs" aria-label="{{'Unexpected Items' | translate}}"><i class="fas fa-fw fa-exclamation-circle"></i></span></span>
|
|
</div>
|
|
<div class="panel-title-text">
|
|
<span tooltip data-original-title="{{folder.label.length != 0 ? folder.id : ''}}">{{folder.label.length != 0 ? folder.label : folder.id}}</span>
|
|
</div>
|
|
</h4>
|
|
</button>
|
|
<div id="folder-{{$index}}" class="panel-collapse collapse">
|
|
<div class="panel-body">
|
|
<table class="table table-condensed table-striped table-auto">
|
|
<tbody>
|
|
<tr ng-show="folder.label != undefined && folder.label.length > 0">
|
|
<th><span class="fas fa-fw fa-info-circle"></span> <span translate>Folder ID</span></th>
|
|
<td class="text-right no-overflow-ellipse">{{folder.id}}</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-folder-open"></span> <span translate>Folder Path</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{folder.path}}">{{folder.path}}</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="!folder.paused && (model[folder.id].invalid || model[folder.id].error)">
|
|
<th><span class="fas fa-fw fa-exclamation-triangle"></span> <span translate>Error</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{model[folder.id].invalid || model[folder.id].error}}">{{model[folder.id].invalid || model[folder.id].error}}</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="!folder.paused">
|
|
<th><span class="fas fa-fw fa-globe"></span> <span translate>Global State</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{model[folder.id].globalFiles | alwaysNumber | localeNumber}} {{'files' | translate}}, {{model[folder.id].globalDirectories | alwaysNumber | localeNumber}} {{'directories' | translate}}, ~{{model[folder.id].globalBytes | binary}}B">
|
|
<span class="far fa-copy"></span> {{model[folder.id].globalFiles | alwaysNumber | localeNumber}} 
|
|
<span class="far fa-folder"></span> {{model[folder.id].globalDirectories | alwaysNumber | localeNumber}} 
|
|
<span class="far fa-hdd"></span> ~{{model[folder.id].globalBytes | binary}}B
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="!folder.paused">
|
|
<th><span class="fas fa-fw fa-home"></span> <span translate>Local State</span></th>
|
|
<td class="text-right">
|
|
<div>
|
|
<span tooltip data-original-title="{{model[folder.id].localFiles | alwaysNumber | localeNumber}} {{'files' | translate}}, {{model[folder.id].localDirectories | alwaysNumber | localeNumber}} {{'directories' | translate}}, ~{{model[folder.id].localBytes | binary}}B">
|
|
<span class="far fa-copy"></span> {{model[folder.id].localFiles | alwaysNumber | localeNumber}} 
|
|
<span class="far fa-folder"></span> {{model[folder.id].localDirectories | alwaysNumber | localeNumber}} 
|
|
<span class="far fa-hdd"></span> ~{{model[folder.id].localBytes | binary}}B
|
|
</span>
|
|
</div>
|
|
<div ng-if="model[folder.id].ignorePatterns">
|
|
<a href="" ng-click="editFolderExisting(folder, '#folder-ignores')"><i class="small" translate>Reduced by ignore patterns</i></a>
|
|
</div>
|
|
<div ng-if="folder.ignoreDelete">
|
|
<i class="small">
|
|
<span translate>Altered by ignoring deletes.</span>
|
|
<a href="{{docsURL('advanced/folder-ignoredelete')}}" target="_blank">
|
|
<span class="fas fa-question-circle"></span> <span translate>Help</span>
|
|
</a>
|
|
</i>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="model[folder.id].needTotalItems > 0">
|
|
<th><span class="fas fa-fw fa-cloud-download-alt"></span> <span translate>Out of Sync Items</span></th>
|
|
<td class="text-right">
|
|
<a href="" ng-click="showNeed(folder.id)">{{model[folder.id].needTotalItems | alwaysNumber | localeNumber}} <span translate>items</span>, ~{{model[folder.id].needBytes | binary}}B</a>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folderStatus(folder) === 'scanning' && scanRate(folder.id) > 0">
|
|
<th><span class="fas fa-fw fa-hourglass-half"></span> <span translate>Scan Time Remaining</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{scanRate(folder.id) | binary}}B/s">~ {{scanRemaining(folder.id)}}</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="hasFailedFiles(folder.id)">
|
|
<th><span class="fas fa-fw fa-exclamation-circle"></span> <span translate>Failed Items</span></th>
|
|
<!-- Show the number of failed items as a link to bring up the list. -->
|
|
<td class="text-right">
|
|
<a href="" ng-click="showFailed(folder.id)">{{model[folder.id].pullErrors | alwaysNumber | localeNumber}} <span translate>items</span></a>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="hasReceiveOnlyChanged(folder)">
|
|
<th><span class="fas fa-fw fa-exclamation-circle"></span> <span translate>Locally Changed Items</span></th>
|
|
<td class="text-right">
|
|
<a href="" ng-click="showLocalChanged(folder.id, folder.type)">{{model[folder.id].receiveOnlyTotalItems | alwaysNumber | localeNumber}} <span translate>items</span>, ~{{model[folder.id].receiveOnlyChangedBytes | binary}}B</a>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folder.type != 'sendreceive'">
|
|
<th><span class="fas fa-fw fa-folder"></span> <span translate>Folder Type</span></th>
|
|
<td class="text-right">
|
|
<span ng-if="folder.type == 'sendonly'" translate>Send Only</span>
|
|
<span ng-if="folder.type == 'receiveonly'" translate>Receive Only</span>
|
|
<span ng-if="folder.type == 'receiveencrypted'" translate>Receive Encrypted</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folder.ignorePerms">
|
|
<th><span class="far fa-fw fa-minus-square"></span> <span translate>Ignore Permissions</span></th>
|
|
<td class="text-right">
|
|
<span translate>Yes</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-refresh"></span> <span translate>Rescans</span></th>
|
|
<td class="text-right">
|
|
<div ng-if="folder.rescanIntervalS > 0">
|
|
<span ng-if="!folder.fsWatcherEnabled" tooltip data-original-title="{{'Periodic scanning at given interval and disabled watching for changes' | translate}}">
|
|
<span class="far fa-clock"></span> {{folder.rescanIntervalS | duration}} 
|
|
<span class="fas fa-eye-slash"></span> <span translate>Disabled</span>
|
|
</span>
|
|
<span ng-if="folder.fsWatcherEnabled && (!model[folder.id].watchError || folder.paused || folderStatus(folder) === 'stopped')" tooltip data-original-title="{{'Periodic scanning at given interval and enabled watching for changes' | translate}}">
|
|
<span class="far fa-clock"></span> {{folder.rescanIntervalS | duration}} 
|
|
<span class="fas fa-eye"></span> <span translate>Enabled</span>
|
|
</span>
|
|
<span ng-if="folder.fsWatcherEnabled && !folder.paused && folderStatus(folder) !== 'stopped' && model[folder.id].watchError" tooltip data-original-title="{{'Periodic scanning at given interval and failed setting up watching for changes, retrying every 1m:' | translate}}<br/>{{model[folder.id].watchError}}">
|
|
<span class="far fa-clock"></span> {{folder.rescanIntervalS | duration}} 
|
|
<span class="fas fa-eye-slash"></span> <span translate>Failed to setup, retrying</span>
|
|
</span>
|
|
</div>
|
|
<div ng-if="folder.rescanIntervalS <= 0">
|
|
<span ng-if="!folder.fsWatcherEnabled" tooltip data-original-title="{{'Disabled periodic scanning and disabled watching for changes' | translate}}">
|
|
<span class="far fa-clock"></span> <span translate>Disabled</span> 
|
|
<span class="fas fa-eye-slash"></span> <span translate>Disabled</span>
|
|
</span>
|
|
<span ng-if="folder.fsWatcherEnabled && (!model[folder.id].watchError || folder.paused || folderStatus(folder) === 'stopped')" tooltip data-original-title="{{'Disabled periodic scanning and enabled watching for changes' | translate}}">
|
|
<span class="far fa-clock"></span> <span translate>Disabled</span> 
|
|
<span class="fas fa-eye"></span> <span translate>Enabled</span>
|
|
</span>
|
|
<span ng-if="folder.fsWatcherEnabled && !folder.paused && folderStatus(folder) !== 'stopped' && model[folder.id].watchError" tooltip data-original-title="{{'Disabled periodic scanning and failed setting up watching for changes, retrying every 1m:' | translate}}<br/>{{model[folder.id].watchError}}">
|
|
<span class="far fa-clock"></span> <span translate>Disabled</span> 
|
|
<span class="fas fa-eye-slash"></span> <span translate>Failed to setup, retrying</span>
|
|
</span>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folder.order != 'random' && folder.type != 'sendonly'">
|
|
<th><span class="fas fa-fw fa-sort"></span> <span translate>File Pull Order</span></th>
|
|
<td class="text-right" ng-switch="folder.order">
|
|
<span ng-switch-when="random" translate>Random</span>
|
|
<span ng-switch-when="alphabetic" translate>Alphabetic</span>
|
|
<span ng-switch-when="smallestFirst" translate>Smallest First</span>
|
|
<span ng-switch-when="largestFirst" translate>Largest First</span>
|
|
<span ng-switch-when="oldestFirst" translate>Oldest First</span>
|
|
<span ng-switch-when="newestFirst" translate>Newest First</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folder.versioning.type">
|
|
<th><span class="fa fa-fw fa-file"></span> <span translate>File Versioning</span></th>
|
|
<td class="text-right">
|
|
<span ng-switch="folder.versioning.type">
|
|
<span ng-switch-when="trashcan" translate>Trash Can</span>
|
|
<span ng-switch-when="simple" translate>Simple</span>
|
|
<span ng-switch-when="staggered" translate>Staggered</span>
|
|
<span ng-switch-when="external" tooltip data-original-title="<span class='text-monospace'>{{folder.versioning.params.command}}</span>" translate>External</span>
|
|
</span>
|
|
<span ng-if="folder.versioning.type != 'external'">
|
|
<span ng-if="(folder.versioning.type == 'trashcan' || folder.versioning.type == 'simple') && folder.versioning.params.cleanoutDays != versioningDefaults.trashcanClean" tooltip data-original-title="{{'Clean out after' | translate}}">
|
|
 <span class="fa fa-calendar"></span> {{folder.versioning.params.cleanoutDays * 86400 | duration:"d"}}
|
|
</span>
|
|
<span ng-if="folder.versioning.type == 'simple' && folder.versioning.params.keep != versioningDefaults.simpleKeep" tooltip data-original-title="{{'Keep Versions' | translate}}">
|
|
 <span class="fa fa-file-archive-o"></span> {{folder.versioning.params.keep}}
|
|
</span>
|
|
<span ng-if="folder.versioning.type == 'staggered' && folder.versioning.params.maxAge / 86400 != versioningDefaults.staggeredMaxAge" tooltip data-original-title="{{'Maximum Age' | translate}}">
|
|
 <span class="fa fa-calendar"></span> <span ng-if="folder.versioning.params.maxAge == 0" translate>Forever</span><span ng-if="folder.versioning.params.maxAge > 0">{{folder.versioning.params.maxAge | duration}}</span>
|
|
</span>
|
|
<span ng-if="folder.versioning.cleanupIntervalS != versioningDefaults.cleanupIntervalS" tooltip data-original-title="{{'Cleanup Interval' | translate}}">
|
|
 <span class="fa fa-recycle"></span> <span ng-if="folder.versioning.cleanupIntervalS == 0" translate>Disabled</span><span ng-if="folder.versioning.cleanupIntervalS > 0">{{folder.versioning.cleanupIntervalS | duration}}</span>
|
|
</span>
|
|
<!-- Keep the path last, so that it truncates without pushing other information out of the screen. -->
|
|
<span ng-if="folder.versioning.fsPath != ''" tooltip data-original-title="{{folder.versioning.fsPath}}">
|
|
 <span class="fa fa-folder-open-o"></span> {{folder.versioning.fsPath}}
|
|
</span>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-share-alt"></span> <span translate>Shared With</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{sharesFolder(folder)}} {{folderHasUnacceptedDevices(folder) ? '<br/>(<sup>1</sup>' + ('The remote device has not accepted sharing this folder.' | translate) + ')' : ''}} {{folderHasPausedDevices(folder) ? '<br/>(<sup>2</sup>' + ('The remote device has paused this folder.' | translate) + ')' : ''}}" ng-bind-html="sharesFolder(folder)"></span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folderStats[folder.id].lastScan">
|
|
<th><span class="far fa-fw fa-clock"></span> <span translate>Last Scan</span></th>
|
|
<td translate ng-if="folderStats[folder.id].lastScanDays >= 365" class="text-right">Never</td>
|
|
<td ng-if="folderStats[folder.id].lastScanDays < 365" class="text-right">
|
|
<span>{{folderStats[folder.id].lastScan | date:'yyyy-MM-dd HH:mm:ss'}}</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="folder.type != 'sendonly' && folder.type != 'receiveencrypted' && folderStats[folder.id].lastFile && folderStats[folder.id].lastFile.filename">
|
|
<th><span class="fas fa-fw fa-exchange-alt"></span> <span translate>Latest Change</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{folderStats[folder.id].lastFile.filename}} @ {{folderStats[folder.id].lastFile.at | date:'yyyy-MM-dd HH:mm:ss'}}">
|
|
<span translate translate-value-file="{{folderStats[folder.id].lastFile.filename | basename}}" ng-if="!folderStats[folder.id].lastFile.deleted">Updated {%file%}</span>
|
|
<span translate translate-value-file="{{folderStats[folder.id].lastFile.filename | basename}}" ng-if="folderStats[folder.id].lastFile.deleted">Deleted {%file%}</span>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<div class="panel-footer">
|
|
<button type="button" class="btn btn-sm btn-danger pull-left" ng-click="revertOverrideConfirmationModal('override', folder.id)" ng-if="folderStatus(folder) == 'outofsync' && folder.type == 'sendonly'">
|
|
<span class="fas fa-arrow-circle-up"></span> <span translate>Override Changes</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-danger pull-left" ng-click="revertOverrideConfirmationModal('revert', folder.id)" ng-if="hasReceiveOnlyChanged(folder)">
|
|
<span class="fa fa-arrow-circle-down"></span> <span translate>Revert Local Changes</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-danger pull-left" ng-click="revertOverrideConfirmationModal('deleteEnc', folder.id)" ng-if="hasReceiveEncryptedItems(folder)">
|
|
<span class="fa fa-minus-circle"></span> <span translate>Delete Unexpected Items</span>
|
|
</button>
|
|
<span class="pull-right">
|
|
<button ng-if="!folder.paused" type="button" class="btn btn-sm btn-default" ng-click="setFolderPause(folder.id, true)">
|
|
<span class="fas fa-pause"></span> <span translate>Pause</span>
|
|
</button>
|
|
<button ng-if="folder.paused" type="button" class="btn btn-sm btn-default" ng-click="setFolderPause(folder.id, false)">
|
|
<span class="fas fa-play"></span> <span translate>Resume</span>
|
|
</button>
|
|
<button type="button" class="btn btn-default btn-sm" ng-click="restoreVersions.show(folder.id)" ng-if="folder.versioning.type && folder.versioning.type != 'external'" ng-disabled="folder.paused">
|
|
<span class="fas fa-undo"></span> <span translate>Versions</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="rescanFolder(folder.id)" ng-disabled="['idle', 'stopped', 'unshared', 'outofsync', 'faileditems', 'localadditions'].indexOf(folderStatus(folder)) < 0">
|
|
<span class="fas fa-refresh"></span> <span translate>Rescan</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="editFolderExisting(folder)">
|
|
<span class="fas fa-pencil-alt"></span> <span translate>Edit</span>
|
|
</button>
|
|
</span>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<span class="pull-right">
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="setAllFoldersPause(true)" ng-if="isAtleastOneFolderPausedStateSetTo(false)">
|
|
<span class="fas fa-pause"></span> <span translate>Pause All</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="setAllFoldersPause(false)" ng-if="isAtleastOneFolderPausedStateSetTo(true)">
|
|
<span class="fas fa-play"></span> <span translate>Resume All</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="rescanAllFolders()" ng-if="folderList().length > 0" ng-disabled="!isAtleastOneFolderPausedStateSetTo(false)">
|
|
<span class="fas fa-refresh"></span> <span translate>Rescan All</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="addFolder()">
|
|
<span class="fas fa-plus"></span> <span translate>Add Folder</span>
|
|
</button>
|
|
</span>
|
|
<div class="clearfix"></div>
|
|
<hr class="visible-sm"/>
|
|
</div>
|
|
|
|
<!-- Device list (top right) -->
|
|
|
|
<!-- This device -->
|
|
|
|
<div class="col-md-6" aria-label="{{'Devices' | translate}}" role="region">
|
|
<h3 translate>This Device</h3>
|
|
<div class="panel panel-default" ng-repeat="deviceCfg in [thisDevice()]">
|
|
<button class="btn panel-heading" data-toggle="collapse" data-target="#device-this" aria-expanded="true">
|
|
<h4 class="panel-title">
|
|
<identicon class="panel-icon" data-value="deviceCfg.deviceID"></identicon>
|
|
<div class="panel-title-text">{{deviceName(deviceCfg)}}</div>
|
|
</h4>
|
|
</button>
|
|
<div id="device-this" class="panel-collapse collapse in">
|
|
<div class="panel-body">
|
|
<table class="table table-condensed table-striped table-auto">
|
|
<tbody>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-cloud-download-alt"></span> <span translate>Download Rate</span></th>
|
|
<td class="text-right">
|
|
<a href="#" class="toggler" ng-click="toggleUnits()">
|
|
<span ng-if="!metricRates">{{connectionsTotal.inbps | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{connectionsTotal.inbps*8 | metric}}bps</span>
|
|
({{connectionsTotal.inBytesTotal | binary}}B)
|
|
<small ng-if="config.options.maxRecvKbps > 0"><br/>
|
|
<i class="text-muted"><span translate>Limit</span>:
|
|
<span ng-if="!metricRates">{{config.options.maxRecvKbps*1024 | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{config.options.maxRecvKbps*1024*8 | metric}}bps</span>
|
|
</i>
|
|
</small>
|
|
</a>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-cloud-upload-alt"></span> <span translate>Upload Rate</span></th>
|
|
<td class="text-right">
|
|
<a href="#" class="toggler" ng-click="toggleUnits()">
|
|
<span ng-if="!metricRates">{{connectionsTotal.outbps | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{connectionsTotal.outbps*8 | metric}}bps</span>
|
|
({{connectionsTotal.outBytesTotal | binary}}B)
|
|
<small ng-if="config.options.maxSendKbps > 0"><br/>
|
|
<i class="text-muted"><span translate>Limit</span>:
|
|
<span ng-if="!metricRates">{{config.options.maxSendKbps*1024 | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{config.options.maxSendKbps*1024*8 | metric}}bps</span>
|
|
</i>
|
|
</small>
|
|
</a>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-home"></span> <span translate>Local State (Total)</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{localStateTotal.files | alwaysNumber | localeNumber}} {{'files' | translate}}, {{ localStateTotal.directories | alwaysNumber | localeNumber}} {{'directories' | translate}}, ~{{ localStateTotal.bytes | binary}}B">
|
|
<span class="far fa-copy"></span> {{localStateTotal.files | alwaysNumber | localeNumber}} 
|
|
<span class="far fa-folder"></span> {{localStateTotal.directories| alwaysNumber | localeNumber}} 
|
|
<span class="far fa-hdd"></span> ~{{localStateTotal.bytes | binary}}B
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-sitemap"></span> <span translate>Listeners</span></th>
|
|
<td class="text-right">
|
|
<span class="data" tooltip data-original-title="{{'Show detailed listener status' | translate}}.">
|
|
<a href="" ng-class="{'text-success': listenersTotal > 0 && listenersFailed.length == 0, 'text-danger': listenersTotal > 0 && listenersFailed.length == listenersTotal}" ng-click="showListenerStatus()">
|
|
{{listenersTotal-listenersFailed.length}}/{{listenersTotal}}
|
|
</a>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="system.discoveryEnabled">
|
|
<th><span class="fas fa-fw fa-map-signs"></span> <span translate>Discovery</span></th>
|
|
<td class="text-right">
|
|
<span class="data" tooltip data-original-title="{{'Show detailed discovery status' | translate}}.">
|
|
<a href="" ng-class="{'text-success': discoveryFailed.length == 0, 'text-danger': discoveryFailed.length == discoveryTotal}" ng-click="showDiscoveryStatus()">
|
|
{{discoveryTotal-discoveryFailed.length}}/{{discoveryTotal}}
|
|
</a>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="far fa-fw fa-clock"></span> <span translate>Uptime</span></th>
|
|
<td class="text-right">{{system.uptime | duration:"m"}}</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-qrcode"></span> <span translate>Identification</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{'Click to see full identification string and QR code.' | translate}}">
|
|
<a href="" ng-click="showDeviceIdentification(thisDevice())">{{deviceShortID(deviceCfg.deviceID)}}</a>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-tag"></span> <span translate>Version</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{versionString()}}">{{versionString()}}</span>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Remote devices -->
|
|
<h3 translate>Remote Devices</h3>
|
|
<div class="panel-group" id="devices">
|
|
<div class="panel panel-default" ng-repeat="deviceCfg in otherDevices()">
|
|
<button class="btn panel-heading" data-toggle="collapse" data-parent="#devices" data-target="#device-{{$index}}" aria-expanded="false">
|
|
<div class="panel-progress" ng-show="deviceStatus(deviceCfg) == 'syncing'" ng-attr-style="width: {{completion[deviceCfg.deviceID]._total | percent}}"></div>
|
|
<h4 class="panel-title">
|
|
<identicon class="panel-icon" data-value="deviceCfg.deviceID"></identicon>
|
|
<span ng-switch="deviceStatus(deviceCfg)" class="pull-right text-{{deviceClass(deviceCfg)}}">
|
|
<span ng-switch-when="insync"><span class="hidden-xs" translate>Up to Date</span><span class="visible-xs" aria-label="{{'Up to Date' | translate}}"><i class="fas fa-fw fa-check"></i></span></span>
|
|
<span ng-switch-when="unused-insync"><span class="hidden-xs" translate>Connected (Unused)</span><span class="visible-xs" aria-label="{{'Connected (Unused)' | translate}}"><i class="fas fa-fw fa-unlink"></i></span></span>
|
|
<span ng-switch-when="syncing">
|
|
<span class="hidden-xs" translate>Syncing</span> ({{completion[deviceCfg.deviceID]._total | percent}}, {{completion[deviceCfg.deviceID]._needBytes | binary}}B)
|
|
</span>
|
|
<span ng-switch-when="paused"><span class="hidden-xs" translate>Paused</span><span class="visible-xs" aria-label="{{'Paused' | translate}}"><i class="fas fa-fw fa-pause"></i></span></span>
|
|
<span ng-switch-when="unused-paused"><span class="hidden-xs" translate>Paused (Unused)</span><span class="visible-xs" aria-label="{{'Paused (Unused)' | translate}}"><i class="fas fa-fw fa-unlink"></i></span></span>
|
|
<span ng-switch-when="disconnected"><span class="hidden-xs" translate>Disconnected</span><span class="visible-xs" aria-label="{{'Disconnected' | translate}}"><i class="fas fa-fw fa-power-off"></i></span></span>
|
|
<span ng-switch-when="unused-disconnected"><span class="hidden-xs" translate>Disconnected (Unused)</span><span class="visible-xs" aria-label="{{'Disconnected (Unused)' | translate}}"><i class="fas fa-fw fa-unlink"></i></span></span>
|
|
</span>
|
|
<div class="panel-title-text">{{deviceName(deviceCfg)}}</div>
|
|
</h4>
|
|
</button>
|
|
<div id="device-{{$index}}" class="panel-collapse collapse">
|
|
<div class="panel-body">
|
|
<table class="table table-condensed table-striped table-auto">
|
|
<tbody>
|
|
<tr ng-if="!connections[deviceCfg.deviceID].connected">
|
|
<th><span class="fas fa-fw fa-eye"></span> <span translate>Last seen</span></th>
|
|
<td translate ng-if="!deviceStats[deviceCfg.deviceID].lastSeenDays || deviceStats[deviceCfg.deviceID].lastSeenDays >= 365" class="text-right">Never</td>
|
|
<td ng-if="deviceStats[deviceCfg.deviceID].lastSeenDays < 365" class="text-right">{{deviceStats[deviceCfg.deviceID].lastSeen | date:"yyyy-MM-dd HH:mm:ss"}}</td>
|
|
</tr>
|
|
<tr ng-if="!connections[deviceCfg.deviceID].connected && deviceFolders(deviceCfg).length > 0">
|
|
<th><span class="fas fa-fw fa-cloud"></span> <span translate>Sync Status</span></th>
|
|
<td translate ng-if="completion[deviceCfg.deviceID]._total == 100" class="text-right">Up to Date</td>
|
|
<td ng-if="completion[deviceCfg.deviceID]._total < 100" class="text-right">
|
|
<span class="hidden-xs" translate>Out of Sync</span> ({{completion[deviceCfg.deviceID]._total | percent}}, {{completion[deviceCfg.deviceID]._needBytes | binary}}B)
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="connections[deviceCfg.deviceID].connected">
|
|
<th><span class="fas fa-fw fa-cloud-download-alt"></span> <span translate>Download Rate</span></th>
|
|
<td class="text-right">
|
|
<a href="#" class="toggler" ng-click="toggleUnits()">
|
|
<span ng-if="!metricRates">{{connections[deviceCfg.deviceID].inbps | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{connections[deviceCfg.deviceID].inbps*8 | metric}}bps</span>
|
|
({{connections[deviceCfg.deviceID].inBytesTotal | binary}}B)
|
|
<small ng-if="deviceCfg.maxRecvKbps > 0"><br/>
|
|
<i class="text-muted"><span translate>Limit</span>:
|
|
<span ng-if="!metricRates">{{deviceCfg.maxRecvKbps*1024 | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{deviceCfg.maxRecvKbps*1024*8 | metric}}bps</span>
|
|
</i>
|
|
</small>
|
|
</a>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="connections[deviceCfg.deviceID].connected">
|
|
<th><span class="fas fa-fw fa-cloud-upload-alt"></span> <span translate>Upload Rate</span></th>
|
|
<td class="text-right">
|
|
<a href="#" class="toggler" ng-click="toggleUnits()">
|
|
<span ng-if="!metricRates">{{connections[deviceCfg.deviceID].outbps | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{connections[deviceCfg.deviceID].outbps*8 | metric}}bps</span>
|
|
({{connections[deviceCfg.deviceID].outBytesTotal | binary}}B)
|
|
<small ng-if="deviceCfg.maxSendKbps > 0"><br/>
|
|
<i class="text-muted"><span translate>Limit</span>:
|
|
<span ng-if="!metricRates">{{deviceCfg.maxSendKbps*1024 | binary}}B/s</span>
|
|
<span ng-if="metricRates">{{deviceCfg.maxSendKbps*1024*8 | metric}}bps</span>
|
|
</i>
|
|
</small>
|
|
</a>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="deviceStatus(deviceCfg) == 'syncing'">
|
|
<th><span class="fas fa-fw fa-exchange-alt"></span> <span translate>Out of Sync Items</span></th>
|
|
<td class="text-right">
|
|
<a href="" ng-click="showRemoteNeed(deviceCfg)">{{completion[deviceCfg.deviceID]._needItems | alwaysNumber | localeNumber}} <span translate>items</span>, ~{{completion[deviceCfg.deviceID]._needBytes | binary}}B</a>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-link"></span> <span translate>Address</span></th>
|
|
<td ng-if="connections[deviceCfg.deviceID].connected" class="text-right">
|
|
<span tooltip data-original-title="{{ connections[deviceCfg.deviceID].type.indexOf('Relay') > -1 ? '' : connections[deviceCfg.deviceID].type }} {{ connections[deviceCfg.deviceID].crypto }}">
|
|
{{deviceAddr(deviceCfg)}}
|
|
</span>
|
|
</td>
|
|
<td ng-if="!connections[deviceCfg.deviceID].connected" class="text-right">
|
|
<span ng-repeat="addr in deviceCfg.addresses">
|
|
<span tooltip data-original-title="{{'Configured' | translate}}">{{addr}}</span><br>
|
|
<small ng-if="system.lastDialStatus[addr].error" tooltip data-original-title="{{system.lastDialStatus[addr].error}}" class="text-danger">{{abbreviatedError(addr)}}<br></small>
|
|
</span>
|
|
<span ng-repeat="addr in discoveryCache[deviceCfg.deviceID].addresses">
|
|
<span tooltip data-original-title="{{'Discovered' | translate}}">{{addr}}</span><br>
|
|
<small ng-if="system.lastDialStatus[addr].error" tooltip data-original-title="{{system.lastDialStatus[addr].error}}" class="text-danger">{{abbreviatedError(addr)}}<br></small>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="connections[deviceCfg.deviceID].connected && connections[deviceCfg.deviceID].type.indexOf('Relay') > -1" tooltip data-original-title="Connections via relays might be rate limited by the relay">
|
|
<th><span class="fas fa-fw fa-exclamation-triangle text-danger"></span> <span translate>Connection Type</span></th>
|
|
<td class="text-right">{{connections[deviceCfg.deviceID].type}}</td>
|
|
</tr>
|
|
<tr ng-if="deviceCfg.allowedNetworks.length > 0">
|
|
<th><span class="fas fa-fw fa-filter"></span> <span translate>Allowed Networks</span></th>
|
|
<td class="text-right">
|
|
<span>{{deviceCfg.allowedNetworks.join(", ")}}</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="deviceCfg.compression != 'metadata'">
|
|
<th><span class="fas fa-fw fa-compress"></span> <span translate>Compression</span></th>
|
|
<td class="text-right">
|
|
<span ng-if="deviceCfg.compression == 'always'" translate>All Data</span>
|
|
<span ng-if="deviceCfg.compression == 'never'" translate>Off</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="deviceCfg.introducer">
|
|
<th><span class="far fa-fw fa-thumbs-up"></span> <span translate>Introducer</span></th>
|
|
<td translate class="text-right">Yes</td>
|
|
</tr>
|
|
<tr ng-if="deviceCfg.introducedBy">
|
|
<th><span class="far fa-fw fa-handshake-o"></span> <span translate>Introduced By</span></th>
|
|
<td class="text-right">{{ deviceName(devices[deviceCfg.introducedBy]) || deviceCfg.introducedBy.substring(0, 5) }}</td>
|
|
</tr>
|
|
<tr>
|
|
<th><span class="fas fa-fw fa-qrcode"></span> <span translate>Identification</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{'Click to see full identification string and QR code.' | translate}}">
|
|
<a href="" ng-click="showDeviceIdentification(deviceCfg)">{{deviceShortID(deviceCfg.deviceID)}}</a>
|
|
</span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="connections[deviceCfg.deviceID].clientVersion">
|
|
<th><span class="fas fa-fw fa-tag"></span> <span translate>Version</span></th>
|
|
<td class="text-right">{{connections[deviceCfg.deviceID].clientVersion}}</td>
|
|
</tr>
|
|
<tr ng-if="deviceFolders(deviceCfg).length > 0">
|
|
<th><span class="fas fa-fw fa-folder"></span> <span translate>Folders</span></th>
|
|
<td class="text-right">
|
|
<span tooltip data-original-title="{{sharedFolders(deviceCfg)}} {{deviceHasUnacceptedFolders(deviceCfg) ? '<br/>(<sup>1</sup>' + ('The remote device has not accepted sharing this folder.' | translate) + ')' : '' }} {{deviceHasPausedFolders(deviceCfg) ? '<br/>(<sup>2</sup>' + ('The remote device has paused this folder.' | translate) + ')' : '' }}" ng-bind-html="sharedFolders(deviceCfg)"></span>
|
|
</td>
|
|
</tr>
|
|
<tr ng-if="deviceCfg.remoteGUIPort > 0">
|
|
<th><span class="fas fa-fw fa-desktop"></span> <span translate>Remote GUI</span></th>
|
|
<td class="text-right" ng-attr-title="Port {{deviceCfg.remoteGUIPort}}">
|
|
<!-- Apply RFC6874 encoding for IPv6 link-local zone identifier -->
|
|
<a ng-if="hasRemoteGUIAddress(deviceCfg)" href="{{remoteGUIAddress(deviceCfg).replace('%', '%25')}}">{{remoteGUIAddress(deviceCfg)}}</a>
|
|
<span translate ng-if="!hasRemoteGUIAddress(deviceCfg)">Unknown</span>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<div class="panel-footer">
|
|
<span class="pull-right">
|
|
<button ng-if="!deviceCfg.paused" type="button" class="btn btn-sm btn-default" ng-click="setDevicePause(deviceCfg.deviceID, true)">
|
|
<span class="fas fa-pause"></span> <span translate>Pause</span>
|
|
</button>
|
|
<button ng-if="deviceCfg.paused" type="button" class="btn btn-sm btn-default" ng-click="setDevicePause(deviceCfg.deviceID, false)">
|
|
<span class="fas fa-play"></span> <span translate>Resume</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="editDeviceExisting(deviceCfg)">
|
|
<span class="fas fa-pencil-alt"></span> <span translate>Edit</span>
|
|
</button>
|
|
</span>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="form-group">
|
|
<span class="pull-right">
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="setAllDevicesPause(true)" ng-if="isAtleastOneDevicePausedStateSetTo(false)">
|
|
<span class="fas fa-pause"></span> <span translate>Pause All</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="setAllDevicesPause(false)" ng-if="isAtleastOneDevicePausedStateSetTo(true)">
|
|
<span class="fas fa-play"></span> <span translate>Resume All</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="globalChanges()">
|
|
<span class="fas fa-fw fa-info-circle"></span> <span translate>Recent Changes</span>
|
|
</button>
|
|
<button type="button" class="btn btn-sm btn-default" ng-click="addDevice()">
|
|
<span class="fas fa-plus"></span> <span translate>Add Remote Device</span>
|
|
</button>
|
|
</span>
|
|
<div class="clearfix"></div>
|
|
</div>
|
|
</div>
|
|
</div> <!-- /row -->
|
|
|
|
</div> <!-- /container -->
|
|
</div> <!-- /ng-cloak -->
|
|
|
|
<!-- Bottom bar -->
|
|
|
|
<nav class="navbar navbar-default navbar-fixed-bottom">
|
|
<div class="container">
|
|
<ul class="nav navbar-nav">
|
|
<li><a class="navbar-link" href="https://syncthing.net/" target="_blank"><span class="fas fa-home"></span> <span translate>Home page</span></a></li>
|
|
<li><a class="navbar-link" href="{{docsURL()}}" target="_blank"><span class="fas fa-book"></span> <span translate>Documentation</span></a></li>
|
|
<li><a class="navbar-link" href="https://forum.syncthing.net" target="_blank"><span class="fas fa-question-circle"></span> <span translate>Support</span></a></li>
|
|
<li><a class="navbar-link" href="https://data.syncthing.net/" target="_blank"><span class="fas fa-bar-chart"></span> <span translate>Statistics</span></a></li>
|
|
<li><a class="navbar-link" href="https://github.com/syncthing/syncthing/releases" target="_blank"><span class="far fa-file-alt"></span> <span translate>Changelog</span></a></li>
|
|
<li><a class="navbar-link" href="https://github.com/syncthing/syncthing/issues" target="_blank"><span class="fas fa-bug"></span> <span translate>Bugs</span></a></li>
|
|
<li><a class="navbar-link" href="https://github.com/syncthing/syncthing" target="_blank"><span class="fas fa-wrench"></span> <span translate>Source Code</span></a></li>
|
|
<li><a class="navbar-link" href="https://twitter.com/syncthing" target="_blank"><span class="fab fa-twitter"></span> <span translate>Twitter</span></a></li>
|
|
</ul>
|
|
</div>
|
|
</nav>
|
|
|
|
<ng-include src="'syncthing/core/networkErrorDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/httpErrorDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/restartingDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/upgradingDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/shutdownDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/device/idqrModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/device/editDeviceModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/device/globalChangesModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/folder/editFolderModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/folder/restoreVersionsModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/folder/restoreVersionsConfirmation.html'"></ng-include>
|
|
<ng-include src="'syncthing/settings/settingsModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/settings/advancedSettingsModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/settings/discardChangesConfirmation.html'"></ng-include>
|
|
<ng-include src="'syncthing/usagereport/usageReportModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/usagereport/usageReportPreviewModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/transfer/neededFilesModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/transfer/failedFilesModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/transfer/remoteNeededFilesModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/transfer/localChangedFilesModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/upgradeModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/majorUpgradeModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/aboutModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/connectivityStatusModalView.html'"></ng-include>
|
|
<ng-include src="'syncthing/folder/removeFolderDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/folder/revertOverrideView.html'"></ng-include>
|
|
<ng-include src="'syncthing/device/removeDeviceDialogView.html'"></ng-include>
|
|
<ng-include src="'syncthing/core/logViewerModalView.html'"></ng-include>
|
|
|
|
<!-- vendor scripts -->
|
|
<script type="text/javascript" src="vendor/jquery/jquery-2.2.2.js"></script>
|
|
<script type="text/javascript" src="vendor/angular/angular.js"></script>
|
|
<script type="text/javascript" src="vendor/angular/angular-sanitize.js"></script>
|
|
<script type="text/javascript" src="vendor/angular/angular-translate.js"></script>
|
|
<script type="text/javascript" src="vendor/angular/angular-translate-loader-static-files.js"></script>
|
|
<script type="text/javascript" src="vendor/angular/angular-dirPagination.js"></script>
|
|
<script type="text/javascript" src="vendor/moment/moment.js"></script>
|
|
<script type="text/javascript" src="vendor/bootstrap/js/bootstrap.js"></script>
|
|
<script type="text/javascript" src="vendor/daterangepicker/daterangepicker.js"></script>
|
|
<script type="text/javascript" src="vendor/fancytree/jquery.fancytree-all-deps.js"></script>
|
|
<!-- / vendor scripts -->
|
|
|
|
<!-- gui application code -->
|
|
<script type="text/javascript" src="syncthing/core/module.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/alwaysNumberFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/basenameFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/binaryFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/localeNumberFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/percentFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/durationFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/eventService.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/identiconDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/languageSelectDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/lastErrorComponentFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/localeService.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/modalDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/metricFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/notificationDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/pathIsSubDirDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/popoverDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/syncthingController.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/tooltipDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/uncamelFilter.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/uniqueFolderDirective.js"></script>
|
|
<script type="text/javascript" src="syncthing/core/validDeviceidDirective.js"></script>
|
|
<script type="text/javascript" src="assets/lang/valid-langs.js"></script>
|
|
<script type="text/javascript" src="assets/lang/prettyprint.js"></script>
|
|
<script type="text/javascript" src="meta.js"></script>
|
|
<script type="text/javascript" src="syncthing/app.js"></script>
|
|
<!-- / gui application code -->
|
|
|
|
</body>
|
|
</html>
|