随着 WinRT 8.1 API 的发布,Windows 8.1 和 Windows Phone 8.1 (基于 WinRT) 应用程序的开发模型经历了戏剧性的收敛性。与一些特定于平台的考虑,我们现在可以为 Windows 和使用几乎相同的 XAML 框架和 API 来开发 Windows Phone 开发应用程序。

"旧的"基于 Windows Phone 8.0 Silverlight API 将继续得到支持和改善,但基于 WinRT 的融合的模型显然是要在未来走的路。

所以抽时间实现了wp8.1实现二维码扫描的功能

1. 在项目中添加对Zxing的引用

Zxing示例代码下载地址:http://zxingnet.codeplex.com/

2.添加页面代码

 <Grid x:Name="LayoutRoot"  Background="Black">
<Grid Background="Black">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="50"/>
<ColumnDefinition Width="300"/>
<ColumnDefinition Width="50"/>
</Grid.ColumnDefinitions>
<Grid Grid.Column="0" Height="366" Margin="0,0,0,0"> </Grid>
<!--取景器区域-->
<Grid Grid.Column="1" x:Name="ScanViewGrid" Margin="0,0,0,0">
<Grid.Resources>
<Storyboard x:Name="myStoryboard">
<DoubleAnimation
Storyboard.TargetName="imgScan"
Storyboard.TargetProperty="(Canvas.Top)"
Duration="0:0:2.5"
From="0"
To="300"
RepeatBehavior="Forever"/>
</Storyboard>
</Grid.Resources>
<!--扫描线--> <Canvas Width="300" Canvas.Top="170" Height="300">
<CaptureElement Stretch="UniformToFill" x:Name="VideoCapture" Width="300" Height="300" > </CaptureElement>
<Image x:Name="imgScan" Source="/Assets/img/light.png" Width="300" Height="40" Stretch="Fill"/> <Image x:Name="CaptureImage" Width="300" Height="300" Visibility="Collapsed" />
</Canvas> </Grid>
<Grid Grid.Column="2" Margin="0,0,0,0"/>
</Grid> <TextBlock x:Name="Error" VerticalAlignment="Bottom" FontSize="32" Width="400" TextAlignment="Center" />
<TextBlock x:Name="ScanResult" Text="hhh" FontSize="25" Foreground="White" VerticalAlignment="Bottom" TextAlignment="Center" Width="400"/>
</Grid>

3.功能实现

  public sealed partial class QCodeView : Page
{
private readonly MediaCapture _mediaCapture = new MediaCapture();
private Result _result;
private NavigationHelper navigationHelper;
private ObservableDictionary defaultViewModel = new ObservableDictionary();
private DispatcherTimer _timer; public QCodeView()
{
this.InitializeComponent();
//设备屏幕方向
DisplayInformation.AutoRotationPreferences = DisplayOrientations.Portrait;
///隐藏StatusBar
var statusBar = StatusBar.GetForCurrentView();
statusBar.HideAsync();
this.navigationHelper = new NavigationHelper(this);
this.navigationHelper.LoadState += this.NavigationHelper_LoadState;
this.navigationHelper.SaveState += this.NavigationHelper_SaveState;
} /// <summary>
/// 获取与此 <see cref="Page"/> 关联的 <see cref="NavigationHelper"/>。
/// </summary>
public NavigationHelper NavigationHelper
{
get { return this.navigationHelper; }
} /// <summary>
/// 获取此 <see cref="Page"/> 的视图模型。
/// 可将其更改为强类型视图模型。
/// </summary>
public ObservableDictionary DefaultViewModel
{
get { return this.defaultViewModel; }
} /// <summary>
/// 使用在导航过程中传递的内容填充页。 在从以前的会话
/// 重新创建页时,也会提供任何已保存状态。
/// </summary>
/// <param name="sender">
/// 事件的来源; 通常为 <see cref="NavigationHelper"/>
/// </param>
/// <param name="e">事件数据,其中既提供在最初请求此页时传递给
/// <see cref="Frame.Navigate(Type, Object)"/> 的导航参数,又提供
/// 此页在以前会话期间保留的状态的
/// 字典。 首次访问页面时,该状态将为 null。</param>
private void NavigationHelper_LoadState(object sender, LoadStateEventArgs e)
{
} /// <summary>
/// 保留与此页关联的状态,以防挂起应用程序或
/// 从导航缓存中放弃此页。值必须符合
/// <see cref="SuspensionManager.SessionState"/> 的序列化要求。
/// </summary>
/// <param name="sender">事件的来源;通常为 <see cref="NavigationHelper"/></param>
///<param name="e">提供要使用可序列化状态填充的空字典
///的事件数据。</param>
private void NavigationHelper_SaveState(object sender, SaveStateEventArgs e)
{ } #region NavigationHelper 注册 /// <summary>
/// 此部分中提供的方法只是用于使
/// NavigationHelper 可响应页面的导航方法。
/// <para>
/// 应将页面特有的逻辑放入用于
/// <see cref="NavigationHelper.LoadState"/>
/// 和 <see cref="NavigationHelper.SaveState"/> 的事件处理程序中。
/// 除了在会话期间保留的页面状态之外
/// LoadState 方法中还提供导航参数。
/// </para>
/// </summary>
/// <param name="e">提供导航方法数据和
/// 无法取消导航请求的事件处理程序。</param>
protected override void OnNavigatedTo(NavigationEventArgs e)
{ myStoryboard.Begin();
InitVideoCapture();
InitVideoTimer(); this.navigationHelper.OnNavigatedTo(e);
} /// <summary>
/// 初始化摄像
/// </summary>
private async void InitVideoCapture()
{
///摄像头的检测
var cameras = await DeviceInformation.FindAllAsync(DeviceClass.VideoCapture);
if (cameras.Count < 1)
{
Error.Text = "设备没有摄像头,读取本地资源";
await DecodeStaticResource();
return;
} /// 创建 MediaCaptureInitializationSettings 对象的新实例。
var settings = new MediaCaptureInitializationSettings
{
StreamingCaptureMode = StreamingCaptureMode.AudioAndVideo,
MediaCategory = MediaCategory.Other,
AudioProcessing = Windows.Media.AudioProcessing.Default,
VideoDeviceId = cameras[1].Id
};
//if (cameras.Count == 1)
//{
// settings = new MediaCaptureInitializationSettings { VideoDeviceId = cameras[0].Id }; // 0 => front, 1 => back
//}
//else
//{
// settings = new MediaCaptureInitializationSettings { VideoDeviceId = cameras[1].Id }; // 0 => front, 1 => back
//} await _mediaCapture.InitializeAsync(settings);
VideoCapture.Source = _mediaCapture;
await _mediaCapture.StartPreviewAsync();
}
private void InitVideoTimer()
{
_timer = new DispatcherTimer();
_timer.Interval = TimeSpan.FromSeconds(6);
_timer.Tick += _timer_Tick;
_timer.Start(); }
private bool IsBusy = false;
async void _timer_Tick(object sender, object e)
{
try
{
while (_result == null)
{
if (!IsBusy)
{
IsBusy = true;
///获取焦点
//var autoFocusSupported = _mediaCapture.VideoDeviceController.FocusControl.SupportedFocusModes.Contains(FocusMode.Auto);
//if (autoFocusSupported)
//{
// var focusSettings = new FocusSettings
// {
// Mode = FocusMode.Auto,
// AutoFocusRange = AutoFocusRange.Normal
// };
// _mediaCapture.VideoDeviceController.FocusControl.Configure(focusSettings);
// //await _mediaCapture.VideoDeviceController.FocusControl.FocusAsync().AsTask(_cancellationTokenSource.Token);
//} var photoStorageFile = await KnownFolders.PicturesLibrary.CreateFileAsync("qcode.jpg", CreationCollisionOption.ReplaceExisting);
await _mediaCapture.CapturePhotoToStorageFileAsync(ImageEncodingProperties.CreateJpeg(), photoStorageFile); var stream = await photoStorageFile.OpenReadAsync();
// initialize with 1,1 to get the current size of the image
var writeableBmp = new WriteableBitmap(1, 1);
writeableBmp.SetSource(stream);
// and create it again because otherwise the WB isn't fully initialized and decoding
// results in a IndexOutOfRange
writeableBmp = new WriteableBitmap(writeableBmp.PixelWidth, writeableBmp.PixelHeight);
stream.Seek(0);
writeableBmp.SetSource(stream);
_result = ScanBitmap(writeableBmp);
await photoStorageFile.DeleteAsync(StorageDeleteOption.PermanentDelete);
}
if (_result != null)
{
await _mediaCapture.StopPreviewAsync();
VideoCapture.Visibility = Visibility.Collapsed;
CaptureImage.Visibility = Visibility.Visible;
ScanResult.Text = _result.Text;
_timer.Stop();
myStoryboard.Stop();
imgScan.Visibility = Visibility.Collapsed; }
IsBusy = false; }
}
catch (Exception ex)
{ IsBusy = false;
//Error.Text = ex.Message;
}
}
/// <summary>
/// 读取本地资源
/// </summary>
/// <returns></returns>
private async System.Threading.Tasks.Task DecodeStaticResource()
{
var file = await Windows.ApplicationModel.Package.Current.InstalledLocation.GetFileAsync(@"Assets\1.png");
var stream = await file.OpenReadAsync();
// initialize with 1,1 to get the current size of the image
var writeableBmp = new WriteableBitmap(1, 1);
writeableBmp.SetSource(stream);
// and create it again because otherwise the WB isn't fully initialized and decoding
// results in a IndexOutOfRange
writeableBmp = new WriteableBitmap(writeableBmp.PixelWidth, writeableBmp.PixelHeight);
stream.Seek(0);
writeableBmp.SetSource(stream);
CaptureImage.Source = writeableBmp;
VideoCapture.Visibility = Visibility.Collapsed;
CaptureImage.Visibility = Visibility.Visible; _result = ScanBitmap(writeableBmp);
if (_result != null)
{
ScanResult.Text += _result.Text;
}
return;
}
/// <summary>
/// 解析二维码图片
/// </summary>
/// <param name="writeableBmp">拍摄的图片</param>
/// <returns></returns>
private Result ScanBitmap(WriteableBitmap writeableBmp)
{
var barcodeReader = new BarcodeReader
{
TryHarder = true,
AutoRotate = true
};
var result = barcodeReader.Decode(writeableBmp); if (result != null)
{
CaptureImage.Source = writeableBmp;
} return result;
}
/// <summary>
/// 页面离开时对_mediaCapture对象进行释放
/// 防止内存溢出及资源占用
/// </summary>
/// <param name="e"></param>
protected override async void OnNavigatedFrom(NavigationEventArgs e)
{
try
{
await _mediaCapture.StopPreviewAsync();
await _mediaCapture.StopRecordAsync();
_mediaCapture.Dispose();
}
catch (Exception)
{ }
this.navigationHelper.OnNavigatedFrom(e);
}
#endregion
}

源码下载地址:http://vdisk.weibo.com/s/ztJnMX2jYqaK3

注意事项: 在调试时如果出现死机现象 ,请长按锁屏键+音量(-)键 方可重启手机。

学习交流QQ群:157153754 (入群成员请将您的昵称改为: 城市-昵称 )

WindowsPhone8.1 开发-- 二维码扫描的更多相关文章

  1. iOS开发-二维码扫描和应用跳转

    iOS开发-二维码扫描和应用跳转   序言 前面我们已经调到过怎么制作二维码,在我们能够生成二维码之后,如何对二维码进行扫描呢? 在iOS7之前,大部分应用中使用的二维码扫描是第三方的扫描框架,例如Z ...

  2. ipad开发:二维码扫描,摄像头旋转角度问题解决办法

    之前一直是在手机上开发,用系统原生二维码扫描功能,一点问题都没有,但是在ipad上,用户是横屏操作的,虽然界面旋转了,是横屏的,但是摄像头里显示的依然是竖屏效果,也就是说从摄像头里看到的和人眼看到的内 ...

  3. H5混合开发二维码扫描以及调用本地摄像头

    今天主管给了我个需求,说要用混合开发,用H5调用本地摄像头进行扫描二维码,我之前有做过原生安卓的二维码扫一扫,主要是通过调用zxing插件进行操作的,其中还弄了个闪光灯.但是纯H5的没接触过,心里没底 ...

  4. Android开发--二维码开发应用(转载!)

    android项目开发 二维码扫描   基于android平台的二维码扫描项目,可以查看结果并且链接网址 工具/原料 zxing eclipse 方法/步骤   首先需要用到google提供的zxin ...

  5. IOS开发小功能2:二维码扫描界面的设计(横线上下移动)

    效果图如上,实现的是一个二维码扫描界面. 下面我贴出线条上下移动的代码,至于二维码的代码是用的第三方库. 首先是整体的结构: 注意下面的库文件一个都不能少,否则会报错. TLTiltHighlight ...

  6. iOS开发之二维码扫描

    二维码扫描 01-导入系统库 02 新建继承自UIView的 LHQPreView 2.1导入系统库头文件 #import <AVFoundation/AVFoundation.h> 2. ...

  7. IOS开发---菜鸟学习之路--(二十)-二维码扫描功能的实现

    本章将讲解如何实现二维码扫描的功能 首先在github上下载ZBar SDK地址https://github.com/bmorton/ZBarSDK 然后将如下的相关类库添加进去 AVFoundati ...

  8. iOS开发技术 - 二维码扫描、生成

    QRecLevel:QR_ECLEVEL_H // 二维码容错率,最高为30%(即QR_ECLEVEL_H),即LOGO有大                                       ...

  9. [Unity3D]自制UnityForAndroid二维码扫描插件

    一周左右终于将二维码生成和扫描功能给实现了,终于能舒缓一口气了,从一开始的疑惑为啥不同的扫码客户端为啥扫出来的效果不同?通用的扫描器扫出来就是一个下载APK,自制的扫描器扫出来是想要的有效信息,然后分 ...

随机推荐

  1. 精尽 MyBatis 源码分析 - MyBatis 初始化(一)之加载 mybatis-config.xml

    该系列文档是本人在学习 Mybatis 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释(Mybatis源码分析 GitHub 地址.Mybatis-Spring 源码分析 GitHub ...

  2. 巧妙使用MindManager图像功能,能够让你的思维导图更精彩

    MindManager是一款多功能思维导图工具软件,有其他软件无法媲美的项目管理和商业规划高级功能.用户们制作思维导图时一定要注意图文并茂,单纯的文字会过于单调.所以接下来,小编就为大家详细介绍Min ...

  3. C3PO数据库连接池

    1 <?xml version="1.0" encoding="UTF-8"?> 2 3 -<c3p0-config> 4 5 6 -& ...

  4. kafka producer 概要(看源码前,最好能掌握)

        kafakproducer概要(看源码前,最好能理解) 摘要 kafak 被设计用来作为一个统一的平台来处理庞大的数据的实时工具,在设计上有诸多变态的要求 它必须具有高吞吐量才能支持大量事件流 ...

  5. Java反射——java.lang.Class和类的加载

    反射的基础: java.lang.Class Class类的实例对象,用于记录类描述信息. 源码说:represent classes and interfaces in a running Java ...

  6. JQuery案例:折叠菜单

    折叠菜单(jquery) <html> <head> <meta charset="UTF-8"> <title>accordion ...

  7. 导出mysql内数据 python建倒排索引

    根据mysql内数据,python建倒排索引,再导回mysql内. 先把mysql内的数据导出,先导出为csv文件,因为有中文,直接打开csv文件会乱码,再直接改文件的后缀为txt,这样打开时不会是乱 ...

  8. 经典面试题解析:这道 C 编程面试题居然有如此多的解法!

    问题描述 任意给定一个32位无符号整数n,求n的二进制表示中1的个数,比如n = 5(0101)时,返回2,n = 15(1111)时,返回4 这也是一道比较经典的题目了,相信不少人面试的时候可能遇到 ...

  9. 创建topic

    sh kafka-topics.sh --create --zookeeper localhost:2181 --replication-factor 1 --partitions 1 --topic ...

  10. 机场&代理商-关系图

    机场&代理商-关系图 思路 ①首先统计机场活跃度Top10的机场名称,以下是我的表结构,以及查询语句 表结构: 查询语句:SELECT * from 2020csale ORDER BY cn ...