所谓的选择联动,就是指,当我DateTimePicker1选择2月4号的时候,我DateTimePicker2只能选择2月4号和2月5号两天,当然你可以自行规定要选择的日期。这在一些图表查询条件里面是很常用的一个功能。下面我们就来看看如何设计。

DateTimePicker的选取与使用

在这里,我们使用的DateTimePicker是一个开源的组件,他的model名称为:ui.bootstrap.datetimepicker,我们可以去这个网址找到其相关的内容:http://dalelotts.github.io/angular-bootstrap-datetimepicker/,然后下载其相应的包,最后放到项目中,并进行引用即可(注意不能少了moment.js,它是构建与这个组件的基础上):

    <!--DateTimePicker Part-->
<link href="~/Content/front/angular-datetimepicker/css/datetimepicker.css" rel="stylesheet" />
<script src="~/Content/front/angular-datetimepicker/js/moment.js"></script>
<script src="~/Content/front/angular-datetimepicker/js/zh-cn.js"></script>
<script src="~/Content/front/angular-datetimepicker/js/datetimepicker.js"></script>

然后在module中进行注册:

var app = angular.module('dsBootstrap', [
'ui.grid',
'ui.grid.selection',
'ui.grid.pagination',
'ngCookies',
'ui.bootstrap.datetimepicker'
]);

最后在HTML页面进行排版布局使用即可:

  <div class="row" ng-show="visible">
<div class="col-md-4">
<div class="input-group">
<span class="input-group-addon" id="basic-addon1">开始时间:</span>
<div class="dropdown">
<a class="dropdown-toggle" id="starttime" role="button" data-toggle="dropdown" data-target="#" href="#">
<div class="input-group">
<input type="text" class="form-control" data-ng-model="starttime|date:'yyyy-MM-dd' "><span class="input-group-addon"><i class="glyphicon glyphicon-calendar"></i></span>
</div>
</a>
<ul class="dropdown-menu" role="menu" aria-labelledby="dLabel">
<datetimepicker data-ng-model="starttime" data-on-set-time="onTimeSet(newDate, oldDate)" data-datetimepicker-config="{ dropdownSelector: '#starttime',startView:'day', minView:'day' }" />
</ul>
</div>
</div>
</div>
<div class="col-md-4">
<div class="input-group">
<span class="input-group-addon" id="basic-addon1">结束时间:</span>
<div class="dropdown">
<a class="dropdown-toggle" id="endtime" role="button" data-toggle="dropdown" data-target="#" href="#">
<div class="input-group">
<input type="text" class="form-control" data-ng-model="endtime|date:'yyyy-MM-dd' "><span class="input-group-addon"><i class="glyphicon glyphicon-calendar"></i></span>
</div>
</a>
<ul class="dropdown-menu" id="endContainer" role="menu" aria-labelledby="dLabel"> </ul>
</div>
</div>
</div> </div>

这样我们就完成了第一步的工作了。从上面的Html代码我们可以看出,starttime和endtime是传递到controller中的所选择的日期值。而且我们的endtime DateTimePicker我们并没有放到前台,我们需要在后台动态生成。为什么呢?因为我们需要根据StartTime的选取值,来计算出endTime的选取范围,所以这里需要动态生成绑定。

DateTimePicker的联动

下面我们开始设计其联动工作。

首先,当选择开始时间后,会进入onTimeSet事件,在这个事件中,我们先动态生成结束时间选择器:

    $scope.onTimeSet = function (newDate, oldDate) {
var startTimeFmt = moment($scope.starttime).format("YYYY-MM-DD");
var endTimeFmt = moment(startTimeFmt).add(1, 'day').format("YYYY-MM-DD"); //时间框置空
$("#endtime input").val("");
//移除原有的datetimepicker对象
$("#endContainer").children().remove();
//设置config
$scope.config = { dropdownSelector: "#endtime", startView: "day", minView: "day" };
//动态编译datetimepicker directive
var compiledeHTML = $compile('<datetimepicker data-ng-model="endtime" data-before-render="beforeRender($view, $dates, $leftDate, $upDate, $rightDate)" data-datetimepicker-config="{{config}}" />')($scope);
//放入html容器
$("#endContainer").append(compiledeHTML);
}

然后,当上面的方法执行到$compile的时候,就会触发其beforeRender事件,此事件允许在控件加载前,进行一些相关操作。我们利用这个事件,抛出一个days-check事件,以便动态加载时间范围:

    //当选择开始时间,会进入onTimeSet事件,执行到$compile的时候,就会触发下面的beforeRender事件
//触发beforeReder事件后,会抛出一个days-check事件出去,并附带所有的当页时间对象。
$scope.beforeRender = function ($view, $dates, $leftDate, $upDate, $rightDate) {
$timeout(function () {
$scope.$broadcast('days-check', $dates);
});
}

最后,我们接收这个事件:

    //接收事件,并重置页面
$scope.$on('days-check', function (e, d) {
for (var i = 0; i < d.length; i++) {
//初始设置为不可选状态,不选中状态
d[i].active = false;
d[i].selectable = false;
//当前loop的值
var currentDate = moment(d[i].utcDateValue).format("YYYY-MM-DD");
//当前选定的开始时间
var startTimeFmt = moment($scope.starttime).format("YYYY-MM-DD");
//允许选定的最大的结束时间
var endTimeFmt = moment(startTimeFmt).add(1, 'day').format("YYYY-MM-DD");
//比较并设置可选日期
if (currentDate >= startTimeFmt && currentDate <= endTimeFmt) {
d[i].selectable = true;
}
}
});

当我们收到这个days-check事件的时候,我们会首先重置所有日期为不可选状态,然后根据预先设定的值,来确定选取范围。 这样,通过上面的设置,我们的结束时间就能根据开始时间的选择而进行联动了。是不是很方便呢?最后附上controller总体代码:

app.controller('collectorController', ['$scope', '$cookies', '$timeout','$compile', 'baseService', 'collectorService', 'uiGridConstants', function ($scope, $cookies,$timeout,$compile, baseService, collectorService, uiGridConstants) {
var self = this; $scope.ProvinceData = null;
$scope.CityData = null;
$scope.DistrictData = null;
$scope.CompanyData = null;
$scope.MachineData = null;
$scope.RealTimeData = null;
$scope.HistoryData = null; $scope.selectedProvince = null;
$scope.selectedCity = null;
$scope.selectedDistrict = null;
$scope.selectedCompany = null;
$scope.selectedMachine = null; $scope.starttime = null;
$scope.endtime = null; $scope.visible = false; //datetimepicker是否显示 //art.dialog({ title: '加载提示', icon: 'face-smile', fixed: true,left:'50%',top:0, time:3, content: "日期选择跨度不要过大,否则会因为数据量过大而无法加载图表和列表!", padding: 0 }); //监测省份的变化,如果发生了变化,则加载城市列表
$scope.$watchCollection('selectedProvince', function (oldval, newval) {
$scope.GetCityList();
}); //监测城市变化,如果发生了变化,则加载地区列表
$scope.$watchCollection('selectedCity', function (oldval, newval) {
$scope.GetDistrictList();
});
//监测地区变化,如果发生了变化,则加载公司列表
$scope.$watchCollection('selectedDistrict', function (oldval, newval) {
$scope.GetCompanyList();
});
//监测公司变化,如果发生了变化,则加载机器列表
$scope.$watchCollection('selectedCompany', function (oldval, newval) {
$scope.GetMachineList();
}); //绑定列表
$scope.gridOptions = {
enableRowSelection: true,
enableSelectAll: false,
selectionRowHeaderWidth: 35,
rowHeight: 35,
showGridFooter: false,
multiSelect: true,
enablePaginationControls: true,
paginationPageSizes: [9, 15, 20],
paginationPageSize: 9
}; $scope.gridOptions.columnDefs = [
{ name: 'Param_name', displayName: '参数名称' },
{ name: 'Param_unit', displayName: '参数单位' },
{ name: 'Param_data', displayName: '参数值' },
{ name: 'Param_time', displayName: '采集时间' }
]; $scope.gridOptions.onRegisterApi = function (gridApi) {
$scope.gridApi = gridApi;
}; //省份绑定
collectorService.GetProvinceData().then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.ProvinceData = data.data.data;
$scope.selectedProvince = baseService.getSelectedDataMapper($scope.ProvinceData, 'province');
}
}, null); $scope.GetCityList = function () {
var selectedProvinceId;
if ($scope.selectedProvince != undefined)
selectedProvinceId = $scope.selectedProvince.id;
else
return;
//市区绑定
collectorService.GetCityData(selectedProvinceId).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.CityData = data.data.data;
$scope.selectedCity = baseService.getSelectedDataMapper($scope.CityData, 'city');
}
}, null);
} $scope.GetDistrictList = function () {
var selectedCityId;
if ($scope.selectedCity != undefined)
selectedCityId = $scope.selectedCity.id;
else
return;
//区县绑定
collectorService.GetDistrictData(selectedCityId).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.DistrictData = data.data.data;
$scope.selectedDistrict = baseService.getSelectedDataMapper($scope.DistrictData, 'district');
}
}, null);
} $scope.GetCompanyList = function () {
var selectedDistrictId;
if ($scope.selectedDistrict != undefined)
selectedDistrictId = $scope.selectedDistrict.id;
else
return;
//公司绑定
collectorService.GetCompanyData(selectedDistrictId).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.CompanyData = data.data.data;
$scope.selectedCompany = baseService.getSelectedDataMapper($scope.CompanyData, 'company');
}
}, null);
} $scope.GetMachineList = function () {
var selectedCompanyId;
if ($scope.selectedCompany != undefined)
selectedCompanyId = $scope.selectedCompany.id;
else
return;
//设备绑定
collectorService.GetMachineList(selectedCompanyId).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.MachineData = data.data.data;
$scope.selectedMachine = baseService.getSelectedDataMapper($scope.MachineData, 'machine');
}
}, null);
} //获取实时数据
$scope.GetRealTimeDataByMachine = function () {
timeCheck($scope.starttime, $scope.endtime);
var starttimeFmt = timeFmt($scope.starttime);
var endtimeFmt = timeFmt($scope.endtime);
//获取实时数据
var machineId = $scope.selectedMachine.machine_id;
collectorService.GetRealDataList(machineId).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.RealTimeData = data.data.data;
}
}, null);
//获取列表数据
collectorService.GetHistoryDataList(machineId,starttimeFmt,endtimeFmt).then(function (data) {
var flag = data.data.success;
if(flag)
{
$scope.HistoryData = data.data.data;
$scope.gridOptions.data = data.data.list;
}
}, null);
//将级联列表项放到cookie中,以便于之后的操作简易化
var expireDate = new Date();
expireDate.setDate(expireDate.getDate() + 7);
delete $cookies['frontselection']; var cookieData = JSON.stringify({
province: $scope.selectedProvince,
city: $scope.selectedCity,
district: $scope.selectedDistrict,
company: $scope.selectedCompany,
machine: $scope.selectedMachine
});
$cookies.put('frontselection', cookieData, { 'expires': expireDate });
} $scope.ClickToGetParamDataList = function (paramId) { timeCheck($scope.starttime,$scope.endtime); var starttimeFmt = timeFmt($scope.starttime);
var endtimeFmt = timeFmt($scope.endtime); var machineId = $scope.selectedMachine.machine_id;
collectorService.GetHistoryDataListByParamId(machineId, paramId, starttimeFmt, endtimeFmt).then(function (data) {
var flag = data.data.success;
if (flag) {
$scope.gridOptions.data = data.data.list;
}
});
} //显示隐藏时间段选择
$scope.ClickToShowTimePicker = function () {
$scope.visible = !$scope.visible;
} $scope.onTimeSet = function (newDate, oldDate) {
var startTimeFmt = moment($scope.starttime).format("YYYY-MM-DD");
var endTimeFmt = moment(startTimeFmt).add(1, 'day').format("YYYY-MM-DD"); //时间框置空
$("#endtime input").val("");
//移除原有的datetimepicker对象
$("#endContainer").children().remove();
//设置config
$scope.config = { dropdownSelector: "#endtime", startView: "day", minView: "day" };
//动态编译datetimepicker directive
var compiledeHTML = $compile('<datetimepicker data-ng-model="endtime" data-before-render="beforeRender($view, $dates, $leftDate, $upDate, $rightDate)" data-datetimepicker-config="{{config}}" />')($scope);
//放入html容器
$("#endContainer").append(compiledeHTML);
} //接收事件,并重置页面
$scope.$on('days-check', function (e, d) {
for (var i = 0; i < d.length; i++) {
//初始设置为不可选状态,不选中状态
d[i].active = false;
d[i].selectable = false;
//当前loop的值
var currentDate = moment(d[i].utcDateValue).format("YYYY-MM-DD");
//当前选定的开始时间
var startTimeFmt = moment($scope.starttime).format("YYYY-MM-DD");
//允许选定的最大的结束时间
var endTimeFmt = moment(startTimeFmt).add(1, 'day').format("YYYY-MM-DD");
//比较并设置可选日期
if (currentDate >= startTimeFmt && currentDate <= endTimeFmt) {
d[i].selectable = true;
}
}
}); //当选择开始时间,会进入onTimeSet事件,执行到$compile的时候,就会触发下面的beforeRender事件
//触发beforeReder事件后,会抛出一个days-check事件出去,并附带所有的当页时间对象。
$scope.beforeRender = function ($view, $dates, $leftDate, $upDate, $rightDate) {
$timeout(function () {
$scope.$broadcast('days-check', $dates);
});
} var timeCheck = function(start,end)
{
if (start == null && end!=null)
{
art.dialog({ title: '提示', icon: 'error', time: 6, content: "必须选择开始日期,请重试!", padding: 0 });
return;
}
if (start!=null && end == null)
{
art.dialog({ title: '提示', icon: 'error', time: 6, content: "必须选择结束日期,请重试!", padding: 0 });
return;
}
if (start != null && end != null && start > end) {
art.dialog({ title: '提示', icon: 'error', time: 6, content: "开始时间不能大于结束时间,请重试!", padding: 0 });
return;
}
} var timeFmt = function(time)
{
if (time == null)
time = "";
else
time = time.toLocaleDateString();
return time;
}
}]);

物联网平台设计心得:DateTimePicker实现选择联动的更多相关文章

  1. 物联网平台设计心得:你所不知道的CRC校验

    在物联网平台设计过程中,我的中间件一方面需要处理来自于硬件端的包,另一方面需要处理来自于用户端的包,用户端包括web端和手机端等等.所以编写一个统一的CRC认证是非常必须要. 那么,在设计开始,CRC ...

  2. 物联网平台设计心得:管中窥豹之HeartBeat

    前言 在写这篇文章之前,我曾对HeartBeat做过诸多的研究,也做过诸多的项目,在这些项目中,有客户端为了维持状态,而定时向服务端发送的HeartBeat:有服务端为了维持客户端连接状态而处理Hea ...

  3. 国内物联网平台(3):QQ物联智能硬件开放平台

    国内物联网平台(3)——QQ物联·智能硬件开放平台 马智 平台定位 将QQ帐号体系.好友关系链.QQ消息通道及音视频服务等核心能力提供给可穿戴设备.智能家居.智能车载.传统硬件等领域的合作伙伴,实现用 ...

  4. 国内物联网平台(5):机智云IoT物联网云服务平台及智能硬件自助开发平台

    国内物联网平台(5)——机智云IoT物联网云服务平台及智能硬件自助开发平台 马智 平台定位 机智云平台是致力于物联网.智能硬件云服务的开放平台.平台提供了从定义产品.设备端开发调试.应用开发.产测.运 ...

  5. 阿里云物联网平台体验(树莓派+Python篇)

    阿里云物联网平台体验(树莓派+Python篇) 虽然对阿里云物联网平台比较熟悉了,从一开始就有幸参与了飞凤平台(Link Develop 一站式开发平台的前身)的一些偏硬件接入的工作.但是同时也见证了 ...

  6. 读懂SAP Leonardo物联网平台

    读懂SAP Leonardo物联网平台 https://blog.csdn.net/weixin_42137700/article/details/81903290 本文比较系统.全面地介绍了SAP ...

  7. 国外物联网平台(8):Telit

    国外物联网平台(8) ——Telit 马智 定位 We Bring IoT to Life Telit提供世界上最全面的高性能物联网模块.连接服务和软件. 产品体系 模块 Telit提供丰富专业的物联 ...

  8. 国外物联网平台(7):FogHorn

    国外物联网平台(7) ——FogHorn 马智 引言: 据外媒在本月20日报道,硅谷初创公司FogHorn正在与谷歌合作以简化工业物联网应用的部署.本文对FogHorn的技术.产品.应用和生态进行了分 ...

  9. 国外物联网平台(6):Electric Imp

    国外物联网平台(6)——Electric Imp 马智 公司背景 Electric Imp成立于2011年,公司设立在美国加利福尼亚州洛斯阿尔托斯和英国剑桥 公司投资者包括:富士康技术集团.PTI创投 ...

随机推荐

  1. RequireJS入门之一——实现第一个例子

    为什么学习RequireJS? 像我这种菜鸟,会提到海量文章里提到的AMD.JS模块化编程.异步... ... 等等 RequireJS是一个Javascript 文件和模块框架,它可以帮我们去管理j ...

  2. RequireJS 模块的定义与加载

    模块不同于传统的脚本文件,它良好地定义了一个作用域来避免全局名称空间污染.它可以显式地列出其依赖关系,并以函数(定义此模块的那个函数)参数的形式将这些依赖进行注入,而无需引用全局变量.RequireJ ...

  3. tomcat远程调试javaweb

    当把一个本地项目部署到远程服务器后有可能出现意想不到错误,这个时候通过远程调试能够更清楚的找到bug所在位置. 目前百度tomcat绝大多数方法都是在startup.sh或者catalina.sh增加 ...

  4. C语言中把数字转换为字符串 【转】

    在将各种类型的数据构造成字符串时,sprintf 的强大功能很少会让你失望.由于sprintf 跟printf 在用法上几乎一样,只是打印的目的地不同而已,前者打印到字符串中,后者则直接在命令行上输出 ...

  5. SVG简介

    最近遇到SVG这个名词,于是查阅资料,做个笔记. 前言 图片的数字化.将图片存储为数据有两种方案. 位图.也被称为光栅图.即是以自然的光学的眼光将图片看成在平面上密集排布的点的集合.每个点发出的光有独 ...

  6. iOS中,在类的源文件(.m)中,@interface部分的作用?

      此@interface部分为类扩展(extension). 其被设计出来就是为了解决两个问题的 其一,定义类私有方法的地方. 其二,实现public readonly,private readwr ...

  7. AES 加密工具类

    /** * AES 是一种可逆加密算法,对用户的敏感信息加密处理 对原始数据进行AES加密后,在进行Base64编码转化: */public class AESOperator { /* * 加密用的 ...

  8. 会话技术( Cookie ,Session)

    会话技术:    会话:浏览器访问服务器端,发送多次请求,接受多次响应.直到有一方断开连接.会话结束.        解决问题:可以使用会话技术,在一次会话的多次请求之间共享数据.           ...

  9. iOS -[PFPASIDataCompressor compressBytes:length:error:shouldFinish:] in PFPGZIPInvocationCompressor.o

    添加动态库报错 "_deflate", referenced from: -[PFPASIDataCompressor compressBytes:length:error:sho ...

  10. Web客户端数据存储学习笔记——Cookie

    今天对登录访问的安全以及web客户端存储做了一些大致的学习,决定在这方面加深理解,记录在博客里.第一个接触到的是Cookie... WHAT? WHY? HOW? 在学习cookie的使用时发现其名称 ...