Skip to content

Commit 4d68079

Browse files
defccyyx990803
authored andcommitted
fix(parser): the first newline following pre and textarea tag should be ignored (#6022)
fix #6022
1 parent a1d1145 commit 4d68079

File tree

2 files changed

+30
-6
lines changed

2 files changed

+30
-6
lines changed

src/compiler/parser/html-parser.js

+14-4
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,10 @@ const decodingMap = {
5959
const encodedAttr = /&(?:lt|gt|quot|amp);/g
6060
const encodedAttrWithNewLines = /&(?:lt|gt|quot|amp|#10);/g
6161

62+
// #5992
63+
const isIgnoreNewlineTag = makeMap('pre,textarea', true)
64+
const shouldIgnoreFirstNewline = (tag, html) => tag && isIgnoreNewlineTag(tag) && html[0] === '\n'
65+
6266
function decodeAttr (value, shouldDecodeNewlines) {
6367
const re = shouldDecodeNewlines ? encodedAttrWithNewLines : encodedAttr
6468
return value.replace(re, match => decodingMap[match])
@@ -75,6 +79,9 @@ export function parseHTML (html, options) {
7579
last = html
7680
// Make sure we're not in a plaintext content element like script/style
7781
if (!lastTag || !isPlainTextElement(lastTag)) {
82+
if (shouldIgnoreFirstNewline(lastTag, html)) {
83+
advance(1)
84+
}
7885
let textEnd = html.indexOf('<')
7986
if (textEnd === 0) {
8087
// Comment:
@@ -152,16 +159,19 @@ export function parseHTML (html, options) {
152159
options.chars(text)
153160
}
154161
} else {
155-
var stackedTag = lastTag.toLowerCase()
156-
var reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)(</' + stackedTag + '[^>]*>)', 'i'))
157-
var endTagLength = 0
158-
var rest = html.replace(reStackedTag, function (all, text, endTag) {
162+
let endTagLength = 0
163+
const stackedTag = lastTag.toLowerCase()
164+
const reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)(</' + stackedTag + '[^>]*>)', 'i'))
165+
const rest = html.replace(reStackedTag, function (all, text, endTag) {
159166
endTagLength = endTag.length
160167
if (!isPlainTextElement(stackedTag) && stackedTag !== 'noscript') {
161168
text = text
162169
.replace(/<!--([\s\S]*?)-->/g, '$1')
163170
.replace(/<!\[CDATA\[([\s\S]*?)]]>/g, '$1')
164171
}
172+
if (shouldIgnoreFirstNewline(stackedTag, text)) {
173+
text = text.slice(1)
174+
}
165175
if (options.chars) {
166176
options.chars(text)
167177
}

test/unit/modules/compiler/parser.spec.js

+16-2
Original file line numberDiff line numberDiff line change
@@ -495,6 +495,21 @@ describe('parser', () => {
495495
expect(span.children[0].text).toBe(' ')
496496
})
497497

498+
// #5992
499+
it('ignore the first newline in <pre> tag', function () {
500+
const options = extend({}, baseOptions)
501+
const ast = parse('<div><pre>\nabc</pre>\ndef<pre>\n\nabc</pre></div>', options)
502+
const pre = ast.children[0]
503+
expect(pre.children[0].type).toBe(3)
504+
expect(pre.children[0].text).toBe('abc')
505+
const text = ast.children[1]
506+
expect(text.type).toBe(3)
507+
expect(text.text).toBe('\ndef')
508+
const pre2 = ast.children[2]
509+
expect(pre2.children[0].type).toBe(3)
510+
expect(pre2.children[0].text).toBe('\nabc')
511+
})
512+
498513
it('forgivingly handle < in plain text', () => {
499514
const options = extend({}, baseOptions)
500515
const ast = parse('<p>1 < 2 < 3</p>', options)
@@ -530,8 +545,7 @@ describe('parser', () => {
530545
expect(whitespace.children.length).toBe(1)
531546
expect(whitespace.children[0].type).toBe(3)
532547
// textarea is whitespace sensitive
533-
expect(whitespace.children[0].text).toBe(`
534-
<p>Test 1</p>
548+
expect(whitespace.children[0].text).toBe(` <p>Test 1</p>
535549
test2
536550
`)
537551

0 commit comments

Comments
 (0)