import React from 'react' import axios from 'axios' import Spinner from 'react-spinner' import { range, fileToDataURL } from '../lib/util' const RAND_RANGE = 1000000 const UPDATE_SIZE = 20 const WALLPAPER_COLLECTION_ID = 136026 const RANDOM_WALLPAPER_URL = `https://source.unsplash.com/collection/${WALLPAPER_COLLECTION_ID}/240x320` const largerImage = url => url.replace(/w=\d+/, 'w=1920').replace(/&h=\d+/, '') const smallerImage = url => url.replace(/w=\d+/, 'w=240') const imageUrls = {} let cache = [] async function getImage() { // circumvent browser caching const sig = Math.floor(Math.random() * RAND_RANGE) const res = await axios.get(`${RANDOM_WALLPAPER_URL}?sig=${sig}`, { responseType: 'blob' }) // image already in cache? if (imageUrls[res.request.responseURL]) return imageUrls[res.request.responseURL] = true return { url: res.request.responseURL, dataURL: await fileToDataURL(res.data) } } export default class extends React.Component { constructor() { super() this.state = { cacheIndex: 0, loading: false } this.selectImage = this.selectImage.bind(this) this.updateCache = this.updateCache.bind(this) this.nextImage = this.nextImage.bind(this) } // fetch images in browser (we require window.FileReader) componentDidMount() { // clear cache when remounted cache = [] this.updateCache() } selectImage() { this.setState({ loading: true }) axios .get(largerImage(cache[this.state.cacheIndex].url), { responseType: 'blob' }) .then(res => res.data) .then(this.props.onChange) .then(() => this.setState({ loading: false })) } updateCache() { this.setState({ loading: true }) Promise.all(range(UPDATE_SIZE).map(getImage)) .then(imgs => imgs.filter(img => img)) // remove null .then(imgs => (cache = cache.concat(imgs))) .then(() => this.setState({ loading: false })) } nextImage() { if (this.state.loading) return this.setState({ cacheIndex: this.state.cacheIndex + 1 }) if (this.state.cacheIndex > cache.length - 2) { this.updateCache() } } render() { const bgImage = cache[this.state.cacheIndex] && cache[this.state.cacheIndex].dataURL return (
Use Image Try Another
{this.state.loading && }
) } }