Skip to content
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

feat: add review item component #37

Merged
merged 4 commits into from
Jun 13, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 34 additions & 0 deletions components/review-item/review-item.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
.review-item {
display: flex;
flex-direction: column;
align-items: center;
border-radius: 10px;
padding: 41px 51px 31px;
background-color: #efefef;
max-width: 568px;
}
.review-img-text-container {
display: flex;
align-items: center;
}
.review-img {
width: 104px;
height: 104px;
flex-shrink: 0;
margin-right: 31px;
}
.review-img img {
width: 100%;
height: 100%;
padding-top: 7px;
border-radius: 50%;
background-color: #ffffff;
}
.review-text {
font-size: 14px;
}
.review-name {
width: 100%;
margin-top: 14px;
text-align: right;
}
56 changes: 56 additions & 0 deletions components/review-item/review-item.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
class ReviewItem extends HTMLElement {
constructor() {
super();
this.attachShadow({ mode: "open" });
}

// initial render in here to ensure the element is fully connected to the DOM.
connectedCallback() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

connectedCallback을 사용하면 더 안전하게 render할 수 있겠군요! 참고하겠습니다!

this.render();
}

static get observedAttributes() {
return ["text", "name", "img"];
}

attributeChangedCallback(attrName, oldVal, newVal) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

혹시 constructor 나 connectedCallback이 아닌 observedAttributes과 attributeChangedCallback을 사용에 render method를 넣었을 때 얻을 수 있는 이점을 알 수 있을까요?

Copy link
Contributor Author

@yolophg yolophg Jun 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 질문이 혹시, 동적 변경 관련 메소드를 사용한 것에 대한 질문일까요? 아니면, constructor/connectedCallback 말고, attributeChangedCallback에서 render()를 호출했을 때의 차이에 대해 질문하신걸까요? 아니면 둘 다 인걸까요?😄

전자의 대해 답변 드리자면, 첫 의도는 일단, 추후 확장성이나 유연성을 혹시라도 고려하는게 좋지 않을까라는 생각과, 사실 동적인게 더 익숙하고 편안해서 일단은 제가 그렇게 만들고 싶어서 작성을 해본 것도 있는 것 같습니다. 😀
근데, 동시에 한편으론, 완전 정적으로 유지보수가 될 것이고, 해당 컴포넌트가 추가적인 기능을 하게 될 가능성이 없을 듯 하여, 코드 간소화를 위해서라도 당장 불필요한 동적 변경 기능을 하는 메소드들은 제거를 하려했던 상태입니다...! 한 번 더 이 부분에 대해 짚어주셔서 감사합니다 :) 제거하고 다시 푸쉬할 예정입니다!

후자의 경우에 대해 답변 드리자면, constructor/connectedCallback에서 호출한 render(저의 경우에는 connectedCallback에서 호출했습니다)와 attributedChangedCallback에서 호출한 render의 의미는 완전히 다르게 의도해서 작성했습니다!

  • connectedCallback에서 호출한 render : 엘리먼트가 DOM에 추가된 이후에 초기 렌더링을 위함입니다.
  • attributeChangedCallback에서 호출한 render : 동적으로 구독된 attribute 중 하나가 변경되는 사항이 있을 때, 재렌더링을 해주기 위함입니다.

하지만, 결론적으로 동적 변경 기능을 하는 해당 메소드를 제거할 예정이라, render()함수를 포함하고 있던 불필요했던 메소드 또한 사라질테니, 최종적으로 유지보수에 더 좋은 방향일거라고 판단됩니다. 질문해주셔서 감사합니다 :)

Copy link
Contributor

@SamTheKorean SamTheKorean Jun 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

상세한 답변 감사드립니다! 지금 제 질문을 다시 보니 충분히 구체적으로 질문드리지 않아 오해의 소지가 있고 애매모호한 점이 많네요...ㅎㅎ 우선 질문은 전자에 대한 것 하나입니다! 저도 비슷한 생각으로 확장성을 고려해 동적 변경을 고려하여 개발했다가 전에 달레님께서 정적 유지보수로만 진행할 거라고 짚어주신 기억이 있고 테크리드님의 피드백도 있어서 관련 메서드를 뺐습니다! 혹시나 Helena님께서도 비숫한 생각으로 개발하신 것 같아 여쭤봤습니다! 부족한 질문이었음에도 불구하고 상세히 설명해주셔서 감사드립니다!

Copy link
Contributor

@DaleSeo DaleSeo Jun 13, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

우리 다음 모임 때 지금까지 각자 시도해보신 여러 스타일의 웹 컴포넌트 작성 방식을 장단점을 비교해보고 Best Practice를 정립해보는 시간을 가져보면 어떨까요? 안 그럼, 제가 나중에 유지보수하거나 다음 기수 개발 팀에게 인계해주기가 상당히 힘들 것 같다는 생각이 들었어요.

@sounmind @yolophg 두 분 중 한 분이 지금까지 나온 스타일을 정리하셔서 간단히 문서화해주시고 관련 토론을 리딩해주실 수 있으실까요?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

좋습니다, 어제 미팅에서도 각자 스타일대로 컴포넌트를 작성 중이었던 점 잠깐 언급됐었는데 요부분을 어느정도는 통일감을 가져가는 것이 유지보수 측면에서도 그렇고 저희가 당장 협업하면서도 혼란이 적을 것 같습니다.
이 부분은 제가 진행토록 하겠습니다, #44 에 이번 이터레이션 이슈로 만들어두었습니다!

if (oldVal !== newVal) {
this.render();
}
}

// getter to make it clean to access the attribute value and ensure a default empty string if the attribute is missing.
get text() {
return this.getAttribute("text") || "";
}

get name() {
return this.getAttribute("name") || "";
}

get img() {
return this.getAttribute("img") || "";
}

render() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Inspired from React? 😆

try {
this.shadowRoot.innerHTML = `
<link rel="stylesheet" href="components/review-item/review-item.css">
<div class="review-item">
<div class="review-img-text-container">
<div class="review-img">
<img src="${this.img}" alt="Person">
</div>
<div class="review-text">${this.text}</div>
</div>
<div class="review-name">${this.name}</div>
</div>
`;
} catch (error) {
// log any rendering errors.
console.error("Error during render:", error);
}
}
}

customElements.define("review-item", ReviewItem);
123 changes: 34 additions & 89 deletions index.html
Original file line number Diff line number Diff line change
Expand Up @@ -143,10 +143,10 @@
}

.review-card-list {
margin-top: 4rem;
display: flex;
flex-wrap: wrap;
gap: 0.9rem;
display: grid;
grid-template-columns: repeat(2, 568px);
gap: 10px 9px;
padding-top: 40px;
}

.review-card {
Expand Down Expand Up @@ -279,91 +279,36 @@
<div id="member-review-container">
<span>참가자 후기</span>
<div class="review-card-list">
<div class="review-card">
<div class="picture-placeholder">
<img src="images/human.png" />
</div>
<div class="review">
<p>
해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은
선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를
풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며
실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수
있었습니다.
</p>
<div class="name">
<span>Sam</span>
</div>
</div>
</div>
<div class="review-card">
<div class="picture-placeholder">
<img src="images/human.png" />
</div>
<div class="review">
<p>
해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은
선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를
풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며
실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수
있었습니다.
</p>
<div class="name">
<span>Sam</span>
</div>
</div>
</div>
<div class="review-card">
<div class="picture-placeholder">
<img src="images/human.png" />
</div>
<div class="review">
<p>
해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은
선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를
풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며
실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수
있었습니다.
</p>
<div class="name">
<span>Sam</span>
</div>
</div>
</div>
<div class="review-card">
<div class="picture-placeholder">
<img src="images/human.png" />
</div>
<div class="review">
<p>
해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은
선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를
풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며
실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수
있었습니다.
</p>
<div class="name">
<span>Sam</span>
</div>
</div>
</div>
<div class="review-card">
<div class="picture-placeholder">
<img src="images/human.png" />
</div>
<div class="review">
<p>
해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은
선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를
풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며
실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수
있었습니다.
</p>
<div class="name">
<span>Sam</span>
</div>
</div>
</div>
<review-item
img="images/human.png"
text="해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은 선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를 풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며 실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수 있었습니다."
name="John Doe"
>
</review-item>
<review-item
img="images/human.png"
text="해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은 선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를 풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며 실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수 있었습니다."
name="John Doe"
>
</review-item>
<review-item
img="images/human.png"
text="해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은 선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를 풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며 실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수 있었습니다."
name="John Doe"
>
</review-item>
<review-item
img="images/human.png"
text="해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은 선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를 풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며 실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수 있었습니다."
name="John Doe"
>
</review-item>
<review-item
img="images/human.png"
text="해외취업을 목표로 리트코드 스터디 모임에 참여한 것은 좋은 선택이었습니다. 특히 혼자 공부하는 것보다 멤버들과 함께 문제를 풀며 시너지를 낼 수 있었습니다. 모임 중에 다양한 질문을 주고받으며 실력을 쌓을 수 있었고, 해외 취업 준비에 실질적인 도움을 받을 수 있었습니다."
name="John Doe"
>
</review-item>
</div>
</div>

Expand Down
1 change: 1 addition & 0 deletions main.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,5 @@ import "./components/divider/divider.js";
import "./components/footer-link/footer-link.js";
import "./components/hero/hero.js";
import "./components/link-button/link-button.js";
import "./components/review-item/review-item.js";
import "./components/seo-meta-tag/seo-meta-tag.js";