Фильтрация Puppeter Sharp's ElementHandle [] с помощью async linq - PullRequest
0 голосов
/ 04 ноября 2018

Я переписывал свое приложение Selenium в Puppeter Sharp, чтобы узнать больше об этой платформе. Мое приложение работает на сайте, к которому у меня нет доступа, чтобы изменить его источник, я только собираю некоторую информацию.

Исходя из этого, я написал функцию с использованием Selenium, которая может фильтровать нужный элемент внутри IEnumerable<IWebElement>, таким образом:

var element = Driver.FindElements(By.Id("User"))
    .First(x => 
           x.Text.ToLower().Trim() == "AAA" &&
           x.FindElement(By.Id("contact")).Text.ToLower().Trim() == "BBB")
    .FindElement(By.XPath("../input"));

Но я не смог написать код, который делает то же самое, используя Puppeteer Sharp. Самый точный подход, который я написал, был:

var users = await Page.QuerySelectorAllAsync("#User");
var baseElement = users.First(async x =>
    (await x.JsonValueAsync<string>()).ToLower().Trim() == "AAA" &&
    (await (await x.QuerySelectorAsync("#contact")).JsonValueAsync<string>()).ToLower().Trim() == "BBB");
var element = await baseElement.XPathAsync("../input");

Код выше все еще возвращает мне ошибку:

Невозможно преобразовать асинхронное лямбда-выражение в тип делегата 'Func'.

Да, это имеет смысл, но я не смог найти другой подход с linq, чтобы он был хотя бы разборчивым и маленьким.

Редактировать

Пример HTML ниже: мне нужно вернуть радио вход, основанный на названии и типе атаки. Я понимаю, что могу написать отличный, уродливый и неразборчивый xpath, чтобы запросить его у меня, но я попытался сделать его более читабельным и гибким, чтобы его было проще обслуживать.

<div id="player">
    <div class="attklist">
        <h2>Choose an Attack</h2>
        <ul>
            <li class="batsel" style="cursor: pointer;">
                <input type="radio" value="1" name="selected" checked="checked" id="atkopt1">
                <label for="atkopt1">Strength
                    <br><span class="tbtn tbtn-normal">normal</span>
                </label>
            </li>
            <li style="cursor: pointer;">
                <input type="radio" value="2" name="selected" id="atkopt2">
                <label for="atkopt2">Strength
                    <br><span class="tbtn tbtn-fighting">fighting</span>
                </label>
            </li>
            <li style="cursor: pointer;">
                <input type="radio" value="3" name="selected" id="atkopt3">
                <label for="atkopt3">Spectre Slash
                    <br><span class="tbtn tbtn-ghost">ghost</span>
                </label>
            </li>
            <li style="cursor: pointer;">
                <input type="radio" value="4" name="selected" id="atkopt4">
                <label for="atkopt4">Fire Puch
                    <br><span class="tbtn tbtn-fire">fire</span>
                </label>
            </li>
        </ul>
    </div>
</div>

1 Ответ

0 голосов
/ 04 ноября 2018

Технически говоря, это недостающая особенность методов Linq. Кодировать функции Async Linq не так просто.

С другой стороны, я считаю, что это гораздо проще реализовать прямо в javascript. Зачем? потому что вы можете проверить свой скрипт в браузере, а затем просто переместить его в код C #.

Например, этот метод найдет все li элементы, найдет li с интервалом с текстом fire и затем щелкните по входу.

await page.EvaluateFunctionAsync(@"() => {
     Array.from(document.querySelectorAll('li'))
         .find(l => l.querySelector('span').innerText === 'fire').querySelector('INPUT').click();
}");
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...