Duilib界面解析1---简单的仿360窗口

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 最近开始研究Duilib,并逐个分析Duilib的demo例子,记录下来方便更多的人学习 。界面运行起来如下图所示,该Demo主要包含 BaseDialog.h和BaseDialog.cpp,以及界面布局文件的详细解释,每行注释已经添加。
最近开始研究Duilib,并逐个分析Duilib的demo例子,记录下来方便更多的人学习 。界面运行起来如下图所示,该Demo主要包含 BaseDialog.h和BaseDialog.cpp,以及界面布局文件的详细解释,每行注释已经添加。



BaseDialog头文件源文件 以及布局文件如下,可自行下载Duilib 根据注释学习该例子 只针对新手~~~

#ifndef BaseDialog_H
#define BaseDialog_H

#include "UIlib.h"

#include <map>
using namespace DuiLib;

//继承自CWindowWnd  INotifyUI  
//拥有创建窗口和接受窗口事件通知的能力
class BaseDialog : public CWindowWnd, public INotifyUI
{

public:
	BaseDialog(){};
protected:
	//虚函数 用于设置窗口的CLASS NAME
	virtual LPCTSTR GetWindowClassName() const { return _T("USHER TEST DLG"); }
	//用于时间通告消息回调 
	virtual void Notify(TNotifyUI& msg);
	//处理
	virtual LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam);
	// 自定义消息映射
	typedef HRESULT (BaseDialog::*CustomMsgHandler)(WPARAM, LPARAM, BOOL&);
	typedef std::map<UINT, CustomMsgHandler> MessageMap;
	virtual MessageMap* InitMessageMap();
	// 自定义消息处理 在窗口创建的时候
	HRESULT OnCreate(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//在窗口销毁的时候
	HRESULT OnDestory(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//擦除窗口背景
	HRESULT OnErasebkgnd(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//尺寸改变的时候
	HRESULT OnSize(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//非客户区重绘
	HRESULT OnNcPaint(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//非客户区激活
	HRESULT OnNcActive(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//非客户区计算大小
	HRESULT OnNcCalSize(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//非客户区点击测试
	HRESULT OnNcHitTest(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//系统命令处理
	LRESULT OnSysCommand(WPARAM wParam, LPARAM lParam, BOOL& bHandled);
	//最大最小化信息
	LRESULT OnGetMinMaxInfo(WPARAM wParam, LPARAM lParam, BOOL& bHandled);

private:
	CPaintManagerUI m_pm;
};
#endif // BaseDialog_H


#include "StdAfx.h"
#include "BaseDialog.h"

#include <memory>


//消息通告
void BaseDialog::Notify(TNotifyUI& msg)
{   
	 //如果是点击消息那么通过控件名字判断是哪个控件
	if ( msg.sType == _T("click"))
	{
		if( msg.pSender == static_cast<CButtonUI*>(m_pm.FindControl(_T("minbtn"))) )
			SendMessage(WM_SYSCOMMAND, SC_MINIMIZE, 0);

		if( msg.pSender == static_cast<CButtonUI*>(m_pm.FindControl(_T("closebtn"))) )
			PostQuitMessage(0);

		if( msg.pSender == static_cast<CButtonUI*>(m_pm.FindControl(_T("maxbtn"))) )
			::IsZoomed(*this) ? SendMessage(WM_SYSCOMMAND, SC_RESTORE, 0) : SendMessage(WM_SYSCOMMAND, SC_MAXIMIZE, 0);
		if (msg.pSender == static_cast<CButtonUI*>(m_pm.FindControl(_T("check_normal")))) {
			::MessageBox(NULL, L"开始扫描", L"提示", MB_OK);
		}
	}
}

//首先启动消息循环会进入此虚函数进行消息处理 
LRESULT BaseDialog::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{  
	
	//初始化消息映射MAP  利用auto_ptr维护指针 static保证只创建一次
	static std::auto_ptr<MessageMap> customMessageMap(InitMessageMap());

	BOOL bHandled = TRUE;
	LRESULT lRes = 0;

	//将消息在消息映射map中进行查找 找到响应的消息处理函数
	if ( customMessageMap->find(uMsg) != customMessageMap->end() )
	{   
		//typedef HRESULT (BaseDialog::*CustomMsgHandler)(WPARAM, LPARAM, BOOL&);  
		//如果找到 查找响应的消息响应函数
		CustomMsgHandler handler = (*customMessageMap)[uMsg];
		//通过this->(*handler)进行消息响应函数的调用
		lRes = (this->*handler)(wParam, lParam, bHandled);
		//如果 bHandled返回True没有被修改那么说明消息已经被处理 返回
		if ( bHandled ) return lRes;
	}
	//CPaintManagerUI丢给PaintManagerUI进行处理 如果处理了 那么会返回True 否则返回false继续走
	if( m_pm.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;
	//最后丢给默认的windows消息处理函数
	
	return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
}

//初始化消息循环对应的消息响应函数
BaseDialog::MessageMap* BaseDialog::InitMessageMap()
{
	MessageMap* map = new MessageMap;
	(*map)[WM_CREATE] = &BaseDialog::OnCreate;
	(*map)[WM_DESTROY] = &BaseDialog::OnDestory;
	(*map)[WM_ERASEBKGND] = &BaseDialog::OnErasebkgnd;
	(*map)[WM_SIZE] = &BaseDialog::OnSize;
	(*map)[WM_NCPAINT] = &BaseDialog::OnNcPaint;
	//以下三个消息用于屏蔽系统标题栏
	(*map)[WM_NCACTIVATE] = &BaseDialog::OnNcActive;
	(*map)[WM_NCCALCSIZE] = &BaseDialog::OnNcCalSize;
	(*map)[WM_NCHITTEST] = &BaseDialog::OnNcHitTest;
	(*map)[WM_SYSCOMMAND] = &BaseDialog::OnSysCommand;
	(*map)[WM_GETMINMAXINFO] = &BaseDialog::OnGetMinMaxInfo;
	return map;
}

//窗口创建时候
HRESULT BaseDialog::OnCreate( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{
	//获取当前窗口风格
	LONG styleValue = ::GetWindowLong(*this, GWL_STYLE);
	styleValue &= ~WS_CAPTION;
	//设置STYLE
	::SetWindowLong(*this, GWL_STYLE, styleValue | WS_CLIPSIBLINGS | WS_CLIPCHILDREN);
	//初始化界面渲染器
	m_pm.Init(m_hWnd); 
	//D
	CDialogBuilder builder;
	//通过xml 以及渲染器渲染界面UI
	CControlUI* pRoot = builder.Create(_T("main_dlg.xml"), (UINT)0, NULL, &m_pm);
	//附加界面UI到对话框容器 
	m_pm.AttachDialog(pRoot);
	//增加消息处理  因为实现了 INotifyUI接口
	m_pm.AddNotifier(this);
	return 0;
}


HRESULT BaseDialog::OnDestory( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{
    //发送退出消息
	::PostQuitMessage(0L);
	return 0;
}

//擦除背景
HRESULT BaseDialog::OnErasebkgnd( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{ 
	return 1;
}

//当窗口的尺寸发生改编的时候
HRESULT BaseDialog::OnSize( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{   
	//在窗口大小改变的时候设置圆角
	SIZE szRoundCorner = m_pm.GetRoundCorner();
	if( !::IsIconic(*this) && (szRoundCorner.cx != 0 || szRoundCorner.cy != 0) ) 
	{
		CRect rcWnd;
		::GetWindowRect(*this, &rcWnd);
		rcWnd.Offset(-rcWnd.left, -rcWnd.top);
		rcWnd.right++; 
		rcWnd.bottom++;
		HRGN hRgn = ::CreateRoundRectRgn(rcWnd.left, rcWnd.top, rcWnd.right, 
			rcWnd.bottom, szRoundCorner.cx, szRoundCorner.cy);
		::SetWindowRgn(*this, hRgn, TRUE);
		::DeleteObject(hRgn);
		return 0;
	}
	bHandled = FALSE;
	return 0;
}

HRESULT BaseDialog::OnNcPaint( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{
	return 0;
}

HRESULT BaseDialog::OnNcActive( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{
	if( ::IsIconic(*this) ) bHandled = FALSE;
	return (wParam == 0) ? TRUE : FALSE;
}


//如果不处理那么就会导致DUILIB 不停调用系统消息进行处理
// 屏蔽系统标题栏 似乎不屏蔽一定会出问题
HRESULT BaseDialog::OnNcCalSize( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{
	return 0;
}

HRESULT BaseDialog::OnNcHitTest( WPARAM wParam, LPARAM lParam, BOOL& bHandled )
{  
	//获取客户区相对坐标
	POINT pt; 
	pt.x = GET_X_LPARAM(lParam); 
	pt.y = GET_Y_LPARAM(lParam);
	::ScreenToClient(*this, &pt);

	//获取客户区域
	RECT rcClient;
	::GetClientRect(*this, &rcClient);

	//如果窗口没有最大化
	if( !::IsZoomed(*this) ) 
	{  
		//获取非客户区域sizebox
		RECT rcSizeBox = m_pm.GetSizeBox();
		//如果y<窗口区域top+sizebox.top 算上非客户区
		if( pt.y < rcClient.top + rcSizeBox.top ) 
		{  
			//判断是否在左上边
			if( pt.x < rcClient.left + rcSizeBox.left ) return HTTOPLEFT;
			//判断是否在右上边
			if( pt.x > rcClient.right - rcSizeBox.right ) return HTTOPRIGHT;
			//返回顶端测试
			return HTTOP;
		}
		//否则在bottom
		else if( pt.y > rcClient.bottom - rcSizeBox.bottom ) 
		{   
			//左下
			if( pt.x < rcClient.left + rcSizeBox.left ) return HTBOTTOMLEFT;
			//右下
			if( pt.x > rcClient.right - rcSizeBox.right ) return HTBOTTOMRIGHT;
			//默认下边
			return HTBOTTOM;
		}
		//如果不再 top 或者bottom 那么就是在左边 右边的非客户区
		if( pt.x < rcClient.left + rcSizeBox.left ) return HTLEFT;
		if( pt.x > rcClient.right - rcSizeBox.right ) return HTRIGHT;
	}

	//获取标题栏的矩形区域  
	//并且判断鼠标是否在该区域中 如果在返回
	RECT rcCaption = m_pm.GetCaptionRect();
	if( pt.x >= rcClient.left + rcCaption.left 
		&& pt.x < rcClient.right - rcCaption.right
		&& pt.y >= rcCaption.top 
		&& pt.y < rcCaption.bottom ) 
	{      
		  //ButtonUI OptionUI 只有这两种类型当作标题栏
			CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(pt));
			if( pControl 
				&& _tcscmp(pControl->GetClass(), _T("ButtonUI")) != 0 
				&& _tcscmp(pControl->GetClass(), _T("OptionUI")) != 0 )
			{
				return HTCAPTION;
			}
	}
	//其余部分是客户区
	return HTCLIENT;
}


//系统命令处理
LRESULT BaseDialog::OnSysCommand(WPARAM wParam, LPARAM lParam, BOOL& bHandled)
{
	if( wParam == SC_CLOSE ) 
	{
		::PostQuitMessage(0L);
		bHandled = TRUE;
		return 0;
	}

	BOOL bZoomed = ::IsZoomed(*this);
	LRESULT lRes = CWindowWnd::HandleMessage(WM_SYSCOMMAND, wParam, lParam);

	return 1L;
}

LRESULT BaseDialog::OnGetMinMaxInfo(WPARAM wParam, LPARAM lParam, BOOL& bHandled)
{
	MONITORINFO oMonitor = {};
	oMonitor.cbSize = sizeof(oMonitor);
	::GetMonitorInfo(::MonitorFromWindow(*this, MONITOR_DEFAULTTOPRIMARY), &oMonitor);
	CRect rcWork = oMonitor.rcWork;
	rcWork.Offset(-rcWork.left, -rcWork.top);

	/// 窗口最大化时裁剪阴影所占区域
	LPMINMAXINFO lpMMI = (LPMINMAXINFO) lParam;
	lpMMI->ptMaxPosition.x	= rcWork.left-5;
	lpMMI->ptMaxPosition.y	= rcWork.top-3;
	lpMMI->ptMaxSize.x		= rcWork.right+10;
	lpMMI->ptMaxSize.y		= rcWork.bottom+10;

	bHandled = FALSE;
	return 0;
}


<?xml version="1.0" encoding="utf-8"?>
<Window size="910,610" sizebox="10,10,12,12" roundcorner="0,0" caption="0,0,0,90" mininfo="910,610" bktrans="true" shadow="true" shadowimage="shadow.png" shadowtopleft="0,0,5,3" shadowtopright="205,0,5,3" shadowbottomleft="0,136,5,7" shadowbottomright="205,136,5,7">
 <!--根据顺序记录字体可通过int 索引引用-->
  <Font name="微软雅黑" size="12" />
  <Font name="微软雅黑" size="16" />
  <Font name="微软雅黑" size="22" />
  <Font name="宋体" size="15" />
  <!--整体垂直布局-->
  <VerticalLayout   bkimage="file='bg.png'">
    <!--容器的内边距放置到一行  水平占满-->
    <HorizontalLayout height="40" inset="0,3,0,0">
      <!--titile标题的位置占位 根据内容决定大小  在一个水平布局中没有设置宽度的布局会默认填充所有内容-->
      <HorizontalLayout>
        <!--float true使用绝对定位  设置位置 文本颜色 字体索引0-->
        <Text text="360安全卫士9.2" pos="15, 10, 200, 50" float="true" textcolor="#FFFFFF" font="0" />
      </HorizontalLayout>
      <!--徽章位置宽度40 height 30-->
	    <HorizontalLayout width="40" height="30">
		   <Button maxwidth="26" inset="200,10,0,0" normalimage="medal.png"/>
	    </HorizontalLayout>
      <!---宽度是150的菜单栏-->
      <HorizontalLayout width="150">
        <!--设置按钮的 宽度 高度  普通图片  hover图片 按钮按下的图片-->
		   <Button name="skinbtn" maxwidth="27" maxheight="22" normalimage="skin_normal.png" hotimage="skin_hover.png" pushedimage="skin_pressed.png"/>
		   <Button name="feedbackbtn" maxwidth="27" maxheight="22" normalimage="feedback_normal.png" hotimage="feedback_hover.png" pushedimage="feedback_pressed.png"/>
		   <Button name="maxbtn" maxwidth="27" maxheight="22" normalimage="menu_normal.png" hotimage="menu_hover.png" pushedimage="menu_pressed.png"/>
		   <Button name="minbtn" maxwidth="27" maxheight="22" normalimage="min_normal.png" hotimage="min_hover.png" pushedimage="min_pressed.png"/>
       <Button name="closebtn" maxwidth="27" maxheight="22" normalimage="close_normal.png" hotimage="close_hover.png" pushedimage="close_pressed.png"/>
      </HorizontalLayout>
    </HorizontalLayout>
   <!--第二行工具按钮 -->
	<HorizontalLayout height="85">
		<HorizontalLayout>
      <!---前景背景图片都是相对坐标除非 float会变成绝对zu-->
			<Option pos="15,10,85,85" float="true" align="bottom" text="电脑体检"   textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_Examine.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'" selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="true"/>
			<Option pos="97,10,167,85" float="true" align="bottom" text="木马查杀" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_dsmain.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="179,10,249,85" float="true" align="bottom" text="系统修复" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_SysRepair.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="261,10,331,85" float="true" align="bottom" text="电脑清理" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_TraceCleaner.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="343,10,413,85" float="true" align="bottom" text="优化加速" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_SpeedupOpt.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="425,10,495,85" float="true" align="bottom" text="电脑专家" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_expert.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="507,10,577,85" float="true" align="bottom" text="电脑门诊" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_diannaomenzhen.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
			<Option pos="589,10,659,85" float="true" align="bottom" text="软件管家" textpadding="11,55,0,0" font="0" textcolor="#FFFFFF" selected="true" foreimage="file='ico_softmgr.png' dest='11,5,59,53'" normalimage="toolbar_normal.png" hotimage="file='toolbar_hover.png' source='80,0,160,75'" pushedimage="file='toolbar_hover.png' source='80,0,160,75'"  selectedimage="file='toolbar_hover.png' source='80,0,160,75'" group="main_toolbar" selected="false"/>
		</HorizontalLayout>
    <!---LOGO insert位置 相对位置  -->
    <!--bkcolor="0xFFFF0000" insert是内部内容的相对位置  LayOut布局默认是从右边开始-->
    <HorizontalLayout width="180" inset="10,15,1,0">
			<Button maxwidth="148" maxheight="62" bkimage="file='logo.png'"/>
		</HorizontalLayout>
	</HorizontalLayout>
    <!--窗体内容背景-->
	<HorizontalLayout bkimage="1.bmp" bordercolor="0xFFFF0000" >
    <!--垂直布局-->
		<VerticalLayout bordercolor="0xFFFF0000">
      <!---pos如果是float=true,那么是绝对布局。控件那么 指定位置和大小 如何不是那么 只指定大小-->
      <!---relativepos跟最大化后剧中显示有关 前两个是位移值  后两个是缩放一般不写 属性列表没有给出这个 前两个是大小改变的时候横纵向位移 默认单位50表示一个单位,  后两个是缩放 x,y,scaleX,scaleY-->
      <!---常用50,50,0,0 让绝对定位拥有相对定位布局的特性-->
			<Button pos="50, 50, 196, 168"  bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" normalimage="error.png" />
			<Text pos="250, 60, 350, 160" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" text="您的电脑已经" font="1" textcolor="#1C1C1C" />
			<Text pos="360, 55, 400, 160" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" text="24" font="2" textcolor="#0000FF" />
			<Text pos="400, 60, 500, 160" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" text="天没有体检,建议立即体检!" font="1" textcolor="#1C1C1C" />
			<Text pos="250, 100, 350, 150" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" text="系统可能已经存在大量风险,安全性和性能都在急速下降," font="0" textcolor="#555555" />
			<Text pos="250, 120, 350, 150" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" text="建议您每天坚持电脑体检,提高电脑的安全和性能" font="0" textcolor="#555555" />
			<Button pos="230, 250, 396, 316" name="check_normal" bordercolor="0xFFFF0000" relativepos="50,50,0,0"  float="true" normalimage="check_normal.png" hotimage="check_hover.png" pushedimage="check_pressed.png" />
		</VerticalLayout>
		<VerticalLayout width="250">
      <!---对于此处来说50代表水平位置自动适应 垂直位置不变-->
			<Button maxwidth="1"   bordercolor="0xFFFF0000"  minheight="1000" normalimage="fenge_line.png" />
			<Button pos="10, 15, 230, 126"    bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" minheight="1000" normalimage="register_bg.png" />
			<Button pos="0, 140, 245, 141"    bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" normalimage="rp_line.png" />
			<Button pos="80, 140, 81, 225"    bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" normalimage="fenge_line.png" />
			<Button pos="160, 140, 161, 225"  bordercolor="0xFFFF0000"  relativepos="50,0,0,0"  float="true" normalimage="fenge_line.png" />
			<Button pos="15, 150, 65, 195"    bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" normalimage="firewall_open_normal.png" hotimage="firewall_open_hover.png"/>
			<Text pos="10, 200, 65, 210"      bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" text="木马防火墙" font="0" textcolor="#1C1C1C" />
			<Button pos="95, 150, 145, 195"   bordercolor="0xFFFF0000"  relativepos="50,0,0,0"  float="true" normalimage="guard_open_normal.png" hotimage="guard_open_hover.png"/>
			<Text pos="95, 200, 105, 210"     bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" text="360保镖" font="0" textcolor="#1C1C1C" />
			<Button pos="175, 150, 225, 195"  bordercolor="0xFFFF0000"  relativepos="50,0,0,0"  float="true" normalimage="payinsure_close_normal.png" hotimage="payinsure_close_hover.png"/>
			<Text pos="175, 200, 185, 210"    bordercolor="0xFFFF0000" relativepos="50,0,0,0"  float="true" text="网购先赔" font="0" textcolor="#1C1C1C" />
			<Button pos="0, 225, 245, 226"    bordercolor="0xFFFF0000"  relativepos="50,0,0,0"  float="true" normalimage="rp_line.png" />
		</VerticalLayout>
	</HorizontalLayout>
  </VerticalLayout>
</Window>


代码加注释下载地址.......

http://download.csdn.net/detail/yue7603835/9185031









目录
相关文章
|
前端开发 JavaScript Java
【Layui】入门必看:登录注册界面搭建与功能开发解析
Layui是一款面向前端开发者的轻量级JavaScript库,旨在简化网页开发过程。它提供了丰富的基础UI组件和常用的工具函数,使开发者可以更快速地构建出优秀的网页界面。Layui具备以下特点和功能:简洁易用:Layui的设计理念是简洁而易用,它提供了直观的API和简单的配置,使开发者可以更快速地上手并快速构建界面。模块化加载:Layui采用模块化的加载方式,每个功能都是以独立的模块形式存在,开发者可以根据自己的需求按需加载所需的模块,避免了不必要的资源浪费。基础UI组件。
|
3月前
|
前端开发 开发者 容器
构建响应式Web界面:Flexbox与Grid布局的深度解析
【10月更文挑战第11天】本文深入解析了CSS3中的Flexbox和Grid布局,探讨了它们的特点、应用场景及使用方法。Flexbox适用于一维布局,如导航栏;Grid布局则适用于二维布局,如复杂网格。通过示例代码和核心属性介绍,帮助开发者灵活构建响应式Web界面。
64 5
|
3月前
|
存储 SQL 数据挖掘
TDengine 流计算与窗口机制的深度解析:揭示计数窗口的关键作用
在 TDengine 3.2.3.0 版本中,我们针对流式计算新增了计数窗口,进一步优化了流式数据处理的能力。本文将为大家解读流式计算与几大窗口的关系,并针对新增的计数窗口进行详细的介绍,帮助大家进一步了解 TDengine 流式计算,以便更好地进行应用。
70 1
|
5月前
|
开发者 编解码
界面适应奥秘:从自适应布局到图片管理,Xamarin响应式设计全解析
【8月更文挑战第31天】在 Xamarin 的世界里,构建灵活且适应性强的界面是每位开发者的必修课。本文将带您探索 Xamarin 的响应式设计技巧,包括自适应布局、设备服务协商和高效图片管理,帮助您的应用在各种设备上表现出色。通过 Grid 和 StackLayout 实现弹性空间分配,利用 Device 类检测设备类型以加载最优布局,以及使用 Image 控件自动选择合适图片资源,让您轻松应对不同屏幕尺寸的挑战。掌握这些技巧,让您的应用在多变的市场中持续领先。
49 0
|
7月前
|
数据采集 JavaScript 前端开发
HTML表单深度解析:构建互动的网页界面
HTML表单深度解析:构建互动的网页界面
|
7月前
|
缓存 前端开发 JavaScript
【前端性能优化】深入解析重绘和回流,构建高性能Web界面
【前端性能优化】深入解析重绘和回流,构建高性能Web界面
83 1
|
8月前
|
前端开发 开发者 UED
构建响应式Web界面:Flexbox与Grid布局的深度解析
【4月更文挑战第6天】 在现代前端开发中,为了适配不同设备并提供流畅的用户体验,理解并掌握响应式设计变得至关重要。本文将深入探讨两种主要的CSS布局模式——Flexbox和Grid。我们将剖析它们的核心概念、使用场景以及如何结合它们来创建复杂且灵活的响应式界面。通过实例演示和对比分析,帮助开发者提升界面布局技能,从而设计出能够适应多变设备的Web界面。
|
8月前
|
前端开发 开发者 UED
构建响应式Web界面:Flexbox与Grid布局的深度解析
【2月更文挑战第15天】在现代前端开发中,创建灵活且响应式的用户界面至关重要。本文深入探讨了两种强大的CSS布局模块——Flexbox和Grid,它们如何帮助我们实现复杂布局的挑战。通过比较这两种技术的特点、适用场景及其兼容性问题,我们将理解如何有效地将它们应用于日常开发中,以提升界面设计的灵活性和用户体验。
|
8月前
|
前端开发 开发者 UED
构建响应式Web界面:Flexbox与Grid布局的深度解析
【2月更文挑战第28天】 在现代前端开发中,打造灵活且适应不同屏幕尺寸的用户界面是至关重要的。随着移动设备的普及,响应式设计已经成为网页制作不可或缺的一部分。本文将深入探讨两种强大的CSS布局模块——Flexbox和Grid,它们如何简化布局创建过程,并赋予设计师更大的灵活性去构建动态和流畅的响应式界面。通过对这两种技术的比较、使用场景分析以及代码示例,读者将能够更好地理解何时以及如何使用这些工具来提升前端项目的质量和效率。
114 0
|
8月前
|
前端开发 开发者 UED
构建响应式Web界面:Flexbox与Grid布局的深度解析
【2月更文挑战第14天】 在现代前端开发中,为了适配不同设备并提供流畅的用户体验,理解并掌握响应式设计变得至关重要。本文将深入探讨两种主要的CSS布局模式——Flexbox和Grid。我们将剖析它们的核心概念、使用场景以及如何结合它们来创建复杂且灵活的响应式界面。通过实例演示和对比分析,帮助开发者提升界面布局技能,从而设计出能够适应多变设备的Web界面。

推荐镜像

更多
下一篇
开通oss服务