写在前面

最近的工作一直在弄一些h5的单页应用,然后嵌入到app的webview中。之前一直在用angularjs+html+ashx的一套东西。实在是玩腻了。然后就尝试通过asp.net mvc的方式构建单页应用。用到的技术angularjs+webapi+mvc。在网上找到了一些相关的文章,关于anguar的位置也没有一个比较好的一个标准。这里也是抛砖引玉,希望通过讨论,得到一个更好的结构。

项目结构

结构说明:

_Layout.cshtml:该文件作为模板文件,这里将应用的js,及css文件,都凡在该页。如图所示:

Controllers/api文件夹:存放webapi接口。

Controllers/*.cs:存放控制器。

将angularjs的mvc结构,都存放在了Scripts文件夹,这样做,也是为了操作方便(文件多的话,来回的切换窗口,确实很麻烦)。

Scripts/App:存放angularjs的文件,以及app.js

app.js的定义如下,包括module的注册,以及服务信息:

  1. var app = angular.module('app_store', ['ngRoute', 'StoreService', ]);
  2.  
  3. //服务
  4. var StoreService = angular.module('StoreService', []);
  5. //请求服务
  6. StoreService.factory('requestService', function ($http, $q) {
  7. var request = {
  8. method: 'POST',
  9. url: '',
  10. headers: { 'Content-Type': 'application/json' },
  11. data: {}
  12. };
  13. var postData = {
  14. lists: function (type) {
  15. request.method = "get";
  16. request.url = "../api/order/lists/" + type + "";
  17. return requestService($http, $q, request);
  18. },
  19. submit_product: function (data) {
  20. request.method = "post";
  21. request.url = "../api/order";
  22. request.data = data;
  23. return requestService($http, $q, request);
  24. }
  25. };
  26. return postData;
  27. });
  28. function requestService($http, $q, request) {
  29. var deferred = $q.defer(); // 声明延后执行,表示要去监控后面的执行
  30. $http(request).
  31. success(function (data, status, headers, config) {
  32. deferred.resolve(data); // 声明执行成功,即http请求数据成功,可以返回数据了
  33. }).
  34. error(function (data, status, headers, config) {
  35.  
  36. deferred.reject(data); // 声明执行失败,即服务器返回错误
  37. });
  38. return deferred.promise; // 返回承诺,这里并不是最终数据,而是访问最终数据的API
  39. };

将modlue的定义放在了该js文件中,其中也包括请求的服务,考虑到减少一次静态文件的请求,所以将服务也放在了该文件中。

Scripts/Controllers:这是angularjs的控制器。用来定义前端的controller。关于这个你可以根据用途,分成不同的控制器。也可以对应于web api的方式定义。我建议如果功能不是太多,还是放在一个里面,如果定义太多的js文件,一是静态文件的请求次数会很多,二是开发起来确实很头大,每次开发在vs打开n个tab页面,你会发现会让你非常的头大。

Scripts/Filter:存放angularjs自定义的过滤器,(如果过滤器不多,建议还是合并到app.js文件中。)

Scripts/Route:angularjs路由,如果路由不多,仍建议放在app.js中。

Scripts/Views:angularjs视图,存放视图模板。这个分法,不好说,可以参考asp.net mvc的分发,按控制器名称建文件夹。如果视图不多,我是一股脑的都塞到views文件夹了。

一个例子

列举一个根据关键字搜索商品的列表的例子。

Scripts/Controllers/StoreController.js

  1. app.controller('StoreController', function ($scope, $http, $location, $routeParams, requestService) {
  2. console.log('StoreController');
  3. if (!$scope.productKey) {
  4. $scope.productKey = "飞机";
  5. };
  6. requestService.lists($scope.productKey).then(function (data) {
  7. console.log(data);
  8. if (data._code === 200) {
  9. $scope.orders = data._data;
  10. };
  11. });
  12. });

Scripts/Route/app-route.js

  1. app.config(['$routeProvider', function ($routeProvider) {
  2. $routeProvider
  3. .when('/', { templateUrl: '../Scripts/Views/OrderList.html', controller: 'StoreController' })
  4. .when('/error', { templateUrl: '../Scripts/Views/Error.html', controller: 'ErrorController' })
  5. .otherwise({ redirectTo: '/error' });
  6. }]);

Scripts/Views/OrderList.html

  1. <div class="address_serace">
  2. <input class="form-control" ng-change="" ng-model="productKey" placeholder="搜索商品">
  3. </div>
  4. <div class="address_div">
  5. <dl class="address_dl" ng-repeat="item in orders">
  6. <dt class="address_checkbox">
  7. <img class="address_check" src="../Images/icon-xx01@2x.png" />
  8. </dt>
  9. <dt class="address_user"><img class="address_user" src="../Images/dingy.png" /></dt>
  10. <dd class="address_font">
  11. <p class="address_font_t">{{item.Name}}</p>
  12. <p>单价:{{item.Price}}</p>
  13. </dd>
  14. </dl>
  15.  
  16. </div>

asp.net mvc 控制器StoreController.cs 中Index的action添加视图,作为呈现的页面。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Web;
  5. using System.Web.Mvc;
  6.  
  7. namespace Wolfy.MvsSinglePage.Controllers
  8. {
  9. public class StoreController : Controller
  10. {
  11. // GET: Store
  12. public ActionResult Index()
  13. {
  14. return View();
  15. }
  16. }
  17. }

Index.cshtml,很简单,一个添加指定ng-view的div,用来呈现Views中的html模板的。

  1. @{
  2. ViewBag.Title = "Index";
  3. }
  4.  
  5. <div ng-view></div>

web api:OrderController

  1. using Newtonsoft.Json;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Net;
  6. using System.Net.Http;
  7. using System.Threading.Tasks;
  8. using System.Web.Http;
  9. using Wolfy.MvsSinglePage.Models;
  10.  
  11. namespace Wolfy.MvsSinglePage.Controllers.api
  12. {
  13. public class OrderController : ApiController
  14. {
  15. // GET: api/Order
  16. [HttpGet]
  17. [Route("api/order/lists/{key?}")]
  18. public async Task<HttpResponseMessage> Get(string key)
  19. {
  20. return await Task.Run(() =>
  21. {
  22. HttpResponseMessage response = new HttpResponseMessage(HttpStatusCode.Accepted);
  23. List<Order> lst = new List<Order>() {
  24. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机", Price=},
  25. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机2", Price=},
  26. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机3", Price=},
  27. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机4", Price=},
  28. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机5", Price=},
  29. new Order(){ Id=Guid.NewGuid(), Dt=DateTime.Now, Name="飞机6", Price=},
  30. };
  31. var results = string.IsNullOrEmpty(key) ? lst : lst.Where(x => x.Name.Contains(key));
  32. response = new HttpResponseMessage(HttpStatusCode.OK)
  33. {
  34. Content = new StringContent(JsonConvert.SerializeObject(new { _code = , _data = results }))
  35. };
  36. return response;
  37. });
  38. }
  39. }
  40. }

最后不要忘了,为_Layout.cshtml的html标签添加指令ng-app。

运行测试:

总结

这是在实际工作中,摸索出的一种分层的方式,如果你有更好的建议,可以分享一下,在网上也找了一些资料,并没有具体的分层方式。我这里抛砖引玉,希望有个更好的方案。

Demo Url: https://git.oschina.net/wolfy/Wolfy.Angularjs_Mvc_SPA

[Angularjs]asp.net mvc+angularjs+web api单页应用的更多相关文章

  1. ABP示例程序-使用AngularJs,ASP.NET MVC,Web API和EntityFramework创建N层的单页面Web应用

    本片文章翻译自ABP在CodeProject上的一个简单示例程序,网站上的程序是用ABP之前的版本创建的,模板创建界面及工程文档有所改变,本文基于最新的模板创建.通过这个简单的示例可以对ABP有个更深 ...

  2. vs 2013下自定义ASP.net MVC 5/Web API 2 模板(T4 视图模板/控制器模板)

    vs 2013下自定义ASP.net MVC 5/Web API 2  模板(T4 视图模板/控制器模板): Customizing ASP.NET MVC 5/Web API 2 Scaffoldi ...

  3. ASP.NET MVC+Knockout+Web API+SignalR

    架构设计(ASP.NET MVC+Knockout+Web API+SignalR) 架构设计(ASP.NET MVC+Knockout+Web API+SignalR) 2014-01-16 18: ...

  4. Visual Studio 2013 Preview - ASP.NET, MVC 5, Web API 2新功能搶先看

    Visual Studio 2013 Preview - ASP.NET, MVC 5, Web API 2新功能搶先看 來自TechEd North America 2013的第一手消息 以下資訊均 ...

  5. ASP.NET MVC和Web API中的Angular2 - 第2部分

    下载源码 内容 第1部分:Visual Studio 2017中的Angular2设置,基本CRUD应用程序,第三方模态弹出控件 第2部分:使用Angular2管道进行过滤/搜索,全局错误处理,调试客 ...

  6. [Asp.Net] MVC 和Web API Action 获取参数的区别

    Asp.net MVC 和web api 的action 在获取从前台传入的数据是有很大不同 前台使用ajax的方式向后台发起post的请求 Content-Type:application/json ...

  7. Asp.Net MVC及Web API框架配置会碰到的几个问题及解决方案(转)

      前言 刚开始创建MVC与Web API的混合项目时,碰到好多问题,今天拿出来跟大家一起分享下.有朋友私信我问项目的分层及文件夹结构在我的第一篇博客中没说清楚,那么接下来我就准备从这些文件怎么分文件 ...

  8. Asp.Net MVC及Web API框架配置会碰到的几个问题及解决方案

    前言 刚开始创建MVC与Web API的混合项目时,碰到好多问题,今天拿出来跟大家一起分享下.有朋友私信我问项目的分层及文件夹结构在我的第一篇博客中没说清楚,那么接下来我就准备从这些文件怎么分文件夹说 ...

  9. 转-Asp.Net MVC及Web API框架配置会碰到的几个问题及解决方案

    前言 刚开始创建MVC与Web API的混合项目时,碰到好多问题,今天拿出来跟大家一起分享下.有朋友私信我问项目的分层及文件夹结构在我的第一篇博客中没说清楚,那么接下来我就准备从这些文件怎么分文件夹说 ...

随机推荐

  1. livewriter写Blog 神秘失踪?

    现在习惯用livewriter来总结/记录一些知识并发布为Blog 与同行交流,但是今天发生了一个怪事,上午我整理了两篇文档当时就用livewriter发送到了Blog上,但是晚上来看的时候之前发送的 ...

  2. Object-Oriented CSS

    1.指导思想: http://oocss.org/ 2.reset.css http://meyerweb.com/eric/tools/css/reset/ 3.normalize.css http ...

  3. C#6.0特性(快来围观)

    说明一下,很多博友一进来就认为仅仅是语法糖,C#语法的更新,代表着它的进步,语法糖是为了让我们更好的实现语句和功能,增加了易读性和易用性.而且它的每次进步,也会给我们带来新的支持和改进.比如C#(4. ...

  4. Dropbox的可用Hosts文件

    108.160.167.203 www.dropbox.com 108.160.167.203 dropbox.com 108.160.165.211 dl-client677.dropbox.com ...

  5. 转摘http://blog.csdn.net/hulihui/article/details/3351922#s6

    译文:构建DataGridView的定制NumericUpDown单元格(Cell)和表格列(Column) 分类: DataGridView控件 2008-11-22 20:58 3555人阅读 评 ...

  6. 【BZOJ1008】【HNOI2008】越狱(数学排列组合题)

    1008: [HNOI2008]越狱 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3140  Solved: 1317[Submit][Status] ...

  7. javascript函数自调用

    1. 函数是由事件驱动的或者当它被调用时执行的可重复使用的代码块. 2.  将函数用 “()”括起来, 后面再加一个“()” 3.  javascript函数的内置对象arguments对象,  它包 ...

  8. Html设置图标icon

    html head添加: <link rel="icon" href="/favicon.ico" type="image/x-icon&quo ...

  9. Java基础-继承-子类与父类执行顺序

    代码 public class Test { public static void main(String[] args) { new Circle(); } } class Draw { publi ...

  10. 【CodeForces 602C】H - Approximating a Constant Range(dijk)

    Description through n) and m bidirectional railways. There is also an absurdly simple road network — ...