用Taro搭建一个简易的电商类小程序

Taro 是一套遵循React语法规范的多端开发解决方案,可以快速开发微信小程序,并且支持多端开发转化,喜欢React语法的可以多试试(适合了解微信小程序和React的铁子们)!

这次做一个电商类的小程序,主要做首页(轮播图,商品列表)、商品详情页、购物车以及登录等主要功能页,疫情期间,逆战一波吧,铁子们...

1.安装脚手架

# 使用 npm 安装 CLI

$ npm install -g @tarojs/cli

# 使用 yarn 安装 CLI

$ yarn global add @tarojs/cli

# 安装了 cnpm,使用 cnpm 安装 CLI

$ cnpm install -g @tarojs/cli

2.创建项目

taro init myApp

**不建议在此处使用 npx 创建**

3.运行项目

 3.1 运行到浏览器端

npm run dev:h5

npm run build:h5

3.2 运行微信小程序

npm run dev:weapp

npm run build:weapp

4.熟悉项目结构以及代码规范

**所有的自己写的代码都在src文件夹下**

5、创建项目的基本路由

5.1 创建了 页面

在pages文件夹下新建页面文件

pages/home/index.jsx

pages/kind/index.jsx

pages/cart/index.jsx

pages/user/index.jsx

如下图:

在index.jsx中编辑,以首页为例

import Taro, { Component } from '@tarojs/taro'

// 为什呢 View 要单独引入,react说明组件的首字母一定要大写,小写被当做html标签

import { View } from '@tarojs/components'

class Index extends Component {

  render () {

    return (

      <View>

        首页

      </View>

    )

  }

}

export default Index

5.2 配置路由

同微信小程序一样,我们可以在src/app.jsx文件中进行全局配置

// src/app.jsx

pages: [

 // 路由

  'pages/home/index',

  'pages/kind/index',

  'pages/cart/index',

  'pages/user/index',

  'pages/index/index'

],

5.3 配置底部选项卡(tabBar)

注意:

key值一律不加引号

value只加单引号

key与value值之间需要冒号后加 空格

图片使用相对路径

// src/app.jsx

tabBar: {

  color: '#333',

  selectedColor: '#f66',

  backgroundColor: '#efefef',

  borderStyle: 'white',

  list: [ // 图片资源必须使用  相对路径(图片资源保存在src/resources文件夹中)

    {

      pagePath: 'pages/home/index',

      text: '首页',

      iconPath: './resources/home.png',

      selectedIconPath: './resources/home_active.png'

    },

    {

      pagePath: 'pages/kind/index',

      text: '分类',

      iconPath: './resources/kind.png',

      selectedIconPath: './resources/kind_active.png'

    },

    {

      pagePath: 'pages/cart/index',

      text: '购物车',

      iconPath: './resources/cart.png',

      selectedIconPath: './resources/cart_active.png'

    },

    {

      pagePath: 'pages/user/index',

      text: '我的',

      iconPath: './resources/user.png',

      selectedIconPath: './resources/user_active.png'

    }

  ]

}

如图:

5.4 页面的配置

参考微信小程序的页面配置,多端化的会有兼容问题(H5和rn兼容问题居多,很多都不支持)

以首页为例,添加页面的配置

// pages/home/index.jsx

config = { //在微信小程序中会生成一个home/index.json文件 --- 页面配置

  navigationBarTitleText: 'taro-首页',

  enablePullDownRefresh: true,

  backgroundColor: '#efefef'

}

6、生命周期钩子函数

6.1 全局生命周期钩子函数

// app.jsx

1.// onLaunch 在微信/百度/字节跳动/支付宝小程序中这一生命周期方法对应 app 的 onLaunch

1.componentWillMount() {}

2.// onLaunch 在微信/百度/字节跳动/支付宝小程序中这一生命周期方法对应 app 的 onLaunch,在 componentWillMount 后执行

2.componentDidMount () {}

3.// onShow

3.componentDidShow () {}

4.// onHide

4.componentDidHide () {}

5.// onError

5.componentDidCatchError () {}

6.// onPageNotFound 404

6.componentDidNotFound () {}

6.2 页面级生命周期钩子函数

componentWillMount()

页面加载时触发,一个页面只会调用一次,此时页面 DOM 尚未准备好,还不能和视图层进行交互 ----- react一致 --- 一般不使用(以前的react版本支持在此处请求数据,但是现在不建议)

componentDidMount()

页面初次渲染完成时触发,一个页面只会调用一次,代表页面已经准备妥当,可以和视图层进行交互 -   react一致 --- 请求数据、DOM操作

shouldComponentUpdate(nextProps, nextState)

页面是否需要更新,返回 false 不继续更新,否则继续走更新流程 --- react 一致  ---  提升react组件性能的关键

componentWillUpdate(nextProps, nextState)

页面即将更新  --- react一致  --- 一般不使用

componentDidUpdate(prevProps, prevState)

页面更新完毕  ----  react一致  --- 可以请求数据但是不建议,如果非要请求,必须得加判断 ---- 不能在此请求数据(错误的),如果你对更新前后的 props 进行了比较,也可以选择在此处进行网络请求

componentDidUpdate(prevProps) {

  // 典型用法(不要忘记比较 props):

  if (this.props.userID !== prevProps.userID) {

    this.fetchData(this.props.userID);

  }

}

componentWillUnmount()

页面卸载时触发,如 redirectTo 或 navigateBack 到其他页面时 --- 小程序这块 缓存我们的页面 

componentDidShow()

页面显示/切入前台时触发  ----  onShow()

componentDidHide() --- onHide()

页面隐藏/切入后台时触发, 如 navigateTo 或底部 tab 切换到其他页面,小程序切入后台等

在以上所有的生命周期方法中,都可以通过 this.$router.params 获取打开当前页面路径中的参数。

**其他的函数**

onPullDownRefresh()

onReachBottom()

onPageScroll(Object)

onShareAppMessage(Object)

6.3 组件生命周期 参考的就是react的生命周期

7.熟悉taro中的各种语法

状态、属性、列表渲染、条件的渲染、事件的处理...(详情可以去Taro官网查看)

8.项目首页的相关实现

8.1实现首页轮播图

(轮播图状态、生命周期钩子中请求数据、修改状态、渲染数据)

1.状态设置

// constructor 写了必写 super ,要不就不写 constructor,super事关 this指向的问题   ---  es6的构造函数中的机制

  // ES5 的继承,实质是先创造子类的实例对象this,然后再将父类的方法添加到this上面(Parent.apply(this))。ES6 的继承机制完全不同,实质是先将父类实例对象的属性和方法,加到this上面(所以必须先调用super方法),然后再用子类的构造函数修改this。

//src/pages/home/index.jsx

constructor (props) {

  super(props)

  // 1.设置状态

  this.state = {

    bannerlist: []

  }

}

2、请求数据

* 封装数据库的请求 src/utils/index.js,按照自己的后台接口来封装

// src/utils/index.js

import Taro from '@tarojs/taro'

const baseUrl = '后台接口(这里不可以复制哈~)'

export function request (options) {

  const { url, data, method, header } = options

  Taro.showLoading({

    title: '加载中'

  })

  return new Promise((resolve, reject) => {

    Taro.request({

      url: baseUrl + url,

      data: data || {},

      method: method || 'GET',

      header: header || {},

      success: res => {

        resolve(res)

      },

      fail: err => {

        reject(err)

      },

      complete: () => {

        Taro.hideLoading()

      }

    })

  })

}

```

* pages/home/index.jsx 引入请求数据,修改状态

//pages/home/index.jsx

// 2.请求数据

import { request } from './../../utils'

componentDidMount () {

  // vue /react  axios /fetch ajax

  // wx  wx.request

  // uni uni.request

  // taro taro.request

  request({

    url: '/pro/banner'

  }).then(res => {

    console.log(res.data)

    this.setState({

      bannerlist: res.data.data

    })

  })

}

//渲染数据

render () {

  return (

    <View>

      <Swiper indicatorDots autoplay circular>

        {

          this.state.bannerlist.map((item, index) => (

            <SwiperItem key={index}>

              <Image className="bannerimg" style={ {width: '100%'} }  mode="aspectFit" src={'http://daxun.kuboy.top' + item} />

            </SwiperItem>

          ))

        }

      </Swiper>

    </View>

  )

}

```

8.2 实现列表的效果

1.自定义一个列表组件

新建components文件夹,按需存放各类组件

// src/components/prolist/index.jsx

import Taro from '@tarojs/taro'

import { View } from '@tarojs/components'

class Index extends Taro.Component {

  render () {

    return (

      <View>

        列表

      </View>

    )

  }

}

export default Index

8.2 首页引入列表组件测试

// src/pages/home/index.jsx

import Prolist from './../../components/prolist'

render () {

  return (

    <View>

      <Swiper indicatorDots autoplay circular>    //可以在官网查看swiper组件使用方法

        ...

      </Swiper>

      <Prolist />

    </View>

  )

}

8.3 构建组件的样式

// prolist/index.jsx

import Taro from '@tarojs/taro'

import { View, Image } from '@tarojs/components'

import './index.scss'

class Index extends Taro.Component {

  render () {

    return (

      <View className="prolist">

        <View className="proitem">

          <View className="itemimg">

            <Image className='img' src=''></Image>

          </View>

          <View className="iteminfo">

            <View className="title">item.proname</View>

            <View className="title">item.sales / item.stock</View>

            <View className="price">¥item.price</View>

          </View>

        </View>

      </View>

    )

  }

}

export default Index

// prolist/index.scss

.prolist .proitem {

  width: 100%;

  height: 100PX;

  display: flex;

  box-sizing: border-box;

  border-bottom: 1px solid #ccc;

}

/* gulp */

.prolist .proitem .itemimg {

  width: 100PX;

  height: 100PX;

}

.prolist .proitem .itemimg .img{

  width: 90PX;

  height: 90PX;

  box-sizing: border-box;

  /* 如果是网页开发,需要写一个物理像素 ---- scss库  ---- 1PX边框 */

  border: 1px solid #ccc;

  margin: 5PX;

}

.prolist .proitem .iteminfo {

  flex: 1;

  padding: 3PX 5PX;

}

8.4 首页请求数据,并且传递数据给子组件

//src/pages/home/index.jsx

class Index extends Component {

  config = {

    ...

  }

  constructor (props) {

    super(props)

    // 1.设置状态

    this.state = {

      bannerlist: [],

      prolist: [] 

    }

  }

  // 2.请求数据

  componentDidMount () {

    request({

      url: '/pro'

    }).then(res => {

      console.log(res.data)

      this.setState({

        prolist: res.data.data

      })

    })

  }

  render () {

    return (

      <View>

        <Swiper indicatorDots autoplay circular>

          ...

        </Swiper>

        {/* 

          在父组件调用子组件的地方,添加一个自定义的属性,属性的值就是要传递给子组件的值,如果值是一个变量,boolean,或者是number类,需要使用{}包裹

        */}

        <Prolist prolist={ this.state.prolist }/>

      </View>

    )

  }

}

export default Index

8.5 子组件接收数据并且校验数据

先行安装 prop-types 数据校验模块

$ cnpm i prop-types -S

//src/components/prolist/index.jsx

import Taro from '@tarojs/taro'

import PropTypes from 'prop-types'

import { View, Image } from '@tarojs/components'

import './index.scss'

/**

 * 子组件 使用 prop-types 进行数据的校验,校验完毕。

 * 在子组件(类组件)通过 this.props.自定义的属性名  就可以访问数据

 * 如果组件时函数式组件,通过 props.自定义的属性名  访问数据

 */

class Index extends Taro.Component {

  render () {

    return (

      <View className="prolist">

        ....

      </View>

    )

  }

}

// 校验数据格式

Index.propTypes = {

  prolist: PropTypes.array

}

export default Index

8.6 渲染列表数据

//src/components/prolist/index.jsx

import Taro from '@tarojs/taro'

import PropTypes from 'prop-types'

import { View, Image } from '@tarojs/components'

import './index.scss'

/**

 * 子组件 使用 prop-types 进行数据的校验,校验完毕。

 * 在子组件(类组件)通过 this.props.自定义的属性名  就可以访问数据

 * 如果组件时函数式组件,通过 props.自定义的属性名  访问数据

 */

class Index extends Taro.Component {

  render () {

    return (

      <View className="prolist">

        {

          this.props.prolist.map(item => (

            <View className="proitem" key={ item.proid }>

              <View className="itemimg">

                <Image className="img" src={ item.proimg }></Image>

              </View>

              <View className="iteminfo">

                <View className="title">{ item.proname }</View>

                <View className="title">{ item.sales } / { item.stock }</View>

                <View className="price">¥{ item.price }</View>

              </View>

            </View>

          ))

        }

      </View>

    )

  }

}

// 校验数据格式

Index.propTypes = {

  prolist: PropTypes.array

}

export default Index

```

### 8.7 下拉刷新 --- 不支持 h5模式

```

constructor (props) {

  super(props)

  // 1.设置状态

  this.state = {

    bannerlist: [],

    prolist: [],

    pageCode: 1 // ++++++++++++++++++++++++++

  }

}

onPullDownRefresh () { // 不支持H5模式

  request({

    url: '/pro'

  }).then(res => {

    console.log(res.data)

    this.setState({

      prolist: res.data.data,

      pageCode: 1 // 必须重置页码

    })

    Taro.stopPullDownRefresh() // 这句话一定要加,真机测试时一直处于加载状态

  })

}

```

### 8.8 上拉加载实现

```

onReachBottom () {

  request({

    url: '/pro',

    data: {

      pageCode: this.state.pageCode

    }

  }).then(res => {

    console.log(res.data)

    if (res.data.code === '10000') {

      Taro.showToast({

        title: '没有更多数据了'

      })

    } else {

      // 获取数据  处理数据  修改状态

      let prolist = this.state.prolist

      let pageCode = this.state.pageCode

      prolist = [...prolist, ...res.data.data]

      pageCode += 1

      this.setState({

        prolist,

        pageCode

      })

    }

  })

}

8.9 返回顶部

定位一个小图标于 右下角,点击小图标时 调用 taro提供的api返回顶部即可

当滚动条滚动到一定位置时 小图标出现

//src/pages/home/index.jsx

<View className="backtop" onClick={ () => {

  console.log('1111')

  Taro.pageScrollTo({

    scrollTop: 0,

    duration: 500

  })

} }>↑</View>

// pages/home/index.scss

.backtop {

  position: fixed;

  right: 10Px;

  bottom: 60Px;

  width: 30PX;

  height: 30PX;

  background-color: rgba(0,0,0,0.4);

  color: #fff;

  text-align: center;

  line-height: 30Px;

  border-radius: 50%;

}

9.列表进入详情并且渲染

9.1 设计一个详情页面

// pages/detail/index.jsx

import Taro, { Component } from '@tarojs/taro'

import { View, Image } from '@tarojs/components'

class Index extends Component {

  constructor (props) {

    super(props)

    this.state = {

      proid: '',

      proname: '',

      proimg: '',

      price: 0

    }

  }

  render () {

    return (

      <View>

        <Image src={ this.state.proimg }></Image>

        <View>{ this.state.proname }</View>

        <View>{ this.state.price }</View>

      </View>

    )

  }

}

export default Index

// src/app.jsx中在pages中注册页面 ---- 修改配置文件 ---- 配置文件修改需重启服务器

config = { // 最终编译成为 app.json

  pages: [ // 路由

    'pages/home/index',

    'pages/kind/index',

    'pages/cart/index',

    'pages/user/index',

    'pages/index/index',

    'pages/detail/index' // +++++++++++++

  ],

}  

9.2 列表跳转至详情 并且传递参数

// react 声明式跳转(Link, NavLink) react-router-dom

// react 编程式跳转 this.props.history .push() .replace() .goBack()

// 如果组件中 的this.props 没有history这个属性,那么如何编程式跳转

//    找到距离最近的父组件,传this.props给这个组件

//    withRouter react-router-dom

// taro 声明式跳转 ---  Navigator url指明路径 可以传递参数

// components/prolist/index.jsx

render () {

  return (

    <View className="prolist">

      {

        this.props.prolist.map(item => (

          <Navigator url={ '/pages/detail/index?proid=' + item.proid } className="proitem" key={ item.proid }>

            <View className="itemimg">

              <Image className="img" src={ item.proimg }></Image>

            </View>

            <View className="iteminfo">

              <View className="title">{ item.proname }</View>

              <View className="title">{ item.sales } / { item.stock }</View>

              <View className="price">¥{ item.price }</View>

            </View>

          </Navigator>

        ))

      }

    </View>

  )

}

9.3 详情页面获取数据

// react this.props 获取参数。

// taro  this.$router

```

// pages/detail/index.jsx

componentDidMount () {

  console.log(this.$router.params) // 获取参数信息

  const { proid } = this.$router.params

  request({

    url: '/pro/detail', // get请求的参数也可以放在data中

    data: {

      proid

    }

  }).then(res => {

    const { proname, proimg, price } = res.data.data

    this.setState({

      proname, proimg, price, proid

    })

    Taro.setNavigationBarTitle({// 修改详情的标题

      title: proname

    }) 

  })

}

9.4 编程式导航跳转

// react this.props.history. push replace goBack

// taro Taro.navigateTo Taro.switchTab Taro.redirectTo 

```

// components/prolist/index.jsx

class Index extends Taro.Component {

  render () {

    return (

      <View className="prolist">

        {

          this.props.prolist.map(item => (

            <View className="proitem" key={ item.proid } onClick={ () => {

              Taro.navigateTo({

                url: '/pages/detail/index?proid=' + item.proid

              })

            } }>

              <View className="itemimg">

                <Image className="img" src={ item.proimg }></Image>

              </View>

              <View className="iteminfo">

                <View className="title">{ item.proname }</View>

                <View className="title">{ item.sales } / { item.stock }</View>

                <View className="price">¥{ item.price }</View>

              </View>

            </View>

          ))

        }

      </View>

    )

  }

}

// 校验数据格式

Index.propTypes = {

  prolist: PropTypes.array

}

export default Index

10.登陆注册

10.1 设计登陆页面 

// pages/login/index.jsx

import Taro, { Component } from '@tarojs/taro'

import { View } from '@tarojs/components'

class Index extends Component {

  render () {

    return (

      <View>

        登陆

      </View>

    )

  }

}

export default Index

// app.jsx 注册路由

pages: [ // 路由

  'pages/home/index',

  'pages/kind/index',

  'pages/cart/index',

  'pages/user/index',

  'pages/index/index',

  'pages/detail/index',

  'pages/login/index' // +++++++++++

],

10.2 完成登陆的表单

安装Taro-ui库(同uni-app一样,最好不要使用第三方ui库)

 cnpm i taro-ui -S

config/index.js中h5的标识处添加如下代码

h5: {

  esnextModules: ['taro-ui']

}

// pages/login/index.jsx

// 找到taro-ui中的需要的组件,在jsx中导入组件,在scss文件中引入 需要的样式表

import Taro, { Component } from '@tarojs/taro'

import { View } from '@tarojs/components'

import { AtInput, AtForm, AtButton } from 'taro-ui'

import './index.scss'

class Index extends Component {

  constructor (props) {

    super(props)

    this.state = {

      tel: '12345678912',

      password: '123456'

    }

  }

  render () {

    return (

      <View>

        <AtForm>

          <AtInput

            name='tel'

            title='手机号码为11位'

            type='text'

            placeholder='手机号码'

            value={this.state.tel}

          />

          <AtInput

            name='password'

            title='密码长度不能少于6位'

            type='password'

            placeholder='密码'

            value={this.state.password}

          />

          <AtButton type='secondary' size='normal'>登陆</AtButton>

        </AtForm>

      </View>

    )

  }

}

export default Index

// pages/login/index.scss

@import "~taro-ui/dist/style/components/input.scss";

@import "~taro-ui/dist/style/components/icon.scss";

@import "~taro-ui/dist/style/components/button.scss";

@import "~taro-ui/dist/style/components/loading.scss";

10.3 表单校验

//pages/login/index.jsx

import Taro, { Component } from '@tarojs/taro'

import { View } from '@tarojs/components'

import { AtInput, AtForm, AtButton } from 'taro-ui'

import './index.scss'

class Index extends Component {

  constructor (props) {

    super(props)

    this.state = {

      tel: '12345678912',

      password: '123456',

      telflag: false,

      passwordflag: false

    }

  }

  render () {

    return (

      <View>

        <AtForm>

          <AtInput

            error={ this.state.telflag }

            name='tel'

            clear

            title='手机号码'

            type='text'

            placeholder='手机号码'

            value={this.state.tel}

            onErrorClick={ () => {

              console.log('手机号码长度为11位')

              Taro.showToast({

                title: '手机号码长度为11位',

                icon: 'none',

                duration: 5000

              })

            }}

            onChange={ (value) => {

              console.log(value)

              let telflag = this.state.telflag

              if (value.length !== 11) {

                telflag = true

              } else {

                telflag = false

              }

              this.setState({

                tel: value,

                telflag

              })

            } }

          />

          <AtInput

            error={ this.state.passwordflag}

            name='password'

            title='密码'

            clear

            type='password'

            placeholder='密码'

            value={this.state.password}

            onErrorClick={ () => {

              console.log('密码长度应该大于6')

              Taro.showToast({

                title: '密码长度应该大于6',

                icon: 'none',

                duration: 5000

              })

            }}

            onChange={ (value) => {

              // let passwordflag = this.state.passwordflag

              // if (value.length < 6) {

              //   passwordflag = true

              // } else {

              //   passwordflag = false

              // }

              let passwordflag = value.length < 6 ? true : false;

              this.setState({

                password: value,

                passwordflag

              })

            }}

          />

          {/* <AtButton disabled={ this.state.tel.length !== 11 || this.state.password.length < 6 } type='secondary' size='normal'>登陆</AtButton> */}

          {/* 推荐使用下面写法,因为你的业务逻辑可能是复杂的正则验证 */}

          <AtButton disabled={ this.state.telflag || this.state.passwordflag } type= { this.state.telflag && this.state.passwordflag ? 'secondary' : 'primary'} size='normal' >登陆</AtButton>

        </AtForm>

      </View>

    )

  }

}

export default Index

10.4 登陆

<AtButton disabled={ this.state.telflag || this.state.passwordflag } type= { this.state.telflag && this.state.passwordflag ? 'secondary' : 'primary'} size='normal' onClick={ () => {

  request({

    url: '/users/login',

    method: 'POST',

    data: {

      tel: this.state.tel,

      password: this.state.password

    },

    // 如果实际注册现实未注册,加上头信息

    header: {'content-type': "application/json; charset=utf-8"}

  }).then(res => {

    console.log(res)

    if (res.data.code === '10006') {

      Taro.showToast({

        title: '用户未注册,请先注册',

        icon: 'none',

        duration: 3000

      })

    } else if (res.data.code === '10007') {

      Taro.showToast({

        title: '密码错误',

        icon: 'none',

        duration: 3000

      })

    } else {

      Taro.showToast({

        title: '登陆成功',

        icon: 'none',

        duration: 3000

      })

      try {

        Taro.setStorageSync('userid', res.data.data.userid)

        Taro.setStorageSync('token', res.data.data.token)

        Taro.navigateBack()

      } catch (error) {


      }

    }

  })

} }>登陆</AtButton>

11.购物车

vue (uniapp)直接改变数据 ---  (获取数据,处理数据、修改状态)

11.1 加入购物车

// pages/detail/index.js

addCart () {

  try {

    let userid = Taro.getStorageSync('userid')

    let token = Taro.getStorageSync('token')

    console.log(userid, token)

    if (userid && token) {

      request({

        url: '/cart/add',

        method: 'POST',

        data: {

          userid,

          token,

          proid: this.state.proid,

          num: 1

        },

        // Taro Post

        header: {'content-type': "application/json; charset=utf-8"}

      }).then( res => {

        if (res.data.code === '10119') {

          Taro.showToast({

            title: '还未登陆,请先登陆',

            icon: 'none'

          })

          Taro.navigateTo({

            url: '/pages/login/index'

          })

        } else {

          Taro.showToast({

            title: '加入购物车成功',

            icon: 'none'

          })

        }

      })

    } else {

      Taro.showToast({

        title: '还未登陆,请先登陆',

        icon: 'none'

      })

      Taro.navigateTo({

        url: '/pages/login/index'

      })

    }

  } catch (error) {


  }

}

render () {

  return (

    <View>

      <Image src={ this.state.proimg }></Image>

      <View>{ this.state.proname }</View>

      <View>{ this.state.price }</View>

      <Button onClick={ this.addCart.bind(this) }>加入购物车</Button>

    </View>

  )

}

11.2 查看购物车

import Taro, { Component } from '@tarojs/taro'

// 为什呢 View 要单独引入,react 说明 组件的首字母一定要大写,小写被当做html标签

import { View, Checkbox, CheckboxGroup } from '@tarojs/components'

import { request } from './../../utils'

import './index.scss'

class Index extends Component {

  constructor (props) {

    super(props)

    this.state = {

      cartlist: [],

      isTrue: true,

      totalNum: 0,

      totalPrice: 0,

      allSelected: true

    }

  }

  componentDidShow () {

    try {

      // 本地判断是否登陆

      let userid = Taro.getStorageSync('userid')

      let token = Taro.getStorageSync('token')

      if ( userid && token) {

        request({

          url: '/cart',

          data: {

            userid,

            token

          }

        }).then( res => {

          console.log(res.data.code)

          if (res.data.code === '10119') {

            Taro.showToast({

              title: '还未登陆,请先登陆',

              icon: 'none'

            })

            Taro.navigateTo({

              url: '/pages/login/index'

            })

          } else if (res.data.code === '10112') {

            Taro.showToast({

              title: '购物车空空如也,请加购',

              icon: 'none'

            })

            this.setState({

              isTrue: true

            })

          } else {

            res.data.data.map( item => {

              item.flag = true

            })

            this.setState({

              isTrue: false,

              cartlist: res.data.data

            }, () => {

              this.count() // 用来计算总价和总数

            })


          }

        })

      } else {

        Taro.showToast({

          title: '还未登陆,请先登陆',

          icon: 'none'

        })

        Taro.navigateTo({

          url: '/pages/login/index'

        })

      }

    } catch (error) {


    }

  }

  count () {

    console.log(this.state.cartlist)

    let num = 0

    let price = 0

    this.state.cartlist.map( item => { // 选中才会计算

      item.flag ? num += item.num : num += 0

      item.flag ? price += item.num * item.price : price += 0

    })

    this.setState({

      totalNum: num,

      totalPrice: price

    })

  }

  render () {

    const { cartlist= [] } = this.state

    return (

      <View>

        {

          this.state.isTrue ? <View>购物车空空如也,请加购</View> : <View>

            <CheckboxGroup onChange={

              (event) => {

                console.log(event.detail)

                // 获取长度  表明是否被选中

                let len = event.detail.value.length

                console.log(len)

                // 通过长度获取到 选中和不选中变量

                let flag = len === 1 ? true : false

                // 获取列表数据

                let list = this.state.cartlist

                // 处理数据

                list.map(item => {

                  flag ? item.flag = true : item.flag = false

                })

                // 修改状态

                this.setState({

                  allSelected: flag,

                  cartlist: list

                }, () => { // 计算总价和总数

                  this.count()

                })

              }

            }>

              <Checkbox checked={ this.state.allSelected } />全选

            </CheckboxGroup>

            {

              cartlist.map((item, index) => {

                return (

                  <View key={ item.proid }>

                  <CheckboxGroup onChange={

                    (event) => {

                      // console.log(event.detail.value)

                      // 根据选中的数组的长度判断是否被选中

                      let len = event.detail.value.length

                      // console.log(len)

                      // console.log('index', index)

                      // 设置标识 表明当前的这个是不是被选中

                      let flag = len === 1 ? true : false

                      // 获取数据 准备处理以及修改数据用

                      let list = this.state.cartlist

                      // list[index].flag = flag

                      console.log('flag', flag)

                      // 如果当前的数据被选中

                      if (flag === true) {

                        // console.log('111', list[index].flag)

                        // 处理当前的数据源中的标识为选中状态

                        list[index].flag = true

                        // 检测其他的选项是否都是选中状态

                        let test = list.every(val => {

                          console.log('val', val.flag)

                          return val.flag === true

                        })

                        // console.log('test', test)

                        // 如果都被选中

                        if (test) {

                          // 全选设置为选中状态,更新状态,计算总价总数

                          this.setState({

                            allSelected: true,

                            cartlist: list

                          }, () => {

                            this.count()

                          })

                        } else {

                          // 其余项有没被选中,更新状态,计算总价总数

                          this.setState({

                            allSelected: false,

                            cartlist: list

                          }, () => {

                            this.count()

                          })

                        }

                      } else {

                        // 点击当前未被选中,需要将当前的数据值为 false

                        list[index].flag = false

                        // 更新状态,计算总价总数

                        this.setState({

                          allSelected: false,

                          cartlist: list

                        }, () => {

                          this.count()

                        })

                      }

                    }

                  }>

                    <Checkbox checked={item.flag}/>

                  </CheckboxGroup>


                  { item.proname } - { item.price } - 

                  ...

                  </View>

                )

              })

            }

            <View>

              总数: { this.state.totalNum }

            </View>

            <View>

              总价: { this.state.totalPrice }

            </View>

          </View>

        }

      </View>

    )

  }

}

export default Index


大致流程就是这样,新手上路,不喜勿喷~谢谢大家!(点击返回顶部~~~~手动滑稽)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351