From d4f0e98faf4903f074b74b8e14576d654bfe3bee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=EA=B9=80=EB=B3=91=ED=98=84?= Date: Sun, 17 Sep 2023 22:30:45 +0900 Subject: [PATCH] =?UTF-8?q?[Design]=EC=99=BC=EC=AA=BD=20=EC=B0=BD=20?= =?UTF-8?q?=EB=94=94=EC=9E=90=EC=9D=B8=20=EC=88=98=EC=A0=95=20=EC=A0=84?= =?UTF-8?q?=EC=B2=B4=EC=A2=85=EB=AA=A9,=20=EA=B4=80=EC=8B=AC=EC=A2=85?= =?UTF-8?q?=EB=AA=A9=EC=97=90=EC=84=9C=20=EA=B8=80=EC=94=A8=20=ED=81=AC?= =?UTF-8?q?=EA=B8=B0=20=EC=88=98=EC=A0=95,=20=EA=B8=80=EC=94=A8=20?= =?UTF-8?q?=EC=83=89=20=EC=88=98=EC=A0=95=20=ED=98=B8=EB=B2=84=20=EC=8B=9C?= =?UTF-8?q?=20=ED=95=98=EB=82=98=EC=9D=98=20=EC=95=84=EC=9D=B4=ED=85=9C?= =?UTF-8?q?=EC=97=90=EC=84=9C=20=EB=B3=80=EB=8F=99=EB=A5=A0/=EB=B3=80?= =?UTF-8?q?=EB=8F=99=EA=B0=80=EA=B2=A9=20=EB=B3=80=ED=99=94=EA=B0=80=20?= =?UTF-8?q?=EC=9D=BC=EC=96=B4=EB=82=98=EB=8F=84=EB=A1=9D=20=EC=88=98?= =?UTF-8?q?=EC=A0=95=20=EA=B0=80=EB=A1=9C=EC=84=A0=20=EC=9C=84=EC=B9=98=20?= =?UTF-8?q?=EC=88=98=EC=A0=95=20=EB=A9=94=EB=89=B4=20=EB=B2=84=ED=8A=BC=20?= =?UTF-8?q?=EB=94=94=EC=9E=90=EC=9D=B8=EC=9D=84=20=EB=B0=9D=EA=B3=A0=20?= =?UTF-8?q?=ED=81=AC=EA=B2=8C=20=EC=88=98=EC=A0=95=20Issues=20#19?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../src/components/EntireList/EntireList.tsx | 43 ++++++----- client/src/components/EntireList/Header.tsx | 32 +++++--- .../src/components/EntireList/StockItem.tsx | 34 ++++++--- client/src/components/HoldingList/Header.tsx | 31 +++++--- .../components/HoldingList/HoldingList.tsx | 62 ++++++++------- .../src/components/HoldingList/StockItem.tsx | 17 +++-- client/src/components/watchlist/Header.tsx | 34 ++++++--- client/src/components/watchlist/StockItem.tsx | 33 ++++---- .../watchlist/StockSearchComponent.tsx | 3 +- client/src/components/watchlist/WatchList.tsx | 75 +++++++++---------- 10 files changed, 212 insertions(+), 152 deletions(-) diff --git a/client/src/components/EntireList/EntireList.tsx b/client/src/components/EntireList/EntireList.tsx index 9279ffa2..9e8ac5c7 100644 --- a/client/src/components/EntireList/EntireList.tsx +++ b/client/src/components/EntireList/EntireList.tsx @@ -28,10 +28,14 @@ const EntireList: React.FC = ({ currentListType, onChangeListTy return ( -
- - {isLogin == 0 ? ("로그인이 필요한 서비스 입니다.") : (현금 보유량: {holdingsAmount}원)} - + +
+ + + + {isLogin == 0 ? (로그인이 필요한 서비스 입니다.) : (현금 보유량: {holdingsAmount}원)} + + {isLoading ?
Loading...
: isError ?
Error fetching data
: companiesList.map((company) => )}
@@ -55,11 +59,21 @@ const WatchListContainer = styled.div` align-items: flex-start; `; -const Divider1 = styled.div` - margin: 0px; - padding: 0px; +const Header1Container =styled.div` + height: 48px; + display: flex; + +`; + +const Header2Container =styled.div` + height: 43.5px; + display: flex; + justify-content:center; + align-items: center; +`; + +const Divider = styled.div` width: 100%; - height: 10px; display: flex; flex-direction: row; border-bottom: 1px solid #2f4f4f; @@ -68,21 +82,10 @@ const Divider1 = styled.div` const HoldingsAmount = styled.div` font-size: 16px; font-weight: bold; - margin: 8px 12px; - text-align: center; + margin-left: 20px; color: darkslategray; // 현금 보유량을 파란색으로 표시 `; -const Divider2 = styled.div` - margin: 0px; - padding: 0px; - width: 100%; - height: 4.5px; - display: flex; - flex-direction: row; - border-bottom: 1px solid #2f4f4f; -`; - const StockList = styled.div` height: 100%; width: 100%; diff --git a/client/src/components/EntireList/Header.tsx b/client/src/components/EntireList/Header.tsx index 50ea82a7..20c02df3 100644 --- a/client/src/components/EntireList/Header.tsx +++ b/client/src/components/EntireList/Header.tsx @@ -17,7 +17,7 @@ const Header: React.FC = ({ currentListType, onChangeListType, isMe {currentListType} {isMenuOpen && ( - { onChangeListType(ALL_LIST); setMenuOpen(false); }}>{ALL_LIST} + { onChangeListType(ALL_LIST); setMenuOpen(false); }}>{ALL_LIST} { onChangeListType(INTEREST_LIST); setMenuOpen(false); }}>{INTEREST_LIST} { onChangeListType(HOLDING_LIST); setMenuOpen(false); }}>{HOLDING_LIST} @@ -41,16 +41,18 @@ const HeaderWrapper = styled.div` `; const Icon = styled.img` - margin-top: 9.5px; - margin-left: 10px; + margin: 0px 10px 0px 10px; width: 24px; height: 24px; cursor: pointer; - margin-right: 10px; + + &:hover { + background-color: #f2f2f2; + } `; const HeaderText = styled.span` - margin-top: 9.5px; + margin-top: 2px; font-size: 18px; `; @@ -58,19 +60,31 @@ const SlideMenu = styled.div` position: absolute; top: 100%; left: 0; - width: 248px; + width: 247px; background-color: #f7f7f7; - border: 1px solid #e0e0e0; /* 밑에 가로줄 추가 */ + border-top: 1px solid black; + border-right: 1px solid black; + border-left: 1px solid black; display: flex; flex-direction: column; `; const MenuItem = styled.button` - padding: 8px 16px; + height: 40px; + padding: 8px 44px; border: none; - background-color: transparent; + background-color: white; cursor: pointer; text-align: left; + border-bottom:1px solid black; + font-size: 17px; + + &:hover { + background-color: #f2f2f2; + } +`; +const MenuItem1 = styled(MenuItem)` + border-left: 4px solid darkslategray; `; export default Header; diff --git a/client/src/components/EntireList/StockItem.tsx b/client/src/components/EntireList/StockItem.tsx index f06bb014..5462a851 100644 --- a/client/src/components/EntireList/StockItem.tsx +++ b/client/src/components/EntireList/StockItem.tsx @@ -1,26 +1,31 @@ -import React from 'react'; +import React, { useState } from 'react'; import styled from 'styled-components'; import logo from '../../asset/images/StockHolmImage.png'; -const StockItem: React.FC = ({ company, setShowChangePrice, showChangePrice }) => { +const StockItem: React.FC = ({ company }) => { const isPositiveChange = parseFloat(company.stockChangeRate) > 0; - const priceColor = isPositiveChange ? 'red' : 'blue'; + const priceColor1 = isPositiveChange ? "#ed2926" : "#2679ed"; + const priceColor2 = isPositiveChange ? "#f87369" : "#5a99f8"; + + + const [showChangePrice, setShowChangePrice] = useState(false); // 상태를 여기로 이동 return ( - + setShowChangePrice(true)} // StockItemWrapper에 이벤트 리스너 적용 + onMouseLeave={() => setShowChangePrice(false)} + > {company.korName} {company.code} - {company.stockPrice} + {company.stockPrice} setShowChangePrice(true)} - onMouseLeave={() => setShowChangePrice(false)} + change={priceColor2} > - {showChangePrice ? `${company.stockChangeAmount}%` : `${company.stockChangeRate}%`} + {showChangePrice ? `${company.stockChangeAmount}` : `${company.stockChangeRate}%`} @@ -58,7 +63,7 @@ const Logo = styled.img` border-radius: 50%; width: 40px; height: 40px; - margin-right: 12px; + margin-right: 10px; `; const StockInfo = styled.div` @@ -69,18 +74,21 @@ const StockInfo = styled.div` `; const StockName = styled.span` - font-weight: bold; + font-weight: 500; `; const StockCode = styled.span` color: gray; + font-weight: 400; + font-size: 14px; `; const StockPriceSection = styled.div` display: flex; flex-direction: column; - align-items: flex-start; + align-items: flex-end; margin-left: auto; /* 자동으로 왼쪽 여백 추가 */ + margin-right:10px; `; const StockPrice = styled.span<{ change: string }>` @@ -90,6 +98,8 @@ const StockPrice = styled.span<{ change: string }>` const StockChange = styled.span<{ change: string }>` color: ${(props) => props.change}; cursor: pointer; + font-size:14px; + `; export default StockItem; diff --git a/client/src/components/HoldingList/Header.tsx b/client/src/components/HoldingList/Header.tsx index 664d63b9..5dab7850 100644 --- a/client/src/components/HoldingList/Header.tsx +++ b/client/src/components/HoldingList/Header.tsx @@ -19,13 +19,16 @@ const Header: React.FC = ({ currentListType, onChangeListType, isMe { onChangeListType(ALL_LIST); setMenuOpen(false); }}>{ALL_LIST} { onChangeListType(INTEREST_LIST); setMenuOpen(false); }}>{INTEREST_LIST} - { onChangeListType(HOLDING_LIST); setMenuOpen(false); }}>{HOLDING_LIST} + { onChangeListType(HOLDING_LIST); setMenuOpen(false); }}>{HOLDING_LIST} )} ); }; +export default Header; + + type HeaderProps = { currentListType: string; onChangeListType: (type: "전체종목" | "관심종목" | "보유종목") => void; @@ -41,16 +44,15 @@ const HeaderWrapper = styled.div` `; const Icon = styled.img` - margin-top: 9.5px; width: 24px; height: 24px; cursor: pointer; margin-right: 10px; - margin-left:10px; + margin-left: 10px; `; const HeaderText = styled.span` - margin-top: 9.5px; + margin-top: 2px; font-size: 18px; `; @@ -58,19 +60,30 @@ const SlideMenu = styled.div` position: absolute; top: 100%; left: 0; - width: 248px; + width: 247px; background-color: #f7f7f7; - border: 1px solid #e0e0e0; /* 밑에 가로줄 추가 */ + border-top: 1px solid black; + border-right: 1px solid black; + border-left: 1px solid black; display: flex; flex-direction: column; `; const MenuItem = styled.button` - padding: 8px 16px; + height: 40px; + padding: 8px 44px; border: none; - background-color: transparent; + background-color: white; cursor: pointer; text-align: left; + border-bottom:1px solid black; + font-size: 17px; + + &:hover { + background-color: #f2f2f2; + } +`; +const MenuItem1 = styled(MenuItem)` + border-left: 4px solid darkslategray; `; -export default Header; diff --git a/client/src/components/HoldingList/HoldingList.tsx b/client/src/components/HoldingList/HoldingList.tsx index f0610297..c422e67a 100644 --- a/client/src/components/HoldingList/HoldingList.tsx +++ b/client/src/components/HoldingList/HoldingList.tsx @@ -22,15 +22,21 @@ const HoldingList: React.FC = ({ currentListType, onChangeListTy return ( -
- - 평가 수익금: {totalEvaluationProfit.toLocaleString()}원 - + +
+ + + + + 평가 수익금: {totalEvaluationProfit.toLocaleString()}원 + + + {isLoading || isCompanyDataLoading ? (
Loading...
@@ -71,33 +77,35 @@ const WatchListContainer = styled.div` align-items: flex-start; `; -const Divider1 = styled.div` - margin: 0px; - padding: 0px; - width: 100%; - height: 10px; +const Header1Container =styled.div` + height: 48px; display: flex; - flex-direction: row; - border-bottom: 1px solid #2f4f4f; `; -const EvaluationProfit = styled.div` - font-size: 16px; - font-weight: bold; - margin: 8px 12px; - text-align: center; - color: red; + +const Header2Container =styled.div` + height: 43.5px; + display: flex; + justify-content:center; + align-items: center; `; -const Divider2 = styled.div` - margin: 0px; - padding: 0px; +const Divider = styled.div` width: 100%; - height: 4.5px; display: flex; flex-direction: row; border-bottom: 1px solid #2f4f4f; `; - +const EvaluationProfit = styled.div<{ profit: number }>` + font-size: 16px; + font-weight: bold; + margin-left: 20px; + text-align: center; + color: ${props => + props.profit === 0 ? '#000' : + props.profit > 0 ? '#e22926' : + '#2679ed' + }; +`; const StockList = styled.div` height: 100%; diff --git a/client/src/components/HoldingList/StockItem.tsx b/client/src/components/HoldingList/StockItem.tsx index 6555fe17..6f69733e 100644 --- a/client/src/components/HoldingList/StockItem.tsx +++ b/client/src/components/HoldingList/StockItem.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React,{ useState } from 'react'; import styled from 'styled-components'; import logo from '../../asset/images/StockHolmImage.png'; @@ -28,7 +28,8 @@ export type StockItemProps = { -const StockItem: React.FC = ({ companyData, stockData, setShowChangePrice, showChangePrice }) => { +const StockItem: React.FC = ({ companyData, stockData }) => { + const [showChangePrice, setShowChangePrice] = useState(false); // local state const { stockCount, reserveSellStockCount, totalPrice, percentage, stockReturn } = stockData; const totalStocksHeld = stockCount + reserveSellStockCount; const company = companyData ? companyData : undefined; @@ -41,13 +42,15 @@ const StockItem: React.FC = ({ companyData, stockData, setShowCh stockChangeRate = '' } = company || {}; - // Format percentage to two decimal places - const formattedPercentage = parseFloat(percentage.toFixed(2)); + // Format percentage to two decimal places + const formattedPercentage = parseFloat(percentage.toFixed(2)); return ( <> - + setShowChangePrice(true)} // Mouse event handlers + onMouseLeave={() => setShowChangePrice(false)}> + {korName} @@ -165,8 +168,8 @@ const DetailData = styled.span` `; const getColorByValue = (value: string) => { - if (value.startsWith('')) return 'red'; - if (value.startsWith('-')) return 'blue'; + if (value.startsWith('')) return '#ed2926'; + if (value.startsWith('-')) return '#2679ed'; return 'black'; }; diff --git a/client/src/components/watchlist/Header.tsx b/client/src/components/watchlist/Header.tsx index 664d63b9..424449fe 100644 --- a/client/src/components/watchlist/Header.tsx +++ b/client/src/components/watchlist/Header.tsx @@ -18,7 +18,7 @@ const Header: React.FC = ({ currentListType, onChangeListType, isMe {isMenuOpen && ( { onChangeListType(ALL_LIST); setMenuOpen(false); }}>{ALL_LIST} - { onChangeListType(INTEREST_LIST); setMenuOpen(false); }}>{INTEREST_LIST} + { onChangeListType(INTEREST_LIST); setMenuOpen(false); }}>{INTEREST_LIST} { onChangeListType(HOLDING_LIST); setMenuOpen(false); }}>{HOLDING_LIST} )} @@ -26,6 +26,8 @@ const Header: React.FC = ({ currentListType, onChangeListType, isMe ); }; +export default Header; + type HeaderProps = { currentListType: string; onChangeListType: (type: "전체종목" | "관심종목" | "보유종목") => void; @@ -33,7 +35,6 @@ type HeaderProps = { setMenuOpen: React.Dispatch>; }; - const HeaderWrapper = styled.div` display: flex; align-items: center; @@ -41,36 +42,45 @@ const HeaderWrapper = styled.div` `; const Icon = styled.img` - margin-top: 9.5px; + margin: 0px 10px 0px 10px; width: 24px; height: 24px; cursor: pointer; - margin-right: 10px; - margin-left:10px; `; const HeaderText = styled.span` - margin-top: 9.5px; + margin-top: 2px; font-size: 18px; `; - const SlideMenu = styled.div` position: absolute; top: 100%; left: 0; - width: 248px; + width: 247px; background-color: #f7f7f7; - border: 1px solid #e0e0e0; /* 밑에 가로줄 추가 */ + border-top: 1px solid black; + border-right: 1px solid black; + border-left: 1px solid black; display: flex; flex-direction: column; `; const MenuItem = styled.button` - padding: 8px 16px; + height: 40px; + padding: 8px 44px; border: none; - background-color: transparent; + background-color: white; cursor: pointer; text-align: left; + border-bottom:1px solid black; + font-size: 17px; + + &:hover { + background-color: #f2f2f2; + } +`; +const MenuItem1 = styled(MenuItem)` + border-left: 4px solid darkslategray; `; -export default Header; + diff --git a/client/src/components/watchlist/StockItem.tsx b/client/src/components/watchlist/StockItem.tsx index f06bb014..a6fda6ac 100644 --- a/client/src/components/watchlist/StockItem.tsx +++ b/client/src/components/watchlist/StockItem.tsx @@ -1,26 +1,27 @@ -import React from 'react'; +import React,{ useState } from 'react'; import styled from 'styled-components'; import logo from '../../asset/images/StockHolmImage.png'; -const StockItem: React.FC = ({ company, setShowChangePrice, showChangePrice }) => { +const StockItem: React.FC = ({ company }) => { + const [showChangePrice, setShowChangePrice] = useState(false); const isPositiveChange = parseFloat(company.stockChangeRate) > 0; - const priceColor = isPositiveChange ? 'red' : 'blue'; + const priceColor1 = isPositiveChange ? "#ed2926" : "#2679ed"; + const priceColor2 = isPositiveChange ? "#f87369" : "#5a99f8"; return ( - + setShowChangePrice(true)} + onMouseLeave={() => setShowChangePrice(false)} + > {company.korName} {company.code} - {company.stockPrice} - setShowChangePrice(true)} - onMouseLeave={() => setShowChangePrice(false)} - > - {showChangePrice ? `${company.stockChangeAmount}%` : `${company.stockChangeRate}%`} + {company.stockPrice} + + {showChangePrice ? `${company.stockChangeAmount}` : `${company.stockChangeRate}%`} @@ -38,8 +39,6 @@ type NewCompanyData = { type StockItemProps = { company: NewCompanyData; - setShowChangePrice: React.Dispatch>; - showChangePrice: boolean; }; const StockItemWrapper = styled.div` @@ -69,18 +68,21 @@ const StockInfo = styled.div` `; const StockName = styled.span` - font-weight: bold; + font-weight: 500; `; const StockCode = styled.span` color: gray; + font-weight: 400; + font-size: 14px; `; const StockPriceSection = styled.div` display: flex; flex-direction: column; - align-items: flex-start; + align-items: flex-end; margin-left: auto; /* 자동으로 왼쪽 여백 추가 */ + margin-right: 10px; `; const StockPrice = styled.span<{ change: string }>` @@ -90,6 +92,7 @@ const StockPrice = styled.span<{ change: string }>` const StockChange = styled.span<{ change: string }>` color: ${(props) => props.change}; cursor: pointer; + font-size:14px; `; export default StockItem; diff --git a/client/src/components/watchlist/StockSearchComponent.tsx b/client/src/components/watchlist/StockSearchComponent.tsx index 114cdd3d..c202b65d 100644 --- a/client/src/components/watchlist/StockSearchComponent.tsx +++ b/client/src/components/watchlist/StockSearchComponent.tsx @@ -79,8 +79,9 @@ const StyledSearchInput = styled.input.attrs({ width: 100%; padding: 0.5rem; border: 1px solid #ccc; - border-radius: 4px; + border-radius: 5px; flex: 1; + margin-left:8px; `; const StyledSearchButton = styled.button` diff --git a/client/src/components/watchlist/WatchList.tsx b/client/src/components/watchlist/WatchList.tsx index a0cbeaae..dca41a43 100644 --- a/client/src/components/watchlist/WatchList.tsx +++ b/client/src/components/watchlist/WatchList.tsx @@ -9,7 +9,6 @@ import { RootState } from '../../store/config.ts'; // Redux store의 RootState const WatchList: React.FC = ({ currentListType, onChangeListType }) => { const [isMenuOpen, setMenuOpen] = useState(false); - const [showChangePrice, setShowChangePrice] = useState(false); const loginStatus = useSelector((state: RootState) => state.login); @@ -42,35 +41,34 @@ const WatchList: React.FC = ({ currentListType, onChangeListType return ( -
- - - - - {isLoading ? ( -
Loading...
- ) : isError ? ( -
Error fetching data
- ) : loginStatus === 1 ? ( - companiesList - .filter(company => searchedCompanyIds.includes(company.companyId)) - .map((company) => ( - +
- )) - ) : ( -
로그인이 필요합니다.
- )} - + + + + + + + + {isLoading ? ( +
Loading...
+ ) : isError ? ( +
Error fetching data
+ ) : loginStatus === 1 ? ( + companiesList + .filter(company => searchedCompanyIds.includes(company.companyId)) + .map((company) => ( + + )) + ) : ( +
로그인이 필요합니다.
+ ) } +
); @@ -90,21 +88,18 @@ const WatchListContainer = styled.div` align-items: flex-start; `; -const Divider1 = styled.div` - margin:0px; - padding:0px; - width: 100%; - height: 10px; +const Header1Container =styled.div` + height: 48px; + display: flex; +`; + +const Header2Container =styled.div` + height: 43.5px; display: flex; - flex-direction: row; - border-bottom: 1px solid #2f4f4f; `; -const Divider2 = styled.div` - margin:0px; - padding:0px; +const Divider = styled.div` width: 100%; - height: 4.5px; display: flex; flex-direction: row; border-bottom: 1px solid #2f4f4f;