草庐IT

走进小程序【四】小程序自定义Component如何使用,手把手封装一个底部Tabbar栏

忆凡_ 2023-04-14 原文

文章目录


🌟前言

哈喽小伙伴们,上一期为大家讲解了一下小程序的代码构成并且搭建了一个简单的钢琴小程序,不知道大家有没有扫码去体验一下呢;这期给大家说一说小程序的自定义组件,并且搭建一个自定义的底部Tabbar栏;话不多说,咱们直接开整!🤘

🌟先看效果

🌟什么是小程序自定义Component

🌟Component 组件化

写Vue的小伙伴们应该都知道,Vue有一个很重要的思想就是组件化,为的是一次封装,多次复用;带来的好处就是当我们的业务功能重复繁琐时候就可以把主要功能抽离出来封装成组件,便于我们在多个地方去使用;还有就是框架的组件并不符合我们的实际业务功能,那么这个时候也需要我们去独立封装一个适用的组件。在我们微信小程序里边,也有组件化的思想,那它是怎么定义的呢?

🌟小程序的自定义Component

小程序自定义组件

开发者可以将页面内的功能模块抽象成自定义组件,以便在不同的页面中重复使用;也可以将复杂的页面拆分成多个低耦合的模块,有助于代码维护。自定义组件在使用时与基础组件非常相似。

从小程序基础库版本 1.6.3 开始,小程序支持简洁的组件化编程。所有自定义组件相关特性都需要基础库版本 1.6.3 或更高。

🌟创建自定义组件

类似于页面,一个自定义组件由 json、 wxml、 wxss、 js 4个文件组成。要编写一个自定义组件,首先需要在 json 文件中进行自定义组件声明(将 component 字段设为 true 可将这一组文件设为自定义组件):

{
  "component": true
}

同时,还要在 wxml 文件中编写组件模板,在 wxss 文件中加入组件样式,它们的写法与页面的写法类似。

代码示例:

<!-- 这是自定义组件的内部WXML结构 -->
<view class="inner">
  {{innerText}}
</view>
<slot></slot>
/* 这里的样式只应用于这个自定义组件 */
.inner {
  color: red;
}

注意:在组件wxss中不应使用ID选择器、属性选择器和标签名选择器。

在自定义组件的 js 文件中,需要使用 Component() 来注册组件,并提供组件的属性定义、内部数据和自定义方法。 组件的属性值和内部数据将被用于组件 wxml 的渲染,其中,属性值是可由组件外部传入的。

代码示例:

Component({
  properties: {
    // 这里定义了innerText属性,属性值可以在组件使用时指定
    innerText: {
      type: String,
      value: 'default value',
    }
  },
  data: {
    // 这里是一些组件内部数据
    someData: {}
  },
  methods: {
    // 这里是一个自定义方法
    customMethod: function(){}
  }
})

这里的代码结构我们可以使用开发者工具的快捷生成不用我们去写

🌟使用自定义组件

使用已注册的自定义组件前,首先要在页面的 json 文件中进行引用声明。此时需要提供每个自定义组件的标签名和对应的自定义组件文件路径:

{
  "usingComponents": {
    "component-tag-name": "path/to/the/custom/component"
  }
}

这样,在页面的 wxml 中就可以像使用基础组件一样使用自定义组件。节点名即自定义组件的标签名,节点属性即传递给组件的属性值。

开发者工具 1.02.1810190 及以上版本支持在 app.json 中声明 usingComponents 字段,在此处声明的自定义组件视为全局自定义组件,在小程序内的页面或自定义组件中可以直接使用而无需再声明。

代码示例:

在开发者工具中预览效果

<view>
  <!-- 以下是对一个自定义组件的引用 -->
  <component-tag-name inner-text="Some text"></component-tag-name>
</view>

自定义组件的 wxml 节点结构在与数据结合之后,将被插入到引用位置内。

一些需要注意的细节:

  • 因为 WXML 节点标签名只能是小写字母、中划线和下划线的组合,所以自定义组件的标签名也只能包含这些字符。
  • 自定义组件也是可以引用自定义组件的,引用方法类似于页面引用自定义组件的方式(使用 usingComponents 字段)。
  • 自定义组件和页面所在项目根目录名不能以“wx-”为前缀,否则会报错。

注意,是否在页面文件中使用 usingComponents 会使得页面的 this 对象的原型稍有差异,包括:

  • 使用 usingComponents 页面的原型与不使用时不一致,即 Object.getPrototypeOf(this) 结果不同。
  • 使用 usingComponents 时会多一些方法,如 selectComponent 。
  • 出于性能考虑,使用 usingComponents 时, setData 内容不会被直接深复制,即 this.setData({ field: obj }) 后 this.data.field === obj 。(深拷贝会在这个值被组件间传递时发生。)

如果页面比较复杂,新增或删除 usingComponents 定义段时建议重新测试一下。

🌟封装Tabbar自定义组件

🌟初始化文件结构

根目录下创建component文件夹,在其下创建tabbar文件夹,然后再tabbar文件夹里创建名为index的Component

🌟component/tabbar/index.json

component 设置为 true,即为自定义组件

{
    "component": true
}

🌟component/tabbar/index.wxml

<view class="tabbar" style="width: 500rpx;height: 300rpx;">
  <view class="item {{index==idx?'active':''}}" wx:for="{{tabBar}}" wx:for-index="idx" bindtap="goto" data-index='{{idx}}' data-path="{{item.path}}">
    <i class="iconfont {{item.icon}} icon"></i>
    <view class="text">{{item.name}}</view>
  </view>
  <view class="move" style="left:{{115+(index*210)}}rpx"></view>
</view>

🌟component/tabbar/index.wxss

.tabbar{
    width: 100% !important;
    height: 140rpx !important;
    position: fixed;
    bottom: 0;
    left: 0;
    background: #181c27;
    display: flex;
    justify-content: center;
    box-sizing: border-box;
    padding:0 60rpx;
  }
  
  
  .tabbar .item{
    flex: 1;
    color: #fff;
    height: 100rpx;
    display: flex;
    flex-wrap: wrap;
    justify-content: center;
    align-items: center;
    position: relative;
    z-index: 10;
  
  }
  .tabbar .item .text{
    position: absolute;
    width: 100%;
    bottom: 10rpx;
    text-align: center;
    font-size: 22rpx;
    opacity: 0;
    transition: all .8s;
    transform: scale(0.8);
    width: 100%;
  }
  .tabbar .item.active .text{
    opacity: 1;
    transform: scale(1);
  }
  
  
  .tabbar .item.active .icon{
    color: #3561f5;
    transform: translateY(-55rpx);
  }
  
  .tabbar .item .icon{
    font-size: 50rpx!important;
    text-align: center;
    transition: all .8s;
    
  }
  page{
    width: 100%;
    height: 100%;
    /* background: #404655; */
  }
  
  .move{
    width: 100rpx;
    height: 100rpx;
    background: #ffffff;
    box-shadow: 0rpx 0rpx 50rpx 2rpx rgba(255, 255, 255, 0.9);
    border-radius: 50%;
    position: absolute;
    left: 115rpx;
    top: -35%;
    transition: all .3s;
    z-index: 1;
  }

🌟component/tabbar/index.js

Component({
    options: {
        addGlobalClass: true
    },
    behaviors: [],
    properties: { // 类似于Vue的props,用来接收父组件传递的参数
        myProperty: { // 属性名
            type: String,
            value: ''
        },
        myProperty2: String // 简化的定义方式
    },

    data: {
        index: 0, // 默认显示第一个(首页)
        
        // tabbar栏切换,这三个页面要提前建好,放在根目录下的pages里
        tabBar: [{
                name: '首页',
                icon: 'icon-home',
                path: '../../pages/home/index'
            },
            {
                name: '商品',
                icon: 'icon-all',
                path: '../../pages/shop/shop'
            },
            {
                name: '我的',
                icon: 'icon-bussiness-man',
                path: '../../pages/my/my'
            },
        ]
    }, // 私有数据,可用于模板渲染

    lifetimes: {
        // 生命周期函数,可以为函数,或一个在methods段中定义的方法名
        attached: function () {},
        moved: function () {},
        detached: function () {},
    },

    // 生命周期函数,可以为函数,或一个在methods段中定义的方法名
    attached: function () {}, // 此处attached的声明会被lifetimes字段中的声明覆盖
    ready: function () {},

    pageLifetimes: {
        // 组件所在页面的生命周期函数
        show: function () {},
        hide: function () {},
        resize: function () {},
    },

    methods: {

        goto(e) {
            // console.log(e.currentTarget.dataset.index);
            if (e.currentTarget.dataset.index != this.data.index) {
                this.setData({
                    index: e.currentTarget.dataset.index
                })
            }
            // this.triggerEvent 类似于Vue的 this.$emit; 用来子组件给父组件传递参数
            // 这里我们把每次点击的Tab栏下标传递给父组件
            this.triggerEvent("itemChange",{index:e.currentTarget.dataset.index});
        },
        onMyButtonTap: function () {
            this.setData({
                // 更新属性和数据的方法与更新页面数据的方法类似
            })
        },
        // 内部方法建议以下划线开头
        _myPrivateMethod: function () {
            // 这里将 data.A[0].B 设为 'myPrivateData'
            this.setData({
                'A[0].B': 'myPrivateData'
            })
        },
        _propertyChange: function (newVal, oldVal) {

        }
    }

})

🌟页面中使用

找到pages/home文件夹:

🌟pages/home/index.json

{
  "usingComponents": {
      "tab-bar":"../../component/tabbar1/index",
      "my":"../../pages/my/my",
      "shop":"../../pages/shop/shop"
  },
  "navigationBarTitleText": "首页"
}

把我们封装好的tab-bar组件注册进来,属性名随便写,类似Vue的组件名;值为该组件的路径。
Tabbar其余页面组件也注册进来,后面会用到(页面跳转切换)。

🌟pages/home/index.wxml

<view>
    <!-- 【首页】展示的页面 -->
    <block wx:if="{{currentIndex === 0}}">
        <van-grid column-num="3" border="{{ false }}">
            <van-grid-item use-slot wx:for="{{ 8 }}" wx:for-item="index">
                <image style="width: 100%; height: 90px;" src="https://img.yzcdn.cn/vant/apple-{{ index + 1 }}.jpg" />
            </van-grid-item>
        </van-grid>
    </block>
    <!-- Tabbar切换至 【商店】 展示的页面 -->
    <shop wx:if="{{currentIndex === 1}}"></shop>
    <!-- Tabbar切换至 【我的】 展示的页面 -->
    <my wx:if="{{currentIndex === 2}}"></my>
    <!-- 刚才自定义的 【Tabbar】 -->
    <tab-bar bind:itemChange="handleItemChange"></tab-bar>
</view>

🌟pages/home/index.js

// index.js
// 获取应用实例
const app = getApp()

Page({
    data: {
        currentIndex: 0,  // 初始下标,默认显示首页
    },
    handleItemChange(e) {
        this.setData({
            currentIndex: e.detail.index
        })
        if (this.data.currentIndex === 0) {
            wx.setNavigationBarTitle({
                title: '首页' //页面切换,更换页面标题
            })
        } else if (this.data.currentIndex === 1) {
            wx.setNavigationBarTitle({
                title: '商店' //页面切换,更换页面标题
            })
        } else(
            wx.setNavigationBarTitle({
                title: '我的' //页面切换,更换页面标题
            })
        )
    },
    onLoad() {},
})

🌟结语

这篇为小伙伴们讲解了在小程序当中,我们如何去自定义或者封装我们自己的组件;大家自己也要尝试一下,思想和Vue差不太多,相信小伙伴们一定也可以的,加油;各位小伙伴让我们 let’s be prepared at all times!

✨原创不易,还希望各位大佬支持一下!
👍 点赞,你的认可是我创作的动力!
⭐️ 收藏,你的青睐是我努力的方向!
✏️ 评论,你的意见是我进步的财富!

有关走进小程序【四】小程序自定义Component如何使用,手把手封装一个底部Tabbar栏的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  3. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  4. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  5. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  6. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  7. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  8. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  9. ruby-on-rails - 如何生成传递一些自定义参数的 `link_to` URL? - 2

    我正在使用RubyonRails3.0.9,我想生成一个传递一些自定义参数的link_toURL。也就是说,有一个articles_path(www.my_web_site_name.com/articles)我想生成如下内容:link_to'Samplelinktitle',...#HereIshouldimplementthecode#=>'http://www.my_web_site_name.com/articles?param1=value1¶m2=value2&...我如何编写link_to语句“alàRubyonRailsWay”以实现该目的?如果我想通过传递一些

  10. ruby-on-rails - 如何在 Rails 3 中创建自定义脚手架生成器? - 2

    有这些railscast。http://railscasts.com/episodes/218-making-generators-in-rails-3有了这个,你就会知道如何创建样式表和脚手架生成器。http://railscasts.com/episodes/216-generators-in-rails-3通过这个,您可以了解如何添加一些文件来修改脚手架View。我想把两者结合起来。我想创建一个生成器,它也可以创建脚手架View。有点像RyanBates漂亮的生成器或web_app_themegem(https://github.com/pilu/web-app-theme)。我

随机推荐