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,32 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
describe EmailsController, "handling GET /emails" do
|
4
|
+
MEDIATOR = [ :account, :campaign, :contact, :lead, :opportunity ].freeze
|
5
|
+
|
6
|
+
before(:each) do
|
7
|
+
require_user
|
8
|
+
end
|
9
|
+
|
10
|
+
# DELETE /emails/1
|
11
|
+
# DELETE /emails/1.xml AJAX
|
12
|
+
#----------------------------------------------------------------------------
|
13
|
+
describe "responding to DELETE destroy" do
|
14
|
+
describe "AJAX request" do
|
15
|
+
describe "with valid params" do
|
16
|
+
MEDIATOR.each do |asset|
|
17
|
+
it "should destroy the requested email and render [destroy] template" do
|
18
|
+
@asset = FactoryGirl.create(asset)
|
19
|
+
@email = FactoryGirl.create(:email, :mediator => @asset, :user => @current_user)
|
20
|
+
Email.stub!(:new).and_return(@email)
|
21
|
+
|
22
|
+
xhr :delete, :destroy, :id => @email.id
|
23
|
+
lambda { Email.find(@email) }.should raise_error(ActiveRecord::RecordNotFound)
|
24
|
+
response.should render_template("emails/destroy")
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
end
|
32
|
+
|
@@ -0,0 +1,99 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
describe HomeController do
|
4
|
+
|
5
|
+
# GET /
|
6
|
+
#----------------------------------------------------------------------------
|
7
|
+
describe "responding to GET /" do
|
8
|
+
before(:each) do
|
9
|
+
require_user
|
10
|
+
end
|
11
|
+
|
12
|
+
it "should get a list of activities" do
|
13
|
+
@activity = FactoryGirl.create(:activity, :subject => FactoryGirl.create(:account, :user => @current_user))
|
14
|
+
controller.should_receive(:get_activities).once.and_return([ @activity ])
|
15
|
+
|
16
|
+
get :index
|
17
|
+
assigns[:activities].should == [ @activity ]
|
18
|
+
end
|
19
|
+
|
20
|
+
it "should assign @hello and call hook" do
|
21
|
+
require_user
|
22
|
+
controller.should_receive(:hook).at_least(:once)
|
23
|
+
|
24
|
+
get :index
|
25
|
+
assigns[:hello].should == "Hello world"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# GET /home/options AJAX
|
30
|
+
#----------------------------------------------------------------------------
|
31
|
+
describe "responding to GET options" do
|
32
|
+
before(:each) do
|
33
|
+
require_user
|
34
|
+
end
|
35
|
+
|
36
|
+
it "should assign instance variables for user preferences" do
|
37
|
+
@asset = FactoryGirl.create(:preference, :user => @current_user, :name => "activity_asset", :value => Base64.encode64(Marshal.dump("tasks")))
|
38
|
+
@user = FactoryGirl.create(:preference, :user => @current_user, :name => "activity_user", :value => Base64.encode64(Marshal.dump("Billy Bones")))
|
39
|
+
@duration = FactoryGirl.create(:preference, :user => @current_user, :name => "activity_duration", :value => Base64.encode64(Marshal.dump("two days")))
|
40
|
+
|
41
|
+
xhr :get, :options
|
42
|
+
assigns[:asset].should == "tasks"
|
43
|
+
assigns[:user].should == "Billy Bones"
|
44
|
+
assigns[:duration].should == "two days"
|
45
|
+
assigns[:all_users].should == User.order("first_name, last_name").all
|
46
|
+
end
|
47
|
+
|
48
|
+
it "should not assign instance variables when hiding options" do
|
49
|
+
xhr :get, :options, :cancel => "true"
|
50
|
+
assigns[:asset].should == nil
|
51
|
+
assigns[:user].should == nil
|
52
|
+
assigns[:duration].should == nil
|
53
|
+
assigns[:all_users].should == nil
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
# POST /home/redraw AJAX
|
58
|
+
#----------------------------------------------------------------------------
|
59
|
+
describe "responding to POST redraw" do
|
60
|
+
before(:each) do
|
61
|
+
require_user
|
62
|
+
end
|
63
|
+
|
64
|
+
it "should save user selected options" do
|
65
|
+
xhr :post, :redraw, :asset => "tasks", :user => "Billy Bones", :duration => "two days"
|
66
|
+
@current_user.pref[:activity_asset].should == "tasks"
|
67
|
+
@current_user.pref[:activity_user].should == "Billy Bones"
|
68
|
+
@current_user.pref[:activity_duration].should == "two days"
|
69
|
+
end
|
70
|
+
|
71
|
+
it "should get a list of activities" do
|
72
|
+
@activity = FactoryGirl.create(:activity, :subject => FactoryGirl.create(:account, :user => @current_user))
|
73
|
+
controller.should_receive(:get_activities).once.and_return([ @activity ])
|
74
|
+
|
75
|
+
get :index
|
76
|
+
assigns[:activities].should == [ @activity ]
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
# GET /home/toggle AJAX
|
81
|
+
#----------------------------------------------------------------------------
|
82
|
+
describe "responding to GET toggle" do
|
83
|
+
it "should toggle expand/collapse state of form section in the session (delete existing session key)" do
|
84
|
+
session[:hello] = "world"
|
85
|
+
|
86
|
+
xhr :get, :toggle, :id => "hello"
|
87
|
+
session.keys.should_not include(:hello)
|
88
|
+
end
|
89
|
+
|
90
|
+
it "should toggle expand/collapse state of form section in the session (save new session key)" do
|
91
|
+
session.delete(:hello)
|
92
|
+
|
93
|
+
xhr :get, :toggle, :id => "hello"
|
94
|
+
session[:hello].should == true
|
95
|
+
end
|
96
|
+
end
|
97
|
+
|
98
|
+
end
|
99
|
+
|
@@ -0,0 +1,1047 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
describe LeadsController do
|
4
|
+
|
5
|
+
before(:each) do
|
6
|
+
require_user
|
7
|
+
set_current_tab(:leads)
|
8
|
+
end
|
9
|
+
|
10
|
+
# GET /leads
|
11
|
+
# GET /leads.xml AJAX and HTML
|
12
|
+
#----------------------------------------------------------------------------
|
13
|
+
describe "responding to GET index" do
|
14
|
+
|
15
|
+
it "should expose all leads as @leads and render [index] template" do
|
16
|
+
@leads = [ FactoryGirl.create(:lead, :user => @current_user) ]
|
17
|
+
|
18
|
+
get :index
|
19
|
+
assigns[:leads].should == @leads
|
20
|
+
response.should render_template("leads/index")
|
21
|
+
end
|
22
|
+
|
23
|
+
it "should collect the data for the leads sidebar" do
|
24
|
+
@leads = [ FactoryGirl.create(:lead, :user => @current_user) ]
|
25
|
+
@status = Setting.lead_status.dup
|
26
|
+
|
27
|
+
get :index
|
28
|
+
(assigns[:lead_status_total].keys.map(&:to_sym) - (@status << :all << :other)).should == []
|
29
|
+
end
|
30
|
+
|
31
|
+
it "should filter out leads by status" do
|
32
|
+
controller.session[:filter_by_lead_status] = "new,contacted"
|
33
|
+
@leads = [
|
34
|
+
FactoryGirl.create(:lead, :status => "new", :user => @current_user),
|
35
|
+
FactoryGirl.create(:lead, :status => "contacted", :user => @current_user)
|
36
|
+
]
|
37
|
+
|
38
|
+
# This one should be filtered out.
|
39
|
+
FactoryGirl.create(:lead, :status => "rejected", :user => @current_user)
|
40
|
+
|
41
|
+
get :index
|
42
|
+
# Note: can't compare campaigns directly because of BigDecimals.
|
43
|
+
assigns[:leads].size.should == 2
|
44
|
+
assigns[:leads].map(&:status).sort.should == %w(contacted new)
|
45
|
+
end
|
46
|
+
|
47
|
+
it "should perform lookup using query string" do
|
48
|
+
@billy_bones = FactoryGirl.create(:lead, :user => @current_user, :first_name => "Billy", :last_name => "Bones")
|
49
|
+
@captain_flint = FactoryGirl.create(:lead, :user => @current_user, :first_name => "Captain", :last_name => "Flint")
|
50
|
+
|
51
|
+
get :index, :query => "bill"
|
52
|
+
assigns[:leads].should == [ @billy_bones ]
|
53
|
+
assigns[:current_query].should == "bill"
|
54
|
+
session[:leads_current_query].should == "bill"
|
55
|
+
end
|
56
|
+
|
57
|
+
describe "AJAX pagination" do
|
58
|
+
it "should pick up page number from params" do
|
59
|
+
@leads = [ FactoryGirl.create(:lead, :user => @current_user) ]
|
60
|
+
xhr :get, :index, :page => 42
|
61
|
+
|
62
|
+
assigns[:current_page].to_i.should == 42
|
63
|
+
assigns[:leads].should == [] # page #42 should be empty if there's only one lead ;-)
|
64
|
+
session[:leads_current_page].to_i.should == 42
|
65
|
+
response.should render_template("leads/index")
|
66
|
+
end
|
67
|
+
|
68
|
+
it "should pick up saved page number from session" do
|
69
|
+
session[:leads_current_page] = 42
|
70
|
+
@leads = [ FactoryGirl.create(:lead, :user => @current_user) ]
|
71
|
+
xhr :get, :index
|
72
|
+
|
73
|
+
assigns[:current_page].should == 42
|
74
|
+
assigns[:leads].should == []
|
75
|
+
response.should render_template("leads/index")
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
describe "with mime type of JSON" do
|
80
|
+
it "should render all leads as JSON" do
|
81
|
+
@controller.should_receive(:get_list_of_records).and_return(leads = mock("Array of Leads"))
|
82
|
+
leads.should_receive(:to_json).and_return("generated JSON")
|
83
|
+
|
84
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
85
|
+
get :index
|
86
|
+
response.body.should == "generated JSON"
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
describe "with mime type of XML" do
|
91
|
+
it "should render all leads as xml" do
|
92
|
+
@controller.should_receive(:get_list_of_records).and_return(leads = mock("Array of Leads"))
|
93
|
+
leads.should_receive(:to_xml).and_return("generated XML")
|
94
|
+
|
95
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
96
|
+
get :index
|
97
|
+
response.body.should == "generated XML"
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
101
|
+
|
102
|
+
# GET /leads/1
|
103
|
+
# GET /leads/1.xml HTML
|
104
|
+
#----------------------------------------------------------------------------
|
105
|
+
describe "responding to GET show" do
|
106
|
+
|
107
|
+
describe "with mime type of HTML" do
|
108
|
+
before(:each) do
|
109
|
+
@lead = FactoryGirl.create(:lead, :id => 42, :user => @current_user)
|
110
|
+
@comment = Comment.new
|
111
|
+
end
|
112
|
+
|
113
|
+
it "should expose the requested lead as @lead and render [show] template" do
|
114
|
+
get :show, :id => 42
|
115
|
+
assigns[:lead].should == @lead
|
116
|
+
assigns[:comment].attributes.should == @comment.attributes
|
117
|
+
response.should render_template("leads/show")
|
118
|
+
end
|
119
|
+
|
120
|
+
it "should update an activity when viewing the lead" do
|
121
|
+
Activity.should_receive(:log).with(@current_user, @lead, :viewed).once
|
122
|
+
get :show, :id => @lead.id
|
123
|
+
end
|
124
|
+
end
|
125
|
+
|
126
|
+
describe "with mime type of JSON" do
|
127
|
+
it "should render the requested lead as JSON" do
|
128
|
+
Lead.stub_chain(:my, :find).and_return(lead = mock_model(Lead, :name => ''))
|
129
|
+
lead.should_receive(:to_json).and_return("generated JSON")
|
130
|
+
|
131
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
132
|
+
get :show, :id => 42
|
133
|
+
response.body.should == "generated JSON"
|
134
|
+
end
|
135
|
+
end
|
136
|
+
|
137
|
+
describe "with mime type of XML" do
|
138
|
+
it "should render the requested lead as xml" do
|
139
|
+
Lead.stub_chain(:my, :find).and_return(lead = mock_model(Lead, :name => ''))
|
140
|
+
lead.should_receive(:to_xml).and_return("generated XML")
|
141
|
+
|
142
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
143
|
+
get :show, :id => 42
|
144
|
+
response.body.should == "generated XML"
|
145
|
+
end
|
146
|
+
end
|
147
|
+
|
148
|
+
describe "lead got deleted or otherwise unavailable" do
|
149
|
+
it "should redirect to lead index if the lead got deleted" do
|
150
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
151
|
+
@lead.destroy
|
152
|
+
|
153
|
+
get :show, :id => @lead.id
|
154
|
+
flash[:warning].should_not == nil
|
155
|
+
response.should redirect_to(leads_path)
|
156
|
+
end
|
157
|
+
|
158
|
+
it "should redirect to lead index if the lead is protected" do
|
159
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
160
|
+
|
161
|
+
get :show, :id => @private.id
|
162
|
+
flash[:warning].should_not == nil
|
163
|
+
response.should redirect_to(leads_path)
|
164
|
+
end
|
165
|
+
|
166
|
+
it "should return 404 (Not Found) JSON error" do
|
167
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
168
|
+
@lead.destroy
|
169
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
170
|
+
|
171
|
+
get :show, :id => @lead.id
|
172
|
+
response.code.should == "404" # :not_found
|
173
|
+
end
|
174
|
+
|
175
|
+
it "should return 404 (Not Found) XML error" do
|
176
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
177
|
+
@lead.destroy
|
178
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
179
|
+
|
180
|
+
get :show, :id => @lead.id
|
181
|
+
response.code.should == "404" # :not_found
|
182
|
+
end
|
183
|
+
end
|
184
|
+
|
185
|
+
end
|
186
|
+
|
187
|
+
# GET /leads/new
|
188
|
+
# GET /leads/new.xml AJAX
|
189
|
+
#----------------------------------------------------------------------------
|
190
|
+
describe "responding to GET new" do
|
191
|
+
|
192
|
+
it "should expose a new lead as @lead and render [new] template" do
|
193
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :campaign => nil)
|
194
|
+
Lead.stub!(:new).and_return(@lead)
|
195
|
+
@users = [ FactoryGirl.create(:user) ]
|
196
|
+
@campaigns = [ FactoryGirl.create(:campaign, :user => @current_user) ]
|
197
|
+
|
198
|
+
xhr :get, :new
|
199
|
+
assigns[:lead].attributes.should == @lead.attributes
|
200
|
+
assigns[:users].should == @users
|
201
|
+
assigns[:campaigns].should == @campaigns
|
202
|
+
response.should render_template("leads/new")
|
203
|
+
end
|
204
|
+
|
205
|
+
it "should create related object when necessary" do
|
206
|
+
@campaign = FactoryGirl.create(:campaign, :id => 123)
|
207
|
+
|
208
|
+
xhr :get, :new, :related => "campaign_123"
|
209
|
+
assigns[:campaign].should == @campaign
|
210
|
+
end
|
211
|
+
|
212
|
+
describe "(when creating related lead)" do
|
213
|
+
it "should redirect to parent asset's index page with the message if parent asset got deleted" do
|
214
|
+
@campaign = FactoryGirl.create(:campaign)
|
215
|
+
@campaign.destroy
|
216
|
+
|
217
|
+
xhr :get, :new, :related => "campaign_#{@campaign.id}"
|
218
|
+
flash[:warning].should_not == nil
|
219
|
+
response.body.should == 'window.location.href = "/campaigns";'
|
220
|
+
end
|
221
|
+
|
222
|
+
it "should redirect to parent asset's index page with the message if parent asset got protected" do
|
223
|
+
@campaign = FactoryGirl.create(:campaign, :access => "Private")
|
224
|
+
|
225
|
+
xhr :get, :new, :related => "campaign_#{@campaign.id}"
|
226
|
+
flash[:warning].should_not == nil
|
227
|
+
response.body.should == 'window.location.href = "/campaigns";'
|
228
|
+
end
|
229
|
+
end
|
230
|
+
|
231
|
+
end
|
232
|
+
|
233
|
+
# GET /leads/1/edit AJAX
|
234
|
+
#----------------------------------------------------------------------------
|
235
|
+
describe "responding to GET edit" do
|
236
|
+
|
237
|
+
it "should expose the requested lead as @lead and render [edit] template" do
|
238
|
+
@lead = FactoryGirl.create(:lead, :id => 42, :user => @current_user, :campaign => nil)
|
239
|
+
@users = [ FactoryGirl.create(:user) ]
|
240
|
+
@campaigns = [ FactoryGirl.create(:campaign, :user => @current_user) ]
|
241
|
+
|
242
|
+
xhr :get, :edit, :id => 42
|
243
|
+
assigns[:lead].should == @lead
|
244
|
+
assigns[:users].should == @users
|
245
|
+
assigns[:campaigns].should == @campaigns
|
246
|
+
response.should render_template("leads/edit")
|
247
|
+
end
|
248
|
+
|
249
|
+
it "should find previous lead when necessary" do
|
250
|
+
@lead = FactoryGirl.create(:lead, :id => 42)
|
251
|
+
@previous = FactoryGirl.create(:lead, :id => 321)
|
252
|
+
|
253
|
+
xhr :get, :edit, :id => 42, :previous => 321
|
254
|
+
assigns[:previous].should == @previous
|
255
|
+
end
|
256
|
+
|
257
|
+
describe "lead got deleted or is otherwise unavailable" do
|
258
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
259
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
260
|
+
@lead.destroy
|
261
|
+
|
262
|
+
xhr :get, :edit, :id => @lead.id
|
263
|
+
flash[:warning].should_not == nil
|
264
|
+
response.body.should == "window.location.reload();"
|
265
|
+
end
|
266
|
+
|
267
|
+
it "should reload current page with the flash message if the lead is protected" do
|
268
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
269
|
+
|
270
|
+
xhr :get, :edit, :id => @private.id
|
271
|
+
flash[:warning].should_not == nil
|
272
|
+
response.body.should == "window.location.reload();"
|
273
|
+
end
|
274
|
+
end
|
275
|
+
|
276
|
+
describe "(previous lead got deleted or is otherwise unavailable)" do
|
277
|
+
before(:each) do
|
278
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
279
|
+
@previous = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user))
|
280
|
+
end
|
281
|
+
|
282
|
+
it "should notify the view if previous lead got deleted" do
|
283
|
+
@previous.destroy
|
284
|
+
|
285
|
+
xhr :get, :edit, :id => @lead.id, :previous => @previous.id
|
286
|
+
flash[:warning].should == nil # no warning, just silently remove the div
|
287
|
+
assigns[:previous].should == @previous.id
|
288
|
+
response.should render_template("leads/edit")
|
289
|
+
end
|
290
|
+
|
291
|
+
it "should notify the view if previous lead got protected" do
|
292
|
+
@previous.update_attribute(:access, "Private")
|
293
|
+
|
294
|
+
xhr :get, :edit, :id => @lead.id, :previous => @previous.id
|
295
|
+
flash[:warning].should == nil
|
296
|
+
assigns[:previous].should == @previous.id
|
297
|
+
response.should render_template("leads/edit")
|
298
|
+
end
|
299
|
+
end
|
300
|
+
|
301
|
+
end
|
302
|
+
|
303
|
+
# POST /leads
|
304
|
+
# POST /leads.xml AJAX
|
305
|
+
#----------------------------------------------------------------------------
|
306
|
+
describe "responding to POST create" do
|
307
|
+
|
308
|
+
describe "with valid params" do
|
309
|
+
|
310
|
+
it "should expose a newly created lead as @lead and render [create] template" do
|
311
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :campaign => nil)
|
312
|
+
Lead.stub!(:new).and_return(@lead)
|
313
|
+
@users = [ FactoryGirl.create(:user) ]
|
314
|
+
@campaigns = [ FactoryGirl.create(:campaign, :user => @current_user) ]
|
315
|
+
|
316
|
+
xhr :post, :create, :lead => { :first_name => "Billy", :last_name => "Bones" }, :users => %w(1 2 3)
|
317
|
+
assigns(:lead).should == @lead
|
318
|
+
assigns(:users).should == @users
|
319
|
+
assigns(:campaigns).should == @campaigns
|
320
|
+
assigns[:lead_status_total].should be_nil
|
321
|
+
response.should render_template("leads/create")
|
322
|
+
end
|
323
|
+
|
324
|
+
it "should copy selected campaign permissions unless asked otherwise" do
|
325
|
+
he = FactoryGirl.create(:user, :id => 7)
|
326
|
+
she = FactoryGirl.create(:user, :id => 8)
|
327
|
+
@campaign = FactoryGirl.build(:campaign, :access => "Shared")
|
328
|
+
@campaign.permissions << FactoryGirl.build(:permission, :user => he, :asset => @campaign)
|
329
|
+
@campaign.permissions << FactoryGirl.build(:permission, :user => she, :asset => @campaign)
|
330
|
+
@campaign.save
|
331
|
+
|
332
|
+
@lead = FactoryGirl.build(:lead, :campaign => @campaign, :user => @current_user, :access => "Shared")
|
333
|
+
Lead.stub!(:new).and_return(@lead)
|
334
|
+
|
335
|
+
xhr :put, :create, :lead => { :first_name => "Billy", :last_name => "Bones", :access => "Campaign" }, :campaign => @campaign.id, :users => %w(7 8)
|
336
|
+
@lead.reload.access.should == "Shared"
|
337
|
+
@lead.permissions.map(&:user_id).sort.should == [ 7, 8 ]
|
338
|
+
@lead.permissions.map(&:asset_id).should == [ @lead.id, @lead.id ]
|
339
|
+
@lead.permissions.map(&:asset_type).should == %w(Lead Lead)
|
340
|
+
end
|
341
|
+
|
342
|
+
it "should get the data to update leads sidebar if called from leads index" do
|
343
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :campaign => nil)
|
344
|
+
Lead.stub!(:new).and_return(@lead)
|
345
|
+
|
346
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
347
|
+
xhr :post, :create, :lead => { :first_name => "Billy", :last_name => "Bones" }, :users => %w(1 2 3)
|
348
|
+
assigns[:lead_status_total].should be_an_instance_of(HashWithIndifferentAccess)
|
349
|
+
end
|
350
|
+
|
351
|
+
it "should reload leads to update pagination if called from leads index" do
|
352
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :campaign => nil)
|
353
|
+
Lead.stub!(:new).and_return(@lead)
|
354
|
+
|
355
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
356
|
+
xhr :post, :create, :lead => { :first_name => "Billy", :last_name => "Bones" }, :users => %w(1 2 3)
|
357
|
+
assigns[:leads].should == [ @lead ]
|
358
|
+
end
|
359
|
+
|
360
|
+
it "should reload lead campaign if called from campaign landing page" do
|
361
|
+
@campaign = FactoryGirl.create(:campaign)
|
362
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :campaign => @campaign)
|
363
|
+
|
364
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
365
|
+
xhr :put, :create, :lead => { :first_name => "Billy", :last_name => "Bones"}, :campaign => @campaign.id
|
366
|
+
assigns[:campaign].should == @campaign
|
367
|
+
end
|
368
|
+
|
369
|
+
end
|
370
|
+
|
371
|
+
describe "with invalid params" do
|
372
|
+
|
373
|
+
it "should expose a newly created but unsaved lead as @lead and still render [create] template" do
|
374
|
+
@lead = FactoryGirl.build(:lead, :user => @current_user, :first_name => nil, :campaign => nil)
|
375
|
+
Lead.stub!(:new).and_return(@lead)
|
376
|
+
@users = [ FactoryGirl.create(:user) ]
|
377
|
+
@campaigns = [ FactoryGirl.create(:campaign, :user => @current_user) ]
|
378
|
+
|
379
|
+
xhr :post, :create, :lead => { :first_name => nil }, :users => nil
|
380
|
+
assigns(:lead).should == @lead
|
381
|
+
assigns(:users).should == @users
|
382
|
+
assigns(:campaigns).should == @campaigns
|
383
|
+
assigns[:lead_status_total].should == nil
|
384
|
+
response.should render_template("leads/create")
|
385
|
+
end
|
386
|
+
|
387
|
+
end
|
388
|
+
|
389
|
+
end
|
390
|
+
|
391
|
+
# PUT /leads/1
|
392
|
+
# PUT /leads/1.xml
|
393
|
+
#----------------------------------------------------------------------------
|
394
|
+
describe "responding to PUT update" do
|
395
|
+
|
396
|
+
describe "with valid params" do
|
397
|
+
|
398
|
+
it "should update the requested lead, expose it as @lead, and render [update] template" do
|
399
|
+
@lead = FactoryGirl.create(:lead, :first_name => "Billy", :user => @current_user)
|
400
|
+
|
401
|
+
xhr :put, :update, :id => @lead.id, :lead => { :first_name => "Bones" }
|
402
|
+
@lead.reload.first_name.should == "Bones"
|
403
|
+
assigns[:lead].should == @lead
|
404
|
+
assigns[:lead_status_total].should == nil
|
405
|
+
response.should render_template("leads/update")
|
406
|
+
end
|
407
|
+
|
408
|
+
it "should update lead status" do
|
409
|
+
@lead = FactoryGirl.create(:lead, :status => "new", :user => @current_user)
|
410
|
+
|
411
|
+
xhr :put, :update, :id => @lead.id, :lead => { :status => "rejected" }
|
412
|
+
@lead.reload.status.should == "rejected"
|
413
|
+
end
|
414
|
+
|
415
|
+
it "should update lead source" do
|
416
|
+
@lead = FactoryGirl.create(:lead, :source => "campaign", :user => @current_user)
|
417
|
+
|
418
|
+
xhr :put, :update, :id => @lead.id, :lead => { :source => "cald_call" }
|
419
|
+
@lead.reload.source.should == "cald_call"
|
420
|
+
end
|
421
|
+
|
422
|
+
it "should update lead campaign" do
|
423
|
+
@campaigns = { :old => FactoryGirl.create(:campaign), :new => FactoryGirl.create(:campaign) }
|
424
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaigns[:old])
|
425
|
+
|
426
|
+
xhr :put, :update, :id => @lead.id, :lead => { :campaign_id => @campaigns[:new].id }
|
427
|
+
@lead.reload.campaign.should == @campaigns[:new]
|
428
|
+
end
|
429
|
+
|
430
|
+
it "should decrement campaign leads count if campaign has been removed" do
|
431
|
+
@campaign = FactoryGirl.create(:campaign)
|
432
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaign)
|
433
|
+
@count = @campaign.reload.leads_count
|
434
|
+
|
435
|
+
xhr :put, :update, :id => @lead, :lead => { :campaign_id => nil }
|
436
|
+
@lead.reload.campaign.should == nil
|
437
|
+
@campaign.reload.leads_count.should == @count - 1
|
438
|
+
end
|
439
|
+
|
440
|
+
it "should increment campaign leads count if campaign has been assigned" do
|
441
|
+
@campaign = FactoryGirl.create(:campaign)
|
442
|
+
@lead = FactoryGirl.create(:lead, :campaign => nil)
|
443
|
+
@count = @campaign.leads_count
|
444
|
+
|
445
|
+
xhr :put, :update, :id => @lead, :lead => { :campaign_id => @campaign.id }
|
446
|
+
@lead.reload.campaign.should == @campaign
|
447
|
+
@campaign.reload.leads_count.should == @count + 1
|
448
|
+
end
|
449
|
+
|
450
|
+
it "should update both campaign leads counts if reassigned to a new campaign" do
|
451
|
+
@campaigns = { :old => FactoryGirl.create(:campaign), :new => FactoryGirl.create(:campaign) }
|
452
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaigns[:old])
|
453
|
+
@counts = { :old => @campaigns[:old].reload.leads_count, :new => @campaigns[:new].leads_count }
|
454
|
+
|
455
|
+
xhr :put, :update, :id => @lead, :lead => { :campaign_id => @campaigns[:new].id }
|
456
|
+
@lead.reload.campaign.should == @campaigns[:new]
|
457
|
+
@campaigns[:old].reload.leads_count.should == @counts[:old] - 1
|
458
|
+
@campaigns[:new].reload.leads_count.should == @counts[:new] + 1
|
459
|
+
end
|
460
|
+
|
461
|
+
it "should update shared permissions for the campaign" do
|
462
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
463
|
+
he = FactoryGirl.create(:user, :id => 7)
|
464
|
+
she = FactoryGirl.create(:user, :id => 8)
|
465
|
+
|
466
|
+
xhr :put, :update, :id => @lead.id, :lead => { :access => "Shared" }, :users => %w(7 8)
|
467
|
+
@lead.permissions.map(&:user_id).sort.should == [ 7, 8 ]
|
468
|
+
end
|
469
|
+
|
470
|
+
it "should get the data for leads sidebar when called from leads index" do
|
471
|
+
@lead = FactoryGirl.create(:lead)
|
472
|
+
|
473
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
474
|
+
xhr :put, :update, :id => @lead.id, :lead => { :first_name => "Billy" }
|
475
|
+
assigns[:lead_status_total].should_not be_nil
|
476
|
+
assigns[:lead_status_total].should be_an_instance_of(HashWithIndifferentAccess)
|
477
|
+
end
|
478
|
+
|
479
|
+
it "should reload lead campaign if called from campaign landing page" do
|
480
|
+
@campaign = FactoryGirl.create(:campaign)
|
481
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaign)
|
482
|
+
|
483
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
484
|
+
xhr :put, :update, :id => @lead.id, :lead => { :first_name => "Hello" }
|
485
|
+
assigns[:campaign].should == @campaign
|
486
|
+
end
|
487
|
+
|
488
|
+
describe "lead got deleted or otherwise unavailable" do
|
489
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
490
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
491
|
+
@lead.destroy
|
492
|
+
|
493
|
+
xhr :put, :update, :id => @lead.id
|
494
|
+
flash[:warning].should_not == nil
|
495
|
+
response.body.should == "window.location.reload();"
|
496
|
+
end
|
497
|
+
|
498
|
+
it "should reload current page with the flash message if the lead is protected" do
|
499
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
500
|
+
|
501
|
+
xhr :put, :update, :id => @private.id
|
502
|
+
flash[:warning].should_not == nil
|
503
|
+
response.body.should == "window.location.reload();"
|
504
|
+
end
|
505
|
+
end
|
506
|
+
end
|
507
|
+
|
508
|
+
describe "with invalid params" do
|
509
|
+
|
510
|
+
it "should not update the lead, but still expose it as @lead and render [update] template" do
|
511
|
+
@lead = FactoryGirl.create(:lead, :id => 42, :user => @current_user, :campaign => nil)
|
512
|
+
@users = [ FactoryGirl.create(:user) ]
|
513
|
+
@campaigns = [ FactoryGirl.create(:campaign, :user => @current_user) ]
|
514
|
+
|
515
|
+
xhr :put, :update, :id => 42, :lead => { :first_name => nil }
|
516
|
+
assigns[:lead].should == @lead
|
517
|
+
assigns[:users].should == @users
|
518
|
+
assigns[:campaigns].should == @campaigns
|
519
|
+
response.should render_template("leads/update")
|
520
|
+
end
|
521
|
+
|
522
|
+
end
|
523
|
+
|
524
|
+
end
|
525
|
+
|
526
|
+
# DELETE /leads/1
|
527
|
+
# DELETE /leads/1.xml AJAX and HTML
|
528
|
+
#----------------------------------------------------------------------------
|
529
|
+
describe "responding to DELETE destroy" do
|
530
|
+
|
531
|
+
before(:each) do
|
532
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
533
|
+
end
|
534
|
+
|
535
|
+
describe "AJAX request" do
|
536
|
+
it "should destroy the requested lead and render [destroy] template" do
|
537
|
+
xhr :delete, :destroy, :id => @lead.id
|
538
|
+
|
539
|
+
assigns[:leads].should == nil # @lead got deleted
|
540
|
+
lambda { Lead.find(@lead) }.should raise_error(ActiveRecord::RecordNotFound)
|
541
|
+
response.should render_template("leads/destroy")
|
542
|
+
end
|
543
|
+
|
544
|
+
describe "when called from Leads index page" do
|
545
|
+
before(:each) do
|
546
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
547
|
+
end
|
548
|
+
|
549
|
+
it "should get data for the sidebar" do
|
550
|
+
@another_lead = FactoryGirl.create(:lead, :user => @current_user)
|
551
|
+
|
552
|
+
xhr :delete, :destroy, :id => @lead.id
|
553
|
+
assigns[:leads].should == [ @another_lead ] # @lead got deleted
|
554
|
+
assigns[:lead_status_total].should_not be_nil
|
555
|
+
assigns[:lead_status_total].should be_an_instance_of(HashWithIndifferentAccess)
|
556
|
+
response.should render_template("leads/destroy")
|
557
|
+
end
|
558
|
+
|
559
|
+
it "should try previous page and render index action if current page has no leads" do
|
560
|
+
session[:leads_current_page] = 42
|
561
|
+
|
562
|
+
xhr :delete, :destroy, :id => @lead.id
|
563
|
+
session[:leads_current_page].should == 41
|
564
|
+
response.should render_template("leads/index")
|
565
|
+
end
|
566
|
+
|
567
|
+
it "should render index action when deleting last lead" do
|
568
|
+
session[:leads_current_page] = 1
|
569
|
+
|
570
|
+
xhr :delete, :destroy, :id => @lead.id
|
571
|
+
session[:leads_current_page].should == 1
|
572
|
+
response.should render_template("leads/index")
|
573
|
+
end
|
574
|
+
end
|
575
|
+
|
576
|
+
describe "when called from campaign landing page" do
|
577
|
+
before(:each) do
|
578
|
+
@campaign = FactoryGirl.create(:campaign)
|
579
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user, :campaign => @campaign)
|
580
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
581
|
+
end
|
582
|
+
|
583
|
+
it "should reset current page to 1" do
|
584
|
+
xhr :delete, :destroy, :id => @lead.id
|
585
|
+
session[:leads_current_page].should == 1
|
586
|
+
response.should render_template("leads/destroy")
|
587
|
+
end
|
588
|
+
|
589
|
+
it "should reload campaiign to be able to refresh its summary" do
|
590
|
+
xhr :delete, :destroy, :id => @lead.id
|
591
|
+
assigns[:campaign].should == @campaign
|
592
|
+
response.should render_template("leads/destroy")
|
593
|
+
end
|
594
|
+
end
|
595
|
+
|
596
|
+
describe "lead got deleted or otherwise unavailable" do
|
597
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
598
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
599
|
+
@lead.destroy
|
600
|
+
|
601
|
+
xhr :delete, :destroy, :id => @lead.id
|
602
|
+
flash[:warning].should_not == nil
|
603
|
+
response.body.should == "window.location.reload();"
|
604
|
+
end
|
605
|
+
|
606
|
+
it "should reload current page with the flash message if the lead is protected" do
|
607
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
608
|
+
|
609
|
+
xhr :delete, :destroy, :id => @private.id
|
610
|
+
flash[:warning].should_not == nil
|
611
|
+
response.body.should == "window.location.reload();"
|
612
|
+
end
|
613
|
+
end
|
614
|
+
end
|
615
|
+
|
616
|
+
describe "HTML request" do
|
617
|
+
it "should redirect to Leads index when a lead gets deleted from its landing page" do
|
618
|
+
delete :destroy, :id => @lead.id
|
619
|
+
flash[:notice].should_not == nil
|
620
|
+
session[:leads_current_page].should == 1
|
621
|
+
response.should redirect_to(leads_path)
|
622
|
+
end
|
623
|
+
|
624
|
+
it "should redirect to lead index with the flash message is the lead got deleted" do
|
625
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
626
|
+
@lead.destroy
|
627
|
+
|
628
|
+
delete :destroy, :id => @lead.id
|
629
|
+
flash[:warning].should_not == nil
|
630
|
+
response.should redirect_to(leads_path)
|
631
|
+
end
|
632
|
+
|
633
|
+
it "should redirect to lead index with the flash message if the lead is protected" do
|
634
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
635
|
+
|
636
|
+
delete :destroy, :id => @private.id
|
637
|
+
flash[:warning].should_not == nil
|
638
|
+
response.should redirect_to(leads_path)
|
639
|
+
end
|
640
|
+
end
|
641
|
+
end
|
642
|
+
|
643
|
+
# GET /leads/1/convert
|
644
|
+
# GET /leads/1/convert.xml AJAX
|
645
|
+
#----------------------------------------------------------------------------
|
646
|
+
describe "responding to GET convert" do
|
647
|
+
|
648
|
+
it "should should collect necessary data and render [convert] template" do
|
649
|
+
@campaign = FactoryGirl.create(:campaign, :user => @current_user)
|
650
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user, :campaign => @campaign, :source => "cold_call")
|
651
|
+
@users = [ FactoryGirl.create(:user) ]
|
652
|
+
@accounts = [ FactoryGirl.create(:account, :user => @current_user) ]
|
653
|
+
@account = Account.new(:user => @current_user, :name => @lead.company, :access => "Lead")
|
654
|
+
@opportunity = Opportunity.new(:user => @current_user, :access => "Lead", :stage => "prospecting", :campaign => @lead.campaign, :source => @lead.source)
|
655
|
+
|
656
|
+
xhr :get, :convert, :id => @lead.id
|
657
|
+
assigns[:lead].should == @lead
|
658
|
+
assigns[:users].should == @users
|
659
|
+
assigns[:accounts].should == @accounts
|
660
|
+
assigns[:account].attributes.should == @account.attributes
|
661
|
+
assigns[:opportunity].attributes.should == @opportunity.attributes
|
662
|
+
assigns[:opportunity].campaign.should == @opportunity.campaign
|
663
|
+
response.should render_template("leads/convert")
|
664
|
+
end
|
665
|
+
|
666
|
+
describe "(lead got deleted or is otherwise unavailable)" do
|
667
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
668
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
669
|
+
@lead.destroy
|
670
|
+
|
671
|
+
xhr :get, :convert, :id => @lead.id
|
672
|
+
flash[:warning].should_not == nil
|
673
|
+
response.body.should == "window.location.reload();"
|
674
|
+
end
|
675
|
+
|
676
|
+
it "should reload current page with the flash message if the lead is protected" do
|
677
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
678
|
+
|
679
|
+
xhr :get, :convert, :id => @private.id
|
680
|
+
flash[:warning].should_not == nil
|
681
|
+
response.body.should == "window.location.reload();"
|
682
|
+
end
|
683
|
+
end
|
684
|
+
|
685
|
+
describe "(previous lead got deleted or is otherwise unavailable)" do
|
686
|
+
before(:each) do
|
687
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
688
|
+
@previous = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user))
|
689
|
+
end
|
690
|
+
|
691
|
+
it "should notify the view if previous lead got deleted" do
|
692
|
+
@previous.destroy
|
693
|
+
|
694
|
+
xhr :get, :convert, :id => @lead.id, :previous => @previous.id
|
695
|
+
flash[:warning].should == nil # no warning, just silently remove the div
|
696
|
+
assigns[:previous].should == @previous.id
|
697
|
+
response.should render_template("leads/convert")
|
698
|
+
end
|
699
|
+
|
700
|
+
it "should notify the view if previous lead got protected" do
|
701
|
+
@previous.update_attribute(:access, "Private")
|
702
|
+
|
703
|
+
xhr :get, :convert, :id => @lead.id, :previous => @previous.id
|
704
|
+
flash[:warning].should == nil
|
705
|
+
assigns[:previous].should == @previous.id
|
706
|
+
response.should render_template("leads/convert")
|
707
|
+
end
|
708
|
+
end
|
709
|
+
end
|
710
|
+
|
711
|
+
# PUT /leads/1/promote
|
712
|
+
# PUT /leads/1/promote.xml AJAX
|
713
|
+
#----------------------------------------------------------------------------
|
714
|
+
describe "responding to PUT promote" do
|
715
|
+
|
716
|
+
it "on success: should change lead's status to [converted] and render [promote] template" do
|
717
|
+
@lead = FactoryGirl.create(:lead, :id => 42, :user => @current_user, :campaign => nil)
|
718
|
+
@users = [ FactoryGirl.create(:user) ]
|
719
|
+
@account = FactoryGirl.create(:account, :id => 123, :user => @current_user)
|
720
|
+
@opportunity = FactoryGirl.build(:opportunity, :user => @current_user, :campaign => @lead.campaign,
|
721
|
+
:account => @account)
|
722
|
+
Opportunity.stub!(:new).and_return(@opportunity)
|
723
|
+
@contact = FactoryGirl.build(:contact, :user => @current_user, :lead => @lead)
|
724
|
+
Contact.stub!(:new).and_return(@contact)
|
725
|
+
|
726
|
+
xhr :put, :promote, :id => 42, :account => { :id => 123 }, :opportunity => { :name => "Hello" }
|
727
|
+
@lead.reload.status.should == "converted"
|
728
|
+
assigns[:lead].should == @lead
|
729
|
+
assigns[:users].should == @users
|
730
|
+
assigns[:account].should == @account
|
731
|
+
assigns[:accounts].should == [ @account ]
|
732
|
+
assigns[:opportunity].should == @opportunity
|
733
|
+
assigns[:contact].should == @contact
|
734
|
+
assigns[:stage].should be_instance_of(Array)
|
735
|
+
response.should render_template("leads/promote")
|
736
|
+
end
|
737
|
+
|
738
|
+
it "should copy lead permissions to newly created account and opportunity when asked so" do
|
739
|
+
he = FactoryGirl.create(:user, :id => 7)
|
740
|
+
she = FactoryGirl.create(:user, :id => 8)
|
741
|
+
@lead = FactoryGirl.build(:lead, :access => "Shared")
|
742
|
+
@lead.permissions << FactoryGirl.build(:permission, :user => he, :asset => @lead)
|
743
|
+
@lead.permissions << FactoryGirl.build(:permission, :user => she, :asset => @lead)
|
744
|
+
@lead.save
|
745
|
+
@account = FactoryGirl.build(:account, :user => @current_user, :access => "Shared")
|
746
|
+
@account.permissions << FactoryGirl.create(:permission, :user => he, :asset => @account)
|
747
|
+
@account.permissions << FactoryGirl.create(:permission, :user => she, :asset => @account)
|
748
|
+
@account.stub!(:new).and_return(@account)
|
749
|
+
@opportunity = FactoryGirl.build(:opportunity, :user => @current_user, :access => "Shared")
|
750
|
+
@opportunity.permissions << FactoryGirl.create(:permission, :user => he, :asset => @opportunity)
|
751
|
+
@opportunity.permissions << FactoryGirl.create(:permission, :user => she, :asset => @opportunity)
|
752
|
+
@opportunity.stub!(:new).and_return(@opportunity)
|
753
|
+
|
754
|
+
xhr :put, :promote, :id => @lead.id, :access => "Lead", :account => { :name => "Hello", :access => "Lead", :user_id => @current_user.id }, :opportunity => { :name => "World", :access => "Lead", :user_id => @current_user.id }
|
755
|
+
@account.access.should == "Shared"
|
756
|
+
@account.permissions.map(&:user_id).sort.should == [ 7, 8 ]
|
757
|
+
@account.permissions.map(&:asset_id).should == [ @account.id, @account.id ]
|
758
|
+
@account.permissions.map(&:asset_type).should == %w(Account Account)
|
759
|
+
@opportunity.access.should == "Shared"
|
760
|
+
@opportunity.permissions.map(&:user_id).sort.should == [ 7, 8 ]
|
761
|
+
@opportunity.permissions.map(&:asset_id).should == [ @opportunity.id, @opportunity.id ]
|
762
|
+
@opportunity.permissions.map(&:asset_type).should == %w(Opportunity Opportunity)
|
763
|
+
end
|
764
|
+
|
765
|
+
it "should assign lead's campaign to the newly created opportunity" do
|
766
|
+
@campaign = FactoryGirl.create(:campaign)
|
767
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user, :campaign => @campaign)
|
768
|
+
|
769
|
+
xhr :put, :promote, :id => @lead.id, :account => { :name => "Hello" }, :opportunity => { :name => "Hello", :campaign_id => @campaign.id }
|
770
|
+
assigns[:opportunity].campaign.should == @campaign
|
771
|
+
end
|
772
|
+
|
773
|
+
it "should assign lead's source to the newly created opportunity" do
|
774
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user, :source => "cold_call")
|
775
|
+
|
776
|
+
xhr :put, :promote, :id => @lead.id, :account => { :name => "Hello" }, :opportunity => { :name => "Hello", :source => @lead.source }
|
777
|
+
assigns[:opportunity].source.should == @lead.source
|
778
|
+
end
|
779
|
+
|
780
|
+
it "should get the data for leads sidebar when called from leads index" do
|
781
|
+
@lead = FactoryGirl.create(:lead)
|
782
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
783
|
+
|
784
|
+
xhr :put, :promote, :id => @lead.id, :account => { :name => "Hello" }, :opportunity => {}
|
785
|
+
assigns[:lead_status_total].should_not be_nil
|
786
|
+
assigns[:lead_status_total].should be_an_instance_of(HashWithIndifferentAccess)
|
787
|
+
end
|
788
|
+
|
789
|
+
it "should reload lead campaign if called from campaign landing page" do
|
790
|
+
@campaign = FactoryGirl.create(:campaign)
|
791
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaign)
|
792
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
793
|
+
|
794
|
+
xhr :put, :promote, :id => @lead.id, :account => { :name => "Hello" }, :opportunity => {}
|
795
|
+
assigns[:campaign].should == @campaign
|
796
|
+
end
|
797
|
+
|
798
|
+
it "on failure: should not change lead's status and still render [promote] template" do
|
799
|
+
@lead = FactoryGirl.create(:lead, :id => 42, :user => @current_user, :status => "new")
|
800
|
+
@users = [ FactoryGirl.create(:user) ]
|
801
|
+
@account = FactoryGirl.create(:account, :id => 123, :user => @current_user)
|
802
|
+
@contact = FactoryGirl.build(:contact, :first_name => nil) # make it fail
|
803
|
+
Contact.stub!(:new).and_return(@contact)
|
804
|
+
|
805
|
+
xhr :put, :promote, :id => 42, :account => { :id => 123 }
|
806
|
+
@lead.reload.status.should == "new"
|
807
|
+
response.should render_template("leads/promote")
|
808
|
+
end
|
809
|
+
|
810
|
+
describe "lead got deleted or otherwise unavailable" do
|
811
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
812
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
813
|
+
@lead.destroy
|
814
|
+
|
815
|
+
xhr :put, :promote, :id => @lead.id
|
816
|
+
flash[:warning].should_not == nil
|
817
|
+
response.body.should == "window.location.reload();"
|
818
|
+
end
|
819
|
+
|
820
|
+
it "should reload current page with the flash message if the lead is protected" do
|
821
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
822
|
+
|
823
|
+
xhr :put, :promote, :id => @private.id
|
824
|
+
flash[:warning].should_not == nil
|
825
|
+
response.body.should == "window.location.reload();"
|
826
|
+
end
|
827
|
+
end
|
828
|
+
end
|
829
|
+
|
830
|
+
# PUT /leads/1/reject
|
831
|
+
# PUT /leads/1/reject.xml AJAX and HTML
|
832
|
+
#----------------------------------------------------------------------------
|
833
|
+
describe "responding to PUT reject" do
|
834
|
+
|
835
|
+
before(:each) do
|
836
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user, :status => "new")
|
837
|
+
end
|
838
|
+
|
839
|
+
describe "AJAX request" do
|
840
|
+
it "should reject the requested lead and render [reject] template" do
|
841
|
+
xhr :put, :reject, :id => @lead.id
|
842
|
+
|
843
|
+
assigns[:lead].should == @lead.reload
|
844
|
+
@lead.status.should == "rejected"
|
845
|
+
response.should render_template("leads/reject")
|
846
|
+
end
|
847
|
+
|
848
|
+
it "should get the data for leads sidebar when called from leads index" do
|
849
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
850
|
+
xhr :put, :reject, :id => @lead.id
|
851
|
+
assigns[:lead_status_total].should_not be_nil
|
852
|
+
assigns[:lead_status_total].should be_an_instance_of(HashWithIndifferentAccess)
|
853
|
+
end
|
854
|
+
|
855
|
+
it "should reload lead campaign if called from campaign landing page" do
|
856
|
+
@campaign = FactoryGirl.create(:campaign)
|
857
|
+
@lead = FactoryGirl.create(:lead, :campaign => @campaign)
|
858
|
+
|
859
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
860
|
+
xhr :put, :reject, :id => @lead.id
|
861
|
+
assigns[:campaign].should == @campaign
|
862
|
+
end
|
863
|
+
|
864
|
+
describe "lead got deleted or otherwise unavailable" do
|
865
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
866
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
867
|
+
@lead.destroy
|
868
|
+
|
869
|
+
xhr :put, :reject, :id => @lead.id
|
870
|
+
flash[:warning].should_not == nil
|
871
|
+
response.body.should == "window.location.reload();"
|
872
|
+
end
|
873
|
+
|
874
|
+
it "should reload current page with the flash message if the lead is protected" do
|
875
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
876
|
+
|
877
|
+
xhr :put, :reject, :id => @private.id
|
878
|
+
flash[:warning].should_not == nil
|
879
|
+
response.body.should == "window.location.reload();"
|
880
|
+
end
|
881
|
+
end
|
882
|
+
end
|
883
|
+
|
884
|
+
describe "HTML request" do
|
885
|
+
it "should redirect to Leads index when a lead gets rejected from its landing page" do
|
886
|
+
put :reject, :id => @lead.id
|
887
|
+
|
888
|
+
assigns[:lead].should == @lead.reload
|
889
|
+
@lead.status.should == "rejected"
|
890
|
+
flash[:notice].should_not == nil
|
891
|
+
response.should redirect_to(leads_path)
|
892
|
+
end
|
893
|
+
|
894
|
+
describe "lead got deleted or otherwise unavailable" do
|
895
|
+
it "should redirect to lead index if the lead got deleted" do
|
896
|
+
@lead = FactoryGirl.create(:lead, :user => @current_user)
|
897
|
+
@lead.destroy
|
898
|
+
|
899
|
+
put :reject, :id => @lead.id
|
900
|
+
flash[:warning].should_not == nil
|
901
|
+
response.should redirect_to(leads_path)
|
902
|
+
end
|
903
|
+
|
904
|
+
it "should redirect to lead index if the lead is protected" do
|
905
|
+
@private = FactoryGirl.create(:lead, :user => FactoryGirl.create(:user), :access => "Private")
|
906
|
+
|
907
|
+
put :reject, :id => @private.id
|
908
|
+
flash[:warning].should_not == nil
|
909
|
+
response.should redirect_to(leads_path)
|
910
|
+
end
|
911
|
+
end
|
912
|
+
end
|
913
|
+
end
|
914
|
+
|
915
|
+
# PUT /leads/1/attach
|
916
|
+
# PUT /leads/1/attach.xml AJAX
|
917
|
+
#----------------------------------------------------------------------------
|
918
|
+
describe "responding to PUT attach" do
|
919
|
+
describe "tasks" do
|
920
|
+
before do
|
921
|
+
@model = FactoryGirl.create(:lead)
|
922
|
+
@attachment = FactoryGirl.create(:task, :asset => nil)
|
923
|
+
end
|
924
|
+
it_should_behave_like("attach")
|
925
|
+
end
|
926
|
+
end
|
927
|
+
|
928
|
+
# PUT /leads/1/attach
|
929
|
+
# PUT /leads/1/attach.xml AJAX
|
930
|
+
#----------------------------------------------------------------------------
|
931
|
+
describe "responding to PUT attach" do
|
932
|
+
describe "tasks" do
|
933
|
+
before do
|
934
|
+
@model = FactoryGirl.create(:lead)
|
935
|
+
@attachment = FactoryGirl.create(:task, :asset => nil)
|
936
|
+
end
|
937
|
+
it_should_behave_like("attach")
|
938
|
+
end
|
939
|
+
end
|
940
|
+
|
941
|
+
# POST /leads/1/discard
|
942
|
+
# POST /leads/1/discard.xml AJAX
|
943
|
+
#----------------------------------------------------------------------------
|
944
|
+
describe "responding to POST discard" do
|
945
|
+
before(:each) do
|
946
|
+
@attachment = FactoryGirl.create(:task, :assigned_to => @current_user)
|
947
|
+
@model = FactoryGirl.create(:lead)
|
948
|
+
@model.tasks << @attachment
|
949
|
+
end
|
950
|
+
|
951
|
+
it_should_behave_like("discard")
|
952
|
+
end
|
953
|
+
|
954
|
+
# POST /leads/auto_complete/query AJAX
|
955
|
+
#----------------------------------------------------------------------------
|
956
|
+
describe "responding to POST auto_complete" do
|
957
|
+
before(:each) do
|
958
|
+
@auto_complete_matches = [ FactoryGirl.create(:lead, :first_name => "Hello", :last_name => "World", :user => @current_user) ]
|
959
|
+
end
|
960
|
+
|
961
|
+
it_should_behave_like("auto complete")
|
962
|
+
end
|
963
|
+
|
964
|
+
# GET /leads/options AJAX
|
965
|
+
#----------------------------------------------------------------------------
|
966
|
+
describe "responding to GET options" do
|
967
|
+
it "should set current user preferences when showing options" do
|
968
|
+
@per_page = FactoryGirl.create(:preference, :user => @current_user, :name => "leads_per_page", :value => Base64.encode64(Marshal.dump(42)))
|
969
|
+
@outline = FactoryGirl.create(:preference, :user => @current_user, :name => "leads_outline", :value => Base64.encode64(Marshal.dump("option_long")))
|
970
|
+
@sort_by = FactoryGirl.create(:preference, :user => @current_user, :name => "leads_sort_by", :value => Base64.encode64(Marshal.dump("leads.first_name ASC")))
|
971
|
+
@naming = FactoryGirl.create(:preference, :user => @current_user, :name => "leads_naming", :value => Base64.encode64(Marshal.dump("option_after")))
|
972
|
+
|
973
|
+
xhr :get, :options
|
974
|
+
assigns[:per_page].should == 42
|
975
|
+
assigns[:outline].should == "option_long"
|
976
|
+
assigns[:sort_by].should == "leads.first_name ASC"
|
977
|
+
assigns[:naming].should == "option_after"
|
978
|
+
end
|
979
|
+
|
980
|
+
it "should not assign instance variables when hiding options" do
|
981
|
+
xhr :get, :options, :cancel => "true"
|
982
|
+
assigns[:per_page].should == nil
|
983
|
+
assigns[:outline].should == nil
|
984
|
+
assigns[:sort_by].should == nil
|
985
|
+
assigns[:naming].should == nil
|
986
|
+
end
|
987
|
+
end
|
988
|
+
|
989
|
+
# POST /leads/redraw AJAX
|
990
|
+
#----------------------------------------------------------------------------
|
991
|
+
describe "responding to POST redraw" do
|
992
|
+
it "should save user selected lead preference" do
|
993
|
+
xhr :post, :redraw, :per_page => 42, :outline => "long", :sort_by => "first_name", :naming => "after"
|
994
|
+
@current_user.preference[:leads_per_page].should == "42"
|
995
|
+
@current_user.preference[:leads_outline].should == "long"
|
996
|
+
@current_user.preference[:leads_sort_by].should == "leads.first_name ASC"
|
997
|
+
@current_user.preference[:leads_naming].should == "after"
|
998
|
+
end
|
999
|
+
|
1000
|
+
it "should set similar options for Contacts" do
|
1001
|
+
xhr :post, :redraw, :sort_by => "first_name", :naming => "after"
|
1002
|
+
@current_user.pref[:contacts_sort_by].should == "contacts.first_name ASC"
|
1003
|
+
@current_user.pref[:contacts_naming].should == "after"
|
1004
|
+
end
|
1005
|
+
|
1006
|
+
it "should reset current page to 1" do
|
1007
|
+
xhr :post, :redraw, :per_page => 42, :outline => "long", :sort_by => "first_name", :naming => "after"
|
1008
|
+
session[:leads_current_page].should == 1
|
1009
|
+
end
|
1010
|
+
|
1011
|
+
it "should select @leads and render [index] template" do
|
1012
|
+
@leads = [
|
1013
|
+
FactoryGirl.create(:lead, :first_name => "Alice", :user => @current_user),
|
1014
|
+
FactoryGirl.create(:lead, :first_name => "Bobby", :user => @current_user)
|
1015
|
+
]
|
1016
|
+
|
1017
|
+
xhr :post, :redraw, :per_page => 1, :sort_by => "first_name"
|
1018
|
+
assigns(:leads).should == [ @leads.first ]
|
1019
|
+
response.should render_template("leads/index")
|
1020
|
+
end
|
1021
|
+
end
|
1022
|
+
|
1023
|
+
# POST /leads/filter AJAX
|
1024
|
+
#----------------------------------------------------------------------------
|
1025
|
+
describe "responding to POST filter" do
|
1026
|
+
|
1027
|
+
it "should filter out leads as @leads and render :index action" do
|
1028
|
+
session[:filter_by_lead_status] = "contacted,rejected"
|
1029
|
+
|
1030
|
+
@leads = [ FactoryGirl.create(:lead, :user => @current_user, :status => "new") ]
|
1031
|
+
xhr :post, :filter, :status => "new"
|
1032
|
+
assigns[:leads].should == @leads
|
1033
|
+
response.should be_a_success
|
1034
|
+
response.should render_template("leads/index")
|
1035
|
+
end
|
1036
|
+
|
1037
|
+
it "should reset current page to 1" do
|
1038
|
+
@leads = []
|
1039
|
+
xhr :post, :filter, :status => "new"
|
1040
|
+
|
1041
|
+
session[:leads_current_page].should == 1
|
1042
|
+
end
|
1043
|
+
|
1044
|
+
end
|
1045
|
+
|
1046
|
+
end
|
1047
|
+
|