Taro Taro, a team partner – A ji, about the use of hands-on practice of an article, I hope to help you use.
1. Introduction of Taro
Taro is a multi-terminal development framework developed by Bump Lab that follows the React syntax specification.
Using Taro, we can only write one set of code, and then compile the source code separately into codes that can run on different ends (wechat applet, H5, App, etc.) through Taro’s compilation tool. Taro currently supports compiling codes that support wechat applet and H5 running. Support for RN, Kuaibao and Alipay applet is still under development.
Detailed see article multiterminal unified development framework – Taro, and making https://github.com/NervJS/taro warehouse
2. The preface
In order to learn Taro, I found in Github zhihu small program demo, this article by modifying the code, to achieve the Taro version of Zhihu H5, small program demo, students interested in Taro can star or fork down to learn, Github address.
3. The installation
Install the Taro development tool @tarojs/cli
Use NPM or YARN for global installation
NPM install -g@tarojs /cli // or YARN global add @tarojs/cliCopy the code
Download the code
git clone https://github.com/jimczj/taro_zhihu
# install dependencies
cd taro_zhihu
npm i
Copy the code
Run the code
The file directories are as follows:
├ ─ ─ dist directory compiled results ├ ─ ─ the config configuration directory | ├ ─ ─ dev. The development of a js configuration | ├ ─ ─ index. The default configuration | js └ ─ ─ the prod. Js packaging configuration ├ ─ ─ the SRC source directory | ├ ─ ─ pages Page file directory | | ├ ─ ─ index index page directory | | | ├ ─ ─ index. The js index page logic | | | └ ─ ─ index. The CSS style index page | ├ ─ ─ app. The CSS style of total project general | └ ─ ─ ├ ── package.json app.js project entry fileCopy the code
Enter the project directory to start development, you can choose the mini program preview mode or H5 preview mode. If you use the wechat mini program preview mode, you need to download and open the wechat developer tool by yourself, and select the preview project root directory.
Wechat applet compilation preview mode:
# npm script
npm run dev:weapp
# or global installation only
taro build --type weapp --watch
Copy the code
H5 Compiler preview mode:
# npm script
npm run dev:h5
# or global installation only
taro build --type h5 --watch
Copy the code
5. Pay attention before development
If you use the wechat applet preview mode, you need to download and use the wechat developer tool to add projects for preview. At this time, you need to pay attention to the project Settings of the wechat developer tool
- You need to disable the ES6 to ES5 function. If this function is enabled, an error may occur
- Need to set to turn off the upload code style automatic completion, enable may report an error
- You need to set the disable code compression upload, enable may report an error
6. Detailed explanation of function implementation
6.1 Applet Global Configuration
The app.json file is used for global configuration of wechat applets, determining the path of page files, window performance, setting network timeout, setting multiple tabs, etc. There is almost no cost to convert the original app.json to the Taro project’s app.js, and the configuration is basically the same. However, it is important to note that the static image resources folder should be placed under the SRC directory so that the code will copy and package the image resources when compiling and packaging. I started with the static image resource folder at the same level as SRC, and then couldn’t find the image resource and wasted a lot of time. Well, without further ado, if you look at the code below, you can see how much easier the migration is, and how much easier it looks when you write React.
Original wechat small programapp.jsonThe code is as follows:
{
'pages': ['pages/index/index'.'pages/discovery/discovery'.'pages/more/more'.'pages/answer/answer'.'pages/question/question'].'window': {'backgroundTextStyle':'light'.'navigationBarBackgroundColor': '#0068C4'.'navigationBarTitleText': 'zhihu'.'navigationBarTextStyle':'white'.'enablePullDownRefresh':true
},
'tabBar': {
'color': '# 626567'.'selectedColor': '#2A8CE5'.'backgroundColor': '#FBFBFB'.'borderStyle': 'white'.'list': [{
'pagePath': 'pages/index/index'.'text': 'home'.'iconPath': 'images/index.png'.'selectedIconPath': 'images/index_focus.png'
}, {
'pagePath': 'pages/discovery/discovery'.'text': 'found'.'iconPath': 'images/discovery.png'.'selectedIconPath': 'images/discovery_focus.png'
}, {
'pagePath': 'pages/more/more'.'text': 'I'.'iconPath': 'images/burger.png'.'selectedIconPath': 'images/burger_focus.png'}}}]Copy the code
Translate the Taro code as follows:
import Taro, { Component } from '@tarojs/taro'
import Index from './pages/index'
import './app.scss'
class App extends Component {
config = {
pages: [
'pages/index/index'.'pages/discovery/discovery'.'pages/more/more'.'pages/answer/answer'.'pages/question/question'].window: {
backgroundTextStyle: 'light'.navigationBarBackgroundColor: '#0068C4'.navigationBarTitleText: 'Taro zhihu'.navigationBarTextStyle: 'white'.enablePullDownRefresh: true
},
tabBar: {
color: '# 626567'.selectedColor: '#2A8CE5'.backgroundColor: '#FBFBFB'.borderStyle: 'white'.list: [{
pagePath: 'pages/index/index'.text: 'home'.iconPath: './asset/images/index.png'.selectedIconPath: './asset/images/index_focus.png'}, {pagePath: 'pages/discovery/discovery'.text: 'found'.iconPath: './asset/images/discovery.png'.selectedIconPath: './asset/images/discovery_focus.png'
},
{
pagePath: 'pages/more/more'.text: 'I'.iconPath: './asset/images/burger.png'.selectedIconPath: './asset/images/burger_focus.png'
}]
}
}
render () {
return (
<Index />
)
}
}
Taro.render(<App />, document.getElementById('app'))
Copy the code
6.2 the home page
The page looks like this:
Function description: slide up refresh, pull down load more, data request, click jump
The action of refreshing and continuing to load depends on the ScrollView component, And bind onScrolltoupper and onScrolltolower on the component to bind events triggered by scrolling to the top and bottom. Meanwhile, upperThreshold and lowerThreshold can adjust the distance from the boundary when triggered.
Data request API uses Taro. Request, which is basically the same as wx. Request, except that Taro. Request naturally supports promise, and mock data is provided using easy-Mock.
NavigateTo is basically the same as wx.navigateTo. When writing the jump, I wanted to use the form of anonymous function to write, but FOUND that Taro does not support it yet, it is said that it will support, Taro’s iteration speed is still very fast.
When redesigning the homepage, the most time-consuming problem should be the conversion of WXSS to SCSS, and the conversion of Taro components into wechat applet and Web, the tags are different. For example, the Image component will be changed into Image or IMG tags, but the original WXSS uses tag names to name CSS. In this way, the micro channel applet is inconsistent with the web style. Therefore, you are advised to use class instead of label names.
export default class Index extends Component {
config = {
navigationBarTitleText: 'home'
}
constructor() {
super(... arguments)this.state = {
loading:true.list:[]
}
}
componentDidMount () {
// Get remote data
this.updateList()
}
updateList() {
Taro.showLoading({title: 'Loading'})
Taro.request({
url: 'https://easy-mock.com/mock/5b21d97f6b88957fa8a502f2/example/feed'
}).then(res= > {
Taro.hideLoading()
if (res.data.success) {
this.setState({
loading:false.list:res.data.data
})
}
})
}
appendNextPageList() {
Taro.showLoading({title: 'Loading'})
Taro.request({
url: 'https://easy-mock.com/mock/5b21d97f6b88957fa8a502f2/example/feed'
}).then(res= > {
Taro.hideLoading()
if (res.data.success) {
this.setState({
list: this.state.list.concat(res.data.data)
})
}
})
}
render () {
return(<ScrollView className='container' scrollY scrollWithAnimation scrollTop='0' lowerThreshold='10' upperThreshold='10' onScrolltoupper={this.updateList} onScrolltolower={this.appendNextPageList} > <View className='search flex-wrp'> <View className='search-left flex-item'> <View className='flex-wrp'> <View className='flex1'><Image SRC ={searchPng}></Image></View> <View className='flex6'><Input type='text' placeholder=' 'placeholderClass='search-placeholder'/></View> </View> </View> </View> <View className='search-right flex-item'> <Image src={lightingPng}></Image> </View> </View> { this.state.loading ? <View className='txcenter'><Text> Load </Text></View> : this.state.list.map((item,index)=>{ return <Feed key={item} />}) } </ScrollView> ) } }Copy the code
6.3 Taro components
In fact, I have never understood that the Component of wechat applets is different from the Page life cycle function. Page life cycle methods include onLoad, onReady, onUnload, etc. In contrast, created, attached, and ready are created in components. Taro is relatively unified. Both pages and components are written in the same way as React life cycle, and unified API development is much smoother.
The Taro component does not support this. Props. Children. Props cannot pass JSX;
When abstracting components, the following are the main considerations
- In writing,
Taro
Components should be capitalized and humped, as inwxml
The label on the inside isView, scrollview, image
In theTaro
To be writtenView, ScrollView, Image
.Taro
The event bindings of theon
Start with a hump nomenclature
// Small program code
<scroll-view scroll-y='true' class='container' bindscrolltoupper='upper' upper-threshold='10' lower-threshold='5' bindscrolltolower='lower'>
</scroll-view>/ / Taro code<ScrollView className='container'
scrollY
scrollWithAnimation
scrollTop='0'
lowerThreshold='10'
upperThreshold='10'
onScrolltoupper={this.upper.bind(this)}
onScrolltolower={this.lower.bind(this)}
>
</ScrollView>
Copy the code
- Applets reference local static resources directly in
src
I’ll write the relative path,Taro
Local static resources need to be referenced firstimport
In order to ensure that h5 deployment does not fail the image path, it is best to put the image on the server, and then write HTTP path directly
// Applets reference local static resources
<image src='.. /.. /images/search.png'></image>
// Taro references local static resources
import searchPng from '.. /.. /asset/images/search.png'
/ /... Countless code omitted here
<Image src={searchPng}></Image>
// It is better to put the image on the server and write the HTTP path
<Image src='https://image.ibb.co/kUissy/search.png'></Image>
Copy the code
- Traversal list, applets using template language, while
Taro
usejsx
/ / small program < block wx: for = '{{feed}}' wx: the for - index = 'independence idx' wx: the for - item = 'item' data - independence idx = "{{independence idx}}" > < view class = 'feed - item >... </view> </block> // Taro code {this.state.list.map((item,index)=>{return <Feed {... item} key={index} /> }) }Copy the code
In the process of abstraction, the answer component wants to be written directly in pure functional form
Const Text = ({answer}) => <p>{answer}</p>Copy the code
It is not supported yet, so we write it as class:
export default class Feed extends Component {
navigateTo(url) {
Taro.navigateTo({url:url})
}
render() {
return (
<View className='feed-item'>
<View className='feed-source'>
<View className='avatar flex1'>
<Image src={this.props.feed_source_img}></Image>
</View>
<View className='flex8'>
<Text className='feed-source-txt'>{this.props.feed_source_name}{this.props.feed_source_txt}</Text>
</View>
<View className='flex1'>
<Image className='item-more' mode='aspectFit' src={more}></Image>
</View>
</View>
<View className='feed-content'>
<View className='question' onClick={this.navigateTo.bind(this,'/pages/question/question')} >
<View className='question-link'>
<Text>{this.props.question}</Text>
</View>
</View>
<View className='answer-body'>
<View>
<Text className='answer-txt' onClick={this.navigateTo.bind(this,'/pages/answer/answer')} >{this.props.answer_ctnt}</Text>
</View>
<View className='answer-actions'>
<View className='like dot'>
<View>{this. Props. Good_num}</View>
</View>
<View className='comments dot'>
<View>Comment ment_num {this.props.com}</View>
</View>
<View className='follow-it'>
<View>Pay attention to the problem</View>
</View>
</View>
</View>
</View>
</View>)}}Copy the code
When using a component, you want to use {… Item} to deconstruct the assignment, but found that it was not supported for the time being.
this.state.list.map((item,index) = > {
return <Feed
feed_source_img={item.feed_source_img}
feed_source_txt={item.feed_source_txt}
question={item.question}
answer_ctnt={item.answer_ctnt}
good_num={item.good_num}
comment_num={item.comment_num}
key={index} />
})
Copy the code
6.4 TAB Switching in Discover Page
TAB switching principle:
onClick
this.state.currentNavtab
tab
this.switchTab.bind(this,index)
export default class Discovery extends Component {
constructor() {
super(... arguments)this.state = {
currentNavtab: 0.navTab: ['recommendations'.'round'.'hot'.'collection'],
}
}
switchTab(index,e) {
this.setState({
currentNavtab: index
});
}
render () {
return (
<View>
<View className='top-tab flex-wrp flex-tab' >
{
this.state.navTab.map((item,index) => {
return (<View className={this.state.currentNavtab= = =index ? 'toptab flex-item active' : 'toptab flex-item'}key={index} onClick={this.switchTab.bind(this,index)}>
{item}
</View>)})}</View>
<ScrollView scroll-y className='container discovery withtab'>
<View className='ctnt0' hidden={this.state.currentNavtab= =0 ? false : true} >.</View>
<View className='txcenter' hidden={this.state.currentNavtab= =1 ? false : true} >
<Text>The round table</Text>
</View>
<View className='txcenter' hidden={this.state.currentNavtab= =2 ? false : true} >
<Text>hot</Text>
</View>
<View className='txcenter' hidden={this.state.currentNavtab= =3 ? false : true} >
<Text>collection</Text>
</View>
</ScrollView>
</View>)}}Copy the code
6.5 Round play function
Swiper is used in the Swiper page, and the parameters are one-to-one corresponding to the small program. For details, you can see the detailed documents. In the reconstruction process, WXML is mainly changed into THE form of JSX.
<Swiper className='activity' indicatorDots='true'
autoplay='true' interval='5000' duration='500'>
{this.state.imgUrls.map((item,index) = > {
return (<SwiperItem key={index}>
<Image src={item} className='slide-image' width='355' height='375' />
</SwiperItem>)})}</Swiper>
Copy the code
7. Summary of problems found in use
Taro
The components converted into wechat applets are not the same as the tags after the Web, for exampleImage
The component becomesimage
orimg
Tag. Therefore, you are not advised to use the label name of the CSS. You are advised to use class instead- Current event bindings do not support anonymous functions
<View onClick={()=> this. NavigateTo ()'/pages/answer/answer')} > </View>
Copy the code
Taro
The component has many properties and events that are not currently supported on the Web side- Some apis need to be handled differently depending on the operating environment, for example
wx.getUserInfo
It doesn’t exist on the Web side, so we need to determine the environment to execute the code
if (Taro.getEnv() === Taro.ENV_TYPE.WEAPP) {
// Applet environment
} else if (Taro.getEnv() === Taro.ENV_TYPE.WEB ) {
/ / WEB (H5) environment
}
Copy the code
Taro
Not currently supportedSVG
There are few third-party libraries out there (after allTaro
It’s just coming out. Hopefully there will be various UI libraries in the community.Taro
Components currently do not support pure functions and do not support destruct assignments{... item}
8. To summarize
Migrating the project to Taro was not costly, and most of the work involved syntax changes. Taro uses the React development method to write wechat small program, the overall experience is very good, have React development experience friends believe can quickly start. It is amazing that the same code can support both the Web side and the applet side. Although the support for the Web side is still to be improved, the direction is right, and it is only a matter of time. Look forward to Taro’s next performance. Finally, if you are interested in the project, you can download the code and run the GitHub address.