reduced_fat_crm 0.14.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +36 -0
- data/.rubocop.yml +5 -0
- data/.teatro.yml +3 -0
- data/.travis.yml +41 -0
- data/CHANGELOG +1923 -0
- data/CONTRIBUTORS +91 -0
- data/Capfile +25 -0
- data/Dockerfile +18 -0
- data/Gemfile +88 -0
- data/Gemfile.lock +430 -0
- data/Guardfile +24 -0
- data/MIT-LICENSE +21 -0
- data/Procfile +1 -0
- data/README.md +153 -0
- data/Rakefile +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.gif +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/notifications.png +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/fields.js.coffee +72 -0
- data/app/assets/javascripts/application.js.erb +41 -0
- data/app/assets/javascripts/crm.js.coffee +428 -0
- data/app/assets/javascripts/crm_chosen.js.coffee +29 -0
- data/app/assets/javascripts/crm_classes.js.coffee +215 -0
- data/app/assets/javascripts/crm_comments.js.coffee +28 -0
- data/app/assets/javascripts/crm_loginout.js.coffee +27 -0
- data/app/assets/javascripts/crm_select2.js.coffee +31 -0
- data/app/assets/javascripts/crm_sortable.js.coffee +32 -0
- data/app/assets/javascripts/crm_tags.js.coffee +22 -0
- data/app/assets/javascripts/crm_textarea_autocomplete.js.coffee +33 -0
- data/app/assets/javascripts/datepicker.js.coffee +20 -0
- data/app/assets/javascripts/format_buttons.js.coffee +52 -0
- data/app/assets/javascripts/lists.js.coffee +54 -0
- data/app/assets/javascripts/pagination.js.coffee +16 -0
- data/app/assets/javascripts/search.js.coffee +55 -0
- data/app/assets/javascripts/timeago.js.coffee +21 -0
- data/app/assets/stylesheets/about.css.scss +10 -0
- data/app/assets/stylesheets/admin/fields.css.scss +17 -0
- data/app/assets/stylesheets/advanced_search.css.scss +108 -0
- data/app/assets/stylesheets/application.css.erb +37 -0
- data/app/assets/stylesheets/base.scss +78 -0
- data/app/assets/stylesheets/common.scss +765 -0
- data/app/assets/stylesheets/ffcrm_chosen.scss +69 -0
- data/app/assets/stylesheets/fields.scss +27 -0
- data/app/assets/stylesheets/format_buttons.css.scss +40 -0
- data/app/assets/stylesheets/groups.css.scss +8 -0
- data/app/assets/stylesheets/header.scss +121 -0
- data/app/assets/stylesheets/index_headers.css.scss +5 -0
- data/app/assets/stylesheets/lists.css.scss +26 -0
- data/app/assets/stylesheets/print.css.scss +77 -0
- data/app/assets/stylesheets/rails.scss +133 -0
- data/app/assets/stylesheets/safari.scss +11 -0
- data/app/assets/stylesheets/textarea_autocomplete.scss +47 -0
- data/app/controllers/admin/application_controller.rb +30 -0
- data/app/controllers/admin/field_groups_controller.rb +83 -0
- data/app/controllers/admin/fields_controller.rb +118 -0
- data/app/controllers/admin/groups_controller.rb +65 -0
- data/app/controllers/admin/plugins_controller.rb +17 -0
- data/app/controllers/admin/settings_controller.rb +14 -0
- data/app/controllers/admin/tags_controller.rb +71 -0
- data/app/controllers/admin/users_controller.rb +144 -0
- data/app/controllers/application_controller.rb +276 -0
- data/app/controllers/authentications_controller.rb +51 -0
- data/app/controllers/comments_controller.rb +93 -0
- data/app/controllers/emails_controller.rb +18 -0
- data/app/controllers/entities/accounts_controller.rb +158 -0
- data/app/controllers/entities/campaigns_controller.rb +197 -0
- data/app/controllers/entities/contacts_controller.rb +178 -0
- data/app/controllers/entities/leads_controller.rb +256 -0
- data/app/controllers/entities/opportunities_controller.rb +222 -0
- data/app/controllers/entities_controller.rb +214 -0
- data/app/controllers/home_controller.rb +156 -0
- data/app/controllers/lists_controller.rb +40 -0
- data/app/controllers/passwords_controller.rb +65 -0
- data/app/controllers/tasks_controller.rb +219 -0
- data/app/controllers/users_controller.rb +159 -0
- data/app/helpers/accounts_helper.rb +110 -0
- data/app/helpers/addresses_helper.rb +17 -0
- data/app/helpers/admin/application_helper.rb +15 -0
- data/app/helpers/admin/field_groups_helper.rb +25 -0
- data/app/helpers/admin/fields_helper.rb +18 -0
- data/app/helpers/admin/plugins_helper.rb +7 -0
- data/app/helpers/admin/settings_helper.rb +7 -0
- data/app/helpers/admin/tags_helper.rb +10 -0
- data/app/helpers/admin/users_helper.rb +51 -0
- data/app/helpers/application_helper.rb +527 -0
- data/app/helpers/authentications_helper.rb +7 -0
- data/app/helpers/campaigns_helper.rb +34 -0
- data/app/helpers/comments_helper.rb +11 -0
- data/app/helpers/contacts_helper.rb +21 -0
- data/app/helpers/emails_helper.rb +7 -0
- data/app/helpers/fields_helper.rb +7 -0
- data/app/helpers/groups_helper.rb +7 -0
- data/app/helpers/home_helper.rb +66 -0
- data/app/helpers/javascript_helper.rb +30 -0
- data/app/helpers/leads_helper.rb +85 -0
- data/app/helpers/lists_helper.rb +7 -0
- data/app/helpers/opportunities_helper.rb +46 -0
- data/app/helpers/passwords_helper.rb +7 -0
- data/app/helpers/remote_link_pagination_helper.rb +8 -0
- data/app/helpers/tags_helper.rb +29 -0
- data/app/helpers/tasks_helper.rb +139 -0
- data/app/helpers/users_helper.rb +35 -0
- data/app/helpers/versions_helper.rb +32 -0
- data/app/inputs/date_pair_input.rb +65 -0
- data/app/inputs/date_time_input.rb +36 -0
- data/app/inputs/datetime_pair_input.rb +25 -0
- data/app/inputs/text_input.rb +12 -0
- data/app/mailers/dropbox_mailer.rb +18 -0
- data/app/mailers/subscription_mailer.rb +30 -0
- data/app/mailers/user_mailer.rb +32 -0
- data/app/models/entities/account.rb +142 -0
- data/app/models/entities/account_contact.rb +28 -0
- data/app/models/entities/account_opportunity.rb +26 -0
- data/app/models/entities/campaign.rb +116 -0
- data/app/models/entities/contact.rb +215 -0
- data/app/models/entities/contact_opportunity.rb +27 -0
- data/app/models/entities/lead.rb +197 -0
- data/app/models/entities/opportunity.rb +190 -0
- data/app/models/fields/core_field.rb +37 -0
- data/app/models/fields/custom_field.rb +148 -0
- data/app/models/fields/custom_field_date_pair.rb +42 -0
- data/app/models/fields/custom_field_datetime_pair.rb +12 -0
- data/app/models/fields/custom_field_pair.rb +42 -0
- data/app/models/fields/field.rb +129 -0
- data/app/models/fields/field_group.rb +63 -0
- data/app/models/list.rb +17 -0
- data/app/models/observers/entity_observer.rb +43 -0
- data/app/models/observers/lead_observer.rb +29 -0
- data/app/models/observers/opportunity_observer.rb +47 -0
- data/app/models/observers/task_observer.rb +31 -0
- data/app/models/polymorphic/address.rb +59 -0
- data/app/models/polymorphic/avatar.rb +55 -0
- data/app/models/polymorphic/comment.rb +73 -0
- data/app/models/polymorphic/email.rb +59 -0
- data/app/models/polymorphic/tag.rb +21 -0
- data/app/models/polymorphic/tagging.rb +8 -0
- data/app/models/polymorphic/task.rb +284 -0
- data/app/models/polymorphic/version.rb +81 -0
- data/app/models/setting.rb +103 -0
- data/app/models/users/ability.rb +59 -0
- data/app/models/users/authentication.rb +54 -0
- data/app/models/users/group.rb +13 -0
- data/app/models/users/permission.rb +27 -0
- data/app/models/users/preference.rb +50 -0
- data/app/models/users/user.rb +184 -0
- data/app/views/accounts/_account.html.haml +5 -0
- data/app/views/accounts/_contact_info.html.haml +37 -0
- data/app/views/accounts/_edit.html.haml +21 -0
- data/app/views/accounts/_index_brief.html.haml +32 -0
- data/app/views/accounts/_index_long.html.haml +42 -0
- data/app/views/accounts/_new.html.haml +21 -0
- data/app/views/accounts/_sidebar_index.html.haml +20 -0
- data/app/views/accounts/_sidebar_show.html.haml +57 -0
- data/app/views/accounts/_title_bar.html.haml +11 -0
- data/app/views/accounts/_top_section.html.haml +29 -0
- data/app/views/accounts/create.js.haml +17 -0
- data/app/views/accounts/destroy.js.haml +6 -0
- data/app/views/accounts/edit.js.haml +31 -0
- data/app/views/accounts/index.html.haml +18 -0
- data/app/views/accounts/index.js.haml +11 -0
- data/app/views/accounts/index.xls.builder +84 -0
- data/app/views/accounts/new.js.haml +11 -0
- data/app/views/accounts/show.html.haml +26 -0
- data/app/views/accounts/show.js.haml +5 -0
- data/app/views/accounts/update.js.haml +19 -0
- data/app/views/admin/custom_fields/_base_field.html.haml +21 -0
- data/app/views/admin/custom_fields/_check_boxes_field.html.haml +6 -0
- data/app/views/admin/custom_fields/_date_field.html.haml +14 -0
- data/app/views/admin/custom_fields/_date_pair_field.html.haml +21 -0
- data/app/views/admin/custom_fields/_datetime_field.html.haml +14 -0
- data/app/views/admin/custom_fields/_datetime_pair_field.html.haml +21 -0
- data/app/views/admin/custom_fields/_radio_field.html.haml +6 -0
- data/app/views/admin/custom_fields/_select_field.html.haml +25 -0
- data/app/views/admin/custom_fields/_string_field.html.haml +2 -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 +28 -0
- data/app/views/admin/field_groups/_new.html.haml +10 -0
- data/app/views/admin/field_groups/_top_section.html.haml +14 -0
- data/app/views/admin/field_groups/confirm.js.haml +7 -0
- data/app/views/admin/field_groups/create.js.haml +14 -0
- data/app/views/admin/field_groups/destroy.js.haml +9 -0
- data/app/views/admin/field_groups/edit.js.haml +12 -0
- data/app/views/admin/field_groups/new.js.haml +8 -0
- data/app/views/admin/field_groups/update.js.haml +8 -0
- data/app/views/admin/fields/_field.html.haml +17 -0
- data/app/views/admin/fields/_form.html.haml +35 -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/_subform.html.haml +6 -0
- data/app/views/admin/fields/create.js.haml +17 -0
- data/app/views/admin/fields/destroy.js.haml +8 -0
- data/app/views/admin/fields/edit.js.haml +3 -0
- data/app/views/admin/fields/index.html.haml +34 -0
- data/app/views/admin/fields/show.html.haml +41 -0
- data/app/views/admin/fields/update.js.haml +13 -0
- data/app/views/admin/groups/_edit.html.haml +12 -0
- data/app/views/admin/groups/_form.html.haml +11 -0
- data/app/views/admin/groups/_group.html.haml +16 -0
- data/app/views/admin/groups/_new.html.haml +12 -0
- data/app/views/admin/groups/create.js.haml +10 -0
- data/app/views/admin/groups/destroy.js.haml +9 -0
- data/app/views/admin/groups/edit.js.haml +14 -0
- data/app/views/admin/groups/index.html.haml +18 -0
- data/app/views/admin/groups/index.js.haml +2 -0
- data/app/views/admin/groups/new.js.haml +9 -0
- data/app/views/admin/groups/show.html.haml +11 -0
- data/app/views/admin/groups/update.js.haml +9 -0
- data/app/views/admin/plugins/_plugin.html.haml +13 -0
- data/app/views/admin/plugins/index.html.haml +3 -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.haml +7 -0
- data/app/views/admin/tags/create.js.haml +11 -0
- data/app/views/admin/tags/destroy.js.haml +10 -0
- data/app/views/admin/tags/edit.js.haml +20 -0
- data/app/views/admin/tags/index.html.haml +17 -0
- data/app/views/admin/tags/new.js.haml +7 -0
- data/app/views/admin/tags/update.js.haml +9 -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 +54 -0
- data/app/views/admin/users/_sidebar_index.html.haml +2 -0
- data/app/views/admin/users/_user.html.haml +66 -0
- data/app/views/admin/users/confirm.js.haml +7 -0
- data/app/views/admin/users/create.js.haml +10 -0
- data/app/views/admin/users/destroy.js.haml +10 -0
- data/app/views/admin/users/edit.js.haml +20 -0
- data/app/views/admin/users/index.html.haml +13 -0
- data/app/views/admin/users/index.js.haml +2 -0
- data/app/views/admin/users/new.js.haml +9 -0
- data/app/views/admin/users/reactivate.js.haml +3 -0
- data/app/views/admin/users/show.html.haml +5 -0
- data/app/views/admin/users/suspend.js.haml +3 -0
- data/app/views/admin/users/update.js.haml +9 -0
- data/app/views/application/_auto_complete.html.haml +10 -0
- data/app/views/application/index.atom.builder +34 -0
- data/app/views/application/index.rss.builder +31 -0
- data/app/views/application/show.atom.builder +32 -0
- data/app/views/application/show.rss.builder +29 -0
- data/app/views/authentications/new.html.haml +19 -0
- data/app/views/campaigns/_campaign.html.haml +5 -0
- data/app/views/campaigns/_edit.html.haml +18 -0
- data/app/views/campaigns/_index_brief.html.haml +15 -0
- data/app/views/campaigns/_index_long.html.haml +20 -0
- data/app/views/campaigns/_metrics.html.haml +16 -0
- data/app/views/campaigns/_new.html.haml +18 -0
- data/app/views/campaigns/_objectives.html.haml +30 -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/_title_bar.html.haml +11 -0
- data/app/views/campaigns/_top_section.html.haml +29 -0
- data/app/views/campaigns/create.js.haml +20 -0
- data/app/views/campaigns/destroy.js.haml +6 -0
- data/app/views/campaigns/edit.js.haml +31 -0
- data/app/views/campaigns/index.html.haml +18 -0
- data/app/views/campaigns/index.js.haml +11 -0
- data/app/views/campaigns/index.xls.builder +77 -0
- data/app/views/campaigns/new.js.haml +11 -0
- data/app/views/campaigns/show.html.haml +26 -0
- data/app/views/campaigns/show.js.haml +5 -0
- data/app/views/campaigns/update.js.haml +22 -0
- data/app/views/comments/_comment.html.haml +29 -0
- data/app/views/comments/_edit.html.haml +11 -0
- data/app/views/comments/_new.html.haml +35 -0
- data/app/views/comments/_subscription_links.html.haml +13 -0
- data/app/views/comments/create.js.haml +14 -0
- data/app/views/comments/destroy.js.haml +5 -0
- data/app/views/comments/edit.js.haml +11 -0
- data/app/views/comments/update.js.haml +9 -0
- data/app/views/contacts/_contact.html.haml +5 -0
- data/app/views/contacts/_contacts.html.haml +7 -0
- data/app/views/contacts/_edit.html.haml +19 -0
- data/app/views/contacts/_extra.html.haml +36 -0
- data/app/views/contacts/_index_brief.html.haml +25 -0
- data/app/views/contacts/_index_full.html.haml +57 -0
- data/app/views/contacts/_index_long.html.haml +44 -0
- data/app/views/contacts/_new.html.haml +20 -0
- data/app/views/contacts/_section_extra.html.haml +26 -0
- data/app/views/contacts/_section_general.html.haml +26 -0
- data/app/views/contacts/_sidebar_index.html.haml +0 -0
- data/app/views/contacts/_sidebar_show.html.haml +52 -0
- data/app/views/contacts/_title_bar.html.haml +11 -0
- data/app/views/contacts/_top_section.html.haml +45 -0
- data/app/views/contacts/_web.html.haml +31 -0
- data/app/views/contacts/create.js.haml +23 -0
- data/app/views/contacts/destroy.js.haml +9 -0
- data/app/views/contacts/edit.js.haml +34 -0
- data/app/views/contacts/index.html.haml +18 -0
- data/app/views/contacts/index.js.haml +11 -0
- data/app/views/contacts/index.xls.builder +104 -0
- data/app/views/contacts/new.js.haml +13 -0
- data/app/views/contacts/show.html.haml +22 -0
- data/app/views/contacts/show.js.haml +5 -0
- data/app/views/contacts/update.js.haml +24 -0
- data/app/views/dropbox_mailer/dropbox_notification.html.haml +9 -0
- data/app/views/emails/_email.html.haml +36 -0
- data/app/views/emails/destroy.js.haml +5 -0
- data/app/views/entities/_basic_search.html.haml +24 -0
- data/app/views/entities/_permissions.html.haml +36 -0
- data/app/views/entities/_search.html.haml +16 -0
- data/app/views/entities/_section_custom_fields.html.haml +11 -0
- data/app/views/entities/_sort_fields.html.haml +3 -0
- data/app/views/entities/_title_bar.html.haml +12 -0
- data/app/views/entities/attach.js.haml +21 -0
- data/app/views/entities/contacts.js.haml +3 -0
- data/app/views/entities/discard.js.haml +6 -0
- data/app/views/entities/leads.js.haml +3 -0
- data/app/views/entities/opportunities.js.haml +3 -0
- data/app/views/entities/subscription_update.js.haml +4 -0
- data/app/views/entities/versions.js.haml +3 -0
- data/app/views/fields/_edit_custom_field_group.html.haml +8 -0
- data/app/views/fields/_group.html.haml +11 -0
- data/app/views/fields/_group_table.html.haml +13 -0
- data/app/views/fields/_group_view.html.haml +10 -0
- data/app/views/fields/_groups.html.haml +8 -0
- data/app/views/fields/_sidebar_show.html.haml +9 -0
- data/app/views/fields/group.js.erb +3 -0
- data/app/views/home/_account.html.haml +29 -0
- data/app/views/home/_activity.html.haml +35 -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/_events_menu.html.haml +8 -0
- data/app/views/home/_opportunity.html.haml +41 -0
- data/app/views/home/_options.html.haml +12 -0
- data/app/views/home/_task.html.haml +48 -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 +46 -0
- data/app/views/home/index.js.haml +7 -0
- data/app/views/home/index.rss.builder +20 -0
- data/app/views/home/options.js.haml +7 -0
- data/app/views/layouts/500.html.haml +17 -0
- data/app/views/layouts/_about.html.haml +17 -0
- data/app/views/layouts/_footer.html.haml +9 -0
- data/app/views/layouts/_header.html.haml +20 -0
- data/app/views/layouts/_jumpbox.html.haml +34 -0
- data/app/views/layouts/_sidebar.html.haml +12 -0
- data/app/views/layouts/_tabbed.html.haml +15 -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 +24 -0
- data/app/views/layouts/application.html.haml +48 -0
- data/app/views/layouts/header.xls.builder +8 -0
- data/app/views/leads/_contact.html.haml +35 -0
- data/app/views/leads/_convert.html.haml +25 -0
- data/app/views/leads/_convert_permissions.html.haml +31 -0
- data/app/views/leads/_edit.html.haml +23 -0
- data/app/views/leads/_index_brief.html.haml +41 -0
- data/app/views/leads/_index_long.html.haml +62 -0
- data/app/views/leads/_lead.html.haml +5 -0
- data/app/views/leads/_leads.html.haml +6 -0
- data/app/views/leads/_new.html.haml +24 -0
- data/app/views/leads/_opportunity.html.haml +37 -0
- data/app/views/leads/_sidebar_index.html.haml +20 -0
- data/app/views/leads/_sidebar_show.html.haml +65 -0
- data/app/views/leads/_status.html.haml +36 -0
- data/app/views/leads/_title_bar.html.haml +16 -0
- data/app/views/leads/_top_section.html.haml +30 -0
- data/app/views/leads/_web.html.haml +31 -0
- data/app/views/leads/convert.js.haml +35 -0
- data/app/views/leads/create.js.haml +21 -0
- data/app/views/leads/destroy.js.haml +9 -0
- data/app/views/leads/edit.js.haml +33 -0
- data/app/views/leads/index.html.haml +18 -0
- data/app/views/leads/index.js.haml +11 -0
- data/app/views/leads/index.xls.builder +102 -0
- data/app/views/leads/new.js.haml +11 -0
- data/app/views/leads/promote.js.haml +26 -0
- data/app/views/leads/reject.js.haml +12 -0
- data/app/views/leads/show.html.haml +20 -0
- data/app/views/leads/show.js.haml +5 -0
- data/app/views/leads/update.js.haml +28 -0
- data/app/views/lists/_lists.html.haml +6 -0
- data/app/views/lists/_sidebar.html.haml +29 -0
- data/app/views/lists/create.js.haml +9 -0
- data/app/views/lists/destroy.js.haml +1 -0
- data/app/views/opportunities/_edit.html.haml +18 -0
- data/app/views/opportunities/_index_brief.html.haml +29 -0
- data/app/views/opportunities/_index_long.html.haml +59 -0
- data/app/views/opportunities/_new.html.haml +18 -0
- data/app/views/opportunities/_opportunities.html.haml +6 -0
- data/app/views/opportunities/_opportunity.html.haml +5 -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/_title_bar.html.haml +11 -0
- data/app/views/opportunities/_top_section.html.haml +59 -0
- data/app/views/opportunities/contacts.js.haml +3 -0
- data/app/views/opportunities/create.js.haml +29 -0
- data/app/views/opportunities/destroy.js.haml +13 -0
- data/app/views/opportunities/edit.js.haml +34 -0
- data/app/views/opportunities/index.html.haml +20 -0
- data/app/views/opportunities/index.js.haml +11 -0
- data/app/views/opportunities/index.xls.builder +71 -0
- data/app/views/opportunities/new.js.haml +13 -0
- data/app/views/opportunities/show.html.haml +22 -0
- data/app/views/opportunities/show.js.haml +5 -0
- data/app/views/opportunities/update.js.haml +27 -0
- data/app/views/passwords/edit.html.haml +15 -0
- data/app/views/passwords/new.html.haml +10 -0
- data/app/views/shared/_add_comment.html.haml +8 -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 +2 -0
- data/app/views/shared/_inline_styles.html.haml +31 -0
- data/app/views/shared/_naming.html.haml +9 -0
- data/app/views/shared/_paginate.haml +2 -0
- data/app/views/shared/_paginate_with_per_page.html.haml +12 -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 +15 -0
- data/app/views/shared/_select_popup.html.haml +19 -0
- data/app/views/shared/_tags.html.haml +10 -0
- data/app/views/shared/_tasks.html.haml +10 -0
- data/app/views/shared/_timeline.html.haml +2 -0
- data/app/views/shared/_total.html.haml +4 -0
- data/app/views/subscription_mailer/comment_notification.text.erb +7 -0
- data/app/views/tasks/_assigned.html.haml +47 -0
- data/app/views/tasks/_completed.html.haml +25 -0
- data/app/views/tasks/_edit.html.haml +18 -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 +32 -0
- data/app/views/tasks/_selector.html.haml +8 -0
- data/app/views/tasks/_sidebar_index.html.haml +18 -0
- data/app/views/tasks/_tasks.html.haml +3 -0
- data/app/views/tasks/_title.html.haml +8 -0
- data/app/views/tasks/_top_section.html.haml +34 -0
- data/app/views/tasks/complete.js.haml +14 -0
- data/app/views/tasks/create.js.haml +39 -0
- data/app/views/tasks/destroy.js.haml +7 -0
- data/app/views/tasks/discard.js.haml +1 -0
- data/app/views/tasks/edit.js.haml +26 -0
- data/app/views/tasks/filter.js.haml +4 -0
- data/app/views/tasks/index.html.haml +20 -0
- data/app/views/tasks/index.xls.builder +49 -0
- data/app/views/tasks/new.js.haml +8 -0
- data/app/views/tasks/uncomplete.js.haml +12 -0
- data/app/views/tasks/update.js.haml +19 -0
- data/app/views/user_mailer/assigned_entity_notification.html.haml +3 -0
- data/app/views/user_mailer/password_reset_instructions.html.haml +6 -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 +33 -0
- data/app/views/users/_user_report.html.haml +6 -0
- data/app/views/users/avatar.js.haml +9 -0
- data/app/views/users/change_password.js.haml +13 -0
- data/app/views/users/edit.js.haml +9 -0
- data/app/views/users/index.html.haml +3 -0
- data/app/views/users/new.html.haml +19 -0
- data/app/views/users/opportunities_overview.html.haml +16 -0
- data/app/views/users/password.js.haml +10 -0
- data/app/views/users/show.html.haml +15 -0
- data/app/views/users/update.js.haml +9 -0
- data/app/views/users/upload_avatar.js.haml +7 -0
- data/app/views/versions/_version.html.haml +17 -0
- data/app/views/versions/_version_item.html.haml +8 -0
- data/app/views/versions/_versions.html.haml +11 -0
- data/bin/bundle +3 -0
- data/bin/rails +4 -0
- data/bin/rake +4 -0
- data/config.ru +4 -0
- data/config/application.rb +77 -0
- data/config/boot.rb +10 -0
- data/config/database.mysql.mac.yml +32 -0
- data/config/database.mysql.yml +33 -0
- data/config/database.postgres.docker.yml +25 -0
- data/config/database.postgres.yml +26 -0
- data/config/database.sqlite.yml +23 -0
- data/config/deploy.example.rb +77 -0
- data/config/environment.rb +10 -0
- data/config/environments/development.rb +46 -0
- data/config/environments/production.rb +66 -0
- data/config/environments/staging.rb +66 -0
- data/config/environments/test.rb +54 -0
- data/config/initializers/action_mailer.rb +22 -0
- data/config/initializers/assets.rb +14 -0
- data/config/initializers/authlogic.rb +6 -0
- data/config/initializers/backtrace_silencers.rb +7 -0
- data/config/initializers/constants.rb +6 -0
- data/config/initializers/cookies_serializer.rb +3 -0
- data/config/initializers/custom_field_ransack_translations.rb +17 -0
- data/config/initializers/filter_parameter_logging.rb +4 -0
- data/config/initializers/gravatar.rb +12 -0
- data/config/initializers/inflections.rb +16 -0
- data/config/initializers/locale.rb +18 -0
- data/config/initializers/mime_types.rb +12 -0
- data/config/initializers/paginate_arrays.rb +11 -0
- data/config/initializers/ransack.rb +20 -0
- data/config/initializers/relative_url_root.rb +10 -0
- data/config/initializers/secret_token.rb +25 -0
- data/config/initializers/session_store.rb +5 -0
- data/config/initializers/simple_form.rb +100 -0
- data/config/initializers/views.rb +47 -0
- data/config/initializers/wrap_parameters.rb +14 -0
- data/config/locales/cz.yml +247 -0
- data/config/locales/cz_fat_free_crm.yml +610 -0
- data/config/locales/de.yml +237 -0
- data/config/locales/de_fat_free_crm.yml +842 -0
- data/config/locales/de_ransack.yml +91 -0
- data/config/locales/en-GB.yml +219 -0
- data/config/locales/en-GB_fat_free_crm.yml +937 -0
- data/config/locales/en-GB_ransack.yml +91 -0
- data/config/locales/en-US.yml +219 -0
- data/config/locales/en-US_fat_free_crm.yml +937 -0
- data/config/locales/en-US_ransack.yml +91 -0
- data/config/locales/en.yml +23 -0
- data/config/locales/es-CL.yml +227 -0
- data/config/locales/es-CL_fat_free_crm.yml +929 -0
- data/config/locales/es-CL_ransack.yml +91 -0
- data/config/locales/es.yml +219 -0
- data/config/locales/es_fat_free_crm.yml +917 -0
- data/config/locales/es_ransack.yml +91 -0
- data/config/locales/fr-CA.yml +224 -0
- data/config/locales/fr-CA_fat_free_crm.yml +570 -0
- data/config/locales/fr.yml +227 -0
- data/config/locales/fr_fat_free_crm.yml +934 -0
- data/config/locales/it.yml +219 -0
- data/config/locales/it_fat_free_crm.yml +592 -0
- data/config/locales/ja.yml +219 -0
- data/config/locales/ja_fat_free_crm.yml +557 -0
- data/config/locales/ja_ransack.yml +91 -0
- data/config/locales/nl.yml +210 -0
- data/config/locales/nl_fat_free_crm.yml +920 -0
- data/config/locales/nl_ransack.yml +91 -0
- data/config/locales/pl.yml +230 -0
- data/config/locales/pl_fat_free_crm.yml +570 -0
- data/config/locales/pt-BR.yml +232 -0
- data/config/locales/pt-BR_fat_free_crm.yml +650 -0
- data/config/locales/ru.yml +326 -0
- data/config/locales/ru_fat_free_crm.yml +977 -0
- data/config/locales/sv-SE.yml +215 -0
- data/config/locales/sv-SE_fat_free_crm.yml +565 -0
- data/config/locales/th.rb +204 -0
- data/config/locales/th_fat_free_crm.yml +558 -0
- data/config/locales/zh-CN.yml +241 -0
- data/config/locales/zh-CN_fat_free_crm.yml +937 -0
- data/config/locales/zh-CN_ransack.yml +91 -0
- data/config/routes.rb +199 -0
- data/config/settings.default.yml +400 -0
- data/config/unicorn.rb +32 -0
- data/custom_plan.rb +9 -0
- data/db/demo/account_contacts.yml +24 -0
- data/db/demo/account_opportunities.yml +24 -0
- data/db/demo/accounts.yml +47 -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/field_groups.yml +14 -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 +245 -0
- data/db/migrate/20100928030598_create_sessions.rb +21 -0
- data/db/migrate/20100928030599_create_users.rb +45 -0
- data/db/migrate/20100928030600_create_openid_tables.rb +23 -0
- data/db/migrate/20100928030601_create_accounts.rb +26 -0
- data/db/migrate/20100928030602_create_permissions.rb +15 -0
- data/db/migrate/20100928030603_create_settings.rb +15 -0
- data/db/migrate/20100928030604_create_preferences.rb +15 -0
- data/db/migrate/20100928030605_create_campaigns.rb +34 -0
- data/db/migrate/20100928030606_create_leads.rb +38 -0
- data/db/migrate/20100928030607_create_contacts.rb +38 -0
- data/db/migrate/20100928030608_create_opportunities.rb +27 -0
- data/db/migrate/20100928030609_create_account_contacts.rb +14 -0
- data/db/migrate/20100928030610_create_account_opportunities.rb +14 -0
- data/db/migrate/20100928030611_create_contact_opportunities.rb +15 -0
- data/db/migrate/20100928030612_create_tasks.rb +26 -0
- data/db/migrate/20100928030613_create_comments.rb +16 -0
- data/db/migrate/20100928030614_create_activities.rb +19 -0
- data/db/migrate/20100928030615_create_avatars.rb +16 -0
- data/db/migrate/20100928030616_rename_remember_token.rb +11 -0
- data/db/migrate/20100928030617_drop_openid_tables.rb +23 -0
- data/db/migrate/20100928030618_add_admin_to_users.rb +11 -0
- data/db/migrate/20100928030619_add_suspended_to_users.rb +11 -0
- data/db/migrate/20100928030620_remove_uuid.rb +24 -0
- data/db/migrate/20100928030621_add_email_to_accounts.rb +9 -0
- data/db/migrate/20100928030622_add_background_info_to_models.rb +19 -0
- data/db/migrate/20100928030623_create_addresses.rb +49 -0
- data/db/migrate/20100928030624_add_index_on_permissions.rb +9 -0
- data/db/migrate/20100928030625_create_emails.rb +26 -0
- data/db/migrate/20100928030626_add_state_to_timeline_objects.rb +13 -0
- data/db/migrate/20100928030627_acts_as_taggable_on_migration.rb +29 -0
- data/db/migrate/20101221123456_add_single_access_token_to_users.rb +9 -0
- data/db/migrate/20101221345678_add_rating_and_category_to_accounts.rb +11 -0
- data/db/migrate/20110719082054_add_skype_to_contacts_and_leads.rb +11 -0
- data/db/migrate/20111101083437_create_fields.rb +27 -0
- data/db/migrate/20111101090312_create_field_groups.rb +15 -0
- data/db/migrate/20111116091952_add_field_groups_tag_id.rb +9 -0
- data/db/migrate/20111117041311_change_fields_collection_to_text.rb +9 -0
- data/db/migrate/20111201030535_add_field_groups_klass_name.rb +24 -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 +10 -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/migrate/20120309070209_add_versions_related.rb +6 -0
- data/db/migrate/20120314080441_add_subscribed_users_to_entities.rb +23 -0
- data/db/migrate/20120316045804_activities_to_versions.rb +35 -0
- data/db/migrate/20120405080727_change_subscribed_users_to_set.rb +24 -0
- data/db/migrate/20120405080742_change_further_subscribed_users_to_set.rb +27 -0
- data/db/migrate/20120406082136_create_groups.rb +19 -0
- data/db/migrate/20120413034923_add_index_on_versions_item_type.rb +5 -0
- data/db/migrate/20120510025219_add_not_null_constraints_for_timestamp_columns.rb +22 -0
- data/db/migrate/20120528102124_increase_length_of_version_events.rb +9 -0
- data/db/migrate/20120801032706_add_pair_id_to_fields.rb +5 -0
- data/db/migrate/20121003063155_add_settings_to_custom_fields.rb +5 -0
- data/db/migrate/20121221033947_fix_country_mapping.rb +31 -0
- data/db/migrate/20131207033244_add_user_id_to_lists.rb +6 -0
- data/db/migrate/20140916011927_add_created_at_index_on_versions.rb +5 -0
- data/db/migrate/20140916012922_add_indexes_to_model_associations.rb +6 -0
- data/db/migrate/20141126031837_increase_email_to254_chars.rb +21 -0
- data/db/migrate/20141230021159_add_transaction_id_column_to_versions.rb +11 -0
- data/db/migrate/20141230205453_add_missing_unique_indices.acts_as_taggable_on_engine.rb +20 -0
- data/db/migrate/20141230205454_add_taggings_counter_cache_to_tags.acts_as_taggable_on_engine.rb +15 -0
- data/db/migrate/20141230205455_add_missing_taggable_index.acts_as_taggable_on_engine.rb +10 -0
- data/db/migrate/20150123060900_convert_radio_to_radio_buttons.rb +10 -0
- data/db/migrate/20150227123054_remove_last_request_at_from_users.rb +5 -0
- data/db/migrate/20150427131956_create_index_related_type.rb +9 -0
- data/db/migrate/20160511053730_add_account_contacts_index.rb +5 -0
- data/db/schema.rb +459 -0
- data/db/seeds.rb +9 -0
- data/db/seeds/fields.rb +93 -0
- data/docker-compose.yml +13 -0
- data/fat_free_crm.gemspec +63 -0
- data/lib/development_tasks/gem.rake +13 -0
- data/lib/development_tasks/license.rake +77 -0
- data/lib/development_tasks/rdoc.rake +20 -0
- data/lib/development_tasks/rspec.rake +24 -0
- data/lib/fat_free_crm.rb +57 -0
- data/lib/fat_free_crm/callback.rb +144 -0
- data/lib/fat_free_crm/comment_extensions.rb +28 -0
- data/lib/fat_free_crm/core_ext.rb +6 -0
- data/lib/fat_free_crm/core_ext/nil.rb +14 -0
- data/lib/fat_free_crm/core_ext/string.rb +48 -0
- data/lib/fat_free_crm/custom_fields.rb +12 -0
- data/lib/fat_free_crm/engine.rb +27 -0
- data/lib/fat_free_crm/errors.rb +37 -0
- data/lib/fat_free_crm/exceptions.rb +21 -0
- data/lib/fat_free_crm/export_csv.rb +33 -0
- data/lib/fat_free_crm/exportable.rb +49 -0
- data/lib/fat_free_crm/fields.rb +95 -0
- data/lib/fat_free_crm/gem_dependencies.rb +40 -0
- data/lib/fat_free_crm/gem_ext.rb +10 -0
- data/lib/fat_free_crm/gem_ext/action_controller/base.rb +16 -0
- data/lib/fat_free_crm/gem_ext/active_record/schema_dumper.rb +27 -0
- data/lib/fat_free_crm/gem_ext/active_support/buffered_logger.rb +17 -0
- data/lib/fat_free_crm/gem_ext/rails/engine.rb +15 -0
- data/lib/fat_free_crm/gem_ext/rake/task.rb +13 -0
- data/lib/fat_free_crm/gem_ext/simple_form/action_view_extensions/form_helper.rb +24 -0
- data/lib/fat_free_crm/i18n.rb +42 -0
- data/lib/fat_free_crm/load_settings.rb +18 -0
- data/lib/fat_free_crm/mail_processor/base.rb +227 -0
- data/lib/fat_free_crm/mail_processor/comment_replies.rb +69 -0
- data/lib/fat_free_crm/mail_processor/dropbox.rb +273 -0
- data/lib/fat_free_crm/permissions.rb +109 -0
- data/lib/fat_free_crm/renderers.rb +11 -0
- data/lib/fat_free_crm/secret_token_generator.rb +59 -0
- data/lib/fat_free_crm/sortable.rb +43 -0
- data/lib/fat_free_crm/tabs.rb +23 -0
- data/lib/fat_free_crm/version.rb +15 -0
- data/lib/fat_free_crm/view_factory.rb +81 -0
- data/lib/gravatar_image_tag.rb +58 -0
- data/lib/missing_translation_detector.rb +50 -0
- data/lib/tasks/.gitkeep +0 -0
- data/lib/tasks/.gitkeep~master +0 -0
- data/lib/tasks/ffcrm/comment_replies.rake +27 -0
- data/lib/tasks/ffcrm/config.rake +40 -0
- data/lib/tasks/ffcrm/demo.rake +62 -0
- data/lib/tasks/ffcrm/dropbox.rake +28 -0
- data/lib/tasks/ffcrm/missing_translations.rake +34 -0
- data/lib/tasks/ffcrm/secret.rake +15 -0
- data/lib/tasks/ffcrm/settings.rake +20 -0
- data/lib/tasks/ffcrm/setup.rake +81 -0
- data/lib/tasks/ffcrm/update_data.rake +238 -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/favicon.ico +0 -0
- data/public/robots.txt +5 -0
- data/script/rails +6 -0
- data/spec/controllers/admin/groups_controller_spec.rb +9 -0
- data/spec/controllers/admin/users_controller_spec.rb +305 -0
- data/spec/controllers/applications_controller_spec.rb +41 -0
- data/spec/controllers/authentications_controller_spec.rb +148 -0
- data/spec/controllers/comments_controller_spec.rb +193 -0
- data/spec/controllers/emails_controller_spec.rb +35 -0
- data/spec/controllers/entities/accounts_controller_spec.rb +605 -0
- data/spec/controllers/entities/campaigns_controller_spec.rb +643 -0
- data/spec/controllers/entities/contacts_controller_spec.rb +702 -0
- data/spec/controllers/entities/leads_controller_spec.rb +1007 -0
- data/spec/controllers/entities/opportunities_controller_spec.rb +902 -0
- data/spec/controllers/entities_controller_spec.rb +40 -0
- data/spec/controllers/home_controller_spec.rb +231 -0
- data/spec/controllers/lists_controller_spec.rb +9 -0
- data/spec/controllers/passwords_controller_spec.rb +30 -0
- data/spec/controllers/tasks_controller_spec.rb +512 -0
- data/spec/controllers/users_controller_spec.rb +431 -0
- data/spec/factories/account_factories.rb +38 -0
- data/spec/factories/campaign_factories.rb +28 -0
- data/spec/factories/contact_factories.rb +43 -0
- data/spec/factories/field_factories.rb +37 -0
- data/spec/factories/lead_factories.rb +34 -0
- data/spec/factories/list_factories.rb +11 -0
- data/spec/factories/opportunity_factories.rb +37 -0
- data/spec/factories/sequences.rb +31 -0
- data/spec/factories/setting_factories.rb +13 -0
- data/spec/factories/shared_factories.rb +66 -0
- data/spec/factories/subscription_factories.rb +11 -0
- data/spec/factories/tag_factories.rb +10 -0
- data/spec/factories/task_factories.rb +27 -0
- data/spec/factories/user_factories.rb +64 -0
- data/spec/features/acceptance_helper.rb +9 -0
- data/spec/features/accounts_spec.rb +109 -0
- data/spec/features/admin/groups_spec.rb +30 -0
- data/spec/features/admin/users_spec.rb +37 -0
- data/spec/features/campaigns_spec.rb +103 -0
- data/spec/features/contacts_spec.rb +129 -0
- data/spec/features/dashboard_spec.rb +66 -0
- data/spec/features/leads_spec.rb +129 -0
- data/spec/features/opportunities_overview_spec.rb +87 -0
- data/spec/features/opportunities_spec.rb +105 -0
- data/spec/features/support/browser.rb +18 -0
- data/spec/features/support/headless.rb +17 -0
- data/spec/features/support/helpers.rb +30 -0
- data/spec/features/support/maintain_sessions.rb +10 -0
- data/spec/features/support/paths.rb +46 -0
- data/spec/features/support/selector_helpers.rb +31 -0
- data/spec/features/tasks_spec.rb +93 -0
- data/spec/fixtures/rails.png +0 -0
- data/spec/helpers/accounts_helper_spec.rb +14 -0
- data/spec/helpers/admin/field_groups_helper_spec.rb +18 -0
- data/spec/helpers/admin/plugins_helper_spec.rb +9 -0
- data/spec/helpers/admin/settings_helper_spec.rb +9 -0
- data/spec/helpers/admin/users_helper_spec.rb +9 -0
- data/spec/helpers/application_helper_spec.rb +83 -0
- data/spec/helpers/authentications_helper_spec.rb +14 -0
- data/spec/helpers/campaigns_helper_spec.rb +14 -0
- data/spec/helpers/comments_helper_spec.rb +14 -0
- data/spec/helpers/contacts_helper_spec.rb +14 -0
- data/spec/helpers/emails_helper_spec.rb +9 -0
- data/spec/helpers/fields_helper_spec.rb +10 -0
- data/spec/helpers/groups_helper_spec.rb +19 -0
- data/spec/helpers/home_helper_spec.rb +14 -0
- data/spec/helpers/leads_helper_spec.rb +14 -0
- data/spec/helpers/lists_helper_spec.rb +19 -0
- data/spec/helpers/opportunities_helper_spec.rb +14 -0
- data/spec/helpers/passwords_helper_spec.rb +14 -0
- data/spec/helpers/tasks_helper_spec.rb +18 -0
- data/spec/helpers/users_helper_spec.rb +37 -0
- data/spec/lib/comment_extensions_spec.rb +34 -0
- data/spec/lib/core_ext/string_spec.rb +21 -0
- data/spec/lib/errors_spec.rb +30 -0
- data/spec/lib/fields_spec.rb +99 -0
- data/spec/lib/mail_processor/base_spec.rb +167 -0
- data/spec/lib/mail_processor/comment_replies_spec.rb +68 -0
- data/spec/lib/mail_processor/dropbox_spec.rb +304 -0
- data/spec/lib/mail_processor/sample_emails/dropbox.rb +171 -0
- data/spec/lib/permissions_spec.rb +134 -0
- data/spec/lib/secret_token_generator_spec.rb +67 -0
- data/spec/lib/view_factory_spec.rb +67 -0
- data/spec/mailers/subscription_mailer_spec.rb +47 -0
- data/spec/mailers/user_mailer_spec.rb +139 -0
- data/spec/models/entities/account_contact_spec.rb +31 -0
- data/spec/models/entities/account_opportunity_spec.rb +31 -0
- data/spec/models/entities/account_spec.rb +178 -0
- data/spec/models/entities/campaign_spec.rb +136 -0
- data/spec/models/entities/contact_opportunity_spec.rb +32 -0
- data/spec/models/entities/contact_spec.rb +208 -0
- data/spec/models/entities/lead_spec.rb +102 -0
- data/spec/models/entities/opportunity_spec.rb +287 -0
- data/spec/models/fields/custom_field_date_pair_spec.rb +88 -0
- data/spec/models/fields/custom_field_pair_spec.rb +74 -0
- data/spec/models/fields/custom_field_spec.rb +125 -0
- data/spec/models/fields/field_group_spec.rb +27 -0
- data/spec/models/fields/field_spec.rb +65 -0
- data/spec/models/list_spec.rb +17 -0
- data/spec/models/observers/entity_observer_spec.rb +72 -0
- data/spec/models/polymorphic/address_spec.rb +36 -0
- data/spec/models/polymorphic/avatar_spec.rb +44 -0
- data/spec/models/polymorphic/comment_spec.rb +47 -0
- data/spec/models/polymorphic/email_spec.rb +40 -0
- data/spec/models/polymorphic/task_spec.rb +385 -0
- data/spec/models/polymorphic/version_spec.rb +244 -0
- data/spec/models/setting_spec.rb +54 -0
- data/spec/models/users/abilities/user_ability_spec.rb +63 -0
- data/spec/models/users/authentication_spec.rb +17 -0
- data/spec/models/users/group_spec.rb +9 -0
- data/spec/models/users/permission_spec.rb +48 -0
- data/spec/models/users/preference_spec.rb +69 -0
- data/spec/models/users/user_spec.rb +237 -0
- data/spec/routing/accounts_routing_spec.rb +58 -0
- data/spec/routing/admin/users_routing_spec.rb +38 -0
- data/spec/routing/campaigns_routing_spec.rb +62 -0
- data/spec/routing/comments_routing_spec.rb +30 -0
- data/spec/routing/contacts_routing_spec.rb +58 -0
- data/spec/routing/emails_routing_spec.rb +38 -0
- data/spec/routing/leads_routing_spec.rb +86 -0
- data/spec/routing/opportunities_routing_spec.rb +62 -0
- data/spec/routing/tasks_routing_spec.rb +66 -0
- data/spec/routing/users_routing_spec.rb +82 -0
- data/spec/shared/controllers.rb +103 -0
- data/spec/shared/models.rb +139 -0
- data/spec/spec_helper.rb +107 -0
- data/spec/support/assert_select.rb +164 -0
- data/spec/support/auth_macros.rb +48 -0
- data/spec/support/macros.rb +44 -0
- data/spec/support/mail_processor_mocks.rb +35 -0
- data/spec/support/uploaded_file.rb +8 -0
- data/spec/views/accounts/_edit.haml_spec.rb +42 -0
- data/spec/views/accounts/_new.haml_spec.rb +40 -0
- data/spec/views/accounts/create.js.haml_spec.rb +51 -0
- data/spec/views/accounts/destroy.js.haml_spec.rb +32 -0
- data/spec/views/accounts/edit.js.haml_spec.rb +64 -0
- data/spec/views/accounts/index.haml_spec.rb +41 -0
- data/spec/views/accounts/index.js.haml_spec.rb +32 -0
- data/spec/views/accounts/new.js.haml_spec.rb +42 -0
- data/spec/views/accounts/show.haml_spec.rb +33 -0
- data/spec/views/accounts/update.js.haml_spec.rb +94 -0
- data/spec/views/admin/field_groups/create.js.haml_spec.rb +29 -0
- data/spec/views/admin/field_groups/destroy.js.haml_spec.rb +29 -0
- data/spec/views/admin/field_groups/edit.js.haml_spec.rb +22 -0
- data/spec/views/admin/field_groups/new.js.haml_spec.rb +23 -0
- data/spec/views/admin/field_groups/update.js.haml_spec.rb +28 -0
- data/spec/views/admin/users/_create.haml_spec.rb +21 -0
- data/spec/views/admin/users/create.js.haml_spec.rb +40 -0
- data/spec/views/admin/users/destroy.js.haml_spec.rb +51 -0
- data/spec/views/admin/users/edit.js.haml_spec.rb +42 -0
- data/spec/views/admin/users/index.haml_spec.rb +20 -0
- data/spec/views/admin/users/index.js.haml_spec.rb +24 -0
- data/spec/views/admin/users/new.js.haml_spec.rb +32 -0
- data/spec/views/admin/users/reactivate.js.haml_spec.rb +20 -0
- data/spec/views/admin/users/show.haml_spec.rb +16 -0
- data/spec/views/admin/users/suspend.js.haml_spec.rb +20 -0
- data/spec/views/admin/users/update.js.haml_spec.rb +36 -0
- data/spec/views/application/auto_complete.haml_spec.rb +46 -0
- data/spec/views/authentications/new.haml_spec.rb +29 -0
- data/spec/views/campaigns/_edit.haml_spec.rb +42 -0
- data/spec/views/campaigns/_new.haml_spec.rb +39 -0
- data/spec/views/campaigns/create.js.haml_spec.rb +48 -0
- data/spec/views/campaigns/destroy.js.haml_spec.rb +30 -0
- data/spec/views/campaigns/edit.js.haml_spec.rb +64 -0
- data/spec/views/campaigns/index.haml_spec.rb +35 -0
- data/spec/views/campaigns/index.js.haml_spec.rb +32 -0
- data/spec/views/campaigns/new.js.haml_spec.rb +48 -0
- data/spec/views/campaigns/show.haml_spec.rb +33 -0
- data/spec/views/campaigns/update.js.haml_spec.rb +80 -0
- data/spec/views/comments/edit.js.haml_spec.rb +29 -0
- data/spec/views/contacts/_edit.haml_spec.rb +71 -0
- data/spec/views/contacts/_new.haml_spec.rb +50 -0
- data/spec/views/contacts/create.js.haml_spec.rb +64 -0
- data/spec/views/contacts/destroy.js.haml_spec.rb +43 -0
- data/spec/views/contacts/edit.js.haml_spec.rb +72 -0
- data/spec/views/contacts/index.haml_spec.rb +35 -0
- data/spec/views/contacts/index.js.html_spec.rb +32 -0
- data/spec/views/contacts/new.js.haml_spec.rb +45 -0
- data/spec/views/contacts/show.haml_spec.rb +30 -0
- data/spec/views/contacts/update.js.haml_spec.rb +130 -0
- data/spec/views/home/index.haml_spec.rb +34 -0
- data/spec/views/home/index.js.haml_spec.rb +31 -0
- data/spec/views/home/options.js.haml_spec.rb +51 -0
- data/spec/views/leads/_convert.haml_spec.rb +28 -0
- data/spec/views/leads/_edit.haml_spec.rb +45 -0
- data/spec/views/leads/_new.haml_spec.rb +43 -0
- data/spec/views/leads/_sidebar_show.haml_spec.rb +29 -0
- data/spec/views/leads/convert.js.haml_spec.rb +77 -0
- data/spec/views/leads/create.js.haml_spec.rb +68 -0
- data/spec/views/leads/destroy.js.haml_spec.rb +47 -0
- data/spec/views/leads/edit.js.haml_spec.rb +74 -0
- data/spec/views/leads/index.haml_spec.rb +35 -0
- data/spec/views/leads/index.js.haml_spec.rb +32 -0
- data/spec/views/leads/new.js.haml_spec.rb +45 -0
- data/spec/views/leads/promote.js.haml_spec.rb +141 -0
- data/spec/views/leads/reject.js.haml_spec.rb +46 -0
- data/spec/views/leads/show.haml_spec.rb +27 -0
- data/spec/views/leads/update.js.haml_spec.rb +124 -0
- data/spec/views/opportunities/_edit.haml_spec.rb +68 -0
- data/spec/views/opportunities/_new.haml_spec.rb +49 -0
- data/spec/views/opportunities/create.js.haml_spec.rb +86 -0
- data/spec/views/opportunities/destroy.js.haml_spec.rb +65 -0
- data/spec/views/opportunities/edit.js.haml_spec.rb +73 -0
- data/spec/views/opportunities/index.haml_spec.rb +36 -0
- data/spec/views/opportunities/index.js.haml_spec.rb +33 -0
- data/spec/views/opportunities/new.js.haml_spec.rb +52 -0
- data/spec/views/opportunities/show.haml_spec.rb +30 -0
- data/spec/views/opportunities/update.js.haml_spec.rb +149 -0
- data/spec/views/tasks/_edit.haml_spec.rb +48 -0
- data/spec/views/tasks/complete.js.haml_spec.rb +66 -0
- data/spec/views/tasks/create.js.haml_spec.rb +119 -0
- data/spec/views/tasks/destroy.js.haml_spec.rb +56 -0
- data/spec/views/tasks/edit.js.haml_spec.rb +74 -0
- data/spec/views/tasks/index.haml_spec.rb +44 -0
- data/spec/views/tasks/new.js.haml_spec.rb +47 -0
- data/spec/views/tasks/uncomplete.js.haml_spec.rb +44 -0
- data/spec/views/tasks/update.js.haml_spec.rb +141 -0
- data/spec/views/users/avatar.js.haml_spec.rb +33 -0
- data/spec/views/users/change_password.js.haml_spec.rb +50 -0
- data/spec/views/users/edit.js.haml_spec.rb +33 -0
- data/spec/views/users/password.js.haml_spec.rb +34 -0
- data/spec/views/users/update.js.haml_spec.rb +45 -0
- data/spec/views/users/upload_avatar.js.haml_spec.rb +44 -0
- data/vendor/assets/images/calendar_date_select/calendar.gif +0 -0
- data/vendor/assets/images/chosen-sprite.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_0_aaaaaa_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_0_eeeeee_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_100_ffffff_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_25_3875d7_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_55_ffffff_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_65_3875d7_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_flat_75_ffffff_40x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-bg_highlight-soft_50_dddddd_1x100.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-icons_0073ea_256x240.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-icons_466bb1_256x240.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-icons_ff0084_256x240.png +0 -0
- data/vendor/assets/images/jquery-ui/ui-icons_ffffff_256x240.png +0 -0
- data/vendor/assets/javascripts/ajax-chosen-jquery.js +2 -0
- data/vendor/assets/javascripts/ajax-chosen.jquery.coffee +171 -0
- data/vendor/assets/javascripts/chosen-jquery.js +3 -0
- data/vendor/assets/javascripts/chosen.jquery.coffee +574 -0
- data/vendor/assets/javascripts/jquery.disable.js +23 -0
- data/vendor/assets/javascripts/jquery_timeago/index.js +19 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.cz.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.de.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.en-GB.js +20 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.en-US.js +20 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.es-CL.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.es.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.fr-CA.js +17 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.fr.js +17 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.it.js +16 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.ja.js +19 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.js +194 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.nl.js +20 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.pl.js +31 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.pt-BR.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.ru.js +34 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.sv-SE.js +18 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.th.js +20 -0
- data/vendor/assets/javascripts/jquery_timeago/jquery.timeago.zh-CN.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-addon.js +1882 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-af.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-ca.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-cz.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-de.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-el.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-es-CL.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-es.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-et.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-fi.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-fr-CA.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-fr.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-gl.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-he.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-hu.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-id.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-it.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-ja.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-ko.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-lt.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-nl.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-no.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-pl.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-pt-BR.js +22 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-pt.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-ro.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-ru.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-sk.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-sv-SE.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-th.js +19 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-tr.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-vi.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-zh-CN.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery-ui-timepicker-zh-TW.js +20 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-cz.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-de.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-en-GB.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-es-CL.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-es.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-fr-CA.js +21 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-fr.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-it.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-ja.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-nl.js +21 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-pl.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-pt-BR.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-pt.js +21 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-ru.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-sv-SE.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-th.js +23 -0
- data/vendor/assets/javascripts/jquery_ui_datepicker/jquery.ui.datepicker-zh-CN.js +23 -0
- data/vendor/assets/javascripts/lib/abstract-chosen.coffee +110 -0
- data/vendor/assets/javascripts/lib/select-parser.coffee +51 -0
- data/vendor/assets/javascripts/rating.js +162 -0
- data/vendor/assets/javascripts/textarea_autocomplete.js +605 -0
- data/vendor/assets/stylesheets/chosen.css.sass +361 -0
- data/vendor/assets/stylesheets/jquery-ui.custom.scss +566 -0
- data/vendor/assets/stylesheets/modalbox.css +107 -0
- data/vendor/gems/globby-0.1.2/LICENSE.txt +20 -0
- data/vendor/gems/globby-0.1.2/README.md +65 -0
- data/vendor/gems/globby-0.1.2/Rakefile +9 -0
- data/vendor/gems/globby-0.1.2/lib/globby.rb +47 -0
- data/vendor/gems/globby-0.1.2/lib/globby/glob.rb +90 -0
- data/vendor/gems/globby-0.1.2/lib/globby/globject.rb +18 -0
- data/vendor/gems/globby-0.1.2/lib/globby/result.rb +20 -0
- data/vendor/gems/globby-0.1.2/spec/gitignore_spec.rb +109 -0
- data/vendor/gems/globby-0.1.2/spec/globby_spec.rb +93 -0
- data/zeus.json +22 -0
- metadata +1566 -0
|
@@ -0,0 +1,1007 @@
|
|
|
1
|
+
# Copyright (c) 2008-2013 Michael Dvorkin and contributors.
|
|
2
|
+
#
|
|
3
|
+
# Fat Free CRM is freely distributable under the terms of MIT license.
|
|
4
|
+
# See MIT-LICENSE file or http://www.opensource.org/licenses/mit-license.php
|
|
5
|
+
#------------------------------------------------------------------------------
|
|
6
|
+
require File.expand_path(File.dirname(__FILE__) + '/../../spec_helper')
|
|
7
|
+
|
|
8
|
+
describe LeadsController do
|
|
9
|
+
before(:each) do
|
|
10
|
+
require_user
|
|
11
|
+
set_current_tab(:leads)
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
# GET /leads
|
|
15
|
+
# GET /leads.xml AJAX and HTML
|
|
16
|
+
#----------------------------------------------------------------------------
|
|
17
|
+
describe "responding to GET index" do
|
|
18
|
+
it "should expose all leads as @leads and render [index] template" do
|
|
19
|
+
@leads = [FactoryGirl.create(:lead, user: current_user)]
|
|
20
|
+
|
|
21
|
+
get :index
|
|
22
|
+
expect(assigns[:leads]).to eq(@leads)
|
|
23
|
+
expect(response).to render_template("leads/index")
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
it "should collect the data for the leads sidebar" do
|
|
27
|
+
@leads = [FactoryGirl.create(:lead, user: current_user)]
|
|
28
|
+
@status = Setting.lead_status.dup
|
|
29
|
+
|
|
30
|
+
get :index
|
|
31
|
+
expect(assigns[:lead_status_total].keys.map(&:to_sym) - (@status << :all << :other)).to eq([])
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
it "should filter out leads by status" do
|
|
35
|
+
controller.session[:leads_filter] = "new,contacted"
|
|
36
|
+
@leads = [
|
|
37
|
+
FactoryGirl.create(:lead, status: "new", user: current_user),
|
|
38
|
+
FactoryGirl.create(:lead, status: "contacted", user: current_user)
|
|
39
|
+
]
|
|
40
|
+
|
|
41
|
+
# This one should be filtered out.
|
|
42
|
+
FactoryGirl.create(:lead, status: "rejected", user: current_user)
|
|
43
|
+
|
|
44
|
+
get :index
|
|
45
|
+
# Note: can't compare campaigns directly because of BigDecimals.
|
|
46
|
+
expect(assigns[:leads].size).to eq(2)
|
|
47
|
+
expect(assigns[:leads].map(&:status).sort).to eq(%w(contacted new))
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
it "should perform lookup using query string" do
|
|
51
|
+
@billy_bones = FactoryGirl.create(:lead, user: current_user, first_name: "Billy", last_name: "Bones")
|
|
52
|
+
@captain_flint = FactoryGirl.create(:lead, user: current_user, first_name: "Captain", last_name: "Flint")
|
|
53
|
+
|
|
54
|
+
get :index, query: "bill"
|
|
55
|
+
expect(assigns[:leads]).to eq([@billy_bones])
|
|
56
|
+
expect(assigns[:current_query]).to eq("bill")
|
|
57
|
+
expect(session[:leads_current_query]).to eq("bill")
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
describe "AJAX pagination" do
|
|
61
|
+
it "should pick up page number from params" do
|
|
62
|
+
@leads = [FactoryGirl.create(:lead, user: current_user)]
|
|
63
|
+
xhr :get, :index, page: 42
|
|
64
|
+
|
|
65
|
+
expect(assigns[:current_page].to_i).to eq(42)
|
|
66
|
+
expect(assigns[:leads]).to eq([]) # page #42 should be empty if there's only one lead ;-)
|
|
67
|
+
expect(session[:leads_current_page].to_i).to eq(42)
|
|
68
|
+
expect(response).to render_template("leads/index")
|
|
69
|
+
end
|
|
70
|
+
|
|
71
|
+
it "should pick up saved page number from session" do
|
|
72
|
+
session[:leads_current_page] = 42
|
|
73
|
+
session[:leads_current_query] = "bill"
|
|
74
|
+
@leads = [FactoryGirl.create(:lead, user: current_user)]
|
|
75
|
+
xhr :get, :index, query: "bill"
|
|
76
|
+
|
|
77
|
+
expect(assigns[:current_page]).to eq(42)
|
|
78
|
+
expect(assigns[:leads]).to eq([])
|
|
79
|
+
expect(response).to render_template("leads/index")
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
it "should reset current_page when query is altered" do
|
|
83
|
+
session[:leads_current_page] = 42
|
|
84
|
+
session[:leads_current_query] = "bill"
|
|
85
|
+
@leads = [FactoryGirl.create(:lead, user: current_user)]
|
|
86
|
+
xhr :get, :index
|
|
87
|
+
|
|
88
|
+
expect(assigns[:current_page]).to eq(1)
|
|
89
|
+
expect(assigns[:leads]).to eq(@leads)
|
|
90
|
+
expect(response).to render_template("leads/index")
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
describe "with mime type of JSON" do
|
|
95
|
+
it "should render all leads as JSON" do
|
|
96
|
+
expect(@controller).to receive(:get_leads).and_return(leads = double("Array of Leads"))
|
|
97
|
+
expect(leads).to receive(:to_json).and_return("generated JSON")
|
|
98
|
+
|
|
99
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
|
100
|
+
get :index
|
|
101
|
+
expect(response.body).to eq("generated JSON")
|
|
102
|
+
end
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
describe "with mime type of XML" do
|
|
106
|
+
it "should render all leads as xml" do
|
|
107
|
+
expect(@controller).to receive(:get_leads).and_return(leads = double("Array of Leads"))
|
|
108
|
+
expect(leads).to receive(:to_xml).and_return("generated XML")
|
|
109
|
+
|
|
110
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
|
111
|
+
get :index
|
|
112
|
+
expect(response.body).to eq("generated XML")
|
|
113
|
+
end
|
|
114
|
+
end
|
|
115
|
+
end
|
|
116
|
+
|
|
117
|
+
# GET /leads/1
|
|
118
|
+
# GET /leads/1.xml HTML
|
|
119
|
+
#----------------------------------------------------------------------------
|
|
120
|
+
describe "responding to GET show" do
|
|
121
|
+
describe "with mime type of HTML" do
|
|
122
|
+
before(:each) do
|
|
123
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user)
|
|
124
|
+
@comment = Comment.new
|
|
125
|
+
end
|
|
126
|
+
|
|
127
|
+
it "should expose the requested lead as @lead and render [show] template" do
|
|
128
|
+
get :show, id: 42
|
|
129
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
130
|
+
expect(assigns[:comment].attributes).to eq(@comment.attributes)
|
|
131
|
+
expect(response).to render_template("leads/show")
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
it "should update an activity when viewing the lead" do
|
|
135
|
+
get :show, id: @lead.id
|
|
136
|
+
expect(@lead.versions.last.event).to eq('view')
|
|
137
|
+
end
|
|
138
|
+
end
|
|
139
|
+
|
|
140
|
+
describe "with mime type of JSON" do
|
|
141
|
+
it "should render the requested lead as JSON" do
|
|
142
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user)
|
|
143
|
+
expect(Lead).to receive(:find).and_return(@lead)
|
|
144
|
+
expect(@lead).to receive(:to_json).and_return("generated JSON")
|
|
145
|
+
|
|
146
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
|
147
|
+
get :show, id: 42
|
|
148
|
+
expect(response.body).to eq("generated JSON")
|
|
149
|
+
end
|
|
150
|
+
end
|
|
151
|
+
|
|
152
|
+
describe "with mime type of XML" do
|
|
153
|
+
it "should render the requested lead as xml" do
|
|
154
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user)
|
|
155
|
+
expect(Lead).to receive(:find).and_return(@lead)
|
|
156
|
+
expect(@lead).to receive(:to_xml).and_return("generated XML")
|
|
157
|
+
|
|
158
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
|
159
|
+
get :show, id: 42
|
|
160
|
+
expect(response.body).to eq("generated XML")
|
|
161
|
+
end
|
|
162
|
+
end
|
|
163
|
+
|
|
164
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
165
|
+
it "should redirect to lead index if the lead got deleted" do
|
|
166
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
167
|
+
@lead.destroy
|
|
168
|
+
|
|
169
|
+
get :show, id: @lead.id
|
|
170
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
171
|
+
expect(response).to redirect_to(leads_path)
|
|
172
|
+
end
|
|
173
|
+
|
|
174
|
+
it "should redirect to lead index if the lead is protected" do
|
|
175
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
176
|
+
|
|
177
|
+
get :show, id: @private.id
|
|
178
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
179
|
+
expect(response).to redirect_to(leads_path)
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
it "should return 404 (Not Found) JSON error" do
|
|
183
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
184
|
+
@lead.destroy
|
|
185
|
+
request.env["HTTP_ACCEPT"] = "application/json"
|
|
186
|
+
|
|
187
|
+
get :show, id: @lead.id
|
|
188
|
+
expect(response.code).to eq("404") # :not_found
|
|
189
|
+
end
|
|
190
|
+
|
|
191
|
+
it "should return 404 (Not Found) XML error" do
|
|
192
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
193
|
+
@lead.destroy
|
|
194
|
+
request.env["HTTP_ACCEPT"] = "application/xml"
|
|
195
|
+
|
|
196
|
+
get :show, id: @lead.id
|
|
197
|
+
expect(response.code).to eq("404") # :not_found
|
|
198
|
+
end
|
|
199
|
+
end
|
|
200
|
+
end
|
|
201
|
+
|
|
202
|
+
# GET /leads/new
|
|
203
|
+
# GET /leads/new.xml AJAX
|
|
204
|
+
#----------------------------------------------------------------------------
|
|
205
|
+
describe "responding to GET new" do
|
|
206
|
+
it "should expose a new lead as @lead and render [new] template" do
|
|
207
|
+
@lead = FactoryGirl.build(:lead, user: current_user, campaign: nil)
|
|
208
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
209
|
+
@campaigns = [FactoryGirl.create(:campaign, user: current_user)]
|
|
210
|
+
|
|
211
|
+
xhr :get, :new
|
|
212
|
+
expect(assigns[:lead].attributes).to eq(@lead.attributes)
|
|
213
|
+
expect(assigns[:campaigns]).to eq(@campaigns)
|
|
214
|
+
expect(response).to render_template("leads/new")
|
|
215
|
+
end
|
|
216
|
+
|
|
217
|
+
it "should create related object when necessary" do
|
|
218
|
+
@campaign = FactoryGirl.create(:campaign, id: 123)
|
|
219
|
+
|
|
220
|
+
xhr :get, :new, related: "campaign_123"
|
|
221
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
222
|
+
end
|
|
223
|
+
|
|
224
|
+
describe "(when creating related lead)" do
|
|
225
|
+
it "should redirect to parent asset's index page with the message if parent asset got deleted" do
|
|
226
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
227
|
+
@campaign.destroy
|
|
228
|
+
|
|
229
|
+
xhr :get, :new, related: "campaign_#{@campaign.id}"
|
|
230
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
231
|
+
expect(response.body).to eq('window.location.href = "/campaigns";')
|
|
232
|
+
end
|
|
233
|
+
|
|
234
|
+
it "should redirect to parent asset's index page with the message if parent asset got protected" do
|
|
235
|
+
@campaign = FactoryGirl.create(:campaign, access: "Private")
|
|
236
|
+
|
|
237
|
+
xhr :get, :new, related: "campaign_#{@campaign.id}"
|
|
238
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
239
|
+
expect(response.body).to eq('window.location.href = "/campaigns";')
|
|
240
|
+
end
|
|
241
|
+
end
|
|
242
|
+
end
|
|
243
|
+
|
|
244
|
+
# GET /leads/1/edit AJAX
|
|
245
|
+
#----------------------------------------------------------------------------
|
|
246
|
+
describe "responding to GET edit" do
|
|
247
|
+
it "should expose the requested lead as @lead and render [edit] template" do
|
|
248
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user, campaign: nil)
|
|
249
|
+
@campaigns = [FactoryGirl.create(:campaign, user: current_user)]
|
|
250
|
+
|
|
251
|
+
xhr :get, :edit, id: 42
|
|
252
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
253
|
+
expect(assigns[:campaigns]).to eq(@campaigns)
|
|
254
|
+
expect(response).to render_template("leads/edit")
|
|
255
|
+
end
|
|
256
|
+
|
|
257
|
+
it "should find previous lead when necessary" do
|
|
258
|
+
@lead = FactoryGirl.create(:lead, id: 42)
|
|
259
|
+
@previous = FactoryGirl.create(:lead, id: 321)
|
|
260
|
+
|
|
261
|
+
xhr :get, :edit, id: 42, previous: 321
|
|
262
|
+
expect(assigns[:previous]).to eq(@previous)
|
|
263
|
+
end
|
|
264
|
+
|
|
265
|
+
describe "lead got deleted or is otherwise unavailable" do
|
|
266
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
267
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
268
|
+
@lead.destroy
|
|
269
|
+
|
|
270
|
+
xhr :get, :edit, id: @lead.id
|
|
271
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
272
|
+
expect(response.body).to eq("window.location.reload();")
|
|
273
|
+
end
|
|
274
|
+
|
|
275
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
276
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
277
|
+
|
|
278
|
+
xhr :get, :edit, id: @private.id
|
|
279
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
280
|
+
expect(response.body).to eq("window.location.reload();")
|
|
281
|
+
end
|
|
282
|
+
end
|
|
283
|
+
|
|
284
|
+
describe "(previous lead got deleted or is otherwise unavailable)" do
|
|
285
|
+
before(:each) do
|
|
286
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
287
|
+
@previous = FactoryGirl.create(:lead, user: FactoryGirl.create(:user))
|
|
288
|
+
end
|
|
289
|
+
|
|
290
|
+
it "should notify the view if previous lead got deleted" do
|
|
291
|
+
@previous.destroy
|
|
292
|
+
|
|
293
|
+
xhr :get, :edit, id: @lead.id, previous: @previous.id
|
|
294
|
+
expect(flash[:warning]).to eq(nil) # no warning, just silently remove the div
|
|
295
|
+
expect(assigns[:previous]).to eq(@previous.id)
|
|
296
|
+
expect(response).to render_template("leads/edit")
|
|
297
|
+
end
|
|
298
|
+
|
|
299
|
+
it "should notify the view if previous lead got protected" do
|
|
300
|
+
@previous.update_attribute(:access, "Private")
|
|
301
|
+
|
|
302
|
+
xhr :get, :edit, id: @lead.id, previous: @previous.id
|
|
303
|
+
expect(flash[:warning]).to eq(nil)
|
|
304
|
+
expect(assigns[:previous]).to eq(@previous.id)
|
|
305
|
+
expect(response).to render_template("leads/edit")
|
|
306
|
+
end
|
|
307
|
+
end
|
|
308
|
+
end
|
|
309
|
+
|
|
310
|
+
# POST /leads
|
|
311
|
+
# POST /leads.xml AJAX
|
|
312
|
+
#----------------------------------------------------------------------------
|
|
313
|
+
describe "responding to POST create" do
|
|
314
|
+
describe "with valid params" do
|
|
315
|
+
it "should expose a newly created lead as @lead and render [create] template" do
|
|
316
|
+
@lead = FactoryGirl.build(:lead, user: current_user, campaign: nil)
|
|
317
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
318
|
+
@campaigns = [FactoryGirl.create(:campaign, user: current_user)]
|
|
319
|
+
|
|
320
|
+
xhr :post, :create, lead: { first_name: "Billy", last_name: "Bones" }
|
|
321
|
+
expect(assigns(:lead)).to eq(@lead)
|
|
322
|
+
expect(assigns(:campaigns)).to eq(@campaigns)
|
|
323
|
+
expect(assigns[:lead_status_total]).to be_nil
|
|
324
|
+
expect(response).to render_template("leads/create")
|
|
325
|
+
end
|
|
326
|
+
|
|
327
|
+
it "should copy selected campaign permissions unless asked otherwise" do
|
|
328
|
+
he = FactoryGirl.create(:user, id: 7)
|
|
329
|
+
she = FactoryGirl.create(:user, id: 8)
|
|
330
|
+
@campaign = FactoryGirl.build(:campaign, access: "Shared")
|
|
331
|
+
@campaign.permissions << FactoryGirl.build(:permission, user: he, asset: @campaign)
|
|
332
|
+
@campaign.permissions << FactoryGirl.build(:permission, user: she, asset: @campaign)
|
|
333
|
+
@campaign.save
|
|
334
|
+
|
|
335
|
+
@lead = FactoryGirl.build(:lead, campaign: @campaign, user: current_user, access: "Shared")
|
|
336
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
337
|
+
|
|
338
|
+
xhr :post, :create, lead: { first_name: "Billy", last_name: "Bones", access: "Campaign", user_ids: %w(7 8) }, campaign: @campaign.id
|
|
339
|
+
expect(assigns(:lead)).to eq(@lead)
|
|
340
|
+
expect(@lead.reload.access).to eq("Shared")
|
|
341
|
+
expect(@lead.permissions.map(&:user_id).sort).to eq([7, 8])
|
|
342
|
+
expect(@lead.permissions.map(&:asset_id)).to eq([@lead.id, @lead.id])
|
|
343
|
+
expect(@lead.permissions.map(&:asset_type)).to eq(%w(Lead Lead))
|
|
344
|
+
end
|
|
345
|
+
|
|
346
|
+
it "should get the data to update leads sidebar if called from leads index" do
|
|
347
|
+
@lead = FactoryGirl.build(:lead, user: current_user, campaign: nil)
|
|
348
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
349
|
+
|
|
350
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
351
|
+
xhr :post, :create, lead: { first_name: "Billy", last_name: "Bones" }
|
|
352
|
+
expect(assigns[:lead_status_total]).to be_an_instance_of(HashWithIndifferentAccess)
|
|
353
|
+
end
|
|
354
|
+
|
|
355
|
+
it "should reload leads to update pagination if called from leads index" do
|
|
356
|
+
@lead = FactoryGirl.build(:lead, user: current_user, campaign: nil)
|
|
357
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
358
|
+
|
|
359
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
360
|
+
xhr :post, :create, lead: { first_name: "Billy", last_name: "Bones" }
|
|
361
|
+
expect(assigns[:leads]).to eq([@lead])
|
|
362
|
+
end
|
|
363
|
+
|
|
364
|
+
it "should reload lead campaign if called from campaign landing page" do
|
|
365
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
366
|
+
@lead = FactoryGirl.build(:lead, user: current_user, campaign: @campaign)
|
|
367
|
+
|
|
368
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
|
369
|
+
xhr :put, :create, lead: { first_name: "Billy", last_name: "Bones" }, campaign: @campaign.id
|
|
370
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
371
|
+
end
|
|
372
|
+
|
|
373
|
+
it "should add a new comment to the newly created lead when specified" do
|
|
374
|
+
@lead = FactoryGirl.create(:lead)
|
|
375
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
376
|
+
xhr :post, :create, lead: { first_name: "Test", last_name: "Lead" }, comment_body: "This is an important lead."
|
|
377
|
+
expect(@lead.reload.comments.map(&:comment)).to include("This is an important lead.")
|
|
378
|
+
end
|
|
379
|
+
end
|
|
380
|
+
|
|
381
|
+
describe "with invalid params" do
|
|
382
|
+
it "should expose a newly created but unsaved lead as @lead and still render [create] template" do
|
|
383
|
+
@lead = FactoryGirl.build(:lead, user: current_user, first_name: nil, campaign: nil)
|
|
384
|
+
allow(Lead).to receive(:new).and_return(@lead)
|
|
385
|
+
@campaigns = [FactoryGirl.create(:campaign, user: current_user)]
|
|
386
|
+
|
|
387
|
+
xhr :post, :create, lead: { first_name: nil }
|
|
388
|
+
expect(assigns(:lead)).to eq(@lead)
|
|
389
|
+
expect(assigns(:campaigns)).to eq(@campaigns)
|
|
390
|
+
expect(assigns[:lead_status_total]).to eq(nil)
|
|
391
|
+
expect(response).to render_template("leads/create")
|
|
392
|
+
end
|
|
393
|
+
end
|
|
394
|
+
end
|
|
395
|
+
|
|
396
|
+
# PUT /leads/1
|
|
397
|
+
# PUT /leads/1.xml
|
|
398
|
+
#----------------------------------------------------------------------------
|
|
399
|
+
describe "responding to PUT update" do
|
|
400
|
+
describe "with valid params" do
|
|
401
|
+
it "should update the requested lead, expose it as @lead, and render [update] template" do
|
|
402
|
+
@lead = FactoryGirl.create(:lead, first_name: "Billy", user: current_user)
|
|
403
|
+
|
|
404
|
+
xhr :put, :update, id: @lead.id, lead: { first_name: "Bones" }
|
|
405
|
+
expect(@lead.reload.first_name).to eq("Bones")
|
|
406
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
407
|
+
expect(assigns[:lead_status_total]).to eq(nil)
|
|
408
|
+
expect(response).to render_template("leads/update")
|
|
409
|
+
end
|
|
410
|
+
|
|
411
|
+
it "should update lead status" do
|
|
412
|
+
@lead = FactoryGirl.create(:lead, status: "new", user: current_user)
|
|
413
|
+
|
|
414
|
+
xhr :put, :update, id: @lead.id, lead: { status: "rejected" }
|
|
415
|
+
expect(@lead.reload.status).to eq("rejected")
|
|
416
|
+
end
|
|
417
|
+
|
|
418
|
+
it "should update lead source" do
|
|
419
|
+
@lead = FactoryGirl.create(:lead, source: "campaign", user: current_user)
|
|
420
|
+
|
|
421
|
+
xhr :put, :update, id: @lead.id, lead: { source: "cald_call" }
|
|
422
|
+
expect(@lead.reload.source).to eq("cald_call")
|
|
423
|
+
end
|
|
424
|
+
|
|
425
|
+
it "should update lead campaign" do
|
|
426
|
+
@campaigns = { old: FactoryGirl.create(:campaign), new: FactoryGirl.create(:campaign) }
|
|
427
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaigns[:old])
|
|
428
|
+
|
|
429
|
+
xhr :put, :update, id: @lead.id, lead: { campaign_id: @campaigns[:new].id }
|
|
430
|
+
expect(@lead.reload.campaign).to eq(@campaigns[:new])
|
|
431
|
+
end
|
|
432
|
+
|
|
433
|
+
it "should decrement campaign leads count if campaign has been removed" do
|
|
434
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
435
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaign)
|
|
436
|
+
@count = @campaign.reload.leads_count
|
|
437
|
+
|
|
438
|
+
xhr :put, :update, id: @lead, lead: { campaign_id: nil }
|
|
439
|
+
expect(@lead.reload.campaign).to eq(nil)
|
|
440
|
+
expect(@campaign.reload.leads_count).to eq(@count - 1)
|
|
441
|
+
end
|
|
442
|
+
|
|
443
|
+
it "should increment campaign leads count if campaign has been assigned" do
|
|
444
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
445
|
+
@lead = FactoryGirl.create(:lead, campaign: nil)
|
|
446
|
+
@count = @campaign.leads_count
|
|
447
|
+
|
|
448
|
+
xhr :put, :update, id: @lead, lead: { campaign_id: @campaign.id }
|
|
449
|
+
expect(@lead.reload.campaign).to eq(@campaign)
|
|
450
|
+
expect(@campaign.reload.leads_count).to eq(@count + 1)
|
|
451
|
+
end
|
|
452
|
+
|
|
453
|
+
it "should update both campaign leads counts if reassigned to a new campaign" do
|
|
454
|
+
@campaigns = { old: FactoryGirl.create(:campaign), new: FactoryGirl.create(:campaign) }
|
|
455
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaigns[:old])
|
|
456
|
+
@counts = { old: @campaigns[:old].reload.leads_count, new: @campaigns[:new].leads_count }
|
|
457
|
+
|
|
458
|
+
xhr :put, :update, id: @lead, lead: { campaign_id: @campaigns[:new].id }
|
|
459
|
+
expect(@lead.reload.campaign).to eq(@campaigns[:new])
|
|
460
|
+
expect(@campaigns[:old].reload.leads_count).to eq(@counts[:old] - 1)
|
|
461
|
+
expect(@campaigns[:new].reload.leads_count).to eq(@counts[:new] + 1)
|
|
462
|
+
end
|
|
463
|
+
|
|
464
|
+
it "should update shared permissions for the lead" do
|
|
465
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
466
|
+
he = FactoryGirl.create(:user, id: 7)
|
|
467
|
+
she = FactoryGirl.create(:user, id: 8)
|
|
468
|
+
|
|
469
|
+
xhr :put, :update, id: @lead.id, lead: { access: "Shared", user_ids: %w(7 8) }
|
|
470
|
+
expect(@lead.user_ids.sort).to eq([7, 8])
|
|
471
|
+
end
|
|
472
|
+
|
|
473
|
+
it "should get the data for leads sidebar when called from leads index" do
|
|
474
|
+
@lead = FactoryGirl.create(:lead)
|
|
475
|
+
|
|
476
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
477
|
+
xhr :put, :update, id: @lead.id, lead: { first_name: "Billy" }
|
|
478
|
+
expect(assigns[:lead_status_total]).not_to be_nil
|
|
479
|
+
expect(assigns[:lead_status_total]).to be_an_instance_of(HashWithIndifferentAccess)
|
|
480
|
+
end
|
|
481
|
+
|
|
482
|
+
it "should reload lead campaign if called from campaign landing page" do
|
|
483
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
484
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaign)
|
|
485
|
+
|
|
486
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
|
487
|
+
xhr :put, :update, id: @lead.id, lead: { first_name: "Hello" }
|
|
488
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
489
|
+
end
|
|
490
|
+
|
|
491
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
492
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
493
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
494
|
+
@lead.destroy
|
|
495
|
+
|
|
496
|
+
xhr :put, :update, id: @lead.id
|
|
497
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
498
|
+
expect(response.body).to eq("window.location.reload();")
|
|
499
|
+
end
|
|
500
|
+
|
|
501
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
502
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
503
|
+
|
|
504
|
+
xhr :put, :update, id: @private.id
|
|
505
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
506
|
+
expect(response.body).to eq("window.location.reload();")
|
|
507
|
+
end
|
|
508
|
+
end
|
|
509
|
+
end
|
|
510
|
+
|
|
511
|
+
describe "with invalid params" do
|
|
512
|
+
it "should not update the lead, but still expose it as @lead and render [update] template" do
|
|
513
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user, campaign: nil)
|
|
514
|
+
@campaigns = [FactoryGirl.create(:campaign, user: current_user)]
|
|
515
|
+
|
|
516
|
+
xhr :put, :update, id: 42, lead: { first_name: nil }
|
|
517
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
518
|
+
expect(assigns[:campaigns]).to eq(@campaigns)
|
|
519
|
+
expect(response).to render_template("leads/update")
|
|
520
|
+
end
|
|
521
|
+
end
|
|
522
|
+
end
|
|
523
|
+
|
|
524
|
+
# DELETE /leads/1
|
|
525
|
+
# DELETE /leads/1.xml AJAX and HTML
|
|
526
|
+
#----------------------------------------------------------------------------
|
|
527
|
+
describe "responding to DELETE destroy" do
|
|
528
|
+
before(:each) do
|
|
529
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
530
|
+
end
|
|
531
|
+
|
|
532
|
+
describe "AJAX request" do
|
|
533
|
+
it "should destroy the requested lead and render [destroy] template" do
|
|
534
|
+
xhr :delete, :destroy, id: @lead.id
|
|
535
|
+
|
|
536
|
+
expect(assigns[:leads]).to eq(nil) # @lead got deleted
|
|
537
|
+
expect { Lead.find(@lead.id) }.to raise_error(ActiveRecord::RecordNotFound)
|
|
538
|
+
expect(response).to render_template("leads/destroy")
|
|
539
|
+
end
|
|
540
|
+
|
|
541
|
+
describe "when called from Leads index page" do
|
|
542
|
+
before(:each) do
|
|
543
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
544
|
+
end
|
|
545
|
+
|
|
546
|
+
it "should get data for the sidebar" do
|
|
547
|
+
@another_lead = FactoryGirl.create(:lead, user: current_user)
|
|
548
|
+
|
|
549
|
+
xhr :delete, :destroy, id: @lead.id
|
|
550
|
+
expect(assigns[:leads]).to eq([@another_lead]) # @lead got deleted
|
|
551
|
+
expect(assigns[:lead_status_total]).not_to be_nil
|
|
552
|
+
expect(assigns[:lead_status_total]).to be_an_instance_of(HashWithIndifferentAccess)
|
|
553
|
+
expect(response).to render_template("leads/destroy")
|
|
554
|
+
end
|
|
555
|
+
|
|
556
|
+
it "should try previous page and render index action if current page has no leads" do
|
|
557
|
+
session[:leads_current_page] = 42
|
|
558
|
+
|
|
559
|
+
xhr :delete, :destroy, id: @lead.id
|
|
560
|
+
expect(session[:leads_current_page]).to eq(41)
|
|
561
|
+
expect(response).to render_template("leads/index")
|
|
562
|
+
end
|
|
563
|
+
|
|
564
|
+
it "should render index action when deleting last lead" do
|
|
565
|
+
session[:leads_current_page] = 1
|
|
566
|
+
|
|
567
|
+
xhr :delete, :destroy, id: @lead.id
|
|
568
|
+
expect(session[:leads_current_page]).to eq(1)
|
|
569
|
+
expect(response).to render_template("leads/index")
|
|
570
|
+
end
|
|
571
|
+
end
|
|
572
|
+
|
|
573
|
+
describe "when called from campaign landing page" do
|
|
574
|
+
before(:each) do
|
|
575
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
576
|
+
@lead = FactoryGirl.create(:lead, user: current_user, campaign: @campaign)
|
|
577
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
|
578
|
+
end
|
|
579
|
+
|
|
580
|
+
it "should reset current page to 1" do
|
|
581
|
+
xhr :delete, :destroy, id: @lead.id
|
|
582
|
+
expect(session[:leads_current_page]).to eq(1)
|
|
583
|
+
expect(response).to render_template("leads/destroy")
|
|
584
|
+
end
|
|
585
|
+
|
|
586
|
+
it "should reload campaiign to be able to refresh its summary" do
|
|
587
|
+
xhr :delete, :destroy, id: @lead.id
|
|
588
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
589
|
+
expect(response).to render_template("leads/destroy")
|
|
590
|
+
end
|
|
591
|
+
end
|
|
592
|
+
|
|
593
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
594
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
595
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
596
|
+
@lead.destroy
|
|
597
|
+
|
|
598
|
+
xhr :delete, :destroy, id: @lead.id
|
|
599
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
600
|
+
expect(response.body).to eq("window.location.reload();")
|
|
601
|
+
end
|
|
602
|
+
|
|
603
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
604
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
605
|
+
|
|
606
|
+
xhr :delete, :destroy, id: @private.id
|
|
607
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
608
|
+
expect(response.body).to eq("window.location.reload();")
|
|
609
|
+
end
|
|
610
|
+
end
|
|
611
|
+
end
|
|
612
|
+
|
|
613
|
+
describe "HTML request" do
|
|
614
|
+
it "should redirect to Leads index when a lead gets deleted from its landing page" do
|
|
615
|
+
delete :destroy, id: @lead.id
|
|
616
|
+
expect(flash[:notice]).not_to eq(nil)
|
|
617
|
+
expect(session[:leads_current_page]).to eq(1)
|
|
618
|
+
expect(response).to redirect_to(leads_path)
|
|
619
|
+
end
|
|
620
|
+
|
|
621
|
+
it "should redirect to lead index with the flash message is the lead got deleted" do
|
|
622
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
623
|
+
@lead.destroy
|
|
624
|
+
|
|
625
|
+
delete :destroy, id: @lead.id
|
|
626
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
627
|
+
expect(response).to redirect_to(leads_path)
|
|
628
|
+
end
|
|
629
|
+
|
|
630
|
+
it "should redirect to lead index with the flash message if the lead is protected" do
|
|
631
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
632
|
+
|
|
633
|
+
delete :destroy, id: @private.id
|
|
634
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
635
|
+
expect(response).to redirect_to(leads_path)
|
|
636
|
+
end
|
|
637
|
+
end
|
|
638
|
+
end
|
|
639
|
+
|
|
640
|
+
# GET /leads/1/convert
|
|
641
|
+
# GET /leads/1/convert.xml AJAX
|
|
642
|
+
#----------------------------------------------------------------------------
|
|
643
|
+
describe "responding to GET convert" do
|
|
644
|
+
it "should should collect necessary data and render [convert] template" do
|
|
645
|
+
@campaign = FactoryGirl.create(:campaign, user: current_user)
|
|
646
|
+
@lead = FactoryGirl.create(:lead, user: current_user, campaign: @campaign, source: "cold_call")
|
|
647
|
+
@accounts = [FactoryGirl.create(:account, user: current_user)]
|
|
648
|
+
@account = Account.new(user: current_user, name: @lead.company, access: "Lead")
|
|
649
|
+
@opportunity = Opportunity.new(user: current_user, access: "Lead", stage: "prospecting", campaign: @lead.campaign, source: @lead.source)
|
|
650
|
+
|
|
651
|
+
xhr :get, :convert, id: @lead.id
|
|
652
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
653
|
+
expect(assigns[:accounts]).to eq(@accounts)
|
|
654
|
+
expect(assigns[:account].attributes).to eq(@account.attributes)
|
|
655
|
+
expect(assigns[:opportunity].attributes).to eq(@opportunity.attributes)
|
|
656
|
+
expect(assigns[:opportunity].campaign).to eq(@opportunity.campaign)
|
|
657
|
+
expect(response).to render_template("leads/convert")
|
|
658
|
+
end
|
|
659
|
+
|
|
660
|
+
describe "(lead got deleted or is otherwise unavailable)" do
|
|
661
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
662
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
663
|
+
@lead.destroy
|
|
664
|
+
|
|
665
|
+
xhr :get, :convert, id: @lead.id
|
|
666
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
667
|
+
expect(response.body).to eq("window.location.reload();")
|
|
668
|
+
end
|
|
669
|
+
|
|
670
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
671
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
672
|
+
|
|
673
|
+
xhr :get, :convert, id: @private.id
|
|
674
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
675
|
+
expect(response.body).to eq("window.location.reload();")
|
|
676
|
+
end
|
|
677
|
+
end
|
|
678
|
+
|
|
679
|
+
describe "(previous lead got deleted or is otherwise unavailable)" do
|
|
680
|
+
before(:each) do
|
|
681
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
682
|
+
@previous = FactoryGirl.create(:lead, user: FactoryGirl.create(:user))
|
|
683
|
+
end
|
|
684
|
+
|
|
685
|
+
it "should notify the view if previous lead got deleted" do
|
|
686
|
+
@previous.destroy
|
|
687
|
+
|
|
688
|
+
xhr :get, :convert, id: @lead.id, previous: @previous.id
|
|
689
|
+
expect(flash[:warning]).to eq(nil) # no warning, just silently remove the div
|
|
690
|
+
expect(assigns[:previous]).to eq(@previous.id)
|
|
691
|
+
expect(response).to render_template("leads/convert")
|
|
692
|
+
end
|
|
693
|
+
|
|
694
|
+
it "should notify the view if previous lead got protected" do
|
|
695
|
+
@previous.update_attribute(:access, "Private")
|
|
696
|
+
|
|
697
|
+
xhr :get, :convert, id: @lead.id, previous: @previous.id
|
|
698
|
+
expect(flash[:warning]).to eq(nil)
|
|
699
|
+
expect(assigns[:previous]).to eq(@previous.id)
|
|
700
|
+
expect(response).to render_template("leads/convert")
|
|
701
|
+
end
|
|
702
|
+
end
|
|
703
|
+
end
|
|
704
|
+
|
|
705
|
+
# PUT /leads/1/promote
|
|
706
|
+
# PUT /leads/1/promote.xml AJAX
|
|
707
|
+
#----------------------------------------------------------------------------
|
|
708
|
+
describe "responding to PUT promote" do
|
|
709
|
+
it "on success: should change lead's status to [converted] and render [promote] template" do
|
|
710
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user, campaign: nil)
|
|
711
|
+
@account = FactoryGirl.create(:account, id: 123, user: current_user)
|
|
712
|
+
@opportunity = FactoryGirl.build(:opportunity, user: current_user, campaign: @lead.campaign,
|
|
713
|
+
account: @account)
|
|
714
|
+
allow(Opportunity).to receive(:new).and_return(@opportunity)
|
|
715
|
+
@contact = FactoryGirl.build(:contact, user: current_user, lead: @lead)
|
|
716
|
+
allow(Contact).to receive(:new).and_return(@contact)
|
|
717
|
+
|
|
718
|
+
xhr :put, :promote, id: 42, account: { id: 123 }, opportunity: { name: "Hello" }
|
|
719
|
+
expect(@lead.reload.status).to eq("converted")
|
|
720
|
+
expect(assigns[:lead]).to eq(@lead)
|
|
721
|
+
expect(assigns[:account]).to eq(@account)
|
|
722
|
+
expect(assigns[:accounts]).to eq([@account])
|
|
723
|
+
expect(assigns[:opportunity]).to eq(@opportunity)
|
|
724
|
+
expect(assigns[:contact]).to eq(@contact)
|
|
725
|
+
expect(assigns[:stage]).to be_instance_of(Array)
|
|
726
|
+
expect(response).to render_template("leads/promote")
|
|
727
|
+
end
|
|
728
|
+
|
|
729
|
+
it "should copy lead permissions to newly created account and opportunity when asked so" do
|
|
730
|
+
he = FactoryGirl.create(:user, id: 7)
|
|
731
|
+
she = FactoryGirl.create(:user, id: 8)
|
|
732
|
+
@lead = FactoryGirl.build(:lead, access: "Shared")
|
|
733
|
+
@lead.permissions << FactoryGirl.build(:permission, user: he, asset: @lead)
|
|
734
|
+
@lead.permissions << FactoryGirl.build(:permission, user: she, asset: @lead)
|
|
735
|
+
@lead.save
|
|
736
|
+
@account = FactoryGirl.build(:account, user: current_user, access: "Shared")
|
|
737
|
+
@account.permissions << FactoryGirl.create(:permission, user: he, asset: @account)
|
|
738
|
+
@account.permissions << FactoryGirl.create(:permission, user: she, asset: @account)
|
|
739
|
+
allow(@account).to receive(:new).and_return(@account)
|
|
740
|
+
@opportunity = FactoryGirl.build(:opportunity, user: current_user, access: "Shared")
|
|
741
|
+
@opportunity.permissions << FactoryGirl.create(:permission, user: he, asset: @opportunity)
|
|
742
|
+
@opportunity.permissions << FactoryGirl.create(:permission, user: she, asset: @opportunity)
|
|
743
|
+
allow(@opportunity).to receive(:new).and_return(@opportunity)
|
|
744
|
+
|
|
745
|
+
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 }
|
|
746
|
+
expect(@account.access).to eq("Shared")
|
|
747
|
+
expect(@account.permissions.map(&:user_id).sort).to eq([7, 8])
|
|
748
|
+
expect(@account.permissions.map(&:asset_id)).to eq([@account.id, @account.id])
|
|
749
|
+
expect(@account.permissions.map(&:asset_type)).to eq(%w(Account Account))
|
|
750
|
+
expect(@opportunity.access).to eq("Shared")
|
|
751
|
+
expect(@opportunity.permissions.map(&:user_id).sort).to eq([7, 8])
|
|
752
|
+
expect(@opportunity.permissions.map(&:asset_id)).to eq([@opportunity.id, @opportunity.id])
|
|
753
|
+
expect(@opportunity.permissions.map(&:asset_type)).to eq(%w(Opportunity Opportunity))
|
|
754
|
+
end
|
|
755
|
+
|
|
756
|
+
it "should assign lead's campaign to the newly created opportunity" do
|
|
757
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
758
|
+
@lead = FactoryGirl.create(:lead, user: current_user, campaign: @campaign)
|
|
759
|
+
|
|
760
|
+
xhr :put, :promote, id: @lead.id, account: { name: "Hello" }, opportunity: { name: "Hello", campaign_id: @campaign.id }
|
|
761
|
+
expect(assigns[:opportunity].campaign).to eq(@campaign)
|
|
762
|
+
end
|
|
763
|
+
|
|
764
|
+
it "should assign lead's source to the newly created opportunity" do
|
|
765
|
+
@lead = FactoryGirl.create(:lead, user: current_user, source: "cold_call")
|
|
766
|
+
|
|
767
|
+
xhr :put, :promote, id: @lead.id, account: { name: "Hello" }, opportunity: { name: "Hello", source: @lead.source }
|
|
768
|
+
expect(assigns[:opportunity].source).to eq(@lead.source)
|
|
769
|
+
end
|
|
770
|
+
|
|
771
|
+
it "should get the data for leads sidebar when called from leads index" do
|
|
772
|
+
@lead = FactoryGirl.create(:lead)
|
|
773
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
774
|
+
|
|
775
|
+
xhr :put, :promote, id: @lead.id, account: { name: "Hello" }, opportunity: {}
|
|
776
|
+
expect(assigns[:lead_status_total]).not_to be_nil
|
|
777
|
+
expect(assigns[:lead_status_total]).to be_an_instance_of(HashWithIndifferentAccess)
|
|
778
|
+
end
|
|
779
|
+
|
|
780
|
+
it "should reload lead campaign if called from campaign landing page" do
|
|
781
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
782
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaign)
|
|
783
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
|
784
|
+
|
|
785
|
+
xhr :put, :promote, id: @lead.id, account: { name: "Hello" }, opportunity: {}
|
|
786
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
787
|
+
end
|
|
788
|
+
|
|
789
|
+
it "on failure: should not change lead's status and still render [promote] template" do
|
|
790
|
+
@lead = FactoryGirl.create(:lead, id: 42, user: current_user, status: "new")
|
|
791
|
+
@account = FactoryGirl.create(:account, id: 123, user: current_user)
|
|
792
|
+
@contact = FactoryGirl.build(:contact, first_name: nil) # make it fail
|
|
793
|
+
allow(Contact).to receive(:new).and_return(@contact)
|
|
794
|
+
|
|
795
|
+
xhr :put, :promote, id: 42, account: { id: 123 }
|
|
796
|
+
expect(@lead.reload.status).to eq("new")
|
|
797
|
+
expect(response).to render_template("leads/promote")
|
|
798
|
+
end
|
|
799
|
+
|
|
800
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
801
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
802
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
803
|
+
@lead.destroy
|
|
804
|
+
|
|
805
|
+
xhr :put, :promote, id: @lead.id
|
|
806
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
807
|
+
expect(response.body).to eq("window.location.reload();")
|
|
808
|
+
end
|
|
809
|
+
|
|
810
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
811
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
812
|
+
|
|
813
|
+
xhr :put, :promote, id: @private.id
|
|
814
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
815
|
+
expect(response.body).to eq("window.location.reload();")
|
|
816
|
+
end
|
|
817
|
+
end
|
|
818
|
+
end
|
|
819
|
+
|
|
820
|
+
# PUT /leads/1/reject
|
|
821
|
+
# PUT /leads/1/reject.xml AJAX and HTML
|
|
822
|
+
#----------------------------------------------------------------------------
|
|
823
|
+
describe "responding to PUT reject" do
|
|
824
|
+
before(:each) do
|
|
825
|
+
@lead = FactoryGirl.create(:lead, user: current_user, status: "new")
|
|
826
|
+
end
|
|
827
|
+
|
|
828
|
+
describe "AJAX request" do
|
|
829
|
+
it "should reject the requested lead and render [reject] template" do
|
|
830
|
+
xhr :put, :reject, id: @lead.id
|
|
831
|
+
|
|
832
|
+
expect(assigns[:lead]).to eq(@lead.reload)
|
|
833
|
+
expect(@lead.status).to eq("rejected")
|
|
834
|
+
expect(response).to render_template("leads/reject")
|
|
835
|
+
end
|
|
836
|
+
|
|
837
|
+
it "should get the data for leads sidebar when called from leads index" do
|
|
838
|
+
request.env["HTTP_REFERER"] = "http://localhost/leads"
|
|
839
|
+
xhr :put, :reject, id: @lead.id
|
|
840
|
+
expect(assigns[:lead_status_total]).not_to be_nil
|
|
841
|
+
expect(assigns[:lead_status_total]).to be_an_instance_of(HashWithIndifferentAccess)
|
|
842
|
+
end
|
|
843
|
+
|
|
844
|
+
it "should reload lead campaign if called from campaign landing page" do
|
|
845
|
+
@campaign = FactoryGirl.create(:campaign)
|
|
846
|
+
@lead = FactoryGirl.create(:lead, campaign: @campaign)
|
|
847
|
+
|
|
848
|
+
request.env["HTTP_REFERER"] = "http://localhost/campaigns/#{@campaign.id}"
|
|
849
|
+
xhr :put, :reject, id: @lead.id
|
|
850
|
+
expect(assigns[:campaign]).to eq(@campaign)
|
|
851
|
+
end
|
|
852
|
+
|
|
853
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
854
|
+
it "should reload current page with the flash message if the lead got deleted" do
|
|
855
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
856
|
+
@lead.destroy
|
|
857
|
+
|
|
858
|
+
xhr :put, :reject, id: @lead.id
|
|
859
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
860
|
+
expect(response.body).to eq("window.location.reload();")
|
|
861
|
+
end
|
|
862
|
+
|
|
863
|
+
it "should reload current page with the flash message if the lead is protected" do
|
|
864
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
865
|
+
|
|
866
|
+
xhr :put, :reject, id: @private.id
|
|
867
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
868
|
+
expect(response.body).to eq("window.location.reload();")
|
|
869
|
+
end
|
|
870
|
+
end
|
|
871
|
+
end
|
|
872
|
+
|
|
873
|
+
describe "HTML request" do
|
|
874
|
+
it "should redirect to Leads index when a lead gets rejected from its landing page" do
|
|
875
|
+
put :reject, id: @lead.id
|
|
876
|
+
|
|
877
|
+
expect(assigns[:lead]).to eq(@lead.reload)
|
|
878
|
+
expect(@lead.status).to eq("rejected")
|
|
879
|
+
expect(flash[:notice]).not_to eq(nil)
|
|
880
|
+
expect(response).to redirect_to(leads_path)
|
|
881
|
+
end
|
|
882
|
+
|
|
883
|
+
describe "lead got deleted or otherwise unavailable" do
|
|
884
|
+
it "should redirect to lead index if the lead got deleted" do
|
|
885
|
+
@lead = FactoryGirl.create(:lead, user: current_user)
|
|
886
|
+
@lead.destroy
|
|
887
|
+
|
|
888
|
+
put :reject, id: @lead.id
|
|
889
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
890
|
+
expect(response).to redirect_to(leads_path)
|
|
891
|
+
end
|
|
892
|
+
|
|
893
|
+
it "should redirect to lead index if the lead is protected" do
|
|
894
|
+
@private = FactoryGirl.create(:lead, user: FactoryGirl.create(:user), access: "Private")
|
|
895
|
+
|
|
896
|
+
put :reject, id: @private.id
|
|
897
|
+
expect(flash[:warning]).not_to eq(nil)
|
|
898
|
+
expect(response).to redirect_to(leads_path)
|
|
899
|
+
end
|
|
900
|
+
end
|
|
901
|
+
end
|
|
902
|
+
end
|
|
903
|
+
|
|
904
|
+
# PUT /leads/1/attach
|
|
905
|
+
# PUT /leads/1/attach.xml AJAX
|
|
906
|
+
#----------------------------------------------------------------------------
|
|
907
|
+
describe "responding to PUT attach" do
|
|
908
|
+
describe "tasks" do
|
|
909
|
+
before do
|
|
910
|
+
@model = FactoryGirl.create(:lead)
|
|
911
|
+
@attachment = FactoryGirl.create(:task, asset: nil)
|
|
912
|
+
end
|
|
913
|
+
it_should_behave_like("attach")
|
|
914
|
+
end
|
|
915
|
+
end
|
|
916
|
+
|
|
917
|
+
# PUT /leads/1/attach
|
|
918
|
+
# PUT /leads/1/attach.xml AJAX
|
|
919
|
+
#----------------------------------------------------------------------------
|
|
920
|
+
describe "responding to PUT attach" do
|
|
921
|
+
describe "tasks" do
|
|
922
|
+
before do
|
|
923
|
+
@model = FactoryGirl.create(:lead)
|
|
924
|
+
@attachment = FactoryGirl.create(:task, asset: nil)
|
|
925
|
+
end
|
|
926
|
+
it_should_behave_like("attach")
|
|
927
|
+
end
|
|
928
|
+
end
|
|
929
|
+
|
|
930
|
+
# POST /leads/1/discard
|
|
931
|
+
# POST /leads/1/discard.xml AJAX
|
|
932
|
+
#----------------------------------------------------------------------------
|
|
933
|
+
describe "responding to POST discard" do
|
|
934
|
+
before(:each) do
|
|
935
|
+
@attachment = FactoryGirl.create(:task, assigned_to: current_user)
|
|
936
|
+
@model = FactoryGirl.create(:lead)
|
|
937
|
+
@model.tasks << @attachment
|
|
938
|
+
end
|
|
939
|
+
|
|
940
|
+
it_should_behave_like("discard")
|
|
941
|
+
end
|
|
942
|
+
|
|
943
|
+
# POST /leads/auto_complete/query AJAX
|
|
944
|
+
#----------------------------------------------------------------------------
|
|
945
|
+
describe "responding to POST auto_complete" do
|
|
946
|
+
before(:each) do
|
|
947
|
+
@auto_complete_matches = [FactoryGirl.create(:lead, first_name: "Hello", last_name: "World", user: current_user)]
|
|
948
|
+
end
|
|
949
|
+
|
|
950
|
+
it_should_behave_like("auto complete")
|
|
951
|
+
end
|
|
952
|
+
|
|
953
|
+
# GET /leads/redraw AJAX
|
|
954
|
+
#----------------------------------------------------------------------------
|
|
955
|
+
describe "responding to GET redraw" do
|
|
956
|
+
it "should save user selected lead preference" do
|
|
957
|
+
xhr :get, :redraw, per_page: 42, view: "long", sort_by: "first_name", naming: "after"
|
|
958
|
+
expect(current_user.preference[:leads_per_page]).to eq("42")
|
|
959
|
+
expect(current_user.preference[:leads_index_view]).to eq("long")
|
|
960
|
+
expect(current_user.preference[:leads_sort_by]).to eq("leads.first_name ASC")
|
|
961
|
+
expect(current_user.preference[:leads_naming]).to eq("after")
|
|
962
|
+
end
|
|
963
|
+
|
|
964
|
+
it "should set similar options for Contacts" do
|
|
965
|
+
xhr :get, :redraw, sort_by: "first_name", naming: "after"
|
|
966
|
+
expect(current_user.pref[:contacts_sort_by]).to eq("contacts.first_name ASC")
|
|
967
|
+
expect(current_user.pref[:contacts_naming]).to eq("after")
|
|
968
|
+
end
|
|
969
|
+
|
|
970
|
+
it "should reset current page to 1" do
|
|
971
|
+
xhr :get, :redraw, per_page: 42, view: "long", sort_by: "first_name", naming: "after"
|
|
972
|
+
expect(session[:leads_current_page]).to eq(1)
|
|
973
|
+
end
|
|
974
|
+
|
|
975
|
+
it "should select @leads and render [index] template" do
|
|
976
|
+
@leads = [
|
|
977
|
+
FactoryGirl.create(:lead, first_name: "Alice", user: current_user),
|
|
978
|
+
FactoryGirl.create(:lead, first_name: "Bobby", user: current_user)
|
|
979
|
+
]
|
|
980
|
+
|
|
981
|
+
xhr :get, :redraw, per_page: 1, sort_by: "first_name"
|
|
982
|
+
expect(assigns(:leads)).to eq([@leads.first])
|
|
983
|
+
expect(response).to render_template("leads/index")
|
|
984
|
+
end
|
|
985
|
+
end
|
|
986
|
+
|
|
987
|
+
# POST /leads/filter AJAX
|
|
988
|
+
#----------------------------------------------------------------------------
|
|
989
|
+
describe "responding to POST filter" do
|
|
990
|
+
it "should filter out leads as @leads and render :index action" do
|
|
991
|
+
session[:leads_filter] = "contacted,rejected"
|
|
992
|
+
|
|
993
|
+
@leads = [FactoryGirl.create(:lead, user: current_user, status: "new")]
|
|
994
|
+
xhr :post, :filter, status: "new"
|
|
995
|
+
expect(assigns[:leads]).to eq(@leads)
|
|
996
|
+
expect(response).to be_a_success
|
|
997
|
+
expect(response).to render_template("leads/index")
|
|
998
|
+
end
|
|
999
|
+
|
|
1000
|
+
it "should reset current page to 1" do
|
|
1001
|
+
@leads = []
|
|
1002
|
+
xhr :post, :filter, status: "new"
|
|
1003
|
+
|
|
1004
|
+
expect(session[:leads_current_page]).to eq(1)
|
|
1005
|
+
end
|
|
1006
|
+
end
|
|
1007
|
+
end
|