单页面微信分享(html+vue)
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
<meta http-equiv="pragma" content="no-cache">
<meta http-equiv="cache-control" content="no-cache">
<meta http-equiv="expires" content="0">
<link rel="icon" type="image/x-icon" href="./static/favicon.ico">
<title></title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/vant@2.12/lib/index.css" />
<style>
.body-container {
background: rgba(0, 35, 116, 1);
}
.flex {
display: flex;
}
.flex-wrap-between {
display: flex;
flex-wrap: wrap;
justify-content: space-between;
}
/* 嘉宾信息 */
.guest-info-wrap {
padding: .95rem .4rem 0 .4rem;
padding-bottom: .95rem;
position: relative;
}
.guest-info-wrap .guest-info-box {
background: rgba(13, 44, 131, 0.8);
box-shadow: 0 0 4px 0 rgba(20, 13, 109, 0.5), 0 0 5px 0 rgba(37, 240, 255, 0.32);
border-radius: .12rem;
border: 1px solid #4778FF;
padding: 0 .45rem;
position: relative;
/* padding-top: .87rem; */
/* margin-bottom: .95rem; */
max-height: 14.44rem;
overflow-y: scroll;
scrollbar-width: none;
/* Firefox */
}
.guest-info-wrap .guest-info-box::-webkit-scrollbar {
display: none;
/* Chrome Safari */
}
.guest-info-wrap::before {
content: '';
width: 6.08rem;
height: .63rem;
background: url('https://eventimg.oss-cn-shenzhen.aliyuncs.com/kabrita/live_pic_word_info.png') center center;
background-size: 100% 100%;
position: absolute;
top: .68rem;
left: 0;
right: 0;
margin: auto;
z-index: 9;
}
.guest-info-wrap .guest-info-box .guest-info-item {
padding-bottom: .4rem;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-title {
font-size: .24rem;
font-family: PingFangSC-Medium, PingFang SC;
font-weight: 500;
color: #FFFFFF;
line-height: .32rem;
padding-bottom: .22rem;
position: relative;
display: inline-block;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont {
background: #1D3DA0;
border-radius: .08rem;
padding: .38rem .32rem .16rem;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .sub-title {
font-size: .22rem;
font-family: PingFangSC-Regular, PingFang SC;
font-weight: 400;
color: rgba(255, 255, 255, 0.8);
line-height: .32rem;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .sub-img-box {
width: 100%;
/* height: 2.9rem; */
margin: .12rem auto .16rem;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .sub-img-box .sub-img {
display: block;
width: 100%;
height: 100%;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .share-info {
padding: .16rem 0;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .share-info .date {
font-size: .22rem;
font-family: HelveticaNeue;
color: rgba(255, 255, 255, 0.8);
line-height: .32rem;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .share-info .share-btn {
font-size: .22rem;
font-family: PingFangSC-Medium, PingFang SC;
font-weight: 500;
color: #47F1FF;
line-height: .32rem;
position: relative;
}
.guest-info-wrap .guest-info-box .guest-info-item .info-item-cont .share-info .share-btn::after {
content: '';
width: .26rem;
height: .26rem;
background: url('https://eventimg.oss-cn-shenzhen.aliyuncs.com/kabrita/share_icon.png') center center;
background-size: 100% 100%;
display: inline-block;
transform: translateY(2px);
}
.van-overlay {
display: flex;
justify-content: center;
align-items: center;
z-index: 20;
/* overflow-y: scroll;
padding: 1rem 0;
box-sizing: border-box; */
}
.share-pic-word {
width: 100%;
/* height: 100%; */
/* overflow: scroll; */
/* padding: 1rem 0; */
/* box-sizing: border-box; */
}
/* 图文分享弹框 */
.share-pic-word .share-pic-word-cont {
width: 89%;
/* max-height: 8.94rem;
overflow-y: scroll; */
margin: auto;
background: #0B2672;
/* box-shadow: 0px 0px 4px 0px rgba(20, 13, 109, 0.5), 0px 0px 5px 0px rgba(37, 240, 255, 0.32); */
/* border-radius: .12rem; */
/* border: 1px solid #4778FF; */
box-sizing: border-box;
padding: .4rem .45rem .3rem;
}
.share-pic-word .share-pic-word-cont .share-pic-word-title {
font-size: .24rem;
font-family: PingFangSC-Medium, PingFang SC;
font-weight: 500;
color: #FFFFFF;
line-height: .32rem;
}
.share-pic-word .share-pic-word-cont .time-box {
font-size: .24rem;
font-family: PingFangSC-Regular, PingFang SC;
font-weight: 400;
color: #FFFFFF;
line-height: .24rem;
position: relative;
padding: .24rem 0 .14rem;
}
.share-pic-word .share-pic-word-cont .time-box img {
display: block;
width: .24rem;
height: .24rem;
margin-right: .1rem;
}
.share-pic-word .share-pic-word-cont .share-pic-word-sub-title {
font-size: .22rem;
font-family: PingFangSC-Regular, PingFang SC;
font-weight: 400;
color: rgba(255, 255, 255, 0.8);
line-height: .32rem;
padding-bottom: .12rem;
}
.share-pic-word .share-pic-word-cont .sub-img-box {
width: 100%;
/* height: 3.26rem; */
}
.share-pic-word .share-pic-word-cont .sub-img-box .sub-img {
width: 100%;
height: 100%;
display: block;
}
.share-pic-word .share-pic-word-cont .scan-code-box {
padding: .21rem .31rem;
align-items: center;
display: flex;
}
.share-pic-word .share-pic-word-cont .scan-code-box span {
font-size: .34rem;
font-family: Alibaba-PuHuiTi-H, Alibaba-PuHuiTi;
font-weight: normal;
color: #FFFFFF;
line-height: 1;
letter-spacing: 1px;
text-shadow: 0px 1px 2px rgba(0, 0, 0, 0.5);
}
.share-pic-word .share-pic-word-cont .scan-code-box img {
width: 1.62rem;
height: 1.62rem;
display: block;
border: 5px solid #fff;
box-sizing: border-box;
}
.share-pic-word .save-btn {
font-size: .28rem;
font-family: SourceHanSansCN-Medium, SourceHanSansCN;
font-weight: 500;
color: #FFFFFF;
line-height: 1;
letter-spacing: 2px;
margin-top: .32rem;
display: block;
text-align: center;
}
.poster-img {
display: block;
width: 89%;
border-radius: .12rem;
border: 1px solid #4778FF;
margin: auto;
}
.no-before::before {
display: none !important;
}
.body-embed {
padding-bottom: 0 !important;
}
.body-embed::after {
display: none !important;
}
#qrcode {
width: 1.62rem;
height: 1.62rem;
display: block;
border: 5px solid #fff;
box-sizing: border-box;
overflow: hidden;
}
#qrcode img {
display: block !important;
width: 100% !important;
height: 100% !important;
}
.poster-img-box {
overflow: scroll;
}
.share-mask {
display: flex;
justify-content: center;
align-items: center;
z-index: 20;
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, .7);
}
.lock-body {
overflow: hidden !important;
}
[v-cloak] {
display: none;
}
</style>
</head>
<body class="body-container">
<div id="app" v-cloak>
<div class="guest-info-wrap">
<ul class="guest-info-box">
<li class="guest-info-item" v-for="(item, index) in picWordList" :key="index">
<div class="info-item-title">{{ item.graphicInformationTitle }}</div>
<div class="info-item-cont">
<div class="sub-title">
{{ item.graphicInformationDetailContent }}
</div>
<div class="sub-img-box">
<img :src="item.graphicInformationPicture" v-if="item.graphicInformationPicture" alt="" class="sub-img">
</div>
<div class="share-info flex-wrap-between">
<span class="date">{{ item.createDate }}</span>
<a href="javascript:;" class="share-btn" @click="showShareInfo(index)">
图文分享
</a>
</div>
</div>
</li>
</ul>
</div>
<!-- 图文分享弹框 -->
<div class="share-mask" v-if="shareShow" @click="shareShow = false">
<div class="share-pic-word" @click.stop>
<div class="poster-img-box" v-if="posterImg" :style="{maxHeight: posterBoxMaxH}">
<img :src="posterImg" class="poster-img" alt="">
</div>
<div class="share-pic-word-cont" v-else>
<div class="share-pic-word-title van-multi-ellipsis--l2">
{{ sharePicWord.graphicInformationTitle }}
</div>
<div class="time-box flex">
<img :src="timeIcon" alt="">
{{ sharePicWord.createDate }}
</div>
<div class="share-pic-word-sub-title">
{{ sharePicWord.graphicInformationDetailContent }}
</div>
<div class="sub-img-box">
<img :src="sharePicWord.graphicInformationPicture" v-if="sharePicWord.graphicInformationPicture" class="sub-img"
alt="" @load="afterLoad">
</div>
<div class="scan-code-box flex-wrap-between" v-show="JSON.stringify(sharePicWord) !== '{}'">
<span>更多快讯请扫码查看</span>
<img :src="shareLinkImg" alt="" crossorigin="anonymous" @load="afterLoad">
</div>
</div>
<a href="javascript:;" class="save-btn">长按图片发送给好友</a>
</div>
</div>
</div>
</body>
<script src="https://res2.wx.qq.com/open/js/jweixin-1.6.0.js"></script>
<script src="https://unpkg.com/axios/dist/axios.min.js"></script>
<script src="https://unpkg.com/vue/dist/vue.js"></script>
<script src="https://cdn.jsdelivr.net/npm/vant@2.10/lib/vant.min.js"></script>
<script src="https://cdn.bootcss.com/html2canvas/0.5.0-beta4/html2canvas.js"></script>
<!-- <script src="https://cdn.bootcss.com/vConsole/3.3.0/vconsole.min.js"></script> -->
<script>
// new VConsole();
(function(w) {
var resizeEvt =
"orientationchange" in window ?
"orientationchange" :
"resize";
var docEl = document.documentElement;
function resizeFn() {
var docW = docEl.clientWidth;
if (docW > w) {
docW = w;
}
docEl.style.fontSize = (docW * 100) / w + "px";
}
resizeFn();
window.addEventListener(resizeEvt, resizeFn);
})(750);
var showLoading = options => {
toastInstance = vant.Toast.loading(options ? options : {
message: '玩命加载中...',
forbidClick: true,
duration: 0, // 持续展示 toast
});
}
// 隐藏loading
var hideLoading = () => {
toastInstance && toastInstance.clear();
}
var axiosGet = (url, params) => {
return new Promise((resolve, reject) => {
axios.get(url, params).then(res => {
resolve(res.data)
}).catch(err => {
reject(err)
})
})
}
// 使用axios的post方法
var axiosPost = (url, params) => {
return new Promise((resolve, reject) => {
let isUploadFile = url.indexOf('uploadFile') !== -1;
let temp = params;
if (isUploadFile) {
temp = new FormData();
temp.append('file', params.file);
temp.append('name', params.file.name);
temp.append('type', params.type);
}
axios.post(url, temp).then(res => {
resolve(res.data)
}).catch(err => {
reject(err)
})
})
}
// api地址
const APP_API_HOST = 'https://api-dev.tuoluohuodong.com/api';
// const APP_API_HOST = 'https://api-prod.tuoluohuodong.com/api';
// 分享图文快讯的时间图标
const timeIcon =
''
// 图片链接前缀
const APP_IMG_HOST = 'https://eventimg.oss-cn-shenzhen.aliyuncs.com/kabrita';
var getGraphicList = params => axiosPost(`${APP_API_HOST}/h5/graphic/information/share/list`, params);
var fileToBase64 = params => axiosPost(`${APP_API_HOST}/file/common/file/to/base64`, params);
var getGraphicShare = params => axiosGet(`${APP_API_HOST}/h5/graphic/information/share/${params}`); //微信分享接口
const indexPage = window.location.hre;
const appId = APP_API_HOST.indexOf('prod') !== -1 ? 'wxfb54b0a71143dd15' : 'wxaf0e73a687201e7e';
const wxShare = function(title, desc, link, imgUrl) {
axios({
method: 'post',
headers: {
'Content-Type': 'application/json; charset=utf-8'
},
url: APP_API_HOST + '/common/wx/config',
data: JSON.stringify({
'url': encodeURIComponent(link),
appId
})
}).then(res => {
if (res && res.data.code === 0) {
let data = res.data.data;
wx.config({
//debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
appId: data.appId, // 必填,公众号的唯一标识
timestamp: parseInt(data.timestamp), // 必填,生成签名的时间戳
nonceStr: data.nonceStr, // 必填,生成签名的随机串
signature: data.signature, // 必填,签名,见附录1
jsApiList: [
'onMenuShareTimeline',
'onMenuShareAppMessage',
'onMenuShareQQ',
'onMenuShareWeibo',
'onMenuShareQZone',
] // 必填,需要使用的JS接口列表,所有JS接口列表见附录2
});
wx.ready(function() {
wx.updateAppMessageShareData({
title: title, // 分享标题
desc: desc, // 分享描述
link: link, // 分享链接
imgUrl: imgUrl, // 分享图标
success: function() {
// 用户确认分享后执行的回调函数
},
cancel: function() {
// 用户取消分享后执行的回调函数
},
});
wx.onMenuShareAppMessage({
title: title, // 分享标题
desc: desc, // 分享描述
link: link, // 分享链接
imgUrl: imgUrl, // 分享图标
type: 'link', // 分享类型,music、video或link,不填默认为link
dataUrl: '', // 如果type是music或video,则要提供数据链接,默认为空
success: function() {
// 用户确认分享后执行的回调函数
},
cancel: function() {
// 用户取消分享后执行的回调函数
}
});
wx.onMenuShareQQ({
title: title, // 分享标题
desc: desc, // 分享描述
link: link, // 分享链接
imgUrl: imgUrl, // 分享图标
success: function() {
// 用户确认分享后执行的回调函数
},
cancel: function() {
// 用户取消分享后执行的回调函数
}
});
wx.onMenuShareWeibo({
title: title, // 分享标题
desc: desc, // 分享描述
link: link, // 分享链接
imgUrl: imgUrl, // 分享图标
success: function() {
// 用户确认分享后执行的回调函数
},
cancel: function() {
// 用户取消分享后执行的回调函数
}
});
wx.onMenuShareQZone({
title: title, // 分享标题
desc: desc, // 分享描述
link: link, // 分享链接
imgUrl: imgUrl, // 分享图标
success: function() {
// 用户确认分享后执行的回调函数
},
cancel: function() {
// 用户取消分享后执行的回调函数
}
});
});
}
}).catch((err) => {
console.log(err, 'err');
});
}
new Vue({
el: '#app',
data: {
logo: `${APP_IMG_HOST}/kabrita.png`,
slogan: `${APP_IMG_HOST}/kabrita_slogan.png`,
avatorInfoBg: `${APP_IMG_HOST}/avator_info_bg.png`,
startPosition: 0,
picWordList: [],
shareShow: false,
sharePicWord: {},
page: 1,
pageSize: 10,
totalPage: 1,
shareLinkImg: `${APP_IMG_HOST}/template.png`,
timeIcon,
posterImg: '',
graphicInfoPicture: '',
curIndex: 0, // 当前查看的图文
shareLink: 'https://tuoluohuodong.com/',
shareData: {}, //分享相关数据
graphicInformationId: '', //id
posterBoxMaxH: `${document.body.clientHeight + 520}px`
},
mounted() {
console.log(window.location.href, '路径')
console.log(location.href.split('#')[0]);
this.getGraphicList()
this.$nextTick(() => {
document.body.className = 'body-container body-box';
// 监听滚动事件
let scrollTargetBox = document.querySelector('.guest-info-box');
scrollTargetBox.onscroll = () => {
var scrollHeight = scrollTargetBox.scrollHeight; //251
var scrollTop = scrollTargetBox.scrollTop; //0-18
var clientHeight = scrollTargetBox.clientHeight; //233
if (scrollHeight - clientHeight == scrollTop) {
//滚动条滚到最底部
if (this.page >= this.totalPage) return
this.page++;
this.getGraphicList()
}
}
})
},
watch: {
'shareShow': {
handler(newVal, oldVal) {
if (!newVal) {
this.posterImg = '';
this.sharePicWord = {};
this.shareLinkImg = `${APP_IMG_HOST}/template.png`;
this.$nextTick(() => {
document.body.className = 'body-container body-box';
})
hideLoading();
}
}
}
},
methods: {
showShareInfo(index) {
this.curIndex = index;
this.sharePicWord = JSON.parse(JSON.stringify(this.picWordList[index]));
let image = new Image();
image.src =
`https://api.qrserver.com/v1/create-qr-code?data=${this.sharePicWord.shareLink}&bgcolor=255-255-255&color=000000`;
image.onload = () => {
this.shareLinkImg = image.src;
}
this.generatePoster();
this.$nextTick(() => {
this.shareShow = true;
document.body.className = 'body-container body-box lock-body';
let timer = setTimeout(() => {
clearTimeout(timer)
document.querySelector('.share-pic-word-cont').scrollTop = 0;
}, 0);
})
},
// 生成海报
generatePoster() {
showLoading();
if (this.sharePicWord.graphicInformationPicture && this.sharePicWord.graphicInformationPicture.indexOf('http') !==
-1) {
fileToBase64({
fileUrl: this.sharePicWord.graphicInformationPicture
}).then(res => {
let {
code,
data
} = res;
if (code === 0 && data) {
this.sharePicWord.graphicInformationPicture = `data:image/png;base64,${data}`
this.picWordList[this.curIndex].graphicInformationPicture = `data:image/png;base64,${data}`
}
}).catch(() => {})
}
},
// 图片加载完毕再生成海报
afterLoad() {
// return
// 图片已经转成base64格式
if (this.shareShow && this.sharePicWord.graphicInformationPicture && this.sharePicWord.graphicInformationPicture
.indexOf('http') === -1 && this.shareLinkImg !== `${APP_IMG_HOST}/template.png`) {
let shareContent = document.querySelector('.share-pic-word-cont');
let width = shareContent.offsetWidth // 获取dom 宽度
let height = shareContent.offsetHeight // 获取dom 高度
let canvas = document.createElement('canvas') // 创建一个canvas节点
var scale = 4 // 定义任意放大倍数 支持小数
canvas.width = Math.floor(width * scale) // 定义canvas 宽度 * 缩放
canvas.height = Math.floor(height * scale) // 定义canvas高度 *缩放
var rect = shareContent.getBoundingClientRect() // 获取元素相对于视口的
canvas.getContext('2d').scale(scale, scale) // 获取context,设置scale
let scrollTop = document.documentElement.scrollTop || document.body.scrollTop // 如果页面根元素有滚动的话,否则可以忽略此项
let scrollLeft = document.documentElement.scrollLeft || document.body.scrollLeft
// canvas.getContext('2d').translate(-rect.left - options.scrollX, -rect.left - options.scrollY);
canvas.getContext('2d').translate(-rect.left, -rect.top);
html2canvas(document.querySelector('.share-pic-word-cont'), {
logging: false, //日志开关,便于查看html2canvas的内部执行流程
scrollY: 0,
scrollX: 0,
useCORS: true, // 【重要】开启跨域配置
canvas,
background: "#ffffff", // 一定要添加背景颜色,否则出来的图片,背景全部都是透明的
}).then(canvas => {
hideLoading();
this.posterImg = canvas.toDataURL("image/png");
})
}
},
//获取链接参数
getRequest() {
var url = window.location.href;
var theRequest = new Object();
if (url.indexOf("?") != -1) {
var str = url.split("?")[1];
var strs = str.split("&");
for (var i = 0; i < strs.length; i++) {
theRequest[strs[i].split("=")[0]] = unescape(strs[i].split("=")[1]);
}
}
return theRequest;
},
// 获取图文快讯分页
getGraphicList() {
showLoading();
let {
page,
pageSize
} = this;
getGraphicList({
// graphicInformationId: this.getRequest()['id'],//获取链接里的id
graphicInformationId:"bj00N9qfFwM8WGzxK3FuKAkFvwtDcPnQ",
page,
pageSize
}).then(res => {
hideLoading();
let {
code,
data
} = res;
if (code === 0 && data && data.list && data.list.length) {
this.picWordList = [...this.picWordList, ...data.list];
this.totalPage = data.pagination.totalPage;
this.shareOut();
}
}).catch(() => {})
// },
},
// 获取微信分享
shareOut() {
// let graphicInformationId = this.picWordList[0].graphicInformationId;//获取列表里的id
let graphicInformationId = '2509165008675733504'
getGraphicShare(graphicInformationId).then(res => {
let {
code,
data
} = res;
if (code === 0 && data) {
this.shareData = data;
wxShare(data.shareTitle, data.shareSign, window.location.href, data.sharePic || "https://tuoluohuodong.oss-cn-shenzhen.aliyuncs.com/digital_system/tuoluoicon.png");
}
}).catch(() => {})
},
}
})
</script>
</html>