@ndla/image-search 6.0.132 → 6.0.134
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 +3 -3
- package/es/ImageSearchResult.js +1 -1
- package/es/PreviewImage.js +2 -2
- package/es/index.js +8 -0
- package/lib/ImageSearch.js +3 -3
- package/lib/ImageSearchResult.js +1 -1
- package/lib/PreviewImage.js +2 -2
- package/lib/index.d.ts +7 -0
- package/lib/index.js +7 -0
- package/package.json +9 -9
package/es/ImageSearch.js
CHANGED
|
@@ -11,9 +11,9 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
|
|
|
11
11
|
import { Component } from 'react';
|
|
12
12
|
import { css } from '@emotion/react';
|
|
13
13
|
import { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';
|
|
14
|
-
import Pager from '@ndla/pager';
|
|
15
14
|
import { InputContainer, InputV3 } from '@ndla/forms';
|
|
16
15
|
import { Search as SearchIcon } from '@ndla/icons/common';
|
|
16
|
+
import Pager from '@ndla/pager';
|
|
17
17
|
import ImageSearchResult from './ImageSearchResult';
|
|
18
18
|
import { jsx as _jsx } from "@emotion/react/jsx-runtime";
|
|
19
19
|
import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
|
|
@@ -56,14 +56,14 @@ const ImageSearchWrapper = /*#__PURE__*/_styled("div", {
|
|
|
56
56
|
until: breakpoints.wide
|
|
57
57
|
}), "{.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({
|
|
58
58
|
from: breakpoints.wide
|
|
59
|
-
}), "{.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"]} */"));
|
|
59
|
+
}), "{.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 { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\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"]} */"));
|
|
60
60
|
const searchIconCss = process.env.NODE_ENV === "production" ? {
|
|
61
61
|
name: "imhgxk-searchIconCss",
|
|
62
62
|
styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;"
|
|
63
63
|
} : {
|
|
64
64
|
name: "imhgxk-searchIconCss",
|
|
65
65
|
styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;",
|
|
66
|
-
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"]} */",
|
|
66
|
+
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 { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\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"]} */",
|
|
67
67
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
68
68
|
};
|
|
69
69
|
class ImageSearch extends Component {
|
package/es/ImageSearchResult.js
CHANGED
|
@@ -22,7 +22,7 @@ const StyledButton = /*#__PURE__*/_styled(ButtonV2, {
|
|
|
22
22
|
} : {
|
|
23
23
|
name: "1fttcpj",
|
|
24
24
|
styles: "display:flex;flex-direction:column",
|
|
25
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
25
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkltYWdlU2VhcmNoUmVzdWx0LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFjcUMiLCJmaWxlIjoiSW1hZ2VTZWFyY2hSZXN1bHQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTctcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgeyBCdXR0b25WMiB9IGZyb20gJ0BuZGxhL2J1dHRvbic7XG5pbXBvcnQgeyBJSW1hZ2VNZXRhSW5mb3JtYXRpb25WMyB9IGZyb20gJ0BuZGxhL3R5cGVzLWJhY2tlbmQvaW1hZ2UtYXBpJztcbmltcG9ydCBQcmV2aWV3SW1hZ2UgZnJvbSAnLi9QcmV2aWV3SW1hZ2UnO1xuaW1wb3J0IHsgZ2V0UHJldmlld1NyY1NldHMgfSBmcm9tICcuL3V0aWwvaW1hZ2VVdGlsJztcblxuY29uc3QgU3R5bGVkQnV0dG9uID0gc3R5bGVkKEJ1dHRvblYyKWBcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbmA7XG5cbmludGVyZmFjZSBQcm9wcyB7XG4gIGltYWdlOiBJSW1hZ2VNZXRhSW5mb3JtYXRpb25WMztcbiAgb25JbWFnZUNsaWNrOiAoaW1hZ2U6IElJbWFnZU1ldGFJbmZvcm1hdGlvblYzKSA9PiB2b2lkO1xuICBzZWxlY3RlZEltYWdlPzogSUltYWdlTWV0YUluZm9ybWF0aW9uVjM7XG4gIG9uU2VsZWN0SW1hZ2U6IChpbWFnZTogSUltYWdlTWV0YUluZm9ybWF0aW9uVjMsIHNhdmVBc01ldGFJbWFnZTogYm9vbGVhbikgPT4gdm9pZDtcbiAgdXNlSW1hZ2VUaXRsZTogc3RyaW5nO1xuICBzaG93Q2hlY2tib3g6IGJvb2xlYW47XG4gIGNoZWNrYm94TGFiZWw/OiBzdHJpbmc7XG4gIHVzZUFzTWV0YUltYWdlTGFiZWw/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIEltYWdlU2VhcmNoUmVzdWx0KHtcbiAgaW1hZ2UsXG4gIG9uSW1hZ2VDbGljayxcbiAgc2VsZWN0ZWRJbWFnZSxcbiAgb25TZWxlY3RJbWFnZSxcbiAgdXNlSW1hZ2VUaXRsZSxcbiAgc2hvd0NoZWNrYm94LFxuICBjaGVja2JveExhYmVsLFxufTogUHJvcHMpIHtcbiAgY29uc3QgYWN0aXZlID0gc2VsZWN0ZWRJbWFnZSAmJiBzZWxlY3RlZEltYWdlLmlkID09PSBpbWFnZS5pZCA/ICdhY3RpdmUnIDogJyc7XG5cbiAgcmV0dXJuIChcbiAgICA8ZGl2IGtleT17aW1hZ2UuaWR9IGNsYXNzTmFtZT17YGxpc3QtaXRlbSAke2FjdGl2ZX1gfT5cbiAgICAgIDxkaXYgY2xhc3NOYW1lPVwibGlzdC1pdGVtLWlubmVyXCI+XG4gICAgICAgIDxTdHlsZWRCdXR0b24gdmFyaWFudD1cInN0cmlwcGVkXCIgZGF0YS10ZXN0aWQ9XCJzZWxlY3QtaW1hZ2UtZnJvbS1saXN0XCIgb25DbGljaz17KCkgPT4gb25JbWFnZUNsaWNrKGltYWdlKX0+XG4gICAgICAgICAgPGltZ1xuICAgICAgICAgICAgcm9sZT1cInByZXNlbnRhdGlvblwiXG4gICAgICAgICAgICBhbHQ9XCJwcmVzZW50YXRpb25cIlxuICAgICAgICAgICAgc3JjU2V0PXtnZXRQcmV2aWV3U3JjU2V0cyhpbWFnZS5pbWFnZS5pbWFnZVVybCl9XG4gICAgICAgICAgICBzcmM9e2ltYWdlLmltYWdlLmltYWdlVXJsfVxuICAgICAgICAgIC8+XG4gICAgICAgICAgPHNwYW4gY2xhc3NOYW1lPVwibGlzdC1pdGVtLXRpdGxlXCI+e2ltYWdlLnRpdGxlLnRpdGxlfTwvc3Bhbj5cbiAgICAgICAgPC9TdHlsZWRCdXR0b24+XG4gICAgICA8L2Rpdj5cbiAgICAgIHtzZWxlY3RlZEltYWdlICYmIHNlbGVjdGVkSW1hZ2UuaWQgPT09IGltYWdlLmlkID8gKFxuICAgICAgICA8UHJldmlld0ltYWdlXG4gICAgICAgICAgaW1hZ2U9e3NlbGVjdGVkSW1hZ2V9XG4gICAgICAgICAgb25TZWxlY3RJbWFnZT17b25TZWxlY3RJbWFnZX1cbiAgICAgICAgICB1c2VJbWFnZVRpdGxlPXt1c2VJbWFnZVRpdGxlfVxuICAgICAgICAgIGNoZWNrYm94TGFiZWw9e2NoZWNrYm94TGFiZWx9XG4gICAgICAgICAgc2hvd0NoZWNrYm94PXtzaG93Q2hlY2tib3h9XG4gICAgICAgICAgYXJpYS1oaWRkZW49e3RydWV9XG4gICAgICAgIC8+XG4gICAgICApIDogKFxuICAgICAgICAnJ1xuICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKTtcbn1cbiJdfQ== */",
|
|
26
26
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
27
27
|
});
|
|
28
28
|
export default function ImageSearchResult(_ref) {
|
package/es/PreviewImage.js
CHANGED
|
@@ -9,10 +9,10 @@
|
|
|
9
9
|
import { useState } from 'react';
|
|
10
10
|
import { useTranslation } from 'react-i18next';
|
|
11
11
|
import { ButtonV2 } from '@ndla/button';
|
|
12
|
-
import { uuid } from '@ndla/util';
|
|
13
12
|
import { CheckboxItem } from '@ndla/forms';
|
|
14
|
-
import {
|
|
13
|
+
import { uuid } from '@ndla/util';
|
|
15
14
|
import ImageMeta from './ImageMeta';
|
|
15
|
+
import { getSrcSets } from './util/imageUtil';
|
|
16
16
|
import { jsx as _jsx } from "@emotion/react/jsx-runtime";
|
|
17
17
|
import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
|
|
18
18
|
const PreviewImage = _ref => {
|
package/es/index.js
CHANGED
|
@@ -1,3 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2017-present, NDLA.
|
|
3
|
+
*
|
|
4
|
+
* This source code is licensed under the GPLv3 license found in the
|
|
5
|
+
* LICENSE file in the root directory of this source tree.
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
8
|
+
|
|
1
9
|
import ImageSearch from './ImageSearch';
|
|
2
10
|
export { default as ImageMeta } from './ImageMeta';
|
|
3
11
|
export default ImageSearch;
|
package/lib/ImageSearch.js
CHANGED
|
@@ -8,9 +8,9 @@ var _base = _interopRequireDefault(require("@emotion/styled/base"));
|
|
|
8
8
|
var _react = require("react");
|
|
9
9
|
var _react2 = require("@emotion/react");
|
|
10
10
|
var _core = require("@ndla/core");
|
|
11
|
-
var _pager = _interopRequireDefault(require("@ndla/pager"));
|
|
12
11
|
var _forms = require("@ndla/forms");
|
|
13
12
|
var _common = require("@ndla/icons/common");
|
|
13
|
+
var _pager = _interopRequireDefault(require("@ndla/pager"));
|
|
14
14
|
var _ImageSearchResult = _interopRequireDefault(require("./ImageSearchResult"));
|
|
15
15
|
var _jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
16
16
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
@@ -60,14 +60,14 @@ const ImageSearchWrapper = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
60
60
|
until: _core.breakpoints.wide
|
|
61
61
|
}), "{.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({
|
|
62
62
|
from: _core.breakpoints.wide
|
|
63
|
-
}), "{.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"]} */"));
|
|
63
|
+
}), "{.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 { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\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"]} */"));
|
|
64
64
|
const searchIconCss = process.env.NODE_ENV === "production" ? {
|
|
65
65
|
name: "imhgxk-searchIconCss",
|
|
66
66
|
styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;"
|
|
67
67
|
} : {
|
|
68
68
|
name: "imhgxk-searchIconCss",
|
|
69
69
|
styles: "border:0;background:transparent;margin:0;padding:0;label:searchIconCss;",
|
|
70
|
-
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"]} */",
|
|
70
|
+
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 { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { fonts, colors, spacing, mq, breakpoints } from '@ndla/core';\nimport { InputContainer, InputV3 } from '@ndla/forms';\nimport { Search as SearchIcon } from '@ndla/icons/common';\nimport Pager from '@ndla/pager';\nimport { IImageMetaInformationV3, ISearchResultV3, ISearchParams } from '@ndla/types-backend/image-api';\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"]} */",
|
|
71
71
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
72
72
|
};
|
|
73
73
|
class ImageSearch extends _react.Component {
|
package/lib/ImageSearchResult.js
CHANGED
|
@@ -26,7 +26,7 @@ const StyledButton = /*#__PURE__*/(0, _base.default)(_button.ButtonV2, {
|
|
|
26
26
|
} : {
|
|
27
27
|
name: "1fttcpj",
|
|
28
28
|
styles: "display:flex;flex-direction:column",
|
|
29
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
29
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkltYWdlU2VhcmNoUmVzdWx0LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFjcUMiLCJmaWxlIjoiSW1hZ2VTZWFyY2hSZXN1bHQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTctcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgeyBCdXR0b25WMiB9IGZyb20gJ0BuZGxhL2J1dHRvbic7XG5pbXBvcnQgeyBJSW1hZ2VNZXRhSW5mb3JtYXRpb25WMyB9IGZyb20gJ0BuZGxhL3R5cGVzLWJhY2tlbmQvaW1hZ2UtYXBpJztcbmltcG9ydCBQcmV2aWV3SW1hZ2UgZnJvbSAnLi9QcmV2aWV3SW1hZ2UnO1xuaW1wb3J0IHsgZ2V0UHJldmlld1NyY1NldHMgfSBmcm9tICcuL3V0aWwvaW1hZ2VVdGlsJztcblxuY29uc3QgU3R5bGVkQnV0dG9uID0gc3R5bGVkKEJ1dHRvblYyKWBcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbmA7XG5cbmludGVyZmFjZSBQcm9wcyB7XG4gIGltYWdlOiBJSW1hZ2VNZXRhSW5mb3JtYXRpb25WMztcbiAgb25JbWFnZUNsaWNrOiAoaW1hZ2U6IElJbWFnZU1ldGFJbmZvcm1hdGlvblYzKSA9PiB2b2lkO1xuICBzZWxlY3RlZEltYWdlPzogSUltYWdlTWV0YUluZm9ybWF0aW9uVjM7XG4gIG9uU2VsZWN0SW1hZ2U6IChpbWFnZTogSUltYWdlTWV0YUluZm9ybWF0aW9uVjMsIHNhdmVBc01ldGFJbWFnZTogYm9vbGVhbikgPT4gdm9pZDtcbiAgdXNlSW1hZ2VUaXRsZTogc3RyaW5nO1xuICBzaG93Q2hlY2tib3g6IGJvb2xlYW47XG4gIGNoZWNrYm94TGFiZWw/OiBzdHJpbmc7XG4gIHVzZUFzTWV0YUltYWdlTGFiZWw/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIEltYWdlU2VhcmNoUmVzdWx0KHtcbiAgaW1hZ2UsXG4gIG9uSW1hZ2VDbGljayxcbiAgc2VsZWN0ZWRJbWFnZSxcbiAgb25TZWxlY3RJbWFnZSxcbiAgdXNlSW1hZ2VUaXRsZSxcbiAgc2hvd0NoZWNrYm94LFxuICBjaGVja2JveExhYmVsLFxufTogUHJvcHMpIHtcbiAgY29uc3QgYWN0aXZlID0gc2VsZWN0ZWRJbWFnZSAmJiBzZWxlY3RlZEltYWdlLmlkID09PSBpbWFnZS5pZCA/ICdhY3RpdmUnIDogJyc7XG5cbiAgcmV0dXJuIChcbiAgICA8ZGl2IGtleT17aW1hZ2UuaWR9IGNsYXNzTmFtZT17YGxpc3QtaXRlbSAke2FjdGl2ZX1gfT5cbiAgICAgIDxkaXYgY2xhc3NOYW1lPVwibGlzdC1pdGVtLWlubmVyXCI+XG4gICAgICAgIDxTdHlsZWRCdXR0b24gdmFyaWFudD1cInN0cmlwcGVkXCIgZGF0YS10ZXN0aWQ9XCJzZWxlY3QtaW1hZ2UtZnJvbS1saXN0XCIgb25DbGljaz17KCkgPT4gb25JbWFnZUNsaWNrKGltYWdlKX0+XG4gICAgICAgICAgPGltZ1xuICAgICAgICAgICAgcm9sZT1cInByZXNlbnRhdGlvblwiXG4gICAgICAgICAgICBhbHQ9XCJwcmVzZW50YXRpb25cIlxuICAgICAgICAgICAgc3JjU2V0PXtnZXRQcmV2aWV3U3JjU2V0cyhpbWFnZS5pbWFnZS5pbWFnZVVybCl9XG4gICAgICAgICAgICBzcmM9e2ltYWdlLmltYWdlLmltYWdlVXJsfVxuICAgICAgICAgIC8+XG4gICAgICAgICAgPHNwYW4gY2xhc3NOYW1lPVwibGlzdC1pdGVtLXRpdGxlXCI+e2ltYWdlLnRpdGxlLnRpdGxlfTwvc3Bhbj5cbiAgICAgICAgPC9TdHlsZWRCdXR0b24+XG4gICAgICA8L2Rpdj5cbiAgICAgIHtzZWxlY3RlZEltYWdlICYmIHNlbGVjdGVkSW1hZ2UuaWQgPT09IGltYWdlLmlkID8gKFxuICAgICAgICA8UHJldmlld0ltYWdlXG4gICAgICAgICAgaW1hZ2U9e3NlbGVjdGVkSW1hZ2V9XG4gICAgICAgICAgb25TZWxlY3RJbWFnZT17b25TZWxlY3RJbWFnZX1cbiAgICAgICAgICB1c2VJbWFnZVRpdGxlPXt1c2VJbWFnZVRpdGxlfVxuICAgICAgICAgIGNoZWNrYm94TGFiZWw9e2NoZWNrYm94TGFiZWx9XG4gICAgICAgICAgc2hvd0NoZWNrYm94PXtzaG93Q2hlY2tib3h9XG4gICAgICAgICAgYXJpYS1oaWRkZW49e3RydWV9XG4gICAgICAgIC8+XG4gICAgICApIDogKFxuICAgICAgICAnJ1xuICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKTtcbn1cbiJdfQ== */",
|
|
30
30
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
31
31
|
});
|
|
32
32
|
function ImageSearchResult(_ref) {
|
package/lib/PreviewImage.js
CHANGED
|
@@ -7,10 +7,10 @@ exports.default = void 0;
|
|
|
7
7
|
var _react = require("react");
|
|
8
8
|
var _reactI18next = require("react-i18next");
|
|
9
9
|
var _button = require("@ndla/button");
|
|
10
|
-
var _util = require("@ndla/util");
|
|
11
10
|
var _forms = require("@ndla/forms");
|
|
12
|
-
var
|
|
11
|
+
var _util = require("@ndla/util");
|
|
13
12
|
var _ImageMeta = _interopRequireDefault(require("./ImageMeta"));
|
|
13
|
+
var _imageUtil = require("./util/imageUtil");
|
|
14
14
|
var _jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
15
15
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
16
16
|
/**
|
package/lib/index.d.ts
CHANGED
|
@@ -1,3 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2017-present, NDLA.
|
|
3
|
+
*
|
|
4
|
+
* This source code is licensed under the GPLv3 license found in the
|
|
5
|
+
* LICENSE file in the root directory of this source tree.
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
1
8
|
import ImageSearch from './ImageSearch';
|
|
2
9
|
export { default as ImageMeta } from './ImageMeta';
|
|
3
10
|
export default ImageSearch;
|
package/lib/index.js
CHANGED
|
@@ -13,4 +13,11 @@ exports.default = void 0;
|
|
|
13
13
|
var _ImageSearch = _interopRequireDefault(require("./ImageSearch"));
|
|
14
14
|
var _ImageMeta = _interopRequireDefault(require("./ImageMeta"));
|
|
15
15
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
16
|
+
/**
|
|
17
|
+
* Copyright (c) 2017-present, NDLA.
|
|
18
|
+
*
|
|
19
|
+
* This source code is licensed under the GPLv3 license found in the
|
|
20
|
+
* LICENSE file in the root directory of this source tree.
|
|
21
|
+
*
|
|
22
|
+
*/
|
|
16
23
|
var _default = exports.default = _ImageSearch.default;
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ndla/image-search",
|
|
3
|
-
"version": "6.0.
|
|
3
|
+
"version": "6.0.134",
|
|
4
4
|
"description": "A simple library for searching images from NDLA",
|
|
5
5
|
"license": "GPL-3.0",
|
|
6
6
|
"main": "lib/index.js",
|
|
@@ -26,13 +26,13 @@
|
|
|
26
26
|
"es"
|
|
27
27
|
],
|
|
28
28
|
"dependencies": {
|
|
29
|
-
"@ndla/button": "^12.0.
|
|
30
|
-
"@ndla/core": "^4.2.
|
|
31
|
-
"@ndla/forms": "^5.2.
|
|
32
|
-
"@ndla/icons": "^4.2.
|
|
33
|
-
"@ndla/pager": "^2.2.
|
|
34
|
-
"@ndla/ui": "^50.
|
|
35
|
-
"@ndla/util": "^4.0.
|
|
29
|
+
"@ndla/button": "^12.0.17",
|
|
30
|
+
"@ndla/core": "^4.2.9",
|
|
31
|
+
"@ndla/forms": "^5.2.6",
|
|
32
|
+
"@ndla/icons": "^4.2.4",
|
|
33
|
+
"@ndla/pager": "^2.2.42",
|
|
34
|
+
"@ndla/ui": "^50.4.0",
|
|
35
|
+
"@ndla/util": "^4.0.3",
|
|
36
36
|
"pretty-bytes": "^5.6.0"
|
|
37
37
|
},
|
|
38
38
|
"devDependencies": {
|
|
@@ -51,5 +51,5 @@
|
|
|
51
51
|
"publishConfig": {
|
|
52
52
|
"access": "public"
|
|
53
53
|
},
|
|
54
|
-
"gitHead": "
|
|
54
|
+
"gitHead": "5c65dfc8de95fb464182f86a40b74868ca30a29e"
|
|
55
55
|
}
|