今天,我发布了CloudNotes的一个更新版本:1.0.5484.36793。这个版本与1.0.5472.20097不同的是,它拥有增强的笔记列表,与之前单调的列表系统相比,新的笔记列表不仅可以显示笔记的摘要内容,而且还可以从笔记中抽取第一张图片,并显示图片的详细信息:

怎么样?相比之前的笔记列表,现在的设计是不是能够展示更丰富的信息呢?

升级到最新版本

如果在读完我的第一篇关于CloudNotes的文章,《CloudNotes:一个云端个人笔记系统》之后,已经安装并体验了上一个版本的CloudNotes,那么,当你重新打开CloudNotes桌面客户端时,你将在登录界面,或者主界面的状态栏部分看到发现新版本更新的通知信息:

你可以点击这个信息来进入新版本更新系统。然而不幸的是,更新系统会告诉你,更新失败,原因是因为你打开了Windows 7的用户帐户控制(UAC)的功能,更新程序不具备访问C:\Program Files (x86)目录的权限。解决方案有如下两种:

  1. 进入控制面板,关闭用户帐户控制(UAC)功能
  2. 点击这里】下载CloudNotes更新程序的补丁包,下载以后,将zip文件解压并覆盖CloudNotes安装目录下的Updater目录中(比如:C:\Program Files (x86)\daxnet\CloudNotes Desktop Client)的所有文件,然后重新通过CloudNotes桌面客户端进入升级程序,即可完成升级

如果你是第一次看到有CloudNotes这么个玩意儿,并且打算尝试使用的话,请直接【点击这里】下载版本1.0.5484.36793,它是截止到本文撰写时的最新版本,包含了这个最新的日志列表和修复的CloudNotes更新程序。

技术实现

在此大致介绍一下我是如何实现这种全新的笔记列表界面的,并简要介绍一下如何让应用程序在UAC下具有访问文件系统资源的权限。

增强的笔记列表

说起来也有趣,有一天我看到有个同事在用Evernote,觉得它的笔记列表视图做得挺不错,还有缩略图:

给人的感觉就是能够在简单的视图中体现更多的笔记信息,让人更容易地找到需要查看的笔记内容。于是我也在想,我的CloudNotes桌面客户端是否也可以实现类似的效果。

经过一番研究之后,我决定对目前笔记列表所使用的树形控件进行自定义:从System.Windows.Forms.TreeView类继承一个自定义的TreeViewEx类型,并使用OwnerDrawText的自定义绘画方式,对每个树状节点(TreeNode)进行重绘,以达到类似的效果。

在TreeNode进行重绘时(就是在OnDrawNode事件处理过程中),TreeViewEx会根据当前节点上的TreeNodeExItem数据,对树状节点进行重绘,从而达到上面截图中展示的效果。实现起来其实并不困难,就是需要有耐心,文字的颜色、定位、图片的尺寸等等,都需要花时间慢慢调整。这部分代码我也不多做解释了,朋友们请自行参考CloudNotes.DesktopClient项目下Controls\TreeViewEx.cs源代码文件中的实现即可。相比之下,更为有趣技术点主要有:

从HTML中获得纯文本信息

这就是用于显示笔记的摘要文字部分。由于笔记是HTML格式存储的,而摘要部分却仅需要笔记文字的开头一段即可,因此,就需要从HTML中去掉HTML的标记,从中提取可读的纯文本信息。在CloudNotes中,我是使用下面的扩展方法来实现这个功能的。该方法同时被WebAPI和桌面客户端使用。通过代码可以看到,我仅仅将笔记中的前100个文字作为笔记的摘要信息。

  1. /// <summary>
  2. /// Extract the description from the html.
  3. /// </summary>
  4. /// <param name="html"></param>
  5. /// <returns></returns>
  6. public static string ExtractDescription(this string html)
  7. {
  8. var plainText = html.RemoveHtmlTags();
  9. return plainText.Substring(0, plainText.Length < 100 ? plainText.Length : 100);
  10. }
  11.  
  12. /// <summary>
  13. /// Removes all the HTML tags and bad characters from the given HTML string.
  14. /// </summary>
  15. /// <param name="html">The source HTML string.</param>
  16. /// <returns></returns>
  17. private static string RemoveHtmlTags(this string html)
  18. {
  19. html = HttpUtility.UrlDecode(html);
  20. html = HttpUtility.HtmlDecode(html);
  21.  
  22. html = RemoveTag(html, "<!--", "-->");
  23. html = RemoveTag(html, "<script", "</script>");
  24. html = RemoveTag(html, "<style", "</style>");
  25.  
  26. //replace matches of these regexes with space
  27. html = Tags.Replace(html, " ");
  28. html = NotOkCharacter.Replace(html, " ");
  29. html = SingleSpacedTrim(html);
  30.  
  31. return html;
  32. }

提取笔记中的第一张图片

为了显示笔记图片的缩略图,首先需要提取笔记中的第一张图片,然后再通过图像处理函数产生缩略图。话不多说,直接上代码:

  1. /// <summary>
  2. /// The regular expression for extracting the Src value from an HTML Img tag.
  3. /// </summary>
  4. public const string ImgSrcFormatPattern = @"<img[^>]*?src\s*=\s*[""']?([^'"" >]+?)[ '""][^>]*?>";
  5.  
  6. /// <summary>
  7. /// Extract the thumbnail image from the html.
  8. /// </summary>
  9. /// <param name="html"></param>
  10. /// <returns></returns>
  11. public static string ExtractThumbnailImageBase64(this string html)
  12. {
  13. var imageBase64List = html.GetImgSrcBase64FromHtml();
  14. string result = null;
  15. if (imageBase64List != null && imageBase64List.Any())
  16. {
  17. result = imageBase64List.First();
  18. }
  19. return result;
  20. }
  21.  
  22. /// <summary>
  23. ///
  24. /// </summary>
  25. /// <param name="html"></param>
  26. /// <returns></returns>
  27. private static IEnumerable<string> GetImgSrcBase64FromHtml(this string html)
  28. {
  29. var matchesImgSrc = Regex.Matches(html, Constants.ImgSrcFormatPattern,
  30. RegexOptions.IgnoreCase | RegexOptions.Singleline);
  31. if (matchesImgSrc.Count == 0)
  32. return null;
  33. List<string> result = new List<string>();
  34. foreach (Match m in matchesImgSrc)
  35. {
  36. var href = m.Groups[1].Value;
  37. var pos = href.IndexOf("base64,", StringComparison.InvariantCultureIgnoreCase);
  38. pos += 7;
  39. result.Add(href.Substring(pos, href.Length - pos).Trim());
  40. }
  41. return result;
  42. }

缩略图的产生

缩略图的产生代码是在TreeViewEx控件中实现的,基本思路其实很简单,就是根据图片的长宽进行等比缩小即可。代码如下:

  1. private static Image FixedSize(Image imgPhoto, int width, int height, Color clearColor)
  2. {
  3. int sourceWidth = imgPhoto.Width;
  4. int sourceHeight = imgPhoto.Height;
  5. int sourceX = 0;
  6. int sourceY = 0;
  7. int destX = 0;
  8. int destY = 0;
  9.  
  10. float nPercent;
  11. float nPercentW;
  12. float nPercentH;
  13.  
  14. nPercentW = ((float)width / (float)sourceWidth);
  15. nPercentH = ((float)height / (float)sourceHeight);
  16. if (nPercentH < nPercentW)
  17. {
  18. nPercent = nPercentH;
  19. destX = Convert.ToInt16((width - (sourceWidth * nPercent)) / 2);
  20. }
  21. else
  22. {
  23. nPercent = nPercentW;
  24. destY = Convert.ToInt16((height - (sourceHeight * nPercent)) / 2);
  25. }
  26.  
  27. int destWidth = (int)(sourceWidth * nPercent);
  28. int destHeight = (int)(sourceHeight * nPercent);
  29.  
  30. Bitmap bmPhoto = new Bitmap(width, height,
  31. PixelFormat.Format24bppRgb);
  32. bmPhoto.SetResolution(imgPhoto.HorizontalResolution,
  33. imgPhoto.VerticalResolution);
  34.  
  35. Graphics grPhoto = Graphics.FromImage(bmPhoto);
  36. grPhoto.Clear(clearColor);
  37. grPhoto.InterpolationMode =
  38. InterpolationMode.HighQualityBicubic;
  39.  
  40. grPhoto.DrawImage(imgPhoto,
  41. new Rectangle(destX, destY, destWidth, destHeight),
  42. new Rectangle(sourceX, sourceY, sourceWidth, sourceHeight),
  43. GraphicsUnit.Pixel);
  44.  
  45. grPhoto.Dispose();
  46. return bmPhoto;
  47. }

让应用程序在UAC下具有访问文件系统资源的权限

这是之前版本中CloudNotes桌面客户端更新程序遇到的一个问题。如果用户将CloudNotes安装在系统目录中,更新程序在试图更新CloudNotes桌面客户端的版本的时候,就会遇到错误,提示无法更新。根本原因是用户在Windows系统中启用了用户帐户控制(UAC),即使当前登录系统的帐户是管理员,也并不代表该用户对系统中的所有资源都具备管理员权限。在这种情况下,即使是由管理员启动的更新程序,也无法将下载并解压好的文件复制到系统目录下。

要解决这一问题,就需要在.NET应用程序的Manifest里指定requestedExecutionLevel,将其指定为requireAdministrator,于是,在执行更新程序的时候,会弹出以下标准对话框,让用户对应用程序所作出的行为进行确认:

此时只需点击“是”按钮即可完成更新。

更改Manifest其实很简单,只需要在应用程序项目上,通过Visual Studio的“添加项目”对话框,即可添加App.manifest文件,将文件中的requestedExecutionLevel改为requireAdministrator即可:

有关CloudNotes WebAPI以及桌面客户端的其它内容,请大家直接上https://github.com/daxnet/CloudNotes站点直接查看源代码即可。不懂的地方可以在此留言。

接下来??

我是打算一步步对CloudNotes进行功能和性能完善的,接下来要做的事情还有太多。在每次完成新功能更新时,我都会出博客文章进行介绍。如果没有新功能,我也会穿插介绍已有功能的相关技术实现。就目前而言,打算接下来的版本逐步提供以下功能改进:

  • 本地缓存和服务器同步系统 - 目前每次打开和保存笔记,都是与服务器直接相连的,不仅增大服务器负载,而且用户体验也不够流畅
  • 插件系统 - 任何感兴趣的朋友都可以通过插件系统,为CloudNotes桌面客户端编写插件,比如,直接将网页抓取成日志等
  • 用户之间的互信和笔记共享服务
  • 文档结构视图 - 通过分析HTML文档,提供层级的文档结构视图,方便用CloudNotes进行写作的用户

等等等等。。。。。。让我们一起期待吧。。

CloudNotes之桌面客户端篇:增强的笔记列表的更多相关文章

  1. CloudNotes之桌面客户端篇:笔记撰写样式的支持

    最近在CloudNotes桌面客户端中新增了笔记撰写样式的功能.当用户新建笔记的时候,可以在输入笔记标题的同时,选择笔记撰写样式,由安装包默认提供的样式主要有默认样式(Default).羊皮纸样式(L ...

  2. CloudNotes之桌面客户端篇:插件系统的实现

    [CloudNotes版本更新历史与各版本下载地址请点击此处] [CloudNotes中文系列文章汇总列表请点击此处] [查看CloudNotes源代码请点击此处] 有时候,同一个名词,针对不同的人群 ...

  3. Windows Server 2012 R2超级虚拟化之七 远程桌面服务的增强

    Windows Server 2012 R2超级虚拟化之七  远程桌面服务的增强 在Windows Server 2012提供的远程桌面服务角色,使用户能够连接到虚拟桌面. RemoteApp程序.基 ...

  4. C#Windows Service服务程序的安装/卸载、启动/停止 桌面客户端管理程序设计

    C#Windows Service服务程序的安装/卸载.启动/停止 桌面客户端管理程序设计 关于Windows Service程序的安装与卸载如果每次使用命令行操作,那简直要奔溃了,太麻烦而且还容易出 ...

  5. Java多线程编程实战指南(核心篇)读书笔记(四)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76690961冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  6. Java多线程编程实战指南(核心篇)读书笔记(三)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76686044冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  7. "xaml+cs"桌面客户端跨平台初体验

    "Xaml+C#"桌面客户端跨平台初体验 前言   随着 .Net 5的到来,微软在 .Net 跨平台路上又开始了一个更高的起点.回顾.Net Core近几年的成果,可谓是让.Ne ...

  8. [原创]用windows7连接windows2003的终端服务器时,出现"由于这台计算机没有远程桌面客户端访问许可证,远程会话被中断"的问题

    用windows7连接windows2003的终端服务器时,出现"由于这台计算机没有远程桌面客户端访问许可证,远程会话被中断"的问题,原因是终端服务器授权方式设置为了"每 ...

  9. Atitit 全屏模式的cs桌面客户端软件gui h5解决方案 Kiosk模式

    Atitit 全屏模式的cs桌面客户端软件gui h5解决方案 Kiosk模式 1.1. Kiosk Software广泛用于公共电脑或者嵌入系统,最常用的就是ATM机.自动服务机之类的系统了.,1 ...

随机推荐

  1. Nginx模块参考手册:HTTP核心模块

    FROM: http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=17238776&id=2982697 这些模块默认会全部编 ...

  2. C#中ToString格式大全

    更多资源:http://denghejun.github.io C 货币 2.5.ToString("C") ¥2.50 D 十进制数 25.ToString("D5&q ...

  3. Ubuntu安装Gnome3

    参考:How To Install GNOME In Ubuntu 14.04 . Ubuntu11.10安装GNOME3,卸载UNITY和UNITY2D操作 和How to install Gnom ...

  4. matlab size、numel、length、fix函数的使用,补充nargin

    size():获取矩阵的行数和列数 (1)s=size(A), 当只有一个输出参数时,返回一个行向量,该行向量的第一个元素时矩阵的行数,第二个元素是矩阵的列数.(2)[r,c]=size(A), 当有 ...

  5. .NET 4.0 版本号

    .NET 4.5.1, .NET 4.5 和 .NET 4.0 均基于 .NET 4.0 CLR,而 .NET 4.5 对 CLR进行了升级和Bug修复. .NET 4.0 - 4.0.30319.1 ...

  6. 深圳电信光纤用户必备:简单破解中兴ZXA10 F460光电猫,实现WIFI和自动拨号功能

    最近搬家,ADSL转成光纤宽带,下载速度非常给力.原来的ADSL路由器派不上用场,电信的安装人员也给开通了wifi功能,只是wifi无法上网,而且拨号一定需要用电脑连网线进行拨号.以前是直接把拨号账户 ...

  7. .NET单元测试的艺术-2.核心技术

    开篇:上一篇我们学习基本的单元测试基础知识和入门实例.但是,如果我们要测试的方法依赖于一个外部资源,如文件系统.数据库.Web服务或者其他难以控制的东西,那又该如何编写测试呢?为了解决这些问题,我们需 ...

  8. Hadoop学习笔记—7.计数器与自定义计数器

    一.Hadoop中的计数器 计数器:计数器是用来记录job的执行进度和状态的.它的作用可以理解为日志.我们通常可以在程序的某个位置插入计数器,用来记录数据或者进度的变化情况,它比日志更便利进行分析. ...

  9. Google分布式构建软件之一:获取源代码

    本文原文在google开发者工具组的博客上[需要FQ],以下是我的翻译,欢迎转载,但尊重作者版权,注名原文地址. 在Google,所有的产品都是在主干上开发的.这样的好处:每个人都可以查看和修改代码, ...

  10. New Year's resolution for 2016

    A New Year's resolution is a traditional for me to celebrate a new beginning. For the past year, I h ...