如何获取(当前模板的)父模板实例
How to get the parent template instance (of the current template)
有没有一种干净的方法来获取当前模板的父模板? Meteor 的 API 中没有任何正式记录。
我说的是 Blaze.TemplateInstance
,而不是上下文(即不是 Template.parentData
)。
Is there a clean way to get the parent template of the current
template?
目前,据我所知,none,但这应该在未来某个时间发生,作为计划 "better API for designing reusable components" 的一部分(这在 Meteor post 1.0 roadmap 中讨论)。
目前,这是我在我的项目中使用的解决方法:
// extend Blaze.View prototype to mimick jQuery's closest for views
_.extend(Blaze.View.prototype,{
closest:function(viewName){
var view=this;
while(view){
if(view.name=="Template."+viewName){
return view;
}
view=view.parentView;
}
return null;
}
});
// extend Blaze.TemplateInstance to expose added Blaze.View functionalities
_.extend(Blaze.TemplateInstance.prototype,{
closestInstance:function(viewName){
var view=this.view.closest(viewName);
return view?view.templateInstance():null;
}
});
请注意,这仅支持命名的父模板,并且应该以与 jQuery closest
相同的方式工作,以遍历从子模板到最顶层模板(正文)的父视图节点, 搜索适当命名的模板。
在您的客户端代码中的某处注册了 Blaze 的扩展后,您可以执行以下操作:
HTML
<template name="parent">
<div style="background-color:{{backgroundColor}};">
{{> child}}
</div>
</template>
<template name="child">
<button type="button">Click me to change parent color !</button>
</template>
JS
Template.parent.created=function(){
this.backgroundColor=new ReactiveVar("green");
};
Template.parent.helpers({
backgroundColor:function(){
return Template.instance().backgroundColor.get();
}
});
Template.child.events({
"click button":function(event,template){
var parent=template.closestInstance("parent");
var backgroundColor=parent.backgroundColor.get();
switch(backgroundColor){
case "green":
parent.backgroundColor.set("red");
break;
case "red":
parent.backgroundColor.set("green");
break;
}
}
});
最后,我用 Meteor 的 parentData
类似地扩展了模板实例,如下所示:
/**
* Get the parent template instance
* @param {Number} [levels] How many levels to go up. Default is 1
* @returns {Blaze.TemplateInstance}
*/
Blaze.TemplateInstance.prototype.parentTemplate = function (levels) {
var view = this.view;
if (typeof levels === "undefined") {
levels = 1;
}
while (view) {
if (view.name.substring(0, 9) === "Template." && !(levels--)) {
return view.templateInstance();
}
view = view.parentView;
}
};
用法示例:someTemplate.parentTemplate()
获取直接父级
到目前为止我一直在做的是,如果我需要访问 child 模板函数中的 parent 实例,我会尝试重构此函数以在 parent 模板,然后将其作为参数传递给 child,后者可以执行它。
例如,假设我想在 child 模板中增加 parent 模板上的模板变量。我可以这样写:
Template.parentTemplate.onCreated(function () {
var parentInstance = this;
parentInstance.count = new ReactiveVar(1);
});
Template.parentTemplate.helpers({
incrementHandler: function () {
var parentInstance = Template.instance();
var count = parentInstance.count.get();
return function () {
var newCount = count + 1;
parentInstance.count.set(newCount);
};
}
});
然后包括我的 child 模板:
{{> childTemplate handler=loadMoreHandler}}
并设置我的活动:
Template.childTemplate.events({
'click .increment-button': function (event, childInstance) {
event.preventDefault();
childInstance.data.handler();
}
});
如果您不想扩展 Blaze.TemplateInstance
,您可以像这样访问父实例:
Template.exampleTemplate.onRendered(function () {
const instance = this;
const parentInstance = instance.view.parentView.templateInstance();
});
仅在 Meteor 1 中测试。4.x
您可以使用 Aldeed's template-extension
这样的包
那里提供了以下方法:
templateInstance.parent(numLevels, includeBlockHelpers)
有没有一种干净的方法来获取当前模板的父模板? Meteor 的 API 中没有任何正式记录。
我说的是 Blaze.TemplateInstance
,而不是上下文(即不是 Template.parentData
)。
Is there a clean way to get the parent template of the current template?
目前,据我所知,none,但这应该在未来某个时间发生,作为计划 "better API for designing reusable components" 的一部分(这在 Meteor post 1.0 roadmap 中讨论)。
目前,这是我在我的项目中使用的解决方法:
// extend Blaze.View prototype to mimick jQuery's closest for views
_.extend(Blaze.View.prototype,{
closest:function(viewName){
var view=this;
while(view){
if(view.name=="Template."+viewName){
return view;
}
view=view.parentView;
}
return null;
}
});
// extend Blaze.TemplateInstance to expose added Blaze.View functionalities
_.extend(Blaze.TemplateInstance.prototype,{
closestInstance:function(viewName){
var view=this.view.closest(viewName);
return view?view.templateInstance():null;
}
});
请注意,这仅支持命名的父模板,并且应该以与 jQuery closest
相同的方式工作,以遍历从子模板到最顶层模板(正文)的父视图节点, 搜索适当命名的模板。
在您的客户端代码中的某处注册了 Blaze 的扩展后,您可以执行以下操作:
HTML
<template name="parent">
<div style="background-color:{{backgroundColor}};">
{{> child}}
</div>
</template>
<template name="child">
<button type="button">Click me to change parent color !</button>
</template>
JS
Template.parent.created=function(){
this.backgroundColor=new ReactiveVar("green");
};
Template.parent.helpers({
backgroundColor:function(){
return Template.instance().backgroundColor.get();
}
});
Template.child.events({
"click button":function(event,template){
var parent=template.closestInstance("parent");
var backgroundColor=parent.backgroundColor.get();
switch(backgroundColor){
case "green":
parent.backgroundColor.set("red");
break;
case "red":
parent.backgroundColor.set("green");
break;
}
}
});
最后,我用 Meteor 的 parentData
类似地扩展了模板实例,如下所示:
/**
* Get the parent template instance
* @param {Number} [levels] How many levels to go up. Default is 1
* @returns {Blaze.TemplateInstance}
*/
Blaze.TemplateInstance.prototype.parentTemplate = function (levels) {
var view = this.view;
if (typeof levels === "undefined") {
levels = 1;
}
while (view) {
if (view.name.substring(0, 9) === "Template." && !(levels--)) {
return view.templateInstance();
}
view = view.parentView;
}
};
用法示例:someTemplate.parentTemplate()
获取直接父级
到目前为止我一直在做的是,如果我需要访问 child 模板函数中的 parent 实例,我会尝试重构此函数以在 parent 模板,然后将其作为参数传递给 child,后者可以执行它。
例如,假设我想在 child 模板中增加 parent 模板上的模板变量。我可以这样写:
Template.parentTemplate.onCreated(function () {
var parentInstance = this;
parentInstance.count = new ReactiveVar(1);
});
Template.parentTemplate.helpers({
incrementHandler: function () {
var parentInstance = Template.instance();
var count = parentInstance.count.get();
return function () {
var newCount = count + 1;
parentInstance.count.set(newCount);
};
}
});
然后包括我的 child 模板:
{{> childTemplate handler=loadMoreHandler}}
并设置我的活动:
Template.childTemplate.events({
'click .increment-button': function (event, childInstance) {
event.preventDefault();
childInstance.data.handler();
}
});
如果您不想扩展 Blaze.TemplateInstance
,您可以像这样访问父实例:
Template.exampleTemplate.onRendered(function () {
const instance = this;
const parentInstance = instance.view.parentView.templateInstance();
});
仅在 Meteor 1 中测试。4.x
您可以使用 Aldeed's template-extension
这样的包那里提供了以下方法:
templateInstance.parent(numLevels, includeBlockHelpers)