-
Notifications
You must be signed in to change notification settings - Fork 48
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[고급 레이아웃 컴포넌트] 가브리엘(윤주현) 미션 제출합니다. #62
Changes from all commits
0b2381c
8e49320
fe96857
9afcfda
f0b32f9
0d55c95
7b75ae2
f6f6880
e9dafd1
b2f4827
109919b
cf46080
8ed89ff
8d140a4
d079e66
c8684b5
fd1aa03
f9c0d8e
30489bb
88d5b41
b6a6056
0634417
6eee7fb
61a63d8
df11769
55600f1
2821779
9fca256
c20ccc6
6e1b28c
fe4448e
6bcbf25
d0aecaf
1d706ee
f6ff274
e3429ed
9ceeb46
4295de7
f85c013
bf5f8cb
3d51cc4
1d7b81a
36a8f45
debd0ce
b9dbe63
8c8b981
a3ea011
bf49e5a
d7cf2ca
3e54c5d
f12fabf
8e5abc3
2a60d90
9eef417
8cb7c86
7d7915a
5cd49ee
91c12a4
5e77b85
902470d
cc09fae
c68852d
4a8d256
f24ada6
0d9f49a
f472124
4e37a25
ab9debe
338b838
ff44c2e
fd69069
b621af7
fb8a9f4
790ccdb
e2cd604
fa867ad
72b6979
37143e8
7b05fb0
0b97ba0
3231d5c
ffea388
154b692
63089e3
7bf23c6
60b2f74
e68197f
6f7daf0
0903fc8
7dc2aad
ab50987
d8f6f68
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
export {default as Container} from "./layouts/Container/Container"; | ||
export {default as Flex} from "./layouts/Flex/Flex"; | ||
export {default as Grid} from "./layouts/Grid/Grid"; | ||
export {default as SplitPane} from "./layouts/SplitPane/SplitPane"; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
import React, {Component, ErrorInfo, ReactNode} from "react"; | ||
|
||
interface Props { | ||
children?: ReactNode; | ||
} | ||
|
||
interface State { | ||
hasError: boolean; | ||
error: Error | null; | ||
errorInfo?: ErrorInfo | null; | ||
} | ||
|
||
class ErrorBoundary extends Component<Props, State> { | ||
public state: State = { | ||
hasError: false, | ||
error: null, | ||
}; | ||
|
||
public static getDerivedStateFromError(error: Error): State { | ||
return { | ||
hasError: true, | ||
error, | ||
}; | ||
} | ||
|
||
public componentDidCatch(error: Error, errorInfo: ErrorInfo) { | ||
console.error("Uncaught error:", error, errorInfo); | ||
this.setState({errorInfo}); | ||
} | ||
|
||
public render() { | ||
if (this.state.hasError) { | ||
return ( | ||
<div> | ||
<h1>무언가 잘못됐어요...!</h1> | ||
<p>{this.state.error?.toString()}</p> | ||
<button onClick={() => this.setState({hasError: false})}>다시 시도하기</button> | ||
</div> | ||
); | ||
} | ||
|
||
return this.props.children; | ||
} | ||
} | ||
|
||
export default ErrorBoundary; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
import type {Meta} from "@storybook/react"; | ||
import SplitPane from "./index.ts"; | ||
import {SplitPaneProps} from "./SplitPane.tsx"; | ||
|
||
const meta = { | ||
title: "Components/SplitPane", | ||
component: SplitPane, | ||
tags: ["autodocs"], | ||
args: { | ||
defaultSize: "20%", | ||
minSize: "10%", | ||
maxSize: "90%", | ||
}, | ||
argTypes: { | ||
defaultSize: { | ||
control: { | ||
type: "text", | ||
}, | ||
description: "SplitPane의 기본 사이즈를 설정합니다.", | ||
}, | ||
minSize: { | ||
control: { | ||
type: "text", | ||
}, | ||
description: "SplitPane 좌측 영역의 최소 사이즈를 설정합니다." | ||
}, | ||
maxSize: { | ||
control: { | ||
type: "text", | ||
}, | ||
description: "SplitPane 좌측 영역의 최대 사이즈를 설정합니다." | ||
} | ||
}, | ||
} satisfies Meta<typeof SplitPane>; | ||
|
||
export default meta; | ||
|
||
const loremIpsum = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. Duis vulputate nunc eu tempor tempor. Maecenas ipsum nisi, placerat sit amet arcu eu, egestas luctus orci. Etiam sed ante eu metus aliquet dapibus. Nullam eget dolor diam. Praesent et lobortis massa. Proin ornare, risus quis pretium vulputate, lacus lacus vehicula ex, eleifend vestibulum sapien augue et quam. Nulla id augue felis. Quisque vitae elit felis. Vestibulum vitae rhoncus elit. Duis imperdiet tortor vitae tellus vulputate, et sagittis est ullamcorper. Fusce dignissim ultricies tristique. Nullam congue mi ut venenatis pellentesque. Proin egestas sodales volutpat.`; | ||
|
||
export const Default = (args: SplitPaneProps) => ( | ||
<SplitPane {...args}> | ||
<div style={{backgroundColor: 'yellow', height: '100px'}}/> | ||
<div style={{backgroundColor: 'yellowgreen', height: '100px'}}/> | ||
</SplitPane> | ||
); | ||
|
||
export const Texts = (args: SplitPaneProps) => ( | ||
<SplitPane {...args}> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); | ||
|
||
export const Duplicated = (args: SplitPaneProps) => ( | ||
<SplitPane {...args}> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<SplitPane {...args}> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
</SplitPane> | ||
); | ||
|
||
export const WrongChildren = (args: SplitPaneProps) => ( | ||
<SplitPane {...args}> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); | ||
|
||
export const WrongArgs = () => ( | ||
<SplitPane defaultSize="20%" minSize="10%" maxSize="110%"> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); | ||
|
||
export const WrongArgs2 = () => ( | ||
<SplitPane defaultSize="20%" minSize="10%" maxSize="10%"> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); | ||
|
||
export const WrongArgs3 = () => ( | ||
<SplitPane defaultSize="20" minSize="10%" maxSize="90%"> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); | ||
|
||
export const WrongArgs4 = () => ( | ||
<SplitPane defaultSize="20%" minSize="1a0%" maxSize="90%"> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
<div> | ||
{loremIpsum} | ||
</div> | ||
</SplitPane> | ||
); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
import styled from "styled-components"; | ||
|
||
interface ResizablePaneProps { | ||
$size: string; | ||
} | ||
|
||
export const SplitPaneContainer = styled.div` | ||
display: flex; | ||
width: 100%; | ||
`; | ||
|
||
export const ResizablePane = styled.div<ResizablePaneProps>` | ||
overflow: auto; | ||
width: ${({$size}) => $size}; | ||
`; | ||
|
||
export const Resizer = styled.div` | ||
padding: 0px 1px; | ||
margin: 0px 2px; | ||
border-left: 1px solid #b5b5b5; | ||
border-right: 1px solid #b5b5b5; | ||
cursor: col-resize; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 오 새로운 속성 배워갑니당 |
||
`; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
import {ReactNode} from 'react'; | ||
import ErrorBoundary from "../ErrorBoundary.tsx"; | ||
import SplitPaneRenderer from "./SplitPaneRenderer.tsx"; | ||
|
||
export interface SplitPaneProps { | ||
defaultSize: string; | ||
minSize: string; | ||
maxSize: string; | ||
children: ReactNode[]; | ||
} | ||
|
||
const SplitPane = ({defaultSize, minSize, maxSize, children}: SplitPaneProps) => { | ||
|
||
return ( | ||
<ErrorBoundary> | ||
<SplitPaneRenderer defaultSize={defaultSize} minSize={minSize} maxSize={maxSize}>{children}</SplitPaneRenderer> | ||
</ErrorBoundary> | ||
); | ||
} | ||
|
||
export default SplitPane; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
import {useSplitPane} from "./useSplitPane.ts"; | ||
import {checkValidProps,} from "./validate.ts"; | ||
import {SplitPaneProps} from "./SplitPane.tsx"; | ||
import {ResizablePane, Resizer, SplitPaneContainer} from "./SplitPane.styles.ts"; | ||
|
||
function SplitPaneRenderer({defaultSize, minSize, maxSize, children}: SplitPaneProps) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. renderer로 분리하신 이유가 무엇인가요?? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 에러바운더리를 적용하려면 에러가 하위 컴포넌트에서 발생해야 감지가 됐기 때문입니다...! |
||
|
||
const { | ||
containerRef, | ||
leftPaneRef, | ||
rightPaneRef, | ||
paneSize, | ||
handleMouseDown, | ||
} = useSplitPane(defaultSize, minSize, maxSize); | ||
|
||
checkValidProps({defaultSize, minSize, maxSize, children}); | ||
|
||
return ( | ||
<SplitPaneContainer ref={containerRef}> | ||
<ResizablePane $size={paneSize} ref={leftPaneRef}> | ||
{children[0]} | ||
</ResizablePane> | ||
<Resizer onMouseDown={handleMouseDown}/> | ||
<ResizablePane $size={`calc(100% - ${paneSize} - 3px)`} ref={rightPaneRef}> | ||
{children[1]} | ||
</ResizablePane> | ||
</SplitPaneContainer> | ||
); | ||
} | ||
|
||
export default SplitPaneRenderer; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
export const ERROR_MESSAGE_CHECK_ENDS_WITH_PERCENTAGE = 'defaultSize, minSize, maxSize는 반드시 %로 끝나야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_DEFAULT_SIZE_LESS_THAN_MIN_SIZE = 'defaultSize는 minSize보다 작을 수 없습니다.'; | ||
export const ERROR_MESSAGE_CHECK_DEFAULT_SIZE_GREATER_THAN_MAX_SIZE = 'defaultSize는 maxSize보다 클 수 없습니다.'; | ||
export const ERROR_MESSAGE_CHECK_CHILDREN_LENGTH = 'SplitPane 컴포넌트의 children은 반드시 2개여야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_DEFAULT_SIZE = 'defaultSize는 0% ~ 100% 사이의 값이어야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_MIN_SIZE = 'minSize는 0% ~ 100% 사이의 값이어야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_MAX_SIZE = 'maxSize는 0% ~ 100% 사이의 값이어야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_DEFAULT_NUMBER_PERCENTAGE = 'defaultSize는 숫자% 형태여야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_MIN_SIZE_NUMBER_PERCENTAGE = 'minSize는 숫자% 형태여야 합니다.'; | ||
export const ERROR_MESSAGE_CHECK_MAX_SIZE_NUMBER_PERCENTAGE = 'maxSize는 숫자% 형태여야 합니다.'; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
import SplitPane from "./SplitPane.tsx"; | ||
|
||
export default SplitPane; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
오 아예 에러 바운더리로 처리하셨군요 bb