dynamic-fields-for 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/LICENSE +21 -0
- data/README.md +139 -0
- data/app/assets/javascripts/dynamic-fields-for.js +114 -0
- data/lib/dynamic-fields-for.rb +9 -0
- data/lib/dynamic-fields-for/form_helper.rb +69 -0
- data/lib/dynamic-fields-for/railtie.rb +12 -0
- data/lib/dynamic-fields-for/version.rb +3 -0
- metadata +116 -0
data/LICENSE
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2015 Sergey Tokarenko
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
13
|
+
copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
21
|
+
SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,139 @@
|
|
1
|
+
DynamicFieldsFor
|
2
|
+
================
|
3
|
+
|
4
|
+
DynamicFieldsFor is a Rails plugin which provides the dynamic association fieldsets to your forms without a pain. And it does nothing more.
|
5
|
+
|
6
|
+
The main features are:
|
7
|
+
* Not breaks the HTML layout - no any wrappers, additional divs etc;
|
8
|
+
* Works with fields block, i.e. not requires the separated partial for them;
|
9
|
+
* Not provides new form helpers, but extend the existing one;
|
10
|
+
* Simple and predictable interface and behavior;
|
11
|
+
* Not requires any special HTML intities inside templates;
|
12
|
+
* Supports [Simple Form](https://github.com/plataformatec/simple_form).
|
13
|
+
|
14
|
+
## Alternatives
|
15
|
+
* [coccon](https://github.com/nathanvda/cocoon)
|
16
|
+
* [Nested Form](https://github.com/ryanb/nested_form)
|
17
|
+
|
18
|
+
## Dependencies
|
19
|
+
* [rails](https://github.com/rails/rails) >= 3.2.13
|
20
|
+
* [jquery-rails](https://github.com/rails/jquery-rails)
|
21
|
+
* [association-soft-build](https://github.com/stokarenko/association-soft-build)
|
22
|
+
|
23
|
+
## Getting started
|
24
|
+
|
25
|
+
Add to your Gemfile:
|
26
|
+
|
27
|
+
```ruby
|
28
|
+
gem 'dynamic-fields-for'
|
29
|
+
```
|
30
|
+
|
31
|
+
Run the bundle command to install it.
|
32
|
+
|
33
|
+
Add to `app/assets/javascripts/application.js`:
|
34
|
+
```js
|
35
|
+
//= require dynamic-fields-for
|
36
|
+
```
|
37
|
+
|
38
|
+
## Usage
|
39
|
+
Lets say that we have the models:
|
40
|
+
|
41
|
+
```ruby
|
42
|
+
class User < ActiveRecord::Base
|
43
|
+
has_many :roles
|
44
|
+
end
|
45
|
+
|
46
|
+
class Role < ActiveRecod::Base
|
47
|
+
belongs_to :user
|
48
|
+
|
49
|
+
validates :user, presence: true
|
50
|
+
end
|
51
|
+
```
|
52
|
+
|
53
|
+
First, apply `inverse_of` to User's `:roles` associations, otherwise no chance to pass
|
54
|
+
the validation of Role's user presence on user creation:
|
55
|
+
```ruby
|
56
|
+
class User < ActiveRecord::Base
|
57
|
+
has_many :roles, inverse_of: :user
|
58
|
+
end
|
59
|
+
```
|
60
|
+
|
61
|
+
Add to User model:
|
62
|
+
```ruby
|
63
|
+
accepts_nested_attributes_for :roles, allow_destroy: true
|
64
|
+
```
|
65
|
+
|
66
|
+
Skip `allow_destroy` definition if you don't need to use `remove_fields_link` helper).
|
67
|
+
|
68
|
+
Take care about strong parameters in controller like that:
|
69
|
+
```ruby
|
70
|
+
params.require(:user).permit(roles_attributes: [:id, :_destroy])
|
71
|
+
```
|
72
|
+
|
73
|
+
It is important to permit `id` role's parameter, don't miss it. As for `_destroy`,
|
74
|
+
skip it if you don't need to use `remove_fields_link` helper.
|
75
|
+
|
76
|
+
Then, in view:
|
77
|
+
```haml
|
78
|
+
= form_for resource do |f|
|
79
|
+
= f.text_field :user_name
|
80
|
+
|
81
|
+
= f.fields_for :roles, dynamic: true do |rf|
|
82
|
+
= rf.text_field :role_name
|
83
|
+
= rf.remove_fields_link 'Remove role'
|
84
|
+
|
85
|
+
= f.add_fields_link :roles, 'Add role'
|
86
|
+
|
87
|
+
= f.submit
|
88
|
+
```
|
89
|
+
|
90
|
+
DynamicFieldsFor supports SimpleForm:
|
91
|
+
```haml
|
92
|
+
= simple_form_for resource do |f|
|
93
|
+
= f.input :user_name
|
94
|
+
|
95
|
+
= f.simple_fields_for :roles, dynamic: true do |rf|
|
96
|
+
= rf.input :role_name
|
97
|
+
= rf.remove_fields_link 'Remove role'
|
98
|
+
|
99
|
+
= f.add_fields_link :roles, 'Add role'
|
100
|
+
|
101
|
+
= f.submit
|
102
|
+
```
|
103
|
+
|
104
|
+
## JavaScript events
|
105
|
+
There are the events which will be triggered on `add_fields_link` click, in actual order:
|
106
|
+
* `dynamic-fields:before-add-into` touched to dynamic fields parent node;
|
107
|
+
* `dynamic-fields:after-add` touched to each first-level elements which was inserted;
|
108
|
+
* `dynamic-fields:after-add-into` touched to dynamic fields parent node;
|
109
|
+
|
110
|
+
Like that, these events will be triggered on `add_fields_link` click, in actual order:
|
111
|
+
* `dynamic-fields:before-remove-from` touched to dynamic fields parent node;
|
112
|
+
* `dynamic-fields:before-remove` touched to each first-level elements which going to be removed;
|
113
|
+
* `dynamic-fields:after-remove-from` touched to dynamic fields parent node;
|
114
|
+
|
115
|
+
Typical callback for dynamic fields parent node looks like:
|
116
|
+
```js
|
117
|
+
$(document).on('dynamic-fields:after-add-into', function(event){
|
118
|
+
$(event.target).find('li').order();
|
119
|
+
})
|
120
|
+
```
|
121
|
+
|
122
|
+
As for first-level elements, need to remember that compatible callbacks
|
123
|
+
will be triggered to each of them. To deal with this,
|
124
|
+
use `$.find2` javascript helper, which provided by DynamicFieldsFor:
|
125
|
+
```js
|
126
|
+
$('#some_id').find2('.some_class');
|
127
|
+
// doing the same as...
|
128
|
+
$('#some_id').find('.some_class').add($('#some_id').filter('.some_class'));
|
129
|
+
```
|
130
|
+
|
131
|
+
Typical event callback first-level elements should looks like:
|
132
|
+
```js
|
133
|
+
$(document).on('dynamic-fields:after-add', function(event){
|
134
|
+
$(event.target).find2('.datepicker').datetimepicker();
|
135
|
+
})
|
136
|
+
```
|
137
|
+
|
138
|
+
## License
|
139
|
+
MIT License. Copyright (c) 2015 Sergey Tokarenko
|
@@ -0,0 +1,114 @@
|
|
1
|
+
(function() {
|
2
|
+
var DynamicFields;
|
3
|
+
|
4
|
+
$.fn.extend({
|
5
|
+
find2: function(selector) {
|
6
|
+
return this.filter(selector).add(this.find(selector));
|
7
|
+
}
|
8
|
+
});
|
9
|
+
|
10
|
+
DynamicFields = (function() {
|
11
|
+
DynamicFields._counter = 0;
|
12
|
+
|
13
|
+
DynamicFields.add = function(element) {
|
14
|
+
var lazy_dynamic_fields;
|
15
|
+
lazy_dynamic_fields = this._find($(element).data('dynamicFieldsAdd'));
|
16
|
+
if (lazy_dynamic_fields) {
|
17
|
+
return lazy_dynamic_fields.add();
|
18
|
+
}
|
19
|
+
};
|
20
|
+
|
21
|
+
DynamicFields.remove = function(element) {
|
22
|
+
var lazy_dynamic_fields;
|
23
|
+
lazy_dynamic_fields = this._find($(element).data('dynamicFieldsRemove'));
|
24
|
+
if (lazy_dynamic_fields) {
|
25
|
+
return lazy_dynamic_fields.remove($(element));
|
26
|
+
}
|
27
|
+
};
|
28
|
+
|
29
|
+
DynamicFields._find = function(fields_id) {
|
30
|
+
var $fields_begin, fields_begin;
|
31
|
+
$fields_begin = this._$anchor('begin', fields_id);
|
32
|
+
fields_begin = $fields_begin.get(0);
|
33
|
+
if (!fields_begin) {
|
34
|
+
return null;
|
35
|
+
}
|
36
|
+
return fields_begin.lazy_dynamic_fields || (fields_begin.lazy_dynamic_fields = new this($fields_begin));
|
37
|
+
};
|
38
|
+
|
39
|
+
DynamicFields._anchor = function(postfix, fields_id) {
|
40
|
+
return "[data-dynamic-fields-" + postfix + "=" + fields_id + "]";
|
41
|
+
};
|
42
|
+
|
43
|
+
DynamicFields._$anchor = function(postfix, fields_id) {
|
44
|
+
return $(this._anchor(postfix, fields_id));
|
45
|
+
};
|
46
|
+
|
47
|
+
function DynamicFields($fields_begin) {
|
48
|
+
this.fields_id = $fields_begin.data('dynamicFieldsBegin');
|
49
|
+
this.$fields_begin = $fields_begin;
|
50
|
+
this.$fields_end = this._$anchor('end');
|
51
|
+
this.$fields_parent = this.$fields_begin.parent();
|
52
|
+
}
|
53
|
+
|
54
|
+
DynamicFields.prototype.add = function() {
|
55
|
+
var fields_add;
|
56
|
+
fields_add = $(this._render_insertion('Add'));
|
57
|
+
this.$fields_parent.trigger('dynamic-fields:before-add-into');
|
58
|
+
this.$fields_end.before(fields_add);
|
59
|
+
fields_add.not('script').trigger('dynamic-fields:after-add');
|
60
|
+
return this.$fields_parent.trigger('dynamic-fields:after-add-into');
|
61
|
+
};
|
62
|
+
|
63
|
+
DynamicFields.prototype.remove = function($element) {
|
64
|
+
var destroy_inputs, fields_item_begin, fields_remove, node, object_id;
|
65
|
+
node = $element;
|
66
|
+
while ((fields_item_begin = node.prevAll((this._anchor('item-begin')) + ":first"), fields_item_begin.length === 0)) {
|
67
|
+
node = node.parent();
|
68
|
+
if (node.length === 0) {
|
69
|
+
break;
|
70
|
+
}
|
71
|
+
}
|
72
|
+
if (fields_item_begin.length === 0) {
|
73
|
+
return;
|
74
|
+
}
|
75
|
+
if ((object_id = $element.data('dynamicFieldsRemoveId'))) {
|
76
|
+
destroy_inputs = this._render_insertion('Remove').replace(/(["])dynamic_fields_object_id(["])/g, "$1" + object_id + "$2");
|
77
|
+
this.$fields_begin.after(destroy_inputs);
|
78
|
+
}
|
79
|
+
fields_remove = fields_item_begin.nextUntil((this._anchor('item-begin')) + ", " + (this._anchor('end'))).andSelf();
|
80
|
+
this.$fields_parent.trigger('dynamic-fields:before-remove-from');
|
81
|
+
fields_remove.not('script').trigger('dynamic-fields:before-remove');
|
82
|
+
fields_remove.remove();
|
83
|
+
return this.$fields_parent.trigger('dynamic-fields:after-remove-from');
|
84
|
+
};
|
85
|
+
|
86
|
+
DynamicFields.prototype._anchor = function(postfix) {
|
87
|
+
return this.constructor._anchor(postfix, this.fields_id);
|
88
|
+
};
|
89
|
+
|
90
|
+
DynamicFields.prototype._$anchor = function(postfix) {
|
91
|
+
return this.constructor._$anchor(postfix, this.fields_id);
|
92
|
+
};
|
93
|
+
|
94
|
+
DynamicFields.prototype._render_insertion = function(template_type) {
|
95
|
+
var template;
|
96
|
+
template = this.$fields_begin.data("dynamicFields" + template_type + "Template");
|
97
|
+
return template.replace(/([_\[])dynamic_fields_index([_\]])/g, "$1" + (new Date().getTime() + this.constructor._counter++) + "$2");
|
98
|
+
};
|
99
|
+
|
100
|
+
return DynamicFields;
|
101
|
+
|
102
|
+
})();
|
103
|
+
|
104
|
+
$(document).on('click', '[data-dynamic-fields-remove]', function(event) {
|
105
|
+
event.preventDefault();
|
106
|
+
return DynamicFields.remove(this);
|
107
|
+
});
|
108
|
+
|
109
|
+
$(document).on('click', '[data-dynamic-fields-add]', function(event) {
|
110
|
+
event.preventDefault();
|
111
|
+
return DynamicFields.add(this);
|
112
|
+
});
|
113
|
+
|
114
|
+
}).call(this);
|
@@ -0,0 +1,69 @@
|
|
1
|
+
module DynamicFieldsFor
|
2
|
+
module FormHelper
|
3
|
+
extend ActiveSupport::Concern
|
4
|
+
|
5
|
+
included do
|
6
|
+
alias_method_chain :fields_for, :dynamic_fields
|
7
|
+
alias_method_chain :fields_for_nested_model, :dynamic_fields
|
8
|
+
end
|
9
|
+
|
10
|
+
def fields_for_with_dynamic_fields(association, record_object = nil, options = {}, &block)
|
11
|
+
#Inherit the native parameters adjustment
|
12
|
+
options, record_object = record_object, nil if record_object.is_a?(Hash) && record_object.extractable_options?
|
13
|
+
|
14
|
+
return fields_for_without_dynamic_fields(association, record_object, options, &block) unless options.delete(:dynamic)
|
15
|
+
|
16
|
+
new_object = @object.send(association).soft_build
|
17
|
+
|
18
|
+
options[:child_index] = 'dynamic_fields_index'
|
19
|
+
remove_template = fields_for_without_dynamic_fields(association, new_object, options) do |f|
|
20
|
+
f.hidden_field(:id, value: 'dynamic_fields_object_id') +
|
21
|
+
f.hidden_field(:_destroy, value: true)
|
22
|
+
end
|
23
|
+
|
24
|
+
options[:dynamic_fields_id] = dynamic_fields_id(association)
|
25
|
+
add_template = fields_for_without_dynamic_fields(association, new_object, options, &block)
|
26
|
+
|
27
|
+
options.delete(:child_index)
|
28
|
+
collection_output = fields_for_without_dynamic_fields(association, record_object, options, &block)
|
29
|
+
|
30
|
+
@template.content_tag(:script, nil, data: {
|
31
|
+
'dynamic-fields-begin' => options[:dynamic_fields_id],
|
32
|
+
'dynamic-fields-add-template' => CGI.escapeHTML(add_template).html_safe,
|
33
|
+
'dynamic-fields-remove-template' => CGI.escapeHTML(remove_template).html_safe
|
34
|
+
}) +
|
35
|
+
collection_output +
|
36
|
+
@template.content_tag(:script, nil, data: {
|
37
|
+
'dynamic-fields-end' => options[:dynamic_fields_id]
|
38
|
+
})
|
39
|
+
end
|
40
|
+
|
41
|
+
def add_fields_link(association, label, options = {})
|
42
|
+
@template.link_to(label, '#', {data: {'dynamic-fields-add' => dynamic_fields_id(association)}}.deep_merge(options))
|
43
|
+
end
|
44
|
+
|
45
|
+
def remove_fields_link(label, options = {})
|
46
|
+
data_options = {data: {'dynamic-fields-remove' => @options[:dynamic_fields_id]}}
|
47
|
+
data_options[:data]['dynamic-fields-remove-id'] = @object.id if @object.try(:persisted?)
|
48
|
+
|
49
|
+
@template.link_to(label, '#', data_options.deep_merge(options))
|
50
|
+
end
|
51
|
+
|
52
|
+
def fields_for_nested_model_with_dynamic_fields(name, object, fields_options, block)
|
53
|
+
dynamic_item_begin_mark = fields_options.has_key?(:dynamic_fields_id) ?
|
54
|
+
@template.content_tag(:script, nil, data: {
|
55
|
+
'dynamic-fields-item-begin' => fields_options[:dynamic_fields_id]
|
56
|
+
}) : ''.html_safe
|
57
|
+
|
58
|
+
dynamic_item_begin_mark +
|
59
|
+
fields_for_nested_model_without_dynamic_fields(name, object, fields_options, block)
|
60
|
+
end
|
61
|
+
|
62
|
+
private
|
63
|
+
|
64
|
+
def dynamic_fields_id(association)
|
65
|
+
"#{self.object_id}-#{association}"
|
66
|
+
end
|
67
|
+
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,12 @@
|
|
1
|
+
require 'active_record/railtie'
|
2
|
+
|
3
|
+
module DynamicFieldsFor
|
4
|
+
class Engine < Rails::Engine
|
5
|
+
|
6
|
+
initializer 'dynamic-fields-for.configure_rails_initialization' do
|
7
|
+
ActiveSupport.on_load :action_view do
|
8
|
+
ActionView::Helpers::FormBuilder.send :include, DynamicFieldsFor::FormHelper
|
9
|
+
end
|
10
|
+
end
|
11
|
+
end
|
12
|
+
end
|
metadata
ADDED
@@ -0,0 +1,116 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: dynamic-fields-for
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 1.0.0
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Sergey Tokarenko
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2015-05-09 00:00:00.000000000 Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: rails
|
16
|
+
requirement: !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ! '>='
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: 3.2.13
|
22
|
+
type: :runtime
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: !ruby/object:Gem::Requirement
|
25
|
+
none: false
|
26
|
+
requirements:
|
27
|
+
- - ! '>='
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: 3.2.13
|
30
|
+
- !ruby/object:Gem::Dependency
|
31
|
+
name: jquery-rails
|
32
|
+
requirement: !ruby/object:Gem::Requirement
|
33
|
+
none: false
|
34
|
+
requirements:
|
35
|
+
- - ! '>='
|
36
|
+
- !ruby/object:Gem::Version
|
37
|
+
version: '0'
|
38
|
+
type: :runtime
|
39
|
+
prerelease: false
|
40
|
+
version_requirements: !ruby/object:Gem::Requirement
|
41
|
+
none: false
|
42
|
+
requirements:
|
43
|
+
- - ! '>='
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: '0'
|
46
|
+
- !ruby/object:Gem::Dependency
|
47
|
+
name: association-soft-build
|
48
|
+
requirement: !ruby/object:Gem::Requirement
|
49
|
+
none: false
|
50
|
+
requirements:
|
51
|
+
- - ! '>='
|
52
|
+
- !ruby/object:Gem::Version
|
53
|
+
version: '0'
|
54
|
+
type: :runtime
|
55
|
+
prerelease: false
|
56
|
+
version_requirements: !ruby/object:Gem::Requirement
|
57
|
+
none: false
|
58
|
+
requirements:
|
59
|
+
- - ! '>='
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '0'
|
62
|
+
- !ruby/object:Gem::Dependency
|
63
|
+
name: bundler
|
64
|
+
requirement: !ruby/object:Gem::Requirement
|
65
|
+
none: false
|
66
|
+
requirements:
|
67
|
+
- - ! '>='
|
68
|
+
- !ruby/object:Gem::Version
|
69
|
+
version: '0'
|
70
|
+
type: :development
|
71
|
+
prerelease: false
|
72
|
+
version_requirements: !ruby/object:Gem::Requirement
|
73
|
+
none: false
|
74
|
+
requirements:
|
75
|
+
- - ! '>='
|
76
|
+
- !ruby/object:Gem::Version
|
77
|
+
version: '0'
|
78
|
+
description: Dynamic association fieldsets without pain.
|
79
|
+
email: private.tokarenko.sergey@gmail.com
|
80
|
+
executables: []
|
81
|
+
extensions: []
|
82
|
+
extra_rdoc_files: []
|
83
|
+
files:
|
84
|
+
- app/assets/javascripts/dynamic-fields-for.js
|
85
|
+
- lib/dynamic-fields-for/form_helper.rb
|
86
|
+
- lib/dynamic-fields-for/railtie.rb
|
87
|
+
- lib/dynamic-fields-for/version.rb
|
88
|
+
- lib/dynamic-fields-for.rb
|
89
|
+
- LICENSE
|
90
|
+
- README.md
|
91
|
+
homepage: https://github.com/stokarenko/dynamic-fields-for
|
92
|
+
licenses:
|
93
|
+
- MIT
|
94
|
+
post_install_message:
|
95
|
+
rdoc_options: []
|
96
|
+
require_paths:
|
97
|
+
- lib
|
98
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
99
|
+
none: false
|
100
|
+
requirements:
|
101
|
+
- - ! '>='
|
102
|
+
- !ruby/object:Gem::Version
|
103
|
+
version: '0'
|
104
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
105
|
+
none: false
|
106
|
+
requirements:
|
107
|
+
- - ! '>='
|
108
|
+
- !ruby/object:Gem::Version
|
109
|
+
version: '0'
|
110
|
+
requirements: []
|
111
|
+
rubyforge_project:
|
112
|
+
rubygems_version: 1.8.23.2
|
113
|
+
signing_key:
|
114
|
+
specification_version: 3
|
115
|
+
summary: Dynamic association fieldsets without pain.
|
116
|
+
test_files: []
|