jormungandr-bite/src/client/components/link.vue

92 lines
1.9 KiB
Vue
Raw Normal View History

2020-02-09 11:48:45 -07:00
<template>
2020-02-10 04:51:17 -07:00
<component :is="self ? 'router-link' : 'a'" class="xlcxczvw _link" :[attr]="self ? url.substr(local.length) : url" :rel="rel" :target="target"
2020-02-09 11:48:45 -07:00
@mouseover="onMouseover"
@mouseleave="onMouseleave"
:title="url"
>
<slot></slot>
<fa :icon="faExternalLinkSquareAlt" v-if="target === '_blank'" class="icon"/>
</component>
</template>
<script lang="ts">
import Vue from 'vue';
import { faExternalLinkSquareAlt } from '@fortawesome/free-solid-svg-icons';
import { url as local } from '../config';
2020-03-21 23:38:33 -06:00
import MkUrlPreview from './url-preview-popup.vue';
import { isDeviceTouch } from '../scripts/is-device-touch';
2020-02-09 11:48:45 -07:00
export default Vue.extend({
props: {
url: {
type: String,
required: true,
},
rel: {
type: String,
required: false,
}
},
data() {
2020-02-10 04:51:17 -07:00
const self = this.url.startsWith(local);
2020-02-09 11:48:45 -07:00
return {
local,
2020-02-10 04:51:17 -07:00
self: self,
attr: self ? 'to' : 'href',
target: self ? null : '_blank',
2020-02-09 11:48:45 -07:00
showTimer: null,
hideTimer: null,
preview: null,
faExternalLinkSquareAlt
};
},
methods: {
showPreview() {
if (!document.body.contains(this.$el)) return;
if (this.preview) return;
2020-03-21 23:38:33 -06:00
this.preview = new MkUrlPreview({
2020-02-09 11:48:45 -07:00
parent: this,
propsData: {
url: this.url,
source: this.$el
}
}).$mount();
document.body.appendChild(this.preview.$el);
},
closePreview() {
if (this.preview) {
this.preview.destroyDom();
this.preview = null;
}
},
onMouseover() {
if (isDeviceTouch()) return;
2020-02-09 11:48:45 -07:00
clearTimeout(this.showTimer);
clearTimeout(this.hideTimer);
this.showTimer = setTimeout(this.showPreview, 500);
},
onMouseleave() {
if (isDeviceTouch()) return;
2020-02-09 11:48:45 -07:00
clearTimeout(this.showTimer);
clearTimeout(this.hideTimer);
this.hideTimer = setTimeout(this.closePreview, 500);
}
}
});
</script>
<style lang="scss" scoped>
.xlcxczvw {
word-break: break-all;
> .icon {
padding-left: 2px;
font-size: .9em;
font-weight: 400;
font-style: normal;
}
}
</style>