小六教你用ReactNative做一个拼图游戏(二) 优化与重布局..
上一篇文章我们一张完整的猫猫图片切割成了零碎的图片
但是我们发现所有的代码都在index.js,这样并不是我们应该去做的事情。
重构
这次我们首先将上次的代码进行重构。
首先我们建立一个js文件,帮助我们处理图片。
import { ImageEditor, Image, Dimensions } from "react-native"; const WIDTH = Dimensions.get("window").width; export default class Croper { constructor(imageUri) { this.imageUri = imageUri; } /* * 方法进行Promise的转化 * */ _cropImage(img, cropData) { return new Promise(function (resolve, reject) { ImageEditor.cropImage(img, cropData, resolve, reject); }); }; _getSize(img) { return new Promise(function (resolve, reject) { Image.getSize(img, (width, height) => { resolve({width, height}) }, reject); }); }; async crop(column, row) { //好习惯,首先写返回值 let result = {}; //获取图片宽和高 let {width: imgWidth, height: imgHeight} = await this._getSize(this.imageUri); //进行切割 3*3布局 // 基本功两次循环 let _column = column; let _row = row; let _columnWidth = imgWidth / _column; let _rowHeight = imgHeight / _row; result.width = _columnWidth; result.height = _rowHeight; result.imageList = []; for (let r = 0; r < _row; r++) { for (let c = 0; c < _column; c++) { let _cropData = { offset: { x: c * _columnWidth, y: r * _rowHeight, }, size: { width: _columnWidth, height: _rowHeight, }, } console.log(_cropData.offset.x); let uri = await this._cropImage(this.imageUri, _cropData); result.imageList.push({ uri: uri, width: _columnWidth, height: _rowHeight, fullWidth: WIDTH / _column, fullHeight: (WIDTH / imgWidth) * imgHeight / _row, index: r + c, r: r, c: c, }); } } return result; } }
注意我们这里返回的结果中,加入了更多属性 比如index ,坐标,重新计算后的宽度
然后我们继续处理,这一次我们把图片按照我们想要的进行拼接~
可爱的猫猫又要被我们拼回去了~。
不过这次我们要留下一个最后一个块作为空白快(大家应该都玩过拼图游戏吧)。
然后加上一些黑色的边框,这样更加带感。
按钮
我们添加一些按钮,用于以后操纵拼图游戏
<View style={{flexDirection: "row"}}> <Button title="↑" onPress={this._move.bind(this, "up")}></Button> <Button title="↓" onPress={this._move.bind(this, "down")}></Button> <Button title="←" onPress={this._move.bind(this, "left")}></Button> <Button title="→" onPress={this._move.bind(this, "right")}></Button> </View>
打乱
我们现在需要将之前的图片进行打乱,同时跟他们加上currentIndex标记
//打乱图片 并且加上索引 抠出最后一张图 _upset(arr) { var result = arr; var indexList = []; for (let i = 0; i < arr.length; i++) { indexList.push(i); } //打乱顺序 indexList = indexList.sort(function () { return (0.5 - Math.random()); }); for (let i = 0; i < arr.length; i++) { result[i].currentIndex = indexList[i]; if (indexList[i] == arr.length - 1) { result[i].isNull = true; } } return result; }
这个时候我们需要将图片显示的函数进行一些调整
<Image style={{ width: img.fullWidth, height: img.fullHeight, position: "absolute", top: parseInt(img.currentIndex / 3) * img.fullHeight, left: img.currentIndex % 3 * img.fullWidth, borderWidth: 1, borderColor: "#000000" }} key={index} source={{uri: img.uri}}/>
完整的代码如下。
import React, {Component} from 'react'; import { Platform, StyleSheet, Dimensions, ScrollView, Text, ImageEditor, Button, Image, View } from 'react-native'; import Croper from "./Croper"; const WIDTH = Dimensions.get("window").width; const _IMG = ""; export default class App extends Component<{}> { constructor() { super(); this.state = { myUrl: {uri: _IMG}, imgList: [], } let _this = this; let croper = new Croper(_IMG); croper.crop(3, 3).then((json) => { this._getActBox(); this.setState({imgList: this._upset(json.imageList)}); }); //存放哪些块 this.actBox = { up: {}, down: {}, left: {}, right: {}, } } //打乱图片 并且加上索引 抠出最后一张图 _upset(arr) { var result = arr; var indexList = []; for (let i = 0; i < arr.length; i++) { indexList.push(i); } //打乱顺序 indexList = indexList.sort(function () { return (0.5 - Math.random()); }); for (let i = 0; i < arr.length; i++) { result[i].currentIndex = indexList[i]; if (indexList[i] == arr.length - 1) { result[i].isNull = true; } } return result; } _getActBox() { var blackBox = this.state.imgList.findIndex((element) => { return element == null }); } _move(direction) { } render() { return ( <View style={styles.container}> <View style={{flex: 1}}> {this.state.imgList.map((img, index) => { if (img.isNull) return null; return ( <Image style={{ width: img.fullWidth, height: img.fullHeight, position: "absolute", top: parseInt(img.currentIndex / 3) * img.fullHeight, left: img.currentIndex % 3 * img.fullWidth, borderWidth: 1, borderColor: "#000000" }} key={index} source={{uri: img.uri}}/> ) })} </View> <View style={{flexDirection: "row"}}> <Button title="↑" onPress={this._move.bind(this, "up")}></Button> <Button title="↓" onPress={this._move.bind(this, "down")}></Button> <Button title="←" onPress={this._move.bind(this, "left")}></Button> <Button title="→" onPress={this._move.bind(this, "right")}></Button> </View> </View> ); } } const styles = StyleSheet.create({ container: { flex: 1, }, opView: {}, canvas: { flex: 1 }, });
丛刻我们就又把猫猫变成完整的猫了~。~
到此为止,我们的准备工作全部做好了,之后我们就开始改造和动画
未完待续。