vuepress-theme-reco/components/FriendLink.vue
2020-01-21 16:15:36 +08:00

197 lines
5.1 KiB
Vue

<template>
<div class="friend-link-wrapper">
<div
class="friend-link-item"
v-for="(item, index) in dataAddColor"
:key="index"
@mouseenter="showDetail($event)"
@mouseleave="hideDetail($event)"
target="_blank">
<span
class="list-style"
:style="{ 'backgroundColor': item.color }">
</span>
{{item.title}}
<transition name="fade">
<div class="popup-window-wrapper">
<div
class="popup-window"
:style="popupWindowStyle"
ref="popupWindow">
<div class="logo">
<img :src="getImgUrl(item)" />
</div>
<div class="info">
<div class="title">
<h4>{{ item.title }}</h4>
<a
class="btn-go"
:style="{ 'backgroundColor': item.color }"
:href="item.link"
target="_blank">GO</a>
</div>
<p v-if="item.desc">{{ item.desc }}</p>
</div>
</div>
</div>
</transition>
</div>
</div>
</template>
<script>
import md5 from 'md5'
import { getOneColor } from '@theme/helpers/other'
export default {
data () {
return {
popupWindowStyle: {}
}
},
computed: {
dataAddColor () {
let { friendLink } = this.$themeConfig
if (friendLink && friendLink.length > 0) {
friendLink = friendLink.map(item => ({
...item,
color: getOneColor()
}))
return friendLink
}
return []
}
},
methods: {
getMd5 (str) {
return md5(str)
},
showDetail (e) {
const currentDom = e.target
const popupWindowWrapper = currentDom.querySelector('.popup-window-wrapper')
const popupWindow = currentDom.querySelector('.popup-window')
popupWindowWrapper.style.display = 'block'
const { clientWidth } = currentDom
const {
clientWidth: windowWidth,
clientHeight: windowHeight
} = popupWindow
this.popupWindowStyle = {
left: (clientWidth - windowWidth) / 2 + 'px',
top: -windowHeight + 'px'
}
this.$nextTick(() => {
this._adjustPosition(currentDom.querySelector('.popup-window'))
})
},
hideDetail (e) {
const currentDom = e.target
currentDom.querySelector('.popup-window-wrapper').style.display = 'none'
},
getImgUrl (info) {
const { logo, email } = info
if (logo && /^http/.test(logo)) return logo
if (logo && !/^http/.test(logo)) return this.$withBase(logo)
return `//1.gravatar.com/avatar/${this.getMd5(email || '')}?s=50&amp;d=mm&amp;r=x`
},
_adjustPosition (dom) {
const { offsetWidth } = document.body
const { x, width } = dom.getBoundingClientRect()
const distanceToRight = offsetWidth - (x + width)
if (distanceToRight < 0) {
const { offsetLeft } = dom
this.popupWindowStyle = {
...this.popupWindowStyle,
left: offsetLeft + distanceToRight + 'px'
}
}
}
}
}
</script>
<style lang="stylus" scoped>
@require '../styles/mode.styl'
.friend-link-wrapper
position relative
margin 30px 0
.friend-link-item
position relative
vertical-align: middle;
margin: 4px 4px 10px;
padding: 4px 8px 4px 20px;
line-height 20px
display: inline-block;
cursor: pointer;
border-radius: $borderRadius
font-size: 13px;
box-shadow var(--box-shadow)
transition: all .5s
.list-style
position absolute
left .4rem
top 0
bottom 0
margin auto
display block
width .4rem
height .4rem
border-radius .1rem
background $accentColor
content ''
.popup-window-wrapper
display none
.popup-window
position absolute
display flex
background var(--background-color)
box-shadow var(--box-shadow)
border-radius $borderRadius
box-sizing border-box
padding .8rem 1rem
width 300px
.logo
margin-right .4rem
width 2rem
height 2rem
flex 0 0 2rem
border-radius $borderRadius
overflow hidden
img
width 2rem
height 2rem
.info
flex 0 0 85%
width 85%
.title
display flex
align-items center
justify-content space-between
height 2rem
h4
margin .2rem 0
flex 0 0 86%
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
.btn-go
width 1.4rem
height 1.2rem
border-radius $borderRadius
font-size .1rem
color #ffffff
text-align center
line-height 1.2rem
cursor pointer
transition all .5s
&:hover
transform scale(1.1)
.fade-enter-active, .fade-leave-active
transition opacity .5s
.fade-enter, .fade-leave-to
opacity 0
</style>