diff --git a/package-lock.json b/package-lock.json
index 9e0465f..ff09938 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1524,7 +1524,7 @@
"version": "18.2.11",
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.11.tgz",
"integrity": "sha512-zq6Dy0EiCuF9pWFW6I6k6W2LdpUixLE4P6XjXU1QHLfak3GPACQfLwEuHzY5pOYa4hzj1d0GxX/P141aFjZsyg==",
- "devOptional": true,
+ "dev": true,
"dependencies": {
"@types/react": "*"
}
diff --git a/src/App.tsx b/src/App.tsx
new file mode 100644
index 0000000..a21e1a8
--- /dev/null
+++ b/src/App.tsx
@@ -0,0 +1,33 @@
+import { BrowserRouter as Router, Routes, Route } from 'react-router-dom';
+import MainPage from './components/MainPage/MainPage';
+import Favorite from './components/Favorites/Favorite';
+import LoginPage from './components/Login/LoginPage';
+import Offer from './components/Offer/Offer';
+
+type Offer = {
+ id: number;
+ title: string;
+ price: number;
+ rating: number;
+ type: string;
+ isPremium: boolean;
+ previewImage: string;
+ NumberOfPlaces: number;
+};
+
+type AppProps = {
+ offers: Offer[];
+};
+
+export default function App({ offers }: AppProps) {
+ return (
+
+
+ } />
+ } />
+ } />
+ } />
+
+
+ );
+}
diff --git a/src/components/Favorites/Favorite.tsx b/src/components/Favorites/Favorite.tsx
index 132d12f..80c81da 100644
--- a/src/components/Favorites/Favorite.tsx
+++ b/src/components/Favorites/Favorite.tsx
@@ -1,3 +1,36 @@
+import OfferCard from '../Offer/OfferCard';
+type Offer = {
+ id: number;
+ title: string;
+ price: number;
+ rating: number;
+ type: string;
+ isPremium: boolean;
+ previewImage: string;
+ NumberOfPlaces: number;
+};
+type FavoriteProps = {
+ offers: Offer[];
+};
+const Favorite = ({ offers }: FavoriteProps) => (
+
+
+
+
+ Saved listings
+
+ {offers.map((offer) => (
+
+ ))}
+
+
+
+
+
+);
+
+export default Favorite;
+=======
export const Favorite = () => (
@@ -225,4 +258,5 @@ export const Favorite = () => (
+
);
diff --git a/src/components/Login/LoginPage.tsx b/src/components/Login/LoginPage.tsx
index f60425b..dcebe77 100644
--- a/src/components/Login/LoginPage.tsx
+++ b/src/components/Login/LoginPage.tsx
@@ -1,3 +1,67 @@
+export default function LoginPage () {
+ return (
+
+ );
+}
+=======
export const LoginPage = () => (
@@ -58,4 +122,4 @@ export const LoginPage = () => (
-);
+);
\ No newline at end of file
diff --git a/src/components/MainPage/MainPage.tsx b/src/components/MainPage/MainPage.tsx
index 47cca7b..ee9a53a 100644
--- a/src/components/MainPage/MainPage.tsx
+++ b/src/components/MainPage/MainPage.tsx
@@ -1,9 +1,21 @@
import {FC} from 'react';
-import Card from '../MainPageCard/Card.tsx';
-import {cardProperties}from '../../index.tsx';
+import OfferList from '../Offer/OfferList';
+type Offer = {
+ id: number;
+ title: string;
+ price: number;
+ rating: number;
+ type: string;
+ isPremium: boolean;
+ previewImage: string;
+ NumberOfPlaces: number;
+};
+type MainPageProps = {
+ offers: Offer[];
+};
-export const MainPage : FC<{ CardProps: cardProperties[] }> = ({ CardProps }) =>
+export const MainPage : FC = ({ offers }) =>
(
@@ -77,7 +89,7 @@ export const MainPage : FC<{ CardProps: cardProperties[] }> = ({ CardProps }) =>
Places
- {CardProps[0].NumberOfPlaces} places to stay in Amsterdam
+ {offers.length} places to stay in Amsterdam
-
-
-
-
+
@@ -108,4 +117,4 @@ export const MainPage : FC<{ CardProps: cardProperties[] }> = ({ CardProps }) =>
);
-
+export default MainPage;
diff --git a/src/components/Offer/Offer.tsx b/src/components/Offer/Offer.tsx
index f8602a6..6c9511b 100644
--- a/src/components/Offer/Offer.tsx
+++ b/src/components/Offer/Offer.tsx
@@ -1,478 +1,485 @@
-export const Offer = () => (
-
-
-
-
-
-
-
-
+import SendCommentForm from '../SendCommentForm/SendCommentForm';
+export default function Offer () {
+ return (
+
-
+
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
-
-
-
-
- Premium
-
-
-
- Beautiful & luxurious studio at great location
-
-
-
-
-
- To bookmarks
-
-
-
-
-
-
Rating
+
+
-
-
- Apartment
-
-
- 3 Bedrooms
-
-
- Max 4 adults
-
-
-
- €120
- night
-
-
-
What's inside
-
- Wi-Fi
- Washing machine
- Towels
- Heating
- Coffee machine
- Baby seat
- Kitchen
- Dishwasher
- Cabel TV
- Fridge
-
-
-
-
Meet the host
-
-
-
+
+
+ Beautiful & luxurious studio at great location
+
+
+
+
+
+ To bookmarks
+
+
+
+
+
+ Rating
-
Angelina
-
Pro
+
4.8
-
-
- A quiet cozy and picturesque that hides behind a a river by
- the unique lightness of Amsterdam. The building is green and
- from 18th century.
-
-
- An independent House, strategically located between Rembrand
- Square and National Opera, but where the bustle of the city
- comes to rest in this alley flowery and colorful.
-
+
+
+ Apartment
+
+
+ 3 Bedrooms
+
+
+ Max 4 adults
+
+
+
+ €120
+ night
-
-
-
- Reviews · 1
-
-
-
-
-
-
-
-
Max
+
+
What's inside
+
+ Wi-Fi
+ Washing machine
+ Towels
+ Heating
+ Coffee machine
+ Baby seat
+ Kitchen
+ Dishwasher
+ Cabel TV
+ Fridge
+
+
+
+
Meet the host
+
+
+
-
-
-
-
- Rating
+ Angelina
+ Pro
+
+
+
+ A quiet cozy and picturesque that hides behind a a river by
+ the unique lightness of Amsterdam. The building is green and
+ from 18th century.
+
+
+ An independent House, strategically located between Rembrand
+ Square and National Opera, but where the bustle of the city
+ comes to rest in this alley flowery and colorful.
+
+
+
+
-
-
-
-
-
-
-
-
- Other places in the neighbourhood
-
-
-
-
-
-
-
- €80
-
- / night
-
+
+
+
+
+
+
-
-
+
+
+ To submit review please make sure to set{' '}
+ rating and describe
+ your stay with at least{' '}
+ 50 characters .
+
+
-
-
- In bookmarks
-
+ Submit
+
+
+
+
+
+
+
+
+
+
+
+ Other places in the neighbourhood
+
+
+
+
-
-
-
-
Rating
+
+
+
+ €80
+
+ / night
+
+
+
+
+
+
+ In bookmarks
+
+
+
+
+
Room
-
-
Room
-
-
+
-
-
-
-
-
- €132
-
- / night
-
-
-
-
-
-
- To bookmarks
-
+
+
-
-
-
-
Rating
+
+
+
+ €132
+
+ / night
+
+
+
+
+
+
+ To bookmarks
+
+
+
+
+
Apartment
-
-
Apartment
-
-
+
-
-
- Premium
-
-
-
-
-
- €180
-
- / night
-
-
-
-
-
-
- To bookmarks
-
+
+
+ Premium
+
+
-
-
-
-
Rating
+
+
+
+ €180
+
+ / night
+
+
+
+
+
+
+ To bookmarks
+
+
+
+
+
Apartment
-
-
Apartment
-
-
-
-
-
-
-
-);
+
+
+
+
+
+
Reviews
+ {}
+
+
+
+ );
+}
diff --git a/src/components/Offer/OfferCard.tsx b/src/components/Offer/OfferCard.tsx
new file mode 100644
index 0000000..f872325
--- /dev/null
+++ b/src/components/Offer/OfferCard.tsx
@@ -0,0 +1,51 @@
+import { Link } from 'react-router-dom';
+
+type Offer = {
+ id: number;
+ title: string;
+ price: number;
+ rating: number;
+ type: string;
+ isPremium: boolean;
+ previewImage: string;
+ NumberOfPlaces: number;
+};
+
+type OfferCardProps = {
+ offer: Offer;
+};
+
+const OfferCard = ({ offer }: OfferCardProps) => (
+
+ {offer.isPremium && (
+
+ Premium
+
+ )}
+
+
+
+
+
+
+
+
+ €{offer.price}
+ / night
+
+
+
+
+ {offer.title}
+
+
{offer.type}
+
+
+);
+
+export default OfferCard;
diff --git a/src/components/Offer/OfferList.tsx b/src/components/Offer/OfferList.tsx
new file mode 100644
index 0000000..d84aa13
--- /dev/null
+++ b/src/components/Offer/OfferList.tsx
@@ -0,0 +1,38 @@
+import { useState } from 'react';
+import OfferCard from './OfferCard';
+
+type Offer = {
+ id: number;
+ title: string;
+ price: number;
+ rating: number;
+ type: string;
+ isPremium: boolean;
+ previewImage: string;
+ NumberOfPlaces: number;
+};
+
+type OfferListProps = {
+ offers: Offer[];
+};
+
+const OfferList = ({ offers }: OfferListProps) => {
+ const [activeOfferId, setActiveOfferId] = useState(null);
+
+ return (
+
+ {offers.map((offer) => (
+
setActiveOfferId(offer.id)}
+ onMouseLeave={() => setActiveOfferId(null)}
+ >
+
+
+ ))}
+
{activeOfferId &&
Active Offer ID: {activeOfferId}
}
+
+ );
+};
+
+export default OfferList;
diff --git a/src/components/SendCommentForm/SendCommentForm.tsx b/src/components/SendCommentForm/SendCommentForm.tsx
new file mode 100644
index 0000000..5c4af3b
--- /dev/null
+++ b/src/components/SendCommentForm/SendCommentForm.tsx
@@ -0,0 +1,60 @@
+import React, { useState } from 'react';
+
+const SendCommentForm = () => {
+ const [rating, setRating] = useState(0);
+ const [comment, setComment] = useState('');
+
+ const handleSubmit = (e: React.FormEvent) => {
+ e.preventDefault();
+ };
+ const numberofstars = [5, 4, 3, 2, 1];
+ return (
+
+ Your review
+
+ {numberofstars.map((star) => (
+
+ setRating(star)}
+ />
+
+
+
+
+
+
+ ))}
+
+
+ setComment(e.target.value)}
+ />
+
+
+
+ To submit review please make sure to set rating and describe your stay
+ with at least 10 characters .
+
+
+ Submit
+
+
+
+ );
+};
+
+export default SendCommentForm;
diff --git a/src/index.tsx b/src/index.tsx
index e17c5bb..15c15ff 100644
--- a/src/index.tsx
+++ b/src/index.tsx
@@ -1,7 +1,7 @@
import React from 'react';
import ReactDOM from 'react-dom/client';
-import {Main} from './components/Main/main';
-
+import App from './App';
+import { offers } from './mock/offers';
const root = ReactDOM.createRoot(
document.getElementById('root') as HTMLElement
);
@@ -64,8 +64,6 @@ export default {mainPageCardInfo} ;
root.render(
-
-
-
+
);
diff --git a/src/mock/offers.ts b/src/mock/offers.ts
new file mode 100644
index 0000000..ac5ea3e
--- /dev/null
+++ b/src/mock/offers.ts
@@ -0,0 +1,42 @@
+export const offers = [
+ {
+ id: 1,
+ title: 'Luxury Apartment in Downtown',
+ price: 250,
+ rating: 4.9,
+ type: 'Apartment',
+ isPremium: true,
+ previewImage: 'img/apartment-01.jpg',
+ NumberOfPlaces: 1,
+ },
+ {
+ id: 2,
+ title: 'Cozy House in Suburbs',
+ price: 100,
+ rating: 4.7,
+ type: 'House',
+ isPremium: false,
+ previewImage: 'img/apartment-02.jpg',
+ NumberOfPlaces: 1,
+ },
+ {
+ id: 3,
+ title: 'Stylish Studio in City Center',
+ price: 150,
+ rating: 4.8,
+ type: 'Studio',
+ isPremium: false,
+ previewImage: 'img/studio-01.jpg',
+ NumberOfPlaces: 1,
+ },
+ {
+ id: 4,
+ title: 'Modern Loft with River View',
+ price: 200,
+ rating: 5.0,
+ type: 'Loft',
+ isPremium: true,
+ previewImage: 'img/studio-01.jpg',
+ NumberOfPlaces: 1,
+ },
+];