원글 페이지 : 바로가기
파일구조 // 페이지(컴포넌트) 임포트
import Home from “./public/pages/Home.js”;
import Design from “./public/pages/Design.js”;
import Board from “./public/pages/Board.js”;
import Submit from “./public/pages/Submit.js”;
// 경로 문자열을 정규 표현식으로 변환하는 함수
const pathToRegex = path => new RegExp(“^” + path.replace(/\//g, “\\/”).replace(/:\w+/g, “(.+)”) + “$”);
// 매개변수와 일치하는 값을 추출하는 함수
const getParams = match => {
const values = match.result.slice(1);
const keys = Array.from(match.route.path.matchAll(/:(\w+)/g)).map(result => result[1]);
return Object.fromEntries(keys.map((key, i) => {
return [key, values[i]];
}));
};
// URL을 변경하고 새로운 페이지를 로드하는 함수
const navigateTo = url => {
history.pushState(null, null, url);
window.scrollTo(0, 0);
router(); // 새로운 페이지 로드
}
// 라우터 함수
const router = async () => {
// 라우트와 페이지(컴포넌트) 매핑
const routes = [
{ path : “/”, pages : Home},
{ path : “/design”, pages : Design},
{ path : “/board”, pages : Board},
{ path : “/submit”, pages : Submit}
];
// 현재 URL 경로에 대응하는 라우트를 찾음
const potentialMatches = routes.map(route => {
return {
route : route,
result : location.pathname.match(pathToRegex(route.path))
};
});
// 매칭된 라우트 찾기
let match = potentialMatches.find(potentialMatch => potentialMatch.result !== null);
// 매칭된 라우트가 없을 경우 기본 라우트로 설정
if (!match) {
match = {
route : routes[0],
result : [location.pathname]
};
}
// 매칭된 라우트에 대응하는 페이지 인스턴스 생성
const pages = new match.route.pages(getParams(match));
// 페이지의 HTML을 가져와서 #app 요소에 삽입
document.querySelector(“#app”).innerHTML = await pages.getHTML();
// 페이지의 실행 스크립트 실행
await pages.executeScript();
};
// 브라우저 히스토리의 상태 변화에 대응하여 라우터 함수 호출
window.addEventListener(“popstate”, router);
// 문서가 완전히 로딩되면 실행되는 이벤트
document.addEventListener(“DOMContentLoaded”, ()=> {
// 클릭 이벤트를 리스닝하여 [data-link] 속성이 있는 링크가 클릭되면 페이지 전환 처리
document.body.addEventListener(“click”, e => {
if(e.target.matches(“[data-link]”)){
e.preventDefault();
navigateTo(e.target.href);
}
});
// 초기 로딩 시에도 라우터 함수를 호출하여 현재 URL에 대응하는 페이지 로드
router();
}); 아래의 클래스는 라우터를 통해 동적으로 로드되는 각 페이지(컴포넌트)에서 공통적으로 사용되는 메서드와 속성을 정의함 // 클래스 정의 및 내보내기
export default class {
// 생성자 메서드
constructor(params) {
// 클래스 인스턴스의 params 속성에 전달받은 매개변수(params) 설정
this.params = params;
// 매개변수를 콘솔에 출력
console.log(this.params);
}
// 페이지 제목을 설정하는 메서드
setTitle(title) {
document.title = title;
}
// 비동기 메서드 (아직 구현되지 않음)
async fetch(url) {
return “”; // 아무런 동작을 하지 않고 빈 문자열을 반환
}
// 뷰 스크립트를 실행하는 비동기 메서드 (아직 구현되지 않음)
async executeViewScript() {
return “”; // 아무런 동작을 하지 않고 빈 문자열을 반환
}
} 위의 layout클래스를 상속받은 클래스는 아래처럼 구성됨. // layout 모듈에서 가져온 클래스를 확장(상속)하는 클래스 정의 및 내보내기
export default class extends layout {
// 생성자 메서드
constructor(params) {
// 부모 클래스의 생성자 호출
super(params);
// 페이지 제목 설정
this.setTitle(“Home”);
}
// 비동기 메서드: HTML을 반환하는 메서드
async getHTML() {
return `
Craft Your
Dreams
인테리어 스타일링과 예상 견적 서비스를 제공합니다.
`;
}
// 스크립트를 실행하는 메서드
executeScript() {
console.log(“Hello, This is the Main Page”);
}
} parameter값과 Title을 Submit으로 동적으로 구성함. getHTML메서드를 통해서 뷰를 비동기 함수로 전달한다. 위의 구조를 보면 babel과 webpack이 있는데..! 그건 아래글 참고 (–스크랩글) https://seo-tory.tistory.com/74 [JS] 웹팩(Webpack)과 바벨(Babel): 바닐라 자바스크립트에 처음부터 적용시켜보기 최근 바닐라 자바스크립트로 과제를 만들 기회가 있었다. 요즘 계속 React를 많이 사용했었기 때문에 JSX의 편리함에 익숙해져있던 편이었지만 다행히도 바닐라 자바스크립트를 사용해서 코딩하 seo-tory.tistory.com 함께 참고하면 좋은 블로그 글 https://velog.io/@jhyj0521/FE-%EB%B0%94%EB%8B%90%EB%9D%BC-%EC%9E%90%EB%B0%94%EC%8A%A4%ED%81%AC%EB%A6%BD%ED%8A%B8%EB%A1%9C-%EC%9B%B9-%EC%BB%B4%ED%8F%AC%EB%84%8C%ED%8A%B8-%EB%A7%8C%EB%93%A4%EA%B8%B0-feat.-POT-%ED%94%84%EB%A1%9C%EC%A0%9D%ED%8A%B8 [FE] 바닐라 자바스크립트로 웹 컴포넌트 만들기! – feat. ‘POT’ 프로젝트 최근에 웹 컴포넌트라는 개념을 처음 듣고 관심이 생기게 되었다 😀새로 알게 된 기술을 학습할 때 공식 문서를 참조하여 간단하게 작은 것이라도 만들어 보고자 하는 편인데, 더 좋은 방법은 velog.io 강의 : 노마드코더 바닐라 자바스크립트 강의