【WP8.1】WebView笔记
之前在WP8的时候做过WebBrowser相关的笔记,在WP8.1的WebView和WebBrowser有些不一样,在这里做一些笔记
下面分为几个部分
1、禁止缩放
2、JS通知后台C#代码(notify)
3、C#调用JS方法
动态加载JS文件,动态注册事件方法(eval)
4、WebView导航
5、手势(WinJS)
6、常见问题
1、禁用缩放
body {
/* Block area from manipulation actions (zoom, pan) */
touch-action: pan-y;
}
这个值可以禁用掉缩放和横向手势
关于touch-action参见:https://msdn.microsoft.com/en-us/library/windows/apps/xaml/hh767313.aspx?f=255&MSPPError=-2147217396
2、JS通知后台C#代码(notify)
window.external.notify(msg);
注意:这里的方法名是小写的,在WP8上后面的Notify方法的首字母是大小写都可以
3、C#调用JS方法(InvokeScriptAsync)
通过后台代码动态加载css,js,动态绑定事件和方法
3.1、先定义传输的数据格式
/// <summary>
/// JS消息格式
/// </summary>
public class JsInvokeModel
{
[JsonProperty("type")]
public string Type { get; set; } [JsonProperty("content1")]
public string Content1 { get; set; } [JsonProperty("content2")]
public string Content2 { get; set; } [JsonProperty("content3")]
public string Content3 { get; set; }
}
3.2、XAML
<WebView x:Name="WebView" DOMContentLoaded="WebView_OnDOMContentLoaded" ScriptNotify="WebView_OnScriptNotify" />
3.3、下面是事件方法
//DOM树加载完成后执行
private async void WebView_OnDOMContentLoaded(WebView sender, WebViewDOMContentLoadedEventArgs args)
{
//1、动态加载css
var js = @"var myCss = document.createElement(""link"");
myCss.rel = ""stylesheet"";
myCss.type = ""text/css"";
myCss.href = ""ms-appx-web:///Assets/Html/css/default.css"";
document.body.appendChild(myCss);";
await sender.InvokeScriptAsync("eval", new[] { js }); //2、动态加载js库(json2)
js = @"var myScript = document.createElement(""script"");
myScript.type = ""text/javascript"";
myScript.src = ""ms-appx-web:///Scripts/json2.min.js"";
document.body.appendChild(myScript);";
await sender.InvokeScriptAsync("eval", new[] {js}); //3、调用js执行自定义代码(为图片添加点击事件,并通知)
js = @"var imgs = document.getElementsByTagName(""img"");
for (var i = 0, len = imgs.length; i < len; i++) {
imgs[i].onclick = function (e) {
var jsonObj = { type: 'image', content1: this.src };
window.external.notify(JSON.stringify(jsonObj));
};
}";
await sender.InvokeScriptAsync("eval", new[] {js}); //4、动态加载手势
js = @"var myScript = document.createElement(""script"");
myScript.type = ""text/javascript"";
myScript.src = ""ms-appx-web:///Assets/Html/js/gesture.js"";
document.body.appendChild(myScript);
window.external.notify(myScript.src+"""");";
await sender.InvokeScriptAsync("eval", new[] { js }); //5、为body添加手势监听
js = @"var target = document.getElementsByTagName(""body"")[0];
prepareTarget(target, eventListener);";
await sender.InvokeScriptAsync("eval", new[] { js });
}
3.4、Notify监听
private void WebView_OnScriptNotify(object sender, NotifyEventArgs e)
{
//这个事件函数可以监听到JS通知的消息,消息类型为文本
//这里统一消息格式为:JsInvokeModel
var model = JsonConvert.DeserializeObject<JsInvokeModel>(e.Value);
switch (model.Type)
{
case "image":
Info.Text = e.Value;
break;
case "swiperight":
//右滑
Info.Text = e.Value;
break;
case "swipeleft":
//左滑
Info.Text = e.Value;
break;
case "text":
Info.Text = e.Value;
break;
}
}
WebView虽然提供了同步方法InvokeScript,但是在WP8.1没有实现
通过InvokeScriptAsync,可以做更多操作,例如,相信对于更换颜色(夜间模式),修改字体大小等
4、WebView导航
两种方式
//后退
//WebView.GoBack();
await WebView.InvokeScriptAsync("eval", new []{"history.go(-1)"}); //刷新
//WebView.Refresh();
await WebView.InvokeScriptAsync("eval", new[] { "history.go()" }); //前进
//WebView.GoForward();
await WebView.InvokeScriptAsync("eval", new[] { "history.go(1)" });
5、手势
由于WebView的内部结构与WebBrowser不同,WebView无法监听到Manipulation事件
场景:当我们需要在PivotItem中放置WebView的时候,左右滑动无法切换PivotItem,下面通过JS手势监听WebView上面的手势操作,然后传到后台代码进行处理,这里没有做实时处理,只是监听了手势离开时的速度判断左右滑动
5.1、定义手势监听事件方法
var gesture;
//记录手势操作开始位置
var gestureStartX; //触发Id,防止重复触发,触发Id与手势Id
var gestureId = 1;
var lastGestureId = 0; //速度触发
var gestureVector = 1.5; //注册手势事件
function prepareTarget(target, eventListener) {
//var target = document.getElementById(targetId);
target.addEventListener("MSGestureStart", eventListener, false);
target.addEventListener("MSGestureEnd", eventListener, false);
target.addEventListener("MSGestureChange", eventListener, false);
target.addEventListener("MSInertiaStart", eventListener, false);
//target.addEventListener("MSGestureTap", eventListener, false);
//target.addEventListener("MSGestureHold", eventListener, false);
target.addEventListener("pointerdown", onPointDown, false);
target.addEventListener("pointerup", onPointUp, false); gesture = new MSGesture();
gesture.target = target;
} function onPointUp(e) {
//把触发时间参数传到gesture
gesture.addPointer(e.pointerId);
} function onPointDown(e) {
//把触发时间参数传到gesture
gesture.addPointer(e.pointerId);
} //手势事件
//具体的属性参见:https://msdn.microsoft.com/zh-cn/library/ie/hh772076%28v=vs.85%29.aspx
function eventListener(evt) {
var myGesture = evt.gestureObject;
if (evt.type == "MSGestureStart") {
//开始触发,记录初始位置
gestureStartX = evt.clientX;
}
else if (evt.type == "MSInertiaStart") {
if (lastGestureId == gestureId || evt.velocityX == "undefined") {
return;
} else {
//释放时触发惯性事件,判断手势释放时的速度
if (evt.velocityX > gestureVector) {
var jsonObj = { type: "swiperight" };
window.external.notify(JSON.stringify(jsonObj));
lastGestureId = gestureId;
} else if (evt.velocityX < -gestureVector) {
jsonObj = { type: "swipeleft" };
window.external.notify(JSON.stringify(jsonObj));
lastGestureId = gestureId;
}
}
}
else if (evt.type == "MSGestureChange") {
//if (lastGestureId == gestureId) {
// return;
//} else {
// var change = evt.clientX - gestureStartX;
// window.external.notify("clientX:" + change);
//}
} else if (evt.type == "MSGestureEnd") {
//手势结束,Id+1
gestureId = gestureId + 1;
myGesture.reset();
}
}
gesture.js
5.2、在WebView加载完成后,动态加载改JS文件,获取body标签,然后监听事件(参见3.3)
5.3、当事件触发的时候改变Pivot.SelectedIndex,这样就能实现在WebView上滑动切换PivotItem
private void WebView_OnScriptNotify(object sender, NotifyEventArgs e)
{
//这个事件函数可以监听到JS通知的消息,消息类型为文本
//这里统一消息格式为:JsInvokeModel
var model = JsonConvert.DeserializeObject<JsInvokeModel>(e.Value);
switch (model.Type)
{
case "image":
Info.Text = e.Value;
break;
case "swiperight":
Info.Text = e.Value;
if (pivot.Items != null)
{
if (pivot.SelectedIndex > )
{
pivot.SelectedIndex--;
}
else
{
pivot.SelectedIndex = pivot.Items.Count - ;
}
}
break;
case "swipeleft":
Info.Text = e.Value;
if (pivot.Items != null)
{
if (pivot.SelectedIndex < pivot.Items.Count - )
{
pivot.SelectedIndex++;
}
else
{
pivot.SelectedIndex = ;
}
}
break;
case "text":
Info.Text = e.Value;
break;
}
}
6、常见问题
6.1、alert和prompt方法在WebView失效,如果需要,可以使用 window.external.notify('message');然后再后台代码进行处理
6.2、如果是导航到本地路径(ms-appdata:///local/),需要注意
6.2.1、必须要将html需要的资源和html文件放在同一个文件夹下,html的资源路径不管是不是”/”开头,webview都认为是相对路径
6.2.2、如果html中要的本地文件,如果在对应目录中不存在,wp上会直接跳出应用,大概过1分钟左右崩溃,且捕获不到异常。Windows上没有这现象
6.2.3、不支持js的window.exteranl.notify功能
webView1.Navigate(new Uri("ms-appdata:///local/data/html/sample.htm"));
不知道为什么WebView在导航到本地路径的时候 window.external.notify 会失效,如果需要可以考虑使用 内容流的方式加载本地文件
下面做简单的演示
1、定义内容留转换器
public class HtmlStreamUriResolver : IUriToStreamResolver
{
public IAsyncOperation<IInputStream> UriToStreamAsync(Uri uri)
{
string path = uri.AbsolutePath;
return GetContent(path).AsAsyncOperation(); } // 根据 uri 返回对应的内容流
private async Task<IInputStream> GetContent(string path)
{
if (path.StartsWith("http"))
{
// http 方式获取内容数据
var client = new HttpClient();
var response = await client.GetAsync(new Uri(path), HttpCompletionOption.ResponseHeadersRead);
return (await response.Content.ReadAsStreamAsync()).AsInputStream();
}
else if (path.StartsWith("/local/cachedata/html/"))
{
path = string.Format("ms-appdata://{0}", path);
// 获取本地数据
var fileRead = await StorageFile.GetFileFromApplicationUriAsync(new Uri(path));
return await fileRead.OpenAsync(FileAccessMode.Read);
}
else
{
// 获取本地数据
var fileRead = await StorageFile.GetFileFromApplicationUriAsync(new Uri(path, UriKind.Absolute));
return await fileRead.OpenAsync(FileAccessMode.Read);
}
}
}
HtmlStreamUriResolver
2、生成,导航,html中的所有引用都通过HtmlStreamUriResolver返回,具体逻辑在里面处理,通过这样导航的页面可以使用 window.external.notify 方法
//生成内容流
var htmlStreamUriResolver = new HtmlStreamUriResolver();
var uri = webView.BuildLocalStreamUri("tag", "/local/cachedata/html/sample.html");
webView.NavigateToLocalStreamUri(uri, htmlStreamUriResolver);
7、Demo
http://files.cnblogs.com/files/bomo/WebViewDemo.zip
8、参考链接
手势事件参数说明:https://msdn.microsoft.com/zh-cn/library/ie/hh772076%28v=vs.85%29.aspx?f=255&MSPPError=-2147217396
WinJS手势:https://msdn.microsoft.com/zh-cn/library/ie/hh968249(v=vs.85).aspx
Win8.1和WP8.1在UniversalAPP中WebView的研究:http://blog.csdn.net/hzdinglai/article/details/41073739
重新想象 Windows 8.1 Store Apps (81) : WebView:http://www.cnblogs.com/webabcd/p/3803384.html
关于WP8的WebBrowser,请移步:http://www.cnblogs.com/bomo/p/3949994.html
个人能力有限,如果上文有误或者您有更好的实现,可以给我留言
转载请注明出处:http://www.cnblogs.com/bomo/p/4320077.html
【WP8.1】WebView笔记的更多相关文章
- webview笔记
1. 用户上传文件 webChromeClient的onShowFileChooser这个方法,这将打开一个文件选择器,如果要取消这个请求则是调用filePathCallback.onReceiveV ...
- Android WebView 笔记
WebView(网络视图)能载入显示载入网页.将网页内容载入到手机client,它使用了WebKit渲染引擎载入显示网页,实现WebView有下面方法: 首先要实如今手机client显示必需要求注冊一 ...
- Android SDK组件:webview笔记
1.安卓手机中内置了一款webkit内核的浏览器,在SDK中封装为WebView组件. 2.该组件可以在自己的应用程序中显示本地或者Internet上的网页,也可以把它当作一个浏览器来时用. 3.We ...
- <WP8开发学习笔记>获取手机的常用型号(如Lumia920,而非RM-822)
之前WP7时代可以用API获得WP手机的型号如lumia510,但是到了WP8后用APi只能获得硬件版本号了如RM-822,这种型号可以让我们更详细的了解具体的硬件版本,比如国行和港行,设备版本号不一 ...
- 1. WP8.1学习笔记
数据绑定 含义:将对象绑定到控件上 2.基本名词 控件:绑定目标 对象:绑定源(数据源) 控件与对象属性的联系:路径 如何绑定 创建对象,设置控件 在控件需要数据绑定的地方使用拓展语法 <But ...
- 0. WP8.1学习笔记
应用程序生命周期: 运行: 在程序NotRunning状态下点击图标,应用将处于Running状态,这会触发一个Actived事件 挂起: 在程序Running状态下, 点击返回键或win键会触发一个 ...
- wp8.1 学习笔记 001 动态生成图片 并更改图片位置
1.在xaml中划分表格 <Grid Name="gr"> <Grid.ColumnDefinitions> <ColumnDefinition> ...
- <WP8开发学习笔记>ApplicationBar(任务栏)的切换以及“黑条问题”
ApplicationBar(以下简称AppBar)是WP应用相当常见的控件,也很方便.常见的做法是pivot或者panorama的页面切换的时候,AppBar跟随切换对应的按钮或者不显示按钮,如下图 ...
- <WP8开发学习笔记>动态修改启动时导航的第一个页面(如登录前启动页为LoginPage,登录后变为MainPage)
很多时候我们需要在启动程序的时候根据状态改变初始导航页面,比如程序在启动的时候判断用户是否登录, 如果未登录则跳转到LoginPage.xaml否则跳转到MainPage界面. 这时候就要分析程序的启 ...
随机推荐
- Java 堆内存与栈内存异同(Java Heap Memory vs Stack Memory Difference)
--reference Java Heap Memory vs Stack Memory Difference 在数据结构中,堆和栈可以说是两种最基础的数据结构,而Java中的栈内存空间和堆内存空间有 ...
- Windos环境用Nginx配置反向代理和负载均衡
Windos环境用Nginx配置反向代理和负载均衡 引言:在前后端分离架构下,难免会遇到跨域问题.目前的解决方案大致有JSONP,反向代理,CORS这三种方式.JSONP兼容性良好,最大的缺点是只支持 ...
- javascript 笔记!
1.通过javascript向文档中输出文本 document是javascript的内置对象,代表浏览器的文档部分 document.write("Hello Javascript&quo ...
- arcgis api for js入门开发系列六地图分屏对比(含源代码)
上一篇实现了demo的地图标绘模块,本篇新增地图地图分屏对比模块,截图如下(源代码见文章底部): 对效果图的简单介绍一下,在demo只采用了两分屏对比,感兴趣的话,可以在两分屏的基础上拓展,修改css ...
- SAP CRM 将组件整合至导航栏中
到现在,我们已经可以让组件独立地显示.我们只是运行它.让它显示在Web UI中.让我们把组件整合进导航栏,使我们可以在正常登录Web UI时访问它. 步骤一: 为你的UI组件主窗体创建一个内向插件. ...
- ios label 自动计算行高详解
在OC当中自动计算行高主要调用系统的 p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 11.0px Menlo; color: #ffffff } span ...
- 基于jQuery左右滑动切换特效 附源码
分享一款基于脚jQuery左右滑动切换特效.这是一款鼠标点击左右箭头按钮图片滚动切换,鼠标移到图片上显示透明边框特效. 效果图如下: 废话不多说,代码奉上! html代码: <div ...
- ORACLE从共享池删除指定SQL的执行计划
Oracle 11g在DBMS_SHARED_POOL包中引入了一个名为PURGE的新存储过程,用于从对象库缓存中刷新特定对象,例如游标,包,序列,触发器等.也就是说可以删除.清理特定SQL的执行计划 ...
- OpenSUSE下编译安装OpenFoam
在不是Ubuntu系统下安装OpenFoam,需要采用编译安装的方式.以下以OpenSuSE为例进行编译安装. 1 软件包准备 需要下载两个程序包: OpenFOAM-4.x-version-4.1. ...
- 《AngularJS深度剖析与最佳实践》简介
由于年末将至,前阵子一直忙于工作的事务,不得已暂停了微信订阅号的更新,我将会在后续的时间里尽快的继续为大家推送更多的博文.毕竟一个人的力量微薄,精力有限,希望大家能理解,仍然能一如既往的关注和支持sh ...