3

我正在做一些 HTML 解析,我正在使用HtmlAgilityPack,我正在尝试检查如果在浏览器中呈现 html,节点元素是否可见。

通过可见,我可能对检查displayvisibility样式值感到满意。(除非有什么额外的事情我应该担心?)。

那么,我该怎么做呢?有没有简单的方法?我可以使用一些 XPath 魔法吗?(目前我对 XPath 的了解不多)。

我曾考虑过手动解析样式值,但宁愿将其保存为最后的手段。或者这是我唯一的选择?

仅供参考,我正在使用的对象是这样的:

HtmlAgilityPack.HtmlNode node = GetNode();
4

1 回答 1

5

好的,所以我已经设法做到了,至少满足我的需要。但是请注意,正如其他评论所说,这不允许您检查最终用户是否可以看到元素(在屏幕上)。

我采用的方法简单检查了一些基本规则:如果元素的样式属性包含display:noneor ,则元素“不可见” visibility:hidden,或者祖先元素具有相同的样式规则。

考虑到这一点,这是为我完成这项工作的代码:

private static bool IsNodeVisible(HtmlAgilityPack.HtmlNode node)
{
    var attribute = node.Attributes["style"];

    bool thisVisible = false;

    if (attribute == null || CheckStyleVisibility(attribute.Value))
        thisVisible = true;

    if (thisVisible && node.ParentNode != null)
        return IsNodeVisible(node.ParentNode);

    return thisVisible;
}

private static bool CheckStyleVisibility(string style)
{
    if (string.IsNullOrWhiteSpace(style))
        return true;

    var keys = ParseHtmlStyleString(style);

    if (keys.Keys.Contains("display"))
    {
        string display = keys["display"];
        if (display != null && display == "none")
            return false;
    }

    if (keys.Keys.Contains("visibility"))
    {
        string visibility = keys["visibility"];
        if (visibility != null && visibility == "hidden")
            return false;
    }

    return true;
}

public static Dictionary<string, string> ParseHtmlStyleString(string style)
{
    Dictionary<string, string> result = new Dictionary<string, string>();

    style = style.Replace(" ", "").ToLowerInvariant();

    string[] settings = style.Split(new char[] { ';' }, StringSplitOptions.RemoveEmptyEntries);

    foreach (string s in settings)
    {
        if (!s.Contains(':'))
            continue;
        string[] data = s.Split(':');
        result.Add(data[0], data[1]);
    }

    return result;
}

入口点是IsNodeVisible并将检查HtmlNode传递给它的可见性。

于 2013-02-21T14:55:28.567 回答