123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- // @flow
-
- import React, { Component, Fragment, PureComponent } from 'react';
- import { Dimensions, Image, Platform, View } from 'react-native';
- import FastImage, {
- type CacheControls,
- type Priorities
- } from 'react-native-fast-image';
-
- import { ColorPalette } from '../../styles';
-
- import styles from './styles';
-
- /**
- * The default image/source to be used in case none is specified or the
- * specified one fails to load.
- *
- * XXX The relative path to the default/stock (image) file is defined by the
- * {@code const} {@code DEFAULT_AVATAR_RELATIVE_PATH}. Unfortunately, the
- * packager of React Native cannot deal with it early enough for the following
- * {@code require} to succeed at runtime. Anyway, be sure to synchronize the
- * relative path on Web and mobile for the purposes of consistency.
- *
- * @private
- * @type {string}
- */
- const _DEFAULT_SOURCE = require('../../../../../images/avatar.png');
-
- /**
- * The type of the React {@link Component} props of {@link Avatar}.
- */
- type Props = {
-
- /**
- * The size for the {@link Avatar}.
- */
- size: number,
-
-
- /**
- * The URI of the {@link Avatar}.
- */
- uri: string
- };
-
- /**
- * The type of the React {@link Component} state of {@link Avatar}.
- */
- type State = {
-
- /**
- * Background color for the locally generated avatar.
- */
- backgroundColor: string,
-
- /**
- * Error indicator for non-local avatars.
- */
- error: boolean,
-
- /**
- * Indicates if the non-local avatar was loaded or not.
- */
- loaded: boolean,
-
- /**
- * Source for the non-local avatar.
- */
- source: {
- uri?: string,
- headers?: Object,
- priority?: Priorities,
- cache?: CacheControls,
- }
- };
-
- /**
- * Implements a React Native/mobile {@link Component} wich renders the content
- * of an Avatar.
- */
- class AvatarContent extends Component<Props, State> {
- /**
- * Initializes a new Avatar instance.
- *
- * @param {Props} props - The read-only React Component props with which
- * the new instance is to be initialized.
- */
- constructor(props: Props) {
- super(props);
-
- // Set the image source. The logic for the character # below is as
- // follows:
- // - Technically, URI is supposed to start with a scheme and scheme
- // cannot contain the character #.
- // - Technically, the character # in a URI signals the start of the
- // fragment/hash.
- // - Technically, the fragment/hash does not imply a retrieval
- // action.
- // - Practically, the fragment/hash does not always mandate a
- // retrieval action. For example, an HTML anchor with an href that
- // starts with the character # does not cause a Web browser to
- // initiate a retrieval action.
- // So I'll use the character # at the start of URI to not initiate
- // an image retrieval action.
- const source = {};
-
- if (props.uri && !props.uri.startsWith('#')) {
- source.uri = props.uri;
- }
-
- this.state = {
- backgroundColor: this._getBackgroundColor(props),
- error: false,
- loaded: false,
- source
- };
-
- // Bind event handlers so they are only bound once per instance.
- this._onAvatarLoaded = this._onAvatarLoaded.bind(this);
- this._onAvatarLoadError = this._onAvatarLoadError.bind(this);
- }
-
- /**
- * Computes if the default avatar (ie, locally generated) should be used
- * or not.
- */
- get useDefaultAvatar() {
- const { error, loaded, source } = this.state;
-
- return !source.uri || error || !loaded;
- }
-
- /**
- * Computes a hash over the URI and returns a HSL background color. We use
- * 75% as lightness, for nice pastel style colors.
- *
- * @param {Object} props - The read-only React {@code Component} props from
- * which the background color is to be generated.
- * @private
- * @returns {string} - The HSL CSS property.
- */
- _getBackgroundColor({ uri }) {
- if (!uri) {
- return ColorPalette.white;
- }
-
- let hash = 0;
-
- /* eslint-disable no-bitwise */
-
- for (let i = 0; i < uri.length; i++) {
- hash = uri.charCodeAt(i) + ((hash << 5) - hash);
- hash |= 0; // Convert to 32-bit integer
- }
-
- /* eslint-enable no-bitwise */
-
- return `hsl(${hash % 360}, 100%, 75%)`;
- }
-
- /**
- * Helper which computes the style for the {@code Image} / {@code FastImage}
- * component.
- *
- * @private
- * @returns {Object}
- */
- _getImageStyle() {
- const { size } = this.props;
-
- return {
- ...styles.avatar,
- borderRadius: size / 2,
- height: size,
- width: size
- };
- }
-
- _onAvatarLoaded: () => void;
-
- /**
- * Handler called when the remote image loading finishes. This doesn't
- * necessarily mean the load was successful.
- *
- * @private
- * @returns {void}
- */
- _onAvatarLoaded() {
- this.setState({ loaded: true });
- }
-
- _onAvatarLoadError: () => void;
-
- /**
- * Handler called when the remote image loading failed.
- *
- * @private
- * @returns {void}
- */
- _onAvatarLoadError() {
- this.setState({ error: true });
- }
-
- /**
- * Renders a default, locally generated avatar image.
- *
- * @private
- * @returns {ReactElement}
- */
- _renderDefaultAvatar() {
- // When using a local image, react-native-fastimage falls back to a
- // regular Image, so we need to wrap it in a view to make it round.
- // https://github.com/facebook/react-native/issues/3198
-
- const { backgroundColor } = this.state;
- const imageStyle = this._getImageStyle();
- const viewStyle = {
- ...imageStyle,
-
- backgroundColor,
-
- // FIXME @lyubomir: Without the opacity below I feel like the
- // avatar colors are too strong. Besides, we use opacity for the
- // ToolbarButtons. That's where I copied the value from and we
- // may want to think about "standardizing" the opacity in the
- // app in a way similar to ColorPalette.
- opacity: 0.1,
- overflow: 'hidden'
- };
-
- return (
- <View style = { viewStyle }>
- <Image
-
- // The Image adds a fade effect without asking, so lets
- // explicitly disable it. More info here:
- // https://github.com/facebook/react-native/issues/10194
- fadeDuration = { 0 }
- resizeMode = 'contain'
- source = { _DEFAULT_SOURCE }
- style = { imageStyle } />
- </View>
- );
- }
-
- /**
- * Renders an avatar using a remote image.
- *
- * @private
- * @returns {ReactElement}
- */
- _renderAvatar() {
- const { source } = this.state;
- let extraStyle;
-
- if (this.useDefaultAvatar) {
- // On Android, the image loading indicators don't work unless the
- // Glide image is actually created, so we cannot use display: none.
- // Instead, render it off-screen, which does the trick.
- if (Platform.OS === 'android') {
- const windowDimensions = Dimensions.get('window');
-
- extraStyle = {
- bottom: -windowDimensions.height,
- right: -windowDimensions.width
- };
- } else {
- extraStyle = { display: 'none' };
- }
- }
-
- return (
- <FastImage
- onError = { this._onAvatarLoadError }
- onLoadEnd = { this._onAvatarLoaded }
- resizeMode = 'contain'
- source = { source }
- style = { [ this._getImageStyle(), extraStyle ] } />
- );
- }
-
- /**
- * Implements React's {@link Component#render()}.
- *
- * @inheritdoc
- */
- render() {
- const { source } = this.state;
-
- return (
- <Fragment>
- { source.uri && this._renderAvatar() }
- { this.useDefaultAvatar && this._renderDefaultAvatar() }
- </Fragment>
- );
- }
- }
-
- /* eslint-disable react/no-multi-comp */
-
- /**
- * Implements an avatar as a React Native/mobile {@link Component}.
- *
- * Note: we use `key` in order to trigger a new component creation in case
- * the URI changes.
- */
- export default class Avatar extends PureComponent<Props> {
- /**
- * Implements React's {@link Component#render()}.
- *
- * @inheritdoc
- */
- render() {
- return (
- <AvatarContent
- key = { this.props.uri }
- { ...this.props } />
- );
- }
- }
|