亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

仿Angular Bootstrap TimePicker創建分鐘數-秒數的輸入控件

2019-11-20 09:34:25
字體:
來源:轉載
供稿:網友

在一個項目中需要一個用來輸入分鐘數和秒數的控件,然而調查了一些開源項目后并未發現合適的控件。在Angular Bootstrap UI中有一個類似的控件TimePicker,但是它并沒有深入到分鐘和秒的精度。
因此,決定參考它的源碼然后自己進行實現。 
最終的效果如下:

 

首先是該directive的定義:

 app.directive('minuteSecondPicker', function() { return { restrict: 'EA', require: ['minuteSecondPicker', '?^ngModel'], controller: 'minuteSecondPickerController', replace: true, scope: {  validity: '=' }, templateUrl: 'partials/directives/minuteSecondPicker.html', link: function(scope, element, attrs, ctrls) {  var minuteSecondPickerCtrl = ctrls[0],  ngModelCtrl = ctrls[1];  if(ngModelCtrl) {  minuteSecondPickerCtrl.init(ngModelCtrl, element.find('input'));  } } };}); 

在以上的link函數中,ctrls是一個數組: ctrls[0]是定義在本directive上的controller實例,ctrls[1]是ngModelCtrl,即ng-model對應的controller實例。這個順序實際上是通過require: ['minuteSecondPicker', '?^ngModel']定義的。
注意到第一個依賴就是directive本身的名字,此時會將該directive中controller聲明的對應實例傳入。第二個依賴的寫法有些奇怪:"?^ngModel",?的含義是即使沒有找到該依賴,也不要拋出異常,即該依賴是一個可選項。^的含義是查找父元素的controller。
然后,定義該directive中用到的一些默認設置,通過constant directive實現:

app.constant('minuteSecondPickerConfig', { minuteStep: 1, secondStep: 1, readonlyInput: false, mousewheel: true}); 

緊接著是directive對應的controller,它的聲明如下:

 app.controller('minuteSecondPickerController', ['$scope', '$attrs', '$parse', 'minuteSecondPickerConfig',  function($scope, $attrs, $parse, minuteSecondPickerConfig) { ...}]); 

在directive的link函數中,調用了此controller的init方法:

 this.init = function(ngModelCtrl_, inputs) { ngModelCtrl = ngModelCtrl_; ngModelCtrl.$render = this.render; var minutesInputEl = inputs.eq(0),  secondsInputEl = inputs.eq(1); var mousewheel = angular.isDefined($attrs.mousewheel) ?   $scope.$parent.$eval($attrs.mousewheel) : minuteSecondPickerConfig.mousewheel; if(mousewheel) {  this.setupMousewheelEvents(minutesInputEl, secondsInputEl); } $scope.readonlyInput = angular.isDefined($attrs.readonlyInput) ?  $scope.$parent.$eval($attrs.readonlyInput) : minuteSecondPickerConfig.readonlyInput; this.setupInputEvents(minutesInputEl, secondsInputEl); }; 

init方法接受的第二個參數是inputs,在link函數中傳入的是:element.find('input')。 所以第一個輸入框用來輸入分鐘,第二個輸入框用來輸入秒。
然后,檢查是否覆蓋了mousewheel屬性,如果沒有覆蓋則使用在constant中設置的默認mousewheel,并進行相關設置如下:

// respond on mousewheel spin this.setupMousewheelEvents = function(minutesInputEl, secondsInputEl) { var isScrollingUp = function(e) {  if(e.originalEvent) {  e = e.originalEvent;  }  // pick correct delta variable depending on event  var delta = (e.wheelData) ? e.wheelData : -e.deltaY;  return (e.detail || delta > 0); }; minutesInputEl.bind('mousewheel wheel', function(e) {  $scope.$apply((isScrollingUp(e)) ? $scope.incrementMinutes() : $scope.decrementMinutes());  e.preventDefault(); }); secondsInputEl.bind('mousewheel wheel', function(e) {  $scope.$apply((isScrollingUp(e)) ? $scope.incrementSeconds() : $scope.decrementSeconds());  e.preventDefault(); }); }; 

init方法最后會對inputs本身進行一些設置: 

// respond on direct input this.setupInputEvents = function(minutesInputEl, secondsInputEl) { if($scope.readonlyInput) {  $scope.updateMinutes = angular.noop;  $scope.updateSeconds = angular.noop;  return; } var invalidate = function(invalidMinutes, invalidSeconds) {  ngModelCtrl.$setViewValue(null);  ngModelCtrl.$setValidity('time', false);  $scope.validity = false;  if(angular.isDefined(invalidMinutes)) {  $scope.invalidMinutes = invalidMinutes;  }  if(angular.isDefined(invalidSeconds)) {  $scope.invalidSeconds = invalidSeconds;  } }; $scope.updateMinutes = function() {  var minutes = getMinutesFromTemplate();  if(angular.isDefined(minutes)) {  selected.minutes = minutes;  refresh('m');  } else {  invalidate(true);  } }; minutesInputEl.bind('blur', function(e) {  if(!$scope.invalidMinutes && $scope.minutes < 10) {  $scope.$apply(function() {   $scope.minutes = pad($scope.minutes);  });  } }); $scope.updateSeconds = function() {  var seconds = getSecondsFromTemplate();  if(angular.isDefined(seconds)) {  selected.seconds = seconds;  refresh('s');  } else {  invalidate(undefined, true);  } }; secondsInputEl.bind('blur', function(e) {  if(!$scope.invalidSeconds && $scope.seconds < 10) {  $scope.$apply(function() {   $scope.seconds = pad($scope.seconds);  });  } }); }; 

此方法中,聲明了用于設置輸入非法的invalidate函數,它會在scope中暴露一個validity = false屬性讓頁面有機會做出合適的反應。
 如果用戶使用了一個變量來表示minuteStep或者secondStep,那么還需要設置相應的watchers:

var minuteStep = minuteSecondPickerConfig.minuteStep; if($attrs.minuteStep) { $scope.parent.$watch($parse($attrs.minuteStep), function(value) {  minuteStep = parseInt(value, 10); }); } var secondStep = minuteSecondPickerConfig.secondStep; if($attrs.secondStep) { $scope.parent.$watch($parse($attrs.secondStep), function(value) {  secondStep = parseInt(value, 10); }); } 

完整的directive實現代碼如下:

var app = angular.module("minuteSecondPickerDemo");app.directive('minuteSecondPicker', function() { return { restrict: 'EA', require: ['minuteSecondPicker', '?^ngModel'], controller: 'minuteSecondPickerController', replace: true, scope: {  validity: '=' }, templateUrl: 'partials/directives/minuteSecondPicker.html', link: function(scope, element, attrs, ctrls) {  var minuteSecondPickerCtrl = ctrls[0],  ngModelCtrl = ctrls[1];  if(ngModelCtrl) {  minuteSecondPickerCtrl.init(ngModelCtrl, element.find('input'));  } } };});app.constant('minuteSecondPickerConfig', { minuteStep: 1, secondStep: 1, readonlyInput: false, mousewheel: true});app.controller('minuteSecondPickerController', ['$scope', '$attrs', '$parse', 'minuteSecondPickerConfig',  function($scope, $attrs, $parse, minuteSecondPickerConfig) { var selected = {  minutes: 0,  seconds: 0 }, ngModelCtrl = {  $setViewValue: angular.noop }; this.init = function(ngModelCtrl_, inputs) { ngModelCtrl = ngModelCtrl_; ngModelCtrl.$render = this.render; var minutesInputEl = inputs.eq(0),  secondsInputEl = inputs.eq(1); var mousewheel = angular.isDefined($attrs.mousewheel) ?   $scope.$parent.$eval($attrs.mousewheel) : minuteSecondPickerConfig.mousewheel; if(mousewheel) {  this.setupMousewheelEvents(minutesInputEl, secondsInputEl); } $scope.readonlyInput = angular.isDefined($attrs.readonlyInput) ?  $scope.$parent.$eval($attrs.readonlyInput) : minuteSecondPickerConfig.readonlyInput; this.setupInputEvents(minutesInputEl, secondsInputEl); }; var minuteStep = minuteSecondPickerConfig.minuteStep; if($attrs.minuteStep) { $scope.parent.$watch($parse($attrs.minuteStep), function(value) {  minuteStep = parseInt(value, 10); }); } var secondStep = minuteSecondPickerConfig.secondStep; if($attrs.secondStep) { $scope.parent.$watch($parse($attrs.secondStep), function(value) {  secondStep = parseInt(value, 10); }); } // respond on mousewheel spin this.setupMousewheelEvents = function(minutesInputEl, secondsInputEl) { var isScrollingUp = function(e) {  if(e.originalEvent) {  e = e.originalEvent;  }  // pick correct delta variable depending on event  var delta = (e.wheelData) ? e.wheelData : -e.deltaY;  return (e.detail || delta > 0); }; minutesInputEl.bind('mousewheel wheel', function(e) {  $scope.$apply((isScrollingUp(e)) ? $scope.incrementMinutes() : $scope.decrementMinutes());  e.preventDefault(); }); secondsInputEl.bind('mousewheel wheel', function(e) {  $scope.$apply((isScrollingUp(e)) ? $scope.incrementSeconds() : $scope.decrementSeconds());  e.preventDefault(); }); }; // respond on direct input this.setupInputEvents = function(minutesInputEl, secondsInputEl) { if($scope.readonlyInput) {  $scope.updateMinutes = angular.noop;  $scope.updateSeconds = angular.noop;  return; } var invalidate = function(invalidMinutes, invalidSeconds) {  ngModelCtrl.$setViewValue(null);  ngModelCtrl.$setValidity('time', false);  $scope.validity = false;  if(angular.isDefined(invalidMinutes)) {  $scope.invalidMinutes = invalidMinutes;  }  if(angular.isDefined(invalidSeconds)) {  $scope.invalidSeconds = invalidSeconds;  } }; $scope.updateMinutes = function() {  var minutes = getMinutesFromTemplate();  if(angular.isDefined(minutes)) {  selected.minutes = minutes;  refresh('m');  } else {  invalidate(true);  } }; minutesInputEl.bind('blur', function(e) {  if(!$scope.invalidMinutes && $scope.minutes < 10) {  $scope.$apply(function() {   $scope.minutes = pad($scope.minutes);  });  } }); $scope.updateSeconds = function() {  var seconds = getSecondsFromTemplate();  if(angular.isDefined(seconds)) {  selected.seconds = seconds;  refresh('s');  } else {  invalidate(undefined, true);  } }; secondsInputEl.bind('blur', function(e) {  if(!$scope.invalidSeconds && $scope.seconds < 10) {  $scope.$apply(function() {   $scope.seconds = pad($scope.seconds);  });  } }); }; this.render = function() { var time = ngModelCtrl.$modelValue ? {  minutes: ngModelCtrl.$modelValue.minutes,  seconds: ngModelCtrl.$modelValue.seconds } : null; // adjust the time for invalid value at first time if(time.minutes < 0) {  time.minutes = 0; } if(time.seconds < 0) {  time.seconds = 0; } var totalSeconds = time.minutes * 60 + time.seconds; time = {  minutes: Math.floor(totalSeconds / 60),  seconds: totalSeconds % 60 }; if(time) {  selected = time;  makeValid();  updateTemplate(); } }; // call internally when the model is valid function refresh(keyboardChange) { makeValid(); ngModelCtrl.$setViewValue({  minutes: selected.minutes,  seconds: selected.seconds }); updateTemplate(keyboardChange); } function makeValid() { ngModelCtrl.$setValidity('time', true); $scope.validity = true; $scope.invalidMinutes = false; $scope.invalidSeconds = false; } function updateTemplate(keyboardChange) { var minutes = selected.minutes,  seconds = selected.seconds; $scope.minutes = keyboardChange === 'm' ? minutes : pad(minutes); $scope.seconds = keyboardChange === 's' ? seconds : pad(seconds); } function pad(value) { return ( angular.isDefined(value) && value.toString().length < 2 ) ? '0' + value : value; } function getMinutesFromTemplate() { var minutes = parseInt($scope.minutes, 10); return (minutes >= 0) ? minutes : undefined; } function getSecondsFromTemplate() { var seconds = parseInt($scope.seconds, 10); if(seconds >= 60) {  seconds = 59; } return (seconds >= 0) ? seconds : undefined; } $scope.incrementMinutes = function() { addSeconds(minuteStep * 60); }; $scope.decrementMinutes = function() { addSeconds(-minuteStep * 60); }; $scope.incrementSeconds = function() { addSeconds(secondStep); }; $scope.decrementSeconds = function() { addSeconds(-secondStep); }; function addSeconds(seconds) { var newSeconds = selected.minutes * 60 + selected.seconds + seconds; if(newSeconds < 0) {  newSeconds = 0; } selected = {  minutes: Math.floor(newSeconds / 60),  seconds: newSeconds % 60 }; refresh(); } $scope.previewTime = function(minutes, seconds) { var totalSeconds = parseInt(minutes, 10) * 60 + parseInt(seconds, 10),  hh = pad(Math.floor(totalSeconds / 3600)),  mm = pad(minutes % 60),  ss = pad(seconds); return hh + ':' + mm + ':' + ss; };}]); 

對應的Template實現: 

<table> <tbody> <tr class="text-center">  <td>  <a ng-click="incrementMinutes()" class="btn btn-link">   <span class="glyphicon glyphicon-chevron-up"></span>  </a>  </td>  <td> </td>  <td>  <a ng-click="incrementSeconds()" class="btn btn-link">   <span class="glyphicon glyphicon-chevron-up"></span>  </a>  </td>  <td> </td> </tr> <tr>  <td style="width:50px;" class="form-group" ng-class="{'has-error': invalidMinutes}">  <input type="text" ng-model="minutes" ng-change="updateMinutes()" class="form-control text-center" ng-mousewheel="incrementMinutes()" ng-readonly="readonlyInput" maxlength="3">  </td>  <td>:</td>  <td style="width:50px;" class="form-group" ng-class="{'has-error': invalidSeconds}">  <input type="text" ng-model="seconds" ng-change="updateSeconds()" class="form-control text-center" ng-mousewheel="incrementSeconds()" ng-readonly="readonlyInput" maxlength="2">  <td>  <!-- preview column -->  <td>  <span class="label label-primary" ng-show="validity">   {{ previewTime(minutes, seconds) }}  </span>  </td> </tr> <tr class="text-center">  <td>  <a ng-click="decrementMinutes()" class="btn btn-link">   <span class="glyphicon glyphicon-chevron-down"></span>  </a>  </td>  <td> </td>  <td>  <a ng-click="decrementSeconds()" class="btn btn-link">   <span class="glyphicon glyphicon-chevron-down"></span>  </a>  </td>  <td> </td> </tr> </tbody></table> 

測試代碼(即前面截圖dialog的源代碼):

 <div class="modal-header"> <h3 class="modal-title">Highlight on <span class="label label-primary">{{ movieName }}</span></h3></div><div class="modal-body"> <div class="row"> <div id="highlight-start" class="col-xs-6">  <h4>Start Time:</h4>  <minute-second-picker ng-model="startTime" validity="startTimeValidity"></minute-second-picker> </div> <div id="highlight-end" class="col-xs-6">  <h4>End Time:</h4>  <minute-second-picker ng-model="endTime" validity="endTimeValidity"></minute-second-picker> </div> </div> <div class="row"> <div class="col-xs-2">  Tags: </div> <div class="col-xs-10">  <tags model="tags" src="s as s.name for s in sourceTags" options="{ addable: 'true' }"></tags> </div> </div></div><div class="modal-footer"> <button class="btn btn-primary" ng-click="ok()" ng-disabled="!startTimeValidity || !endTimeValidity || durationIncorrect(endTime, startTime)">OK</button> <button class="btn btn-warning" ng-click="cancel()">Cancel</button></div>

如果大家還想深入學習,可以點擊這里進行學習,再為大家附3個精彩的專題:

Bootstrap學習教程

Bootstrap實戰教程

Bootstrap插件使用教程

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
色偷偷偷综合中文字幕;dd| 国产精品av在线| 久久精品视频导航| 日韩av免费观影| 久久男人av资源网站| 国产欧美精品一区二区三区介绍| 色偷偷偷综合中文字幕;dd| 一区二区欧美激情| 4k岛国日韩精品**专区| 日韩激情视频在线播放| 亚洲最大的av网站| 亚洲老司机av| 97视频在线免费观看| 久久久在线视频| 色狠狠av一区二区三区香蕉蜜桃| 美女视频黄免费的亚洲男人天堂| 日韩精品免费一线在线观看| 日韩一区二区三区在线播放| 亚洲一区二区国产| 懂色av一区二区三区| 亚洲国产中文字幕在线观看| 久久久久久国产精品久久| 欧美激情第1页| 欧美黑人极品猛少妇色xxxxx| 亚洲网站在线观看| 欧美理论电影网| 成人国产精品色哟哟| 久久精品这里热有精品| 日韩精品视频在线播放| 伊人精品在线观看| 国产精品99蜜臀久久不卡二区| 久久国产精品久久久久久久久久| 在线播放精品一区二区三区| 在线视频国产日韩| 亚洲毛片在线观看| 亚洲成人久久一区| 国产成人91久久精品| 在线观看欧美日韩国产| 国产成人在线视频| 成人写真视频福利网| 久久久久中文字幕2018| 国产精品中文字幕久久久| 2019国产精品自在线拍国产不卡| 欧美亚洲国产视频| 久久久91精品国产| 91精品综合视频| 国产精品爽爽爽爽爽爽在线观看| 久久久久国产一区二区三区| 在线免费看av不卡| 日韩电影免费在线观看中文字幕| 久久99精品视频一区97| 日韩亚洲一区二区| 欧美成人全部免费| 91欧美视频网站| 亚洲国产免费av| 日韩av电影手机在线观看| 欧美黑人国产人伦爽爽爽| 国产日韩精品综合网站| 国产午夜精品麻豆| 久久免费国产精品1| 国模精品视频一区二区| 国产一区二区三区在线视频| 这里只有精品在线观看| 欧美有码在线观看| 成人国产亚洲精品a区天堂华泰| 26uuu另类亚洲欧美日本一| 欧美日韩成人免费| 91久久久久久久久| 日本久久久久亚洲中字幕| 亚洲欧美一区二区三区四区| 亚洲一区中文字幕| 欧美日韩免费区域视频在线观看| 亚洲精品有码在线| 国产精品视频网站| y97精品国产97久久久久久| 亚洲天堂av网| 欧美日韩国产麻豆| 国产成人精品在线视频| 亚洲图片在线综合| 日本欧美一级片| 欧美性高潮在线| 国内成人精品视频| 日韩中文娱乐网| 国内精品久久久| 91精品久久久久久| 国产精品欧美日韩| 色噜噜国产精品视频一区二区| 一个人看的www欧美| 亚洲黄页视频免费观看| 日韩美女在线观看| 中文字幕亚洲在线| 一区二区三区高清国产| 久热精品视频在线| 久久夜色精品国产亚洲aⅴ| 精品日本高清在线播放| 国产不卡一区二区在线播放| 国产精品第2页| 亚洲福利视频二区| 美女视频久久黄| 日韩美女视频在线观看| 日韩亚洲第一页| 中文字幕在线日韩| 亚洲高清免费观看高清完整版| 亚洲国产欧美日韩精品| 26uuu亚洲国产精品| 在线成人免费网站| 色先锋久久影院av| 亚洲欧美自拍一区| 欧美性生交大片免费| 亚洲精品午夜精品| 91精品啪在线观看麻豆免费| 国产成人精品久久久| 亚洲一区中文字幕在线观看| 亚洲精品成人免费| 欧美最猛黑人xxxx黑人猛叫黄| 国产精品久久久久久久电影| 欧美日韩在线观看视频小说| 国产一区二区视频在线观看| 亚洲精品suv精品一区二区| 欧美高跟鞋交xxxxxhd| 久久精品色欧美aⅴ一区二区| 国产精品mp4| 久久久亚洲欧洲日产国码aⅴ| 日韩中文理论片| 欧美精品一本久久男人的天堂| 青青草国产精品一区二区| 成人黄色大片在线免费观看| 亚洲精品一区中文字幕乱码| 欧美在线视频观看免费网站| 亚洲iv一区二区三区| 欧美成人在线免费视频| 亚洲电影免费观看高清完整版在线| 欧美中文在线观看国产| 国产亚洲精品激情久久| 欧美极度另类性三渗透| 久久久久中文字幕| 国产精品中文字幕久久久| 国产精品久久久久7777婷婷| 色综合伊人色综合网| 久久精品国产一区二区三区| 国内伊人久久久久久网站视频| 国产精品视频内| 国产美女搞久久| 久久人人爽人人爽爽久久| 中文日韩电影网站| 日韩视频永久免费观看| 日本午夜在线亚洲.国产| 亚洲区一区二区| 日韩成人黄色av| 26uuu另类亚洲欧美日本一| 韩国欧美亚洲国产| 成人免费在线视频网站| 性欧美长视频免费观看不卡| 一本色道久久综合亚洲精品小说| 国产成人av网址| 欧美老少做受xxxx高潮| 98精品国产高清在线xxxx天堂| 国产精品video| 热久久这里只有精品| 久久视频在线免费观看| 国产精品成人播放| 国产一区私人高清影院| 日韩在线一区二区三区免费视频| 亚洲自拍偷拍福利|