-
Notifications
You must be signed in to change notification settings - Fork 0
/
active_model_basics.html
1836 lines (1660 loc) · 145 KB
/
active_model_basics.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
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!doctype html>
<html dir="ltr" lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Conceptos Básicos de Active Model — Ruby on Rails Guides</title>
<link rel="stylesheet" type="text/css" href="stylesheets/style-v2.css" data-turbo-track="reload">
<link rel="stylesheet" type="text/css" href="stylesheets/print-v2.css" media="print">
<link rel="stylesheet" type="text/css" href="stylesheets/highlight-v2.css" data-turbo-track="reload">
<link rel="icon" href="images/favicon.ico" sizes="any">
<link rel="apple-touch-icon" href="images/icon.png">
<script src="javascripts/@hotwired--turbo.js" data-turbo-track="reload"></script>
<script src="javascripts/clipboard.js" data-turbo-track="reload"></script>
<script src="javascripts/guides.js" data-turbo-track="reload"></script>
<meta property="og:title" content="Conceptos Básicos de Active Model — Ruby on Rails Guides" />
<meta name="description" content="NO LEA ESTE ARCHIVO EN GITHUB, LAS GUÍAS ESTÁN PUBLICADAS EN https://guides.rubyonrails.org.Conceptos Básicos de Active ModelEsta guía le proporcionará lo que necesita para comenzar a usar Active Model. Active Model proporciona una forma para que los ayudantes de Action Pack y Action View interactúen con objetos Ruby simples. También ayuda a construir ORM personalizados para su uso fuera del marco de Rails.Después de leer esta guía, sabrá: Qué es Active Model y cómo se relaciona con Active Record. Los diferentes módulos que se incluyen en Active Model. Cómo usar Active Model en sus clases." />
<meta property="og:description" content="NO LEA ESTE ARCHIVO EN GITHUB, LAS GUÍAS ESTÁN PUBLICADAS EN https://guides.rubyonrails.org.Conceptos Básicos de Active ModelEsta guía le proporcionará lo que necesita para comenzar a usar Active Model. Active Model proporciona una forma para que los ayudantes de Action Pack y Action View interactúen con objetos Ruby simples. También ayuda a construir ORM personalizados para su uso fuera del marco de Rails.Después de leer esta guía, sabrá: Qué es Active Model y cómo se relaciona con Active Record. Los diferentes módulos que se incluyen en Active Model. Cómo usar Active Model en sus clases." />
<meta property="og:locale" content="en_US" />
<meta property="og:site_name" content="Ruby on Rails Guides" />
<meta property="og:image" content="https://avatars.githubusercontent.com/u/4223" />
<meta property="og:type" content="website" />
<link rel="preconnect" href="https://fonts.googleapis.com">
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
<link href="https://fonts.googleapis.com/css2?family=Noto+Sans+Arabic:[email protected]&display=swap" rel="stylesheet">
<link href="https://fonts.googleapis.com/css2?family=Heebo:[email protected]&family=Noto+Sans+Arabic:[email protected]&display=swap" rel="stylesheet">
<meta name="theme-color" content="#C81418">
</head>
<body class="guide">
<nav id="topNav" aria-label="Secondary">
<div class="wrapper">
<strong class="more-info-label">Más en <a href="https://rubyonrails.org/">rubyonrails.org:</a> </strong>
<span class="red-button more-info-button">
Más Ruby on Rails
</span>
<ul class="more-info-links s-hidden">
<li class="more-info"><a href="https://rubyonrails.org/blog">Blog</a></li>
<li class="more-info"><a href="https://guides.rubyonrails.org/">Guías</a></li>
<li class="more-info"><a href="https://api.rubyonrails.org/">API</a></li>
<li class="more-info"><a href="https://discuss.rubyonrails.org/">Foro</a></li>
<li class="more-info"><a href="https://github.com/rails/rails">Contribuir en GitHub</a></li>
</ul>
</div>
</nav>
<header id="page_header">
<div class="wrapper clearfix">
<nav id="feature_nav">
<div class="header-logo">
<a href="index.html" title="Regresar a la página principal de Guías para Edge">Guías</a>
<span id="version_switcher">
Versión:
<select class="guides-version">
<option value="https://edgeguides.rubyonrails.org/" selected>Edge</option>
<option value="https://guides.rubyonrails.org/v7.2/">7.2</option>
<option value="https://guides.rubyonrails.org/v7.1/">7.1</option>
<option value="https://guides.rubyonrails.org/v7.0/">7.0</option>
<option value="https://guides.rubyonrails.org/v6.1/">6.1</option>
<option value="https://guides.rubyonrails.org/v6.0/">6.0</option>
<option value="https://guides.rubyonrails.org/v5.2/">5.2</option>
<option value="https://guides.rubyonrails.org/v5.1/">5.1</option>
<option value="https://guides.rubyonrails.org/v5.0/">5.0</option>
<option value="https://guides.rubyonrails.org/v4.2/">4.2</option>
<option value="https://guides.rubyonrails.org/v4.1/">4.1</option>
<option value="https://guides.rubyonrails.org/v4.0/">4.0</option>
<option value="https://guides.rubyonrails.org/v3.2/">3.2</option>
<option value="https://guides.rubyonrails.org/v3.1/">3.1</option>
<option value="https://guides.rubyonrails.org/v3.0/">3.0</option>
<option value="https://guides.rubyonrails.org/v2.3/">2.3</option>
</select>
</span>
</div>
<ul class="nav">
<li><a class="nav-item" id="home_nav" href="https://rubyonrails.org/">Inicio</a></li>
<li class="guides-index guides-index-large">
<a href="index.html" id="guidesMenu" class="guides-index-item nav-item">Índice de Guías</a>
<div id="guides" class="clearfix" style="display: none;">
<hr />
<dl class="guides-section-container">
<div class="guides-section">
<dt>Comienza Aquí</dt>
<dd><a href="getting_started.html">Primeros Pasos con Rails</a></dd>
</div>
<div class="guides-section">
<dt>Modelos</dt>
<dd><a href="active_record_basics.html">Conceptos Básicos de Active Record</a></dd>
<dd><a href="active_record_migrations.html">Migraciones de Active Record</a></dd>
<dd><a href="active_record_validations.html">Validaciones de Active Record</a></dd>
</div>
<div class="guides-section">
<dt>Vistas</dt>
<dd><a href="action_view_overview.html">Resumen de Action View</a></dd>
<dd><a href="layouts_and_rendering.html">Diseños y Renderizado en Rails</a></dd>
</div>
<div class="guides-section">
<dt>Controladores</dt>
<dd><a href="action_controller_overview.html">Resumen de Action Controller</a></dd>
<dd><a href="routing.html">Enrutamiento en Rails desde el Exterior</a></dd>
</div>
<div class="guides-section">
<dt>Otros Componentes</dt>
<dd><a href="active_support_core_extensions.html">Extensiones Básicas de Active Support</a></dd>
<dd><a href="action_mailer_basics.html">Conceptos Básicos de Action Mailer</a></dd>
<dd><a href="action_mailbox_basics.html">Conceptos Básicos de Action Mailbox</a></dd>
<dd><a href="action_text_overview.html">Resumen de Action Text</a></dd>
<dd><a href="active_job_basics.html">Conceptos Básicos de Active Job</a></dd>
</div>
<div class="guides-section">
<dt>Políticas</dt>
<dd><a href="maintenance_policy.html">Política de Mantenimiento</a></dd>
</div>
<div class="guides-section">
<dt>Notas de Lanzamiento</dt>
<dd><a href="upgrading_ruby_on_rails.html">Actualizando Ruby on Rails</a></dd>
<dd><a href="7_2_release_notes.html">Versión 7.2 - ?</a></dd>
<dd><a href="7_1_release_notes.html">Versión 7.1 - Octubre 2023</a></dd>
<dd><a href="7_0_release_notes.html">Versión 7.0 - Diciembre 2021</a></dd>
<dd><a href="6_1_release_notes.html">Versión 6.1 - Diciembre 2020</a></dd>
</div>
</dl>
</div>
</li>
<li><a class="nav-item" href="contributing_to_ruby_on_rails.html">Contribuir</a></li>
<li class="guides-index guides-index-small">
<select class="guides-index-item nav-item">
<option value="index.html">Índice de Guías</option>
<optgroup label="Comienza Aquí">
<option value="getting_started.html">Primeros Pasos con Rails</option>
</optgroup>
<optgroup label="Modelos">
<option value="active_record_basics.html">Conceptos Básicos de Active Record</option>
<option value="active_record_migrations.html">Migraciones de Active Record</option>
<option value="active_record_validations.html">Validaciones de Active Record</option>
</optgroup>
<optgroup label="Vistas">
<option value="action_view_overview.html">Resumen de Action View</option>
<option value="layouts_and_rendering.html">Diseños y Renderizado en Rails</option>
</optgroup>
<optgroup label="Controladores">
<option value="action_controller_overview.html">Resumen de Action Controller</option>
<option value="routing.html">Enrutamiento en Rails desde el Exterior</option>
</optgroup>
<optgroup label="Otros Componentes">
<option value="active_support_core_extensions.html">Extensiones Básicas de Active Support</option>
<option value="action_mailer_basics.html">Conceptos Básicos de Action Mailer</option>
<option value="action_mailbox_basics.html">Conceptos Básicos de Action Mailbox</option>
<option value="action_text_overview.html">Resumen de Action Text</option>
<option value="active_job_basics.html">Conceptos Básicos de Active Job</option>
</optgroup>
<optgroup label="Políticas">
<option value="maintenance_policy.html">Política de Mantenimiento</option>
</optgroup>
<optgroup label="Notas de Lanzamiento">
<option value="upgrading_ruby_on_rails.html">Actualizando Ruby on Rails</option>
<option value="7_2_release_notes.html">Versión 7.2 - ?</option>
<option value="7_1_release_notes.html">Versión 7.1 - Octubre 2023</option>
<option value="7_0_release_notes.html">Versión 7.0 - Diciembre 2021</option>
<option value="6_1_release_notes.html">Versión 6.1 - Diciembre 2020</option>
</optgroup>
</select>
</li>
</ul>
</nav>
</div>
</header>
<hr class="hide" />
<section id="feature">
<div class="wrapper">
<p><strong>NO LEA ESTE ARCHIVO EN GITHUB, LAS GUÍAS ESTÁN PUBLICADAS EN <a href="https://guides.rubyonrails.org">https://guides.rubyonrails.org</a>.</strong></p><h1>Conceptos Básicos de Active Model</h1><p>Esta guía le proporcionará lo que necesita para comenzar a usar Active Model. Active Model proporciona una forma para que los ayudantes de Action Pack y Action View interactúen con objetos Ruby simples. También ayuda a construir ORM personalizados para su uso fuera del marco de Rails.</p><p>Después de leer esta guía, sabrá:</p>
<ul>
<li>Qué es Active Model y cómo se relaciona con Active Record.</li>
<li>Los diferentes módulos que se incluyen en Active Model.</li>
<li>Cómo usar Active Model en sus clases.</li>
</ul>
<nav id="subCol">
<h3 class="chapter">
<picture>
<!-- Using the `source` HTML tag to set the dark theme image -->
<source
srcset="images/icon_book-close-bookmark-1-wht.svg"
media="(prefers-color-scheme: dark)"
/>
<img src="images/icon_book-close-bookmark-1.svg" alt="Chapter Icon" />
</picture>
Chapters
</h3>
<ol class="chapters">
<li><a href="#¿qué-es-active-model-questionmark">¿Qué es Active Model?</a>
<ul>
<li><a href="#api">API</a></li>
<li><a href="#model">Model</a></li>
<li><a href="#attributes">Attributes</a></li>
<li><a href="#asignación-de-atributos">Asignación de Atributos</a></li>
<li><a href="#métodos-de-atributo">Métodos de Atributo</a></li>
<li><a href="#callbacks">Callbacks</a></li>
<li><a href="#conversión">Conversión</a></li>
<li><a href="#dirty">Dirty</a></li>
<li><a href="#nombramiento">Nombramiento</a></li>
<li><a href="#securepassword">SecurePassword</a></li>
<li><a href="#serialización">Serialización</a></li>
<li><a href="#traducción">Traducción</a></li>
<li><a href="#validaciones">Validaciones</a></li>
<li><a href="#pruebas-de-lint">Pruebas de Lint</a></li>
</ul></li>
</ol>
</nav>
<hr>
</div>
</section>
<main id="container">
<div class="wrapper">
<div id="mainCol">
<h2 id="¿qué-es-active-model-questionmark"><a class="anchorlink" href="#¿qué-es-active-model-questionmark"><span>1</span> ¿Qué es Active Model?</a></h2><p>Para entender Active Model, necesita saber un poco sobre <a href="active_record_basics.html">Active Record</a>. Active Record es un ORM (Mapeador de Objetos Relacional) que conecta objetos cuyos datos requieren almacenamiento persistente a una base de datos relacional. Sin embargo, tiene funcionalidades que son útiles fuera del ORM, algunas de estas incluyen validaciones, callbacks, traducciones, la capacidad de crear atributos personalizados, etc.</p><p>Parte de esta funcionalidad fue abstraída de Active Record para formar Active Model. Active Model es una biblioteca que contiene varios módulos que se pueden usar en objetos Ruby simples que requieren características similares a los modelos pero no están vinculados a ninguna tabla en una base de datos.</p><p>En resumen, mientras que Active Record proporciona una interfaz para definir modelos que corresponden a tablas de bases de datos, Active Model proporciona funcionalidad para construir clases Ruby similares a modelos que no necesariamente necesitan estar respaldadas por una base de datos. Active Model se puede usar independientemente de Active Record.</p><p>Algunos de estos módulos se explican a continuación.</p><h3 id="api"><a class="anchorlink" href="#api"><span>1.1</span> API</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/API.html"><code>ActiveModel::API</code></a> agrega la capacidad para que una clase funcione con <a href="https://edgeapi.rubyonrails.org/files/actionpack/README_rdoc.html">Action Pack</a> y <a href="action_view_overview.html">Action View</a> directamente.</p><p>Al incluir <code>ActiveModel::API</code>, se incluyen otros módulos por defecto que le permiten obtener características como:</p>
<ul>
<li><a href="#asignación-de-atributos">Asignación de Atributos</a></li>
<li><a href="#conversión">Conversión</a></li>
<li><a href="#nombramiento">Nombramiento</a></li>
<li><a href="#traducción">Traducción</a></li>
<li><a href="#validaciones">Validaciones</a></li>
</ul>
<p>Aquí hay un ejemplo de una clase que incluye <code>ActiveModel::API</code> y cómo se puede usar:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">EmailContact</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">API</span>
<span class="nb">attr_accessor</span> <span class="ss">:name</span><span class="p">,</span> <span class="ss">:email</span><span class="p">,</span> <span class="ss">:message</span>
<span class="n">validates</span> <span class="ss">:name</span><span class="p">,</span> <span class="ss">:email</span><span class="p">,</span> <span class="ss">:message</span><span class="p">,</span> <span class="ss">presence: </span><span class="kp">true</span>
<span class="k">def</span> <span class="nf">deliver</span>
<span class="k">if</span> <span class="n">valid?</span>
<span class="c1"># Entregar correo electrónico</span>
<span class="k">end</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class EmailContact
include ActiveModel::API
attr_accessor :name, :email, :message
validates :name, :email, :message, presence: true
def deliver
if valid?
# Entregar correo electrónico
end
end
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">email_contact</span> <span class="o">=</span> <span class="no">EmailContact</span><span class="p">.</span><span class="nf">new</span><span class="p">(</span><span class="ss">name: </span><span class="s2">"David"</span><span class="p">,</span> <span class="ss">email: </span><span class="s2">"[email protected]"</span><span class="p">,</span> <span class="ss">message: </span><span class="s2">"Hello World"</span><span class="p">)</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">email_contact</span><span class="p">.</span><span class="nf">name</span> <span class="c1"># Asignación de Atributos</span>
<span class="p">=></span> <span class="s2">"David"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">email_contact</span><span class="p">.</span><span class="nf">to_model</span> <span class="o">==</span> <span class="n">email_contact</span> <span class="c1"># Conversión</span>
<span class="p">=></span> <span class="kp">true</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">email_contact</span><span class="p">.</span><span class="nf">model_name</span><span class="p">.</span><span class="nf">name</span> <span class="c1"># Nombramiento</span>
<span class="p">=></span> <span class="s2">"EmailContact"</span>
<span class="gp">irb></span><span class="w"> </span><span class="no">EmailContact</span><span class="p">.</span><span class="nf">human_attribute_name</span><span class="p">(</span><span class="s2">"name"</span><span class="p">)</span> <span class="c1"># Traducción si la configuración regional está establecida</span>
<span class="p">=></span> <span class="s2">"Name"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">email_contact</span><span class="p">.</span><span class="nf">valid?</span> <span class="c1"># Validaciones</span>
<span class="p">=></span> <span class="kp">true</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">empty_contact</span> <span class="o">=</span> <span class="no">EmailContact</span><span class="p">.</span><span class="nf">new</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">empty_contact</span><span class="p">.</span><span class="nf">valid?</span>
<span class="p">=></span> <span class="kp">false</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="email_contact = EmailContact.new(name: "David", email: "[email protected]", message: "Hello World")
email_contact.name # Asignación de Atributos
email_contact.to_model == email_contact # Conversión
email_contact.model_name.name # Nombramiento
EmailContact.human_attribute_name("name") # Traducción si la configuración regional está establecida
email_contact.valid? # Validaciones
empty_contact = EmailContact.new
empty_contact.valid?
">Copy</button>
</div>
<p>Cualquier clase que incluya <code>ActiveModel::API</code> se puede usar con <code>form_with</code>, <code>render</code> y cualquier otro <a href="https://edgeapi.rubyonrails.org/classes/ActionView/Helpers.html">método de ayuda de Action View</a>, al igual que los objetos de Active Record.</p><p>Por ejemplo, <code>form_with</code> se puede usar para crear un formulario para un objeto <code>EmailContact</code> de la siguiente manera:</p><div class="interstitial code">
<pre><code class="highlight erb"><span class="cp"><%=</span> <span class="n">form_with</span> <span class="ss">model: </span><span class="no">EmailContact</span><span class="p">.</span><span class="nf">new</span> <span class="k">do</span> <span class="o">|</span><span class="n">form</span><span class="o">|</span> <span class="cp">%></span>
<span class="cp"><%=</span> <span class="n">form</span><span class="p">.</span><span class="nf">text_field</span> <span class="ss">:name</span> <span class="cp">%></span>
<span class="cp"><%</span> <span class="k">end</span> <span class="cp">%></span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="<%= form_with model: EmailContact.new do |form| %>
<%= form.text_field :name %>
<% end %>
">Copy</button>
</div>
<p>lo que resulta en el siguiente HTML:</p><div class="interstitial code">
<pre><code class="highlight html"><span class="nt"><form</span> <span class="na">action=</span><span class="s">"/email_contacts"</span> <span class="na">method=</span><span class="s">"post"</span><span class="nt">></span>
<span class="nt"><input</span> <span class="na">type=</span><span class="s">"text"</span> <span class="na">name=</span><span class="s">"email_contact[name]"</span> <span class="na">id=</span><span class="s">"email_contact_name"</span><span class="nt">></span>
<span class="nt"></form></span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="<form action="/email_contacts" method="post">
<input type="text" name="email_contact[name]" id="email_contact_name">
</form>
">Copy</button>
</div>
<p><code>render</code> se puede usar para renderizar un parcial con el objeto:</p><div class="interstitial code">
<pre><code class="highlight erb"><span class="cp"><%=</span> <span class="n">render</span> <span class="vi">@email_contact</span> <span class="cp">%></span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="<%= render @email_contact %>
">Copy</button>
</div>
<p>NOTA: Puede aprender más sobre cómo usar <code>form_with</code> y <code>render</code> con objetos compatibles con <code>ActiveModel::API</code> en las guías <a href="form_helpers.html">Ayudantes de Formularios de Action View</a> y <a href="layouts_and_rendering.html">Layouts y Renderizado</a>, respectivamente.</p><h3 id="model"><a class="anchorlink" href="#model"><span>1.2</span> Model</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/Model.html"><code>ActiveModel::Model</code></a> incluye <a href="#api">ActiveModel::API</a> para interactuar con Action Pack y Action View por defecto, y es el enfoque recomendado para implementar clases Ruby similares a modelos. Se extenderá en el futuro para agregar más funcionalidad.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Model</span>
<span class="nb">attr_accessor</span> <span class="ss">:name</span><span class="p">,</span> <span class="ss">:age</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::Model
attr_accessor :name, :age
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span><span class="p">(</span><span class="ss">name: </span><span class="s1">'bob'</span><span class="p">,</span> <span class="ss">age: </span><span class="s1">'18'</span><span class="p">)</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span> <span class="c1"># => "bob"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">age</span> <span class="c1"># => "18"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new(name: 'bob', age: '18')
person.name # => "bob"
person.age # => "18"
">Copy</button>
</div>
<h3 id="attributes"><a class="anchorlink" href="#attributes"><span>1.3</span> Attributes</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/Attributes.html"><code>ActiveModel::Attributes</code></a> le permite definir tipos de datos, establecer valores predeterminados y manejar la conversión y serialización en objetos Ruby simples. Esto puede ser útil para datos de formularios que producirán conversiones similares a Active Record para cosas como fechas y booleanos en objetos regulares.</p><p>Para usar <code>Attributes</code>, incluya el módulo en su clase de modelo y defina sus atributos usando la macro <code>attribute</code>. Acepta un nombre, un tipo de conversión, un valor predeterminado y cualquier otra opción compatible con el tipo de atributo.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Attributes</span>
<span class="n">attribute</span> <span class="ss">:name</span><span class="p">,</span> <span class="ss">:string</span>
<span class="n">attribute</span> <span class="ss">:date_of_birth</span><span class="p">,</span> <span class="ss">:date</span>
<span class="n">attribute</span> <span class="ss">:active</span><span class="p">,</span> <span class="ss">:boolean</span><span class="p">,</span> <span class="ss">default: </span><span class="kp">true</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::Attributes
attribute :name, :string
attribute :date_of_birth, :date
attribute :active, :boolean, default: true
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span> <span class="o">=</span> <span class="s2">"Jane"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="s2">"Jane"</span>
<span class="c"># Convierte la cadena a una fecha establecida por el atributo
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">date_of_birth</span> <span class="o">=</span> <span class="s2">"2020-01-01"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">date_of_birth</span>
<span class="p">=></span> <span class="no">Wed</span><span class="p">,</span> <span class="mo">01</span> <span class="no">Jan</span> <span class="mi">2020</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">date_of_birth</span><span class="p">.</span><span class="nf">class</span>
<span class="p">=></span> <span class="no">Date</span>
<span class="c"># Usa el valor predeterminado establecido por el atributo
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">active</span>
<span class="p">=></span> <span class="kp">true</span>
<span class="c"># Convierte el entero a un booleano establecido por el atributo
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">active</span> <span class="o">=</span> <span class="mi">0</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">active</span>
<span class="p">=></span> <span class="kp">false</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.name = "Jane"
person.name
person.date_of_birth = "2020-01-01"
person.date_of_birth
person.date_of_birth.class
person.active
person.active = 0
person.active
">Copy</button>
</div>
<p>Algunos métodos adicionales descritos a continuación están disponibles cuando se utiliza <code>ActiveModel::Attributes</code>.</p><h4 id="método-attribute-names"><a class="anchorlink" href="#método-attribute-names"><span>1.3.1</span> Método: <code>attribute_names</code></a></h4><p>El método <code>attribute_names</code> devuelve un array de nombres de atributos.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="no">Person</span><span class="p">.</span><span class="nf">attribute_names</span>
<span class="p">=></span> <span class="p">[</span><span class="s2">"name"</span><span class="p">,</span> <span class="s2">"date_of_birth"</span><span class="p">,</span> <span class="s2">"active"</span><span class="p">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Person.attribute_names
">Copy</button>
</div>
<h4 id="método-attributes"><a class="anchorlink" href="#método-attributes"><span>1.3.2</span> Método: <code>attributes</code></a></h4><p>El método <code>attributes</code> devuelve un hash de todos los atributos con sus nombres como claves y los valores de los atributos como valores.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">attributes</span>
<span class="p">=></span> <span class="p">{</span><span class="s2">"name"</span> <span class="o">=></span> <span class="s2">"Jane"</span><span class="p">,</span> <span class="s2">"date_of_birth"</span> <span class="o">=></span> <span class="no">Wed</span><span class="p">,</span> <span class="mo">01</span> <span class="no">Jan</span> <span class="mi">2020</span><span class="p">,</span> <span class="s2">"active"</span> <span class="o">=></span> <span class="kp">false</span><span class="p">}</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person.attributes
">Copy</button>
</div>
<h3 id="asignación-de-atributos"><a class="anchorlink" href="#asignación-de-atributos"><span>1.4</span> Asignación de Atributos</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/AttributeAssignment.html"><code>ActiveModel::AttributeAssignment</code></a> le permite establecer los atributos de un objeto pasando un hash de atributos con claves que coinciden con los nombres de los atributos. Esto es útil cuando desea establecer múltiples atributos a la vez.</p><p>Considere la siguiente clase:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">AttributeAssignment</span>
<span class="nb">attr_accessor</span> <span class="ss">:name</span><span class="p">,</span> <span class="ss">:date_of_birth</span><span class="p">,</span> <span class="ss">:active</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::AttributeAssignment
attr_accessor :name, :date_of_birth, :active
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="err">
</span><span class="c"># Establecer múltiples atributos a la vez
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">assign_attributes</span><span class="p">(</span><span class="ss">name: </span><span class="s2">"John"</span><span class="p">,</span> <span class="ss">date_of_birth: </span><span class="s2">"1998-01-01"</span><span class="p">,</span> <span class="ss">active: </span><span class="kp">false</span><span class="p">)</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="s2">"John"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">date_of_birth</span>
<span class="p">=></span> <span class="no">Thu</span><span class="p">,</span> <span class="mo">01</span> <span class="no">Jan</span> <span class="mi">1998</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">active</span>
<span class="p">=></span> <span class="kp">false</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.assign_attributes(name: "John", date_of_birth: "1998-01-01", active: false)
person.name
person.date_of_birth
person.active
">Copy</button>
</div>
<p>Si el hash pasado responde al método <code>permitted?</code> y el valor de retorno de este método es <code>false</code>, se lanza una excepción <code>ActiveModel::ForbiddenAttributesError</code>.</p><p>NOTA: <code>permitted?</code> se utiliza para la integración de <a href="https://guides.rubyonrails.org/action_controller_overview.html#strong-parameters">parámetros fuertes</a> en la que está asignando un atributo de parámetros de una solicitud.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="err">
</span><span class="c"># Usando comprobaciones de parámetros fuertes, construir un hash de atributos similar a los parámetros de una solicitud
</span><span class="gp">irb></span><span class="w"> </span><span class="n">params</span> <span class="o">=</span> <span class="no">ActionController</span><span class="o">::</span><span class="no">Parameters</span><span class="p">.</span><span class="nf">new</span><span class="p">(</span><span class="ss">name: </span><span class="s2">"John"</span><span class="p">)</span>
<span class="p">=></span> <span class="kt">#<</span><span class="no">ActionController</span><span class="o">::</span><span class="no">Parameters</span> <span class="p">{</span><span class="s2">"name"</span> <span class="o">=</span><span class="kt">></span> <span class="s2">"John"</span><span class="p">}</span> <span class="ss">permitted: </span><span class="kp">false</span><span class="o">></span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">assign_attributes</span><span class="p">(</span><span class="n">params</span><span class="p">)</span>
<span class="p">=></span> <span class="c1"># Raises ActiveModel::ForbiddenAttributesError</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="kp">nil</span>
<span class="c"># Permitir los atributos que queremos permitir la asignación
</span><span class="gp">irb></span><span class="w"> </span><span class="n">permitted_params</span> <span class="o">=</span> <span class="n">params</span><span class="p">.</span><span class="nf">permit</span><span class="p">(</span><span class="ss">:name</span><span class="p">)</span>
<span class="p">=></span> <span class="kt">#<</span><span class="no">ActionController</span><span class="o">::</span><span class="no">Parameters</span> <span class="p">{</span><span class="s2">"name"</span> <span class="o">=</span><span class="kt">></span> <span class="s2">"John"</span><span class="p">}</span> <span class="ss">permitted: </span><span class="kp">true</span><span class="o">></span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">assign_attributes</span><span class="p">(</span><span class="n">permitted_params</span><span class="p">)</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="s2">"John"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
params = ActionController::Parameters.new(name: "John")
person.assign_attributes(params)
person.name
permitted_params = params.permit(:name)
person.assign_attributes(permitted_params)
person.name
">Copy</button>
</div>
<h4 id="alias-del-método-attributes"><a class="anchorlink" href="#alias-del-método-attributes"><span>1.4.1</span> Alias del Método: <code>attributes=</code></a></h4><p>El método <code>assign_attributes</code> tiene un alias <code>attributes=</code>.</p><div class="interstitial info"><p>Un alias de método es un método que realiza la misma acción que otro método, pero se llama de manera diferente. Los alias existen por el bien de la legibilidad y la conveniencia.</p></div><p>El siguiente ejemplo demuestra el uso del método <code>attributes=</code> para establecer múltiples atributos a la vez:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">attributes</span> <span class="o">=</span> <span class="p">{</span> <span class="ss">name: </span><span class="s2">"John"</span><span class="p">,</span> <span class="ss">date_of_birth: </span><span class="s2">"1998-01-01"</span><span class="p">,</span> <span class="ss">active: </span><span class="kp">false</span> <span class="p">}</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="s2">"John"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">date_of_birth</span>
<span class="p">=></span> <span class="s2">"1998-01-01"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.attributes = { name: "John", date_of_birth: "1998-01-01", active: false }
person.name
person.date_of_birth
">Copy</button>
</div>
<div class="interstitial info"><p><code>assign_attributes</code> y <code>attributes=</code> son ambas llamadas de método y aceptan el hash de atributos para asignar como argumento. En muchos casos, Ruby permite que se omitan los paréntesis <code>()</code> de las llamadas a métodos y las llaves <code>{}</code> de las definiciones de hash.<br><br>
Los métodos "setter" como <code>attributes=</code> se escriben comúnmente sin <code>()</code>, aunque incluirlos funciona de la misma manera y requieren que el hash siempre incluya <code>{}</code>. <code>person.attributes=({ name: "John" })</code> está bien, pero <code>person.attributes = name: "John"</code> resulta en un <code>SyntaxError</code>.<br><br>
Otras llamadas a métodos como <code>assign_attributes</code> pueden o no contener tanto paréntesis <code>()</code> como <code>{}</code> para el argumento hash. Por ejemplo, <code>assign_attributes name: "John"</code> y <code>assign_attributes({ name: "John" })</code> son ambos códigos Ruby perfectamente válidos, sin embargo, <code>assign_attributes { name: "John" }</code> no lo es, porque Ruby no puede diferenciar ese argumento hash de un bloque y lanzará un <code>SyntaxError</code>.</p></div><h3 id="métodos-de-atributo"><a class="anchorlink" href="#métodos-de-atributo"><span>1.5</span> Métodos de Atributo</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/AttributeMethods.html"><code>ActiveModel::AttributeMethods</code></a> proporciona una forma de definir métodos dinámicamente para los atributos de un modelo. Este módulo es particularmente útil para simplificar el acceso y manipulación de atributos, y puede agregar prefijos y sufijos personalizados a los métodos de una clase. Puede definir los prefijos y sufijos y qué métodos en el objeto los usarán de la siguiente manera:</p>
<ol>
<li>Incluya <code>ActiveModel::AttributeMethods</code> en su clase.</li>
<li>Llame a cada uno de los métodos que desea agregar, como <code>attribute_method_suffix</code>, <code>attribute_method_prefix</code>, <code>attribute_method_affix</code>.</li>
<li>Llame a <code>define_attribute_methods</code> después de los otros métodos para declarar el(los) atributo(s) que deberían tener prefijos y sufijos.</li>
<li>Defina los diversos métodos genéricos <code>_attribute</code> que ha declarado. El parámetro <code>attribute</code> en estos métodos será reemplazado por el argumento pasado en <code>define_attribute_methods</code>. En el ejemplo a continuación es <code>name</code>.</li>
</ol>
<p>NOTA: <code>attribute_method_prefix</code> y <code>attribute_method_suffix</code> se utilizan para definir los prefijos y sufijos que se usarán para crear los métodos. <code>attribute_method_affix</code> se utiliza para definir tanto el prefijo como el sufijo al mismo tiempo.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">AttributeMethods</span>
<span class="n">attribute_method_affix</span> <span class="ss">prefix: </span><span class="s2">"reset_"</span><span class="p">,</span> <span class="ss">suffix: </span><span class="s2">"_to_default!"</span>
<span class="n">attribute_method_prefix</span> <span class="s2">"first_"</span><span class="p">,</span> <span class="s2">"last_"</span>
<span class="n">attribute_method_suffix</span> <span class="s2">"_short?"</span>
<span class="n">define_attribute_methods</span> <span class="s2">"name"</span>
<span class="nb">attr_accessor</span> <span class="ss">:name</span>
<span class="kp">private</span>
<span class="c1"># Llamada al método de atributo para 'first_name'</span>
<span class="k">def</span> <span class="nf">first_attribute</span><span class="p">(</span><span class="n">attribute</span><span class="p">)</span>
<span class="n">public_send</span><span class="p">(</span><span class="n">attribute</span><span class="p">).</span><span class="nf">split</span><span class="p">.</span><span class="nf">first</span>
<span class="k">end</span>
<span class="c1"># Llamada al método de atributo para 'last_name'</span>
<span class="k">def</span> <span class="nf">last_attribute</span><span class="p">(</span><span class="n">attribute</span><span class="p">)</span>
<span class="n">public_send</span><span class="p">(</span><span class="n">attribute</span><span class="p">).</span><span class="nf">split</span><span class="p">.</span><span class="nf">last</span>
<span class="k">end</span>
<span class="c1"># Llamada al método de atributo para 'name_short?'</span>
<span class="k">def</span> <span class="nf">attribute_short?</span><span class="p">(</span><span class="n">attribute</span><span class="p">)</span>
<span class="n">public_send</span><span class="p">(</span><span class="n">attribute</span><span class="p">).</span><span class="nf">length</span> <span class="o"><</span> <span class="mi">5</span>
<span class="k">end</span>
<span class="c1"># Llamada al método de atributo 'reset_name_to_default!'</span>
<span class="k">def</span> <span class="nf">reset_attribute_to_default!</span><span class="p">(</span><span class="n">attribute</span><span class="p">)</span>
<span class="n">public_send</span><span class="p">(</span><span class="s2">"</span><span class="si">#{</span><span class="n">attribute</span><span class="si">}</span><span class="s2">="</span><span class="p">,</span> <span class="s2">"Default Name"</span><span class="p">)</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::AttributeMethods
attribute_method_affix prefix: "reset_", suffix: "_to_default!"
attribute_method_prefix "first_", "last_"
attribute_method_suffix "_short?"
define_attribute_methods "name"
attr_accessor :name
private
# Llamada al método de atributo para 'first_name'
def first_attribute(attribute)
public_send(attribute).split.first
end
# Llamada al método de atributo para 'last_name'
def last_attribute(attribute)
public_send(attribute).split.last
end
# Llamada al método de atributo para 'name_short?'
def attribute_short?(attribute)
public_send(attribute).length < 5
end
# Llamada al método de atributo 'reset_name_to_default!'
def reset_attribute_to_default!(attribute)
public_send("#{attribute}=", "Default Name")
end
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span> <span class="o">=</span> <span class="s2">"Jane Doe"</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">first_name</span>
<span class="p">=></span> <span class="s2">"Jane"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">last_name</span>
<span class="p">=></span> <span class="s2">"Doe"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name_short?</span>
<span class="p">=></span> <span class="kp">false</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">reset_name_to_default!</span>
<span class="p">=></span> <span class="s2">"Default Name"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.name = "Jane Doe"
person.first_name
person.last_name
person.name_short?
person.reset_name_to_default!
">Copy</button>
</div>
<p>Si llama a un método que no está definido, lanzará un error <code>NoMethodError</code>.</p><h4 id="método-alias-attribute"><a class="anchorlink" href="#método-alias-attribute"><span>1.5.1</span> Método: <code>alias_attribute</code></a></h4><p><code>ActiveModel::AttributeMethods</code> proporciona alias de métodos de atributo usando <code>alias_attribute</code>.</p><p>El ejemplo a continuación crea un atributo alias para <code>name</code> llamado <code>full_name</code>. Devuelven el mismo valor, pero el alias <code>full_name</code> refleja mejor que el atributo incluye un nombre y un apellido.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">AttributeMethods</span>
<span class="n">attribute_method_suffix</span> <span class="s2">"_short?"</span>
<span class="n">define_attribute_methods</span> <span class="ss">:name</span>
<span class="nb">attr_accessor</span> <span class="ss">:name</span>
<span class="n">alias_attribute</span> <span class="ss">:full_name</span><span class="p">,</span> <span class="ss">:name</span>
<span class="kp">private</span>
<span class="k">def</span> <span class="nf">attribute_short?</span><span class="p">(</span><span class="n">attribute</span><span class="p">)</span>
<span class="n">public_send</span><span class="p">(</span><span class="n">attribute</span><span class="p">).</span><span class="nf">length</span> <span class="o"><</span> <span class="mi">5</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::AttributeMethods
attribute_method_suffix "_short?"
define_attribute_methods :name
attr_accessor :name
alias_attribute :full_name, :name
private
def attribute_short?(attribute)
public_send(attribute).length < 5
end
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span> <span class="o">=</span> <span class="s2">"Joe Doe"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name</span>
<span class="p">=></span> <span class="s2">"Joe Doe"</span>
<span class="c"># `full_name` es el alias de `name` y devuelve el mismo valor
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">full_name</span>
<span class="p">=></span> <span class="s2">"Joe Doe"</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">name_short?</span>
<span class="p">=></span> <span class="kp">false</span>
<span class="c"># `full_name_short?` es el alias de `name_short?` y devuelve el mismo valor
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">full_name_short?</span>
<span class="p">=></span> <span class="kp">false</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.name = "Joe Doe"
person.name
person.full_name
person.name_short?
person.full_name_short?
">Copy</button>
</div>
<h3 id="callbacks"><a class="anchorlink" href="#callbacks"><span>1.6</span> Callbacks</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/Callbacks.html"><code>ActiveModel::Callbacks</code></a> proporciona a los objetos Ruby simples <a href="active_record_callbacks.html">callbacks al estilo de Active Record</a>. Los callbacks le permiten engancharse en eventos del ciclo de vida del modelo, como <code>before_update</code> y <code>after_create</code>, así como definir lógica personalizada para ejecutarse en puntos específicos en el ciclo de vida del modelo.</p><p>Puede implementar <code>ActiveModel::Callbacks</code> siguiendo los pasos a continuación:</p>
<ol>
<li>Extienda <code>ActiveModel::Callbacks</code> dentro de su clase.</li>
<li>Emplee <code>define_model_callbacks</code> para establecer una lista de métodos que deberían tener callbacks asociados. Cuando designe un método como <code>:update</code>, incluirá automáticamente los tres callbacks predeterminados (<code>before</code>, <code>around</code> y <code>after</code>) para el evento <code>:update</code>.</li>
<li>Dentro del método definido, utilice <code>run_callbacks</code>, que ejecutará la cadena de callbacks cuando se active el evento específico.</li>
<li>En su clase, puede utilizar los métodos <code>before_update</code>, <code>after_update</code> y <code>around_update</code> de la misma manera que los usaría en un modelo de Active Record.</li>
</ol>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">extend</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Callbacks</span>
<span class="n">define_model_callbacks</span> <span class="ss">:update</span>
<span class="n">before_update</span> <span class="ss">:reset_me</span>
<span class="n">after_update</span> <span class="ss">:finalize_me</span>
<span class="n">around_update</span> <span class="ss">:log_me</span>
<span class="c1"># Método `define_model_callbacks` que contiene `run_callbacks` que ejecuta el callback(s) para el evento dado</span>
<span class="k">def</span> <span class="nf">update</span>
<span class="n">run_callbacks</span><span class="p">(</span><span class="ss">:update</span><span class="p">)</span> <span class="k">do</span>
<span class="nb">puts</span> <span class="s2">"update method called"</span>
<span class="k">end</span>
<span class="k">end</span>
<span class="kp">private</span>
<span class="c1"># Cuando se llama a update en un objeto, este método es llamado por el callback `before_update`</span>
<span class="k">def</span> <span class="nf">reset_me</span>
<span class="nb">puts</span> <span class="s2">"reset_me method: called before the update method"</span>
<span class="k">end</span>
<span class="c1"># Cuando se llama a update en un objeto, este método es llamado por el callback `after_update`</span>
<span class="k">def</span> <span class="nf">finalize_me</span>
<span class="nb">puts</span> <span class="s2">"finalize_me method: called after the update method"</span>
<span class="k">end</span>
<span class="c1"># Cuando se llama a update en un objeto, este método es llamado por el callback `around_update`</span>
<span class="k">def</span> <span class="nf">log_me</span>
<span class="nb">puts</span> <span class="s2">"log_me method: called around the update method"</span>
<span class="k">yield</span>
<span class="nb">puts</span> <span class="s2">"log_me method: block successfully called"</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
extend ActiveModel::Callbacks
define_model_callbacks :update
before_update :reset_me
after_update :finalize_me
around_update :log_me
# Método `define_model_callbacks` que contiene `run_callbacks` que ejecuta el callback(s) para el evento dado
def update
run_callbacks(:update) do
puts "update method called"
end
end
private
# Cuando se llama a update en un objeto, este método es llamado por el callback `before_update`
def reset_me
puts "reset_me method: called before the update method"
end
# Cuando se llama a update en un objeto, este método es llamado por el callback `after_update`
def finalize_me
puts "finalize_me method: called after the update method"
end
# Cuando se llama a update en un objeto, este método es llamado por el callback `around_update`
def log_me
puts "log_me method: called around the update method"
yield
puts "log_me method: block successfully called"
end
end
">Copy</button>
</div>
<p>La clase anterior producirá lo siguiente, lo que indica el orden en que se están llamando los callbacks:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">update</span>
<span class="go">reset_me method: called before the update method
log_me method: called around the update method
update method called
log_me method: block successfully called
finalize_me method: called after the update method
</span><span class="p">=></span> <span class="kp">nil</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.update
">Copy</button>
</div>
<p>Según el ejemplo anterior, al definir un callback 'around', recuerde <code>yield</code> al bloque, de lo contrario, no se ejecutará.</p><p>NOTA: <code>method_name</code> pasado a <code>define_model_callbacks</code> no debe terminar con <code>!</code>, <code>?</code> o <code>=</code>. Además, definir el mismo callback varias veces sobrescribirá las definiciones de callbacks anteriores.</p><h4 id="definición-de-callbacks-específicos"><a class="anchorlink" href="#definición-de-callbacks-específicos"><span>1.6.1</span> Definición de Callbacks Específicos</a></h4><p>Puede optar por crear callbacks específicos pasando la opción <code>only</code> al método <code>define_model_callbacks</code>:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="n">define_model_callbacks</span> <span class="ss">:update</span><span class="p">,</span> <span class="ss">:create</span><span class="p">,</span> <span class="ss">only: </span><span class="p">[</span><span class="ss">:after</span><span class="p">,</span> <span class="ss">:before</span><span class="p">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="define_model_callbacks :update, :create, only: [:after, :before]
">Copy</button>
</div>
<p>Esto creará solo los callbacks <code>before_create</code> / <code>after_create</code> y <code>before_update</code> / <code>after_update</code>, pero omitirá los <code>around_*</code>. La opción se aplicará a todos los callbacks definidos en esa llamada de método. Es posible llamar a <code>define_model_callbacks</code> varias veces para especificar diferentes eventos del ciclo de vida:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="n">define_model_callbacks</span> <span class="ss">:create</span><span class="p">,</span> <span class="ss">only: :after</span>
<span class="n">define_model_callbacks</span> <span class="ss">:update</span><span class="p">,</span> <span class="ss">only: :before</span>
<span class="n">define_model_callbacks</span> <span class="ss">:destroy</span><span class="p">,</span> <span class="ss">only: :around</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="define_model_callbacks :create, only: :after
define_model_callbacks :update, only: :before
define_model_callbacks :destroy, only: :around
">Copy</button>
</div>
<p>Esto creará solo los métodos <code>after_create</code>, <code>before_update</code> y <code>around_destroy</code>.</p><h4 id="definición-de-callbacks-con-una-clase"><a class="anchorlink" href="#definición-de-callbacks-con-una-clase"><span>1.6.2</span> Definición de Callbacks con una Clase</a></h4><p>Puede pasar una clase a <code>before_<type></code>, <code>after_<type></code> y <code>around_<type></code> para tener más control sobre cuándo y en qué contexto se activan sus callbacks. El callback activará el método <code><action>_<type></code> de esa clase, pasando una instancia de la clase como argumento.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">extend</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Callbacks</span>
<span class="n">define_model_callbacks</span> <span class="ss">:create</span>
<span class="n">before_create</span> <span class="no">PersonCallbacks</span>
<span class="k">end</span>
<span class="k">class</span> <span class="nc">PersonCallbacks</span>
<span class="k">def</span> <span class="nc">self</span><span class="o">.</span><span class="nf">before_create</span><span class="p">(</span><span class="n">obj</span><span class="p">)</span>
<span class="c1"># `obj` es la instancia de Person en la que se está llamando el callback</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
extend ActiveModel::Callbacks
define_model_callbacks :create
before_create PersonCallbacks
end
class PersonCallbacks
def self.before_create(obj)
# `obj` es la instancia de Person en la que se está llamando el callback
end
end
">Copy</button>
</div>
<h4 id="abortando-callbacks"><a class="anchorlink" href="#abortando-callbacks"><span>1.6.3</span> Abortando Callbacks</a></h4><p>La cadena de callbacks se puede abortar en cualquier momento lanzando <code>:abort</code>. Esto es similar a cómo funcionan los callbacks de Active Record.</p><p>En el ejemplo a continuación, dado que lanzamos <code>:abort</code> antes de una actualización en el método <code>reset_me</code>, la cadena de callbacks restante, incluido <code>before_update</code>, se abortará, y el cuerpo del método <code>update</code> no se ejecutará.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">extend</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Callbacks</span>
<span class="n">define_model_callbacks</span> <span class="ss">:update</span>
<span class="n">before_update</span> <span class="ss">:reset_me</span>
<span class="n">after_update</span> <span class="ss">:finalize_me</span>
<span class="n">around_update</span> <span class="ss">:log_me</span>
<span class="k">def</span> <span class="nf">update</span>
<span class="n">run_callbacks</span><span class="p">(</span><span class="ss">:update</span><span class="p">)</span> <span class="k">do</span>
<span class="nb">puts</span> <span class="s2">"update method called"</span>
<span class="k">end</span>
<span class="k">end</span>
<span class="kp">private</span>
<span class="k">def</span> <span class="nf">reset_me</span>
<span class="nb">puts</span> <span class="s2">"reset_me method: called before the update method"</span>
<span class="kp">throw</span> <span class="ss">:abort</span>
<span class="nb">puts</span> <span class="s2">"reset_me method: some code after abort"</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">finalize_me</span>
<span class="nb">puts</span> <span class="s2">"finalize_me method: called after the update method"</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">log_me</span>
<span class="nb">puts</span> <span class="s2">"log_me method: called around the update method"</span>
<span class="k">yield</span>
<span class="nb">puts</span> <span class="s2">"log_me method: block successfully called"</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
extend ActiveModel::Callbacks
define_model_callbacks :update
before_update :reset_me
after_update :finalize_me
around_update :log_me
def update
run_callbacks(:update) do
puts "update method called"
end
end
private
def reset_me
puts "reset_me method: called before the update method"
throw :abort
puts "reset_me method: some code after abort"
end
def finalize_me
puts "finalize_me method: called after the update method"
end
def log_me
puts "log_me method: called around the update method"
yield
puts "log_me method: block successfully called"
end
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span>
<span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">update</span>
<span class="go">reset_me method: called before the update method
</span><span class="p">=></span> <span class="kp">false</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new
person.update
">Copy</button>
</div>
<h3 id="conversión"><a class="anchorlink" href="#conversión"><span>1.7</span> Conversión</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/Conversion.html"><code>ActiveModel::Conversion</code></a> es una colección de métodos que le permiten convertir su objeto a diferentes formas para diferentes propósitos. Un caso de uso común es convertir su objeto a una cadena o un entero para construir URLs, campos de formulario y más.</p><p>El módulo <code>ActiveModel::Conversion</code> agrega los siguientes métodos: <code>to_model</code>, <code>to_key</code>, <code>to_param</code> y <code>to_partial_path</code> a las clases.</p><p>Los valores de retorno de los métodos dependen de si <code>persisted?</code> está definido y si se proporciona un <code>id</code>. El método <code>persisted?</code> debe devolver <code>true</code> si el objeto ha sido guardado en la base de datos o almacén, de lo contrario, debe devolver <code>false</code>. El <code>id</code> debe hacer referencia al id del objeto o nil si el objeto no está guardado.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Conversion</span>
<span class="nb">attr_accessor</span> <span class="ss">:id</span>
<span class="k">def</span> <span class="nf">initialize</span><span class="p">(</span><span class="nb">id</span><span class="p">)</span>
<span class="vi">@id</span> <span class="o">=</span> <span class="nb">id</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">persisted?</span>
<span class="nb">id</span><span class="p">.</span><span class="nf">present?</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::Conversion
attr_accessor :id
def initialize(id)
@id = id
end
def persisted?
id.present?
end
end
">Copy</button>
</div>
<h4 id="to-model"><a class="anchorlink" href="#to-model"><span>1.7.1</span> to_model</a></h4><p>El método <code>to_model</code> devuelve el propio objeto.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span> <span class="o">=</span> <span class="no">Person</span><span class="p">.</span><span class="nf">new</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span>
<span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">to_model</span> <span class="o">==</span> <span class="n">person</span>
<span class="p">=></span> <span class="kp">true</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person = Person.new(1)
person.to_model == person
">Copy</button>
</div>
<p>Si su modelo no actúa como un objeto Active Model, entonces debe definir <code>:to_model</code> usted mismo devolviendo un objeto proxy que envuelva su objeto con métodos compatibles con Active Model.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="k">def</span> <span class="nf">to_model</span>
<span class="c1"># Un objeto proxy que envuelve su objeto con métodos compatibles con Active Model.</span>
<span class="no">PersonModel</span><span class="p">.</span><span class="nf">new</span><span class="p">(</span><span class="nb">self</span><span class="p">)</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
def to_model
# Un objeto proxy que envuelve su objeto con métodos compatibles con Active Model.
PersonModel.new(self)
end
end
">Copy</button>
</div>
<h4 id="to-key"><a class="anchorlink" href="#to-key"><span>1.7.2</span> to_key</a></h4><p>El método <code>to_key</code> devuelve un array de los atributos clave del objeto si alguno de los atributos está configurado, independientemente de si el objeto está persistido. Devuelve nil si no hay atributos clave.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">to_key</span>
<span class="p">=></span> <span class="p">[</span><span class="mi">1</span><span class="p">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person.to_key
">Copy</button>
</div>
<p>NOTA: Un atributo clave es un atributo que se utiliza para identificar el objeto. Por ejemplo, en un modelo respaldado por una base de datos, el atributo clave es la clave primaria.</p><h4 id="to-param"><a class="anchorlink" href="#to-param"><span>1.7.3</span> to_param</a></h4><p>El método <code>to_param</code> devuelve una representación <code>string</code> de la clave del objeto adecuada para su uso en URLs, o <code>nil</code> en el caso de que <code>persisted?</code> sea <code>false</code>.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">to_param</span>
<span class="p">=></span> <span class="s2">"1"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person.to_param
">Copy</button>
</div>
<h4 id="to-partial-path"><a class="anchorlink" href="#to-partial-path"><span>1.7.4</span> to_partial_path</a></h4><p>El método <code>to_partial_path</code> devuelve una <code>string</code> que representa la ruta asociada con el objeto. Action Pack utiliza esto para encontrar un parcial adecuado para representar el objeto.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">person</span><span class="p">.</span><span class="nf">to_partial_path</span>
<span class="p">=></span> <span class="s2">"people/person"</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="person.to_partial_path
">Copy</button>
</div>
<h3 id="dirty"><a class="anchorlink" href="#dirty"><span>1.8</span> Dirty</a></h3><p><a href="https://edgeapi.rubyonrails.org/classes/ActiveModel/Dirty.html"><code>ActiveModel::Dirty</code></a> es útil para rastrear cambios realizados en los atributos del modelo antes de que se guarden. Esta funcionalidad le permite determinar qué atributos han sido modificados, cuáles eran sus valores anteriores y actuales, y realizar acciones basadas en esos cambios. Es particularmente útil para auditoría, validación y lógica condicional dentro de su aplicación. Proporciona una forma de rastrear cambios en su objeto de la misma manera que Active Record.</p><p>Un objeto se vuelve "sucio" cuando ha pasado por uno o más cambios en sus atributos y no ha sido guardado. Tiene métodos de acceso basados en atributos.</p><p>Para usar <code>ActiveModel::Dirty</code>, necesita:</p>
<ol>
<li>Incluir el módulo en su clase.</li>
<li>Definir los métodos de atributo para los que desea rastrear cambios, usando <code>define_attribute_methods</code>.</li>
<li>Llamar a <code>[attr_name]_will_change!</code> antes de cada cambio en el atributo rastreado.</li>
<li>Llamar a <code>changes_applied</code> después de que los cambios se hayan persistido.</li>
<li>Llamar a <code>clear_changes_information</code> cuando desee restablecer la información de cambios.</li>
<li>Llamar a <code>restore_attributes</code> cuando desee restaurar los datos anteriores.</li>
</ol>
<p>Luego puede usar los métodos proporcionados por <code>ActiveModel::Dirty</code> para consultar el objeto por su lista de todos los atributos cambiados, los valores originales de los atributos cambiados y los cambios realizados en los atributos.</p><p>Consideremos una clase <code>Person</code> con los atributos <code>first_name</code> y <code>last_name</code> y determinemos cómo podemos usar <code>ActiveModel::Dirty</code> para rastrear cambios en estos atributos.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Person</span>
<span class="kp">include</span> <span class="no">ActiveModel</span><span class="o">::</span><span class="no">Dirty</span>
<span class="nb">attr_reader</span> <span class="ss">:first_name</span><span class="p">,</span> <span class="ss">:last_name</span>
<span class="n">define_attribute_methods</span> <span class="ss">:first_name</span><span class="p">,</span> <span class="ss">:last_name</span>
<span class="k">def</span> <span class="nf">initialize</span>
<span class="vi">@first_name</span> <span class="o">=</span> <span class="kp">nil</span>
<span class="vi">@last_name</span> <span class="o">=</span> <span class="kp">nil</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">first_name</span><span class="o">=</span><span class="p">(</span><span class="n">value</span><span class="p">)</span>
<span class="n">first_name_will_change!</span> <span class="k">unless</span> <span class="n">value</span> <span class="o">==</span> <span class="vi">@first_name</span>
<span class="vi">@first_name</span> <span class="o">=</span> <span class="n">value</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">last_name</span><span class="o">=</span><span class="p">(</span><span class="n">value</span><span class="p">)</span>
<span class="n">last_name_will_change!</span> <span class="k">unless</span> <span class="n">value</span> <span class="o">==</span> <span class="vi">@last_name</span>
<span class="vi">@last_name</span> <span class="o">=</span> <span class="n">value</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">save</span>
<span class="c1"># Persistir datos - borra datos sucios y mueve `changes` a `previous_changes`.</span>
<span class="n">changes_applied</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">reload!</span>
<span class="c1"># Borra todos los datos sucios: cambios actuales y cambios anteriores.</span>
<span class="n">clear_changes_information</span>
<span class="k">end</span>
<span class="k">def</span> <span class="nf">rollback!</span>
<span class="c1"># Restaura todos los datos anteriores de los atributos proporcionados.</span>
<span class="n">restore_attributes</span>
<span class="k">end</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Person
include ActiveModel::Dirty
attr_reader :first_name, :last_name
define_attribute_methods :first_name, :last_name
def initialize
@first_name = nil
@last_name = nil
end
def first_name=(value)
first_name_will_change! unless value == @first_name
@first_name = value
end