forked from cherimarie/gdi-rails
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathclass2.html
472 lines (411 loc) · 18.1 KB
/
class2.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>Intro to Rails</title>
<meta name="description" content="This is an Intro to Rails course, intended for eventual inclusion in the Girl Develop It Core Curriculum. All material by Cheri Allen, inspired by Railsbridge.
The course is meant to be taught in four two-hour workshops. Each of the slides and practice files are customizable according to the needs of a given class or audience.">
<meta name="author" content="Girl Develop It">
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" />
<link rel="stylesheet" href="reveal.js/css/reveal.css">
<link rel="stylesheet" href="reveal.js/css/theme/gdicool.css" id="theme">
<link rel="stylesheet" href="overwrite.css">
<!-- For syntax highlighting -->
<link rel="stylesheet" href="reveal.js/lib/css/light.css">
<!-- <link rel="stylesheet" href="reveal.js/lib/css/dark.css"> -->
<!-- If use the PDF print sheet so students can print slides-->
<link rel="stylesheet" href="reveal.js/css/print/pdf.css" type="text/css" media="print">
<script>document.write('<script src="http://' + (location.host || 'localhost').split(':')[0] + ':35729/livereload.js?snipver=1"></' + 'script>')</script>
</head>
<body>
<div class="reveal">
<!-- Any section element inside of this container is displayed as a slide -->
<div class="slides">
<!-- Opening slide -->
<section>
<img src = "images/gdi_logo_badge.png" alt="Girl Develop It Logo">
<h3>Intro to Rails</h3>
<p>slides: http://gdibtv.github.io/gdi-rails</p>
</section>
<!-- Welcome-->
<section>
<h3>Welcome!</h3>
<div class = "left-align">
<p>Girl Develop It is here to provide affordable and accessible programs to learn software through mentorship and hands-on instruction.</p>
<p class ="green">Some "rules"</p>
<ul>
<li>We are here for you!</li>
<li>Every question is important</li>
<li>Help each other</li>
<li>Have fun</li>
</ul>
</div>
</section>
<section>
<h3>What we will cover today</h3>
<ul>
<li>Homework review</li>
<li>Write user stories for new app</li>
<li>Plan app architecture</li>
<li>Build models</li>
<li>Build data associations</li>
</ul>
</section>
<section>
<h3>Homework Discussion</h3>
<p>Questions?</p>
</section>
<section>
<h3>Best Song App</h3>
<h5>Description</h5>
<p>An application that lists artists and allows users to vote for their favorite songs by artist, and suggest new songs to vote on.</p>
</section>
<section>
<h3>User Stories</h3>
<ul>
<li>Stories guide development- how will your product be used?</li>
<li>Should be for small, testable features</li>
<li>Written with input from the stakeholders</li>
<li>Formal format is: As a (role) I want (goal) so that (reason).</li>
<li>Give them a priority ("required", "nice to have", "v2")</li>
<li>Give them an estimated time/size (how hard will this be to do?)</li>
<li>During the project, stories may change, be broken into smaller pieces, be re-prioritized or re-estimated</li>
</ul>
</section>
<section>
<h3>User Story Examples</h3>
<ul>
<li>As a user, I can backup my entire hard drive.</li>
<li>As a user, I can indicate folders not to backup so that my backup drive isn't filled up with things I don't need saved.</li>
<li>Parking passes can be paid via credit cards.</li>
<li>Students can only enroll in seminars for which they have prerequisites.</li>
</ul>
<a href="http://www.allaboutagile.com/writing-good-user-stories/">More info on writing good user stories...</a>
</section>
<section>
<h3>Best Song App User Stories</h3>
<p>????</p>
<!-- 10 minute discussion, list student suggestions on whiteboard if possible -->
</section>
<section>
<h3>App Architecture</h3>
<ul>
<li>Models</li>
<li>Data Associations</li>
<li>Controllers</li>
<li>Views</li>
</ul>
<br><br>
<p>What do we need?</p>
<!-- 10 minute discussion, write out requirements on a whiteboard if possible -->
</section>
<section>
<h3>Let's Build an App!</h3>
<!-- Change these models, etc. to reflect your class' requirements-->
<h4>Generating Models</h4>
<pre><code class="command-line">
$ rails generate model Artist full_name:string
$ rails generate model Song title:string artist:belongs_to
$ rails generate model Vote song:belongs_to
</code></pre>
</section>
<section>
<h3>Migrations</h3>
<p>Generating models also generated database migrations for us. Let's check them out and make sure they list the attributes we want.</p>
</section>
<section>
<h3>Migrations</h3>
<p>Let's add 'current hairstyle' attribute to Artist table!</p>
<p>Since the migration hasn't been run yet, we can just edit it to add the new column.</p>
<pre><code class="ruby">
def change
create_table :artists do |t|
t.string :full_name
t.string :current_hairstyle
t.timestamps
end
end
</code></pre>
<p>Ok, now that it looks good, run the migrations.</p>
<pre><code class="command-line">
$ rake db:migrate
</code></pre>
</section>
<section>
<h3>Migrations</h3>
<ul>
<li>Migrations describe changes that will be made to the database. </li>
<li>They use a Ruby DSL so you don't have to write SQL.</li>
<li>Each migration modifies the database's schema, which you can see in your db/schema.rb file.</li>
</ul>
<a href="http://guides.rubyonrails.org/migrations.html">Migrations in more detail...</a>
</section>
<section>
<h3>Migrations</h3>
<p class="smalltext">It would be pretty cool if songs has an attribute called 'optimal volume'. Let's create a migration to add that to the table. Run this in the terminal:</p>
<pre><code class="command-line">
$ rails generate migration AddOptimalVolumetoSongs
</code></pre>
<p class="smalltext">And this will be the migration file it generates, with your additions:</p>
<pre><code class="ruby">
class AddOptimalVolumetoSongs < ActiveRecord::Migration
def change
add_column :songs, :optimal_volume, :string
end
end
</code></pre>
<p class="smalltext">Sweet. Now run the migration ('rake db:migrate') so the schema is up to date.</p>
</section>
<section>
<h3>Let's Develop It!</h3>
<p>Generate a migration to add a column called 'hometown' with a type of 'string' to the Artists table.</p>
</section>
<section>
<h3>Data Associations</h3>
<p>The second part of building data associations is in the models.</p>
<pre><code class="ruby">
class Artist < ActiveRecord::Base
has_many :songs
end
class Song < ActiveRecord::Base
belongs_to :artist
has_many :votes
end
class Vote < ActiveRecord::Base
belongs_to :song
end
</code></pre>
</section>
<section>
<h3>Data Associations</h3>
<p>Rails supports six types of associations: </p>
<ul>
<li>belongs_to</li>
<li>has_one</li>
<li>has_many</li>
<li>has_many :through</li>
<li>has_one :through</li>
<li>has_and_belongs_to_many</li>
</ul>
<p>Note: the last two are not used very often.</p>
<a href="http://guides.rubyonrails.org/association_basics.html">Associations in more detail...</a>
</section>
<section>
<h3>Data Associations</h3>
<h4>belongs_to</h4>
<p>This sets up either a one-to-one or one-to-many model with another mdoel.</p>
<ul>
<li>If order <strong>belongs to</strong> customer, each order can be assigned to exactly <strong>one</strong> customer.</li>
<li>In the order table, there will be an entry for 'customer_id: integer'.</li>
<li>The order model will say 'belongs_to :customer'. Note that 'customer' is singular!</li>
<li>The customer model will need to either 'have one' or 'have many' orders.</li>
</ul>
</section>
<section>
<h3>Data Associations</h3>
<h4>belongs_to</h4>
<a href="http://guides.rubyonrails.org/v3.2.21/association_basics.html#the-belongs_to-association"><img src="images/belongs_to.png" alt="belongs to association" /></a>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_one</h4>
<p>This sets up a one-to-one connection with a another model.</p>
<ul>
<li>if supplier <strong>has one</strong> account, each supplier can have exactly <strong>one</strong> associated account.</li>
<li>in the supplier table, there will not be a reference to accounts.</li>
<li>the supplier model will say 'has_one :account'. note that 'account' is singular!</li>
<li>the account model will need to 'belong to' supplier.</li>
</ul>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_one</h4>
<pre>
<code>
class Supplier < ActiveRecord::Base
has_one :account
end
</code>
</pre>
<a href="http://guides.rubyonrails.org/v3.2.21/association_basics.html#the-has_one-association"><img src="images/has_one.png" alt="has one association" /></a>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_many</h4>
<p>This sets up a one-to-many connection with another model.</p>
<ul>
<li>If customer <strong>has many</strong> orders, each customer may be associated with zero or more orders.</li>
<li>In the customer table, there will not be a reference to orders.</li>
<li>The customer model will say 'has_many :orders'. Note that 'orders' is plural!</li>
<li>Orders will need to 'belong to' customer.</li>
</ul>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_many</h4>
<pre>
<code>
class Customer < ActiveRecord::Base
has_many :orders
end
</code>
</pre>
<a href="http://guides.rubyonrails.org/v3.2.21/association_basics.html#the-has_many-association"><img src="images/has_many.png" alt="has many association" /></a>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_many :through</h4>
<p>This sets up a many-to-many connection to another model, through a third model.</p>
<ul>
<li>A patient can have many physicians <strong>through</strong> an appointment and a physician can have many patients <strong>through</strong> an appointment.</li>
<li>In the appointments table, there will be an entry for both physician_id and patient_id.</li>
<li>The appointment model will say 'belongs_to :physician, belongs_to :patient', and the patient model will say 'has_many :appointments, has_many :physicians, through: :appointments'.</li>
</ul>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_many :through</h4>
<pre>
<code>
class Physician < ActiveRecord::Base
has_many :appointments
has_many :patients, :through => :appointments
end
class Appointment < ActiveRecord::Base
belongs_to :physician
belongs_to :patient
end
class Patient < ActiveRecord::Base
has_many :appointments
has_many :physicians, :through => :appointments
end
</code>
</pre>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_many :through</h4>
<a href="http://guides.rubyonrails.org/v3.2.21/association_basics.html#the-has_many-through-association"><img src="images/has_many_through.png" alt="has many through association" /></a>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_and_belongs_to_many</h4>
<p class="smalltext">
A has_and_belongs_to_many association creates a direct many-to-many connection with another model, with no intervening model. For example, if your application includes assemblies and parts, with each assembly having many parts and each part appearing in many assemblies, you could declare the models this way:
</p>
<pre>
<code class="ruby">
class Assembly < ActiveRecord::Base
has_and_belongs_to_many :parts
end
class Part < ActiveRecord::Base
has_and_belongs_to_many :assemblies
end
</code>
</pre>
</section>
<section>
<h3>Data Associations</h3>
<h4>has_and_belongs_to_many (habtm)</h4>
<a href="http://guides.rubyonrails.org/v3.2.21/association_basics.html#the-has_and_belongs_to_many-association"><img src="images/habtm.png" alt="has and belongs to many association" /></a>
</section>
<section>
<h3>Data Associations</h3>
<h4>Choosing between has_many :through and habtm</h4>
<br>
<ol>
<li>Set up a has_many :through relationship if:</li>
<ul>
<li>
You need to work with the relationship model as an independent entity
</li>
</ul>
<br>
<li>Set up a has_and_belongs_to_many relationship if:</li>
<ul>
<li>
You don’t need to do anything with the relationship model
</li>
</ul>
</ol>
<br><br>
<p>
You will need to remember to setup the intermediate table in the database
</p>
</section>
<section>
<h3>Data Associations</h3>
<h4>Index</h4>
<p>A database index is a data structure that improves the speed of data retrieval at the cost of more storage space. An index is a copy of the selected columns, which can be searched very efficiently.</p>
</section>
<section>
<h3>Data Associations</h3>
<h4>Index</h4>
<img src="images/table_index_diagram.jpg" alt="table index diagram" />
</section>
<section>
<h3>Summary</h3>
<p>Today we did this:</p>
<ul>
<li>Planned our application by writing user stories </li>
<li>Discussing the architecture</li>
<li>Then, we generated the model</li>
<li>And hooked up the associations</li>
</ul>
</section>
<section>
<h3>Questions?</h3>
</section>
<section>
<h3>Homework 1</h3>
<p class="smalltext"><a href="http://railsforzombies.org/levels/1">Complete Rails for Zombies</a>
</p>
<img src="images/rails_for_zombies.png" alt="rails for zombies logo" />
</section>
<section>
<h3>Homework 2</h3>
<p>
Use Rails Console with your Best Song App
</p>
<p>Try some commands</p>
<ul>
<li>Create records</li>
<li>Search for records</li>
<li>Update record attributes</li>
<li>Delete records</li>
<li>Associate some record with another one</li>
</ul>
</section>
</div>
<footer>
<div class="copyright">
Intro to Rails
<a rel="license" href="http://creativecommons.org/licenses/by-nc/3.0/deed.en_US"><img alt="Creative Commons License" style="border-width:0" src="http://i.creativecommons.org/l/by-nc/3.0/80x15.png" /></a>
</div>
</footer>
</div>
<script src="reveal.js/lib/js/head.min.js"></script>
<script src="reveal.js/js/reveal.min.js"></script>
<script>
// Full list of configuration options available here:
// https://github.com/hakimel/reveal.js#configuration
Reveal.initialize({
controls: true,
progress: true,
history: true,
theme: Reveal.getQueryHash().theme, // available themes are in /css/theme
transition: Reveal.getQueryHash().transition || 'default', // default/cube/page/concave/zoom/linear/none
// Optional libraries used to extend on reveal.js
dependencies: [
{ src: 'reveal.js/lib/js/classList.js', condition: function() { return !document.body.classList; } },
{ src: 'reveal.js/plugin/markdown/showdown.js', condition: function() { return !!document.querySelector( '[data-markdown]' ); } },
{ src: 'reveal.js/plugin/markdown/markdown.js', condition: function() { return !!document.querySelector( '[data-markdown]' ); } },
{ src: 'reveal.js/plugin/highlight/highlight.js', async: true, callback: function() { hljs.initHighlightingOnLoad(); } },
{ src: 'reveal.js/plugin/zoom-js/zoom.js', async: true, condition: function() { return !!document.body.classList; } },
{ src: 'reveal.js/plugin/notes/notes.js', async: true, condition: function() { return !!document.body.classList; } }
]
});
</script>
</body>
</html>