继上一篇高仿QQ即时聊天软件开发系列之一开端之后,开始做登录窗口

废话不多说,先看效果,只有界面

可能还有一些细节地方没有做,例如那个LOGO嘛,不要在意这些细节

GIF虽短,可是这做起来真难,好吧因为我没玩过WPF所以难,因为感觉做出来之后也就那样

整体布局

整体是上下分,下面是左中右分

 <Grid>
<Grid.RowDefinitions>
<RowDefinition Height="27"><!--用于放窗口右上角关闭、最小化按钮--></RowDefinition>
<RowDefinition Height="153"><!--和上一行共同用于放LOGO--></RowDefinition>
<RowDefinition Height="150"><!--窗口下半部分--></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="340"><!--占地方--></ColumnDefinition>
<ColumnDefinition><!--放最小化关闭按钮--></ColumnDefinition>
</Grid.ColumnDefinitions>
<Grid Background="#EBF2F9" Grid.Row="2" Grid.ColumnSpan="2">
<Grid.RowDefinitions>
<RowDefinition Height="10"></RowDefinition>
<RowDefinition Height="80"><!--窗口左下方的图片--></RowDefinition>
<RowDefinition><!--登录按钮--></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="30"></ColumnDefinition>
<ColumnDefinition Width="80">
<!--窗口左下方的图片-->
</ColumnDefinition>
<ColumnDefinition><!--用户名密码输入框--></ColumnDefinition>
<ColumnDefinition Width="80"><!--注册账号找回密码--></ColumnDefinition>
<ColumnDefinition Width="30"></ColumnDefinition>
</Grid.ColumnDefinitions>
</Grid>
</Grid>

整体布局

无边框

这个很简单,只需设置WindowStyle就行

 <Window xmlns:xctk="http://schemas.xceed.com/wpf/xaml/toolkit"  xmlns:my="http://schemas.microsoft.com/wpf/2008/toolkit"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
x:Class="XC.LoginWindow"
Title="LoginWindow" Height="330" WindowStyle="None" ResizeMode="NoResize" Width="421" Icon="Resources/images/app_big.ico" WindowStartupLocation="CenterScreen" MouseDown="Window_MouseDown" Name="login" Loaded="login_Loaded">
</Window>

无边框

右上角关闭最小化按钮、可拖动效果

代码都比较简单,认真看一下都能看懂,不过多解释

 <StackPanel Panel.ZIndex="1" Grid.Column="1" Orientation="Horizontal" FlowDirection="RightToLeft">
<Button Name="btnClose" Click="btnClose_Click" Width="30" Style="{DynamicResource styleBtnClose}"></Button>
<Button Name="btnMin" Click="btnMin_Click" Width="30" Style="{DynamicResource styleBtnMin}"></Button>
</StackPanel>
<Style x:Key="styleBtnClose" TargetType="Button">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<StackPanel>
<Image Name="imgNormal" Source="Resources/images/sysbtn_close_normal.png" Visibility="Visible"></Image>
<Image Name="imgHover" Source="Resources/images/sysbtn_close_hover.png" Visibility="Collapsed"></Image>
<Image Name="imgDown" Source="Resources/images/sysbtn_close_down.png" Visibility="Collapsed"></Image>
</StackPanel>
<ControlTemplate.Triggers>
<Trigger Property="Button.IsMouseOver" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgNormal" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgHover" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
<Trigger Property="Button.IsPressed" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgHover" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgDown" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
<Style x:Key="styleBtnMin" TargetType="Button">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<StackPanel>
<Image Name="imgNormal" Source="Resources/images/sysbtn_min_normal.png" Visibility="Visible"></Image>
<Image Name="imgHover" Source="Resources/images/sysbtn_min_hover.png" Visibility="Collapsed"></Image>
<Image Name="imgDown" Source="Resources/images/sysbtn_min_down.png" Visibility="Collapsed"></Image>
</StackPanel>
<ControlTemplate.Triggers>
<Trigger Property="Button.IsMouseOver" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgNormal" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgHover" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
<Trigger Property="Button.IsPressed" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgHover" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgDown" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
  private void btnClose_Click(object sender, RoutedEventArgs e)
{
Close();
} private void btnMin_Click(object sender, RoutedEventArgs e)
{
WindowState = System.Windows.WindowState.Minimized;
} private void Window_MouseDown(object sender, MouseButtonEventArgs e)
{
if(e.LeftButton== MouseButtonState.Pressed)
{
DragMove();
}
}
密码输入框

做到这个地方的时候,我就一直在想,微软你就不能支持一下placeholder效果吗?

为了能实现placeholder效果,疯狂谷歌,选择了一种看起来最简单的方案

 <Border CornerRadius="0,0,4,4" Grid.Row="1"  BorderBrush="#B5B5B5" BorderThickness="1,1,1,1">
<PasswordBox Margin="1" KeyUp="pbPassword_KeyUp" Name="pbPassword" BorderBrush="Transparent" Grid.Row="1" VerticalContentAlignment="Center" Background="{DynamicResource burshPassword}">
</PasswordBox>
</Border>
<VisualBrush x:Key="burshPassword" AlignmentX="Left" AlignmentY="Center" Stretch="None">
<VisualBrush.Visual>
<Label Content="密码" Foreground="Gray" />
</VisualBrush.Visual>
</VisualBrush>

最终代码就这么几行,可当时找这个方案差点没气死

解释一下,那Border是用于让密码框的边框为圆角,CornerRadius是指定圆角的半径,密码框只需要下边两个角是圆角,所以像代码中那样写。

重点在Background,这个是实现placeholder的关键。代码中为Background指定了一个动态资源,Key为brushPassword,而下面定义了这个资源也就是VisualBrush,这个东东可以把一个控件显示成为另一个控件的背景,像上面那样,把Label控件显示成为了PasswordBox的背景,然后Label上写个“密码”,设置一下颜色,placeholder效果就实现了

但这还不够,因为这样的话就算用户输入了字符,那个背景仍然会存在,所以

 private void pbPassword_KeyUp(object sender, KeyEventArgs e)
{
if (string.IsNullOrEmpty(pbPassword.Password))
{
pbPassword.Background = (VisualBrush)Resources["burshPassword"];
}
else
{
pbPassword.Background = null;
}
}

隐藏背景

这代码就简单多了,不多解释。这样,一个很简单的placeholder方案就出来了。在谷歌上面找的其他方案看着就晕了,不过下面这个还真的要晕过去。

用户名/邮箱输入组合框

这个是登录框的重头戏,因为ComboBox居然不支持Background属性,设置了半天一点效果都没有,我#@*&$^#@,吐槽完毕,详见另一篇文章高仿QQ即时聊天软件开发系列之三登录窗口用户选择下拉框

登录按钮
 <Button Name="btnLogin" Click="btnLogin_Click" Content="登录" Grid.Column="" Grid.Row="" Style="{DynamicResource styleBtnLogin}" Margin="0,13,0,0"></Button>
<Style x:Key="styleBtnLogin" TargetType="Button">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<StackPanel>
<Grid Name="imgNormal">
<TextBlock Text="登 录" VerticalAlignment="Center" HorizontalAlignment="Center" Foreground="White" Panel.ZIndex="">
</TextBlock>
<Image Panel.ZIndex="" Source="Resources/images/button_login_normal.png" Visibility="Visible"></Image>
</Grid>
<Grid Name="imgHover" Visibility="Collapsed">
<TextBlock Text="登 录" VerticalAlignment="Center" HorizontalAlignment="Center" Foreground="White" Panel.ZIndex="">
</TextBlock>
<Image Panel.ZIndex="" Source="Resources/images/button_login_hover.png" Visibility="Visible"></Image>
</Grid>
<Grid Name="imgDown" Visibility="Collapsed">
<TextBlock Text="登 录" VerticalAlignment="Center" HorizontalAlignment="Center" Foreground="White" Panel.ZIndex="">
</TextBlock>
<Image Panel.ZIndex="" Source="Resources/images/button_login_down.png" Visibility="Visible"></Image>
</Grid>
</StackPanel>
<ControlTemplate.Triggers>
<Trigger Property="Button.IsMouseOver" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgNormal" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgHover" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
<Trigger Property="Button.IsPressed" Value="True">
<Trigger.Setters>
<Setter Property="Visibility" TargetName="imgHover" Value="Collapsed"></Setter>
<Setter Property="Visibility" TargetName="imgDown" Value="Visible"></Setter>
</Trigger.Setters>
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>

登录按钮

其他
<StackPanel Grid.Column="" Grid.Row="">
<TextBlock Margin="0,8,0,0" Height="" VerticalAlignment="Center">
<Hyperlink TextDecorations="None">
注册账号
</Hyperlink>
</TextBlock>
<TextBlock Height="">
<Hyperlink TextDecorations="None">
找回密码
</Hyperlink>
</TextBlock>
</StackPanel>

好了,整个界面就是这样了,除了用户组合框有点难(其实我感觉是特别难)以外,其他的都没什么。图片啥的到QQ安装路径找rdb文件解包吧

高仿QQ即时聊天软件开发系列之二登录窗口界面的更多相关文章

  1. 高仿QQ即时聊天软件开发系列之三登录窗口用户选择下拉框

    上一篇高仿QQ即时聊天软件开发系列之二登录窗口界面写了一个大概的布局和原理 这一篇详细说下拉框的实现原理 先上最终效果图 一开始其实只是想给下拉框加一个placeholder效果,让下拉框在未选择未输 ...

  2. 高仿QQ即时聊天软件开发系列之一开端

    前段时间在园子里看到一个大神做了一个GG2014IM软件,仿QQ的,那感觉···,赶快下载源码过来试试,还真能直接跑起来,效果也不错.但一看源码,全都给封装到了ESFramework里面了,音视频那部 ...

  3. SignalR快速入门 ~ 仿QQ即时聊天,消息推送,单聊,群聊,多群公聊(基础=》提升)

     SignalR快速入门 ~ 仿QQ即时聊天,消息推送,单聊,群聊,多群公聊(基础=>提升,5个Demo贯彻全篇,感兴趣的玩才是真的学) 官方demo:http://www.asp.net/si ...

  4. 仿QQ局域网聊天软件

    1 目的   想复习一下TCP/IP协议,再结合一下以前学的Qt的知识,加上前段时间学的MySQL数据库操作,所以写了个"仿QQ局域网聊天软件"小项目,只实现了一部分功能,还没写完 ...

  5. Android 高仿微信即时聊天 百度云为基础的推

    转载请注明出处:http://blog.csdn.net/lmj623565791/article/details/38799363 ,本文出自:[张鸿洋的博客] 一直在仿微信界面,今天最终有幸利用百 ...

  6. “仿QQ局域网聊天软件”项目-常用编程技巧总结

    1 信号槽篇 qqLogin loginDialog; QQ mainDialog; loginDialog.show(); //连接登陆窗口和主窗口 QObject::connect(&lo ...

  7. 高仿QQ的即时通讯应用带服务端软件安装

    Android 基于xmpp协议,smack包,openfire服务端(在下面)的高仿QQ的即时通讯实现.实现了注册,登录,读取好友列表,搜索好友,添加分组,添加好友,删除好友,修改心情,两个客户端之 ...

  8. 高仿qq聊天界面

    高仿qq聊天界面,给有需要的人,界面效果如下: 真心觉得做界面非常痛苦,给有需要的朋友. chat.xml <?xml version="1.0" encoding=&quo ...

  9. 基于环信的仿QQ即时通讯的简单实现

    代码地址如下:http://www.demodashi.com/demo/11645.html 我的博客地址 之前一直想实现聊天的功能,但是感觉有点困难,今天看了环信的API,就利用下午的时间动手试了 ...

随机推荐

  1. sitecore(key\value\language)的灵活应用

    1.当我们在做网站的时候是否会因为一个页面的文字变动来回改变.这样的麻烦sitecore都帮我们解决了. 2.sitecore分类key和value和语言几个维度.不同的key会因为不同的语言显示不同 ...

  2. android popupwindow低版本报空指针

    在项目中使用Popupwindow pop=new Popupwindow();在2.3版本报 异常信息: Exception: null 堆栈信息: android.widget.PopupWind ...

  3. Ubuntu12.04 下安装Chrome浏览器

    第一步 下载. 32位:https://dl.google.com/linux/direct/google-chrome-stable_current_i386.deb 64位:https://dl. ...

  4. 指定Action、Category调用系统Activity

    1.Intent对象详解 Android的应用程序包含三种重要组件:Activity.Service.BroadcastReceiver,应用程序采用一致的方式来启动它们----都是依靠Intent来 ...

  5. Retina 显示屏

    Retina 直接翻译是视网膜的意思.在IT上,是Apple 公司提出的.意思是指一个显示屏的颗粒度 px 密度高到人类无法看见.要了解细节必须先了解基础知识inch 英寸 1 inch = 2.52 ...

  6. poj 1819 Disks

    http://poj.org/problem?id=1819 #include <cstdio> #include <cstring> #include <cmath&g ...

  7. Android 添加子视图(addView和setView)

    我们在添加视图文件的时候有两种方式,一种是通过在xml文件定义layout,另一种方式是在java代码中动态生成布局文件. 在xml中定义的layout要想转化为view,需要使用到LayoutInf ...

  8. Smarty include使用

    {include} {include}用于载入其他模板到当前模板中. 在包含模板中可用的变量,载入后在当前模板仍然可用. {include}必须设置file 属性,设置载入的文件资源路径. 设置了可选 ...

  9. cocos2d-x 找不到资源文件问题

    问题描述: 在项目中引用到了图片,但是运行时报错: Unhandled exception at 0x001049DE in hello.exe: 0xC0000005: Access violati ...

  10. poj2409 & 2154 polya计数+欧拉函数优化

    这两个题都是项链珠子的染色问题 也是polya定理的最基本和最经典的应用之一 题目大意: 用m种颜色染n个珠子构成的项链,问最终形成的等价类有多少种 项链是一个环.通过旋转或者镜像对称都可以得到置换 ...