import { Q } from '@nozbe/watermelondb'; import React, { useEffect, useRef, useState } from 'react'; import { shallowEqual, useSelector } from 'react-redux'; import { Observable, Subscription } from 'rxjs'; import Button from '../Button'; import { IApplicationState, TSubscriptionModel, TUserModel } from '../../definitions'; import database from '../../lib/database'; import { getUserSelector } from '../../selectors/login'; import Avatar from './Avatar'; import { IAvatar } from './interfaces'; import I18n from '../../i18n'; import { useTheme } from '../../theme'; import styles from './styles'; const AvatarContainer = ({ style, text = '', avatar, emoji, size, borderRadius, type, children, onPress, getCustomEmoji, isStatic, rid, handleEdit, isUserProfile }: IAvatar & { isUserProfile?: boolean; handleEdit?: () => void }): React.ReactElement => { const subscription = useRef(); const [avatarETag, setAvatarETag] = useState(''); const previousAvatarEtag = useRef(''); const { colors } = useTheme(); const isDirect = () => type === 'd'; const server = useSelector((state: IApplicationState) => state.share.server.server || state.server.server); const serverVersion = useSelector((state: IApplicationState) => state.share.server.version || state.server.version); const { id, token, avatarETagUser } = useSelector( (state: IApplicationState) => ({ id: getUserSelector(state).id, token: getUserSelector(state).token, avatarETagUser: getUserSelector(state).avatarETag }), shallowEqual ); const externalProviderUrl = useSelector( (state: IApplicationState) => state.settings.Accounts_AvatarExternalProviderUrl as string ); const blockUnauthenticatedAccess = useSelector( (state: IApplicationState) => (state.share.settings?.Accounts_AvatarBlockUnauthenticatedAccess as boolean) ?? state.settings.Accounts_AvatarBlockUnauthenticatedAccess ?? true ); const init = async () => { const db = database.active; const usersCollection = db.get('users'); const subsCollection = db.get('subscriptions'); let record; try { if (isDirect()) { const [user] = await usersCollection.query(Q.where('username', text)).fetch(); record = user; } else if (rid) { record = await subsCollection.find(rid); } } catch { // Record not found } if (record) { const observable = record.observe() as Observable; subscription.current = observable.subscribe(r => { setAvatarETag(r.avatarETag); }); } }; useEffect(() => { if (!avatarETag || avatarETag !== previousAvatarEtag.current) { init(); } return () => { if (subscription?.current?.unsubscribe) { subscription.current.unsubscribe(); } }; }, [text, type, size, avatarETag, externalProviderUrl]); useEffect(() => { previousAvatarEtag.current = avatarETag; }, [avatarETag]); return ( <> {handleEdit ? (