I'm using nextUntil
method to get all stuff between two elements. But this method does not include text nodes
to output. It gives an array like [<br>, <br>, <br>]
. How can I get all stuff including text nodes?
This is the HTML code:
$('.content a:contains("spoiler").b:even').each(function() {
$(this).nextUntil('.content a:contains("spoiler").b')
.wrapAll('<div style="border:solid 1px black;"></div>');
});
<script src=".3.1/jquery.min.js"></script>
<div class="content">
--- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br>
<br> dangerous text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br> safe text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br>
<br> dangerous text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
</div>
I'm using nextUntil
method to get all stuff between two elements. But this method does not include text nodes
to output. It gives an array like [<br>, <br>, <br>]
. How can I get all stuff including text nodes?
This is the HTML code:
$('.content a:contains("spoiler").b:even').each(function() {
$(this).nextUntil('.content a:contains("spoiler").b')
.wrapAll('<div style="border:solid 1px black;"></div>');
});
<script src="https://cdnjs.cloudflare./ajax/libs/jquery/3.3.1/jquery.min.js"></script>
<div class="content">
--- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br>
<br> dangerous text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br> safe text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
<br>
<br> dangerous text here
<br> --- <a class="b" href="/?q=spoiler">spoiler</a> ---
</div>
JSFiddle: http://jsfiddle/Lwk97rvq/1/
Share Improve this question edited Apr 19, 2019 at 13:19 double-beep 5,53719 gold badges40 silver badges49 bronze badges asked Sep 16, 2014 at 16:11 Dorukcan KişinDorukcan Kişin 1,1312 gold badges14 silver badges29 bronze badges 3-
6
Only
.contents()
returns text nodes. – iCollect.it Ltd Commented Sep 16, 2014 at 16:12 -
how can i modify code for
contents()
? – Dorukcan Kişin Commented Sep 16, 2014 at 16:26 - 1 Potential answer below (If I understand the aim correctly) – iCollect.it Ltd Commented Sep 16, 2014 at 16:28
2 Answers
Reset to default 8You can create your own jquery plugin which does the same as nextUntil
but includes text nodes:
$.fn.nextUntilWithTextNodes = function (until) {
var matched = $.map(this, function (elem, i, until) {
var matched = [];
while ((elem = elem.nextSibling) && elem.nodeType !== 9) {
if (elem.nodeType === 1 || elem.nodeType === 3) {
if (until && jQuery(elem).is(until)) {
break;
}
matched.push(elem);
}
}
return matched;
}, until);
return this.pushStack(matched);
};
So you can call this nextUntilWithTextNodes
instead of nextUntil
and you are good to go.
Only the jQuery .contents()
method returns all nodes (including text nodes, normally ignored).
So maybe something like this?:
http://jsfiddle/ykv3gf5L/2/
$('.content').each(function () {
var open = false;
var result = $();
$(this).contents().each(function () {
var $this = $(this);
if ($this.text() == "spoiler") {
if (open) {
result.wrapAll('<div style="border:solid 1px black;"></div>');
open = false;
} else {
result = $();
open = true;
}
} else {
result = result.add($this)
}
});
if (open) {
result.wrapAll('<div style="border:solid 1px black;"></div>');
}
});
It just iterate all nodes and based on a flag starts a new collection, or wraps the nodes found.
The final if (open)
allows for an unclosed spolier block within a content
classed div.
Notes:
$()
is an empty jQuery collection (like an empty array but for jQuery objects)- I suggest you use a style for your spoilers and use a class e.g.
result.wrapAll('<div class="spoiler"></div>');
e.g. http://jsfiddle/ykv3gf5L/3/