以下是个人封装的一个组件希望对大家有所帮助
示例:

封装过程如下:
我们先在小程序里面新建component文件夹用来存放公共组件

如大家所见,里面已经新建了popup文件夹及其相关文件
首先
1、popup.wxml
<view class="wx-popup" hidden="{{flag}}">
<view class='popup-container'>
<view class="wx-popup-con">{{content}}</view>
<view class="wx-popup-btn">
<text class="btn-no" bindtap='_error'>{{btn_no}}</text>
<text class="btn-ok" style="border-right:0;color: #FFAA01;" bindtap='_success'>{{btn_ok}}</text>
</view>
</view>
</view>
2、popup.wxss
/* pages/actiondetail/index.wxss */
page{
height: 100%;
}
.wx-popup {
position: absolute;
left: 0;
top: 0;
100%;
height: 100%;
background: rgba(0, 0, 0, .5);
z-index: 999;
}
.popup-container {
position: absolute;
left: 50%;
top: 50%;
height: 368rpx;
background: #FFFFFF;
border-radius: 20rpx;
80%;
max- 600rpx;
box-sizing: bordre-box;
transform: translate(-50%, -50%);
overflow: hidden;
}
.wx-popup-con {
height: 260rpx;
display: flex;
justify-content: center;
align-items: center;
100%;
font-size: 32rpx;
font-weight: 600;
color: #333333;
}
.wx-popup-btn {
height: calc(100% - 260rpx);
100%;
display: flex;
}
.wx-popup-btn text{
display: flex;
align-items: center;
justify-content: center;
50%;
height: 100%;
border-top: 1rpx solid #E6E6E6;
border-right: 1rpx solid #E6E6E6;
color: #999999;
font-size: 32rpx;
font-weight: 600;
}
3、popup.js
Component({
options: {
multipleSlots: true // 在组件定义时的选项中启用多slot支持
},
/**
* 组件的属性列表
*/
properties: {
// title: { // 属性名
// type: String, // 类型(必填),目前接受的类型包括:String, Number, Boolean, Object, Array, null(表示任意类型)
// value: '标题' // 属性初始值(可选),如果未指定则会根据类型选择一个
// },
// 弹窗内容
content: {
type: String,
value: '内容'
},
// 弹窗取消按钮文字
btn_no: {
type: String,
value: '取消'
},
// 弹窗确认按钮文字
btn_ok: {
type: String,
value: '确定'
}
},
/**
* 组件的初始数据
*/
data: {
flag: true,
},
/**
* 组件的方法列表
*/
methods: {
//隐藏弹框
hidePopup: function () {
this.setData({
flag: !this.data.flag
})
},
//展示弹框
showPopup () {
this.setData({
flag: !this.data.flag
})
},
/*
* 内部私有方法建议以下划线开头
* triggerEvent 用于触发事件
*/
_error () {
//触发取消回调
this.triggerEvent("error")
},
_success () {
//触发成功回调
this.triggerEvent("success");
}
}
})
公共组件我们已经写好了 ,接下来就是如何使用
1、大家根据自己实际情况引用组件哦,我这里是在date这个页面引入,就当做例子示范了
先配置我们的json文件(usingComponents)就是将组件引入到这个页面

2、在页面使用他( date.wxml)
<popup id='popup'
content='是否要安排当日训练计划日程'
btn_no='否'
btn_ok='是'
bind:error="_error"
bind:success="_success">
</popup>
date.js
this.popup.showPopup() ----- 这个是判断弹框显示的 可写在你需要让他弹出的事件里
某个事件: function (){
this.popup.showPopup()
},
//取消事件
_error() {
console.log('你点击了取消');
this.popup.hidePopup();
},
//确认事件
_success() {
console.log('你点击了确定');
this.popup.hidePopup();
},
我们想要的组件就封装完毕了
捎带这里加一个 date的页面 大家可以看也可以不看 就是记录下
date.wxml
<view style="height:100%;">
<view class="message disf">
<image class="img1" src="/imgs/icon-Message-notification.png"></image>
<view>把训练计划加入日程里,可以有效安排训练时间和监督自己</view>
<image src="/imgs/icon-messages-turning-off.png" class="img2"></image>
</view>
<image src="/imgs/1photo-my.png" class="bgImg"></image>
<view class="date">
<view class="date-y-m">
<view class="y-m disf" style="padding-left: 25rpx;" bindtap="yearTap">
<view>{{year}}</view>
<image src="/imgs/pagedown.png"></image>
</view>
<view class="y-m disf" style="padding-right: 25rpx;" bindtap="monthTap">
<view>{{month}}月</view>
<image src="/imgs/pagedown.png"></image>
</view>
</view>
<view class="date-d">
<view class='header'>
<view wx:for='{{date}}' wx:key="index" class='{{(index == todayIndex) && isTodayWeek ? "weekMark" : ""}}'>{{item}}<view></view></view>
</view>
<view class='date-box'>
<view wx:for='{{dateArr}}' wx:key="index" class='{{isToday == item.isToday ? "nowDay" : ""}}' data-date='{{item.isToday}}' data-index="{{index}}" bindtap="dateTap">
<view class='date-head disf' style="{{item.status?'background: #FF4D3A;color: #FFFFFF;':''}}">
<view class="view" style="{{isToday == item.isToday ? 'transform: translateY(-2px);':''}}">{{item.dateNum}}</view>
<view class='{{isToday == item.isToday ? "dian" : "text"}}'>{{isToday == item.isToday ? "" : "练"}}</view>
</view>
</view>
</view>
</view>
</view>
<view class="datePopup" wx:if="{{datePopupShow}}">
<view class="datePopup-bottom">
<view class="popup-header">
<view class="no" data-type="0" bindtap="datePopupTap">取消</view>
<view class="yes" data-type="1" bindtap="datePopupTap">确定</view>
</view>
<view class="popup-footer">
<picker-view indicator-style="height: 50px;" style="100%;height:240rpx;" value="{{value}}" bindchange="bindChange">
<picker-view-column>
<view wx:for="{{years}}" class="disf" wx:key="index" style="height:80rpx;">{{item}}年</view>
</picker-view-column>
<picker-view-column>
<view wx:for="{{months}}" class="disf" wx:key="index" style="height:80rpx;">{{item}}月</view>
</picker-view-column>
</picker-view>
</view>
</view>
</view>
<popup id='popup'
content='是否要安排当日训练计划日程'
btn_no='否'
btn_ok='是'
bind:error="_error"
bind:success="_success">
</popup>
</view>
date.wxss
/* pages/actiondetail/index.wxss */
page{
height: 100%;
}
.disf{
display: flex;
justify-content: center;
align-items: center;
}
.message{
100%;
height: 80rpx;
background: rgba(255, 51, 51, 0.1);
font-size: 24rpx;
color: #FF3333;
position: absolute;
top: 0;
z-index: 20;
}
.message .img1{
34rpx;
height: 26rpx;
}
.message .img2{
13rpx;
height: 13rpx;
margin-left: 10rpx;
}
.bgImg{
height: 480rpx;
100%;
position: relative;
top: -150rpx;
}
.date{
position: relative;
top: -335rpx;
calc(100% - 48rpx);
padding: 24rpx;
background: rgba(255, 51, 51, 0);
}
.date .date-y-m{
315rpx;
height: 68rpx;
margin: 0 auto;
background: #FFFFFF;
box-shadow: 0px 0px 125rpx 0px rgba(169, 172, 189, 0.25);
border-radius: 33rpx;
display: flex;
}
.date .date-y-m .y-m{
50%;
height: 100%;
flex-direction: row;
}
.date .date-y-m .y-m view{
color: #333333;
font-size: 22rpx;
margin-right: 16rpx;
}
.date .date-y-m .y-m image{
20rpx;
height: 10rpx;
}
.date .date-d{
height: 692rpx;
100%;
background-image: url();
background-size: 100%;
background-repeat: no-repeat;
}
.header{
font-size: 0;
padding: 80rpx 24rpx 0 24rpx;
}
.header>view{
display: inline-block;
14.285%;
color: #333;
font-size: 30rpx;
text-align: center;
padding: 20rpx 0;
}
.weekMark{
position: relative;
}
.weekMark view{
position: absolute;
bottom: 0;
left: 0;
100%;
}
.date-box{
font-size: 0;
padding: 10rpx 25rpx;
}
.date-box>view{
position: relative;
display: inline-block;
14.285%;
color: #020202;
font-size: 40rpx;
text-align: center;
vertical-align: middle;
margin: 7rpx 0;
}
.date-head{
70rpx;
height: 70rpx;
font-size: 28rpx;
color: #333333;
border-radius: 50%;
flex-direction: column;
margin: 0 auto;
}
.nowDay .date-head{
color: #FF4D3A;
padding-bottom: 16rpx;
height: 48rpx;
}
.date-head .dian{
10rpx;
height: 10rpx;
background: #FF4D3A;
border-radius: 50%;
}
.date-head .text{
font-size: 22rpx;
color: #FFFFFF;
}
.date-weight{
font-size: 22rpx;
padding: 15rpx 0;
}
.nowDay .date-weight{
color: #22A7F6;
}
.datePopup{
height: 100%;
100%;
background: rgba(0, 0, 0, .28);
position: absolute;
top: 0;
z-index: 999;
}
.datePopup .datePopup-bottom{
100%;
height: 500rpx;
background: #FFFFFF;
border-radius: 30rpx 30rpx 0rpx 0rpx;
display: flex;
flex-direction: column;
position: fixed;
bottom: 0;
}
.datePopup .popup-header{
margin-top: 48rpx;
padding: 0 30rpx;
calc(100% - 60rpx);
display: flex;
flex-direction: row;
justify-content: space-between;
}
.popup-header .no{
color: #999999;
font-size: 30rpx;
}
.popup-header .yes{
color: #FFAA01;
font-size: 30rpx;
}
.popup-footer{
height: 240rpx;
622rpx;
margin: 80rpx auto;
overflow: hidden;
display: flex;
}
.intro {
margin: 30px;
text-align: center;
}
date.js
// pages/actiondetail/index.js
Page({
/**
* 页面的初始数据
*/
data: {
name: '',
year: 0,
month: 0,
date: ['日', '一', '二', '三', '四', '五', '六'],
dateArr: [],
isToday: 0,
isTodayWeek: false,
todayIndex: 0,
datePopupShow: false, // 日历弹窗
years: [],// 2000-2040年
months: [1,2,3,4,5,6,7,8,9,10,11,12], // 12个月
value: [21, 0],
},
/**
* 生命周期函数--监听页面加载
*/
onLoad: function (options) {
let now = new Date();
let year = now.getFullYear();
let month = now.getMonth() + 1;
console.log(year, month)
this.dateInit();
var arr = []
for( let i = 0; i < 40; i++) {
arr.push(2000+i)
}
this.setData({
year: year,
month: month,
isToday: '' + year + month + now.getDate(),
years: arr
})
},
dateInit: function (setYear, setMonth) { // 公用方法 获取日期天数
//全部时间的月份都是按0~11基准,显示月份才+1
let dateArr = []; //需要遍历的日历数组数据
let arrLen = 0; //dateArr的数组长度
let now = setYear ? new Date(setYear, setMonth) : new Date();
let year = setYear || now.getFullYear();
let nextYear = 0;
let month = setMonth || now.getMonth() + 1; //没有+1方便后面计算当月总天数
let nextMonth = (month) > 11 ? 1 : (month);
let startWeek = new Date(year + ',' + (month) + ',' + 1).getDay(); //目标月1号对应的星期
let dayNums = new Date(year, nextMonth, 0).getDate(); //获取目标月有多少天
let obj = {};
let num = 0;
if (month > 11) {
nextYear = year + 1;
dayNums = new Date(nextYear, nextMonth, 0).getDate();
}
arrLen = startWeek + dayNums;
for (let i = 0; i < arrLen; i++) {
if (i >= startWeek) {
num = i - startWeek + 1;
obj = {
isToday: '' + year + (month) + num,
dateNum: num,
weight: 5,
status: false
}
} else {
obj = {};
}
dateArr[i] = obj;
}
// console.log('dateArr', dateArr)
this.setData({
dateArr: dateArr
})
let nowDate = new Date();
let nowYear = nowDate.getFullYear();
let nowMonth = nowDate.getMonth() + 1;
let nowWeek = nowDate.getDay();
let getYear = setYear || nowYear;
let getMonth = setMonth >= 0 ? (setMonth + 1) : nowMonth;
if (nowYear == getYear && nowMonth == getMonth) {
this.setData({
isTodayWeek: true,
todayIndex: nowWeek
})
} else {
this.setData({
isTodayWeek: false,
todayIndex: -1
})
}
},
yearMonthTap () { // 显示日期选择框
this.setData({
datePopupShow: true
})
},
datePopupTap (e) { // 日期选择框点击确定时
var type = e.currentTarget.dataset.type
if (type == 0) {
this.setData({
datePopupShow: false
})
} else {
this.setData({
year: this.data.years[this.data.value[0]],
month: this.data.months[this.data.value[1]],
datePopupShow: false
})
let year = this.data.year
let month = this.data.month
console.log(year, month)
this.dateInit(year, month)
}
},
bindChange: function (e) { // 日期选择器框发生改变
const val = e.detail.value
this.setData({
value: val
})
},
dateTap (e) { // 点击某一天日期触发事件
this.popup.showPopup()
var index = e.currentTarget.dataset.index
var fictionList = this.data.dateArr
var fictionStatus = 'dateArr[' + index + '].status'
for (let i = 0; i < fictionList.length; i++) {
// var otherFiction = 'dateArr[' + i + '].status'
if (i == index) {
this.setData({
[fictionStatus]: true
})
}
}
},
//取消事件
_error() {
console.log('你点击了取消');
this.popup.hidePopup();
},
//确认事件
_success() {
console.log('你点击了确定');
this.popup.hidePopup();
},
/**
* 生命周期函数--监听页面初次渲染完成
*/
onReady: function () {
//获得popup组件
this.popup = this.selectComponent("#popup");
},
/**
* 生命周期函数--监听页面显示
*/
onShow: function () {
},
/**
* 生命周期函数--监听页面隐藏
*/
onHide: function () {
},
/**
* 生命周期函数--监听页面卸载
*/
onUnload: function () {
},
/**
* 页面相关事件处理函数--监听用户下拉动作
*/
onPullDownRefresh: function () {
},
/**
* 页面上拉触底事件的处理函数
*/
onReachBottom: function () {
},
/**
* 用户点击右上角分享
*/
onShareAppMessage: function () {
}
})