web-dev-qa-db-fra.com

Meilleure fonction typée pour générer des requêtes multimédias

Je suis ce post: https://medium.com/@samuelresua/easy-media-queries-in-styled-components-690b78f5005

J'ai créé ce qui suit dans TypeScript, mais j'ai dû recourir à la saisie de any plus que je ne le dois, j'en suis sûr:

const breakpoints: ObjectMap<number> = {
  small: 767,
  medium: 992,
  large: 1200,
  extraLarge: 1240
};

export const media = Object.keys(breakpoints).reduce((acc: { [key: string]: (...args: any) => any }, label) => {
  acc[label] = (...args) => css`
     @media (min-width: ${breakpoints[label]}px) {
        ${css(...args as [any])};
     }
  `;
  return acc;
}, {});

En conséquence, je n'ai aucune aide dans mon IDE lorsque j'écris des styles dans mes blocs de requête multimédia:

styled.button<Props>`
  background: red; /* this must be a valid style */
  ${({ theme }) => theme.media.large`
      background: blue;
      foo: bar; /* this isn't caught */
   `

Quelqu'un sait-il comment je peux améliorer ma fonction media?

8
Mister Epic

styled-components vous permet d'utiliser des objets pour les propriétés CSS ainsi que des chaînes de modèles, et les objets sont bien mieux compris par le compilateur TypeScript. Essaye ça:

const breakpoints: ObjectMap<number> = {
  small: 767,
  medium: 992,
  large: 1200,
  extraLarge: 1240
};

export const media = (size: keyof typeof breakpoints, properties: CSSObject) => ({[`@media (min-width: ${breakpoints[size]})`]: properties});

const MyButton = styled.button<Props>(({ theme }) => ({
  background: red, // this must be a valid style
  ...theme.media('large', {
    background: blue // this is also validated :)
  })
}));
0
Robert Moore

Si je comprends bien, vous voulez vous débarrasser de any.

Voici comment vous pouvez le faire:

import {
  css,
  CSSObject,
  SimpleInterpolation,
  FlattenSimpleInterpolation,
} from 'styled-components';

type ObjectMap<T> = {[key: string]: T};

const breakpoints: ObjectMap<number> = {
  small: 767,
  medium: 992,
  large: 1200,
  extraLarge: 1240,
};

export const media = Object.keys(breakpoints).reduce(
  (
    acc: {
      [key: string]: (
        first: TemplateStringsArray | CSSObject,
        ...interpolations: SimpleInterpolation[]
      ) => FlattenSimpleInterpolation;
      // ^^^ this is the first implementation of  `BaseThemedCssFunction`
    },
    label
  ) => {
    acc[label] = (...args) => css`
      @media (min-width: ${breakpoints[label]}px) {
        ${css(...args)};
      }
    `;
    return acc;
  },
  {}
);
1
Maxim Mazurok