Skip to content

Commit

Permalink
finish ios and android cacheimage
Browse files Browse the repository at this point in the history
  • Loading branch information
fov42550564 committed Dec 7, 2015
1 parent 8b55a73 commit df587fd
Show file tree
Hide file tree
Showing 8 changed files with 677 additions and 15 deletions.
4 changes: 3 additions & 1 deletion .gitignore
Expand Up @@ -2,9 +2,11 @@
*.DS_Store
.DS_Store
*Thumbs.db
*.iml
.gradle
.idea
node_modules
npm-debug.log
/android/RCTDes/build
/android/build
/ios/**/*xcuserdata*
/ios/**/*xcshareddata*
12 changes: 12 additions & 0 deletions .npmignore
@@ -0,0 +1,12 @@
*.DS_Store
.DS_Store
*Thumbs.db
.gradle
.idea
*.iml
npm-debug.log
node_modules
/android/build
/ios/**/*xcuserdata*
/ios/**/*xcshareddata*

4 changes: 2 additions & 2 deletions README.md
@@ -1,7 +1,7 @@
# React Native cache-image (remobile)
# React Native CacheImage (remobile)
A cache-image for react-native

## Installation
```sh
npm install react-native-cache-image --save
npm install @remobile/react-native-cache-image --save
```
1 change: 1 addition & 0 deletions image.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

301 changes: 290 additions & 11 deletions index.js 100644 → 100755
@@ -1,25 +1,304 @@
/*
* (The MIT License)
* Copyright (c) 2015-2016 YunJiang.Fang <42550564@qq.com>
* @providesModule cache-image
* @providesModule CacheImage
* @flow-weak
*/
'use strict';

var React = require('react-native');

var {
View,
Image,
Text,
StyleSheet,
} = React;

module.exports = React.createClass({
render() {

var md5 = require("./md5.js");
var image = require('./image.js');
var StorageMgr = require('./storageMgr.js');
var fs = require('react-native-fs');
var TimerMixin = require('react-timer-mixin');


/*list status change graph
*
*STATUS_NONE->[STATUS_LOADING]
*STATUS_LOADING->[STATUS_LOADED, STATUS_UNLOADED]
*
*/
var
STATUS_NONE = 0,
STATUS_LOADING = 1,
STATUS_LOADED = 2,
STATUS_UNLOADED = 3;

var storageMgr = new StorageMgr();
var db = StorageMgr.db;
var syncImageSource = {};

var CacheImage = React.createClass({
mixins: [TimerMixin],
addImageRef(url, size) {
return new Promise((resolve, reject) => {
db.transaction((tx)=>{
var ref = size?'1':'ref+1';
tx.executeSql('UPDATE '+StorageMgr.TABLE_CACHE_IMAGE+' SET ref='+ref+' WHERE url=?', [url], (tx, rs)=>{
if (rs.rowsAffected == 0) {
tx.executeSql('INSERT INTO '+StorageMgr.TABLE_CACHE_IMAGE+' (url, ref, size, time) VALUES (?, ?, ?, ?)', [url, 1, size, parseInt(Date.now()/1000)], (tx, rs)=>{
console.log('subImageRef <insert>', url, size);
resolve(true);
});
} else {
console.log('subImageRef <undo>', url, size);
resolve(true);
}
});
}, (error)=>{
console.log('subImageRef <error>', url, size, error);
resolve(false);
});
});
},
subImageRef(url) {
var self = this;
return new Promise((resolve, reject) => {
db.transaction((tx)=>{
tx.executeSql('SELECT ref,size FROM '+StorageMgr.TABLE_CACHE_IMAGE+' WHERE url=?', [url], function (tx, rs) {
var item = rs.rows.item(0);
var ref = item.ref;
var size = item.size;
if (ref == 1) {
tx.executeSql('DELETE FROM '+StorageMgr.TABLE_CACHE_IMAGE+' WHERE url=?', [url], async(tx, rs)=>{
await fs.unlink(storageMgr.getCacheFilePath(url));
await storageMgr.updateStorage(-size);
console.log('subImageRef <delete>', url);
resolve(true);
});
} else {
tx.executeSql('UPDATE '+StorageMgr.TABLE_CACHE_IMAGE+' SET ref=ref-1 WHERE url=?', [url], (tx, rs)=>{
console.log('subImageRef <update>', url);
resolve(true);
});
}
});
}, (error)=>{
console.log('subImageRef <error>', url, error);
resolve(false);
});
});
},
checkCacheId(id, url, size) {
var self = this;
return new Promise((resolve, reject) => {
db.transaction((tx)=>{
tx.executeSql('SELECT url FROM '+StorageMgr.TABLE_CACHE_ID+' WHERE id=?', [id], (tx, rs)=>{
if (rs.rows.length) {
var item = rs.rows.item(0);
var oldurl = item.url;
if (url !== oldurl) {
tx.executeSql('UPDATE '+StorageMgr.TABLE_CACHE_ID+' SET url=? WHERE id=?', [url, id], async(tx, rs)=>{
console.log('checkCacheId <update oldurl>', id, url, size, oldurl);
await self.addImageRef(url, size);
await self.subImageRef(oldurl);
self.unlock();
resolve(true);
});
} else {
self.unlock();
resolve(true);
}
} else {
tx.executeSql('INSERT INTO '+StorageMgr.TABLE_CACHE_ID+' (id, url) VALUES (?, ?)', [id, url], async(tx, rs)=>{
console.log('checkCacheId <insert new url>', id, url, size);
await self.addImageRef(url, size);
self.unlock();
resolve(true);
});
}
});
}, (error)=>{
resolve(false);
console.log('checkCacheId <error>', id, url, size, error);
self.unlock();
});
});
},
deleteCacheImage(storage) {
var self = this;
return new Promise((resolve, reject) => {
db.transaction((tx)=>{
tx.executeSql('SELECT url,size FROM '+StorageMgr.TABLE_CACHE_IMAGE+' WHERE time=(SELECT MIN(time) FROM '+StorageMgr.TABLE_CACHE_IMAGE+')', [], function (tx, rs) {
if (rs.rows.length) {
var item = rs.rows.item(0);
var url = item.url;
var size = item.size;
tx.executeSql('DELETE FROM '+StorageMgr.TABLE_CACHE_IMAGE+' WHERE url=?', [url], (tx, rs)=>{
tx.executeSql('DELETE FROM '+StorageMgr.TABLE_CACHE_ID+' WHERE url=?', [url], async(tx, rs)=>{
console.log('deleteCacheImage <delete>', url, size);
storage -= size;
await fs.unlink(storageMgr.getCacheFilePath(url));
await storageMgr.updateStorage(-size);
resolve(storage);
});
});
}
});
}, (error)=>{
console.log('deleteCacheImage <error>', error);
reject(error);
});
});
},
checkCacheStorage(size) {
var self = this;
return new Promise(async(resolve, reject) => {
var storage = storageMgr.storage + size;
console.log('target:', storage);
while (storage >= StorageMgr.CACHE_IMAGE_SIZE) {
storage = await self.deleteCacheImage(storage);
console.log('after:', storage);
}
resolve();
});
},
isFileExist(filepath) {
return new Promise((resolve, reject) => {
fs.stat(filepath).then((rs) => {
resolve(true);
}).catch((err) => {
resolve(false);
});
});
},
downloadImage(url, filepath, cacheId, filename) {
var self = this;
var ret = fs.downloadFile(url, filepath).then(async (res)=>{
self.setState({
status:STATUS_LOADED,
source:{uri:'file://'+filepath},
});
console.log(self.state);
await self.checkCacheId(cacheId, filename, res.bytesWritten);
await storageMgr.updateStorage(res.bytesWritten);
await self.checkCacheStorage(res.bytesWritten);
}).catch(
(err)=>{
console.log(err);
this.unlock();
self.setState({
status:STATUS_UNLOADED,
});
}
);
},
checkImageSource(cacheId, url) {
var type = url.replace(/.*\.(.*)/, '$1');
var filename = md5(url)+'.'+type;
var filepath = storageMgr.getCacheFilePath(filename);
this.param = {cacheId:cacheId, url:url, filename:filename, filepath:filepath};
this.syncCheckImageSource();
},
lock() {
syncImageSource[this.param.filename] = true;
},
unlock() {
delete syncImageSource[this.param.filename];
},
islock() {
return syncImageSource[this.param.filename];
},
syncCheckImageSource() {
if (this.islock()) {
this.setTimeout(this.syncCheckImageSource, 100);
} else {
this.doCheckImageSource();
}
},
async doCheckImageSource() {
var {cacheId, url, filename, filepath} = this.param;
this.lock();
var isExist = await this.isFileExist(filepath);
console.log(this.param);
console.log('Is File exist', isExist);
if (isExist) {
this.setState({
status:STATUS_LOADED,
source:{uri:'file://'+filepath},
});
console.log(this.state);
this.checkCacheId(cacheId, filename);
} else {
this.downloadImage(url, filepath, cacheId, filename);
}
},
getInitialState() {
return {
status:STATUS_NONE,
}
},
componentWillMount() {
let {cacheId, url} = this.props;
this.setState({status:STATUS_LOADING});
this.checkImageSource(cacheId, url);
},
renderLoading() {
return (
<View style={{flex:1, justifyContent: 'center', alignItems: 'center',}}>
<Text>
come soon
</Text>
</View>
)
}
<Image
{...this.props}
style={[this.props.style, {justifyContent:'center', alignItems:'center'}]}
source={this.props.defaultImage}
>
<Image
style={styles.spinner}
source={{
uri: '',
isStatic: true
}}
>
</Image>
{this.props.children}
</Image>
);
},
renderLocalFile() {
return (
<Image
{...this.props}
source={this.state.source}
>
{this.props.children}
</Image>
);
},
render() {
if (this.state.status === STATUS_LOADING) {
return this.renderLoading();
} else if (this.state.status === STATUS_LOADED) {
return this.renderLocalFile();
} else {
return (
<Image
{...this.props}
source={this.props.defaultImage}
>
{this.props.children}
</Image>
)
}
},
});

CacheImage.clear = storageMgr.clear;
module.exports = CacheImage;

var styles = StyleSheet.create({
spinner: {
alignItems: 'center',
justifyContent: 'center',
width: 25,
height: 25,
backgroundColor: 'transparent',
},
});

0 comments on commit df587fd

Please sign in to comment.