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

Feature/article list #92

Merged
merged 15 commits into from
Jan 22, 2024
55 changes: 54 additions & 1 deletion cigaradvisor/blocks/article-list/article-list.css
Original file line number Diff line number Diff line change
@@ -1 +1,54 @@
/** Dummy File for now **/
@import url('../article-teaser/article-teaser.css');
bstopp marked this conversation as resolved.
Show resolved Hide resolved

.article-list.block .pagination-container {
display: flex;
justify-content: center;
align-items: center;
padding-top: 30px;

}

.article-list.block .pagination {
display: flex;
padding-left: 0;
margin: 20px 0;
border-radius: 4px;
}

.article-list.block .pagination li {
display: inline;
}

.article-list.block .pagination a {
border: .5px solid var(--tan);
margin: auto 2px;
font-size: 15px;
bstopp marked this conversation as resolved.
Show resolved Hide resolved
min-width: 40px;
display: flex;
justify-content: center;
align-items: center;
min-height: 40px;
font-weight: 400;
position: relative;
float: left;
padding: 6px 12px;
line-height: 1.4286;
text-decoration: none;
background-color: var(--clr-white);

}

.article-list.block .pagination a:hover {
background-color: var(--tan);
border-color: var(--tan);
color: var(--clr-white);
}

.article-list.block .pagination .gap:hover {
background-color: var(--neutral-grey);
}

.article-list.block .pagination a.active {
background-color: var(--tan);
color: var(--clr-white);;
}
137 changes: 136 additions & 1 deletion cigaradvisor/blocks/article-list/article-list.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,138 @@
import { readBlockConfig } from '../../scripts/aem.js';
import { fetchData, getRelativePath } from '../../scripts/scripts.js';
import { buildArticleTeaser } from '../article-teaser/article-teaser.js';

// Function to create ellipsis
function createEllipsis() {
const listItem = document.createElement('li');
const a = document.createElement('a');
const span = document.createElement('span');
a.className = 'gap';
span.textContent = '...';
a.appendChild(span);
listItem.appendChild(a);
return listItem;
}

// Function to create a page link
function createPageLink(pageNumber, text, className) {
const listItem = document.createElement('li');
const link = document.createElement('a');
const currentPagePath = window.location.pathname;
link.href = `${currentPagePath}?page=${pageNumber}`;
link.textContent = text;

if (className) {
link.classList.add(className);
}

listItem.appendChild(link);
return listItem;
}

function generatePagination(currentPage, totalPages) {
const displayPages = 7;
const paginationList = document.createElement('ol');
paginationList.className = 'pagination';

// Previous page link
if (currentPage > 1) {
paginationList.appendChild(createPageLink(currentPage - 1, '«', 'prev'));
}

// Page links
const startPage = Math.max(1, currentPage - Math.floor(displayPages / 2));
const endPage = Math.min(totalPages, startPage + displayPages - 1);

if (startPage > 1) {
paginationList.appendChild(createPageLink(1, '1'));
if (startPage > 2) {
paginationList.appendChild(createEllipsis());
}
}

for (let i = startPage; i <= endPage; i += 1) {
if (i === currentPage) {
paginationList.appendChild(createPageLink(i, i, 'active'));
} else {
paginationList.appendChild(createPageLink(i, i));
}
}

if (endPage < totalPages) {
if (endPage < totalPages - 1) {
paginationList.appendChild(createEllipsis());
}
paginationList.appendChild(createPageLink(totalPages, totalPages));
}

// Next page link
if (currentPage < totalPages) {
paginationList.appendChild(createPageLink(currentPage + 1, '»', 'next'));
}
return paginationList;
}

export default async function decorate(block) {
block.innerHTML = '';
const configs = readBlockConfig(block);
const { category } = configs;
const { author } = configs;
const { limit } = configs;
const parentDiv = document.createElement('div');
parentDiv.classList.add('section');
parentDiv.dataset.layout = '50/50';
const leftDiv = document.createElement('div');
leftDiv.classList.add('left-grid');
const rightDiv = document.createElement('div');
rightDiv.classList.add('right-grid');
let current = rightDiv;
let currentPage = 1;
if (category || author) {
let articles;
if (category) {
articles = await fetchData(getRelativePath(category), '/cigaradvisor/drafts/Kailas/query-index.json', 'category');
} else if (author) {
articles = await fetchData(getRelativePath(author), '/cigaradvisor/drafts/Kailas/query-index.json', 'author');
}
const totalArticles = articles.length;
const totalPages = Math.ceil(totalArticles / limit);
let categoryInfo;
let authorInfo;
if (category) {
categoryInfo = await fetchData(getRelativePath(category), '/cigaradvisor/drafts/Kailas/query-index.json');
}
if (author) {
authorInfo = await fetchData(getRelativePath(author), '/cigaradvisor/author/query-index.json');
}
if (!articles || articles.length === 0) {
return;
}
const urlParams = new URLSearchParams(window.location.search);
currentPage = urlParams.get('page') ? parseInt(urlParams.get('page'), 10) : 1;
// eslint-disable-next-line max-len
const articlePromises = [...articles].slice((currentPage - 1) * limit, currentPage * limit).map(async (article) => {
const articletTeaserWrapper = document.createElement('div');
articletTeaserWrapper.classList.add('article-teaser-wrapper');
const articleTeaser = document.createElement('div');
articleTeaser.classList.add('article-teaser');
articleTeaser.classList.add('block');
articletTeaserWrapper.append(articleTeaser);
current = (current === leftDiv) ? rightDiv : leftDiv;
current.append(articletTeaserWrapper);
if (categoryInfo) {
await buildArticleTeaser(articleTeaser, article, categoryInfo[0]);
} else if (authorInfo) {
await buildArticleTeaser(articleTeaser, article, null, authorInfo[0]);
}
});
await Promise.all(articlePromises);

parentDiv.append(leftDiv);
parentDiv.append(rightDiv);
block.replaceChildren(parentDiv);
const pageinationContainer = document.createElement('div');
pageinationContainer.classList.add('pagination-container');
pageinationContainer.appendChild(generatePagination(currentPage, totalPages));
block.append(pageinationContainer);
}
}
44 changes: 31 additions & 13 deletions cigaradvisor/blocks/article-teaser/article-teaser.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,21 +13,28 @@ function formatDate(originalDateString) {
return `${formattedDate}|${dateTimeAttribute}`;
}

export default async function decorate(block) {
const filterPath = block.querySelector('a').getAttribute('href');
block.classList.add('article-teaser');
const articleInfo = await fetchData(getRelativePath(filterPath), '/cigaradvisor/posts/query-index.json');
if (!articleInfo) {
return;
// eslint-disable-next-line max-len
export async function buildArticleTeaser(parentElement, articleInfo, articleCategoryInfo, articleAuthorInfo) {
bstopp marked this conversation as resolved.
Show resolved Hide resolved
let categoryInfo = articleCategoryInfo;
let authorInfo = articleAuthorInfo;
if (!articleCategoryInfo) {
const articleCategoryLink = articleInfo.category;
const fetchedCategoryInfo = await fetchData(getRelativePath(articleCategoryLink), '/cigaradvisor/query-index.json');
if (fetchedCategoryInfo && fetchedCategoryInfo.length !== 0) {
[categoryInfo] = fetchedCategoryInfo;
}
}
if (!articleAuthorInfo) {
const articleAuthorLink = articleInfo.author;
const fetchedAuthorInfo = await fetchData(getRelativePath(articleAuthorLink), '/cigaradvisor/author/query-index.json');
if (fetchedAuthorInfo && fetchedAuthorInfo.length !== 0) {
[authorInfo] = fetchedAuthorInfo;
}
}
const articleCategoryLink = articleInfo.category;
const articleCategoryInfo = await fetchData(getRelativePath(articleCategoryLink));
const [formattedDate, datetimeAttr] = formatDate(articleInfo.published).split('|');
const articleAuthorLink = articleInfo.author;
const articleAuthorInfo = await fetchData(getRelativePath(articleAuthorLink), '/cigaradvisor/author/query-index.json');
block.innerHTML = `
parentElement.innerHTML += `
<article class="article article-thumbnail">
<a class="article-category" href="${articleCategoryLink}" data-category="${(articleCategoryInfo && articleCategoryInfo.title) ? articleCategoryInfo.title : ''}" title="${(articleCategoryInfo && articleCategoryInfo.title) ? articleCategoryInfo.title : ''}">${(articleCategoryInfo && articleCategoryInfo.title) ? articleCategoryInfo.title : ''}</a>
<a class="article-category" href="${categoryInfo ? categoryInfo.path : ''}" data-category="${(categoryInfo && categoryInfo.title) ? categoryInfo.title : ''}" title="${(categoryInfo && categoryInfo.title) ? categoryInfo.title : ''}">${(categoryInfo && categoryInfo.title) ? categoryInfo.title : ''}</a>
<div class="article-image">
${createOptimizedPicture(articleInfo.image).outerHTML}
</div>
Expand All @@ -37,7 +44,7 @@ export default async function decorate(block) {
<a class="article-title-link" href="${articleInfo.path}" title="${articleInfo.title}">${articleInfo.title}</a>
</h2>
<div class="article-meta">
<a class="article-authorLink" href="${articleAuthorLink}" title="By ${(articleAuthorInfo && articleAuthorInfo.name) ? articleAuthorInfo.name : ''}">By ${(articleAuthorInfo && articleAuthorInfo.name) ? articleAuthorInfo.name : ''}</a>
<a class="article-authorLink" href="${authorInfo ? authorInfo.path : ''}" title="By ${(authorInfo && authorInfo.name) ? authorInfo.name : ''}">By ${(authorInfo && authorInfo.name) ? authorInfo.name : ''}</a>
<time class="article-pubdate" datetime="${datetimeAttr}">${formattedDate}</time>
</div>
</articleheader>
Expand All @@ -51,3 +58,14 @@ export default async function decorate(block) {
</article>
`;
}

export default async function decorate(block) {
const filterPath = block.querySelector('a').getAttribute('href');
block.classList.add('article-teaser');
const articleInfo = await fetchData(getRelativePath(filterPath), '/cigaradvisor/posts/query-index.json');
if (!articleInfo || articleInfo.length === 0) {
return;
}
block.innerHTML = '';
buildArticleTeaser(block, articleInfo[0]);
}
12 changes: 6 additions & 6 deletions cigaradvisor/blocks/articleheader/articleheader.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,20 +9,20 @@ export default async function decorate(block) {
const articleInfo = document.createElement('div');
articleInfo.classList.add('article-info');
const categoryLink = block.querySelector('p.category').innerText;
const category = await fetchData(getRelativePath(categoryLink));
if (category) {
const category = await fetchData(getRelativePath(categoryLink), '/cigaradvisor/query-index.json');
if (category[0]) {
const categoryLinkEl = document.createElement('div');
categoryLinkEl.classList.add('article-category');
categoryLinkEl.innerHTML = `<a href="${categoryLink}">${category.title}</a>`;
categoryLinkEl.innerHTML = `<a href="${categoryLink}">${category[0].title}</a>`;
articleInfo.append(categoryLinkEl);
}
articleInfo.append(block.querySelector('h1'));
const authorLink = block.querySelector('p.author').innerText;
const author = await fetchData(getRelativePath(authorLink));
const author = await fetchData(getRelativePath(authorLink), '/cigaradvisor/author/query-index.json');
const authorLinkEl = document.createElement('div');
authorLinkEl.classList.add('article-author');
if (author) {
authorLinkEl.innerHTML = `<a href="${authorLink}">By ${author.title}</a>`;
if (author[0]) {
authorLinkEl.innerHTML = `<a href="${authorLink}">By ${author[0].title}</a>`;
articleInfo.append(authorLinkEl);
}
const publishedDate = block.querySelector('p.published-date').innerText;
Expand Down
6 changes: 3 additions & 3 deletions cigaradvisor/blocks/author-card/author-card.js
Original file line number Diff line number Diff line change
Expand Up @@ -35,12 +35,12 @@ export default async function decorate(block) {
authorWrapperSection.innerHTML = '';
const authorPromises = [...authors].map(async (authorPage) => {
const authorInfo = await fetchData(getRelativePath(authorPage), '/cigaradvisor/author/query-index.json');
if (authorInfo) {
if (authorInfo[0]) {
return `<div class="author-content">
<div class="overlay-image">
${createOptimizedPicture(authorInfo.image).outerHTML}
${createOptimizedPicture(authorInfo[0].image).outerHTML}
<div class="overlay-content">
<p class="align-center"><a href="${authorInfo.page}">${authorInfo.name}</a></p>
<p class="align-center"><a href="${authorInfo[0].path}">${authorInfo[0].name}</a></p>
</div>
</div>
</div>`;
Expand Down
18 changes: 9 additions & 9 deletions cigaradvisor/scripts/scripts.js
Original file line number Diff line number Diff line change
Expand Up @@ -206,17 +206,17 @@ export function getRelativePath(path) {
let indexData = '';
let authorIndexData = '';
let articleIndexData = '';

/**
* Fetches data from a specified URL and returns the filtered
* data based on the provided filter path.
* @param {string} filterPath - The path used to filter the data.
* @param {string} [fetchUrl='/query-index.json'] - The URL to fetch the data
* from. Defaults to '/query-index.json'.
* @returns {Promise<Object>} - A promise that resolves to the filtered data object.
* Fetches data based on the search value and fetch URL.
* @param {string} searchValue - The value to search for.
* @param {string} fetchUrl - The URL to fetch the data from. Default value is '/query-index.json'.
* @param {string} searchParam - The parameter to search for in the data. Default value is 'path'.
* @returns {Promise<Array>} - A promise that resolves to an array of filtered data.
*/
export async function fetchData(filterPath, fetchUrl = '/query-index.json') {
export async function fetchData(searchValue, fetchUrl = '/cigaradvisor/drafts/Kailas/query-index.json', searchParam = 'path') {
let responeData = '';
if (fetchUrl === '/query-index.json') {
if (fetchUrl === '/cigaradvisor/drafts/Kailas/query-index.json') {
if (!indexData) {
const resp = await fetch(fetchUrl);
let jsonData = '';
Expand Down Expand Up @@ -251,7 +251,7 @@ export async function fetchData(filterPath, fetchUrl = '/query-index.json') {
}
let filteredData = '';
if (responeData) {
filteredData = responeData.find((obj) => obj.path === filterPath);
filteredData = responeData.filter((obj) => obj[searchParam] === searchValue);
}
return filteredData;
}
Expand Down
7 changes: 4 additions & 3 deletions cigaradvisor/styles/styles.css
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
--very-light-grey: #ddd;
--vv-light-grey: #e0e0e0;
--medium-grey: #888;
--neutral-grey: #eee;
--grey: #333;
--light-grey: #d3d3d3;
--dark-grey: #3c3a3b;
Expand Down Expand Up @@ -445,15 +446,15 @@ main .button-container .button:hover {
font-size: 45px;
}

main .section {
main > .section {
margin-bottom: 40px;
}

main .section.background-grey {
main > .section.background-grey {
background-color: var(--grey-background);
}

main .section > .default-content-wrapper > h2 {
main > .section > .default-content-wrapper > h2 {
font-size: 45px;
}
}
Expand Down