Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

starter-redux-assessment : Submitting the code #2

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions src/features/photos/create/index.js
Original file line number Diff line number Diff line change
@@ -1,13 +1,16 @@
import { useState } from 'react';

// Task 2: Import the `useDispatch()` method from the appropriate package
import { useDispatch } from 'react-redux';
// Task 3: Import the `addPhoto()` action creator from the photos slice
import { addPhoto } from '../photos.slice';

import './create.css';

export default function CreatePhoto() {
const [formData, setFormData] = useState({ imageUrl: '', caption: '' });
// Task 4: Store a reference to the Redux store's dispatch method in a variable called `dispatch`
const dispatch = useDispatch();

function handleChange({ target: { name, value } }) {
setFormData({
Expand All @@ -19,6 +22,7 @@ export default function CreatePhoto() {
function handleSubmit(event) {
event.preventDefault();
// Task 5: Dispatch the `addPhoto()` action creator, passing in the form data
dispatch(addPhoto(formData));
setFormData({ imageUrl: '', caption: '' });
}

Expand Down
10 changes: 7 additions & 3 deletions src/features/photos/list/index.js
Original file line number Diff line number Diff line change
@@ -1,18 +1,22 @@
import { useSelector, useDispatch } from 'react-redux';
import {
// Task 7: Import the `removePhoto()` action creator from the photos slice
selectAllPhotos,
removePhoto,
// Task 13: Import the `selectFilteredPhotos()` selector from the photos slice
selectFilteredPhotos,
} from '../photos.slice';
import './list.css';

export default function PhotosList() {
// Task 14: Call `useSelector()` below with `selectFilteredPhotos` instead of `selectAllPhotos`
const photos = useSelector(selectAllPhotos);
const photos = useSelector(selectFilteredPhotos);
// Task 8: Store a reference to the Redux store's dispatch method in a variable called `dispatch`
const dispatch = useDispatch();

function handleDeleteButtonClick(id) {

function handleDeleteButtonClick(photoId) {
// Task 9: Dispatch the `removePhoto()` action creator, passing in the id
dispatch(removePhoto({ id: photoId }));
}

const photosListItems = photos.map(({ id, caption, imageUrl }) => (
Expand Down
12 changes: 12 additions & 0 deletions src/features/photos/photos.slice.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,20 @@ const options = {
// Task 1: Create an `addPhoto()` case reducer that adds a photo to state.photos.
// Task 1 Hint: You can use state.photos.unshift()
// `unshift()` documentation: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/unshift
addPhoto: (state, action) => {
state.photos.unshift(action.payload);
},

// Task 6: Create an `removePhoto()` case reducer that removes a photo from state.photos
// Task 6 Hint: You can use state.photos.splice()
// `splice()` documentation: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/splice
// Task 6: Create a `removePhoto()` case reducer that removes a photo from state.photos.
removePhoto: (state, action) => {
const index = state.photos.findIndex(photo => photo.id === action.payload.id);
if (index !== -1) {
state.photos.splice(index, 1);
}
}
},
};

Expand All @@ -29,4 +39,6 @@ export default photosSlice.reducer;
export const selectAllPhotos = (state) => state.photos.photos;
export const selectFilteredPhotos = (state) => {
// Task 12: Complete `selectFilteredPhotos()` selector to return a filtered list of photos whose captions match the user's search term
const searchTerm = selectSearchTerm(state);
return state.photos.photos.filter(photo => photo.caption.toLowerCase().includes(searchTerm.toLowerCase()));
};
2 changes: 2 additions & 0 deletions src/features/search/search-bar/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,11 @@ import './search-bar.css';
export default function SearchBar() {
const searchTerm = useSelector(selectSearchTerm);
// Task 10: Store a reference to the Redux store's dispatch method in a variable called `dispatch`
const dispatch = useDispatch();

function handleChange({ target: { value } }) {
// Task 11: Dispatch the `setSearchTerm()` action creator, passing in the value of the search input
dispatch(setSearchTerm(value));
}

return (
Expand Down
8 changes: 6 additions & 2 deletions src/features/suggestion/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,20 +4,23 @@ import {
fetchSuggestion,
selectError,
selectLoading,
selectSuggestion
// Task 18: Import the `selectSuggestion()` selector from the suggestion slice
} from './suggestion.slice';
import './suggestion.css';

export default function Suggestion() {
// Task 19: Call useSelector() with the selectSuggestion() selector
// The component needs to access the `imageUrl` and `caption` properties of the suggestion object.
const suggestion = useSelector(selectSuggestion);
const loading = useSelector(selectLoading);
const error = useSelector(selectError);
const dispatch = useDispatch();

useEffect(() => {
async function loadSuggestion() {
// Task 20: Dispatch the fetchSuggestion() action creator
dispatch(fetchSuggestion());
}
loadSuggestion();
}, [dispatch]);
Expand All @@ -29,10 +32,11 @@ export default function Suggestion() {
render = <h3>Sorry, we're having trouble loading the suggestion.</h3>;
} else {
// Task 21: Enable the two JSX lines below needed to display the suggestion on the page
const { imageUrl, caption } = suggestion;
render = (
<>
{/* <img alt={caption} src={imageUrl} />
<p>{imageUrl}</p> */}
<img alt={caption} src={imageUrl} />
<p>{caption}</p>
</>
);
}
Expand Down
26 changes: 24 additions & 2 deletions src/features/suggestion/suggestion.slice.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,15 @@
import { createAsyncThunk, createSlice } from '@reduxjs/toolkit';

export const fetchSuggestion =
createAsyncThunk(/* Task 15: Complete the `createAsyncThunk()` function to load a suggestion from this URL: http://localhost:3004/api/suggestion */);
createAsyncThunk(/* Task 15: Complete the `createAsyncThunk()` function to load a suggestion from this URL: http://localhost:3004/api/suggestion */

'suggestion/fetchSuggestion',
async () => {
const response = await fetch('http://localhost:3004/api/suggestion');
return response.json();
}

);

const initialState = {
suggestion: '',
Expand All @@ -13,8 +21,21 @@ const options = {
name: 'suggestion',
initialState,
reducers: {},
extraReducers: {
extraReducers: builder => {
/* Task 16: Inside `extraReducers`, add reducers to handle all three promise lifecycle states - pending, fulfilled, and rejected - for the `fetchSuggestion()` call */
builder
// Task 16: Handle promise lifecycle states for fetchSuggestion call
.addCase(fetchSuggestion.pending, state => {
state.status = 'loading';
})
.addCase(fetchSuggestion.fulfilled, (state, action) => {
state.status = 'succeeded';
state.suggestion = action.payload;
})
.addCase(fetchSuggestion.rejected, (state, action) => {
state.status = 'failed';
state.error = action.error.message;
});
},
};

Expand All @@ -23,6 +44,7 @@ const suggestionSlice = createSlice(options);
export default suggestionSlice.reducer;

// Task 17: Create a selector, called `selectSuggestion`, for the `suggestion` state variable and export it from the file
export const selectSuggestion = state => state.suggestion.suggestion;

export const selectLoading = (state) => state.suggestion.loading;
export const selectError = (state) => state.suggestion.error;