Debugging Ember
(without tears)
@vaidehijoshi
data:image/s3,"s3://crabby-images/dd3e3/dd3e3e5216f1e6880926973bb673b616eb382164" alt=""
data:image/s3,"s3://crabby-images/bf5a9/bf5a9853b8d7ef20da0ca8b8991b4f3bef3cc97d" alt=""
💕
💕
community
documentation
easy to use
...most of the time
Debugging
🐞
🐝
data:image/s3,"s3://crabby-images/bb5ec/bb5eca517b4b769811729b62c13057124324cd3e" alt=""
It's hard.
And frustrating.
Maybe I should give up?
get stuck
un
let's debug together!
Bug #1
🐛
data:image/s3,"s3://crabby-images/847f1/847f154fe9a562e70ac1ed148aae6fbbd0a65bea" alt=""
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
data:image/s3,"s3://crabby-images/be450/be4507369ca63fcae3e8c18f8be0353c13fb621b" alt=""
😮
wait...what!?
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
data:image/s3,"s3://crabby-images/be450/be4507369ca63fcae3e8c18f8be0353c13fb621b" alt=""
narrow down the of the problem
scope
the view
<h3>Recipes</h3>
<table>
<thead>
<tr>
<th>Name</th>
<th>Date</th>
</tr>
</thead>
<tbody>
{{#each sortedRecipes key='id' as |recipe|}}
<tr>
<td>
{{#link-to 'recipe' recipe}}{{recipe.name}}{{/link-to}}
</td>
<td>
{{time-format recipe.published_at 'l'}}
</td>
</tr>
{{/each}}
</tbody>
</table>
<hr>
<button {{action "newRecipe"}}>New Recipe</button>
👍
{{#link-to 'recipe' recipe}}
{{recipe.name}}
{{/link-to}}
👍
the route
it's not the view.
maybe it's
// router.js
import Ember from 'ember';
Router.map(function() {
this.route('recipe', {
path: '/recipe/:id'
});
});
dynamic segment
// recipe.js
import Ember from 'ember';
export default Ember.Route.extend({
});
the model
what does our route have access to?
where
does it get access to it?
// recipe.js
import Ember from 'ember';
export default Ember.Route.extend({
});
// model hook!
model(params) {
return this.store.find('recipe', params.id);
}
return a promise
Ember Data record
javaScript object
javaScript array
model(params) {
debugger;
return this.store.find('recipe', params.id);
}
yah yah yah, but will it work?
(also: how does it work?)
data:image/s3,"s3://crabby-images/847f1/847f154fe9a562e70ac1ed148aae6fbbd0a65bea" alt=""
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
data:image/s3,"s3://crabby-images/3cbb4/3cbb432d1636c8b3aeddbc0fd82695f9b33ffd18" alt=""
and when we refresh:
🎊
data:image/s3,"s3://crabby-images/19e94/19e94a5bacf8f550d582c8f2bfa7ba5725227643" alt=""
data:image/s3,"s3://crabby-images/a0afb/a0afbccf73481953505251a8979735af1c0cfcbf" alt=""
in our ember inspector:
import Ember from 'ember';
export default Ember.Route.extend({
// model(params) {
// return this.store.find('recipe', params.id);
// }
});
so.
what's different?
where did we start?
the view
1/
link-to
2/
the route directly
documentation is our friend.
data:image/s3,"s3://crabby-images/86323/8632383d1b0e85c075797d39d572ebcd4889be8e" alt=""
data:image/s3,"s3://crabby-images/6579e/6579e1a0db4f4b17c7697f5bfc6914499b63dd8a" alt=""
our route never knew how to get its model!
🙊
model hooks don't
fire when using
link-to helpers
🐛
Bug #2
🐜
data:image/s3,"s3://crabby-images/e0122/e0122977c0f7b921c91a7b506ece4feb45bf0605" alt=""
Ember Meetup
Party Supplies
data:image/s3,"s3://crabby-images/4e152/4e15201e353f1b721554700cd787950e4b348a61" alt=""
Party Supplies
1. Avocados
2. Pepper Jack Cheese
😱
why are our model properties wrong?
where do model properties come from?
the route
import Ember from 'ember';
export default Ember.Route.extend({
model() {
return this.store.find('grocery-list', params.id);
}
});
import Ember from 'ember';
export default Ember.Controller.extend({
name: null,
items: null
});
data:image/s3,"s3://crabby-images/4e152/4e15201e353f1b721554700cd787950e4b348a61" alt=""
Party Supplies
1. Avocados
2. Pepper Jack Cheese
our controller had values set at least once.
maybe we need to set them again?
does a controller get its properties from?
where
...are we setting them correctly?
data:image/s3,"s3://crabby-images/6579e/6579e1a0db4f4b17c7697f5bfc6914499b63dd8a" alt=""
data:image/s3,"s3://crabby-images/00b17/00b17f5df7ff1944a1a6ceeb14099eb61c4c3491" alt=""
// grocery-list.js
import Ember from 'ember';
export default Ember.Route.extend({
model(params) {
return this.store.find('grocery-list', params.id);
},
setupController(controller, model) {
this._super(controller, model);
}
});
setupController(controller, model) {
this._super(controller, model);
controller.setProperties(
model.getProperties([
'name',
'items'
])
);
}
find model
pull values from it
set them on our controller
data:image/s3,"s3://crabby-images/e0122/e0122977c0f7b921c91a7b506ece4feb45bf0605" alt=""
Ember Meetup
Party Supplies
data:image/s3,"s3://crabby-images/4e152/4e15201e353f1b721554700cd787950e4b348a61" alt=""
Ember Meetup
Tomster food!
we never told our controller to set its properties!
🙊
controllers are singletons
persisted state
🐜
controllers are singletons
routeable components!
controllers are singletons
Bug #3
🐌
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
😭
what do we have
to?
access
what's being passed in?
{{recipe-editor recipe=model afterDestroy='afterDestroy'}}
where do they come from?
import Ember from 'ember';
export default Ember.Route.extend({
model(params) {
return this.store.find('recipe', params.id);
},
actions: {
afterDestroy() {
this.transitionTo('recipes');
}
}
});
<button {{action 'deleteRecipe'}}>
Delete
</button>
how is it being passed?
import Ember from 'ember';
export default Ember.Component.extend({
recipe: null,
saveRecipe() {
// logic to save recipe
},
deleteRecipe() {
const recipe = this.get('recipe');
if (confirm('Are you sure?')) {
recipe.destroyRecord().then(() => {
afterDestroy();
});
}
}
});
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
data:image/s3,"s3://crabby-images/b4d9e/b4d9e8cfc492e13ecc2843ad3f44460cef68c4b0" alt=""
data:image/s3,"s3://crabby-images/57ade/57adeb61d7cc583b312aa60085c969841bce1e35" alt=""
{{recipe-editor recipe=model afterDestroy='afterDestroy'}}
deleteRecipe() {
const recipe = this.get('recipe');
if (confirm('Are you sure?')) {
recipe.destroyRecord().then(() => {
afterDestroy();
});
}
}
this.afterDestroy();
this.sendAction('afterDestroy');
afterDestroy();
👎
🚫
❓
deleteRecipe() {
const recipe = this.get('recipe');
if (confirm('Are you sure?')) {
recipe.destroyRecord().then(() => {
this.sendAction('afterDestroy');
});
}
}
data:image/s3,"s3://crabby-images/6741b/6741b998ffab383d505fee455cb0f8a9728ad3d2" alt=""
data:image/s3,"s3://crabby-images/07c7e/07c7e707bd15c09f38d38e31d929729af263931f" alt=""
🎉
data:image/s3,"s3://crabby-images/cdecd/cdecd95000acd1129f47ff6275008089738b0d9d" alt=""
data:image/s3,"s3://crabby-images/a84c9/a84c9247c17d708ac88101961914b60727a9173c" alt=""
data:image/s3,"s3://crabby-images/6579e/6579e1a0db4f4b17c7697f5bfc6914499b63dd8a" alt=""
// recipe-editor/template.js
<recipe-editor recipe=model destroy={{action 'afterDestroy'}}>
// recipe-editor/component.js
export default Ember.Component.extend({
actions: {
deleteRecipe() {
const recipe = this.get('recipe');
recipe.destroyRecord().then(() => {
this.attrs.destroy();
}
}
}
});
angle-bracket way!
actions are fired on our current context
🐌
data:image/s3,"s3://crabby-images/1886c/1886c0a64fd093cdfb750f31369d53c0bbb0c2a2" alt=""
developing with ember:
superpowers!
Computers
¯\_(ツ)_/¯
data:image/s3,"s3://crabby-images/ae245/ae2451ebab174afe404db14cb38905b60e2dc50e" alt=""
understand what's happening in our code?
not cry while debugging?
understand what's happening in our code?
🐞
getting stuck.
getting unstuck.
asking questions!
all the
check our assumptions
narrow down the scope of the problem
figure out what we have access to
lean on resources + docs
1/
2/
3/
4/
code's
point of view
data:image/s3,"s3://crabby-images/4568f/4568fc6ab9aa2c5976ade92afe0f13aec95ff193" alt=""
frustrated
💖
fun
frustrated
Thanks!
@vaidehijoshi
🐞
Debugging Ember
By Vaidehi Joshi
Debugging Ember
- 1,324