添加图片懒加载,图片预览,优化grid view mode

This commit is contained in:
Aicirou 2020-05-16 19:58:25 +08:00
parent 40f9a15294
commit 53ef30577e
9 changed files with 238 additions and 108 deletions

26
package-lock.json generated
View File

@ -12226,6 +12226,22 @@
"resolved": "https://registry.npm.taobao.org/uuid/download/uuid-3.4.0.tgz?cache=0&sync_timestamp=1585683718911&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fuuid%2Fdownload%2Fuuid-3.4.0.tgz", "resolved": "https://registry.npm.taobao.org/uuid/download/uuid-3.4.0.tgz?cache=0&sync_timestamp=1585683718911&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fuuid%2Fdownload%2Fuuid-3.4.0.tgz",
"integrity": "sha1-sj5DWK+oogL+ehAK8fX4g/AgB+4=" "integrity": "sha1-sj5DWK+oogL+ehAK8fX4g/AgB+4="
}, },
"v-viewer": {
"version": "1.5.1",
"resolved": "https://registry.npm.taobao.org/v-viewer/download/v-viewer-1.5.1.tgz",
"integrity": "sha1-lIBNg4FMylZuaRM5/h5keTSG8gA=",
"requires": {
"throttle-debounce": "^2.0.1",
"viewerjs": "^1.5.0"
},
"dependencies": {
"throttle-debounce": {
"version": "2.1.0",
"resolved": "https://registry.npm.taobao.org/throttle-debounce/download/throttle-debounce-2.1.0.tgz",
"integrity": "sha1-JX5kjwpWvZ5U/g8TLEq4YR304dU="
}
}
},
"v8-compile-cache": { "v8-compile-cache": {
"version": "2.1.0", "version": "2.1.0",
"resolved": "https://registry.npm.taobao.org/v8-compile-cache/download/v8-compile-cache-2.1.0.tgz", "resolved": "https://registry.npm.taobao.org/v8-compile-cache/download/v8-compile-cache-2.1.0.tgz",
@ -12263,6 +12279,11 @@
"extsprintf": "^1.2.0" "extsprintf": "^1.2.0"
} }
}, },
"viewerjs": {
"version": "1.5.0",
"resolved": "https://registry.npm.taobao.org/viewerjs/download/viewerjs-1.5.0.tgz?cache=0&sync_timestamp=1574486727393&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fviewerjs%2Fdownload%2Fviewerjs-1.5.0.tgz",
"integrity": "sha1-3r6pSSGJ+kEiPMVn3pFdeUNYmdA="
},
"vm-browserify": { "vm-browserify": {
"version": "1.1.2", "version": "1.1.2",
"resolved": "https://registry.npm.taobao.org/vm-browserify/download/vm-browserify-1.1.2.tgz", "resolved": "https://registry.npm.taobao.org/vm-browserify/download/vm-browserify-1.1.2.tgz",
@ -12503,6 +12524,11 @@
"resolved": "https://registry.npm.taobao.org/vue-infinite-scroll/download/vue-infinite-scroll-2.0.2.tgz", "resolved": "https://registry.npm.taobao.org/vue-infinite-scroll/download/vue-infinite-scroll-2.0.2.tgz",
"integrity": "sha1-yjepH+ku4K07dKz4aCwAkXFEtxE=" "integrity": "sha1-yjepH+ku4K07dKz4aCwAkXFEtxE="
}, },
"vue-lazyload": {
"version": "1.3.3",
"resolved": "https://registry.npm.taobao.org/vue-lazyload/download/vue-lazyload-1.3.3.tgz",
"integrity": "sha1-TfUKJxvem3TDyveiKNbgr1DVaC8="
},
"vue-loader": { "vue-loader": {
"version": "15.9.1", "version": "15.9.1",
"resolved": "https://registry.npm.taobao.org/vue-loader/download/vue-loader-15.9.1.tgz?cache=0&sync_timestamp=1584584267698&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fvue-loader%2Fdownload%2Fvue-loader-15.9.1.tgz", "resolved": "https://registry.npm.taobao.org/vue-loader/download/vue-loader-15.9.1.tgz?cache=0&sync_timestamp=1584584267698&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fvue-loader%2Fdownload%2Fvue-loader-15.9.1.tgz",

View File

@ -22,11 +22,13 @@
"markdown-it": "^10.0.0", "markdown-it": "^10.0.0",
"node-sass": "^4.14.0", "node-sass": "^4.14.0",
"sass-loader": "^8.0.2", "sass-loader": "^8.0.2",
"v-viewer": "^1.5.1",
"vue": "^2.6.11", "vue": "^2.6.11",
"vue-axios": "^2.1.5", "vue-axios": "^2.1.5",
"vue-clipboard2": "^0.3.1", "vue-clipboard2": "^0.3.1",
"vue-i18n": "^8.17.3", "vue-i18n": "^8.17.3",
"vue-infinite-scroll": "^2.0.2", "vue-infinite-scroll": "^2.0.2",
"vue-lazyload": "^1.3.3",
"vue-router": "^3.1.6", "vue-router": "^3.1.6",
"vue2-ace-editor": "0.0.15", "vue2-ace-editor": "0.0.15",
"vuex": "^3.4.0" "vuex": "^3.4.0"

View File

@ -0,0 +1,33 @@
<template>
<span
class="icon"
v-show="$route.meta.view === 'list'"
:title="
mode === 'list' ? $t('list.view.gridMode') : $t('list.view.listMode')
"
@click="toggleMode"
>
<i
:class="'fa' + (mode === 'list' ? ' fa-th' : ' fa-th-list')"
aria-hidden="true"
></i>
</span>
</template>
<script>
import { mapState, mapActions } from "vuex";
export default {
data() {
return {};
},
computed: {
...mapState("acrou/view", ["mode"]),
},
methods: {
...mapActions("acrou/view", ["toggle"]),
toggleMode() {
this.toggle(this.mode === "list" ? "grid" : "list");
},
},
};
</script>

View File

@ -12,6 +12,9 @@ import i18n from './i18n'
import store from '@/store/index' import store from '@/store/index'
import VueClipboard from 'vue-clipboard2' import VueClipboard from 'vue-clipboard2'
import infiniteScroll from 'vue-infinite-scroll' import infiniteScroll from 'vue-infinite-scroll'
import VueLazyload from 'vue-lazyload'
import Viewer from 'v-viewer'
import 'viewerjs/dist/viewer.css'
import "@/assets/style/theme/register.scss" import "@/assets/style/theme/register.scss"
Vue.config.productionTip = false Vue.config.productionTip = false
@ -19,6 +22,11 @@ Vue.use(ElementUI)
Vue.use(VueAxios,axios) Vue.use(VueAxios,axios)
Vue.use(VueClipboard) Vue.use(VueClipboard)
Vue.use(infiniteScroll) Vue.use(infiniteScroll)
Vue.use(VueLazyload,{
// error: 'dist/error.png',
loading: 'https://cloud.jsonpop.cn/go2index/vue/airplane.gif'
})
Vue.use(Viewer)
new Vue({ new Vue({
router, router,

View File

@ -1,7 +1,7 @@
<template> <template>
<div> <div>
<nav <nav
class="breadcrumb is-hidden-mobile is-hidden-touch level g2-breadcrumb" class="breadcrumb level g2-breadcrumb is-hidden-mobile is-hidden-touch"
aria-label="breadcrumbs" aria-label="breadcrumbs"
> >
<div class="level-left"> <div class="level-left">
@ -28,22 +28,8 @@
</div> </div>
</div> </div>
<div class="level-right"> <div class="level-right">
<div <div class="level-item">
class="level-item" <view-mode />
v-show="$route.meta.view === 'list'"
:title="
mode === 'list'
? $t('list.view.gridMode')
: $t('list.view.listMode')
"
@click="changeView"
>
<span class="icon">
<i
:class="'fa' + (mode === 'list' ? ' fa-th' : ' fa-th-list')"
aria-hidden="true"
></i>
</span>
</div> </div>
</div> </div>
</nav> </nav>
@ -51,10 +37,13 @@
</template> </template>
<script> <script>
import { mapState, mapActions } from "vuex"; import ViewMode from "@/layout/viewmode";
import { decode64 } from "@utils/AcrouUtil"; import { decode64 } from "@utils/AcrouUtil";
export default { export default {
props: ["name"], props: ["name"],
components: {
ViewMode,
},
data: function() { data: function() {
return { return {
navs: [], navs: [],
@ -67,11 +56,7 @@ export default {
watch: { watch: {
$route: "render", $route: "render",
}, },
computed: {
...mapState("acrou/view", ["mode"]),
},
methods: { methods: {
...mapActions("acrou/view", ["toggle"]),
go(path) { go(path) {
this.$router.push({ this.$router.push({
path: path, path: path,
@ -114,9 +99,6 @@ export default {
this.navs = navs; this.navs = navs;
} }
}, },
changeView() {
this.toggle(this.mode === "list" ? "grid" : "list");
},
}, },
}; };
</script> </script>

View File

@ -71,6 +71,9 @@
<i class="fab fa-github"></i> <i class="fab fa-github"></i>
</a> </a>
<header-setting /> <header-setting />
<a class="navbar-item is-hidden-desktop" @click.stop="$refs.viewMode.toggleMode">
<view-mode ref="viewMode"/>
</a>
</div> </div>
</div> </div>
</div> </div>
@ -80,10 +83,12 @@
<script> <script>
import headerLocales from "@/layout/header-aside/components/header-locales"; import headerLocales from "@/layout/header-aside/components/header-locales";
import headerSetting from "@/layout/header-aside/components/header-setting"; import headerSetting from "@/layout/header-aside/components/header-setting";
import ViewMode from "@/layout/viewmode";
export default { export default {
components: { components: {
headerLocales, headerLocales,
headerSetting, headerSetting,
ViewMode,
}, },
created() { created() {
this.siteName = document.getElementsByTagName("title")[0].innerText; this.siteName = document.getElementsByTagName("title")[0].innerText;
@ -107,7 +112,8 @@ export default {
currgd: {}, currgd: {},
gds: [], gds: [],
isActive: false, isActive: false,
eyes: '' eyes:
"",
}; };
}, },
methods: { methods: {

View File

@ -85,6 +85,7 @@
v-if="mode !== 'list'" v-if="mode !== 'list'"
:getIcon="getIcon" :getIcon="getIcon"
:go="go" :go="go"
:thum="thum"
/> />
<div <div
v-show="files.length === 0" v-show="files.length === 0"
@ -102,6 +103,24 @@
</div> </div>
<hr /> <hr />
<readmemd :option="readmemd" v-if="readmemd.display"></readmemd> <readmemd :option="readmemd" v-if="readmemd.display"></readmemd>
<viewer
v-if="images && images.length > 0"
:images="images"
class="is-hidden"
ref="viewer"
:options="{ toolbar: true, url: 'data-source' }"
@inited="inited"
>
<img
v-for="image in images"
:src="thum(image.thumbnailLink)"
:data-source="image.path"
:key="image.path"
:alt="image.name"
class="image"
/>
</viewer>
</div> </div>
</template> </template>
@ -185,18 +204,30 @@ export default {
}, },
buildFiles() { buildFiles() {
var path = this.$route.path; var path = this.$route.path;
return this.files.map((item) => { return this.files
var p = path + checkoutPath(item.name, item); .map((item) => {
let isFolder = item.mimeType === "application/vnd.google-apps.folder"; var p = path + checkoutPath(item.name, item);
let size = isFolder ? "-" : formatFileSize(item.size); let isFolder = item.mimeType === "application/vnd.google-apps.folder";
return { let size = isFolder ? "-" : formatFileSize(item.size);
path: p, return {
...item, path: p,
modifiedTime: formatDate(item.modifiedTime), ...item,
size: size, modifiedTime: formatDate(item.modifiedTime),
isFolder: isFolder, size: size,
}; isFolder: isFolder,
}); };
})
.sort((a, b) => {
if (a.isFolder && b.isFolder) {
return 0;
}
return a.isFolder ? -1 : 1;
});
},
images() {
return this.buildFiles.filter(
(file) => file.mimeType.indexOf("image") != -1
);
}, },
}, },
created() { created() {
@ -287,7 +318,19 @@ export default {
}); });
}); });
}, },
thum(url) {
return url ? `/${this.$route.params.id}:view?url=${url}` : "";
},
inited(viewer) {
this.$viewer = viewer;
},
go(file, target) { go(file, target) {
if (file.mimeType.indexOf("image") != -1) {
this.$viewer.show();
// const viewer = this.$el.querySelector(".images").$viewer;
// viewer.show();
return;
}
let path = file.path; let path = file.path;
let cmd = this.$route.params.cmd; let cmd = this.$route.params.cmd;
if (cmd && cmd === "search") { if (cmd && cmd === "search") {

View File

@ -20,31 +20,35 @@
<span class="icon"> <span class="icon">
<i class="fa fa-play-circle" aria-hidden="true"></i> <i class="fa fa-play-circle" aria-hidden="true"></i>
</span> </span>
{{$t('page.video.play')}} / {{ $t("page.video.play") }} /
<span class="icon"> <span class="icon">
<i class="fa fa-download" aria-hidden="true"></i> <i class="fa fa-download" aria-hidden="true"></i>
</span> </span>
{{$t('page.video.download')}} {{ $t("page.video.download") }}
</p> </p>
</header> </header>
<div class="card-content"> <div class="card-content">
<div class="content"> <div class="content">
<div class="field"> <div class="field">
<label class="label">{{$t('page.video.link')}}</label> <label class="label">{{ $t("page.video.link") }}</label>
<div class="control"> <div class="control">
<input class="input" type="text" :value="videourl" /> <input class="input" type="text" :value="videourl" />
</div> </div>
</div> </div>
<div class="columns is-mobile is-multiline has-text-centered"> <div class="columns is-mobile is-multiline has-text-centered">
<div class="column" v-for="(item,index) in players" v-bind:key="index"> <div
class="column"
v-for="(item, index) in players"
v-bind:key="index"
>
<p class="heading"> <p class="heading">
<a :href="item.scheme+(item.name==='Thunder'?getThunder:videourl)"> <a :href="item.scheme">
<figure class="image is-48x48" style="margin: 0 auto;"> <figure class="image is-48x48" style="margin: 0 auto;">
<img class="icon" :src="item.icon" /> <img class="icon" :src="item.icon" />
</figure> </figure>
</a> </a>
</p> </p>
<p class>{{item.name}}</p> <p class>{{ item.name }}</p>
</div> </div>
</div> </div>
</div> </div>
@ -56,65 +60,87 @@
<script> <script>
import { decode64 } from "@utils/AcrouUtil"; import { decode64 } from "@utils/AcrouUtil";
export default { export default {
data: function () { data: function() {
return { return {
apiurl: "", apiurl: "",
videourl: "", videourl: "",
players: [
{
name: "IINA",
icon: "https://www.iina.io/images/iina-icon-60.png",
scheme: "iina://weblink?url="
},
{
name: "PotPlayer",
icon: "https://cloud.jsonpop.cn/go2index/player/potplayer.png",
scheme: "potplayer://"
},
{
name: "VLC",
icon: "https://cloud.jsonpop.cn/go2index/player/vlc.png",
scheme: "vlc://"
},
{
name: "Thunder",
icon: "https://cloud.jsonpop.cn/go2index/player/thunder.png",
scheme: "thunder://"
},
{
name: "MXPlayer",
icon: "https://cloud.jsonpop.cn/go2index/player/mxplayer.png",
scheme: "intent:"
},
{
name: "nPlayer",
icon: "https://cloud.jsonpop.cn/go2index/player/nplayer.png",
scheme: "nplayer-"
}
]
}; };
}, },
methods: { methods: {
render () { render() {
// 便 // 便
this.videourl = window.location.origin + encodeURI(this.url); this.videourl = window.location.origin + encodeURI(this.url);
this.apiurl = this.apiurl =
"https://api.jsonpop.cn/demo/blplyaer/?url=" + this.videourl; "https://api.jsonpop.cn/demo/blplyaer/?url=" + this.videourl;
} },
}, },
activated () { activated() {
this.render(); this.render();
}, },
computed: { computed: {
url () { url() {
if (this.$route.params.path) { if (this.$route.params.path) {
return decode64(this.$route.params.path); return decode64(this.$route.params.path);
} }
return '' return "";
}, },
getThunder () { players() {
return [
{
name: "IINA",
icon: "https://www.iina.io/images/iina-icon-60.png",
scheme: "iina://weblink?url=" + this.videourl,
},
{
name: "PotPlayer",
icon: "https://cloud.jsonpop.cn/go2index/player/potplayer.png",
scheme: "potplayer://" + this.videourl,
},
{
name: "VLC",
icon: "https://cloud.jsonpop.cn/go2index/player/vlc.png",
scheme: "vlc://" + this.videourl,
},
{
name: "Thunder",
icon: "https://cloud.jsonpop.cn/go2index/player/thunder.png",
scheme: "thunder://" + this.getThunder,
},
{
name: "Aria2",
icon: "https://cloud.jsonpop.cn/go2index/player/aria2.png",
scheme: 'javascript:alert("暂未实现")',
},
{
name: "nPlayer",
icon: "https://cloud.jsonpop.cn/go2index/player/nplayer.png",
scheme: "nplayer-" + this.videourl,
},
{
name: "MXPlayer(Free)",
icon: "https://cloud.jsonpop.cn/go2index/player/mxplayer.png",
scheme:
"intent:" +
this.videourl +
"#Intent;package=com.mxtech.videoplayer.ad;S.title=" +
this.title +
";end",
},
{
name: "MXPlayer(Pro)",
icon: "https://cloud.jsonpop.cn/go2index/player/mxplayer.png",
scheme:
"intent:" +
this.videourl +
"#Intent;package=com.mxtech.videoplayer.pro;S.title=" +
this.title +
";end",
},
];
},
getThunder() {
return Buffer.from("AA" + this.videourl + "ZZ").toString("base64"); return Buffer.from("AA" + this.videourl + "ZZ").toString("base64");
} },
} },
}; };
</script> </script>

View File

@ -3,16 +3,11 @@
<div class="columns is-multiline"> <div class="columns is-multiline">
<div <div
class="column is-one-quarter" class="column is-one-quarter"
v-for="(file, index) in imgData" v-for="(file, index) in folders"
:key="index" :key="'folder_' + index"
@click=" @click="go(file)"
go(
file,
file.mimeType !== 'application/vnd.google-apps.folder' ? 'view' : ''
)
"
> >
<div class="card g2-view-box g2-grid-folder" v-if="file.isFolder"> <div class="card g2-grid-view-card g2-grid-view-folder">
<div class="media"> <div class="media">
<div class="content"> <div class="content">
<svg class="iconfont" aria-hidden="true"> <svg class="iconfont" aria-hidden="true">
@ -22,12 +17,21 @@
</div> </div>
</div> </div>
</div> </div>
<div class="card g2-view-box" v-if="!file.isFolder"> </div>
</div>
<div class="columns is-multiline">
<div
class="column is-one-quarter"
v-for="(file, index) in files"
:key="'file_' + index"
@click="go(file, 'view')"
>
<div class="card g2-grid-view-card">
<div class="card-image"> <div class="card-image">
<figure class="image is-square"> <figure class="image is-square">
<img <img
v-if="file.thumbnailLink" v-if="file.thumbnailLink"
:src="thum(file.thumbnailLink)" v-lazy="thum(file.thumbnailLink)"
alt="Placeholder image" alt="Placeholder image"
/> />
</figure> </figure>
@ -60,22 +64,22 @@ export default {
go: { go: {
type: Function, type: Function,
}, },
thum: {
type: Function,
},
}, },
data: function() { data: function() {
return {}; return {};
}, },
computed: { computed: {
imgData() { folders() {
// return this.data.filter((file) => file.mimeType.indexOf("image/") != -1); return this.data.filter((item) => item.isFolder);
// return this.data.filter((file) => !file.isFolder); },
return this.data; files() {
}, return this.data.filter((item) => !item.isFolder);
},
methods: {
thum(url) {
return url ? `/${this.$route.params.id}:view?url=${url}` : "";
}, },
}, },
methods: {},
}; };
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@ -95,14 +99,14 @@ export default {
.column { .column {
cursor: pointer; cursor: pointer;
} }
.g2-view-box { .g2-grid-view-card {
box-shadow: 0 4px 10px rgba(0, 0, 0, 0.05), 0 0 1px rgba(0, 0, 0, 0.1); box-shadow: 0 4px 10px rgba(0, 0, 0, 0.05), 0 0 1px rgba(0, 0, 0, 0.1);
border-radius: 1rem; border-radius: 1rem;
&:hover { &:hover {
box-shadow: 0 4px 10px rgba(0, 0, 0, 0.2), 0 0 1px rgba(0, 0, 0, 0.05); box-shadow: 0 4px 10px rgba(0, 0, 0, 0.2), 0 0 1px rgba(0, 0, 0, 0.05);
} }
} }
.g2-grid-folder { .g2-grid-view-folder {
padding: 10px; padding: 10px;
} }
.iconfont { .iconfont {