代码之家  ›  专栏  ›  技术社区  ›  Peter

如何在WinForms WebBrowser控件中生成的HTML中引用本地资源?

  •  3
  • Peter  · 技术社区  · 16 年前

    我正在使用winforms浏览器控件在windows窗体应用程序中显示一些内容。我正在使用DocumentText属性编写生成的HTML。这部分工作得非常出色。现在我想在标记中使用一些图像。(不过,我也更喜欢使用链接的CSS和JavaScript,只需嵌入即可解决。)

    我在谷歌上搜索了几天,似乎找不到标题问题的答案。

    我尝试使用相对引用:应用程序exe位于bin\debug中。图像位于项目根目录的“images”目录中。我已将图像设置为在编译时复制到输出目录,因此它们最终会出现在bin\debug\images*中。因此,我使用这样的引用“Images…”,认为它与exe有关。但是,当我查看嵌入式浏览器窗口中的图像属性时,我看到图像URL为“about:blankImages/*”。当HTML写入控件时,一切似乎都与“about:blank”有关。由于缺少位置上下文,我无法弄清楚如何使用相对文件资源引用。

    我仔细研究了控件的属性,看看是否有办法设置一些东西来解决这个问题。我创建了一个空白的html页面,并使用“Navigate()”方法将浏览器指向它,使用文件的完整本地路径。这与浏览器报告本地“file:///...“空白页的路径。然后我再次写入浏览器,这次使用Document.Write()。同样,浏览器现在报告”about:blank“作为URL。

    除了将动态HTML结果写入真实文件之外,难道没有其他方法可以引用文件资源吗?

    我将尝试最后一件事:构造图像的绝对文件路径并将其写入HTML。我的HTML是使用序列化对象的XML的XSL转换生成的,因此我需要使用一些XSL参数,这将需要额外的时间,因为我对它们不太熟悉。

    5 回复  |  直到 14 年前
        1
  •  4
  •   Ken Wootton    16 年前

    以下是我们所做的,尽管我应该提到的是,我们使用自定义web浏览器来删除诸如右键单击和查看旧IE上下文菜单等功能:

    public class HtmlFormatter
    {
        /// <summary>
        /// Indicator that this is a URI referencing the local
        /// file path.
        /// </summary>
        public static readonly string FILE_URL_PREFIX = 
            "file://";
    
        /// <summary>
        /// The path separator for HTML paths.
        /// </summary>
        public const string PATH_SEPARATOR = "/";
    }
    
    
    // We need to add the proper paths to each image source
    // designation that match where they are being placed on disk.
    String html = HtmlFormatter.ReplaceImagePath(
        myHtml, 
        HtmlFormatter.FILE_URL_PREFIX + ApplicationPath.FullAppPath + 
        HtmlFormatter.PATH_SEPARATOR);
    

    基本上,您需要一个具有文件URI的图像路径,例如。

    <img src="file://ApplicationPath/images/myImage.gif">
    
        2
  •  1
  •   Peter    16 年前

    我弄明白了。

    我只是将exe目录的完整解析url传递给XSL转换,该转换包含带有图像标签的HTML输出:

    XsltArgumentList lstArgs = new XsltArgumentList();
    lstArgs.AddParam("absoluteRoot", string.Empty, Path.GetFullPath("."));
    

    然后,我只是用参数值作为所有图像的前缀:

    <img src="{$absoluteRoot}/Images/SilkIcons/comment_add.gif" align="middle" border="0" />
    
        3
  •  0
  •   Peter    16 年前

    我最终使用了与肯建议基本相同的东西。但是,我没有手动附加文件前缀,而是使用UriBuilder类使用“文件”协议构建完整的URI。

    这也解决了我们在更真实的位置Program Files测试应用程序时出现的后续问题。空格已编码,但当使用标准系统路径(即“C:\Program%20Files…”)引用文件时,操作系统无法处理编码字符。使用真实URI值(file:///C:/Program文件/。..)工作。

        4
  •  0
  •   Roark Fan    16 年前

    或者,保留正常样式的相对链接,删除HTML转换代码,而是嵌入一个C#web服务器,如 this 在exe中,然后将WebControl指向您的内部URL,如localhost:8199/myapp/

        5
  •  0
  •   Orwellophile    16 年前

    Ken的代码缺少一些工作所需的东西。我已经对其进行了修改,并创建了一个新的方法,可以使事情自动化一点。

    只需按如下方式调用静态方法:

    html = HtmlFormatter.ReplaceImagePathAuto(html);
    

    以及html中所有匹配的链接file://ApplicationPath/将与当前工作目录交换。如果要指定备用位置,则包含原始静态方法(加上缺少的位)。

    public class HtmlFormatter
    {
    
        public static readonly string FILE_URL_PREFIX = "file://";
        public static readonly string PATH_SEPARATOR = "/";
        public static String ReplaceImagePath(String html, String path)
        {
            return html.Replace("file://ApplicationPath/", path);
        }
        /// <summary>
        /// Replaces URLs matching file://ApplicationPath/... with Executable Path
        /// </summary>
        /// <param name="html"></param>
        /// <returns></returns>
        public static String ReplaceImagePathAuto(String html)
        {
            String executableName = System.Windows.Forms.Application.ExecutablePath;
            System.IO.FileInfo executableFileInfo = new System.IO.FileInfo(executableName);
            String executableDirectoryName = executableFileInfo.DirectoryName;
            String replaceWith = HtmlFormatter.FILE_URL_PREFIX
                + executableDirectoryName
                + HtmlFormatter.PATH_SEPARATOR;
    
            return ReplaceImagePath(html, replaceWith);
        }
    }