이런거 markup language로 만드거임
파일명이랑 프로젝트명 다르면 package.json에 name 변경해줘야한다
JSX -> 함수형 클래스형 두 가지로 나뉘는데 우리는 함수형
으로 진행할 예정
index.html root - > potato로 바꾸면 아무것도 뜨지 않는다
이렇게 오류가 난다.
potato.jsx생성
화면 구성 차이점 앞이 SPRING 뒤가 리엑트
자바스크립트를 사용하면서 이용한다.
function Movie(){
return(
<h2>나는 감자가 조아</h2>
);
}
function App() {
// 이곳은 일반적인 js 소스코드 부분
//return 화면을 생성하는 가상 DOM 부분 (virtual dom)
return (
//부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
//jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
//자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
//css선택자인 class를 className으로 사용한다
<div className="App">
{/* SPA: Single Page Application 의 줄임말, (spring은 multipage application)
index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
{/* SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
{/* 컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}
<h2>리액트 사용하기</h2>
<Movie/>
<Movie/>
</div>
function 이름 바뀌어도 잘 작동된다
리엑트에서 dom개념 설명해 놓은 블로그
https://velog.io/@mollog/React%EC%97%90%EC%84%9C%EC%9D%98-%EA%B0%80%EC%83%81%EB%8F%94-%EA%B0%9C%EB%85%90
<Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>
이렇게 콘솔창에 뜬다
import './App.css';
// import Potatao from "./Potatao";
// 반드시 경로명을 적어야 한다
// 이렇게 한 파일에서 function을 만들어 넣어도 된다
// props : 부모 컴포넌트에서 자식 컴포넌트로 데이터를 전달하는 매개변수
// props는 읽기 전용
function Food(props){
const some = props.something;
console.log(props);
return(
<h2>I love {props.fav}</h2>
);
}
function App() {
// 이곳은 일반적인 js 소스코드 부분
//return 화면을 생성하는 가상 DOM 부분 (virtual dom)
return (
//부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
//jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
//자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
//css선택자인 class를 className으로 사용한다
<div className="App">
{/* SPA: Single Page Application 의 줄임말, (spring은 multipage application)
index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
{/* SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
{/* 컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}
{/* {Food라는 컴포넌트에 변수명이 kimchi인 데이터를 전달한다}*/}
{/*{}를 사용해야만 데이터가 전달이 된다 그냥 다 감싸 */}
<Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>
<Food fav={"돈까스"}/>
<Food fav={"햄버거"}/>
<Food fav={"스시"}/>
<h2>리액트 사용하기</h2>
{/* 함수안에 들어있는 정보를 화면에 뿌려준다*/}
</div>
// 오로지 div 안에서만 구성해야 한다
);
}
// return 화면 렌더링 - 무조건 한 개의 html태그만 사용한다
// 주로 <div>로 부모태그만들고 안에 화면을 그린다. return 뒤에 있는 ()는 구분하는 용도. 어디서부터 어디까지 return 되는지 헷갈리니까
// 안 붙여도 괜찮지만 붙이는 것이 더 바람직하다. (한 줄 이상 될 경우)
//export에 default가 붙어있다면 다른 곳에서 import할 때 {}를 붙여야하는데 안 붙이고 바로 호출하는 것이 가능하다
export default App;
이런 형태로 props사용해도 된다
컴포넌트와 props는 꼭 기억해야한다!
import './App.css';
// import Potatao from "./Potatao";
// 반드시 경로명을 적어야 한다
// 이렇게 한 파일에서 function을 만들어 넣어도 된다
// props : 부모 컴포넌트에서 자식 컴포넌트로 데이터를 전달하는 매개변수
// props는 읽기 전용
// 사용법 :
// 1) 컴포넌트 명(props){
// {props.부모가 전달한 키 이름}
// }
// 2) 컴포넌트 명({부모가 전달한 키 이름, ...}) {
// {키 이름1}, {키 이름2}, ...
// console.log({키이름1});
// }
// 3) 컴포넌트명(props){
// const {키이름, ...} = props;
// console.log({키이름});
// }
// 1번
// function Food(props){
// const some = props.something;
// console.log(props);
// return(
// <h2>I love {props.fav}</h2>
// );
// }
// 2번
function Food({name, pic}){
// const some = name.something;
// console.log(name);
return(
<div>
<h2>I love {name}</h2>
<img src={pic}/>
</div>
);
}
// 3번
// function Food(props){
// const some = props.something;
// console.log(props);
// return(
// <h2>I love {props.fav}</h2>
// );
// }
// 새로운 예제
const foodList = [
{
name: "돈까스",
image : "https://t1.daumcdn.net/thumb/R720x0/?fname=http://t1.daumcdn.net/brunch/" +
"service/user/3YN/image/XIsief9ks0Re5EL1N87ztiBfaok.jpg"
},
{
name :"순대국밥",
image: "" +
"1NbmozXDGRW4ArRNbilONorEVkV6uOoxFYismtaNgo9HWsRXq9RCYmsE141rXAM1msVmgAyK9Na140" +
"TjYVhS+VeO1aCuONhWaxWze9EB//2Q=="
},
{
name : "햄버거",
image: "" +
"s6DTKjBdBHqar6mcwvYvBp/ucx/4mZvoTH0pxguG2rX+ztW0/yIq/kKt15TqEVhFcqknlnte0UUwoUUUUAFFFFABRRRQAUUUUAf/Z"
}
];
function App() {
// 이곳은 일반적인 js 소스코드 부분
// 원래 이렇게 넣어야 했다
// foodList.map(function(dish:{image,name}){
// <Food name={dish.name}/>
// });
//return 화면을 생성하는 가상 DOM 부분 (virtual dom)
return (
//부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
//jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
//자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
//css선택자인 class를 className으로 사용한다
<div className="App">
{/* SPA: Single Page Application 의 줄임말, (spring은 multipage application)
index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
{/* SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
{/* 컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}
{/*{} : jsx 문법에서 {}데이터를 표현하는 데 사용함, 간단한 연산식도 사용가능*/}
{/* {Food라는 컴포넌트에 변수명이 kimchi인 데이터를 전달한다}*/}
{/*{}를 사용해야만 데이터가 전달이 된다 그냥 다 감싸 */}
{foodList.map(dish => <Food name = {dish.name} pic={dish.image}/>)}
{/*<Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>*/}
{/*<Food fav={"돈까스"}/>*/}
{/*<Food fav={"햄버거"}/>*/}
{/*<Food fav={"스시"}/>*/}
<h2>리액트 사용하기</h2>
{/* 함수안에 들어있는 정보를 화면에 뿌려준다*/}
</div>
// 오로지 div 안에서만 구성해야 한다
);
}
// return 화면 렌더링 - 무조건 한 개의 html태그만 사용한다
// 주로 <div>로 부모태그만들고 안에 화면을 그린다. return 뒤에 있는 ()는 구분하는 용도. 어디서부터 어디까지 return 되는지 헷갈리니까
// 안 붙여도 괜찮지만 붙이는 것이 더 바람직하다. (한 줄 이상 될 경우)
//export에 default가 붙어있다면 다른 곳에서 import할 때 {}를 붙여야하는데 안 붙이고 바로 호출하는 것이 가능하다
export default App;
unique key값이 없기 때문에 오류
prop-type 설치해야 한다. 터미널에다 쳐야함
npm install prop-types
삿갓 표시는 이 숫자 이상이면 잘 작동된다는 의미다
Food.propTypes = {
name : PropTypes.string.isRequired, //값 안들어오면 오류체크하라는 의미
pic : PropTypes.string.isRequired,
rating : PropTypes.string.isRequired
};
데이터형이 맞지 않다고 뜸
Food.propTypes = {
name : PropTypes.string.isRequired, //값 안들어오면 오류체크하라는 의미
img : PropTypes.string.isRequired,
rating : PropTypes.number.isRequired
};
변수 이름이 달라지면 틀렸다고 알려줌
원래 pic인데 img로 바꾸어보았다
ppt 5장
App2.jsx생성
클래스로 import하는 방법
constructor(props) {
super(props);
console.log('생성자 실행');
}
componentDidMount() {
console.log('컴포넌트 생성 완료')
}
componentDidUpdate(prevProps, prevState, snapshot) {
console.log("컴포넌트 업데이트 완료")
}
componentWillUnmount() {
console.log('컴포넌트 제거 완료')
}
+버튼 누르면
마이너스 버튼 누르면
App2.jsx 최종
import React from "react";
// 클래스 타입의 컴포넌트 (요즘 많이 사용하지는 않는 편)
// 클래스 컴포넌트는 React.Component를 상속받아 사용함
// 함수형이 더 간편하다
class App2 extends React.Component {
// state : 현재 컴포넌트에서 사용하는 상태 값
// props는 부모 컴포넌트에서 전달되는 값이지만 state는 현재 컴포넌트에서만 사용되는 값으로 수정 가능함
// props는 수정 불가능.
// state의 값이 변경되면 리엑트는 화면을 다시 렌더링을 함
// state의 값을 직접적으로 변경하는 것은 불가능, setState()함수를 사용하여 값을 수정
state = {
count : 0,
};
plus = () => {
this.setState({count:this.state.count + 1}) //이렇게 해야 값이 변동이 된다.
console.log('plus');
}
minus = () => {
console.log('minus')
this.setState({count: this.state.count-1})
}
// render() : 클래스 컴포넌트에서 화면을 렌더링하기 위한 메서드
// render은 함수 컴포넌트와 사용 방법이 동일하다
render() {
return(
<div>
<h1>클래스 컴포넌트 App2</h1>
<h3>카운트 수 : {this.state.count}</h3>
{/* 버튼 추가*/}
<button onClick={this.plus}>plus</button>
<button onClick = {this.minus}>minus</button>
</div>
);
}
constructor(props) {
super(props);
console.log('생성자 실행');
}
//componentDidMount,componentDidUpdate,componentWillUnmount는 리액트의 생명주기에 관련되어 있는 이벤트 함수
// componentDidMount() : 해당 컴포넌트가 DOM에 추가된 후 동작하는 이벤트
// componentDidUpdate() : 해당 컴포넌트의 데이터가 변경된 후 동작하는 이벤트
// componentWillUnmount() : 해당 컴포넌트가 더이상 사용되지 않아 삭제된 후 동작하는 이벤트
//클래스 컴포넌트에서만 사용할 수 있음
// 함수 컴포넌트에서는 해당 이벤트 함수들을 사용할 수 없음
// 함수 컴포넌트에서 해당 이벤트 함수들을 사용하기 위해서 HOOKS라는 기능을 추가하여 해당 이벤트를 구현함
componentDidMount() {
console.log('컴포넌트 생성 완료')
}
componentDidUpdate(prevProps, prevState, snapshot) {
console.log("컴포넌트 업데이트 완료")
}
// componentWillUnmount() {
// console.log('컴포넌트 제거 완료') //삭제할 때 사용됨
// }
}
export default App2;
import React from "react";
const styles={
wrapper : {
margin:8,
padding:8,
display: 'flex',
flexDirection : 'row', //밑으로 쌓이게끔
boarder :'1px solid gray',
borderRadius : 16,
},
messageText : {
color :'black',
fontSize:16,
},
};
class Notification extends React.Component {
constructor(props) {
super(props);
this.state={};
}
render() {
return(
<div style={styles.wrapper}>
<span style = {styles.messageText}>{this.props.message}</span>
</div>
);
}
}
export default Notification;
//NotificationList.jsx
import React from "react";
import Notification from "./Notification";
import notification from "./Notification";
//js 배열은 다른 언어의 list와 같다.
// 댓글 데이터 생성
const reservedNotifications = [
{id:1,message : "안녕하세요, 오늘 일정 알려드립니다"},
{id:2,message: "오후 수업 시간입니다."},
{id:3,message: "이제 곧 쉬는 시간입니다"},
];
// 자바스크립트 타이머 객체 정보를 저장하는 변수
// setTimeout : 1회용 타이머 , 지정된 시간 이후에 한 번 동작, 실행시 타이머 정보를 반환
// 타이머 삭제 시 clearTimeout()을 사용
// setInterval : 지정된 시간마다 동작하는 타이머,실행시 타이머 정보를 반환
// 멈출수가 없음. 타이머 삭제 시 clearInterval()을 사용한다
var timer;
class NotificationList extends React.Component {
constructor(props) {
super(props);
// 현재 컴포넌트의 상태인 notifications 선언
this.state = {
// 빈 배열 타입인 state가 생성
notifications:[],
// notifications: reservedNotifications,
};
}
// 화면에 처음 그려질 내용
render() {
return (
// state의 notifications 배열을 가지고 화면을 그려줌
// notifications 배열의 기본값이 비었기 때문에 화면에 아무것도 그리지 않는다
// notifications:[] 이 값을 가져와 돌린다.
// 지금 빈 배열이라 돌릴 수 없다. 그래서 밑에 부분이 화면에 그려지지 않는다.
<div>
{
this.state.notifications.map((item) => {
return <Notification key={item.id} message={item.message}/>
})
}
{/*원래는 아래 처럼 해야 했다. 하지만 귀찮으니까 위의 것 처럼 행한다.*/}
{/*<Notification key={reservedNotifications[0].id} message = {reservedNotifications[0].message}/>,*/}
{/*<Notification key={reservedNotifications[1].id} message = {reservedNotifications[0].message}/>,*/}
{/*<Notification key={reservedNotifications[2].id} message = {reservedNotifications[0].message}/>,*/}
</div>
)
}
// render() 함수 실행 후 동작(다 끝내고 동작하는 함수)
componentDidMount() {
// object 타입의 확장 표현식을 통해서 변수 notifications에 state가 가지고 있는 notifications의 데이터를
// 대입한다.
//const/let [변수명1, 변수명2, ...] = [원본배열] -> 배열 값을 차례대로 하나씩 대입시킨다.
// const val1 = 10; 이거랑 유사하다.
// const val2 = 20;
const {notifications} = this.state;
//state의 notification의 데이터가 비어 있음
// const {notificaions} = this.state.notifications
// 랑 같은 거.(확장표현식)
// 타이머를 사용하여 지정된 시간마다 동작하도록 설정함
// timer = setInterval(() => {
// if (notifications.length < reservedNotifications.length) { //notifications.length는 무조건 0으로 뜨고 reservedNotifications.length는 3
// const index = notifications.length;
// // 배열 notifications에 데이터 추가
// notifications.push(reservedNotifications[index]); //데이터 하나를 추가해준다
//
// //state의 상태 수정
// this.setState({
// // this.state에 있는 notifications에 현재 componentDidMount 안에 있는 지역변수
// // notifications의 데이터를 저장한다
// notifications: notifications,
// });
// } else {
// clearInterval(timer);
// }
// }, 2000);
setTimeout(() => {
this.setState({
notifications : reservedNotifications,
});
console.log("데이터 추가 완료");
}, 3000);
console.log("마운트 완료")
}
componentDidUpdate() {
console.log("상태 업데이트")
}
}
export default NotificationList;
import React from 'react';
import ReactDOM from 'react-dom/client';
import './index.css';
// default로 되어있어서 {}없이 바로 import 가능하다
import App from './App';
import reportWebVitals from './reportWebVitals';
import App2 from "./App2";
import NotificationList from "./mounts/NotificationList";
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
<React.StrictMode>
{/* js의 문법. js의 return부분을 사용하겠다는 의미.
jsx = js + html + xml
여기에 넣을 때 전부 시작태그 끝 태그 다 있어야 한다.
<APP></APP> 혹은 <APP / > 둘 중 하나를 골라 해야한다.*/}
{/* 무조건 대문자A를 사용해줘야 한다.*/}
{/* jsx문법에서 화면을 렌더링하기 위한 태그의 이름은 반드시 첫글자가 "대문자"이어야 함*/}
{/* 대문자를 사용하는 이유는 일반적으로 html 태그를 모두 소문자로 사용하기 때문에 일반 html태그인지
jsx문법으로 생성된 태그인지를 구분하기 위함이다*/}
<App />
<br/><hr /><br />
<App2/>
<NotificationList/> <----- 여기 추가됨!!
<br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br />
</React.StrictMode>
// <h2>이렇게 부모태그가 2개가 되면 오류가 발생한다.</h2>
);
reportWebVitals();