MVC视图展现模式之移动布局解析-续集
网站就必须用响应式布局吗?MVC视图展现模式之移动布局:http://www.cnblogs.com/dunitian/p/5213787.html
demo:http://pan.baidu.com/s/1bnTUaKJ
有人会疑问,为什么他能识别.mobile的后缀却不能识别例如:.mac .dnt 等等后缀呢?这些又是放在哪里的呢?
mobile 这个后缀其实是存放在:DisplayModeProvider.Instance.Modes 里面的,我们监视一下,发现里面就一个mobile,还有一个是默认的
可以猜想,运行的时候是从上往下匹配的,“”的是通用匹配,那么我们加入一个自定义的后缀看看==>(可以思考一下,为什么用 insert 不用 add)
//添加一个自定义后缀
DisplayModeProvider.Instance.Modes.Insert(0, new DefaultDisplayMode("dnt")
{
ContextCondition = (Context) => Context.Request.UserAgent.Contains("dnt")
});
可能你不是很理解 DefaultDisplayMode,看看反编译吧----构造函数为suffix赋值(后缀)
添加一个自定义的后缀视图
打开谷歌浏览器,设置一下User-Agent Switcher的浏览模式
附录:
DisplayModeProvider:
// Generated by .NET Reflector from F:\Work\Net\Mobile\packages\Microsoft.AspNet.WebPages.3.2.3\lib\net45\System.Web.WebPages.dll
namespace System.Web.WebPages
{
using System;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using System.Runtime.CompilerServices;
using System.Web; public sealed class DisplayModeProvider
{
private static readonly object _displayModeKey = new object();
private readonly List<IDisplayMode> _displayModes;
private static readonly DisplayModeProvider _instance = new DisplayModeProvider();
public static readonly string DefaultDisplayModeId = string.Empty;
public static readonly string MobileDisplayModeId = "Mobile"; internal DisplayModeProvider()
{
List<IDisplayMode> list = new List<IDisplayMode>();
DefaultDisplayMode item = new DefaultDisplayMode(MobileDisplayModeId) {
ContextCondition = context => context.GetOverriddenBrowser().IsMobileDevice
};
list.Add(item);
list.Add(new DefaultDisplayMode());
this._displayModes = list;
} private int FindFirstAvailableDisplayMode(IDisplayMode currentDisplayMode, bool requireConsistentDisplayMode)
{
if (!requireConsistentDisplayMode || (currentDisplayMode == null))
{
return ;
}
int index = this._displayModes.IndexOf(currentDisplayMode);
if (index < )
{
return this._displayModes.Count;
}
return index;
} public IEnumerable<IDisplayMode> GetAvailableDisplayModesForContext(HttpContextBase httpContext, IDisplayMode currentDisplayMode)
{
return this.GetAvailableDisplayModesForContext(httpContext, currentDisplayMode, this.RequireConsistentDisplayMode);
} internal IEnumerable<IDisplayMode> GetAvailableDisplayModesForContext(HttpContextBase httpContext, IDisplayMode currentDisplayMode, bool requireConsistentDisplayMode)
{
int iteratorVariable0 = this.FindFirstAvailableDisplayMode(currentDisplayMode, requireConsistentDisplayMode);
for (int i = iteratorVariable0; i < this._displayModes.Count; i++)
{
IDisplayMode iteratorVariable2 = this._displayModes[i];
if (iteratorVariable2.CanHandleContext(httpContext))
{
yield return iteratorVariable2;
}
}
} public DisplayInfo GetDisplayInfoForVirtualPath(string virtualPath, HttpContextBase httpContext, Func<string, bool> virtualPathExists, IDisplayMode currentDisplayMode)
{
return this.GetDisplayInfoForVirtualPath(virtualPath, httpContext, virtualPathExists, currentDisplayMode, this.RequireConsistentDisplayMode);
} internal DisplayInfo GetDisplayInfoForVirtualPath(string virtualPath, HttpContextBase httpContext, Func<string, bool> virtualPathExists, IDisplayMode currentDisplayMode, bool requireConsistentDisplayMode)
{
for (int i = this.FindFirstAvailableDisplayMode(currentDisplayMode, requireConsistentDisplayMode); i < this._displayModes.Count; i++)
{
IDisplayMode mode = this._displayModes[i];
if (mode.CanHandleContext(httpContext))
{
DisplayInfo info = mode.GetDisplayInfo(httpContext, virtualPath, virtualPathExists);
if (info != null)
{
return info;
}
}
}
return null;
} internal static IDisplayMode GetDisplayMode(HttpContextBase context)
{
if (context == null)
{
return null;
}
return (context.Items[_displayModeKey] as IDisplayMode);
} internal static void SetDisplayMode(HttpContextBase context, IDisplayMode displayMode)
{
if (context != null)
{
context.Items[_displayModeKey] = displayMode;
}
} public static DisplayModeProvider Instance
{
get
{
return _instance;
}
} public IList<IDisplayMode> Modes
{
get
{
return this._displayModes;
}
} public bool RequireConsistentDisplayMode { get; set; } [CompilerGenerated]
private sealed class <GetAvailableDisplayModesForContext>d__4 : IEnumerable<IDisplayMode>, IEnumerable, IEnumerator<IDisplayMode>, IEnumerator, IDisposable
{
private int <>1__state;
private IDisplayMode <>2__current;
public IDisplayMode <>3__currentDisplayMode;
public HttpContextBase <>3__httpContext;
public bool <>3__requireConsistentDisplayMode;
public DisplayModeProvider <>4__this;
private int <>l__initialThreadId;
public int <first>5__5;
public int <i>5__6;
public IDisplayMode <mode>5__7;
public IDisplayMode currentDisplayMode;
public HttpContextBase httpContext;
public bool requireConsistentDisplayMode; [DebuggerHidden]
public <GetAvailableDisplayModesForContext>d__4(int <>1__state)
{
this.<>1__state = <>1__state;
this.<>l__initialThreadId = Environment.CurrentManagedThreadId;
} private bool MoveNext()
{
switch (this.<>1__state)
{
case :
this.<>1__state = -;
this.<first>5__5 = this.<>4__this.FindFirstAvailableDisplayMode(this.currentDisplayMode, this.requireConsistentDisplayMode);
this.<i>5__6 = this.<first>5__5;
goto Label_00A5; case :
this.<>1__state = -;
break; default:
goto Label_00BD;
}
Label_0097:
this.<i>5__6++;
Label_00A5:
if (this.<i>5__6 < this.<>4__this._displayModes.Count)
{
this.<mode>5__7 = this.<>4__this._displayModes[this.<i>5__6];
if (this.<mode>5__7.CanHandleContext(this.httpContext))
{
this.<>2__current = this.<mode>5__7;
this.<>1__state = ;
return true;
}
goto Label_0097;
}
Label_00BD:
return false;
} [DebuggerHidden]
IEnumerator<IDisplayMode> IEnumerable<IDisplayMode>.GetEnumerator()
{
DisplayModeProvider.<GetAvailableDisplayModesForContext>d__4 d__;
if ((Environment.CurrentManagedThreadId == this.<>l__initialThreadId) && (this.<>1__state == -))
{
this.<>1__state = ;
d__ = this;
}
else
{
d__ = new DisplayModeProvider.<GetAvailableDisplayModesForContext>d__4() {
<>4__this = this.<>4__this
};
}
d__.httpContext = this.<>3__httpContext;
d__.currentDisplayMode = this.<>3__currentDisplayMode;
d__.requireConsistentDisplayMode = this.<>3__requireConsistentDisplayMode;
return d__;
} [DebuggerHidden]
IEnumerator IEnumerable.GetEnumerator()
{
return this.System.Collections.Generic.IEnumerable<System.Web.WebPages.IDisplayMode>.GetEnumerator();
} [DebuggerHidden]
void IEnumerator.Reset()
{
throw new NotSupportedException();
} void IDisposable.Dispose()
{
} IDisplayMode IEnumerator<IDisplayMode>.Current
{
[DebuggerHidden]
get
{
return this.<>2__current;
}
} object IEnumerator.Current
{
[DebuggerHidden]
get
{
return this.<>2__current;
}
}
}
}
}
DefaultDisplayMode:
// Generated by .NET Reflector from F:\Work\Net\Mobile\packages\Microsoft.AspNet.WebPages.3.2.3\lib\net45\System.Web.WebPages.dll
namespace System.Web.WebPages
{
using System;
using System.IO;
using System.Runtime.CompilerServices;
using System.Web; public class DefaultDisplayMode : IDisplayMode
{
private readonly string _suffix; public DefaultDisplayMode() : this(DisplayModeProvider.DefaultDisplayModeId)
{
} public DefaultDisplayMode(string suffix)
{
this._suffix = suffix ?? string.Empty;
} public bool CanHandleContext(HttpContextBase httpContext)
{
if (this.ContextCondition != null)
{
return this.ContextCondition(httpContext);
}
return true;
} public virtual DisplayInfo GetDisplayInfo(HttpContextBase httpContext, string virtualPath, Func<string, bool> virtualPathExists)
{
string arg = this.TransformPath(virtualPath, this._suffix);
if ((arg != null) && virtualPathExists(arg))
{
return new DisplayInfo(arg, this);
}
return null;
} protected virtual string TransformPath(string virtualPath, string suffix)
{
if (string.IsNullOrEmpty(suffix))
{
return virtualPath;
}
string extension = Path.GetExtension(virtualPath);
return Path.ChangeExtension(virtualPath, suffix + extension);
} public Func<HttpContextBase, bool> ContextCondition { get; set; } public virtual string DisplayModeId
{
get
{
return this._suffix;
}
}
}
}
参考:https://msdn.microsoft.com/en-us/magazine/hh975347.aspx
http://www.asp.net/mvc/overview/older-versions/aspnet-mvc-4-mobile-features
http://stackoverflow.com/questions/9354188/asp-net-mvc-4-mobile-display-modes-stop-working
MVC视图展现模式之移动布局解析-续集的更多相关文章
- 网站就必须用响应式布局吗?MVC视图展现模式之移动布局
本文先引入给读者一个自己研究的机会,下次深入说明一下: 废话不多说,直接上图 新建一个mvc的项目 在视图里面添加一个移动端视图 正常访问一下 Bootstrap自带的响应式的方式(页面代码并没有改变 ...
- MVC视图展现模式之移动布局
参考:http://www.cnblogs.com/dunitian/p/5218140.html 简单点,直接上用法 新建MVC项目,在golbal.asax中添加如下代码 //添加一个自定义后缀 ...
- Spring MVC视图解析器
Spring MVC提供的视图解析器使用ViewResolver进行视图解析,实现浏览器中渲染模型.ViewResolver能够解析JSP.Velocity模板.FreeMarker模板和XSLT等多 ...
- ASP.NET Core 入门教程 6、ASP.NET Core MVC 视图布局入门
一.前言 1.本教程主要内容 ASP.NET Core MVC (Razor)视图母版页教程 ASP.NET Core MVC (Razor)带有Section的视图母版页教程 ASP.NET Cor ...
- 模型-视图-控制器模式(MVC模式,10种常见体系架构模式之一)
一.简介: 架构模式是一个通用的.可重用的解决方案,用于在给定上下文中的软件体系结构中经常出现的问题.架构模式与软件设计模式类似,但具有更广泛的范围. 模型-视图-控制器模式,也称为MVC模式.是软件 ...
- ASP.NET Core 入门笔记7,ASP.NET Core MVC 视图布局入门
一.前言 1.本教程主要内容 ASP.NET Core MVC (Razor)视图母版页教程 ASP.NET Core MVC (Razor)带有Section的视图母版页教程 ASP.NET Cor ...
- 二、ASP.NET MVC Controller 控制器(一:深入解析控制器运行原理)
阅读目录: 1.开篇介绍 2.ASP.NETMVC Controller 控制器的入口(Controller的执行流程) 3.ASP.NETMVC Controller 控制器的入口(Controll ...
- NET/ASP.NET MVC Controller 控制器(一:深入解析控制器运行原理)
阅读目录: 1.开篇介绍 2.ASP.NETMVC Controller 控制器的入口(Controller的执行流程) 3.ASP.NETMVC Controller 控制器的入口(Controll ...
- Asp.net MVC 视图之公用代码
一.公共模板 转自:http://www.cnblogs.com/kissdodog/archive/2013/01/07/2848881.html 1.@RenderBody() 在网站公用部分通过 ...
随机推荐
- .NET Core的文件系统[5]:扩展文件系统构建一个简易版“云盘”
FileProvider构建了一个抽象文件系统,作为它的两个具体实现,PhysicalFileProvider和EmbeddedFileProvider则分别为我们构建了一个物理文件系统和程序集内嵌文 ...
- 有趣的 CSS 像素艺术
原文地址:https://css-tricks.com/fun-times-css-pixel-art/#article-header-id-4 译者:nzbin 友情提示:由于国内网络的原因,Cod ...
- Android公共title的应用
我们在开发Android应用中,写每一个页面的时候都会建一个title,不是写一个LinearLayout就是写一个RelativeLayout,久而久之就会觉得这样繁琐,尤其几个页面是只是标题不一样 ...
- 【算法】C语言实现数组的动态分配
C语言实现数组的动态分配 作者:白宁超 2016年10月27日20:13:13 摘要:数据结构和算法对于编程的意义不言而喻,具有指导意义的.无论从事算法优化方向研究,还是大数据处理,亦或者网站开发AP ...
- Javascript 代理模式模拟一个文件同步功能
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- C++随笔:.NET CoreCLR之GC探索(4)
今天继续来 带大家讲解CoreCLR之GC,首先我们继续看这个GCSample,这篇文章是上一篇文章的继续,如果有不清楚的,还请翻到我写的上一篇随笔.下面我们继续: // Initialize fre ...
- notepad++设置默认打开txt文件失效的解决方法
1.系统环境 win10企业版,64位系统 2.初步设置 设置txt默认为notepad++打开,菜单:设置->首选项->文件关联 选择对应的文件扩展,点击"关闭"按钮 ...
- Git(1)
安装Git 完毕 (在开始菜单打开的话,打开的不是你想要的路径,切换路径很麻烦) 1.D盘新建 GitTest 文件夹 2.打开GitTest , 在空白的地方右键, 3.单击 Git Bash He ...
- Oracle 10g安装教程
首先下载安装文件,打开后文件结构如图所示: 安装之前请关闭Windows防火墙并断开网络. xp系统下直接双击运行(本经验以XP系统安装为例进行讲述). 如果是在win7上安装,如图:在setup文件 ...
- 使用nginx反向代理,一个80端口下,配置多个微信项目
我们要接入微信公众号平台开发,需要填写服务器配置,然后依据接口文档才能实现业务逻辑.但是微信公众号接口只支持80接口(80端口).我们因业务需求需要在一个公众号域名下面,发布两个需要微信授权的项目,怎 ...