Skip to content

Commit 46b349c

Browse files
committed
Parse media queries and throw an error for invalid queries
1 parent b3ec8c4 commit 46b349c

File tree

7 files changed

+304
-583
lines changed

7 files changed

+304
-583
lines changed

.prettierignore

+2
Original file line numberDiff line numberDiff line change
@@ -1 +1,3 @@
11
dist
2+
package.json
3+
package-lock.json

package.json

+18-5
Original file line numberDiff line numberDiff line change
@@ -30,20 +30,33 @@
3030
}
3131
},
3232
"lint-staged": {
33-
"*.{js,json,md}": ["prettier --write", "git add"]
33+
"*.{js,json,md}": [
34+
"prettier --write",
35+
"git add"
36+
]
3437
},
3538
"prettier": {
3639
"trailingComma": "all"
3740
},
3841
"dependencies": {
3942
"css": "^2.2.1",
43+
"css-mediaquery": "^0.1.2",
4044
"css-to-react-native": "^2.1.2"
4145
},
4246
"repository": {
4347
"type": "git",
44-
"url":
45-
"git+https://github.com/kristerkari/css-to-react-native-transform.git"
48+
"url": "git+https://github.com/kristerkari/css-to-react-native-transform.git"
4649
},
47-
"keywords": ["React", "ReactNative", "styles", "CSS"],
48-
"files": ["dist", "src", "CHANGELOG.md", "README.md"]
50+
"keywords": [
51+
"React",
52+
"ReactNative",
53+
"styles",
54+
"CSS"
55+
],
56+
"files": [
57+
"dist",
58+
"src",
59+
"CHANGELOG.md",
60+
"README.md"
61+
]
4962
}

src/index.js

+33
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,17 @@
11
import parseCSS from "css/lib/parse";
22
import transformCSS from "css-to-react-native";
3+
import mediaQuery from "css-mediaquery";
34
import { remToPx } from "./transforms/rem";
45
import { camelCase } from "./utils/camelCase";
56
import { allEqual } from "./utils/allEqual";
67
import { values } from "./utils/values";
8+
import { mediaQueryTypes } from "./transforms/media-queries/types";
9+
import {
10+
mediaQueryFeatures,
11+
dimensionFeatures,
12+
} from "./transforms/media-queries/features";
713

14+
const lengthRe = /^(0$|(?:[+-]?(?:\d*\.)?\d+(?:[Ee][+-]?\d+)?)(?=px|rem$))/;
815
const shorthandBorderProps = [
916
"border-radius",
1017
"border-width",
@@ -56,6 +63,32 @@ const transform = (css, options) => {
5663
options != null &&
5764
options.parseMediaQueries === true
5865
) {
66+
const parsed = mediaQuery.parse(rule.media);
67+
68+
parsed.forEach(mq => {
69+
if (mediaQueryTypes.indexOf(mq.type) === -1) {
70+
throw new Error(`Failed to parse media query type "${mq.type}"`);
71+
}
72+
73+
mq.expressions.forEach(e => {
74+
const mf = e.modifier ? `${e.modifier}-${e.feature}` : e.feature;
75+
const val = e.value ? `: ${e.value}` : "";
76+
77+
if (mediaQueryFeatures.indexOf(e.feature) === -1) {
78+
throw new Error(`Failed to parse media query feature "${mf}"`);
79+
}
80+
81+
if (
82+
dimensionFeatures.indexOf(e.feature) > -1 &&
83+
lengthRe.test(e.value) === false
84+
) {
85+
throw new Error(
86+
`Failed to parse media query expression "(${mf}${val})"`,
87+
);
88+
}
89+
});
90+
});
91+
5992
const media = "@media " + rule.media;
6093
for (const r in rule.rules) {
6194
const ruleRule = rule.rules[r];

src/index.spec.js

+195
Original file line numberDiff line numberDiff line change
@@ -2232,4 +2232,199 @@ describe("media queries", () => {
22322232
},
22332233
});
22342234
});
2235+
2236+
it("should support screen type", () => {
2237+
expect(
2238+
transform(
2239+
`
2240+
.foo {
2241+
color: blue;
2242+
}
2243+
@media screen and (min-height: 50px) and (max-height: 150px) {
2244+
.foo {
2245+
color: red;
2246+
}
2247+
}
2248+
@media screen and (min-height: 150px) and (max-height: 200px) {
2249+
.foo {
2250+
color: green;
2251+
}
2252+
}
2253+
`,
2254+
{
2255+
parseMediaQueries: true,
2256+
},
2257+
),
2258+
).toEqual({
2259+
foo: { color: "blue" },
2260+
"@media screen and (min-height: 50px) and (max-height: 150px)": {
2261+
foo: { color: "red" },
2262+
},
2263+
"@media screen and (min-height: 150px) and (max-height: 200px)": {
2264+
foo: { color: "green" },
2265+
},
2266+
});
2267+
});
2268+
2269+
it("should support all type", () => {
2270+
expect(
2271+
transform(
2272+
`
2273+
.foo {
2274+
color: blue;
2275+
}
2276+
@media all and (min-height: 50px) and (max-height: 150px) {
2277+
.foo {
2278+
color: red;
2279+
}
2280+
}
2281+
@media all and (min-height: 150px) and (max-height: 200px) {
2282+
.foo {
2283+
color: green;
2284+
}
2285+
}
2286+
`,
2287+
{
2288+
parseMediaQueries: true,
2289+
},
2290+
),
2291+
).toEqual({
2292+
foo: { color: "blue" },
2293+
"@media all and (min-height: 50px) and (max-height: 150px)": {
2294+
foo: { color: "red" },
2295+
},
2296+
"@media all and (min-height: 150px) and (max-height: 200px)": {
2297+
foo: { color: "green" },
2298+
},
2299+
});
2300+
});
2301+
2302+
it("should throw for invalid types", () => {
2303+
expect(() =>
2304+
transform(
2305+
`
2306+
.foo {
2307+
color: blue;
2308+
}
2309+
2310+
@media screens {
2311+
.foo {
2312+
color: red;
2313+
}
2314+
}
2315+
`,
2316+
{
2317+
parseMediaQueries: true,
2318+
},
2319+
),
2320+
).toThrow('Failed to parse media query type "screens"');
2321+
expect(() =>
2322+
transform(
2323+
`
2324+
.foo {
2325+
color: blue;
2326+
}
2327+
@media sdfgsdfg {
2328+
.foo {
2329+
color: red;
2330+
}
2331+
}
2332+
`,
2333+
{
2334+
parseMediaQueries: true,
2335+
},
2336+
),
2337+
).toThrow('Failed to parse media query type "sdfgsdfg"');
2338+
});
2339+
2340+
it("should throw for invalid features", () => {
2341+
expect(() =>
2342+
transform(
2343+
`
2344+
.foo {
2345+
color: blue;
2346+
}
2347+
@media (min-heigh: 50px) and (max-height: 150px) {
2348+
.foo {
2349+
color: red;
2350+
}
2351+
}
2352+
`,
2353+
{
2354+
parseMediaQueries: true,
2355+
},
2356+
),
2357+
).toThrow('Failed to parse media query feature "min-heigh"');
2358+
expect(() =>
2359+
transform(
2360+
`
2361+
.foo {
2362+
color: blue;
2363+
}
2364+
@media (orientations: landscape) {
2365+
.foo {
2366+
color: red;
2367+
}
2368+
}
2369+
`,
2370+
{
2371+
parseMediaQueries: true,
2372+
},
2373+
),
2374+
).toThrow('Failed to parse media query feature "orientations"');
2375+
});
2376+
2377+
it("should throw for values without units", () => {
2378+
expect(() =>
2379+
transform(
2380+
`
2381+
.foo {
2382+
color: blue;
2383+
}
2384+
@media (min-height: 50) and (max-height: 150px) {
2385+
.foo {
2386+
color: red;
2387+
}
2388+
}
2389+
`,
2390+
{
2391+
parseMediaQueries: true,
2392+
},
2393+
),
2394+
).toThrow('Failed to parse media query expression "(min-height: 50)"');
2395+
expect(() =>
2396+
transform(
2397+
`
2398+
.foo {
2399+
color: blue;
2400+
}
2401+
@media (min-height: 50px) and (max-height: 150) {
2402+
.foo {
2403+
color: red;
2404+
}
2405+
}
2406+
`,
2407+
{
2408+
parseMediaQueries: true,
2409+
},
2410+
),
2411+
).toThrow('Failed to parse media query expression "(max-height: 150)"');
2412+
expect(() =>
2413+
transform(
2414+
`
2415+
.foo {
2416+
color: blue;
2417+
}
2418+
@media (min-width) {
2419+
.foo {
2420+
color: red;
2421+
}
2422+
}
2423+
`,
2424+
{
2425+
parseMediaQueries: true,
2426+
},
2427+
),
2428+
).toThrow('Failed to parse media query expression "(min-width)"');
2429+
});
22352430
});
+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
export const dimensionFeatures = [
2+
"width",
3+
"height",
4+
"device-width",
5+
"device-height",
6+
];
7+
export const mediaQueryFeatures = [
8+
"orientation",
9+
"scan",
10+
"resolution",
11+
"aspect-ratio",
12+
"device-aspect-ratio",
13+
"grid",
14+
"color",
15+
"color-index",
16+
"monochrome",
17+
].concat(dimensionFeatures);

src/transforms/media-queries/types.js

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
export const defaultTypes = [
2+
"all",
3+
"braille",
4+
"embossed",
5+
"handheld",
6+
"print",
7+
"projection",
8+
"screen",
9+
"speech",
10+
"tty",
11+
"tv",
12+
];
13+
export const cssnextMediaQueryTypes = ["pointer", "hover", "block-overflow"];
14+
export const reactNativeMediaQueryTypes = ["android", "ios"];
15+
export const mediaQueryTypes = defaultTypes
16+
.concat(cssnextMediaQueryTypes)
17+
.concat(reactNativeMediaQueryTypes);

0 commit comments

Comments
 (0)