gatsby-core-theme 45.0.0-beta.1 → 45.0.0-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,3 +1,19 @@
1
+ # [45.0.0-beta.2](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/compare/v45.0.0-beta.1...v45.0.0-beta.2) (2025-06-13)
2
+
3
+
4
+ ### Bug Fixes
5
+
6
+ * merge conflict bug ([2f72a23](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/2f72a2341858cd254daccb61d79b33c5beadeafd))
7
+ * merge conflicts ([6467656](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/6467656ec248e670f59b346fe9eb66f49025a609))
8
+ * phase one ([59040df](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/59040df4a9acabb7ab079123421eddddadb08e62))
9
+ * phase three - testing ([7f320d1](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/7f320d181c6b4f903d7867c62bf21662ac91c7ac))
10
+ * phase two ([b4562f3](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/b4562f36fc376a5bb6a1446a868e2de265a6b244))
11
+
12
+
13
+ * Merge branch 'tm-5493-user-comments' into 'beta' ([bd6381e](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/bd6381e86bba4ea84439e01a01e702ba8bfdee7e))
14
+ * Merge branch 'master' into tm-5493-user-comments ([88c32c5](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/88c32c57c503e4bb8dd1fb155b8a572e5a89fc2e))
15
+ * Merge branch 'master' into tm-5493-user-comments ([a4fb3af](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/commit/a4fb3af3e7e1ba0b9bda2764b024bc4d1ca72565))
16
+
1
17
  # [45.0.0-beta.1](https://gitlab.com/g2m-gentoo/team-floyd/themes/gatsby-themes/compare/v44.2.6...v45.0.0-beta.1) (2025-06-13)
2
18
 
3
19
 
package/gatsby-node.mjs CHANGED
@@ -1,3 +1,4 @@
1
+ /* eslint-disable no-nested-ternary */
1
2
  /* eslint-disable no-underscore-dangle */
2
3
  /* eslint-disable import/no-extraneous-dependencies */
3
4
  /* eslint-disable prefer-destructuring */
@@ -63,6 +64,7 @@ let schemaData = null;
63
64
  let templatesData = null;
64
65
  let defaultLanguage = null;
65
66
  const pagesToCreate = [];
67
+ let commentsData = null;
66
68
  const relations = {};
67
69
 
68
70
  // eslint-disable-next-line import/prefer-default-export
@@ -174,6 +176,7 @@ export const createPages = async (
174
176
  autogenerated,
175
177
  siteSchema,
176
178
  authors,
179
+ comments: commentsData?.[page?.id],
177
180
  lang: page.language,
178
181
  ...themeOptions,
179
182
  });
@@ -374,6 +377,7 @@ export const onPreBootstrap = async () => {
374
377
  cryptoExchangesData,
375
378
  cryptoBrokersData,
376
379
  countriesData,
380
+ commentsData
377
381
  } = await fetchSiteSettings(process.env.GATSBY_SITE_NAME));
378
382
  };
379
383
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "gatsby-core-theme",
3
- "version": "45.0.0-beta.1",
3
+ "version": "45.0.0-beta.2",
4
4
  "description": "Gatsby Theme NPM Package",
5
5
  "author": "",
6
6
  "license": "ISC",
@@ -29,7 +29,7 @@ describe('Contact Us form Component', () => {
29
29
  );
30
30
 
31
31
  await waitFor(() => {
32
- expect(container).toBeTruthy();
32
+ expect(container).toBeInTheDocument();
33
33
  expect(container.querySelector('H1').innerHTML).toBe('Contact Us');
34
34
  });
35
35
  });
@@ -51,7 +51,7 @@ describe('Contact Us form Component', () => {
51
51
  );
52
52
 
53
53
  await waitFor(() => {
54
- expect(container.querySelector('.contactForm')).toBeTruthy();
54
+ expect(container.querySelector('.contactForm')).toBeInTheDocument();
55
55
  expect(container.querySelector('.contactForm').querySelectorAll('input')).toHaveLength(4);
56
56
  expect(
57
57
  container.querySelector('.contactForm').querySelectorAll('input[type="text"]')
@@ -72,7 +72,7 @@ describe('Contact Us form Component', () => {
72
72
  );
73
73
 
74
74
  expect(container.querySelectorAll('input[type="text"]').length).toEqual(2);
75
- expect(container.querySelector('textarea')).toBeTruthy();
75
+ expect(container.querySelector('textarea')).toBeInTheDocument();
76
76
  expect(container.querySelector('.formGroup').querySelector('label')).toBeFalsy();
77
77
  expect(container.querySelectorAll('label').length).toEqual(1);
78
78
  });
@@ -0,0 +1,100 @@
1
+ .commentContainer{
2
+ @include flex-direction(column);
3
+
4
+ gap: 1.6rem;
5
+ }
6
+
7
+ .commentTopArea{
8
+ @include flex-align(center, start);
9
+
10
+ gap: .8rem;
11
+ }
12
+
13
+ .commentName{
14
+ font-size: 18px;
15
+ font-weight: 600;
16
+ line-height: 28px;
17
+ }
18
+
19
+ .commentJobTitle{
20
+ height: 24px;
21
+ padding: .8rem;
22
+ font-size: 1.2rem;
23
+ font-weight: 400;
24
+ line-height: 1.8rem;
25
+
26
+ @include flex-align(center, center);
27
+
28
+ border-radius: 5rem;
29
+ background: #CFC7C0;
30
+ }
31
+
32
+ .commentDate{
33
+ color: #64748B;
34
+ font-size: 1.6rem;
35
+ font-weight: 400;
36
+ line-height: 2.6rem;
37
+ }
38
+
39
+ .commentContent{
40
+ border-radius: .8rem;
41
+ border: .1rem solid #F4F4F4;
42
+ background: #FFF;
43
+ position: relative;
44
+ padding: 2.4rem;
45
+ }
46
+
47
+ .isReplyThread::before{
48
+ content: "";
49
+ position: absolute;
50
+ right: 100%;
51
+ border-left: .1rem solid #CFC7C0;
52
+ border-bottom: .1rem solid #CFC7C0;
53
+ border-bottom-left-radius: 2rem;
54
+ height: 30rem;
55
+ width: 3rem;
56
+ bottom: 50%;
57
+ z-index: -1;
58
+ }
59
+
60
+ .commentActions{
61
+ margin-left: auto;
62
+
63
+ @include flex-direction(row);
64
+ }
65
+
66
+ .buttonGroup{
67
+ background: #CFC7C0;
68
+ font-size: 1.2rem;
69
+ padding: .8rem;
70
+ gap: .4rem;
71
+
72
+ @include flex-align(center, center);
73
+
74
+ &.left{
75
+ border-bottom-left-radius: 5rem;
76
+ border-top-left-radius: 5rem;
77
+ padding-right: 0;
78
+ }
79
+
80
+ &.right{
81
+ border-bottom-right-radius: 5rem;
82
+ border-top-right-radius: 5rem;
83
+ }
84
+ }
85
+
86
+ .buttonGroupIcon{
87
+ width: 1.6rem;
88
+ height: 1.6rem;
89
+ }
90
+
91
+ .replyButton{
92
+ border-radius: 10rem;
93
+ border: 1.5px solid #161128;
94
+ font-size: 1.2rem;
95
+ font-weight: 700;
96
+ line-height: 1.8rem;
97
+ text-transform: capitalize;
98
+ margin-left: .8rem;
99
+ padding: 0 1.6rem;
100
+ }
@@ -0,0 +1,65 @@
1
+ /* eslint-disable react-hooks/exhaustive-deps */
2
+ /* eslint-disable jsx-a11y/click-events-have-key-events */
3
+ /* eslint-disable import/no-extraneous-dependencies */
4
+ import React from 'react';
5
+ import PropTypes from 'prop-types';
6
+ import LazyImage from '~hooks/lazy-image';
7
+
8
+ import useTranslate from '~hooks/useTranslate/useTranslate';
9
+ import styles from './comment.module.scss';
10
+
11
+ const Comment = ({ comment, authors, isReply }) => {
12
+ const commentName = comment?.author_id ? authors?.[comment?.author_id]?.name : comment.name;
13
+ const commentJobTitle = comment?.author_id ? authors?.[comment?.author_id]?.author_title : undefined;
14
+
15
+ const date = new Date(comment.updated_at);
16
+ const day = String(date.getUTCDate()).padStart(2, '0');
17
+ const month = String(date.getUTCMonth() + 1).padStart(2, '0');
18
+ const year = date.getUTCFullYear();
19
+
20
+ const hours = String(date.getUTCHours()).padStart(2, '0');
21
+ const minutes = String(date.getUTCMinutes()).padStart(2, '0');
22
+ const commentDate = `${day}/${month}/${year} | ${hours}:${minutes}`;
23
+
24
+
25
+ return <div className={`${styles.commentContainer} ${isReply ? styles.isReply : ''}`}>
26
+ <div className={styles.commentTopArea}>
27
+ <LazyImage src='/images/anon-user.svg' alt={`${commentName} user image`} />
28
+ <span className={styles.commentName}>{commentName}</span>
29
+ {comment?.author_id && <span className={styles.commentJobTitle}>{commentJobTitle}</span>}
30
+ <span className={styles.commentDate}>{commentDate}</span>
31
+ </div>
32
+ <div className={`${styles.commentContent} ${isReply ? styles.isReplyThread : ''}`}>
33
+ <p>{comment.comment}</p>
34
+ </div>
35
+ <div className={styles.commentActions}>
36
+ <button aria-label='Like Button' type='button' className={`${styles.buttonGroup} ${styles.left}`}>
37
+ <LazyImage className={styles.buttonGroupIcon} src='/images/like.svg' />
38
+ {comment.votes_up}
39
+ </button>
40
+ <button aria-label='Dislike Button' type='button' className={`${styles.buttonGroup} ${styles.right}`}>
41
+ <LazyImage className={styles.buttonGroupIcon} src='/images/dislike.svg' />
42
+ {comment.votes_down}
43
+ </button>
44
+ <button aria-label='Reply Button' type='button' className={styles.replyButton}>
45
+ {useTranslate('reply_button', 'Reply')}
46
+ </button>
47
+ </div>
48
+ </div>
49
+ };
50
+
51
+ Comment.propTypes = {
52
+ comment: PropTypes.shape({
53
+ author_id: PropTypes.number,
54
+ name: PropTypes.string,
55
+ email: PropTypes.string,
56
+ comment: PropTypes.string,
57
+ updated_at: PropTypes.string,
58
+ votes_up: PropTypes.number,
59
+ votes_down: PropTypes.number,
60
+ }),
61
+ authors: PropTypes.shape({}),
62
+ isReply: PropTypes.bool
63
+ };
64
+
65
+ export default Comment;
@@ -0,0 +1,81 @@
1
+ /* eslint-disable react-hooks/exhaustive-deps */
2
+ /* eslint-disable jsx-a11y/click-events-have-key-events */
3
+ /* eslint-disable import/no-extraneous-dependencies */
4
+ import React from 'react';
5
+
6
+ import Form from "gatsby-core-theme/src/components/organisms/form";
7
+ import PropTypes from 'prop-types';
8
+ import { FaArrowRight } from '@react-icons/all-files/fa/FaArrowRight';
9
+ import { commentForm } from '../../../constants/forms';
10
+ import styles from './leave-comment-form.module.scss';
11
+
12
+ const LeaveCommentForm = ({
13
+ page,
14
+ showLabels = true,
15
+ showButtonIcon = false,
16
+ buttonLabel = 'Add Comment',
17
+ submitUrl = `${process.env.GATSBY_API_URL}/v0.1/content/comments`,
18
+ successMessage = 'Comment Added',
19
+ failMessage = 'Comment Not Added',
20
+ validationMessage = 'Fill all required fields',
21
+ buttonIcon = <FaArrowRight fontSize={20} title="Right-pointing Arrow Icon" />,
22
+ }) => {
23
+ const activeMarket = page?.market;
24
+
25
+ const customSubmit = (form) => {
26
+ const formData = new FormData(form);
27
+ const data = Object.fromEntries(formData.entries());
28
+
29
+ // if(data?.["g-recaptcha-response"] === '') return;
30
+
31
+ data.reference_id = page?.id;
32
+ data.site_id = process.env.SITE_ID;
33
+ data.market_short_code = activeMarket;
34
+ delete data["g-recaptcha-response"];
35
+
36
+ return fetch(submitUrl, {
37
+ method: 'POST',
38
+ cache: 'no-cache',
39
+ headers: {
40
+ 'Content-Type': 'application/json',
41
+ Accept: 'application/json',
42
+ },
43
+ body: JSON.stringify(data),
44
+ });
45
+ };
46
+
47
+ return <div className={`${styles.commentFormContainer}`}>
48
+ <Form
49
+ formOptions={commentForm[activeMarket] || commentForm.default}
50
+ showLabels={showLabels}
51
+ hasButton
52
+ showButtonIcon={showButtonIcon}
53
+ buttonLabel={buttonLabel}
54
+ customeSubmit={customSubmit}
55
+ successMessage={successMessage}
56
+ failMessage={failMessage}
57
+ validationMessage={validationMessage}
58
+ path={page?.path}
59
+ buttonIcon={buttonIcon}
60
+ customClass={styles.commentForm || ''}
61
+ />
62
+ </div>
63
+ };
64
+
65
+ LeaveCommentForm.propTypes = {
66
+ page: PropTypes.shape({
67
+ market: PropTypes.string,
68
+ id: PropTypes.number,
69
+ path: PropTypes.string,
70
+ }).isRequired,
71
+ submitUrl: PropTypes.string,
72
+ successMessage: PropTypes.string,
73
+ failMessage: PropTypes.string,
74
+ validationMessage: PropTypes.string,
75
+ buttonLabel: PropTypes.string,
76
+ showButtonIcon: PropTypes.bool,
77
+ showLabels: PropTypes.bool,
78
+ buttonIcon: PropTypes.element,
79
+ };
80
+
81
+ export default LeaveCommentForm;
@@ -0,0 +1,23 @@
1
+ .commentForm{
2
+ background-color: transparent;
3
+ box-shadow: none;
4
+ padding: 0;
5
+ margin-top: 3.5rem;
6
+
7
+ > form{
8
+ display: grid;
9
+ grid-template-columns: 1fr 1fr;
10
+
11
+ > button{
12
+ border: .2rem solid #161128;
13
+ padding: 8px 16px;
14
+ width: auto;
15
+ margin-left: auto;
16
+ border-radius: 5rem;
17
+ font-size: 1.4rem;
18
+ font-weight: 700;
19
+ line-height: 2.2rem;
20
+ text-transform: capitalize;
21
+ }
22
+ }
23
+ }
@@ -61,6 +61,8 @@ const Modules = ({ module, page, pageContext, modulePosition }) => {
61
61
  return lazy(() => import("~molecules/bonus/template-one"));
62
62
  case "image":
63
63
  return lazy(() => import("~atoms/image"));
64
+ case "comments":
65
+ return lazy(() => import("~organisms/comments"));
64
66
  case "spotlights":
65
67
  if (module?.mode === "image_text") {
66
68
  if (module?.style === "template_two") {
@@ -0,0 +1,29 @@
1
+ import React from "react";
2
+
3
+ import PropTypes, { arrayOf } from 'prop-types';
4
+ import keygen from '~helpers/keygen';
5
+ import Comment from "../../../molecules/comment";
6
+
7
+ const CommentTree = React.memo(({ comment, authors, depth = 0 }) => (
8
+ <div key={keygen()} style={{ marginLeft: depth * 56 }}>
9
+ <Comment comment={comment} authors={authors} isReply={depth > 0} />
10
+ {comment.replies?.map(reply => (
11
+ <CommentTree
12
+ key={keygen()}
13
+ comment={reply}
14
+ authors={authors}
15
+ depth={depth + 1}
16
+ />
17
+ ))}
18
+ </div>
19
+ ));
20
+
21
+ CommentTree.propTypes = {
22
+ comment: PropTypes.shape({
23
+ replies: arrayOf(PropTypes.shape({}))
24
+ }),
25
+ authors: PropTypes.shape({}),
26
+ depth: PropTypes.number
27
+ };
28
+
29
+ export default CommentTree;
@@ -0,0 +1,26 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import CommentTree from './comment-tree';
4
+ import LeaveCommentForm from '../../molecules/leave-comment-form';
5
+
6
+ const Comments = ({ pageContext }) => {
7
+ const { comments, authors } = pageContext;
8
+ if(!comments) return;
9
+
10
+ return (
11
+ <div>
12
+ {comments.map((comment) => <CommentTree comment={comment} authors={authors} depth={0} />)}
13
+ <LeaveCommentForm page={pageContext.page} />
14
+ </div>
15
+ );
16
+ };
17
+
18
+ Comments.propTypes = {
19
+ pageContext: PropTypes.shape({
20
+ comments: PropTypes.arrayOf({}),
21
+ authors: PropTypes.shape({}),
22
+ page: PropTypes.shape({})
23
+ })
24
+ };
25
+
26
+ export default Comments;
@@ -83,6 +83,58 @@ export const contactUsForm = {
83
83
  },
84
84
  };
85
85
 
86
+ export const commentForm = {
87
+ default: {
88
+ title: {
89
+ label: 'Leave a comment',
90
+ translationKey: 'leave_a_comment',
91
+ },
92
+ hasReCAPTCHA: true,
93
+ reCaptcha: {
94
+ // validations: {
95
+ // label: 'Please confirm',
96
+ // translationKey: 'valid_recaptcha',
97
+ // icon: <IoMdCloseCircleOutline />,
98
+ // }
99
+ },
100
+ fields: [
101
+ {
102
+ label: 'Name',
103
+ translationKey: 'name_label',
104
+ id: 'name',
105
+ type: 'text',
106
+ placeholder: {
107
+ label: 'Name',
108
+ translationKey: 'name_placeholder',
109
+ },
110
+ twoCol: true,
111
+ },
112
+ {
113
+ label: 'Email Address',
114
+ id: 'email',
115
+ type: 'email',
116
+ placeholder: {
117
+ label: 'Email@placeholder.com',
118
+ translationKey: 'email_placeholder',
119
+ },
120
+ translationKey: 'email_label',
121
+ twoCol: true,
122
+ },
123
+ {
124
+ id: 'comment',
125
+ type: 'textarea',
126
+ placeholder: {
127
+ label: 'Leave a comment',
128
+ translationKey: 'comment_placeholder',
129
+ },
130
+ translationKey: 'textarea_label',
131
+ maxLength: '1000',
132
+ twoCol: false,
133
+ },
134
+ ],
135
+ },
136
+ };
137
+
86
138
  export const newsLetterForm = {
87
139
  default: {
88
140
  title: {
@@ -0,0 +1,27 @@
1
+ // eslint-disable-next-line import/prefer-default-export
2
+ export function transformComments(comments) {
3
+ const transformedComments = {};
4
+
5
+ for (const comment of Object.values(comments || {})) {
6
+ const refId = comment.reference_id;
7
+ if (!transformedComments[refId]) {
8
+ transformedComments[refId] = [];
9
+ }
10
+ if(comment.approved){
11
+ transformedComments[refId].push({
12
+ replies: transformComments(comment?.replies)[refId],
13
+ email: comment?.email,
14
+ name: comment?.name,
15
+ comment: comment?.comment,
16
+ rate: comment?.rate,
17
+ author_id: comment?.author_id,
18
+ votes_down: comment.votes_down,
19
+ votes_up: comment.votes_up,
20
+ created_at: comment?.created_at,
21
+ updated_at: comment?.updated_at,
22
+ });
23
+ }
24
+ }
25
+
26
+ return transformedComments;
27
+ }
@@ -34,78 +34,78 @@ export function generateHreflangs(page, data) {
34
34
 
35
35
  return href;
36
36
  });
37
- }
38
-
39
- export function transformCategories(categoryIds, data) {
40
- return categoryIds.map((id) => ({
41
- name: data.categories[id].name,
42
- short_name: data.categories[id].short_name,
43
- }));
44
- }
45
-
46
- export function addBreadcrumbsToPage(breadcrumbIds, market, data) {
47
- if (!breadcrumbIds || breadcrumbIds.length === 0) return []; // Return an empty array if no breadcrumb IDs
48
-
49
- const pathPrefix = data.site_markets[market]?.path_prefix;
50
-
51
- const breadcrumbsArray = breadcrumbIds.map((id) => {
52
- // Check if the breadcrumb ID exists in data.breadcrumbs
53
- const breadcrumbData = data.breadcrumbs[id];
54
-
55
- // If breadcrumbData is not found (ID doesn't exist), return undefined
56
- if (!breadcrumbData) {
57
- return undefined; // Return undefined for non-existent breadcrumbs
58
- }
59
-
60
- // Clone the existing breadcrumb data
61
- const clonedBreadcrumbData = cloneDeep(breadcrumbData);
62
-
63
- // Update path if pathPrefix is available
64
- if (pathPrefix && clonedBreadcrumbData.path) {
65
- clonedBreadcrumbData.path = `${pathPrefix}/${clonedBreadcrumbData.path}`;
66
- }
67
-
68
- return clonedBreadcrumbData;
69
- }).filter(Boolean); // Filter out any undefined values
70
-
71
- return breadcrumbsArray;
72
- }
37
+ }
38
+
39
+ export function transformCategories(categoryIds, data) {
40
+ return categoryIds.map((id) => ({
41
+ name: data.categories[id].name,
42
+ short_name: data.categories[id].short_name,
43
+ }));
44
+ }
73
45
 
74
- export function addMultipleAuthorsToPage(page, data) {
75
- // Check if the page has multiple authors
76
- if (page.authors && page.authors.length > 1) {
77
- const allAuthors = page.authors.map((authorId) => {
78
- if (data.authors[authorId] === null) return undefined; // Return undefined for non-existent authors
79
-
80
- const author = data.authors[authorId];
81
- let review;
82
-
83
- // Check if author review per page per author is available
84
- if (page.author_reviews && page.author_reviews[authorId]?.length) {
85
- review = data.content[page.author_reviews[authorId]];
86
- }
87
-
88
- return { ...author, author_review: review || "" }; // Return the author object with review or empty string
89
- }).filter(Boolean); // Filter out undefined values
90
-
91
- return allAuthors; // Add all authors to the transformed page
46
+ export function addBreadcrumbsToPage(breadcrumbIds, market, data) {
47
+ if (!breadcrumbIds || breadcrumbIds.length === 0) return []; // Return an empty array if no breadcrumb IDs
48
+
49
+ const pathPrefix = data.site_markets[market]?.path_prefix;
50
+
51
+ const breadcrumbsArray = breadcrumbIds.map((id) => {
52
+ // Check if the breadcrumb ID exists in data.breadcrumbs
53
+ const breadcrumbData = data.breadcrumbs[id];
54
+
55
+ // If breadcrumbData is not found (ID doesn't exist), return undefined
56
+ if (!breadcrumbData) {
57
+ return undefined; // Return undefined for non-existent breadcrumbs
92
58
  }
93
- return [];
59
+
60
+ // Clone the existing breadcrumb data
61
+ const clonedBreadcrumbData = cloneDeep(breadcrumbData);
62
+
63
+ // Update path if pathPrefix is available
64
+ if (pathPrefix && clonedBreadcrumbData.path) {
65
+ clonedBreadcrumbData.path = `${pathPrefix}/${clonedBreadcrumbData.path}`;
66
+ }
67
+
68
+ return clonedBreadcrumbData;
69
+ }).filter(Boolean); // Filter out any undefined values
70
+
71
+ return breadcrumbsArray;
72
+ }
73
+
74
+ export function addMultipleAuthorsToPage(page, data) {
75
+ // Check if the page has multiple authors
76
+ if (page.authors && page.authors.length > 1) {
77
+ const allAuthors = page.authors.map((authorId) => {
78
+ if (data.authors[authorId] === null) return undefined; // Return undefined for non-existent authors
79
+
80
+ const author = data.authors[authorId];
81
+ let review;
82
+
83
+ // Check if author review per page per author is available
84
+ if (page.author_reviews && page.author_reviews[authorId]?.length) {
85
+ review = data.content[page.author_reviews[authorId]];
86
+ }
87
+
88
+ return { ...author, author_review: review || "" }; // Return the author object with review or empty string
89
+ }).filter(Boolean); // Filter out undefined values
90
+
91
+ return allAuthors; // Add all authors to the transformed page
94
92
  }
93
+ return [];
94
+ }
95
95
 
96
- export function mapAuthorsWithCountries(authors, countries) {
97
- return Object.keys(authors).reduce((acc, authorId) => {
98
- const author = authors[authorId];
99
- const country = countries[author.country_id] ?
100
- pick(
101
- {
102
- ...countries[author.country_id],
103
- logo: countries[author.country_id].standardised_logo_filename_object,
104
- },
105
- pickRelationKeys.country
106
- ) : null
107
-
108
- acc[authorId] = { ...author, country };
109
- return acc;
110
- }, {});
111
- }
96
+ export function mapAuthorsWithCountries(authors, countries) {
97
+ return Object.keys(authors).reduce((acc, authorId) => {
98
+ const author = authors[authorId];
99
+ const country = countries[author.country_id] ?
100
+ pick(
101
+ {
102
+ ...countries[author.country_id],
103
+ logo: countries[author.country_id].standardised_logo_filename_object,
104
+ },
105
+ pickRelationKeys.country
106
+ ) : null
107
+
108
+ acc[authorId] = { ...author, country };
109
+ return acc;
110
+ }, {});
111
+ }
@@ -48,7 +48,7 @@ export async function getResponsableGamings(siteName) {
48
48
 
49
49
  const allMarketsData = Object.values(affSettings)[0].markets_data;
50
50
  const allRespGamings = {};
51
- Object.keys(allMarketsData).map((key) => {
51
+ Object.keys(allMarketsData).forEach((key) => {
52
52
  const value = allMarketsData[key];
53
53
  const marketRespGamings = value.responsible_gaming_ids.map(
54
54
  (id) => allResponsibleGamingByID[id][0]
@@ -130,6 +130,10 @@ export async function getSports(siteName) {
130
130
  return callAPIV2(`v0.1/sports/sorted/sorted-v2/${siteName}`);
131
131
  }
132
132
 
133
+ export async function getSiteComments(siteName) {
134
+ return callAPIV2(`v0.1/content/comments/sorted-v2/${siteName}`);
135
+ }
136
+
133
137
  export async function getCryptoBrokers() {
134
138
  return callAPIV2(`v0.1/cryptos/brokers/sorted`);
135
139
  }
@@ -1,3 +1,4 @@
1
+ /* eslint-disable no-unused-vars */
1
2
  /* eslint-disable import/prefer-default-export */
2
3
  import {
3
4
  getSiteSettings,
@@ -19,6 +20,7 @@ import {
19
20
  getCryptoBrokers,
20
21
  getCryptoExchanges,
21
22
  getCryptoWallets,
23
+ getSiteComments
22
24
  } from "./api.mjs";
23
25
  import { groupBy } from "../helpers/getters.mjs";
24
26
  import { transformOperators } from "../resolver/operators.mjs";
@@ -26,6 +28,7 @@ import { transformCryptosData } from "../resolver/cryptos.mjs";
26
28
  import { transformGames, transformGamesJackpot } from "../resolver/games.mjs";
27
29
  import { transformPayments } from "../resolver/payment.mjs";
28
30
  import { transformProvider } from "../resolver/providers.mjs";
31
+ import { transformComments } from "../resolver/comments.mjs";
29
32
 
30
33
  export const fetchSiteSettings = async (siteName, previewPath = null) => {
31
34
  const siteSettingsPromise = getSiteSettings(siteName);
@@ -61,6 +64,8 @@ export const fetchSiteSettings = async (siteName, previewPath = null) => {
61
64
 
62
65
  const contentPromise = getContent(process.env.GATSBY_SITE_NAME);
63
66
 
67
+ const userCommentsPromise = getSiteComments(process.env.GATSBY_SITE_NAME);
68
+
64
69
  const pagesPromise = getPages(process.env.GATSBY_SITE_NAME, previewPath);
65
70
 
66
71
  let sportsPromise = Promise.resolve({});
@@ -85,7 +90,8 @@ export const fetchSiteSettings = async (siteName, previewPath = null) => {
85
90
  sportsData,
86
91
  cryptoBrokers,
87
92
  cryptoExchanges,
88
- cryptoWallets
93
+ cryptoWallets,
94
+ userComments
89
95
  ] = await Promise.all([
90
96
  ribbonsPromise,
91
97
  responsableGamingPromise,
@@ -103,7 +109,8 @@ export const fetchSiteSettings = async (siteName, previewPath = null) => {
103
109
  sportsPromise,
104
110
  cryptoBrokersPromise,
105
111
  cryptoExchangesPromise,
106
- cryptoWalletsPromise
112
+ cryptoWalletsPromise,
113
+ userCommentsPromise,
107
114
  ]);
108
115
 
109
116
  const allMarketPrefixes = Object.keys(allMarketsData)
@@ -118,6 +125,8 @@ export const fetchSiteSettings = async (siteName, previewPath = null) => {
118
125
 
119
126
  const paymentData = transformPayments(payments);
120
127
 
128
+ const commentsData = transformComments( userComments );
129
+
121
130
  const gamesData = transformGames(games, {
122
131
  providers: providersData,
123
132
  jackpot_data: transformGamesJackpot(gamesJackpotData)
@@ -179,6 +188,7 @@ export const fetchSiteSettings = async (siteName, previewPath = null) => {
179
188
  gamesData,
180
189
  sportsData,
181
190
  cryptoExchangesData,
182
- cryptoBrokersData
191
+ cryptoBrokersData,
192
+ commentsData
183
193
  };
184
194
  };
@@ -0,0 +1,15 @@
1
+ <svg width="50" height="50" viewBox="0 0 50 50" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
2
+ <circle cx="25" cy="25" r="24" fill="url(#pattern0_8007_12905)"/>
3
+ <circle cx="25" cy="25" r="24" fill="url(#pattern1_8007_12905)"/>
4
+ <circle cx="25" cy="25" r="24" stroke="#ECECEC"/>
5
+ <defs>
6
+ <pattern id="pattern0_8007_12905" patternContentUnits="objectBoundingBox" width="1" height="1">
7
+ <use xlink:href="#image0_8007_12905" transform="translate(0 -0.25) scale(0.003125)"/>
8
+ </pattern>
9
+ <pattern id="pattern1_8007_12905" patternContentUnits="objectBoundingBox" width="1" height="1">
10
+ <use xlink:href="#image1_8007_12905" transform="scale(0.00462963)"/>
11
+ </pattern>
12
+ <image id="image0_8007_12905" width="320" height="480" preserveAspectRatio="none" xlink:href=""/>
13
+ <image id="image1_8007_12905" width="216" height="216" preserveAspectRatio="none" xlink:href=""/>
14
+ </defs>
15
+ </svg>