diff --git a/.eslintignore b/.eslintignore index 70368638..d6c991a5 100644 --- a/.eslintignore +++ b/.eslintignore @@ -1,4 +1,5 @@ .next/* node_modules out/* -storybook-static/* \ No newline at end of file +storybook-static/* +lib/api/noirlab/codegen/* diff --git a/app/[locale]/[...uriSegments]/page.tsx b/app/[locale]/[...uriSegments]/page.tsx index 8cd55db8..0ae71fa8 100644 --- a/app/[locale]/[...uriSegments]/page.tsx +++ b/app/[locale]/[...uriSegments]/page.tsx @@ -2,7 +2,6 @@ import { FunctionComponent } from "react"; import { Metadata, ResolvingMetadata } from "next"; import { draftMode } from "next/headers"; import { notFound, redirect } from "next/navigation"; -import { OpenGraph } from "next/dist/lib/metadata/types/opengraph-types"; import striptags from "striptags"; import { getBreadcrumbsById, getEntryMetadataByUri } from "@/lib/api/metadata"; import { @@ -10,8 +9,8 @@ import { getEntrySectionByUri, } from "@/lib/api/entries/index"; import { getEntryDataByUri } from "@/lib/api/entry"; -import { getReleaseOpenGraph } from "@/lib/api/noirlab"; -import { resizeCantoImage } from "@/lib/api/canto/resize"; +import { generateReleaseMetadata } from "@/lib/api/noirlab"; +import { pickFeaturedImage } from "@/lib/helpers/metadata"; import PageTemplate from "@/components/templates/Page"; import NewsPageTemplate from "@/components/templates/NewsPage"; import GlossaryPageTemplate from "@/components/templates/GlossaryPage"; @@ -20,39 +19,6 @@ import SlideshowPageTemplate from "@/components/templates/SlideshowPage"; import StaffPageTemplate from "@/components/templates/StaffPage"; import EventPageTemplate from "@/components/templates/EventPage"; -const pickFeaturedImage = async ( - locale: string, - image?: any, - cantoAsset?: any, - pressReleaseId?: string -): Promise => { - if (pressReleaseId) { - const releaseOpenGraphImage = await getReleaseOpenGraph( - locale, - pressReleaseId - ); - - return releaseOpenGraphImage; - } - - if (cantoAsset) { - const { - width, - height, - url: { directUrlPreview }, - } = cantoAsset; - const url = resizeCantoImage(directUrlPreview, 800); - - return { url, width, height }; - } - - if (image) { - const { url, width, height, altText: alt } = image; - - return { url, width, height, alt }; - } -}; - export async function generateMetadata( { params: { locale, uriSegments }, searchParams = {} }: UriSegmentProps, parent: ResolvingMetadata @@ -80,14 +46,14 @@ export async function generateMetadata( pressReleaseId, } = entry; - const featuredImage = await pickFeaturedImage( - locale, - image[0], - cantoAssetSingle[0], - pressReleaseId - ); const previousImages = (await parent).openGraph?.images || []; + if (pressReleaseId) { + return generateReleaseMetadata(pressReleaseId, locale); + } + + const featuredImage = await pickFeaturedImage(image[0], cantoAssetSingle[0]); + return { title, description: striptags(description), diff --git a/app/[locale]/page.tsx b/app/[locale]/page.tsx index d4d1c3d3..030ab00b 100644 --- a/app/[locale]/page.tsx +++ b/app/[locale]/page.tsx @@ -1,14 +1,15 @@ import { FunctionComponent } from "react"; -import { Metadata } from "next"; +import { Metadata, ResolvingMetadata } from "next"; import { notFound } from "next/navigation"; import { draftMode } from "next/headers"; +import { pickFeaturedImage } from "@/lib/helpers/metadata"; import { getHomepage, getHomepageMetadata } from "@/lib/api/homepage"; import HomePageTemplate from "@/templates/HomePage"; -export async function generateMetadata({ - params: { locale }, - searchParams = {}, -}: LocaleProps): Promise { +export async function generateMetadata( + { params: { locale }, searchParams = {} }: LocaleProps, + parent: ResolvingMetadata +): Promise { let previewToken: string | undefined; if (draftMode().isEnabled) { @@ -17,11 +18,22 @@ export async function generateMetadata({ : searchParams?.preview; } + const { openGraph } = await parent; + const { - entry: { title, description }, + entry: { title, description, hero, cantoHero }, } = await getHomepageMetadata(locale, previewToken); - return { title, description }; + const images = + (await pickFeaturedImage(hero[0], cantoHero[0])) || openGraph?.images; + + return { + title, + description, + openGraph: { + images, + }, + }; } const RootPage: FunctionComponent = async ({ @@ -43,7 +55,7 @@ const RootPage: FunctionComponent = async ({ notFound(); } - return ; + return ; }; export default RootPage; diff --git a/components/content-blocks/AccordionGroup/index.js b/components/content-blocks/AccordionGroup/index.js index a87daa5a..97e6194f 100644 --- a/components/content-blocks/AccordionGroup/index.js +++ b/components/content-blocks/AccordionGroup/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import { Accordion, Container } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/Callout/index.js b/components/content-blocks/Callout/index.js index 02bd43ca..bf543745 100644 --- a/components/content-blocks/Callout/index.js +++ b/components/content-blocks/Callout/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import CalloutTwoTone from "./CalloutTwoTone"; import CalloutMain from "./CalloutMain"; diff --git a/components/content-blocks/ComplexTable/index.js b/components/content-blocks/ComplexTable/index.js index e11da26e..bb8be0a5 100644 --- a/components/content-blocks/ComplexTable/index.js +++ b/components/content-blocks/ComplexTable/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import * as EPOLib from "@rubin-epo/epo-react-lib"; import { useGlobalData } from "@/lib/utils"; diff --git a/components/content-blocks/Contact/index.js b/components/content-blocks/Contact/index.js index 621ac8c3..0593114b 100644 --- a/components/content-blocks/Contact/index.js +++ b/components/content-blocks/Contact/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import ContactInfo from "@/global/ContactInfo"; diff --git a/components/content-blocks/ContactStaff/index.js b/components/content-blocks/ContactStaff/index.js index 2b12607c..8cc1a9a8 100644 --- a/components/content-blocks/ContactStaff/index.js +++ b/components/content-blocks/ContactStaff/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import { Container } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/DownloadList/index.js b/components/content-blocks/DownloadList/index.js index a9701b4f..f501e85d 100644 --- a/components/content-blocks/DownloadList/index.js +++ b/components/content-blocks/DownloadList/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { fileSize } from "@/helpers"; import * as Styled from "./styles"; diff --git a/components/content-blocks/Embed/index.js b/components/content-blocks/Embed/index.js index d6e1a80a..139b4f6c 100644 --- a/components/content-blocks/Embed/index.js +++ b/components/content-blocks/Embed/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import { containerFullBleed } from "@/styles/globalStyles"; diff --git a/components/content-blocks/GridBlock/index.js b/components/content-blocks/GridBlock/index.js index 13ddee46..6fee9199 100644 --- a/components/content-blocks/GridBlock/index.js +++ b/components/content-blocks/GridBlock/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import { Container, MixedLink } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/Image/styles.js b/components/content-blocks/Image/styles.js index 0a77dda9..e1e746f8 100644 --- a/components/content-blocks/Image/styles.js +++ b/components/content-blocks/Image/styles.js @@ -1,3 +1,4 @@ +"use client"; import styled from "styled-components"; import { containerNarrow, respond, tokens } from "@/styles/globalStyles"; diff --git a/components/content-blocks/KeyNumbersGridBlock/index.js b/components/content-blocks/KeyNumbersGridBlock/index.js index 71eb43f2..34e938a1 100644 --- a/components/content-blocks/KeyNumbersGridBlock/index.js +++ b/components/content-blocks/KeyNumbersGridBlock/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { mixedLinkShape } from "@/shapes/link"; import * as Styled from "./styles"; diff --git a/components/content-blocks/PeopleBlock/index.js b/components/content-blocks/PeopleBlock/index.js index 24c2a005..4987cc1d 100644 --- a/components/content-blocks/PeopleBlock/index.js +++ b/components/content-blocks/PeopleBlock/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container } from "@rubin-epo/epo-react-lib"; import { imageShaper } from "@/lib/utils"; diff --git a/components/content-blocks/PublicationsList/index.js b/components/content-blocks/PublicationsList/index.js index 08f10759..951a0f71 100644 --- a/components/content-blocks/PublicationsList/index.js +++ b/components/content-blocks/PublicationsList/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container } from "@rubin-epo/epo-react-lib"; import { mixedLinkShape } from "@/shapes/link"; diff --git a/components/content-blocks/Schedule/index.js b/components/content-blocks/Schedule/index.js index cfdaac33..ba96e453 100644 --- a/components/content-blocks/Schedule/index.js +++ b/components/content-blocks/Schedule/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import { Container } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/ShareBlock/index.js b/components/content-blocks/ShareBlock/index.js index 8eb0286a..d2ae55a8 100644 --- a/components/content-blocks/ShareBlock/index.js +++ b/components/content-blocks/ShareBlock/index.js @@ -1,3 +1,4 @@ +"use client"; import { Share } from "@/components/atomic"; import { Container } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/SimpleTable/index.js b/components/content-blocks/SimpleTable/index.js index 8ac2bb48..2c22cf5c 100644 --- a/components/content-blocks/SimpleTable/index.js +++ b/components/content-blocks/SimpleTable/index.js @@ -1,3 +1,4 @@ +"use client"; import React from "react"; import PropTypes from "prop-types"; import * as EPOLib from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/SlideBlock/index.js b/components/content-blocks/SlideBlock/index.js index c386bcce..2552344c 100644 --- a/components/content-blocks/SlideBlock/index.js +++ b/components/content-blocks/SlideBlock/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import styled from "styled-components"; import striptags from "striptags"; diff --git a/components/content-blocks/SummitStatus/index.js b/components/content-blocks/SummitStatus/index.js index e01e1839..600740be 100644 --- a/components/content-blocks/SummitStatus/index.js +++ b/components/content-blocks/SummitStatus/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container } from "@rubin-epo/epo-react-lib"; import { WeatherUnitProvider } from "@/contexts/WeatherUnit"; diff --git a/components/content-blocks/TableGroup/index.js b/components/content-blocks/TableGroup/index.js index 0c2e4b11..9d87f9a8 100644 --- a/components/content-blocks/TableGroup/index.js +++ b/components/content-blocks/TableGroup/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container } from "@rubin-epo/epo-react-lib"; import { Tab } from "@headlessui/react"; diff --git a/components/content-blocks/Text/index.js b/components/content-blocks/Text/index.js index e8a60005..58de9647 100644 --- a/components/content-blocks/Text/index.js +++ b/components/content-blocks/Text/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container } from "@rubin-epo/epo-react-lib"; diff --git a/components/content-blocks/Video/index.js b/components/content-blocks/Video/index.js index 8df8510f..5b0995b8 100644 --- a/components/content-blocks/Video/index.js +++ b/components/content-blocks/Video/index.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Container, Video, Figure } from "@rubin-epo/epo-react-lib"; import { useIsMounted } from "@/hooks"; diff --git a/components/dynamic/TempList/index.js b/components/dynamic/TempList/index.js index 0e07199b..2190327d 100644 --- a/components/dynamic/TempList/index.js +++ b/components/dynamic/TempList/index.js @@ -1,3 +1,4 @@ +"use client"; import React from "react"; import styled from "styled-components"; import PropTypes from "prop-types"; diff --git a/components/factories/ContentBlockFactory.js b/components/factories/ContentBlockFactory.js index 329b885a..15753c74 100644 --- a/components/factories/ContentBlockFactory.js +++ b/components/factories/ContentBlockFactory.js @@ -1,3 +1,4 @@ +"use client"; import PropTypes from "prop-types"; import { Image, diff --git a/components/layout/Tabs/index.js b/components/layout/Tabs/index.js index fd68f73f..df7a854e 100644 --- a/components/layout/Tabs/index.js +++ b/components/layout/Tabs/index.js @@ -1,3 +1,4 @@ +"use client"; import React, { useState, useRef, Children, cloneElement } from "react"; import PropTypes from "prop-types"; import styled from "styled-components"; diff --git a/components/molecules/Hero/index.tsx b/components/molecules/Hero/index.tsx new file mode 100644 index 00000000..7d6e8703 --- /dev/null +++ b/components/molecules/Hero/index.tsx @@ -0,0 +1,42 @@ +import { FunctionComponent, PropsWithChildren } from "react"; +import * as Styled from "./styles"; + +interface HeroProps { + data: Array; + focalPointX?: number; + focalPointY?: number; + className?: string; +} +const Hero: FunctionComponent> = ({ + data, + focalPointX = 50, + focalPointY = 50, + className, + children, +}) => { + const imageData = data && data[0]; + + if (!imageData?.url) return null; + + const { srcSet = [] } = imageData; + + const blurImage = srcSet[0]?.src ? `url('${srcSet[0]?.src}')` : undefined; + + return ( + + + {children} + + ); +}; + +Hero.displayName = "Molecule.Hero"; + +export default Hero; diff --git a/components/molecules/Hero/styles.ts b/components/molecules/Hero/styles.ts new file mode 100644 index 00000000..8fa0b84e --- /dev/null +++ b/components/molecules/Hero/styles.ts @@ -0,0 +1,29 @@ +"use client"; +import styled from "styled-components"; +import { fluidScale } from "@/styles/globalStyles"; +import Image from "@rubin-epo/epo-react-lib/Image"; + +export const HeroContainer = styled.div` + display: flex; + justify-content: center; + align-items: center; + + height: var(--Hero-height, ${fluidScale("540px", "400px")}); + overflow: hidden; + position: relative; +`; + +export const HeroImage = styled(Image)` + color: transparent; + background-image: var(--image-background-hero, none); + background-size: cover; + background-position: var(--Hero-object-position, center); + background-repeat: no-repeat; + + width: unset; + height: unset; + inline-size: 100%; + block-size: 100%; + object-fit: cover; + object-position: var(--Hero-object-position, center); +`; diff --git a/components/page/Hero/index.js b/components/page/Hero/index.js deleted file mode 100644 index c3956684..00000000 --- a/components/page/Hero/index.js +++ /dev/null @@ -1,37 +0,0 @@ -import PropTypes from "prop-types"; -import imageShape from "@/shapes/image"; -import * as Styled from "./styles"; - -export default function Hero({ - data, - focalPointX, - focalPointY, - className, - children, -}) { - const imageData = data && data[0]; - - if (!imageData?.url) return null; - - return ( - - - {children} - - ); -} - -Hero.displayName = "Global.Hero"; - -Hero.propTypes = { - data: PropTypes.arrayOf(imageShape), - children: PropTypes.node, - className: PropTypes.string, - focalPointX: PropTypes.number, - focalPointY: PropTypes.number, -}; diff --git a/components/page/Hero/styles.js b/components/page/Hero/styles.js deleted file mode 100644 index cc432a93..00000000 --- a/components/page/Hero/styles.js +++ /dev/null @@ -1,21 +0,0 @@ -import styled from "styled-components"; -import { fluidScale, containerFullBleed } from "@/styles/globalStyles"; -import { Image } from "@rubin-epo/epo-react-lib"; - -export const HeroContainer = styled.div` - ${containerFullBleed("CONTAINER_FULL")} - position: relative; - height: var(--Hero-height, ${fluidScale("540px", "400px")}); - overflow: auto; -`; - -export const HeroImage = styled(Image)` - --Hero-object-position: ${({ $focalPointX, $focalPointY }) => - `${$focalPointX}% ${$focalPointY}%;`} - width: 100%; - - /* stylelint-disable declaration-no-important */ - height: 100% !important; - object-fit: cover; - object-position: var(--Hero-object-position, center); -`; diff --git a/components/page/PageContent/styles.js b/components/page/PageContent/styles.js index 053e8ce4..e4acd26d 100644 --- a/components/page/PageContent/styles.js +++ b/components/page/PageContent/styles.js @@ -1,5 +1,5 @@ import styled from "styled-components"; -import HeroComponent from "@/page/Hero"; +import HeroComponent from "@/components/molecules/Hero"; import { fluidScale, respond, @@ -23,8 +23,6 @@ export const Hero = styled(HeroComponent)` WIDE_BREAKPOINT, MOBILE_BREAKPOINT )}; - --Hero-object-position: ${({ $focalPointX, $focalPointY }) => - `${$focalPointX}% ${$focalPointY}%;`} --hero-overlap: ${HERO_OVERLAP}; `; diff --git a/components/templates/EventPage/index.js b/components/templates/EventPage/index.js index 2225e213..6e90a9e0 100644 --- a/components/templates/EventPage/index.js +++ b/components/templates/EventPage/index.js @@ -6,7 +6,7 @@ import { createLocationString, useCustomBreadcrumbs, } from "@/lib/utils"; -import Hero from "@/page/Hero"; +import Hero from "@/components/molecules/Hero"; import ContentBlockFactory from "@/factories/ContentBlockFactory"; import { Share } from "@/content-blocks"; import Breadcrumbs from "@/page/Breadcrumbs"; diff --git a/components/templates/HomePage/index.js b/components/templates/HomePage/index.js index efc52429..94f4d539 100644 --- a/components/templates/HomePage/index.js +++ b/components/templates/HomePage/index.js @@ -1,29 +1,33 @@ -"use client"; import PropTypes from "prop-types"; -import { useTranslation } from "react-i18next"; +import truncate from "lodash/truncate"; +import { damAssetToImage } from "@/lib/canto"; +import { useTranslation } from "@/lib/i18n"; import ContentBlockFactory from "@/factories/ContentBlockFactory"; -import Hero from "@/page/Hero"; -import { Buttonish, MixedLink } from "@rubin-epo/epo-react-lib"; -import { makeDateString, makeTruncatedString } from "@/lib/utils"; +import Hero from "@/components/molecules/Hero"; +import Buttonish from "@rubin-epo/epo-react-lib/Buttonish"; +import MixedLink from "@rubin-epo/epo-react-lib/MixedLink"; +import { makeDateString } from "@/lib/utils/dates"; import { SlideBlock } from "@/components/content-blocks"; import Tabs from "@/components/layout/Tabs"; import TempList from "@/components/dynamic/TempList"; import * as Styled from "./styles"; -export default function HomePage({ +export default async function HomePage({ data: { contentBlocks, customHero, description, hero, + cantoHero, focalPointX, focalPointY, id, newsEntry, title, }, + locale, }) { - const { t } = useTranslation(); + const { t } = await useTranslation(locale); // HERO AREA // If there is a newsEntry or customHero, display alternate hero area @@ -95,7 +99,7 @@ export default function HomePage({ {!custom && news?.date &&

{makeDateString(news.date)}

} -
{makeTruncatedString(altTeaser, 30)}
+
{truncate(altTeaser, 30)}
{custom?.mixedLink?.element || custom?.mixedLink?.url ? ( @@ -111,7 +115,14 @@ export default function HomePage({ ) : ( - + 0 + ? [damAssetToImage(locale, cantoHero[0])] + : hero + } + {...{ focalPointX, focalPointY }} + />

{title}

@@ -142,4 +153,5 @@ HomePage.displayName = "Template.HomePage"; HomePage.propTypes = { data: PropTypes.object, + locale: PropTypes.string.isRequired, }; diff --git a/components/templates/HomePage/styles.js b/components/templates/HomePage/styles.js index 2ec826ae..c6d815f7 100644 --- a/components/templates/HomePage/styles.js +++ b/components/templates/HomePage/styles.js @@ -1,3 +1,4 @@ +"use client"; import styled from "styled-components"; import { respond } from "@/styles/globalStyles"; diff --git a/components/templates/NewsPage/Contacts.js b/components/templates/NewsPage/Contacts.js index 5183a290..ec7fb8b5 100644 --- a/components/templates/NewsPage/Contacts.js +++ b/components/templates/NewsPage/Contacts.js @@ -1,5 +1,4 @@ import PropTypes from "prop-types"; -import Link from "next/link"; import { useTranslation } from "react-i18next"; import * as Styled from "./styles"; @@ -10,73 +9,71 @@ export default function Contacts({ contacts }) { return ( <> {t(`news.contacts`)} -
+ {contacts.map((contact, i) => { const { name, affiliation, city, telephone, email } = contact; return ( - - {name && ( - - - - -
-

{name}

-
-
- )} - {affiliation && ( - - - - -
-

{affiliation}

-
-
- )} - {city && ( - - - - -
-

{city}

-
-
- )} - {telephone && ( - - - - -
- {telephone} - - {telephone} - -
-
- )} - {email && ( - - - - -
- - {email} - -
-
- )} -
+
  • + + {name && ( + + + + +
    {name}
    +
    + )} + {affiliation && ( + + + + +
    {affiliation}
    +
    + )} + {city && ( + + + + +
    {city}
    +
    + )} + {telephone && ( + + + + + + + )} + {email && ( + + + + +
    + {email} +
    +
    + )} +
    +
  • ); })} -
    + ); } diff --git a/components/templates/NewsPage/NewsArticle.js b/components/templates/NewsPage/NewsArticle.js index 9fed67d2..eee45f9f 100644 --- a/components/templates/NewsPage/NewsArticle.js +++ b/components/templates/NewsPage/NewsArticle.js @@ -22,16 +22,13 @@ export default function NewsArticle({ data }) { subtitle, contacts, links, - /* eslint-disable camelcase */ - more_information: moreInformation, - release_date: releaseDate, - /* eslint-enable camelcase */ + moreInformation, headline, releaseUrl, } = data; const { t } = useTranslation(); - const localizedDate = useDateString(date || releaseDate); + const localizedDate = useDateString(date); const { ref } = useResizeObserver({ box: "border-box", onResize: ({ height }) => { @@ -58,7 +55,7 @@ export default function NewsArticle({ data }) { {subtitle && ( {subtitle} )} - + {localizedDate} {description || headline} @@ -69,6 +66,7 @@ export default function NewsArticle({ data }) { {releaseDescription && (
    diff --git a/components/templates/NewsPage/NewsHero.js b/components/templates/NewsPage/NewsHero.js index bfe8fd75..30123978 100644 --- a/components/templates/NewsPage/NewsHero.js +++ b/components/templates/NewsPage/NewsHero.js @@ -1,6 +1,7 @@ import PropTypes from "prop-types"; import imageShape from "@/shapes/image"; import * as Styled from "./styles"; +import Hero from "@/components/molecules/Hero"; export default function NewsHero({ data, @@ -17,27 +18,13 @@ export default function NewsHero({ if (caption) return ( - - - + ); return ( - - - {children} - + {children} ); } diff --git a/components/templates/NewsPage/index.js b/components/templates/NewsPage/client.js similarity index 82% rename from components/templates/NewsPage/index.js rename to components/templates/NewsPage/client.js index 92021ed2..d378d12b 100644 --- a/components/templates/NewsPage/index.js +++ b/components/templates/NewsPage/client.js @@ -1,24 +1,16 @@ "use client"; import PropTypes from "prop-types"; import { useTranslation } from "react-i18next"; -import { - getSiteString, - useCustomBreadcrumbs, - makeReleaseFeature, -} from "@/lib/utils"; -import { useRelease } from "@/lib/api/noirlabReleases"; +import { useCustomBreadcrumbs } from "@/lib/utils"; import Breadcrumbs from "@/page/Breadcrumbs"; import NewsHero from "./NewsHero"; import NewsArticle from "./NewsArticle"; import NewsList from "@/dynamic/NewsList"; import NewsAside from "@/components/page/Aside/patterns/Media"; import * as Styled from "./styles"; +import { getReleaseHero } from "@/lib/api/noirlab"; export default function NewsPage({ data }) { - const { data: entryWithRelease } = useRelease( - getSiteString(data?.siteHandle || "default"), - data - ); const { contentBlocksNews = [], hero = [], @@ -32,10 +24,11 @@ export default function NewsPage({ data }) { uri, images: releaseImages, videos: releaseVideos, - } = entryWithRelease || data; + } = data; - const heroImage = - hero?.length > 0 ? hero : makeReleaseFeature(releaseImages, "banner1920"); + if (hero.length === 0 && releaseImages) { + hero.push(getReleaseHero(releaseImages)); + } const { t } = useTranslation(); @@ -74,7 +67,7 @@ export default function NewsPage({ data }) { } }); // If there is a hero then combine it with the content block media assets - if (heroBlock) mediaContentBlocks.unshift(heroBlock); + if (heroBlock && !releaseImages) mediaContentBlocks.unshift(heroBlock); // Only show the aside if there are news assets const showAside = @@ -89,14 +82,12 @@ export default function NewsPage({ data }) { - {(entryWithRelease || data) && ( - - )} + {data && } {showAside && ( = async ({ data, section, locale }) => { + const { pressReleaseId } = data; + + if (pressReleaseId) { + const { data: release, error } = await ReleasesService.releasesRetrieve({ + path: { id: pressReleaseId }, + query: { + lang: locale as Locale, + translation_mode: "fallback", + }, + }); + + if (!error && release) { + const { + title, + url: releaseUrl, + description, + headline, + subtitle, + images, + videos, + release_date: date, + more_information: moreInformation, + links, + contacts, + } = release; + + const combinedData = { + ...data, + title, + releaseUrl, + headline, + subtitle, + releaseDescription: description ? sanitizeHtml(description) : undefined, + moreInformation: moreInformation + ? sanitizeHtml(moreInformation) + : undefined, + links: links ? sanitizeHtml(links) : links, + contacts, + date, + images, + videos, + }; + + return ; + } + } + + return ; +}; + +export default NewsPage; diff --git a/components/templates/NewsPage/styles.js b/components/templates/NewsPage/styles.js index a901ba13..baa04a4f 100644 --- a/components/templates/NewsPage/styles.js +++ b/components/templates/NewsPage/styles.js @@ -2,20 +2,12 @@ import styled from "styled-components"; import { containerWide, containerFullBleed, - fluidScale, ptToEm, respond, tokens, } from "@/styles/globalStyles"; import { aHidden } from "@/styles/mixins/appearance"; -import { IconComposer, Image, token } from "@rubin-epo/epo-react-lib"; - -export const Hero = styled.div` - ${containerFullBleed("CONTAINER_FULL")} - position: relative; - height: var(--Hero-height, ${fluidScale("540px", "400px")}); - overflow: auto; -`; +import { IconComposer, token } from "@rubin-epo/epo-react-lib"; export const HeroFigure = styled.figure` ${containerFullBleed("CONTAINER_FULL")} @@ -27,13 +19,6 @@ export const HeroFigure = styled.figure` } `; -export const HeroImageContainer = styled.div` - ${containerFullBleed("CONTAINER_FULL")} - position: relative; - height: var(--Hero-height, ${fluidScale("540px", "400px")}); - overflow: auto; -`; - export const HeroFigCaption = styled.figcaption` ${aHidden} `; @@ -45,17 +30,6 @@ export const HeroCaption = styled.div` line-height: 1.428; `; -export const HeroImage = styled(Image)` - --Hero-object-position: ${({ $focalPointX, $focalPointY }) => - `${$focalPointX}% ${$focalPointY}%;`} - - width: 100%; - /* stylelint-disable declaration-no-important */ - height: 100% !important; - object-fit: cover; - object-position: var(--Hero-object-position, center); -`; - export const Article = styled.article` z-index: 1; margin-top: var(--Hero-caption-offset); @@ -102,7 +76,7 @@ export const NewsDetail = styled.div` } `; -export const Pretitle = styled.div` +export const Pretitle = styled.time` padding-bottom: 10px; `; @@ -122,18 +96,18 @@ export const ArticleHeading = styled.h3` export const ContactList = styled.ul` display: block; - + ul { - margin-top: 25px; + & > * + * { + margin-block-start: 25px; } `; -export const ContactListItem = styled.li` +export const Contact = styled.address` + font-style: normal; +`; + +export const ContactRow = styled.div` display: flex; align-items: center; - - + & { - margin-top: 10px; - } `; export const IconWrapper = styled.div` diff --git a/cypress/e2e/homepage.cy.js b/cypress/e2e/homepage.cy.js index db8a64f7..7d23b879 100644 --- a/cypress/e2e/homepage.cy.js +++ b/cypress/e2e/homepage.cy.js @@ -1,4 +1,4 @@ -describe("Search from Homepage", () => { +describe("Homepage", () => { beforeEach(() => { // Cypress starts out with a blank slate for each test // so we must tell it to visit our website with the `cy.visit()` command. @@ -6,7 +6,13 @@ describe("Search from Homepage", () => { // we include it in our beforeEach function so that it runs before each test cy.visit("/"); }); - + it("has a hero image with priority loading", () => { + cy.get("[data-cy=hero] > img") + .invoke("attr", "src") + .should("have.length.gt", 0); + cy.get("[data-cy=hero] > img").should("have.attr", "loading", "eager"); + cy.get("[data-cy=hero] > img").should("have.attr", "decoding", "sync"); + }); it("Searching yields results", () => { cy.get(".c-search-bar__toggle").click(); cy.get("#headerSearchBar").should("be.visible").type("rubin{enter}"); diff --git a/cypress/e2e/press-release.cy.js b/cypress/e2e/press-release.cy.js new file mode 100644 index 00000000..74b26fbb --- /dev/null +++ b/cypress/e2e/press-release.cy.js @@ -0,0 +1,47 @@ +describe("Press release", () => { + beforeEach(() => { + // Cypress starts out with a blank slate for each test + // so we must tell it to visit our website with the `cy.visit()` command. + // Since we want to visit the same URL at the start of all our tests, + // we include it in our beforeEach function so that it runs before each test + cy.visit("/news/rubin-reveal-stellar-streams"); + }); + + it("Has page metadata", () => { + cy.document() + .get('head meta[name="description"]') + .should("have.attr", "content"); + }); + it("Has a banner image", () => { + cy.get("[data-cy=hero] > img") + .invoke("attr", "alt") + .should("have.length.gt", 0); + cy.get("[data-cy=hero] > img") + .invoke("attr", "src") + .should("have.length.gt", 0); + cy.get("[data-cy=hero] > img").should("have.attr", "loading", "eager"); + }); + it("Has a headline", () => { + cy.get("h1").invoke("text").should("have.length.gt", 0); + }); + it("Has a release date", () => { + cy.get("time").invoke("text").should("have.length.gt", 0); + cy.get("time").should("have.attr", "datetime"); + }); + it("Has some text content", () => { + cy.get("[data-cy=press-release-text]") + .invoke("text") + .should("have.length.gt", 0); + }); + it("Has additional information", () => { + cy.contains("h3", "More Information") + .siblings() + .should("have.class", "c-content-rte"); + }); + it("Has links", () => { + cy.contains("h3", "Links").siblings().contains("ul"); + }); + it("Has contacts", () => { + cy.contains("h3", "Contacts").siblings().contains("li"); + }); +}); diff --git a/lib/api/homepage/index.ts b/lib/api/homepage/index.ts index c3aea5c2..96d7716d 100644 --- a/lib/api/homepage/index.ts +++ b/lib/api/homepage/index.ts @@ -27,6 +27,20 @@ export const getHomepage = async (locale: string, previewToken?: string) => { ${getImageFields("crop", 1920, 1067)} } } + cantoHero { + url { + directUrlPreview + directUrlOriginal + } + width + height + metadata: additional { + AltTextEN + AltTextES + CaptionEN + CaptionES + } + } focalPointX focalPointY newsEntry { @@ -94,6 +108,25 @@ export const getHomepageMetadata = async ( ... on homepage_homepage_Entry { title description + hero { + ...on heroes_Asset { + ${getImageFields("crop", 900, 550)} + } + } + cantoHero { + url { + directUrlPreview + directUrlOriginal + } + width + height + metadata: additional { + AltTextEN + AltTextES + CaptionEN + CaptionES + } + } } } } diff --git a/lib/api/noirlab/codegen/index.ts b/lib/api/noirlab/codegen/index.ts new file mode 100644 index 00000000..1cb041de --- /dev/null +++ b/lib/api/noirlab/codegen/index.ts @@ -0,0 +1,4 @@ +// This file is auto-generated by @hey-api/openapi-ts +export * from "./schemas.gen"; +export * from "./services.gen"; +export * from "./types.gen"; diff --git a/lib/api/noirlab/codegen/schemas.gen.ts b/lib/api/noirlab/codegen/schemas.gen.ts new file mode 100644 index 00000000..ed8f983d --- /dev/null +++ b/lib/api/noirlab/codegen/schemas.gen.ts @@ -0,0 +1,2185 @@ +// This file is auto-generated by @hey-api/openapi-ts + +export const ActivitySchema = { + type: "object", + properties: { + id: { + type: "string", + description: "ID of the activity, also used in URLs", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + name: { + type: "string", + maxLength: 100, + }, + title: { + type: "string", + nullable: true, + description: + "title to be displayed in the activity description when the activity is joined with other activities", + maxLength: 100, + }, + slogan: { + type: "string", + maxLength: 255, + }, + observatory: { + type: "string", + maxLength: 50, + }, + key_visual_en: { + type: "string", + readOnly: true, + }, + key_visual_es: { + type: "string", + readOnly: true, + }, + duration: { + type: "string", + description: "Format: HH:MM", + }, + safety_tech_doc: { + type: "string", + readOnly: true, + }, + conduct_tech_doc: { + type: "string", + readOnly: true, + }, + liability_tech_doc: { + type: "string", + readOnly: true, + }, + safety_tech_doc_es: { + type: "string", + readOnly: true, + }, + conduct_tech_doc_es: { + type: "string", + readOnly: true, + }, + liability_tech_doc_es: { + type: "string", + readOnly: true, + }, + group_safety_tech_doc: { + type: "string", + readOnly: true, + }, + group_liability_tech_doc: { + type: "string", + readOnly: true, + }, + group_safety_tech_doc_es: { + type: "string", + readOnly: true, + }, + group_liability_tech_doc_es: { + type: "string", + readOnly: true, + }, + }, + required: [ + "conduct_tech_doc", + "conduct_tech_doc_es", + "duration", + "group_liability_tech_doc", + "group_liability_tech_doc_es", + "group_safety_tech_doc", + "group_safety_tech_doc_es", + "id", + "key_visual_en", + "key_visual_es", + "liability_tech_doc", + "liability_tech_doc_es", + "name", + "observatory", + "safety_tech_doc", + "safety_tech_doc_es", + ], +} as const; + +export const AnnouncementSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + url: { + type: "string", + readOnly: true, + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + subtitle: { + type: "string", + description: "Optional subtitle to be shown just above the headline.", + maxLength: 255, + }, + description: { + type: "string", + }, + contacts: { + type: "string", + description: "Contacts", + }, + links: { + type: "string", + description: "Links", + }, + featured: { + type: "boolean", + }, + release_date: { + type: "string", + readOnly: true, + }, + programs: { + type: "array", + items: { + $ref: "#/components/schemas/Program", + }, + }, + images: { + type: "array", + items: { + $ref: "#/components/schemas/ImageMini", + }, + readOnly: true, + }, + videos: { + type: "array", + items: { + $ref: "#/components/schemas/VideoMini", + }, + readOnly: true, + }, + }, + required: [ + "id", + "images", + "programs", + "release_date", + "title", + "url", + "videos", + ], +} as const; + +export const AnnouncementMiniSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + url: { + type: "string", + readOnly: true, + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + subtitle: { + type: "string", + description: "Optional subtitle to be shown just above the headline.", + maxLength: 255, + }, + release_date: { + type: "string", + readOnly: true, + }, + programs: { + type: "array", + items: { + $ref: "#/components/schemas/Program", + }, + }, + main_image: { + allOf: [ + { + $ref: "#/components/schemas/ImageMini", + }, + ], + readOnly: true, + }, + }, + required: ["id", "main_image", "programs", "release_date", "title", "url"], +} as const; + +export const AuthorDescriptionSchema = { + type: "object", + properties: { + name: { + type: "string", + readOnly: true, + }, + description: { + type: "string", + description: + 'Optional description, e.g.: "Author: ", or "Interview with"', + maxLength: 100, + }, + photo: { + type: "string", + readOnly: true, + }, + static_photo: { + type: "string", + description: "Direct link to a JPG image, recommended size: 350px wide", + readOnly: true, + }, + }, + required: ["name", "photo", "static_photo"], +} as const; + +export const BlankEnumSchema = { + enum: [""], +} as const; + +export const CategorySchema = { + type: "object", + properties: { + slug: { + type: "string", + }, + name: { + type: "string", + description: "Title of query to be displayed to the user.", + maxLength: 255, + }, + logo_url: { + type: "string", + format: "uri", + nullable: true, + maxLength: 255, + }, + }, + required: ["name", "slug"], +} as const; + +export const ImageSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Id of image - used in the URL for the image as well as the filename for the different formats.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + source: { + type: "string", + description: + "Id of image - used in the URL for the image as well as the filename for the different formats.", + nullable: true, + title: "Translation source", + }, + title: { + type: "string", + description: "General descriptive title given to the image resource.", + maxLength: 255, + }, + headline: { + type: "string", + nullable: true, + description: "Short description of the full caption.", + }, + description: { + type: "string", + nullable: true, + description: + "Full caption and related description text for the image resource.", + }, + categories: { + type: "array", + items: { + $ref: "#/components/schemas/Category", + }, + }, + type: { + nullable: true, + description: `The type of image/media resource. + +* \`Observation\` - Observation +* \`Artwork\` - Artwork +* \`Photographic\` - Photographic +* \`Planetary\` - Planetary +* \`Simulation\` - Simulation +* \`Collage\` - Collage +* \`Chart\` - Chart`, + oneOf: [ + { + $ref: "#/components/schemas/TypeEnum", + }, + { + $ref: "#/components/schemas/BlankEnum", + }, + { + $ref: "#/components/schemas/NullEnum", + }, + ], + }, + credit: { + type: "string", + description: + "The minimum information that the Publisher would like to see mentioned when the resource is used.", + }, + release_date: { + type: "string", + readOnly: true, + }, + width: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Width in pixels of the image resource.", + }, + height: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Height in pixels of the image resource.", + }, + featured: { + type: "boolean", + }, + subject_name: { + type: "array", + items: { + type: "string", + }, + }, + resources: { + type: "array", + items: { + type: "object", + properties: { + ResourceType: { + type: "string", + }, + MediaType: { + type: "string", + }, + URL: { + type: "string", + }, + FileSize: { + type: "integer", + nullable: true, + }, + Dimensions: { + type: "array", + items: { + type: "integer", + }, + nullable: true, + }, + ProjectionType: { + type: "string", + nullable: true, + }, + }, + }, + readOnly: true, + }, + formats: { + type: "object", + properties: { + banner1920: { + type: "string", + nullable: true, + }, + eps: { + type: "string", + nullable: true, + }, + illustrator: { + type: "string", + nullable: true, + }, + illustrator_text: { + type: "string", + nullable: true, + }, + large: { + type: "string", + nullable: true, + }, + medium: { + type: "string", + nullable: true, + }, + news: { + type: "string", + nullable: true, + }, + newsfeature: { + type: "string", + nullable: true, + }, + newsmini: { + type: "string", + nullable: true, + }, + original: { + type: "string", + nullable: true, + }, + pdf: { + type: "string", + nullable: true, + }, + pl_original: { + type: "string", + nullable: true, + }, + pl_screen: { + type: "string", + nullable: true, + }, + pl_thumbs: { + type: "string", + nullable: true, + }, + portrait1080: { + type: "string", + nullable: true, + }, + poster400y: { + type: "string", + nullable: true, + }, + potwmedium: { + type: "string", + nullable: true, + }, + publicationjpg: { + type: "string", + nullable: true, + }, + publicationtiff: { + type: "string", + nullable: true, + }, + publicationtiff10k: { + type: "string", + nullable: true, + }, + publicationtiff25k: { + type: "string", + nullable: true, + }, + publicationtiff40k: { + type: "string", + nullable: true, + }, + screen: { + type: "string", + nullable: true, + }, + screen640: { + type: "string", + nullable: true, + }, + thumb150y: { + type: "string", + nullable: true, + }, + thumb300y: { + type: "string", + nullable: true, + }, + thumb350x: { + type: "string", + nullable: true, + }, + thumb700x: { + type: "string", + nullable: true, + }, + thumbs: { + type: "string", + nullable: true, + }, + wallpaper1: { + type: "string", + nullable: true, + }, + wallpaper2: { + type: "string", + nullable: true, + }, + wallpaper3: { + type: "string", + nullable: true, + }, + wallpaper4: { + type: "string", + nullable: true, + }, + wallpaper5: { + type: "string", + nullable: true, + }, + wallpaperthumbs: { + type: "string", + nullable: true, + }, + zoomable: { + type: "string", + nullable: true, + }, + }, + readOnly: true, + }, + }, + required: [ + "categories", + "formats", + "id", + "release_date", + "resources", + "subject_name", + "title", + "url", + ], +} as const; + +export const ImageItemSchema = { + oneOf: [ + { + $ref: "#/components/schemas/ImageTiny", + }, + { + $ref: "#/components/schemas/ImageMini", + }, + ], +} as const; + +export const ImageMiniSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Id of image - used in the URL for the image as well as the filename for the different formats.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + source: { + type: "string", + description: + "Id of image - used in the URL for the image as well as the filename for the different formats.", + nullable: true, + title: "Translation source", + }, + title: { + type: "string", + description: "General descriptive title given to the image resource.", + maxLength: 255, + }, + width: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Width in pixels of the image resource.", + }, + height: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Height in pixels of the image resource.", + }, + featured: { + type: "boolean", + }, + categories: { + type: "array", + items: { + $ref: "#/components/schemas/Category", + }, + }, + formats: { + type: "object", + properties: { + banner1920: { + type: "string", + nullable: true, + }, + eps: { + type: "string", + nullable: true, + }, + illustrator: { + type: "string", + nullable: true, + }, + illustrator_text: { + type: "string", + nullable: true, + }, + large: { + type: "string", + nullable: true, + }, + medium: { + type: "string", + nullable: true, + }, + news: { + type: "string", + nullable: true, + }, + newsfeature: { + type: "string", + nullable: true, + }, + newsmini: { + type: "string", + nullable: true, + }, + original: { + type: "string", + nullable: true, + }, + pdf: { + type: "string", + nullable: true, + }, + pl_original: { + type: "string", + nullable: true, + }, + pl_screen: { + type: "string", + nullable: true, + }, + pl_thumbs: { + type: "string", + nullable: true, + }, + portrait1080: { + type: "string", + nullable: true, + }, + poster400y: { + type: "string", + nullable: true, + }, + potwmedium: { + type: "string", + nullable: true, + }, + publicationjpg: { + type: "string", + nullable: true, + }, + publicationtiff: { + type: "string", + nullable: true, + }, + publicationtiff10k: { + type: "string", + nullable: true, + }, + publicationtiff25k: { + type: "string", + nullable: true, + }, + publicationtiff40k: { + type: "string", + nullable: true, + }, + screen: { + type: "string", + nullable: true, + }, + screen640: { + type: "string", + nullable: true, + }, + thumb150y: { + type: "string", + nullable: true, + }, + thumb300y: { + type: "string", + nullable: true, + }, + thumb350x: { + type: "string", + nullable: true, + }, + thumb700x: { + type: "string", + nullable: true, + }, + thumbs: { + type: "string", + nullable: true, + }, + wallpaper1: { + type: "string", + nullable: true, + }, + wallpaper2: { + type: "string", + nullable: true, + }, + wallpaper3: { + type: "string", + nullable: true, + }, + wallpaper4: { + type: "string", + nullable: true, + }, + wallpaper5: { + type: "string", + nullable: true, + }, + wallpaperthumbs: { + type: "string", + nullable: true, + }, + zoomable: { + type: "string", + nullable: true, + }, + }, + readOnly: true, + }, + }, + required: ["categories", "formats", "id", "title", "url"], +} as const; + +export const ImageTinySchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Id of image - used in the URL for the image as well as the filename for the different formats.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + title: { + type: "string", + description: "General descriptive title given to the image resource.", + maxLength: 255, + }, + width: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Width in pixels of the image resource.", + }, + height: { + type: "integer", + maximum: 2147483647, + minimum: 0, + nullable: true, + description: "Height in pixels of the image resource.", + }, + formats: { + type: "object", + properties: { + thumb300y: { + type: "string", + nullable: true, + }, + screen: { + type: "string", + nullable: true, + }, + thumb700x: { + type: "string", + nullable: true, + }, + }, + readOnly: true, + }, + }, + required: ["formats", "id", "title", "url"], +} as const; + +export const Model3dSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + description: { + type: "string", + }, + credit: { + type: "string", + readOnly: true, + }, + priority: { + type: "integer", + maximum: 32767, + minimum: 0, + description: + "Priority of product (100 highest, 0 lowest) - high priority products are ranked higher in search results than low priority products.", + }, + release_date: { + type: "string", + readOnly: true, + }, + assets: { + type: "string", + readOnly: true, + }, + }, + required: ["assets", "credit", "id", "release_date", "title"], +} as const; + +export const NullEnumSchema = { + enum: [null], +} as const; + +export const PageSchema = { + type: "object", + properties: { + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + content: { + type: "string", + }, + description: { + type: "string", + description: + "The metadata description is normally shown in search engine results, making the description an effective way of capturing users attention. Description should be a clear description of the content and less the 200 characters long. Also used when sharing page on social media", + }, + keywords: { + type: "string", + description: + "Comma-separated list of keywords for this page. Mainly used internally as search engines rarely use keywords to rank pages.", + }, + }, + required: ["title"], +} as const; + +export const PageRequestSchema = { + type: "object", + properties: { + title: { + type: "string", + minLength: 1, + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + content: { + type: "string", + }, + description: { + type: "string", + description: + "The metadata description is normally shown in search engine results, making the description an effective way of capturing users attention. Description should be a clear description of the content and less the 200 characters long. Also used when sharing page on social media", + }, + keywords: { + type: "string", + description: + "Comma-separated list of keywords for this page. Mainly used internally as search engines rarely use keywords to rank pages.", + }, + }, + required: ["title"], +} as const; + +export const PaginatedAnnouncementMiniListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/AnnouncementMini", + }, + }, + }, +} as const; + +export const PaginatedImageItemListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/ImageItem", + }, + }, + }, +} as const; + +export const PaginatedImageListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/Image", + }, + }, + }, +} as const; + +export const PaginatedModel3dListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/Model3d", + }, + }, + }, +} as const; + +export const PaginatedPodcastListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/Podcast", + }, + }, + }, +} as const; + +export const PaginatedReleaseMiniListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/ReleaseMini", + }, + }, + }, +} as const; + +export const PaginatedVideoListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/Video", + }, + }, + }, +} as const; + +export const PaginatedVideoMiniListSchema = { + type: "object", + properties: { + count: { + type: "integer", + example: 123, + }, + next: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=4", + }, + previous: { + type: "string", + nullable: true, + format: "uri", + example: "http://api.example.org/accounts/?page=2", + }, + results: { + type: "array", + items: { + $ref: "#/components/schemas/VideoMini", + }, + }, + }, +} as const; + +export const PatchedPageRequestSchema = { + type: "object", + properties: { + title: { + type: "string", + minLength: 1, + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + content: { + type: "string", + }, + description: { + type: "string", + description: + "The metadata description is normally shown in search engine results, making the description an effective way of capturing users attention. Description should be a clear description of the content and less the 200 characters long. Also used when sharing page on social media", + }, + keywords: { + type: "string", + description: + "Comma-separated list of keywords for this page. Mainly used internally as search engines rarely use keywords to rank pages.", + }, + }, +} as const; + +export const PodcastSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 200, + }, + description: { + type: "string", + }, + credit: { + type: "string", + readOnly: true, + }, + priority: { + type: "integer", + maximum: 32767, + minimum: 0, + description: + "Priority of product (100 highest, 0 lowest) - high priority products are ranked higher in search results than low priority products.", + }, + release_date: { + type: "string", + readOnly: true, + }, + assets: { + type: "string", + readOnly: true, + }, + }, + required: ["assets", "credit", "id", "release_date", "title"], +} as const; + +export const PostSchema = { + type: "object", + properties: { + id: { + type: "string", + readOnly: true, + }, + slug: { + type: "string", + description: "Used for the URL, this cannot be updated later", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + title: { + type: "string", + maxLength: 255, + }, + subtitle: { + type: "string", + description: "Optional subtitle", + maxLength: 255, + }, + banner: { + type: "string", + readOnly: true, + }, + authors: { + type: "array", + items: { + $ref: "#/components/schemas/AuthorDescription", + }, + }, + category: { + allOf: [ + { + $ref: "#/components/schemas/Category", + }, + ], + readOnly: true, + }, + lede: { + type: "string", + }, + release_date: { + type: "string", + format: "date-time", + nullable: true, + }, + }, + required: [ + "authors", + "banner", + "category", + "id", + "lede", + "slug", + "title", + "url", + ], +} as const; + +export const ProgramSchema = { + type: "object", + properties: { + slug: { + type: "string", + }, + name: { + type: "string", + description: "Title of query to be displayed to the user.", + maxLength: 255, + }, + logo_url: { + type: "string", + format: "uri", + nullable: true, + maxLength: 255, + }, + }, + required: ["name", "slug"], +} as const; + +export const ReleaseSchema = { + type: "object", + properties: { + id: { + type: "string", + description: "Id of release - e.g. heic0801. The id must be unique.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + url: { + type: "string", + readOnly: true, + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 255, + }, + release_type: { + type: "string", + readOnly: true, + }, + subtitle: { + type: "string", + description: "Optional subtitle to be shown just above the headline.", + maxLength: 255, + }, + headline: { + type: "string", + description: + "HTML code in lead is not allowed. The lead is further more normally shown in search engine results, making the description an effective way of capturing users attention.", + }, + release_date: { + type: "string", + readOnly: true, + }, + description: { + type: "string", + }, + notes: { + type: "string", + }, + more_information: { + type: "string", + }, + links: { + type: "string", + description: "Help text", + }, + disclaimer: { + type: "string", + description: + "Disclaimer for press release - usually e.g. retractions of previously issued press releases.", + }, + programs: { + type: "array", + items: { + $ref: "#/components/schemas/Program", + }, + }, + images: { + type: "array", + items: { + $ref: "#/components/schemas/ImageMini", + }, + readOnly: true, + }, + videos: { + type: "array", + items: { + $ref: "#/components/schemas/VideoMini", + }, + readOnly: true, + }, + contacts: { + type: "array", + items: { + $ref: "#/components/schemas/ReleaseContact", + }, + }, + }, + required: [ + "contacts", + "id", + "images", + "programs", + "release_date", + "release_type", + "title", + "url", + "videos", + ], +} as const; + +export const ReleaseContactSchema = { + type: "object", + properties: { + name: { + type: "string", + maxLength: 255, + }, + email: { + type: "string", + maxLength: 255, + }, + telephone: { + type: "string", + maxLength: 255, + }, + cellular: { + type: "string", + maxLength: 255, + }, + affiliation: { + type: "string", + maxLength: 255, + }, + address: { + type: "string", + maxLength: 255, + }, + city: { + type: "string", + maxLength: 255, + }, + state_province: { + type: "string", + maxLength: 255, + }, + postal_code: { + type: "string", + maxLength: 255, + }, + country: { + type: "string", + maxLength: 255, + }, + }, +} as const; + +export const ReleaseMiniSchema = { + type: "object", + properties: { + id: { + type: "string", + description: "Id of release - e.g. heic0801. The id must be unique.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + url: { + type: "string", + readOnly: true, + }, + release_type: { + type: "string", + readOnly: true, + }, + title: { + type: "string", + description: + "Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages.", + maxLength: 255, + }, + subtitle: { + type: "string", + description: "Optional subtitle to be shown just above the headline.", + maxLength: 255, + }, + headline: { + type: "string", + description: + "HTML code in lead is not allowed. The lead is further more normally shown in search engine results, making the description an effective way of capturing users attention.", + }, + release_date: { + type: "string", + readOnly: true, + }, + programs: { + type: "array", + items: { + $ref: "#/components/schemas/Program", + }, + }, + main_image: { + allOf: [ + { + $ref: "#/components/schemas/ImageMini", + }, + ], + readOnly: true, + }, + }, + required: [ + "id", + "main_image", + "programs", + "release_date", + "release_type", + "title", + "url", + ], +} as const; + +export const ShowingSchema = { + type: "object", + properties: { + id: { + type: "integer", + readOnly: true, + }, + start_time: { + type: "string", + readOnly: true, + }, + formatted_start_time: { + type: "string", + readOnly: true, + }, + free_spaces: { + type: "integer", + }, + }, + required: ["formatted_start_time", "free_spaces", "id", "start_time"], +} as const; + +export const TypeEnumSchema = { + enum: [ + "Observation", + "Artwork", + "Photographic", + "Planetary", + "Simulation", + "Collage", + "Chart", + ], + type: "string", + description: `* \`Observation\` - Observation +* \`Artwork\` - Artwork +* \`Photographic\` - Photographic +* \`Planetary\` - Planetary +* \`Simulation\` - Simulation +* \`Collage\` - Collage +* \`Chart\` - Chart`, +} as const; + +export const VideoSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Id of video - used in the URL for the image as well as the filename for the different formats.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + source: { + type: "string", + description: + "Id of video - used in the URL for the image as well as the filename for the different formats.", + nullable: true, + title: "Translation source", + }, + title: { + type: "string", + description: "General descriptive title given to the image resource.", + maxLength: 255, + }, + headline: { + type: "string", + nullable: true, + description: "Short description of the full caption.", + }, + description: { + type: "string", + nullable: true, + description: + "Full caption and related description text for the image resource.", + }, + categories: { + type: "array", + items: { + $ref: "#/components/schemas/Category", + }, + }, + type: { + nullable: true, + description: `The type of image/media resource. + +* \`Observation\` - Observation +* \`Artwork\` - Artwork +* \`Photographic\` - Photographic +* \`Planetary\` - Planetary +* \`Simulation\` - Simulation +* \`Collage\` - Collage +* \`Chart\` - Chart`, + oneOf: [ + { + $ref: "#/components/schemas/TypeEnum", + }, + { + $ref: "#/components/schemas/BlankEnum", + }, + { + $ref: "#/components/schemas/NullEnum", + }, + ], + }, + credit: { + type: "string", + description: + "The minimum information that the Publisher would like to see mentioned when the resource is used.", + }, + release_date: { + type: "string", + readOnly: true, + }, + featured: { + type: "boolean", + }, + duration: { + type: "string", + nullable: true, + readOnly: true, + }, + youtube_video_id: { + type: "string", + nullable: true, + title: "YouTube VideoID", + maxLength: 11, + }, + use_youtube: { + type: "boolean", + title: "Use YouTube player", + }, + formats: { + type: "object", + properties: { + broadcast_sd: { + type: "string", + nullable: true, + }, + cylindrical_16kmaster: { + type: "string", + nullable: true, + }, + cylindrical_4kmaster: { + type: "string", + nullable: true, + }, + cylindrical_8kmaster: { + type: "string", + nullable: true, + }, + cylindrical_preview: { + type: "string", + nullable: true, + }, + dome_2kmaster: { + type: "string", + nullable: true, + }, + dome_2kplayback: { + type: "string", + nullable: true, + }, + dome_4kmaster: { + type: "string", + nullable: true, + }, + dome_4kplayback: { + type: "string", + nullable: true, + }, + dome_8kmaster: { + type: "string", + nullable: true, + }, + dome_mov: { + type: "string", + nullable: true, + }, + dome_preview: { + type: "string", + nullable: true, + }, + ext_highres: { + type: "string", + nullable: true, + }, + ext_playback: { + type: "string", + nullable: true, + }, + hd_1080_broadcast: { + type: "string", + nullable: true, + }, + hd_1080_screen: { + type: "string", + nullable: true, + }, + hd_1080p25_broadcast: { + type: "string", + nullable: true, + }, + hd_1080p25_screen: { + type: "string", + nullable: true, + }, + hd_and_apple: { + type: "string", + nullable: true, + }, + hd_broadcast_720p25: { + type: "string", + nullable: true, + }, + hd_broadcast_720p50: { + type: "string", + nullable: true, + }, + large_qt: { + type: "string", + nullable: true, + }, + medium_flash: { + type: "string", + nullable: true, + }, + medium_mpeg1: { + type: "string", + nullable: true, + }, + medium_podcast: { + type: "string", + nullable: true, + }, + news: { + type: "string", + nullable: true, + }, + newsfeature: { + type: "string", + nullable: true, + }, + newsmini: { + type: "string", + nullable: true, + }, + old_video: { + type: "string", + nullable: true, + }, + original: { + type: "string", + nullable: true, + }, + potwmedium: { + type: "string", + nullable: true, + }, + qtvr: { + type: "string", + nullable: true, + }, + script: { + type: "string", + nullable: true, + }, + small_flash: { + type: "string", + nullable: true, + }, + small_qt: { + type: "string", + nullable: true, + }, + thumb: { + type: "string", + nullable: true, + }, + thumb300y: { + type: "string", + nullable: true, + }, + thumb350x: { + type: "string", + nullable: true, + }, + ultra_hd: { + type: "string", + nullable: true, + }, + ultra_hd_8k_broadcast: { + type: "string", + nullable: true, + }, + ultra_hd_8k_h265: { + type: "string", + nullable: true, + }, + ultra_hd_broadcast: { + type: "string", + nullable: true, + }, + ultra_hd_h265: { + type: "string", + nullable: true, + }, + videoframe: { + type: "string", + nullable: true, + }, + vr_16kmaster: { + type: "string", + nullable: true, + }, + vr_2k_sos: { + type: "string", + nullable: true, + }, + vr_4k: { + type: "string", + nullable: true, + }, + vr_4k_sos: { + type: "string", + nullable: true, + }, + vr_4kmaster: { + type: "string", + nullable: true, + }, + vr_8k: { + type: "string", + nullable: true, + }, + vr_8kmaster: { + type: "string", + nullable: true, + }, + }, + readOnly: true, + }, + }, + required: [ + "categories", + "duration", + "formats", + "id", + "release_date", + "title", + "url", + ], +} as const; + +export const VideoMiniSchema = { + type: "object", + properties: { + id: { + type: "string", + description: + "Id of video - used in the URL for the image as well as the filename for the different formats.", + maxLength: 50, + pattern: "^[-a-zA-Z0-9_]+$", + }, + url: { + type: "string", + readOnly: true, + }, + lang: { + type: "string", + title: "Language", + maxLength: 7, + }, + source: { + type: "string", + description: + "Id of video - used in the URL for the image as well as the filename for the different formats.", + nullable: true, + title: "Translation source", + }, + title: { + type: "string", + description: "General descriptive title given to the image resource.", + maxLength: 255, + }, + featured: { + type: "boolean", + }, + duration: { + type: "string", + nullable: true, + readOnly: true, + }, + categories: { + type: "array", + items: { + $ref: "#/components/schemas/Category", + }, + }, + youtube_video_id: { + type: "string", + nullable: true, + title: "YouTube VideoID", + maxLength: 11, + }, + use_youtube: { + type: "boolean", + title: "Use YouTube player", + }, + formats: { + type: "object", + properties: { + broadcast_sd: { + type: "string", + nullable: true, + }, + cylindrical_16kmaster: { + type: "string", + nullable: true, + }, + cylindrical_4kmaster: { + type: "string", + nullable: true, + }, + cylindrical_8kmaster: { + type: "string", + nullable: true, + }, + cylindrical_preview: { + type: "string", + nullable: true, + }, + dome_2kmaster: { + type: "string", + nullable: true, + }, + dome_2kplayback: { + type: "string", + nullable: true, + }, + dome_4kmaster: { + type: "string", + nullable: true, + }, + dome_4kplayback: { + type: "string", + nullable: true, + }, + dome_8kmaster: { + type: "string", + nullable: true, + }, + dome_mov: { + type: "string", + nullable: true, + }, + dome_preview: { + type: "string", + nullable: true, + }, + ext_highres: { + type: "string", + nullable: true, + }, + ext_playback: { + type: "string", + nullable: true, + }, + hd_1080_broadcast: { + type: "string", + nullable: true, + }, + hd_1080_screen: { + type: "string", + nullable: true, + }, + hd_1080p25_broadcast: { + type: "string", + nullable: true, + }, + hd_1080p25_screen: { + type: "string", + nullable: true, + }, + hd_and_apple: { + type: "string", + nullable: true, + }, + hd_broadcast_720p25: { + type: "string", + nullable: true, + }, + hd_broadcast_720p50: { + type: "string", + nullable: true, + }, + large_qt: { + type: "string", + nullable: true, + }, + medium_flash: { + type: "string", + nullable: true, + }, + medium_mpeg1: { + type: "string", + nullable: true, + }, + medium_podcast: { + type: "string", + nullable: true, + }, + news: { + type: "string", + nullable: true, + }, + newsfeature: { + type: "string", + nullable: true, + }, + newsmini: { + type: "string", + nullable: true, + }, + old_video: { + type: "string", + nullable: true, + }, + original: { + type: "string", + nullable: true, + }, + potwmedium: { + type: "string", + nullable: true, + }, + qtvr: { + type: "string", + nullable: true, + }, + script: { + type: "string", + nullable: true, + }, + small_flash: { + type: "string", + nullable: true, + }, + small_qt: { + type: "string", + nullable: true, + }, + thumb: { + type: "string", + nullable: true, + }, + thumb300y: { + type: "string", + nullable: true, + }, + thumb350x: { + type: "string", + nullable: true, + }, + ultra_hd: { + type: "string", + nullable: true, + }, + ultra_hd_8k_broadcast: { + type: "string", + nullable: true, + }, + ultra_hd_8k_h265: { + type: "string", + nullable: true, + }, + ultra_hd_broadcast: { + type: "string", + nullable: true, + }, + ultra_hd_h265: { + type: "string", + nullable: true, + }, + videoframe: { + type: "string", + nullable: true, + }, + vr_16kmaster: { + type: "string", + nullable: true, + }, + vr_2k_sos: { + type: "string", + nullable: true, + }, + vr_4k: { + type: "string", + nullable: true, + }, + vr_4k_sos: { + type: "string", + nullable: true, + }, + vr_4kmaster: { + type: "string", + nullable: true, + }, + vr_8k: { + type: "string", + nullable: true, + }, + vr_8kmaster: { + type: "string", + nullable: true, + }, + }, + readOnly: true, + }, + }, + required: ["categories", "duration", "formats", "id", "title", "url"], +} as const; diff --git a/lib/api/noirlab/codegen/services.gen.ts b/lib/api/noirlab/codegen/services.gen.ts new file mode 100644 index 00000000..3e74a814 --- /dev/null +++ b/lib/api/noirlab/codegen/services.gen.ts @@ -0,0 +1,45 @@ +// This file is auto-generated by @hey-api/openapi-ts + +import { + createClient, + createConfig, + type Options, +} from "@hey-api/client-fetch"; +import type { + ReleasesListData, + ReleasesListError, + ReleasesListResponse, + ReleasesRetrieveData, + ReleasesRetrieveError, + ReleasesRetrieveResponse, +} from "./types.gen"; + +export const client = createClient(createConfig()); + +export class ReleasesService { + public static releasesList( + options?: Options + ) { + return (options?.client ?? client).get< + ReleasesListResponse, + ReleasesListError, + ThrowOnError + >({ + ...options, + url: "/public/api/v2/releases/", + }); + } + + public static releasesRetrieve( + options: Options + ) { + return (options?.client ?? client).get< + ReleasesRetrieveResponse, + ReleasesRetrieveError, + ThrowOnError + >({ + ...options, + url: "/public/api/v2/releases/{id}/", + }); + } +} diff --git a/lib/api/noirlab/codegen/types.gen.ts b/lib/api/noirlab/codegen/types.gen.ts new file mode 100644 index 00000000..3f361bf9 --- /dev/null +++ b/lib/api/noirlab/codegen/types.gen.ts @@ -0,0 +1,818 @@ +// This file is auto-generated by @hey-api/openapi-ts + +export type Activity = { + /** + * ID of the activity, also used in URLs + */ + id: string; + name: string; + /** + * title to be displayed in the activity description when the activity is joined with other activities + */ + title?: string | null; + slogan?: string; + observatory: string; + readonly key_visual_en: string; + readonly key_visual_es: string; + /** + * Format: HH:MM + */ + duration: string; + readonly safety_tech_doc: string; + readonly conduct_tech_doc: string; + readonly liability_tech_doc: string; + readonly safety_tech_doc_es: string; + readonly conduct_tech_doc_es: string; + readonly liability_tech_doc_es: string; + readonly group_safety_tech_doc: string; + readonly group_liability_tech_doc: string; + readonly group_safety_tech_doc_es: string; + readonly group_liability_tech_doc_es: string; +}; + +export type Announcement = { + /** + * Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item. + */ + id: string; + lang?: string; + readonly url: string; + /** + * Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages. + */ + title: string; + /** + * Optional subtitle to be shown just above the headline. + */ + subtitle?: string; + description?: string; + /** + * Contacts + */ + contacts?: string; + /** + * Links + */ + links?: string; + featured?: boolean; + readonly release_date: string; + programs: Array; + readonly images: Array; + readonly videos: Array; +}; + +export type AnnouncementMini = { + /** + * Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item. + */ + id: string; + lang?: string; + readonly url: string; + /** + * Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages. + */ + title: string; + /** + * Optional subtitle to be shown just above the headline. + */ + subtitle?: string; + readonly release_date: string; + programs: Array; + readonly main_image: ImageMini; +}; + +export type AuthorDescription = { + readonly name: string; + /** + * Optional description, e.g.: "Author: ", or "Interview with" + */ + description?: string; + readonly photo: string; + /** + * Direct link to a JPG image, recommended size: 350px wide + */ + readonly static_photo: string; +}; + +export type BlankEnum = ""; + +export const BlankEnum = { + EMPTY_STRING: "", +} as const; + +export type Category = { + slug: string; + /** + * Title of query to be displayed to the user. + */ + name: string; + logo_url?: string | null; +}; + +export type Image = { + /** + * Id of image - used in the URL for the image as well as the filename for the different formats. + */ + id: string; + readonly url: string; + lang?: string; + /** + * Id of image - used in the URL for the image as well as the filename for the different formats. + */ + source?: string | null; + /** + * General descriptive title given to the image resource. + */ + title: string; + /** + * Short description of the full caption. + */ + headline?: string | null; + /** + * Full caption and related description text for the image resource. + */ + description?: string | null; + categories: Array; + /** + * The type of image/media resource. + * + * * `Observation` - Observation + * * `Artwork` - Artwork + * * `Photographic` - Photographic + * * `Planetary` - Planetary + * * `Simulation` - Simulation + * * `Collage` - Collage + * * `Chart` - Chart + */ + type?: (TypeEnum | BlankEnum | NullEnum) | null; + /** + * The minimum information that the Publisher would like to see mentioned when the resource is used. + */ + credit?: string; + readonly release_date: string; + /** + * Width in pixels of the image resource. + */ + width?: number | null; + /** + * Height in pixels of the image resource. + */ + height?: number | null; + featured?: boolean; + subject_name: Array; + readonly resources: Array<{ + ResourceType?: string; + MediaType?: string; + URL?: string; + FileSize?: number | null; + Dimensions?: Array | null; + ProjectionType?: string | null; + }>; + readonly formats: { + banner1920?: string | null; + eps?: string | null; + illustrator?: string | null; + illustrator_text?: string | null; + large?: string | null; + medium?: string | null; + news?: string | null; + newsfeature?: string | null; + newsmini?: string | null; + original?: string | null; + pdf?: string | null; + pl_original?: string | null; + pl_screen?: string | null; + pl_thumbs?: string | null; + portrait1080?: string | null; + poster400y?: string | null; + potwmedium?: string | null; + publicationjpg?: string | null; + publicationtiff?: string | null; + publicationtiff10k?: string | null; + publicationtiff25k?: string | null; + publicationtiff40k?: string | null; + screen?: string | null; + screen640?: string | null; + thumb150y?: string | null; + thumb300y?: string | null; + thumb350x?: string | null; + thumb700x?: string | null; + thumbs?: string | null; + wallpaper1?: string | null; + wallpaper2?: string | null; + wallpaper3?: string | null; + wallpaper4?: string | null; + wallpaper5?: string | null; + wallpaperthumbs?: string | null; + zoomable?: string | null; + }; +}; + +export type ImageItem = ImageTiny | ImageMini; + +export type ImageMini = { + /** + * Id of image - used in the URL for the image as well as the filename for the different formats. + */ + id: string; + readonly url: string; + lang?: string; + /** + * Id of image - used in the URL for the image as well as the filename for the different formats. + */ + source?: string | null; + /** + * General descriptive title given to the image resource. + */ + title: string; + /** + * Width in pixels of the image resource. + */ + width?: number | null; + /** + * Height in pixels of the image resource. + */ + height?: number | null; + featured?: boolean; + categories: Array; + readonly formats: { + banner1920?: string | null; + eps?: string | null; + illustrator?: string | null; + illustrator_text?: string | null; + large?: string | null; + medium?: string | null; + news?: string | null; + newsfeature?: string | null; + newsmini?: string | null; + original?: string | null; + pdf?: string | null; + pl_original?: string | null; + pl_screen?: string | null; + pl_thumbs?: string | null; + portrait1080?: string | null; + poster400y?: string | null; + potwmedium?: string | null; + publicationjpg?: string | null; + publicationtiff?: string | null; + publicationtiff10k?: string | null; + publicationtiff25k?: string | null; + publicationtiff40k?: string | null; + screen?: string | null; + screen640?: string | null; + thumb150y?: string | null; + thumb300y?: string | null; + thumb350x?: string | null; + thumb700x?: string | null; + thumbs?: string | null; + wallpaper1?: string | null; + wallpaper2?: string | null; + wallpaper3?: string | null; + wallpaper4?: string | null; + wallpaper5?: string | null; + wallpaperthumbs?: string | null; + zoomable?: string | null; + }; +}; + +export type ImageTiny = { + /** + * Id of image - used in the URL for the image as well as the filename for the different formats. + */ + id: string; + readonly url: string; + lang?: string; + /** + * General descriptive title given to the image resource. + */ + title: string; + /** + * Width in pixels of the image resource. + */ + width?: number | null; + /** + * Height in pixels of the image resource. + */ + height?: number | null; + readonly formats: { + thumb300y?: string | null; + screen?: string | null; + thumb700x?: string | null; + }; +}; + +export type Model3D = { + /** + * Ids are only allowed to contain letters, numbers, underscores or hyphens. They are used in URLs for the archive item. + */ + id: string; + /** + * Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages. + */ + title: string; + description?: string; + readonly credit: string; + /** + * Priority of product (100 highest, 0 lowest) - high priority products are ranked higher in search results than low priority products. + */ + priority?: number; + readonly release_date: string; + readonly assets: string; +}; + +export type NullEnum = unknown; + +export type Page = { + /** + * Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages. + */ + title: string; + content?: string; + /** + * The metadata description is normally shown in search engine results, making the description an effective way of capturing users attention. Description should be a clear description of the content and less the 200 characters long. Also used when sharing page on social media + */ + description?: string; + /** + * Comma-separated list of keywords for this page. Mainly used internally as search engines rarely use keywords to rank pages. + */ + keywords?: string; +}; + +export type PageRequest = { + /** + * Title is shown in browser window. Use a good informative title, since search engines normally display the title on their result pages. + */ + title: string; + content?: string; + /** + * The metadata description is normally shown in search engine results, making the description an effective way of capturing users attention. Description should be a clear description of the content and less the 200 characters long. Also used when sharing page on social media + */ + description?: string; + /** + * Comma-separated list of keywords for this page. Mainly used internally as search engines rarely use keywords to rank pages. + */ + keywords?: string; +}; + +export type PaginatedAnnouncementMiniList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedImageItemList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedImageList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedModel3dList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedPodcastList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedReleaseMiniList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array; +}; + +export type PaginatedVideoList = { + count?: number; + next?: string | null; + previous?: string | null; + results?: Array