国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

angularJs關(guān)于指令的一些冷門屬性詳解

2019-11-20 08:40:54
字體:
供稿:網(wǎng)友

我們使用ng的時(shí)候,經(jīng)常會(huì)使用到指令,大家所熟知的屬性我在這里就不介紹了,講講大家沒怎么留意的屬性

1.multiElement

這是指定指令作用區(qū)間的功能,最常用的就是ng-repeat-start和ng-repeat-end了。

2.priority

指令優(yōu)先級(jí),優(yōu)先級(jí)越高,指令越早執(zhí)行。

3.terminal

是否允許優(yōu)先級(jí)低的指令起作用,如果是true,那么只有比當(dāng)前指令或跟當(dāng)前指令等級(jí)相同的指令才可以執(zhí)行。最典型的就是ngIf

4.templateNamespace

聲明模板的格式有三種選擇 svg、html、math

5.transclude

或許有人疑問了,transclude也算是冷門屬性嗎?其實(shí)大家對(duì)transclude了解并沒有想象的那么深,transclude是一個(gè)挺復(fù)雜的屬性,一般大家會(huì)用到的也僅僅是true,false。這兩個(gè)屬性我在這里就不講了,在這里我主要講的是transclude:element,我google了一整天都沒找到正確描述這個(gè)屬性的方法。我覺得google出來的答案太文檔化了。最后在研究$transclude才看出來這個(gè)屬性的功能究竟在哪里。再講功能前我們先了解下$transclude

無論在指令的compile還是link時(shí)期我們的最后一個(gè)參數(shù)就是$transclude了,這里其實(shí)我們看看源碼是如何定義的,我看的源碼是ng1.5.3的

function controllersBoundTransclude(scope, cloneAttachFn, futureParentElement, slotName) {     var transcludeControllers;     // No scope passed in:     if (!isScope(scope)) {      slotName = futureParentElement;      futureParentElement = cloneAttachFn;      cloneAttachFn = scope;      scope = undefined;     }     if (hasElementTranscludeDirective) {      transcludeControllers = elementControllers;     }     if (!futureParentElement) {      futureParentElement = hasElementTranscludeDirective ? $element.parent() : $element;     }     if (slotName) {      // slotTranscludeFn can be one of three things:      // * a transclude function - a filled slot      // * `null` - an optional slot that was not filled      // * `undefined` - a slot that was not declared (i.e. invalid)      var slotTranscludeFn = boundTranscludeFn.$$slots[slotName];      if (slotTranscludeFn) {       return slotTranscludeFn(scope, cloneAttachFn, transcludeControllers, futureParentElement, scopeToChild);      } else if (isUndefined(slotTranscludeFn)) {       throw $compileMinErr('noslot',        'No parent directive that requires a transclusion with slot name "{0}". ' +        'Element: {1}',        slotName, startingTag($element));      }     } else {      return boundTranscludeFn(scope, cloneAttachFn, transcludeControllers, futureParentElement, scopeToChild);     }    }

還有一個(gè)另一個(gè)函數(shù)要特別指出來,就是最后返回的 boundTranscludeFn 這個(gè)方法,下面是他的源碼

function createBoundTranscludeFn(scope, transcludeFn, previousBoundTranscludeFn) {   function boundTranscludeFn(transcludedScope, cloneFn, controllers, futureParentElement, containingScope) {    if (!transcludedScope) {     transcludedScope = scope.$new(false, containingScope);     transcludedScope.$$transcluded = true;    }    return transcludeFn(transcludedScope, cloneFn, {     parentBoundTranscludeFn: previousBoundTranscludeFn,     transcludeControllers: controllers,     futureParentElement: futureParentElement    });   }

這兩個(gè)方法到底是在做什么呢?其實(shí)就是克隆了當(dāng)前指令的節(jié)點(diǎn),并生成子作用域。克隆的節(jié)點(diǎn)由transclude定義,如果你的屬性是true,則克隆的是指令模板中的ng-transclude所在的DOM節(jié)點(diǎn),及其子節(jié)點(diǎn)。如果屬性是element則克隆整個(gè)模板的節(jié)點(diǎn)。

這是兩個(gè)指令的代碼

angular.module('MyApp', [])      .directive('dropPanel', function() {        return {          transclude: 'element',          replace: true,          template: "<div class='drop-panel'>" +            "<span ng-transclude class='111'></span>" +            "</div>",          link: function(scope, el, c, d, $transclude) {            $transclude(function ngRepeatTransclude(clone, scope) {              console.log(clone);            })          }        }      })      .directive('dropPanel2', function() {        return {          transclude: true,          replace: true,          template: "<div class='drop-panel'>" +            "<span ng-transclude class='111'></span>" +            "</div>",          link: function(scope, el, c, d, $transclude) {            $transclude(function ngRepeatTransclude(clone, scope) {              console.log(clone);            })          }        }      })

如果你覺得replace干擾了對(duì)結(jié)果的理解,你可以注釋掉,然后查看控制臺(tái)中打印出來的clone,你就能知道所謂transclude的屬性聲明為element的作用了,我們打開replace目的在于能較清楚的查看DOM節(jié)點(diǎn),來獲得結(jié)論,下面就是兩者編譯后DOM節(jié)點(diǎn)的區(qū)別了

看完上面的圖,你可以明顯的區(qū)別到兩者對(duì)DOM的克隆不一樣的,另外如果在聲明屬性為‘element'時(shí),需要聲明replace為true,才能渲染出來。我查了很多資料,最終用斷點(diǎn)得出了我認(rèn)為對(duì)的結(jié)論,斷點(diǎn)追蹤的結(jié)果是發(fā)現(xiàn)如果不聲明replace,好像就不會(huì)執(zhí)行ngTransclude指令,這點(diǎn)我很奇怪,正因?yàn)檫@樣子所以導(dǎo)致沒有成功渲染。二歸根結(jié)底其實(shí)是兩者的操作的DOM元素不同,在聲明transclude為element時(shí),replace為true,你取到的DOM節(jié)點(diǎn)是含有transclude屬性的節(jié)點(diǎn)(子節(jié)點(diǎn)),而為false你拿到的并不是含有transclude屬性的節(jié)點(diǎn)(父節(jié)點(diǎn)),而ng本身不對(duì)其節(jié)點(diǎn)進(jìn)行遍歷,導(dǎo)致沒能執(zhí)行ngTransclude指令

我看到一個(gè)觀點(diǎn)覺得不錯(cuò),大概意思就是:源于功能的考慮,在使用element屬性的時(shí)候,一般都是起占位符的作用,你需要做的操作是對(duì)DOM的添加時(shí)候,才會(huì)用到這個(gè)克隆功能。

我覺得這個(gè)觀點(diǎn)不錯(cuò),看過很多關(guān)于ngrepeat的介紹,很多文章都說ngrepeat源碼是通過$scope.$new()來生成子作用域的,實(shí)際上并不完全正確,他的確是通過$scope.$new產(chǎn)生子作用域的,但是這個(gè)產(chǎn)生功能是交給$transclude函數(shù)去做得,實(shí)際上ngrepeat的源碼上是通過$transclude來生成子作用域和添加DOM節(jié)點(diǎn)的。與上面的觀點(diǎn)有相似之處。

以上就是小編為大家?guī)淼腶ngularJs關(guān)于指令的一些冷門屬性詳解全部?jī)?nèi)容了,希望大家多多支持武林網(wǎng)~

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 同德县| 赤峰市| 斗六市| 大宁县| 桑植县| 马关县| 宜兴市| 金川县| 阜宁县| 咸丰县| 墨竹工卡县| 陆良县| 翼城县| 东乡族自治县| 扶风县| 郁南县| 京山县| 梁山县| 颍上县| 西青区| 黑水县| 康平县| 渝北区| 大宁县| 敦化市| 芜湖市| 临武县| 榕江县| 南皮县| 葫芦岛市| 兴仁县| 和平县| 大埔县| 武穴市| 长岛县| 阿拉尔市| 门头沟区| 巴东县| 峡江县| 休宁县| 梅河口市|