├── .gitignore ├── babel.config.json ├── murple_logo.png ├── src ├── assets │ └── img │ │ ├── home.png │ │ ├── user.png │ │ ├── comment.png │ │ ├── reservation.png │ │ ├── icons8-multiply-24.png │ │ └── icons8-menu-rounded-24.png ├── modules │ ├── apiUrls.js │ ├── itemCounter.js │ ├── commentCounter.js │ ├── reservationCounter.js │ ├── domSelector.js │ ├── displayItems.js │ ├── popupComment.js │ └── popupReservation.js ├── index.js ├── Test │ ├── test_item_count.test.js │ ├── test_comments.test.js │ └── test_reservations.test.js ├── index.html └── style.css ├── dist ├── assets │ └── img │ │ ├── user.png │ │ ├── icons8-multiply-24.png │ │ └── icons8-menu-rounded-24.png ├── index.bundle.js.LICENSE.txt ├── index.html ├── runtime.bundle.js ├── index.css └── index.bundle.js ├── .babelrc ├── .hintrc ├── .eslintrc.json ├── .stylelintrc.json ├── LICENSE ├── webpack.config.js ├── package.json ├── .github └── workflows │ └── linters.yml └── README.md /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ 2 | coverage/ -------------------------------------------------------------------------------- /babel.config.json: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | "@babel/preset-env" 4 | ] 5 | } -------------------------------------------------------------------------------- /murple_logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/murple_logo.png -------------------------------------------------------------------------------- /src/assets/img/home.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/home.png -------------------------------------------------------------------------------- /src/assets/img/user.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/user.png -------------------------------------------------------------------------------- /dist/assets/img/user.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/dist/assets/img/user.png -------------------------------------------------------------------------------- /src/assets/img/comment.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/comment.png -------------------------------------------------------------------------------- /src/assets/img/reservation.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/reservation.png -------------------------------------------------------------------------------- /dist/assets/img/icons8-multiply-24.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/dist/assets/img/icons8-multiply-24.png -------------------------------------------------------------------------------- /src/assets/img/icons8-multiply-24.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/icons8-multiply-24.png -------------------------------------------------------------------------------- /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "test": { 4 | "plugins": ["@babel/plugin-transform-modules-commonjs"] 5 | } 6 | } 7 | } -------------------------------------------------------------------------------- /dist/assets/img/icons8-menu-rounded-24.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/dist/assets/img/icons8-menu-rounded-24.png -------------------------------------------------------------------------------- /src/assets/img/icons8-menu-rounded-24.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raihan2bd/js-capstone/HEAD/src/assets/img/icons8-menu-rounded-24.png -------------------------------------------------------------------------------- /dist/index.bundle.js.LICENSE.txt: -------------------------------------------------------------------------------- 1 | /*! regenerator-runtime -- Copyright (c) 2014-present, Facebook, Inc. -- license (MIT): https://github.com/facebook/regenerator/blob/main/LICENSE */ 2 | -------------------------------------------------------------------------------- /src/modules/apiUrls.js: -------------------------------------------------------------------------------- 1 | export const BASE_URL = 'https://us-central1-involvement-api.cloudfunctions.net/capstoneApi/apps/Y1Ocl2k5LoJdVEhHia5O'; 2 | export const MOVIE_API = 'https://api.tvmaze.com/seasons/1/episodes'; 3 | -------------------------------------------------------------------------------- /src/modules/itemCounter.js: -------------------------------------------------------------------------------- 1 | const itemCounter = (countContainer, itemContainer) => { 2 | const count = itemContainer.childElementCount; 3 | countContainer.innerText = `(${count})`; 4 | return count; 5 | }; 6 | 7 | export default itemCounter; 8 | -------------------------------------------------------------------------------- /src/modules/commentCounter.js: -------------------------------------------------------------------------------- 1 | const commentCounter = (countContainer, commentContainer) => { 2 | const count = commentContainer.childElementCount; 3 | countContainer.innerText = `(${count})`; 4 | return count; 5 | }; 6 | 7 | export default commentCounter; 8 | -------------------------------------------------------------------------------- /src/modules/reservationCounter.js: -------------------------------------------------------------------------------- 1 | const reservationCounter = (countContainer, reservationContainer) => { 2 | const count = reservationContainer.childElementCount; 3 | countContainer.innerText = `(${count})`; 4 | return count; 5 | }; 6 | 7 | export default reservationCounter; 8 | -------------------------------------------------------------------------------- /.hintrc: -------------------------------------------------------------------------------- 1 | { 2 | "connector": { 3 | "name": "local", 4 | "options": { 5 | "pattern": ["**", "!.git/**", "!node_modules/**"] 6 | } 7 | }, 8 | "extends": ["development"], 9 | "formatters": ["stylish"], 10 | "hints": [ 11 | "button-type", 12 | "disown-opener", 13 | "html-checker", 14 | "meta-charset-utf-8", 15 | "meta-viewport", 16 | "no-inline-styles:error" 17 | ] 18 | } -------------------------------------------------------------------------------- /src/modules/domSelector.js: -------------------------------------------------------------------------------- 1 | export const menuBtn = document.getElementById('mobile-menu'); 2 | export const closeBtn = document.getElementById('cross-menu'); 3 | export const mobileNavlinks = document.querySelectorAll( 4 | '.mobile-nav-group .nav-link', 5 | ); 6 | export const mobNavGroup = document.querySelector('.mobile-nav-group'); 7 | export const modalContainer = document.querySelector('.popup-modal'); 8 | export const listItemsContainer = document.querySelector('.show-group'); 9 | export const itemCountContainer = document.getElementById('count_item'); 10 | -------------------------------------------------------------------------------- /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "browser": true, 4 | "es6": true, 5 | "jest": true 6 | }, 7 | "parser": "babel-eslint", 8 | "parserOptions": { 9 | "ecmaVersion": 2018, 10 | "sourceType": "module" 11 | }, 12 | "extends": ["airbnb-base"], 13 | "rules": { 14 | "no-shadow": "off", 15 | "no-param-reassign": "off", 16 | "eol-last": "off", 17 | "import/extensions": [ 1, { 18 | "js": "always", "json": "always" 19 | }] 20 | }, 21 | "ignorePatterns": [ 22 | "dist/", 23 | "build/" 24 | ] 25 | } -------------------------------------------------------------------------------- /.stylelintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": ["stylelint-config-standard"], 3 | "plugins": ["stylelint-scss", "stylelint-csstree-validator"], 4 | "rules": { 5 | "at-rule-no-unknown": [ 6 | true, 7 | { 8 | "ignoreAtRules": ["tailwind", "apply", "variants", "responsive", "screen"] 9 | } 10 | ], 11 | "scss/at-rule-no-unknown": [ 12 | true, 13 | { 14 | "ignoreAtRules": ["tailwind", "apply", "variants", "responsive", "screen"] 15 | } 16 | ], 17 | "csstree/validator": true 18 | }, 19 | "ignoreFiles": ["build/**", "dist/**", "**/reset*.css", "**/bootstrap*.css", "**/*.js", "**/*.jsx"] 20 | } -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | // import static files 2 | import './style.css'; 3 | import './assets/img/icons8-menu-rounded-24.png'; 4 | import './assets/img/icons8-multiply-24.png'; 5 | import './assets/img/user.png'; 6 | 7 | import fetchTvShow from './modules/displayItems.js'; 8 | import { 9 | closeBtn, 10 | menuBtn, 11 | mobNavGroup, 12 | mobileNavlinks, 13 | } from './modules/domSelector.js'; 14 | 15 | // toggle the menu 16 | menuBtn.addEventListener('click', () => { 17 | mobNavGroup.classList.toggle('display-flex'); 18 | }); 19 | 20 | closeBtn.addEventListener('click', () => { 21 | mobNavGroup.classList.remove('display-flex'); 22 | }); 23 | 24 | mobileNavlinks.forEach((item) => { 25 | item.addEventListener('click', () => { 26 | mobNavGroup.classList.remove('display-flex'); 27 | }); 28 | }); 29 | 30 | // load the item list on the fly. 31 | window.onload = () => { 32 | fetchTvShow(); 33 | }; 34 | -------------------------------------------------------------------------------- /src/Test/test_item_count.test.js: -------------------------------------------------------------------------------- 1 | import itemCounter from '../modules/itemCounter.js'; 2 | 3 | describe('Test Display Item counter', () => { 4 | document.body.innerHTML = ` 5 |
6 |
74 |
75 |
(back to top)
81 |
82 |
83 | ## 💻 Getting Started
84 |
85 | >Please follow the instructions to clone the repo
86 | To get a local copy, follow these simple example steps.
87 | Clone this repository or download the Zip folder:
88 |
89 | ### Prerequisites
90 |
91 | >In order to check the linters errors make sure you have installed the [nodejs](https://nodejs.org)
92 |
93 |
94 | ### Setup
95 |
96 | >Clone this repository to your desired folder: cd [folder] to navigate and run the below comand to clone the project
97 |
98 | ```sh
99 | git clone https://github.com/raihan2bd/js-capstone.git
100 | ```
101 |
102 |
103 | ### Install
104 |
105 | Install this project with:
106 |
107 | download npm the packages
108 | ```sh
109 | npm install --save-dev
110 | ```
111 |
112 | ### Usage
113 |
114 | > To see the project in your local browser first of all, go to the dist folder then open the index.html file.
115 |
116 |
117 |
118 |
119 |
120 |
121 | ## 👥 Author
122 |
123 |
124 |
125 | 👤 **Abu Raihan**
126 |
127 | - GitHub: [@githubhandle](https://github.com/githubhandle)
128 | - Twitter: [@twitterhandle](https://twitter.com/raihan2bd)
129 | - LinkedIn: [LinkedIn](https://linkedin.com/in/raihan2bd)
130 |
131 | 👤 **Salwa Ballouti**
132 |
133 | - GitHub: [@githubhandle](https://github.com/Salwa99)
134 | - Twitter: [@twitterhandle](https://twitter.com/salwa-ballouti)
135 | - LinkedIn: [LinkedIn](https://linkedin.com/in/salwa-ballouti)
136 |
137 | 👤 **Lugard Agu**
138 |
139 | - GitHub: [@githubhandle](https://github.com/lugard1)
140 | - Twitter: [@twitterhandle](https://twitter.com/Dsn3kings)
141 | - LinkedIn: [LinkedIn](https://www.linkedin.com/in/lugard-agu-45bb05b6/)
142 |
143 |
144 |
145 |
146 |
147 |
148 |
149 | ## How to access the database?
150 |
151 | - It's not available for now I'm still working on it, once it's available I'll update this answer.
152 |
153 |
154 |
155 | ## 🤝 Contributing
156 |
157 | Contributions, issues, and feature requests are welcome!
158 |
159 | Feel free to check the [issues page](../../issues/).
160 |
161 |
162 |
163 |
164 | ## ⭐️ Show your support
165 |
166 | > Give a ⭐️ if you like this project!
167 |
168 |
169 |
170 |
171 |
172 | ## 🙏 Acknowledgments
173 |
174 | > Without Microverse Help this project not compleated at all so thanks a lot Microverse for giving us this kind of opertunity.
175 |
176 |
177 |
178 |
179 |
180 |
181 | ## 📝 License
182 |
183 | This project is [MIT](./LICENSE) licensed.
184 |
185 |
186 |
--------------------------------------------------------------------------------
/src/modules/displayItems.js:
--------------------------------------------------------------------------------
1 | import { itemCountContainer, listItemsContainer } from './domSelector.js';
2 | import fetchSingleShowComment from './popupComment.js';
3 | import fetchSingleShow from './popupReservation.js';
4 | import { BASE_URL, MOVIE_API } from './apiUrls.js';
5 | import itemCounter from './itemCounter.js';
6 |
7 | // createNew like
8 | const createNewLike = async (id, likeCount, btnLike) => {
9 | btnLike.setAttribute('disabled', '');
10 | const response = await fetch(`${BASE_URL}/likes`, {
11 | method: 'POST',
12 | headers: {
13 | 'Content-Type': 'application/json',
14 | },
15 | body: JSON.stringify({ item_id: id }),
16 | });
17 |
18 | if (!response.ok && response.status !== 201) {
19 | return;
20 | }
21 |
22 | const responseLikes = await fetch(`${BASE_URL}/likes`);
23 | const result = await responseLikes.json();
24 | const likeData = result.find((item) => item.item_id === id);
25 | if (likeData) {
26 | likeCount.innerText = likeData.likes > 1 ? `${likeData.likes} likes` : `${likeData.likes} like`;
27 | btnLike.innerHTML = `
28 |
29 | `;
36 | }
37 | };
38 |
39 | // render the item list
40 | const render = (data) => {
41 | if (data.length > 0) {
42 | listItemsContainer.innerHTML = '';
43 |
44 | data.forEach((i) => {
45 | const item = document.createElement('li');
46 | item.id = i.id;
47 | item.className = 'show-item';
48 |
49 | // create sho Img element
50 | const showImg = document.createElement('div');
51 | showImg.className = 'show-item-img';
52 | showImg.innerHTML = `No Data Found
'; 128 | } 129 | }; 130 | 131 | const fetchTvShows = async () => { 132 | const response = await fetch(MOVIE_API); 133 | const result = await response.json(); 134 | 135 | // call the Involment api to get likes 136 | const responseInvolvement = await fetch(`${BASE_URL}/likes/`); 137 | const likesResult = await responseInvolvement.json(); 138 | 139 | // Distribute likes with correct array. 140 | let joinArr = []; 141 | if (!(likesResult.length <= 0)) { 142 | joinArr = result.map((movieItem) => { 143 | let likes = 0; 144 | const findLikes = likesResult.find((likeItem) => likeItem.item_id === movieItem.id); 145 | if (findLikes) { 146 | likes = findLikes.likes; 147 | } 148 | return { ...movieItem, likes }; 149 | }); 150 | } 151 | 152 | // call render function to display the item list 153 | render(joinArr); 154 | }; 155 | 156 | export default fetchTvShows; 157 | -------------------------------------------------------------------------------- /src/modules/popupComment.js: -------------------------------------------------------------------------------- 1 | import { BASE_URL } from './apiUrls.js'; 2 | import commentCounter from './commentCounter.js'; 3 | import { modalContainer } from './domSelector.js'; 4 | 5 | const createNewComment = async (url, data, commentsContainer, Form) => { 6 | const response = await fetch(url, { 7 | method: 'POST', 8 | headers: { 9 | 'Content-Type': 'application/json', 10 | }, 11 | body: JSON.stringify(data), 12 | }); 13 | 14 | if (!response.ok && response.status !== 201) { 15 | return; 16 | } 17 | 18 | const fetchComUrl = `${BASE_URL}/comments?item_id=${data.item_id}`; 19 | 20 | const commentResponse = await fetch(fetchComUrl); 21 | 22 | const result = await commentResponse.json(); 23 | 24 | // manupulate the dom 25 | const commentTite = document.createElement('div'); 26 | commentTite.className = 'comment-title'; 27 | const commentHeader = document.createElement('h3'); 28 | commentHeader.className = 'comment-header'; 29 | commentHeader.innerHTML = 'Comments'; 30 | const commentCount = document.createElement('p'); 31 | commentCount.id = 'show_comment_count'; 32 | commentCount.innerText = 0; 33 | 34 | commentTite.append(commentHeader, commentCount); 35 | 36 | const commentGroup = document.createElement('ul'); 37 | commentGroup.className = 'comment-goup'; 38 | 39 | let commentItems = ''; 40 | 41 | if (result.length > 0) { 42 | result.forEach((item) => { 43 | commentItems += `${item.comment}
52 |${item.comment}
139 |").concat(t.comment,"
\n").concat(t.comment,"
\nNo Data Found
';case 21:case"end":return t.stop()}var p,m,h}),t)})));return function(){return t.apply(this,arguments)}}();r.addEventListener("click",(function(){a.classList.toggle("display-flex")})),o.addEventListener("click",(function(){a.classList.remove("display-flex")})),i.forEach((function(t){t.addEventListener("click",(function(){a.classList.remove("display-flex")}))})),window.onload=function(){Y()}}},t=>{t(t.s=963)}]); 3 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.bundle.js","mappings":"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","sources":["webpack://js-capstone/./src/modules/domSelector.js","webpack://js-capstone/./src/modules/apiUrls.js","webpack://js-capstone/./src/modules/commentCounter.js","webpack://js-capstone/./src/modules/popupComment.js","webpack://js-capstone/./src/modules/reservationCounter.js","webpack://js-capstone/./src/modules/popupReservation.js","webpack://js-capstone/./src/modules/displayItems.js","webpack://js-capstone/./src/modules/itemCounter.js","webpack://js-capstone/./src/index.js"],"sourcesContent":["export const menuBtn = document.getElementById('mobile-menu');\nexport const closeBtn = document.getElementById('cross-menu');\nexport const mobileNavlinks = document.querySelectorAll(\n  '.mobile-nav-group .nav-link',\n);\nexport const mobNavGroup = document.querySelector('.mobile-nav-group');\nexport const modalContainer = document.querySelector('.popup-modal');\nexport const listItemsContainer = document.querySelector('.show-group');\nexport const itemCountContainer = document.getElementById('count_item');\n","export const BASE_URL = 'https://us-central1-involvement-api.cloudfunctions.net/capstoneApi/apps/Y1Ocl2k5LoJdVEhHia5O';\nexport const MOVIE_API = 'https://api.tvmaze.com/seasons/1/episodes';\n","const commentCounter = (countContainer, commentContainer) => {\n  const count = commentContainer.childElementCount;\n  countContainer.innerText = `(${count})`;\n  return count;\n};\n\nexport default commentCounter;\n","import { BASE_URL } from './apiUrls.js';\nimport commentCounter from './commentCounter.js';\nimport { modalContainer } from './domSelector.js';\n\nconst createNewComment = async (url, data, commentsContainer, Form) => {\n  const response = await fetch(url, {\n    method: 'POST',\n    headers: {\n      'Content-Type': 'application/json',\n    },\n    body: JSON.stringify(data),\n  });\n\n  if (!response.ok && response.status !== 201) {\n    return;\n  }\n\n  const fetchComUrl = `${BASE_URL}/comments?item_id=${data.item_id}`;\n\n  const commentResponse = await fetch(fetchComUrl);\n\n  const result = await commentResponse.json();\n\n  // manupulate the dom\n  const commentTite = document.createElement('div');\n  commentTite.className = 'comment-title';\n  const commentHeader = document.createElement('h3');\n  commentHeader.className = 'comment-header';\n  commentHeader.innerHTML = 'Comments';\n  const commentCount = document.createElement('p');\n  commentCount.id = 'show_comment_count';\n  commentCount.innerText = 0;\n\n  commentTite.append(commentHeader, commentCount);\n\n  const commentGroup = document.createElement('ul');\n  commentGroup.className = 'comment-goup';\n\n  let commentItems = '';\n\n  if (result.length > 0) {\n    result.forEach((item) => {\n      commentItems += `<li class='comment-item'>\n      <div class='user-info'>\n        <div class='user-avatar'>\n          <img src='./assets/img/user.png' alt='${item.username}'/>\n        </div>\n        <h4 class='user-name'>${item.username}</h4>\n        <span class='submit-date'>${item.creation_date}</span>\n      </div>\n      <p class='user-data'>${item.comment}</p>\n    </li>`;\n    });\n  }\n\n  commentGroup.innerHTML = commentItems; // append comment list\n  commentsContainer.innerHTML = '';\n\n  commentCounter(commentCount, commentGroup);\n\n  commentsContainer.append(commentTite, commentGroup);\n\n  Form.elements.name.value = '';\n  Form.elements.insight.value = '';\n};\n\nconst render = (data) => {\n  // modal card\n  const modalCard = document.createElement('div');\n  modalCard.className = 'modal-card';\n\n  // button\n  const btnCross = document.createElement('button');\n  btnCross.className = 'btn-cross';\n  btnCross.innerHTML = '<img src=\"./assets/img/icons8-multiply-24.png\" alt=\"X\" />';\n  btnCross.addEventListener('click', () => {\n    modalContainer.classList.remove('show');\n  });\n\n  // card img\n  const cardImg = document.createElement('div');\n  cardImg.className = 'card-img';\n  cardImg.innerHTML = `<img\n  src=\"${data.image.original}\"\n  alt=\"${data.name}\"\n/>`;\n\n  // card title\n  const title = document.createElement('h2');\n  title.className = 'card-title';\n  title.innerText = data.name;\n  // card des\n  const des = document.createElement('div');\n  des.className = 'card-des';\n  des.innerHTML = data.summary;\n\n  // card spec group\n  const cardSpec = document.createElement('div');\n  cardSpec.className = 'card-spec-group';\n  cardSpec.innerHTML = `<ul class=\"card-spec\">\n  <li class=\"spec-item\">Season: ${data.season}</li>\n  <li class=\"spec-item\">Duration: ${data.runtime}</li>\n</ul>\n<ul class=\"card-spec\">\n  <li class=\"spec-item\">airdate: ${data.airdate}</li>\n  <li class=\"spec-item\">Rating: ${data.rating.average}</li>\n</ul>`;\n\n  const commentSection = document.createElement('div');\n  commentSection.className = 'comment-section';\n\n  const commentTite = document.createElement('div');\n  commentTite.className = 'comment-title';\n  const commentHeader = document.createElement('h3');\n  commentHeader.className = 'comment-header';\n  commentHeader.innerHTML = 'Comments';\n  const commentCount = document.createElement('p');\n  commentCount.id = 'show_comment_count';\n  commentCount.innerText = 0;\n\n  commentTite.append(commentHeader, commentCount);\n\n  const commentGroup = document.createElement('ul');\n  commentGroup.className = 'comment-goup';\n\n  let commentItems = '';\n\n  if (data.comments.length > 0) {\n    data.comments.forEach((item) => {\n      commentItems += `<li class='comment-item'>\n        <div class='user-info'>\n          <div class='user-avatar'>\n            <img src='./assets/img/user.png' alt='${item.username}'/>\n          </div>\n          <h4 class='user-name'>${item.username}</h4>\n          <span class='submit-date'>${item.creation_date}</span>\n        </div>\n        <p class='user-data'>${item.comment}</p>\n      </li>`;\n    });\n  }\n\n  commentGroup.innerHTML = commentItems; // append comment list\n\n  commentCounter(commentCount, commentGroup);\n\n  commentSection.append(commentTite, commentGroup);\n\n  // Add comment\n  const commentForm = document.createElement('div');\n  commentForm.className = 'comment-form';\n  const addComment = document.createElement('h3');\n  addComment.className = 'comment-title';\n  addComment.innerText = 'Add a comment';\n  const Form = document.createElement('form');\n  Form.innerHTML = `\n<input type=\"text\"  name=\"name\" id=\"name\" placeholder=\"Your name\" maxlength=\"10\" required>\n<textarea id=\"insight\" name=\"insight\" placeholder=\"Your insights\"></textarea>\n<button type=\"submit\" id=\"submitBtn\">Submit</button>`;\n\n  commentForm.append(addComment, Form);\n\n  Form.addEventListener('submit', async (e) => {\n    e.preventDefault();\n    const name = Form.elements.name.value;\n    const insight = Form.elements.insight.value;\n    const commentData = {\n      item_id: data.id,\n      username: name,\n      comment: insight,\n    };\n    const url = `${BASE_URL}/comments`;\n    createNewComment(url, commentData, commentSection, Form);\n  });\n  // append child elements in cardModal\n  modalCard.append(\n    btnCross,\n    cardImg,\n    title,\n    des,\n    cardSpec,\n    commentSection,\n    commentForm,\n  );\n\n  modalContainer.innerHTML = '';\n  modalContainer.append(modalCard); // append cardModal\n};\n\nconst fetchSingleShowComment = async (e) => {\n  modalContainer.classList.add('show');\n  const { id } = e.target.parentElement.parentElement;\n  const url = `https://api.tvmaze.com/episodes/${id}`;\n  const response = await fetch(url);\n  const result = await response.json();\n\n  const commentResponse = await fetch(`${BASE_URL}/comments?item_id=${id}`);\n\n  let filterResult = { ...result };\n\n  const commentResult = await commentResponse.json();\n  if (commentResult && !commentResult.error) {\n    filterResult = { ...filterResult, comments: commentResult };\n  } else {\n    filterResult = { ...filterResult, comments: [] };\n  }\n\n  render(filterResult);\n};\n\nexport default fetchSingleShowComment;\n","const reservationCounter = (countContainer, reservationContainer) => {\n  const count = reservationContainer.childElementCount;\n  countContainer.innerText = `(${count})`;\n  return count;\n};\n\nexport default reservationCounter;\n","import { BASE_URL } from './apiUrls.js';\nimport { modalContainer } from './domSelector.js';\nimport reservationCounter from './reservationCounter.js';\n\nconst createNewReservation = async (url, data, reservationContainer, Form) => {\n  const response = await fetch(url, {\n    method: 'POST',\n    headers: {\n      'Content-Type': 'application/json',\n    },\n    body: JSON.stringify(data),\n  });\n\n  if (!response.ok && response.status !== 201) {\n    return;\n  }\n\n  const fetchComUrl = `${BASE_URL}/reservations?item_id=${data.item_id}`;\n\n  const reservationResponse = await fetch(fetchComUrl);\n\n  const result = await reservationResponse.json();\n\n  // manupulate the dom\n  const reservationHeader = document.createElement('h3');\n  reservationHeader.className = 'reservation-title';\n  reservationHeader.innerHTML = \"Reservations <span id='show_comment_count'>(0)</span>\";\n\n  const reservationGroup = document.createElement('ul');\n  reservationGroup.className = 'reservation-group';\n\n  let reservationItems = '';\n\n  if (result.length > 0) {\n    result.forEach((item) => {\n      reservationItems += `<li class='reservation-item'>\n      <div class='user-info'>\n        <div class='user-avatar'>\n          <img src='./assets/img/user.png' alt='${item.username}'/>\n        </div>\n        <h4 class='user-name'>${item.username}</h4>\n        <span class='submit-date'>${item.date_start} - ${item.date_end}</span>\n      </div>\n    </li>`;\n    });\n  }\n\n  reservationGroup.innerHTML = reservationItems; // append reservation list\n  reservationContainer.innerHTML = '';\n\n  reservationCounter(reservationHeader.children[0], reservationGroup);\n\n  reservationContainer.append(reservationHeader, reservationGroup);\n\n  Form.elements.name.value = '';\n  Form.elements.start_date.value = '';\n  Form.elements.end_date.value = '';\n};\n\nconst render = (data) => {\n  // modal card\n  const modalCard = document.createElement('div');\n  modalCard.className = 'modal-card';\n\n  // button\n  const btnCross = document.createElement('button');\n  btnCross.className = 'btn-cross';\n  btnCross.innerHTML = '<img src=\"./assets/img/icons8-multiply-24.png\" alt=\"X\" />';\n  btnCross.addEventListener('click', () => {\n    modalContainer.classList.remove('show');\n  });\n\n  // card img\n  const cardImg = document.createElement('div');\n  cardImg.className = 'card-img';\n  cardImg.innerHTML = `<img\n  src=\"${data.image.original}\"\n  alt=\"${data.name}\"\n/>`;\n\n  // card title\n  const title = document.createElement('h2');\n  title.className = 'card-title';\n  title.innerText = data.name;\n\n  // card des\n  const des = document.createElement('div');\n  des.className = 'card-des';\n  des.innerHTML = data.summary;\n\n  // card spec group\n  const cardSpec = document.createElement('div');\n  cardSpec.className = 'card-spec-group';\n  cardSpec.innerHTML = `<ul class=\"card-spec\">\n  <li class=\"spec-item\">Season: ${data.season}</li>\n  <li class=\"spec-item\">Duration: ${data.runtime}</li>\n</ul>\n<ul class=\"card-spec\">\n  <li class=\"spec-item\">airdate: ${data.airdate}</li>\n  <li class=\"spec-item\">Rating: ${data.rating.average}</li>\n</ul>`;\n\n  const reservationSection = document.createElement('div');\n  reservationSection.className = 'reservation-section';\n\n  const reservationHeader = document.createElement('h3');\n  reservationHeader.className = 'reservation-title';\n  reservationHeader.innerHTML = \"Reservations <span id='show_comment_count'>(0)</span>\";\n\n  const reservationGroup = document.createElement('ul');\n  reservationGroup.className = 'reservation-group';\n\n  let reservationItems = '';\n\n  if (data.reservations.length > 0) {\n    data.reservations.forEach((item) => {\n      reservationItems += `<li class='reservation-item'>\n      <div class='user-info'>\n        <div class='user-avatar'>\n          <img src='./assets/img/user.png' alt='${item.username}'/>\n        </div>\n        <h4 class='user-name'>${item.username}</h4>\n        <span class='submit-date'>${item.date_start} - ${item.date_end}</span>\n      </div>\n    </li>`;\n    });\n  }\n\n  reservationGroup.innerHTML = reservationItems; // append reservation list\n\n  reservationCounter(reservationHeader.children[0], reservationGroup);\n\n  reservationSection.append(reservationHeader, reservationGroup);\n\n  // Add comment\n  const reservationForm = document.createElement('div');\n  reservationForm.className = 'reservation-form';\n  const addReservation = document.createElement('h3');\n  addReservation.className = 'reservation-title';\n  addReservation.innerText = 'Add a Reservation';\n  const Form = document.createElement('form');\n  Form.innerHTML = `\n  <input type=\"text\"  name=\"name\" id=\"name\" placeholder=\"Your name\" maxlength=\"10\" required>\n  <label for='start_date'>Add your start date below</label>\n  <input type=\"date\"  name=\"start_date\" id=\"start_date\" placeholder=\"Add your start date here.\" required>\n  <label for='end_date'>Add your end date below</label>\n  <input type=\"date\"  name=\"end_date\" id=\"end_date\" placeholder=\"Add your end date here.\" required>\n  <button type=\"submit\" id=\"submitBtn\">Reserve</button>`;\n\n  reservationForm.append(addReservation, Form);\n\n  Form.addEventListener('submit', async (e) => {\n    e.preventDefault();\n    const name = Form.elements.name.value;\n    const startDate = Form.elements.start_date.value;\n    const endDate = Form.elements.end_date.value;\n    const reserveData = {\n      item_id: data.id,\n      username: name,\n      date_start: startDate,\n      date_end: endDate,\n    };\n    const url = `${BASE_URL}/reservations/`;\n    createNewReservation(url, reserveData, reservationSection, Form);\n  });\n\n  // append child elements in cardModal\n  // eslint-disable-next-line max-len\n  modalCard.append(\n    btnCross,\n    cardImg,\n    title,\n    des,\n    cardSpec,\n    reservationSection,\n    reservationForm,\n  );\n\n  modalContainer.innerHTML = '';\n  modalContainer.append(modalCard); // append cardModal\n};\n\nconst fetchSingleShow = async (e) => {\n  modalContainer.classList.add('show');\n  const { id } = e.target.parentElement.parentElement;\n  const url = `https://api.tvmaze.com/episodes/${id}`;\n  const response = await fetch(url);\n  const result = await response.json();\n\n  const reservationResponse = await fetch(\n    `${BASE_URL}/reservations?item_id=${id}`,\n  );\n\n  let filterResult = { ...result };\n\n  const reservationResult = await reservationResponse.json();\n  if (reservationResult && !reservationResult.error) {\n    filterResult = { ...filterResult, reservations: reservationResult };\n  } else {\n    filterResult = { ...filterResult, reservations: [] };\n  }\n\n  render(filterResult);\n};\n\nexport default fetchSingleShow;\n","import { itemCountContainer, listItemsContainer } from './domSelector.js';\nimport fetchSingleShowComment from './popupComment.js';\nimport fetchSingleShow from './popupReservation.js';\nimport { BASE_URL, MOVIE_API } from './apiUrls.js';\nimport itemCounter from './itemCounter.js';\n\n// createNew like\nconst createNewLike = async (id, likeCount, btnLike) => {\n  btnLike.setAttribute('disabled', '');\n  const response = await fetch(`${BASE_URL}/likes`, {\n    method: 'POST',\n    headers: {\n      'Content-Type': 'application/json',\n    },\n    body: JSON.stringify({ item_id: id }),\n  });\n\n  if (!response.ok && response.status !== 201) {\n    return;\n  }\n\n  const responseLikes = await fetch(`${BASE_URL}/likes`);\n  const result = await responseLikes.json();\n  const likeData = result.find((item) => item.item_id === id);\n  if (likeData) {\n    likeCount.innerText = likeData.likes > 1 ? `${likeData.likes} likes` : `${likeData.likes} like`;\n    btnLike.innerHTML = `<?xml version=\"1.0\" encoding=\"iso-8859-1\"?>\n    <!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->\n    <svg fill=\"#ff7b00\" height=\"22px\" width=\"22px\" version=\"1.1\" id=\"Layer_1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" \n       viewBox=\"0 0 455 455\" xml:space=\"preserve\">\n    <path d=\"M326.632,10.346c-38.733,0-74.991,17.537-99.132,46.92c-24.141-29.383-60.399-46.92-99.132-46.92\n      C57.586,10.346,0,67.931,0,138.714c0,55.426,33.049,119.535,98.23,190.546c50.162,54.649,104.729,96.96,120.257,108.626l9.01,6.769\n      l9.009-6.768c15.53-11.667,70.099-53.979,120.26-108.625C421.95,258.251,455,194.141,455,138.714\n      C455,67.931,397.414,10.346,326.632,10.346z\"/>\n    </svg>`;\n  }\n};\n\n// render the item list\nconst render = (data) => {\n  if (data.length > 0) {\n    listItemsContainer.innerHTML = '';\n\n    data.forEach((i) => {\n      const item = document.createElement('li');\n      item.id = i.id;\n      item.className = 'show-item';\n\n      // create sho Img element\n      const showImg = document.createElement('div');\n      showImg.className = 'show-item-img';\n      showImg.innerHTML = `<img src='${i.image.original}'>`;\n\n      // create show info\n      const showInfo = document.createElement('div');\n      showInfo.className = 'show-info';\n\n      // create tile for show info\n      const title = document.createElement('h3');\n      title.className = 'show-title';\n      title.innerText = i.name;\n\n      // like action\n      const showLikeAction = document.createElement('div');\n      showLikeAction.className = 'shwo-like-action';\n\n      // like button\n      const btnLike = document.createElement('button');\n      btnLike.className = 'btn-like';\n      btnLike.innerHTML = `<?xml version=\"1.0\" encoding=\"iso-8859-1\"?>\n      <!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->\n      <svg fill=\"#ff7b00\" height=\"22px\" width=\"22px\" version=\"1.1\" id=\"Layer_1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" \n         viewBox=\"0 0 455 455\" xml:space=\"preserve\">\n      <path d=\"M326.632,10.346c-38.733,0-74.991,17.537-99.132,46.92c-24.141-29.384-60.398-46.92-99.132-46.92\n        C57.586,10.346,0,67.931,0,138.714c0,55.426,33.05,119.535,98.23,190.546c50.161,54.647,104.728,96.959,120.257,108.626l9.01,6.769\n        l9.01-6.768c15.529-11.667,70.098-53.978,120.26-108.625C421.949,258.251,455,194.141,455,138.714\n        C455,67.931,397.414,10.346,326.632,10.346z M334.666,308.974c-41.259,44.948-85.648,81.283-107.169,98.029\n        c-21.52-16.746-65.907-53.082-107.166-98.03C61.236,244.592,30,185.717,30,138.714c0-54.24,44.128-98.368,98.368-98.368\n        c35.694,0,68.652,19.454,86.013,50.771l13.119,23.666l13.119-23.666c17.36-31.316,50.318-50.771,86.013-50.771\n        c54.24,0,98.368,44.127,98.368,98.368C425,185.719,393.763,244.594,334.666,308.974z\"/>\n      </svg>`;\n\n      // like count\n      const likeCount = document.createElement('span');\n      if (i.likes > 1) {\n        likeCount.innerText = `${i.likes} likes`;\n      } else {\n        likeCount.innerHTML = `${i.likes} like`;\n      }\n\n      // btn like event to create new like\n      btnLike.addEventListener('click', () => {\n        createNewLike(i.id, likeCount, btnLike);\n      });\n\n      showLikeAction.append(btnLike, likeCount); // append like actions child element.\n\n      showInfo.append(title, showLikeAction); // append in showInfo\n\n      // show action\n      const showActions = document.createElement('div');\n      showActions.className = 'show-actions';\n\n      // create child btn\n      const commentBtn = document.createElement('button');\n      commentBtn.className = 'btn-action btn-comment';\n      commentBtn.innerText = 'Comments';\n      commentBtn.addEventListener('click', (e) => {\n        fetchSingleShowComment(e);\n      });\n\n      const reservationBtn = document.createElement('button');\n      reservationBtn.className = 'btn-action btn-reservation';\n      reservationBtn.innerText = 'Reservations';\n      reservationBtn.addEventListener('click', (e) => {\n        fetchSingleShow(e);\n      });\n\n      showActions.append(commentBtn, reservationBtn); // append child action buttons in showActions\n\n      item.append(showImg, showInfo, showActions); // append clild all the elements in item.\n\n      listItemsContainer.appendChild(item);\n    });\n    itemCounter(itemCountContainer, listItemsContainer);\n  } else {\n    listItemsContainer.innerHTML = '<p class=\"no-data\">No Data Found</p>';\n  }\n};\n\nconst fetchTvShows = async () => {\n  const response = await fetch(MOVIE_API);\n  const result = await response.json();\n\n  // call the Involment api to get likes\n  const responseInvolvement = await fetch(`${BASE_URL}/likes/`);\n  const likesResult = await responseInvolvement.json();\n\n  // filter Array thats have Likes\n  const filterArrWithLikes = [];\n  result.forEach((item) => {\n    likesResult.forEach((likeItem) => {\n      if (item.id === likeItem.item_id) {\n        filterArrWithLikes.push({ ...item, likes: likeItem.likes });\n      }\n    });\n  });\n\n  // filter Array thats have not likes\n  let filterWithoutLikes = [];\n  filterWithoutLikes = result.filter(\n    (el) => !filterArrWithLikes.find((element) => element.id === el.id),\n  ); //eslint-disable-line\n\n  // modify the filterWithout array likes count 0;\n  const modifiyFilterWithoutLikes = [];\n  filterWithoutLikes.forEach((item) => {\n    modifiyFilterWithoutLikes.push({ ...item, likes: 0 });\n  });\n\n  // join and sort the arrays\n  const joinArr = modifiyFilterWithoutLikes.concat(filterArrWithLikes);\n  joinArr.sort((a, b) => a.id - b.id);\n\n  // call render function to display the item list\n  render(joinArr);\n};\n\nexport default fetchTvShows;\n","const itemCounter = (countContainer, itemContainer) => {\n  const count = itemContainer.childElementCount;\n  countContainer.innerText = `(${count})`;\n  return count;\n};\n\nexport default itemCounter;\n","// import static files\nimport './style.css';\nimport './assets/img/icons8-menu-rounded-24.png';\nimport './assets/img/icons8-multiply-24.png';\nimport './assets/img/user.png';\n\nimport fetchTvShow from './modules/displayItems.js';\nimport {\n  closeBtn,\n  menuBtn,\n  mobNavGroup,\n  mobileNavlinks,\n} from './modules/domSelector.js';\n\n// toggle the menu\nmenuBtn.addEventListener('click', () => {\n  mobNavGroup.classList.toggle('display-flex');\n});\n\ncloseBtn.addEventListener('click', () => {\n  mobNavGroup.classList.remove('display-flex');\n});\n\nmobileNavlinks.forEach((item) => {\n  item.addEventListener('click', () => {\n    mobNavGroup.classList.remove('display-flex');\n  });\n});\n\n// load the item list on the fly.\nwindow.onload = () => {\n  fetchTvShow();\n};\n"],"names":["menuBtn","document","getElementById","closeBtn","mobileNavlinks","querySelectorAll","mobNavGroup","querySelector","modalContainer","listItemsContainer","itemCountContainer","BASE_URL","countContainer","commentContainer","count","childElementCount","innerText","i","createNewComment","url","data","commentsContainer","Form","fetch","method","headers","body","JSON","stringify","response","ok","status","fetchComUrl","item_id","commentResponse","json","result","commentTite","createElement","className","commentHeader","innerHTML","commentCount","id","append","commentGroup","commentItems","length","forEach","item","username","creation_date","comment","commentCounter","elements","name","value","insight","render","modalCard","btnCross","addEventListener","classList","remove","cardImg","image","original","title","des","summary","cardSpec","season","runtime","airdate","rating","average","commentSection","comments","commentForm","addComment","e","preventDefault","commentData","add","target","parentElement","filterResult","commentResult","error","reservationContainer","createNewReservation","reservationResponse","reservationHeader","reservationGroup","reservationItems","date_start","date_end","reservationCounter","children","start_date","end_date","reservationSection","reservations","reservationForm","addReservation","startDate","endDate","reserveData","reservationResult","createNewLike","likeCount","btnLike","setAttribute","responseLikes","likeData","find","likes","responseInvolvement","likesResult","filterArrWithLikes","likeItem","push","filterWithoutLikes","filter","el","element","modifiyFilterWithoutLikes","joinArr","concat","sort","a","b","showImg","showInfo","showLikeAction","showActions","commentBtn","fetchSingleShowComment","reservationBtn","fetchSingleShow","appendChild","toggle","window","onload","fetchTvShow"],"sourceRoot":""} --------------------------------------------------------------------------------