Como obter elemento por innerText

Respostas:

147

Você terá que atravessar à mão.

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`.
August Lilleaas
fonte
1
@AutoSponge Na verdade, innerHTML é padrão. não innerText não funcionar no FF
AnaMaria
Atualizado o exemplo, textContent provavelmente é o que você deseja neste caso. Obrigado, pessoal :)
agosto Lilleaas
1
@AugustLilleaas, o que há com o i < il? O que isso está fazendo?
David Sawyer
1
Descobri que, se você tiver <span> <span> texto de pesquisa </span> </span>, esse método poderá retornar a extensão externa em vez da interna.
Kevin Wheeler
5
Não, esta pergunta é sobre JavaScript e HTML, não Java
agosto Lilleaas
159

Você pode usar o xpath para fazer isso

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

Você também pode procurar um elemento que contenha algum texto usando este xpath:

var xpath = "//a[contains(text(),'Searching')]";
carlin.scott
fonte
7
Essa deve ser a resposta principal. XPath pode fazer muito mais, como selecionar nó valor do atributo, selecione conjuntos nó ... introdução simples: w3schools.com/xml/xpath_syntax.asp
Timathon
1
A pergunta é, qual é a penalidade de desempenho para este truque
vsync
2
@ vsync Acho que isso será mais rápido do que qualquer outra resposta, pois o xpath é executado por um algoritmo fornecido pelo navegador, em vez de executar em javascript, como todas as outras respostas aqui. É uma pergunta interessante, no entanto.
Carlin.scott #
1
Parece que Document.evaluate() não é suposto no IE navegador
vsync
1
Não sei por que, mas de alguma forma var xpath = "//a[text()='SearchingText']"; Isso não está funcionando, mas var xpath = "//a[contains(text(),'Searching')]"; funciona. Esteja ciente do caractere excedido, como \ '\'.
Joey Cho #
38

Usando a sintaxe mais moderna disponível no momento, isso pode ser feito de maneira muito limpa assim:

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

Ou com um filtro separado:

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

Naturalmente, os navegadores herdados não lidam com isso, mas você pode usar um transpiler se for necessário suporte legado.

user1106925
fonte
Abordagem de filtro <3
John Vandivier 9/04/19
36

Você pode usar o jQuery : contains () Selector

var element = $( "a:contains('SearchingText')" );
Mouneer
fonte
Eu recebo: Error: <![EX[["Tried to get element with id of \"%s\" but it is not present on the page","a:contains('SearchingText')"]]]> TAAL[1]embora eu tenha elementos com "SearchingText" neles.
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>

Obviamente, uma maneira um pouco mais simples ainda é:

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>

Referências:

David diz para restabelecer Monica
fonte
14

Abordagem funcional. Retorna a matriz de todos os elementos correspondentes e apara os espaços ao redor durante a verificação.

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

Uso

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

se você estiver procurando por diferentes tags, como extensão ou botão

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

A tag de valor padrão = 'a' precisará do Babel para navegadores antigos

Pawel
fonte
Isso está incorreto porque também inclui resultados para todos os nós filhos. Ou seja, se o nó filho de airá conter str- elserá incluído no getElementsByTextresultado; o que está errado.
perfil
@ avalanche1 depende se isso é indesejável. Poderá ser necessário selecionar pelo texto, mesmo se ele está envolvido em outra tag ou seja, <span> </ span>
Pawel
5

Basta passar sua substring para a seguinte linha:

HTML externo

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

HTML interno

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

Você pode usá-los para pesquisar em todo o documento e recuperar as tags que contêm seu termo de pesquisa:

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

Uso :

Quantas vezes o usuário "T3rm1" é mencionado nesta página?

get_elements_by_inner("T3rm1").length

1

Quantas vezes o jQuery é mencionado?

get_elements_by_inner("jQuery").length

3

Obtenha todos os elementos que contêm a palavra "Cybernetic":

get_elements_by_inner("Cybernetic")

insira a descrição da imagem aqui

Cibernético
fonte
Isso retorna verdadeiro ou falso, mas não o elemento.
T3rm1
Você pode usar a condição de verdade para iterar pelos elementos recuperados e obter o que precisar desses elementos. Veja a resposta atualizada.
Cibernética
4

Eu achei o uso da sintaxe mais recente um pouco menor em comparação com as outras respostas. Então aqui está a minha proposta:

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

Amin NAIRI
fonte
2

Para obter o método de filtro do usuário1106925 trabalhando em <= IE11, se necessário

Você pode substituir o operador de propagação por:

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

e inclui a chamada com a.textContent.match("your search term")

que funciona muito bem:

[].slice.call(document.querySelectorAll("a"))
   .filter(a => a.textContent.match("your search term"))
   .forEach(a => console.log(a.textContent))
Alkie
fonte
Eu gosto deste método Você também pode, em Array.fromvez de [].slice.call. Por exemplo: Array.from(document.querySelectorAll('a'))
Richard
1

Embora seja possível entender o texto interno, acho que você está seguindo o caminho errado. Essa cadeia interna é gerada dinamicamente? Nesse caso, você pode atribuir à tag uma classe ou - melhor ainda - ID quando o texto for inserido lá. Se é estático, é ainda mais fácil.

Zack Marrapese
fonte
1

Você pode usar a TreeWalkerpara passar pelos nós do DOM, localizar todos os nós de texto que contêm o texto e retornar seus pais:

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>

Ori Drori
fonte
0

Acho que você precisará ser um pouco mais específico para ajudá-lo.

  1. Como você está encontrando isso? Javascript? PHP? Perl?
  2. Você pode aplicar um atributo de ID à tag?

Se o texto for único (ou realmente, se não for, mas você precisaria executar uma matriz), você poderia executar uma expressão regular para encontrá-lo. Usar o preg_match () do PHP funcionaria para isso.

Se você estiver usando Javascript e puder inserir um atributo de ID, poderá usar getElementById ('id'). Você pode acessar os atributos do elemento retornado através do DOM: https://developer.mozilla.org/en/DOM/element.1 .

Jeff Meyers
fonte
0

Eu só precisava de uma maneira de obter o elemento que contém um texto específico e foi isso que eu criei.

Use document.getElementsByInnerText()para obter vários elementos (vários elementos podem ter o mesmo texto exato) e use document.getElementByInnerText()para obter apenas um elemento (primeira correspondência).

Além disso, você pode localizar a pesquisa usando um elemento (por exemplo someElement.getElementByInnerText()) em vez de document.

Pode ser necessário ajustá-lo para torná-lo entre navegadores ou satisfazer suas necessidades.

Eu acho que o código é auto-explicativo, então vou deixar como está.

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>

akinuri
fonte
0

Isso faz o trabalho.
Retorna uma matriz de nós que contêm 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())
    );
  }
avalanche1
fonte