我使用 React.js 建立了一個電影 Web 應用程式。當我呼叫API上的資料時,資料運作正常。但是,當我嘗試透過 API 呼叫 id
來切換頁面時,資料變得未定義。我使用 Redux 工具包來呼叫資料。
但是當我嘗試按下卡片時,我設定的從所選卡獲取資料之一的頁面卻像下圖一樣發生
我嘗試檢查控制台日誌,結果如下。
如果你想檢查程式碼
.env
sorry i can share api key VITE_BASE_URL=https://api.themoviedb.org/3 VITE_IMG_PATH=https://image.tmdb.org/t/p/w500 VITE_VIDEO_PATH=https://api.themoviedb.org/3/movie
api.js
import Axios from 'axios' export const movieList = Axios.create({ baseURL: `${import.meta.env.VITE_BASE_URL}` })
action.js
import { createAsyncThunk } from "@reduxjs/toolkit"; import { movieList } from "../../service/api/api"; export const fetchMovieAll = createAsyncThunk('movie/fetchMovieAll', async() => { try { const api = `api_key=${import.meta.env.VITE_TMDB_KEY}` const respone = await movieList.get(`discover/movie?${api}`) return respone.data.results } catch (error) { return error } }) export const fetchMovie = createAsyncThunk('movie/fetchMovie', async(id) => { try { const respone = await movieList.get(`/movie/${id}?api_key=${import.meta.env.VITE_TMDB_KEY}`) return respone.data.results } catch (error) { return error } })
切片.js
#import {createSlice} from '@reduxjs/toolkit' import * as movieAction from '../action' const movieDb = createSlice({ name: 'movieDb', initialState: { loading: false, data: [], entity: { id: null, title: '', release_date: '', popularity: '', poster_path: '', overview: '', vote_average: '', backdrop_path: '' }, error: null, }, extraReducers: (builder) => builder.addCase(movieAction.fetchMovieAll.pending, (state) => { state.loading = true }) .addCase(movieAction.fetchMovieAll.fulfilled, (state, action) => { state.loading = false, state.data = action.payload }) .addCase(movieAction.fetchMovieAll.rejected, (state) => { state.loading = true, state.error = action.payload }) .addCase(movieAction.fetchMovie.pending, (state) => { state.loading = true }) .addCase(movieAction.fetchMovie.fulfilled, (state, action) => { state.loading = false, state.entity = action.payload }) .addCase(movieAction.fetchMovie.rejected, (state) => { state.loading = true, state.error = action.payload }) }) export default movieDb.reducer;
store.js
import { configureStore } from "@reduxjs/toolkit"; import movieDbReducer from '../slice' export default configureStore({ reducer: { movieDb: movieDbReducer, } })
CardMovie.jsx
import React, { useEffect } from 'react' import { useDispatch, useSelector } from 'react-redux' import { fetchMovieAll } from '../../../manage/action' import style from './style.module.css' import { useNavigate } from 'react-router-dom' function CardMovie() { const dispatch = useDispatch() const {data} = useSelector((state) => state.movieDb) const navigate = useNavigate() const fetchDataMovie = async() => { dispatch(fetchMovieAll()) } const goToDetail = (id) => { navigate(`/detail/${id}`) } useEffect(() => { fetchDataMovie() console.log(data) },[]) return ( <> {data.map((item) => { return ( <div key={item.id} className={style['main-card']} onClick={() => goToDetail(item.id)}> <div className={style['card-content']}> <img src={`${import.meta.env.VITE_IMG_PATH}${item.poster_path}`} alt="poster" /> <div className={style['card-title']}> <p>{item.title}</p> </div> </div> </div> ) })} </> ) } export default CardMovie
詳細資訊.jsx
import React, { useEffect } from 'react' import { useDispatch, useSelector } from 'react-redux' import { useNavigate, useParams } from 'react-router-dom' import { fetchMovie } from '../../manage/action' function Detail() { const {id} = useParams() const dispatch = useDispatch() // const navigate = useNavigate() const {entity} = useSelector((state) => state.movieDb) const fetchDataMovie = async(movieId) => { dispatch(fetchMovie(movieId)) } useEffect(() => { fetchDataMovie(id) console.log(entity) },[]) return ( <> <div> <h1>Detail</h1> {/* <h5>{entity.title}</h5> */} </div> </> ) } export default Detail
App.jsx
import Home from "./page/Home" import Genre from "./page/Genre" import Detail from "./page/Detail" import './App.css' import {Outlet, RouterProvider, createBrowserRouter} from 'react-router-dom' function App() { const router = createBrowserRouter([ { path: '/', children: [ { index: true, element: <Home/> }, { path: 'genre', element: <Genre/> }, { path: 'detail/:id', element: <Detail/> } ] } ]) return ( <> <RouterProvider router={router}/> <Outlet/> </> ) } export default App
index.jsx
import React from 'react' import ReactDOM from 'react-dom/client' import App from './App.jsx' import 'bootstrap/dist/css/bootstrap.css' import 'bootstrap/dist/js/bootstrap.js' import { Provider } from 'react-redux' import store from './manage/store/index.js' ReactDOM.createRoot(document.getElementById('root')).render( <React.StrictMode> <Provider store={store}> <App /> </Provider> </React.StrictMode>, )
我正在嘗試做這個網站參考的事情。
如果我點擊一些卡片,結果就會變成這樣
但是我得到的是這樣的 和這個
P粉7162282452024-01-30 12:22:42
根據我所看到的, 問題應該出在這個 api 呼叫上
export const fetchMovie = createAsyncThunk('movie/fetchMovie', async(id) => { try { const respone = await movieList.get(`/movie/${id}?api_key=${import.meta.env.VITE_TMDB_KEY}`) return respone.data.results } catch (error) { return error }
})
應該是
export const fetchMovie = createAsyncThunk('movie/fetchMovie', async(id) => { try { const respone = await movieList.get(`/movie/${id}? api_key=${import.meta.env.VITE_TMDB_KEY}`) return respone.data } catch (error) { return error }