【WP开发】读写剪贴板_移动开发_编程开发_程序员俱乐部

中国优秀的程序员网站程序员频道CXYCLUB技术地图
热搜:
更多>>
 
您所在的位置: 程序员俱乐部 > 编程开发 > 移动开发 > 【WP开发】读写剪贴板

【WP开发】读写剪贴板

 2015/3/26 18:09:35  东邪独孤  程序员俱乐部  我要评论(0)
  • 摘要:在WP8.1中只有SilverlightApp支持操作剪贴板的API,RuntimeApp并不支持。不过,在WP10中也引入了可以操作剪贴板的API。顺便说点题外话,有人会说,我8.1的开发还没学了,Win10又来了,那我怎么学得过来?放心,学得过来的D,因为只要你对有WPF的基础,再加上对RTApp的学习就足够应付Win10上的app开发了。实际上,面向10的API都是在8.1的基础上增加一点新的API而已,你不需要学习新的知识就可以直接玩10的开发了。而且,你在10上依然可以继续使用8
  • 标签:开发

在WP 8.1中只有Silverlight App支持操作剪贴板的API,Runtime App并不支持。不过,在WP 10中也引入了可以操作剪贴板的API。

顺便说点题外话,有人会说,我8.1的开发还没学了,Win10又来了,那我怎么学得过来?放心,学得过来的D,因为只要你对有WPF的基础,再加上对RT App的学习就足够应付Win 10上的app开发了。实际上,面向10的API都是在8.1的基础上增加一点新的API而已,你不需要学习新的知识就可以直接玩10的开发了。

而且,你在10上依然可以继续使用8.1的API。当然我指的是RuntimeApp,Silverlight的API有可能将来不用。虽然RT API使用起来看着和以往的.net类库没什么区别,而实际上RT库是基于COM的形式存在的,如果你足够细心的话,通过在RT应用发生异常的时候,你都会看到来自COM的HRESULT值,在调用试的时候,你也能看到许多RT库的类型显示为COM对象。这是很有力的证据。

所以我大胆估计,RT库就是为了优化性能而出现的,随同新的编译器和.net 4.6的发布,.net的性能会有更可观的优化,这是肯定的事情。而对于Win 10,微软也确实在努力优化,尽管现在是预览版,问题肯定会多一些,但我使用Build 10041后,确实感觉到反应速度快了很多,因为我的笔记本配置不算很高,对性能是有一点儿敏感的。

好了,屁话说得太多了,为了避免大家向我扔石头(如果你要扔的话,建议你扔玛瑙),下面我再简单说说“通用应用程序项目”(UAP),耍W10开发必须了解这个。

通过框架是在面向8.1(VS UD2)的时候出现,在这个通用应用中是分为三个项目的,相信大家有所了解。

1、面向Windows Phone的应用。

2、面向Windows的应用。

3、共享代码。

平台共用的代码通常会放到第三个项目中,不过这样做有时候也觉得代码管理起来不方便。

 

在面向10的项目中直接就合并成一个项目,而不再按平台分开,直接就统一到一个项目中,生成的应用程序你喜欢在手机上运行也行,在平板上运行也行,没有平板在普通PC上耍也可以。

在VS的“对象浏览器”窗口中,你会发现有一个名为Universal App Platform的子集,它就是包含通用API的各个库的集合,把多个平台可以共享的API都放到一起了,开发者就不必去管理共享代码了。

当然你也可以继续使用面向8.1的项目,这些项目模板在“Store Apps”节点下面。

 

好了,有了上面的介绍,再进入咱们今天的正题就轻松了。

首先,参照上面截图自己新项一个Windows 10的空白应用程序项目。

这里我给出的示例比较简单,就是复制和粘贴一张图片。还是和以前一样,用XAML代码来布局UI。

    <Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
        <Grid.RowDefinitions>
            <RowDefinition Height="*"/>
            <RowDefinition Height="*"/>
        </Grid.RowDefinitions>
        <StackPanel Orientation="Horizontal" VerticalAlignment="Top">
            <Image Name="imgSrc" Width="200" Height="200" />
            <Button Content="复制" Click="OnCopy"/>
        </StackPanel>
        <StackPanel Grid.Row="1" VerticalAlignment="Top" Orientation="Horizontal">
            <Image Name="imgDes" Width="200" Height="200"/>
            <Button Content="粘贴" Click="OnSet"/>
        </StackPanel>
    </Grid>

然后在页面加载后从项目目录中读出一个图像文件。

        public MainPage()
        {
            this.InitializeComponent();
            this.Loaded += OnPageLoaded;
        }

        private async void OnPageLoaded(object sender, RoutedEventArgs e)
        {
            // 加载项目中的图片
            StorageFile imgFile = await StorageFile.GetFileFromApplicationUriAsync(new Uri("ms-appx:///1.jpg"));
            // 打开文件流
            IRandomAccessStream stream = await imgFile.OpenReadAsync();
            // 显示图像
            BitmapImage bmp = new BitmapImage();
            bmp.DecodePixelWidth = 250;
            await bmp.SetSourceAsync(stream);
            this.imgSrc.Source = bmp;
            // 将流对象放入Tag属性中暂存,稍后会用得上
            this.Tag = RandomAccessStreamReference.CreateFromStream(stream.CloneStream());
            stream.Dispose();
        }

文件流一般来说,用完就应该马上关闭,不要借着别人的钱不还,但是,我们随后要把文件流中的内容写入剪贴板,而文件流被关闭后又无法操作了,于是,我调用CloneStream方法直接将流复制一份,这样原来的文件流可以关闭了,并且不影响后面的操作。并且赋值给Tag属性,随后在其他代码中可以从Tag属性取出流对象。
下面代码将流中的数据写入剪贴板:

            // 从Tag属性中取出文件流
            RandomAccessStreamReference stream = this.Tag as RandomAccessStreamReference;
            if (stream != null)
            {
                // 创建数据包装
                DataPackage package = new DataPackage();
                // 放入内容
                package.SetBitmap(stream);
                // 将数据放入ClipBoard
                Clipboard.SetContent(package);
            }

下面代码从剪贴板读出数据,即“粘贴”:

            // 从clip board中取出数据
            DataPackageView pack = Clipboard.GetContent();

            // 如果Count为0,表示剪贴板中没有可用的内容
            if (pack.AvailableFormats.Count == 0)
            {
                return;
            }
            else
            {
                foreach (string format in pack.AvailableFormats)
                {
                    System.Diagnostics.Debug.WriteLine(format);
                }
            }
            // 验证格式是否正确
            if (pack.AvailableFormats.Where(s=>s == StandardDataFormats.Bitmap).Count() > 0)
            {

                // 读出流
                RandomAccessStreamReference streamref = await pack.GetBitmapAsync();
                using (IRandomAccessStream streamIn = await streamref.OpenReadAsync())
                {
                    // 显示图片
                    BitmapImage bmp = new BitmapImage();
                    bmp.DecodePixelWidth = 250;
                    await bmp.SetSourceAsync(streamIn);
                    this.imgDes.Source = bmp;
                }
                Clipboard.Clear(); //清除剪贴板中的内容

            }


完事了,例子就这么简单。结果如下图。

 

下面就给大家介绍一下知识点。

1、操作剪贴板用的是Clipboard类(命名空间:Windows.ApplicationModel.DataTransfer),这个类是静态的,也就是说它的成员也是静态的,不用new就可以直接用。

2、要向剪贴板写入数据,就用SetContent方法,要从剪贴板中读出数据就用GetContent方法。

3、在写入的时候,数据是通过DataPackage类来包装的,而读取时则是通过DataPackageView类来操作的。那么,这两个家伙有什么区别呢?

看到了吧,一个是SetXXXXX,一个是GetXXXXXX。因此,DataPackage是用于创建数据包的,而DataPackageView是用来读数据包的。

4、操作也不难,写什么类型的数据就用什么方法,如写文本用SetText,读文本用GetTextAsync。本示例是读写图像,所以写的时候调用SetBitmap方法,读的时候调用GetBitmapAsync方法。注意,图像的内容是用RandomAccessStreamReference类来封装的流对象,通过静态的CreateFromStream方法可以从现存的流创建新的RandomAccessStreamReference实例。

5、用DataPackageView方法检索内容之前,应当访问AvailableFormats属性,检查一下剪贴板上的内容是否可用,或者是空的。该属性是一个字符串列表,如果列表中元素个数为0,说明剪贴板中的数据无效。

6、Clipboard.Clear()方法清理剪贴板中的内容。

 

随后,我把示例源代码上传,大家可以下载下来玩玩。

下载地址:http://files.cnblogs.com/files/tcjiaan/ClipboardApp.zip

===============================

这里补充一下,现在Win10和VS2015都是预览版,如果你害怕出现小问题,那你得谨慎使用,如果你像我这样,不干正经事,只是体验一下,可以在电脑上装来耍耍。我比较穷,只有一台笔记本,买不起新电脑,就花几百块钱买了一台国产的山寨平板,8寸的屏幕是小了点,不过还是可以当电脑用的。所以我的笔记本用来开发实验,山寨板用来看看电子书,看看电影电视剧,和林妹妹聊聊天,还是可以用的。

要注意的是,装Win10预览版一定要装最新发布的Build 10041版本,否则面向Win 10通用应用的XAML设计器无法加载。一开始我也犯了这个错误,后来查看一下清单中的XML文件,才发现它要求的是Build 10030以上的版本,而大于这个版本的就只有最近发布的Build 10041了。

另外,有一件事,需要向各位读者表示道歉的,有位细心的朋友发现《精通C# 5.0》中第54页,定义的Book结构有一个属性应该是ISBN,我弄错为ISBM。这是我的疏忽,偏偏编辑在审稿时没有注意,我在定稿时也没看到。因此,向各位表示歉意。

 

发表评论
用户名: 匿名