给 Xamarin.Form For Windows Phone APP 加个漂亮的 "头"
Windows Phone 是那个1%, 我也是那个1%, 不喜勿喷.
WP 向来给 android / ios 的粉们一个最直观的印象: 丑.
其实"丑"这个东西会一直下去,而且是个解不开的死循环.
WP 下, 做的华丽的, 目前我用过的只有少数几个, 网易云音乐 和 智机社区, 做的漂亮而且流畅, 其它的只能用渣渣来形容.




Xamarin.Form 的 APP 在 Android 和 IOS 下默认都会有一个"头", 用于显示页面名称和导航按钮.
在 WP 下却没有, 像这样:

Xamarin.Form 在 WP 的 MainPage.xaml.cs 中这样定义:
public partial class MainPage : FormsApplicationPage{
...
public MainPage() {
InitializeComponent();
SupportedOrientations = SupportedPageOrientation.PortraitOrLandscape;
global::Xamarin.Forms.Forms.Init();
LoadApplication(new LBC.Mobi.App());
}
FormsApplicationPage.LoadApplication 方法的定义:
protected void LoadApplication(Xamarin.Forms.Application application)
{
Xamarin.Forms.Application.Current = application;
application.PropertyChanged += new PropertyChangedEventHandler(this.ApplicationOnPropertyChanged);
this.application = application;
application.SendStart();
this.SetMainPage();
}
private void SetMainPage()
{
if (this.platform == null)
this.platform = new Platform((PhoneApplicationPage) this);
this.platform.SetPage(this.application.MainPage);
if (this.Content == this.platform)
return;
this.Content = (UIElement) this.platform;
}
这样一个调用链:
LoadApplication -> SetMainPage -> 直接给整个页面的 Content 赋值.
所以, 无论你在 MainPage.xaml 中写什么内容, 最终都是无法显示的.
为了达成目的, 我们需要修改 SetMainPage 方法 , 使 this.platform 只作为 MainPage 的一部分, 而不是整个 Content.
但是反编译一下, 可以看到 Xamarin.Form 的封装很蛋疼, 很多都是 internal , private, 不带 virtual, 所以, 通过正常手段是无法对 Xamarin.Form 进行扩展的.

还好, Xamarin 对反射支持的很到位, 无法正常通过继承/重写来实现, 我们照样可以用反射来完成它.
1, 在 MainPage.xaml.cs 中新增方法 LoadApplication:
new protected void LoadApplication(Xamarin.Forms.Application application) {
//Xamarin.Forms.Application.Current = application;
typeof(Xamarin.Forms.Application).GetProperty("Current", BindingFlags.Static | BindingFlags.Public)
.SetValue(Xamarin.Forms.Application.Current, application);
application.PropertyChanged += new PropertyChangedEventHandler(this.ApplicationOnPropertyChanged);
typeof(FormsApplicationPage).GetField("application", BindingFlags.NonPublic | BindingFlags.Instance)
.SetValue(this, application);
//application.SendStart();
application.GetType().GetMethod("SendStart", BindingFlags.NonPublic | BindingFlags.Instance)
.Invoke(application, null);
this.SetMainPage();
17 var mp = (Xamarin.Forms.NavigationPage)application.MainPage;
18 mp.PropertyChanged += MainPage_PropertyChanged;
19 this.Title = mp.Title;
}
为了能显示当前页的标题, 我在上面的方法中加了一段(红色标注), 实际中, 你的 application.MainPage 可能不是 NavigationPage, 需要自行修改.
2,
void MainPage_PropertyChanged(object sender, PropertyChangedEventArgs e) {
if (e.PropertyName.Equals("CurrentPage")) {
this.Title = ((Xamarin.Forms.NavigationPage)sender).CurrentPage.Title;
this.PropertyChanged(this, new PropertyChangedEventArgs("Title"));
}
}
3, 在 MainPage.xaml.cs 中添加 SetMainPage :
private void SetMainPage() {
if (this.Platform == null) {
this.Platform = new Platform((PhoneApplicationPage)this);
}
this.Platform.SetPage(Xamarin.Forms.Application.Current.MainPage);
if (this.mainBody.Content != null && this.mainBody.Content.Equals(this.Platform))
return;
this.mainBody.Content = (UIElement)this.Platform;
}
4, 修改 MainPage.xaml
<winPhone:FormsApplicationPage
x:Class="Discuz.WinPhone.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:phone="clr-namespace:Microsoft.Phone.Controls;assembly=Microsoft.Phone"
xmlns:shell="clr-namespace:Microsoft.Phone.Shell;assembly=Microsoft.Phone"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:winPhone="clr-namespace:Xamarin.Forms.Platform.WinPhone;assembly=Xamarin.Forms.Platform.WP8"
xmlns:local="clr-namespace:Discuz.WinPhone"
mc:Ignorable="d"
FontFamily="{StaticResource PhoneFontFamilyNormal}"
FontSize="{StaticResource PhoneFontSizeNormal}"
Foreground="{StaticResource PhoneForegroundBrush}"
SupportedOrientations="Portrait" Orientation="Portrait"
shell:SystemTray.IsVisible="True"
shell:SystemTray.Opacity="0"
Background="#1e5263"
>
<!--shell:SystemTray.BackgroundColor="#2ba9d3"-->
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="15" /><!-- for SystemTray -->
<RowDefinition Height="auto" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<local:TilePanel TileWidth="5" TileHeight="5" Height="65" Grid.Row="0" Grid.RowSpan="2">
<local:TilePanel.Image>
<ImageBrush ImageSource="texture.png" />
</local:TilePanel.Image>
</local:TilePanel>
<StackPanel Orientation="Horizontal" Margin="10,0" Grid.Row="1" VerticalAlignment="Center">
<Image Source="icon.png" Width="40" Height="40" Margin="0,0,10,0" />
<TextBlock Text="{Binding Title}" VerticalAlignment="Center" />
</StackPanel>
<ContentPresenter x:Name="mainBody" Grid.Row="2" />
</Grid>
</winPhone:FormsApplicationPage>
好啦, 一个带标题的 XF For WP 的 APP 改造完了.
最终效果:


具体参见:
https://github.com/gruan01/Discuz.Mobi/tree/master/Discuz/Discuz.WinPhone
--------------------
OK, 完
给 Xamarin.Form For Windows Phone APP 加个漂亮的 "头"的更多相关文章
- FlipView For Xamarin.Form 之 IOS
之前写过两篇博文 是关于 Android 和 Windows Phone 下的 FlipView 的实现. 上上周,有个印度佬通过 GitHub 找到我, 问我有没有打算个 ios 端的,还说比较了相 ...
- Xamarin.Form 实例: Discuz BBS 客户端 源码分享
感谢台风, 这个十一长假让我好好的休息了一回, 睡觉到腰酸背疼, 看电影看到眼发红. 今天最后一天, 不敢出去逛, 不知道哪会还会下暴雨... 嗯嗯..这个项目其实在十一之前就开始了, 工作无聊,没有 ...
- Visual Studio + C# + Xamarin = iOS/Android/Windows Apps
Visual Studio 跨平台開發實戰 (1) -- Hello Xamarin! 前言 應用程式發展的腳步,從來沒有停過.從早期的 Windows 應用程式, 到網路時代的 web 應用程式,再 ...
- Windows Phone App的dump 文件分析
前言 我们在发布了自己的App以后,Windows Phone的Error Report机制会帮助我们收集程序的崩溃信息并发送到微软的服务器上,这可以辅助开发者提高App的稳定性. 那么如何利用这些d ...
- [Windows Phone] APP上架,遇到错误2001的解决方案。(Error:2001)
[Windows Phone] APP上架,遇到错误2001的解决方案.(Error:2001) 问题情景 最近在开始玩Windows Phone的开发,开发的过程中虽然有点小挫折,但是参考网络许多前 ...
- Xamarin Android Fragment的两种加载方式
android Fragment的重点: 3.0版本后引入,即minSdk要大于11 Fragment需要嵌套在Activity中使用,当然也可以嵌套到另外一个Fragment中,但这个被嵌套的Fra ...
- Xamarin.Form怎么调用原生方法
---恢复内容开始--- Xamarin.Form怎么调用原生包 今天我想和大家分享的是有关Xamarin如何调用安卓的原生代码,下面的例子以大家可能会经常用到的微信WX方法的调用. 首先我们新建一个 ...
- Xamarin.Form与Xamarin.Android或Xamarin.IOS的区别简述
Xamarin.Form与Xamarin.Android或Xamarin.IOS的区别简述: 可能刚刚接触Xamarin的人来说,对于这个概念比较的模糊,认为这说的不都是同一个东西吗?事实并不是这样的 ...
- 在桌面程序上和Metro/Modern/Windows store app的交互(相互打开,配置读取)
这个标题真是取得我都觉得蛋疼..微软改名狂魔搞得我都不知道要叫哪个好.. 这边记录一下自己的桌面程序跟windows store app交互的过程. 由于某些原因,微软的商店应用的安全沙箱导致很多事情 ...
随机推荐
- oracle 得到新插入数据的ID并使用
DECLARE newID varchar2(50);begininsert into table1 (aa,bb) values('7777','8888') RETURNING ID INTO ...
- poj 3159 Candies 差分约束
Candies Time Limit: 1500MS Memory Limit: 131072K Total Submissions: 22177 Accepted: 5936 Descrip ...
- fcntl 获取文件状态标志
int fcntl(int fd,int cmd,...) 函数fcntl提供了非常丰富的功能.主要依赖于cmd的各种参数: 复制已有的文件描述符 F_DUPFD,F_DUPFD_CLOEXEC 获取 ...
- [转]响应式网页设计:rem、em设置网页字体大小自适应
本文转自:http://www.cnblogs.com/aimyfly/archive/2013/07/19/3200742.html 「rem」是指根元素(root element,html)的字体 ...
- JVM的内存区域划分
JVM的内存区域划分 学过C语言的朋友都知道C编译器在划分内存区域的时候经常将管理的区域划分为数据段和代码段,数据段包括堆.栈以及静态数据区.那么在Java语言当中,内存又是如何划分的 ...
- hdu1042 N!
/* N! Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others) Total Subm ...
- Linux 系统常用命令汇总(五) 磁盘管理
磁盘管理 命令 选项 注解 示例 df [选项] 显示磁盘空间使用情况 显示磁盘空间是员工情况,以M显示: df -m -i 使用inodes显示结果 -k(m) 使用KB(MB)显示结果 du ...
- 如此低价的ZBrush,你能想象?
作为3D艺术的狂热者,你是否曾为找不到一款适合自己的雕刻软件而苦恼?要么,你已经找到了,却因为昂贵的价格而迟迟不肯入手? 作为改变整个三维行业的业界先进的数字雕刻和绘画软件,ZBrush向来拥有广 ...
- ZBrush中的SubTool工具该怎样使用
今天的ZBrush教程中将为大家引入一个新的工具SubTool,使用SubTool您可以添加PolyMesh至当前编辑的模型中,它的出现改变了过去ZBrush不能同时编辑多个模型的弊端. 查看详细的视 ...
- HDU 5105 Math Problem --数学,求导
官方题解: f(x)=|a∗x3+b∗x2+c∗x+d|, 求最大值.令g(x)=a∗x3+b∗x2+c∗x+d,f(x)的最大值即为g(x)的正最大值,或者是负最小值.a!=0时, g′(x)=3∗ ...