日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > windows >内容正文

windows

MAUI Blazor 如何通过url使用本地文件

發(fā)布時間:2023/11/29 windows 46 coder
生活随笔 收集整理的這篇文章主要介紹了 MAUI Blazor 如何通过url使用本地文件 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

前言

上一篇文章 MAUI Blazor 顯示本地圖片的新思路 中, 提出了通過webview攔截,從而在前端中顯示本地圖片的思路。不過當時還不完善,隨后也發(fā)現(xiàn)了很多問題。比如,

  1. 不同平臺上的url不統(tǒng)一。這對于需要存儲圖片路徑并且多端互通的需求來說,并不友好。至少 FileSystem.AppDataDirectoryFileSystem.CacheDirectory 下的文件生成的url應(yīng)該統(tǒng)一。
  2. 音頻文件和視頻文件無法使用。理論上可以用于各種文件,但是音頻和視頻不能播放,應(yīng)該是需要相應(yīng)的處理
  3. Windows上有限制。大于9~10M的圖片不顯示
  4. iOS/ Mac有跨域問題。尤其是調(diào)用用于截圖的js庫,圖片會由于跨域不出現(xiàn)在截圖中

所以,在這篇文章中,對這個思路進行完善,使之成為一個可行的方案。

例如 <img src='appdata/Image/image1.jpg' > 會顯示 FileSystem.AppDataDirectory 文件夾下的 Image 文件夾下的 image1.jpg 這個圖片
<video src='cache/Video/video1.mp4' controls > 會播放 FileSystem.CacheDirectory 文件夾下的 Video 文件夾下的 video1.mp4 這個視頻
對于其他路徑的文件來說,url設(shè)為 file/ 加上轉(zhuǎn)義后的完整路徑

正文

準備工作

新建一個MAUI Blazor項目

參考 配置基于文件名的多目標 ,更改項目文件(以.csproj結(jié)尾的文件),添加以下代碼

<!-- Android -->
<ItemGroup Condition="$(TargetFramework.StartsWith('net8.0-android')) != true">
  <Compile Remove="**\**\*.Android.cs" />
  <None Include="**\**\*.Android.cs" Exclude="$(DefaultItemExcludes);$(DefaultExcludesInProjectFolder)" />
</ItemGroup>

<!-- Both iOS and Mac Catalyst -->
<ItemGroup Condition="$(TargetFramework.StartsWith('net8.0-ios')) != true AND $(TargetFramework.StartsWith('net8.0-maccatalyst')) != true">
  <Compile Remove="**\**\*.MaciOS.cs" />
  <None Include="**\**\*.MaciOS.cs" Exclude="$(DefaultItemExcludes);$(DefaultExcludesInProjectFolder)" />
</ItemGroup>

<!-- iOS -->
<ItemGroup Condition="$(TargetFramework.StartsWith('net8.0-ios')) != true">
  <Compile Remove="**\**\*.iOS.cs" />
  <None Include="**\**\*.iOS.cs" Exclude="$(DefaultItemExcludes);$(DefaultExcludesInProjectFolder)" />
</ItemGroup>

<!-- Mac Catalyst -->
<ItemGroup Condition="$(TargetFramework.StartsWith('net8.0-maccatalyst')) != true">
  <Compile Remove="**\**\*.MacCatalyst.cs" />
  <None Include="**\**\*.MacCatalyst.cs" Exclude="$(DefaultItemExcludes);$(DefaultExcludesInProjectFolder)" />
</ItemGroup>

<!-- Windows -->
<ItemGroup Condition="$(TargetFramework.Contains('-windows')) != true">
  <Compile Remove="**\*.Windows.cs" />
  <None Include="**\*.Windows.cs" Exclude="$(DefaultItemExcludes);$(DefaultExcludesInProjectFolder)" />
</ItemGroup>

添加一個處理ContentType的靜態(tài)類

用來獲取文件的ContentType
沒找到什么太好的方法,偶然看到Maui的源碼中的一段,還不錯,不過是internal修飾的,就直接抄來了
新建Utilities/MimeType文件夾,在里面添加 StaticContentProvider.cs
代碼如下:

#nullable disable
// Copyright (c) .NET Foundation. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.

using System.Diagnostics.CodeAnalysis;

namespace MauiBlazorLocalMediaFile.Utilities
{
    internal partial class StaticContentProvider
	{
		private static readonly FileExtensionContentTypeProvider ContentTypeProvider = new();

		internal static string GetResponseContentTypeOrDefault(string path)
			=> ContentTypeProvider.TryGetContentType(path, out var matchedContentType)
			? matchedContentType
			: "application/octet-stream";

		internal static IDictionary<string, string> GetResponseHeaders(string contentType)
			=> new Dictionary<string, string>(StringComparer.Ordinal)
			{
				{ "Content-Type", contentType },
				{ "Cache-Control", "no-cache, max-age=0, must-revalidate, no-store" },
			};

		internal class FileExtensionContentTypeProvider
		{
			// Notes:
			// - This table was initially copied from IIS and has many legacy entries we will maintain for backwards compatibility.
			// - We only plan to add new entries where we expect them to be applicable to a majority of developers such as being
			// used in the project templates.
			#region Extension mapping table
			/// <summary>
			/// Creates a new provider with a set of default mappings.
			/// </summary>
			public FileExtensionContentTypeProvider()
				: this(new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
				{
				{ ".323", "text/h323" },
				{ ".3g2", "video/3gpp2" },
				{ ".3gp2", "video/3gpp2" },
				{ ".3gp", "video/3gpp" },
				{ ".3gpp", "video/3gpp" },
				{ ".aac", "audio/aac" },
				{ ".aaf", "application/octet-stream" },
				{ ".aca", "application/octet-stream" },
				{ ".accdb", "application/msaccess" },
				{ ".accde", "application/msaccess" },
				{ ".accdt", "application/msaccess" },
				{ ".acx", "application/internet-property-stream" },
				{ ".adt", "audio/vnd.dlna.adts" },
				{ ".adts", "audio/vnd.dlna.adts" },
				{ ".afm", "application/octet-stream" },
				{ ".ai", "application/postscript" },
				{ ".aif", "audio/x-aiff" },
				{ ".aifc", "audio/aiff" },
				{ ".aiff", "audio/aiff" },
				{ ".appcache", "text/cache-manifest" },
				{ ".application", "application/x-ms-application" },
				{ ".art", "image/x-jg" },
				{ ".asd", "application/octet-stream" },
				{ ".asf", "video/x-ms-asf" },
				{ ".asi", "application/octet-stream" },
				{ ".asm", "text/plain" },
				{ ".asr", "video/x-ms-asf" },
				{ ".asx", "video/x-ms-asf" },
				{ ".atom", "application/atom+xml" },
				{ ".au", "audio/basic" },
				{ ".avi", "video/x-msvideo" },
				{ ".axs", "application/olescript" },
				{ ".bas", "text/plain" },
				{ ".bcpio", "application/x-bcpio" },
				{ ".bin", "application/octet-stream" },
				{ ".bmp", "image/bmp" },
				{ ".c", "text/plain" },
				{ ".cab", "application/vnd.ms-cab-compressed" },
				{ ".calx", "application/vnd.ms-office.calx" },
				{ ".cat", "application/vnd.ms-pki.seccat" },
				{ ".cdf", "application/x-cdf" },
				{ ".chm", "application/octet-stream" },
				{ ".class", "application/x-java-applet" },
				{ ".clp", "application/x-msclip" },
				{ ".cmx", "image/x-cmx" },
				{ ".cnf", "text/plain" },
				{ ".cod", "image/cis-cod" },
				{ ".cpio", "application/x-cpio" },
				{ ".cpp", "text/plain" },
				{ ".crd", "application/x-mscardfile" },
				{ ".crl", "application/pkix-crl" },
				{ ".crt", "application/x-x509-ca-cert" },
				{ ".csh", "application/x-csh" },
				{ ".css", "text/css" },
				{ ".csv", "text/csv" }, // https://tools.ietf.org/html/rfc7111#section-5.1
                { ".cur", "application/octet-stream" },
				{ ".dcr", "application/x-director" },
				{ ".deploy", "application/octet-stream" },
				{ ".der", "application/x-x509-ca-cert" },
				{ ".dib", "image/bmp" },
				{ ".dir", "application/x-director" },
				{ ".disco", "text/xml" },
				{ ".dlm", "text/dlm" },
				{ ".doc", "application/msword" },
				{ ".docm", "application/vnd.ms-word.document.macroEnabled.12" },
				{ ".docx", "application/vnd.openxmlformats-officedocument.wordprocessingml.document" },
				{ ".dot", "application/msword" },
				{ ".dotm", "application/vnd.ms-word.template.macroEnabled.12" },
				{ ".dotx", "application/vnd.openxmlformats-officedocument.wordprocessingml.template" },
				{ ".dsp", "application/octet-stream" },
				{ ".dtd", "text/xml" },
				{ ".dvi", "application/x-dvi" },
				{ ".dvr-ms", "video/x-ms-dvr" },
				{ ".dwf", "drawing/x-dwf" },
				{ ".dwp", "application/octet-stream" },
				{ ".dxr", "application/x-director" },
				{ ".eml", "message/rfc822" },
				{ ".emz", "application/octet-stream" },
				{ ".eot", "application/vnd.ms-fontobject" },
				{ ".eps", "application/postscript" },
				{ ".etx", "text/x-setext" },
				{ ".evy", "application/envoy" },
				{ ".exe", "application/vnd.microsoft.portable-executable" }, // https://www.iana.org/assignments/media-types/application/vnd.microsoft.portable-executable
                { ".fdf", "application/vnd.fdf" },
				{ ".fif", "application/fractals" },
				{ ".fla", "application/octet-stream" },
				{ ".flr", "x-world/x-vrml" },
				{ ".flv", "video/x-flv" },
				{ ".gif", "image/gif" },
				{ ".gtar", "application/x-gtar" },
				{ ".gz", "application/x-gzip" },
				{ ".h", "text/plain" },
				{ ".hdf", "application/x-hdf" },
				{ ".hdml", "text/x-hdml" },
				{ ".hhc", "application/x-oleobject" },
				{ ".hhk", "application/octet-stream" },
				{ ".hhp", "application/octet-stream" },
				{ ".hlp", "application/winhlp" },
				{ ".hqx", "application/mac-binhex40" },
				{ ".hta", "application/hta" },
				{ ".htc", "text/x-component" },
				{ ".htm", "text/html" },
				{ ".html", "text/html" },
				{ ".htt", "text/webviewhtml" },
				{ ".hxt", "text/html" },
				{ ".ical", "text/calendar" },
				{ ".icalendar", "text/calendar" },
				{ ".ico", "image/x-icon" },
				{ ".ics", "text/calendar" },
				{ ".ief", "image/ief" },
				{ ".ifb", "text/calendar" },
				{ ".iii", "application/x-iphone" },
				{ ".inf", "application/octet-stream" },
				{ ".ins", "application/x-internet-signup" },
				{ ".isp", "application/x-internet-signup" },
				{ ".IVF", "video/x-ivf" },
				{ ".jar", "application/java-archive" },
				{ ".java", "application/octet-stream" },
				{ ".jck", "application/liquidmotion" },
				{ ".jcz", "application/liquidmotion" },
				{ ".jfif", "image/pjpeg" },
				{ ".jpb", "application/octet-stream" },
				{ ".jpe", "image/jpeg" },
				{ ".jpeg", "image/jpeg" },
				{ ".jpg", "image/jpeg" },
				{ ".js", "application/javascript" },
				{ ".json", "application/json" },
				{ ".jsx", "text/jscript" },
				{ ".latex", "application/x-latex" },
				{ ".lit", "application/x-ms-reader" },
				{ ".lpk", "application/octet-stream" },
				{ ".lsf", "video/x-la-asf" },
				{ ".lsx", "video/x-la-asf" },
				{ ".lzh", "application/octet-stream" },
				{ ".m13", "application/x-msmediaview" },
				{ ".m14", "application/x-msmediaview" },
				{ ".m1v", "video/mpeg" },
				{ ".m2ts", "video/vnd.dlna.mpeg-tts" },
				{ ".m3u", "audio/x-mpegurl" },
				{ ".m4a", "audio/mp4" },
				{ ".m4v", "video/mp4" },
				{ ".man", "application/x-troff-man" },
				{ ".manifest", "application/x-ms-manifest" },
				{ ".map", "text/plain" },
				{ ".markdown", "text/markdown" },
				{ ".md", "text/markdown" },
				{ ".mdb", "application/x-msaccess" },
				{ ".mdp", "application/octet-stream" },
				{ ".me", "application/x-troff-me" },
				{ ".mht", "message/rfc822" },
				{ ".mhtml", "message/rfc822" },
				{ ".mid", "audio/mid" },
				{ ".midi", "audio/mid" },
				{ ".mix", "application/octet-stream" },
				{ ".mmf", "application/x-smaf" },
				{ ".mno", "text/xml" },
				{ ".mny", "application/x-msmoney" },
				{ ".mov", "video/quicktime" },
				{ ".movie", "video/x-sgi-movie" },
				{ ".mp2", "video/mpeg" },
				{ ".mp3", "audio/mpeg" },
				{ ".mp4", "video/mp4" },
				{ ".mp4v", "video/mp4" },
				{ ".mpa", "video/mpeg" },
				{ ".mpe", "video/mpeg" },
				{ ".mpeg", "video/mpeg" },
				{ ".mpg", "video/mpeg" },
				{ ".mpp", "application/vnd.ms-project" },
				{ ".mpv2", "video/mpeg" },
				{ ".ms", "application/x-troff-ms" },
				{ ".msi", "application/octet-stream" },
				{ ".mso", "application/octet-stream" },
				{ ".mvb", "application/x-msmediaview" },
				{ ".mvc", "application/x-miva-compiled" },
				{ ".nc", "application/x-netcdf" },
				{ ".nsc", "video/x-ms-asf" },
				{ ".nws", "message/rfc822" },
				{ ".ocx", "application/octet-stream" },
				{ ".oda", "application/oda" },
				{ ".odc", "text/x-ms-odc" },
				{ ".ods", "application/oleobject" },
				{ ".oga", "audio/ogg" },
				{ ".ogg", "video/ogg" },
				{ ".ogv", "video/ogg" },
				{ ".ogx", "application/ogg" },
				{ ".one", "application/onenote" },
				{ ".onea", "application/onenote" },
				{ ".onetoc", "application/onenote" },
				{ ".onetoc2", "application/onenote" },
				{ ".onetmp", "application/onenote" },
				{ ".onepkg", "application/onenote" },
				{ ".osdx", "application/opensearchdescription+xml" },
				{ ".otf", "font/otf" },
				{ ".p10", "application/pkcs10" },
				{ ".p12", "application/x-pkcs12" },
				{ ".p7b", "application/x-pkcs7-certificates" },
				{ ".p7c", "application/pkcs7-mime" },
				{ ".p7m", "application/pkcs7-mime" },
				{ ".p7r", "application/x-pkcs7-certreqresp" },
				{ ".p7s", "application/pkcs7-signature" },
				{ ".pbm", "image/x-portable-bitmap" },
				{ ".pcx", "application/octet-stream" },
				{ ".pcz", "application/octet-stream" },
				{ ".pdf", "application/pdf" },
				{ ".pfb", "application/octet-stream" },
				{ ".pfm", "application/octet-stream" },
				{ ".pfx", "application/x-pkcs12" },
				{ ".pgm", "image/x-portable-graymap" },
				{ ".pko", "application/vnd.ms-pki.pko" },
				{ ".pma", "application/x-perfmon" },
				{ ".pmc", "application/x-perfmon" },
				{ ".pml", "application/x-perfmon" },
				{ ".pmr", "application/x-perfmon" },
				{ ".pmw", "application/x-perfmon" },
				{ ".png", "image/png" },
				{ ".pnm", "image/x-portable-anymap" },
				{ ".pnz", "image/png" },
				{ ".pot", "application/vnd.ms-powerpoint" },
				{ ".potm", "application/vnd.ms-powerpoint.template.macroEnabled.12" },
				{ ".potx", "application/vnd.openxmlformats-officedocument.presentationml.template" },
				{ ".ppam", "application/vnd.ms-powerpoint.addin.macroEnabled.12" },
				{ ".ppm", "image/x-portable-pixmap" },
				{ ".pps", "application/vnd.ms-powerpoint" },
				{ ".ppsm", "application/vnd.ms-powerpoint.slideshow.macroEnabled.12" },
				{ ".ppsx", "application/vnd.openxmlformats-officedocument.presentationml.slideshow" },
				{ ".ppt", "application/vnd.ms-powerpoint" },
				{ ".pptm", "application/vnd.ms-powerpoint.presentation.macroEnabled.12" },
				{ ".pptx", "application/vnd.openxmlformats-officedocument.presentationml.presentation" },
				{ ".prf", "application/pics-rules" },
				{ ".prm", "application/octet-stream" },
				{ ".prx", "application/octet-stream" },
				{ ".ps", "application/postscript" },
				{ ".psd", "application/octet-stream" },
				{ ".psm", "application/octet-stream" },
				{ ".psp", "application/octet-stream" },
				{ ".pub", "application/x-mspublisher" },
				{ ".qt", "video/quicktime" },
				{ ".qtl", "application/x-quicktimeplayer" },
				{ ".qxd", "application/octet-stream" },
				{ ".ra", "audio/x-pn-realaudio" },
				{ ".ram", "audio/x-pn-realaudio" },
				{ ".rar", "application/octet-stream" },
				{ ".ras", "image/x-cmu-raster" },
				{ ".rf", "image/vnd.rn-realflash" },
				{ ".rgb", "image/x-rgb" },
				{ ".rm", "application/vnd.rn-realmedia" },
				{ ".rmi", "audio/mid" },
				{ ".roff", "application/x-troff" },
				{ ".rpm", "audio/x-pn-realaudio-plugin" },
				{ ".rtf", "application/rtf" },
				{ ".rtx", "text/richtext" },
				{ ".scd", "application/x-msschedule" },
				{ ".sct", "text/scriptlet" },
				{ ".sea", "application/octet-stream" },
				{ ".setpay", "application/set-payment-initiation" },
				{ ".setreg", "application/set-registration-initiation" },
				{ ".sgml", "text/sgml" },
				{ ".sh", "application/x-sh" },
				{ ".shar", "application/x-shar" },
				{ ".sit", "application/x-stuffit" },
				{ ".sldm", "application/vnd.ms-powerpoint.slide.macroEnabled.12" },
				{ ".sldx", "application/vnd.openxmlformats-officedocument.presentationml.slide" },
				{ ".smd", "audio/x-smd" },
				{ ".smi", "application/octet-stream" },
				{ ".smx", "audio/x-smd" },
				{ ".smz", "audio/x-smd" },
				{ ".snd", "audio/basic" },
				{ ".snp", "application/octet-stream" },
				{ ".spc", "application/x-pkcs7-certificates" },
				{ ".spl", "application/futuresplash" },
				{ ".spx", "audio/ogg" },
				{ ".src", "application/x-wais-source" },
				{ ".ssm", "application/streamingmedia" },
				{ ".sst", "application/vnd.ms-pki.certstore" },
				{ ".stl", "application/vnd.ms-pki.stl" },
				{ ".sv4cpio", "application/x-sv4cpio" },
				{ ".sv4crc", "application/x-sv4crc" },
				{ ".svg", "image/svg+xml" },
				{ ".svgz", "image/svg+xml" },
				{ ".swf", "application/x-shockwave-flash" },
				{ ".t", "application/x-troff" },
				{ ".tar", "application/x-tar" },
				{ ".tcl", "application/x-tcl" },
				{ ".tex", "application/x-tex" },
				{ ".texi", "application/x-texinfo" },
				{ ".texinfo", "application/x-texinfo" },
				{ ".tgz", "application/x-compressed" },
				{ ".thmx", "application/vnd.ms-officetheme" },
				{ ".thn", "application/octet-stream" },
				{ ".tif", "image/tiff" },
				{ ".tiff", "image/tiff" },
				{ ".toc", "application/octet-stream" },
				{ ".tr", "application/x-troff" },
				{ ".trm", "application/x-msterminal" },
				{ ".ts", "video/vnd.dlna.mpeg-tts" },
				{ ".tsv", "text/tab-separated-values" },
				{ ".ttc", "application/x-font-ttf" },
				{ ".ttf", "application/x-font-ttf" },
				{ ".tts", "video/vnd.dlna.mpeg-tts" },
				{ ".txt", "text/plain" },
				{ ".u32", "application/octet-stream" },
				{ ".uls", "text/iuls" },
				{ ".ustar", "application/x-ustar" },
				{ ".vbs", "text/vbscript" },
				{ ".vcf", "text/x-vcard" },
				{ ".vcs", "text/plain" },
				{ ".vdx", "application/vnd.ms-visio.viewer" },
				{ ".vml", "text/xml" },
				{ ".vsd", "application/vnd.visio" },
				{ ".vss", "application/vnd.visio" },
				{ ".vst", "application/vnd.visio" },
				{ ".vsto", "application/x-ms-vsto" },
				{ ".vsw", "application/vnd.visio" },
				{ ".vsx", "application/vnd.visio" },
				{ ".vtx", "application/vnd.visio" },
				{ ".wasm", "application/wasm" },
				{ ".wav", "audio/wav" },
				{ ".wax", "audio/x-ms-wax" },
				{ ".wbmp", "image/vnd.wap.wbmp" },
				{ ".wcm", "application/vnd.ms-works" },
				{ ".wdb", "application/vnd.ms-works" },
				{ ".webm", "video/webm" },
				{ ".webmanifest", "application/manifest+json" }, // https://w3c.github.io/manifest/#media-type-registration
                { ".webp", "image/webp" },
				{ ".wks", "application/vnd.ms-works" },
				{ ".wm", "video/x-ms-wm" },
				{ ".wma", "audio/x-ms-wma" },
				{ ".wmd", "application/x-ms-wmd" },
				{ ".wmf", "application/x-msmetafile" },
				{ ".wml", "text/vnd.wap.wml" },
				{ ".wmlc", "application/vnd.wap.wmlc" },
				{ ".wmls", "text/vnd.wap.wmlscript" },
				{ ".wmlsc", "application/vnd.wap.wmlscriptc" },
				{ ".wmp", "video/x-ms-wmp" },
				{ ".wmv", "video/x-ms-wmv" },
				{ ".wmx", "video/x-ms-wmx" },
				{ ".wmz", "application/x-ms-wmz" },
				{ ".woff", "application/font-woff" }, // https://www.w3.org/TR/WOFF/#appendix-b
                { ".woff2", "font/woff2" }, // https://www.w3.org/TR/WOFF2/#IMT
                { ".wps", "application/vnd.ms-works" },
				{ ".wri", "application/x-mswrite" },
				{ ".wrl", "x-world/x-vrml" },
				{ ".wrz", "x-world/x-vrml" },
				{ ".wsdl", "text/xml" },
				{ ".wtv", "video/x-ms-wtv" },
				{ ".wvx", "video/x-ms-wvx" },
				{ ".x", "application/directx" },
				{ ".xaf", "x-world/x-vrml" },
				{ ".xaml", "application/xaml+xml" },
				{ ".xap", "application/x-silverlight-app" },
				{ ".xbap", "application/x-ms-xbap" },
				{ ".xbm", "image/x-xbitmap" },
				{ ".xdr", "text/plain" },
				{ ".xht", "application/xhtml+xml" },
				{ ".xhtml", "application/xhtml+xml" },
				{ ".xla", "application/vnd.ms-excel" },
				{ ".xlam", "application/vnd.ms-excel.addin.macroEnabled.12" },
				{ ".xlc", "application/vnd.ms-excel" },
				{ ".xlm", "application/vnd.ms-excel" },
				{ ".xls", "application/vnd.ms-excel" },
				{ ".xlsb", "application/vnd.ms-excel.sheet.binary.macroEnabled.12" },
				{ ".xlsm", "application/vnd.ms-excel.sheet.macroEnabled.12" },
				{ ".xlsx", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" },
				{ ".xlt", "application/vnd.ms-excel" },
				{ ".xltm", "application/vnd.ms-excel.template.macroEnabled.12" },
				{ ".xltx", "application/vnd.openxmlformats-officedocument.spreadsheetml.template" },
				{ ".xlw", "application/vnd.ms-excel" },
				{ ".xml", "text/xml" },
				{ ".xof", "x-world/x-vrml" },
				{ ".xpm", "image/x-xpixmap" },
				{ ".xps", "application/vnd.ms-xpsdocument" },
				{ ".xsd", "text/xml" },
				{ ".xsf", "text/xml" },
				{ ".xsl", "text/xml" },
				{ ".xslt", "text/xml" },
				{ ".xsn", "application/octet-stream" },
				{ ".xtp", "application/octet-stream" },
				{ ".xwd", "image/x-xwindowdump" },
				{ ".z", "application/x-compress" },
				{ ".zip", "application/x-zip-compressed" },
				})
			{
			}
			#endregion

			/// <summary>
			/// Creates a lookup engine using the provided mapping.
			/// It is recommended that the IDictionary instance use StringComparer.OrdinalIgnoreCase.
			/// </summary>
			/// <param name="mapping"></param>
			public FileExtensionContentTypeProvider(IDictionary<string, string> mapping)
			{
				if (mapping == null)
				{
					throw new ArgumentNullException(nameof(mapping));
				}
				Mappings = mapping;
			}

			/// <summary>
			/// The cross reference table of file extensions and content-types.
			/// </summary>
			public IDictionary<string, string> Mappings { get; private set; }

			/// <summary>
			/// Given a file path, determine the MIME type
			/// </summary>
			/// <param name="subpath">A file path</param>
			/// <param name="contentType">The resulting MIME type</param>
			/// <returns>True if MIME type could be determined</returns>
			public bool TryGetContentType(string subpath, [MaybeNullWhen(false)] out string contentType)
			{
				var extension = GetExtension(subpath);
				if (extension == null)
				{
					contentType = null;
					return false;
				}
				return Mappings.TryGetValue(extension, out contentType);
			}

			private static string GetExtension(string path)
			{
				// Don't use Path.GetExtension as that may throw an exception if there are
				// invalid characters in the path. Invalid characters should be handled
				// by the FileProviders

				if (string.IsNullOrWhiteSpace(path))
				{
					return null;
				}

				int index = path.LastIndexOf('.');
				if (index < 0)
				{
					return null;
				}

				return path.Substring(index);
			}
		}
	}
}

創(chuàng)建自定義的BlazorWebViewHandler類

BlazorWebViewHandler是MAUI Blazor中處理BlazorWebView相關(guān)的一個類,我們自定義一個類替換它,添加自己需要的一些處理邏輯

Maui Blazor中iOS / Mac和其他平臺的baseUrl是不統(tǒng)一的,iOS / Mac是 app://0.0.0.0 (原因在Maui源碼的注釋中有寫到,iOS WKWebView doesn't allow handling 'http'/'https' schemes, so we use the fake 'app' scheme),其他平臺是 https://0.0.0.0 ,所以我們的url設(shè)為相對路徑才能統(tǒng)一,而且與頁面同源,不會有跨域問題(筆者之前在iOS / Mac上的做法就是注冊自定義協(xié)議,使用html2canvas截圖時,結(jié)果發(fā)生了跨域問題)。

添加MauiBlazorWebViewHandler.cs
代碼如下

using Microsoft.AspNetCore.Components.WebView.Maui;

namespace MauiBlazorLocalMediaFile
{
    public partial class MauiBlazorWebViewHandler : BlazorWebViewHandler
    {
        private const string AppHostAddress = "0.0.0.0";
#if IOS || MACCATALYST
        public const string BaseUri = $"app://{AppHostAddress}/";
#else
        public const string BaseUri = $"https://{AppHostAddress}/";
#endif
        public readonly static Dictionary<string, string> AppFilePathMap = new()
        {
            { FileSystem.AppDataDirectory, "appdata" },
            { FileSystem.CacheDirectory, "cache" },
        };

        private static readonly string OtherFileMapPath = "file";

        //把真實的文件路徑轉(zhuǎn)化為url相對路徑
        public static string FilePathToUrlRelativePath(string filePath)
        {
            foreach (var item in AppFilePathMap)
            {
                if (filePath.StartsWith(item.Key))
                {
                    return item.Value + filePath[item.Key.Length..].Replace(Path.DirectorySeparatorChar, '/');
                }
            }

            return OtherFileMapPath + "/" + Uri.EscapeDataString(filePath);
        }

        //把url相對路徑轉(zhuǎn)化為真實的文件路徑
        public static string UrlRelativePathToFilePath(string urlRelativePath)
        {
            UrlRelativePathToFilePath(urlRelativePath, out string path);
            return path;
        }

        private static bool Intercept(string uri, out string path)
        {
            if (!uri.StartsWith(BaseUri))
            {
                path = string.Empty;
                return false;
            }

            var urlRelativePath = uri[BaseUri.Length..];
            return UrlRelativePathToFilePath(urlRelativePath, out path);
        }

        private static bool UrlRelativePathToFilePath(string urlRelativePath, out string path)
        {
            if (string.IsNullOrEmpty(urlRelativePath))
            {
                path = string.Empty;
                return false;
            }

            urlRelativePath = Uri.UnescapeDataString(urlRelativePath);

            foreach (var item in AppFilePathMap)
            {
                if (urlRelativePath.StartsWith(item.Value + '/'))
                {
                    string urlRelativePathSub = urlRelativePath[(item.Value.Length + 1)..];
                    path = Path.Combine(item.Key, urlRelativePathSub.Replace('/', Path.DirectorySeparatorChar));
                    if (File.Exists(path))
                    {
                        return true;
                    }
                }
            }

            if (urlRelativePath.StartsWith(OtherFileMapPath + '/'))
            {
                string urlRelativePathSub = urlRelativePath[(OtherFileMapPath.Length + 1)..];
                path = urlRelativePathSub.Replace('/', Path.DirectorySeparatorChar);
                if (File.Exists(path))
                {
                    return true;
                }
            }

            path = string.Empty;
            return false;
        }
    }
}


Android

添加MauiBlazorWebViewHandler.Android.cs
代碼如下

using Android.Webkit;
using MauiBlazorLocalMediaFile.Utilities;
using WebView = Android.Webkit.WebView;

namespace MauiBlazorLocalMediaFile
{
    public partial class MauiBlazorWebViewHandler
    {
#pragma warning disable CA1416 // 驗證平臺兼容性
        protected override void ConnectHandler(WebView platformView)
        {
            base.ConnectHandler(platformView);
            platformView.SetWebViewClient(new MyWebViewClient(platformView.WebViewClient));
        }

#nullable disable
        private class MyWebViewClient : WebViewClient
        {
            private WebViewClient WebViewClient { get; }

            public MyWebViewClient(WebViewClient webViewClient)
            {
                WebViewClient = webViewClient;
            }

            public override bool ShouldOverrideUrlLoading(Android.Webkit.WebView view, IWebResourceRequest request)
            {
                return WebViewClient.ShouldOverrideUrlLoading(view, request);
            }

            public override WebResourceResponse ShouldInterceptRequest(Android.Webkit.WebView view, IWebResourceRequest request)
            {
                var intercept = InterceptCustomPathRequest(request, out WebResourceResponse webResourceResponse);
                if (intercept)
                {
                    return webResourceResponse;
                }

                return WebViewClient.ShouldInterceptRequest(view, request);
            }

            public override void OnPageFinished(Android.Webkit.WebView view, string url)
                => WebViewClient.OnPageFinished(view, url);

            protected override void Dispose(bool disposing)
            {
                if (!disposing)
                    return;

                WebViewClient.Dispose();
            }

            private static bool InterceptCustomPathRequest(IWebResourceRequest request, out WebResourceResponse webResourceResponse)
            {
                webResourceResponse = null;

                var uri = request.Url.ToString();
                if (!Intercept(uri, out string path))
                {
                    return false;
                }

                if (!File.Exists(path))
                {
                    return false;
                }

                webResourceResponse = CreateWebResourceResponse(request, path);
                return true;
            }

            private static WebResourceResponse CreateWebResourceResponse(IWebResourceRequest request, string path)
            {
                string contentType = StaticContentProvider.GetResponseContentTypeOrDefault(path);
                var headers = StaticContentProvider.GetResponseHeaders(contentType);
                FileStream stream = File.OpenRead(path);
                var length = stream.Length;
                long rangeStart = 0;
                long rangeEnd = length - 1;

                string encoding = "UTF-8";
                int stateCode = 200;
                string reasonPhrase = "OK";

                //適用于音頻視頻文件資源的響應(yīng)
                bool partial = request.RequestHeaders.TryGetValue("Range", out string rangeString);
                if (partial)
                {
                    //206,可斷點續(xù)傳
                    stateCode = 206;
                    reasonPhrase = "Partial Content";

                    var ranges = rangeString.Split('=');
                    if (ranges.Length > 1 && !string.IsNullOrEmpty(ranges[1]))
                    {
                        string[] rangeDatas = ranges[1].Split("-");
                        rangeStart = Convert.ToInt64(rangeDatas[0]);
                        if (rangeDatas.Length > 1 && !string.IsNullOrEmpty(rangeDatas[1]))
                        {
                            rangeEnd = Convert.ToInt64(rangeDatas[1]);
                        }
                    }

                    headers.Add("Accept-Ranges", "bytes");
                    headers.Add("Content-Range", $"bytes {rangeStart}-{rangeEnd}/{length}");
                }

                //這一行刪去似乎也不影響
                headers.Add("Content-Length", (rangeEnd - rangeStart + 1).ToString());

                var response = new WebResourceResponse(contentType, encoding, stateCode, reasonPhrase, headers, stream);
                return response;
            }

        }
    }
}


iOS / Mac

iOS / Mac中我們要替換Maui對于app://自定義協(xié)議的注冊,但是很多類、方法、屬性、字段是不公開的,也就是internal和private,所以我們的代碼用了很多反射。

添加 MauiBlazorWebViewHandler.MaciOS.cs
代碼如下

using Foundation;
using Microsoft.AspNetCore.Components.WebView;
using Microsoft.AspNetCore.Components.WebView.Maui;
using Microsoft.Extensions.Logging;
using MauiBlazorLocalMediaFile.Utilities;
using System.Globalization;
using System.Reflection;
using System.Runtime.Versioning;
using UIKit;
using WebKit;
using RectangleF = CoreGraphics.CGRect;

namespace MauiBlazorLocalMediaFile
{
#nullable disable
    public partial class MauiBlazorWebViewHandler
    {
        private BlazorWebViewHandlerReflection _base;

        private BlazorWebViewHandlerReflection Base => _base ??= new(this);

        [SupportedOSPlatform("ios11.0")]
        protected override WKWebView CreatePlatformView()
        {
            Base.LoggerCreatingWebKitWKWebView();

            var config = new WKWebViewConfiguration();

            // By default, setting inline media playback to allowed, including autoplay
            // and picture in picture, since these things MUST be set during the webview
            // creation, and have no effect if set afterwards.
            // A custom handler factory delegate could be set to disable these defaults
            // but if we do not set them here, they cannot be changed once the
            // handler's platform view is created, so erring on the side of wanting this
            // capability by default.
            if (OperatingSystem.IsMacCatalystVersionAtLeast(10) || OperatingSystem.IsIOSVersionAtLeast(10))
            {
                config.AllowsPictureInPictureMediaPlayback = true;
                config.AllowsInlineMediaPlayback = true;
                config.MediaTypesRequiringUserActionForPlayback = WKAudiovisualMediaTypes.None;
            }

            VirtualView.BlazorWebViewInitializing(new BlazorWebViewInitializingEventArgs()
            {
                Configuration = config
            });

            // Legacy Developer Extras setting.
            config.Preferences.SetValueForKey(NSObject.FromObject(Base.DeveloperToolsEnabled), new NSString("developerExtrasEnabled"));

            config.UserContentController.AddScriptMessageHandler(Base.CreateWebViewScriptMessageHandler(), "webwindowinterop");
            config.UserContentController.AddUserScript(new WKUserScript(
                new NSString(Base.BlazorInitScript), WKUserScriptInjectionTime.AtDocumentEnd, true));

            // iOS WKWebView doesn't allow handling 'http'/'https' schemes, so we use the fake 'app' scheme
            config.SetUrlSchemeHandler(new SchemeHandler(this), urlScheme: "app");

            var webview = new WKWebView(RectangleF.Empty, config)
            {
                BackgroundColor = UIColor.Clear,
                AutosizesSubviews = true
            };

            if (OperatingSystem.IsIOSVersionAtLeast(16, 4) || OperatingSystem.IsMacCatalystVersionAtLeast(13, 3))
            {
                // Enable Developer Extras for Catalyst/iOS builds for 16.4+
                webview.SetValueForKey(NSObject.FromObject(Base.DeveloperToolsEnabled), new NSString("inspectable"));
            }

            VirtualView.BlazorWebViewInitialized(Base.CreateBlazorWebViewInitializedEventArgs(webview));

            Base.LoggerCreatedWebKitWKWebView();

            return webview;
        }

        private class SchemeHandler : NSObject, IWKUrlSchemeHandler
        {
            private readonly MauiBlazorWebViewHandler _webViewHandler;

            public SchemeHandler(MauiBlazorWebViewHandler webViewHandler)
            {
                _webViewHandler = webViewHandler;
            }

            [Export("webView:startURLSchemeTask:")]
            [SupportedOSPlatform("ios11.0")]
            public void StartUrlSchemeTask(WKWebView webView, IWKUrlSchemeTask urlSchemeTask)
            {
                var intercept = InterceptCustomPathRequest(urlSchemeTask);
                if (intercept)
                {
                    return;
                }

                var responseBytes = GetResponseBytes(urlSchemeTask.Request.Url?.AbsoluteString ?? "", out var contentType, statusCode: out var statusCode);
                if (statusCode == 200)
                {
                    using (var dic = new NSMutableDictionary<NSString, NSString>())
                    {
                        dic.Add((NSString)"Content-Length", (NSString)(responseBytes.Length.ToString(CultureInfo.InvariantCulture)));
                        dic.Add((NSString)"Content-Type", (NSString)contentType);
                        // Disable local caching. This will prevent user scripts from executing correctly.
                        dic.Add((NSString)"Cache-Control", (NSString)"no-cache, max-age=0, must-revalidate, no-store");
                        if (urlSchemeTask.Request.Url != null)
                        {
                            using var response = new NSHttpUrlResponse(urlSchemeTask.Request.Url, statusCode, "HTTP/1.1", dic);
                            urlSchemeTask.DidReceiveResponse(response);
                        }

                    }
                    urlSchemeTask.DidReceiveData(NSData.FromArray(responseBytes));
                    urlSchemeTask.DidFinish();
                }
            }

            private byte[] GetResponseBytes(string? url, out string contentType, out int statusCode)
            {
                var allowFallbackOnHostPage = _webViewHandler.Base.IsBaseOfPage(_webViewHandler.Base.AppOriginUri, url);
                url = _webViewHandler.Base.QueryStringHelperRemovePossibleQueryString(url);

                _webViewHandler.Base.LoggerHandlingWebRequest(url);

                if (_webViewHandler.Base.TryGetResponseContentInternal(url, allowFallbackOnHostPage, out statusCode, out var statusMessage, out var content, out var headers))
                {
                    statusCode = 200;
                    using var ms = new MemoryStream();

                    content.CopyTo(ms);
                    content.Dispose();

                    contentType = headers["Content-Type"];

                    _webViewHandler?.Base.LoggerResponseContentBeingSent(url, statusCode);

                    return ms.ToArray();
                }
                else
                {
                    _webViewHandler?.Base.LoggerReponseContentNotFound(url);

                    statusCode = 404;
                    contentType = string.Empty;
                    return Array.Empty<byte>();
                }
            }

            [Export("webView:stopURLSchemeTask:")]
            public void StopUrlSchemeTask(WKWebView webView, IWKUrlSchemeTask urlSchemeTask)
            {
            }

            private static bool InterceptCustomPathRequest(IWKUrlSchemeTask urlSchemeTask)
            {
                var uri = urlSchemeTask.Request.Url.ToString();
                if (uri == null)
                {
                    return false;
                }

                if (!Intercept(uri, out string path))
                {
                    return false;
                }

                if (!File.Exists(path))
                {
                    return false;
                }

                long length = new FileInfo(path).Length;
                string contentType = StaticContentProvider.GetResponseContentTypeOrDefault(path);
                using (var dic = new NSMutableDictionary<NSString, NSString>())
                {
                    dic.Add((NSString)"Content-Length", (NSString)(length.ToString(CultureInfo.InvariantCulture)));
                    dic.Add((NSString)"Content-Type", (NSString)contentType);
                    // Disable local caching. This will prevent user scripts from executing correctly.
                    dic.Add((NSString)"Cache-Control", (NSString)"no-cache, max-age=0, must-revalidate, no-store");
                    using var response = new NSHttpUrlResponse(urlSchemeTask.Request.Url, 200, "HTTP/1.1", dic);
                    urlSchemeTask.DidReceiveResponse(response);
                }

                urlSchemeTask.DidReceiveData(NSData.FromFile(path));
                urlSchemeTask.DidFinish();
                return true;
            }
        }
    }

    public class BlazorWebViewHandlerReflection
    {
        public BlazorWebViewHandlerReflection(BlazorWebViewHandler blazorWebViewHandler)
        {
            _blazorWebViewHandler = blazorWebViewHandler;
            _logger = new(() =>
            {
                var property = Type.GetProperty("Logger", BindingFlags.NonPublic | BindingFlags.Instance);
                return (ILogger)property?.GetValue(_blazorWebViewHandler);
            });
            _blazorInitScript = new(() =>
            {
                var property = Type.GetField("BlazorInitScript", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static);
                return (string)property?.GetValue(_blazorWebViewHandler);
            });
            _appOriginUri = new(() =>
            {
                var property = Type.GetField("AppOriginUri", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static);
                return (Uri)property?.GetValue(_blazorWebViewHandler);
            });
        }

        private readonly BlazorWebViewHandler _blazorWebViewHandler;

        private static readonly Type Type = typeof(BlazorWebViewHandler);

        private static readonly Assembly Assembly = Type.Assembly;

        private static readonly Type TypeLog = Assembly.GetType("Microsoft.AspNetCore.Components.WebView.Log")!;

        private readonly Lazy<ILogger> _logger;

        private readonly Lazy<string> _blazorInitScript;

        private readonly Lazy<Uri> _appOriginUri;

        private object WebviewManager;

        private MethodInfo MethodTryGetResponseContentInternal;

        private MethodInfo MethodIsBaseOfPage;

        private MethodInfo MethodQueryStringHelperRemovePossibleQueryString;

        public ILogger Logger => _logger.Value;

        public string BlazorInitScript => _blazorInitScript.Value;

        public Uri AppOriginUri => _appOriginUri.Value;

        public bool DeveloperToolsEnabled => GetDeveloperToolsEnabled();

        public void LoggerCreatingWebKitWKWebView()
        {
            var method = TypeLog.GetMethod("CreatingWebKitWKWebView");
            method?.Invoke(null, new object[] { Logger });
        }

        public void LoggerCreatedWebKitWKWebView()
        {
            var method = TypeLog.GetMethod("CreatedWebKitWKWebView");
            method?.Invoke(null, new object[] { Logger });
        }

        public void LoggerHandlingWebRequest(string url)
        {
            var method = TypeLog.GetMethod("HandlingWebRequest");
            method?.Invoke(null, new object[] { Logger, url });
        }

        public void LoggerResponseContentBeingSent(string url, int statusCode)
        {
            var method = TypeLog.GetMethod("ResponseContentBeingSent");
            method?.Invoke(null, new object[] { Logger, url, statusCode });
        }

        public void LoggerReponseContentNotFound(string url)
        {
            var method = TypeLog.GetMethod("ReponseContentNotFound");
            method?.Invoke(null, new object[] { Logger, url });
        }

        private bool GetDeveloperToolsEnabled()
        {
            var PropertyDeveloperTools = Type.GetProperty("DeveloperTools", BindingFlags.NonPublic | BindingFlags.Instance);
            var DeveloperTools = PropertyDeveloperTools.GetValue(_blazorWebViewHandler);

            var type = DeveloperTools.GetType();
            var Enabled = type.GetProperty("Enabled", BindingFlags.Public | BindingFlags.Instance);
            return (bool)Enabled?.GetValue(DeveloperTools);
        }

        public IWKScriptMessageHandler CreateWebViewScriptMessageHandler()
        {
            Type webViewScriptMessageHandlerType = Type.GetNestedType("WebViewScriptMessageHandler", BindingFlags.NonPublic);

            if (webViewScriptMessageHandlerType != null)
            {
                // 獲取 MessageReceived 方法信息
                MethodInfo messageReceivedMethod = Type.GetMethod("MessageReceived", BindingFlags.Instance | BindingFlags.NonPublic);

                if (messageReceivedMethod != null)
                {
                    // 創(chuàng)建 WebViewScriptMessageHandler 實例
                    object webViewScriptMessageHandlerInstance = Activator.CreateInstance(webViewScriptMessageHandlerType, new object[] { Delegate.CreateDelegate(typeof(Action<Uri, string>), _blazorWebViewHandler, messageReceivedMethod) });
                    return (IWKScriptMessageHandler)webViewScriptMessageHandlerInstance;
                }
            }

            return null;
        }

        public BlazorWebViewInitializedEventArgs CreateBlazorWebViewInitializedEventArgs(WKWebView wKWebView)
        {
            var blazorWebViewInitializedEventArgs = new BlazorWebViewInitializedEventArgs();
            PropertyInfo property = typeof(BlazorWebViewInitializedEventArgs).GetProperty("WebView", BindingFlags.Public | BindingFlags.Instance);
            property.SetValue(blazorWebViewInitializedEventArgs, wKWebView);
            return blazorWebViewInitializedEventArgs;
        }

        public bool TryGetResponseContentInternal(string uri, bool allowFallbackOnHostPage, out int statusCode, out string statusMessage, out Stream content, out IDictionary<string, string> headers)
        {
            if (MethodTryGetResponseContentInternal == null)
            {
                var Field_webviewManager = Type.GetField("_webviewManager", BindingFlags.NonPublic | BindingFlags.Instance);
                WebviewManager = Field_webviewManager.GetValue(_blazorWebViewHandler);

                MethodTryGetResponseContentInternal = WebviewManager.GetType().GetMethod("TryGetResponseContentInternal", BindingFlags.NonPublic | BindingFlags.Instance);
            }
            // 定義參數(shù)
            object[] parameters = new object[] { uri, allowFallbackOnHostPage, 0, null, null, null };

            bool result = (bool)MethodTryGetResponseContentInternal.Invoke(WebviewManager, parameters);

            // 獲取返回值和輸出參數(shù)
            statusCode = (int)parameters[2];
            statusMessage = (string)parameters[3];
            content = (Stream)parameters[4];
            headers = (IDictionary<string, string>)parameters[5];
            return result;
        }

        public bool IsBaseOfPage(Uri baseUri, string? uriString)
        {
            if (MethodIsBaseOfPage == null)
            {
                var type = Assembly.GetType("Microsoft.AspNetCore.Components.WebView.Maui.UriExtensions")!;
                MethodIsBaseOfPage = type.GetMethod("IsBaseOfPage", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static);
            }

            return (bool)MethodIsBaseOfPage.Invoke(null, new object[] { baseUri, uriString });
        }

        public string QueryStringHelperRemovePossibleQueryString(string? url)
        {
            if (MethodQueryStringHelperRemovePossibleQueryString == null)
            {
                var type = Assembly.GetType("Microsoft.AspNetCore.Components.WebView.QueryStringHelper")!;
                MethodQueryStringHelperRemovePossibleQueryString = type.GetMethod("RemovePossibleQueryString", BindingFlags.Public | BindingFlags.Static | BindingFlags.Instance);
            }

            return (string)MethodQueryStringHelperRemovePossibleQueryString.Invoke(null, new object[] { url });
        }
    }
}

Windows

添加MauiBlazorWebViewHandler.Windows.cs
代碼如下

using MauiBlazorLocalMediaFile.Utilities;
using Microsoft.Web.WebView2.Core;
using System.Runtime.InteropServices.WindowsRuntime;
using Windows.Storage.Streams;
using WebView2Control = Microsoft.UI.Xaml.Controls.WebView2;

namespace MauiBlazorLocalMediaFile
{
    public partial class MauiBlazorWebViewHandler
    {
        protected override void ConnectHandler(WebView2Control platformView)
        {
            base.ConnectHandler(platformView);
            platformView.CoreWebView2Initialized += CoreWebView2Initialized;
        }

        protected override void DisconnectHandler(WebView2Control platformView)
        {
            platformView.CoreWebView2Initialized -= CoreWebView2Initialized;
            base.DisconnectHandler(platformView);
        }

        private void CoreWebView2Initialized(WebView2Control sender, Microsoft.UI.Xaml.Controls.CoreWebView2InitializedEventArgs args)
        {
            var webview2 = sender.CoreWebView2;
            webview2.WebResourceRequested += WebView2WebResourceRequested;
        }

        async void WebView2WebResourceRequested(CoreWebView2 webview2, CoreWebView2WebResourceRequestedEventArgs args)
        {
            await InterceptCustomPathRequest(webview2, args);
        }

        static async Task<bool> InterceptCustomPathRequest(CoreWebView2 webview2, CoreWebView2WebResourceRequestedEventArgs args)
        {
            string uri = args.Request.Uri;
            if (!Intercept(uri, out string filePath))
            {
                return false;
            }

            if (File.Exists(filePath))
            {
                args.Response = await CreateWebResourceResponse(webview2, args, filePath);
            }
            else
            {
                args.Response = webview2.Environment.CreateWebResourceResponse(null, 404, "Not Found", string.Empty);
            }

            return true;

            static string GetHeaderString(IDictionary<string, string> headers) =>
                string.Join(Environment.NewLine, headers.Select(kvp => $"{kvp.Key}: {kvp.Value}"));

            static async Task<CoreWebView2WebResourceResponse> CreateWebResourceResponse(CoreWebView2 webview2, CoreWebView2WebResourceRequestedEventArgs args, string filePath)
            {
                var contentType = StaticContentProvider.GetResponseContentTypeOrDefault(filePath);
                var headers = StaticContentProvider.GetResponseHeaders(contentType);
                using var contentStream = File.OpenRead(filePath);
                var length = contentStream.Length;
                long rangeStart = 0;
                long rangeEnd = length - 1;

                int statusCode = 200;
                string reasonPhrase = "OK";

                //適用于音頻視頻文件資源的響應(yīng)
                bool partial = args.Request.Headers.Contains("Range");
                if (partial)
                {
                    statusCode = 206;
                    reasonPhrase = "Partial Content";

                    var rangeString = args.Request.Headers.GetHeader("Range");
                    var ranges = rangeString.Split('=');
                    if (ranges.Length > 1 && !string.IsNullOrEmpty(ranges[1]))
                    {
                        string[] rangeDatas = ranges[1].Split("-");
                        rangeStart = Convert.ToInt64(rangeDatas[0]);
                        if (rangeDatas.Length > 1 && !string.IsNullOrEmpty(rangeDatas[1]))
                        {
                            rangeEnd = Convert.ToInt64(rangeDatas[1]);
                        }
                        else
                        {
                            //每次加載4Mb,不能設(shè)置太多
                            rangeEnd = Math.Min(rangeEnd, rangeStart + 4 * 1024 * 1024);
                        }
                    }

                    headers.Add("Accept-Ranges", "bytes");
                    headers.Add("Content-Range", $"bytes {rangeStart}-{rangeEnd}/{length}");
                }

                headers.Add("Content-Length", (rangeEnd - rangeStart + 1).ToString());
                var headerString = GetHeaderString(headers);
                IRandomAccessStream stream = await ReadStreamRange(contentStream, rangeStart, rangeEnd);
                return webview2.Environment.CreateWebResourceResponse(stream, statusCode, reasonPhrase, headerString);
            }

            static async Task<IRandomAccessStream> ReadStreamRange(Stream contentStream, long start, long end)
            {
                long length = end - start + 1;
                contentStream.Position = start;

                using var memoryStream = new MemoryStream();

                StreamCopy(contentStream, memoryStream, length);
                // 將內(nèi)存流的位置重置為起始位置
                memoryStream.Seek(0, SeekOrigin.Begin);

                var randomAccessStream = new InMemoryRandomAccessStream();
                await randomAccessStream.WriteAsync(memoryStream.GetWindowsRuntimeBuffer());

                return randomAccessStream;
            }

            // 輔助方法,用于限制StreamCopy復(fù)制的數(shù)據(jù)長度
            static void StreamCopy(Stream source, Stream destination, long length)
            {
                //緩沖區(qū)設(shè)為1Mb,應(yīng)該是夠了
                byte[] buffer = new byte[1024 * 1024];
                int bytesRead;

                while (length > 0 && (bytesRead = source.Read(buffer, 0, (int)Math.Min(buffer.Length, length))) > 0)
                {
                    destination.Write(buffer, 0, bytesRead);
                    length -= bytesRead;
                }
            }
        }
    }
}


在MauiProgram.cs中添加

添加在builder.Services.AddMauiBlazorWebView();的下面

builder.Services.ConfigureMauiHandlers(delegate (IMauiHandlersCollection handlers)
{
    handlers.AddHandler<IBlazorWebView>((IServiceProvider _) => new MauiBlazorWebViewHandler());
});

試驗一下

添加一個復(fù)制文件的靜態(tài)類

簡單寫一個方法,用于把選中的文件復(fù)制到指定目錄,并且返回所需要的url相對路徑
為了防止文件被重復(fù)復(fù)制,我們以md5作為文件名

添加文件夾Utilities/File,在里面添加一個靜態(tài)類MediaResourceFile.cs
代碼如下

using System.Security.Cryptography;

namespace MauiBlazorLocalMediaFile.Utilities
{
    public static class MediaResourceFile
    {
        public static async Task<string?> CreateMediaResourceFileAsync(string targetDirectoryPath, string? sourceFilePath)
        {
            if (string.IsNullOrEmpty(sourceFilePath))
            {
                return null;
            }

            using Stream stream = File.OpenRead(sourceFilePath);
            //新的文件以文件的md5為文件名,確保文件不會重復(fù)存在
            //獲取文件的md5有一點耗時,暫時沒想到更好的方案
            var fn = stream.CreateMD5() + Path.GetExtension(sourceFilePath);
            var targetFilePath = Path.Combine(targetDirectoryPath, fn);
            //如果文件存在就不用復(fù)制了
            if (!File.Exists(targetFilePath))
            {
                if (sourceFilePath.StartsWith(FileSystem.CacheDirectory))
                {
                    stream.Close();
                    await FileMoveAsync(sourceFilePath, targetFilePath);
                }
                else
                {
                    //將流的位置重置為起始位置
                    stream.Seek(0, SeekOrigin.Begin);
                    await FileCopyAsync(targetFilePath, stream);
                }
            }

            return MauiBlazorWebViewHandler.FilePathToUrlRelativePath(targetFilePath);
        }

        private static async Task FileCopyAsync(string targetFilePath, Stream sourceStream)
        {
            CreateFileDirectory(targetFilePath);

            using (FileStream localFileStream = File.OpenWrite(targetFilePath))
            {
                await sourceStream.CopyToAsync(localFileStream, 1024 * 1024);
            };
        }

        private static Task FileMoveAsync(string sourceFilePath, string targetFilePath)
        {
            CreateFileDirectory(targetFilePath);
            File.Move(sourceFilePath, targetFilePath);
            return Task.CompletedTask;
        }

        private static void CreateFileDirectory(string filePath)
        {
            string? directoryPath = Path.GetDirectoryName(filePath);
            if (!Directory.Exists(directoryPath))
            {
                Directory.CreateDirectory(directoryPath!);
            }
        }

        private static string CreateMD5(this Stream stream, int bufferSize = 1024 * 1024)
        {
            using MD5 md5 = MD5.Create();
            byte[] buffer = new byte[bufferSize];
            int bytesRead;

            while ((bytesRead = stream.Read(buffer, 0, buffer.Length)) > 0)
            {
                md5.TransformBlock(buffer, 0, bytesRead, buffer, 0);
            }

            md5.TransformFinalBlock(buffer, 0, 0);
            
            byte[] hash = md5.Hash ?? [];
            return BitConverter.ToString(hash).Replace("-", "").ToLower();
        }
    }
}


寫一個選中視頻文件并顯示的示例頁面
@page "/video"
@using MauiBlazorLocalMediaFile.Utilities

<h1>Video</h1>

<video src="@Src" controls style="max-width:100%;"></video>

<div style="word-break: break-all;">Src="@Src"</div>

<button class="btn btn-primary" @onclick="()=>Pick(true)">選中并復(fù)制到AppDataDirectory</button>
<button class="btn btn-primary" @onclick="()=>Pick(false)">選中不復(fù)制(僅限Windows)</button>

@code {
    private string? Src;

    private async void Pick(bool copy)
    {
#if !WINDOWS
        if (!copy)
        {
            return;
        }
#endif

        var result = await MediaPicker.Default.PickVideoAsync();
        var path = result?.FullPath;
        if (path is null)
        {
            return;
        }

        if (copy)
        {
            var targetDirectoryPath = Path.Combine(FileSystem.AppDataDirectory, "Video");
            Src = await MediaResourceFile.CreateMediaResourceFileAsync(targetDirectoryPath, path);
        }
        else
        {
            Src = MauiBlazorWebViewHandler.FilePathToUrlRelativePath(path);
        }

        await InvokeAsync(StateHasChanged);
    }
}


截圖

用筆者比較喜歡的動畫電影《魁拔》作為視頻文件,大約500MB多一些

Windows

Android

iOS / Mac選中視頻會被壓縮,特別慢,所以就用一個短的視頻了

iOS

Mac

后來補的一張音頻的截圖,用的原始路徑

后記

這篇文章改了又改,總覺得有不妥之處。實在改不動了,就這么地吧,可能寫的還是不夠詳細。

筆者水平有限,性能上可能還存在優(yōu)化的空間,希望各位大佬不吝賜教,提出寶貴意見

源碼

本文中的例子的源碼放到 Github 和 Gitee 了

有需要的可以去看一下

Github: https://github.com/Yu-Core/MauiBlazorLocalMediaFile

Gitee: https://gitee.com/Yu-core/MauiBlazorLocalMediaFile

總結(jié)

以上是生活随笔為你收集整理的MAUI Blazor 如何通过url使用本地文件的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

伊人六月 | 国产在线资源 | 国产亚洲精品电影 | 国产中文字幕视频在线观看 | 一区二区中文字幕在线 | 日日夜操 | 精品国产综合区久久久久久 | 国产资源在线视频 | 国产精品成人一区二区三区 | 国产一区在线视频 | 黄色三级免费网址 | 久久国产精品成人免费浪潮 | 婷婷丁香在线观看 | 在线观看视频免费播放 | 最新日韩在线观看 | 久草在线免费播放 | 国产一区免费在线 | 中文字幕在线国产精品 | 一区二区三区四区精品 | 福利视频入口 | 精品一区免费 | 国产中文字幕av | 三级在线播放视频 | 亚洲va欧美va人人爽春色影视 | 人人爱爱 | 日韩中文字幕免费在线播放 | 久久9精品| 91麻豆精品国产91久久久久 | www.888av| 久久精久久精 | 黄色成人影视 | www.午夜视频 | 国产成人a亚洲精品 | 欧美一级日韩三级 | 午夜精品福利一区二区三区蜜桃 | 在线观看黄色av | 日韩va欧美va亚洲va久久 | 国产中年夫妇高潮精品视频 | 中文字幕一区二区三区在线视频 | 久久激五月天综合精品 | 成人av在线影视 | 在线成人国产 | 久久精品电影网 | 日本久久高清视频 | 久久久午夜精品福利内容 | 天天综合成人 | 国产精品久久久久av | 久久久久久久久久影视 | 伊人导航| 亚洲人成人99网站 | 91在线免费看片 | 国产精品视频app | 国产专区精品视频 | 91看片淫黄大片一级在线观看 | 国产一级91 | 色婷婷激情五月 | 成人免费观看大片 | 中文字幕在线观看视频一区二区三区 | 中文字幕一区二区三区乱码在线 | 美女视频a美女大全免费下载蜜臀 | 国产高清久久久 | 在线亚洲免费视频 | 午夜三级理论 | 最近2019年日本中文免费字幕 | 丁香九月激情 | 操操操夜夜操 | 欧美小视频在线观看 | 在线亚洲人成电影网站色www | 在线播放一区二区三区 | 日韩中文字幕亚洲一区二区va在线 | 欧美午夜精品久久久久久孕妇 | 丁香网五月天 | 18久久久久 | 91在线小视频| 日韩影视精品 | 日韩精品免费在线 | 久久婷婷国产色一区二区三区 | 久99精品 | 亚洲精品乱码久久久久 | 五月婷婷丁香六月 | 成人黄色在线看 | 欧美日韩高清不卡 | 久久久久久久久久久久电影 | 亚洲精品一区二区三区新线路 | 欧美肥妇free | 成年人免费观看国产 | 免费观看一级成人毛片 | 中文字幕在线视频一区 | 人成电影网 | 日韩在线视频看看 | 久久深夜 | 超碰在线观看av.com | 国产精品久久久999 国产91九色视频 | 亚洲人人网 | 婷婷久久五月天 | 国产一级电影在线 | 亚州欧美视频 | 在线观看视频国产 | av丝袜制服 | 亚洲精品国产欧美在线观看 | 在线观看国产区 | av免费看电影 | 天天狠狠操 | 国产福利av| 日韩黄色网络 | 69国产精品视频免费观看 | 国产91成人| 亚洲mv大片欧洲mv大片免费 | 精品国产一区二区三区日日嗨 | 中文字幕在线看视频 | 亚洲色图色| 色久五月| 国产精品video | 国产中文| 色婷婷天天干 | 91桃色在线免费观看 | 人人澡人人添人人爽一区二区 | 亚洲精品乱码久久久久久蜜桃91 | 人人插人人射 | 伊人狠狠干 | 成人免费xyz网站 | 国产无遮挡猛进猛出免费软件 | 日韩视频在线不卡 | 韩日精品在线 | 亚洲国产小视频在线观看 | 免费精品视频在线观看 | www.天天色.com | 亚洲黄色成人 | 国产手机在线观看视频 | 亚洲综合最新在线 | 五月天天在线 | 国产在线观看中文字幕 | 亚洲一区二区三区精品在线观看 | 精品久久久久久亚洲综合网站 | 一区二区三区免费在线播放 | 国产色区 | 久久私人影院 | 国产一区二区不卡视频 | 久久精品精品电影网 | 婷婷5月色| 国产午夜精品一区二区三区欧美 | 激情视频综合网 | 国产一区二区在线播放视频 | 波多野结衣精品视频 | 久草在线免费播放 | 在线免费观看不卡av | 成人观看视频 | 免费在线观看成人av | 日韩欧美高清不卡 | 国产日韩在线观看一区 | 操操综合网 | 99色人| 超碰在线99| 精品国产视频在线 | 亚洲成人精品影院 | 亚洲精品国产精品国自产 | 久久在线看 | 国产白浆在线观看 | 美女视频黄频大全免费 | 国产专区在线 | 免费日韩 精品中文字幕视频在线 | 国产一区视频在线 | 在线小视频国产 | 日韩三级.com | 国产99久久久精品视频 | 亚洲最新在线视频 | 国产精品乱码高清在线看 | 日本免费一二三区 | 成人三级黄色 | 九九在线高清精品视频 | 国产手机在线精品 | 国产看片免费 | 五月婷婷丁香激情 | 好看av在线 | 色94色欧美 | 国产精品手机在线 | 摸bbb搡bbb搡bbbb| 国产成人精品久久久久 | 91精品国产91久久久久久三级 | 国产精品久久99综合免费观看尤物 | 日韩91在线 | 日韩一区二区三免费高清在线观看 | 日韩精品免费在线观看 | 激情网色 | 麻豆影视网站 | 国产私拍在线 | 国产夫妻性生活自拍 | 亚洲欧美日韩精品久久奇米一区 | 天天爽天天碰狠狠添 | 免费视频a | 国产免费不卡av | 91看片在线 | 日韩久久久久久久久 | 夜夜爽www | 国产原创在线视频 | 久久爱www.| 99久久精品国产免费看不卡 | 黄色资源在线 | 色综合久久综合网 | 一区二区中文字幕在线观看 | 国产超碰97 | 91久久精品日日躁夜夜躁国产 | 中文字幕日韩免费视频 | 天天艹天天 | www.狠狠色 | 国产成人精品av久久 | 97国产小视频 | 91激情在线视频 | 最新日韩在线 | 欧美日韩性视频 | 日韩精品视频网站 | 亚洲黄色精品 | 亚洲国产午夜 | 日韩精品一区二区三区视频播放 | 高清视频一区二区三区 | 五月综合激情 | 黄色字幕网| 国产色爽 | 91在线视频免费观看 | 狠狠色综合网站久久久久久久 | a国产精品| 欧美日韩国产在线一区 | 日韩av手机在线看 | 国产原创中文在线 | 在线观看欧美成人 | 精品视频| 首页中文字幕 | 狠狠躁夜夜躁人人爽视频 | 亚洲一级片免费观看 | 日日操日日插 | 综合久久五月天 | 午夜精品视频福利 | 精品久久久久久一区二区里番 | 亚州av网站 | 久久久精品国产一区二区三区 | 亚洲香蕉视频 | 黄色app网站在线观看 | 狠狠干网址 | 亚洲一级黄色片 | 久热av| 日韩欧美在线高清 | 日本不卡一区二区 | 免费视频 你懂的 | 特级大胆西西4444www | av黄网站| 亚洲精品乱码久久久久久按摩 | 国产精品va最新国产精品视频 | 一区 二区电影免费在线观看 | 444av| 国产一区麻豆 | 久久久久久美女 | 97福利| 亚洲欧美国产日韩在线观看 | 在线观看 国产 | 日韩视频在线一区 | 在线免费观看欧美日韩 | 国内精品久久久久久久97牛牛 | 精品婷婷 | 久久高清免费视频 | 五月天综合网站 | av丝袜制服 | a级国产乱理论片在线观看 伊人宗合网 | 亚洲va综合va国产va中文 | 精品美女在线视频 | 91麻豆精品国产91久久久使用方法 | 国产精品毛片一区视频播不卡 | 天天射天天做 | 国产精品国产三级国产aⅴ无密码 | 麻豆传媒视频在线播放 | 男女激情网址 | 天天操夜夜干 | 免费看黄色毛片 | av成人动漫在线观看 | 亚洲作爱 | 久久人人97超碰com | 免费色视频网址 | 香蕉久草 | 丁香一区二区 | 玖玖在线观看视频 | 日韩有码欧美 | 最近免费观看的电影完整版 | 成人中文字幕av | 六月丁香激情网 | 国产精品18毛片一区二区 | 国产18精品乱码免费看 | 日韩精品久久久久久久电影竹菊 | 欧美亚洲另类在线视频 | 久久久久免费电影 | 欧美伦理电影一区二区 | 免费a v观看 | 国产精品高清在线观看 | 日日夜夜天天射 | 高清国产一区 | 久久精品79国产精品 | 欧美小视频在线 | 欧美国产91 | 久久精品视频18 | 91人人澡人人爽 | 亚洲一级电影 | 精品国产一区二区在线 | 成人a免费视频 | 国产成人一区二区三区在线观看 | 在线观看91av | 视频在线精品 | 91成人精品一区在线播放 | 福利视频第一页 | 成人午夜影院 | 亚洲性xxxx| 婷婷久草 | 国产高清视频免费 | h网站免费在线观看 | 欧美大片在线看免费观看 | 亚洲精品高清一区二区三区四区 | 婷婷婷国产在线视频 | 久草精品网 | 国产视频一区在线播放 | 日韩av影片在线观看 | 一区二区中文字幕在线观看 | 91丨porny丨九色 | 东方av在线免费观看 | 国产视频在线观看免费 | 国产麻豆视频网站 | 福利视频导航网址 | 不卡的av在线播放 | 欧美激情综合色综合啪啪五月 | 天天操偷偷干 | 国产精品扒开做爽爽的视频 | 色婷婷综合久久久久中文字幕1 | 日韩资源在线播放 | 成人av高清在线观看 | 美女久久网站 | 西西大胆免费视频 | 99久久婷婷国产一区二区三区 | 伊人久久国产精品 | 日韩激情免费视频 | 丁香五月亚洲综合在线 | 综合久久五月天 | av久久在线 | 91精品国产欧美一区二区 | 久草视频在 | 91亚洲欧美激情 | 超碰97中文| 97精品国产97久久久久久久久久久久 | 亚洲国产精品va在线看黑人动漫 | 麻花豆传媒一二三产区 | 国产中文字幕在线免费观看 | 亚洲成a人片在线观看中文 中文字幕在线视频第一页 狠狠色丁香婷婷综合 | 国产精品午夜8888 | 免费在线色视频 | 91看片在线播放 | 国产黄在线看 | 欧美国产精品久久久久久免费 | 日韩av成人免费看 | 亚洲精品玖玖玖av在线看 | 国产精品av在线免费观看 | 国产午夜视频在线观看 | 涩涩网站在线播放 | 成人综合婷婷国产精品久久免费 | 免费高清在线观看成人 | 日批在线看 | 久久公开视频 | 狠狠狠色丁香综合久久天下网 | 狠狠干成人综合网 | 精品国产一区二区三区在线观看 | 精品欧美在线视频 | aav在线 | 一区二区三区高清不卡 | 最近中文字幕大全中文字幕免费 | 久久精品一级片 | 婷婷五天天在线视频 | av电影免费在线看 | 欧美另类tv| 久久精品国产亚洲a | 天天色成人网 | 黄色网www | 欧美日韩亚洲在线 | 久久av免费观看 | 狠狠操狠狠干天天操 | 欧美精品亚洲精品日韩精品 | 久久久久久久国产精品影院 | 久久欧美在线电影 | av一级片网站 | 午夜精品久久久久久久久久久 | 激情综合色综合久久综合 | 久久久久久免费视频 | 免费日韩在线 | www.久久视频 | 97超碰资源网 | 国产黄色在线看 | 五月婷婷在线综合 | 精品国产资源 | 中文字幕在线观看2018 | 色黄视频免费观看 | 国产不卡在线视频 | 久久久久久美女 | 午夜在线免费观看视频 | 毛片网在线 | 日韩动态视频 | 成人免费中文字幕 | 久久大视频 | 亚洲欧美国产精品 | 午夜av免费看 | 少妇精品久久久一区二区免费 | 午夜国产一区二区 | 欧美最爽乱淫视频播放 | 日韩在线观看免费 | 美女网色 | 国产人成看黄久久久久久久久 | 免费在线91| 日韩视频中文字幕在线观看 | 右手影院亚洲欧美 | 五月综合色婷婷 | 麻豆手机在线 | 日韩在线视 | 人人躁| 在线国产精品视频 | 超碰资源在线 | 久久精品视频网 | 亚洲国产视频在线 | 亚洲韩国一区二区三区 | 免费国产在线观看 | 国产无套精品久久久久久 | 涩涩网站在线观看 | 日韩一级黄色av | av网址aaa| 中文字幕高清免费日韩视频在线 | 免费黄色av电影 | 亚洲另类视频在线 | 欧美日韩国产在线一区 | 色就色,综合激情 | 免费观看完整版无人区 | 超碰激情在线 | 日韩素人在线观看 | 天天爱天天操天天干 | 看全黄大色黄大片 | 国产高清不卡一区二区三区 | 国产九九在线 | 日本中文字幕久久 | 福利视频导航网址 | 欧美性猛片 | 久久国产精品色av免费看 | 人人爽人人片 | 免费看黄在线观看 | 久久精品99精品国产香蕉 | 狠狠婷婷| 99视频在线观看视频 | 日本在线视频一区二区三区 | 亚洲激情精品 | 欧美爽爽爽| 在线不卡中文字幕播放 | 精品国产一区二区三区久久久蜜月 | 国产va在线 | 又黄又刺激又爽的视频 | 成年人在线免费视频观看 | 久操伊人 | 毛片精品免费在线观看 | 国产无限资源在线观看 | 在线导航av | 亚洲精品美女久久 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 亚洲狠狠干 | 国产超碰97 | 亚洲在线高清 | 日日碰狠狠躁久久躁综合网 | 日日夜操| 狠狠色丁婷婷日日 | 国产精品一区二区免费视频 | 久久国产精品久久久久 | 在线精品视频免费观看 | 超碰个人在线 | 亚洲人在线 | 看毛片网站 | 一区二区三区免费播放 | 尤物97国产精品久久精品国产 | 国产成人精品亚洲精品 | 午夜婷婷综合 | 三级动态视频在线观看 | 天天天操操操 | 天天夜夜狠狠操 | 蜜臀一区二区三区精品免费视频 | 日日日日干 | 日韩欧美精品在线观看视频 | 国产精品视频app | 久久欧美在线电影 | 九九99| 成人av电影在线 | 色婷婷视频网 | 亚洲精品白浆高清久久久久久 | 日本爱爱免费视频 | 国产福利91精品张津瑜 | 天天天操天天天干 | 欧美最猛性xxxxx免费 | 国产在线高清 | 亚洲国内精品视频 | 亚洲综合激情小说 | 国产精品国产三级国产不产一地 | 高潮久久久久久 | 日韩网站视频 | www亚洲一区 | 伊人中文在线 | 色婷婷色 | 免费进去里的视频 | 免费在线观看日韩欧美 | 69热国产视频 | 99在线精品观看 | 五月激情六月丁香 | 成人在线中文字幕 | 免费观看一级特黄欧美大片 | 人人澡人人舔 | 伊人久久五月天 | 一区精品在线 | 91在线影院| 人操人 | 成年人免费在线观看 | 国产成人精品一区二区三区福利 | 久久精品播放 | 日韩城人在线 | 一区二区视 | 97超级碰碰碰碰久久久久 | 国产精品刺激对白麻豆99 | 久久不色 | 免费一级片视频 | 国产二区视频在线 | 久久一区二 | 亚洲不卡av一区二区三区 | 天天操夜夜操国产精品 | 久久成人人人人精品欧 | 91探花系列在线播放 | 日韩高清成人 | 一本一道久久a久久精品蜜桃 | 日韩二区在线播放 | 亚洲精品国精品久久99热一 | 婷婷在线网 | 亚洲综合精品在线 | 成片免费观看视频 | 午夜久久网 | 亚洲欧洲国产精品 | 亚洲精品在线二区 | 色网站中文字幕 | 国产a网站 | 中文字幕一区二区三区乱码不卡 | 91视频com| 免费看片亚洲 | 免费看黄电影 | 日本丰满少妇免费一区 | 天天操天天拍 | 久久99热精品这里久久精品 | 91九色在线播放 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 亚洲天堂视频在线 | 日本在线精品视频 | 一级性视频 | 91亚瑟视频 | www.夜夜爽 | 99人成在线观看视频 | 亚洲资源 | 91九色网站 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 久热精品国产 | 久久尤物电影视频在线观看 | 午夜精品久久久久久久久久 | 成人蜜桃 | 国产成人久久77777精品 | 久久视了 | 蜜臀av麻豆 | 在线观看成人国产 | 精品国产人成亚洲区 | 激情婷婷综合网 | 欧美日韩电影在线播放 | 天天爱天天爽 | 国产资源中文字幕 | 久久久精选| 波多野结衣小视频 | 国产成人精品综合久久久久99 | 九九色网 | 国产成人av电影在线观看 | 国产亚洲精品成人av久久影院 | 免费在线观看成年人视频 | 精品亚洲视频在线观看 | 亚洲精品国产第一综合99久久 | 在线成人一区二区 | 精品美女久久久久久免费 | 国产一级片在线播放 | 97超碰人人爱 | 麻豆视频免费入口 | av电影免费在线看 | 久久综合影音 | 久久久久国产一区二区三区四区 | 午夜精品一区二区三区在线观看 | 久久久久久久久久久电影 | 三级a视频 | 99久久毛片 | 在线观看国产中文字幕 | 九九免费精品视频在线观看 | 国产男女无遮挡猛进猛出在线观看 | 久久99国产精品二区护士 | 在线视频一区观看 | 一区二区三区在线观看免费 | 国产成人久久精品一区二区三区 | 日韩丝袜 | 亚洲精品91天天久久人人 | 日韩va欧美va亚洲va久久 | 日韩免费一区二区在线观看 | www..com黄色片 | 久久激情五月婷婷 | 黄色特一级片 | 91视频啪| 亚洲不卡123 | 日韩三级在线观看 | 日日夜夜天天综合 | 337p西西人体大胆瓣开下部 | av黄色在线观看 | 女人魂免费观看 | 久久久国产精品一区二区中文 | 天天插视频 | 免费在线观看av的网站 | 一 级 黄 色 片免费看的 | 精品久久久久免费极品大片 | 国产精品白丝jk白祙 | 成人免费色 | 久久久久免费精品国产小说色大师 | 91精品国产九九九久久久亚洲 | 日韩欧美大片免费观看 | 亚洲精品66 | 国产久视频 | 国产成人一区二区三区在线观看 | 国产精品入口麻豆www | 久草| 久久综合久色欧美综合狠狠 | 国产午夜精品福利视频 | 97超碰在线久草超碰在线观看 | 97超碰站| 高潮久久久 | 欧美色综合天天久久综合精品 | 国产亚洲精品久久19p | 天天躁天天躁天天躁婷 | 992tv又爽又黄的免费视频 | 夜夜操天天 | 亚洲欧美视频一区二区三区 | 日本在线观看视频一区 | 看全黄大色黄大片 | 欧美日韩另类在线 | 热热热热热色 | 日韩精品免费在线观看 | 天天插天天狠 | 免费特级黄毛片 | 97人人模人人爽人人少妇 | 国产一区免费视频 | 亚洲激精日韩激精欧美精品 | 黄色网大全 | 国产黄视频在线观看 | 精品国产aⅴ麻豆 | 国产精品一区二区久久精品爱微奶 | 国产精品美 | 久热av | 久久婷婷网 | 亚洲另类在线视频 | 91九色精品国产 | 日韩精品视频在线免费观看 | www日日夜夜 | 亚洲精品免费观看 | 婷婷精品在线视频 | 国产精品videossex国产高清 | 久久精久久精 | 久久久国产视频 | 亚洲va欧洲va国产va不卡 | 中文字幕第一页在线vr | 国产精品久久99综合免费观看尤物 | www.色爱| 五月婷婷六月丁香 | 99久久国产免费看 | 99精品视频免费观看 | 天堂av网站 | 久久九九久久精品 | 中文字幕在线一二 | 国产亚洲一区二区在线观看 | 激情av综合 | 久久精品视频网址 | 国产一区二区三区免费在线 | 日韩精品一区二区免费 | 一区二区久久 | 久久精品视频2 | 精品久久久久国产免费第一页 | 久久国产精品久久久久 | 香蕉视频免费看 | 国产精品毛片一区二区 | 亚洲国产精品小视频 | 五月天堂色 | 日韩影视大全 | 欧美日韩国内在线 | 免费久久99精品国产 | 国产999精品久久久影片官网 | 国产又粗又硬又爽视频 | av国产在线观看 | 中文字幕亚洲在线观看 | 色婷婷骚婷婷 | 97在线看 | 亚洲精品456在线播放第一页 | 伊人影院在线观看 | av先锋中文字幕 | 免费看一级特黄a大片 | 亚洲精品视频免费在线 | 国产在线a不卡 | 天天爱综合 | 99夜色| 最新日韩在线观看视频 | 人人草人人草 | 色偷偷中文字幕 | 亚洲综合色播 | 在线观看免费高清视频大全追剧 | www九九热 | 国内精品久久久 | 狠狠色香婷婷久久亚洲精品 | 激情偷乱人伦小说视频在线观看 | 射久久| 国产99久久九九精品免费 | 99婷婷狠狠成为人免费视频 | 黄色一级动作片 | 超碰电影在线观看 | 一区二区国产精品 | 国产美女免费 | 天天操狠狠干 | 久久久久国产成人精品亚洲午夜 | 三级黄色a | 91av九色| 国偷自产中文字幕亚洲手机在线 | 深爱激情五月综合 | 亚洲精品网站在线 | 热久在线 | 99国产免费网址 | 69国产在线观看 | 国产91丝袜在线播放动漫 | 亚洲狠狠操 | 91片网| 久久99精品久久久久蜜臀 | 美女视频黄免费网站 | 看片黄网站 | 成人毛片在线观看视频 | 五月天色综合 | 日本在线观看黄色 | av在线影视 | 91高清免费看 | 色九九在线 | 最近中文字幕mv免费高清在线 | 日韩1页 | 91在线影视| 成人在线视频免费观看 | 深爱开心激情网 | 免费中文字幕 | 米奇狠狠狠888 | 四虎在线观看网址 | 成人久久18免费网站图片 | 在线观看av免费 | 国产精品一区免费在线观看 | 日韩理论片在线观看 | 九九热国产视频 | 亚洲精品国精品久久99热 | 成人黄色小视频 | 在线小视频国产 | 日本精品视频免费观看 | 国产精品成人一区 | 国产视频精品免费 | 欧美激情综合色综合啪啪五月 | 成人黄色在线 | 99产精品成人啪免费网站 | 国产精品午夜在线 | a级片韩国 | 日本天天色 | 超碰97久久 | 国产精品入口a级 | 91在线中文| 久久美女高清视频 | 国产亚洲精品久久久久久大师 | 国产精品久久久久久一区二区三区 | 亚洲毛片在线观看. | 一级欧美一级日韩 | 国产在线黄 | 久久国产精品二国产精品中国洋人 | 五月综合久久 | 成年人电影免费在线观看 | 在线精品一区二区 | 黄色小说网站在线 | 国产成人黄色在线 | 91传媒免费在线观看 | 国产精品午夜在线观看 | 亚洲aⅴ乱码精品成人区 | 久久好看免费视频 | 91精品免费 | 六月丁香综合网 | 久久都是精品 | 天天色天天操天天爽 | www.xxx.性狂虐 | av超碰免费在线 | 激情五月综合网 | 欧美精品久久天天躁 | 精品一区二区免费视频 | 中文字幕在线日亚洲9 | 亚洲综合网 | 日韩电影中文,亚洲精品乱码 | 久久久亚洲成人 | 欧美做受高潮电影o | 久久综合久久综合久久综合 | 在线视频欧美亚洲 | 色网免费观看 | 激情欧美xxxx | 大胆欧美gogo免费视频一二区 | 精品久久久久久久久久久久久 | 国产视频二 | 在线a人片免费观看视频 | 日韩精品一区二区在线观看 | 天天干天天在线 | 亚洲欧美日韩国产精品一区午夜 | 一二区av| 久久久91精品国产 | 五月婷婷另类国产 | 欧美日韩裸体免费视频 | 播五月婷婷 | 欧美另类高清 videos | 日韩精品一区二区三区免费观看 | 日本久久综合网 | 91自拍视频在线观看 | 国产精品久久久久久麻豆一区 | 国产麻豆精品一区 | 亚洲人成免费 | 国产97免费 | 国产亚洲精品久久久久久久久久 | 国产又黄又爽无遮挡 | 欧美色图亚洲图片 | 日韩av网页 | 国产精品高潮呻吟久久久久 | 欧美精品久久久久久久久久白贞 | 美女视频a美女大全免费下载蜜臀 | 日本aaaa级毛片在线看 | 免费a视频| 2019中文 | 精品国产1区 | 久久综合色影院 | 久久免费试看 | 欧美一级视频免费 | 午夜在线观看一区 | 99在线观看 | av网站有哪些 | 亚洲电影图片小说 | 精品欧美在线视频 | 狠狠躁日日躁夜夜躁av | 亚洲免费观看视频 | 2019久久精品 | 久久草精品 | 福利一区在线 | 黄网av在线 | 中文字幕色综合网 | 伊人激情综合 | 999成人国产 | 国产精品不卡在线播放 | 久久成人一区二区 | 亚洲精品白浆高清久久久久久 | 黄色片免费看 | 成人a在线观看高清电影 | 免费国产一区二区 | 国产精品va最新国产精品视频 | mm1313亚洲精品国产 | 国产美女精品在线 | 久久亚洲精品电影 | 午夜色场 | 久久免费视频网 | 成人免费 在线播放 | 久久伊人精品天天 | 国产小视频免费在线网址 | 91超级碰| 日韩欧美区 | 超碰夜夜 | 看片黄网站 | 一区二区三区四区免费视频 | 欧美日韩高清一区 | 在线观看成人小视频 | 在线 视频 亚洲 | 免费视频成人 | 奇米影视8888在线观看大全免费 | 久久久精品视频网站 | 日韩网站免费观看 | 欧美va天堂在线电影 | 日韩有码在线播放 | 91女神的呻吟细腰翘臀美女 | 欧美在线一二区 | 成人h在线播放 | 免费看片日韩 | a精品视频 | 中文字幕国语官网在线视频 | 国产精品麻豆91 | 五月婷婷六月丁香在线观看 | 亚洲日日夜夜 | 国产91精品在线播放 | 免费国产视频 | 国产婷婷在线观看 | 精品 一区 在线 | 六月激情网 | 在线日韩| 在线免费观看av网站 | 色婷婷中文 | 亚洲aⅴ免费在线观看 | 色噜噜狠狠狠狠色综合久不 | 麻豆国产在线播放 | 国产麻豆电影在线观看 | 69久久99精品久久久久婷婷 | 天天搞夜夜骑 | 亚洲精品乱码久久 | 日韩欧美在线中文字幕 | 99热这里是精品 | 国产精品九九热 | av一区在线播放 | 中文字幕一区在线 | 亚洲综合在线播放 | 国产成人av电影在线 | 午夜神马福利 | 国产成人61精品免费看片 | 麻豆国产在线视频 | 中文字幕中文字幕在线中文字幕三区 | 久草手机视频 | 99在线观看 | 国产69久久久欧美一级 | www狠狠| 免费电影播放 | 国产 在线观看 | 天天舔天天射天天操 | 国产精品久久毛片 | 性色av香蕉一区二区 | 日韩有码网站 | 波多野结衣电影一区二区 | 五月天婷亚洲天综合网鲁鲁鲁 | 国产亚洲精品久久久久动 | 欧美国产精品一区二区 | 五月婷婷一区 | 免费日韩一区二区 | 91大神免费视频 | 精品99在线 | 欧美一二三专区 | 一级免费观看 | 999久久a精品合区久久久 | 黄网站色| 久久99最新地址 | 99久久er热在这里只有精品66 | 婷婷在线免费视频 | 在线韩国电影免费观影完整版 | 久久精品国产精品亚洲精品 | 色婷婷亚洲综合 | 精品视频免费看 | 粉嫩一二三区 | 在线精品视频免费观看 | 97电影在线 | 狠狠狠综合 | www.com黄色| 奇米7777狠狠狠琪琪视频 | 国产精品久久久久av免费 | 99久久精品国产免费看不卡 | 99久久精品国产欧美主题曲 | 国产成人一区二区三区电影 | 亚洲精品在线看 | 综合婷婷久久 | 一区二区三区精品在线视频 | 手机在线欧美 | 亚洲一区免费在线 | 欧美一区二区精美视频 | 国产69精品久久99的直播节目 | 日韩理论电影在线 | 国产亚洲精品综合一区91 | 午夜影院一级片 | 国产视频综合在线 | 亚洲精品中文字幕在线 | 久久久久久亚洲精品 | 在线观看国产亚洲 | 亚洲精品网页 | 欧美一级在线 | 久久成人毛片 | 综合久久影院 | 国产精品欧美日韩在线观看 | 免费午夜网站 | 欧美一二三视频 | 日韩伦理片一区二区三区 | 亚洲国内在线 | 精品亚洲免费 | 天天草天天 | 日韩性片 | 黄色福利视频网站 | 精品视频免费播放 | 丝袜av网站 | 欧美精品做受xxx性少妇 | 亚洲 中文 欧美 日韩vr 在线 | 国产成人专区 | 亚洲精品一区二区三区在线观看 | 丁香六月在线 | 国产成人一区二区三区电影 | 女人18片| 免费视频网 | av一区二区在线观看中文字幕 | 日韩在线观看小视频 | 亚洲香蕉视频 | 国产精品99久久久久久久久久久久 | aav在线|