fat_free_crm 0.11.0

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of fat_free_crm might be problematic. Click here for more details.

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