¿Cómo implementar rutas y permisos avanzados en tu aplicación?
Gestionar rutas y permisos avanzados en una aplicación puede parecer un desafío monumental, pero con el enfoque adecuado, se transforma en uno de los aspectos más gratificantes del desarrollo web. Al aprender a manejar rutas dinámicas y condiciones de autorización, aseguramos un comportamiento lógico y seguro para todos los usuarios de la plataforma. Aquí, te mostraremos cómo abordar un reto que implica estas habilidades.
¿Por qué son importantes las rutas dinámicas?
Las rutas dinámicas nos permiten crear URLs específicas para cada usuario, personalizando la experiencia de navegación y organización dentro de la aplicación. En el caso de Platzi, por ejemplo, las rutas de perfil son únicas para cada usuario, comparables a cómo Twitter gestiona sus URLs de perfil.
Para lograr esto, utilizamos el hook useParams de React Router que nos permite capturar parámetros dinámicos de la URL y mostrarlos de manera adecuada.
import{ useParams }from'react-router-dom';constUserProfile=()=>{const{ username }=useParams();// lógica para mostrar el perfil del usuario};
¿Cómo se gestionan los permisos de edición?
No todas las rutas son solo para mostrar información; algunas requieren capacidades de edición. La clave está en permitir que solo el dueño del perfil o un administrador autorizado puedan realizar cambios.
Verificación de la cuenta: Comparamos la información del usuario autenticado con la del perfil accedido. Si coinciden, mostramos opciones adicionales de edición.
Roles de usuario: Implementamos roles, donde el usuario con permisos de administrador puede modificar cualquier perfil, mientras que los usuarios estándar solo pueden editar el suyo.
¿Cuál es el siguiente paso en la aplicación de estos conceptos?
Una vez que comprendas y pratices la creación de rutas dinámicas y la gestión de permisos a detalle, es crucial aplicar tu conocimiento en proyectos reales. La mejor manera de consolidar lo aprendido es enfrentarte a retos, como el que hemos propuesto, para afinar estas habilidades.
A medida que te adentras en este proceso, recuerda:
Interactuar con la comunidad: Comparte tus avances y problemas en foros o plataformas como Platzi. Aprender de las experiencias de tus pares es invaluable.
Documentar tus intentos: Sin importar el resultado, documentar tus pasos te permite evaluar qué funcionó y qué no, y compartirlo puede ser una fuente de aprendizaje para otros.
Iteración y mejora continua: El desarrollo es un viaje de aprendizaje constante. Experimenta con distintas soluciones y mejoras a medida que avanzas.
Siguiendo estos consejos y enfrentando los desafíos con determinación y curiosidad, estarás un paso más cerca de dominar la creación de aplicaciones complejas con React y React Router. ¡Adelante y sigue aprendiendo!
Aquí les comparto mi progreso del curso y los retos aplicados. Si alguien le sirve de referencia o si alguien gusta proporcionar algo de feedback, ya sea en buenas practicas o sugerencias, se agradece y mucho.
Repo.
GitHub page.
Advierto que sólo trabajé funcionalidad, por lo que no esperen un trabajo estético xD.
Los usuarios admin del sistema son: Irisval, RetaxMaster, freddier, alex. Los usuarios "normales" son juandc, nameless.
Hardcoded data.
Para manejar los datos "hardcodeados" opté por crear archivos para la información del usuario (user.js), blogs (blogdata.js) y los roles de usuario (roles.js). Todos ubicados en src/data/.
Inspirandome en aportaciones dadas por compañeros en clases anteriores, removí la lista de Admins y agregué nuevas propiedades como phonenumber, description y roles, este último siendo un objeto el cual guarda un arreglo de roles.
Para el manejo de roles implementé el uso de constantes. En este caso sólo creé dos roles: ADMIN (admin) y USER (user).
Agregué a los blogs una propeidad comments el cual, obviamente, guarda un arreglo de comentarios.
La estructura de un comentario es la de un objeto con las propiedades content y author.
Services.
Las funcionalidades de Auth del archivo auth.js visto en el curso las trabajé como un servicio. Por lo que creé la carpeta src/services/.
Por comodidad creé un servicio user.js. Esto con el fin de manipular la data del usuario y desarrollar funciones acorde a este. Buscando mantener auth.js lo más posible a sólo cosas relacionadas a funciones de autentificación.
Creé un servicio que funciona como la BD de usuarios. usersDB.js. Esto con el fin de servir de apoyo para las funciones de actualización de usuarios.
Para la manipulación de los blogs opté por crear un servicio al igual que auth: blog.js. Siendo este una copia del auth.js ya que utilizo la mismas funciones o características. Un provider, un useBlog y BlogRoute (ojo en este último).
BlogRoute. Durante el proceso me di cuenta que si accedía a un blog que no existe la app truena. Por lo que redirecciono al usuario en caso de que no exista el blog que indica en la url.
BlogProvider/BlogContext. Además de gestionar los blogs, también sus comentarios. Dejé todos los métodos dentro de este servicio, aunque otra opción sería crear un servicio de comentarios para manipularlos desde ahí utilizando useBlog. Algo parecido a useUser.
Gestión de blogs.
Habilité las funciones de eliminar blogs. Permitiendo dicha función unicamente al propietario del blog o los usuarios admin.
Añadí un formulario para la creación de nuevos blogs (sólo para usuarios con una sesión iniciada).
Añadí una sección de comentarior en el detalle de un blog y la función de agregar un nuevo comentario (sólo para usuarios con una sesión iniciada).
Al igual que los blogs, los usuarios pueden borrar sus comentarios. Pero un usuario admin puede cualquier comentario.
UX login.
En caso de no tener una sesión iniciada, la vista blog cuenta con un botón para dirigir al usuario a la vista de login. Una vez iniciada la sesión, el usuario volverá a la vista de blog.
La funcionalidad anterior también fue implementada desde la sección de comentarios de un blog.
Editar perfil de usuario.
Además de agregar más datos para el perfil de los usuarios, también agregué la posibilidad de cambiar estos datos.
Únicamente permito actualizar el número de teléfono y la descripción del usuario.
username no es un campo válido a modificar (al menos en este proyecto) debido a que funciona como el identificador del usuario y los blogs y comentarios los relaciono a partir de este.
Es posible ver el perfil de otros usuarios ingresando su username en la ruta: e.j. profile/juandc.
Los usuarios admin pueden modificar el perfil de otros usuarios.
Posibles mejoras.
Agregar estilos (por supuesto xD).
Implementar función de Editar (blogs y comentarios).
Guardar comentarios con algún identificador y/o datetime para facilitar la gestión de estos.
El propietario del blog sea capaz de eliminar u ocultar comentarios de sus blogs.
Uso de algún ID para los usuarios (number, hash, etc.).
Editar roles de usuario.
Ajustes en los servicios.
Fuaaaa de lujo!! Muy buen trabajo :clap:
Está muy bueno Martín! Un montón de funcionalidades. Algunas mejoras que espero te ayuden:
Poder entrar con un usuario no logeado (pestaña sign up)
Agregar más usuarios (si hay, no los pude encontrar desde la página)
Arreglar error que salta al pulsar "enter' cuando se quiere agregar un comentario o un blog,
De nuevo, muy buen trabajo! Y, me podrías decir como solucionaste el reto de esta clase? Ya que no encuentro más usuarios, no puedo acceder a sus perfiles.
Lo logré!
Después de muchos intentos y muchas pruebas, pude hacerlo sin buscar la respuesta en internet.
.
Lo primero que hice fue cambiar la forma en la que se logeaban los usuarios. Agregue la clase User para que todos tengan la misma estructura.
Esa fue la parte fácil. Lo segundo fue usar useParams() en ProfilePage. Cambie la ruta de '/profile' a '/profile/:username', luego hice comprobaciones para saber si se trataba de la cuenta del usuario logeado o no.
ProfilePage.js
export function ProfilePage() {
const { user, users } = useAuth()
const { username } = useParams()
let isAuthor
const userProfile = users.find(user => user.username === username)
if (!userProfile) return <p>sorry! {username} isn't a logged user.</p>
if (userProfile === user) isAuthor = true
if (isAuthor)
return (
<>
<h2>Hello, {username}</h2>
<button>Edit your profile here!</button>
</>
)
return <h2>Profile of {username}</h2>
}
Surgió el problema que más me costó resolver: el link en Menu redirigia a '/profile', pero esa ruta ya no existía. Probé muchas cosas.
Cambiar la ruta a '/profile/${user.username}', pero el menu no se re-renderizaba al cambiar el user, resultando en que siempre lleve al primer usuario logeado.
Agregar un Navigate a ProfilePage si username no existía, pero claro, la ruta '/profile' en ningún momento llevaba a mi componente.
Usar Outlet (ni yo sabía como) o renderProps (no me acuerdo por qué pensé que eso tendría sentido)
Agregar Navigate to='/profile/{user.username}'/> en App, pero App no tiene acceso a user.
Finalmente, agregué un componente ProfileRedirect en ProfilePage.js
ProfilePage.js
exportfunctionProfileRedirect(){const{ user }=useAuth()if(user)return<Navigate to={`/profile/${user.username}`}/>return<NotFound/>}
Siento que no quería llegar a esta solución, no me parece la más elegante, pero es la que conseguí después de muchos intentos. Y funciona perfecto!
.
Por último, cualquier sugerencia va a ser agradecida. Gracias por leer :D
Esta es una GRAN solución, Cande. Felicitaciones. :clap::clap: :star2:
Muchísimas gracias!!
Mi solución
Cuando Juan mencionó todo el reto, por mi mente pasó un "Aaaaihhggg.... por dónde empiezo...."
.
Les describiré cómo se fue creando mi solución en mi cabeza:
.
Cambiar estructura de profiles
Antes solo podíamos mirar nuestro perfil, ahora tenemos que hacer que cualquier perfil pueda ver cualquier otro perfil.
La ruta "/profile" contiene un input con un buscador de perfiles (hice un array con perfiles hardcodeados). La ruta "/profile/:username" contiene la vista del perfil del usuario.
En el buscador de perfiles, verifico si el username ingresado en el input existe; si no existe, lanzo un alert(), si sí existe, navego hacia la ruta "/profile/${username}"
functionSearchUserPage(){const navigate =useNavigate();const[username, setUsername]=useState("");constsearchUser=(ev)=>{ ev.preventDefault();if(!users.some((user)=> user.username=== username)){alert("This user does not exist. Try another user.");return;}navigate(`/profile/${username}`);};return(<><h3>SearchUser</h3><form onSubmit={searchUser}><label htmlFor="username">Type an username</label><input
type="text" name="username" id="username" value={username} onChange={(ev)=>setUsername(ev.target.value)}/><button type="submit">Search</button></form></>);}
.
Renderizar y Autorización en Profile Page
En Profile Page, recojo el user autenticado y el user enviado por la URL.
Si los dos son el mismo user, tiene permiso para editar el perfil.
Si el user autenticado tiene el rol de admin, puede editar el perfil.
functionProfilePage(){const{ user }=useAuth();const params =useParams();const data = users.find((user)=> user.username=== params.username);constcanEditProfile=()=>{if(user.role.name==="admin")returntrue;if(user.username=== data.username)returntrue;returnfalse;};return(<><h2>ProfilePage</h2><h3>{data.username}</h3><img src={data.img.src} alt={data.img.alt} width="400"/><p>{data.bio}</p>{canEditProfile()&&<button>EditProfile</button>}</>);}
.
.
Mi filosofía para este reto
Lo que hago es simular una base de datos teniendo un array de usuarios en un archivo aparte, y en cada búsqueda o autenticación, lo consulto. Y dentro de cada componente hago la lógica necesaria trabajando los datos que tengo: array de usuarios y user autenticado (o no autenticado)
Llegue a tu misma solucion! 👩💻
Jaja me alegro! Viendo mis comentarios 3 años despues
Reto completado , muy muyyyy retador pero se logro
Repo :
Page :
Trabaje con Tailwind un diseño responsivo
Cuenta con usuarios con distintos permisos como Admin, Editor, Tester, User, solo tienes que seleccionar alguno, ya tiene la info predefinida, si quieres crear tu usuario propio lo puedes hacer mediante el storage ya que todo es manejado con localStorageAsincrono con un tiempo de retraso de 1segundo para simular traer la data de la base de datos
De igual forma los blogs se manejan mediante el LocalStorage, manejo datos por default que se insertan cuando el localStorage es nullo por tanto si rompes la app pones en la consola LocalStorage.clear() la info se resetea
Solo los tester y admin pueden ser capaces de crear blogs
Los editor pueden editar cualquier blog
Los usuarios cuyo blogs son propios puedes editarlo
Cualquier usuario logeado puede crear sus comentarios en cualquier blog, únicamente su creador los puede editar
El admin puede editar o borrar el blog
El usuario que coincide con el perfil de el puede editarlo, para editarlo necesita la contraseña, no puedes cambiar el rol
El admin puede editarlo y eliminarlo, al editar ya tiene la contraseña dada y puede cambiar los permiso, un admin puede hacer a todos admin
Toda la data sincronizada
Posibles mejoras
Que se pueda crear cualquier usuario y verificar que no colapse con usuarios existentes, y en el caso que quiera cambiar contraseña lo pueda hacer
Cualquier feedback es bien recibido, os invito a comentan si tienten dudas de como realice algo o como lo podría mejorar , comenten, saludos compañeros de Platzi
Estas son las vistas del admin
Por si desean usar el diseño este es el figma que diseñe
Hola amigos, les comparto el deploy de mi app y el repositorio por si a alguien le sirve
Github Pages - Deploy
Repositorio
La lista de admins son [ JuanDC - Freddier - Retaxmaster ]
La aplicación tiene persistencia en local storage de los blogs y usuarios, puedes crear tus propios blogs, puedes eliminar tus blogs pero si inicias sesión con otro user ya no podrás hacerlo, a no ser que sean un administrador, un admin puede hacer todo
Reto 1 😰 Reto 2 😃 Reto 3 😰
No lo logré. Comparto una forma de no hacerlo. Yo creé un array aparte para simular un "backend" y creé las rutas de la misma forma que en Blogpage y BlogPost
//Este es el "backend"const publicData =[];publicData.push({name:'Jhonny',slug_Url:'@jhonny-c',nickname:'@jhonny-c',});....
Utilize el contexto Para compartir el "backend". También creé un método signUp para crear una cuenta.
functionAuthProvider({ children }){const[newUser, setNewUser]=React.useState(publicData);...constsignup=({ username, nickname })=>{ un objeto que representa a un nuevo usuario
const newSignUp ={name: username,slug_Url:'@'+ nickname,nickname:'@'+ nickname,}//Crear una copia del "backend"const createUser =[...newUser]//concatenar el nuevo usuario al "backend" createUser.push(newSignUp);//cambiar el "backend" para toda la appsetNewUser(createUser);//esta parte es para comprobar si el usuario existeconst findUser = newUser.find(user=> user.name=== username);//el hook useLocationconstfrom=location.state?.from?.pathname ||-1;//si el usuario no existe findUser ===undefined?//crearlosetUser({name: username,nickname: nickname,role: roles.visitor})//si existe es auth.user:setUser(findUser)//el hook useLocationnavigate(from,{replace:true});}const auth ={ user,... newUser,};
La sub-ruta de ProfilePage es PublicProfile asi se ve desde App.js
functionPublicProfile(){const{ slug }=useParams();const{ user, newUser}=useAuth();const profile = newUser?.filter(person=> person.slug_Url=== slug);const verified = user === profile.name;return(<>{profile &&!verified && profile.map(p=><ProfilesPub props={p}/>)}{profile && verified && profile.map(p=><Profiles props={p}/>)}</>);}//este componente es para cuando hace login y le permite modificar el perfilfunctionProfiles({props}){return(<><div><h2>Profile</h2><h4>{props.name}</h4><button>editar</button><div></div><button>editar</button><div>Nickname:{props.nickname}</div><button>editar</button></div></>);}//este componente solo muestra el perfil público y no permite modificaciones.functionProfilesPub({ props }){return(<><div><h2>Profile</h2><h4>{props.name}</h4><div></div><div>Nickname:{props.nickname}</div></div></>);}
Reitero esta es una forma que no funciona. Desde LoginPage o SignUpPage se crea la sub-ruta directamente. Además, la sub-ruta solo es disponible después de hacer logout.
La solución que logré construir está basada en un arreglo que contiene la hipotética información privada de cada usuario registrado en la App, la cual alojé en un componente al que le dí el nombre de userdata.jsx:
Inicialmente construí el botón "Editar Perfil" dentro del componente ProfilePage.jsx, que al final de todo el proceso, me quedó de la siguiente forma:
import{ useAuth }from"./auth";import{ useNavigate, useParams,Outlet}from"react-router-dom";functionProfilePage(){const auth =useAuth();const navigate =useNavigate();const{username}=useParams();const isOwnProfile = username === auth.user?.username;const canEdit =(isOwnProfile || auth.user?.isAdmin)&& auth.user?.validUser;consteditProfile=()=>{navigate(`/profile/${username}/edit`);};return(<><h1>ProfilePage</h1>{isOwnProfile ?(<><p>Welcome,{auth.user.username}</p></>):(<><p>Este es el perfil público de {username}</p></>)}{canEdit &&<button onClick={editProfile}>Editar perfil</button>}<Outlet/></>);}export{ProfilePage};
Para validar la existencia del usuario logueado en alguna de las 2 listas (adminList o userdate), fue necesario replantear el login dentro del componente auth.jsx, que al final de todo el proceso, me quedó así:
Para que el parámetro dinámico del username sea aceptado, fue necesario ajustar el sistema de rutas del componente App.jsx, que al final del proceso, lucía así:
Finalmente, al componente que renderiza la información privada de los usuarios registrados en la App se le asignó el nombre de EditProfile.jsx y después de construir toda la solución, luce de la sigiente forma:
import{useParams, useNavigate }from'react-router-dom';import{ userdata }from"./userdata";functionEditProfile(){const navigate =useNavigate();const{username}=useParams();const user = userdata.find((u)=> u.name=== username);const goBack =()=>navigate(-1);// volver a la vista anteriorreturn(<><h2>Editando perfil de {user.name}</h2><p><strong>Email:</strong>{user.email}</p><p><strong>Ciudad:</strong>{user.city}</p><p><strong>Edad:</strong>{user.age}</p><p><strong>Rol:</strong>{user.role}</p><button onClick={goBack}>Volver</button>);}export{EditProfile};
Mi solución:
1) Para crear las rutas dinámicas, es básicamente mirar lo que hicimos anteriormente con el contenido de los blogs.
Primero crear un componente donde renderizaremos las paginas dinámicas.
y con ello hacer la modificación permitente en nuestro Router
<Route path="/profile" element={<AuthRoute><Profile/></AuthRoute>}><Route path=":users" element={<ProfileUser/>}/></Route>```2) Modificar el valor de nuestra ruta profile que tenemos en menu.jsx
```js
{to:`/profile/${auth.user?.userName}`,text:'Profile',private:true,}```Con esto ya logramos el tener paginas dinámicas al ir a profile según el usuario con el que ingresemos en login.
3\) ahora es hora de dar cierta personalización en cada pagina, para ello elegí darle mas información a cada usuario, este array decidí agregarlo un estado, pero al tener mas de 3 estados en auth.jsx opte por crear un estado compuesto para tener un mejor manejo:
```js
const userList =[{user:'SrPizza',nickname:'Sr_Pizza',location:'Bogotá, Col',role:'admin'},{user:'Percy',nickname:'Perrito_Percy',location:'Bogotá, Col',role:'admin'},{user:'Nicolas',nickname:'XnicolasG',location:'NY, US',role:'poster'}];
const[authState, setAuthState]=useState({user:null,dataBlog:Blogdata,prevPath:null,userList: userList
})```con esto podemos renderizar la información en cada perfil según el usuario primero ubicando los datos en el array segun nuestro slug en este caso lo llame 'users'
```js
const{ users }=useParams();letUsers= auth.userList.find(us=> us.user=== users)```con esto ya podemos renderizar la información de nuestro usuario especifico dandole cierta diferencia de si el usuario se encuentra registrado en nuestro array o no
```js
return(<section>{Users?(<><h2>{Users.user}-({Users.role})</h2><h4>{Users.nickname}</h4><p>{Users.location}</p></>):(<h2>{auth.user.userName}-(Visitor)</h2>)}</section>)```4) Por último decidí que se pueda editar el nickname, pero solamente si es la persona es dueña del perfil, o si su rol es admin.
Lo primero es dictar un condicional inicialmente creando una variable que use para la condición 
```js
const canChangeName =Users|| auth.userList.some((us)=> us.user=== auth.user?.userName && us.role==='admin')```una vez creada ya podemos generar la condición en la cual genere un pequeño formulario con un input y un boton:
```js
return(<section>{Users?(<><h2>{Users.user}-({Users.role})</h2><h4>{Users.nickname}</h4><p>{Users.location}</p>{ canChangeName &&(<form onSubmit={changeName}><label htmlFor="Nuevo nombre"></label><input
value={newName} onChange={(e)=>setNewName(e.target.value)} type="text"/><button type='submit'>Cambiar nombre</button></form>)}</>):(<h2>{auth.user.userName}-(Visitor)</h2>)}</section>)}```5) para generar la acción de editar el nickname y vincularlo al formulario lo hice de esta manera:
```js
constchangeName=(e)=>{ e.preventDefault();const profileUser =Users.user;const findUser = auth.userList.find((us)=> us.user=== profileUser);if(findUser){ findUser.nickname= newName; auth.setAuthState((prevState)=>({...prevState,userList:Array.isArray(prevState.userList)?[...prevState.userList]:[]}))}}
Para la variable de quien puede editar o no los perfiles tuve que realizar la siguiente modificación
Complete los 3 retos, la parte a la que agregé mayor funcionalidad fue al blog, ya que se puede editar y eliminar comentarios, el titulo y el contenido del BlogPost. Para lograr todas las validaciones de los roles y las acciones permitidas y no permitidas en el blog use un nuevo custom hook que hice yo mismo. Espero les agrade mi solución. No creo que sea la forma más optima de hacerlo, pero me sirvió para reforzar mas cosas sobre el estado, la composición de componentes, useReducer, y los closures.import React, { useReducer } from "react";
const initialState = { //role:"visitor", read:true, selfWrite:false, selfDelete:false, elseWrite:false, elseDelete:false,};
//Action Typesconst actionTypes = { admin: 'admin', editor: 'editor', customer: 'customer', visitor: 'visitor',}
const reducerObject = (state, action) => { const actions = { [actionTypes.admin]: { ...state, userName:action.payload, role:actionTypes.admin, selfWrite:true, selfDelete:true, elseWrite:true, elseDelete:true, }, [actionTypes.editor]: { ...state, userName:action.payload, role:actionTypes.editor, selfWrite:true, selfDelete:true, elseWrite:false, elseDelete:false, }, [actionTypes.customer]: { ...state, userName:action.payload, role:actionTypes.customer, selfWrite:true, selfDelete:true, elseWrite:false, elseDelete:false, }, }; return actions[action.type] || initialState;};
export function useAuthReducer() {
const [state, dispatch] = useReducer(reducerObject, initialState); //Action creators const onAdmin = (userName) => {dispatch({type: actionTypes.admin, payload:userName});} const onEditor = (userName) => {dispatch({type: actionTypes.editor, payload:userName});} const onCustomer = (userName) => {dispatch({type: actionTypes.customer, payload:userName});} const onvisitor = () => {dispatch(initialState);} // const onWrite = (eventValue) => {// dispatch({type: actionTypes.write, payload:eventValue});// }//create a switch expression executing the 3 above functions according to the userRole parameter const dispatchUser = (user)=>{ if(!!user){ switch (user.userRole) { case 'admin': onAdmin(user.userName); break; case 'editor': onEditor(user.userName); break; case 'customer': onCustomer(user.userName); break; default: break; }}else{ onvisitor(); } } return [state, dispatchUser] // const onDelete = (eventValue) => {}
Custom hook, useBlogContent,js:
importReactfrom'react'exportconstuseBlogContent=({ contentToChange, actions, setBlogdataLocal })=>{const{ onEdit, onDelete }= actions
const[edit, setEdit]=React.useState(false);const[inputContent, setInputContent]=React.useState(contentToChange);const[loading, setLoading]=React.useState(false);consthandleEdit=()=>{setEdit(true);};consthandleSave=()=>{setLoading(true);setInputContent(inputContent);setBlogdataLocal(onEdit);setEdit(false);setLoading(false);};consthandleDelete=()=>{setLoading(true);setBlogdataLocal(onDelete);setLoading(false);};consthandleCancelEdit=()=>{setEdit(false);const currentValue = inputContent;setInputContent(currentValue);};return{ edit, inputContent, loading, setInputContent, handleEdit, handleSave, handleDelete, handleCancelEdit,};};```Componente BlogPost (Composicion de componente, permisos, usa el customhook useBlogContent):
```js
importReactfrom'react'import{ useParams, useNavigate }from'react-router-dom'//import { blogdata } from './blogData'import{ useAuth }from'./auth'importCommentsfrom'./Comments/Comments'importCommentfrom'./Comments/Comment'importRequestButtonfrom'./Buttons/RequestButton'import{ useBlogContent }from'./useBlogContent'functionBlogPost({blogDataStates}){const[blogdataLocal, setBlogdataLocal]= blogDataStates
const navigate =useNavigate()const auth =useAuth()const{ user }= auth
const{ slug }=useParams()const{ title, content, author, comments }= blogdataLocal.find(post=> post.slug=== slug)constreturnToBlog=()=>{navigate('/blog')}const updateActions ={'EDIT':'EDIT',}constupdateTitlePost=(action, inputContent)=>( blogdataLocal.map(post=>{if(post.slug=== slug){if(action === updateActions.EDIT){return{...post,title: inputContent
};}}return post;}))constupdateContentPost=(action, inputContent)=>( blogdataLocal.map(post=>{if(post.slug=== slug){if(action === updateActions.EDIT){return{...post,content: inputContent
};}}return post;}))constdeletePost=()=>{const updatedBlogData = blogdataLocal.filter(post=> post.slug!== slug);setBlogdataLocal(updatedBlogData);navigate('/blog')}const{edit:editTitle,inputContent: inputContentTitle,loading: loadingTitle,setInputContent: setInputContentTitle,handleEdit: handleEditTitle,handleCancelEdit: handleCancelEditTitle,handleSave: handleSaveTitle,}=useBlogContent({contentToChange: title,actions:{onEdit:()=>updateTitlePost(updateActions.EDIT, inputContentTitle),}, setBlogdataLocal,});const{edit:editContent,inputContent: inputContentPostContent,loading: loadingContent,setInputContent: setInputContentPostContent,handleEdit: handleEditContent,handleCancelEdit: handleCancelEditContent,handleSave: handleSaveContent,}=useBlogContent({contentToChange: content,actions:{onEdit:()=>updateContentPost(updateActions.EDIT, inputContentPostContent),}, setBlogdataLocal,});return(<><header>{!editTitle &&<h3>{title}</h3>}{editTitle &&(<input
type="text" value={inputContentTitle} onChange={(e)=>setInputContentTitle(e.target.value)}/>)}{!editTitle &&(<div>{(user?.elseWrite || user?.userName === author)&&(<button onClick={handleEditTitle}>Editar título</button>)}</div>)}{editTitle &&(<div><RequestButton loading={loadingTitle} requestType={"EDIT"} onClick={handleSaveTitle}>Guardar</RequestButton><button onClick={handleCancelEditTitle}>Cancelar</button></div>)}</header><br /><button onClick={returnToBlog}>Volver al blog</button><p>{author}</p>{!editContent &&<p>{content}</p>}{editContent &&(<textarea
value={inputContentPostContent} onChange={(e)=>setInputContentPostContent(e.target.value)}/>)}{!editContent &&(<>{user?.elseWrite &&<button onClick={handleEditContent}>EditarContenido</button>}{user?.userName === author &&(!user?.elseWrite ||!user?.elseDelete)&&<button onClick={handleEditContent}>EditarContenido</button>}</>)}{editContent &&(<div><RequestButton loading={loadingContent} requestType={"EDIT"} onClick={handleSaveContent}>Guardar</RequestButton><button onClick={handleCancelEditContent}>Cancelar</button></div>)}<br />{user?.elseDelete &&<RequestButton requestType={"DELETE"} onClick={deletePost}>BorrarPost</RequestButton>}{user?.userName === author &&(!user?.elseWrite ||!user?.elseDelete)&&<RequestButton requestType={"DELETE"} onClick={deletePost}>BorrarPost</RequestButton>}<br /><Comments>{comments.map(({ idUser, content }, index)=>(<Comment key={index} idUser={idUser} content={content} slug={slug} setBlogdataLocal={setBlogdataLocal} blogdataLocal={blogdataLocal}/>))}</Comments>{/*crea */}</>);}exportdefaultBlogPost```Componente Comment (Composicion de componente, permisos, usa el customhook useBlogContent):
```js
importReactfrom'react'import{ users, useAuth }from'../auth'importRequestButtonfrom'../Buttons/RequestButton';import{ useBlogContent }from'../useBlogContent.js';functionComment({idUser, content, slug, blogdataLocal, setBlogdataLocal}){const author =findUser(idUser)const{ user }=useAuth()const updateActions ={'EDIT':'EDIT','DELETE':'DELETE'}constupdatedBlogData=(action, inputContent)=>( blogdataLocal.map(post=>{if(post.slug=== slug){let updatedComments;if(action === updateActions.DELETE){ updatedComments = post.comments.filter(comment=> comment.idUser!== idUser);}elseif(action === updateActions.EDIT){ updatedComments = post.comments.map(comment=>{if(comment.idUser=== idUser){return{...comment,content: inputContent
};}return comment;});}return{...post,comments: updatedComments
};}return post;}))const{ edit, inputContent, loading, setInputContent, handleEdit, handleCancelEdit,handleDelete: handleDeleteComment, handleSave,}=useBlogContent({contentToChange: content,actions:{onEdit:()=>updatedBlogData(updateActions.EDIT,inputContent),onDelete:()=>updatedBlogData(updateActions.DELETE, inputContent),}, setBlogdataLocal,});return(<article><h5>{author.userName}</h5><div>{!edit &&<p>{content}</p>}{edit &&<textarea value={inputContent} onChange={(e)=>setInputContent(e.target.value)}/>}</div>{!edit &&<div>{user?.elseWrite &&<button onClick={handleEdit}>Editar</button>}{user?.elseDelete &&<RequestButton requestType={'DELETE'} onClick={handleDeleteComment} loading={loading}>Borrar</RequestButton>}{user?.userName === author.userName&&(!user?.elseWrite||!user?.elseDelete)&&(<><button onClick={handleEdit}>Editar</button><RequestButton requestType={'DELETE'} loading={loading} onClick={handleDeleteComment}>Borrar</RequestButton></>)}</div>}{ edit &&<div><RequestButton loading={loading} requestType={'EDIT'} onClick={handleSave}>Guardar</RequestButton><button onClick={handleCancelEdit}>Cancelar</button></div>}</article>);}//a function that iterates over a list of objects and return the object that have the same userName that i am looking forexportfunctionfindUser(idUser){let user = users.find(user=> user.id=== idUser) user = user ? user:{userName:'Anónimo'}return user
}exportdefaultComment
auth.js (aqui resuelvo el segundo y tercer reto, además uso el reducer para permitir la escalabilidad de mas roles):
importReactfrom'react'import{ useNavigate,Navigate, useLocation }from'react-router-dom';import{ useAuthReducer }from'./AuthReducer'constAuthContext=React.createContext();const roles ={admin:'admin',editor:'editor',customer:'customer',visitor:'visitor',};exportconst users =[{id:1,userName:"Andres",role: roles.admin,},{id:2,userName:"Felipe",role: roles.editor,},{id:3,userName:"Pedro",role: roles.customer,},];exportfunctionAuthProvider({children}){//const [user, setUser] = useState(null)const[user, dispatchUser]=useAuthReducer();const navigate =useNavigate()constlogin=({ userName, callback })=>{//revisar si el usuario existe o lo crea como visitanteconst userFound = users.find((usu)=> usu.userName=== userName);if(userFound !==undefined){dispatchUser({userName: userFound.userName,userRole: userFound.role})}else{dispatchUser(null);}if(callback){callback();}else{navigate("/profile");}}constlogout=()=>{dispatchUser(null);navigate('/')}const auth ={ user, login, logout}return(<AuthContext.Provider value={auth}>{children}</AuthContext.Provider>)}exportfunctionuseAuth(){const auth =React.useContext(AuthContext)return auth
}exportfunctionRequireAuth(props){const auth =useAuth();letlocation=useLocation();if(!auth.user.role){return<Navigate to="/login" state={{from:location}} replace/>}return props.children}```AuthReducer.js:
```js
importReact,{ useReducer }from"react";const initialState ={//role:"visitor",read:true,selfWrite:false,selfDelete:false,elseWrite:false,elseDelete:false,};//Action Typesconst actionTypes ={admin:'admin',editor:'editor',customer:'customer',visitor:'visitor',}constreducerObject=(state, action)=>{const actions ={[actionTypes.admin]:{...state,userName:action.payload,role:actionTypes.admin,selfWrite:true,selfDelete:true,elseWrite:true,elseDelete:true,},[actionTypes.editor]:{...state,userName:action.payload,role:actionTypes.editor,selfWrite:true,selfDelete:true,elseWrite:false,elseDelete:false,},[actionTypes.customer]:{...state,userName:action.payload,role:actionTypes.customer,selfWrite:true,selfDelete:true,elseWrite:false,elseDelete:false,},};return actions[action.type]|| initialState;};exportfunctionuseAuthReducer(){const[state, dispatch]=useReducer(reducerObject, initialState);//Action creatorsconstonAdmin=(userName)=>{dispatch({type: actionTypes.admin,payload:userName});}constonEditor=(userName)=>{dispatch({type: actionTypes.editor,payload:userName});}constonCustomer=(userName)=>{dispatch({type: actionTypes.customer,payload:userName});}constonvisitor=()=>{dispatch(initialState);}//a switch expression executing the 3 above functions according to the userRole parameterconstdispatchUser=(user)=>{if(!!user){switch(user.userRole){case'admin':onAdmin(user.userName);break;case'editor':onEditor(user.userName);break;case'customer':onCustomer(user.userName);break;default:break;}}else{onvisitor();}}return[state, dispatchUser]// const onDelete = (eventValue) => {}
Buenas, me demore un poco pero lo logre.
Le puse algo de amor con bootstrap ❤️ My Blog
Hola a comunidad, les dejo mi código del proyecto en typescript.
Tome varias ideas interesantes de otros comentarios, espero el mio tambien les sea de ayuda:
exportinterfaceIUser{username: string,alias: string,description: string,rol:Rol,}exportenumRol{Admin='admin',User='user',}exportconstusersData:IUser[]=[{username:'Steve',alias:'Capitan America',description:'I am Steve Rogers',rol:Rol.User,},{username:'Tony',alias:'Iron Man',description:'I am Tony Stark',rol:Rol.Admin,},{username:'Bruce',alias:'Hulk',description:'I am Bruce Banner',rol:Rol.User,},{username:'Hank',alias:'Hombre Hormiga',description:'I am Hank Pym',rol:Rol.Admin,}]
UFFFFFFFFFFFFFFFFFFF! DESPUES DE 3 DIAS! Tuve que refactorizar todo, estaba muy perdido. Entonces comenze un nuevo proyecto desde 0.
Implemente una pagina Profiles que solo tiene acceso el rol admin. apartir de esto una ruta publica de profiles/:user para ver el perfil de un usuario.
En resumen: CREE LAS FUNCIONALIDADES DE:
-Crear un usuario.
-Editar un usuario
-Ver la informacion del usuario
-Crear un blog
-Editar un blog
Dejo el repositorio. Me quedan los estilos y proteger algunas rutas. pero esta es mi forma de solucionar hasta el momento los retos dados hasta ahora:
Que mas... pues nada, se me complico un poco implementar el componente ProtectedRoute y algunas protecciones son dadads en el mismo componente.
Condiciones:
La persona debe estar autenticada.
Si es dueño de la cuenta; puede ver el botón de edit profile en su perfil.
Si el usuario está autenticado y viaja a la ruta de otro perfil no podrá ver el botón de edit profile, a menos que este usuario autenticado sea un admin.
Para realizar el reto:
Cambié la ruta del componente <ProfilePage> a una ruta dinámica:
En mi custom hook useAuth.jsx tengo los roles de los usuarios y manejo el estado del username logueado. Los envío en el return del hook.
Ahora debo compartir estos datos al componente <Menu> y al componente <ProfilePage>; esto lo hago con el context del outlet y con las props para el <Menu>. Todos estos componentes están contenidos en un único componente <Layout>
Bueno, ahora hay que decirle al menú que el link que lleva a ProfilePage es dinámico; se hará de esta manera, uso una constante donde guardo la ruta con el dato dinámico según el usuario logueado y luego se la paso como valor al parámetro "to" del objeto que contiene la información del link a ProfilePage.
Ahora finalmente en el componente <ProfilePage> se debe renderizar un botón de edit profile dependiendo si el usuario es el dueño de la cuenta o no. Se haría de esta manera:
Guardo el parámetro y traigo los datos del contexto, luego, valido si el rol del usuario autenticado es un admin o no. Ahora con estos datos renderizo o no el saludo de la primera etiqueta p y la información de la etiqueta button validando si es o no dueño de la cuenta o admin el usuario autenticado gracias a la dirección de la ruta.
Así se ve cuando es un usuario en su cuenta:
Así si ve si navega a otro perfil:
Espero haber entendido lo que se pedía en el reto. Pero queda algo pendiente, hacer que el botón actualice datos. La cosa es que mi información es hardcodeada en su mayoría ya que no uso un objeto literal que tenga todos esos datos y luego pueda cambiar sus propiedades, tampoco uso una API, base de datos o localstorage. Pero si tuviera la forma de cambiar los datos que está debajo de Count information:
Sería actualizando un estado que contenga un objeto literal con la información de ese perfil. La actualización sería con un componente formulario que actualice el estado y luego navegue al profilepage, el profilpage renderizaría datos nuevos.
Saludos, playzinautas
les comparto un proyecto que hice con los retos de las clases, me faltan detalles por mejorar, pero espero con ansias su feedback
GitHub Page.
usuario admin clave admin123
usuario con role creator
juandc clave admin123
usuarios con role editor
nicobytes clave admin123
diannerd clave123
se me ocurrió separar los componentes del blog, y hacer lo mismo que hicimos en las rutas cuando usamos "AuthRoute" pero lo tome como un AuthAccess con el valido los posibles roles y ya sé que componente mostrar y cual no, pero me parece que se complica ya que se debe generar uno para cada componente
Aquí mi repositorio de lo que hice se pueden editar y eliminar los blogs solo si tienes los permisos se puede acceder a el de forma publica cree un custom hook para la info de los blogs, acciones y comparto la información en base a los patrones de render del curso anterior no se si es exactamente lo que se buscaba pero me gusto como quedo
Esto fue lo que hice para el reto (agregué una sección de comments donde sólo puedes comentar si estás loggeado):
Adicionalmente hice una ligera modificación en el hook de Auth para redirigir de forma inteligentemente si el login fue a través de la sección de comments o si fue de alguna otra parte del sitio: