오늘 한 일

  • 검색창을 띄우는 버튼을 누르면 검색창으로 페이지 이동이 완전하게 이루어져야 하는데, 자꾸만 검색창이 페이지 가운데에 뜨고 양 옆 부분으로는 다른 컴포넌트가 보이는 기현상이 일어났다. z-index 문제인가 싶어서 검색창을 0까지 내려보기도 했는데 , 그런 식으로 해결되는 문제는 아니었다. 왜냐하면 두 컴포넌트의 url이 겹쳤기 때문이었다...^^ 아무리 관련 있는 컴포넌트라지만 url 분기를 제대로 안 하다니, 순식간에 해결되는 걸 보고 허무와 충격을 동시에 느꼈다.
  • 카테고리 이름을 클릭하면 카테고리별로 뉴스 목록이 나타나는 페이지에서는 카테고리가 달라질 때마다 브라우저 주소창 위 타이틀이 달라져야 했다. 구글링 해서 react-helmet 을 설치하니 컴포넌트별로 메타 태그에도 손을 댈 수 있었다. 사용법도 간단했다. 앞으로도 애용할 것 같다.
  • 사용자가 이메일 주소와 닉네임을 입력하면 해당 데이터를 서버로 넘겨서, 서버에서 메일을 발송하기로 했다. axios를 적용해 작업을 완료했다.
  • 검색창에서 검색어를 입력하면 주소창에 검색어를 들어가게 한 후, 곧바로 검색 결과 페이지로 이동시킨 다음 거기서 그 주소창의 검색어를 다시 따서 서버에 보내 관련 뉴스 목록을 받아오는 axios 코드를 다른 프론트 분과 함께 짰다. 버튼을 클릭하면 최신순과 관련순으로도 받아올 수 있도록 했다.
  • 배포해도 수정은 가능하니 일단 배포를 했다. 백엔드 분들도 데이터가 어떻게 펼쳐지는지를 보고 싶으실 텐데 지금까지 프론트에서 공유해주는 화면만 보셨으니 답답하셨을 것 같다는 생각이 들었기 때문이다. 수고했다는 말을 들을 때마다 프론트가 들이는 시간이 상대적으로 많다는 이유로 항상 공이 프론트에게 많이 돌아가게 되는 것 같아 민망스럽다. 협업해서 함께 만든 건데. 언제나 한결같이 배려해주시는 백엔드 분들 덕분에 수월하게 마무리 되어 가는 중이다.

내일 할 일

  • 다른 프론트 분이 로딩 스피너를 달아주셨는데, 스피너 자체가 조금 늦게 뜨거나 해당 컴포넌트에 데이터를 불러오지 않은 모습을 한번 띄운 후 뜨는 미운 모습을 보여서 다시 함께 방법을 연구해보기로 했다.
  • 카테고리 메뉴는 클릭하면 그 밑에 언더라인이 뜨는데 다른 페이지로 이동하기 전까지는 그게 유지된다. 하지만 내가 구현한 걸로는 마우스 클릭이 유지되는 상태에서만 언더라인이 잠깐 뜨고 사라진다. 이 부분 해결이 필요하다.
  • footer에 마우스를 올리면 글자가 옆으로 흐르는 애니메이션이 적용되어 있다. 구현해보도록 한다.
  • 상세 페이지에서는 프로그레스바가 뜨는데, 스크롤 막대의 위치에 따라 최상단에서는 보이지 않다가 어느 지점에 이르면 나타난다. 이것도 연구하기.
  • 상세 페이지 뉴스 제목 위 카테고리를 클릭하면 해당 카테고리의 뉴스 목록 페이지로 이동하게 한다. 어려울 것 같진 않다.

오늘 한 일

  • 메인 페이지 뉴스 목록 아래에는 '더보기' 버튼이 있어서, 처음에는 목록을 몇 개만 불러오고 그걸 눌렀을 때 추가로 더 붙여서 보여주는 식이다. 버튼을 누를 때마다 다음 목록이 나오는 거니까 페이지 버튼만 없을 뿐, 결국 화면에 나오지 않는 모든 데이터의 목록을 균등하게 나누는 페이지네이션이 필요했다. 주특기 주간 과제로 무한스크롤을 구현할 때 페이지네이션이 너무 어려웠던 기억이 나서, 이번만큼은 강의자료를 보는 대신 구글링을 먼저 했다. 운좋게도 나에게 필요한 수준의 간단한 페이지네이션을 아주 쉽게 설명해 놓은 블로그를 찾았다. 눈으로 쭉 코드만 훑었을 때에도 '이걸로 이게 된다고?' 싶을 정도로 짧았다. 역시 간결한 코드가 좋은 코드인 걸까, 단번에 실행됐다. 따라 치며 이해를 한 후에는 그 코드를 내 방식대로 더 짧게 만들어 보았다.
const CardList = () => {
	const [end, setEnd] = useState(12);     	  // 한 번에 불러올 개수 입력
	const all_news = useSelector(state => state.news.list);
	const news_list = all_news.slice(0, end); 	  // 0부터 (end-1)번째까지 출력

	const loadMoreNews = () => {
		if (end >= all_news.length) {             // 다음 목록이 없으면,
			return;                           // 리턴
		}
		setEnd(end + 12);                         // 다음 목록이 있으면 end를 증가
	};
}
  • 메인 페이지 뉴스 목록에 데이터가 잘 들어오기는 하는데 CSS에 하자가 좀 있어 오늘은 그 부분을 뜯어 고쳤다. 코드가 내 마음에 들게 실행되는 순간만큼 기쁠 때가 또 있을까.
  • 뉴스 목록에 뜨는 날짜들이 뒤죽박죽인 걸 깨닫고 나서는 날짜 내림차순으로 정렬해서 다시 배치했다.
  • 카드 리스트 안에 카드들이 들어있는 형태인데, 각자가 테두리 선을 가지고 있어서 겹치는 부분은 굵어졌다. 구글링을 통해 찾은 border-collapse: collapse; 로도 해결이 되지 않아 box-shadow 속성을 이용하는 방법을 찾아서 적용해봤다. 정석대로 처리한 건 아닌 것 같지만 전혀 생각지 못한 방법으로도 해볼 수 있다는 걸 배웠다.
  • 카테고리 메뉴를 누르면 카테고리별로 서버에서 데이터를 받아와 메인에서와 같은 카드 리스트 안에 담아 띄우도록 구현했다. 

내일 할 일

  • 사용자가 이메일 주소와 닉네임을 입력하면 해당 데이터를 서버에 넘겨주는 axios를 짠다.
  • 검색창 CSS가 알 수 없는 이유로 완전히 뭉개졌다...ㅠ.ㅠ 고치는 것을 시도해봐야겠다.

오늘 한 일

  • 쉬는 날이어서 오늘도 진짜 쉬었다. 지난 프로젝트 때 일요일 쉬었다가 그 후에 너무 바빠졌지만, 그래도 이번에도 쉬었다.
  • 이번에 함께 프론트를 맡은 분은 지난 주에 재정비반에서 복습을 하셨기 때문에 이번 협업 프로젝트의 진행을 낯설어 하셨다. 팀 회의를 진행할 때도 협업의 과정도, 용어도 잘 모르니 회의 내용을 이해하기가 어렵다고 하셨다. 나도 백엔드와의 협업 프로젝트를 처음 시작했던 지난 주에 회의에서 말 한 마디 나올 때마다 스스로에게, 또는 구글에게, 또는 백엔드 팀원분들에게 많은 질문을 던졌었다. 그때로부터 단 1주일 지났을 뿐인데 그 고충을 잊어버리다니. 내가 미처 챙기지 못하고 지나쳐서 죄송했다. 오늘 잠깐 짬이 난 김에 내게도 딱 한 번뿐이긴 하지만 그 협업 경험을 바탕으로 만든 간단한 자료를 노션에 올려 공유하고 설명해드렸다. 그리고 게더의 화면 공유 기능을 이용해, 실습(?) 겸 해서 백엔드에서 주신 API URL을 연결하고 axios를 실제 코드에 적용해 데이터를 불러오는 과정을 보여드렸다. 비록 구멍 나 있는 CSS 때문에 예쁘게 보이지는 않았지만 다행히 생각했던 대로 데이터 자체는 잘 불러올 수 있었다.

내일 할 일

  • 메인 페이지 CSS의 구멍난 부분을 메우고, 메인 페이지의 데이터를 예쁘게 불러올 수 있는지 확인한다.
  • 상세 페이지에 서버를 연결하고 데이터가 예쁘게 띄워지는지 확인한다.

오늘 한 일

  • 애증의 리덕스도 몇 번 써보니 코드를 짜는 데엔 큰 어려움은 없는 것 같다. 다만 리듀서에서 데이터를 스테이트에 반영하는 부분이 좀 부족해 오류 발생이 잦다. 한가할 때 꼭 리덕스에 대해서 다시 공부해봐야겠다. 아니, 이게 아닌데. 오늘 한 일을 써야 하는데... 아주 그냥 의식의 흐름대로 쓰고 있다. 뷰를 잡을 때 단위가 작은 컴포넌트들부터 만들면서 시작하게 되면 처음에는 시간이 많이 걸린다는 단점이 있지만, 뒤로 갈수록 속도가 붙어 뚝딱뚝딱 조립하게 된다. 클론코딩이라 참고할 자료들이 있다는 특수성이 더해진 프로젝트이긴 하지만 말이다. 클론코딩의 장점은, 뷰를 잡고 나면 뿌듯함이 크다는 것이다. 누군가 피땀 흘려 디자인한 아이디어를, 코딩 연습이랍시고 내가 막 갖다 쓰는 게 아닌가 하는 생각도 든다. 아무튼 뷰는 세부적인 부분만 시간 날 때 조금씩 들여다보며 고쳐가면 될 것 같고, api 리스트 중 1, 2번째인 메인 페이지 전체 뉴스 목록 불러오기와 상세 페이지 개별 뉴스 내용 불러오기에 대한 코드를 짰다. 리덕스를 내가 붙이겠다고 하고 시작한 작업이었지만 axios까지 적고 나니 아차 나 혼자 너무 멀리 왔나 싶었다.

내일 할 일

  • 일요일이라 쉬는 날이지만, 마음이 무거워서 하루종일 쉴 수는 없을 것 같다. 아직 구현하지 않은 남은 api 리스트 두 항목도 코드를 완성해봐야겠다.
  • 다만 그 전에, 이번에는 나 혼자 가지 말고 꼭 천천히 함께 이해하며 넘어가자.

오늘 한 일

  • 대망의 클론코딩 주간이 밝았다. 미니 프로젝트 때도 너무나 짧게 느껴졌던 1주일이라는 시간 안에 어떻게 이걸 해내라는 건지는 여전히 의문이지만(그리고 심지어 이 모자란 내가 팀장만 4번째...), 아무튼 우리 조는 뉴닉을 클론코딩의 대상으로 정했다. 스코프는 나중에라도 늘릴 수 있기 때문에 일단은 좁게 잡아야 한다고 해서 메인 페이지에서 전체 기사 목록 보여주고, 카테고리별로 나눠서도 보여주고, 상세 페이지에서 기사 내용 보여주고, 검색창에서 키워드로 검색하면 검색 결과 보여주고... 로그인과 회원가입 기능은 빼고 이 정도만 구현해보는 걸로 결정했다.
  • 깃헙 저장소를 만들었는데 왜인지 로컬 저장소와 연결하는 처음부터 난리난리가 나서 결국 밀어버리고 다시 시작했다. 프로젝트 극초반이라면 사람과 마찬가지로 저장소도 고쳐 쓰지는 말자. 이미 만들어둔 저장소에 미련이 남아 밀어버리기로 결정하기까지 시간이 다소 소요되었지만 밀고 나니 작업의 진행 속도가 붙었다.
  • 아무리 클론코딩할 사이트를 개발자 도구로 뜯어본다고 해도 최대한 비슷하게 구현하는 것은 추가적인 노력이 필요한 것 같다. 최선을 다해 따라하는데 어째 그 모양새가 안 나온다. 그래도 오늘, 그렇게나 멀게만 느껴졌던 반응형 페이지 설정하는 방법을 우연한 기회에 알게 되어 조금 더 완성도가 높은 뷰를 제작할 수 있게 되었다. 원래 쓰던 모듈에서 뭘 더 추가적으로 설치할 필요 없이! 이것만 해도 이번 주간의 충분히 큰 수확이다. (*/ω\*) ~ ♬

내일 할 일

  • 메인 페이지 뷰의 부족한 부분을 마저 채우고 나머지 페이지들의 뷰 잡는 작업도 시작한다. 기초 단위 컴포넌트들을 오늘 거의 만들어 놓아서 메인 페이지만큼 시간이 많이 걸릴 것 같지는 않다.
  • 서버단에서 임시로 쓸 수 있는 api를 넘겨 주시면 연결을 해본다. (어쩌면 임시가 아닐 수도 있다)

Day38 한 일

  • 상세한 내막을 잘은 모르지만 백엔드에서는 비교적 우리보다 빠르게 작업을 마쳤고, 배포까지 끝낸 상태에서 우리가 끝내기를 기다리고 있었다. 그게 저녁 나절이었나. 벌써 기억이 가물하지만 어떤 이유에선지 앞으로 해야 할 작업들을 가볍게 생각한 나는 큰코를 다치고야 말았다. 처음 하는 일이었지만 다행히 큰 어려움 없이 JWT 토큰을 쿠키에 저장하는 데 성공했고, 게시글 작성 여부를 판단하기 위해 사용자를 구별하는 정보가 추가적으로 필요해서 회원가입과 동시에 자동으로 부여되는 사용자의 고유번호를 따로 보관해야 했다. 하지만 일전에 적었듯이 리액트스럽지 못하게 페이지 이동을 강제하는 새로고침을 넣은 상황에서, 바로 그 새로고침 때문에 기껏 보존해둔 state 데이터가 매번 모조리 날아가게 되어 어려움을 겪었다. login을 마치면 쿠키에 토큰이 저장되고 리덕스에 사용자 정보가 들어가는데, login이 끝남과 동시에 메인 페이지로 이동시키는 새로고침이 발생해 정보가 깨끗하게 증발해버리고 마는 것이었다. 이 문제로 인해 몇 시간을 구글링하고 머리 싸맨 끝에 동환님이 id가 데이터에 들어간 것이 확인된 후 history.push('/') 하게 만드는 식으로 해결했다. 이게 가장 완벽하고 적절한 해결책이라고 할 수는 없겠지만 새벽 3시에 내릴 수 있었던 최선의 선택이었음은 분명해 보인다.

오늘 한 일

  • 늦게 잔 것치고는 제때 일어나 늦지 않게 출석했다. 자정까지 과제를 제출해야 하는 날이었는데 자꾸만 새로이 발견되는 오류들과, 후딱 해치워버리고 싶지만 의외로 사람을 끝까지 물고 늘어지는 자잘한 문제들에 온종일 시달렸다. 우리가 고전하느라 백엔드에서는 마냥 손놓고 기다려야 할 뿐이어서 마음이 무거웠다. 실력이 부족한 내가 원망스러운 날이었다ㅠㅠ.. 깃은 왜 그렇게 안 도와주는지 푸시 대상인 파일을 처리해주지 않아 어이없게 누락되고, 결국에는 로컬 저장소를 밀어버리고 다시 통째로 클론해오고 하는 일이 빈번했다. 그래, 이러지 않으면 마지막 날이 아니지.
  • 그래도 완성하고 나니 (비록 그 후에도 몇 번이나 수정->재빌드->재배포 반복) 뿌듯했다. 사실 안 그래도 되지만 괜히 완전한 주소로 남겨놓고 싶어 도메인도 새로 구입해서 달았다. 원래 자정까지 배포한 후 제출하는 게 과제였지만 다행이라고 해야 할지, 제출 링크의 DB에 뭔가 문제가 생겼다고 해서 내일 아침으로 미뤄졌다. 그 덕에 조금 더 여유롭게 마무리했고, 나름대로 훈훈하게 마무리를 지었다. 도메인은 http://dev-moim.shop/
  • 과제 제출로 일찍(?) 끝난 기념으로 해산 후 객체 지향 프로그래밍에 관한 유튜브 영상을 3편 정도 봤다. 아직 코딩과 프로그래밍에 대해 모르는 게 너무 많다.

내일 할 일

  • 과제 제출 확인 후 코드 리뷰를 하고 깃헙 저장소를 정비하고 다른 팀의 과제를 구경한다.
  • 내일은 드디어 클론코딩 주간의 시작이다. 내가 뭘 안다고 벌써 클론코딩을 하는 건지, 말도 안되고 당황스럽기 그지없지만 하다 보면 또 하게 되는 것 아닐까. 다만 팀에 조금이라도 더 도움이 되는 사람이 되고 싶다.

오늘 한 일

  • 어제 TIL을 쓰고 아침 7시에 잠들었다. 새벽에 가짜 데이터를 넣어놓은 json 파일 때문에 터진 오류를 해결하다가 결국 데이터 구조 문제에 봉착해서 포기하고 잤지만, 어쨌든 늦게 잔 후폭풍은 지대했다.. 출근도 1시간이나 늦었고ㅠㅠ 오전에는 게더가 터져 버려서 줌에서 간신히 미팅을 진행했고, 새벽에 맞닥뜨린 문제를 공유했다. 지금 생각해보니 API URL이 사이트 주소와 전혀 상관없다는 걸 어제 배웠었는데... 머리가 잘 돌아가지 않아서 그걸 다시 떠올리지 못하고 다시 비슷한 질문을 했던 것 같다. 아무튼 데이터 구조는 API 리스트에 적힌 거 말고는 당연하게도 새삼 혼자 고민하고 말고 할 게 없다는 것!
  • 점심 나절에 백엔드에서 서버를 배포했고, 우리의 컴포넌트를 서버에 연결했다. 연결하고 나니 위에서 고민했던 그 문제는 처음부터 없었던 것과 다름없는 상태가 되었다. API 리스트의 데이터 구조 그대로를 적용한 백엔드의 서버에 직접 연결한 거니까, 적혀 있는 대로 불러오기만 하면 되는 거였다. 그리고 나 혼자 진행하는 프로젝트에서 파이어베이스를 붙이고 거기서 데이터를 끌어오는 것보다 axios로 서버에서 데이터를 가져오는 게 코드도 짧고 여러 모로 훨씬 수월했다.
  • 전체 게시글의 정보는 메인 페이지에서 받고, 그 메인 페이지의 게시글을 클릭하면 해당 게시글 정보를 상세 페이지에 props로 넘기면서 상세 페이지가 렌더링 된다. 하지만 상세 페이지의 링크를 타고 직접 접속해버리면 메인 페이지를 거치지 않으므로 props가 undefined 되어 게시글 내용을 불러올 수 없다. 이 점을 보완하기 위해 props가 없는 경우 상세 페이지 자체적으로 서버에 연결해 데이터를 가져오는 코드를 짰다.
  • 상세 페이지에서 불러온 댓글을 수정하고 삭제하는 코드를 짰다. 그런데 주특기 기본 주간 과제와 마찬가지로 페이지를 새로고침 해야만 변경된 데이터가 적용되었다. 새로고침 하는 함수를 넣자니 전체 화면이 반짝거리는 게 마땅찮고, 또 컴포넌트를 기본으로 하는 리액트의 사용 취지에도 맞지 않는 것 같아 고민이 된다. 저번 과제 때는 이걸 해결하지 못해서, 글을 작성해서 등록하고 나면 팝업창을 띄워서 메인 페이지로 이동하는 버튼을 누르게끔 만들었었다. ...일단은 더 좋은 방법을 찾을 때까지는 부모 컴포넌트의 window.location.reload();로 버텨보기로. 나는야 리액트바보

내일 할 일

  • 로그인, 회원가입 페이지를 서버와 연결하고, 로그인 여부에 따라서 구현되는 화면 구성을 분기한다.
  • 상세 페이지의 신청하기 버튼을 누르면 해당 사용자의 정보를 서버로 넘길 수 있도록 한다.

오늘 한 일

  • 오늘 TIL에 올릴 내용 많다고 아까(?) 밤 9시쯤 회의할 때 다같이 얘기했었는데 시간이 이쯤 되고 보니 하루종일 뭘 했는지 벌써 가물가물하다. TIL에 올려야지, 했던 내용 중 하나는 API URL에 대한 것이었다. 프론트가 백 없이 작업할 때에는 API 요청 없이 파이어베이스에 연동해서 직접 데이터베이스를 만지다 보니, 우리가 주로 보는 '주소'란 것은 주소창에 들어가는 말 그대로의 주소뿐이었다. 하지만 API URL은 서버에 요청할 때 사용하는 주소이므로 클라이언트단에서 보여지는 주소창 주소와는 전혀 상관없는 것이었다. 난 그걸 몰랐다ㅎ.ㅎ... 백에 무지한 프론트... axios를 쓰면서도, 심지어 거기에 API URL을 적어 넣으면서도, 나는 그걸 몰랐다. 그냥 생각해 본 적도 없었다고 말하는 게 적당할 것 같다. 관심 범위 밖의 일이었다, 지금까지는. (Thanks to 하영님!)
  • 오늘은 손가락 움직임이 더뎌서 한 것도 별로 없었는데 오류는 역대급으로 많이 봤다. 그래도 이제는 그새 조금 익숙해졌다고 오류 뜨는 대로 바로바로 해결할 수 있는 것들이 생겼고, 또 반대로 자주 봐도 여전히 해결책을 모르겠는 것들도 있다. 리덕스와 axios를 만지면서 컴포넌트에 데이터를 주입하려는데 변수를 이렇게 저렇게 설정해봐도, 콘솔에 아무리 찍어봐도 내가 원하는 데이터를 불러오지 못하는 상황도 있었다. 오류 속에 갇혀 있다가 이대로 질식할 것 같아서(ㅋㅋㅋ) 결국 헬프를 쳤고 너무 허망하게도 일이 간단히 해결되고 말았다. 나의 몇 시간 고민을 무색하게 만든 간단한 방법이었다. props로 넘긴 데이터에 또 props라는 이름을 붙인 바람에, 그 데이터를 가져다 쓰기 위해서는 앞에 props.props.를 붙여야만 했다... 이걸 캐치 못한 내가 원망스러우면서도, 어쨌든 해결되어서 기분은 좋았다. (Thanks to 동환님!)
  • 자정을 넘겨서 아까 한 2시쯤, 대충 만든 가짜 데이터 json 파일을 mock api를 사용해서 서버로 돌린 다음 연결하려고 하는데, 적절하다고 생각한 URL을 넣어도 브라우저에는 자꾸 { } 이렇게 빈 중괄호만 떴다. 보통 코드를 짤 때 자신 없는 부분에서는 한 줄 한 줄 콘솔에 다 찍어보는 나로서는 당황스러운 상황이었다. 데이터가 없는 경로를 찍은 것도 아닌데 어떻게 아무것도 안 뜰까? 구글링을 해도 속시원한 답변을 찾을 수가 없었다. 시간도 늦어서 누구한테 물어보기도 애매했는데 마침 요즘 어쩌다 보면 밤 시간에 자주 마주치게 되는 지혁님이 눈에 띄었다. 매우 바쁘셨을 테지만 그때 나는 너무 절망하고 있었다. 나는 프론트와 백 사이에 약속해 놓은 API URL을 타고 들어가는 게 아니라면 데이터를 받아오지 못한다고 생각하고 있었는데, 그 경로에 들어갈 수가 없다면 그 부분을 포함하는 좀 더 넓은 범위의 데이터를 받아온 후 잘라서 사용해보라고 하셨다. 그렇게 해도 되는 건지 안 그래도 궁금했는데 먼저 그렇게 말씀해주셔서 다행이었다. (Thanks to 지혁님!) 그런데 문제는, 그걸 해결하고 나니 다시 (나는) 알 수 없는 이유로 404 에러가 떴다....^^ 오늘은 그만 좀 하자 제발....

내일 할 일

  • 점심 때쯤 시범 배포를 해보기로 했는데, 내가 하고 있는 일에 진전이 없어 어떻게 될지 모르겠다.. 내일, 아니 오늘은 이따가 백에 데이터 샘플(?)을 한번 요청해봐야겠다. 혹시라도 지금 내가 가지고 있는 json 파일보다 접근성이 좋을지도 모르는 일이니까? 주시는 데이터가 뭐든 간에 지금 내가 처한 것보다는 나은 환경을 만들어 줄 것 같다ㅠㅠ
  • json 파일의 데이터 구조는 우리가 애초에 사용하던 임시의 것이었는데 그걸로 하다 보니 실제 서버 연결 때 문제가 많을 것 같아 다시 바꿨다. 바꾼 파일에 대해서 공유해야 한다.
  • CSS 기능적 부분에 약간의 수정이 필요하다.
  • 로그인, 회원가입 페이지 뷰를 확정해야 한다.

+ Recent posts