【uwp】浅谈China Daily 中划词翻译的实现
学习uwp开发也有一段时间了,最近上架了一个小应用(China Daily),现在准备将开发中所学到的一些东西拿出来跟大家分享交流一下。
先给出应用的下载链接:China Daily , 感兴趣的童鞋可以看一看。
废话就扯到这里,接下来,我们来看看这个应用中的划词翻译功能是如何实现的(也顺带谈谈点击正文中的图片显示详情)。
新闻的主体是放在一个WebView里面的,所以,说白了就是解决WebView的问题(JS通知后台C#以及C#调用JS)。
1.XAML
<WebView x:Name="webView"
NavigationCompleted="webView_NavigationCompleted"
ScriptNotify="webView_ScriptNotify"
/>
2.添加对alert的监听
private async void webView_NavigationCompleted(WebView sender, WebViewNavigationCompletedEventArgs args)
{
//检测alert
var inject = WebViewUtil.RiseNotification(); await webView.InvokeScriptAsync("eval", new List<string>() { inject });
}
public static string RiseNotification()
{
string func = @"window.alert = function(arg) {
window.external.notify(arg);
};";
return func;
}
3.监听到alert时的处理
private void webView_ScriptNotify(object sender, NotifyEventArgs e)
{
var data = e.Value.Trim(); if (data.Contains("http://"))
{
// 判断点击的是图片
LoadImage(data);
}
else
{
// 判断点击的是文本
LoadTranslation(data);
}
}
此处的data是WebView通过JS方法传递过来的信息。如果点击的是图片,则传递图片的url;如果点击(选中)的是文本,则把文本传递过来。
当然,WebView会这么机智的将我们想要的信息准确的传递过来吗?答案是肯定的,前提是我们要告诉它应该怎么做。
4.交给WebView的锦囊
4.1 明确任务
China Daily 接口没有直接返回完整的html,但它将主要的内容都交到了我们手里,而我们所需要完成的,就是把这些资源给组合起来,再交给WebView。
4.2 前期准备
public static int GetBaseFontSize() => DeviceUtils.IsMobile ? : ; public static int GetBaseLineHeight() => DeviceUtils.IsMobile ? : ; /// <summary>
/// 内容样式
/// </summary>
/// <returns></returns>
public static string GetContentCSS()
{
string commonStyle = "touch-action: pan-y; font-family:'微软雅黑';" + (DeviceUtils.IsMobile ? "padding:2px 19px 2px 19px;" : "text-align:justify; padding:2px 16px 2px 5px;"); string colorstyle = "background-color:#ffffff; color:#000000;" ; string webStyle = "body{" + commonStyle + colorstyle +
$"font-size:{GetBaseFontSize()}px; line-height:{GetBaseLineHeight()}px" +
"}";
string css = "<style>" + webStyle + "</style>"; return css;
}
/// <summary>
/// JS方法
/// </summary>
/// <returns></returns>
public static string GetContentJS()
{
string js = @"window.lastOriginData = ''; function mouseUp (param) {
if(param == 'y') return;
if (window.getSelection) {
var str = window.getSelection().toString();
window.lastOriginData = str.trim();
alert(str);
}
};
function getImageUrl(url) {
alert(url);
}
window.onload = function () {
var as = document.getElementsByTagName('a');
for (var i = 0; i < as.length; i++) {
as[i].onclick = function (event) {
window.external.notify(JSON.stringify({ 'type': 'HyperLink', 'content': this.href }));
return false;
}
}
};"; return js;
}
GetContentCSS方法只是用于给整段新闻添加一个样式,使布局看起来美观一点,大家不用太在意。
可以关注下的是GetContentJS方法。小小的说明一下,window.onload部分不用太在意;mouseUp部分用于在光标离开的时候,通知后台C#用户所选中的文本;而getImageUrl方法则是用于传递所点击的图片的url。当然,接口给定的内容中一般来说不可能给出img的onclick事件,因此,这个部分也需要我们进行相关处理:
private const string imgStyle = "style='max-width: 100%' onclick='getImageUrl(this.src)'";
/// <summary>
/// 处理新闻内容(主要是过滤Style和为图片添加点击事件)
/// </summary>
/// <param name="content"></param>
/// <returns></returns>
public static string UpdateContentStyle(string content)
{
if (string.IsNullOrEmpty(content))
return null; // 正则匹配img,统一替换它的Style,并添加点击事件(此处该如何匹配还应视具体情况而定)
var matches = Regex.Matches(content, @"<img[\s\S]*?(style='[\s\S]*?')[\s\S]*?/?>");
List<string> list = new List<string>(); if (matches.Count > )
list.Add(matches[].Groups[].Value); for (int i = ; i < matches.Count - ; i++)
{
if (!list.Contains(matches[i].Groups[].Value))
list.Add(matches[i].Groups[].Value);
} foreach (var item in list)
{
content = content.Replace(item, imgStyle);
} return content;
}
4.3 整合资源
/// <summary>
/// 将新闻主体部分拼凑成一段完整的html
/// </summary>
/// <returns></returns>
public string ProcessNewsContent()
{
var newsContent = WebViewUtil.UpdateContentStyle(news.Content); string webContent = "<html><head>" + contentCSS + "<script>" + js + "</script>" + "</head>" + "<body><div onmouseup='mouseUp()'> " + newsContent + "</div>" + "</body></html>"; return webContent;
}
简单说明一下,contentCSS就是上方GetContentCSS方法返回的结果,js是GetContentJS方法返回的结果,news则是用于展示的新闻,news.Content则是新闻内容,
它的大概模样如下(截取部分):
<p>
<strong>Today three top economists look at the state of the global economy and give their outlooks for 2017.</strong>
</p>
<p>
<img attachmentid=""232694"" src=""http://iosnews.chinadaily.com.cn/newsdata/news/201612/26/432220/picture_232694.jpg"" style='max-width: 100%' />
</p>
4.4 交货
webView.NavigateToString(content);
此处的content自然就是我们整合好的资源了(一段字符串)。
5.存在的问题
至此,关键的部分均已实现,让我们按照以上套路运行一番。(假设LoadImage和LoadTranslation方法都只是将JS传递过来的信息直接显示出来)
我们发现,PC上正常运行无压力;而手机上却是另一番光景——选中文本的时候没有任何反应,取消选中的时候反倒是显示信息了。(尚未搞清楚是为何)
因此,我采用了一个纠正措施,如下:
/// <summary>
/// 为wp修正文本选中问题
/// </summary>
public async void ProcessTextSelected4Phone(string text)
{
if (DeviceUtils.IsMobile)//wp上表现和电脑上不一样...
{
// 判断是否选中文本
DataPackage dataPackage = await webView.CaptureSelectedContentToDataPackageAsync();
if (dataPackage == null)// 表示未选中文本
{
LoadTranslation(null);// 隐藏翻译栏
return;
} var handleMouseUp = string.IsNullOrEmpty(text) ? false : true;
await webView.InvokeScriptAsync("mouseUp", new[] { handleMouseUp ? "y" : "" });// 若参数为y,则不触发事件
}
}
其实质就是在手机端,在正常的mouseUp触发基础上,通过代码主动多触发一次mouseUp,以达到修正目的。
6.Demo
http://files.cnblogs.com/files/lary/Demo.rar
7.参考
【WP8.1】WebView笔记:http://www.cnblogs.com/bomo/p/4320077.html
初次写博客,多多包涵
花开成景,花落成诗。不问花开几许,只问浅笑安然。(●'◡'●)
【uwp】浅谈China Daily 中划词翻译的实现的更多相关文章
- 【uwp】浅谈China Daily中数据同步到One Drive的实现
新版China Daily与旧版相比新增了数据同步的功能,那这个功能具体是如何实现的呢,现在让我们来一起看看. 1.注册应用 开发者中心的应用注册就不用多说了(https://developer.mi ...
- 转: 浅谈C/C++中的指针和数组(二)
转自:http://www.cnblogs.com/dolphin0520/archive/2011/11/09/2242419.html 浅谈C/C++中的指针和数组(二) 前面已经讨论了指针和数组 ...
- 转:浅谈C/C++中的指针和数组(一)
再次读的时候实践了一下代码,结果和原文不一致 error C2372: 'p' : redefinition; different types of indirection 不同类型的间接寻址 /// ...
- 转载 浅谈C/C++中的static和extern关键字
浅谈C/C++中的static和extern关键字 2011-04-21 16:57 海子 博客园 字号:T | T static是C++中常用的修饰符,它被用来控制变量的存贮方式和可见性.ext ...
- 浅谈C语言中的强符号、弱符号、强引用和弱引用
摘自http://www.jb51.net/article/56924.htm 浅谈C语言中的强符号.弱符号.强引用和弱引用 投稿:hebedich 字体:[增加 减小] 类型:转载 时间:2014- ...
- 【sql注入】浅谈sql注入中的Post注入
[sql注入]浅谈sql注入中的Post注入 本文来源:i春秋学院 00x01在许多交流群中,我看见很多朋友对于post注入很是迷茫,曾几何,我也是这样,因为我们都被复杂化了,想的太辅助了所以导致现在 ...
- 浅谈关于QT中Webkit内核浏览器
关于QT中Webkit内核浏览器是本文要介绍的内容,主要是来学习QT中webkit中浏览器的使用.提起WebKit,大家自然而然地想到浏览器.作为浏览器内部的主要构件,WebKit的主要工作是渲染.给 ...
- 浅谈JAVA GUI中,AWT与Swing的区别、联系及优缺点
浅谈JAVA GUI中,AWT与Swing的区别.联系及优缺点 A.区别 1.发布的时间 AWT是在JDK 1.0版本时提出的 Swing是在AWT之后提出的(JAVA 2) 2. ”重量” AWT是 ...
- 浅谈 Swift 2 中的 Objective-C 指针
浅谈 Swift 2 中的 Objective-C 指针 2015-09-07 499 文章目录 1. 在 Swift 中读 C 指针 2. 在 Swift 中创建 C 指针 3. 总结 作者:Ja ...
随机推荐
- Oracle分析函数入门
一.Oracle分析函数入门 分析函数是什么?分析函数是Oracle专门用于解决复杂报表统计需求的功能强大的函数,它可以在数据中进行分组然后计算基于组的某种统计值,并且每一组的每一行都可以返回一个统计 ...
- 【知识必备】内存泄漏全解析,从此拒绝ANR,让OOM远离你的身边,跟内存泄漏say byebye
一.写在前面 对于C++来说,内存泄漏就是new出来的对象没有delete,俗称野指针:而对于java来说,就是new出来的Object放在Heap上无法被GC回收:而这里就把我之前的一篇内存泄漏的总 ...
- 继电器是如何成为CPU的(1)
继电器是如何成为CPU的(1) ——<穿越计算机的迷雾>整理和总结 究竟是如何设计的电路,具有计算和控制的智力? 这一点也不高深.本系列文章从初中学的最简单的电路图说起,看看能不能从最初的 ...
- Partition1:新建分区表
未分区的表,只能存储在一个FileGroup中:对Table进行分区后,每一个分区都存储在一个FileGroup,或分布式存储在不同的FileGroup中.对表进行分区的过程,是将逻辑上完整的一个表, ...
- 执行 $Gulp 时发生了什么 —— 基于 Gulp 的前端集成解决方案(二)
前言 文章 在windows下安装gulp —— 基于 Gulp 的前端集成解决方案(一) 中,已经完成对 gulp 的安装,由于是window环境,文中特意提到了可以通过安装 gitbash 来代替 ...
- javascript单元测试框架mochajs详解
关于单元测试的想法 对于一些比较重要的项目,每次更新代码之后总是要自己测好久,担心一旦上线出了问题影响的服务太多,此时就希望能有一个比较规范的测试流程.在github上看到牛逼的javascript开 ...
- [.NET] 利用 async & await 的异步编程
利用 async & await 的异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/5922573.html 目录 异步编程的简介 异 ...
- c# 基础 object ,new操作符,类型转换
参考页面: http://www.yuanjiaocheng.net/webapi/config-webapi.html http://www.yuanjiaocheng.net/webapi/web ...
- Android—Volley:接收服务端发送的json数据乱码问题解决
new JsonObjectRequest中重写方法parseNetworkResponse,内容如下: /** * 重写此方法不会导致乱码 */ @Override protected Respon ...
- 安卓自定义组合控件--toolbar
最近在学习安卓APP的开发,用到了toolbar这个控件, 最开始使用时include layout这种方法,不过感觉封装性不好,就又改成了自定义组合控件的方式. 使用的工具为android stud ...