@newskit-render/core 0.0.0-d55b7d88
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/.ci/trigger-circleci-workflow.sh +37 -0
- package/.eslintignore +23 -0
- package/.eslintrc +74 -0
- package/.prettierignore +19 -0
- package/.prettierrc +7 -0
- package/CHANGELOG.md +7638 -0
- package/Dockerfile +14 -0
- package/Dockerfile.withNewRelic +15 -0
- package/README.md +274 -0
- package/__tests__/pages/__snapshots__/brightcove.test.tsx.snap +20 -0
- package/__tests__/pages/__snapshots__/home.test.tsx.snap +1195 -0
- package/__tests__/pages/brightcove.test.tsx +34 -0
- package/__tests__/pages/empty.test.tsx +10 -0
- package/__tests__/pages/home.test.tsx +16 -0
- package/app-context/InstrumentationContextProvider.tsx +36 -0
- package/assets/fontFamily.ts +416 -0
- package/assets/fonts/Graphik-Black-App.ttf +0 -0
- package/assets/fonts/Graphik-BlackItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Bold-App.ttf +0 -0
- package/assets/fonts/Graphik-BoldItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Extralight-App.ttf +0 -0
- package/assets/fonts/Graphik-ExtralightItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Light-App.ttf +0 -0
- package/assets/fonts/Graphik-LightItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Medium-App.ttf +0 -0
- package/assets/fonts/Graphik-Medium-Web.woff2 +0 -0
- package/assets/fonts/Graphik-MediumItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Regular-App.ttf +0 -0
- package/assets/fonts/Graphik-Regular-Web.woff2 +0 -0
- package/assets/fonts/Graphik-RegularItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Semibold-App.ttf +0 -0
- package/assets/fonts/Graphik-Semibold-Web.woff2 +0 -0
- package/assets/fonts/Graphik-SemiboldItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Super-App.ttf +0 -0
- package/assets/fonts/Graphik-SuperItalic-App.ttf +0 -0
- package/assets/fonts/Graphik-Thin-App.ttf +0 -0
- package/assets/fonts/Graphik-ThinItalic-App.ttf +0 -0
- package/assets/fonts/Montserrat-Black.ttf +0 -0
- package/assets/fonts/Montserrat-BlackItalic.ttf +0 -0
- package/assets/fonts/Montserrat-Bold.ttf +0 -0
- package/assets/fonts/Montserrat-BoldItalic.ttf +0 -0
- package/assets/fonts/Montserrat-ExtraBold.ttf +0 -0
- package/assets/fonts/Montserrat-ExtraBoldItalic.ttf +0 -0
- package/assets/fonts/Montserrat-ExtraLight.ttf +0 -0
- package/assets/fonts/Montserrat-ExtraLightItalic.ttf +0 -0
- package/assets/fonts/Montserrat-Italic.ttf +0 -0
- package/assets/fonts/Montserrat-Light.ttf +0 -0
- package/assets/fonts/Montserrat-LightItalic.ttf +0 -0
- package/assets/fonts/Montserrat-Medium.ttf +0 -0
- package/assets/fonts/Montserrat-MediumItalic.ttf +0 -0
- package/assets/fonts/Montserrat-Regular.ttf +0 -0
- package/assets/fonts/Montserrat-SemiBold.ttf +0 -0
- package/assets/fonts/Montserrat-SemiBoldItalic.ttf +0 -0
- package/assets/fonts/Montserrat-Thin.ttf +0 -0
- package/assets/fonts/Montserrat-ThinItalic.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Black-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-BlackItalic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Bold-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Bold-Web.woff2 +0 -0
- package/assets/fonts/PublicoHeadline-BoldItalic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Extrabold-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-ExtraboldItalic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Italic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Light-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-LightItalic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Medium-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-MediumItalic-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Roman-App.ttf +0 -0
- package/assets/fonts/PublicoHeadline-Roman-Web.woff2 +0 -0
- package/assets/fonts/PublicoText-Bold-App.ttf +0 -0
- package/assets/fonts/PublicoText-BoldItalic-App.ttf +0 -0
- package/assets/fonts/PublicoText-Italic-App.ttf +0 -0
- package/assets/fonts/PublicoText-Italic-Web.woff2 +0 -0
- package/assets/fonts/PublicoText-Roman-App.ttf +0 -0
- package/assets/fonts/PublicoText-Roman-Web.woff2 +0 -0
- package/assets/fonts/PublicoText-Semibold-App.ttf +0 -0
- package/assets/fonts/PublicoText-SemiboldItalic-App.ttf +0 -0
- package/assets/fonts/Roboto-Bold.ttf +0 -0
- package/assets/fonts/Roboto-Light.ttf +0 -0
- package/assets/fonts/Roboto-Medium.ttf +0 -0
- package/assets/fonts/Roboto-Regular.ttf +0 -0
- package/assets/fonts/TheSun-Bold.ttf +0 -0
- package/assets/fonts/TheSun-BoldItalic.ttf +0 -0
- package/assets/fonts/TheSun-HeavyCondensed.ttf +0 -0
- package/assets/fonts/TheSun-HeavyNarrow.ttf +0 -0
- package/assets/fonts/TheSun-Italic.ttf +0 -0
- package/assets/fonts/TheSun-Medium.ttf +0 -0
- package/assets/fonts/TheSun-MediumItalic.ttf +0 -0
- package/assets/fonts/TheSun-Regular.ttf +0 -0
- package/assets/fonts/TimesDigital-Bold.ttf +0 -0
- package/assets/fonts/TimesDigital-BoldItalic.ttf +0 -0
- package/assets/fonts/TimesDigital-BoldSC.ttf +0 -0
- package/assets/fonts/TimesDigital-Italic.ttf +0 -0
- package/assets/fonts/TimesDigital-Regular.ttf +0 -0
- package/assets/fonts/TimesDigital-RegularSC.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-Bold.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-BoldItalic.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-BoldSC.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-Italic.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-Regular.ttf +0 -0
- package/assets/fonts/TimesDigitalW04-RegularSC.ttf +0 -0
- package/assets/fonts/TimesModern-Regular.otf +0 -0
- package/assets/fonts/attribuitions.mdx +35 -0
- package/assets/fonts/bitter-medium.woff +0 -0
- package/assets/fonts/bitter-medium.woff2 +0 -0
- package/assets/fonts/bitter-mediumitalic.woff +0 -0
- package/assets/fonts/bitter-mediumitalic.woff2 +0 -0
- package/assets/fonts/bitter-regular.woff +0 -0
- package/assets/fonts/bitter-regular.woff2 +0 -0
- package/assets/fonts/bitter-semibold.woff +0 -0
- package/assets/fonts/bitter-semibold.woff2 +0 -0
- package/assets/fonts/dmmono-medium.woff +0 -0
- package/assets/fonts/dmmono-medium.woff2 +0 -0
- package/assets/fonts/dmsans-bold-webfont.woff +0 -0
- package/assets/fonts/dmsans-bold-webfont.woff2 +0 -0
- package/assets/fonts/dmsans-bolditalic-webfont.woff +0 -0
- package/assets/fonts/dmsans-bolditalic-webfont.woff2 +0 -0
- package/assets/fonts/dmsans-italic-webfont.woff +0 -0
- package/assets/fonts/dmsans-italic-webfont.woff2 +0 -0
- package/assets/fonts/dmsans-medium-webfont.woff +0 -0
- package/assets/fonts/dmsans-medium-webfont.woff2 +0 -0
- package/assets/fonts/dmsans-mediumitalic-webfont.woff +0 -0
- package/assets/fonts/dmsans-mediumitalic-webfont.woff2 +0 -0
- package/assets/fonts/dmsans-regular-webfont.woff +0 -0
- package/assets/fonts/dmsans-regular-webfont.woff2 +0 -0
- package/assets/fonts/notosans-bold-webfont.woff +0 -0
- package/assets/fonts/notosans-bold-webfont.woff2 +0 -0
- package/assets/fonts/notosans-italic-webfont.woff +0 -0
- package/assets/fonts/notosans-italic-webfont.woff2 +0 -0
- package/assets/fonts/notosans-medium-webfont.woff +0 -0
- package/assets/fonts/notosans-medium-webfont.woff2 +0 -0
- package/assets/fonts/notosans-regular-webfont.woff +0 -0
- package/assets/fonts/notosans-regular-webfont.woff2 +0 -0
- package/assets/fonts/poppins-bold-webfont.woff +0 -0
- package/assets/fonts/poppins-bold-webfont.woff2 +0 -0
- package/assets/fonts/poppins-bolditalic-webfont.woff +0 -0
- package/assets/fonts/poppins-bolditalic-webfont.woff2 +0 -0
- package/assets/fonts/poppins-extrabold-webfont.woff +0 -0
- package/assets/fonts/poppins-extrabold-webfont.woff2 +0 -0
- package/assets/fonts/poppins-extrabolditalic-webfont.woff +0 -0
- package/assets/fonts/poppins-extrabolditalic-webfont.woff2 +0 -0
- package/assets/fonts/poppins-italic-webfont.woff +0 -0
- package/assets/fonts/poppins-italic-webfont.woff2 +0 -0
- package/assets/fonts/poppins-light-webfont.woff +0 -0
- package/assets/fonts/poppins-light-webfont.woff2 +0 -0
- package/assets/fonts/poppins-lightitalic-webfont.woff +0 -0
- package/assets/fonts/poppins-lightitalic-webfont.woff2 +0 -0
- package/assets/fonts/poppins-medium-webfont.woff +0 -0
- package/assets/fonts/poppins-medium-webfont.woff2 +0 -0
- package/assets/fonts/poppins-mediumitalic-webfont.woff +0 -0
- package/assets/fonts/poppins-mediumitalic-webfont.woff2 +0 -0
- package/assets/fonts/poppins-regular-webfont.woff +0 -0
- package/assets/fonts/poppins-regular-webfont.woff2 +0 -0
- package/assets/fonts/poppins-semibold-webfont.woff +0 -0
- package/assets/fonts/poppins-semibold-webfont.woff2 +0 -0
- package/assets/fonts/poppins-semibolditalic-webfont.woff +0 -0
- package/assets/fonts/poppins-semibolditalic-webfont.woff2 +0 -0
- package/assets/fonts/source-serif-pro-600.woff +0 -0
- package/assets/fonts/source-serif-pro-600.woff2 +0 -0
- package/assets/fonts/source-serif-pro-600italic.woff +0 -0
- package/assets/fonts/source-serif-pro-600italic.woff2 +0 -0
- package/assets/fonts/source-serif-pro-700.woff +0 -0
- package/assets/fonts/source-serif-pro-700.woff2 +0 -0
- package/assets/fonts/source-serif-pro-700italic.woff +0 -0
- package/assets/fonts/source-serif-pro-700italic.woff2 +0 -0
- package/assets/fonts/source-serif-pro-italic.woff +0 -0
- package/assets/fonts/source-serif-pro-italic.woff2 +0 -0
- package/assets/fonts/source-serif-pro-regular.woff +0 -0
- package/assets/fonts/source-serif-pro-regular.woff2 +0 -0
- package/components/AccountDropdown/AccountDropdown.tsx +65 -0
- package/components/AccountDropdown/__tests__/AcountDropdown.test.tsx +83 -0
- package/components/AccountDropdown/accounts.ts +47 -0
- package/components/AccountDropdown/index.tsx +2 -0
- package/components/AccountDropdown/utils.ts +21 -0
- package/components/ErrorPage/ErrorPage.tsx +135 -0
- package/components/ad/index.tsx +35 -0
- package/components/article/Highlights.tsx +77 -0
- package/components/article/MetaBox.tsx +50 -0
- package/components/article/RelatedArticles.tsx +53 -0
- package/components/article/RelatedTopics.tsx +59 -0
- package/components/article/ShareBarComponent.tsx +53 -0
- package/components/article/__tests__/__snapshots__/index.test.tsx.snap +1568 -0
- package/components/article/__tests__/index.test.tsx +171 -0
- package/components/article/__tests__/mocks.ts +105 -0
- package/components/article/index.tsx +320 -0
- package/components/common/BackToHomepage.tsx +37 -0
- package/components/common/NavLink.tsx +26 -0
- package/components/common/NewskitLogo.tsx +52 -0
- package/components/common/NewskitLogoShort.tsx +48 -0
- package/components/common/SectionTitleBar.tsx +57 -0
- package/components/common/ShareButton.tsx +18 -0
- package/components/common/ViewMoreButton.tsx +20 -0
- package/components/common/iconNavLink/IconNavLink.tsx +70 -0
- package/components/common/iconNavLink/__tests__/IconNavLink.test.tsx +35 -0
- package/components/common/iconNavLink/index.ts +3 -0
- package/components/common/iconNavLink/types.ts +24 -0
- package/components/common/icons/IconAccount.tsx +13 -0
- package/components/common/icons/IconFilledTwitter.tsx +14 -0
- package/components/common/icons/StyledIconFilledChevronRight.tsx +12 -0
- package/components/footer/__snapshots__/index.test.tsx.snap +192 -0
- package/components/footer/index.test.tsx +13 -0
- package/components/footer/index.tsx +151 -0
- package/components/header/banner-messages.ts +39 -0
- package/components/header/index.tsx +52 -0
- package/components/header/navigation-links.ts +20 -0
- package/components/layout/Gutter.tsx +11 -0
- package/components/layout/LayoutTemplate.tsx +26 -0
- package/components/layout/MainGrid.tsx +66 -0
- package/components/layout/index.tsx +3 -0
- package/components/layout/layoutTypes.ts +13 -0
- package/components/section/ArticleSlice.tsx +35 -0
- package/components/section/CollectionBlock.tsx +54 -0
- package/components/section/SectionContext.tsx +9 -0
- package/components/section/__tests__/ArticleSlice.test.tsx +128 -0
- package/components/section/__tests__/CollectionBlock.test.tsx +83 -0
- package/components/section/__tests__/pageBlock.test.tsx +24 -0
- package/components/section/__tests__/sectionUtils.test.ts +94 -0
- package/components/section/index.tsx +39 -0
- package/components/section/layouts/Block.tsx +79 -0
- package/components/section/layouts/FallBack.tsx +24 -0
- package/components/section/layouts/Lead.tsx +43 -0
- package/components/section/layouts/Rows.tsx +82 -0
- package/components/section/layouts/SectionTitle.tsx +57 -0
- package/components/section/layouts/__tests__/Lead.test.tsx +37 -0
- package/components/section/layouts/__tests__/Rows.test.tsx +40 -0
- package/components/section/layouts/__tests__/SectionTitle.test.tsx +37 -0
- package/components/section/layouts/__tests__/__snapshots__/Lead.test.tsx.snap +180 -0
- package/components/section/layouts/__tests__/__snapshots__/SectionTitle.test.tsx.snap +354 -0
- package/components/section/layouts/gridUtils.ts +21 -0
- package/components/section/layouts/index.tsx +4 -0
- package/components/section/layouts/types.ts +13 -0
- package/components/section/pageBlock.tsx +23 -0
- package/components/section/sectionUtils.ts +61 -0
- package/components/teaser/index.test.tsx +50 -0
- package/components/teaser/index.tsx +64 -0
- package/components/teaser/teaserVariants.ts +33 -0
- package/components/teaser/variant-types.ts +49 -0
- package/components/teaser/variants/featureVariant.ts +42 -0
- package/components/teaser/variants/horizontal.ts +14 -0
- package/components/teaser/variants/titleTeaserVariant.ts +32 -0
- package/components/teaser/variants/titleVertical.ts +24 -0
- package/components/utils/index.test.ts +12 -0
- package/components/utils/index.ts +6 -0
- package/config/__tests__/index.test.ts +54 -0
- package/config/environment.ts +80 -0
- package/config/index.ts +2 -0
- package/config/multiTenancy.ts +11 -0
- package/constants/index.ts +2 -0
- package/context/app-context/AppContext.test.tsx +56 -0
- package/context/app-context/holidayStopListContextOverrides.ts +18 -0
- package/context/app-context/index.tsx +34 -0
- package/context/app-context/paymentMethodContext.tsx +70 -0
- package/context/index.tsx +2 -0
- package/context/multi-tenancy/MultiTenancy.test.tsx +48 -0
- package/context/multi-tenancy/index.tsx +32 -0
- package/css/index.ts +15 -0
- package/cypress/axe/terminal-log.js +19 -0
- package/cypress/config/visual.config.ts +11 -0
- package/cypress/config/visual.skip.config.ts +10 -0
- package/cypress/e2e/account/accessibility.cy.js +88 -0
- package/cypress/e2e/account/banners.cy.js +169 -0
- package/cypress/e2e/account/cancellation.cy.js +219 -0
- package/cypress/e2e/account/holiday-stops.cy.js +109 -0
- package/cypress/e2e/account/main-api.cy.js +102 -0
- package/cypress/e2e/account/newsletters-and-alerts.cy.js +149 -0
- package/cypress/e2e/account/newsletters-page.cy.js +69 -0
- package/cypress/e2e/account/payment-page.cy.js +98 -0
- package/cypress/e2e/account/personal-details.cy.js +1161 -0
- package/cypress/e2e/account/subscription-and-billing.cy.js +434 -0
- package/cypress/e2e/checkout/account-creation.cy.js +8 -0
- package/cypress/e2e/checkout/payment-details.cy.js +19 -0
- package/cypress/e2e/core/home-page.cy.js +6 -0
- package/cypress/e2e/help-hub/accessibility.cy.js +35 -0
- package/cypress/e2e/help-hub/article-page.cy.js +115 -0
- package/cypress/e2e/help-hub/landing-page.cy.js +94 -0
- package/cypress/e2e/help-hub/result-page.cy.js +117 -0
- package/cypress/fixtures/empty.html +10 -0
- package/cypress/fixtures/example.json +5 -0
- package/cypress/fixtures/holiday-stops.json +40 -0
- package/cypress/fixtures/paymentFailure.js +24 -0
- package/cypress/fixtures/testDates.js +26 -0
- package/cypress/support/commands.js +125 -0
- package/cypress/support/consentValues.js +5839 -0
- package/cypress/support/e2e.js +22 -0
- package/cypress/support/users.js +58 -0
- package/cypress/visual/account/print-visual-regression.cy.js +9 -0
- package/cypress/visual/account/visual-regression.cy.js +40 -0
- package/cypress/visual/empty/empty.cy.js +5 -0
- package/cypress.config.ts +38 -0
- package/helpers/__tests__/getUser.test.ts +45 -0
- package/helpers/__tests__/logger.test.ts +62 -0
- package/helpers/__tests__/useThemeDropdownObject.test.ts +49 -0
- package/helpers/a11y.ts +7 -0
- package/helpers/addCacheHeaders.ts +8 -0
- package/helpers/getUser.ts +37 -0
- package/helpers/getYear.ts +1 -0
- package/helpers/global-types.ts +186 -0
- package/helpers/logger/getWinstonLogger.ts +19 -0
- package/helpers/logger/index.ts +20 -0
- package/helpers/logger/replaceConsoleWithLogger.ts +14 -0
- package/helpers/mocks/articleMock.ts +50 -0
- package/helpers/mocks/getPageMock.ts +369 -0
- package/helpers/mocks/getRadioPostMock.ts +71 -0
- package/helpers/mocks/getRadioPostsMock.ts +26 -0
- package/helpers/mocks/getRecommendationsMock.ts +20 -0
- package/helpers/mocks/index.ts +7 -0
- package/helpers/multiTenancy.ts +19 -0
- package/helpers/setupTests.ts +5 -0
- package/helpers/test-utils.tsx +33 -0
- package/helpers/useThemeDropdownObject.tsx +73 -0
- package/infrastructure/.circleci/config.yml +1187 -0
- package/infrastructure/INFRASTRUCTURE.md +142 -0
- package/infrastructure/build_and_deploy.png +0 -0
- package/infrastructure/helm/Chart.yaml +21 -0
- package/infrastructure/helm/templates/_helpers.tpl +87 -0
- package/infrastructure/helm/templates/deployment.yaml +45 -0
- package/infrastructure/helm/templates/horizontalpodautoscaler.yaml +21 -0
- package/infrastructure/helm/templates/ingress.yaml +31 -0
- package/infrastructure/helm/templates/networkpolicy.yaml +26 -0
- package/infrastructure/helm/templates/secret.yaml +12 -0
- package/infrastructure/helm/templates/service.yaml +15 -0
- package/infrastructure/helm/values-dev.yaml +39 -0
- package/infrastructure/helm/values-local.yaml +30 -0
- package/infrastructure/helm/values-pr.yaml +40 -0
- package/infrastructure/helm/values-prod.yaml +39 -0
- package/infrastructure/helm/values-staging.yaml +39 -0
- package/infrastructure/helm/values.yaml +44 -0
- package/infrastructure/pull_request.png +0 -0
- package/infrastructure/release-documentation-cli.config.json +18 -0
- package/infrastructure/remove_pr.png +0 -0
- package/infrastructure/terraform-ecr/dev.tfvars +3 -0
- package/infrastructure/terraform-ecr/ecr.tf +69 -0
- package/infrastructure/terraform-ecr/main.tf +3 -0
- package/infrastructure/terraform-ecr/variables.tf +19 -0
- package/infrastructure/terraform-ecr/versions.tf +12 -0
- package/infrastructure/terraform-newrelic/alerts.tf +290 -0
- package/infrastructure/terraform-newrelic/dashboards.tf +347 -0
- package/infrastructure/terraform-newrelic/dev.tfvars +3 -0
- package/infrastructure/terraform-newrelic/lighthouse-script-dev.js +53 -0
- package/infrastructure/terraform-newrelic/lighthouse-script-prod.js +54 -0
- package/infrastructure/terraform-newrelic/lighthouse-script-stage.js +54 -0
- package/infrastructure/terraform-newrelic/main.tf +11 -0
- package/infrastructure/terraform-newrelic/monitors.tf +11 -0
- package/infrastructure/terraform-newrelic/prod.tfvars +3 -0
- package/infrastructure/terraform-newrelic/staging.tfvars +3 -0
- package/infrastructure/terraform-newrelic/variables.tf +33 -0
- package/infrastructure/terraform-newrelic/versions.tf +18 -0
- package/jest.config.js +40 -0
- package/jest.config.pact.js +18 -0
- package/newrelic.js +86 -0
- package/next-env.d.ts +5 -0
- package/next.config.js +95 -0
- package/package.json +127 -0
- package/pages/_app.tsx +117 -0
- package/pages/_document.tsx +112 -0
- package/pages/_error.tsx +68 -0
- package/pages/account/add/[field].tsx +34 -0
- package/pages/account/cancellation/index.tsx +22 -0
- package/pages/account/delete/confirm/index.tsx +16 -0
- package/pages/account/delete/error/index.tsx +16 -0
- package/pages/account/delete/index.tsx +16 -0
- package/pages/account/delete/success/index.tsx +16 -0
- package/pages/account/dream-team/index.tsx +19 -0
- package/pages/account/edit/[field].tsx +45 -0
- package/pages/account/family/index.tsx +19 -0
- package/pages/account/holiday-stop/index.tsx +19 -0
- package/pages/account/holiday-stop/previous-holiday-stops.tsx +19 -0
- package/pages/account/holiday-stop/upcoming-holiday-stops.tsx +19 -0
- package/pages/account/index.tsx +33 -0
- package/pages/account/newsletters/index.tsx +19 -0
- package/pages/account/newsletters-and-alerts/index.tsx +24 -0
- package/pages/account/payment/[paymentMethod].tsx +37 -0
- package/pages/account/payment/index.tsx +23 -0
- package/pages/account/payment-methods/index.tsx +21 -0
- package/pages/account/subscription-and-billing/index.tsx +29 -0
- package/pages/api/account/get-session/index.ts +33 -0
- package/pages/api/account/invoice/index.ts +35 -0
- package/pages/api/account/mutate/index.ts +8 -0
- package/pages/api/account/query/index.ts +7 -0
- package/pages/api/auth/[...nextauth].ts +11 -0
- package/pages/api/competitions-sitemap.ts +19 -0
- package/pages/api/feed.ts +23 -0
- package/pages/api/health-check.ts +7 -0
- package/pages/api/news-sitemap.ts +18 -0
- package/pages/api/pages-sitemap.ts +27 -0
- package/pages/api/robots.ts +19 -0
- package/pages/api/sitemap.ts +24 -0
- package/pages/checkout/account-creation/index.tsx +13 -0
- package/pages/checkout/payment-details/index.tsx +10 -0
- package/pages/empty.tsx +6 -0
- package/pages/help-hub/article/[title]/index.tsx +25 -0
- package/pages/help-hub/error.tsx +23 -0
- package/pages/help-hub/index.tsx +51 -0
- package/pages/help-hub/results.tsx +25 -0
- package/pages/index.tsx +19 -0
- package/pages/player/brightcove.tsx +19 -0
- package/pages/rss/create/index.tsx +18 -0
- package/pages/rss/feed/[feedId]/index.tsx +21 -0
- package/pages/rss/index.tsx +18 -0
- package/public/ads.min.js +8175 -0
- package/public/assets/display-base-header.svg +25 -0
- package/public/assets/display-personalDetails-header-sun.svg +55 -0
- package/public/assets/display-personalDetails-header-times.svg +90 -0
- package/public/assets/display-personalDetails-header-vr.svg +46 -0
- package/public/assets/dream-team.svg +10 -0
- package/public/assets/navigationPrimary-brandMark-sun.svg +9 -0
- package/public/assets/navigationPrimary-brandMark-times.svg +18 -0
- package/public/assets/navigationPrimary-brandMark-vr.svg +10 -0
- package/public/assets/newsletters/bestoftimes-101.png +0 -0
- package/public/assets/newsletters/books-115.png +0 -0
- package/public/assets/newsletters/environment-140.png +0 -0
- package/public/assets/newsletters/fashion-108.png +0 -0
- package/public/assets/newsletters/food-106.png +0 -0
- package/public/assets/newsletters/football-111.png +0 -0
- package/public/assets/newsletters/ireland-152.png +0 -0
- package/public/assets/newsletters/law-121.png +0 -0
- package/public/assets/newsletters/money-107.png +0 -0
- package/public/assets/newsletters/moneymentor-150.png +0 -0
- package/public/assets/newsletters/placeholder.png +0 -0
- package/public/assets/newsletters/politics-119.png +0 -0
- package/public/assets/newsletters/property-113.png +0 -0
- package/public/assets/newsletters/puzzles-125.png +0 -0
- package/public/assets/newsletters/scotland-134.png +0 -0
- package/public/assets/newsletters/theatre-127.png +0 -0
- package/public/assets/newsletters/travel-149.png +0 -0
- package/public/assets/newsletters/tv-133.png +0 -0
- package/public/assets/newsletters/us-153.png +0 -0
- package/public/assets/no-subscription.svg +15 -0
- package/public/assets/pending-activation.svg +16 -0
- package/public/assets/personal-details-header.svg +91 -0
- package/public/assets/plchldr150x100.png +0 -0
- package/public/assets/previous-subscription.svg +15 -0
- package/public/assets/primary-navigation-logo-white.svg +6 -0
- package/public/assets/primary-navigation-logo.svg +6 -0
- package/public/assets/tls-logo-white.svg +3 -0
- package/public/favicon.ico +0 -0
- package/public/icon.png +0 -0
- package/public/prebid.min.js +1 -0
- package/queries/getPage.ts +100 -0
- package/queries/getRadioPost.ts +61 -0
- package/queries/getRadioPosts.ts +16 -0
- package/queries/getUserSubscription.ts +9 -0
- package/queries/index.ts +3 -0
- package/scripts/k8s-local.sh +137 -0
- package/scripts/nr-exports.sh +7 -0
- package/temp/_app.tsx +15 -0
- package/temp/_document.tsx +61 -0
- package/temp/app-context-test.tsx +51 -0
- package/temp/app-context.tsx +25 -0
- package/temp/config-index.ts +1 -0
- package/temp/context-index.tsx +1 -0
- package/temp/header.tsx +45 -0
- package/temp/next.config.js +56 -0
- package/tsconfig.json +51 -0
- package/tsconfig.test.json +6 -0
- package/validation/index.tsx +24 -0
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import CollectionBlock from '../CollectionBlock'
|
|
3
|
+
import { getPageMock } from '../../../helpers/mocks'
|
|
4
|
+
import { CollectionBlock as Collection } from '../../../helpers/global-types'
|
|
5
|
+
import { renderWithTheme } from '../../../helpers/test-utils'
|
|
6
|
+
import { SectionContext } from '../SectionContext'
|
|
7
|
+
|
|
8
|
+
jest.mock('../ArticleSlice', () => () => <div data-testid="article-slice" />)
|
|
9
|
+
|
|
10
|
+
describe('CollectionBlock', () => {
|
|
11
|
+
test('renders section title with articles', () => {
|
|
12
|
+
const { getByTestId, getAllByTestId } = renderWithTheme(
|
|
13
|
+
SectionContext.Provider,
|
|
14
|
+
{
|
|
15
|
+
value: {
|
|
16
|
+
isIndexPage: false,
|
|
17
|
+
},
|
|
18
|
+
children: (
|
|
19
|
+
<CollectionBlock collection={getPageMock.page.body[0]} index={1} />
|
|
20
|
+
),
|
|
21
|
+
}
|
|
22
|
+
)
|
|
23
|
+
|
|
24
|
+
expect(getByTestId('SectionTitleBar')).toBeInTheDocument()
|
|
25
|
+
expect(getAllByTestId('article-slice').length).toEqual(6)
|
|
26
|
+
})
|
|
27
|
+
|
|
28
|
+
test('should not render section title', () => {
|
|
29
|
+
const { queryByTestId } = renderWithTheme(SectionContext.Provider, {
|
|
30
|
+
value: {
|
|
31
|
+
isIndexPage: true,
|
|
32
|
+
},
|
|
33
|
+
children: (
|
|
34
|
+
<CollectionBlock collection={getPageMock.page.body[0]} index={1} />
|
|
35
|
+
),
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
expect(queryByTestId('SectionTitleBar')).not.toBeInTheDocument()
|
|
39
|
+
})
|
|
40
|
+
|
|
41
|
+
test('should not render section title when index is 0', () => {
|
|
42
|
+
const { queryByTestId } = renderWithTheme(SectionContext.Provider, {
|
|
43
|
+
value: {
|
|
44
|
+
isIndexPage: false,
|
|
45
|
+
},
|
|
46
|
+
children: (
|
|
47
|
+
<CollectionBlock collection={getPageMock.page.body[0]} index={0} />
|
|
48
|
+
),
|
|
49
|
+
})
|
|
50
|
+
|
|
51
|
+
expect(queryByTestId('SectionTitleBar')).not.toBeInTheDocument()
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
test('should handle topic-author-slice', () => {
|
|
55
|
+
const topic = {
|
|
56
|
+
...getPageMock.page.body[0],
|
|
57
|
+
}
|
|
58
|
+
topic.children[0].type = 'topic-author-slice'
|
|
59
|
+
topic.children[1].type = 'topic-author-slice'
|
|
60
|
+
|
|
61
|
+
const { getAllByTestId } = renderWithTheme(CollectionBlock, {
|
|
62
|
+
collection: topic as Collection,
|
|
63
|
+
index: 1,
|
|
64
|
+
})
|
|
65
|
+
|
|
66
|
+
expect(getAllByTestId('article-slice').length).toEqual(4)
|
|
67
|
+
})
|
|
68
|
+
|
|
69
|
+
test('should handle topic-author-slice', () => {
|
|
70
|
+
const topic = {
|
|
71
|
+
...getPageMock.page.body[0],
|
|
72
|
+
}
|
|
73
|
+
topic.children[0].type = 'unknown'
|
|
74
|
+
topic.children[1].type = 'unknown'
|
|
75
|
+
|
|
76
|
+
const { getAllByTestId } = renderWithTheme(CollectionBlock, {
|
|
77
|
+
collection: topic as Collection,
|
|
78
|
+
index: 1,
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
expect(getAllByTestId('article-slice').length).toEqual(4)
|
|
82
|
+
})
|
|
83
|
+
})
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import pageBlock from '../pageBlock'
|
|
3
|
+
import { getPageMock } from '../../../helpers/mocks'
|
|
4
|
+
import CollectionBlock from '../CollectionBlock'
|
|
5
|
+
|
|
6
|
+
jest.mock('../CollectionBlock', () => () => (
|
|
7
|
+
<div data-testid="collectionBlock" />
|
|
8
|
+
))
|
|
9
|
+
|
|
10
|
+
describe('test pageBlock', () => {
|
|
11
|
+
const collection = getPageMock.page.body[0]
|
|
12
|
+
test('pageBlock returns CollectionBlock', () => {
|
|
13
|
+
expect(pageBlock('collection')(collection, 0)).toStrictEqual(
|
|
14
|
+
<CollectionBlock collection={collection} index={0} />
|
|
15
|
+
)
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
test('pageBlock returns null', () => {
|
|
19
|
+
expect(pageBlock('image')(collection)).toBe(null)
|
|
20
|
+
expect(pageBlock('paragraph')(collection)).toBe(null)
|
|
21
|
+
expect(pageBlock('tweet')(collection)).toBe(null)
|
|
22
|
+
expect(pageBlock('doesNotExist')(collection)).toBe(null)
|
|
23
|
+
})
|
|
24
|
+
})
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { Video } from '../../../helpers/global-types'
|
|
2
|
+
import { teaserURL, teaserSummary, getMedia } from '../sectionUtils'
|
|
3
|
+
|
|
4
|
+
describe('Section Utils', () => {
|
|
5
|
+
test('teaserURL', () => {
|
|
6
|
+
expect(
|
|
7
|
+
teaserURL({
|
|
8
|
+
id: 'demo-1',
|
|
9
|
+
headline: 'Something Something',
|
|
10
|
+
categories: [
|
|
11
|
+
{
|
|
12
|
+
slug: 'catagory-one',
|
|
13
|
+
},
|
|
14
|
+
],
|
|
15
|
+
})
|
|
16
|
+
).toBe('catagory-one/demo-1/something-something')
|
|
17
|
+
expect(teaserURL({})).toBe('uncategorized')
|
|
18
|
+
expect(
|
|
19
|
+
teaserURL({
|
|
20
|
+
categories: [
|
|
21
|
+
{
|
|
22
|
+
slug: 'catagory-one',
|
|
23
|
+
},
|
|
24
|
+
],
|
|
25
|
+
})
|
|
26
|
+
).toBe('catagory-one')
|
|
27
|
+
expect(
|
|
28
|
+
teaserURL({
|
|
29
|
+
id: 'demo-1',
|
|
30
|
+
headline: 'Something Something',
|
|
31
|
+
categories: [
|
|
32
|
+
// @ts-ignore
|
|
33
|
+
{},
|
|
34
|
+
],
|
|
35
|
+
})
|
|
36
|
+
).toBe('uncategorized/demo-1/something-something')
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
test('teaserSummary', () => {
|
|
40
|
+
const summary = {
|
|
41
|
+
children: [
|
|
42
|
+
{
|
|
43
|
+
text: 'one',
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
text: 'two',
|
|
47
|
+
},
|
|
48
|
+
],
|
|
49
|
+
}
|
|
50
|
+
expect(teaserSummary(summary)).toBe('one')
|
|
51
|
+
expect(teaserSummary({})).toBe('')
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
test('getMedia', () => {
|
|
55
|
+
expect(getMedia(undefined)).toEqual({
|
|
56
|
+
src: '',
|
|
57
|
+
alt: '',
|
|
58
|
+
loadingAspectRatio: '',
|
|
59
|
+
})
|
|
60
|
+
|
|
61
|
+
const image = {
|
|
62
|
+
crops: [
|
|
63
|
+
{
|
|
64
|
+
url: 'image-src',
|
|
65
|
+
alt: 'image-alt',
|
|
66
|
+
aspectRatio: '1:1',
|
|
67
|
+
},
|
|
68
|
+
],
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
expect(getMedia(image)).toEqual({
|
|
72
|
+
src: image.crops[0].url,
|
|
73
|
+
alt: image.crops[0].alt,
|
|
74
|
+
loadingAspectRatio: image.crops[0].aspectRatio,
|
|
75
|
+
})
|
|
76
|
+
|
|
77
|
+
const video = {
|
|
78
|
+
videoId: '123',
|
|
79
|
+
posterImage: {
|
|
80
|
+
crop: {
|
|
81
|
+
url: 'image-src',
|
|
82
|
+
alt: 'image-alt',
|
|
83
|
+
aspectRatio: '1:1',
|
|
84
|
+
},
|
|
85
|
+
},
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
expect(getMedia(video as Video)).toEqual({
|
|
89
|
+
src: video.posterImage.crop.url,
|
|
90
|
+
alt: video.posterImage.crop.alt,
|
|
91
|
+
loadingAspectRatio: video.posterImage.crop.aspectRatio,
|
|
92
|
+
})
|
|
93
|
+
})
|
|
94
|
+
})
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { Block, Cell } from 'newskit'
|
|
3
|
+
import { UserData } from '@newskit-render/my-account'
|
|
4
|
+
import { Page } from '../../helpers/global-types'
|
|
5
|
+
import Layout from '../layout'
|
|
6
|
+
import pageBlock from './pageBlock'
|
|
7
|
+
import SectionTitleBar from '../common/SectionTitleBar'
|
|
8
|
+
import { SectionContext } from './SectionContext'
|
|
9
|
+
import { BackToHomepage } from '../common/BackToHomepage'
|
|
10
|
+
|
|
11
|
+
const gridOverride = {
|
|
12
|
+
xsMargin: 'space000',
|
|
13
|
+
xsRowGutter: 'space000',
|
|
14
|
+
xsColumnGutter: 'space000',
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const SectionPage: React.FC<{
|
|
18
|
+
page: Page
|
|
19
|
+
isIndexPage?: boolean
|
|
20
|
+
user?: UserData
|
|
21
|
+
}> = ({ page, user, isIndexPage = false }) => {
|
|
22
|
+
return (
|
|
23
|
+
<SectionContext.Provider value={{ isIndexPage }}>
|
|
24
|
+
<Layout dataTestId="SectionGrid" gridOverride={gridOverride} user={user}>
|
|
25
|
+
<Cell xs={12} md={10} mdOffset={1} data-testid="SectionCell">
|
|
26
|
+
<Block marginBlockEnd="space070" />
|
|
27
|
+
{isIndexPage && <SectionTitleBar title={page.title} />}
|
|
28
|
+
{Array.isArray(page.body) &&
|
|
29
|
+
page.body.map((block, i) =>
|
|
30
|
+
pageBlock(block.type as string)(block, i)
|
|
31
|
+
)}
|
|
32
|
+
{isIndexPage && <BackToHomepage />}
|
|
33
|
+
</Cell>
|
|
34
|
+
</Layout>
|
|
35
|
+
</SectionContext.Provider>
|
|
36
|
+
)
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export default SectionPage
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import {
|
|
3
|
+
ArticleTeaser,
|
|
4
|
+
Media,
|
|
5
|
+
SliceBlock,
|
|
6
|
+
TeaserSummary,
|
|
7
|
+
} from '../../../helpers/global-types'
|
|
8
|
+
import { teaserURL, teaserSummary, getMedia } from '../sectionUtils'
|
|
9
|
+
import Teaser from '../../teaser'
|
|
10
|
+
import { VariantsName } from '../../teaser/teaserVariants'
|
|
11
|
+
|
|
12
|
+
export const getBlockProps = (sliceBlock: SliceBlock) => {
|
|
13
|
+
// Temp remove later
|
|
14
|
+
const blockTypes = {
|
|
15
|
+
'article-block': 'article' in sliceBlock && sliceBlock.article,
|
|
16
|
+
'external-reference': {
|
|
17
|
+
headline: 'TODO - External Reference',
|
|
18
|
+
id: 'author' in sliceBlock && sliceBlock.author.id,
|
|
19
|
+
},
|
|
20
|
+
'video-brightcove': {
|
|
21
|
+
headline: 'TODO - Video',
|
|
22
|
+
id: 'accountId' in sliceBlock && sliceBlock.id,
|
|
23
|
+
},
|
|
24
|
+
default: {
|
|
25
|
+
headline: `Unknown slice block - ${sliceBlock.type}`,
|
|
26
|
+
},
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
return blockTypes[sliceBlock.type] || blockTypes.default
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export const getBlock = (
|
|
33
|
+
sliceBlock: SliceBlock,
|
|
34
|
+
variant: VariantsName = 'base'
|
|
35
|
+
) => {
|
|
36
|
+
const { headline, summary, media } = ('article' in sliceBlock &&
|
|
37
|
+
sliceBlock.article) as ArticleTeaser
|
|
38
|
+
const blockTypes = {
|
|
39
|
+
'article-block': (
|
|
40
|
+
<Teaser
|
|
41
|
+
variant={variant}
|
|
42
|
+
url={teaserURL(
|
|
43
|
+
('article' in sliceBlock && sliceBlock.article) as ArticleTeaser
|
|
44
|
+
)}
|
|
45
|
+
media={getMedia(media as Media)}
|
|
46
|
+
title={headline as string}
|
|
47
|
+
teaser={teaserSummary(summary as TeaserSummary)}
|
|
48
|
+
/>
|
|
49
|
+
),
|
|
50
|
+
'external-reference': (
|
|
51
|
+
<Teaser
|
|
52
|
+
variant={variant}
|
|
53
|
+
media={{
|
|
54
|
+
src: 'https://plchldr.co/i/802x451?bg=F0F0F0&fc=111111&text=img',
|
|
55
|
+
}}
|
|
56
|
+
title="TODO - External Reference"
|
|
57
|
+
/>
|
|
58
|
+
),
|
|
59
|
+
'video-brightcove': (
|
|
60
|
+
<Teaser
|
|
61
|
+
variant={variant}
|
|
62
|
+
media={{
|
|
63
|
+
src: 'https://plchldr.co/i/802x451?bg=F0F0F0&fc=111111&text=img',
|
|
64
|
+
}}
|
|
65
|
+
title="TODO - Video"
|
|
66
|
+
/>
|
|
67
|
+
),
|
|
68
|
+
default: (
|
|
69
|
+
<Teaser
|
|
70
|
+
variant={variant}
|
|
71
|
+
media={{
|
|
72
|
+
src: 'https://plchldr.co/i/802x451?bg=F0F0F0&fc=111111&text=img',
|
|
73
|
+
}}
|
|
74
|
+
title={`Unknown slice block - ${sliceBlock.type}`}
|
|
75
|
+
/>
|
|
76
|
+
),
|
|
77
|
+
}
|
|
78
|
+
return blockTypes[sliceBlock.type] || blockTypes.default
|
|
79
|
+
}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { NextLink } from '@newskit-render/shared-components'
|
|
3
|
+
import { getBlockProps } from './Block'
|
|
4
|
+
import { teaserURL } from '../sectionUtils'
|
|
5
|
+
import { LayoutProps } from './types'
|
|
6
|
+
|
|
7
|
+
// Fallback for unimplemented layouts. Remove once all have been implemented.
|
|
8
|
+
|
|
9
|
+
export const FallBack: React.FC<LayoutProps> = ({ slice }) => (
|
|
10
|
+
<div className={slice.name}>
|
|
11
|
+
<h2>Layout: {slice.name}</h2>
|
|
12
|
+
{slice.children.map((block, i) => {
|
|
13
|
+
const teaserProps = getBlockProps(block)
|
|
14
|
+
const { id, headline } = teaserProps
|
|
15
|
+
return (
|
|
16
|
+
<p key={id || i}>
|
|
17
|
+
<NextLink href={teaserURL(teaserProps)} type="standalone">
|
|
18
|
+
{headline}
|
|
19
|
+
</NextLink>
|
|
20
|
+
</p>
|
|
21
|
+
)
|
|
22
|
+
})}
|
|
23
|
+
</div>
|
|
24
|
+
)
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { GridLayout, Block } from 'newskit'
|
|
3
|
+
import { outerGridOverride, innerGridOverride } from './gridUtils'
|
|
4
|
+
import { getBlock } from './Block'
|
|
5
|
+
import { LayoutProps } from './types'
|
|
6
|
+
|
|
7
|
+
export const Lead: React.FC<LayoutProps> = ({ slice }) => (
|
|
8
|
+
<GridLayout
|
|
9
|
+
{...outerGridOverride}
|
|
10
|
+
columns={{ lg: '2fr 1fr' }}
|
|
11
|
+
columnGap="space050"
|
|
12
|
+
data-testid={`${slice.name}-Grid`}
|
|
13
|
+
>
|
|
14
|
+
<Block data-testid="featureVerticalCell">
|
|
15
|
+
{getBlock(slice.children[0], 'featureVertical')}
|
|
16
|
+
</Block>
|
|
17
|
+
<GridLayout
|
|
18
|
+
columnGap={{ md: '20px' }}
|
|
19
|
+
columns={{ xs: '1fr', md: '1fr 1fr', lg: '1fr' }}
|
|
20
|
+
justifyContent={{ lg: 'stretch' }}
|
|
21
|
+
alignContent={{ lg: 'space-between' }}
|
|
22
|
+
overrides={{ height: '100%' }}
|
|
23
|
+
{...innerGridOverride}
|
|
24
|
+
>
|
|
25
|
+
{slice.children.slice(1).map((block, i) => {
|
|
26
|
+
return (
|
|
27
|
+
<Block
|
|
28
|
+
key={`titleVerticalCell-${i}`}
|
|
29
|
+
data-testid={`titleVerticalCell-${i}`}
|
|
30
|
+
>
|
|
31
|
+
{getBlock(block, 'horizontal')}
|
|
32
|
+
</Block>
|
|
33
|
+
)
|
|
34
|
+
})}
|
|
35
|
+
</GridLayout>
|
|
36
|
+
</GridLayout>
|
|
37
|
+
)
|
|
38
|
+
|
|
39
|
+
export const LeadFullWidth: React.FC<LayoutProps> = ({ slice }) => (
|
|
40
|
+
<Block data-testid={`${slice.name}-Grid`}>
|
|
41
|
+
{getBlock(slice.children[0], 'featureVertical')}
|
|
42
|
+
</Block>
|
|
43
|
+
)
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { Block, GridLayout } from 'newskit'
|
|
3
|
+
import { ArticleRecommendation } from '@newskit-render/standalone-components'
|
|
4
|
+
import { LayoutProps } from './types'
|
|
5
|
+
import { outerGridOverride } from './gridUtils'
|
|
6
|
+
import { getBlock } from './Block'
|
|
7
|
+
|
|
8
|
+
export const BasicRow: React.FC<LayoutProps> = ({
|
|
9
|
+
slice,
|
|
10
|
+
articles,
|
|
11
|
+
colums,
|
|
12
|
+
variant = 'titleTeaserVertical',
|
|
13
|
+
'data-testid': testid,
|
|
14
|
+
}) => {
|
|
15
|
+
const dataTestid =
|
|
16
|
+
(testid && testid) || (slice && `${slice.name}-Grid`) || 'Articles-Grid'
|
|
17
|
+
return (
|
|
18
|
+
<GridLayout
|
|
19
|
+
columns={colums}
|
|
20
|
+
columnGap="space050"
|
|
21
|
+
data-testid={dataTestid}
|
|
22
|
+
{...outerGridOverride}
|
|
23
|
+
>
|
|
24
|
+
{articles
|
|
25
|
+
? articles.map((article, i) => (
|
|
26
|
+
<Block
|
|
27
|
+
marginBlockEnd="space050"
|
|
28
|
+
key={article.title}
|
|
29
|
+
data-testid={`articleRec-${i}`}
|
|
30
|
+
>
|
|
31
|
+
<ArticleRecommendation article={article} size="large" />
|
|
32
|
+
</Block>
|
|
33
|
+
))
|
|
34
|
+
: slice.children.map((block, i) => (
|
|
35
|
+
<Block
|
|
36
|
+
key={'article' in block ? block.article.id : i}
|
|
37
|
+
data-testid={`titleTeaserVertical-${i}`}
|
|
38
|
+
>
|
|
39
|
+
{getBlock(block, variant)}
|
|
40
|
+
</Block>
|
|
41
|
+
))}
|
|
42
|
+
</GridLayout>
|
|
43
|
+
)
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export const Row3: React.FC<LayoutProps> = ({ slice }) => (
|
|
47
|
+
<GridLayout
|
|
48
|
+
areas={{
|
|
49
|
+
xs: `
|
|
50
|
+
"A"
|
|
51
|
+
"B"
|
|
52
|
+
"C"
|
|
53
|
+
`,
|
|
54
|
+
md: `
|
|
55
|
+
"A B"
|
|
56
|
+
"A C"
|
|
57
|
+
`,
|
|
58
|
+
lg: `
|
|
59
|
+
"A B C"
|
|
60
|
+
`,
|
|
61
|
+
}}
|
|
62
|
+
rowGap={{ xs: 'space010', md: 'space040' }}
|
|
63
|
+
columnGap={{ md: 'space050', lg: 'space050' }}
|
|
64
|
+
data-testid={`${slice.name}-Grid`}
|
|
65
|
+
>
|
|
66
|
+
{(Areas) =>
|
|
67
|
+
Object.entries(Areas).map((Area, i) => {
|
|
68
|
+
const AreaComp = Area[1]
|
|
69
|
+
const block = slice.children[i]
|
|
70
|
+
|
|
71
|
+
return (
|
|
72
|
+
<AreaComp
|
|
73
|
+
key={('article' in block && block.article.id) || i}
|
|
74
|
+
data-testid={`titleTeaserVertical-${i}`}
|
|
75
|
+
>
|
|
76
|
+
{getBlock(block, 'horizontal')}
|
|
77
|
+
</AreaComp>
|
|
78
|
+
)
|
|
79
|
+
})
|
|
80
|
+
}
|
|
81
|
+
</GridLayout>
|
|
82
|
+
)
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { NextLink } from '@newskit-render/shared-components'
|
|
3
|
+
import { Block, Divider, getColorCssFromTheme, styled } from 'newskit'
|
|
4
|
+
import { LayoutProps } from './types'
|
|
5
|
+
import StyledIconFilledChevronRight from '../../common/icons/StyledIconFilledChevronRight'
|
|
6
|
+
import SectionTitleBar from '../../common/SectionTitleBar'
|
|
7
|
+
|
|
8
|
+
const link = (href: string, stylePreset?: string) => (
|
|
9
|
+
<NextLink
|
|
10
|
+
type="standalone"
|
|
11
|
+
href={href}
|
|
12
|
+
overrides={{
|
|
13
|
+
stylePreset: stylePreset || 'inkInverse',
|
|
14
|
+
typographyPreset: 'utilityButton020',
|
|
15
|
+
spaceInline: 'space020',
|
|
16
|
+
}}
|
|
17
|
+
>
|
|
18
|
+
Link
|
|
19
|
+
<StyledIconFilledChevronRight overrides={{ size: 'iconSize020' }} />
|
|
20
|
+
</NextLink>
|
|
21
|
+
)
|
|
22
|
+
|
|
23
|
+
const StyledDivider = styled(Divider)`
|
|
24
|
+
${getColorCssFromTheme('background', 'transparent')};
|
|
25
|
+
${getColorCssFromTheme('borderColor', 'transparent')};
|
|
26
|
+
`
|
|
27
|
+
|
|
28
|
+
type SectionTitleProps = {
|
|
29
|
+
title: string
|
|
30
|
+
titleBarStylePreset?: string
|
|
31
|
+
titleBarColour?: string
|
|
32
|
+
addTopSpace?: boolean
|
|
33
|
+
} & LayoutProps
|
|
34
|
+
|
|
35
|
+
export const SectionTitle: React.FC<SectionTitleProps> = ({
|
|
36
|
+
title,
|
|
37
|
+
sectionURL,
|
|
38
|
+
titleBarStylePreset,
|
|
39
|
+
titleBarColour,
|
|
40
|
+
addTopSpace,
|
|
41
|
+
}) => {
|
|
42
|
+
return (
|
|
43
|
+
<>
|
|
44
|
+
{addTopSpace && (
|
|
45
|
+
<Block marginBlockEnd="space060" data-testid="addTopSpace">
|
|
46
|
+
<StyledDivider />
|
|
47
|
+
</Block>
|
|
48
|
+
)}
|
|
49
|
+
<SectionTitleBar
|
|
50
|
+
title={title}
|
|
51
|
+
actionItem={() => link(sectionURL as string, titleBarStylePreset)}
|
|
52
|
+
stylePreset={titleBarStylePreset}
|
|
53
|
+
colour={titleBarColour}
|
|
54
|
+
/>
|
|
55
|
+
</>
|
|
56
|
+
)
|
|
57
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { renderWithTheme } from '../../../../helpers/test-utils'
|
|
2
|
+
import { Lead, LeadFullWidth } from '../Lead'
|
|
3
|
+
import { getPageMock } from '../../../../helpers/mocks'
|
|
4
|
+
import { ArticleSlice } from '../../../../helpers/global-types'
|
|
5
|
+
|
|
6
|
+
describe('Lead', () => {
|
|
7
|
+
test('should render Lead', () => {
|
|
8
|
+
const { asFragment } = renderWithTheme(Lead, {
|
|
9
|
+
slice: getPageMock.page.body[0].children[0] as ArticleSlice,
|
|
10
|
+
})
|
|
11
|
+
expect(asFragment()).toMatchSnapshot()
|
|
12
|
+
})
|
|
13
|
+
|
|
14
|
+
test('should render Lead with cells', () => {
|
|
15
|
+
const { getAllByTestId, queryByTestId } = renderWithTheme(Lead, {
|
|
16
|
+
slice: getPageMock.page.body[0].children[0] as ArticleSlice,
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
expect(
|
|
20
|
+
queryByTestId('SUPPLEMENT_LEAD_AND_4_STACK-Grid')
|
|
21
|
+
).toBeInTheDocument()
|
|
22
|
+
expect(queryByTestId('featureVerticalCell')).toBeInTheDocument()
|
|
23
|
+
expect(getAllByTestId(/titleVerticalCell-/).length).toEqual(2)
|
|
24
|
+
})
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
describe('Lead full width', () => {
|
|
28
|
+
test('should render Lead full width', () => {
|
|
29
|
+
const { queryByTestId } = renderWithTheme(LeadFullWidth, {
|
|
30
|
+
slice: getPageMock.page.body[0].children[0] as ArticleSlice,
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
expect(
|
|
34
|
+
queryByTestId('SUPPLEMENT_LEAD_AND_4_STACK-Grid')
|
|
35
|
+
).toBeInTheDocument()
|
|
36
|
+
})
|
|
37
|
+
})
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { renderWithTheme } from '../../../../helpers/test-utils'
|
|
2
|
+
import { BasicRow, Row3 } from '../Rows'
|
|
3
|
+
import { getPageMock } from '../../../../helpers/mocks'
|
|
4
|
+
import { ArticleSlice } from '../../../../helpers/global-types'
|
|
5
|
+
import { relatedArticles } from '../../../../helpers/mocks/articleMock'
|
|
6
|
+
|
|
7
|
+
describe('Rows', () => {
|
|
8
|
+
test('should render a row with 2 columns', () => {
|
|
9
|
+
const slice = {
|
|
10
|
+
...getPageMock.page.body[0].children[0],
|
|
11
|
+
children: getPageMock.page.body[0].children[0].children.slice(0, 2),
|
|
12
|
+
}
|
|
13
|
+
const { getAllByTestId } = renderWithTheme(BasicRow, {
|
|
14
|
+
slice: slice as ArticleSlice,
|
|
15
|
+
colums: { xs: '1fr', md: '1fr 1fr' },
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
expect(getAllByTestId(/titleTeaserVertical-/).length).toEqual(2)
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
test('should render a row with 3 columns', () => {
|
|
22
|
+
const { getAllByTestId } = renderWithTheme(Row3, {
|
|
23
|
+
slice: getPageMock.page.body[0].children[0] as ArticleSlice,
|
|
24
|
+
colums: { xs: '1fr', md: '1fr 1fr' },
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
expect(getAllByTestId(/titleTeaserVertical-/).length).toEqual(3)
|
|
28
|
+
})
|
|
29
|
+
|
|
30
|
+
describe('Recommended articles', () => {
|
|
31
|
+
test('should render with recommended articles', () => {
|
|
32
|
+
const { getAllByTestId } = renderWithTheme(BasicRow, {
|
|
33
|
+
articles: relatedArticles,
|
|
34
|
+
colums: { xs: '1fr', md: '1fr 1fr', lg: '1fr 1fr 1fr' },
|
|
35
|
+
})
|
|
36
|
+
|
|
37
|
+
expect(getAllByTestId(/articleRec-/).length).toEqual(4)
|
|
38
|
+
})
|
|
39
|
+
})
|
|
40
|
+
})
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { renderWithTheme } from '../../../../helpers/test-utils'
|
|
2
|
+
import { SectionTitle } from '../SectionTitle'
|
|
3
|
+
import { getPageMock } from '../../../../helpers/mocks'
|
|
4
|
+
import { ArticleSlice } from '../../../../helpers/global-types'
|
|
5
|
+
|
|
6
|
+
describe('SectionTitle', () => {
|
|
7
|
+
test('SectionTitle with title', () => {
|
|
8
|
+
const { asFragment } = renderWithTheme(SectionTitle, {
|
|
9
|
+
slice: getPageMock.page.body[0].children[1] as ArticleSlice,
|
|
10
|
+
sectionURL: getPageMock.page.body[0].link.url,
|
|
11
|
+
title: getPageMock.page.body[0].title,
|
|
12
|
+
})
|
|
13
|
+
expect(asFragment()).toMatchSnapshot()
|
|
14
|
+
})
|
|
15
|
+
|
|
16
|
+
test('SectionTitle without extra space', () => {
|
|
17
|
+
const document = renderWithTheme(SectionTitle, {
|
|
18
|
+
slice: getPageMock.page.body[0].children[1] as ArticleSlice,
|
|
19
|
+
sectionURL: getPageMock.page.body[0].link.url,
|
|
20
|
+
title: getPageMock.page.body[0].title,
|
|
21
|
+
addTopSpace: true,
|
|
22
|
+
})
|
|
23
|
+
expect(document.getByTestId('addTopSpace')).toBeInTheDocument()
|
|
24
|
+
expect(document).toMatchSnapshot()
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
test('SectionTitle alternative titlebar', () => {
|
|
28
|
+
const { asFragment } = renderWithTheme(SectionTitle, {
|
|
29
|
+
slice: getPageMock.page.body[0].children[1] as ArticleSlice,
|
|
30
|
+
sectionURL: getPageMock.page.body[0].link.url,
|
|
31
|
+
title: getPageMock.page.body[0].title,
|
|
32
|
+
titleBarStylePreset: 'inkContrast',
|
|
33
|
+
titleBarColour: 'transparent',
|
|
34
|
+
})
|
|
35
|
+
expect(asFragment()).toMatchSnapshot()
|
|
36
|
+
})
|
|
37
|
+
})
|