rollable 0.0.1

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.
Files changed (73) hide show
  1. data/.gitignore +4 -0
  2. data/Gemfile +4 -0
  3. data/LICENCE +19 -0
  4. data/README.textile +67 -0
  5. data/Rakefile +2 -0
  6. data/lib/generators/rollable/setup/setup_generator.rb +55 -0
  7. data/lib/generators/rollable/setup/templates/migration.rb +15 -0
  8. data/lib/generators/rollable/setup/templates/role.rb +4 -0
  9. data/lib/rollable.rb +3 -0
  10. data/lib/rollable/base.rb +92 -0
  11. data/lib/rollable/railtie.rb +7 -0
  12. data/lib/rollable/version.rb +3 -0
  13. data/rollable.gemspec +23 -0
  14. data/spec/rollable_test/.gitignore +4 -0
  15. data/spec/rollable_test/.rspec +1 -0
  16. data/spec/rollable_test/Gemfile +32 -0
  17. data/spec/rollable_test/README +256 -0
  18. data/spec/rollable_test/Rakefile +7 -0
  19. data/spec/rollable_test/app/controllers/application_controller.rb +3 -0
  20. data/spec/rollable_test/app/helpers/application_helper.rb +2 -0
  21. data/spec/rollable_test/app/models/horse.rb +3 -0
  22. data/spec/rollable_test/app/models/pig.rb +3 -0
  23. data/spec/rollable_test/app/models/role.rb +4 -0
  24. data/spec/rollable_test/app/models/user.rb +6 -0
  25. data/spec/rollable_test/app/views/layouts/application.html.erb +14 -0
  26. data/spec/rollable_test/config.ru +4 -0
  27. data/spec/rollable_test/config/application.rb +42 -0
  28. data/spec/rollable_test/config/boot.rb +6 -0
  29. data/spec/rollable_test/config/database.yml +22 -0
  30. data/spec/rollable_test/config/environment.rb +5 -0
  31. data/spec/rollable_test/config/environments/development.rb +26 -0
  32. data/spec/rollable_test/config/environments/production.rb +49 -0
  33. data/spec/rollable_test/config/environments/test.rb +35 -0
  34. data/spec/rollable_test/config/initializers/backtrace_silencers.rb +7 -0
  35. data/spec/rollable_test/config/initializers/inflections.rb +10 -0
  36. data/spec/rollable_test/config/initializers/mime_types.rb +5 -0
  37. data/spec/rollable_test/config/initializers/secret_token.rb +7 -0
  38. data/spec/rollable_test/config/initializers/session_store.rb +8 -0
  39. data/spec/rollable_test/config/locales/en.yml +5 -0
  40. data/spec/rollable_test/config/routes.rb +58 -0
  41. data/spec/rollable_test/db/migrate/20110421101457_create_users.rb +12 -0
  42. data/spec/rollable_test/db/migrate/20110421101514_create_horses.rb +12 -0
  43. data/spec/rollable_test/db/migrate/20110421101523_create_pigs.rb +12 -0
  44. data/spec/rollable_test/db/migrate/20110423152854_create_roles.rb +15 -0
  45. data/spec/rollable_test/db/schema.rb +39 -0
  46. data/spec/rollable_test/db/seeds.rb +7 -0
  47. data/spec/rollable_test/doc/README_FOR_APP +2 -0
  48. data/spec/rollable_test/lib/tasks/.gitkeep +0 -0
  49. data/spec/rollable_test/public/404.html +26 -0
  50. data/spec/rollable_test/public/422.html +26 -0
  51. data/spec/rollable_test/public/500.html +26 -0
  52. data/spec/rollable_test/public/favicon.ico +0 -0
  53. data/spec/rollable_test/public/images/rails.png +0 -0
  54. data/spec/rollable_test/public/index.html +239 -0
  55. data/spec/rollable_test/public/javascripts/application.js +2 -0
  56. data/spec/rollable_test/public/javascripts/controls.js +965 -0
  57. data/spec/rollable_test/public/javascripts/dragdrop.js +974 -0
  58. data/spec/rollable_test/public/javascripts/effects.js +1123 -0
  59. data/spec/rollable_test/public/javascripts/prototype.js +6001 -0
  60. data/spec/rollable_test/public/javascripts/rails.js +191 -0
  61. data/spec/rollable_test/public/robots.txt +5 -0
  62. data/spec/rollable_test/public/stylesheets/.gitkeep +0 -0
  63. data/spec/rollable_test/script/rails +6 -0
  64. data/spec/rollable_test/spec/models/horse_spec.rb +5 -0
  65. data/spec/rollable_test/spec/models/pig_spec.rb +5 -0
  66. data/spec/rollable_test/spec/models/role_spec.rb +26 -0
  67. data/spec/rollable_test/spec/models/user_spec.rb +23 -0
  68. data/spec/rollable_test/spec/spec_helper.rb +27 -0
  69. data/spec/rollable_test/test/performance/browsing_test.rb +9 -0
  70. data/spec/rollable_test/test/test_helper.rb +13 -0
  71. data/spec/rollable_test/vendor/plugins/.gitkeep +0 -0
  72. data/spec/spec_helper.rb +28 -0
  73. metadata +147 -0
@@ -0,0 +1,191 @@
1
+ (function() {
2
+ // Technique from Juriy Zaytsev
3
+ // http://thinkweb2.com/projects/prototype/detecting-event-support-without-browser-sniffing/
4
+ function isEventSupported(eventName) {
5
+ var el = document.createElement('div');
6
+ eventName = 'on' + eventName;
7
+ var isSupported = (eventName in el);
8
+ if (!isSupported) {
9
+ el.setAttribute(eventName, 'return;');
10
+ isSupported = typeof el[eventName] == 'function';
11
+ }
12
+ el = null;
13
+ return isSupported;
14
+ }
15
+
16
+ function isForm(element) {
17
+ return Object.isElement(element) && element.nodeName.toUpperCase() == 'FORM'
18
+ }
19
+
20
+ function isInput(element) {
21
+ if (Object.isElement(element)) {
22
+ var name = element.nodeName.toUpperCase()
23
+ return name == 'INPUT' || name == 'SELECT' || name == 'TEXTAREA'
24
+ }
25
+ else return false
26
+ }
27
+
28
+ var submitBubbles = isEventSupported('submit'),
29
+ changeBubbles = isEventSupported('change')
30
+
31
+ if (!submitBubbles || !changeBubbles) {
32
+ // augment the Event.Handler class to observe custom events when needed
33
+ Event.Handler.prototype.initialize = Event.Handler.prototype.initialize.wrap(
34
+ function(init, element, eventName, selector, callback) {
35
+ init(element, eventName, selector, callback)
36
+ // is the handler being attached to an element that doesn't support this event?
37
+ if ( (!submitBubbles && this.eventName == 'submit' && !isForm(this.element)) ||
38
+ (!changeBubbles && this.eventName == 'change' && !isInput(this.element)) ) {
39
+ // "submit" => "emulated:submit"
40
+ this.eventName = 'emulated:' + this.eventName
41
+ }
42
+ }
43
+ )
44
+ }
45
+
46
+ if (!submitBubbles) {
47
+ // discover forms on the page by observing focus events which always bubble
48
+ document.on('focusin', 'form', function(focusEvent, form) {
49
+ // special handler for the real "submit" event (one-time operation)
50
+ if (!form.retrieve('emulated:submit')) {
51
+ form.on('submit', function(submitEvent) {
52
+ var emulated = form.fire('emulated:submit', submitEvent, true)
53
+ // if custom event received preventDefault, cancel the real one too
54
+ if (emulated.returnValue === false) submitEvent.preventDefault()
55
+ })
56
+ form.store('emulated:submit', true)
57
+ }
58
+ })
59
+ }
60
+
61
+ if (!changeBubbles) {
62
+ // discover form inputs on the page
63
+ document.on('focusin', 'input, select, texarea', function(focusEvent, input) {
64
+ // special handler for real "change" events
65
+ if (!input.retrieve('emulated:change')) {
66
+ input.on('change', function(changeEvent) {
67
+ input.fire('emulated:change', changeEvent, true)
68
+ })
69
+ input.store('emulated:change', true)
70
+ }
71
+ })
72
+ }
73
+
74
+ function handleRemote(element) {
75
+ var method, url, params;
76
+
77
+ var event = element.fire("ajax:before");
78
+ if (event.stopped) return false;
79
+
80
+ if (element.tagName.toLowerCase() === 'form') {
81
+ method = element.readAttribute('method') || 'post';
82
+ url = element.readAttribute('action');
83
+ params = element.serialize();
84
+ } else {
85
+ method = element.readAttribute('data-method') || 'get';
86
+ url = element.readAttribute('href');
87
+ params = {};
88
+ }
89
+
90
+ new Ajax.Request(url, {
91
+ method: method,
92
+ parameters: params,
93
+ evalScripts: true,
94
+
95
+ onComplete: function(request) { element.fire("ajax:complete", request); },
96
+ onSuccess: function(request) { element.fire("ajax:success", request); },
97
+ onFailure: function(request) { element.fire("ajax:failure", request); }
98
+ });
99
+
100
+ element.fire("ajax:after");
101
+ }
102
+
103
+ function handleMethod(element) {
104
+ var method = element.readAttribute('data-method'),
105
+ url = element.readAttribute('href'),
106
+ csrf_param = $$('meta[name=csrf-param]')[0],
107
+ csrf_token = $$('meta[name=csrf-token]')[0];
108
+
109
+ var form = new Element('form', { method: "POST", action: url, style: "display: none;" });
110
+ element.parentNode.insert(form);
111
+
112
+ if (method !== 'post') {
113
+ var field = new Element('input', { type: 'hidden', name: '_method', value: method });
114
+ form.insert(field);
115
+ }
116
+
117
+ if (csrf_param) {
118
+ var param = csrf_param.readAttribute('content'),
119
+ token = csrf_token.readAttribute('content'),
120
+ field = new Element('input', { type: 'hidden', name: param, value: token });
121
+ form.insert(field);
122
+ }
123
+
124
+ form.submit();
125
+ }
126
+
127
+
128
+ document.on("click", "*[data-confirm]", function(event, element) {
129
+ var message = element.readAttribute('data-confirm');
130
+ if (!confirm(message)) event.stop();
131
+ });
132
+
133
+ document.on("click", "a[data-remote]", function(event, element) {
134
+ if (event.stopped) return;
135
+ handleRemote(element);
136
+ event.stop();
137
+ });
138
+
139
+ document.on("click", "a[data-method]", function(event, element) {
140
+ if (event.stopped) return;
141
+ handleMethod(element);
142
+ event.stop();
143
+ });
144
+
145
+ document.on("submit", function(event) {
146
+ var element = event.findElement(),
147
+ message = element.readAttribute('data-confirm');
148
+ if (message && !confirm(message)) {
149
+ event.stop();
150
+ return false;
151
+ }
152
+
153
+ var inputs = element.select("input[type=submit][data-disable-with]");
154
+ inputs.each(function(input) {
155
+ input.disabled = true;
156
+ input.writeAttribute('data-original-value', input.value);
157
+ input.value = input.readAttribute('data-disable-with');
158
+ });
159
+
160
+ var element = event.findElement("form[data-remote]");
161
+ if (element) {
162
+ handleRemote(element);
163
+ event.stop();
164
+ }
165
+ });
166
+
167
+ document.on("ajax:after", "form", function(event, element) {
168
+ var inputs = element.select("input[type=submit][disabled=true][data-disable-with]");
169
+ inputs.each(function(input) {
170
+ input.value = input.readAttribute('data-original-value');
171
+ input.removeAttribute('data-original-value');
172
+ input.disabled = false;
173
+ });
174
+ });
175
+
176
+ Ajax.Responders.register({
177
+ onCreate: function(request) {
178
+ var csrf_meta_tag = $$('meta[name=csrf-token]')[0];
179
+
180
+ if (csrf_meta_tag) {
181
+ var header = 'X-CSRF-Token',
182
+ token = csrf_meta_tag.readAttribute('content');
183
+
184
+ if (!request.options.requestHeaders) {
185
+ request.options.requestHeaders = {};
186
+ }
187
+ request.options.requestHeaders[header] = token;
188
+ }
189
+ }
190
+ });
191
+ })();
@@ -0,0 +1,5 @@
1
+ # See http://www.robotstxt.org/wc/norobots.html for documentation on how to use the robots.txt file
2
+ #
3
+ # To ban all spiders from the entire site uncomment the next two lines:
4
+ # User-Agent: *
5
+ # Disallow: /
File without changes
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env ruby
2
+ # This command will automatically be run when you run "rails" with Rails 3 gems installed from the root of your application.
3
+
4
+ APP_PATH = File.expand_path('../../config/application', __FILE__)
5
+ require File.expand_path('../../config/boot', __FILE__)
6
+ require 'rails/commands'
@@ -0,0 +1,5 @@
1
+ require 'spec_helper'
2
+
3
+ describe Horse do
4
+ pending "add some examples to (or delete) #{__FILE__}"
5
+ end
@@ -0,0 +1,5 @@
1
+ require 'spec_helper'
2
+
3
+ describe Pig do
4
+ pending "add some examples to (or delete) #{__FILE__}"
5
+ end
@@ -0,0 +1,26 @@
1
+ require 'spec_helper'
2
+
3
+ describe Role do
4
+ context "Rollable" do
5
+ it "should validate inclusion of roll names" do
6
+ role = Role.new
7
+ role.should_not be_valid
8
+ role.user = User.create
9
+ role.should_not be_valid
10
+ role.rollable = Horse.create
11
+ role.should_not be_valid
12
+ role.name = User.role_names.first
13
+ role.should be_valid
14
+ end
15
+
16
+ it "should validate inclusion of rollable types" do
17
+ u = User.create
18
+ r = u.roles.create(:name => "owner" )
19
+ r.should be_valid
20
+ r.rollable = u
21
+ r.should_not be_valid
22
+ r.rollable = Horse.create
23
+ r.should be_valid
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,23 @@
1
+ require 'spec_helper'
2
+
3
+ describe User do
4
+ context "Rollable helpers" do
5
+ it "should respond to helper methods" do
6
+ @user = User.new
7
+ @user.should respond_to(:is_owner_of?)
8
+ @user.should respond_to(:is_rider_on?)
9
+ end
10
+
11
+ it "should be able to get roles" do
12
+ @user = User.create
13
+ @user.roles.create(:rollable => Horse.create, :name => User.role_names.first )
14
+ @user.roles.count.should == 1
15
+ end
16
+
17
+ it "should be able to use helpers" do
18
+ @user = User.create
19
+ @user.roles.create(:name => "owner", :rollable => Horse.create)
20
+ @user.is_owner_of?(Horse.first).should be_true
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,27 @@
1
+ # This file is copied to spec/ when you run 'rails generate rspec:install'
2
+ ENV["RAILS_ENV"] ||= 'test'
3
+ require File.expand_path("../../config/environment", __FILE__)
4
+ require 'rspec/rails'
5
+
6
+ # Requires supporting ruby files with custom matchers and macros, etc,
7
+ # in spec/support/ and its subdirectories.
8
+ Dir[Rails.root.join("spec/support/**/*.rb")].each {|f| require f}
9
+
10
+ RSpec.configure do |config|
11
+ # == Mock Framework
12
+ #
13
+ # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line:
14
+ #
15
+ # config.mock_with :mocha
16
+ # config.mock_with :flexmock
17
+ # config.mock_with :rr
18
+ config.mock_with :rspec
19
+
20
+ # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures
21
+ config.fixture_path = "#{::Rails.root}/spec/fixtures"
22
+
23
+ # If you're not using ActiveRecord, or you'd prefer not to run each of your
24
+ # examples within a transaction, remove the following line or assign false
25
+ # instead of true.
26
+ config.use_transactional_fixtures = true
27
+ end
@@ -0,0 +1,9 @@
1
+ require 'test_helper'
2
+ require 'rails/performance_test_help'
3
+
4
+ # Profiling results for each test method are written to tmp/performance.
5
+ class BrowsingTest < ActionDispatch::PerformanceTest
6
+ def test_homepage
7
+ get '/'
8
+ end
9
+ end
@@ -0,0 +1,13 @@
1
+ ENV["RAILS_ENV"] = "test"
2
+ require File.expand_path('../../config/environment', __FILE__)
3
+ require 'rails/test_help'
4
+
5
+ class ActiveSupport::TestCase
6
+ # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order.
7
+ #
8
+ # Note: You'll currently still have to declare fixtures explicitly in integration tests
9
+ # -- they do not yet inherit this setting
10
+ fixtures :all
11
+
12
+ # Add more helper methods to be used by all tests here...
13
+ end
File without changes
@@ -0,0 +1,28 @@
1
+ # This file is copied to spec/ when you run 'rails generate rspec:install'
2
+ ENV["RAILS_ENV"] ||= 'test'
3
+ require File.expand_path("../rollable_test/config/environment", __FILE__)
4
+ require 'rspec/rails'
5
+
6
+ # Requires supporting ruby files with custom matchers and macros, etc,
7
+ # in spec/support/ and its subdirectories.
8
+ #Dir[Rails.root.join("spec/support/**/*.rb")].each {|f| require f}
9
+ Dir["#{File.dirname(__FILE__)}/support/**/*.rb"].each{ |f| require f }
10
+
11
+ RSpec.configure do |config|
12
+ # == Mock Framework
13
+ #
14
+ # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line:
15
+ #
16
+ # config.mock_with :mocha
17
+ # config.mock_with :flexmock
18
+ # config.mock_with :rr
19
+ config.mock_with :rspec
20
+
21
+ # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures
22
+ config.fixture_path = "#{::Rails.root}/spec/fixtures"
23
+
24
+ # If you're not using ActiveRecord, or you'd prefer not to run each of your
25
+ # examples within a transaction, remove the following line or assign false
26
+ # instead of true.
27
+ config.use_transactional_fixtures = true
28
+ end
metadata ADDED
@@ -0,0 +1,147 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: rollable
3
+ version: !ruby/object:Gem::Version
4
+ prerelease:
5
+ version: 0.0.1
6
+ platform: ruby
7
+ authors:
8
+ - Timon Vonk
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+
13
+ date: 2011-04-23 00:00:00 Z
14
+ dependencies:
15
+ - !ruby/object:Gem::Dependency
16
+ name: rails
17
+ prerelease: false
18
+ requirement: &id001 !ruby/object:Gem::Requirement
19
+ none: false
20
+ requirements:
21
+ - - ">="
22
+ - !ruby/object:Gem::Version
23
+ version: 3.0.0
24
+ type: :runtime
25
+ version_requirements: *id001
26
+ - !ruby/object:Gem::Dependency
27
+ name: rspec
28
+ prerelease: false
29
+ requirement: &id002 !ruby/object:Gem::Requirement
30
+ none: false
31
+ requirements:
32
+ - - ">="
33
+ - !ruby/object:Gem::Version
34
+ version: "0"
35
+ type: :runtime
36
+ version_requirements: *id002
37
+ description: This gem adds agnostic roles for authorization to Rails.
38
+ email:
39
+ - mail@timonv.nl
40
+ executables: []
41
+
42
+ extensions: []
43
+
44
+ extra_rdoc_files: []
45
+
46
+ files:
47
+ - .gitignore
48
+ - Gemfile
49
+ - LICENCE
50
+ - README.textile
51
+ - Rakefile
52
+ - lib/generators/rollable/setup/setup_generator.rb
53
+ - lib/generators/rollable/setup/templates/migration.rb
54
+ - lib/generators/rollable/setup/templates/role.rb
55
+ - lib/rollable.rb
56
+ - lib/rollable/base.rb
57
+ - lib/rollable/railtie.rb
58
+ - lib/rollable/version.rb
59
+ - rollable.gemspec
60
+ - spec/rollable_test/.gitignore
61
+ - spec/rollable_test/.rspec
62
+ - spec/rollable_test/Gemfile
63
+ - spec/rollable_test/README
64
+ - spec/rollable_test/Rakefile
65
+ - spec/rollable_test/app/controllers/application_controller.rb
66
+ - spec/rollable_test/app/helpers/application_helper.rb
67
+ - spec/rollable_test/app/models/horse.rb
68
+ - spec/rollable_test/app/models/pig.rb
69
+ - spec/rollable_test/app/models/role.rb
70
+ - spec/rollable_test/app/models/user.rb
71
+ - spec/rollable_test/app/views/layouts/application.html.erb
72
+ - spec/rollable_test/config.ru
73
+ - spec/rollable_test/config/application.rb
74
+ - spec/rollable_test/config/boot.rb
75
+ - spec/rollable_test/config/database.yml
76
+ - spec/rollable_test/config/environment.rb
77
+ - spec/rollable_test/config/environments/development.rb
78
+ - spec/rollable_test/config/environments/production.rb
79
+ - spec/rollable_test/config/environments/test.rb
80
+ - spec/rollable_test/config/initializers/backtrace_silencers.rb
81
+ - spec/rollable_test/config/initializers/inflections.rb
82
+ - spec/rollable_test/config/initializers/mime_types.rb
83
+ - spec/rollable_test/config/initializers/secret_token.rb
84
+ - spec/rollable_test/config/initializers/session_store.rb
85
+ - spec/rollable_test/config/locales/en.yml
86
+ - spec/rollable_test/config/routes.rb
87
+ - spec/rollable_test/db/migrate/20110421101457_create_users.rb
88
+ - spec/rollable_test/db/migrate/20110421101514_create_horses.rb
89
+ - spec/rollable_test/db/migrate/20110421101523_create_pigs.rb
90
+ - spec/rollable_test/db/migrate/20110423152854_create_roles.rb
91
+ - spec/rollable_test/db/schema.rb
92
+ - spec/rollable_test/db/seeds.rb
93
+ - spec/rollable_test/doc/README_FOR_APP
94
+ - spec/rollable_test/lib/tasks/.gitkeep
95
+ - spec/rollable_test/public/404.html
96
+ - spec/rollable_test/public/422.html
97
+ - spec/rollable_test/public/500.html
98
+ - spec/rollable_test/public/favicon.ico
99
+ - spec/rollable_test/public/images/rails.png
100
+ - spec/rollable_test/public/index.html
101
+ - spec/rollable_test/public/javascripts/application.js
102
+ - spec/rollable_test/public/javascripts/controls.js
103
+ - spec/rollable_test/public/javascripts/dragdrop.js
104
+ - spec/rollable_test/public/javascripts/effects.js
105
+ - spec/rollable_test/public/javascripts/prototype.js
106
+ - spec/rollable_test/public/javascripts/rails.js
107
+ - spec/rollable_test/public/robots.txt
108
+ - spec/rollable_test/public/stylesheets/.gitkeep
109
+ - spec/rollable_test/script/rails
110
+ - spec/rollable_test/spec/models/horse_spec.rb
111
+ - spec/rollable_test/spec/models/pig_spec.rb
112
+ - spec/rollable_test/spec/models/role_spec.rb
113
+ - spec/rollable_test/spec/models/user_spec.rb
114
+ - spec/rollable_test/spec/spec_helper.rb
115
+ - spec/rollable_test/test/performance/browsing_test.rb
116
+ - spec/rollable_test/test/test_helper.rb
117
+ - spec/rollable_test/vendor/plugins/.gitkeep
118
+ - spec/spec_helper.rb
119
+ homepage: http://www.timonv.nl
120
+ licenses: []
121
+
122
+ post_install_message:
123
+ rdoc_options: []
124
+
125
+ require_paths:
126
+ - lib
127
+ required_ruby_version: !ruby/object:Gem::Requirement
128
+ none: false
129
+ requirements:
130
+ - - ">="
131
+ - !ruby/object:Gem::Version
132
+ version: "0"
133
+ required_rubygems_version: !ruby/object:Gem::Requirement
134
+ none: false
135
+ requirements:
136
+ - - ">="
137
+ - !ruby/object:Gem::Version
138
+ version: "0"
139
+ requirements: []
140
+
141
+ rubyforge_project: rollable
142
+ rubygems_version: 1.7.2
143
+ signing_key:
144
+ specification_version: 3
145
+ summary: Agnostic roles for rails
146
+ test_files: []
147
+