fat_free_crm 0.11.0
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of fat_free_crm might be problematic. Click here for more details.
- data/.gitignore +32 -0
- data/.travis.yml +48 -0
- data/CHANGELOG +1839 -0
- data/CONTRIBUTORS +50 -0
- data/Gemfile +40 -0
- data/Gemfile.ci +13 -0
- data/Gemfile.lock +256 -0
- data/Guardfile +45 -0
- data/LICENSE +620 -0
- data/Procfile +1 -0
- data/README.md +239 -0
- data/Rakefile +9 -0
- data/acceptance/acceptance_helper.rb +7 -0
- data/acceptance/accounts_spec.rb +74 -0
- data/acceptance/support/browser.rb +13 -0
- data/acceptance/support/database_cleaner.rb +16 -0
- data/acceptance/support/headless.rb +15 -0
- data/acceptance/support/helpers.rb +19 -0
- data/acceptance/support/maintain_sessions.rb +5 -0
- data/acceptance/support/paths.rb +9 -0
- data/app/assets/images/1x1.gif +0 -0
- data/app/assets/images/asterisk.gif +0 -0
- data/app/assets/images/avatar.jpg +0 -0
- data/app/assets/images/blog.gif +0 -0
- data/app/assets/images/facebook-close.gif +0 -0
- data/app/assets/images/facebook.gif +0 -0
- data/app/assets/images/iconset_attribution.png +0 -0
- data/app/assets/images/info.png +0 -0
- data/app/assets/images/info_tiny.png +0 -0
- data/app/assets/images/linkedin.gif +0 -0
- data/app/assets/images/loading.gif +0 -0
- data/app/assets/images/skype.gif +0 -0
- data/app/assets/images/sortable.gif +0 -0
- data/app/assets/images/stars.gif +0 -0
- data/app/assets/images/twitter.gif +0 -0
- data/app/assets/javascripts/admin/field_groups.js.coffee +3 -0
- data/app/assets/javascripts/application.js.erb +42 -0
- data/app/assets/javascripts/crm.js +487 -0
- data/app/assets/javascripts/crm_chosen.js.coffee +28 -0
- data/app/assets/javascripts/crm_classes.js +230 -0
- data/app/assets/javascripts/crm_fields.js +47 -0
- data/app/assets/javascripts/crm_loginout.js +41 -0
- data/app/assets/javascripts/jquery-noconflict.js +1 -0
- data/app/assets/javascripts/lists.js.coffee +37 -0
- data/app/assets/javascripts/search.js.coffee +40 -0
- data/app/assets/stylesheets/admin/field_groups.css.scss +3 -0
- data/app/assets/stylesheets/application.css.erb +37 -0
- data/app/assets/stylesheets/base.scss +90 -0
- data/app/assets/stylesheets/common.scss +688 -0
- data/app/assets/stylesheets/ffcrm_chosen.scss +64 -0
- data/app/assets/stylesheets/fields.scss +8 -0
- data/app/assets/stylesheets/header.scss +129 -0
- data/app/assets/stylesheets/lists.css.scss +3 -0
- data/app/assets/stylesheets/print.css.scss +80 -0
- data/app/assets/stylesheets/rails.scss +136 -0
- data/app/assets/stylesheets/safari.scss +23 -0
- data/app/controllers/accounts_controller.rb +207 -0
- data/app/controllers/admin/application_controller.rb +42 -0
- data/app/controllers/admin/field_groups_controller.rb +128 -0
- data/app/controllers/admin/fields_controller.rb +148 -0
- data/app/controllers/admin/plugins_controller.rb +33 -0
- data/app/controllers/admin/settings_controller.rb +31 -0
- data/app/controllers/admin/tags_controller.rb +131 -0
- data/app/controllers/admin/users_controller.rb +200 -0
- data/app/controllers/application_controller.rb +219 -0
- data/app/controllers/authentications_controller.rb +66 -0
- data/app/controllers/base_controller.rb +155 -0
- data/app/controllers/campaigns_controller.rb +205 -0
- data/app/controllers/comments_controller.rb +145 -0
- data/app/controllers/contacts_controller.rb +214 -0
- data/app/controllers/emails_controller.rb +64 -0
- data/app/controllers/home_controller.rb +170 -0
- data/app/controllers/leads_controller.rb +286 -0
- data/app/controllers/lists_controller.rb +27 -0
- data/app/controllers/opportunities_controller.rb +256 -0
- data/app/controllers/passwords_controller.rb +78 -0
- data/app/controllers/tasks_controller.rb +213 -0
- data/app/controllers/users_controller.rb +163 -0
- data/app/helpers/accounts_helper.rb +54 -0
- data/app/helpers/addresses_helper.rb +28 -0
- data/app/helpers/admin/application_helper.rb +29 -0
- data/app/helpers/admin/field_groups_helper.rb +36 -0
- data/app/helpers/admin/fields_helper.rb +32 -0
- data/app/helpers/admin/plugins_helper.rb +20 -0
- data/app/helpers/admin/settings_helper.rb +20 -0
- data/app/helpers/admin/tags_helper.rb +22 -0
- data/app/helpers/admin/users_helper.rb +65 -0
- data/app/helpers/application_helper.rb +421 -0
- data/app/helpers/authentications_helper.rb +20 -0
- data/app/helpers/campaigns_helper.rb +55 -0
- data/app/helpers/comments_helper.rb +20 -0
- data/app/helpers/contacts_helper.rb +34 -0
- data/app/helpers/crm_tags_helper.rb +50 -0
- data/app/helpers/emails_helper.rb +19 -0
- data/app/helpers/fields_helper.rb +20 -0
- data/app/helpers/home_helper.rb +70 -0
- data/app/helpers/leads_helper.rb +116 -0
- data/app/helpers/lists_helper.rb +2 -0
- data/app/helpers/opportunities_helper.rb +53 -0
- data/app/helpers/passwords_helper.rb +20 -0
- data/app/helpers/tasks_helper.rb +176 -0
- data/app/helpers/users_helper.rb +40 -0
- data/app/inputs/date_time_input.rb +39 -0
- data/app/inputs/text_input.rb +6 -0
- data/app/models/base/account.rb +138 -0
- data/app/models/base/account_contact.rb +37 -0
- data/app/models/base/account_opportunity.rb +37 -0
- data/app/models/base/campaign.rb +123 -0
- data/app/models/base/contact.rb +191 -0
- data/app/models/base/contact_opportunity.rb +38 -0
- data/app/models/base/lead.rb +188 -0
- data/app/models/base/opportunity.rb +188 -0
- data/app/models/base/task.rb +241 -0
- data/app/models/fields/core_field.rb +48 -0
- data/app/models/fields/custom_field.rb +138 -0
- data/app/models/fields/field.rb +124 -0
- data/app/models/fields/field_group.rb +81 -0
- data/app/models/list.rb +8 -0
- data/app/models/mailers/notifier.rb +44 -0
- data/app/models/observers/activity_observer.rb +84 -0
- data/app/models/polymorphic/activity.rb +106 -0
- data/app/models/polymorphic/address.rb +58 -0
- data/app/models/polymorphic/avatar.rb +58 -0
- data/app/models/polymorphic/comment.rb +55 -0
- data/app/models/polymorphic/email.rb +69 -0
- data/app/models/polymorphic/tag.rb +14 -0
- data/app/models/polymorphic/tagging.rb +2 -0
- data/app/models/setting.rb +142 -0
- data/app/models/users/authentication.rb +64 -0
- data/app/models/users/permission.rb +36 -0
- data/app/models/users/preference.rb +62 -0
- data/app/models/users/user.rb +171 -0
- data/app/views/accounts/_account.html.haml +35 -0
- data/app/views/accounts/_contact_info.html.haml +37 -0
- data/app/views/accounts/_edit.html.haml +19 -0
- data/app/views/accounts/_new.html.haml +19 -0
- data/app/views/accounts/_options.html.haml +20 -0
- data/app/views/accounts/_permissions.html.haml +24 -0
- data/app/views/accounts/_sidebar_index.html.haml +21 -0
- data/app/views/accounts/_sidebar_show.html.haml +57 -0
- data/app/views/accounts/_top_section.html.haml +29 -0
- data/app/views/accounts/contacts.js.rjs +3 -0
- data/app/views/accounts/create.js.rjs +15 -0
- data/app/views/accounts/destroy.js.rjs +3 -0
- data/app/views/accounts/edit.js.rjs +35 -0
- data/app/views/accounts/index.html.haml +25 -0
- data/app/views/accounts/index.js.rjs +6 -0
- data/app/views/accounts/new.js.rjs +11 -0
- data/app/views/accounts/opportunities.js.rjs +3 -0
- data/app/views/accounts/options.js.rjs +10 -0
- data/app/views/accounts/show.html.haml +42 -0
- data/app/views/accounts/update.js.rjs +17 -0
- data/app/views/admin/field_groups/_confirm.html.haml +5 -0
- data/app/views/admin/field_groups/_edit.html.haml +8 -0
- data/app/views/admin/field_groups/_field_group.html.haml +27 -0
- data/app/views/admin/field_groups/_new.html.haml +10 -0
- data/app/views/admin/field_groups/_top_section.html.haml +13 -0
- data/app/views/admin/field_groups/confirm.js.rjs +6 -0
- data/app/views/admin/field_groups/create.js.rjs +21 -0
- data/app/views/admin/field_groups/destroy.js.rjs +11 -0
- data/app/views/admin/field_groups/edit.js.rjs +14 -0
- data/app/views/admin/field_groups/new.js.rjs +9 -0
- data/app/views/admin/field_groups/update.js.rjs +9 -0
- data/app/views/admin/fields/_edit.html.haml +9 -0
- data/app/views/admin/fields/_field.html.haml +15 -0
- data/app/views/admin/fields/_new.html.haml +10 -0
- data/app/views/admin/fields/_options.html.haml +12 -0
- data/app/views/admin/fields/_sidebar_index.html.haml +1 -0
- data/app/views/admin/fields/_sidebar_show.html.haml +1 -0
- data/app/views/admin/fields/_sort_by.html.haml +10 -0
- data/app/views/admin/fields/_top_section.html.haml +54 -0
- data/app/views/admin/fields/create.js.rjs +16 -0
- data/app/views/admin/fields/destroy.js.rjs +11 -0
- data/app/views/admin/fields/edit.js.rjs +35 -0
- data/app/views/admin/fields/index.html.haml +36 -0
- data/app/views/admin/fields/index.js.rjs +6 -0
- data/app/views/admin/fields/new.js.rjs +8 -0
- data/app/views/admin/fields/options.js.rjs +10 -0
- data/app/views/admin/fields/show.html.haml +41 -0
- data/app/views/admin/fields/update.js.rjs +17 -0
- data/app/views/admin/plugins/_plugin.html.haml +13 -0
- data/app/views/admin/plugins/index.html.haml +9 -0
- data/app/views/admin/settings/index.html.haml +3 -0
- data/app/views/admin/tags/_confirm.html.haml +5 -0
- data/app/views/admin/tags/_edit.html.haml +9 -0
- data/app/views/admin/tags/_new.html.haml +10 -0
- data/app/views/admin/tags/_tag.html.haml +29 -0
- data/app/views/admin/tags/_top_section.html.haml +6 -0
- data/app/views/admin/tags/confirm.js.rjs +6 -0
- data/app/views/admin/tags/create.js.rjs +12 -0
- data/app/views/admin/tags/destroy.js.rjs +11 -0
- data/app/views/admin/tags/edit.js.rjs +21 -0
- data/app/views/admin/tags/index.html.haml +17 -0
- data/app/views/admin/tags/new.js.rjs +8 -0
- data/app/views/admin/tags/update.js.rjs +10 -0
- data/app/views/admin/users/_confirm.html.haml +5 -0
- data/app/views/admin/users/_edit.html.haml +11 -0
- data/app/views/admin/users/_new.html.haml +11 -0
- data/app/views/admin/users/_profile.html.haml +43 -0
- data/app/views/admin/users/_sidebar_index.html.haml +2 -0
- data/app/views/admin/users/_user.html.haml +56 -0
- data/app/views/admin/users/confirm.js.rjs +6 -0
- data/app/views/admin/users/create.js.rjs +11 -0
- data/app/views/admin/users/destroy.js.rjs +11 -0
- data/app/views/admin/users/edit.js.rjs +21 -0
- data/app/views/admin/users/index.html.haml +13 -0
- data/app/views/admin/users/index.js.rjs +2 -0
- data/app/views/admin/users/new.js.rjs +8 -0
- data/app/views/admin/users/reactivate.js.rjs +3 -0
- data/app/views/admin/users/show.html.haml +5 -0
- data/app/views/admin/users/suspend.js.rjs +3 -0
- data/app/views/admin/users/update.js.rjs +10 -0
- data/app/views/authentications/new.html.haml +31 -0
- data/app/views/base/_advanced_search.html.haml +12 -0
- data/app/views/base/_condition_fields.html.haml +14 -0
- data/app/views/base/_grouping_fields.html.haml +12 -0
- data/app/views/base/_sort_fields.html.haml +3 -0
- data/app/views/base/advanced_search.js.rjs +9 -0
- data/app/views/campaigns/_campaign.html.haml +17 -0
- data/app/views/campaigns/_edit.html.haml +16 -0
- data/app/views/campaigns/_metrics.html.haml +16 -0
- data/app/views/campaigns/_new.html.haml +15 -0
- data/app/views/campaigns/_objectives.html.haml +30 -0
- data/app/views/campaigns/_options.html.haml +20 -0
- data/app/views/campaigns/_permissions.html.haml +24 -0
- data/app/views/campaigns/_sidebar_index.html.haml +20 -0
- data/app/views/campaigns/_sidebar_show.html.haml +91 -0
- data/app/views/campaigns/_status.html.haml +24 -0
- data/app/views/campaigns/_top_section.html.haml +29 -0
- data/app/views/campaigns/create.js.rjs +21 -0
- data/app/views/campaigns/destroy.js.rjs +3 -0
- data/app/views/campaigns/edit.js.rjs +38 -0
- data/app/views/campaigns/index.html.haml +25 -0
- data/app/views/campaigns/index.js.rjs +6 -0
- data/app/views/campaigns/leads.js.rjs +3 -0
- data/app/views/campaigns/new.js.rjs +13 -0
- data/app/views/campaigns/opportunities.js.rjs +3 -0
- data/app/views/campaigns/options.js.rjs +10 -0
- data/app/views/campaigns/show.html.haml +44 -0
- data/app/views/campaigns/update.js.rjs +23 -0
- data/app/views/comments/_comment.html.haml +25 -0
- data/app/views/comments/_edit.html.haml +12 -0
- data/app/views/comments/_new.html.haml +27 -0
- data/app/views/comments/create.js.rjs +11 -0
- data/app/views/comments/destroy.js.rjs +6 -0
- data/app/views/comments/edit.js.rjs +11 -0
- data/app/views/comments/new.js.rjs +12 -0
- data/app/views/comments/update.js.rjs +11 -0
- data/app/views/contacts/_contact.html.haml +34 -0
- data/app/views/contacts/_edit.html.haml +17 -0
- data/app/views/contacts/_extra.html.haml +36 -0
- data/app/views/contacts/_new.html.haml +17 -0
- data/app/views/contacts/_options.html.haml +21 -0
- data/app/views/contacts/_permissions.html.haml +24 -0
- data/app/views/contacts/_sidebar_index.html.haml +3 -0
- data/app/views/contacts/_sidebar_show.html.haml +41 -0
- data/app/views/contacts/_top_section.html.haml +54 -0
- data/app/views/contacts/_web.html.haml +31 -0
- data/app/views/contacts/create.js.rjs +20 -0
- data/app/views/contacts/destroy.js.rjs +7 -0
- data/app/views/contacts/edit.js.rjs +38 -0
- data/app/views/contacts/index.html.haml +26 -0
- data/app/views/contacts/index.js.rjs +6 -0
- data/app/views/contacts/new.js.rjs +13 -0
- data/app/views/contacts/opportunities.js.rjs +3 -0
- data/app/views/contacts/options.js.rjs +10 -0
- data/app/views/contacts/show.html.haml +38 -0
- data/app/views/contacts/update.js.rjs +22 -0
- data/app/views/emails/_email.html.haml +27 -0
- data/app/views/emails/destroy.js.rjs +6 -0
- data/app/views/fields/_group.html.haml +20 -0
- data/app/views/fields/_groups.html.haml +11 -0
- data/app/views/fields/_sidebar_show.html.haml +8 -0
- data/app/views/fields/group.js.rjs +7 -0
- data/app/views/home/_actions_menu.html.haml +8 -0
- data/app/views/home/_activity.html.haml +22 -0
- data/app/views/home/_assets_menu.html.haml +7 -0
- data/app/views/home/_duration_menu.html.haml +8 -0
- data/app/views/home/_options.html.haml +14 -0
- data/app/views/home/_users_menu.html.haml +8 -0
- data/app/views/home/index.atom.builder +24 -0
- data/app/views/home/index.html.haml +19 -0
- data/app/views/home/index.js.rjs +6 -0
- data/app/views/home/index.rss.builder +20 -0
- data/app/views/home/options.js.rjs +8 -0
- data/app/views/layouts/500.html.haml +29 -0
- data/app/views/layouts/_about.html.haml +17 -0
- data/app/views/layouts/_footer.html.haml +13 -0
- data/app/views/layouts/_header.html.haml +20 -0
- data/app/views/layouts/_jumpbox.html.haml +32 -0
- data/app/views/layouts/_sidebar.html.haml +14 -0
- data/app/views/layouts/_tabbed.html.haml +19 -0
- data/app/views/layouts/_tabless.html.haml +3 -0
- data/app/views/layouts/admin/_header.html.haml +14 -0
- data/app/views/layouts/admin/application.html.haml +25 -0
- data/app/views/layouts/application.html.haml +43 -0
- data/app/views/leads/_contact.html.haml +38 -0
- data/app/views/leads/_convert.html.haml +36 -0
- data/app/views/leads/_convert_permissions.html.haml +26 -0
- data/app/views/leads/_edit.html.haml +21 -0
- data/app/views/leads/_lead.html.haml +43 -0
- data/app/views/leads/_new.html.haml +21 -0
- data/app/views/leads/_opportunity.html.haml +37 -0
- data/app/views/leads/_options.html.haml +21 -0
- data/app/views/leads/_permissions.html.haml +28 -0
- data/app/views/leads/_sidebar_index.html.haml +20 -0
- data/app/views/leads/_sidebar_show.html.haml +58 -0
- data/app/views/leads/_status.html.haml +36 -0
- data/app/views/leads/_top_section.html.haml +29 -0
- data/app/views/leads/_web.html.haml +31 -0
- data/app/views/leads/convert.js.rjs +38 -0
- data/app/views/leads/create.js.rjs +20 -0
- data/app/views/leads/destroy.js.rjs +7 -0
- data/app/views/leads/edit.js.rjs +38 -0
- data/app/views/leads/index.html.haml +25 -0
- data/app/views/leads/index.js.rjs +6 -0
- data/app/views/leads/new.js.rjs +11 -0
- data/app/views/leads/options.js.rjs +10 -0
- data/app/views/leads/promote.js.rjs +28 -0
- data/app/views/leads/reject.js.rjs +10 -0
- data/app/views/leads/show.html.haml +35 -0
- data/app/views/leads/update.js.rjs +27 -0
- data/app/views/lists/create.js.rjs +9 -0
- data/app/views/lists/destroy.js.rjs +1 -0
- data/app/views/notifier/dropbox_ack_notification.html.haml +9 -0
- data/app/views/notifier/password_reset_instructions.html.haml +6 -0
- data/app/views/opportunities/_edit.html.haml +16 -0
- data/app/views/opportunities/_new.html.haml +16 -0
- data/app/views/opportunities/_opportunity.html.haml +45 -0
- data/app/views/opportunities/_options.html.haml +20 -0
- data/app/views/opportunities/_permissions.html.haml +24 -0
- data/app/views/opportunities/_sidebar_index.html.haml +20 -0
- data/app/views/opportunities/_sidebar_show.html.haml +61 -0
- data/app/views/opportunities/_top_section.html.haml +65 -0
- data/app/views/opportunities/contacts.js.rjs +3 -0
- data/app/views/opportunities/create.js.rjs +31 -0
- data/app/views/opportunities/destroy.js.rjs +11 -0
- data/app/views/opportunities/edit.js.rjs +39 -0
- data/app/views/opportunities/index.html.haml +28 -0
- data/app/views/opportunities/index.js.rjs +6 -0
- data/app/views/opportunities/new.js.rjs +16 -0
- data/app/views/opportunities/options.js.rjs +10 -0
- data/app/views/opportunities/show.html.haml +36 -0
- data/app/views/opportunities/update.js.rjs +27 -0
- data/app/views/passwords/edit.html.haml +15 -0
- data/app/views/passwords/new.html.haml +10 -0
- data/app/views/shared/_address.html.haml +44 -0
- data/app/views/shared/_address_show.html.haml +24 -0
- data/app/views/shared/_comment.html.haml +11 -0
- data/app/views/shared/_edit_comment.html.haml +13 -0
- data/app/views/shared/_empty.html.haml +8 -0
- data/app/views/shared/_export.html.haml +1 -0
- data/app/views/shared/_inline_styles.html.haml +31 -0
- data/app/views/shared/_lists.html.haml +25 -0
- data/app/views/shared/_naming.html.haml +9 -0
- data/app/views/shared/_outline.html.haml +9 -0
- data/app/views/shared/_paginate.haml +1 -0
- data/app/views/shared/_per_page.html.haml +10 -0
- data/app/views/shared/_recent.html.haml +4 -0
- data/app/views/shared/_recently.html.haml +7 -0
- data/app/views/shared/_search.html.haml +11 -0
- data/app/views/shared/_select_popup.html.haml +21 -0
- data/app/views/shared/_sort_by.html.haml +10 -0
- data/app/views/shared/_tags.html.haml +9 -0
- data/app/views/shared/_tasks.html.haml +10 -0
- data/app/views/shared/_timeline.html.haml +3 -0
- data/app/views/shared/_total.html.haml +4 -0
- data/app/views/shared/attach.js.rjs +24 -0
- data/app/views/shared/auto_complete.html.haml +10 -0
- data/app/views/shared/discard.rjs +7 -0
- data/app/views/shared/index.atom.builder +34 -0
- data/app/views/shared/index.rss.builder +31 -0
- data/app/views/tasks/_assigned.html.haml +47 -0
- data/app/views/tasks/_completed.html.haml +23 -0
- data/app/views/tasks/_edit.html.haml +19 -0
- data/app/views/tasks/_empty.html.haml +4 -0
- data/app/views/tasks/_new.html.haml +14 -0
- data/app/views/tasks/_pending.html.haml +52 -0
- data/app/views/tasks/_related.html.haml +33 -0
- data/app/views/tasks/_selector.html.haml +8 -0
- data/app/views/tasks/_sidebar_index.html.haml +18 -0
- data/app/views/tasks/_title.html.haml +8 -0
- data/app/views/tasks/_top_section.html.haml +32 -0
- data/app/views/tasks/complete.js.rjs +17 -0
- data/app/views/tasks/create.js.rjs +42 -0
- data/app/views/tasks/destroy.js.rjs +7 -0
- data/app/views/tasks/discard.rjs +1 -0
- data/app/views/tasks/edit.js.rjs +25 -0
- data/app/views/tasks/filter.js.rjs +1 -0
- data/app/views/tasks/index.html.haml +20 -0
- data/app/views/tasks/new.js.rjs +11 -0
- data/app/views/tasks/update.js.rjs +22 -0
- data/app/views/users/_avatar.html.haml +20 -0
- data/app/views/users/_languages.html.haml +9 -0
- data/app/views/users/_password.html.haml +27 -0
- data/app/views/users/_profile.html.haml +67 -0
- data/app/views/users/_user.html.haml +28 -0
- data/app/views/users/avatar.js.rjs +10 -0
- data/app/views/users/change_password.js.rjs +15 -0
- data/app/views/users/edit.js.rjs +10 -0
- data/app/views/users/index.html.haml +3 -0
- data/app/views/users/new.html.haml +19 -0
- data/app/views/users/password.js.rjs +11 -0
- data/app/views/users/show.html.haml +16 -0
- data/app/views/users/update.js.rjs +10 -0
- data/app/views/users/upload_avatar.js.rjs +8 -0
- data/app/views/versions/_version.html.haml +30 -0
- data/config.ru +4 -0
- data/config/application.rb +75 -0
- data/config/boot.rb +6 -0
- data/config/database.mysql.mac.yml +31 -0
- data/config/database.mysql.yml +32 -0
- data/config/database.postgres.yml +25 -0
- data/config/database.sqlite.yml +22 -0
- data/config/environment.rb +5 -0
- data/config/environments/development.rb +39 -0
- data/config/environments/production.rb +67 -0
- data/config/environments/staging.rb +14 -0
- data/config/environments/test.rb +48 -0
- data/config/initializers/action_mailer.rb +3 -0
- data/config/initializers/authlogic.rb +1 -0
- data/config/initializers/mime_types.rb +8 -0
- data/config/initializers/paginate_arrays.rb +7 -0
- data/config/initializers/paper_trail.rb +1 -0
- data/config/initializers/sass.rb +13 -0
- data/config/initializers/secret_token.rb +13 -0
- data/config/initializers/simple_form.rb +96 -0
- data/config/locales/cz.yml +213 -0
- data/config/locales/cz_fat_free_crm.yml +695 -0
- data/config/locales/de.yml +193 -0
- data/config/locales/de_fat_free_crm.yml +637 -0
- data/config/locales/en-GB.yml +180 -0
- data/config/locales/en-GB_fat_free_crm.yml +670 -0
- data/config/locales/en-US.yml +182 -0
- data/config/locales/en-US_fat_free_crm.yml +832 -0
- data/config/locales/es.yml +184 -0
- data/config/locales/es_fat_free_crm.yml +642 -0
- data/config/locales/fr-CA.yml +187 -0
- data/config/locales/fr-CA_fat_free_crm.yml +635 -0
- data/config/locales/fr.yml +182 -0
- data/config/locales/fr_fat_free_crm.yml +633 -0
- data/config/locales/it.yml +183 -0
- data/config/locales/it_fat_free_crm.yml +665 -0
- data/config/locales/ja.yml +188 -0
- data/config/locales/ja_fat_free_crm.yml +636 -0
- data/config/locales/pl.yml +194 -0
- data/config/locales/pl_fat_free_crm.yml +646 -0
- data/config/locales/pt-BR.yml +199 -0
- data/config/locales/pt-BR_fat_free_crm.yml +639 -0
- data/config/locales/ru.yml +303 -0
- data/config/locales/ru_fat_free_crm.yml +661 -0
- data/config/locales/simple_form.en.yml +24 -0
- data/config/locales/sv-SE.yml +232 -0
- data/config/locales/sv-SE_fat_free_crm.yml +638 -0
- data/config/locales/th.rb +200 -0
- data/config/locales/th_fat_free_crm.yml +643 -0
- data/config/locales/zh-CN.yml +210 -0
- data/config/locales/zh-CN_fat_free_crm.yml +631 -0
- data/config/routes.rb +172 -0
- data/config/settings.default.yml +325 -0
- data/db/demo/account_contacts.yml +24 -0
- data/db/demo/account_opportunities.yml +24 -0
- data/db/demo/accounts.yml +48 -0
- data/db/demo/activities.yml +16 -0
- data/db/demo/addresses.yml +46 -0
- data/db/demo/avatars.yml +16 -0
- data/db/demo/campaigns.yml +144 -0
- data/db/demo/comments.yml +30 -0
- data/db/demo/contact_opportunities.yml +25 -0
- data/db/demo/contacts.yml +70 -0
- data/db/demo/emails.yml +53 -0
- data/db/demo/leads.yml +71 -0
- data/db/demo/opportunities.yml +48 -0
- data/db/demo/permissions.yml +13 -0
- data/db/demo/preferences.yml +13 -0
- data/db/demo/settings.yml +12 -0
- data/db/demo/tasks.yml +64 -0
- data/db/demo/users.yml +249 -0
- data/db/migrate/20100928030598_create_sessions.rb +17 -0
- data/db/migrate/20100928030599_create_users.rb +46 -0
- data/db/migrate/20100928030600_create_openid_tables.rb +24 -0
- data/db/migrate/20100928030601_create_accounts.rb +27 -0
- data/db/migrate/20100928030602_create_permissions.rb +16 -0
- data/db/migrate/20100928030603_create_settings.rb +16 -0
- data/db/migrate/20100928030604_create_preferences.rb +16 -0
- data/db/migrate/20100928030605_create_campaigns.rb +35 -0
- data/db/migrate/20100928030606_create_leads.rb +39 -0
- data/db/migrate/20100928030607_create_contacts.rb +39 -0
- data/db/migrate/20100928030608_create_opportunities.rb +28 -0
- data/db/migrate/20100928030609_create_account_contacts.rb +15 -0
- data/db/migrate/20100928030610_create_account_opportunities.rb +15 -0
- data/db/migrate/20100928030611_create_contact_opportunities.rb +16 -0
- data/db/migrate/20100928030612_create_tasks.rb +27 -0
- data/db/migrate/20100928030613_create_comments.rb +17 -0
- data/db/migrate/20100928030614_create_activities.rb +20 -0
- data/db/migrate/20100928030615_create_avatars.rb +17 -0
- data/db/migrate/20100928030616_rename_remember_token.rb +12 -0
- data/db/migrate/20100928030617_drop_openid_tables.rb +23 -0
- data/db/migrate/20100928030618_add_admin_to_users.rb +12 -0
- data/db/migrate/20100928030619_add_suspended_to_users.rb +12 -0
- data/db/migrate/20100928030620_remove_uuid.rb +24 -0
- data/db/migrate/20100928030621_add_email_to_accounts.rb +10 -0
- data/db/migrate/20100928030622_add_background_info_to_models.rb +20 -0
- data/db/migrate/20100928030623_create_addresses.rb +51 -0
- data/db/migrate/20100928030624_add_index_on_permissions.rb +10 -0
- data/db/migrate/20100928030625_create_emails.rb +27 -0
- data/db/migrate/20100928030626_add_state_to_timeline_objects.rb +14 -0
- data/db/migrate/20100928030627_acts_as_taggable_on_migration.rb +30 -0
- data/db/migrate/20101221123456_add_single_access_token_to_users.rb +10 -0
- data/db/migrate/20101221345678_add_rating_and_category_to_accounts.rb +12 -0
- data/db/migrate/20110719082054_add_skype_to_contacts_and_leads.rb +12 -0
- data/db/migrate/20111101083437_create_fields.rb +28 -0
- data/db/migrate/20111101090312_create_field_groups.rb +16 -0
- data/db/migrate/20111116091952_add_field_groups_tag_id.rb +10 -0
- data/db/migrate/20111117041311_change_fields_collection_to_text.rb +10 -0
- data/db/migrate/20111201030535_add_field_groups_klass_name.rb +21 -0
- data/db/migrate/20120121054235_create_lists.rb +10 -0
- data/db/migrate/20120216031616_create_versions.rb +18 -0
- data/db/migrate/20120216042541_is_paranoid_to_paper_trail.rb +13 -0
- data/db/migrate/20120220233724_add_versions_object_changes.rb +9 -0
- data/db/migrate/20120224073107_remove_default_value_and_clear_settings.rb +16 -0
- data/db/schema.rb +415 -0
- data/db/seeds.rb +10 -0
- data/db/seeds/fields.rb +94 -0
- data/fat_free_crm.gemspec +45 -0
- data/lib/country_select/MIT-LICENSE +20 -0
- data/lib/country_select/README +14 -0
- data/lib/country_select/init.rb +1 -0
- data/lib/country_select/install.rb +2 -0
- data/lib/country_select/lib/country_select.rb +551 -0
- data/lib/country_select/uninstall.rb +2 -0
- data/lib/development_tasks/gem.rake +8 -0
- data/lib/development_tasks/license.rake +99 -0
- data/lib/development_tasks/rdoc.rake +15 -0
- data/lib/development_tasks/rspec.rake +23 -0
- data/lib/dynamic_form/MIT-LICENSE +20 -0
- data/lib/dynamic_form/README +13 -0
- data/lib/dynamic_form/Rakefile +10 -0
- data/lib/dynamic_form/dynamic_form.gemspec +12 -0
- data/lib/dynamic_form/init.rb +2 -0
- data/lib/dynamic_form/lib/action_view/helpers/dynamic_form.rb +301 -0
- data/lib/dynamic_form/lib/action_view/locale/en-US.yml +8 -0
- data/lib/dynamic_form/lib/dynamic_form.rb +6 -0
- data/lib/dynamic_form/test/dynamic_form_i18n_test.rb +42 -0
- data/lib/dynamic_form/test/dynamic_form_test.rb +370 -0
- data/lib/dynamic_form/test/test_helper.rb +10 -0
- data/lib/fat_free_crm.rb +52 -0
- data/lib/fat_free_crm/callback.rb +144 -0
- data/lib/fat_free_crm/core_ext.rb +18 -0
- data/lib/fat_free_crm/core_ext/array.rb +95 -0
- data/lib/fat_free_crm/core_ext/nil.rb +29 -0
- data/lib/fat_free_crm/core_ext/string.rb +74 -0
- data/lib/fat_free_crm/dropbox.rb +429 -0
- data/lib/fat_free_crm/engine.rb +9 -0
- data/lib/fat_free_crm/errors.rb +51 -0
- data/lib/fat_free_crm/exceptions.rb +34 -0
- data/lib/fat_free_crm/exportable.rb +63 -0
- data/lib/fat_free_crm/fields.rb +76 -0
- data/lib/fat_free_crm/gem_dependencies.rb +16 -0
- data/lib/fat_free_crm/gem_ext.rb +8 -0
- data/lib/fat_free_crm/gem_ext/action_controller/base.rb +28 -0
- data/lib/fat_free_crm/gem_ext/active_model/serializers/xml/serializer/attribute.rb +29 -0
- data/lib/fat_free_crm/gem_ext/active_record/schema_dumper.rb +22 -0
- data/lib/fat_free_crm/gem_ext/active_support/buffered_logger.rb +32 -0
- data/lib/fat_free_crm/gem_ext/authlogic/session/cookies.rb +11 -0
- data/lib/fat_free_crm/gem_ext/rails/engine.rb +17 -0
- data/lib/fat_free_crm/gem_ext/rails/text_helper.rb +125 -0
- data/lib/fat_free_crm/gem_ext/rake/task.rb +8 -0
- data/lib/fat_free_crm/gem_ext/simple_form/action_view_extensions/form_helper.rb +20 -0
- data/lib/fat_free_crm/i18n.rb +51 -0
- data/lib/fat_free_crm/permissions.rb +111 -0
- data/lib/fat_free_crm/plugin.rb +106 -0
- data/lib/fat_free_crm/plugin_dependencies.rb +6 -0
- data/lib/fat_free_crm/renderers.rb +21 -0
- data/lib/fat_free_crm/sortable.rb +59 -0
- data/lib/fat_free_crm/syck_yaml.rb +4 -0
- data/lib/fat_free_crm/tabs.rb +35 -0
- data/lib/fat_free_crm/version.rb +27 -0
- data/lib/gravatar_image_tag/Gemfile +8 -0
- data/lib/gravatar_image_tag/README.textile +108 -0
- data/lib/gravatar_image_tag/ROADMAP.textile +33 -0
- data/lib/gravatar_image_tag/Rakefile +50 -0
- data/lib/gravatar_image_tag/VERSION +1 -0
- data/lib/gravatar_image_tag/gravatar_image_tag.gemspec +45 -0
- data/lib/gravatar_image_tag/init.rb +2 -0
- data/lib/gravatar_image_tag/lib/gravatar_image_tag.rb +63 -0
- data/lib/gravatar_image_tag/spec/gravatar_image_tag_spec.rb +83 -0
- data/lib/gravatar_image_tag/spec/test_helper.rb +12 -0
- data/lib/responds_to_parent/MIT-LICENSE +20 -0
- data/lib/responds_to_parent/README +47 -0
- data/lib/responds_to_parent/Rakefile +22 -0
- data/lib/responds_to_parent/init.rb +2 -0
- data/lib/responds_to_parent/install.rb +2 -0
- data/lib/responds_to_parent/lib/responds_to_parent.rb +70 -0
- data/lib/responds_to_parent/test/responds_to_parent_test.rb +11 -0
- data/lib/responds_to_parent/test/test_helper.rb +7 -0
- data/lib/responds_to_parent/uninstall.rb +2 -0
- data/lib/tasks/.gitkeep +0 -0
- data/lib/tasks/.gitkeep~master +0 -0
- data/lib/tasks/demo.rake +72 -0
- data/lib/tasks/dropbox.rake +32 -0
- data/lib/tasks/fat_free_crm.rake +130 -0
- data/lib/tasks/plugins.rake +77 -0
- data/lib/tasks/schema_upgrade.rake +31 -0
- data/lib/templates/erb/scaffold/_form.html.erb +13 -0
- data/public/404.html +26 -0
- data/public/422.html +26 -0
- data/public/500.html +26 -0
- data/public/blank_iframe.html +2 -0
- data/public/favicon.ico +0 -0
- data/public/robots.txt +5 -0
- data/script/rails +6 -0
- data/spec/controllers/accounts_controller_spec.rb +626 -0
- data/spec/controllers/admin/users_controller_spec.rb +309 -0
- data/spec/controllers/authentications_controller_spec.rb +147 -0
- data/spec/controllers/campaigns_controller_spec.rb +666 -0
- data/spec/controllers/comments_controller_spec.rb +274 -0
- data/spec/controllers/contacts_controller_spec.rb +699 -0
- data/spec/controllers/emails_controller_spec.rb +32 -0
- data/spec/controllers/home_controller_spec.rb +99 -0
- data/spec/controllers/leads_controller_spec.rb +1047 -0
- data/spec/controllers/lists_controller_spec.rb +5 -0
- data/spec/controllers/opportunities_controller_spec.rb +932 -0
- data/spec/controllers/passwords_controller_spec.rb +11 -0
- data/spec/controllers/tasks_controller_spec.rb +508 -0
- data/spec/controllers/users_controller_spec.rb +344 -0
- data/spec/factories/account_factories.rb +35 -0
- data/spec/factories/campaign_factories.rb +23 -0
- data/spec/factories/contact_factories.rb +39 -0
- data/spec/factories/field_factories.rb +34 -0
- data/spec/factories/lead_factories.rb +29 -0
- data/spec/factories/list_factories.rb +6 -0
- data/spec/factories/opportunity_factories.rb +20 -0
- data/spec/factories/sequences.rb +26 -0
- data/spec/factories/setting_factories.rb +8 -0
- data/spec/factories/shared_factories.rb +70 -0
- data/spec/factories/tag_factories.rb +5 -0
- data/spec/factories/task_factories.rb +18 -0
- data/spec/factories/user_factories.rb +57 -0
- data/spec/fixtures/rails.png +0 -0
- data/spec/helpers/accounts_helper_spec.rb +12 -0
- data/spec/helpers/admin/field_groups_helper_spec.rb +13 -0
- data/spec/helpers/admin/plugins_helper_spec.rb +5 -0
- data/spec/helpers/admin/settings_helper_spec.rb +5 -0
- data/spec/helpers/admin/users_helper_spec.rb +5 -0
- data/spec/helpers/application_helper_spec.rb +67 -0
- data/spec/helpers/authentications_helper_spec.rb +12 -0
- data/spec/helpers/campaigns_helper_spec.rb +12 -0
- data/spec/helpers/comments_helper_spec.rb +12 -0
- data/spec/helpers/contacts_helper_spec.rb +12 -0
- data/spec/helpers/emails_helper_spec.rb +5 -0
- data/spec/helpers/fields_helper_spec.rb +6 -0
- data/spec/helpers/home_helper_spec.rb +12 -0
- data/spec/helpers/leads_helper_spec.rb +12 -0
- data/spec/helpers/lists_helper_spec.rb +15 -0
- data/spec/helpers/opportunities_helper_spec.rb +12 -0
- data/spec/helpers/passwords_helper_spec.rb +12 -0
- data/spec/helpers/tasks_helper_spec.rb +12 -0
- data/spec/helpers/users_helper_spec.rb +12 -0
- data/spec/lib/core_ext/string.rb +17 -0
- data/spec/lib/dropbox/email_samples.rb +77 -0
- data/spec/lib/dropbox_spec.rb +410 -0
- data/spec/lib/errors_spec.rb +25 -0
- data/spec/models/base/account_contact_spec.rb +27 -0
- data/spec/models/base/account_opportunity_spec.rb +27 -0
- data/spec/models/base/account_spec.rb +136 -0
- data/spec/models/base/campaign_spec.rb +135 -0
- data/spec/models/base/contact_opportunity_spec.rb +28 -0
- data/spec/models/base/contact_spec.rb +160 -0
- data/spec/models/base/lead_spec.rb +101 -0
- data/spec/models/base/opportunity_spec.rb +173 -0
- data/spec/models/base/task_spec.rb +263 -0
- data/spec/models/fields/custom_field_spec.rb +106 -0
- data/spec/models/fields/field_group_spec.rb +23 -0
- data/spec/models/fields/field_spec.rb +64 -0
- data/spec/models/list_spec.rb +12 -0
- data/spec/models/polymorphic/activity_spec.rb +303 -0
- data/spec/models/polymorphic/address_spec.rb +33 -0
- data/spec/models/polymorphic/avatar_spec.rb +41 -0
- data/spec/models/polymorphic/comment_spec.rb +29 -0
- data/spec/models/polymorphic/email_spec.rb +36 -0
- data/spec/models/setting_spec.rb +51 -0
- data/spec/models/users/authentication_spec.rb +13 -0
- data/spec/models/users/permission_spec.rb +27 -0
- data/spec/models/users/preference_spec.rb +60 -0
- data/spec/models/users/user_spec.rb +169 -0
- data/spec/routing/accounts_routing_spec.rb +55 -0
- data/spec/routing/admin/users_routing_spec.rb +35 -0
- data/spec/routing/campaigns_routing_spec.rb +58 -0
- data/spec/routing/comments_routing_spec.rb +35 -0
- data/spec/routing/contacts_routing_spec.rb +55 -0
- data/spec/routing/emails_routing_spec.rb +35 -0
- data/spec/routing/leads_routing_spec.rb +83 -0
- data/spec/routing/opportunities_routing_spec.rb +59 -0
- data/spec/routing/tasks_routing_spec.rb +62 -0
- data/spec/routing/users_routing_spec.rb +79 -0
- data/spec/shared/controllers.rb +102 -0
- data/spec/shared/models.rb +37 -0
- data/spec/spec_helper.rb +110 -0
- data/spec/support/assert_select.rb +158 -0
- data/spec/support/auth_macros.rb +44 -0
- data/spec/support/macros.rb +40 -0
- data/spec/support/rjs_support.rb +9 -0
- data/spec/views/accounts/_edit.haml_spec.rb +38 -0
- data/spec/views/accounts/_new.haml_spec.rb +37 -0
- data/spec/views/accounts/create.rjs_spec.rb +54 -0
- data/spec/views/accounts/destroy.rjs_spec.rb +30 -0
- data/spec/views/accounts/edit.rjs_spec.rb +69 -0
- data/spec/views/accounts/index.haml_spec.rb +33 -0
- data/spec/views/accounts/index.rjs_spec.rb +32 -0
- data/spec/views/accounts/new.rjs_spec.rb +47 -0
- data/spec/views/accounts/options.rjs_spec.rb +59 -0
- data/spec/views/accounts/show.haml_spec.rb +30 -0
- data/spec/views/accounts/update.rjs_spec.rb +99 -0
- data/spec/views/admin/users/_create.haml_spec.rb +17 -0
- data/spec/views/admin/users/create.rjs_spec.rb +42 -0
- data/spec/views/admin/users/destroy.rjs_spec.rb +47 -0
- data/spec/views/admin/users/edit.rjs_spec.rb +45 -0
- data/spec/views/admin/users/index.haml_spec.rb +16 -0
- data/spec/views/admin/users/index.rjs_spec.rb +23 -0
- data/spec/views/admin/users/new.rjs_spec.rb +31 -0
- data/spec/views/admin/users/reactivate.rjs_spec.rb +17 -0
- data/spec/views/admin/users/show.haml_spec.rb +12 -0
- data/spec/views/admin/users/suspend.rjs_spec.rb +17 -0
- data/spec/views/admin/users/update.rjs_spec.rb +36 -0
- data/spec/views/authentications/new.haml_spec.rb +25 -0
- data/spec/views/campaigns/_edit.haml_spec.rb +37 -0
- data/spec/views/campaigns/_new.haml_spec.rb +35 -0
- data/spec/views/campaigns/create.rjs_spec.rb +52 -0
- data/spec/views/campaigns/destroy.rjs_spec.rb +28 -0
- data/spec/views/campaigns/edit.rjs_spec.rb +78 -0
- data/spec/views/campaigns/index.haml_spec.rb +28 -0
- data/spec/views/campaigns/index.rjs_spec.rb +32 -0
- data/spec/views/campaigns/new.rjs_spec.rb +57 -0
- data/spec/views/campaigns/options.rjs_spec.rb +60 -0
- data/spec/views/campaigns/show.haml_spec.rb +31 -0
- data/spec/views/campaigns/update.rjs_spec.rb +87 -0
- data/spec/views/comments/new.rjs_spec.rb +21 -0
- data/spec/views/common/auto_complete.haml_spec.rb +43 -0
- data/spec/views/contacts/_edit.haml_spec.rb +67 -0
- data/spec/views/contacts/_new.haml_spec.rb +47 -0
- data/spec/views/contacts/create.rjs_spec.rb +69 -0
- data/spec/views/contacts/destroy.rjs_spec.rb +41 -0
- data/spec/views/contacts/edit.rjs_spec.rb +77 -0
- data/spec/views/contacts/index.haml_spec.rb +28 -0
- data/spec/views/contacts/index.rjs_spec.rb +32 -0
- data/spec/views/contacts/new.rjs_spec.rb +51 -0
- data/spec/views/contacts/options.rjs_spec.rb +61 -0
- data/spec/views/contacts/show.haml_spec.rb +27 -0
- data/spec/views/contacts/update.rjs_spec.rb +141 -0
- data/spec/views/home/index.haml_spec.rb +26 -0
- data/spec/views/home/index.rjs_spec.rb +30 -0
- data/spec/views/home/options.rjs_spec.rb +52 -0
- data/spec/views/leads/_convert.haml_spec.rb +25 -0
- data/spec/views/leads/_edit.haml_spec.rb +41 -0
- data/spec/views/leads/_new.haml_spec.rb +39 -0
- data/spec/views/leads/_sidebar_show.haml_spec.rb +25 -0
- data/spec/views/leads/convert.rjs_spec.rb +83 -0
- data/spec/views/leads/create.rjs_spec.rb +70 -0
- data/spec/views/leads/destroy.rjs_spec.rb +46 -0
- data/spec/views/leads/edit.rjs_spec.rb +79 -0
- data/spec/views/leads/index.haml_spec.rb +28 -0
- data/spec/views/leads/index.rjs_spec.rb +32 -0
- data/spec/views/leads/new.rjs_spec.rb +52 -0
- data/spec/views/leads/options.rjs_spec.rb +61 -0
- data/spec/views/leads/promote.rjs_spec.rb +154 -0
- data/spec/views/leads/reject.rjs_spec.rb +49 -0
- data/spec/views/leads/show.haml_spec.rb +24 -0
- data/spec/views/leads/update.rjs_spec.rb +134 -0
- data/spec/views/opportunities/_edit.haml_spec.rb +64 -0
- data/spec/views/opportunities/_new.haml_spec.rb +46 -0
- data/spec/views/opportunities/create.rjs_spec.rb +94 -0
- data/spec/views/opportunities/destroy.rjs_spec.rb +64 -0
- data/spec/views/opportunities/edit.rjs_spec.rb +79 -0
- data/spec/views/opportunities/index.haml_spec.rb +29 -0
- data/spec/views/opportunities/index.rjs_spec.rb +34 -0
- data/spec/views/opportunities/new.rjs_spec.rb +60 -0
- data/spec/views/opportunities/options.rjs_spec.rb +60 -0
- data/spec/views/opportunities/show.haml_spec.rb +27 -0
- data/spec/views/opportunities/update.rjs_spec.rb +162 -0
- data/spec/views/tasks/_edit.haml_spec.rb +45 -0
- data/spec/views/tasks/complete.rjs_spec.rb +68 -0
- data/spec/views/tasks/create.rjs_spec.rb +124 -0
- data/spec/views/tasks/destroy.rjs_spec.rb +53 -0
- data/spec/views/tasks/edit.rjs_spec.rb +78 -0
- data/spec/views/tasks/index.haml_spec.rb +40 -0
- data/spec/views/tasks/new.rjs_spec.rb +62 -0
- data/spec/views/tasks/update.rjs_spec.rb +152 -0
- data/spec/views/users/avatar.rjs_spec.rb +32 -0
- data/spec/views/users/change_password.rjs_spec.rb +49 -0
- data/spec/views/users/edit.rjs_spec.rb +32 -0
- data/spec/views/users/password.rjs_spec.rb +32 -0
- data/spec/views/users/update.rjs_spec.rb +50 -0
- data/spec/views/users/upload_avatar.rjs_spec.rb +42 -0
- data/vendor/assets/images/calendar_date_select/calendar.gif +0 -0
- data/vendor/assets/images/delete.png +0 -0
- data/vendor/assets/images/tab_icons/accounts.png +0 -0
- data/vendor/assets/images/tab_icons/accounts_active.png +0 -0
- data/vendor/assets/images/tab_icons/campaigns.png +0 -0
- data/vendor/assets/images/tab_icons/campaigns_active.png +0 -0
- data/vendor/assets/images/tab_icons/contacts.png +0 -0
- data/vendor/assets/images/tab_icons/contacts_active.png +0 -0
- data/vendor/assets/images/tab_icons/dashboard.png +0 -0
- data/vendor/assets/images/tab_icons/dashboard_active.png +0 -0
- data/vendor/assets/images/tab_icons/leads.png +0 -0
- data/vendor/assets/images/tab_icons/leads_active.png +0 -0
- data/vendor/assets/images/tab_icons/opportunities.png +0 -0
- data/vendor/assets/images/tab_icons/opportunities_active.png +0 -0
- data/vendor/assets/images/tab_icons/tasks.png +0 -0
- data/vendor/assets/images/tab_icons/tasks_active.png +0 -0
- data/vendor/assets/javascripts/calendar_date_select/calendar_date_select.js +446 -0
- data/vendor/assets/javascripts/calendar_date_select/format_american.js +35 -0
- data/vendor/assets/javascripts/calendar_date_select/format_danish.js +31 -0
- data/vendor/assets/javascripts/calendar_date_select/format_db.js +27 -0
- data/vendor/assets/javascripts/calendar_date_select/format_euro_24hr.js +7 -0
- data/vendor/assets/javascripts/calendar_date_select/format_euro_24hr_ymd.js +7 -0
- data/vendor/assets/javascripts/calendar_date_select/format_finnish.js +32 -0
- data/vendor/assets/javascripts/calendar_date_select/format_hyphen_ampm.js +37 -0
- data/vendor/assets/javascripts/calendar_date_select/format_iso_date.js +29 -0
- data/vendor/assets/javascripts/calendar_date_select/format_italian.js +24 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/ar.js +10 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/da.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/de.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/es.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/fi.js +10 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/fr.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/it.js +9 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/ja.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/nl.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/pl.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/pt.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/ru.js +10 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/sl.js +11 -0
- data/vendor/assets/javascripts/calendar_date_select/locale/sv.js +9 -0
- data/vendor/assets/javascripts/event.simulate.js +64 -0
- data/vendor/assets/javascripts/facebooklist.js +548 -0
- data/vendor/assets/javascripts/facebooklist.simulate.js +28 -0
- data/vendor/assets/javascripts/modalbox.js +506 -0
- data/vendor/assets/javascripts/rating.js +162 -0
- data/vendor/assets/stylesheets/calendar_date_select/blue.css +130 -0
- data/vendor/assets/stylesheets/calendar_date_select/default.css +135 -0
- data/vendor/assets/stylesheets/calendar_date_select/green.css +142 -0
- data/vendor/assets/stylesheets/calendar_date_select/plain.css +128 -0
- data/vendor/assets/stylesheets/calendar_date_select/red.css +135 -0
- data/vendor/assets/stylesheets/calendar_date_select/silver.css +133 -0
- data/vendor/assets/stylesheets/facebooklist.css +47 -0
- data/vendor/assets/stylesheets/modalbox.css +107 -0
- data/vendor/plugins/.gitkeep +0 -0
- data/vendor/plugins/.gitkeep~master +0 -0
- metadata +1196 -0
@@ -0,0 +1,932 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
describe OpportunitiesController do
|
4
|
+
|
5
|
+
def get_data_for_sidebar
|
6
|
+
@stage = Setting.unroll(:opportunity_stage)
|
7
|
+
end
|
8
|
+
|
9
|
+
before do
|
10
|
+
require_user
|
11
|
+
set_current_tab(:opportunities)
|
12
|
+
end
|
13
|
+
|
14
|
+
# GET /opportunities
|
15
|
+
# GET /opportunities.xml
|
16
|
+
#----------------------------------------------------------------------------
|
17
|
+
describe "responding to GET index" do
|
18
|
+
|
19
|
+
before do
|
20
|
+
get_data_for_sidebar
|
21
|
+
end
|
22
|
+
|
23
|
+
it "should expose all opportunities as @opportunities and render [index] template" do
|
24
|
+
@opportunities = [ FactoryGirl.create(:opportunity, :user => @current_user) ]
|
25
|
+
|
26
|
+
get :index
|
27
|
+
assigns[:opportunities].should == @opportunities
|
28
|
+
response.should render_template("opportunities/index")
|
29
|
+
end
|
30
|
+
|
31
|
+
it "should expose the data for the opportunities sidebar" do
|
32
|
+
get :index
|
33
|
+
assigns[:stage].should == @stage
|
34
|
+
(assigns[:opportunity_stage_total].keys.map(&:to_sym) - (@stage.map(&:last) << :all << :other)).should == []
|
35
|
+
end
|
36
|
+
|
37
|
+
it "should filter out opportunities by stage" do
|
38
|
+
controller.session[:filter_by_opportunity_stage] = "prospecting,negotiation"
|
39
|
+
@opportunities = [
|
40
|
+
FactoryGirl.create(:opportunity, :user => @current_user, :stage => "negotiation"),
|
41
|
+
FactoryGirl.create(:opportunity, :user => @current_user, :stage => "prospecting")
|
42
|
+
]
|
43
|
+
# This one should be filtered out.
|
44
|
+
FactoryGirl.create(:opportunity, :user => @current_user, :stage => "analysis")
|
45
|
+
|
46
|
+
get :index
|
47
|
+
# Note: can't compare opportunities directly because of BigDecimal objects.
|
48
|
+
assigns[:opportunities].size.should == 2
|
49
|
+
assigns[:opportunities].map(&:stage).sort.should == %w(negotiation prospecting)
|
50
|
+
end
|
51
|
+
|
52
|
+
it "should perform lookup using query string" do
|
53
|
+
@first = FactoryGirl.create(:opportunity, :user => @current_user, :name => "The first one")
|
54
|
+
@second = FactoryGirl.create(:opportunity, :user => @current_user, :name => "The second one")
|
55
|
+
|
56
|
+
get :index, :query => "second"
|
57
|
+
assigns[:opportunities].should == [ @second ]
|
58
|
+
assigns[:current_query].should == "second"
|
59
|
+
session[:opportunities_current_query].should == "second"
|
60
|
+
end
|
61
|
+
|
62
|
+
describe "AJAX pagination" do
|
63
|
+
it "should pick up page number from params" do
|
64
|
+
@opportunities = [ FactoryGirl.create(:opportunity, :user => @current_user) ]
|
65
|
+
xhr :get, :index, :page => 42
|
66
|
+
|
67
|
+
assigns[:current_page].to_i.should == 42
|
68
|
+
assigns[:opportunities].should == [] # page #42 should be empty if there's only one opportunity ;-)
|
69
|
+
session[:opportunities_current_page].to_i.should == 42
|
70
|
+
response.should render_template("opportunities/index")
|
71
|
+
end
|
72
|
+
|
73
|
+
it "should pick up saved page number from session" do
|
74
|
+
session[:opportunities_current_page] = 42
|
75
|
+
@opportunities = [ FactoryGirl.create(:opportunity, :user => @current_user) ]
|
76
|
+
xhr :get, :index
|
77
|
+
|
78
|
+
assigns[:current_page].should == 42
|
79
|
+
assigns[:opportunities].should == []
|
80
|
+
response.should render_template("opportunities/index")
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
describe "with mime type of JSON" do
|
85
|
+
it "should render all opportunities as JSON" do
|
86
|
+
@controller.should_receive(:get_list_of_records).and_return(opportunities = mock("Array of Opportunities"))
|
87
|
+
opportunities.should_receive(:to_json).and_return("generated JSON")
|
88
|
+
|
89
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
90
|
+
get :index
|
91
|
+
response.body.should == "generated JSON"
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
describe "with mime type of JSON" do
|
96
|
+
it "should render all opportunities as JSON" do
|
97
|
+
@controller.should_receive(:get_list_of_records).and_return(opportunities = mock("Array of Opportunities"))
|
98
|
+
opportunities.should_receive(:to_json).and_return("generated JSON")
|
99
|
+
|
100
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
101
|
+
get :index
|
102
|
+
response.body.should == "generated JSON"
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
describe "with mime type of XML" do
|
107
|
+
it "should render all opportunities as xml" do
|
108
|
+
@controller.should_receive(:get_list_of_records).and_return(opportunities = mock("Array of Opportunities"))
|
109
|
+
opportunities.should_receive(:to_xml).and_return("generated XML")
|
110
|
+
|
111
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
112
|
+
get :index
|
113
|
+
response.body.should == "generated XML"
|
114
|
+
end
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
# GET /opportunities/1
|
119
|
+
# GET /opportunities/1.xml HTML
|
120
|
+
#----------------------------------------------------------------------------
|
121
|
+
describe "responding to GET show" do
|
122
|
+
|
123
|
+
describe "with mime type of HTML" do
|
124
|
+
before do
|
125
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
126
|
+
@stage = Setting.unroll(:opportunity_stage)
|
127
|
+
@comment = Comment.new
|
128
|
+
end
|
129
|
+
|
130
|
+
it "should expose the requested opportunity as @opportunity and render [show] template" do
|
131
|
+
get :show, :id => 42
|
132
|
+
assigns[:opportunity].should == @opportunity
|
133
|
+
assigns[:stage].should == @stage
|
134
|
+
assigns[:comment].attributes.should == @comment.attributes
|
135
|
+
response.should render_template("opportunities/show")
|
136
|
+
end
|
137
|
+
|
138
|
+
it "should update an activity when viewing the opportunity" do
|
139
|
+
Activity.should_receive(:log).with(@current_user, @opportunity, :viewed).once
|
140
|
+
get :show, :id => @opportunity.id
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
describe "with mime type of JSON" do
|
145
|
+
it "should render the requested opportunity as JSON" do
|
146
|
+
Opportunity.stub_chain(:my, :find).and_return(opportunity = mock_model(Opportunity, :name => ''))
|
147
|
+
opportunity.should_receive(:to_json).and_return("generated JSON")
|
148
|
+
|
149
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
150
|
+
get :show, :id => 42
|
151
|
+
response.body.should == "generated JSON"
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
describe "with mime type of XML" do
|
156
|
+
it "should render the requested opportunity as xml" do
|
157
|
+
Opportunity.stub_chain(:my, :find).and_return(opportunity = mock_model(Opportunity, :name => ''))
|
158
|
+
opportunity.should_receive(:to_xml).and_return("generated XML")
|
159
|
+
|
160
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
161
|
+
get :show, :id => 42
|
162
|
+
response.body.should == "generated XML"
|
163
|
+
end
|
164
|
+
end
|
165
|
+
|
166
|
+
describe "opportunity got deleted or otherwise unavailable" do
|
167
|
+
it "should redirect to opportunity index if the opportunity got deleted" do
|
168
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
169
|
+
@opportunity.destroy
|
170
|
+
|
171
|
+
get :show, :id => @opportunity.id
|
172
|
+
flash[:warning].should_not == nil
|
173
|
+
response.should redirect_to(opportunities_path)
|
174
|
+
end
|
175
|
+
|
176
|
+
it "should redirect to opportunity index if the opportunity is protected" do
|
177
|
+
@private = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user), :access => "Private")
|
178
|
+
|
179
|
+
get :show, :id => @private.id
|
180
|
+
flash[:warning].should_not == nil
|
181
|
+
response.should redirect_to(opportunities_path)
|
182
|
+
end
|
183
|
+
|
184
|
+
it "should return 404 (Not Found) JSON error" do
|
185
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
186
|
+
@opportunity.destroy
|
187
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
188
|
+
|
189
|
+
get :show, :id => @opportunity.id
|
190
|
+
response.code.should == "404" # :not_found
|
191
|
+
end
|
192
|
+
|
193
|
+
it "should return 404 (Not Found) XML error" do
|
194
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
195
|
+
@opportunity.destroy
|
196
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
197
|
+
|
198
|
+
get :show, :id => @opportunity.id
|
199
|
+
response.code.should == "404" # :not_found
|
200
|
+
end
|
201
|
+
end
|
202
|
+
|
203
|
+
end
|
204
|
+
|
205
|
+
# GET /opportunities/new
|
206
|
+
# GET /opportunities/new.xml AJAX
|
207
|
+
#----------------------------------------------------------------------------
|
208
|
+
describe "responding to GET new" do
|
209
|
+
|
210
|
+
it "should expose a new opportunity as @opportunity and render [new] template" do
|
211
|
+
@opportunity = Opportunity.new(:user => @current_user, :access => Setting.default_access, :stage => "prospecting")
|
212
|
+
@account = Account.new(:user => @current_user, :access => Setting.default_access)
|
213
|
+
@users = [ FactoryGirl.create(:user) ]
|
214
|
+
@accounts = [ FactoryGirl.create(:account, :user => @current_user) ]
|
215
|
+
|
216
|
+
xhr :get, :new
|
217
|
+
assigns[:opportunity].attributes.should == @opportunity.attributes
|
218
|
+
assigns[:account].attributes.should == @account.attributes
|
219
|
+
assigns[:users].should == @users
|
220
|
+
assigns[:accounts].should == @accounts
|
221
|
+
response.should render_template("opportunities/new")
|
222
|
+
end
|
223
|
+
|
224
|
+
it "should created an instance of related object when necessary" do
|
225
|
+
@contact = FactoryGirl.create(:contact, :id => 42)
|
226
|
+
|
227
|
+
xhr :get, :new, :related => "contact_42"
|
228
|
+
assigns[:contact].should == @contact
|
229
|
+
end
|
230
|
+
|
231
|
+
describe "(when creating related opportunity)" do
|
232
|
+
it "should redirect to parent asset's index page with the message if parent asset got deleted" do
|
233
|
+
@account = FactoryGirl.create(:account)
|
234
|
+
@account.destroy
|
235
|
+
|
236
|
+
xhr :get, :new, :related => "account_#{@account.id}"
|
237
|
+
flash[:warning].should_not == nil
|
238
|
+
response.body.should == 'window.location.href = "/accounts";'
|
239
|
+
end
|
240
|
+
|
241
|
+
it "should redirect to parent asset's index page with the message if parent asset got protected" do
|
242
|
+
@account = FactoryGirl.create(:account, :access => "Private")
|
243
|
+
|
244
|
+
xhr :get, :new, :related => "account_#{@account.id}"
|
245
|
+
flash[:warning].should_not == nil
|
246
|
+
response.body.should == 'window.location.href = "/accounts";'
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
250
|
+
end
|
251
|
+
|
252
|
+
# GET /opportunities/1/edit AJAX
|
253
|
+
#----------------------------------------------------------------------------
|
254
|
+
describe "responding to GET edit" do
|
255
|
+
|
256
|
+
it "should expose the requested opportunity as @opportunity and render [edit] template" do
|
257
|
+
# Note: campaign => nil makes sure campaign factory is not invoked which has a side
|
258
|
+
# effect of creating an extra (campaign) user.
|
259
|
+
@account = FactoryGirl.create(:account, :user => @current_user)
|
260
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :user => @current_user, :campaign => nil,
|
261
|
+
:account => @account)
|
262
|
+
@users = [ FactoryGirl.create(:user) ]
|
263
|
+
@stage = Setting.unroll(:opportunity_stage)
|
264
|
+
@accounts = [ @account ]
|
265
|
+
|
266
|
+
xhr :get, :edit, :id => 42
|
267
|
+
@opportunity.reload
|
268
|
+
assigns[:opportunity].should == @opportunity
|
269
|
+
assigns[:account].attributes.should == @opportunity.account.attributes
|
270
|
+
assigns[:accounts].should == @accounts
|
271
|
+
assigns[:users].should == @users
|
272
|
+
assigns[:stage].should == @stage
|
273
|
+
assigns[:previous].should == nil
|
274
|
+
response.should render_template("opportunities/edit")
|
275
|
+
end
|
276
|
+
|
277
|
+
it "should expose previous opportunity as @previous when necessary" do
|
278
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
279
|
+
@previous = FactoryGirl.create(:opportunity, :id => 41)
|
280
|
+
|
281
|
+
xhr :get, :edit, :id => 42, :previous => 41
|
282
|
+
assigns[:previous].should == @previous
|
283
|
+
end
|
284
|
+
|
285
|
+
describe "opportunity got deleted or is otherwise unavailable" do
|
286
|
+
it "should reload current page with the flash message if the opportunity got deleted" do
|
287
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
288
|
+
@opportunity.destroy
|
289
|
+
|
290
|
+
xhr :get, :edit, :id => @opportunity.id
|
291
|
+
flash[:warning].should_not == nil
|
292
|
+
response.body.should == "window.location.reload();"
|
293
|
+
end
|
294
|
+
|
295
|
+
it "should reload current page with the flash message if the opportunity is protected" do
|
296
|
+
@private = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user), :access => "Private")
|
297
|
+
|
298
|
+
xhr :get, :edit, :id => @private.id
|
299
|
+
flash[:warning].should_not == nil
|
300
|
+
response.body.should == "window.location.reload();"
|
301
|
+
end
|
302
|
+
end
|
303
|
+
|
304
|
+
describe "(previous opportunity got deleted or is otherwise unavailable)" do
|
305
|
+
before do
|
306
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
307
|
+
@previous = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user))
|
308
|
+
end
|
309
|
+
|
310
|
+
it "should notify the view if previous opportunity got deleted" do
|
311
|
+
@previous.destroy
|
312
|
+
|
313
|
+
xhr :get, :edit, :id => @opportunity.id, :previous => @previous.id
|
314
|
+
flash[:warning].should == nil # no warning, just silently remove the div
|
315
|
+
assigns[:previous].should == @previous.id
|
316
|
+
response.should render_template("opportunities/edit")
|
317
|
+
end
|
318
|
+
|
319
|
+
it "should notify the view if previous opportunity got protected" do
|
320
|
+
@previous.update_attribute(:access, "Private")
|
321
|
+
|
322
|
+
xhr :get, :edit, :id => @opportunity.id, :previous => @previous.id
|
323
|
+
flash[:warning].should == nil
|
324
|
+
assigns[:previous].should == @previous.id
|
325
|
+
response.should render_template("opportunities/edit")
|
326
|
+
end
|
327
|
+
end
|
328
|
+
end
|
329
|
+
|
330
|
+
# POST /opportunities
|
331
|
+
# POST /opportunities.xml AJAX
|
332
|
+
#----------------------------------------------------------------------------
|
333
|
+
describe "responding to POST create" do
|
334
|
+
|
335
|
+
describe "with valid params" do
|
336
|
+
|
337
|
+
before do
|
338
|
+
@opportunity = FactoryGirl.build(:opportunity, :user => @current_user)
|
339
|
+
Opportunity.stub!(:new).and_return(@opportunity)
|
340
|
+
@stage = Setting.unroll(:opportunity_stage)
|
341
|
+
end
|
342
|
+
|
343
|
+
it "should expose a newly created opportunity as @opportunity and render [create] template" do
|
344
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :account => { :name => "Hello again" }, :users => %w(1 2 3)
|
345
|
+
assigns(:opportunity).should == @opportunity
|
346
|
+
assigns(:stage).should == @stage
|
347
|
+
assigns(:opportunity_stage_total).should be_nil
|
348
|
+
response.should render_template("opportunities/create")
|
349
|
+
end
|
350
|
+
|
351
|
+
it "should get sidebar data if called from opportunities index" do
|
352
|
+
request.env["HTTP_REFERER"] = "http://localhost/opportunities"
|
353
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :account => { :name => "Hello again" }, :users => %w(1 2 3)
|
354
|
+
assigns(:opportunity_stage_total).should be_an_instance_of(HashWithIndifferentAccess)
|
355
|
+
end
|
356
|
+
|
357
|
+
it "should find related account if called from account landing page" do
|
358
|
+
@account = FactoryGirl.create(:account, :user => @current_user)
|
359
|
+
request.env["HTTP_REFERER"] = "http://localhost/accounts/#{@account.id}"
|
360
|
+
|
361
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :account => { :id => @account.id }, :users => %w(1 2 3)
|
362
|
+
assigns(:account).should == @account
|
363
|
+
end
|
364
|
+
|
365
|
+
it "should find related campaign if called from campaign landing page" do
|
366
|
+
@campaign = FactoryGirl.create(:campaign, :user => @current_user)
|
367
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
368
|
+
|
369
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :campaign => @campaign.id, :account => { :name => "Hello again" }, :users => %w(1 2 3)
|
370
|
+
assigns(:campaign).should == @campaign
|
371
|
+
end
|
372
|
+
|
373
|
+
it "should reload opportunities to update pagination if called from opportunities index" do
|
374
|
+
request.env["HTTP_REFERER"] = "http://localhost/opportunities"
|
375
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :account => { :name => "Hello again" }, :users => %w(1 2 3)
|
376
|
+
assigns[:opportunities].should == [ @opportunity ]
|
377
|
+
end
|
378
|
+
|
379
|
+
it "should associate opportunity with the campaign when called from campaign landing page" do
|
380
|
+
@campaign = FactoryGirl.create(:campaign)
|
381
|
+
|
382
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
383
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :campaign => @campaign.id, :account => { :name => "Test Account" }, :users => []
|
384
|
+
assigns(:opportunity).should == @opportunity
|
385
|
+
assigns(:campaign).should == @campaign
|
386
|
+
@opportunity.campaign.should == @campaign
|
387
|
+
end
|
388
|
+
|
389
|
+
it "should associate opportunity with the contact when called from contact landing page" do
|
390
|
+
@contact = FactoryGirl.create(:contact, :id => 42)
|
391
|
+
|
392
|
+
request.env["HTTP_REFERER"] = "http://localhost/contacts/42"
|
393
|
+
xhr :post, :create, :opportunity => { :name => "Hello" }, :contact => 42, :account => { :name => "Hello again" }, :users => []
|
394
|
+
assigns(:opportunity).should == @opportunity
|
395
|
+
@opportunity.contacts.should include(@contact)
|
396
|
+
@contact.opportunities.should include(@opportunity)
|
397
|
+
end
|
398
|
+
|
399
|
+
it "should create new account and associate it with the opportunity" do
|
400
|
+
xhr :put, :create, :opportunity => { :name => "Hello" }, :account => { :name => "new account" }
|
401
|
+
assigns(:opportunity).should == @opportunity
|
402
|
+
@opportunity.account.name.should == "new account"
|
403
|
+
end
|
404
|
+
|
405
|
+
it "should associate opportunity with the existing account" do
|
406
|
+
@account = FactoryGirl.create(:account, :id => 42)
|
407
|
+
|
408
|
+
xhr :post, :create, :opportunity => { :name => "Hello world" }, :account => { :id => 42 }, :users => []
|
409
|
+
assigns(:opportunity).should == @opportunity
|
410
|
+
@opportunity.account.should == @account
|
411
|
+
@account.opportunities.should include(@opportunity)
|
412
|
+
end
|
413
|
+
|
414
|
+
it "should update related campaign revenue if won" do
|
415
|
+
@campaign = FactoryGirl.create(:campaign, :revenue => 0)
|
416
|
+
@opportunity = FactoryGirl.build(:opportunity, :user => @current_user, :stage => "won", :amount => 1100, :discount => 100)
|
417
|
+
Opportunity.stub!(:new).and_return(@opportunity)
|
418
|
+
|
419
|
+
xhr :post, :create, :opportunity => { :name => "Hello world" }, :campaign => @campaign.id, :account => { :name => "Test Account" }
|
420
|
+
assigns(:opportunity).should == @opportunity
|
421
|
+
@opportunity.campaign.should == @campaign.reload
|
422
|
+
@campaign.revenue.to_i.should == 1000 # 1000 - 100 discount.
|
423
|
+
end
|
424
|
+
end
|
425
|
+
|
426
|
+
describe "with invalid params" do
|
427
|
+
|
428
|
+
it "should expose a newly created but unsaved opportunity as @opportunity with blank @account and render [create] template" do
|
429
|
+
@account = Account.new(:user => @current_user)
|
430
|
+
@opportunity = FactoryGirl.build(:opportunity, :name => nil, :campaign => nil, :user => @current_user,
|
431
|
+
:account => @account)
|
432
|
+
Opportunity.stub!(:new).and_return(@opportunity)
|
433
|
+
@stage = Setting.unroll(:opportunity_stage)
|
434
|
+
@users = [ FactoryGirl.create(:user) ]
|
435
|
+
@accounts = [ FactoryGirl.create(:account, :user => @current_user) ]
|
436
|
+
|
437
|
+
# Expect to redraw [create] form with blank account.
|
438
|
+
xhr :post, :create, :opportunity => {}, :account => { :user_id => @current_user.id }
|
439
|
+
assigns(:opportunity).should == @opportunity
|
440
|
+
assigns(:users).should == @users
|
441
|
+
assigns(:account).attributes.should == @account.attributes
|
442
|
+
assigns(:accounts).should == @accounts
|
443
|
+
response.should render_template("opportunities/create")
|
444
|
+
end
|
445
|
+
|
446
|
+
it "should expose a newly created but unsaved opportunity as @opportunity with existing @account and render [create] template" do
|
447
|
+
@account = FactoryGirl.create(:account, :id => 42, :user => @current_user)
|
448
|
+
@opportunity = FactoryGirl.build(:opportunity, :name => nil, :campaign => nil, :user => @current_user,
|
449
|
+
:account => @account)
|
450
|
+
Opportunity.stub!(:new).and_return(@opportunity)
|
451
|
+
@stage = Setting.unroll(:opportunity_stage)
|
452
|
+
@users = [ FactoryGirl.create(:user) ]
|
453
|
+
|
454
|
+
# Expect to redraw [create] form with selected account.
|
455
|
+
xhr :post, :create, :opportunity => {}, :account => { :id => 42, :user_id => @current_user.id }
|
456
|
+
assigns(:opportunity).should == @opportunity
|
457
|
+
assigns(:users).should == @users
|
458
|
+
assigns(:account).should == @account
|
459
|
+
assigns(:accounts).should == [ @account ]
|
460
|
+
response.should render_template("opportunities/create")
|
461
|
+
end
|
462
|
+
|
463
|
+
it "should preserve the campaign when called from campaign landing page" do
|
464
|
+
@campaign = FactoryGirl.create(:campaign, :id => 42)
|
465
|
+
|
466
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/42"
|
467
|
+
xhr :post, :create, :opportunity => { :name => nil }, :campaign => 42, :account => { :name => "Test Account" }, :users => []
|
468
|
+
assigns(:campaign).should == @campaign
|
469
|
+
response.should render_template("opportunities/create")
|
470
|
+
end
|
471
|
+
|
472
|
+
it "should preserve the contact when called from contact landing page" do
|
473
|
+
@contact = FactoryGirl.create(:contact, :id => 42)
|
474
|
+
|
475
|
+
request.env["HTTP_REFERER"] = "http://localhost/contacts/42"
|
476
|
+
xhr :post, :create, :opportunity => { :name => nil }, :contact => 42, :account => { :name => "Test Account" }, :users => []
|
477
|
+
assigns(:contact).should == @contact
|
478
|
+
response.should render_template("opportunities/create")
|
479
|
+
end
|
480
|
+
|
481
|
+
end
|
482
|
+
|
483
|
+
end
|
484
|
+
|
485
|
+
# PUT /opportunities/1
|
486
|
+
# PUT /opportunities/1.xml AJAX
|
487
|
+
#----------------------------------------------------------------------------
|
488
|
+
describe "responding to PUT update" do
|
489
|
+
|
490
|
+
describe "with valid params" do
|
491
|
+
|
492
|
+
it "should update the requested opportunity, expose it as @opportunity, and render [update] template" do
|
493
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
494
|
+
@stage = Setting.unroll(:opportunity_stage)
|
495
|
+
|
496
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello world" }, :account => { :name => "Test Account" }, :users => %w(1 2 3)
|
497
|
+
@opportunity.reload.name.should == "Hello world"
|
498
|
+
assigns(:opportunity).should == @opportunity
|
499
|
+
assigns(:stage).should == @stage
|
500
|
+
assigns(:opportunity_stage_total).should == nil
|
501
|
+
response.should render_template("opportunities/update")
|
502
|
+
end
|
503
|
+
|
504
|
+
it "should get sidebar data if called from opportunities index" do
|
505
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
506
|
+
|
507
|
+
request.env["HTTP_REFERER"] = "http://localhost/opportunities"
|
508
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello world" }, :account => { :name => "Test Account" }
|
509
|
+
assigns(:opportunity_stage_total).should be_an_instance_of(HashWithIndifferentAccess)
|
510
|
+
end
|
511
|
+
|
512
|
+
it "should find related account if called from account landing page" do
|
513
|
+
@account = FactoryGirl.create(:account, :user => @current_user)
|
514
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :account => @account)
|
515
|
+
request.env["HTTP_REFERER"] = "http://localhost/accounts/#{@account.id}"
|
516
|
+
|
517
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello world" }
|
518
|
+
assigns(:account).should == @account
|
519
|
+
end
|
520
|
+
|
521
|
+
it "should remove related account if blank :account param is given" do
|
522
|
+
@account = FactoryGirl.create(:account, :user => @current_user)
|
523
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :account => @account)
|
524
|
+
request.env["HTTP_REFERER"] = "http://localhost/accounts/#{@account.id}"
|
525
|
+
|
526
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello world" }, :account => { :id => "" }
|
527
|
+
assigns(:account).should == nil
|
528
|
+
end
|
529
|
+
|
530
|
+
it "should find related campaign if called from campaign landing page" do
|
531
|
+
@campaign = FactoryGirl.create(:campaign, :user => @current_user)
|
532
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :user => @current_user)
|
533
|
+
@campaign.opportunities << @opportunity
|
534
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
535
|
+
|
536
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello world", :campaign_id => @campaign.id }, :account => {}
|
537
|
+
assigns(:campaign).should == @campaign
|
538
|
+
end
|
539
|
+
|
540
|
+
it "should be able to create an account and associate it with updated opportunity" do
|
541
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
542
|
+
|
543
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello" }, :account => { :name => "new account" }
|
544
|
+
assigns[:opportunity].should == @opportunity
|
545
|
+
@opportunity.reload
|
546
|
+
@opportunity.account.should_not be_nil
|
547
|
+
@opportunity.account.name.should == "new account"
|
548
|
+
end
|
549
|
+
|
550
|
+
it "should be able to create an account and associate it with updated opportunity" do
|
551
|
+
@old_account = FactoryGirl.create(:account, :id => 111)
|
552
|
+
@new_account = FactoryGirl.create(:account, :id => 999)
|
553
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :account => @old_account)
|
554
|
+
|
555
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello" }, :account => { :id => 999 }
|
556
|
+
@opportunity.reload
|
557
|
+
assigns[:opportunity].should == @opportunity
|
558
|
+
@opportunity.account.should == @new_account
|
559
|
+
end
|
560
|
+
|
561
|
+
it "should update opportunity permissions when sharing with specific users" do
|
562
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :access => "Public")
|
563
|
+
he = FactoryGirl.create(:user, :id => 7)
|
564
|
+
she = FactoryGirl.create(:user, :id => 8)
|
565
|
+
|
566
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => "Hello", :access => "Shared" }, :users => %w(7 8), :account => { :name => "Test Account" }
|
567
|
+
@opportunity.reload.access.should == "Shared"
|
568
|
+
@opportunity.permissions.map(&:user_id).sort.should == [ 7, 8 ]
|
569
|
+
assigns[:opportunity].should == @opportunity
|
570
|
+
end
|
571
|
+
|
572
|
+
it "should reload opportunity campaign if called from campaign landing page" do
|
573
|
+
@campaign = FactoryGirl.create(:campaign)
|
574
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaign)
|
575
|
+
|
576
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
577
|
+
xhr :put, :update, :id => @opportunity.id, :opportunity => { :name => "Hello" }, :account => { :name => "Test Account" }
|
578
|
+
assigns[:campaign].should == @campaign
|
579
|
+
end
|
580
|
+
|
581
|
+
describe "updating campaign revenue (same campaign)" do
|
582
|
+
it "should add to actual revenue when opportunity is closed/won" do
|
583
|
+
@campaign = FactoryGirl.create(:campaign, :revenue => 1000)
|
584
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaign, :stage => nil, :amount => 1100, :discount => 100)
|
585
|
+
|
586
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => "won" }, :account => { :name => "Test Account" }
|
587
|
+
@campaign.reload.revenue.to_i.should == 2000 # 1000 -> 2000
|
588
|
+
end
|
589
|
+
|
590
|
+
it "should substract from actual revenue when opportunity is no longer closed/won" do
|
591
|
+
@campaign = FactoryGirl.create(:campaign, :revenue => 1000)
|
592
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaign, :stage => "won", :amount => 1100, :discount => 100)
|
593
|
+
# @campaign.revenue is now $2000 since we created winning opportunity.
|
594
|
+
|
595
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => nil }, :account => { :name => "Test Account" }
|
596
|
+
@campaign.reload.revenue.to_i.should == 1000 # Should be adjusted back to $1000.
|
597
|
+
end
|
598
|
+
|
599
|
+
it "should not update actual revenue when opportunity is not closed/won" do
|
600
|
+
@campaign = FactoryGirl.create(:campaign, :revenue => 1000)
|
601
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaign, :stage => nil, :amount => 1100, :discount => 100)
|
602
|
+
|
603
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => "lost" }, :account => { :name => "Test Account" }
|
604
|
+
@campaign.reload.revenue.to_i.should == 1000 # Stays the same.
|
605
|
+
end
|
606
|
+
end
|
607
|
+
|
608
|
+
describe "updating campaign revenue (diferent campaigns)" do
|
609
|
+
it "should update newly assigned campaign when opportunity is closed/won" do
|
610
|
+
@campaigns = { :old => FactoryGirl.create(:campaign, :revenue => 1000), :new => FactoryGirl.create(:campaign, :revenue => 1000) }
|
611
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaigns[:old], :stage => nil, :amount => 1100, :discount => 100)
|
612
|
+
|
613
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => "won", :campaign_id => @campaigns[:new].id }, :account => { :name => "Test Account" }
|
614
|
+
|
615
|
+
@campaigns[:old].reload.revenue.to_i.should == 1000 # Stays the same.
|
616
|
+
@campaigns[:new].reload.revenue.to_i.should == 2000 # 1000 -> 2000
|
617
|
+
end
|
618
|
+
|
619
|
+
it "should update old campaign when opportunity is no longer closed/won" do
|
620
|
+
@campaigns = { :old => FactoryGirl.create(:campaign, :revenue => 1000), :new => FactoryGirl.create(:campaign, :revenue => 1000) }
|
621
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaigns[:old], :stage => "won", :amount => 1100, :discount => 100)
|
622
|
+
# @campaign.revenue is now $2000 since we created winning opportunity.
|
623
|
+
|
624
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => nil, :campaign_id => @campaigns[:new].id }, :account => { :name => "Test Account" }
|
625
|
+
@campaigns[:old].reload.revenue.to_i.should == 1000 # Should be adjusted back to $1000.
|
626
|
+
@campaigns[:new].reload.revenue.to_i.should == 1000 # Stays the same.
|
627
|
+
end
|
628
|
+
|
629
|
+
it "should not update campaigns when opportunity is not closed/won" do
|
630
|
+
@campaigns = { :old => FactoryGirl.create(:campaign, :revenue => 1000), :new => FactoryGirl.create(:campaign, :revenue => 1000) }
|
631
|
+
@opportunity = FactoryGirl.create(:opportunity, :campaign => @campaigns[:old], :stage => nil, :amount => 1100, :discount => 100)
|
632
|
+
|
633
|
+
xhr :put, :update, :id => @opportunity, :opportunity => { :stage => "lost", :campaign_id => @campaigns[:new].id }, :account => { :name => "Test Account" }
|
634
|
+
@campaigns[:old].reload.revenue.to_i.should == 1000 # Stays the same.
|
635
|
+
@campaigns[:new].reload.revenue.to_i.should == 1000 # Stays the same.
|
636
|
+
end
|
637
|
+
end
|
638
|
+
|
639
|
+
describe "opportunity got deleted or otherwise unavailable" do
|
640
|
+
it "should reload current page with the flash message if the opportunity got deleted" do
|
641
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
642
|
+
@opportunity.destroy
|
643
|
+
|
644
|
+
xhr :put, :update, :id => @opportunity.id
|
645
|
+
flash[:warning].should_not == nil
|
646
|
+
response.body.should == "window.location.reload();"
|
647
|
+
end
|
648
|
+
|
649
|
+
it "should reload current page with the flash message if the opportunity is protected" do
|
650
|
+
@private = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user), :access => "Private")
|
651
|
+
|
652
|
+
xhr :put, :update, :id => @private.id
|
653
|
+
flash[:warning].should_not == nil
|
654
|
+
response.body.should == "window.location.reload();"
|
655
|
+
end
|
656
|
+
end
|
657
|
+
end
|
658
|
+
|
659
|
+
describe "with invalid params" do
|
660
|
+
|
661
|
+
it "should not update the requested opportunity but still expose it as @opportunity, and render [update] template" do
|
662
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42, :name => "Hello people")
|
663
|
+
|
664
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => nil }, :account => { :name => "Test Account" }
|
665
|
+
@opportunity.reload.name.should == "Hello people"
|
666
|
+
assigns(:opportunity).should == @opportunity
|
667
|
+
assigns(:opportunity_stage_total).should == nil
|
668
|
+
response.should render_template("opportunities/update")
|
669
|
+
end
|
670
|
+
|
671
|
+
it "should expose existing account as @account if selected" do
|
672
|
+
@account = FactoryGirl.create(:account, :id => 99)
|
673
|
+
@opportunity = FactoryGirl.create(:opportunity, :id => 42)
|
674
|
+
FactoryGirl.create(:account_opportunity, :account => @account, :opportunity => @opportunity)
|
675
|
+
|
676
|
+
xhr :put, :update, :id => 42, :opportunity => { :name => nil }, :account => { :id => 99 }
|
677
|
+
assigns(:account).should == @account
|
678
|
+
end
|
679
|
+
|
680
|
+
end
|
681
|
+
|
682
|
+
end
|
683
|
+
|
684
|
+
# DELETE /opportunities/1
|
685
|
+
# DELETE /opportunities/1.xml AJAX
|
686
|
+
#----------------------------------------------------------------------------
|
687
|
+
describe "responding to DELETE destroy" do
|
688
|
+
before do
|
689
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
690
|
+
end
|
691
|
+
|
692
|
+
describe "AJAX request" do
|
693
|
+
it "should destroy the requested opportunity and render [destroy] template" do
|
694
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
695
|
+
|
696
|
+
lambda { Opportunity.find(@opportunity) }.should raise_error(ActiveRecord::RecordNotFound)
|
697
|
+
assigns(:opportunity_stage_total).should == nil
|
698
|
+
response.should render_template("opportunities/destroy")
|
699
|
+
end
|
700
|
+
|
701
|
+
describe "when called from Opportunities index page" do
|
702
|
+
before do
|
703
|
+
request.env["HTTP_REFERER"] = "http://localhost/opportunities"
|
704
|
+
end
|
705
|
+
|
706
|
+
it "should get sidebar data if called from opportunities index" do
|
707
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
708
|
+
assigns(:opportunity_stage_total).should be_an_instance_of(HashWithIndifferentAccess)
|
709
|
+
end
|
710
|
+
|
711
|
+
it "should try previous page and render index action if current page has no opportunities" do
|
712
|
+
session[:opportunities_current_page] = 42
|
713
|
+
|
714
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
715
|
+
session[:opportunities_current_page].should == 41
|
716
|
+
response.should render_template("opportunities/index")
|
717
|
+
end
|
718
|
+
|
719
|
+
it "should render index action when deleting last opportunity" do
|
720
|
+
session[:opportunities_current_page] = 1
|
721
|
+
|
722
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
723
|
+
session[:opportunities_current_page].should == 1
|
724
|
+
response.should render_template("opportunities/index")
|
725
|
+
end
|
726
|
+
end
|
727
|
+
|
728
|
+
describe "when called from related asset page" do
|
729
|
+
it "should reset current page to 1" do
|
730
|
+
request.env["HTTP_REFERER"] = "http://localhost/accounts/123"
|
731
|
+
|
732
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
733
|
+
session[:opportunities_current_page].should == 1
|
734
|
+
response.should render_template("opportunities/destroy")
|
735
|
+
end
|
736
|
+
|
737
|
+
it "should reload campaiign to be able to refresh its summary" do
|
738
|
+
@account = FactoryGirl.create(:account)
|
739
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user, :account => @account)
|
740
|
+
request.env["HTTP_REFERER"] = "http://localhost/accounts/#{@account.id}"
|
741
|
+
|
742
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
743
|
+
assigns[:account].should == @account
|
744
|
+
response.should render_template("opportunities/destroy")
|
745
|
+
end
|
746
|
+
|
747
|
+
it "should reload campaiign to be able to refresh its summary" do
|
748
|
+
@campaign = FactoryGirl.create(:campaign)
|
749
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user, :campaign => @campaign)
|
750
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
751
|
+
|
752
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
753
|
+
assigns[:campaign].should == @campaign
|
754
|
+
response.should render_template("opportunities/destroy")
|
755
|
+
end
|
756
|
+
end
|
757
|
+
|
758
|
+
describe "opportunity got deleted or otherwise unavailable" do
|
759
|
+
it "should reload current page is the opportunity got deleted" do
|
760
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
761
|
+
@opportunity.destroy
|
762
|
+
|
763
|
+
xhr :delete, :destroy, :id => @opportunity.id
|
764
|
+
flash[:warning].should_not == nil
|
765
|
+
response.body.should == "window.location.reload();"
|
766
|
+
end
|
767
|
+
|
768
|
+
it "should reload current page with the flash message if the opportunity is protected" do
|
769
|
+
@private = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user), :access => "Private")
|
770
|
+
|
771
|
+
xhr :delete, :destroy, :id => @private.id
|
772
|
+
flash[:warning].should_not == nil
|
773
|
+
response.body.should == "window.location.reload();"
|
774
|
+
end
|
775
|
+
end
|
776
|
+
end
|
777
|
+
|
778
|
+
describe "HTML request" do
|
779
|
+
it "should redirect to Opportunities index when an opportunity gets deleted from its landing page" do
|
780
|
+
delete :destroy, :id => @opportunity.id
|
781
|
+
flash[:notice].should_not == nil
|
782
|
+
response.should redirect_to(opportunities_path)
|
783
|
+
end
|
784
|
+
|
785
|
+
it "should redirect to opportunity index with the flash message is the opportunity got deleted" do
|
786
|
+
@opportunity = FactoryGirl.create(:opportunity, :user => @current_user)
|
787
|
+
@opportunity.destroy
|
788
|
+
|
789
|
+
delete :destroy, :id => @opportunity.id
|
790
|
+
flash[:warning].should_not == nil
|
791
|
+
response.should redirect_to(opportunities_path)
|
792
|
+
end
|
793
|
+
|
794
|
+
it "should redirect to opportunity index with the flash message if the opportunity is protected" do
|
795
|
+
@private = FactoryGirl.create(:opportunity, :user => FactoryGirl.create(:user), :access => "Private")
|
796
|
+
|
797
|
+
delete :destroy, :id => @private.id
|
798
|
+
flash[:warning].should_not == nil
|
799
|
+
response.should redirect_to(opportunities_path)
|
800
|
+
end
|
801
|
+
end
|
802
|
+
end
|
803
|
+
|
804
|
+
# PUT /opportunities/1/attach
|
805
|
+
# PUT /opportunities/1/attach.xml AJAX
|
806
|
+
#----------------------------------------------------------------------------
|
807
|
+
describe "responding to PUT attach" do
|
808
|
+
describe "tasks" do
|
809
|
+
before do
|
810
|
+
@model = FactoryGirl.create(:opportunity)
|
811
|
+
@attachment = FactoryGirl.create(:task, :asset => nil)
|
812
|
+
end
|
813
|
+
it_should_behave_like("attach")
|
814
|
+
end
|
815
|
+
|
816
|
+
describe "contacts" do
|
817
|
+
before do
|
818
|
+
@model = FactoryGirl.create(:opportunity)
|
819
|
+
@attachment = FactoryGirl.create(:contact)
|
820
|
+
end
|
821
|
+
it_should_behave_like("attach")
|
822
|
+
end
|
823
|
+
end
|
824
|
+
|
825
|
+
# POST /opportunities/1/discard
|
826
|
+
# POST /opportunities/1/discard.xml AJAX
|
827
|
+
#----------------------------------------------------------------------------
|
828
|
+
describe "responding to POST discard" do
|
829
|
+
describe "tasks" do
|
830
|
+
before do
|
831
|
+
@model = FactoryGirl.create(:opportunity)
|
832
|
+
@attachment = FactoryGirl.create(:task, :asset => @model)
|
833
|
+
end
|
834
|
+
it_should_behave_like("discard")
|
835
|
+
end
|
836
|
+
|
837
|
+
describe "contacts" do
|
838
|
+
before do
|
839
|
+
@attachment = FactoryGirl.create(:contact)
|
840
|
+
@model = FactoryGirl.create(:opportunity)
|
841
|
+
@model.contacts << @attachment
|
842
|
+
end
|
843
|
+
it_should_behave_like("discard")
|
844
|
+
end
|
845
|
+
end
|
846
|
+
|
847
|
+
# POST /opportunities/auto_complete/query AJAX
|
848
|
+
#----------------------------------------------------------------------------
|
849
|
+
describe "responding to POST auto_complete" do
|
850
|
+
before do
|
851
|
+
@auto_complete_matches = [ FactoryGirl.create(:opportunity, :name => "Hello World", :user => @current_user) ]
|
852
|
+
end
|
853
|
+
|
854
|
+
it_should_behave_like("auto complete")
|
855
|
+
end
|
856
|
+
|
857
|
+
# GET /opportunities/options AJAX
|
858
|
+
#----------------------------------------------------------------------------
|
859
|
+
describe "responding to GET options" do
|
860
|
+
it "should set current user preferences when showing options" do
|
861
|
+
@per_page = FactoryGirl.create(:preference, :user => @current_user, :name => "opportunities_per_page", :value => Base64.encode64(Marshal.dump(42)))
|
862
|
+
@outline = FactoryGirl.create(:preference, :user => @current_user, :name => "opportunities_outline", :value => Base64.encode64(Marshal.dump("option_long")))
|
863
|
+
@sort_by = FactoryGirl.create(:preference, :user => @current_user, :name => "opportunities_sort_by", :value => Base64.encode64(Marshal.dump("opportunities.name ASC")))
|
864
|
+
|
865
|
+
xhr :get, :options
|
866
|
+
assigns[:per_page].should == 42
|
867
|
+
assigns[:outline].should == "option_long"
|
868
|
+
assigns[:sort_by].should == "opportunities.name ASC"
|
869
|
+
end
|
870
|
+
|
871
|
+
it "should not assign instance variables when hiding options" do
|
872
|
+
xhr :get, :options, :cancel => "true"
|
873
|
+
assigns[:per_page].should == nil
|
874
|
+
assigns[:outline].should == nil
|
875
|
+
assigns[:sort_by].should == nil
|
876
|
+
end
|
877
|
+
end
|
878
|
+
|
879
|
+
# POST /opportunities/redraw AJAX
|
880
|
+
#----------------------------------------------------------------------------
|
881
|
+
describe "responding to POST redraw" do
|
882
|
+
it "should save user selected opportunity preference" do
|
883
|
+
xhr :post, :redraw, :per_page => 42, :outline => "brief", :sort_by => "name"
|
884
|
+
@current_user.preference[:opportunities_per_page].should == "42"
|
885
|
+
@current_user.preference[:opportunities_outline].should == "brief"
|
886
|
+
@current_user.preference[:opportunities_sort_by].should == "opportunities.name ASC"
|
887
|
+
end
|
888
|
+
|
889
|
+
it "should reset current page to 1" do
|
890
|
+
xhr :post, :redraw, :per_page => 42, :outline => "brief", :sort_by => "name"
|
891
|
+
session[:opportunities_current_page].should == 1
|
892
|
+
end
|
893
|
+
|
894
|
+
it "should select @opportunities and render [index] template" do
|
895
|
+
@opportunities = [
|
896
|
+
FactoryGirl.create(:opportunity, :name => "A", :user => @current_user),
|
897
|
+
FactoryGirl.create(:opportunity, :name => "B", :user => @current_user)
|
898
|
+
]
|
899
|
+
|
900
|
+
xhr :post, :redraw, :per_page => 1, :sort_by => "name"
|
901
|
+
assigns(:opportunities).should == [ @opportunities.first ]
|
902
|
+
response.should render_template("opportunities/index")
|
903
|
+
end
|
904
|
+
end
|
905
|
+
|
906
|
+
# POST /opportunities/filter AJAX
|
907
|
+
#----------------------------------------------------------------------------
|
908
|
+
describe "responding to GET filter" do
|
909
|
+
|
910
|
+
it "should expose filtered opportunities as @opportunity and render [filter] template" do
|
911
|
+
session[:filter_by_opportunity_stage] = "negotiation,analysis"
|
912
|
+
@opportunities = [ FactoryGirl.create(:opportunity, :stage => "prospecting", :user => @current_user) ]
|
913
|
+
@stage = Setting.unroll(:opportunity_stage)
|
914
|
+
|
915
|
+
xhr :get, :filter, :stage => "prospecting"
|
916
|
+
assigns(:opportunities).should == @opportunities
|
917
|
+
assigns[:stage].should == @stage
|
918
|
+
response.should be_a_success
|
919
|
+
response.should render_template("opportunities/index")
|
920
|
+
end
|
921
|
+
|
922
|
+
it "should reset current page to 1" do
|
923
|
+
@opportunities = []
|
924
|
+
xhr :get, :filter, :status => "new"
|
925
|
+
|
926
|
+
session[:opportunities_current_page].should == 1
|
927
|
+
end
|
928
|
+
|
929
|
+
end
|
930
|
+
|
931
|
+
end
|
932
|
+
|