Compare commits
130 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f73518029c | ||
|
|
c782f21b0f | ||
|
|
0942fc7042 | ||
|
|
c1987237d0 | ||
|
|
cf85404dd2 | ||
|
|
6f226fa549 | ||
|
|
228ebea66c | ||
|
|
bb19834042 | ||
|
|
7870e89bc0 | ||
|
|
8888b9f446 | ||
|
|
f6e5c6f0de | ||
|
|
e7659ea36b | ||
|
|
7730ccd611 | ||
|
|
80890075e8 | ||
|
|
9b04004120 | ||
|
|
a73d7f14b7 | ||
|
|
ffe960a8c2 | ||
|
|
73c80732d9 | ||
|
|
8e2663bf7b | ||
|
|
e697e58164 | ||
|
|
c01496624a | ||
|
|
8906408a8f | ||
|
|
3ec7951380 | ||
|
|
b30aefa522 | ||
|
|
bc8a750dfe | ||
|
|
f1f7f17ade | ||
|
|
9182d33e1c | ||
|
|
7d836a3728 | ||
|
|
010d16fc1d | ||
|
|
fa89ba4665 | ||
|
|
a0752904c1 | ||
|
|
371718634b | ||
|
|
0f4f8751f2 | ||
|
|
ec45ee471f | ||
|
|
6fffcbac4e | ||
|
|
2948589fcd | ||
|
|
ecd0b2ee0d | ||
|
|
205f11d677 | ||
|
|
949f0f277f | ||
|
|
665e45889c | ||
|
|
8d87e0d5f9 | ||
|
|
46d80464d2 | ||
|
|
829ed9fb6d | ||
|
|
988d3e5bdd | ||
|
|
6eb3ab0635 | ||
|
|
c2e03bbfab | ||
|
|
608a0015ee | ||
|
|
f81857acce | ||
|
|
b1e0d5b39f | ||
|
|
68cf7a2173 | ||
|
|
89d1c06441 | ||
|
|
2bebb5f0f8 | ||
|
|
683b11d265 | ||
|
|
4d1b9dd211 | ||
|
|
b8f35ce932 | ||
|
|
a078f0b787 | ||
|
|
7401d16e45 | ||
|
|
958a44f95e | ||
|
|
e08239781f | ||
|
|
c29698dffa | ||
|
|
81de95632a | ||
|
|
7f2d221083 | ||
|
|
74b7cd8e81 | ||
|
|
6cb51b4eb4 | ||
|
|
f09bf3e1d0 | ||
|
|
6f345be3e4 | ||
|
|
ddd4ffa4ca | ||
|
|
deabc80fd7 | ||
|
|
b6a51bed51 | ||
|
|
0426629a59 | ||
|
|
0358e42d2c | ||
|
|
3768e3345f | ||
|
|
16e434be66 | ||
|
|
bf303c536a | ||
|
|
43a460993c | ||
|
|
7f0673ee70 | ||
|
|
4c3099a086 | ||
|
|
f0bc9167b1 | ||
|
|
23d646c456 | ||
|
|
76add9e527 | ||
|
|
c63cc5a2d2 | ||
|
|
25c8788390 | ||
|
|
aa52b07bb1 | ||
|
|
76b466f649 | ||
|
|
8ecc2da947 | ||
|
|
8650d2ffe7 | ||
|
|
34d7d2c8c4 | ||
|
|
201329abce | ||
|
|
f2b5dd3787 | ||
|
|
5072bbb2cb | ||
|
|
6b19ab6613 | ||
|
|
730be5ef6b | ||
|
|
46ee595389 | ||
|
|
dee465ab86 | ||
|
|
209f9fa77f | ||
|
|
ba8c09f454 | ||
|
|
16a34defc0 | ||
|
|
7d1e03075d | ||
|
|
1c25f6ee69 | ||
|
|
aa172b8bb5 | ||
|
|
4711e7bcd5 | ||
|
|
8a47aee137 | ||
|
|
190cb99a79 | ||
|
|
603203848a | ||
|
|
5e6f14b5dc | ||
|
|
976eb5583d | ||
|
|
b92152693f | ||
|
|
7ec24d9d77 | ||
|
|
20ebbf6611 | ||
|
|
ba7e71a7c3 | ||
|
|
8973c4598f | ||
|
|
18889ad725 | ||
|
|
73ccbe912f | ||
|
|
84e3a98303 | ||
|
|
7dd5b34d42 | ||
|
|
4470d0a704 | ||
|
|
a76e01d2b7 | ||
|
|
2697093ac1 | ||
|
|
59f9964e80 | ||
|
|
1516d9932b | ||
|
|
fcb115f42d | ||
|
|
e410272e6b | ||
|
|
87f1881b42 | ||
|
|
c0d85f3d85 | ||
|
|
98d79b8ed9 | ||
|
|
fe80730bb1 | ||
|
|
6c8ee96e6a | ||
|
|
b521dec8f9 | ||
|
|
e9baf0c4b6 | ||
|
|
e1a6f593e1 |
@@ -1,3 +1,3 @@
|
|||||||
*
|
*
|
||||||
!.docker.json
|
!docker/*
|
||||||
!filebrowser
|
!filebrowser
|
||||||
12
.github/ISSUE_TEMPLATE/bug_report.md
vendored
12
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -4,19 +4,19 @@ about: Create a report to help us improve
|
|||||||
---
|
---
|
||||||
|
|
||||||
**Description**
|
**Description**
|
||||||
A clear and concise description of what the issue is about. What are you trying to do?
|
<!-- A clear and concise description of what the issue is about. What are you trying to do? -->
|
||||||
|
|
||||||
**Expected behaviour**
|
**Expected behaviour**
|
||||||
What did you expect to happen?
|
<!-- What did you expect to happen? -->
|
||||||
|
|
||||||
**What is happening instead?**
|
**What is happening instead?**
|
||||||
Please, give full error messages and/or log.
|
<!-- Please, give full error messages and/or log. -->
|
||||||
|
|
||||||
**Additional context**
|
**Additional context**
|
||||||
Add any other context about the problem here. If applicable, add screenshots to help explain your problem.
|
<!-- Add any other context about the problem here. If applicable, add screenshots to help explain your problem. -->
|
||||||
|
|
||||||
**How to reproduce?**
|
**How to reproduce?**
|
||||||
Tell us how to reproduce this issue. How can someone who is starting from scratch reproduce this behaviour as minimally as possible?
|
<!-- Tell us how to reproduce this issue. How can someone who is starting from scratch reproduce this behaviour as minimally as possible? -->
|
||||||
|
|
||||||
**Files**
|
**Files**
|
||||||
A list of relevant files for this issue. Large files can be uploaded one-by-one or in a tarball/zipfile.
|
<!-- A list of relevant files for this issue. Large files can be uploaded one-by-one or in a tarball/zipfile. -->
|
||||||
|
|||||||
8
.github/ISSUE_TEMPLATE/feature_request.md
vendored
8
.github/ISSUE_TEMPLATE/feature_request.md
vendored
@@ -4,13 +4,13 @@ about: Suggest an idea for this project
|
|||||||
---
|
---
|
||||||
|
|
||||||
**Is your feature request related to a problem? Please describe.**
|
**Is your feature request related to a problem? Please describe.**
|
||||||
Add a clear and concise description of what the problem is. E.g. *I'm always frustrated when [...]*
|
<!-- Add a clear and concise description of what the problem is. E.g. *I'm always frustrated when [...]* -->
|
||||||
|
|
||||||
**Describe the solution you'd like**
|
**Describe the solution you'd like**
|
||||||
Add a clear and concise description of what you want to happen.
|
<!-- Add a clear and concise description of what you want to happen. -->
|
||||||
|
|
||||||
**Describe alternatives you've considered**
|
**Describe alternatives you've considered**
|
||||||
Add a clear and concise description of any alternative solutions or features you've considered.
|
<!-- Add a clear and concise description of any alternative solutions or features you've considered. -->
|
||||||
|
|
||||||
**Additional context**
|
**Additional context**
|
||||||
Add any other context or screenshots about the feature request here.
|
<!-- Add any other context or screenshots about the feature request here. -->
|
||||||
|
|||||||
4
.github/PULL_REQUEST_TEMPLATE.md
vendored
4
.github/PULL_REQUEST_TEMPLATE.md
vendored
@@ -1,6 +1,8 @@
|
|||||||
**Description**
|
**Description**
|
||||||
|
<!--
|
||||||
Please explain the changes you made here.
|
Please explain the changes you made here.
|
||||||
If the feature changes current behaviour, explain why your solution is better.
|
If the feature changes current behaviour, explain why your solution is better.
|
||||||
|
-->
|
||||||
|
|
||||||
:rotating_light: Before submitting your PR, please read [community](https://github.com/filebrowser/community), and indicate which issues (in any of the repos) are either fixed or closed by this PR. See [GitHub Help: Closing issues using keywords](https://help.github.com/articles/closing-issues-via-commit-messages/).
|
:rotating_light: Before submitting your PR, please read [community](https://github.com/filebrowser/community), and indicate which issues (in any of the repos) are either fixed or closed by this PR. See [GitHub Help: Closing issues using keywords](https://help.github.com/articles/closing-issues-via-commit-messages/).
|
||||||
|
|
||||||
@@ -11,6 +13,8 @@ If the feature changes current behaviour, explain why your solution is better.
|
|||||||
- [ ] AVOID breaking the continuous integration build.
|
- [ ] AVOID breaking the continuous integration build.
|
||||||
|
|
||||||
**Further comments**
|
**Further comments**
|
||||||
|
<!--
|
||||||
If this is a relatively large or complex change, kick off the discussion by explaining why you chose the solution you did, what alternatives you considered, etc.
|
If this is a relatively large or complex change, kick off the discussion by explaining why you chose the solution you did, what alternatives you considered, etc.
|
||||||
|
|
||||||
:heart: Thank you!
|
:heart: Thank you!
|
||||||
|
-->
|
||||||
|
|||||||
60
.github/workflows/main.yaml
vendored
60
.github/workflows/main.yaml
vendored
@@ -9,29 +9,63 @@ on:
|
|||||||
pull_request:
|
pull_request:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
# linters
|
||||||
|
lint-frontend:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '14'
|
||||||
|
- run: make lint-frontend
|
||||||
|
lint-backend:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-go@v2
|
||||||
|
with:
|
||||||
|
go-version: 1.17
|
||||||
|
- run: make lint-backend
|
||||||
|
lint-commits:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '14'
|
||||||
|
- run: make lint-commits
|
||||||
lint:
|
lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [lint-frontend, lint-backend, lint-commits]
|
||||||
|
steps:
|
||||||
|
- run: echo "done"
|
||||||
|
|
||||||
|
# tests
|
||||||
|
test-frontend:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: 1.16
|
|
||||||
- uses: actions/setup-node@v2
|
- uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: '14'
|
node-version: '14'
|
||||||
- run: npm i -g commitlint
|
- run: make test-frontend
|
||||||
- run: make lint
|
test-backend:
|
||||||
test:
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-go@v2
|
- uses: actions/setup-go@v2
|
||||||
with:
|
with:
|
||||||
go-version: 1.16
|
go-version: 1.17
|
||||||
- uses: actions/setup-node@v2
|
- run: make test-backend
|
||||||
with:
|
test:
|
||||||
node-version: '14'
|
runs-on: ubuntu-latest
|
||||||
- run: make test
|
needs: [test-frontend, test-backend]
|
||||||
|
steps:
|
||||||
|
- run: echo "done"
|
||||||
|
|
||||||
|
# release
|
||||||
release:
|
release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [lint, test]
|
needs: [lint, test]
|
||||||
@@ -42,7 +76,7 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- uses: actions/setup-go@v2
|
- uses: actions/setup-go@v2
|
||||||
with:
|
with:
|
||||||
go-version: 1.16
|
go-version: 1.17
|
||||||
- uses: actions/setup-node@v2
|
- uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: '14'
|
node-version: '14'
|
||||||
@@ -50,7 +84,7 @@ jobs:
|
|||||||
uses: docker/setup-qemu-action@v1
|
uses: docker/setup-qemu-action@v1
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v1
|
uses: docker/setup-buildx-action@v1
|
||||||
- name: Build fronetend
|
- name: Build frontend
|
||||||
run: make build-frontend
|
run: make build-frontend
|
||||||
- name: Login to Docker Hub
|
- name: Login to Docker Hub
|
||||||
uses: docker/login-action@v1
|
uses: docker/login-action@v1
|
||||||
|
|||||||
6
.gitignore
vendored
6
.gitignore
vendored
@@ -1,11 +1,11 @@
|
|||||||
*.db
|
*.db
|
||||||
*.lock
|
|
||||||
*.bak
|
*.bak
|
||||||
_old
|
_old
|
||||||
rice-box.go
|
rice-box.go
|
||||||
.idea/
|
.idea/
|
||||||
filebrowser
|
/filebrowser
|
||||||
filebrowser.exe
|
/filebrowser.exe
|
||||||
|
/dist
|
||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
node_modules
|
node_modules
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ linters-settings:
|
|||||||
funlen:
|
funlen:
|
||||||
lines: 100
|
lines: 100
|
||||||
statements: 50
|
statements: 50
|
||||||
|
gci:
|
||||||
|
local-prefixes: github.com/filebrowser/filebrowser
|
||||||
goconst:
|
goconst:
|
||||||
min-len: 2
|
min-len: 2
|
||||||
min-occurrences: 2
|
min-occurrences: 2
|
||||||
@@ -26,8 +28,6 @@ linters-settings:
|
|||||||
min-complexity: 15
|
min-complexity: 15
|
||||||
goimports:
|
goimports:
|
||||||
local-prefixes: github.com/filebrowser/filebrowser
|
local-prefixes: github.com/filebrowser/filebrowser
|
||||||
golint:
|
|
||||||
min-confidence: 0
|
|
||||||
gomnd:
|
gomnd:
|
||||||
settings:
|
settings:
|
||||||
mnd:
|
mnd:
|
||||||
@@ -58,26 +58,25 @@ linters:
|
|||||||
- dogsled
|
- dogsled
|
||||||
- dupl
|
- dupl
|
||||||
- errcheck
|
- errcheck
|
||||||
|
- exportloopref
|
||||||
|
- exhaustive
|
||||||
- funlen
|
- funlen
|
||||||
- gochecknoinits
|
- gochecknoinits
|
||||||
- goconst
|
- goconst
|
||||||
- gocritic
|
- gocritic
|
||||||
- gocyclo
|
- gocyclo
|
||||||
- goimports
|
- goimports
|
||||||
- golint
|
|
||||||
- gomnd
|
- gomnd
|
||||||
- goprintffuncname
|
- goprintffuncname
|
||||||
- gosec
|
- gosec
|
||||||
- gosimple
|
- gosimple
|
||||||
- govet
|
- govet
|
||||||
- ineffassign
|
- ineffassign
|
||||||
- interfacer
|
|
||||||
- lll
|
- lll
|
||||||
- misspell
|
- misspell
|
||||||
- nakedret
|
- nakedret
|
||||||
- nolintlint
|
- nolintlint
|
||||||
- rowserrcheck
|
- rowserrcheck
|
||||||
- scopelint
|
|
||||||
- staticcheck
|
- staticcheck
|
||||||
- structcheck
|
- structcheck
|
||||||
- stylecheck
|
- stylecheck
|
||||||
@@ -89,19 +88,6 @@ linters:
|
|||||||
- whitespace
|
- whitespace
|
||||||
- prealloc
|
- prealloc
|
||||||
|
|
||||||
# don't enable:
|
|
||||||
# - asciicheck
|
|
||||||
# - exhaustive (TODO: enable after next release; current release at time of writing is v1.27)
|
|
||||||
# - gochecknoglobals
|
|
||||||
# - gocognit
|
|
||||||
# - godot
|
|
||||||
# - godox
|
|
||||||
# - goerr113
|
|
||||||
# - maligned
|
|
||||||
# - nestif
|
|
||||||
# - testpackage
|
|
||||||
# - wsl
|
|
||||||
|
|
||||||
issues:
|
issues:
|
||||||
exclude-rules:
|
exclude-rules:
|
||||||
- path: cmd/.*.go
|
- path: cmd/.*.go
|
||||||
@@ -118,6 +104,9 @@ issues:
|
|||||||
- text: "Auther"
|
- text: "Auther"
|
||||||
linters:
|
linters:
|
||||||
- misspell
|
- misspell
|
||||||
|
- text: "strconv.Parse"
|
||||||
|
linters:
|
||||||
|
- gomnd
|
||||||
|
|
||||||
run:
|
run:
|
||||||
skip-dirs:
|
skip-dirs:
|
||||||
|
|||||||
110
.goreleaser.yml
110
.goreleaser.yml
@@ -41,7 +41,7 @@ archives:
|
|||||||
dockers:
|
dockers:
|
||||||
-
|
-
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
use_buildx: true
|
use: buildx
|
||||||
build_flag_templates:
|
build_flag_templates:
|
||||||
- "--pull"
|
- "--pull"
|
||||||
- "--label=org.opencontainers.image.created={{.Date}}"
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
@@ -56,10 +56,10 @@ dockers:
|
|||||||
- "filebrowser/filebrowser:{{ .Tag }}-amd64"
|
- "filebrowser/filebrowser:{{ .Tag }}-amd64"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-amd64"
|
- "filebrowser/filebrowser:v{{ .Major }}-amd64"
|
||||||
extra_files:
|
extra_files:
|
||||||
- .docker.json
|
- docker_config.json
|
||||||
-
|
-
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
use_buildx: true
|
use: buildx
|
||||||
build_flag_templates:
|
build_flag_templates:
|
||||||
- "--pull"
|
- "--pull"
|
||||||
- "--label=org.opencontainers.image.created={{.Date}}"
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
@@ -74,10 +74,10 @@ dockers:
|
|||||||
- "filebrowser/filebrowser:{{ .Tag }}-arm64"
|
- "filebrowser/filebrowser:{{ .Tag }}-arm64"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-arm64"
|
- "filebrowser/filebrowser:v{{ .Major }}-arm64"
|
||||||
extra_files:
|
extra_files:
|
||||||
- .docker.json
|
- docker_config.json
|
||||||
-
|
-
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
use_buildx: true
|
use: buildx
|
||||||
build_flag_templates:
|
build_flag_templates:
|
||||||
- "--pull"
|
- "--pull"
|
||||||
- "--label=org.opencontainers.image.created={{.Date}}"
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
@@ -93,10 +93,10 @@ dockers:
|
|||||||
- "filebrowser/filebrowser:{{ .Tag }}-armv6"
|
- "filebrowser/filebrowser:{{ .Tag }}-armv6"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-armv6"
|
- "filebrowser/filebrowser:v{{ .Major }}-armv6"
|
||||||
extra_files:
|
extra_files:
|
||||||
- .docker.json
|
- docker_config.json
|
||||||
-
|
-
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
use_buildx: true
|
use: buildx
|
||||||
build_flag_templates:
|
build_flag_templates:
|
||||||
- "--pull"
|
- "--pull"
|
||||||
- "--label=org.opencontainers.image.created={{.Date}}"
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
@@ -112,7 +112,82 @@ dockers:
|
|||||||
- "filebrowser/filebrowser:{{ .Tag }}-armv7"
|
- "filebrowser/filebrowser:{{ .Tag }}-armv7"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-armv7"
|
- "filebrowser/filebrowser:v{{ .Major }}-armv7"
|
||||||
extra_files:
|
extra_files:
|
||||||
- .docker.json
|
- docker_config.json
|
||||||
|
## s6 based docker images
|
||||||
|
-
|
||||||
|
dockerfile: Dockerfile.s6
|
||||||
|
use: buildx
|
||||||
|
build_flag_templates:
|
||||||
|
- "--pull"
|
||||||
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
|
- "--label=org.opencontainers.image.name={{.ProjectName}}"
|
||||||
|
- "--label=org.opencontainers.image.revision={{.FullCommit}}"
|
||||||
|
- "--label=org.opencontainers.image.version={{.Version}}"
|
||||||
|
- "--label=org.opencontainers.image.source={{.GitURL}}"
|
||||||
|
- "--platform=linux/amd64"
|
||||||
|
goos: linux
|
||||||
|
goarch: amd64
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-amd64-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-amd64-s6"
|
||||||
|
extra_files:
|
||||||
|
- docker/root
|
||||||
|
-
|
||||||
|
dockerfile: Dockerfile.s6.aarch64
|
||||||
|
use: buildx
|
||||||
|
build_flag_templates:
|
||||||
|
- "--pull"
|
||||||
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
|
- "--label=org.opencontainers.image.name={{.ProjectName}}"
|
||||||
|
- "--label=org.opencontainers.image.revision={{.FullCommit}}"
|
||||||
|
- "--label=org.opencontainers.image.version={{.Version}}"
|
||||||
|
- "--label=org.opencontainers.image.source={{.GitURL}}"
|
||||||
|
- "--platform=linux/arm64"
|
||||||
|
goos: linux
|
||||||
|
goarch: arm64
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-arm64-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-arm64-s6"
|
||||||
|
extra_files:
|
||||||
|
- docker/root
|
||||||
|
-
|
||||||
|
dockerfile: Dockerfile.s6.armhf
|
||||||
|
use: buildx
|
||||||
|
build_flag_templates:
|
||||||
|
- "--pull"
|
||||||
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
|
- "--label=org.opencontainers.image.name={{.ProjectName}}"
|
||||||
|
- "--label=org.opencontainers.image.revision={{.FullCommit}}"
|
||||||
|
- "--label=org.opencontainers.image.version={{.Version}}"
|
||||||
|
- "--label=org.opencontainers.image.source={{.GitURL}}"
|
||||||
|
- "--platform=linux/arm/v6"
|
||||||
|
goos: linux
|
||||||
|
goarch: arm
|
||||||
|
goarm: '6'
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv6-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-armv6-s6"
|
||||||
|
extra_files:
|
||||||
|
- docker/root
|
||||||
|
-
|
||||||
|
dockerfile: Dockerfile.s6.armhf
|
||||||
|
use: buildx
|
||||||
|
build_flag_templates:
|
||||||
|
- "--pull"
|
||||||
|
- "--label=org.opencontainers.image.created={{.Date}}"
|
||||||
|
- "--label=org.opencontainers.image.name={{.ProjectName}}"
|
||||||
|
- "--label=org.opencontainers.image.revision={{.FullCommit}}"
|
||||||
|
- "--label=org.opencontainers.image.version={{.Version}}"
|
||||||
|
- "--label=org.opencontainers.image.source={{.GitURL}}"
|
||||||
|
- "--platform=linux/arm/v7"
|
||||||
|
goos: linux
|
||||||
|
goarch: arm
|
||||||
|
goarm: '7'
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv7-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-armv7-s6"
|
||||||
|
extra_files:
|
||||||
|
- docker/root
|
||||||
docker_manifests:
|
docker_manifests:
|
||||||
- name_template: "filebrowser/filebrowser:latest"
|
- name_template: "filebrowser/filebrowser:latest"
|
||||||
image_templates:
|
image_templates:
|
||||||
@@ -132,6 +207,25 @@ docker_manifests:
|
|||||||
- "filebrowser/filebrowser:v{{ .Major }}-arm64"
|
- "filebrowser/filebrowser:v{{ .Major }}-arm64"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-armv6"
|
- "filebrowser/filebrowser:v{{ .Major }}-armv6"
|
||||||
- "filebrowser/filebrowser:v{{ .Major }}-armv7"
|
- "filebrowser/filebrowser:v{{ .Major }}-armv7"
|
||||||
|
## s6 image manifests
|
||||||
|
- name_template: "filebrowser/filebrowser:s6"
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-amd64-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-arm64-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv6-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv7-s6"
|
||||||
|
- name_template: "filebrowser/filebrowser:{{ .Tag }}-s6"
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-amd64-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-arm64-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv6-s6"
|
||||||
|
- "filebrowser/filebrowser:{{ .Tag }}-armv7-s6"
|
||||||
|
- name_template: "filebrowser/filebrowser:v{{ .Major }}-s6"
|
||||||
|
image_templates:
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-amd64-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-arm64-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-armv6-s6"
|
||||||
|
- "filebrowser/filebrowser:v{{ .Major }}-armv7-s6"
|
||||||
brews:
|
brews:
|
||||||
- name: filebrowser
|
- name: filebrowser
|
||||||
tap:
|
tap:
|
||||||
|
|||||||
181
CHANGELOG.md
181
CHANGELOG.md
@@ -2,6 +2,187 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
||||||
|
|
||||||
|
## [2.21.0](https://github.com/filebrowser/filebrowser/compare/v2.20.1...v2.21.0) (2022-02-21)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add colorized file type icons ([2948589](https://github.com/filebrowser/filebrowser/commit/2948589fcde6d1dca7f3ea52a621d8213fa3300c))
|
||||||
|
* add gallery view mode ([8888b9f](https://github.com/filebrowser/filebrowser/commit/8888b9f44640394df9e3583db4392472d7027a4b))
|
||||||
|
* add Ukrainian translation / update Russian translation ([#1753](https://github.com/filebrowser/filebrowser/issues/1753)) ([665e458](https://github.com/filebrowser/filebrowser/commit/665e45889cd333f1e3500e4bf38d15d229c9fe2a))
|
||||||
|
* add upload file list with progress ([#1825](https://github.com/filebrowser/filebrowser/issues/1825)) ([cf85404](https://github.com/filebrowser/filebrowser/commit/cf85404dd25cd7fdd73aa32878b4dc5f85ee3e96))
|
||||||
|
* smaller column width to fit 2 columns in landscape mobiles ([7870e89](https://github.com/filebrowser/filebrowser/commit/7870e89bc04f1494f2705795476b5f1c9d621e38))
|
||||||
|
* use real image path to calculate cache key ([c198723](https://github.com/filebrowser/filebrowser/commit/c1987237d05adcce77c614e5247a181ae5cdfacd))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* correctly handle non-ascii passwords for shared resources ([c782f21](https://github.com/filebrowser/filebrowser/commit/c782f21b0fa4511a15e7015117d075eaf5ea332c))
|
||||||
|
* don't expose scope for non-admin users ([0942fc7](https://github.com/filebrowser/filebrowser/commit/0942fc7042fd949cce91855169d0bcf16eb75771))
|
||||||
|
* open all the pdf files correctly ([#1742](https://github.com/filebrowser/filebrowser/issues/1742)) ([949f0f2](https://github.com/filebrowser/filebrowser/commit/949f0f277f6004904b3edfa716a8365ec93fa0fa))
|
||||||
|
|
||||||
|
|
||||||
|
### Build
|
||||||
|
|
||||||
|
* **deps:** bump browserslist from 4.16.3 to 4.19.1 in /frontend ([8089007](https://github.com/filebrowser/filebrowser/commit/80890075e802e2a4217edbb01d6417122d702f5e))
|
||||||
|
* **deps:** bump dns-packet from 1.3.1 to 1.3.4 in /frontend ([a73d7f1](https://github.com/filebrowser/filebrowser/commit/a73d7f14b787935c6ebe525dba64b65f8ed733e2))
|
||||||
|
* **deps:** bump follow-redirects from 1.13.3 to 1.14.8 in /frontend ([f1f7f17](https://github.com/filebrowser/filebrowser/commit/f1f7f17ade8d40fc6cfb22c79960bce299876b56))
|
||||||
|
* **deps:** bump hosted-git-info from 2.8.8 to 2.8.9 in /frontend ([e7659ea](https://github.com/filebrowser/filebrowser/commit/e7659ea36bdf780ce17005f7170a2fef02a2d5e5))
|
||||||
|
* **deps:** bump path-parse from 1.0.6 to 1.0.7 in /frontend ([c014966](https://github.com/filebrowser/filebrowser/commit/c01496624a7ebfc8a7c256bd919a400367281cbb))
|
||||||
|
* **deps:** bump postcss from 7.0.35 to 7.0.39 in /frontend ([9182d33](https://github.com/filebrowser/filebrowser/commit/9182d33e1cc375473fb18989a92d20252884f096))
|
||||||
|
* **deps:** bump ssri from 6.0.1 to 6.0.2 in /frontend ([3717186](https://github.com/filebrowser/filebrowser/commit/371718634b11f32e68165f31c51b6b1139c829ec))
|
||||||
|
* **deps:** bump tar from 6.1.0 to 6.1.11 in /frontend ([010d16f](https://github.com/filebrowser/filebrowser/commit/010d16fc1d8f0200e5662943aef17ee89c5877b7))
|
||||||
|
* **deps:** bump url-parse from 1.5.1 to 1.5.4 in /frontend ([8906408](https://github.com/filebrowser/filebrowser/commit/8906408a8f0ed86d1e11ea90fc573b36815c9c0d))
|
||||||
|
* **deps:** bump url-parse from 1.5.4 to 1.5.7 in /frontend ([228ebea](https://github.com/filebrowser/filebrowser/commit/228ebea66cc871b33459406590a80ef906298e7d))
|
||||||
|
* **deps:** bump ws from 6.2.1 to 6.2.2 in /frontend ([73c8073](https://github.com/filebrowser/filebrowser/commit/73c80732d934bc8802a6d7c7a559cad37df405f0))
|
||||||
|
|
||||||
|
### [2.20.1](https://github.com/filebrowser/filebrowser/compare/v2.20.0...v2.20.1) (2021-12-21)
|
||||||
|
|
||||||
|
|
||||||
|
### Build
|
||||||
|
|
||||||
|
* revert to using the default alpine based docker image ([46d8046](https://github.com/filebrowser/filebrowser/commit/46d80464d2a67927b06a11b83fb137ad364a90ed))
|
||||||
|
|
||||||
|
## [2.20.0](https://github.com/filebrowser/filebrowser/compare/v2.19.0...v2.20.0) (2021-12-20)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* detect multiple subtitle languages ([#1723](https://github.com/filebrowser/filebrowser/issues/1723)) ([c2e03bb](https://github.com/filebrowser/filebrowser/commit/c2e03bbfab97fc6716bcdd59158e9d5129bf0ea7))
|
||||||
|
* use linuxserver based docker image ([b8f35ce](https://github.com/filebrowser/filebrowser/commit/b8f35ce9322c2b0dbf954cfd3ff584bc9f742fdd))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* set correct default database path in the config ([988d3e5](https://github.com/filebrowser/filebrowser/commit/988d3e5bdd224509ddc2f08444560e3087e9c67d))
|
||||||
|
* upgrade vulnerable versions of the library ([6eb3ab0](https://github.com/filebrowser/filebrowser/commit/6eb3ab063509a015ad630ab704ae3791461d0982))
|
||||||
|
|
||||||
|
|
||||||
|
### Build
|
||||||
|
|
||||||
|
* refactor makefile ([f81857a](https://github.com/filebrowser/filebrowser/commit/f81857acce25936a700945db5ef4af545eaeb1cf))
|
||||||
|
* remove deprecated goreleaser use_buildx param ([4d1b9dd](https://github.com/filebrowser/filebrowser/commit/4d1b9dd2112002a93bb26cece07dcfd81c31dc2c))
|
||||||
|
|
||||||
|
## [2.19.0](https://github.com/filebrowser/filebrowser/compare/v2.18.0...v2.19.0) (2021-11-24)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* prefetch previous and next images in preview. ([#1627](https://github.com/filebrowser/filebrowser/issues/1627)) ([7401d16](https://github.com/filebrowser/filebrowser/commit/7401d16e457bb232fd7dd7ef427e8960d465705c))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* empty file listing on share ([e082397](https://github.com/filebrowser/filebrowser/commit/e08239781f61e7bb25d9b8c5c6cce90f34621a76))
|
||||||
|
* relative font sizes ([c29698d](https://github.com/filebrowser/filebrowser/commit/c29698dffac769077ab7c7869569a902979ee3d7))
|
||||||
|
|
||||||
|
## [2.18.0](https://github.com/filebrowser/filebrowser/compare/v2.17.2...v2.18.0) (2021-10-31)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add ability to select file modified time format ([#1536](https://github.com/filebrowser/filebrowser/issues/1536)) ([0426629](https://github.com/filebrowser/filebrowser/commit/0426629a59c712849570d3e29956948ae7725a4a))
|
||||||
|
* add manifest theme color param ([#1542](https://github.com/filebrowser/filebrowser/issues/1542)) ([0358e42](https://github.com/filebrowser/filebrowser/commit/0358e42d2c206732fffa77714f5a66f4fe50a69d))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* back button behaviour in preview ([#1573](https://github.com/filebrowser/filebrowser/issues/1573)) ([deabc80](https://github.com/filebrowser/filebrowser/commit/deabc80fd7670983039dfcd29531b45002ca5d9e))
|
||||||
|
* fix sidebar navigation on mobile devices ([#1618](https://github.com/filebrowser/filebrowser/issues/1618)) ([f09bf3e](https://github.com/filebrowser/filebrowser/commit/f09bf3e1d076b27d29ba8a91cf448a99993bc444))
|
||||||
|
* search box is misaligned when the browser preferred font size is other than 16px ([#1613](https://github.com/filebrowser/filebrowser/issues/1613)) ([6f345be](https://github.com/filebrowser/filebrowser/commit/6f345be3e47ba57ecc1eb9a62587ab949078c125))
|
||||||
|
* security issue in command runner (closes [#1621](https://github.com/filebrowser/filebrowser/issues/1621)) ([74b7cd8](https://github.com/filebrowser/filebrowser/commit/74b7cd8e81840537a8206317344f118093153e8d))
|
||||||
|
* set correct editor height regardless of preferred font size ([#1614](https://github.com/filebrowser/filebrowser/issues/1614)) ([ddd4ffa](https://github.com/filebrowser/filebrowser/commit/ddd4ffa4caa6b292a3a644ecd897aba1237c7503))
|
||||||
|
* zoom pics when dlclick at first time ([#1561](https://github.com/filebrowser/filebrowser/issues/1561)) ([b6a51be](https://github.com/filebrowser/filebrowser/commit/b6a51bed516814944f8aa41440652242d57824c5))
|
||||||
|
|
||||||
|
### [2.17.2](https://github.com/filebrowser/filebrowser/compare/v2.17.1...v2.17.2) (2021-08-27)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* bug with inlineLink not creating url properly ([#1515](https://github.com/filebrowser/filebrowser/issues/1515)) ([43a4609](https://github.com/filebrowser/filebrowser/commit/43a460993c3f0d158b876db4b20caa7963e9f361))
|
||||||
|
|
||||||
|
### [2.17.1](https://github.com/filebrowser/filebrowser/compare/v2.17.0...v2.17.1) (2021-08-23)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* internal server error if --disable-preview-resize flag is set (closes [#1510](https://github.com/filebrowser/filebrowser/issues/1510)) ([4c3099a](https://github.com/filebrowser/filebrowser/commit/4c3099a086c206dcb3bc70ee8c8da02eee61c30b))
|
||||||
|
|
||||||
|
## [2.17.0](https://github.com/filebrowser/filebrowser/compare/v2.16.1...v2.17.0) (2021-08-21)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* open file option on preview ([76add9e](https://github.com/filebrowser/filebrowser/commit/76add9e5274b0373c6b983e3b20e387a14ea6c9e))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* 401 error in share view open file button ([#1495](https://github.com/filebrowser/filebrowser/issues/1495)) ([25c8788](https://github.com/filebrowser/filebrowser/commit/25c87883908babde073390a2e2320a8e5880a87c))
|
||||||
|
* escape quote on index template ([23d646c](https://github.com/filebrowser/filebrowser/commit/23d646c456876d06cf48e71c1e57b69de99511f0)), closes [#1501](https://github.com/filebrowser/filebrowser/issues/1501)
|
||||||
|
* file caching directive ([c63cc5a](https://github.com/filebrowser/filebrowser/commit/c63cc5a2d25909cc4e2f2e7235f276ec66c32bf2))
|
||||||
|
|
||||||
|
### [2.16.1](https://github.com/filebrowser/filebrowser/compare/v2.16.0...v2.16.1) (2021-08-04)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* check symlink target type (closes [#1488](https://github.com/filebrowser/filebrowser/issues/1488)) ([76b466f](https://github.com/filebrowser/filebrowser/commit/76b466f6492e74cf13e66a33e7e5f597ac92b240))
|
||||||
|
|
||||||
|
## [2.16.0](https://github.com/filebrowser/filebrowser/compare/v2.15.0...v2.16.0) (2021-07-26)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* browser cache directives ([190cb99](https://github.com/filebrowser/filebrowser/commit/190cb99a79a0d438eca2da13539f8c6449ad73ac))
|
||||||
|
* display error messages on settings ([6032038](https://github.com/filebrowser/filebrowser/commit/603203848a8b2221158088b6d849609db4c0c46c))
|
||||||
|
* file name on page title ([16a34de](https://github.com/filebrowser/filebrowser/commit/16a34defc02554a77c6ac47b9e17e69d098a09fe))
|
||||||
|
* gzip encoding for static js files ([aa172b8](https://github.com/filebrowser/filebrowser/commit/aa172b8bb5f17d5f5cb9666bfb5ee650d8091fb5))
|
||||||
|
* loading spinner on views navigation ([976eb55](https://github.com/filebrowser/filebrowser/commit/976eb5583dae474125fd7ddec5dc19b6c291f98f))
|
||||||
|
* message for connection error ([5e6f14b](https://github.com/filebrowser/filebrowser/commit/5e6f14b5dcb9c5efdf526f1346e09c2d0b2f6974))
|
||||||
|
* mod time title on file info ([7d1e030](https://github.com/filebrowser/filebrowser/commit/7d1e03075d2c27148f60813defa0f68403d1d3c2))
|
||||||
|
* open file option on share ([1c25f6e](https://github.com/filebrowser/filebrowser/commit/1c25f6ee69bd71eed82af7020006d0e27537a967))
|
||||||
|
* show more button on share ([ba8c09f](https://github.com/filebrowser/filebrowser/commit/ba8c09f454feeadf4a1e97547a34151a81b389d5))
|
||||||
|
* support for IE11 browser ([7ec24d9](https://github.com/filebrowser/filebrowser/commit/7ec24d9d7794fa37825f64ca2d1575f568fb1362))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* break resource create/update handlers on error (closes [#1464](https://github.com/filebrowser/filebrowser/issues/1464)) ([5072bbb](https://github.com/filebrowser/filebrowser/commit/5072bbb2cbf5b29d041629faa8367f15e4d145a2))
|
||||||
|
* copying files with special characters ([20ebbf6](https://github.com/filebrowser/filebrowser/commit/20ebbf6611b734371426fb1b9cb5e388be90bf7e))
|
||||||
|
* delete image cache when moving ([8973c45](https://github.com/filebrowser/filebrowser/commit/8973c4598ff817647f1f1ad6ee36480054cd2776))
|
||||||
|
* don't remove files on unsuccessful updates (closes [#1456](https://github.com/filebrowser/filebrowser/issues/1456)) ([6b19ab6](https://github.com/filebrowser/filebrowser/commit/6b19ab6613b12be7f075299cd98f4b41d43827c7))
|
||||||
|
* failure on broken symlink deletion ([8650d2f](https://github.com/filebrowser/filebrowser/commit/8650d2ffe7a29cbafa800efcecbf6a61598a9f0c))
|
||||||
|
* inconsistent double click on listing item ([ba7e71a](https://github.com/filebrowser/filebrowser/commit/ba7e71a7c3b0cc71012e5adf94b1c642e554972e))
|
||||||
|
* no items displayed on file listing ([18889ad](https://github.com/filebrowser/filebrowser/commit/18889ad725f7f7e5a7e3f7abcf156487556dbeaf))
|
||||||
|
* omit file content ([209f9fa](https://github.com/filebrowser/filebrowser/commit/209f9fa77f751054512355f2b74b9b7258465d0b))
|
||||||
|
* short commit sha and typo fix in Makefile ([#1411](https://github.com/filebrowser/filebrowser/issues/1411)) ([46ee595](https://github.com/filebrowser/filebrowser/commit/46ee59538914dc2859f0da6b32e2d062d0a01b10))
|
||||||
|
|
||||||
|
## [2.15.0](https://github.com/filebrowser/filebrowser/compare/v2.14.1...v2.15.0) (2021-04-06)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add EXIF thumbnail support for JPEG files ([#1234](https://github.com/filebrowser/filebrowser/issues/1234)) ([7dd5b34](https://github.com/filebrowser/filebrowser/commit/7dd5b34d425dfbc2782152310483cbecf85c800a))
|
||||||
|
* dynamic autoplay on previewer ([a76e01d](https://github.com/filebrowser/filebrowser/commit/a76e01d2b78a785f3665a8b3532c7cc566bfabce))
|
||||||
|
* dynamic item count on file listing ([6c8ee96](https://github.com/filebrowser/filebrowser/commit/6c8ee96e6a21fae5d4608bdc7a5c5a161d7dafd3))
|
||||||
|
* dynamic zoom limit on previewer ([e410272](https://github.com/filebrowser/filebrowser/commit/e410272e6be6a0b660efe8d4eee6c6e9dd834cc5))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* buttons without permission on header ([1516d99](https://github.com/filebrowser/filebrowser/commit/1516d9932bf9926ac8b4cb3e738a5f51e80d5b1d))
|
||||||
|
* check modify permission on file overwrite ([59f9964](https://github.com/filebrowser/filebrowser/commit/59f9964e80c8233775f27be33a4c16a31bfe848a))
|
||||||
|
* empty archive name on directory download ([2697093](https://github.com/filebrowser/filebrowser/commit/2697093ac151f74eea3022951d128acfe04d1dcf))
|
||||||
|
* empty text file on editor ([e9baf0c](https://github.com/filebrowser/filebrowser/commit/e9baf0c4b688fab291cdc842ec464c7a7a816499))
|
||||||
|
* error causes panic on upload ([e1a6f59](https://github.com/filebrowser/filebrowser/commit/e1a6f593e1824e7fa4345a61dff5b1bb8cd22d05))
|
||||||
|
* hidden editor header on Safari ([b521dec](https://github.com/filebrowser/filebrowser/commit/b521dec8f9b14dd92248c429e902ebc639046389))
|
||||||
|
* image quality switch on previewer ([c0d85f3](https://github.com/filebrowser/filebrowser/commit/c0d85f3d85926c8790757bf142140d19455ae8ca))
|
||||||
|
* list item interactions on share ([87f1881](https://github.com/filebrowser/filebrowser/commit/87f1881b429877a740ea84a8e783ad4103248289))
|
||||||
|
* missing bold variation for Roboto font ([98d79b8](https://github.com/filebrowser/filebrowser/commit/98d79b8ed955df5691a306d709b4ab60d91da408))
|
||||||
|
* mouse wheel zoom on previewer ([fcb115f](https://github.com/filebrowser/filebrowser/commit/fcb115f42d33db2be7a4d428ec53d65d6050320b))
|
||||||
|
* no header button animations on file listing ([fe80730](https://github.com/filebrowser/filebrowser/commit/fe80730bb135b38e4d9de470c75cbe10b1aec201))
|
||||||
|
|
||||||
### [2.14.1](https://github.com/filebrowser/filebrowser/compare/v2.14.0...v2.14.1) (2021-03-21)
|
### [2.14.1](https://github.com/filebrowser/filebrowser/compare/v2.14.0...v2.14.1) (2021-03-21)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \
|
|||||||
VOLUME /srv
|
VOLUME /srv
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
|
|
||||||
COPY .docker.json /.filebrowser.json
|
COPY docker_config.json /.filebrowser.json
|
||||||
COPY filebrowser /filebrowser
|
COPY filebrowser /filebrowser
|
||||||
|
|
||||||
ENTRYPOINT [ "/filebrowser" ]
|
ENTRYPOINT [ "/filebrowser" ]
|
||||||
16
Dockerfile.s6
Normal file
16
Dockerfile.s6
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
FROM ghcr.io/linuxserver/baseimage-alpine:3.14
|
||||||
|
|
||||||
|
RUN apk --update add ca-certificates \
|
||||||
|
mailcap \
|
||||||
|
curl
|
||||||
|
|
||||||
|
HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \
|
||||||
|
CMD curl -f http://localhost/health || exit 1
|
||||||
|
|
||||||
|
# copy local files
|
||||||
|
COPY docker/root/ /
|
||||||
|
COPY filebrowser /usr/bin/filebrowser
|
||||||
|
|
||||||
|
# ports and volumes
|
||||||
|
VOLUME /srv /config /database
|
||||||
|
EXPOSE 80
|
||||||
16
Dockerfile.s6.aarch64
Normal file
16
Dockerfile.s6.aarch64
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
FROM ghcr.io/linuxserver/baseimage-alpine:arm64v8-3.14
|
||||||
|
|
||||||
|
RUN apk --update add ca-certificates \
|
||||||
|
mailcap \
|
||||||
|
curl
|
||||||
|
|
||||||
|
HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \
|
||||||
|
CMD curl -f http://localhost/health || exit 1
|
||||||
|
|
||||||
|
# copy local files
|
||||||
|
COPY docker/root/ /
|
||||||
|
COPY filebrowser /usr/bin/filebrowser
|
||||||
|
|
||||||
|
# ports and volumes
|
||||||
|
VOLUME /srv /config /database
|
||||||
|
EXPOSE 80
|
||||||
16
Dockerfile.s6.armhf
Normal file
16
Dockerfile.s6.armhf
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
FROM ghcr.io/linuxserver/baseimage-alpine:arm32v7-3.14
|
||||||
|
|
||||||
|
RUN apk --update add ca-certificates \
|
||||||
|
mailcap \
|
||||||
|
curl
|
||||||
|
|
||||||
|
HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \
|
||||||
|
CMD curl -f http://localhost/health || exit 1
|
||||||
|
|
||||||
|
# copy local files
|
||||||
|
COPY docker/root/ /
|
||||||
|
COPY filebrowser /usr/bin/filebrowser
|
||||||
|
|
||||||
|
# ports and volumes
|
||||||
|
VOLUME /srv /config /database
|
||||||
|
EXPOSE 80
|
||||||
104
Makefile
104
Makefile
@@ -1,94 +1,68 @@
|
|||||||
SHELL := /bin/bash
|
include common.mk
|
||||||
BASE_PATH := $(shell dirname $(realpath $(lastword $(MAKEFILE_LIST))))
|
include tools.mk
|
||||||
VERSION ?= $(shell git describe --tags --always --match=v* 2> /dev/null || \
|
|
||||||
cat $(CURDIR)/.version 2> /dev/null || echo v0)
|
|
||||||
VERSION_HASH = $(shell git rev-parse HEAD)
|
|
||||||
|
|
||||||
BIN = $(BASE_PATH)/bin
|
LDFLAGS += -X "$(MODULE)/version.Version=$(VERSION)" -X "$(MODULE)/version.CommitSHA=$(VERSION_HASH)"
|
||||||
PATH := $(BIN):$(PATH)
|
|
||||||
export PATH
|
|
||||||
|
|
||||||
# printing
|
## Build:
|
||||||
V = 0
|
|
||||||
Q = $(if $(filter 1,$V),,@)
|
|
||||||
M = $(shell printf "\033[34;1m▶\033[0m")
|
|
||||||
|
|
||||||
GO = GOGC=off go
|
|
||||||
# go module
|
|
||||||
MODULE = $(shell env GO111MODULE=on $(GO) list -m)
|
|
||||||
|
|
||||||
DATE ?= $(shell date +%FT%T%z)
|
|
||||||
VERSION ?= $(shell git describe --tags --always --match=v* 2> /dev/null || \
|
|
||||||
cat $(CURDIR)/.version 2> /dev/null || echo v0)
|
|
||||||
VERSION_HASH = $(shell git rev-parse HEAD)
|
|
||||||
BRANCH = $(shell git rev-parse --abbrev-ref HEAD)
|
|
||||||
|
|
||||||
LDFLAGS += -X "$(MODULE)/varsion.Version=$(VERSION)" -X "$(MODULE)/varsion.CommitSHA=$(VERSION_HASH)"
|
|
||||||
|
|
||||||
# tools
|
|
||||||
$(BIN):
|
|
||||||
@mkdir -p $@
|
|
||||||
$(BIN)/%: | $(BIN) ; $(info $(M) installing $(PACKAGE)…)
|
|
||||||
$Q env GOBIN=$(BIN) $(GO) install $(PACKAGE)
|
|
||||||
|
|
||||||
GOLANGCI_LINT = $(BIN)/golangci-lint
|
|
||||||
$(BIN)/golangci-lint: PACKAGE=github.com/golangci/golangci-lint/cmd/golangci-lint@v1.37.1
|
|
||||||
|
|
||||||
GOIMPORTS = $(BIN)/goimports
|
|
||||||
$(BIN)/goimports: PACKAGE=golang.org/x/tools/cmd/goimports@v0.1.0
|
|
||||||
|
|
||||||
## build: Build
|
|
||||||
.PHONY: build
|
.PHONY: build
|
||||||
build: | build-frontend build-backend ; $(info $(M) building…)
|
build: | build-frontend build-backend ## Build binary
|
||||||
|
|
||||||
## build-frontend: Build frontend
|
|
||||||
.PHONY: build-frontend
|
.PHONY: build-frontend
|
||||||
build-frontend: | ; $(info $(M) building frontend…)
|
build-frontend: ## Build frontend
|
||||||
$Q cd frontend && npm ci && npm run build
|
$Q cd frontend && npm ci && npm run build
|
||||||
|
|
||||||
## build-backend: Build backend
|
|
||||||
.PHONY: build-backend
|
.PHONY: build-backend
|
||||||
build-backend: | ; $(info $(M) building backend…)
|
build-backend: ## Build backend
|
||||||
$Q $(GO) build -ldflags '$(LDFLAGS)' -o .
|
$Q $(go) build -ldflags '$(LDFLAGS)' -o .
|
||||||
|
|
||||||
## test: Run all tests
|
|
||||||
.PHONY: test
|
.PHONY: test
|
||||||
test: | test-frontend test-backend ; $(info $(M) running tests…)
|
test: | test-frontend test-backend ## Run all tests
|
||||||
|
|
||||||
## test-frontend: Run frontend tests
|
|
||||||
.PHONY: test-frontend
|
.PHONY: test-frontend
|
||||||
test-frontend: | ; $(info $(M) running frontend tests…)
|
test-frontend: ## Run frontend tests
|
||||||
|
|
||||||
## test-backend: Run backend tests
|
|
||||||
.PHONY: test-backend
|
.PHONY: test-backend
|
||||||
test-backend: | ; $(info $(M) running backend tests…)
|
test-backend: ## Run backend tests
|
||||||
$Q $(GO) test -v ./...
|
$Q $(go) test -v ./...
|
||||||
|
|
||||||
## lint: Lint
|
|
||||||
.PHONY: lint
|
.PHONY: lint
|
||||||
lint: lint-frontend lint-backend lint-commits | ; $(info $(M) running all linters…)
|
lint: lint-frontend lint-backend lint-commits ## Run all linters
|
||||||
|
|
||||||
## lint-frontend: Lint frontend
|
|
||||||
.PHONY: lint-frontend
|
.PHONY: lint-frontend
|
||||||
lint-frontend: | ; $(info $(M) running frontend linters…)
|
lint-frontend: ## Run frontend linters
|
||||||
$Q cd frontend && npm ci && npm run lint
|
$Q cd frontend && npm ci && npm run lint
|
||||||
|
|
||||||
## lint-backend: Lint backend
|
|
||||||
.PHONY: lint-backend
|
.PHONY: lint-backend
|
||||||
lint-backend: | $(GOLANGCI_LINT) ; $(info $(M) running backend linters…)
|
lint-backend: | $(golangci-lint) ## Run backend linters
|
||||||
$Q $(GOLANGCI_LINT) run
|
$Q $(golangci-lint) run -v
|
||||||
|
|
||||||
## lint-commits: Lint commits
|
|
||||||
.PHONY: lint-commits
|
.PHONY: lint-commits
|
||||||
lint-commits: | ; $(info $(M) running commitlint…)
|
lint-commits: $(commitlint) ## Run commit linters
|
||||||
$Q ./scripts/commitlint.sh
|
$Q ./scripts/commitlint.sh
|
||||||
|
|
||||||
## bump-version: Bump app version
|
fmt: $(goimports) ## Format source files
|
||||||
|
$Q $(goimports) -local $(MODULE) -w $$(find . -type f -name '*.go' -not -path "./vendor/*")
|
||||||
|
|
||||||
|
clean: clean-tools ## Clean
|
||||||
|
|
||||||
|
## Release:
|
||||||
|
|
||||||
.PHONY: bump-version
|
.PHONY: bump-version
|
||||||
bump-version: | ; $(info $(M) creating a new release…)
|
bump-version: $(standard-version) ## Bump app version
|
||||||
$Q ./scripts/bump_version.sh
|
$Q ./scripts/bump_version.sh
|
||||||
|
|
||||||
## help: Show this help
|
## Help:
|
||||||
.PHONY: help
|
help: ## Show this help
|
||||||
help:
|
@echo ''
|
||||||
@sed -n 's/^## //p' $(MAKEFILE_LIST) | column -t -s ':' | sed -e 's/^/ /' | sort
|
@echo 'Usage:'
|
||||||
|
@echo ' ${YELLOW}make${RESET} ${GREEN}<target> [options]${RESET}'
|
||||||
|
@echo ''
|
||||||
|
@echo 'Options:'
|
||||||
|
@$(call global_option, "V [0|1]", "enable verbose mode (default:0)")
|
||||||
|
@echo ''
|
||||||
|
@echo 'Targets:'
|
||||||
|
@awk 'BEGIN {FS = ":.*?## "} { \
|
||||||
|
if (/^[a-zA-Z_-]+:.*?##.*$$/) {printf " ${YELLOW}%-20s${GREEN}%s${RESET}\n", $$1, $$2} \
|
||||||
|
else if (/^## .*$$/) {printf " ${CYAN}%s${RESET}\n", substr($$1,4)} \
|
||||||
|
}' $(MAKEFILE_LIST)
|
||||||
26
SECURITY.md
Normal file
26
SECURITY.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Supported Versions
|
||||||
|
|
||||||
|
Use this section to tell people about which versions of your project are
|
||||||
|
currently being supported with security updates.
|
||||||
|
|
||||||
|
| Version | Supported |
|
||||||
|
| ------- | ------------------ |
|
||||||
|
| 2.x | :white_check_mark: |
|
||||||
|
| < 2.0 | :x: |
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
Vulnerabilities should be reported to filebrowser@googlegroups.com - which is a private, maintainer-only group. Maintainers will attempt to respond to/confirm reports within 2-3 days, but if you believe your report to be "critical" to user safety and security, please note as such in the subject. We have tens of thousands of users using our software, and take security vulnerabilities seriously.
|
||||||
|
|
||||||
|
When reporting an issue, where possible, please provide at least:
|
||||||
|
|
||||||
|
* The commit version the issue was identified at
|
||||||
|
* A proof of concept (plaintext; no binaries)
|
||||||
|
* Steps to reproduce
|
||||||
|
* Your recommended remediation(s), if any.
|
||||||
|
|
||||||
|
The FileBrowser team is a volunteer-only effort, and may reach back out for clarification.
|
||||||
|
|
||||||
|
> Note: Please do not open public issues for security issues, as GitHub does not provide facility for private issues, and deleting the issue makes it hard to triage/respond back to the reporter.
|
||||||
@@ -41,6 +41,7 @@ func addConfigFlags(flags *pflag.FlagSet) {
|
|||||||
flags.String("recaptcha.secret", "", "ReCaptcha secret")
|
flags.String("recaptcha.secret", "", "ReCaptcha secret")
|
||||||
|
|
||||||
flags.String("branding.name", "", "replace 'File Browser' by this name")
|
flags.String("branding.name", "", "replace 'File Browser' by this name")
|
||||||
|
flags.String("branding.color", "", "set the theme color")
|
||||||
flags.String("branding.files", "", "path to directory with images and custom styles")
|
flags.String("branding.files", "", "path to directory with images and custom styles")
|
||||||
flags.Bool("branding.disableExternal", false, "disable external links such as GitHub links")
|
flags.Bool("branding.disableExternal", false, "disable external links such as GitHub links")
|
||||||
}
|
}
|
||||||
@@ -121,7 +122,7 @@ func getAuthentication(flags *pflag.FlagSet, defaults ...interface{}) (settings.
|
|||||||
}
|
}
|
||||||
|
|
||||||
func printSettings(ser *settings.Server, set *settings.Settings, auther auth.Auther) {
|
func printSettings(ser *settings.Server, set *settings.Settings, auther auth.Auther) {
|
||||||
w := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0)
|
w := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0) //nolint:gomnd
|
||||||
|
|
||||||
fmt.Fprintf(w, "Sign up:\t%t\n", set.Signup)
|
fmt.Fprintf(w, "Sign up:\t%t\n", set.Signup)
|
||||||
fmt.Fprintf(w, "Create User Dir:\t%t\n", set.CreateUserDir)
|
fmt.Fprintf(w, "Create User Dir:\t%t\n", set.CreateUserDir)
|
||||||
@@ -131,6 +132,7 @@ func printSettings(ser *settings.Server, set *settings.Settings, auther auth.Aut
|
|||||||
fmt.Fprintf(w, "\tName:\t%s\n", set.Branding.Name)
|
fmt.Fprintf(w, "\tName:\t%s\n", set.Branding.Name)
|
||||||
fmt.Fprintf(w, "\tFiles override:\t%s\n", set.Branding.Files)
|
fmt.Fprintf(w, "\tFiles override:\t%s\n", set.Branding.Files)
|
||||||
fmt.Fprintf(w, "\tDisable external links:\t%t\n", set.Branding.DisableExternal)
|
fmt.Fprintf(w, "\tDisable external links:\t%t\n", set.Branding.DisableExternal)
|
||||||
|
fmt.Fprintf(w, "\tColor:\t%s\n", set.Branding.Color)
|
||||||
fmt.Fprintln(w, "\nServer:")
|
fmt.Fprintln(w, "\nServer:")
|
||||||
fmt.Fprintf(w, "\tLog:\t%s\n", ser.Log)
|
fmt.Fprintf(w, "\tLog:\t%s\n", ser.Log)
|
||||||
fmt.Fprintf(w, "\tPort:\t%s\n", ser.Port)
|
fmt.Fprintf(w, "\tPort:\t%s\n", ser.Port)
|
||||||
|
|||||||
@@ -51,6 +51,8 @@ you want to change. Other options will remain unchanged.`,
|
|||||||
set.Shell = convertCmdStrToCmdArray(mustGetString(flags, flag.Name))
|
set.Shell = convertCmdStrToCmdArray(mustGetString(flags, flag.Name))
|
||||||
case "branding.name":
|
case "branding.name":
|
||||||
set.Branding.Name = mustGetString(flags, flag.Name)
|
set.Branding.Name = mustGetString(flags, flag.Name)
|
||||||
|
case "branding.color":
|
||||||
|
set.Branding.Color = mustGetString(flags, flag.Name)
|
||||||
case "branding.disableExternal":
|
case "branding.disableExternal":
|
||||||
set.Branding.DisableExternal = mustGetBool(flags, flag.Name)
|
set.Branding.DisableExternal = mustGetBool(flags, flag.Name)
|
||||||
case "branding.files":
|
case "branding.files":
|
||||||
|
|||||||
10
cmd/root.go
10
cmd/root.go
@@ -3,8 +3,8 @@ package cmd
|
|||||||
import (
|
import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"errors"
|
"errors"
|
||||||
|
"io"
|
||||||
"io/fs"
|
"io/fs"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
@@ -61,10 +61,10 @@ func addServerFlags(flags *pflag.FlagSet) {
|
|||||||
flags.StringP("key", "k", "", "tls key")
|
flags.StringP("key", "k", "", "tls key")
|
||||||
flags.StringP("root", "r", ".", "root to prepend to relative paths")
|
flags.StringP("root", "r", ".", "root to prepend to relative paths")
|
||||||
flags.String("socket", "", "socket to listen to (cannot be used with address, port, cert nor key flags)")
|
flags.String("socket", "", "socket to listen to (cannot be used with address, port, cert nor key flags)")
|
||||||
flags.Uint32("socket-perm", 0666, "unix socket file permissions")
|
flags.Uint32("socket-perm", 0666, "unix socket file permissions") //nolint:gomnd
|
||||||
flags.StringP("baseurl", "b", "", "base url")
|
flags.StringP("baseurl", "b", "", "base url")
|
||||||
flags.String("cache-dir", "", "file cache directory (disabled if empty)")
|
flags.String("cache-dir", "", "file cache directory (disabled if empty)")
|
||||||
flags.Int("img-processors", 4, "image processors count")
|
flags.Int("img-processors", 4, "image processors count") //nolint:gomnd
|
||||||
flags.Bool("disable-thumbnails", false, "disable image thumbnails")
|
flags.Bool("disable-thumbnails", false, "disable image thumbnails")
|
||||||
flags.Bool("disable-preview-resize", false, "disable resize of image previews")
|
flags.Bool("disable-preview-resize", false, "disable resize of image previews")
|
||||||
flags.Bool("disable-exec", false, "disables Command Runner feature")
|
flags.Bool("disable-exec", false, "disables Command Runner feature")
|
||||||
@@ -128,7 +128,7 @@ user created with the credentials from options "username" and "password".`,
|
|||||||
cacheDir, err := cmd.Flags().GetString("cache-dir")
|
cacheDir, err := cmd.Flags().GetString("cache-dir")
|
||||||
checkErr(err)
|
checkErr(err)
|
||||||
if cacheDir != "" {
|
if cacheDir != "" {
|
||||||
if err := os.MkdirAll(cacheDir, 0700); err != nil { //nolint:govet
|
if err := os.MkdirAll(cacheDir, 0700); err != nil { //nolint:govet,gomnd
|
||||||
log.Fatalf("can't make directory %s: %s", cacheDir, err)
|
log.Fatalf("can't make directory %s: %s", cacheDir, err)
|
||||||
}
|
}
|
||||||
fileCache = diskcache.New(afero.NewOsFs(), cacheDir)
|
fileCache = diskcache.New(afero.NewOsFs(), cacheDir)
|
||||||
@@ -299,7 +299,7 @@ func setupLog(logMethod string) {
|
|||||||
case "stderr":
|
case "stderr":
|
||||||
log.SetOutput(os.Stderr)
|
log.SetOutput(os.Stderr)
|
||||||
case "":
|
case "":
|
||||||
log.SetOutput(ioutil.Discard)
|
log.SetOutput(io.Discard)
|
||||||
default:
|
default:
|
||||||
log.SetOutput(&lumberjack.Logger{
|
log.SetOutput(&lumberjack.Logger{
|
||||||
Filename: logMethod,
|
Filename: logMethod,
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ You can also specify an optional parameter (index_end) so
|
|||||||
you can remove all commands from 'index' to 'index_end',
|
you can remove all commands from 'index' to 'index_end',
|
||||||
including 'index_end'.`,
|
including 'index_end'.`,
|
||||||
Args: func(cmd *cobra.Command, args []string) error {
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
if err := cobra.RangeArgs(1, 2)(cmd, args); err != nil {
|
if err := cobra.RangeArgs(1, 2)(cmd, args); err != nil { //nolint:gomnd
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ var usersCmd = &cobra.Command{
|
|||||||
}
|
}
|
||||||
|
|
||||||
func printUsers(usrs []*users.User) {
|
func printUsers(usrs []*users.User) {
|
||||||
w := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0)
|
w := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0) //nolint:gomnd
|
||||||
fmt.Fprintln(w, "ID\tUsername\tScope\tLocale\tV. Mode\tS.Click\tAdmin\tExecute\tCreate\tRename\tModify\tDelete\tShare\tDownload\tPwd Lock")
|
fmt.Fprintln(w, "ID\tUsername\tScope\tLocale\tV. Mode\tS.Click\tAdmin\tExecute\tCreate\tRename\tModify\tDelete\tShare\tDownload\tPwd Lock")
|
||||||
|
|
||||||
for _, u := range usrs {
|
for _, u := range usrs {
|
||||||
@@ -53,7 +53,7 @@ func printUsers(usrs []*users.User) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func parseUsernameOrID(arg string) (username string, id uint) {
|
func parseUsernameOrID(arg string) (username string, id uint) {
|
||||||
id64, err := strconv.ParseUint(arg, 10, 0)
|
id64, err := strconv.ParseUint(arg, 10, 64) //nolint:gomnd
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return arg, 0
|
return arg, 0
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -72,7 +72,7 @@ func dbExists(path string) (bool, error) {
|
|||||||
d := filepath.Dir(path)
|
d := filepath.Dir(path)
|
||||||
_, err = os.Stat(d)
|
_, err = os.Stat(d)
|
||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
if err := os.MkdirAll(d, 0700); err != nil { //nolint:govet
|
if err := os.MkdirAll(d, 0700); err != nil { //nolint:govet,gomnd
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
return false, nil
|
return false, nil
|
||||||
|
|||||||
28
common.mk
Normal file
28
common.mk
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
SHELL := /bin/bash
|
||||||
|
DATE ?= $(shell date +%FT%T%z)
|
||||||
|
BASE_PATH := $(shell dirname $(realpath $(lastword $(MAKEFILE_LIST))))
|
||||||
|
VERSION ?= $(shell git describe --tags --always --match=v* 2> /dev/null || \
|
||||||
|
cat $(CURDIR)/.version 2> /dev/null || echo v0)
|
||||||
|
VERSION_HASH = $(shell git rev-parse HEAD)
|
||||||
|
BRANCH = $(shell git rev-parse --abbrev-ref HEAD)
|
||||||
|
|
||||||
|
go = GOGC=off go
|
||||||
|
MODULE = $(shell env GO111MODULE=on go list -m)
|
||||||
|
|
||||||
|
# printing
|
||||||
|
# $Q (quiet) is used in the targets as a replacer for @.
|
||||||
|
# This macro helps to print the command for debugging by setting V to 1. Example `make test-unit V=1`
|
||||||
|
V = 0
|
||||||
|
Q = $(if $(filter 1,$V),,@)
|
||||||
|
# $M is a macro to print a colored ▶ character. Example `$(info $(M) running coverage tests…)` will print "▶ running coverage tests…"
|
||||||
|
M = $(shell printf "\033[34;1m▶\033[0m")
|
||||||
|
|
||||||
|
GREEN := $(shell tput -Txterm setaf 2)
|
||||||
|
YELLOW := $(shell tput -Txterm setaf 3)
|
||||||
|
WHITE := $(shell tput -Txterm setaf 7)
|
||||||
|
CYAN := $(shell tput -Txterm setaf 6)
|
||||||
|
RESET := $(shell tput -Txterm sgr0)
|
||||||
|
|
||||||
|
define global_option
|
||||||
|
printf " ${YELLOW}%-20s${GREEN}%s${RESET}\n" $(1) $(2)
|
||||||
|
endef
|
||||||
@@ -6,7 +6,7 @@ import (
|
|||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sync"
|
"sync"
|
||||||
@@ -37,11 +37,11 @@ func (f *FileCache) Store(ctx context.Context, key string, value []byte) error {
|
|||||||
defer mu.Unlock()
|
defer mu.Unlock()
|
||||||
|
|
||||||
fileName := f.getFileName(key)
|
fileName := f.getFileName(key)
|
||||||
if err := f.fs.MkdirAll(filepath.Dir(fileName), 0700); err != nil {
|
if err := f.fs.MkdirAll(filepath.Dir(fileName), 0700); err != nil { //nolint:gomnd
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := afero.WriteFile(f.fs, fileName, value, 0700); err != nil {
|
if err := afero.WriteFile(f.fs, fileName, value, 0700); err != nil { //nolint:gomnd
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -55,7 +55,7 @@ func (f *FileCache) Load(ctx context.Context, key string) (value []byte, exist b
|
|||||||
}
|
}
|
||||||
defer r.Close()
|
defer r.Close()
|
||||||
|
|
||||||
value, err = ioutil.ReadAll(r)
|
value, err = io.ReadAll(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, false, err
|
return nil, false, err
|
||||||
}
|
}
|
||||||
|
|||||||
8
docker/root/defaults/settings.json
Normal file
8
docker/root/defaults/settings.json
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"port": 80,
|
||||||
|
"baseURL": "",
|
||||||
|
"address": "",
|
||||||
|
"log": "stdout",
|
||||||
|
"database": "/database/filebrowser.db",
|
||||||
|
"root": "/srv"
|
||||||
|
}
|
||||||
15
docker/root/etc/cont-init.d/20-config
Normal file
15
docker/root/etc/cont-init.d/20-config
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#!/usr/bin/with-contenv bash
|
||||||
|
|
||||||
|
# make folders
|
||||||
|
mkdir -p /database
|
||||||
|
|
||||||
|
# copy config
|
||||||
|
if [ ! -f "/config/settings.json" ]; then
|
||||||
|
cp -a /defaults/settings.json /config/settings.json
|
||||||
|
fi
|
||||||
|
|
||||||
|
# permissions
|
||||||
|
chown abc:abc \
|
||||||
|
/config/settings.json \
|
||||||
|
/database \
|
||||||
|
/srv
|
||||||
3
docker/root/etc/services.d/filebrowser/run
Normal file
3
docker/root/etc/services.d/filebrowser/run
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/usr/bin/with-contenv bash
|
||||||
|
|
||||||
|
exec s6-setuidgid abc filebrowser -c /config/settings.json -d /database/filebrowser.db;
|
||||||
120
files/file.go
120
files/file.go
@@ -34,6 +34,7 @@ type FileInfo struct {
|
|||||||
ModTime time.Time `json:"modified"`
|
ModTime time.Time `json:"modified"`
|
||||||
Mode os.FileMode `json:"mode"`
|
Mode os.FileMode `json:"mode"`
|
||||||
IsDir bool `json:"isDir"`
|
IsDir bool `json:"isDir"`
|
||||||
|
IsSymlink bool `json:"isSymlink"`
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
Subtitles []string `json:"subtitles,omitempty"`
|
Subtitles []string `json:"subtitles,omitempty"`
|
||||||
Content string `json:"content,omitempty"`
|
Content string `json:"content,omitempty"`
|
||||||
@@ -50,6 +51,7 @@ type FileOptions struct {
|
|||||||
ReadHeader bool
|
ReadHeader bool
|
||||||
Token string
|
Token string
|
||||||
Checker rules.Checker
|
Checker rules.Checker
|
||||||
|
Content bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewFileInfo creates a File object from a path and a given user. This File
|
// NewFileInfo creates a File object from a path and a given user. This File
|
||||||
@@ -60,12 +62,73 @@ func NewFileInfo(opts FileOptions) (*FileInfo, error) {
|
|||||||
return nil, os.ErrPermission
|
return nil, os.ErrPermission
|
||||||
}
|
}
|
||||||
|
|
||||||
info, err := opts.Fs.Stat(opts.Path)
|
file, err := stat(opts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
file := &FileInfo{
|
if opts.Expand {
|
||||||
|
if file.IsDir {
|
||||||
|
if err := file.readListing(opts.Checker, opts.ReadHeader); err != nil { //nolint:govet
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return file, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
err = file.detectType(opts.Modify, opts.Content, true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return file, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func stat(opts FileOptions) (*FileInfo, error) {
|
||||||
|
var file *FileInfo
|
||||||
|
|
||||||
|
if lstaterFs, ok := opts.Fs.(afero.Lstater); ok {
|
||||||
|
info, _, err := lstaterFs.LstatIfPossible(opts.Path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
file = &FileInfo{
|
||||||
|
Fs: opts.Fs,
|
||||||
|
Path: opts.Path,
|
||||||
|
Name: info.Name(),
|
||||||
|
ModTime: info.ModTime(),
|
||||||
|
Mode: info.Mode(),
|
||||||
|
IsDir: info.IsDir(),
|
||||||
|
IsSymlink: IsSymlink(info.Mode()),
|
||||||
|
Size: info.Size(),
|
||||||
|
Extension: filepath.Ext(info.Name()),
|
||||||
|
Token: opts.Token,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// regular file
|
||||||
|
if file != nil && !file.IsSymlink {
|
||||||
|
return file, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// fs doesn't support afero.Lstater interface or the file is a symlink
|
||||||
|
info, err := opts.Fs.Stat(opts.Path)
|
||||||
|
if err != nil {
|
||||||
|
// can't follow symlink
|
||||||
|
if file != nil && file.IsSymlink {
|
||||||
|
return file, nil
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// set correct file size in case of symlink
|
||||||
|
if file != nil && file.IsSymlink {
|
||||||
|
file.Size = info.Size()
|
||||||
|
file.IsDir = info.IsDir()
|
||||||
|
return file, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
file = &FileInfo{
|
||||||
Fs: opts.Fs,
|
Fs: opts.Fs,
|
||||||
Path: opts.Path,
|
Path: opts.Path,
|
||||||
Name: info.Name(),
|
Name: info.Name(),
|
||||||
@@ -77,21 +140,7 @@ func NewFileInfo(opts FileOptions) (*FileInfo, error) {
|
|||||||
Token: opts.Token,
|
Token: opts.Token,
|
||||||
}
|
}
|
||||||
|
|
||||||
if opts.Expand {
|
|
||||||
if file.IsDir {
|
|
||||||
if err := file.readListing(opts.Checker, opts.ReadHeader); err != nil { //nolint:govet
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return file, nil
|
return file, nil
|
||||||
}
|
|
||||||
|
|
||||||
err = file.detectType(opts.Modify, true, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return file, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Checksum checksums a given File for a given User, using a specific
|
// Checksum checksums a given File for a given User, using a specific
|
||||||
@@ -136,6 +185,19 @@ func (i *FileInfo) Checksum(algo string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (i *FileInfo) RealPath() string {
|
||||||
|
if realPathFs, ok := i.Fs.(interface {
|
||||||
|
RealPath(name string) (fPath string, err error)
|
||||||
|
}); ok {
|
||||||
|
realPath, err := realPathFs.RealPath(i.Path)
|
||||||
|
if err == nil {
|
||||||
|
return realPath
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i.Path
|
||||||
|
}
|
||||||
|
|
||||||
//nolint:goconst
|
//nolint:goconst
|
||||||
//TODO: use constants
|
//TODO: use constants
|
||||||
func (i *FileInfo) detectType(modify, saveContent, readHeader bool) error {
|
func (i *FileInfo) detectType(modify, saveContent, readHeader bool) error {
|
||||||
@@ -170,7 +232,10 @@ func (i *FileInfo) detectType(modify, saveContent, readHeader bool) error {
|
|||||||
case strings.HasPrefix(mimetype, "image"):
|
case strings.HasPrefix(mimetype, "image"):
|
||||||
i.Type = "image"
|
i.Type = "image"
|
||||||
return nil
|
return nil
|
||||||
case (strings.HasPrefix(mimetype, "text") || (len(buffer) > 0 && !isBinary(buffer))) && i.Size <= 10*1024*1024: // 10 MB
|
case strings.HasSuffix(mimetype, "pdf"):
|
||||||
|
i.Type = "pdf"
|
||||||
|
return nil
|
||||||
|
case (strings.HasPrefix(mimetype, "text") || !isBinary(buffer)) && i.Size <= 10*1024*1024: // 10 MB
|
||||||
i.Type = "text"
|
i.Type = "text"
|
||||||
|
|
||||||
if !modify {
|
if !modify {
|
||||||
@@ -203,7 +268,7 @@ func (i *FileInfo) readFirstBytes() []byte {
|
|||||||
}
|
}
|
||||||
defer reader.Close()
|
defer reader.Close()
|
||||||
|
|
||||||
buffer := make([]byte, 512)
|
buffer := make([]byte, 512) //nolint:gomnd
|
||||||
n, err := reader.Read(buffer)
|
n, err := reader.Read(buffer)
|
||||||
if err != nil && err != io.EOF {
|
if err != nil && err != io.EOF {
|
||||||
log.Print(err)
|
log.Print(err)
|
||||||
@@ -222,11 +287,17 @@ func (i *FileInfo) detectSubtitles() {
|
|||||||
i.Subtitles = []string{}
|
i.Subtitles = []string{}
|
||||||
ext := filepath.Ext(i.Path)
|
ext := filepath.Ext(i.Path)
|
||||||
|
|
||||||
// TODO: detect multiple languages. Base.Lang.vtt
|
// detect multiple languages. Base*.vtt
|
||||||
|
// TODO: give subtitles descriptive names (lang) and track attributes
|
||||||
fPath := strings.TrimSuffix(i.Path, ext) + ".vtt"
|
parentDir := strings.TrimRight(i.Path, i.Name)
|
||||||
if _, err := i.Fs.Stat(fPath); err == nil {
|
dir, err := afero.ReadDir(i.Fs, parentDir)
|
||||||
i.Subtitles = append(i.Subtitles, fPath)
|
if err == nil {
|
||||||
|
base := strings.TrimSuffix(i.Name, ext)
|
||||||
|
for _, f := range dir {
|
||||||
|
if !f.IsDir() && strings.HasPrefix(f.Name(), base) && strings.HasSuffix(f.Name(), ".vtt") {
|
||||||
|
i.Subtitles = append(i.Subtitles, path.Join(parentDir, f.Name()))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -251,7 +322,9 @@ func (i *FileInfo) readListing(checker rules.Checker, readHeader bool) error {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isSymlink := false
|
||||||
if IsSymlink(f.Mode()) {
|
if IsSymlink(f.Mode()) {
|
||||||
|
isSymlink = true
|
||||||
// It's a symbolic link. We try to follow it. If it doesn't work,
|
// It's a symbolic link. We try to follow it. If it doesn't work,
|
||||||
// we stay with the link information instead of the target's.
|
// we stay with the link information instead of the target's.
|
||||||
info, err := i.Fs.Stat(fPath)
|
info, err := i.Fs.Stat(fPath)
|
||||||
@@ -267,6 +340,7 @@ func (i *FileInfo) readListing(checker rules.Checker, readHeader bool) error {
|
|||||||
ModTime: f.ModTime(),
|
ModTime: f.ModTime(),
|
||||||
Mode: f.Mode(),
|
Mode: f.Mode(),
|
||||||
IsDir: f.IsDir(),
|
IsDir: f.IsDir(),
|
||||||
|
IsSymlink: isSymlink,
|
||||||
Extension: filepath.Ext(name),
|
Extension: filepath.Ext(name),
|
||||||
Path: fPath,
|
Path: fPath,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -40,13 +40,13 @@ func CopyFile(fs afero.Fs, source, dest string) error {
|
|||||||
|
|
||||||
// Makes the directory needed to create the dst
|
// Makes the directory needed to create the dst
|
||||||
// file.
|
// file.
|
||||||
err = fs.MkdirAll(filepath.Dir(dest), 0666)
|
err = fs.MkdirAll(filepath.Dir(dest), 0666) //nolint:gomnd
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the destination file.
|
// Create the destination file.
|
||||||
dst, err := fs.OpenFile(dest, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0775)
|
dst, err := fs.OpenFile(dest, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0775) //nolint:gomnd
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
//go:build !dev
|
||||||
// +build !dev
|
// +build !dev
|
||||||
|
|
||||||
package frontend
|
package frontend
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
//go:build dev
|
||||||
// +build dev
|
// +build dev
|
||||||
|
|
||||||
package frontend
|
package frontend
|
||||||
|
|||||||
735
frontend/package-lock.json
generated
735
frontend/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -5,7 +5,7 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"serve": "vue-cli-service serve",
|
"serve": "vue-cli-service serve",
|
||||||
"build": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --no-clean",
|
"build": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --no-clean",
|
||||||
"lint": "npx vue-cli-service lint --no-fix",
|
"lint": "npx vue-cli-service lint --no-fix --max-warnings=0",
|
||||||
"fix": "npx vue-cli-service lint",
|
"fix": "npx vue-cli-service lint",
|
||||||
"watch": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --watch --no-clean"
|
"watch": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --watch --no-clean"
|
||||||
},
|
},
|
||||||
@@ -13,10 +13,11 @@
|
|||||||
"ace-builds": "^1.4.7",
|
"ace-builds": "^1.4.7",
|
||||||
"clipboard": "^2.0.4",
|
"clipboard": "^2.0.4",
|
||||||
"core-js": "^3.9.1",
|
"core-js": "^3.9.1",
|
||||||
|
"css-vars-ponyfill": "^2.4.3",
|
||||||
"js-base64": "^2.5.1",
|
"js-base64": "^2.5.1",
|
||||||
"lodash.clonedeep": "^4.5.0",
|
"lodash.clonedeep": "^4.5.0",
|
||||||
"lodash.throttle": "^4.1.1",
|
"lodash.throttle": "^4.1.1",
|
||||||
"material-design-icons": "^3.0.1",
|
"material-icons": "^1.10.5",
|
||||||
"moment": "^2.24.0",
|
"moment": "^2.24.0",
|
||||||
"normalize.css": "^8.0.1",
|
"normalize.css": "^8.0.1",
|
||||||
"noty": "^3.2.0-beta",
|
"noty": "^3.2.0-beta",
|
||||||
@@ -27,7 +28,8 @@
|
|||||||
"vue-lazyload": "^1.3.3",
|
"vue-lazyload": "^1.3.3",
|
||||||
"vue-router": "^3.1.3",
|
"vue-router": "^3.1.3",
|
||||||
"vuex": "^3.1.2",
|
"vuex": "^3.1.2",
|
||||||
"vuex-router-sync": "^5.0.0"
|
"vuex-router-sync": "^5.0.0",
|
||||||
|
"whatwg-fetch": "^3.6.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@vue/cli-plugin-babel": "^4.1.2",
|
"@vue/cli-plugin-babel": "^4.1.2",
|
||||||
@@ -35,6 +37,7 @@
|
|||||||
"@vue/cli-service": "^4.1.2",
|
"@vue/cli-service": "^4.1.2",
|
||||||
"@vue/eslint-config-prettier": "^6.0.0",
|
"@vue/eslint-config-prettier": "^6.0.0",
|
||||||
"babel-eslint": "^10.1.0",
|
"babel-eslint": "^10.1.0",
|
||||||
|
"compression-webpack-plugin": "^6.0.3",
|
||||||
"eslint": "^6.7.2",
|
"eslint": "^6.7.2",
|
||||||
"eslint-plugin-prettier": "^3.3.1",
|
"eslint-plugin-prettier": "^3.3.1",
|
||||||
"eslint-plugin-vue": "^6.2.2",
|
"eslint-plugin-vue": "^6.2.2",
|
||||||
@@ -64,6 +67,6 @@
|
|||||||
"browserslist": [
|
"browserslist": [
|
||||||
"> 1%",
|
"> 1%",
|
||||||
"last 2 versions",
|
"last 2 versions",
|
||||||
"not ie <= 8"
|
"not ie < 11"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
<!-- Add to home screen for Android and modern mobile browsers -->
|
<!-- Add to home screen for Android and modern mobile browsers -->
|
||||||
<link rel="manifest" id="manifestPlaceholder" crossorigin="use-credentials">
|
<link rel="manifest" id="manifestPlaceholder" crossorigin="use-credentials">
|
||||||
<meta name="theme-color" content="#2979ff">
|
<meta name="theme-color" content="[{[ if .Color -]}][{[ .Color ]}][{[ else ]}]#2979ff[{[ end ]}]">
|
||||||
|
|
||||||
<!-- Add to home screen for Safari on iOS/iPadOS -->
|
<!-- Add to home screen for Safari on iOS/iPadOS -->
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
@@ -26,11 +26,11 @@
|
|||||||
|
|
||||||
<!-- Add to home screen for Windows -->
|
<!-- Add to home screen for Windows -->
|
||||||
<meta name="msapplication-TileImage" content="[{[ .StaticURL ]}]/img/icons/mstile-144x144.png">
|
<meta name="msapplication-TileImage" content="[{[ .StaticURL ]}]/img/icons/mstile-144x144.png">
|
||||||
<meta name="msapplication-TileColor" content="#2979ff">
|
<meta name="msapplication-TileColor" content="[{[ if .Color -]}][{[ .Color ]}][{[ else ]}]#2979ff[{[ end ]}]">
|
||||||
|
|
||||||
<!-- Inject Some Variables and generate the manifest json -->
|
<!-- Inject Some Variables and generate the manifest json -->
|
||||||
<script>
|
<script>
|
||||||
window.FileBrowser = JSON.parse(`[{[ .Json ]}]`);
|
window.FileBrowser = JSON.parse('[{[ .Json ]}]');
|
||||||
|
|
||||||
var fullStaticURL = window.location.origin + window.FileBrowser.StaticURL;
|
var fullStaticURL = window.location.origin + window.FileBrowser.StaticURL;
|
||||||
var dynamicManifest = {
|
var dynamicManifest = {
|
||||||
@@ -51,7 +51,7 @@
|
|||||||
"start_url": window.location.origin + window.FileBrowser.BaseURL,
|
"start_url": window.location.origin + window.FileBrowser.BaseURL,
|
||||||
"display": "standalone",
|
"display": "standalone",
|
||||||
"background_color": "#ffffff",
|
"background_color": "#ffffff",
|
||||||
"theme_color": "#455a64"
|
"theme_color": window.FileBrowser.Color || "#455a64"
|
||||||
}
|
}
|
||||||
|
|
||||||
const stringManifest = JSON.stringify(dynamicManifest);
|
const stringManifest = JSON.stringify(dynamicManifest);
|
||||||
@@ -77,7 +77,7 @@
|
|||||||
opacity: 0;
|
opacity: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.spinner {
|
#loading .spinner {
|
||||||
width: 70px;
|
width: 70px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@@ -87,7 +87,7 @@
|
|||||||
transform: translate(-50%, -50%);
|
transform: translate(-50%, -50%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.spinner > div {
|
#loading .spinner > div {
|
||||||
width: 18px;
|
width: 18px;
|
||||||
height: 18px;
|
height: 18px;
|
||||||
background-color: #333;
|
background-color: #333;
|
||||||
@@ -97,12 +97,12 @@
|
|||||||
animation: sk-bouncedelay 1.4s infinite ease-in-out both;
|
animation: sk-bouncedelay 1.4s infinite ease-in-out both;
|
||||||
}
|
}
|
||||||
|
|
||||||
.spinner .bounce1 {
|
#loading .spinner .bounce1 {
|
||||||
-webkit-animation-delay: -0.32s;
|
-webkit-animation-delay: -0.32s;
|
||||||
animation-delay: -0.32s;
|
animation-delay: -0.32s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.spinner .bounce2 {
|
#loading .spinner .bounce2 {
|
||||||
-webkit-animation-delay: -0.16s;
|
-webkit-animation-delay: -0.16s;
|
||||||
animation-delay: -0.16s;
|
animation-delay: -0.16s;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ body {
|
|||||||
#loading {
|
#loading {
|
||||||
background: var(--background);
|
background: var(--background);
|
||||||
}
|
}
|
||||||
#loading .spinner div, #previewer .loading .spinner div {
|
#loading .spinner div, main .spinner div {
|
||||||
background: var(--icon);
|
background: var(--icon);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
__webpack_public_path__ = window.FileBrowser.StaticURL + "/";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "app",
|
name: "app",
|
||||||
mounted() {
|
mounted() {
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ export async function fetch(url, password = "") {
|
|||||||
url = removePrefix(url);
|
url = removePrefix(url);
|
||||||
|
|
||||||
const res = await fetchURL(`/api/public/share${url}`, {
|
const res = await fetchURL(`/api/public/share${url}`, {
|
||||||
headers: { "X-SHARE-PASSWORD": password },
|
headers: { "X-SHARE-PASSWORD": encodeURIComponent(password) },
|
||||||
});
|
});
|
||||||
|
|
||||||
if (res.status === 200) {
|
if (res.status === 200) {
|
||||||
|
|||||||
@@ -8,13 +8,18 @@ export async function fetchURL(url, opts) {
|
|||||||
|
|
||||||
let { headers, ...rest } = opts;
|
let { headers, ...rest } = opts;
|
||||||
|
|
||||||
const res = await fetch(`${baseURL}${url}`, {
|
let res;
|
||||||
|
try {
|
||||||
|
res = await fetch(`${baseURL}${url}`, {
|
||||||
headers: {
|
headers: {
|
||||||
"X-Auth": store.state.jwt,
|
"X-Auth": store.state.jwt,
|
||||||
...headers,
|
...headers,
|
||||||
},
|
},
|
||||||
...rest,
|
...rest,
|
||||||
});
|
});
|
||||||
|
} catch (error) {
|
||||||
|
return { status: 0 };
|
||||||
|
}
|
||||||
|
|
||||||
if (res.headers.get("X-Renew-Token") === "true") {
|
if (res.headers.get("X-Renew-Token") === "true") {
|
||||||
await renew(store.state.jwt);
|
await renew(store.state.jwt);
|
||||||
|
|||||||
BIN
frontend/src/assets/fonts/roboto/bold-cyrillic-ext.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-cyrillic-ext.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-cyrillic.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-cyrillic.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-greek-ext.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-greek-ext.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-greek.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-greek.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-latin-ext.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-latin-ext.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-latin.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-latin.woff2
Executable file
Binary file not shown.
BIN
frontend/src/assets/fonts/roboto/bold-vietnamese.woff2
Executable file
BIN
frontend/src/assets/fonts/roboto/bold-vietnamese.woff2
Executable file
Binary file not shown.
@@ -1,15 +1,15 @@
|
|||||||
<template>
|
<template>
|
||||||
<nav :class="{ active }">
|
<nav :class="{ active }">
|
||||||
<template v-if="isLogged">
|
<template v-if="isLogged">
|
||||||
<router-link
|
<button
|
||||||
class="action"
|
class="action"
|
||||||
to="/files/"
|
@click="toRoot"
|
||||||
:aria-label="$t('sidebar.myFiles')"
|
:aria-label="$t('sidebar.myFiles')"
|
||||||
:title="$t('sidebar.myFiles')"
|
:title="$t('sidebar.myFiles')"
|
||||||
>
|
>
|
||||||
<i class="material-icons">folder</i>
|
<i class="material-icons">folder</i>
|
||||||
<span>{{ $t("sidebar.myFiles") }}</span>
|
<span>{{ $t("sidebar.myFiles") }}</span>
|
||||||
</router-link>
|
</button>
|
||||||
|
|
||||||
<div v-if="user.perm.create">
|
<div v-if="user.perm.create">
|
||||||
<button
|
<button
|
||||||
@@ -34,15 +34,15 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<router-link
|
<button
|
||||||
class="action"
|
class="action"
|
||||||
to="/settings"
|
@click="toSettings"
|
||||||
:aria-label="$t('sidebar.settings')"
|
:aria-label="$t('sidebar.settings')"
|
||||||
:title="$t('sidebar.settings')"
|
:title="$t('sidebar.settings')"
|
||||||
>
|
>
|
||||||
<i class="material-icons">settings_applications</i>
|
<i class="material-icons">settings_applications</i>
|
||||||
<span>{{ $t("sidebar.settings") }}</span>
|
<span>{{ $t("sidebar.settings") }}</span>
|
||||||
</router-link>
|
</button>
|
||||||
|
|
||||||
<button
|
<button
|
||||||
v-if="authMethod == 'json'"
|
v-if="authMethod == 'json'"
|
||||||
@@ -125,6 +125,14 @@ export default {
|
|||||||
authMethod: () => authMethod,
|
authMethod: () => authMethod,
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
toRoot() {
|
||||||
|
this.$router.push({ path: "/files/" }, () => {});
|
||||||
|
this.$store.commit("closeHovers");
|
||||||
|
},
|
||||||
|
toSettings() {
|
||||||
|
this.$router.push({ path: "/settings" }, () => {});
|
||||||
|
this.$store.commit("closeHovers");
|
||||||
|
},
|
||||||
help() {
|
help() {
|
||||||
this.$store.commit("showHover", "help");
|
this.$store.commit("showHover", "help");
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -29,14 +29,6 @@ export default {
|
|||||||
type: Number,
|
type: Number,
|
||||||
default: () => 200,
|
default: () => 200,
|
||||||
},
|
},
|
||||||
maxScale: {
|
|
||||||
type: Number,
|
|
||||||
default: () => 4,
|
|
||||||
},
|
|
||||||
minScale: {
|
|
||||||
type: Number,
|
|
||||||
default: () => 0.25,
|
|
||||||
},
|
|
||||||
classList: {
|
classList: {
|
||||||
type: Array,
|
type: Array,
|
||||||
default: () => [],
|
default: () => [],
|
||||||
@@ -45,10 +37,6 @@ export default {
|
|||||||
type: Number,
|
type: Number,
|
||||||
default: () => 0.25,
|
default: () => 0.25,
|
||||||
},
|
},
|
||||||
autofill: {
|
|
||||||
type: Boolean,
|
|
||||||
default: () => false,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
@@ -56,6 +44,7 @@ export default {
|
|||||||
lastX: null,
|
lastX: null,
|
||||||
lastY: null,
|
lastY: null,
|
||||||
inDrag: false,
|
inDrag: false,
|
||||||
|
touches: 0,
|
||||||
lastTouchDistance: 0,
|
lastTouchDistance: 0,
|
||||||
moveDisabled: false,
|
moveDisabled: false,
|
||||||
disabledTimer: null,
|
disabledTimer: null,
|
||||||
@@ -64,6 +53,8 @@ export default {
|
|||||||
center: { x: 0, y: 0 },
|
center: { x: 0, y: 0 },
|
||||||
relative: { x: 0, y: 0 },
|
relative: { x: 0, y: 0 },
|
||||||
},
|
},
|
||||||
|
maxScale: 4,
|
||||||
|
minScale: 0.25,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
@@ -88,6 +79,10 @@ export default {
|
|||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
src: function () {
|
src: function () {
|
||||||
|
if (!this.decodeUTIF()) {
|
||||||
|
this.$refs.imgex.src = this.src;
|
||||||
|
}
|
||||||
|
|
||||||
this.scale = 1;
|
this.scale = 1;
|
||||||
this.setZoom();
|
this.setZoom();
|
||||||
this.setCenter();
|
this.setCenter();
|
||||||
@@ -122,6 +117,21 @@ export default {
|
|||||||
img.classList.add("image-ex-img-ready");
|
img.classList.add("image-ex-img-ready");
|
||||||
|
|
||||||
document.addEventListener("mouseup", this.onMouseUp);
|
document.addEventListener("mouseup", this.onMouseUp);
|
||||||
|
|
||||||
|
let realSize = img.naturalWidth;
|
||||||
|
let displaySize = img.offsetWidth;
|
||||||
|
|
||||||
|
// Image is in portrait orientation
|
||||||
|
if (img.naturalHeight > img.naturalWidth) {
|
||||||
|
realSize = img.naturalHeight;
|
||||||
|
displaySize = img.offsetHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Scale needed to display the image on full size
|
||||||
|
const fullScale = realSize / displaySize;
|
||||||
|
|
||||||
|
// Full size plus additional zoom
|
||||||
|
this.maxScale = fullScale + 4;
|
||||||
},
|
},
|
||||||
onMouseUp() {
|
onMouseUp() {
|
||||||
this.inDrag = false;
|
this.inDrag = false;
|
||||||
@@ -251,7 +261,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
wheelMove(event) {
|
wheelMove(event) {
|
||||||
this.scale += (event.wheelDeltaY / 100) * this.zoomStep;
|
this.scale += -Math.sign(event.deltaY) * this.zoomStep;
|
||||||
this.setZoom();
|
this.setZoom();
|
||||||
},
|
},
|
||||||
setZoom() {
|
setZoom() {
|
||||||
|
|||||||
@@ -8,9 +8,8 @@
|
|||||||
@dragover="dragOver"
|
@dragover="dragOver"
|
||||||
@drop="drop"
|
@drop="drop"
|
||||||
@click="itemClick"
|
@click="itemClick"
|
||||||
@dblclick="dblclick"
|
|
||||||
@touchstart="touchstart"
|
|
||||||
:data-dir="isDir"
|
:data-dir="isDir"
|
||||||
|
:data-type="type"
|
||||||
:aria-label="name"
|
:aria-label="name"
|
||||||
:aria-selected="isSelected"
|
:aria-selected="isSelected"
|
||||||
>
|
>
|
||||||
@@ -19,7 +18,7 @@
|
|||||||
v-if="readOnly == undefined && type === 'image' && isThumbsEnabled"
|
v-if="readOnly == undefined && type === 'image' && isThumbsEnabled"
|
||||||
v-lazy="thumbnailUrl"
|
v-lazy="thumbnailUrl"
|
||||||
/>
|
/>
|
||||||
<i v-else class="material-icons">{{ icon }}</i>
|
<i v-else class="material-icons"></i>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
@@ -69,13 +68,6 @@ export default {
|
|||||||
isSelected() {
|
isSelected() {
|
||||||
return this.selected.indexOf(this.index) !== -1;
|
return this.selected.indexOf(this.index) !== -1;
|
||||||
},
|
},
|
||||||
icon() {
|
|
||||||
if (this.isDir) return "folder";
|
|
||||||
if (this.type === "image") return "insert_photo";
|
|
||||||
if (this.type === "audio") return "volume_up";
|
|
||||||
if (this.type === "video") return "movie";
|
|
||||||
return "insert_drive_file";
|
|
||||||
},
|
|
||||||
isDraggable() {
|
isDraggable() {
|
||||||
return this.readOnly == undefined && this.user.perm.rename;
|
return this.readOnly == undefined && this.user.perm.rename;
|
||||||
},
|
},
|
||||||
@@ -96,7 +88,7 @@ export default {
|
|||||||
// reload the image when the file is replaced
|
// reload the image when the file is replaced
|
||||||
const key = Date.parse(this.modified);
|
const key = Date.parse(this.modified);
|
||||||
|
|
||||||
return `${baseURL}/api/preview/thumb/${path}?auth=${this.jwt}&inline=true&k=${key}`;
|
return `${baseURL}/api/preview/thumb/${path}?k=${key}&inline=true`;
|
||||||
},
|
},
|
||||||
isThumbsEnabled() {
|
isThumbsEnabled() {
|
||||||
return enableThumbs;
|
return enableThumbs;
|
||||||
@@ -108,6 +100,9 @@ export default {
|
|||||||
return filesize(this.size);
|
return filesize(this.size);
|
||||||
},
|
},
|
||||||
humanTime: function () {
|
humanTime: function () {
|
||||||
|
if (this.readOnly == undefined && this.user.dateFormat) {
|
||||||
|
return moment(this.modified).format("L LT");
|
||||||
|
}
|
||||||
return moment(this.modified).fromNow();
|
return moment(this.modified).fromNow();
|
||||||
},
|
},
|
||||||
dragStart: function () {
|
dragStart: function () {
|
||||||
@@ -153,13 +148,13 @@ export default {
|
|||||||
for (let i of this.selected) {
|
for (let i of this.selected) {
|
||||||
items.push({
|
items.push({
|
||||||
from: this.req.items[i].url,
|
from: this.req.items[i].url,
|
||||||
to: this.url + this.req.items[i].name,
|
to: this.url + encodeURIComponent(this.req.items[i].name),
|
||||||
name: this.req.items[i].name,
|
name: this.req.items[i].name,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let base = el.querySelector(".name").innerHTML + "/";
|
// Get url from ListingItem instance
|
||||||
let path = this.$route.path + base;
|
let path = el.__vue__.url;
|
||||||
let baseItems = (await api.fetch(path)).items;
|
let baseItems = (await api.fetch(path)).items;
|
||||||
|
|
||||||
let action = (overwrite, rename) => {
|
let action = (overwrite, rename) => {
|
||||||
@@ -200,6 +195,16 @@ export default {
|
|||||||
},
|
},
|
||||||
click: function (event) {
|
click: function (event) {
|
||||||
if (!this.singleClick && this.selectedCount !== 0) event.preventDefault();
|
if (!this.singleClick && this.selectedCount !== 0) event.preventDefault();
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
this.touches = 0;
|
||||||
|
}, 300);
|
||||||
|
|
||||||
|
this.touches++;
|
||||||
|
if (this.touches > 1) {
|
||||||
|
this.open();
|
||||||
|
}
|
||||||
|
|
||||||
if (this.$store.state.selected.indexOf(this.index) !== -1) {
|
if (this.$store.state.selected.indexOf(this.index) !== -1) {
|
||||||
this.removeSelected(this.index);
|
this.removeSelected(this.index);
|
||||||
return;
|
return;
|
||||||
@@ -235,19 +240,6 @@ export default {
|
|||||||
this.resetSelected();
|
this.resetSelected();
|
||||||
this.addSelected(this.index);
|
this.addSelected(this.index);
|
||||||
},
|
},
|
||||||
dblclick: function () {
|
|
||||||
if (!this.singleClick) this.open();
|
|
||||||
},
|
|
||||||
touchstart() {
|
|
||||||
setTimeout(() => {
|
|
||||||
this.touches = 0;
|
|
||||||
}, 300);
|
|
||||||
|
|
||||||
this.touches++;
|
|
||||||
if (this.touches > 1) {
|
|
||||||
this.open();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
open: function () {
|
open: function () {
|
||||||
this.$router.push({ path: this.url });
|
this.$router.push({ path: this.url });
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
<strong>{{ $t("prompts.size") }}:</strong>
|
<strong>{{ $t("prompts.size") }}:</strong>
|
||||||
<span id="content_length"></span> {{ humanSize }}
|
<span id="content_length"></span> {{ humanSize }}
|
||||||
</p>
|
</p>
|
||||||
<p v-if="selected.length < 2">
|
<p v-if="selected.length < 2" :title="modTime">
|
||||||
<strong>{{ $t("prompts.lastModified") }}:</strong> {{ humanTime }}
|
<strong>{{ $t("prompts.lastModified") }}:</strong> {{ humanTime }}
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
@@ -110,6 +110,9 @@ export default {
|
|||||||
|
|
||||||
return moment(this.req.items[this.selected[0]].modified).fromNow();
|
return moment(this.req.items[this.selected[0]].modified).fromNow();
|
||||||
},
|
},
|
||||||
|
modTime: function () {
|
||||||
|
return new Date(Date.parse(this.req.modified)).toLocaleString();
|
||||||
|
},
|
||||||
name: function () {
|
name: function () {
|
||||||
return this.selectedCount === 0
|
return this.selectedCount === 0
|
||||||
? this.req.name
|
? this.req.name
|
||||||
|
|||||||
@@ -56,6 +56,12 @@ export default {
|
|||||||
|
|
||||||
let prompt = this.$refs.currentComponent;
|
let prompt = this.$refs.currentComponent;
|
||||||
|
|
||||||
|
// Esc!
|
||||||
|
if (event.keyCode === 27) {
|
||||||
|
event.stopImmediatePropagation();
|
||||||
|
this.$store.commit("closeHovers");
|
||||||
|
}
|
||||||
|
|
||||||
// Enter
|
// Enter
|
||||||
if (event.keyCode == 13) {
|
if (event.keyCode == 13) {
|
||||||
switch (this.show) {
|
switch (this.show) {
|
||||||
|
|||||||
@@ -11,11 +11,11 @@
|
|||||||
<div class="card-action full">
|
<div class="card-action full">
|
||||||
<div @click="uploadFile" class="action">
|
<div @click="uploadFile" class="action">
|
||||||
<i class="material-icons">insert_drive_file</i>
|
<i class="material-icons">insert_drive_file</i>
|
||||||
<div class="title">File</div>
|
<div class="title">{{ $t("buttons.file") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div @click="uploadFolder" class="action">
|
<div @click="uploadFolder" class="action">
|
||||||
<i class="material-icons">folder</i>
|
<i class="material-icons">folder</i>
|
||||||
<div class="title">Folder</div>
|
<div class="title">{{ $t("buttons.folder") }}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
62
frontend/src/components/prompts/UploadFiles.vue
Normal file
62
frontend/src/components/prompts/UploadFiles.vue
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
<template>
|
||||||
|
<div
|
||||||
|
v-if="filesInUploadCount > 0"
|
||||||
|
class="upload-files"
|
||||||
|
v-bind:class="{ closed: !open }"
|
||||||
|
>
|
||||||
|
<div class="card floating">
|
||||||
|
<div class="card-title">
|
||||||
|
<h2>{{ $t("prompts.uploadFiles", { files: filesInUploadCount }) }}</h2>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="action"
|
||||||
|
@click="toggle"
|
||||||
|
aria-label="Toggle file upload list"
|
||||||
|
title="Toggle file upload list"
|
||||||
|
>
|
||||||
|
<i class="material-icons">{{
|
||||||
|
open ? "keyboard_arrow_down" : "keyboard_arrow_up"
|
||||||
|
}}</i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-content file-icons">
|
||||||
|
<div
|
||||||
|
class="file"
|
||||||
|
v-for="file in filesInUpload"
|
||||||
|
:key="file.id"
|
||||||
|
:data-type="file.type"
|
||||||
|
:aria-label="file.name"
|
||||||
|
>
|
||||||
|
<div class="file-name">
|
||||||
|
<i class="material-icons"></i> {{ file.name }}
|
||||||
|
</div>
|
||||||
|
<div class="file-progress">
|
||||||
|
<div v-bind:style="{ width: file.progress + '%' }"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import { mapGetters } from "vuex";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "uploadFiles",
|
||||||
|
data: function () {
|
||||||
|
return {
|
||||||
|
open: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
...mapGetters(["filesInUpload", "filesInUploadCount"]),
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
toggle: function () {
|
||||||
|
this.open = !this.open;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
@@ -28,7 +28,10 @@ export default {
|
|||||||
pt: "pt",
|
pt: "pt",
|
||||||
ro: "ro",
|
ro: "ro",
|
||||||
ru: "ru",
|
ru: "ru",
|
||||||
|
sk: "sk",
|
||||||
"sv-se": "svSE",
|
"sv-se": "svSE",
|
||||||
|
tr: "tr",
|
||||||
|
ua: "ua",
|
||||||
"zh-cn": "zhCN",
|
"zh-cn": "zhCN",
|
||||||
"zh-tw": "zhTW",
|
"zh-tw": "zhTW",
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -34,7 +34,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.share__box__info {
|
.share__box__info {
|
||||||
flex: 1 1 auto;
|
flex: 1 1 18em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.share__box__element {
|
.share__box__element {
|
||||||
@@ -43,6 +43,15 @@
|
|||||||
word-break: break-all;
|
word-break: break-all;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.share__box__element .button {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.share__box__element .button i {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
.share__box__items {
|
.share__box__items {
|
||||||
text-align: left;
|
text-align: left;
|
||||||
flex: 10 0 25em;
|
flex: 10 0 25em;
|
||||||
|
|||||||
@@ -4,4 +4,11 @@
|
|||||||
--red: #F44336;
|
--red: #F44336;
|
||||||
--dark-red: #D32F2F;
|
--dark-red: #D32F2F;
|
||||||
--moon-grey: #f2f2f2;
|
--moon-grey: #f2f2f2;
|
||||||
|
|
||||||
|
--icon-red: #da4453;
|
||||||
|
--icon-orange: #f47750;
|
||||||
|
--icon-yellow: #fdbc4b;
|
||||||
|
--icon-green: #2ecc71;
|
||||||
|
--icon-blue: #1d99f3;
|
||||||
|
--icon-violet: #9b59b6;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
body {
|
body {
|
||||||
font-family: 'Roboto', sans-serif;
|
font-family: "Roboto", sans-serif;
|
||||||
padding-top: 4em;
|
padding-top: 4em;
|
||||||
background-color: #fafafa;
|
background-color: #fafafa;
|
||||||
color: #333333;
|
color: #333333;
|
||||||
@@ -13,7 +13,7 @@ body {
|
|||||||
*:hover,
|
*:hover,
|
||||||
*:active,
|
*:active,
|
||||||
*:focus {
|
*:focus {
|
||||||
outline: 0
|
outline: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
@@ -44,7 +44,7 @@ i.spin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#app {
|
#app {
|
||||||
transition: .2s ease padding;
|
transition: 0.2s ease padding;
|
||||||
}
|
}
|
||||||
|
|
||||||
#app.multiple {
|
#app.multiple {
|
||||||
@@ -63,17 +63,17 @@ nav .action {
|
|||||||
display: block;
|
display: block;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
font-size: 1.1em;
|
font-size: 1.1em;
|
||||||
padding: .5em;
|
padding: 0.5em;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
}
|
}
|
||||||
|
|
||||||
nav>div {
|
nav > div {
|
||||||
border-top: 1px solid rgba(0, 0, 0, 0.05);
|
border-top: 1px solid rgba(0, 0, 0, 0.05);
|
||||||
}
|
}
|
||||||
|
|
||||||
nav .action>* {
|
nav .action > * {
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -97,19 +97,25 @@ main {
|
|||||||
|
|
||||||
.breadcrumbs a {
|
.breadcrumbs a {
|
||||||
color: inherit;
|
color: inherit;
|
||||||
transition: .1s ease-in;
|
transition: 0.1s ease-in;
|
||||||
border-radius: .125em;
|
border-radius: 0.125em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumbs a:hover {
|
.breadcrumbs a:hover {
|
||||||
background-color: rgba(0,0,0, 0.05);
|
background-color: rgba(0, 0, 0, 0.05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumbs span a {
|
.breadcrumbs span a {
|
||||||
padding: .2em;
|
padding: 0.2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
#progress {
|
.files {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 30px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
@@ -118,11 +124,11 @@ main {
|
|||||||
z-index: 9999999999;
|
z-index: 9999999999;
|
||||||
}
|
}
|
||||||
|
|
||||||
#progress div {
|
.progress div {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background-color: #40c4ff;
|
background-color: #40c4ff;
|
||||||
width: 0;
|
width: 0;
|
||||||
transition: .2s ease width;
|
transition: 0.2s ease width;
|
||||||
}
|
}
|
||||||
|
|
||||||
.break-word {
|
.break-word {
|
||||||
|
|||||||
@@ -110,4 +110,64 @@
|
|||||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215, U+E0FF, U+EFFD, U+F000;
|
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215, U+E0FF, U+EFFD, U+F000;
|
||||||
}
|
}
|
||||||
|
|
||||||
@import "~material-design-icons/iconfont/material-icons.css";
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-cyrillic-ext.woff2) format('woff2');
|
||||||
|
unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-cyrillic.woff2) format('woff2');
|
||||||
|
unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-greek-ext.woff2) format('woff2');
|
||||||
|
unicode-range: U+1F00-1FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-greek.woff2) format('woff2');
|
||||||
|
unicode-range: U+0370-03FF;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-vietnamese.woff2) format('woff2');
|
||||||
|
unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-latin-ext.woff2) format('woff2');
|
||||||
|
unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Roboto';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
src: local('Roboto Bold'), local('Roboto-Bold'), url(../assets/fonts/roboto/bold-latin.woff2) format('woff2');
|
||||||
|
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215, U+E0FF, U+EFFD, U+F000;
|
||||||
|
}
|
||||||
|
|
||||||
|
@import '~material-icons/iconfont/filled.css';
|
||||||
|
|
||||||
|
.material-icons {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
}
|
||||||
@@ -84,7 +84,8 @@ header .menu-button {
|
|||||||
#search #input {
|
#search #input {
|
||||||
background-color: #f5f5f5;
|
background-color: #f5f5f5;
|
||||||
display: flex;
|
display: flex;
|
||||||
padding: 0.75em;
|
height: 100%;
|
||||||
|
padding: 0em 0.75em;
|
||||||
border-radius: 0.3em;
|
border-radius: 0.3em;
|
||||||
transition: .1s ease all;
|
transition: .1s ease all;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|||||||
203
frontend/src/css/listing-icons.css
Normal file
203
frontend/src/css/listing-icons.css
Normal file
@@ -0,0 +1,203 @@
|
|||||||
|
/* Icons */
|
||||||
|
|
||||||
|
/* General */
|
||||||
|
|
||||||
|
.file-icons [aria-label^="."] { opacity: 0.33 }
|
||||||
|
.file-icons [aria-label$=".bak"] { opacity: 0.33 }
|
||||||
|
|
||||||
|
.file-icons [data-type=audio] i::before { content: 'volume_up' }
|
||||||
|
.file-icons [data-type=blob] i::before { content: 'insert_drive_file' }
|
||||||
|
.file-icons [data-type=image] i::before { content: 'image' }
|
||||||
|
.file-icons [data-type=pdf] i::before { content: 'description' }
|
||||||
|
.file-icons [data-type=text] i::before { content: 'description' }
|
||||||
|
.file-icons [data-type=video] i::before { content: 'movie' }
|
||||||
|
|
||||||
|
/* #f90 - Image */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".ai"] i::before,
|
||||||
|
.file-icons [aria-label$=".odg"] i::before,
|
||||||
|
.file-icons [aria-label$=".xcf"] i::before
|
||||||
|
{ content: 'image' }
|
||||||
|
|
||||||
|
/* #f90 - Presentation */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".odp"] i::before,
|
||||||
|
.file-icons [aria-label$=".ppt"] i::before,
|
||||||
|
.file-icons [aria-label$=".pptx"] i::before
|
||||||
|
{ content: 'slideshow' }
|
||||||
|
|
||||||
|
/* #0f0 - Spreadsheet/Database */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".csv"] i::before,
|
||||||
|
.file-icons [aria-label$=".db"] i::before,
|
||||||
|
.file-icons [aria-label$=".odb"] i::before,
|
||||||
|
.file-icons [aria-label$=".ods"] i::before,
|
||||||
|
.file-icons [aria-label$=".xls"] i::before,
|
||||||
|
.file-icons [aria-label$=".xlsx"] i::before
|
||||||
|
{ content: 'border_all' }
|
||||||
|
|
||||||
|
/* #00f - Document */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".doc"] i::before,
|
||||||
|
.file-icons [aria-label$=".docx"] i::before,
|
||||||
|
.file-icons [aria-label$=".log"] i::before,
|
||||||
|
.file-icons [aria-label$=".odt"] i::before,
|
||||||
|
.file-icons [aria-label$=".rtf"] i::before
|
||||||
|
{ content: 'description' }
|
||||||
|
|
||||||
|
/* #999 - Code */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".c"] i::before,
|
||||||
|
.file-icons [aria-label$=".cpp"] i::before,
|
||||||
|
.file-icons [aria-label$=".cs"] i::before,
|
||||||
|
.file-icons [aria-label$=".css"] i::before,
|
||||||
|
.file-icons [aria-label$=".go"] i::before,
|
||||||
|
.file-icons [aria-label$=".h"] i::before,
|
||||||
|
.file-icons [aria-label$=".html"] i::before,
|
||||||
|
.file-icons [aria-label$=".java"] i::before,
|
||||||
|
.file-icons [aria-label$=".js"] i::before,
|
||||||
|
.file-icons [aria-label$=".json"] i::before,
|
||||||
|
.file-icons [aria-label$=".kt"] i::before,
|
||||||
|
.file-icons [aria-label$=".php"] i::before,
|
||||||
|
.file-icons [aria-label$=".py"] i::before,
|
||||||
|
.file-icons [aria-label$=".rb"] i::before,
|
||||||
|
.file-icons [aria-label$=".rs"] i::before,
|
||||||
|
.file-icons [aria-label$=".vue"] i::before,
|
||||||
|
.file-icons [aria-label$=".xml"] i::before,
|
||||||
|
.file-icons [aria-label$=".yml"] i::before
|
||||||
|
{ content: 'code' }
|
||||||
|
|
||||||
|
/* #999 - Executable */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".apk"] i::before,
|
||||||
|
.file-icons [aria-label$=".bat"] i::before,
|
||||||
|
.file-icons [aria-label$=".exe"] i::before,
|
||||||
|
.file-icons [aria-label$=".jar"] i::before,
|
||||||
|
.file-icons [aria-label$=".ps1"] i::before,
|
||||||
|
.file-icons [aria-label$=".sh"] i::before
|
||||||
|
{ content: 'web_asset' }
|
||||||
|
|
||||||
|
/* #999 - Installer */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".deb"] i::before,
|
||||||
|
.file-icons [aria-label$=".msi"] i::before,
|
||||||
|
.file-icons [aria-label$=".pkg"] i::before,
|
||||||
|
.file-icons [aria-label$=".rpm"] i::before
|
||||||
|
{ content: 'archive' }
|
||||||
|
|
||||||
|
/* #999 - Compressed */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".7z"] i::before,
|
||||||
|
.file-icons [aria-label$=".bz2"] i::before,
|
||||||
|
.file-icons [aria-label$=".cab"] i::before,
|
||||||
|
.file-icons [aria-label$=".gz"] i::before,
|
||||||
|
.file-icons [aria-label$=".rar"] i::before,
|
||||||
|
.file-icons [aria-label$=".tar"] i::before,
|
||||||
|
.file-icons [aria-label$=".xz"] i::before,
|
||||||
|
.file-icons [aria-label$=".zip"] i::before,
|
||||||
|
.file-icons [aria-label$=".zst"] i::before
|
||||||
|
{ content: 'folder_zip' }
|
||||||
|
|
||||||
|
/* #999 - Disk */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".ccd"] i::before,
|
||||||
|
.file-icons [aria-label$=".dmg"] i::before,
|
||||||
|
.file-icons [aria-label$=".iso"] i::before,
|
||||||
|
.file-icons [aria-label$=".mdf"] i::before,
|
||||||
|
.file-icons [aria-label$=".vdi"] i::before,
|
||||||
|
.file-icons [aria-label$=".vhd"] i::before,
|
||||||
|
.file-icons [aria-label$=".vmdk"] i::before,
|
||||||
|
.file-icons [aria-label$=".wim"] i::before
|
||||||
|
{ content: 'album' }
|
||||||
|
|
||||||
|
/* #999 - Font */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".otf"] i::before,
|
||||||
|
.file-icons [aria-label$=".ttf"] i::before,
|
||||||
|
.file-icons [aria-label$=".woff"] i::before,
|
||||||
|
.file-icons [aria-label$=".woff2"] i::before
|
||||||
|
{ content: 'font_download' }
|
||||||
|
|
||||||
|
/* Colors */
|
||||||
|
|
||||||
|
/* General */
|
||||||
|
|
||||||
|
.file-icons [data-type=audio] i { color: var(--icon-yellow) }
|
||||||
|
.file-icons [data-type=image] i { color: var(--icon-orange) }
|
||||||
|
.file-icons [data-type=video] i { color: var(--icon-violet) }
|
||||||
|
|
||||||
|
/* #f00 - Adobe/Oracle */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".ai"] i,
|
||||||
|
.file-icons [aria-label$=".java"] i,
|
||||||
|
.file-icons [aria-label$=".jar"] i,
|
||||||
|
.file-icons [aria-label$=".psd"] i,
|
||||||
|
.file-icons [aria-label$=".rb"] i,
|
||||||
|
.file-icons [data-type=pdf] i
|
||||||
|
{ color: var(--icon-red) }
|
||||||
|
|
||||||
|
/* #f90 - Image/Presentation */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".html"] i,
|
||||||
|
.file-icons [aria-label$=".odg"] i,
|
||||||
|
.file-icons [aria-label$=".odp"] i,
|
||||||
|
.file-icons [aria-label$=".ppt"] i,
|
||||||
|
.file-icons [aria-label$=".pptx"] i,
|
||||||
|
.file-icons [aria-label$=".vue"] i,
|
||||||
|
.file-icons [aria-label$=".xcf"] i
|
||||||
|
{ color: var(--icon-orange) }
|
||||||
|
|
||||||
|
/* #ff0 - Various */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".css"] i,
|
||||||
|
.file-icons [aria-label$=".js"] i,
|
||||||
|
.file-icons [aria-label$=".json"] i,
|
||||||
|
.file-icons [aria-label$=".zip"] i
|
||||||
|
{ color: var(--icon-yellow) }
|
||||||
|
|
||||||
|
/* #0f0 - Spreadsheet/Google */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".apk"] i,
|
||||||
|
.file-icons [aria-label$=".dex"] i,
|
||||||
|
.file-icons [aria-label$=".go"] i,
|
||||||
|
.file-icons [aria-label$=".ods"] i,
|
||||||
|
.file-icons [aria-label$=".xls"] i,
|
||||||
|
.file-icons [aria-label$=".xlsx"] i
|
||||||
|
{ color: var(--icon-green) }
|
||||||
|
|
||||||
|
/* #00f - Document/Microsoft/Apple/Closed */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".aac"] i,
|
||||||
|
.file-icons [aria-label$=".bat"] i,
|
||||||
|
.file-icons [aria-label$=".cab"] i,
|
||||||
|
.file-icons [aria-label$=".cs"] i,
|
||||||
|
.file-icons [aria-label$=".dmg"] i,
|
||||||
|
.file-icons [aria-label$=".doc"] i,
|
||||||
|
.file-icons [aria-label$=".docx"] i,
|
||||||
|
.file-icons [aria-label$=".emf"] i,
|
||||||
|
.file-icons [aria-label$=".exe"] i,
|
||||||
|
.file-icons [aria-label$=".ico"] i,
|
||||||
|
.file-icons [aria-label$=".mp2"] i,
|
||||||
|
.file-icons [aria-label$=".mp3"] i,
|
||||||
|
.file-icons [aria-label$=".mp4"] i,
|
||||||
|
.file-icons [aria-label$=".mpg"] i,
|
||||||
|
.file-icons [aria-label$=".msi"] i,
|
||||||
|
.file-icons [aria-label$=".odt"] i,
|
||||||
|
.file-icons [aria-label$=".ps1"] i,
|
||||||
|
.file-icons [aria-label$=".rtf"] i,
|
||||||
|
.file-icons [aria-label$=".vob"] i,
|
||||||
|
.file-icons [aria-label$=".wim"] i
|
||||||
|
{ color: var(--icon-blue) }
|
||||||
|
|
||||||
|
/* #60f - Various */
|
||||||
|
|
||||||
|
.file-icons [aria-label$=".iso"] i,
|
||||||
|
.file-icons [aria-label$=".php"] i,
|
||||||
|
.file-icons [aria-label$=".rar"] i
|
||||||
|
{ color: var(--icon-violet) }
|
||||||
|
|
||||||
|
/* Overrides */
|
||||||
|
|
||||||
|
.file-icons [data-dir=true] i { color: var(--icon-blue) }
|
||||||
|
.file-icons [data-dir=true] i::before { content: 'folder' }
|
||||||
|
.file-icons [aria-selected=true] i { color: var(--item-selected) }
|
||||||
@@ -1,3 +1,7 @@
|
|||||||
|
#listing {
|
||||||
|
--item-selected: white;
|
||||||
|
}
|
||||||
|
|
||||||
#listing h2 {
|
#listing h2 {
|
||||||
margin: 0 0 0 0.5em;
|
margin: 0 0 0 0.5em;
|
||||||
font-size: .9em;
|
font-size: .9em;
|
||||||
@@ -25,6 +29,7 @@
|
|||||||
transition: .1s ease background, .1s ease opacity;
|
transition: .1s ease background, .1s ease opacity;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
#listing .item div:last-of-type {
|
#listing .item div:last-of-type {
|
||||||
@@ -55,6 +60,7 @@
|
|||||||
#listing .item img {
|
#listing .item img {
|
||||||
width: 4em;
|
width: 4em;
|
||||||
height: 4em;
|
height: 4em;
|
||||||
|
object-fit: cover;
|
||||||
margin-right: 0.1em;
|
margin-right: 0.1em;
|
||||||
vertical-align: bottom;
|
vertical-align: bottom;
|
||||||
}
|
}
|
||||||
@@ -104,6 +110,41 @@
|
|||||||
width: calc(100% - 5vw);
|
width: calc(100% - 5vw);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#listing.mosaic.gallery .item div:first-of-type {
|
||||||
|
width: 100%;
|
||||||
|
height: 12em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#listing.mosaic.gallery .item div:last-of-type {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0.5em;
|
||||||
|
padding: 1em;
|
||||||
|
width: calc(100% - 1em);
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
#listing.mosaic.gallery .item[data-type=image] div:last-of-type {
|
||||||
|
color: white;
|
||||||
|
background: linear-gradient(#0000, #0009);
|
||||||
|
}
|
||||||
|
|
||||||
|
#listing.mosaic.gallery .item i {
|
||||||
|
width: 100%;
|
||||||
|
margin-right: 0;
|
||||||
|
font-size: 8em;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
#listing.mosaic.gallery .item img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#listing.gallery .size,
|
||||||
|
#listing.gallery .modified {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
#listing.list {
|
#listing.list {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -125,7 +166,7 @@
|
|||||||
|
|
||||||
#listing .item[aria-selected=true] {
|
#listing .item[aria-selected=true] {
|
||||||
background: var(--blue) !important;
|
background: var(--blue) !important;
|
||||||
color: #fff !important;
|
color: var(--item-selected) !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
#listing.list .item div:first-of-type {
|
#listing.list .item div:first-of-type {
|
||||||
@@ -236,5 +277,5 @@
|
|||||||
|
|
||||||
#listing #multiple-selection p,
|
#listing #multiple-selection p,
|
||||||
#listing #multiple-selection i {
|
#listing #multiple-selection i {
|
||||||
color: #fff;
|
color: var(--item-selected);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,8 @@
|
|||||||
@import "./base.css";
|
@import "./base.css";
|
||||||
@import "./header.css";
|
@import "./header.css";
|
||||||
@import "./listing.css";
|
@import "./listing.css";
|
||||||
|
@import "./listing-icons.css";
|
||||||
|
@import "./upload-files.css";
|
||||||
@import "./dashboard.css";
|
@import "./dashboard.css";
|
||||||
@import "./login.css";
|
@import "./login.css";
|
||||||
|
|
||||||
@@ -17,6 +19,48 @@
|
|||||||
color: var(--blue);
|
color: var(--blue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
main .spinner {
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
line-height: 0;
|
||||||
|
padding: 1em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
main .spinner > div {
|
||||||
|
width: .8em;
|
||||||
|
height: .8em;
|
||||||
|
margin: 0 .1em;
|
||||||
|
font-size: 1em;
|
||||||
|
background-color: rgba(0, 0, 0, 0.3);
|
||||||
|
border-radius: 100%;
|
||||||
|
display: inline-block;
|
||||||
|
animation: sk-bouncedelay 1.4s infinite ease-in-out both;
|
||||||
|
}
|
||||||
|
|
||||||
|
main .spinner .bounce1 {
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
main .spinner .bounce2 {
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delayed {
|
||||||
|
animation: delayed linear 100ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes delayed {
|
||||||
|
0% {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
99% {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* * * * * * * * * * * * * * * *
|
/* * * * * * * * * * * * * * * *
|
||||||
* ACTION *
|
* ACTION *
|
||||||
* * * * * * * * * * * * * * * */
|
* * * * * * * * * * * * * * * */
|
||||||
@@ -109,6 +153,7 @@
|
|||||||
|
|
||||||
#previewer {
|
#previewer {
|
||||||
background-color: rgba(0, 0, 0, 0.9);
|
background-color: rgba(0, 0, 0, 0.9);
|
||||||
|
padding-top: 4em;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
@@ -142,7 +187,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#previewer .preview {
|
#previewer .preview {
|
||||||
margin-top: 4em;
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
height: calc(100vh - 4em);
|
height: calc(100vh - 4em);
|
||||||
}
|
}
|
||||||
@@ -163,6 +207,34 @@
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#previewer .preview .info {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
font-size: 1.5em;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
#previewer .preview .info .title {
|
||||||
|
margin-bottom: 1em;
|
||||||
|
}
|
||||||
|
#previewer .preview .info .title i {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: .1em;
|
||||||
|
font-size: 4em;
|
||||||
|
}
|
||||||
|
#previewer .preview .info .button {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
#previewer .preview .info .button:hover {
|
||||||
|
background-color: rgba(255, 255, 255, 0.2)
|
||||||
|
}
|
||||||
|
#previewer .preview .info .button i {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-size: 1.3em;
|
||||||
|
}
|
||||||
|
|
||||||
#previewer .pdf {
|
#previewer .pdf {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -204,26 +276,38 @@
|
|||||||
right: 0.5em;
|
right: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#previewer .spinner {
|
||||||
|
text-align: center;
|
||||||
|
position: fixed;
|
||||||
|
top: calc(50% + 1.85em);
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
}
|
||||||
|
|
||||||
|
#previewer .spinner > div {
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
background-color: white;
|
||||||
|
}
|
||||||
|
|
||||||
/* EDITOR */
|
/* EDITOR */
|
||||||
|
|
||||||
#editor-container {
|
#editor-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
background-color: #fafafa;
|
background-color: #fafafa;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
margin-top: 4em;
|
padding-top: 4em;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
z-index: 9999;
|
z-index: 9999;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
#previewer .loading {
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
#editor-container #editor {
|
#editor-container #editor {
|
||||||
height: calc(100vh - 8.4em);
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
#editor-container .breadcrumbs {
|
#editor-container .breadcrumbs {
|
||||||
@@ -232,11 +316,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#editor-container .breadcrumbs span {
|
#editor-container .breadcrumbs span {
|
||||||
font-size: 12px;
|
font-size: .75rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
#editor-container .breadcrumbs i {
|
#editor-container .breadcrumbs i {
|
||||||
font-size: 16px;
|
font-size: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* * * * * * * * * * * * * * * *
|
/* * * * * * * * * * * * * * * *
|
||||||
@@ -252,7 +336,7 @@
|
|||||||
background: rgba(0, 0, 0, 0.05);
|
background: rgba(0, 0, 0, 0.05);
|
||||||
border: 1px solid rgba(0,0,0,0.1);
|
border: 1px solid rgba(0,0,0,0.1);
|
||||||
box-shadow: 0 0 0 0;
|
box-shadow: 0 0 0 0;
|
||||||
font-size: 14px;
|
font-size: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* * * * * * * * * * * * * * * *
|
/* * * * * * * * * * * * * * * *
|
||||||
@@ -283,8 +367,7 @@
|
|||||||
|
|
||||||
@keyframes spin {
|
@keyframes spin {
|
||||||
100% {
|
100% {
|
||||||
-webkit-transform: rotate(-360deg);
|
transform: rotate(360deg);
|
||||||
transform: rotate(-360deg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
61
frontend/src/css/upload-files.css
Normal file
61
frontend/src/css/upload-files.css
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
.upload-files .card.floating {
|
||||||
|
left: auto;
|
||||||
|
top: auto;
|
||||||
|
margin: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
transform: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .file {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .file .file-name {
|
||||||
|
font-size: 1.1em;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .file .file-name i {
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .file .file-progress {
|
||||||
|
margin-top: 2px;
|
||||||
|
width: 100%;
|
||||||
|
height: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .file .file-progress div {
|
||||||
|
height: 100%;
|
||||||
|
background-color: #40c4ff;
|
||||||
|
width: 0;
|
||||||
|
transition: 0.2s ease width;
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files.closed .card-content {
|
||||||
|
display: none;
|
||||||
|
padding: 0em 1em 1em 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files .card .card-title {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 0.8em;
|
||||||
|
padding: 1em 1em 0em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-files.closed .card-title {
|
||||||
|
font-size: 0.7em;
|
||||||
|
padding: 0.5em 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 450px) {
|
||||||
|
.upload-files .card.floating {
|
||||||
|
max-width: 100%;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -8,6 +8,8 @@
|
|||||||
"create": "Create",
|
"create": "Create",
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"download": "Download",
|
"download": "Download",
|
||||||
|
"file": "File",
|
||||||
|
"folder": "Folder",
|
||||||
"hideDotfiles": "Hide dotfiles",
|
"hideDotfiles": "Hide dotfiles",
|
||||||
"info": "Info",
|
"info": "Info",
|
||||||
"more": "More",
|
"more": "More",
|
||||||
@@ -33,7 +35,8 @@
|
|||||||
"switchView": "Switch view",
|
"switchView": "Switch view",
|
||||||
"toggleSidebar": "Toggle sidebar",
|
"toggleSidebar": "Toggle sidebar",
|
||||||
"update": "Update",
|
"update": "Update",
|
||||||
"upload": "Upload"
|
"upload": "Upload",
|
||||||
|
"openFile": "Open file"
|
||||||
},
|
},
|
||||||
"download": {
|
"download": {
|
||||||
"downloadFile": "Download File",
|
"downloadFile": "Download File",
|
||||||
@@ -43,7 +46,8 @@
|
|||||||
"errors": {
|
"errors": {
|
||||||
"forbidden": "You don't have permissions to access this.",
|
"forbidden": "You don't have permissions to access this.",
|
||||||
"internal": "Something really went wrong.",
|
"internal": "Something really went wrong.",
|
||||||
"notFound": "This location can't be reached."
|
"notFound": "This location can't be reached.",
|
||||||
|
"connection": "The server can't be reached."
|
||||||
},
|
},
|
||||||
"files": {
|
"files": {
|
||||||
"body": "Body",
|
"body": "Body",
|
||||||
@@ -61,7 +65,8 @@
|
|||||||
"size": "Size",
|
"size": "Size",
|
||||||
"sortByLastModified": "Sort by last modified",
|
"sortByLastModified": "Sort by last modified",
|
||||||
"sortByName": "Sort by name",
|
"sortByName": "Sort by name",
|
||||||
"sortBySize": "Sort by size"
|
"sortBySize": "Sort by size",
|
||||||
|
"noPreview": "Preview is not available for this file."
|
||||||
},
|
},
|
||||||
"help": {
|
"help": {
|
||||||
"click": "select file or directory",
|
"click": "select file or directory",
|
||||||
@@ -93,7 +98,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "Romanian",
|
"ro": "Romanian",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "Swedish (Sweden)",
|
"svSE": "Swedish (Sweden)",
|
||||||
|
"tr": "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
@@ -143,6 +151,7 @@
|
|||||||
"show": "Show",
|
"show": "Show",
|
||||||
"size": "Size",
|
"size": "Size",
|
||||||
"upload": "Upload",
|
"upload": "Upload",
|
||||||
|
"uploadFiles": "Uploading {files} files...",
|
||||||
"uploadMessage": "Select an option to upload.",
|
"uploadMessage": "Select an option to upload.",
|
||||||
"optionalPassword": "Optional password"
|
"optionalPassword": "Optional password"
|
||||||
},
|
},
|
||||||
@@ -211,9 +220,11 @@
|
|||||||
"rules": "Rules",
|
"rules": "Rules",
|
||||||
"rulesHelp": "Here you can define a set of allow and disallow rules for this specific user. The blocked files won't show up in the listings and they wont be accessible to the user. We support regex and paths relative to the users scope.\n",
|
"rulesHelp": "Here you can define a set of allow and disallow rules for this specific user. The blocked files won't show up in the listings and they wont be accessible to the user. We support regex and paths relative to the users scope.\n",
|
||||||
"scope": "Scope",
|
"scope": "Scope",
|
||||||
|
"setDateFormat": "Set exact date format",
|
||||||
"settingsUpdated": "Settings updated!",
|
"settingsUpdated": "Settings updated!",
|
||||||
"shareDuration": "Share Duration",
|
"shareDuration": "Share Duration",
|
||||||
"shareManagement": "Share Management",
|
"shareManagement": "Share Management",
|
||||||
|
"shareDeleted": "Share deleted!",
|
||||||
"singleClick": "Use single clicks to open files and directories",
|
"singleClick": "Use single clicks to open files and directories",
|
||||||
"themes": {
|
"themes": {
|
||||||
"dark": "Dark",
|
"dark": "Dark",
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -16,6 +16,8 @@ import pt from "./pt.json";
|
|||||||
import ptBR from "./pt-br.json";
|
import ptBR from "./pt-br.json";
|
||||||
import ro from "./ro.json";
|
import ro from "./ro.json";
|
||||||
import ru from "./ru.json";
|
import ru from "./ru.json";
|
||||||
|
import sk from "./sk.json";
|
||||||
|
import ua from "./ua.json";
|
||||||
import svSE from "./sv-se.json";
|
import svSE from "./sv-se.json";
|
||||||
import zhCN from "./zh-cn.json";
|
import zhCN from "./zh-cn.json";
|
||||||
import zhTW from "./zh-tw.json";
|
import zhTW from "./zh-tw.json";
|
||||||
@@ -70,6 +72,12 @@ export function detectLocale() {
|
|||||||
case /^ko.*/i.test(locale):
|
case /^ko.*/i.test(locale):
|
||||||
locale = "ko";
|
locale = "ko";
|
||||||
break;
|
break;
|
||||||
|
case /^sk.*/i.test(locale):
|
||||||
|
locale = "sk";
|
||||||
|
break;
|
||||||
|
case /^ua.*/i.test(locale):
|
||||||
|
locale = "ua";
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
locale = "en";
|
locale = "en";
|
||||||
}
|
}
|
||||||
@@ -107,7 +115,9 @@ const i18n = new VueI18n({
|
|||||||
pt: removeEmpty(pt),
|
pt: removeEmpty(pt),
|
||||||
ru: removeEmpty(ru),
|
ru: removeEmpty(ru),
|
||||||
ro: removeEmpty(ro),
|
ro: removeEmpty(ro),
|
||||||
|
sk: removeEmpty(sk),
|
||||||
"sv-se": removeEmpty(svSE),
|
"sv-se": removeEmpty(svSE),
|
||||||
|
ua: removeEmpty(ua),
|
||||||
"zh-cn": removeEmpty(zhCN),
|
"zh-cn": removeEmpty(zhCN),
|
||||||
"zh-tw": removeEmpty(zhTW),
|
"zh-tw": removeEmpty(zhTW),
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Portugees (Brazilië)",
|
"ptBR": "Portugees (Brazilië)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Russisch",
|
"ru": "Russisch",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "Chinees (vereenvoudigd)",
|
"zhCN": "Chinees (vereenvoudigd)",
|
||||||
"zhTW": "Chinees (traditioneel)"
|
"zhTW": "Chinees (traditioneel)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Russo",
|
"ru": "Russo",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "Chinês simplificado",
|
"zhCN": "Chinês simplificado",
|
||||||
"zhTW": "Chinês tradicional"
|
"zhTW": "Chinês tradicional"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -8,6 +8,8 @@
|
|||||||
"create": "Создать",
|
"create": "Создать",
|
||||||
"delete": "Удалить",
|
"delete": "Удалить",
|
||||||
"download": "Скачать",
|
"download": "Скачать",
|
||||||
|
"file": "Файл",
|
||||||
|
"folder": "Папка",
|
||||||
"hideDotfiles": "Скрыть точечные файлы",
|
"hideDotfiles": "Скрыть точечные файлы",
|
||||||
"info": "Инфо",
|
"info": "Инфо",
|
||||||
"more": "Еще",
|
"more": "Еще",
|
||||||
@@ -29,10 +31,12 @@
|
|||||||
"selectMultiple": "Мультивыбор",
|
"selectMultiple": "Мультивыбор",
|
||||||
"share": "Поделиться",
|
"share": "Поделиться",
|
||||||
"shell": "Командная строка",
|
"shell": "Командная строка",
|
||||||
|
"submit": "Отправить",
|
||||||
"switchView": "Вид",
|
"switchView": "Вид",
|
||||||
"toggleSidebar": "Боковая панель",
|
"toggleSidebar": "Боковая панель",
|
||||||
"update": "Обновить",
|
"update": "Обновить",
|
||||||
"upload": "Загрузить"
|
"upload": "Загрузить",
|
||||||
|
"openFile": "Открыть файл"
|
||||||
},
|
},
|
||||||
"download": {
|
"download": {
|
||||||
"downloadFile": "Скачать файл",
|
"downloadFile": "Скачать файл",
|
||||||
@@ -42,7 +46,8 @@
|
|||||||
"errors": {
|
"errors": {
|
||||||
"forbidden": "У вас нет прав доступа к этому.",
|
"forbidden": "У вас нет прав доступа к этому.",
|
||||||
"internal": "Что-то пошло не так.",
|
"internal": "Что-то пошло не так.",
|
||||||
"notFound": "Неправильная ссылка."
|
"notFound": "Неправильная ссылка.",
|
||||||
|
"connection": "Нет подключения к серверу."
|
||||||
},
|
},
|
||||||
"files": {
|
"files": {
|
||||||
"body": "Тело",
|
"body": "Тело",
|
||||||
@@ -60,7 +65,8 @@
|
|||||||
"size": "Размер",
|
"size": "Размер",
|
||||||
"sortByLastModified": "Сортировка по дате изменения",
|
"sortByLastModified": "Сортировка по дате изменения",
|
||||||
"sortByName": "Сортировка по имени",
|
"sortByName": "Сортировка по имени",
|
||||||
"sortBySize": "Сортировка по размеру"
|
"sortBySize": "Сортировка по размеру",
|
||||||
|
"noPreview": "Предварительный просмотр для этого файла недоступен."
|
||||||
},
|
},
|
||||||
"help": {
|
"help": {
|
||||||
"click": "выбрать файл или каталог",
|
"click": "выбрать файл или каталог",
|
||||||
@@ -82,17 +88,20 @@
|
|||||||
"en": "English",
|
"en": "English",
|
||||||
"es": "Español",
|
"es": "Español",
|
||||||
"fr": "Français",
|
"fr": "Français",
|
||||||
"is": "",
|
"is": "Icelandic",
|
||||||
"it": "Italiano",
|
"it": "Italiano",
|
||||||
"ja": "日本語",
|
"ja": "日本語",
|
||||||
"ko": "한국어",
|
"ko": "한국어",
|
||||||
"nlBE": "",
|
"nlBE": "Dutch (Belgium)",
|
||||||
"pl": "Polski",
|
"pl": "Polski",
|
||||||
"pt": "Português",
|
"pt": "Português",
|
||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "Romanian",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
"svSE": "",
|
"sk": "Slovenčina",
|
||||||
|
"svSE": "Swedish (Sweden)",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
@@ -115,10 +124,11 @@
|
|||||||
"currentlyNavigating": "Текущий каталог:",
|
"currentlyNavigating": "Текущий каталог:",
|
||||||
"deleteMessageMultiple": "Удалить эти файлы ({count})?",
|
"deleteMessageMultiple": "Удалить эти файлы ({count})?",
|
||||||
"deleteMessageSingle": "Удалить этот файл/каталог?",
|
"deleteMessageSingle": "Удалить этот файл/каталог?",
|
||||||
|
"deleteMessageShare": "Удалить этот общий файл/каталог ({path})?",
|
||||||
"deleteTitle": "Удалить файлы",
|
"deleteTitle": "Удалить файлы",
|
||||||
"displayName": "Отображаемое имя:",
|
"displayName": "Отображаемое имя:",
|
||||||
"download": "Скачать файлы",
|
"download": "Скачать файлы",
|
||||||
"downloadMessage": "Выберите формат а котором хотите скачать.",
|
"downloadMessage": "Выберите формат в котором хотите скачать.",
|
||||||
"error": "Ошибка",
|
"error": "Ошибка",
|
||||||
"fileInfo": "Информация о файле",
|
"fileInfo": "Информация о файле",
|
||||||
"filesSelected": "Файлов выбрано: {count}.",
|
"filesSelected": "Файлов выбрано: {count}.",
|
||||||
@@ -141,7 +151,8 @@
|
|||||||
"show": "Показать",
|
"show": "Показать",
|
||||||
"size": "Размер",
|
"size": "Размер",
|
||||||
"upload": "Загрузить",
|
"upload": "Загрузить",
|
||||||
"uploadMessage": "Выберите вариант для загрузки."
|
"uploadMessage": "Выберите вариант для загрузки.",
|
||||||
|
"optionalPassword": "Необязательный пароль"
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"images": "Изображения",
|
"images": "Изображения",
|
||||||
@@ -182,7 +193,7 @@
|
|||||||
"hideDotfiles": "Скрыть точечные файлы",
|
"hideDotfiles": "Скрыть точечные файлы",
|
||||||
"insertPath": "Вставьте путь",
|
"insertPath": "Вставьте путь",
|
||||||
"insertRegex": "Вставить регулярное выражение",
|
"insertRegex": "Вставить регулярное выражение",
|
||||||
"instanceName": "Instance name",
|
"instanceName": "Текущее название программы",
|
||||||
"language": "Язык",
|
"language": "Язык",
|
||||||
"lockPassword": "Запретить пользователю менять пароль",
|
"lockPassword": "Запретить пользователю менять пароль",
|
||||||
"newPassword": "Новый пароль",
|
"newPassword": "Новый пароль",
|
||||||
@@ -208,9 +219,11 @@
|
|||||||
"rules": "Права",
|
"rules": "Права",
|
||||||
"rulesHelp": "Здесь вы можете определить набор разрешающих и запрещающих правил для этого конкретного пользователь. Блокированные файлы не будут отображаться в списках, и не будут доступны для пользователя. Есть поддержка регулярных выражений и относительных путей.\n",
|
"rulesHelp": "Здесь вы можете определить набор разрешающих и запрещающих правил для этого конкретного пользователь. Блокированные файлы не будут отображаться в списках, и не будут доступны для пользователя. Есть поддержка регулярных выражений и относительных путей.\n",
|
||||||
"scope": "Корень",
|
"scope": "Корень",
|
||||||
|
"setDateFormat": "Установить точный формат даты",
|
||||||
"settingsUpdated": "Настройки применены!",
|
"settingsUpdated": "Настройки применены!",
|
||||||
"shareDuration": "Время расшаренной ссылки",
|
"shareDuration": "Время расшаренной ссылки",
|
||||||
"shareManagement": "Управление расшаренными ссылками",
|
"shareManagement": "Управление расшаренными ссылками",
|
||||||
|
"shareDeleted": "Расшаренная ссылка удалена!",
|
||||||
"singleClick": "Открытие файлов и каталогов одним кликом",
|
"singleClick": "Открытие файлов и каталогов одним кликом",
|
||||||
"themes": {
|
"themes": {
|
||||||
"dark": "Темная",
|
"dark": "Темная",
|
||||||
|
|||||||
266
frontend/src/i18n/sk.json
Normal file
266
frontend/src/i18n/sk.json
Normal file
@@ -0,0 +1,266 @@
|
|||||||
|
{
|
||||||
|
"buttons": {
|
||||||
|
"cancel": "Zrušiť",
|
||||||
|
"close": "Zavrieť",
|
||||||
|
"copy": "Kopírovať",
|
||||||
|
"copyFile": "Kopírovať súbor",
|
||||||
|
"copyToClipboard": "Kopírovať do schránky",
|
||||||
|
"create": "Vytvoriť",
|
||||||
|
"delete": "Odstrániť",
|
||||||
|
"download": "Stiahnuť",
|
||||||
|
"file": "Súbor",
|
||||||
|
"folder": "Priečinok",
|
||||||
|
"hideDotfiles": "Skryť súbory začínajúce bodkou",
|
||||||
|
"info": "Info",
|
||||||
|
"more": "Viac",
|
||||||
|
"move": "Presunúť",
|
||||||
|
"moveFile": "Presunúť súbory",
|
||||||
|
"new": "Nový",
|
||||||
|
"next": "Ďalšie",
|
||||||
|
"ok": "OK",
|
||||||
|
"permalink": "Získať trvalý odkaz",
|
||||||
|
"previous": "Predošlé",
|
||||||
|
"publish": "Zverejniť",
|
||||||
|
"rename": "Premenovať",
|
||||||
|
"replace": "Nahradiť",
|
||||||
|
"reportIssue": "Nahlásiť problém",
|
||||||
|
"save": "Uložiť",
|
||||||
|
"schedule": "Naplánovať",
|
||||||
|
"search": "Hľadať",
|
||||||
|
"select": "Vybrať",
|
||||||
|
"selectMultiple": "Vybrať viaceré",
|
||||||
|
"share": "Zdieľať",
|
||||||
|
"shell": "Prepnúť shell",
|
||||||
|
"submit": "Poslať",
|
||||||
|
"switchView": "Prepnúť pohľad",
|
||||||
|
"toggleSidebar": "Prepnúť sidebar",
|
||||||
|
"update": "Aktualizovať",
|
||||||
|
"upload": "Nahrať",
|
||||||
|
"openFile": "Otvoriť súbor"
|
||||||
|
},
|
||||||
|
"download": {
|
||||||
|
"downloadFile": "Stiahnuť súbor",
|
||||||
|
"downloadFolder": "Stiahnuť priečinok",
|
||||||
|
"downloadSelected": "Stiahnuť vybraté"
|
||||||
|
},
|
||||||
|
"errors": {
|
||||||
|
"forbidden": "You don't have permissions to access this.",
|
||||||
|
"internal": "Something really went wrong.",
|
||||||
|
"notFound": "This location can't be reached.",
|
||||||
|
"connection": "The server can't be reached."
|
||||||
|
},
|
||||||
|
"files": {
|
||||||
|
"body": "Telo",
|
||||||
|
"clear": "Zrušiť výber",
|
||||||
|
"closePreview": "Zavrieť náhľad",
|
||||||
|
"files": "Súbory",
|
||||||
|
"folders": "Priečinky",
|
||||||
|
"home": "Domov",
|
||||||
|
"lastModified": "Posledná zmena",
|
||||||
|
"loading": "Načítanie...",
|
||||||
|
"lonely": "Je tu tak pusto...",
|
||||||
|
"metadata": "Metadata",
|
||||||
|
"multipleSelectionEnabled": "Zapnutý viacnásobný výber",
|
||||||
|
"name": "Názov",
|
||||||
|
"size": "Veľkosť",
|
||||||
|
"sortByLastModified": "Zoradiť podľa dátumu",
|
||||||
|
"sortByName": "Zoradiť podľa názvu",
|
||||||
|
"sortBySize": "Zoradiť podľa veľkosti",
|
||||||
|
"noPreview": "Pre tento súbor nie je dostupný náhľad."
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"click": "vyberie súbor alebo priečinok",
|
||||||
|
"ctrl": {
|
||||||
|
"click": "vyberie viac súborov alebo priečinkov",
|
||||||
|
"f": "otvorí vyhľadávanie",
|
||||||
|
"s": "uloží súbor alebo stiahne priečinok tam kde ste"
|
||||||
|
},
|
||||||
|
"del": "odstráni vybraté položky",
|
||||||
|
"doubleClick": "otvorí súbor alebo priečinok",
|
||||||
|
"esc": "zruší výber a/alebo zavrie okno",
|
||||||
|
"f1": "tieto informácie",
|
||||||
|
"f2": "premenuje súbor",
|
||||||
|
"help": "Pomoc"
|
||||||
|
},
|
||||||
|
"languages": {
|
||||||
|
"ar": "العربية",
|
||||||
|
"de": "Deutsch",
|
||||||
|
"en": "English",
|
||||||
|
"es": "Español",
|
||||||
|
"fr": "Français",
|
||||||
|
"is": "Icelandic",
|
||||||
|
"it": "Italiano",
|
||||||
|
"ja": "日本語",
|
||||||
|
"ko": "한국어",
|
||||||
|
"nlBE": "Dutch (Belgium)",
|
||||||
|
"pl": "Polski",
|
||||||
|
"pt": "Português",
|
||||||
|
"ptBR": "Português (Brasil)",
|
||||||
|
"ro": "Romanian",
|
||||||
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
|
"svSE": "Swedish (Sweden)",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
|
"zhCN": "中文 (简体)",
|
||||||
|
"zhTW": "中文 (繁體)"
|
||||||
|
},
|
||||||
|
"login": {
|
||||||
|
"createAnAccount": "Vytvoriť účet",
|
||||||
|
"loginInstead": "Už mám účet",
|
||||||
|
"password": "Heslo",
|
||||||
|
"passwordConfirm": "Potvrdenie hesla",
|
||||||
|
"passwordsDontMatch": "Heslá nesúhlasia",
|
||||||
|
"signup": "Registrovať",
|
||||||
|
"submit": "Prihlásiť",
|
||||||
|
"username": "Používateľské meno",
|
||||||
|
"usernameTaken": "Meno je už obsadené",
|
||||||
|
"wrongCredentials": "Nesprávne prihlasovacie údaje"
|
||||||
|
},
|
||||||
|
"permanent": "Trvalé",
|
||||||
|
"prompts": {
|
||||||
|
"copy": "Kopírovať",
|
||||||
|
"copyMessage": "Zvoľte miesto, kde chcete kopírovať súbory:",
|
||||||
|
"currentlyNavigating": "Aktuálna cesta:",
|
||||||
|
"deleteMessageMultiple": "Naozaj chcete odstrániť {count} súbor(ov)?",
|
||||||
|
"deleteMessageSingle": "Naozaj chcete odstrániť tento súbor/priečinok?",
|
||||||
|
"deleteMessageShare": "Naozaj chcete odstrániť toto zdieľanie({path})?",
|
||||||
|
"deleteTitle": "Odstránenie súborov",
|
||||||
|
"displayName": "Zobrazený názov:",
|
||||||
|
"download": "Stiahnuť súbory",
|
||||||
|
"downloadMessage": "Vyberte formát, ktorý chcete stiahnuť.",
|
||||||
|
"error": "Niečo sa pokazilo",
|
||||||
|
"fileInfo": "Informácie o súbore",
|
||||||
|
"filesSelected": "{count} súborov vybratých.",
|
||||||
|
"lastModified": "Dátum zmeny",
|
||||||
|
"move": "Presunúť",
|
||||||
|
"moveMessage": "Zvoľte nový domov pre vaše súbory/priečinky:",
|
||||||
|
"newArchetype": "Vytvorí nový príspevok z archetypu. Nový súbor sa vytvorí v priečinku s obsahom.",
|
||||||
|
"newDir": "Nový priečinok",
|
||||||
|
"newDirMessage": "Napíšte názov nového priečinka.",
|
||||||
|
"newFile": "Nový súbor",
|
||||||
|
"newFileMessage": "Napíšte názov nového súboru.",
|
||||||
|
"numberDirs": "Počet priečinkov",
|
||||||
|
"numberFiles": "Počet súborov",
|
||||||
|
"rename": "Premenovať",
|
||||||
|
"renameMessage": "Zadajte nový názov pre",
|
||||||
|
"replace": "Nahradiť",
|
||||||
|
"replaceMessage": "Niektorý nahrávaný súbor je v konflikte názvov. Chcete nahradiť existujúci súbor?\n",
|
||||||
|
"schedule": "Naplánovať",
|
||||||
|
"scheduleMessage": "Pick a date and time to schedule the publication of this post.",
|
||||||
|
"show": "Zobraziť",
|
||||||
|
"size": "Veľkosť",
|
||||||
|
"upload": "Nahrať",
|
||||||
|
"uploadMessage": "Zvoľte možnosť nahrávania.",
|
||||||
|
"optionalPassword": "Voliteľné heslo"
|
||||||
|
},
|
||||||
|
"search": {
|
||||||
|
"images": "Obrázky",
|
||||||
|
"music": "Hudba",
|
||||||
|
"pdf": "PDF",
|
||||||
|
"pressToSearch": "Vyhľadáte stlačením Enter...",
|
||||||
|
"search": "Hľadať...",
|
||||||
|
"typeToSearch": "Vyhľadáte písaním...",
|
||||||
|
"types": "Typy",
|
||||||
|
"video": "Video"
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"admin": "Admin",
|
||||||
|
"administrator": "Administrátor",
|
||||||
|
"allowCommands": "Vykonávať príkazy",
|
||||||
|
"allowEdit": "Upravovať, premenovať a odstraňovať súbory a priečinky",
|
||||||
|
"allowNew": "Vytvárať nové súbory a priečinky",
|
||||||
|
"allowPublish": "Zverejňovať nové príspevky a stránky",
|
||||||
|
"allowSignup": "Povoliť registráciu používateľov",
|
||||||
|
"avoidChanges": "(nechajte prázdne, aby sa nezmenilo)",
|
||||||
|
"branding": "Vlastný vzhľad",
|
||||||
|
"brandingDirectoryPath": "Cesta k priečinku s vlastným vzhľadom",
|
||||||
|
"brandingHelp": "Môžete si prispôsobiť ako bude vyzerá váš File Browser instance zmenou jeho názvu, výmenou loga a pridaním vlastný štýlov alebo vypnutím externých odkazov na GitHub.\nViac informácií o vlastnom vzhľade nájdete na {0}.",
|
||||||
|
"changePassword": "Zmeniť heslo",
|
||||||
|
"commandRunner": "Spúšťač príkazov",
|
||||||
|
"commandRunnerHelp": "Sem môžete nastaviť príkazy, ktoré sa vykonajú pri určitých udalostiach. Musíte písať jeden na riadok. Premenné prostredia {0} a {1} sú k dispozícii, s tým že {0} relatívne k {1}. Viac informácií o tejto funkcionalite a dostupných premenných prostredia nájdete na {2}.",
|
||||||
|
"commandsUpdated": "Príkazy upravené!",
|
||||||
|
"createUserDir": "Automaticky vytvoriť domovský priečinok pri pridaní používateľa",
|
||||||
|
"customStylesheet": "Vlastný Stylesheet",
|
||||||
|
"defaultUserDescription": "Toto sú predvolané nastavenia nového používateľa.",
|
||||||
|
"disableExternalLinks": "Vypnúť externé odkazy (okrem dokumentácie)",
|
||||||
|
"documentation": "dokumentácia",
|
||||||
|
"examples": "Príklady",
|
||||||
|
"executeOnShell": "Vykonať cez shell",
|
||||||
|
"executeOnShellDescription": "Predvolene File Browser vykonáva príkazy volaním priamo ich binárok. Ak ich chcete spúšťať cez shell (napr. Bash alebo PowerShell), môžete ho napísať sem a pridať potrebné argumenty a flagy. Ak je nastavený, tak sa príkazy budú spúšťať pridaním na koniec ako argument. Toto sa týka používateľských príkazov aj udalostí.",
|
||||||
|
"globalRules": "Toto je globálne nastavenie pravidiel. Aplikujú sa na všetkých používateľov. Môžete definovať špecifické pravidlá pre každého používateľa a prekryť tak pravidlá nastavené tu.",
|
||||||
|
"globalSettings": "Globálne nastavenia",
|
||||||
|
"hideDotfiles": "Skryť súroby začínajúce bodkou",
|
||||||
|
"insertPath": "Vložte cestu",
|
||||||
|
"insertRegex": "Vložte regex výraz",
|
||||||
|
"instanceName": "Názov inštalácie",
|
||||||
|
"language": "Jazyk",
|
||||||
|
"lockPassword": "Zabrániť používateľovi meniť heslo",
|
||||||
|
"newPassword": "Nové heslo",
|
||||||
|
"newPasswordConfirm": "Potvrenie nového hesla",
|
||||||
|
"newUser": "Nový používateľ",
|
||||||
|
"password": "Heslo",
|
||||||
|
"passwordUpdated": "Heslo zmenené!",
|
||||||
|
"path": "Cesta",
|
||||||
|
"perm": {
|
||||||
|
"create": "Vytvárať súbory a priečinky",
|
||||||
|
"delete": "Odstraňovať súbory a priečinky",
|
||||||
|
"download": "Stiahnuť",
|
||||||
|
"execute": "Vykonávať príkazy",
|
||||||
|
"modify": "Upravovať súbory",
|
||||||
|
"rename": "Premenovať a presúvať súbory a priečinky",
|
||||||
|
"share": "Zdieľať súbory"
|
||||||
|
},
|
||||||
|
"permissions": "Práva",
|
||||||
|
"permissionsHelp": "Môžete nastaviť používateľa, aby bol administrátorom alebo vybrať práva jednotlivo. Ak zvolíte \"Administrator\", všetky ďalši budú automaticky zaškrtnuté. Manažment používateľov ostáva v správe administrátora.\n",
|
||||||
|
"profileSettings": "Nastavenia profilu",
|
||||||
|
"ruleExample1": "blokuje prístup ku všetkým súborom začínajúcim bodkou (napríklad .git, .gitignore) v každom priečinku.\n",
|
||||||
|
"ruleExample2": "blokuje prístup k súborom s názvom Caddyfile v koreňovom priečinku.",
|
||||||
|
"rules": "Pravidlá",
|
||||||
|
"rulesHelp": "Tu môžete definovať pravidilá pre konkrétneho používateľa. Blokované súbory používateľ nebude vidieť a ani nebude k nim mať prístup. Podporujeme regex a cesty relatívne k používateľovi.\n",
|
||||||
|
"scope": "Scope",
|
||||||
|
"settingsUpdated": "Nastavenia upravené!",
|
||||||
|
"shareDuration": "Trvanie zdieľania",
|
||||||
|
"shareManagement": "Správa zdieľania",
|
||||||
|
"shareDeleted": "Zdieľanie odstránené!",
|
||||||
|
"singleClick": "Používať jeden klik na otváranie súborov a priečinkov",
|
||||||
|
"themes": {
|
||||||
|
"dark": "Tmavá",
|
||||||
|
"light": "Svetlá",
|
||||||
|
"title": "Téma"
|
||||||
|
},
|
||||||
|
"user": "Používateľ",
|
||||||
|
"userCommands": "Príkazy",
|
||||||
|
"userCommandsHelp": "Zoznam povolených príkazov oddelených medzerou pre tohoto používateľa. Napríklad:\n",
|
||||||
|
"userCreated": "Používateľ vytvorený!",
|
||||||
|
"userDefaults": "Predovolené nastavenia používateľa",
|
||||||
|
"userDeleted": "Používateľ odstránený!",
|
||||||
|
"userManagement": "Správa používateľov",
|
||||||
|
"userUpdated": "Používateľ upravený!",
|
||||||
|
"username": "Meno používateľa",
|
||||||
|
"users": "Používatelia"
|
||||||
|
},
|
||||||
|
"sidebar": {
|
||||||
|
"help": "Pomoc",
|
||||||
|
"hugoNew": "Nový Hugo",
|
||||||
|
"login": "Prihlásiť",
|
||||||
|
"logout": "Odhlásiť",
|
||||||
|
"myFiles": "Moje súbory",
|
||||||
|
"newFile": "Nový súbor",
|
||||||
|
"newFolder": "Nový priečinok",
|
||||||
|
"preview": "Náhľad",
|
||||||
|
"settings": "Nastavenia",
|
||||||
|
"signup": "Registrovať",
|
||||||
|
"siteSettings": "Nastavenia stránky"
|
||||||
|
},
|
||||||
|
"success": {
|
||||||
|
"linkCopied": "Odkaz skopírovaný!"
|
||||||
|
},
|
||||||
|
"time": {
|
||||||
|
"days": "Dni",
|
||||||
|
"hours": "Hodiny",
|
||||||
|
"minutes": "Minúty",
|
||||||
|
"seconds": "Sekundy",
|
||||||
|
"unit": "Jednotka času"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "",
|
"ro": "",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "",
|
"svSE": "",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
264
frontend/src/i18n/tr.json
Normal file
264
frontend/src/i18n/tr.json
Normal file
@@ -0,0 +1,264 @@
|
|||||||
|
{
|
||||||
|
"buttons": {
|
||||||
|
"cancel": "Vazgeç",
|
||||||
|
"close": "Kapat",
|
||||||
|
"copy": "Kopyala",
|
||||||
|
"copyFile": "Dosyayı kopyala",
|
||||||
|
"copyToClipboard": "Panoya kopyala",
|
||||||
|
"create": "Oluştur",
|
||||||
|
"delete": "Sil",
|
||||||
|
"download": "İndir",
|
||||||
|
"hideDotfiles": "Nokta dosyalarını gizle",
|
||||||
|
"info": "Bilgi",
|
||||||
|
"more": "Daha fazla",
|
||||||
|
"move": "Taşı",
|
||||||
|
"moveFile": "Dosyayı taşı",
|
||||||
|
"new": "Yeni",
|
||||||
|
"next": "Sonraki",
|
||||||
|
"ok": "Tamam",
|
||||||
|
"permalink": "Kalıcı Bağlantı Alın",
|
||||||
|
"previous": "Önceki",
|
||||||
|
"publish": "Yayınla",
|
||||||
|
"rename": "Yeniden anlandır",
|
||||||
|
"replace": "Değiştir",
|
||||||
|
"reportIssue": "Sorun bildir",
|
||||||
|
"save": "Kaydet",
|
||||||
|
"schedule": "Planla",
|
||||||
|
"search": "Ara",
|
||||||
|
"select": "Seç",
|
||||||
|
"selectMultiple": "Çoklu seçim",
|
||||||
|
"share": "Paylaş",
|
||||||
|
"shell": "Komut satırı aç/kapat",
|
||||||
|
"submit": "Gönder",
|
||||||
|
"switchView": "Görünümü değiştir",
|
||||||
|
"toggleSidebar": "Menüyü aç/kapat",
|
||||||
|
"update": "Güncelle",
|
||||||
|
"upload": "Yükle",
|
||||||
|
"openFile": "Dosyayı aç"
|
||||||
|
},
|
||||||
|
"download": {
|
||||||
|
"downloadFile": "Dosyayı indir",
|
||||||
|
"downloadFolder": "Klasörü indir",
|
||||||
|
"downloadSelected": "Seçilileri indir"
|
||||||
|
},
|
||||||
|
"errors": {
|
||||||
|
"forbidden": "Buna erişim izniniz yok.",
|
||||||
|
"internal": "Bir şeyler ters gitti.",
|
||||||
|
"notFound": "Bu konuma ulaşılamıyor.",
|
||||||
|
"connection": "Sunucuya ulaşılamıyor."
|
||||||
|
},
|
||||||
|
"files": {
|
||||||
|
"body": "Sayfa",
|
||||||
|
"clear": "Temizle",
|
||||||
|
"closePreview": "Önizlemeyi kapat",
|
||||||
|
"files": "Dosyalar",
|
||||||
|
"folders": "Klasörler",
|
||||||
|
"home": "Ana dizin",
|
||||||
|
"lastModified": "Son güncellenme",
|
||||||
|
"loading": "Yükleniyor...",
|
||||||
|
"lonely": "Burada yalnızlık hissediyorum...",
|
||||||
|
"metadata": "meta veri",
|
||||||
|
"multipleSelectionEnabled": "Çoklu seçim etkin",
|
||||||
|
"name": "İsim",
|
||||||
|
"size": "Boyut",
|
||||||
|
"sortByLastModified": "Güncelleme tarihine göre sırala",
|
||||||
|
"sortByName": "İsme göre sırala",
|
||||||
|
"sortBySize": "Boyuta göre sırala",
|
||||||
|
"noPreview": "Bu dosya için önizleme aktif değil"
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"click": "dosya veya klasör seçin",
|
||||||
|
"ctrl": {
|
||||||
|
"click": "çoklu dosya ve klasör seçin",
|
||||||
|
"f": "Aramayı aç",
|
||||||
|
"s": "bir dosyayı kaydedin veya bulunduğunuz dizini indirin"
|
||||||
|
},
|
||||||
|
"del": "seçilileri sil",
|
||||||
|
"doubleClick": "dosya veya dizini açın",
|
||||||
|
"esc": "seçimi temizle veya kapatın",
|
||||||
|
"f1": "bu bilgi",
|
||||||
|
"f2": "dosyayı yeniden adlandır",
|
||||||
|
"help": "Yardım"
|
||||||
|
},
|
||||||
|
"languages": {
|
||||||
|
"ar": "العربية",
|
||||||
|
"de": "Deutsch",
|
||||||
|
"en": "English",
|
||||||
|
"es": "Español",
|
||||||
|
"fr": "Français",
|
||||||
|
"is": "Icelandic",
|
||||||
|
"it": "Italiano",
|
||||||
|
"ja": "日本語",
|
||||||
|
"ko": "한국어",
|
||||||
|
"nlBE": "Dutch (Belgium)",
|
||||||
|
"pl": "Polski",
|
||||||
|
"pt": "Português",
|
||||||
|
"ptBR": "Português (Brasil)",
|
||||||
|
"ro": "Romanian",
|
||||||
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
|
"svSE": "Swedish (Sweden)",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
|
"zhCN": "中文 (简体)",
|
||||||
|
"zhTW": "中文 (繁體)"
|
||||||
|
},
|
||||||
|
"login": {
|
||||||
|
"createAnAccount": "Bir hesap oluşturun",
|
||||||
|
"loginInstead": "Zaten hesabınız var mı",
|
||||||
|
"password": "Şifre",
|
||||||
|
"passwordConfirm": "Şifre tekrarı",
|
||||||
|
"passwordsDontMatch": "Şifreler uyuşmuyor",
|
||||||
|
"signup": "Üye Ol",
|
||||||
|
"submit": "Giriş",
|
||||||
|
"username": "Kullanıcı adı",
|
||||||
|
"usernameTaken": "Kullanıcı adı mevcut",
|
||||||
|
"wrongCredentials": "Yanlış hesap bilgileri"
|
||||||
|
},
|
||||||
|
"permanent": "Kalıcı",
|
||||||
|
"prompts": {
|
||||||
|
"copy": "Kopyala",
|
||||||
|
"copyMessage": "Dosyalarınızı kopyalayacağınız yeri seçin:",
|
||||||
|
"currentlyNavigating": "Şu anki lokasyon:",
|
||||||
|
"deleteMessageMultiple": "{count} dosyayı/dosyaları silmek istediğinizden emin misiniz?",
|
||||||
|
"deleteMessageSingle": "Bu dosyayı/klasörü silmek istediğinizden emin misiniz?",
|
||||||
|
"deleteMessageShare": "Bu paylaşımı({path}) silmek istediğinizden emin misiniz?",
|
||||||
|
"deleteTitle": "Dosyaları sil",
|
||||||
|
"displayName": "Görünen Ad:",
|
||||||
|
"download": "Dosyaları indirŞ",
|
||||||
|
"downloadMessage": "İndirmek istediğiniz formatı seçin.",
|
||||||
|
"error": "Bir şeyler yanlış gitti",
|
||||||
|
"fileInfo": "Dosya bilgisi",
|
||||||
|
"filesSelected": "{count} dosya seçildi.",
|
||||||
|
"lastModified": "Son güncellenme",
|
||||||
|
"move": "Taşı",
|
||||||
|
"moveMessage": "Dosya(lar)ınız/klasör(ler)iniz için yeni ana dizin seçin:",
|
||||||
|
"newArchetype": "Bir prototip temelinde yeni bir gönderi oluşturun. Dosyanız içerik klasöründe oluşturulacaktır.",
|
||||||
|
"newDir": "Yeni dizin",
|
||||||
|
"newDirMessage": "Yeni dizinin adını yazın.",
|
||||||
|
"newFile": "Yeni dosya",
|
||||||
|
"newFileMessage": "Yeni dosyanın adını yazın.",
|
||||||
|
"numberDirs": "Dizin sayısı",
|
||||||
|
"numberFiles": "Dosya sayısı",
|
||||||
|
"rename": "Yeniden adlandır",
|
||||||
|
"renameMessage": "için yeni bir ad girin",
|
||||||
|
"replace": "Değiştir",
|
||||||
|
"replaceMessage": "Yüklemeye çalıştığınız dosyalardan biri, adı nedeniyle çakışıyor. Mevcut olanı değiştirmek istiyor musunuz?\n",
|
||||||
|
"schedule": "Planla",
|
||||||
|
"scheduleMessage": "Bu paylaşımın yayınlanmasını planlamak için bir tarih ve saat seçin.",
|
||||||
|
"show": "Göster",
|
||||||
|
"size": "Boyut",
|
||||||
|
"upload": "Gönder",
|
||||||
|
"uploadMessage": "Yüklemek için bir seçenek belirleyin.",
|
||||||
|
"optionalPassword": "İsteğe bağlı şifre"
|
||||||
|
},
|
||||||
|
"search": {
|
||||||
|
"images": "Görseller",
|
||||||
|
"music": "Müzik",
|
||||||
|
"pdf": "PDF",
|
||||||
|
"pressToSearch": "Aramak için enter'a basın...",
|
||||||
|
"search": "Ara...",
|
||||||
|
"typeToSearch": "Aramak için yazın...",
|
||||||
|
"types": "Türler",
|
||||||
|
"video": "Video"
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"admin": "Yönetim",
|
||||||
|
"administrator": "Yönetici",
|
||||||
|
"allowCommands": "Komutları çalıştır",
|
||||||
|
"allowEdit": "Dosyaları veya dizinleri düzenleyin, yeniden adlandırın ve silin",
|
||||||
|
"allowNew": "Yeni dosyalar ve dizinler oluşturun",
|
||||||
|
"allowPublish": "Yeni linkler ve sayfaları yayınlayın",
|
||||||
|
"allowSignup": "Kullanıcıların kaydolmasına izin ver",
|
||||||
|
"avoidChanges": "(değişiklikleri önlemek için boş bırakın)",
|
||||||
|
"branding": "Marka",
|
||||||
|
"brandingDirectoryPath": "Marka dizin yolu",
|
||||||
|
"brandingHelp": "Adını değiştirerek, logoyu değiştirerek, özel stiller ekleyerek ve hatta GitHub'a harici bağlantıları devre dışı bırakarak Filebrowser örneğinizin görünüşünü ve hissini özelleştirebilirsiniz.\nÖzel marka bilinci oluşturma hakkında daha fazla bilgi için lütfen {0} sayfasına göz atın.",
|
||||||
|
"changePassword": "Şifre Değiştir",
|
||||||
|
"commandRunner": "Komut satırı",
|
||||||
|
"commandRunnerHelp": "Burada, adlandırılmış olaylarda yürütülen komutları ayarlayabilirsiniz. Her satıra bir tane yazmalısınız. {0} ve {1} ortam değişkenleri, {1}'ye göre {0} olacak şekilde kullanılabilir olacaktır. Bu özellik ve mevcut ortam değişkenleri hakkında daha fazla bilgi için lütfen {2}'yi okuyun.",
|
||||||
|
"commandsUpdated": "Komutlar güncellendi!",
|
||||||
|
"createUserDir": "Kullanıcı eklerken, kullanıcı ana dizinini otomatik oluştur",
|
||||||
|
"customStylesheet": "Özel CSS",
|
||||||
|
"defaultUserDescription": "Bu, yeni kullanıcılar için varsayılan ayarlardır.",
|
||||||
|
"disableExternalLinks": "Harici bağlantıları devre dışı bırakın (dökümantasyon hariç)",
|
||||||
|
"documentation": "dökümantasyon",
|
||||||
|
"examples": "Örnekler",
|
||||||
|
"executeOnShell": "Komut satırında çalıştır",
|
||||||
|
"executeOnShellDescription": "Varsayılan olarak, FileBrowser komutları doğrudan dosyaları çağırarak yürütür. Bunları komut satırında çalıştırmak istiyorsanız (Bash veya PowerShell gibi), burada gerekli argümanlar ve flagler tanımlayabilirsiniz. Ayarlanırsa, yürüttüğünüz komut argüman olarak eklenir. Bu, hem kullanıcı komutları hem de event hooklar için geçerlidir.",
|
||||||
|
"globalRules": "Bu, genel bir izin verme ve izin vermeme kurallar bütünüdür. Her kullanıcı için geçerlidirler. Bunları geçersiz kılmak için her kullanıcının ayarlarında belirli kurallar tanımlayabilirsiniz.",
|
||||||
|
"globalSettings": "Genel Ayarlar",
|
||||||
|
"hideDotfiles": ". ile başlayan dosyaları gizle",
|
||||||
|
"insertPath": "Dizini ekle",
|
||||||
|
"insertRegex": "Regex ifadesini ekle",
|
||||||
|
"instanceName": "Instance adı",
|
||||||
|
"language": "Dil",
|
||||||
|
"lockPassword": "Kullanıcının parolayı değiştirmesini engelle",
|
||||||
|
"newPassword": "Yeni şifre",
|
||||||
|
"newPasswordConfirm": "Yeni şifre tekrarı",
|
||||||
|
"newUser": "Yeni Kullanıcı",
|
||||||
|
"password": "Şifre",
|
||||||
|
"passwordUpdated": "Şifre güncellendi",
|
||||||
|
"path": "Yol",
|
||||||
|
"perm": {
|
||||||
|
"create": "Dosyalar ve dizinler oluşturun",
|
||||||
|
"delete": "Dosyalar ve dizinleri silin",
|
||||||
|
"download": "İndir",
|
||||||
|
"execute": "Komutları çalıştır",
|
||||||
|
"modify": "Dosyaları değiştir",
|
||||||
|
"rename": "Dosyaları ve dizinleri yeniden adlandırın veya taşıyın",
|
||||||
|
"share": "Dosyaları paylaş"
|
||||||
|
},
|
||||||
|
"permissions": "İzinler",
|
||||||
|
"permissionsHelp": "Kullanıcıyı yönetici olarak ayarlayabilir veya izinleri ayrı ayrı seçebilirsiniz. \"Yönetici\"yi seçerseniz, diğer tüm seçenekler otomatik olarak kontrol edilecektir. Kullanıcıların yönetimi, bir yöneticinin yetkisi olarak kalır.\n",
|
||||||
|
"profileSettings": "Profil ayarları",
|
||||||
|
"ruleExample1": "her klasördeki herhangi bir noktalı dosyaya (.git, .gitignore gibi) erişimi engeller.\n",
|
||||||
|
"ruleExample2": "Root erişimidenki CaddyFile dosyalarına erişimi engelle.",
|
||||||
|
"rules": "Kurallar",
|
||||||
|
"rulesHelp": "Burada, bu belirli kullanıcı için bir dizi izin verme ve izin vermeme kuralı tanımlayabilirsiniz. Engellenen dosyalar listelerde görünmeyecek ve kullanıcı bunlara erişemeyecek. Kullanıcı erişimine göre regex ifadeleri destekliyoruz.\n",
|
||||||
|
"scope": "Kapsam",
|
||||||
|
"settingsUpdated": "Ayarlar güncellendi!",
|
||||||
|
"shareDuration": "Paylaşım süresi",
|
||||||
|
"shareManagement": "Paylaşım yönetimi",
|
||||||
|
"shareDeleted": "Paylaşım silindi!",
|
||||||
|
"singleClick": "Dosyaları ve dizinleri açmak için tek tıklamayı kullanın",
|
||||||
|
"themes": {
|
||||||
|
"dark": "Dark",
|
||||||
|
"light": "Light",
|
||||||
|
"title": "Theme"
|
||||||
|
},
|
||||||
|
"user": "Kullanıcı",
|
||||||
|
"userCommands": "Komutları",
|
||||||
|
"userCommandsHelp": "Bu kullanıcı için mevcut komutları içeren boşlukla ayrılmış bir liste. Örnek:\n",
|
||||||
|
"userCreated": "Kullanıcı oluşturuldu!",
|
||||||
|
"userDefaults": "Kullanıcı varsayılan ayarları",
|
||||||
|
"userDeleted": "Kullanıcı silindi!",
|
||||||
|
"userManagement": "Kullanıcı yönetimi",
|
||||||
|
"userUpdated": "Kullanıcı güncellendi!",
|
||||||
|
"username": "Kullanıcı adı",
|
||||||
|
"users": "Kullanıcılar"
|
||||||
|
},
|
||||||
|
"sidebar": {
|
||||||
|
"help": "Yardım",
|
||||||
|
"hugoNew": "Yeni Hugo",
|
||||||
|
"login": "Giriş",
|
||||||
|
"logout": "Çıkış",
|
||||||
|
"myFiles": "Dosyalarım",
|
||||||
|
"newFile": "Yeni dosya",
|
||||||
|
"newFolder": "Yeni klasör",
|
||||||
|
"preview": "Önizleme",
|
||||||
|
"settings": "Ayarlar",
|
||||||
|
"signup": "Kayıt",
|
||||||
|
"siteSettings": "Site ayarları!"
|
||||||
|
},
|
||||||
|
"success": {
|
||||||
|
"linkCopied": "Link kopyalandı!"
|
||||||
|
},
|
||||||
|
"time": {
|
||||||
|
"days": "Gün",
|
||||||
|
"hours": "Saat",
|
||||||
|
"minutes": "Dakika",
|
||||||
|
"seconds": "Saniye",
|
||||||
|
"unit": "Zaman birimi"
|
||||||
|
}
|
||||||
|
}
|
||||||
267
frontend/src/i18n/ua.json
Normal file
267
frontend/src/i18n/ua.json
Normal file
@@ -0,0 +1,267 @@
|
|||||||
|
{
|
||||||
|
"buttons": {
|
||||||
|
"cancel": "Відмінити",
|
||||||
|
"close": "Закрити",
|
||||||
|
"copy": "Копіювати",
|
||||||
|
"copyFile": "Копіювати файл",
|
||||||
|
"copyToClipboard": "Копіювати в буфер обміну",
|
||||||
|
"create": "Створити",
|
||||||
|
"delete": "Видалити",
|
||||||
|
"download": "Завантажити",
|
||||||
|
"file": "Файл",
|
||||||
|
"folder": "Папка",
|
||||||
|
"hideDotfiles": "Приховати точкові файли",
|
||||||
|
"info": "Інфо",
|
||||||
|
"more": "Більше",
|
||||||
|
"move": "Перемістити",
|
||||||
|
"moveFile": "Перемістити файл",
|
||||||
|
"new": "Новий",
|
||||||
|
"next": "Далі",
|
||||||
|
"ok": "ОК",
|
||||||
|
"permalink": "Отримати постійне посилання",
|
||||||
|
"previous": "Назад",
|
||||||
|
"publish": "Опублікувати",
|
||||||
|
"rename": "Перейменувати",
|
||||||
|
"replace": "Замінити",
|
||||||
|
"reportIssue": "Повідомити про помилку",
|
||||||
|
"save": "Зберегти",
|
||||||
|
"schedule": "Планування",
|
||||||
|
"search": "Пошук",
|
||||||
|
"select": "Вибрати",
|
||||||
|
"selectMultiple": "Мультивибір",
|
||||||
|
"share": "Поділитися",
|
||||||
|
"shell": "Командний рядок",
|
||||||
|
"submit": "Відправити",
|
||||||
|
"switchView": "Вид",
|
||||||
|
"toggleSidebar": "Бічна панель",
|
||||||
|
"update": "Оновити",
|
||||||
|
"upload": "Завантажити",
|
||||||
|
"openFile": "Відкрити файл"
|
||||||
|
},
|
||||||
|
"download": {
|
||||||
|
"downloadFile": "Завантажити файл",
|
||||||
|
"downloadFolder": "Завантажити папку",
|
||||||
|
"downloadSelected": "Завантажити вибране"
|
||||||
|
},
|
||||||
|
"errors": {
|
||||||
|
"forbidden": "У вас немає прав доступу до цього.",
|
||||||
|
"internal": "Щось пішло не так.",
|
||||||
|
"notFound": "Неправильне посилання.",
|
||||||
|
"connection": "Немає підключення до сервера."
|
||||||
|
},
|
||||||
|
"files": {
|
||||||
|
"body": "Тіло",
|
||||||
|
"clear": "Очистити",
|
||||||
|
"closePreview": "Закрити",
|
||||||
|
"files": "Файли",
|
||||||
|
"folders": "Папки",
|
||||||
|
"home": "Домівка",
|
||||||
|
"lastModified": "Останній раз змінено",
|
||||||
|
"loading": "Завантаження...",
|
||||||
|
"lonely": "Тут пусто...",
|
||||||
|
"metadata": "Метадані",
|
||||||
|
"multipleSelectionEnabled": "Мультивибір включений",
|
||||||
|
"name": "Ім'я",
|
||||||
|
"size": "Розмір",
|
||||||
|
"sortByLastModified": "Сортувати за останнім зміненням",
|
||||||
|
"sortByName": "Сортувати за іменем",
|
||||||
|
"sortBySize": "Сортувати за розміром",
|
||||||
|
"noPreview": "Попередній перегляд для цього файлу недоступний."
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"click": "вибрати файл чи каталог",
|
||||||
|
"ctrl": {
|
||||||
|
"click": "вибрати кілька файлів чи каталогів",
|
||||||
|
"f": "відкрити пошук",
|
||||||
|
"s": "скачати файл або поточний каталог"
|
||||||
|
},
|
||||||
|
"del": "видалити вибрані елементи",
|
||||||
|
"doubleClick": "відкрити файл чи каталог",
|
||||||
|
"esc": "очистити виділення та/або закрити вікно",
|
||||||
|
"f1": "допомога",
|
||||||
|
"f2": "перейменувати файл",
|
||||||
|
"help": "Допомога"
|
||||||
|
},
|
||||||
|
"languages": {
|
||||||
|
"ar": "العربية",
|
||||||
|
"de": "Deutsch",
|
||||||
|
"en": "English",
|
||||||
|
"es": "Español",
|
||||||
|
"fr": "Français",
|
||||||
|
"is": "Icelandic",
|
||||||
|
"it": "Italiano",
|
||||||
|
"ja": "日本語",
|
||||||
|
"ko": "한국어",
|
||||||
|
"nlBE": "Dutch (Belgium)",
|
||||||
|
"pl": "Polski",
|
||||||
|
"pt": "Português",
|
||||||
|
"ptBR": "Português (Brasil)",
|
||||||
|
"ro": "Romanian",
|
||||||
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
|
"svSE": "Swedish (Sweden)",
|
||||||
|
"tr": "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
|
"zhCN": "中文 (简体)",
|
||||||
|
"zhTW": "中文 (繁體)"
|
||||||
|
},
|
||||||
|
"login": {
|
||||||
|
"createAnAccount": "Створити обліковий запис",
|
||||||
|
"loginInstead": "Вже є обліковий запис",
|
||||||
|
"password": "Пароль",
|
||||||
|
"passwordConfirm": "Підтвердження паролю",
|
||||||
|
"passwordsDontMatch": "Паролі не співпадають",
|
||||||
|
"signup": "Зареєструватися",
|
||||||
|
"submit": "Увійти",
|
||||||
|
"username": "Ім'я користувача",
|
||||||
|
"usernameTaken": "Ім'я користувача вже використовується",
|
||||||
|
"wrongCredentials": "Невірне ім'я користувача або пароль"
|
||||||
|
},
|
||||||
|
"permanent": "Постійний",
|
||||||
|
"prompts": {
|
||||||
|
"copy": "Копіювати",
|
||||||
|
"copyMessage": "Копіювати в:",
|
||||||
|
"currentlyNavigating": "Поточний каталог:",
|
||||||
|
"deleteMessageMultiple": "Видалити ці файли ({count})?",
|
||||||
|
"deleteMessageSingle": "Видалити цей файл/каталог?",
|
||||||
|
"deleteMessageShare": "Видалити цей спільний файл/каталог ({path})?",
|
||||||
|
"deleteTitle": "Видалити файлы",
|
||||||
|
"displayName": "Відображене ім'я:",
|
||||||
|
"download": "Завантажити файлы",
|
||||||
|
"downloadMessage": "Виберіть формат, в якому хочете завантажити.",
|
||||||
|
"error": "Помилка",
|
||||||
|
"fileInfo": "Інформація про файл",
|
||||||
|
"filesSelected": "Файлів вибрано: {count}.",
|
||||||
|
"lastModified": "Останній раз змінено",
|
||||||
|
"move": "Перемістити",
|
||||||
|
"moveMessage": "Перемістити в:",
|
||||||
|
"newArchetype": "Створіть новий запис на основі архетипу. Файл буде створено у каталозі.",
|
||||||
|
"newDir": "Новий каталог",
|
||||||
|
"newDirMessage": "Ім'я нового каталогу.",
|
||||||
|
"newFile": "Новий файл",
|
||||||
|
"newFileMessage": "Ім'я нового файлу.",
|
||||||
|
"numberDirs": "Кількість каталогів",
|
||||||
|
"numberFiles": "Кількість файлів",
|
||||||
|
"rename": "Перейменувати",
|
||||||
|
"renameMessage": "Нове ім'я",
|
||||||
|
"replace": "Замінити",
|
||||||
|
"replaceMessage": "Ім'я одного з файлів, що завантажуються, збігається з вже існуючим файлом. Ви бажаєте замінити існуючий?\n",
|
||||||
|
"schedule": "Планування",
|
||||||
|
"scheduleMessage": "Запланувати дату та час публікації.",
|
||||||
|
"show": "Показати",
|
||||||
|
"size": "Розмір",
|
||||||
|
"upload": "Завантажити",
|
||||||
|
"uploadMessage": "Виберіть варіант для завантаження.",
|
||||||
|
"optionalPassword": "Необов'язковий пароль"
|
||||||
|
},
|
||||||
|
"search": {
|
||||||
|
"images": "Зображення",
|
||||||
|
"music": "Музика",
|
||||||
|
"pdf": "PDF",
|
||||||
|
"pressToSearch": "Натисніть ENTER для пошуку",
|
||||||
|
"search": "Пошук...",
|
||||||
|
"typeToSearch": "Введіть ім'я файлу...",
|
||||||
|
"types": "Типи",
|
||||||
|
"video": "Відео"
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"admin": "Адмін",
|
||||||
|
"administrator": "Адміністратор",
|
||||||
|
"allowCommands": "Запуск команд",
|
||||||
|
"allowEdit": "Редагування, перейменування та видалення файлів чи каталогів",
|
||||||
|
"allowNew": "Створення нових файлів або каталогів",
|
||||||
|
"allowPublish": "Публікація нових записів та сторінок",
|
||||||
|
"allowSignup": "Дозволити користувачам реєструватися",
|
||||||
|
"avoidChanges": "(залишіть поле порожнім, щоб уникнути змін)",
|
||||||
|
"branding": "Брендинг",
|
||||||
|
"brandingDirectoryPath": "Шлях до каталогу брендів",
|
||||||
|
"brandingHelp": "Ви можете налаштувати зовнішній вигляд файлового браузера, змінивши його ім'я, замінивши логотип, додавши власні стилі та навіть відключивши зовнішні посилання на GitHub.\nДодаткову інформацію про персоналізований брендинг можна знайти на сторінці {0}.",
|
||||||
|
"changePassword": "Зміна пароля",
|
||||||
|
"commandRunner": "Запуск команд",
|
||||||
|
"commandRunnerHelp": "Тут ви можете встановити команди, які будуть виконуватися у зазначених подіях. Ви повинні вказати по одній команді в кожному рядку. Змінні середовища {0} та {1} будуть доступні, будучи {0} щодо {1}. Додаткові відомості про цю функцію та доступні змінні середовища див. у {2}.",
|
||||||
|
"commandsUpdated": "Команди оновлені!",
|
||||||
|
"createUserDir": "Автоматичне створення домашнього каталогу користувача при додаванні нового користувача",
|
||||||
|
"customStylesheet": "Свій стиль",
|
||||||
|
"defaultUserDescription": "Це налаштування за замовчуванням для нових користувачів.",
|
||||||
|
"disableExternalLinks": "Вимкнути зовнішні посилання (крім документації)",
|
||||||
|
"documentation": "документація",
|
||||||
|
"examples": "Приклади",
|
||||||
|
"executeOnShell": "Виконати в командному рядку",
|
||||||
|
"executeOnShellDescription": "За замовчуванням File Browser виконує команди, безпосередньо викликаючи їх бінарні файли. Якщо ви хочете замість цього запускати їх в оболонці (наприклад, Bash або PowerShell), ви можете визначити їх тут з необхідними аргументами та прапорами. Якщо встановлено, виконуєма вами команда буде додана як аргумент. Це стосується як користувацьких команд, так і обробників подій.",
|
||||||
|
"globalRules": "Це глобальний набір дозволяючих та забороняючих правил. Вони застосовні до кожного користувача. Ви можете визначити певні правила для налаштувань кожного користувача, щоб перевизначити їх.",
|
||||||
|
"globalSettings": "Глобальні налаштування",
|
||||||
|
"hideDotfiles": "Приховати точкові файли",
|
||||||
|
"insertPath": "Вставте шлях",
|
||||||
|
"insertRegex": "Вставити регулярний вираз",
|
||||||
|
"instanceName": "Поточна назва програми",
|
||||||
|
"language": "Мова",
|
||||||
|
"lockPassword": "Заборонити користувачеві змінювати пароль",
|
||||||
|
"newPassword": "Новий пароль",
|
||||||
|
"newPasswordConfirm": "Підтвердження нового пароля",
|
||||||
|
"newUser": "Новий користувач",
|
||||||
|
"password": "Пароль",
|
||||||
|
"passwordUpdated": "Пароль оновлено!",
|
||||||
|
"path": "Шлях",
|
||||||
|
"perm": {
|
||||||
|
"create": "Створювати файли та каталоги",
|
||||||
|
"delete": "Видаляти файли та каталоги",
|
||||||
|
"download": "Завантажувати",
|
||||||
|
"execute": "Виконувати команди",
|
||||||
|
"modify": "Редагувати файли",
|
||||||
|
"rename": "Перейменовувати або переміщувати файли та каталоги",
|
||||||
|
"share": "Ділітися файлами"
|
||||||
|
},
|
||||||
|
"permissions": "Дозволи",
|
||||||
|
"permissionsHelp": "Можна настроїти користувача як адміністратора або вибрати індивідуальні дозволи. При виборі \"Адміністратор\" всі інші параметри будуть автоматично вибрані. Керування користувачами - привілей адміністратора.\n",
|
||||||
|
"profileSettings": "Налаштування профілю",
|
||||||
|
"ruleExample1": "запобігти доступу до будь-якого прихованого файлу (наприклад: .git, .gitignore) у кожній папці.\n",
|
||||||
|
"ruleExample2": "блокує доступ до файлу з ім'ям Caddyfile у кореневій області.",
|
||||||
|
"rules": "Права",
|
||||||
|
"rulesHelp": "Тут ви можете визначити набір дозволяючих та забороняючих правил для цього конкретного користувача. Блоковані файли не відображатимуться у списках, і не будуть доступні для користувача. Є підтримка регулярних виразів та відносних шляхів.\n",
|
||||||
|
"scope": "Корінь",
|
||||||
|
"setDateFormat": "Встановити точний формат дати",
|
||||||
|
"settingsUpdated": "Налаштування застосовані!",
|
||||||
|
"shareDuration": "Тривалість спільного посилання",
|
||||||
|
"shareManagement": "Управління спільними посиланнями",
|
||||||
|
"shareDeleted": "Спільне посилання видалено!",
|
||||||
|
"singleClick": "Відкриття файлів та каталогів одним кліком",
|
||||||
|
"themes": {
|
||||||
|
"dark": "Темна",
|
||||||
|
"light": "Світла",
|
||||||
|
"title": "Тема"
|
||||||
|
},
|
||||||
|
"user": "Користувач",
|
||||||
|
"userCommands": "Команди",
|
||||||
|
"userCommandsHelp": "Список команд, доступних користувачу, розділений пробілами. Приклад:\n",
|
||||||
|
"userCreated": "Користувач створений!",
|
||||||
|
"userDefaults": "Налаштування користувача за замовчуванням",
|
||||||
|
"userDeleted": "Користувач видалений!",
|
||||||
|
"userManagement": "Керування користувачами",
|
||||||
|
"userUpdated": "Користувач змінений!",
|
||||||
|
"username": "Ім'я користувача",
|
||||||
|
"users": "Користувачі"
|
||||||
|
},
|
||||||
|
"sidebar": {
|
||||||
|
"help": "Допомога",
|
||||||
|
"hugoNew": "Hugo New",
|
||||||
|
"login": "Увійти",
|
||||||
|
"logout": "Вийти",
|
||||||
|
"myFiles": "Файли",
|
||||||
|
"newFile": "Новий файл",
|
||||||
|
"newFolder": "Новий каталог",
|
||||||
|
"preview": "Перегляд",
|
||||||
|
"settings": "Налаштування",
|
||||||
|
"signup": "Зареєструватися",
|
||||||
|
"siteSettings": "Налаштування сайту"
|
||||||
|
},
|
||||||
|
"success": {
|
||||||
|
"linkCopied": "Посилання скопійоване!"
|
||||||
|
},
|
||||||
|
"time": {
|
||||||
|
"days": "Дні",
|
||||||
|
"hours": "Години",
|
||||||
|
"minutes": "Хвилини",
|
||||||
|
"seconds": "Секунди",
|
||||||
|
"unit": "Одиниця часу"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,12 +8,14 @@
|
|||||||
"create": "创建",
|
"create": "创建",
|
||||||
"delete": "删除",
|
"delete": "删除",
|
||||||
"download": "下载",
|
"download": "下载",
|
||||||
"hideDotfiles": "不显示隐藏的文件",
|
"file": "文件",
|
||||||
|
"folder": "文件夹",
|
||||||
|
"hideDotfiles": "不显示隐藏文件",
|
||||||
"info": "信息",
|
"info": "信息",
|
||||||
"more": "更多",
|
"more": "更多",
|
||||||
"move": "移动",
|
"move": "移动",
|
||||||
"moveFile": "移动文件",
|
"moveFile": "移动文件",
|
||||||
"new": "新",
|
"new": "新建",
|
||||||
"next": "下一个",
|
"next": "下一个",
|
||||||
"ok": "确定",
|
"ok": "确定",
|
||||||
"permalink": "获取永久链接",
|
"permalink": "获取永久链接",
|
||||||
@@ -28,12 +30,13 @@
|
|||||||
"select": "选择",
|
"select": "选择",
|
||||||
"selectMultiple": "选择多个",
|
"selectMultiple": "选择多个",
|
||||||
"share": "分享",
|
"share": "分享",
|
||||||
"shell": "激活 shell",
|
"shell": "激活 Shell",
|
||||||
"submit": "提交",
|
"submit": "提交",
|
||||||
"switchView": "切换显示方式",
|
"switchView": "切换显示方式",
|
||||||
"toggleSidebar": "切换侧边栏",
|
"toggleSidebar": "切换侧边栏",
|
||||||
"update": "更新",
|
"update": "更新",
|
||||||
"upload": "上传"
|
"upload": "上传",
|
||||||
|
"openFile": "打开文件"
|
||||||
},
|
},
|
||||||
"download": {
|
"download": {
|
||||||
"downloadFile": "下载文件",
|
"downloadFile": "下载文件",
|
||||||
@@ -43,7 +46,8 @@
|
|||||||
"errors": {
|
"errors": {
|
||||||
"forbidden": "你无权限访问",
|
"forbidden": "你无权限访问",
|
||||||
"internal": "服务器出了点问题。",
|
"internal": "服务器出了点问题。",
|
||||||
"notFound": "找不到文件。"
|
"notFound": "找不到文件。",
|
||||||
|
"connection": "无法连接到服务器。"
|
||||||
},
|
},
|
||||||
"files": {
|
"files": {
|
||||||
"body": "内容",
|
"body": "内容",
|
||||||
@@ -61,7 +65,8 @@
|
|||||||
"size": "大小",
|
"size": "大小",
|
||||||
"sortByLastModified": "按最后修改时间排序",
|
"sortByLastModified": "按最后修改时间排序",
|
||||||
"sortByName": "按名称排序",
|
"sortByName": "按名称排序",
|
||||||
"sortBySize": "按大小排序"
|
"sortBySize": "按大小排序",
|
||||||
|
"noPreview": "此文件无法预览。"
|
||||||
},
|
},
|
||||||
"help": {
|
"help": {
|
||||||
"click": "选择文件或目录",
|
"click": "选择文件或目录",
|
||||||
@@ -93,7 +98,10 @@
|
|||||||
"ptBR": "Português(Brasil)",
|
"ptBR": "Português(Brasil)",
|
||||||
"ro": "Romanian",
|
"ro": "Romanian",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "Swedish(Sweden)",
|
"svSE": "Swedish(Sweden)",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文(简体)",
|
"zhCN": "中文(简体)",
|
||||||
"zhTW": "中文(繁體)"
|
"zhTW": "中文(繁體)"
|
||||||
},
|
},
|
||||||
@@ -116,7 +124,7 @@
|
|||||||
"currentlyNavigating": "当前目录:",
|
"currentlyNavigating": "当前目录:",
|
||||||
"deleteMessageMultiple": "你确定要删除这 {count} 个文件吗?",
|
"deleteMessageMultiple": "你确定要删除这 {count} 个文件吗?",
|
||||||
"deleteMessageSingle": "你确定要删除这个文件/文件夹吗?",
|
"deleteMessageSingle": "你确定要删除这个文件/文件夹吗?",
|
||||||
"deleteMessageShare": "你确定要删除这个分享({path})吗?",
|
"deleteMessageShare": "你确定要删除这个分享({path})吗?",
|
||||||
"deleteTitle": "删除文件",
|
"deleteTitle": "删除文件",
|
||||||
"displayName": "名称:",
|
"displayName": "名称:",
|
||||||
"download": "下载文件",
|
"download": "下载文件",
|
||||||
@@ -137,7 +145,7 @@
|
|||||||
"rename": "重命名",
|
"rename": "重命名",
|
||||||
"renameMessage": "请输入新名称,旧名称为:",
|
"renameMessage": "请输入新名称,旧名称为:",
|
||||||
"replace": "替换",
|
"replace": "替换",
|
||||||
"replaceMessage": "您尝试上传的文件中有一个与现有文件的名称存在冲突。是否替换现有的同名文件?",
|
"replaceMessage": "您尝试上传的文件中有一个与现有文件的名称存在冲突。是否替换现有的同名文件?\n",
|
||||||
"schedule": "计划",
|
"schedule": "计划",
|
||||||
"scheduleMessage": "请选择发布这篇帖子的日期与时间。",
|
"scheduleMessage": "请选择发布这篇帖子的日期与时间。",
|
||||||
"show": "点击以显示",
|
"show": "点击以显示",
|
||||||
@@ -150,16 +158,16 @@
|
|||||||
"images": "图像",
|
"images": "图像",
|
||||||
"music": "音乐",
|
"music": "音乐",
|
||||||
"pdf": "PDF",
|
"pdf": "PDF",
|
||||||
"pressToSearch": "回车搜索...",
|
"pressToSearch": "输入回车以搜索...",
|
||||||
"search": "搜索...",
|
"search": "搜索...",
|
||||||
"typeToSearch": "输入搜索...",
|
"typeToSearch": "输入以搜索...",
|
||||||
"types": "类型",
|
"types": "类型",
|
||||||
"video": "视频"
|
"video": "视频"
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"admin": "管理员",
|
"admin": "管理员",
|
||||||
"administrator": "管理员",
|
"administrator": "管理员",
|
||||||
"allowCommands": "执行命令(shell 命令)",
|
"allowCommands": "执行命令(Shell 命令)",
|
||||||
"allowEdit": "编辑、重命名或删除文件/目录",
|
"allowEdit": "编辑、重命名或删除文件/目录",
|
||||||
"allowNew": "创建新文件和目录",
|
"allowNew": "创建新文件和目录",
|
||||||
"allowPublish": "发布新的帖子与页面",
|
"allowPublish": "发布新的帖子与页面",
|
||||||
@@ -167,10 +175,10 @@
|
|||||||
"avoidChanges": "(留空以避免更改)",
|
"avoidChanges": "(留空以避免更改)",
|
||||||
"branding": "品牌",
|
"branding": "品牌",
|
||||||
"brandingDirectoryPath": "品牌信息文件夹路径",
|
"brandingDirectoryPath": "品牌信息文件夹路径",
|
||||||
"brandingHelp": "您可以通过改变实例名称,更换Logo,加入自定义样式,甚至禁用到Github的外部链接来自定义File Browser的外观和给人的感觉。\n想获得更多信息,请查看 {0} 。",
|
"brandingHelp": "您可以通过改变实例名称,更换 Logo,加入自定义样式,甚至禁用到 Github 的外部链接来自定义 File Browser 的外观和感觉。\n想获得更多信息,请查看 {0}。",
|
||||||
"changePassword": "更改密码",
|
"changePassword": "更改密码",
|
||||||
"commandRunner": "命令执行器",
|
"commandRunner": "命令执行器",
|
||||||
"commandRunnerHelp": "在这里你可以设置在下面的事件中执行的命令。每行必须写一条命令。可以在命令中使用环境变量 {0} 和 {1}。关于此功能和可用环境变量的更多信息,请阅读{2}.",
|
"commandRunnerHelp": "你可以在此设置在下列事件中执行的命令。每行必须写一条命令。可以在命令中使用环境变量 {0} 和 {1},使 {0} 与 {1} 相关联。关于此功能和可用环境变量的更多信息,请阅读 {2}。",
|
||||||
"commandsUpdated": "命令已更新!",
|
"commandsUpdated": "命令已更新!",
|
||||||
"createUserDir": "在添加新用户的同时自动创建用户的个人目录",
|
"createUserDir": "在添加新用户的同时自动创建用户的个人目录",
|
||||||
"customStylesheet": "自定义样式表(CSS)",
|
"customStylesheet": "自定义样式表(CSS)",
|
||||||
@@ -178,18 +186,18 @@
|
|||||||
"disableExternalLinks": "禁止外部链接(帮助文档除外)",
|
"disableExternalLinks": "禁止外部链接(帮助文档除外)",
|
||||||
"documentation": "帮助文档",
|
"documentation": "帮助文档",
|
||||||
"examples": "例子",
|
"examples": "例子",
|
||||||
"executeOnShell": "在Shell中执行",
|
"executeOnShell": "在 Shell 中执行",
|
||||||
"executeOnShellDescription": "默认情况下,File Browser通过直接调用命令的二进制包来执行命令,如果想在shell中执行(如Bash、PowerShell),你可以在这里定义所使用的shell和参数。如果设置了这个选项,所执行的命令会作为参数追加在后面。本选项对用户命令和事件钩子都生效。",
|
"executeOnShellDescription": "默认情况下,File Browser 通过直接调用命令的二进制包来执行命令,如果想在 Shell中 执行(如 Bash 或 PowerShell),你可以在这里定义所使用的 Shell 和参数。设置后,您所执行的命令会作为参数追加。本设置对用户命令和事件钩子都生效。",
|
||||||
"globalRules": "这是全局允许与禁止规则。它们作用于所有用户。您可以给每个用户定义单独的特殊规则来覆盖全局规则。",
|
"globalRules": "这是全局允许与禁止规则。它们作用于所有用户。您可以给每个用户定义单独的特殊规则来覆盖全局规则。",
|
||||||
"globalSettings": "全局设置",
|
"globalSettings": "全局设置",
|
||||||
"hideDotfiles": "",
|
"hideDotfiles": "不显示隐藏文件",
|
||||||
"insertPath": "插入路径",
|
"insertPath": "插入路径",
|
||||||
"insertRegex": "插入正则表达式",
|
"insertRegex": "插入正则表达式",
|
||||||
"instanceName": "实例名称",
|
"instanceName": "实例名称",
|
||||||
"language": "语言",
|
"language": "语言",
|
||||||
"lockPassword": "禁止用户修改密码",
|
"lockPassword": "禁止用户修改密码",
|
||||||
"newPassword": "您的新密码",
|
"newPassword": "您的新密码",
|
||||||
"newPasswordConfirm": "重输一遍新密码",
|
"newPasswordConfirm": "再次输入以确认您的新密码",
|
||||||
"newUser": "新建用户",
|
"newUser": "新建用户",
|
||||||
"password": "密码",
|
"password": "密码",
|
||||||
"passwordUpdated": "密码已更新!",
|
"passwordUpdated": "密码已更新!",
|
||||||
@@ -204,25 +212,27 @@
|
|||||||
"share": "分享文件"
|
"share": "分享文件"
|
||||||
},
|
},
|
||||||
"permissions": "权限",
|
"permissions": "权限",
|
||||||
"permissionsHelp": "您可以将该用户设置为管理员,也可以单独选择各项权限。如果选择了“管理员”,则其他的选项会被自动勾上,同时该用户可以管理其他用户。",
|
"permissionsHelp": "您可以将该用户设置为管理员或单独选择各项权限。如果您选择了“管理员”,则其他的选项会被自动选中,同时该用户可以管理其他用户。\n",
|
||||||
"profileSettings": "个人设置",
|
"profileSettings": "个人设置",
|
||||||
"ruleExample1": "阻止用户访问所有文件夹下任何以 . 开头的文件(隐藏文件, 例如: .git, .gitignore)。",
|
"ruleExample1": "阻止用户访问所有文件夹下任何以 . 开头的文件(隐藏文件, 例如: .git, .gitignore)。\n",
|
||||||
"ruleExample2": "阻止用户访问其目录范围的根目录下名为 Caddyfile 的文件。",
|
"ruleExample2": "阻止用户访问其目录范围的根目录下名为 Caddyfile 的文件。",
|
||||||
"rules": "规则",
|
"rules": "规则",
|
||||||
"rulesHelp": "您可以为该用户制定一组黑名单或白名单式的规则,被屏蔽的文件将不会显示在列表中,用户也无权限访问,支持相对于目录范围的路径。",
|
"rulesHelp": "您可以为该用户制定一组黑名单或白名单式的规则,被屏蔽的文件将不会显示在列表中,用户也无权限访问,支持正则表达式和相对于用户范围的路径。\n",
|
||||||
"scope": "目录范围",
|
"scope": "目录范围",
|
||||||
|
"setDateFormat": "显示精确的日期格式",
|
||||||
"settingsUpdated": "设置已更新!",
|
"settingsUpdated": "设置已更新!",
|
||||||
"shareDuration": "分享期限",
|
"shareDuration": "分享期限",
|
||||||
"shareManagement": "分享管理",
|
"shareManagement": "分享管理",
|
||||||
"singleClick": "",
|
"shareDeleted": "分享已删除!",
|
||||||
|
"singleClick": "使用单击来打开文件和目录",
|
||||||
"themes": {
|
"themes": {
|
||||||
"dark": "深色",
|
"dark": "深色",
|
||||||
"light": "浅色",
|
"light": "浅色",
|
||||||
"title": "主题"
|
"title": "主题"
|
||||||
},
|
},
|
||||||
"user": "用户",
|
"user": "用户",
|
||||||
"userCommands": "用户命令(shell 命令)",
|
"userCommands": "用户命令(Shell 命令)",
|
||||||
"userCommandsHelp": "指定该用户可以执行的命令(shell 代码),用空格分隔。例如:",
|
"userCommandsHelp": "指定该用户可以执行的命令(Shell 命令),用空格分隔。例如:\n",
|
||||||
"userCreated": "用户已创建!",
|
"userCreated": "用户已创建!",
|
||||||
"userDefaults": "用户默认设置",
|
"userDefaults": "用户默认设置",
|
||||||
"userDeleted": "用户已删除!",
|
"userDeleted": "用户已删除!",
|
||||||
@@ -233,7 +243,7 @@
|
|||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"help": "帮助",
|
"help": "帮助",
|
||||||
"hugoNew": "Hugo New",
|
"hugoNew": "Hugo 新建",
|
||||||
"login": "登录",
|
"login": "登录",
|
||||||
"logout": "登出",
|
"logout": "登出",
|
||||||
"myFiles": "我的文件",
|
"myFiles": "我的文件",
|
||||||
|
|||||||
@@ -92,7 +92,10 @@
|
|||||||
"ptBR": "Português (Brasil)",
|
"ptBR": "Português (Brasil)",
|
||||||
"ro": "Romanian",
|
"ro": "Romanian",
|
||||||
"ru": "Русский",
|
"ru": "Русский",
|
||||||
|
"sk": "Slovenčina",
|
||||||
"svSE": "Swedish(Sweden)",
|
"svSE": "Swedish(Sweden)",
|
||||||
|
"tr" : "Türkçe",
|
||||||
|
"ua": "Українська",
|
||||||
"zhCN": "中文 (简体)",
|
"zhCN": "中文 (简体)",
|
||||||
"zhTW": "中文 (繁體)"
|
"zhTW": "中文 (繁體)"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import "whatwg-fetch";
|
||||||
|
import cssVars from "css-vars-ponyfill";
|
||||||
import { sync } from "vuex-router-sync";
|
import { sync } from "vuex-router-sync";
|
||||||
import store from "@/store";
|
import store from "@/store";
|
||||||
import router from "@/router";
|
import router from "@/router";
|
||||||
@@ -7,6 +9,8 @@ import { recaptcha, loginPage } from "@/utils/constants";
|
|||||||
import { login, validateLogin } from "@/utils/auth";
|
import { login, validateLogin } from "@/utils/auth";
|
||||||
import App from "@/App";
|
import App from "@/App";
|
||||||
|
|
||||||
|
cssVars();
|
||||||
|
|
||||||
sync(store, router);
|
sync(store, router);
|
||||||
|
|
||||||
async function start() {
|
async function start() {
|
||||||
|
|||||||
@@ -8,8 +8,36 @@ const getters = {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let totalSize = state.upload.sizes.reduce((a, b) => a + b, 0);
|
||||||
|
|
||||||
let sum = state.upload.progress.reduce((acc, val) => acc + val);
|
let sum = state.upload.progress.reduce((acc, val) => acc + val);
|
||||||
return Math.ceil((sum / state.upload.size) * 100);
|
return Math.ceil((sum / totalSize) * 100);
|
||||||
|
},
|
||||||
|
filesInUploadCount: (state) => {
|
||||||
|
let total =
|
||||||
|
Object.keys(state.upload.uploads).length + state.upload.queue.length;
|
||||||
|
return total;
|
||||||
|
},
|
||||||
|
filesInUpload: (state) => {
|
||||||
|
let files = [];
|
||||||
|
|
||||||
|
for (let index in state.upload.uploads) {
|
||||||
|
let upload = state.upload.uploads[index];
|
||||||
|
let id = upload.id;
|
||||||
|
let type = upload.type;
|
||||||
|
let name = decodeURIComponent(upload.path.replace(/^.*[\\/]/, ""));
|
||||||
|
let progress = state.upload.progress[id];
|
||||||
|
let size = state.upload.sizes[id];
|
||||||
|
|
||||||
|
files.push({
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
progress: Math.ceil((progress / size) * 100),
|
||||||
|
type,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return files.sort((a, b) => a.progress - b.progress);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ const UPLOADS_LIMIT = 5;
|
|||||||
|
|
||||||
const state = {
|
const state = {
|
||||||
id: 0,
|
id: 0,
|
||||||
size: 0,
|
sizes: [],
|
||||||
progress: [],
|
progress: [],
|
||||||
queue: [],
|
queue: [],
|
||||||
uploads: {},
|
uploads: {},
|
||||||
@@ -19,12 +19,12 @@ const mutations = {
|
|||||||
},
|
},
|
||||||
reset: (state) => {
|
reset: (state) => {
|
||||||
state.id = 0;
|
state.id = 0;
|
||||||
state.size = 0;
|
state.sizes = [];
|
||||||
state.progress = [];
|
state.progress = [];
|
||||||
},
|
},
|
||||||
addJob: (state, item) => {
|
addJob: (state, item) => {
|
||||||
state.queue.push(item);
|
state.queue.push(item);
|
||||||
state.size += item.file.size;
|
state.sizes[state.id] = item.file.size;
|
||||||
state.id++;
|
state.id++;
|
||||||
},
|
},
|
||||||
moveJob(state) {
|
moveJob(state) {
|
||||||
@@ -33,6 +33,7 @@ const mutations = {
|
|||||||
Vue.set(state.uploads, item.id, item);
|
Vue.set(state.uploads, item.id, item);
|
||||||
},
|
},
|
||||||
removeJob(state, id) {
|
removeJob(state, id) {
|
||||||
|
Vue.delete(state.uploads, id);
|
||||||
delete state.uploads[id];
|
delete state.uploads[id];
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -12,6 +12,8 @@ export function parseToken(token) {
|
|||||||
|
|
||||||
const data = JSON.parse(Base64.decode(parts[1]));
|
const data = JSON.parse(Base64.decode(parts[1]));
|
||||||
|
|
||||||
|
document.cookie = `auth=${token}; path=/`;
|
||||||
|
|
||||||
localStorage.setItem("jwt", token);
|
localStorage.setItem("jwt", token);
|
||||||
store.commit("setJWT", token);
|
store.commit("setJWT", token);
|
||||||
store.commit("setUser", data.user);
|
store.commit("setUser", data.user);
|
||||||
@@ -81,6 +83,8 @@ export async function signup(username, password) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function logout() {
|
export function logout() {
|
||||||
|
document.cookie = "auth=; expires=Thu, 01 Jan 1970 00:00:01 GMT; path=/";
|
||||||
|
|
||||||
store.commit("setJWT", "");
|
store.commit("setJWT", "");
|
||||||
store.commit("setUser", null);
|
store.commit("setUser", null);
|
||||||
localStorage.setItem("jwt", null);
|
localStorage.setItem("jwt", null);
|
||||||
|
|||||||
@@ -99,6 +99,15 @@ export function scanFiles(dt) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function detectType(mimetype) {
|
||||||
|
if (mimetype.startsWith("video")) return "video";
|
||||||
|
if (mimetype.startsWith("audio")) return "audio";
|
||||||
|
if (mimetype.startsWith("image")) return "image";
|
||||||
|
if (mimetype.startsWith("pdf")) return "pdf";
|
||||||
|
if (mimetype.startsWith("text")) return "text";
|
||||||
|
return "blob";
|
||||||
|
}
|
||||||
|
|
||||||
export function handleFiles(files, base, overwrite = false) {
|
export function handleFiles(files, base, overwrite = false) {
|
||||||
for (let i = 0; i < files.length; i++) {
|
for (let i = 0; i < files.length; i++) {
|
||||||
let id = store.state.upload.id;
|
let id = store.state.upload.id;
|
||||||
@@ -120,6 +129,7 @@ export function handleFiles(files, base, overwrite = false) {
|
|||||||
path,
|
path,
|
||||||
file,
|
file,
|
||||||
overwrite,
|
overwrite,
|
||||||
|
type: detectType(file.type),
|
||||||
};
|
};
|
||||||
|
|
||||||
store.dispatch("upload/upload", item);
|
store.dispatch("upload/upload", item);
|
||||||
|
|||||||
@@ -3,8 +3,8 @@
|
|||||||
<header-bar v-if="showHeader" showMenu showLogo />
|
<header-bar v-if="showHeader" showMenu showLogo />
|
||||||
|
|
||||||
<h2 class="message">
|
<h2 class="message">
|
||||||
<i class="material-icons">{{ icon }}</i>
|
<i class="material-icons">{{ info.icon }}</i>
|
||||||
<span>{{ message }}</span>
|
<span>{{ $t(info.message) }}</span>
|
||||||
</h2>
|
</h2>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -13,6 +13,10 @@
|
|||||||
import HeaderBar from "@/components/header/HeaderBar";
|
import HeaderBar from "@/components/header/HeaderBar";
|
||||||
|
|
||||||
const errors = {
|
const errors = {
|
||||||
|
0: {
|
||||||
|
icon: "cloud_off",
|
||||||
|
message: "errors.connection",
|
||||||
|
},
|
||||||
403: {
|
403: {
|
||||||
icon: "error",
|
icon: "error",
|
||||||
message: "errors.forbidden",
|
message: "errors.forbidden",
|
||||||
@@ -33,11 +37,17 @@ export default {
|
|||||||
HeaderBar,
|
HeaderBar,
|
||||||
},
|
},
|
||||||
props: ["errorCode", "showHeader"],
|
props: ["errorCode", "showHeader"],
|
||||||
data: function () {
|
computed: {
|
||||||
return {
|
code() {
|
||||||
icon: errors[this.errorCode].icon,
|
return this.errorCode === "0" ||
|
||||||
message: this.$t(errors[this.errorCode].message),
|
this.errorCode === "404" ||
|
||||||
};
|
this.errorCode === "403"
|
||||||
|
? parseInt(this.errorCode)
|
||||||
|
: 500;
|
||||||
|
},
|
||||||
|
info() {
|
||||||
|
return errors[this.code];
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,13 +1,18 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<header-bar v-if="error || !req.type" showMenu showLogo />
|
<header-bar v-if="error || req.type == null" showMenu showLogo />
|
||||||
|
|
||||||
<breadcrumbs base="/files" />
|
<breadcrumbs base="/files" />
|
||||||
|
|
||||||
<errors v-if="error" :errorCode="errorCode" />
|
<errors v-if="error" :errorCode="error.message" />
|
||||||
<component v-else-if="currentView" :is="currentView"></component>
|
<component v-else-if="currentView" :is="currentView"></component>
|
||||||
<div v-else>
|
<div v-else>
|
||||||
<h2 class="message">
|
<h2 class="message delayed">
|
||||||
|
<div class="spinner">
|
||||||
|
<div class="bounce1"></div>
|
||||||
|
<div class="bounce2"></div>
|
||||||
|
<div class="bounce3"></div>
|
||||||
|
</div>
|
||||||
<span>{{ $t("files.loading") }}</span>
|
<span>{{ $t("files.loading") }}</span>
|
||||||
</h2>
|
</h2>
|
||||||
</div>
|
</div>
|
||||||
@@ -62,11 +67,6 @@ export default {
|
|||||||
return "preview";
|
return "preview";
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
errorCode() {
|
|
||||||
return this.error.message === "404" || this.error.message === "403"
|
|
||||||
? parseInt(this.error.message)
|
|
||||||
: 500;
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.fetchData();
|
this.fetchData();
|
||||||
@@ -116,7 +116,7 @@ export default {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.$store.commit("updateRequest", res);
|
this.$store.commit("updateRequest", res);
|
||||||
document.title = res.name;
|
document.title = `${res.name} - ${this.$route.name}`;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.error = e;
|
this.error = e;
|
||||||
} finally {
|
} finally {
|
||||||
@@ -124,15 +124,6 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
keyEvent(event) {
|
keyEvent(event) {
|
||||||
if (this.show !== null) {
|
|
||||||
// Esc!
|
|
||||||
if (event.keyCode === 27) {
|
|
||||||
this.$store.commit("closeHovers");
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// F1!
|
// F1!
|
||||||
if (event.keyCode === 112) {
|
if (event.keyCode === 112) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div id="progress">
|
<div v-if="progress" class="progress">
|
||||||
<div v-bind:style="{ width: this.progress + '%' }"></div>
|
<div v-bind:style="{ width: this.progress + '%' }"></div>
|
||||||
</div>
|
</div>
|
||||||
<sidebar></sidebar>
|
<sidebar></sidebar>
|
||||||
@@ -9,6 +9,7 @@
|
|||||||
<shell v-if="isExecEnabled && isLogged && user.perm.execute" />
|
<shell v-if="isExecEnabled && isLogged && user.perm.execute" />
|
||||||
</main>
|
</main>
|
||||||
<prompts></prompts>
|
<prompts></prompts>
|
||||||
|
<upload-files></upload-files>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -17,6 +18,7 @@ import { mapState, mapGetters } from "vuex";
|
|||||||
import Sidebar from "@/components/Sidebar";
|
import Sidebar from "@/components/Sidebar";
|
||||||
import Prompts from "@/components/prompts/Prompts";
|
import Prompts from "@/components/prompts/Prompts";
|
||||||
import Shell from "@/components/Shell";
|
import Shell from "@/components/Shell";
|
||||||
|
import UploadFiles from "../components/prompts/UploadFiles";
|
||||||
import { enableExec } from "@/utils/constants";
|
import { enableExec } from "@/utils/constants";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
@@ -25,6 +27,7 @@ export default {
|
|||||||
Sidebar,
|
Sidebar,
|
||||||
Prompts,
|
Prompts,
|
||||||
Shell,
|
Shell,
|
||||||
|
UploadFiles,
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapGetters(["isLogged", "progress"]),
|
...mapGetters(["isLogged", "progress"]),
|
||||||
|
|||||||
@@ -10,26 +10,22 @@
|
|||||||
{{ $t("settings.profileSettings") }}
|
{{ $t("settings.profileSettings") }}
|
||||||
</li></router-link
|
</li></router-link
|
||||||
>
|
>
|
||||||
<router-link to="/settings/shares"
|
<router-link to="/settings/shares" v-if="user.perm.share"
|
||||||
><li :class="{ active: $route.path === '/settings/shares' }">
|
><li :class="{ active: $route.path === '/settings/shares' }">
|
||||||
{{ $t("settings.shareManagement") }}
|
{{ $t("settings.shareManagement") }}
|
||||||
</li></router-link
|
</li></router-link
|
||||||
>
|
>
|
||||||
<router-link to="/settings/global"
|
<router-link to="/settings/global" v-if="user.perm.admin"
|
||||||
><li
|
><li :class="{ active: $route.path === '/settings/global' }">
|
||||||
:class="{ active: $route.path === '/settings/global' }"
|
|
||||||
v-if="user.perm.admin"
|
|
||||||
>
|
|
||||||
{{ $t("settings.globalSettings") }}
|
{{ $t("settings.globalSettings") }}
|
||||||
</li></router-link
|
</li></router-link
|
||||||
>
|
>
|
||||||
<router-link to="/settings/users"
|
<router-link to="/settings/users" v-if="user.perm.admin"
|
||||||
><li
|
><li
|
||||||
:class="{
|
:class="{
|
||||||
active:
|
active:
|
||||||
$route.path === '/settings/users' || $route.name === 'User',
|
$route.path === '/settings/users' || $route.name === 'User',
|
||||||
}"
|
}"
|
||||||
v-if="user.perm.admin"
|
|
||||||
>
|
>
|
||||||
{{ $t("settings.userManagement") }}
|
{{ $t("settings.userManagement") }}
|
||||||
</li></router-link
|
</li></router-link
|
||||||
@@ -38,6 +34,17 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div v-if="loading">
|
||||||
|
<h2 class="message delayed">
|
||||||
|
<div class="spinner">
|
||||||
|
<div class="bounce1"></div>
|
||||||
|
<div class="bounce2"></div>
|
||||||
|
<div class="bounce3"></div>
|
||||||
|
</div>
|
||||||
|
<span>{{ $t("files.loading") }}</span>
|
||||||
|
</h2>
|
||||||
|
</div>
|
||||||
|
|
||||||
<router-view></router-view>
|
<router-view></router-view>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -53,7 +60,7 @@ export default {
|
|||||||
HeaderBar,
|
HeaderBar,
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(["user"]),
|
...mapState(["user", "loading"]),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -19,7 +19,50 @@
|
|||||||
|
|
||||||
<breadcrumbs :base="'/share/' + hash" />
|
<breadcrumbs :base="'/share/' + hash" />
|
||||||
|
|
||||||
<div v-if="!loading">
|
<div v-if="loading">
|
||||||
|
<h2 class="message delayed">
|
||||||
|
<div class="spinner">
|
||||||
|
<div class="bounce1"></div>
|
||||||
|
<div class="bounce2"></div>
|
||||||
|
<div class="bounce3"></div>
|
||||||
|
</div>
|
||||||
|
<span>{{ $t("files.loading") }}</span>
|
||||||
|
</h2>
|
||||||
|
</div>
|
||||||
|
<div v-else-if="error">
|
||||||
|
<div v-if="error.message === '401'">
|
||||||
|
<div class="card floating" id="password">
|
||||||
|
<div v-if="attemptedPasswordLogin" class="share__wrong__password">
|
||||||
|
{{ $t("login.wrongCredentials") }}
|
||||||
|
</div>
|
||||||
|
<div class="card-title">
|
||||||
|
<h2>{{ $t("login.password") }}</h2>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-content">
|
||||||
|
<input
|
||||||
|
v-focus
|
||||||
|
type="password"
|
||||||
|
:placeholder="$t('login.password')"
|
||||||
|
v-model="password"
|
||||||
|
@keyup.enter="fetchData"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="card-action">
|
||||||
|
<button
|
||||||
|
class="button button--flat"
|
||||||
|
@click="fetchData"
|
||||||
|
:aria-label="$t('buttons.submit')"
|
||||||
|
:title="$t('buttons.submit')"
|
||||||
|
>
|
||||||
|
{{ $t("buttons.submit") }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<errors v-else :errorCode="error.message" />
|
||||||
|
</div>
|
||||||
|
<div v-else>
|
||||||
<div class="share">
|
<div class="share">
|
||||||
<div class="share__box share__box__info">
|
<div class="share__box share__box__info">
|
||||||
<div class="share__box__header">
|
<div class="share__box__header">
|
||||||
@@ -35,16 +78,30 @@
|
|||||||
<div class="share__box__element">
|
<div class="share__box__element">
|
||||||
<strong>{{ $t("prompts.displayName") }}</strong> {{ req.name }}
|
<strong>{{ $t("prompts.displayName") }}</strong> {{ req.name }}
|
||||||
</div>
|
</div>
|
||||||
<div class="share__box__element">
|
<div class="share__box__element" :title="modTime">
|
||||||
<strong>{{ $t("prompts.lastModified") }}:</strong> {{ humanTime }}
|
<strong>{{ $t("prompts.lastModified") }}:</strong> {{ humanTime }}
|
||||||
</div>
|
</div>
|
||||||
<div class="share__box__element">
|
<div class="share__box__element">
|
||||||
<strong>{{ $t("prompts.size") }}:</strong> {{ humanSize }}
|
<strong>{{ $t("prompts.size") }}:</strong> {{ humanSize }}
|
||||||
</div>
|
</div>
|
||||||
<div class="share__box__element share__box__center">
|
<div class="share__box__element share__box__center">
|
||||||
<a target="_blank" :href="link" class="button button--flat">{{
|
<a target="_blank" :href="link" class="button button--flat">
|
||||||
$t("buttons.download")
|
<div>
|
||||||
}}</a>
|
<i class="material-icons">file_download</i
|
||||||
|
>{{ $t("buttons.download") }}
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
target="_blank"
|
||||||
|
:href="inlineLink"
|
||||||
|
class="button button--flat"
|
||||||
|
v-if="!req.isDir"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<i class="material-icons">open_in_new</i
|
||||||
|
>{{ $t("buttons.openFile") }}
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="share__box__element share__box__center">
|
<div class="share__box__element share__box__center">
|
||||||
<qrcode-vue :value="fullLink" size="200" level="M"></qrcode-vue>
|
<qrcode-vue :value="fullLink" size="200" level="M"></qrcode-vue>
|
||||||
@@ -57,7 +114,7 @@
|
|||||||
<div class="share__box__header" v-if="req.isDir">
|
<div class="share__box__header" v-if="req.isDir">
|
||||||
{{ $t("files.files") }}
|
{{ $t("files.files") }}
|
||||||
</div>
|
</div>
|
||||||
<div id="listing" class="list">
|
<div id="listing" class="list file-icons">
|
||||||
<item
|
<item
|
||||||
v-for="item in req.items.slice(0, this.showLimit)"
|
v-for="item in req.items.slice(0, this.showLimit)"
|
||||||
:key="base64(item.name)"
|
:key="base64(item.name)"
|
||||||
@@ -71,7 +128,11 @@
|
|||||||
readOnly
|
readOnly
|
||||||
>
|
>
|
||||||
</item>
|
</item>
|
||||||
<div v-if="req.items.length > showLimit" class="item">
|
<div
|
||||||
|
v-if="req.items.length > showLimit"
|
||||||
|
class="item"
|
||||||
|
@click="showLimit += 100"
|
||||||
|
>
|
||||||
<div>
|
<div>
|
||||||
<p class="name">+ {{ req.items.length - showLimit }}</p>
|
<p class="name">+ {{ req.items.length - showLimit }}</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -106,39 +167,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="error">
|
|
||||||
<div v-if="error.message === '401'">
|
|
||||||
<div class="card floating" id="password">
|
|
||||||
<div v-if="attemptedPasswordLogin" class="share__wrong__password">
|
|
||||||
{{ $t("login.wrongCredentials") }}
|
|
||||||
</div>
|
|
||||||
<div class="card-title">
|
|
||||||
<h2>{{ $t("login.password") }}</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="card-content">
|
|
||||||
<input
|
|
||||||
v-focus
|
|
||||||
type="password"
|
|
||||||
:placeholder="$t('login.password')"
|
|
||||||
v-model="password"
|
|
||||||
@keyup.enter="fetchData"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div class="card-action">
|
|
||||||
<button
|
|
||||||
class="button button--flat"
|
|
||||||
@click="fetchData"
|
|
||||||
:aria-label="$t('buttons.submit')"
|
|
||||||
:title="$t('buttons.submit')"
|
|
||||||
>
|
|
||||||
{{ $t("buttons.submit") }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<errors v-else :errorCode="errorCode" />
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -168,14 +196,18 @@ export default {
|
|||||||
},
|
},
|
||||||
data: () => ({
|
data: () => ({
|
||||||
error: null,
|
error: null,
|
||||||
showLimit: 500,
|
showLimit: 100,
|
||||||
password: "",
|
password: "",
|
||||||
attemptedPasswordLogin: false,
|
attemptedPasswordLogin: false,
|
||||||
hash: null,
|
hash: null,
|
||||||
token: null,
|
token: null,
|
||||||
}),
|
}),
|
||||||
watch: {
|
watch: {
|
||||||
$route: "fetchData",
|
$route: function () {
|
||||||
|
this.showLimit = 100;
|
||||||
|
|
||||||
|
this.fetchData();
|
||||||
|
},
|
||||||
},
|
},
|
||||||
created: async function () {
|
created: async function () {
|
||||||
const hash = this.$route.params.pathMatch.split("/")[0];
|
const hash = this.$route.params.pathMatch.split("/")[0];
|
||||||
@@ -207,6 +239,11 @@ export default {
|
|||||||
const path = this.$route.path.split("/").splice(2).join("/");
|
const path = this.$route.path.split("/").splice(2).join("/");
|
||||||
return `${baseURL}/api/public/dl/${path}${queryArg}`;
|
return `${baseURL}/api/public/dl/${path}${queryArg}`;
|
||||||
},
|
},
|
||||||
|
inlineLink: function () {
|
||||||
|
let url = new URL(this.fullLink);
|
||||||
|
url.searchParams.set("inline", "true");
|
||||||
|
return url.href;
|
||||||
|
},
|
||||||
fullLink: function () {
|
fullLink: function () {
|
||||||
return window.location.origin + this.link;
|
return window.location.origin + this.link;
|
||||||
},
|
},
|
||||||
@@ -220,10 +257,8 @@ export default {
|
|||||||
humanTime: function () {
|
humanTime: function () {
|
||||||
return moment(this.req.modified).fromNow();
|
return moment(this.req.modified).fromNow();
|
||||||
},
|
},
|
||||||
errorCode() {
|
modTime: function () {
|
||||||
return this.error.message === "404" || this.error.message === "403"
|
return new Date(Date.parse(this.req.modified)).toLocaleString();
|
||||||
? parseInt(this.error.message)
|
|
||||||
: 500;
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@@ -256,9 +291,11 @@ export default {
|
|||||||
this.token = file.token || "";
|
this.token = file.token || "";
|
||||||
|
|
||||||
this.updateRequest(file);
|
this.updateRequest(file);
|
||||||
this.setLoading(false);
|
document.title = `${file.name} - ${this.$route.name}`;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.error = e;
|
this.error = e;
|
||||||
|
} finally {
|
||||||
|
this.setLoading(false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
keyEvent(event) {
|
keyEvent(event) {
|
||||||
|
|||||||
@@ -4,14 +4,13 @@
|
|||||||
<action icon="close" :label="$t('buttons.close')" @action="close()" />
|
<action icon="close" :label="$t('buttons.close')" @action="close()" />
|
||||||
<title>{{ req.name }}</title>
|
<title>{{ req.name }}</title>
|
||||||
|
|
||||||
<template #actions>
|
|
||||||
<action
|
<action
|
||||||
|
v-if="user.perm.modify"
|
||||||
id="save-button"
|
id="save-button"
|
||||||
icon="save"
|
icon="save"
|
||||||
:label="$t('buttons.save')"
|
:label="$t('buttons.save')"
|
||||||
@action="save()"
|
@action="save()"
|
||||||
/>
|
/>
|
||||||
</template>
|
|
||||||
</header-bar>
|
</header-bar>
|
||||||
|
|
||||||
<breadcrumbs base="/files" noLink />
|
<breadcrumbs base="/files" noLink />
|
||||||
|
|||||||
@@ -25,18 +25,21 @@
|
|||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
v-if="headerButtons.copy"
|
v-if="headerButtons.copy"
|
||||||
|
id="copy-button"
|
||||||
icon="content_copy"
|
icon="content_copy"
|
||||||
:label="$t('buttons.copyFile')"
|
:label="$t('buttons.copyFile')"
|
||||||
show="copy"
|
show="copy"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
v-if="headerButtons.move"
|
v-if="headerButtons.move"
|
||||||
|
id="move-button"
|
||||||
icon="forward"
|
icon="forward"
|
||||||
:label="$t('buttons.moveFile')"
|
:label="$t('buttons.moveFile')"
|
||||||
show="move"
|
show="move"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
v-if="headerButtons.delete"
|
v-if="headerButtons.delete"
|
||||||
|
id="delete-button"
|
||||||
icon="delete"
|
icon="delete"
|
||||||
:label="$t('buttons.delete')"
|
:label="$t('buttons.delete')"
|
||||||
show="delete"
|
show="delete"
|
||||||
@@ -50,18 +53,21 @@
|
|||||||
@action="$store.commit('toggleShell')"
|
@action="$store.commit('toggleShell')"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
:icon="user.viewMode === 'mosaic' ? 'view_list' : 'view_module'"
|
:icon="viewIcon"
|
||||||
:label="$t('buttons.switchView')"
|
:label="$t('buttons.switchView')"
|
||||||
@action="switchView"
|
@action="switchView"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
|
v-if="headerButtons.download"
|
||||||
icon="file_download"
|
icon="file_download"
|
||||||
:label="$t('buttons.download')"
|
:label="$t('buttons.download')"
|
||||||
@action="download"
|
@action="download"
|
||||||
:counter="selectedCount"
|
:counter="selectedCount"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
|
v-if="headerButtons.upload"
|
||||||
icon="file_upload"
|
icon="file_upload"
|
||||||
|
id="upload-button"
|
||||||
:label="$t('buttons.upload')"
|
:label="$t('buttons.upload')"
|
||||||
@action="upload"
|
@action="upload"
|
||||||
/>
|
/>
|
||||||
@@ -108,8 +114,13 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="$store.state.loading">
|
<div v-if="loading">
|
||||||
<h2 class="message">
|
<h2 class="message delayed">
|
||||||
|
<div class="spinner">
|
||||||
|
<div class="bounce1"></div>
|
||||||
|
<div class="bounce2"></div>
|
||||||
|
<div class="bounce3"></div>
|
||||||
|
</div>
|
||||||
<span>{{ $t("files.loading") }}</span>
|
<span>{{ $t("files.loading") }}</span>
|
||||||
</h2>
|
</h2>
|
||||||
</div>
|
</div>
|
||||||
@@ -135,7 +146,12 @@
|
|||||||
multiple
|
multiple
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div v-else id="listing" :class="user.viewMode">
|
<div
|
||||||
|
v-else
|
||||||
|
id="listing"
|
||||||
|
ref="listing"
|
||||||
|
:class="user.viewMode + ' file-icons'"
|
||||||
|
>
|
||||||
<div>
|
<div>
|
||||||
<div class="item header">
|
<div class="item header">
|
||||||
<div></div>
|
<div></div>
|
||||||
@@ -248,11 +264,13 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import Vue from "vue";
|
||||||
import { mapState, mapGetters, mapMutations } from "vuex";
|
import { mapState, mapGetters, mapMutations } from "vuex";
|
||||||
import { users, files as api } from "@/api";
|
import { users, files as api } from "@/api";
|
||||||
import { enableExec } from "@/utils/constants";
|
import { enableExec } from "@/utils/constants";
|
||||||
import * as upload from "@/utils/upload";
|
import * as upload from "@/utils/upload";
|
||||||
import css from "@/utils/css";
|
import css from "@/utils/css";
|
||||||
|
import throttle from "lodash.throttle";
|
||||||
|
|
||||||
import HeaderBar from "@/components/header/HeaderBar";
|
import HeaderBar from "@/components/header/HeaderBar";
|
||||||
import Action from "@/components/header/Action";
|
import Action from "@/components/header/Action";
|
||||||
@@ -270,12 +288,22 @@ export default {
|
|||||||
data: function () {
|
data: function () {
|
||||||
return {
|
return {
|
||||||
showLimit: 50,
|
showLimit: 50,
|
||||||
|
columnWidth: 280,
|
||||||
dragCounter: 0,
|
dragCounter: 0,
|
||||||
width: window.innerWidth,
|
width: window.innerWidth,
|
||||||
|
itemWeight: 0,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(["req", "selected", "user", "show", "multiple", "selected"]),
|
...mapState([
|
||||||
|
"req",
|
||||||
|
"selected",
|
||||||
|
"user",
|
||||||
|
"show",
|
||||||
|
"multiple",
|
||||||
|
"selected",
|
||||||
|
"loading",
|
||||||
|
]),
|
||||||
...mapGetters(["selectedCount"]),
|
...mapGetters(["selectedCount"]),
|
||||||
nameSorted() {
|
nameSorted() {
|
||||||
return this.req.sorting.by === "name";
|
return this.req.sorting.by === "name";
|
||||||
@@ -334,6 +362,14 @@ export default {
|
|||||||
|
|
||||||
return "arrow_upward";
|
return "arrow_upward";
|
||||||
},
|
},
|
||||||
|
viewIcon() {
|
||||||
|
const icons = {
|
||||||
|
list: "view_module",
|
||||||
|
mosaic: "grid_view",
|
||||||
|
"mosaic gallery": "view_list",
|
||||||
|
};
|
||||||
|
return icons[this.user.viewMode];
|
||||||
|
},
|
||||||
headerButtons() {
|
headerButtons() {
|
||||||
return {
|
return {
|
||||||
upload: this.user.perm.create,
|
upload: this.user.perm.create,
|
||||||
@@ -350,15 +386,37 @@ export default {
|
|||||||
return this.width <= 736;
|
return this.width <= 736;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
watch: {
|
||||||
|
req: function () {
|
||||||
|
// Reset the show value
|
||||||
|
this.showLimit = 50;
|
||||||
|
|
||||||
|
// Ensures that the listing is displayed
|
||||||
|
Vue.nextTick(() => {
|
||||||
|
// How much every listing item affects the window height
|
||||||
|
this.setItemWeight();
|
||||||
|
|
||||||
|
// Fill and fit the window with listing items
|
||||||
|
this.fillWindow(true);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
mounted: function () {
|
mounted: function () {
|
||||||
// Check the columns size for the first time.
|
// Check the columns size for the first time.
|
||||||
this.resizeEvent();
|
this.colunmsResize();
|
||||||
|
|
||||||
|
// How much every listing item affects the window height
|
||||||
|
this.setItemWeight();
|
||||||
|
|
||||||
|
// Fill and fit the window with listing items
|
||||||
|
this.fillWindow(true);
|
||||||
|
|
||||||
// Add the needed event listeners to the window and document.
|
// Add the needed event listeners to the window and document.
|
||||||
window.addEventListener("keydown", this.keyEvent);
|
window.addEventListener("keydown", this.keyEvent);
|
||||||
window.addEventListener("resize", this.resizeEvent);
|
|
||||||
window.addEventListener("scroll", this.scrollEvent);
|
window.addEventListener("scroll", this.scrollEvent);
|
||||||
window.addEventListener("resize", this.windowsResize);
|
window.addEventListener("resize", this.windowsResize);
|
||||||
|
|
||||||
|
if (!this.user.perm.create) return;
|
||||||
document.addEventListener("dragover", this.preventDefault);
|
document.addEventListener("dragover", this.preventDefault);
|
||||||
document.addEventListener("dragenter", this.dragEnter);
|
document.addEventListener("dragenter", this.dragEnter);
|
||||||
document.addEventListener("dragleave", this.dragLeave);
|
document.addEventListener("dragleave", this.dragLeave);
|
||||||
@@ -367,9 +425,10 @@ export default {
|
|||||||
beforeDestroy() {
|
beforeDestroy() {
|
||||||
// Remove event listeners before destroying this page.
|
// Remove event listeners before destroying this page.
|
||||||
window.removeEventListener("keydown", this.keyEvent);
|
window.removeEventListener("keydown", this.keyEvent);
|
||||||
window.removeEventListener("resize", this.resizeEvent);
|
|
||||||
window.removeEventListener("scroll", this.scrollEvent);
|
window.removeEventListener("scroll", this.scrollEvent);
|
||||||
window.removeEventListener("resize", this.windowsResize);
|
window.removeEventListener("resize", this.windowsResize);
|
||||||
|
|
||||||
|
if (this.user && !this.user.perm.create) return;
|
||||||
document.removeEventListener("dragover", this.preventDefault);
|
document.removeEventListener("dragover", this.preventDefault);
|
||||||
document.removeEventListener("dragenter", this.dragEnter);
|
document.removeEventListener("dragenter", this.dragEnter);
|
||||||
document.removeEventListener("dragleave", this.dragLeave);
|
document.removeEventListener("dragleave", this.dragLeave);
|
||||||
@@ -460,7 +519,7 @@ export default {
|
|||||||
for (let i of this.selected) {
|
for (let i of this.selected) {
|
||||||
items.push({
|
items.push({
|
||||||
from: this.req.items[i].url,
|
from: this.req.items[i].url,
|
||||||
name: encodeURIComponent(this.req.items[i].name),
|
name: this.req.items[i].name,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -485,7 +544,7 @@ export default {
|
|||||||
const from = item.from.endsWith("/")
|
const from = item.from.endsWith("/")
|
||||||
? item.from.slice(0, -1)
|
? item.from.slice(0, -1)
|
||||||
: item.from;
|
: item.from;
|
||||||
const to = this.$route.path + item.name;
|
const to = this.$route.path + encodeURIComponent(item.name);
|
||||||
items.push({ from, to, name: item.name });
|
items.push({ from, to, name: item.name });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -543,20 +602,36 @@ export default {
|
|||||||
|
|
||||||
action(overwrite, rename);
|
action(overwrite, rename);
|
||||||
},
|
},
|
||||||
resizeEvent() {
|
colunmsResize() {
|
||||||
// Update the columns size based on the window width.
|
// Update the columns size based on the window width.
|
||||||
let columns = Math.floor(
|
let columns = Math.floor(
|
||||||
document.querySelector("main").offsetWidth / 300
|
document.querySelector("main").offsetWidth / this.columnWidth
|
||||||
);
|
);
|
||||||
let items = css(["#listing.mosaic .item", ".mosaic#listing .item"]);
|
let items = css(["#listing.mosaic .item", ".mosaic#listing .item"]);
|
||||||
if (columns === 0) columns = 1;
|
if (columns === 0) columns = 1;
|
||||||
items.style.width = `calc(${100 / columns}% - 1em)`;
|
items.style.width = `calc(${100 / columns}% - 1em)`;
|
||||||
},
|
},
|
||||||
scrollEvent() {
|
scrollEvent: throttle(function () {
|
||||||
if (window.innerHeight + window.scrollY >= document.body.offsetHeight) {
|
const totalItems = this.req.numDirs + this.req.numFiles;
|
||||||
this.showLimit += 50;
|
|
||||||
|
// All items are displayed
|
||||||
|
if (this.showLimit >= totalItems) return;
|
||||||
|
|
||||||
|
const currentPos = window.innerHeight + window.scrollY;
|
||||||
|
|
||||||
|
// Trigger at the 75% of the window height
|
||||||
|
const triggerPos = document.body.offsetHeight - window.innerHeight * 0.25;
|
||||||
|
|
||||||
|
if (currentPos > triggerPos) {
|
||||||
|
// Quantity of items needed to fill 2x of the window height
|
||||||
|
const showQuantity = Math.ceil(
|
||||||
|
(window.innerHeight * 2) / this.itemWeight
|
||||||
|
);
|
||||||
|
|
||||||
|
// Increase the number of displayed items
|
||||||
|
this.showLimit += showQuantity;
|
||||||
}
|
}
|
||||||
},
|
}, 100),
|
||||||
dragEnter() {
|
dragEnter() {
|
||||||
this.dragCounter++;
|
this.dragCounter++;
|
||||||
|
|
||||||
@@ -591,22 +666,20 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let base = "";
|
let files = await upload.scanFiles(dt);
|
||||||
|
let items = this.req.items;
|
||||||
|
let path = this.$route.path.endsWith("/")
|
||||||
|
? this.$route.path
|
||||||
|
: this.$route.path + "/";
|
||||||
|
|
||||||
if (
|
if (
|
||||||
el !== null &&
|
el !== null &&
|
||||||
el.classList.contains("item") &&
|
el.classList.contains("item") &&
|
||||||
el.dataset.dir === "true"
|
el.dataset.dir === "true"
|
||||||
) {
|
) {
|
||||||
base = el.querySelector(".name").innerHTML + "/";
|
// Get url from ListingItem instance
|
||||||
}
|
path = el.__vue__.url;
|
||||||
|
|
||||||
let files = await upload.scanFiles(dt);
|
|
||||||
let path = this.$route.path.endsWith("/")
|
|
||||||
? this.$route.path + base
|
|
||||||
: this.$route.path + "/" + base;
|
|
||||||
let items = this.req.items;
|
|
||||||
|
|
||||||
if (base !== "") {
|
|
||||||
try {
|
try {
|
||||||
items = (await api.fetch(path)).items;
|
items = (await api.fetch(path)).items;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -707,9 +780,19 @@ export default {
|
|||||||
this.$store.commit("multiple", !this.multiple);
|
this.$store.commit("multiple", !this.multiple);
|
||||||
this.$store.commit("closeHovers");
|
this.$store.commit("closeHovers");
|
||||||
},
|
},
|
||||||
windowsResize() {
|
windowsResize: throttle(function () {
|
||||||
|
this.colunmsResize();
|
||||||
this.width = window.innerWidth;
|
this.width = window.innerWidth;
|
||||||
},
|
|
||||||
|
// Listing element is not displayed
|
||||||
|
if (this.$refs.listing == null) return;
|
||||||
|
|
||||||
|
// How much every listing item affects the window height
|
||||||
|
this.setItemWeight();
|
||||||
|
|
||||||
|
// Fill but not fit the window
|
||||||
|
this.fillWindow();
|
||||||
|
}, 100),
|
||||||
download() {
|
download() {
|
||||||
if (this.selectedCount === 1 && !this.req.items[this.selected[0]].isDir) {
|
if (this.selectedCount === 1 && !this.req.items[this.selected[0]].isDir) {
|
||||||
api.download(null, this.req.items[this.selected[0]].url);
|
api.download(null, this.req.items[this.selected[0]].url);
|
||||||
@@ -738,25 +821,64 @@ export default {
|
|||||||
switchView: async function () {
|
switchView: async function () {
|
||||||
this.$store.commit("closeHovers");
|
this.$store.commit("closeHovers");
|
||||||
|
|
||||||
const data = {
|
const modes = {
|
||||||
id: this.user.id,
|
list: "mosaic",
|
||||||
viewMode: this.user.viewMode === "mosaic" ? "list" : "mosaic",
|
mosaic: "mosaic gallery",
|
||||||
|
"mosaic gallery": "list",
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
const data = {
|
||||||
await users.update(data, ["viewMode"]);
|
id: this.user.id,
|
||||||
this.$store.commit("updateUser", data);
|
viewMode: modes[this.user.viewMode] || "list",
|
||||||
} catch (e) {
|
};
|
||||||
this.$showError(e);
|
|
||||||
}
|
users.update(data, ["viewMode"]).catch(this.$showError);
|
||||||
|
|
||||||
|
// Await ensures correct value for setItemWeight()
|
||||||
|
await this.$store.commit("updateUser", data);
|
||||||
|
|
||||||
|
this.setItemWeight();
|
||||||
|
this.fillWindow();
|
||||||
},
|
},
|
||||||
upload: function () {
|
upload: function () {
|
||||||
if (typeof DataTransferItem.prototype.webkitGetAsEntry !== "undefined") {
|
if (
|
||||||
|
typeof window.DataTransferItem !== "undefined" &&
|
||||||
|
typeof DataTransferItem.prototype.webkitGetAsEntry !== "undefined"
|
||||||
|
) {
|
||||||
this.$store.commit("showHover", "upload");
|
this.$store.commit("showHover", "upload");
|
||||||
} else {
|
} else {
|
||||||
document.getElementById("upload-input").click();
|
document.getElementById("upload-input").click();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
setItemWeight() {
|
||||||
|
// Listing element is not displayed
|
||||||
|
if (this.$refs.listing == null) return;
|
||||||
|
|
||||||
|
let itemQuantity = this.req.numDirs + this.req.numFiles;
|
||||||
|
if (itemQuantity > this.showLimit) itemQuantity = this.showLimit;
|
||||||
|
|
||||||
|
// How much every listing item affects the window height
|
||||||
|
this.itemWeight = this.$refs.listing.offsetHeight / itemQuantity;
|
||||||
|
},
|
||||||
|
fillWindow(fit = false) {
|
||||||
|
const totalItems = this.req.numDirs + this.req.numFiles;
|
||||||
|
|
||||||
|
// More items are displayed than the total
|
||||||
|
if (this.showLimit >= totalItems && !fit) return;
|
||||||
|
|
||||||
|
const windowHeight = window.innerHeight;
|
||||||
|
|
||||||
|
// Quantity of items needed to fill 2x of the window height
|
||||||
|
const showQuantity = Math.ceil(
|
||||||
|
(windowHeight + windowHeight * 2) / this.itemWeight
|
||||||
|
);
|
||||||
|
|
||||||
|
// Less items to display than current
|
||||||
|
if (this.showLimit > showQuantity && !fit) return;
|
||||||
|
|
||||||
|
// Set the number of displayed items
|
||||||
|
this.showLimit = showQuantity > totalItems ? totalItems : showQuantity;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -17,12 +17,14 @@
|
|||||||
<template #actions>
|
<template #actions>
|
||||||
<action
|
<action
|
||||||
:disabled="loading"
|
:disabled="loading"
|
||||||
|
v-if="user.perm.rename"
|
||||||
icon="mode_edit"
|
icon="mode_edit"
|
||||||
:label="$t('buttons.rename')"
|
:label="$t('buttons.rename')"
|
||||||
show="rename"
|
show="rename"
|
||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
:disabled="loading"
|
:disabled="loading"
|
||||||
|
v-if="user.perm.delete"
|
||||||
icon="delete"
|
icon="delete"
|
||||||
:label="$t('buttons.delete')"
|
:label="$t('buttons.delete')"
|
||||||
@action="deleteFile"
|
@action="deleteFile"
|
||||||
@@ -30,6 +32,7 @@
|
|||||||
/>
|
/>
|
||||||
<action
|
<action
|
||||||
:disabled="loading"
|
:disabled="loading"
|
||||||
|
v-if="user.perm.download"
|
||||||
icon="file_download"
|
icon="file_download"
|
||||||
:label="$t('buttons.download')"
|
:label="$t('buttons.download')"
|
||||||
@action="download"
|
@action="download"
|
||||||
@@ -43,19 +46,32 @@
|
|||||||
</template>
|
</template>
|
||||||
</header-bar>
|
</header-bar>
|
||||||
|
|
||||||
<div class="loading" v-if="loading">
|
<div class="loading delayed" v-if="loading">
|
||||||
<div class="spinner">
|
<div class="spinner">
|
||||||
<div class="bounce1"></div>
|
<div class="bounce1"></div>
|
||||||
<div class="bounce2"></div>
|
<div class="bounce2"></div>
|
||||||
<div class="bounce3"></div>
|
<div class="bounce3"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<template v-else>
|
||||||
<template v-if="!loading">
|
|
||||||
<div class="preview">
|
<div class="preview">
|
||||||
<ExtendedImage v-if="req.type == 'image'" :src="raw"></ExtendedImage>
|
<ExtendedImage v-if="req.type == 'image'" :src="raw"></ExtendedImage>
|
||||||
<audio v-else-if="req.type == 'audio'" :src="raw" controls></audio>
|
<audio
|
||||||
<video v-else-if="req.type == 'video'" :src="raw" controls>
|
v-else-if="req.type == 'audio'"
|
||||||
|
ref="player"
|
||||||
|
:src="raw"
|
||||||
|
controls
|
||||||
|
:autoplay="autoPlay"
|
||||||
|
@play="autoPlay = true"
|
||||||
|
></audio>
|
||||||
|
<video
|
||||||
|
v-else-if="req.type == 'video'"
|
||||||
|
ref="player"
|
||||||
|
:src="raw"
|
||||||
|
controls
|
||||||
|
:autoplay="autoPlay"
|
||||||
|
@play="autoPlay = true"
|
||||||
|
>
|
||||||
<track
|
<track
|
||||||
kind="captions"
|
kind="captions"
|
||||||
v-for="(sub, index) in subtitles"
|
v-for="(sub, index) in subtitles"
|
||||||
@@ -73,12 +89,31 @@
|
|||||||
class="pdf"
|
class="pdf"
|
||||||
:data="raw"
|
:data="raw"
|
||||||
></object>
|
></object>
|
||||||
<a v-else-if="req.type == 'blob'" :href="downloadUrl">
|
<div v-else-if="req.type == 'blob'" class="info">
|
||||||
<h2 class="message">
|
<div class="title">
|
||||||
{{ $t("buttons.download") }}
|
<i class="material-icons">feedback</i>
|
||||||
<i class="material-icons">file_download</i>
|
{{ $t("files.noPreview") }}
|
||||||
</h2>
|
</div>
|
||||||
|
<div>
|
||||||
|
<a target="_blank" :href="downloadUrl" class="button button--flat">
|
||||||
|
<div>
|
||||||
|
<i class="material-icons">file_download</i
|
||||||
|
>{{ $t("buttons.download") }}
|
||||||
|
</div>
|
||||||
</a>
|
</a>
|
||||||
|
<a
|
||||||
|
target="_blank"
|
||||||
|
:href="downloadUrl + '&inline=true'"
|
||||||
|
class="button button--flat"
|
||||||
|
v-if="!req.isDir"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<i class="material-icons">open_in_new</i
|
||||||
|
>{{ $t("buttons.openFile") }}
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -102,6 +137,8 @@
|
|||||||
>
|
>
|
||||||
<i class="material-icons">chevron_right</i>
|
<i class="material-icons">chevron_right</i>
|
||||||
</button>
|
</button>
|
||||||
|
<link rel="prefetch" :href="previousRaw" />
|
||||||
|
<link rel="prefetch" :href="nextRaw" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -111,7 +148,6 @@ import { files as api } from "@/api";
|
|||||||
import { baseURL, resizePreview } from "@/utils/constants";
|
import { baseURL, resizePreview } from "@/utils/constants";
|
||||||
import url from "@/utils/url";
|
import url from "@/utils/url";
|
||||||
import throttle from "lodash.throttle";
|
import throttle from "lodash.throttle";
|
||||||
|
|
||||||
import HeaderBar from "@/components/header/HeaderBar";
|
import HeaderBar from "@/components/header/HeaderBar";
|
||||||
import Action from "@/components/header/Action";
|
import Action from "@/components/header/Action";
|
||||||
import ExtendedImage from "@/components/files/ExtendedImage";
|
import ExtendedImage from "@/components/files/ExtendedImage";
|
||||||
@@ -136,6 +172,9 @@ export default {
|
|||||||
showNav: true,
|
showNav: true,
|
||||||
navTimeout: null,
|
navTimeout: null,
|
||||||
hoverNav: false,
|
hoverNav: false,
|
||||||
|
autoPlay: false,
|
||||||
|
previousRaw: "",
|
||||||
|
nextRaw: "",
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -158,11 +197,9 @@ export default {
|
|||||||
if (this.req.type === "image" && !this.fullSize) {
|
if (this.req.type === "image" && !this.fullSize) {
|
||||||
return `${baseURL}/api/preview/big${url.encodePath(
|
return `${baseURL}/api/preview/big${url.encodePath(
|
||||||
this.req.path
|
this.req.path
|
||||||
)}?auth=${this.jwt}&k=${key}`;
|
)}?k=${key}`;
|
||||||
}
|
}
|
||||||
return `${baseURL}/api/raw${url.encodePath(this.req.path)}?auth=${
|
return `${baseURL}/api/raw${url.encodePath(this.req.path)}?k=${key}`;
|
||||||
this.jwt
|
|
||||||
}&k=${key}`;
|
|
||||||
},
|
},
|
||||||
raw() {
|
raw() {
|
||||||
return `${this.previewUrl}&inline=true`;
|
return `${this.previewUrl}&inline=true`;
|
||||||
@@ -207,11 +244,11 @@ export default {
|
|||||||
},
|
},
|
||||||
prev() {
|
prev() {
|
||||||
this.hoverNav = false;
|
this.hoverNav = false;
|
||||||
this.$router.push({ path: this.previousLink });
|
this.$router.replace({ path: this.previousLink });
|
||||||
},
|
},
|
||||||
next() {
|
next() {
|
||||||
this.hoverNav = false;
|
this.hoverNav = false;
|
||||||
this.$router.push({ path: this.nextLink });
|
this.$router.replace({ path: this.nextLink });
|
||||||
},
|
},
|
||||||
key(event) {
|
key(event) {
|
||||||
if (this.show !== null) {
|
if (this.show !== null) {
|
||||||
@@ -230,9 +267,17 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
async updatePreview() {
|
async updatePreview() {
|
||||||
|
if (
|
||||||
|
this.$refs.player &&
|
||||||
|
this.$refs.player.paused &&
|
||||||
|
!this.$refs.player.ended
|
||||||
|
) {
|
||||||
|
this.autoPlay = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (this.req.subtitles) {
|
if (this.req.subtitles) {
|
||||||
this.subtitles = this.req.subtitles.map(
|
this.subtitles = this.req.subtitles.map(
|
||||||
(sub) => `${baseURL}/api/raw${sub}?auth=${this.jwt}&inline=true`
|
(sub) => `${baseURL}/api/raw${sub}?inline=true`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -260,13 +305,14 @@ export default {
|
|||||||
for (let j = i - 1; j >= 0; j--) {
|
for (let j = i - 1; j >= 0; j--) {
|
||||||
if (mediaTypes.includes(this.listing[j].type)) {
|
if (mediaTypes.includes(this.listing[j].type)) {
|
||||||
this.previousLink = this.listing[j].url;
|
this.previousLink = this.listing[j].url;
|
||||||
|
this.previousRaw = this.prefetchUrl(this.listing[j]);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let j = i + 1; j < this.listing.length; j++) {
|
for (let j = i + 1; j < this.listing.length; j++) {
|
||||||
if (mediaTypes.includes(this.listing[j].type)) {
|
if (mediaTypes.includes(this.listing[j].type)) {
|
||||||
this.nextLink = this.listing[j].url;
|
this.nextLink = this.listing[j].url;
|
||||||
|
this.nextRaw = this.prefetchUrl(this.listing[j]);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -274,6 +320,16 @@ export default {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
prefetchUrl: function (item) {
|
||||||
|
const key = Date.parse(item.modified);
|
||||||
|
if (item.type === "image" && !this.fullSize) {
|
||||||
|
return `${baseURL}/api/preview/big${item.path}?k=${key}&inline=true`;
|
||||||
|
} else if (item.type === "image") {
|
||||||
|
return `${baseURL}/api/raw${item.path}?k=${key}&inline=true`;
|
||||||
|
} else {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
},
|
||||||
openMore() {
|
openMore() {
|
||||||
this.$store.commit("showHover", "more");
|
this.$store.commit("showHover", "more");
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="row" v-if="settings !== null">
|
<errors v-if="error" :errorCode="error.message" />
|
||||||
|
<div class="row" v-else-if="!loading">
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<form class="card" @submit.prevent="save">
|
<form class="card" @submit.prevent="save">
|
||||||
<div class="card-title">
|
<div class="card-title">
|
||||||
@@ -170,12 +171,13 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapState } from "vuex";
|
import { mapState, mapMutations } from "vuex";
|
||||||
import { settings as api } from "@/api";
|
import { settings as api } from "@/api";
|
||||||
|
import { enableExec } from "@/utils/constants";
|
||||||
import UserForm from "@/components/settings/UserForm";
|
import UserForm from "@/components/settings/UserForm";
|
||||||
import Rules from "@/components/settings/Rules";
|
import Rules from "@/components/settings/Rules";
|
||||||
import Themes from "@/components/settings/Themes";
|
import Themes from "@/components/settings/Themes";
|
||||||
import { enableExec } from "@/utils/constants";
|
import Errors from "@/views/Errors";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "settings",
|
name: "settings",
|
||||||
@@ -183,19 +185,23 @@ export default {
|
|||||||
Themes,
|
Themes,
|
||||||
UserForm,
|
UserForm,
|
||||||
Rules,
|
Rules,
|
||||||
|
Errors,
|
||||||
},
|
},
|
||||||
data: function () {
|
data: function () {
|
||||||
return {
|
return {
|
||||||
|
error: null,
|
||||||
originalSettings: null,
|
originalSettings: null,
|
||||||
settings: null,
|
settings: null,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(["user"]),
|
...mapState(["user", "loading"]),
|
||||||
isExecEnabled: () => enableExec,
|
isExecEnabled: () => enableExec,
|
||||||
},
|
},
|
||||||
async created() {
|
async created() {
|
||||||
try {
|
try {
|
||||||
|
this.setLoading(true);
|
||||||
|
|
||||||
const original = await api.get();
|
const original = await api.get();
|
||||||
let settings = { ...original, commands: [] };
|
let settings = { ...original, commands: [] };
|
||||||
|
|
||||||
@@ -211,10 +217,13 @@ export default {
|
|||||||
this.originalSettings = original;
|
this.originalSettings = original;
|
||||||
this.settings = settings;
|
this.settings = settings;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.$showError(e);
|
this.error = e;
|
||||||
|
} finally {
|
||||||
|
this.setLoading(false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
...mapMutations(["setLoading"]),
|
||||||
capitalize(name, where = "_") {
|
capitalize(name, where = "_") {
|
||||||
if (where === "caps") where = /(?=[A-Z])/;
|
if (where === "caps") where = /(?=[A-Z])/;
|
||||||
let splitted = name.split(where);
|
let splitted = name.split(where);
|
||||||
|
|||||||
@@ -15,6 +15,10 @@
|
|||||||
<input type="checkbox" v-model="singleClick" />
|
<input type="checkbox" v-model="singleClick" />
|
||||||
{{ $t("settings.singleClick") }}
|
{{ $t("settings.singleClick") }}
|
||||||
</p>
|
</p>
|
||||||
|
<p>
|
||||||
|
<input type="checkbox" v-model="dateFormat" />
|
||||||
|
{{ $t("settings.setDateFormat") }}
|
||||||
|
</p>
|
||||||
<h3>{{ $t("settings.language") }}</h3>
|
<h3>{{ $t("settings.language") }}</h3>
|
||||||
<languages
|
<languages
|
||||||
class="input input--block"
|
class="input input--block"
|
||||||
@@ -83,6 +87,7 @@ export default {
|
|||||||
passwordConf: "",
|
passwordConf: "",
|
||||||
hideDotfiles: false,
|
hideDotfiles: false,
|
||||||
singleClick: false,
|
singleClick: false,
|
||||||
|
dateFormat: false,
|
||||||
locale: "",
|
locale: "",
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@@ -103,12 +108,14 @@ export default {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
|
this.setLoading(false);
|
||||||
this.locale = this.user.locale;
|
this.locale = this.user.locale;
|
||||||
this.hideDotfiles = this.user.hideDotfiles;
|
this.hideDotfiles = this.user.hideDotfiles;
|
||||||
this.singleClick = this.user.singleClick;
|
this.singleClick = this.user.singleClick;
|
||||||
|
this.dateFormat = this.user.dateFormat;
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapMutations(["updateUser"]),
|
...mapMutations(["updateUser", "setLoading"]),
|
||||||
async updatePassword(event) {
|
async updatePassword(event) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
|
||||||
@@ -134,8 +141,14 @@ export default {
|
|||||||
locale: this.locale,
|
locale: this.locale,
|
||||||
hideDotfiles: this.hideDotfiles,
|
hideDotfiles: this.hideDotfiles,
|
||||||
singleClick: this.singleClick,
|
singleClick: this.singleClick,
|
||||||
|
dateFormat: this.dateFormat,
|
||||||
};
|
};
|
||||||
await api.update(data, ["locale", "hideDotfiles", "singleClick"]);
|
await api.update(data, [
|
||||||
|
"locale",
|
||||||
|
"hideDotfiles",
|
||||||
|
"singleClick",
|
||||||
|
"dateFormat",
|
||||||
|
]);
|
||||||
this.updateUser(data);
|
this.updateUser(data);
|
||||||
this.$showSuccess(this.$t("settings.settingsUpdated"));
|
this.$showSuccess(this.$t("settings.settingsUpdated"));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="row">
|
<errors v-if="error" :errorCode="error.message" />
|
||||||
|
<div class="row" v-else-if="!loading">
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<div class="card-title">
|
<div class="card-title">
|
||||||
<h2>{{ $t("settings.shareManagement") }}</h2>
|
<h2>{{ $t("settings.shareManagement") }}</h2>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="card-content full">
|
<div class="card-content full" v-if="links.length > 0">
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<th>{{ $t("settings.path") }}</th>
|
<th>{{ $t("settings.path") }}</th>
|
||||||
@@ -52,6 +53,10 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
<h2 class="message" v-else>
|
||||||
|
<i class="material-icons">sentiment_dissatisfied</i>
|
||||||
|
<span>{{ $t("files.lonely") }}</span>
|
||||||
|
</h2>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -59,21 +64,28 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { share as api, users } from "@/api";
|
import { share as api, users } from "@/api";
|
||||||
import moment from "moment";
|
|
||||||
import { baseURL } from "@/utils/constants";
|
import { baseURL } from "@/utils/constants";
|
||||||
|
import { mapState, mapMutations } from "vuex";
|
||||||
|
import moment from "moment";
|
||||||
import Clipboard from "clipboard";
|
import Clipboard from "clipboard";
|
||||||
import { mapState } from "vuex";
|
import Errors from "@/views/Errors";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "shares",
|
name: "shares",
|
||||||
computed: mapState(["user"]),
|
components: {
|
||||||
|
Errors,
|
||||||
|
},
|
||||||
|
computed: mapState(["user", "loading"]),
|
||||||
data: function () {
|
data: function () {
|
||||||
return {
|
return {
|
||||||
|
error: null,
|
||||||
links: [],
|
links: [],
|
||||||
clip: null,
|
clip: null,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
async created() {
|
async created() {
|
||||||
|
this.setLoading(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
let links = await api.list();
|
let links = await api.list();
|
||||||
if (this.user.perm.admin) {
|
if (this.user.perm.admin) {
|
||||||
@@ -87,7 +99,9 @@ export default {
|
|||||||
}
|
}
|
||||||
this.links = links;
|
this.links = links;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.$showError(e);
|
this.error = e;
|
||||||
|
} finally {
|
||||||
|
this.setLoading(false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
@@ -100,6 +114,7 @@ export default {
|
|||||||
this.clip.destroy();
|
this.clip.destroy();
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
...mapMutations(["setLoading"]),
|
||||||
deleteLink: async function (event, link) {
|
deleteLink: async function (event, link) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
|
||||||
@@ -108,8 +123,13 @@ export default {
|
|||||||
confirm: () => {
|
confirm: () => {
|
||||||
this.$store.commit("closeHovers");
|
this.$store.commit("closeHovers");
|
||||||
|
|
||||||
|
try {
|
||||||
api.remove(link.hash);
|
api.remove(link.hash);
|
||||||
this.links = this.links.filter((item) => item.hash !== link.hash);
|
this.links = this.links.filter((item) => item.hash !== link.hash);
|
||||||
|
this.$showSuccess(this.$t("settings.shareDeleted"));
|
||||||
|
} catch (e) {
|
||||||
|
this.$showError(e);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user