listAll 来自 firebase 的 getDownloadURL 存储在数组中以呈现图像的 FlatList 不起作用

发布于 2025-01-10 20:00:11 字数 4376 浏览 0 评论 0原文

在我的 React Native 组件中,我尝试获取 Firebase 存储中所有图像的列表:

listAll(listRef).then((res) => {       
        res.items.forEach((itemRef) => {

对于每个图像,我执行 getDownloadURL:

 const downloadUrl = getDownloadURL(ref(storage, itemRef))

切片令牌(稍后用于密钥),创建一个对象并将该对象推入数组中:

downloadUrl.then((url) => {
                // Find token in url
                const indexOfToken = url.indexOf("&token=")

                // slice token from url [+7 is the length of the word '&token=']
                const token = url.slice(indexOfToken + 7)

                // create a image object
                const image = {
                    "imageUrl": url,
                    "imageToken": token
                }

                // store the image in eventImages array
                eventImages.push(image);

如果我尝试使用此数组渲染平面列表。

<FlatList data={imageData}
   keyExtractor={item => item.imageToken}
   renderItem={({ item }) => (                      
     <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} 
     source={{ uri: imageData.imageUrl }} alt="Alternate Text" size="xl" />
     )}/>

数组为空。为什么?


完整代码

import React, { useState } from 'react'
import { StyleSheet, SafeAreaView } from 'react-native'
import { Image, FlatList } from "native-base"
import EventGalleryHeader from '../components/EventGalleryHeader.js'
import { getStorage, ref, getDownloadURL, list, listAll } from "firebase/storage"
import { LongPressGestureHandler, State } from 'react-native-gesture-handler'

export default function EventScreen(props) {

    const [imageData, setImageData] = useState();
    const eventImages = []
    const storage = getStorage();
    const listRef = ref(storage, '/eventimages/1645351776501/');

    listAll(listRef).then((res) => {
        res.items.forEach((itemRef) => {
            const downloadUrl = getDownloadURL(ref(storage, itemRef))
            downloadUrl.then((url) => {
                const indexOfToken = url.indexOf("&token=")
                const token = url.slice(indexOfToken + 7)
                const image = {
                    "imageUrl": url,
                    "imageToken": token
                }
                eventImages.push(image);
            }).catch((error) => {
                switch (error.code) {
                    case 'storage/object-not-found':
                        break;
                    case 'storage/unauthorized':
                        break;
                    case 'storage/canceled':
                        break;
                    case 'storage/unknown':
                        break;
                }
            });
            console.log(eventImages)
        });
    }).catch((error) => {
    });

    const onLongPress = (event) => {
        if (event.nativeEvent.state === State.ACTIVE) {
            alert("I've been pressed for 800 milliseconds");
        }
    };
    return (
        <SafeAreaView style={styles.container} >
            <FlatList _dark={{ bg: "blueGray.900" }} _light={{ bg: "blueGray.50" }}
                style={styles.list}
                numColumns={2}
                ListHeaderComponent={<EventGalleryHeader data={props.route.params.eventData} />}
                data={imageData}
                keyExtractor={item => item.imageToken}
                renderItem={({ item }) => (
                    <LongPressGestureHandler
                        onHandlerStateChange={onLongPress}
                        minDurationMs={800}
                    >
                        <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} source={{ uri: imageData.imageUrl }} alt="Alternate Text" size="xl" />
                    </LongPressGestureHandler>
                )}
            />
        </SafeAreaView>
    );
}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        marginTop: 25,
    },
    image: {
        maxHeight: 450,
        width: '100%',
        height: 200,
        overflow: 'hidden',
    },
    list: {
        alignSelf: 'center',
    },
    gallery: {
        flex: 1,
        width: '100%',
        flexDirection: 'row',

    }
})

In my react native component I try to get a list of all images in my firebase store:

listAll(listRef).then((res) => {       
        res.items.forEach((itemRef) => {

For each image I do the getDownloadURL:

 const downloadUrl = getDownloadURL(ref(storage, itemRef))

Slice the token (for key later), create an object and push this object into an array:

downloadUrl.then((url) => {
                // Find token in url
                const indexOfToken = url.indexOf("&token=")

                // slice token from url [+7 is the length of the word '&token=']
                const token = url.slice(indexOfToken + 7)

                // create a image object
                const image = {
                    "imageUrl": url,
                    "imageToken": token
                }

                // store the image in eventImages array
                eventImages.push(image);

If I try to render a flatlist with this array.

<FlatList data={imageData}
   keyExtractor={item => item.imageToken}
   renderItem={({ item }) => (                      
     <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} 
     source={{ uri: imageData.imageUrl }} alt="Alternate Text" size="xl" />
     )}/>

The Array is empty. Why?


Fullcode

import React, { useState } from 'react'
import { StyleSheet, SafeAreaView } from 'react-native'
import { Image, FlatList } from "native-base"
import EventGalleryHeader from '../components/EventGalleryHeader.js'
import { getStorage, ref, getDownloadURL, list, listAll } from "firebase/storage"
import { LongPressGestureHandler, State } from 'react-native-gesture-handler'

export default function EventScreen(props) {

    const [imageData, setImageData] = useState();
    const eventImages = []
    const storage = getStorage();
    const listRef = ref(storage, '/eventimages/1645351776501/');

    listAll(listRef).then((res) => {
        res.items.forEach((itemRef) => {
            const downloadUrl = getDownloadURL(ref(storage, itemRef))
            downloadUrl.then((url) => {
                const indexOfToken = url.indexOf("&token=")
                const token = url.slice(indexOfToken + 7)
                const image = {
                    "imageUrl": url,
                    "imageToken": token
                }
                eventImages.push(image);
            }).catch((error) => {
                switch (error.code) {
                    case 'storage/object-not-found':
                        break;
                    case 'storage/unauthorized':
                        break;
                    case 'storage/canceled':
                        break;
                    case 'storage/unknown':
                        break;
                }
            });
            console.log(eventImages)
        });
    }).catch((error) => {
    });

    const onLongPress = (event) => {
        if (event.nativeEvent.state === State.ACTIVE) {
            alert("I've been pressed for 800 milliseconds");
        }
    };
    return (
        <SafeAreaView style={styles.container} >
            <FlatList _dark={{ bg: "blueGray.900" }} _light={{ bg: "blueGray.50" }}
                style={styles.list}
                numColumns={2}
                ListHeaderComponent={<EventGalleryHeader data={props.route.params.eventData} />}
                data={imageData}
                keyExtractor={item => item.imageToken}
                renderItem={({ item }) => (
                    <LongPressGestureHandler
                        onHandlerStateChange={onLongPress}
                        minDurationMs={800}
                    >
                        <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} source={{ uri: imageData.imageUrl }} alt="Alternate Text" size="xl" />
                    </LongPressGestureHandler>
                )}
            />
        </SafeAreaView>
    );
}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        marginTop: 25,
    },
    image: {
        maxHeight: 450,
        width: '100%',
        height: 200,
        overflow: 'hidden',
    },
    list: {
        alignSelf: 'center',
    },
    gallery: {
        flex: 1,
        width: '100%',
        flexDirection: 'row',

    }
})

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

银河中√捞星星 2025-01-17 20:00:11

我认为您没有将图像添加到 imageData 状态

listAll(listRef).then((res) => {
    res.items.forEach((itemRef) => {
        const downloadUrl = getDownloadURL(ref(storage, itemRef))
        downloadUrl.then((url) => {
            const indexOfToken = url.indexOf("&token=")
            const token = url.slice(indexOfToken + 7)
            const image = {
                "imageUrl": url,
                "imageToken": token
            }
            eventImages.push(image);
        }).catch((error) => {
            switch (error.code) {
                case 'storage/object-not-found':
                    break;
                case 'storage/unauthorized':
                    break;
                case 'storage/canceled':
                    break;
                case 'storage/unknown':
                    break;
            }
        });
        console.log(eventImages)
        setImageData(eventImages) // <---- Add the images to the imageData state
    });
}).catch((error) => {
});

I think you are not adding the images to the imageData state

listAll(listRef).then((res) => {
    res.items.forEach((itemRef) => {
        const downloadUrl = getDownloadURL(ref(storage, itemRef))
        downloadUrl.then((url) => {
            const indexOfToken = url.indexOf("&token=")
            const token = url.slice(indexOfToken + 7)
            const image = {
                "imageUrl": url,
                "imageToken": token
            }
            eventImages.push(image);
        }).catch((error) => {
            switch (error.code) {
                case 'storage/object-not-found':
                    break;
                case 'storage/unauthorized':
                    break;
                case 'storage/canceled':
                    break;
                case 'storage/unknown':
                    break;
            }
        });
        console.log(eventImages)
        setImageData(eventImages) // <---- Add the images to the imageData state
    });
}).catch((error) => {
});
夏见 2025-01-17 20:00:11

我重新考虑了我的代码并且是错误的。希望对于初学者来说是正常的。

这是我的工作编码器,用于直接从 Google Fireabase 商店加载图库。

import React, { Component } from 'react'
import { StyleSheet, SafeAreaView } from 'react-native'
import { Image, FlatList } from "native-base"
import EventGalleryHeader from '../components/EventGalleryHeader.js'
import { getStorage, ref, getDownloadURL, list, listAll } from "firebase/storage"
import { LongPressGestureHandler, State } from 'react-native-gesture-handler'

export default class EventScreen extends Component {

    constructor(props) {
        super(props);
        this.storage = getStorage()
        this.pathToImages = '/eventimages/'
        this.eventImageSource = this.props.route.params.eventData.key
        this.imagesRef = this.pathToImages + this.eventImageSource
        this.state = {
            isLoading: true,
            images: [],
            event: {
                adress: this.props.route.params.eventData.adress,
                hosts: this.props.route.params.eventData.hosts,
                description: this.props.route.params.eventData.description,
                eventtitle: this.props.route.params.eventData.eventtitle,
                invitecode: this.props.route.params.eventData.invitecode,
                key: this.props.route.params.eventData.key,
                timestamp: this.props.route.params.eventData.timestamp,
            }
        }
    }

    componentDidMount() {
        this.getEventImageData()
    }

    componentWillUnmount() {
    }

    getEventImageData() {
        const images = []
        const event = {
            adress: this.props.route.params.eventData.adress,
            description: this.props.route.params.eventData.description,
            eventtitle: this.props.route.params.eventData.eventtitle,
            key: this.props.route.params.eventData.key,
            timestamp: this.props.route.params.eventData.timestamp,
        }

        listAll(ref(this.storage, this.imagesRef))
            .then((res) => {
                res.items.forEach((itemRef) => {
                    // console.log(itemRef._location.path_)

                    getDownloadURL(itemRef)
                        .then((url) => {

                            const indexOfToken = url.indexOf("&token=")
                            const token = url.slice(indexOfToken + 7)
                            images.push({
                                "imageUrl": url,
                                "imageToken": token
                            });
                            this.setState({
                                images,
                                isLoading: false,
                                event
                            });
                            // console.log(this.state.images)
                        })
                        .catch((error) => {
                            switch (error.code) {
                                case 'storage/object-not-found':
                                    break;
                                case 'storage/unauthorized':
                                    break;
                                case 'storage/canceled':
                                    break;
                                case 'storage/unknown':
                                    break;
                            }
                        });


                });

            }).catch((error) => {
            });

    }


    onLongPress = (event) => {
        if (event.nativeEvent.state === State.ACTIVE) {
            alert("I've been pressed for 800 milliseconds");
        }
    };



    render() {

        return (
            <SafeAreaView style={styles.container} >
                <FlatList _dark={{ bg: "blueGray.900" }} _light={{ bg: "blueGray.50" }}
                    style={styles.list}
                    numColumns={2}
                    ListHeaderComponent={<EventGalleryHeader data={this.state.event} />}
                    data={this.state.images}
                    keyExtractor={item => item.imageToken}
                    renderItem={({ item }) => (
                        <LongPressGestureHandler
                            onHandlerStateChange={this.onLongPress}
                            minDurationMs={800}
                        >
                            <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} source={{ uri: item.imageUrl }} alt="Alternate Text" size="xl" />
                        </LongPressGestureHandler>
                    )}
                />
            </SafeAreaView>
        );

    };

}
const styles = StyleSheet.create({
    container: {
        flex: 1,
        marginTop: 25,

    },
    image: {
        maxHeight: 450,
        width: '100%',
        height: 200,
        overflow: 'hidden',
    },
    list: {
        alignSelf: 'center',
    },
    gallery: {
        flex: 1,
        width: '100%',
        flexDirection: 'row',

    }
})

I've reconsidered my code and was wrong. Hopefully normal for a beginner.

Here is my working coder to load a gallery directly from Google Fireabase Store.

import React, { Component } from 'react'
import { StyleSheet, SafeAreaView } from 'react-native'
import { Image, FlatList } from "native-base"
import EventGalleryHeader from '../components/EventGalleryHeader.js'
import { getStorage, ref, getDownloadURL, list, listAll } from "firebase/storage"
import { LongPressGestureHandler, State } from 'react-native-gesture-handler'

export default class EventScreen extends Component {

    constructor(props) {
        super(props);
        this.storage = getStorage()
        this.pathToImages = '/eventimages/'
        this.eventImageSource = this.props.route.params.eventData.key
        this.imagesRef = this.pathToImages + this.eventImageSource
        this.state = {
            isLoading: true,
            images: [],
            event: {
                adress: this.props.route.params.eventData.adress,
                hosts: this.props.route.params.eventData.hosts,
                description: this.props.route.params.eventData.description,
                eventtitle: this.props.route.params.eventData.eventtitle,
                invitecode: this.props.route.params.eventData.invitecode,
                key: this.props.route.params.eventData.key,
                timestamp: this.props.route.params.eventData.timestamp,
            }
        }
    }

    componentDidMount() {
        this.getEventImageData()
    }

    componentWillUnmount() {
    }

    getEventImageData() {
        const images = []
        const event = {
            adress: this.props.route.params.eventData.adress,
            description: this.props.route.params.eventData.description,
            eventtitle: this.props.route.params.eventData.eventtitle,
            key: this.props.route.params.eventData.key,
            timestamp: this.props.route.params.eventData.timestamp,
        }

        listAll(ref(this.storage, this.imagesRef))
            .then((res) => {
                res.items.forEach((itemRef) => {
                    // console.log(itemRef._location.path_)

                    getDownloadURL(itemRef)
                        .then((url) => {

                            const indexOfToken = url.indexOf("&token=")
                            const token = url.slice(indexOfToken + 7)
                            images.push({
                                "imageUrl": url,
                                "imageToken": token
                            });
                            this.setState({
                                images,
                                isLoading: false,
                                event
                            });
                            // console.log(this.state.images)
                        })
                        .catch((error) => {
                            switch (error.code) {
                                case 'storage/object-not-found':
                                    break;
                                case 'storage/unauthorized':
                                    break;
                                case 'storage/canceled':
                                    break;
                                case 'storage/unknown':
                                    break;
                            }
                        });


                });

            }).catch((error) => {
            });

    }


    onLongPress = (event) => {
        if (event.nativeEvent.state === State.ACTIVE) {
            alert("I've been pressed for 800 milliseconds");
        }
    };



    render() {

        return (
            <SafeAreaView style={styles.container} >
                <FlatList _dark={{ bg: "blueGray.900" }} _light={{ bg: "blueGray.50" }}
                    style={styles.list}
                    numColumns={2}
                    ListHeaderComponent={<EventGalleryHeader data={this.state.event} />}
                    data={this.state.images}
                    keyExtractor={item => item.imageToken}
                    renderItem={({ item }) => (
                        <LongPressGestureHandler
                            onHandlerStateChange={this.onLongPress}
                            minDurationMs={800}
                        >
                            <Image style={{ marginRight: 2, marginTop: 2, width: '50%', opacity: 1 }} source={{ uri: item.imageUrl }} alt="Alternate Text" size="xl" />
                        </LongPressGestureHandler>
                    )}
                />
            </SafeAreaView>
        );

    };

}
const styles = StyleSheet.create({
    container: {
        flex: 1,
        marginTop: 25,

    },
    image: {
        maxHeight: 450,
        width: '100%',
        height: 200,
        overflow: 'hidden',
    },
    list: {
        alignSelf: 'center',
    },
    gallery: {
        flex: 1,
        width: '100%',
        flexDirection: 'row',

    }
})
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文