วิธีรับองค์ประกอบโดย innerText


คำตอบ:


148

คุณจะต้องเคลื่อนที่ด้วยมือ

var aTags = document.getElementsByTagName("a");
var searchText = "SearchingText";
var found;

for (var i = 0; i < aTags.length; i++) {
  if (aTags[i].textContent == searchText) {
    found = aTags[i];
    break;
  }
}

// Use `found`.

1
@AutoSponge จริง ๆ แล้ว HTML เป็นมาตรฐาน innerText ไม่ทำงานใน FF
AnaMaria

อัปเดตตัวอย่าง textContent น่าจะเป็นสิ่งที่คุณต้องการในกรณีนี้ ขอบคุณ folks :)
Lilleaas

1
@AugustLilleaas เกิดอะไรขึ้นกับi < il? กำลังทำอะไรอยู่
David Sawyer

1
ฉันพบว่าหากคุณมี <span> <span> ข้อความค้นหา </span> </span> วิธีนี้อาจส่งคืนช่วงนอกโดยรอบแทนที่จะเป็นส่วนใน
Kevin Wheeler

5
ไม่คำถามนี้เกี่ยวกับ JavaScript และ HTML ไม่ใช่ Java
August Lilleaas

160

คุณสามารถใช้ xpath เพื่อทำสิ่งนี้ให้สำเร็จ

var xpath = "//a[text()='SearchingText']";
var matchingElement = document.evaluate(xpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;

คุณยังสามารถค้นหาองค์ประกอบที่มีข้อความโดยใช้ xpath นี้:

var xpath = "//a[contains(text(),'Searching')]";

7
นี่ควรเป็นคำตอบที่ดีที่สุด XPath สามารถทำได้มากกว่านี้เช่นเลือกโหนดด้วยค่าแอททริบิวเลือกชุดโหนด ... คำนำง่าย ๆ : w3schools.com/xml/xpath_syntax.asp
Timathon

1
คำถามคืออะไรคือประสิทธิภาพการลงโทษสำหรับเล่ห์เหลี่ยมนี้
vsync

2
@vsync ฉันคิดว่านี่จะเร็วกว่าคำตอบอื่น ๆ เนื่องจาก xpath นั้นถูกเรียกใช้โดยเบราว์เซอร์ที่มีอัลกอริทึมแทนที่จะใช้ใน javascript เหมือนกับคำตอบอื่น ๆ ทั้งหมดที่นี่ มันเป็นคำถามที่น่าสนใจ
carlin.scott

1
ดูเหมือนว่าDocument.evaluate() ไม่ควรในเบราว์เซอร์IE
vsync

1
ฉันไม่รู้ว่าทำไม แต่อย่างใดvar xpath = "//a[text()='SearchingText']"; นี่ไม่ทำงาน แต่ใช้ var xpath = "//a[contains(text(),'Searching')]"; งานได้ ระวังตัวละครที่ถูกทำให้ตกใจเช่น \ '\'
Joey Cho

38

การใช้ไวยากรณ์ที่ทันสมัยที่สุดในขณะนี้สามารถทำได้อย่างหมดจดเช่นนี้:

for (const a of document.querySelectorAll("a")) {
  if (a.textContent.includes("your search term")) {
    console.log(a.textContent)
  }
}

หรือด้วยตัวกรองแยก:

[...document.querySelectorAll("a")]
   .filter(a => a.textContent.includes("your search term"))
   .forEach(a => console.log(a.textContent))

โดยปกติเบราว์เซอร์รุ่นเก่าจะไม่จัดการสิ่งนี้ แต่คุณสามารถใช้ transpiler หากต้องการการสนับสนุนแบบดั้งเดิม


วิธีการกรอง <3
John Vandivier

36

คุณสามารถใช้ jQuery : contain () Selector

var element = $( "a:contains('SearchingText')" );

ฉันได้รับ: Error: <![EX[["Tried to get element with id of \"%s\" but it is not present on the page","a:contains('SearchingText')"]]]> TAAL[1]แม้ว่าฉันจะมีองค์ประกอบที่มี "LookupText" อยู่ในนั้น
Rishabh Agrahari

15

function findByTextContent(needle, haystack, precise) {
  // needle: String, the string to be found within the elements.
  // haystack: String, a selector to be passed to document.querySelectorAll(),
  //           NodeList, Array - to be iterated over within the function:
  // precise: Boolean, true - searches for that precise string, surrounded by
  //                          word-breaks,
  //                   false - searches for the string occurring anywhere
  var elems;

  // no haystack we quit here, to avoid having to search
  // the entire document:
  if (!haystack) {
    return false;
  }
  // if haystack is a string, we pass it to document.querySelectorAll(),
  // and turn the results into an Array:
  else if ('string' == typeof haystack) {
    elems = [].slice.call(document.querySelectorAll(haystack), 0);
  }
  // if haystack has a length property, we convert it to an Array
  // (if it's already an array, this is pointless, but not harmful):
  else if (haystack.length) {
    elems = [].slice.call(haystack, 0);
  }

  // work out whether we're looking at innerText (IE), or textContent 
  // (in most other browsers)
  var textProp = 'textContent' in document ? 'textContent' : 'innerText',
    // creating a regex depending on whether we want a precise match, or not:
    reg = precise === true ? new RegExp('\\b' + needle + '\\b') : new RegExp(needle),
    // iterating over the elems array:
    found = elems.filter(function(el) {
      // returning the elements in which the text is, or includes,
      // the needle to be found:
      return reg.test(el[textProp]);
    });
  return found.length ? found : false;;
}


findByTextContent('link', document.querySelectorAll('li'), false).forEach(function(elem) {
  elem.style.fontSize = '2em';
});

findByTextContent('link3', 'a').forEach(function(elem) {
  elem.style.color = '#f90';
});
<ul>
  <li><a href="#">link1</a>
  </li>
  <li><a href="#">link2</a>
  </li>
  <li><a href="#">link3</a>
  </li>
  <li><a href="#">link4</a>
  </li>
  <li><a href="#">link5</a>
  </li>
</ul>

แน่นอนว่าวิธีที่เรียบง่ายกว่าคือ:

var textProp = 'textContent' in document ? 'textContent' : 'innerText';

// directly converting the found 'a' elements into an Array,
// then iterating over that array with Array.prototype.forEach():
[].slice.call(document.querySelectorAll('a'), 0).forEach(function(aEl) {
  // if the text of the aEl Node contains the text 'link1':
  if (aEl[textProp].indexOf('link1') > -1) {
    // we update its style:
    aEl.style.fontSize = '2em';
    aEl.style.color = '#f90';
  }
});
<ul>
  <li><a href="#">link1</a>
  </li>
  <li><a href="#">link2</a>
  </li>
  <li><a href="#">link3</a>
  </li>
  <li><a href="#">link4</a>
  </li>
  <li><a href="#">link5</a>
  </li>
</ul>

อ้างอิง:


14

วิธีการใช้งาน ส่งคืนอาร์เรย์ขององค์ประกอบที่ตรงกันทั้งหมดและตัดขอบช่องว่างรอบ ๆ ขณะตรวจสอบ

function getElementsByText(str, tag = 'a') {
  return Array.prototype.slice.call(document.getElementsByTagName(tag)).filter(el => el.textContent.trim() === str.trim());
}

การใช้

getElementsByText('Text here'); // second parameter is optional tag (default "a")

หากคุณกำลังดูแท็กที่แตกต่างกันเช่น span หรือปุ่ม

getElementsByText('Text here', 'span');
getElementsByText('Text here', 'button');

แท็กค่าเริ่มต้น = 'a' จะต้องใช้เบเบลสำหรับเบราว์เซอร์รุ่นเก่า


สิ่งนี้ไม่ถูกต้องเนื่องจากมันรวมผลลัพธ์สำหรับโหนดลูกทั้งหมด คือถ้าโหนดย่อยaจะมีstr- elจะถูกรวมเข้าไปในgetElementsByTextผลลัพธ์ ซึ่งผิด
avalanche1

@ avalanche1 มันขึ้นอยู่กับว่ามันไม่เป็นที่พึงปรารถนา อาจจำเป็นต้องเลือกตามข้อความแม้ว่าจะอยู่ในแท็กอื่นเช่น <span> </span>
Pawel

5

เพียงส่งสตริงย่อยของคุณไปยังบรรทัดต่อไปนี้:

HTML ภายนอก

document.documentElement.outerHTML.includes('substring')

HTML ภายใน

document.documentElement.innerHTML.includes('substring')

คุณสามารถใช้สิ่งเหล่านี้เพื่อค้นหาเอกสารทั้งหมดและดึงแท็กที่มีข้อความค้นหาของคุณ:

function get_elements_by_inner(word) {
    res = []
    elems = [...document.getElementsByTagName('a')];
    elems.forEach((elem) => { 
        if(elem.outerHTML.includes(word)) {
            res.push(elem)
        }
    })
    return(res)
}

การใช้งาน :

มีการพูดถึงผู้ใช้ "T3rm1" กี่ครั้งในหน้านี้

get_elements_by_inner("T3rm1").length

1

มีการพูดถึง jQuery กี่ครั้ง?

get_elements_by_inner("jQuery").length

3

รับองค์ประกอบทั้งหมดที่มีคำว่า "Cybernetic":

get_elements_by_inner("Cybernetic")

ป้อนคำอธิบายรูปภาพที่นี่


สิ่งนี้จะคืนค่าจริงหรือเท็จ แต่ไม่ใช่องค์ประกอบ
T3rm1

คุณสามารถใช้เงื่อนไขความจริงเพื่อย้ำผ่านองค์ประกอบที่ดึงมาและหยิบสิ่งที่คุณต้องการจากองค์ประกอบเหล่านั้น ดูคำตอบที่อัปเดต
Cybernetic

4

ฉันพบว่าการใช้ไวยากรณ์ใหม่กว่าสั้นกว่าเล็กน้อยเมื่อเปรียบเทียบกับคำตอบอื่น ๆ ดังนั้นนี่คือข้อเสนอของฉัน:

const callback = element => element.innerHTML == 'My research'

const elements = Array.from(document.getElementsByTagName('a'))
// [a, a, a, ...]

const result = elements.filter(callback)

console.log(result)
// [a]

JSfiddle.net


2

เพื่อให้ได้วิธีการกรองจากuser1106925ทำงานใน <= IE11 ถ้าจำเป็น

คุณสามารถแทนที่โอเปอเรเตอร์การแพร่กระจายด้วย:

[].slice.call(document.querySelectorAll("a"))

และรวมถึงการโทรด้วย a.textContent.match("your search term")

ซึ่งใช้งานได้อย่างเรียบร้อย:

[].slice.call(document.querySelectorAll("a"))
   .filter(a => a.textContent.match("your search term"))
   .forEach(a => console.log(a.textContent))

ฉันชอบวิธีนี้ คุณยังสามารถแทนArray.from [].slice.callตัวอย่างเช่น: Array.from(document.querySelectorAll('a'))
Richard

1

ในขณะที่เป็นไปได้ที่จะได้รับจากข้อความภายในฉันคิดว่าคุณกำลังมุ่งหน้าไปทางที่ผิด สตริงภายในนั้นถูกสร้างแบบไดนามิกหรือไม่? ถ้าเป็นเช่นนั้นคุณสามารถให้แท็กในชั้นเรียนหรือ - ดีขึ้นยัง - ID เมื่อข้อความไปที่นั่น หากเป็นแบบคงที่ก็จะยิ่งง่ายขึ้น


1

คุณสามารถใช้ a TreeWalkerเพื่อข้ามโหนด DOM และค้นหาโหนดข้อความทั้งหมดที่มีข้อความและส่งคืนพาเรนต์:

const findNodeByContent = (text, root = document.body) => {
  const treeWalker = document.createTreeWalker(root, NodeFilter.SHOW_TEXT);

  const nodeList = [];

  while (treeWalker.nextNode()) {
    const node = treeWalker.currentNode;

    if (node.nodeType === Node.TEXT_NODE && node.textContent.includes(text)) {
      nodeList.push(node.parentNode);
    }
  };

  return nodeList;
}

const result = findNodeByContent('SearchingText');

console.log(result);
<a ...>SearchingText</a>


1

มันทำงานได้ดี ส่งกลับอาร์เรย์ของโหนดที่มี
text

function get_nodes_containing_text(selector, text) {
    const elements = [...document.querySelectorAll(selector)];

    return elements.filter(
      (element) =>
        element.childNodes[0]
        && element.childNodes[0].nodeValue
        && RegExp(text, "u").test(element.childNodes[0].nodeValue.trim())
    );
  }

0

ฉันคิดว่าคุณจะต้องเจาะจงมากขึ้นสำหรับเราที่จะช่วยคุณ

  1. คุณค้นพบสิ่งนี้ได้อย่างไร Javascript? PHP? Perl?
  2. คุณสามารถใช้แอตทริบิวต์ ID กับแท็กได้หรือไม่

หากข้อความนั้นไม่ซ้ำกัน (หรือจริง ๆ ถ้าไม่ใช่ แต่คุณต้องเรียกใช้อาร์เรย์) คุณสามารถเรียกใช้นิพจน์ทั่วไปเพื่อค้นหาได้ การใช้ preg_match ของ PHP () จะใช้งานได้

หากคุณใช้ Javascript และสามารถแทรกแอตทริบิวต์ ID คุณสามารถใช้ getElementById ('id') จากนั้นคุณสามารถเข้าถึงคุณลักษณะองค์ประกอบกลับผ่าน DOM นี้: https://developer.mozilla.org/en/DOM/element.1


0

ฉันแค่ต้องการวิธีที่จะได้รับองค์ประกอบที่มีข้อความเฉพาะและนี่คือสิ่งที่ฉันมาด้วย

ใช้document.getElementsByInnerText()เพื่อรับองค์ประกอบหลายรายการ (หลายองค์ประกอบอาจมีข้อความที่แน่นอนเหมือนกัน) และใช้document.getElementByInnerText()เพื่อรับองค์ประกอบเดียว (คู่แรก)

นอกจากนี้คุณสามารถ จำกัด การค้นหาโดยใช้องค์ประกอบ (เช่นsomeElement.getElementByInnerText()) documentแทน

คุณอาจต้องปรับแต่งมันเพื่อให้มันข้ามเบราว์เซอร์หรือตอบสนองความต้องการของคุณ

ฉันคิดว่ารหัสนั้นอธิบายตัวเองได้ดังนั้นฉันจะทิ้งมันไว้เหมือนเดิม

HTMLElement.prototype.getElementsByInnerText = function (text, escape) {
    var nodes  = this.querySelectorAll("*");
    var matches = [];
    for (var i = 0; i < nodes.length; i++) {
        if (nodes[i].innerText == text) {
            matches.push(nodes[i]);
        }
    }
    if (escape) {
        return matches;
    }
    var result = [];
    for (var i = 0; i < matches.length; i++) {
        var filter = matches[i].getElementsByInnerText(text, true);
        if (filter.length == 0) {
            result.push(matches[i]);
        }
    }
    return result;
};
document.getElementsByInnerText = HTMLElement.prototype.getElementsByInnerText;

HTMLElement.prototype.getElementByInnerText = function (text) {
    var result = this.getElementsByInnerText(text);
    if (result.length == 0) return null;
    return result[0];
}
document.getElementByInnerText = HTMLElement.prototype.getElementByInnerText;

console.log(document.getElementsByInnerText("Text1"));
console.log(document.getElementsByInnerText("Text2"));
console.log(document.getElementsByInnerText("Text4"));
console.log(document.getElementsByInnerText("Text6"));

console.log(document.getElementByInnerText("Text1"));
console.log(document.getElementByInnerText("Text2"));
console.log(document.getElementByInnerText("Text4"));
console.log(document.getElementByInnerText("Text6"));
<table>
    <tr>
        <td>Text1</td>
    </tr>
    <tr>
        <td>Text2</td>
    </tr>
    <tr>
        <td>
            <a href="#">Text2</a>
        </td>
    </tr>
    <tr>
        <td>
            <a href="#"><span>Text3</span></a>
        </td>
    </tr>
    <tr>
        <td>
            <a href="#">Special <span>Text4</span></a>
        </td>
    </tr>
    <tr>
        <td>
            Text5
            <a href="#">Text6</a>
            Text7
        </td>
    </tr>
</table>

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.