diff --git a/.vscode/settings.json b/.vscode/settings.json index 7bd07e2e82..7f1bff12d5 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -6,5 +6,8 @@ }, "[javascript]": { "editor.defaultFormatter": "vscode.typescript-language-features" + }, + "[javascriptreact]": { + "editor.defaultFormatter": "vscode.typescript-language-features" } } diff --git a/Pipfile.lock b/Pipfile.lock index 68cca804a6..6d7703c2a2 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -377,7 +377,6 @@ ], "markers": "python_version >= '3.9'", "version": "==2.10.1" - }, "python-dotenv": { "hashes": [ diff --git a/migrations/versions/5cf725964086_.py b/migrations/versions/cb544095a50b_.py similarity index 93% rename from migrations/versions/5cf725964086_.py rename to migrations/versions/cb544095a50b_.py index 3c4b693934..373038b2e3 100644 --- a/migrations/versions/5cf725964086_.py +++ b/migrations/versions/cb544095a50b_.py @@ -1,8 +1,8 @@ """empty message -Revision ID: 5cf725964086 -Revises: e882aaa7e01e -Create Date: 2024-11-30 11:31:26.832067 +Revision ID: cb544095a50b +Revises: d6e7917535b2 +Create Date: 2024-12-11 22:58:32.170013 """ from alembic import op @@ -10,8 +10,8 @@ # revision identifiers, used by Alembic. -revision = '5cf725964086' -down_revision = 'e882aaa7e01e' +revision = 'cb544095a50b' +down_revision = 'd6e7917535b2' branch_labels = None depends_on = None @@ -21,7 +21,7 @@ def upgrade(): op.create_table('breed', sa.Column('id', sa.Integer(), nullable=False), sa.Column('name', sa.String(length=30), nullable=True), - sa.Column('species', sa.Enum('perro', 'gato', 'ave', 'conejo', 'otro', name='species'), nullable=False), + sa.Column('species', sa.Enum('perro', 'gato', 'ave', 'conejo', 'reptil', 'otro', name='species'), nullable=False), sa.PrimaryKeyConstraint('id') ) op.create_table('pet', diff --git a/migrations/versions/65ec0a935ab9_.py b/migrations/versions/d6e7917535b2_.py similarity index 89% rename from migrations/versions/65ec0a935ab9_.py rename to migrations/versions/d6e7917535b2_.py index 6b1dc1c9a0..226461e361 100644 --- a/migrations/versions/65ec0a935ab9_.py +++ b/migrations/versions/d6e7917535b2_.py @@ -1,8 +1,8 @@ """empty message -Revision ID: 65ec0a935ab9 +Revision ID: d6e7917535b2 Revises: -Create Date: 2024-11-27 23:26:32.249259 +Create Date: 2024-12-03 00:34:48.862682 """ from alembic import op @@ -10,7 +10,7 @@ # revision identifiers, used by Alembic. -revision = '65ec0a935ab9' +revision = 'd6e7917535b2' down_revision = None branch_labels = None depends_on = None diff --git a/migrations/versions/e882aaa7e01e_.py b/migrations/versions/e882aaa7e01e_.py deleted file mode 100644 index 45864ac075..0000000000 --- a/migrations/versions/e882aaa7e01e_.py +++ /dev/null @@ -1,35 +0,0 @@ -"""empty message - -Revision ID: e882aaa7e01e -Revises: -Create Date: 2024-11-30 11:14:24.658166 - -""" -from alembic import op -import sqlalchemy as sa - - -# revision identifiers, used by Alembic. -revision = 'e882aaa7e01e' -down_revision = None -branch_labels = None -depends_on = None - - -def upgrade(): - # ### commands auto generated by Alembic - please adjust! ### - op.create_table('user', - sa.Column('id', sa.Integer(), nullable=False), - sa.Column('email', sa.String(length=120), nullable=False), - sa.Column('password', sa.String(length=80), nullable=False), - sa.Column('is_active', sa.Boolean(), nullable=False), - sa.PrimaryKeyConstraint('id'), - sa.UniqueConstraint('email') - ) - # ### end Alembic commands ### - - -def downgrade(): - # ### commands auto generated by Alembic - please adjust! ### - op.drop_table('user') - # ### end Alembic commands ### diff --git a/src/api/admin.py b/src/api/admin.py index 3eecb64140..12b3d411ea 100644 --- a/src/api/admin.py +++ b/src/api/admin.py @@ -1,7 +1,7 @@ import os from flask_admin import Admin -from .models import db, User +from .models import db, User, Pet, Post_Description, Breed from flask_admin.contrib.sqla import ModelView def setup_admin(app): @@ -12,6 +12,9 @@ def setup_admin(app): # Add your models here, for example this is how we add a the User model to the admin admin.add_view(ModelView(User, db.session)) + admin.add_view(ModelView(Pet, db.session)) + admin.add_view(ModelView(Post_Description, db.session)) + admin.add_view(ModelView(Breed, db.session)) # You can duplicate that line to add mew models # admin.add_view(ModelView(YourModelName, db.session)) \ No newline at end of file diff --git a/src/api/models.py b/src/api/models.py index 534e858ab4..82704652f4 100644 --- a/src/api/models.py +++ b/src/api/models.py @@ -48,7 +48,7 @@ class Pet(db.Model): __tablename__ ="pet" id = db.Column (db.Integer, primary_key=True) name = db.Column (db.String(30)) - breed = db.Column (db.Integer, db.ForeignKey("breed.id")) + breed = db.Column (db.Integer, db.ForeignKey("breed.id")) # bread_id gender = db.Column (db.Enum(Genders)) color = db.Column (db.String(15)) photo_1 = db.Column (db.String(120), nullable=False) @@ -58,15 +58,14 @@ class Pet(db.Model): user_id = db.Column (db.Integer, db.ForeignKey("user.id")) user = db.relationship ("User", back_populates="pet") post = db.relationship("Post_Description", back_populates="pet_relationship") - breed_relationship = db.relationship("Breed", back_populates= "pets") + breed_relationship = db.relationship("Breed", back_populates= "pets") # breed def __repr__(self): - return f'' + return f'' def serialize(self): return{ "name" : self.name, - "species" : self.species, "breed" : self.breed, "color": self.color, "photo_1": self.photo_1, diff --git a/src/app.py b/src/app.py index 8aa49f6c42..403fe831d8 100644 --- a/src/app.py +++ b/src/app.py @@ -218,6 +218,15 @@ def create_post_description(): return jsonify({'msg':'Post creado exitosamente', 'data': new_post.serialize()}), 201 ####### +#GET All pets Matias 17:46PM 3/12/24..Update: working 11:43AM 4/12/24 +@app.route('/pets', methods=['GET']) +def get_all_pets(): + pets = Pet.query.all() + pets_serialized = [] + for pet in pets: + pets_serialized.append(pet.serialize()) + return jsonify({'msg': 'ok', 'data': pets_serialized}), 200 + # any other endpoint will try to serve it like a static file @app.route('/', methods=['GET']) diff --git a/src/front/js/App.js b/src/front/js/App.js new file mode 100644 index 0000000000..9f848d4022 --- /dev/null +++ b/src/front/js/App.js @@ -0,0 +1,18 @@ +import React from "react"; +import { PasswordResetProvider } from "./PasswordResetContext"; // Importamos el proveedor +import ResetPassword from "./ResetPassword"; // Importamos el componente de recuperación de contraseña +import PetsView from "./PetsView"; // Vista de mascotas + +function App() { + return ( + +
+

Aplicación de Recuperación de Contraseña y Mascotas

+ {/* Componente de recuperación de contraseña */} + {/* Componente de vista de mascotas */} +
+
+ ); +} + +export default App; diff --git a/src/front/js/component/Filters.js b/src/front/js/component/Filters.js new file mode 100644 index 0000000000..c561a1cf71 --- /dev/null +++ b/src/front/js/component/Filters.js @@ -0,0 +1,46 @@ +import React from "react"; +import "../../styles/pets-view.css"; + +const Filters = ({ onFilterChange }) => { + return ( +
+

Filter

+
+ + +
+
+ + + +
+
+ + +
+
+ ); +}; + +export default Filters; diff --git a/src/front/js/component/PetCard.jsx b/src/front/js/component/PetCard.jsx new file mode 100644 index 0000000000..11338a6329 --- /dev/null +++ b/src/front/js/component/PetCard.jsx @@ -0,0 +1,225 @@ +import React, { useContext, useState, useEffect } from "react"; +import { Context } from "../store/appContext"; +import { useParams } from "react-router-dom"; + +const PetCard = () => { + const { store } = useContext(Context); + const { theid } = useParams(); + const [detail, setDetail] = useState(null); + const [mainImage, setMainImage] = useState("https://cdn.pixabay.com/photo/2023/11/12/17/12/puppy-8383633_1280.jpg"); // Imagen principal inicial + + const findDetail = () => { + const result = store.petData.find((item) => item.id == theid); + setDetail(result); + setMainImage(result?.image || "https://cdn.pixabay.com/photo/2023/11/12/17/12/puppy-8383633_1280.jpg"); // Establecemos la imagen principal + }; + + useEffect(() => { + findDetail(); + }, [theid]); + + return ( +
+ {/* Navigation path */} +
+ + Home + {" "} + >{" "} + + Mascotas + {" "} + > Más información +
+ +
+ {/* Imagen y carrousel */} +
+ {/* Imagen principal */} +
+ Main Pet +
+ + {/* Miniaturas debajo de la imagen principal */} +
+ Thumbnail 1 setMainImage("https://via.placeholder.com/600x400")} + /> + Thumbnail 2 setMainImage("https://via.placeholder.com/600x400/ff7f7f")} + /> + Thumbnail 3 setMainImage("https://via.placeholder.com/600x400/7fff7f")} + /> + Thumbnail 4 setMainImage("https://cdn.pixabay.com/photo/2023/11/12/17/12/puppy-8383633_1280.jpg")} + /> +
+ + {/* Botón Share debajo de la carrousel */} +
+
Compartir
+ +
+
+ + {/* Pet details */} +
+

{detail?.name}

+

+ Perdido/Encontrado el {detail?.lostDate} +

+ + Comunicarse + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
NOMBRE:{detail?.name}
SEXO:{detail?.gender}
EDAD:{detail?.age}
TAMAÑO: + {detail?.size}
COLOR:{detail?.color}
SE PERDIÓ EN:{detail?.location}
+

+ INFORMACIÓN ADICIONAL: {detail?.additionalInfo} +

+
+
+ + {/* Modal para contacto */} + +
+ ); +}; + +export default PetCard; diff --git a/src/front/js/component/PetView.jsx b/src/front/js/component/PetView.jsx new file mode 100644 index 0000000000..3074bc2fc6 --- /dev/null +++ b/src/front/js/component/PetView.jsx @@ -0,0 +1,269 @@ +import React, { useState, useEffect } from "react"; +import { Link } from "react-router-dom"; + +const PetsView = () => { + // Estado para los datos y los filtros + const [pets, setPets] = useState([]); + const [filteredPets, setFilteredPets] = useState([]); + const [filters, setFilters] = useState({ + type: "", // Tipo de mascota (perro, gato, etc.) + color: "", // Color de la mascota + size: "", // Tamaño de la mascota + sex: "", // Sexo de la mascota (macho o hembra) + }); + +const BringAllPetsAPI = () => { + fetch('https://super-duper-dollop-x5rqv9jj6pv7cvx4p-3001.app.github.dev/pets') + .then((resp) => resp.json()) + .then((data) => { + setPets(data.data); + setFilteredPets(data.data); // Mostrar todo inicialmente + }) +} + + + // Simulación de datos o carga desde una API + useEffect(() => { + const petData = [ + { id: 1, name: "Max", type: "Perro", color: "Negro", sex: "Macho", size: "Grande", image: "" }, + { id: 2, name: "Bella", type: "Gato", color: "Amarillo", sex: "Hembra", size: "Pequeño", image:"" }, + { id: 3, name: "Rocky", type: "Perro", color: "marrón", size: "Mediano", image: "" }, + { id: 4, name: "Spike", type: "gato", color: "verde", sex:"macho", size: "Pequeño", image: "" }, + { id: 5, name: "Luna", type: "otro", color: "gris", size: "Grande", image: "" }, + { id: 6, name: "Chameleon", type: "reptil", color: "multicolor", size: "Pequeño", image: "https://mundoreptil.net/wp-content/uploads/primer-plano-increible-habilidad.webp" }, + { id: 7, name: "Blanquita", type: "otro", color: "Blanco y Negro", size: "Pequeño", image: "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSfkhMZAkUKlthC2a_1jMkEhoMzNakBrJcegQ&s" }, + { id: 8, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "https://vitalcan.es/wp-content/uploads/kitty-2903812_1280.jpg" }, + { id: 9, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "" }, + // Más datos... + ]; + setPets(petData); + setFilteredPets(petData); // Mostrar todo inicialmente + }, []); + + // Manejar cambios en los filtros + const handleFilterChange = (e) => { + const { name, value } = e.target; + setFilters({ + ...filters, + [name]: value, + }); + }; + + // Filtrar mascotas + useEffect(() => { + const filtered = pets.filter((pet) => { + return ( + (filters.type === "" || pet.type === filters.type) && + (filters.color === "" || pet.color === filters.color) && + (filters.size === "" || pet.size === filters.size) && + (filters.sex === "" || pet.sex === filters.sex) + ); + }); + setFilteredPets(filtered); + }, [filters, pets]); + return ( +
+
+ {/* Columna de Filtros */} +
+
+
Filtrar por:
+ {/* Filtro por sexo */} +
+ + +
+ {/* Filtro por tipo */} +
+ + +
+ + {/* Filtro por color */} +
+ +
+ + + + + + + + + + + +
+
+ {/* Filtro por tamaño */} +
+ + +
+
+
+ + {/* Columna de Tarjetas */} +
+
+ {filteredPets.map((pet) => ( +
+
+ {pet.name} +
+
{pet.name}
+

+ Tipo: {pet.type}
+ Color: {pet.color}
+ Sexo: {pet.size} +

+ Más información +
+
+
+ ))} +
+
+
+
+ ); +}; + +export default PetsView; diff --git a/src/front/js/component/ResetPassword.jsx b/src/front/js/component/ResetPassword.jsx new file mode 100644 index 0000000000..4483e345a4 --- /dev/null +++ b/src/front/js/component/ResetPassword.jsx @@ -0,0 +1,116 @@ +import React, { useState } from "react"; + +const ResetPassword = () => { + const [securityAnswer, setSecurityAnswer] = useState(""); + const [newPassword, setNewPassword] = useState(""); + const [confirmPassword, setConfirmPassword] = useState(""); + const [message, setMessage] = useState(""); + const [error, setError] = useState(""); + + const handleSubmit = async (e) => { + e.preventDefault(); + + // Validación simple del formulario + if (!securityAnswer || !newPassword || !confirmPassword) { + setError("Todos los campos son obligatorios."); + setMessage(""); + return; + } + + if (newPassword !== confirmPassword) { + setError("Las contraseñas no coinciden."); + setMessage(""); + return; + } + + try { + // Llamada a la API + const response = await fetch("https://sturdy-space-invention-q7947gjpqj76fx57r-3001.app.github.dev/user/1", { + method: "PUT", + headers: { + "Content-Type": "application/json", + }, + body: JSON.stringify({ + security_question: securityAnswer, + new_password: newPassword, + }), + }); + + if (response.ok) { + const data = await response.json(); + setMessage(data.msg || "Contraseña cambiada correctamente."); + setError(""); + } else { + const errorData = await response.json(); + setError(errorData.msg || "Ocurrió un error. Intenta nuevamente."); + setMessage(""); + } + } catch (err) { + setError("Error al conectar con el servidor."); + setMessage(""); + } + }; + + return ( +
+
+
+

¿Olvidó su contraseña?

+
+
+ + setSecurityAnswer(e.target.value)} + /> +
+ +
+ + setNewPassword(e.target.value)} + /> +
+ +
+ + setConfirmPassword(e.target.value)} + /> +
+ +
+ +
+
+ + {message &&

{message}

} + {error &&

{error}

} +
+
+
+ ); +}; + +export default ResetPassword; diff --git a/src/front/js/component/footer.js b/src/front/js/component/footer.js index 670323e091..cda777bf4a 100755 --- a/src/front/js/component/footer.js +++ b/src/front/js/component/footer.js @@ -6,5 +6,6 @@ export const Footer = () => ( Made with by{" "} 4Geeks Academy

+ ); diff --git a/src/front/js/component/navbar.js b/src/front/js/component/navbar.js index 795c8348d7..416dcc7ab0 100755 --- a/src/front/js/component/navbar.js +++ b/src/front/js/component/navbar.js @@ -1,6 +1,8 @@ import React from "react"; import { Link } from "react-router-dom"; import logotipo from "../../img/PatasperdidasPNG.png" + + export const Navbar = () => { return ( ); -}; +} diff --git a/src/front/js/component/ubication_map.js b/src/front/js/component/ubication_map.js index 935ec60f6c..11dd67ac6a 100644 --- a/src/front/js/component/ubication_map.js +++ b/src/front/js/component/ubication_map.js @@ -1,25 +1,25 @@ import React, { useState } from "react"; -import "leaflet/dist/leaflet.css"; -import { MapContainer, TileLayer, useMapEvents, Marker, Popup } from 'react-leaflet' -import { Icon, divIcon, map } from "leaflet" -import MarkerClusterGroup from 'react-leaflet-cluster'; -import L from "leaflet" +// import "leaflet/dist/leaflet.css"; +// import { MapContainer, TileLayer, useMapEvents, Marker, Popup } from 'react-leaflet' +// import { Icon, divIcon, map } from "leaflet" +// import MarkerClusterGroup from 'react-leaflet-cluster'; +// import L from "leaflet" const UbicationMap = ({ coordenates }) => { - const CustomIcon = L.divIcon({ - html: '', - className: "custom-icon", - iconSize: [24, 24], - iconAnchor: [12, 24] - }) + // const CustomIcon = L.divIcon({ + // html: '', + // className: "custom-icon", + // iconSize: [24, 24], + // iconAnchor: [12, 24] + // }) const [position, setPosition] = useState(null) function ClickHandler() { useMapEvents({ click: (e) => { const { lat, lng } = e.latlng; - const newPosition = {lat, lng} + const newPosition = { lat, lng } setPosition(newPosition); console.log(`Latitud: ${lat}, Longitud: ${lng}`); if (coordenates) { diff --git a/src/front/js/layout.js b/src/front/js/layout.js index 7774e671eb..180f6275ca 100755 --- a/src/front/js/layout.js +++ b/src/front/js/layout.js @@ -16,6 +16,9 @@ import { Navbar } from "./component/navbar"; import { Footer } from "./component/footer"; import NewPetLost from "./pages/newPetLost.js"; import NewFoundPet from "./pages/newFoundPet.js"; +import ResetPassword from "./component/ResetPassword.jsx"; +import PetView from "./component/PetView.jsx"; +import PetCard from "./component/PetCard.jsx"; import UbicationMap from "./component/ubication_map.js"; //create your first component @@ -41,6 +44,10 @@ const Layout = () => { } path="/single/:theid" /> } path="/user" /> } path="/login" /> + } path="/forgot-password" /> + } path="/petview"/> + } path="/petcard/:theid" /> + } path="/signup" /> Not found!} /> diff --git a/src/front/js/pages/newPetLost.js b/src/front/js/pages/newPetLost.js index 5995abdda5..891850a87c 100644 --- a/src/front/js/pages/newPetLost.js +++ b/src/front/js/pages/newPetLost.js @@ -3,8 +3,8 @@ import React, { useState, useContext } from "react"; import { useNavigate } from "react-router-dom"; import { Context } from "../store/appContext"; import Cloudinary from "../component/cloudinary"; -import "leaflet/dist/leaflet.css"; -import L from "leaflet" +// import "leaflet/dist/leaflet.css"; +// import L from "leaflet" import UbicationMap from "../component/ubication_map"; const NewPetLost = () => { diff --git a/src/front/js/store/flux.js b/src/front/js/store/flux.js index c8b60e1b57..52804f7a55 100755 --- a/src/front/js/store/flux.js +++ b/src/front/js/store/flux.js @@ -1,10 +1,8 @@ const getState = ({ getStore, getActions, setStore }) => { return { store: { - message: null, demo: [ - { title: "FIRST", background: "white", @@ -15,7 +13,19 @@ const getState = ({ getStore, getActions, setStore }) => { background: "white", initial: "white" } - ] + ], + petData:[ + { id: 1, name: "Max", type: "perro", color: "negro", size: "grande", image: "" }, + { id: 2, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "/img/bella.jpg" }, + { id: 3, name: "Rocky", type: "conejo", color: "marrón", size: "mediano", image: "/img/rocky.jpg" }, + { id: 4, name: "Spike", type: "reptil", color: "verde", size: "pequeño", image: "/img/spike.jpg" }, + { id: 5, name: "Luna", type: "otro", color: "gris", size: "grande", image: "/img/luna.jpg" }, + { id: 6, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "/img/bella.jpg" }, + { id: 7, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "/img/bella.jpg" }, + { id: 8, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "/img/bella.jpg" }, + { id: 9, name: "Bella", type: "gato", color: "blanco", size: "pequeño", image: "/img/bella.jpg" }, + + ], }, actions: { diff --git a/src/front/styles/home.css b/src/front/styles/home.css index 723acfb61b..41035aeaea 100755 --- a/src/front/styles/home.css +++ b/src/front/styles/home.css @@ -1,10 +1,4 @@ -/* - home.css: This website contains selectors only used in home.css - All pages share the styles on index.css but you should create - one more css for each page that will contain the selected used - on that page only (the ones not reused in other pages). -*/ .adlam-display-regular { font-family: "ADLaM Display"; font-weight: 400; @@ -26,6 +20,167 @@ body { .btnStart{ background-color: darkblue; } + +.forgot-password { + border-radius: 20px; +} + +/* Contenedor principal */ +.pet-card-container { + max-width: 80%; + margin: 30px auto; + background-color: #fff; + border-radius: 10px; + box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); + padding: 20px; +} + +/* Diseño 50/50 */ +.pet-card { + display: flex; + gap: 20px; +} + +.image-section { + flex: 1; +} + +.main-image { + width: 100%; + height: auto; + border-radius: 10px; + object-fit: cover; +} + +.info-section { + flex: 1; + padding: 10px; +} + +.info-section h3 { + font-size: 24px; + margin-bottom: 10px; +} + +.text-danger { + color: #ff4d4d; + font-weight: bold; +} + +a.btn-primary { + display: inline-block; + background-color: #007bff; + color: #fff; + text-decoration: none; + padding: 10px 15px; + border-radius: 5px; + margin: 10px 0; +} + +a.btn-primary:hover { + background-color: #0056b3; +} + +.table { + margin-top: 15px; + width: 100%; + border-collapse: collapse; +} + +.table th { + text-align: left; + font-weight: bold; + padding: 5px 0; +} + +.table td { + padding: 5px 0; +} + +p { + font-size: 14px; +} + +.additional-info { + margin-top: 15px; + font-style: italic; +} + +/* Galería */ +.gallery { + display: flex; + gap: 10px; + margin-bottom: 10px; +} + +.gallery img { + width: 70px; + height: 70px; + object-fit: cover; + border-radius: 5px; + cursor: pointer; + transition: transform 0.2s ease; +} + +.gallery img:hover { + transform: scale(1.1); +} + +/* Íconos de compartir */ +.share-icons span { + font-weight: bold; + margin-right: 10px; +} + +.share-icons a { + color: #6c757d; + margin-right: 5px; + font-size: 18px; +} + +.share-icons a:hover { + color: #007bff; +} +.filters-container { + padding: 10px; + border: 1px solid #ccc; + border-radius: 5px; + background-color: #f9f9f9; +} + +.filters-container h5 { + margin-bottom: 15px; +} + +.filter label { + font-weight: bold; +} + +.filter input[type="radio"] { + margin-right: 10px; +} +.main-image { + width: 100%; + height: auto; + margin-bottom: 15px; +} + +.gallery img { + cursor: pointer; + transition: transform 0.2s; +} + +.gallery img:hover { + transform: scale(1.1); +} +.navigation-path a { + color: #6c757d; + text-decoration: none; +} + +.navigation-path a:hover { + text-decoration: underline; +} .bg-success { --bs-bg-opacity: 0.2; background-color: rgba(var(--bs-primary-rgb), var(--bs-bg-opacity)) !important; @@ -34,3 +189,21 @@ body { color: darkblue; font-size: 30px; } + + +.btn-primary:hover { + background-color: #004a99; /* Azul oscuro más claro al pasar el cursor */ +} + +/* Color del nombre */ +.text-primary { + color: #002f6c !important; /* Azul oscuro */ +} + +/* Iconos sin recuadro */ +.btn-outline-secondary { + border: none; /* Sin recuadro */ + background: none; /* Sin fondo */ + padding: 0; +} + diff --git a/src/front/styles/pet-views.css b/src/front/styles/pet-views.css new file mode 100644 index 0000000000..b16153de77 --- /dev/null +++ b/src/front/styles/pet-views.css @@ -0,0 +1,92 @@ +/* Barra de navegación */ +.navbar { + display: flex; + justify-content: space-between; + align-items: center; + background-color: white; + padding: 10px 20px; + box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); +} + +/* Contenedor de vista de mascotas (Filtros y tarjetas) */ +.pets-view { + display: flex; + gap: 20px; + justify-content: space-between; +} + +/* Filtros */ +.filters { + width: 200px; /* Ajusta el ancho de los filtros */ + display: flex; + flex-direction: column; + gap: 20px; + position: sticky; + top: 20px; +} + +/* Categorías de filtros */ +.filter-category { + display: flex; + flex-direction: column; + gap: 10px; +} + +.filter-category h3 { + margin: 0; + font-size: 1.2em; + font-weight: bold; +} + +/* Estilo para los filtros (circulitos y cuadrados) */ +label { + display: flex; + align-items: center; + gap: 10px; +} + +/* Círculos de color */ +.color-circle { + width: 15px; + height: 15px; + border-radius: 50%; + display: inline-block; +} + +.color-circle.red { + background-color: red; +} + +.color-circle.blue { + background-color: blue; +} + +.color-circle.green { + background-color: green; +} + +/* Cuadrados de tamaño */ +.size-square { + width: 15px; + height: 15px; + display: inline-block; + border: 1px solid #ccc; + background-color: #f0f0f0; +} + +/* Contenedor principal de las tarjetas de mascotas */ +.pet-cards { + display: grid; + grid-template-columns: repeat(3, 1fr); + gap: 20px; + width: calc(100% - 220px); /* Ajusta el ancho para dar espacio a los filtros */ +} + +/* Tarjeta individual */ +.pet-card { + background-color: white; + padding: 10px; + border-radius: 5px; + box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); + text-align: center; +}