Skip to content

Commit 7d9cfeb

Browse files
committed
fix(ssr): properly handle invalid and numeric style properties
Ignores values that are not string or numbers, and append px as default unit to appropriate properties. There will still be certain cases where the user simply provides an invalid string value to a property which will be too costly to detect (it's possible by using the `cssstyle` package, but very heavy). But in such cases the user would already notice the style is not working on the client, so it's not really worth it for the perf implications. fix #9231
1 parent fe2b27f commit 7d9cfeb

File tree

4 files changed

+108
-4
lines changed

4 files changed

+108
-4
lines changed

src/platforms/web/server/modules/style.js

+22-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/* @flow */
22

3-
import { escape } from '../util'
3+
import { escape, noUnitNumericStyleProps } from '../util'
44
import { hyphenate } from 'shared/util'
55
import { getStyle } from 'web/util/style'
66

@@ -11,15 +11,34 @@ export function genStyle (style: Object): string {
1111
const hyphenatedKey = hyphenate(key)
1212
if (Array.isArray(value)) {
1313
for (let i = 0, len = value.length; i < len; i++) {
14-
styleText += `${hyphenatedKey}:${value[i]};`
14+
styleText += normalizeValue(hyphenatedKey, value[i])
1515
}
1616
} else {
17-
styleText += `${hyphenatedKey}:${value};`
17+
styleText += normalizeValue(hyphenatedKey, value)
1818
}
1919
}
2020
return styleText
2121
}
2222

23+
function normalizeValue(key: string, value: any): string {
24+
if (typeof value === 'string') {
25+
return `${key}:${value};`
26+
} else if (typeof value === 'number') {
27+
// Handle numeric values.
28+
// Turns out all evergreen browsers plus IE11 already support setting plain
29+
// numbers on the style object and will automatically convert it to px if
30+
// applicable, so we should support that on the server too.
31+
if (noUnitNumericStyleProps[key]) {
32+
return `${key}:${value};`
33+
} else {
34+
return `${key}:${value}px;`
35+
}
36+
} else {
37+
// invalid values
38+
return ``
39+
}
40+
}
41+
2342
export default function renderStyle (vnode: VNodeWithData): ?string {
2443
const styleText = genStyle(getStyle(vnode, false))
2544
if (styleText !== '') {

src/platforms/web/server/util.js

+45
Original file line numberDiff line numberDiff line change
@@ -54,3 +54,48 @@ export function escape (s: string) {
5454
function escapeChar (a) {
5555
return ESC[a] || a
5656
}
57+
58+
export const noUnitNumericStyleProps = {
59+
"animation-iteration-count": true,
60+
"border-image-outset": true,
61+
"border-image-slice": true,
62+
"border-image-width": true,
63+
"box-flex": true,
64+
"box-flex-group": true,
65+
"box-ordinal-group": true,
66+
"column-count": true,
67+
"columns": true,
68+
"flex": true,
69+
"flex-grow": true,
70+
"flex-positive": true,
71+
"flex-shrink": true,
72+
"flex-negative": true,
73+
"flex-order": true,
74+
"grid-row": true,
75+
"grid-row-end": true,
76+
"grid-row-span": true,
77+
"grid-row-start": true,
78+
"grid-column": true,
79+
"grid-column-end": true,
80+
"grid-column-span": true,
81+
"grid-column-start": true,
82+
"font-weight": true,
83+
"line-clamp": true,
84+
"line-height": true,
85+
"opacity": true,
86+
"order": true,
87+
"orphans": true,
88+
"tab-size": true,
89+
"widows": true,
90+
"z-index": true,
91+
"zoom": true,
92+
// SVG
93+
"fill-opacity": true,
94+
"flood-opacity": true,
95+
"stop-opacity": true,
96+
"stroke-dasharray": true,
97+
"stroke-dashoffset": true,
98+
"stroke-miterlimit": true,
99+
"stroke-opacity": true,
100+
"stroke-width": true
101+
}

src/platforms/web/util/style.js

-1
Original file line numberDiff line numberDiff line change
@@ -69,4 +69,3 @@ export function getStyle (vnode: VNodeWithData, checkChild: boolean): Object {
6969
}
7070
return res
7171
}
72-

test/ssr/ssr-string.spec.js

+41
Original file line numberDiff line numberDiff line change
@@ -1499,6 +1499,47 @@ describe('SSR: renderToString', () => {
14991499
done()
15001500
})
15011501
})
1502+
1503+
it('invalid style value', done => {
1504+
renderVmWithOptions({
1505+
template: '<div :style="style"><p :style="style2"/></div>',
1506+
data: {
1507+
// all invalid, should not even have "style" attribute
1508+
style: {
1509+
opacity: {},
1510+
color: null
1511+
},
1512+
// mix of valid and invalid
1513+
style2: {
1514+
opacity: 0,
1515+
color: null
1516+
}
1517+
}
1518+
}, result => {
1519+
expect(result).toContain(
1520+
'<div data-server-rendered="true"><p style="opacity:0;"></p></div>'
1521+
)
1522+
done()
1523+
})
1524+
})
1525+
1526+
it('numeric style value', done => {
1527+
renderVmWithOptions({
1528+
template: '<div :style="style"></div>',
1529+
data: {
1530+
style: {
1531+
opacity: 0, // opacity should display as-is
1532+
top: 0, // top and margin should be converted to '0px'
1533+
marginTop: 10
1534+
}
1535+
}
1536+
}, result => {
1537+
expect(result).toContain(
1538+
'<div data-server-rendered="true" style="opacity:0;top:0px;margin-top:10px;"></div>'
1539+
)
1540+
done()
1541+
})
1542+
})
15021543
})
15031544

15041545
function renderVmWithOptions (options, cb) {

0 commit comments

Comments
 (0)