@ndla/image-search 6.0.120 → 6.0.122

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/es/ImageSearch.js CHANGED
@@ -25,7 +25,7 @@ import { Component } from 'react';
25
25
  import { css } from '@emotion/react';
26
26
  import { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';
27
27
  import Pager from '@ndla/pager';
28
- import { Input } from '@ndla/forms';
28
+ import { InputContainer, InputV3 } from '@ndla/forms';
29
29
  import { Search as SearchIcon } from '@ndla/icons/common';
30
30
  import ImageSearchResult from './ImageSearchResult';
31
31
  import { jsx as _jsx } from "@emotion/react/jsx-runtime";
@@ -69,14 +69,14 @@ var ImageSearchWrapper = /*#__PURE__*/_styled("div", {
69
69
  until: breakpoints.wide
70
70
  }), "{.list-item .image-preview{width:400%;}.list-item:nth-of-type(4n - 2) .image-preview{margin-left:-100%;}.list-item:nth-of-type(4n - 1) .image-preview{margin-left:-200%;}.list-item:nth-of-type(4n) .image-preview{margin-left:-300%;}}", mq.range({
71
71
  from: breakpoints.wide
72
- }), "{.list-item .image-preview{width:500%;}.list-item:nth-of-type(5n - 3) .image-preview{margin-left:-100%;}.list-item:nth-of-type(5n - 2) .image-preview{margin-left:-200%;}.list-item:nth-of-type(5n - 1) .image-preview{margin-left:-300%;}.list-item:nth-of-type(5n) .image-preview{margin-left:-400%;}}@keyframes fadeInSearchPreview{0%{display:none;opacity:0;}1%{opacity:0;display:flex;transform:translateY(-20px);}100%{opacity:1;transform:translateY(0px);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AAkBqC","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { Input } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <Input\n          placeholder={searchPlaceholder}\n          // eslint-disable-next-line jsx-a11y/no-autofocus\n          autoFocus\n          iconRight={\n            <button\n              css={searchIconCss}\n              aria-label={searchButtonTitle}\n              type=\"button\"\n              onClick={() => {\n                this.searchImages({ query: queryString, page: 1 });\n              }}\n            >\n              <SearchIcon />\n            </button>\n          }\n          value={queryString}\n          onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n          onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n            if (evt.key === 'Enter') {\n              evt.preventDefault();\n              this.searchImages({ query: queryString, page: 1 });\n            }\n          }}\n        />\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */"));
72
+ }), "{.list-item .image-preview{width:500%;}.list-item:nth-of-type(5n - 3) .image-preview{margin-left:-100%;}.list-item:nth-of-type(5n - 2) .image-preview{margin-left:-200%;}.list-item:nth-of-type(5n - 1) .image-preview{margin-left:-300%;}.list-item:nth-of-type(5n) .image-preview{margin-left:-400%;}}@keyframes fadeInSearchPreview{0%{display:none;opacity:0;}1%{opacity:0;display:flex;transform:translateY(-20px);}100%{opacity:1;transform:translateY(0px);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AAkBqC","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <InputContainer>\n          <InputV3\n            placeholder={searchPlaceholder}\n            // eslint-disable-next-line jsx-a11y/no-autofocus\n            autoFocus\n            value={queryString}\n            onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n            onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n              if (evt.key === 'Enter') {\n                evt.preventDefault();\n                this.searchImages({ query: queryString, page: 1 });\n              }\n            }}\n          />\n          <button\n            css={searchIconCss}\n            aria-label={searchButtonTitle}\n            type=\"button\"\n            onClick={() => {\n              this.searchImages({ query: queryString, page: 1 });\n            }}\n          >\n            <SearchIcon />\n          </button>\n        </InputContainer>\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */"));
73
73
  var searchIconCss = process.env.NODE_ENV === "production" ? {
74
74
  name: "imhgxk-searchIconCss",
75
75
  styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;"
76
76
  } : {
77
77
  name: "imhgxk-searchIconCss",
78
78
  styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;",
79
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AA2QyB","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { Input } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <Input\n          placeholder={searchPlaceholder}\n          // eslint-disable-next-line jsx-a11y/no-autofocus\n          autoFocus\n          iconRight={\n            <button\n              css={searchIconCss}\n              aria-label={searchButtonTitle}\n              type=\"button\"\n              onClick={() => {\n                this.searchImages({ query: queryString, page: 1 });\n              }}\n            >\n              <SearchIcon />\n            </button>\n          }\n          value={queryString}\n          onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n          onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n            if (evt.key === 'Enter') {\n              evt.preventDefault();\n              this.searchImages({ query: queryString, page: 1 });\n            }\n          }}\n        />\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */",
79
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AA2QyB","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <InputContainer>\n          <InputV3\n            placeholder={searchPlaceholder}\n            // eslint-disable-next-line jsx-a11y/no-autofocus\n            autoFocus\n            value={queryString}\n            onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n            onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n              if (evt.key === 'Enter') {\n                evt.preventDefault();\n                this.searchImages({ query: queryString, page: 1 });\n              }\n            }}\n          />\n          <button\n            css={searchIconCss}\n            aria-label={searchButtonTitle}\n            type=\"button\"\n            onClick={() => {\n              this.searchImages({ query: queryString, page: 1 });\n            }}\n          >\n            <SearchIcon />\n          </button>\n        </InputContainer>\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */",
80
80
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
81
81
  };
82
82
  var ImageSearch = /*#__PURE__*/function (_Component) {
@@ -187,12 +187,28 @@ var ImageSearch = /*#__PURE__*/function (_Component) {
187
187
  var page = queryObject.page;
188
188
  var noResultsFound = !searching && images.length === 0;
189
189
  return _jsxs(ImageSearchWrapper, {
190
- children: [_jsx(Input, {
191
- placeholder: searchPlaceholder
192
- // eslint-disable-next-line jsx-a11y/no-autofocus
193
- ,
194
- autoFocus: true,
195
- iconRight: _jsx("button", {
190
+ children: [_jsxs(InputContainer, {
191
+ children: [_jsx(InputV3, {
192
+ placeholder: searchPlaceholder
193
+ // eslint-disable-next-line jsx-a11y/no-autofocus
194
+ ,
195
+ autoFocus: true,
196
+ value: queryString,
197
+ onChange: function onChange(evt) {
198
+ return _this4.setState({
199
+ queryString: evt.target.value
200
+ });
201
+ },
202
+ onKeyPress: function onKeyPress(evt) {
203
+ if (evt.key === 'Enter') {
204
+ evt.preventDefault();
205
+ _this4.searchImages({
206
+ query: queryString,
207
+ page: 1
208
+ });
209
+ }
210
+ }
211
+ }), _jsx("button", {
196
212
  css: searchIconCss,
197
213
  "aria-label": searchButtonTitle,
198
214
  type: "button",
@@ -203,22 +219,7 @@ var ImageSearch = /*#__PURE__*/function (_Component) {
203
219
  });
204
220
  },
205
221
  children: _jsx(SearchIcon, {})
206
- }),
207
- value: queryString,
208
- onChange: function onChange(evt) {
209
- return _this4.setState({
210
- queryString: evt.target.value
211
- });
212
- },
213
- onKeyPress: function onKeyPress(evt) {
214
- if (evt.key === 'Enter') {
215
- evt.preventDefault();
216
- _this4.searchImages({
217
- query: queryString,
218
- page: 1
219
- });
220
- }
221
- }
222
+ })]
222
223
  }), noResultsFound && this.props.noResults, _jsx("div", {
223
224
  className: "list",
224
225
  children: images.map(function (image) {
@@ -73,14 +73,14 @@ var ImageSearchWrapper = /*#__PURE__*/(0, _base.default)("div", {
73
73
  until: _core.breakpoints.wide
74
74
  }), "{.list-item .image-preview{width:400%;}.list-item:nth-of-type(4n - 2) .image-preview{margin-left:-100%;}.list-item:nth-of-type(4n - 1) .image-preview{margin-left:-200%;}.list-item:nth-of-type(4n) .image-preview{margin-left:-300%;}}", _core.mq.range({
75
75
  from: _core.breakpoints.wide
76
- }), "{.list-item .image-preview{width:500%;}.list-item:nth-of-type(5n - 3) .image-preview{margin-left:-100%;}.list-item:nth-of-type(5n - 2) .image-preview{margin-left:-200%;}.list-item:nth-of-type(5n - 1) .image-preview{margin-left:-300%;}.list-item:nth-of-type(5n) .image-preview{margin-left:-400%;}}@keyframes fadeInSearchPreview{0%{display:none;opacity:0;}1%{opacity:0;display:flex;transform:translateY(-20px);}100%{opacity:1;transform:translateY(0px);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AAkBqC","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { Input } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <Input\n          placeholder={searchPlaceholder}\n          // eslint-disable-next-line jsx-a11y/no-autofocus\n          autoFocus\n          iconRight={\n            <button\n              css={searchIconCss}\n              aria-label={searchButtonTitle}\n              type=\"button\"\n              onClick={() => {\n                this.searchImages({ query: queryString, page: 1 });\n              }}\n            >\n              <SearchIcon />\n            </button>\n          }\n          value={queryString}\n          onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n          onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n            if (evt.key === 'Enter') {\n              evt.preventDefault();\n              this.searchImages({ query: queryString, page: 1 });\n            }\n          }}\n        />\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */"));
76
+ }), "{.list-item .image-preview{width:500%;}.list-item:nth-of-type(5n - 3) .image-preview{margin-left:-100%;}.list-item:nth-of-type(5n - 2) .image-preview{margin-left:-200%;}.list-item:nth-of-type(5n - 1) .image-preview{margin-left:-300%;}.list-item:nth-of-type(5n) .image-preview{margin-left:-400%;}}@keyframes fadeInSearchPreview{0%{display:none;opacity:0;}1%{opacity:0;display:flex;transform:translateY(-20px);}100%{opacity:1;transform:translateY(0px);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AAkBqC","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <InputContainer>\n          <InputV3\n            placeholder={searchPlaceholder}\n            // eslint-disable-next-line jsx-a11y/no-autofocus\n            autoFocus\n            value={queryString}\n            onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n            onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n              if (evt.key === 'Enter') {\n                evt.preventDefault();\n                this.searchImages({ query: queryString, page: 1 });\n              }\n            }}\n          />\n          <button\n            css={searchIconCss}\n            aria-label={searchButtonTitle}\n            type=\"button\"\n            onClick={() => {\n              this.searchImages({ query: queryString, page: 1 });\n            }}\n          >\n            <SearchIcon />\n          </button>\n        </InputContainer>\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */"));
77
77
  var searchIconCss = process.env.NODE_ENV === "production" ? {
78
78
  name: "imhgxk-searchIconCss",
79
79
  styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;"
80
80
  } : {
81
81
  name: "imhgxk-searchIconCss",
82
82
  styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;",
83
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AA2QyB","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { Input } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <Input\n          placeholder={searchPlaceholder}\n          // eslint-disable-next-line jsx-a11y/no-autofocus\n          autoFocus\n          iconRight={\n            <button\n              css={searchIconCss}\n              aria-label={searchButtonTitle}\n              type=\"button\"\n              onClick={() => {\n                this.searchImages({ query: queryString, page: 1 });\n              }}\n            >\n              <SearchIcon />\n            </button>\n          }\n          value={queryString}\n          onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n          onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n            if (evt.key === 'Enter') {\n              evt.preventDefault();\n              this.searchImages({ query: queryString, page: 1 });\n            }\n          }}\n        />\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */",
83
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageSearch.tsx"],"names":[],"mappings":"AA2QyB","file":"ImageSearch.tsx","sourcesContent":["/**\n * Copyright (c) 2017-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ChangeEvent, Component, ReactNode, KeyboardEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport ImageSearchResult from './ImageSearchResult';\n\nconst ImageSearchWrapper = styled.div`\n  .text {\n    text-align: center;\n  }\n  .list-item {\n    position: relative;\n    float: left;\n    height: 210px;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet })} {\n      width: 50%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide })} {\n      width: 33.3%;\n    }\n    ${mq.range({ from: breakpoints.desktop })} {\n      width: 25%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 20%;\n    }\n    &.active {\n      height: inherit;\n    }\n  }\n\n  .list {\n    display: flex;\n    align-items: stretch;\n    flex-flow: row wrap;\n    position: relative;\n    margin-left: -${spacing.small};\n    margin-right: -${spacing.small};\n    margin-top: ${spacing.normal};\n  }\n\n  .list-item-inner {\n    padding: ${spacing.small};\n    text-align: center;\n    height: 210px;\n\n    .list-item-title {\n      margin: ${spacing.xsmall} 0;\n      ${fonts.sizes('14px', 1.2)};\n      overflow: hidden;\n      text-overflow: ellipsis;\n      display: -webkit-box;\n      -webkit-box-orient: vertical;\n      -webkit-line-clamp: 3;\n      max-height: ${spacing.large};\n    }\n  }\n\n  .list-item-inner img {\n    max-height: 135px;\n    max-width: 100%;\n    border: 2px solid white;\n    transition: border-color 100ms ease;\n  }\n\n  .list-item-inner:hover {\n    img {\n      border: 2px solid ${colors.brand.primary};\n    }\n  }\n\n  .list-item-inner > .list-item-inner img {\n    border: 2px solid ${colors.brand.primary};\n  }\n\n  .list-item.active > .list-item-inner::after {\n    animation: fadeInSearchPreview 300ms ease;\n    top: 190px;\n    left: 50%;\n    border: solid transparent;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n    border-color: rgba(136, 183, 213, 0);\n    border-bottom-color: ${colors.brand.lighter};\n    border-width: ${spacing.normal};\n    margin-left: -${spacing.normal};\n  }\n\n  .image-preview {\n    animation: fadeInSearchPreview 300ms ease;\n    position: relative;\n    width: 100%;\n    ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n      width: 200%;\n    }\n    ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n      width: 300%;\n    }\n    ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n      width: 400%;\n    }\n    ${mq.range({ from: breakpoints.wide })} {\n      width: 500%;\n    }\n    background-color: ${colors.brand.lighter};\n    border-radius: 2px;\n    margin: 20px 0;\n    display: flex;\n    align-items: flex-start;\n    justify-content: flex-start;\n\n    ${mq.range({ until: breakpoints.mobileWide })} {\n      display: block;\n    }\n\n    .image {\n      max-width: 50%;\n      padding: ${spacing.small};\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n      }\n    }\n\n    .info {\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .image img {\n      max-width: 100%;\n      max-height: 300px;\n    }\n\n    .information {\n      width: 50%;\n      padding: calc(${spacing.normal} - ${spacing.xsmall}) ${spacing.normal} ${spacing.normal} ${spacing.small};\n      word-break: initial;\n      ${mq.range({ until: breakpoints.mobileWide })} {\n        width: 100%;\n        padding: 0 ${spacing.small} ${spacing.normal};\n      }\n    }\n\n    .information > * {\n      margin-top: ${spacing.small};\n    }\n\n    .title {\n      padding-top: 0;\n      margin: 0;\n      line-height: 1.3;\n    }\n    .text--left {\n      width: 20%;\n      display: inline-block;\n    }\n\n    .text--right {\n      width: 80%;\n      display: inline-block;\n    }\n    .tags > b {\n      display: block;\n      margin-bottom: ${spacing.normal};\n    }\n\n    .tags > .tag_item {\n      font-weight: ${fonts.weight.semibold};\n      margin-right: ${spacing.xsmall};\n      margin-bottom: ${spacing.xsmall};\n      display: inline-block;\n      ${fonts.sizes('16px', 1.3)}\n    }\n\n    .tags > .tag_item:hover {\n      text-decoration: none;\n    }\n\n    .clear {\n      clear: both;\n    }\n\n    width: 100%;\n  }\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    .list-item .image-preview {\n      width: 200%;\n    }\n    .list-item:nth-of-type(2n) .image-preview {\n      margin-left: -100%;\n    }\n  }\n  ${mq.range({ from: breakpoints.tabletWide, until: breakpoints.desktop })} {\n    .list-item .image-preview {\n      width: 300%;\n    }\n    .list-item:nth-of-type(3n - 1) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(3n) .image-preview {\n      margin-left: -200%;\n    }\n  }\n  ${mq.range({ from: breakpoints.desktop, until: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 400%;\n    }\n    .list-item:nth-of-type(4n - 2) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(4n - 1) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(4n) .image-preview {\n      margin-left: -300%;\n    }\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    .list-item .image-preview {\n      width: 500%;\n    }\n    .list-item:nth-of-type(5n - 3) .image-preview {\n      margin-left: -100%;\n    }\n    .list-item:nth-of-type(5n - 2) .image-preview {\n      margin-left: -200%;\n    }\n    .list-item:nth-of-type(5n - 1) .image-preview {\n      margin-left: -300%;\n    }\n    .list-item:nth-of-type(5n) .image-preview {\n      margin-left: -400%;\n    }\n  }\n\n  @keyframes fadeInSearchPreview {\n    0% {\n      display: none;\n      opacity: 0;\n    }\n    1% {\n      opacity: 0;\n      display: flex;\n      transform: translateY(-20px);\n    }\n    100% {\n      opacity: 1;\n      transform: translateY(0px);\n    }\n  }\n`;\n\nconst searchIconCss = css`\n  border: 0;\n  background: transparent;\n  margin: 0;\n  padding: 0;\n`;\n\ninterface Props {\n  onImageSelect: (image: IImageMetaInformationV3) => void;\n  searchImages: (query: string | undefined, page: number | undefined) => Promise<ISearchResultV3>;\n  fetchImage: (id: number) => Promise<IImageMetaInformationV3>;\n  onError: (err: any) => void;\n  searchPlaceholder: string;\n  searchButtonTitle: string;\n  locale: string;\n  useImageTitle: string;\n  noResults?: ReactNode;\n  checkboxAction?: (image: IImageMetaInformationV3) => void;\n  showCheckbox?: boolean;\n  checkboxLabel?: string;\n}\n\ninterface State {\n  queryObject: ISearchParams;\n  images: IImageMetaInformationV3[];\n  selectedImage?: IImageMetaInformationV3;\n  lastPage: number;\n  searching: boolean;\n  queryString?: string;\n}\nclass ImageSearch extends Component<Props, State> {\n  constructor(props: Props) {\n    super(props);\n    this.state = {\n      queryObject: {\n        query: undefined,\n        page: 1,\n        pageSize: 16,\n      },\n      images: [],\n      selectedImage: undefined,\n      lastPage: 0,\n      searching: false,\n    };\n\n    this.searchImages = this.searchImages.bind(this);\n    this.onImageClick = this.onImageClick.bind(this);\n    this.onSelectImage = this.onSelectImage.bind(this);\n  }\n\n  componentDidMount() {\n    this.searchImages(this.state.queryObject);\n  }\n\n  onImageClick(image: IImageMetaInformationV3) {\n    const { onError, fetchImage } = this.props;\n    const { selectedImage } = this.state;\n    if (!selectedImage || image.id !== selectedImage.id) {\n      fetchImage(parseInt(image.id))\n        .then((result) => {\n          this.setState({ selectedImage: result });\n        })\n        .catch((err) => {\n          onError(err);\n        });\n    }\n  }\n\n  onSelectImage(image: IImageMetaInformationV3, saveAsMetaImage: boolean) {\n    const { onImageSelect, checkboxAction } = this.props;\n    this.setState({ selectedImage: undefined });\n    onImageSelect(image);\n    if (saveAsMetaImage) {\n      checkboxAction && checkboxAction(image);\n    }\n  }\n\n  searchImages(queryObject: ISearchParams) {\n    const { searchImages, onError } = this.props;\n    this.setState({ searching: true });\n    searchImages(queryObject.query, queryObject.page)\n      .then((result) => {\n        this.setState({\n          queryObject: {\n            query: queryObject.query,\n            pageSize: result.pageSize,\n            page: queryObject.page,\n          },\n          images: result.results,\n          lastPage: Math.ceil(result.totalCount / result.pageSize),\n          searching: false,\n        });\n      })\n      .catch((err) => {\n        onError(err);\n        this.setState({ searching: false });\n      });\n  }\n\n  render() {\n    const { searchPlaceholder, searchButtonTitle, useImageTitle, showCheckbox, checkboxLabel } = this.props;\n\n    const { queryObject, images, selectedImage, lastPage, searching, queryString } = this.state;\n\n    const { page } = queryObject;\n    const noResultsFound = !searching && images.length === 0;\n\n    return (\n      <ImageSearchWrapper>\n        <InputContainer>\n          <InputV3\n            placeholder={searchPlaceholder}\n            // eslint-disable-next-line jsx-a11y/no-autofocus\n            autoFocus\n            value={queryString}\n            onChange={(evt: ChangeEvent<HTMLInputElement>) => this.setState({ queryString: evt.target.value })}\n            onKeyPress={(evt: KeyboardEvent<HTMLInputElement>) => {\n              if (evt.key === 'Enter') {\n                evt.preventDefault();\n                this.searchImages({ query: queryString, page: 1 });\n              }\n            }}\n          />\n          <button\n            css={searchIconCss}\n            aria-label={searchButtonTitle}\n            type=\"button\"\n            onClick={() => {\n              this.searchImages({ query: queryString, page: 1 });\n            }}\n          >\n            <SearchIcon />\n          </button>\n        </InputContainer>\n        {noResultsFound && this.props.noResults}\n        <div className=\"list\">\n          {images.map((image) => (\n            <ImageSearchResult\n              key={image.id}\n              image={image}\n              onImageClick={this.onImageClick}\n              selectedImage={selectedImage}\n              onSelectImage={this.onSelectImage}\n              useImageTitle={useImageTitle}\n              showCheckbox={!!showCheckbox}\n              checkboxLabel={checkboxLabel}\n            />\n          ))}\n        </div>\n        <Pager\n          page={page ?? 1}\n          pathname=\"\"\n          lastPage={lastPage}\n          query={queryObject}\n          onClick={this.searchImages}\n          pageItemComponentClass=\"button\"\n        />\n      </ImageSearchWrapper>\n    );\n  }\n}\n\nexport default ImageSearch;\n"]} */",
84
84
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
85
85
  };
86
86
  var ImageSearch = /*#__PURE__*/function (_Component) {
@@ -191,12 +191,28 @@ var ImageSearch = /*#__PURE__*/function (_Component) {
191
191
  var page = queryObject.page;
192
192
  var noResultsFound = !searching && images.length === 0;
193
193
  return (0, _jsxRuntime.jsxs)(ImageSearchWrapper, {
194
- children: [(0, _jsxRuntime.jsx)(_forms.Input, {
195
- placeholder: searchPlaceholder
196
- // eslint-disable-next-line jsx-a11y/no-autofocus
197
- ,
198
- autoFocus: true,
199
- iconRight: (0, _jsxRuntime.jsx)("button", {
194
+ children: [(0, _jsxRuntime.jsxs)(_forms.InputContainer, {
195
+ children: [(0, _jsxRuntime.jsx)(_forms.InputV3, {
196
+ placeholder: searchPlaceholder
197
+ // eslint-disable-next-line jsx-a11y/no-autofocus
198
+ ,
199
+ autoFocus: true,
200
+ value: queryString,
201
+ onChange: function onChange(evt) {
202
+ return _this4.setState({
203
+ queryString: evt.target.value
204
+ });
205
+ },
206
+ onKeyPress: function onKeyPress(evt) {
207
+ if (evt.key === 'Enter') {
208
+ evt.preventDefault();
209
+ _this4.searchImages({
210
+ query: queryString,
211
+ page: 1
212
+ });
213
+ }
214
+ }
215
+ }), (0, _jsxRuntime.jsx)("button", {
200
216
  css: searchIconCss,
201
217
  "aria-label": searchButtonTitle,
202
218
  type: "button",
@@ -207,22 +223,7 @@ var ImageSearch = /*#__PURE__*/function (_Component) {
207
223
  });
208
224
  },
209
225
  children: (0, _jsxRuntime.jsx)(_common.Search, {})
210
- }),
211
- value: queryString,
212
- onChange: function onChange(evt) {
213
- return _this4.setState({
214
- queryString: evt.target.value
215
- });
216
- },
217
- onKeyPress: function onKeyPress(evt) {
218
- if (evt.key === 'Enter') {
219
- evt.preventDefault();
220
- _this4.searchImages({
221
- query: queryString,
222
- page: 1
223
- });
224
- }
225
- }
226
+ })]
226
227
  }), noResultsFound && this.props.noResults, (0, _jsxRuntime.jsx)("div", {
227
228
  className: "list",
228
229
  children: images.map(function (image) {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ndla/image-search",
3
- "version": "6.0.120",
3
+ "version": "6.0.122",
4
4
  "description": "A simple library for searching images from NDLA",
5
5
  "license": "GPL-3.0",
6
6
  "main": "lib/index.js",
@@ -26,12 +26,12 @@
26
26
  "es"
27
27
  ],
28
28
  "dependencies": {
29
- "@ndla/button": "^12.0.12",
29
+ "@ndla/button": "^12.0.13",
30
30
  "@ndla/core": "^4.2.7",
31
- "@ndla/forms": "^5.1.2",
32
- "@ndla/icons": "^4.2.0",
33
- "@ndla/pager": "^2.2.37",
34
- "@ndla/ui": "^49.0.12",
31
+ "@ndla/forms": "^5.2.0",
32
+ "@ndla/icons": "^4.2.1",
33
+ "@ndla/pager": "^2.2.38",
34
+ "@ndla/ui": "^50.0.0",
35
35
  "@ndla/util": "^4.0.1",
36
36
  "pretty-bytes": "^5.6.0"
37
37
  },
@@ -46,5 +46,5 @@
46
46
  "publishConfig": {
47
47
  "access": "public"
48
48
  },
49
- "gitHead": "1764d9f407bcf76cbf7163f367c1bee33cfdaa54"
49
+ "gitHead": "027f7a38317f7e30dd59eae70f48c6c4be2d7de0"
50
50
  }