30-days-of-react-native源码分析

Github地址

入口-index.ios.js

Swiper插件

import Swiper from 'react-native-swiper';
<Swiper height={150} showsButtons={false} autoplay={true}  activeDot={}>
 //Todo
</Swiper>

Day01 - A stopwatch

设置顶部状态栏

import {StatusBar } from 'react-native';
...
componentDidMount() {
    StatusBar.setBarStyle(0);
  }

Day02 - A weather app

Day03 - Twitter

createAnimatedComponent使得任何一个React组件支持动画。用它来创建Animated.View等等。
加载完成之后页面图标放大淡出动画:

const AnimatedIcon = Animated.createAnimatedComponent(Icon);
......
componentDidMount() {
    Animated.timing(
       this.state.transformAnim,    
       {toValue: 50,
        duration: 1200,
        delay:2000,
        easing: Easing.elastic(2),
      },          
    ).start();
    Animated.timing(         
       this.state.opacityAnim,    
       {toValue: 0,
        duration: 800,
        easing: Easing.elastic(1),
        delay:2200,
      },          
     ).start();
    setTimeout(() => {
      this.props.hideThis();
    }, 3300);              
  }

  render () {
    return(
      <Animated.View style={[styles.entrance,{opacity:this.state.opacityAnim}]}>
        <AnimatedIcon size={60} style={[styles.twitter,{transform:[{scale:this.state.transformAnim}]}]} name="logo-twitter"></AnimatedIcon>
      </Animated.View>
    )
  }

scrollView下拉刷新

 <ScrollView
      refreshControl={
          <RefreshControl
              //决定加载进去指示器是否为活跃状态,也表明当前是否在刷新中
              refreshing={this.state.isRefreshing}  
              //当视图开始刷新的时候调用
              onRefresh={()=>this._onRefresh()}
              // iOS平台适用  设置加载进度指示器的颜色
              tintColor="red" 
          />
        }
      >
           ......
</ScrollView>

Day04 - Cocoapods

Day05 - Find my location

MapView组件在React Native0.42以后已被淘汰,使用react-native-maps模块替代。

Day06 - Spotify

import Video from 'react-native-video';

Day07 - Moveable Circle

手势系统——拖拽组件

class MoveableCircle extends Component{
  constructor() {
    super();
    this.state = {
      color: "rgba(255,255,255,0.7)",
    };
  }

  // 初始left值
  _previousLeft = Util.size.width/2-40;
  // 初始top值
  _previousTop = Util.size.height/2-50;
  //最大top值
  _maxTop = Util.size.height-110;
  //最大left值
  _maxLeft = Util.size.width-98;
  _circleStyles = {};
  //http://stackoverflow.com/questions/32721630/javascript-syntax-null
  circle = (null : ?{ setNativeProps(props: Object): void });

  _updatePosition() {
    this.circle && this.circle.setNativeProps(this._circleStyles);
  }

  _endMove(evt, gestureState) {
    this._previousLeft += gestureState.dx;
    this._previousTop += gestureState.dy;
    this.setState({
      color: "rgba(255,255,255,0.7)"
    });
  }

  componentWillMount() {
    this._panResponder = PanResponder.create({
        //要求成为响应者:
        onStartShouldSetPanResponder: (evt, gestureState) => true,
        onStartShouldSetPanResponderCapture: (evt, gestureState) => true,
        onMoveShouldSetPanResponder: (evt, gestureState) => true,
        onMoveShouldSetPanResponderCapture: (evt, gestureState) => true,

        // 开始手势操作。给用户一些视觉反馈,让他们知道发生了什么事情!
        onPanResponderGrant: (evt, gestureState) => {
          this.setState({
            color: "white",
          })
        },

        onPanResponderMove: (evt, gestureState) => {
            // 最近一次的移动距离为gestureState.move{X,Y}
            // 从成为响应者开始时的累计手势移动距离为gestureState.d{x,y}
           this._circleStyles.style.left = this._previousLeft + gestureState.dx;
           this._circleStyles.style.top = this._previousTop + gestureState.dy;
           if (this._circleStyles.style.left<0) {
              this._circleStyles.style.left = 0;
           };
           if (this._circleStyles.style.top<5) {
              this._circleStyles.style.top = 5;
           };
           if (this._circleStyles.style.left>this._maxLeft) {
              this._circleStyles.style.left = this._maxLeft;
           };
           if (this._circleStyles.style.top>this._maxTop) {
              this._circleStyles.style.top = this._maxTop;
           };
           this._updatePosition();
        },
        onPanResponderTerminationRequest: (evt, gestureState) => true,

        // 用户放开了所有的触摸点,且此时视图已经成为了响应者。
        // 一般来说这意味着一个手势操作已经成功完成。
        onPanResponderRelease: (evt, gestureState) => this._endMove(evt, gestureState),
        // 另一个组件已经成为了新的响应者,所以当前手势将被取消。
        onPanResponderTerminate: (evt, gestureState) => this._endMove(evt, gestureState),
    });

    this._circleStyles = {
      style: {
        left: this._previousLeft,
        top: this._previousTop,
      },
    };

  }

  componentDidMount() {
    this._updatePosition();
  }

  render() {
    return(
      <View ref={(circle) => {this.circle = circle;}} style={styles.MoveableCircle} {...this._panResponder.panHandlers}>
        <Icon ref="baseball" name="ios-baseball" color={this.state.color} size={120}></Icon>
      </View>
    )
  }
}

Day08 - Swipe Left Menu

手势系统——DrawerNavigator(⭐️⭐️⭐️)

Day09 - Twitter Parallax View

手势系统——位置、尺寸、透明度对着拖动而渐变(⭐️⭐️⭐️⭐️)

class TwitterUser extends Component{
    constructor() {
        super();
        this.state = {
            scrollEnabled: false, //scrollView是否可以滚动(scrollView与最外层view只有一个可以滚动)
            scale: 1,             //图标的初始scale值
            iconTop: 95,          //图标的初始top值
            bannerTop:0,          //banner的初始top值
            opacity:0,            //view的初始opacity值
        };

        this._scrollEnabled = false; //scrollView是否可以滚动

        this._previousTop = 0;  //上一次拖动之后的最外层view的top值
        this._iconTop = 95;     //上一次拖动之后图标的top值
        this._scale = 1;        //上一次拖动之后图标的scale值
        this._bannerTop = 0;    //上一次拖动之后banner的top值
        this._opacity = 0;      //上一次拖动之后banner的opacity值
        this._minTop = -192;    //最外层view的最小top值
        this._userStyle = {};   //最外层view的style样式
        this.user = (null : ?{ setNativeProps(props: Object): void }); //对最外层view进行flow类型检查
    }

    //最外层view作为手势的响应者,使用setNativeProps直接修改属性
    _updatePosition() {
       this.user && this.user.setNativeProps(this._userStyles);
    }

    //手指松开当前最外层view的响应者,或者有其他组件成为新的响应者时,执行该函数
    _endMove(evt, gestureState) {
        //使用最新的top值赋值给this._previousTop
        this._previousTop = this._userStyles.style.top;
    }

    componentWillMount() {
        //组件将要渲染时,创建手势系统
        this._panResponder = PanResponder.create({
            //要求成为响应者:
            onStartShouldSetPanResponder: (evt, gestureState) => true,
            onStartShouldSetPanResponderCapture: (evt, gestureState) => true,
            onMoveShouldSetPanResponder: (evt, gestureState) => {
                return gestureState.dy/gestureState.dx!=0;
            },
            // 开始手势操作。
            onPanResponderGrant: (evt, gestureState) => {
                //给用户一些视觉反馈,让他们知道发生了什么事情!
                // gestureState.{x,y}0 现在会被设置为0
            },
            onPanResponderMove: (evt, gestureState) => {
                // 最近一次的移动距离为gestureState.move{X,Y}
                // 从成为响应者开始时的累计手势移动距离为gestureState.d{x,y}
                // 外层view的top值为0 + gestureState.dy   注意gestureState.dy为负值
                this._userStyles.style.top = this._previousTop + gestureState.dy;
                //图标的尺寸scale
                this._scale = 1 + this._userStyles.style.top/162.5;
                //图标的top值
                this._iconTop = 95 - this._userStyles.style.top/4.16;
                //banner的top值
                this._bannerTop = 0;
                //view的opacity值
                this._opacity = 0;
                this._scrollEnabled = false;

                if (this._userStyles.style.top< -62.5) {
                    this._scale = 0.6;
                    this._iconTop = 110;
                    this._bannerTop = -this._userStyles.style.top-62.5;
                    //Math.pow(x, y)得到x的y次幂
                    this._opacity = Math.pow((-this._userStyles.style.top-62.5)/129.5,0.5)
                };
                if (this._userStyles.style.top>0) {
                    this._userStyles.style.top = 0;
                    this._scale = 1;
                    this._iconTop = 95
                }
                if (this._userStyles.style.top < this._minTop) {
                    this._userStyles.style.top = this._minTop;
                    this._opacity = 1;
                    this._bannerTop = 129.5;
                    this._scrollEnabled = true;
                };
                this.setState({
                    scrollEnabled: this._scrollEnabled,
                    scale: this._scale,
                    iconTop: this._iconTop,
                    bannerTop: this._bannerTop,
                    opacity: this._opacity
                });
                this._updatePosition();
            },
            onPanResponderTerminationRequest: (evt, gestureState) => true,
            // 用户放开了所有的触摸点,且此时视图已经成为了响应者。
            // 一般来说这意味着一个手势操作已经成功完成。
            onPanResponderRelease: (evt, gestureState) => this._endMove(evt, gestureState),
            // 另一个组件已经成为了新的响应者,所以当前手势将被取消。
            onPanResponderTerminate: (evt, gestureState) => this._endMove(evt, gestureState),
            onShouldBlockNativeResponder: (event, gestureState) => true,
        });

        this._userStyles = {
            style: {
                top: this._previousTop,
            },
        };
    }

    componentDidMount() {
        this._updatePosition();
    }

    render () {
        let panProps = this.state.scrollEnabled?{}:{...this._panResponder.panHandlers};
        return(
            <View ref={(user) => {this.user = user}} style={styles.userContainer} {...panProps}>
                 <View style={styles.userPanel}>
                     <Image style={[styles.banner,{top: this.state.bannerTop}]} source={{uri:'banner'}} />
                     <View style={[styles.iconContainer,{top:this.state.iconTop,transform:[{scale:this.state.scale}]}]}>
                        <Image style={styles.icon} source={{uri:"icon"}}></Image>
                     </View>
                     <View style={styles.userControl}>
                        <TouchableHighlight style={styles.controlIcon}>
                            <Icon name="ios-settings" color="#8999a5" size={20}></Icon>
                         </TouchableHighlight>
                        <TouchableHighlight style={styles.controlBtn}>
                          <Icon name="ios-people" color="#8999a5" size={20}></Icon>
                        </TouchableHighlight>
                        <TouchableHighlight style={styles.controlBtn2}>
                          <Text style={styles.controlBtnText}>编辑个人资料</Text>
                        </TouchableHighlight>
                     </View>
                     <View style={styles.userInfo}>
                         <Text style={styles.userInfoName}>Github</Text>
                         <Text style={styles.userInfoAccount}>@Github</Text>
                         <View style={styles.userInfoFollow}>
                             <Text style={styles.userInfoFollowing}><Text style={styles.fontEm}>183</Text> 正在关注</Text>
                             <Text style={styles.userInfoFollower}><Text style={styles.fontEm}>830k</Text> 关注者</Text>
                         </View>
                     </View>
                      {this.state.bannerTop<=0?<View></View>:<Image style={[styles.banner,{top: this.state.bannerTop}]} source={{uri:'banner'}}></Image>}
                      {this.state.bannerTop<=0?<View></View>:<Image style={[styles.banner,{top: this.state.bannerTop, opacity:this.state.opacity}]} source={{uri:'bannerBlur'}}></Image>}
                      <Text style={{position:"absolute",left:Util.size.width/2-30, fontSize:20, fontWeight:"500", top: this.state.bannerTop+90,opacity:this.state.opacity, backgroundColor:"transparent", color:"#fff"}}>Github</Text>
                      <View style={styles.segment}>
                        <SegmentedControlIOS values={['推文', '媒体', '喜欢']}  selectedIndex={0} tintColor="#2aa2ef"/>
                      </View>
                 </View>
                <ScrollView contentInset={{top:0}} style={styles.detailScroll} scrollEnabled={this.state.scrollEnabled}>
                    <View style={{width:Util.size.width,backgroundColor:"#f5f8fa"}}>
                        <Image style={{width:Util.size.width, height:0.835*Util.size.width, resizeMode:"contain"}} source={{uri:'moreinfo'}}></Image>
                    </View>
                </ScrollView>
            </View>
        )
    }
}

Day10 - Tumblr Menu

Easing的使用

export default class extends Component{
  constructor() {
    super();
    this.state = {
      shift: new Animated.Value(-120),
      show:false,
    };
  }

  _pushMenu() {
    this.setState({
      show: true,
    });

    Animated.timing(         
       this.state.shift,    
       {toValue: Util.size.width === 375? 50:30,
        duration: 200,
        delay:100,
        easing: Easing.elastic(1),
      },          
    ).start();
  }

  _popMenu() {
    Animated.timing(         
       this.state.shift,    
       {toValue: -120,
        duration: 200,
        delay:100,
        easing: Easing.elastic(1),
      },          
    ).start();

    setTimeout(()=>{
      this.setState({
        show: false,
      })
    },500);
  }

  componentDidMount() {
    StatusBar.setBarStyle(1);
  }
  ......
}

Day11 - Using OpenGL with React native

gl-react-native

Day12 - Charts

Day13 - A twitter tweet UI

Day14 - Tinder Like Swipe

react-native-tinder-swipe-cards

Day15 - Time picker

Modal组件可以用来覆盖包含React Native根视图的原生视图(如UIViewController,Activity)。在嵌入React Native的混合应用中可以使用Modal。Modal可以使你应用中RN编写的那部分内容覆盖在原生视图上显示。

 <Modal  animationType="slide" transparent={false} visible={this.state.showModal}>
      ······
</Modal>

Day16 - Unlock with gesture

react-native-gesture-password

Day17 - Native search bar and Fuzzy search

react-native-search-bar
f使用 for in 循环遍历对象的属性时,原型链上的所有属性都将被访问,推荐总是使用hasOwnProperty方法, 这将会避免原型对象扩展带来的干扰。

const stateData = {a:1 ,b:2,c:3}
this.states = [];
for (let key in stateData) {
  if (stateData.hasOwnProperty(key)) {
    this.states.push(stateData[key]);
  }
}

Day 18 - Sortable List

手势系统——拖拽排序(⭐️⭐️⭐️⭐️⭐️)
代码
LayoutAnimation的使用
官网文档
React Native超棒的LayoutAnimation(布局动画)
React Native API模块之LayoutAnimation布局动画详解

this.animations = {
  duration: 200,
  create: {
    type: LayoutAnimation.Types.linear,
  },
  update: {
    type: LayoutAnimation.Types.linear,
    springDamping: 0.7,
  },
};
...
if (this.finalIndex != this.index) {
  this.index = this.finalIndex;
  this.setState({
    selected: this.finalIndex,
  });
}
LayoutAnimation.configureNext(this.animations);
...
box.setNativeProps({style: {top,left,...shadowStyle}});
LayoutAnimation.configureNext(this.animations);

Day 19 - Unlock app with touchID

react-native-touch-id

Day 20 - Sigle page Reminder

LayoutAnimation的使用

this.animations = {
  duration: 200,
  create: {
    type: LayoutAnimation.Types.linear,
  },
  update: {
    type: LayoutAnimation.Types.linear,
    springDamping: 0.7,
  },
};
......
this.setState({
  listData,
  numOfItems,
});
LayoutAnimation.configureNext(this.animations);

Day21 - Multi page Reminder

Day22 - Google Now

Day23 - Local WebView An example using D3.js

Day 24 - Youtube scrollable tab

react-native-scrollable-tab-view

Day25 - TO BE UPDATED

Day26 - TO BE UPDATED

Day27 - iMessage Gradient

The chat bubble changes its gradient color with its pageY.
react-native-linear-gradient

Day28 - iMessage Image Picker.

Day29 - TO BE UPDATED

Day30 - Push Notification.

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

推荐阅读更多精彩内容

  • 持续更新中...... 一套企业级的 UI 设计语言和 React 实现。 https://mobile.ant....
    日不落000阅读 5,675评论 0 35
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,907评论 25 707
  • 简短说明 收录一些好用的RN第三方组件,以方便日常的使用,大家有什么推荐的也可以跟我说,我加进去。如有冒犯,可以联...
    以德扶人阅读 43,626评论 44 214
  • 多年之后,你要去那边境小镇,那里人迹稀少,悠长的叫卖声走街串巷,不时会有炊烟四起,清晨你走到竹海边,采摘新鲜的蘑菇...
    半只肥秃阅读 368评论 0 0
  • 今天在一连听了好多堂课,信息量有些大,边听边做笔记,需要后续的复习、内化和应用。 1、利用中午时间,在下班加油站回...
    向着美好奔跑阅读 176评论 0 0