React Native (简称RN)是Facebook于2015年4月开源的跨平台移动应用开发框架,是Facebook早先开源的UI框架 React 在原生移动应用平台的衍生产物,目前支持iOS和安卓两大平台。RN使用Javascript语言,类似于HTML的JSX,以及CSS来开发移动应用,因此熟悉Web前端开发的技术人员只需很少的学习就可以进入移动应用开发领域。(百度百科)

      小编有幸参与了公司APP的开发,该APP采用混合式开发,本人负责RN的几个功能的开发,其中最复杂的就是应用中心的功能,如下图所示。模块分为上下两部分,下面是所有应用,上面是下方勾选的应用,可以上下左右滑动(图中只有一行,后面改成了两行,故可以上下或者斜着滑动,偷懒就不截图了)

Android图标跟随系统主题变化_Image

         所以一共要实现接收数据、选择应用、调整顺序、提交这个几个功能,下面重点讲述调整顺序的代码,其他的代码可能会掠过。

1、获取数据

定一个几个变量,用于保存我的应用和所有应用,获取数据调用了原生的方法,大家用RN的获取数据的方法是一样的。

 

在构造器里面的变量
this.myapps = [];
this.application= [];

 

//获取数据
dsfectchData(){
    RNBridgeManager.queryAppList((error, result) => {
        if(error){
            console.log("---q--->error:" + error);
        }else{
            var myapp = [];
            for(var i=0;i<result.length;i++){
                if(result[i].myAppList == 1){
                    myapp.push(result[i]);
                    console.log( i + " : " + result[i].appId);
                }
            }
            this.myapps = myapp;   //保存我的应用数据
            console.log("this.myapps" + this.myapps.length);
            this.application = result; // 保存所有数据
        }
    });
}
2、展示数据

重点来看我的应用的数据展示方式,如下图第一个红色标记的是将数据放入到一个数组中,这个数据记录
顺序;第二个红色标记用于和上一个标记中的order数组一起,用于排序和重新展示;第三个红色标记的
部分是每个可拖动的图标的定位,当然首先要将定位设定为:position = 'absolute'。绿色底标注
的为可拖动的核心,即触摸操作。


 


<View style={{flexDirection: 'row',height:25,backgroundColor: '#fff',alignItems: 'center',paddingTop:10,marginTop:px2pt(10)}}>
<Image
        source={{uri:this.state.navbar}}
        style={{width:3,height:15,marginLeft:px2pt(10)}}
    />
    <Text style={styles.itemTitle}>我的应用</Text>
    <Text style={styles.itemTitleDes}>(按住拖动调整顺序)</Text>
</View>
<View style={styles.containerMy}>
    {this.myapps.map((item, i)=>{
        this.order.push(item);
        return (
            <View
                {...this._panResponder.panHandlers}
                ref={(ref) => this.items[i] = ref}
                key={i}
                style={[styles.item, {top: (parseInt(i/4))*90 + 10,left:(i%4)*this.widthSreen}]}>
                <Image
                    source={{uri:this.state.shortcut_icon_bg}}
                    style={styles.thumbnail}
                />
                <Image
                    source={this._getApplicationIcons(item.appId)}
                    style={styles.thumbnailInside}
                />
                <TextActivity text={item.appName} width={containerWidth}/>
                <TouchableWithoutFeedback onPress={this._pressIcon.bind(this,this.myapps,i,1)} style={styles.buttonRightTop}>
                    <Image
                        source={{uri:this.state.shortcut_icon_del}}
                        style={styles.imageRightTop}
                    />
                </TouchableWithoutFeedback>
            </View>
        );
    })}
</View>
3、滑动操作部分,此部分依次为

1)点击时判定是哪个图标(根据坐标)

2)判定移动位置是否已经和其他的图标有重合,如果有,则交换两个图标的位置

3)释放后的显示结果


componentWillMount(){
    this._panResponder = PanResponder.create({
        onStartShouldSetPanResponder: (evt, gestureState) => true,
        onMoveShouldSetPanResponder: (evt, gestureState) => true,
        onPanResponderGrant: (evt, gestureState) => {
            const {pageX,pageY, locationX,locationY} = evt.nativeEvent;
            console.log("--eric-->:pageX:" + pageX + "--pageY:" + pageY + "--locationX:" + locationX + "--locationY:" + locationY);
            this.index = this._getIdByPosition(pageX,pageY);
            //逻辑应该是点击的一瞬间,判定是第几个 (parseInt(i/4))*90,left:(i%4)*this.widthSreen}
            this.preX = (this.index%4)*this.widthSreen;
            this.preY = (parseInt(this.index/4))*90 + 10;
            //get the taped item and highlight it
            let item = this.items[this.index];
            if(item!=null){
                item.setNativeProps({
                    style: {
                        shadowColor: "#000",
                        shadowOpacity: 0.3,
                        shadowRadius: 5,
                        shadowOffset: {height: 0, width: 2},
                        elevation: 5
                    }
                });
            }
        },
        // 移动判定是否进入了某个图标的领域,如果进入则交换
         onPanResponderMove: (evt, gestureState) => {
            let left = this.preX + gestureState.dx;
            let top = this.preY + gestureState.dy;
            console.log("--eric-->top:" + top + ":" + gestureState.dy);
            let item = this.items[this.index];
            if(item!=null) {
                item.setNativeProps({
                    style: {left: left,top: top}
                });
            }
            let collideIndex = this._getIdByPosition(evt.nativeEvent.pageX,evt.nativeEvent.pageY);
            if(collideIndex !== this.index && collideIndex !== -1) {
                let collideItem = this.items[collideIndex];
                if(collideItem!=null){
                    collideItem.setNativeProps({
                        style: {left: this._getLeftValueYById(this.index),top:this._getTopValueYById(this.index)}
                    });
                    //swap two values
                    [this.items[this.index], this.items[collideIndex]] = [this.items[collideIndex], this.items[this.index]];
                    [this.order[this.index], this.order[collideIndex]] = [this.order[collideIndex], this.order[this.index]];
                    this.index = collideIndex;
                }
            }
        },
        
        // 释放后的显示结果
        onPanResponderTerminationRequest: (evt, gestureState) => true,
        onPanResponderRelease: (evt, gestureState) => {
            const shadowStyle = {
                shadowColor: "#000",
                shadowOpacity: 0,
                shadowRadius: 0,
                shadowOffset: {height: 0, width: 0,},
                elevation: 0
            };
            let item = this.items[this.index];
            //go back the correct position
            if(item!=null){
                item.setNativeProps({
                    style: {...shadowStyle, left: this._getLeftValueYById(this.index),top: this._getTopValueYById(this.index)}
                });
            }
            console.log(this.order);
        },
        onPanResponderTerminate: (evt, gestureState) => {
            // Another component has become the responder, so this gesture
            // should be cancelled
        }
    });
}