将 Ember 组件附加到不受 Ember 管理的 DOM 元素

Appending Ember Component to a DOM element not managed by Ember

我想将 Ember 组件 ComponentB 附加到 DOM 元素,该元素由 Ember UI 上的某些非 UI 库生成=15=] 的 ComponentA,结果类似于

<div class='ember-view component-a'>
   <div class='i-know-nothing-of-ember'>
      <div class='ember-view component-b'></div>
   </div>
</div>

我知道 appendTo(element) 方法,但是它因断言

而失败

You cannot append to an existing Ember.View. Consider using Ember.ContainerView instead.

我还尝试在组件 B 上调用 createElement,然后通过 jQuery 将其附加到 DOM - 哪种方法可行,但最终失败并出现错误

Cannot set property '_elementInserted' of null

http://emberjs.jsbin.com/cofebo/2/

实现上述目标的正确方法是什么?如果可能的话,动作和其他行为应该好像 i-know-nothing-of-ember 将由组件 A 模板生成。

我建议使用容器来查找组件并在需要时将其附加到任何地方。

方法 1 - 在路线内检索容器

http://emberjs.jsbin.com/libipazavu/1/edit?html,js,output

js

App = Ember.Application.create();

App.IndexRoute = Em.Route.extend({
  setupController:function(controller,model){
    controller.set("container",this.container);
  }
});

App.IndexView = Em.View.extend({

  appendNonEmberUILibrary:function(){
    callNonEmberUILibrary();
    var componentB = this.get("controller.container").lookup("component:component-b");
    componentB.appendTo(".non-ember-ui");
  }.on("didInsertElement")
});

App.ComponentBComponent = Em.Component.extend({
  layoutName:"components/component-b",
  prop1:"test-option-1"
});


function callNonEmberUILibrary(){
  $("body").append("<div class='non-ember-ui' style='border:1px solid;'>element from non-ember ui lib</div>");
}

hbs

<script type="text/x-handlebars">
    <h2>Welcome to Ember.js</h2>
    <h3>Adding Ember Component to non-Ember DOM Element using <u><i>container</i></u></h3>

    {{outlet}}
  </script>

  <script type="text/x-handlebars" data-template-name="components/component-b">
  <br/>
  <div style="border:1px dashed #F5664D">
  This is componentB ->  {{prop1}}
  </div>
  <br/>
  </script>

方法 2 - 在初始化程序中检索容器

http://emberjs.jsbin.com/hijedowacu/1/edit?html,js,output

js

App = Ember.Application.create();

App.initializer({
  name:"main",
  initialize:function(container,application){
     application.register('main:compB', container.lookup("component:component-b"), { instantiate: false });
     application.inject("controller:index","theComponentB","main:compB");
  }
});

App.IndexView = Em.View.extend({

  appendNonEmberUILibrary:function(){
    callNonEmberUILibrary();
    var componentB = this.get("controller.theComponentB");
    componentB.appendTo(".non-ember-ui");
  }.on("didInsertElement")
});

App.ComponentBComponent = Em.Component.extend({
  layoutName:"components/component-b",
  prop1:"test-option-1"
});


function callNonEmberUILibrary(){
  $("body").append("<div class='non-ember-ui' style='border:1px solid;'>element from non-ember ui lib</div>");
}

hbs

<script type="text/x-handlebars">
    <h2>Welcome to Ember.js</h2>
    <h3>Adding Ember Component to non-Ember DOM Element using <u><i>container</i></u></h3>

    {{outlet}}
  </script>

  <script type="text/x-handlebars" data-template-name="components/component-b">
  <br/>
  <div style="border:1px dashed #F5664D">
  This is componentB ->  {{prop1}}
  </div>
  <br/>
  </script>

方法 3 - 使用 jquery 重新分配添加的组件(根据评论更新)

或者,您可以在模板中将 component-b 添加为不可见,在 didInsertElement 中,您可以使用 jquery.

在任何需要的地方重新分配和显示它

示例http://jsbin.com/maginovexa/1/edit?html,js,output

js

App.IndexView = Em.View.extend({
  prop2:"prop-from-index-view",
  appendNonEmberUILibrary:function(){
    callNonEmberUILibrary();
    //var componentB = this.get("controller.container").lookup("component:component-b");
    //componentB.appendTo(".non-ember-ui");
    var componentBDOM = this.get("componentB").$().detach();
    $(".non-ember-ui").append(componentBDOM);
  }.on("didInsertElement"),
  click:function(){this.set("prop2",Date.now());}
});

App.ComponentBComponent = Em.Component.extend({
  layoutName:"components/component-b",
  prop1:"test-option-1"
});


function callNonEmberUILibrary(){
  $(".inner").append("<div class='non-ember-ui' style='border:1px solid;'>element from non-ember ui lib</div>");
}

hbs

<script type="text/x-handlebars">
    <h2>Welcome to Ember.js</h2>
    <h3>Adding Ember Component to non-Ember DOM Element using <u><i>container</i></u></h3>
    <div class='inner'></div>
    {{outlet}}
  </script>

  <script type="text/x-handlebars" data-template-name="index">
  this is index (click here to change prop2 of component)
  <div style="display:none">
    {{component-b prop2=view.prop2 viewName="componentB"}}
    </div>


  </script>
....

根据 Simon Jesenko 的要求(操作).

您尝试过 运行 循环吗?像这样,

//after comp.createElement();

parent = this;

Ember.run.schedule('afterRender', this, function() {

  parent.$('.i-know-nothing-of-ember').append($(comp.element));

});