Taro自定义Modal对话框组件|taro仿微信、android弹窗

基于Taro多端实践TaroPop:自定义模态框|dialog对话框|msg消息框|Toast提示

taro自定义弹出框支持编译到多端H5/小程序/ReactNative,还可以自定义弹窗类型/弹窗样式、多按钮事件/样式、自动关闭、遮罩层、弹窗显示位置及自定义内容模板

用法

 ▍在相应页面引入组件

import TaroPop from ‘@components/taroPop‘

import Taro from ‘@tarojs/taro‘
import { View, Text } from ‘@tarojs/components‘

// 引入自定义弹窗组件
import TaroPop from ‘@components/taroPop‘

export default class TaroPopDemo extends Taro.Component {
    ...

    render() {
        return (
            <View className="taro-container">
                ...

                {/* 引入弹窗模板 */}
                <TaroPop ref="taroPop" />
            </View>
        );
    }
}

通过ref方式调用组件内show、close方法

this.refs.taroPop.show({...options})

this.refs.taroPop.close()

 ▍自定义弹窗模板内容(如下图)

只需把页面上的模板写成如下即可,调用方式还和上面一样

<TaroPop ref="taroPopTpl">
    ...
</TaroPop>

支持多种参数配置:

/**
 * @ 弹窗默认配置
 */
static defaultProps = {
    isVisible: false,       //弹窗显示

    title: ‘‘,              //标题
    content: ‘‘,            //内容
    contentStyle: null,     //内容样式
    style: null,            //自定义弹窗样式
    skin: ‘‘,               //弹窗风格
    icon: ‘‘,               //弹窗图标
    xclose: false,          //自定义关闭按钮

    shade: true,            //遮罩层
    shadeClose: true,       //点击遮罩关闭
    opacity: ‘‘,            //遮罩透明度
    time: 0,                //自动关闭时间
    end: null,              //销毁弹窗回调函数

    position: ‘‘,           //弹窗位置显示

    btns: null,             //弹窗按钮 [{...args}, {...args}]
}
/**
 * 显示弹窗事件
 */
show = (options) => {
    this.setState({
        ...this.props, ...options, isVisible: true
    })
}

/**
 * 关闭弹窗事件
 */
close = () => {
    this.setState({...this.props})

    this.timer && clearTimeout(this.timer)
    delete this.timer

    typeof this.state.end === ‘function‘ && this.state.end.call(this)
}

/**
 * 点击遮罩关闭
 */
shadeClick = () => {
    if(!this.state.shadeClose) return
    this.close()
}

◆ msg消息框提示

this.refs.taroPop.show({
    content: ‘Taro自定义模态Modal弹窗‘,
    shadeClose: false,
    style: {backgroundColor: ‘rgba(0,0,0,.7)‘, borderRadius: 6},
    contentStyle: {color: ‘#fff‘, fontSize: 12, padding: 12},
    time: 3,
    opacity: .2,
})

◆ Toast轻提示效果(success | error | info | loading四种图标)

let taroPop = this.refs.taroPop
taroPop.show({
    skin: ‘toast‘,
    content: ‘loading‘,
    icon: ‘loading‘, //success | info | error | loading
    shade: false,
    time: 3
})

◆ android弹窗效果

let taroPop = this.refs.taroPop
taroPop.show({
    skin: ‘android‘,
    title: ‘邮件提醒‘,
    content: ‘系统检测到你未开启新邮件提醒功能,为了保证新邮件能及时收到提醒,请前往系统 [设置] - [应用] 中开启‘,
    shadeClose: false,

    btns: [
        {
            text: ‘取消‘,
            onClick() {
                taroPop.close()
            }
        },
        {
            text: ‘前往设置‘,
            style: {color: ‘#4eca33‘},
            onClick() {
                console.log(‘您点击了前往设置!‘)
            }
        }
    ]
})

emmmm,看了如上展示及调用方式,是否觉得还不错哟!哈哈哈,这可是花了无数个日夜采坑的结果。

尤其是编译到reactNative端,各种千奇百怪的问题,有些抓狂~~

另外对于不同端的一些兼容性处理,需要判断各端环境并渲染相应模板,对于RN,则使用Modal

let taroEnv = process.env.TARO_ENV

// 渲染窗体
if (taroEnv === ‘rn‘) {
    return (
        <Modal transparent={true} visible={isVisible} onRequestClose={this.close}>
            {renderTpl}
        </Modal>
    )
}else if (taroEnv === ‘h5‘ || taroEnv === ‘weapp‘){
    return isVisible && renderTpl
}

另外在样式处理上也需注意RN端兼容性。

/**
 * @Title     Taro自定义弹窗组件 - taroPop.js
 * @Time     andy by 2019-11-28
 * @About     Q:282310962  wx:xy190310
 */

import Taro from ‘@tarojs/taro‘
import { View, Text, Image } from ‘@tarojs/components‘
import { Modal, ActivityIndicator, TouchableHighlight } from ‘react-native‘
import classNames from ‘classnames‘
import ‘./index.scss‘

export default class TaroPop extends Taro.Component {
    /**
     * @ 弹窗默认配置
     */
    static defaultProps = {
        isVisible: false,       //弹窗显示

        title: ‘‘,              //标题
        content: ‘‘,            //内容
        contentStyle: null,     //内容样式
        style: null,            //自定义弹窗样式
        skin: ‘‘,               //弹窗风格
        icon: ‘‘,               //弹窗图标
        xclose: false,          //自定义关闭按钮

        shade: true,            //遮罩层
        shadeClose: true,       //点击遮罩关闭
        opacity: ‘‘,            //遮罩透明度
        time: 0,                //自动关闭时间
        end: null,              //销毁弹窗回调函数

        anim: ‘scaleIn‘,        //弹窗动画
        position: ‘‘,           //弹窗位置显示

        btns: null,             //弹窗按钮 [{...args}, {...args}]
    }

    constructor(props) {
        super(props)
        this.state = {
            ...this.props,
        }
        this.timer = null
    }

    /**
     * @ 显示弹窗事件
     */
    show = (options) => {
        this.setState({
            ...this.props, ...options, isVisible: true
        })
    }

    /**
     * @ 关闭弹窗事件
     */
    close = () => {
        this.setState({...this.props})

        this.timer && clearTimeout(this.timer)
        delete this.timer

        typeof this.state.end === ‘function‘ && this.state.end.call(this)
    }

    /**
     * @ 点击遮罩关闭
     */
    shadeClick = () => {
        if(!this.state.shadeClose) return
        this.close()
    }

    render() {
        let { isVisible, title, content, contentStyle, style, skin, icon, xclose, shade, shadeClose, opacity, time, end, anim, position, btns } = this.state

        let toastIcon = {
            loading: require(‘./skin/loading.png‘),
            success: require(‘./skin/success.png‘),
            error: require(‘./skin/error.png‘),
            info: require(‘./skin/info.png‘),
        }

        let taroEnv = process.env.TARO_ENV

        ...

        // 渲染H5、RN模板
        const renderTpl = (
            <View className="taroPop">
                {/* 遮罩 */}
                {shade ? <View className="atpop__ui_mask" style={{opacity: opacity == ‘‘ ? .6 : opacity}} onClick={this.shadeClick} /> : null}
                {/* 窗体 */}
                <View className="atpop__ui_main">
                    <View className={classNames(‘atpop__ui_child‘, skin && ‘atpop__‘ + skin, position && ‘atpop__ui_child-‘ + position)} style={style}>
                        {/* 标题 */}
                        {title ? <Text className={classNames(‘atpop__ui_tit‘, skin && ‘atpop__ui_tit-‘ + skin)}>{title}</Text> : null}
                        {/* 内容 */}
                        {content ? <View className="atpop__ui_cnt">
                            {/* toast内容 */}
                            {icon && skin === ‘toast‘ ?
                                <View className="atpop__ui_toast">
                                    {icon === ‘loading‘ && taroEnv === ‘rn‘ ?
                                    <ActivityIndicator color="rgba(255,255,255,.5)" size={24} /> : <Image className={classNames(‘atpop__ui_toast-img‘, icon==‘loading‘ && ‘atpop__ui_toast-img-loading‘)} src={toastIcon[icon]} mode="aspectFit" />
                                    }
                                </View>
                                :
                                null
                            }
                            {/* 文本内容 */}
                            <Text className={classNames(‘atpop__ui_cntxt‘, skin && ‘atpop__ui_cntxt-‘ + skin)} style={contentStyle}>{content}</Text>
                        </View>
                        :
                        this.props.children
                        }
                        {/* 按钮 */}
                        {btns ? <View className={classNames(‘atpop__ui_btns‘, skin && ‘atpop__ui_btns-‘ + skin)}>
                            {btns.map((item, i) => {
                                return taroEnv === ‘rn‘ ?
                                <TouchableHighlight className={classNames(‘atpop__ui_btn‘, skin && ‘atpop__ui_btn-‘ + skin)} activeOpacity={1} underlayColor=‘rgba(200,200,200,.3)‘ key={i} onPress={item.onClick}>
                                    <Text className={classNames(‘atpop__ui_btntxt‘, skin && ‘atpop__ui_btntxt-‘ + skin)} style={item.style}>{item.text}</Text>
                                </TouchableHighlight>
                                :
                                <View className={classNames(‘atpop__ui_btn‘, skin && ‘atpop__ui_btn-‘ + skin)} key={i} onClick={item.onClick}>
                                    <Text className={classNames(‘atpop__ui_btntxt‘, skin && ‘atpop__ui_btntxt-‘ + skin)} style={item.style}>{item.text}</Text>
                                </View>
                            })}
                        </View>
                        :
                        null
                        }
                    </View>
                    {/* xclose */}
                    {xclose ? <View className="atpop__ui_xclose" onClick={this.close}><Image className="atpop__ui_xclose-img" src={require(‘./skin/error.png‘)} mode="aspectFit" /></View> : null}
                </View>
            </View>
        )

        // 渲染窗体
        if (taroEnv === ‘rn‘) {
            return (
                <Modal transparent={true} visible={isVisible} onRequestClose={this.close}>
                    {renderTpl}
                </Modal>
            )
        }else if (taroEnv === ‘h5‘ || taroEnv === ‘weapp‘){
            return isVisible && renderTpl
        }
    }
}

好了,以上就是taro自定义弹窗组件实现方式,希望能有帮助??~~

原文地址:https://www.cnblogs.com/xiaoyan2017/p/11969729.html

时间: 2024-10-12 23:05:17

Taro自定义Modal对话框组件|taro仿微信、android弹窗的相关文章

安卓开发复习笔记——Fragment+ViewPager组件(高仿微信界面)

什么是ViewPager? 关于ViewPager的介绍和使用,在之前我写过一篇相关的文章<安卓开发复习笔记——ViewPager组件(仿微信引导界面)>,不清楚的朋友可以看看,这里就不再重复. 什么是Fragment? Fragment是Android3.0后新增的概念,Fragment名为碎片,不过却和Activity十分相似,具有自己的生命周期,它是用来描述一些行为或一部分用户界面在一个Activity中,我们可以合并多个Fragment在一个单独的activity中建立多个UI面板,或

转-Fragment+ViewPager组件(高仿微信界面)

http://www.cnblogs.com/lichenwei/p/3982302.html 什么是ViewPager? 关于ViewPager的介绍和使用,在之前我写过一篇相关的文章<安卓开发复习笔记——ViewPager组件(仿微信引导界面)>,不清楚的朋友可以看看,这里就不再重复. 什么是Fragment? Fragment是Android3.0后新增的概念,Fragment名为碎片,不过却和Activity十分相似,具有自己的生命周期,它是用来描述一些行为或一部分用户界面在一个Act

安卓开发复习笔记——ViewPager组件(仿微信引导界面&gt;)

这2天事情比较多,都没时间更新博客,趁周末,继续继续~ 今天来讲个比较新潮的组件——ViewPager 什么是ViewPager? ViewPager是安卓3.0之后提供的新特性,继承自ViewGroup,专门用以实现左右滑动切换View的效果. 如果想向下兼容就必须要android-support-v4.jar这个包的支持,这是一个来自google提供的一个附加包. 通俗点来讲,就是现在市面上大多数app,安装完第一次打开软件会出现的一个左右滑动的引导界面. 先来看下效果图:     这是一个

【Android UI设计与开发】第14期:顶部标题栏(五)两种方式实现仿微信标题栏弹窗效果

转载请注明出处:http://blog.csdn.net/yangyu20121224/article/details/9093821         博主在这篇文章中将会继续围绕顶部标题栏专题来进行实例讲解,今天要讲解的主题是分别使用PopupWindow和Activity两种不同的方式来实现仿微信顶部标题栏弹窗的这样一个效果. 一.实现效果图 这里为了演示方便,我将两种方法放在一个应用程序中演示,这个是主界面 虽然两种实现的方式不一样,但是最终的效果图都是差不多的     二.项目结构图  

h5聊天室web端(仿微博、微信)|h5仿微信网页端|仿微信界面弹窗

html5开发的仿微博.微信聊天web端案例,h5仿微信聊天网页版,采用html5+css3+jquery+swiper+wcPop等技术进行布控架构开发,弹窗插件wcPop.js进行了一次全面api升级,修复编辑器插入表情时光标定位错误bug,新增了上传附件及自定义推送内容,另外也新增了个人名片.上传附件.分享等样式,功能上实现了消息.表情的发送,图片.视频全屏预览. 项目运行图: /* --- 用户设置.Start ---*/ // 联系人/群聊切换 $("body").on(&q

小程序超实用组件:仿微信通讯录

效果图 因为是使用的手机录屏,视频格式为MP4,上传到文章时发现只支持图片,还好电脑自动录屏功能,所以简单的录制了一下,完后又提示只能4M,只能再去压缩图片,所以画质略渣,各位客官讲究的看看吧. 特色功能介绍 用户只需按照格式传入参数,组件能够自动将参数按首字母分组,简单方便: 组件右侧首字母导航无需另外传值,并且根据参数具体有哪些首字母显示(没有的咱就不要): 用户进行上下滑动时,左右相互联动: 点击右侧导航,组件会相应的上下滚动. 实现基础 说到滚动当然少不了小程序的基础组件scroll-v

(仿微信Android)聊天+红包+直播+朋友圈源码发布了

今天给大伙推荐一个安卓开源项目-“凡信”的最新版本,很值得开发和产品同学们研究一下. 功能概览:IM聊天单聊/群聊/聊天室--基于环信sdk红包功能:1.一对一红包2.群红包(抢红包.答题红包.专属红包)3.钱包(资金及账户管理)直播功能:Ucloud推流 .Ucloud拉流.电台拉流朋友圈.用户体系.好友体系凡信自己服务器维护管理数据详细介绍(凡信)背景介绍:https://github.com/huangfangyi/FanXin2.0_IMhttp://www.imgeek.org/art

Taro聊天室|react+taro仿微信聊天App界面|taro聊天实例

一.项目简述 taro-chatroom是基于Taro多端实例聊天项目,运用Taro+react+react-redux+taroPop+react-native等技术开发的仿微信App界面聊天室,实现了发送消息/emoj表情.gif表情大图.图片预览.发红包.动态圈等功能. 二.预览效果 编译到H5端.小程序.App端效果如下:(后续大图均为APP端) 三.技术栈 编码/技术:Vscode + react/taro/redux/RN iconfont图标:阿里字体图标库 自定义导航栏Navig

微信小程序自定义弹窗wcPop插件|仿微信弹窗样式

微信小程序自定义组件弹窗wcPop|小程序消息提示框|toast自定义模板弹窗 平时在开发小程序的时候,弹窗应用场景还是蛮广泛的,但是微信官方提供的弹窗比较有局限性,不能自定义修改.这个时候首先想到的是自定义组件化开发,就是把弹出框封装成一个组件,然后多处调用. 解决了小程序开发自定义弹窗出现后,遮罩层下的页面仍可以滚动的方法: 给遮罩层的最外层view中加入catchtouchmove="preventTouchMove" 即可解决该遮罩层点透问题. 根据需要还可以自定义多个按钮及事