/** * Copyright (c) 2013-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. * * The examples provided by Facebook are for non-commercial testing and * evaluation purposes only. * * Facebook reserves all rights not expressly granted. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL * FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * * @providesModule CameraRollView * @flow */ 'use strict'; var React = require('react'); const PropTypes = require('prop-types'); var ReactNative = require('react-native'); var { ActivityIndicator, Alert, CameraRoll, Image, ListView, PermissionsAndroid, Platform, StyleSheet, View, } = ReactNative; var groupByEveryN = require('groupByEveryN'); var logError = require('logError'); var propTypes = { /** * The group where the photos will be fetched from. Possible * values are 'Album', 'All', 'Event', 'Faces', 'Library', 'PhotoStream' * and SavedPhotos. */ groupTypes: PropTypes.oneOf([ 'Album', 'All', 'Event', 'Faces', 'Library', 'PhotoStream', 'SavedPhotos', ]), /** * Number of images that will be fetched in one page. */ batchSize: PropTypes.number, /** * A function that takes a single image as a parameter and renders it. */ renderImage: PropTypes.func, /** * imagesPerRow: Number of images to be shown in each row. */ imagesPerRow: PropTypes.number, /** * The asset type, one of 'Photos', 'Videos' or 'All' */ assetType: PropTypes.oneOf([ 'Photos', 'Videos', 'All', ]), }; var CameraRollView = React.createClass({ // $FlowFixMe(>=0.41.0) propTypes: propTypes, getDefaultProps: function(): Object { return { groupTypes: 'SavedPhotos', batchSize: 5, imagesPerRow: 1, assetType: 'Photos', renderImage: function(asset) { var imageSize = 150; var imageStyle = [styles.image, {width: imageSize, height: imageSize}]; return ( ); }, }; }, getInitialState: function() { var ds = new ListView.DataSource({rowHasChanged: this._rowHasChanged}); return { assets: ([]: Array), groupTypes: this.props.groupTypes, lastCursor: (null : ?string), assetType: this.props.assetType, noMore: false, loadingMore: false, dataSource: ds, }; }, /** * This should be called when the image renderer is changed to tell the * component to re-render its assets. */ rendererChanged: function() { var ds = new ListView.DataSource({rowHasChanged: this._rowHasChanged}); this.state.dataSource = ds.cloneWithRows( // $FlowFixMe(>=0.41.0) groupByEveryN(this.state.assets, this.props.imagesPerRow) ); }, componentDidMount: function() { this.fetch(); }, componentWillReceiveProps: function(nextProps: {groupTypes?: string}) { if (this.props.groupTypes !== nextProps.groupTypes) { this.fetch(true); } }, _fetch: async function(clear?: boolean) { if (clear) { this.setState(this.getInitialState(), this.fetch); return; } if (Platform.OS === 'android') { const result = await PermissionsAndroid.request( PermissionsAndroid.PERMISSIONS.READ_EXTERNAL_STORAGE, { title: 'Permission Explanation', message: 'UIExplorer would like to access your pictures.', }, ); if (result !== 'granted') { Alert.alert('Access to pictures was denied.'); return; } } const fetchParams: Object = { first: this.props.batchSize, groupTypes: this.props.groupTypes, assetType: this.props.assetType, }; if (Platform.OS === 'android') { // not supported in android delete fetchParams.groupTypes; } if (this.state.lastCursor) { fetchParams.after = this.state.lastCursor; } try { const data = await CameraRoll.getPhotos(fetchParams); this._appendAssets(data); } catch (e) { logError(e); } }, /** * Fetches more images from the camera roll. If clear is set to true, it will * set the component to its initial state and re-fetch the images. */ fetch: function(clear?: boolean) { if (!this.state.loadingMore) { this.setState({loadingMore: true}, () => { this._fetch(clear); }); } }, render: function() { return ( ); }, _rowHasChanged: function(r1: Array, r2: Array): boolean { if (r1.length !== r2.length) { return true; } for (var i = 0; i < r1.length; i++) { if (r1[i] !== r2[i]) { return true; } } return false; }, _renderFooterSpinner: function() { if (!this.state.noMore) { return ; } return null; }, // rowData is an array of images _renderRow: function(rowData: Array, sectionID: string, rowID: string) { var images = rowData.map((image) => { if (image === null) { return null; } // $FlowFixMe(>=0.41.0) return this.props.renderImage(image); }); return ( {images} ); }, _appendAssets: function(data: Object) { var assets = data.edges; var newState: Object = { loadingMore: false }; if (!data.page_info.has_next_page) { newState.noMore = true; } if (assets.length > 0) { newState.lastCursor = data.page_info.end_cursor; newState.assets = this.state.assets.concat(assets); newState.dataSource = this.state.dataSource.cloneWithRows( // $FlowFixMe(>=0.41.0) groupByEveryN(newState.assets, this.props.imagesPerRow) ); } this.setState(newState); }, _onEndReached: function() { if (!this.state.noMore) { this.fetch(); } }, }); var styles = StyleSheet.create({ row: { flexDirection: 'row', flex: 1, }, url: { fontSize: 9, marginBottom: 14, }, image: { margin: 4, }, info: { flex: 1, }, container: { flex: 1, }, }); module.exports = CameraRollView;