MailPoet Newsletters (Previous) - Version 2.6.8

Version Description

  • 2014-07-04 =
  • Fixed security issue reported by Dominic
Download this release

Release Info

Developer wysija
Plugin Icon 128x128 MailPoet Newsletters (Previous)
Version 2.6.8
Comparing to
See all releases

Version 2.6.8

Files changed (319) hide show
  1. add-ons/add-ons-list.php +324 -0
  2. add-ons/add-ons.php +263 -0
  3. add-ons/index.html +0 -0
  4. classes/WJ_Analytics.php +793 -0
  5. classes/WJ_Export.php +216 -0
  6. classes/WJ_Field.php +283 -0
  7. classes/WJ_FieldHandler.php +59 -0
  8. classes/WJ_FieldRender.php +249 -0
  9. classes/WJ_FieldUser.php +130 -0
  10. classes/WJ_Import.php +828 -0
  11. classes/WJ_Migrator.php +53 -0
  12. classes/WJ_Option.php +29 -0
  13. classes/WJ_Settings.php +30 -0
  14. classes/WJ_Stats.php +474 -0
  15. classes/WJ_Upgrade.php +390 -0
  16. classes/WJ_Utils.php +69 -0
  17. classes/index.html +0 -0
  18. controllers/ajax/campaigns.php +1004 -0
  19. controllers/ajax/config.php +460 -0
  20. controllers/ajax/index.html +0 -0
  21. controllers/ajax/statistics.php +95 -0
  22. controllers/ajax/subscribers.php +131 -0
  23. controllers/ajax/tmce.php +139 -0
  24. controllers/back.php +790 -0
  25. controllers/back/campaigns.php +2759 -0
  26. controllers/back/config.php +383 -0
  27. controllers/back/index.html +0 -0
  28. controllers/back/premium.php +34 -0
  29. controllers/back/statistics.php +197 -0
  30. controllers/back/subscribers.php +1051 -0
  31. controllers/front.php +54 -0
  32. controllers/front/confirm.php +308 -0
  33. controllers/front/email.php +95 -0
  34. controllers/front/index.html +0 -0
  35. controllers/front/module.php +38 -0
  36. controllers/front/stats.php +56 -0
  37. controllers/front/subscribers.php +135 -0
  38. controllers/index.html +0 -0
  39. core/autoloader.php +23 -0
  40. core/base.php +1578 -0
  41. core/constants.php +46 -0
  42. core/controller.php +99 -0
  43. core/index.html +0 -0
  44. core/model.php +1034 -0
  45. core/module/module.php +248 -0
  46. core/module/statistics.php +33 -0
  47. core/module/statistics_model.php +139 -0
  48. core/module/statisticschart.php +11 -0
  49. core/module/statisticstable.php +5 -0
  50. core/view.php +292 -0
  51. css/add-ons.css +1 -0
  52. css/admin-campaigns-articles.css +1 -0
  53. css/admin-campaigns-autopost.css +1 -0
  54. css/admin-campaigns-bookmarks.css +1 -0
  55. css/admin-campaigns-dividers.css +1 -0
  56. css/admin-campaigns-editDetails.css +1 -0
  57. css/admin-campaigns-editTemplate.css +1 -0
  58. css/admin-campaigns-medias.css +1 -0
  59. css/admin-campaigns-themes.css +1 -0
  60. css/admin-campaigns-viewstats.css +1 -0
  61. css/admin-campaigns-welcome_new.css +1 -0
  62. css/admin-campaigns.css +1 -0
  63. css/admin-config-form_widget_settings.css +1 -0
  64. css/admin-config.css +1 -0
  65. css/admin-dashicons.css +1 -0
  66. css/admin-global.css +1 -0
  67. css/admin-premium.css +1 -0
  68. css/admin-statistics.css +1 -0
  69. css/admin-subscribers-addlist.css +1 -0
  70. css/admin-subscribers-edit.css +1 -0
  71. css/admin-subscribers-export.css +1 -0
  72. css/admin-subscribers-exportlist.css +1 -0
  73. css/admin-subscribers-import.css +1 -0
  74. css/admin-subscribers-importmatch.css +1 -0
  75. css/admin-subscribers-lists.css +0 -0
  76. css/admin-subscribers.css +1 -0
  77. css/admin-widget.css +1 -0
  78. css/admin.css +1 -0
  79. css/adminPopup.css +1 -0
  80. css/index.html +0 -0
  81. css/jquery/ui/themes/base/images/animated-overlay.gif +0 -0
  82. css/jquery/ui/themes/base/images/ui-bg_flat_0_aaaaaa_40x100.png +0 -0
  83. css/jquery/ui/themes/base/images/ui-bg_flat_75_ffffff_40x100.png +0 -0
  84. css/jquery/ui/themes/base/images/ui-bg_glass_55_fbf9ee_1x400.png +0 -0
  85. css/jquery/ui/themes/base/images/ui-bg_glass_65_ffffff_1x400.png +0 -0
  86. css/jquery/ui/themes/base/images/ui-bg_glass_75_dadada_1x400.png +0 -0
  87. css/jquery/ui/themes/base/images/ui-bg_glass_75_e6e6e6_1x400.png +0 -0
  88. css/jquery/ui/themes/base/images/ui-bg_glass_95_fef1ec_1x400.png +0 -0
  89. css/jquery/ui/themes/base/images/ui-bg_highlight-soft_75_cccccc_1x100.png +0 -0
  90. css/jquery/ui/themes/base/images/ui-icons_222222_256x240.png +0 -0
  91. css/jquery/ui/themes/base/images/ui-icons_2e83ff_256x240.png +0 -0
  92. css/jquery/ui/themes/base/images/ui-icons_454545_256x240.png +0 -0
  93. css/jquery/ui/themes/base/images/ui-icons_888888_256x240.png +0 -0
  94. css/jquery/ui/themes/base/images/ui-icons_cd0a0a_256x240.png +0 -0
  95. css/jquery/ui/themes/base/jquery.ui.core.min.css +5 -0
  96. css/jquery/ui/themes/base/jquery.ui.datepicker.min.css +5 -0
  97. css/rtl.css +1 -0
  98. css/select2/select2.css +615 -0
  99. css/smoothness/images/calendar.gif +0 -0
  100. css/smoothness/images/index.html +0 -0
  101. css/smoothness/images/ui-bg_flat_0_aaaaaa_40x100.png +0 -0
  102. css/smoothness/images/ui-bg_flat_75_ffffff_40x100.png +0 -0
  103. css/smoothness/images/ui-bg_glass_55_fbf9ee_1x400.png +0 -0
  104. css/smoothness/images/ui-bg_glass_65_ffffff_1x400.png +0 -0
  105. css/smoothness/images/ui-bg_glass_75_dadada_1x400.png +0 -0
  106. css/smoothness/images/ui-bg_glass_75_e6e6e6_1x400.png +0 -0
  107. css/smoothness/images/ui-bg_glass_95_fef1ec_1x400.png +0 -0
  108. css/smoothness/images/ui-bg_highlight-soft_75_cccccc_1x100.png +0 -0
  109. css/smoothness/images/ui-icons_222222_256x240.png +0 -0
  110. css/smoothness/images/ui-icons_2e83ff_256x240.png +0 -0
  111. css/smoothness/images/ui-icons_454545_256x240.png +0 -0
  112. css/smoothness/images/ui-icons_888888_256x240.png +0 -0
  113. css/smoothness/images/ui-icons_cd0a0a_256x240.png +0 -0
  114. css/smoothness/index.html +0 -0
  115. css/smoothness/jquery-ui-1.8.20.custom.css +565 -0
  116. css/tmce/editor.css +2317 -0
  117. css/tmce/index.html +0 -0
  118. css/tmce/panelbtns.css +23 -0
  119. css/tmce/style.css +7 -0
  120. css/tmce/subscribers_count.css +37 -0
  121. css/tmce/widget.css +44 -0
  122. css/validationEngine.jquery.css +1 -0
  123. css/vendor/bootstrap.tooltip.css +101 -0
  124. css/wysija-editor.css +1 -0
  125. css/wysija-form-editor.css +1 -0
  126. data/bookmarks/index.html +0 -0
  127. data/bookmarks/medium/01/credit.txt +3 -0
  128. data/bookmarks/medium/01/facebook.png +0 -0
  129. data/bookmarks/medium/01/google.png +0 -0
  130. data/bookmarks/medium/01/linkedin.png +0 -0
  131. data/bookmarks/medium/01/twitter.png +0 -0
  132. data/bookmarks/medium/02/credit.txt +3 -0
  133. data/bookmarks/medium/02/facebook.png +0 -0
  134. data/bookmarks/medium/02/google.png +0 -0
  135. data/bookmarks/medium/02/linkedin.png +0 -0
  136. data/bookmarks/medium/02/twitter.png +0 -0
  137. data/bookmarks/medium/03/credit.txt +5 -0
  138. data/bookmarks/medium/03/facebook.png +0 -0
  139. data/bookmarks/medium/03/google.png +0 -0
  140. data/bookmarks/medium/03/linkedin.png +0 -0
  141. data/bookmarks/medium/03/twitter.png +0 -0
  142. data/bookmarks/medium/04/credit.txt +5 -0
  143. data/bookmarks/medium/04/facebook.png +0 -0
  144. data/bookmarks/medium/04/google.png +0 -0
  145. data/bookmarks/medium/04/linkedin.png +0 -0
  146. data/bookmarks/medium/04/twitter.png +0 -0
  147. data/bookmarks/medium/06/credit.txt +5 -0
  148. data/bookmarks/medium/06/facebook.png +0 -0
  149. data/bookmarks/medium/06/google.png +0 -0
  150. data/bookmarks/medium/06/linkedin.png +0 -0
  151. data/bookmarks/medium/06/twitter.png +0 -0
  152. data/bookmarks/medium/07/credit.txt +5 -0
  153. data/bookmarks/medium/07/facebook.png +0 -0
  154. data/bookmarks/medium/07/google.png +0 -0
  155. data/bookmarks/medium/07/linkedin.png +0 -0
  156. data/bookmarks/medium/07/twitter.png +0 -0
  157. data/bookmarks/medium/08/credit.txt +5 -0
  158. data/bookmarks/medium/08/facebook.png +0 -0
  159. data/bookmarks/medium/08/google.png +0 -0
  160. data/bookmarks/medium/08/linkedin.png +0 -0
  161. data/bookmarks/medium/08/twitter.png +0 -0
  162. data/bookmarks/medium/09/credit.txt +5 -0
  163. data/bookmarks/medium/09/facebook.png +0 -0
  164. data/bookmarks/medium/09/google.png +0 -0
  165. data/bookmarks/medium/09/linkedin.png +0 -0
  166. data/bookmarks/medium/09/twitter.png +0 -0
  167. data/bookmarks/medium/10/credit.txt +3 -0
  168. data/bookmarks/medium/10/facebook.png +0 -0
  169. data/bookmarks/medium/10/google.png +0 -0
  170. data/bookmarks/medium/10/linkedin.png +0 -0
  171. data/bookmarks/medium/10/twitter.png +0 -0
  172. data/bookmarks/medium/11/credit.txt +5 -0
  173. data/bookmarks/medium/11/facebook.png +0 -0
  174. data/bookmarks/medium/11/google.png +0 -0
  175. data/bookmarks/medium/11/linkedin.png +0 -0
  176. data/bookmarks/medium/11/twitter.png +0 -0
  177. data/bookmarks/medium/12/credit.txt +5 -0
  178. data/bookmarks/medium/12/facebook.png +0 -0
  179. data/bookmarks/medium/12/google.png +0 -0
  180. data/bookmarks/medium/12/linkedin.png +0 -0
  181. data/bookmarks/medium/12/twitter.png +0 -0
  182. data/bookmarks/medium/13/credit.txt +5 -0
  183. data/bookmarks/medium/13/facebook.png +0 -0
  184. data/bookmarks/medium/13/google.png +0 -0
  185. data/bookmarks/medium/13/linkedin.png +0 -0
  186. data/bookmarks/medium/13/twitter.png +0 -0
  187. data/bookmarks/medium/14/credit.txt +3 -0
  188. data/bookmarks/medium/14/facebook.png +0 -0
  189. data/bookmarks/medium/14/google.png +0 -0
  190. data/bookmarks/medium/14/linkedin.png +0 -0
  191. data/bookmarks/medium/14/twitter.png +0 -0
  192. data/bookmarks/medium/15/credit.txt +3 -0
  193. data/bookmarks/medium/15/facebook.png +0 -0
  194. data/bookmarks/medium/15/google.png +0 -0
  195. data/bookmarks/medium/15/linkedin.png +0 -0
  196. data/bookmarks/medium/15/twitter.png +0 -0
  197. data/bookmarks/medium/16/credit.txt +3 -0
  198. data/bookmarks/medium/16/facebook.png +0 -0
  199. data/bookmarks/medium/16/google.png +0 -0
  200. data/bookmarks/medium/16/linkedin.png +0 -0
  201. data/bookmarks/medium/16/twitter.png +0 -0
  202. data/bookmarks/medium/17/facebook.png +0 -0
  203. data/bookmarks/medium/17/google.png +0 -0
  204. data/bookmarks/medium/17/linkedin.png +0 -0
  205. data/bookmarks/medium/17/twitter.png +0 -0
  206. data/bookmarks/medium/18/credit.txt +3 -0
  207. data/bookmarks/medium/18/facebook.png +0 -0
  208. data/bookmarks/medium/18/google.png +0 -0
  209. data/bookmarks/medium/18/linkedin.png +0 -0
  210. data/bookmarks/medium/18/twitter.png +0 -0
  211. data/bookmarks/medium/19/credit.txt +3 -0
  212. data/bookmarks/medium/19/facebook.png +0 -0
  213. data/bookmarks/medium/19/google.png +0 -0
  214. data/bookmarks/medium/19/linkedin.png +0 -0
  215. data/bookmarks/medium/19/twitter.png +0 -0
  216. data/bookmarks/medium/20/credit.txt +3 -0
  217. data/bookmarks/medium/20/facebook.png +0 -0
  218. data/bookmarks/medium/20/google.png +0 -0
  219. data/bookmarks/medium/20/linkedin.png +0 -0
  220. data/bookmarks/medium/20/twitter.png +0 -0
  221. data/bookmarks/medium/21/credit.txt +3 -0
  222. data/bookmarks/medium/21/facebook.png +0 -0
  223. data/bookmarks/medium/21/google.png +0 -0
  224. data/bookmarks/medium/21/linkedin.png +0 -0
  225. data/bookmarks/medium/21/twitter.png +0 -0
  226. data/bookmarks/medium/22/credit.txt +3 -0
  227. data/bookmarks/medium/22/facebook.png +0 -0
  228. data/bookmarks/medium/22/google.png +0 -0
  229. data/bookmarks/medium/22/linkedin.png +0 -0
  230. data/bookmarks/medium/22/twitter.png +0 -0
  231. data/bookmarks/medium/23/facebook.png +0 -0
  232. data/bookmarks/medium/23/google.png +0 -0
  233. data/bookmarks/medium/23/linkedin.png +0 -0
  234. data/bookmarks/medium/23/twitter.png +0 -0
  235. data/bookmarks/medium/24/facebook.png +0 -0
  236. data/bookmarks/medium/24/google.png +0 -0
  237. data/bookmarks/medium/24/linkedin.png +0 -0
  238. data/bookmarks/medium/24/twitter.png +0 -0
  239. data/bookmarks/medium/index.html +0 -0
  240. data/bookmarks/small/01/credit.txt +3 -0
  241. data/bookmarks/small/01/facebook.gif +0 -0
  242. data/bookmarks/small/01/google.gif +0 -0
  243. data/bookmarks/small/01/linkedin.gif +0 -0
  244. data/bookmarks/small/01/twitter.gif +0 -0
  245. data/bookmarks/small/02/credit.txt +3 -0
  246. data/bookmarks/small/02/facebook.png +0 -0
  247. data/bookmarks/small/02/google.png +0 -0
  248. data/bookmarks/small/02/linkedin.png +0 -0
  249. data/bookmarks/small/02/twitter.png +0 -0
  250. data/bookmarks/small/03/credit.txt +5 -0
  251. data/bookmarks/small/03/facebook.png +0 -0
  252. data/bookmarks/small/03/google.png +0 -0
  253. data/bookmarks/small/03/linkedin.png +0 -0
  254. data/bookmarks/small/03/twitter.png +0 -0
  255. data/bookmarks/small/04/credit.txt +5 -0
  256. data/bookmarks/small/04/facebook.png +0 -0
  257. data/bookmarks/small/04/google.png +0 -0
  258. data/bookmarks/small/04/linkedin.png +0 -0
  259. data/bookmarks/small/04/twitter.png +0 -0
  260. data/bookmarks/small/05/credit.txt +3 -0
  261. data/bookmarks/small/05/facebook.png +0 -0
  262. data/bookmarks/small/05/google.png +0 -0
  263. data/bookmarks/small/05/linkedin.png +0 -0
  264. data/bookmarks/small/05/twitter.png +0 -0
  265. data/bookmarks/small/06/credit.txt +3 -0
  266. data/bookmarks/small/06/facebook.png +0 -0
  267. data/bookmarks/small/06/google.png +0 -0
  268. data/bookmarks/small/06/linkedin.png +0 -0
  269. data/bookmarks/small/06/twitter.png +0 -0
  270. data/bookmarks/small/index.html +0 -0
  271. data/dividers/artcore-white.png +0 -0
  272. data/dividers/artcore.png +0 -0
  273. data/dividers/bar.png +0 -0
  274. data/dividers/business.png +0 -0
  275. data/dividers/business2.png +0 -0
  276. data/dividers/cafe.png +0 -0
  277. data/dividers/drop1.png +0 -0
  278. data/dividers/drop2.png +0 -0
  279. data/dividers/drop3.png +0 -0
  280. data/dividers/flashback.png +0 -0
  281. data/dividers/index.html +0 -0
  282. data/dividers/juicy.png +0 -0
  283. data/dividers/plus.png +0 -0
  284. data/dividers/school.png +0 -0
  285. data/dividers/sketchy.png +0 -0
  286. data/dividers/solid.jpg +0 -0
  287. data/dividers/vintage.png +0 -0
  288. data/index.html +0 -0
  289. data/themes/default/index.html +0 -0
  290. data/themes/default/screenshot.jpg +0 -0
  291. data/themes/default/style.css +48 -0
  292. data/themes/default/thumbnail.jpg +0 -0
  293. data/themes/index.html +0 -0
  294. docs/german-disclaimer.md +4 -0
  295. font/icon-email.eot +0 -0
  296. font/icon-email.svg +11 -0
  297. font/icon-email.ttf +0 -0
  298. font/icon-email.woff +0 -0
  299. helpers/articles.php +432 -0
  300. helpers/autonews.php +390 -0
  301. helpers/back.php +529 -0
  302. helpers/backloader.php +406 -0
  303. helpers/bookmarks.php +113 -0
  304. helpers/bounce.php +1099 -0
  305. helpers/campaigns.php +52 -0
  306. helpers/charts.php +66 -0
  307. helpers/conflicts.php +103 -0
  308. helpers/cron.php +184 -0
  309. helpers/dividers.php +64 -0
  310. helpers/email.php +166 -0
  311. helpers/encoding.php +40 -0
  312. helpers/file.php +271 -0
  313. helpers/form_engine.php +842 -0
  314. helpers/forms.php +266 -0
  315. helpers/front.php +306 -0
  316. helpers/http.php +119 -0
  317. helpers/image.php +89 -0
  318. helpers/index.html +0 -0
  319. helpers/install.php +466 -0
add-ons/add-ons-list.php ADDED
@@ -0,0 +1,324 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * List of all add-ons available for MailPoet
4
+ */
5
+ function add_ons_list(){
6
+ $available_add_ons = array(
7
+ array(
8
+ 'name' => 'WooCommerce: Subscribe on Checkout',
9
+ 'search' => 'MailPoet WooCommerce Add-on',
10
+ 'thumbnail' => 'woocommerce.png',
11
+ 'description' => __( 'Add a checkbox on checkout page for your customers to subscribe to your MailPoet newsletters.', WYSIJA ),
12
+ 'plugin_url' => 'mailpoet-woocommerce-addon/mailpoet-woocommerce-addon.php',
13
+ 'version' => '1.0.2',
14
+ 'config_url' => 'admin.php?page=woocommerce_settings&tab=mailpoet',
15
+ 'official' => true,
16
+ 'on_wordpress.org' => true,
17
+ 'free' => true,
18
+ 'service' => false,
19
+ 'requires' => 'woocommerce/woocommerce.php',
20
+ 'requires_name' => 'WooCommerce',
21
+ ),
22
+
23
+ array(
24
+ 'name' => 'Autoresponders for WooCommerce Products',
25
+ 'search' => 'Autoresponders for WooCommerce Products',
26
+ 'thumbnail' => 'woocommerce.png',
27
+ 'description' => __( 'Decide which emails a new customer gets automatically, based on the product he just purchased. This a Premium add-on for WooCommerce.', WYSIJA ),
28
+ 'review' => '',
29
+ 'plugin_url' => '',
30
+ 'author_url' => 'http://clicky.me/woocommerce-autoresponder',
31
+ 'purchase_url' => 'http://clicky.me/woocommerce-autoresponder',
32
+ 'version' => '1.0.1',
33
+ 'official' => true,
34
+ 'on_wordpress.org' => false,
35
+ 'free' => false,
36
+ 'service' => false,
37
+ ),
38
+
39
+ array(
40
+ 'name' => 'Jigoshop: Subscribe on Checkout',
41
+ 'search' => 'MailPoet Jigoshop Add-on',
42
+ 'thumbnail' => 'jigoshop.png',
43
+ 'description' => __( 'Add a checkbox on checkout page for your customers to subscribe to your MailPoet newsletters.', WYSIJA ),
44
+ 'plugin_url' => 'mailpoet-jigoshop-addon/mailpoet-jigoshop-addon.php',
45
+ 'version' => '1.0.3',
46
+ 'config_url' => 'admin.php?page=jigoshop_settings&tab=mailpoet',
47
+ 'official' => true,
48
+ 'on_wordpress.org' => true,
49
+ 'free' => true,
50
+ 'service' => false,
51
+ 'requires' => 'jigoshop/jigoshop.php',
52
+ 'requires_name' => 'Jigoshop',
53
+ ),
54
+
55
+ array(
56
+ 'name' => 'Gravity Forms: Subscribe Add-on',
57
+ 'search' => 'MailPoet Gravity Forms Add-on',
58
+ 'thumbnail' => 'gravity-forms.png',
59
+ 'description' => __( 'Add a new field to add to your forms so your visitors can subscriber to your MailPoet newsletters.', WYSIJA ),
60
+ 'plugin_url' => 'mailpoet-gravity-forms-addon/mailpoet-gravity-forms-addon.php',
61
+ 'version' => '1.0.0',
62
+ 'official' => true,
63
+ 'on_wordpress.org' => true,
64
+ 'free' => true,
65
+ 'service' => false,
66
+ 'requires' => 'gravityforms/gravityforms.php',
67
+ 'requires_name' => 'Gravity Forms',
68
+ ),
69
+
70
+ array(
71
+ 'name' => 'WP e-Commerce: Subscribe on Checkout',
72
+ 'search' => 'MailPoet WP e-Commerce Add-on',
73
+ 'thumbnail' => 'wp-e-commerce.png',
74
+ 'description' => __( 'Add a checkbox on checkout page for your customers to subscribe to your MailPoet newsletters.', WYSIJA ),
75
+ 'plugin_url' => 'mailpoet-wp-ecommerce-addon/mailpoet-wp-ecommerce-addon.php',
76
+ 'version' => '1.0.0',
77
+ 'config_url' => 'edit.php?post_type=wpsc-product&page=mailpoet',
78
+ 'official' => true,
79
+ 'on_wordpress.org' => true,
80
+ 'free' => true,
81
+ 'service' => false,
82
+ 'requires' => 'wp-e-commerce/wp-shopping-cart.php',
83
+ 'requires_name' => 'WP e-Commerce',
84
+ ),
85
+
86
+ array(
87
+ 'name' => 'Restrict Content Pro: Registration Add-on',
88
+ 'search' => 'Restrict Content Pro Wysija',
89
+ 'thumbnail' => 'pippins.png',
90
+ 'description' => __( 'Add a checkbox on the registration form of this plugin so your new users join a list of your choice.', WYSIJA ),
91
+ 'plugin_url' => 'restrict-content-pro-wysija/rcp-wysija.php',
92
+ 'version' => '1.0',
93
+ 'official' => true,
94
+ 'on_wordpress.org' => true,
95
+ 'free' => true,
96
+ 'service' => false,
97
+ 'requires' => 'restrict-content-pro/restrict-content-pro.php',
98
+ 'requires_name' => 'Restrict Content Pro',
99
+ ),
100
+
101
+ array(
102
+ 'name' => 'Contact Form 7: MailPoet Add-on',
103
+ 'search' => 'MailPoet Contact Form 7',
104
+ 'thumbnail' => 'cf7.png',
105
+ 'description' => __( 'Add a checkbox in your forms, and choose which lists the user gets added to.', WYSIJA ),
106
+ 'plugin_url' => 'mailpoet-contact-form-7/mailpoet-contact-form-7.php',
107
+ 'author_url' => '',
108
+ 'version' => '1.0.0',
109
+ 'official' => true,
110
+ 'on_wordpress.org' => true,
111
+ 'free' => true,
112
+ 'service' => false,
113
+ 'requires' => 'contact-form-7/wp-contact-form-7.php',
114
+ 'requires_name' => 'Contact Form 7',
115
+ ),
116
+
117
+ array(
118
+ 'name' => 'WangGuard: MailPoet Connector',
119
+ 'search' => 'WangGuard Wysija Newsletter Connector',
120
+ 'thumbnail' => 'wangguard.png',
121
+ 'description' => __( 'Stop fake signups by robots with WangGuard, a free white listing service, and plugin.', WYSIJA ),
122
+ 'review' => __( 'If you have this problem, then this is your solution. Totally free too.', WYSIJA ),
123
+ 'plugin_url' => 'wangguard-wysija-newsletter-connector/wangguard-wysija.php',
124
+ 'author_url' => 'http://clicky.me/wangguard',
125
+ 'version' => '1.0.0',
126
+ 'official' => true,
127
+ 'on_wordpress.org' => true,
128
+ 'free' => true,
129
+ 'service' => false,
130
+ 'requires' => '',
131
+ ),
132
+
133
+ array(
134
+ 'name' => 'Leaky Paywall: MailPoet Add-on',
135
+ 'search' => 'Leaky Paywall: MailPoet Add-on',
136
+ 'thumbnail' => 'leakypw.png',
137
+ 'description' => __( 'Leaky Paywall is a Premium plugin to put content behind a paywall. This add-on manages your customers\' customers subscriptions.', WYSIJA ),
138
+ 'plugin_url' => 'leaky-paywall-mail-poet/issuem-leaky-paywall-mailpoet.php',
139
+ 'author_url' => 'http://clicky.me/leakypw',
140
+ 'version' => '1.0.0',
141
+ 'official' => true,
142
+ 'on_wordpress.org' => true,
143
+ 'free' => true,
144
+ 'service' => false,
145
+ ),
146
+
147
+ array(
148
+ 'name' => 'GetConversion: Bar and Box Forms Add-on',
149
+ 'search' => 'GC MailPoet EX',
150
+ 'thumbnail' => 'getconversion.png',
151
+ 'description' => __( 'Put a form in a top or bottom bar of your site, or in a nice box within your posts with GetConversion\'s free plugins. Connect them to MailPoet with this add-on.<br/></br>Get 30% off the Premium version of their add-on by clicking <a href="http://getconversion.com/products/gc-mailpoet-ex-pro/?coupon=MAILPOET">here</a>.', WYSIJA ),
152
+ 'review' => __( 'Try it yourself, we were impressed by the free versions.', WYSIJA ),
153
+ 'plugin_url' => 'gc-message-bar/main.php',
154
+ 'author_url' => 'http://clicky.me/GC-add-on',
155
+ 'version' => '1.0.0',
156
+ 'official' => true,
157
+ 'on_wordpress.org' => true,
158
+ 'free' => true,
159
+ 'service' => false,
160
+ ),
161
+
162
+ /* Below is the list of services that work with MailPoet */
163
+
164
+ array(
165
+ 'name' => 'OptinMonster: Slick Forms',
166
+ 'search' => 'OptinMonster: slick forms',
167
+ 'thumbnail' => 'optinmonster.png',
168
+ 'description' => __( 'Put a subscription form in a popup, footer bar or slide-in to add subscribers with this Premium plugin.', WYSIJA ),
169
+ 'review' => __( 'The authors have done a pretty awesome job.', WYSIJA ),
170
+ 'plugin_url' => '',
171
+ 'author_url' => 'http://clicky.me/optinmonsteraddon',
172
+ 'purchase_url' => '',
173
+ 'version' => '',
174
+ 'official' => false,
175
+ 'on_wordpress.org' => false,
176
+ 'free' => false,
177
+ 'service' => true,
178
+ 'premium_offer' => __( 'Premium users save 20% with this coupon: MP20', WYSIJA ),
179
+ ),
180
+
181
+ array(
182
+ 'name' => 'Pippity: Form in a Popup',
183
+ 'search' => 'Pippity: form in a popup',
184
+ 'thumbnail' => 'pippity.png',
185
+ 'description' => __( 'This Premium plugin allows you to add a MailPoet form in a popup.', WYSIJA ),
186
+ 'review' => __( 'You\'ll need to know CSS to design it to your taste.', WYSIJA ),
187
+ 'plugin_url' => '',
188
+ 'author_url' => 'http://clicky.me/pippity',
189
+ 'purchase_url' => '',
190
+ 'version' => '',
191
+ 'official' => false,
192
+ 'on_wordpress.org' => false,
193
+ 'free' => false,
194
+ 'service' => true,
195
+ 'premium_offer' => __( 'Premium users save 25% with this coupon: mailpoet25', WYSIJA ),
196
+ ),
197
+
198
+
199
+ array(
200
+ 'name' => 'Hybrid Connect: Forms and Facebook',
201
+ 'search' => 'Hybrid Connect: forms and Facebook',
202
+ 'thumbnail' => 'hybrid-connect.png',
203
+ 'description' => __( 'This Premium plugin allows you to add MailPoet\'s forms in many places, and in Facebook.', WYSIJA ),
204
+ 'review' => __( 'Very powerful, if not a quirky user interface.', WYSIJA ),
205
+ 'plugin_url' => '',
206
+ 'author_url' => 'http://clicky.me/hybrid-connect',
207
+ 'purchase_url' => '',
208
+ 'version' => '',
209
+ 'official' => false,
210
+ 'on_wordpress.org' => false,
211
+ 'free' => false,
212
+ 'service' => true,
213
+ 'premium_offer' => '',
214
+ ),
215
+
216
+ array(
217
+ 'name' => 'Magic Action Box: Pretty forms in post',
218
+ 'search' => 'Magic Action Box: pretty forms in post',
219
+ 'thumbnail' => 'magic-action-box.png',
220
+ 'description' => __( 'This free plugin allows you to design and position a form at the beginning or bottom of a post.', WYSIJA ),
221
+ 'review' => __( 'It\'s free, and has proven popular with our users.', WYSIJA ),
222
+ 'plugin_url' => '',
223
+ 'author_url' => 'http://clicky.me/magic-action-box',
224
+ 'purchase_url' => '',
225
+ 'version' => '',
226
+ 'official' => false,
227
+ 'on_wordpress.org' => false,
228
+ 'free' => false,
229
+ 'service' => true,
230
+ 'premium_offer' => '',
231
+ ),
232
+
233
+ array(
234
+ 'name' => 'Optin Revolution: Popup form',
235
+ 'search' => 'Optin Revolution: popup form',
236
+ 'thumbnail' => 'optin-revolution.png',
237
+ 'description' => __( 'This free plugin allows you to put a form in a popup, that\'s it.', WYSIJA ),
238
+ 'review' => __( 'This is the free version of a more powerful Premium version.', WYSIJA ),
239
+ 'plugin_url' => '',
240
+ 'author_url' => 'http://clicky.me/optin-revolution',
241
+ 'purchase_url' => '',
242
+ 'version' => '',
243
+ 'official' => false,
244
+ 'on_wordpress.org' => false,
245
+ 'free' => false,
246
+ 'service' => true,
247
+ 'premium_offer' => '',
248
+ ),
249
+
250
+ array(
251
+ 'name' => 'Ultimate Coming Soon Page',
252
+ 'search' => 'Ultimate Coming Soon Page',
253
+ 'thumbnail' => 'ultimate-coming-soon.png',
254
+ 'description' => __( 'Your new site is not yet launched? Put a coming soon page with a form to capture emails with this Premium plugin.', WYSIJA ),
255
+ 'review' => __( 'A well designed plugin for a simple idea.', WYSIJA ),
256
+ 'plugin_url' => '',
257
+ 'author_url' => 'http://clicky.me/coming-soon',
258
+ 'purchase_url' => '',
259
+ 'version' => '',
260
+ 'official' => false,
261
+ 'on_wordpress.org' => false,
262
+ 'free' => false,
263
+ 'service' => true,
264
+ 'premium_offer' => __( 'Premium users save 20% with this coupon: MAILPOET20', WYSIJA ),
265
+ ),
266
+
267
+ array(
268
+ 'name' => 'Plugmatter: form below your header',
269
+ 'search' => 'Plugmatter',
270
+ 'thumbnail' => 'plugmatter.png',
271
+ 'description' => __( 'This Premium plugin allows you to put a pretty form right below the header. Nothing more, nothing less.', WYSIJA ),
272
+ 'review' => '',
273
+ 'plugin_url' => '',
274
+ 'author_url' => 'http://clicky.me/plugmatter',
275
+ 'purchase_url' => '',
276
+ 'version' => '',
277
+ 'official' => false,
278
+ 'on_wordpress.org' => false,
279
+ 'free' => false,
280
+ 'service' => true,
281
+ 'premium_offer' => '',
282
+ ),
283
+
284
+ array(
285
+ 'name' => 'EDD: MailPoet: Subscribe on Checkout',
286
+ 'search' => 'Easy Digital Downloads - MailPoet',
287
+ 'thumbnail' => 'easy-digital-downloads.png',
288
+ 'description' => __( 'A checkbox is added as option for your customers to signup for your newsletter while purchasing from your digital store.', WYSIJA ),
289
+ 'review' => '',
290
+ 'plugin_url' => '',
291
+ 'author_url' => 'http://clicky.me/easy-digital-downloads',
292
+ 'purchase_url' => 'http://clicky.me/easy-digital-downloads',
293
+ 'version' => '1.1.2',
294
+ 'official' => false,
295
+ 'on_wordpress.org' => false,
296
+ 'free' => false,
297
+ 'service' => false,
298
+ 'requires' => 'easy-digital-downloads/easy-digital-downloads.php',
299
+ 'requires_name' => 'Easy Digital Downloads',
300
+ 'premium_offer' => '',
301
+ ),
302
+
303
+ array(
304
+ 'name' => 'Ninja Forms: Subscribe Add-on',
305
+ 'search' => 'Ninja Forms: Subscribe Add-on',
306
+ 'thumbnail' => 'ninja-forms.png',
307
+ 'description' => __( 'Quickly create newsletter signup forms for your MailPoet mailing lists using the power and flexibility that Ninja Forms provides.', WYSIJA ),
308
+ 'review' => '',
309
+ 'plugin_url' => 'ninja-forms-mailpoet/nf-mailpoet.php',
310
+ 'author_url' => 'http://clicky.me/ninja-forms',
311
+ 'purchase_url' => 'http://clicky.me/ninja-forms',
312
+ 'version' => '1.0.1',
313
+ 'official' => false,
314
+ 'on_wordpress.org' => false,
315
+ 'free' => true,
316
+ 'service' => false,
317
+ 'requires' => 'ninja-forms/ninja-forms.php',
318
+ 'requires_name' => 'Ninja Forms',
319
+ 'premium_offer' => '',
320
+ ),
321
+ );
322
+
323
+ return $available_add_ons;
324
+ }
add-ons/add-ons.php ADDED
@@ -0,0 +1,263 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ if ( ! defined( 'ABSPATH' ) ){
3
+ exit; // Exit if accessed directly
4
+ }
5
+
6
+ class MailPoet_Add_ons {
7
+ /**
8
+ * Constructor
9
+ */
10
+ public function __construct(){
11
+ $this->plugin_path = WYSIJA_DIR;
12
+ $this->wp_plugin_path = str_replace( 'wysija-newsletters', '', $this->plugin_path );
13
+ $this->plugin_url = WYSIJA_URL;
14
+ $this->image_url = 'http://ps.w.org/wysija-newsletters/assets/add-ons/';
15
+
16
+ $this->mailpoet_add_on_activated_notice();
17
+ $this->mailpoet_add_on_deactivated_notice();
18
+ }
19
+
20
+ /**
21
+ * Runs when the plugin is initialized.
22
+ */
23
+ public function init_mail_poet_add_ons(){
24
+ // Load JavaScript and stylesheets.
25
+ $this->register_scripts_and_styles();
26
+ }
27
+
28
+ /**
29
+ * Registers and enqueues stylesheets for the
30
+ * administration panel and the public facing site.
31
+ */
32
+ public function register_scripts_and_styles(){
33
+ if ( is_admin() ) {
34
+ wp_register_style( 'mail_poet_add_ons', WYSIJA_URL . 'css/add-ons.css' );
35
+ wp_enqueue_style( 'mail_poet_add_ons' );
36
+ } // end if
37
+ } // end register_scripts_and_styles
38
+
39
+ /**
40
+ * This notifies the user that the add-on plugin
41
+ * is now activated and returns them back to the
42
+ * add-ons page.
43
+ */
44
+ public function mailpoet_add_on_activated_notice(){
45
+ global $current_screen;
46
+
47
+ require_once(ABSPATH.'/wp-admin/includes/plugin.php');
48
+
49
+ if ( isset($_GET['action'] ) && $_GET['action'] == 'activate' && isset( $_GET['module'] ) ) {
50
+
51
+ $plugin = plugin_basename( $_GET['module'] );
52
+ $plugin_data = get_plugin_data( $this->wp_plugin_path . $plugin );
53
+
54
+ $plugin_name = esc_attr( str_replace( ' ', '_', $plugin_data['Name'] ) );
55
+ $plugin_name = esc_attr( str_replace( '&#039;', '_', $plugin_name ) );
56
+
57
+ if ( isset( $_GET['requires'] ) ) {
58
+ if ( file_exists( $this->wp_plugin_path . plugin_basename( $_GET['requires'] ) ) ) {
59
+ if ( ! WYSIJA::is_plugin_active( $_GET['requires'] ) ) {
60
+ $location = admin_url( 'admin.php?page=wysija_config&status=not-activated&add-on=' . $plugin_name . '&requires=' . esc_attr( str_replace( ' ', '_', $_GET['requires_name'] ) ) . '#tab-add-ons' );
61
+ wp_safe_redirect( $location );
62
+ exit;
63
+ }
64
+ } else {
65
+ $location = admin_url( 'admin.php?page=wysija_config&status=not-installed&add-on=' . $plugin_name . '&requires=' . esc_attr( str_replace( ' ', '_', $_GET['requires_name'] ) ) . '#tab-add-ons' );
66
+ wp_safe_redirect( $location );
67
+ exit;
68
+ }
69
+ }
70
+
71
+ // Activate the add-on plugin.
72
+ activate_plugin( $plugin );
73
+
74
+ // Return back to add-on page.
75
+ $location = admin_url( 'admin.php?page=wysija_config&status=activated&add-on=' . $plugin_name . '#tab-add-ons' );
76
+ wp_safe_redirect( $location );
77
+ exit;
78
+ }
79
+
80
+ /**
81
+ * Display message if the plugin was not able to activate due
82
+ * to a required plugin is not active first.
83
+ */
84
+ if ( $current_screen->parent_base == 'wysija_campaigns' && isset( $_GET['status'] ) && $_GET['status'] == 'not-activated' || isset( $_GET['status'] ) && $_GET['status'] == 'not-installed' ){
85
+ echo
86
+ '<div id="message" class="error fade" style="display:block !important;">' .
87
+ '<p>' .
88
+ '<strong>' . esc_attr( str_replace( '_', ' ', $_GET['add-on'] ) ) . '</strong> ' .
89
+ wp_kses( sprintf(
90
+ __( 'was not activated as it requires <strong><a href="%s">%s</a></strong> to be installed and active first.', WYSIJA ),
91
+ esc_url( admin_url( 'plugin-install.php?tab=search&type=term&s=' . esc_attr( strtolower( str_replace( ' ', '+', $_GET['requires'] ) ) ) ) ),
92
+ str_replace( '_', ' ', $_GET['requires'] )
93
+ ), array( 'a' => array( 'href' => array() ), 'strong' => array(), 'b' => array(), 'em' => array() ) ) .
94
+ ' <input type="button" class="button" value="' . esc_attr__( 'Hide this message', WYSIJA ) . '" onclick="document.location.href=\'' . esc_url( admin_url( 'admin.php?page=wysija_config#tab-add_ons' ) ) . '\';">' .
95
+ '</p>' .
96
+ '</div>';
97
+ }
98
+
99
+ // Display message once the add-on has been activated.
100
+ if ( $current_screen->parent_base == 'wysija_campaigns' && isset( $_GET['status'] ) && $_GET['status'] == 'activated' ){
101
+ echo '<div id="message" class="updated fade" style="display:block !important;"><p><strong>' . esc_attr( str_replace( '_', ' ', $_GET['add-on'] ) ) . '</strong> ' . esc_attr__( 'has been activated.', WYSIJA ) . '</p></div>';
102
+ }
103
+ }
104
+
105
+ /**
106
+ * This notifies the user that the add-on plugin
107
+ * is now deactivated and returns them back to the
108
+ * add-ons page.
109
+ */
110
+ public function mailpoet_add_on_deactivated_notice(){
111
+ global $current_screen;
112
+
113
+ require_once ABSPATH . '/wp-admin/includes/plugin.php';
114
+
115
+ if ( isset( $_GET['action'] ) && $_GET['action'] == 'deactivate' && isset( $_GET['module'] ) ) {
116
+ $plugin = plugin_basename( $_GET['module'] );
117
+ $plugin_data = get_plugin_data( $this->wp_plugin_path . $plugin );
118
+
119
+ // Deactivate the add-on plugin.
120
+ deactivate_plugins( $plugin );
121
+
122
+ // Return back to add-on page.
123
+ $location = admin_url( 'admin.php?page=wysija_config&status=deactivated&add-on=' . esc_html( str_replace( ' ', '_', $plugin_data['Name'] ) ) . '#tab-add-ons' );
124
+ wp_safe_redirect( $location );
125
+ exit;
126
+ }
127
+
128
+ // Display message once the add-on has been deactivated.
129
+ if ( $current_screen->parent_base == 'wysija_campaigns' && isset( $_GET['status'] ) && $_GET['status'] == 'deactivated' ) {
130
+ echo '<div id="message" class="updated fade" style="display:block !important;"><p><strong>' . esc_attr( str_replace( '_', ' ', $_GET['add-on'] ) ) . '</strong> ' . esc_attr__( 'has been de-activated.', WYSIJA ) . '</p></div>';
131
+ }
132
+
133
+ }
134
+
135
+ /**
136
+ * Displays the add ons page and lists
137
+ * the plugins and services available.
138
+ */
139
+ public function add_ons_page(){
140
+ require_once WYSIJA_DIR . '/add-ons/add-ons-list.php';
141
+
142
+ echo '<div class="module-container">';
143
+ foreach ( add_ons_list() as $plugin => $product ){
144
+ $status = ''; // Status class.
145
+
146
+ /**
147
+ * Queries if the plugin is installed,
148
+ * active and meets the requirements
149
+ * it requires if any.
150
+ */
151
+ if ( file_exists( $this->wp_plugin_path . plugin_basename( $product['plugin_url'] ) ) ) {
152
+ $status .= ' installed';
153
+ } else {
154
+ $status .= ' not-installed';
155
+ }
156
+
157
+ if ( WYSIJA::is_plugin_active( $product['plugin_url'] ) ) {
158
+ $status .= ' active';
159
+ } else {
160
+ $status .= ' inactive';
161
+ }
162
+
163
+ if ( empty( $product['requires'] ) ) {
164
+ $status .= ' ready';
165
+ } elseif ( ! empty( $product['requires'] ) && file_exists( $this->wp_plugin_path . plugin_basename( $product['requires'] ) ) ) {
166
+ $status .= ' ready';
167
+ if ( WYSIJA::is_plugin_active( $product['requires'] ) ) {
168
+ $status .= ' ready';
169
+ } else {
170
+ $status .= ' not-ready';
171
+ }
172
+ } elseif ( ! empty( $product['requires'] ) && ! file_exists( $this->wp_plugin_path . plugin_basename( $product['requires'] ) ) ) {
173
+ $status .= ' not-ready';
174
+ }
175
+
176
+ if ( WYSIJA::is_plugin_active( 'wysija-newsletters-premium/index.php' ) ) {
177
+ $status .= ' premium-active';
178
+ }
179
+
180
+ echo
181
+ '<div class="mailpoet-module' . esc_attr( $status ) . '" id="product">' .
182
+ '<h3>' . esc_attr( $product['name'] ) . '</h3>';
183
+
184
+ if ( ! empty( $product['thumbnail'] ) ) {
185
+ echo '<div class="mailpoet-module-image"><img src="' . esc_url( $this->image_url . $product['thumbnail'] ) . '" width="100%" title="' . esc_attr( $product['name'] ) . '" alt=""></div>';
186
+ }
187
+
188
+ echo
189
+ '<div class="mailpoet-module-content">' .
190
+ '<div class="mailpoet-module-description">' .
191
+ '<p>' . wp_kses( $product['description'], array() ) . '</p>';
192
+
193
+ if ( ! empty( $product['review'] ) ) {
194
+ echo '<p><strong>' . esc_attr__( 'MailPoet says: ', WYSIJA ) . '<em>' . esc_attr( $product['review'] ) . '</em>' . '</strong></p>';
195
+ }
196
+
197
+ if ( WYSIJA::is_plugin_active( 'wysija-newsletters-premium/index.php' ) && ! empty( $product['premium_offer'] ) ) {
198
+ echo '<p><strong>' . esc_attr( $product['premium_offer'] ) . '</strong></p>';
199
+ }
200
+ echo
201
+ '</div>' .
202
+ '</div>' .
203
+
204
+ '<div class="mailpoet-module-actions">';
205
+
206
+ if ( ! empty( $product['author_url'] ) ) {
207
+ echo '<a href="' . esc_url( $product['author_url'] ) . '" target="_blank" rel="external" class="button-primary website">' . esc_attr__( 'Website', WYSIJA ) . '</a>&nbsp;';
208
+ }
209
+
210
+ if ( $product['free'] == false && ! empty( $product['purchase_url'] ) ) {
211
+ if ( ! empty( $product['plugin_url'] ) && ! file_exists( $this->wp_plugin_path . plugin_basename( $product['plugin_url'] ) ) ) {
212
+ echo '<a href="' . esc_url( $product['purchase_url'] ) . '" target="_blank" rel="external" class="button-primary purchase">' . esc_attr__( 'Purchase', WYSIJA ) . '</a>&nbsp;';
213
+ } // end if plugin is installed, don't show purchase button.
214
+ } // end if product is not free.
215
+
216
+ if ( $product['service'] == false ){
217
+ if ( $product['on_wordpress.org'] == true ){
218
+ if ( ! file_exists( $this->wp_plugin_path . plugin_basename( $product['plugin_url'] ) ) ) {
219
+ echo '<a href="' . esc_url( admin_url( 'plugin-install.php?tab=search&type=term&s=' . strtolower( str_replace( ' ', '+', $product['search'] ) ) ) ) . '" class="button-primary install">' . esc_attr__( 'Install from WordPress.org', WYSIJA ) . '</a>&nbsp;';
220
+ }
221
+ } // end if $product['on_wordpress.org'];
222
+
223
+ if ( ! empty( $product['plugin_url'] ) && file_exists( $this->wp_plugin_path . plugin_basename( $product['plugin_url'] ) ) ) {
224
+ if ( ! WYSIJA::is_plugin_active( $product['plugin_url'] ) ) {
225
+ if ( ! empty( $product['requires'] ) ) {
226
+ $requires = '&amp;requires=' . $product['requires'] . '&amp;requires_name=' . $product['requires_name'];
227
+ } else {
228
+ $requires = '';
229
+ }
230
+ echo '<a href="' . esc_url( admin_url( 'admin.php?page=wysija_config&amp;action=activate&amp;module=' . $product['plugin_url'] . $requires ) ) . '" class="button-primary activate">' . esc_attr__( 'Activate', WYSIJA ) . '</a>&nbsp;';
231
+ } else {
232
+ if ( ! empty( $product['config_url'] ) ) {
233
+ echo '<a href="' . esc_url( $product['config_url'] ) . '" class="mailpoet-configure-button button-secondary">' . esc_attr__( 'Configure', WYSIJA ) . '</a>';
234
+ }
235
+ }
236
+ }
237
+ }
238
+
239
+ echo
240
+ '</div>' .
241
+ '</div>';
242
+ } // end if local is yes.
243
+
244
+ echo
245
+ '<div class="submit-idea">' .
246
+ '<p>' . wp_kses( sprintf( __( 'Don\'t see the add-on you\'re looking for? <a href="%s">Submit it</a> in our contact form.', WYSIJA ), 'http://www.mailpoet.com/contact/" target="blank' ), array( 'a' => array( 'href' => array() ) ) ) . '</p>' .
247
+ '</div>' .
248
+ '</div>';
249
+ }
250
+ } // end class
251
+
252
+ /**
253
+ * This loads the add ons class and displays the page.
254
+ *
255
+ * @init_mail_poet_add_ons();
256
+ * @add_ons_page();
257
+ */
258
+ function load_add_ons_manager(){
259
+ $mailpoet_add_ons = new MailPoet_Add_ons();
260
+ $mailpoet_add_ons->init_mail_poet_add_ons();
261
+ $mailpoet_add_ons->add_ons_page();
262
+ }
263
+ load_add_ons_manager();
add-ons/index.html ADDED
File without changes
classes/WJ_Analytics.php ADDED
@@ -0,0 +1,793 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ Class Analytics.
5
+ It's a sort of useful stats and numbers generator about MailPoet usage.
6
+ It also handles the MixPanel integration.
7
+ $analytics = new WJ_Analytics();
8
+ $analytics->generate_data();
9
+ $analytics->send();
10
+ */
11
+ class WJ_Analytics {
12
+
13
+ // All analytics data to be sent to JS.
14
+ private $analytics_data = array(
15
+ 'monthly_emails_sent' => array(
16
+ 'label' => 'Monthly emails sent',
17
+ 'value' => ''
18
+ ),
19
+ 'lists_with_more_than_25' => array(
20
+ 'label' => 'Lists with more than 25 subscribers',
21
+ 'value' => ''
22
+ ),
23
+ 'confirmed_subscribers' => array(
24
+ 'label' => 'Confirmed subscribers',
25
+ 'value' => ''
26
+ ),
27
+ 'unconfirmed_subscribers' => array(
28
+ 'label' => 'Unconfirmed subscribers',
29
+ 'value' => ''
30
+ ),
31
+ 'standard_newsletters' => array(
32
+ 'label' => 'Standard newsletters',
33
+ 'value' => ''
34
+ ),
35
+ 'auto_newsletters' => array(
36
+ 'label' => 'Auto newsletters',
37
+ 'value' => ''
38
+ ),
39
+ 'wordpress_version' => array(
40
+ 'label' => 'WordPress Version',
41
+ 'value' => ''
42
+ ),
43
+ 'plugin_version' => array(
44
+ 'label' => 'Plugin Version',
45
+ 'value' => ''
46
+ ),
47
+ 'license_type' => array(
48
+ 'label' => 'License type',
49
+ 'value' => ''
50
+ ),
51
+ 'sending_method' => array(
52
+ 'label' => 'Sending method',
53
+ 'value' => ''
54
+ ),
55
+ 'smtp_hostname' => array(
56
+ 'label' => 'Smtp hostname',
57
+ 'value' => ''
58
+ ),
59
+ 'activation_email_status' => array(
60
+ 'label' => 'Activation Email',
61
+ 'value' => ''
62
+ ),
63
+ 'average_open_rate' => array(
64
+ 'label' => 'Open rate',
65
+ 'value' => ''
66
+ ),
67
+ 'average_click_rate' => array(
68
+ 'label' => 'Click rate',
69
+ 'value' => ''
70
+ ),
71
+ 'industry' => array(
72
+ 'label' => 'Industry',
73
+ 'value' => ''
74
+ ),
75
+ 'wordpress_language' => array(
76
+ 'label' => 'WordPress Language',
77
+ 'value' => ''
78
+ ),
79
+ 'rtl' => array(
80
+ 'label' => 'Rtl',
81
+ 'value' => ''
82
+ ),
83
+ 'beta' => array(
84
+ 'label' => 'Beta',
85
+ 'value' => ''
86
+ ),
87
+ 'archive_page' => array(
88
+ 'label' => 'Archive Page',
89
+ 'value' => ''
90
+ ),
91
+ 'dkim_status' => array(
92
+ 'label' => 'DKIM Active',
93
+ 'value' => ''
94
+ ),
95
+ 'subscribe_in_comments' => array(
96
+ 'label' => 'Subscribe in comments',
97
+ 'value' => ''
98
+ ),
99
+ 'subscribe_on_register' => array(
100
+ 'label' => 'Subscribe on registration',
101
+ 'value' => ''
102
+ ),
103
+ 'browser_link' => array(
104
+ 'label' => 'View in browser link',
105
+ 'value' => ''
106
+ ),
107
+ 'profile_edit'=> array(
108
+ 'label' => 'Subcsribers can edit profile',
109
+ 'value' => ''
110
+ ),
111
+ 'html_edit' => array(
112
+ 'label' => 'Allow HTML edit',
113
+ 'value' => ''
114
+ ),
115
+ 'mailpoet_cron' => array(
116
+ 'label' => 'MailPoet Cron Enabled',
117
+ 'value' => ''
118
+ ),
119
+ 'forms_number' => array(
120
+ 'label' => 'Total number of forms',
121
+ 'value' => ''
122
+ ),
123
+ 'using_custom_fields' => array(
124
+ 'label' => 'Using custom fields',
125
+ 'value' => ''
126
+ ),
127
+ 'custom_fields_input' => array(
128
+ 'label' => 'Custom Fields: Input field',
129
+ 'value' => ''
130
+ ),
131
+ 'custom_fields_textarea' => array(
132
+ 'label' => 'Custom Fields: Textarea',
133
+ 'value' => ''
134
+ ),
135
+ 'custom_fields_select' => array(
136
+ 'label' => 'Custom Fields: Select',
137
+ 'value' => ''
138
+ ),
139
+ 'custom_fields_checkbox' => array(
140
+ 'label' => 'Custom Fields: Checkbox',
141
+ 'value' => ''
142
+ ),
143
+ 'custom_fields_radio' => array(
144
+ 'label' => 'Custom Fields: Radio',
145
+ 'value' => ''
146
+ ),
147
+ 'active_last_week' => array(
148
+ 'label' => 'Active last week',
149
+ 'value' => ''
150
+ ),
151
+ 'is_multisite' => array(
152
+ 'label' => 'Using Multisite',
153
+ 'value' => ''
154
+ )
155
+ );
156
+
157
+ function __construct() {
158
+
159
+ }
160
+
161
+ /**
162
+ * Send data to Mixpanel by enqueuing the analytics JS file.
163
+ * @return
164
+ */
165
+ public function send() {
166
+
167
+ // Enqueue analytics Javascript.
168
+ wp_enqueue_script('analytics', WYSIJA_URL . 'js/analytics.js', array(), WYSIJA::get_version());
169
+ // Make analytics data available in JS.
170
+ wp_localize_script('analytics', 'analytics_data', $this->analytics_data);
171
+ }
172
+
173
+ /**
174
+ * Generate fresh data and store it in the $analytics_data Class property.
175
+ * @return
176
+ */
177
+ public function generate_data() {
178
+
179
+ foreach ($this->analytics_data as $key => $data) {
180
+ $method = $key;
181
+ $this->analytics_data[$key]['value'] = $this->$method();
182
+ }
183
+
184
+ }
185
+
186
+ /**
187
+ * Calculate Emails sent in the last 30 days.
188
+ * @return Int
189
+ */
190
+ private function monthly_emails_sent() {
191
+
192
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
193
+ $query = 'SELECT COUNT(*) as total_emails
194
+ FROM ' . '[wysija]' . $model_email_user_stat->table_name . '
195
+ WHERE DATE_SUB(CURDATE(),INTERVAL 30 DAY) <= sent_at';
196
+ $result = $model_email_user_stat->query('get_res', $query);
197
+
198
+ $total_emails = $result[0]['total_emails'];
199
+ switch (true) {
200
+ case ($total_emails <= 1000):
201
+ $track = 'Less than 1000';
202
+ break;
203
+ case ((1001 <= $total_emails) && ($total_emails <= 2500)):
204
+ $track = '1001 to 2500';
205
+ break;
206
+ case ((2501 <= $total_emails) && ($total_emails <= 5000)):
207
+ $track = '2501 to 5000';
208
+ break;
209
+ case ((5001 <= $total_emails) && ($total_emails <= 10000)):
210
+ $track = '5001 to 10000';
211
+ break;
212
+ case ((10001 <= $total_emails) && ($total_emails <= 25000)):
213
+ $track = '10001 to 25000';
214
+ break;
215
+ case ((25001 <= $total_emails) && ($total_emails <= 50000)):
216
+ $track = '25001 to 50000';
217
+ break;
218
+ case ((50001 <= $total_emails) && ($total_emails <= 100000)):
219
+ $track = '50001 to 100000';
220
+ break;
221
+ case (100001 <= $total_emails):
222
+ $track = 'More than 100001';
223
+ break;
224
+ default:
225
+ $track = 'No emails sent';
226
+ break;
227
+ }
228
+
229
+ return $track;
230
+
231
+ }
232
+
233
+ /*
234
+ Pass through the WordPress version.
235
+ */
236
+ private function wordpress_version() {
237
+ return get_bloginfo('version');
238
+ }
239
+
240
+ /*
241
+ Pass through the WordPress language.
242
+ */
243
+ private function wordpress_language() {
244
+ return get_bloginfo('language');
245
+ }
246
+
247
+ /*
248
+ The plugin version.
249
+ */
250
+ private function plugin_version() {
251
+ return WYSIJA::get_version();
252
+ }
253
+
254
+ /**
255
+ * Calculate lists with more than 25 subscribers.
256
+ * @return Int
257
+ */
258
+ private function lists_with_more_than_25() {
259
+
260
+ $model_user_list = WYSIJA::get('user_list', 'model');
261
+ $query = 'SELECT list_id, COUNT(*) as count
262
+ FROM ' . '[wysija]' . $model_user_list->table_name . '
263
+ GROUP BY list_id
264
+ HAVING COUNT(*) >= 25';
265
+ $result = $model_user_list->query('get_res', $query);
266
+ $lists_count = count($result);
267
+
268
+ return $lists_count;
269
+ }
270
+
271
+ /**
272
+ * Calculate total subscribers.
273
+ * @return Int
274
+ */
275
+ private function total_subcsribers() {
276
+
277
+ $model_user = WYSIJA::get('user', 'model');
278
+ $query = 'SELECT COUNT(*) as total_subscribers
279
+ FROM ' . '[wysija]' . $model_user->table_name;
280
+ $result = $model_user->query('get_res', $query);
281
+
282
+ return $result[0]['total_subscribers'];
283
+ }
284
+
285
+ /**
286
+ * Calculate confirmed subscribers.
287
+ * @return Int
288
+ */
289
+ private function confirmed_subscribers() {
290
+
291
+ $model_user = WYSIJA::get('user', 'model');
292
+ $query = 'SELECT COUNT(*) as confirmed_subscribers
293
+ FROM ' . '[wysija]' . $model_user->table_name . '
294
+ WHERE status = 1';
295
+ $result = $model_user->query('get_res', $query);
296
+
297
+ $confirmed_subscribers = $result[0]['confirmed_subscribers'];
298
+ $total_subscribers = $this->total_subcsribers();
299
+ $confirmed_percentage = round(($confirmed_subscribers * 100) / $total_subscribers);
300
+
301
+ return $confirmed_percentage;
302
+ }
303
+
304
+ /**
305
+ * Calculate unconfirmed subscribers.
306
+ * @return Int
307
+ */
308
+ public function unconfirmed_subscribers() {
309
+
310
+ $model_user = WYSIJA::get('user', 'model');
311
+ $query = 'SELECT COUNT(*) as unconfirmed_subscribers
312
+ FROM ' . '[wysija]' . $model_user->table_name . '
313
+ WHERE status = 0';
314
+ $result = $model_user->query('get_res', $query);
315
+
316
+ return $result[0]['unconfirmed_subscribers'];
317
+ }
318
+
319
+ /**
320
+ * Calculate standard newsletters total.
321
+ * @return Int
322
+ */
323
+ private function standard_newsletters() {
324
+
325
+ $model_email = WYSIJA::get('email', 'model');
326
+ $query = 'SELECT COUNT(*) as standard_newsletters
327
+ FROM ' . '[wysija]' . $model_email->table_name . '
328
+ WHERE type = 1
329
+ AND status = 2';
330
+ $result = $model_email->query('get_res', $query);
331
+
332
+ return $result[0]['standard_newsletters'];
333
+ }
334
+
335
+ /**
336
+ * Calculate auto newsletters total.
337
+ * @return Int
338
+ */
339
+ private function auto_newsletters() {
340
+
341
+ $model_email = WYSIJA::get('email', 'model');
342
+ $query = 'SELECT COUNT(*) as auto_newsletters
343
+ FROM ' . '[wysija]' . $model_email->table_name . '
344
+ WHERE type = 2';
345
+ $result = $model_email->query('get_res', $query);
346
+
347
+ return $result[0]['auto_newsletters'];
348
+ }
349
+
350
+ /**
351
+ * Check license type in use.
352
+ * @return String Free | Premium
353
+ */
354
+ private function license_type() {
355
+
356
+ $model_config = WYSIJA::get('config', 'model');
357
+ $is_premium = $model_config->getValue('premium_key');
358
+
359
+ if ($is_premium) {
360
+ $license_type = 'Premium';
361
+ } else {
362
+ $license_type = 'Free';
363
+ }
364
+
365
+ return $license_type;
366
+ }
367
+
368
+ /**
369
+ * Get sending method in use.
370
+ * @return String
371
+ */
372
+ private function sending_method() {
373
+
374
+ $model_config = WYSIJA::get('config', 'model');
375
+ return $model_config->getValue('sending_method');
376
+ }
377
+
378
+ /**
379
+ * Get smtp hostname in use.
380
+ * @return String
381
+ */
382
+ private function smtp_hostname() {
383
+
384
+ $model_config = WYSIJA::get('config', 'model');
385
+ return $model_config->getValue('smtp_host');
386
+ }
387
+
388
+ /**
389
+ * Get activation email status.
390
+ * @return String On | Off
391
+ */
392
+ private function activation_email_status() {
393
+
394
+ $model_config = WYSIJA::get('config', 'model');
395
+ $activation_email_status = $model_config->getValue('confirm_dbleoptin');
396
+
397
+ if ($activation_email_status === 1) {
398
+ $result = 'On';
399
+ } else {
400
+ $result = 'Off';
401
+ }
402
+
403
+ return $result;
404
+ }
405
+
406
+ /**
407
+ * Get DKIM status.
408
+ * @return String Yes | No
409
+ */
410
+ private function dkim_status() {
411
+ $model_config = WYSIJA::get('config', 'model');
412
+ $dkim_status = $model_config->getValue('dkim_active');
413
+ if ($dkim_status === 1) {
414
+ $result = 'Yes';
415
+ } else {
416
+ $result = 'No';
417
+ }
418
+ return $result;
419
+ }
420
+
421
+ /**
422
+ * Get subscribe in comments.
423
+ * @return String Yes | No
424
+ */
425
+ private function subscribe_in_comments() {
426
+ $model_config = WYSIJA::get('config', 'model');
427
+ $subscribe_in_comments = $model_config->getValue('commentform');
428
+ if ($subscribe_in_comments == 1) {
429
+ $result = 'Yes';
430
+ } else {
431
+ $result = 'No';
432
+ }
433
+ return $result;
434
+ }
435
+
436
+ /**
437
+ * Get subscribe during registration option.
438
+ * @return String Yes | No
439
+ */
440
+ private function subscribe_on_register() {
441
+ $model_config = WYSIJA::get('config', 'model');
442
+ $subscribe_on_register = $model_config->getValue('registerform');
443
+ if ($subscribe_on_register == 1) {
444
+ $result = 'Yes';
445
+ } else {
446
+ $result = 'No';
447
+ }
448
+ return $result;
449
+ }
450
+
451
+ /**
452
+ * Get view in browser option.
453
+ * @return String Yes | No
454
+ */
455
+ private function browser_link() {
456
+ $model_config = WYSIJA::get('config', 'model');
457
+ $browser_link = $model_config->getValue('viewinbrowser');
458
+ if ($browser_link == 1) {
459
+ $result = 'Yes';
460
+ } else {
461
+ $result = 'No';
462
+ }
463
+ return $result;
464
+ }
465
+
466
+ /**
467
+ * Get profile edit option.
468
+ * @return String Yes | No
469
+ */
470
+ private function profile_edit() {
471
+ $model_config = WYSIJA::get('config', 'model');
472
+ $profile_edit = $model_config->getValue('manage_subscriptions');
473
+ if ($profile_edit == 1) {
474
+ $result = 'Yes';
475
+ } else {
476
+ $result = 'No';
477
+ }
478
+ return $result;
479
+ }
480
+
481
+ /**
482
+ * Get html edit in newsletter option.
483
+ * @return String Yes | No
484
+ */
485
+ private function html_edit() {
486
+ $model_config = WYSIJA::get('config', 'model');
487
+ $html_edit = $model_config->getValue('html_source');
488
+ if ($html_edit == 1) {
489
+ $result = 'Yes';
490
+ } else {
491
+ $result = 'No';
492
+ }
493
+ return $result;
494
+ }
495
+
496
+ /**
497
+ * Get mailpoet cron option.
498
+ * @return String Yes | No
499
+ */
500
+ private function mailpoet_cron() {
501
+ $model_config = WYSIJA::get('config', 'model');
502
+ $cron_option = $model_config->getValue('cron_manual');
503
+ if ($cron_option == 1) {
504
+ $result = 'Yes';
505
+ } else {
506
+ $result = 'No';
507
+ }
508
+ return $result;
509
+ }
510
+
511
+ /**
512
+ * Calculate average open rate.
513
+ * @return Int
514
+ */
515
+ private function average_open_rate() {
516
+
517
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
518
+ $query = 'SELECT COUNT(*) as opened_emails
519
+ FROM ' . '[wysija]' . $model_email_user_stat->table_name . '
520
+ WHERE status = 1';
521
+ $result = $model_email_user_stat->query('get_res', $query);
522
+
523
+ $opened_emails = $result[0]['opened_emails'];
524
+ $total_emails = $this->total_emails_sent();
525
+
526
+ if ($total_emails == 0) {
527
+ $average_open_rate = 0;
528
+ } else {
529
+ $average_open_rate = round(($opened_emails * 100) / $total_emails);
530
+ }
531
+
532
+ return $average_open_rate;
533
+ }
534
+
535
+ /**
536
+ * Calculate average click rate.
537
+ * @return String opened/total
538
+ */
539
+ private function average_click_rate() {
540
+
541
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
542
+ $query = 'SELECT COUNT(*) as clicked_emails
543
+ FROM ' . '[wysija]' . $model_email_user_stat->table_name . '
544
+ WHERE status = 2';
545
+ $result = $model_email_user_stat->query('get_res', $query);
546
+
547
+ $clicked_emails = $result[0]['clicked_emails'];
548
+ $total_emails = $this->total_emails_sent();
549
+
550
+ if ($total_emails == 0) {
551
+ $average_click_rate = 0;
552
+ } else {
553
+ $average_click_rate = round(($clicked_emails * 100) / $total_emails);
554
+ }
555
+
556
+ return $average_click_rate;
557
+ }
558
+
559
+ /**
560
+ * Get all emails sent.
561
+ * @return Int
562
+ */
563
+ private function total_emails_sent() {
564
+
565
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
566
+ $query = 'SELECT COUNT(*) as all_emails
567
+ FROM ' . '[wysija]' . $model_email_user_stat->table_name . '';
568
+ $result = $model_email_user_stat->query('get_res', $query);
569
+
570
+ return $result[0]['all_emails'];
571
+ }
572
+
573
+ /**
574
+ * Get total number of forms.
575
+ * @return Int
576
+ */
577
+ private function forms_number() {
578
+
579
+ $model_form = WYSIJA::get('forms', 'model');
580
+ $query = 'SELECT COUNT(*) as forms_number
581
+ FROM ' . '[wysija]' . $model_form->table_name . '';
582
+ $result = $model_form->query('get_res', $query);
583
+
584
+ return $result[0]['forms_number'];
585
+ }
586
+
587
+ /**
588
+ * Get Industry specified in the settings page.
589
+ * @return String
590
+ */
591
+ public function industry() {
592
+ $model_config = WYSIJA::get('config', 'model');
593
+ return $model_config->getValue('industry');
594
+ }
595
+
596
+ /**
597
+ * Get if is using right to left language.
598
+ * @return String
599
+ */
600
+ private function rtl() {
601
+
602
+ if (is_rtl()) {
603
+ $is_rtl = 'Yes';
604
+ } else {
605
+ $is_rtl = 'No';
606
+ }
607
+
608
+ return $is_rtl;
609
+ }
610
+
611
+ /**
612
+ * Check if it's multisite.
613
+ * @return String
614
+ */
615
+ private function is_multisite() {
616
+
617
+ if (is_multisite()) {
618
+ $is_multisite = 'Yes';
619
+ } else {
620
+ $is_multisite = 'No';
621
+ }
622
+
623
+ return $is_multisite;
624
+ }
625
+
626
+ /**
627
+ * Get if is using beta mode
628
+ * @return String
629
+ */
630
+ private function beta() {
631
+
632
+ $model_config = WYSIJA::get('config', 'model');
633
+
634
+ if ($model_config->getValue('beta_mode')) {
635
+ $is_beta = 'Yes';
636
+ } else {
637
+ $is_beta = 'No';
638
+ }
639
+
640
+ return $is_beta;
641
+ }
642
+
643
+ /**
644
+ * Checks if user used the archive page feature.
645
+ * @return String true | false
646
+ */
647
+ private function archive_page() {
648
+
649
+ $model_config = WYSIJA::get('config', 'model');
650
+ $archive_lists = $model_config->getValue('archive_lists');
651
+
652
+ if (!empty($archive_lists)) {
653
+ $used_archive = 'true';
654
+ } else {
655
+ $used_archive = 'false';
656
+ }
657
+
658
+ return $used_archive;
659
+ }
660
+
661
+ /*
662
+ Check if user is using custom fields or not.
663
+ # => Yes | No
664
+ */
665
+ private function using_custom_fields() {
666
+ $fields = WJ_Field::get_all();
667
+ if ($fields != null) {
668
+ $result = 'Yes';
669
+ } else {
670
+ $result = 'No';
671
+ }
672
+ return $result;
673
+ }
674
+
675
+ /*
676
+ How many input custom fields are in use.
677
+ # => int | null
678
+ */
679
+ private function custom_fields_input() {
680
+ global $wpdb;
681
+ $field = new WJ_Field();
682
+ $table_name = $field->get_table_name();
683
+ $result = $wpdb->get_var(
684
+ "SELECT COUNT(*) as inputs
685
+ FROM $table_name
686
+ WHERE type = 'input'"
687
+ );
688
+ if ($result == null) {
689
+ $result = '0';
690
+ }
691
+ return $result;
692
+ }
693
+
694
+ /*
695
+ How many textarea custom fields are in use.
696
+ # => int | null
697
+ */
698
+ private function custom_fields_textarea() {
699
+ global $wpdb;
700
+ $field = new WJ_Field();
701
+ $table_name = $field->get_table_name();
702
+ $result = $wpdb->get_var(
703
+ "SELECT COUNT(*) as textareas
704
+ FROM $table_name
705
+ WHERE type = 'textarea'"
706
+ );
707
+ if ($result == null) {
708
+ $result = '0';
709
+ }
710
+ return $result;
711
+ }
712
+
713
+ /*
714
+ How many select custom fields are in use.
715
+ # => int | null
716
+ */
717
+ private function custom_fields_select() {
718
+ global $wpdb;
719
+ $field = new WJ_Field();
720
+ $table_name = $field->get_table_name();
721
+ $result = $wpdb->get_var(
722
+ "SELECT COUNT(*) as selects
723
+ FROM $table_name
724
+ WHERE type = 'select'"
725
+ );
726
+ if ($result == null) {
727
+ $result = '0';
728
+ }
729
+ return $result;
730
+ }
731
+
732
+ /*
733
+ How many checkbox custom fields are in use.
734
+ # => int | null
735
+ */
736
+ private function custom_fields_checkbox() {
737
+ global $wpdb;
738
+ $field = new WJ_Field();
739
+ $table_name = $field->get_table_name();
740
+ $result = $wpdb->get_var(
741
+ "SELECT COUNT(*) as checkboxes
742
+ FROM $table_name
743
+ WHERE type = 'checkbox'"
744
+ );
745
+ if ($result == null) {
746
+ $result = '0';
747
+ }
748
+ return $result;
749
+ }
750
+
751
+ /*
752
+ How many checkbox custom fields are in use.
753
+ # => int | null
754
+ */
755
+ private function custom_fields_radio() {
756
+ global $wpdb;
757
+ $field = new WJ_Field();
758
+ $table_name = $field->get_table_name();
759
+ $result = $wpdb->get_var(
760
+ "SELECT COUNT(*) as radios
761
+ FROM $table_name
762
+ WHERE type = 'radio'"
763
+ );
764
+ if ($result == null) {
765
+ $result = '0';
766
+ }
767
+ return $result;
768
+ }
769
+
770
+ /*
771
+ Check if user has been active in the last week.
772
+ This means he sent at least one email.
773
+ # => Yes | No
774
+ */
775
+ private function active_last_week() {
776
+ global $wpdb;
777
+ $model_stats = WYSIJA::get('email_user_stat', 'model');
778
+ $table_name = '[wysija]' . $model_stats->table_name;
779
+
780
+ $query = 'SELECT COUNT(*) as activities
781
+ FROM ' . $table_name .
782
+ ' WHERE sent_at > UNIX_TIMESTAMP(date_sub(now(), interval 1 week))';
783
+
784
+ $result = $model_stats->query('get_res', $query);
785
+ $result = $result[0]['activities'];
786
+
787
+ if ($result > 0) {
788
+ return 'Yes';
789
+ }
790
+ return 'No';
791
+ }
792
+
793
+ }
classes/WJ_Export.php ADDED
@@ -0,0 +1,216 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Class Export.
5
+ *
6
+ * Exporting subscribers
7
+ */
8
+ class WJ_Export extends WYSIJA_object {
9
+
10
+ private $_file_header = '';
11
+ private $_file_handle = null;
12
+ private $_user_ids_rows = 0;
13
+ private $_user_ids = array();
14
+ private $_fields = array();
15
+ private $_export_batch = 2000;
16
+ private $_filter_list = '';
17
+ private $_filter_confirmed = '';
18
+ private $_fields_separator = ';';
19
+ private $_lines_separator = "\n";
20
+ private $_base_encode = 'UTF-8';
21
+ private $_output_encode = 'UTF-8//TRANSLIT//IGNORE';
22
+
23
+ //needed for tung's batch actions
24
+ public $batch_select = null;
25
+
26
+ function __construct() {
27
+ if ( ! empty( $_POST['wysija']['export']['fields'] ) ) {
28
+ $this->_fields = $_POST['wysija']['export']['fields'];
29
+ }
30
+
31
+ if ( ! empty( $_POST['wysija']['export']['user_ids'] ) ) {
32
+ $this->_user_ids = $this->_get_posted_user_ids();
33
+ }
34
+ if ( ! empty($_POST['wysija']['export']['filter']['list'] ) ) {
35
+ $this->_filter_list = $_POST['wysija']['export']['filter']['list'];
36
+ }
37
+
38
+ // how do we separate the values in the files commas or semy colons ?
39
+ if ( ! empty($_POST['wysija']['export']['format'] ) ) {
40
+ $this->_fields_separator = $_POST['wysija']['export']['format'];
41
+ }
42
+
43
+ if ( ! empty($_POST['wysija']['export']['filter']['confirmed'] ) ) {
44
+ $this->_filter_confirmed = $_POST['wysija']['export']['filter']['confirmed'];
45
+ }
46
+ }
47
+
48
+ /**
49
+ * export the subscribers
50
+ * @return type
51
+ */
52
+ public function export_subscribers() {
53
+
54
+ //generate temp file
55
+ $helper_file = WYSIJA::get( 'file', 'helper' );
56
+ $this->_prepare_headers();
57
+ $result_file = $helper_file->temp( $this->_file_header, 'export', '.csv' );
58
+
59
+ //open the created file in append mode
60
+ $this->_file_handle = fopen( $result_file['path'], 'a' );
61
+
62
+ //get a list of user_ids to export
63
+ if ( ! empty( $this->_user_ids ) && empty( $this->batch_select ) ) {
64
+
65
+ $this->_user_ids_rows = count( $this->_user_ids );
66
+ $this->_push_data_to_export_file();
67
+ } else {
68
+
69
+ $this->_get_chunks_user_ids();
70
+ }
71
+
72
+ fclose( $this->_file_handle );
73
+ return $result_file['path'];
74
+ }
75
+
76
+ /**
77
+ * get the number of rows exported
78
+ * @return type
79
+ */
80
+ public function get_user_ids_rows() {
81
+ return $this->_user_ids_rows;
82
+ }
83
+
84
+ /**
85
+ * get an array of user_ids from the global $_POST
86
+ * @return type
87
+ */
88
+ private function _get_posted_user_ids() {
89
+ return unserialize( base64_decode( $_POST['wysija']['export']['user_ids'] ) );
90
+ }
91
+
92
+ /**
93
+ * get the query used to select a bung of ids
94
+ * @return string
95
+ */
96
+ private function _get_query_users_ids() {
97
+
98
+ // based on filters prepare a query to get a list of user_ids
99
+ if ( ! empty( $this->batch_select ) ) { // batch select and export
100
+ $this->_user_ids_rows = $this->batch_select['count'];
101
+ $qry = $this->batch_select['original_query'];
102
+ } else { // export all list
103
+ // prepare the filters
104
+ $filters = array();
105
+ if ( ! empty( $this->_filter_list ) ) {
106
+ if ( ! is_array( $this->_filter_list ) ) {
107
+ $this->_filter_list = array( $this->_filter_list );
108
+ }
109
+ }
110
+ $filters['lists'] = $this->_filter_list;
111
+
112
+ // include also unsubscribed and unconfirmed
113
+ if ( ! empty( $this->_filter_confirmed ) ) {
114
+ $filters['status'] = 'subscribed';
115
+ }
116
+
117
+ $model_user = WYSIJA::get( 'user', 'model' );
118
+ $select = array( 'A.user_id' );
119
+ $qry = $model_user->get_subscribers( $select, $filters, '', $return_query = true );
120
+ }
121
+
122
+ return $qry;
123
+ }
124
+
125
+ /**
126
+ * get chunks of subscribers ids and push them step by step to the export file
127
+ */
128
+ private function _get_chunks_user_ids() {
129
+
130
+ $model_user = WYSIJA::get( 'user', 'model' );
131
+ $this->_user_ids = array();
132
+ $query_user_ids = $this->_get_query_users_ids();
133
+ $query_count = str_replace( array( 'DISTINCT(A.user_id)', 'DISTINCT(B.user_id)' ), 'COUNT(DISTINCT(A.user_id))', $query_user_ids );
134
+
135
+ if ( empty( $this->_user_ids_rows ) ) {
136
+ $useridsrows_result = $model_user->getResults( $query_count, ARRAY_N );
137
+ $this->_user_ids_rows = (int) $useridsrows_result[0][0];
138
+ }
139
+
140
+ if ( $this->_user_ids_rows <= $this->_export_batch ) {
141
+ $user_ids_db = $model_user->getResults( $query_user_ids, ARRAY_N );
142
+
143
+ foreach ( $user_ids_db as $uarr ) {
144
+ $this->_user_ids[] = $uarr[0];
145
+ }
146
+
147
+ $this->_push_data_to_export_file();
148
+ } else {
149
+ $pages = ceil( $this->_user_ids_rows / $this->_export_batch ); //pagination
150
+ for ( $i = 0; $i < $pages; $i++ ) {
151
+ $query_batch = $query_user_ids . ' ORDER BY user_id ASC LIMIT ' . ($i * $this->_export_batch) . ',' . $this->_export_batch;
152
+ $user_ids_db = $model_user->getResults( $query_batch, ARRAY_N );
153
+ foreach ( $user_ids_db as $uarr ) {
154
+ $this->_user_ids[] = $uarr[0];
155
+ }
156
+ $this->_push_data_to_export_file();
157
+
158
+ unset($user_ids_db); //free memory
159
+ }
160
+ }
161
+ }
162
+
163
+ /**
164
+ * split the user_ids array into chunks, load the fields of all the concerned
165
+ * users and push the data to the file
166
+ */
167
+ function _push_data_to_export_file() {
168
+ $user_ids_chunks = array(); // chunk rows into separated batchs, limit by $this->_export_batch
169
+ $user_ids_chunks = array_chunk( $this->_user_ids, 200 );
170
+ $this->_user_ids = null; // free memory
171
+
172
+ $model_user = WYSIJA::get( 'user', 'model' );
173
+ foreach ( $user_ids_chunks as $user_id_chunk ) {
174
+ // get the full data for that specific chunk of ids
175
+ $data = $model_user->get( $this->_fields, array( 'user_id' => $user_id_chunk ) );
176
+
177
+ if ( in_array( 'created_at', $this->_fields ) ) {
178
+ foreach ( $data as $key => $row ) {
179
+ $data[$key]['created_at'] = date_i18n( get_option( 'date_format' ), $row['created_at'] );
180
+ }
181
+ }
182
+
183
+ $rows_count = count( $data );
184
+
185
+ // As required in Wysija/plugin#798 removed BOM from file
186
+ // fwrite( $this->_file_handle, "\xEF\xBB\xBF" );
187
+
188
+ // append content to the file
189
+ foreach ( $data as $k => $row ) {
190
+ $row_string = implode( $this->_fields_separator, $row );
191
+ $encoded_string = iconv( $this->_base_encode, $this->_output_encode, $row_string );
192
+ fwrite( $this->_file_handle, $encoded_string . ( $rows_count !== $k ? $this->_lines_separator : '' ) );
193
+ }
194
+ }
195
+ }
196
+
197
+ /**
198
+ * simply prepare the header of the file based on the fields
199
+ */
200
+ private function _prepare_headers() {
201
+ $model_user = WYSIJA::get( 'user_field', 'model' );
202
+ $database_fields = $model_user->getFields();
203
+
204
+ $name_fields = array();
205
+ //prepare the columns that need to be exported
206
+ foreach ( $this->_fields as $key_field ) {
207
+ $name_fields[] = $database_fields[$key_field];
208
+ }
209
+
210
+ //create the export file step by step
211
+ $row_string = implode( $this->_fields_separator, $name_fields ) . $this->_lines_separator;
212
+ $encoded_string = iconv( $this->_base_encode, $this->_output_encode, $row_string );
213
+ $this->_file_header = $encoded_string;
214
+ }
215
+
216
+ }
classes/WJ_Field.php ADDED
@@ -0,0 +1,283 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ A custom field.
5
+ It represents a custom field that
6
+ can be added to a form.
7
+ Example:
8
+ $custom_field = new WJ_Field();
9
+ $custom_field->set(array(
10
+ 'name' => 'Fruits',
11
+ 'type' => 'select',
12
+ 'required' => false,
13
+ 'settings' => array(
14
+ 'label' => 'Select a fruit:',
15
+ 'values' => array(
16
+ 'Apple', 'Orange', 'Banana'
17
+ ),
18
+ 'default_value' => 'Orange'
19
+ )
20
+ ));
21
+ $custom_field->save();
22
+ */
23
+
24
+ class WJ_Field {
25
+
26
+ // The custom fields DB table.
27
+ private $table;
28
+ // The Users table.
29
+ private $user_table;
30
+ // The ID of the custom field.
31
+ public $id;
32
+ // The name of the custom field.
33
+ public $name;
34
+ // input, textarea, checkbox, radio, select
35
+ public $type;
36
+ // true | false
37
+ public $required;
38
+ // Settings.
39
+ public $settings = array();
40
+
41
+ /*
42
+ Just set the correct tables on creation.
43
+ $custom_field = new WJ_Field();
44
+ */
45
+ function __construct() {
46
+ $this->table = WJ_Settings::db_prefix( 'custom_field' );
47
+ $this->user_table = WJ_Settings::db_prefix( 'user' );
48
+ }
49
+
50
+ /*
51
+ Get the table name.
52
+ Useful for statistics and counts on Custom Fields.
53
+ Generally when we want to run DB queries.
54
+ */
55
+ public function get_table_name() {
56
+ return $this->table;
57
+ }
58
+
59
+ /*
60
+ Static method to get a Custom field by id.
61
+ WJ_Field::get(id);
62
+ # => WJCustomField Object
63
+ */
64
+ public static function get( $id ) {
65
+ $object = new self();
66
+ global $wpdb;
67
+ $result = $wpdb->get_row( $wpdb->prepare( "SELECT * from {$object->table} WHERE id = %d", $id ), ARRAY_A );
68
+ if ( $result != null ) {
69
+ $result['settings'] = unserialize( $result['settings'] );
70
+ $result['required'] = WJ_Utils::to_bool( $result['required'] );
71
+ $object->set( $result );
72
+ return $object;
73
+ } else {
74
+ return null;
75
+ }
76
+ }
77
+
78
+ /*
79
+ Get all custom fields.
80
+ WJ_Field::get_all();
81
+ # => Array of WJ_Field
82
+ */
83
+ public static function get_all( $options = array() ) {
84
+ $object = new self();
85
+ global $wpdb;
86
+
87
+ // default order by
88
+ $order_by = 'id ASC';
89
+ if ( isset($options['order_by'] ) ){
90
+ $order_by = $options['order_by'];
91
+ }
92
+
93
+ // fetch rows from db
94
+ $results = $wpdb->get_results( $wpdb->prepare( "SELECT * FROM {$object->table} ORDER BY %s", $order_by ), ARRAY_A );
95
+
96
+ if ( $results != null ) {
97
+ $collection = array();
98
+ foreach ( $results as $result ) {
99
+ $result['settings'] = unserialize( $result['settings'] );
100
+ $result['required'] = WJ_Utils::to_bool( $result['required'] );
101
+ $field = new self();
102
+ $field->set( $result );
103
+ $collection[] = $field;
104
+ }
105
+ return $collection;
106
+ } else {
107
+ return null;
108
+ }
109
+ }
110
+
111
+
112
+ /*
113
+ Get a custom fields names list.
114
+ WJ_Field::get_names_list();
115
+ # => array(1 => 'Address', 2 => 'Gender')
116
+ */
117
+ public static function get_all_names() {
118
+ $fields = self::get_all();
119
+ $fields_list = array();
120
+ if ( isset( $fields ) ) {
121
+ foreach ( $fields as $field ) {
122
+ $fields_list[$field->id] = $field->name;
123
+ }
124
+ }
125
+ return $fields_list;
126
+ }
127
+
128
+ /*
129
+ Set all object properties.
130
+ $custom_field->set(array(
131
+ 'name' => 'First Name',
132
+ 'type' => 'text',
133
+ 'required' => true
134
+ ));
135
+ */
136
+ public function set( $args ) {
137
+ if ( isset( $args['id'] ) ) {
138
+ $this->id = $args['id'];
139
+ }
140
+ $this->name = $args['name'];
141
+ $this->type = $args['type'];
142
+ $this->required = $args['required'];
143
+ $this->settings = $args['settings'];
144
+ }
145
+
146
+ /*
147
+ Store Custom Field in DB.
148
+ If already stored, updates it.
149
+ $custom_field->save();
150
+ */
151
+ public function save() {
152
+ // Check if it's a new object or an update.
153
+ if ( isset( $this->id ) ) {
154
+ $this->update();
155
+ } else {
156
+ $this->create();
157
+ }
158
+ }
159
+
160
+ // Delete custom field from DB.
161
+ public function delete() {
162
+ global $wpdb;
163
+ $result = $wpdb->delete( $this->table, array( 'id' => $this->id ), array( '%d' ) );
164
+ $this->delete_user_col( $this->id );
165
+ return $result;
166
+ }
167
+
168
+ /*
169
+ Generates user column name;
170
+ $custom_field->user_column_name();
171
+ # => 'cf_1'
172
+ */
173
+ public function user_column_name() {
174
+ $column_name = 'cf_' . $this->id;
175
+ return $column_name;
176
+ }
177
+
178
+ /*
179
+ Creates the Custom Field.
180
+ It also creates the user column, depending on type.
181
+ */
182
+ private function create() {
183
+ global $wpdb;
184
+ $required = WJ_Utils::to_int( $this->required );
185
+ $wpdb->insert(
186
+ $this->table,
187
+ array(
188
+ 'name' => $this->name,
189
+ 'type' => $this->type,
190
+ 'required' => $required,
191
+ 'settings' => serialize( $this->settings ),
192
+ ),
193
+ array( '%s', '%s', '%d', '%s', '%s' )
194
+ );
195
+ // ! No id in user col?
196
+ if ( $wpdb->insert_id ) {
197
+ $this->id = $wpdb->insert_id;
198
+ $this->create_user_col();
199
+ } else {
200
+ return false;
201
+ }
202
+ }
203
+
204
+ /*
205
+ Updates the value of the custom field.
206
+ $custom_field->update('New address');
207
+ */
208
+ private function update() {
209
+ global $wpdb;
210
+ $required = WJ_Utils::to_int( $this->required );
211
+ $result = $wpdb->update(
212
+ $this->table,
213
+ array(
214
+ 'name' => $this->name,
215
+ 'type' => $this->type,
216
+ 'required' => $required,
217
+ 'settings' => serialize( $this->settings ),
218
+ ),
219
+ array( 'id' => $this->id ),
220
+ array( '%s', '%s', '%d' ),
221
+ array( '%d' )
222
+ );
223
+ return $result;
224
+ }
225
+
226
+ /*
227
+ Creates the correct user columnn, named cf_x, in the
228
+ user table. x is the ID if the custom field.
229
+ */
230
+ private function create_user_col() {
231
+ global $wpdb;
232
+ $column_name = $this->user_column_name();
233
+ $column_type = $this->generate_column_type();
234
+ $result = $wpdb->query(
235
+ "ALTER TABLE $this->user_table ADD COLUMN $column_name $column_type"
236
+ );
237
+ return $result;
238
+ }
239
+
240
+ /*
241
+ Calculates the correct column type, based on custom field type.
242
+ $custom_field->generate_column_type();
243
+ # => 'VARCHAR(100)'
244
+ */
245
+ private function generate_column_type() {
246
+ switch ( $this->type ) {
247
+ case 'input':
248
+ $column_type = 'VARCHAR(100)';
249
+ break;
250
+ case 'textarea':
251
+ $column_type = 'VARCHAR(255)';
252
+ break;
253
+ case 'checkbox':
254
+ $column_type = 'TINYINT(1)';
255
+ break;
256
+ case 'radio':
257
+ $column_type = 'VARCHAR(255)';
258
+ break;
259
+ case 'select':
260
+ $column_type = 'VARCHAR(255)';
261
+ break;
262
+ case 'date':
263
+ $column_type = 'INT(10) UNSIGNED';
264
+ break;
265
+ default:
266
+ $column_type = 'VARCHAR(255)';
267
+ break;
268
+ }
269
+ return $column_type;
270
+ }
271
+
272
+ /*
273
+ Deletes the user column in the user table.
274
+ Needed when we remove a custom field.
275
+ */
276
+ private function delete_user_col( $custom_field_id ) {
277
+ global $wpdb;
278
+ $cf_column = 'cf_' . $custom_field_id;
279
+ $result = $wpdb->query( "ALTER TABLE $this->user_table DROP COLUMN $cf_column" );
280
+ return $result;
281
+ }
282
+
283
+ }
classes/WJ_FieldHandler.php ADDED
@@ -0,0 +1,59 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ Class WJ_FieldHandler
5
+ It handles the custom fields in a controller.
6
+ */
7
+ class WJ_FieldHandler {
8
+
9
+ public function __construct() {
10
+ }
11
+
12
+ /*
13
+ Handler to save all the fields, by giving an array
14
+ that specifies the custom fields columns names, and
15
+ the ID of the user.
16
+ WJ_FieldHandler::handle_all(array(
17
+ 'cf_1' => 'Value',
18
+ 'cf_2' => 'Value 2'
19
+ ), 1
20
+ );
21
+ */
22
+ public static function handle_all(array $fields_values, $user_id) {
23
+ $user_fields = WJ_FieldUser::get_all($user_id);
24
+
25
+ if(isset($user_fields) && !empty($user_fields)) {
26
+ foreach ($user_fields as $user_field) {
27
+ $key = $user_field->column_name();
28
+
29
+ // check if there is a value for this custom field
30
+ if(array_key_exists($key, $fields_values)) {
31
+
32
+ // set value
33
+ $new_value = $fields_values[$key];
34
+
35
+ // extra process for specific types
36
+ if($user_field->field->type === 'checkbox') {
37
+ // limit the value between [0,1]
38
+ $new_value = min(max(0, (int)$new_value), 1);
39
+
40
+ } else if($user_field->field->type === 'date') {
41
+ // get date parameters
42
+ $year = (isset($fields_values[$key]['year'])) ? (int)$fields_values[$key]['year'] : (int)strftime('%Y');
43
+ $month = (isset($fields_values[$key]['month'])) ? (int)$fields_values[$key]['month'] : 1;
44
+ $day = (isset($fields_values[$key]['day'])) ? (int)$fields_values[$key]['day'] : 1;
45
+
46
+ $new_value = null;
47
+ if($year !== 0 && $month !== 0 && $day !== 0) {
48
+ // make timestamp from date parameters
49
+ $new_value = mktime(0, 0, 0, $month, $day, $year);
50
+ }
51
+ }
52
+
53
+ // update value
54
+ $user_field->update($new_value);
55
+ }
56
+ }
57
+ }
58
+ }
59
+ }
classes/WJ_FieldRender.php ADDED
@@ -0,0 +1,249 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ Class WJ_FieldRender
5
+ It is able to render Custom Fields in a form.
6
+ */
7
+
8
+ class WJ_FieldRender {
9
+
10
+ // WJ_FieldUser Object.
11
+ private $field_user;
12
+ // The column, like 'cf_1'
13
+ private $identifier;
14
+ // Name of the field.
15
+ private $name;
16
+ // Value of the field.
17
+ private $value;
18
+
19
+ public function __construct($field_user) {
20
+ $this->field_user = $field_user;
21
+ $this->identifier = $field_user->column_name();
22
+ $this->name = esc_attr($field_user->field->name);
23
+ $this->value = esc_attr($field_user->value());
24
+ }
25
+
26
+ public function validation() {
27
+ $rules = array();
28
+ $validation_class = '';
29
+
30
+ // If there are any validations, add correct class names.
31
+ if (!empty($this->field_user->field->settings['validate'])) {
32
+ $validation_value = $this->field_user->field->settings['validate'];
33
+ $rules[] = 'custom[' . $validation_value . ']';
34
+ }
35
+
36
+ // If there's any validation, join all classes.
37
+ if (!empty($rules)) {
38
+ $validation_class = 'class="validate[' . join(',', $rules) . ']"';
39
+ }
40
+
41
+ return $validation_class;
42
+ }
43
+
44
+ /*
45
+ Returns the HTML label for this field, to be used
46
+ in a form.
47
+ $field_render->label();
48
+ # => '<label ...>Address</label>'
49
+ */
50
+ public function label() {
51
+ $form_field_name = $this->identifier;
52
+ $label =
53
+ '<label for="' . $form_field_name . '">' .
54
+ $this->name .
55
+ '</label>';
56
+ return $label;
57
+ }
58
+
59
+ /*
60
+ Returns the input related to the type of the field,
61
+ to be used in a form.
62
+ $field_render->input();
63
+ # => '<input type=..../>
64
+ */
65
+ public function input() {
66
+ $input = '';
67
+
68
+ switch ($this->field_user->field->type) {
69
+ case 'input':
70
+ $input =
71
+ '<input type="text" size="40" id="'. $this->identifier . '"' .
72
+ 'value="' . $this->value .
73
+ '" name="wysija[field][' . $this->identifier . ']" ' .
74
+ $this->validation() .
75
+ ' />';
76
+ break;
77
+ case 'textarea':
78
+ $input =
79
+ '<textarea id="'. $this->identifier . '"' .
80
+ 'name="wysija[field][' . $this->identifier . ']" ' .
81
+ $this->validation() .
82
+ ' />' .
83
+ $this->value .
84
+ '</textarea>';
85
+ break;
86
+ case 'checkbox':
87
+ $check_value = '';
88
+ $label = $this->field_user->field->settings['values'][0]['value'];
89
+ if ($this->value == 1) {
90
+ $check_value = ' checked="checked"';
91
+ }
92
+ $input =
93
+ '<label for="' . $this->identifier . '">' .
94
+ '<input type="hidden" '.
95
+ $check_value .
96
+ ' name="wysija[field][' . $this->identifier . ']"' .
97
+ ' value="0" ' .
98
+ ' />' .
99
+ '<input type="checkbox" id="'. $this->identifier . '"' .
100
+ $check_value .
101
+ ' name="wysija[field][' . $this->identifier . ']"' .
102
+ ' value="1" ' .
103
+ $this->validation() .
104
+ ' />' . $label . '</label>';
105
+ break;
106
+ case 'radio':
107
+ $field = $this->field_user->field;
108
+ foreach ($field->settings['values'] as $index => $content) {
109
+ $check_value = '';
110
+ if ($this->value == $content['value']) {
111
+ $check_value = 'checked="checked"';
112
+ }
113
+ $local_identifier = $this->identifier . '_' . $content['value'];
114
+ $input .=
115
+ '<label for="' . $local_identifier . '">' .
116
+ '<input type="radio" id="'. $local_identifier . '"' .
117
+ $check_value .
118
+ ' name="wysija[field][' . $this->identifier . ']"' .
119
+ ' value="' . $content['value'] . '" />' .
120
+ $content['value'] .
121
+ '</label>';
122
+ }
123
+ break;
124
+ case 'select':
125
+ $input = '<select id="' . $this->identifier . '"' .
126
+ ' name="wysija[field][' . $this->identifier . ']" ' .
127
+ $this->validation() .
128
+ ' />';
129
+ $field = $this->field_user->field;
130
+ foreach ($field->settings['values'] as $index => $content) {
131
+ $check_value = '';
132
+ if ($this->value == $content['value']) {
133
+ $check_value = 'selected="selected"';
134
+ }
135
+ $input .=
136
+ '<option ' . $check_value .
137
+ ' value="' . $content['value'] . '" >' .
138
+ $content['value'] .
139
+ '</option>';
140
+ }
141
+ $input .= '</select>';
142
+ break;
143
+
144
+ case 'date':
145
+ // get date format from field settings
146
+ $field = $this->field_user->field;
147
+
148
+ // get timestamp value
149
+ $value = ((int)$this->value > 0) ? (int)$this->value : null;
150
+
151
+ // get date type (defaults to year + month + day)
152
+ $date_type = (isset($field->settings['date_type'])) ? $field->settings['date_type'] : 'year_month_day';
153
+ // get an array of all required date components (year, month, day)
154
+ $display_date_fields = explode('_', $date_type);
155
+ // form engine to get date data
156
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
157
+
158
+ // year selection
159
+ if(in_array('year', $display_date_fields)) {
160
+ $years = $helper_form_engine->get_years();
161
+
162
+ $selected_year = null;
163
+ if($value !== null) {
164
+ $selected_year = (int)strftime('%Y', $value);
165
+ }
166
+
167
+ // select
168
+ $input .= '<select name="wysija[field]['.$this->identifier.'][year]">';
169
+ $input .= '<option value="">'.__('Year').'</option>';
170
+ foreach($years as $year) {
171
+ $is_selected = ((int)$year['year'] === $selected_year) ? ' selected="selected"' : '';
172
+ $input .= '<option value="'.$year['year'].'"'.$is_selected.'>'.$year['year'].'</option>';
173
+ }
174
+ $input .= '</select>';
175
+ }
176
+
177
+ // month selection
178
+ if(in_array('month', $display_date_fields)) {
179
+ $months = $helper_form_engine->get_months();
180
+
181
+ $selected_month = null;
182
+ if($value !== null) {
183
+ $selected_month = (int)strftime('%m', $value);
184
+ }
185
+
186
+ // select
187
+ $input .= '<select name="wysija[field]['.$this->identifier.'][month]">';
188
+ $input .= '<option value="">'.__('Month').'</option>';
189
+ foreach($months as $month) {
190
+ $is_selected = ((int)$month['month'] === $selected_month) ? ' selected="selected"' : '';
191
+ $input .= '<option value="'.$month['month'].'"'.$is_selected.'>'.$month['month_name'].'</option>';
192
+ }
193
+ $input .= '</select>';
194
+ }
195
+
196
+ // day selection
197
+ if(in_array('day', $display_date_fields)) {
198
+ $days = $helper_form_engine->get_days();
199
+
200
+ $selected_day = null;
201
+ if($value !== null) {
202
+ $selected_day = (int)strftime('%d', $value);
203
+ }
204
+
205
+ // select
206
+ $input .= '<select name="wysija[field]['.$this->identifier.'][day]">';
207
+ $input .= '<option value="">'.__('Day').'</option>';
208
+ foreach($days as $day) {
209
+ $is_selected = ((int)$day['day'] === $selected_day) ? ' selected="selected"' : '';
210
+ $input .= '<option value="'.$day['day'].'"'.$is_selected.'>'.$day['day'].'</option>';
211
+ }
212
+ $input .= '</select>';
213
+ }
214
+ break;
215
+ default:
216
+ $input = '';
217
+ break;
218
+ }
219
+ return $input;
220
+ }
221
+
222
+ /*
223
+ Render all custom fields in a table, given the user id.
224
+ $field_render::render_all(1);
225
+ # => '<tr><th><label...<input...</td></tr>'
226
+ */
227
+ public static function render_all($user_id) {
228
+ $fields = WJ_FieldUser::get_all($user_id);
229
+ if (isset($fields)) {
230
+ $output = '';
231
+ foreach ($fields as $field) {
232
+ $field_render = new self($field);
233
+ $output .=
234
+ '<tr>' .
235
+ '<th scope="row">' .
236
+ $field_render->label() .
237
+ '</th><td>' .
238
+ $field_render->input() .
239
+ '</td></tr>';
240
+ }
241
+ return $output;
242
+ } else {
243
+ return null;
244
+ }
245
+ }
246
+
247
+
248
+ }
249
+
classes/WJ_FieldUser.php ADDED
@@ -0,0 +1,130 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ An user field is a custom field identified by ID,
5
+ linked to a particular user, identified by ID.
6
+ */
7
+
8
+ class WJ_FieldUser {
9
+
10
+ // User Table.
11
+ private $user_table;
12
+ // Custom fields table.
13
+ private $field_table;
14
+ // User ID.
15
+ public $user_id;
16
+ // The field object.
17
+ public $field;
18
+ // The field value.
19
+ public $value;
20
+
21
+ /*
22
+ $field_user = new WJ_FieldUser();
23
+ Just setup correct tables names.
24
+ */
25
+ function __construct() {
26
+ $this->user_table = WJ_Settings::db_prefix('user');
27
+ $this->field_table = WJ_Settings::db_prefix('custom_field');
28
+ }
29
+
30
+ /*
31
+ After object creation, you can set the user id,
32
+ and the field id. The object will now contain the right
33
+ custom field, and the user id.
34
+ $field_user->set(array(
35
+ 'user_id' => 1,
36
+ 'field_id' => 2
37
+ ));
38
+ */
39
+ public function set($args) {
40
+ $this->user_id = $args['user_id'];
41
+ $this->field = WJ_Field::get($args['field_id']);
42
+ }
43
+
44
+ /*
45
+ Updates the field user value.
46
+ $field_user->update('Main Street');
47
+ */
48
+ public function update($value) {
49
+ global $wpdb;
50
+ $column_name = $this->field->user_column_name();
51
+ // Cast value to the correct column type.
52
+ switch ($this->field->type) {
53
+ case 'checkbox':
54
+ $validation = '%d';
55
+ $value = (int)$value;
56
+ break;
57
+ default:
58
+ // We default to a string.
59
+ $validation = '%s';
60
+ $value = (string)$value;
61
+ break;
62
+ }
63
+ $result = $wpdb->update(
64
+ $this->user_table,
65
+ array(
66
+ $column_name => $value
67
+ ), array("user_id" => $this->user_id),
68
+ array($validation),
69
+ array("%d")
70
+ );
71
+ if ($result != false) {
72
+ $this->value = $value;
73
+ }
74
+ return $result;
75
+ }
76
+
77
+ /*
78
+ Get the user field value.
79
+ $field_user->value();
80
+ # => 'Main Street'
81
+ */
82
+ public function value() {
83
+ $value = '';
84
+ if (isset($this->value)) {
85
+ $value = $this->value;
86
+ } else {
87
+ $column_name = $this->field->user_column_name();
88
+ global $wpdb;
89
+ $result = $wpdb->get_row($wpdb->prepare(
90
+ "SELECT $column_name FROM $this->user_table
91
+ WHERE user_id = %d",
92
+ array($this->user_id)
93
+ ), ARRAY_A);
94
+ $this->value = $result[$column_name];
95
+ $value = $result[$column_name];
96
+ }
97
+ return $value;
98
+ }
99
+
100
+ /*
101
+ Get the user unique column name.
102
+ $field_user->column_name();
103
+ # => 'cf_1'
104
+ */
105
+ public function column_name() {
106
+ return $this->field->user_column_name();
107
+ }
108
+
109
+ /*
110
+ Get all UserFields by User ID.
111
+ $WJ_FieldUser::get_all();
112
+ # => Array of WJ_FieldUser
113
+ */
114
+ public static function get_all($user_id) {
115
+ $fields = WJ_Field::get_all();
116
+ if(isset($fields) && !empty($fields)) {
117
+ $collection = array();
118
+ foreach ($fields as $field) {
119
+ $user_field = new self();
120
+ $user_field->user_id = $user_id;
121
+ $user_field->field = $field;
122
+ $collection[] = $user_field;
123
+ }
124
+ return $collection;
125
+ } else {
126
+ return null;
127
+ }
128
+ }
129
+
130
+ }
classes/WJ_Import.php ADDED
@@ -0,0 +1,828 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Class Import.
5
+ *
6
+ * Importing subscribers
7
+ */
8
+ class WJ_Import extends WYSIJA_object {
9
+
10
+ private $_header_row = array();
11
+ private $_csv_data = array();
12
+ private $_match = array();
13
+ private $_ignore_invalid_date = array(); // specific for date fields
14
+ private $_custom_fields = array();
15
+ private $_email_key = '';
16
+ private $_data_to_insert = array();
17
+ private $_number_list = 0;
18
+ private $_data_numbers = array();
19
+ private $_line_delimiter = "\n";
20
+ private $_duplicate_emails_count = array(); // detect the emails that are duplicates in the import file
21
+ private $_csv_array = array(); // containing the csv into an array
22
+ private $_ignored_row_count = 0; // the count of rows we ignore because there was no valid email present
23
+ private $_lines_count = 0; // count the number of valid lines inserted in the DB
24
+ private $_chunks_count = 0; // number of chunks we chopped from the csv
25
+ private $_first_row_is_data = false; // means that there is no header on that csv file
26
+ private $_emails_inserted_in_current_chunk = array(); // array of emails
27
+ private $_csv_file_string = '';
28
+ private $_data_result = array(); // used for importmatch refactor
29
+ private $_regex_new_field = "/^new_field\|([^\|]+)\|(.+)$/i";
30
+
31
+ function __construct() {
32
+ if (!empty($_POST['wysija']['match']))
33
+ $this->_match = $_POST['wysija']['match'];
34
+ if (!empty($_POST['wysija']['ignore_invalid_date']))
35
+ $this->_ignore_invalid_date = $_POST['wysija']['ignore_invalid_date'];
36
+ if (!empty($_REQUEST['wysija']['user_list']['list']))
37
+ $this->_number_list = count($_REQUEST['wysija']['user_list']['list']);
38
+ if (!empty($_POST['firstrowisdata']))
39
+ $this->_first_row_is_data = true;
40
+ $this->_data_numbers = array('invalid' => array(), 'inserted' => 0, 'valid_email_processed' => 0, 'list_added' => 0, 'list_user_ids' => 0, 'list_list_ids' => $this->_number_list, 'emails_queued' => 0);
41
+ }
42
+
43
+ /**
44
+ * loading file data passed in a global variable
45
+ * @return type
46
+ */
47
+ private function _get_temp_file_info() {
48
+ if (!empty($_REQUEST['wysija']['dataImport']))
49
+ return unserialize(base64_decode($_REQUEST['wysija']['dataImport']));
50
+ }
51
+
52
+ /**
53
+ * loading the file based on global parameters
54
+ * @return string
55
+ */
56
+ private function _loading_file_content() {
57
+ // try to access the temporary file created in the previous step
58
+ $this->_csv_data = $this->_get_temp_file_info();
59
+
60
+ $helper_file = WYSIJA::get('file', 'helper');
61
+ $result_file = $helper_file->get($this->_csv_data['csv'], 'import');
62
+
63
+ if (!$result_file) {
64
+ $upload_dir = wp_upload_dir();
65
+ $this->error(sprintf(__('Cannot access CSV file. Verify access rights to this directory "%1$s"', WYSIJA), $upload_dir['basedir']), true);
66
+ return false;
67
+ }
68
+
69
+ // get the temp csv file
70
+ $this->_csv_file_string = file_get_contents($result_file);
71
+ }
72
+
73
+ private function _set_custom_fields(){
74
+ $WJ_Field = new WJ_Field();
75
+ $_custom_fields = $WJ_Field->get_all();
76
+
77
+ if(!empty($_custom_fields)){
78
+ foreach($_custom_fields as $key => &$row){
79
+ $this->_custom_fields['cf_'.$row->id] = $row;
80
+ }
81
+ }
82
+ }
83
+
84
+ /**
85
+ * Validate if a column is already matched previously
86
+ * @param string $column_name
87
+ * @return boolean
88
+ */
89
+ private function _is_column_matched($column_name) {
90
+ if (empty($this->_data_to_insert)) {
91
+ return false;
92
+ }
93
+ $column_names = array_values($this->_data_to_insert);
94
+ return in_array(trim($column_name), $column_names);
95
+ }
96
+ /**
97
+ * match columns together with the csv data based on the data passed
98
+ */
99
+ private function _match_columns_to_insert() {
100
+
101
+ // we're going through all of the selected value in each dropdown when importing
102
+ foreach($this->_match as $csv_column_number => $column_in_user_table){
103
+ // Reduce matching twice the same column
104
+ if ($this->_is_column_matched($column_in_user_table))
105
+ continue;
106
+
107
+ // Ignore `nomatch` columns
108
+ if ($column_in_user_table == 'nomatch')
109
+ continue;
110
+
111
+ // Check if maybe it's a new field
112
+ preg_match($this->_regex_new_field, $column_in_user_table, $maybe_newfield);
113
+ if ( !empty($maybe_newfield) && in_array($maybe_newfield[1], array('date', 'input')) ){
114
+ // TODO need to change to WJ_Field I guess when Marco is done moving the files
115
+ // saving a new custom field
116
+ $custom_field = new WJ_Field();
117
+
118
+ $custom_field->set(array(
119
+ 'name' => $maybe_newfield[2],
120
+ 'type' => $maybe_newfield[1],
121
+ 'required' => false,
122
+ 'settings' => array(
123
+ 'label' => $maybe_newfield[2],
124
+ 'date_type' => 'year_month_day',
125
+ 'is_default_today' => 0,
126
+ 'date_order' => 'mm/dd/yyyy'
127
+ )
128
+ ));
129
+
130
+ $custom_field->save();
131
+
132
+ // this is the column name in the database so this is where we need to import that field
133
+ $column_in_user_table = $custom_field->user_column_name();
134
+ $this->_match[$csv_column_number] = $column_in_user_table;
135
+ }
136
+
137
+ // keep the match of CSV column number to column key in our database
138
+ // not sure why do we trim the column key ...
139
+ $this->_data_to_insert[$csv_column_number] = trim($column_in_user_table);
140
+
141
+ // this column is the email column, let's keep track of it for later validation etc..
142
+ if($column_in_user_table == 'email'){
143
+ $this->_email_key = $csv_column_number;
144
+ }
145
+ }
146
+
147
+ $this->_set_custom_fields();
148
+
149
+ // if the status column is not matched, we make sure that we have an entry for the status column so that we default it to some value on import
150
+ if(!in_array('status',$this->_data_to_insert)){
151
+ $this->_data_to_insert['status'] = 'status';
152
+ }
153
+ }
154
+
155
+ /**
156
+ * build the header of the import query
157
+ * @return type
158
+ */
159
+ private function _get_import_query_header() {
160
+ return 'INSERT IGNORE INTO [wysija]user (`' . implode('` ,`', $this->_data_to_insert) . '`,`created_at`) VALUES ';
161
+ }
162
+
163
+ /**
164
+ *
165
+ * @param type $array_csv
166
+ */
167
+ private function _check_duplicate_emails($array_csv) {
168
+ // look for duplicate emails
169
+ foreach ($array_csv as $keyline => $csv_line) {
170
+ if (isset($csv_line[$this->_email_key])) {
171
+ if (isset($this->_duplicate_emails_count[$csv_line[$this->_email_key]])) {
172
+ $this->_duplicate_emails_count[$csv_line[$this->_email_key]]++;
173
+ //$arra[$keyline]
174
+ } else {
175
+ $this->_duplicate_emails_count[$csv_line[$this->_email_key]] = 1;
176
+ }
177
+ } else {
178
+ //if the record doesn't have the attribute email then we just ignore it
179
+ $this->_ignored_row_count++;
180
+ unset($array_csv[$keyline]);
181
+ }
182
+ }
183
+ }
184
+
185
+ /**
186
+ * save new column/field match to improve usability the next time our admin
187
+ * import a field with similar headers/columns names
188
+ * @return boolean
189
+ */
190
+ private function _smart_column_match_recording() {
191
+ if ($this->_first_row_is_data === false) {
192
+ //save the importing fields to be able to match them the next time
193
+ $import_fields = get_option('wysija_import_fields');
194
+
195
+ foreach($this->_match as $csv_column_number => $column_in_user_table){
196
+ if($column_in_user_table != 'nomatch') {
197
+
198
+ // make a column name key
199
+ $column_name_key = str_replace(array(' ','-','_'),'',strtolower($this->_header_row[$csv_column_number]));
200
+ // fill in the array of "csv column name" / "user table column" matches
201
+ $import_fields[$column_name_key] = $column_in_user_table;
202
+
203
+ }
204
+ }
205
+
206
+ WYSIJA::update_option('wysija_import_fields' , $import_fields);
207
+ return true;
208
+ }
209
+ return false;
210
+ }
211
+
212
+ /**
213
+ * import a csv type into wysija's subscribers' table
214
+ * @return type
215
+ */
216
+ public function import_subscribers() {
217
+
218
+ // import the contacts
219
+ // 1-check that a list is selected and that there is a csv file pasted
220
+ // 2-save the list if necessary
221
+ // 3-save the contacts and record them for each list selected
222
+ $this->_loading_file_content();
223
+ // convert the csv file to an array of lines
224
+ $this->_csv_array = $this->_csv_to_array( $this->_csv_file_string , 0 , $this->_csv_data['fsep'] , $this->_csv_data['fenc']);
225
+
226
+ // check what columns of the csv have been matched together with our subscribers' table
227
+ $this->_match_columns_to_insert();
228
+
229
+ $this->_header_row = array_map('trim', $this->_csv_array[0]);
230
+
231
+ // we process the sql insertion 200 by 200 so that we are safe with the server's memory and cpu
232
+ $csv_chunks = array_chunk($this->_csv_array, 200);
233
+ $this->_csv_array = null;
234
+ $this->_chunks_count = 0;
235
+ $this->_lines_count = 0;
236
+
237
+ // setting up a timeout value on the sql side to avoid timeout when importing a lot of data apparently.
238
+ global $wpdb;
239
+ $wpdb->query('set session wait_timeout=600');
240
+
241
+ // loop and insert the data chunks by chunks
242
+ foreach ($csv_chunks as $key_chunk => $csv_chunk) {
243
+
244
+ $this->_check_duplicate_emails($csv_chunk);
245
+
246
+ $result = $this->_import_rows($csv_chunk);
247
+
248
+ if ($result !== false)
249
+ $this->_chunks_count++;
250
+ else {
251
+ // there was an error we try 3 more times the same chunk and se how it goes
252
+ $try = 0;
253
+ while ($result === false && $try < 3) {
254
+ $result = $this->_import_rows($csv_chunk);
255
+ if ($result !== false) {
256
+ $this->_chunks_count++;
257
+ break;
258
+ }
259
+ $try++;
260
+ }
261
+
262
+ if ($result === false) {
263
+ $this->error(__('There seems to be an error with the list you\'re trying to import.', WYSIJA), true);
264
+ return false;
265
+ }
266
+ }
267
+ // increment the lines count
268
+ $this->_lines_count += $result;
269
+ // free up some memory
270
+ unset($csv_chunks[$key_chunk]);
271
+ }
272
+
273
+ // useful the next time we import a file with the same format
274
+ $this->_smart_column_match_recording();
275
+
276
+ // refresh the total count of users in wysija
277
+ $helper_user = WYSIJA::get('user','helper');
278
+ $helper_user->refreshUsers();
279
+
280
+ // keep only the real duplicate emails unset the unique ones
281
+ // TODO check that this email duplicate function could be a memory sink hole
282
+ // especially that right now we don't use its value
283
+ foreach ($this->_duplicate_emails_count as $email_address => $times_email_in_csv) {
284
+ if ($times_email_in_csv == 1)
285
+ unset($this->_duplicate_emails_count[$email_address]);
286
+ }
287
+
288
+ // how come we need to do that sometimes? how a lines count could become negative?
289
+ if ($this->_lines_count < 0)
290
+ $this->_lines_count = 0;
291
+
292
+ // all of these numbers were useful at some point when we were showing more information after an import
293
+ $this->_data_numbers['ignored'] = ($this->_data_numbers['valid_email_processed'] - $this->_data_numbers['inserted']);
294
+ $this->_data_numbers['ignored_list'] = ( ($this->_data_numbers['list_user_ids'] * $this->_data_numbers['list_list_ids']) - $this->_data_numbers['list_added'] );
295
+
296
+
297
+ return $this->_data_numbers;
298
+ }
299
+
300
+ public function get_duplicate_emails_count() {
301
+ return $this->_duplicate_emails_count;
302
+ }
303
+
304
+ /**
305
+ * convert a csv string to an array
306
+ * @param type $csv_file_content
307
+ * @param type $rows_to_read
308
+ * @param type $delimiter
309
+ * @param type $enclosure
310
+ * @return array
311
+ */
312
+ private function _csv_to_array($csv_file_content, $rows_to_read = 0, $delimiter = ',', $enclosure = '') {
313
+ $data = array();
314
+
315
+ // the new way for splitting a string into an array of lines
316
+ $csv_data_array = explode( $this->_line_delimiter, $csv_file_content );
317
+
318
+ $i=1;
319
+ foreach($csv_data_array as $csv_line){
320
+ if($rows_to_read!=0 && $i> $rows_to_read) return $data;
321
+
322
+ // str_getcsv only exists in php5 and is a faster and cleaner function than our csv_explode
323
+ if (!function_exists('str_getcsv')) {
324
+ $data[] = $this->_lines_explode($csv_line, $delimiter, $enclosure);
325
+ } else {
326
+ $data[] = str_getcsv($csv_line, $delimiter, $enclosure);
327
+ }
328
+
329
+ $i++;
330
+ }
331
+
332
+ return $data;
333
+ }
334
+
335
+ /**
336
+ * explode lines to columns
337
+ * @param type $csv_line
338
+ * @param type $delimiter
339
+ * @param type $enclose
340
+ * @param type $preserve
341
+ * @return type
342
+ */
343
+ private function _lines_explode($csv_line, $delimiter, $enclose, $preserve = false) {
344
+ $resArr = array();
345
+ $n = 0;
346
+ if (empty($enclose)) {
347
+ $resArr = explode($delimiter, $csv_line);
348
+ } else {
349
+ $expEncArr = explode($enclose, $csv_line);
350
+ foreach ($expEncArr as $EncItem) {
351
+ if ($n++ % 2) {
352
+ array_push($resArr, array_pop($resArr) . ($preserve ? $enclose : '') . $EncItem . ( $preserve ? $enclose : ''));
353
+ } else {
354
+ $expDelArr = explode($delimiter, $EncItem);
355
+ array_push($resArr, array_pop($resArr) . array_shift($expDelArr));
356
+ $resArr = array_merge($resArr, $expDelArr);
357
+ }
358
+ }
359
+ }
360
+
361
+ return $resArr;
362
+ }
363
+
364
+ /**
365
+ * function processing a chunk of a csv array to import it in the DB
366
+ * @global object $wpdb
367
+ * @param array $csv_chunk
368
+ * @return boolean|string
369
+ */
370
+ private function _import_rows($csv_chunk) {
371
+
372
+ global $wpdb;
373
+
374
+ $this->_emails_inserted_in_current_chunk = array();
375
+ $time = time();
376
+ $lines = array();
377
+ $lines_count = count($csv_chunk);
378
+ $columns_count = count($this->_data_to_insert);
379
+
380
+ $query = $this->_get_import_query_header();
381
+
382
+ // make sure that each line has the right numbers of columns if it doesn't then we can skip it
383
+ foreach ($csv_chunk as $k => $line) {
384
+ if (!(count($line) >= (count($this->_data_to_insert) - 1))) {
385
+ unset($csv_chunk[$k]);
386
+ $lines_count--;
387
+ }
388
+ }
389
+
390
+ $valid_email_processed = 0;
391
+ $j = 1;
392
+
393
+ foreach ($csv_chunk as $key_line => $line) {
394
+
395
+ // if first row is not data but header then we just skip it only on the first chunk
396
+ if ($this->_first_row_is_data === false && $j == 1 && $this->_chunks_count == 0) {
397
+ $j++;
398
+ continue;
399
+ }
400
+
401
+ $i = 1;
402
+ $values = array();
403
+
404
+ // TODO maybe we should check the value of the status column so that if we export a wysija's subscribers' list
405
+ // and import it again in another site then we keep the status
406
+ if (isset($this->_data_to_insert['status']))
407
+ $line['status'] = 1;
408
+
409
+ foreach ($line as $key_column => &$value_column) {
410
+
411
+ // make sure this column is a column we want to insert in our DB
412
+ if (isset($this->_data_to_insert[$key_column])) {
413
+ $column_name = $this->_data_to_insert[$key_column];
414
+ $original_value_column = $value_column;
415
+ $value_column = $this->_validate_value($column_name, $value_column);
416
+
417
+ // if it is a a date column, we convert it as a unixtimestamp
418
+ if(isset($this->_custom_fields[$column_name]) && $this->_custom_fields[$column_name]->type == 'date'){
419
+ $value_column = strtotime($value_column);
420
+ }
421
+
422
+ // this kind of result invalidates the whole row
423
+ if ($value_column === false) {
424
+ // record the invalid row and continue with the loop
425
+ $this->_data_numbers['invalid'][] = $original_value_column;
426
+ unset($csv_chunk[$key_line]);
427
+ $lines_count--;
428
+ continue 2;
429
+ } else {
430
+ // only if this is the email row we record an entry in the recorded emails and the email processed count
431
+ if ($this->_email_key === $key_column) {
432
+ $this->_emails_inserted_in_current_chunk[] = $value_column;
433
+ $valid_email_processed++;
434
+ }
435
+ }
436
+
437
+ // prepare the query
438
+ $values[] = "'" . esc_sql($value_column, $wpdb->dbh) . "'";
439
+ }
440
+ }
441
+
442
+ $values[] = $time;
443
+ $lines[] .= "(" . implode(', ', $values) . ")";
444
+ }
445
+
446
+ $query .= implode(', ', $lines);
447
+
448
+ // replace query to import the subscribers
449
+ $model_wysija = new WYSIJA_model();
450
+ $import_query = $model_wysija->query($query);
451
+
452
+ $lines_count = $wpdb->rows_affected;
453
+ $this->_data_numbers['inserted'] += $wpdb->rows_affected;
454
+ $this->_data_numbers['valid_email_processed'] += $valid_email_processed;
455
+
456
+ if ($import_query === false) {
457
+ $this->error(__('Error when inserting emails.', WYSIJA), true);
458
+ return false;
459
+ }
460
+ $time_now = time();
461
+ $result_query_import_list = $this->_import_new_users_into_lists($time_now);
462
+
463
+ $this->_trigger_active_autoresponders($time_now);
464
+
465
+ if ($result_query_import_list === false) {
466
+ $this->error(__('Error when inserting list.', WYSIJA), true);
467
+ return false;
468
+ }
469
+
470
+ if ($import_query == 0)
471
+ return '0';
472
+
473
+ return $lines_count;
474
+ }
475
+
476
+ /**
477
+ * used to validate or cast values before importing
478
+ * TODO should we add a type for import of custom fields ?
479
+ * Comment : Marco, feel free to modify entirely
480
+ * @param type $column_name
481
+ * @param type $value
482
+ */
483
+ function _validate_value($column_name, $value) {
484
+ $value = trim($value);
485
+
486
+ switch ($column_name) {
487
+ case 'email':
488
+ return is_email($value);
489
+ break;
490
+ case 'status':
491
+
492
+ if (in_array(strtolower($value), array('subscribed', 'confirmed', 1, '1', 'true'))) {
493
+ return 1;
494
+ } elseif (in_array(strtolower($value), array('unsubscribed', -1, '-1', 'false'))) {
495
+ return -1;
496
+ } elseif (in_array(strtolower($value), array('unconfirmed', 0, '0'))) {
497
+ return 0;
498
+ }
499
+ else
500
+ return 1;
501
+ break;
502
+ default :
503
+ return $value;
504
+ }
505
+ }
506
+
507
+ /**
508
+ * take care of active autoresponders retro-activity
509
+ * @param type $time_now
510
+ * @return boolean
511
+ */
512
+ private function _trigger_active_autoresponders($time_now) {
513
+ $helper_email = WYSIJA::get('email', 'helper');
514
+ $model_wysija = new WYSIJA_model();
515
+
516
+ // list the active auto responders emails
517
+ $active_autoresponders_per_list = $helper_email->get_active_follow_ups(array('email_id', 'params'), true);
518
+
519
+ if (!empty($active_autoresponders_per_list)) {
520
+ foreach ($_REQUEST['wysija']['user_list']['list'] as $list_id) {
521
+ // checking if this list has a list of follow ups
522
+ if (isset($active_autoresponders_per_list[$list_id])) {
523
+ // for each follow up of that list we queu an email
524
+ foreach ($active_autoresponders_per_list[$list_id] as $key_queue => $follow_up) {
525
+ // insert query per active followup
526
+ $query_queue = 'INSERT IGNORE INTO [wysija]queue (`email_id` ,`user_id`,`send_at`) ';
527
+ $query_queue .= ' SELECT ' . $follow_up['email_id'] . ' , B.user_id , ' . ($time_now + $follow_up['delay']);
528
+ $query_queue .= ' FROM [wysija]user_list as B';
529
+ $query_queue .= ' WHERE B.list_id=' . (int) $list_id . ' AND sub_date=' . $time_now;
530
+
531
+ $model_wysija->query($query_queue);
532
+
533
+ $this->_data_numbers['emails_queued'] += $wpdb->rows_affected;
534
+ }
535
+ }
536
+ }
537
+ return true;
538
+ }
539
+ return false;
540
+ }
541
+
542
+ /**
543
+ *
544
+ * @global type $wpdb
545
+ * @param type $time_now
546
+ * @return type
547
+ */
548
+ private function _import_new_users_into_lists($time_now) {
549
+ global $wpdb;
550
+ $wpdb->rows_affected = 0;
551
+ $model_wysija = new WYSIJA_model();
552
+
553
+ $user_ids = $this->_get_imported_user_ids();
554
+
555
+ // insert query per list
556
+ $query = 'INSERT IGNORE INTO [wysija]user_list (`list_id` ,`user_id`,`sub_date`) VALUES ';
557
+
558
+ foreach ($_REQUEST['wysija']['user_list']['list'] as $keyl => $list_id) {
559
+ if (empty($list_id))
560
+ continue;
561
+ // for each list pre selected go through that process
562
+ foreach ($user_ids as $key => $user_data) {
563
+
564
+ // inserting each user id to this list
565
+ $query.='(' . $list_id . ' , ' . $user_data['user_id'] . ' , ' . $time_now . ')';
566
+
567
+ // if this is not the last row we put a comma for the next row
568
+ if (count($user_ids) > ($key + 1)) {
569
+ $query.=' , ';
570
+ }
571
+ }
572
+
573
+ // if this is not the last row we put a comma for the next row
574
+ if (count($_REQUEST['wysija']['user_list']['list']) > ($keyl + 1)) {
575
+ $query.=',';
576
+ }
577
+ }
578
+
579
+ $result_query = $model_wysija->query($query);
580
+
581
+ $this->_data_numbers['list_added']+=$wpdb->rows_affected;
582
+ $this->_data_numbers['list_user_ids']+=count($user_ids);
583
+ return $result_query;
584
+ }
585
+
586
+ /**
587
+ * get a list of user_ids freshly imported
588
+ * @return type
589
+ */
590
+ private function _get_imported_user_ids() {
591
+ $model_user = WYSIJA::get('user', 'model');
592
+ // select query to get all of the ids of the emails that have just been inserted
593
+ return $model_user->get(array('user_id'), array('email' => $this->_emails_inserted_in_current_chunk));
594
+ }
595
+
596
+
597
+ /**
598
+ * this save a default list of column matching to ease the import process, this is done only the first time when the
599
+ * field is not populated yet
600
+ */
601
+ private function _save_default_import_field_match(){
602
+ $import_fields = get_option('wysija_import_fields');
603
+ if (!$import_fields) {
604
+ $import_fields = array(
605
+ 'fname' => 'firstname',
606
+ 'firstname' => 'firstname',
607
+ 'prenom' => 'firstname',
608
+ 'nom' => 'lastname',
609
+ 'name' => 'lastname',
610
+ 'lastname' => 'lastname',
611
+ 'lname' => 'lastname',
612
+ 'ipaddress' => 'ip',
613
+ 'ip' => 'ip',
614
+ 'addresseip' => 'ip',
615
+ );
616
+ WYSIJA::update_option('wysija_import_fields', $import_fields);
617
+ }
618
+ }
619
+
620
+
621
+ /**
622
+ * put the whole file or posted string into one string and clean up the string that may cause trouble during import
623
+ * @return boolean
624
+ */
625
+ private function _get_csv_file_cleaned_up(){
626
+ // is it a text import or a file import?
627
+ if($_POST['wysija']['import']['type'] == 'copy'){
628
+ if(!isset($_POST['wysija']['user_list']['csv'])){
629
+ // memory limit has been reached
630
+ $this->error(__('The list you\'ve pasted is too big for the browser. <strong>Upload the file</strong> instead.', WYSIJA), true);
631
+ return false;
632
+ }
633
+ $this->_csv_file_string = trim(stripslashes($_POST['wysija']['user_list']['csv']));
634
+
635
+ }else{
636
+ // move_uploaded_file($_importfile, $destination);
637
+ $this->_csv_file_string = trim(file_get_contents($_FILES['importfile']['tmp_name']));
638
+ }
639
+
640
+ $this->_csv_file_string = str_replace(array("\r", "\n\n", "\n\t\t\n\t\n\t\n\t\n", "\n\t\t\n\t\n\t\n", "\xEF\xBB\xBF", "\n\t\n", "\n(+1)"), array("\n", "\n", "\n ;", "\n", '', ';', ''), $this->_csv_file_string);
641
+
642
+
643
+ // this might be gmail recipients(copy-paste from a gmail account) rare paste ...
644
+ if(!preg_match_all('/<([a-z0-9_\'&\.\-\+])+\@(([a-z0-9\-])+\.)+([a-z0-9]{2,10})+>/i' , $this->_csv_file_string , $matches)){
645
+ //return false;
646
+ }else{
647
+
648
+ if (substr($this->_csv_file_string, -1) != ",")
649
+ $this->_csv_file_string = trim($this->_csv_file_string) . ',';
650
+
651
+
652
+ if (count($matches[0]) == count($matches)) {
653
+ //this is gmail simple paste
654
+ $this->_csv_file_string = str_replace(array('>,', '<'), array("\n", ','), $this->_csv_file_string);
655
+ }
656
+ $this->_csv_file_string = trim($this->_csv_file_string);
657
+ }
658
+ }
659
+
660
+
661
+ /**
662
+ * try to figure out the format of that CSV file, which separators and enclosure strings does it use
663
+ * @return boolean
664
+ */
665
+ private function _run_test_on_csv_file(){
666
+ // try different set of enclosure and separator for the csv which can have different look depending on the data carried
667
+ $field_separators_to_test = array( ',', ';', "\t" );
668
+ $field_enclosers_to_test = array( '"', '' );
669
+ $this->_csv_data['fsep'] = false;
670
+ $this->_csv_data['fenc'] = '';
671
+ $helper_user = WYSIJA::get('user','helper');
672
+
673
+ foreach($field_enclosers_to_test as $enclosure){
674
+ foreach($field_separators_to_test as $fsep){
675
+
676
+ // testing different combinations of separator and enclosers
677
+ $this->_csv_array = $this->_csv_to_array( $this->_csv_file_string, 10, $fsep, $enclosure );
678
+
679
+ // make sure that our CSV has more than one row and that it has the same number of values in the first row and the second row
680
+ if ( ( count( $this->_csv_array ) > 1 && count( $this->_csv_array[0] ) == count( $this->_csv_array[1] ) ) ) {
681
+ if ( count( $this->_csv_array[0] ) > 1 || $helper_user->validEmail( trim( $this->_csv_array[0][0] ) ) || $helper_user->validEmail( trim( $this->_csv_array[1][0] ) ) ) {
682
+ $this->_csv_data['fsep'] = $fsep;
683
+ $this->_csv_data['fenc'] = $enclosure;
684
+ break(2);
685
+ }
686
+ }
687
+ }
688
+ }
689
+
690
+ // if we didn't manage to find a separator in that file then it is not a csv file and we come out
691
+ if(empty($this->_csv_data['fsep'])){
692
+ $this->notice( sprintf(
693
+ "%s <a href='http://support.mailpoet.com/knowledgebase/importing-subscribers-with-a-csv-file/' target='_blank'>%s</a>",
694
+ __("The data you are trying to import doesn't appear to be in the CSV format (Comma Separated Values).", WYSIJA),
695
+ __("Read More", WYSIJA)
696
+ ) );
697
+ $this->notice(__('The first line of a CSV file should be the column headers : "email","lastname","firstname".',WYSIJA));
698
+ $this->notice(__('The second line of a CSV file should be a set of values : "joeeg@example.com","Average","Joe".',WYSIJA));
699
+
700
+ $this->notice(__('The two first lines of the file you\'ve uploaded are as follow:', WYSIJA));
701
+
702
+ $arraylines = explode("\n", $this->_csv_file_string);
703
+
704
+ if (empty($arraylines[0]))
705
+ $text = __('Line is empty', WYSIJA);
706
+ else
707
+ $text = $arraylines[0];
708
+ $this->notice('<strong>' . $text . '</strong>');
709
+
710
+ if (empty($arraylines[1]))
711
+ $text = __('Line is empty', WYSIJA);
712
+ else
713
+ $text = $arraylines[1];
714
+ $this->notice('<strong>' . $text . '</strong>');
715
+
716
+ return false;
717
+ }
718
+
719
+ // test the size of the file
720
+ $temp_csv_array = $this->_csv_to_array($this->_csv_file_string, 0, $this->_csv_data['fsep'], $this->_csv_data['fenc']);
721
+
722
+ $this->_data_result['totalrows'] = count($temp_csv_array);
723
+ end($temp_csv_array);
724
+ $this->_data_result['lastrow'] = current($temp_csv_array);
725
+ }
726
+
727
+
728
+ /**
729
+ * save the CSV string into a file so that we can use it later in the next step of the process
730
+ * @return boolean|string
731
+ */
732
+ private function _save_csv_file(){
733
+ // try to make a wysija dir to save the import file
734
+ $helper_file = WYSIJA::get('file', 'helper');
735
+ $result_dir = $helper_file->makeDir('import');
736
+ if (!$result_dir) {
737
+ return false;
738
+ }
739
+
740
+ $file_name = 'import-' . time() . '.csv';
741
+ $handle = fopen($result_dir . $file_name, 'w');
742
+ fwrite($handle, $this->_csv_file_string);
743
+ fclose($handle);
744
+
745
+ return $file_name;
746
+ }
747
+
748
+
749
+ /**
750
+ * detect which column is an email one
751
+ */
752
+ private function _test_csv_emails(){
753
+ $found_email = 0;
754
+ $this->_email_key = array();
755
+ $helper_user = WYSIJA::get('user', 'helper');
756
+ foreach ($this->_csv_array as $csv_row) {
757
+ foreach ($csv_row as $key_column => $value_column) {
758
+ if ($helper_user->validEmail(trim($value_column))) {
759
+ $found_email++;
760
+
761
+ $this->_email_key[$key_column] = $this->_csv_array[0][$key_column];
762
+ }
763
+ }
764
+ }
765
+
766
+ $this->_data_result['errormatch'] = false;
767
+ $check_again = __('Check again what you\'re trying to import.',WYSIJA);
768
+ if (($found_email < 1)) {
769
+ $this->error(__('We couldn\'t find any valid addresses in the first 10 rows.', WYSIJA).' '.$check_again, true);
770
+ }
771
+
772
+ if ((count($this->_csv_array) < 2)) {
773
+ $this->error(__('You\'re import file is not valid.', WYSIJA).' '.$check_again, true);
774
+ }
775
+ }
776
+
777
+
778
+ /**
779
+ * function used to scan the CSV before trying to match each column with our own fields
780
+ * @return boolean
781
+ */
782
+ public function scan_csv_file(){
783
+ $this->_data_result = array();
784
+
785
+ // make sure the import match will be as easy as possible with default matches
786
+ if( $this->_save_default_import_field_match() === false) return false;
787
+ // get the csv into a string and check for messy email address etc
788
+ if( $this->_get_csv_file_cleaned_up() === false) return false;
789
+ // find out the format of that CSV file comma semi colon etc ..
790
+ if( $this->_run_test_on_csv_file() === false) return false;
791
+
792
+ // save the string into a file for later use
793
+ $file_name = $this->_save_csv_file();
794
+ if ($file_name === false)
795
+ return false;
796
+
797
+ // look for the email column
798
+ $this->_test_csv_emails();
799
+
800
+ // stock the data we will use in the next page
801
+ $this->_data_result['csv'] = $this->_csv_array;
802
+
803
+ $data_import = array(
804
+ 'csv' => $file_name,
805
+ 'fsep' => $this->_csv_data['fsep'],
806
+ 'fenc' => $this->_csv_data['fenc']);
807
+ $this->_data_result['dataImport'] = base64_encode(serialize($data_import));
808
+
809
+ $this->_data_result['keyemail'] = $this->_email_key;
810
+
811
+ //test if the first row is data or not
812
+ //test the email column
813
+ foreach ($this->_data_result['keyemail'] as $k)
814
+ $this->_email_key = $k;
815
+
816
+
817
+ // test whether the first row is a data row or is a descriptive header row
818
+ $helper_user = WYSIJA::get('user','helper');
819
+ if($helper_user->validEmail( $this->_email_key )){
820
+ $this->_data_result['firstrowisdata']=true;
821
+ }else{
822
+ $this->_data_result['totalrows']--;
823
+ }
824
+
825
+ return $this->_data_result;
826
+ }
827
+
828
+ }
classes/WJ_Migrator.php ADDED
@@ -0,0 +1,53 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ // We need the dbDelta migrator from Wordpress.
4
+ require_once( ABSPATH . 'wp-admin/includes/upgrade.php' );
5
+
6
+ /**
7
+ Migrator.
8
+ A class that uses Wordpress dbDelta function
9
+ to handle incremental migrations.
10
+ Beware that deltas have specific syntax, not
11
+ identical to SQL syntax.
12
+ $migrator = new WJ_Migrator('2.1.5');
13
+ $migrator->setup();
14
+ */
15
+ class WJ_Migrator {
16
+
17
+ private $db_prefix;
18
+ private $new_version;
19
+
20
+ function __construct($new_version) {
21
+ // Global Wordpress db object.
22
+ $this->db_prefix = WJ_Settings::db_prefix();
23
+ $this->new_version = $new_version;
24
+ }
25
+
26
+ /*
27
+ Add here your specific version number case.
28
+ It will be executed on version change.
29
+ */
30
+ public function setup() {
31
+
32
+ $sql = '';
33
+
34
+ switch ($this->new_version) {
35
+ case '2.5.2.1':
36
+ $table_name = $this->db_prefix . 'custom_field';
37
+ $sql = "CREATE TABLE $table_name (
38
+ id mediumint(9) NOT NULL AUTO_INCREMENT,
39
+ name tinytext NOT NULL,
40
+ type tinytext NOT NULL,
41
+ required tinyint(1) DEFAULT '0' NOT NULL,
42
+ default_value text,
43
+ field_values text,
44
+ PRIMARY KEY (id)
45
+ );";
46
+ break;
47
+ }
48
+
49
+ return dbDelta($sql);
50
+
51
+ }
52
+
53
+ }
classes/WJ_Option.php ADDED
@@ -0,0 +1,29 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Option.
5
+ * It creates and modifies Wordpress options.
6
+ * $option = new WJ_Option('my_option');
7
+ * $option->set('my_value');
8
+ * # => true
9
+ * $option->get();
10
+ * # => 'my_value'
11
+ */
12
+ class WJ_Option {
13
+
14
+ private $prefix = 'wysija_premium_';
15
+ private $option_name;
16
+
17
+ function __construct($name) {
18
+ $this->option_name = $this->prefix . $name;
19
+ }
20
+
21
+ public function get() {
22
+ return get_option($this->option_name);
23
+ }
24
+
25
+ public function set($option_value) {
26
+ return update_option($this->option_name, $option_value);
27
+ }
28
+
29
+ }
classes/WJ_Settings.php ADDED
@@ -0,0 +1,30 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Settings.
5
+ * Global Wysija Premium settings singleton.
6
+ */
7
+ class WJ_Settings {
8
+
9
+ static private $db_prefix;
10
+ static private $name = 'wysija';
11
+
12
+ private function __construct() {
13
+ }
14
+
15
+ // db_prefix();
16
+ // # => wp_wysija_
17
+ // db_prefix('custom_field');
18
+ // # => wp_wysija_custom_field
19
+ static function db_prefix($table_name = false) {
20
+ global $wpdb;
21
+ self::$db_prefix = $wpdb->prefix . self::$name . '_';
22
+ if ($table_name) {
23
+ $prefixed = self::$db_prefix . $table_name;
24
+ } else {
25
+ $prefixed = self::$db_prefix;
26
+ }
27
+ return $prefixed;
28
+ }
29
+
30
+ }
classes/WJ_Stats.php ADDED
@@ -0,0 +1,474 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Class Stats.
5
+ *
6
+ * Handling subscribers click and open action of an email
7
+ */
8
+ class WJ_Stats extends WYSIJA_object {
9
+
10
+ public $email_id = null;
11
+ public $user_id = null;
12
+ public $clicked_url = '';
13
+ public $is_preview = false;
14
+
15
+ /**
16
+ * Email object
17
+ * @var Wysija_email
18
+ */
19
+ private $_email;
20
+
21
+ /**
22
+ * User object
23
+ * @var Wysija_user
24
+ */
25
+ private $_user;
26
+
27
+ private $_url_id = null;
28
+ private $_unique_click = false;
29
+ public $decoded_url = '';
30
+
31
+
32
+ /**
33
+ * load some global variables into the class
34
+ */
35
+ function __construct() {
36
+ if (!empty($_REQUEST['email_id']))
37
+ $this->email_id = (int) $_REQUEST['email_id'];
38
+ if (!empty($_REQUEST['user_id']))
39
+ $this->user_id = (int) $_REQUEST['user_id'];
40
+ if (!empty($_REQUEST['demo']))
41
+ $this->is_preview = true;
42
+ $this->_get_clicked_url();
43
+
44
+ // $this->_user can be empty, it's helpful in case an email was sent as a preview to a non-existing user, then a link is opened from there
45
+ $this->_user = WYSIJA::get('user', 'model')->getOne(false,(int)$this->user_id);
46
+
47
+ $this->_email = WYSIJA::get('email', 'model')->getOne(false,(int)$this->email_id);
48
+ if (empty($this->_email))
49
+ exit;
50
+
51
+ // consider to get / insert URL record here
52
+ // $this->_url = $this->_get_url($this->clicked_url);
53
+ }
54
+
55
+ /**
56
+ * Get user object
57
+ * @param type $user_id
58
+ */
59
+ public function get_user() {
60
+ return $this->_user;
61
+ }
62
+
63
+ /**
64
+ * Get email object
65
+ * @param type $user_id
66
+ */
67
+ public function get_email() {
68
+ return $this->_email;
69
+ }
70
+
71
+ /**
72
+ * count the action as an open and display the empty picture
73
+ */
74
+ public function subscriber_opened() {
75
+ if (!$this->_is_open_action())
76
+ exit;
77
+
78
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
79
+ $model_email_user_stat->reset();
80
+ $model_email_user_stat->update(
81
+ array('status' => 1, 'opened_at' => time()), array('email_id' => $this->email_id, 'user_id' => $this->user_id, 'status' => 0));
82
+
83
+ $this->_update_user(array('last_opened' => time()));
84
+
85
+ header('Cache-Control: no-store, no-cache, must-revalidate');
86
+ header('Cache-Control: post-check=0, pre-check=0', false);
87
+ header('Pragma: no-cache');
88
+
89
+ if (empty($picture))
90
+ $picture = WYSIJA_DIR_IMG . 'statpicture.png';
91
+ $handle = fopen($picture, 'r');
92
+
93
+ if (!$handle)
94
+ exit;
95
+ header('Content-type: image/png');
96
+ $contents = fread($handle, filesize($picture));
97
+ fclose($handle);
98
+ echo $contents;
99
+ exit;
100
+ }
101
+
102
+ public function subscriber_clicked() {
103
+ if (!$this->_is_click_action())
104
+ exit;
105
+
106
+ if ($this->email_id && $this->is_preview === false) { //if not email_id that means it is an email preview
107
+ $this->_record_url();
108
+
109
+ $this->_record_user_url();
110
+
111
+ $this->_record_url_mail();
112
+
113
+ $redirect = $this->_record_email_user_stat();
114
+ if (!empty($redirect))
115
+ $this->decoded_url = $redirect;
116
+ $this->_update_user(array('last_clicked' => time()));
117
+ }else {
118
+ $this->_get_url_preview();
119
+ }
120
+ //sometimes this will be a life saver :)
121
+ $this->decoded_url = str_replace('&amp;', '&', $this->decoded_url);
122
+ if ($this->is_browser_link($this->decoded_url)) {
123
+ $this->decoded_url = $this->attach_user($this->decoded_url);
124
+ }
125
+ return $this->decoded_url;
126
+ }
127
+
128
+ /**
129
+ * Attach user_id as a param of an url
130
+ * @param string $url
131
+ * @param int $user_id
132
+ * @return string
133
+ */
134
+ protected function attach_user($url) {
135
+ if (!empty($this->_user) && !empty($this->_user['user_id'])) {
136
+ $url_components = parse_url($url);
137
+ $arr_params = array();
138
+ if (!empty($url_components['query']))
139
+ parse_str($url_components['query'],$arr_params);
140
+ if (empty($arr_params['user_id'])) {
141
+ $url .= ((strpos($url, '?') !== false) ? '&' : '?');
142
+ $url .= 'user_id='.$this->_user['user_id'];
143
+ }
144
+ }
145
+ return $url;
146
+ }
147
+
148
+ /**
149
+ * record entry into the table email_user_url
150
+ */
151
+ private function _record_user_url() {
152
+ //look for email_user_url entry and insert if not exists
153
+ $model_email_user_url = WYSIJA::get('email_user_url', 'model');
154
+ $data_email_user_url = array('email_id' => $this->email_id, 'user_id' => $this->user_id, 'url_id' => $this->_url_id);
155
+ $email_user_url_array = $model_email_user_url->getOne(false, $data_email_user_url);
156
+
157
+ if (!$email_user_url_array && $this->email_id > 0 && $this->user_id > 0 && $this->_url_id > 0) {
158
+ //we need to insert in email_user_url
159
+ $model_email_user_url->reset();
160
+ $query_EmailUserUrl = 'INSERT IGNORE INTO [wysija]email_user_url (`email_id` ,`user_id`,`url_id`) ';
161
+ $query_EmailUserUrl .= 'VALUES (' . $this->email_id . ', ' . $this->user_id . ', ' . $this->_url_id . ')';
162
+
163
+ $model_email_user_url->query($query_EmailUserUrl);
164
+
165
+ //$modelEmailUserUrl->insert($dataEmailUserUrl);
166
+ $this->_unique_click = true;
167
+ }
168
+
169
+ //increment stats counter on email_user_url clicked
170
+ $model_email_user_url = WYSIJA::get('email_user_url', 'model');
171
+ $model_email_user_url->update(array('clicked_at' => time(), 'number_clicked' => '[increment]'), $data_email_user_url);
172
+ }
173
+
174
+ /**
175
+ * record entry into the table url_mail
176
+ */
177
+ private function _record_url_mail() {
178
+ //look for url_mail entry and insert if not exists
179
+ $model_url_mail = WYSIJA::get('url_mail', 'model');
180
+ $data_url_mail = array('email_id' => $this->email_id, 'url_id' => $this->_url_id);
181
+ $urlMailObj = $model_url_mail->getOne(false, $data_url_mail);
182
+ if (!$urlMailObj) {
183
+ //we need to insert in url_mail
184
+ $model_url_mail->reset();
185
+ $model_url_mail->insert($data_url_mail);
186
+ }
187
+
188
+ $data_update = array('total_clicked' => '[increment]');
189
+ if (!$this->_unique_click)
190
+ $data_update['unique_clicked'] = '[increment]';
191
+ //increment stats counter on url_mail clicked
192
+ $model_url_mail->update($data_update, $data_url_mail);
193
+ }
194
+
195
+ /**
196
+ * record entry into the table email_user_stat
197
+ */
198
+ private function _record_email_user_stat() {
199
+ // clicked status
200
+ $status_email_user_stat = 2;
201
+
202
+ // this is the system url case (unsubscribe, view in browser and subscriptions)
203
+ if (in_array($this->clicked_url, array('[unsubscribe_link]', '[subscriptions_link]', '[view_in_browser_link]'))) {
204
+ $this->subscriberClass = WYSIJA::get('user', 'model');
205
+ $this->subscriberClass->getFormat = OBJECT;
206
+
207
+ //check if the security hash is passed to insure privacy
208
+ $receiver = $link = false;
209
+ if (isset($_REQUEST['hash'])) {
210
+ if ($_REQUEST['hash'] == md5(AUTH_KEY . $this->clicked_url . $this->user_id)) {
211
+ $receiver = $this->subscriberClass->getOne(array('user_id' => $this->user_id));
212
+ } else {
213
+ die('Security check failure.');
214
+ }
215
+ } else {
216
+ //link is not valid anymore
217
+ //propose to resend the newsletter with good links ?
218
+ $link = $this->subscriberClass->old_get_new_link_for_expired_links($this->user_id, $this->email_id);
219
+ }
220
+
221
+
222
+ switch ($this->clicked_url) {
223
+ case '[unsubscribe_link]':
224
+ //we need to make sure that this link belongs to that user
225
+ if ($receiver) {
226
+ $link = $this->subscriberClass->getUnsubLink($receiver, true);
227
+ // unsubscribe status
228
+ $status_email_user_stat = 3;
229
+ }
230
+ break;
231
+ case '[subscriptions_link]':
232
+ if ($receiver) {
233
+ $link = $this->subscriberClass->getEditsubLink($receiver, true);
234
+ }
235
+ break;
236
+ case '[view_in_browser_link]':
237
+ $model_email = WYSIJA::get('email', 'model');
238
+ $data_email = $model_email->getOne(false, array('email_id' => $this->email_id));
239
+ $helper_email = WYSIJA::get('email', 'helper');
240
+ $link = $helper_email->getVIB($data_email);
241
+ break;
242
+ }
243
+
244
+ //if the subscriber still exists in the DB we will have a link
245
+ if ($link) {
246
+ $this->decoded_url = $link;
247
+ } else {
248
+ //the subscriber doesn't appear in the DB we can redirect to the web version
249
+ $this->decoded_url = $this->_get_browser_link();
250
+
251
+ return $this->decoded_url;
252
+ }
253
+ } else {
254
+ // this is the standard non-system url case
255
+ if (strpos($this->decoded_url, 'http://') === false && strpos($this->decoded_url, 'https://') === false)
256
+ $this->decoded_url = 'http://' . $this->decoded_url;
257
+ // check that there is no broken unsubscribe link such as http://[unsubscribe_link]
258
+ if (strpos($this->decoded_url, '[unsubscribe_link]') !== false) {
259
+ $this->subscriberClass = WYSIJA::get('user', 'model');
260
+ $this->subscriberClass->getFormat = OBJECT;
261
+ $receiver = $this->subscriberClass->getOne($this->user_id);
262
+ $this->decoded_url = $this->subscriberClass->getUnsubLink($receiver, true);
263
+ }
264
+
265
+ if (strpos($this->decoded_url, '[view_in_browser_link]') !== false) {
266
+ $link = $this->_get_browser_link();
267
+ $this->decoded_url = $link;
268
+ }
269
+ }
270
+
271
+ $model_email_user_stat = WYSIJA::get('email_user_stat', 'model');
272
+ $exists = $model_email_user_stat->getOne(false, array('equal' => array('email_id' => $this->email_id, 'user_id' => $this->user_id), 'less' => array('status' => $status_email_user_stat)));
273
+ $data_update = array('status' => $status_email_user_stat);
274
+ if ($exists && isset($exists['opened_at']) && !(int) $exists['opened_at']) {
275
+ $data_update['opened_at'] = time();
276
+ }
277
+
278
+ $model_email_user_stat->reset();
279
+ $model_email_user_stat->colCheck = false;
280
+ $model_email_user_stat->update($data_update, array('equal' => array('email_id' => $this->email_id, 'user_id' => $this->user_id), 'less' => array('status' => $status_email_user_stat)));
281
+ }
282
+
283
+ /**
284
+ * update the user last_clicked or last_opened parameter
285
+ * @param type $data_update
286
+ */
287
+ private function _update_user($data_update) {
288
+ if (!empty($data_update)) {
289
+ $model_user = WYSIJA::get('user', 'model');
290
+ $model_user->update($data_update, array('user_id' => $this->user_id));
291
+ }
292
+ }
293
+
294
+ /**
295
+ * when the email sent is a preview email, we just need to emulate actions, no need to count stats or to unsubscribe user etc ...
296
+ * @return boolean
297
+ */
298
+ private function _get_url_preview() {
299
+ // we're in the case of an email preview
300
+ if (in_array($this->clicked_url, array('[unsubscribe_link]', '[subscriptions_link]', '[view_in_browser_link]'))) {
301
+ $model_user = WYSIJA::get('user', 'model');
302
+ $model_user->getFormat = OBJECT;
303
+ $user_object = $model_user->getOne(false, array('wpuser_id' => get_current_user_id()));
304
+ switch ($this->clicked_url) {
305
+ case '[unsubscribe_link]':
306
+ $link = $model_user->getConfirmLink($user_object, 'unsubscribe', false, true) . '&demo=1';
307
+
308
+ break;
309
+ case '[subscriptions_link]':
310
+ $link = $model_user->getConfirmLink($user_object, 'subscriptions', false, true) . '&demo=1';
311
+ break;
312
+ case 'view_in_browser_link':
313
+ case '[view_in_browser_link]':
314
+ if (!$this->email_id)
315
+ $this->email_id = $_REQUEST['id'];
316
+
317
+ $link = $this->_get_browser_link();
318
+ break;
319
+ }
320
+
321
+ $this->decoded_url = $link;
322
+ }else {
323
+ if (strpos($this->decoded_url, 'http://') === false && strpos($this->decoded_url, 'https://') === false)
324
+ $this->decoded_url = 'http://' . $this->decoded_url;
325
+ }
326
+ return true;
327
+ }
328
+
329
+ /**
330
+ * construct the view in your browser link
331
+ * @return type
332
+ */
333
+ private function _get_browser_link() {
334
+ $params_url = array(
335
+ 'wysija-page' => 1,
336
+ 'controller' => 'email',
337
+ 'action' => 'view',
338
+ 'email_id' => $this->email_id,
339
+ 'user_id' => 0
340
+ );
341
+ $config = WYSIJA::get('config', 'model');
342
+ return WYSIJA::get_permalink($config->getValue('confirm_email_link'), $params_url);
343
+ }
344
+
345
+ /**
346
+ * Detect if the current link is a browser link
347
+ * @param string $url
348
+ * @return boolean
349
+ */
350
+ protected function is_browser_link($url) {
351
+ $flag = false;
352
+ $helper_toolbox = WYSIJA::get('toolbox', 'helper');
353
+ if ($helper_toolbox->is_internal_link($url)) {
354
+ $url_components = parse_url($url);
355
+ if (!empty($url_components['query'])) {
356
+ $params = array();
357
+ parse_str($url_components['query'], $params);
358
+ if (!empty($params['controller']) && strtolower($params['controller']) == 'email'
359
+ && !empty($params['action']) && strtolower($params['action']) == 'view'
360
+ ) {
361
+ $flag = true;
362
+ }
363
+ }
364
+ }
365
+ return $flag;
366
+ }
367
+
368
+ /**
369
+ * check that this is a valid open action
370
+ * @return boolean
371
+ */
372
+ private function _is_open_action() {
373
+ if (empty($this->email_id) || empty($this->user_id))
374
+ return false;
375
+
376
+ return true;
377
+ }
378
+
379
+ /**
380
+ * check that this is a valid click action
381
+ * @return boolean
382
+ */
383
+ private function _is_click_action() {
384
+ if ((empty($this->email_id) || empty($this->user_id) || empty($this->clicked_url)) && $this->is_preview === false)
385
+ return false;
386
+
387
+ return true;
388
+ }
389
+
390
+ /**
391
+ * record the url into the db if not recorded already
392
+ * @return boolean
393
+ */
394
+ private function _record_url() {
395
+ //look for url entry and insert if not exists
396
+ $model_url = WYSIJA::get('url', 'model');
397
+ $url_found = $model_url->getOne(false, array('url' => $this->clicked_url));
398
+
399
+ if ($url_found) {
400
+ // we need to keep it
401
+ $this->_url_id = $url_found['url_id'];
402
+ } else {
403
+ // we need to record in database
404
+ $this->_url_id = $model_url->insert(array('url' => $this->clicked_url));
405
+ }
406
+ }
407
+
408
+ /**
409
+ * get/alter clicked url based on the global parameters
410
+ */
411
+ private function _get_clicked_url() {
412
+
413
+ if (isset($_REQUEST['urlencoded'])) {
414
+ $this->clicked_url = $_REQUEST['urlencoded'];
415
+ } elseif (isset($_REQUEST['urlpassed'])) {
416
+ $this->clicked_url = $_REQUEST['urlpassed'];
417
+ }
418
+
419
+ // make sure the url is or is not base64 encoded, some server cannot handle long url or url with encoded parameter which is the default behaviour
420
+ if (isset($_REQUEST['no64'])) {
421
+ $this->decoded_url = $this->clicked_url;
422
+ } else {
423
+ $this->clicked_url = $this->decoded_url = base64_decode($this->clicked_url);
424
+ }
425
+
426
+ if (strpos($this->clicked_url, 'utm_source') !== false) {
427
+ $this->clicked_url = $this->_clean_params_url(array('utm_source', 'utm_campaign', 'utm_medium'), $this->clicked_url);
428
+ }
429
+ return true;
430
+ }
431
+
432
+ /**
433
+ * remove any params from a url
434
+ * @param array $params_to_remove
435
+ * @param string $url
436
+ * @return string
437
+ */
438
+ private function _clean_params_url($params_to_remove = array(), $url = '') {
439
+ if (!$url){
440
+ return $url;
441
+ }
442
+
443
+ $url_splitted = explode('?', $url);
444
+ $params_in_url = array();
445
+
446
+ // lets parse the parameters of the url
447
+ parse_str($url_splitted[1], $params_in_url);
448
+
449
+ foreach ($params_to_remove as $param_key){
450
+ unset($params_in_url[$param_key]);
451
+ }
452
+
453
+ // let's use the base of that url to rebuild it without the cleaned out parameters.
454
+ $new_url = $url_splitted[0];
455
+
456
+ // when there are params left other than the ones we cleaned out of the url, we stick them back together
457
+ if($params_in_url) {
458
+ $new_url .= '?';
459
+ $i = 0;
460
+ foreach ($params_in_url as $k => $v) {
461
+ if ($i > 0){
462
+ $new_url .= '&';
463
+ }
464
+
465
+ // parse_str keeps the ampersand as a html number, let's clean it out so that the url look good when recorded
466
+ $new_url .= str_replace($k,'&#38;','') . '=' . $v;
467
+ $i++;
468
+ }
469
+ }
470
+
471
+ return $new_url;
472
+ }
473
+
474
+ }
classes/WJ_Upgrade.php ADDED
@@ -0,0 +1,390 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class WJ_Upgrade extends WYSIJA_object {
3
+ /**
4
+ * A static variable that holds a dinamic instance of the class
5
+ * @var [object||null]
6
+ */
7
+ public static $instance = null;
8
+
9
+ public static $plugins = array( 'wysija-newsletters/index.php', 'wysija-newsletters-premium/index.php' );
10
+
11
+ public static $baseurl = array(
12
+ 'core' => 'https://downloads.wordpress.org/plugin/',
13
+ 'packager' => 'http://packager.mailpoet.com/release/',
14
+ );
15
+
16
+ public static function hook(){
17
+ null === self::$instance and self::$instance = new self;
18
+
19
+ if ( ! is_admin() ) {
20
+ return;
21
+ }
22
+
23
+ self::$baseurl = (object) self::$baseurl;
24
+
25
+ add_action( 'current_screen', array( self::$instance, 'setup_bulk_screen' ) );
26
+ add_action( 'shutdown', array( self::$instance, 'setup_bulk_screen_footer' ) );
27
+ add_action( 'current_screen', array( self::$instance, 'iframe_intercept' ) );
28
+
29
+ add_action( 'init', array( self::$instance, 'update_warning' ) );
30
+
31
+ add_filter( 'pre_set_site_transient_update_plugins', array( self::$instance, 'pre_set_site_transient_update_plugins' ), 100 );
32
+ }
33
+
34
+ public function update_warning() {
35
+ if ( ! is_admin() ){
36
+ return;
37
+ }
38
+
39
+ if ( defined( 'DOING_AJAX' ) && DOING_AJAX ) {
40
+ return;
41
+ }
42
+
43
+ if ( ! ( ( is_multisite() && current_user_can( 'manage_network' ) ) || current_user_can( 'update_plugins' ) ) ){
44
+ return;
45
+ }
46
+
47
+ if ( ! function_exists( 'get_plugin_data' ) ){
48
+ include_once ABSPATH . 'wp-admin/includes/plugin.php';
49
+ }
50
+
51
+ $current = get_site_transient( 'update_plugins' );
52
+
53
+ foreach ( self::$plugins as $plugin ){
54
+ if ( isset( $current->response[$plugin] ) ){
55
+ $data = self::get_plugin_data( $plugin );
56
+
57
+ if ( version_compare( $current->response[$plugin]->new_version, $data->info->Version, '<=' ) ){
58
+ continue;
59
+ }
60
+
61
+ $this->notice(
62
+ sprintf(
63
+ __( 'Hey! %1$s has an update (version %2$s), <a href="%3$s">click here to update</a>.', WYSIJA ),
64
+ '<strong>' . esc_attr( $data->info->Name ) . '</strong>',
65
+ $current->response[$plugin]->new_version,
66
+ wp_nonce_url( self_admin_url( 'update.php?action=upgrade-plugin&plugin=' ) . $plugin, 'upgrade-plugin_' . $plugin )
67
+ ),
68
+ true,
69
+ true
70
+ );
71
+ }
72
+ }
73
+ }
74
+
75
+ public function iframe_intercept( $current_screen ) {
76
+ if ( $current_screen->base !== 'mailpoet_page_wysija_config' ){
77
+ return;
78
+ }
79
+
80
+ if ( ! isset( $_GET['action'] ) || $_GET['action'] !== 'packager-switch' ){
81
+ return;
82
+ }
83
+
84
+ // Verify if it's has been created within the last 12 hours (nonce)
85
+ if ( wp_verify_nonce( $_GET['_wpnonce'], $_GET['action'] ) !== 1 ){
86
+ return;
87
+ }
88
+
89
+ // Require the Updater classes
90
+ include_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
91
+
92
+ $to = (isset($_GET['stable']) && $_GET['stable']?'stable':'beta');
93
+
94
+ add_filter( 'pre_site_transient_update_plugins', array( $this, 'pre_site_transient_update_plugins' ) );
95
+
96
+ $plugins = array();
97
+
98
+ // Check for the action, it might be upgrading or installing
99
+ $action = 'upgrade';
100
+ if ( isset( $_GET['_mp_action'] ) && in_array( $_GET['_mp_action'], array( 'upgrade', 'install' ) ) ){
101
+ $action = strtolower( $_GET['_mp_action'] );
102
+ }
103
+
104
+ foreach ( self::$plugins as $k => $plugin ) {
105
+ if ( is_plugin_active( $plugin ) ){
106
+ $plugins[] = $plugin;
107
+ }
108
+ }
109
+
110
+ // Ajust the Padding/margin of the iFrame
111
+ define( 'IFRAME_REQUEST', true );
112
+ echo "<div style='margin: 0 20px;'>";
113
+
114
+ // Thats how WordPress calls for an iFrame page
115
+ wp_enqueue_script( 'jquery' );
116
+ iframe_header();
117
+
118
+ if ( $action === 'upgrade' ) {
119
+ $upgrader = new Plugin_Upgrader( new Bulk_Plugin_Upgrader_Skin( compact( 'nonce', 'url' ) ) );
120
+ $upgrader->bulk_upgrade( $plugins );
121
+ } elseif ( $action === 'install' ) {
122
+ // If the action is install, it will only happen if it's the Premium
123
+ $upgrader = new Plugin_Upgrader( new Plugin_Installer_Skin() );
124
+ $result = $upgrader->install( self::get_url( self::$plugins[1], WYSIJA::is_beta(), 'zip' ) );
125
+ }
126
+
127
+ iframe_footer();
128
+ echo "</div>";
129
+
130
+ remove_filter( 'pre_site_transient_update_plugins', array( $this, 'pre_site_transient_update_plugins' ) );
131
+
132
+ $model_config = WYSIJA::get( 'config', 'model' );
133
+ $model_config->save( array( 'beta_mode' => ( $to === 'stable' ? false : true ) ) );
134
+ set_site_transient( 'update_plugins', '' );
135
+
136
+ exit();
137
+ }
138
+
139
+ /**
140
+ * A static method to grab the url from the packager to grab the ZIP
141
+ * or the version of the plugin there.
142
+ *
143
+ * @uses bool_from_yn
144
+ * @uses esc_url
145
+ * @uses add_query_arg
146
+ *
147
+ * @param string $package Which package you want to grab
148
+ * @param boolean|string $beta Beta URL or not
149
+ * @param string $action Which kind of URL you need? [zip|check]
150
+ * @param string $version The version you want the URL to be related to
151
+ * @return string|null It will return the URL from the packager related to the asked action
152
+ */
153
+ public static function get_url( $package = null, $beta = false, $action = 'zip', $version = null ){
154
+ if ( is_string( $beta ) ){
155
+ if ( $beta === 'beta' ) {
156
+ $beta = true;
157
+ } else {
158
+ $beta = bool_from_yn( $beta );
159
+ }
160
+ } else {
161
+ $beta = (bool) $beta;
162
+ }
163
+
164
+ if ( ! in_array( $action, array( 'zip', 'check' ) ) ) {
165
+ return null;
166
+ }
167
+
168
+ $slug = self::get_slug( $package );
169
+
170
+ if ( true === $beta || 'wysija-newsletters-premium' === $slug ) {
171
+ $url = self::$baseurl->packager . $action;
172
+
173
+ $params = array(
174
+ 'key' => self::get_slug( $package ),
175
+ );
176
+
177
+ if ( $beta === true )
178
+ $params['beta'] = 'true';
179
+
180
+ $url = add_query_arg( $params, $url );
181
+
182
+ return (string) $url;
183
+ } else {
184
+ return (string) self::$baseurl->core . $slug . '.zip';
185
+ }
186
+ }
187
+
188
+ public static function get_version( $package = null, $beta = false ){
189
+ $request = wp_remote_get( self::get_url( $package, $beta, 'check' ) );
190
+
191
+ if ( is_wp_error( $request ) )
192
+ return false;
193
+
194
+ $version = wp_remote_retrieve_body( $request );
195
+
196
+ return $version;
197
+ }
198
+
199
+ public static function get_slug( $package = null ){
200
+ switch ( $package ) {
201
+ case self::$plugins[1]:
202
+ case 'premium':
203
+ case 'wysija-newsletters-premium':
204
+ return 'wysija-newsletters-premium';
205
+ break;
206
+
207
+ case self::$plugins[0]:
208
+ case 'base':
209
+ case 'wysija-newsletters':
210
+ default:
211
+ return 'wysija-newsletters';
212
+ break;
213
+ }
214
+ }
215
+
216
+ public static function get_plugin_data( $package = null, $beta = false, $new_version = false ){
217
+ $data = (object) array();
218
+ if ( is_null( $package ) )
219
+ return $data;
220
+
221
+ $data->id = 27505;
222
+ $data->slug = self::get_slug( $package );
223
+ $data->package = self::get_url( $package, $beta, 'zip' );
224
+
225
+ if ( function_exists( 'get_plugin_data' ) ){
226
+ $data->info = (object) get_plugin_data( plugin_dir_path( dirname( dirname( __FILE__ ) ) ) . $package );
227
+ }
228
+
229
+ if ( $data->slug === 'wysija-newsletters' ){
230
+ $data->url = "https://wordpress.org/plugins/{$data->slug}/";
231
+ } else {
232
+ $data->url = 'http://www.mailpoet.com/wordpress-newsletter-plugin-premium/';
233
+ }
234
+ $data->url = esc_url( $data->url );
235
+
236
+ if ( $new_version !== false ){
237
+ $data->new_version = (string) $new_version;
238
+ }
239
+
240
+ return $data;
241
+ }
242
+
243
+ public function pre_set_site_transient_update_plugins( $update_data ){
244
+
245
+ if ( ! function_exists( 'get_plugin_data' ) ){
246
+ return (object) array();
247
+ }
248
+
249
+ if ( ! is_object( $update_data ) && strlen( trim( $update_data ) ) === 0 ){
250
+ return (object) array();
251
+ }
252
+
253
+ if ( ! isset( $update_data->last_checked ) ){
254
+ $update_data->last_checked = 0;
255
+ }
256
+
257
+ if ( ( time() - ( 60 * 60 * 12 ) ) > ( $update_data->last_checked ) ) { // Just check once every 12 hours
258
+ return $update_data;
259
+ }
260
+
261
+ foreach ( self::$plugins as $plugin ){
262
+ if ( ! function_exists( 'is_plugin_active' ) ) {
263
+ include_once( ABSPATH . 'wp-admin/includes/plugin.php' );
264
+ }
265
+
266
+ if ( ! is_plugin_active( $plugin ) ){
267
+ continue;
268
+ }
269
+
270
+ if ( ! WYSIJA::is_beta() && $plugin === 'wysija-newsletter' ) {
271
+ continue;
272
+ }
273
+
274
+ $version = self::get_version( $plugin, WYSIJA::is_beta() );
275
+ $update_data->last_checked = time();
276
+
277
+ if ( version_compare( WYSIJA::get_version( $plugin ), $version, '>=' ) ){
278
+ continue;
279
+ }
280
+
281
+ $update_data->response[$plugin] = self::get_plugin_data( $plugin, WYSIJA::is_beta(), $version );
282
+ }
283
+
284
+ return $update_data;
285
+ }
286
+
287
+ public function pre_site_transient_update_plugins( $transient ) {
288
+ $update_data = (object) array(
289
+ 'last_checked' => time() - 10,
290
+ 'response' => array()
291
+ );
292
+ $to = (isset($_GET['stable']) && $_GET['stable']?'stable':'beta');
293
+
294
+ foreach ( self::$plugins as $plugin ) {
295
+ if ( ! function_exists( 'is_plugin_active' ) ){
296
+ include_once( ABSPATH . 'wp-admin/includes/plugin.php' );
297
+ }
298
+
299
+ if ( ! is_plugin_active( $plugin ) ) {
300
+ continue;
301
+ }
302
+
303
+ $update_data->response[$plugin] = self::get_plugin_data( $plugin, $to, self::get_version( $plugin, $to ) );
304
+ }
305
+
306
+ return $update_data;
307
+ }
308
+
309
+ public function setup_bulk_screen( $current_screen ) {
310
+ global $title, $parent_file, $submenu_file;
311
+
312
+ if ( strtoupper( $_SERVER['REQUEST_METHOD'] ) === 'POST' && in_array( $current_screen->id, array( 'update-core', 'plugins' ) ) ) {
313
+ if ( !isset( $_POST['checked'] ) ){
314
+ return;
315
+ }
316
+
317
+ $plugins = (array) $_POST['checked'];
318
+ $plugins = array_map( 'urldecode', $plugins );
319
+
320
+ $__intersection = array_intersect( $plugins, self::$plugins );
321
+
322
+ if ( empty( $__intersection ) )
323
+ return;
324
+
325
+ switch ( $_POST['action'] ){
326
+ case 'delete-selected':
327
+ break;
328
+
329
+ case 'deactivate-selected':
330
+ if ( in_array( self::$plugins[0], $plugins ) && ! in_array( self::$plugins[1], $plugins ) && is_plugin_active( self::$plugins[1] ) ){
331
+ $plugins[] = self::$plugins[1];
332
+ }
333
+ break;
334
+
335
+ case 'update-selected':
336
+ case 'activate-selected':
337
+ if ( in_array( self::$plugins[1], $plugins ) && ! in_array( self::$plugins[0], $plugins ) ){
338
+ $plugins[] = self::$plugins[0];
339
+ }
340
+
341
+ break;
342
+ }
343
+
344
+ $_POST['checked'] = $plugins;
345
+
346
+ return;
347
+ }
348
+
349
+
350
+ if ( $current_screen->id !== 'update' ){
351
+ return;
352
+ }
353
+
354
+ if ( $_GET['action'] !== 'upgrade-plugin' ){
355
+ return;
356
+ }
357
+
358
+ if ( $_GET['action'] === 'upgrade-plugin' && ! in_array( $_GET['plugin'], self::$plugins ) ){
359
+ return;
360
+ }
361
+
362
+ foreach ( self::$plugins as $plugin ) {
363
+ if ( ! is_plugin_active( $plugin ) ) {
364
+ return;
365
+ }
366
+ }
367
+
368
+ $_GET['action'] = $_REQUEST['action'] = 'update-selected';
369
+ $_GET['plugins'] = $_REQUEST['plugins'] = implode( ',', array_map( 'urlencode', self::$plugins ) );
370
+ $_GET['_wpnonce'] = $_REQUEST['_wpnonce'] = wp_create_nonce( 'bulk-update-plugins' );
371
+ $_GET['_wysija_bulk_screen'] = $_REQUEST['_wysija_bulk_screen'] = true;
372
+
373
+ $title = esc_attr__( 'Update Plugin' );
374
+ $parent_file = 'plugins.php';
375
+ $submenu_file = 'plugins.php';
376
+ require_once(ABSPATH . 'wp-admin/admin-header.php');
377
+ echo
378
+ "<div class='wrap'>" .
379
+ "<h2>{$title}</h2>";
380
+
381
+ }
382
+
383
+ public function setup_bulk_screen_footer(){
384
+ if ( ! isset( $_GET['_wysija_bulk_screen'] ) ){
385
+ return;
386
+ }
387
+ echo "</div>";
388
+ include(ABSPATH . 'wp-admin/admin-footer.php');
389
+ }
390
+ }
classes/WJ_Utils.php ADDED
@@ -0,0 +1,69 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Wysija Premium Utils.
5
+ * Utilities singleton not related to a specific model.
6
+ */
7
+ class WJ_Utils {
8
+
9
+ private function __construct() {
10
+ }
11
+
12
+ public static function to_int($value) {
13
+ $boolean = (bool) $value;
14
+ if ($boolean) {
15
+ $int = 1;
16
+ } else {
17
+ $int = 0;
18
+ }
19
+ return $int;
20
+ }
21
+
22
+ public static function to_bool($value) {
23
+ $int = (int) $value;
24
+ if ($int == 0) {
25
+ $boolean = false;
26
+ } else {
27
+ $boolean = true;
28
+ }
29
+ return $boolean;
30
+ }
31
+
32
+ /**
33
+ * Retuns plain domain name.
34
+ *
35
+ * @return string $domain
36
+ */
37
+ public static function get_domain() {
38
+ if(isset($_SERVER['SERVER_NAME']) && strlen(trim($_SERVER['SERVER_NAME'])) > 0) {
39
+ $domain = strtolower($_SERVER['SERVER_NAME']);
40
+ } else {
41
+ $domain = preg_replace('@http[s]?:\/\/@', '', get_site_url());
42
+ }
43
+
44
+ return preg_replace('@^www\.@', '', $domain);
45
+ }
46
+
47
+ /**
48
+ * Reruns necessary data for tooltip.
49
+ *
50
+ * @return array $data
51
+ */
52
+ public static function get_tip_data() {
53
+ $model_config = WYSIJA::get('config', 'model');
54
+ $is_gmail = 'false';
55
+
56
+ if ('gmail' === $model_config->getValue('sending_method')) {
57
+ $is_gmail = 'true';
58
+ }
59
+
60
+ $data = array(
61
+ 'text' => sprintf(__('Use an email from your domain, like <strong>info@%s</strong> to avoid the spam folder, or even be blocked.', WYSIJA), self::get_domain()),
62
+ 'gmailText' => sprintf(__('If you want to use a Gmail address, you need to send with Gmail. Check <a id="tip-send-with" href="%s">sending settings</a>.', WYSIJA), admin_url('admin.php?page=wysija_config#tab-sendingmethod')),
63
+ 'domain' => self::get_domain(),
64
+ 'isGmail' => $is_gmail
65
+ );
66
+
67
+ return $data;
68
+ }
69
+ }
classes/index.html ADDED
File without changes
controllers/ajax/campaigns.php ADDED
@@ -0,0 +1,1004 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_control_back_campaigns extends WYSIJA_control{
4
+
5
+ function WYSIJA_control_back_campaigns(){
6
+ if(!WYSIJA::current_user_can('wysija_newsletters')) die('Action is forbidden.');
7
+ parent::WYSIJA_control();
8
+ }
9
+
10
+ function save_poll(){
11
+ $model_config = WYSIJA::get('config','model');
12
+ $model_config->save(array('poll_origin' => $_REQUEST['how'] , 'poll_origin_url' => $_REQUEST['where']));
13
+
14
+ $res['result'] = true;
15
+ $res['msg'] = '<span><span class="checkmark">---</span>'. __('Thanks!',WYSIJA). '</span>';
16
+ return $res;
17
+ }
18
+
19
+ function search_terms( $request = null ){
20
+ $response = (object) array(
21
+ 'status' => false,
22
+ 'message' => __( 'Your request has failed', WYSIJA ),
23
+ 'results' => array(),
24
+ 'more' => true,
25
+ );
26
+
27
+ if ( ( ! defined( 'DOING_AJAX' ) && is_null( $request ) ) || ! is_user_logged_in() ){
28
+ return $response;
29
+ }
30
+
31
+ $request = (object) wp_parse_args(
32
+ $request,
33
+ array(
34
+ 'search' => isset( $_GET['search'] ) ? $_GET['search'] : '',
35
+ 'post_type' => isset( $_GET['post_type'] ) ? $_GET['post_type'] : null,
36
+ 'page' => absint( isset( $_GET['page'] ) ? $_GET['page'] : 0 ),
37
+ 'page_limit' => absint( isset( $_GET['page_limit'] ) ? $_GET['page_limit'] : 10 ),
38
+ )
39
+ );
40
+
41
+ if ( is_null( $request->post_type ) ){
42
+ return $response;
43
+ }
44
+
45
+ $response->status = true;
46
+ $response->message = __( 'Request successful', WYSIJA );
47
+
48
+ $response->post_type = get_post_types( array( 'name' => $request->post_type ) );
49
+ $response->post_type = reset( $response->post_type );
50
+
51
+ preg_match( '/@(\w+)/i', $request->search, $response->regex );
52
+
53
+ if ( ! empty( $response->regex ) ){
54
+ $request->search = array_filter( array_map( 'trim', explode( '|', str_replace( $response->regex[0], '|', $request->search ) ) ) );
55
+ $request->search = reset( $request->search );
56
+ $taxonomies = $response->regex[1];
57
+ } else {
58
+ $taxonomies = get_object_taxonomies( $response->post_type );
59
+ }
60
+ $response->taxonomies = get_object_taxonomies( $response->post_type, 'objects' );
61
+
62
+ $response->results = get_terms(
63
+ (array) $taxonomies,
64
+ array(
65
+ 'hide_empty' => false,
66
+ 'search' => $request->search,
67
+ 'number' => $request->page_limit,
68
+ 'offset' => $request->page_limit * ( $request->page - 1 ),
69
+ )
70
+ );
71
+
72
+ if ( empty( $response->results ) || count( $response->results ) < $request->page_limit ){
73
+ $response->more = false;
74
+ }
75
+
76
+ return $response;
77
+ }
78
+
79
+ function switch_theme() {
80
+ if(isset($_POST['wysijaData'])) {
81
+ $rawData = $_POST['wysijaData'];
82
+ // avoid using stripslashes as it's not reliable depending on the magic quotes settings
83
+ $rawData = str_replace('\"', '"', $rawData);
84
+ // decode JSON data
85
+ $rawData = json_decode($rawData, true);
86
+
87
+ $theme = (isset($rawData['theme'])) ? $rawData['theme'] : 'default';
88
+
89
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
90
+ $res['templates'] = $helper_wj_engine->renderTheme($theme);
91
+
92
+ $email_id = (int)$_REQUEST['id'];
93
+
94
+ $campaignsHelper = WYSIJA::get('campaigns', 'helper');
95
+
96
+ if(isset($res['templates']['divider_options'])) {
97
+ // save divider
98
+ $campaignsHelper->saveParameters($email_id, 'divider', $res['templates']['divider_options']);
99
+ }
100
+
101
+ // save theme used
102
+ $campaignsHelper->saveParameters($email_id, 'theme', $theme);
103
+
104
+ $res['templates']['theme'] = $theme;
105
+ $res['styles'] = $helper_wj_engine->renderThemeStyles($theme);
106
+ } else {
107
+ $res['msg'] = __("The theme you selected could not be loaded.",WYSIJA);
108
+ $res['result'] = false;
109
+ }
110
+ return $res;
111
+ }
112
+
113
+ function save_editor() {
114
+ // decode json data and convert to array
115
+ $rawData = '';
116
+ if(isset($_POST['wysijaData'])) {
117
+ $rawData = $_POST['wysijaData'];
118
+ // avoid using stripslashes as it's not reliable depending on the magic quotes settings
119
+ $rawData = str_replace('\"', '"', $rawData);
120
+ // decode JSON data
121
+ $rawData = json_decode($rawData, true);
122
+ }
123
+
124
+ if(!$rawData){
125
+ $this->error('Error saving',false);
126
+ return array('result' => false);
127
+ }
128
+
129
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
130
+ $helper_wj_engine->setData($rawData);
131
+ $result = false;
132
+
133
+ // get email id
134
+ $email_id = (int)$_REQUEST['id'];
135
+
136
+ $model_email = WYSIJA::get('email', 'model');
137
+ $emailData = $model_email->getOne(array('wj_styles', 'subject', 'params', 'email_id', 'campaign_id'), array('email_id' => $email_id));
138
+
139
+ $helper_wj_engine->setStyles($emailData['wj_styles'], true);
140
+
141
+ $values = array('wj_data' => $helper_wj_engine->getEncoded('data'));
142
+ $values['body'] = $helper_wj_engine->renderEmail($emailData);
143
+ $values['email_id'] = $email_id;
144
+
145
+ $updated_email = $helper_wj_engine->getEmailData();
146
+
147
+ // update modified_at timestamp
148
+ $model_email->columns['modified_at']['autoup']=1;
149
+
150
+ // update data in DB
151
+ $result = $model_email->update($values, array('email_id' => $email_id));
152
+
153
+ if(!$result) {
154
+ // throw error
155
+ $this->error(__('Your email could not be saved', WYSIJA));
156
+ } else {
157
+ // save successful
158
+ $this->notice(__('Your email has been saved', WYSIJA));
159
+ }
160
+
161
+ return array('result' => $result);
162
+ }
163
+
164
+ function save_styles() {
165
+ // decode json data and convert to array
166
+ $rawData = '';
167
+ if(isset($_POST['wysijaStyles'])) {
168
+ $rawData = $_POST['wysijaStyles'];
169
+ // avoid using stripslashes as it's not reliable depending on the magic quotes settings
170
+ $rawData = str_replace('\"', '"', $rawData);
171
+ // decode JSON data
172
+ $rawData = json_decode($rawData, true);
173
+
174
+ }
175
+
176
+ // handle checkboxes
177
+ if(array_key_exists('a-underline', $rawData) === false) {
178
+ $rawData['a-underline'] = -1;
179
+ }
180
+
181
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
182
+ $helper_wj_engine->setStyles($helper_wj_engine->formatStyles($rawData));
183
+
184
+ $result = false;
185
+
186
+ $values = array(
187
+ 'wj_styles' => $helper_wj_engine->getEncoded('styles')
188
+ );
189
+
190
+ // get email id
191
+ $email_id = (int)$_REQUEST['id'];
192
+
193
+ // update data in DB
194
+ $model_email = WYSIJA::get('email', 'model');
195
+ $result = $model_email->update($values, array('email_id' => $email_id));
196
+
197
+ if(!$result) {
198
+ // throw error
199
+ $this->error(__('Styles could not be saved', WYSIJA));
200
+ } else {
201
+ // save successful
202
+ $this->notice(__('Styles have been saved', WYSIJA));
203
+ }
204
+
205
+ return array(
206
+ 'styles' => $helper_wj_engine->renderStyles(),
207
+ 'result' => $result
208
+ );
209
+ }
210
+
211
+ function deleteimg(){
212
+
213
+ if(isset($_REQUEST['imgid']) && $_REQUEST['imgid']>0){
214
+ /* delete the image with id imgid */
215
+ $result=wp_delete_attachment($_REQUEST['imgid'],true);
216
+ if($result){
217
+ $this->notice(__('Image has been deleted.',WYSIJA));
218
+ }
219
+ }
220
+
221
+ $res=array();
222
+ $res['result'] = $result;
223
+ return $res;
224
+ }
225
+
226
+ function deleteTheme(){
227
+ if(isset($_REQUEST['themekey']) && $_REQUEST['themekey']){
228
+ /* delete the image with id imgid */
229
+ $helperTheme=WYSIJA::get('themes','helper');
230
+ $result=$helperTheme->delete($_REQUEST['themekey']);
231
+ }
232
+
233
+ $res=array();
234
+ $res['result'] = $result;
235
+ return $res;
236
+ }
237
+
238
+ // set newsletter default theme
239
+ function setDefaultTheme() {
240
+ if(isset($_REQUEST['theme']) && $_REQUEST['theme']) {
241
+ // check that the theme exists
242
+ // TODO
243
+ $theme_exists = true;
244
+ if($theme_exists === true) {
245
+ // update config
246
+ $model_config = WYSIJA::get('config', 'model');
247
+ $model_config->save(array('newsletter_default_theme' => $_REQUEST['theme']));
248
+
249
+ $result = true;
250
+ } else {
251
+ $result = false;
252
+ }
253
+ }
254
+
255
+ return array('result' => $result);
256
+ }
257
+
258
+ function save_IQS() {
259
+ // decode json data and convert to array
260
+ $wysijaIMG = '';
261
+ if(isset($_POST['wysijaIMG'])) {
262
+ $wysijaIMG = json_decode(stripslashes($_POST['wysijaIMG']), TRUE);
263
+ }
264
+ $values = array(
265
+ 'params' => array('quickselection'=>$wysijaIMG)
266
+ );
267
+
268
+ // get email id
269
+ $email_id = (int)$_REQUEST['id'];
270
+ $values['email_id']=$email_id;
271
+
272
+ // update data in DB
273
+ $model_email = WYSIJA::get('email', 'model');
274
+ $result = $model_email->update($values, array('email_id' => $email_id));
275
+
276
+ if(!$result) {
277
+ // throw error
278
+ $this->error(__('Image selection has not been saved.', WYSIJA));
279
+ } else {
280
+ // save successful
281
+ $this->notice(__('Image selection has been saved.', WYSIJA));
282
+ }
283
+
284
+ return array('result' => $result);
285
+ }
286
+
287
+
288
+ function view_NL() {
289
+ // get campaign id
290
+ $email_id = (int)$_REQUEST['id'];
291
+
292
+ // update data in DB
293
+ $model_email = WYSIJA::get('email', 'model');
294
+ $result = $model_email->getOne(false,array('email_id' => $email_id));
295
+
296
+ echo $result['body'];
297
+ exit;
298
+ }
299
+
300
+ function display_NL() {
301
+ // get email id
302
+ $email_id = (int)$_REQUEST['id'];
303
+
304
+ // update data in DB
305
+ $model_email = WYSIJA::get('email', 'model');
306
+ $email= $model_email->getOne(false,array('email_id' => $email_id));
307
+
308
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
309
+ $helper_wj_engine->setStyles($result['wj_styles'], true);
310
+ $helper_wj_engine->setData($result['wj_data'], true);
311
+ $html = $helper_wj_engine->renderEmail($email);
312
+ print $html;
313
+ exit;
314
+ }
315
+
316
+ /**
317
+ * returns the images attached to displayed posts
318
+ */
319
+ function get_post_images() {
320
+ // get parameters
321
+ $params = array(
322
+
323
+ );
324
+
325
+ $result = true;
326
+
327
+ if($params['post_id'] === null) {
328
+
329
+ }
330
+
331
+ return array(
332
+ 'result' => $result,
333
+ 'images' => $images
334
+ );
335
+ }
336
+
337
+ /**
338
+ * returns a list of articles to the popup in the visual editor
339
+ * @global type $wpdb
340
+ * @return boolean
341
+ */
342
+ function get_articles(){
343
+ // fixes issue with pcre functions
344
+ @ini_set('pcre.backtrack_limit', 1000000);
345
+
346
+ // get parameters
347
+ $raw_data = $_REQUEST['data'];
348
+ $params = array();
349
+ foreach ($raw_data as $value) {
350
+ $params[$value['name']] = $value['value'];
351
+ }
352
+
353
+ // get options
354
+ $model_config = WYSIJA::get('config', 'model');
355
+ $interpret_shortcode = (bool)$model_config->getValue('interp_shortcode');
356
+
357
+ // post statuses
358
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
359
+ $post_statuses = $helper_wp_tools->get_post_statuses();
360
+ $post_types = $helper_wp_tools->get_post_types();
361
+
362
+ // filter by post_type
363
+ if(isset($params['post_type'])) {
364
+ $post_types_filter = array();
365
+ if(strlen(trim($params['post_type'])) === 0) {
366
+ $post_types_filter = array_keys($post_types);
367
+ $post_types_filter[] = 'post';
368
+ $post_types_filter[] = 'page';
369
+ } else {
370
+ $post_types_filter = trim($params['post_type']);
371
+ }
372
+ // set condition on post type
373
+ $params['post_type'] = $post_types_filter;
374
+ }
375
+
376
+ // query offset when doing incremental loading
377
+ $query_offset = (isset($_REQUEST['query_offset']) && (int)$_REQUEST['query_offset'] >= 0) ? (int)$_REQUEST['query_offset'] : 0;
378
+ $params['query_offset'] = $query_offset;
379
+
380
+ // fetch posts
381
+ $helper_articles = WYSIJA::get('articles', 'helper');
382
+
383
+ // set is_search_query (true) to get a count in addition to the results
384
+ $params['is_search_query'] = true;
385
+
386
+ $model_wp_posts = WYSIJA::get('wp_posts','model');
387
+ $data = $model_wp_posts->get_posts($params);
388
+
389
+ // extract data
390
+ $posts = $data['rows'];
391
+ // contains the total number of rows available
392
+ $count = $data['count'];
393
+
394
+ // return results
395
+ $result = array(
396
+ 'result' => true,
397
+ 'append' => ($query_offset > 0)
398
+ );
399
+
400
+ if(empty($posts) === false) {
401
+ foreach($posts as $key => $post) {
402
+ // interpret shortcodes
403
+ if($interpret_shortcode === true) {
404
+ $posts[$key]['post_content'] = apply_filters('the_content', $posts[$key]['post_content']);
405
+ }
406
+
407
+ // get thumbnail
408
+ $posts[$key]['post_image'] = $helper_articles->getImage($post);
409
+
410
+ // set post status
411
+ $post_status_label = '';
412
+ if(isset($post_statuses[$posts[$key]['post_status']])) {
413
+ $post_status_label = $post_statuses[$posts[$key]['post_status']];
414
+ }
415
+ $posts[$key]['post_status'] = $post_status_label;
416
+ }
417
+ $result['posts'] = $posts;
418
+ $result['total'] = (int)$count['total'];
419
+ }else {
420
+ $result['msg'] = __('There are no posts corresponding to that search.', WYSIJA);
421
+ $result['result'] = false;
422
+ }
423
+
424
+ return $result;
425
+ }
426
+
427
+ function insert_articles() {
428
+ // get raw params
429
+ $raw_params = $_REQUEST['data'];
430
+
431
+ // format params
432
+ $params = array();
433
+ foreach($raw_params as $value) {
434
+ $params[$value['name']] = $value['value'];
435
+ }
436
+
437
+ if($params['show_divider'] === 'yes') {
438
+ // get divider
439
+ $divider = $_REQUEST['divider'];
440
+ } else {
441
+ $divider = null;
442
+ }
443
+ $params['divider'] = $divider;
444
+
445
+ // get post ids
446
+ $post_ids = array();
447
+ if(isset($_REQUEST['post_ids']) && strlen(trim($_REQUEST['post_ids'])) > 0) {
448
+ $post_ids = explode(',', $_REQUEST['post_ids']);
449
+ }
450
+
451
+ if(empty($post_ids)) {
452
+ // return error
453
+ $res['msg'] = __('Please select an article.', WYSIJA);
454
+ $res['result'] = false;
455
+ return $res;
456
+ }
457
+
458
+ // specify custom fields to get from posts
459
+ $post_params = array('include' => $post_ids);
460
+
461
+ // include sort by parameter into post params
462
+ $post_params['sort_by'] = $params['sort_by'];
463
+
464
+ // get posts
465
+ $model_wp_posts = WYSIJA::get('wp_posts', 'model');
466
+ $posts = $model_wp_posts->get_posts($post_params);
467
+
468
+ // check if we need to interpret shortcodes
469
+ $model_config = WYSIJA::get('config', 'model');
470
+ $interpret_shortcode = (bool)$model_config->getValue('interp_shortcode');
471
+
472
+ // get some model and helpers
473
+ $helper_articles = WYSIJA::get('articles', 'helper');
474
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
475
+
476
+ $output = '';
477
+
478
+ // save parameters for next time
479
+ $model_config->save(array('insert_post_parameters' => $helper_wj_engine->encodeParameters($params)));
480
+
481
+ foreach($posts as $key => $post) {
482
+ if($interpret_shortcode === true) {
483
+ // interpret shortcodes
484
+ $posts[$key]['post_content'] = apply_filters('the_content', $post['post_content']);
485
+ }
486
+ // get thumbnail
487
+ $posts[$key]['post_image'] = $helper_articles->getImage($post);
488
+ }
489
+
490
+ $output .= base64_encode($helper_wj_engine->renderPostsToBlocks($posts, $params));
491
+
492
+ if(strlen($output) > 0) {
493
+ $res['result'] = true;
494
+ $res['posts'] = $output;
495
+ }else {
496
+ $res['msg'] = __('There are no posts corresponding to that search.',WYSIJA);
497
+ $res['result'] = false;
498
+ }
499
+
500
+ return $res;
501
+ }
502
+
503
+ function send_preview($spamtest=false){
504
+ $mailer=WYSIJA::get('mailer','helper');
505
+ $email_id = $_REQUEST['id'];
506
+ $resultarray=array();
507
+
508
+ // update data in DB
509
+ $model_email = WYSIJA::get('email', 'model');
510
+ $model_email->getFormat=OBJECT;
511
+ $email_object = $model_email->getOne(false,array('email_id' => $email_id));
512
+ $mailer->testemail=true;
513
+
514
+
515
+ if(isset($_REQUEST['data'])){
516
+ $dataTemp=$_REQUEST['data'];
517
+ $_REQUEST['data']=array();
518
+ foreach($dataTemp as $val) $_REQUEST['data'][$val['name']]=$val['value'];
519
+ unset($dataTemp);
520
+ foreach($_REQUEST['data'] as $k =>$v){
521
+ $newkey=str_replace(array('wysija[email][',']'),'',$k);
522
+ $configVal[$newkey]=$v;
523
+ }
524
+ if(isset($configVal['from_name'])){
525
+ $params=array(
526
+ 'from_name'=>$configVal['from_name'],
527
+ 'from_email'=>$configVal['from_email'],
528
+ 'replyto_name'=>$configVal['replyto_name'],
529
+ 'replyto_email'=>$configVal['replyto_email']);
530
+ if(isset($configVal['subject'])) $email_object->subject=$configVal['subject'];
531
+ }
532
+
533
+ }else{
534
+ $params=array(
535
+ 'from_name'=>$email_object->from_name,
536
+ 'from_email'=>$email_object->from_email,
537
+ 'replyto_name'=>$email_object->replyto_name,
538
+ 'replyto_email'=>$email_object->replyto_email
539
+ );
540
+ }
541
+ if(strpos($_REQUEST['receiver'], ',')) {
542
+ $receivers = explode(',',$_REQUEST['receiver']);
543
+ } else if(strpos($_REQUEST['receiver'], ';')) {
544
+ $receivers = explode(';',$_REQUEST['receiver']);
545
+ } else {
546
+ $receivers = array($_REQUEST['receiver']);
547
+ }
548
+
549
+ $user_model = WYSIJA::get('user', 'model');
550
+ foreach($receivers as $key => $receiver){
551
+ $receiver = trim($receiver);
552
+ $dummy_receiver = $user_model->get_object_by_email($receiver);
553
+ if(empty($dummy_receiver)){
554
+ $dummy_receiver = new stdClass();
555
+ $dummy_receiver->user_id = 0;
556
+ $dummy_receiver->email = $receiver;
557
+ $dummy_receiver->status = 1;
558
+ $dummy_receiver->lastname = $dummy_receiver->firstname = '';
559
+ }
560
+
561
+ if($spamtest){
562
+ $langextra = '';
563
+ $dummy_receiver->firstname ='Mail Tester';
564
+
565
+ $wp_lang = get_locale();
566
+ if(!empty($wp_lang)) $langextra ='&lang='.$wp_lang;
567
+ $resultarray['urlredirect']='http://www.mail-tester.com/check.php?id='.urlencode($dummy_receiver->email).$langextra;
568
+ }
569
+ $receivers[$key] = $dummy_receiver;
570
+
571
+ }
572
+
573
+ $email_clone=array();
574
+ foreach($email_object as $kk=>$vv) $email_clone[$kk]=$vv;
575
+
576
+
577
+ $wj_engine = WYSIJA::get('wj_engine', 'helper');
578
+ // set data & styles
579
+ if(isset($email_clone['wj_data'])) { $wj_engine->setData($email_clone['wj_data'], true); } else { $wj_engine->setData(); }
580
+ if(isset($email_clone['wj_styles'])) { $wj_engine->setStyles($email_clone['wj_styles'], true); } else { $wj_engine->setStyles(); }
581
+
582
+ // generate email html body
583
+ $body = $wj_engine->renderEmail($email_clone);
584
+
585
+ // get back email data as it will be updated during the rendering (articles ids + articles count)
586
+ $email_child = $wj_engine->getEmailData();
587
+
588
+ // [total] [number] and [post_title] are only valid for post notifications newsletter
589
+ if((int)$email_child['type'] === 2 && isset($email_child['params']['autonl']['event']) &&
590
+ $email_child['params']['autonl']['event'] === 'new-articles' && isset($email_child['params']['autonl']['articles'])){
591
+
592
+ $item_count = 0;
593
+ $total_count = 1;
594
+ $first_subject = '';
595
+
596
+ if(isset($email_child['params']['autonl']['articles']['count'])) $item_count = (int)$email_child['params']['autonl']['articles']['count'];
597
+ if(isset($email_child['params']['autonl']['articles']['first_subject'])) $first_subject = $email_child['params']['autonl']['articles']['first_subject'];
598
+ if(isset($email_child['params']['autonl']['total_child'])) $total_count = (int)$email_child['params']['autonl']['total_child'] + 1;
599
+
600
+ $email_object->subject = str_replace(
601
+ array('[total]','[number]','[post_title]'),
602
+ array($item_count, $total_count, $first_subject),
603
+ $email_child['subject']
604
+ );
605
+ }
606
+ $successmsg = __('Your email preview has been sent to %1$s', WYSIJA);
607
+
608
+ // correction added for post notifications with the tag [newsletter:post_title] failing to send
609
+ if(isset($email_object->params['autonl']) && isset($email_child['params']['autonl'])){
610
+ $email_object->params['autonl']=$email_child['params']['autonl'];
611
+ }
612
+
613
+ if(isset($email_object->params)) {
614
+ $params['params']=$email_object->params;
615
+
616
+ if(isset($configVal['params[googletrackingcode'])){
617
+ $paramsemail=array();
618
+ if(!is_array($email_object->params)) $paramsemail=unserialize(base64_decode($email_object->params));
619
+
620
+ if(trim($configVal['params[googletrackingcode'])) {
621
+ $paramsemail['googletrackingcode']=$configVal['params[googletrackingcode'];
622
+ }
623
+ else {
624
+ unset($paramsemail['googletrackingcode']);
625
+ }
626
+ $params['params'] = base64_encode(serialize($paramsemail));
627
+ }
628
+ }
629
+
630
+ $params['email_id'] = $email_object->email_id;
631
+ $receiversList = array();
632
+ $res = false;
633
+ foreach($receivers as $receiver){
634
+ if($mailer->sendSimple($receiver, stripslashes($email_object->subject),$email_object->body,$params)) {
635
+ $res = true;
636
+ $receiversList[] = $receiver->email;
637
+ }
638
+ WYSIJA::log('preview_sent', $mailer, 'manual');
639
+ }
640
+
641
+ if($res === true) {
642
+ $this->notice(sprintf($successmsg, implode(', ', $receiversList)));
643
+ }
644
+
645
+ $resultarray['result'] = $res;
646
+
647
+ return $resultarray;
648
+ }
649
+
650
+ /**
651
+ * send spam test function step 2 of the newsletter edition process
652
+ */
653
+ function send_spamtest(){
654
+ return apply_filters('wysija_send_spam_test','',$this);
655
+ }
656
+
657
+ function set_divider()
658
+ {
659
+ $src = isset($_POST['wysijaData']['src']) ? $_POST['wysijaData']['src'] : NULL;
660
+ $width = isset($_POST['wysijaData']['width']) ? (int)$_POST['wysijaData']['width'] : NULL;
661
+ $height = isset($_POST['wysijaData']['height']) ? (int)$_POST['wysijaData']['height'] : NULL;
662
+
663
+ if($src === NULL OR $width === NULL OR $height === NULL) {
664
+ // there is a least one missing parameter, fallback to default divider
665
+ $dividersHelper = WYSIJA::get('dividers', 'helper');
666
+ $divider = $dividersHelper->getDefault();
667
+ } else {
668
+ // use provided params
669
+ $divider = array(
670
+ 'src' => $src,
671
+ 'width' => $width,
672
+ 'height' => $height
673
+ );
674
+ }
675
+
676
+ // update campaign parameters
677
+ $email_id = (int)$_REQUEST['id'];
678
+ $campaignsHelper = WYSIJA::get('campaigns', 'helper');
679
+ $campaignsHelper->saveParameters($email_id, 'divider', $divider);
680
+
681
+ // set params
682
+ $block = array_merge(array('no-block' => true, 'type' => 'divider'), $divider);
683
+
684
+ $helper_wj_engine=WYSIJA::get('wj_engine','helper');
685
+ return base64_encode($helper_wj_engine->renderEditorBlock($block));
686
+ }
687
+
688
+ function get_social_bookmarks() {
689
+ $size = isset($_POST['wysijaData']['size']) ? $_POST['wysijaData']['size'] : NULL;
690
+ $theme = isset($_POST['wysijaData']['theme']) ? $_POST['wysijaData']['theme'] : NULL;
691
+
692
+ $bookmarksHelper = WYSIJA::get('bookmarks', 'helper');
693
+ $bookmarks = $bookmarksHelper->getAll($size, $theme);
694
+ return json_encode(array('icons' => $bookmarks));
695
+ }
696
+
697
+ function generate_social_bookmarks() {
698
+
699
+ $size = 'medium';
700
+ $iconset = '01';
701
+
702
+ if(isset($_POST['wysijaData']) && !empty($_POST['wysijaData'])) {
703
+ $data = $_POST['wysijaData'];
704
+ $items = array();
705
+
706
+ foreach($data as $key => $values) {
707
+ if($values['name'] === 'bookmarks-size') {
708
+ // get size
709
+ $size = $values['value'];
710
+ } else if($values['name'] === 'bookmarks-theme') {
711
+ // get theme name
712
+ $theme = $values['value'];
713
+ } else if($values['name'] === 'bookmarks-iconset') {
714
+ // get iconset
715
+ $iconset = $values['value'];
716
+ if(strlen(trim($iconset)) === 0) {
717
+ $this->error('No iconset specified', false);
718
+ return false;
719
+ }
720
+ } else {
721
+ $keys = explode('-', $values['name']);
722
+ $network = $keys[1];
723
+ $property = $keys[2];
724
+ if(array_key_exists($network, $items)) {
725
+ $items[$network][$property] = $values['value'];
726
+ } else {
727
+ $items[$network] = array($property => $values['value']);
728
+ }
729
+ }
730
+ }
731
+ }
732
+
733
+ $urls = array();
734
+ // check data and remove network with an empty url
735
+ foreach($items as $network => $item) {
736
+ if(strlen(trim($item['url'])) === 0) {
737
+ // empty url
738
+ unset($items[$network]);
739
+ } else {
740
+ // url specified
741
+ $urls[$network] = $item['url'];
742
+ }
743
+ }
744
+
745
+ // check if there's at least one url left
746
+ if(empty($urls)) {
747
+ $this->error('No url specified', false);
748
+ return false;
749
+ }
750
+
751
+ // save url in config
752
+ $config=WYSIJA::get('config','model');
753
+ $config->save(array('social_bookmarks' => $urls));
754
+
755
+ // get iconset icons
756
+ $bookmarksHelper = WYSIJA::get('bookmarks', 'helper');
757
+
758
+ // if the iconset is 00, then it's the theme's bookmarks
759
+ if($iconset === '00') {
760
+ $icons = $bookmarksHelper->getAllByTheme($theme);
761
+ } else {
762
+ // otherwise it's a basic iconset
763
+ $icons = $bookmarksHelper->getAllByIconset($size, $iconset);
764
+ }
765
+
766
+
767
+ // format data
768
+ $block = array(
769
+ 'position' => 1,
770
+ 'type' => 'gallery',
771
+ 'items' => array(),
772
+ 'alignment' => 'center'
773
+ );
774
+
775
+ $width = 0;
776
+ foreach($items as $key => $item) {
777
+ $block['items'][] = array_merge($item, $icons[$key], array('alt' => ucfirst($key)));
778
+ $width += (int)$icons[$key]['width'];
779
+ }
780
+ // add margin between icons
781
+ $width += (count($block['items']) - 1) * 10;
782
+ // set optimal width
783
+ $block['width'] = max(0, min($width, 564));
784
+
785
+ $helper_wj_engine=WYSIJA::get('wj_engine','helper');
786
+ return base64_encode($helper_wj_engine->renderEditorBlock($block));
787
+ }
788
+
789
+ function install_theme() {
790
+ if( isset($_REQUEST['theme_id'])){
791
+ global $wp_version;
792
+ //check if theme is premium if you have the premium licence
793
+ if(isset($_REQUEST['premium']) && $_REQUEST['premium']){
794
+ $getpremiumtheme=apply_filters('wysija_install_theme_premium', false);
795
+
796
+ if(!$getpremiumtheme){
797
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
798
+ $themes = $helper_wj_engine->renderThemes();
799
+ return array('result'=>false, 'themes' => $themes);
800
+ }
801
+ }
802
+
803
+ $helperToolbox = WYSIJA::get('toolbox','helper');
804
+ $domain_name = $helperToolbox->_make_domain_name(admin_url('admin.php'));
805
+
806
+ $request = 'http://api.mailpoet.com/download/zip/'.$_REQUEST['theme_id'].'?domain='.$domain_name;
807
+
808
+ $args = array(
809
+ 'timeout' => 30,
810
+ 'body' => array( ),
811
+ 'user-agent' => 'WordPress/' . $wp_version . '; ' . get_bloginfo( 'url' )
812
+ );
813
+ $raw_response = wp_remote_post( $request, $args );
814
+
815
+ if ( is_wp_error( $raw_response ) || 200 != wp_remote_retrieve_response_code( $raw_response ) ){
816
+ if(method_exists($raw_response, 'get_error_messages')){
817
+ $this->error($raw_response->get_error_messages());
818
+ }
819
+ $ZipfileResult = false;
820
+ }else{
821
+ $ZipfileResult = maybe_unserialize( wp_remote_retrieve_body( $raw_response ) );
822
+ }
823
+
824
+ if($ZipfileResult === false){
825
+ $result = false;
826
+ $this->error(__('We were unable to contact the API, the site may be down. Please try again later.',WYSIJA),true);
827
+ }else{
828
+ $themesHelp=WYSIJA::get('themes','helper');
829
+ $result = $themesHelp->installTheme($ZipfileResult);
830
+
831
+ // refresh themes list
832
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
833
+ $themes = $helper_wj_engine->renderThemes();
834
+ }
835
+ }else{
836
+ $result = false;
837
+ $themes = '';
838
+ $this->notice('missing info');
839
+ }
840
+
841
+ return array('result' => $result, 'themes' => $themes);
842
+ }
843
+
844
+ function refresh_themes() {
845
+ // refresh themes list
846
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
847
+ return array('result'=>true, 'themes' => $helper_wj_engine->renderThemes());
848
+ }
849
+
850
+ function generate_auto_post() {
851
+ // get params and generate html
852
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
853
+ $helper_articles = WYSIJA::get('articles', 'helper');
854
+
855
+ // get parameters
856
+ $block_params = array();
857
+ if(isset($_POST['wysijaData'])) {
858
+ // store category ids (TBR)
859
+ $category_ids = array();
860
+
861
+ foreach($_POST['wysijaData'] as $pairs) {
862
+ // special cases
863
+ switch($pairs['name']) {
864
+ case 'author_label':
865
+ case 'category_label':
866
+ case 'readmore':
867
+ case 'nopost_message':
868
+ $block_params[] = array('key' => $pairs['name'], 'value' => base64_encode(stripslashes($pairs['value'])));
869
+ break;
870
+ case 'category_ids':
871
+ $category_ids = array_filter( array_map( 'absint', explode( ',', $pairs['value'] ) ) );
872
+ break;
873
+ default:
874
+ $block_params[] = array('key' => $pairs['name'], 'value' => $pairs['value']);
875
+ }
876
+ }
877
+
878
+ // make sure we have only unique ids in categories
879
+ $block_params[] = array('key' => 'category_ids', 'value' => join(',', array_unique($category_ids)));
880
+ }
881
+
882
+ if(empty($block_params)) {
883
+ // an error occurred, do something!
884
+ return false;
885
+ } else {
886
+ $data = array(
887
+ 'type' => 'auto-post',
888
+ 'params' => $block_params
889
+ );
890
+ return base64_encode($helper_wj_engine->renderEditorBlock($data));
891
+ }
892
+ }
893
+
894
+ function load_auto_post() {
895
+ $params = array();
896
+
897
+ if(isset($_POST['wysijaData'])) {
898
+
899
+ $pairs = explode('&', $_POST['wysijaData']);
900
+
901
+ foreach($pairs as $pair) {
902
+ list($key, $value) = explode('=', $pair);
903
+ switch($key) {
904
+ case 'autopost_count':
905
+ $params[$key] = (int)$value;
906
+ break;
907
+ case 'readmore':
908
+ case 'author_label':
909
+ case 'category_label':
910
+ case 'nopost_message':
911
+ $params[$key] = base64_decode($value);
912
+ break;
913
+ case 'exclude':
914
+ $params[$key] = explode(',', $value);
915
+ break;
916
+ default:
917
+ $params[$key] = $value;
918
+ }
919
+ }
920
+ }
921
+
922
+ if(empty($params)) {
923
+ // an error occurred, do something!
924
+ return false;
925
+ } else {
926
+
927
+ // get email params
928
+ $email_id = (int)$_REQUEST['id'];
929
+ $model_email = WYSIJA::get('email', 'model');
930
+ $email = $model_email->getOne(array('params','sent_at','campaign_id'), array('email_id' => $email_id));
931
+
932
+ $helper_articles = WYSIJA::get('articles', 'helper');
933
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
934
+
935
+ // see if posts have already been sent
936
+ if(!empty($email['params']['autonl']['articles']['ids'])) {
937
+ if(!isset($params['exclude'])) { $params['exclude'] = array(); }
938
+
939
+ $params['exclude'] = array_unique(array_merge($email['params']['autonl']['articles']['ids'], $params['exclude']));
940
+ }
941
+
942
+ //we set the post_date to filter articles only older than that one
943
+ if(isset($email['params']['autonl']['firstSend'])){
944
+ $params['post_date'] = $email['params']['autonl']['firstSend'];
945
+ }
946
+
947
+ // if immediate let it know to the get post
948
+ if(isset($email['params']['autonl']['articles']['immediatepostid'])){
949
+ $params['include'] = $email['params']['autonl']['articles']['immediatepostid'];
950
+ $params['post_limit'] = 1;
951
+ }else{
952
+ //we set the post_date to filter articles only older than the last time we sent articles
953
+ if(isset($email['params']['autonl']['lastSend'])){
954
+ $params['post_date'] = $email['params']['autonl']['lastSend'];
955
+ }else{
956
+ //get the latest child newsletter sent_at value
957
+ $mEmail=WYSIJA::get('email','model');
958
+ $mEmail->reset();
959
+ $mEmail->orderBy('email_id','DESC');
960
+ $lastEmailSent=$mEmail->getOne(false,array('campaign_id'=>$email['campaign_id'],'type'=>'1'));
961
+
962
+ if(!empty($lastEmailSent)) $params['post_date'] = $lastEmailSent['sent_at'];
963
+ }
964
+ }
965
+
966
+ // get posts
967
+ $model_wp_posts = WYSIJA::get('wp_posts','model');
968
+ $posts = $model_wp_posts->get_posts($params);
969
+
970
+ if(empty($posts)) {
971
+ // nothing to display
972
+ $posts = array();
973
+ } else {
974
+ // used to keep track of post ids present in the auto post
975
+ $post_ids = array();
976
+
977
+ // cleanup post and get image
978
+ foreach($posts as $key => $post) {
979
+ if($params['image_alignment'] !== 'none') {
980
+ // attempt to get post image
981
+ $posts[$key]['post_image'] = $helper_articles->getImage($post);
982
+ }
983
+
984
+ // store article id
985
+ $post_ids[] = $post['ID'];
986
+ }
987
+ // store article ids
988
+ $params['post_ids'] = join(',', $post_ids);
989
+ }
990
+
991
+ // get divider if necessary (for immediate post notification, the "show_divider" parameter is not available)
992
+ if(isset($params['show_divider']) && $params['show_divider'] === 'yes') {
993
+ if(isset($email['params']['divider'])) {
994
+ $params['divider'] = $email['params']['divider'];
995
+ } else {
996
+ $helper_dividers = WYSIJA::get('dividers', 'helper');
997
+ $params['divider'] = $helper_dividers->getDefault();
998
+ }
999
+ }
1000
+
1001
+ return base64_encode($helper_wj_engine->renderEditorAutoPost($posts, $params));
1002
+ }
1003
+ }
1004
+ }
controllers/ajax/config.php ADDED
@@ -0,0 +1,460 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_control_back_config extends WYSIJA_control{
4
+
5
+ function WYSIJA_control_back_config(){
6
+ if(!WYSIJA::current_user_can('wysija_config')) die("Action is forbidden.");
7
+ parent::WYSIJA_control();
8
+ }
9
+
10
+ function _displayErrors(){
11
+ if(version_compare(phpversion(), '5.4')>= 0){
12
+ error_reporting(E_ALL ^ E_STRICT);
13
+
14
+ }else{
15
+ error_reporting(E_ALL);
16
+ }
17
+ @ini_set("display_errors", 1);
18
+ }
19
+
20
+ function _hideErrors(){
21
+ error_reporting(0);
22
+ @ini_set('display_errors', '0');
23
+ }
24
+
25
+ function send_test_mail(){
26
+ $this->_displayErrors();
27
+ /*switch the send method*/
28
+ $configVal=$this->_convertPostedInarray();
29
+
30
+ /*send a test mail*/
31
+ $hEmail=WYSIJA::get('email','helper');
32
+ $res['result']=$hEmail->send_test_mail($configVal);
33
+
34
+ if($res['result']){
35
+ $modelConf=WYSIJA::get('config','model');
36
+ $modelConf->save(array('sending_emails_ok'=>$res['result']));
37
+ }
38
+ $this->_hideErrors();
39
+ return $res;
40
+ }
41
+
42
+ function send_test_mail_ms(){
43
+ $this->_displayErrors();
44
+ /*switch the send method*/
45
+ $configVal=$this->_convertPostedInarray();
46
+
47
+ /*send a test mail*/
48
+ $hEmail=WYSIJA::get('email','helper');
49
+ $res['result']=$hEmail->send_test_mail($configVal,true);
50
+ if($res['result']){
51
+ $modelConf=WYSIJA::get('config','model');
52
+ $modelConf->save(array('ms_sending_emails_ok'=>$res['result']));
53
+ }
54
+ //$this->_hideErrors();
55
+ return $res;
56
+ }
57
+
58
+ function bounce_connect(){
59
+
60
+ $configVal=$this->_convertPostedInarray();
61
+ /*try to connect to thebounce server*/
62
+ $bounceClass=WYSIJA::get('bounce','helper');
63
+ $bounceClass->report = true;
64
+ $res['result']=false;
65
+ if($bounceClass->init($configVal)){
66
+ if($bounceClass->connect()){
67
+ $nbMessages = $bounceClass->getNBMessages();
68
+ $this->notice(sprintf(__('Successfully connected to %1$s',WYSIJA),$bounceClass->config->getValue('bounce_login')));
69
+ $this->notice(sprintf(__('There are %1$s messages in your mailbox',WYSIJA),$nbMessages));
70
+ $bounceClass->close();
71
+ if((int)$nbMessages >0) $res['result']=true;
72
+ else $this->notice(sprintf(__('There are no bounced messages to process right now!',WYSIJA),$nbMessages));
73
+ if(!empty($nbMessages)){
74
+ //$app->enqueueMessage('<a class="modal" style="text-decoration:blink" rel="{handler: \'iframe\', size: {x: 640, y: 480}}" href="'.acymailing_completeLink("bounces&task=process",true ).'">'.__("CLICK HERE to handle the messages",WYSIJA).'</a>');
75
+ }
76
+ }else{
77
+ $errors = $bounceClass->getErrors();
78
+ if(!empty($errors)){
79
+ $this->error($errors,true);
80
+ $errorString = implode(' ',$errors);
81
+ $port = $bounceClass->config->getValue('bounce_port','');
82
+ if(preg_match('#certificate#i',$errorString) && !$bounceClass->config->getValue('bounce_selfsigned',false)){
83
+ $this->notice(__('You may need to turn ON the option <i>Self-signed certificates</i>', WYSIJA));
84
+ }elseif(!empty($port) AND !in_array($port,array('993','143','110'))){
85
+ $this->notice(__('Are you sure you selected the right port? You can leave it empty if you do not know what to specify',WYSIJA));
86
+ }
87
+ }
88
+ }
89
+ }
90
+
91
+
92
+ return $res;
93
+ }
94
+
95
+ /**
96
+ * processing the bounce manually through the config
97
+ * @return type
98
+ */
99
+ function bounce_process(){
100
+ // bounce handling
101
+ $helper_bounce = WYSIJA::get('bounce','helper');
102
+
103
+ // in a multisite case we process first the bounce recording into the bounce table
104
+ if(is_multisite()){
105
+ $helper_bounce->record_bounce_ms();
106
+
107
+ // then we take actions from what has been returned by the bounce
108
+ return $helper_bounce->process_bounce_ms();
109
+ }else{
110
+ return $helper_bounce->process_bounce();
111
+ }
112
+ }
113
+
114
+ function linkignore(){
115
+ $this->_displayErrors();
116
+
117
+ $modelConf=WYSIJA::get('config','model');
118
+
119
+ $ignore_msgs=$modelConf->getValue('ignore_msgs');
120
+ if(!$ignore_msgs) $ignore_msgs=array();
121
+
122
+ $ignore_msgs[$_REQUEST['ignorewhat']]=1;
123
+ $modelConf->save(array('ignore_msgs'=>$ignore_msgs));
124
+
125
+ $res['result']=true;
126
+ $this->_hideErrors();
127
+ return $res;
128
+ }
129
+
130
+ // Ajax called function to enable analytics sharing from welcome page.
131
+ function share_analytics() {
132
+ $this->_displayErrors();
133
+
134
+ $model_config = WYSIJA::get('config','model');
135
+ $model_config->save(array('analytics' => 1));
136
+
137
+ $res['result'] = true;
138
+ $this->_hideErrors();
139
+ return $res;
140
+ }
141
+
142
+ function validate(){
143
+ $helpLic=WYSIJA::get('licence','helper');
144
+ $res=$helpLic->check();
145
+
146
+ if(!isset($res['result'])) $res['result']=false;
147
+ return $res;
148
+ }
149
+
150
+ function devalidate(){
151
+
152
+ $modelCOnfig=WYSIJA::get('config','model');
153
+ $res=$modelCOnfig->save(array('premium_key'=>false));
154
+
155
+ if(!isset($res['result'])) $res['result']=false;
156
+ return $res;
157
+ }
158
+
159
+
160
+
161
+ function _convertPostedInarray(){
162
+ $_POST = stripslashes_deep($_POST);
163
+ $dataTemp=$_POST['data'];
164
+ $_POST['data']=array();
165
+ foreach($dataTemp as $val) $_POST['data'][$val['name']]=$val['value'];
166
+ $dataTemp=null;
167
+ foreach($_POST['data'] as $k =>$v){
168
+ $newkey=str_replace(array('wysija[config][',']'),'',$k);
169
+ $configVal[$newkey]=$v;
170
+ }
171
+ return $configVal;
172
+ }
173
+
174
+ // WYSIJA Form Editor
175
+ function wysija_form_generate_template() {
176
+ $data = $this->_wysija_form_get_data();
177
+
178
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
179
+ return base64_encode($helper_form_engine->render_editor_template($data));
180
+ }
181
+
182
+ function wysija_form_manage_field() {
183
+ $response = array('result' => true, 'error' => null);
184
+
185
+ // get data
186
+ $data = $this->_wysija_form_get_data();
187
+ $form_id = (int)$_REQUEST['form_id'];
188
+
189
+ // check for required fields
190
+ if(!isset($data['type']) || isset($data['type']) && strlen(trim($data['type'])) === 0) {
191
+ $response['error'] = __('You need to select a type for this field', WYSIJA);
192
+ $response['result'] = false;
193
+ }
194
+ if(!isset($data['name']) || isset($data['name']) && strlen(trim($data['name'])) === 0) {
195
+ $response['error'] = __('You need to specify a name for this field', WYSIJA);
196
+ $response['result'] = false;
197
+ }
198
+
199
+ // only proceed if there is no error
200
+ if($response['error'] === null) {
201
+ $is_required = (isset($data['params']['required']) ? WJ_Utils::to_bool($data['params']['required']) : false);
202
+
203
+ if(isset($data['field_id']) && (int)$data['field_id'] > 0) {
204
+ // it's an update
205
+ $custom_field = WJ_Field::get($data['field_id']);
206
+
207
+ if($custom_field !== NULL) {
208
+ $data['params'] = array_merge($custom_field->settings, $data['params']);
209
+ // update fields
210
+ $custom_field->name = $data['name'];
211
+ $custom_field->type = $data['type'];
212
+ $custom_field->required = $is_required;
213
+ $custom_field->settings = $data['params'];
214
+ $custom_field->save();
215
+ } else {
216
+ // throw error if field does not exist
217
+ $response['error'] = __('This field does not exist', WYSIJA);
218
+ $response['result'] = false;
219
+ }
220
+ } else {
221
+ // create new custom field
222
+ $custom_field = new WJ_Field();
223
+ $custom_field->set(array(
224
+ 'name' => $data['name'],
225
+ 'type' => $data['type'],
226
+ 'required' => $is_required,
227
+ 'settings' => $data['params']
228
+ ));
229
+ $custom_field->save();
230
+ }
231
+
232
+ if($response['error'] === null) {
233
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
234
+
235
+ // need to update each block instance of this custom field
236
+ $block = $helper_form_engine->refresh_custom_field($form_id, array(
237
+ 'name' => $data['name'],
238
+ 'field' => $custom_field->user_column_name(),
239
+ 'type' => $data['type'],
240
+ 'required' => $is_required,
241
+ 'settings' => $data['params']
242
+ ));
243
+
244
+ // render editor toolbar & templates
245
+ $response['data'] = array(
246
+ 'toolbar' => base64_encode($helper_form_engine->render_editor_toolbar()),
247
+ 'templates' => base64_encode($helper_form_engine->render_editor_templates())
248
+ );
249
+
250
+ if($block !== null) {
251
+ // refresh block using this custom field in the current form
252
+ $block_template = $helper_form_engine->render_editor_template($block);
253
+
254
+ if($block_template !== null) {
255
+ $response['data']['block'] = base64_encode($block_template);
256
+ }
257
+ }
258
+ }
259
+ }
260
+
261
+ return $response;
262
+ }
263
+
264
+ // get wysija form data from post (auto decoding)
265
+ private function _wysija_form_get_data() {
266
+ if(isset($_POST['wysijaData'])) {
267
+ // decode the data string
268
+ $decoded_data = base64_decode($_POST['wysijaData']);
269
+
270
+ // avoid using stripslashes as it's not reliable depending on the magic quotes settings
271
+ $json_data = str_replace('\"', '"', $decoded_data);
272
+ return json_decode($json_data, true);
273
+ }
274
+ return array();
275
+ }
276
+
277
+ // remove a custom field
278
+ function form_field_delete() {
279
+ $data = $this->_wysija_form_get_data();
280
+
281
+ // check for field_id parameter
282
+ if(isset($data['field_id']) && (int)$data['field_id'] > 0) {
283
+ // get custom field by id
284
+ $custom_field = WJ_Field::get($data['field_id']);
285
+
286
+ // if the custom field exists
287
+ if($custom_field !== null) {
288
+ // we need to remove the field in any form
289
+ // get all forms
290
+ $model_forms = WYSIJA::get('forms', 'model');
291
+ $forms = $model_forms->getRows();
292
+
293
+ // get custom field name
294
+ $field_name = $custom_field->user_column_name();
295
+ if(is_array($forms) && count($forms) > 0) {
296
+ // loop through each form
297
+ foreach ($forms as $i => $form) {
298
+ $requires_update = false;
299
+
300
+ // decode form data
301
+ $data = unserialize(base64_decode($form['data']));
302
+
303
+ // loop through each block
304
+ foreach($data['body'] as $j => $block) {
305
+ // in case we find a text block
306
+ if($block['field'] === $field_name) {
307
+ unset($data['body'][$j]);
308
+ // flag form to be updated
309
+ $requires_update = true;
310
+ }
311
+ }
312
+
313
+ // if the form requires update, let's do it
314
+ if($requires_update === true) {
315
+ $model_forms->reset();
316
+ $model_forms->update(array('data' => base64_encode(serialize($data))), array('form_id' => (int)$form['form_id']));
317
+ }
318
+ }
319
+ }
320
+
321
+ // delete custom field
322
+ $custom_field->delete();
323
+ }
324
+ }
325
+ }
326
+
327
+ function form_name_save() {
328
+ // get name from post and stripslashes it
329
+ $form_name = trim(stripslashes($_POST['name']));
330
+ // get form_id from post
331
+ $form_id = (int)$_POST['form_id'];
332
+
333
+ if(strlen($form_name) > 0 && $form_id > 0) {
334
+ // update the form name within the database
335
+ $model_forms = WYSIJA::get('forms', 'model');
336
+ $model_forms->update(array('name' => $form_name), array('form_id' => $form_id));
337
+ }
338
+ return array('name' => $form_name);
339
+ }
340
+
341
+ function form_save() {
342
+ // get form id
343
+ $form_id = null;
344
+ if(isset($_POST['form_id']) && (int)$_POST['form_id'] > 0) {
345
+ $form_id = (int)$_POST['form_id'];
346
+ }
347
+
348
+ // decode json data and convert to array
349
+ $raw_data = null;
350
+ if(isset($_POST['wysijaData'])) {
351
+ // decode the data string
352
+ $decoded_data = base64_decode($_POST['wysijaData']);
353
+
354
+ // avoid using stripslashes as it's not reliable depending on the magic quotes settings
355
+ $json_data = str_replace('\"', '"', $decoded_data);
356
+ // decode JSON data
357
+ $raw_data = json_decode($json_data, true);
358
+ }
359
+
360
+ if($form_id === null or $raw_data === null) {
361
+ $this->error('Error saving', false);
362
+ return array('result' => false);
363
+ } else {
364
+
365
+ // flag to see if the user can select his own lists
366
+ $has_list_selection = false;
367
+ $raw_data['settings']['lists_selected_by'] = 'admin';
368
+
369
+ // special case for block params, as we base64_encode the values and serialize arrays, so let's decode it before saving it
370
+ foreach($raw_data['body'] as $block_id => $block) {
371
+ if(isset($block['params']) && !empty($block['params'])) {
372
+ $params = array();
373
+
374
+ foreach($block['params'] as $key => $value) {
375
+ $value = base64_decode($value);
376
+ if(is_serialized($value) === true) {
377
+ $value = unserialize($value);
378
+ }
379
+ $params[$key] = $value;
380
+ }
381
+
382
+ if(!empty($params)) {
383
+ $raw_data['body'][$block_id]['params'] = $params;
384
+ }
385
+ }
386
+ // special case when the list selection widget is present
387
+ if($block['type'] === 'list') {
388
+ $has_list_selection = true;
389
+
390
+ $lists = array();
391
+ foreach($params['values'] as $list) {
392
+ $lists[] = (int)$list['list_id'];
393
+ }
394
+
395
+ // override lists in form settings
396
+ $raw_data['settings']['lists'] = $lists;
397
+ $raw_data['settings']['lists_selected_by'] = 'user';
398
+ }
399
+ }
400
+
401
+ // make sure the lists parameter is an array, otherwise it's not gonna work for a single list
402
+ if($has_list_selection === false) {
403
+ if(!is_array($raw_data['settings']['lists'])) {
404
+ $raw_data['settings']['lists'] = array((int)$raw_data['settings']['lists']);
405
+ }
406
+ }
407
+
408
+ // set form id into data so we can track who subscribed through it
409
+ $raw_data['form_id'] = $form_id;
410
+
411
+ // set data in form engine so we can generate the render the web version
412
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
413
+ $helper_form_engine->set_data($raw_data);
414
+
415
+ // check if the form has already been inserted in a widget and therefore display different success message
416
+ $widgets = get_option('widget_wysija');
417
+ $is_form_added_as_widget = false;
418
+ if($widgets !== false) {
419
+ foreach($widgets as $widget) {
420
+ if(is_array($widget) && isset($widget['form']) && (int)$widget['form'] === $form_id) {
421
+ $is_form_added_as_widget = true;
422
+ }
423
+
424
+ }
425
+ }
426
+ if($is_form_added_as_widget === true) {
427
+ $save_message = __('Saved! The changes are already active in your widget.', WYSIJA);
428
+ } else {
429
+ $save_message = str_replace(array(
430
+ '[link_widget]',
431
+ '[/link_widget]'
432
+ ), array(
433
+ '<a href="'.admin_url('widgets.php').'" target="_blank">',
434
+ '</a>'
435
+ ),
436
+ __('Saved! Add this form to [link_widget]a widget[/link_widget].', WYSIJA)
437
+ );
438
+ }
439
+
440
+ // update form data in DB
441
+ $model_forms = WYSIJA::get('forms', 'model');
442
+ $model_forms->reset();
443
+ $result = $model_forms->update(array(
444
+ // get encoded data to store it in the database
445
+ 'data' => $helper_form_engine->get_encoded('data')
446
+ ), array('form_id' => $form_id));
447
+
448
+ // return response depending on db save result
449
+ if(!$result) {
450
+ // throw error
451
+ $this->error(__('Your form could not be saved.', WYSIJA));
452
+ } else {
453
+ // save successful
454
+ $this->notice(__('Your form has been saved.', WYSIJA));
455
+ }
456
+ }
457
+
458
+ return array('result' => $result, 'save_message' => base64_encode($save_message), 'exports' => base64_encode($helper_form_engine->render_editor_export($form_id)));
459
+ }
460
+ }
controllers/ajax/index.html ADDED
File without changes
controllers/ajax/statistics.php ADDED
@@ -0,0 +1,95 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ require_once(WYSIJA_CORE.'module'.DS.'statistics.php'); // @todo
5
+
6
+ class WYSIJA_control_back_statistics extends WYSIJA_control
7
+ {
8
+
9
+ /**
10
+ * Main model of this controller
11
+ * @var string
12
+ */
13
+ public $model = 'statistics';
14
+
15
+ /**
16
+ * Main view of this controller
17
+ * @var string
18
+ */
19
+ public $view = 'statistics';
20
+
21
+ /**
22
+ * Render a hook of a specific module
23
+ * @return string
24
+ */
25
+ public function get_block()
26
+ {
27
+ if (empty($_REQUEST['block']))
28
+ return '';
29
+ $module = $_REQUEST['block'];
30
+ $hook_name = 'hook_stats';
31
+ return apply_filters('custom_module_hook', '', $module, $hook_name, $this->get_post_params());
32
+ }
33
+
34
+ protected function get_post_params()
35
+ {
36
+ $params = array( );
37
+ $order_by = !empty($_REQUEST['filter']['orderBy']) ? $_REQUEST['filter']['orderBy'] : null;
38
+ switch (strtolower($order_by))
39
+ {
40
+ case 'sent':
41
+ $order_by = WYSIJA_module_statistics::ORDER_BY_SENT;
42
+ break;
43
+ case 'open':
44
+ $order_by = WYSIJA_module_statistics::ORDER_BY_OPEN;
45
+ break;
46
+ case 'click':
47
+ $order_by = WYSIJA_module_statistics::ORDER_BY_CLICK;
48
+ break;
49
+ case 'unsubscribe':
50
+ $order_by = WYSIJA_module_statistics::ORDER_BY_UNSUBSCRIBE;
51
+ break;
52
+ default:
53
+ $order_by = null;
54
+ break;
55
+ }
56
+ $order_direction = !empty($_REQUEST['filter']['orderDirection']) ? $_REQUEST['filter']['orderDirection'] : null;
57
+ switch (strtolower($order_direction))
58
+ {
59
+ case 'asc':
60
+ $order_direction = WYSIJA_module_statistics::ORDER_DIRECTION_ASC;
61
+ break;
62
+ case 'desc':
63
+ default:
64
+ $order_direction = WYSIJA_module_statistics::ORDER_DIRECTION_DESC;
65
+ break;
66
+ }
67
+ $params['top'] = !empty($_REQUEST['filter']['itemPerPage']) ? (int)$_REQUEST['filter']['itemPerPage'] : WYSIJA_module_statistics::DEFAULT_TOP_RECORDS;
68
+ $params['from'] = !empty($_REQUEST['filter']['from']) ? $_REQUEST['filter']['from'] : null;
69
+ $params['to'] = !empty($_REQUEST['filter']['to']) ? $_REQUEST['filter']['to'] : null;
70
+ $params['order_by'] = $order_by;
71
+ $params['order_direction'] = $order_direction;
72
+
73
+ // this doesn't work when php is less than 5.3, this is the case on my host (ben) which is very popular in France, SPain and UK
74
+ // we cannot use functions from php 5.3
75
+ if (function_exists('date_diff'))
76
+ {
77
+ $this->data['date_interval'] = date_diff(date_create($params['from']), date_create($params['to']));
78
+ }
79
+ else
80
+ {
81
+ $duration = strtotime($params['to']) - strtotime($params['from']);
82
+ $helper_toolbox = WYSIJA::get('toolbox', 'helper');
83
+ $this->data['date_interval'] = (object)$helper_toolbox->convert_seconds_to_array($duration, false);
84
+ }
85
+ $params['group_by'] = ( $this->data['date_interval']->days == 0 || $this->data['date_interval']->days > WYSIJA_module_statistics::SWITCHING_DATE_TO_MONTH_THRESHOLD) ?
86
+ WYSIJA_module_statistics::GROUP_BY_MONTH :
87
+ WYSIJA_module_statistics::GROUP_BY_DATE; // $date_interval->days == 0, means, no begin date, no end date
88
+
89
+ // Hack!
90
+ $_REQUEST['limit_pp'] = $params['top'];// Pagination, mark current selected value
91
+
92
+ return $params;
93
+ }
94
+
95
+ }
controllers/ajax/subscribers.php ADDED
@@ -0,0 +1,131 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ include(dirname(dirname(__FILE__)).DS.'front.php');
4
+ class WYSIJA_control_back_subscribers extends WYSIJA_control_front{
5
+ var $model='user';
6
+ var $view='';
7
+
8
+ function WYSIJA_control_back_subscribers(){
9
+ parent::WYSIJA_control_front();
10
+ $data=array();
11
+ foreach($_REQUEST['data'] as $vals){
12
+ $data[$vals['name']]=$vals['value'];
13
+ }
14
+ if(isset($data['message_success'])){
15
+ $this->messages['insert'][true]=$data['message_success'];
16
+ }else{
17
+ $this->messages['insert'][true]=__('User has been inserted.',WYSIJA);
18
+ }
19
+
20
+ $this->messages['insert'][false]=__('User has not been inserted.',WYSIJA);
21
+ $this->messages['update'][true]=__('User has been updated.',WYSIJA);
22
+ $this->messages['update'][false]=__('User has not been updated.',WYSIJA);
23
+ }
24
+
25
+ function save(){
26
+ $datarequested=array();
27
+ $i=0;
28
+ foreach($_REQUEST['data'] as $vals){
29
+ if($vals['name']=='wysija[user_list][list_id][]'){
30
+ $datarequested[str_replace('wysija[user_list][list_id][]', 'wysija[user_list][list_id]['.$i.']', $vals['name'])]=$vals['value'];
31
+ $i++;
32
+ }else $datarequested[$vals['name']]=$vals['value'];
33
+ }
34
+
35
+ $data=$this->convertUserData($datarequested);
36
+
37
+ $helperUser=WYSIJA::get('user','helper');
38
+ if(!$helperUser->checkData($data))return false;
39
+ $user_id = $helperUser->addSubscriber($data);
40
+
41
+ if((int)$user_id > 0) {
42
+ // Handle custom fields
43
+ if(isset($data['user_field'])) {
44
+ WJ_FieldHandler::handle_all($data['user_field'], $user_id);
45
+ }
46
+ }
47
+
48
+ return true;
49
+ }
50
+
51
+ // REFACTOR: Insanely complicated and inefficient
52
+ function convertUserData($datarequested){
53
+ $data=array();
54
+
55
+ //get the lists
56
+ if(isset($datarequested['wysija[user_list][list_ids]'])){
57
+ $listids=explode(',',$datarequested['wysija[user_list][list_ids]']);
58
+ $subdate=time();
59
+ unset($datarequested['wysija[user_list][list_ids]']);
60
+ }else{
61
+ $i=0;
62
+ $listids=array();
63
+ for ($i = 0; $i <= 25; $i++) {
64
+ $testkey='wysija[user_list][list_id]['.$i.']';
65
+ if(isset($datarequested[$testkey])) $listids[]=$datarequested[$testkey];
66
+ unset($datarequested[$testkey]);
67
+ }
68
+ }
69
+ $data['user_list']['list_ids']=$listids;
70
+
71
+ // define array for custom user fields
72
+ $data['user_field'] = array();
73
+
74
+ //get the user info and the rest of the data posted
75
+ foreach($datarequested as $key => $val){
76
+ if(strpos($key, 'wysija[user]')!== false) {
77
+ $keymodified=str_replace(array('wysija[','][',']'),array('','#',''),$key);
78
+ $keystabcol=explode('#',$keymodified);
79
+ switch(count($keystabcol)){
80
+ case 2:
81
+ $data[$keystabcol[0]][$keystabcol[1]]=$val;
82
+ break;
83
+ case 3:
84
+ $data[$keystabcol[0]][$keystabcol[1]][$keystabcol[2]]=$val;
85
+ break;
86
+ }
87
+ } else if(strpos($key, 'wysija[field]')!== false){
88
+ $keymodified=str_replace(array('wysija[','][',']'),array('','#',''),$key);
89
+ $keystabcol=explode('#',$keymodified);
90
+
91
+ switch(count($keystabcol)){
92
+ case 2:
93
+ $data['user_field'][$keystabcol[1]] = $val;
94
+ break;
95
+ case 3:
96
+ $data['user_field'][$keystabcol[1]][$keystabcol[2]] = $val;
97
+ break;
98
+ }
99
+ } else {
100
+ if(!isset($data[$key])) $data[$key]=$val;
101
+ }
102
+ }
103
+
104
+ return $data;
105
+ }
106
+
107
+ function registerToLists($data,$uid){
108
+ $model=WYSIJA::get('user_list','model');
109
+ if(isset($data['wysija[user_list][list_ids]'])){
110
+ $listids=explode(',',$data['wysija[user_list][list_ids]']);
111
+
112
+ $subdate=time();
113
+
114
+
115
+ }else{
116
+ $i=0;
117
+ $listids=array();
118
+ for ($i = 0; $i <= 25; $i++) {
119
+ $testkey='wysija[user_list][list_id]['.$i.']';
120
+ if(isset($data[$testkey])) $listids[]=$data[$testkey];
121
+ }
122
+
123
+ //$listids=$data['wysija[user_list][list_id]'];
124
+ }
125
+ foreach($listids as $listid){
126
+ $model->replace(array('list_id'=>$listid,'user_id'=>$uid,'sub_date'=>$subdate));
127
+ $model->reset();
128
+ }
129
+
130
+ }
131
+ }
controllers/ajax/tmce.php ADDED
@@ -0,0 +1,139 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_control_back_tmce extends WYSIJA_control{
4
+
5
+ function WYSIJA_control_back_tmce(){
6
+ parent::WYSIJA_control();
7
+ $this->viewObj=WYSIJA::get('tmce','view');
8
+ }
9
+
10
+ function subscribersAdd(){
11
+ $this->viewObj->title=__('Insert Total of Subscribers',WYSIJA);
12
+ $data = array();
13
+ $data['lists'] = WYSIJA::get('list','model')->getLists();
14
+ foreach ($data['lists'] as $k => $list){
15
+ if (empty($list['is_enabled']) OR empty($list['is_public'])) unset($data['lists'][$k]);
16
+ }
17
+ $data['confirm_dbleoptin'] = WYSIJA::get('config','model')->getValue('confirm_dbleoptin');
18
+
19
+ $this->viewObj->subscribersAdd($data);
20
+ exit;
21
+ }
22
+
23
+ function subscribersEdit(){
24
+
25
+ }
26
+
27
+ function registerAdd(){
28
+ $this->viewObj->title=__('Insert Subscription Form',WYSIJA);
29
+
30
+ $this->viewObj->registerAdd($this->getData());
31
+ exit;
32
+ }
33
+
34
+ function registerEdit(){
35
+ $this->viewObj->title=__('Insert Subscription Form',WYSIJA);
36
+
37
+ $this->viewObj->registerAdd($this->getData(),true);
38
+ exit;
39
+ }
40
+
41
+
42
+ function getData(){
43
+ $data_widget=array();
44
+
45
+ // if this parameter is passed that means we have an old widget let's import it and select
46
+ if(isset($_REQUEST['widget-data64'])){
47
+ $data_widget=unserialize(base64_decode($_REQUEST['widget-data64']));
48
+ $post=get_post($_REQUEST['post_id']);
49
+
50
+ // we need a title to identify the form
51
+ $data_widget['title'] = 'Form on '.$post->post_type.': '.$post->post_title;
52
+
53
+ $model_forms = WYSIJA::get('forms', 'model');
54
+ $model_forms->reset();
55
+ $form = $model_forms->getOne(false,array('name' => $data_widget['title']));
56
+
57
+ // this form doesn't exist yet in the new format so let's try to import it
58
+ if(empty($form)){
59
+ $helper_update=WYSIJA::get('update','helper');
60
+
61
+ $form_id = $helper_update->convert_widget_to_form($data_widget);
62
+ if($form_id!==false) {
63
+ $data_widget['default_form'] = $form_id;
64
+ }
65
+ }
66
+ }
67
+
68
+ if(isset($_POST['widget-wysija'])){
69
+ $data_widget=array('widget_id'=>$_POST['widget_id'],'preview'=>true);
70
+ $custom_fields=array();
71
+
72
+ foreach($_POST['widget-wysija'] as $arra){
73
+ foreach($arra as $k => $v) {
74
+ switch($k){
75
+ case 'lists':
76
+ if(isset($data_widget[$k])) $data_widget[$k][]=$v[0];
77
+ else $data_widget[$k]=array($v[0]);
78
+ break;
79
+ case 'lists_name':
80
+ foreach($v as $kv=>$vv){
81
+ if(isset($data_widget[$k])) $data_widget[$k][$kv]=$vv;
82
+ else $data_widget[$k]=array($kv=>$vv);
83
+ }
84
+
85
+ break;
86
+ case 'customfields':
87
+
88
+ $found=false;
89
+ foreach($custom_fields as $keycol => $params){
90
+ if(isset($v[$keycol])){
91
+ $custom_fields[$keycol]= array_merge($custom_fields[$keycol],$v[$keycol]);
92
+ $found=true;
93
+ }
94
+ }
95
+
96
+ if(!$found) $custom_fields=array_merge($custom_fields,$v);
97
+ break;
98
+
99
+ default:
100
+ $data_widget[$k]=stripslashes($v);
101
+ }
102
+
103
+ }
104
+ }
105
+
106
+
107
+ $count=count($custom_fields);
108
+ if($count>1){
109
+ foreach($custom_fields as $keycol=>$paraval){
110
+ if($keycol!='email' && !isset($paraval['column_name'])) unset($custom_fields[$keycol]);
111
+ }
112
+ }
113
+
114
+ $count=count($custom_fields);
115
+ if($count>0){
116
+ if(!isset($custom_fields['email'])){
117
+ $custom_fields['email']['column_name']='email';
118
+ $custom_fields['email']['label']=__('Email',WYSIJA);
119
+ }
120
+ }
121
+
122
+ $count=count($custom_fields);
123
+ if($count==1 && isset($custom_fields['email'])){
124
+
125
+ $custom_fields=array();
126
+ }
127
+
128
+ if($custom_fields) $data_widget['customfields']=$custom_fields;
129
+
130
+ }
131
+
132
+ if(!isset($data_widget['customfields']) && isset($data_widget['labelswithin']) && $data_widget['labelswithin']=='labels_within'){
133
+ $data_widget['customfields']=array('email'=>array('label'=>__('Email',WYSIJA)));
134
+ }
135
+
136
+ return $data_widget;
137
+ }
138
+
139
+ }
controllers/back.php ADDED
@@ -0,0 +1,790 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ global $wysi_location;
4
+ class WYSIJA_control_back extends WYSIJA_control{
5
+ var $redirectAfterSave=true;
6
+ var $searchable=array();
7
+ var $data=array();
8
+ var $jsTrans=array();
9
+ var $msgOnSave=true;
10
+ var $pref=array();
11
+ var $statuses=array();
12
+ var $viewShow=null;
13
+ var $_affected_rows = 0; //affected rows by batch select
14
+ var $target_action_form = ''; // default target of a form
15
+
16
+ function WYSIJA_control_back(){
17
+ parent::WYSIJA_control();
18
+ global $wysija_msg,$wysija_queries,$wysija_queries_errors;
19
+ $wysija_msgTemp=get_option('wysija_msg');
20
+
21
+ if(is_array($wysija_msgTemp) && count($wysija_msgTemp)>0){
22
+ $wysija_msg=$wysija_msgTemp;
23
+ }
24
+
25
+ $wysija_qryTemp=get_option('wysija_queries');
26
+ $wysija_qryErrors=get_option('wysija_queries_errors');
27
+ if(is_array($wysija_qryTemp) && count($wysija_qryTemp)>0){
28
+ $wysija_queries=$wysija_qryTemp;
29
+ }
30
+
31
+ if(is_array($wysija_qryErrors) && count($wysija_qryErrors)>0){
32
+ $wysija_queries_errors=$wysija_qryErrors;
33
+ }
34
+
35
+ WYSIJA::update_option('wysija_queries','');
36
+ WYSIJA::update_option('wysija_queries_errors','');
37
+ WYSIJA::update_option('wysija_msg','');
38
+ global $wysija_installing;
39
+ if($wysija_installing===true) return;
40
+ $this->pref=get_user_meta(WYSIJA::wp_get_userdata('ID'),'wysija_pref',true);
41
+
42
+ $prefupdate=false;
43
+ if($this->pref) {
44
+ $prefupdate=true;
45
+ $this->pref=unserialize(base64_decode($this->pref));
46
+ }else{
47
+ $this->pref=array();
48
+ }
49
+
50
+ if(!isset($_GET['action'])) $action='default';
51
+ else $action=$_GET['action'];
52
+
53
+ if(isset($_REQUEST['limit_pp'])){
54
+ $this->pref[$_REQUEST['page']][$action]['limit_pp']=$_REQUEST['limit_pp'];
55
+ }
56
+
57
+ if($this->pref && isset($_REQUEST['page']) && $_REQUEST['page'] && isset($this->pref[$_REQUEST['page']][$action]['limit_pp'])){
58
+ $this->viewObj->limit_pp=$this->pref[$_REQUEST['page']][$action]['limit_pp'];
59
+ $this->modelObj->limit_pp=$this->pref[$_REQUEST['page']][$action]['limit_pp'];
60
+ }
61
+
62
+ if($prefupdate){
63
+ update_user_meta(WYSIJA::wp_get_userdata('ID'),'wysija_pref',base64_encode(serialize($this->pref)));
64
+ }else{
65
+ add_user_meta(WYSIJA::wp_get_userdata('ID'),'wysija_pref',base64_encode(serialize($this->pref)));
66
+ }
67
+ add_action('wysija_various_check',array($this,'variousCheck'));
68
+ do_action('wysija_various_check');
69
+
70
+ $this->target_action_form = $this->data['target_action_form'] = $this->_getTargetActionForm(); // set default value of any form action
71
+ }
72
+
73
+
74
+ function variousCheck(){
75
+ $model_config=WYSIJA::get('config','model');
76
+ if(get_option('wysicheck')){
77
+ $helper_licence=WYSIJA::get('licence','helper');
78
+ $result=$helper_licence->check(true);
79
+ if($result['nocontact']){
80
+ // redirect instantly to a page with a javascript file where we check the domain is ok
81
+ $data=get_option('wysijey');
82
+ // remotely connect to host
83
+ wp_enqueue_script('wysija-verif-licence', 'http://www.mailpoet.com/?wysijap=checkout&wysijashop-page=1&controller=customer&action=checkDomain&js=1&data='.$data, array( 'jquery' ), time());
84
+ }
85
+ }
86
+
87
+ // check if the name of the site or the upload folder has changed of name :
88
+ if(WYSIJA_UPLOADS_URL!=$model_config->getValue('uploadurl')){
89
+
90
+ }
91
+
92
+ }
93
+
94
+
95
+ function errorInstall(){
96
+ $this->viewObj->renderErrorInstall();
97
+ }
98
+
99
+ function _resetGlobMsg(){
100
+ global $wysija_msg,$wysija_queries,$wysija_queries_errors;
101
+
102
+ $wysija_msg=$wysija_queries=$wysija_queries_errors=array();
103
+ }
104
+ function defaultDisplay(){
105
+ $this->viewShow=$this->action='main';
106
+
107
+ /* if it has not been enqueud in the head we print it here(can happens based on the action after a save or so)*/
108
+ $this->js[]='wysija-admin-list';
109
+
110
+ /*get the filters*/
111
+ if(isset($_REQUEST['search']) && $_REQUEST['search']){
112
+ $this->filters['like']=array();
113
+ foreach($this->searchable as $searchable){
114
+ $this->filters['like'][$searchable]=$_REQUEST['search'];
115
+ }
116
+
117
+ }
118
+
119
+ if($this->filters){
120
+ $this->modelObj->setConditions($this->filters);
121
+ }
122
+
123
+ if($this->joins){
124
+ $this->modelObj->setJoin($this->joins);
125
+ }
126
+
127
+ if($this->statuses){
128
+ //we count by statuses
129
+ $query='SELECT count('.$this->modelObj->pk.') as count, status FROM `[wysija]'.$this->modelObj->table_name.'` GROUP BY status';
130
+ $countss=$this->modelObj->query('get_res',$query,ARRAY_A);
131
+ $counts=array();
132
+ $this->modelObj->countRows=0;
133
+
134
+ foreach($countss as $count){
135
+ $mystat=(int)$count['status'];
136
+
137
+ $this->statuses[$mystat]['count']=$count['count'];
138
+ $this->statuses[$mystat]['uri']=$this->getDefaultUrl(false).'&link_filter='.$this->statuses[$mystat]['key'];
139
+
140
+ $this->modelObj->countRows=$this->modelObj->countRows+$count['count'];
141
+ $this->viewObj->statuses=$this->statuses;
142
+ }
143
+
144
+ }else{
145
+ $this->modelObj->countRows=$this->modelObj->count();
146
+ }
147
+
148
+
149
+
150
+
151
+ if(isset($_REQUEST['orderby'])){
152
+ $this->modelObj->orderBy($_REQUEST['orderby'],strtoupper($_REQUEST['ordert']));
153
+ }else{
154
+ $this->modelObj->orderBy($this->modelObj->getPk(),'DESC');
155
+ }
156
+ $this->modelObj->limitON=true;
157
+
158
+ $data=$this->modelObj->getRows($this->list_columns);
159
+
160
+ $methodDefaultData='defaultData';
161
+ if(method_exists($this,$methodDefaultData )){
162
+ $this->$methodDefaultData($data);
163
+ }
164
+
165
+ }
166
+
167
+ function defaultData($data){
168
+ $this->data=$data;
169
+ }
170
+
171
+
172
+ function render(){
173
+
174
+ $this->viewObj->render($this->viewShow,$this->data);
175
+ }
176
+
177
+ /**
178
+ * by default this is the first method called from a controller this is from where we route to other methods
179
+ */
180
+ function main(){
181
+ $this->WYSIJA_control_back();
182
+ if($this->model){
183
+ if(isset($_REQUEST['action'])) $action=$_REQUEST['action'];
184
+ else $action='defaultDisplay';
185
+ if(!$action) $action='defaultDisplay';
186
+
187
+ if($action){
188
+ $this->_tryAction($action);
189
+ }
190
+ }else{
191
+ $this->error('No Model is linked to this controller : '. get_class($this));
192
+ return false;
193
+ }
194
+
195
+ return true;
196
+ }
197
+
198
+ function _getTargetActionForm(){
199
+ $url = parse_url($_SERVER['REQUEST_URI']);
200
+ $target_action_form = $url['path'];
201
+ if(!empty($url['query'])){
202
+ parse_str($url['query'], $params);
203
+ if(isset($params['redirect']))
204
+ unset($params['redirect']);
205
+ $url['query'] = http_build_query($params);
206
+ $target_action_form = $url['path'].'?'.$url['query'];
207
+ }
208
+ return $target_action_form;
209
+ }
210
+ function __setMetaTitle(){
211
+ global $title;
212
+
213
+ if(isset($this->title))$title=$this->title;
214
+ else $title=$this->viewObj->title;
215
+ }
216
+
217
+ function _tryAction($action){
218
+ $action=strip_tags($action);
219
+ $_REQUEST = stripslashes_deep($_REQUEST);
220
+ $_POST = stripslashes_deep($_POST);
221
+
222
+ $is_batch_select = $this->_batchSelect();
223
+ $this->_affected_rows = $is_batch_select ? $this->_batch_select['count'] : (!empty($_REQUEST['wysija']['user']['user_id']) ? count($_REQUEST['wysija']['user']['user_id']) : 0);
224
+ if(method_exists($this, $action)){
225
+ /* in some bulk actions we need to specify the action name and one or few variables*/
226
+ $this->action=$action;
227
+ $this->viewShow=$this->action;
228
+ if(!$this->viewShow) $this->viewShow='defaultDisplay';
229
+
230
+ if(strpos($action, 'bulk_')===false)$this->$action();
231
+ else {
232
+ $this->$action($_REQUEST['wysija'][$this->model][$this->modelObj->pk]);
233
+ }
234
+
235
+ $this->__setMetaTitle();
236
+ }else{
237
+ /* in some bulk actions we need to specify the action name and one or few variables*/
238
+ if(strpos($action,'actionvar_')!== false){
239
+ $data=explode('-',$action);
240
+ $datas=array();
241
+
242
+ foreach($data as $dt){
243
+ $res=explode('_',$dt);
244
+ $datas[$res[0]]=$res[1];
245
+ }
246
+
247
+ $action =$datas['actionvar'];
248
+ unset($datas['actionvar']);
249
+ $this->action=$action;
250
+
251
+ if(method_exists($this, $this->action)){
252
+ $this->viewShow=$this->action;
253
+ $this->$action($datas);
254
+ $this->__setMetaTitle();
255
+
256
+ }else{
257
+ $this->error("Action '".$action."' does not exist in controller : ". get_class($this));
258
+ $this->redirect();
259
+ }
260
+ }else{
261
+ $this->error("Action '".$action."' does not exist in controller : ". get_class($this));
262
+ $this->redirect();
263
+ //$this->defaultDisplay();
264
+ }
265
+
266
+ }
267
+
268
+ if(defined('WYSIJA_REDIRECT')) $this->redirectProcess();
269
+
270
+ $this->checkTotalSubscribers();
271
+ }
272
+
273
+ function checkTotalSubscribers(){
274
+ add_action('wysija_check_total_subscribers',array($this,'_checkTotalSubscribers'));
275
+ do_action('wysija_remove_action_check_total_subscribers');
276
+ do_action('wysija_check_total_subscribers');
277
+ }
278
+
279
+ /**
280
+ * Batch select process
281
+ * - Currently, is for subscribers only
282
+ * - Get all matched subscribers and override to $_REQUEST['wysija']['user']['user_id']
283
+ */
284
+ function _batchSelect(){
285
+ if(empty($_REQUEST['wysija']['user']['force_select_all']))
286
+ return FALSE;
287
+ if (!(bool)$_REQUEST['wysija']['user']['force_select_all'])
288
+ return FALSE;
289
+ if(empty($_REQUEST['wysija']['user']['timestamp']))
290
+ return FALSE;
291
+ //$_POST['wysija']['filter'] = array(
292
+ // link_filter => '', //[subscribed, unsubscribed, unsubscribed, all]
293
+ // filter_list => int
294
+ //);
295
+ //
296
+ //$_POST['wysija']['user']['timestamp'] = int
297
+ //
298
+ //select all users which match to $_POST['wysija']['filter'] and create_at <= $_POST['wysija']['user']['timestamp']
299
+ // - build query
300
+
301
+ $select = array( '[wysija]user.user_id');
302
+
303
+ // filters for unsubscribed
304
+ $filters = $this->modelObj->detect_filters();
305
+
306
+
307
+ $this->_batch_select = array();
308
+
309
+
310
+ $this->_batch_select['query'] = $this->modelObj->get_subscribers( $select, $filters, '', true );
311
+ $this->_batch_select['query_count'] = $this->modelObj->get_subscribers( array( 'COUNT(DISTINCT([wysija]user.user_id))'), $filters, '', true );
312
+
313
+
314
+ //Create a temporary table
315
+ $temp_table_name = '[wysija]user'. time();
316
+ $temp_table_create = 'CREATE TEMPORARY TABLE IF NOT EXISTS '.$temp_table_name . ' (user_id int (10) NOT NULL, PRIMARY KEY (user_id)) ENGINE=MyISAM';
317
+ $temp_table_insert = 'INSERT IGNORE INTO '.$temp_table_name.' ' . $this->_batch_select['query'];
318
+ $model_user = WYSIJA::get('user','model');
319
+
320
+ $model_user->query($temp_table_create);
321
+ $model_user->query($temp_table_insert);
322
+
323
+ //Override the queres with temporary table
324
+ unset($this->_batch_select['where']);
325
+ $row_count = $model_user->query('get_row', 'SELECT COUNT(*) as row_count FROM '.$temp_table_name);
326
+ $this->_batch_select['original_query'] = $this->_batch_select['query']; // useful for export feature; in this case, we don't use temporary table
327
+ $this->_batch_select['select'] = 'SELECT DISTINCT user_id';
328
+ $this->_batch_select['from'] = 'FROM '.$temp_table_name . ' A';
329
+ $this->_batch_select['query'] = 'SELECT user_id FROM '.$temp_table_name;
330
+ $this->_batch_select['count'] = $row_count['row_count'];
331
+ return true;
332
+ }
333
+
334
+
335
+
336
+ function _checkTotalSubscribers(){
337
+
338
+ $config=WYSIJA::get('config','model');
339
+ $totalSubscribers=$config->getValue('total_subscribers');
340
+ $helper_licence = WYSIJA::get('licence','helper');
341
+
342
+ if((int)$totalSubscribers>1900){
343
+ if((int)$totalSubscribers>2000){
344
+ $url_checkout = $helper_licence->get_url_checkout('over200');
345
+ $this->error(str_replace(array('[link]','[/link]'),
346
+ array('<a title="'.__('Get Premium now',WYSIJA).'" target="_blank" href="'.$url_checkout.'">','</a>'),
347
+ sprintf(__('Yikes. You\'re over the limit of 2000 subscribers for the free version of MailPoet (%1$s in total). Sending is disabled now. Please upgrade your version to [link]premium[/link] to send without limits.',WYSIJA)
348
+ ,$totalSubscribers)),true);
349
+ }else{
350
+ $url_checkout = $helper_licence->get_url_checkout('near200');
351
+ $this->notice(str_replace(array('[link]','[/link]'),
352
+ array('<a title="'.__('Get Premium now',WYSIJA).'" target="_blank" href="'.$url_checkout.'">','</a>'),
353
+ sprintf(__('Yikes! You\'re near the limit of %1$s subscribers for MailPoet\'s free version. Upgrade to [link]Premium[/link] to send without limits, and more.',WYSIJA)
354
+ ,"2000")));
355
+ }
356
+ }
357
+ }
358
+
359
+ function edit($id=false){
360
+
361
+ if(isset($_REQUEST['id']) || $id){
362
+ if(!$id) $id=$_REQUEST['id'];
363
+ $this->data[$this->modelObj->table_name]=$this->modelObj->getOne($this->form_columns,array($this->modelObj->pk=>$id));
364
+
365
+ //$this->viewObj->render($this->action,$data);
366
+
367
+ }else{
368
+ $this->error('Cannot edit element primary key is missing : '. get_class($this));
369
+ }
370
+
371
+ }
372
+
373
+ function view($id=false){
374
+
375
+ if(isset($_REQUEST['id']) || $id){
376
+ if(!$id) $id=$_REQUEST['id'];
377
+ $this->data[$this->modelObj->table_name]=$this->modelObj->getOne($this->form_columns,array($this->modelObj->pk=>$id));
378
+
379
+ }else{
380
+ $this->error('Cannot view element primary key is missing : '. get_class($this));
381
+ }
382
+
383
+ }
384
+
385
+ function add($dataPost=false){
386
+
387
+ if(!$dataPost){
388
+ $data=array();
389
+ foreach($this->form_columns as $key){
390
+ $data[$key]='';
391
+ }
392
+ }else{
393
+
394
+ $data=array();
395
+ foreach($this->form_columns as $key){
396
+ if($key != $this->viewObj->model->pk) $data[$key]=$dataPost[$key];
397
+ }
398
+ $data[$this->viewObj->model->pk]='';
399
+ }
400
+
401
+
402
+ //$this->viewObj->render('edit',$data);
403
+ }
404
+
405
+ function save(){
406
+ $this->requireSecurity();
407
+ //see if it's an update or an insert
408
+ //get the pk and its value as a conditions where pk = pkval
409
+ $conditions=$this->getPKVal($this->modelObj);
410
+
411
+ if($conditions){
412
+ //this an update
413
+
414
+ $result=$this->modelObj->update($_POST['wysija'][$this->model],$conditions);
415
+
416
+ if($this->msgOnSave){
417
+
418
+ // Create the update success message and add edit again link.
419
+ $update_success = str_replace(array('[link]','[/link]'),array('<a href="admin.php?page=wysija_subscribers&action=edit&id='.$result.'" >',"</a>"), $this->messages['update'][true]);
420
+
421
+ if ($result) {
422
+ $this->notice($update_success);
423
+ } else {
424
+ if($result==0){
425
+
426
+ }else{
427
+ $this->error($this->messages['update'][false],true);
428
+ }
429
+
430
+ }
431
+ }
432
+
433
+
434
+ if($this->redirectAfterSave){
435
+ if(isset($this->modelObj->stay)){
436
+ $this->action='edit';
437
+ $this->redirect();
438
+ //$this->edit($result);
439
+ }else{
440
+ $this->action='edit';
441
+ $this->redirect();
442
+ //$this->edit($result);
443
+ }
444
+ }
445
+
446
+ }else{
447
+ //this is an insert
448
+ unset($_POST['wysija'][$this->model][$this->modelObj->pk]);
449
+ $result=$this->modelObj->insert($_POST['wysija'][$this->model]);
450
+
451
+ if($this->msgOnSave){
452
+ if($result) $this->notice($this->messages['insert'][true]);
453
+ else{
454
+ $this->error($this->messages['insert'][false],true);
455
+ }
456
+ }
457
+
458
+
459
+ if($this->redirectAfterSave){
460
+ if(isset($this->modelObj->stay)){
461
+ $this->action='add';
462
+ $this->add($_POST['wysija'][$this->model]);
463
+ }else{
464
+ $this->action='main';
465
+ $this->redirect();
466
+ }
467
+ }
468
+
469
+ }
470
+
471
+ //now we redirect to the edit page with the data in it
472
+ return $result;
473
+ }
474
+
475
+ function bulk_delete($ids){
476
+ $this->requireSecurity();
477
+ foreach($ids as $id){
478
+
479
+ $conditions=$this->getPKVal($this->modelObj);
480
+ if(!$conditions) $this->error('Cannot obtain PKVal from GET or POST.');
481
+
482
+ $result=$this->modelObj->delete($conditions);
483
+ $this->modelObj->reset();
484
+ }
485
+ $this->notice(__('Elements deleted',WYSIJA));
486
+ $this->redirect();
487
+ }
488
+
489
+ function delete(){
490
+ // see if it's an update or an insert
491
+ $this->requireSecurity();
492
+ $conditions=$this->getPKVal($this->modelObj);
493
+ if(!$conditions) $this->error('Cannot obtain PKVal from GET or POST.');
494
+
495
+ $result=$this->modelObj->delete($conditions);
496
+ if($result){
497
+ $this->notice(__('Element has been deleted.',WYSIJA));
498
+ }
499
+
500
+
501
+ $this->modelObj->reset();
502
+ //now we redirect to the edit page with the data in it
503
+ $this->action='main';
504
+ $this->redirect();
505
+ }
506
+
507
+ function redirect($location=false){
508
+ global $wysi_location;
509
+ define('WYSIJA_REDIRECT',true);
510
+ if($location)
511
+ {
512
+ $url = parse_url($location);
513
+ if(!empty($url['query'])) {
514
+ $location .= '&';
515
+ } else {
516
+ $location .= '?';
517
+ }
518
+
519
+ $location .= 'redirect=1';
520
+ }
521
+ $wysi_location=$location;
522
+ }
523
+
524
+ function redirectProcess(){
525
+ global $wysi_location;
526
+
527
+ if(!$wysi_location) {
528
+ $wysi_location=$this->getDefaultUrl();
529
+ }
530
+ WYSIJA::redirect($wysi_location);
531
+
532
+ }
533
+
534
+ function popupReturn($viewFunc) {
535
+ return wp_iframe( array($this->viewObj,'popup_'.$viewFunc), $this->data);
536
+ }
537
+
538
+ function _addTab($defaulttab){
539
+ return $this->iframeTabs;
540
+ }
541
+
542
+ function popupContent(){
543
+ // remove auth check
544
+ remove_action('admin_enqueue_scripts', 'wp_auth_check_load');
545
+
546
+ // add popup css
547
+ wp_enqueue_style('custom_popup_css', WYSIJA_URL.'css/adminPopup.css', array(), WYSIJA::get_version(), 'screen');
548
+
549
+ global $viewMedia;
550
+ $viewMedia=$this->viewObj;
551
+ $_GET['type']=$_REQUEST['type']='image';
552
+
553
+ $config=WYSIJA::get('config','model');
554
+ $_GET['post_id']=$_REQUEST['post_id']=$config->getValue('confirm_email_link');
555
+ $post_id = isset($_GET['post_id'])? (int) $_GET['post_id'] : 0;
556
+ if(file_exists(ABSPATH.'wp-admin'.DS.'admin.php')) require_once(ABSPATH.'wp-admin'.DS.'admin.php');
557
+
558
+ @header('Content-Type: ' . get_option('html_type') . '; charset=' . get_option('blog_charset'));
559
+
560
+ add_filter('media_upload_tabs', array($this,'_addTab'));
561
+
562
+ if(!isset($this->iframeTabs)) {
563
+
564
+
565
+ //if wp version includes plupload then let's use it
566
+ if(version_compare(get_bloginfo('version'), '3.3.0')>= 0){
567
+ $this->iframeTabs=array(
568
+ 'special_new_wordp_upload'=>__('Upload',WYSIJA));
569
+ }else{
570
+ $this->iframeTabs=array(
571
+ 'special_wordp_upload'=>__('Upload',WYSIJA));
572
+ }
573
+
574
+ $this->iframeTabs['special_wysija_browse']=__('Newsletter Images',WYSIJA);
575
+ $this->iframeTabs['special_wordp_browse']=__('WordPress Posts\' Images',WYSIJA);
576
+
577
+ foreach($this->iframeTabs as $actionKey =>$actionTitle)
578
+ add_action('media_upload_'.$actionKey, array($this,$actionKey));
579
+ }else add_action('media_upload_standard', array($this,'popupReturn'));
580
+
581
+ // upload type: image, video, file, ..?
582
+ if ( isset($_GET['type']) )
583
+ $type = strval($_GET['type']);
584
+ else
585
+ $type = apply_filters('media_upload_default_type', 'file');
586
+
587
+ // tab: gallery, library, or type-specific
588
+ if ( isset($_GET['tab']) )
589
+ $tab = strval($_GET['tab']);
590
+ else
591
+ $tab ='special_wysija_browse';
592
+
593
+ $body_id = 'media-upload';
594
+ // let the action code decide how to handle the request
595
+ if ( $tab == 'type' || $tab == 'type_url' )
596
+ //i'm not so sure we need that line
597
+ do_action("media_upload_$type");
598
+ else{
599
+ if(strpos($tab, 'special_')!==false){
600
+ do_action("media_upload_$tab");
601
+ }else{
602
+ do_action('media_upload_standard',$tab);
603
+ }
604
+ }
605
+
606
+ exit;
607
+
608
+ }
609
+
610
+ function getDefaultUrl($filter=true){
611
+ $location="admin.php?page=".$_REQUEST['page'];
612
+
613
+ if($filter){
614
+ if(isset($_REQUEST['search']) && $_REQUEST['search']){
615
+ $location.='&search='.$_REQUEST['search'];
616
+ }
617
+
618
+ if(isset($_REQUEST['filter-list']) && $_REQUEST['filter-list']){
619
+ $location.='&filter-list='.$_REQUEST['filter-list'];
620
+ }
621
+
622
+ if(isset($_REQUEST['link_filter']) && $_REQUEST['link_filter']){
623
+ $location.='&link_filter='.$_REQUEST['link_filter'];
624
+ }
625
+
626
+ if(isset($_REQUEST['orderby']) && $_REQUEST['orderby']){
627
+ $location.='&orderby='.$_REQUEST['orderby'];
628
+ }
629
+
630
+ if(isset($_REQUEST['ordert']) && $_REQUEST['ordert']){
631
+ $location.='&ordert='.$_REQUEST['ordert'];
632
+ }
633
+ }
634
+
635
+ return $location;
636
+ }
637
+
638
+ /**
639
+ * to remove the conflicts in wysija's interfaces
640
+ * @param boolean $themes
641
+ */
642
+ function get_conflictive_plugins($themes=false){
643
+
644
+ /**
645
+ * List of all the conflictive extensions which invite themselves on our interfaces and break some of our js:
646
+ * tribulant newsletter
647
+ */
648
+ $conflictivePlugins = array(
649
+ 'tribulant-wp-mailinglist' => array(
650
+ 'file' => 'wp-mailinglist/wp-mailinglist.php',
651
+ 'version' => '3.8.7',
652
+ 'clean' => array(
653
+ 'admin_head' => array(
654
+ '10' => array(
655
+ 'objects' => array('wpMail')
656
+ )
657
+ )
658
+ )
659
+ ),
660
+ 'wp-events' => array(
661
+ 'file' => 'wp-events/wp-events.php',
662
+ 'version' => '',
663
+ 'clean' => array(
664
+ 'admin_head' => array(
665
+ '10' => array(
666
+ 'function' => 'events_editor_admin_head'
667
+ )
668
+ )
669
+ )
670
+ ),
671
+ 'email-users' => array(
672
+ 'file' => 'email-users/email-users.php',
673
+ 'version' => '',
674
+ 'clean' => array(
675
+ 'admin_head' => array(
676
+ '10' => array(
677
+ 'function' => 'editor_admin_head'
678
+ )
679
+ )
680
+ )
681
+ ),
682
+ 'acf' => array(
683
+ 'file' => 'advanced-custom-fields/acf.php',
684
+ 'version' => '3.1.7',
685
+ 'clean' => array(
686
+ 'init' => array(
687
+ '10' => array(
688
+ 'objects' => array('Acf')
689
+ )
690
+ )
691
+ )
692
+ ),
693
+ 'wptofacebook' => array(
694
+ 'file' => 'wptofacebook/index.php',
695
+ 'version' => '1.2.3',
696
+ 'clean' => array(
697
+ 'admin_head' => array(
698
+ '10' => array(
699
+ 'function' => 'WpToFb::wptofb_editor_admin_head'
700
+ )
701
+ )
702
+ )
703
+ ),
704
+ 'mindvalley-pagemash' => array(
705
+ 'file' => 'mindvalley-pagemash/pagemash.php',
706
+ 'version' => '1.1',
707
+ 'clean' => array(
708
+ 'admin_print_scripts' => array(
709
+ '10' => array(
710
+ 'function' => 'pageMash_head'
711
+ )
712
+ )
713
+ )
714
+ ),
715
+ 'wp-polls' => array(
716
+ 'file' => 'wp-polls/wp-polls.php',
717
+ 'version' => '2.63',
718
+ 'clean' => array(
719
+ 'wp_enqueue_scripts' => array(
720
+ '10' => array(
721
+ 'function' => 'poll_scripts'
722
+ )
723
+ )
724
+ )
725
+ ),
726
+ 'wp_rokajaxsearch' => array(
727
+ 'file' => 'wp_rokajaxsearch/rokajaxsearch.php',
728
+ 'version' => '',
729
+ 'clean' => array(
730
+ 'init' => array(
731
+ '-50' => array(
732
+ 'function' => 'rokajaxsearch_mootools_init'
733
+ )
734
+ )
735
+ )
736
+ ),
737
+ 'wp_rokstories' => array(
738
+ 'file' => 'wp_rokstories/rokstories.php',
739
+ 'version' => '',
740
+ 'clean' => array(
741
+ 'init' => array(
742
+ '-50' => array(
743
+ 'function' => 'rokstories_mootools_init'
744
+ )
745
+ )
746
+ )
747
+ ),
748
+ 'simple-links' => array(
749
+ 'file' => 'simple-links/simple-links.php',
750
+ 'version' => '1.5',
751
+ 'clean' => array(
752
+ 'admin_print_scripts' => array(
753
+ '10' => array(
754
+ 'objects' => array('simple_links_admin')
755
+ )
756
+ )
757
+ )
758
+ )
759
+ );
760
+
761
+ $conflictiveThemes = array(
762
+ 'smallbiz' => array(
763
+ 'clean' => array(
764
+ 'admin_head' => array(
765
+ '10' => array(
766
+ 'function' => 'smallbiz_on_admin_head'
767
+ )
768
+ )
769
+ )
770
+ ),
771
+ 'balance' => array(
772
+ 'clean' => array(
773
+ 'admin_enqueue_scripts' => array(
774
+ '10' => array(
775
+ 'functions' => array('al_admin_scripts', 'al_adminpanel_scripts', 'al_pricing_tables_scripts')
776
+ )
777
+ ),
778
+ 'admin_head' => array(
779
+ '10' => array(
780
+ 'function' => 'al_admin_head'
781
+ )
782
+ )
783
+ )
784
+ )
785
+ );
786
+
787
+ if($themes) return $conflictiveThemes;
788
+ return $conflictivePlugins;
789
+ }
790
+ }
controllers/back/campaigns.php ADDED
@@ -0,0 +1,2759 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ global $viewMedia;
4
+ defined('WYSIJA') or die('Restricted access');
5
+
6
+ class WYSIJA_control_back_campaigns extends WYSIJA_control_back {
7
+
8
+ var $model = 'campaign';
9
+ var $view = 'campaigns';
10
+ var $list_columns = array('campaign_id', 'name', 'description');
11
+ var $searchable = array('name', 'subject');
12
+ var $filters = array();
13
+ var $base_url = 'admin.php';
14
+
15
+ function WYSIJA_control_back_campaigns() {
16
+
17
+ }
18
+
19
+ function installation(){
20
+
21
+ $start_time = time();
22
+ $check_max_execution_time = get_option( 'mpoet_check_max_execution_time' );
23
+ $ini_max_execution_time = ini_get( 'max_execution_time' );
24
+
25
+ // 1 - we haven't run the max execution time extension test yet, let's do it
26
+ // this test should be run in the background, it may take more than 5 minutes to validate
27
+ if( empty( $check_max_execution_time ) ){
28
+
29
+ // we set the max_execution_time to infinite
30
+ @ini_set( 'max_execution_time' , 0 );
31
+ // we keep running the script even if the user abort his request
32
+ @ignore_user_abort( true );
33
+
34
+ $reset_max_execution_time = ini_get( 'max_execution_time' );
35
+
36
+ // FAILURE some servers the "max execution time" returned is Array we make sure our value is numeric
37
+ if( !is_numeric( $ini_max_execution_time ) ){
38
+ WYSIJA::update_option( 'mpoet_check_max_execution_time' , $ini_max_execution_time-1 );
39
+ return;
40
+ }
41
+
42
+ // FAILURE this server doesn't allow us to set the max_execution_time
43
+ if( $reset_max_execution_time == $ini_max_execution_time ){
44
+ WYSIJA::update_option( 'mpoet_check_max_execution_time' , $ini_max_execution_time-1 );
45
+ return;
46
+ }
47
+
48
+ // we set a FAILURE value to our flag so we know if the process crash while executing
49
+ WYSIJA::update_option( 'mpoet_check_max_execution_time' , $ini_max_execution_time );
50
+
51
+ // the common default timeout value is 300, so if by default the timelimit is 0 (which might have been set by another application)
52
+ // we default it to 300 known as a common default value and we run the extension test
53
+ if( $ini_max_execution_time == 0 ) $ini_max_execution_time = 300;
54
+
55
+ // we run asleep script during the initial max execution time to a larger value
56
+ $total_time_to_run_script = $ini_max_execution_time + 20;
57
+ $sleeping_intervals = 5;
58
+ $duration_so_far = 0;
59
+
60
+ // we sleep until the extended total timeout value is reached or it breaks before but we have the value we need
61
+ while($total_time_to_run_script >0){
62
+ $total_time_to_run_script = $total_time_to_run_script - $sleeping_intervals;
63
+ sleep( $sleeping_intervals );
64
+
65
+ $duration_so_far = time() - $start_time;
66
+ WYSIJA::update_option( 'mpoet_check_max_execution_time' , $duration_so_far );
67
+ }
68
+
69
+ // SUCCESS the scripts didn't die we set our flag in the db so that we know the test is a success
70
+ WYSIJA::update_option( 'mpoet_check_max_execution_time' , $duration_so_far );
71
+
72
+ }else{
73
+ // 2 - the max execution time extension test has been run, this is the results we get
74
+ if( $check_max_execution_time > $ini_max_execution_time ){
75
+ echo 'max_execution_time('.$ini_max_execution_time.') can be extended('.$check_max_execution_time.')';
76
+ }else{
77
+ echo 'max_execution_time('.$ini_max_execution_time.') cannot be extended( max value : '.$check_max_execution_time.')';
78
+ }
79
+ }
80
+ exit;
81
+ }
82
+
83
+ /**
84
+ * Welcome page first time install
85
+ * @return boolean
86
+ */
87
+ function welcome_new() {
88
+ $this->title = $this->viewObj->title = __('Welcome Page!', WYSIJA);
89
+ $this->jsTrans['instalwjp'] = __('Installing MailPoet Newsletter Premium plugin', WYSIJA);
90
+ $helper_readme = WYSIJA::get('readme', 'helper');
91
+ $helper_readme->scan();
92
+ $this->data = array();
93
+ $this->data['abouttext'] = __('A Brand New MailPoet. Let the Fun Begin.', WYSIJA);
94
+
95
+ $model_config = WYSIJA::get('config', 'model');
96
+ $is_multisite = is_multisite();
97
+ $is_network_admin = WYSIJA::current_user_can('manage_network');
98
+ if ($is_multisite && $is_network_admin) {
99
+ $model_config->save(array('ms_wysija_whats_new' => WYSIJA::get_version()));
100
+ } else {
101
+ $model_config->save(array('wysija_whats_new' => WYSIJA::get_version()));
102
+ }
103
+
104
+ //add a new language code with a new video
105
+ $video_language=array();
106
+ //$video_language['en_EN'] = '<iframe width="853" height="480" src="http://www.youtube.com/embed/pYzaHDTg5Jk" frameborder="0" allowfullscreen></iframe>';
107
+ $video_language['en_EN'] = '<iframe src="//player.vimeo.com/video/81479899" width="500" height="281" frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe>';
108
+ //$video_language['fr_FR'] = '<iframe width="853" height="480" src="http://www.youtube.com/embed/W5EyW5w7aWQ" frameborder="0" allowfullscreen></iframe>';
109
+ // $video_language['sv_SE']='<iframe width="853" height="480" src="http://www.youtube.com/embed/O8_t_dekx74" frameborder="0" allowfullscreen></iframe>';
110
+ //$video_language['ar']='<iframe width="853" height="480" src="http://www.youtube.com/embed/cyDHlX_qgOo" frameborder="0" allowfullscreen></iframe>';
111
+
112
+ $wp_lang = get_locale();
113
+ if (!empty($wp_lang) && isset($video_language[$wp_lang])) {
114
+ $welcome_video_link = $video_language[$wp_lang];
115
+ } else {
116
+ $welcome_video_link = $video_language['en_EN'];
117
+ }
118
+
119
+ $this->data['sections'][] = array(
120
+ 'title' => __('Hey, we\'re curious! How did you find out about us?', WYSIJA) . '<span id="poll_result"></span>',
121
+ 'format' => 'normal',
122
+ 'paragraphs' => array(
123
+ '<ul class="welcome_poll">
124
+ <li>
125
+ <input type="radio" id="how_did_you_find_us_1" value="repository" name="how_did_you_find_us">
126
+ <label value="lab1" for="how_did_you_find_us_1">' . __('WordPress.org plugin repository', WYSIJA) . '</label>
127
+ </li>
128
+ <li>
129
+ <input type="radio" id="how_did_you_find_us_2" value="search_engine" name="how_did_you_find_us">
130
+ <label value="lab2" for="how_did_you_find_us_2">' . __('Google or other search engine', WYSIJA) . '</label>
131
+ </li>
132
+ <li>
133
+ <input type="radio" id="how_did_you_find_us_3" value="friend" name="how_did_you_find_us">
134
+ <label value="lab3" for="how_did_you_find_us_3">' . __('Friend recommendation', WYSIJA) . '</label>
135
+ </li>
136
+ <li>
137
+ <input type="radio" id="how_did_you_find_us_4" value="url" name="how_did_you_find_us">
138
+ <label value="lab4" for="how_did_you_find_us_4">' . __('Blog post, online review, forum:', WYSIJA) . '</label>
139
+ <input type="text" id="how_did_you_find_us_4_url" name="how_did_you_find_us_url" placeholder="' . __('Please enter the address where you\'ve found out about us', WYSIJA) . '">
140
+ </li>
141
+ </ul>'
142
+ )
143
+ );
144
+
145
+ $this->data['sections'][]=array(
146
+ 'title'=>__('First Time? See it in Action',WYSIJA),
147
+ 'format'=>'normal',
148
+ 'paragraphs'=>array(
149
+ __('You can start by watching this video by one of our users.',WYSIJA),
150
+ $welcome_video_link
151
+ )
152
+ );
153
+
154
+ // Check if user is already sharing data, and hide the share link.
155
+ if (!$model_config->getValue('analytics')) {
156
+ $share_section = array(
157
+ 'title' => __('Share your data', WYSIJA),
158
+ 'content' => str_replace(
159
+ array('[link]', '[/link]', '[ajaxlink]', '[/ajaxlink]'), array('<a title="Anonymous Data" target="_blank" href="http://support.mailpoet.com/knowledgebase/share-your-data/?utm_source=wpadmin&utm_campaign=welcome_page">', '</a>', '<a id="share_analytics" href="javascript:;">', '</a>'), __("We know too little about our users. We're looking for [link]anonymous data[/link] to build a better plugin. [ajaxlink]Yes, count me in![/ajaxlink]", WYSIJA))
160
+ );
161
+ } else {
162
+ $share_section = array(
163
+ 'title' => __('Share your data', WYSIJA),
164
+ 'content' => str_replace(
165
+ array('[link]', '[/link]'), array('<a title="Anonymous Data" target="_blank" href="http://support.mailpoet.com/knowledgebase/share-your-data/?utm_source=wpadmin&utm_campaign=welcome_page">', '</a>', '<a id="share_analytics" href="javascript:;">', '</a>'), __("We know too little about our users. We're looking for [link]anonymous data[/link] to build a better plugin. Thanks, you're already sharing!", WYSIJA))
166
+ );
167
+ }
168
+
169
+ $this->data['sections'][] = array(
170
+ 'title' => __('What You Can Do', WYSIJA),
171
+ 'cols' => array(
172
+ array(
173
+ 'title' => __('5 minute newbie guide', WYSIJA),
174
+ 'content' => __('Your MailPoet comes with an example newsletter. You\'ll see it when you close this welcome page. Edit it to start playing with it.', WYSIJA)
175
+ ),
176
+ $share_section,
177
+ array(
178
+ 'title' => __('Help yourself. Or let us help you.', WYSIJA),
179
+ 'content' => str_replace(
180
+ array('[link]', '[/link]'), array('<a href="http://support.mailpoet.com/" target="_blank" title="On our blog!">', '</a>'), __('We got documentation and a ticket system on [link]support.mailpoet.com[/link]. We answer within 24h.', WYSIJA))
181
+ )
182
+ ),
183
+ 'format' => 'three-col',
184
+ );
185
+
186
+ $this->viewObj->skip_header = true;
187
+
188
+ return true;
189
+ }
190
+
191
+ /**
192
+ * Welcome page for updaters
193
+ * @return boolean
194
+ */
195
+ function whats_new() {
196
+
197
+ $this->title = $this->viewObj->title = __('What\'s new?', WYSIJA);
198
+ $this->jsTrans['instalwjp'] = __('Installing MailPoet Newsletter Premium plugin', WYSIJA);
199
+ wp_enqueue_style('wysija-admin-css-premium', WYSIJA_URL.'css/admin-premium.css',array(),WYSIJA::get_version());
200
+
201
+ $this->viewObj->skip_header = true;
202
+ return true;
203
+ }
204
+
205
+
206
+
207
+ /* START prem check hook */
208
+
209
+ // when curl or any php remote function not available mailpoet.com returns lcheck to that function
210
+ function licok() {
211
+ parent::WYSIJA_control_back();
212
+ $dt = get_option('wysijey');
213
+
214
+ if (isset($_REQUEST['xtz']) && $dt === $_REQUEST['xtz']) {
215
+ $dataconf = array('premium_key' => base64_encode(get_option('home') . time()), 'premium_val' => time());
216
+ $this->notice(__('Premium version is valid for your site.', WYSIJA));
217
+ } else {
218
+ $dataconf = array('premium_key' => '', 'premium_val' => '');
219
+
220
+ $helper_licence = WYSIJA::get('licence', 'helper');
221
+ $url_premium = 'http://www.mailpoet.com/checkout/?wysijadomain=' . $dt . '&nc=1&utm_source=wpadmin&utm_campaign=error_licence_activation';
222
+
223
+ $this->error(str_replace(array('[link]', '[/link]'), array('<a href="' . $url_premium . '" target="_blank">', '</a>'), __('Premium licence does not exist for your site. Purchase it [link]here[/link].', WYSIJA)), 1);
224
+ }
225
+ WYSIJA::update_option('wysicheck', false);
226
+ $modelConf = WYSIJA::get('config', 'model');
227
+ $modelConf->save($dataconf);
228
+
229
+ $this->redirect('admin.php?page=wysija_config#tab-premium');
230
+ }
231
+
232
+ /* END prem check hook */
233
+
234
+ function validateLic() {
235
+ $helpLic = WYSIJA::get('licence', 'helper');
236
+ $res = $helpLic->check();
237
+
238
+ $this->redirect();
239
+ }
240
+
241
+ /**
242
+ * this function is triggered when sending manually the emails with the "Don't wait and send right now" button
243
+ * @param type $dataPost
244
+ */
245
+ function manual_send($dataPost = false) {
246
+ $modelQ = WYSIJA::get('queue', 'model');
247
+ $config = WYSIJA::get('config', 'model');
248
+ if ((int) $config->getValue('total_subscribers') < 2000) {
249
+ if ($modelQ->count() > 0) {
250
+ $helperQ = WYSIJA::get('queue', 'helper');
251
+ $emailid = false;
252
+ if ($_REQUEST['emailid']) {
253
+ $emailid = $_REQUEST['emailid'];
254
+ }
255
+ $helperQ->process($emailid);
256
+ } else {
257
+ echo '<strong style="font-family: Arial; font-weight: bold; font-size: 12px;">' . __('Queue is empty!', WYSIJA) . '</strong>';
258
+ }
259
+ exit;
260
+ } else {
261
+ //deprecated
262
+ do_action('wysija_send_test_editor');
263
+
264
+ do_action('wysija_manual_send');
265
+ }
266
+
267
+ exit;
268
+ }
269
+
270
+ /**
271
+ * test the bounce handling maybe this should move somewhere else like config controller
272
+ * @return boolean
273
+ */
274
+ function test_bounce() {
275
+ // bounce handling
276
+ $helper_bounce = WYSIJA::get('bounce', 'helper');
277
+
278
+ // in a multisite case we process first the bounce recording into the bounce table
279
+ if (is_multisite()) {
280
+ $helper_bounce->record_bounce_ms();
281
+
282
+ // then we take actions from what has been returned by the bounce
283
+ $helper_bounce->process_bounce_ms();
284
+ } else {
285
+ $helper_bounce->process_bounce();
286
+ }
287
+ exit;
288
+ }
289
+
290
+ function add($dataPost = false) {
291
+ $this->title = sprintf(__('Step %1$s', WYSIJA), 1);
292
+ $this->js[] = 'wysija-validator';
293
+
294
+ $this->js[] = 'wysija-edit-autonl';
295
+ $this->js['admin-campaigns-edit'] = 'admin-campaigns-edit';
296
+ $this->jsTrans['descauto'] = str_replace(array('[newsletter:number]', '[newsletter:total]', '[newsletter:post_title]'), array('<b>[newsletter:number]</b>', '<b>[newsletter:total]</b>', '<b>[newsletter:post_title]</b>'), __('Insert [newsletter:total] to show number of posts, [newsletter:post_title] to show the latest post\'s title & [newsletter:number] to display the issue number.', WYSIJA));
297
+ $this->jsTrans['descstandard'] = __('The first thing your subscribers see. Be creative and increase your open rate!', WYSIJA);
298
+ $this->immediateWarning();
299
+ $this->viewObj->title = __('First step: main details', WYSIJA);
300
+ $this->viewShow = 'add';
301
+ $this->data = array();
302
+ $this->data['campaign'] = array('name' => '', 'description' => '');
303
+ $modelConfig = WYSIJA::get('config', 'model');
304
+ $this->data['email'] = array('subject' => '', 'from_email' => $modelConfig->getValue('from_email'), 'from_name' => $modelConfig->getValue('from_name'));
305
+ $this->data['lists'] = $this->__getLists(false, true, true);
306
+
307
+ $this->dataAutoNl();
308
+ $this->jsLoc['admin-campaigns-edit']['autofields'] = $this->data['autonl']['fields'];
309
+ }
310
+
311
+ /**
312
+ * get the fields and fields value necessary when dealing with automatic newsletters
313
+ */
314
+ function dataAutoNl() {
315
+ $dataFrequencyNoImmediate = $dataFrequency = array('daily' => __('once a day at...', WYSIJA),
316
+ 'weekly' => __('weekly on...', WYSIJA),
317
+ 'monthly' => __('monthly on the...', WYSIJA),
318
+ 'monthlyevery' => __('monthly every...', WYSIJA),
319
+ 'immediate' => __('immediately.', WYSIJA));
320
+
321
+ unset($dataFrequencyNoImmediate['immediate']);
322
+
323
+ $times = array();
324
+ $time = strtotime('00:00:00');
325
+ $toolboxH = WYSIJA::get('toolbox', 'helper');
326
+ $times['00:00:00'] = $toolboxH->localtime($time);
327
+
328
+ for ($i = 1; $i < 24; $i++) {
329
+ $time = strtotime('+ 1hour', $time);
330
+ $key = date('H:i:s', $time);
331
+ $times[$key] = $toolboxH->localtime($time);
332
+ }
333
+
334
+ $daysvalues = $toolboxH->getday();
335
+
336
+ $numberweeks = $toolboxH->getweeksnumber();
337
+ $daynumbers = $toolboxH->getdaynumber();
338
+
339
+
340
+ $dataLists = array();
341
+
342
+ foreach ($this->data['lists'] as $datal) {
343
+ if ($datal['is_enabled'])
344
+ $dataLists[$datal['list_id']] = $datal['name'];
345
+ }
346
+
347
+ // Get all available roles
348
+ $wptoolsH = WYSIJA::get('wp_tools', 'helper');
349
+ $roles = $wptoolsH->wp_get_all_roles();
350
+ $available_roles = array('any' => __('in any WordPress role', WYSIJA));
351
+ foreach ($roles as $role => $name) {
352
+ $available_roles[$role] = $name;
353
+ }
354
+
355
+ $this->data['autonl']['fields'] = array(
356
+ 'event' => array(
357
+ 'values' => array(
358
+ 'new-articles' => __('When there\'s new content...', WYSIJA),
359
+ 'subs-2-nl' => __('When someone subscribes to the list...', WYSIJA),
360
+ 'new-user' => __('When a new WordPress user is added to your site...', WYSIJA),
361
+ ),
362
+ 'valueshow' => array(
363
+ 'new-articles' => array('when-article'),
364
+ 'subs-2-nl' => array('subscribetolist', 'numberafter', 'numberofwhat', 'unique_send'),
365
+ 'new-user' => array('roles', 'numberafter', 'numberofwhat', 'unique_send'),
366
+ ),
367
+ 'style' => 'width:300px;'
368
+ ),
369
+ 'when-article' => array(
370
+ 'values' => $dataFrequency,
371
+ 'valueshow' => array(
372
+ 'daily' => array('time'),
373
+ 'weekly' => array('dayname', 'time'),
374
+ 'monthly' => array('daynumber', 'time'),
375
+ 'monthlyevery' => array('dayevery', 'dayname', 'time'),
376
+ )
377
+ ),
378
+ 'subscribetolist' => array(
379
+ 'values' => $dataLists,
380
+ 'style' => 'width:300px;'
381
+ ),
382
+ 'roles' => array(
383
+ 'values' => $available_roles
384
+ ),
385
+ 'numberafter' => array(
386
+ 'type' => 'input',
387
+ 'style' => 'width:35px;',
388
+ 'class' => 'validate[required,custom[integer],min[1]]',
389
+ ),
390
+ 'numberofwhat' => array(
391
+ 'values' => array(
392
+ 'immediate' => __('immediately.', WYSIJA),
393
+ 'hours' => __('hour(s) after.', WYSIJA),
394
+ 'days' => __('day(s) after.', WYSIJA),
395
+ 'weeks' => __('week(s) after.', WYSIJA)
396
+ ),
397
+ 'valuesunit' => array(
398
+ 'immediate' => __('immediately', WYSIJA),
399
+ 'hours' => __('hour(s)', WYSIJA),
400
+ 'days' => __('day(s)', WYSIJA),
401
+ 'weeks' => __('week(s)', WYSIJA)
402
+ ),
403
+ ),
404
+ 'dayevery' => array(
405
+ 'values' => $numberweeks,
406
+ ),
407
+ 'dayname' => array(
408
+ 'values' => $daysvalues,
409
+ ),
410
+ 'daynumber' => array(
411
+ 'values' => $daynumbers,
412
+ ),
413
+ 'time' => array(
414
+ 'values' => $times,
415
+ ),
416
+ /* 'unique_send'=>array(
417
+ 'label_before'=>__('Send this email only once.',WYSIJA),
418
+ 'type'=>'checkbox'
419
+ ), */
420
+ );
421
+ $helpersEvent = WYSIJA::get('autonews', 'helper');
422
+ $extraEvents = $helpersEvent->events();
423
+
424
+ /* if there are plugin to add autonewsletter event they are adding their customized field over here */
425
+ if ($extraEvents) {
426
+ foreach ($extraEvents as $k => $v) {
427
+
428
+ $this->data['autonl']['fields']['event']['values'][$k] = $v['title'];
429
+ foreach ($v['fields'] as $fieldCKEY => $fieldCVAL) {
430
+ if (isset($this->data['autonl']['fields'][$fieldCKEY]))
431
+ continue;
432
+ }
433
+ $this->data['autonl']['fields']['event']['valueshow'][$k] = array_keys($v['fields']);
434
+ }
435
+ }
436
+ }
437
+
438
+ function __getLists($enabled = true, $count = false, $notgetalllistid = false) {
439
+ $modelList = WYSIJA::get('list', 'model');
440
+ //get lists which have users and are enabled */
441
+ if ($enabled)
442
+ $enabledstrg = ' is_enabled>0 and';
443
+ else
444
+ $enabledstrg = '';
445
+
446
+ $extrasql = '';
447
+ if (!$notgetalllistid)
448
+ $extrasql = 'WHERE list_id in (SELECT distinct(list_id) from [wysija]user_list )';
449
+ $query = 'SELECT * FROM [wysija]list ' . $extrasql;
450
+ $listres = $modelList->query('get_res', $query);
451
+
452
+ if ($count) {
453
+ $mConfig = WYSIJA::get('config', 'model');
454
+ $condit = '>=';
455
+ if ($mConfig->getValue('confirm_dbleoptin'))
456
+ $condit = '>';
457
+ $qry1 = "SELECT count(distinct A.user_id) as nbsub,A.list_id FROM `[wysija]user_list` as A LEFT JOIN `[wysija]user` as B on A.user_id=B.user_id WHERE B.status $condit 0 and A.sub_date>0 and A.unsub_date=0 GROUP BY list_id";
458
+
459
+ $total = $modelList->getResults($qry1);
460
+
461
+ foreach ($total as $tot) {
462
+ foreach ($listres as $key => $res) {
463
+ if ($tot['list_id'] == $res['list_id'])
464
+ $listres[$key]['count'] = $tot['nbsub'];
465
+ }
466
+ }
467
+ }
468
+ foreach ($listres as $key => $res) {
469
+ if (!isset($res['count']))
470
+ $listres[$key]['count'] = 0;
471
+ }
472
+ return $listres;
473
+ }
474
+
475
+ function edit($dataPost = false) {
476
+ if (!$this->_checkEmailExists($_REQUEST['id']))
477
+ return;
478
+ $this->add();
479
+
480
+ $modelEmail = WYSIJA::get('email', 'model');
481
+
482
+ $this->data['email'] = $modelEmail->getOne(false, array('email_id' => $_REQUEST['id']));
483
+
484
+ if ($this->data['email']['status'] > 0) {
485
+ $this->redirect();
486
+ }
487
+ $this->title = sprintf(__('Step %1$s', WYSIJA), 1) . ' | ' . $this->data['email']['subject'];
488
+ $modelCamp = WYSIJA::get('campaign', 'model');
489
+ $this->data['campaign'] = $modelCamp->getOne(false, array('campaign_id' => $this->data['email']['campaign_id']));
490
+
491
+ $modelCL = WYSIJA::get('campaign_list', 'model');
492
+ $this->data['campaign_list'] = $modelCL->get(false, array('campaign_id' => $this->data['email']['campaign_id']));
493
+ }
494
+
495
+ function editTemplate() {
496
+ // make sure the editor content is not cached
497
+ header('Cache-Control: no-cache, max-age=0, must-revalidate, no-store'); // HTTP/1.1
498
+ header('Expires: Fri, 9 Mar 1984 00:00:00 GMT');
499
+
500
+ if (!$this->_checkEmailExists($_REQUEST['id']))
501
+ return;
502
+ $this->viewShow = 'editTemplate';
503
+
504
+ wp_enqueue_style('thickbox');
505
+
506
+ $wjEngine = WYSIJA::get('wj_engine', 'helper');
507
+ /* WJ editor translations */
508
+ $this->jsTrans = array_merge($this->jsTrans, $wjEngine->getTranslations(), $wjEngine->getApplicationData());
509
+
510
+ $this->jsTrans['savingnl'] = __('Saving newsletter...', WYSIJA);
511
+ $this->jsTrans['errorsavingnl'] = __('Error Saving newsletter...', WYSIJA);
512
+ $this->jsTrans['savednl'] = __('Newsletter has been saved.', WYSIJA);
513
+ $this->jsTrans['previewemail'] = __('Sending preview...', WYSIJA);
514
+ $this->jsTrans['spamtestresult'] = __('Spam test results', WYSIJA);
515
+
516
+ /* WJ editor JS */
517
+ $this->js[] = 'wysija-editor';
518
+ $this->js[] = 'wysija-admin-ajax-proto';
519
+ $this->js[] = 'wysija-admin-ajax';
520
+ $this->js[] = 'wysija-base-script-64';
521
+ $this->js[] = 'media-upload';
522
+ $this->js['admin-campaigns-editDetails'] = 'admin-campaigns-editDetails';
523
+ $modelEmail = WYSIJA::get('email', 'model');
524
+ $this->data = array();
525
+ $this->data['email'] = $modelEmail->getOne(false, array('email_id' => $_REQUEST['id']));
526
+
527
+ $this->checkIsEditable();
528
+
529
+ $this->viewObj->title = sprintf(__('Second step: "%1$s"', WYSIJA), $this->data['email']['subject']);
530
+ $this->title = sprintf(__('Step %1$s', WYSIJA), 2) . " | " . $this->data['email']['subject'];
531
+
532
+ // check if html source is enabled in the config (this will add the "html source" button in tinymce)
533
+ $model_config = WYSIJA::get('config', 'model');
534
+ $this->jsTrans['html_source_enabled'] = (int) $model_config->getValue('html_source');
535
+ }
536
+
537
+ function checkIsEditable() {
538
+ if (
539
+ !($this->data['email'] == 2 || isset($this->data['email']['params']['schedule']['isscheduled'])) && $this->data['email']['status'] > 0
540
+ ) {
541
+ $this->redirect();
542
+ }
543
+ }
544
+
545
+ function pause() {
546
+ /* pause the campaign entry */
547
+ if (isset($_REQUEST['id']) && $_REQUEST['id']) {
548
+ $modelEmail = WYSIJA::get('email', 'model');
549
+ $myemail = $modelEmail->getOne(false, array('email_id' => $_REQUEST['id']));
550
+ $modelEmail->reset();
551
+ $modelEmail->columns['modified_at']['autoup'] = 1;
552
+ $modelEmail->update(array('status' => -1), array('email_id' => $_REQUEST['id']));
553
+
554
+ if ($myemail['type'] == 2) {
555
+ return $this->redirect('admin.php?page=wysija_campaigns&id=' . $myemail['email_id'] . '&action=edit');
556
+ } else {
557
+ $this->notice(__('Sending is now paused.', WYSIJA));
558
+ }
559
+ }
560
+
561
+ $this->redirect();
562
+ }
563
+
564
+ function resume() {
565
+ /* pause the campaign entry */
566
+ if (isset($_REQUEST['id']) && $_REQUEST['id']) {
567
+ $modelEmail = WYSIJA::get('email', 'model');
568
+ $modelEmail->columns['modified_at']['autoup'] = 1;
569
+ $modelEmail->update(array('status' => 99), array('email_id' => $_REQUEST['id']));
570
+ $this->notice(__('Sending has resumed.', WYSIJA));
571
+ }
572
+
573
+ $this->redirect();
574
+ }
575
+
576
+ function duplicate() {
577
+
578
+ /* 1 - copy the campaign entry */
579
+
580
+ $model = WYSIJA::get('campaign', 'model');
581
+ $query = 'INSERT INTO `[wysija]campaign` (`name`,`description`)
582
+ SELECT concat("' . stripslashes(__('Copy of ', WYSIJA)) . '",`name`),`description` FROM [wysija]campaign
583
+ WHERE campaign_id=' . (int) $_REQUEST['id'];
584
+ $campaignid = $model->query($query);
585
+
586
+ /* 2 - copy the email entry */
587
+ $query = 'INSERT INTO `[wysija]email` (`campaign_id`,`subject`,`body`,`type`,`params`,`wj_data`,`wj_styles`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status`,`created_at`,`modified_at`)
588
+ SELECT ' . $campaignid . ', concat("' . stripslashes(__('Copy of ', WYSIJA)) . '",`subject`),`body`,`type`,`params`,`wj_data`,`wj_styles`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,0,' . time() . ',' . time() . ' FROM [wysija]email
589
+ WHERE email_id=' . (int) $_REQUEST['email_id'];
590
+ $emailid = $model->query($query);
591
+
592
+ //let's reset the count of total childs for auto newsletter
593
+ $mEmail = WYSIJA::get('email', 'model');
594
+ $emailData = $mEmail->getOne(false, array('email_id' => $emailid));
595
+
596
+ if ($emailData['type'] == 2) {
597
+ $paramsReseted = $emailData['params'];
598
+ if (isset($paramsReseted['autonl']['total_child']))
599
+ $paramsReseted['autonl']['total_child'] = 0;
600
+ if (isset($paramsReseted['autonl']['nextSend']))
601
+ $paramsReseted['autonl']['nextSend'] = 0;
602
+ if (isset($paramsReseted['autonl']['firstSend']))
603
+ unset($paramsReseted['autonl']['firstSend']);
604
+ if (isset($paramsReseted['autonl']['lastSend']))
605
+ unset($paramsReseted['autonl']['lastSend']);
606
+ if (isset($paramsReseted['autonl']['articles']['ids']))
607
+ unset($paramsReseted['autonl']['articles']['ids']);
608
+
609
+
610
+ $mEmail->update(array('params' => $paramsReseted), array('email_id' => $emailid));
611
+ }
612
+
613
+ /* 3 - copy the campaign_list entry */
614
+ $query = "INSERT INTO `[wysija]campaign_list` (`campaign_id`,`list_id`,`filter`)
615
+ SELECT $campaignid,`list_id`,`filter` FROM [wysija]campaign_list
616
+ WHERE campaign_id=" . (int) $_REQUEST['id'];
617
+ $model->query($query);
618
+
619
+ $this->notice(__('The newsletter has been duplicated.', WYSIJA));
620
+
621
+ $this->redirect('admin.php?page=wysija_campaigns&id=' . $emailid . '&action=edit');
622
+ }
623
+
624
+ function duplicateEmail() {
625
+ if (!$this->_checkEmailExists($_REQUEST['id']))
626
+ return;
627
+
628
+ $model = WYSIJA::get('campaign', 'model');
629
+ /* 2 - copy the email entry */
630
+ $query = 'INSERT INTO `[wysija]email` (`campaign_id`,`subject`,`body`,`params`,`wj_data`,`wj_styles`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status`,`created_at`,`modified_at`)
631
+ SELECT `campaign_id`, concat("' . stripslashes(__("Copy of ", WYSIJA)) . '",`subject`),`body`,`params`,`wj_data`,`wj_styles`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,0,' . time() . ',' . time() . ' FROM [wysija]email
632
+ WHERE email_id=' . (int) $_REQUEST['id'];
633
+ $emailid = $model->query($query);
634
+
635
+ $this->notice(__('The newsletter has been duplicated.', WYSIJA));
636
+
637
+ $this->redirect('admin.php?page=wysija_campaigns&id=' . $emailid . '&action=edit');
638
+ }
639
+
640
+ function immediateWarning() {
641
+ $model_config = WYSIJA::get('config', 'model');
642
+ $is_multisite = is_multisite();
643
+
644
+ //$is_multisite=true;//PROD comment that line
645
+ if ($is_multisite && $model_config->getValue('sending_method') == 'network') {
646
+ $sending_emails_each = $model_config->getValue('ms_sending_emails_each');
647
+ $number = $model_config->getValue('ms_sending_emails_number');
648
+ } else {
649
+ $sending_emails_each = $model_config->getValue('sending_emails_each');
650
+ $number = $model_config->getValue('sending_emails_number');
651
+ }
652
+
653
+ $formsHelp = WYSIJA::get('forms', 'helper');
654
+
655
+ $timespan = $formsHelp->eachValuesSec[$sending_emails_each];
656
+ $tb = WYSIJA::get('toolbox', 'helper');
657
+
658
+ $this->immediatewarning = str_replace(
659
+ array('[link]', '[/link]', '[settings]'), array('<a href="#">', '</a>', sprintf(__('%1$s emails every %2$s', WYSIJA), $number, trim($tb->duration_string($timespan, true)))), __('Your sending settings ([settings]) can\'t send that quickly to [number] subscribers. Expect delivery delays. [link]Read more[/link]', WYSIJA));
660
+ $this->viewObj->immediatewarning = '<span class="warning-msg" id="immediatewarning">' . $this->immediatewarning . '</span>';
661
+
662
+ $this->jsTrans['immediatewarning'] = $this->immediatewarning;
663
+
664
+ //how many emails can be sent in 12 hours
665
+ //if the frequency is less than 12hours
666
+ if ($timespan < 43200) {
667
+ $ratio = floor(43200 / $timespan);
668
+ $this->jsTrans['possibleemails'] = $ratio * $number;
669
+ } else {
670
+ if ($timespan == 43200) {
671
+ $this->jsTrans['possibleemails'] = $number;
672
+ } else {
673
+ $ratio = floor($timespan / 43200);
674
+ $this->jsTrans['possibleemails'] = $number / $ratio;
675
+ }
676
+ }
677
+ }
678
+
679
+ function editDetails() {
680
+ if (!$this->_checkEmailExists($_REQUEST['id']))
681
+ return;
682
+ $this->viewObj->title = __('Final step: last details', WYSIJA);
683
+ $this->viewShow = 'editDetails';
684
+ $this->js[] = 'wysija-validator';
685
+ $this->jsTrans['previewemail'] = __('Sending preview...', WYSIJA);
686
+ $this->jsTrans['pickadate'] = __('Pick a date', WYSIJA);
687
+ $this->jsTrans['saveclose'] = __('Save & close', WYSIJA);
688
+ $this->jsTrans['sendlater'] = __('Send later', WYSIJA);
689
+
690
+ $this->jsTrans['schedule'] = __('Schedule', WYSIJA);
691
+
692
+ $this->jsTrans['emailCheck'] = WJ_Utils::get_tip_data();
693
+
694
+
695
+ $this->js[] = 'jquery-ui-datepicker';
696
+ $this->js[] = 'wysija-tooltip';
697
+
698
+ $model_list = WYSIJA::get('list', 'model');
699
+ $model_list->limitON = false;
700
+ $this->data = array();
701
+ $this->data['lists'] = $this->__getLists(false, true, true);
702
+
703
+ $model_email = WYSIJA::get('email', 'model');
704
+ $this->data['email'] = $model_email->getOne(false, array('email_id' => $_REQUEST['id']));
705
+
706
+ // The first newsletter, we don't have replyto_email and replyto_name
707
+ if (empty($this->data['email']['replyto_email']) || empty($this->data['email']['replyto_name'])) {
708
+ $current_user = wp_get_current_user();
709
+ $this->data['email']['replyto_email'] = $current_user->data->user_email;
710
+ $this->data['email']['replyto_name'] = $current_user->data->display_name;
711
+ }
712
+
713
+
714
+ if ((int) $this->data['email']['type'] == 2) {
715
+ $this->js['wysija-edit-autonl'] = 'wysija-edit-autonl';
716
+ $this->jsTrans['autonl'] = true;
717
+ $this->immediateWarning();
718
+ $this->jsTrans['send'] = __('Activate now', WYSIJA);
719
+ } else {
720
+ $this->jsTrans['autonl'] = true;
721
+ $this->viewObj->immediatewarning = '';
722
+ $this->jsTrans['send'] = __('Send', WYSIJA);
723
+ }
724
+
725
+ if ((int) $this->data['email']['type'] == 1) {
726
+ $this->jsTrans['alertsend'] = __('You are about to send this newsletter. Please confirm.', WYSIJA);
727
+ } else {
728
+ if (isset($this->data['email']['params']['autonl']['event']) && $this->data['email']['params']['autonl']['event'] == 'subs-2-nl') {
729
+ $this->data['autoresponder'] = 1;
730
+ foreach ($this->data['lists'] as $list) {
731
+ if ($list['list_id'] == $this->data['email']['params']['autonl']['subscribetolist']) {
732
+ break;
733
+ }
734
+ }
735
+
736
+ $this->jsTrans['ignoreprevious'] = sprintf(__('Are you sure you want to ignore the %1$s subscribers of the list %2$s?', WYSIJA), '"' . $list['count'] . '"', '"' . $list['name'] . '"');
737
+ }
738
+ }
739
+
740
+ $this->checkIsEditable();
741
+
742
+ $this->title = sprintf(__('Step %1$s', WYSIJA), 3) . " | " . $this->data['email']['subject'];
743
+ $this->dataAutoNl();
744
+
745
+ $this->jsLoc['wysija-edit-autonl']['autofields'] = $this->data['autonl']['fields'];
746
+
747
+ $modelCL = WYSIJA::get('campaign_list', 'model');
748
+ $this->data['campaign_list'] = $modelCL->get(false, array('campaign_id' => $this->data['email']['campaign_id']));
749
+ }
750
+
751
+ function delete() {
752
+ $this->requireSecurity();
753
+
754
+ $campaign_ids = array();
755
+ if(isset($_REQUEST['id'])) $campaign_ids[] = $_REQUEST['id'];
756
+ if(isset($_REQUEST['campaign']['campaign_id'])) $campaign_ids[] = $_REQUEST['campaign']['campaign_id'];
757
+ if(isset($_REQUEST['wysija']['campaign']['campaign_id'][0])){
758
+ $campaign_ids = array_merge($campaign_ids, $_REQUEST['wysija']['campaign']['campaign_id']);
759
+ }
760
+
761
+ if (!empty($campaign_ids)) {
762
+ foreach($campaign_ids as $campaign_id){
763
+ if($campaign_id > 0){
764
+ $model_campaign = WYSIJA::get('campaign', 'model');
765
+ $model_campaign->delete(array('campaign_id' => $campaign_id));
766
+
767
+ $model_campaign_list = WYSIJA::get('campaign_list', 'model');
768
+ $model_campaign_list->delete(array('campaign_id' => $campaign_id));
769
+
770
+ $modelEmail = WYSIJA::get('email', 'model');
771
+ $modelEmail->delete(array('campaign_id' => $campaign_id));
772
+ }
773
+ }
774
+
775
+ $this->notice(_n(__('Newsletter deleted.', WYSIJA), __('Newsletters deleted.', WYSIJA), count($campaign_ids), WYSIJA));
776
+ } else {
777
+ $this->notice(__('Newsletter can\'t be deleted.', WYSIJA));
778
+ }
779
+
780
+ // retrieve saved filter
781
+ if (!empty($_REQUEST['action']))
782
+ unset($_REQUEST['action']);
783
+ if (!empty($_REQUEST['id']))
784
+ unset($_REQUEST['id']);
785
+ if (!empty($_REQUEST['_wpnonce']))
786
+ unset($_REQUEST['_wpnonce']);
787
+ $redirect = $this->base_url . '?' . http_build_query($_REQUEST);
788
+ $this->redirect($redirect);
789
+ }
790
+
791
+ /**
792
+ *
793
+ * this function is to delete an email that belongs to a campaign
794
+ * when we have a post notification all emails belong to the same campaign
795
+ * we don't want to delete an entire campaign when we delete a single email
796
+ */
797
+ function deleteEmail(){
798
+ $this->requireSecurity();
799
+ if(!$this->_checkEmailExists($_REQUEST['id'])) return;
800
+
801
+ if(isset($_REQUEST['id'])){
802
+ $modelEmail=WYSIJA::get('email','model');
803
+ $modelEmail->delete(array('email_id'=>$_REQUEST['id']));
804
+ $this->notice(__('Newsletter deleted.',WYSIJA));
805
+ }else{
806
+ $this->notice(__('Newsletter can\'t be deleted.',WYSIJA));
807
+ }
808
+
809
+ $this->redirect();
810
+ }
811
+
812
+ function savecamp() {
813
+ $this->redirectAfterSave = false;
814
+
815
+ /* update email */
816
+ $data = array();
817
+
818
+ // in case the newsletter already exists
819
+ if (isset($_REQUEST['id'])) {
820
+ $modelEmail = WYSIJA::get('email', 'model');
821
+ $modelEmail->fieldValid = false;
822
+ $emaildataarr = $modelEmail->getOne(false, array('email_id' => $_REQUEST['id']));
823
+
824
+ $model_campaign = WYSIJA::get('campaign', 'model');
825
+ $model_campaign->update(array('name' => $_POST['wysija']['email']['subject'], 'description' => ''), array('campaign_id' => $emaildataarr['campaign_id']));
826
+
827
+ $campaign_id = $emaildataarr['campaign_id'];
828
+ $email_id = $emaildataarr['email_id'];
829
+ $dataEmail = array(
830
+ 'campaign_id' => $campaign_id,
831
+ 'subject' => $_POST['wysija']['email']['subject'],
832
+ 'type' => $_POST['wysija']['email']['type']);
833
+
834
+ if((int)$dataEmail['type'] === 2) {
835
+ // set autonl params
836
+ $dataEmail['params'] = array('autonl' => $_POST['wysija']['email']['params']['autonl']);
837
+
838
+ // WTF?
839
+ if(!isset($newparams['autonl']['unique_send'])) {
840
+ unset($dataEmail['params']['autonl']['unique_send']);
841
+ } else {
842
+ $dataEmail['params']['autonl']['unique_send'] = true;
843
+ }
844
+ }
845
+
846
+ // check if the newsletter used to be an automatic newsletter (if )
847
+ if((int)$dataEmail['type'] === 1 && (int)$emaildataarr['type'] === 2) {
848
+ // make sure we remove any kind of auto-post block
849
+ $helper_autonews = WYSIJA::get('autonews', 'helper');
850
+ $updated_email_data = $helper_autonews->remove_autopost_blocks($emaildataarr['wj_data']);
851
+ if($updated_email_data !== false) {
852
+ // if the email data has been changed, make sure to update it in the DB
853
+ $dataEmail['wj_data'] = $updated_email_data;
854
+ }
855
+ }
856
+
857
+ $modelEmail->columns['modified_at']['autoup'] = 1;
858
+ $modelEmail->debugupdate = true;
859
+ $dataEmail['email_id'] = $_REQUEST['id'];
860
+
861
+ if(isset($_REQUEST['save-reactivate'])) {
862
+ //if the button save and reactivate has been clicked then we reactivate and redirect to the newsletter page
863
+ $dataEmail['status'] = 99;
864
+ $_REQUEST['return'] = 1;
865
+ }
866
+
867
+ $data['email']['email_id'] = $modelEmail->update($dataEmail, array('email_id' => (int)$_REQUEST['id']));
868
+ } else {
869
+ // get default theme
870
+ $model_config = WYSIJA::get('config', 'model');
871
+ $default_theme = $model_config->getValue('newsletter_default_theme', 'default');
872
+
873
+ $helper_themes = WYSIJA::get('themes', 'helper');
874
+ $theme_data = $helper_themes->getData($default_theme);
875
+
876
+ // get campaign data
877
+ $model_campaign = WYSIJA::get('campaign', 'model');
878
+ $campaign_id = $model_campaign->insert(array('name' => $_POST['wysija']['email']['subject'], 'description' => ''));
879
+
880
+ $modelEmail = WYSIJA::get('email', 'model');
881
+ $modelEmail->fieldValid = false;
882
+ $emaildata = array(
883
+ 'campaign_id' => $campaign_id,
884
+ 'subject' => $_POST['wysija']['email']['subject'],
885
+ 'type' => (int) $_POST['wysija']['email']['type']
886
+ );
887
+
888
+
889
+ // create autonl parameters if necessary
890
+ if ((int) $_POST['wysija']['email']['type'] === 2 && isset($_POST['wysija']['email']['params']['autonl'])) {
891
+ $emaildata['params'] = array('autonl' => $_POST['wysija']['email']['params']['autonl']);
892
+ }
893
+
894
+ // create sample data depending on newsletter's type
895
+ if ((int) $_POST['wysija']['email']['type'] === 2 && $_POST['wysija']['email']['params']['autonl']['event'] === 'new-articles') {
896
+
897
+ // if immediate, post_limit is set to 1
898
+ if ($emaildata['params']['autonl']['when-article'] === 'immediate') {
899
+ $autopostParams = array(
900
+ array('key' => 'category_ids', 'value' => null),
901
+ array('key' => 'title_tag', 'value' => 'h2'),
902
+ array('key' => 'title_alignment', 'value' => 'left'),
903
+ array('key' => 'title_position', 'value' => 'inside'),
904
+ array('key' => 'image_alignment', 'value' => 'alternate'),
905
+ array('key' => 'image_width', 'value' => 325),
906
+ array('key' => 'post_content', 'value' => 'excerpt'),
907
+ array('key' => 'readmore', 'value' => base64_encode(__('Read more.', WYSIJA))),
908
+ array('key' => 'show_divider', 'value' => 'yes'),
909
+ array('key' => 'post_limit', 'value' => 1),
910
+ array('key' => 'post_type', 'value' => 'post'),
911
+ array('key' => 'author_show', 'value' => 'no'),
912
+ array('key' => 'author_label', 'value' => base64_encode(__('Author:', WYSIJA))),
913
+ array('key' => 'category_show', 'value' => 'no'),
914
+ array('key' => 'category_label', 'value' => base64_encode(__('Categories:', WYSIJA))),
915
+ array('key' => 'nopost_message', 'value' => base64_encode(__('Latest content already sent.', WYSIJA))),
916
+ array('key' => 'bgcolor1', 'value' => null),
917
+ array('key' => 'bgcolor2', 'value' => null),
918
+ array('key' => 'sort_by', 'value' => 'newest')
919
+ );
920
+ } else {
921
+ $autopostParams = array(
922
+ array('key' => 'category_ids', 'value' => null),
923
+ array('key' => 'title_tag', 'value' => 'h2'),
924
+ array('key' => 'title_alignment', 'value' => 'left'),
925
+ array('key' => 'title_position', 'value' => 'inside'),
926
+ array('key' => 'image_alignment', 'value' => 'alternate'),
927
+ array('key' => 'image_width', 'value' => 325),
928
+ array('key' => 'post_content', 'value' => 'excerpt'),
929
+ array('key' => 'readmore', 'value' => base64_encode(__('Read more.', WYSIJA))),
930
+ array('key' => 'show_divider', 'value' => 'yes'),
931
+ array('key' => 'post_limit', 'value' => 2),
932
+ array('key' => 'post_type', 'value' => 'post'),
933
+ array('key' => 'author_show', 'value' => 'no'),
934
+ array('key' => 'author_label', 'value' => base64_encode(__('Author:', WYSIJA))),
935
+ array('key' => 'category_show', 'value' => 'no'),
936
+ array('key' => 'category_label', 'value' => base64_encode(__('Categories:', WYSIJA))),
937
+ array('key' => 'nopost_message', 'value' => base64_encode(__('Latest content already sent.', WYSIJA))),
938
+ array('key' => 'bgcolor1', 'value' => null),
939
+ array('key' => 'bgcolor2', 'value' => null),
940
+ array('key' => 'sort_by', 'value' => 'newest')
941
+ );
942
+ }
943
+
944
+ // sample data for post notifications
945
+ $newwjdata = array(
946
+ 'version' => WYSIJA::get_version(),
947
+ 'header' => array(
948
+ 'text' => NULL,
949
+ 'image' => array(
950
+ 'src' => WYSIJA_EDITOR_IMG . 'transparent.png',
951
+ 'width' => 600,
952
+ 'height' => 86,
953
+ 'alignment' => 'center',
954
+ 'static' => true,
955
+ ),
956
+ 'alignment' => 'center',
957
+ 'static' => true,
958
+ 'type' => 'header'
959
+ ),
960
+ 'body' => array(
961
+ 'block-1' => array(
962
+ 'text' => array(
963
+ 'value' => '<h3 class="align-right">' . sprintf(__("The posts below were added with the widget %sAutomatic latest content%s", WYSIJA), '<strong>', '</strong>') . '</h3>'
964
+ ),
965
+ 'image' => array(
966
+ 'src' => WYSIJA_EDITOR_IMG . 'default-newsletter/autonewsletter/arrow-up.png',
967
+ 'width' => 45,
968
+ 'height' => 45,
969
+ 'alignment' => 'right',
970
+ 'static' => false
971
+ ),
972
+ 'alignment' => 'right',
973
+ 'static' => false,
974
+ 'position' => '1',
975
+ 'type' => 'content'
976
+ ),
977
+ 'block-2' => array(
978
+ 'text' => array(
979
+ 'value' => '<h3>' . sprintf(__('%sTo edit%s, mouse over to show edit button below.', WYSIJA), '<strong>', '</strong>') . '</h3>'
980
+ ),
981
+ 'image' => array(
982
+ 'src' => WYSIJA_EDITOR_IMG . 'default-newsletter/autonewsletter/arrow-down.png',
983
+ 'width' => 150,
984
+ 'height' => 53,
985
+ 'alignment' => 'left',
986
+ 'static' => false
987
+ ),
988
+ 'alignment' => 'left',
989
+ 'static' => false,
990
+ 'position' => '2',
991
+ 'type' => 'content'
992
+ ),
993
+ 'block-3' => array(
994
+ 'params' => $autopostParams,
995
+ 'position' => '3',
996
+ 'type' => 'auto-post'
997
+ )
998
+ ),
999
+ 'footer' => array(
1000
+ 'text' => NULL,
1001
+ 'image' => array(
1002
+ 'src' => WYSIJA_EDITOR_IMG . 'transparent.png',
1003
+ 'width' => 600,
1004
+ 'height' => 86,
1005
+ 'alignment' => 'center',
1006
+ 'static' => true,
1007
+ ),
1008
+ 'alignment' => 'center',
1009
+ 'static' => true,
1010
+ 'type' => 'footer'
1011
+ )
1012
+ );
1013
+ } else {
1014
+ if(!isset($emaildata['params'])) {
1015
+ $emaildata['params'] = array();
1016
+ }
1017
+
1018
+ $emaildata['params']['quickselection'] = array(
1019
+ 'wp-301' => array(
1020
+ 'identifier' => 'wp-301',
1021
+ 'width' => 281,
1022
+ 'height' => 190,
1023
+ 'url' => WYSIJA_EDITOR_IMG . 'default-newsletter/newsletter/pigeon.png',
1024
+ 'thumb_url' => WYSIJA_EDITOR_IMG . 'default-newsletter/newsletter/pigeon-150x150.png'
1025
+ )
1026
+ );
1027
+
1028
+ if($theme_data['divider'] === null) {
1029
+ // default theme does not exist anymore or there is no divider associated to the theme
1030
+ // we need to get the default divider in this case
1031
+ $helper_dividers = WYSIJA::get('dividers', 'helper');
1032
+ $default_divider = $helper_dividers->getDefault();
1033
+ } else {
1034
+ // set default divider
1035
+ $default_divider = $theme_data['divider'];
1036
+ }
1037
+ // set default divider in email parameters
1038
+ $emaildata['params']['divider'] = $default_divider;
1039
+
1040
+ // get bookmarks from iconset 2
1041
+ $helper_bookmarks = WYSIJA::get('bookmarks', 'helper');
1042
+ $bookmarks = $helper_bookmarks->getAllByIconset('medium', '02');
1043
+
1044
+ // sample data for regular newsletter
1045
+ $newwjdata = array(
1046
+ 'version' => WYSIJA::get_version(),
1047
+ 'header' => array(
1048
+ 'text' => null,
1049
+ 'image' => array(
1050
+ // 'src' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/header.png',
1051
+ 'src' => WYSIJA_EDITOR_IMG . 'transparent.png',
1052
+ 'width' => 600,
1053
+ 'height' => 86,
1054
+ 'alignment' => 'center',
1055
+ 'static' => true
1056
+ ),
1057
+ 'alignment' => 'center',
1058
+ 'static' => true,
1059
+ 'type' => 'header'
1060
+ ),
1061
+ 'body' => array(
1062
+ 'block-1' => array(
1063
+ 'text' => array(
1064
+ 'value' => '<h2><strong>' . __('Step 1:', WYSIJA) . '</strong> ' . __('hey, click on this text!', WYSIJA) . '</h2>' . '<p>' . __('To edit, simply click on this block of text.', WYSIJA) . '</p>'
1065
+ ),
1066
+ 'image' => null,
1067
+ 'alignment' => 'left',
1068
+ 'static' => false,
1069
+ 'position' => 1,
1070
+ 'type' => 'content'
1071
+ ),
1072
+ 'block-2' => array_merge(array(
1073
+ 'position' => 2,
1074
+ 'type' => 'divider'
1075
+ ),
1076
+ $default_divider
1077
+ ),
1078
+ 'block-3' => array(
1079
+ 'text' => array(
1080
+ 'value' => '<h2><strong>' . __('Step 2:', WYSIJA) . '</strong> ' . __('play with this image', WYSIJA) . '</h2>'
1081
+ ),
1082
+ 'image' => null,
1083
+ 'alignment' => 'left',
1084
+ 'static' => false,
1085
+ 'position' => 3,
1086
+ 'type' => 'content'
1087
+ ),
1088
+ 'block-4' => array(
1089
+ 'text' => array(
1090
+ 'value' => '<p>' . __('Position your mouse over the image to the left.', WYSIJA) . '</p>'
1091
+ ),
1092
+ 'image' => array(
1093
+ 'src' => WYSIJA_EDITOR_IMG . 'default-newsletter/newsletter/pigeon.png',
1094
+ 'width' => 281,
1095
+ 'height' => 190,
1096
+ 'alignment' => 'left',
1097
+ 'static' => false
1098
+ ),
1099
+ 'alignment' => 'left',
1100
+ 'static' => false,
1101
+ 'position' => 4,
1102
+ 'type' => 'content'
1103
+ ),
1104
+ 'block-5' => array_merge(array(
1105
+ 'position' => 5,
1106
+ 'type' => 'divider'
1107
+ ), $default_divider
1108
+ ),
1109
+ 'block-6' => array(
1110
+ 'text' => array(
1111
+ 'value' => '<h2><strong>' . __('Step 3:', WYSIJA) . '</strong> ' . __('drop content here', WYSIJA) . '</h2>' .
1112
+ '<p>' . sprintf(__('Drag and drop %1$stext, posts, dividers.%2$s Look on the right!', WYSIJA), '<strong>', '</strong>') . '</p>' .
1113
+ '<p>' . sprintf(__('You can even %1$ssocial bookmarks%2$s like these:', WYSIJA), '<strong>', '</strong>') . '</p>'
1114
+ ),
1115
+ 'image' => null,
1116
+ 'alignment' => 'left',
1117
+ 'static' => false,
1118
+ 'position' => 6,
1119
+ 'type' => 'content'
1120
+ ),
1121
+ 'block-7' => array(
1122
+ 'width' => 184,
1123
+ 'alignment' => 'center',
1124
+ 'items' => array(
1125
+ array_merge(array(
1126
+ 'url' => 'http://www.facebook.com/mailpoetplugin',
1127
+ 'alt' => 'Facebook',
1128
+ 'cellWidth' => 61,
1129
+ 'cellHeight' => 32
1130
+ ), $bookmarks['facebook']),
1131
+ array_merge(array(
1132
+ 'url' => 'http://www.twitter.com/mail_poet',
1133
+ 'alt' => 'Twitter',
1134
+ 'cellWidth' => 61,
1135
+ 'cellHeight' => 32
1136
+ ), $bookmarks['twitter']),
1137
+ array_merge(array(
1138
+ 'url' => 'https://plus.google.com/+Mailpoet',
1139
+ 'alt' => 'Google',
1140
+ 'cellWidth' => 61,
1141
+ 'cellHeight' => 32
1142
+ ), $bookmarks['google'])
1143
+ ),
1144
+ 'position' => 7,
1145
+ 'type' => 'gallery'
1146
+ ),
1147
+ 'block-8' => array_merge(array(
1148
+ 'position' => 8,
1149
+ 'type' => 'divider'
1150
+ ), $default_divider
1151
+ ),
1152
+ 'block-9' => array(
1153
+ 'text' => array(
1154
+ 'value' => '<h2><strong>' . __('Step 4:', WYSIJA) . '</strong> ' . __('and the footer?', WYSIJA) . '</h2>' .
1155
+ '<p>' . sprintf(__('Change the footer\'s content in MailPoet\'s %1$sSettings%2$s page.', WYSIJA), '<strong>', '</strong>') . '</p>'
1156
+ ),
1157
+ 'image' => null,
1158
+ 'alignment' => 'left',
1159
+ 'static' => false,
1160
+ 'position' => 9,
1161
+ 'type' => 'content'
1162
+ )
1163
+ ),
1164
+ 'footer' => array(
1165
+ 'text' => NULL,
1166
+ 'image' => array(
1167
+ // 'src' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/footer.png',
1168
+ 'src' => WYSIJA_EDITOR_IMG . 'transparent.png',
1169
+ 'width' => 600,
1170
+ 'height' => 86,
1171
+ 'alignment' => 'center',
1172
+ 'static' => true,
1173
+ ),
1174
+ 'alignment' => 'center',
1175
+ 'static' => true,
1176
+ 'type' => 'footer'
1177
+ )
1178
+ );
1179
+ }
1180
+
1181
+ // set default styles
1182
+ $helper_engine = WYSIJA::get('wj_engine', 'helper');
1183
+ $styles = $helper_engine->getDefaultStyles();
1184
+ // end - set default styles
1185
+
1186
+ // set theme specific data
1187
+ if($theme_data['header'] !== null) {
1188
+ $newwjdata['header'] = $theme_data['header'];
1189
+ }
1190
+ if($theme_data['footer'] !== null) {
1191
+ $newwjdata['footer'] = $theme_data['footer'];
1192
+ }
1193
+ if($theme_data['divider'] !== null) {
1194
+ $newwjdata['widgets'] = array('divider' => $theme_data['divider']);
1195
+ }
1196
+ // end - set theme specific data
1197
+
1198
+ $emaildata['wj_data'] = base64_encode(serialize($newwjdata));
1199
+ $emaildata['wj_styles'] = base64_encode(serialize($styles));
1200
+
1201
+ $email_id = $data['email']['email_id'] = $modelEmail->insert($emaildata);
1202
+
1203
+ $this->notice(__('Newsletter successfully created.', WYSIJA));
1204
+ }
1205
+
1206
+ $this->_saveLists($campaign_id, true);
1207
+
1208
+ if (isset($_REQUEST['return']))
1209
+ $this->redirect();
1210
+ else {
1211
+ $this->redirect('admin.php?page=wysija_campaigns&action=editTemplate&id=' . $email_id);
1212
+ }
1213
+ }
1214
+
1215
+ function saveemail() {
1216
+ $this->redirectAfterSave = false;
1217
+ //$this->requireSecurity();
1218
+ $modelEmail = WYSIJA::get("email", "model");
1219
+ $modelEmail->fieldValid = false;
1220
+ $emaildataarr = $modelEmail->getOne(array('email_id' => $_REQUEST['id']));
1221
+
1222
+ if (isset($_REQUEST['save-reactivate'])) {
1223
+ //if the button save and reactivate has been clicked then we reactivate and redirect to the newsletter page
1224
+ $dataEmail['status'] = 99;
1225
+ $_REQUEST['return'] = 1;
1226
+ }
1227
+
1228
+ if (isset($_REQUEST['return']))
1229
+ $this->redirect();
1230
+ else {
1231
+ $this->redirect('admin.php?page=wysija_campaigns&action=editDetails&id=' . $emaildataarr['email_id']);
1232
+ }
1233
+ }
1234
+
1235
+ function savelast() {
1236
+ $this->redirectAfterSave = false;
1237
+ $post_notification = false;
1238
+ $this->requireSecurity();
1239
+
1240
+ if (!isset($_POST['wysija']['email']['from_name']) || !isset($_POST['wysija']['email']['from_email']) || !isset($_POST['wysija']['email']['replyto_name']) || !isset($_POST['wysija']['email']['replyto_email'])) {
1241
+ $this->error(__('Information is missing.', WYSIJA));
1242
+ return $this->editDetails();
1243
+ }
1244
+ if (isset($_REQUEST['wysija']['email']['params']['googletrackingcode']) && $_REQUEST['wysija']['email']['params']['googletrackingcode'] &&
1245
+ (!is_string($_REQUEST['wysija']['email']['params']['googletrackingcode']) OR
1246
+ preg_match('#[^a-z0-9_\-\s]#i', $_REQUEST['wysija']['email']['params']['googletrackingcode']) !== 0 )) {
1247
+ //force to simple text
1248
+ $_REQUEST['wysija']['email']['params']['googletrackingcode'] = preg_replace('#[^a-z0-9_\-\s]#i', '_', $_REQUEST['wysija']['email']['params']['googletrackingcode']);
1249
+ $this->error(__('Your Google Campaign can only contain latin characters, numbers, spaces and hyphens!', WYSIJA), 1);
1250
+ return $this->editDetails();
1251
+ }
1252
+
1253
+ $update_email = array(
1254
+ 'email_id' => $_POST['wysija']['email']['email_id'],
1255
+ 'from_name' => $_POST['wysija']['email']['from_name'],
1256
+ 'from_email' => $_POST['wysija']['email']['from_email'],
1257
+ 'replyto_name' => $_POST['wysija']['email']['replyto_name'],
1258
+ 'replyto_email' => $_POST['wysija']['email']['replyto_email'],
1259
+ 'subject' => $_POST['wysija']['email']['subject'],
1260
+ );
1261
+ $model_email = WYSIJA::get('email', 'model');
1262
+ if (isset($_POST['wysija']['email']['params']))
1263
+ $update_email['params'] = $_POST['wysija']['email']['params'];
1264
+
1265
+ //insert into campaigns lists
1266
+ $this->_saveLists($_POST['wysija']['campaign']['campaign_id']);
1267
+ $email_data = $model_email->getOne($_POST['wysija']['email']['email_id']);
1268
+
1269
+ // if we just save the draf we don't go through the big sending process setup
1270
+ if (isset($_POST['submit-draft']) || isset($_POST['submit-pause']) || (isset($_REQUEST['wj_redir']) && $_REQUEST['wj_redir'] == 'savelastback')) {
1271
+ if (isset($_POST['wysija']['email']['params']['schedule']['isscheduled']))
1272
+ $this->notice(__('Newsletter has been scheduled.', WYSIJA));
1273
+ else
1274
+ $this->notice(__('Newsletter has been saved as a draft.', WYSIJA));
1275
+ }else {
1276
+ // we update the param attribute with what's has been posted
1277
+ foreach ($update_email as $ki => $vi) {
1278
+ if ($ki == 'params') {
1279
+ foreach ($vi as $parake => $paraval) {
1280
+ $email_data['params'][$parake] = $paraval;
1281
+ }
1282
+ $update_email[$ki] = $email_data[$ki];
1283
+ }
1284
+ else
1285
+ $email_data[$ki] = $vi;
1286
+ }
1287
+
1288
+ // if the checkbox to ignore retroactivity is here we just tell the class
1289
+ if (isset($_POST['wysija']['email']['ignore_subscribers'])) {
1290
+ $model_email->retro_active_autoresponders = false;
1291
+ }
1292
+
1293
+ // activate or send the email depending on the typ
1294
+ $model_email->send_activate($email_data);
1295
+ }
1296
+
1297
+ // update email
1298
+ $update_email['type'] = $email_data['type'];
1299
+
1300
+ if ($post_notification) {
1301
+ $helper_autonews = WYSIJA::get('autonews', 'helper');
1302
+ $update_email['params']['autonl']['nextSend'] = $helper_autonews->getNextSend($update_email);
1303
+ }
1304
+
1305
+ $model_email->reset();
1306
+ $model_email->columns['modified_at']['autoup'] = 1;
1307
+
1308
+ // update some fields of the email
1309
+ $model_email->update($update_email);
1310
+
1311
+ // update the campaign subject which ispretty much useless but good to keep in sync with the email
1312
+ $model_campaign = WYSIJA::get('campaign', 'model');
1313
+ $model_campaign->reset();
1314
+ $update_campaign = array('campaign_id' => $_REQUEST['id'], 'name' => $_POST['wysija']['email']['subject']);
1315
+ $model_campaign->update($update_campaign);
1316
+
1317
+ if (isset($_REQUEST['wj_redir']) && $_REQUEST['wj_redir'] == 'savelastback') {
1318
+ return $this->redirect('admin.php?page=wysija_campaigns&action=editTemplate&id=' . $_POST['wysija']['email']['email_id']);
1319
+ }
1320
+ else
1321
+ return $this->redirect();
1322
+ }
1323
+
1324
+ function _saveLists($campaignId, $flagup = false) {
1325
+ //record the list that we have in that campaign
1326
+ $modelCampL = WYSIJA::get('campaign_list', 'model');
1327
+ if ($flagup || (int) $campaignId > 0) {
1328
+ $modelCampL->delete(array('equal' => array('campaign_id' => $campaignId)));
1329
+ $modelCampL->reset();
1330
+ }
1331
+
1332
+ if (isset($_POST['wysija']['campaign_list']['list_id'])) {
1333
+ //$modelCampL=WYSIJA::get("campaign_list","model");
1334
+ foreach ($_POST['wysija']['campaign_list']['list_id'] as $listid) {
1335
+ $modelCampL->insert(array('campaign_id' => $campaignId, "list_id" => $listid));
1336
+ }
1337
+ }
1338
+ }
1339
+
1340
+ function _addLinkFilter($status, $type = 'status') {
1341
+ switch ($type) {
1342
+ case 'status':
1343
+ switch ($status) {
1344
+ case 'draft':
1345
+ $this->filters['equal'] = array('status' => 0);
1346
+ break;
1347
+ case 'sending':
1348
+ $this->filters['equal'] = array('status' => 99);
1349
+ break;
1350
+ case 'sent':
1351
+ $this->filters['equal'] = array('status' => 2);
1352
+ break;
1353
+ case 'paused':
1354
+ $this->filters['equal'] = array('status' => -1);
1355
+ break;
1356
+ case 'scheduled':
1357
+ $this->filters['equal'] = array('status' => 4);
1358
+ break;
1359
+ }
1360
+ break;
1361
+ case 'type':
1362
+ switch ($status) {
1363
+ case 'regular':
1364
+ $this->filters['equal'] = array('type' => 1);
1365
+ break;
1366
+ case 'autonl':
1367
+ $this->filters['equal'] = array('type' => 2);
1368
+ break;
1369
+ }
1370
+ break;
1371
+ }
1372
+ }
1373
+
1374
+ /**
1375
+ * Get all campaigns, based on the filters
1376
+ */
1377
+ protected function get_campaigns() {
1378
+ $order_by = ' ORDER BY ';
1379
+
1380
+ if (isset($_REQUEST['orderby'])) {
1381
+ if (!is_string($_REQUEST['orderby']) OR preg_match('|[^a-z0-9#_.-]|i', $_REQUEST['orderby']) !== 0) {
1382
+ $_REQUEST['orderby'] = '';
1383
+ }
1384
+ if (!in_array(strtoupper($_REQUEST['ordert']), array('DESC', 'ASC')))
1385
+ $_REQUEST['ordert'] = 'DESC';
1386
+ $order_by.=$_REQUEST['orderby'] . ' ' . $_REQUEST['ordert'];
1387
+ }else {
1388
+ $order_by.='FIELD(B.status, 99,3,1,0,2), ';
1389
+ $order_by.='B.status desc, ';
1390
+ $order_by.='B.modified_at desc, ';
1391
+ $order_by.='B.sent_at desc, ';
1392
+ $order_by.='B.type desc, ';
1393
+ $order_by.='A.' . $this->modelObj->getPk() . ' DESC';
1394
+ }
1395
+
1396
+ $query = '
1397
+ SELECT
1398
+ A.`campaign_id`,
1399
+ A.`name` as `campaign_name`,
1400
+ B.`subject` as `name`,
1401
+ A.`description`,
1402
+ B.`params`,
1403
+ B.`type`,
1404
+ B.`number_sent`,
1405
+ B.`number_opened`,
1406
+ B.`number_clicked`,
1407
+ B.`number_unsub`,
1408
+ (B.`number_sent` +
1409
+ B.`number_opened` +
1410
+ B.`number_clicked` +
1411
+ B.`number_unsub` +
1412
+ B.`number_bounce` +
1413
+ B.`number_forward`
1414
+ ) AS `number_total`,
1415
+ B.`status`,
1416
+ B.`created_at`,
1417
+ B.`modified_at`,
1418
+ B.`sent_at`,
1419
+ B.`email_id`
1420
+ FROM
1421
+ `[wysija]' . $this->modelObj->table_name . '` AS A
1422
+ LEFT JOIN
1423
+ `[wysija]email` AS B on A.`campaign_id` = B.`campaign_id`
1424
+ LEFT JOIN
1425
+ `[wysija]campaign_list` AS C on A.`campaign_id` = C.`campaign_id`';
1426
+ $campaigns = $this->modelObj->getResults($query . $this->modelObj->makeWhere() . ' GROUP BY B.email_id' . $order_by . $this->modelObj->setLimit());
1427
+
1428
+ // calculate percetange of open / click / unsubscribe
1429
+ $helper_numbers = WYSIJA::get('numbers', 'helper');
1430
+ foreach ($campaigns as &$campaign) {
1431
+ // open rate, based on sent number
1432
+ $campaign['rate_opened'] = $helper_numbers->calculate_percetage($campaign['number_opened'], $campaign['number_total'], 1);
1433
+ // click rate, based on opened number
1434
+ $campaign['rate_clicked'] = $helper_numbers->calculate_percetage($campaign['number_clicked'], $campaign['number_total'], 1);
1435
+ // unsubscribe rate, based on opened number
1436
+ $campaign['rate_unsub'] = $helper_numbers->calculate_percetage($campaign['number_unsub'], $campaign['number_total'], 1);
1437
+ }
1438
+ return $campaigns;
1439
+ }
1440
+
1441
+ /**
1442
+ * Get the first campaign in history
1443
+ */
1444
+ protected function get_oldest_compaign() {
1445
+ $query = '
1446
+ SELECT
1447
+ MIN(B.created_at) as datemin
1448
+ FROM `[wysija]' . $this->modelObj->table_name . '` as A
1449
+ LEFT JOIN
1450
+ `[wysija]email` AS B on A.campaign_id = B.campaign_id
1451
+ LEFT JOIN
1452
+ `[wysija]campaign_list` as C on A.campaign_id = C.campaign_id';
1453
+
1454
+ return $this->modelObj->query('get_row', $query . $this->modelObj->makeWhere());
1455
+ }
1456
+
1457
+ /**
1458
+ * Count ALL emails of each email-status
1459
+ * @todo: move to model
1460
+ */
1461
+ protected function count_emails_by_status() {
1462
+ $query = '
1463
+ SELECT
1464
+ COUNT(`email_id`) AS `campaigns`,
1465
+ `status`
1466
+ FROM
1467
+ `[wysija]email`
1468
+ WHERE
1469
+ `campaign_id` > 0
1470
+ GROUP BY `status`';
1471
+
1472
+ $countss = $this->modelObj->query('get_res', $query);
1473
+ $counts = array();
1474
+ $total = 0;
1475
+
1476
+ foreach ($countss as $count) {
1477
+ switch ($count['status']) {
1478
+ case '0':
1479
+ $type = 'draft';
1480
+ break;
1481
+ case '1':
1482
+ case '3':
1483
+ case '99':
1484
+ $type = 'sending';
1485
+ break;
1486
+ case '2':
1487
+ $type = 'sent';
1488
+ break;
1489
+ case '-1':
1490
+ $type = 'paused';
1491
+ break;
1492
+ case '4':
1493
+ $type = 'scheduled';
1494
+ break;
1495
+ }
1496
+ $total = $total + $count['campaigns'];
1497
+ $counts['status-' . $type] = $count['campaigns'];
1498
+ }
1499
+ return $counts;
1500
+ }
1501
+
1502
+ /**
1503
+ * Count emails which matched the filters
1504
+ */
1505
+ protected function count_emails() {
1506
+ $query = '
1507
+ SELECT
1508
+ COUNT(DISTINCT B.`email_id`) AS `campaigns`
1509
+ FROM
1510
+ `[wysija]' . $this->modelObj->table_name . '` AS A
1511
+ LEFT JOIN
1512
+ `[wysija]email` AS B ON A.`campaign_id` = B.`campaign_id`
1513
+ LEFT JOIN
1514
+ `[wysija]campaign_list` AS C ON A.`campaign_id` = C.`campaign_id`';
1515
+ return $this->modelObj->count($query . $this->modelObj->makeWhere(), 'campaigns');
1516
+ }
1517
+
1518
+ /**
1519
+ * Count ALL emails of each type of email
1520
+ */
1521
+ protected function count_emails_by_type() {
1522
+ $query = '
1523
+ SELECT
1524
+ COUNT(`email_id`) AS `campaigns`,
1525
+ `type`
1526
+ FROM
1527
+ `[wysija]email`
1528
+ WHERE
1529
+ `campaign_id` > 0
1530
+ GROUP BY `type`';
1531
+ $countss = $this->modelObj->query('get_res', $query, ARRAY_A);
1532
+ $counts = array();
1533
+ foreach ($countss as $count) {
1534
+ switch ($count['type']) {
1535
+ case '1':
1536
+ $type = 'regular';
1537
+ break;
1538
+ case '2':
1539
+ $type = 'autonl';
1540
+ break;
1541
+ }
1542
+ $counts['type-' . $type] = $count['campaigns'];
1543
+ }
1544
+ return $counts;
1545
+ }
1546
+
1547
+ /**
1548
+ * Get all existing lists
1549
+ */
1550
+ protected function get_lists() {
1551
+ $model_list = WYSIJA::get('list', 'model');
1552
+ $query = '
1553
+ SELECT
1554
+ A.`list_id`,
1555
+ A.`name`,
1556
+ A.`is_enabled`,
1557
+ COUNT( B.`campaign_id` ) AS `users`
1558
+ FROM
1559
+ `[wysija]' . $model_list->table_name . '` as A
1560
+ LEFT JOIN
1561
+ `[wysija]campaign_list` AS B on A.`list_id` = B.`list_id`
1562
+ GROUP BY A.`list_id`';
1563
+ $result = $model_list->getResults($query);
1564
+ $lists = array();
1565
+ foreach ($result as $list_obj) {
1566
+ $lists[$list_obj['list_id']] = $list_obj;
1567
+ }
1568
+ return $lists;
1569
+ }
1570
+
1571
+ function defaultDisplay() {
1572
+ $this->data['base_url'] = $this->base_url . '?' . http_build_query($_REQUEST); // saved filter
1573
+ $this->title = __('Newsletters', WYSIJA);
1574
+ $this->viewShow = $this->action = 'main';
1575
+ $this->js[] = 'wysija-admin-list';
1576
+ $this->jsTrans["selecmiss"] = __('Please select a newsletter.', WYSIJA);
1577
+ $this->jsTrans['suredelete'] = __('Delete this newsletter for ever?', WYSIJA);
1578
+ $this->jsTrans['suredelete_bulk'] = __('Delete these newsletters for ever?', WYSIJA);
1579
+ $this->jsTrans['processqueue'] = __('Sending batch of emails...', WYSIJA);
1580
+ $this->jsTrans['viewnews'] = __('View newsletter', WYSIJA);
1581
+ $this->jsTrans['confirmpauseedit'] = __('The newsletter will be deactivated, you will need to reactivate it once you\'re over editing it. Do you want to proceed?', WYSIJA);
1582
+
1583
+
1584
+ //get the filters
1585
+ if (isset($_REQUEST['search']) && $_REQUEST['search']) {
1586
+ $this->filters['like'] = array();
1587
+ foreach ($this->searchable as $field)
1588
+ $this->filters['like'][$field] = $_REQUEST['search'];
1589
+ }
1590
+
1591
+ if (isset($_REQUEST['filter-list']) && $_REQUEST['filter-list']) {
1592
+ $this->filters['equal'] = array('C.list_id' => $_REQUEST['filter-list']);
1593
+ }
1594
+
1595
+ if (isset($_REQUEST['filter-date']) && $_REQUEST['filter-date']) {
1596
+ $this->filters['greater_eq'] = array('created_at' => $_REQUEST['filter-date']);
1597
+ $this->filters['less_eq'] = array('created_at' => strtotime('+1 month', $_REQUEST['filter-date']));
1598
+ }
1599
+
1600
+ $this->filters['is'] = array('type' => 'IS NOT NULL');
1601
+
1602
+
1603
+ if (isset($_REQUEST['link_filter']) && $_REQUEST['link_filter']) {
1604
+ $linkfilters = explode('-', $_REQUEST['link_filter']);
1605
+
1606
+ if (count($linkfilters) > 1) {
1607
+ $this->_addLinkFilter($linkfilters[1], $linkfilters[0]);
1608
+ } else {
1609
+ $this->_addLinkFilter($_REQUEST['link_filter']);
1610
+ }
1611
+ }
1612
+
1613
+ $this->modelObj->noCheck = true;
1614
+ $this->modelObj->reset();
1615
+ if ($this->filters)
1616
+ $this->modelObj->setConditions($this->filters);
1617
+
1618
+
1619
+ // Count emails by status and type
1620
+ $emails_by_status = $this->count_emails_by_status();
1621
+ $emails_by_type = $this->count_emails_by_type();
1622
+ $counts = array_merge($emails_by_status, $emails_by_type);
1623
+ $counts['all'] = array_sum($emails_by_status);
1624
+
1625
+ // collect data
1626
+ $this->data['campaigns'] = $this->get_campaigns();
1627
+ $this->data['datemin'] = $this->get_oldest_compaign();
1628
+ $lists = $this->get_lists(); // $lists is in use later within this scope
1629
+ $this->data['lists'] = $lists;
1630
+
1631
+ // for paging
1632
+ $this->modelObj->countRows = $this->filters ? $this->count_emails () : $counts['all'];
1633
+
1634
+
1635
+ // count queue
1636
+ $email_ids = array();
1637
+ foreach ($this->data['campaigns'] as $emailcamp) {
1638
+ if (in_array($emailcamp['status'], array(1, 3, 99)))
1639
+ $email_ids[] = $emailcamp['email_id'];
1640
+ }
1641
+ $model_queue = WYSIJA::get('queue', 'model');
1642
+ $model_queue->setConditions(array("email_id" => $email_ids));
1643
+ $model_queue->groupBy('email_id');
1644
+ $queue = $model_queue->count();
1645
+ if ($queue) {
1646
+ $this->viewObj->queuedemails = $queue;
1647
+ }
1648
+
1649
+ $this->modelObj->reset();
1650
+
1651
+ //make a loop from the first created to now and increment an array of months
1652
+ $now = time();
1653
+ $this->data['dates'] = array();
1654
+
1655
+ if ((int) $this->data['datemin']['datemin'] > 1) {
1656
+ setlocale(LC_TIME, 'en_US');
1657
+ $formtlettres = "1 " . date('F', $this->data['datemin']['datemin']) . ' ' . date("Y", $this->data['datemin']['datemin']);
1658
+ $month_start = strtotime($formtlettres);
1659
+
1660
+ if ($month_start > 0) {
1661
+ for ($i = $month_start; $i < $now; $i = strtotime('+1 month', $i)) {
1662
+ $this->data['dates'][$i] = date_i18n('F Y', $i); //date('F Y',$i);
1663
+ }
1664
+ }
1665
+ }
1666
+
1667
+
1668
+
1669
+ $campaign_ids_sent = $campaign_ids = array();
1670
+ foreach ($this->data['campaigns'] as &$campaign) {
1671
+ $campaign_ids[] = $campaign['campaign_id'];
1672
+ $modelEmail = WYSIJA::get('email', 'model');
1673
+ $modelEmail->getParams($campaign);
1674
+ if (in_array((int) $campaign['status'], array(-1, 1, 2, 3, 99)))
1675
+ $campaign_ids_sent[] = $campaign['campaign_id'];
1676
+ }
1677
+
1678
+ // 3 - campaign_list request & count request for queue */
1679
+ if ($campaign_ids) {
1680
+ $modeluList = WYSIJA::get('campaign_list', 'model');
1681
+ $userlists = $modeluList->get(array('list_id', 'campaign_id'), array('campaign_id' => $campaign_ids));
1682
+
1683
+ if ($campaign_ids_sent) {
1684
+ $modeluList = WYSIJA::get("email_user_stat", "model");
1685
+ $statstotal = $modeluList->getResults("SELECT COUNT(A.user_id) as count,B.email_id FROM `[wysija]queue` as A
1686
+ JOIN `[wysija]email` as B on A.email_id=B.email_id
1687
+ WHERE B.campaign_id IN (" . implode(",", $campaign_ids_sent) . ") group by B.email_id");
1688
+
1689
+ $senttotalgroupedby = $modeluList->getResults("SELECT COUNT(A.user_id) as count,B.campaign_id,B.email_id,B.type,B.status,A.status as statususer FROM `[wysija]" . $modeluList->table_name . "` as A
1690
+ JOIN `[wysija]email` as B on A.email_id=B.email_id
1691
+ WHERE B.campaign_id IN (" . implode(",", $campaign_ids_sent) . ") group by A.status,B.email_id"); //,A.status
1692
+
1693
+
1694
+ $updateEmail = array();
1695
+ $columnnamestatus = array(0 => "number_sent", 1 => "number_opened", 2 => "number_clicked", 3 => "number_unsub", -1 => "number_bounce");
1696
+ foreach ($senttotalgroupedby as $sentbystatus) {
1697
+ if ($sentbystatus['statususer'] != "-2")
1698
+ $updateEmail[$sentbystatus['email_id']][$columnnamestatus[$sentbystatus['statususer']]] = $sentbystatus['count'];
1699
+ if (isset($senttotal[$sentbystatus['email_id']])) {
1700
+ $senttotal[$sentbystatus['email_id']]['count'] = (int) $senttotal[$sentbystatus['email_id']]['count'] + (int) $sentbystatus['count'];
1701
+ } else {
1702
+ unset($sentbystatus['statususer']);
1703
+ $senttotal[$sentbystatus['email_id']] = $sentbystatus;
1704
+ }
1705
+ }
1706
+
1707
+ $modelEmail = WYSIJA::get('email', 'model');
1708
+
1709
+ foreach ($updateEmail as $emailid => $update) {
1710
+
1711
+ foreach ($columnnamestatus as $v) {
1712
+ if (!isset($update[$v]))
1713
+ $update[$v] = 0;
1714
+ }
1715
+
1716
+ $modelEmail->update($update, array('email_id' => $emailid));
1717
+ $modelEmail->reset();
1718
+ }
1719
+
1720
+
1721
+ /**/
1722
+ $modelC = WYSIJA::get('config', 'model');
1723
+ $running = false;
1724
+
1725
+ $is_multisite = is_multisite();
1726
+
1727
+ //$is_multisite=true;//PROD comment that line
1728
+ if ($is_multisite && $modelC->getValue('sending_method') == 'network') {
1729
+ $sending_emails_each = $modelC->getValue('ms_sending_emails_each');
1730
+ } else {
1731
+ $sending_emails_each = $modelC->getValue('sending_emails_each');
1732
+ }
1733
+
1734
+ if ($modelC->getValue('cron_manual')) {
1735
+ $formsHelp = WYSIJA::get('forms', 'helper');
1736
+ $queue_frequency = $formsHelp->eachValuesSec[$sending_emails_each];
1737
+ $queue_scheduled = WYSIJA::get_cron_schedule('queue');
1738
+
1739
+ $next_scheduled_queue = $queue_scheduled['next_schedule'];
1740
+ $running = $queue_scheduled['running'];
1741
+
1742
+ if ($running) {
1743
+ $helperToolbox = WYSIJA::get('toolbox', 'helper');
1744
+ $running = time() - $running;
1745
+ $running = $helperToolbox->duration_string($running, true, 4);
1746
+ }
1747
+ } else {
1748
+ $schedules = wp_get_schedules();
1749
+ $queue_frequency = $schedules[wp_get_schedule('wysija_cron_queue')]['interval'];
1750
+ $next_scheduled_queue = wp_next_scheduled('wysija_cron_queue');
1751
+ }
1752
+
1753
+
1754
+
1755
+ $status_sent_complete = array();
1756
+ if (isset($senttotal) && $senttotal) {
1757
+ foreach ($senttotal as $sentot) {
1758
+ if ($sentot) {
1759
+ $this->data['sent'][$sentot['email_id']]['total'] = $sentot['count'];
1760
+ $this->data['sent'][$sentot['email_id']]['to'] = $sentot['count'];
1761
+ } else {
1762
+ $this->data['sent'][$sentot['email_id']]['total'] = $this->data['sent'][$sentot['email_id']]['to'] = 0;
1763
+ }
1764
+ $this->data['sent'][$sentot['email_id']]['status'] = $sentot['status'];
1765
+ $this->data['sent'][$sentot['email_id']]['type'] = $sentot['type'];
1766
+ $this->data['sent'][$sentot['email_id']]['left'] = (int) $this->data['sent'][$sentot['email_id']]['total'] - (int) $this->data['sent'][$sentot['email_id']]['to'];
1767
+ }
1768
+ }
1769
+
1770
+ foreach ($statstotal as $sentot) {
1771
+ if (!isset($this->data['sent'][$sentot['email_id']])) {
1772
+ $this->data['sent'][$sentot['email_id']]['total'] = 0;
1773
+ $this->data['sent'][$sentot['email_id']]['to'] = 0;
1774
+ }
1775
+ $this->data['sent'][$sentot['email_id']]['total'] = $this->data['sent'][$sentot['email_id']]['total'] + $sentot['count'];
1776
+ $this->data['sent'][$sentot['email_id']]['left'] = (int) $this->data['sent'][$sentot['email_id']]['total'] - (int) $this->data['sent'][$sentot['email_id']]['to'];
1777
+ }
1778
+
1779
+ //$is_multisite=true;//PROD comment that line
1780
+ if ($is_multisite && $modelC->getValue('sending_method') == 'network') {
1781
+ $sending_emails_number = $modelC->getValue('ms_sending_emails_number');
1782
+ } else {
1783
+ $sending_emails_number = $modelC->getValue('sending_emails_number');
1784
+ }
1785
+
1786
+ if (isset($this->data['sent'])) {
1787
+ foreach ($this->data['sent'] as $key => &$camp) {
1788
+ if ($this->data['sent'][$key]['left'] > 0) {
1789
+ $cronsneeded = ceil($this->data['sent'][$key]['left'] / $sending_emails_number);
1790
+ $this->data['sent'][$key]['remaining_time'] = $cronsneeded * $queue_frequency;
1791
+ $this->data['sent'][$key]['running_for'] = $running;
1792
+ $this->data['sent'][$key]['next_batch'] = $next_scheduled_queue - time();
1793
+ $this->data['sent'][$key]['remaining_time'] = $this->data['sent'][$key]['remaining_time'] - ($queue_frequency) + $this->data['sent'][$key]['next_batch'];
1794
+ } else {
1795
+ if ((in_array($this->data['sent'][$key]['status'], array(1, 3, 99))) && $this->data['sent'][$key]['type'] == 1)
1796
+ $status_sent_complete[] = $key;
1797
+ }
1798
+ }
1799
+ }
1800
+
1801
+
1802
+ // status update to sent for the one that are sent
1803
+ if (count($status_sent_complete) > 0) {
1804
+ $modelEmail = WYSIJA::get('email', 'model');
1805
+ $modelEmail->noCheck = true;
1806
+ $modelEmail->reset();
1807
+ $modelEmail->update(array('status' => 2), array('equal' => array('email_id' => $status_sent_complete)));
1808
+ }
1809
+ }
1810
+ }
1811
+
1812
+ $this->data['counts'] = array_reverse($counts);
1813
+
1814
+ /* regrouping all the data in the same array */
1815
+ foreach ($this->data['campaigns'] as &$campaign) {
1816
+ /* default key while we don't have the data */
1817
+ //TODO add data for stats about emails opened clicked etc
1818
+ $campaign["emails"] = 0;
1819
+ $campaign["opened"] = 0;
1820
+ $campaign["clicked"] = 0;
1821
+
1822
+ if ($userlists) {
1823
+ foreach ($userlists as $key => $userlist) {
1824
+ if ($campaign["campaign_id"] == $userlist["campaign_id"] && isset($lists[$userlist["list_id"]])) {
1825
+ if (!isset($campaign["lists"]))
1826
+ $campaign["lists"] = $lists[$userlist["list_id"]]["name"];
1827
+ else
1828
+ $campaign["lists"].=", " . $lists[$userlist["list_id"]]["name"];
1829
+ }
1830
+ }
1831
+ }
1832
+ if (isset($campaign["lists"]) && !$campaign["lists"])
1833
+ unset($campaign["lists"]);
1834
+
1835
+ if (((isset($campaign['params']['schedule']['isscheduled']) ||
1836
+ ($campaign['type'] == 2 && isset($campaign['params']['autonl']['event']) && in_array($campaign['params']['autonl']['event'], array('new-articles'/* ,'subs-2-nl' */)))
1837
+ ) && $campaign['status'] != 2 && !isset($campaign["lists"])) || ($campaign['type'] == 2 && isset($campaign['params']['autonl']['event']) && in_array($campaign['params']['autonl']['event'], array('subs-2-nl')) && $campaign['status'] != 2 && (!isset($campaign['params']['autonl']['subscribetolist']) || !isset($lists[$campaign['params']['autonl']['subscribetolist']]) ))
1838
+ ) {
1839
+ $campaign['classRow'] = " listmissing ";
1840
+ $campaign['msgListEdit'] = '<strong>' . __('The list has been deleted.', WYSIJA) . '</strong>';
1841
+ $campaign['msgSendSuspended'] = '<strong>' . __('Sending suspended.', WYSIJA) . '</strong>';
1842
+ }
1843
+ }
1844
+
1845
+ $this->dataAutoNl();
1846
+ if (!$this->data['campaigns']) {
1847
+ $this->notice(__('There are no newsletters.', WYSIJA));
1848
+ }
1849
+ }
1850
+
1851
+ function setviewStatsfilter() {
1852
+ /* get the filters */
1853
+ $this->searchable = array("email", "firstname", "lastname");
1854
+ $this->filters = array();
1855
+ if (isset($_REQUEST['search']) && $_REQUEST['search']) {
1856
+ $this->filters["like"] = array();
1857
+ foreach ($this->searchable as $field)
1858
+ $this->filters["like"][$field] = $_REQUEST['search'];
1859
+ }
1860
+ $this->tableQuery = 'email_user_stat';
1861
+ $this->statusemail = 'B.status as umstatus';
1862
+ if (isset($_REQUEST['link_filter']) && $_REQUEST['link_filter']) {
1863
+ switch ($_REQUEST['link_filter']) {
1864
+ case 'inqueue':
1865
+ $this->tableQuery = 'queue';
1866
+ $this->statusemail = '-2 as umstatus';
1867
+ break;
1868
+ case 'sent':
1869
+ $this->filters['equal'] = array('B.status' => 0);
1870
+ break;
1871
+ case 'bounced':
1872
+ $this->filters['equal'] = array('B.status' => -1);
1873
+ break;
1874
+ case 'opened':
1875
+ $this->filters['equal'] = array('B.status' => 1);
1876
+ break;
1877
+ case 'clicked':
1878
+ $this->filters['equal'] = array('B.status' => 2);
1879
+ break;
1880
+ case 'unsubscribe':
1881
+ $this->filters['equal'] = array('B.status' => 3);
1882
+ break;
1883
+ case 'notsent':
1884
+ $this->filters['equal'] = array('B.status' => -2);
1885
+ break;
1886
+ }
1887
+ }
1888
+ // filter by url id
1889
+ if (isset($_REQUEST['url_id']) && (int) $_REQUEST['url_id'] > 0) {
1890
+ $this->tableQuery = 'email_user_url';
1891
+ $this->filters['equal'] = array('B.url_id' => (int) $_REQUEST['url_id']);
1892
+ $this->statusemail = '2 as umstatus'; //by default, when filter by url_id, all subscribers had clicked
1893
+ }
1894
+ }
1895
+
1896
+ function viewstats() {
1897
+ $this->js[] = 'wysija-admin-list';
1898
+ $this->js[] = 'wysija-charts';
1899
+ $this->viewShow = 'viewstats';
1900
+
1901
+ $this->modelObj = WYSIJA::get("email", "model");
1902
+ $this->modelObj->limitON = false;
1903
+
1904
+ $email_object = $this->modelObj->getOne(false, array("email_id" => $_REQUEST['id']));
1905
+ $this->viewObj->model = $this->modelObj;
1906
+ $this->viewObj->namecampaign = $email_object['subject'];
1907
+ $this->viewObj->title = sprintf(__('Stats : %1$s', WYSIJA), $email_object['subject']);
1908
+
1909
+ $modelObjCamp = WYSIJA::get("campaign", "model");
1910
+ $limit_pp = false;
1911
+ if (isset($modelObjCamp->limit_pp))
1912
+ $limit_pp = $modelObjCamp->limit_pp;
1913
+ $modelObjCamp->limitON = false;
1914
+ $campaign = $modelObjCamp->getOne(false, array("campaign_id" => $email_object['campaign_id']));
1915
+
1916
+
1917
+ $this->setviewStatsfilter();
1918
+
1919
+ $this->modelObj->reset();
1920
+ $this->modelObj->noCheck = true;
1921
+
1922
+ /* 0 - counting request */
1923
+ $queryCmmonStart = 'SELECT count(distinct B.user_id) as users FROM `[wysija]user` as A';
1924
+ $queryCmmonStart.=' LEFT JOIN `[wysija]' . $this->tableQuery . '` as B on A.user_id=B.user_id';
1925
+
1926
+ /* all the counts query */
1927
+ $query = "SELECT count(user_id) as users, status FROM `[wysija]email_user_stat` as A
1928
+ WHERE A.email_id=" . $email_object['email_id'] . " GROUP BY status";
1929
+ $countss = $this->modelObj->query("get_res", $query, ARRAY_A);
1930
+
1931
+ /* we also count what is in the queue */
1932
+ $query = "SELECT count(user_id) as users FROM `[wysija]queue` as A
1933
+ WHERE A.email_id=" . $email_object['email_id'];
1934
+ $countss[-2]['status'] = -3;
1935
+ $countss[-2]['users'] = $this->modelObj->count($query, 'users');
1936
+
1937
+ $counts = array();
1938
+ $truetotal = $total = 0;
1939
+
1940
+ foreach ($countss as $count) {
1941
+ switch ($count['status']) {
1942
+ case "-3":
1943
+ $type = 'inqueue';
1944
+ break;
1945
+ case "-2":
1946
+ $type = 'notsent';
1947
+ break;
1948
+ case "-1":
1949
+ $type = 'bounced';
1950
+ break;
1951
+ case "0":
1952
+ $type = 'sent';
1953
+ break;
1954
+ case "1":
1955
+ $type = 'opened';
1956
+ break;
1957
+ case "2":
1958
+ $type = 'clicked';
1959
+ break;
1960
+ case "3":
1961
+ $type = 'unsubscribe';
1962
+ break;
1963
+ }
1964
+ if ($count['status'] != "-2")
1965
+ $total = $total + $count['users'];
1966
+ $truetotal = $truetotal + $count['users'];
1967
+ $counts[$type] = $count['users'];
1968
+ }
1969
+
1970
+ $counts['allsent'] = $total;
1971
+ $counts['all'] = $truetotal;
1972
+
1973
+ $this->modelObj->reset();
1974
+ $this->filters['equal']["B.email_id"] = $email_object['email_id'];
1975
+
1976
+ $this->modelObj->noCheck = true;
1977
+ if ($this->filters)
1978
+ $this->modelObj->setConditions($this->filters);
1979
+
1980
+ //$this->modelObj->setConditions(array("equal"=>array("B.email_id"=>$emailObj['email_id'])));
1981
+
1982
+ /* 1 - subscriber request */
1983
+ $query = 'SELECT A.user_id, A.firstname, A.lastname,A.status as ustatus,' . $this->statusemail . ' , A.email, B.* FROM `[wysija]user` as A';
1984
+ $query.=' LEFT JOIN `[wysija]' . $this->tableQuery . '` as B on A.user_id=B.user_id';
1985
+ $queryFinal = $this->modelObj->makeWhere();
1986
+
1987
+ /* without filter we already have the total number of subscribers */
1988
+ if ($this->filters)
1989
+ $this->modelObj->countRows = $this->modelObj->count($queryCmmonStart . $queryFinal, 'users');
1990
+ else
1991
+ $this->modelObj->countRows = $counts['all'];
1992
+
1993
+ $orderby = " ORDER BY ";
1994
+ /**
1995
+ * Until now, we have
1996
+ * - 3 possible values of $this->tableQuery (queue, email_user_url, email_user_stat), set by $this->setviewStatsfilter()
1997
+ * - 2 possible values of $_REQUEST['orderby']
1998
+ * => 3x2 = 6 cases
1999
+ */
2000
+ if (isset($_REQUEST['orderby'])) {
2001
+ switch ($this->tableQuery) {
2002
+ case 'email_user_url':
2003
+ case 'email_user_stat':
2004
+ $orderby .= 'B.' . $_REQUEST['orderby'] . " " . $_REQUEST['ordert'];
2005
+ break;
2006
+
2007
+ case 'queue':
2008
+ default:
2009
+ $orderby .= 'A.user_id DESC';
2010
+ break;
2011
+ }
2012
+ } else {
2013
+ switch ($this->tableQuery) {
2014
+ case 'email_user_url':
2015
+ $orderby .= 'B.clicked_at DESC, B.number_clicked DESC'; // by default, sort by last clicked and biggest hit
2016
+ break;
2017
+
2018
+ case 'email_user_stat':
2019
+ $orderby .= 'B.opened_at DESC, B.status DESC'; // by default, sort by last open and its staus value
2020
+ break;
2021
+
2022
+ case 'queue':
2023
+ default:
2024
+ $orderby .= 'A.user_id DESC';
2025
+ break;
2026
+ }
2027
+ }
2028
+ $this->data['tableQuery'] = $this->tableQuery;
2029
+ $this->modelObj->limitON = true;
2030
+
2031
+ $subscribers = array();
2032
+ $hook_params = array(
2033
+ 'email_id' => $email_object['email_id'],
2034
+ 'url_id' => isset($_REQUEST['url_id']) && $_REQUEST['url_id'] ? $_REQUEST['url_id'] : false,
2035
+ 'subscribers' => &$subscribers
2036
+ );
2037
+ $this->data['subscribers'] = $this->modelObj->getResults($query . $queryFinal . " GROUP BY A.user_id" . $orderby . $this->modelObj->setLimit(0, (int)$limit_pp));
2038
+ $this->modelObj->reset();
2039
+
2040
+ /* make the data object for the listing view */
2041
+ $modelList = WYSIJA::get("list", "model");
2042
+
2043
+ /* 2 - list request */
2044
+ $query = "SELECT A.list_id, A.name,A.is_enabled, count( B.user_id ) AS users FROM `[wysija]" . $modelList->table_name . "` as A";
2045
+ $query.=" LEFT JOIN `[wysija]user_list` as B on A.list_id = B.list_id";
2046
+ $query.=" GROUP BY A.list_id";
2047
+ $listsDB = $modelList->getResults($query);
2048
+
2049
+ $lists = array();
2050
+ foreach ($listsDB as $listobj) {
2051
+ $lists[$listobj["list_id"]] = $listobj;
2052
+ }
2053
+
2054
+ $listsDB = null;
2055
+
2056
+ $user_ids = array();
2057
+ foreach ($this->data['subscribers'] as $subscriber) {
2058
+ $user_ids[] = $subscriber['user_id'];
2059
+ }
2060
+
2061
+ /* 3 - user_list request */
2062
+ if ($user_ids) {
2063
+ $modeluList = WYSIJA::get("user_list", "model");
2064
+ $userlists = $modeluList->get(array("list_id", "user_id"), array("user_id" => $user_ids));
2065
+ }
2066
+
2067
+
2068
+ $this->data['lists'] = $lists;
2069
+ $this->data['counts'] = array_reverse($counts);
2070
+
2071
+ /* regrouping all the data in the same array */
2072
+ foreach ($this->data['subscribers'] as $keysus => $subscriber) {
2073
+ /* default key while we don't have the data */
2074
+ //TODO add data for stats about emails opened clicked etc
2075
+ $this->data['subscribers'][$keysus]["emails"] = 0;
2076
+ $this->data['subscribers'][$keysus]["opened"] = 0;
2077
+ $this->data['subscribers'][$keysus]["clicked"] = 0;
2078
+
2079
+ if ($userlists) {
2080
+ foreach ($userlists as $key => $userlist) {
2081
+ if ($subscriber["user_id"] == $userlist["user_id"] && isset($lists[$userlist["list_id"]])) {
2082
+ if (!isset($this->data['subscribers'][$keysus]["lists"]))
2083
+ $this->data['subscribers'][$keysus]["lists"] = $lists[$userlist["list_id"]]["name"];
2084
+ else
2085
+ $this->data['subscribers'][$keysus]["lists"].=", " . $lists[$userlist["list_id"]]["name"];
2086
+ }
2087
+ }
2088
+ }
2089
+ }
2090
+
2091
+ $this->data['email'] = $email_object;
2092
+
2093
+ if (!$this->data['subscribers']) {
2094
+ $this->notice(__("Your request can't retrieve any subscribers. Change your filters!", WYSIJA));
2095
+ }
2096
+
2097
+ // execute hooks
2098
+ $hook_params = array(
2099
+ 'email_id' => $_REQUEST['id'],
2100
+ 'email_object' => $email_object,
2101
+ 'url_id' => !empty($_REQUEST['url_id']) ? (int)$_REQUEST['url_id'] : null
2102
+ );
2103
+
2104
+ $this->data['hooks']['hook_newsletter_top'] = apply_filters('hook_newsletter_top', WYSIJA_module::execute_hook('hook_newsletter_top', $hook_params), $hook_params);
2105
+ $this->data['hooks']['hook_newsletter_bottom'] = apply_filters('hook_newsletter_bottom', WYSIJA_module::execute_hook('hook_newsletter_bottom', $hook_params), $hook_params);
2106
+ }
2107
+
2108
+ function getListSubscriberQry($selectcolumns) {
2109
+ $this->modelObj = WYSIJA::get("email", "model");
2110
+ $this->emailObj = $this->modelObj->getOne(false, array('email_id' => $_REQUEST['id']));
2111
+
2112
+ /* use the filter if there is */
2113
+ $this->setviewStatsfilter();
2114
+
2115
+ if ($selectcolumns == "B.user_id") {
2116
+ //unset($this->filters["like"]);
2117
+ }
2118
+
2119
+ $this->filters['equal']["B.email_id"] = $this->emailObj['email_id'];
2120
+ $this->modelObj->noCheck = true;
2121
+ if ($this->filters)
2122
+ $this->modelObj->setConditions($this->filters);
2123
+
2124
+ /* select insert all the subscribers from that campaign into user_list */
2125
+ if ($selectcolumns == "B.user_id") {
2126
+ $query = "SELECT $selectcolumns FROM `[wysija]" . $this->tableQuery . "` as B";
2127
+ $query.=$this->modelObj->makeWhere();
2128
+ } else {
2129
+ $query = "SELECT $selectcolumns FROM `[wysija]user` as A";
2130
+ $query.=" LEFT JOIN `[wysija]" . $this->tableQuery . "` as B on A.user_id=B.user_id";
2131
+ $query.=$this->modelObj->makeWhere();
2132
+ }
2133
+
2134
+ return $query;
2135
+ }
2136
+
2137
+ function createnewlist() {
2138
+ /* get the email subject */
2139
+ $emailModel = WYSIJA::get('email', 'model');
2140
+ $email = $emailModel->getOne(array('subject'), array('email_id' => $_REQUEST['id']));
2141
+
2142
+ $this->modelObj->reset();
2143
+
2144
+ /* set the name of the new list */
2145
+ $prefix = "";
2146
+ if (isset($_REQUEST['link_filter']))
2147
+ $prefix = ' (' . $this->viewObj->getTransStatusEmail($_REQUEST['link_filter']) . ')';
2148
+ $listname = sprintf(__('Segment of %1$s', WYSIJA), $email['subject'] . $prefix);
2149
+
2150
+ /* insert new list */
2151
+ $modelL = WYSIJA::get('list', 'model');
2152
+ $listid = $modelL->insert(array('is_enabled' => 1, 'name' => $listname, 'description' => __('List created based on a newsletter segment.', WYSIJA)));
2153
+
2154
+ /* get list of subscribers filtered or not */
2155
+ $query = $this->getListSubscriberQry($listid . ', A.user_id, ' . time() . ', 0');
2156
+
2157
+ $query2 = 'INSERT INTO `[wysija]user_list` (`list_id`,`user_id`,`sub_date`,`unsub_date`) ' . $query;
2158
+
2159
+ $this->modelObj->query($query2);
2160
+
2161
+ $this->notice(sprintf(__('A new list "%1$s" has been created out of this segment.', WYSIJA), $listname));
2162
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id=' . $_REQUEST['id']);
2163
+ }
2164
+
2165
+ function unsubscribeall() {
2166
+ // Update user_list, set unsubdate and sub_date
2167
+ $query = $this->getListSubscriberQry('B.user_id');
2168
+ $query1 = "
2169
+ UPDATE `[wysija]user_list`
2170
+ SET
2171
+ `unsub_date` = ".time().",
2172
+ `sub_date` = 0
2173
+ WHERE
2174
+ `user_id` IN ($query)
2175
+ AND `list_id` NOT IN (SELECT `list_id` FROM `[wysija]list` WHERE `is_enabled` < 1)
2176
+ ";
2177
+ $this->modelObj->query($query1);
2178
+
2179
+ // unsubscribe from user where select from email_user_stat
2180
+ $query2 = "UPDATE `[wysija]user` SET `status`=-1 WHERE `user_id` IN ($query)";
2181
+ $this->modelObj->query($query2);
2182
+
2183
+ $this->notice(__('The segment has been unsubscribed from all the lists.', WYSIJA));
2184
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id=' . $_REQUEST['id']);
2185
+ }
2186
+
2187
+ function sendconfirmation() {
2188
+ //delete from user_lists where select from email_user_stat
2189
+ $query = $this->getListSubscriberQry('B.user_id ');
2190
+
2191
+ $user_ids = $this->modelObj->query('get_res', $query);
2192
+
2193
+ $uids = array();
2194
+ foreach ($user_ids as $data) {
2195
+ $uids[] = $data['user_id'];
2196
+ }
2197
+
2198
+ $helperUser = WYSIJA::get('user', 'helper');
2199
+ $helperUser->sendConfirmationEmail($uids);
2200
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id=' . $_REQUEST['id']);
2201
+ }
2202
+
2203
+ function removequeue() {
2204
+ /* delete from queue where select from email_user_stat */
2205
+ $query = $this->getListSubscriberQry('B.user_id');
2206
+ $query2 = "DELETE FROM `[wysija]queue` where user_id IN ($query) AND email_id=" . $this->emailObj['email_id'];
2207
+ $this->modelObj->query($query2);
2208
+
2209
+ $this->notice(__('The segment has been removed from the queue of this newsletter.', WYSIJA));
2210
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id=' . $_REQUEST['id']);
2211
+ }
2212
+
2213
+ function export() {
2214
+ /* select from email_user_stat left join user */
2215
+ $query = $this->getListSubscriberQry('B.user_id');
2216
+ $result = $this->modelObj->query('get_res', $query);
2217
+ $user_ids = array();
2218
+ foreach ($result as $user)
2219
+ $user_ids[] = $user['user_id'];
2220
+
2221
+ $fileHelp = WYSIJA::get('file', 'helper');
2222
+ $tempfilename = $fileHelp->temp(implode(',', $user_ids), 'export_userids', '.txt');
2223
+
2224
+ //$this->redirect("admin.php?page=wysija_campaigns&action=viewstats&id=".$_REQUEST['id']."&user_ids=".serialize($result));
2225
+ $this->redirect('admin.php?page=wysija_subscribers&action=exportcampaign&camp_id=' . $_REQUEST['id'] . '&file_name=' . base64_encode($tempfilename['path']));
2226
+ }
2227
+
2228
+ function unsubscribelist($data) {
2229
+
2230
+ $modelL = WYSIJA::get('list', 'model');
2231
+ $list = $modelL->getOne(false, array('list_id' => $data['listid']));
2232
+ if ($list['is_enabled']) {
2233
+ /* delete from user_lists where select from email_user_stat */
2234
+ $query = $this->getListSubscriberQry("B.user_id");
2235
+ $query2 = "DELETE FROM `[wysija]user_list` where user_id IN ($query) and list_id=" . $data['listid'];
2236
+ $this->modelObj->query($query2);
2237
+
2238
+ $this->notice(sprintf(__('The segment has been unsubscribed from the list "%1$s".', WYSIJA), $list['name']));
2239
+ } else {
2240
+ $this->notice(sprintf(__('The segment cannot be unsubscribed from an [IMPORT] list.', WYSIJA), $list['name']));
2241
+ }
2242
+
2243
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id=' . $_REQUEST['id']);
2244
+ }
2245
+
2246
+ function articles() {
2247
+ $this->iframeTabs = array('articles' => __("Post Selection", WYSIJA));
2248
+
2249
+ // required js files
2250
+ $this->js[] = 'wysija-admin-ajax';
2251
+ $this->js[] = 'wysija-base-script-64';
2252
+ $this->js[] = 'wysija-scriptaculous';
2253
+ $this->js[] = 'wysija-colorpicker';
2254
+
2255
+ // translations
2256
+ $this->jsTrans['show_advanced'] = __('Display and insert options', WYSIJA);
2257
+ $this->jsTrans['hide_advanced'] = __('Back to selection', WYSIJA);
2258
+ $this->jsTrans['loading_results'] = __('Loading results...', WYSIJA);
2259
+ $this->jsTrans['inserting_selection'] = __('Inserting selected articles...', WYSIJA);
2260
+ $this->jsTrans['post_selected'] = __('selected', WYSIJA);
2261
+
2262
+ // default tab in popup (this needs to be removed at some point)
2263
+ $_GET['tab'] = 'articles';
2264
+
2265
+ // get parameters
2266
+ $params = array(
2267
+ 'category_ids' => null,
2268
+ 'category' => null,
2269
+ 'title_tag' => 'h2',
2270
+ 'title_alignment' => 'left',
2271
+ 'title_position' => 'inside',
2272
+ 'image_alignment' => 'alternate',
2273
+ 'image_width' => 325,
2274
+ 'post_content' => 'excerpt',
2275
+ 'readmore' => __('Read more.', WYSIJA),
2276
+ 'show_divider' => 'yes',
2277
+ 'post_limit' => 5,
2278
+ 'post_type' => 'post',
2279
+ 'author_show' => 'no',
2280
+ 'author_label' => __('Author:', WYSIJA),
2281
+ 'category_show' => 'no',
2282
+ 'category_label' => __('Categories:', WYSIJA),
2283
+ 'nopost_message' => __('Latest content already sent.', WYSIJA),
2284
+ 'bgcolor1' => null,
2285
+ 'bgcolor2' => null,
2286
+ 'sort_by' => 'newest'
2287
+ );
2288
+
2289
+ // check if GET parameters are specified
2290
+ foreach($params as $key => $value) {
2291
+ if(array_key_exists($key, $_GET)) {
2292
+ switch($key) {
2293
+ case 'autopost_count':
2294
+ $params[$key] = (int)$_GET[$key];
2295
+ break;
2296
+ case 'readmore':
2297
+ case 'nopost_message':
2298
+ $params[$key] = base64_decode($_GET[$key]);
2299
+ break;
2300
+ default:
2301
+ $params[$key] = $_GET[$key];
2302
+ }
2303
+ }
2304
+ }
2305
+
2306
+ $model_config = WYSIJA::get('config', 'model');
2307
+ $insert_post_parameters = $model_config->getValue('insert_post_parameters');
2308
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
2309
+
2310
+ if($insert_post_parameters !== false) {
2311
+ // there are user params
2312
+ $params = $helper_wj_engine->decodeParameters(array_merge($params, $insert_post_parameters));
2313
+ }
2314
+
2315
+ // get autopost count
2316
+ $this->data['autopost_count'] = (array_key_exists('autopost_count', $_GET)) ? (int) $_GET['autopost_count'] : 0;
2317
+
2318
+ // get autopost type (single or multiple)
2319
+ $this->data['autopost_type'] = (array_key_exists('autopost_type', $_GET)) ? $_GET['autopost_type'] : 'multiple';
2320
+
2321
+ // if only one group of post can be added, change default alignment to left
2322
+ if($this->data['autopost_type'] === 'single' && $params['image_alignment'] === 'alternate') {
2323
+ $params['image_alignment'] = 'left';
2324
+ }
2325
+
2326
+ // get post categories (even when there's no post)
2327
+ $post_categories = get_categories(array('hide_empty' => 0));
2328
+ $categories = array();
2329
+ foreach ($post_categories as $category) {
2330
+ $categories[] = array('id' => $category->cat_ID, 'name' => $category->name);
2331
+ }
2332
+ $this->data['categories'] = $categories;
2333
+
2334
+ // max number of posts
2335
+ $this->data['post_limits'] = array(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 20, 30, 50);
2336
+
2337
+ $this->data['params'] = $params;
2338
+
2339
+ return $this->popupContent();
2340
+ }
2341
+
2342
+ function themeupload() {
2343
+ $helperNumbers = WYSIJA::get('numbers', 'helper');
2344
+ $bytes = $helperNumbers->get_max_file_upload();
2345
+
2346
+ if (isset($_SERVER['CONTENT_LENGTH']) && $_SERVER['CONTENT_LENGTH'] > $bytes['maxbytes']) {
2347
+ if (isset($_FILES['my-theme']['name']) && $_FILES['my-theme']['name']) {
2348
+ $filename = $_FILES['my-theme']['name'];
2349
+ } else {
2350
+ $filename = "";
2351
+ }
2352
+
2353
+ $this->error(sprintf(__('Upload error, file %1$s is too large! (MAX:%2$s)', WYSIJA), $filename, $bytes['maxmegas']), true);
2354
+ $this->redirect('admin.php?page=wysija_campaigns&action=themes');
2355
+
2356
+ return false;
2357
+ }
2358
+
2359
+
2360
+ $ZipfileResult = trim(file_get_contents($_FILES['my-theme']['tmp_name']));
2361
+
2362
+ $themesHelp = WYSIJA::get('themes', 'helper');
2363
+ $result = $themesHelp->installTheme($_FILES['my-theme']['tmp_name'], true);
2364
+ $this->redirect('admin.php?page=wysija_campaigns&action=themes&reload=1');
2365
+
2366
+ return true;
2367
+ }
2368
+
2369
+ function themes() {
2370
+ $this->iframeTabs = array('themes' => __('Install Themes', WYSIJA));
2371
+ $this->js[] = 'wysija-admin-ajax';
2372
+ $this->js[] = 'wysija-base-script-64';
2373
+ $this->jsTrans['viewinfos'] = __('Details & PSD', WYSIJA);
2374
+ $this->jsTrans['viewback'] = __('<< Back', WYSIJA);
2375
+ $this->jsTrans['install'] = __('Download', WYSIJA);
2376
+ $this->jsTrans['reinstall'] = __('Reinstall', WYSIJA);
2377
+ $this->jsTrans['premiumonly'] = __('Premium', WYSIJA);
2378
+
2379
+ $model_config = WYSIJA::get('config', 'model');
2380
+ //change the translation of the button when it's premium
2381
+ if ($model_config->getValue('premium_key'))
2382
+ $this->jsTrans['ispremium'] = 1;
2383
+ else
2384
+ $this->jsTrans['ispremium'] = 0;
2385
+
2386
+ $this->jsTrans['premiumfiles'] = __('Photoshop file available as part of [link]Premium features[/link].', WYSIJA);
2387
+
2388
+ $helper_licence = WYSIJA::get('licence', 'helper');
2389
+ $url_checkout = $helper_licence->get_url_checkout('themes');
2390
+ $this->jsTrans['premiumfiles'] = str_replace(array('[link]', '[/link]'), array('<a href="' . $url_checkout . '" target="_blank" >', '</a>'), $this->jsTrans['premiumfiles']);
2391
+
2392
+ $this->jsTrans['showallthemes'] = __('Show all themes', WYSIJA);
2393
+ $this->jsTrans['totalvotes'] = __('(%1$s votes)', WYSIJA);
2394
+ $this->jsTrans['voterecorded'] = __("Your vote has been recorded.", WYSIJA);
2395
+ $this->jsTrans['votenotrecorded'] = __("Your vote could not be recorded.", WYSIJA);
2396
+ $this->jsTrans['reinstallwarning'] = __('Watch out! If you reinstall this theme all the files which are in the folder:/wp-content/uploads/wysija/themes/%1$s will be overwritten. Are you sure you want to reinstall?', WYSIJA);
2397
+ $this->jsTrans['errorconnecting'] = __("We were unable to contact the API, the site may be down. Please try again later.", WYSIJA);
2398
+
2399
+ $this->jsTrans['viewallthemes'] = __('View all themes by %1$s', WYSIJA);
2400
+ $this->jsTrans['downloadpsd'] = __("Download original Photoshop file", WYSIJA);
2401
+ $this->jsTrans['downloadzip'] = __("Download as .zip", WYSIJA);
2402
+ $this->jsTrans['viewauthorsite'] = __("View author's website", WYSIJA);
2403
+ $this->jsTrans['stars'] = __('Average rating: %1$s', WYSIJA);
2404
+ $this->jsTrans['starsyr'] = __('My rating: %1$s', WYSIJA);
2405
+ $this->jsTrans['downloads'] = __('Downloads: %1$s', WYSIJA);
2406
+ $this->jsTrans['tags'] = __('Tags: %1$s', WYSIJA);
2407
+ $this->jsTrans['lastupdated'] = __('Last updated: %1$s', WYSIJA);
2408
+ $this->jsTrans['includes'] = __('Includes: %1$s', WYSIJA);
2409
+
2410
+ $helper_themes = WYSIJA::get('themes', 'helper');
2411
+
2412
+ $this->jsTrans['installedthemes'] = $helper_themes->getInstalled();
2413
+
2414
+ $url = admin_url('admin.php');
2415
+ $helper_toolbox = WYSIJA::get("toolbox", "helper");
2416
+ $domain_name = $helper_toolbox->_make_domain_name($url);
2417
+ $this->jsTrans['domainname'] = $domain_name;
2418
+
2419
+ $_GET['tab'] = 'themes';
2420
+
2421
+ return $this->popupContent();
2422
+ }
2423
+
2424
+ function bookmarks() {
2425
+ $this->iframeTabs = array('bookmarks' => __('Bookmarks Selection', WYSIJA));
2426
+ $this->js[] = 'wysija-admin-ajax';
2427
+
2428
+ $_GET['tab'] = 'bookmarks';
2429
+
2430
+ $networks = array(
2431
+ 'facebook' => array(
2432
+ 'label' => 'Facebook',
2433
+ 'url' => null,
2434
+ 'placeholder' => 'https://www.facebook.com/mailpoetplugin'
2435
+ ),
2436
+ 'twitter' => array(
2437
+ 'label' => 'Twitter',
2438
+ 'url' => null,
2439
+ 'placeholder' => 'http://www.twitter.com/mail_poet'
2440
+ ),
2441
+ 'google' => array(
2442
+ 'label' => 'Google+',
2443
+ 'url' => null,
2444
+ 'placeholder' => null
2445
+ ),
2446
+ 'linkedin' => array(
2447
+ 'label' => 'LinkedIn',
2448
+ 'url' => null,
2449
+ 'placeholder' => null
2450
+ )
2451
+ );
2452
+
2453
+ // get networks' url from config
2454
+ $model_config = WYSIJA::get('config', 'model');
2455
+ $urls = $model_config->getValue('social_bookmarks');
2456
+
2457
+ // set url from config for each network if specified
2458
+ foreach ($networks as $network => $values) {
2459
+ if (isset($urls[$network]) and strlen(trim($urls[$network])) > 0) {
2460
+ $networks[$network]['url'] = $urls[$network];
2461
+ }
2462
+ }
2463
+
2464
+ $this->data['networks'] = $networks;
2465
+ $this->data['size'] = 'medium';
2466
+ $this->data['theme'] = isset($_REQUEST['theme']) ? $_REQUEST['theme'] : 'default';
2467
+
2468
+ return $this->popupContent();
2469
+ }
2470
+
2471
+ function dividers() {
2472
+ $this->iframeTabs = array('dividers' => __("Dividers Selection", WYSIJA));
2473
+ $this->js[] = 'wysija-admin-ajax';
2474
+ $this->js[] = 'wysija-base-script-64';
2475
+
2476
+ $_GET['tab'] = 'dividers';
2477
+
2478
+ $model_email = WYSIJA::get('email', 'model');
2479
+ $this->data['email'] = $email = $model_email->getOne(false, array('email_id' => $_REQUEST['emailId']));
2480
+
2481
+ // get dividers
2482
+ $helper_dividers = WYSIJA::get('dividers', 'helper');
2483
+ $dividers = $helper_dividers->getAll();
2484
+
2485
+ // get theme divider if it's not the default theme
2486
+ if (isset($email['params']['theme'])) {
2487
+ $helper_themes = WYSIJA::get('themes', 'helper');
2488
+ $themeDivider = $helper_themes->getDivider($email['params']['theme']);
2489
+ if ($themeDivider !== NULL) {
2490
+ array_unshift($dividers, $themeDivider);
2491
+ }
2492
+ }
2493
+
2494
+ // get selected divider
2495
+ if (isset($email['params']['divider'])) {
2496
+ $selected_divider = $email['params']['divider'];
2497
+ } else {
2498
+ $helper_dividers = WYSIJA::get('dividers', 'helper');
2499
+ $selected_divider = $helper_dividers->getDefault();
2500
+ }
2501
+
2502
+ // set selected divider in first position
2503
+ array_unshift($dividers, $selected_divider);
2504
+
2505
+ // remove selected divider if present in the list
2506
+ for ($i = 1; $i < count($dividers); $i++) {
2507
+ if ($dividers[$i]['src'] === $selected_divider['src']) {
2508
+ unset($dividers[$i]);
2509
+ break;
2510
+ }
2511
+ }
2512
+
2513
+ $this->data['selected'] = $selected_divider;
2514
+ $this->data['dividers'] = $dividers;
2515
+ return $this->popupContent();
2516
+ }
2517
+
2518
+ function autopost() {
2519
+ $this->iframeTabs = array('autopost' => __("Add / Edit group of posts", WYSIJA));
2520
+ $this->js[] = 'wysija-admin-ajax';
2521
+ $this->js[] = 'wysija-base64';
2522
+ $this->js[] = 'wysija-scriptaculous';
2523
+ $this->js[] = 'wysija-colorpicker';
2524
+ $this->js[] = 'mailpoet-select2';
2525
+
2526
+ // translations
2527
+ $this->jsTrans['show_advanced'] = __('Show display options', WYSIJA);
2528
+ $this->jsTrans['hide_advanced'] = __('Hide display options', WYSIJA);
2529
+
2530
+ $_GET['tab'] = 'autopost';
2531
+
2532
+ // get parameters
2533
+ $params = array(
2534
+ 'category_ids' => null,
2535
+ 'category_condition' => 'include',
2536
+ 'title_tag' => 'h2',
2537
+ 'title_alignment' => 'left',
2538
+ 'title_position' => 'inside',
2539
+ 'image_alignment' => 'alternate',
2540
+ 'image_width' => 325,
2541
+ 'post_content' => 'excerpt',
2542
+ 'readmore' => __('Read more.', WYSIJA),
2543
+ 'show_divider' => 'yes',
2544
+ 'post_limit' => 5,
2545
+ 'post_type' => 'post',
2546
+ 'author_show' => 'no',
2547
+ 'author_label' => __('Author:', WYSIJA),
2548
+ 'category_show' => 'no',
2549
+ 'category_label' => __('Categories:', WYSIJA),
2550
+ 'nopost_message' => __('Latest content already sent.', WYSIJA),
2551
+ 'bgcolor1' => null,
2552
+ 'bgcolor2' => null,
2553
+ 'sort_by' => 'newest'
2554
+ );
2555
+
2556
+ // backwards compatibility since we replaced the 'cpt' parameter by 'post_type' in 2.6
2557
+ if(isset($_GET['cpt']) && strlen(trim($_GET['cpt'])) > 0) {
2558
+ $params['post_type'] = trim($_GET['cpt']);
2559
+ }
2560
+
2561
+ // check if GET parameters are specified
2562
+ foreach ($params as $key => $value) {
2563
+ if (array_key_exists($key, $_GET)) {
2564
+ switch ($key) {
2565
+ case 'autopost_count':
2566
+ $params[$key] = (int)$_GET[$key];
2567
+ break;
2568
+ case 'author_label':
2569
+ case 'category_label':
2570
+ case 'readmore':
2571
+ case 'nopost_message':
2572
+ $params[$key] = base64_decode($_GET[$key]);
2573
+ break;
2574
+ default:
2575
+ $params[$key] = trim($_GET[$key]);
2576
+ }
2577
+ }
2578
+ }
2579
+
2580
+ // get autopost count
2581
+ $this->data['autopost_count'] = (array_key_exists('autopost_count', $_GET)) ? (int) $_GET['autopost_count'] : 0;
2582
+
2583
+ // get autopost type (single or multiple)
2584
+ $this->data['autopost_type'] = (array_key_exists('autopost_type', $_GET)) ? $_GET['autopost_type'] : 'multiple';
2585
+
2586
+ // if only one group of post can be added, change default alignment to left
2587
+ if ($this->data['autopost_type'] === 'single') {
2588
+ if ($params['image_alignment'] === 'alternate')
2589
+ $params['image_alignment'] = 'left';
2590
+ }
2591
+
2592
+ // we use that now, because categories from a post are different than categories from a CPT
2593
+ // $helper_wp_tools = WYSIJA::get('wp_tools','helper');
2594
+ // $this->data['categories'] = $helper_wp_tools->get_categories();
2595
+
2596
+ // max number of posts
2597
+ $this->data['post_limits'] = array(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 20, 30, 50);
2598
+
2599
+ $this->data['params'] = $params;
2600
+
2601
+ return $this->popupContent();
2602
+ }
2603
+
2604
+ function image_data() {
2605
+ $this->data['url'] = (isset($_GET['url']) && $_GET['url'] !== '') ? trim(urldecode($_GET['url'])) : null;
2606
+ $this->data['alt'] = (isset($_GET['alt'])) ? trim(urldecode($_GET['alt'])) : '';
2607
+
2608
+ $this->iframeTabs = array('image_data' => __("Image Parameters", WYSIJA));
2609
+ $_GET['tab'] = 'image_data';
2610
+ return $this->popupContent();
2611
+ }
2612
+
2613
+ function medias() {
2614
+ $this->popupContent();
2615
+ }
2616
+
2617
+ function special_wysija_browse() {
2618
+ $this->_wysija_subaction();
2619
+ $this->jsTrans['deleteimg'] = __('Delete image for all newsletters?', WYSIJA);
2620
+ return wp_iframe(array($this->viewObj, 'popup_wysija_browse'), array());
2621
+ }
2622
+
2623
+ function special_wordp_browse() {
2624
+ $this->_wysija_subaction();
2625
+ $this->jsTrans['deleteimg'] = __('This image might be in an article. Delete anyway?', WYSIJA);
2626
+ return wp_iframe(array($this->viewObj, 'popup_wp_browse'), array());
2627
+ }
2628
+
2629
+ function _wysija_subaction() {
2630
+
2631
+ if (isset($_REQUEST['subaction'])) {
2632
+ if ($_REQUEST['subaction'] === 'delete') {
2633
+ if (isset($_REQUEST['imgid']) && (int) $_REQUEST['imgid'] > 0) {
2634
+ /* delete the image with id imgid */
2635
+ $res = wp_delete_attachment((int) $_REQUEST['imgid'], true);
2636
+ if ($res) {
2637
+ $this->notice(__('Image has been deleted.', WYSIJA));
2638
+ }
2639
+ }
2640
+ }
2641
+ }
2642
+ return true;
2643
+ }
2644
+
2645
+ function special_new_wordp_upload() {
2646
+
2647
+ //wp_enqueue_script('plupload-all');
2648
+ wp_enqueue_script('wysija-plupload-handlers', WYSIJA_URL . 'js/jquery/pluploadHandler.js', array('plupload-all', 'jquery'));
2649
+ $uploader_l10n = array(
2650
+ 'queue_limit_exceeded' => __('You have attempted to queue too many files.'),
2651
+ 'file_exceeds_size_limit' => __('%s exceeds the maximum upload size for this site.'),
2652
+ 'zero_byte_file' => __('This file is empty. Please try another.'),
2653
+ 'invalid_filetype' => __('This file type is not allowed. Please try another.'),
2654
+ 'not_an_image' => __('This file is not an image. Please try another.'),
2655
+ 'image_memory_exceeded' => __('Memory exceeded. Please try another smaller file.'),
2656
+ 'image_dimensions_exceeded' => __('This is larger than the maximum size. Please try another.'),
2657
+ 'default_error' => __('An error occurred in the upload. Please try again later.'),
2658
+ 'missing_upload_url' => __('There was a configuration error. Please contact the server administrator.'),
2659
+ 'upload_limit_exceeded' => __('You may only upload 1 file.'),
2660
+ 'http_error' => __('HTTP error.'),
2661
+ 'upload_failed' => __('Upload failed.'),
2662
+ 'big_upload_failed' => __('Please try uploading this file with the %1$sbrowser uploader%2$s.'),
2663
+ 'big_upload_queued' => __('%s exceeds the maximum upload size for the multi-file uploader when used in your browser.'),
2664
+ 'io_error' => __('IO error.'),
2665
+ 'security_error' => __('Security error.'),
2666
+ 'file_cancelled' => __('File canceled.'),
2667
+ 'upload_stopped' => __('Upload stopped.'),
2668
+ 'dismiss' => __('Dismiss'),
2669
+ 'crunching' => __('Crunching&hellip;'),
2670
+ 'deleted' => __('moved to the trash.'),
2671
+ 'error_uploading' => __('&#8220;%s&#8221; has failed to upload.')
2672
+ );
2673
+
2674
+ wp_localize_script('wysija-plupload-handlers', 'pluploadL10n', $uploader_l10n);
2675
+
2676
+ wp_enqueue_script('image-edit');
2677
+ wp_enqueue_script('set-post-thumbnail');
2678
+ wp_enqueue_style('imgareaselect');
2679
+ wp_enqueue_script('media-gallery');
2680
+
2681
+ /* wp_register_style('myplupload', '/adjust-this-url/myplupload.css');
2682
+ wp_enqueue_style('myplupload'); */
2683
+
2684
+
2685
+ $errors = array();
2686
+ return wp_iframe(array($this->viewObj, 'popup_new_wp_upload'), $errors);
2687
+ }
2688
+
2689
+ function special_wordp_upload() {
2690
+
2691
+ wp_enqueue_script('swfupload-all');
2692
+ wp_enqueue_script('swfupload-handlers');
2693
+ wp_enqueue_script('wysija-upload-handlers', WYSIJA_URL . "js/jquery/uploadHandlers.js");
2694
+ wp_enqueue_script('image-edit');
2695
+ wp_enqueue_script('set-post-thumbnail');
2696
+ wp_enqueue_style('imgareaselect');
2697
+
2698
+ $errors = array();
2699
+ $id = 0;
2700
+ if (isset($_GET['flash']))
2701
+ $_GET['flash'] = 1;
2702
+ if (isset($_POST['html-upload']) && !empty($_FILES)) {
2703
+ // Upload File button was clicked
2704
+ $id = media_handle_upload('async-upload', $_REQUEST['post_id']);
2705
+ unset($_FILES);
2706
+ if (is_wp_error($id)) {
2707
+ $errors['upload_error'] = $id;
2708
+ $id = false;
2709
+ }
2710
+ }
2711
+
2712
+ if (!empty($_POST['insertonlybutton'])) {
2713
+ $href = $_POST['insertonly']['href'];
2714
+ if (!empty($href) && !strpos($href, '://'))
2715
+ $href = "http://$href";
2716
+
2717
+ $title = esc_attr($_POST['insertonly']['title']);
2718
+ if (empty($title))
2719
+ $title = basename($href);
2720
+ if (!empty($title) && !empty($href))
2721
+ $html = "<a href='" . esc_url($href) . "' >$title</a>";
2722
+ $html = apply_filters('file_send_to_editor_url', $html, esc_url_raw($href), $title);
2723
+ return media_send_to_editor($html);
2724
+ }
2725
+
2726
+ if (!empty($_POST)) {
2727
+ $return = media_upload_form_handler();
2728
+
2729
+ if (is_string($return))
2730
+ return $return;
2731
+ if (is_array($return))
2732
+ $errors = $return;
2733
+ }
2734
+
2735
+ if (isset($_POST['save'])) {
2736
+ $errors['upload_notice'] = __('Saved.', WYSIJA);
2737
+ return media_upload_gallery();
2738
+ }
2739
+
2740
+
2741
+ return wp_iframe(array($this->viewObj, 'popup_wp_upload'), $errors);
2742
+ }
2743
+
2744
+ function _checkEmailExists($emailId) {
2745
+ $result = false;
2746
+ $modelEmail = WYSIJA::get('email', 'model');
2747
+
2748
+ if ($modelEmail->exists(array('email_id' => $emailId)))
2749
+ $result = true;
2750
+
2751
+ if (!$result) {
2752
+ $this->error(__("The newsletter doesn't exist.", WYSIJA), 1);
2753
+ $this->redirect('admin.php?page=wysija_campaigns');
2754
+ }
2755
+ else
2756
+ return true;
2757
+ }
2758
+
2759
+ }
controllers/back/config.php ADDED
@@ -0,0 +1,383 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_control_back_config extends WYSIJA_control_back{
4
+ var $view='config';
5
+ var $model='config';
6
+
7
+ function WYSIJA_control_back_config(){
8
+
9
+ }
10
+
11
+ function main() {
12
+ parent::WYSIJA_control_back();
13
+ wp_enqueue_style('thickbox');
14
+
15
+ if(!isset($_REQUEST['action'])) $this->action='main';
16
+ else $this->action=$_REQUEST['action'];
17
+ $this->jsTrans['testemail'] = __('Sending a test email', WYSIJA);
18
+ $this->jsTrans['bounceconnect'] = __('Bounce handling connection test', WYSIJA);
19
+ $this->jsTrans['processbounceT'] = __('Bounce handling processing', WYSIJA);
20
+ $this->jsTrans['doubleoptinon'] = __('Subscribers will now need to activate their subscription by email in order to receive your newsletters. This is recommended.', WYSIJA);
21
+ $this->jsTrans['doubleoptinoff'] = __('Unconfirmed subscribers will receive your newsletters from now on without the need to activate their subscriptions.', WYSIJA);
22
+ $this->jsTrans['processbounce'] = __('Process bounce handling now!', WYSIJA);
23
+ $this->jsTrans['errorbounceforward'] = __('When setting up the bounce system, you need to have a different address for the bounce email and the forward to address', WYSIJA);
24
+
25
+ // form list
26
+ $this->jsTrans['suredelete'] = __('Are you sure you want to delete this form?', WYSIJA);
27
+
28
+ switch($this->action) {
29
+ case 'log':
30
+ case 'save':
31
+ case 'clearlog':
32
+ wp_enqueue_script('wysija-config-settings', WYSIJA_URL.'js/admin-config-settings.js', array('wysija-admin-js-global'), WYSIJA::get_version());
33
+ wp_localize_script('wysija-config-settings', 'mpEmailCheck', WJ_Utils::get_tip_data());
34
+ wp_enqueue_script('jquery-cookie', WYSIJA_URL.'js/jquery/jquery.cookie.js', array('jquery'), WYSIJA::get_version());
35
+ case 'form_add':
36
+ case 'form_edit':
37
+ case 'form_duplicate':
38
+ case 'form_delete':
39
+ case 'form_widget_settings':
40
+ case 'form_add_field':
41
+ return $this->{$this->action}();
42
+ break;
43
+ case 'reinstall':
44
+ $this->reinstall();
45
+ return;
46
+ break;
47
+ case 'dkimcheck':
48
+ $this->dkimcheck();
49
+ if(defined('WYSIJA_REDIRECT')) $this->redirectProcess();
50
+ return;
51
+ break;
52
+ case 'doreinstall':
53
+ $this->doreinstall();
54
+ if(defined('WYSIJA_REDIRECT')){
55
+ global $wysi_location;
56
+ $wysi_location='admin.php?page=wysija_campaigns';
57
+ $this->redirectProcess();
58
+ }
59
+ return;
60
+ break;
61
+ default:
62
+ wp_enqueue_script( 'mailpoet.tooltip', WYSIJA_URL . 'js/vendor/bootstrap.tooltip.js', array( 'jquery' ), WYSIJA::get_version(), true );
63
+ wp_enqueue_style( 'mailpoet.tooltip', WYSIJA_URL . 'css/vendor/bootstrap.tooltip.css', array(), WYSIJA::get_version(), 'screen' );
64
+ wp_enqueue_script('wysija-config-settings', WYSIJA_URL.'js/admin-config-settings.js', array('wysija-admin-js-global'), WYSIJA::get_version(), true);
65
+ wp_localize_script('wysija-config-settings', 'mpEmailCheck', WJ_Utils::get_tip_data());
66
+ wp_enqueue_script('jquery-cookie', WYSIJA_URL.'js/jquery/jquery.cookie.js', array('jquery'), WYSIJA::get_version());
67
+ }
68
+
69
+ if(WYSIJA_DBG > 1) {
70
+ $this->viewObj->arrayMenus = array('log' => 'View log');
71
+ }
72
+
73
+ $this->data = array();
74
+ $hook_settings_super_advanced_params = array();
75
+ $this->data['hooks']['hook_settings_super_advanced'] = apply_filters('hook_settings_super_advanced',WYSIJA_module::execute_hook('hook_settings_super_advanced', $hook_settings_super_advanced_params), $hook_settings_super_advanced_params);
76
+ $this->action='main';
77
+
78
+ if(isset($_REQUEST['validate'])){
79
+ $this->notice(str_replace(array('[link]','[/link]'),
80
+ array('<a title="'.__('Get Premium now',WYSIJA).'" class="premium-activate" href="javascript:;">','</a>'),
81
+ __('You\'re almost there. Click this [link]link[/link] to activate the licence you have just purchased.',WYSIJA)));
82
+
83
+ }
84
+
85
+ }
86
+
87
+ function dkimcheck(){
88
+
89
+ if(isset($_POST['xtz'])){
90
+
91
+ $dataconf=json_decode(base64_decode($_POST['xtz']));
92
+ if(isset($dataconf->dkim_pubk->key) && isset($dataconf->dkim_privk)){
93
+ $modelConf=WYSIJA::get('config','model');
94
+ $dataconfsave=array('dkim_pubk'=>$dataconf->dkim_pubk->key, 'dkim_privk'=>$dataconf->dkim_privk,'dkim_1024'=>1);
95
+
96
+ $modelConf->save($dataconfsave);
97
+ WYSIJA::update_option('dkim_autosetup',false);
98
+ }
99
+ }
100
+
101
+ $this->redirect('admin.php?page=wysija_config');
102
+ return true;
103
+ }
104
+
105
+ function save(){
106
+ $_REQUEST = stripslashes_deep($_REQUEST);
107
+ $_POST = stripslashes_deep($_POST);
108
+ $this->requireSecurity();
109
+
110
+ $hook_settings_before_save = array(
111
+ 'REQUEST' =>& $_REQUEST
112
+ );
113
+ apply_filters('hook_settings_before_save',WYSIJA_module::execute_hook('hook_settings_before_save', $hook_settings_before_save),$hook_settings_before_save);
114
+
115
+ $this->modelObj->save($_REQUEST['wysija']['config'],true);
116
+
117
+ $hook_settings_super_advanced_params = array();
118
+ $this->data['hooks']['hook_settings_super_advanced'] = apply_filters('hook_settings_super_advanced',WYSIJA_module::execute_hook('hook_settings_super_advanced', $hook_settings_super_advanced_params),$hook_settings_super_advanced_params);
119
+ // redirect so that javascript values get updated
120
+ wp_redirect('admin.php?page=wysija_config'.$_REQUEST['redirecttab']);
121
+ }
122
+
123
+ function reinstall(){
124
+ $this->viewObj->title=__('Reinstall MailPoet?',WYSIJA);
125
+ return true;
126
+ }
127
+
128
+ function changeMode(){
129
+ $helperFile=WYSIJA::get('file','helper');
130
+ $helperFile->chmodr(WYSIJA_UPLOADS_DIR, 0666, 0777);
131
+ $this->redirect('admin.php?page=wysija_config');
132
+ return true;
133
+ }
134
+
135
+ function doreinstall(){
136
+
137
+ if(isset($_REQUEST['postedfrom']) && $_REQUEST['postedfrom'] === 'reinstall') {
138
+ $uninstaller=WYSIJA::get('uninstall','helper');
139
+ $uninstaller->reinstall();
140
+ }
141
+ $this->redirect('admin.php?page=wysija_config');
142
+ return true;
143
+ }
144
+
145
+ function render(){
146
+ $this->checkTotalSubscribers();
147
+ $this->viewObj->render($this->action,$this->data);
148
+ }
149
+
150
+ function log(){
151
+ $this->viewObj->arrayMenus=array('clearlog'=>'Clear log');
152
+ $this->viewObj->title='MailPoet\'s log';
153
+ return true;
154
+ }
155
+
156
+ function clearlog(){
157
+ update_option('wysija_log', array());
158
+ $this->redirect('admin.php?page=wysija_config&action=log');
159
+ return true;
160
+ }
161
+
162
+ // WYSIJA Form Editor
163
+ function form_add() {
164
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
165
+ // set default form data
166
+ $helper_form_engine->set_data();
167
+
168
+ // create form in database with default data
169
+ $form = array('name' => __('New Form', WYSIJA));
170
+
171
+ // insert into form table
172
+ $model_forms = WYSIJA::get('forms', 'model');
173
+ $form_id = $model_forms->insert($form);
174
+
175
+ if($form_id !== null && (int)$form_id > 0) {
176
+ // merge form_id into form data for later use
177
+ $data = array_merge(array('form_id' => $form_id), $helper_form_engine->get_data());
178
+ // update form data in form engine
179
+ $helper_form_engine->set_data($data);
180
+ // update form data in database
181
+ $model_forms->update(array('data' => $helper_form_engine->get_encoded('data')), array('form_id' => (int)$form_id));
182
+
183
+ // redirect to form editor, passing along the newly created form id
184
+ WYSIJA::redirect('admin.php?page=wysija_config&action=form_edit&id='.$form_id);
185
+ } else {
186
+ WYSIJA::redirect('admin.php?page=wysija_config#tab-forms');
187
+ }
188
+ return true;
189
+ }
190
+
191
+ function form_duplicate() {
192
+
193
+ if(isset($_GET['id']) && (int)$_GET['id'] > 0) {
194
+ $form_id = (int)$_GET['id'];
195
+
196
+ $model_forms = WYSIJA::get('forms', 'model');
197
+
198
+ // get form data
199
+ $form = $model_forms->getOne(array('name', 'data', 'styles'), array('form_id' => $form_id));
200
+
201
+ if(empty($form)) {
202
+ $this->error(__('This form does not exist', WYSIJA), true);
203
+ } else {
204
+ // reset model forms
205
+ $model_forms->reset();
206
+
207
+ // add "copy" to the name
208
+ $form['name'] = $form['name'].' - '.__('Copy', WYSIJA);
209
+
210
+ // insert form (duplicated)
211
+ $model_forms->insert($form);
212
+
213
+ // display notice
214
+ $this->notice(sprintf(__('The form named "%1$s" has been created.', WYSIJA), $form['name']));
215
+ }
216
+ }
217
+
218
+ WYSIJA::redirect('admin.php?page=wysija_config#tab-forms');
219
+ }
220
+
221
+ function form_delete() {
222
+
223
+ $this->requireSecurity();
224
+
225
+ if(isset($_GET['id']) && (int)$_GET['id'] > 0) {
226
+ $form_id = (int)$_GET['id'];
227
+
228
+ $model_forms = WYSIJA::get('forms', 'model');
229
+
230
+ // get form data
231
+ $form = $model_forms->getOne(array('name'), array('form_id' => $form_id));
232
+
233
+ if(empty($form)) {
234
+ $this->error(__('This form has already been deleted.', WYSIJA), true);
235
+ } else {
236
+ // delete the form in the database
237
+ $model_forms->reset();
238
+ $model_forms->delete(array('form_id' => $form_id));
239
+
240
+ // display notice
241
+ $this->notice(sprintf(__('The form named "%1$s" has been deleted.', WYSIJA), $form['name']));
242
+ }
243
+ }
244
+
245
+ WYSIJA::redirect('admin.php?page=wysija_config#tab-forms');
246
+ }
247
+
248
+ function form_edit() {
249
+ // define whether the form can be edited
250
+ $this->data['can_edit'] = true;
251
+
252
+ // wysija form editor javascript files
253
+ $this->js[]='wysija-form-editor';
254
+ $this->js[]='wysija-admin-ajax-proto';
255
+ // $this->js[]='wysija-admin-ajax';
256
+ $this->js[]='wysija-base-script-64';
257
+ $this->js[] = 'mailpoet-select2';
258
+
259
+ // make sure the editor content is not cached
260
+ //header('Cache-Control: no-cache, max-age=0, must-revalidate, no-store'); // HTTP/1.1
261
+ //header('Expires: Fri, 9 Mar 1984 00:00:00 GMT');
262
+
263
+ // get form id
264
+ $form_id = (isset($_REQUEST['id']) && (int)$_REQUEST['id'] > 0) ? (int)$_REQUEST['id'] : null;
265
+ $form = array('name' => __('New form', WYSIJA));
266
+
267
+ // if no form id was specified, then it's a new form
268
+ if($form_id !== null) {
269
+ // try to get form data based on form id
270
+ $model_forms = WYSIJA::get('forms', 'model');
271
+ $form = $model_forms->getOne($form_id);
272
+
273
+ // if the form does not exist
274
+ if(empty($form)) {
275
+ // redirect to forms list
276
+ $this->error(__('This form does not exist.', WYSIJA), true);
277
+ WYSIJA::redirect('admin.php?page=wysija_config#tab-forms');
278
+ } else {
279
+ // pass form id to the view
280
+ $this->data['form_id'] = (int)$form['form_id'];
281
+ }
282
+ }
283
+ // pass form to the view
284
+ $this->data['form'] = $form;
285
+
286
+ $helper_form_engine = WYSIJA::get('form_engine', 'helper');
287
+ $lists = $helper_form_engine->get_lists();
288
+ $this->data['lists'] = $lists;
289
+
290
+ // disable editing capability when there is no list
291
+ if(empty($lists)) {
292
+ $this->data['can_edit'] = false;
293
+ }
294
+
295
+ // get custom fields
296
+ $this->data['custom_fields'] = $helper_form_engine->get_custom_fields();
297
+
298
+ // translations
299
+ $this->jsTrans = array_merge($this->jsTrans, $helper_form_engine->get_translations());
300
+ }
301
+
302
+ /*
303
+ * Handles the settings popup of wysija form widgets
304
+ */
305
+ function form_widget_settings() {
306
+ $this->iframeTabs = array('form_widget_settings' => __('Widget Settings', WYSIJA));
307
+ $this->js[] = 'wysija-admin-ajax';
308
+ $this->js[] = 'wysija-base-script-64';
309
+ $this->js[] = 'wysija-scriptaculous';
310
+
311
+ $_GET['tab'] = 'form_widget_settings';
312
+
313
+ // if there is a field id, let's get all that from this field
314
+ if(isset($_REQUEST['field_id'])) {
315
+ $field_id = ((int)$_REQUEST['field_id'] > 0) ? (int)$_REQUEST['field_id'] : 0;
316
+
317
+ // if the id is positive then try to fetch field data
318
+ $custom_field = WJ_Field::get($field_id);
319
+
320
+ // if field has been found
321
+ if($custom_field !== NULL) {
322
+ $this->data['name'] = (isset($_REQUEST['name'])) ? $_REQUEST['name'] : $custom_field->name;
323
+ $this->data['type'] = (isset($_REQUEST['type'])) ? $_REQUEST['type'] : $custom_field->type;
324
+ $this->data['field'] = $custom_field->user_column_name();
325
+ $this->data['params'] = $custom_field->settings;
326
+ } else {
327
+ $this->data['name'] = (isset($_REQUEST['name'])) ? $_REQUEST['name'] : null;
328
+ $this->data['type'] = (isset($_REQUEST['type'])) ? $_REQUEST['type'] : null;
329
+ $this->data['field'] = null;
330
+ $this->data['params'] = null;
331
+ }
332
+
333
+ $this->data['field_id'] = $field_id;
334
+ } else {
335
+ // extract parameters from url
336
+ $params = array();
337
+ if(isset($_REQUEST['params']) && trim(strlen($_REQUEST['params'])) > 0) {
338
+ $pairs = explode('|', $_REQUEST['params']);
339
+ if(count($pairs) > 0) {
340
+ foreach($pairs as $pair) {
341
+ // extract both key and value
342
+ list($key, $value) = explode(':', $pair);
343
+
344
+ // decode value
345
+ $value = base64_decode($value);
346
+ // unserialize if necessary (using is_serialized from WordPress)
347
+ if(is_serialized($value) === true) {
348
+ $value = unserialize($value);
349
+ }
350
+ $params[$key] = $value;
351
+ }
352
+ }
353
+ }
354
+
355
+ // common widget data
356
+ $this->data['name'] = (isset($_REQUEST['name'])) ? $_REQUEST['name'] : null;
357
+ $this->data['type'] = (isset($_REQUEST['type'])) ? $_REQUEST['type'] : null;
358
+ $this->data['field'] = (isset($_REQUEST['field'])) ? $_REQUEST['field'] : null;
359
+
360
+ // widget params
361
+ $this->data['params'] = $params;
362
+
363
+ // extra data that needs to be fetched for some widget
364
+ $extra = array();
365
+
366
+ switch($this->data['type']) {
367
+ // in case of the list widget, we need to pass an array of all available lists
368
+ case 'list':
369
+ $model_list = WYSIJA::get('list', 'model');
370
+
371
+ // get lists users can subscribe to (aka "enabled list")
372
+ $extra['lists'] = $model_list->get(array('name', 'list_id', 'is_public'), array('is_enabled' => 1));
373
+ break;
374
+ }
375
+
376
+ $this->data['extra'] = $extra;
377
+ }
378
+
379
+ return $this->popupContent();
380
+ exit;
381
+ }
382
+ // End: WYSIJA Form Editor
383
+ }
controllers/back/index.html ADDED
File without changes
controllers/back/premium.php ADDED
@@ -0,0 +1,34 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ defined('WYSIJA') or die('Restricted access');
4
+
5
+ class WYSIJA_control_back_premium extends WYSIJA_control_back {
6
+
7
+ /**
8
+ * Main view of this controller
9
+ * @var string
10
+ */
11
+ public $view = 'premium';
12
+ public $model = 'config';
13
+
14
+
15
+ /**
16
+ * Constructor
17
+ */
18
+ function WYSIJA_control_back_premium() {
19
+ }
20
+ function defaultDisplay() {
21
+ $this->jsTrans['premium_activate'] = __('Already paid? Click here to activate', WYSIJA);
22
+ $this->jsTrans['premium_activating'] = __('Checking license', WYSIJA);
23
+ }
24
+
25
+ function premium_ben() {
26
+ $this->jsTrans['premium_activate'] = __('Already paid? Click here to activate', WYSIJA);
27
+ $this->jsTrans['premium_activating'] = __('Checking license', WYSIJA);
28
+ }
29
+
30
+ function premium_kim() {
31
+ $this->jsTrans['premium_activate'] = __('Already paid? Click here to activate', WYSIJA);
32
+ $this->jsTrans['premium_activating'] = __('Checking license', WYSIJA);
33
+ }
34
+ }
controllers/back/statistics.php ADDED
@@ -0,0 +1,197 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ require_once(WYSIJA_CORE.'module'.DS.'statistics.php'); // @todo
4
+
5
+ class WYSIJA_control_back_statistics extends WYSIJA_control_back
6
+ {
7
+
8
+ /**
9
+ * Main model of this controller
10
+ * @var string
11
+ */
12
+ public $model = 'statistics';
13
+
14
+ /**
15
+ * Main view of this controller
16
+ * @var string
17
+ */
18
+ public $view = 'statistics';
19
+
20
+ /**
21
+ * Base URL of all requests
22
+ * @var string
23
+ */
24
+ public $base_url = 'admin.php';
25
+
26
+ /**
27
+ * Load blocks at a same time (FALSE) or one by one (TRUE)
28
+ * @var TRUE
29
+ */
30
+ protected $lazy_load = true;
31
+
32
+ /**
33
+ * list of pre-defined dates
34
+ * @var Array
35
+ */
36
+ protected $pre_defined_dates = array( );
37
+
38
+ protected $date_format = 'Y/m/d';
39
+
40
+ protected $js_date_format = 'yy/mm/dd';
41
+
42
+ /**
43
+ * Constructor
44
+ */
45
+ function WYSIJA_control_back_statistics()
46
+ {
47
+
48
+ }
49
+
50
+ public function defaultDisplay()
51
+ {
52
+ $this->pre_defined_dates = $this->get_pre_defined_dates();
53
+ // Define view
54
+ $this->viewShow = $this->action = 'main';
55
+ $this->js['jquery.core'] = 'jquery/ui/jquery.ui.core';
56
+ $this->js['jquery.datepicker'] = 'jquery/ui/jquery.ui.datepicker';
57
+ $this->js['wysijalazyload'] = 'wysija-lazyload';
58
+ $this->js['admin-statistics-filter'] = 'admin-statistics-filter';
59
+ wp_enqueue_style('jquery.core', WYSIJA_URL.'css/jquery/ui/themes/base/jquery.ui.core.min.css', array( ), WYSIJA::get_version());
60
+ wp_enqueue_style('jquery.core', WYSIJA_URL.'css/jquery/ui/themes/base/jquery.ui.theme.min.css', array( ), WYSIJA::get_version());
61
+
62
+ // date filter
63
+ $default_duration = $this->get_default_duration();
64
+ if (function_exists('date_diff'))
65
+ {
66
+ $this->data['date_interval'] = date_diff(date_create($default_duration['from']), date_create($default_duration['to']));
67
+ }
68
+ else
69
+ {
70
+ $duration = strtotime($default_duration['to']) - strtotime($default_duration['from']);
71
+ $helper_toolbox = WYSIJA::get('toolbox', 'helper');
72
+ $this->data['date_interval'] = (object)$helper_toolbox->convert_seconds_to_array($duration, false);
73
+ }
74
+
75
+ $this->data['custom_dates'] = $this->pre_defined_dates;
76
+ $this->data['default_duration'] = $default_duration;
77
+ $this->data['js_date_format'] = $this->js_date_format;
78
+
79
+ // Process and push data into view
80
+ $this->data['lazy_load'] = $this->lazy_load;
81
+ $hook_name = 'hook_stats';
82
+ $hook_params = array( );
83
+ $hook_params['top'] = WYSIJA_module_statistics::DEFAULT_TOP_RECORDS;
84
+ $hook_params['from'] = !empty($_REQUEST['filter']['from']) ? $_REQUEST['filter']['from'] : $default_duration['from'];
85
+ $hook_params['to'] = !empty($_REQUEST['filter']['to']) ? $_REQUEST['filter']['to'] : $default_duration['to'];
86
+ $hook_params['group_by'] = ($this->data['date_interval']->days == 0 || $this->data['date_interval']->days > WYSIJA_module_statistics::SWITCHING_DATE_TO_MONTH_THRESHOLD) ?
87
+ WYSIJA_module_statistics::GROUP_BY_MONTH :
88
+ WYSIJA_module_statistics::GROUP_BY_DATE; // $this->data['date_interval']->days == 0, means, no begin date, no end date
89
+
90
+ // Hack!
91
+ $_REQUEST['limit_pp'] = $hook_params['top'];// Pagination, mark current selected value
92
+
93
+ $modules = WYSIJA_module::get_modules_from_hook($hook_name);
94
+ $this->data['modules'] = $modules;
95
+ $this->data['lazy_load_modules'] = array( );
96
+ $this->data['first_module'] = '';
97
+
98
+ if (!$this->lazy_load)
99
+ {
100
+ $this->data['hooks'][$hook_name] = apply_filters('hook_stats', '', $hook_params);
101
+ }
102
+ else
103
+ {
104
+ if (!empty($modules))
105
+ {
106
+ $first_module = array_shift($modules);
107
+ // List of lazy loaded modules
108
+ $this->data['lazy_load_modules'] = $modules;
109
+
110
+ // Evenly we are lazy loading, we always load the first module by default
111
+ $this->data['first_module'] = apply_filters('custom_module_hook', '', $first_module, $hook_name, $hook_params);
112
+ }
113
+ }
114
+ }
115
+
116
+ /**
117
+ * get pre defined dates (duration)
118
+ * @return type
119
+ */
120
+ protected function get_pre_defined_dates()
121
+ {
122
+ return array(
123
+ array(
124
+ 'value' => 7,
125
+ 'label' => __('Last 7 days', WYSIJA),
126
+ 'selected' => false,
127
+ 'from' => date($this->date_format, strtotime('-7 days')),
128
+ 'to' => date($this->date_format, strtotime('tomorrow'))
129
+ ),
130
+ array(
131
+ 'value' => 'last_month',
132
+ 'label' => __('Last month', WYSIJA),
133
+ 'selected' => false,
134
+ 'from' => date($this->date_format, mktime(0, 0, 0, date('m')-1, 1, date('Y'))),
135
+ 'to' => date($this->date_format, mktime(0, 0, 0, date('m'), 0, date('Y')))
136
+
137
+
138
+ ),
139
+ array(
140
+ 'value' => 30,
141
+ 'label' => __('Last 30 days', WYSIJA),
142
+ 'selected' => false,
143
+ 'from' => date($this->date_format, strtotime('-30 days')),
144
+ 'to' => date($this->date_format, strtotime('tomorrow'))
145
+ ),
146
+ array(
147
+ 'value' => 90,
148
+ 'label' => __('Last 90 days', WYSIJA),
149
+ 'selected' => true,
150
+ 'from' => date($this->date_format, strtotime('-90 days')),
151
+ 'to' => date($this->date_format, strtotime('tomorrow'))
152
+ ),
153
+ array(
154
+ 'value' => 180,
155
+ 'label' => __('Last 180 days', WYSIJA),
156
+ 'selected' => false,
157
+ 'from' => date($this->date_format, strtotime('-180 days')),
158
+ 'to' => date($this->date_format, strtotime('tomorrow'))
159
+ ),
160
+ array(
161
+ 'value' => 365,
162
+ 'label' => __('Last 365 days', WYSIJA),
163
+ 'selected' => false,
164
+ 'from' => date($this->date_format, strtotime('-365 days')),
165
+ 'to' => date($this->date_format, strtotime('tomorrow'))
166
+ ),
167
+ array(
168
+ 'value' => 0,
169
+ 'label' => __('Custom dates', WYSIJA),
170
+ 'selected' => false,
171
+ 'from' => '',
172
+ 'to' => ''
173
+ ),
174
+ );
175
+ }
176
+
177
+ /**
178
+ * Get default duration of stats
179
+ * @return int
180
+ */
181
+ protected function get_default_duration()
182
+ {
183
+ foreach ($this->pre_defined_dates as $duration)
184
+ if (isset($duration['selected']) && $duration['selected'])
185
+ return $duration;
186
+ return end($this->pre_defined_dates);
187
+ }
188
+
189
+ function date_diff($time_start, $time_end)
190
+ {
191
+ $result = null;
192
+ $duration = $time_end - $time_start;
193
+ $result->days = floor($duration / (60 * 60 * 24));
194
+ return $result;
195
+ }
196
+
197
+ }
controllers/back/subscribers.php ADDED
@@ -0,0 +1,1051 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_control_back_subscribers extends WYSIJA_control_back{
4
+ var $model='user';
5
+ var $view='subscribers';
6
+ var $list_columns=array('user_id','firstname', 'lastname','email','created_at');
7
+ var $searchable=array('email','firstname', 'lastname');
8
+ var $_separators = array(',', ';'); // csv separator; comma is for standard csv, semi-colon is good for Excel
9
+ var $_default_separator = ';';
10
+
11
+ /**
12
+ * Inactive users = users who never opened or clicked
13
+ * @todo: disabled on 2.6. Once it's enabled, please double check in term of "inactive users".
14
+ * OR - users who never opened or clicked
15
+ * OR - users who never opened or clicked AND received at least 1 newsletter.
16
+ * @var boolean
17
+ */
18
+ var $_filter_by_inactive_users = false;
19
+
20
+ function WYSIJA_control_back_subscribers(){
21
+ WYSIJA_control_back::WYSIJA_control_back();
22
+ if ($this->_filter_by_inactive_users) {
23
+ if (
24
+ // default display
25
+ empty($_REQUEST['action'])
26
+ // bulk action
27
+ || !empty($_REQUEST['doaction']) && trim(strtolower($_REQUEST['doaction'])) === 'apply'
28
+ ) {
29
+ $this->modelObj->prepare_inactive_users_table();
30
+ }
31
+ }
32
+ }
33
+
34
+ function save(){
35
+ $this->redirectAfterSave=false;
36
+ $helperUser=WYSIJA::get('user','helper');
37
+ if(isset($_REQUEST['id'])){
38
+ $id=$_REQUEST['id'];
39
+ parent::save();
40
+
41
+ //run the unsubscribe process if needed
42
+ if((int)$_REQUEST['wysija']['user']['status']==-1){
43
+ $helperUser->unsubscribe($id);
44
+ }
45
+
46
+ /* update subscriptions */
47
+ $modelUL=WYSIJA::get('user_list','model');
48
+ $modelUL->backSave=true;
49
+ /* list of core list */
50
+ $modelLIST=WYSIJA::get('list','model');
51
+ $results=$modelLIST->get(array('list_id'),array('is_enabled'=>'0'));
52
+ $core_listids=array();
53
+ foreach($results as $res){
54
+ $core_listids[]=$res['list_id'];
55
+ }
56
+
57
+ //0 - get current lists of the user
58
+ $userlists=$modelUL->get(array('list_id','unsub_date'),array('user_id'=>$id));
59
+
60
+ $oldlistids=$newlistids=array();
61
+ foreach($userlists as $listdata) $oldlistids[$listdata['list_id']]=$listdata['unsub_date'];
62
+
63
+ $config=WYSIJA::get('config','model');
64
+ $dbloptin=$config->getValue('confirm_dbleoptin');
65
+ //1 - insert new user_list
66
+ if(isset($_POST['wysija']['user_list']) && $_POST['wysija']['user_list']){
67
+ $modelUL->reset();
68
+ $modelUL->update(array('sub_date'=>time()),array('user_id'=>$id));
69
+ if(!empty($_POST['wysija']['user_list']['list_id'])){
70
+ foreach($_POST['wysija']['user_list']['list_id'] as $list_id){
71
+ //if the list is not already recorded for the user then we will need to insert it
72
+ if(!isset($oldlistids[$list_id])){
73
+ $modelUL->reset();
74
+ $newlistids[]=$list_id;
75
+ $dataul=array('user_id'=>$id,'list_id'=>$list_id,'sub_date'=>time());
76
+ //if double optin is on and user is unconfirmed or unsubscribed, then we need to set it as unconfirmed subscription
77
+ if($dbloptin && (int)$_POST['wysija']['user']['status']<1) unset($dataul['sub_date']);
78
+ $modelUL->insert($dataul);
79
+ //if the list is recorded already then let's check the status, if it is an unsubed one then we update it
80
+ }else{
81
+ if($oldlistids[$list_id]>0){
82
+ $modelUL->reset();
83
+ $modelUL->update(array('unsub_date'=>0,'sub_date'=>time()),array('user_id'=>$id,'list_id'=>$list_id));
84
+ }
85
+ }
86
+ }
87
+ }
88
+
89
+ }else{
90
+ // if no list is selected we unsubscribe them all
91
+ $modelUL->reset();
92
+ $modelUL->update(array('unsub_date'=>time(),'sub_date'=>0),array('user_id'=>$id));
93
+ }
94
+
95
+ //if a confirmation email needs to be sent then we send it
96
+ if($dbloptin && (int)$_POST['wysija']['user']['status']==0 && !empty($newlistids)){
97
+ $hUser=WYSIJA::get('user','helper');
98
+ $hUser->sendConfirmationEmail($id,true,$newlistids);
99
+ }
100
+
101
+ if((int)$_POST['wysija']['user']['status']==0 || (int)$_POST['wysija']['user']['status']==1){
102
+ $modelUL->reset();
103
+ $modelUL->update(array('unsub_date'=>0,'sub_date'=>time()),array('user_id'=>$id,'list_id'=>$core_listids));
104
+ }
105
+
106
+ $arrayLists=array();
107
+ if(isset($_POST['wysija']['user_list']['list_id'])) $arrayLists=$_POST['wysija']['user_list']['list_id'];
108
+ $notEqual=array_merge($core_listids, $arrayLists);
109
+
110
+ //unsubscribe from lists which exist in the old list but does not exist in the new list
111
+ $unsubsribe_list = array_diff(array_keys($oldlistids), $arrayLists);
112
+ if(!empty($unsubsribe_list))
113
+ {
114
+ $modelUL->reset();
115
+ $modelUL->update(array('unsub_date'=>time()),array('user_id'=>$id,'list_id'=>$unsubsribe_list));
116
+ }
117
+ $modelUL->reset();
118
+
119
+ /*
120
+ Custom Fields.
121
+ */
122
+ if (isset($_POST['wysija']['field'])) {
123
+ WJ_FieldHandler::handle_all(
124
+ $_POST['wysija']['field'], $id
125
+ );
126
+ }
127
+
128
+
129
+ }else{
130
+ //instead of going through a classic save we should save through the helper
131
+ $data=$_REQUEST['wysija'];
132
+ $data['user_list']['list_ids'] = !empty($data['user_list']['list_id']) ? $data['user_list']['list_id'] : array();
133
+ unset($data['user_list']['list_id']);
134
+ $data['message_success']=__('Subscriber has been saved.',WYSIJA);
135
+ $id=$helperUser->addSubscriber($data,true);
136
+ //$id= parent::save();
137
+ if(!$id) {
138
+ $this->viewShow=$this->action='add';
139
+ $data=array('details'=>$_REQUEST['wysija']['user']);
140
+ return $this->add($data);
141
+ }
142
+ }
143
+ $this->redirect();
144
+ return true;
145
+ }
146
+
147
+
148
+ /**
149
+ * Get selected lists
150
+ * @return array
151
+ */
152
+ protected function get_selected_lists() {
153
+ $result = array();
154
+ if (isset($_REQUEST['wysija']['filter']['filter_list'])) {
155
+ $result[] = $_REQUEST['wysija']['filter']['filter_list'];
156
+ } elseif (!empty($_REQUEST['filter-list'])) {
157
+ $lists = explode(',', trim($_REQUEST['filter-list']));// currently, only single list is allowed.
158
+ if (!empty($lists)) {
159
+ $result = array_merge ($result, $lists);
160
+ }
161
+ }
162
+ return $result;
163
+ }
164
+
165
+ function defaultDisplay(){
166
+ $this->viewShow=$this->action='main';
167
+ $this->js[]='wysija-admin-list';
168
+ $this->viewObj->msgPerPage = __('Subscribers per page:',WYSIJA);
169
+
170
+ $this->jsTrans['selecmiss'] = __('Select at least 1 subscriber!',WYSIJA);
171
+
172
+ // get the total count for subscribed, unsubscribed and unconfirmed users
173
+ $select = array( 'COUNT(`user_id`) AS users' , 'status' , 'MAX(`created_at`) AS `max_create_at`');
174
+ $count_group_by = 'status';
175
+ $count_by_status = $this->modelObj->get_subscribers( $select , array() , $count_group_by );
176
+ if ($this->_filter_by_inactive_users) {
177
+ $inactive_users = $this->modelObj->count_inactive_users();
178
+ if ($inactive_users) {
179
+ array_unshift($count_by_status, array(
180
+ 'users' => $inactive_users['count'],
181
+ 'status' => -99,//-99 = inactive
182
+ 'max_create_at' => $inactive_users['max_created_at']
183
+ ));
184
+ }
185
+ }
186
+
187
+
188
+ $counts = $this->modelObj->structure_user_status_count_array($count_by_status);
189
+ $arr_max_create_at = $this->modelObj->get_max_create($count_by_status);
190
+
191
+ // count the rows based on the filters
192
+ $filters = $this->modelObj->detect_filters();
193
+
194
+ $select = array( 'COUNT(DISTINCT([wysija]user.user_id)) as total_users', 'MAX([wysija]user.created_at) as max_create_at');
195
+ $count_rows = $this->modelObj->get_subscribers( $select, $filters);
196
+
197
+ // without filter we already have the total number of subscribers
198
+ $this->data['max_create_at'] = null; //max value of create_at field of current list of users
199
+ if(!empty($filters)){
200
+ // used for pagination
201
+ $this->modelObj->countRows = $count_rows['total_users'];
202
+ // used for
203
+ $this->data['max_create_at'] = $count_rows['max_create_at'];
204
+ }else{
205
+ $this->data['max_create_at'] = !empty($arr_max_create_at) ? max($arr_max_create_at) : 0;
206
+ $this->modelObj->countRows=$counts['all'];
207
+ }
208
+
209
+ $select = array(
210
+ '[wysija]user.firstname',
211
+ '[wysija]user.lastname',
212
+ '[wysija]user.status',
213
+ '[wysija]user.email',
214
+ '[wysija]user.created_at',
215
+ '[wysija]user.last_opened',
216
+ '[wysija]user.last_clicked',
217
+ '[wysija]user_list.user_id'
218
+ );
219
+
220
+ $this->data['subscribers'] = $this->modelObj->get_subscribers($select , $filters, '', false, true);
221
+
222
+ $this->data['current_counts'] = $this->modelObj->countRows;
223
+ $this->data['show_batch_select'] = ($this->modelObj->limit >= $this->modelObj->countRows) ? false : true;
224
+ $this->data['selected_lists'] = $this->get_selected_lists();
225
+ $this->modelObj->reset();
226
+
227
+
228
+ // make the data object for the listing view
229
+ $model_list = WYSIJA::get('list','model');
230
+ $lists_db = $model_list->getLists();
231
+
232
+ $lists=array();
233
+
234
+ foreach($lists_db as $listobj){
235
+ $lists[$listobj['list_id']]=$listobj;
236
+ }
237
+
238
+ $user_ids=array();
239
+ foreach($this->data['subscribers'] as $subscriber){
240
+ $user_ids[]=$subscriber['user_id'];
241
+ }
242
+
243
+ // 3 - user_list request
244
+ if($user_ids){
245
+ $modeluList=WYSIJA::get('user_list','model');
246
+ $userlists=$modeluList->get(array('list_id','user_id','unsub_date'),array('user_id'=>$user_ids));
247
+ }
248
+
249
+ $this->data['lists']=$lists;
250
+ $this->data['counts']=array_reverse($counts);
251
+
252
+ // regrouping all the data in the same array
253
+ foreach($this->data['subscribers'] as $keysus=>$subscriber){
254
+ // default key while we don't have the data
255
+ //TODO add data for stats about emails opened clicked etc
256
+ $this->data['subscribers'][$keysus]['emails']=0;
257
+ $this->data['subscribers'][$keysus]['opened']=0;
258
+ $this->data['subscribers'][$keysus]['clicked']=0;
259
+
260
+ if($userlists){
261
+ foreach($userlists as $key=>$userlist){
262
+ if($subscriber['user_id']==$userlist['user_id'] && isset($lists[$userlist['list_id']])){
263
+ //what kind of list ist it ? unsubscribed ? or not
264
+
265
+ if($userlist['unsub_date']>0){
266
+ if(!isset($this->data['subscribers'][$keysus]['unsub_lists']) ){
267
+ $this->data['subscribers'][$keysus]['unsub_lists']=$this->data['lists'][$userlist['list_id']]['name'];
268
+ }else{
269
+ $this->data['subscribers'][$keysus]['unsub_lists'].=', '.$this->data['lists'][$userlist['list_id']]['name'];
270
+ }
271
+ }else{
272
+ if(!isset($this->data['subscribers'][$keysus]['lists']) ){
273
+ $this->data['subscribers'][$keysus]['lists']=$this->data['lists'][$userlist['list_id']]['name'];
274
+ }else{
275
+ $this->data['subscribers'][$keysus]['lists'].=', '.$this->data['lists'][$userlist['list_id']]['name'];
276
+ }
277
+ }
278
+ }
279
+ }
280
+ }
281
+ }
282
+ if(!$this->data['subscribers']){
283
+ $this->notice(__('Yikes! Couldn\'t find any subscribers.',WYSIJA));
284
+ }
285
+
286
+ }
287
+
288
+ function main(){
289
+ $this->messages['insert'][true]=__('Subscriber has been saved.',WYSIJA);
290
+ $this->messages['insert'][false]=__('Subscriber has not been saved.',WYSIJA);
291
+ $this->messages['update'][true]=__('Subscriber has been modified. [link]Edit again[/link].',WYSIJA);
292
+ $this->messages['update'][false]=__('Subscriber has not been modified.',WYSIJA);
293
+ $this->cleanup_form();
294
+ parent::WYSIJA_control_back();
295
+
296
+ //we change the default model of the controller based on the action
297
+ if(isset($_REQUEST['action'])){
298
+ switch($_REQUEST['action']){
299
+ case 'listsedit':
300
+ case 'savelist':
301
+ case 'lists':
302
+ $this->model='list';
303
+ break;
304
+ default:
305
+ $this->model='user';
306
+ }
307
+ }
308
+
309
+ $this->WYSIJA_control();
310
+ if(!isset($_REQUEST['action']) || !$_REQUEST['action']) {
311
+ $this->defaultDisplay();
312
+ $this->checkTotalSubscribers();
313
+ } else {
314
+ $this->_tryAction($_REQUEST['action']);
315
+ }
316
+ if ( isset($_REQUEST['method']) && !empty($_REQUEST['method'])){
317
+ $this->_tryAction($_REQUEST['method']);
318
+ }
319
+ }
320
+
321
+ /**
322
+ * We are using the same form for different purposes:
323
+ * - Bulk actions
324
+ * - Filter by list
325
+ * We need to remove all un-necessary values from interface
326
+ */
327
+ protected function cleanup_form() {
328
+ if (!empty($_REQUEST['doaction'])) {
329
+ $action_type = strtolower(trim($_REQUEST['doaction']));
330
+ switch ($action_type)
331
+ {
332
+ // Filter by list
333
+ case 'filter':
334
+ if (!empty($_REQUEST['wysija']['user']))
335
+ unset($_REQUEST['wysija']['user']);
336
+ if (!empty($_REQUEST['action']))
337
+ unset($_REQUEST['action']);
338
+ break;
339
+ // Bulk action. Nothing to do, because we will invoke _tryAction() directly right after this step
340
+ case 'apply':
341
+ default:
342
+ break;
343
+ }
344
+ }
345
+ }
346
+
347
+ /**
348
+ * bulk action copy to list
349
+ * @global type $wpdb
350
+ * @param type $data
351
+ */
352
+ function copytolist($data){
353
+ $helpU=WYSIJA::get('user','helper');
354
+ if(empty($this->_batch_select))
355
+ $helpU->addToList($data['listid'],$_POST['wysija']['user']['user_id']);
356
+ else
357
+ $helpU->addToList($data['listid'],$this->_batch_select, true);
358
+
359
+ $modelL=WYSIJA::get('list','model');
360
+ $result=$modelL->getOne(array('name'),array('list_id'=>$data['listid']));
361
+
362
+ if($this->_affected_rows > 1)
363
+ $this->notice(sprintf(__('%1$s subscribers have been added to "%2$s".',WYSIJA),$this->_affected_rows,$result['name']));
364
+ else
365
+ $this->notice(sprintf(__('%1$s subscriber have been added to "%2$s".',WYSIJA),$this->_affected_rows,$result['name']));
366
+ $this->redirect_after_bulk_action();
367
+ }
368
+
369
+ /**
370
+ * Moves subscriber to another list.
371
+ *
372
+ * @param array $data List id to move, $data = array('listid' => 1);
373
+ */
374
+ function movetolist($data) {
375
+ $helper_user = WYSIJA::get('user', 'helper');
376
+
377
+ if (!empty($this->_batch_select)) {
378
+ $helper_user->moveToList($data['listid'], $this->_batch_select, true);
379
+ } elseif (isset($_POST['wysija']['user']['user_id'])) {
380
+ $helper_user->moveToList($data['listid'], $_POST['wysija']['user']['user_id']);
381
+ }
382
+
383
+ $model_list = WYSIJA::get('list','model');
384
+ $result = $model_list->getOne(array('name'), array('list_id' => $data['listid']));
385
+
386
+ if ($this->_affected_rows > 1) {
387
+ $this->notice(sprintf(__('%1$s subscribers have been moved to "%2$s".',WYSIJA), $this->_affected_rows, $result['name']));
388
+ } else {
389
+ $this->notice(sprintf(__('%1$s subscriber have been moved to "%2$s".',WYSIJA), $this->_affected_rows, $result['name']));
390
+ }
391
+
392
+ $this->redirect_after_bulk_action();
393
+ }
394
+
395
+ /**
396
+ * After performing a bulk action, let's keep the current list filter
397
+ */
398
+ protected function redirect_after_bulk_action() {
399
+ $filter_list = !empty($_REQUEST['wysija']['filter']['filter_list']) ? $_REQUEST['wysija']['filter']['filter_list'] : 0;
400
+ if (empty($filter_list)) {// view all lists
401
+ $this->redirect('admin.php?page=wysija_subscribers');
402
+ } elseif (is_numeric($filter_list)) {
403
+ $this->redirect('admin.php?page=wysija_subscribers&filter-list='.$filter_list);
404
+ } else {// subscribers in no list
405
+ $this->redirect('admin.php?page=wysija_subscribers&filter-list=orphaned');
406
+ }
407
+ }
408
+
409
+ /**
410
+ * Bulk action remove subscribers from all existing lists
411
+ * @param type $data = array('list_id'=>?)
412
+ */
413
+ function removefromalllists($data){
414
+ $helpU=WYSIJA::get('user','helper');
415
+ if(!empty($this->_batch_select))
416
+ $helpU->removeFromLists(array(),$this->_batch_select, true);
417
+ else
418
+ $helpU->removeFromLists(array(),$_POST['wysija']['user']['user_id']);
419
+
420
+ if($this->_affected_rows > 1)
421
+ $this->notice(sprintf(__('%1$s subscribers have been removed from all existing lists.',WYSIJA),$this->_affected_rows));
422
+ else
423
+ $this->notice(sprintf(__('%1$s subscriber have been removed from all existing lists.',WYSIJA),$this->_affected_rows));
424
+ $this->redirect_after_bulk_action();
425
+ }
426
+
427
+ /**
428
+ * Bulk action remove subscribers from all existing lists
429
+ * @param type $data = array('list_id'=>?)
430
+ */
431
+ function removefromlist($data = array()){
432
+ $helpU=WYSIJA::get('user','helper');
433
+ if(!empty($this->_batch_select))
434
+ $helpU->removeFromLists(array($data['listid']),$this->_batch_select, true);
435
+ else
436
+ $helpU->removeFromLists(array($data['listid']),$_POST['wysija']['user']['user_id']);
437
+ $modelL=WYSIJA::get('list','model');
438
+ $result=$modelL->getOne(array('name'),array('list_id'=>$data['listid']));
439
+
440
+ if($this->_affected_rows > 1)
441
+ $this->notice(sprintf(__('%1$s subscribers have been removed from "%2$s".',WYSIJA),$this->_affected_rows, $result['name']));
442
+ else
443
+ $this->notice(sprintf(__('%1$s subscriber have been removed from "%2$s".',WYSIJA),$this->_affected_rows, $result['name']));
444
+
445
+ $this->redirect_after_bulk_action();
446
+ }
447
+
448
+ /**
449
+ * Bulk confirm users
450
+ */
451
+ function confirmusers(){
452
+ $helpU=WYSIJA::get('user','helper');
453
+ if(!empty($this->_batch_select))
454
+ $helpU->confirmUsers($this->_batch_select, true);
455
+ else
456
+ $helpU->confirmUsers($_POST['wysija']['user']['user_id']);
457
+
458
+ if($this->_affected_rows > 1)
459
+ $this->notice(sprintf(__('%1$s subscribers have been confirmed.',WYSIJA),$this->_affected_rows));
460
+ else
461
+ $this->notice(sprintf(__('%1$s subscriber have been confirmed.',WYSIJA),$this->_affected_rows));
462
+ $this->redirect_after_bulk_action();
463
+ }
464
+
465
+ /**
466
+ * bulk action copy to list
467
+ * @global type $wpdb
468
+ * @param type $data
469
+ */
470
+ /*function unsubscribemany(){
471
+ $helperUser=WYSIJA::get('user','helper');
472
+ foreach($_POST['wysija']['user']['user_id'] as $uid) $helperUser->unsubscribe($uid,true);
473
+ $count=count($_POST['wysija']['user']['user_id']);
474
+ $this->notice(sprintf(__('%1$d Subscribers have been unsubscribed.',WYSIJA),$count));
475
+ $this->redirect();
476
+ }*/
477
+
478
+ function lists(){
479
+ $this->js[]='wysija-admin-list';
480
+ $this->_commonlists();
481
+
482
+ $this->modelObj=WYSIJA::get('list','model');
483
+ $this->viewObj->title=__('Edit lists',WYSIJA);
484
+ $this->modelObj->countRows=$this->modelObj->count();
485
+
486
+ $this->viewObj->model=$this->modelObj;
487
+ $this->data['form']=$this->_getForm();
488
+ }
489
+
490
+ function editlist(){
491
+ $this->_commonlists();
492
+ $this->data['form']=$this->_getForm($_REQUEST['id']);
493
+
494
+ $this->viewObj->title=sprintf(__('Editing list %1$s',WYSIJA), '<b><i>'.$this->data['form']['name'].'</i></b>');
495
+ }
496
+
497
+ function addlist(){
498
+ $this->_commonlists();
499
+ $this->viewObj->title=__('How about a new list?',WYSIJA);
500
+ $this->data['form']=$this->_getForm();
501
+ }
502
+
503
+ function duplicatelist(){
504
+
505
+ /* get the list's email id
506
+ * 0 duplicate the list's welcome email
507
+ * 1 duplicate the list
508
+ * 2 duplicate the list's subscribers
509
+ */
510
+ $model=WYSIJA::get('list','model');
511
+ $data=$model->getOne(array('name','namekey','welcome_mail_id','unsub_mail_id'),array('list_id'=>(int)$_REQUEST['id']));
512
+
513
+ $query='INSERT INTO `[wysija]email` (`created_at`,`campaign_id`,`subject`,`body`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status`)
514
+ SELECT '.time().',`campaign_id`,`subject`,`body`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status` FROM [wysija]email
515
+ WHERE email_id='.(int)$data['welcome_mail_id'];
516
+ $emailWelcomeid=$model->query($query);
517
+
518
+
519
+ $query='INSERT INTO `[wysija]email` (`created_at`,`campaign_id`,`subject`,`body`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status`)
520
+ SELECT '.time().',`campaign_id`,`subject`,`body`,`from_email`,`from_name`,`replyto_email`,`replyto_name`,`attachments`,`status` FROM [wysija]email
521
+ WHERE email_id='.(int)$data['unsub_mail_id'];
522
+ $emailUnsubid=$model->query($query);
523
+
524
+
525
+ $query='INSERT INTO `[wysija]list` (`created_at`,`name`,`namekey`,`description`,`welcome_mail_id`,`unsub_mail_id`,`is_enabled`,`ordering`)
526
+ SELECT '.time().',"'.stripslashes(__('Copy of ',WYSIJA)).$data['name'].'" ,"copy_'.$data['namekey'].time().'" ,`description`,'.$emailWelcomeid.','.$emailUnsubid.' ,1,`ordering` FROM [wysija]list
527
+ WHERE list_id='.(int)$_REQUEST['id'];
528
+
529
+ $listid=$model->query($query);
530
+
531
+ $query='INSERT INTO `[wysija]user_list` (`list_id`,`user_id`,`sub_date`,`unsub_date`)
532
+ SELECT '.$listid.',`user_id`,`sub_date`,`unsub_date` FROM [wysija]user_list
533
+ WHERE list_id='.(int)$_REQUEST['id'];
534
+
535
+ $model->query($query);
536
+
537
+ $this->notice(sprintf(__('List "%1$s" has been duplicated.',WYSIJA),$data['name']));
538
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
539
+
540
+ }
541
+
542
+ function add($data=false){
543
+ $this->js[]='wysija-validator';
544
+ $this->viewObj->add=true;
545
+
546
+ $this->title=$this->viewObj->title=__('Add Subscriber',WYSIJA);
547
+
548
+ $this->data=array();
549
+ $this->data['user']=false;
550
+ if($data)$this->data['user']=$data;
551
+ $modelList=WYSIJA::get('list','model');
552
+ $modelList->limitON=false;
553
+ $this->data['list']=$modelList->get(false,array('greater'=>array('is_enabled'=>'0') ));
554
+
555
+ }
556
+
557
+ function back(){
558
+ $this->redirect();
559
+ }
560
+
561
+ function backtolist(){
562
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
563
+ }
564
+
565
+ function edit($id=false){
566
+
567
+ if (empty($_REQUEST['id']) && empty($id)){
568
+ $this->error('Cannot edit element primary key is missing : '. get_class($this));
569
+ return;
570
+ }
571
+
572
+ if(!$id) $id=$_REQUEST['id'];
573
+
574
+ // get detail info of current user
575
+ $this->data['user']=$this->modelObj->getDetails(array('user_id'=>$id));
576
+ if(!$this->data['user']){
577
+ $this->notice(__('No subscriber found, most probably because he was deleted.',WYSIJA));
578
+ return $this->redirect();
579
+ }
580
+
581
+ // get list info
582
+ $model_list=WYSIJA::get('list','model');
583
+ $model_list->limitON=false;
584
+ $model_list->orderBy('is_enabled','DESC');
585
+ $this->data['list']=$model_list->get(false,array('greater'=>array('is_enabled'=>'-1') ));
586
+ $this->viewObj->title=__('Edit',WYSIJA).' '.$this->data['user']['details']['email'];
587
+
588
+ // execute hooks
589
+ $hook_params = array(
590
+ 'user_id' => $id
591
+ );
592
+ $this->data['hooks']['hook_subscriber_left'] = apply_filters('hook_subscriber_left',WYSIJA_module::execute_hook('hook_subscriber_left', $hook_params), $hook_params);
593
+ $this->data['hooks']['hook_subscriber_right'] = apply_filters('hook_subscriber_right',WYSIJA_module::execute_hook('hook_subscriber_right', $hook_params), $hook_params);
594
+ $this->data['hooks']['hook_subscriber_bottom'] = apply_filters('hook_subscriber_bottom',WYSIJA_module::execute_hook('hook_subscriber_bottom', $hook_params), $hook_params);
595
+
596
+
597
+ // prepare js, for rendering
598
+ $this->js[]='wysija-validator';
599
+ }
600
+
601
+ function deletelist(){
602
+ $this->requireSecurity();
603
+
604
+ /* get the list's email id
605
+ * 0 delete the welcome email corresponding to that list
606
+ * 1 delete the list subscribers reference
607
+ * 2 delete the list campaigns references
608
+ * 4 delete the list
609
+ */
610
+ $model_list=WYSIJA::get('list','model');
611
+ $data=$model_list->getOne(array('name','namekey','welcome_mail_id'),array('list_id'=>(int)$_REQUEST['id']));
612
+
613
+ if($data && isset($data['namekey']) && ($data['namekey']!='users')){
614
+
615
+ //there is no welcome email per list that's old stuff
616
+ $model_user_list=WYSIJA::get('user_list','model');
617
+ $model_user_list->delete(array('list_id'=>$_REQUEST['id']));
618
+
619
+ $model_campaign_list=WYSIJA::get('campaign_list','model');
620
+ $model_campaign_list->delete(array('list_id'=>$_REQUEST['id']));
621
+
622
+ $model_list->reset();
623
+ $model_list->delete(array('list_id'=>$_REQUEST['id']));
624
+
625
+ $this->notice(sprintf(__('List "%1$s" has been deleted.',WYSIJA),$data['name']));
626
+ }else{
627
+ $this->error(__('The list does not exists or cannot be deleted.',WYSIJA),true);
628
+ }
629
+
630
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
631
+
632
+ }
633
+
634
+
635
+ function synchlist(){
636
+ $this->requireSecurity();
637
+
638
+ $helper_user=WYSIJA::get('user','helper');
639
+ $helper_user->synchList($_REQUEST['id']);
640
+
641
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
642
+ }
643
+
644
+ function synchlisttotal(){
645
+ $this->requireSecurity();
646
+
647
+ global $current_user;
648
+
649
+ if(is_multisite() && is_super_admin( $current_user->ID )){
650
+ $helper_user=WYSIJA::get('user','helper');
651
+ $helper_user->synchList($_REQUEST['id'],true);
652
+ }
653
+
654
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
655
+ }
656
+
657
+
658
+ function savelist(){
659
+ $this->_resetGlobMsg();
660
+ $update=false;
661
+
662
+ if($_REQUEST['wysija']['list']['list_id']) $update=true;
663
+ /* save the result */
664
+ /* 1-save the welcome email*/
665
+ /* 2-save the list*/
666
+ if(isset($_REQUEST['wysija']['list']['is_public'])){
667
+ if($_REQUEST['wysija']['list']['is_public']=='on')$_REQUEST['wysija']['list']['is_public']=1;
668
+ else $_REQUEST['wysija']['list']['is_public']=0;
669
+ }else{
670
+ //$_REQUEST['wysija']['list']['is_public']=0; It's wrong. If is_public is not passed by interface, leave it as it is.
671
+ }
672
+
673
+ if($update){
674
+ $this->modelObj->update($_REQUEST['wysija']['list']);
675
+ $this->notice(__('List has been updated.',WYSIJA));
676
+ }else{
677
+ $_REQUEST['wysija']['list']['created_at']=time();
678
+ $_REQUEST['wysija']['list']['is_enabled']=1;
679
+
680
+ $this->modelObj->insert($_REQUEST['wysija']['list']);
681
+ $this->notice(__('Your brand-new list awaits its first subscriber.',WYSIJA));
682
+ }
683
+
684
+
685
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
686
+ }
687
+
688
+
689
+
690
+ function importpluginsave($id=false){
691
+ $this->requireSecurity();
692
+ $this->_resetGlobMsg();
693
+ $model_config=WYSIJA::get('config','model');
694
+ $helper_import=WYSIJA::get('plugins_import','helper');
695
+ $plugins_importable=$model_config->getValue('pluginsImportableEgg');
696
+ $plugins_imported=array();
697
+ foreach($_REQUEST['wysija']['import'] as $table_name =>$result){
698
+ $connection_info=$helper_import->getPluginsInfo($table_name);
699
+
700
+ if($result){
701
+ $plugins_imported[]=$table_name;
702
+ if(!$connection_info) $connection_info=$plugins_importable[$table_name];
703
+ $helper_import->import($table_name,$connection_info);
704
+ sleep(2);
705
+ $this->notice(sprintf(__('Import from plugin %1$s has been completed.',WYSIJA),"<strong>'".$connection_info['name']."'</strong>"));
706
+ }else{
707
+ $this->notice(sprintf(__('Import from plugin %1$s has been cancelled.',WYSIJA),"<strong>'".$connection_info['name']."'</strong>"));
708
+ }
709
+
710
+ }
711
+
712
+ $model_config->save(array('pluginsImportedEgg'=>$plugins_imported));
713
+
714
+ $this->redirect('admin.php?page=wysija_subscribers&action=lists');
715
+ }
716
+
717
+ function importplugins($id=false){
718
+ $this->js[]='wysija-validator';
719
+
720
+ $this->viewObj->title=__('Import subscribers from plugins',WYSIJA);
721
+
722
+ $model_config=WYSIJA::get('config','model');
723
+
724
+ $this->data=array();
725
+ $this->data['plugins']=$model_config->getValue('pluginsImportableEgg');
726
+ $imported_plugins=$model_config->getValue('pluginsImportedEgg');
727
+
728
+ if($imported_plugins){
729
+ foreach($imported_plugins as $tablename){
730
+ unset( $this->data['plugins'][$tablename]);
731
+ }
732
+ }
733
+
734
+
735
+ if(!$this->data['plugins']){
736
+ $this->notice(__('There is no plugin to import from.',WYSIJA));
737
+ return $this->redirect();
738
+ }
739
+ $this->viewShow='importplugins';
740
+
741
+ }
742
+
743
+ function import($id=false){
744
+ $this->js[]='wysija-validator';
745
+ $this->viewObj->title=__('Import Subscribers',WYSIJA);
746
+ $this->viewShow='import';
747
+ }
748
+
749
+ function importmatch(){
750
+ $this->jsTrans['subscribers_import_match_confirmation_1'] = __('The selected value is already matched to another column.', WYSIJA);
751
+ $this->jsTrans['subscribers_import_match_confirmation_2'] = __('Can you confirm that this column is corresponding to that field?', WYSIJA);
752
+ $this->js[] = 'wysija-validator';
753
+ wp_enqueue_script('jquery-matchColumn', WYSIJA_URL.'js/jquery/jquery.matchColumn.js', array('jquery'), WYSIJA::get_version());
754
+ $helper_numbers = WYSIJA::get('numbers','helper');
755
+ $bytes = $helper_numbers->get_max_file_upload();
756
+
757
+ if(isset($_SERVER['CONTENT_LENGTH']) && $_SERVER['CONTENT_LENGTH']>$bytes['maxbytes']){
758
+ if(isset($_FILES['importfile']['name']) && $_FILES['importfile']['name']){
759
+ $file_name = $_FILES['importfile']['name'];
760
+ }else{
761
+ $file_name = __('which you have pasted',WYSIJA);
762
+ }
763
+
764
+ $this->error(sprintf(__('Upload error, file %1$s is too large! (MAX:%2$s)',WYSIJA) , $file_name , $bytes['maxmegas']),true);
765
+ $this->redirect('admin.php?page=wysija_subscribers&action=import');
766
+ return false;
767
+ }
768
+
769
+ $import = new WJ_Import();
770
+ $this->data = $import->scan_csv_file();
771
+
772
+ if($this->data === false) $this->redirect('admin.php?page=wysija_subscribers&action=import');
773
+
774
+ $this->viewObj->title=__('Import Subscribers',WYSIJA);
775
+ $this->viewShow='importmatch';
776
+
777
+ }
778
+
779
+ /**
780
+ *
781
+ * @param type $input
782
+ * @param type $rowstoread
783
+ * @param type $delimiter
784
+ * @param type $enclosure
785
+ * @param type $linedelimiter
786
+ * @return array
787
+ */
788
+ function _csvToArray($input,$rowstoread=0 , $delimiter=',',$enclosure='',$linedelimiter="\n"){
789
+ $header = null;
790
+ $data = array();
791
+
792
+ $csvData = explode($linedelimiter,$input);
793
+ $i=1;
794
+ foreach($csvData as $csvLine){
795
+ if($rowstoread!=0 && $i>$rowstoread) return $data;
796
+
797
+ /* str_getcsv only exists in php5 ...*/
798
+ if(!function_exists("str_getcsv")){
799
+ $data[]= $this->csv_explode($csvLine, $delimiter,$enclosure);
800
+ }else{
801
+ $data[] = str_getcsv($csvLine, $delimiter,$enclosure);
802
+ }
803
+
804
+ $i++;
805
+ }
806
+
807
+ return $data;
808
+ }
809
+
810
+ function csv_explode($str,$delim, $enclose, $preserve=false){
811
+ $resArr = array();
812
+ $n = 0;
813
+ if(empty($enclose)){
814
+ $resArr = explode($delim, $str);
815
+ }else{
816
+ $expEncArr = explode($enclose, $str);
817
+ foreach($expEncArr as $EncItem){
818
+ if($n++%2){
819
+ array_push($resArr, array_pop($resArr) . ($preserve?$enclose:'') . $EncItem.($preserve?$enclose:''));
820
+ }else{
821
+ $expDelArr = explode($delim, $EncItem);
822
+ array_push($resArr, array_pop($resArr) . array_shift($expDelArr));
823
+ $resArr = array_merge($resArr, $expDelArr);
824
+ }
825
+ }
826
+ }
827
+
828
+ return $resArr;
829
+ }
830
+
831
+
832
+ function import_save(){
833
+ @ini_set('max_execution_time',0);
834
+
835
+ $this->requireSecurity();
836
+ $this->_resetGlobMsg();
837
+
838
+ //we need to save a new list in that situation
839
+ if(!empty($_REQUEST['wysija']['list']['newlistname'])){
840
+ $model_list = WYSIJA::get('list','model');
841
+ $data_list = array();
842
+ $data_list['is_enabled'] = 1;
843
+ $data_list['name'] = $_REQUEST['wysija']['list']['newlistname'];
844
+ $_REQUEST['wysija']['user_list']['list'][] = $model_list->insert($data_list);
845
+ }
846
+
847
+ //if there is no list selected, we return to the same form prompting the user to take action
848
+ if(!isset($_REQUEST['wysija']['user_list']['list']) || !$_REQUEST['wysija']['user_list']['list']){
849
+ $this->error(__('You need to select at least one list.',WYSIJA),true);
850
+ return $this->importmatch();
851
+ }
852
+
853
+ $import = new WJ_Import();
854
+ $data_numbers = $import->import_subscribers();
855
+ $duplicate_emails_count = $import->get_duplicate_emails_count();
856
+
857
+ if($data_numbers === false){
858
+ return $this->redirect('admin.php?page=wysija_subscribers&action=import');
859
+ }
860
+
861
+ //get a list of list name
862
+ $model = WYSIJA::get('list','model');
863
+ $results = $model->get(array('name'),array('list_id'=>$_REQUEST['wysija']['user_list']['list']));
864
+
865
+ $list_names=array();
866
+ foreach($results as $k =>$v) $list_names[]=$v['name'];
867
+
868
+ $this->notice( sprintf(__('%1$s subscribers added to %2$s.', WYSIJA),
869
+ $data_numbers['list_user_ids'],
870
+ '"'.implode('", "',$list_names).'"'
871
+ ) );
872
+
873
+ if(count($duplicate_emails_count)>0){
874
+ $list_emails = '';
875
+ $i = 0;
876
+ foreach($duplicate_emails_count as $email_address => $occurences){
877
+ if( $i > 0 )$list_emails.=', ';
878
+ $list_emails.= $email_address.' ('.$occurences.')';
879
+ $i++;
880
+ }
881
+ //$emailsalreadyinserted=array_keys($emailsCount);
882
+ $this->notice(sprintf(__('%1$s emails appear more than once in your file : %2$s.',WYSIJA),count($duplicate_emails_count),$list_emails),0);
883
+ }
884
+
885
+ if(count($data_numbers['invalid'])>0){
886
+ $string = sprintf(__('%1$s emails are not valid : %2$s.',WYSIJA),count($data_numbers['invalid']), utf8_encode(implode(', ',$data_numbers['invalid'])));
887
+ $this->notice($string,0);
888
+ }
889
+
890
+ $this->redirect();
891
+ }
892
+
893
+
894
+ function export(){
895
+ $this->js[]='wysija-validator';
896
+
897
+ $this->viewObj->title=__('Export Subscribers',WYSIJA);
898
+ $this->data=array();
899
+ //$this->data['lists']=$this->_getLists();
900
+ $this->data['lists']=$modelList=WYSIJA::get('list','model');
901
+ $listsDB=$modelList->getLists();
902
+
903
+ $lists=array();
904
+
905
+ foreach($listsDB as $listobj){
906
+ $lists[$listobj['list_id']]=$listobj;
907
+ }
908
+ $this->data['lists']=$lists;
909
+
910
+ $this->viewShow='export';
911
+ }
912
+
913
+ function exportcampaign(){
914
+ if(isset($_REQUEST['file_name'])){
915
+ $content=file_get_contents(base64_decode($_REQUEST['file_name']));
916
+ $user_ids=explode(",",$content);
917
+ }
918
+ $_REQUEST['wysija']['user']['user_id']=$user_ids;
919
+
920
+ $this->exportlist();
921
+ }
922
+
923
+ function exportlist(){
924
+
925
+ if(!empty($_REQUEST['wysija']['user']['force_select_all'])){
926
+
927
+ $select = array( 'COUNT(DISTINCT([wysija]user.user_id)) as total_users');
928
+ if(!empty($_REQUEST['wysija']['filter']['filter_list'])){
929
+ $select[] = '[wysija]user_list.list_id';
930
+ }
931
+
932
+ // filters for unsubscribed
933
+ $filters = $this->modelObj->detect_filters();
934
+
935
+ $count = $this->modelObj->get_subscribers( $select, $filters );
936
+ $number = $count['total_users'];
937
+ } else {
938
+ $number = count($_REQUEST['wysija']['user']['user_id']);
939
+ }
940
+
941
+ $this->viewObj->title = sprintf(__('Exporting %1$s subscribers',WYSIJA),$number);
942
+ $this->data=array();
943
+
944
+ $this->data['subscribers'] = $_REQUEST['wysija']['user']['user_id'];
945
+ $this->data['user'] = $_REQUEST['wysija']['user'];//for batch-selecting
946
+
947
+ if(!empty($_REQUEST['search'])) $_REQUEST['wysija']['filter']['search'] = $_REQUEST['search'];
948
+
949
+ $this->data['filter'] = $_REQUEST['wysija']['filter'];//for batch-selecting
950
+ $this->viewShow = 'export';
951
+ }
952
+
953
+
954
+
955
+ function sendconfirmation(){
956
+ $helperUser=WYSIJA::get('user','helper');
957
+ $helperUser->sendConfirmationEmail($_POST['wysija']['user']['user_id']);
958
+ $this->redirect();
959
+ }
960
+
961
+ /**
962
+ * bulk delete option
963
+ */
964
+ function deleteusers(){
965
+ $helper_user=WYSIJA::get('user','helper');
966
+ if(!empty($this->_batch_select))
967
+ $helper_user->delete($this->_batch_select, false, true);
968
+ else
969
+ $helper_user->delete($_POST['wysija']['user']['user_id']);
970
+ if($this->_affected_rows > 1)
971
+ $this->notice(sprintf(__(' %1$s subscribers have been deleted.',WYSIJA),$this->_affected_rows));
972
+ else
973
+ $this->notice(sprintf(__(' %1$s subscriber have been deleted.',WYSIJA),$this->_affected_rows));
974
+
975
+ // make sure the total count of subscribers is updated
976
+ $helper_user->refreshUsers();
977
+ $this->redirect_after_bulk_action();
978
+ }
979
+
980
+ /**
981
+ * function generating an export file based on an array of user_ids
982
+ */
983
+ function export_get(){
984
+ @ini_set('max_execution_time',0);
985
+
986
+ $export = new WJ_Export();
987
+
988
+ if(!empty($this->_batch_select)) $export->batch_select = $this->_batch_select;
989
+
990
+ $file_path_result = $export->export_subscribers();
991
+
992
+ $url=get_bloginfo('wpurl').'/wp-admin/admin.php?page=wysija_subscribers&action=exportedFileGet&file='.base64_encode($file_path_result);
993
+ $this->notice(str_replace(
994
+ array('[link]','[/link]'),
995
+ array('<a href="'.$url.'" target="_blank" class="exported-file" >','</a>'),
996
+ sprintf(__('%1$s subscribers were exported. Get the exported file [link]here[/link].',WYSIJA),$export->get_user_ids_rows())));
997
+
998
+ if(isset($_REQUEST['camp_id'])){
999
+ $this->redirect('admin.php?page=wysija_campaigns&action=viewstats&id='.$_REQUEST['camp_id']);
1000
+ }else{
1001
+ $this->redirect();
1002
+ }
1003
+ }
1004
+
1005
+ function exportedFileGet(){
1006
+ if(isset($_REQUEST['file'])){
1007
+ $helper=WYSIJA::get('file','helper');
1008
+ $helper->send(base64_decode($_REQUEST['file']));
1009
+ }
1010
+ }
1011
+
1012
+
1013
+
1014
+ function bulk_action(){
1015
+ return true;
1016
+ }
1017
+
1018
+
1019
+
1020
+
1021
+ /*
1022
+ * common task to all the list actions
1023
+ */
1024
+ function _commonlists(){
1025
+ $this->js[]='wysija-validator';
1026
+
1027
+ $this->data=array();
1028
+ $this->data['list']=$this->_getLists(10);
1029
+
1030
+ }
1031
+
1032
+ function _getLists($limit=false){
1033
+
1034
+ $modelList=WYSIJA::get('list','model');
1035
+ $modelList->escapingOn=true;
1036
+ $modelList->_limitison=$limit;
1037
+ return $modelList->getLists();
1038
+ }
1039
+
1040
+ function _getForm($id=false){
1041
+ if($id){
1042
+ $model_list=WYSIJA::get('list','model');
1043
+
1044
+ return $model_list->get_one_list($id);
1045
+ }else{
1046
+ $array=array('name'=>'','list_id'=>'','description'=>'','is_public'=>true,'is_enabled'=>true);
1047
+ return $array;
1048
+ }
1049
+
1050
+ }
1051
+ }
controllers/front.php ADDED
@@ -0,0 +1,54 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+
5
+ class WYSIJA_control_front extends WYSIJA_control{
6
+
7
+ function WYSIJA_control_front($extension="wysija-newsletters"){
8
+ $this->extension=$extension;
9
+ parent::WYSIJA_control();
10
+ $_REQUEST = stripslashes_deep($_REQUEST);
11
+ $_POST = stripslashes_deep($_POST);
12
+ $this->action = (isset($_REQUEST['action'])) ? $_REQUEST['action'] : 'index';
13
+ }
14
+
15
+ function save(){
16
+ $this->requireSecurity();
17
+ /* see if it's an update or an insert */
18
+ /*get the pk and its value as a conditions where pk = pkval*/
19
+ $conditions=$this->getPKVal($this->modelObj);
20
+
21
+ if($conditions){
22
+ /* this an update */
23
+
24
+ $result=$this->modelObj->update($_REQUEST['wysija'][$this->model],$conditions);
25
+
26
+ if($result) $this->notice($this->messages['update'][true]);
27
+ else{
28
+ $this->error($this->messages['update'][false],true);
29
+ }
30
+
31
+ }else{
32
+ /* this is an insert */
33
+ unset($_REQUEST['wysija'][$this->modelObj->pk]);
34
+
35
+ $result=$this->modelObj->insert($_REQUEST['wysija'][$this->model]);
36
+
37
+ if($result) $this->notice($this->messages['insert'][true]);
38
+ else{
39
+ $this->error($this->messages['insert'][false],true);
40
+ }
41
+
42
+ }
43
+ return $result;
44
+ }
45
+
46
+ function redirect($location) {
47
+ // make sure we encode square brackets as wp_redirect will strip them off
48
+ $location = str_replace(array('[', ']'), array('%5B', '%5D'), $location);
49
+
50
+ // redirect to specified location
51
+ wp_redirect($location);
52
+ exit;
53
+ }
54
+ }
controllers/front/confirm.php ADDED
@@ -0,0 +1,308 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+
5
+ class WYSIJA_control_front_confirm extends WYSIJA_control_front{
6
+ var $model='user';
7
+ var $view='confirm';
8
+
9
+ function WYSIJA_control_front_confirm(){
10
+ parent::WYSIJA_control_front();
11
+ }
12
+
13
+ function _testKeyuser(){
14
+ $this->helperUser=WYSIJA::get('user','helper');
15
+
16
+ $this->userData=$this->helperUser->checkUserKey();
17
+ add_action('init',array($this,'testsession'));
18
+
19
+ if(!$this->userData){
20
+ $this->title=__('Page does not exist.',WYSIJA);
21
+ $this->subtitle=__('Please verify your link to this page.',WYSIJA);
22
+ return false;
23
+ }
24
+ return true;
25
+ }
26
+
27
+ /**
28
+ * confirm subscription page
29
+ * return boolean
30
+ */
31
+ function subscribe(){
32
+ $helper_user = WYSIJA::get('user','helper');
33
+ if(!isset($_REQUEST['demo'])){
34
+ $helper_user->confirm_user();
35
+
36
+ if(!empty($helper_user->title)) $this->title = $helper_user->title;
37
+ if(!empty($helper_user->subtitle)) $this->optional_subtitle = $helper_user->subtitle;
38
+ }else{
39
+ $model_config=WYSIJA::get('config','model');
40
+
41
+ // we need to call the translation otherwise it will not be loaded and translated
42
+ $model_config->add_translated_default();
43
+
44
+ $this->title = sprintf($model_config->getValue('subscribed_title'), 'demo');
45
+ $this->optional_subtitle=$model_config->getValue('subscribed_subtitle');
46
+ }
47
+
48
+ return true;
49
+ }
50
+
51
+ function unsubscribe(){
52
+ $model_config=WYSIJA::get('config','model');
53
+
54
+ // we need to call the translation otherwise it will not be loaded and translated
55
+ $model_config->add_translated_default();
56
+
57
+ $this->title = $model_config->getValue('unsubscribed_title');
58
+ if(!isset($model_config->values['unsubscribed_title'])) $this->title = __("You've unsubscribed!",WYSIJA);
59
+
60
+ $this->optional_subtitle = $model_config->getValue('unsubscribed_subtitle');
61
+ if(!isset($model_config->values['unsubscribed_subtitle'])) $this->optional_subtitle = __("Great, you'll never hear from us again!",WYSIJA);
62
+
63
+ $undo_paramsurl = array(
64
+ 'wysija-page'=>1,
65
+ 'controller'=>'confirm',
66
+ 'action'=>'undounsubscribe',
67
+ 'wysija-key'=>$_REQUEST['wysija-key']
68
+ );
69
+
70
+ if(!isset($_REQUEST['demo'])){
71
+ if($this->_testKeyuser()){
72
+ $statusint=(int)$this->userData['details']['status'];
73
+ if( ($model_config->getValue('confirm_dbleoptin') && $statusint>0) || (!$model_config->getValue('confirm_dbleoptin') && $statusint>=0)){
74
+ $listids=$this->helperUser->unsubscribe($this->userData['details']['user_id']);
75
+ $this->helperUser->uid=$this->userData['details']['user_id'];
76
+ if($model_config->getValue('emails_notified') && $model_config->getValue('emails_notified_when_unsub')) $this->helperUser->_notify($this->userData['details']['email'],false,$listids);
77
+ }else{
78
+ $this->title=__('You are already unsubscribed.',WYSIJA);
79
+ return false;
80
+ }
81
+ }
82
+ }else{
83
+ $undo_paramsurl['demo'] = 1;
84
+ }
85
+
86
+ $link_undo = WYSIJA::get_permalink($model_config->getValue('unsubscribe_page'),$undo_paramsurl);
87
+
88
+
89
+ $this->undo_unsubscribe = str_replace(
90
+ array('[link]','[/link]'),
91
+ array('<a href="'.$link_undo.'">','</a>'),
92
+ '<p><b>'.__('You made a mistake? [link]Undo unsubscribe[/link].',WYSIJA)).'</b></p>';
93
+ return true;
94
+ }
95
+
96
+ function undounsubscribe(){
97
+ $model_config=WYSIJA::get('config','model');
98
+
99
+ // we need to call the translation otherwise it will not be loaded and translated
100
+ $model_config->add_translated_default();
101
+
102
+ $this->title =__("You've been subscribed!",WYSIJA);
103
+ $user_object = false;
104
+ if(!isset($_REQUEST['demo'])){
105
+ if($this->_testKeyuser()){
106
+ $user_object = (object)$this->userData['details'];
107
+ $this->helperUser->undounsubscribe($this->userData['details']['user_id']);
108
+ }
109
+ }
110
+
111
+ //manage subcription link
112
+ if(($model_config->getValue('manage_subscriptions'))){
113
+ $helper_user = WYSIJA::get('config','helper');
114
+ $model_user = WYSIJA::get('user','model');
115
+ $editsubscriptiontxt = $model_config->getValue('manage_subscriptions_linkname');
116
+ if(empty($editsubscriptiontxt)) $editsubscriptiontxt =__('Edit your subscription',WYSIJA);
117
+ $this->subtitle = '<p>'.$model_user->getEditsubLink($user_object,false,'').'.</p>';
118
+ }
119
+ return true;
120
+ }
121
+
122
+ function subscriptions(){
123
+ $data=array();
124
+
125
+ //get the user_id out of the params passed
126
+ if($this->_testKeyuser()){
127
+
128
+ $data['user']=$this->userData;
129
+ //get the list of user
130
+ $model_list=WYSIJA::get('list','model');
131
+ $model_list->orderBy('ordering','ASC');
132
+ $data['list']=$model_list->get(array('list_id','name','description'),array('is_enabled'=>true,'is_public'=>true));
133
+
134
+ $this->title=sprintf(__('Edit your subscriber profile: %1$s',WYSIJA),$data['user']['details']['email']);
135
+
136
+ $this->subtitle=$this->viewObj->subscriptions($data);
137
+
138
+
139
+ return true;
140
+ }
141
+
142
+
143
+ }
144
+
145
+
146
+
147
+ function resend(){
148
+ $this->title='The link you clicked has expired';
149
+
150
+ $this->subtitle=$this->viewObj->resend();
151
+ }
152
+
153
+
154
+ function resendconfirm(){
155
+ //make sure the user has the right to access this action
156
+ if($this->requireSecurity()){
157
+ //resend email
158
+ $helper_mailer=WYSIJA::get('mailer','helper');
159
+ $helper_mailer->sendOne((int)$_REQUEST['email_id'],(int)$_REQUEST['user_id']);
160
+ $this->title='Please check your inbox!';
161
+
162
+ $this->subtitle='<h3>A new email with working links has been sent to you.<h3/>';
163
+ }
164
+ }
165
+
166
+
167
+ function save(){
168
+
169
+ //get the user_id out of the params passed */
170
+ if($this->_testKeyuser()){
171
+ //update the general details */
172
+ $userid=$_REQUEST['wysija']['user']['user_id'];
173
+ unset($_REQUEST['wysija']['user']['user_id']);
174
+ $model_config=WYSIJA::get('config','model');
175
+ // we need to call the translation otherwise it will not be loaded and translated
176
+ $model_config->add_translated_default();
177
+ $this->helperUser->uid=$userid;
178
+ //if the status changed we might need to send notifications */
179
+ if((int)$_REQUEST['wysija']['user']['status'] !=(int)$this->userData['details']['status']){
180
+ if($_REQUEST['wysija']['user']['status']>0){
181
+ if($model_config->getValue('emails_notified_when_sub')) $this->helperUser->_notify($this->userData['details']['email']);
182
+ }else{
183
+ if($model_config->getValue('emails_notified_when_unsub')) $this->helperUser->_notify($this->userData['details']['email'],false);
184
+ }
185
+ }
186
+
187
+ //check whether the email address has changed if so then we should make sure that the new address doesnt exists already
188
+ if(isset($_REQUEST['wysija']['user']['email'])){
189
+ $_REQUEST['wysija']['user']['email']=trim($_REQUEST['wysija']['user']['email']);
190
+ if($this->userData['details']['email']!=$_REQUEST['wysija']['user']['email']){
191
+ $this->modelObj->reset();
192
+ $result=$this->modelObj->getOne(false,array('email'=>$_REQUEST['wysija']['user']['email']));
193
+ if($result){
194
+ $this->error(sprintf(__('Email %1$s already exists.',WYSIJA),$_REQUEST['wysija']['user']['email']),1);
195
+ unset($_REQUEST['wysija']['user']['email']);
196
+ }
197
+ }
198
+ }
199
+
200
+ $this->modelObj->update($_REQUEST['wysija']['user'],array('user_id'=>$userid));
201
+ $id=$userid;
202
+
203
+ $hUser=WYSIJA::get('user','helper');
204
+ //update the list subscriptions */
205
+ //run the unsubscribe process if needed
206
+ if((int)$_REQUEST['wysija']['user']['status']==-1){
207
+ $hUser->unsubscribe($id);
208
+ }
209
+
210
+ //update subscriptions */
211
+ $modelUL=WYSIJA::get('user_list','model');
212
+ $modelUL->backSave=true;
213
+ /* list of core list */
214
+ $modelLIST=WYSIJA::get('list','model');
215
+
216
+ // Using "like" condition in order to force sql query to OR (instead of AND). It'll be incorrct if status contains other values than 0/1.
217
+ $results=$modelLIST->get(array('list_id'),array('like' => array('is_enabled'=>0, 'is_public' => 0)));
218
+ $static_listids=array();
219
+ foreach($results as $res){
220
+ $static_listids[]=$res['list_id'];
221
+ }
222
+
223
+ //0 - get current lists of the user
224
+ $userlists=$modelUL->get(array('list_id','unsub_date'),array('user_id'=>$id));
225
+
226
+ $oldlistids=$new_list_ids=array();
227
+ foreach($userlists as $listdata) $oldlistids[$listdata['list_id']]=$listdata['unsub_date'];
228
+
229
+ $config=WYSIJA::get('config','model');
230
+ $dbloptin=$config->getValue('confirm_dbleoptin');
231
+ //1 - insert new user_list
232
+ if(isset($_POST['wysija']['user_list']['list_id']) && $_POST['wysija']['user_list']['list_id']){
233
+ $modelUL->reset();
234
+ $modelUL->update(array('sub_date'=>time()),array('user_id'=>$id));
235
+ foreach($_POST['wysija']['user_list']['list_id'] as $list_id){
236
+ //if the list is not already recorded for the user then we will need to insert it
237
+ if(!isset($oldlistids[$list_id])){
238
+ $modelUL->reset();
239
+ $new_list_ids[]=$list_id;
240
+ $dataul=array('user_id'=>$id,'list_id'=>$list_id,'sub_date'=>time());
241
+ //if double optin is on then we want to send a confirmation email for newly added subscription
242
+ if($dbloptin){
243
+ unset($dataul['sub_date']);
244
+ $modelUL->nohook=true;
245
+ }
246
+ $modelUL->insert($dataul);
247
+ //if the list is recorded already then let's check the status, if it is an unsubed one then we update it
248
+ }else{
249
+ if($oldlistids[$list_id]>0){
250
+ $modelUL->reset();
251
+ $modelUL->update(array('unsub_date'=>0,'sub_date'=>time()),array('user_id'=>$id,'list_id'=>$list_id));
252
+ }
253
+ //$alreadysubscribelistids[]=$list_id;
254
+ }
255
+ }
256
+ }
257
+
258
+
259
+
260
+
261
+ //if a confirmation email needs to be sent then we send it
262
+
263
+ if($dbloptin && !empty($new_list_ids)){
264
+ $send_confirmation = true;
265
+ $send_confirmation = apply_filters('mpoet_confirm_new_list_subscriptions_page', $send_confirmation);
266
+
267
+ if($send_confirmation === true){
268
+ $hUser->sendConfirmationEmail($id,true,$new_list_ids);
269
+ }else{
270
+ // this case has been made so that when subscribers add themselves to a
271
+ // list through the edit your subscription form they don't receive a confirmation email they already confirmed.
272
+ // so they receive also the autorespo,nders correspondign to that list immediately.
273
+ $helper_user = WYSIJA::get('user','helper');
274
+ $_REQUEST['wysiconf'] = base64_encode(serialize($new_list_ids));
275
+ $helper_user->confirm_user();
276
+ }
277
+ }
278
+
279
+ // list ids
280
+ $list_ids = !empty($_POST['wysija']['user_list']['list_id']) ? $_POST['wysija']['user_list']['list_id'] : array();
281
+ if(is_array($list_ids) === false) $list_ids = array();
282
+
283
+ $notEqual = array_merge($static_listids, $list_ids);
284
+
285
+ //delete the lists from which you've removed yourself
286
+ $condiFirst = array('notequal'=>array('list_id'=> $notEqual), 'equal' => array('user_id' => $id, 'unsub_date' => 0));
287
+ $modelUL=WYSIJA::get('user_list','model');
288
+ $modelUL->update(array('unsub_date'=>time()),$condiFirst);
289
+ $modelUL->reset();
290
+
291
+ /*
292
+ Custom Fields.
293
+ */
294
+ if (isset($_POST['wysija']['field'])) {
295
+ WJ_FieldHandler::handle_all(
296
+ $_POST['wysija']['field'], $id
297
+ );
298
+ }
299
+
300
+ $this->notice(__('Newsletter profile has been updated.',WYSIJA));
301
+ $this->subscriptions();
302
+
303
+ //reset post otherwise wordpress will not recognise the post !!!
304
+ $_POST=array();
305
+ }
306
+ return true;
307
+ }
308
+ }
controllers/front/email.php ADDED
@@ -0,0 +1,95 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ class WYSIJA_control_front_email extends WYSIJA_control_front{
5
+ var $model='email';
6
+ var $view='email';
7
+
8
+ function WYSIJA_control_front_email(){
9
+ parent::WYSIJA_control_front();
10
+ }
11
+
12
+ function view(){
13
+
14
+ $data=array();
15
+
16
+ header('Content-type:text/html; charset=utf-8');
17
+
18
+ // Get email model as object.
19
+ $emailM = WYSIJA::get('email','model');
20
+ $emailM->getFormat = OBJECT;
21
+ // Get config model
22
+ $configM = WYSIJA::get('config','model');
23
+ $configM->add_translated_default();
24
+ // Helpers
25
+ $emailH = WYSIJA::get('email','helper');
26
+ $mailerH = WYSIJA::get('mailer','helper');
27
+
28
+ $email_id = (int)$_REQUEST['email_id'];
29
+ // Get current email object.
30
+ $current_email = $emailM->getOne($email_id);
31
+
32
+ if($current_email->type==2){
33
+
34
+ $emailM->reset();
35
+ $autonewsHelper = WYSIJA::get('autonews','helper');
36
+ $autonewsHelper->refresh_automatic_content(array($email_id));
37
+ $emailM->getFormat = OBJECT;
38
+ $current_email = $emailM->getOne($email_id);
39
+ }
40
+
41
+
42
+
43
+
44
+ // Get current user object if possible
45
+ $current_user = null;
46
+ if(isset($_REQUEST['user_id']) && (int)$_REQUEST['user_id'] > 0 ){
47
+ // Get User Model
48
+ $userM = WYSIJA::get('user','model');
49
+ $userM->getFormat = OBJECT;
50
+ $current_user = $userM->getOne((int)$_REQUEST['user_id']);
51
+ }
52
+ // Parse and replace user tags.
53
+ $mailerH->parseUserTags($current_email);
54
+ $mailerH->parseSubjectUserTags($current_email);
55
+ $mailerH->replaceusertags($current_email, $current_user);
56
+ // Start tracking if this preview is opened from a newsletter
57
+ if ($current_user !== null) {
58
+ $mailerH->tracker_replaceusertags($current_email, $current_user);
59
+ }
60
+
61
+ // Set Body
62
+ $email_render = $current_email->body;
63
+
64
+ // Parse old shortcodes that we are parsing in the queue.
65
+ $find = array('[unsubscribe_linklabel]');
66
+ $replace = array($configM->getValue('unsubscribe_linkname'));
67
+ if (isset($current_email->params['autonl']['articles']['first_subject'])){
68
+ $find[] = '[post_title]';
69
+ $replace[] = $current_email->params['autonl']['articles']['first_subject'];
70
+ }
71
+ if (isset($current_email->params['autonl']['articles']['total'])){
72
+ $find[] = '[total]';
73
+ $replace[] = $current_email->params['autonl']['articles']['total'];
74
+ }
75
+
76
+ // if (isset($current_email->params['autonl']['articles']['ids'])){
77
+ // $find[] = '[number]';
78
+ // $replace[] = count($current_email->params['autonl']['articles']['ids']);
79
+ // }
80
+
81
+ if (isset($current_email->params['autonl']['total_child'])){
82
+ $find[] = '[number]';
83
+ $replace[] = $current_email->params['autonl']['total_child'];
84
+ }
85
+
86
+ $email_render1 = str_replace($find, $replace, $email_render);
87
+ // Strip unsubscribe links.
88
+ $email_render2 = $emailH->stripPersonalLinks($email_render1);
89
+
90
+ echo apply_filters('wysija_preview',$email_render2);
91
+
92
+ exit;
93
+ }
94
+
95
+ }
controllers/front/index.html ADDED
File without changes
controllers/front/module.php ADDED
@@ -0,0 +1,38 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * http://wy351s.local/?wysija-page=1&controller=confirm&wysija-key=4d5c04e327932b99d394f0963fa71dc4&action=subscriptions&wysijap=subscriptions&demo=1
5
+ * http://wy351s.local/?wysija-page=1&controller=module&action=init&wysijap=subscriptions&module=a&hook=newsletter
6
+ */
7
+ defined('WYSIJA') or die('Restricted access');
8
+
9
+ class WYSIJA_control_front_module extends WYSIJA_control_front{
10
+ /**
11
+ *
12
+ * @var string title of the page
13
+ */
14
+ public $subtitle;
15
+
16
+ /**
17
+ *
18
+ * @var html content of the page
19
+ */
20
+ public $title;
21
+
22
+ /**
23
+ * Dispatcher of module
24
+ */
25
+ public function init() {
26
+ if (empty($_REQUEST['module']) || empty($_REQUEST['hook'])) {
27
+ wp_redirect( home_url() ); exit;
28
+ }
29
+ $module_name = $_REQUEST['module'];
30
+ $hook = $_REQUEST['hook'];
31
+ $extension = !empty($_REQUEST['extension'])
32
+ ?
33
+ in_array($_REQUEST['extension'], array(WYSIJA, WYSIJANLP)) ? $_REQUEST['extension'] : WYSIJA
34
+ : WYSIJA;
35
+ $hook_params = array_merge($_REQUEST, array('controller_object' => $this));
36
+ $this->subtitle = WYSIJA_module::get_instance_by_name($module_name, $extension)->{'hook_'.$hook}($hook_params);
37
+ }
38
+ }
controllers/front/stats.php ADDED
@@ -0,0 +1,56 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+
5
+ class WYSIJA_control_front_stats extends WYSIJA_control_front{
6
+ var $model='' ;
7
+ var $view='';
8
+
9
+ /**
10
+ * Possible characters to be url encoded
11
+ * @var array
12
+ */
13
+ protected $characters_to_encode = array(
14
+ '@'
15
+ );
16
+
17
+ function WYSIJA_control_front_stats(){
18
+ parent::WYSIJA_control_front();
19
+ }
20
+
21
+ /**
22
+ * count the click statistic and redirect to the right url
23
+ * @return boolean
24
+ */
25
+ function analyse(){
26
+ if(isset($_REQUEST['email_id']) && isset($_REQUEST['user_id'])){
27
+
28
+ $WJ_Stats = new WJ_Stats();
29
+ if(!empty($WJ_Stats->clicked_url)){
30
+ // clicked stats
31
+ $url = $this->encode_url($WJ_Stats->subscriber_clicked());
32
+ do_action('mpoet_click_stats', $WJ_Stats);
33
+ $this->redirect($url);
34
+ }else{
35
+ // opened stat
36
+ $WJ_Stats->subscriber_opened();
37
+ }
38
+ }
39
+
40
+ return true;
41
+ }
42
+
43
+ /**
44
+ * Encode some special characters in url
45
+ * @param string $url
46
+ * @return string
47
+ */
48
+ protected function encode_url($url) {
49
+ return str_replace(
50
+ $this->characters_to_encode,
51
+ array_map('urlencode', $this->characters_to_encode),
52
+ $url
53
+ );
54
+ }
55
+
56
+ }
controllers/front/subscribers.php ADDED
@@ -0,0 +1,135 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ class WYSIJA_control_front_subscribers extends WYSIJA_control_front{
5
+ var $model='user';
6
+ var $view='widget_nl';
7
+
8
+ function WYSIJA_control_front_subscribers(){
9
+ parent::WYSIJA_control_front();
10
+ if(isset($_REQUEST['message_success'])){
11
+ $this->messages['insert'][true]=$_REQUEST['message_success'];
12
+ }else{
13
+ $this->messages['insert'][true]=__('User has been inserted.',WYSIJA);
14
+ }
15
+
16
+ $this->messages['insert'][false]=__('User has not been inserted.',WYSIJA);
17
+ $this->messages['update'][true]=__('User has been updated.',WYSIJA);
18
+ $this->messages['update'][false]=__('User has not been updated.',WYSIJA);
19
+ }
20
+
21
+ function save(){
22
+ $config=WYSIJA::get('config','model');
23
+
24
+ if(!$config->getValue('allow_no_js')){
25
+ $this->notice(__('Subscription without JavaScript is disabled.',WYSIJA));
26
+ return false;
27
+ }
28
+
29
+ if(isset($_REQUEST['wysija']['user_list']['list_id'])){
30
+ $_REQUEST['wysija']['user_list']['list_ids']=$_REQUEST['wysija']['user_list']['list_id'];
31
+ unset($_REQUEST['wysija']['user_list']['list_id']);
32
+ }elseif(isset($_REQUEST['wysija']['user_list']['list_ids'])){
33
+ $_REQUEST['wysija']['user_list']['list_ids']=explode(',',$_REQUEST['wysija']['user_list']['list_ids']);
34
+ }
35
+
36
+ $data=$_REQUEST['wysija'];
37
+ unset($_REQUEST['wysija']);
38
+
39
+ foreach($_REQUEST as $key => $val){
40
+ if(!isset($data[$key])) $data[$key]=$val;
41
+ }
42
+
43
+ $helperUser=WYSIJA::get('user','helper');
44
+ if(!$helperUser->checkData($data))return false;
45
+
46
+ $helperUser->addSubscriber($data);
47
+
48
+ return true;
49
+ }
50
+
51
+ /**
52
+ * handles the form generation in iframe mode, basically wysija's iframes call that action to generate the html of the body
53
+ */
54
+ function wysija_outter() {
55
+
56
+ //params used to generate the html in the widget class
57
+ $widget_data=array();
58
+
59
+ if(isset($_REQUEST['wysija_form']) && (int)$_REQUEST['wysija_form'] > 0) {
60
+ // this a wysija form made with the form editor
61
+ // if it's a preview, we need to dynamically render the form
62
+ // get form data
63
+
64
+ $widget_data['form']=(int)$_REQUEST['wysija_form'];
65
+ $widget_data['form_type']='iframe';
66
+
67
+ } else {
68
+
69
+ //this is the old way, we need to keep it for backward compatibility
70
+ if(isset($_REQUEST['encodedForm'])){
71
+ $encoded_form=json_decode(base64_decode(urldecode($_REQUEST['encodedForm'])));
72
+ } else {
73
+ if(isset($_REQUEST['fullWysijaForm'])){
74
+ $encoded_form=json_decode(base64_decode(urldecode($_REQUEST['fullWysijaForm'])));
75
+ } else {
76
+ if(isset($_REQUEST['widgetnumber'])){
77
+
78
+ $widgets=get_option('widget_wysija');
79
+ if(isset($widgets[$_REQUEST['widgetnumber']])){
80
+ $encoded_form=$widgets[$_REQUEST['widgetnumber']];
81
+ }
82
+
83
+ }else{
84
+ $encoded_form=$_REQUEST['formArray'];
85
+ $encoded_form=stripslashes_deep($encoded_form);
86
+ }
87
+
88
+ }
89
+ }
90
+
91
+
92
+ //fill the widget data array based on the parameters found earlier
93
+ if($encoded_form){
94
+ foreach($encoded_form as $key =>$val) {
95
+ $widget_data[$key]=$val;
96
+
97
+ //if the value is an object we need to loop through and make an array of it
98
+ //I think we could simply cast the object as an array not sure if that works on objects within objects...
99
+ if(is_object($val)){
100
+ $object_to_array=array();
101
+ foreach($val as $key_in =>$val_in){
102
+ $object_to_array[$key_in]=$val_in;
103
+ if(is_object($val_in)){
104
+ $object_to_array_second_level=array();
105
+ foreach($val_in as $k_in => $v_in){
106
+ $object_to_array_second_level[$k_in]=$v_in;
107
+ }
108
+ $object_to_array[$key_in]=$object_to_array_second_level;
109
+ }
110
+ }
111
+ $widget_data[$key]=$object_to_array;
112
+ }
113
+ }
114
+ }else{
115
+ if(current_user_can('switch_themes')) echo '<b>'.str_replace(array('[link]','[/link]'),array('<a target="_blank" href="'. admin_url('widgets.php').'">','</a>'),__('It seems your widget has been deleted from the WordPress\' [link]widgets area[/link].',WYSIJA)).'</b>';
116
+ exit;
117
+ }
118
+
119
+ //create a unique identifier for the form (the old way)
120
+ if(isset($_REQUEST['widgetnumber'])) $form_identifier=$_REQUEST['widgetnumber'];
121
+ else $form_identifier=rand(5, 1500);
122
+ $widget_data['widget_id']='wysija-nl-iframe-'.$form_identifier;
123
+ }
124
+
125
+
126
+ require_once(WYSIJA_WIDGETS.'wysija_nl.php');
127
+ $widget_NL=new WYSIJA_NL_Widget(true);
128
+ $widget_NL->iFrame=true;
129
+ $subscription_form = $widget_NL->widget($widget_data,$widget_data);
130
+
131
+ echo $subscription_form;
132
+ exit;
133
+ }
134
+
135
+ }
controllers/index.html ADDED
File without changes
core/autoloader.php ADDED
@@ -0,0 +1,23 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /**
4
+ * Classes Autoloader.
5
+ * It loads automatically the right class on class instantation.
6
+ * Since we still can't use namespaces, we use 'WJ_' as a prefix for our classes.
7
+ * @param Class $class Class name
8
+ * @return
9
+ */
10
+ function wysija_classes_autoloader($class) {
11
+ // Check if the class name has our prefix.
12
+ if (strpos($class, 'WJ_') !== false) {
13
+ // Class file path.
14
+ $class_path = WYSIJA_CLASSES . $class . '.php';
15
+ // If the class file exists, let's load it.
16
+ if (file_exists($class_path)) {
17
+ require_once $class_path;
18
+ }
19
+ }
20
+ }
21
+
22
+ // This is the global PHP autoload register, where we register our autoloaders.
23
+ spl_autoload_register('wysija_classes_autoloader');
core/base.php ADDED
@@ -0,0 +1,1578 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ // Require constants.
3
+ require_once( dirname( __FILE__ ) . DIRECTORY_SEPARATOR . 'constants.php' );
4
+
5
+ // Require global classes autoloader
6
+ require_once( dirname( __FILE__ ) . DIRECTORY_SEPARATOR . 'autoloader.php' );
7
+
8
+ defined( 'WYSIJA' ) or die( 'Restricted access' );
9
+
10
+ global $wysija_msg, $wysija_wpmsg;
11
+ if ( ! $wysija_msg ) {
12
+ $wysija_msg = array();
13
+ }
14
+ $wysija_wpmsg = array();
15
+
16
+ class WYSIJA_object{
17
+
18
+ /**
19
+ * Static variable holding core MailPoet's version
20
+ * @var array
21
+ */
22
+ static $version = '2.6.8';
23
+
24
+ function WYSIJA_object(){
25
+
26
+ }
27
+
28
+ /**
29
+ * Order an array by param name string compare
30
+ *
31
+ * @param array $a Array with the param to compare
32
+ * @param array $b Array with the param to compare
33
+ * @return int Sorting result from strcmp
34
+ */
35
+ public static function sort_by_name( $a, $b ){
36
+ return strcmp( strtolower( $a['name'] ), strtolower( $b['name'] ) );
37
+ }
38
+
39
+ /**
40
+ * Returns the version of based on a path
41
+ *
42
+ * @filter mailpoet/get_version
43
+ * @filter mailpoet/package
44
+ *
45
+ * @param string $path
46
+ * @return string Version of the package
47
+ */
48
+ public static function get_version( $path = null ) {
49
+ return apply_filters( 'mailpoet/get_version', self::$version, apply_filters( 'mailpoet/package', 'core', $path ) );
50
+ }
51
+
52
+ /**
53
+ * get the current_user data in a safe manner making sure a field exists before returning it's value
54
+ * @global type $current_user
55
+ * @param string $field
56
+ * @return mixed
57
+ */
58
+ public static function wp_get_userdata( $field = false ) {
59
+ //WordPress globals be careful there
60
+ global $current_user;
61
+ if ( $field ) {
62
+ if ( function_exists( 'get_currentuserinfo' ) ) {
63
+ // Here is an exception because of one of the weirdest bug
64
+ // the idea is to make sure we don't call get_currentuserinfo on the wysija_subscribers page when on a multisite
65
+ if ( ! ( isset( $_GET['page'] ) && $_GET['page'] === 'wysija_subscribers' && is_multisite() ) ){
66
+ get_currentuserinfo();
67
+ }
68
+ }
69
+ if ( isset( $current_user->{$field} ) ){
70
+ return $current_user->{$field};
71
+ } elseif ( isset( $current_user->data->{$field} ) ){
72
+ return $current_user->data->{$field};
73
+ } else {
74
+ return $current_user;
75
+ }
76
+ }
77
+ return $current_user;
78
+ }
79
+
80
+ /**
81
+ * set a global notice message
82
+ * @global array $wysija_wpmsg
83
+ * @param type $msg
84
+ */
85
+ function wp_notice( $msg ){
86
+ global $wysija_wpmsg;
87
+
88
+ //add the hook only once
89
+ if ( ! $wysija_wpmsg ) {
90
+ add_action( 'admin_notices', array( $this, 'wp_msgs' ) );
91
+ }
92
+
93
+ //record msgs
94
+ $wysija_wpmsg['updated'][] = $msg;
95
+ }
96
+
97
+ /**
98
+ * set a global error message
99
+ * @global array $wysija_wpmsg
100
+ * @param type $msg
101
+ */
102
+ function wp_error( $msg ){
103
+ global $wysija_wpmsg;
104
+
105
+ //add the hook only once
106
+ if ( ! $wysija_wpmsg ){
107
+ add_action( 'admin_notices', array( $this, 'wp_msgs' ) );
108
+ }
109
+
110
+ //record msgs
111
+ $wysija_wpmsg['error'][] = $msg;
112
+ }
113
+
114
+ /**
115
+ * prints a global message in the WordPress' backend identified as belonging to wysija
116
+ * we tend to avoid as much as possible printing messages globally, since this is ugly
117
+ * and make the administrators immune to beige-yellowish messages :/
118
+ * @global array $wysija_wpmsg
119
+ */
120
+ function wp_msgs() {
121
+ global $wysija_wpmsg;
122
+ foreach ( $wysija_wpmsg as $keymsg => $wp2 ){
123
+ $msgs = '<div class="' . $keymsg . ' fade">';
124
+ foreach ( $wp2 as $mymsg ){
125
+ $msgs .= '<p><strong>MailPoet</strong> : ' . $mymsg . '</p>';
126
+ }
127
+ $msgs .= '</div>';
128
+ }
129
+
130
+ // This is bad, we should be checking for valid HTML here.
131
+ echo $msgs;
132
+ }
133
+
134
+ /**
135
+ * returns an error message, it will appear as a red pinkish message in our interfaces
136
+ * @param string $msg
137
+ * @param boolean $public if set to true it will appear as a full message, otherwise it will appear behind a "Show more details." link
138
+ * @param boolean $global if set to true it will appear on all of the backend interfaces, not only wysija's own
139
+ */
140
+ function error( $msg, $public = false, $global = false ){
141
+ $status = 'error';
142
+ if ( $global ){
143
+ $status = 'g-' . $status;
144
+ }
145
+ $this->setInfo( $status, $msg, $public );
146
+ }
147
+
148
+ /**
149
+ * returns a success message, it will appear as a beige yellowish message in our interfaces
150
+ * @param string $msg
151
+ * @param boolean $public if set to true it will appear as a full message, otherwise it will appear behind a "Show more details." link
152
+ * @param boolean $global if set to true it will appear on all of the backend interfaces, not only wysija's own
153
+ */
154
+ function notice( $msg, $public = true, $global = false ){
155
+ $status = 'updated';
156
+ if ( $global ){
157
+ $status = 'g-' . $status;
158
+ }
159
+ $this->setInfo( $status, $msg, $public );
160
+ }
161
+
162
+ /**
163
+ * store all of the error and success messages in a global variable
164
+ * @global type $wysija_msg
165
+ * @param type $status whether this is a success message or an error message
166
+ * @param type $msg
167
+ * @param type $public if set to true it will appear as a full message, otherwise it will appear behind a "Show more details." link
168
+ */
169
+ static function setInfo($status,$msg,$public=false){
170
+ global $wysija_msg;
171
+ if(!$public) {
172
+
173
+ if(!isset($wysija_msg['private'][$status])){
174
+ $wysija_msg['private']=array();
175
+ $wysija_msg['private'][$status]=array();
176
+ }
177
+ array_push($wysija_msg['private'][$status], $msg);
178
+ }else{
179
+ if(!isset($wysija_msg[$status])) $wysija_msg[$status]=array();
180
+ array_push($wysija_msg[$status], $msg);
181
+ }
182
+
183
+ }
184
+
185
+ /**
186
+ * read the global function containing all of the error messages and print them
187
+ * @global type $wysija_msg
188
+ * @return type
189
+ */
190
+ function getMsgs(){
191
+ global $wysija_msg;
192
+
193
+ if(isset($wysija_msg['private']['error'])){
194
+ $wysija_msg['error'][]=str_replace(array('[link]','[/link]'),array('<a class="showerrors" href="javascript:;">','</a>'),__('An error occured. [link]Show more details.[/link]',WYSIJA));
195
+ }
196
+
197
+ if(isset($wysija_msg['private']['updated'])){
198
+ $wysija_msg['updated'][]=str_replace(array('[link]','[/link]'),array('<a class="shownotices" href="javascript:;">','</a>'),__('[link]Show more details.[/link]',WYSIJA));
199
+ }
200
+ if(isset($wysija_msg['private'])){
201
+ $prv=$wysija_msg['private'];
202
+ unset($wysija_msg['private']);
203
+ if(isset($prv['error'])) $wysija_msg['xdetailed-errors']=$prv['error'];
204
+ if(isset($prv['updated'])) $wysija_msg['xdetailed-updated']=$prv['updated'];
205
+ }
206
+ return $wysija_msg;
207
+ }
208
+ }
209
+
210
+
211
+ class WYSIJA_help extends WYSIJA_object{
212
+ var $controller = null;
213
+
214
+ static $admin_body_class_runner = false;
215
+
216
+ function WYSIJA_help(){
217
+ add_action( 'widgets_init', array( $this, 'widgets_init' ), 1 );
218
+
219
+ // Only load this when ajax is not used
220
+ if ( !( defined( 'DOING_AJAX' ) && DOING_AJAX ) ) {
221
+ add_action( 'init', array( $this, 'register_scripts' ), 1 );
222
+ }
223
+
224
+ add_action( 'admin_enqueue_scripts', array( $this, 'admin_enqueue_scripts' ) );
225
+ add_filter( 'admin_body_class', array( $this, 'admin_body_class' ) );
226
+ }
227
+
228
+ function widgets_init() {
229
+ //load the widget file
230
+ require_once(WYSIJA_WIDGETS.'wysija_nl.php');
231
+ register_widget( 'WYSIJA_NL_Widget' );
232
+ }
233
+
234
+ public function admin_enqueue_scripts(){
235
+ if ( WYSIJA_ITF ){
236
+ wp_enqueue_script( 'mailpoet-global' );
237
+ }
238
+ wp_enqueue_style( 'mailpoet-dashicons' );
239
+ }
240
+
241
+ public function admin_body_class( $body_class_str ){
242
+
243
+ if ( WYSIJA_help::$admin_body_class_runner === true ){
244
+ return $body_class_str;
245
+ }
246
+
247
+ WYSIJA_help::$admin_body_class_runner = true;
248
+
249
+ global $wp_version;
250
+
251
+ $class = array();
252
+ if ( ! empty( $body_class_str ) ){
253
+ $class = explode( ' ', $body_class_str );
254
+ }
255
+
256
+ if ( '3.8' === $wp_version ){
257
+ $class[] = 'mp-menu-icon-font';
258
+ }
259
+
260
+ if ( version_compare( $wp_version, '3.8', '<' ) ){
261
+ $class[] = 'mp-menu-icon-bg';
262
+ } else {
263
+ $class[] = 'mpoet-ui';
264
+ }
265
+
266
+ return implode( ' ', $class );
267
+ }
268
+
269
+ function register_scripts(){
270
+ $helper_toolbox = WYSIJA::get('toolbox','helper');
271
+ $wp_language_code = $helper_toolbox->get_language_code();
272
+ $valid_language = array(
273
+ 'ar',
274
+ 'ca',
275
+ 'cs',
276
+ 'cz',
277
+ 'da',
278
+ 'de',
279
+ 'el',
280
+ 'es',
281
+ 'et',
282
+ 'fa',
283
+ 'fi',
284
+ 'fr',
285
+ 'he',
286
+ 'hr',
287
+ 'hu',
288
+ 'id',
289
+ 'it',
290
+ 'ja',
291
+ 'lt',
292
+ 'nl',
293
+ 'no',
294
+ 'pl',
295
+ 'pt',
296
+ 'pt_BR',
297
+ 'ro',
298
+ 'ru',
299
+ 'sv',
300
+ 'tr',
301
+ 'vi',
302
+ 'zh_CN',
303
+ 'zh_TW',
304
+ );
305
+
306
+
307
+ if ( in_array( $wp_language_code, $valid_language ) ) {
308
+ wp_register_script('wysija-validator-lang',WYSIJA_URL.'js/validate/languages/jquery.validationEngine-'.$wp_language_code.'.js', array( 'jquery' ),WYSIJA::get_version(),true );
309
+ }else{
310
+ wp_register_script('wysija-validator-lang',WYSIJA_URL.'js/validate/languages/jquery.validationEngine-en.js', array( 'jquery' ),WYSIJA::get_version(),true );
311
+ }
312
+ wp_register_script('wysija-validator',WYSIJA_URL.'js/validate/jquery.validationEngine.js', array( 'jquery' ),WYSIJA::get_version(),true );
313
+ wp_register_script('wysija-front-subscribers', WYSIJA_URL.'js/front-subscribers.js', array( 'jquery' ),WYSIJA::get_version(),true);
314
+
315
+ wp_register_style( 'mailpoet-dashicons', WYSIJA_URL . 'css/admin-dashicons.css', array(), WYSIJA::get_version() );
316
+
317
+ wp_register_script('wysija-form', WYSIJA_URL.'js/forms.js', array( 'jquery' ),WYSIJA::get_version());
318
+ wp_register_style('validate-engine-css',WYSIJA_URL.'css/validationEngine.jquery.css',array(),WYSIJA::get_version());
319
+ wp_register_script('wysija-admin-ajax', WYSIJA_URL.'js/admin-ajax.js',array(),WYSIJA::get_version());
320
+ wp_register_script('wysija-admin-ajax-proto', WYSIJA_URL.'js/admin-ajax-proto.js',array(),WYSIJA::get_version());
321
+ wp_register_script( 'mailpoet-global', WYSIJA_URL.'js/admin-global.js', array( 'jquery', 'underscore' ), WYSIJA::get_version() );
322
+
323
+ if(defined('WYSIJA_SIDE') && WYSIJA_SIDE=='front') wp_enqueue_style('validate-engine-css');
324
+
325
+ }
326
+
327
+
328
+ /**
329
+ * All the ajax requests are routed through here
330
+ */
331
+ function ajax() {
332
+
333
+ $result_array = array();
334
+ if( !$_REQUEST || !isset( $_REQUEST['controller'] ) || !isset( $_REQUEST['task'] ) ){
335
+ $result_array = array( 'result' => false );
336
+ }else{
337
+ $plugin_requesting_ajax = 'wysija-newsletters';
338
+
339
+ // we override the plugin resquesting ajax if specified in the request
340
+ if( isset( $_REQUEST['wysijaplugin'] ) ){
341
+ $plugin_requesting_ajax = $_REQUEST['wysijaplugin'];
342
+ }
343
+
344
+ // fetching the right controller
345
+ $this->controller = WYSIJA::get( $_REQUEST['controller'] , 'controller' , false, $plugin_requesting_ajax );
346
+
347
+ // let's make sure the requested task exist
348
+ if( method_exists( $this->controller , $_REQUEST['task'] ) ){
349
+ $result_array['result'] = $this->controller->$_REQUEST['task']();
350
+ }else{
351
+ $this->error( 'Method "' . $_REQUEST['task'] . '" doesn\'t exist for controller : "'.$_REQUEST['controller'] );
352
+ }
353
+ }
354
+
355
+ // get the appended messages triggerred during the task execution
356
+ $result_array['msgs'] = $this->getMsgs();
357
+
358
+ // this header will allow ajax request from the home domain, this can be a lifesaver when domain mapping is on
359
+ if(function_exists('home_url')){
360
+ header('Access-Control-Allow-Origin: '.home_url());
361
+ }
362
+
363
+ // let's encode our response in the json format
364
+ $json_data = json_encode($result_array);
365
+
366
+ // in some case scenarios our client will have jQuery forcing the jsonp so we need to adapt ourselves
367
+ if(isset($_REQUEST['callback'])) {
368
+ // special header for json-p
369
+ header('Content-type: application/javascript');
370
+
371
+ $helper_jsonp = WYSIJA::get('jsonp', 'helper');
372
+ if($helper_jsonp->isValidCallback($_REQUEST['callback'])) {
373
+ print $_REQUEST['callback'] . '('.$json_data.');';
374
+ }
375
+ } else {
376
+ // standard header for unwrapped classic json response
377
+ header('Content-type: application/json');
378
+ print $json_data;
379
+ }
380
+ // our ajax response is printed, no need to let WordPress or 3rd party plugin execute more code
381
+ die();
382
+ }
383
+ }
384
+
385
+
386
+ class WYSIJA extends WYSIJA_object{
387
+
388
+ function WYSIJA(){
389
+
390
+ }
391
+
392
+ /**
393
+ * function created at the beginning to handle particular cases with WP get_permalink it got much smaller recently
394
+ * @param int $pageid
395
+ * @param array $params
396
+ * @param boolean $simple
397
+ * @return type
398
+ */
399
+ public static function get_permalink( $pageid, $params = array(), $simple = false ){
400
+ $hWPtools = WYSIJA::get( 'wp_tools', 'helper' );
401
+ return $hWPtools->get_permalink( $pageid, $params, $simple );
402
+ }
403
+
404
+ /**
405
+ * translate the plugin
406
+ * @staticvar boolean $extensionloaded
407
+ * @param type $extendedplugin
408
+ * @return boolean
409
+ */
410
+ public static function load_lang( $extendedplugin = false ){
411
+ static $extensionloaded = false;
412
+
413
+ //we return the entire array of extensions loaded if non is specified
414
+ if ( ! $extendedplugin ) {
415
+ return $extensionloaded;
416
+ }
417
+
418
+ //we only need to load this translation loader once on init
419
+ if(!$extensionloaded){
420
+ add_action('init', array('WYSIJA','load_lang_init'));
421
+ }
422
+ //each plugin has a different name
423
+ if ( !$extensionloaded || !isset($extensionloaded[$extendedplugin])) {
424
+ $transstring = null;
425
+ switch($extendedplugin){
426
+ case 'wysija-newsletters':
427
+ $transstring=WYSIJA;
428
+ break;
429
+ case 'wysijashop':
430
+ $transstring=WYSIJASHOP;
431
+ break;
432
+ case 'wysijacrons':
433
+ $transstring=WYSIJACRONS;
434
+ break;
435
+ case 'wysija-newsletters-premium':
436
+ $transstring=WYSIJANLP;
437
+ break;
438
+ case 'get_all':
439
+ return $extensionloaded;
440
+ }
441
+
442
+ //store all the required translations to be loaded in the static variable
443
+ if($transstring !== null) {
444
+ $extensionloaded[$extendedplugin] = $transstring;
445
+ }
446
+ }
447
+ }
448
+
449
+ /**
450
+ * check if the user is tech support as this can be used to switch the language back to english when helping our customers
451
+ * @global type $current_user
452
+ * @param type $debugmode
453
+ * @return type
454
+ */
455
+ public static function is_wysija_admin($debugmode=false){
456
+ //to allow wysija team members to work in english mode if debug is activated
457
+ global $current_user;
458
+
459
+ if((int)$debugmode > 0 && empty($current_user)) return true;
460
+
461
+ if(isset($current_user->data->user_email) && (strpos($current_user->data->user_email, '@mailpoet.com') !== false)) {
462
+ return true;
463
+ }
464
+ return false;
465
+ }
466
+
467
+ /**
468
+ * this function exists just to fix the issue with qtranslate :/ (it only fix it partially)
469
+ * @param type $extended_plugin
470
+ */
471
+ public static function load_lang_init($extended_plugin=false){
472
+ $model_config = WYSIJA::get('config','model');
473
+ $debug_mode = (int)$model_config->getValue('debug_new');
474
+
475
+ if($debug_mode === 0 || ($debug_mode > 0 && WYSIJA::is_wysija_admin($debug_mode) === false)) {
476
+ $extensions_loaded = WYSIJA::load_lang('get_all');
477
+ foreach($extensions_loaded as $extended_plugin => $translation_string){
478
+
479
+ // check for translation file overriding from transstring wp-content/languages/wysija-newsletters/wysija-newsletters-en_US.mo
480
+ $filename = WP_CONTENT_DIR.DS.'languages'.DS.$extended_plugin.DS.$translation_string.'-'.get_locale().'.mo';
481
+
482
+ if( !file_exists($filename) ){
483
+ // get the translation file in our local file
484
+ $filename = WYSIJA_PLG_DIR.$extended_plugin.DS.'languages'.DS.$translation_string.'-'.get_locale().'.mo';
485
+ }
486
+
487
+ // load the translation file with WP's load_textdomain
488
+ if( file_exists( $filename ) ){
489
+ load_textdomain( $translation_string, $filename );
490
+ }
491
+ }
492
+ }
493
+ }
494
+
495
+ /**
496
+ * function to generate objects of different types, managing file requiring in order to be the most efficient
497
+ * @staticvar array $arrayOfObjects
498
+ * @param string $name
499
+ * @param string $type : in which folder do we go and pick the class
500
+ * @param boolean $force_side : this parameter is almost never set to true,
501
+ * it will be useful for instance if you want to get a back controller
502
+ * from the frontend, it was used maybe in the shop but it can be ignored for wysija-newsletters
503
+ * @param type $extended_plugin : used only when calling the url from a different plugin it is used watch those files :
504
+ * -core/controller.php line 21, 23 ,24
505
+ * @param type $load_lang : the load lang is in the get to be sure we don't forget to load the language file for each plugin at least once
506
+ * the way I see it it could be moved to the index.php of each plugin. for now only wysija-newsletters is translated anyway
507
+ * @return boolean
508
+ */
509
+ public static function get($name,$type,$force_side=false,$extended_plugin='wysija-newsletters',$load_lang=true){
510
+ static $array_of_objects;
511
+
512
+ if($load_lang) WYSIJA::load_lang($extended_plugin);
513
+
514
+ // store all the objects made so that we can reuse them accross the application if the object is already set we return it immediately
515
+ if(isset($array_of_objects[$extended_plugin][$type.$name])) {
516
+ return $array_of_objects[$extended_plugin][$type.$name];
517
+ }
518
+
519
+ // which folder do we pick for controllersand views ? back or front ?
520
+ if($force_side) {
521
+ $side=$force_side;
522
+ } else {
523
+ $side=WYSIJA_SIDE;
524
+ }
525
+
526
+ // for each plugin we will define the $extended_constant variable if it's not defined already
527
+ // also we will defined the $extended_plugin_name which corresponds to the folder name and also will be used to build the class to be called
528
+ switch($extended_plugin){
529
+ case 'wysija-newsletters-premium':
530
+ $extended_constant='WYSIJANLP';
531
+ if(!defined($extended_constant)) define($extended_constant,$extended_constant);
532
+ $extended_plugin_name='wysijanlp';
533
+ break;
534
+ case 'wysija-newsletters':
535
+ $extended_constant='WYSIJA';
536
+ if(!defined($extended_constant)) define($extended_constant,$extended_constant);
537
+ $extended_plugin_name='wysija';
538
+ break;
539
+ default :
540
+ $extended_constant=strtoupper($extended_plugin);
541
+ if(!defined($extended_constant)) define($extended_constant,$extended_constant);
542
+ $extended_plugin_name=$extended_plugin;
543
+ }
544
+
545
+ // security to protect against dangerous ./../ includes
546
+ $name = preg_replace('#[^a-z0-9_]#i','',$name);
547
+
548
+ // this switch will require_once the file needed and build a the class name depending on the parameters passed to the function
549
+ switch($type){
550
+ case 'controller':
551
+ // require the parent class necessary
552
+ require_once(WYSIJA_CORE.'controller.php');
553
+
554
+ $ctrdir=WYSIJA_PLG_DIR.$extended_plugin.DS.'controllers'.DS;
555
+ // Require module concept
556
+ require_once(WYSIJA_CORE.'module'.DS.'module.php');
557
+
558
+ // if we are doing ajax we don't go to one side, ajax is for frontend or backend in the same folder
559
+ if(defined('DOING_AJAX')) {
560
+ $class_path=$ctrdir.'ajax'.DS.$name.'.php';
561
+ }else {
562
+ // the other controllers are called in a side folder back or front
563
+ $class_path=$ctrdir.$side.DS.$name.'.php';
564
+ // require the side specific controller file
565
+ require_once(WYSIJA_CTRL.$side.'.php');
566
+ }
567
+ $class_name = strtoupper($extended_plugin_name).'_control_'.$side.'_'.$name;
568
+ break;
569
+ case 'view':
570
+ $viewdir=WYSIJA_PLG_DIR.$extended_plugin.DS.'views'.DS;
571
+ // let's get the right path for the view front or back and the right class_name
572
+ $class_path=$viewdir.$side.DS.$name.'.php';
573
+ $class_name = strtoupper($extended_plugin_name).'_view_'.$side.'_'.$name;
574
+
575
+ // require the common view file and the side view file
576
+ require_once(WYSIJA_CORE.'view.php');
577
+ require_once(WYSIJA_VIEWS.$side.'.php');
578
+ break;
579
+ case 'helper':
580
+ $helpdir=WYSIJA_PLG_DIR.$extended_plugin.DS.'helpers'.DS;
581
+ $class_path=$helpdir.$name.'.php';
582
+ $class_name = strtoupper($extended_plugin_name).'_help_'.$name;
583
+
584
+ break;
585
+ case 'model':
586
+ $modeldir=WYSIJA_PLG_DIR.$extended_plugin.DS.'models'.DS;
587
+ $class_path=$modeldir.$name.'.php';
588
+ $class_name = strtoupper($extended_plugin_name).'_model_'.$name;
589
+ // require the parent class necessary
590
+ require_once(WYSIJA_CORE.'model.php');
591
+ break;
592
+ case 'widget':
593
+ $modeldir=WYSIJA_PLG_DIR.$extended_plugin.DS.'widgets'.DS;
594
+ $class_path=$modeldir.$name.'.php';
595
+ if($name=='wysija_nl') $class_name='WYSIJA_NL_Widget';
596
+ else $class_name = strtoupper($extended_plugin_name).'_widget_'.$name;
597
+ break;
598
+
599
+ case 'module':
600
+ $moduledir = WYSIJA_PLG_DIR . $extended_plugin . DS . 'modules' . DS;
601
+ if (file_exists($moduledir . $name . '.php'))
602
+ $class_path = $moduledir . $name . '.php';
603
+ elseif (file_exists($moduledir . $name . DS . $name . '.php'))
604
+ $class_path = $moduledir . $name . DS . $name . '.php';
605
+ else
606
+ return;
607
+ $class_name = strtoupper($extended_plugin_name) . '_module_' . $name;
608
+ // require the parent class necessary
609
+ //require_once(WYSIJA_CORE.'module'.DS.'module.php');
610
+ require_once(WYSIJA_CORE . 'module' . DS . 'statistics_model.php');
611
+ require_once(WYSIJA_CORE . 'module' . DS . 'statistics.php');
612
+ require_once(WYSIJA_CORE . 'module' . DS . 'statisticschart.php');
613
+ require_once(WYSIJA_CORE . 'module' . DS . 'statisticstable.php');
614
+
615
+ break;
616
+
617
+ default:
618
+ WYSIJA::setInfo('error','WYSIJA::get does not accept this type of file "'.$type.'" .');
619
+ return false;
620
+ }
621
+
622
+ if(!file_exists($class_path)) {
623
+ WYSIJA::setInfo('error','file has not been recognised '.$class_path);
624
+ WYSIJA::setInfo('error',$class_name);
625
+ WYSIJA::setInfo('error',$type);
626
+ return;
627
+ }
628
+
629
+ // require the file needed once and store & return the object needed
630
+ require_once($class_path);
631
+ return $array_of_objects[$extended_plugin][$type.$name]=new $class_name($extended_plugin_name);
632
+
633
+ }
634
+
635
+ /**
636
+ * log function to spot some strange issues when sending emails for instance
637
+ * @param type $key
638
+ * @param type $data
639
+ * @param type $category
640
+ * @return type
641
+ */
642
+ public static function log($key='default',$data='empty',$category='default'){
643
+ $config=WYSIJA::get('config','model');
644
+
645
+ if((int)$config->getValue('debug_new')>1 && $category && $config->getValue('debug_log_'.$category)){
646
+
647
+ $optionlog=get_option('wysija_log');
648
+
649
+
650
+ $optionlog[$category][(string)microtime(true)][$key]=$data;
651
+
652
+ WYSIJA::update_option('wysija_log' , $optionlog);
653
+ }
654
+ return false;
655
+ }
656
+
657
+ /**
658
+ * the filter to add option to the cron frequency instead of being stuck with hourly, daily and twicedaily...
659
+ * we can add filters but we cannot delete other values such as the default ones, as this might break other plugins crons
660
+ * @param type $param
661
+ * @return type
662
+ */
663
+ public static function filter_cron_schedules( $param ) {
664
+ $frequencies=array(
665
+ 'one_min' => array(
666
+ 'interval' => 60,
667
+ 'display' => __( 'Once every minute',WYSIJA)
668
+ ),
669
+ 'two_min' => array(
670
+ 'interval' => 120,
671
+ 'display' => __( 'Once every two minutes',WYSIJA)
672
+ ),
673
+ 'five_min' => array(
674
+ 'interval' => 300,
675
+ 'display' => __( 'Once every five minutes',WYSIJA)
676
+ ),
677
+ 'ten_min' => array(
678
+ 'interval' => 600,
679
+ 'display' => __( 'Once every ten minutes',WYSIJA)
680
+ ),
681
+ 'fifteen_min' => array(
682
+ 'interval' => 900,
683
+ 'display' => __( 'Once every fifteen minutes',WYSIJA)
684
+ ),
685
+ 'thirty_min' => array(
686
+ 'interval' => 1800,
687
+ 'display' => __( 'Once every thirty minutes',WYSIJA)
688
+ ),
689
+ 'two_hours' => array(
690
+ 'interval' => 7200,
691
+ 'display' => __( 'Once every two hours',WYSIJA)
692
+ ),
693
+ 'eachweek' => array(
694
+ 'interval' => 604800,
695
+ 'display' => __( 'Once a week',WYSIJA)
696
+ ),
697
+ 'each28days' => array(
698
+ 'interval' => 2419200,
699
+ 'display' => __( 'Once every 28 days',WYSIJA)
700
+ ),
701
+ );
702
+
703
+ return array_merge($param, $frequencies);
704
+ }
705
+
706
+ /**
707
+ * scheduled task for sending the emails in the queue, the frequency is set in the settings
708
+ */
709
+ public static function croned_queue( $check_scheduled_newsletter = true) {
710
+
711
+ // check the scheduled tasks only if it's a standard WP scheduled task free only
712
+ if($check_scheduled_newsletter){
713
+ WYSIJA::check_scheduled_newsletters();
714
+ }
715
+
716
+ $model_config = WYSIJA::get('config','model');
717
+ // check that the 2000 limit is not passed and process the queue
718
+
719
+ if((int)$model_config->getValue('total_subscribers') < 2000 ){
720
+ $helper_queue = WYSIJA::get('queue','helper');
721
+ $helper_queue->report=false;
722
+ WYSIJA::log('croned_queue process',true,'cron');
723
+
724
+ $helper_queue->process();
725
+ }
726
+
727
+ }
728
+
729
+ public static function check_scheduled_newsletters(){
730
+ $last_scheduled_check = get_option('wysija_last_scheduled_check');
731
+
732
+ // if the latest post notification check was done more than five minutes ago let's check it again
733
+ if(empty($last_scheduled_check) || ( time() > ($last_scheduled_check + 300) ) ){
734
+ // create the scheduled automatic post notifications email if there are any
735
+ $helper_autonews = WYSIJA::get('autonews','helper');
736
+ $helper_autonews->checkPostNotif();
737
+
738
+ // queue the scheduled newsletter also if there are any
739
+ $helper_autonews->checkScheduled();
740
+ WYSIJA::update_option('wysija_last_scheduled_check', time());
741
+ }
742
+
743
+ // send daily report about emails sent
744
+ $model_config = WYSIJA::get('config','model');
745
+ if($model_config->getValue('emails_notified_when_dailysummary')){
746
+ $helper_notification = WYSIJA::get('notifications','helper');
747
+ $helper_notification->send_daily_report();
748
+ }
749
+
750
+ }
751
+
752
+
753
+ /**
754
+ * everyday we make sure not to leave any trash files
755
+ * remove temporary files
756
+ */
757
+ public static function croned_daily() {
758
+
759
+ @ini_set('max_execution_time',0);
760
+
761
+ /*user refresh count total*/
762
+ $helper_user = WYSIJA::get('user','helper');
763
+ $helper_user->refreshUsers();
764
+
765
+ /*user domain generation*/
766
+ $helper_user->generate_domains();
767
+
768
+ /*clear temporary folders*/
769
+ $helper_file = WYSIJA::get('file','helper');
770
+ $helper_file->clear();
771
+
772
+ /*clear queue from unsubscribed*/
773
+ $helper_queue = WYSIJA::get('queue','helper');
774
+ $helper_queue->clear();
775
+
776
+
777
+ }
778
+
779
+ // Weekly cron
780
+ public static function croned_weekly() {
781
+
782
+ @ini_set('max_execution_time',0);
783
+
784
+ $model_config = WYSIJA::get('config','model');
785
+
786
+ // If enabled, flag MixPanel sending on next page load.
787
+ if ($model_config->getValue('analytics') == 1) {
788
+ $model_config->save(array('send_analytics_now' => 1));
789
+ }
790
+
791
+ }
792
+
793
+ // Monthly cron
794
+ public static function croned_monthly() {
795
+
796
+ @ini_set('max_execution_time',0);
797
+
798
+ $model_config = WYSIJA::get('config','model');
799
+
800
+ /* send daily report about emails sent */
801
+ if ($model_config->getValue('sharedata')) {
802
+ $helper_stats = WYSIJA::get('stats','helper');
803
+ $helper_stats->share();
804
+ }
805
+
806
+ }
807
+
808
+ /**
809
+ * when we deactivate the plugin we clear the WP install from those cron records
810
+ */
811
+ public static function deactivate() {
812
+ wp_clear_scheduled_hook('wysija_cron_queue');
813
+ wp_clear_scheduled_hook('wysija_cron_bounce');
814
+ wp_clear_scheduled_hook('wysija_cron_daily');
815
+ wp_clear_scheduled_hook('wysija_cron_weekly');
816
+ wp_clear_scheduled_hook('wysija_cron_monthly');
817
+ }
818
+
819
+
820
+ /**
821
+ * wysija's redirect allows to save some variables for the next page load such as notices etc..
822
+ * @global type $wysija_msg
823
+ * @global type $wysija_queries
824
+ * @global type $wysija_queries_errors
825
+ * @param type $location
826
+ */
827
+ public static function redirect($location) {
828
+ //save the messages
829
+ global $wysija_msg,$wysija_queries,$wysija_queries_errors;
830
+ WYSIJA::update_option('wysija_msg',$wysija_msg);
831
+ WYSIJA::update_option('wysija_queries',$wysija_queries);
832
+ WYSIJA::update_option('wysija_queries_errors',$wysija_queries_errors);
833
+
834
+ // make sure we encode square brackets as wp_redirect will strip them off
835
+ $location = str_replace(array('[', ']'), array('%5B', '%5D'), $location);
836
+
837
+ // redirect to specified location
838
+ wp_redirect($location);
839
+ exit;
840
+ }
841
+
842
+ /**
843
+ * custom post type for wysija is call wysijap as in wysija's post
844
+ */
845
+ public static function create_post_type() {
846
+
847
+ //by default there is url rewriteing on wysijap custom post, though in one client case I had to deactivate it.
848
+ //as this is rare we just need to set this setting to activate it
849
+ //by default let's deactivate the url rewriting of the wysijap confirmation page because it is breaking in some case.
850
+ $show_interface=false;
851
+ if(defined('WYSIJA_DBG') && WYSIJA_DBG>1) $show_interface=true;
852
+ register_post_type( 'wysijap',
853
+ array(
854
+ 'labels' => array(
855
+ 'name' => __('MailPoet page'),
856
+ 'singular_name' => __('MailPoet page')
857
+ ),
858
+ 'public' => true,
859
+ 'has_archive' => false,
860
+ 'show_ui' =>$show_interface,
861
+ 'show_in_menu' =>$show_interface,
862
+ 'rewrite' => false,
863
+ 'show_in_nav_menus'=>false,
864
+ 'can_export'=>false,
865
+ 'publicly_queryable'=>true,
866
+ 'exclude_from_search'=>true,
867
+ )
868
+ );
869
+
870
+ if(!get_option('wysija_post_type_updated')) {
871
+ $modelPosts=new WYSIJA_model();
872
+ $modelPosts->tableWP=true;
873
+ $modelPosts->table_prefix='';
874
+ $modelPosts->table_name='posts';
875
+ $modelPosts->noCheck=true;
876
+ $modelPosts->pk='ID';
877
+ if($modelPosts->exists(array('post_type'=>'wysijapage'))){
878
+ $modelPosts->update(array('post_type'=>'wysijap'),array('post_type'=>'wysijapage'));
879
+ flush_rewrite_rules( false );
880
+ }
881
+ WYSIJA::update_option('wysija_post_type_updated',time());
882
+ }
883
+
884
+ if(!get_option('wysija_post_type_created')) {
885
+ flush_rewrite_rules( false );
886
+ WYSIJA::update_option('wysija_post_type_created',time());
887
+ }
888
+
889
+ }
890
+
891
+ /**
892
+ * wysija update_option function is very similar to WordPress' one but it
893
+ * can also manage new options not automatically loaded each time
894
+ * @param type $option_name
895
+ * @param type $newvalue
896
+ * @param type $defaultload this parameter is the advantage other Wp's update_option here
897
+ */
898
+ public static function update_option($option_name,$newvalue,$defaultload='no'){
899
+ if ( get_option( $option_name ) != $newvalue ) {
900
+ update_option( $option_name, $newvalue );
901
+ } else {
902
+ add_option( $option_name, $newvalue, '', $defaultload );
903
+ }
904
+ }
905
+
906
+ /**
907
+ * When a WordPress user is added we also need to add it to the subscribers list
908
+ * @param type $user_id
909
+ * @return type
910
+ */
911
+ public static function hook_add_WP_subscriber($user_id) {
912
+ $data=get_userdata($user_id);
913
+
914
+
915
+ //check first if a subscribers exists if it doesn't then let's insert it
916
+ $model_config=WYSIJA::get('config','model');
917
+ $model_user=WYSIJA::get('user','model');
918
+ $model_user->getFormat=ARRAY_A; // there is one case where we were getting an object instead of an array
919
+ $subscriber_exists=$model_user->getOne(array('user_id'),array('email'=>$data->user_email));
920
+
921
+ $first_name=$data->first_name;
922
+ $last_name=$data->last_name;
923
+ if(!$data->first_name && !$data->last_name) $first_name=$data->display_name;
924
+
925
+ $model_user->reset();
926
+ if($subscriber_exists){
927
+ $user_id=$subscriber_exists['user_id'];
928
+ // we need to update the current subscriber using it's id
929
+ $model_user->update(array('wpuser_id'=>$data->ID,'firstname'=>$first_name,'lastname'=>$last_name),array('user_id'=>$user_id));
930
+ }else{
931
+ $model_user->noCheck=true;
932
+ $user_id=$model_user->insert(array('email'=>$data->user_email,'wpuser_id'=>$data->ID,'firstname'=>$first_name,'lastname'=>$last_name,'status'=>$model_config->getValue('confirm_dbleoptin')));
933
+ }
934
+
935
+ $model_user_list=WYSIJA::get('user_list','model');
936
+ $model_user_list->insert(array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id'),'sub_date'=>time()),true);
937
+
938
+ $helper_user=WYSIJA::get('user','helper');
939
+ $helper_user->sendAutoNl($user_id,$data,'new-user');
940
+ return true;
941
+ }
942
+
943
+ /**
944
+ * when a WordPress user is updated we also need to update the corresponding subscriber
945
+ * @param type $user_id
946
+ * @return type
947
+ */
948
+ public static function hook_edit_WP_subscriber($user_id) {
949
+ $data=get_userdata($user_id);
950
+
951
+ //check first if a subscribers exists if it doesn't then let's insert it
952
+ $model_user=WYSIJA::get('user','model');
953
+ $model_config=WYSIJA::get('config','model');
954
+ $model_user_list=WYSIJA::get('user_list','model');
955
+ $model_user->getFormat = ARRAY_A;
956
+ $subscriber_exists=$model_user->getOne(array('user_id'),array('email'=>$data->user_email));
957
+
958
+ $model_user->reset();
959
+
960
+ $first_name=$data->first_name;
961
+ $last_name=$data->last_name;
962
+ if(!$data->first_name && !$data->last_name) $first_name=$data->display_name;
963
+
964
+ if($subscriber_exists){
965
+ $user_id=$subscriber_exists['user_id'];
966
+
967
+ $model_user->update(array('wpuser_id'=>$data->ID, 'email'=>$data->user_email,'firstname'=>$first_name,'lastname'=>$last_name),array('user_id'=>$user_id));
968
+
969
+ $result=$model_user_list->getOne(false,array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id')));
970
+ $model_user_list->reset();
971
+ if(!$result)
972
+ $model_user_list->insert(array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id'),'sub_date'=>time()));
973
+ }else{
974
+ //chck that we didnt update the email
975
+ $subscriber_exists=$model_user->getOne(false,array('wpuser_id'=>$data->ID));
976
+
977
+ if($subscriber_exists){
978
+ $user_id=$subscriber_exists['user_id'];
979
+
980
+ $model_user->update(array('email'=>$data->user_email,'firstname'=>$first_name,'lastname'=>$last_name),array('wpuser_id'=>$data->ID));
981
+
982
+ $result=$model_user_list->getOne(false,array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id')));
983
+ $model_user_list->reset();
984
+ if(!$result)
985
+ $model_user_list->insert(array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id'),'sub_date'=>time()));
986
+ }else{
987
+ $model_user->noCheck=true;
988
+ $user_id=$model_user->insert(array('email'=>$data->user_email,'wpuser_id'=>$data->ID,'firstname'=>$first_name,'lastname'=>$last_name,'status'=>$model_config->getValue('confirm_dbleoptin')));
989
+ $model_user_list->insert(array('user_id'=>$user_id,'list_id'=>$model_config->getValue('importwp_list_id'),'sub_date'=>time()));
990
+ }
991
+ }
992
+ return true;
993
+ }
994
+
995
+ /**
996
+ * when a wp user is deleted we also delete the subscriber corresponding to it
997
+ * @param type $user_id
998
+ */
999
+ public static function hook_del_WP_subscriber($user_id) {
1000
+ $model_config=WYSIJA::get('config','model');
1001
+ $model_user=WYSIJA::get('user','model');
1002
+ $data = $model_user->getOne(array('email','user_id'),array('wpuser_id'=>$user_id));
1003
+ if(isset($data['email'])) {
1004
+ $model_user->delete(array('email'=>$data['email']));
1005
+ }
1006
+ if(isset($data['user_id'])) {
1007
+ $model_user = WYSIJA::get('user_list','model');
1008
+ $model_user->delete(array('user_id'=>$data['user_id'],'list_id'=>$model_config->getValue('importwp_list_id')));
1009
+ }
1010
+ }
1011
+
1012
+
1013
+ public static function hook_auto_newsletter_refresh($post_id) {
1014
+ $helper_autonews = WYSIJA::get('autonews', 'helper');
1015
+ $helper_autonews->refresh_automatic_content();
1016
+
1017
+ return true;
1018
+ }
1019
+
1020
+ /**
1021
+ * post notification transition hook, know when a post really gets published
1022
+ * @param type $new_status
1023
+ * @param type $old_status
1024
+ * @param type $post
1025
+ * @return type
1026
+ */
1027
+ public static function hook_postNotification_transition($new_status, $old_status, $post) {
1028
+ //we run some process only if the status of the post changes from something to publish
1029
+ if($new_status === 'publish' && $old_status !== $new_status) {
1030
+ $model_email = WYSIJA::get('email', 'model');
1031
+ $emails = $model_email->get(false, array('type' => 2, 'status' => array(1, 3, 99)));
1032
+
1033
+ if(!empty($emails)) {
1034
+ //we loop through all of the automatic emails
1035
+ foreach($emails as $key => $email) {
1036
+
1037
+ //we will try to give birth to a child email only if the automatic newsletter is a post notification email and in immediate mode
1038
+ if(is_array($email) && $email['params']['autonl']['event'] === 'new-articles' && $email['params']['autonl']['when-article'] === 'immediate') {
1039
+ // set default include/exclude categories
1040
+ $include_category_ids = array();
1041
+ $exclude_category_ids = array();
1042
+
1043
+ // ALC need to check for post_type on each block of the autoposts
1044
+ $wj_data = maybe_unserialize( base64_decode( $email['wj_data'] ) );
1045
+ $post_types = array();
1046
+ $has_alc_blocks = false;
1047
+
1048
+ foreach ( $wj_data['body'] as $block_key => $block ){
1049
+ if ( $block['type'] !== 'auto-post' ){
1050
+ continue;
1051
+ }
1052
+
1053
+ $has_alc_blocks = true;
1054
+
1055
+ // get post type and post categories from block parameters
1056
+ foreach( $block['params'] as $param_data ) {
1057
+ if(in_array($param_data['key'], array('post_type', 'cpt')) && strlen(trim($param_data['value'])) > 0) {
1058
+ // store post type
1059
+ $post_types[] = trim($param_data['value']);
1060
+ } else if($param_data['key'] === 'category_ids' && strlen(trim($param_data['value'])) > 0) {
1061
+ // store post category ids
1062
+ $include_category_ids = array_map('intval', explode(',', trim($param_data['value'])));
1063
+ }
1064
+ }
1065
+ }
1066
+
1067
+ if ( $has_alc_blocks === true && ! in_array( $post->post_type, $post_types ) ) {
1068
+ continue;
1069
+ }
1070
+
1071
+ // get post categories
1072
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
1073
+ $taxonomies = $helper_wp_tools->get_post_category_ids($post);
1074
+
1075
+ // assume the post has to be sent
1076
+ $do_send_post = true;
1077
+
1078
+ // post categories have to match at least one of the email's included categories
1079
+ $include_intersection = array_intersect($taxonomies, $include_category_ids);
1080
+ if(!empty($include_category_ids) && empty($include_intersection)) {
1081
+ $do_send_post = false;
1082
+ }
1083
+
1084
+ $exclude_intersection = array_intersect($taxonomies, $exclude_category_ids);
1085
+ // post categories should not match any one of the email's excluded categories
1086
+ if(!empty($exclude_category_ids) && !empty($exclude_intersection)) {
1087
+ $do_send_post = false;
1088
+ }
1089
+
1090
+ if($do_send_post) {
1091
+ WYSIJA::log('post_transition_hook_give_birth', array(
1092
+ 'post_id' => $post->ID,
1093
+ 'post_title' => $post->post_title,
1094
+ 'newsletter' => $email,
1095
+ 'old_status' => $old_status,
1096
+ 'new_status' => $new_status
1097
+ ),'post_notif');
1098
+
1099
+ $model_email->reset();
1100
+ $model_email->give_birth($email, $post->ID);
1101
+ }
1102
+ }
1103
+ }
1104
+ }
1105
+
1106
+ // we check for automatic latest content widget in automatic newsletter
1107
+ $helper_autonews = WYSIJA::get('autonews', 'helper');
1108
+ $helper_autonews->refresh_automatic_content();
1109
+ }
1110
+
1111
+ return true;
1112
+ }
1113
+
1114
+ /**
1115
+ * uninstall process not used
1116
+ */
1117
+ public static function uninstall(){
1118
+ $helperUS=WYSIJA::get('uninstall','helper');
1119
+ $helperUS->uninstall();
1120
+ }
1121
+
1122
+ /**
1123
+ * this function is run when wysija gets activated
1124
+ * there is no installation process here, all is about checking the global status of the app
1125
+ */
1126
+ public static function activate(){
1127
+ $encoded_option=get_option('wysija');
1128
+ $installApp=false;
1129
+ if($encoded_option){
1130
+ $values=unserialize(base64_decode($encoded_option));
1131
+ if(isset($values['installed'])) $installApp=true;
1132
+ }
1133
+
1134
+ //test again for plugins on reactivation
1135
+ if($installApp){
1136
+ $helper_import=WYSIJA::get('plugins_import','helper');
1137
+ $helper_import->testPlugins();
1138
+
1139
+ //resynch wordpress list
1140
+ $helper_user=WYSIJA::get('user','helper');
1141
+ $helper_user->synchList($values['importwp_list_id']);
1142
+ }
1143
+ }
1144
+
1145
+ /**
1146
+ * the is_plugin_active functions from WordPress sometimes are not loaded so here is one that works for single and multisites anywhere in the code
1147
+ * @param type $pluginName
1148
+ * @return type
1149
+ */
1150
+ public static function is_plugin_active($pluginName){
1151
+ $arrayactiveplugins=get_option('active_plugins');
1152
+ //we check in the list of the site options if the plugin is activated
1153
+ if(in_array($pluginName, $arrayactiveplugins)) {
1154
+ //plugin is activated for that site
1155
+ return true;
1156
+ }
1157
+
1158
+ //if this is a multisite it might not be activated in the site option but network activated though
1159
+ if(is_multisite()){
1160
+ $plugins = get_site_option('active_sitewide_plugins');
1161
+ //plugin is activated for that multisite
1162
+ if(isset($plugins[$pluginName])){
1163
+ return true;
1164
+ }
1165
+ }
1166
+ return false;
1167
+ }
1168
+
1169
+ /**
1170
+ * make sure that the current user has the good access rights corresponding to its role
1171
+ * @global type $current_user
1172
+ * @return type
1173
+ */
1174
+ public static function update_user_caps(){
1175
+ global $current_user;
1176
+
1177
+ if(empty($current_user) && function_exists('get_currentuserinfo')) get_currentuserinfo();
1178
+ if(empty($current_user)) return false;
1179
+ $current_user->get_role_caps();
1180
+
1181
+ return true;
1182
+ }
1183
+
1184
+ /**
1185
+ * test whether the plugin is a beta version or not
1186
+ * 2.4.4.4 is a beta
1187
+ * 2.4.4 is a bug fix release
1188
+ * 2.4 is a feature release
1189
+ * @param string $plugin_name
1190
+ */
1191
+ public static function is_beta($plugin_name=false){
1192
+ // exceptions
1193
+ $not_beta_versions = array('2.5.9.1', '2.5.9.2', '2.5.9.3', "2.5.9.4");
1194
+ $mailpoet_version = WYSIJA::get_version($plugin_name);
1195
+ if(in_array($mailpoet_version, $not_beta_versions)) return false;
1196
+
1197
+ // standard way of defining a beta version
1198
+ if(count(explode('.', $mailpoet_version)) > 3 ) return true;
1199
+ return false;
1200
+ }
1201
+
1202
+ /**
1203
+ * depending where it's used the base function from WordPress doesn't work, so this one will work anywhere
1204
+ * @param type $capability
1205
+ * @return type
1206
+ */
1207
+ public static function current_user_can($capability){
1208
+ if(!$capability) return false;
1209
+ WYSIJA::update_user_caps();
1210
+ if(!current_user_can($capability)) return false;
1211
+ return true;
1212
+ }
1213
+
1214
+ /**
1215
+ * this function get and sets the cron schedules when MailPoet's own cron system is active
1216
+ * @staticvar type $cron_schedules
1217
+ * @param type $schedule
1218
+ * @return type
1219
+ */
1220
+ public static function get_cron_schedule($schedule='queue'){
1221
+ static $cron_schedules;
1222
+
1223
+ //if the cron schedules are already loaded statically then we just have to return the right schedule value
1224
+ if(!empty($cron_schedules)){
1225
+ if($schedule=='all') return $cron_schedules;
1226
+ if(isset($cron_schedules[$schedule])) {
1227
+ return $cron_schedules[$schedule];
1228
+ }else{
1229
+ WYSIJA::set_cron_schedule($schedule);
1230
+ return false;
1231
+ }
1232
+ }else{
1233
+ //this is the first time this function is executed so let's get them from the db and store them statically
1234
+ $cron_schedules=get_option('wysija_schedules',array());
1235
+ if(!empty($cron_schedules)){
1236
+ if(isset($cron_schedules[$schedule])) return $cron_schedules[$schedule];
1237
+ else return false;
1238
+ }else{
1239
+ WYSIJA::set_cron_schedule();
1240
+ return false;
1241
+ }
1242
+ }
1243
+ return false;
1244
+ }
1245
+
1246
+ /**
1247
+ * return the frequency for each cron task needed by MailPoet
1248
+ * @return type an array of frequencies
1249
+ */
1250
+ public static function get_cron_frequencies(){
1251
+ $model_config = WYSIJA::get('config','model');
1252
+ $helper_forms = WYSIJA::get('forms','helper');
1253
+
1254
+ if(is_multisite() && $model_config->getValue('sending_method')=='network'){
1255
+ $sending_emails_each = $model_config->getValue('ms_sending_emails_each');
1256
+ }else{
1257
+ $sending_emails_each = $model_config->getValue('sending_emails_each');
1258
+ }
1259
+
1260
+ $queue_frequency = $helper_forms->eachValuesSec[$sending_emails_each];
1261
+ $bounce_frequency = 99999999999999;
1262
+ if(isset($helper_forms->eachValuesSec[$model_config->getValue('bouncing_emails_each')])){
1263
+ $bounce_frequency = $helper_forms->eachValuesSec[$model_config->getValue('bouncing_emails_each')];
1264
+ }
1265
+ return array('queue'=>$queue_frequency,'bounce'=>$bounce_frequency,'daily'=>86400,'weekly'=>604800,'monthly'=>2419200);
1266
+ }
1267
+
1268
+ /**
1269
+ * set the next cron schedule
1270
+ * TODO : needs probably to make the difference of running process for the next schedule, so that there is no delay(this is only problematic on some slow servers)
1271
+ * @param string $schedule
1272
+ * @param int $last_saved
1273
+ * @param boolean $set_running
1274
+ * @return boolean
1275
+ */
1276
+ public static function set_cron_schedule($schedule = false , $last_saved = 0 , $set_running = false){
1277
+ $cron_schedules = array();
1278
+
1279
+ $start_time = $last_saved;
1280
+ if(!$start_time) $start_time = time();
1281
+ $processes = WYSIJA::get_cron_frequencies();
1282
+ if(!$schedule){
1283
+ foreach($processes as $process => $frequency){
1284
+ $next_schedule = $start_time + $frequency;
1285
+ $prev_schedule = 0;
1286
+ if(isset($cron_schedules[$process]['running']) && $cron_schedules[$process]['running']) $prev_schedule=$cron_schedules[$process]['running'];
1287
+ $cron_schedules[$process]=array(
1288
+ 'next_schedule' => $next_schedule,
1289
+ 'prev_schedule' => $prev_schedule,
1290
+ 'running' => false);
1291
+ }
1292
+ }else{
1293
+ $cron_schedules = WYSIJA::get_cron_schedule('all');
1294
+ if($set_running){
1295
+ $cron_schedules[$schedule]['running'] = $set_running;
1296
+ }else{
1297
+ $running = 0;
1298
+ if(isset($cron_schedules[$schedule]['running'])) $running = $cron_schedules[$schedule]['running'];
1299
+ // if the process is not running or has been running for more than 15 minutes then we set the next_schedule date
1300
+ $process_frequency = $processes[$schedule];
1301
+
1302
+ if(!$running || ( time() > ($running + $process_frequency) ) ){
1303
+
1304
+ $next_schedule = $start_time + $process_frequency;
1305
+ // if the next schedule is already behind, we give it 30 seconds before it can triggers again
1306
+ if( $next_schedule < $start_time ){
1307
+ $next_schedule = $start_time + 30;
1308
+ }
1309
+
1310
+ $cron_schedules[$schedule] = array(
1311
+ 'next_schedule' => $next_schedule,
1312
+ 'prev_schedule' => $running,
1313
+ 'running' => false);
1314
+ }
1315
+ }
1316
+ }
1317
+ WYSIJA::update_option( 'wysija_schedules' , $cron_schedules , 'yes' );
1318
+ return true;
1319
+ }
1320
+
1321
+ /**
1322
+ * check that there is no passed schedules that need to be executed now
1323
+ * @return void
1324
+ */
1325
+ public static function cron_check() {
1326
+
1327
+ $cron_schedules = WYSIJA::get_cron_schedule('all');
1328
+ if(empty($cron_schedules)) return;
1329
+ else{
1330
+ $processes = WYSIJA::get_cron_frequencies();
1331
+
1332
+ $updated_sched = false;
1333
+ foreach($cron_schedules as $schedule => &$params){
1334
+ $running = 0;
1335
+ $time_now = time();
1336
+ if(isset($params['running'])) $running = $params['running'];
1337
+ //if the process has timedout we reschedule the next execution
1338
+ if($running && ( $time_now> ($running + $processes[$schedule]) ) ){
1339
+ //WYSIJA::setInfo('error','modifying next schedule for '.$proc);
1340
+ $process_frequency = $processes[$schedule];
1341
+
1342
+ $next_schedule = $running + $process_frequency;
1343
+ // if the next schedule is already behind, we give it 30 seconds before it can trigger again
1344
+ if( $next_schedule < $time_now ){
1345
+ $next_schedule = $time_now + 30;
1346
+ }
1347
+ $params=array(
1348
+ 'next_schedule' => $next_schedule,
1349
+ 'prev_schedule' => $running,
1350
+ 'running' => false);
1351
+ $updated_sched=true;
1352
+ }
1353
+ }
1354
+ if($updated_sched){
1355
+ //WYSIJA::setInfo('error','updating scheds');
1356
+ WYSIJA::update_option( 'wysija_schedules' , $cron_schedules , 'yes' );
1357
+ }
1358
+
1359
+ }
1360
+
1361
+ $time_now = time();
1362
+ $processesToRun = array();
1363
+ foreach($cron_schedules as $schedule => $scheduled_times){
1364
+ if(strpos($schedule, '(bounce handling not activated)')!==false) continue;
1365
+ $process_frequency = $processes[$schedule];
1366
+ if( ( !$scheduled_times['running'] || (int)$scheduled_times['running'] + $process_frequency < $time_now ) && $scheduled_times['next_schedule'] < $time_now){
1367
+ $processesToRun[] = $schedule;
1368
+ }
1369
+ }
1370
+
1371
+ $model_config = WYSIJA::get('config','model');
1372
+ $page_view_trigger = (int)$model_config->getValue('cron_page_hit_trigger');
1373
+ if(!empty($processesToRun) && $page_view_trigger === 1){
1374
+ //call the cron url
1375
+ // do not call that more than once per 5 minutes attempt at reducing the CPU load for some users
1376
+ // http://wordpress.org/support/topic/wysija-newsletters-slowing-down-my-site-1
1377
+ $last_cron_time_plus_5min = (int)get_option('wysija_last_php_cron_call') + (5*60);
1378
+
1379
+ if($last_cron_time_plus_5min < time()){
1380
+ $cron_url = site_url( 'wp-cron.php').'?'.WYSIJA_CRON.'&action=wysija_cron&process='.implode(',',$processesToRun).'&silent=1';
1381
+ $cron_request = apply_filters( 'cron_request', array(
1382
+ 'url' => $cron_url,
1383
+ 'args' => array( 'timeout' => 0.01, 'blocking' => false, 'sslverify' => apply_filters( 'https_local_ssl_verify', true ) )
1384
+ ) );
1385
+
1386
+ wp_remote_post( $cron_url, $cron_request['args'] );
1387
+ WYSIJA::update_option('wysija_last_php_cron_call', time());
1388
+ }
1389
+
1390
+
1391
+ }
1392
+ }
1393
+
1394
+ /**
1395
+ * Function somehow necessary to avoid some conflicts in windows server and WordPress autoload of plugins language file
1396
+ * @param type $boolean
1397
+ * @param type $domain
1398
+ * @param type $mofile
1399
+ * @return boolean
1400
+ */
1401
+ public static function override_load_textdomain($boolean, $domain, $mofile){
1402
+ $extensionloaded=WYSIJA::load_lang('get_all');
1403
+
1404
+ if(isset($extensionloaded[$domain]) && !@file_exists($mofile)){
1405
+ return true;
1406
+ }
1407
+
1408
+ return false;
1409
+ }
1410
+
1411
+ /**
1412
+ * function to rewrite the path of the file if the file doesn't exist
1413
+ * @param type $mofile
1414
+ * @param type $domain
1415
+ * @return type
1416
+ */
1417
+ public static function load_textdomain_mofile($mofile, $domain){
1418
+ $extensionloaded=WYSIJA::load_lang('get_all');
1419
+
1420
+ if(isset($extensionloaded[$domain]) && !file_exists($mofile)){
1421
+ return WYSIJA_PLG_DIR.$domain.DS.'languages'.DS.$extensionloaded[$domain].'-'.get_locale().'.mo';
1422
+ }
1423
+ return $mofile;
1424
+ }
1425
+ }
1426
+
1427
+ // subscribers/wp-user synch hooks
1428
+ add_action('user_register', array('WYSIJA', 'hook_add_WP_subscriber'), 1);
1429
+ add_action('added_existing_user', array('WYSIJA', 'hook_add_WP_subscriber'), 1);
1430
+ add_action('profile_update', array('WYSIJA', 'hook_edit_WP_subscriber'), 1);
1431
+ // for standard blog
1432
+ add_action('delete_user', array('WYSIJA', 'hook_del_WP_subscriber'), 1);
1433
+ // for multisite
1434
+ add_action('deleted_user', array('WYSIJA', 'hook_del_WP_subscriber'), 1);
1435
+ add_action('remove_user_from_blog', array('WYSIJA', 'hook_del_WP_subscriber'), 1);
1436
+
1437
+ // Load the Upgrader Class
1438
+ add_action('init', array('WJ_Upgrade', 'hook'), 9);
1439
+
1440
+ // post notif trigger
1441
+ add_action('transition_post_status', array('WYSIJA', 'hook_postNotification_transition'), 1, 3);
1442
+ // refresh auto newsletter content when a post is modified
1443
+ //add_action('save_post', array('WYSIJA', 'hook_auto_newsletter_refresh'), 1, 1);
1444
+ add_action('delete_post', array('WYSIJA', 'hook_auto_newsletter_refresh'), 1, 1);
1445
+
1446
+ // add image size for emails
1447
+ add_image_size('wysija-newsletters-max', 600, 9999);
1448
+
1449
+ $modelConf=WYSIJA::get('config','model');
1450
+ if($modelConf->getValue('installed_time')){
1451
+
1452
+ // START all that concerns the CRON
1453
+ // make sure we check when is the schedule due with wysija's cron
1454
+ if($modelConf->getValue('cron_manual')){
1455
+ // if WP cron tasks are still set, we clear them
1456
+ if(wp_get_schedule('wysija_cron_queue')) WYSIJA::deactivate();
1457
+
1458
+ // set the crons schedule for each process
1459
+ WYSIJA::get_cron_schedule();
1460
+
1461
+ // check that there is no late cron schedules if we are using wysija's cron option and that the cron option is triggerred by any page view
1462
+ if(!isset($_REQUEST['process'])){
1463
+ WYSIJA::cron_check();
1464
+ }
1465
+
1466
+ // this action is triggerred only by a cron job
1467
+ // if we're entering the wysija's cron part, it should end here
1468
+ if(isset($_REQUEST['action']) && $_REQUEST['action']=='wysija_cron'){
1469
+ // priority is hundred so that the messages such as unsubscribe or view in your browser have time to be translated(they get translated around 96, 97)
1470
+ add_action('init', 'init_wysija_cron',100);
1471
+
1472
+ function init_wysija_cron(){
1473
+ $hCron=WYSIJA::get('cron','helper');
1474
+ $hCron->run();
1475
+ }
1476
+ }
1477
+
1478
+ // make sure the scheduled tasks are recorded when using WordPress' cron
1479
+ }else{
1480
+
1481
+ // filter to add new possible frequencies to the cron
1482
+ add_filter( 'cron_schedules', array( 'WYSIJA', 'filter_cron_schedules' ) );
1483
+
1484
+ // action to handle the scheduled tasks in wysija
1485
+ add_action( 'wysija_cron_queue', array( 'WYSIJA', 'croned_queue' ) );
1486
+ add_action( 'wysija_cron_daily', array( 'WYSIJA', 'croned_daily' ) );
1487
+ add_action( 'wysija_cron_weekly', array( 'WYSIJA', 'croned_weekly' ) );
1488
+ add_action( 'wysija_cron_monthly', array( 'WYSIJA', 'croned_monthly' ) );
1489
+
1490
+ // same with the weekly task
1491
+ if(!wp_next_scheduled('wysija_cron_weekly')){
1492
+ wp_schedule_event( $modelConf->getValue('last_save') , 'eachweek', 'wysija_cron_weekly' );
1493
+ }
1494
+ // the monthly task...
1495
+ if(!wp_next_scheduled('wysija_cron_monthly')){
1496
+ wp_schedule_event( $modelConf->getValue('last_save') , 'each28days', 'wysija_cron_monthly' );
1497
+ }
1498
+
1499
+ // the daily task...
1500
+ if(!wp_next_scheduled('wysija_cron_daily')){
1501
+ wp_schedule_event( $modelConf->getValue('last_save') , 'daily', 'wysija_cron_daily' );
1502
+ }
1503
+
1504
+
1505
+ if(is_multisite()){
1506
+
1507
+ // in the case of multisite and the network's method we schedule with a different frequency
1508
+ // this option contains the list of sites already scheduled
1509
+ $ms_wysija_bounce_cron = get_site_option('ms_wysija_bounce_cron');
1510
+ global $blog_id;
1511
+
1512
+ // if this blog is not recorded in our wysija_sending_cron option then we clear its scheduled so that we can reinitialize it
1513
+ if(!$ms_wysija_bounce_cron || !isset($ms_wysija_bounce_cron[$blog_id])){
1514
+ wp_clear_scheduled_hook('wysija_cron_bounce');
1515
+ WYSIJA::set_cron_schedule('queue');
1516
+ $ms_wysija_bounce_cron[$blog_id] = 1;
1517
+ update_site_option('ms_wysija_bounce_cron',$ms_wysija_bounce_cron);
1518
+ }
1519
+
1520
+ }
1521
+
1522
+ // if the bounce task is not scheduled then we initialize it
1523
+ if(!wp_next_scheduled('wysija_cron_bounce')){
1524
+ wp_schedule_event( $modelConf->getValue('last_save') , $modelConf->getValue('bouncing_emails_each'), 'wysija_cron_bounce' );
1525
+ }
1526
+
1527
+ // and the queue processing task ...
1528
+ // if we are in a multisite case we make sure that the ms frequency hasn't been changed, if it has we reset it
1529
+ if(is_multisite() && $modelConf->getValue('sending_method')=='network'){
1530
+ // in the case of multisite and the network's method we schedule with a different frequency
1531
+ // this option contains the list of sites already scheduled
1532
+ $ms_wysija_sending_cron=get_site_option('ms_wysija_sending_cron');
1533
+ global $blog_id;
1534
+
1535
+ // if this blog is not recorded in our wysija_sending_cron option then we clear its scheduled so that we can reinitialize it
1536
+ if(!$ms_wysija_sending_cron || !isset($ms_wysija_sending_cron[$blog_id])){
1537
+ wp_clear_scheduled_hook('wysija_cron_queue');
1538
+ WYSIJA::set_cron_schedule('queue');
1539
+ $ms_wysija_sending_cron[$blog_id]=1;
1540
+ update_site_option('ms_wysija_sending_cron',$ms_wysija_sending_cron);
1541
+ }
1542
+ }
1543
+
1544
+
1545
+ // simply schedule the queue
1546
+ if(!wp_next_scheduled('wysija_cron_queue')){
1547
+
1548
+ // in the case of multisite and the network's method we schedule with a different frequency
1549
+ if(is_multisite() && $modelConf->getValue('sending_method')=='network'){
1550
+ $sending_emails_each=$modelConf->getValue('ms_sending_emails_each');
1551
+ }else{
1552
+ $sending_emails_each=$modelConf->getValue('sending_emails_each');
1553
+ }
1554
+ wp_schedule_event( $modelConf->getValue('last_save') , $sending_emails_each, 'wysija_cron_queue' );
1555
+ }
1556
+
1557
+ }// END all that concerns the CRON
1558
+
1559
+ // filter fixing a bug with automatic load_text_domain_from WP didn't understand yet why this was necessary...
1560
+ // somehow wp_register_script(which is irrelevant) was triggerring this kind of notice
1561
+ // Warning: is_readable() [function.is-readable]: open_basedir restriction in effect. File(C:\Domains\website.com\wwwroot\web/wp-content/plugins/C:\Domains\website.com\wwwroot\web\wp-content\plugins\wysija-newsletters/languages/wysija-newsletters-en_US.mo) is not within the allowed path(s): (.;C:\Domains\;C:\PHP\;C:\Sites\;C:\SitesData\;/) in C:\Domains\website.com\wwwroot\web\wp-includes\l10n.php on line 339
1562
+ // the only solution is to make sure on our end that the file exists and rewrite it if necessary
1563
+ add_filter( 'override_load_textdomain', array( 'WYSIJA', 'override_load_textdomain' ), 10, 3);
1564
+ add_filter('load_textdomain_mofile', array( 'WYSIJA', 'load_textdomain_mofile' ), 10, 2);
1565
+ }
1566
+
1567
+ // not yet used but the purpose is to override any notification sent through wp_mail
1568
+ if($modelConf->getValue('wp_notifications')){
1569
+ $hWPnotif=WYSIJA::get('wp_notifications','helper');
1570
+ }
1571
+
1572
+ register_deactivation_hook(WYSIJA_FILE, array( 'WYSIJA', 'deactivate' ));
1573
+ register_activation_hook(WYSIJA_FILE, array( 'WYSIJA', 'activate' ));
1574
+ add_action( 'init', array('WYSIJA','create_post_type') );
1575
+
1576
+
1577
+ // launch application
1578
+ $helper = WYSIJA::get(WYSIJA_SIDE,'helper');
core/constants.php ADDED
@@ -0,0 +1,46 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*make the difference between frontend and backend routing*/
4
+ /*require what is needed for common purpose in backend such as backend menus*/
5
+
6
+ if(defined('WP_ADMIN')) {
7
+ define('WYSIJA_SIDE','back');
8
+ }else define('WYSIJA_SIDE','front');
9
+
10
+ $plugin_name='wysija-newsletters';
11
+ $plugin_folder_name=dirname(dirname(plugin_basename(__FILE__)));
12
+ $current_folder=dirname(dirname(__FILE__));
13
+
14
+ if(!defined('DS')) define('DS', DIRECTORY_SEPARATOR);
15
+ define('WYSIJA', $plugin_name);
16
+ define('WYSIJAFUTURE','wysija-future');
17
+ if(!defined('WYSIJA_PLG_DIR')) define('WYSIJA_PLG_DIR', dirname($current_folder).DS);
18
+ define('WYSIJA_DIR', $current_folder.DS);
19
+ define('WYSIJA_DATA_DIR', WYSIJA_DIR.'data'.DS);
20
+ define('WYSIJA_FILE',WYSIJA_DIR.'index.php');
21
+
22
+ define('WYSIJA_URL',plugins_url().'/'.strtolower('wysija-newsletters').'/');
23
+
24
+ $upload_dir = wp_upload_dir();
25
+
26
+ define('WYSIJA_UPLOADS_DIR',str_replace('/',DS,$upload_dir['basedir']).DS.'wysija'.DS);
27
+ define('WYSIJA_UPLOADS_URL',$upload_dir['baseurl'].'/wysija/');
28
+ if(is_multisite()){
29
+ define('WYSIJA_UPLOADS_MS_DIR',str_replace(get_option( 'upload_path' ), get_blog_option(1, 'upload_path'), $upload_dir['basedir']).DS.'wysija'.DS);
30
+ define('WYSIJA_UPLOADS_MS_URL',get_blog_option(1, 'siteurl').'/'.get_blog_option(1, 'upload_path').'/wysija/');
31
+ }
32
+
33
+ define('WYSIJA_INC',WYSIJA_DIR.'inc'.DS);
34
+ define('WYSIJA_CORE',WYSIJA_DIR.'core'.DS);
35
+ define('WYSIJA_VIEWS',WYSIJA_DIR.'views'.DS);
36
+ define('WYSIJA_MODELS',WYSIJA_DIR.'models'.DS);
37
+ define('WYSIJA_HELPERS',WYSIJA_DIR.'helpers'.DS);
38
+ define('WYSIJA_CLASSES',WYSIJA_DIR.'classes'.DS);
39
+ define('WYSIJA_CTRL',WYSIJA_DIR.'controllers'.DS);
40
+ define('WYSIJA_WIDGETS',WYSIJA_DIR.'widgets'.DS);
41
+
42
+ define('WYSIJA_DIR_IMG',WYSIJA_DIR.'img'.DS);
43
+ define('WYSIJA_EDITOR_IMG',WYSIJA_URL.'img/');
44
+ define('WYSIJA_EDITOR_TOOLS',WYSIJA_DIR.'tools'.DS);
45
+ global $blog_id;
46
+ define('WYSIJA_CRON',md5(__FILE__.$blog_id));
core/controller.php ADDED
@@ -0,0 +1,99 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+
5
+ class WYSIJA_control extends WYSIJA_object{
6
+ var $model="";
7
+ var $view="";
8
+ var $action="";
9
+ var $list_columns=array();
10
+ var $form_columns=array();
11
+ var $filters=array();
12
+ var $js=array();
13
+ var $jsLoc=array();
14
+ var $extension="wysija-newsletters";
15
+ var $joins=array();
16
+ var $title="";
17
+
18
+ function WYSIJA_control(){
19
+ //setup some required objects for the request
20
+ if(!defined('DOING_AJAX')){
21
+ if($this->view) $this->viewObj = WYSIJA::get($this->view,"view",false,$this->extension);
22
+ if(empty($this->viewObj)) $this->viewObj = new stdClass (); // In some cases, viewObj can not be created
23
+ if($this->model){
24
+ $this->modelObj=WYSIJA::get($this->model,"model",false,$this->extension);
25
+ $this->viewObj->model=WYSIJA::get($this->model,"model",false,$this->extension);
26
+ }
27
+ }
28
+
29
+ // Security test, we should never do anything before WordPress Init; It's just wrong
30
+ add_action('init', array(&$this, 'nonce_verification'));
31
+ }
32
+
33
+ public function nonce_verification(){
34
+ if(!empty($_REQUEST['_wpnonce'])){
35
+ $_REQUEST['wpnonceback']=$_REQUEST['_wpnonce'];
36
+
37
+ if($_REQUEST['action']=='wysija_ajax'){
38
+ $actionnonce='wysija_ajax';
39
+ }else{
40
+ //backend case
41
+ if(is_admin() && !empty($_REQUEST['page'])){
42
+ $actionnonce=$_REQUEST['page'].'-action_'.$_REQUEST['action'];
43
+ if(!empty($_REQUEST['id'])) $actionnonce.='-id_'.$_REQUEST['id'];
44
+ //frontend case
45
+ } elseif (!empty($_REQUEST['controller'])) {
46
+ $actionnonce=$_REQUEST['controller'].'-action_'.$_REQUEST['action'];
47
+ if(!empty($_REQUEST['id'])) $actionnonce.='-id_'.$_REQUEST['id'];
48
+ }
49
+ }
50
+
51
+ if(!$_REQUEST['action'])
52
+ return true;
53
+
54
+ //if the wp_nonce has been set up then we test it against the one here if it fails we just die
55
+ $nonce=$_REQUEST['_wpnonce'];
56
+
57
+ if(!wp_verify_nonce($nonce, $actionnonce) ){
58
+ wp_die("<h2>" . __('Security failure during request') . "</h2>", __("Security Problem"), array(
59
+ 'response' => 403,
60
+ 'back_link' => true
61
+ ));
62
+ }
63
+ }
64
+ }
65
+
66
+ /**
67
+ * if a controller calls that page then it needs those global parameters to be set
68
+ * @return boolean
69
+ */
70
+ function requireSecurity(){
71
+
72
+ if(!isset($_REQUEST['wpnonceback']) && !isset($_REQUEST['_wpnonce'])) {
73
+ die('Your request is not safe.');
74
+ }else{
75
+ return true;
76
+ }
77
+ }
78
+
79
+
80
+ /**
81
+ * prepare an array of condition for a where statement with the pk and its value.
82
+ * note: not sure this function should be here though.
83
+ * @return array
84
+ */
85
+ function getPKVal(){
86
+
87
+ if(isset($_POST['wysija'][$this->modelObj->table_name][$this->modelObj->pk]) && $_POST['wysija'][$this->modelObj->table_name][$this->modelObj->pk]){
88
+ //this is an update
89
+ $conditions=array($this->modelObj->pk =>$_POST['wysija'][$this->modelObj->table_name][$this->modelObj->pk]);
90
+ unset($_POST['wysija'][$this->modelObj->table_name][$this->modelObj->pk]);
91
+ }elseif(isset($_GET['id'])){
92
+ $conditions=array($this->modelObj->pk =>$_GET['id']);
93
+ }else{
94
+ $conditions=array();
95
+ }
96
+
97
+ return $conditions;
98
+ }
99
+ }
core/index.html ADDED
File without changes
core/model.php ADDED
@@ -0,0 +1,1034 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_model extends WYSIJA_object{
4
+
5
+ var $table_prefix='wysija';
6
+ var $table_name='';
7
+ var $pk='';
8
+ var $values=array();
9
+ var $conditions=array();
10
+ var $orderby=array();
11
+ var $groupby=false;
12
+ var $noCheck =false;
13
+ var $replaceQRY=false;
14
+ var $limitON=false;
15
+ var $dbg=false;
16
+ var $colCheck=true;
17
+ var $getFormat=ARRAY_A;
18
+ var $getOne=false;
19
+ var $fieldValid=true;
20
+ var $specialUpdate=false;
21
+ var $escapeFields=array();
22
+ var $escapingOn=false;
23
+ var $tableWP=false;
24
+ var $columns=array();
25
+ var $joins=array();
26
+ var $ignore = false;
27
+ var $sql_error = false;
28
+ var $last_error = '';
29
+ var $comparisonKeys = array('equal', 'notequal', 'like', 'greater', 'less', 'greater_eq', 'less_eq', 'is');
30
+ var $time_start = 0;
31
+ var $query_duration = 0;
32
+
33
+ function WYSIJA_model($extensions=''){
34
+ if(defined('WYSIJA_DBG') && WYSIJA_DBG>0) $this->dbg=true;
35
+ global $wpdb;
36
+ $this->wpprefix=$wpdb->prefix;
37
+ if($extensions) $this->table_prefix=$extensions;
38
+ //fix for radiokapi
39
+ //$this->wpprefix=$wpdb->base_prefix;
40
+ }
41
+ /**
42
+ * since we reuse the same objects accross the whole application
43
+ * once in a while for instance between a delete and a select we need to reset the objects to update the conditions
44
+ */
45
+ function reset(){
46
+ $this->values = array();
47
+ $this->conditions = array();
48
+ $this->orderby = array();
49
+ $this->groupby = false;
50
+ $this->getFormat = ARRAY_A;
51
+ $this->getOne = false;
52
+ $this->limitON = false;
53
+ $this->sql_error = false;
54
+ $this->last_error = '';
55
+ }
56
+
57
+ /**
58
+ *
59
+ * @param type $columnsOrPKval
60
+ * @param type $conditions
61
+ * @return type
62
+ */
63
+ function get($columnsOrPKval=false,$conditions=array()){
64
+ /*then columns becomes the pk value*/
65
+ if(!$conditions){
66
+ $conditions=array('equal'=>array($this->pk=>$columnsOrPKval));
67
+ $columnsOrPKval=false;
68
+ $this->noCheck=true;
69
+ }
70
+
71
+ /* if we pass just the id strong in the get conditions then it's the pk*/
72
+ if($conditions && !is_array($conditions)){
73
+ $conditions=array('equal'=>array($this->pk=>$conditions));
74
+ }
75
+ if($this->setConditions($conditions)){
76
+ if($this->getOne) $results=$this->getRows($columnsOrPKval,0,1);
77
+ else $results=$this->getRows($columnsOrPKval);
78
+ //$this->escapeQuotesFromRes($results);
79
+ if($this->getOne && count($results)==1){
80
+ switch($this->getFormat){
81
+ case ARRAY_A:
82
+ foreach($results as $res)return $res;
83
+ break;
84
+ case OBJECT:
85
+ foreach($results as $res)return $res;
86
+ break;
87
+ }
88
+ }
89
+ else return $results;
90
+ }
91
+
92
+ return false;
93
+ }
94
+
95
+ function getOne($columnsOrPKval=false,$conditions=array()){
96
+ $this->getOne=true;
97
+ $this->limitON=true;
98
+
99
+ return $this->get($columnsOrPKval,$conditions);
100
+ }
101
+
102
+ /**
103
+ * get a list of result based on a select query
104
+ * @global type $wpdb
105
+ * @param type $columns
106
+ * @param type $page
107
+ * @param type $limit
108
+ * @return type
109
+ */
110
+ function getRows($columns=false,$page=0,$limit=false){
111
+
112
+ /*set the columns*/
113
+ if($columns){
114
+ if(is_array($columns)){
115
+ $columns=implode(', ',$columns);
116
+ }
117
+ }else $columns='*';
118
+
119
+
120
+ $query='SELECT '.$columns.' FROM `'.$this->getSelectTableName()."`";
121
+ $query.=$this->makeJoins();
122
+ $query.=$this->makeWhere();
123
+ $query.=$this->makeGroupBY();
124
+ $query.=$this->makeOrderBY();
125
+
126
+ if($this->limitON) $query.=$this->setLimit($page,$limit);
127
+ $results=$this->query('get_res',$query,$this->getFormat);
128
+
129
+ //$this->escapeQuotesFromRes($results);
130
+
131
+ return $results;
132
+ }
133
+
134
+ function escapeQuotesFromRes(&$results){
135
+ if(!$this->escapingOn) return false;
136
+ foreach($results as $k =>$r){
137
+
138
+ if(in_array($this->getFormat,array(ARRAY_A,ARRAY_N))){
139
+ foreach($r as $k1 =>$v1){
140
+ if(in_array($k1,$this->escapeFields)){
141
+ $results[$k][$k1]= stripslashes($v1);
142
+ }
143
+ }
144
+ }
145
+ }
146
+ }
147
+
148
+ function setLimit($page=0,$limit=false){
149
+ /*set the limit of the selection*/
150
+
151
+ if(!$this->getOne){
152
+ if($page==0){
153
+ if(isset($_REQUEST['pagi'])){
154
+ $page=(int)$_REQUEST['pagi'];
155
+ if($page!=0) $page=$page-1;
156
+ }
157
+
158
+ }else $page=$page-1;
159
+ }
160
+
161
+ if(!$limit){
162
+ if(isset($this->limit_pp)) $limit=$this->limit_pp;
163
+ else{
164
+ $config=WYSIJA::get('config','model');
165
+ $limit=$config->getValue('limit_listing');
166
+ }
167
+ }
168
+
169
+ $this->limit=(int)$limit;
170
+ $this->page=$page;
171
+ $this->limit_start=(int)($this->page*$this->limit);
172
+ $this->limit_end=(int)($this->limit_start+$this->limit);
173
+
174
+ return " LIMIT $this->limit_start , $this->limit";
175
+ }
176
+
177
+ /**
178
+ * DEPRECATED
179
+ * to have a custom query through the model and get the result immediately
180
+ * @param type $query
181
+ * @return type
182
+ */
183
+ function getResults($query,$type=ARRAY_A){
184
+ return $this->query('get_res',$query, $type);
185
+ }
186
+
187
+ /**
188
+ * to have a custom query through the model and get the result immediately
189
+ * @param type $query
190
+ * @return type
191
+ */
192
+ public function get_results($query,$type=ARRAY_A){
193
+ return $this->getResults($query,$type);
194
+ }
195
+
196
+
197
+ function getSelectTableName(){
198
+ if($this->joins && isset($this->joins['tablestart'])){
199
+ if(isset($this->joins['prefstart'])) return $this->wpprefix.$this->joins['prefstart'].'_'.$this->joins['tablestart'];
200
+ else return $this->getPrefix().$this->joins['tablestart'];
201
+ }else return $this->getPrefix().$this->table_name;
202
+ }
203
+
204
+ /**
205
+ * simple SQL count
206
+ * @global type $wpdb
207
+ * @return type
208
+ */
209
+ function count($query=false,$keygetcount=false){
210
+ if(!$query){
211
+ $groupBy=$this->makeGroupBY();
212
+ $columnMore='';
213
+ if($groupBy) $columnMore=','.$this->groupby;
214
+ $query='SELECT COUNT('.$this->getPk().') as count '.$columnMore.' FROM `'.$this->getSelectTableName().'`';
215
+ $query.=$this->makeJoins();
216
+
217
+ $query.=$this->makeWhere();
218
+ $query.=$groupBy;
219
+ }
220
+
221
+
222
+ // if dbg is on we track the duration of the query
223
+ if($this->dbg){
224
+ $this->timer_start();
225
+ }
226
+ $results=$this->query('get_res',$query,$this->getFormat);
227
+
228
+ // if dbg is on we track the duration of the query
229
+ if($this->dbg){
230
+ $this->timer_stop();
231
+ $this->keepQry('count');
232
+ }
233
+
234
+ if(!$results || count($results)>1) return $results;
235
+ else {
236
+ if($keygetcount) return $results[0][$keygetcount];
237
+ else{
238
+ foreach($results[0] as $key => $count) return $count;
239
+ }
240
+ }
241
+
242
+
243
+ return $results;
244
+ }
245
+
246
+ /**
247
+ * make the SQL WHERE condition string
248
+ * @return string
249
+ */
250
+ function makeWhere(){
251
+ $query='';
252
+ if($this->conditions){
253
+ /*set the WHERE clause*/
254
+ $conditions=array();
255
+ foreach($this->conditions as $type=>$values){
256
+ if(!in_array($type, $this->comparisonKeys)){
257
+ $conditionsss=$this->conditions;
258
+ $this->conditions=array();
259
+ $this->conditions['equal']=$conditionsss;
260
+
261
+ break;
262
+ }
263
+ }
264
+ foreach($this->conditions as $type=>$values){
265
+ if($type=='like' && count($values)>1){
266
+ if(is_array($values)){
267
+ $total=count($values);
268
+ $i=1;
269
+ $likeCond='';
270
+ foreach($values as $qfield => $qval){
271
+ $likeCond.=$qfield." LIKE '%".esc_sql(addcslashes($qval, '%_' ))."%'";
272
+ if($i<$total){
273
+ $likeCond.=' OR ';
274
+ }
275
+ $i++;
276
+ }
277
+ $conditions[]='('.$likeCond.')';
278
+ }
279
+ continue;
280
+ }
281
+ foreach($values as $condK => $condVal){
282
+
283
+ //secure from injections
284
+ $this->_secureFieldVal($condK, $condVal);
285
+
286
+ switch($type){
287
+ case 'equal':
288
+ if(is_array($condVal)){
289
+ $conditions[]=$condK.' IN ("'.implode('","', $condVal).'")';
290
+ }else{
291
+ if(is_null($condVal)) {
292
+ $conditions[] = $condK.' IS NULL';
293
+ } else {
294
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
295
+ $conditions[] = $condK.'='.$condVal;
296
+ }
297
+ }
298
+ break;
299
+ case 'notequal':
300
+ if(is_array($condVal)){
301
+ $conditions[]=$condK.' NOT IN ("'.implode('","', $condVal).'")';
302
+ }else{
303
+ //this means that if I delete something with a list of ids and the array happens to be empty array of ids it will just delete everything by
304
+ if(is_null($condVal)) {
305
+ $conditions[] = $condK.' IS NOT NULL';
306
+ } else {
307
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
308
+ $conditions[] = $condK.' != '.$condVal;
309
+ }
310
+ }
311
+ break;
312
+ case 'like':
313
+ $conditions[]=$condK." LIKE '%".esc_sql(addcslashes($condVal, '%_' ))."%'";
314
+ break;
315
+ case 'greater':
316
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
317
+ $conditions[]=$condK.' > '.$condVal;
318
+ break;
319
+ case 'less':
320
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
321
+ $conditions[]=$condK.' < '.$condVal;
322
+ break;
323
+ case 'greater_eq':
324
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
325
+ $conditions[]=$condK.' >= '.$condVal;
326
+ break;
327
+ case 'less_eq':
328
+ if(is_numeric($condVal) === false) $condVal = '"'.$condVal.'"';
329
+ $conditions[]=$condK.' <= '.$condVal;
330
+ break;
331
+ case 'is':
332
+
333
+ $conditions[]=$condK.' '.$condVal;
334
+ break;
335
+ }
336
+ }
337
+
338
+ }
339
+
340
+ $query.=' WHERE '.implode(' AND ',$conditions);
341
+ }
342
+
343
+ return $query;
344
+ }
345
+
346
+ /**
347
+ * make the SQL ORDER BY condition string
348
+ * @return string
349
+ */
350
+ function makeOrderBY(){
351
+ $query=' ORDER BY ';
352
+ if($this->orderby){
353
+ /*set the ORDER BY clause*/
354
+ $query.=$this->orderby.' '.$this->orderbyt;
355
+ }else{
356
+ /*by default we order by pk desc*/
357
+ if(is_array($this->pk)) return '';
358
+ $query.=$this->pk.' DESC';
359
+ }
360
+ return $query;
361
+ }
362
+
363
+
364
+ function makeJoins(){
365
+
366
+ if($this->joins){
367
+ $join=' as A';
368
+ $arrayLetters=array('B','C','D','E');
369
+ foreach($this->joins['tablejoins'] as $table => $fk){
370
+ $letter=array_shift($arrayLetters);
371
+ $join.=' JOIN `'.$this->getPrefix().$table.'` AS '.$letter." on $letter.$fk=A.".$this->joins['keystart'].' ';
372
+ }
373
+ /*set the ORDER BY clause*/
374
+ return $join;
375
+ }else return '';
376
+
377
+ }
378
+ /**
379
+ * make the SQL ORDER BY condition string
380
+ * @return string
381
+ */
382
+ function makeGroupBY(){
383
+
384
+ if($this->groupby){
385
+ /*set the ORDER BY clause*/
386
+ return ' GROUP BY '.$this->groupby;
387
+ }else return '';
388
+
389
+ }
390
+ function groupBy($name){
391
+
392
+ if (!is_string($name) OR preg_match('|[^a-z0-9#_.-]|i',$name) !== 0 ){
393
+ $this->groupby=false;
394
+ }else $this->groupby=$name;
395
+ }
396
+ function orderBy($name,$type = 'ASC'){
397
+
398
+ if(is_array($name) and count($name) > 0) {
399
+ // set order by to empty string
400
+ $this->orderby = '';
401
+ $this->ordert = '';
402
+
403
+ // count number of arguments
404
+ $count = count($name);
405
+
406
+ // build order by query
407
+ for($i = 0; $i < $count; $i++) {
408
+
409
+ $value = current($name);
410
+
411
+ //security escaping
412
+ if(!is_string(key($name)) OR preg_match('|[^a-z0-9#_.-]|i',key($name)) !== 0 ){
413
+ $orderByCol="";
414
+ }else $orderByCol=key($name);
415
+ //security escaping
416
+ if(!is_string($value) OR preg_match('|[^a-z0-9#_.-]|i',$value) !== 0 ){
417
+ $orderByVal="";
418
+ }else $orderByVal=$value;
419
+
420
+ if($i === ($count - 1)) {
421
+ $this->orderby .= $orderByCol;
422
+ $this->ordert = $orderByVal;
423
+ } else {
424
+ $this->orderby .=$orderByCol.' '.$orderByVal;
425
+ $this->orderby .= ', ';
426
+ next($name);
427
+ }
428
+ }
429
+ } else if(!is_string($name) OR preg_match('|[^a-z0-9#_.-]|i',$name) !== 0 ){
430
+ $this->orderby="";
431
+ }else {
432
+ $this->orderby=$name;
433
+ }
434
+
435
+ if(!in_array($type,array('DESC','ASC'))) $type = 'DESC';
436
+ $this->orderbyt=$type;
437
+ }
438
+
439
+
440
+
441
+ /**
442
+ * prepare for an insert procedure
443
+ * @param type $values
444
+ */
445
+ function insert($values,$ignore=false){
446
+ if($ignore)$this->ignore=true;
447
+ if($this->setValues($values)){
448
+ return $this->save();
449
+ }else{
450
+ $this->error(sprintf('missing values in model insert : %1$s.', get_class($this)));
451
+ }
452
+ }
453
+
454
+
455
+ function replace($values=array()){
456
+ $this->replaceQRY=true;
457
+ $this->insert($values);
458
+ $this->replaceQRY=false;
459
+ }
460
+
461
+ /**
462
+ * prepare for an update procedure
463
+ * @param type $values
464
+ * @param type $conditions
465
+ */
466
+ function update($values=array(),$conditions=array()){
467
+
468
+ if($this->setValues($values)){
469
+ /*if no condition is set then we set it mannualy based on the primary key*/
470
+ if(!$conditions){
471
+ if(!$this->conditions){
472
+ if(isset($values[$this->pk]) && $values[$this->pk]){
473
+
474
+ $this->setConditions(array($this->pk => $values[$this->pk]), true);
475
+
476
+ unset($values[$this->pk]);
477
+
478
+ return $this->save(true);
479
+
480
+ }else{
481
+ $this->error(sprintf('missing pk value in model update : %1$s.', get_class($this)));
482
+ }
483
+ }
484
+
485
+ }else{
486
+ if($this->setConditions($conditions,true)){
487
+ return $this->save(true);
488
+ }else{
489
+ $this->error(sprintf('missing conditions in model update : %1$s.', get_class($this)));
490
+ }
491
+ }
492
+
493
+ }else{
494
+ $this->error(sprintf('missing values in model update : %1$s.', get_class($this)));
495
+ }
496
+ }
497
+
498
+ /**
499
+ * UPDATE with a special where condition
500
+ * @param type $table
501
+ * @param type $data
502
+ * @param type $where
503
+ * @param type $format
504
+ * @param type $where_format
505
+ * @return type
506
+ */
507
+ function specialUpdate( $table, $data, $where, $format = null, $where_format = null ) {
508
+ if ( ! is_array( $data ) || ! is_array( $where ) )
509
+ return false;
510
+
511
+ $formats = $format = (array) $format;
512
+ $bits = $wheres = array();
513
+
514
+ $i=0;
515
+ foreach ( $data as $field => $val) {
516
+ $this->_secureFieldVal($field,$val);
517
+
518
+ switch($format[$i]){
519
+ case "%d":
520
+ $bits[] = "`$field` = ".(int)$val;
521
+ break;
522
+ case '[increment]':
523
+ $bits[] = "`$field` = ".$field.'+1';
524
+ break;
525
+ case '[decrement]':
526
+ $bits[] = "`$field` = ".$field.'-1';
527
+ break;
528
+ default :
529
+ $bits[] = "`$field` = '".$val."'";
530
+ }
531
+ $i++;
532
+ }
533
+
534
+ $sql = "UPDATE `$table` SET " . implode( ', ', $bits ) . ' ' . $this->makeWhere();
535
+ return $this->query( $sql );
536
+ }
537
+
538
+
539
+ function _secureFieldVal( &$field, &$mixed ) {
540
+ if ( ! is_string( $field ) || preg_match( '|[^a-z0-9#_.-]|i', $field ) !== 0 ) {
541
+ die('field "'.$field .'" not secured');
542
+ }
543
+ if ( is_string( $mixed ) || is_numeric( $mixed ) || is_bool( $mixed ) ) {
544
+ $mixed = esc_sql( $mixed );
545
+ } else {
546
+ if(!empty($mixed) && is_array($mixed)){
547
+ foreach ( $mixed as $key => &$value ) {
548
+ $this->_secureFieldVal( $field, $value );
549
+ }
550
+ }
551
+ }
552
+ }
553
+ /**
554
+ * save information as an update or an insert
555
+ * @global type $wpdb
556
+ * @param type $update
557
+ * @return type
558
+ */
559
+ function save($update=false){
560
+
561
+ if($update)$updateStr='Update';
562
+ else $updateStr='Insert';
563
+ $beforeSave='before'.$updateStr;
564
+ $afterSave='after'.$updateStr;
565
+
566
+
567
+
568
+ if(!$update && isset($this->columns['created_at']))$this->values['created_at']=time();
569
+ foreach($this->columns as $key => $params){
570
+ /*check for auto columns */
571
+ if((isset($params['autoup']) && $update) || (!$update && $key!='sent_at')){
572
+ if(isset($params['type']) && !isset($this->values[$key])){
573
+ switch($params['type']){
574
+ case 'date':
575
+ $this->values[$key]=time();
576
+ break;
577
+ case 'ip':
578
+ $userHelper=WYSIJA::get("user","helper");
579
+ /*record the ip and save the user*/
580
+ $this->values[$key]=$userHelper->getIP();
581
+ break;
582
+ case 'referer':
583
+ /*record the ip and save the user*/
584
+ $this->values[$key]=$_SERVER['HTTP_REFERER'];
585
+ break;
586
+ }
587
+ }
588
+
589
+ }
590
+ }
591
+
592
+ if(method_exists($this,$beforeSave)){
593
+ if(!$this->$beforeSave()){
594
+ //$this->error(sprintf('Problem during validation "%2$s" in model : %1$s.', get_class($this), $beforeSave));
595
+ return false;
596
+ }
597
+ }
598
+
599
+ /*prepare a format list for the update and insert function*/
600
+ $fieldsFormats=array();
601
+ if(!is_array($this->pk) && isset($this->values[$this->pk])) unset($this->values[$this->pk]);
602
+ foreach($this->values as $key =>$val){
603
+ if(!isset($this->columns[$key]['html'])) $this->values[$key]=strip_tags($val);
604
+ /* let's correct the type of the values based on the one defined in the model*/
605
+ if(in_array($val, array('[increment]','[decrement]'))){
606
+ $fieldsFormats[]=$val;
607
+ $this->specialUpdate=true;
608
+ }else{
609
+ //dbg($this->values);
610
+ if(!isset($this->columns[$key]['type'])){
611
+ $this->columns[$key]['type']='default';
612
+ }
613
+ switch($this->columns[$key]['type']){
614
+ case 'integer':
615
+ case 'boolean':
616
+ $fieldsFormats[]="%d";
617
+ break;
618
+ default:
619
+ $fieldsFormats[]="%s";
620
+
621
+ }
622
+ }
623
+
624
+ }
625
+
626
+ if($this->fieldValid && !$this->validateFields()) {
627
+ $this->error(__('Error Validating the fields',WYSIJA),true);
628
+ $this->stay=true;
629
+ return false;
630
+ }
631
+
632
+ global $wpdb;
633
+
634
+ // if dbg is on we track the duration of the query
635
+ if($this->dbg){
636
+ $this->timer_start();
637
+ }
638
+
639
+ if($update){
640
+
641
+ if( $this->specialUpdate || isset($this->conditions['equal']) || isset($this->conditions['notequal']) || isset($this->conditions['like'])){
642
+
643
+ $resultSave=$this->specialUpdate($this->getPrefix().$this->table_name,$this->values,$this->conditions,$fieldsFormats);
644
+ $this->logError();
645
+ }else{
646
+
647
+ $wpdb->update($this->getPrefix().$this->table_name,$this->values,$this->conditions,$fieldsFormats);
648
+ $this->logError();
649
+ $resultSave=$wpdb->result;
650
+ }
651
+
652
+ }else{
653
+ if($this->replaceQRY){
654
+ $resultSave=$wpdb->replace($this->getPrefix().$this->table_name,$this->values,$fieldsFormats);
655
+ $this->logError();
656
+ }else{
657
+
658
+ if($this->ignore) $resultSave=$wpdb->insert($this->getPrefix().$this->table_name,$this->values,$fieldsFormats);
659
+ else $resultSave=$wpdb->insert($this->getPrefix().$this->table_name,$this->values,$fieldsFormats);
660
+
661
+ $this->logError();
662
+ //dbg('hello');
663
+ }
664
+
665
+ }
666
+
667
+ // if dbg is on we track the duration of the query
668
+ if($this->dbg){
669
+ $this->timer_stop();
670
+ $this->keepQry('save');
671
+ }
672
+
673
+ if(!$resultSave){
674
+ $wpdb->show_errors();
675
+ return false;
676
+ }else{
677
+ if($update){
678
+ if(isset($this->conditions[$this->getPk()])){
679
+ $resultSave=$this->conditions[$this->getPk()];
680
+ }else{
681
+ if(isset($this->conditions[$this->getPk(1)])) $resultSave=$this->conditions[$this->getPk(1)];
682
+ }
683
+
684
+ }else{
685
+ $resultSave=$wpdb->insert_id;
686
+ }
687
+ }
688
+
689
+ $wpdb->flush();
690
+
691
+ if(method_exists($this,$afterSave)){
692
+ if(!$this->$afterSave($resultSave)){
693
+ //$this->error(sprintf('Problem during validation "%2$s" in model : %1$s.', get_class($this), $afterSave));
694
+ return false;
695
+ }
696
+ }
697
+ return $resultSave;
698
+ }
699
+
700
+
701
+ function insertMany($values){
702
+ $fields=array_keys($values[0]);
703
+
704
+ $query='INSERT INTO `'.$this->getPrefix().$this->table_name.'` (`' . implode( '`,`', $fields ) . '`) VALUES ';
705
+
706
+ $total=count($values);
707
+ $i=1;
708
+ foreach($values as &$vals){
709
+ foreach($vals as &$v) $v=esc_sql($v);
710
+ $query.= "('" . implode( "','", $vals )."')";
711
+ if($i<$total) $query.=',';
712
+ $i++;
713
+ }
714
+
715
+ $this->query($query.$myvalues);
716
+
717
+ }
718
+
719
+ /**
720
+ * validate the fields type(defined in each model) in the save procedure
721
+ * @return type
722
+ */
723
+ function validateFields(){
724
+ $error=false;
725
+ foreach($this->values as $key =>$val){
726
+ if(isset($this->columns[$key]['req']) && !$val && $this->columns[$key]['type']!='boolean'){
727
+ $this->error(sprintf(__('Field "%1$s" is required in table "%2$s".',WYSIJA), $key,$this->table_name),true);
728
+ $error=true;
729
+ }
730
+ /* let's correct the type of the values based on the one defined in the model*/
731
+ switch($this->columns[$key]['type']){
732
+ case "email":
733
+ $userHelper = WYSIJA::get('user','helper');
734
+ if(!$userHelper->validEmail($val)){
735
+ $this->error(sprintf(__('Field "%1$s" needs to be a valid Email.',WYSIJA), $key),true);
736
+ $error=true;
737
+ }
738
+ break;
739
+ }
740
+ }
741
+
742
+ if($error) return false;
743
+ return true;
744
+ }
745
+
746
+ /**
747
+ * delete procedure
748
+ * @global type $wpdb
749
+ * @param type $conditions
750
+ * @return type
751
+ */
752
+ function delete($conditions){
753
+ $query='DELETE FROM `'.$this->getPrefix().$this->table_name.'`';
754
+
755
+ if($this->setConditions($conditions)){
756
+ $whereQuery=$this->makeWhere();
757
+ if(!$whereQuery){
758
+ $this->error('Cannot delete element without conditions in model : '.get_class($this));
759
+ }
760
+ }else{
761
+ $this->error('Cannot delete element without conditions in model : '.get_class($this));
762
+ return false;
763
+ }
764
+ $result=$this->beforeDelete($conditions);
765
+ if($result) $result=$this->query($query.$whereQuery);
766
+ else return false;
767
+ $this->afterDelete();
768
+
769
+ return true;
770
+ }
771
+
772
+ function exists($conditions){
773
+
774
+ $query='SELECT '.$this->getPk().' FROM `'.$this->getSelectTableName().'`';
775
+
776
+ $query.=$this->makeJoins();
777
+ if($this->setConditions($conditions)){
778
+ $whereQuery=$this->makeWhere();
779
+ if(!$whereQuery){
780
+ $this->error('Cannot test element without conditions in model : '.get_class($this));
781
+ }
782
+ }else{
783
+ $this->error('Cannot test element without conditions in model : '.get_class($this));
784
+ return false;
785
+ }
786
+ $res=$this->query('get_res',$query.$whereQuery, ARRAY_A);
787
+ if($res) return $res;
788
+ else return false;
789
+ }
790
+
791
+ function getPk($numb=0){
792
+ $pk=$this->pk;
793
+ if(is_array($pk)) $pk=$pk[$numb];
794
+ return $pk;
795
+ }
796
+
797
+ /**
798
+ * set the values after verifying them
799
+ * @param type $values
800
+ * @return type
801
+ */
802
+ function setValues($values){
803
+ if($this->colCheck && !$this->checkAreColumns($values)) return false;
804
+
805
+ $this->values=array();
806
+ $this->values=$values;
807
+ return true;
808
+ }
809
+
810
+ /**
811
+ *
812
+ * @param type $values
813
+ * @return type
814
+ */
815
+ function setJoin($joins){
816
+ $this->joins=$joins;
817
+ return true;
818
+ }
819
+
820
+ /**
821
+ * set the conditions after verifying them
822
+ * @param type $conditions
823
+ * @return type
824
+ */
825
+ function setConditions($conditions,$update=false){
826
+ if($conditions && is_array($conditions)){
827
+
828
+ $this->conditions=array();
829
+ if($update){
830
+ foreach($conditions as $key =>$cond){
831
+ if($this->colCheck && !$this->checkAreColumns($conditions)) return false;
832
+
833
+ if(is_array($cond)){
834
+ $this->specialUpdate=true;
835
+ $this->conditions=$conditions;
836
+
837
+ return true;
838
+ }else $this->conditions[$key]=$cond;
839
+
840
+ }
841
+ } else {
842
+ foreach($conditions as $key => $cond) {
843
+ if(!in_array($key, $this->comparisonKeys /*array('like','equal','notequal','greater','less','greater_eq','less_eq')*/)){
844
+ if($this->colCheck && !$this->checkAreColumns($conditions)) return false;
845
+ if(array_key_exists('equal', $this->conditions) === false) $this->conditions['equal'] = array();
846
+ $this->conditions['equal'][$key] = $cond;
847
+ }else{
848
+ if($this->colCheck && !$this->checkAreColumns($cond)) return false;
849
+ $this->conditions[$key]=$cond;
850
+ }
851
+
852
+ }
853
+ }
854
+
855
+ return true;
856
+ }else return false;
857
+ }
858
+
859
+ /**
860
+ * check that the columns corresponds to the columns in the model
861
+ * @param type $arrayColumns
862
+ * @return type
863
+ */
864
+ function checkAreColumns($columns){
865
+ if($this->noCheck) return true;
866
+ foreach($columns as $column => $values) {
867
+ // skip when column is a comparison key
868
+ if(in_array($column, $this->comparisonKeys)) continue;
869
+
870
+ $columnName = $column;
871
+ if(!isset($this->columns[$columnName])){
872
+ $this->error(sprintf('Column %1$s does not exist in model : %2$s', $columnName, get_class($this)));
873
+ return false;
874
+ }
875
+ }
876
+ return true;
877
+ }
878
+
879
+ function timer_start() {
880
+ $this->query_duration = 0;
881
+ $this->time_start = microtime( true );
882
+ return true;
883
+ }
884
+
885
+ function timer_stop() {
886
+ $this->query_duration = ( microtime( true ) - $this->time_start );
887
+ }
888
+
889
+ function query($query,$arg2='',$arg3=ARRAY_A){
890
+ global $wpdb;
891
+ $this->sql_error = false;
892
+ if(!$arg2) $query = str_replace(array('[wysija]','[wp]'),array($this->getPrefix(),$wpdb->prefix),$query);
893
+ else $arg2 = str_replace(array('[wysija]','[wp]'),array($this->getPrefix(),$wpdb->prefix),$arg2);
894
+
895
+ // if dbg is on we track the duration of the query
896
+ if($this->dbg){
897
+ $this->timer_start();
898
+ }
899
+
900
+ switch($query){
901
+ case 'get_row':
902
+ $result = $wpdb->get_row($arg2,$arg3);
903
+ $this->logError();
904
+ break;
905
+ case 'get_res':
906
+ $result = $wpdb->get_results($arg2,$arg3);
907
+ //$this->escapeQuotesFromRes($results);
908
+ $this->logError();
909
+ break;
910
+ default:
911
+ $result = $wpdb->query($query);
912
+ $this->logError();
913
+
914
+ // get the last insert id if it's an insert query
915
+ if(substr($query, 0, 6) == 'INSERT') $result = $wpdb->insert_id;
916
+
917
+ }
918
+ // if dbg is on we track the duration of the query
919
+ if($this->dbg){
920
+ $this->timer_stop();
921
+ $this->keepQry('query');
922
+ }
923
+ return $result;
924
+ }
925
+
926
+ function logError(){
927
+ if(defined('WYSIJA_DBG') && WYSIJA_DBG>1){
928
+ global $wysija_queries_errors, $wpdb;
929
+ if(!$wysija_queries_errors) $wysija_queries_errors = array();
930
+
931
+ $this->sql_error = $wpdb->last_error;
932
+
933
+ if( $this->sql_error &&
934
+ ( empty( $this->last_error ) || $this->last_error != $this->sql_error )) {
935
+ $this->last_error = $wysija_queries_errors[] = $this->sql_error;
936
+ $this->sql_error = false;
937
+ WYSIJA::log('queries_errors' , $this->sql_error , 'query_errors');
938
+ }
939
+
940
+ }
941
+
942
+ }
943
+
944
+ function keepQry($from = 'wpdb'){
945
+ global $wpdb,$wysija_queries;
946
+ $wysija_queries[$from][] = array('duration' => $this->query_duration , 'query' => $wpdb->last_query);
947
+ }
948
+
949
+ function getAffectedRows(){
950
+ global $wpdb;
951
+ return $wpdb->rows_affected;
952
+ }
953
+
954
+ function getErrorMsg(){
955
+ global $wpdb;
956
+ return $wpdb->show_errors();
957
+ }
958
+ /**
959
+ * get the full prefix for the table
960
+ * @global type $wpdb
961
+ * @return type
962
+ */
963
+ function getPrefix(){
964
+ if($this->tableWP) return $this->wpprefix.$this->table_prefix;
965
+ else return $this->wpprefix.$this->table_prefix.'_';
966
+ }
967
+
968
+ /**
969
+ * this function allows you to get the prefix from the main site on a multisite
970
+ * @return type
971
+ */
972
+ function get_site_prefix($blog_id=1){
973
+
974
+ switch_to_blog( $blog_id );
975
+ global $wpdb;
976
+ $main_site_prefix=$wpdb->prefix;
977
+ restore_current_blog();
978
+
979
+ if($this->tableWP) return $main_site_prefix.$this->table_prefix;
980
+ else return $main_site_prefix.$this->table_prefix.'_';
981
+ }
982
+
983
+ /**
984
+ *
985
+ * @param type $field_name name of field which will become a key
986
+ * @param array $dataset list of records
987
+ * @param boolean $removing_field_name decide if we should remove field name from output dataset
988
+ * @param string $field_name_as_value a field in which we consider its value as value of $field_name
989
+ * @return array field based indexed dataset
990
+ */
991
+ protected function indexing_dataset_by_field($field_name, Array $dataset, $removing_field_name = false, $field_name_as_value = null){
992
+ if (empty($dataset))
993
+ return array();
994
+ $tmp = array();
995
+ foreach ($dataset as $record){
996
+ if (isset($record[$field_name]))
997
+ {
998
+ if (!empty($field_name_as_value)){
999
+ $tmp[$record[$field_name]] = isset($record[$field_name_as_value]) ? $record[$field_name_as_value] : null;
1000
+ continue;
1001
+ }
1002
+ $tmp[$record[$field_name]] = $record;
1003
+ if ($removing_field_name)
1004
+ unset($tmp[$record[$field_name]][$field_name]);
1005
+ }
1006
+
1007
+ }
1008
+ return $tmp;
1009
+ }
1010
+
1011
+ function beforeInsert(){
1012
+ return true;
1013
+ }
1014
+
1015
+ function afterInsert($resultSaveID){
1016
+ return true;
1017
+ }
1018
+ function beforeDelete($conditions){
1019
+ return true;
1020
+ }
1021
+
1022
+ function afterDelete(){
1023
+ return true;
1024
+ }
1025
+
1026
+ function beforeUpdate($id = null){
1027
+ return true;
1028
+ }
1029
+
1030
+ function afterUpdate($resultSaveID){
1031
+ return true;
1032
+ }
1033
+
1034
+ }
core/module/module.php ADDED
@@ -0,0 +1,248 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ defined('WYSIJA') or die('Restricted access');
4
+
5
+ class WYSIJA_module extends WYSIJA_control{
6
+
7
+ /**
8
+ * ID of module
9
+ * @var string
10
+ */
11
+ protected $name;
12
+
13
+ /**
14
+ * view class of module
15
+ * @var string
16
+ */
17
+ public $view;
18
+
19
+ /**
20
+ * instance of view class of module
21
+ * @var string
22
+ */
23
+ protected $view_obj;
24
+
25
+
26
+ /**
27
+ * action/view of a hook
28
+ * @var string
29
+ */
30
+
31
+ protected $view_show;
32
+
33
+ /**
34
+ * data which view class will pull from
35
+ * @var array
36
+ */
37
+ protected $data;
38
+
39
+ protected $extended_plugin='wysija-newsletters';
40
+
41
+
42
+
43
+ /**
44
+ * Define hook name and list of modules of each hook
45
+ * @var Array
46
+ * @todo: implement hook management which allows to manage hooks from admin side
47
+ */
48
+
49
+ protected $is_premium = false;
50
+
51
+ public static $hooks = array(
52
+ 'hook_stats' => array(
53
+ // 'stats_newsletters_in_time',
54
+ 'stats_top_newsletters',
55
+ 'stats_top_subscribers',
56
+ 'stats_top_links',
57
+ 'stats_new_subscribers',
58
+ 'stats_subscriptions',
59
+ 'stats_top_domains'
60
+ ),
61
+
62
+ // the left block in the page "subscriber detail"
63
+ 'hook_subscriber_left' => array(
64
+ ),
65
+
66
+ // the righ block in the page "subscriber detail"
67
+ 'hook_subscriber_right' => array(
68
+ 'stats_subscriber'
69
+ ),
70
+ 'hook_subscriber_bottom' => array(
71
+ 'stats_subscribers_std',
72
+ // 'stats_newsletters_in_time' // not in use
73
+ ),
74
+ // top of newsletter (viewstats) page
75
+ 'hook_newsletter_top' => array(
76
+
77
+ 'stats_newsletter_std',
78
+ 'stats_newsletter',
79
+ // 'stats_reportcards_std'
80
+ ),
81
+
82
+ // Newsletters >> Newsletter detail: bottom block
83
+ 'hook_newsletter_bottom' => array(
84
+ //'stats_newsletter_std',
85
+ // 'stats_newsletters_in_time'
86
+ ),
87
+
88
+ // the block "super advanced" in Settings >> Advanced tab
89
+ 'hook_settings_super_advanced' => array(
90
+ 'archive_std'
91
+ ),
92
+
93
+ // event: before saving settings (Admin)
94
+ 'hook_settings_before_save' => array(
95
+ 'archive_std'
96
+ )
97
+ );
98
+ /**
99
+ * Constructor
100
+ * This is neccessary to override default action of WYSIJA_control::WYSIJA_control(),
101
+ * which always tries to load a default view object
102
+ */
103
+ public function __construct() {
104
+ if (!empty($this->model)){
105
+ $class_name = $this->model;
106
+ $this->model_obj = new $class_name();
107
+ $this->model_obj->limit = 0; // quickfix "Undefined property: WYSIJA_model_statistics::$limit in views\back.php::limitPerPage()"
108
+ }
109
+ $this->get_view_obj($this->extended_plugin);
110
+ if (!empty($this->view_obj) && !empty($this->model_obj)){
111
+ $this->view_obj->model = $this->model_obj;
112
+ }
113
+
114
+ $this->data['module_name'] = $this->name;
115
+
116
+
117
+ $model_config=WYSIJA::get('config','model');
118
+ if($model_config->getValue('premium_key'))
119
+ $this->is_premium = true;
120
+ $this->data['is_premium'] = $this->is_premium;
121
+ }
122
+
123
+ /**
124
+ * get name of module
125
+ * @return string
126
+ */
127
+ public function get_name(){
128
+ return $this->name;
129
+ }
130
+
131
+ /**
132
+ * Get unique link to the module and hook. This link will be displayed as an independent page and actually it renders [wysijap] postype
133
+ * @param string $module_name
134
+ * @param string $hook_name
135
+ * @param array $params (key => value, key => value)
136
+ * @return type
137
+ */
138
+ public static function get_module_link($module_name, $hook_name, $extended_plugin='wysija-newsletters', Array $params = array()) {
139
+ $model_config=WYSIJA::get('config','model');
140
+ $params = array_merge($params, array(
141
+ 'wysija-page' => 1,
142
+ 'controller'=>'module',
143
+ 'action' => 'init',
144
+ 'module' => $module_name,
145
+ 'extension' => $extended_plugin,
146
+ 'hook' => $hook_name
147
+ ));
148
+ return WYSIJA::get_permalink($model_config->getValue('confirm_email_link'),$params);
149
+ }
150
+
151
+ /**
152
+ * Return Hooks List
153
+ * @param string $hook_name name of hook
154
+ * @module_name string $module_name name of a specific module
155
+ * @return Array list of modules
156
+ */
157
+ public static function get_modules_from_hook($hook_name, $module_name = null){
158
+ $module_list = self::get_hook_module_list();
159
+ $modules = !empty($module_list[$hook_name]) ? $module_list[$hook_name] : array();
160
+ if ($module_name)
161
+ return isset($modules[$module_name]) ? array($modules[$module_name]) : array();
162
+ return $modules;
163
+ }
164
+
165
+ /**
166
+ * Get all registered hooks and modules
167
+ * @return Array
168
+ */
169
+ public static function get_hook_module_list(){
170
+ return self::$hooks;
171
+ }
172
+
173
+ /**
174
+ * Execute a hook, module by module, from first one to last one
175
+ * @param string $hook_name
176
+ * @param string $params
177
+ * @param string $extended_plugin
178
+ *
179
+ * @todo Performance factor:
180
+ * We are calling the same method for free / Premium version.
181
+ * Some modules don't exist at free side.
182
+ * Some modules don't exist at Premium side.
183
+ * This fact leads to an other fact: we have to check_exist() in both cases.
184
+ * Solution 1: cache by using a static attribute, within this class
185
+ * Solution 2: populate data to an external file (xml), and load that file into this static attribute (with solution 1)
186
+ */
187
+ public static function execute_hook($hook_name, $params, $extended_plugin='wysija-newsletters'){
188
+ $hook_output = '';
189
+ if (!empty(self::$hooks[$hook_name])){
190
+ foreach (self::$hooks[$hook_name] as $module_name){
191
+ $module = WYSIJA::get($module_name,'module',false,$extended_plugin);
192
+ if(!empty($module) && method_exists($module, $hook_name))
193
+ $hook_output .= $module->$hook_name($params);
194
+ }
195
+ }
196
+ return $hook_output;
197
+ }
198
+
199
+ /**
200
+ * get an instance of a module class
201
+ * @param string $module_name module to be loaded
202
+ * @param type $extended_plugin : used only when calling the url from a different plugin it is used watch those files :
203
+ * -core/controller.php line 21, 23 ,24
204
+ * @return an instance of WYSIJA_module or its derived classes
205
+ */
206
+ public static function get_instance_by_name($module_name,$extended_plugin='wysija-newsletters'){
207
+ return WYSIJA::get($module_name,'module',false, $extended_plugin);
208
+ }
209
+
210
+ /**
211
+ * Render a view/action
212
+ * @return string
213
+ */
214
+ public function render($buffering_output = true){
215
+ if (!empty($this->view))
216
+ {
217
+ if ($buffering_output)
218
+ ob_start();
219
+ if (!$buffering_output)
220
+ return $this->get_view_obj()->render($this->view_show, $this->data, true);
221
+ else{
222
+ $this->get_view_obj()->render($this->view_show, $this->data, true);
223
+ $view = ob_get_contents();
224
+ ob_end_clean();
225
+ return $view;
226
+ }
227
+ }
228
+ }
229
+
230
+ /**
231
+ * initialize WYSIJA_view instance
232
+ * @return WYSIJA_view
233
+ */
234
+ protected function get_view_obj(){
235
+ require_once(WYSIJA_CORE.'view.php');
236
+ require_once(WYSIJA_VIEWS.WYSIJA_SIDE.'.php');
237
+ if (empty($this->view_obj)){
238
+
239
+ $view_dir=WYSIJA_PLG_DIR.$this->extended_plugin.DS.'modules'.DS.$this->name; // quickfix, @todo
240
+ $class_path=$view_dir.DS.$this->view.'.php';// @todo: check exists
241
+ $class_name = strtoupper('wysija').'_module_view_'.$this->view;
242
+ require_once(WYSIJA_CORE.'view.php');
243
+ require_once($class_path);
244
+ $this->view_obj = new $class_name();
245
+ }
246
+ return $this->view_obj;
247
+ }
248
+ }
core/module/statistics.php ADDED
@@ -0,0 +1,33 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ class WYSIJA_module_statistics extends WYSIJA_module{
5
+ const GROUP_BY_YEAR = 1;
6
+ const GROUP_BY_MONTH = 2;
7
+ const GROUP_BY_DATE = 3;
8
+ const GROUP_BY_HOUR = 4;
9
+ const GROUP_BY_MINUTE = 5;
10
+
11
+ const ORDER_BY_SENT = 1;
12
+ const ORDER_BY_OPEN = 2;
13
+ const ORDER_BY_CLICK = 3;
14
+ const ORDER_BY_UNSUBSCRIBE = 4;
15
+
16
+ const ORDER_DIRECTION_ASC = 1;
17
+ const ORDER_DIRECTION_DESC = 2;
18
+
19
+ const SWITCHING_DATE_TO_MONTH_THRESHOLD = 90;// if the days between FROM and TO is greater than this value, we will group data by month instead of by date. Useful for charts.
20
+
21
+ const DEFAULT_TOP_RECORDS = 5; // default number of how many first records we should retrieve
22
+
23
+ public function __construct() {
24
+ parent::__construct();
25
+ $this->data['messages'] = $this->init_messages();
26
+ }
27
+
28
+ protected function init_messages() {
29
+ return array(
30
+ 'data_not_available' => __("There's no stats to load!", WYSIJA)
31
+ );
32
+ }
33
+ }
core/module/statistics_model.php ADDED
@@ -0,0 +1,139 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ defined('WYSIJA') or die('Restricted access');
4
+
5
+ class WYSIJA_module_statistics_model extends WYSIJA_model {
6
+
7
+ const STATS_DATE_OF_CREATION = 'STATS_DATE_OF_CREATION';
8
+ /**
9
+ * Time to live of stats data
10
+ */
11
+ const STATS_DATA_LIFE_TIME = 3600; // 1h * 60mins * 60s
12
+ /**
13
+ * Time to live of stats's table (structure)
14
+ */
15
+ const STATS_TABLE_LIFE_TIME = 86400; // 24h * 60mins * 60s
16
+
17
+ const STATS_PREFIX = 'stats_cache_'; // prefix of stat tables
18
+
19
+ public function __construct() {
20
+ parent::WYSIJA_model();
21
+ $this->clean_up_out_of_date_tables();
22
+ }
23
+
24
+ /**
25
+ *
26
+ * @param array $params array of input params
27
+ */
28
+
29
+ public function get_hash(Array $params) {
30
+ return md5(get_class($this) . json_encode($params));
31
+ }
32
+
33
+ /**
34
+ * set a date of creating temporary tables, useful for caching
35
+ * @return type
36
+ */
37
+ protected function set_date_of_creation() {
38
+ $config = WYSIJA::get('config', 'model');
39
+ return $config->save(array(self::STATS_DATE_OF_CREATION => time()));
40
+ }
41
+
42
+ /**
43
+ * Check if cache life time is out of date
44
+ * @return boolean
45
+ */
46
+ protected function is_data_out_of_date() {
47
+ $config = WYSIJA::get('config', 'model');
48
+ $date_of_creation = $config->getValue(self::STATS_DATE_OF_CREATION);
49
+ return (time() - $date_of_creation >= self::STATS_DATA_LIFE_TIME);
50
+ }
51
+
52
+ /**
53
+ * Get the list of tables which are out of date, based on create_time
54
+ * @return type
55
+ */
56
+ protected function get_out_of_date_tables() {
57
+ $query = '
58
+ SELECT
59
+ TABLE_NAME as table_name
60
+ FROM
61
+ INFORMATION_SCHEMA.TABLES
62
+ WHERE
63
+ TABLE_SCHEMA IN (SELECT DATABASE())
64
+ AND TABLE_NAME LIKE "[wysija]'.self::STATS_PREFIX.'%"
65
+ AND TIMESTAMPDIFF(SECOND,CREATE_TIME, NOW()) >= '.self::STATS_TABLE_LIFE_TIME.';
66
+ ';
67
+ return $this->get_results($query);
68
+ }
69
+
70
+ /**
71
+ * Auto cleanup out-of-date tables
72
+ */
73
+ protected function clean_up_out_of_date_tables() {
74
+ $tables = $this->get_out_of_date_tables();
75
+ if (!empty($tables) && is_array($tables)) {
76
+ $_temp = array();
77
+ foreach ($tables as $table)
78
+ if (!empty($table['table_name']))
79
+ $_temp[] = $table['table_name'];
80
+ }
81
+ if (!empty($_temp)) {
82
+ $query = 'DROP TABLE IF EXISTS `'. implode('`,`', $_temp).'`';
83
+ $this->get_results($query);
84
+ }
85
+ }
86
+
87
+ /**
88
+ * Check if a table exists
89
+ * @param string $table_name table name
90
+ * @return boolean
91
+ */
92
+ protected function does_table_exists($table_name = null) {
93
+ if (empty($table_name) OR !is_string($table_name))
94
+ return false;
95
+ $query = "SHOW TABLES LIKE '$table_name'";
96
+ $result = $this->get_results($query);
97
+ return !empty($result) ? true : false;
98
+ }
99
+
100
+ /**
101
+ * Generate a table name, based on input params
102
+ * @param type $params
103
+ */
104
+ protected function get_table_name($params) {
105
+ $hash = $this->get_hash($params);
106
+ return '[wysija]' . self::STATS_PREFIX . $hash;
107
+ }
108
+
109
+ /**
110
+ * Generate a cached table
111
+ * @param type $table_name
112
+ * @param array $queries_create_table query to create a new cached table
113
+ * @param array $queries_insert_data query to collect and insert data to the newly created/truncated table
114
+ * @return boolean
115
+ */
116
+ protected function generate_table($table_name, Array $queries_create_table, Array $queries_insert_data) {
117
+ $is_out_of_date = $this->is_data_out_of_date();
118
+ $does_table_exists = $this->does_table_exists($table_name);
119
+
120
+ if (!$is_out_of_date && $does_table_exists)
121
+ return true;
122
+ if ($does_table_exists) {
123
+ $this->query('TRUNCATE TABLE `' . $table_name . '`');
124
+ } else {
125
+ foreach ($queries_create_table as $query_create_table)
126
+ $this->query($query_create_table);
127
+ }
128
+
129
+ foreach ($queries_insert_data as $query_insert_data)
130
+ $this->query($query_insert_data);
131
+
132
+ if ($is_out_of_date) {
133
+ $this->set_date_of_creation();
134
+ }
135
+
136
+ return true;
137
+ }
138
+
139
+ }
core/module/statisticschart.php ADDED
@@ -0,0 +1,11 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ abstract class WYSIJA_module_statisticschart extends WYSIJA_module_statistics{
5
+ public function __construct() {
6
+ parent::__construct();
7
+ $this->data['js_date_format'] = 'yy/mm/dd';
8
+ }
9
+ public function init(){
10
+ }
11
+ }
core/module/statisticstable.php ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ abstract class WYSIJA_module_statisticstable extends WYSIJA_module_statistics{
5
+ }
core/view.php ADDED
@@ -0,0 +1,292 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_view extends WYSIJA_object{
4
+
5
+ var $title="DEFAULT TITLE";
6
+ var $icon="icon-edit";
7
+ var $links=array();
8
+ var $search=array();
9
+ var $cols_nks=array();//correspondance between user_id and user-id once processed
10
+
11
+ static $color_coordinates = array();
12
+
13
+ static $cache_color_schemes = array();
14
+
15
+ /**
16
+ * Color schemes of MailPoet
17
+ * @var array
18
+ */
19
+ var $color_schemes = array(
20
+ array('#21759b', '#6697BF', '#487192', '#284c69', '#000333'),// like blue
21
+ array('#388e71', '#2c6f58','#16523d','#0d3c2c','#092e21'),// like green
22
+ array('#c97575','#a76262', '#854f4f', '#743a3a', '#5a2a2a'),// like red
23
+ array('#00CC00', '#269926', '#008500', '#39E639', '#67E667'), // http://colorschemedesigner.com/#2P11Tw0w0w0w0 // green
24
+ array('#FF0000', '#BF3030', '#A60000', '#FF4040', '#FF7373') // http://colorschemedesigner.com/#0011Tw0w0w0w0 // red
25
+ );
26
+
27
+ /**
28
+ * Default font family
29
+ * @var type
30
+ */
31
+ var $font_family = array('Arial');
32
+
33
+ /**
34
+ * Default font size in pixel
35
+ * @var int
36
+ */
37
+ var $font_size = 12;
38
+
39
+
40
+ function WYSIJA_view(){
41
+
42
+ }
43
+
44
+
45
+ /**
46
+ * Swap color schemes, to make sure, we don't use 2 colors in a same scheme continously
47
+ * @return array()
48
+ */
49
+ protected function swap_color_schemes() {
50
+ if (empty(self::$cache_color_schemes)) {
51
+ // swap colors from axis X => Y, Y => X
52
+ $x = count($this->color_schemes[0][0]);
53
+ $y = count($this->color_schemes[0]);
54
+ $tmp = array();
55
+ foreach ($this->color_schemes as $y => $colors) {
56
+ foreach ($colors as $x => $color) {
57
+ $tmp[$x][$y] = $color;
58
+ }
59
+ }
60
+ self::$cache_color_schemes = $tmp;
61
+ }
62
+ return self::$cache_color_schemes;
63
+ }
64
+ /**
65
+ * Get one random color from schemes
66
+ * @return type
67
+ *
68
+ * Depreciated
69
+ */
70
+ function get_random_color() {
71
+ return $this->get_next_color();
72
+ }
73
+
74
+ public function get_next_color() {
75
+
76
+ $color_schemes = $this->swap_color_schemes();
77
+
78
+ $class_name = get_class($this);
79
+ if (empty(self::$color_coordinates[$class_name]))
80
+ self::$color_coordinates[$class_name] = array('x'=> 0, 'y' => 0);
81
+
82
+ $current_color = $color_schemes[self::$color_coordinates[$class_name]['x']][self::$color_coordinates[$class_name]['y']];
83
+
84
+ // find out and set a next color
85
+ $flag = false;
86
+ $detected_new_color = false;
87
+ foreach ($color_schemes as $x => $colors) {
88
+ if ($detected_new_color) break;
89
+ foreach ($colors as $y => $color) {
90
+ if ($flag) {
91
+ self::$color_coordinates[$class_name]['x'] = $x;
92
+ self::$color_coordinates[$class_name]['y'] = $y;
93
+ $detected_new_color = true;
94
+ break;
95
+ }
96
+ if ($x == self::$color_coordinates[$class_name]['x'] && $y == self::$color_coordinates[$class_name]['y'])
97
+ $flag = true;
98
+
99
+ }
100
+ }
101
+ if (!$detected_new_color) {
102
+ self::$color_coordinates[$class_name]['x'] = 0;
103
+ self::$color_coordinates[$class_name]['y'] = 0;
104
+ }
105
+ return $current_color;
106
+ }
107
+
108
+ /**
109
+ * Get all colors from schemes
110
+ * @return type
111
+ */
112
+ function get_all_colors() {
113
+ $color_schemes = $this->swap_color_schemes();// this one, to make sure, we don't put same color tone continously
114
+
115
+ $tmp = array();
116
+ foreach ($color_schemes as $colors) {
117
+ $tmp = array_merge($tmp, $colors);
118
+ }
119
+ return $tmp;
120
+ }
121
+
122
+
123
+
124
+ function renderErrorInstall(){
125
+ $this->title=__("Your server's configuration doesn't allow us to complete MailPoet's Installation!",WYSIJA);
126
+ $this->header();
127
+ $this->footer();
128
+ }
129
+
130
+ /**
131
+ *
132
+ * @param type $type
133
+ * @param type $data
134
+ * @param bool $is_module is rendering a module view
135
+ */
136
+ function render($type,$data, $is_module = false){
137
+ $this->action=$type;
138
+ if (!$is_module){
139
+ $this->header($data);
140
+ }
141
+ if($type !== NULL) {
142
+ $this->$type($data);
143
+ }
144
+ if (!$is_module){
145
+ $this->footer();
146
+ }
147
+ }
148
+
149
+ /**
150
+ * display all the messages that have queued
151
+ * @global type $wysija_msg
152
+ */
153
+ function messages($noglobal=false){
154
+ $wysija_msg = $this->getMsgs();
155
+
156
+ if(isset($wysija_msg['g-updated'])) {
157
+ if(!$noglobal) {
158
+ if(isset($wysija_msg['updated'])) $wysija_msg['updated']=array_merge((array)$wysija_msg['updated'], $wysija_msg['g-updated']);
159
+ else $wysija_msg['updated']= $wysija_msg['g-updated'];
160
+ }
161
+ unset($wysija_msg['g-updated']);
162
+ }
163
+ if(isset($wysija_msg['g-error'])) {
164
+ if(!$noglobal){
165
+ if(isset($wysija_msg['error'])) $wysija_msg['error']=array_merge((array)$wysija_msg['error'], $wysija_msg['g-error']);
166
+ else $wysija_msg['error']= $wysija_msg['g-error'];
167
+ }
168
+ unset($wysija_msg['g-error']);
169
+ }
170
+ $wpnonce='<input type="hidden" value="'.wp_create_nonce("wysija_ajax").'" id="wysijax" />';
171
+ if(!$wysija_msg) return '<div class="wysija-msg ajax"></div>'.$wpnonce;
172
+ $html='<div class="wysija-msg">';
173
+ foreach($wysija_msg as $level =>$messages){
174
+ $msg_class = '';
175
+ switch($level){
176
+ case 'updated':
177
+ $msg_class = 'notice-msg updated';
178
+ break;
179
+ case 'error':
180
+ $msg_class = 'error-msg error';
181
+ break;
182
+ case 'xdetailed-updated':
183
+ $msg_class = 'xdetailed-updated';
184
+ break;
185
+ case 'xdetailed-errors':
186
+ $msg_class = 'xdetailed-errors';
187
+ break;
188
+
189
+ }
190
+
191
+ $html.='<div class="'.$msg_class.'">';
192
+ $html.='<ul>';
193
+
194
+ if(count($messages)>0){
195
+ foreach($messages as $msg){
196
+ // check type of msg variable
197
+ if(is_array($msg)) {
198
+ $msg = var_export($msg, true);
199
+ }
200
+
201
+ // display message
202
+ $html.='<li>'.$msg.'</li>';
203
+ }
204
+ }
205
+
206
+
207
+ $html.='</ul>';
208
+ $html.='</div>';
209
+ }
210
+ $html.='</div><div class="wysija-msg ajax"></div>'.$wpnonce;
211
+
212
+ return $html;
213
+ }
214
+
215
+ /**
216
+ * this function let us generate a nonce which is an encrypted unique word based n the user info and some other stuff.
217
+ * by default it will create an hidden input nonce field
218
+ * @param type $params
219
+ * @param type $get
220
+ * @return type
221
+ */
222
+ static function secure($params=array(),$get=false,$echo=true){
223
+ $controller='';
224
+ if(!is_array($params)) $action=$params;
225
+ else{
226
+ $action=$params['action'];
227
+ if(isset($params['controller'])) $controller=$params['controller'];
228
+ elseif(isset($_REQUEST['page'])) $controller=$_REQUEST['page'];
229
+ }
230
+ $nonceaction=$controller.'-action_'.$action;
231
+
232
+ if(is_array($params) && isset($params['id']) && $params['id']) $nonceaction.='-id_'.$params['id'];
233
+
234
+ if($get){
235
+ return wp_create_nonce($nonceaction);
236
+ }else{
237
+ return wp_nonce_field($nonceaction,'_wpnonce',true,$echo);
238
+ }
239
+
240
+ }
241
+
242
+ /**
243
+ * this allows us to get a field class to be validated by when making a form field
244
+ * @param type $params
245
+ * @param string $prefixclass
246
+ * @return string
247
+ */
248
+ function getClassValidate($params,$returnAttr=false,$prefixclass=""){
249
+ $class_validate = '';
250
+ $recognised_types = array('email','url');
251
+
252
+ if(isset($params['req'])){
253
+ $class_validate='required';
254
+ if(isset($params['type']) && in_array($params['type'], $recognised_types)) {
255
+ $class_validate.=',custom['.$params['type'].']';
256
+ }
257
+ }else{
258
+ if(isset($params['type']) && in_array($params['type'],$recognised_types )) {
259
+ $class_validate.='custom['.$params['type'].']';
260
+ }
261
+ }
262
+
263
+ if($prefixclass) $prefixclass.=' ';
264
+ if($class_validate) $class_validate='validate['.$class_validate.']';
265
+ if(!$returnAttr && $class_validate) $class_validate= ' class="'.$prefixclass.$class_validate.'" ';
266
+
267
+ return $class_validate;
268
+ }
269
+
270
+ /**
271
+ * central function to return a translated formated date
272
+ * @param type $val
273
+ * @param type $format
274
+ * @return string
275
+ */
276
+ function fieldListHTML_created_at($val,$format=''){
277
+ if(!$val) return '---';
278
+
279
+ //offset the time to the time of the WP site not the server
280
+ $hToolbox = WYSIJA::get('toolbox','helper');
281
+ // get current time taking timezone into account.
282
+
283
+ $val = $hToolbox->servertime_to_localtime($val);
284
+
285
+ if($format) return date_i18n($format,$val);
286
+ else return date_i18n(get_option('date_format'),$val);
287
+ }
288
+
289
+ function fieldListHTML_created_at_time($val){
290
+ return $this->fieldListHTML_created_at($val,get_option('date_format').', '.get_option('time_format'));
291
+ }
292
+ }
css/add-ons.css ADDED
@@ -0,0 +1 @@
 
1
+ .module-container{overflow:hidden;width:100%;min-width:676px;margin:20px 0px}.module-container p,.module-container h2,.module-container h3{margin-left:6px}.mailpoet-module{position:relative;float:left;width:320px;min-height:330px;margin:0px 16px 18px 0px;border:1px solid #e5e5e5;background:#fff;padding:0px 0 14px 0px;box-shadow:0 1px 1px rgba(0,0,0,0.04);overflow:hidden}.mailpoet-module.not-ready{border-color:#eee;cursor:pointer}.mailpoet-module .mailpoet-module-image{margin:0px 12px}.mailpoet-module .mailpoet-module-content p{margin:0px 0px 8px 0px}.mailpoet-module .mailpoet-module-content p span.mailpoet-module-badge{background-color:#df1d4f;color:#fff;font-weight:bold;padding:6px 10px;margin-right:4px;border:1px solid #cc1148;border-radius:0px 0px 3px 3px}.mailpoet-module .mailpoet-module-content p span.mailpoet-module-badge a{color:#fff !important;text-decoration:none;border-bottom:1px dotted #fff}.mailpoet-module .mailpoet-module-content p span.mailpoet-module-badge.new{float:right !important;margin-top:-6px;background-color:#054e6e !important;border:1px solid #054e6e !important}.mailpoet-module .mailpoet-module-actions{display:block;position:absolute;bottom:6px;left:10px}.mailpoet-module .mailpoet-module-actions a{position:relative;margin-bottom:4px}.mailpoet-module-content{margin:0 1em}.mailpoet-module h3{line-height:1.1;margin:6px 12px;padding:3px 0px;font-size:14px;font-weight:bold;font-style:normal;color:#000}.mailpoet-module-description p{line-height:150%;font-size:12px;margin-bottom:20px;color:#666}.submit-idea{clear:both;display:block;padding:8px 15px;margin:8px 4px;overflow:hidden;line-height:180%}
css/admin-campaigns-articles.css ADDED
@@ -0,0 +1 @@
 
1
+ #results{min-height:430px;padding:0 0 50px 0;width:98%}#results .loading{margin:10px 0 0 0}#results li{border:1px solid #ddd;height:38px;margin:0 0 10px 0;padding:0;background-color:#FFF;position:relative}#results li .checkbox_container{width:38px;height:38px;text-align:center;line-height:35px;float:left;position:relative}#results li .checkbox{cursor:pointer;margin:0 0 0 11px}#results .thumbnail{width:60px;height:38px;overflow:hidden;float:left;margin:0 10px 0 0}#results li label{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;user-select:none;margin:0 0 0 0;line-height:38px;height:38px;overflow:hidden;float:left;width:510px}#results li label.with_thumbnail{width:440px}#results li.selected{border-color:#0074A2;cursor:pointer}#results li.selected label{color:#0074A2}#results hr{border:0 none;border-top:1px solid #dbdbdb;height:1px;line-height:1px}#basic .filters-box{float:left;margin:0}.filters-box #post_type{width:120px}.filters-box .post_category{width:160px}.filters-box #post_status{width:140px}#basic .search-box{float:right;margin:0}input#search{width:160px}#search-submit{font-size:13px;margin:0 14px 0 0;padding:1px 10px}#toggle-advanced{position:relative;right:10px;top:0}#advanced{padding:0 0 50px 0;display:none}#advanced div.block{padding:0.8em 0 0.8em 1em;margin:0}#advanced div.block p{margin:0 0 5px 0;padding:0}#advanced label{float:left;width:200px;padding:0}#advanced div.block p label{float:none}#image_width_slider{float:left;width:252px;margin:15px 0 0 5px;height:0px;position:relative;cursor:pointer;border:2px solid #808080;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px}#slider_handle{width:16px;height:16px;cursor:pointer;position:absolute;top:-9px;margin:0 0 0 -2px;-webkit-border-radius:10px;-moz-border-radius:10px;border-radius:10px;background-color:#fff;box-shadow:inset 0px 0px 2px rgba(0,0,0,0.1);border:1px solid #ccc}#slider_info{float:left;font-weight:bold;margin:7px}#advanced label span.label{float:none;display:block;font-size:10px;font-style:italic;line-height:1.2em;width:120px}#advanced label.radio{float:left;width:auto;margin:0 9px 0 0}#advanced label.radio input{margin:0 2px 0 0}#advanced div.group{float:left}.cpt-type{background-color:#F1F1F1;border-left:1px solid #ccc;bottom:0;color:#21759B;font-style:italic;padding:0 10px;position:absolute;right:0;top:0;line-height:38px}.cpt-type small{color:#888}.modcoder_excolor_clrbox{width:17px;height:17px;margin:4px 0 0 0;color:transparent;padding:0}
css/admin-campaigns-autopost.css ADDED
@@ -0,0 +1 @@
 
1
+ .autopost{padding:0 10px 10px 0;display:block}.autopost div.block{padding:0.8em 0 0.8em 1em;margin:0}.autopost div.block p{margin:0 0 5px 0;padding:0}.autopost label{float:left;width:200px;padding:0}.autopost div.block p label{float:none}.autopost label span.label{float:none;display:block;font-size:10px;font-style:italic;line-height:1.2em;width:120px}.autopost label.radio{float:left;width:auto;margin:0 9px 0 0}.autopost label.radio input{margin:0 2px 0 0}.autopost div.group{float:left}#image_width_slider{float:left;width:252px;margin:15px 0 0 5px;height:0px;position:relative;cursor:pointer;border:2px solid #808080;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px}#slider_handle{width:16px;height:16px;cursor:pointer;position:absolute;top:-9px;margin:0 0 0 -2px;-webkit-border-radius:10px;-moz-border-radius:10px;border-radius:10px;background-color:#fff;box-shadow:inset 0px 0px 2px rgba(0,0,0,0.1);border:1px solid #ccc}#slider_info{float:left;font-weight:bold;margin:7px}select#category_ids{width:400px}#toggle-advanced{margin:0 0 0 12px}#advanced{display:none}.modcoder_excolor_clrbox{width:17px;height:17px;margin:4px 0 0 0;color:transparent;padding:0}
css/admin-campaigns-bookmarks.css ADDED
@@ -0,0 +1 @@
 
1
+ .bookmarks .networks li{margin:0 0 10px 0}.bookmarks .networks label{width:70px;float:left;font-weight:normal}.bookmarks .networks input{width:350px;float:left}.bookmarks .sizes{margin:20px 0 0 0}.bookmarks .sizes a{margin:0 0 0 10px}.bookmarks .sizes a.selected{font-weight:bold}.bookmarks .sizes .small{font-size:12px}.bookmarks .sizes .medium{font-size:14px}.bookmarks .sizes .large{font-size:16px}.bookmarks .icons{height:230px;overflow:auto}.bookmarks .icons li{margin:10px}.bookmarks .icons li a{padding:5px;margin:0;display:-moz-inline-box;display:inline-block;*display:inline;*float:left}.bookmarks .icons li a.selected{border:2px solid #DD561A;padding:3px}.bookmarks .icons img{margin:0 5px;vertical-align:middle}
css/admin-campaigns-dividers.css ADDED
@@ -0,0 +1 @@
 
1
+ .wysija_popup_content.divider{width:620px}.dividers ul{height:350px;width:620px;overflow:auto;margin:0 0 20px 0}.dividers li{text-align:center;padding:5px 2px}.dividers li a{padding:2px;display:block}.dividers li a.selected{padding:0;border:2px solid #DD561A}.dividers li img{vertical-align:middle}
css/admin-campaigns-editDetails.css ADDED
@@ -0,0 +1 @@
 
1
+ #datepicker-day{margin-left:10px;min-width:90px}
css/admin-campaigns-editTemplate.css ADDED
@@ -0,0 +1 @@
 
1
+ #TB_window.default,#TB_window.default #TB_iframeContent{width:750px !important;height:550px !important}#TB_window.addlink,#TB_window.addlink #TB_iframeContent{width:350px !important;height:205px !important}#TB_window.themes,#TB_window.themes #TB_iframeContent{height:535px !important;overflow:hidden;width:750px !important}#TB_window.dividers,#TB_window.dividers #TB_iframeContent{height:470px !important;width:665px !important}#TB_window.bookmarks,#TB_window.bookmarks #TB_iframeContent{height:475px !important;width:670px !important}#TB_window p.notice{font-style:italic;font-size:11px}.wysija-msg,.updated,.error{display:none}.wysija-msg.ajax,.wysija-msg.ajax .updated,.wysija-msg.ajax .error,.ui-dialog-content .error,.ui-dialog-content .updated{display:block}.sendtestemail{background-color:#EAEAEA;border-color:#BFBFBF;border-radius:14px 14px 14px 14px;border-style:dashed;border-width:2px;height:32px;margin:30px 0 10px;padding-left:10px;padding-top:9px;width:300px}.ui-widget-overlay{z-index:9998 !important;background-image:none !important}.wj_themes span.default-wrap,.wj_themes span.delete-wrap,.wj_images span.delete-wrap{background-color:#DFDFDF;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);border:1px solid #CCCCCC;border-radius:2px 2px 2px 2px;box-shadow:0 1px 0 rgba(0,0,0,0.15),0 0 2px 1px #fff inset;height:20px;position:absolute;width:20px;z-index:9999;text-indent:-99999px;overflow:hidden}.wj_themes span.default-wrap{right:28px;top:2px}.wj_themes span.delete-wrap,.wj_images span.delete-wrap{right:2px;top:2px}.wj_themes span.default,.wj_themes span.delete,.wj_images span.delete{cursor:pointer;display:block;height:20px;text-indent:-9999px;width:20px;z-index:9999}.wj_themes span.delete,.wj_images span.delete{background:url("../img/controls/icons.png") no-repeat scroll -100px 0 transparent}.wj_themes span.default{background:url("../img/controls/icons.png") no-repeat scroll -240px 0 transparent}.wj_themes span.default:hover{background:url("../img/controls/icons.png") no-repeat scroll -240px -20px transparent}.wj_themes span.default.selected,.wj_themes span.default.selected:hover{background:url("../img/controls/icons.png") no-repeat scroll -260px -20px transparent}#screen-meta{z-index:10000}.marginl{margin-left:10px}
css/admin-campaigns-medias.css ADDED
@@ -0,0 +1 @@
 
1
+ .swfupload{vertical-align:middle}ul#sidemenu{bottom:-1px;float:none !important;font-weight:normal;left:0;margin:0 5px;overflow:hidden;float:none !important}.wysija-thumb{float:left;height:150px;margin:10px;width:150px;position:relative;border:2px dashed transparent;filter:alpha(opacity=60);-moz-opacity:0.6;opacity:0.6;margin:8px}.wysija-thumb:hover,.wysija-thumb.selected:hover{border-color:#000;border-style:dashed}.wysija-thumb.selected{border-color:#4a5b04;border-style:solid;filter:alpha(opacity=100);-moz-opacity:1;opacity:1}.wysija-thumb img{cursor:pointer;max-height:150px;max-width:150px}.wysija-thumb span.delete{background:url("../img/controls/icons.png") no-repeat scroll -100px 0 transparent;cursor:pointer;height:20px;text-indent:-9999px;display:block;width:20px}.wysija-thumb span.delete-wrap{background-color:#DFDFDF;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);border:1px solid #CCCCCC;border-radius:2px 2px 2px 2px;box-shadow:0 1px 0 rgba(0,0,0,0.15),0 0 2px 1px #fff inset;height:20px;position:absolute;right:2px;top:2px;margin:5px 3px 0 0;width:20px;display:none}.wysija-thumb span.thumb_url,.wysija-thumb span.url,.wysija-thumb span.width,.wysija-thumb span.height,.wysija-thumb span.identifier{display:none}.wysija-thumb:hover span.delete-wrap{display:block}.wysija-thumb span.delete:hover{background-position:-100px -20px}.clear{clear:both}.tablenav{margin-right:20px}#media-items{overflow:auto}.media-wp-upload #media-items{display:none}.media-browse #media-items{width:100%;height:340px;margin:0 0 10px 0}.media-wp-browse #media-items{width:100%;height:340px;margin:0 0 10px 0}#media-items em{margin:0 0 0 20px}#media-items,.ml-submit{float:left}.ml-submit{margin-top:15px;margin-left:15px}#loader{left:50%;position:relative;top:50%}#overlay{background-color:#FFFFFF;height:100%;opacity:0.8;position:absolute;width:100%;z-index:999;display:none}#flash-upload-ui{padding:10px 25px}.after-file-upload{padding:0 25px}.after-file-upload,.upload-flash-bypass,.howto{display:none}.wysija-thumb .slidetoggle,.wysija-thumb .toggle{display:none}.max-upload-size{margin:10px 0;display:block}
css/admin-campaigns-themes.css ADDED
@@ -0,0 +1 @@
 
1
+ .swfupload{vertical-align:middle}ul#sidemenu{bottom:-1px;float:none !important;font-weight:normal;left:0;margin:0 5px;overflow:hidden}#loader{left:50%;position:relative;top:50%}.loading{margin:15px 0 0 0}#overlay{background-color:#fff;opacity:0.8;width:100%;height:100%;position:fixed;top:0;bottom:0;left:0;right:0;z-index:999;display:none}#flash-upload-ui{padding:10px 25px}.after-file-upload{padding:0 25px}#close-pop-alt{float:right;margin-right:50px;margin-top:10px}.after-file-upload,.upload-flash-bypass,.howto{display:none}#theme-view .preview,#theme-view .infos{float:left}#theme-view .infos{width:300px;margin:0 0 0 20px}#theme-view .preview{background:url(../img/wpspin_light.gif) no-repeat scroll center center transparent}#theme-view .preview img{width:320px;display:block}#theme-view{font-size:14px;height:480px;overflow:auto}#theme-view h2{color:#555555;font-family:Tahoma;font-size:24px;margin:4px 0 0;line-height:1.2em}#theme-view .actions{margin-bottom:10px}div.star{background-color:transparent;border:medium none;height:100%;left:0;letter-spacing:1ex;position:absolute;top:0}.stars{background:url(../img/stars.png) repeat-x scroll left bottom transparent;height:17px;position:relative;width:95px}.stars .star-rating{background:url(../img/stars.png) repeat-x scroll left top transparent;float:left;height:17px;overflow:hidden;text-indent:100%;white-space:nowrap}.my-rating a{height:17px;width:19px;float:left;margin:0;padding:0}.my-rating a.active,.my-rating a:hover,.my-rating a.on{background:url(../img/stars.png) no-repeat left top transparent}.my-rating a.active.off,.my-rating a.off{background:none}div.paragraph{margin:10px 0}.star-holder.my-rate img{cursor:pointer}.star-holder.my-rate div.star:hover{background-color:#DD0000}div.star.select.star-rating{background-color:#FFCC00}#theme-upload{display:none}.avatar{border:2px solid #CCC;float:right;margin:0}#overwrite{margin:0px 10px 0 0}#filter-selection{position:relative;top:3px}.theme-screenshot{padding:0;width:320px;border:1px solid #ccc;position:relative;background-color:#fff;-webkit-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);-moz-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);box-shadow:0 0 5px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1)}.theme-screenshot:before,.theme-screenshot:after{position:absolute;width:40%;height:10px;content:' ';left:12px;bottom:12px;background:transparent;-webkit-transform:skew(-5deg) rotate(-5deg);-moz-transform:skew(-5deg) rotate(-5deg);-ms-transform:skew(-5deg) rotate(-5deg);-o-transform:skew(-5deg) rotate(-5deg);transform:skew(-5deg) rotate(-5deg);-webkit-box-shadow:0 6px 12px rgba(0,0,0,0.3);-moz-box-shadow:0 6px 12px rgba(0,0,0,0.3);box-shadow:0 6px 12px rgba(0,0,0,0.3);z-index:-1}.theme-screenshot:after{left:auto;right:12px;-webkit-transform:skew(5deg) rotate(5deg);-moz-transform:skew(5deg) rotate(5deg);-ms-transform:skew(5deg) rotate(5deg);-o-transform:skew(5deg) rotate(5deg);transform:skew(5deg) rotate(5deg)}.theme-screenshot img{width:320px}#wj_paginator{float:right;margin-right:10px;font-size:14px}#wj_paginator a{margin:0 1px}#wj_paginator span{font-weight:bold;margin:0 1px}#wj_paginator a.selected{color:#000;text-decoration:none;cursor:default}#themes-list{height:405px;margin:0}.theme{float:left;margin:0px;overflow:hidden;width:152px;margin:25px 8px 0 0}.theme.last{margin-right:0}.theme .thumbnail{height:170px;width:150px;position:relative;border:1px solid #999999;background-color:#000000}.theme img{width:150px;height:170px;position:absolute;z-index:1;top:0px;left:0px}.theme .install{display:none;float:none !important}.theme:hover img{filter:alpha(opacity=25);-moz-opacity:0.25;opacity:0.25}.theme:hover .thumbnail{border:1px solid #2a749f}.theme:hover .install{display:block;position:relative;z-index:100;text-align:center;margin:0px auto;margin-top:80px;width:80px !important}.theme .infos{display:block;width:100%;margin:5px 0 0 0;padding:0;text-align:left;font-size:12px;line-height:1em;color:#999999;text-decoration:none}.theme .infos:hover{color:#21759b;text-decoration:underline}
css/admin-campaigns-viewstats.css ADDED
@@ -0,0 +1 @@
 
1
+ #posts-filter{margin-top:40px}#hook_newsletter_top{margin-top:20px}#hook_newsletter_top .left{float:left;width:50%}#hook_newsletter_top .right{float:right}#hook_newsletter_top .actions{width:45%;margin-bottom:20px;padding-left:10px}.action_buttons a:first-child{margin-left:0}.action_buttons{display:block}table.newsletter-stats-block{margin-top:10px}table.newsletter-stats-block td.label{width:25%;font-weight:bold}.wrap .button-secondary2{top:auto !important}.googletrackingcode .action_buttons{padding-left:0}.googletrackingcode{padding-top:10px}.wrap .button-secondary2:first-child{margin-left:0}
css/admin-campaigns-welcome_new.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-app .welcome_poll li{list-style:none outside none;margin:10px 0;font-size:14px;line-height:20px}#wysija-app .welcome_poll li input{margin:0 10px 0 0}#how_did_you_find_us_4_url{width:50%}#poll_result{font-weight:bold;padding-left:5px}#poll_result span.checkmark{background:url("../img/controls/icons.png") no-repeat scroll -260px -20px transparent;display:inline-block;text-indent:-9999px;width:20px}
css/admin-campaigns.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-app{padding-top:10px}.icon32{margin-top:0px !important}#posts-filter{clear:both}#wysija-app h2{display:inline}#icon-edit-news{background:url(../img/mail-big.png) no-repeat scroll 0 5px transparent}#TB_window{display:block}.moredetails{display:none}.sending,.scheduled{background-color:#d5f0e6}.autonl.sending{background-color:#D7E6F2}#wysija-campaign .listmissing{background-color:#EFD5D5}#preview-receiver{width:230px;vertical-align:middle;margin:0 5px 0 0}.ml-submit{margin:15px 15px 0 15px}.automatic-nl{display:none}#titlediv .titlebox{font-size:1.7em;background-color:#FFFFFF;border-color:#CCCCCC;line-height:100%;outline:medium none;padding:3px 8px;width:100%}label .description{cursor:auto}.about-wrap .feature-section.three-col div{float:left;width:28%}#socials-block div{float:none;margin:5px 5px 2px 0;width:100%}.progress_bar{padding-bottom:10px}.progress_bar .bar{display:block;width:100%;height:22px;float:left;margin-right:10px;background:#eef6fc;border:1px solid #bbbbbb;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;line-height:22px;position:relative}.progress_bar .progress{display:block;background:#69b1e9;height:100%}.progress_bar .percent{height:100%;position:absolute;top:0px;left:10px}#wysija-badge{position:absolute;top:0;right:0;color:#fff;text-shadow:0 1px 0 rgba(0,0,0,0.3);padding-top:91px;height:50px;width:173px;font-weight:bold;font-size:14px;text-align:center;margin:0 -5px;background:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/wysija-badge.png") no-repeat}#review-follow{position:relative;height:177px}#review-follow div{float:left;top:0}#review-follow .review-left{background:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/save-the-kitten.jpg") no-repeat;height:175px;background-color:#f8f8f8;border-top:1px solid #efefef;border-bottom:1px solid #efefef;padding-right:10px}#review-follow h4{margin:0 !important}#review-follow .review-left.small{width:450px}#review-follow .review-left.medium{width:490px}#review-follow .review-left.large{width:540px}#review-follow .small .description{margin-top:24px}#review-follow .medium .description{margin-top:20px}#review-follow .large .description{margin-top:18px}#review-follow .review-left p{margin-left:20px}#review-follow .review-right{background:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow_middle.jpg") no-repeat scroll -30px 0 #f8f8f8;border-bottom:1px solid #EFEFEF;border-top:1px solid #EFEFEF;height:175px;width:24px}#review-follow #mailpoet-subscribe{margin-left:0px;margin-top:10px}#review-follow .subscribe-middle{width:220px;height:175px;background-color:#f8f8f8;border-top:1px solid #efefef;border-bottom:1px solid #efefef}#review-follow .socials{margin-left:30px}#review-follow .follow-left{width:220px;height:175px;background-color:#f8f8f8;border-top:1px solid #efefef;border-bottom:1px solid #efefef}#review-follow .follow-right{background:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow_right.jpg") no-repeat;width:100px;height:175px}#review-follow .follow-right38{background:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow-right-38.jpg") no-repeat;width:100px;height:175px}#review-follow .review-left .description{margin-left:146px}#review-follow .link-cat-review{height:140px;left:20px;position:absolute;top:21px;width:120px;outline:medium none}a.ctaupdate,span.ctaupdate{font-size:12px}.pds-links,.pds-totalvotes-outer{display:none !important}#polldaddy_embed_0{width:680px;margin-left:-30px}span.local_time{margin-left:10px}#hook_newsletter_top .hook-column{width:50%}.card-title{font-weight:bold}.report-card{padding-top:10px}.googletrackingcode .action_buttons{padding-left:44px;padding-top:5px}.googletrackingcode span{font-weight:bold}.mpoet-page h1{font-size:34px}.mpoet-page h1,.mpoet-page h2,.mpoet-page h3,.mpoet-page h4{color:#626262;font-weight:bold}.mpoet-page h1 span.version{color:#5e869b}
css/admin-config-form_widget_settings.css ADDED
@@ -0,0 +1 @@
 
1
+ .form_widget_settings p label{float:left;clear:left;width:200px}#widget-settings-error{margin-right:10px;padding:10px;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;color:#000;font-weight:normal;background-color:#ffebe8;border:1px solid #c00}.sortable li{margin:0;padding:5px}.sortable li:hover{background-color:#eee}.icon span{display:block;height:20px;width:20px}.handle span{cursor:move;background:url(../img/controls/icons.png) no-repeat -120px 0}.handle a.active span,.handle:hover span{background:url(../img/controls/icons.png) no-repeat -120px -20px;cursor:move}.add span{background:url(../img/controls/icons.png) no-repeat -200px 0;float:left}.add.active span,.add:hover span{background:url(../img/controls/icons.png) no-repeat -200px -20px}.remove span{background:url(../img/controls/icons.png) no-repeat -100px 0}.remove.active span,.remove:hover span{background:url(../img/controls/icons.png) no-repeat -100px -20px}#widget-field-settings{margin:0 0 0 10px;display:none}.widget_text textarea,.widget_html textarea{height:120px;width:450px;resize:none}.widget_html input.checkbox,label input[type="checkbox"],label input[type="radio"]{margin:0px 5px 0 0px}.widget_list #lists-available{float:left;padding:0;width:250px}.form_widget_settings .icon.add{line-height:20px;outline:0}.form_widget_settings .selection{background-color:#fff;border:1px solid #dfdfdf;width:100%}.form_widget_settings .selection input[name="is_checked"],.form_widget_settings .selection input.is_checked{margin:0px 0px 2px 3px}.form_widget_settings .selection li{padding:5px 3px;margin:0}.form_widget_settings .selection input{margin:0 5px 0px 0}.form_widget_settings .selection .value{margin:0}.form_widget_settings .selection .is_selected{float:left;margin:3px 0 0 6px !important}.form_widget_settings .selection label{float:left;margin:0 0 0 5px;width:220px;height:20px;line-height:20px;overflow:hidden}.form_widget_settings .selection .handle,.form_widget_settings .selection .remove{float:right;margin:3px 3px 0 0}.form_widget_settings #selection-template{display:none}.form_widget_settings .hidden{display:none}.form_widget_settings label.radio{margin:0 10px 0 0}#date-format-value{display:none}
css/admin-config.css ADDED
@@ -0,0 +1 @@
 
1
+ tr.title_row>td{padding-left:0}#wysija-tabs a{font-size:14px}#wysija-tabs a:focus{outline:0 none}#wysija-settings .premium_activated{color:#0074A2;font-size:1.2em;padding:15px 0 20px 0}#wysija-settings .premium_activated i{font-size:1.5em;margin:0 6px 3px 0;vertical-align:bottom}p.submit{padding:1em 1.4em 1em 0}span.title{font-size:16px;margin:0;color:#000}.mailpoet-frequency_inner_texting{padding:0 10px}.mailpoet-frequency_warning{margin-top:10px}#wysija-settings input[type="checkbox"],#wysija-settings input[type="radio"]{margin-right:6px;vertical-align:baseline}#wysija-settings label input[type="checkbox"],#cron_page_hit_trigger-1{margin-left:0}#wysija-settings .button-secondary{margin-left:6px}#wysija-settings .form-table th{width:300px}#wysija-settings .form-table th.row{border:1px solid #AAAAAA}#warning-msg-frequency{background-color:#FFFFE0;border-color:#E6DB55}#warning-msg-frequency{border-radius:3px 3px 3px 3px;border-style:solid;border-width:1px;margin:5px 15px 2px;padding:0 0.6em}#warning-msg-frequency ul{margin:0}#wysija-settings .form-table th.rolestitle{padding:10px 0 10px 10px;text-align:left;vertical-align:middle;width:220px}#wysija-settings .form-table th.rolestable{padding:3px;text-align:center;vertical-align:middle;width:110px;word-wrap:break-word}#wysija-settings .form-table td.rolestable{padding:6px 0;text-align:center}#wysija-settings .form-table td.title{padding:10px 0 10px 10px;text-align:left;vertical-align:middle}#wysija-settings .form-table td.title p{margin:0}#wysija-settings .form-table td{vertical-align:top}#wysija-settings #bounce .intro{margin:10px 0 20px 10px}#wysija-settings #bounce #bounce-process-auto,#wysija-settings #bounce #bounce-connector{margin-left:0 !important}#wysija-settings #sendingmethod .methods{border-bottom:1px solid #000}#wysija-settings #sendingmethod .choice-sending-method-site{border-top:1px solid #cccccc}#wysija-settings #sendingmethod .choice-sending-method-site th,#wysija-settings #sendingmethod .choice-sending-method-site td{padding-top:18px}label.dkim{margin-right:20px}.cronleft,.cronright{float:left}.cronleft{width:20px;margin-right:10px}#cron_manual_linkname p{margin:0}.labelcheck{float:left;margin-right:15px}.checkbox_optin_label{margin:6px 6px 0 0;float:left}.checkbox_optin_value{float:left;width:500px}#smtp-rest-label{display:none;margin-left:10px}#wysija-settings #multisite #form-ms-config th{width:455px}#wysija-settings #multisite #ms-sendingmethod th{width:300px}#wysija-settings #multisite .intro{margin:0 0 0 10px}#wysija-settings #sendingmethod .methods input,#wysija-settings #multisite .methods input{margin:0 5px 0 0;float:left}#wysija-settings #sendingmethod .methods h3,#wysija-settings #multisite .methods p.title{float:left;margin:0 !important}#wysija-settings #multisite .choice-one-for-all{display:none}#wysija-settings a.view_all{font-style:italic}#wysija-settings .cronright label{display:block;margin-bottom:10px}#wysija-settings .cronright label span{margin-left:36px}#wysija-settings .capabilities_form{width:400px}span.icon{display:block;height:20px;width:20px;cursor:pointer}.geeky-option a,.geeky-option span{float:left}.show-more-geeky-options span{background:url(../img/controls/icons.png) no-repeat -200px 0}.show-more-geeky-options:hover span{background:url(../img/controls/icons.png) no-repeat -200px -20px}.hide-geeky-options span{background:url(../img/controls/icons.png) no-repeat -220px 0}.hide-geeky-options:hover span{background:url(../img/controls/icons.png) no-repeat -220px -20px}@-webkit-keyframes opacity{0%{opacity:1}100%{opacity:0}}@-moz-keyframes opacity{0%{opacity:1}100%{opacity:0}}.wysija-button-loading{text-align:center;margin:100px 0 0 0}.wysija-button-loading span{-webkit-animation-name:opacity;-webkit-animation-duration:1s;-webkit-animation-iteration-count:infinite;-moz-animation-name:opacity;-moz-animation-duration:1s;-moz-animation-iteration-count:infinite}.wysija-button-loading span:nth-child(2){-webkit-animation-delay:100ms;-moz-animation-delay:100ms}.wysija-button-loading span:nth-child(3){-webkit-animation-delay:300ms;-moz-animation-delay:300ms}.links-page{display:none;margin-left:10px}.new_form{margin:16px 0}.new_form a{font-size:16px !important}
css/admin-dashicons.css ADDED
@@ -0,0 +1 @@
 
1
+ @font-face{font-family:'mailpoet-icon';src:url("../font/icon-email.eot?doyy92");src:url("../font/icon-email.eot?#iefixdoyy92") format("embedded-opentype"),url("../font/icon-email.woff?doyy92") format("woff"),url("../font/icon-email.ttf?doyy92") format("truetype"),url("../font/icon-email.svg?doyy92#mailpoet-icon") format("svg");font-weight:normal;font-style:normal}.mp-menu-icon-bg .menu-top.toplevel_page_wysija_campaigns .wp-menu-image{background:transparent url("../img/mail.png") no-repeat center center}.mp-menu-icon-bg .menu-top.toplevel_page_wysija_campaigns .wp-menu-image:before{content:""}.mp-menu-icon-font .menu-top.toplevel_page_wysija_campaigns .wp-menu-image:before{font:normal 18px/1 "mailpoet-icon" !important;speak:none;color:#999;padding:8px 0;height:36px;width:20px;display:inline-block;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;-moz-transition:all .1s ease-in-out;-webkit-transition:all .1s ease-in-out;transition:all .1s ease-in-out}.menu-top.toplevel_page_wysija_campaigns .wp-menu-image:before{content:"\f466"}
css/admin-global.css ADDED
@@ -0,0 +1 @@
 
1
+ .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{display:inline-table}* html .clearfix{height:1%}.clearfix{display:block}.hide{display:none}.wysija-version{margin:0 20px 0 165px;padding-bottom:45px}.wysija-footer h2{color:#21759B;cursor:pointer}.wysija-footer h2:hover{text-decoration:underline}.wysija-version{font-size:12px;color:#777777}.wysija-version .social-foot div{float:left;margin-right:10px}.wysija-version .version{float:left}.wysija-version .help{float:left;margin-left:20px}.wysija-version .social-foot .textsocial{margin-left:8px;margin-top:3px;padding-top:0}.wysija-version .social-foot{clear:both;display:inline-block}.wysija-version .version{float:left;margin-bottom:5px}#upperfoot{display:block;float:none}#submitexport label input[type="checkbox"]{margin-left:0px !important}.clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{display:inline-table}* html .clearfix{height:1%}.clearfix{display:block}.wysija-footer,.wysija-version{margin:0 20px 0 165px;padding-bottom:45px}.wysija-footer h2{color:#21759B;cursor:pointer}.wysija-footer h2:hover{text-decoration:underline}.wysija-version{font-size:12px;color:#777777;height:65px;border-top:1px solid #DFDFDF}.wysija-version .social-foot div{float:left;margin-right:10px}.social-foot .fb{width:80px}.wysija-version .version{float:left}.wysija-version .help{float:left;margin-left:20px}.wysija-version .social-foot .textsocial{margin-left:8px;margin-top:3px;padding-top:0}.wysija-version .social-foot{clear:both;display:inline-block;float:left;margin-top:2px;padding:6px 13px}.wysija-version .social-foot .socials div{margin-left:20px}.wysija-version .social-foot .socials{margin-left:0px}.wysija-version .social-foot .socials .fb,.wysija-version .social-foot .socials .fb div{margin-left:0px}.wysija-version .version{float:left;margin-bottom:5px}.wysija-version .socials{padding-top:9px}#hidesocials{margin-top:2px;font-size:12px}.social-foot .gplus{width:70px}#upperfoot{display:block;float:none}#wysija-app .form-table td{vertical-align:top}#submitexport label input[type="checkbox"]{margin-left:0px !important}.hook-column{float:left;display:block}#wysija-app div.warning{margin:5px 0 15px;background-color:#FFFFE0;border-color:#E6DB55;border-radius:3px 3px 3px 3px;border-style:solid;border-width:1px;padding:10px}.submit-box{border-top:1px solid #dbdbdb;background-color:#FFFFFF;bottom:0;height:50px;left:0;margin:0;padding:0;position:fixed;text-align:right;width:100%}#loading-icon{width:20px;height:20px;position:absolute;left:16px;top:16px}.loading #loading-icon{background:url(../img/wpspin_light.gif) no-repeat 0 0 #fff}.counting #loading-icon{background:url(../img/controls/icons.png) no-repeat -140px -20px #fff}.submit-box #loading-message{font-family:"Trebuchet MS";font-size:14px;left:39px;position:absolute;top:17px}.submit-box #insert-selection,.submit-box #back-selection{margin:10px 10px 0 0}.submit-box #back-selection{display:none}#icon-stats{background:url("../img/stats-icon.png") no-repeat scroll 0 0 transparent}
css/admin-premium.css ADDED
@@ -0,0 +1 @@
 
1
+ #mainmenu .premium{background:url(../img/ui-bg_glass_premium_1x400.png) repeat-x scroll 50% 50% #e6e6e6;border:1px solid #a9abc3}#mainmenu .premium:hover{background:url(../img/ui-bg_glass_premium_hover_1x400.png) repeat-x scroll 50% 50% #e6e6e6;border:1px solid #9c9eb2}#mainmenu .premium a{font-weight:700;color:#5d5e76}#mainmenu .premium a:hover{color:#4b4d69}#mainmenu .ui-state-active{background:white !important;border:1px solid #d8d9e6}.mpoet-page h2{color:#4b4d69;font-size:1.8em;font-weight:200;line-height:1.2em;margin:0}.wysija-premium-wrapper{margin:0 0 20px 0}.wysija-premium-actions{position:fixed;width:1028px;bottom:0;background-color:#eee;border-top:1px solid #ccc;border-left:1px solid #ccc;border-right:1px solid #ccc;-webkit-border-top-left-radius:7px;-webkit-border-top-right-radius:7px;-moz-border-radius-topleft:7px;-moz-border-radius-topright:7px;border-top-left-radius:7px;border-top-right-radius:7px;z-index:999;padding:10px 10px 0}.wysija-premium-actions .licence{margin-right:15px}.wysija-premium-actions p{margin:0px 0px 10px 0;text-align:center}.wysija-premium-actions span.conditions{font-size:12px;margin-right:15px}.wysija-premium-actions .button-primary.wysija-premium-activate{float:none}.wysija-premium-actions-kim{position:fixed;width:1066px;bottom:0;background-color:#eee;border-top:1px solid #ccc;border-left:1px solid #ccc;border-right:1px solid #ccc;z-index:999;margin-left:-30px;padding:10px 10px;text-align:center}.mpoet-page .feature-section.three-col>div{width:29%}.mpoet-page .three-col img{margin:0.5em 0;max-width:100%}a.wysija-premium-purchase{background-color:#FFB30D !important;background:-webkit-gradient(linear, left top, left bottom, from(#fecb19), to(#ff9c00)) !important;background:-webkit-linear-gradient(#fecb19, #ff9c00) !important;background:-moz-linear-gradient(center top, #fecb19 0%, #ff9c00 100%) !important;background:-moz-gradient(center top, #fecb19 0%, #ff9c00 100%) !important;border:1px solid #ff9c00 !important;color:#ffffff;font-family:"Amaranth", Arial, sans-serif;font-style:italic;font-size:15px !important;font-weight:normal;letter-spacing:1px;text-decoration:none;text-shadow:0 1px 0 #434343;padding:10px 30px;border-radius:7px;-moz-border-radius:7px;-webkit-border-radius:7px;box-shadow:none !important;text-shadow:0px 2px 1px rgba(0,0,0,0.3) !important;display:inline-block;text-align:center;width:auto}a.wysija-premium-purchase:hover{box-shadow:none !important}a.wysija-premium-purchase:hover,a.wysija-premium-purchase:active{text-decoration:none !important;background:#ffd547;background:-webkit-gradient(linear, left top, left bottom, from(#ffd648), to(#ffbe38));background:-webkit-linear-gradient(#ffd648, #ffbe38);background:-moz-linear-gradient(center top, #ffd648 20%, #ffbe38 100%);background:-moz-gradient(center top, #ffd648 20%, #ffbe38 100%)}a.wysija-premium-purchase,a.wysija-premium-activate{margin:10px 0}.mpoet-page .bullet-hold{width:600px;margin:15px 30px 15px 5px}.mpoet-page div.description h3{margin:5px 0 4px 0;font-family:"HelveticaNeue-Light","Helvetica Neue Light","Helvetica Neue",sans-serif;font-size:20px;font-weight:normal;color:black}.mpoet-page div.description p{margin:0}#prices_table{margin-top:55px}#prices_table .one-third{display:block;float:left;width:284px}#prices_table #prices_names{height:63px}#prices_table #prices_names .one-third{-moz-border-bottom-colors:none;-moz-border-left-colors:none;-moz-border-right-colors:none;-moz-border-top-colors:none;background:none repeat-x scroll left top rgba(0,0,0,0);border-color:#414141 -moz-use-text-color #414141 #414141;border-image:none;border-style:solid none solid solid;border-width:1px 0 1px 1px;height:61px;text-align:center}#prices_table #prices_names{display:block}#prices_table #prices_names h3{color:#FFF !important;line-height:1em !important}#prices_table #prices_names .blogger{background-color:#0074A2}#prices_table #prices_names .freelance{background-color:#026187}#prices_table #prices_names .agency{background-color:#014661;border-right:1px solid #414141}#prices_table #prices_cost{height:177px}#prices_table #prices_cost p{margin:0}#prices_table #prices_cost .one-third{-moz-border-bottom-colors:none;-moz-border-left-colors:none;-moz-border-right-colors:none;-moz-border-top-colors:none;border-color:-moz-use-text-color -moz-use-text-color #849A97 #849A97;border-image:none;border-style:none none solid solid;border-width:0 0 1px 1px;font-family:'Amaranth',Arial,sans-serif;height:176px}#prices_table #prices_cost .agency{border-right:1px solid #849A97}#prices_table #prices_cost span{display:block}#prices_table #prices_cost .dollars{font-size:80px;height:80px;line-height:normal;padding:20px 55px 0;text-align:center}#prices_table #prices_cost .per_year{font-size:18px;padding:0 60px;text-align:right}#prices_table #prices_cost .blogger .dollars{padding:20px 75px 0}#prices_table #prices_cost .blogger .per_year{padding:0 80px}#prices_table #prices_description{height:62px}#prices_table #prices_description .one-third{-moz-border-bottom-colors:none;-moz-border-left-colors:none;-moz-border-right-colors:none;-moz-border-top-colors:none;border-color:-moz-use-text-color -moz-use-text-color #849A97 #849A97;border-image:none;border-style:none none solid solid;border-width:0 0 1px 1px;font-size:22px;height:61px}#prices_table #prices_description .agency{border-right:1px solid #849A97}#prices_table #prices_description .agency p{display:block;margin:-15px 62px 0 0;text-align:right;font-size:.5em}#prices_table #prices_description span{display:block;padding:17px 0;text-align:center}#prices_table #prices_content{height:158px}#prices_table #prices_content .prices_content{-moz-border-bottom-colors:none;-moz-border-left-colors:none;-moz-border-right-colors:none;-moz-border-top-colors:none;border-color:-moz-use-text-color #849A97 #849A97;border-image:none;border-right:1px solid #849A97;border-style:none solid solid;border-width:0 1px 1px;color:#414141;font-size:17px;padding:16px 0}#prices_table #prices_content .prices_content span{display:block;padding:2px 0;text-align:center}a.buy-button{background:none repeat scroll 0 0 #2EA2CC;border-color:#0074A2;box-shadow:0 1px 0 rgba(120,200,230,0.5) inset,0 1px 0 rgba(0,0,0,0.15);color:#FFFFFF;-moz-box-sizing:border-box;border-radius:3px;border-style:solid;border-width:1px;cursor:pointer;display:inline-block;font-size:20px;margin-top:20px;margin-bottom:60px;padding:7px 10px 7px;text-decoration:none;white-space:nowrap}a.buy-button:hover{color:#ecf3fe}.mpoet-page h1{font-size:34px}#wysija-app .mpoet-page h2{font-size:25px}.mpoet-page h1,.mpoet-page h2,.mpoet-page h3,.mpoet-page h4{color:#626262;font-weight:bold}.mpoet-page h3{margin-top:5px;line-height:24px}.mpoet-page .three-col .argument-cta{display:table;margin-top:5px}#premium-content-b .pick-licence{margin-bottom:15px}.mpoet-page hr{margin-bottom:20px}#premium-content-b .price{font-weight:bold}#footer-upgrade,#footer-left,#wpfooter .alignright,.wysija-footer{display:none}.about-wrap div.updated,.about-wrap div.error{display:block !important}#wysija-app .about-wrap .feature-section{margin-top:20px}
css/admin-statistics.css ADDED
@@ -0,0 +1 @@
 
1
+ .stats_date_filter{padding-top:10px}.statistic-title{clear:both;margin-top:5px}.cl{clear:both}#hook_stats .container{clear:both;padding-top:1px;margin-top:20px}#stats-filter .reset{background:none;border:none;cursor:pointer}#stats-filter .reset:hover{text-decoration:underline}.custom_date{margin-right:10px}.spinner{float:left !important}.hook{margin-top:10px;clear:both}.container-top-links .link_column{width:20%}.container-top-links .click_column{width:6%}.container .check-column{width:2.5em}.container-top-subscribers table th.subscriber{width:30%}.container-top-subscribers table th.clicks{width:7%}.container-top-subscribers table th.opens{width:7%}.container-top-subscribers table th.lists{width:41%}.container-top-subscribers table th.date{width:15%}.container-top-links a.newsletter{display:block;padding-top:5px}.container-top-newsletters th.newsletter{width:19%}.container-top-newsletters th.sent{width:8%}.container-top-newsletters th.opens{width:8%}.container-top-newsletters th.clicks{width:7%}.container-top-newsletters th.unsubscribes{width:11%}.container-top-newsletters th.rates{width:18%}.container-top-newsletters th.sent_at{width:12%}.container-top-newsletters th.lists{width:14%}.container-top-newsletters td a.stats{display:none}.container-top-newsletters td.newsletter:hover a.stats{display:block}.container-top-subscribers th.desc:hover span.sorting-indicator,.container-top-newsletters th.desc:hover span.sorting-indicator{background-position:-7px 0}.stats_feedback{font-weight:bold;margin-top:20px;clear:both}.tip{font-style:italic}.stats_subscriptions-container .tip{margin-left:70px}.container-top-newsletters th.desc:hover span.sorting-indicator:before,.container-top-subscribers th.desc:hover span.sorting-indicator:before,.container-top-newsletters th.asc span.sorting-indicator:before,.container-top-subscribers th.asc span.sorting-indicator:before{content:'\f140'}
css/admin-subscribers-addlist.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-edit .form-table th{width:200px}
css/admin-subscribers-edit.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-edit{padding-bottom:20px}#hook_subscriber_left{width:50%}#hook_subscriber_right{width:50%}.hook-column .submit{text-align:center}#hook_subscriber_bottom{clear:both;padding-top:1px}.container{clear:both;margin-top:30px}.form-table th label{float:right;white-space:nowrap}.labelcheck{float:left;margin-right:15px}#wysija-edit input[type="checkbox"]{margin-right:6px;vertical-align:baseline}#wysija-edit .form-table th{width:60px}.container-stats-subscribers_std table th.newsletter{width:60%}.container-stats-subscribers_std table th.link{width:20%}.container-stats-subscribers_std table th.click{width:5%}.container-stats-subscribers_std .widefat .check-column{width:2.4em}.container-stats-subscribers_std .widefat th.date{width:15%}
css/admin-subscribers-export.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-app table td label{margin-right:15px}#wysija-app table td input[type="radio"]{margin-right:5px}
css/admin-subscribers-exportlist.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-app table td label{margin-right:15px}#wysija-app table td input[type="radio"]{margin-right:5px}
css/admin-subscribers-import.css ADDED
@@ -0,0 +1 @@
 
1
+ .empty{color:#888}
css/admin-subscribers-importmatch.css ADDED
@@ -0,0 +1 @@
 
1
+ #wysija-edit .widefat td{min-width:100px}#wysija-edit .widefat{width:auto}#blocknewlist{width:175px}.import-new-field{display:none;margin-top:4px}.import-new-field label{font-size:12px}.widefat th .import-new-field input{margin:2px 5px 0 4px}.widefat thead{vertical-align:top}#first-row{padding-top:14px}.converted-field-to-date{display:none;margin-left:5px;color:#0e90d2}.converted-field-error{display:none;margin-left:5px;color:#b10000;font-weight:bold}
css/admin-subscribers-lists.css ADDED
File without changes
css/admin-subscribers.css ADDED
@@ -0,0 +1 @@
 
1
+ tr.submit_row>td{padding-left:0}.list .check-column{width:10px}#wysija-app .list .column-date{width:70px}#wysija-app .list .column-list-names{width:100px}
css/admin-widget.css ADDED
@@ -0,0 +1 @@
 
1
+ .wysija-widget-form .lists-block input[type="checkbox"],.wysija-widget-form input[type="radio"]{margin:0 4px 0 0}.wysija-widget-form .labelcheck label{margin:0 4px 0 0}.wysija-widget-form .lists-block{margin:0 3px 0 6px;clear:both;max-height:116px;overflow:auto;float:left;margin:0 10px 0 0}.wysija-widget-form .labelcheck{margin:0 0 5px 0;float:left;margin-left:5px}.wysija-widget-form .divblocks{margin:10px 0 10px 0}#wpfooter .wrap{display:block;margin:3px 0 3px 0 !important}#wpfooter .wrap .add-new-h2{text-decoration:none}
css/admin.css ADDED
@@ -0,0 +1 @@
 
1
+ .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{display:inline-table}* html .clearfix{height:1%}.clearfix{display:block}#wysija-app .column-date{width:125px}#wysija-app .column-clic,#wysija-app .column-opened,#wysija-app .column-emails{width:80px}#wysija-app .column-list-names{width:120px}#wysija-app .menuslinks{margin:20px 0}#statscontainer{height:200px;width:400px}#wysistats1{width:400px}#wysistats1 h3{text-align:center}#wysistats2,#wysistats3{width:200px;padding:30px 0 0 20px}#wysistats3{height:200px;overflow:auto}#wysistats2 ul li,#wysistats .title{font-weight:bold;font-size:14px;margin-top:0}#wysistats3 ol li{font-style:italic;font-size:10px;color:#444}#wysistats3 ol li em{font-size:11px;font-weight:bold}th.sorted,th.sortable{cursor:pointer}div#wysija-app .xdetailed-updated li,div#wysija-app .xdetailed-errors li,div#wysija-app .updated li,div#wysija-app .error li{margin:6px 0}.xdetailed-errors,.xdetailed-updated{border-radius:3px 3px 3px 3px;border-style:solid;border-width:1px;margin:5px 15px 2px;padding:0 0.6em}.xdetailed-updated{background-color:#FFFFE0;border-color:#E6DB55;color:#000;display:none}.xdetailed-errors{background-color:#FFEBE8;border-color:#888888;color:#000;display:none}label input[type="checkbox"],label input[type="radio"]{margin:0 5px 0 15px}label:first-child input[type="checkbox"],label:first-child input[type="radio"]{margin-left:0}label.title{font-weight:bold;font-size:14px}.wrap .button-secondary2{background-color:#6697BF;font-family:sans-serif;font-size:12px;color:#FFFFFF;text-decoration:none;text-shadow:0 1px 0 #777777;box-shadow:1px 1px 0 #AAAAAA;margin-left:4px;padding:3px 8px;position:relative;top:-3px;border-radius:3px;-moz-border-radius:3px;-khtml-border-radius:3px;-webkit-border-radius:3px}.wrap .button-secondary2:hover,.wrap .wcurrent{background-color:#1C73B5}.wrap #theme-view .button-secondary2{top:2px}.dots{margin:0 4px}#wysija-pagination{text-align:center}.tablenav .tablenav-pages a,.tablenav-pages span.current{margin:0 2px}.total-pages{padding:0px 3px}input.searchbox{width:225px}.wysija-premium img{display:none}#TB_load,#TB_window,#TB_overlay,.ui-dialog{z-index:100000 !important}.ui-widget-overlay{z-index:9998 !important;background-image:none !important}p.label{font-size:13px}.wysija_popup_content{padding:10px 20px 0 20px}.wysija_popup_content h3{margin:0 0 20px 0}a.linkignore{color:#BC0B0B}a.linkignore:hover,a.submitsynch:hover{color:#FF0000}a.submitsynch{color:#52aa05}body{min-width:0 !important}.warning-msg{background-color:#FFFFE0;border-color:#E6DB55}.warning-msg{border-radius:3px 3px 3px 3px;border-style:solid;border-width:1px;margin:5px 15px 2px;padding:0 0.6em}.warning-msg ul{margin:0}span.warning-msg{padding:7px}.about-wrap h3{padding-top:0px}.about-text{margin-bottom:1em;margin-top:0;min-height:20px}.wysija-unsubscribed-on{color:#bbb}.clear_select_all,.force_to_select_all_link{text-align:center}.batch-select{background-color:#E0E0E0;color:#555555;border-bottom:1px solid #e5e5e5}.clear_select_all a:hover,.force_to_select_all_link a:hover{text-decoration:underline;cursor:pointer !important}#wysija-app .notice-msg{background-color:#FFFFE0;border-color:#E6DB55}#wysija-app .error-msg{background-color:#FFEBE8;border-color:#CC0000}#wysija-app .wysija-msg .notice-msg,#wysija-app .wysija-msg .error-msg{border-radius:3px 3px 3px 3px;border-style:solid;border-width:1px;margin:5px 0 15px;padding:0 0.6em}.mpoet-page .changelog .review-follow-kitten{overflow:visible}
css/adminPopup.css ADDED
@@ -0,0 +1 @@
 
1
+ html{overflow:hidden;overflow-y:auto;height:auto !important}html.wp-toolbar{padding:0 !important}#wpadminbar{display:none}body{-webkit-backface-visibility:hidden}.popup_content{margin:10px 10px 0 10px}.popup_content.addlink{width:300px}.popup_content.articles{width:700px}.popup_content.dividers{width:620px}.popup_content.autopost{width:700px;min-height:380px}.popup_content.themes{width:800px}.popup_content.bookmarks{width:620px}.popup_content.media-browse,.popup_content.media-wp-browse,.popup_content.media-wp-upload{width:715px}.popup_content.form_widget_settings{width:405px}.popup_content.widget_text,.popup_content.widget_html{width:450px}.popup_content.widget_list{width:450px}p.align-right{text-align:right;margin-bottom:0 !important}div.updated,div.error{margin:10px !important}.popup_content.inline_form label{line-height:2em;font-weight:bold}.popup_content.inline_form p input,.popup_content.inline_form p select,.popup_content.inline_form p .group{float:left;width:200px}.submit_button{text-align:right;margin:20px 10px 20px 0}.popup_content.inline_form .submit_button{margin:20px 4px 0 0}.popup_content.inline_form p input[type="submit"],.popup_content.inline_form .group label,.popup_content.inline_form .group input,.popup_content.inline_form .group select{float:none;clear:none;width:auto}.popup_content.inline_form label.radio,.popup_content.inline_form .group label{font-weight:normal}.popup_content.inline_form input[type="radio"],.popup_content.inline_form input[type="checkbox"]{width:16px;height:16px;margin:0 3px 0 0;padding:0}.popup_content.inline_form input[type="radio"]{-webkit-appearance:radio}.popup_content.inline_form input[type="radio"]:checked:before,.popup_content.inline_form input[type="checkbox"]:checked:before{background:none;content:''}.popup_content.inline_form input[type="checkbox"]{-webkit-appearance:checkbox}.popup_content.inline_form textarea,.popup_content.inline_form input[type="text"],.popup_content.inline_form input[type="password"],.popup_content.inline_form input[type="email"],.popup_content.inline_form input[type="number"],.popup_content.inline_form input[type="search"],.popup_content.inline_form input[type="tel"],.popup_content.inline_form input[type="url"],.popup_content.inline_form select{font-size:13px;line-height:15px;padding:3px 5px}
css/index.html ADDED
File without changes
css/jquery/ui/themes/base/images/animated-overlay.gif ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_flat_0_aaaaaa_40x100.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_flat_75_ffffff_40x100.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_glass_55_fbf9ee_1x400.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_glass_65_ffffff_1x400.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_glass_75_dadada_1x400.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_glass_75_e6e6e6_1x400.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_glass_95_fef1ec_1x400.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-bg_highlight-soft_75_cccccc_1x100.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-icons_222222_256x240.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-icons_2e83ff_256x240.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-icons_454545_256x240.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-icons_888888_256x240.png ADDED
Binary file
css/jquery/ui/themes/base/images/ui-icons_cd0a0a_256x240.png ADDED
Binary file
css/jquery/ui/themes/base/jquery.ui.core.min.css ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ /*! jQuery UI - v1.10.3 - 2013-05-03
2
+ * http://jqueryui.com
3
+ * Copyright 2013 jQuery Foundation and other contributors; Licensed MIT */
4
+
5
+ .ui-helper-hidden{display:none}.ui-helper-hidden-accessible{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px}.ui-helper-reset{margin:0;padding:0;border:0;outline:0;line-height:1.3;text-decoration:none;font-size:100%;list-style:none}.ui-helper-clearfix:before,.ui-helper-clearfix:after{content:"";display:table;border-collapse:collapse}.ui-helper-clearfix:after{clear:both}.ui-helper-clearfix{min-height:0}.ui-helper-zfix{width:100%;height:100%;top:0;left:0;position:absolute;opacity:0;filter:Alpha(Opacity=0)}.ui-front{z-index:100}.ui-state-disabled{cursor:default!important}.ui-icon{display:block;text-indent:-99999px;overflow:hidden;background-repeat:no-repeat}.ui-widget-overlay{position:fixed;top:0;left:0;width:100%;height:100%}
css/jquery/ui/themes/base/jquery.ui.datepicker.min.css ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ /*! jQuery UI - v1.10.3 - 2013-05-03
2
+ * http://jqueryui.com
3
+ * Copyright 2013 jQuery Foundation and other contributors; Licensed MIT */
4
+
5
+ .ui-datepicker{width:17em;padding:.2em .2em 0;display:none}.ui-datepicker .ui-datepicker-header{position:relative;padding:.2em 0}.ui-datepicker .ui-datepicker-prev,.ui-datepicker .ui-datepicker-next{position:absolute;top:2px;width:1.8em;height:1.8em}.ui-datepicker .ui-datepicker-prev-hover,.ui-datepicker .ui-datepicker-next-hover{top:1px}.ui-datepicker .ui-datepicker-prev{left:2px}.ui-datepicker .ui-datepicker-next{right:2px}.ui-datepicker .ui-datepicker-prev-hover{left:1px}.ui-datepicker .ui-datepicker-next-hover{right:1px}.ui-datepicker .ui-datepicker-prev span,.ui-datepicker .ui-datepicker-next span{display:block;position:absolute;left:50%;margin-left:-8px;top:50%;margin-top:-8px}.ui-datepicker .ui-datepicker-title{margin:0 2.3em;line-height:1.8em;text-align:center}.ui-datepicker .ui-datepicker-title select{font-size:1em;margin:1px 0}.ui-datepicker select.ui-datepicker-month-year{width:100%}.ui-datepicker select.ui-datepicker-month,.ui-datepicker select.ui-datepicker-year{width:49%}.ui-datepicker table{width:100%;font-size:.9em;border-collapse:collapse;margin:0 0 .4em}.ui-datepicker th{padding:.7em .3em;text-align:center;font-weight:700;border:0}.ui-datepicker td{border:0;padding:1px}.ui-datepicker td span,.ui-datepicker td a{display:block;padding:.2em;text-align:right;text-decoration:none}.ui-datepicker .ui-datepicker-buttonpane{background-image:none;margin:.7em 0 0;padding:0 .2em;border-left:0;border-right:0;border-bottom:0}.ui-datepicker .ui-datepicker-buttonpane button{float:right;margin:.5em .2em .4em;cursor:pointer;padding:.2em .6em .3em;width:auto;overflow:visible}.ui-datepicker .ui-datepicker-buttonpane button.ui-datepicker-current{float:left}.ui-datepicker.ui-datepicker-multi{width:auto}.ui-datepicker-multi .ui-datepicker-group{float:left}.ui-datepicker-multi .ui-datepicker-group table{width:95%;margin:0 auto .4em}.ui-datepicker-multi-2 .ui-datepicker-group{width:50%}.ui-datepicker-multi-3 .ui-datepicker-group{width:33.3%}.ui-datepicker-multi-4 .ui-datepicker-group{width:25%}.ui-datepicker-multi .ui-datepicker-group-last .ui-datepicker-header,.ui-datepicker-multi .ui-datepicker-group-middle .ui-datepicker-header{border-left-width:0}.ui-datepicker-multi .ui-datepicker-buttonpane{clear:left}.ui-datepicker-row-break{clear:both;width:100%;font-size:0}.ui-datepicker-rtl{direction:rtl}.ui-datepicker-rtl .ui-datepicker-prev{right:2px;left:auto}.ui-datepicker-rtl .ui-datepicker-next{left:2px;right:auto}.ui-datepicker-rtl .ui-datepicker-prev:hover{right:1px;left:auto}.ui-datepicker-rtl .ui-datepicker-next:hover{left:1px;right:auto}.ui-datepicker-rtl .ui-datepicker-buttonpane{clear:right}.ui-datepicker-rtl .ui-datepicker-buttonpane button{float:left}.ui-datepicker-rtl .ui-datepicker-buttonpane button.ui-datepicker-current,.ui-datepicker-rtl .ui-datepicker-group{float:right}.ui-datepicker-rtl .ui-datepicker-group-last .ui-datepicker-header,.ui-datepicker-rtl .ui-datepicker-group-middle .ui-datepicker-header{border-right-width:0;border-left-width:1px}
css/rtl.css ADDED
@@ -0,0 +1 @@
 
1
+ body{direction:rtl;unicode-bidi:embed}input#url{direction:ltr}.xdetailed-errors,.xdetailed-updated{margin:2px 15px 5px}.wysija-footer,.wysija-version{margin:0 165px 0 20px}label input[type="checkbox"],label input[type="radio"]{margin:0 15px 0 5px}input.searchsub.button{float:none}a.button-primary.install{float:none}#review-follow div{float:right}.about-wrap .feature-section.three-col div{float:right}#wysija-badge{left:0;right:auto}.dividers ul{overflow-x:auto;overflow-y:auto}.submit-box{text-align:left}.submit-box #insert-selection,.submit-box #back-selection{margin:10px 0 0 10px}#toggle-advanced{left:10px;top:0;right:auto}.articles #basic .search-box{float:left}.articles #basic .filters-box{float:right}.articles #search-submit{margin:0 0 0 14px}.articles .cpt-type{left:0;right:auto;border-right:1px solid #ccc;border-left:0 none}#advanced div.block{padding:0.8em 1em 0.8em 0}#advanced label{float:right}#image_width_slider{float:right}#slider_info{float:right}#advanced label.radio{float:right;margin:0 0 0 9px}#advanced label.radio input{margin:0 0 0 2px}#advanced div.group{float:right}#results .thumbnail{float:right;margin:0 0 0 10px}#results li .checkbox{cursor:pointer;margin:0 11px 0 0}#results li .checkbox_container{float:right}#results li label{float:right}#loading-icon{left:auto;right:16px;top:16px}.submit-box #loading-message{left:auto;right:39px}.wysija-footer,.wysija-version{margin:0 165px 0 20px}.wysija-version .social-foot{float:right}body.rtl #wysija_wrapper{right:0 !important}#wysija_popup_title h3{float:right;margin-right:10px}#wysija_popup_close{float:left;margin-left:10px}#wysija_toolbar{right:657px}.wj_images li,.wj_themes li{float:right}.wj_images #wj_images_preview,.wj_themes #wj_themes_preview{left:295px;right:auto}.wj_content a.wysija_item,.wysija_widget{padding:0 10px 0 0}.wj_styles form label{margin:0 3px 0 0}.wj_styles #aUnderlineInput{margin-right:10px !important}.wysija_item_settings{left:15px;right:auto}#wysija_toggle_images{left:5px;right:auto}#review-follow .review-left.large{float:right}#review-follow .review-left{background-position:100% 0}#review-follow div{float:right}#review-follow .review-left .description{margin-right:146px;margin-left:0px}#review-follow .review-right{background-image:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow_middle-rtl.jpg");background-position:100% 0}#review-follow .follow-right{background-image:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow_right-rtl.jpg");background-position:100% 0}#review-follow .follow-right38{background-image:url("http://s-plugins.wordpress.org/wysija-newsletters/assets/update_page/arrow-right-38-rtl.jpg");background-position:100% 0}#wysija-settings input[type="checkbox"],#wysija-settings input[type="radio"],#wysija-settings label input[type="checkbox"]{margin-left:6px}label:first-child input[type="checkbox"],label:first-child input[type="radio"],label input[type="checkbox"],label input[type="radio"]{margin:0 15px 0 5px}.linknamecboxes,.labelcheck,.geeky-option a,.geeky-option span{float:right}.checkbox_optin_label{margin:6px 0 0 6px;float:right}.checkbox_optin_value{float:right}#wysija-settings .form-table th.rolestitle{text-align:right}.cronleft{float:right;margin:0 0 0 10px}.popup_content.autopost{width:auto}.popup_content{margin:10px 10px 0 10px}#wysija-settings #sendingmethod .methods h3,#wysija-settings #multisite .methods p.title{float:right}#wysija-settings #sendingmethod .methods input,#wysija-settings #multisite .methods input{float:right;margin:0 0 0 5px}h2.title span,h2.title form{float:right;margin-left:5px;margin-right:0px}#wysija_form_toolbar{right:679px !important;left:auto !important}#wysija_form_toolbar a.wysija_form_item,.wysija_form_widget{padding-right:7px}.form_widget_settings p label{float:right;clear:right;width:200px}.form_widget_settings .selection .is_selected{float:right;margin:3px 6px 0 0 !important}.form_widget_settings .selection label{float:right;margin:0 5px 0 0}.form_widget_settings .selection .handle,.form_widget_settings .selection .remove{float:left;margin:3px 0 0 3px}.popup_content.inline_form p input,.popup_content.inline_form p select,.popup_content.inline_form p .group{float:right}.submit_button{margin:20px 0 20px 10px;text-align:left}.popup_content.inline_form .submit_button{margin:20px 0 0 4px}.popup_content.inline_form label.radio,.popup_content.inline_form .group label{float:right}.popup_content.inline_form input[type="radio"],.popup_content.inline_form input[type="checkbox"]{margin:0 0 0 3px}.wysija_form_item_settings{left:35px;right:auto !important}.wysija_form_item_delete{left:15px;right:auto !important}.bookmarks .networks label,.bookmarks .networks input{float:right}#results li .checkbox{margin-left:10px}.add span{float:right}.themes #wj_paginator{float:left;margin-right:0;margin-left:10px}
css/select2/select2.css ADDED
@@ -0,0 +1,615 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ /*
2
+ Version: 3.4.5 Timestamp: Mon Nov 4 08:22:42 PST 2013
3
+ */
4
+ .select2-container {
5
+ margin: 0;
6
+ position: relative;
7
+ display: inline-block;
8
+ /* inline-block for ie7 */
9
+ zoom: 1;
10
+ *display: inline;
11
+ vertical-align: middle;
12
+ }
13
+
14
+ .select2-container,
15
+ .select2-drop,
16
+ .select2-search,
17
+ .select2-search input {
18
+ /*
19
+ Force border-box so that % widths fit the parent
20
+ container without overlap because of margin/padding.
21
+
22
+ More Info : http://www.quirksmode.org/css/box.html
23
+ */
24
+ -webkit-box-sizing: border-box; /* webkit */
25
+ -moz-box-sizing: border-box; /* firefox */
26
+ box-sizing: border-box; /* css3 */
27
+ }
28
+
29
+ .select2-container .select2-choice {
30
+ display: block;
31
+ height: 26px;
32
+ padding: 0 0 0 8px;
33
+ overflow: hidden;
34
+ position: relative;
35
+
36
+ border: 1px solid #aaa;
37
+ white-space: nowrap;
38
+ line-height: 26px;
39
+ color: #444;
40
+ text-decoration: none;
41
+
42
+ border-radius: 4px;
43
+
44
+ background-clip: padding-box;
45
+
46
+ -webkit-touch-callout: none;
47
+ -webkit-user-select: none;
48
+ -moz-user-select: none;
49
+ -ms-user-select: none;
50
+ user-select: none;
51
+
52
+ background-color: #fff;
53
+ background-image: -webkit-gradient(linear, left bottom, left top, color-stop(0, #eee), color-stop(0.5, #fff));
54
+ background-image: -webkit-linear-gradient(center bottom, #eee 0%, #fff 50%);
55
+ background-image: -moz-linear-gradient(center bottom, #eee 0%, #fff 50%);
56
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr = '#ffffff', endColorstr = '#eeeeee', GradientType = 0);
57
+ background-image: linear-gradient(top, #fff 0%, #eee 50%);
58
+ }
59
+
60
+ .select2-container.select2-drop-above .select2-choice {
61
+ border-bottom-color: #aaa;
62
+
63
+ border-radius: 0 0 4px 4px;
64
+
65
+ background-image: -webkit-gradient(linear, left bottom, left top, color-stop(0, #eee), color-stop(0.9, #fff));
66
+ background-image: -webkit-linear-gradient(center bottom, #eee 0%, #fff 90%);
67
+ background-image: -moz-linear-gradient(center bottom, #eee 0%, #fff 90%);
68
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffff', endColorstr='#eeeeee', GradientType=0);
69
+ background-image: linear-gradient(top, #eee 0%, #fff 90%);
70
+ }
71
+
72
+ .select2-container.select2-allowclear .select2-choice .select2-chosen {
73
+ margin-right: 42px;
74
+ }
75
+
76
+ .select2-container .select2-choice > .select2-chosen {
77
+ margin-right: 26px;
78
+ display: block;
79
+ overflow: hidden;
80
+
81
+ white-space: nowrap;
82
+
83
+ text-overflow: ellipsis;
84
+ }
85
+
86
+ .select2-container .select2-choice abbr {
87
+ display: none;
88
+ width: 12px;
89
+ height: 12px;
90
+ position: absolute;
91
+ right: 24px;
92
+ top: 8px;
93
+
94
+ font-size: 1px;
95
+ text-decoration: none;
96
+
97
+ border: 0;
98
+ background: url('../../img/select2/select2.png') right top no-repeat;
99
+ cursor: pointer;
100
+ outline: 0;
101
+ }
102
+
103
+ .select2-container.select2-allowclear .select2-choice abbr {
104
+ display: inline-block;
105
+ }
106
+
107
+ .select2-container .select2-choice abbr:hover {
108
+ background-position: right -11px;
109
+ cursor: pointer;
110
+ }
111
+
112
+ .select2-drop-mask {
113
+ border: 0;
114
+ margin: 0;
115
+ padding: 0;
116
+ position: fixed;
117
+ left: 0;
118
+ top: 0;
119
+ min-height: 100%;
120
+ min-width: 100%;
121
+ height: auto;
122
+ width: auto;
123
+ opacity: 0;
124
+ z-index: 9998;
125
+ /* styles required for IE to work */
126
+ background-color: #fff;
127
+ filter: alpha(opacity=0);
128
+ }
129
+
130
+ .select2-drop {
131
+ width: 100%;
132
+ margin-top: -1px;
133
+ position: absolute;
134
+ z-index: 9999;
135
+ top: 100%;
136
+
137
+ background: #fff;
138
+ color: #000;
139
+ border: 1px solid #aaa;
140
+ border-top: 0;
141
+
142
+ border-radius: 0 0 4px 4px;
143
+
144
+ -webkit-box-shadow: 0 4px 5px rgba(0, 0, 0, .15);
145
+ box-shadow: 0 4px 5px rgba(0, 0, 0, .15);
146
+ }
147
+
148
+ .select2-drop-auto-width {
149
+ border-top: 1px solid #aaa;
150
+ width: auto;
151
+ }
152
+
153
+ .select2-drop-auto-width .select2-search {
154
+ padding-top: 4px;
155
+ }
156
+
157
+ .select2-drop.select2-drop-above {
158
+ margin-top: 1px;
159
+ border-top: 1px solid #aaa;
160
+ border-bottom: 0;
161
+
162
+ border-radius: 4px 4px 0 0;
163
+
164
+ -webkit-box-shadow: 0 -4px 5px rgba(0, 0, 0, .15);
165
+ box-shadow: 0 -4px 5px rgba(0, 0, 0, .15);
166
+ }
167
+
168
+ .select2-drop-active {
169
+ border: 1px solid #5897fb;
170
+ border-top: none;
171
+ }
172
+
173
+ .select2-drop.select2-drop-above.select2-drop-active {
174
+ border-top: 1px solid #5897fb;
175
+ }
176
+
177
+ .select2-container .select2-choice .select2-arrow {
178
+ display: inline-block;
179
+ width: 18px;
180
+ height: 100%;
181
+ position: absolute;
182
+ right: 0;
183
+ top: 0;
184
+
185
+ border-left: 1px solid #aaa;
186
+ border-radius: 0 4px 4px 0;
187
+
188
+ background-clip: padding-box;
189
+
190
+ background: #ccc;
191
+ background-image: -webkit-gradient(linear, left bottom, left top, color-stop(0, #ccc), color-stop(0.6, #eee));
192
+ background-image: -webkit-linear-gradient(center bottom, #ccc 0%, #eee 60%);
193
+ background-image: -moz-linear-gradient(center bottom, #ccc 0%, #eee 60%);
194
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr = '#eeeeee', endColorstr = '#cccccc', GradientType = 0);
195
+ background-image: linear-gradient(top, #ccc 0%, #eee 60%);
196
+ }
197
+
198
+ .select2-container .select2-choice .select2-arrow b {
199
+ display: block;
200
+ width: 100%;
201
+ height: 100%;
202
+ background: url('../../img/select2/select2.png') no-repeat 0 1px;
203
+ }
204
+
205
+ .select2-search {
206
+ display: inline-block;
207
+ width: 100%;
208
+ min-height: 26px;
209
+ margin: 0;
210
+ padding-left: 4px;
211
+ padding-right: 4px;
212
+
213
+ position: relative;
214
+ z-index: 10000;
215
+
216
+ white-space: nowrap;
217
+ }
218
+
219
+ .select2-search input {
220
+ width: 100%;
221
+ height: auto !important;
222
+ min-height: 26px;
223
+ padding: 4px 20px 4px 5px;
224
+ margin: 0;
225
+
226
+ outline: 0;
227
+ font-family: sans-serif;
228
+ font-size: 1em;
229
+
230
+ border: 1px solid #aaa;
231
+ border-radius: 0;
232
+
233
+ -webkit-box-shadow: none;
234
+ box-shadow: none;
235
+
236
+ background: #fff url('../../img/select2/select2.png') no-repeat 100% -22px;
237
+ background: url('../../img/select2/select2.png') no-repeat 100% -22px, -webkit-gradient(linear, left bottom, left top, color-stop(0.85, #fff), color-stop(0.99, #eee));
238
+ background: url('../../img/select2/select2.png') no-repeat 100% -22px, -webkit-linear-gradient(center bottom, #fff 85%, #eee 99%);
239
+ background: url('../../img/select2/select2.png') no-repeat 100% -22px, -moz-linear-gradient(center bottom, #fff 85%, #eee 99%);
240
+ background: url('../../img/select2/select2.png') no-repeat 100% -22px, linear-gradient(top, #fff 85%, #eee 99%);
241
+ }
242
+
243
+ .select2-drop.select2-drop-above .select2-search input {
244
+ margin-top: 4px;
245
+ }
246
+
247
+ .select2-search input.select2-active {
248
+ background: #fff url('../../img/select2/select2-spinner.gif') no-repeat 100%;
249
+ background: url('../../img/select2/select2-spinner.gif') no-repeat 100%, -webkit-gradient(linear, left bottom, left top, color-stop(0.85, #fff), color-stop(0.99, #eee));
250
+ background: url('../../img/select2/select2-spinner.gif') no-repeat 100%, -webkit-linear-gradient(center bottom, #fff 85%, #eee 99%);
251
+ background: url('../../img/select2/select2-spinner.gif') no-repeat 100%, -moz-linear-gradient(center bottom, #fff 85%, #eee 99%);
252
+ background: url('../../img/select2/select2-spinner.gif') no-repeat 100%, linear-gradient(top, #fff 85%, #eee 99%);
253
+ }
254
+
255
+ .select2-container-active .select2-choice,
256
+ .select2-container-active .select2-choices {
257
+ border: 1px solid #5897fb;
258
+ outline: none;
259
+
260
+ -webkit-box-shadow: 0 0 5px rgba(0, 0, 0, .3);
261
+ box-shadow: 0 0 5px rgba(0, 0, 0, .3);
262
+ }
263
+
264
+ .select2-dropdown-open .select2-choice {
265
+ border-bottom-color: transparent;
266
+ -webkit-box-shadow: 0 1px 0 #fff inset;
267
+ box-shadow: 0 1px 0 #fff inset;
268
+
269
+ border-bottom-left-radius: 0;
270
+ border-bottom-right-radius: 0;
271
+
272
+ background-color: #eee;
273
+ background-image: -webkit-gradient(linear, left bottom, left top, color-stop(0, #fff), color-stop(0.5, #eee));
274
+ background-image: -webkit-linear-gradient(center bottom, #fff 0%, #eee 50%);
275
+ background-image: -moz-linear-gradient(center bottom, #fff 0%, #eee 50%);
276
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#ffffff', GradientType=0);
277
+ background-image: linear-gradient(top, #fff 0%, #eee 50%);
278
+ }
279
+
280
+ .select2-dropdown-open.select2-drop-above .select2-choice,
281
+ .select2-dropdown-open.select2-drop-above .select2-choices {
282
+ border: 1px solid #5897fb;
283
+ border-top-color: transparent;
284
+
285
+ background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #fff), color-stop(0.5, #eee));
286
+ background-image: -webkit-linear-gradient(center top, #fff 0%, #eee 50%);
287
+ background-image: -moz-linear-gradient(center top, #fff 0%, #eee 50%);
288
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#ffffff', GradientType=0);
289
+ background-image: linear-gradient(bottom, #fff 0%, #eee 50%);
290
+ }
291
+
292
+ .select2-dropdown-open .select2-choice .select2-arrow {
293
+ background: transparent;
294
+ border-left: none;
295
+ filter: none;
296
+ }
297
+ .select2-dropdown-open .select2-choice .select2-arrow b {
298
+ background-position: -18px 1px;
299
+ }
300
+
301
+ /* results */
302
+ .select2-results {
303
+ max-height: 200px;
304
+ padding: 0 0 0 4px;
305
+ margin: 4px 4px 4px 0;
306
+ position: relative;
307
+ overflow-x: hidden;
308
+ overflow-y: auto;
309
+ -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
310
+ }
311
+
312
+ .select2-results ul.select2-result-sub {
313
+ margin: 0;
314
+ padding-left: 0;
315
+ }
316
+
317
+ .select2-results ul.select2-result-sub > li .select2-result-label { padding-left: 20px }
318
+ .select2-results ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 40px }
319
+ .select2-results ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 60px }
320
+ .select2-results ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 80px }
321
+ .select2-results ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 100px }
322
+ .select2-results ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 110px }
323
+ .select2-results ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub ul.select2-result-sub > li .select2-result-label { padding-left: 120px }
324
+
325
+ .select2-results li {
326
+ list-style: none;
327
+ display: list-item;
328
+ background-image: none;
329
+ }
330
+
331
+ .select2-results li.select2-result-with-children > .select2-result-label {
332
+ font-weight: bold;
333
+ }
334
+
335
+ .select2-results .select2-result-label {
336
+ padding: 3px 7px 4px;
337
+ margin: 0;
338
+ cursor: pointer;
339
+
340
+ min-height: 1em;
341
+
342
+ -webkit-touch-callout: none;
343
+ -webkit-user-select: none;
344
+ -moz-user-select: none;
345
+ -ms-user-select: none;
346
+ user-select: none;
347
+ }
348
+
349
+ .select2-results .select2-highlighted {
350
+ background: #3875d7;
351
+ color: #fff;
352
+ }
353
+
354
+ .select2-results li em {
355
+ background: #feffde;
356
+ font-style: normal;
357
+ }
358
+
359
+ .select2-results .select2-highlighted em {
360
+ background: transparent;
361
+ }
362
+
363
+ .select2-results .select2-highlighted ul {
364
+ background: #fff;
365
+ color: #000;
366
+ }
367
+
368
+
369
+ .select2-results .select2-no-results,
370
+ .select2-results .select2-searching,
371
+ .select2-results .select2-selection-limit {
372
+ background: #f4f4f4;
373
+ display: list-item;
374
+ }
375
+
376
+ /*
377
+ disabled look for disabled choices in the results dropdown
378
+ */
379
+ .select2-results .select2-disabled.select2-highlighted {
380
+ color: #666;
381
+ background: #f4f4f4;
382
+ display: list-item;
383
+ cursor: default;
384
+ }
385
+ .select2-results .select2-disabled {
386
+ background: #f4f4f4;
387
+ display: list-item;
388
+ cursor: default;
389
+ }
390
+
391
+ .select2-results .select2-selected {
392
+ display: none;
393
+ }
394
+
395
+ .select2-more-results.select2-active {
396
+ background: #f4f4f4 url('../../img/select2/select2-spinner.gif') no-repeat 100%;
397
+ }
398
+
399
+ .select2-more-results {
400
+ background: #f4f4f4;
401
+ display: list-item;
402
+ }
403
+
404
+ /* disabled styles */
405
+
406
+ .select2-container.select2-container-disabled .select2-choice {
407
+ background-color: #f4f4f4;
408
+ background-image: none;
409
+ border: 1px solid #ddd;
410
+ cursor: default;
411
+ }
412
+
413
+ .select2-container.select2-container-disabled .select2-choice .select2-arrow {
414
+ background-color: #f4f4f4;
415
+ background-image: none;
416
+ border-left: 0;
417
+ }
418
+
419
+ .select2-container.select2-container-disabled .select2-choice abbr {
420
+ display: none;
421
+ }
422
+
423
+
424
+ /* multiselect */
425
+
426
+ .select2-container-multi .select2-choices {
427
+ height: auto !important;
428
+ height: 1%;
429
+ margin: 0;
430
+ padding: 0;
431
+ position: relative;
432
+
433
+ border: 1px solid #aaa;
434
+ cursor: text;
435
+ overflow: hidden;
436
+
437
+ background-color: #fff;
438
+ background-image: -webkit-gradient(linear, 0% 0%, 0% 100%, color-stop(1%, #eee), color-stop(15%, #fff));
439
+ background-image: -webkit-linear-gradient(top, #eee 1%, #fff 15%);
440
+ background-image: -moz-linear-gradient(top, #eee 1%, #fff 15%);
441
+ background-image: linear-gradient(top, #eee 1%, #fff 15%);
442
+ }
443
+
444
+ .select2-locked {
445
+ padding: 3px 5px 3px 5px !important;
446
+ }
447
+
448
+ .select2-container-multi .select2-choices {
449
+ min-height: 26px;
450
+ }
451
+
452
+ .select2-container-multi.select2-container-active .select2-choices {
453
+ border: 1px solid #5897fb;
454
+ outline: none;
455
+
456
+ -webkit-box-shadow: 0 0 5px rgba(0, 0, 0, .3);
457
+ box-shadow: 0 0 5px rgba(0, 0, 0, .3);
458
+ }
459
+ .select2-container-multi .select2-choices li {
460
+ float: left;
461
+ list-style: none;
462
+ }
463
+ .select2-container-multi .select2-choices .select2-search-field {
464
+ margin: 0;
465
+ padding: 0;
466
+ white-space: nowrap;
467
+ }
468
+
469
+ .select2-container-multi .select2-choices .select2-search-field input {
470
+ padding: 5px;
471
+ margin: 1px 0;
472
+
473
+ font-family: sans-serif;
474
+ font-size: 100%;
475
+ color: #666;
476
+ outline: 0;
477
+ border: 0;
478
+ -webkit-box-shadow: none;
479
+ box-shadow: none;
480
+ background: transparent !important;
481
+ }
482
+
483
+ .select2-container-multi .select2-choices .select2-search-field input.select2-active {
484
+ background: #fff url('../../img/select2/select2-spinner.gif') no-repeat 100% !important;
485
+ }
486
+
487
+ .select2-default {
488
+ color: #999 !important;
489
+ }
490
+
491
+ .select2-container-multi .select2-choices .select2-search-choice {
492
+ padding: 3px 5px 3px 18px;
493
+ margin: 3px 0 3px 5px;
494
+ position: relative;
495
+
496
+ line-height: 13px;
497
+ color: #333;
498
+ cursor: default;
499
+ border: 1px solid #aaaaaa;
500
+
501
+ border-radius: 3px;
502
+
503
+ -webkit-box-shadow: 0 0 2px #fff inset, 0 1px 0 rgba(0, 0, 0, 0.05);
504
+ box-shadow: 0 0 2px #fff inset, 0 1px 0 rgba(0, 0, 0, 0.05);
505
+
506
+ background-clip: padding-box;
507
+
508
+ -webkit-touch-callout: none;
509
+ -webkit-user-select: none;
510
+ -moz-user-select: none;
511
+ -ms-user-select: none;
512
+ user-select: none;
513
+
514
+ background-color: #e4e4e4;
515
+ filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#f4f4f4', GradientType=0);
516
+ background-image: -webkit-gradient(linear, 0% 0%, 0% 100%, color-stop(20%, #f4f4f4), color-stop(50%, #f0f0f0), color-stop(52%, #e8e8e8), color-stop(100%, #eee));
517
+ background-image: -webkit-linear-gradient(top, #f4f4f4 20%, #f0f0f0 50%, #e8e8e8 52%, #eee 100%);
518
+ background-image: -moz-linear-gradient(top, #f4f4f4 20%, #f0f0f0 50%, #e8e8e8 52%, #eee 100%);
519
+ background-image: linear-gradient(top, #f4f4f4 20%, #f0f0f0 50%, #e8e8e8 52%, #eee 100%);
520
+ }
521
+ .select2-container-multi .select2-choices .select2-search-choice .select2-chosen {
522
+ cursor: default;
523
+ }
524
+ .select2-container-multi .select2-choices .select2-search-choice-focus {
525
+ background: #d4d4d4;
526
+ }
527
+
528
+ .select2-search-choice-close {
529
+ display: block;
530
+ width: 12px;
531
+ height: 13px;
532
+ position: absolute;
533
+ right: 3px;
534
+ top: 4px;
535
+
536
+ font-size: 1px;
537
+ outline: none;
538
+ background: url('../../img/select2/select2.png') right top no-repeat;
539
+ }
540
+
541
+ .select2-container-multi .select2-search-choice-close {
542
+ left: 3px;
543
+ }
544
+
545
+ .select2-container-multi .select2-choices .select2-search-choice .select2-search-choice-close:hover {
546
+ background-position: right -11px;
547
+ }
548
+ .select2-container-multi .select2-choices .select2-search-choice-focus .select2-search-choice-close {
549
+ background-position: right -11px;
550
+ }
551
+
552
+ /* disabled styles */
553
+ .select2-container-multi.select2-container-disabled .select2-choices {
554
+ background-color: #f4f4f4;
555
+ background-image: none;
556
+ border: 1px solid #ddd;
557
+ cursor: default;
558
+ }
559
+
560
+ .select2-container-multi.select2-container-disabled .select2-choices .select2-search-choice {
561
+ padding: 3px 5px 3px 5px;
562
+ border: 1px solid #ddd;
563
+ background-image: none;
564
+ background-color: #f4f4f4;
565
+ }
566
+
567
+ .select2-container-multi.select2-container-disabled .select2-choices .select2-search-choice .select2-search-choice-close { display: none;
568
+ background: none;
569
+ }
570
+ /* end multiselect */
571
+
572
+
573
+ .select2-result-selectable .select2-match,
574
+ .select2-result-unselectable .select2-match {
575
+ text-decoration: underline;
576
+ }
577
+
578
+ .select2-offscreen, .select2-offscreen:focus {
579
+ clip: rect(0 0 0 0) !important;
580
+ width: 1px !important;
581
+ height: 1px !important;
582
+ border: 0 !important;
583
+ margin: 0 !important;
584
+ padding: 0 !important;
585
+ overflow: hidden !important;
586
+ position: absolute !important;
587
+ outline: 0 !important;
588
+ left: 0px !important;
589
+ top: 0px !important;
590
+ }
591
+
592
+ .select2-display-none {
593
+ display: none;
594
+ }
595
+
596
+ .select2-measure-scrollbar {
597
+ position: absolute;
598
+ top: -10000px;
599
+ left: -10000px;
600
+ width: 100px;
601
+ height: 100px;
602
+ overflow: scroll;
603
+ }
604
+ /* Retina-ize icons */
605
+
606
+ @media only screen and (-webkit-min-device-pixel-ratio: 1.5), only screen and (min-resolution: 144dpi) {
607
+ .select2-search input, .select2-search-choice-close, .select2-container .select2-choice abbr, .select2-container .select2-choice .select2-arrow b {
608
+ background-image: url('../../img/select2/select2x2.png') !important;
609
+ background-repeat: no-repeat !important;
610
+ background-size: 60px 40px !important;
611
+ }
612
+ .select2-search input {
613
+ background-position: 100% -21px !important;
614
+ }
615
+ }
css/smoothness/images/calendar.gif ADDED
Binary file
css/smoothness/images/index.html ADDED
File without changes
css/smoothness/images/ui-bg_flat_0_aaaaaa_40x100.png ADDED
Binary file
css/smoothness/images/ui-bg_flat_75_ffffff_40x100.png ADDED
Binary file
css/smoothness/images/ui-bg_glass_55_fbf9ee_1x400.png ADDED
Binary file
css/smoothness/images/ui-bg_glass_65_ffffff_1x400.png ADDED
Binary file
css/smoothness/images/ui-bg_glass_75_dadada_1x400.png ADDED
Binary file
css/smoothness/images/ui-bg_glass_75_e6e6e6_1x400.png ADDED
Binary file
css/smoothness/images/ui-bg_glass_95_fef1ec_1x400.png ADDED
Binary file
css/smoothness/images/ui-bg_highlight-soft_75_cccccc_1x100.png ADDED
Binary file
css/smoothness/images/ui-icons_222222_256x240.png ADDED
Binary file
css/smoothness/images/ui-icons_2e83ff_256x240.png ADDED
Binary file
css/smoothness/images/ui-icons_454545_256x240.png ADDED
Binary file
css/smoothness/images/ui-icons_888888_256x240.png ADDED
Binary file
css/smoothness/images/ui-icons_cd0a0a_256x240.png ADDED
Binary file
css/smoothness/index.html ADDED
File without changes
css/smoothness/jquery-ui-1.8.20.custom.css ADDED
@@ -0,0 +1,565 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ /*!
2
+ * jQuery UI CSS Framework 1.8.20
3
+ *
4
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
5
+ * Dual licensed under the MIT or GPL Version 2 licenses.
6
+ * http://jquery.org/license
7
+ *
8
+ * http://docs.jquery.com/UI/Theming/API
9
+ */
10
+
11
+ /* Layout helpers
12
+ ----------------------------------*/
13
+ .ui-helper-hidden { display: none; }
14
+ .ui-helper-hidden-accessible { position: absolute !important; clip: rect(1px 1px 1px 1px); clip: rect(1px,1px,1px,1px); }
15
+ .ui-helper-reset { margin: 0; padding: 0; border: 0; outline: 0; line-height: 1.3; text-decoration: none; font-size: 100%; list-style: none; }
16
+ .ui-helper-clearfix:before, .ui-helper-clearfix:after { content: ""; display: table; }
17
+ .ui-helper-clearfix:after { clear: both; }
18
+ .ui-helper-clearfix { zoom: 1; }
19
+ .ui-helper-zfix { width: 100%; height: 100%; top: 0; left: 0; position: absolute; opacity: 0; filter:Alpha(Opacity=0); }
20
+
21
+
22
+ /* Interaction Cues
23
+ ----------------------------------*/
24
+ .ui-state-disabled { cursor: default !important; }
25
+
26
+
27
+ /* Icons
28
+ ----------------------------------*/
29
+
30
+ /* states and images */
31
+ .ui-icon { display: block; text-indent: -99999px; overflow: hidden; background-repeat: no-repeat; }
32
+
33
+
34
+ /* Misc visuals
35
+ ----------------------------------*/
36
+
37
+ /* Overlays */
38
+ .ui-widget-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; }
39
+
40
+
41
+ /*!
42
+ * jQuery UI CSS Framework 1.8.20
43
+ *
44
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
45
+ * Dual licensed under the MIT or GPL Version 2 licenses.
46
+ * http://jquery.org/license
47
+ *
48
+ * http://docs.jquery.com/UI/Theming/API
49
+ *
50
+ * To view and modify this theme, visit http://jqueryui.com/themeroller/?ffDefault=Verdana,Arial,sans-serif&fwDefault=normal&fsDefault=1.1em&cornerRadius=4px&bgColorHeader=cccccc&bgTextureHeader=03_highlight_soft.png&bgImgOpacityHeader=75&borderColorHeader=aaaaaa&fcHeader=222222&iconColorHeader=222222&bgColorContent=ffffff&bgTextureContent=01_flat.png&bgImgOpacityContent=75&borderColorContent=aaaaaa&fcContent=222222&iconColorContent=222222&bgColorDefault=e6e6e6&bgTextureDefault=02_glass.png&bgImgOpacityDefault=75&borderColorDefault=d3d3d3&fcDefault=555555&iconColorDefault=888888&bgColorHover=dadada&bgTextureHover=02_glass.png&bgImgOpacityHover=75&borderColorHover=999999&fcHover=212121&iconColorHover=454545&bgColorActive=ffffff&bgTextureActive=02_glass.png&bgImgOpacityActive=65&borderColorActive=aaaaaa&fcActive=212121&iconColorActive=454545&bgColorHighlight=fbf9ee&bgTextureHighlight=02_glass.png&bgImgOpacityHighlight=55&borderColorHighlight=fcefa1&fcHighlight=363636&iconColorHighlight=2e83ff&bgColorError=fef1ec&bgTextureError=02_glass.png&bgImgOpacityError=95&borderColorError=cd0a0a&fcError=cd0a0a&iconColorError=cd0a0a&bgColorOverlay=aaaaaa&bgTextureOverlay=01_flat.png&bgImgOpacityOverlay=0&opacityOverlay=30&bgColorShadow=aaaaaa&bgTextureShadow=01_flat.png&bgImgOpacityShadow=0&opacityShadow=30&thicknessShadow=8px&offsetTopShadow=-8px&offsetLeftShadow=-8px&cornerRadiusShadow=8px
51
+ */
52
+
53
+
54
+ /* Component containers
55
+ ----------------------------------*/
56
+ .ui-widget { font-family: Verdana,Arial,sans-serif; font-size: 1em; }
57
+ .ui-widget .ui-widget { font-size: 1em; }
58
+ .ui-widget input, .ui-widget select, .ui-widget textarea, .ui-widget button { font-family: Verdana,Arial,sans-serif; font-size: 1em; }
59
+ .ui-widget-content { border: 1px solid #aaaaaa; background: #ffffff url(images/ui-bg_flat_75_ffffff_40x100.png) 50% 50% repeat-x; color: #222222; }
60
+ .ui-widget-content a { color: #222222; }
61
+ .ui-widget-header { border: 1px solid #aaaaaa; background: #cccccc url(images/ui-bg_highlight-soft_75_cccccc_1x100.png) 50% 50% repeat-x; color: #222222; font-weight: bold; }
62
+ .ui-widget-header a { color: #222222; }
63
+
64
+ /* Interaction states
65
+ ----------------------------------*/
66
+ .ui-state-default, .ui-widget-content .ui-state-default, .ui-widget-header .ui-state-default { border: 1px solid #d3d3d3; background: #e6e6e6 url(images/ui-bg_glass_75_e6e6e6_1x400.png) 50% 50% repeat-x; font-weight: normal; color: #555555; }
67
+ .ui-state-default a, .ui-state-default a:link, .ui-state-default a:visited { color: #555555; text-decoration: none; }
68
+ .ui-state-hover, .ui-widget-content .ui-state-hover, .ui-widget-header .ui-state-hover, .ui-state-focus, .ui-widget-content .ui-state-focus, .ui-widget-header .ui-state-focus { border: 1px solid #999999; background: #dadada url(images/ui-bg_glass_75_dadada_1x400.png) 50% 50% repeat-x; font-weight: normal; color: #212121; }
69
+ .ui-state-hover a, .ui-state-hover a:hover { color: #212121; text-decoration: none; }
70
+ .ui-state-active, .ui-widget-content .ui-state-active, .ui-widget-header .ui-state-active { border: 1px solid #aaaaaa; background: #ffffff url(images/ui-bg_glass_65_ffffff_1x400.png) 50% 50% repeat-x; font-weight: normal; color: #212121; }
71
+ .ui-state-active a, .ui-state-active a:link, .ui-state-active a:visited { color: #212121; text-decoration: none; }
72
+ .ui-widget :active { outline: none; }
73
+
74
+ /* Interaction Cues
75
+ ----------------------------------*/
76
+ .ui-state-highlight, .ui-widget-content .ui-state-highlight, .ui-widget-header .ui-state-highlight {border: 1px solid #fcefa1; background: #fbf9ee url(images/ui-bg_glass_55_fbf9ee_1x400.png) 50% 50% repeat-x; color: #363636; }
77
+ .ui-state-highlight a, .ui-widget-content .ui-state-highlight a,.ui-widget-header .ui-state-highlight a { color: #363636; }
78
+ .ui-state-error, .ui-widget-content .ui-state-error, .ui-widget-header .ui-state-error {border: 1px solid #cd0a0a; background: #fef1ec url(images/ui-bg_glass_95_fef1ec_1x400.png) 50% 50% repeat-x; color: #cd0a0a; }
79
+ .ui-state-error a, .ui-widget-content .ui-state-error a, .ui-widget-header .ui-state-error a { color: #cd0a0a; }
80
+ .ui-state-error-text, .ui-widget-content .ui-state-error-text, .ui-widget-header .ui-state-error-text { color: #cd0a0a; }
81
+ .ui-priority-primary, .ui-widget-content .ui-priority-primary, .ui-widget-header .ui-priority-primary { font-weight: bold; }
82
+ .ui-priority-secondary, .ui-widget-content .ui-priority-secondary, .ui-widget-header .ui-priority-secondary { opacity: .7; filter:Alpha(Opacity=70); font-weight: normal; }
83
+ .ui-state-disabled, .ui-widget-content .ui-state-disabled, .ui-widget-header .ui-state-disabled { opacity: .35; filter:Alpha(Opacity=35); background-image: none; }
84
+
85
+ /* Icons
86
+ ----------------------------------*/
87
+
88
+ /* states and images */
89
+ .ui-icon { width: 16px; height: 16px; background-image: url(images/ui-icons_222222_256x240.png); }
90
+ .ui-widget-content .ui-icon {background-image: url(images/ui-icons_222222_256x240.png); }
91
+ .ui-widget-header .ui-icon {background-image: url(images/ui-icons_222222_256x240.png); }
92
+ .ui-state-default .ui-icon { background-image: url(images/ui-icons_888888_256x240.png); }
93
+ .ui-state-hover .ui-icon, .ui-state-focus .ui-icon {background-image: url(images/ui-icons_454545_256x240.png); }
94
+ .ui-state-active .ui-icon {background-image: url(images/ui-icons_454545_256x240.png); }
95
+ .ui-state-highlight .ui-icon {background-image: url(images/ui-icons_2e83ff_256x240.png); }
96
+ .ui-state-error .ui-icon, .ui-state-error-text .ui-icon {background-image: url(images/ui-icons_cd0a0a_256x240.png); }
97
+
98
+ /* positioning */
99
+ .ui-icon-carat-1-n { background-position: 0 0; }
100
+ .ui-icon-carat-1-ne { background-position: -16px 0; }
101
+ .ui-icon-carat-1-e { background-position: -32px 0; }
102
+ .ui-icon-carat-1-se { background-position: -48px 0; }
103
+ .ui-icon-carat-1-s { background-position: -64px 0; }
104
+ .ui-icon-carat-1-sw { background-position: -80px 0; }
105
+ .ui-icon-carat-1-w { background-position: -96px 0; }
106
+ .ui-icon-carat-1-nw { background-position: -112px 0; }
107
+ .ui-icon-carat-2-n-s { background-position: -128px 0; }
108
+ .ui-icon-carat-2-e-w { background-position: -144px 0; }
109
+ .ui-icon-triangle-1-n { background-position: 0 -16px; }
110
+ .ui-icon-triangle-1-ne { background-position: -16px -16px; }
111
+ .ui-icon-triangle-1-e { background-position: -32px -16px; }
112
+ .ui-icon-triangle-1-se { background-position: -48px -16px; }
113
+ .ui-icon-triangle-1-s { background-position: -64px -16px; }
114
+ .ui-icon-triangle-1-sw { background-position: -80px -16px; }
115
+ .ui-icon-triangle-1-w { background-position: -96px -16px; }
116
+ .ui-icon-triangle-1-nw { background-position: -112px -16px; }
117
+ .ui-icon-triangle-2-n-s { background-position: -128px -16px; }
118
+ .ui-icon-triangle-2-e-w { background-position: -144px -16px; }
119
+ .ui-icon-arrow-1-n { background-position: 0 -32px; }
120
+ .ui-icon-arrow-1-ne { background-position: -16px -32px; }
121
+ .ui-icon-arrow-1-e { background-position: -32px -32px; }
122
+ .ui-icon-arrow-1-se { background-position: -48px -32px; }
123
+ .ui-icon-arrow-1-s { background-position: -64px -32px; }
124
+ .ui-icon-arrow-1-sw { background-position: -80px -32px; }
125
+ .ui-icon-arrow-1-w { background-position: -96px -32px; }
126
+ .ui-icon-arrow-1-nw { background-position: -112px -32px; }
127
+ .ui-icon-arrow-2-n-s { background-position: -128px -32px; }
128
+ .ui-icon-arrow-2-ne-sw { background-position: -144px -32px; }
129
+ .ui-icon-arrow-2-e-w { background-position: -160px -32px; }
130
+ .ui-icon-arrow-2-se-nw { background-position: -176px -32px; }
131
+ .ui-icon-arrowstop-1-n { background-position: -192px -32px; }
132
+ .ui-icon-arrowstop-1-e { background-position: -208px -32px; }
133
+ .ui-icon-arrowstop-1-s { background-position: -224px -32px; }
134
+ .ui-icon-arrowstop-1-w { background-position: -240px -32px; }
135
+ .ui-icon-arrowthick-1-n { background-position: 0 -48px; }
136
+ .ui-icon-arrowthick-1-ne { background-position: -16px -48px; }
137
+ .ui-icon-arrowthick-1-e { background-position: -32px -48px; }
138
+ .ui-icon-arrowthick-1-se { background-position: -48px -48px; }
139
+ .ui-icon-arrowthick-1-s { background-position: -64px -48px; }
140
+ .ui-icon-arrowthick-1-sw { background-position: -80px -48px; }
141
+ .ui-icon-arrowthick-1-w { background-position: -96px -48px; }
142
+ .ui-icon-arrowthick-1-nw { background-position: -112px -48px; }
143
+ .ui-icon-arrowthick-2-n-s { background-position: -128px -48px; }
144
+ .ui-icon-arrowthick-2-ne-sw { background-position: -144px -48px; }
145
+ .ui-icon-arrowthick-2-e-w { background-position: -160px -48px; }
146
+ .ui-icon-arrowthick-2-se-nw { background-position: -176px -48px; }
147
+ .ui-icon-arrowthickstop-1-n { background-position: -192px -48px; }
148
+ .ui-icon-arrowthickstop-1-e { background-position: -208px -48px; }
149
+ .ui-icon-arrowthickstop-1-s { background-position: -224px -48px; }
150
+ .ui-icon-arrowthickstop-1-w { background-position: -240px -48px; }
151
+ .ui-icon-arrowreturnthick-1-w { background-position: 0 -64px; }
152
+ .ui-icon-arrowreturnthick-1-n { background-position: -16px -64px; }
153
+ .ui-icon-arrowreturnthick-1-e { background-position: -32px -64px; }
154
+ .ui-icon-arrowreturnthick-1-s { background-position: -48px -64px; }
155
+ .ui-icon-arrowreturn-1-w { background-position: -64px -64px; }
156
+ .ui-icon-arrowreturn-1-n { background-position: -80px -64px; }
157
+ .ui-icon-arrowreturn-1-e { background-position: -96px -64px; }
158
+ .ui-icon-arrowreturn-1-s { background-position: -112px -64px; }
159
+ .ui-icon-arrowrefresh-1-w { background-position: -128px -64px; }
160
+ .ui-icon-arrowrefresh-1-n { background-position: -144px -64px; }
161
+ .ui-icon-arrowrefresh-1-e { background-position: -160px -64px; }
162
+ .ui-icon-arrowrefresh-1-s { background-position: -176px -64px; }
163
+ .ui-icon-arrow-4 { background-position: 0 -80px; }
164
+ .ui-icon-arrow-4-diag { background-position: -16px -80px; }
165
+ .ui-icon-extlink { background-position: -32px -80px; }
166
+ .ui-icon-newwin { background-position: -48px -80px; }
167
+ .ui-icon-refresh { background-position: -64px -80px; }
168
+ .ui-icon-shuffle { background-position: -80px -80px; }
169
+ .ui-icon-transfer-e-w { background-position: -96px -80px; }
170
+ .ui-icon-transferthick-e-w { background-position: -112px -80px; }
171
+ .ui-icon-folder-collapsed { background-position: 0 -96px; }
172
+ .ui-icon-folder-open { background-position: -16px -96px; }
173
+ .ui-icon-document { background-position: -32px -96px; }
174
+ .ui-icon-document-b { background-position: -48px -96px; }
175
+ .ui-icon-note { background-position: -64px -96px; }
176
+ .ui-icon-mail-closed { background-position: -80px -96px; }
177
+ .ui-icon-mail-open { background-position: -96px -96px; }
178
+ .ui-icon-suitcase { background-position: -112px -96px; }
179
+ .ui-icon-comment { background-position: -128px -96px; }
180
+ .ui-icon-person { background-position: -144px -96px; }
181
+ .ui-icon-print { background-position: -160px -96px; }
182
+ .ui-icon-trash { background-position: -176px -96px; }
183
+ .ui-icon-locked { background-position: -192px -96px; }
184
+ .ui-icon-unlocked { background-position: -208px -96px; }
185
+ .ui-icon-bookmark { background-position: -224px -96px; }
186
+ .ui-icon-tag { background-position: -240px -96px; }
187
+ .ui-icon-home { background-position: 0 -112px; }
188
+ .ui-icon-flag { background-position: -16px -112px; }
189
+ .ui-icon-calendar { background-position: -32px -112px; }
190
+ .ui-icon-cart { background-position: -48px -112px; }
191
+ .ui-icon-pencil { background-position: -64px -112px; }
192
+ .ui-icon-clock { background-position: -80px -112px; }
193
+ .ui-icon-disk { background-position: -96px -112px; }
194
+ .ui-icon-calculator { background-position: -112px -112px; }
195
+ .ui-icon-zoomin { background-position: -128px -112px; }
196
+ .ui-icon-zoomout { background-position: -144px -112px; }
197
+ .ui-icon-search { background-position: -160px -112px; }
198
+ .ui-icon-wrench { background-position: -176px -112px; }
199
+ .ui-icon-gear { background-position: -192px -112px; }
200
+ .ui-icon-heart { background-position: -208px -112px; }
201
+ .ui-icon-star { background-position: -224px -112px; }
202
+ .ui-icon-link { background-position: -240px -112px; }
203
+ .ui-icon-cancel { background-position: 0 -128px; }
204
+ .ui-icon-plus { background-position: -16px -128px; }
205
+ .ui-icon-plusthick { background-position: -32px -128px; }
206
+ .ui-icon-minus { background-position: -48px -128px; }
207
+ .ui-icon-minusthick { background-position: -64px -128px; }
208
+ .ui-icon-close { background-position: -80px -128px; }
209
+ .ui-icon-closethick { background-position: -96px -128px; }
210
+ .ui-icon-key { background-position: -112px -128px; }
211
+ .ui-icon-lightbulb { background-position: -128px -128px; }
212
+ .ui-icon-scissors { background-position: -144px -128px; }
213
+ .ui-icon-clipboard { background-position: -160px -128px; }
214
+ .ui-icon-copy { background-position: -176px -128px; }
215
+ .ui-icon-contact { background-position: -192px -128px; }
216
+ .ui-icon-image { background-position: -208px -128px; }
217
+ .ui-icon-video { background-position: -224px -128px; }
218
+ .ui-icon-script { background-position: -240px -128px; }
219
+ .ui-icon-alert { background-position: 0 -144px; }
220
+ .ui-icon-info { background-position: -16px -144px; }
221
+ .ui-icon-notice { background-position: -32px -144px; }
222
+ .ui-icon-help { background-position: -48px -144px; }
223
+ .ui-icon-check { background-position: -64px -144px; }
224
+ .ui-icon-bullet { background-position: -80px -144px; }
225
+ .ui-icon-radio-off { background-position: -96px -144px; }
226
+ .ui-icon-radio-on { background-position: -112px -144px; }
227
+ .ui-icon-pin-w { background-position: -128px -144px; }
228
+ .ui-icon-pin-s { background-position: -144px -144px; }
229
+ .ui-icon-play { background-position: 0 -160px; }
230
+ .ui-icon-pause { background-position: -16px -160px; }
231
+ .ui-icon-seek-next { background-position: -32px -160px; }
232
+ .ui-icon-seek-prev { background-position: -48px -160px; }
233
+ .ui-icon-seek-end { background-position: -64px -160px; }
234
+ .ui-icon-seek-start { background-position: -80px -160px; }
235
+ /* ui-icon-seek-first is deprecated, use ui-icon-seek-start instead */
236
+ .ui-icon-seek-first { background-position: -80px -160px; }
237
+ .ui-icon-stop { background-position: -96px -160px; }
238
+ .ui-icon-eject { background-position: -112px -160px; }
239
+ .ui-icon-volume-off { background-position: -128px -160px; }
240
+ .ui-icon-volume-on { background-position: -144px -160px; }
241
+ .ui-icon-power { background-position: 0 -176px; }
242
+ .ui-icon-signal-diag { background-position: -16px -176px; }
243
+ .ui-icon-signal { background-position: -32px -176px; }
244
+ .ui-icon-battery-0 { background-position: -48px -176px; }
245
+ .ui-icon-battery-1 { background-position: -64px -176px; }
246
+ .ui-icon-battery-2 { background-position: -80px -176px; }
247
+ .ui-icon-battery-3 { background-position: -96px -176px; }
248
+ .ui-icon-circle-plus { background-position: 0 -192px; }
249
+ .ui-icon-circle-minus { background-position: -16px -192px; }
250
+ .ui-icon-circle-close { background-position: -32px -192px; }
251
+ .ui-icon-circle-triangle-e { background-position: -48px -192px; }
252
+ .ui-icon-circle-triangle-s { background-position: -64px -192px; }
253
+ .ui-icon-circle-triangle-w { background-position: -80px -192px; }
254
+ .ui-icon-circle-triangle-n { background-position: -96px -192px; }
255
+ .ui-icon-circle-arrow-e { background-position: -112px -192px; }
256
+ .ui-icon-circle-arrow-s { background-position: -128px -192px; }
257
+ .ui-icon-circle-arrow-w { background-position: -144px -192px; }
258
+ .ui-icon-circle-arrow-n { background-position: -160px -192px; }
259
+ .ui-icon-circle-zoomin { background-position: -176px -192px; }
260
+ .ui-icon-circle-zoomout { background-position: -192px -192px; }
261
+ .ui-icon-circle-check { background-position: -208px -192px; }
262
+ .ui-icon-circlesmall-plus { background-position: 0 -208px; }
263
+ .ui-icon-circlesmall-minus { background-position: -16px -208px; }
264
+ .ui-icon-circlesmall-close { background-position: -32px -208px; }
265
+ .ui-icon-squaresmall-plus { background-position: -48px -208px; }
266
+ .ui-icon-squaresmall-minus { background-position: -64px -208px; }
267
+ .ui-icon-squaresmall-close { background-position: -80px -208px; }
268
+ .ui-icon-grip-dotted-vertical { background-position: 0 -224px; }
269
+ .ui-icon-grip-dotted-horizontal { background-position: -16px -224px; }
270
+ .ui-icon-grip-solid-vertical { background-position: -32px -224px; }
271
+ .ui-icon-grip-solid-horizontal { background-position: -48px -224px; }
272
+ .ui-icon-gripsmall-diagonal-se { background-position: -64px -224px; }
273
+ .ui-icon-grip-diagonal-se { background-position: -80px -224px; }
274
+
275
+
276
+ /* Misc visuals
277
+ ----------------------------------*/
278
+
279
+ /* Corner radius */
280
+ .ui-corner-all, .ui-corner-top, .ui-corner-left, .ui-corner-tl { -moz-border-radius-topleft: 4px; -webkit-border-top-left-radius: 4px; -khtml-border-top-left-radius: 4px; border-top-left-radius: 4px; }
281
+ .ui-corner-all, .ui-corner-top, .ui-corner-right, .ui-corner-tr { -moz-border-radius-topright: 4px; -webkit-border-top-right-radius: 4px; -khtml-border-top-right-radius: 4px; border-top-right-radius: 4px; }
282
+ .ui-corner-all, .ui-corner-bottom, .ui-corner-left, .ui-corner-bl { -moz-border-radius-bottomleft: 4px; -webkit-border-bottom-left-radius: 4px; -khtml-border-bottom-left-radius: 4px; border-bottom-left-radius: 4px; }
283
+ .ui-corner-all, .ui-corner-bottom, .ui-corner-right, .ui-corner-br { -moz-border-radius-bottomright: 4px; -webkit-border-bottom-right-radius: 4px; -khtml-border-bottom-right-radius: 4px; border-bottom-right-radius: 4px; }
284
+
285
+ /* Overlays */
286
+ .ui-widget-overlay { background: #aaaaaa url(images/ui-bg_flat_0_aaaaaa_40x100.png) 50% 50% repeat-x; opacity: .30;filter:Alpha(Opacity=30); }
287
+ .ui-widget-shadow { margin: -8px 0 0 -8px; padding: 8px; background: #aaaaaa url(images/ui-bg_flat_0_aaaaaa_40x100.png) 50% 50% repeat-x; opacity: .30;filter:Alpha(Opacity=30); -moz-border-radius: 8px; -khtml-border-radius: 8px; -webkit-border-radius: 8px; border-radius: 8px; }/*!
288
+ * jQuery UI Resizable 1.8.20
289
+ *
290
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
291
+ * Dual licensed under the MIT or GPL Version 2 licenses.
292
+ * http://jquery.org/license
293
+ *
294
+ * http://docs.jquery.com/UI/Resizable#theming
295
+ */
296
+ .ui-resizable { position: relative;}
297
+ .ui-resizable-handle { position: absolute;font-size: 0.1px; display: block; }
298
+ .ui-resizable-disabled .ui-resizable-handle, .ui-resizable-autohide .ui-resizable-handle { display: none; }
299
+ .ui-resizable-n { cursor: n-resize; height: 7px; width: 100%; top: -5px; left: 0; }
300
+ .ui-resizable-s { cursor: s-resize; height: 7px; width: 100%; bottom: -5px; left: 0; }
301
+ .ui-resizable-e { cursor: e-resize; width: 7px; right: -5px; top: 0; height: 100%; }
302
+ .ui-resizable-w { cursor: w-resize; width: 7px; left: -5px; top: 0; height: 100%; }
303
+ .ui-resizable-se { cursor: se-resize; width: 12px; height: 12px; right: 1px; bottom: 1px; }
304
+ .ui-resizable-sw { cursor: sw-resize; width: 9px; height: 9px; left: -5px; bottom: -5px; }
305
+ .ui-resizable-nw { cursor: nw-resize; width: 9px; height: 9px; left: -5px; top: -5px; }
306
+ .ui-resizable-ne { cursor: ne-resize; width: 9px; height: 9px; right: -5px; top: -5px;}/*!
307
+ * jQuery UI Selectable 1.8.20
308
+ *
309
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
310
+ * Dual licensed under the MIT or GPL Version 2 licenses.
311
+ * http://jquery.org/license
312
+ *
313
+ * http://docs.jquery.com/UI/Selectable#theming
314
+ */
315
+ .ui-selectable-helper { position: absolute; z-index: 100; border:1px dotted black; }
316
+ /*!
317
+ * jQuery UI Accordion 1.8.20
318
+ *
319
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
320
+ * Dual licensed under the MIT or GPL Version 2 licenses.
321
+ * http://jquery.org/license
322
+ *
323
+ * http://docs.jquery.com/UI/Accordion#theming
324
+ */
325
+ /* IE/Win - Fix animation bug - #4615 */
326
+ .ui-accordion { width: 100%; }
327
+ .ui-accordion .ui-accordion-header { cursor: pointer; position: relative; margin-top: 1px; zoom: 1; }
328
+ .ui-accordion .ui-accordion-li-fix { display: inline; }
329
+ .ui-accordion .ui-accordion-header-active { border-bottom: 0 !important; }
330
+ .ui-accordion .ui-accordion-header a { display: block; font-size: 1em; padding: .5em .5em .5em .7em; }
331
+ .ui-accordion-icons .ui-accordion-header a { padding-left: 2.2em; }
332
+ .ui-accordion .ui-accordion-header .ui-icon { position: absolute; left: .5em; top: 50%; margin-top: -8px; }
333
+ .ui-accordion .ui-accordion-content { padding: 1em 2.2em; border-top: 0; margin-top: -2px; position: relative; top: 1px; margin-bottom: 2px; overflow: auto; display: none; zoom: 1; }
334
+ .ui-accordion .ui-accordion-content-active { display: block; }
335
+ /*!
336
+ * jQuery UI Autocomplete 1.8.20
337
+ *
338
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
339
+ * Dual licensed under the MIT or GPL Version 2 licenses.
340
+ * http://jquery.org/license
341
+ *
342
+ * http://docs.jquery.com/UI/Autocomplete#theming
343
+ */
344
+ .ui-autocomplete { position: absolute; cursor: default; }
345
+
346
+ /* workarounds */
347
+ * html .ui-autocomplete { width:1px; } /* without this, the menu expands to 100% in IE6 */
348
+
349
+ /*
350
+ * jQuery UI Menu 1.8.20
351
+ *
352
+ * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about)
353
+ * Dual licensed under the MIT or GPL Version 2 licenses.
354
+ * http://jquery.org/license
355
+ *
356
+ * http://docs.jquery.com/UI/Menu#theming
357
+ */
358
+ .ui-menu {
359
+ list-style:none;
360
+ padding: 2px;
361
+ margin: 0;
362
+ display:block;
363
+ float: left;
364
+ }
365
+ .ui-menu .ui-menu {
366
+ margin-top: -3px;
367
+ }
368
+ .ui-menu .ui-menu-item {
369
+ margin:0;
370
+ padding: 0;
371
+ zoom: 1;
372
+ float: left;
373
+ clear: left;
374
+ width: 100%;
375
+ }
376
+ .ui-menu .ui-menu-item a {
377
+ text-decoration:none;
378
+ display:block;
379
+ padding:.2em .4em;
380
+ line-height:1.5;
381
+ zoom:1;
382
+ }
383
+ .ui-menu .ui-menu-item a.ui-state-hover,
384
+ .ui-menu .ui-menu-item a.ui-state-active {
385
+ font-weight: normal;
386
+ margin: -1px;
387
+ }
388
+ /*!
389
+ * jQuery UI Button 1.8.20
390
+ *
391
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
392
+ * Dual licensed under the MIT or GPL Version 2 licenses.
393
+ * http://jquery.org/license
394
+ *
395
+ * http://docs.jquery.com/UI/Button#theming
396
+ */
397
+ .ui-button { display: inline-block; position: relative; padding: 0; margin-right: .1em; text-decoration: none !important; cursor: pointer; text-align: center; zoom: 1; overflow: visible; } /* the overflow property removes extra width in IE */
398
+ .ui-button-icon-only { width: 2.2em; } /* to make room for the icon, a width needs to be set here */
399
+ button.ui-button-icon-only { width: 2.4em; } /* button elements seem to need a little more width */
400
+ .ui-button-icons-only { width: 3.4em; }
401
+ button.ui-button-icons-only { width: 3.7em; }
402
+
403
+ /*button text element */
404
+ .ui-button .ui-button-text { display: block; line-height: 1.4; }
405
+ .ui-button-text-only .ui-button-text { padding: .4em 1em; }
406
+ .ui-button-icon-only .ui-button-text, .ui-button-icons-only .ui-button-text { padding: .4em; text-indent: -9999999px; }
407
+ .ui-button-text-icon-primary .ui-button-text, .ui-button-text-icons .ui-button-text { padding: .4em 1em .4em 2.1em; }
408
+ .ui-button-text-icon-secondary .ui-button-text, .ui-button-text-icons .ui-button-text { padding: .4em 2.1em .4em 1em; }
409
+ .ui-button-text-icons .ui-button-text { padding-left: 2.1em; padding-right: 2.1em; }
410
+ /* no icon support for input elements, provide padding by default */
411
+ input.ui-button { padding: .4em 1em; }
412
+
413
+ /*button icon element(s) */
414
+ .ui-button-icon-only .ui-icon, .ui-button-text-icon-primary .ui-icon, .ui-button-text-icon-secondary .ui-icon, .ui-button-text-icons .ui-icon, .ui-button-icons-only .ui-icon { position: absolute; top: 50%; margin-top: -8px; }
415
+ .ui-button-icon-only .ui-icon { left: 50%; margin-left: -8px; }
416
+ .ui-button-text-icon-primary .ui-button-icon-primary, .ui-button-text-icons .ui-button-icon-primary, .ui-button-icons-only .ui-button-icon-primary { left: .5em; }
417
+ .ui-button-text-icon-secondary .ui-button-icon-secondary, .ui-button-text-icons .ui-button-icon-secondary, .ui-button-icons-only .ui-button-icon-secondary { right: .5em; }
418
+ .ui-button-text-icons .ui-button-icon-secondary, .ui-button-icons-only .ui-button-icon-secondary { right: .5em; }
419
+
420
+ /*button sets*/
421
+ .ui-buttonset { margin-right: 7px; }
422
+ .ui-buttonset .ui-button { margin-left: 0; margin-right: -.3em; }
423
+
424
+ /* workarounds */
425
+ button.ui-button::-moz-focus-inner { border: 0; padding: 0; } /* reset extra padding in Firefox */
426
+ /*!
427
+ * jQuery UI Dialog 1.8.20
428
+ *
429
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
430
+ * Dual licensed under the MIT or GPL Version 2 licenses.
431
+ * http://jquery.org/license
432
+ *
433
+ * http://docs.jquery.com/UI/Dialog#theming
434
+ */
435
+ .ui-dialog { position: absolute; padding: .2em; width: 300px; overflow: hidden; }
436
+ .ui-dialog .ui-dialog-titlebar { padding: .4em 1em; position: relative; }
437
+ .ui-dialog .ui-dialog-title { float: left; margin: .1em 16px .1em 0; }
438
+ .ui-dialog .ui-dialog-titlebar-close { position: absolute; right: .3em; top: 50%; width: 19px; margin: -10px 0 0 0; padding: 1px; height: 18px; }
439
+ .ui-dialog .ui-dialog-titlebar-close span { display: block; }
440
+ .ui-dialog .ui-dialog-titlebar-close:hover, .ui-dialog .ui-dialog-titlebar-close:focus { padding: 0; }
441
+ .ui-dialog .ui-dialog-content { position: relative; border: 0; padding: .5em 1em; background: none; overflow: auto; zoom: 1; }
442
+ .ui-dialog .ui-dialog-buttonpane { text-align: left; border-width: 1px 0 0 0; background-image: none; margin: .5em 0 0 0; padding: .3em 1em .5em .4em; }
443
+ .ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset { float: right; }
444
+ .ui-dialog .ui-dialog-buttonpane button { margin: .5em .4em .5em 0; cursor: pointer; }
445
+ .ui-dialog .ui-resizable-se { width: 14px; height: 14px; right: 3px; bottom: 3px; }
446
+ .ui-draggable .ui-dialog-titlebar { cursor: move; }
447
+ /*!
448
+ * jQuery UI Slider 1.8.20
449
+ *
450
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
451
+ * Dual licensed under the MIT or GPL Version 2 licenses.
452
+ * http://jquery.org/license
453
+ *
454
+ * http://docs.jquery.com/UI/Slider#theming
455
+ */
456
+ .ui-slider { position: relative; text-align: left; }
457
+ .ui-slider .ui-slider-handle { position: absolute; z-index: 2; width: 1.2em; height: 1.2em; cursor: default; }
458
+ .ui-slider .ui-slider-range { position: absolute; z-index: 1; font-size: .7em; display: block; border: 0; background-position: 0 0; }
459
+
460
+ .ui-slider-horizontal { height: .8em; }
461
+ .ui-slider-horizontal .ui-slider-handle { top: -.3em; margin-left: -.6em; }
462
+ .ui-slider-horizontal .ui-slider-range { top: 0; height: 100%; }
463
+ .ui-slider-horizontal .ui-slider-range-min { left: 0; }
464
+ .ui-slider-horizontal .ui-slider-range-max { right: 0; }
465
+
466
+ .ui-slider-vertical { width: .8em; height: 100px; }
467
+ .ui-slider-vertical .ui-slider-handle { left: -.3em; margin-left: 0; margin-bottom: -.6em; }
468
+ .ui-slider-vertical .ui-slider-range { left: 0; width: 100%; }
469
+ .ui-slider-vertical .ui-slider-range-min { bottom: 0; }
470
+ .ui-slider-vertical .ui-slider-range-max { top: 0; }/*!
471
+ * jQuery UI Tabs 1.8.20
472
+ *
473
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
474
+ * Dual licensed under the MIT or GPL Version 2 licenses.
475
+ * http://jquery.org/license
476
+ *
477
+ * http://docs.jquery.com/UI/Tabs#theming
478
+ */
479
+ .ui-tabs { position: relative; padding: .2em; zoom: 1; } /* position: relative prevents IE scroll bug (element with position: relative inside container with overflow: auto appear as "fixed") */
480
+ .ui-tabs .ui-tabs-nav { margin: 0; padding: .2em .2em 0; }
481
+ .ui-tabs .ui-tabs-nav li { list-style: none; float: left; position: relative; top: 1px; margin: 0 .2em 1px 0; border-bottom: 0 !important; padding: 0; white-space: nowrap; }
482
+ .ui-tabs .ui-tabs-nav li a { float: left; padding: .5em 1em; text-decoration: none; }
483
+ .ui-tabs .ui-tabs-nav li.ui-tabs-selected { margin-bottom: 0; padding-bottom: 1px; }
484
+ .ui-tabs .ui-tabs-nav li.ui-tabs-selected a, .ui-tabs .ui-tabs-nav li.ui-state-disabled a, .ui-tabs .ui-tabs-nav li.ui-state-processing a { cursor: text; }
485
+ .ui-tabs .ui-tabs-nav li a, .ui-tabs.ui-tabs-collapsible .ui-tabs-nav li.ui-tabs-selected a { cursor: pointer; } /* first selector in group seems obsolete, but required to overcome bug in Opera applying cursor: text overall if defined elsewhere... */
486
+ .ui-tabs .ui-tabs-panel { display: block; border-width: 0; padding: 1em 1.4em; background: none; }
487
+ .ui-tabs .ui-tabs-hide { display: none !important; }
488
+ /*!
489
+ * jQuery UI Datepicker 1.8.20
490
+ *
491
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
492
+ * Dual licensed under the MIT or GPL Version 2 licenses.
493
+ * http://jquery.org/license
494
+ *
495
+ * http://docs.jquery.com/UI/Datepicker#theming
496
+ */
497
+ .ui-datepicker { width: 17em; padding: .2em .2em 0; display: none; }
498
+ .ui-datepicker .ui-datepicker-header { position:relative; padding:.2em 0; }
499
+ .ui-datepicker .ui-datepicker-prev, .ui-datepicker .ui-datepicker-next { position:absolute; top: 2px; width: 1.8em; height: 1.8em; }
500
+ .ui-datepicker .ui-datepicker-prev-hover, .ui-datepicker .ui-datepicker-next-hover { top: 1px; }
501
+ .ui-datepicker .ui-datepicker-prev { left:2px; }
502
+ .ui-datepicker .ui-datepicker-next { right:2px; }
503
+ .ui-datepicker .ui-datepicker-prev-hover { left:1px; }
504
+ .ui-datepicker .ui-datepicker-next-hover { right:1px; }
505
+ .ui-datepicker .ui-datepicker-prev span, .ui-datepicker .ui-datepicker-next span { display: block; position: absolute; left: 50%; margin-left: -8px; top: 50%; margin-top: -8px; }
506
+ .ui-datepicker .ui-datepicker-title { margin: 0 2.3em; line-height: 1.8em; text-align: center; }
507
+ .ui-datepicker .ui-datepicker-title select { font-size:1em; margin:1px 0; }
508
+ .ui-datepicker select.ui-datepicker-month-year {width: 100%;}
509
+ .ui-datepicker select.ui-datepicker-month,
510
+ .ui-datepicker select.ui-datepicker-year { width: 49%;}
511
+ .ui-datepicker table {width: 100%; font-size: .9em; border-collapse: collapse; margin:0 0 .4em; }
512
+ .ui-datepicker th { padding: .7em .3em; text-align: center; font-weight: bold; border: 0; }
513
+ .ui-datepicker td { border: 0; padding: 1px; }
514
+ .ui-datepicker td span, .ui-datepicker td a { display: block; padding: .2em; text-align: right; text-decoration: none; }
515
+ .ui-datepicker .ui-datepicker-buttonpane { background-image: none; margin: .7em 0 0 0; padding:0 .2em; border-left: 0; border-right: 0; border-bottom: 0; }
516
+ .ui-datepicker .ui-datepicker-buttonpane button { float: right; margin: .5em .2em .4em; cursor: pointer; padding: .2em .6em .3em .6em; width:auto; overflow:visible; }
517
+ .ui-datepicker .ui-datepicker-buttonpane button.ui-datepicker-current { float:left; }
518
+
519
+ /* with multiple calendars */
520
+ .ui-datepicker.ui-datepicker-multi { width:auto; }
521
+ .ui-datepicker-multi .ui-datepicker-group { float:left; }
522
+ .ui-datepicker-multi .ui-datepicker-group table { width:95%; margin:0 auto .4em; }
523
+ .ui-datepicker-multi-2 .ui-datepicker-group { width:50%; }
524
+ .ui-datepicker-multi-3 .ui-datepicker-group { width:33.3%; }
525
+ .ui-datepicker-multi-4 .ui-datepicker-group { width:25%; }
526
+ .ui-datepicker-multi .ui-datepicker-group-last .ui-datepicker-header { border-left-width:0; }
527
+ .ui-datepicker-multi .ui-datepicker-group-middle .ui-datepicker-header { border-left-width:0; }
528
+ .ui-datepicker-multi .ui-datepicker-buttonpane { clear:left; }
529
+ .ui-datepicker-row-break { clear:both; width:100%; font-size:0em; }
530
+
531
+ /* RTL support */
532
+ .ui-datepicker-rtl { direction: rtl; }
533
+ .ui-datepicker-rtl .ui-datepicker-prev { right: 2px; left: auto; }
534
+ .ui-datepicker-rtl .ui-datepicker-next { left: 2px; right: auto; }
535
+ .ui-datepicker-rtl .ui-datepicker-prev:hover { right: 1px; left: auto; }
536
+ .ui-datepicker-rtl .ui-datepicker-next:hover { left: 1px; right: auto; }
537
+ .ui-datepicker-rtl .ui-datepicker-buttonpane { clear:right; }
538
+ .ui-datepicker-rtl .ui-datepicker-buttonpane button { float: left; }
539
+ .ui-datepicker-rtl .ui-datepicker-buttonpane button.ui-datepicker-current { float:right; }
540
+ .ui-datepicker-rtl .ui-datepicker-group { float:right; }
541
+ .ui-datepicker-rtl .ui-datepicker-group-last .ui-datepicker-header { border-right-width:0; border-left-width:1px; }
542
+ .ui-datepicker-rtl .ui-datepicker-group-middle .ui-datepicker-header { border-right-width:0; border-left-width:1px; }
543
+
544
+ /* IE6 IFRAME FIX (taken from datepicker 1.5.3 */
545
+ .ui-datepicker-cover {
546
+ display: none; /*sorry for IE5*/
547
+ display/**/: block; /*sorry for IE5*/
548
+ position: absolute; /*must have*/
549
+ z-index: -1; /*must have*/
550
+ filter: mask(); /*must have*/
551
+ top: -4px; /*must have*/
552
+ left: -4px; /*must have*/
553
+ width: 200px; /*must have*/
554
+ height: 200px; /*must have*/
555
+ }/*!
556
+ * jQuery UI Progressbar 1.8.20
557
+ *
558
+ * Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
559
+ * Dual licensed under the MIT or GPL Version 2 licenses.
560
+ * http://jquery.org/license
561
+ *
562
+ * http://docs.jquery.com/UI/Progressbar#theming
563
+ */
564
+ .ui-progressbar { height:2em; text-align: left; overflow: hidden; }
565
+ .ui-progressbar .ui-progressbar-value {margin: -1px; height:100%; }
css/tmce/editor.css ADDED
@@ -0,0 +1,2317 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+
2
+ /*------------------------------------------------------------------------------
3
+
4
+ TinyMCE and Quicklinks toolbars
5
+ ------------------------------------------------------------------------------*/
6
+
7
+ /* wp_theme/ui.css */
8
+ .wp_themeSkin table,
9
+ .wp_themeSkin tbody,
10
+ .wp_themeSkin a,
11
+ .wp_themeSkin img,
12
+ .wp_themeSkin tr,
13
+ .wp_themeSkin div,
14
+ .wp_themeSkin td,
15
+ .wp_themeSkin iframe,
16
+ .wp_themeSkin span,
17
+ .wp_themeSkin *,
18
+ .wp_themeSkin .mceText {
19
+ border: 0;
20
+ margin: 0;
21
+ padding: 0;
22
+ white-space: nowrap;
23
+ text-decoration: none;
24
+ font-weight: normal;
25
+ cursor: default;
26
+ vertical-align: baseline;
27
+ width: auto;
28
+ border-collapse: separate;
29
+ }
30
+
31
+ .wp_themeSkin a:hover,
32
+ .wp_themeSkin a:link,
33
+ .wp_themeSkin a:visited,
34
+ .wp_themeSkin a:active {
35
+ text-decoration: none;
36
+ font-weight: normal;
37
+ cursor: default;
38
+ }
39
+
40
+ .wp_themeSkin table td {
41
+ vertical-align: middle;
42
+ }
43
+
44
+ .wp_themeSkin *,
45
+ .wp_themeSkin a:hover,
46
+ .wp_themeSkin a:link,
47
+ .wp_themeSkin a:visited,
48
+ .wp_themeSkin a:active {
49
+ color: #555;
50
+ }
51
+
52
+ /* These are part of TinyMCE, used in TinyMCE Advanced, but not WordPress. These are not updated for 3.8's design. */
53
+ .wp_themeSkin span.mce_sup,
54
+ .wp_themeSkin span.mce_sub,
55
+ .wp_themeSkin span.mce_media,
56
+ .wp_themeSkin span.mce_styleprops,
57
+ .wp_themeSkin span.mce_search,
58
+ .wp_themeSkin span.mce_emotions,
59
+ .wp_themeSkin span.mce_print,
60
+ .wp_themeSkin span.mce_attribs,
61
+ .wp_themeSkin span.mce_hr,
62
+ .wp_themeSkin span.mce_cut,
63
+ .wp_themeSkin span.mce_copy,
64
+ .wp_themeSkin span.mce_paste,
65
+ .wp_themeSkin span.mce_cite,
66
+ .wp_themeSkin span.mce_visualchars,
67
+ .wp_themeSkin span.mce_advhr,
68
+ .wp_themeSkin span.mce_insertdate,
69
+ .wp_themeSkin span.mce_anchor,
70
+ .wp_themeSkin span.mce_visualaid,
71
+ .wp_themeSkin span.mce_cleanup,
72
+ .wp_themeSkin span.mce_table,
73
+ .wp_themeSkin span.mce_row_props,
74
+ .wp_themeSkin span.mce_cell_props,
75
+ .wp_themeSkin span.mce_row_before,
76
+ .wp_themeSkin span.mce_row_after,
77
+ .wp_themeSkin span.mce_delete_row,
78
+ .wp_themeSkin span.mce_col_before,
79
+ .wp_themeSkin span.mce_col_after,
80
+ .wp_themeSkin span.mce_delete_col,
81
+ .wp_themeSkin span.mce_split_cells,
82
+ .wp_themeSkin span.mce_merge_cells,
83
+ .wp_themeSkin span.mce_delete_table,
84
+ .wp_themeSkin span.mce_ins,
85
+ .wp_themeSkin span.mce_abbr,
86
+ .wp_themeSkin span.mce_acronym,
87
+ .wp_themeSkin span.mce_del,
88
+ .wp_themeSkin span.mce_replace,
89
+ .wp_themeSkin span.mce_code,
90
+ .wp_themeSkin span.mce_nonbreaking,
91
+ .wp_themeSkin span.mce_inserttime,
92
+ .wp_themeSkin span.mce_insertlayer,
93
+ .wp_themeSkin span.mce_moveforward,
94
+ .wp_themeSkin span.mce_movebackward,
95
+ .wp_themeSkin span.mce_absolute {
96
+ -moz-transition: none;
97
+ -webkit-transition: none;
98
+ transition: none;
99
+ background: url("../js/tinymce/skins/wordpress/images/icons.gif") no-repeat 20px 20px;
100
+ }
101
+
102
+ .wp_themeSkin span.mce_code:after {
103
+ content: '\f475';
104
+ font-family: 'dashicons';
105
+ }
106
+
107
+ /* No @font-face support */
108
+ .no-font-face .wp_themeSkin span.mce_undo,
109
+ .no-font-face .wp_themeSkin span.mce_redo,
110
+ .no-font-face .wp_themeSkin span.mce_bullist,
111
+ .no-font-face .wp_themeSkin span.mce_numlist,
112
+ .no-font-face .wp_themeSkin span.mce_blockquote,
113
+ .no-font-face .wp_themeSkin span.mce_charmap,
114
+ .no-font-face .wp_themeSkin span.mce_bold,
115
+ .no-font-face .wp_themeSkin span.mce_italic,
116
+ .no-font-face .wp_themeSkin span.mce_underline,
117
+ .no-font-face .wp_themeSkin span.mce_justifyleft,
118
+ .no-font-face .wp_themeSkin span.mce_justifyright,
119
+ .no-font-face .wp_themeSkin span.mce_justifycenter,
120
+ .no-font-face .wp_themeSkin span.mce_justifyfull,
121
+ .no-font-face .wp_themeSkin span.mce_indent,
122
+ .no-font-face .wp_themeSkin span.mce_outdent,
123
+ .no-font-face .wp_themeSkin span.mce_link,
124
+ .no-font-face .wp_themeSkin span.mce_unlink,
125
+ .no-font-face .wp_themeSkin span.mce_help,
126
+ .no-font-face .wp_themeSkin span.mce_removeformat,
127
+ .no-font-face .wp_themeSkin span.mce_fullscreen,
128
+ .no-font-face .wp_themeSkin span.mce_wp_fullscreen,
129
+ .no-font-face .wp_themeSkin span.mce_media,
130
+ .no-font-face .wp_themeSkin span.mce_pastetext,
131
+ .no-font-face .wp_themeSkin span.mce_pasteword,
132
+ .no-font-face .wp_themeSkin span.mce_wp_help,
133
+ .no-font-face .wp_themeSkin span.mce_wp_adv,
134
+ .no-font-face .wp_themeSkin span.mce_wp_more,
135
+ .no-font-face .wp_themeSkin span.mce_strikethrough,
136
+ .no-font-face .wp_themeSkin span.mce_spellchecker,
137
+ .no-font-face .wp_themeSkin span.mce_forecolor,
138
+ .no-font-face .wp_themeSkin .mce_forecolorpicker,
139
+ .no-font-face .wp_themeSkin .mceSplitButton .mce_spellchecker span.mce_spellchecker,
140
+ .no-font-face .wp_themeSkin .mceSplitButton .mce_forecolor span.mce_forecolor,
141
+ .no-font-face .wp_themeSkin .mceSplitButton span.mce_numlist,
142
+ .no-font-face .wp_themeSkin .mceSplitButton span.mce_bullist {
143
+ -moz-transition: none;
144
+ -webkit-transition: none;
145
+ transition: none;
146
+ background-image: url('../images/wpicons.png?ver=20120720');
147
+ }
148
+
149
+ /* Theme */
150
+ .no-font-face .wp_themeSkin span.mce_undo {background-position:-500px -20px}
151
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_undo,
152
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_undo {background-position:-500px 0}
153
+
154
+ .no-font-face .wp_themeSkin span.mce_redo {background-position:-480px -20px}
155
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_redo,
156
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_redo {background-position:-480px 0}
157
+
158
+ .no-font-face .wp_themeSkin span.mce_bullist {background-position:-40px -20px}
159
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_bullist,
160
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_bullist,
161
+ .no-font-face .wp_themeSkin .mceSplitButton:hover span.mce_bullist {background-position:-40px 0}
162
+
163
+ .no-font-face .wp_themeSkin span.mce_numlist {background-position:-60px -20px}
164
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_numlist,
165
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_numlist,
166
+ .no-font-face .wp_themeSkin .mceSplitButton:hover span.mce_numlist {background-position:-60px 0}
167
+
168
+ .no-font-face .wp_themeSkin span.mce_blockquote {background-position:-80px -20px}
169
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_blockquote,
170
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_blockquote {background-position:-80px 0}
171
+
172
+ .no-font-face .wp_themeSkin span.mce_charmap {background-position:-420px -20px}
173
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_charmap,
174
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_charmap {background-position:-420px 0}
175
+
176
+ .no-font-face .wp_themeSkin span.mce_bold {background-position:0 -20px}
177
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_bold,
178
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_bold {background-position:0 0}
179
+
180
+ .no-font-face .wp_themeSkin span.mce_italic {background-position:-20px -20px}
181
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_italic,
182
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_italic {background-position:-20px 0}
183
+
184
+ .no-font-face .wp_themeSkin span.mce_underline {background-position:-280px -20px}
185
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_underline,
186
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_underline {background-position:-280px 0}
187
+
188
+ .no-font-face .wp_themeSkin span.mce_justifyleft {background-position:-100px -20px}
189
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_justifyleft,
190
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_justifyleft {background-position:-100px 0}
191
+
192
+ .no-font-face .wp_themeSkin span.mce_justifyright {background-position:-140px -20px}
193
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_justifyright,
194
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_justifyright {background-position:-140px 0}
195
+
196
+ .no-font-face .wp_themeSkin span.mce_justifycenter {background-position:-120px -20px}
197
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_justifycenter,
198
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_justifycenter {background-position:-120px 0}
199
+
200
+ .no-font-face .wp_themeSkin span.mce_justifyfull {background-position:-300px -20px}
201
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_justifyfull,
202
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_justifyfull {background-position:-300px 0}
203
+
204
+ .no-font-face .wp_themeSkin span.mce_indent {background-position:-460px -20px}
205
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_indent,
206
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_indent {background-position:-460px 0}
207
+
208
+ .no-font-face .wp_themeSkin span.mce_outdent {background-position:-440px -20px}
209
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_outdent,
210
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_outdent {background-position:-440px 0}
211
+
212
+ .no-font-face .wp_themeSkin span.mce_link {background-position:-160px -20px}
213
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_link,
214
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_link {background-position:-160px 0}
215
+
216
+ .no-font-face .wp_themeSkin span.mce_unlink {background-position:-180px -20px}
217
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_unlink,
218
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_unlink {background-position:-180px 0}
219
+
220
+ .no-font-face .wp_themeSkin span.mce_help {background-position:-520px -20px}
221
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_help,
222
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_help {background-position:-520px 0}
223
+
224
+ .no-font-face .wp_themeSkin span.mce_removeformat {background-position:-380px -20px}
225
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_removeformat,
226
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_removeformat {background-position:-380px 0}
227
+
228
+ .no-font-face .wp_themeSkin span.mce_strikethrough {background-position:-540px -20px;}
229
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_strikethrough,
230
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_strikethrough {background-position:-540px 0}
231
+
232
+ .no-font-face .wp_themeSkin .mceSplitButton .mce_forecolor span.mce_forecolor {background-position:-320px -20px}
233
+ .no-font-face .wp_themeSkin .mceSplitButtonEnabled:hover span.mce_forecolor,
234
+ .no-font-face .wp_themeSkin .mceSplitButtonSelected span.mce_forecolor {background-position:-320px 0}
235
+
236
+ .no-font-face .wp_themeSkin .mce_forecolorpicker {background-position:-320px -20px}
237
+
238
+ /* Plugins in WP */
239
+ .no-font-face .wp_themeSkin span.mce_fullscreen {background-position:-240px -20px}
240
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_fullscreen,
241
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_fullscreen {background-position:-240px 0}
242
+
243
+ .no-font-face .wp_themeSkin span.mce_wp_fullscreen {background-position:-240px -20px}
244
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_wp_fullscreen,
245
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_wp_fullscreen {background-position:-240px 0}
246
+
247
+ .no-font-face .wp_themeSkin span.mce_media {background-position:-400px -20px}
248
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_media,
249
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_media {background-position:-400px 0}
250
+
251
+ .no-font-face .wp_themeSkin span.mce_pastetext {background-position:-340px -20px}
252
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_pastetext,
253
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_pastetext {background-position:-340px 0}
254
+
255
+ .no-font-face .wp_themeSkin span.mce_pasteword {background-position:-360px -20px}
256
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_pasteword,
257
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_pasteword {background-position:-360px 0}
258
+
259
+ .no-font-face .wp_themeSkin span.mce_spellchecker {background-position:-220px -20px}
260
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_spellchecker,
261
+ .no-font-face .wp_themeSkin .mceSplitButtonEnabled:hover span.mce_spellchecker,
262
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_spellchecker,
263
+ .no-font-face .wp_themeSkin .mceSplitButtonSelected span.mce_spellchecker {background-position:-220px 0}
264
+
265
+ .no-font-face .wp_themeSkin span.mce_wp_help {background-position:-520px -20px}
266
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_wp_help,
267
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_wp_help {background-position:-520px 0}
268
+
269
+ .no-font-face .wp_themeSkin span.mce_wp_adv {background-position:-260px -20px}
270
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_wp_adv,
271
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_wp_adv {background-position:-260px 0}
272
+
273
+ .no-font-face .wp_themeSkin span.mce_wp_more {background-position:-200px -20px}
274
+ .no-font-face .wp_themeSkin .mceButtonEnabled:hover span.mce_wp_more,
275
+ .no-font-face .wp_themeSkin .mceButtonActive span.mce_wp_more {background-position:-200px 0}
276
+
277
+ .no-font-face .mceIcon:before {
278
+ display: none !important;
279
+ }
280
+ /* End no @font-face */
281
+
282
+ /* Containers */
283
+ .wp_themeSkin table {}
284
+
285
+ .wp_themeSkin iframe {
286
+ display: block;
287
+ }
288
+
289
+ .wp_themeSkin #mce_fullscreen_ifr {
290
+ background-color: #fff;
291
+ }
292
+
293
+ .wp_themeSkin .mceToolbar {
294
+ padding: 1px;
295
+ }
296
+
297
+ /* External */
298
+ .wp_themeSkin .mceExternalToolbar {
299
+ position: absolute;
300
+ border-bottom: 0;
301
+ display: none;
302
+ }
303
+
304
+ .wp_themeSkin .mceExternalToolbar td.mceToolbar {
305
+ padding-right: 13px;
306
+ }
307
+
308
+ .wp_themeSkin .mceExternalClose {
309
+ position: absolute;
310
+ top: 3px;
311
+ right: 3px;
312
+ width: 7px;
313
+ height: 7px;
314
+ background: url("../js/tinymce/skins/wordpress/images/icons.gif") -820px 0;
315
+ }
316
+
317
+ /* Layout */
318
+ .wp_themeSkin table.mceToolbar,
319
+ .wp_themeSkin tr.mceFirst .mceToolbar tr td,
320
+ .wp_themeSkin tr.mceLast .mceToolbar tr td {
321
+ border: 0;
322
+ margin: 0;
323
+ padding: 0;
324
+ }
325
+
326
+ .wp_themeSkin table.mceLayout {
327
+ border: 0;
328
+ }
329
+
330
+ .wp_themeSkin .mceStatusbar {
331
+ background: #fff;
332
+ border-top: 1px solid #eee;
333
+ color: #000;
334
+ display: block;
335
+ font-family: sans-serif;
336
+ font-size: 12px;
337
+ height: 20px;
338
+ line-height: 16px;
339
+ padding: 0 0 0 8px;
340
+ overflow: visible;
341
+ }
342
+
343
+ .wp_themeSkin .mceStatusbar * {
344
+ color: #555;
345
+ }
346
+
347
+ .wp_themeSkin .mceStatusbar div {
348
+ float: left;
349
+ padding: 2px;
350
+ }
351
+
352
+ .wp_themeSkin .mceStatusbar a.mceResize {
353
+ display: block;
354
+ float: right;
355
+ background: url("../js/tinymce/skins/wordpress/images/icons.gif") -800px 0;
356
+ width: 20px;
357
+ height: 20px;
358
+ cursor: se-resize;
359
+ }
360
+
361
+ .wp_themeSkin .mceStatusbar a:hover {
362
+ text-decoration: underline;
363
+ }
364
+
365
+ .wp_themeSkin table.mceToolbar {
366
+ margin: 0 6px 2px;
367
+ }
368
+
369
+ .wp_themeSkin table.mceToolbar :active,
370
+ .wp_themeSkin table.mceToolbar :focus,
371
+ .wp_themeSkin table.mceToolbar:focus,
372
+ .wp_themeSkin span.mceSeparator:focus {
373
+ outline: none;
374
+ }
375
+
376
+ .wp_themeSkin #content_toolbar1 {
377
+ margin-top: 2px;
378
+ }
379
+
380
+ .wp_themeSkin .mceToolbar .mceToolbarEndListBox span {
381
+ display: none;
382
+ }
383
+
384
+ .wp_themeSkin span.mceIcon,
385
+ .wp_themeSkin img.mceIcon {
386
+ display: block;
387
+ width: 20px;
388
+ height: 20px;
389
+ }
390
+
391
+ a .mceIcon, .mceAction {
392
+ text-align: center;
393
+ font: normal 20px/1 'dashicons' !important;
394
+ speak: none;
395
+ -webkit-font-smoothing: antialiased;
396
+ -moz-osx-font-smoothing: grayscale;
397
+ }
398
+
399
+ .mceAction {
400
+ line-height:16px;
401
+ }
402
+
403
+ /* Button */
404
+ .wp_themeSkin .mceButton {
405
+ display: block;
406
+ width: 20px;
407
+ height: 20px;
408
+ cursor: default;
409
+ padding: 1px 2px;
410
+ margin: 1px;
411
+ -webkit-border-radius: 2px;
412
+ border-radius: 2px;
413
+ }
414
+
415
+ .wp_themeSkin a.mceButtonEnabled:hover {
416
+ background-image: inherit 0 -10px;
417
+ }
418
+
419
+ .wp_themeSkin .mceOldBoxModel a.mceButton span, .wp_themeSkin .mceOldBoxModel a.mceButton img {
420
+ margin: 0 0 0 1px;
421
+ }
422
+
423
+ .wp_themeSkin .mceButtonDisabled .mceIcon {
424
+ opacity: 0.2;
425
+ filter: alpha(opacity=20);
426
+ }
427
+
428
+ /* Separator */
429
+ .wp_themeSkin .mceSeparator {
430
+ display: none;
431
+ }
432
+
433
+ /* ListBox */
434
+ .wp_themeSkin .mceListBox,
435
+ .wp_themeSkin .mceListBox a {
436
+ display: block;
437
+ }
438
+
439
+ .wp_themeSkin .mceListBox .mceText {
440
+ padding: 1px 4px 1px 5px;
441
+ width: 70px;
442
+ text-align: left;
443
+ text-decoration: none;
444
+ -webkit-border-bottom-left-radius: 2px;
445
+ -webkit-border-top-left-radius: 2px;
446
+ border-bottom-left-radius: 2px;
447
+ border-top-left-radius: 2px;
448
+ font-family: sans-serif;
449
+ font-size: 12px;
450
+ height: 20px;
451
+ line-height: 20px;
452
+ overflow: hidden;
453
+ }
454
+
455
+ .wp_themeSkin .mceListBox {
456
+ margin: 1px;
457
+ direction: ltr;
458
+ background-color: #fff;
459
+ border: 1px solid #ddd;
460
+ -webkit-box-shadow: inset 0 1px 1px -1px rgba(0, 0, 0, .2);
461
+ box-shadow: inset 0 1px 1px -1px rgba(0, 0, 0, .2);
462
+ }
463
+
464
+ .wp_themeSkin .mceListBox .mceOpen {
465
+ width: 12px;
466
+ height: 20px;
467
+ border-collapse: separate;
468
+ padding: 1px;
469
+ -webkit-border-bottom-left-radius: 0;
470
+ -webkit-border-top-left-radius: 0;
471
+ border-bottom-left-radius: 0;
472
+ border-top-left-radius: 0;
473
+ }
474
+
475
+ .wp_themeSkin .mceListBox .mceFirst a {
476
+ border-style: solid;
477
+ border-width: 1px;
478
+ border-bottom-right-radius: 2px;
479
+ border-top-right-radius: 2px;
480
+ }
481
+
482
+ .wp_themeSkin .mceListBoxMenu .mce_formatPreview {
483
+ line-height: normal;
484
+ }
485
+
486
+ .wp_themeSkin .mceListBox .mceOpen,
487
+ .wp_themeSkin .mceListBoxHover .mceOpen,
488
+ .wp_themeSkin .mceListBoxSelected .mceOpen,
489
+ .wp_themeSkin table.mceListBoxEnabled .mceOpen {
490
+ background-image: url("../../../../../wp-includes/images/down_arrow.gif");
491
+ background-position: 3px 1px;
492
+ background-repeat: no-repeat;
493
+ }
494
+
495
+ .wp_themeSkin .mceListBoxDisabled .mceText {
496
+ color: gray;
497
+ }
498
+
499
+ .wp_themeSkin .mceListBoxMenu {
500
+ overflow: auto;
501
+ overflow-x: hidden;
502
+ }
503
+
504
+ .wp_themeSkin .mceOldBoxModel .mceListBox .mceText {
505
+ height: 22px;
506
+ }
507
+
508
+ .wp_themeSkin select.mceListBox {
509
+ font-family: sans-serif;
510
+ font-size: 12px;
511
+ border-color: #b2b2b2;
512
+ background-color: #fff;
513
+ }
514
+
515
+ /* SplitButton */
516
+ .wp_themeSkin .mceSplitButton a,
517
+ .wp_themeSkin .mceSplitButton span {
518
+ display: block;
519
+ height: 20px;
520
+ }
521
+
522
+ .wp_themeSkin .mceSplitButton {
523
+ display: block;
524
+ direction: ltr;
525
+ }
526
+
527
+ .wp_themeSkin table.mceSplitButton td {
528
+ padding: 2px;
529
+ -webkit-border-radius: 2px;
530
+ border-radius: 2px;
531
+ }
532
+
533
+ .wp_themeSkin table.mceSplitButton:hover td {
534
+ background-image: inherit 0 -10px;
535
+ }
536
+
537
+ .wp_themeSkin .mceSplitButton a.mceAction {
538
+ height: 20px;
539
+ width: 20px;
540
+ padding: 1px 2px;
541
+ border-right: 0 none;
542
+ }
543
+
544
+ .wp_themeSkin .mceSplitButton span.mceAction {
545
+ background-image: url("../../../../../wp-includes/js/tinymce/skins/wordpress/images/icons.gif");
546
+ background-repeat: no-repeat;
547
+ background-color: transparent;
548
+ width: 20px;
549
+ }
550
+
551
+ .wp_themeSkin .mceSplitButton span.mceAction.mce_bullist,
552
+ .wp_themeSkin .mceSplitButton span.mceAction.mce_numlist {
553
+ background-image: none;
554
+ }
555
+
556
+ .wp_themeSkin .mceSplitButton a.mceOpen {
557
+ width: 11px;
558
+ height: 20px;
559
+ background-position: 0px 2px;
560
+ background-repeat: no-repeat;
561
+ padding: 1px 0;
562
+ }
563
+
564
+ .wp_themeSkin .mceSplitButton span.mceOpen {
565
+ display: none;
566
+ }
567
+
568
+ .wp_themeSkin .mceSplitButtonDisabled .mceAction {
569
+ opacity: 0.3;
570
+ filter: alpha(opacity=30);
571
+ }
572
+
573
+ .wp_themeSkin .mceListBox a.mceText,
574
+ .wp_themeSkin .mceSplitButton a.mceAction {
575
+ -webkit-border-bottom-left-radius: 2px;
576
+ -webkit-border-top-left-radius: 2px;
577
+ border-bottom-left-radius: 2px;
578
+ border-top-left-radius: 2px;
579
+ }
580
+
581
+ .wp_themeSkin .mceSplitButton a.mceOpen,
582
+ .wp_themeSkin .mceListBox a.mceOpen {
583
+ -webkit-border-bottom-right-radius: 2px;
584
+ -webkit-border-top-right-radius: 2px;
585
+ border-bottom-right-radius: 2px;
586
+ border-top-right-radius: 2px;
587
+ }
588
+
589
+ /* ColorSplitButton */
590
+ .wp_themeSkin div.mceColorSplitMenu table {
591
+ background-color: #ebebeb;
592
+ border-color: #bbb;
593
+ }
594
+
595
+ .wp_themeSkin .mceColorSplitMenu td {
596
+ padding: 2px;
597
+ }
598
+
599
+ .wp_themeSkin .mceColorSplitMenu a {
600
+ display: block;
601
+ width: 9px;
602
+ height: 9px;
603
+ overflow: hidden;
604
+ border-color: #B2B2B2;
605
+ }
606
+
607
+ .wp_themeSkin .mceColorSplitMenu td.mceMoreColors {
608
+ padding: 1px 3px 1px 1px;
609
+ }
610
+
611
+ .wp_themeSkin .mceColorSplitMenu a.mceMoreColors {
612
+ width: 100%;
613
+ height: auto;
614
+ text-align: center;
615
+ font-family: "Open Sans", sans-serif;
616
+ font-size: 11px;
617
+ line-height: 20px;
618
+ border-color: #fff;
619
+ }
620
+
621
+ .wp_themeSkin .mceColorSplitMenu a.mceMoreColors:hover {}
622
+ .wp_themeSkin a.mceMoreColors:hover {}
623
+ .wp_themeSkin .mceColorPreview {
624
+ margin: -5px 0 0 2px;
625
+ width: 16px;
626
+ height: 4px;
627
+ overflow: hidden;
628
+ }
629
+
630
+ /* Menu */
631
+ .wp_themeSkin .mceMenu {
632
+ position: absolute;
633
+ left: 0;
634
+ top: 0;
635
+ z-index: 1000;
636
+ border-color: #ddd;
637
+ direction: ltr;
638
+ }
639
+
640
+ .wp_themeSkin .mceNoIcons span.mceIcon {
641
+ width: 0;
642
+ }
643
+
644
+ .wp_themeSkin .mceNoIcons a .mceText {
645
+ padding-left: 10px;
646
+ }
647
+
648
+ .wp_themeSkin .mceMenu table {
649
+ background-color: #ebeaeb;
650
+ }
651
+
652
+ .wp_themeSkin .mceMenu a,
653
+ .wp_themeSkin .mceMenu span,
654
+ .wp_themeSkin .mceMenu {
655
+ display: block;
656
+ }
657
+
658
+ .wp_themeSkin .mceMenu td {
659
+ height: 20px;overflow: hidden;
660
+ }
661
+
662
+ .wp_themeSkin .mceMenu a {
663
+ position: relative;
664
+ padding: 3px 0 4px 0;
665
+ text-decoration: none !important;
666
+ }
667
+
668
+ .wp_themeSkin .mceMenu .mceText {
669
+ position: relative;
670
+ display: block;
671
+ font-family: "Open Sans", sans-serif;
672
+ cursor: default;
673
+ margin: 0;
674
+ padding: 0 25px;
675
+ color: #000;
676
+ }
677
+
678
+ .wp_themeSkin .mceMenu span.mceText, .wp_themeSkin .mceMenu .mcePreview {
679
+ font-size: 12px;
680
+ }
681
+
682
+ .wp_themeSkin .mceMenu pre.mceText {
683
+ font-family: Monospace;
684
+ }
685
+
686
+ .wp_themeSkin .mceMenu .mceIcon {
687
+ position: absolute;
688
+ top: 0;
689
+ left: 0;
690
+ width: 22px;
691
+ }
692
+
693
+ .wp_themeSkin .mceMenu .mceMenuItemEnabled a:hover,
694
+ .wp_themeSkin .mceMenu .mceMenuItemActive {
695
+ background-color: #f5f5f5;
696
+ }
697
+
698
+ .wp_themeSkin td.mceMenuItemSeparator {
699
+ height: 1px;
700
+ background-color: #aaa;
701
+ }
702
+
703
+ .wp_themeSkin .mceMenuItemTitle a {
704
+ border-top: 0;
705
+ border-right: 0;
706
+ border-left: 0;
707
+ border-bottom: 1px solid #aaa;
708
+ text-decoration: none !important;
709
+ background-color: #ccc;
710
+ }
711
+
712
+ .wp_themeSkin .mceMenuItemTitle span.mceText {
713
+ font-weight: bold;
714
+ padding-left: 4px;
715
+ color: #000;
716
+ }
717
+
718
+ .wp_themeSkin .mceMenuItemSelected .mceIcon {
719
+ background: url("/wp-includes/js/tinymce/skins/wordpress/images/menu_check.gif");
720
+ color: #888;
721
+ }
722
+
723
+ .wp_themeSkin .mceNoIcons .mceMenuItemSelected a {
724
+ background: url("/wp-includes/js/tinymce/skins/wordpress/images/menu_arrow.gif") no-repeat -6px center;
725
+ }
726
+
727
+ .wp_themeSkin .mceMenu span.mceMenuLine {
728
+ display: none;
729
+ }
730
+
731
+ .wp_themeSkin .mceMenuItemSub a {
732
+ background: url("/wp-includes/js/tinymce/skins/wordpress/images/menu_arrow.gif") no-repeat top right;
733
+ }
734
+
735
+ /* Progress,Resize */
736
+ .wp_themeSkin .mceBlocker {
737
+ position: absolute;
738
+ left: 0;
739
+ top: 0;
740
+ z-index: 1000;
741
+ opacity: 0.5;
742
+ filter: alpha(opacity=50);
743
+ background: #FFF;
744
+ }
745
+
746
+ .wp_themeSkin .mceProgress {
747
+ position: absolute;
748
+ left: 0;
749
+ top: 0;
750
+ z-index: 1001;
751
+ background: url("/wp-includes/images/spinner.gif") no-repeat;
752
+ width: 32px;
753
+ height: 32px;
754
+ margin: -16px 0 0 -16px;
755
+ }
756
+
757
+ .wp_themeSkin .mcePlaceHolder {
758
+ border: 1px dotted gray;
759
+ }
760
+
761
+ /* Rtl */
762
+ .mceRtl .mceListBox .mceText {
763
+ text-align: right;
764
+ padding: 0 4px 0 0;
765
+ }
766
+
767
+ .mceRtl .mceMenuItem .mceText {
768
+ text-align: right;
769
+ }
770
+
771
+ /* Formats */
772
+ .wp_themeSkin .mce_p span.mceText {}
773
+ .wp_themeSkin .mce_address span.mceText {
774
+ font-style: italic;
775
+ }
776
+
777
+ .wp_themeSkin .mce_pre span.mceText {
778
+ font-family: monospace;
779
+ }
780
+
781
+ .wp_themeSkin .mce_h1 span.mceText {
782
+ font-weight: bolder;
783
+ font-size: 18px;
784
+ }
785
+
786
+ .wp_themeSkin .mce_h2 span.mceText {
787
+ font-weight: bolder;
788
+ font-size: 14px;
789
+ }
790
+
791
+ .wp_themeSkin .mce_h3 span.mceText {
792
+ font-weight: bolder;
793
+ font-size: 12px;
794
+ }
795
+
796
+ .wp_themeSkin .mce_h4 span.mceText {
797
+ font-weight: bolder;
798
+ font-size: 11px;
799
+ }
800
+
801
+ .wp_themeSkin .mce_h5 span.mceText {
802
+ font-weight: bolder;
803
+ font-size: 11px;
804
+ }
805
+
806
+ .wp_themeSkin .mce_h6 span.mceText {
807
+ font-weight: bolder;
808
+ font-size: 10px;
809
+ }
810
+
811
+ span.mce_bold:before {
812
+ content: '\f200';
813
+ }
814
+
815
+ span.mce_italic:before {
816
+ content: '\f201';
817
+ }
818
+
819
+ span.mce_bullist:before {
820
+ content: '\f203';
821
+ }
822
+
823
+ span.mce_numlist:before {
824
+ content: '\f204';
825
+ }
826
+
827
+ span.mce_blockquote:before {
828
+ content: '\f205';
829
+ }
830
+
831
+ span.mce_justifyleft:before {
832
+ content: '\f206';
833
+ }
834
+
835
+ span.mce_justifycenter:before {
836
+ content: '\f207';
837
+ }
838
+
839
+ span.mce_justifyright:before {
840
+ content: '\f208';
841
+ }
842
+
843
+ span.mce_link:before {
844
+ content: '\f103';
845
+ }
846
+
847
+ span.mce_unlink:before {
848
+ content: '\f225';
849
+ }
850
+
851
+ span.mce_wp_more:before {
852
+ content: '\f209';
853
+ }
854
+
855
+ span.mce_strikethrough:before {
856
+ content: '\f224';
857
+ }
858
+
859
+ span.mce_spellchecker {
860
+ font-size: 20px;
861
+ background: none !important;
862
+ margin-top: 2px;
863
+ }
864
+
865
+ span.mce_spellchecker:before {
866
+ content: '\f210';
867
+ }
868
+
869
+ span.mce_fullscreen:before,
870
+ span.mce_wp_fullscreen:before {
871
+ content: '\f211';
872
+ }
873
+
874
+ span.mce_wp_adv:before {
875
+ content: '\f212';
876
+ }
877
+ span.mce_underline:before {
878
+ content: '\f213';
879
+ }
880
+
881
+ span.mce_justifyfull:before {
882
+ content: '\f214';
883
+ }
884
+
885
+ span.mce_forecolor {
886
+ background: none !important;
887
+ }
888
+
889
+ span.mce_forecolor:before {
890
+ content: '\f215';
891
+ }
892
+
893
+ span.mce_pastetext:before {
894
+ content: '\f217';
895
+ }
896
+
897
+ span.mce_pasteword:before {
898
+ content: '\f216';
899
+ }
900
+
901
+ span.mce_removeformat:before {
902
+ content: '\f218';
903
+ }
904
+
905
+ span.mce_charmap:before {
906
+ content: '\f220';
907
+ }
908
+
909
+ span.mce_outdent:before {
910
+ content: '\f221';
911
+ }
912
+
913
+ span.mce_indent:before {
914
+ content: '\f222';
915
+ }
916
+
917
+ span.mce_undo:before {
918
+ content: '\f171';
919
+ }
920
+
921
+ span.mce_redo:before {
922
+ content: '\f172';
923
+ }
924
+
925
+ span.mce_help:before,
926
+ span.mce_wp_help:before {
927
+ content: '\f223';
928
+ }
929
+
930
+ span.mce_image:before {
931
+ content: '\f104';
932
+ }
933
+
934
+ span.mce_ltr:before {
935
+ content: '\f320';
936
+ }
937
+
938
+ /* Default icons */
939
+ .wp_themeSkin span.mce_cleanup {background-position:-380px -20px}
940
+ .wp_themeSkin span.mce_anchor {background-position:-200px 0}
941
+ .wp_themeSkin span.mce_sub {background-position:-600px 0}
942
+ .wp_themeSkin span.mce_sup {background-position:-620px 0}
943
+ .wp_themeSkin span.mce_newdocument {background-position:-520px 0}
944
+ .wp_themeSkin span.mce_image {background-position:-380px 0}
945
+ .wp_themeSkin span.mce_code {background-position:-260px 0}
946
+ .wp_themeSkin span.mce_hr {background-position:-360px 0}
947
+ .wp_themeSkin span.mce_visualaid {background-position:-660px 0}
948
+ .wp_themeSkin span.mce_paste {background-position:-560px 0}
949
+ .wp_themeSkin span.mce_copy {background-position:-700px 0}
950
+ .wp_themeSkin span.mce_cut {background-position:-680px 0}
951
+ .wp_themeSkin .mce_backcolor span.mceAction {background-position:-760px 0}
952
+ .wp_themeSkin .mce_backcolorpicker {background-position:-760px 0}
953
+
954
+ /* Plugins */
955
+ .wp_themeSkin span.mce_advhr {background-position:-0px -20px}
956
+ .wp_themeSkin span.mce_ltr {background-position:-20px -20px}
957
+ .wp_themeSkin span.mce_rtl {background-position:-40px -20px}
958
+ .wp_themeSkin span.mce_emotions {background-position:-60px -20px}
959
+ .wp_themeSkin span.mce_fullpage {background-position:-80px -20px}
960
+ .wp_themeSkin span.mce_iespell {background-position:-120px -20px}
961
+ .wp_themeSkin span.mce_insertdate {background-position:-140px -20px}
962
+ .wp_themeSkin span.mce_inserttime {background-position:-160px -20px}
963
+ .wp_themeSkin span.mce_absolute {background-position:-180px -20px}
964
+ .wp_themeSkin span.mce_backward {background-position:-200px -20px}
965
+ .wp_themeSkin span.mce_forward {background-position:-220px -20px}
966
+ .wp_themeSkin span.mce_insert_layer {background-position:-240px -20px}
967
+ .wp_themeSkin span.mce_insertlayer {background-position:-260px -20px}
968
+ .wp_themeSkin span.mce_movebackward {background-position:-280px -20px}
969
+ .wp_themeSkin span.mce_moveforward {background-position:-300px -20px}
970
+ .wp_themeSkin span.mce_media {background-position:-320px -20px}
971
+ .wp_themeSkin span.mce_nonbreaking {background-position:-340px -20px}
972
+ .wp_themeSkin span.mce_selectall {background-position:-400px -20px}
973
+ .wp_themeSkin span.mce_preview {background-position:-420px -20px}
974
+ .wp_themeSkin span.mce_print {background-position:-440px -20px}
975
+ .wp_themeSkin span.mce_cancel {background-position:-460px -20px}
976
+ .wp_themeSkin span.mce_save {background-position:-480px -20px}
977
+ .wp_themeSkin span.mce_replace {background-position:-500px -20px}
978
+ .wp_themeSkin span.mce_search {background-position:-520px -20px}
979
+ .wp_themeSkin span.mce_styleprops {background-position:-560px -20px}
980
+ .wp_themeSkin span.mce_table {background-position:-580px -20px}
981
+ .wp_themeSkin span.mce_cell_props {background-position:-600px -20px}
982
+ .wp_themeSkin span.mce_delete_table {background-position:-620px -20px}
983
+ .wp_themeSkin span.mce_delete_col {background-position:-640px -20px}
984
+ .wp_themeSkin span.mce_delete_row {background-position:-660px -20px}
985
+ .wp_themeSkin span.mce_col_after {background-position:-680px -20px}
986
+ .wp_themeSkin span.mce_col_before {background-position:-700px -20px}
987
+ .wp_themeSkin span.mce_row_after {background-position:-720px -20px}
988
+ .wp_themeSkin span.mce_row_before {background-position:-740px -20px}
989
+ .wp_themeSkin span.mce_merge_cells {background-position:-760px -20px}
990
+ .wp_themeSkin span.mce_table_props {background-position:-980px -20px}
991
+ .wp_themeSkin span.mce_row_props {background-position:-780px -20px}
992
+ .wp_themeSkin span.mce_split_cells {background-position:-800px -20px}
993
+ .wp_themeSkin span.mce_template {background-position:-820px -20px}
994
+ .wp_themeSkin span.mce_visualchars {background-position:-840px -20px}
995
+ .wp_themeSkin span.mce_abbr {background-position:-860px -20px}
996
+ .wp_themeSkin span.mce_acronym {background-position:-880px -20px}
997
+ .wp_themeSkin span.mce_attribs {background-position:-900px -20px}
998
+ .wp_themeSkin span.mce_cite {background-position:-920px -20px}
999
+ .wp_themeSkin span.mce_del {background-position:-940px -20px}
1000
+ .wp_themeSkin span.mce_ins {background-position:-960px -20px}
1001
+ .wp_themeSkin span.mce_pagebreak {background-position:0 -40px}
1002
+ .wp_themeSkin span.mce_restoredraft {background-position:-20px -40px}
1003
+ .wp_themeSkin span.mce_visualblocks {background-position: -40px -40px}
1004
+
1005
+ /* border */
1006
+ .wp_themeSkin .mceExternalToolbar,
1007
+ .wp_themeSkin .mceButton,
1008
+ .wp_themeSkin a.mceButtonEnabled:hover,
1009
+ .wp_themeSkin a.mceButtonActive,
1010
+ .wp_themeSkin a.mceButtonSelected,
1011
+ .wp_themeSkin .mceListBox .mceText,
1012
+ .wp_themeSkin .mceListBox .mceOpen,
1013
+ .wp_themeSkin select.mceListBox,
1014
+ .wp_themeSkin .mceSplitButton a.mceAction,
1015
+ .wp_themeSkin .mceSplitButton a.mceOpen,
1016
+ .wp_themeSkin .mceSplitButton a.mceOpen:hover,
1017
+ .wp_themeSkin .mceSplitButtonSelected a.mceOpen,
1018
+ .wp_themeSkin table.mceSplitButtonEnabled:hover a.mceAction,
1019
+ .wp_themeSkin .mceSplitButton a.mceAction:hover,
1020
+ .wp_themeSkin div.mceColorSplitMenu table,
1021
+ .wp_themeSkin .mceColorSplitMenu a,
1022
+ .wp_themeSkin .mceColorSplitMenu a.mceMoreColors,
1023
+ .wp_themeSkin .mceColorSplitMenu a.mceMoreColors:hover,
1024
+ .wp_themeSkin a.mceMoreColors:hover,
1025
+ .wp_themeSkin .mceMenu {
1026
+ border-style: solid;
1027
+ border-width: 1px;
1028
+ }
1029
+
1030
+ .wp_themeSkin .mceListBox .mceText {
1031
+ border-right: 0 none;
1032
+ }
1033
+
1034
+ .wp_themeSkin iframe {
1035
+ background: transparent;
1036
+ }
1037
+
1038
+ .wp_themeSkin .mceButton {
1039
+ border-color: transparent;
1040
+ }
1041
+
1042
+ .wp_themeSkin .mceListBox .mceText,
1043
+ .wp_themeSkin .mceListBox .mceOpen {
1044
+ border-color: transparent;
1045
+ }
1046
+
1047
+ .wp_themeSkin a.mceButtonEnabled:hover,
1048
+ .wp_themeSkin table.mceSplitButton:hover {
1049
+ border-color: #bbb;
1050
+ background: #eee;
1051
+ background-image: -webkit-gradient(linear, left bottom, left top, from(#e5e5e5), to(#fff));
1052
+ background-image: -webkit-linear-gradient(bottom, #e5e5e5, #fff);
1053
+ background-image: -moz-linear-gradient(bottom, #e5e5e5, #fff);
1054
+ background-image: -o-linear-gradient(bottom, #e5e5e5, #fff);
1055
+ background-image: linear-gradient(to top, #e5e5e5, #fff);
1056
+ }
1057
+
1058
+ .wp_themeSkin a.mceButton:active,
1059
+ .wp_themeSkin a.mceButtonEnabled:active,
1060
+ .wp_themeSkin a.mceButtonSelected:active,
1061
+ .wp_themeSkin a.mceButtonActive,
1062
+ .wp_themeSkin a.mceButtonActive:active,
1063
+ .wp_themeSkin a.mceButtonActive:hover,
1064
+ .wp_themeSkin .mceSplitButtonSelected table,
1065
+ .wp_themeSkin .mceSplitButtonSelected table:hover {
1066
+ outline: none;
1067
+ border-color: #999 #ccc #ccc #999;
1068
+ background: #eee;
1069
+ background-image: -webkit-gradient(linear, left bottom, left top, from(#f6f6f6), to(#e3e3e3));
1070
+ background-image: -webkit-linear-gradient(bottom, #f6f6f6, #e3e3e3);
1071
+ background-image: -moz-linear-gradient(bottom, #f6f6f6, #e3e3e3);
1072
+ background-image: -o-linear-gradient(bottom, #f6f6f6, #e3e3e3);
1073
+ background-image: linear-gradient(to top, #f6f6f6, #e3e3e3);
1074
+ }
1075
+
1076
+ .wp_themeSkin .mceSplitButtonSelected table a.mceOpen,
1077
+ .wp_themeSkin .mceSplitButtonSelected table a.mceAction {
1078
+ border-color: #999 #ccc #ccc #999;
1079
+ }
1080
+
1081
+ .wp_themeSkin .mceButtonDisabled {
1082
+ border-color: transparent;
1083
+ }
1084
+
1085
+ .wp_themeSkin .mceListBox .mceOpen {
1086
+ border-left: 0;
1087
+ }
1088
+
1089
+ .wp_themeSkin .mceListBoxEnabled:hover,
1090
+ .wp_themeSkin .mceListBoxEnabled:active,
1091
+ .wp_themeSkin .mceListBoxHover,
1092
+ .wp_themeSkin .mceListBoxHover:active,
1093
+ .wp_themeSkin .mceListBoxSelected {
1094
+ -webkit-box-shadow: inset 0 1px 1px -1px rgba(0, 0, 0, .3);
1095
+ box-shadow: inset 0 1px 1px -1px rgba(0, 0, 0, .3);
1096
+ border-color: #bbb;
1097
+ }
1098
+
1099
+ /* SplitButton */
1100
+ .wp_themeSkin .mceSplitButton .mceLast span.mceOpen .mceIconOnly {
1101
+ display: block;
1102
+ }
1103
+
1104
+ .wp_themeSkin .mceSplitButton a.mceAction,
1105
+ .wp_themeSkin .mceSplitButton a.mceOpen {
1106
+ border-color: transparent;
1107
+ }
1108
+
1109
+ .wp_themeSkin .mceSplitButton:hover a {
1110
+ border-color: #bbb;
1111
+ }
1112
+
1113
+ .wp_themeSkin .mceSplitButtonEnabled a.mceOpen,
1114
+ .wp_themeSkin .mceSplitButtonSelected a.mceOpen,
1115
+ .wp_themeSkin .mceSplitButtonActive a.mceOpen,
1116
+ .wp_themeSkin .mceSplitButtonEnabled:hover a.mceOpen {
1117
+ background-image: url("../../../../../wp-includes/images/down_arrow.gif");
1118
+ background-position: 1px 2px;
1119
+ background-repeat: no-repeat;
1120
+ border-left: 0;
1121
+ }
1122
+
1123
+ .wp_themeSkin .mceSplitButtonActive td {
1124
+ -webkit-border-radius: 3px;
1125
+ border-radius: 3px;
1126
+ }
1127
+
1128
+ .wp_themeSkin .mceColorSplitMenu a.mceMoreColors:hover {
1129
+ border-color: #0A246A;
1130
+ background-color: #B6BDD2;
1131
+ }
1132
+
1133
+ .wp_themeSkin a.mceMoreColors:hover {
1134
+ border-color: #0A246A;
1135
+ }
1136
+
1137
+ .wp_themeSkin .mceMenuItemDisabled .mceText {
1138
+ color: #888;
1139
+ }
1140
+
1141
+ #mceModalBlocker {
1142
+ background: #000;
1143
+ opacity: 0.7;
1144
+ -ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=70)";
1145
+ filter: alpha(opacity=70);
1146
+ }
1147
+
1148
+ /* WP specific */
1149
+ .wp-editor-wrap {
1150
+ position: relative;
1151
+ }
1152
+
1153
+ .wp-editor-tools {
1154
+ position: relative;
1155
+ z-index: 1;
1156
+ }
1157
+
1158
+ .wp-editor-container {
1159
+ clear: both;
1160
+ }
1161
+
1162
+ .wp-editor-area {
1163
+ font-family: Consolas, Monaco, monospace;
1164
+ font-size: 13px;
1165
+ padding: 10px;
1166
+ margin: 1px 0 0;
1167
+ line-height: 150%;
1168
+ border: 0 none;
1169
+ outline: none;
1170
+ display: block;
1171
+ resize: vertical;
1172
+ -moz-box-sizing: border-box;
1173
+ -webkit-box-sizing: border-box;
1174
+ box-sizing: border-box;
1175
+ }
1176
+
1177
+ .wp-editor-tools {
1178
+ padding: 0;
1179
+ }
1180
+
1181
+ .wp-editor-container textarea.wp-editor-area {
1182
+ width: 100%;
1183
+ margin: 0;
1184
+ -webkit-box-shadow: none;
1185
+ box-shadow: none;
1186
+ }
1187
+
1188
+ .quicktags-toolbar,
1189
+ .wp_themeSkin tr.mceFirst td.mceToolbar {
1190
+ border-bottom: 1px solid #dedede;
1191
+ background: #f5f5f5;
1192
+ }
1193
+
1194
+ .wp-editor-tabs {
1195
+ float: right;
1196
+ }
1197
+
1198
+ .wp-switch-editor {
1199
+ background: #ebebeb;
1200
+ border: 1px solid #dedede;
1201
+ color: #777;
1202
+ cursor: pointer;
1203
+ float: right;
1204
+ font: 13px/19px "Open Sans", sans-serif;
1205
+ height: 19px;
1206
+ margin: 5px 0 0 5px;
1207
+ padding: 3px 8px 4px;
1208
+ position: relative;
1209
+ top: 1px;
1210
+ }
1211
+
1212
+ .wp-switch-editor:active {
1213
+ background-color: #f1f1f1;
1214
+ }
1215
+
1216
+ .wp-switch-editor:hover {
1217
+ text-decoration: none !important;
1218
+ background: #fff;
1219
+ }
1220
+
1221
+ .js .tmce-active .wp-editor-area {
1222
+ color: white;
1223
+ }
1224
+
1225
+ .tmce-active .quicktags-toolbar {
1226
+ display: none;
1227
+ }
1228
+
1229
+ .tmce-active .switch-tmce,
1230
+ .html-active .switch-html {
1231
+ background: #f5f5f5;
1232
+ color: #555;
1233
+ height: 20px;
1234
+ border-bottom: none;
1235
+ }
1236
+
1237
+ .wp-media-buttons {
1238
+ float: left;
1239
+ }
1240
+
1241
+ .wp-media-buttons .button {
1242
+ margin-right: 5px;
1243
+ margin-bottom: 4px;
1244
+ padding-left: 7px;
1245
+ padding-right: 7px;
1246
+ }
1247
+
1248
+ .wp-media-buttons .button:active {
1249
+ position: relative;
1250
+ top: 1px;
1251
+ margin-top: -1px;
1252
+ margin-bottom: 1px;
1253
+ }
1254
+
1255
+ .wp-media-buttons .insert-media {
1256
+ padding-left: 5px;
1257
+ }
1258
+
1259
+ .wp-media-buttons a {
1260
+ text-decoration: none;
1261
+ color: #464646;
1262
+ font-size: 12px;
1263
+ }
1264
+
1265
+ .wp-media-buttons img {
1266
+ padding: 0 4px;
1267
+ vertical-align: middle;
1268
+ }
1269
+
1270
+ .wp-media-buttons span.wp-media-buttons-icon {
1271
+ display: inline-block;
1272
+ width: 18px;
1273
+ height: 18px;
1274
+ vertical-align: text-top;
1275
+ margin: 0 2px;
1276
+ }
1277
+
1278
+ .wp-media-buttons .add_media span.wp-media-buttons-icon {
1279
+ background: none;
1280
+ }
1281
+
1282
+ .wp-media-buttons .add_media span.wp-media-buttons-icon:before {
1283
+ font: normal 18px/1 'dashicons';
1284
+ speak: none;
1285
+ -webkit-font-smoothing: antialiased;
1286
+ -moz-osx-font-smoothing: grayscale;
1287
+ }
1288
+
1289
+ .wp-media-buttons .add_media span.wp-media-buttons-icon:before {
1290
+ content: '\f104';
1291
+ }
1292
+
1293
+ .quicktags-toolbar {
1294
+ border-bottom-style: solid;
1295
+ border-bottom-width: 1px;
1296
+ -webkit-border-top-right-radius: 3px;
1297
+ -webkit-border-top-left-radius: 3px;
1298
+ border-top-right-radius: 3px;
1299
+ border-top-left-radius: 3px;
1300
+ padding: 2px 8px 0;
1301
+ min-height: 29px;
1302
+ }
1303
+
1304
+ .quicktags-toolbar > div {
1305
+ padding: 2px 4px 0;
1306
+ }
1307
+
1308
+ .quicktags-toolbar input {
1309
+ margin: 2px 1px 4px;
1310
+ line-height: 18px;
1311
+ display: inline-block;
1312
+ min-width: 26px;
1313
+ padding: 2px 4px;
1314
+ font: 12px/18px "Open Sans", sans-serif;
1315
+ color: #464646;
1316
+ border: 1px solid #c3c3c3;
1317
+ -webkit-border-radius: 3px;
1318
+ border-radius: 3px;
1319
+ background: #eee;
1320
+ background-image: -webkit-gradient(linear, left bottom, left top, from(#e3e3e3), to(#fff));
1321
+ background-image: -webkit-linear-gradient(bottom, #e3e3e3, #fff);
1322
+ background-image: -moz-linear-gradient(bottom, #e3e3e3, #fff);
1323
+ background-image: -o-linear-gradient(bottom, #e3e3e3, #fff);
1324
+ background-image: linear-gradient(to top, #e3e3e3, #fff);
1325
+ }
1326
+
1327
+ .quicktags-toolbar input:hover {
1328
+ border-color: #aaa;
1329
+ background: #ddd;
1330
+ }
1331
+
1332
+ .quicktags-toolbar input[value="link"] {
1333
+ text-decoration: underline;
1334
+ }
1335
+
1336
+ .quicktags-toolbar input[value="del"] {
1337
+ text-decoration: line-through;
1338
+ }
1339
+
1340
+ .quicktags-toolbar input[value="i"] {
1341
+ font-style: italic;
1342
+ }
1343
+
1344
+ .quicktags-toolbar input[value="b"] {
1345
+ font-weight: bold;
1346
+ }
1347
+
1348
+ #wp_editbtns,
1349
+ #wp_gallerybtns {
1350
+ padding: 2px;
1351
+ position: absolute;
1352
+ display: none;
1353
+ z-index: 155000;
1354
+ }
1355
+
1356
+ #wp_editimgbtn,
1357
+ #wp_delimgbtn,
1358
+ #wp_editgallery,
1359
+ #wp_delgallery {
1360
+ border-color: #999;
1361
+ background-color: #eee;
1362
+ margin: 2px;
1363
+ padding: 2px;
1364
+ border-width: 1px;
1365
+ border-style: solid;
1366
+ -webkit-border-radius: 3px;
1367
+ border-radius: 3px;
1368
+ }
1369
+
1370
+ #wp_editimgbtn:hover,
1371
+ #wp_delimgbtn:hover,
1372
+ #wp_editgallery:hover,
1373
+ #wp_delgallery:hover {
1374
+ border-color: #555;
1375
+ background-color: #ccc;
1376
+ }
1377
+
1378
+ /*------------------------------------------------------------------------------
1379
+ wp-link
1380
+ ------------------------------------------------------------------------------*/
1381
+
1382
+ #wp-link {
1383
+ background-color: #F5F5F5;
1384
+ line-height: 1.4em;
1385
+ font-size: 12px;
1386
+ }
1387
+
1388
+ #wp-link ol,
1389
+ #wp-link ul {
1390
+ list-style: none;
1391
+ margin: 0;
1392
+ padding: 0;
1393
+ }
1394
+
1395
+ #wp-link input[type="text"] {
1396
+ -webkit-box-sizing: border-box;
1397
+ }
1398
+
1399
+ #wp-link input[type="text"],
1400
+ #wp-link textarea {
1401
+ border-width: 1px;
1402
+ border-style: solid;
1403
+ -webkit-border-radius: 4px;
1404
+ border-radius: 4px;
1405
+ font-size: 12px;
1406
+ margin: 1px;
1407
+ padding: 3px;
1408
+ }
1409
+
1410
+ #wp-link #link-options {
1411
+ padding: 10px 0 14px;
1412
+ border-bottom: 1px solid #dfdfdf;
1413
+ margin: 0 6px 14px;
1414
+ }
1415
+
1416
+ #wp-link p.howto {
1417
+ margin: 3px;
1418
+ }
1419
+
1420
+ #wp-link #internal-toggle {
1421
+ display: inline-block;
1422
+ cursor: pointer;
1423
+ padding-left: 18px;
1424
+ }
1425
+
1426
+ #wp-link .toggle-arrow {
1427
+ background: transparent url("../../../../../wp-includes/images/toggle-arrow.png") top left no-repeat;
1428
+ height: 23px;
1429
+ line-height: 23px;
1430
+ }
1431
+
1432
+ #wp-link .toggle-arrow-active {
1433
+ background-position: center left;
1434
+ }
1435
+
1436
+ #wp-link label input[type="text"] {
1437
+ width: 360px;
1438
+ margin-top: 5px;
1439
+ }
1440
+
1441
+ #wp-link #link-options label span,
1442
+ #wp-link #search-panel label span.search-label {
1443
+ display: inline-block;
1444
+ width: 80px;
1445
+ text-align: right;
1446
+ padding-right: 5px;
1447
+ }
1448
+
1449
+ #wp-link .link-search-field {
1450
+ float: left;
1451
+ width: 220px;
1452
+ }
1453
+
1454
+ #wp-link .link-search-wrapper {
1455
+ margin: 5px 6px 9px;
1456
+ display: block;
1457
+ overflow: hidden;
1458
+ }
1459
+
1460
+ #wp-link .link-search-wrapper span {
1461
+ float: left;
1462
+ margin-top: 4px;
1463
+ }
1464
+
1465
+ #wp-link .link-search-wrapper .spinner {
1466
+ display: none;
1467
+ vertical-align: text-bottom;
1468
+ }
1469
+
1470
+ #wp-link .link-target {
1471
+ width: auto;
1472
+ padding: 3px 0 0;
1473
+ margin: 0 0 0 87px;
1474
+ font-size: 11px;
1475
+ }
1476
+
1477
+ #wp-link .query-results {
1478
+ border: 1px #dfdfdf solid;
1479
+ margin: 0 5px 5px;
1480
+ background: #fff;
1481
+ height: 185px;
1482
+ overflow: auto;
1483
+ position: relative;
1484
+ }
1485
+
1486
+ #wp-link li,
1487
+ #wp-link .query-notice {
1488
+ clear: both;
1489
+ margin-bottom: 0;
1490
+ border-bottom: 1px solid #f1f1f1;
1491
+ color: #333;
1492
+ padding: 4px 6px;
1493
+ cursor: pointer;
1494
+ position: relative;
1495
+ }
1496
+
1497
+ #wp-link li:hover {
1498
+ background: #eaf2fa;
1499
+ color: #151515;
1500
+ }
1501
+
1502
+ #wp-link li.unselectable {
1503
+ border-bottom: 1px solid #dfdfdf;
1504
+ }
1505
+
1506
+ #wp-link li.unselectable:hover {
1507
+ background: #fff;
1508
+ cursor: auto;
1509
+ color: #333;
1510
+ }
1511
+
1512
+ #wp-link li.selected {
1513
+ background: #ddd;
1514
+ color: #333;
1515
+ }
1516
+
1517
+ #wp-link li.selected .item-title {
1518
+ font-weight: bold;
1519
+ }
1520
+
1521
+ #wp-link .item-title {
1522
+ display: inline-block;
1523
+ width: 80%;
1524
+ }
1525
+
1526
+ #wp-link .item-info {
1527
+ text-transform: uppercase;
1528
+ color: #666;
1529
+ font-size: 11px;
1530
+ position: absolute;
1531
+ right: 5px;
1532
+ top: 4px;
1533
+ bottom: 0;
1534
+ }
1535
+
1536
+ #wp-link #search-results {
1537
+ display: none;
1538
+ }
1539
+
1540
+ #wp-link #search-panel {
1541
+ float: left;
1542
+ width: 100%;
1543
+ }
1544
+
1545
+ #wp-link .river-waiting {
1546
+ display: none;
1547
+ padding: 10px 0;
1548
+ }
1549
+
1550
+ #wp-link .river-waiting .spinner {
1551
+ margin: 0 auto;
1552
+ display: block;
1553
+ }
1554
+
1555
+ #wp-link .submitbox {
1556
+ padding: 5px 10px;
1557
+ font-size: 11px;
1558
+ overflow: auto;
1559
+ height: 29px;
1560
+ }
1561
+
1562
+ #wp-link-cancel {
1563
+ line-height: 25px;
1564
+ float: left;
1565
+ }
1566
+
1567
+ #wp-link-update {
1568
+ line-height: 23px;
1569
+ float: right;
1570
+ }
1571
+
1572
+ /*!
1573
+ * jQuery UI CSS Framework 1.10.1
1574
+ * http://jqueryui.com
1575
+ *
1576
+ * Copyright 2013 jQuery Foundation and other contributors
1577
+ * Released under the MIT license.
1578
+ * http://jquery.org/license
1579
+ *
1580
+ * http://docs.jquery.com/UI/Theming/API
1581
+ */
1582
+
1583
+ /* Layout helpers
1584
+ ----------------------------------*/
1585
+ .ui-helper-hidden {
1586
+ display: none;
1587
+ }
1588
+ .ui-helper-hidden-accessible {
1589
+ border: 0;
1590
+ clip: rect(0 0 0 0);
1591
+ height: 1px;
1592
+ margin: -1px;
1593
+ overflow: hidden;
1594
+ padding: 0;
1595
+ position: absolute;
1596
+ width: 1px;
1597
+ }
1598
+ .ui-helper-reset {
1599
+ margin: 0;
1600
+ padding: 0;
1601
+ border: 0;
1602
+ outline: 0;
1603
+ line-height: 1.3;
1604
+ text-decoration: none;
1605
+ font-size: 100%;
1606
+ list-style: none;
1607
+ }
1608
+ .ui-helper-clearfix:before,
1609
+ .ui-helper-clearfix:after {
1610
+ content: "";
1611
+ display: table;
1612
+ border-collapse: collapse;
1613
+ }
1614
+ .ui-helper-clearfix:after {
1615
+ clear: both;
1616
+ }
1617
+ .ui-helper-clearfix {
1618
+ min-height: 0; /* support: IE7 */
1619
+ }
1620
+ .ui-helper-zfix {
1621
+ width: 100%;
1622
+ height: 100%;
1623
+ top: 0;
1624
+ left: 0;
1625
+ position: absolute;
1626
+ opacity: 0;
1627
+ filter:Alpha(Opacity=0);
1628
+ }
1629
+
1630
+ .ui-front {
1631
+ z-index: 100;
1632
+ }
1633
+
1634
+
1635
+ /* Interaction Cues
1636
+ ----------------------------------*/
1637
+ .ui-state-disabled {
1638
+ cursor: default !important;
1639
+ }
1640
+
1641
+
1642
+ /* Icons
1643
+ ----------------------------------*/
1644
+
1645
+ /* states and images */
1646
+ .ui-icon {
1647
+ display: block;
1648
+ text-indent: -99999px;
1649
+ overflow: hidden;
1650
+ background-repeat: no-repeat;
1651
+ }
1652
+
1653
+
1654
+ /* Misc visuals
1655
+ ----------------------------------*/
1656
+
1657
+ /* Overlays */
1658
+ .ui-widget-overlay {
1659
+ position: fixed;
1660
+ top: 0;
1661
+ left: 0;
1662
+ width: 100%;
1663
+ height: 100%;
1664
+ }
1665
+
1666
+
1667
+ /*!
1668
+ * jQuery UI Resizable 1.10.1
1669
+ * http://jqueryui.com
1670
+ *
1671
+ * Copyright 2013 jQuery Foundation and other contributors
1672
+ * Released under the MIT license.
1673
+ * http://jquery.org/license
1674
+ *
1675
+ * http://docs.jquery.com/UI/Resizable#theming
1676
+ */
1677
+ .ui-resizable {
1678
+ position: relative;
1679
+ }
1680
+ .ui-resizable-handle {
1681
+ position: absolute;
1682
+ font-size: 0.1px;
1683
+ display: block;
1684
+ }
1685
+ .ui-resizable-disabled .ui-resizable-handle,
1686
+ .ui-resizable-autohide .ui-resizable-handle {
1687
+ display: none;
1688
+ }
1689
+ .ui-resizable-n {
1690
+ cursor: n-resize;
1691
+ height: 7px;
1692
+ width: 100%;
1693
+ top: -5px;
1694
+ left: 0;
1695
+ }
1696
+ .ui-resizable-s {
1697
+ cursor: s-resize;
1698
+ height: 7px;
1699
+ width: 100%;
1700
+ bottom: -5px;
1701
+ left: 0;
1702
+ }
1703
+ .ui-resizable-e {
1704
+ cursor: e-resize;
1705
+ width: 7px;
1706
+ right: -5px;
1707
+ top: 0;
1708
+ height: 100%;
1709
+ }
1710
+ .ui-resizable-w {
1711
+ cursor: w-resize;
1712
+ width: 7px;
1713
+ left: -5px;
1714
+ top: 0;
1715
+ height: 100%;
1716
+ }
1717
+ .ui-resizable-se {
1718
+ cursor: se-resize;
1719
+ width: 12px;
1720
+ height: 12px;
1721
+ right: 1px;
1722
+ bottom: 1px;
1723
+ }
1724
+ .ui-resizable-sw {
1725
+ cursor: sw-resize;
1726
+ width: 9px;
1727
+ height: 9px;
1728
+ left: -5px;
1729
+ bottom: -5px;
1730
+ }
1731
+ .ui-resizable-nw {
1732
+ cursor: nw-resize;
1733
+ width: 9px;
1734
+ height: 9px;
1735
+ left: -5px;
1736
+ top: -5px;
1737
+ }
1738
+ .ui-resizable-ne {
1739
+ cursor: ne-resize;
1740
+ width: 9px;
1741
+ height: 9px;
1742
+ right: -5px;
1743
+ top: -5px;
1744
+ }
1745
+
1746
+ /*!
1747
+ * jQuery UI Dialog 1.10.1
1748
+ * http://jqueryui.com
1749
+ *
1750
+ * Copyright 2013 jQuery Foundation and other contributors
1751
+ * Released under the MIT license.
1752
+ * http://jquery.org/license
1753
+ *
1754
+ * http://docs.jquery.com/UI/Dialog#theming
1755
+ */
1756
+ .ui-dialog {
1757
+ position: absolute;
1758
+ top: 0;
1759
+ /* @noflip */
1760
+ left: 0;
1761
+ padding: .2em;
1762
+ outline: 0;
1763
+ }
1764
+ .ui-dialog .ui-dialog-titlebar {
1765
+ padding: .4em 1em;
1766
+ position: relative;
1767
+ }
1768
+ .ui-dialog .ui-dialog-title {
1769
+ float: left;
1770
+ margin: .1em 0;
1771
+ white-space: nowrap;
1772
+ width: 90%;
1773
+ overflow: hidden;
1774
+ text-overflow: ellipsis;
1775
+ }
1776
+ .ui-dialog .ui-dialog-titlebar-close {
1777
+ position: absolute;
1778
+ right: .3em;
1779
+ top: 50%;
1780
+ width: 21px;
1781
+ margin: -10px 0 0 0;
1782
+ padding: 1px;
1783
+ height: 20px;
1784
+ }
1785
+ .ui-dialog .ui-dialog-content {
1786
+ position: relative;
1787
+ border: 0;
1788
+ padding: .5em 1em;
1789
+ background: none;
1790
+ overflow: auto;
1791
+ }
1792
+ .ui-dialog .ui-dialog-buttonpane {
1793
+ text-align: left;
1794
+ border-width: 1px 0 0 0;
1795
+ background-image: none;
1796
+ margin-top: .5em;
1797
+ padding: .3em 1em .5em .4em;
1798
+ }
1799
+ .ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset {
1800
+ float: right;
1801
+ }
1802
+ .ui-dialog .ui-dialog-buttonpane button {
1803
+ margin: .5em .4em .5em 0;
1804
+ cursor: pointer;
1805
+ }
1806
+ .ui-dialog .ui-resizable-se {
1807
+ width: 12px;
1808
+ height: 12px;
1809
+ right: -5px;
1810
+ bottom: -5px;
1811
+ background-position: 16px 16px;
1812
+ }
1813
+ .ui-draggable .ui-dialog-titlebar {
1814
+ cursor: move;
1815
+ }
1816
+
1817
+ /* WP jQuery Dialog Theme */
1818
+ .wp-dialog {
1819
+ padding: 0;
1820
+ z-index: 300002;
1821
+ border: 0;
1822
+ -webkit-box-shadow: 0 5px 15px rgba(0,0,0,0.7);
1823
+ box-shadow: 0 5px 15px rgba(0,0,0,0.7);
1824
+ background-color: #f5f5f5;
1825
+ }
1826
+
1827
+ .wp-dialog .ui-dialog-title {
1828
+ display: block;
1829
+ text-align: center;
1830
+ padding: 0;
1831
+ }
1832
+
1833
+ .wp-dialog .ui-dialog-titlebar {
1834
+ padding: 0 1em;
1835
+ background-color: #444;
1836
+ font-size: 13px;
1837
+ line-height: 24px;
1838
+ color: #fff;
1839
+ }
1840
+
1841
+ .wp-dialog .ui-dialog-content {
1842
+ padding: 0;
1843
+ }
1844
+
1845
+ .wp-dialog .ui-dialog-titlebar-close {
1846
+ cursor: pointer;
1847
+ -webkit-appearance: none;
1848
+ border: 0;
1849
+ width: 30px;
1850
+ height: 20px;
1851
+ top: 13px;
1852
+ right: 6px;
1853
+ background: none;
1854
+ }
1855
+
1856
+ .wp-dialog .ui-dialog-titlebar-close:before {
1857
+ content: '\f158';
1858
+ font: normal 20px/1 'dashicons';
1859
+ speak: none;
1860
+ -webkit-font-smoothing: antialiased;
1861
+ -moz-osx-font-smoothing: grayscale;
1862
+ color: #999;
1863
+ padding-left: 12px;
1864
+ }
1865
+
1866
+ .wp-dialog .ui-dialog-titlebar-close:hover:before {
1867
+ color: #2ea2cc;
1868
+ }
1869
+
1870
+ .wp-dialog .ui-dialog-titlebar-close .ui-button-text {
1871
+ display: none;
1872
+ }
1873
+
1874
+ .wp-dialog .ui-dialog-titlebar-close:hover,
1875
+ .wp-dialog .ui-dialog-titlebar-close:focus {
1876
+ background-position: -87px -32px;
1877
+ }
1878
+
1879
+ .ui-widget-overlay {
1880
+ z-index: 300001;
1881
+ background-color: #000;
1882
+ opacity: 0.6;
1883
+ filter: alpha(opacity=60);
1884
+ }
1885
+
1886
+ /* TinyMCE modal */
1887
+ .clearlooks2 .mceTop {
1888
+ border-bottom: 1px solid #ccc;
1889
+ }
1890
+
1891
+ .clearlooks2 .mceTop span {
1892
+ font: 13px/24px "Open Sans", sans-serif;
1893
+ color: #e5e5e5;
1894
+ }
1895
+
1896
+ .clearlooks2 .mceTop .mceLeft {
1897
+ background: #444444;
1898
+ border-color: transparent;
1899
+ }
1900
+
1901
+ .clearlooks2 .mceTop .mceRight {
1902
+ background: #444444;
1903
+ border-color: transparent;
1904
+ }
1905
+
1906
+ .clearlooks2 .mceMiddle {
1907
+ clip: rect(24px auto auto auto);
1908
+ }
1909
+
1910
+ .clearlooks2 .mceMiddle .mceLeft {
1911
+ background: #f1f1f1;
1912
+ border-color: transparent;
1913
+ }
1914
+
1915
+ .clearlooks2 .mceMiddle .mceRight {
1916
+ background: #f1f1f1;
1917
+ border-color: transparent;
1918
+ }
1919
+
1920
+ .clearlooks2 .mceBottom {
1921
+ background: #f1f1f1;
1922
+ border-color: transparent;
1923
+ }
1924
+
1925
+ .clearlooks2 .mceBottom .mceLeft {
1926
+ background: #f1f1f1;
1927
+ border-color: transparent;
1928
+ }
1929
+
1930
+ .clearlooks2 .mceBottom .mceCenter {
1931
+ background: #f1f1f1;
1932
+ border-color: transparent;
1933
+ }
1934
+
1935
+ .clearlooks2 .mceBottom .mceRight {
1936
+ background: #f1f1f1;
1937
+ border-color: transparent;
1938
+ }
1939
+
1940
+ .clearlooks2 .mceClose,
1941
+ .clearlooks2 .mceFocus .mceClose,
1942
+ .clearlooks2 .mceFocus .mceClose:hover {
1943
+ background-image: none;
1944
+ }
1945
+ .clearlooks2 .mceClose:before {
1946
+ speak: none;
1947
+ -webkit-font-smoothing: antialiased;
1948
+ -moz-osx-font-smoothing: grayscale;
1949
+ color: #999;
1950
+ padding-left: 12px;
1951
+ }
1952
+
1953
+ /* Distraction Free Writing mode
1954
+ * =Overlay Styles
1955
+ -------------------------------------------------------------- */
1956
+ .fullscreen-overlay {
1957
+ z-index: 149999;
1958
+ display: none;
1959
+ position: fixed;
1960
+ top: 0;
1961
+ bottom: 0;
1962
+ left: 0;
1963
+ right: 0;
1964
+ filter: inherit;
1965
+ }
1966
+
1967
+ .fullscreen-active .fullscreen-overlay,
1968
+ .fullscreen-active #wp-fullscreen-body {
1969
+ display: block;
1970
+ }
1971
+
1972
+ .fullscreen-fader {
1973
+ z-index: 200000;
1974
+ }
1975
+
1976
+ .fullscreen-active .fullscreen-fader {
1977
+ display: none;
1978
+ }
1979
+
1980
+ /* =Overlay Body
1981
+ -------------------------------------------------------------- */
1982
+ #wp-fullscreen-body {
1983
+ width: 100%;
1984
+ z-index: 150005;
1985
+ display: none;
1986
+ position: absolute;
1987
+ top: 0;
1988
+ left: 0;
1989
+ font-size: 12px;
1990
+ }
1991
+
1992
+ #wp-fullscreen-wrap {
1993
+ margin: 0 auto 50px;
1994
+ position: relative;
1995
+ padding-top: 60px;
1996
+ }
1997
+
1998
+ #wp-fullscreen-title {
1999
+ font-size: 1.7em;
2000
+ line-height: 100%;
2001
+ outline: medium none;
2002
+ padding: 6px 7px;
2003
+ width: 100%;
2004
+ margin-bottom: 30px;
2005
+ -webkit-box-shadow: none;
2006
+ box-shadow: none;
2007
+ }
2008
+
2009
+ #wp-fullscreen-container {
2010
+ padding: 4px 10px 50px;
2011
+ }
2012
+
2013
+ #wp-fullscreen-title,
2014
+ #wp-fullscreen-container {
2015
+ -webkit-border-radius: 0;
2016
+ border-radius: 0;
2017
+ border: 1px dashed transparent;
2018
+ background: transparent;
2019
+ -moz-transition-property: border-color;
2020
+ -moz-transition-duration: 0.6s;
2021
+ -webkit-transition-property: border-color;
2022
+ -webkit-transition-duration: 0.6s;
2023
+ -o-transition-property: border-color;
2024
+ -o-transition-duration: 0.6s;
2025
+ transition-property: border-color;
2026
+ transition-duration: 0.6s;
2027
+ }
2028
+
2029
+ #wp_mce_fullscreen {
2030
+ width: 100%;
2031
+ min-height: 300px;
2032
+ border: 0;
2033
+ background: transparent;
2034
+ font-family: Consolas, Monaco, monospace;
2035
+ line-height: 1.6em;
2036
+ padding: 0;
2037
+ overflow-y: hidden;
2038
+ outline: none;
2039
+ resize: none;
2040
+ -webkit-box-shadow: none;
2041
+ box-shadow: none;
2042
+ }
2043
+
2044
+ #wp-fullscreen-tagline {
2045
+ color: #BBBBBB;
2046
+ font-size: 18px;
2047
+ float: right;
2048
+ padding-top: 5px;
2049
+ }
2050
+
2051
+ /* =Top bar
2052
+ -------------------------------------------------------------- */
2053
+ #fullscreen-topbar {
2054
+ background: #f5f5f5;
2055
+ border-bottom: 1px solid #fff;
2056
+ height: 40px;
2057
+ left: 0;
2058
+ min-width: 800px;
2059
+ position: fixed;
2060
+ top: 0;
2061
+ width: 100%;
2062
+ z-index: 150050;
2063
+ }
2064
+
2065
+ #wp-fullscreen-toolbar {
2066
+ padding: 6px 10px 0;
2067
+ clear: both;
2068
+ max-width: 1100px;
2069
+ min-width: 820px;
2070
+ margin: 0 auto;
2071
+ }
2072
+
2073
+ #wp-fullscreen-mode-bar,
2074
+ #wp-fullscreen-button-bar,
2075
+ #wp-fullscreen-close,
2076
+ #wp-fullscreen-count {
2077
+ float: left;
2078
+ }
2079
+
2080
+ #wp-fullscreen-save {
2081
+ float: right;
2082
+ padding: 2px 2px 0 5px;
2083
+ }
2084
+
2085
+ #wp-fullscreen-count,
2086
+ #wp-fullscreen-close {
2087
+ padding-top: 5px;
2088
+ }
2089
+
2090
+ #wp-fullscreen-central-toolbar {
2091
+ margin: auto;
2092
+ padding: 0;
2093
+ }
2094
+
2095
+ #wp-fullscreen-buttons > div {
2096
+ float: left;
2097
+ }
2098
+
2099
+ #wp-fullscreen-mode-bar {
2100
+ padding: 1px 14px 0 0;
2101
+ }
2102
+
2103
+ #wp-fullscreen-modes a {
2104
+ display: block;
2105
+ font-size: 11px;
2106
+ text-decoration: none;
2107
+ float: left;
2108
+ margin: 1px 0 0 0;
2109
+ padding: 2px 6px 2px;
2110
+ border-width: 1px 1px 1px 0;
2111
+ border-style: solid;
2112
+ border-color: #bbb;
2113
+ color: #777;
2114
+ text-shadow: 0 1px 0 #fff;
2115
+ background-color: #f4f4f4;
2116
+ background: #f4f4f4;
2117
+ background-image: -webkit-gradient(linear, left bottom, left top, from(#e4e4e4), to(#f9f9f9));
2118
+ background-image: -webkit-linear-gradient(bottom, #e4e4e4, #f9f9f9);
2119
+ background-image: -moz-linear-gradient(bottom, #e4e4e4, #f9f9f9);
2120
+ background-image: -o-linear-gradient(bottom, #e4e4e4, #f9f9f9);
2121
+ background-image: linear-gradient(to top, #e4e4e4, #f9f9f9);
2122
+ }
2123
+
2124
+ #wp-fullscreen-modes a:hover,
2125
+ .wp-html-mode #wp-fullscreen-modes a:last-child,
2126
+ .wp-tmce-mode #wp-fullscreen-modes a:first-child {
2127
+ color: #333;
2128
+ border-color: #999;
2129
+ background: #eee;
2130
+ background-image: -webkit-gradient(linear, left top, left bottom, from(#e4e4e4), to(#f9f9f9));
2131
+ background-image: -webkit-linear-gradient(top, #e4e4e4, #f9f9f9);
2132
+ background-image: -moz-linear-gradient(top, #e4e4e4, #f9f9f9);
2133
+ background-image: -o-linear-gradient(top, #e4e4e4, #f9f9f9);
2134
+ background-image: linear-gradient(to bottom, #e4e4e4, #f9f9f9);
2135
+ }
2136
+
2137
+ #wp-fullscreen-modes a:first-child {
2138
+ border-width: 1px;
2139
+ -webkit-border-top-left-radius: 3px;
2140
+ -webkit-border-bottom-left-radius: 3px;
2141
+ border-top-left-radius: 3px;
2142
+ border-bottom-left-radius: 3px;
2143
+ }
2144
+
2145
+ #wp-fullscreen-modes a:last-child {
2146
+ -webkit-border-top-right-radius: 3px;
2147
+ -webkit-border-bottom-right-radius: 3px;
2148
+ border-top-right-radius: 3px;
2149
+ border-bottom-right-radius: 3px;
2150
+ }
2151
+
2152
+ #wp-fullscreen-buttons .active a {
2153
+ background: inherit;
2154
+ }
2155
+
2156
+ #wp-fullscreen-buttons .hidden {
2157
+ display: none;
2158
+ }
2159
+
2160
+ #wp-fullscreen-buttons .disabled {
2161
+ opacity: 0.5;
2162
+ }
2163
+
2164
+ .wp-html-mode #wp-fullscreen-buttons div {
2165
+ display: none;
2166
+ }
2167
+
2168
+ .wp-html-mode #wp-fullscreen-buttons div.wp-fullscreen-both {
2169
+ display: block;
2170
+ }
2171
+
2172
+ #fullscreen-topbar.fullscreen-make-sticky {
2173
+ display: block !important;
2174
+ }
2175
+
2176
+ #wp-fullscreen-save img {
2177
+ vertical-align: middle;
2178
+ }
2179
+
2180
+ #wp-fullscreen-save img,
2181
+ #wp-fullscreen-save span {
2182
+ padding-right: 4px;
2183
+ display: none;
2184
+ }
2185
+
2186
+ /* =Thickbox Adjustments
2187
+ -------------------------------------------------------------- */
2188
+ .fullscreen-active #TB_overlay {
2189
+ z-index: 150100;
2190
+ }
2191
+
2192
+ .fullscreen-active #TB_window {
2193
+ z-index: 150102;
2194
+ }
2195
+
2196
+ /* =TinyMCE Adjustments
2197
+ -------------------------------------------------------------- */
2198
+ #wp_mce_fullscreen_ifr {
2199
+ background: transparent;
2200
+ }
2201
+
2202
+ #wp_mce_fullscreen_parent #wp_mce_fullscreen_tbl tr.mceFirst {
2203
+ display : none;
2204
+ }
2205
+
2206
+ #wp-fullscreen-container .wp_themeSkin table td {
2207
+ vertical-align: top;
2208
+ }
2209
+
2210
+ /* Colors */
2211
+ .fullscreen-overlay {
2212
+ background: #fff;
2213
+ }
2214
+
2215
+ .wp-fullscreen-focus #wp-fullscreen-title,
2216
+ .wp-fullscreen-focus #wp-fullscreen-container {
2217
+ border-color: #ccc;
2218
+ }
2219
+
2220
+ /* =CSS 3 transitions
2221
+ -------------------------------------------------------------- */
2222
+
2223
+ .fade-1000,
2224
+ .fade-600,
2225
+ .fade-400,
2226
+ .fade-300 {
2227
+ opacity: 0;
2228
+ -moz-transition-property: opacity;
2229
+ -webkit-transition-property: opacity;
2230
+ -o-transition-property: opacity;
2231
+ transition-property: opacity;
2232
+ }
2233
+
2234
+ .fade-1000 {
2235
+ -moz-transition-duration: 1s;
2236
+ -webkit-transition-duration: 1s;
2237
+ -o-transition-duration: 1s;
2238
+ transition-duration: 1s;
2239
+ }
2240
+
2241
+ .fade-600 {
2242
+ -moz-transition-duration: 0.6s;
2243
+ -webkit-transition-duration: 0.6s;
2244
+ -o-transition-duration: 0.6s;
2245
+ transition-duration: 0.6s;
2246
+ }
2247
+
2248
+ .fade-400 {
2249
+ -moz-transition-duration: 0.4s;
2250
+ -webkit-transition-duration: 0.4s;
2251
+ -o-transition-duration: 0.4s;
2252
+ transition-duration: 0.4s;
2253
+ }
2254
+
2255
+ .fade-300 {
2256
+ -moz-transition-duration: 0.3s;
2257
+ -webkit-transition-duration: 0.3s;
2258
+ -o-transition-duration: 0.3s;
2259
+ transition-duration: 0.3s;
2260
+ }
2261
+
2262
+ .fade-trigger {
2263
+ opacity: 1;
2264
+ }
2265
+
2266
+ /* =Localization
2267
+ -------------------------------------------------------------- */
2268
+ .rtl .wp_themeSkin .mceColorSplitMenu a.mceMoreColors,
2269
+ .rtl .wp_themeSkin .mceMenu .mceText,
2270
+ .rtl .wp-switch-editor,
2271
+ .rtl .quicktags-toolbar input,
2272
+ .rtl .clearlooks2 .mceTop span,
2273
+ .rtl .wp_themeSkin .mceColorSplitMenu a.mceMoreColors {
2274
+ font-family: Tahoma, sans-serif;
2275
+ }
2276
+
2277
+ html:lang(he-il) .rtl .wp_themeSkin .mceColorSplitMenu a.mceMoreColors,
2278
+ html:lang(he-il) .rtl .wp_themeSkin .mceMenu .mceText,
2279
+ html:lang(he-il) .rtl .wp-switch-editor,
2280
+ html:lang(he-il) .rtl .quicktags-toolbar input,
2281
+ html:lang(he-il) .rtl .clearlooks2 .mceTop span,
2282
+ html:lang(he-il) .rtl .wp_themeSkin .mceColorSplitMenu a.mceMoreColors {
2283
+ font-family: Arial, sans-serif;
2284
+ }
2285
+
2286
+
2287
+ /* HiDPI */
2288
+ @media print,
2289
+ (-o-min-device-pixel-ratio: 5/4),
2290
+ (-webkit-min-device-pixel-ratio: 1.25),
2291
+ (min-resolution: 120dpi) {
2292
+ .wp-media-buttons .add_media span.wp-media-buttons-icon,
2293
+ #wp-fullscreen-buttons #wp_fs_image span.mce_image {
2294
+ background: none;
2295
+ }
2296
+
2297
+ .wp_themeSkin .mceListBox .mceOpen,
2298
+ .wp_themeSkin .mceListBoxHover .mceOpen,
2299
+ .wp_themeSkin .mceListBoxSelected .mceOpen,
2300
+ .wp_themeSkin table.mceListBoxEnabled .mceOpen {
2301
+ background-image: url('../../../../../wp-includes/images/down_arrow-2x.gif');
2302
+ background-size: 10px 20px;
2303
+ }
2304
+
2305
+ .wp_themeSkin .mceSplitButtonEnabled a.mceOpen,
2306
+ .wp_themeSkin .mceSplitButtonSelected a.mceOpen,
2307
+ .wp_themeSkin .mceSplitButtonActive a.mceOpen,
2308
+ .wp_themeSkin .mceSplitButtonEnabled:hover a.mceOpen {
2309
+ background-image: url('../../../../../wp-includes/images/down_arrow-2x.gif');
2310
+ background-size: 10px 20px;
2311
+ }
2312
+
2313
+ #wp-link .toggle-arrow {
2314
+ background: transparent url('../../../../../wp-includes/images/toggle-arrow-2x.png') top left no-repeat;
2315
+ background-size: 19px 69px;
2316
+ }
2317
+ }
css/tmce/index.html ADDED
File without changes
css/tmce/panelbtns.css ADDED
@@ -0,0 +1,23 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #wp_wysijaregister {
2
+ display: none;
3
+ padding: 2px;
4
+ position: absolute;
5
+ z-index: 999998;
6
+ }
7
+
8
+ #wp_edit_wysinl_btn:hover, #wp_del_wysinl_btn:hover {
9
+ background-color: #CCCCCC;
10
+ border-color: #555555;
11
+ cursor:pointer;
12
+ }
13
+
14
+ #wp_edit_wysinl_btn, #wp_del_wysinl_btn {
15
+ background-color: #EEEEEE;
16
+ border-color: #999999;
17
+ border-radius: 3px 3px 3px 3px;
18
+ border-style: solid;
19
+ border-width: 1px;
20
+ margin: 2px;
21
+ padding: 2px;
22
+ }
23
+
css/tmce/style.css ADDED
@@ -0,0 +1,7 @@
 
 
 
 
 
 
 
1
+ .clear{clear:both;}
2
+ .wysija-register{
3
+ text-indent:-9999px;
4
+ background:url("../../img/nl-placeholder.png") no-repeat scroll center top transparent;
5
+ width:188px;
6
+ height:70px;
7
+ }
css/tmce/subscribers_count.css ADDED
@@ -0,0 +1,37 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #widget-form{
2
+ width: 180px;
3
+ float:left;
4
+ margin: 5px;
5
+ padding: 20px;
6
+ }
7
+
8
+ body,html{
9
+ margin:0 !important;
10
+ overflow: hidden;
11
+ font-family: sans-serif;
12
+ font-size: 12px !important;
13
+ line-height: 1.4em;
14
+ }
15
+
16
+ #formTable{
17
+ padding-top: 10px;
18
+ }
19
+
20
+ .wysija-widget-form .divblocks {
21
+ margin:0 0 10px 0;
22
+ }
23
+
24
+ .wysija-widget-form .divblocks div {
25
+ margin:0 0 5px 0;
26
+ }
27
+ .wysija-widget-form .divblocks select {
28
+ margin:0 !important;
29
+ }
30
+
31
+ .wysija-widget-form{
32
+ margin-bottom: 10px;
33
+ }
34
+
35
+ #widget-form select {
36
+ width:180px;
37
+ }
css/tmce/widget.css ADDED
@@ -0,0 +1,44 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #widget-form{
2
+ width: 180px;
3
+ float:left;
4
+ margin: 5px;
5
+ padding: 20px;
6
+ }
7
+
8
+ #subscriber-ccount-form{
9
+ width: 450px;
10
+ float:left;
11
+ margin: 5px;
12
+ padding: 20px;
13
+ }
14
+
15
+ body,html{
16
+ margin:0 !important;
17
+ overflow: hidden;
18
+ font-family: sans-serif;
19
+ font-size: 12px !important;
20
+ line-height: 1.4em;
21
+ }
22
+
23
+ #formTable{
24
+ padding-top: 10px;
25
+ }
26
+
27
+ .wysija-widget-form .divblocks {
28
+ margin:0 0 10px 0;
29
+ }
30
+
31
+ .wysija-widget-form .divblocks div {
32
+ margin:0 0 5px 0;
33
+ }
34
+ .wysija-widget-form .divblocks select {
35
+ margin:0 !important;
36
+ }
37
+
38
+ .wysija-widget-form{
39
+ margin-bottom: 10px;
40
+ }
41
+
42
+ #widget-form select {
43
+ width:180px;
44
+ }
css/validationEngine.jquery.css ADDED
@@ -0,0 +1 @@
 
1
+ .widget_wysija_cont .inputContainer{position:relative;float:left}.widget_wysija_cont .ajaxSubmit{padding:20px;background:#55ea55;border:1px solid #999;display:none}.formError{position:absolute;top:300px;left:300px;display:block;z-index:5000;cursor:pointer}.formError .formErrorContent{background:none repeat scroll 0 0 #8E2121;border:1px solid #BBBBBB;color:#FFFFFF;font-family:tahoma;font-size:11px;padding:4px 10px;position:relative;width:150px;z-index:5001}.widget_wysija_cont .formError .formErrorArrow{width:15px;margin:-2px 0 0 13px;position:relative;z-index:5006}.widget_wysija_cont .formError .formErrorArrowBottom{margin:0px 0 0 12px;top:2px}.widget_wysija_cont .formError .formErrorArrow div{border-left:2px solid #ddd;border-right:2px solid #ddd;font-size:0px;height:1px;background:#8E2121;margin:0 auto;line-height:0;font-size:0;display:block}.widget_wysija_cont .greenPopup .formErrorArrow div{background:#33be40}.widget_wysija_cont .blackPopup .formErrorArrow div{background:#393939;color:#FFF}.widget_wysija_cont .formError .formErrorArrow .line10{width:15px;border:none}.widget_wysija_cont .formError .formErrorArrow .line9{width:13px;border:none}.widget_wysija_cont .formError .formErrorArrow .line8{width:11px}.widget_wysija_cont .formError .formErrorArrow .line7{width:9px}.widget_wysija_cont .formError .formErrorArrow .line6{width:7px}.widget_wysija_cont .formError .formErrorArrow .line5{width:5px}.widget_wysija_cont .formError .formErrorArrow .line4{width:3px}.widget_wysija_cont .formError .formErrorArrow .line3{width:1px;border-left:2px solid #ddd;border-right:2px solid #ddd;border-bottom:0 solid #ddd}.widget_wysija_cont .formError .formErrorArrow .line2{width:3px;border:none;background:#ddd}.widget_wysija_cont .formError .formErrorArrow .line1{width:1px;border:none;background:#ddd}.greenPopup .formErrorContent{background:none repeat scroll 0 0 #FFFBCC;color:#555555}.widget_wysija_cont .updated,.widget_wysija_cont .error,.widget_wysija_cont .xdetailed-errors{margin:5px 0 15px;color:#333333;font-family:sans-serif;font-size:12px;line-height:1.4em}.widget_wysija_cont .error,.widget_wysija_cont .login #login_error{background-color:#FFEBE8;border-color:#CC0000}.widget_wysija_cont .updated,.widget_wysija_cont .error,.widget_wysija_cont .xdetailed-errors{border-radius:3px;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-style:solid;border-width:1px;margin:5px 15px 2px;padding:0 0.6em;color:#333333}.widget_wysija_cont.updated,.widget_wysija_cont .login .message{background-color:#FFFFE0;border-color:#E6DB55}.widget_wysija_cont .xdetailed-updated{background-color:#FFFFE0;border-color:#E6DB55;color:#000;font-weight:bold;display:none}.widget_wysija_cont .xdetailed-errors{background-color:#A50000;border-color:#888888;color:#DDDDDD;font-weight:bold;display:none}.widget_wysija_cont .instruct{font-style:italic;font-weight:bold;padding:5px 0;font-size:11px}.widget_wysija_cont .needInfo{padding-left:15px;text-decoration:underline}.widget_wysija_cont .wysija-submit{display:block;margin-top:20px}.widget_wysija_cont input.defaultlabels{color:#888}.widget_wysija_cont input.defaultlabels:focus{color:#373737}.widget_wysija_cont .abs-req{display:none}.widget_wysija_cont p label{display:block}.widget_wysija_cont .iframe-hidden{display:none}.widget_wysija_cont .updated,.widget_wysija_cont .error,.widget_wysija_cont .xdetailed-errors{background-color:#FFFFFF;border:1px solid;border-radius:0 0 0 0;font-weight:normal;margin:4px 0;padding:4px}.widget_wysija_cont .error,.widget_wysija_cont .xdetailed-errors{background-color:#e55057;color:#fff}.widget_wysija_cont .error,.widget_wysija_cont .xdetailed-errors{background-color:#e55057;color:#fff;border-color:#CC0000}.widget_wysija_cont .allmsgs ul{padding:0}.widget_wysija_cont .allmsgs li{list-style-type:none}.widget_wysija_cont .updated,.widget_wysija_cont .login .message{background-color:#FFFFE0;border-color:#E6DB55}.widget_wysija_cont .updated ul{margin:0}.widget_wysija_cont .showerrors{color:#DDDDDD}#wysija-subscriptions input.checkboxx{margin-left:0px;margin-right:10px}#wysija-subscriptions th label{margin-right:10px}#wysija-box-after-comment{width:25px}#wysija-box-after-register{margin:0 6px 0 0;padding:3px}.wysija-after-register{padding-bottom:10px}.wysija-unsubscribed-on{color:#bbb}.widget_wysija .wysija-required{font-weight:bold}.widget_wysija .wysija-radio-label,.widget_wysija .wysija-checkbox-label{margin:0 0 5px 0;cursor:pointer}.widget_wysija .wysija-radio,.widget_wysija .wysija-checkbox{margin:0 5px 0 0;vertical-align:middle}.xdetailed-errors,.xdetailed-updated{display:none}
css/vendor/bootstrap.tooltip.css ADDED
@@ -0,0 +1,101 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ .tooltip {
2
+ position: absolute;
3
+ z-index: 1030;
4
+ display: block;
5
+ visibility: visible;
6
+ font-size: 12px;
7
+ line-height: 1.4;
8
+ opacity: 0;
9
+ filter: alpha(opacity=0);
10
+ }
11
+ .tooltip.in {
12
+ opacity: 0.9;
13
+ filter: alpha(opacity=90);
14
+ }
15
+ .tooltip.top {
16
+ margin-top: -3px;
17
+ padding: 5px 0;
18
+ }
19
+ .tooltip.right {
20
+ margin-left: 3px;
21
+ padding: 0 5px;
22
+ }
23
+ .tooltip.bottom {
24
+ margin-top: 3px;
25
+ padding: 5px 0;
26
+ }
27
+ .tooltip.left {
28
+ margin-left: -3px;
29
+ padding: 0 5px;
30
+ }
31
+ .tooltip-inner {
32
+ max-width: 200px;
33
+ padding: 3px 8px;
34
+ color: #ffffff;
35
+ text-align: center;
36
+ text-decoration: none;
37
+ background-color: #000000;
38
+ border-radius: 4px;
39
+ }
40
+ .tooltip-inner a {
41
+ color: #fff;
42
+ }
43
+ .tooltip-arrow {
44
+ position: absolute;
45
+ width: 0;
46
+ height: 0;
47
+ border-color: transparent;
48
+ border-style: solid;
49
+ }
50
+ .tooltip.top .tooltip-arrow {
51
+ bottom: 0;
52
+ left: 50%;
53
+ margin-left: -5px;
54
+ border-width: 5px 5px 0;
55
+ border-top-color: #000000;
56
+ }
57
+ .tooltip.top-left .tooltip-arrow {
58
+ bottom: 0;
59
+ left: 5px;
60
+ border-width: 5px 5px 0;
61
+ border-top-color: #000000;
62
+ }
63
+ .tooltip.top-right .tooltip-arrow {
64
+ bottom: 0;
65
+ right: 5px;
66
+ border-width: 5px 5px 0;
67
+ border-top-color: #000000;
68
+ }
69
+ .tooltip.right .tooltip-arrow {
70
+ top: 50%;
71
+ left: 0;
72
+ margin-top: -5px;
73
+ border-width: 5px 5px 5px 0;
74
+ border-right-color: #000000;
75
+ }
76
+ .tooltip.left .tooltip-arrow {
77
+ top: 50%;
78
+ right: 0;
79
+ margin-top: -5px;
80
+ border-width: 5px 0 5px 5px;
81
+ border-left-color: #000000;
82
+ }
83
+ .tooltip.bottom .tooltip-arrow {
84
+ top: 0;
85
+ left: 50%;
86
+ margin-left: -5px;
87
+ border-width: 0 5px 5px;
88
+ border-bottom-color: #000000;
89
+ }
90
+ .tooltip.bottom-left .tooltip-arrow {
91
+ top: 0;
92
+ left: 5px;
93
+ border-width: 0 5px 5px;
94
+ border-bottom-color: #000000;
95
+ }
96
+ .tooltip.bottom-right .tooltip-arrow {
97
+ top: 0;
98
+ right: 5px;
99
+ border-width: 0 5px 5px;
100
+ border-bottom-color: #000000;
101
+ }
css/wysija-editor.css ADDED
@@ -0,0 +1 @@
 
1
+ .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{display:inline-table}* html .clearfix{height:1%}.clearfix{display:block}.hidden{display:none}strong{font-weight:bold}img{vertical-align:middle}:focus{outline:0 !important}.inline{display:-moz-inline-box;display:inline-block;*display:inline;*float:left}.wysija_tools li,.wysija_tools dd{margin:0 !important}.wysija_controls li{margin:0 0 0 3px !important}#wysija_items{position:fixed;width:275px;right:20px;top:20px;z-index:999}#wysija_items li{line-height:100px;height:100px;outline:1px solid #ccc;text-align:center}.wysija_items li.disabled{-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=20)";filter:alpha(opacity=20);-moz-opacity:0.2;opacity:0.2}.wysija_items li.active a{height:100px;cursor:pointer;display:block}.wysija_header,.wysija_footer{position:relative;display:inline-table;display:block;height:1%;margin:0;padding:0;z-index:98}.wysija_header:after,.wysija_footer:after{content:".";display:block;height:0;clear:both;visibility:hidden}.wysija_block{position:relative;margin:0 10px;padding:10px 18px 10px 18px;display:inline-table;display:block;height:1%;margin:0;z-index:98}.wysija_block.dragging,.wysija_header.dragging,.wysija_footer.dragging{z-index:99000}.wysija_block:after{content:".";display:block;height:0;clear:both;visibility:hidden}.wysija_block.hover{border:1px dashed #bbb;margin:0 0 0 0;padding:9px 17px 9px 17px}.wysija_block.static{padding:0;margin:0}.wysija_block .wysija_controls{background-color:#dfdfdf;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#bbbbbb');background:-webkit-gradient(linear, left bottom, left top, from(#bbb), to(#eee));-moz-border-radius:2px;-webkit-border-radius:2px;-khtml-border-radius:2px;border-radius:2px;border:1px solid #ccc;position:absolute;margin:0;padding:0;width:598px;height:20px;left:-1px;right:0;top:-22px;width:100%}.wysija_block .wysija_controls .handle{width:600px}.wysija_block .wysija_controls li{float:left;width:20px;height:20px}.wysija_block .wysija_controls a{cursor:pointer;float:left;font-size:120%;font-weight:bold;height:20px;line-height:20px;text-align:center;width:20px;color:#000}.wysija_block .handle_container,.wysija_block .handle_container a{float:none;width:40px}.wysija_block .handle_container{left:280px;top:0;position:absolute}.wysija_gallery .wysija_tools,.wysija_image .wysija_tools{position:absolute;height:34px;line-height:34px;margin:0;width:152px;top:-36px;z-index:1010;background-color:#dfdfdf;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#bbbbbb');background:-webkit-gradient(linear, left bottom, left top, from(#bbb), to(#eee));-moz-border-radius:2px;-webkit-border-radius:2px;-khtml-border-radius:2px;border-radius:2px;border:1px solid #ccc;padding:0 0 0 5px}.wysija_gallery .wysija_tools{width:102px}.left .wysija_image .wysija_tools,.center .wysija_image .wysija_tools,.left.wysija_gallery .wysija_tools,.center.wysija_gallery .wysija_tools{left:0;text-align:left}.right .wysija_image .wysija_tools,.right.wysija_gallery .wysija_tools{right:0;text-align:right}.wysija_text .wysija_tools{position:absolute;height:20px;line-height:20px;top:0;right:0;margin:0;z-index:1000;width:20px}.wysija_text.alone .wysija_tools{display:none}.wysija_gallery .wysija_tools li,.wysija_image .wysija_tools li{float:left;width:25px;height:34px}.wysija_text .wysija_tools li{float:left;width:20px;height:20px}.wysija_gallery .wysija_tools a,.wysija_image .wysija_tools a,.wysija_text .wysija_tools a{background-color:#dfdfdf;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#bbbbbb');background:-webkit-gradient(linear, left bottom, left top, from(#bbb), to(#eee));float:left;width:20px;height:20px;margin:5px 3px 0 0;-moz-border-radius:2px;-webkit-border-radius:2px;-khtml-border-radius:2px;border-radius:2px;-webkit-box-shadow:0 1px 0 rgba(0,0,0,0.15),inset 0 0 2px 1px #fff;-moz-box-shadow:0 1px 0 rgba(0,0,0,0.15),inset 0 0 2px 1px #fff;box-shadow:0 1px 0 rgba(0,0,0,0.15),inset 0 0 2px 1px #fff;border:1px solid #ccc;z-index:1010;position:relative}.wysija_text .wysija_tools a{margin:0}.wysija_gallery .wysija_tools a:hover,.wysija_image .wysija_tools a:hover,.wysija_text .wysija_tools a:hover{background-color:#ddd;border-color:#909090}.wysija_gallery .wysija_tools a.active,.wysija_image .wysija_tools a.active,.wysija_text .wysija_tools a.active{background-color:#ddd;background-image:-moz-linear-gradient(center bottom, #eee, #bbb);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#bbbbbb');background:-webkit-gradient(linear, left bottom, left top, from(#eee), to(#bbb));border-color:#909090}.wysija_controls a span,.wysija_gallery .wysija_tools a span,.wysija_image .wysija_tools a span,.wysija_text .wysija_tools a span,.wj_content a span{display:block;height:20px;width:20px}.wysija_controls span input{margin:2px 0 0 0;padding:0;color:transparent}.alignment-left span{background:url(../img/controls/icons.png) no-repeat 0 0}.alignment-left.active span,.alignment-left:hover span{background:url(../img/controls/icons.png) no-repeat 0 -20px}.alignment-center span{background:url(../img/controls/icons.png) no-repeat -20px 0}.alignment-center.active span,.alignment-center:hover span{background:url(../img/controls/icons.png) no-repeat -20px -20px}.alignment-right span{background:url(../img/controls/icons.png) no-repeat -40px 0}.alignment-right.active span,.alignment-right:hover span{background:url(../img/controls/icons.png) no-repeat -40px -20px}.add-link span{background:url(../img/controls/icons.png) no-repeat -60px 0}.add-link.active span,.add-link:hover span{background:url(../img/controls/icons.png) no-repeat -60px -20px}.remove-link span{background:url(../img/controls/icons.png) no-repeat -80px 0}.remove-link.active span,.remove-link:hover span{background:url(../img/controls/icons.png) no-repeat -80px -20px}.remove span{background:url(../img/controls/icons.png) no-repeat -100px 0}.remove.active span,.remove:hover span{background:url(../img/controls/icons.png) no-repeat -100px -20px}.handle span{background:url(../img/controls/grip-handle.png) no-repeat;cursor:move;width:40px !important}.duplicate span{background:url(../img/controls/icons.png) no-repeat -140px 0}.duplicate.active span,.duplicate:hover span{background:url(../img/controls/icons.png) no-repeat -140px -20px}.settings span{background:url(../img/controls/icons.png) no-repeat -160px 0}.settings.active span,.settings:hover span{background:url(../img/controls/icons.png) no-repeat -160px -20px}.icon-plus span{background:url(../img/controls/icons.png) no-repeat -200px 0}.icon-plus.active span,.icon-plus:hover span{background:url(../img/controls/icons.png) no-repeat -200px -20px}.icon-minus span{background:url(../img/controls/icons.png) no-repeat -220px 0}.icon-minus.active span,.icon-minus:hover span{background:url(../img/controls/icons.png) no-repeat -220px -20px}.banner{color:#000;z-index:999;text-align:center;width:600px;position:absolute;top:0;right:0;left:0;bottom:0;white-space:nowrap;font-size:16px !important;-webkit-touch-callout:none;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.banner .banner-message{margin:20px auto 0 auto}.banner .banner-icon{background:url(../img/controls/image-placeholder.png) no-repeat;padding:2px 0 0 28px}.banner .banner-notice{margin:10px auto 0 auto;font-size:12px;font-style:italic}#wysija_wrapper{padding:20px;width:600px;border:1px solid #ccc;position:relative;background-color:#fff;-webkit-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);-moz-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);box-shadow:0 0 5px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1)}#wysija_wrapper:before,#wysija_wrapper:after{position:absolute;width:40%;height:10px;content:' ';left:12px;bottom:12px;background:transparent;-webkit-transform:skew(-5deg) rotate(-5deg);-moz-transform:skew(-5deg) rotate(-5deg);-ms-transform:skew(-5deg) rotate(-5deg);-o-transform:skew(-5deg) rotate(-5deg);transform:skew(-5deg) rotate(-5deg);-webkit-box-shadow:0 6px 12px rgba(0,0,0,0.3);-moz-box-shadow:0 6px 12px rgba(0,0,0,0.3);box-shadow:0 6px 12px rgba(0,0,0,0.3);z-index:-1}#wysija_wrapper:after{left:auto;right:12px;-webkit-transform:skew(5deg) rotate(5deg);-moz-transform:skew(5deg) rotate(5deg);-ms-transform:skew(5deg) rotate(5deg);-o-transform:skew(5deg) rotate(5deg);transform:skew(5deg) rotate(5deg)}body.rtl #wysija_wrapper{right:603px !important}#wysija_container{margin:30px 0 0 0;width:600px;border-collapse:collapse;width:600px;margin:0;padding:0}#wysija_header{padding:0}#wysija_body{vertical-align:top;padding:0}#wysija_footer{padding:0}#wysija_block_placeholder{height:0;padding:0;margin:0}.block_placeholder,.text_placeholder{font-weight:bold;height:0;overflow:hidden;line-height:30px;text-align:center;border:0 none;width:598px;z-index:9500;margin:0 auto;text-indent:-9999px}.wysija_block .block_placeholder,.wysija_block .text_placeholder,.wysija_block .image_placeholder{margin:0}.image_placeholder{margin:1px 0 0 1px}.hover .image_placeholder{margin:0}.wysija_content .empty.editable{font-size:13px;line-height:30px}.wysija_image.empty{height:0;width:0;margin:0 !important;z-index:-1;overflow:hidden}.wysija_image.empty.active{height:auto}.wysija_image.active{background:url(../img/controls/image-placeholder.png) no-repeat center center #f5f5f5;border:1px dashed #dfdfdf}#wysija_header .wysija_image.empty,#wysija_footer .wysija_image.empty{background-image:none;height:auto}#wysija_header .wysija_image.active,#wysija_footer .wysija_image.active{border:0 none}.block_placeholder.active,.text_placeholder.active{text-indent:0;border:1px dashed #dfdfdf;background-color:#f5f5f5;display:block;overflow:auto;height:30px}.image_placeholder.empty.active{display:block}.block_placeholder.hover,.text_placeholder.hover,.image_hover .wysija_image{background-color:#0074a2;border-color:#0074a2;color:#fff !important}.image_placeholder.active{-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=25)";filter:alpha(opacity=25);-moz-opacity:0.25;opacity:0.25;z-index:9501}.wysija_image.empty.active,.wysija_text.empty.active{display:block;margin:0}#wysija_header .wysija_image.active,#wysija_footer .wysija_image.active{left:0;top:0;right:0}.wysija_header .image_placeholder,.wysija_footer .image_placeholder{border:0 none !important;margin:0 !important}.wysija_header .image_placeholder.active,.wysija_footer .image_placeholder.active{border:0 none !important}.wysija_options{display:none}.wysija_content .wysija_image{margin:0 auto 15px auto;cursor:move}.wysija_header .wysija_content.left .wysija_image,.wysija_header .wysija_content.right .wysija_image,.wysija_footer .wysija_content.left .wysija_image,.wysija_footer .wysija_content.right .wysija_image{margin:0 0 0 0 !important}.wysija_content.center .image_placeholder.active{margin:0 0 -2px 0}.wysija_header .wysija_content.center .wysija_image,.wysija_footer .wysija_content.center .wysija_image{margin:0 auto !important}.wysija_header .wysija_content.center .image_placeholder.active,.wysija_footer .wysija_content.center .image_placeholder.active{margin:0}.wysija_block .wysija_content.center .wysija_image.empty.active{margin-bottom:5px !important}.wysija_content.left{float:none}.wysija_gallery.left{float:left}.wysija_content.left .wysija_image{float:left;margin:0 15px 5px 0}.wysija_content.left .wysija_image.active{margin:0 13px 3px 0}.wysija_content.left .wysija_image.empty.active{margin:0 15px 5px 0 !important}.wysija_gallery.right{float:right}.wysija_content.right .wysija_image{float:right;margin:0 0 5px 15px}.wysija_content.right .wysija_image.active{margin:0 0 3px 13px}.wysija_content.right .wysija_image.empty.active{margin:0 0 5px 15px !important}.wysija_content .wysija_image.alone{margin:0 !important}.wysija_content.center .wysija_image.alone{margin:0 auto !important}.wysija_image{position:relative;z-index:999}.wysija_image .url-info{position:absolute;bottom:0;left:0;font-size:0.7em;padding:2px;color:#000;background-color:#fff;-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=80)";filter:alpha(opacity=80);-moz-opacity:0.8;opacity:0.8}.wysija_content.left .wysija_image .url-info{right:0;left:auto}.wysija_text{position:relative;z-index:998}.wysija_text.editing{border:1px solid #dfdfdf}.wysija_content.left .wysija_text.editing{float:left}.wysija_text .editable:hover{cursor:pointer}#wysija_toolbar{left:657px;position:absolute;top:-1000px;width:295px}#wysija_toolbar .wysija_toolbar_tabs{border-bottom:1px solid #dfdfdf;line-height:0}#wysija_toolbar .wysija_toolbar_tabs li,#wysija_toolbar .wysija_toolbar_tabs a{margin:0;height:30px;line-height:30px;padding:0;display:-moz-inline-box;display:inline-block;*display:inline;*float:left;outline:0 none}#wysija_toolbar .wysija_toolbar_tabs a{outline:0 none;text-decoration:none;color:#a6a6a6}#wysija_toolbar .wysija_toolbar_tabs li{margin:0 0 1px 0}#wysija_toolbar .wysija_toolbar_tabs a{font-family:Georgia,"Times New Roman","Bitstream Charter",Times,serif;background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #f5f5f5);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#F5F5F5', endColorstr='#F9F9F9');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#f5f5f5));border:1px solid #DFDFDF;-moz-border-radius:3px 3px 0 0;-webkit-border-radius:3px 3px 0 0;-khtml-border-radius:3px 3px 0 0;border-radius:3px 3px 0 0;box-shadow:0 1px 0 #FFFFFF inset;padding:0 7px}#wysija_toolbar .wysija_toolbar_tabs a:hover{background-color:#eee}#wysija_toolbar .wysija_toolbar_tabs a.selected{color:#000;border-bottom:0 none;background:#fcfcfc;filter:none;padding-bottom:1px}#wysija_toolbar .wysija_toolbar_tabs,#wysija_toolbar .wj_content,#wysija_toolbar .wj_images,#wysija_toolbar .wj_styles,#wysija_toolbar .wj_themes{position:relative;z-index:9998;margin:0}#wysija_toolbar #wysija_notices span{font-size:12px !important;font-weight:normal !important}.wj_content{background-color:#fcfcfc;border:1px solid #dfdfdf;border-top:0 none;padding:0 0 10px 0}.wj_content li{padding:10px 10px 0 10px;position:relative}.wj_content li.notice{font-style:italic;font-size:11px;margin:0 !important;border:0 none !important;background:none !important}.wj_images{z-index:999;background-color:#fcfcfc;border:1px solid #dfdfdf;border-top:0 none;padding:9px}.wj_images ul{height:360px;overflow:auto;margin:0;padding:0}.wj_images li{float:left;width:80px;margin:2px;height:80px;overflow:hidden;position:relative;border:1px solid #dfdfdf}.wj_images li img{max-height:150px;width:80px}.wj_images li a{float:left;display:block;height:80px;width:80px}.wj_button{text-align:center;width:265px;margin:20px 0;height:30px;line-height:30px}.wj_images img{cursor:move;max-width:275px}#wysija-upload-browse,#wysija-themes-browse{float:none}.wj_images #wj_images_preview{border:3px solid #dfdfdf;position:absolute;right:295px;top:0;background-color:#fff}.wj_styles{z-index:999;background-color:#fcfcfc;border:1px solid #dfdfdf;border-top:0 none;padding:9px}.wj_styles p{margin:10px 0}.wj_styles .preview{width:20px;float:none !important;height:20px;vertical-align:middle;display:-moz-inline-box;display:inline-block;*display:inline;*float:left;border:1px solid #333}.wj_styles input,.wj_styles select{height:20px;border:0 none;border:1px solid #CCC;margin:0 3px 0 0 !important;padding:0 !important;vertical-align:middle;line-height:20px;font-size:12px}.wj_styles #aUnderlineInput{margin:0 0 0 10px !important}.wj_styles select{width:75px}.wj_styles input.checkbox,.wj_styles input[type=checkbox]{border:0 none !important}.wj_styles input.color{width:55px;padding:0 0 0 2px !important}.wj_styles input.color.disabled{background-color:#eee !important;color:#eee !important}.wj_styles select.small-size{width:50px}.wj_styles select.type{width:68px}.wj_styles select.size{width:58px}.wj_styles form label{cursor:normal;margin:0 0 0 3px}.wj_themes{z-index:999;background-color:#fcfcfc;border:1px solid #dfdfdf;border-top:0 none;padding:9px}.wj_themes ul{height:360px;overflow:auto;margin:0;padding:0}.wj_themes li{float:left;width:80px;margin:2px;height:80px;overflow:hidden;border:1px solid #dfdfdf;position:relative}.wj_themes li img{max-height:150px;width:80px}.wj_themes li a{display:block}.wj_themes #wj_themes_preview{border:3px solid #dfdfdf;position:absolute;right:295px;top:0;background-color:#fff}.wysija_widget{width:275px;height:25px;line-height:25px;z-index:9999 !important}.wj_content a.wysija_item,.wysija_widget{height:25px;line-height:25px;background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #ececec);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#f9f9f9', endColorstr='#ececec');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#ececec));border:1px solid #DFDFDF;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;box-shadow:0 1px 0 #FFFFFF inset;display:block;font-size:12px;font-weight:bold;padding:0 0 0 7px;cursor:move;color:#222;text-shadow:0 1px 0 #FFFFFF}.wj_content a.wysija_item.disabled{cursor:help;color:#cccccc}.wysija_item_settings{position:absolute;right:15px;top:13px}.wysija_widget.image{z-index:9999 !important;height:auto !important;width:150px !important;background:none !important;border:0 none !important;box-shadow:none !important;border-radius:0 !important;padding:0 !important}#wysija_notices{z-index:20000;position:relative;left:0;top:0;padding:5px;border:1px solid #dfdfdf;border-top:0 none}#wysija_notices.notice{background-color:#FFFFE0;border-color:#E6DB55;z-index:9999}#wysija_notices.error{background-color:#FFEBE8;border-color:#CC0000}#wysija_notice_msg{font-size:14px;font-weight:bold}.align-center{text-align:center}.align-left{text-align:left}.align-right{text-align:right}.align-justify{text-align:justify}.wysija_gallery{position:relative}.wysija_gallery.center{margin:0 auto}.wysija_gallery.right{text-align:right}.wysija_gallery.left{text-align:left}.wysija_row{width:100%}.wysija_cell{float:left;text-align:center}.resize-controls{position:absolute;left:0;right:0;top:0;bottom:0;z-index:99000;border:1px dashed #ccc}.resize-handle{position:absolute;bottom:0;right:0;width:20px;height:20px;border:0 none;z-index:1001;cursor:se-resize;background:transparent url(../img/controls/resize-se.png) no-repeat scroll right bottom}.resize-info{position:absolute;bottom:0;left:4px;font-size:0.8em;padding:0;margin:0;text-shadow:0 1px 0 #F7F5F2;white-space:nowrap}.right .resize-handle{position:absolute;bottom:0;left:0;width:20px;height:20px;border:0 none;z-index:1001;cursor:sw-resize;background:transparent url(../img/controls/resize-sw.png) no-repeat scroll left bottom}.right .resize-info{left:auto !important;right:4px}#adminmenu li .wp-submenu,.folded #adminmenu .wp-has-current-submenu .wp-submenu{z-index:1001 !important}.wysija_divider{margin:0 auto;width:564px}.wysija_block.auto-post.hover .wysija_content *,.wysija_block.auto-post.hover .wysija_divider{z-index:1000;-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=30)";filter:alpha(opacity=30);-moz-opacity:0.3;opacity:0.3}.wysija_auto-post .wysija_divider{text-align:center;margin:0 !important;width:100% !important}.wysija_auto-post .nopost{height:100px}.wysija_auto-post .nopost p{margin:25px 0 0 0}.wysija_settings{left:10px;position:absolute;top:0;z-index:1000}.wysija_settings a{background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #ececec);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#f9f9f9', endColorstr='#ececec');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#ececec));border:1px solid #DFDFDF;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;box-shadow:0 1px 0 #FFFFFF inset;font-size:12px;font-weight:normal;cursor:pointer;color:#222;text-shadow:0 1px 0 #FFFFFF;text-decoration:none;display:block;padding:5px 5px 3px 27px}.wysija_settings a span{width:20px;height:20px;position:absolute;top:3px;left:5px}#wysija_toggle_images{position:absolute;top:5px;right:5px;z-index:1000;width:73px;height:33px;-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=50)";filter:alpha(opacity=50);-moz-opacity:0.5;opacity:0.5;background-color:#fff}#wysija_toggle_images.on{background:url(../img/controls/images-toggle-on.png) no-repeat}#wysija_toggle_images.off{background:url(../img/controls/images-toggle-off.png) no-repeat}#wysija_toggle_images:hover{-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=100)";filter:alpha(opacity=100);-moz-opacity:1.0;opacity:1.0}.wysija_block.auto-post{padding:0 !important}.wysija_auto-post .wysija_content{padding:10px 17px !important}.wysija_auto-post .wysija_content:after{content:".";display:block;height:0;clear:both;visibility:hidden}.wysija_auto-post .wysija_content{display:inline-table}* html .wysija_auto-post .wysija_content{height:1%}.wysija_auto-post .wysija_content{display:block}.wysija_auto-post .wysija_divider{padding:9px 0 !important}.wysija_block.auto-post.hover .wysija_content{padding:10px 17px !important}.wysija_auto-post .wysija_text{pointer-events:none;cursor:default}.wysija_auto-post .wysija_image{pointer-events:none;cursor:default}.auto-post.hover .wysija_content:last-child{padding:9px 17px 9px 17px !important}.wysija_auto-post .loading{padding:10px 17px;font-size:10px}#wysija_popup_overlay{background-color:#000;height:100%;left:0;position:fixed;top:0;width:100%;z-index:999989;-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=75)";filter:alpha(opacity=75);-moz-opacity:0.75;opacity:0.75}#wysija_popup_overlay.loading{background:url(../img/popup/loading.gif) no-repeat 50% 50% #000}#wysija_popup{position:absolute;top:0;left:0;width:600px;height:400px;background-color:#fff;z-index:999999;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;-moz-box-shadow:#000 0 4px 30px;-webkit-box-shadow:#000 0 4px 30px;-khtml-box-shadow:#000 0 4px 30px;box-shadow:#000 0 4px 30px;border:1px solid #555}#wysija_popup_content{padding:0;overflow:hidden;position:relative}#wysija_popup_title{background-color:#222;border:1px solid #333;height:27px;margin:0;padding:0}#wysija_popup_title h3{color:#cfcfcf;font-size:12px;font-weight:normal;margin:6px 0 0 10px;padding:0;float:left;font:12px "Lucida Grande", Verdana, Arial, sans-serif}#wysija_popup_iframe{border:0 none;outline:0 none}#wysija_popup_close{background:url(../img/popup/close.png) no-repeat;height:15px;width:15px;background-color:#333;display:block;float:right;margin:6px 10px 0 0}.mpoet-ui .wysija_item_settings{text-decoration:none}.mpoet-ui .settings span{background:none;color:#999}.mpoet-ui .wysija_item_settings:hover .dashicons-admin-generic:before,.mpoet-ui .settings:hover .dashicons-admin-generic:before{color:#2ea2cc}.wp_themeSkin .mceToolbar{margin:0 !important;padding:0 !important}
css/wysija-form-editor.css ADDED
@@ -0,0 +1 @@
 
1
+ .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{display:inline-table}* html .clearfix{height:1%}.clearfix{display:block}.hidden{display:none}strong{font-weight:bold}img{vertical-align:middle}:focus{outline:0 !important}.inline{display:-moz-inline-box;display:inline-block;*display:inline;*float:left}#form-error{padding:10px;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;color:#000;font-weight:normal;background-color:#ffebe8;border:1px solid #c00}#form-notice{padding:10px;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;color:#555555;font-weight:normal;background-color:#FFFBCC;border:1px solid #E6DB55}#wysija_form_editor{padding:20px;width:600px;border:1px solid #ccc;position:relative;background-color:#fff;-webkit-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);-moz-box-shadow:0 0 4px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1);box-shadow:0 0 5px rgba(0,0,0,0.2),inset 0 0 20px rgba(0,0,0,0.1)}#wysija_form_editor:before,#wysija_form_editor:after{position:absolute;width:40%;height:10px;content:' ';left:12px;bottom:12px;background:transparent;-webkit-transform:skew(-5deg) rotate(-5deg);-moz-transform:skew(-5deg) rotate(-5deg);-ms-transform:skew(-5deg) rotate(-5deg);-o-transform:skew(-5deg) rotate(-5deg);transform:skew(-5deg) rotate(-5deg);-webkit-box-shadow:0 6px 12px rgba(0,0,0,0.3);-moz-box-shadow:0 6px 12px rgba(0,0,0,0.3);box-shadow:0 6px 12px rgba(0,0,0,0.3);z-index:-1}#wysija_form_editor:after{left:auto;right:12px;-webkit-transform:skew(5deg) rotate(5deg);-moz-transform:skew(5deg) rotate(5deg);-ms-transform:skew(5deg) rotate(5deg);-o-transform:skew(5deg) rotate(5deg);transform:skew(5deg) rotate(5deg)}.wysija_form_block .wysija_warning{font-weight:bold;color:#900}.block_placeholder{margin:0;border:0 none}.block_placeholder.active{border:1px dashed #b4b4b4;margin:10px;display:block}.block_placeholder.hover{margin:10px;border:1px dashed #b4b4b4;background-color:#f5f5f5}.wysija_form_block{background-color:#fff;height:20px;border:0 none}.wysija_form_block.highlighted{border:1px solid #5897FB;padding:9px 17px}.wysija_form_block img{max-width:100%}.wysija_form_block p{margin:5px 0;word-wrap:break-word}.wysija_form_block p input.radio,.wysija_form_block p input.checkbox{margin:0 5px 0 0}#wysija_form_container{width:640px;margin:0}#wysija-add-field{float:none}#wysija_form_toolbar{position:absolute;width:295px}#wysija_form_toolbar .wysija_form_toolbar_tabs{border-bottom:1px solid #dfdfdf;line-height:0}#wysija_form_toolbar .add_custom_field{text-align:center;padding:15px 0 5px 0}#wysija_form_toolbar .wysija_form_toolbar_tabs li,#wysija_form_toolbar .wysija_form_toolbar_tabs a{margin:0;height:30px;line-height:30px;padding:0;display:-moz-inline-box;display:inline-block;*display:inline;*float:left;outline:0 none}#wysija_form_toolbar .wysija_form_toolbar_tabs a{outline:0 none;text-decoration:none;color:#a6a6a6}#wysija_form_toolbar .wysija_form_toolbar_tabs li{margin:0 0 1px 0}#wysija_form_toolbar .wysija_form_toolbar_tabs a{font-family:Georgia,"Times New Roman","Bitstream Charter",Times,serif;background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #f5f5f5);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#F5F5F5', endColorstr='#F9F9F9');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#f5f5f5));border:1px solid #DFDFDF;-moz-border-radius:3px 3px 0 0;-webkit-border-radius:3px 3px 0 0;-khtml-border-radius:3px 3px 0 0;border-radius:3px 3px 0 0;box-shadow:0 1px 0 #FFFFFF inset;padding:0 7px}#wysija_form_toolbar .wysija_form_toolbar_tabs a:hover{background-color:#eee}#wysija_form_toolbar .wysija_form_toolbar_tabs a.selected{color:#000;border-bottom:0 none;background:#fcfcfc;filter:none;padding-bottom:1px}#wysija_form_toolbar .wysija_form_toolbar_tabs,#wysija_form_toolbar .wj_content,#wysija_form_toolbar .wj_images,#wysija_form_toolbar .wj_styles,#wysija_form_toolbar .wj_themes{position:relative;z-index:9998;margin:0}.wysija_form_widget{width:275px;height:25px;line-height:25px;z-index:9999 !important}#wysija_form_toolbar a.wysija_form_item,.wysija_form_widget{height:25px;line-height:25px;background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #ececec);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#f9f9f9', endColorstr='#ececec');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#ececec));border:1px solid #DFDFDF;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;box-shadow:0 1px 0 #FFFFFF inset;display:block;font-size:12px;font-weight:bold;padding:0 0 0 7px;cursor:move;color:#222;text-shadow:0 1px 0 #FFFFFF}#wysija_form_toolbar a.wysija_form_item.disabled{cursor:pointer;color:#cccccc;pointer-events:none}.wysija_form_item_settings{position:absolute;right:35px;top:13px}.wysija_form_item_delete{position:absolute;right:15px;top:13px}.wj_content{background-color:#fcfcfc;border:1px solid #dfdfdf;padding:0 0 10px 0}.wj_content li{padding:10px 10px 0 10px;position:relative}.wj_content li.notice{font-style:italic;font-size:11px;margin:0 !important;border:0 none !important;background:none !important}.wysija_form_block{position:relative;margin:0;padding:10px 18px 10px 18px;display:inline-table;display:block;height:1%;margin:0;z-index:98}.wysija_form_block.dragging{z-index:99000;pointer-events:none}.wysija_form_block:after{content:".";display:block;height:0;clear:both;visibility:hidden}.wysija_form_block.hover{border:1px dashed #bbb;margin:0 0 0 0;padding:9px 17px 9px 17px}.wysija_form_block.static{background-color:#999}.wysija_form_block .wysija_controls{background-color:#dfdfdf;background-image:-moz-linear-gradient(center bottom, #bbb, #eee);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#bbbbbb');background:-webkit-gradient(linear, left bottom, left top, from(#bbb), to(#eee));-moz-border-radius:2px;-webkit-border-radius:2px;-khtml-border-radius:2px;border-radius:2px;border:1px solid #ccc;position:absolute;margin:0;padding:0;width:598px;height:20px;left:-1px;right:0;top:-22px}.wysija_form_block .wysija_controls li{float:left;width:20px;height:20px}.wysija_form_block .wysija_controls a{cursor:pointer;float:left;font-size:120%;font-weight:bold;height:20px;line-height:20px;text-align:center;width:20px;color:#000}.wysija_form_block .wysija_controls a.remove{margin:0 0 0 1px}.wysija_form_block .handle_container,.wysija_form_block .handle_container a{float:none;width:40px !important}.wysija_form_block .handle_container{left:280px;top:0;position:absolute}.wysija_controls a span,.wysija_gallery .wysija_tools a span,.wysija_image .wysija_tools a span,.wysija_text .wysija_tools a span,.wj_content a span{display:block;height:20px;width:20px}.wysija_controls span input{margin:2px 0 0 0;padding:0;color:transparent}.alignment-left span{background:url(../img/controls/icons.png) no-repeat 0 0}.alignment-left.active span,.alignment-left:hover span{background:url(../img/controls/icons.png) no-repeat 0 -20px}.alignment-center span{background:url(../img/controls/icons.png) no-repeat -20px 0}.alignment-center.active span,.alignment-center:hover span{background:url(../img/controls/icons.png) no-repeat -20px -20px}.alignment-right span{background:url(../img/controls/icons.png) no-repeat -40px 0}.alignment-right.active span,.alignment-right:hover span{background:url(../img/controls/icons.png) no-repeat -40px -20px}.add-link span{background:url(../img/controls/icons.png) no-repeat -60px 0}.add-link.active span,.add-link:hover span{background:url(../img/controls/icons.png) no-repeat -60px -20px}.remove-link span{background:url(../img/controls/icons.png) no-repeat -80px 0}.remove-link.active span,.remove-link:hover span{background:url(../img/controls/icons.png) no-repeat -80px -20px}.remove span,.delete span{background:url(../img/controls/icons.png) no-repeat -100px 0}.remove.active span,.remove:hover span,.delete.active span,.delete:hover span{background:url(../img/controls/icons.png) no-repeat -100px -20px}.handle span{background:url(../img/controls/grip-handle.png) no-repeat;cursor:move;width:40px !important}.duplicate span{background:url(../img/controls/icons.png) no-repeat -140px 0}.duplicate.active span,.duplicate:hover span{background:url(../img/controls/icons.png) no-repeat -140px -20px}.settings span{background:url(../img/controls/icons.png) no-repeat -160px 0}.settings.active span,.settings:hover span{background:url(../img/controls/icons.png) no-repeat -160px -20px}.icon-plus span{background:url(../img/controls/icons.png) no-repeat -200px 0}.icon-plus.active span,.icon-plus:hover span{background:url(../img/controls/icons.png) no-repeat -200px -20px}.icon-minus span{background:url(../img/controls/icons.png) no-repeat -220px 0}.icon-minus.active span,.icon-minus:hover span{background:url(../img/controls/icons.png) no-repeat -220px -20px}.wysija_options{display:none}.wysija_settings{left:10px;position:absolute;top:0;z-index:1000}.wysija_settings a{background-color:#F5F5F5;background-image:-moz-linear-gradient(center top, #f9f9f9, #ececec);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#f9f9f9', endColorstr='#ececec');background:-webkit-gradient(linear, left top, left bottom, from(#f9f9f9), to(#ececec));border:1px solid #DFDFDF;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;box-shadow:0 1px 0 #FFFFFF inset;font-size:12px;font-weight:normal;cursor:pointer;color:#222;text-shadow:0 1px 0 #FFFFFF;text-decoration:none;display:block;padding:5px 5px 3px 27px}.wysija_settings a span{width:20px;height:20px;position:absolute;top:3px;left:5px}.wysija_form_block label{margin:0 5px 0 0;display:block}#wysija_widget_templates{display:none}#wysija_popup_overlay{background-color:#000;height:100%;left:0;position:fixed;top:0;width:100%;z-index:999998;-ms-filter:"progid:DXImageTransform.Microsoft.Alpha(Opacity=75)";filter:alpha(opacity=75);-moz-opacity:0.75;opacity:0.75}#wysija_popup_overlay.loading{background:url(../img/popup/loading.gif) no-repeat 50% 50% #000}#wysija_popup{position:absolute;top:0;left:0;width:600px;height:400px;background-color:#fff;z-index:999999;-moz-border-radius:3px;-webkit-border-radius:3px;-khtml-border-radius:3px;border-radius:3px;-moz-box-shadow:#000 0 4px 30px;-webkit-box-shadow:#000 0 4px 30px;-khtml-box-shadow:#000 0 4px 30px;box-shadow:#000 0 4px 30px;border:1px solid #555}#wysija_popup_content{padding:0;overflow:hidden;clear:both}#wysija_popup_title{background-color:#222;border:1px solid #333;height:27px;margin:0;padding:0}#wysija_popup_title h3{color:#cfcfcf;font-size:12px;font-weight:normal;margin:6px 0 0 10px;padding:0;float:left;font:12px "Lucida Grande", Verdana, Arial, sans-serif}#wysija_popup_title.empty{background-color:transparent;height:0;border:0 none}#wysija_popup_iframe{border:0 none;outline:0 none}#wysija_popup_close{background:url(../img/popup/close.png) no-repeat;height:15px;width:15px;background-color:#333;display:block;float:right;margin:6px 10px 0 0}h2.title{height:40px;line-height:40px;margin:0 0 15px 0}h2.title span,h2.title form{float:left;margin:0 5px 0 0;vertical-align:middle}h2.title span#form-name{padding:0 3px;margin:0 10px 0 0;font-style:italic}h2.title span#form-name-default{display:none}h2.title #edit-form-name{float:left;margin:10px 5px 0 0}h2.title input{margin:0 10px 0 0}#list-selection{margin:30px 0 0 0}#list-selection label{display:block}#after-submit{margin:30px 0 0 0}#after-submit textarea{width:300px;height:50px}#form-export-links{margin:20px 0 0 0}#form-export textarea{width:100%;height:100px}#wysija_notices{z-index:20000;position:relative;left:0;top:0;padding:5px;border:1px solid #dfdfdf;border-top:0 none}#wysija_notices.notice{background-color:#FFFFE0;border-color:#E6DB55;z-index:9999}#wysija_notices.error{background-color:#FFEBE8;border-color:#CC0000}#wysija_notice_msg{font-size:14px;font-weight:bold}.expand-code{cursor:pointer}.wysija_form_block textarea{resize:none}.mpoet-ui .wysija_form_item_settings,.mpoet-ui .wysija_form_item_delete{text-decoration:none}.mpoet-ui .wysija_form_item_settings:hover .dashicons-admin-generic:before,.mpoet-ui .wysija_form_item_delete:hover .dashicons-dismiss:before,.mpoet-ui .settings:hover .dashicons-admin-generic:before{color:#2ea2cc}.mpoet-ui .settings span,.mpoet-ui .delete span{background:none;color:#999}.mpoet-ui .delete span:before{font-size:21px}
data/bookmarks/index.html ADDED
File without changes
data/bookmarks/medium/01/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.addthis.com/
data/bookmarks/medium/01/facebook.png ADDED
Binary file
data/bookmarks/medium/01/google.png ADDED
Binary file
data/bookmarks/medium/01/linkedin.png ADDED
Binary file
data/bookmarks/medium/01/twitter.png ADDED
Binary file
data/bookmarks/medium/02/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.jankoatwarpspeed.com/post/2009/02/23/Handycons-2-another-free-hand-drawn-icon-set.aspx
data/bookmarks/medium/02/facebook.png ADDED
Binary file
data/bookmarks/medium/02/google.png ADDED
Binary file
data/bookmarks/medium/02/linkedin.png ADDED
Binary file
data/bookmarks/medium/02/twitter.png ADDED
Binary file
data/bookmarks/medium/03/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/03/facebook.png ADDED
Binary file
data/bookmarks/medium/03/google.png ADDED
Binary file
data/bookmarks/medium/03/linkedin.png ADDED
Binary file
data/bookmarks/medium/03/twitter.png ADDED
Binary file
data/bookmarks/medium/04/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/04/facebook.png ADDED
Binary file
data/bookmarks/medium/04/google.png ADDED
Binary file
data/bookmarks/medium/04/linkedin.png ADDED
Binary file
data/bookmarks/medium/04/twitter.png ADDED
Binary file
data/bookmarks/medium/06/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/06/facebook.png ADDED
Binary file
data/bookmarks/medium/06/google.png ADDED
Binary file
data/bookmarks/medium/06/linkedin.png ADDED
Binary file
data/bookmarks/medium/06/twitter.png ADDED
Binary file
data/bookmarks/medium/07/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/07/facebook.png ADDED
Binary file
data/bookmarks/medium/07/google.png ADDED
Binary file
data/bookmarks/medium/07/linkedin.png ADDED
Binary file
data/bookmarks/medium/07/twitter.png ADDED
Binary file
data/bookmarks/medium/08/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/08/facebook.png ADDED
Binary file
data/bookmarks/medium/08/google.png ADDED
Binary file
data/bookmarks/medium/08/linkedin.png ADDED
Binary file
data/bookmarks/medium/08/twitter.png ADDED
Binary file
data/bookmarks/medium/09/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/09/facebook.png ADDED
Binary file
data/bookmarks/medium/09/google.png ADDED
Binary file
data/bookmarks/medium/09/linkedin.png ADDED
Binary file
data/bookmarks/medium/09/twitter.png ADDED
Binary file
data/bookmarks/medium/10/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ We can't find who's the author of these!
data/bookmarks/medium/10/facebook.png ADDED
Binary file
data/bookmarks/medium/10/google.png ADDED
Binary file
data/bookmarks/medium/10/linkedin.png ADDED
Binary file
data/bookmarks/medium/10/twitter.png ADDED
Binary file
data/bookmarks/medium/11/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/11/facebook.png ADDED
Binary file
data/bookmarks/medium/11/google.png ADDED
Binary file
data/bookmarks/medium/11/linkedin.png ADDED
Binary file
data/bookmarks/medium/11/twitter.png ADDED
Binary file
data/bookmarks/medium/12/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/12/facebook.png ADDED
Binary file
data/bookmarks/medium/12/google.png ADDED
Binary file
data/bookmarks/medium/12/linkedin.png ADDED
Binary file
data/bookmarks/medium/12/twitter.png ADDED
Binary file
data/bookmarks/medium/13/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/medium/13/facebook.png ADDED
Binary file
data/bookmarks/medium/13/google.png ADDED
Binary file
data/bookmarks/medium/13/linkedin.png ADDED
Binary file
data/bookmarks/medium/13/twitter.png ADDED
Binary file
data/bookmarks/medium/14/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.shareaholic.com/
data/bookmarks/medium/14/facebook.png ADDED
Binary file
data/bookmarks/medium/14/google.png ADDED
Binary file
data/bookmarks/medium/14/linkedin.png ADDED
Binary file
data/bookmarks/medium/14/twitter.png ADDED
Binary file
data/bookmarks/medium/15/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.blogperfume.com/new-27-circular-social-media-icons-in-3-sizes/
data/bookmarks/medium/15/facebook.png ADDED
Binary file
data/bookmarks/medium/15/google.png ADDED
Binary file
data/bookmarks/medium/15/linkedin.png ADDED
Binary file
data/bookmarks/medium/15/twitter.png ADDED
Binary file
data/bookmarks/medium/16/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://icondock.com/free/vector-social-media-icons
data/bookmarks/medium/16/facebook.png ADDED
Binary file
data/bookmarks/medium/16/google.png ADDED
Binary file
data/bookmarks/medium/16/linkedin.png ADDED
Binary file
data/bookmarks/medium/16/twitter.png ADDED
Binary file
data/bookmarks/medium/17/facebook.png ADDED
Binary file
data/bookmarks/medium/17/google.png ADDED
Binary file
data/bookmarks/medium/17/linkedin.png ADDED
Binary file
data/bookmarks/medium/17/twitter.png ADDED
Binary file
data/bookmarks/medium/18/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://shapshapy.deviantart.com/art/social-media-icons-267541616
data/bookmarks/medium/18/facebook.png ADDED
Binary file
data/bookmarks/medium/18/google.png ADDED
Binary file
data/bookmarks/medium/18/linkedin.png ADDED
Binary file
data/bookmarks/medium/18/twitter.png ADDED
Binary file
data/bookmarks/medium/19/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.softicons.com/free-icons/social-media-icons/social-bookmarks-icons-by-deleket
data/bookmarks/medium/19/facebook.png ADDED
Binary file
data/bookmarks/medium/19/google.png ADDED
Binary file
data/bookmarks/medium/19/linkedin.png ADDED
Binary file
data/bookmarks/medium/19/twitter.png ADDED
Binary file
data/bookmarks/medium/20/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.purtypixels.com/super-social-media-cons/
data/bookmarks/medium/20/facebook.png ADDED
Binary file
data/bookmarks/medium/20/google.png ADDED
Binary file
data/bookmarks/medium/20/linkedin.png ADDED
Binary file
data/bookmarks/medium/20/twitter.png ADDED
Binary file
data/bookmarks/medium/21/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.designkindle.com/2011/10/27/socialis/
data/bookmarks/medium/21/facebook.png ADDED
Binary file
data/bookmarks/medium/21/google.png ADDED
Binary file
data/bookmarks/medium/21/linkedin.png ADDED
Binary file
data/bookmarks/medium/21/twitter.png ADDED
Binary file
data/bookmarks/medium/22/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.designkindle.com/2011/10/27/socialis/
data/bookmarks/medium/22/facebook.png ADDED
Binary file
data/bookmarks/medium/22/google.png ADDED
Binary file
data/bookmarks/medium/22/linkedin.png ADDED
Binary file
data/bookmarks/medium/22/twitter.png ADDED
Binary file
data/bookmarks/medium/23/facebook.png ADDED
Binary file
data/bookmarks/medium/23/google.png ADDED
Binary file
data/bookmarks/medium/23/linkedin.png ADDED
Binary file
data/bookmarks/medium/23/twitter.png ADDED
Binary file
data/bookmarks/medium/24/facebook.png ADDED
Binary file
data/bookmarks/medium/24/google.png ADDED
Binary file
data/bookmarks/medium/24/linkedin.png ADDED
Binary file
data/bookmarks/medium/24/twitter.png ADDED
Binary file
data/bookmarks/medium/index.html ADDED
File without changes
data/bookmarks/small/01/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.addthis.com/
data/bookmarks/small/01/facebook.gif ADDED
Binary file
data/bookmarks/small/01/google.gif ADDED
Binary file
data/bookmarks/small/01/linkedin.gif ADDED
Binary file
data/bookmarks/small/01/twitter.gif ADDED
Binary file
data/bookmarks/small/02/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.jankoatwarpspeed.com/post/2009/02/23/Handycons-2-another-free-hand-drawn-icon-set.aspx
data/bookmarks/small/02/facebook.png ADDED
Binary file
data/bookmarks/small/02/google.png ADDED
Binary file
data/bookmarks/small/02/linkedin.png ADDED
Binary file
data/bookmarks/small/02/twitter.png ADDED
Binary file
data/bookmarks/small/03/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/small/03/facebook.png ADDED
Binary file
data/bookmarks/small/03/google.png ADDED
Binary file
data/bookmarks/small/03/linkedin.png ADDED
Binary file
data/bookmarks/small/03/twitter.png ADDED
Binary file
data/bookmarks/small/04/credit.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ Credit for icons:
2
+
3
+ http://blog.artcore-illustrations.de/
4
+
5
+ Note: these icons are from a MailPoet theme's Photoshop file.
data/bookmarks/small/04/facebook.png ADDED
Binary file
data/bookmarks/small/04/google.png ADDED
Binary file
data/bookmarks/small/04/linkedin.png ADDED
Binary file
data/bookmarks/small/04/twitter.png ADDED
Binary file
data/bookmarks/small/05/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://icondock.com/free/vector-social-media-icons
data/bookmarks/small/05/facebook.png ADDED
Binary file
data/bookmarks/small/05/google.png ADDED
Binary file
data/bookmarks/small/05/linkedin.png ADDED
Binary file
data/bookmarks/small/05/twitter.png ADDED
Binary file
data/bookmarks/small/06/credit.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
1
+ Credit for icons:
2
+
3
+ http://www.softicons.com/free-icons/social-media-icons/social-bookmarks-icons-by-deleket
data/bookmarks/small/06/facebook.png ADDED
Binary file
data/bookmarks/small/06/google.png ADDED
Binary file
data/bookmarks/small/06/linkedin.png ADDED
Binary file
data/bookmarks/small/06/twitter.png ADDED
Binary file
data/bookmarks/small/index.html ADDED
File without changes
data/dividers/artcore-white.png ADDED
Binary file
data/dividers/artcore.png ADDED
Binary file
data/dividers/bar.png ADDED
Binary file
data/dividers/business.png ADDED
Binary file
data/dividers/business2.png ADDED
Binary file
data/dividers/cafe.png ADDED
Binary file
data/dividers/drop1.png ADDED
Binary file
data/dividers/drop2.png ADDED
Binary file
data/dividers/drop3.png ADDED
Binary file
data/dividers/flashback.png ADDED
Binary file
data/dividers/index.html ADDED
File without changes
data/dividers/juicy.png ADDED
Binary file
data/dividers/plus.png ADDED
Binary file
data/dividers/school.png ADDED
Binary file
data/dividers/sketchy.png ADDED
Binary file
data/dividers/solid.jpg ADDED
Binary file
data/dividers/vintage.png ADDED
Binary file
data/index.html ADDED
File without changes
data/themes/default/index.html ADDED
File without changes
data/themes/default/screenshot.jpg ADDED
Binary file
data/themes/default/style.css ADDED
@@ -0,0 +1,48 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ html {
2
+ background-color: #fff;
3
+ }
4
+ .header {
5
+ background-color: #fff;
6
+ }
7
+ body {
8
+ color:#000;
9
+ font-family: Arial;
10
+ font-size: 12px;
11
+ background-color: #fff;
12
+ }
13
+ .footer {
14
+ color: #000;
15
+ font-family: Arial;
16
+ font-size: 12px;
17
+ background-color: #ccc;
18
+ }
19
+ h1 {
20
+ color: #000;
21
+ font-family: Arial;
22
+ font-size: 28px;
23
+ }
24
+ h2 {
25
+ color: #000;
26
+ font-family: Arial;
27
+ font-size: 26px;
28
+ }
29
+ h3 {
30
+ color: #000;
31
+ font-family: Arial;
32
+ font-size: 24px;
33
+ }
34
+ a {
35
+ color: #21759B;
36
+ font-family: Arial;
37
+ font-size: 12px;
38
+ text-decoration:'none';
39
+ }
40
+ .divider {
41
+ border-color: #000;
42
+ border-size: 1px;
43
+ border-style: solid;
44
+ }
45
+
46
+ unsubscribe {
47
+ color:#7f7f7f;
48
+ }
data/themes/default/thumbnail.jpg ADDED
Binary file
data/themes/index.html ADDED
File without changes
docs/german-disclaimer.md ADDED
@@ -0,0 +1,4 @@
 
 
 
 
1
+ ### WICHTIG - HAFTUNGSAUSSCHLUSS
2
+
3
+ Dieses Plugin trägt wesentlich dazu bei, Ihren Newsletter rechtssicher betreiben zu können.
4
+ Es ist durch einen Rechtsanwalt geprüft worden, der ebenfalls die rechtlichen Hinweise & Muster erstellt hat. Sie entsprechen den rechtlichen Anforderungen eines typischen Newsletters. Jedoch kann die rechtliche Sicherheit eines Newsletters nur im Einzelfall geprüft und bestätigt werden, so dass insoweit keine Haftung übernommen werden kann. Daher sind auch die rechtliche Text- & Hinweismuster nur als Vorlagen zu verstehen, deren abschließende rechtliche Prüfung und ggf. Anpassung Ihnen obliegt. Falls Sie eine Prüfung Ihres Newsletters wünschen, können Sie sich an Rechtsanwalt Thomas Schwenke, LL.M. [http://rechtsanwalt-schwenke.de/kontakt](http://rechtsanwalt-schwenke.de/kontakt) wenden.
font/icon-email.eot ADDED
Binary file
font/icon-email.svg ADDED
@@ -0,0 +1,11 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0" standalone="no"?>
2
+ <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
3
+ <svg xmlns="http://www.w3.org/2000/svg">
4
+ <metadata></metadata>
5
+ <defs>
6
+ <font id="icon-email" horiz-adv-x="1024">
7
+ <font-face units-per-em="1024" ascent="960" descent="-64" />
8
+ <missing-glyph horiz-adv-x="1024" />
9
+ <glyph unicode="&#x20;" d="" horiz-adv-x="512" />
10
+ <glyph unicode="&#xf466;" d="M972.8 217.6v460.8q0 31.744-22.528 54.272t-54.272 22.528h-717.312q-31.744 0-54.272-22.528t-22.528-54.272v-460.8q0-31.744 22.528-54.272t54.272-22.528h717.312q31.744 0 54.272 22.528t22.528 54.272zM905.728 684.032q7.68-7.68 8.96-16.64t-2.048-15.104-8.448-11.264l-207.872-190.464 199.68-207.872q13.312-15.36 3.072-26.112-4.608-5.632-14.336-6.144t-14.336 3.584l-223.744 190.976-109.568-99.84-109.056 99.84-223.744-190.976q-4.608-4.096-14.336-3.584t-14.336 6.144q-10.24 10.752 3.072 26.112l199.68 207.872-207.872 190.464q-5.12 5.12-8.448 11.264t-2.048 15.104 8.96 16.64q20.48 20.48 48.64-3.584l319.488-258.048 320 258.048q28.16 24.064 48.64 3.584z" />
11
+ </font></defs></svg>
font/icon-email.ttf ADDED
Binary file
font/icon-email.woff ADDED
Binary file
helpers/articles.php ADDED
@@ -0,0 +1,432 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_articles extends WYSIJA_object {
4
+
5
+ function WYSIJA_help_articles() {
6
+
7
+ }
8
+
9
+ function stripShortcodes($content) {
10
+ if(strlen(trim($content)) === 0) {
11
+ return '';
12
+ }
13
+ // remove captions
14
+ $content = preg_replace("/\[caption.*?\](.*<\/a>)(.*?)\[\/caption\]/", '$1', $content);
15
+
16
+ // remove other shortcodes
17
+ $content = preg_replace('/\[[^\[\]]*\]/', '', $content);
18
+
19
+ return $content;
20
+ }
21
+
22
+ function convertPostToBlock($post, $params = array()) {
23
+
24
+ // defaults
25
+ $defaults = array(
26
+ 'title_tag' => 'h1',
27
+ 'title_alignment' => 'left',
28
+ 'title_position' => 'inside',
29
+ 'image_alignment' => 'left',
30
+ 'image_width' => 325,
31
+ 'readmore' => __('Read online.', WYSIJA),
32
+ 'post_content' => 'full',
33
+ 'author_show' => 'no',
34
+ 'author_label' => '',
35
+ 'category_show' => 'no',
36
+ 'category_label' => ''
37
+ );
38
+
39
+ // merge params with default params
40
+ $params = array_merge($defaults, $params);
41
+
42
+ if($params['post_content'] === 'full') {
43
+ $content = $post['post_content'];
44
+ } else if($params['post_content'] === 'title') {
45
+ $content = $this->getPostTitle($post, $params);
46
+ } else {
47
+ // get excerpt
48
+ if(!empty($post['post_excerpt'])) {
49
+ $content = $post['post_excerpt'];
50
+ } else {
51
+ // strip shortcodes before getting the excerpt
52
+ $post['post_content'] = $this->stripShortcodes($post['post_content']);
53
+
54
+ // if excerpt is empty then try to find the "more" tag
55
+ $excerpts = explode('<!--more-->', $post['post_content']);
56
+ if(count($excerpts) > 1){
57
+ $content = $excerpts[0];
58
+ }else{
59
+ // finally get a made up excerpt if there is no other choice
60
+ $helper_toolbox = WYSIJA::get('toolbox', 'helper');
61
+ $content = $helper_toolbox->excerpt($post['post_content'], apply_filters('mpoet_excerpt_length', 60));
62
+ }
63
+ }
64
+ // strip title tags from excerpt
65
+ $content = preg_replace('/<([\/])?h[123456](.*?)>/', '<$1p$2>', $content);
66
+ }
67
+
68
+ // convert new lines into <p>
69
+ $content = wpautop($content);
70
+
71
+ // remove images
72
+ $content = preg_replace('/<img[^>]+./','', $content);
73
+
74
+ // strip shortcodes
75
+ $content = $this->stripShortcodes($content);
76
+
77
+ // remove wysija nl shortcode
78
+ $content= preg_replace('/\<div class="wysija-register">(.*?)\<\/div>/','',$content);
79
+
80
+ // convert embedded content if necessary
81
+ $content = $this->convertEmbeddedContent($content);
82
+
83
+ // convert h4 h5 h6 to h3
84
+ $content = preg_replace('/<([\/])?h[456](.*?)>/', '<$1h3$2>', $content);
85
+
86
+ // convert ol to ul
87
+ $content = preg_replace('/<([\/])?ol(.*?)>/', '<$1ul$2>', $content);
88
+
89
+ // convert currency signs
90
+ $content = str_replace(array('$', '€', '£', '¥'), array('&#36;', '&euro;', '&pound;', '&#165;'), $content);
91
+
92
+ // strip useless tags
93
+ // TODO should we add table, tr, td and th into that list it could create issues in some cases
94
+ $tags_not_being_stripped = array('<p>','<em>','<span>','<b>','<strong>','<i>','<h1>','<h2>','<h3>','<a>','<ul>','<ol>','<li>','<br>');
95
+
96
+ // filter to modify that list
97
+ $tags_not_being_stripped = apply_filters('mpoet_strip_tags_ignored',$tags_not_being_stripped);
98
+
99
+ $content = strip_tags($content, implode('',$tags_not_being_stripped));
100
+
101
+ // post meta (author, categories)
102
+ $post_meta_above = '';
103
+ // if the author or categories are displayed, open a new paragraph
104
+ if($params['author_show'] === 'above' || $params['category_show'] === 'above') {
105
+ $post_meta_above .= '<p>';
106
+ }
107
+
108
+ // author above
109
+ if($params['author_show'] === 'above') {
110
+ $post_meta_above .= $this->getPostAuthor($post, $params);
111
+ }
112
+ // categories above
113
+ if($params['category_show'] === 'above') {
114
+ // if there is an author already, we need to add an extra break
115
+ if($params['author_show'] === 'above') {
116
+ $post_meta_above .= '<br />';
117
+ }
118
+ // display post categories
119
+ $post_meta_above .= $this->getPostCategories($post, $params);
120
+ }
121
+
122
+ // close the paragraph around author and categories
123
+ if($params['author_show'] === 'above' || $params['category_show'] === 'above') {
124
+ $post_meta_above .= '</p>';
125
+ }
126
+
127
+ if($params['post_content'] !== 'title') {
128
+ if($params['title_position'] === 'inside') {
129
+ // add title
130
+ $content = $this->getPostTitle($post, $params).$post_meta_above.$content;
131
+ } else {
132
+ $content = $post_meta_above.$content;
133
+ }
134
+ } else {
135
+ $content = $post_meta_above.$content;
136
+ }
137
+
138
+ if($params['post_content'] !== 'title') {
139
+ // add read online link
140
+ $content .= '<p><a href="'.get_permalink($post['ID']).'" target="_blank">'.stripslashes($params['readmore']).'</a></p>';
141
+ }
142
+
143
+ // post meta (author, categories) below
144
+ $post_meta_below = '';
145
+
146
+ // if the author or categories are displayed, open a new paragraph
147
+ if($params['author_show'] === 'below' || $params['category_show'] === 'below') {
148
+ $post_meta_below .= '<p>';
149
+ }
150
+
151
+ // author below
152
+ if($params['author_show'] === 'below') {
153
+ $post_meta_below .= $this->getPostAuthor($post, $params);
154
+ }
155
+
156
+ // categories below
157
+ if($params['category_show'] === 'below') {
158
+ // if there is an author already, we need to add an extra break
159
+ if($params['author_show'] === 'below') {
160
+ $post_meta_below .= '<br />';
161
+ }
162
+ $post_meta_below .= $this->getPostCategories($post, $params);
163
+ }
164
+
165
+ // close the paragraph around author and categories
166
+ if($params['author_show'] === 'below' || $params['category_show'] === 'below') {
167
+ $post_meta_below .= '</p>';
168
+ }
169
+
170
+ if($post_meta_below !== '') $content .= $post_meta_below;
171
+
172
+ // set image/text alignment based on present data
173
+ $post_image = null;
174
+
175
+ if(($params['title_tag'] !== 'list') && isset($post['post_image'])) {
176
+ $post_image = $post['post_image'];
177
+
178
+ // set image alignment to match block's
179
+ $post_image['alignment'] = $params['image_alignment'];
180
+
181
+ // constrain size depending on alignment
182
+ if(empty($post_image['height']) or $post_image['height'] === 0) {
183
+ $post_image = null;
184
+ } else {
185
+ $ratio = round(($post_image['width'] / $post_image['height']) * 1000) / 1000;
186
+ switch($params['image_alignment']) {
187
+ case 'alternate':
188
+ case 'left':
189
+ case 'right':
190
+ // constrain width to 325px max
191
+ $post_image['width'] = min($params['image_width'], 325);
192
+ break;
193
+ case 'center':
194
+ // constrain width to 564px max
195
+ $post_image['width'] = min($params['image_width'], 564);
196
+ break;
197
+ }
198
+
199
+ if($ratio > 0) {
200
+ // if ratio has been calculated, deduce image height
201
+ $post_image['height'] = (int)($post_image['width'] / $ratio);
202
+ } else {
203
+ // otherwise skip the image
204
+ $post_image = null;
205
+ }
206
+ }
207
+ }
208
+
209
+ $position = 0;
210
+ if(isset($params['position']) && (int)$params['position'] > 0) {
211
+ $position = (int)$params['position'];
212
+ }
213
+
214
+ $block = array(
215
+ 'position' => $position,
216
+ 'type' => 'content',
217
+ 'text' => array(
218
+ 'value' => base64_encode($content)
219
+ ),
220
+ 'image' => $post_image,
221
+ 'alignment' => $params['image_alignment']
222
+ );
223
+
224
+ return $block;
225
+ }
226
+
227
+ public function getPostAuthor($post = array(), $params = array()) {
228
+ $content = '';
229
+
230
+ if(isset($post['post_author'])) {
231
+ $author_name = get_the_author_meta('display_name', (int)$post['post_author']);
232
+
233
+ // check if the user specified a label to be displayed before the author's name
234
+ if(strlen(trim($params['author_label'])) > 0) {
235
+ $author_name = stripslashes(trim($params['author_label'])).' '.$author_name;
236
+ }
237
+ $content .= $author_name;
238
+ }
239
+
240
+ return $content;
241
+ }
242
+
243
+ public function getPostCategories($post = array(), $params = array()) {
244
+ $content = '';
245
+
246
+ // get categories
247
+ //$categories = get_the_category($post['ID']);
248
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
249
+ $categories = $helper_wp_tools->get_post_categories($post);
250
+
251
+ if(empty($categories) === false) {
252
+ // check if the user specified a label to be displayed before the author's name
253
+ if(strlen(trim($params['category_label'])) > 0) {
254
+ $content = stripslashes($params['category_label']).' ';
255
+ }
256
+
257
+ $content .= join(', ', $categories);
258
+ }
259
+
260
+ return $content;
261
+ }
262
+
263
+ public function getPostTitle($post = array(), $params = array()) {
264
+ $content = '';
265
+
266
+ if(strlen(trim($post['post_title'])) > 0) {
267
+ // cleanup post title and convert currency signs
268
+ $post_title = trim(str_replace(array('$', '€', '£', '¥'), array('&#36;', '&euro;', '&pound;', '&#165;'), strip_tags($post['post_title'])));
269
+
270
+ // open title tag
271
+ if($params['title_tag'] === 'list') {
272
+ $params['title_tag'] = 'li';
273
+ }
274
+
275
+ $content .= '<'.$params['title_tag'].' class="align-'.$params['title_alignment'].'">';
276
+ // set title link
277
+ $content .= '<a href="'.get_permalink($post['ID']).'" target="_blank">';
278
+ // set title
279
+ $content .= $post_title;
280
+ // close title link
281
+ $content .= '</a>';
282
+ // close title tag
283
+ $content .= '</'.$params['title_tag'].'>';
284
+
285
+ }
286
+
287
+ return $content;
288
+ }
289
+
290
+ function getImage($post) {
291
+ $image_info = null;
292
+ $post_image = null;
293
+
294
+ // check if has_post_thumbnail function exists, if not, include wordpress class
295
+ if(!function_exists('has_post_thumbnail')) {
296
+ require_once(ABSPATH.WPINC.'/post-thumbnail-template.php');
297
+ }
298
+
299
+ // check for post thumbnail
300
+ if(has_post_thumbnail($post['ID'])) {
301
+ $post_thumbnail = get_post_thumbnail_id($post['ID']);
302
+
303
+ // get attachment data (src, width, height)
304
+ $image_info = wp_get_attachment_image_src($post_thumbnail, 'single-post-thumbnail');
305
+
306
+ // get alt text
307
+ $altText = trim(strip_tags(get_post_meta($post_thumbnail, '_wp_attachment_image_alt', true)));
308
+ if(strlen($altText) === 0) {
309
+ // if the alt text is empty then use the post title
310
+ $altText = trim(strip_tags($post['post_title']));
311
+ }
312
+ }
313
+
314
+ if($image_info !== null) {
315
+ $post_image = array(
316
+ 'src' => $image_info[0],
317
+ 'width' => $image_info[1],
318
+ 'height' => $image_info[2],
319
+ 'alt' => urlencode($altText)
320
+ );
321
+ } else {
322
+ $matches = $matches2 = array();
323
+
324
+ $output = preg_match_all('/<img.+src=['."'".'"]([^'."'".'"]+)['."'".'"].*>/i', $post['post_content'], $matches);
325
+
326
+ if(isset($matches[0][0])){
327
+ preg_match_all('/(src|height|width|alt)="([^"]*)"/i', $matches[0][0], $matches2);
328
+
329
+ if(isset($matches2[1])){
330
+ foreach($matches2[1] as $k2 => $v2) {
331
+ if(in_array($v2, array('src', 'width', 'height', 'alt'))) {
332
+ if($post_image === null) $post_image = array();
333
+
334
+ if($v2 === 'alt') {
335
+ // special case for alt text as it requireds url encoding
336
+ $post_image[$v2] = urlencode($matches2[2][$k2]);
337
+ } else {
338
+ // otherwise simply get the value
339
+ $post_image[$v2] = $matches2[2][$k2];
340
+ }
341
+ }
342
+ }
343
+ }
344
+ }
345
+ }
346
+
347
+ $helper_images = WYSIJA::get('image','helper');
348
+ $post_image = $helper_images->valid_image($post_image);
349
+
350
+ if($post_image===null) return $post_image;
351
+ return array_merge($post_image, array('url' => get_permalink($post['ID'])));
352
+ }
353
+
354
+
355
+
356
+ function convertEmbeddedContent($content = '') {
357
+ // remove embedded video and replace with links
358
+ $content = preg_replace('#<iframe.*?src=\"(.+?)\".*><\/iframe>#', '<a href="$1">'.__('Click here to view media.', WYSIJA).'</a>', $content);
359
+
360
+ // replace youtube links
361
+ $content = preg_replace('#http://www.youtube.com/embed/([a-zA-Z0-9_-]*)#Ui', 'http://www.youtube.com/watch?v=$1', $content);
362
+
363
+ return $content;
364
+ }
365
+
366
+ function getFilterByType($params = array()) {
367
+ if(array_key_exists('value', $params)) {
368
+ $value = $params['value'];
369
+ } else {
370
+ return;
371
+ }
372
+
373
+ // make sure value is null if it's an empty string
374
+ if($value !== null and strlen(trim($value)) === 0) $value = null;
375
+
376
+ // get all post types
377
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
378
+ $post_types = $helper_wp_tools->get_post_types();
379
+
380
+ // build post type selection
381
+ $output = '<select name="post_type" id="post_type">';
382
+
383
+ if(isset($params['label'])) {
384
+ $output .= '<option value=""'.(($value === null) ? ' selected="selected"' : '').'>'.$params['label'].'</option>';
385
+ }
386
+ $output .= '<option value="post"'.(($value === 'post') ? ' selected="selected"' : '').'>'.__('Posts',WYSIJA).'</option>';
387
+ $output .= '<option value="page"'.(($value === 'page') ? ' selected="selected"' : '').'>'.__('Pages',WYSIJA).'</option>';
388
+
389
+ foreach($post_types as $key=> $object_post_type) {
390
+ $selected = ($value === $key) ? ' selected="selected"' : '';
391
+ $output .= '<option value="'.$key.'"'.$selected.'>'.$object_post_type->labels->name.'</option>';
392
+ }
393
+ $output .= '</select>';
394
+ return $output;
395
+ }
396
+
397
+ function getFilterByCategory($value = null, $post_type = 'post') {
398
+ // make sure value is null if it's an empty string
399
+ if($value !== null and strlen(trim($value)) === 0) $value = null;
400
+
401
+ $helper_wp_tools = WYSIJA::get('wp_tools','helper');
402
+ $categories = $helper_wp_tools->get_categories();
403
+
404
+ $output = '';
405
+ foreach($categories as $cpt => $categories_of_cpt){
406
+ $output .= '<select id="post_category_'.$cpt.'" name="category_ids" class="post_category">';
407
+ $output .= '<option value="">'.__('Filter by category', WYSIJA).'</option>';
408
+ foreach($categories_of_cpt as $category) {
409
+ $output .= '<option value="'.$category['id'].'">'.$category['name'].'</option>';
410
+ }
411
+ $output .= '</select>';
412
+ }
413
+ return $output;
414
+ }
415
+
416
+
417
+ function getFilterByStatus($current_status = 'publish') {
418
+ $output = '';
419
+
420
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
421
+ $statuses = $helper_wp_tools->get_post_statuses();
422
+
423
+ $output .= '<select id="post_status" name="post_status" class="post_status">';
424
+ $output .= '<option value="">'.__('Filter by status', WYSIJA).'</option>';
425
+ foreach($statuses as $key => $label) {
426
+ $is_selected = ($current_status === $key) ? 'selected="selected"' : '';
427
+ $output .= '<option value="'.$key.'" '.$is_selected.'>'.$label.'</option>';
428
+ }
429
+ $output .= '</select>';
430
+ return $output;
431
+ }
432
+ }
helpers/autonews.php ADDED
@@ -0,0 +1,390 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_autonews extends WYSIJA_object {
4
+ function WYSIJA_help_autonews() {
5
+ }
6
+
7
+ function events($key=false,$get=true,$value_set=array()){
8
+ static $events=array();
9
+ if($get){
10
+ if(!$key){
11
+ return $events;
12
+ }else{
13
+ if(isset($events[$key])) return $events[$key];
14
+ return false;
15
+ }
16
+
17
+ }else{
18
+ if(isset($events[$key])) return false;
19
+ $events[$key]=$value_set;
20
+ }
21
+ }
22
+
23
+ function register($key_event,$event=array()){
24
+ $this->events($key_event,false,$event);
25
+ }
26
+
27
+ function get($fieldKey){
28
+ return $this->events($fieldKey);
29
+ }
30
+
31
+ /**
32
+ * I'm not sure if this function is useful anymore
33
+ * @param type $email
34
+ * @return type
35
+ */
36
+ function _deprecated_nextSend($email=false){
37
+ if(!$email) return;
38
+ $model_email=WYSIJA::get('email','model');
39
+
40
+ if(is_array($email)){
41
+ $email_data=$model_email->getOne(false,array('email_id'=>$email['email_id']));
42
+ }else{
43
+ $email_data=$model_email->getOne(false,array('email_id'=>$email));
44
+ }
45
+ return $model_email->give_birth($email_data);
46
+ }
47
+
48
+ /**
49
+ *
50
+ * @param type $email
51
+ * @return type
52
+ */
53
+ function getNextSend($email) {
54
+ $schedule_at = -1;
55
+
56
+ //this condition makes sure that our $email is a post notification
57
+ if((int)$email['type'] === 2 && isset($email['params']['autonl']['event']) && $email['params']['autonl']['event'] === 'new-articles') {
58
+
59
+ $helper_toolbox = WYSIJA::get('toolbox','helper');
60
+ // get current time
61
+ $now = time();
62
+
63
+
64
+ //local time = site's time set by the administrator of the site in WP's settings
65
+ //server time = time return by PHP functions such as time()
66
+ //nextSend is the time scheduled in the future of the local time
67
+ //so if we compare the current time to the next send value we need to offset only one value and that would be the nextSend
68
+ //this way both values are on server time
69
+ if(!isset($email['params']['autonl']['nextSend']) || $now > $helper_toolbox->localtime_to_servertime($email['params']['autonl']['nextSend'])) {
70
+ switch($email['params']['autonl']['when-article']) {
71
+ case 'immediate':
72
+ break;
73
+ case 'daily':
74
+ // get timestamp of when the newsletter is supposed to be sent
75
+ $schedule_at = strtotime($email['params']['autonl']['time']);
76
+
77
+ // check if the scheduled at time has already passed
78
+ if($helper_toolbox->localtime_to_servertime($schedule_at) < $now) {
79
+ // schedule it for tomorrow
80
+ $schedule_at = strtotime('tomorrow '.$email['params']['autonl']['time']);
81
+ }
82
+ break;
83
+ case 'weekly':
84
+ // get timestamp of when the newsletter is supposed to be sent
85
+ $schedule_at = strtotime(ucfirst($email['params']['autonl']['dayname']).' '.$email['params']['autonl']['time']);
86
+
87
+ // check if the scheduled at time has already passed
88
+ if($helper_toolbox->localtime_to_servertime($schedule_at) < $now) {
89
+ // schedule it for next week
90
+ $schedule_at = strtotime('next '.ucfirst($email['params']['autonl']['dayname']).' '.$email['params']['autonl']['time']);
91
+ }
92
+ break;
93
+ case 'monthly':
94
+ $time_current_day=date('d',$now);
95
+ $time_current_month=date('m',$now);
96
+ $time_current_year=date('y',$now);
97
+
98
+ //trigger has to be next month
99
+ if($time_current_day > $email['params']['autonl']['daynumber']) {
100
+
101
+ if((int)$time_current_month === 12) {
102
+ //year +1
103
+ $time_current_month=1;
104
+ $time_current_year++;
105
+ }else{
106
+ //current year
107
+ $time_current_month++;
108
+ }
109
+ }
110
+ $schedule_at=strtotime($time_current_month.'/'.$email['params']['autonl']['daynumber'].'/'.$time_current_year.' '.$email['params']['autonl']['time']);
111
+ break;
112
+ case 'monthlyevery': // monthly every X Day of the week
113
+ $current_day = date('d', $now);
114
+ $current_month = date('m', $now);
115
+ $current_year = date('y', $now);
116
+
117
+ // calculate the timestamp of the Xth day of the week of the current month
118
+ // strtotime('02/01/13 1 Monday 20:00:00') -> this will return the timestamp of the 1st monday of the current month
119
+ $schedule_at = strtotime(
120
+ sprintf('%02d/01/%02d %d %s %s',
121
+ $current_month,
122
+ $current_year,
123
+ $email['params']['autonl']['dayevery'],
124
+ ucfirst($email['params']['autonl']['dayname']),
125
+ $email['params']['autonl']['time']
126
+ ));
127
+
128
+ if($helper_toolbox->localtime_to_servertime($schedule_at) < $now) {
129
+ // get first day timestamp of next month
130
+ $first_day_of_next_month = $this->get_first_day_of_month($schedule_at, 1);
131
+
132
+ // get next month's Xth day of the week
133
+ $schedule_at = strtotime(
134
+ sprintf('%02d/01/%02d %d %s %s',
135
+ date('m', $first_day_of_next_month),
136
+ date('y', $first_day_of_next_month),
137
+ $email['params']['autonl']['dayevery'],
138
+ ucfirst($email['params']['autonl']['dayname']),
139
+ $email['params']['autonl']['time']
140
+ )
141
+ );
142
+ }
143
+ break;
144
+ }
145
+ }
146
+ }
147
+ return $schedule_at;
148
+ }
149
+
150
+ function get_first_day_of_month($time_stamp, $months_to_add = 0) {
151
+ // You can add as many months as you want. mktime will accumulate to the next year.
152
+ $date = getdate($time_stamp); // Covert to Array
153
+ // add number of months
154
+ $date['mon'] = $date['mon'] + (int)$months_to_add;
155
+ // set day to 1
156
+ $date['mday'] = 1;
157
+ // return timestamp
158
+ return mktime($date['hours'], $date['minutes'], $date['seconds'], $date['mon'], $date['mday'], $date['year']);
159
+ }
160
+
161
+ /**
162
+ * get the time of the n dayname of the month
163
+ * @param type $first_day_of_month
164
+ * @param type $day_name
165
+ * @param type $which_number
166
+ * @param type $time_now
167
+ * @return type
168
+ */
169
+ function getNextDay($first_day_of_month,$day_name,$which_number,$time_now){
170
+ $name_first_day = strtolower(date('l', $first_day_of_month));
171
+
172
+ if($name_first_day == strtolower($day_name)) $which_number--;
173
+ for($i=0; $i < $which_number;$i++){
174
+ $first_day_of_month = strtotime('next '.ucfirst($day_name), $first_day_of_month);
175
+ }
176
+ return $first_day_of_month;
177
+ }
178
+
179
+
180
+ /**
181
+ * check if there is post notification needing a child email
182
+ */
183
+ function checkPostNotif(){
184
+ // flag security to make sure that there can't be two checks of the post notif in the same minute
185
+ $current_check = (float)get_option('wysija_check_pn');
186
+
187
+ // there is a check that has been starting to run less than 60 seconds ago
188
+ if(microtime(true) < ($current_check+60)){
189
+ WYSIJA::log('already_running_checkPN', $current_check, 'post_notif');
190
+ return false;
191
+ }
192
+
193
+ // flag is down we process our post notification check and set the start time of the current check
194
+ $current_check=microtime(true);
195
+ WYSIJA::update_option('wysija_check_pn',$current_check);
196
+
197
+ // let's check when do we come here
198
+ WYSIJA::log('check_post_notif_starts', $current_check , 'post_notif');
199
+
200
+ $model_email=WYSIJA::get('email','model');
201
+ $model_email->reset();
202
+ $all_emails=$model_email->get(false,array('type'=>'2','status'=>array('1','3','99')));
203
+
204
+ if($all_emails){
205
+ $helper_toolbox=WYSIJA::get('toolbox','helper');
206
+ foreach($all_emails as $email){
207
+ //post notification make a child newsletter when the timing is immediate otherwise let the cron take care of it
208
+ if($email['params']['autonl']['event']=='new-articles' && $email['params']['autonl']['when-article']!='immediate'){
209
+ //check if the next sending is passed if so then we give birth to one child email
210
+ //IMPORTANT WE COMPARE TO THE OFFSET TIME (time set by the administrator)
211
+ //if the nextSend doesn't exist then we set it
212
+ if(!isset($email['params']['autonl']['nextSend'])){
213
+ WYSIJA::log('check_post_notif_next_send_not_set', $current_check , 'post_notif');
214
+ }else {
215
+ //if the next send is passed we should trigger it
216
+ $time_now_server=time();
217
+ if($time_now_server > $helper_toolbox->localtime_to_servertime($email['params']['autonl']['nextSend'])){
218
+ $how_late=$time_now_server-$helper_toolbox->localtime_to_servertime($email['params']['autonl']['nextSend']);
219
+ //check how late was the previous notification,
220
+ //if it has been more than two hours late then cancel it and change it to the next day
221
+ if(!$this->cancel_late_post_notification($email,$how_late)){
222
+ WYSIJA::log('check_post_notif_before_give_birth', $current_check, 'post_notif');
223
+ //it is not cancel so we can give birth toa a child newsletter
224
+ $model_email->give_birth($email);
225
+ }
226
+ }
227
+ }
228
+ }
229
+ }
230
+ }
231
+ }
232
+
233
+ /**
234
+ * email has a late post notification which needs to be cancelled and postponed
235
+ * @param array $email
236
+ * @param int $how_late
237
+ * @return boolean
238
+ */
239
+ function cancel_late_post_notification($email,$how_late){
240
+ $cancel_it=false;
241
+ switch($email['params']['autonl']['when-article']) {
242
+ case 'daily':
243
+ //cancel a daily notification with more than two hours delay
244
+ if($how_late>(2*3600)){
245
+ $cancel_it=true;
246
+ }
247
+ break;
248
+ case 'weekly':
249
+ //cancel a weekly notification with more than half a day delay
250
+ if($how_late>(12*3600)){
251
+ $cancel_it=true;
252
+ }
253
+ break;
254
+ case 'monthly':
255
+ //cancel a monthly notification with more than a day delay
256
+ if($how_late>(24*3600)){
257
+ $cancel_it=true;
258
+ }
259
+ break;
260
+ case 'monthlyevery':
261
+ //cancel a monthly notification with more than a day delay
262
+ if($how_late>(24*3600)){
263
+ $cancel_it=true;
264
+ }
265
+ break;
266
+ }
267
+ //if the notification is being cancelled then we store the value of that late notification and update the nextSend
268
+ if($cancel_it){
269
+
270
+ $late_send=$email['params']['autonl']['nextSend'];
271
+ WYSIJA::log('cancel_late_post_notification_late_send', $late_send, 'post_notif');
272
+ $next_send=$this->getNextSend($email);
273
+ $email['params']['autonl']['nextSend']=$next_send;
274
+ $email['params']['autonl']['late_send']=$late_send;
275
+
276
+ $model_email=WYSIJA::get('email','model');
277
+ $model_email->reset();
278
+ $model_email->update(array('params'=>$email['params']), array('email_id' => $email['email_id']));
279
+
280
+ return true;
281
+ }
282
+
283
+ return false;
284
+ }
285
+
286
+ /**
287
+ * check if there are any scheduled email not sent yet
288
+ */
289
+ function checkScheduled(){
290
+ $model_email = WYSIJA::get('email','model');
291
+ $helper_toolbox = WYSIJA::get('toolbox','helper');
292
+ $model_email->reset();
293
+
294
+ // select the scheduled emails
295
+ $all_emails = $model_email->get(false,array('type'=>'1','status'=>'4'));
296
+
297
+ if($all_emails){
298
+
299
+ foreach($all_emails as $email){
300
+
301
+ // check if the email is scheduled
302
+ if(isset($email['params']['schedule']['isscheduled'])){
303
+
304
+ $schedule_date = $email['params']['schedule']['day'] . ' ' . $email['params']['schedule']['time'];
305
+ $unix_scheduled_time = strtotime($schedule_date);
306
+
307
+ // if the scheduled time is passed let's send the email
308
+ // we don't compare to the time recorded but to the offset time which is the time set by the user in his time
309
+ if($helper_toolbox->localtime_to_servertime($unix_scheduled_time) < time()){
310
+ $model_email->reset();
311
+ $model_email->send_activate($email);
312
+ }
313
+ }
314
+ }
315
+ }
316
+ }
317
+
318
+ function refresh_automatic_content($email_ids = array()) {
319
+ // TO OPTIMIZE: add a boolean flag for ALP widget being present or not (this way we filter out "static" auto nl)
320
+ $model_email = WYSIJA::get('email', 'model');
321
+
322
+ $conditions = array('type' => 2, 'status' => array(1, 3, 99));
323
+ if(!empty($email_ids)) {
324
+ $conditions['email_id'] = $email_ids;
325
+ }
326
+ // get only the data needed to update an auto nl so we save some resources
327
+ $data_needed = array('campaign_id','email_id','params','wj_styles','wj_data');
328
+ $emails = $model_email->get( $data_needed, $conditions );
329
+
330
+ foreach($emails as $key => $email) {
331
+ if(is_array($email) && isset($email['params']['autonl']['event']) ) {
332
+
333
+ $wj_data = unserialize(base64_decode($email['wj_data']));
334
+ $reload_auto_content = false;
335
+ foreach($wj_data['body'] as $block) {
336
+ if(isset($block['type']) && $block['type'] === 'auto-post') {
337
+ $reload_auto_content = true;
338
+ }
339
+ }
340
+ if(!$reload_auto_content) continue;
341
+
342
+ // we have to regenerate the html rendering of each auto newsletter
343
+ $helper_wj_engine = WYSIJA::get('wj_engine', 'helper');
344
+ $helper_wj_engine->setStyles($email['wj_styles'], true);
345
+ $helper_wj_engine->setData($email['wj_data'], true);
346
+
347
+ // update email data
348
+ $values = array(
349
+ 'email_id' => (int)$email['email_id'],
350
+ 'wj_data' => $helper_wj_engine->getEncoded('data'),
351
+ 'body' => $helper_wj_engine->renderEmail($email)
352
+ );
353
+
354
+ // make sure the modified_at columns get updated
355
+ $model_email->columns['modified_at']['autoup'] = 1;
356
+
357
+ // update data in DB
358
+ $model_email->update($values, array('email_id' => (int)$email['email_id']));
359
+ }
360
+ }
361
+ }
362
+
363
+ // removes any auto-post block
364
+ function remove_autopost_blocks($data = array()) {
365
+ if(empty($data)) {
366
+ return false;
367
+ }
368
+
369
+ // decode data
370
+ $wj_data = unserialize(base64_decode($data));
371
+
372
+ // init updated_wj_data variable with initial wj_data
373
+ $updated_wj_data = $wj_data;
374
+
375
+ foreach($wj_data['body'] as $key => $block) {
376
+ // if we detect an auto-post block, we need to remove it
377
+ if(isset($block['type']) && $block['type'] === 'auto-post') {
378
+ // remove block from updated data
379
+ unset($updated_wj_data['body'][$key]);
380
+ }
381
+ }
382
+ // if the wj_data has not changed, return false
383
+ if($updated_wj_data === $wj_data) {
384
+ return false;
385
+ } else {
386
+ // otherwise return encoded version of updated_wj_data
387
+ return base64_encode(serialize($updated_wj_data));
388
+ }
389
+ }
390
+ }
helpers/back.php ADDED
@@ -0,0 +1,529 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+
5
+ /**
6
+ * class managing the admin vital part to integrate wordpress
7
+ */
8
+ class WYSIJA_help_back extends WYSIJA_help{
9
+
10
+ function WYSIJA_help_back(){
11
+ parent::WYSIJA_help();
12
+ //check that the application has been installed properly
13
+ $config=WYSIJA::get('config','model');
14
+
15
+ define('WYSIJA_DBG',(int)$config->getValue('debug_new'));
16
+ //by default do not show the errors until we get into the debug file
17
+ if(!defined('WP_DEBUG') || !WP_DEBUG){
18
+ error_reporting(0);
19
+ ini_set('display_errors', '0');
20
+ }
21
+
22
+ add_filter('admin_footer_text', array(&$this, 'admin_footer_text'));
23
+ add_filter('update_footer', array(&$this, 'update_footer'), 15);
24
+
25
+ //the controller is backend is it from our pages or from wordpress?
26
+ //are we pluging-in to wordpress interfaces or doing entirely our own page?
27
+ if(isset($_GET['page']) && substr($_GET['page'],0,7)=='wysija_'){
28
+ define('WYSIJA_ITF',TRUE);
29
+ $this->controller=WYSIJA::get(str_replace('wysija_','',$_GET['page']),'controller');
30
+ }else{//check if we are pluging in wordpress interface
31
+ define('WYSIJA_ITF',FALSE);
32
+ }
33
+
34
+ if( WYSIJA_DBG>0 ) include_once(WYSIJA_INC.'debug.php');
35
+
36
+ if(!function_exists('dbg')) {
37
+ function dbg($mixed,$exit=true){}
38
+ }
39
+
40
+
41
+ //we set up the important hooks for backend: menus js css etc
42
+ if(defined('DOING_AJAX')){
43
+ //difference between frontend and backend
44
+
45
+ add_action( 'after_setup_theme', array($this, 'ajax_setup') );
46
+
47
+ }else{
48
+ if(WYSIJA_ITF) {
49
+ add_action('admin_init', array( $this , 'verify_capability'),1);
50
+ add_action('admin_init', array($this->controller, 'main'));
51
+ add_action('after_setup_theme',array($this,'resolveConflicts'));
52
+ }
53
+ //this is a fix for qtranslate as we were loading translatable string quite early
54
+
55
+
56
+ //somehow if we add caps to one role the user with that role doesnt get its caps updated ...
57
+ add_action('after_setup_theme', array('WYSIJA', 'update_user_caps'),11);
58
+ add_action('admin_menu', array($this, 'define_translated_strings'),98);
59
+ add_action('admin_menu', array($this, 'add_menus'),99);
60
+ add_action('admin_enqueue_scripts',array($this, 'add_js'),10,1);
61
+
62
+
63
+ //add specific page script
64
+ add_action('admin_head-post-new.php',array($this,'addCodeToPagePost'));
65
+ add_action('admin_head-post.php',array($this,'addCodeToPagePost'));
66
+
67
+ //make sure that admin and super admin always have the highest access
68
+ $wptools = WYSIJA::get('wp_tools', 'helper');
69
+ $wptools->set_default_rolecaps();
70
+
71
+ // Hook the warning function for premium.
72
+ add_action( 'admin_init', array(&$this, 'warn_action_on_premium') );
73
+ }
74
+
75
+ //if the comment form option is activated then we add an approval action
76
+ if($config->getValue('commentform')){
77
+ add_action('wp_set_comment_status', array($this,'comment_approved'), 60,2);
78
+ }
79
+ }
80
+
81
+
82
+ /**
83
+ * On any of the administration pages related to MailPoet, if the user
84
+ * has the key and doesn't have the premium plugin active a warning will
85
+ * be displayed.
86
+ *
87
+ * @return null
88
+ */
89
+ public function warn_action_on_premium(){
90
+ $mdl_config=WYSIJA::get('config','model');
91
+
92
+ if($mdl_config->getValue('premium_key') && !WYSIJA::is_plugin_active(WJ_Upgrade::$plugins[1])){
93
+ if( file_exists( WPMU_PLUGIN_DIR . DIRECTORY_SEPARATOR . WJ_Upgrade::$plugins[1] ) || file_exists( WP_PLUGIN_DIR . DIRECTORY_SEPARATOR . WJ_Upgrade::$plugins[1] ) ){
94
+ //send a message to the user so that he activates the premium plugin or try to fetch it directly.
95
+ $this->notice('<p>'.__('You need to activate the MailPoet Premium plugin.', WYSIJA).' <a data-warn="' . esc_attr__( "Confirm activating the MailPoet Premium Plugin?", WYSIJA ) . '" class="button-primary" title="' . esc_attr__( "Activate MailPoet Premium Version", WYSIJA ) . '" href="' . wp_nonce_url('plugins.php?action=activate&amp;plugin=' . urlencode(WJ_Upgrade::$plugins[1]) . '&amp;plugin_status=all', 'activate-plugin_' . WJ_Upgrade::$plugins[1]) . '">'.__('Activate now',WYSIJA).'</a></p>');
96
+ } else {
97
+
98
+ $args = array(
99
+ 'page' => 'wysija_config',
100
+ 'action' => 'packager-switch',
101
+ '_mp_action' => 'install',
102
+ '_wpnonce' => wp_create_nonce('packager-switch'),
103
+ );
104
+ if (WYSIJA::is_beta())
105
+ $args["stable"] = 1;
106
+
107
+ $link = esc_attr(add_query_arg($args, admin_url('admin.php')));
108
+
109
+ //send a message to the user so that he gets the premium plugin or try to fetch it directly.
110
+ $this->notice('<p>'.__('Congrats, your Premium license is active. One last step...', WYSIJA).' <a data-warn="' . esc_attr__( "Confirm installing the MailPoet Premium Plugin?", WYSIJA ) . '" id="install-wjp" class="button-primary" title="' . esc_attr__( "Installing MailPoet Premium Version", WYSIJA ) . '" href="' . esc_url($link) . '">'.__('Download the Premium plugin.',WYSIJA).'</a></p>');
111
+ }
112
+ }
113
+
114
+ return null;
115
+ }
116
+
117
+ function comment_approved($cid,$comment_status){
118
+ //if the comment is approved and the meta wysija_comment_subscribe appears, then we have one subscriber more to add
119
+ $metaresult=get_comment_meta($cid, 'wysija_comment_subscribe', true);
120
+
121
+ if($comment_status=='approve' && get_comment_meta($cid, 'wysija_comment_subscribe', true)){
122
+ $mConfig=WYSIJA::get('config','model');
123
+ $comment = get_comment($cid);
124
+ $userHelper=WYSIJA::get('user','helper');
125
+ $data=array('user'=>array('email'=>$comment->comment_author_email,'firstname'=>$comment->comment_author),'user_list'=>array('list_ids'=>$mConfig->getValue('commentform_lists')));
126
+ $userHelper->addSubscriber($data);
127
+ }
128
+ }
129
+
130
+ function ajax_setup(){
131
+ if(!isset($_REQUEST['adminurl']) && !is_user_logged_in()) add_action('wp_ajax_nopriv_wysija_ajax', array($this, 'ajax'));
132
+ else add_action('wp_ajax_wysija_ajax', array($this, 'ajax'));
133
+ }
134
+
135
+
136
+ /**
137
+ * let's fix all the conflicts that we may have
138
+ */
139
+ function resolveConflicts(){
140
+
141
+ // check conflicting themes
142
+ $possibleConflictiveThemes = $this->controller->get_conflictive_plugins(true);
143
+
144
+ $conflictingTheme = null;
145
+ $currentTheme = strtolower(function_exists( 'wp_get_theme' ) ? wp_get_theme() : get_current_theme());
146
+ foreach($possibleConflictiveThemes as $keyTheme => $conflictTheme) {
147
+ if($keyTheme === $currentTheme) {
148
+ $conflictingTheme = $keyTheme;
149
+ }
150
+ }
151
+
152
+ // if the current theme is known to make troubles, let's resolve this
153
+ if($conflictingTheme !== null) {
154
+ $helperConflicts = WYSIJA::get('conflicts', 'helper');
155
+ $helperConflicts->resolve(array($possibleConflictiveThemes[$conflictingTheme]));
156
+ }
157
+
158
+ // check conflicting plugins
159
+ $possibleConflictivePlugins=$this->controller->get_conflictive_plugins();
160
+
161
+ $conflictingPlugins=array();
162
+ foreach($possibleConflictivePlugins as $keyPlg => $conflictPlug){
163
+ if(WYSIJA::is_plugin_active($conflictPlug['file'])) {
164
+ //plugin is activated
165
+ $conflictingPlugins[$keyPlg]=$conflictPlug;
166
+ }
167
+ }
168
+
169
+ if($conflictingPlugins){
170
+ $helperConflicts=WYSIJA::get('conflicts','helper');
171
+ $helperConflicts->resolve($conflictingPlugins);
172
+ }
173
+ }
174
+
175
+ /**
176
+ * this function will check the role of the user executing the action, if it's called from another
177
+ * WordPress admin page than page.php for instance admin-post.php
178
+ * @return boolean
179
+ */
180
+ function verify_capability(){
181
+ if( isset( $_GET['page'] ) && substr( $_GET['page'] ,0 ,7 ) == 'wysija_' ){
182
+
183
+ switch( $_GET['page'] ){
184
+ case 'wysija_campaigns':
185
+ $role_needed = 'wysija_newsletters';
186
+ break;
187
+ case 'wysija_subscribers':
188
+ $role_needed = 'wysija_subscribers';
189
+ break;
190
+ case 'wysija_config':
191
+ $role_needed = 'wysija_config';
192
+ break;
193
+ case 'wysija_statistics':
194
+ $role_needed = 'wysija_stats_dashboard';
195
+ break;
196
+ default:
197
+ $role_needed = 'switch_themes';
198
+ }
199
+
200
+ if( current_user_can( $role_needed ) ){
201
+ return true;
202
+ } else{
203
+ die( 'You are not allowed here.' );
204
+ }
205
+
206
+ }else{
207
+ // this is not a wysija interface/action we can let it pass
208
+ return true;
209
+ }
210
+ }
211
+
212
+ /**
213
+ * translatable strings need to be not loaded to early, this is why we put them ina separate function
214
+ * @global type $wysija_installing
215
+ */
216
+ function define_translated_strings(){
217
+ $config = WYSIJA::get('config','model');
218
+ $linkcontent = __("It doesn't always work the way we want it to, doesn't it? We have a [link]dedicated support website[/link] with documentation and a ticketing system.",WYSIJA);
219
+ $finds = array('[link]','[/link]');
220
+ $replace = array('<a target="_blank" href="http://support.mailpoet.com" title="support.mailpoet.com">','</a>');
221
+ $truelinkhelp = '<p>'.str_replace($finds,$replace,$linkcontent).'</p>';
222
+
223
+ $extra = '<a href="admin.php?page=wysija_config&scroll_to=beta_mode_setting#tab-advanced" title="'.__('Switch to beta',WYSIJA).'">'.__('Switch to beta',WYSIJA).'</a>';
224
+
225
+ $truelinkhelp .= '<p>'.str_replace($finds,$replace,$extra).'</p>';
226
+
227
+ $truelinkhelp .= '<p>'.__('MailPoet Version: ',WYSIJA).'<strong>'.WYSIJA::get_version().'</strong></p>';
228
+
229
+ $this->menus=array(
230
+ 'campaigns'=>array('title'=>'MailPoet'),
231
+ 'subscribers'=>array('title'=>__('Subscribers',WYSIJA)), // if the key "subscribers" is changed, please change in the filter "wysija_menus" as well.
232
+ 'config'=>array('title'=>__('Settings',WYSIJA)),
233
+ 'premium'=>array('title'=>__('Premium',WYSIJA))
234
+ );
235
+ $this->menus = apply_filters('wysija_menus', $this->menus);
236
+ $this->menuHelp = $truelinkhelp;
237
+ if($config->getValue('queue_sends_slow')){
238
+ $msg=$config->getValue('ignore_msgs');
239
+ if(!isset($msg['queuesendsslow'])){
240
+ $this->notice(
241
+ __('Tired of waiting more than 48h to send your emails?',WYSIJA).' '. str_replace(array('[link]','[/link]'), array('<a href="http://support.mailpoet.com/knowledgebase/how-fast-can-i-send-emails-optimal-sending-configurations-explained/?utm_source=wpadmin&utm_campaign=slowqueue" target="_blank">','</a>'), __('[link]Find out[/link] how you can improve this.',WYSIJA)).
242
+ ' <a class="linkignore queuesendsslow" href="javascript:;">'.__('Hide!',WYSIJA).'</a>');
243
+ }
244
+ }
245
+
246
+ if(WYSIJA_ITF){
247
+ global $wysija_installing;
248
+ if( !$config->getValue('sending_emails_ok')){
249
+ $msg=$config->getValue('ignore_msgs');
250
+
251
+ $urlsendingmethod='admin.php?page=wysija_config#tab-sendingmethod';
252
+ if($_REQUEST['page'] === 'wysija_config') {
253
+ $urlsendingmethod='#tab-sendingmethod';
254
+ }
255
+
256
+ }
257
+ }
258
+ }
259
+
260
+
261
+ function add_menus(){
262
+ global $menu,$submenu;// WordPress globals be careful there
263
+ $count=0;
264
+
265
+ //anti conflicting menus code to make sure that another plugin is not at the same level as us
266
+ $position=50;
267
+ $positionplus1=$position+1;
268
+
269
+ while(isset($menu[$position]) || isset($menu[$positionplus1])){
270
+ $position++;
271
+ $positionplus1=$position+1;
272
+ //check that there is no menu at our level neither at ourlevel+1 because that will make us disappear in some case :/
273
+ if(!isset($menu[$position]) && isset($menu[$positionplus1])){
274
+ $position=$position+2;
275
+ }
276
+ }
277
+
278
+ global $wysija_installing;
279
+ foreach($this->menus as $action=> $menutemp){
280
+ $actionFull='wysija_'.$action;
281
+ if (!isset($menutemp['subtitle']))
282
+ $menutemp['subtitle'] = $menutemp['title'];
283
+ if ($action == 'campaigns')
284
+ $roleformenu = 'wysija_newsletters';
285
+ elseif ($action == 'subscribers')
286
+ $roleformenu = 'wysija_subscribers';
287
+ elseif ($action == 'statistics')
288
+ $roleformenu = 'wysija_stats_dashboard';
289
+ else
290
+ $roleformenu = 'wysija_config';
291
+
292
+ if($wysija_installing===true){
293
+ if($count==0){
294
+ $parentmenu=$actionFull;
295
+ $hookname=add_menu_page($menutemp['title'], $menutemp['subtitle'], $roleformenu, $actionFull , array($this->controller, 'errorInstall'), 'div', $position);
296
+ }
297
+ }else{
298
+ if($count==0){
299
+ $parentmenu = $actionFull;
300
+ $hookname = add_menu_page(
301
+ $menutemp['title'],
302
+ $menutemp['subtitle'],
303
+ $roleformenu,
304
+ $actionFull ,
305
+ array($this->controller, 'render'),
306
+ 'div',
307
+ $position
308
+ );
309
+ }else{
310
+ $hookname=add_submenu_page($parentmenu,$menutemp['title'], $menutemp['subtitle'], $roleformenu, $actionFull , array($this->controller, 'render'));
311
+ }
312
+
313
+ //manage wp help tab
314
+ if(WYSIJA_ITF){
315
+ //wp3.3
316
+ if(version_compare(get_bloginfo('version'), '3.3.0')>= 0){
317
+ add_action('load-'.$hookname, array($this,'add_help_tab'));
318
+ }else{
319
+ //wp3.0
320
+ add_contextual_help($hookname, $this->menuHelp);
321
+ }
322
+ }
323
+ }
324
+ $count++;
325
+ }
326
+
327
+ // Correct the text of submenu, in case there is only 1 submenu is enabled
328
+ if(isset($submenu[$parentmenu])) {
329
+ switch ($submenu[$parentmenu][0][2]) {
330
+ case 'wysija_subscribers':
331
+ $textmenu=__('Subscribers',WYSIJA);
332
+ break;
333
+
334
+ case 'wysija_statistics':
335
+ $textmenu=__('Statistics',WYSIJA);
336
+ break;
337
+
338
+ case 'wysija_config':
339
+ $textmenu=__('Settings',WYSIJA);
340
+ break;
341
+
342
+ case 'wysija_campaigns':
343
+ default:
344
+ $textmenu=__('Newsletters',WYSIJA);
345
+ break;
346
+ }
347
+ $submenu[$parentmenu][0][0]=$submenu[$parentmenu][0][3]=$textmenu;
348
+ }
349
+ }
350
+
351
+ function add_help_tab($params){
352
+ $screen = get_current_screen();
353
+
354
+ if(method_exists($screen, "add_help_tab")){
355
+ $screen->add_help_tab(array(
356
+ 'id' => 'wysija_help_tab',
357
+ 'title' => __('Get Help!',WYSIJA),
358
+ 'content'=> $this->menuHelp));
359
+ $tabfunc=true;
360
+ }
361
+ }
362
+
363
+ function add_js($hook) {
364
+ //needed in all the wordpress admin pages including wysija's ones
365
+
366
+ $jstrans=array();
367
+ wp_register_script('wysija-charts', 'https://www.google.com/jsapi', array( 'jquery' ), true);
368
+ wp_register_script('wysija-admin-list', WYSIJA_URL.'js/admin-listing.js', array( 'jquery' ), true, WYSIJA::get_version());
369
+ wp_register_script('wysija-base-script-64', WYSIJA_URL.'js/base-script-64.js', array( 'jquery' ), true, WYSIJA::get_version());
370
+
371
+
372
+ wp_enqueue_style('wysija-admin-css-widget', WYSIJA_URL.'css/admin-widget.css',array(),WYSIJA::get_version());
373
+
374
+ // If Cron enabled sending, send Mixpanel data and reset flag.
375
+ $model_config = WYSIJA::get('config', 'model');
376
+ if ($model_config->getValue('send_analytics_now') == 1) {
377
+ $analytics = new WJ_Analytics();
378
+ $analytics->generate_data();
379
+ $analytics->send();
380
+ // Reset sending flag.
381
+ $model_config->save(array('send_analytics_now' => 0));
382
+ }
383
+
384
+
385
+ //we are in wysija's admin interface
386
+ if(WYSIJA_ITF){
387
+ wp_enqueue_style('wysija-admin-css-global', WYSIJA_URL.'css/admin-global.css',array(),WYSIJA::get_version());
388
+ wp_enqueue_script('wysija-admin-js-global', WYSIJA_URL.'js/admin-wysija-global.js',array(),WYSIJA::get_version());
389
+ $pagename=str_replace('wysija_','',$_REQUEST['page']);
390
+ $backloader=WYSIJA::get('backloader','helper');
391
+ $backloader->init( $this->controller );
392
+
393
+ //$this->controller->jsTrans["ignoremsg"]=__('Are you sure you want to ignore this message?.',WYSIJA);
394
+ $jstrans=$this->controller->jsTrans;
395
+ //if(!in_array('wysija-admin-ajax-proto',$this->controller->js)) $this->controller->js[]='wysija-admin-ajax';
396
+
397
+ $jstrans['gopremium']=__('Go Premium!',WYSIJA);
398
+
399
+ //enqueue all the scripts that have been declared in the controller
400
+ $backloader->parse_js( $this->controller, $pagename, WYSIJA_URL );
401
+
402
+ //this will load automatically existing scripts and stylesheets based on the page and action parameters
403
+ $backloader->load_assets($pagename,WYSIJA_DIR,WYSIJA_URL,$this->controller);
404
+
405
+ //add some translation
406
+ $backloader->localize( $pagename, WYSIJA_DIR, WYSIJA_URL, $this->controller );
407
+
408
+ // add rtl support
409
+ if ( is_rtl() ) {
410
+ wp_enqueue_style('wysija-admin-rtl', WYSIJA_URL.'css/rtl.css',array(),WYSIJA::get_version());
411
+ }
412
+
413
+ }
414
+ $jstrans['newsletters']=__('Newsletters',WYSIJA);
415
+ $jstrans['urlpremium']='admin.php?page=wysija_config#tab-premium';
416
+ if(isset($_REQUEST['page']) && $_REQUEST['page']=='wysija_config'){
417
+ $jstrans['urlpremium']='#tab-premium';
418
+ }
419
+ wp_localize_script('wysija-admin', 'wysijatrans', $jstrans);
420
+ }
421
+
422
+ /**
423
+ * code only executed in the page or post in admin
424
+ */
425
+ function addCodeToPagePost(){
426
+
427
+ //code to add external buttons to the tmce only if the user has the rights to add the forms
428
+ if(get_user_option('rich_editing') == 'true') {
429
+ add_filter("mce_external_plugins", array($this,"addRichPlugin"));
430
+ add_filter('mce_buttons', array($this,'addRichButton1'),999);
431
+ $myStyleUrl = "../../plugins/wysija-newsletters/css/tmce/style.css";
432
+ add_editor_style($myStyleUrl);
433
+ //add_filter('tiny_mce_before_init', array($this,'TMCEinnercss'),12 );
434
+ wp_enqueue_style('custom_TMCE_admin_css', WYSIJA_URL.'css/tmce/panelbtns.css');
435
+ wp_print_styles('custom_TMCE_admin_css');
436
+
437
+ }
438
+ }
439
+
440
+ function addRichPlugin($plugin_array) {
441
+ global $wp_version;
442
+
443
+ if ( version_compare( $wp_version, '3.9', '<' ) ){
444
+ $suffix = '';
445
+ } else {
446
+ $suffix = '_39';
447
+ }
448
+
449
+ $plugin_array['wysija_register'] = WYSIJA_URL.'mce/wysija_register/editor_plugin'.$suffix.'.js';
450
+ //$plugin_array['wysija_subscribers'] = WYSIJA_URL.'mce/wysija_subscribers/editor_plugin.js';
451
+
452
+ return $plugin_array;
453
+ }
454
+
455
+ function addRichButton1($buttons) {
456
+ $newButtons=array();
457
+ foreach($buttons as $value) $newButtons[]=$value;
458
+ //array_push($newButtons, "|", "styleselect");
459
+ array_push($newButtons, '|', 'wysija_register');
460
+ //array_push($newButtons, "|", "wysija_links");
461
+ //array_push($newButtons, '|', 'wysija_subscribers');
462
+ return $newButtons;
463
+ }
464
+
465
+ function admin_footer_text($text) {
466
+ $screen = get_current_screen();
467
+ if (strpos($screen->base, 'wysija')===false)
468
+ return $text;
469
+
470
+ return
471
+ "<a target='_blank' href='http://support.mailpoet.com/feedback/?utm_source=wpadmin&utm_campaign=contact_footer'>" . __( 'Contact Support', WYSIJA ) . "</a>" .
472
+ " | " .
473
+ str_replace(
474
+ array('[stars]','[link]','[/link]'),
475
+ array('<a target="_blank" href="http://clicky.me/wp-reviews" >&#9733;&#9733;&#9733;&#9733;&#9733;</a>','<a target="_blank" href="http://clicky.me/wp-reviews" >','</a>'),
476
+ __('Add your [stars] on [link]wordpress.org[/link] and keep this plugin essentially free.',WYSIJA)
477
+ ) .
478
+ "";
479
+ }
480
+
481
+ function update_footer($text){
482
+ $screen = get_current_screen();
483
+ if (strpos($screen->base, 'wysija')===false)
484
+ return $text;
485
+
486
+ $title = esc_attr((WYSIJA::is_beta()?__('Revert to stable',WYSIJA):__('Switch to beta',WYSIJA)));
487
+ $warn_message = esc_attr((WYSIJA::is_beta()?__('Confirm going back to the stable version?',WYSIJA):__('Great! But the beta version might be buggy, and we expect you to report any bugs. Confirm to installing the Beta version?',WYSIJA)));
488
+
489
+ $args = array(
490
+ 'page' => 'wysija_config',
491
+ 'action' => 'packager-switch',
492
+ '_wpnonce' => wp_create_nonce('packager-switch'),
493
+ );
494
+ if (WYSIJA::is_beta())
495
+ $args["stable"] = 1;
496
+
497
+ $switch_link = esc_attr(add_query_arg($args, admin_url('admin.php')));
498
+
499
+ $switch_text = esc_attr((WYSIJA::is_beta()?__('Revert to Stable', WYSIJA):__('Switch to Beta', WYSIJA)));
500
+
501
+ $version_link = esc_url(add_query_arg(
502
+ array(
503
+ 'page' => 'wysija_campaigns',
504
+ 'action' => 'whats_new',
505
+ ),
506
+ admin_url('admin.php')
507
+ ));
508
+
509
+ $premium = false;
510
+ if (is_plugin_active('wysija-newsletters-premium/index.php')){
511
+ $premium = WYSIJA::get_version('wysija-newsletters-premium/index.php');
512
+ }
513
+
514
+ return
515
+ ((is_multisite() && WYSIJA::current_user_can('manage_network')) || (!is_multisite() && WYSIJA::current_user_can('switch_themes'))?
516
+ "<span class='wrap'>" .
517
+ "<a id='switch_to_package' href='{$switch_link}' title='{$title}' data-warn='{$warn_message}' class='add-new-h2'>{$switch_text}</a>" .
518
+ (WYSIJA::is_beta()?"<a target='_blank' class='add-new-h2' href='http://support.mailpoet.com/feedback/?utm_source=wpadmin&utm_campaign=contact_beta'>" . __( 'Report Bugs', WYSIJA ) . "</a>":"") .
519
+ "</span>"
520
+ :"") .
521
+ "</p>" .
522
+ "<p class='alignright'>" .
523
+ __("MailPoet Version", WYSIJA) . ": <a href='{$version_link}'>" . esc_attr(WYSIJA::get_version()) . "</a>" .
524
+ ($premium?
525
+ " | " .
526
+ __("Premium", WYSIJA) . ": " . esc_attr($premium) . "</a>"
527
+ :"");
528
+ }
529
+ }
helpers/backloader.php ADDED
@@ -0,0 +1,406 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined( 'WYSIJA' ) or die( 'Restricted access' );
3
+
4
+ /**
5
+ * class managing the admin vital part to integrate wordpress
6
+ */
7
+ class WYSIJA_help_backloader extends WYSIJA_help{
8
+
9
+ var $jsVariables = '';
10
+
11
+ function WYSIJA_help_backloader(){
12
+
13
+ parent::WYSIJA_help();
14
+
15
+ }
16
+ /**
17
+ *
18
+ * @param type $controller
19
+ */
20
+ function init( &$controller ) {
21
+ wp_enqueue_style( 'wysija-admin-css', WYSIJA_URL . 'css/admin.css', array(), WYSIJA::get_version() );
22
+ wp_enqueue_script( 'wysija-admin', WYSIJA_URL . 'js/admin.js', array( 'jquery' ), true, WYSIJA::get_version() );
23
+
24
+ /* default script on all wysija interfaces in admin */
25
+ wp_enqueue_script( 'wysija-admin-if ', WYSIJA_URL . 'js/admin-wysija.js', array( 'jquery' ), WYSIJA::get_version() );
26
+
27
+ // TO IMPROVE: This has NOTHING TO DO HERE. It has to be moved to the subscribers controller
28
+ if ( ! $controller->jsTrans ) {
29
+ $controller->jsTrans['selecmiss'] = __( 'Please make a selection first!', WYSIJA );
30
+ $controller->jsTrans['suredelete'] = __( 'Deleting a list will not delete any subscribers.', WYSIJA );
31
+ }
32
+
33
+ $controller->jsTrans['sure_to_switch_package'] = __( 'Do you want to install that version?', WYSIJA );
34
+ $controller->js[] = 'wysija-admin-ajax';
35
+ $controller->js[] = 'thickbox';
36
+ wp_enqueue_style( 'thickbox' );
37
+ }
38
+
39
+ /**
40
+ * help to automatically loads scripts and stylesheets based on the request
41
+ * @param type $pagename
42
+ * @param type $dirname
43
+ * @param type $urlname
44
+ * @param type $controller
45
+ * @param type $extension
46
+ * @return type
47
+ */
48
+ function load_assets( $pagename, $dirname, $urlname, &$controller, $extension = 'newsletter' ) {
49
+
50
+ if ( isset( $_REQUEST['action'] ) ) {
51
+ $action = $_REQUEST['action'];
52
+
53
+ //add form validators script for add and edit
54
+ if ( ( $action == 'edit' || $action == 'add' ) && is_object( $controller ) ) {
55
+ $controller->js[] = 'wysija-validator';
56
+ }
57
+ } else {
58
+ $action = 'default';
59
+
60
+ //load the listing script
61
+ if ( $pagename != 'config' ) {
62
+ wp_enqueue_script( 'wysija-admin-list' );
63
+ }
64
+ }
65
+ //check for files based on this combinations of parameters pagename or pagename and action
66
+ $possibleParameters = array( array( $pagename ), array( $pagename, $action ) );
67
+ $enqueueFileTypes = array( 'wp_enqueue_script' => array( 'js' => 'js', 'php' => 'js' ), 'wp_enqueue_style' => array( 'css' => 'css' ) );
68
+
69
+ // Files that we have, don't use file_exists if we know which files we have
70
+ $files = (object) array(
71
+ 'css' => array(
72
+ 'add-ons',
73
+ 'admin-campaigns-articles',
74
+ 'admin-campaigns-autopost',
75
+ 'admin-campaigns-bookmarks',
76
+ 'admin-campaigns-dividers',
77
+ 'admin-campaigns-editDetails',
78
+ 'admin-campaigns-editTemplate',
79
+ 'admin-campaigns-medias',
80
+ 'admin-campaigns-themes',
81
+ 'admin-campaigns-viewstats',
82
+ 'admin-campaigns-welcome_new',
83
+ 'admin-campaigns',
84
+ 'admin-global',
85
+ 'admin-premium',
86
+ 'admin-statistics',
87
+ 'admin-subscribers-addlist',
88
+ 'admin-subscribers-edit',
89
+ 'admin-subscribers-export',
90
+ 'admin-subscribers-exportlist',
91
+ 'admin-subscribers-import',
92
+ 'admin-subscribers-importmatch',
93
+ 'admin-subscribers-lists',
94
+ 'admin-widget',
95
+ 'admin-config',
96
+ 'admin-config-form_widget_settings',
97
+ ),
98
+ 'js' => array(
99
+ 'admin-ajax-proto',
100
+ 'admin-ajax',
101
+ 'admin-campaigns-articles',
102
+ 'admin-campaigns-autopost',
103
+ 'admin-campaigns-bookmarks',
104
+ 'admin-campaigns-default',
105
+ 'admin-campaigns-dividers',
106
+ 'admin-campaigns-edit',
107
+ 'admin-campaigns-editAutonl',
108
+ 'admin-campaigns-editDetails',
109
+ 'admin-campaigns-editTemplate',
110
+ 'admin-campaigns-image_data',
111
+ 'admin-campaigns-medias',
112
+ 'admin-campaigns-themes',
113
+ 'admin-campaigns-viewstats',
114
+ 'admin-campaigns-welcome_new',
115
+ 'admin-config-form_widget_settings',
116
+ 'admin-config-settings',
117
+ 'admin-global',
118
+ 'admin-listing',
119
+ 'admin-statistics-filter',
120
+ 'admin-statistics',
121
+ 'admin-subscribers-edit-manual',
122
+ 'admin-subscribers-export',
123
+ 'admin-subscribers-import',
124
+ 'admin-subscribers-importmatch',
125
+ 'admin-subscribers',
126
+ 'admin-tmce',
127
+ 'admin-wysija-global',
128
+ 'admin-wysija',
129
+ )
130
+ );
131
+
132
+ foreach ( $possibleParameters as $params ) {
133
+ foreach ( $enqueueFileTypes as $method => $types ) {
134
+ foreach ( $types as $file_type => $file_ext ){
135
+ $file_slug = 'admin-' . implode( '-', $params );
136
+ if ( in_array( $file_slug, $files->{ $file_ext } ) ) {
137
+ $file_id = "wysija-autoinc-{$extension}-{$file_slug}-{$file_ext}";
138
+ $file_url = "{$urlname}{$file_ext}/{$file_slug}.{$file_ext}";
139
+ call_user_func_array( $method, array( $file_id, $file_url, array(), WYSIJA::get_version() ) );
140
+ }
141
+ }
142
+ }
143
+ }
144
+
145
+ return true;
146
+ }
147
+
148
+ /**
149
+ * enqueue and load dif ferent scripts and style based on one script being requested in the controller
150
+ * @param type $controller
151
+ * @param type $pagename
152
+ * @param string $urlbase
153
+ */
154
+ function parse_js( &$controller, $pagename, $urlbase = WYSIJA_URL ){
155
+
156
+ // find out the name of the plugin based on the urlbase parameter
157
+ $plugin = substr( strrchr( substr( $urlbase, 0, strlen( $urlbase ) - 1 ), '/' ), 1 );
158
+
159
+ /* enqueue all the scripts that have been declared in the controller */
160
+ if ( $controller->js ) {
161
+ foreach ( $controller->js as $kjs => $js ) {
162
+ switch ( $js ) {
163
+ case 'jquery-ui-tabs':
164
+ wp_enqueue_script( $js );
165
+ wp_enqueue_style( 'wysija-tabs-css', WYSIJA_URL . 'css/smoothness/jquery-ui-1.8.20.custom.css', array(), WYSIJA::get_version() );
166
+ break;
167
+
168
+ case 'wysija-validator':
169
+ wp_enqueue_script( 'wysija-validator-lang' );
170
+ wp_enqueue_script( $js );
171
+ wp_enqueue_script( 'wysija-form' );
172
+ wp_enqueue_style( 'validate-engine-css' );
173
+ break;
174
+
175
+ case 'wysija-admin-ajax':
176
+ if ( $plugin != 'wysija-newsletters' ){
177
+ $ajaxvarname = $plugin;
178
+ } else {
179
+ $ajaxvarname = 'wysija';
180
+ }
181
+
182
+ $dataajaxxx = array(
183
+ 'action' => 'wysija_ajax',
184
+ 'controller' => $pagename,
185
+ 'wysijaplugin' => $plugin,
186
+ 'dataType' => 'json',
187
+ 'ajaxurl' => admin_url( 'admin-ajax.php', 'relative' ),
188
+ // 'ajaxurl' => plugins_url( 'wysija-newsletters' ).'/core/ajax.php',
189
+ 'pluginurl' => plugins_url( 'wysija-newsletters' ),
190
+ 'loadingTrans' => __( 'Loading...', WYSIJA )
191
+ );
192
+
193
+ if ( is_user_logged_in() ){
194
+ $dataajaxxx['adminurl'] = admin_url( 'admin.php' );
195
+ }
196
+
197
+ wp_localize_script( 'wysija-admin-ajax', $ajaxvarname.'AJAX',$dataajaxxx );
198
+ wp_enqueue_script( 'jquery-ui-dialog' );
199
+ wp_enqueue_script( $js );
200
+ wp_enqueue_style( 'wysija-tabs-css', WYSIJA_URL . 'css/smoothness/jquery-ui-1.8.20.custom.css', array(), WYSIJA::get_version() );
201
+ break;
202
+
203
+ case 'wysija-admin-ajax-proto':
204
+ wp_enqueue_script( $js );
205
+ break;
206
+
207
+ case 'wysija-edit-autonl':
208
+ wp_enqueue_script( 'wysija-edit-autonl', WYSIJA_URL . 'js/admin-campaigns-editAutonl.js', array( 'jquery' ), WYSIJA::get_version() );
209
+ break;
210
+
211
+ case 'wysija-scriptaculous':
212
+ // include prototypeJS + scriptaculous & addons
213
+ wp_enqueue_script( 'wysija-prototype', WYSIJA_URL . 'js/prototype/prototype.js', array(), WYSIJA::get_version() );
214
+ wp_enqueue_script( 'wysija-proto-scriptaculous', WYSIJA_URL . 'js/prototype/scriptaculous.js', array( 'wysija-prototype' ), WYSIJA::get_version() );
215
+ wp_enqueue_script( 'wysija-proto-dragdrop', WYSIJA_URL . 'js/prototype/dragdrop.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
216
+ wp_enqueue_script( 'wysija-proto-controls', WYSIJA_URL . 'js/prototype/controls.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
217
+ wp_enqueue_script( 'wysija-proto-slider', WYSIJA_URL . 'js/prototype/slider.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
218
+ break;
219
+
220
+ case 'mailpoet-select2':
221
+
222
+ wp_enqueue_script( 'mailpoet-select2', WYSIJA_URL . 'js/select2/select2.min.js', array( 'jquery' ), WYSIJA::get_version() );
223
+ wp_enqueue_script( 'mailpoet-select2-l10n', WYSIJA_URL . 'js/select2/select2-l10n.js', array( 'mailpoet-select2', 'underscore' ), WYSIJA::get_version() );
224
+
225
+ wp_enqueue_style( 'mailpoet-select2', WYSIJA_URL . 'css/select2/select2.css', array(), WYSIJA::get_version() );
226
+
227
+ wp_localize_script(
228
+ 'mailpoet-select2-l10n',
229
+ 'mailpoet_l10n_select2',
230
+ array(
231
+ 'noMatches' => __( 'No Results were found', WYSIJA ),
232
+ 'inputTooShort' => __( 'Please enter <%= chars %> more character<%= plural %>', WYSIJA ),
233
+ 'inputTooLong' => __( 'Please delete <%= chars %> character<%= plural %>', WYSIJA ),
234
+ 'selectionTooBig' => __( 'You can only select <%= chars %> item<%= plural %>', WYSIJA ),
235
+ 'loadMore' => __( 'Loading more Results...', WYSIJA ),
236
+ 'searching' => __( 'Searching...', WYSIJA ),
237
+ )
238
+ );
239
+
240
+ break;
241
+
242
+ case 'wysija-form-editor':
243
+ wp_enqueue_script( 'wysija-prototype', WYSIJA_URL . 'js/prototype/prototype.js', array(), WYSIJA::get_version() );
244
+ wp_enqueue_script( 'wysija-proto-scriptaculous', WYSIJA_URL . 'js/prototype/scriptaculous.js', array( 'wysija-prototype' ), WYSIJA::get_version() );
245
+ wp_enqueue_script( 'wysija-proto-dragdrop', WYSIJA_URL . 'js/prototype/dragdrop.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
246
+ wp_enqueue_script( 'wysija-proto-controls', WYSIJA_URL . 'js/prototype/controls.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
247
+ wp_enqueue_script( 'wysija-proto-slider', WYSIJA_URL . 'js/prototype/slider.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
248
+
249
+ // include form editor
250
+ wp_enqueue_script( $js, WYSIJA_URL . 'js/' . $js . '.js', array(), WYSIJA::get_version() );
251
+
252
+ /* MailPoet form editor i18n */
253
+ wp_localize_script( 'wysija-form-editor', 'Wysija_i18n', $controller->jsTrans );
254
+
255
+ // form editor css
256
+ wp_enqueue_style( 'wysija-form-editor-css', WYSIJA_URL . 'css/wysija-form-editor.css', array(), WYSIJA::get_version() );
257
+ break;
258
+
259
+ case 'wysija-amcharts':
260
+ // MailPoet chart
261
+ wp_enqueue_script( 'amcharts', WYSIJA_URL . 'js/amcharts/amcharts.js', array(), WYSIJA::get_version() );
262
+ wp_enqueue_script( 'wysija-amcharts', WYSIJA_URL . 'js/wysija-charts.js', array(), WYSIJA::get_version() );
263
+ break;
264
+
265
+ case 'wysija-editor':
266
+
267
+ wp_enqueue_script( 'wysija-prototype', WYSIJA_URL . 'js/prototype/prototype.js', array(), WYSIJA::get_version() );
268
+ wp_deregister_script( 'thickbox' );
269
+
270
+ wp_register_script( 'thickbox', WYSIJA_URL . 'js/thickbox/thickbox.js', array( 'jquery' ), WYSIJA::get_version() );
271
+
272
+ wp_localize_script(
273
+ 'thickbox',
274
+ 'thickboxL10n',
275
+ array(
276
+ 'next' => __( 'Next &gt;' ),
277
+ 'prev' => __( '&lt; Prev' ),
278
+ 'image' => __( 'Image' ),
279
+ 'of' => __( 'of' ),
280
+ 'close' => __( 'Close' ),
281
+ 'noif rames' => __( 'This feature requires inline frames. You have iframes disabled or your browser does not support them.' ),
282
+ 'l10n_print_after' => 'try{convertEntities( thickboxL10n );}catch( e ){};',
283
+ )
284
+ );
285
+
286
+ wp_enqueue_script( 'wysija-proto-scriptaculous', WYSIJA_URL . 'js/prototype/scriptaculous.js' , array( 'wysija-prototype' ), WYSIJA::get_version() );
287
+ wp_enqueue_script( 'wysija-proto-dragdrop', WYSIJA_URL . 'js/prototype/dragdrop.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
288
+ wp_enqueue_script( 'wysija-proto-controls', WYSIJA_URL . 'js/prototype/controls.js', array( 'wysija-proto-scriptaculous' ), WYSIJA::get_version() );
289
+ wp_enqueue_script( 'wysija-timer', WYSIJA_URL . 'js/timer.js', array(), WYSIJA::get_version() );
290
+ wp_enqueue_script( $js, WYSIJA_URL . 'js/' . $js . '.js', array(), WYSIJA::get_version() );
291
+ wp_enqueue_script( 'wysija-konami', WYSIJA_URL . 'js/konami.js', array(), WYSIJA::get_version() );
292
+ wp_enqueue_script( 'wysija-tinymce', WYSIJA_URL . 'js/tinymce/tiny_mce.js', array(), WYSIJA::get_version() );
293
+
294
+ wp_enqueue_script( 'wysija-tinymce-init', WYSIJA_URL . 'js/tinymce_init.js', array(), WYSIJA::get_version() );
295
+ wp_enqueue_style( 'wysija-editor-css', WYSIJA_URL . 'css/wysija-editor.css', array(), WYSIJA::get_version() );
296
+ wp_enqueue_script( 'wysija-colorpicker', WYSIJA_URL . 'js/excolor/jquery.modcoder.excolor.js', array(), WYSIJA::get_version() );
297
+
298
+ if ( version_compare( $GLOBALS['wp_version'], '3.9', '>=' ) ){
299
+ wp_enqueue_style( 'mailpoet-tinymce', WYSIJA_URL . 'css/tmce/editor.css', array(), WYSIJA::get_version() );
300
+ }
301
+
302
+ /* MailPoet editor i18n */
303
+ wp_localize_script( 'wysija-editor', 'Wysija_i18n', $controller->jsTrans );
304
+ break;
305
+
306
+ case 'wysija-colorpicker':
307
+ wp_enqueue_script( 'wysija-colorpicker', WYSIJA_URL . 'js/excolor/jquery.modcoder.excolor.js', array(), WYSIJA::get_version() );
308
+ break;
309
+
310
+ case 'wysija-tooltip':
311
+ wp_enqueue_script( 'mailpoet.tooltip', WYSIJA_URL . 'js/vendor/bootstrap.tooltip.js', array( 'jquery' ), WYSIJA::get_version(), true );
312
+ wp_enqueue_style( 'mailpoet.tooltip', WYSIJA_URL . 'css/vendor/bootstrap.tooltip.css', array(), WYSIJA::get_version(), 'screen' );
313
+ break;
314
+
315
+ default:
316
+ if ( is_string( $kjs ) ) {
317
+ // check if there's a trailing slash in the urlbase
318
+ if ( substr( $urlbase, -1 ) !== '/' ){
319
+ $urlbase .= '/';
320
+ }
321
+ // check if there's already an extension specif ied for the file
322
+ if ( substr( $urlbase, -3 ) !== '.js' ){
323
+ $js .= '.js';
324
+ }
325
+ // enqueue script
326
+
327
+ wp_enqueue_script( $kjs, $urlbase . 'js/' . $js, array(), WYSIJA::get_version() );
328
+ } else {
329
+ wp_enqueue_script( $js );
330
+ }
331
+ }
332
+ }
333
+ }
334
+ }
335
+
336
+ /**
337
+ * add some js defined variable per script
338
+ * @param type $pagename
339
+ * @param type $dirname
340
+ * @param type $urlname
341
+ * @param type $controller
342
+ * @param type $extension
343
+ */
344
+ function localize( $pagename, $dirname, $urlname, &$controller, $extension = 'newsletter' ){
345
+ if ( $controller->jsLoc ){
346
+ foreach ( $controller->jsLoc as $key => $value ){
347
+ foreach ( $value as $kf => $local ){
348
+
349
+ //this function accepts multidimensional array some version like wp3.2.1 couldn't do that
350
+ $this->localizeme( $key, $kf, $local );
351
+ }
352
+ }
353
+ }
354
+ }
355
+
356
+ /**
357
+ * multidimensional array are possible here
358
+ * @param type $handle
359
+ * @param type $object_name
360
+ * @param type $l10n
361
+ */
362
+ function localizeme( $handle, $object_name, $l10n ) {
363
+
364
+ foreach ( ( array ) $l10n as $key => $value ) {
365
+ if ( ! is_scalar( $value ) ){
366
+ continue;
367
+ }
368
+ $l10n[$key] = html_entity_decode( (string) $value, ENT_QUOTES, 'UTF-8' );
369
+ }
370
+
371
+ $this->jsVariables .= "var $object_name = " . json_encode( $l10n ) . ';';
372
+ add_action( 'admin_head', array( $this, 'localize_print' ) );
373
+ }
374
+
375
+ /**
376
+ * load the variables in the html
377
+ */
378
+ function localize_print(){
379
+ echo "<script type='text/javascript' id='mailpoet-localized'>\n"; // CDATA and type='text/javascript' is not needed for HTML 5
380
+ echo "/* <![CDATA[ */\n";
381
+ echo esc_attr( '' ) . $this->jsVariables . "\n"; // To comply with PHP Code Sniffer WordPress Standards before we "hack" the echo
382
+ echo "/* ]]> */\n";
383
+ echo "</script>\n";
384
+ }
385
+
386
+
387
+
388
+ /**
389
+ * this is for backward compatibility and avoid blank screen on older version of the premium plugin
390
+ */
391
+ function loadScriptsStyles( $pagename, $dirname, $urlname, &$controller, $extension = 'newsletter' ) {
392
+ return $this->load_assets( $pagename, $dirname, $urlname, $controller, $extension );
393
+ }
394
+
395
+ /**
396
+ * this is for backward compatibility and avoid blank screen on older version of the premium plugin
397
+ */
398
+ function initLoad( &$controller ){
399
+ return $this->init( $controller );
400
+ }
401
+
402
+ function jsParse( &$controller, $pagename, $urlbase = WYSIJA_URL ){
403
+ $this->parse_js( $controller, $pagename, $urlbase );
404
+ }
405
+
406
+ }
helpers/bookmarks.php ADDED
@@ -0,0 +1,113 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_bookmarks extends WYSIJA_object {
4
+
5
+ function WYSIJA_help_bookmarks() {
6
+ }
7
+
8
+ /**
9
+ * Get all bookmarks based on size
10
+ * @param string $size
11
+ * @return array
12
+ */
13
+ function getAll($size = 'medium', $theme = 'default') {
14
+ $fileHelper = WYSIJA::get('file', 'helper');
15
+ $dirHandle = $fileHelper->exists('bookmarks'.DS.$size);
16
+
17
+ if($dirHandle['result'] === FALSE) {
18
+ return array();
19
+ } else {
20
+ $bookmarks = array();
21
+
22
+ // if size is medium and the current theme is not default, load theme's bookmarks
23
+ if($size === 'medium' and $theme !== 'default') {
24
+ $themeIcons = $this->getAllByTheme($theme, 'url');
25
+ if(!empty($themeIcons)) {
26
+ $bookmarks['00'] = $themeIcons;
27
+ }
28
+ }
29
+
30
+ $sourceDir = $dirHandle['file'];
31
+ $iconsets = scandir($sourceDir);
32
+ foreach($iconsets as $iconset) {
33
+ // loop through each iconset
34
+ if(in_array($iconset, array('.', '..', '.DS_Store', 'Thumbs.db')) === FALSE and is_dir($sourceDir.DS.$iconset)) {
35
+
36
+ // get all icons from current iconset
37
+ $icons = scandir($sourceDir.DS.$iconset);
38
+ foreach($icons as $icon) {
39
+ if(in_array($icon, array('.', '..', '.DS_Store', 'Thumbs.db')) === FALSE and strrpos($icon, '.txt') === FALSE) {
40
+ $info = pathinfo($sourceDir.DS.$iconset.DS.$icon);
41
+ $bookmarks[$iconset][basename($icon, '.'.$info['extension'])] = $fileHelper->url($icon, 'bookmarks'.DS.$size.DS.$iconset);
42
+ }
43
+ }
44
+ }
45
+ }
46
+ return $bookmarks;
47
+ }
48
+ }
49
+
50
+ /**
51
+ * Get all bookmarks based on size for a given iconset
52
+ * @param string $size
53
+ * @param string $iconset
54
+ * @return array
55
+ */
56
+ function getAllByIconset($size = 'medium', $iconset)
57
+ {
58
+ $fileHelper = WYSIJA::get('file', 'helper');
59
+ $dirHandle = $fileHelper->exists('bookmarks'.DS.$size.DS.$iconset);
60
+
61
+ if($dirHandle['result'] === FALSE) {
62
+ return array();
63
+ } else {
64
+ $bookmarks = array();
65
+ $sourceDir = $dirHandle['file'];
66
+ $icons = scandir($sourceDir);
67
+ foreach($icons as $icon) {
68
+ if(in_array($icon, array('.', '..', '.DS_Store', 'Thumbs.db')) === FALSE and strrpos($icon, '.txt') === FALSE) {
69
+ $info = pathinfo($sourceDir.DS.$icon);
70
+ $dimensions = @getimagesize($sourceDir.DS.$icon);
71
+ $bookmarks[basename($icon, '.'.$info['extension'])] = array(
72
+ 'src' => $fileHelper->url($icon, 'bookmarks/'.$size.'/'.$iconset),
73
+ 'width' => $dimensions[0],
74
+ 'height' => $dimensions[1]
75
+ );
76
+ }
77
+ }
78
+ return $bookmarks;
79
+ }
80
+ }
81
+
82
+ function getAllByTheme($theme, $type = 'all')
83
+ {
84
+ $fileHelper = WYSIJA::get('file', 'helper');
85
+ $dirHandle = $fileHelper->exists('themes'.DS.$theme.DS.'bookmarks');
86
+
87
+ if($dirHandle['result'] === FALSE) {
88
+ return array();
89
+ } else {
90
+ $bookmarks = array();
91
+ $sourceDir = $dirHandle['file'];
92
+ $icons = scandir($sourceDir);
93
+ foreach($icons as $icon) {
94
+ if(in_array($icon, array('.', '..', '.DS_Store', 'Thumbs.db')) === FALSE and strrpos($icon, '.txt') === FALSE) {
95
+
96
+ if($type === 'all') {
97
+ $info = pathinfo($sourceDir.DS.$icon);
98
+ $dimensions = @getimagesize($sourceDir.DS.$icon);
99
+ $bookmarks[basename($icon, '.'.$info['extension'])] = array(
100
+ 'src' => $fileHelper->url($icon, 'themes/'.$theme.'/bookmarks'),
101
+ 'width' => $dimensions[0],
102
+ 'height' => $dimensions[1]
103
+ );
104
+ } else if($type === 'url') {
105
+ $info = pathinfo($sourceDir.DS.$icon);
106
+ $bookmarks[basename($icon, '.'.$info['extension'])] = $fileHelper->url($icon, 'themes/'.$theme.'/bookmarks');
107
+ }
108
+ }
109
+ }
110
+ return $bookmarks;
111
+ }
112
+ }
113
+ }
helpers/bounce.php ADDED
@@ -0,0 +1,1099 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+
4
+ class WYSIJA_help_bounce extends WYSIJA_help {
5
+
6
+ var $report = false;
7
+ var $config;
8
+ var $mailer;
9
+ var $mailbox;
10
+ var $_message;
11
+ var $listsubClass;
12
+ var $subClass;
13
+ var $db;
14
+ var $deletedUsers = array();
15
+ var $unsubscribedUsers = array();
16
+ var $addtolistUsers = array();
17
+ var $bounceMessages = array();
18
+ var $listdetails = array();
19
+ var $usepear = false;
20
+ var $detectEmail = '/[a-z0-9!#$%&\'*+\/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&\'*+\/=?^_`{|}~-]+)*@([a-z0-9\-]+\.)+[a-z0-9]{2,8}/i';
21
+ var $messages = array();
22
+ var $record_ms_bounce = false; // only used in multisite scenario
23
+
24
+ function WYSIJA_help_bounce() {
25
+ $this->config = WYSIJA::get('config', 'model');
26
+ $this->mailer = WYSIJA::get('mailer', 'helper');
27
+ $this->rulesClass = WYSIJA::get('rules', 'helper');
28
+ $this->mailer->report = false;
29
+ $this->subClass = WYSIJA::get('user', 'model'); //acymailing_get('class.subscriber');
30
+ $this->listsubClass = WYSIJA::get('user_list', 'model'); //acymailing_get('class.listsub');
31
+ $this->listsubClass->checkAccess = false;
32
+ $this->listsubClass->sendNotif = false;
33
+ $this->listsubClass->sendConf = false;
34
+ $this->historyClass = WYSIJA::get('user_history', 'model');
35
+ }
36
+
37
+ function init($config = false) {
38
+ if ($config) {
39
+ //unset($this->config->values);
40
+ foreach ($config as $key => $val)
41
+ $this->config->values[$key] = $val;
42
+ }
43
+ if ($this->config->getValue('bounce_connection_method') == 'pear') {
44
+ $this->usepear = true;
45
+ include_once(WYSIJA_INC . 'pear' . DS . 'pop3.php');
46
+ return true;
47
+ }
48
+
49
+ if (extension_loaded('imap') OR function_exists('imap_open'))
50
+ return true;
51
+
52
+ $prefix = (PHP_SHLIB_SUFFIX == 'dll') ? 'php_' : '';
53
+ $EXTENSION = $prefix . 'imap.' . PHP_SHLIB_SUFFIX;
54
+
55
+ if (function_exists('dl')) {
56
+ //We will try to load it on the fly
57
+ $fatalMessage = 'The system tried to load dynamically the ' . $EXTENSION . ' extension';
58
+ $fatalMessage .= '<br/>If you see this message, that means the system could not load this PHP extension';
59
+ $fatalMessage .= '<br/>Please enable the PHP Extension ' . $EXTENSION;
60
+ ob_start();
61
+ echo $fatalMessage;
62
+ //This method could cause a fatal error, but we will still display some messages in that case.
63
+ dl($EXTENSION);
64
+ $warnings = str_replace($fatalMessage, '', ob_get_clean());
65
+ if (extension_loaded('imap') OR function_exists('imap_open'))
66
+ return true;
67
+ }
68
+
69
+ if ($this->report) {
70
+ $this->error('The extension "' . $EXTENSION . '" could not be loaded, please change your PHP configuration to enable it or use the pop3 method without imap extension', true);
71
+ if (!empty($warnings))
72
+ $this->error($warnings, true);
73
+ }
74
+
75
+ return false;
76
+ }
77
+
78
+ function connect() {
79
+ if ($this->usepear)
80
+ return $this->_connectpear();
81
+ return $this->_connectimap();
82
+ }
83
+
84
+ function _connectpear() {
85
+ ob_start();
86
+ $this->mailbox = new Net_POP3();
87
+
88
+ $timeout = $this->config->getValue('bounce_timeout');
89
+ if (!empty($timeout))
90
+ $this->mailbox->setTimeOut($timeout);
91
+
92
+ $port = intval($this->config->getValue('bounce_port', ''));
93
+ if (empty($port))
94
+ $port = '110/pop3/notls';
95
+
96
+ $serverName = $this->config->getValue('bounce_host');
97
+ $secure = $this->config->getValue('bounce_connection_secure', '');
98
+ //We don't add back the ssl:// or tls:// if it's already there
99
+ if (!empty($secure) AND !strpos($serverName, '://'))
100
+ $serverName = $secure . '://' . $serverName;
101
+
102
+ if (!$this->mailbox->connect($serverName, $port)) {
103
+ $warnings = ob_get_clean();
104
+ if ($this->report) {
105
+ $this->error('Error connecting to the server ' . $this->config->getValue('bounce_host') . ' : ' . $port, true);
106
+ return false;
107
+ }
108
+ if (!empty($warnings) AND $this->report)
109
+ $this->error($warnings, true);
110
+ return false;
111
+ }
112
+
113
+ $login = $this->mailbox->login(trim($this->config->getValue('bounce_login')), trim($this->config->getValue('bounce_password')), 'USER');
114
+ if (empty($login) OR isset($login->code)) {
115
+ $warnings = ob_get_clean();
116
+ if ($this->report) {
117
+ $this->error('Identication error ' . $this->config->getValue('bounce_login') . ':' . $this->config->getValue('bounce_password'), true);
118
+ return false;
119
+ }
120
+ if (!empty($warnings) AND $this->report)
121
+ $this->error($warnings, true);
122
+ return false;
123
+ }
124
+
125
+ ob_clean();
126
+
127
+ return true;
128
+ }
129
+
130
+ function _connectimap() {
131
+ ob_start();
132
+ //First we reset the buffer or errors and warnings
133
+ $buff = imap_alerts();
134
+ $buff = imap_errors();
135
+
136
+ $timeout = $this->config->getValue('bounce_timeout');
137
+ if (!empty($timeout))
138
+ imap_timeout(IMAP_OPENTIMEOUT, $timeout);
139
+
140
+ $port = $this->config->getValue('bounce_port', '');
141
+ $secure = $this->config->getValue('bounce_connection_secure', '');
142
+ $protocol = $this->config->getValue('bounce_connection_method', '');
143
+ $serverName = '{' . $this->config->getValue('bounce_host');
144
+ if (empty($port)) {
145
+ if ($secure == 'ssl' && $protocol == 'imap')
146
+ $port = '993';
147
+ elseif ($protocol == 'imap')
148
+ $port = '143';
149
+ elseif ($protocol == 'pop3')
150
+ $port = '110';
151
+ }
152
+
153
+
154
+ if (!empty($port))
155
+ $serverName .= ':' . $port;
156
+ //Add the secure protocol (TLS or SSL)
157
+ if (!empty($secure))
158
+ $serverName .= '/' . $secure;
159
+ if ($this->config->getValue('bounce_selfsigned', false))
160
+ $serverName .= '/novalidate-cert';
161
+ //Add the method (imap by default) ex : pop3
162
+ if (!empty($protocol))
163
+ $serverName .='/service=' . $protocol;
164
+ $serverName .= '}';
165
+ $this->mailbox = imap_open($serverName, trim($this->config->getValue('bounce_login')), trim($this->config->getValue('bounce_password')));
166
+ $warnings = ob_get_clean();
167
+
168
+ if ($this->report) {
169
+ if (!$this->mailbox) {
170
+ $this->error('Error connecting to ' . $serverName, true);
171
+ }
172
+ if (!empty($warnings)) {
173
+ $this->error($warnings, true);
174
+ }
175
+ }
176
+
177
+
178
+ return $this->mailbox ? true : false;
179
+ }
180
+
181
+ function getNBMessages() {
182
+ if ($this->usepear) {
183
+ $this->nbMessages = $this->mailbox->numMsg();
184
+ } else {
185
+ $this->nbMessages = imap_num_msg($this->mailbox);
186
+ }
187
+
188
+ return $this->nbMessages;
189
+ }
190
+
191
+ function getMessage($msgNB) {
192
+ if ($this->usepear) {
193
+ $message = null;
194
+ $message->headerString = $this->mailbox->getRawHeaders($msgNB);
195
+ if (empty($message->headerString))
196
+ return false;
197
+ }else {
198
+ $message = imap_headerinfo($this->mailbox, $msgNB);
199
+ }
200
+
201
+ return $message;
202
+ }
203
+
204
+ function deleteMessage($msgNB) {
205
+ if ($this->usepear) {
206
+ $this->mailbox->deleteMsg($msgNB);
207
+ } else {
208
+ imap_delete($this->mailbox, $msgNB);
209
+ imap_expunge($this->mailbox);
210
+ }
211
+ }
212
+
213
+ function close() {
214
+ if ($this->usepear) {
215
+ $this->mailbox->disconnect();
216
+ } else {
217
+ imap_close($this->mailbox);
218
+ }
219
+ }
220
+
221
+ function decodeMessage() {
222
+ if ($this->usepear) {
223
+ return $this->_decodeMessagepear();
224
+ } else {
225
+ return $this->_decodeMessageimap();
226
+ }
227
+ }
228
+
229
+ function _decodeMessagepear() {
230
+ $this->_message->headerinfo = $this->mailbox->getParsedHeaders($this->_message->messageNB);
231
+
232
+ if (empty($this->_message->headerinfo['subject']))
233
+ return false;
234
+ $this->_message->text = '';
235
+ $this->_message->html = $this->mailbox->getBody($this->_message->messageNB);
236
+ $this->_message->subject = $this->_decodeHeader($this->_message->headerinfo['subject']);
237
+ $this->_message->header->sender_email = @$this->_message->headerinfo['return-path'];
238
+ if (is_array($this->_message->header->sender_email))
239
+ $this->_message->header->sender_email = reset($this->_message->header->sender_email);
240
+ if (preg_match($this->detectEmail, $this->_message->header->sender_email, $results)) {
241
+ $this->_message->header->sender_email = $results[0];
242
+ }
243
+ $this->_message->header->sender_name = strip_tags(@$this->_message->headerinfo['from']);
244
+ $this->_message->header->reply_to_email = $this->_message->header->sender_email;
245
+ $this->_message->header->reply_to_name = $this->_message->header->sender_name;
246
+ $this->_message->header->from_email = $this->_message->header->sender_email;
247
+ $this->_message->header->from_name = $this->_message->header->sender_name;
248
+
249
+ return true;
250
+ }
251
+
252
+ function _decodeMessageimap() {
253
+ $this->_message->structure = imap_fetchstructure($this->mailbox, $this->_message->messageNB);
254
+ if (empty($this->_message->structure))
255
+ return false;
256
+ $this->_message->headerinfo = imap_fetchheader($this->mailbox, $this->_message->messageNB);
257
+ $this->_message->html = '';
258
+ $this->_message->text = '';
259
+
260
+ //Multipart message : type == 1
261
+ if ($this->_message->structure->type == 1) {
262
+ $this->_message->contentType = 2;
263
+ $allParts = $this->_explodeBody($this->_message->structure);
264
+
265
+ $this->_message->text = '';
266
+ foreach ($allParts as $num => $onePart) {
267
+ $charset = $this->_getMailParam($onePart, 'charset');
268
+ if ($onePart->subtype == 'HTML') {
269
+ $this->_message->html = $this->_decodeContent(imap_fetchbody($this->mailbox, $this->_message->messageNB, $num), $onePart);
270
+ } else {
271
+ $this->_message->text .= $this->_decodeContent(imap_fetchbody($this->mailbox, $this->_message->messageNB, $num), $onePart) . "\n\n- - -\n\n";
272
+ }
273
+ }
274
+ } else {
275
+ $charset = $this->_getMailParam($this->_message->structure, 'charset');
276
+ if ($this->_message->structure->subtype == 'HTML') {
277
+ $this->_message->contentType = 1;
278
+ $this->_message->html = $this->_decodeContent(imap_body($this->mailbox, $this->_message->messageNB), $this->_message->structure);
279
+ } else {
280
+ $this->_message->contentType = 0;
281
+ $this->_message->text = $this->_decodeContent(imap_body($this->mailbox, $this->_message->messageNB), $this->_message->structure);
282
+ }
283
+ }
284
+
285
+ //Decode the subject
286
+ $this->_message->subject = $this->_decodeHeader($this->_message->subject);
287
+
288
+ $this->_decodeAddressimap('sender');
289
+ $this->_decodeAddressimap('from');
290
+ $this->_decodeAddressimap('reply_to');
291
+ $this->_decodeAddressimap('to');
292
+ return true;
293
+ }
294
+
295
+ function handleMessages() {
296
+ $model_list = WYSIJA::get('list', 'model');
297
+
298
+ $listdetails = $model_list->getRows(array('name', 'list_id'));
299
+
300
+ foreach ($listdetails as $listinfo) {
301
+ $this->listdetails[$listinfo['list_id']] = $listinfo['name'];
302
+ }
303
+ $maxMessages = min($this->nbMessages, $this->config->getValue('bounce_max', 100));
304
+ if (empty($maxMessages))
305
+ $maxMessages = $this->nbMessages;
306
+
307
+ // we need a report when we are handling the bounce manually through the settings
308
+ if ($this->report) {
309
+ // If we display informations, we directy flush so that we can display in real time!
310
+ if (!headers_sent() AND ob_get_level() > 0) {
311
+ ob_end_flush();
312
+ }
313
+
314
+ // We prepare the area where we will add informations...
315
+ $disp = '<html><head><meta http-equiv="Content-Type" content="text/html;charset=utf-8" />';
316
+ $disp .= '<title>' . addslashes(__('Bounce Handling', WYSIJA)) . '</title>';
317
+ $disp .= '<style>body{font-size:12px;font-family: Arial,Helvetica,sans-serif;} strong{color: black;}</style></head><body>';
318
+ $disp .= "<div style='position:relative; top:3px;left:3px;'>";
319
+ $disp .= __("Bounce Handling", WYSIJA);
320
+ $disp .= ': <span id="counter"/>0</span> / ' . $maxMessages;
321
+ $disp .= '</div>';
322
+ $disp .= '<br/>';
323
+ $disp .= '<script type="text/javascript" language="javascript">';
324
+ $disp .= 'var mycounter = document.getElementById("counter");';
325
+ $disp .= 'function setCounter(val){ mycounter.innerHTML=val;}';
326
+ $disp .= '</script>';
327
+ echo $disp;
328
+ if (function_exists('ob_flush'))
329
+ @ob_flush();
330
+ @flush();
331
+ }
332
+
333
+ //We load all published the rules
334
+ $rules = $this->rulesClass->getRules();
335
+
336
+ $msgNB = $maxMessages;
337
+ $listClass = WYSIJA::get('list', 'model');
338
+ $this->allLists = $listClass->getRows();
339
+
340
+ while (($msgNB > 0) && ($this->_message = $this->getMessage($msgNB))) {
341
+ if ($this->report) {
342
+ echo '<script type="text/javascript" language="javascript">setCounter(' . ($maxMessages - $msgNB + 1) . ')</script>';
343
+ if (function_exists('ob_flush'))
344
+ @ob_flush();
345
+ @flush();
346
+ }
347
+ $this->_message->messageNB = $msgNB;
348
+ $this->decodeMessage();
349
+
350
+ $msgNB--;
351
+ if (empty($this->_message->subject))
352
+ continue;
353
+
354
+ $this->_message->analyseText = $this->_message->html . ' ' . $this->_message->text;
355
+ $this->_display('<strong>' . __('Subject', WYSIJA) . ' : ' . strip_tags($this->_message->subject) . '</strong>', false, $maxMessages - $this->_message->messageNB + 1);
356
+
357
+ // Identify the user and the e-mail... there is not the same info when it is a multisite or not
358
+ $email_identifiers = array();
359
+ if ($this->record_ms_bounce) {
360
+ preg_match('#WY([0-9]+)SI([0-9]+)JA([0-9]+)MS#i', $this->_message->analyseText, $email_identifiers);
361
+ if (!empty($email_identifiers[1]))
362
+ $this->_message->user_id = $email_identifiers[1];
363
+ if (!empty($email_identifiers[2]))
364
+ $this->_message->email_id = $email_identifiers[2];
365
+ if (!empty($email_identifiers[3]))
366
+ $this->_message->site_id = $email_identifiers[3];
367
+ }else {
368
+ preg_match('#WY([0-9]+)SI([0-9]+)JA#i', $this->_message->analyseText, $email_identifiers);
369
+ if (!empty($email_identifiers[1]))
370
+ $this->_message->user_id = $email_identifiers[1];
371
+ if (!empty($email_identifiers[2]))
372
+ $this->_message->email_id = $email_identifiers[2];
373
+ }
374
+
375
+ // if we don't have the user_id set then we need to find the user_id differently
376
+ if (empty($this->_message->user_id)) {
377
+ // We will need the e-mail itself in that case... :p
378
+ $emails_detected_in_the_email = array();
379
+ preg_match_all($this->detectEmail, $this->_message->analyseText, $emails_detected_in_the_email);
380
+ $reply_email = $this->config->getValue('reply_email');
381
+ $from_email = $this->config->getValue('from_email');
382
+ $bounce_email = $this->config->getValue('bounce_email');
383
+ $remove_emails = '#(' . str_replace(array('%'), array('@'), $this->config->getValue('bounce_login'));
384
+ if (!empty($bounce_email))
385
+ $remove_emails .= '|' . $bounce_email;
386
+ if (!empty($from_email))
387
+ $remove_emails .= '|' . $from_email;
388
+ if (!empty($reply_email))
389
+ $remove_emails .= '|' . $reply_email;
390
+ $remove_emails .= ')#i';
391
+ if (!empty($emails_detected_in_the_email[0])) {
392
+ $email_already_checked = array();
393
+ foreach ($emails_detected_in_the_email[0] as $detected_email) {
394
+ // We will find the e-mail if it's not in the list of incorrect e-mail addresses
395
+ if (!preg_match($remove_emails, $detected_email)) {
396
+ // We will keep this one, so we make sure it's strtolower
397
+ $this->_message->subemail = strtolower($detected_email);
398
+ // We already checked this e-mail address... no need to try it a second time
399
+ if (!empty($email_already_checked[$this->_message->subemail]))
400
+ continue;
401
+ $this->subClass->getFormat = OBJECT;
402
+ $result = $this->subClass->getOne(array('user_id'), array('email' => $this->_message->subemail));
403
+
404
+ $this->_message->user_id = $result->user_id;
405
+ $email_already_checked[$this->_message->subemail] = true;
406
+ if (!empty($this->_message->user_id))
407
+ break;
408
+ }
409
+ }
410
+ }
411
+ }
412
+
413
+ // get the email_id if it is not set and the user_id has been found
414
+ if (empty($this->_message->email_id) && !empty($this->_message->user_id)) {
415
+ // We can check if we have a user and only one e-mail sent for this user, it's obviously the e-mail we just sent!!
416
+ $modelEUS = WYSIJA::get('email_user_stat', 'model');
417
+ $emailres = $modelEUS->query('get_row', 'SELECT `email_id` FROM [wysija]' . $modelEUS->table_name . ' WHERE `user_id` = ' . (int) $this->_message->user_id . ' ORDER BY `sent_at` DESC LIMIT 1');
418
+ $this->_message->email_id = $emailres['email_id'];
419
+ //$this->_message->email_id = $this->db->loadResult();
420
+ }
421
+
422
+ foreach ($rules as $one_rule) {
423
+ //We stop as soon as we find a good rule...
424
+ if ($this->_handleRule($one_rule))
425
+ break;
426
+ }
427
+
428
+
429
+ if ($msgNB % 50 == 0){
430
+ if(!$this->record_ms_bounce) $this->_sub_actions();
431
+ }
432
+ }
433
+
434
+ if(!$this->record_ms_bounce) $this->_sub_actions();
435
+
436
+ if ($this->report) {
437
+ //We need to finish the current page properly
438
+ echo '</body></html>';
439
+ }
440
+ }
441
+
442
+
443
+ /**
444
+ * take action on the subscribers based on what data we gathered in the processing part earlier
445
+ */
446
+ function _sub_actions() {
447
+
448
+ // the action is about deleting users
449
+ if (!empty($this->deletedUsers)) {
450
+ $this->subClass->testdelete = true;
451
+ $helper_user = WYSIJA::get('user','helper');
452
+ $helper_user->delete($this->deletedUsers);
453
+ $this->deletedUsers = array();
454
+ }
455
+
456
+ if (!empty($this->unsubscribedUsers)) {
457
+ //unsubscribe user
458
+ $user_helper = WYSIJA::get('user', 'helper');
459
+ if (!is_array($this->unsubscribedUsers)) $this->unsubscribedUsers=array($this->unsubscribedUsers);
460
+
461
+ foreach ($this->unsubscribedUsers as $unsub_user_id) {
462
+ $user_helper->unsubscribe($unsub_user_id, true);
463
+ }
464
+
465
+ $this->unsubscribedUsers = array();
466
+ }
467
+
468
+ if (!empty($this->addtolistUsers)) {
469
+ //unsubscribe user
470
+ $user_helper = WYSIJA::get('user', 'helper');
471
+ foreach ($this->addtolistUsers as $listid => $user_ids) {
472
+ $user_helper->addToList($listid, $user_ids);
473
+ }
474
+
475
+ $this->addtolistUsers = array();
476
+ }
477
+
478
+ if (!empty($this->bounceMessages)) {
479
+ foreach ($this->bounceMessages as $email_id => $bouncedata) {
480
+ if (!empty($bouncedata['user_id'])) {
481
+ //flag email has bounced
482
+ $modelEUS = WYSIJA::get('email_user_stat', 'model');
483
+ $modelEUS->update(array('status' => -1), array('user_id' => $bouncedata['user_id'], 'email_id' => (int) $email_id));
484
+ /* $this->db->setQuery('UPDATE '.acymailing_table('userstats').' SET `bounce` = `bounce` + 1 WHERE `user_id` IN ('.implode(',',$bouncedata['user_id']).') AND `email_id` = '.(int) $email_id);
485
+ $this->db->query(); */
486
+ }
487
+ }
488
+ $this->bounceMessages = array();
489
+ }
490
+ }
491
+
492
+
493
+
494
+
495
+ function _handleRule(&$one_rule) {
496
+
497
+ $regex = $one_rule['regex'];
498
+ if (empty($regex))
499
+ return false;
500
+
501
+ //Do it based on the config of the rule...
502
+
503
+ $analyse_text = '';
504
+ if (isset($one_rule['executed_on']['senderinfo']))
505
+ $analyse_text .= ' ' . $this->_message->header->sender_name . $this->_message->header->sender_email;
506
+ if (isset($one_rule['executed_on']['subject']))
507
+ $analyse_text .= ' ' . $this->_message->subject;
508
+ if (isset($one_rule['executed_on']['body'])) {
509
+ if (!empty($this->_message->html))
510
+ $analyse_text .= ' ' . $this->_message->html;
511
+ if (!empty($this->_message->text))
512
+ $analyse_text .= ' ' . $this->_message->text;
513
+ }
514
+
515
+ //regex multilines
516
+ if (!preg_match('#' . $regex . '#is', $analyse_text))
517
+ return false;
518
+
519
+ $message = $one_rule['name'];
520
+
521
+ if($this->record_ms_bounce) {
522
+ // no need for user action in multisite because we'll do it per site in a second process
523
+ $message .= $this->_action_message_ms($one_rule);
524
+ }else{
525
+ $message .= $this->_action_user($one_rule);
526
+ $message .= $this->_action_message($one_rule);
527
+ }
528
+
529
+
530
+ $this->_display($message, true);
531
+
532
+ return true;
533
+ }
534
+
535
+ function _action_user(&$one_rule) {
536
+ $message = '';
537
+
538
+ if (empty($this->_message->user_id)) {
539
+ $message .= 'user not identified';
540
+ if (!empty($this->_message->subemail))
541
+ $message .= ' ( ' . $this->_message->subemail . ' ) ';
542
+ return $message;
543
+ }
544
+
545
+ if (isset($one_rule['action_user']) && in_array($one_rule['action_user'], array('unsub'))) {
546
+ if (empty($this->_message->subemail)) {
547
+ $currentUser = $this->subClass->getObject($this->_message->user_id);
548
+ if (!empty($currentUser->email))
549
+ $this->_message->subemail = $currentUser->email;
550
+ }
551
+ }
552
+
553
+ if (empty($this->_message->subemail))
554
+ $this->_message->subemail = $this->_message->user_id;
555
+
556
+ //let's handle some actions on the subscriber first
557
+ if (isset($one_rule['action_user_stats'])) {
558
+ //handle this rule in the stats
559
+ if (!empty($this->_message->email_id)) {
560
+ if (empty($this->bounceMessages[$this->_message->email_id]['nbbounces'])) {
561
+ $this->bounceMessages[$this->_message->email_id] = array();
562
+ $this->bounceMessages[$this->_message->email_id]['nbbounces'] = 1;
563
+ } else {
564
+ $this->bounceMessages[$this->_message->email_id]['nbbounces']++;
565
+ }
566
+
567
+ if (!empty($this->_message->user_id) AND ((isset($one_rule['action_user']) && $one_rule['action_user'] != 'delete') || !isset($one_rule['action_user']) )) {
568
+ //Increment the bounce number in the user stat table but only if we don't delete the subscriber
569
+ $this->bounceMessages[$this->_message->email_id]['user_id'][] = intval($this->_message->user_id);
570
+ }
571
+ }
572
+ }
573
+
574
+ //IN WYSIJA THERE ARE 3 POSSIBILITIES
575
+ //1-Delete user
576
+ //2-Unsubscribe user
577
+ //3-Unsubscribe and attach to list "xxx"
578
+ if (isset($one_rule['action_user'])) {
579
+ switch ($one_rule['action_user']) {
580
+ case 'delete'://1 -Delete user
581
+ $message .= ', user ' . $this->_message->subemail . ' deleted';
582
+ $this->deletedUsers[] = intval($this->_message->user_id);
583
+
584
+ break;
585
+ case 'unsub'://2-Unsubscribe user
586
+ //when we unsubscribe somebody automatically we set the status to -2 instead of -1 to make the difference
587
+ $message .= ', user ' . $this->_message->subemail . ' unsubscribed';
588
+ $this->unsubscribedUsers[] = $this->_message->user_id;
589
+
590
+ break;
591
+ default:
592
+ //3 - Unsubscribe user and add to list
593
+ if (strpos($one_rule['action_user'], 'unsub_') !== false) {
594
+ $listid = (int) str_replace('unsub_', '', $one_rule['action_user']);
595
+ $message .= ', user ' . $this->_message->subemail . ' unsubscribed';
596
+ $this->unsubscribedUsers[] = $this->_message->user_id;
597
+ $this->addtolistUsers[$listid][] = $this->_message->user_id;
598
+ $message .= ', user ' . $this->_message->subemail . ' added to list "' . $this->listdetails[$listid] . '"';
599
+ }
600
+ }
601
+ }
602
+
603
+
604
+
605
+ //Make sure we have enough messages to really execute this
606
+ if (!empty($one_rule['action_user_min']) && $one_rule['action_user_min'] > 1) {
607
+ //Let's load the number of bounces the user has and then exit or not...
608
+ $modelEUS = WYSIJA::get('email_user_stat', 'model');
609
+ $res = $modelEUS->query('get_row', 'SELECT COUNT(email_id) as count FROM [wysija]' . $modelEUS->table_name . ' WHERE status = -1 AND user_id = ' . $this->_message->user_id);
610
+ $nb = intval($res['count']) + 1;
611
+
612
+ if ($nb < $one_rule['action_user_min']) {
613
+ $message .= ', ' . sprintf(__('We received %1$s messages from the user %2$s', WYSIJA), $nb, $this->_message->subemail) . ', ' . sprintf(__('Actions will be executed after %1$s messages', WYSIJA), $one_rule['action_user_min']);
614
+ return $message;
615
+ }
616
+ }
617
+
618
+ return $message;
619
+ }
620
+
621
+ function _action_message(&$one_rule) {
622
+
623
+ $message = '';
624
+
625
+ //Handle actions on the message itself
626
+
627
+ if (isset($one_rule['action_message']['save']) && !empty($this->_message->user_id)) {
628
+ //We have a user_id, should we save the message in the database?
629
+ $data = array();
630
+ $data[] = 'SUBJECT::' . $this->_message->subject;
631
+
632
+ if (!empty($this->_message->html))
633
+ $data[] = 'HTML_VERSION::' . htmlentities($this->_message->html);
634
+ if (!empty($this->_message->text))
635
+ $data[] = 'TEXT_VERSION::' . nl2br(htmlentities($this->_message->text));
636
+ $data[] = 'REPLYTO_ADDRESS::' . $this->_message->header->reply_to_name . ' ( ' . $this->_message->header->reply_to_email . ' )';
637
+ $data[] = 'FROM_ADDRESS::' . $this->_message->header->from_name . ' ( ' . $this->_message->header->from_email . ' )';
638
+ $data[] = print_r($this->_message->headerinfo, true);
639
+ $this->historyClass->insert($this->_message->user_id, 'bounce', $data, @$this->_message->email_id);
640
+ $message .= ', message saved (user ' . $this->_message->user_id . ')';
641
+ }
642
+
643
+ if (isset($one_rule['forward'])) {
644
+ if (isset($one_rule['action_message_forwardto']) && !empty($one_rule['action_message_forwardto']) && trim($one_rule['action_message_forwardto']) != trim($this->config->getValue('bounce_email'))) {
645
+ //Get the forward address :
646
+ $this->mailer->clearAll();
647
+ $this->mailer->Subject = 'BOUNCE FORWARD : ' . $this->_message->subject;
648
+ $this->mailer->AddAddress($one_rule['action_message_forwardto']);
649
+ if (!empty($this->_message->html)) {
650
+ $this->mailer->IsHTML(true);
651
+ $this->mailer->Body = $this->_message->html;
652
+ if (!empty($this->_message->text))
653
+ $this->mailer->Body .= '<br/><br/>-------<br/>' . nl2br($this->_message->text);
654
+ }else {
655
+ $this->mailer->IsHTML(false);
656
+ $this->mailer->Body = $this->_message->text;
657
+ }
658
+
659
+ //We add all other extra information just in case of we could use them...
660
+ //original-rcpt-to ? http://tools.ietf.org/html/rfc5965
661
+ $this->mailer->Body .= print_r($this->_message->headerinfo, true);
662
+ $this->mailer->AddReplyTo($this->_message->header->reply_to_email, $this->_message->header->reply_to_name);
663
+ $this->mailer->setFrom($this->_message->header->from_email, $this->_message->header->from_name);
664
+ if ($this->mailer->send()) {
665
+ $message .= ', forwarded to ' . $one_rule['action_message_forwardto'];
666
+ } else {
667
+ $message .= ', error forwarding to ' . $one_rule['action_message_forwardto'];
668
+ }
669
+
670
+ } else {
671
+ //we dont delete the email if the forward email is not specified or if the bounce email is the same as the forward email
672
+ unset($one_rule['action_message']['delete']);
673
+ }
674
+ }
675
+
676
+ if (isset($one_rule['action_message']['delete'])) {
677
+ $message .= ', message deleted';
678
+ $this->deleteMessage($this->_message->messageNB);
679
+ }
680
+
681
+ return $message;
682
+ }
683
+
684
+ function _decodeAddressimap($type) {
685
+ $address = $type . 'address';
686
+ $name = $type . '_name';
687
+ $email = $type . '_email';
688
+ if (empty($this->_message->$type))
689
+ return false;
690
+
691
+ $var = $this->_message->$type;
692
+
693
+ if (!empty($this->_message->$address)) {
694
+ if(!isset($this->_message->header) || is_null($this->_message->header)){
695
+ $this->_message->header = new stdClass();
696
+ }
697
+ } else {
698
+ $this->_message->header->$name = $var[0]->personal;
699
+ }
700
+
701
+ $this->_message->header->$email = $var[0]->mailbox . '@' . @$var[0]->host;
702
+ return true;
703
+ }
704
+
705
+ /**
706
+ * If num is empty then it's a message otherwise it's a send statrus
707
+ */
708
+ function _display($message, $status = '', $num = '') {
709
+ $this->messages[] = $message;
710
+
711
+ if (!$this->report)
712
+ return;
713
+
714
+ $color = $status ? 'black' : 'blue';
715
+ if (!empty($num))
716
+ echo '<br/>' . $num . ' : ';
717
+ else
718
+ echo '<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;';
719
+
720
+ echo '<font style="font-family: Arial;" color="' . $color . '">' . $message . '</font>';
721
+ if (function_exists('ob_flush'))
722
+ @ob_flush();
723
+ @flush();
724
+ }
725
+
726
+ function _decodeHeader($input) {
727
+ // Remove white space between encoded-words
728
+ $input = preg_replace('/(=\?[^?]+\?(q|b)\?[^?]*\?=)(\s)+=\?/i', '\1=?', $input);
729
+ $this->charset = false;
730
+
731
+ // For each encoded-word...
732
+ while (preg_match('/(=\?([^?]+)\?(q|b)\?([^?]*)\?=)/i', $input, $matches)) {
733
+
734
+ $encoded = $matches[1];
735
+ $charset = $matches[2];
736
+ $encoding = $matches[3];
737
+ $text = $matches[4];
738
+
739
+ switch (strtolower($encoding)) {
740
+ case 'b':
741
+ $text = base64_decode($text);
742
+ break;
743
+
744
+ case 'q':
745
+ $text = str_replace('_', ' ', $text);
746
+ preg_match_all('/=([a-f0-9]{2})/i', $text, $matches);
747
+ foreach ($matches[1] as $value)
748
+ $text = str_replace('=' . $value, chr(hexdec($value)), $text);
749
+ break;
750
+ }
751
+ $this->charset = $charset;
752
+ $input = str_replace($encoded, $text, $input);
753
+ }
754
+
755
+ return $input;
756
+ }
757
+
758
+ function _explodeBody($struct, $path = "0", $inline = 0) {
759
+ $allParts = array();
760
+
761
+ if (empty($struct->parts))
762
+ return $allParts;
763
+
764
+ $c = 0; //counts real content
765
+ foreach ($struct->parts as $part) {
766
+ if ($part->type == 1) {
767
+ //There are more parts....:
768
+ if ($part->subtype == "MIXED") { //Mixed:
769
+ $path = $this->_incPath($path, 1); //refreshing current path
770
+ $newpath = $path . ".0"; //create a new path-id (ex.:2.0)
771
+ $allParts = array_merge($this->_explodeBody($part, $newpath), $allParts); //fetch new parts
772
+ } else { //Alternativ / rfc / signed
773
+ $newpath = $this->_incPath($path, 1);
774
+ $path = $this->_incPath($path, 1);
775
+ $allParts = array_merge($this->_explodeBody($part, $newpath, 1), $allParts);
776
+ }
777
+ } else {
778
+ $c++;
779
+ //creating new tree if this is part of a alternativ or rfc message:
780
+ if ($c == 1 && $inline) {
781
+ $path = $path . ".0";
782
+ }
783
+ //saving content:
784
+ $path = $this->_incPath($path, 1);
785
+ //print "<br> Content ".$path."<br>"; //debug information
786
+ $allParts[$path] = $part;
787
+ }
788
+ }
789
+
790
+ return $allParts;
791
+ }
792
+
793
+ //Increases the Path to the parts:
794
+ function _incPath($path, $inc) {
795
+ $newpath = "";
796
+ $path_elements = explode(".", $path);
797
+ $limit = count($path_elements);
798
+ for ($i = 0; $i < $limit; $i++) {
799
+ if ($i == $limit - 1) { //last element
800
+ $newpath .= $path_elements[$i] + $inc; // new Part-Number
801
+ } else {
802
+ $newpath .= $path_elements[$i] . "."; //rebuild "1.2.2"-Chronology
803
+ }
804
+ }
805
+ return $newpath;
806
+ }
807
+
808
+ function _decodeContent($content, $structure) {
809
+ $encoding = $structure->encoding;
810
+
811
+ //First we decode the content properly
812
+ if ($encoding == 2)
813
+ $content = imap_binary($content);
814
+ elseif ($encoding == 3)
815
+ $content = imap_base64($content);
816
+ elseif ($encoding == 4)
817
+ $content = imap_qprint($content);
818
+ //Other cases??
819
+ //added for a client who had issue when message was base64
820
+ if(base64_decode($content,true)!==FALSE)
821
+ $content = base64_decode($content);
822
+
823
+ //Now we convert into utf-8!
824
+ //$charset = $this->_getMailParam($structure,'charset');
825
+ // removes attachment to prevent bounce handling timeout
826
+ // 100 000 characters is plenty
827
+ return substr($content, 0, 100000);
828
+ }
829
+
830
+ function _getMailParam($params, $name) {
831
+ $searchIn = array();
832
+
833
+ if ($params->ifparameters)
834
+ $searchIn = array_merge($searchIn, $params->parameters);
835
+ if ($params->ifdparameters)
836
+ $searchIn = array_merge($searchIn, $params->dparameters);
837
+
838
+ if (empty($searchIn))
839
+ return false;
840
+
841
+ foreach ($searchIn as $num => $values) {
842
+ if (strtolower($values->attribute) == $name) {
843
+ return $values->value;
844
+ }
845
+ }
846
+ }
847
+
848
+ function getErrors() {
849
+ $return = array();
850
+ if ($this->usepear) {
851
+ //TODO : get some errors from the pear interface?
852
+ } else {
853
+ $alerts = imap_alerts();
854
+ $errors = imap_errors();
855
+ if (!empty($alerts))
856
+ $return = array_merge($return, $alerts);
857
+ if (!empty($errors))
858
+ $return = array_merge($return, $errors);
859
+ }
860
+
861
+ return $return;
862
+ }
863
+
864
+
865
+
866
+ /**
867
+ * simple function that launch a bounce process
868
+ * @return boolean
869
+ */
870
+ function process_bounce() {
871
+
872
+ @ini_set('max_execution_time', 0);
873
+
874
+ $model_config = WYSIJA::get('config', 'model');
875
+
876
+ $this->report = true;
877
+ if (!$this->init()) {
878
+ $res['result'] = false;
879
+ return $res;
880
+ }
881
+ if (!$this->connect()) {
882
+ $this->error($this->getErrors());
883
+ $res['result'] = false;
884
+ return $res;
885
+ }
886
+ $this->notice(sprintf(__('Successfully connected to %1$s'), $model_config->getValue('bounce_login')));
887
+ $nbMessages = $this->getNBMessages();
888
+
889
+
890
+ if (empty($nbMessages)) {
891
+ $this->error(__('There are no messages'), true);
892
+ $res['result'] = false;
893
+ return $res;
894
+ } else {
895
+ $this->notice(sprintf(__('There are %1$s messages in your mailbox'), $nbMessages));
896
+ }
897
+
898
+ $this->handleMessages();
899
+
900
+ $this->close();
901
+
902
+ $res['result'] = true;
903
+
904
+ return $res;
905
+ }
906
+
907
+
908
+ /**
909
+ * record the bounce into the bounce table on a multisite
910
+ * @return type
911
+ */
912
+ function record_bounce_ms() {
913
+ // make sure that the bounce recording is not already being processed on another child site
914
+ if (get_site_option('wysija_bounce_being_recorded'))
915
+ return;
916
+
917
+ // flag the current recording
918
+ WYSIJA::update_option('wysija_bounce_being_recorded', true);
919
+
920
+ // set the flag to indicate we are processing the bounce in a multisite manner right now
921
+ $this->record_ms_bounce = true;
922
+
923
+ // will record the bounce in the ms table
924
+ $result = $this->process_bounce();
925
+
926
+ // lower the flag we can process the bounce again
927
+ WYSIJA::update_option('wysija_bounce_being_recorded', false);
928
+
929
+ return $result;
930
+ }
931
+ /**
932
+ * base on the records we have in the bounce table we will take action
933
+ * @return boolean
934
+ */
935
+ function process_bounce_ms() {
936
+
937
+ @ini_set('max_execution_time', 0);
938
+ global $blog_id;
939
+ $main_site_prefix = $this->subClass->get_site_prefix();
940
+ // make a query that will handle the bounce delete for all of the emails recorded in the bounce table for that site
941
+ // join the table bounce and user to make sure the ID's exist get 200 of them
942
+
943
+ // we will delete one by one all of the data from the users that need to be removed
944
+ $tables = array(
945
+ 'user_history',
946
+ 'email_user_url',
947
+ 'email_user_stat',
948
+ 'user_list',
949
+ 'queue',
950
+ );
951
+
952
+ // central query to fetch the id of the bounced emails of the delete action
953
+ $query_join_bounce = 'SELECT B.user_id from '.$main_site_prefix.'bounce as A JOIN [wysija]user as B on A.email = B.email WHERE A.action_taken = "delete"';
954
+
955
+ try{
956
+ foreach ($tables as $table_name){
957
+ $query_delete = 'DELETE FROM [wysija]' . $table_name . ' WHERE user_id IN (' . $query_join_bounce . ')';
958
+ $this->subClass->query($query_delete);
959
+ }
960
+ }catch(Exception $e){
961
+ return false;
962
+ }
963
+
964
+ // delete process from the user table needs to be made through a join since we cannot nest select from the same table we delete from
965
+ $query_delete_user = 'DELETE A.* FROM [wysija]user as A JOIN '.$main_site_prefix.'bounce as B on A.email = B.email WHERE B.action_taken = "delete"';
966
+ $this->subClass->query($query_delete_user);
967
+
968
+ // central query to fetch the id of the bounced emails of the unsubscribe action
969
+ $query_join_bounce = 'SELECT B.user_id from '.$main_site_prefix.'bounce as A JOIN [wysija]user as B on A.email = B.email WHERE A.action_taken = "unsubscribe"';
970
+
971
+ // query to update the status to unsubscribe
972
+ $query_update_user = 'UPDATE [wysija]user as A JOIN '.$main_site_prefix.'bounce as B on A.email = B.email SET A.`status` = -1 WHERE A.`status` != -1';
973
+ //$query_update_user = 'UPDATE [wysija]user SET `status` = -1 WHERE user_id IN (' . $query_join_bounce . ')';
974
+
975
+ $query_update_user_list = 'UPDATE [wysija]user_list as A JOIN [wysija]user as B on A.user_id = B.user_id JOIN '.$main_site_prefix.'bounce as C on B.email = C.email SET A.`unsub_date` = '.time().' , A.`sub_date` = 0 WHERE B.`status` != -1';
976
+ //$query_update_user_list = 'UPDATE [wysija]user_list SET `unsub_date` = '.time().' , `sub_date` = 0 WHERE user_id IN (' . $query_join_bounce . ')';
977
+
978
+ $this->subClass->getResults($query_update_user);
979
+ $this->subClass->getResults($query_update_user_list);
980
+
981
+ // delete what's in the queue for those subscribers
982
+ $query_delete = 'DELETE FROM [wysija]queue WHERE user_id IN (' . $query_join_bounce . ')';
983
+
984
+ $this->subClass->getResults($query_delete);
985
+
986
+ // query to set the boucne value in the email table
987
+ $query_update_email_user_status = 'UPDATE [wysija]email_user_stat as A JOIN '.$main_site_prefix.'bounce as C on (A.user_id = C.user_id AND A.email_id = C.email_id) SET A.`status` = -1 WHERE C.site_id='.$blog_id;
988
+ $this->subClass->getResults($query_update_email_user_status);
989
+
990
+ $res['result'] = true;
991
+
992
+ return $res;
993
+ }
994
+
995
+ /**
996
+ * take action on the subscribers for multisites processing we don't delete or unsubscribe the users, we just record the action to be taken in the db
997
+ */
998
+ function _sub_actions_ms() {
999
+
1000
+ // the action is about deleting users
1001
+ if (!empty($this->deletedUsers)) {
1002
+ $this->subClass->testdelete = true;
1003
+
1004
+ $this->deletedUsers = array();
1005
+ }
1006
+
1007
+ if (!empty($this->unsubscribedUsers)) {
1008
+
1009
+ $this->unsubscribedUsers = array();
1010
+ }
1011
+ }
1012
+
1013
+ function _action_message_ms(&$one_rule) {
1014
+
1015
+ $email_copy = $message = '';
1016
+
1017
+ //Handle actions on the message itself
1018
+
1019
+ if (isset($one_rule['action_message']['save']) && !empty($this->_message->user_id)) {
1020
+ //We have a user_id, should we save the message in the database?
1021
+ $email_saved_as_array = array();
1022
+ $email_saved_as_array[] = 'SUBJECT::' . $this->_message->subject;
1023
+
1024
+ if (!empty($this->_message->html))
1025
+ $email_saved_as_array[] = 'HTML_VERSION::' . htmlentities($this->_message->html);
1026
+ if (!empty($this->_message->text))
1027
+ $email_saved_as_array[] = 'TEXT_VERSION::' . nl2br(htmlentities($this->_message->text));
1028
+ $email_saved_as_array[] = 'REPLYTO_ADDRESS::' . $this->_message->header->reply_to_name . ' ( ' . $this->_message->header->reply_to_email . ' )';
1029
+ $email_saved_as_array[] = 'FROM_ADDRESS::' . $this->_message->header->from_name . ' ( ' . $this->_message->header->from_email . ' )';
1030
+ $email_saved_as_array[] = print_r($this->_message->headerinfo, true);
1031
+ $email_copy = implode("\n",$email_saved_as_array);
1032
+
1033
+ $message .= ', message saved (user ' . $this->_message->user_id . ')';
1034
+
1035
+ }
1036
+
1037
+ if (isset($one_rule['forward'])) {
1038
+ if (isset($one_rule['action_message_forwardto']) && !empty($one_rule['action_message_forwardto']) && trim($one_rule['action_message_forwardto']) != trim($this->config->getValue('bounce_email'))) {
1039
+ //Get the forward address :
1040
+ $this->mailer->clearAll();
1041
+ $this->mailer->Subject = 'BOUNCE FORWARD : ' . $this->_message->subject;
1042
+ $this->mailer->AddAddress($one_rule['action_message_forwardto']);
1043
+ if (!empty($this->_message->html)) {
1044
+ $this->mailer->IsHTML(true);
1045
+ $this->mailer->Body = $this->_message->html;
1046
+ if (!empty($this->_message->text))
1047
+ $this->mailer->Body .= '<br/><br/>-------<br/>' . nl2br($this->_message->text);
1048
+ }else {
1049
+ $this->mailer->IsHTML(false);
1050
+ $this->mailer->Body = $this->_message->text;
1051
+ }
1052
+
1053
+ //We add all other extra information just in case of we could use them...
1054
+ //original-rcpt-to ? http://tools.ietf.org/html/rfc5965
1055
+ $this->mailer->Body .= print_r($this->_message->headerinfo, true);
1056
+ $this->mailer->AddReplyTo($this->_message->header->reply_to_email, $this->_message->header->reply_to_name);
1057
+ $this->mailer->setFrom($this->_message->header->from_email, $this->_message->header->from_name);
1058
+ if ($this->mailer->send()) {
1059
+ $message .= ', forwarded to ' . $one_rule['action_message_forwardto'];
1060
+ } else {
1061
+ $message .= ', error forwarding to ' . $one_rule['action_message_forwardto'];
1062
+ }
1063
+
1064
+ } else {
1065
+ //we dont delete the email if the forward email is not specified or if the bounce email is the same as the forward email
1066
+ unset($one_rule['action_message']['delete']);
1067
+ }
1068
+ }
1069
+
1070
+ // insert the message or just the action taken on that message in the main bounce table
1071
+
1072
+ // get the prefix of the current child site
1073
+ $bounced_site_prefix = $this->subClass->get_site_prefix($this->_message->site_id);
1074
+ // get the email of the bounced message based on the user_id and the site_id
1075
+ $query = 'SELECT email FROM `'.$bounced_site_prefix.'user` WHERE user_id='.(int)$this->_message->user_id.' LIMIT 0 , 1';
1076
+ $result_subscriber = $this->subClass->query('get_res',$query,OBJECT);
1077
+
1078
+ $this->subClass->reset();
1079
+
1080
+ if(strpos($one_rule['action_user'], 'unsub')!==false) $action_taken='unsubscribe';
1081
+ elseif($one_rule['action_user']!='') $action_taken='delete';
1082
+
1083
+ if(!empty($result_subscriber[0]->email)){
1084
+ $main_site_prefix = $this->subClass->get_site_prefix();
1085
+ $query_insert_bounce_ms = 'INSERT IGNORE INTO `'.$main_site_prefix.'bounce` (`email`,`site_id`,`user_id`,`email_id`,`action_taken`,`case`,`message`,`created_at`)';
1086
+ $query_insert_bounce_ms .= " VALUES ('".$result_subscriber[0]->email."','".(int)$this->_message->site_id."','".(int)$this->_message->user_id."','".(int)$this->_message->email_id."','".$action_taken."', '".$one_rule['key']."', '". esc_sql($email_copy)."', '".time()."')";
1087
+
1088
+ $this->subClass->query($query_insert_bounce_ms);
1089
+ }
1090
+
1091
+ if (isset($one_rule['action_message']['delete'])) {
1092
+ $message .= ', message deleted';
1093
+ $this->deleteMessage($this->_message->messageNB);
1094
+ }
1095
+
1096
+ return $message;
1097
+ }
1098
+
1099
+ }
helpers/campaigns.php ADDED
@@ -0,0 +1,52 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ /*
4
+ add_action( 'wp_ajax_mailpoet.search_terms', 'WYSIJA_help_campaigns::ajax_search_terms' );
5
+
6
+
7
+ */
8
+ defined('WYSIJA') or die('Restricted access');
9
+ class WYSIJA_help_campaigns extends WYSIJA_object{
10
+ function WYSIJA_help_campaigns(){
11
+
12
+ }
13
+
14
+ function saveParameters($email_id, $key, $value)
15
+ {
16
+ // 1. get params field for given campaign
17
+ $modelEmail = WYSIJA::get('email', 'model');
18
+ $email = $modelEmail->getOne('params', array('email_id' => $email_id));
19
+ $params = $email['params'];
20
+
21
+ if(!is_array($params)) {
22
+ $params = array();
23
+ }
24
+
25
+ // 2 update data for given key
26
+ if(array_key_exists($key, $params)) {
27
+ $params[$key] = $value;
28
+ } else {
29
+ $params = array_merge($params, array($key => $value));
30
+ }
31
+
32
+ // 3. update campaign
33
+ return $modelEmail->update(array('params' => $params), array('email_id' => $email_id));
34
+ }
35
+
36
+ function getParameters($email_id, $key = null) {
37
+ // 1. get params field for given campaign
38
+ $modelEmail = WYSIJA::get('email', 'model');
39
+ $email = $modelEmail->getOne('params', array('email_id' => $email_id));
40
+ $params = $email['params'];
41
+
42
+ if($key === null) {
43
+ return $params;
44
+ } else {
45
+ if(!is_array($params) or array_key_exists($key, $params) === false) {
46
+ return false;
47
+ } else {
48
+ return $params[$key];
49
+ }
50
+ }
51
+ }
52
+ }
helpers/charts.php ADDED
@@ -0,0 +1,66 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_charts extends WYSIJA_object {
4
+
5
+ function WYSIJA_help_charts() {
6
+
7
+ }
8
+
9
+ function pieChart($id, $options = array()) {
10
+ return $this->generateChart('pie', $id, $options);
11
+ }
12
+
13
+ function serialChart($id, $options = array()) {
14
+ return $this->generateChart('serial', $id, $options);
15
+ }
16
+
17
+ function generateChart($type = 'serial', $id, $options = array()) {
18
+ // format id
19
+ $id = str_replace(' ', '-', $id);
20
+ // chart dimensions
21
+ $width = (isset($options['width'])) ? (int)$options['width'] : 400;
22
+ $height = (isset($options['height'])) ? (int)$options['height'] : 225;
23
+
24
+ // chart title
25
+ $title = (isset($options['title'])) ? $options['title'] : null;
26
+
27
+ // data
28
+ $data = (isset($options['data'])) ? $options['data'] : null;
29
+
30
+ // generate JS code
31
+ $content = '<div id="wysija-chart-'.$id.'" class="wysija-chart" style="width:'.$width.'px;height:'.$height.'px;"></div>';
32
+ $content .= '<script type="text/javascript">';
33
+ $content .= 'AmCharts.ready(function () {';
34
+ $content .= 'WysijaCharts.generateChart("'.$type.'", "wysija-chart-'.$id.'", {';
35
+ // set chart title
36
+ $content .= 'title: "'.$title.'",';
37
+ // set data
38
+ $content .= 'data: '.json_encode($data).',';
39
+
40
+ switch ($type) {
41
+ case 'serial':
42
+ // axes data
43
+ $axes = (isset($options['axes'])) ? $options['axes'] : null;
44
+ // category (the field used to sort by)
45
+ $category = (isset($options['category'])) ? $options['category'] : null;
46
+
47
+ $content .= 'axes: '.json_encode($axes).',';
48
+ $content .= 'category: "'.$category.'"';
49
+ break;
50
+ case 'pie':
51
+ // title and value fields
52
+ $titleField = (isset($options['titleField'])) ? $options['titleField'] : null;
53
+ $valueField = (isset($options['valueField'])) ? $options['valueField'] : null;
54
+
55
+ $content .= 'titleField: "'.$titleField.'",';
56
+ $content .= 'valueField: "'.$valueField.'"';
57
+
58
+ break;
59
+ }
60
+
61
+ $content .= '});';
62
+ $content .= '});';
63
+ $content .= '</script>';
64
+ return $content;
65
+ }
66
+ }
helpers/conflicts.php ADDED
@@ -0,0 +1,103 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_conflicts extends WYSIJA_object{
4
+ var $cleanHooks=array();
5
+ function WYSIJA_help_conflicts(){
6
+
7
+ }
8
+
9
+ /**
10
+ * try to remove hook from plugins inviting themselves onto our interfaces
11
+ * @param type $conflictingPlugins
12
+ */
13
+ function resolve($conflictingPlugins){
14
+
15
+ $this->whatToClean=array();
16
+ foreach($conflictingPlugins as $keyPlg =>$plugin){
17
+ foreach($plugin['clean'] as $action => $details){
18
+ foreach($details as $priority =>$info){
19
+ $this->cleanHooks[$action][$priority][]=$info;
20
+ }
21
+ }
22
+ }
23
+ foreach($this->cleanHooks as $hookToclean => $info){
24
+
25
+ switch($hookToclean){
26
+ case 'admin_head':
27
+ add_action('init', array($this, 'remove_admin_head'), 999);
28
+ break;
29
+ case 'admin_print_scripts':
30
+ add_action('admin_menu', array($this, 'remove_admin_print_scripts'), 999);
31
+ break;
32
+ case 'wp_enqueue_scripts':
33
+ add_action('admin_menu', array($this, 'remove_wp_enqueue_scripts'), 999);
34
+ break;
35
+ case 'admin_enqueue_scripts':
36
+ add_action('admin_menu', array($this, 'remove_admin_enqueue_scripts'), 999);
37
+ break;
38
+ case 'init':
39
+ add_action('after_setup_theme', array($this, 'remove_init'), 999);
40
+ break;
41
+ default:
42
+ add_action('admin_footer', array($this, 'remove_default'), 999);
43
+ }
44
+ }
45
+ }
46
+ function remove_init(){
47
+ global $wp_filter;
48
+ $this->remove_actions('init');
49
+ }
50
+
51
+ function remove_default() {
52
+ $this->remove_actions('admin_init');
53
+ }
54
+
55
+ function remove_admin_head(){
56
+ $this->remove_actions('admin_head');
57
+ }
58
+ function remove_admin_print_scripts(){
59
+ $this->remove_actions('admin_print_scripts');
60
+ }
61
+
62
+ function remove_wp_enqueue_scripts() {
63
+ $this->remove_actions('wp_enqueue_scripts');
64
+ }
65
+
66
+ function remove_admin_enqueue_scripts() {
67
+ $this->remove_actions('admin_enqueue_scripts');
68
+ }
69
+
70
+ function remove_actions($actionsToClear){
71
+ //Carefull WordPress global
72
+ global $wp_filter;
73
+
74
+ foreach($wp_filter[$actionsToClear] as $priority => $callbacks) {
75
+
76
+ if(!isset($this->cleanHooks[$actionsToClear][$priority])) continue;
77
+
78
+ foreach($callbacks as $identifier => $arrayInfo){
79
+
80
+ if(is_array($arrayInfo['function'])){
81
+ foreach($arrayInfo['function'] as $id => $myobject){
82
+ foreach($this->cleanHooks[$actionsToClear][$priority] as $infoClear) {
83
+ if(isset($infoClear['objects']) && is_object($myobject) && in_array(get_class($myobject),$infoClear['objects'])){
84
+ unset($wp_filter[$actionsToClear][$priority][$identifier]);
85
+ }
86
+ }
87
+ }
88
+ } else {
89
+ foreach($this->cleanHooks[$actionsToClear][$priority] as $infoClear){
90
+ // if there is more than one function specified (key: functions, type: array)
91
+ if(isset($infoClear["functions"]) && function_exists($arrayInfo['function']) && in_array($arrayInfo['function'],$infoClear["functions"])){
92
+ unset($wp_filter[$actionsToClear][$priority][$identifier]);
93
+ // if there is only one function to remove (key: function, type: string)
94
+ } else if(array_key_exists('function', $infoClear) && $infoClear['function'] === $arrayInfo['function']) {
95
+ unset($wp_filter[$actionsToClear][$priority][$identifier]);
96
+ }
97
+ }
98
+
99
+ }
100
+ }
101
+ }
102
+ }
103
+ }
helpers/cron.php ADDED
@@ -0,0 +1,184 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_cron extends WYSIJA_object{
4
+
5
+ var $report=false;
6
+
7
+ function WYSIJA_help_cron(){
8
+
9
+ }
10
+
11
+ /**
12
+ * the cron tasks are being run for a certain number of processes (all queue, bounce etc..)
13
+ * @return void
14
+ */
15
+ function run() {
16
+ @ini_set('max_execution_time',0);
17
+ $model_config = WYSIJA::get('config','model');
18
+ $running = false;
19
+ if(!$model_config->getValue('cron_manual')){
20
+ return;
21
+ }
22
+ // get the param from where you want
23
+ $report = $process = false;
24
+ if(isset($_REQUEST['process']) && $_REQUEST['process']){
25
+ $process = $_REQUEST['process'];
26
+ }elseif(!isset($_SERVER['REQUEST_URI']) && isset($_SERVER['SHELL']) && isset($_SERVER['argv'][2]) && $_SERVER['argv'][2]){
27
+ $process = $_SERVER['argv'][2];
28
+ }
29
+
30
+ if(isset($_REQUEST['report']) && $_REQUEST['report']){
31
+ $this->report = $_REQUEST['report'];
32
+ }elseif(!isset($_SERVER['REQUEST_URI']) && isset($_SERVER['SHELL']) && isset($_SERVER['argv'][3]) && $_SERVER['argv'][3]){
33
+ $this->report = $_SERVER['argv'][3];
34
+ }
35
+
36
+ if($process){
37
+ //include the needed parts of wp plus wysija
38
+ if(isset($_REQUEST[WYSIJA_CRON]) || ( isset($_SERVER['argv'][1]) && $_SERVER['argv'][1]==WYSIJA_CRON )) echo '';
39
+ else exit;
40
+ $cron_schedules = get_option('wysija_schedules');
41
+
42
+ $processes = array();
43
+ if(strpos($process, ',')!==false){
44
+ $processes = explode(',', $process);
45
+ }else $processes[] = $process;
46
+
47
+ foreach($processes as $scheduleprocess){
48
+ if($scheduleprocess!='all'){
49
+ $this->check_scheduled_task($cron_schedules,$scheduleprocess);
50
+ }else{
51
+ $allProcesses = array('queue','bounce','daily','weekly','monthly');
52
+ foreach($allProcesses as $processNK){
53
+ $this->check_scheduled_task($cron_schedules,$processNK);
54
+ }
55
+ if($this->report) echo 'processed : All<br/>';
56
+ if(!isset($_REQUEST['silent'])) echo 'MailPoet\'s cron is ready. Simply setup a CRON job on your server (cpanel or other) to trigger this page.';
57
+ exit;
58
+ }
59
+ }
60
+ }
61
+ if(!isset($_REQUEST['silent'])) echo '"MailPoet\'s cron is ready. Simply setup a CRON job on your server (cpanel or other) to trigger this page.' ;
62
+ if($process) exit;
63
+ }
64
+
65
+ /**
66
+ * check that one scheduled task is ready to be executed
67
+ * @param type $cron_schedules list of recorded cron schedules
68
+ * @param type $processNK what to process queue, bounce etc...
69
+ */
70
+ function check_scheduled_task($cron_schedules,$processNK){
71
+ $helper_toolbox = WYSIJA::get('toolbox','helper');
72
+ $time_passed = $time_left = 0;
73
+ $run_scheduled = true;
74
+ $extra_text = $multisite_prefix = '';
75
+ // this is to display a different message whether we're dealing with bounce or not.
76
+ if($processNK == 'bounce'){
77
+ $model_config = WYSIJA::get( 'config' , 'model' );
78
+ // if premium is activated we launch the premium function
79
+ $multisite_prefix = '';
80
+ if(is_multisite()){
81
+ $multisite_prefix = 'ms_';
82
+ }
83
+
84
+ // we don't process the bounce automatically unless the option is ticked
85
+ if(!(defined('WYSIJANLP') && $model_config->getValue( $multisite_prefix . 'bounce_process_auto' )) ){
86
+ $extra_text = ' (bounce handling not activated)';
87
+ $run_scheduled=false;
88
+ }
89
+
90
+ }
91
+
92
+ // calculate the time passed processing a scheduled task
93
+ if(!empty($cron_schedules[$processNK]['running'])){
94
+ $time_passed = time()- $cron_schedules[$processNK]['running'];
95
+ $time_passed = $helper_toolbox->duration_string($time_passed,true,2,5);
96
+ }else{
97
+ $time_left = $cron_schedules[$processNK]['next_schedule'] - time();
98
+ $time_left = $helper_toolbox->duration_string($time_left,true,2,5);
99
+ }
100
+
101
+ if($run_scheduled && $cron_schedules[$processNK]['next_schedule'] < time() && !$cron_schedules[$processNK]['running']){
102
+ if($this->report) echo 'exec process '.$processNK.'<br/>';
103
+ $this->run_scheduled_task($processNK);
104
+ }else{
105
+ if($this->report){
106
+ if($time_passed) $text_time = ' running since : '.$time_passed;
107
+ else $text_time = ' next run : '.$time_left;
108
+ if(!empty($extra_text)) $text_time = $extra_text;
109
+ echo 'skip process <strong>'.$processNK.'</strong>'.$text_time.'<br/>';
110
+ }
111
+ }
112
+ }
113
+
114
+ /**
115
+ * run process if it's not detected as already running
116
+ * @param type $process
117
+ * @return type
118
+ */
119
+ function run_scheduled_task($process = 'queue'){
120
+ //first let's make sure that the process asked to be run is not already running
121
+ $scheduled_times = WYSIJA::get_cron_schedule($process);
122
+ $processes = WYSIJA::get_cron_frequencies();
123
+ $process_frequency = $processes[$process];
124
+
125
+ // check if the scheduled task is already being processed,
126
+ // we consider it timed out once the started running time plus the frequency has been passed
127
+ if(!empty($scheduled_times['running']) && ($scheduled_times['running'] + $process_frequency) > time()){
128
+ if($this->report) echo 'already running : '.$process.'<br/>';
129
+ return;
130
+ }
131
+
132
+ // set schedule as running
133
+ WYSIJA::set_cron_schedule($process,0,time());
134
+
135
+ // execute schedule
136
+ switch($process){
137
+ case 'queue':
138
+ // check if there are any scheduled newsletters ready for action
139
+ WYSIJA::check_scheduled_newsletters();
140
+
141
+ // if premium is activated we execute the premium cron process
142
+ if(defined('WYSIJANLP')){
143
+ $helper_premium = WYSIJA::get('premium', 'helper', false, WYSIJANLP);
144
+ $helper_premium->croned_queue_process();
145
+ }else{
146
+ // run the standard queue process no scheduled tasks will be check since it has already been checked above
147
+ WYSIJA::croned_queue(false);
148
+ }
149
+ break;
150
+ case 'bounce':
151
+ $helper_premium = WYSIJA::get('premium', 'helper', false, WYSIJANLP);
152
+ $model_config = WYSIJA::get( 'config' , 'model' );
153
+ // if premium is activated we launch the premium function
154
+ if(is_multisite()){
155
+ $multisite_prefix='ms_';
156
+ }
157
+
158
+ // we don't process the bounce automatically unless the option is ticked
159
+ if(defined('WYSIJANLP') && $model_config->getValue( $multisite_prefix . 'bounce_process_auto' )){
160
+ $helper_premium->croned_bounce();
161
+ }else{
162
+ $process .= ' (bounce handling not activated)';
163
+ }
164
+
165
+ break;
166
+ case 'daily':
167
+ WYSIJA::croned_daily();
168
+ break;
169
+ case 'weekly':
170
+ if(defined('WYSIJANLP')){
171
+ $helper_premium = WYSIJA::get('premium', 'helper', false, WYSIJANLP);
172
+ $helper_premium->croned_weekly();
173
+ }
174
+ WYSIJA::croned_weekly();
175
+ break;
176
+ case 'monthly':
177
+ WYSIJA::croned_monthly();
178
+ break;
179
+ }
180
+ // set next_schedule details
181
+ WYSIJA::set_cron_schedule($process);
182
+ if($this->report) echo 'processed : '.$process.'<br/>';
183
+ }
184
+ }
helpers/dividers.php ADDED
@@ -0,0 +1,64 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_dividers extends WYSIJA_object {
4
+
5
+ function WYSIJA_help_dividers() {
6
+ }
7
+
8
+ /**
9
+ * Returns all dividers
10
+ * @return array
11
+ */
12
+ function getAll() {
13
+ $fileHelper = WYSIJA::get('file', 'helper');
14
+ $dirHandle = $fileHelper->exists('dividers');
15
+
16
+ if($dirHandle['result'] === FALSE) {
17
+ return array();
18
+ } else {
19
+ $dividers = array();
20
+ $files = scandir($dirHandle['file']);
21
+ foreach($files as $filename) {
22
+ // don't add meta files
23
+ if(in_array($filename, array('.', '..', '.DS_Store', 'Thumbs.db')) === FALSE) {
24
+ // get dimensions of image
25
+ $dimensions = @getimagesize($dirHandle['file'].DS.$filename);
26
+ if($dimensions !== FALSE) {
27
+ $width = (int)$dimensions[0];
28
+ $height = (int)$dimensions[1];
29
+ } else {
30
+ $width = 564;
31
+ $height = 1;
32
+ }
33
+
34
+ // only add divider if height is superior to 0
35
+ if($height > 0) {
36
+ $ratio = round(($width / $height) * 1000) / 1000;
37
+ $width = min($width, 564);
38
+ $height = (int)($width / $ratio);
39
+
40
+ $dividers[] = array(
41
+ 'src' => $fileHelper->url($filename, 'dividers'),
42
+ 'width' => $width,
43
+ 'height' => $height
44
+ );
45
+ }
46
+ }
47
+ }
48
+ return $dividers;
49
+ }
50
+ }
51
+
52
+ /**
53
+ * Get default divider
54
+ * @return array
55
+ */
56
+ function getDefault() {
57
+ $fileHelper = WYSIJA::get('file', 'helper');
58
+ return array(
59
+ 'src' => $fileHelper->url('solid.jpg', 'dividers'),
60
+ 'width' => 564,
61
+ 'height' => 1
62
+ );
63
+ }
64
+ }
helpers/email.php ADDED
@@ -0,0 +1,166 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_email extends WYSIJA_object{
4
+
5
+ function WYSIJA_help_email(){
6
+
7
+ }
8
+
9
+
10
+ /**
11
+ * used to strip the unsubscribe links and the view in browser links from an email html
12
+ * @param type $content
13
+ * @return type
14
+ */
15
+ function stripPersonalLinks($content){
16
+
17
+ //delete the view in browser span
18
+ $content = preg_replace('#<td id="wysija_viewbrowser_content"[^>]*>(.*)</td>#Uis','',$content);
19
+
20
+ //delete the unsubscribe td
21
+ $content = preg_replace('#<td id="wysija_unsubscribe_content"[^>]*>(.*)</td>#Uis','',$content);
22
+
23
+ return $content;
24
+
25
+ }
26
+
27
+ /**
28
+ *
29
+ * @param array $values configuration form's current values
30
+ * @param boolean $testMultisite test the multisite configuration
31
+ * @return boolean
32
+ */
33
+ function send_test_mail($values,$testMultisite=false){
34
+ $content_email=__('Yup, it works. You can start blasting away emails to the moon.',WYSIJA);
35
+
36
+ $options=array(
37
+ 'sending_method'=>'sending_method',
38
+ 'sending_emails_site_method'=>'sending_emails_site_method',
39
+ 'sendmail_path'=>'sendmail_path',
40
+ 'smtp_rest'=>'smtp_rest',
41
+ 'smtp_host'=>'smtp_host',
42
+ 'smtp_port'=>'smtp_port',
43
+ 'smtp_secure'=>'smtp_secure',
44
+ 'smtp_auth'=>'smtp_auth',
45
+ );
46
+
47
+ //add a prefix to each option if we are in ms test case
48
+ if($testMultisite){
49
+ $is_multisite=is_multisite();
50
+
51
+ //$is_multisite=true;//PROD comment that line
52
+ if(!$is_multisite) return false;
53
+ foreach($options as &$option) $option='ms_'.$option;
54
+ }
55
+
56
+ switch($values[$options['sending_method']]){
57
+ case 'site':
58
+ if($values[$options['sending_emails_site_method']]=='phpmail'){
59
+ $send_method='PHP Mail';
60
+ }else{
61
+ $send_method='Sendmail';
62
+ $sendmail_path=$_POST['data']['wysija[config]['.$options['sendmail_path'].']'];
63
+ }
64
+ break;
65
+ case 'smtp':
66
+ $smtp=array();
67
+ $send_method='SMTP';
68
+ $config=WYSIJA::get('config','model');
69
+ if(!isset($values[$options['smtp_rest']])) unset($config->values[$options['smtp_rest']]);
70
+ break;
71
+ case 'gmail':
72
+ $send_method='Gmail';
73
+
74
+ $values[$options['smtp_host']]='smtp.gmail.com';
75
+ $values[$options['smtp_port']]='465';
76
+ $values[$options['smtp_secure']]='ssl';
77
+ $values[$options['smtp_auth']]=true;
78
+
79
+ $content_email=__('You\'re all setup! You\'ve successfully sent with Gmail.',WYSIJA).'<br/><br/>';
80
+ $content_email.=str_replace(
81
+ array('[link]','[/link]'),
82
+ array('<a href="http://support.mailpoet.com/knowledgebase/send-with-smtp-when-using-a-professional-sending-provider/?utm_source=wpadmin&utm_campaign=test_email_result" target="_blank" title="SendGrid partnership">','</a>'),
83
+ __('Looking for a faster method to send? [link]Read more[/link] on sending with a professional SMTP.',WYSIJA));
84
+ break;
85
+ }
86
+
87
+ $mailer=WYSIJA::get('mailer','helper');
88
+ $mailer->WYSIJA_help_mailer('',$values,$testMultisite);
89
+
90
+ $current_user=WYSIJA::wp_get_userdata();
91
+ $mailer->testemail=true;
92
+ $mailer->wp_user=&$current_user->data;
93
+
94
+ $res=$mailer->sendSimple($current_user->data->user_email,str_replace('[send_method]',$send_method,__('[send_method] works with MailPoet',WYSIJA)),$content_email);
95
+
96
+ if($res){
97
+ $this->notice(sprintf(__('Test email successfully sent to %s',WYSIJA),'<b><i>'.$current_user->data->user_email.'</i></b>'));
98
+ return true;
99
+ }else{
100
+ $config=WYSIJA::get('config','model');
101
+ $bounce = $config->getValue('bounce_email');
102
+ if(in_array($config->getValue('sending_method'),array('smtp','gmail')) && $config->getValue('smtp_secure')=='ssl' && !function_exists('openssl_sign')){
103
+ $this->error(__('The PHP Extension openssl is not enabled on your server. Ask your host to enable it if you want to use an SSL connection.',WYSIJA));
104
+ }elseif(!empty($bounce) AND !in_array($config->getValue('sending_method'),array('smtp_com','elasticemail'))){
105
+ $this->error(sprintf(__('The bounce email address "%1$s" might actually cause the problem. Leave the field empty and try again.',WYSIJA),$bounce));
106
+ //Case 2 : you are using SMTP but you didn't add an authentication
107
+ }elseif(in_array($config->getValue('sending_method'),array('smtp','gmail')) AND !$config->getValue('smtp_auth') AND strlen($config->getValue('smtp_password')) > 1){
108
+ $this->error(__('You specified an SMTP password but you don\'t require an authentication, you might want to turn the SMTP authentication ON.',WYSIJA));
109
+ //Case 3 : you are on localhost!
110
+ }elseif((strpos(WYSIJA_URL,'localhost') || strpos(WYSIJA_URL,'127.0.0.1')) && in_array($config->getValue('sending_method'),array('sendmail','qmail','mail'))){
111
+ $this->error(__('Your localhost may not have a mail server. To verify, please log out and click on the "Lost your password?" link on the login page. Do you receive the reset password email from your WordPress?',WYSIJA));
112
+ }
113
+
114
+ $this->error($mailer->reportMessage);
115
+ return false;
116
+ }
117
+ }
118
+
119
+ /**
120
+ * get view in browser link
121
+ * @param array/stdClass $data_email
122
+ * @return string url
123
+ */
124
+ function getVIB($data_email){
125
+ if (!is_array($data_email) && is_object($data_email))
126
+ $data_email = (array) $data_email;
127
+ if(false && isset($data_email['params']['vib_id'])) return WYSIJA::get_permalink($data_email['params']['vib_id'],false);
128
+ else{
129
+ $paramsurl=array(
130
+ 'wysija-page'=>1,
131
+ 'controller'=>'email',
132
+ 'action'=>'view',
133
+ 'email_id'=>$data_email['email_id']
134
+ );
135
+
136
+ $modelConf=WYSIJA::get('config','model');
137
+ return WYSIJA::get_permalink($modelConf->getValue('confirm_email_link'),$paramsurl);
138
+ }
139
+ }
140
+
141
+ /**
142
+ * return an array of active follow_ups
143
+ * @param type $data
144
+ * @param boolean $delay calculate delay and add it to the result
145
+ */
146
+ function get_active_follow_ups($data=array('subject','params'),$delay=false){
147
+ if($delay) $model_queue=WYSIJA::get('queue','model');
148
+
149
+ $model_email=WYSIJA::get('email','model');
150
+ $model_email->setConditions(array('type'=>2,'status'=>99));
151
+ $automatic_emails=$model_email->getRows($data);
152
+
153
+ $follow_ups_per_list=array();
154
+ foreach($automatic_emails as &$auto_email){
155
+ $model_email->getParams($auto_email);
156
+ if($delay) $auto_email['delay']=$model_queue->calculate_delay($auto_email['params']['autonl']);
157
+ if(isset($auto_email['params']['autonl']['event']) && $auto_email['params']['autonl']['event']=='subs-2-nl'){
158
+ if(!isset($follow_ups_per_list[$auto_email['params']['autonl']['subscribetolist']])) $follow_ups_per_list[$auto_email['params']['autonl']['subscribetolist']]=array();
159
+ $follow_ups_per_list[$auto_email['params']['autonl']['subscribetolist']][]=$auto_email;
160
+ }
161
+ }
162
+
163
+ return $follow_ups_per_list;
164
+ }
165
+
166
+ }
helpers/encoding.php ADDED
@@ -0,0 +1,40 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_encoding{
4
+ function change($data,$input,$output){
5
+ $input = strtoupper(trim($input));
6
+ $output = strtoupper(trim($output));
7
+ if($input == $output) return $data;
8
+ if ($input == 'UTF-8' && $output == 'ISO-8859-1'){
9
+ $data = str_replace(array('€','„','“'),array('EUR','"','"'),$data);
10
+ }
11
+ if (function_exists('iconv')){
12
+ set_error_handler('acymailing_error_handler_encoding');
13
+ $encodedData = iconv($input, $output."//IGNORE", $data);
14
+ restore_error_handler();
15
+ if(!acymailing_error_handler_encoding('result')){
16
+ return $encodedData;
17
+ }
18
+ }
19
+ if (function_exists('mb_convert_encoding')){
20
+ return mb_convert_encoding($data, $output, $input);
21
+ }
22
+ if ($input == 'UTF-8' && $output == 'ISO-8859-1'){
23
+ return utf8_decode($data);
24
+ }
25
+ if ($input == 'ISO-8859-1' && $output == 'UTF-8'){
26
+ return utf8_encode($data);
27
+ }
28
+ return $data;
29
+ }
30
+ }//endclass
31
+ function acymailing_error_handler_encoding($errno,$errstr=''){
32
+ static $error = false;
33
+ if(is_string($errno) && $errno=='result'){
34
+ $currentError = $error;
35
+ $error = false;
36
+ return $currentError;
37
+ }
38
+ $error = true;
39
+ return true;
40
+ }
helpers/file.php ADDED
@@ -0,0 +1,271 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_file extends WYSIJA_object{
4
+
5
+ function WYSIJA_help_file(){
6
+
7
+ }
8
+
9
+ /**
10
+ * Get the full path of a file
11
+ * @param type $csvfilename
12
+ * @param type $folder
13
+ * @return boolean
14
+ */
15
+ function exists($fileFolder=false){
16
+ $upload_base_dir = $this->getUploadBaseDir();
17
+
18
+ $filename=str_replace('/',DS,$upload_base_dir).DS.'wysija'.DS.$fileFolder;
19
+ if(!file_exists($filename)){
20
+ return array('result'=>false,'file'=>$filename);
21
+ }
22
+
23
+ return array('result'=>true,'file'=>$filename);
24
+ }
25
+
26
+ /**
27
+ * Get the full path of a file
28
+ * @param type $csvfilename
29
+ * @param type $folder
30
+ * @return boolean
31
+ */
32
+ function get($csvfilename,$folder='temp'){
33
+ $upload_base_dir = $this->getUploadBaseDir();
34
+
35
+ $filename=$upload_base_dir.DS.'wysija'.DS.$folder.DS.$csvfilename;
36
+ if(!file_exists($filename)){
37
+ $filename=$upload_base_dir.DS.$csvfilename;
38
+ if(!file_exists($filename)) $filename=false;
39
+ }
40
+
41
+ return $filename;
42
+ }
43
+
44
+ // Description: create a directory recursively if possible
45
+ // Parameters: (Name of the directory, permissions)
46
+ // Returns: Directory path. False if impossible to create folder.
47
+ function makeDir($folder='temp',$mode=0755){
48
+ $upload_base_dir = $this->getUploadBaseDir();
49
+
50
+ if(strpos(str_replace('/',DS,$folder),str_replace('/',DS,$upload_base_dir))!==false){
51
+ $dirname=$folder;
52
+ }else{
53
+ $dirname=$upload_base_dir.DS.'wysija'.DS.$folder.DS;
54
+ }
55
+ if(!file_exists($dirname)){
56
+ if(!mkdir($dirname, $mode,true)){
57
+ $this->error('Cannot create folder '.$dirname.' try to create the folder manually');
58
+ return false;
59
+ }
60
+ chmod($dirname,$mode);
61
+ }
62
+ return $dirname;
63
+ }
64
+
65
+
66
+ function getUploadDir($folder=false){
67
+ $upload_base_dir = $this->getUploadBaseDir();
68
+
69
+ $dirname=$upload_base_dir.DS.'wysija'.DS;
70
+ if($folder) $dirname.=$folder.DS;
71
+ if(file_exists($dirname)) return $dirname;
72
+ return false;
73
+ }
74
+
75
+ function getUploadBaseDir(){
76
+ $upload_dir = wp_upload_dir();
77
+
78
+ if(!isset($upload_dir['basedir'])){
79
+ if(isset($upload_dir['error'])) $this->wp_error('<b>WordPress error</b> : '.$upload_dir['error'],1);
80
+ return false;
81
+ }
82
+
83
+ //having .. in a path is not safe as it can lead to some parent path where we don't have control
84
+ if(strpos($upload_dir['basedir'], '..')!==false){
85
+ $pathsections=$pathsectionsc=explode(DS, $upload_dir['basedir']);
86
+
87
+ while($key = array_search('..', $pathsections)){
88
+ unset($pathsections[$key]);
89
+ unset($pathsections[$key-1]);
90
+ $newpatharray=array();
91
+ foreach($pathsections as $ky=>$vy){
92
+ $newpatharray[]=$vy;
93
+ }
94
+ $pathsections=$newpatharray;
95
+ }
96
+ $cleanBaseDir=implode(DS, $pathsections);
97
+
98
+ if(file_exists($cleanBaseDir)){
99
+ $upload_dir['basedir']=$cleanBaseDir;
100
+ }
101
+ }
102
+
103
+
104
+ return $upload_dir['basedir'];
105
+ }
106
+
107
+ /**
108
+ * make a temporary file
109
+ * @param type $content
110
+ * @param type $key
111
+ * @param type $format
112
+ * @return type
113
+ */
114
+ function temp($content,$key='temp',$format='.tmp'){
115
+ $tempDir=$this->makeDir();
116
+
117
+ if(!$tempDir) return false;
118
+
119
+
120
+ $filename=$key.'-'.time().$format;
121
+ $handle=fopen($tempDir.$filename, 'w');
122
+ fwrite($handle, $content);
123
+ fclose($handle);
124
+
125
+ return array('path'=>$tempDir.$filename,'name'=>$filename, 'url'=>$this->url($filename,'temp'));
126
+ }
127
+
128
+ /**
129
+ * Get the url of a wysija file based on the filename and the wysija folder
130
+ * @param type $filename
131
+ * @param type $folder
132
+ * @return string
133
+ */
134
+ function url($filename,$folder='temp'){
135
+ $upload_dir = wp_upload_dir();
136
+
137
+ if(file_exists($upload_dir['basedir'].DS.'wysija')){
138
+ $url=$upload_dir['baseurl'].'/wysija/'.$folder.'/'.$filename;
139
+ }else{
140
+ $url=$upload_dir['baseurl'].'/'.$filename;
141
+ }
142
+
143
+ return str_replace(DS,'/',$url);
144
+ }
145
+
146
+ /**
147
+ * send file to be downloaded
148
+ * @param type $path
149
+ */
150
+ function send($path){
151
+ /* submit the file to the admin */
152
+ if(file_exists($path)){
153
+ header('Content-type: application/csv');
154
+ header('Content-Disposition: attachment; filename="export_wysija.csv"');
155
+ readfile($path);
156
+ exit();
157
+ }else $this->error(__('Yikes! We couldn\'t export. Make sure that your folder permissions for /wp-content/uploads/wysija/temp is set to 755.',WYSIJA),true);
158
+
159
+ }
160
+
161
+ /*
162
+ *
163
+ */
164
+ function clear(){
165
+ $foldersToclear=array("import","temp");
166
+ $filenameRemoval=array("import-","export-");
167
+ $deleted=array();
168
+ foreach($foldersToclear as $folder){
169
+ $path=$this->getUploadDir($folder);
170
+ /* get a list of files from this folder and clear them */
171
+ if(!$path) continue;
172
+ $files = scandir($path);
173
+ foreach($files as $filename){
174
+ if(!in_array($filename, array('.','..',".DS_Store","Thumbs.db"))){
175
+ if(preg_match('/('.implode($filenameRemoval,'|').')[0-9]*\.csv/',$filename,$match)){
176
+ $deleted[]=$path.$filename;
177
+
178
+ }
179
+ }
180
+ }
181
+ }
182
+ foreach($deleted as $filename){
183
+ if(file_exists($filename)){
184
+ $filename=str_replace('/',DS,$filename);
185
+ unlink($filename);
186
+ }
187
+ }
188
+
189
+ }
190
+
191
+ function rrmdir($dir) {
192
+ if(strpos($dir, '..')!==false){
193
+ $this->error('Path is not safe, cannot contain ..');
194
+ return false;
195
+ }
196
+ if (is_dir($dir)) {
197
+ $files = scandir($dir);
198
+ foreach ($files as $file){
199
+ if ($file != "." && $file != "..") $this->rrmdir("$dir".DS."$file");
200
+ }
201
+
202
+ if(!rmdir($dir)){
203
+ chmod($dir, 0777);
204
+ rmdir($dir);
205
+ }
206
+
207
+
208
+ }
209
+ else if (file_exists($dir)) {
210
+ $dir=str_replace('/',DS,$dir);
211
+ unlink($dir);
212
+ }
213
+ }
214
+
215
+ function rcopy($src, $dst) {
216
+ if(strpos($src, '..')!==false || strpos($dst, '..')!==false){
217
+ $this->error('src : '.$src);
218
+ $this->error('dst : '.$dst);
219
+ $this->error('Path is not safe, cannot contain ..');
220
+ return false;
221
+ }else{
222
+ if (file_exists($dst)) $this->rrmdir($dst);
223
+ }
224
+
225
+ if (is_dir($src)) {
226
+ mkdir($dst);
227
+ $files = scandir($src);
228
+ foreach ($files as $file){
229
+ if ($file != "." && $file != "..") $this->rcopy("$src/$file", "$dst/$file");
230
+ }
231
+
232
+ }
233
+ else if (file_exists($src)) {
234
+ copy(str_replace('/',DS,$src), str_replace('/',DS,$dst));
235
+ }
236
+ }
237
+
238
+ /*
239
+ * taken from php.net
240
+ */
241
+ function chmodr($path, $filemode=0644, $dirmode=0755) {
242
+ if (is_dir($path) ) {
243
+ if (!chmod($path, $dirmode)) {
244
+ $dirmode_str=decoct($dirmode);
245
+ print "Failed applying filemode '$dirmode_str' on directory '$path'\n";
246
+ print " `-> the directory '$path' will be skipped from recursive chmod\n";
247
+ return;
248
+ }
249
+ $dh = opendir($path);
250
+ while (($file = readdir($dh)) !== false) {
251
+ if($file != '.' && $file != '..') { // skip self and parent pointing directories
252
+ $fullpath = $path.DS.$file;
253
+ $this->chmodr($fullpath, $filemode,$dirmode);
254
+ }
255
+ }
256
+ closedir($dh);
257
+ } else {
258
+ if (is_link($path)) {
259
+ print "link '$path' is skipped\n";
260
+ return;
261
+ }
262
+ if (!chmod($path, $filemode)) {
263
+ $filemode_str=decoct($filemode);
264
+ print "Failed applying filemode '$filemode_str' on file '$path'\n";
265
+ return;
266
+ }
267
+ }
268
+ }
269
+
270
+ }
271
+
helpers/form_engine.php ADDED
@@ -0,0 +1,842 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ /**
4
+ * @class Wysija Engine Helper (PHP5 version)
5
+ */
6
+ class WYSIJA_help_form_engine extends WYSIJA_object {
7
+ // debug mode
8
+ private $_debug = false;
9
+
10
+ // rendering context (editor, web)
11
+ private $_context = 'editor';
12
+ // rendering mode (live, preview)
13
+ private $_mode = 'live';
14
+
15
+ // data holders
16
+ private $_data = null;
17
+ private $_styles = null;
18
+ private $_lists = null;
19
+ private $_fields = null;
20
+
21
+ // static form fields
22
+ private $_static_fields = array('email', 'submit');
23
+
24
+ // unique form fields
25
+ private $_unique_fields = array('firstname', 'lastname', 'list');
26
+
27
+ // constructor
28
+ function __construct() {
29
+
30
+ }
31
+
32
+ // i18n methods
33
+ public function get_translations() {
34
+ return array(
35
+ 'savingnl' => __('Saving form...', WYSIJA),
36
+ 'save' => __('Save', WYSIJA),
37
+ 'edit_settings' => __('Edit display', WYSIJA),
38
+ 'list_cannot_be_empty' => __('You have to select at least 1 list', WYSIJA),
39
+ 'not_enough_options' => __('Your subscriber needs at least 2 options to select from', WYSIJA),
40
+ 'missing_checkbox_label' => __('You need to specify a value for your checkbox', WYSIJA),
41
+ 'add_field' => __('Add new field', WYSIJA),
42
+ 'edit_field' => __('Edit custom field', WYSIJA),
43
+ 'delete_field_confirmation' => __('Are you sure you want to delete this custom field?', WYSIJA),
44
+ 'date_select_year' => __('Year', WYSIJA),
45
+ 'date_select_month' => __('Month', WYSIJA),
46
+ 'date_select_day' => __('Day', WYSIJA)
47
+ );
48
+ }
49
+
50
+ public function get_months($options = array()) {
51
+
52
+ $defaults = array(
53
+ 'selected' => null
54
+ );
55
+ // is default today
56
+ if(isset($options['params']['is_default_today']) && (bool)$options['params']['is_default_today'] === true) {
57
+ $defaults['selected'] = (int)strftime('%m');
58
+ }
59
+
60
+ // merge options with defaults
61
+ $options = array_merge($defaults, $options);
62
+
63
+ $month_names = array(
64
+ __('January', WYSIJA),
65
+ __('February', WYSIJA),
66
+ __('March', WYSIJA),
67
+ __('April', WYSIJA),
68
+ __('May', WYSIJA),
69
+ __('June', WYSIJA),
70
+ __('July', WYSIJA),
71
+ __('August', WYSIJA),
72
+ __('September', WYSIJA),
73
+ __('October', WYSIJA),
74
+ __('November', WYSIJA),
75
+ __('December', WYSIJA)
76
+ );
77
+
78
+ $months = array();
79
+ for($i = 0; $i < 12; $i++) {
80
+ $months[] = array(
81
+ 'month' => ($i + 1),
82
+ 'month_name' => $month_names[$i],
83
+ 'is_selected' => (($i + 1) === $options['selected'])
84
+ );
85
+ }
86
+
87
+ return $months;
88
+ }
89
+
90
+ public function get_years($options = array()) {
91
+ $defaults = array(
92
+ 'selected' => null,
93
+ 'from' => (int)strftime('%Y') - 100,
94
+ 'to' => (int)strftime('%Y')
95
+ );
96
+ // is default today
97
+ if(isset($options['params']['is_default_today']) && (bool)$options['params']['is_default_today'] === true) {
98
+ $defaults['selected'] = (int)strftime('%Y');
99
+ }
100
+
101
+ // merge options with defaults
102
+ $options = array_merge($defaults, $options);
103
+
104
+ $years = array();
105
+
106
+ // return years as an array
107
+ for($i = (int)$options['from']; $i < (int)($options['to'] + 1); $i++) {
108
+ $years[] = array(
109
+ 'year' => $i,
110
+ 'is_selected' => ($i === $options['selected'])
111
+ );
112
+ }
113
+
114
+ return array_reverse($years);
115
+ }
116
+
117
+ public function get_days($options = array()) {
118
+ $defaults = array(
119
+ 'selected' => null
120
+ );
121
+ // is default today
122
+ if(isset($options['params']['is_default_today']) && (bool)$options['params']['is_default_today'] === true) {
123
+ $defaults['selected'] = (int)strftime('%d');
124
+ }
125
+
126
+ // merge options with defaults
127
+ $options = array_merge($defaults, $options);
128
+
129
+ $days = array();
130
+
131
+ // return days as an array
132
+ for($i = 1; $i < 32; $i++) {
133
+ $days[] = array(
134
+ 'day' => $i,
135
+ 'is_selected' => ($i === $options['selected'])
136
+ );
137
+ }
138
+
139
+ return $days;
140
+ }
141
+
142
+ // getters/setters
143
+ public function get_data($type = null) {
144
+ if($type !== null) {
145
+ if(array_key_exists($type, $this->_data)) {
146
+ return $this->_data[$type];
147
+ } else {
148
+ // return default value
149
+ $defaults = $this->get_default_data();
150
+ return $defaults[$type];
151
+ }
152
+ }
153
+ return $this->_data;
154
+ }
155
+
156
+ public function set_data($value = null, $decode = false) {
157
+ if(!$value) {
158
+ $this->_data = $this->get_default_data();
159
+ } else {
160
+ $this->_data = $value;
161
+ if($decode) {
162
+ $this->_data = $this->get_decoded('data');
163
+ }
164
+ }
165
+ }
166
+
167
+ public function set_lists($lists = array()) {
168
+ $this->_lists = $lists;
169
+ }
170
+
171
+ public function get_formatted_lists() {
172
+
173
+ $lists = $this->get_lists();
174
+ $formatted_lists = array();
175
+
176
+ foreach($lists as $list) {
177
+ $formatted_lists[$list['list_id']] = $list['name'];
178
+ }
179
+ return $formatted_lists;
180
+ }
181
+
182
+ public function get_lists() {
183
+ if($this->_lists === null) {
184
+ // get available lists which users can subscribe to
185
+ $model_list = WYSIJA::get('list','model');
186
+
187
+ // get lists users can subscribe to (aka "enabled list")
188
+ $lists = $model_list->get(array('name', 'list_id', 'is_public'), array('is_enabled' => 1));
189
+
190
+ $this->set_lists($lists);
191
+ }
192
+ return $this->_lists;
193
+ }
194
+
195
+ private function get_context() {
196
+ return $this->_context;
197
+ }
198
+
199
+ private function set_context($value = null) {
200
+ if($value !== null) $this->_context = $value;
201
+ }
202
+
203
+ public function set_mode($value = null) {
204
+ if($value !== null) $this->_mode = $value;
205
+ }
206
+
207
+ private function get_mode() {
208
+ return $this->_mode;
209
+ }
210
+
211
+ public function get_encoded($type = 'data') {
212
+ return base64_encode(serialize($this->{'get_'.$type}()));
213
+ }
214
+
215
+ public function get_decoded($type = 'data') {
216
+ return unserialize(base64_decode($this->{'get_'.$type}()));
217
+ }
218
+
219
+ private function get_default_data() {
220
+
221
+ $lists = $this->get_lists();
222
+
223
+ // select default list
224
+ $default_list = array();
225
+ if(!empty($lists)) {
226
+ $default_list[] = $lists[0]['list_id'];
227
+ }
228
+
229
+ return array(
230
+ 'version' => '0.4',
231
+ 'settings' => array(
232
+ 'on_success' => 'message',
233
+ 'success_message' => __('Check your inbox or spam folder now to confirm your subscription.', WYSIJA),
234
+ 'lists' => $default_list,
235
+ 'lists_selected_by' => 'admin'
236
+ ),
237
+ 'body' => array(
238
+ array(
239
+ 'name' => __('Email', WYSIJA),
240
+ 'type' => 'input',
241
+ 'field' => 'email',
242
+ 'params' => array(
243
+ 'label' => __('Email', WYSIJA),
244
+ 'required' => true
245
+ )
246
+ ),
247
+ array(
248
+ 'name' => __('Submit', WYSIJA),
249
+ 'type' => 'submit',
250
+ 'field' => 'submit',
251
+ 'params' => array(
252
+ 'label' => __('Subscribe!', WYSIJA)
253
+ )
254
+ )
255
+ )
256
+ );
257
+ }
258
+
259
+ public function get_setting($key = null) {
260
+ if($key === null) return null;
261
+
262
+ if($this->is_data_valid() === true) {
263
+ $settings = $this->get_data('settings');
264
+ if(array_key_exists($key, $settings)) {
265
+ // otherwise, simply return the value
266
+ return $settings[$key];
267
+ } else {
268
+ return null;
269
+ }
270
+ }
271
+ }
272
+
273
+ // common methods
274
+ private function is_debug() {
275
+ return ($this->_debug === true);
276
+ }
277
+
278
+ private function is_data_valid() {
279
+ return ($this->get_data() !== null);
280
+ }
281
+
282
+ public function get_custom_fields() {
283
+ if($this->_fields === null) {
284
+ // get available custom fields
285
+ $custom_fields = WJ_Field::get_all(array('order_by' => 'name ASC'));
286
+ $user_fields = array();
287
+
288
+ if(!empty($custom_fields)) {
289
+ foreach($custom_fields as $custom_field) {
290
+ $user_fields[] = array(
291
+ 'field_id' => $custom_field->id,
292
+ 'name' => $custom_field->name,
293
+ 'column_name' => $custom_field->user_column_name(),
294
+ 'column_type' => $custom_field->type,
295
+ 'params' => $custom_field->settings
296
+ );
297
+ }
298
+ }
299
+
300
+ // we need to figure out the default list for the "List selection" widget
301
+ $lists = $this->get_lists();
302
+
303
+ // select default list
304
+ $default_list = array();
305
+ if(!empty($lists)) {
306
+ $default_list[] = array(
307
+ 'list_id' => $lists[0]['list_id'],
308
+ 'is_checked' => 0
309
+ );
310
+ }
311
+
312
+ // extra widgets that can be added more than once
313
+ $extra_fields = array(
314
+ array(
315
+ 'name' => __('Divider', WYSIJA),
316
+ 'column_name' => 'divider',
317
+ 'column_type' => 'divider'
318
+ ),
319
+ array(
320
+ 'name' => __('First name', WYSIJA),
321
+ 'column_name' => 'firstname',
322
+ 'column_type' => 'input'
323
+ ),
324
+ array(
325
+ 'name' => __('Last name',WYSIJA),
326
+ 'column_name' => 'lastname',
327
+ 'column_type' => 'input'
328
+ ),
329
+ array(
330
+ 'name' => __('List selection', WYSIJA),
331
+ 'column_name' => 'list',
332
+ 'column_type' => 'list',
333
+ 'params' => array(
334
+ 'label' => __('Select list(s):', WYSIJA),
335
+ 'values' => $default_list
336
+ )
337
+ ),
338
+ array(
339
+ 'name' => __('Random text or HTML', WYSIJA),
340
+ 'column_name' => 'html',
341
+ 'column_type' => 'html',
342
+ 'params' => array(
343
+ 'text' => __('Subscribe to our newsletter and join our [total_subscribers] subscribers.', WYSIJA)
344
+ )
345
+ )
346
+ );
347
+
348
+ // set data to be passed to the view
349
+ $this->_fields = array_merge($user_fields, $extra_fields);
350
+ }
351
+
352
+ return $this->_fields;
353
+ }
354
+
355
+ // editor rendering methods
356
+ public function render_editor_toolbar() {
357
+
358
+ // get custom fields
359
+ $fields = $this->get_custom_fields();
360
+
361
+ $output = '';
362
+ $output .= '<li style="text-align:center;"><a id="wysija-add-field" class="button" href="javascript:;" href2="admin.php?page=wysija_config&action=form_widget_settings&field_id=0">'.__('Add New Field',WYSIJA).'</a></li>';
363
+
364
+ foreach($fields as $field) {
365
+ // get field type or defaults to "input"
366
+ $type = (isset($field['column_type'])) ? $field['column_type'] : 'input';
367
+
368
+ // set unique if the field type matches the unique_fields OR is a custom field
369
+ $is_unique = (bool)(in_array($field['column_name'], $this->_unique_fields) or isset($field['field_id']));
370
+
371
+ // check whether it's an actual custom field
372
+ $is_custom_field = $this->is_custom_field($field);
373
+
374
+ // actions
375
+ $actions = '';
376
+ if($is_custom_field) {
377
+ $actions = '<a class="wysija_form_item_settings settings" title="'.__('Edit field', WYSIJA).'" href="javascript:;" href2="admin.php?page=wysija_config&action=form_widget_settings&field_id='.((int)$field['field_id']).'" data-field-id="'.((int)$field['field_id']).'"><span class="dashicons dashicons-admin-generic"></span></a>';
378
+ $actions .= '<a class="wysija_form_item_delete delete" title="'.__('Delete field', WYSIJA).'" href="javascript:;" data-field-id="'.((int)$field['field_id']).'"><span class="dashicons dashicons-dismiss"></span></a>';
379
+ }
380
+
381
+ // generate html for toolbar item
382
+ $output .= '<li><a class="wysija_form_item" id="'.$field['column_name'].'" wysija_field="'.$field['column_name'].'" wysija_name="'.$field['name'].'" wysija_unique="'.$is_unique.'" wysija_type="'.$type.'">'.$field['name'].'</a>'.$actions.'</li>';
383
+ }
384
+
385
+ return $output;
386
+ }
387
+
388
+ // renders all widgets' templates
389
+ function render_editor_templates() {
390
+ $this->set_context('editor');
391
+
392
+ // get custom fields
393
+ $fields = $this->get_custom_fields();
394
+
395
+ // get parser helper
396
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
397
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
398
+
399
+ // define html output string
400
+ $output = '';
401
+
402
+ foreach($fields as $field) {
403
+ // get field type or defaults to "input"
404
+ $type = (isset($field['column_type'])) ? $field['column_type'] : 'input';
405
+
406
+ // get label from params, defaults to field name
407
+ $label = (isset($field['params']['label'])) ? $field['params']['label'] : $field['name'];
408
+
409
+ // build field data in order to pass it to the widget template
410
+ $block = array(
411
+ 'field' => $field['column_name'],
412
+ 'type' => $type,
413
+ 'name' => $field['name'],
414
+ 'unique' => (in_array($field['column_name'], $this->_unique_fields)),
415
+ 'static' => (in_array($field['column_name'], $this->_static_fields)),
416
+ 'params' => array(
417
+ 'label' => $label
418
+ ),
419
+ 'i18n' => $this->get_translations()
420
+ );
421
+
422
+ // get field extra params if specified
423
+ if(isset($field['params'])) {
424
+ // merge the params
425
+ $block['params'] = array_merge($field['params'], $block['params']);
426
+ }
427
+
428
+ // get extra data depending on field type
429
+ $block = $this->_get_extra_data($block);
430
+
431
+ // render widget templates
432
+ $output .= $helper_render_engine->render($block, 'templates/form/editor/widgets/template.html');
433
+ }
434
+ return $output;
435
+ }
436
+
437
+ private function set_lists_names($block = array()) {
438
+ // get lists using each list id as key
439
+ $lists = $this->get_formatted_lists();
440
+
441
+ if($this->get_context() === 'editor') {
442
+ $block['lists'] = $lists;
443
+ } else {
444
+ // if the block has no list, then simply return the block
445
+ if(!isset($block['params']['values']) or empty($block['params']['values'])) return $block;
446
+
447
+ $values = array();
448
+
449
+ foreach($block['params']['values'] as $list) {
450
+ // check if the list id exists in the lists
451
+ if(isset($lists[$list['list_id']])) {
452
+ $is_checked = (isset($list['is_checked']) ? (int)$list['is_checked'] : 0);
453
+ $values[] = array('name' => $lists[$list['list_id']], 'list_id' => $list['list_id'], 'is_checked' => $is_checked);
454
+ }
455
+ }
456
+
457
+ $block['params']['values'] = $values;
458
+ }
459
+
460
+ return $block;
461
+ }
462
+
463
+ // renders a single widget's template
464
+ public function render_editor_template($block = array()) {
465
+ $this->set_context('editor');
466
+
467
+ if(empty($block)) return null;
468
+
469
+ // get parser helper
470
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
471
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
472
+
473
+ // get extra data depending on field type
474
+ $block = $this->_get_extra_data($block);
475
+
476
+ $block = array_merge($block, array(
477
+ 'unique' => (in_array($block['field'], $this->_unique_fields)),
478
+ 'static' => (in_array($block['field'], $this->_static_fields)),
479
+ 'i18n' => $this->get_translations()
480
+ ));
481
+
482
+ // render widget templates
483
+ return $helper_render_engine->render($block, 'templates/form/editor/widgets/template.html');
484
+ }
485
+
486
+ public function refresh_custom_field($form_id = null, $field = array()) {
487
+ if($form_id === null or empty($field)) return null;
488
+
489
+ // check if refreshing the field is necessary
490
+ if($this->is_custom_field($field)) {
491
+ // get all forms
492
+ $model_forms = WYSIJA::get('forms', 'model');
493
+ $forms = $model_forms->getRows();
494
+
495
+ $updated_block = null;
496
+
497
+ // check if there's at least one form otherwise it's kinda pointless
498
+ if(is_array($forms) && count($forms) > 0) {
499
+ // loop over each form
500
+ foreach ($forms as $i => $form) {
501
+ $requires_update = false;
502
+
503
+ // decode form data
504
+ $data = unserialize(base64_decode($form['data']));
505
+
506
+ // loop through each block
507
+ foreach ($data['body'] as $j => $block) {
508
+ // in case we find an instance of the custom field that needs to be updated
509
+ if($block['field'] === $field['field']) {
510
+ $updated_params = $field['settings'];
511
+ $display_fields = array('label', 'label_within', 'lines');
512
+
513
+ // apply block display options
514
+ foreach($display_fields as $display_field) {
515
+ if(array_key_exists($display_field, $block['params'])) {
516
+ $updated_params[$display_field] = $block['params'][$display_field];
517
+ }
518
+ }
519
+
520
+ // apply new parameters
521
+ $data['body'][$j]['params'] = $updated_params;
522
+ // set flag in order to save changes
523
+ $requires_update = true;
524
+
525
+ // if it's in the current form, we need to return an updated version of this block
526
+ if((int)$form['form_id'] === $form_id) {
527
+ $updated_block = $data['body'][$j];
528
+ }
529
+ }
530
+ }
531
+
532
+ // if the form requires update, let's do it
533
+ if($requires_update === true) {
534
+ $model_forms->reset();
535
+ $model_forms->update(array('data' => base64_encode(serialize($data))), array('form_id' => (int)$form['form_id']));
536
+ }
537
+ }
538
+ }
539
+
540
+ // return false if there's no need to update any block in the current form
541
+ return $updated_block;
542
+ }
543
+ }
544
+
545
+ private function _get_extra_data($block = array()) {
546
+ // special case for lists
547
+ if($block['field'] === 'list') {
548
+ $block = $this->set_lists_names($block);
549
+ }
550
+
551
+ // special case for "date" types
552
+ if($block['type'] === 'date') {
553
+ $block['days'] = $this->get_days($block);
554
+ $block['months'] = $this->get_months($block);
555
+ $block['years'] = $this->get_years($block);
556
+
557
+ $display_date_fields = explode('_', $block['params']['date_type']);
558
+
559
+ // date order
560
+ $date_orders = array(
561
+ 'year_month_day' => array('mm/dd/yyyy', 'dd/mm/yyyy', 'yyyy/mm/dd'),
562
+ 'year_month' => array('mm/yyyy', 'yyyy/mm'),
563
+ 'year' => array('yyyy'),
564
+ 'month' => array('mm')
565
+ );
566
+
567
+ if(isset($block['params']['date_order']) && in_array($block['params']['date_order'], $date_orders[$block['params']['date_type']])) {
568
+ $fields = explode('/', $block['params']['date_order']);
569
+ } else {
570
+ $fields = explode('/', $date_orders[$block['params']['date_type']][0]);
571
+ }
572
+ $block['date_fields'] = $fields;
573
+ }
574
+
575
+ return $block;
576
+ }
577
+
578
+ // renders the editor
579
+ function render_editor() {
580
+ $this->set_context('editor');
581
+
582
+ if($this->is_data_valid() === false) {
583
+ throw new Exception('data is not valid');
584
+ } else {
585
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
586
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
587
+
588
+ $data = array(
589
+ 'body' => $this->render_editor_body(),
590
+ 'is_debug' => $this->is_debug(),
591
+ 'i18n' => $this->get_translations()
592
+ );
593
+
594
+ return $helper_render_engine->render($data, 'templates/form/editor/template.html');
595
+ }
596
+ }
597
+
598
+ // renders editor's body
599
+ function render_editor_body() {
600
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
601
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
602
+
603
+ $blocks = $this->get_data('body');
604
+ if(empty($blocks)) return '';
605
+
606
+ $body = '';
607
+ foreach($blocks as $block) {
608
+ // get extra data depending on field type
609
+ $block = $this->_get_extra_data($block);
610
+
611
+ // generate block template
612
+ $data = array_merge($block, array(
613
+ 'unique' => (in_array($block['field'], $this->_unique_fields)),
614
+ 'static' => (in_array($block['field'], $this->_static_fields)),
615
+ 'i18n' => $this->get_translations())
616
+ );
617
+
618
+ $body .= $helper_render_engine->render($data, 'templates/form/editor/widgets/template.html');
619
+ }
620
+
621
+ return $body;
622
+ }
623
+
624
+ // web rendering methods
625
+ public function render_web($data = array()) {
626
+ $this->set_context('web');
627
+
628
+ if($this->is_data_valid() === false) {
629
+ throw new Exception('data is not valid');
630
+ } else {
631
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
632
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
633
+
634
+ $data = array(
635
+ 'preview' => ($this->get_mode() === 'preview'),
636
+ 'settings' => $this->get_data('settings'),
637
+ 'body' => $this->render_web_body()
638
+ );
639
+
640
+ // in live mode, we need to specify the form id
641
+ if($this->get_mode() === 'live') {
642
+ $data['form_id'] = (int)$this->get_data('form_id');
643
+ }
644
+
645
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
646
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
647
+
648
+ // make sure we get the messages
649
+ $output = '';
650
+
651
+ $posted_form = (isset($_POST['form_id']) && (int)$_POST['form_id'] > 0) ? (int)$_POST['form_id'] : 0;
652
+
653
+ if($data['form_id'] === $posted_form) {
654
+ $view = WYSIJA::get('widget_nl','view','front');
655
+ if(count($view->getMsgs()) > 0) {
656
+ $output .= $view->messages();
657
+ }
658
+ }
659
+
660
+ try {
661
+ $output .= $helper_render_engine->render($data, 'templates/form/web/template.html');
662
+ return $output;
663
+ } catch(Exception $e) {
664
+ return '';
665
+ }
666
+ }
667
+ }
668
+
669
+ protected function get_validation_class($block) {
670
+ $rules = array();
671
+
672
+ // if it's the email field, it's mandatory and needs to be valid
673
+ if($block['field'] === 'email') {
674
+ $rules[] = 'required';
675
+ $rules[] = 'custom[email]';
676
+ }
677
+
678
+ // if it's the list field, at least one option needs to be selected
679
+ if($block['field'] === 'list') {
680
+ $rules[] = 'required';
681
+ }
682
+
683
+ // check if the field is required
684
+ if(isset($block['params']['required']) && (bool)$block['params']['required'] === true) {
685
+ $rules[] = 'required';
686
+ }
687
+
688
+ // check for validation rules
689
+ if(isset($block['params']['validate'])) {
690
+ if(is_array($block['params']['validate'])) {
691
+ // handle multiple validation rules
692
+ foreach($block['params']['validate'] as $rule) {
693
+ $rules[] = 'custom['.$rule.']';
694
+ }
695
+ } else if(strlen(trim($block['params']['validate'])) > 0) {
696
+ // handle single validation rule
697
+ $rules[] = 'custom['.$block['params']['validate'].']';
698
+ }
699
+ }
700
+
701
+ // generate string if there is at least one rule to validate against
702
+ if(empty($rules)) {
703
+ return '';
704
+ } else {
705
+ // make sure rules are not duplicated
706
+ $rules = array_unique($rules);
707
+ return 'validate['.join(',', $rules).']';
708
+ }
709
+ }
710
+
711
+ protected function render_web_body() {
712
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
713
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
714
+
715
+ $blocks = $this->get_data('body');
716
+
717
+ if(empty($blocks)) return '';
718
+
719
+ $body = '';
720
+ foreach($blocks as $key => $block) {
721
+ // get extra data depending on field type
722
+ $block = $this->_get_extra_data($block);
723
+
724
+ // special case for email widget
725
+ if($block['field'] === 'email') {
726
+ $user_email = WYSIJA::wp_get_userdata('user_email');
727
+
728
+ if($user_email && is_string($user_email) && is_user_logged_in() && !current_user_can('switch_themes') && !is_admin()) {
729
+ $block['value'] = $user_email;
730
+ }
731
+ }
732
+
733
+ // set field name 'prefix' depending whether it's a custom field or not
734
+ if($this->is_custom_field($block)) {
735
+ $field_prefix = 'wysija[field]';
736
+ } else {
737
+ $field_prefix = 'wysija[user]';
738
+ }
739
+
740
+ // generate block template
741
+ $data = array_merge($block, array(
742
+ 'field_prefix' => $field_prefix,
743
+ 'preview' => ($this->get_mode() === 'preview'),
744
+ 'i18n' => $this->get_translations(),
745
+ 'validation' => $this->get_validation_class($block)
746
+ ));
747
+ $body .= $helper_render_engine->render($data, 'templates/form/web/widgets/template.html');
748
+ }
749
+
750
+ return $body;
751
+ }
752
+
753
+ public function is_custom_field($field = array()) {
754
+ return (bool)(isset($field['field_id']) || (isset($field['field']) && strpos($field['field'], 'cf_') === 0));
755
+ }
756
+
757
+ public function get_exports($form_id) {
758
+ return array(
759
+ 'iframe' => base64_encode($this->export($form_id, 'iframe')),
760
+ 'php' => base64_encode($this->export($form_id, 'php')),
761
+ 'html' => base64_encode($this->export($form_id, 'html'))
762
+ );
763
+ }
764
+
765
+ public function render_editor_export($form_id) {
766
+ $helper_render_engine = WYSIJA::get('render_engine', 'helper');
767
+ $helper_render_engine->setTemplatePath(WYSIJA_EDITOR_TOOLS);
768
+
769
+ $data = array(
770
+ 'types' => array(
771
+ 'iframe' => $this->export($form_id, 'iframe'),
772
+ 'php' => $this->export($form_id, 'php'),
773
+ 'html' => $this->export($form_id, 'html'),
774
+ 'shortcode' => $this->export($form_id, 'shortcode')
775
+ )
776
+ );
777
+
778
+ return $helper_render_engine->render($data, 'templates/form/web/export.html');
779
+ }
780
+
781
+ public function export($form_id, $type) {
782
+ switch($type) {
783
+ case 'iframe':
784
+ $url_params = array(
785
+ 'wysija-page' => 1,
786
+ 'controller' => 'subscribers',
787
+ 'action' => 'wysija_outter',
788
+ 'wysija_form' => $form_id
789
+ );
790
+
791
+ $url_params['external_site'] = 1;
792
+
793
+ $model_config = WYSIJA::get('config','model');
794
+ $source_url = WYSIJA::get_permalink($model_config->getValue('confirm_email_link'), $url_params, true);
795
+
796
+ return '<iframe width="100%" scrolling="no" frameborder="0" src="'.$source_url.'" class="iframe-wysija" vspace="0" tabindex="0" style="position: static; top: 0pt; margin: 0px; border-style: none; height: 330px; left: 0pt; visibility: visible;" marginwidth="0" marginheight="0" hspace="0" allowtransparency="true" title="'.__('Subscription MailPoet',WYSIJA).'"></iframe>';
797
+ break;
798
+ case 'php':
799
+ $output = array(
800
+ '$widgetNL = new WYSIJA_NL_Widget(true);',
801
+ 'echo $widgetNL->widget(array(\'form\' => '.(int)$form_id.', \'form_type\' => \'php\'));'
802
+ );
803
+ return join("\n", $output);
804
+ break;
805
+ case 'html':
806
+ //need some language for the validation
807
+ $helper_toolbox = WYSIJA::get('toolbox','helper');
808
+ $wp_language_code = $helper_toolbox->get_language_code();
809
+
810
+ $wysija_version = WYSIJA::get_version();
811
+ $scripts_to_include = '<!--START Scripts : this is the script part you can add to the header of your theme-->'."\n";
812
+ $scripts_to_include .= '<script type="text/javascript" src="'.includes_url().'js/jquery/jquery.js'.'?ver='.$wysija_version.'"></script>'."\n";
813
+ if(file_exists(WYSIJA_DIR.'js'.DS.'validate'.DS.'languages'.DS.'jquery.validationEngine-'.$wp_language_code.'.js')){
814
+ $scripts_to_include .= '<script type="text/javascript" src="'.WYSIJA_URL.'js/validate/languages/jquery.validationEngine-'.$wp_language_code.'.js'.'?ver='.$wysija_version.'"></script>'."\n";
815
+ }else{
816
+ $scripts_to_include .= '<script type="text/javascript" src="'.WYSIJA_URL.'js/validate/languages/jquery.validationEngine-en.js'.'?ver='.$wysija_version.'"></script>'."\n";
817
+ }
818
+ $scripts_to_include .= '<script type="text/javascript" src="'.WYSIJA_URL.'js/validate/jquery.validationEngine.js'.'?ver='.$wysija_version.'"></script>'."\n";
819
+ $scripts_to_include .= '<script type="text/javascript" src="'.WYSIJA_URL.'js/front-subscribers.js'.'?ver='.$wysija_version.'"></script>'."\n";
820
+ $scripts_to_include .= '<script type="text/javascript">
821
+ /* <![CDATA[ */
822
+ var wysijaAJAX = {"action":"wysija_ajax","controller":"subscribers","ajaxurl":"'.admin_url('admin-ajax.php','absolute').'","loadingTrans":"'.__('Loading...',WYSIJA).'"};
823
+ /* ]]> */
824
+ </script>';
825
+ $scripts_to_include .= '<script type="text/javascript" src="'.WYSIJA_URL.'js/front-subscribers.js?ver='.$wysija_version.'"></script>'."\n";
826
+ $scripts_to_include .= '<!--END Scripts-->'."\n"."\n";
827
+
828
+ //enqueue the scripts
829
+ $html_result = $scripts_to_include;
830
+
831
+ // add the html for the form
832
+ $widget_NL = new WYSIJA_NL_Widget(true);
833
+ $html_result .= $widget_NL->widget(array('form' => (int)$form_id, 'form_type' => 'html'));
834
+
835
+ return $html_result;
836
+ break;
837
+ case 'shortcode':
838
+ return '[wysija_form id="'.(int)$form_id.'"]';
839
+ break;
840
+ }
841
+ }
842
+ }
helpers/forms.php ADDED
@@ -0,0 +1,266 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ /* some Functions are based on Codeigniter's Form Helper Class (http://www.codeigniter.com) */
4
+ class WYSIJA_help_forms{
5
+ var $eachValues = array();
6
+ var $eachValuesSec = array();
7
+
8
+ function WYSIJA_help_forms(){
9
+ // I believe this is for translation purpose, making sure the correct language is loaded
10
+ add_action( 'init', array( $this, 'apply_filters' ), 20 );
11
+
12
+ $this->eachValues = array(
13
+ 'one_min' => __( 'every minute', WYSIJA ),
14
+ 'two_min' => __( 'every 2 minutes', WYSIJA ),
15
+ 'five_min' => __( 'every 5 minutes', WYSIJA ),
16
+ 'ten_min' => __( 'every 10 minutes', WYSIJA ),
17
+ 'fifteen_min' => __( 'every 15 minutes', WYSIJA ),
18
+ 'thirty_min' => __( 'every 30 minutes', WYSIJA ),
19
+ 'hourly' => __( 'every hour', WYSIJA ),
20
+ 'two_hours' => __( 'every 2 hours', WYSIJA )
21
+ );
22
+
23
+ $this->eachValuesSec = array(
24
+ 'one_min' => '60',
25
+ 'two_min' => '120',
26
+ 'five_min' => '300',
27
+ 'ten_min' => '600',
28
+ 'fifteen_min' => '900',
29
+ 'thirty_min' => '1800',
30
+ 'hourly' => '3600',
31
+ 'two_hours' => '7200',
32
+ 'twicedaily' => '43200',
33
+ 'daily' => '86400',
34
+ );
35
+ }
36
+
37
+ function apply_filters() {
38
+ $this->eachValues = apply_filters( 'mpoet_sending_frequency', $this->eachValues );
39
+ $this->eachValuesSec = apply_filters( 'mpoet_sending_frequency_sec', $this->eachValuesSec );
40
+ }
41
+
42
+ function input($data = '', $value = '', $extra = '') {
43
+ $defaults = array('type' => 'text', 'name' => (( ! is_array($data)) ? $data : ''), 'value' => $value);
44
+ $content="<input ".$this->setAttrib($data, $defaults).$extra." />";
45
+ return $content;
46
+ }
47
+ function password($data = '', $value = '', $extra = '') {
48
+ $defaults = array('type' => 'password', 'name' => (( ! is_array($data)) ? $data : ''), 'value' => $value);
49
+ $content="<input ".$this->setAttrib($data, $defaults).$extra." />";
50
+ return $content;
51
+ }
52
+
53
+ function hidden($data = '', $value = '', $extra = '') {
54
+ $defaults = array('type' => 'hidden', 'name' => (( ! is_array($data)) ? $data : ''), 'value' => $value);
55
+ $content="<input ".$this->setAttrib($data, $defaults).$extra." />";
56
+ return $content;
57
+ }
58
+
59
+ function textarea($data = '', $value = '', $extra = '') {
60
+ $defaults = array('name' => (( ! is_array($data)) ? $data : ''), 'cols' => '90', 'rows' => '12');
61
+
62
+ if ( ! is_array($data) OR ! isset($data['value'])) {
63
+ $val = $value;
64
+ } else {
65
+ $val = $data['value'];
66
+ unset($data['value']);
67
+ }
68
+
69
+ return '<textarea '.$this->setAttrib($data, $defaults).$extra.'>'.$val.'</textarea>';
70
+ }
71
+
72
+
73
+ function tinymce($idName = '', $content = '') {
74
+ $this->the_editor(stripslashes($content), $idName,'title',false);
75
+ }
76
+
77
+ function checkbox($data = '', $value = '', $checked = FALSE, $extra = '') {
78
+ $defaults = array('type' => 'checkbox', 'name' => (( ! is_array($data)) ? $data : ''), 'value' => $value);
79
+
80
+ if (is_array($data) AND array_key_exists('checked', $data)) {
81
+ $checked = $data['checked'];
82
+
83
+ if ($checked == FALSE) unset($data['checked']);
84
+ else $data['checked'] = 'checked';
85
+ }
86
+
87
+ if ($checked == TRUE) $defaults['checked'] = 'checked';
88
+ else unset($defaults['checked']);
89
+
90
+ return "<input ".$this->setAttrib($data, $defaults).$extra." />";
91
+ }
92
+
93
+ function checkboxes($data = '', $values = array(), $value = '', $extra = '') {
94
+ $html='';
95
+ foreach($values as $val => $valtitle){
96
+ $checked=false;
97
+ $data2=$data;
98
+ $data2['id'].="-".$val;
99
+ if($val==$value)$checked=true;
100
+ $html.='<label for="'.$data2['id'].'">'.$this->checkbox($data2, $val, $checked, $extra).$valtitle."</label>";
101
+ }
102
+
103
+ return $html;
104
+ }
105
+
106
+ function radios($data = '', $values = array(), $value = '', $extra = '') {
107
+ $html='';
108
+ foreach($values as $val => $valtitle){
109
+ $checked=false;
110
+ $data2=$data;
111
+ $data2['id'].="-".$val;
112
+ if($val==$value)$checked=true;
113
+ $html.='<label for="'.$data2['id'].'">'.$this->radio($data2, $val, $checked, $extra).$valtitle."</label>";
114
+ }
115
+
116
+ return $html;
117
+ }
118
+
119
+ function radio($data = '', $value = '', $checked = FALSE, $extra = '') {
120
+ if ( ! is_array($data)) {
121
+ $data = array('name' => $data);
122
+ }
123
+
124
+ $data['type'] = 'radio';
125
+ return $this->checkbox($data, $value, $checked, $extra);
126
+ }
127
+
128
+
129
+ function the_editor($content, $id = 'content', $prev_id = 'title', $media_buttons = true, $tab_index = 2){
130
+ $rows = get_option('default_post_edit_rows');
131
+ if (($rows < 3) || ($rows > 100))
132
+ $rows = 12;
133
+
134
+ if ( !current_user_can( 'upload_files' ) )
135
+ $media_buttons = false;
136
+
137
+ $richedit = user_can_richedit();
138
+ $class = '';
139
+
140
+ if ( $richedit || $media_buttons ) { ?>
141
+ <div id="editor-toolbar">
142
+ <?php
143
+ if ( $richedit ) {
144
+ $wp_default_editor=tap_get_option("visual_ed_disable");
145
+ ?>
146
+ <div class="zerosize"><input accesskey="e" type="button" onclick="switchEditors.go('<?php echo $id; ?>')" /></div>
147
+ <?php if ( $wp_default_editor ) {
148
+ add_filter('the_editor_content', 'wp_htmledit_pre'); ?>
149
+ <a id="edButtonHTML" class="edButtonHTML active hide-if-no-js" onclick="switchEditors.go('<?php echo $id; ?>', 'html');"><?php _e('HTML'); ?></a>
150
+ <a id="edButtonPreview" class="edButtonPreview hide-if-no-js" onclick="switchEditors.go('<?php echo $id; ?>', 'tinymce');"><?php _e('Visual'); ?></a>
151
+ <?php } else {
152
+ $class = " class='theEditor'";
153
+ add_filter('the_editor_content', 'wp_richedit_pre'); ?>
154
+ <a id="edButtonHTML" class="edButtonHTML hide-if-no-js" onclick="switchEditors.go('<?php echo $id; ?>', 'html');"><?php _e('HTML'); ?></a>
155
+ <a id="edButtonPreview" class="edButtonPreview active hide-if-no-js" onclick="switchEditors.go('<?php echo $id; ?>', 'tinymce');"><?php _e('Visual'); ?></a>
156
+ <?php }
157
+ }
158
+ ?><div id="media-buttons" class="hide-if-no-js"><?php
159
+ if ( $media_buttons ) { ?>
160
+
161
+ <?php do_action( 'media_buttons' ); ?>
162
+
163
+ <?php
164
+ } ?>
165
+ </div>
166
+ </div>
167
+ <?php
168
+ }
169
+ ?>
170
+ <div id="quicktags"><?php
171
+ wp_print_scripts( 'quicktags' ); ?>
172
+ </div>
173
+
174
+ <?php
175
+ $the_editor = apply_filters('the_editor', "<div id='editorcontainer'><textarea rows='$rows'$class cols='40' name='$id' tabindex='$tab_index' id='$id'>%s</textarea></div>\n");
176
+ $the_editor_content = apply_filters('the_editor_content', $content);
177
+
178
+ printf($the_editor, $the_editor_content);
179
+
180
+ }
181
+
182
+ function titleh($idName = '', $selected = '') {
183
+ $options=array("default"=>"default","h2"=>"h2","h3"=>"h3","h4"=>"h4");
184
+ echo $this->dropdown($idName,$options,$selected);
185
+ }
186
+
187
+ function enabled($idName = '', $selected = '') {
188
+ $options=array(0=>"disabled",1=>"enabled");
189
+ echo $this->dropdown($idName,$options,$selected);
190
+ }
191
+
192
+ function dropdown($data = '', $options = array(), $selected = array(), $extra = '') {
193
+ if ( ! is_array($selected)) {
194
+ $selected = array($selected);
195
+ }
196
+
197
+ if ( empty($options) ) {
198
+ return false;
199
+ }
200
+
201
+ $defaults = array('name' => (( ! is_array($data)) ? $data : ''));
202
+
203
+ /* buggy lines
204
+ * if (count($selected) === 0) {
205
+ if (isset($_POST[$name])) $selected = array($_POST[$name]);
206
+ }*/
207
+
208
+ if ($extra != '') $extra = ' '.$extra;
209
+
210
+ $multiple = (count($selected) > 1 && strpos($extra, 'multiple') === FALSE) ? ' multiple="multiple"' : '';
211
+ $form = '<select '.$this->setAttrib($data, $defaults).$extra.$multiple.">\n";
212
+
213
+ foreach ($options as $key => $val) {
214
+ $key = (string) $key;
215
+ if (is_array($val)) {
216
+ $form .= '<optgroup label="'.$key.'">'."\n";
217
+ foreach ($val as $optgroup_key => $optgroup_val) {
218
+ $sel = (in_array($optgroup_key, $selected)) ? ' selected="selected"' : '';
219
+ $form .= '<option value="'.esc_attr($optgroup_key).'"'.$sel.'>'.(string) $optgroup_val."</option>\n";
220
+ }
221
+ $form .= '</optgroup>'."\n";
222
+ } else {
223
+ $sel = (in_array($key, $selected)) ? ' selected="selected"' : '';
224
+ $form .= '<option value="'.esc_attr($key).'"'.$sel.'>'.(string) $val."</option>\n";
225
+ }
226
+ }
227
+ $form .= '</select>';
228
+ return $form;
229
+ }
230
+
231
+
232
+ function setAttrib($attributes, $default) {
233
+ if (is_array($attributes)) {
234
+ foreach ($default as $key => $val) {
235
+ if (isset($attributes[$key])) {
236
+ $default[$key] = $attributes[$key];
237
+ unset($attributes[$key]);
238
+ }
239
+ }
240
+ if(isset($attributes["default"])){
241
+
242
+ $attributes["onBlur"]="if(this.value=='') {this.value='".$attributes["default"]."';this.style.color='#ccc';this.style.fontStyle='italic';}";
243
+ $attributes["onFocus"]="if(this.value=='".$attributes["default"]."') {this.value='';this.style.color='#000';this.style.fontStyle='normal';}";
244
+ if((!isset($default['value']) || !$default['value'])){
245
+ $default['value']=$attributes["default"];
246
+ $attributes["style"]="color:#ccc;font-style:italic;";
247
+ }
248
+ }
249
+
250
+ if (count($attributes) > 0) {
251
+ $default = array_merge($default, $attributes);
252
+ }
253
+ }
254
+
255
+ $att = '';
256
+
257
+ foreach ($default as $key => $val) {
258
+ if($key=='value') $val=esc_attr($val);
259
+ $att .= $key . '="' . $val . '" ';
260
+ }
261
+
262
+ return $att;
263
+ }
264
+
265
+ }
266
+
helpers/front.php ADDED
@@ -0,0 +1,306 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ /**
4
+ * class managing the admin vital part to integrate
5
+ */
6
+ class WYSIJA_help_front extends WYSIJA_help{
7
+
8
+ function WYSIJA_help_front(){
9
+ parent::WYSIJA_help();
10
+
11
+
12
+ /*if(defined('WYSIJA_DBG_ALL')){
13
+ $config=WYSIJA::get('config','model');
14
+ define('WYSIJA_DBG',(int)$config->getValue('debug_new'));
15
+
16
+ if(WYSIJA_DBG>0) include_once(WYSIJA_INC.'debug.php');
17
+
18
+ if(!function_exists('dbg')) {
19
+ function dbg($mixed,$exit=true){}
20
+ }
21
+ }*/
22
+ /* the controller is frontend if there is any wysija data requested */
23
+
24
+ /*$config=WYSIJA::get("config","model");
25
+ if($config->getValue("debug_on")) include_once(WYSIJA_INC."debug.php");*/
26
+ //include_once(WYSIJA_INC."debug.php");
27
+
28
+ // wysija form shortcode
29
+ add_shortcode('wysija_form', array($this,'scan_form_shortcode'));
30
+ // wysija total of subscribers shortcode
31
+ add_shortcode('wysija_subscribers_count', array($this,'scan_subscribers_count_shortcode'));
32
+
33
+ // init shortcode [wysija_archive]
34
+ require_once(WYSIJA_CORE.'controller.php');
35
+ require_once(WYSIJA_CORE.'module'.DS.'module.php');// @todo: move to autoloader
36
+ $archive_std = WYSIJA_module::get_instance_by_name('archive_std');// implement hook "wysija_front_init()
37
+ if (!empty($archive_std) && is_a($archive_std, 'WYSIJA_module'))
38
+ $archive_std->front_init();
39
+
40
+ /* We try to process the least possible code */
41
+ if(isset($_REQUEST['wysija-page']) || isset($_REQUEST['wysija-launch'])){
42
+
43
+ if(defined('DOING_AJAX')){
44
+ add_action('wp_ajax_nopriv_wysija_ajax', array($this, 'ajax'));
45
+ }else{
46
+ $paramscontroller=$_REQUEST['controller'];
47
+ //this is an exception on one server this params stats was not accepted
48
+ if($paramscontroller=='stat') $paramscontroller='stats';
49
+
50
+ $this->controller=WYSIJA::get($paramscontroller,'controller');
51
+ if(isset($_REQUEST['action']) && method_exists($this->controller, $_REQUEST['action'])){
52
+ add_action('init',array($this->controller,$_REQUEST['action']));
53
+ //$this->controller->$_REQUEST['action']();
54
+ }else $this->error('Action does not exist.');
55
+
56
+ if(isset($_REQUEST['wysija-page'])){
57
+ /* set the content filter to replace the shortcode */
58
+ add_filter('wp_title', array($this,'meta_page_title'));
59
+ add_filter( 'the_title', array($this,'scan_title'));
60
+ add_filter( 'the_content', array($this,'scan_content'),98);
61
+ if(isset($_REQUEST['message_success'])){
62
+ add_filter( 'the_content', array($this,'scan_content_NLform'),99 );
63
+ }
64
+ }
65
+ }
66
+ }else{
67
+ add_filter('the_content', array($this,'scan_content_NLform'),99 );
68
+ //if the comment form checkbox option is activated we add some hooks to process it
69
+ $model_config = WYSIJA::get('config','model');
70
+ if($model_config->getValue('commentform')){
71
+ add_action('comment_form', array($this,'comment_form_extend'));
72
+ add_action('comment_post', array($this,'comment_posted'), 60,2);
73
+ }
74
+
75
+ // if the register form checkbox option is activated we add some hooks to process it
76
+ if($model_config->getValue('registerform')){
77
+ if(is_multisite()){
78
+ add_action('signup_extra_fields', array($this,'register_form_extend'));
79
+ // we need this condition otherwise we will send two confirmation emails when on ms with buddypress
80
+ if(!WYSIJA::is_plugin_active('buddypress/bp-loader.php')){
81
+ add_filter('wpmu_validate_user_signup', array($this,'registerms_posted'), 60,3);
82
+ }
83
+ }else{
84
+ add_action('register_form', array($this,'register_form_extend'));
85
+ add_action('register_post', array($this,'register_posted'), 60,3);
86
+ }
87
+
88
+ // special case when buddypress is activated
89
+ if(WYSIJA::is_plugin_active('buddypress/bp-loader.php')){
90
+ add_action('bp_after_signup_profile_fields', array($this,'register_form_bp_extend'));
91
+ add_action('bp_signup_validate', array($this,'register_bp'),60,3);
92
+
93
+ // we can have just one confirmation email for the wp user and the wysija confirmation when bp and multisite are activated
94
+ if(is_multisite()){
95
+ add_action('wpmu_activate_user', array($this,'wpmu_activate_user'));
96
+ }
97
+ }
98
+ }
99
+ }
100
+ }
101
+
102
+ /**
103
+ * In MS when user account is activated we auto confirm the subscriber
104
+ * @param type $wpuser_id
105
+ * @return boolean
106
+ */
107
+ function wpmu_activate_user($wpuser_id){
108
+ if((int)$wpuser_id>0){
109
+ $model_user = WYSIJA::get('user','model');
110
+ $result_subscriber = $model_user->getOne(false , array('wpuser_id'=>$wpuser_id));
111
+
112
+ if(!empty($result_subscriber)){
113
+ $helper_user = WYSIJA::get('user','helper');
114
+ $helper_user->confirm_user($result_subscriber['user_id']);
115
+ }
116
+ }
117
+ return true;
118
+ }
119
+
120
+ function meta_page_title($title){
121
+ //Here I can echo the result and see that it's actually triggered
122
+ if($this->is_default_mailpoet_page()){
123
+ // when user have kept the default page we modify the title based on the controller
124
+ if(!empty($this->controller->title)) return $this->controller->title;
125
+ }
126
+
127
+ return $title;
128
+ }
129
+
130
+
131
+
132
+ function register_form_bp_extend(){
133
+ if ( !is_user_logged_in()){
134
+ $this->register_form_extend();
135
+ }
136
+ }
137
+
138
+ function register_form_extend(){
139
+ $checkbox= '<p class="wysija-after-register">';
140
+ $checkbox.='<label for="wysija-box-after-register">';
141
+ $checkbox.='<input type="checkbox" id="wysija-box-after-register" value="1" name="wysija[register_subscribe]">';
142
+ $mConfig=WYSIJA::get('config','model');
143
+ $checkbox.=$mConfig->getValue('registerform_linkname').'</label></p>';
144
+
145
+ echo '<div class="register-section" id="profile-details-section-wysija"><div class="editfield">'.$checkbox.'</div></div>';
146
+ }
147
+
148
+
149
+ function register_bp(){
150
+ global $bp;
151
+
152
+ if ( !isset($bp->signup->errors) && isset($_POST['wysija']['register_subscribe']) && $_POST['wysija']['register_subscribe'] ) {
153
+ $model_config=WYSIJA::get('config','model');
154
+ $helper_user=WYSIJA::get('user','helper');
155
+ $data=array('user'=>array('email'=>$bp->signup->email),'user_list'=>array('list_ids'=>$model_config->getValue('registerform_lists')));
156
+
157
+ if(is_multisite()){
158
+ $helper_user->no_confirmation_email=true;
159
+ }
160
+
161
+ $helper_user->addSubscriber($data);
162
+ }
163
+ }
164
+
165
+ function registerms_posted($result){
166
+ if ( empty($result['errors']->errors) && isset($_POST['wysija']['register_subscribe']) && $_POST['wysija']['register_subscribe']) {
167
+ $mConfig=WYSIJA::get('config','model');
168
+ $userHelper=WYSIJA::get('user','helper');
169
+ $data=array('user'=>array('email'=>$result['user_email']),'user_list'=>array('list_ids'=>$mConfig->getValue('registerform_lists')));
170
+ $userHelper->addSubscriber($data);
171
+ }
172
+
173
+ return $result;
174
+ }
175
+
176
+ function register_posted($login,$email,$errors){
177
+
178
+ if ( empty($errors->errors) && isset($_POST['wysija']['register_subscribe']) && $_POST['wysija']['register_subscribe']) {
179
+ $mConfig=WYSIJA::get('config','model');
180
+ $userHelper=WYSIJA::get('user','helper');
181
+ $data=array('user'=>array('email'=>$email),'user_list'=>array('list_ids'=>$mConfig->getValue('registerform_lists')));
182
+ $userHelper->addSubscriber($data);
183
+ }
184
+ }
185
+
186
+
187
+ function comment_form_extend(){
188
+ echo '<p class="wysija-after-comment">';
189
+ echo '<label for="wysija-box-after-comment">';
190
+ echo '<input type="checkbox" id="wysija-box-after-comment" value="1" name="wysija[comment_subscribe]">';
191
+ $mConfig=WYSIJA::get('config','model');
192
+ echo $mConfig->getValue('commentform_linkname').'</label></p>';
193
+ }
194
+
195
+ function comment_posted($cid,$comment){
196
+ $cid = (int) $cid;
197
+ if ( !is_object($comment) )
198
+ $comment = get_comment($cid);
199
+
200
+ //before recording the subscriber, make sure that it is not a spam comment or it needs to be approved first
201
+ if($comment->comment_approved=='spam') return;
202
+
203
+ if(isset($_POST['wysija']['comment_subscribe']) && $_POST['wysija']['comment_subscribe']) {
204
+ if($comment->comment_approved=='0') add_comment_meta($cid, 'wysija_comment_subscribe', 1);
205
+ else{
206
+ $mConfig=WYSIJA::get('config','model');
207
+ $userHelper=WYSIJA::get('user','helper');
208
+ $data=array('user'=>array('email'=>$comment->comment_author_email,'firstname'=>$comment->comment_author),'user_list'=>array('list_ids'=>$mConfig->getValue('commentform_lists')));
209
+ $userHelper->addSubscriber($data);
210
+ }
211
+ }
212
+ }
213
+
214
+ function scan_title($title){
215
+ /*careful WordPress global*/
216
+ global $post;
217
+
218
+ if($this->is_default_mailpoet_page()){
219
+ // when user have kept the default page we modify the title based on the controller
220
+ if(trim($title)==trim(single_post_title( '', false )) && !empty($this->controller->title)){
221
+ $post->comment_status='close';
222
+ $post->post_password='';
223
+ return $this->controller->title;
224
+ }
225
+ }
226
+
227
+ return $title;
228
+
229
+ }
230
+
231
+ function scan_content($content){
232
+ $wysija_content = $wysija_undo = '';
233
+ if(!empty($this->controller->subtitle)) $wysija_content = $this->controller->subtitle;
234
+ if(!empty($this->controller->optional_subtitle)) $wysija_content = $this->controller->optional_subtitle;
235
+ if(!empty($this->controller->undo_unsubscribe)){
236
+ $wysija_undo = $this->controller->undo_unsubscribe;
237
+ }
238
+
239
+ // only force our edit subscription screen at the bottom of the content of the page
240
+ // if it's the right action and there is no wysija_page shortcode in teh page
241
+ if(!empty($_REQUEST['action']) && $_REQUEST['action'] == 'subscriptions' && strpos($content, '[wysija_page]') == false){
242
+ // we append the subscription form at the bottom of the page if we can't detect it
243
+ return $content.'<div class="mpoet_profile_edit">'.$wysija_content.'</div>';
244
+ }else{
245
+ // we replace the shortcode by our automatic content, other wise if there is no shortcode the page stays the same
246
+ return str_replace('[wysija_page]', $wysija_content.$wysija_undo, $content);
247
+ }
248
+
249
+ }
250
+
251
+ function is_default_mailpoet_page(){
252
+ global $post;
253
+ $model_config = WYSIJA::get('config','model');
254
+ if($post->ID == $model_config->getValue('confirm_email_link')) return true;
255
+ else return false;
256
+ }
257
+
258
+ /**
259
+ * this is for the new kind of shortcodes [wysija_form form="1"]
260
+ * @param array $attributes
261
+ * @return string html
262
+ */
263
+ function scan_form_shortcode($attributes) {
264
+ // IMPORTANT: this is to make sure MagicMember won't scan our form and find [user_list] as a code they should replace.
265
+ remove_shortcode('user_list');
266
+
267
+ if(isset($attributes['id']) && (int)$attributes['id']>0){
268
+ $widget_data=array();
269
+ $widget_data['form']=(int)$attributes['id'];
270
+ $widget_data['form_type'] = 'shortcode';
271
+
272
+ $widget_NL=new WYSIJA_NL_Widget(true);
273
+ return $widget_NL->widget($widget_data);
274
+
275
+ }
276
+ return '';
277
+ }
278
+
279
+
280
+ /**
281
+ * this is for the new kind of shortcodes [wysija_form form="1"]
282
+ * @param array $attributes
283
+ * @return string html
284
+ */
285
+ function scan_subscribers_count_shortcode($attributes) {
286
+ $user = WYSIJA::get('user','model');
287
+ $list_ids = !empty($attributes['list_id']) ? explode(',', $attributes['list_id']) : array();
288
+ $confirmed_subscribers = !empty($attributes['confirmed_subscribers']) ? (bool)$attributes['confirmed_subscribers'] : true;
289
+ return $user->countSubscribers($list_ids, $confirmed_subscribers);
290
+ }
291
+
292
+ function scan_content_NLform($content){
293
+ preg_match_all('/\<div class="wysija-register">(.*?)\<\/div>/i',$content,$matches);
294
+ if(!empty($matches[1]) && count($matches[1])>0) require_once(WYSIJA_WIDGETS.'wysija_nl.php');
295
+ foreach($matches[1] as $key => $mymatch){
296
+ if($mymatch){
297
+ $widgetdata=unserialize(base64_decode($mymatch));
298
+ $widgetNL=new WYSIJA_NL_Widget(true);
299
+ $contentTABLE= $widgetNL->widget($widgetdata,$widgetdata);
300
+ $content=str_replace($matches[0][$key],$contentTABLE,$content);
301
+ }//endif
302
+ }//endforeach
303
+ return $content;
304
+ }
305
+
306
+ }
helpers/http.php ADDED
@@ -0,0 +1,119 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_http extends WYSIJA_object{
4
+
5
+
6
+ function WYSIJA_help_http(){
7
+
8
+ }
9
+
10
+ /**
11
+ * try three different methods for http request,
12
+ * @param type $url
13
+ * @return type
14
+ */
15
+ function request($url){
16
+ // use curl_get first if it is activated
17
+ if(function_exists('curl_init')) {
18
+ $this->opts = array(
19
+ CURLOPT_HEADER => FALSE,
20
+ CURLOPT_RETURNTRANSFER => TRUE
21
+ );
22
+ $result=$this->curl_get($url);
23
+ return $result['cr'];
24
+ // then try http_get
25
+ }elseif(function_exists('http_get')){
26
+ return http_parse_message(http_get($url))->body;
27
+
28
+ // finally we have file_get_contents which is quite often deactivated
29
+ }elseif(ini_get('allow_url_fopen')){
30
+ return file_get_contents($url);
31
+ }else{
32
+ $this->error(__('Your server doesn\'t support remote exchanges.',WYSIJA));
33
+ $this->error(__('Contact your administrator to modify that, it should be configurable.',WYSIJA));
34
+ $this->error('<strong>CURL library</strong> DISABLED');
35
+ $this->error('<strong>allow_url_fopen</strong> DISABLED');
36
+ $this->error('<strong>PECL pecl_http >= 0.1.0</strong> DISABLED');
37
+ return false;
38
+ }
39
+ }
40
+
41
+ function wp_request($url){
42
+ global $wp_version;
43
+
44
+ $active = get_option( 'active_plugins', array() );
45
+ $to_send = (object) compact('plugins', 'active');
46
+
47
+ $options = array(
48
+ 'timeout' => ( ( defined('DOING_CRON') && DOING_CRON ) ? 30 : 3),
49
+ 'body' => array( 'plugins' => serialize( $to_send ) ),
50
+ 'user-agent' => 'WordPress/' . $wp_version . '; ' . get_bloginfo( 'url' )
51
+ );
52
+
53
+ $raw_response = wp_remote_post($url, $options);
54
+
55
+ if ( is_wp_error( $raw_response ) || 200 != wp_remote_retrieve_response_code( $raw_response ) ){
56
+ if(method_exists($raw_response, 'get_error_messages')){
57
+ $this->error($raw_response->get_error_messages());
58
+ }
59
+ return false;
60
+ }
61
+
62
+ return maybe_unserialize( wp_remote_retrieve_body( $raw_response ) );
63
+ }
64
+
65
+ function request_timeout($url,$timeout='3'){
66
+ if(function_exists('curl_init')) {
67
+
68
+ $ch = curl_init( $url );
69
+ curl_setopt( $ch, CURLOPT_RETURNTRANSFER, 0 );
70
+ curl_setopt( $ch, CURLOPT_TIMEOUT, $timeout );
71
+ $result = curl_exec( $ch );
72
+
73
+ return curl_close( $ch );
74
+ }elseif(ini_get('allow_url_fopen')){
75
+ ini_set('default_socket_timeout',(int)$timeout);
76
+ return @file_get_contents($url);
77
+ }elseif(function_exists('http_get')){
78
+ return @http_get($url, array('timeout'=>(int)$timeout));
79
+ }else{
80
+ $this->error(__('Your server doesn\'t support remote exchanges.',WYSIJA));
81
+ $this->error(__('Contact your administrator to modify that, it should be configurable.',WYSIJA));
82
+ $this->error('<strong>CURL library</strong> DISABLED');
83
+ $this->error('<strong>allow_url_fopen</strong> DISABLED');
84
+ $this->error('<strong>PECL pecl_http >= 0.1.0</strong> DISABLED');
85
+ return false;
86
+ }
87
+ }
88
+
89
+ function curl_request($ch,$opt){
90
+ # assign global options array
91
+ $opts = $this->opts;
92
+ # assign user's options
93
+ foreach($opt as $k=>$v){$opts[$k] = $v;}
94
+ curl_setopt_array($ch,$opts);
95
+ curl_exec($ch);
96
+ $r['code'] = curl_getinfo($ch,CURLINFO_HTTP_CODE);
97
+ $r['cr'] = curl_exec($ch);
98
+ $r['ce'] = curl_errno($ch);
99
+ curl_close($ch);
100
+ return $r;
101
+ }
102
+
103
+ function curl_get($url='',$opt=array()){
104
+ # create cURL resource
105
+ $ch = curl_init($url);
106
+ return $this->curl_request($ch,$opt);
107
+ }
108
+
109
+ /* function curl_post($url='',$data=array(),$opt=array()){
110
+ # set POST options
111
+ $opts[CURLOPT_POST] = TRUE;
112
+ $opts[CURLOPT_POSTFIELDS] = $data;
113
+
114
+ # create cURL resource
115
+ $ch = curl_init($url);
116
+ return $this->curl_request($ch,$opt);
117
+ }*/
118
+ }
119
+
helpers/image.php ADDED
@@ -0,0 +1,89 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ require_once(dirname(__FILE__).DS.'file.php');
4
+ class WYSIJA_help_image extends WYSIJA_help_file{
5
+
6
+ function WYSIJA_help_image(){
7
+
8
+ }
9
+
10
+ /**
11
+ * get a list of images from a directory
12
+ * @param type $template
13
+ * @return type
14
+ */
15
+ public function get_list_directory($template =' default'){
16
+ $foldersTocheck = 'themes'.DS.$template.DS.'img'.DS.'public';
17
+ $url = 'themes/'.$template.'/img/public';
18
+ $images_types_allowed = array('jpg','png','jpeg');
19
+ $listed = array();
20
+
21
+ $path = $this->getUploadDir($foldersTocheck);
22
+
23
+ /* get a list of files from this folder and clear them */
24
+ if(file_exists($path)){
25
+ $files = scandir($path);
26
+ $i=1;
27
+ foreach($files as $file_name){
28
+ if(!in_array($file_name, array('.','..','.DS_Store','Thumbs.db'))){
29
+ if(preg_match('/.*\.('.implode($images_types_allowed,'|').')/',$file_name,$match)){
30
+ $image_template = array(
31
+ 'path'=> $path.$file_name,
32
+ 'width'=> 0,
33
+ 'height'=> 0,
34
+ 'url'=> $this->url($file_name,$url),
35
+ 'thumb_url'=>$this->url($file_name,$url),
36
+ 'identifier'=>'tmpl-'.$template.$i,
37
+ );
38
+
39
+ $listed['tmpl-'.$template.$i] = $this->valid_image($image_template);
40
+ $i++;
41
+ }
42
+ }
43
+ }
44
+
45
+ return $listed;
46
+ }
47
+
48
+ }
49
+
50
+ /**
51
+ * make sure the image is valid, has a src and has an height and width
52
+ * @param type $post_image
53
+ * @return null
54
+ */
55
+ public function valid_image($post_image){
56
+ if(!isset($post_image['src']) && isset($post_image['url'])) $post_image['src'] = $post_image['url'];
57
+ if(isset($post_image['src'])) {
58
+ // check that height & width have been set, if not try to calculate
59
+ if(empty($post_image['height']) || empty($post_image['width']) || (empty($post_image['height']) && empty($post_image['width']))) {
60
+
61
+ try {
62
+ $image_info = getimagesize($post_image['src']);
63
+
64
+ if($image_info !== false) {
65
+ $post_image['width'] = $image_info[0];
66
+ $post_image['height'] = $image_info[1];
67
+
68
+ }else{
69
+ // if allow_url_fopen is off we need to convert the url image into a local file
70
+ $image_src = dirname(dirname(dirname(WYSIJA_UPLOADS_DIR))).wp_make_link_relative($post_image['src']);
71
+ $image_info = getimagesize($image_src);
72
+ if($image_info !== false) {
73
+ $post_image['width'] = $image_info[0];
74
+ $post_image['height'] = $image_info[1];
75
+ }
76
+
77
+ }
78
+ } catch(Exception $e) {
79
+ return null;
80
+ }
81
+ }
82
+ return $post_image;
83
+ } else {
84
+ return null;
85
+ }
86
+ }
87
+
88
+ }
89
+
helpers/index.html ADDED
File without changes
helpers/install.php ADDED
@@ -0,0 +1,675 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ defined('WYSIJA') or die('Restricted access');
3
+ class WYSIJA_help_install extends WYSIJA_object{
4
+
5
+ function WYSIJA_help_install(){
6
+ if(file_exists(ABSPATH . 'wp-admin'.DS.'includes'.DS.'upgrade.php')) require_once(ABSPATH . 'wp-admin'.DS.'includes'.DS.'upgrade.php');
7
+ }
8
+
9
+ function install(){
10
+ $values=array();
11
+ $model_config=WYSIJA::get('config','model');
12
+ // test server against few things to make sure the installation can be done
13
+ $helper_server=WYSIJA::get('server','helper');
14
+ $missing_capabilities=$helper_server->unhealthy();
15
+
16
+ // if it returns false it means we're all good
17
+ if($missing_capabilities!==false){
18
+ // it will fail only if we have more than one missing capabilities or if we have just one and that's a required function
19
+ if(count($missing_capabilities) > 1 ||
20
+ (count($missing_capabilities)==1 && (!isset($missing_capabilities['functions']) || isset($missing_capabilities['functions']['required']))) ){
21
+ // here we need to return unfortunately
22
+ $this->error(__('Your server cannot run MailPoet.',WYSIJA),1);
23
+
24
+ if(isset($missing_capabilities['functions']['required'])){
25
+ $this->error(sprintf(__('Your server is missing one or many important PHP functions to run properly : %1$s',WYSIJA),'<strong>'.implode(', ', array_keys($missing_capabilities['functions']['required'])).'</strong>').' '.__('Please contact your host or server administrator to fix this.',WYSIJA));
26
+
27
+ }
28
+ return false;
29
+ }else{
30
+ // here we're ok
31
+ }
32
+ }
33
+
34
+ // create the tables there shouldn't be any issue since we've tested it before
35
+ if((int)get_option('installation_step')<1){
36
+ if(!$this->createTables(WYSIJA_DIR.'sql'.DS.'install.sql')) return false;
37
+
38
+ WYSIJA::update_option('installation_step', '1');
39
+ }
40
+
41
+
42
+ // move data to uploads folder: this needs to be done prior to default campaign creation for dependency reasons
43
+ if((int)get_option('installation_step')<4){
44
+ $this->moveData('themes');
45
+ $this->moveData('dividers');
46
+ $this->moveData('bookmarks');
47
+ WYSIJA::update_option('installation_step', '4');
48
+ }
49
+
50
+
51
+ // record custom fields lastname firstname in the user_field table
52
+ if((int)get_option('installation_step')<5){
53
+ $this->recordDefaultUserField();
54
+ WYSIJA::update_option('installation_step', '5');
55
+ }
56
+
57
+ // save default values for the fields : from_name, from_email replyto_name, replyto_email
58
+ if((int)get_option('installation_step')<6){
59
+ $this->defaultSettings($values);
60
+ $model_config->save($values);
61
+ WYSIJA::update_option('installation_step', '6');
62
+ }
63
+
64
+
65
+ //create a default list
66
+ if((int)get_option('installation_step')<7){
67
+ $this->defaultList($values);
68
+ $model_config->save($values);
69
+ WYSIJA::update_option('installation_step', '7');
70
+ }
71
+
72
+
73
+ // create a default campaign
74
+ if((int)get_option('installation_step')<8){
75
+ $this->defaultCampaign($values);
76
+ $model_config->save($values);
77
+ WYSIJA::update_option('installation_step', '8');
78
+ }
79
+
80
+
81
+ // synchronize our user table with wordpress users
82
+ if((int)get_option('installation_step')<9){
83
+ $helper_import=WYSIJA::get('plugins_import','helper');
84
+ $values['importwp_list_id']=$helper_import->importWP();
85
+ $model_config->save($values);
86
+ WYSIJA::update_option('installation_step', '9');
87
+ }
88
+
89
+ // create subscription redirection page
90
+ if((int)get_option('installation_step')<10){
91
+ $this->createPage($values);
92
+ $model_config->save($values);
93
+ WYSIJA::update_option('installation_step', '10');
94
+ }
95
+
96
+
97
+ // create the default dir
98
+ if((int)get_option('installation_step')<11){
99
+ $this->createWYSIJAdir($values);
100
+ $model_config->save($values);
101
+ WYSIJA::update_option('installation_step', '11');
102
+ }
103
+
104
+
105
+ // create default subscription form
106
+ if((int)get_option('installation_step')<12){
107
+ $this->create_default_subscription_form();
108
+ WYSIJA::update_option('installation_step', '12');
109
+ }
110
+
111
+
112
+ // save the confirmation email in the table
113
+
114
+
115
+ if((int)get_option('installation_step')<13){
116
+ // make sure that the activation email is translated
117
+ $model_config->add_translated_default();
118
+ WYSIJA::update_option('installation_step', '13');
119
+ }
120
+
121
+ if((int)get_option('installation_step')<14){
122
+ $model_email=WYSIJA::get('email','model');
123
+ $model_email->blockMe=true;
124
+ $values['confirm_email_id']=$model_email->insert(
125
+ array('type'=>'0',
126
+ 'from_email'=>$values['from_email'],
127
+ 'from_name'=>$values['from_name'],
128
+ 'replyto_email'=>$values['from_email'],
129
+ 'replyto_name'=>$values['from_name'],
130
+ 'subject'=>$model_config->getValue('confirm_email_title'),
131
+ 'body'=>$model_config->getValue('confirm_email_body'),
132
+ 'status'=>99));
133
+ $model_config->save($values);
134
+ WYSIJA::update_option('installation_step', '14');
135
+ }
136
+
137
+
138
+ // look for existing newsletter plugins to import from
139
+ if((int)get_option('installation_step')<15){
140
+ $this->testNLplugins();
141
+
142
+ // administrator caps
143
+ $helper_wp_tools = WYSIJA::get('wp_tools', 'helper');
144
+ $helper_wp_tools->set_default_rolecaps();
145
+
146
+ WYSIJA::update_option('installation_step', '15');
147
+ }
148
+
149
+
150
+
151
+ // save the config into the db
152
+
153
+ if( (int) get_option('installation_step') < 16){
154
+
155
+ $model_config = WYSIJA::get('config','model');
156
+
157
+ $values['installed'] = true;
158
+ $values['manage_subscriptions'] = true;
159
+ $values['installed_time'] = time();
160
+
161
+ $values['wysija_db_version'] = WYSIJA::get_version();
162
+
163
+ $wptoolboxs = WYSIJA::get('toolbox', 'helper');
164
+ $values['dkim_domain'] = $wptoolboxs->_make_domain_name();
165
+
166
+ if( get_option('wysija_reinstall',0) ) $values['wysija_whats_new'] = WYSIJA::get_version();
167
+ $model_config->save($values);
168
+
169
+ WYSIJA::update_option('installation_step', '16');
170
+ }
171
+
172
+
173
+
174
+
175
+ global $wysija_installing;
176
+ $wysija_installing=false;
177
+ WYSIJA::update_option('wysija_reinstall',0);
178
+ return true;
179
+ }
180
+
181
+
182
+ // Description: Creates the default list.
183
+ function defaultList(&$values){
184
+ $model_list=WYSIJA::get('list','model');
185
+ $listname=__('My first list',WYSIJA);
186
+ $defaultListId=$model_list->insert(array(
187
+ 'name'=>$listname,
188
+ 'description'=>__('The list created automatically on install of the MailPoet.',WYSIJA),
189
+ 'is_public'=>1,
190
+ 'is_enabled'=>1));
191
+ $values['default_list_id']=$defaultListId;
192
+ // Add Wordpress user as first subscriber of the default list.
193
+ $helper_user=WYSIJA::get('user','helper');
194
+ $current_user=WYSIJA::wp_get_userdata();
195
+
196
+ $user_ids = array($current_user->ID);
197
+
198
+ $id=$helper_user->addToList($values['default_list_id'],$user_ids);
199
+ }
200
+
201
+ function defaultCampaign($valuesconfig){
202
+ $modelCampaign=WYSIJA::get('campaign','model');
203
+ $campaign_id=$modelCampaign->insert(
204
+ array(
205
+ 'name'=>__('5 Minute User Guide',WYSIJA),
206
+ 'description'=>'',
207
+
208
+ ));
209
+
210
+ $modelEmail=WYSIJA::get('email','model');
211
+ $modelEmail->fieldValid=false;
212
+
213
+ $dataEmail=array(
214
+ 'campaign_id'=>$campaign_id,
215
+ 'subject'=>__('5 Minute User Guide',WYSIJA)
216
+ );
217
+
218
+ // get default styles
219
+ $wjEngine = WYSIJA::get('wj_engine', 'helper');
220
+
221
+ // get solid divider
222
+ $hDividers = WYSIJA::get('dividers', 'helper');
223
+ $defaultDivider = $hDividers->getDefault();
224
+
225
+ // get bookmarks from iconset 2
226
+ $hBookmarks = WYSIJA::get('bookmarks', 'helper');
227
+ $bookmarks = $hBookmarks->getAllByIconset('medium', '02');
228
+
229
+ //--------------
230
+ $dataEmail['wj_data'] = array(
231
+ 'version' => WYSIJA::get_version(),
232
+ 'header' => array(
233
+ 'text' => null,
234
+ 'image' => array(
235
+ 'src' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/header.png',
236
+ 'width' => 600,
237
+ 'height' => 72,
238
+ 'alignment' => 'center',
239
+ 'static' => false
240
+ ),
241
+ 'alignment' => 'center',
242
+ 'static' => false,
243
+ 'type' => 'header'
244
+ ),
245
+ 'body' => array(
246
+ 'block-1' => array(
247
+ 'text' => array(
248
+ 'value' => '<h2><strong>'.__('Step 1:', WYSIJA).'</strong> '.__('hey, click on this text!', WYSIJA).'</h2>'.'<p>'.__('To edit, simply click on this block of text.', WYSIJA).'</p>'
249
+ ),
250
+ 'image' => null,
251
+ 'alignment' => 'left',
252
+ 'static' => false,
253
+ 'position' => 1,
254
+ 'type' => 'content'
255
+ ),
256
+ 'block-2' => array_merge(array(
257
+ 'position' => 2,
258
+ 'type' => 'divider'
259
+ ), $defaultDivider
260
+ ),
261
+ 'block-3' => array(
262
+ 'text' => array(
263
+ 'value' => '<h2><strong>'.__('Step 2:', WYSIJA).'</strong> '.__('play with this image', WYSIJA).'</h2>'
264
+ ),
265
+ 'image' => null,
266
+ 'alignment' => 'left',
267
+ 'static' => false,
268
+ 'position' => 3,
269
+ 'type' => 'content'
270
+ ),
271
+ 'block-4' => array(
272
+ 'text' => array(
273
+ 'value' => '<p>'.__('Position your mouse over the image to the left.', WYSIJA).'</p>'
274
+ ),
275
+ 'image' => array(
276
+ 'src' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/pigeon.png',
277
+ 'width' => 281,
278
+ 'height' => 190,
279
+ 'alignment' => 'left',
280
+ 'static' => false
281
+ ),
282
+ 'alignment' => 'left',
283
+ 'static' => false,
284
+ 'position' => 4,
285
+ 'type' => 'content'
286
+ ),
287
+ 'block-5' => array_merge(array(
288
+ 'position' => 5,
289
+ 'type' => 'divider'
290
+ ), $defaultDivider
291
+ ),
292
+ 'block-6' => array(
293
+ 'text' => array(
294
+ 'value' => '<h2><strong>'.__('Step 3:', WYSIJA).'</strong> '.__('drop content here', WYSIJA).'</h2>'.
295
+ '<p>'.sprintf(__('Drag and drop %1$stext, posts, dividers.%2$s Look on the right!', WYSIJA), '<strong>', '</strong>').'</p>'.
296
+ '<p>'.sprintf(__('You can even %1$ssocial bookmarks%2$s like these:', WYSIJA), '<strong>', '</strong>').'</p>'
297
+ ),
298
+ 'image' => null,
299
+ 'alignment' => 'left',
300
+ 'static' => false,
301
+ 'position' => 6,
302
+ 'type' => 'content'
303
+ ),
304
+ 'block-7' => array(
305
+ 'width' => 184,
306
+ 'alignment' => 'center',
307
+ 'items' => array(
308
+ array_merge(array(
309
+ 'url' => 'http://www.facebook.com/mailpoetplugin',
310
+ 'alt' => 'Facebook',
311
+ 'cellWidth' => 61,
312
+ 'cellHeight' => 32
313
+ ), $bookmarks['facebook']),
314
+ array_merge(array(
315
+ 'url' => 'http://www.twitter.com/mail_poet',
316
+ 'alt' => 'Twitter',
317
+ 'cellWidth' => 61,
318
+ 'cellHeight' => 32
319
+ ), $bookmarks['twitter']),
320
+ array_merge(array(
321
+ 'url' => 'https://plus.google.com/+Mailpoet',
322
+ 'alt' => 'Google',
323
+ 'cellWidth' => 61,
324
+ 'cellHeight' => 32
325
+ ), $bookmarks['google'])
326
+ ),
327
+ 'position' => 7,
328
+ 'type' => 'gallery'
329
+ ),
330
+ 'block-8' => array_merge(array(
331
+ 'position' => 8,
332
+ 'type' => 'divider'
333
+ ), $defaultDivider
334
+ ),
335
+ 'block-9' => array(
336
+ 'text' => array(
337
+ 'value' => '<h2><strong>'.__('Step 4:', WYSIJA).'</strong> '.__('and the footer?', WYSIJA).'</h2>'.
338
+ '<p>'.sprintf(__('Change the footer\'s content in MailPoet\'s %1$sSettings%2$s page.', WYSIJA), '<strong>', '</strong>').'</p>'
339
+ ),
340
+ 'image' => null,
341
+ 'alignment' => 'left',
342
+ 'static' => false,
343
+ 'position' => 9,
344
+ 'type' => 'content'
345
+ )
346
+ ),
347
+ 'footer' => array(
348
+ 'text' => NULL,
349
+ 'image' => array(
350
+ 'src' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/footer.png',
351
+ 'width' => 600,
352
+ 'height' => 46,
353
+ 'alignment' => 'center',
354
+ 'static' => false,
355
+ ),
356
+ 'alignment' => 'center',
357
+ 'static' => false,
358
+ 'type' => 'footer'
359
+ )
360
+ );
361
+
362
+ $dataEmail['wj_styles'] = array(
363
+ 'html' => array(
364
+ 'background' => 'e8e8e8'
365
+ ),
366
+ 'header' => array(
367
+ 'background' => 'e8e8e8'
368
+ ),
369
+ 'body' => array(
370
+ 'color' => '000000',
371
+ 'family' => 'Arial',
372
+ 'size' => 16,
373
+ 'background' => 'ffffff'
374
+ ),
375
+ 'footer' => array(
376
+ 'background' => 'e8e8e8'
377
+ ),
378
+ 'h1' => array(
379
+ 'color' => '000000',
380
+ 'family' => 'Trebuchet MS',
381
+ 'size' => 40
382
+ ),
383
+ 'h2' => array(
384
+ 'color' => '424242',
385
+ 'family' => 'Trebuchet MS',
386
+ 'size' => 30
387
+ ),
388
+ 'h3' => array(
389
+ 'color' => '424242',
390
+ 'family' => 'Trebuchet MS',
391
+ 'size' => 24
392
+ ),
393
+ 'a' => array(
394
+ 'color' => '4a91b0',
395
+ 'underline' => false
396
+ ),
397
+ 'unsubscribe' => array(
398
+ 'color' => '000000'
399
+ ),
400
+ 'viewbrowser' => array(
401
+ 'color' => '000000',
402
+ 'family' => 'Arial',
403
+ 'size' => 12
404
+ )
405
+ );
406
+ //---- END DEFAULT EMAIL ---------
407
+ foreach( $dataEmail['wj_data'] as $key =>&$eachval){
408
+ if($key=="body") {
409
+ foreach($eachval as &$realeachval){
410
+ if(isset($realeachval['text']['value'])) $realeachval['text']['value']=base64_encode($realeachval['text']['value']);
411
+ }
412
+ }
413
+ }
414
+
415
+ $dataEmail['params'] = array(
416
+ 'quickselection' => array(
417
+ 'wp-301' => array(
418
+ 'identifier' => 'wp-301',
419
+ 'width' => 281,
420
+ 'height' => 190,
421
+ 'url' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/pigeon.png',
422
+ 'thumb_url' => WYSIJA_EDITOR_IMG.'default-newsletter/newsletter/pigeon-150x150.png'
423
+ )
424
+ )
425
+ );
426
+ $wjEngine = WYSIJA::get('wj_engine', 'helper');
427
+ $wjEngine->setData($dataEmail['wj_data']);
428
+ $result = false;
429
+ $dataEmail['params'] = base64_encode(serialize($dataEmail['params']));
430
+ $dataEmail['wj_styles'] = base64_encode(serialize($dataEmail['wj_styles']));
431
+ $dataEmail['wj_data'] = base64_encode(serialize($dataEmail['wj_data']));
432
+
433
+
434
+ $dataEmail['replyto_name']=$dataEmail['from_name']=$valuesconfig['from_name'];
435
+ $dataEmail['replyto_email']=$dataEmail['from_email']=$valuesconfig['from_email'];
436
+ $data['email']['email_id']=$modelEmail->insert($dataEmail);
437
+
438
+
439
+
440
+ $modelEmail = WYSIJA::get('email', 'model');
441
+ $emailData = $modelEmail->getOne(array('wj_styles', 'subject', 'params', 'email_id'), array('email_id' => $data['email']['email_id']));
442
+
443
+ $wjEngine->setStyles($emailData['wj_styles'], true);
444
+
445
+ $values = array('wj_data' => $wjEngine->getEncoded('data'));
446
+ $values['body'] = $wjEngine->renderEmail($emailData);
447
+ $result = $modelEmail->update($values, array('email_id' => $data['email']['email_id']));
448
+
449
+ }
450
+
451
+ /**
452
+ * this function creates the table from a specified sql file and test that the tables have been properly created.
453
+ * TODO it should just need a SQL file and no other parameter
454
+ * @global type $wpdb
455
+ * @param type $sql_file
456
+ * @param type $main_table_ms
457
+ * @return boolean
458
+ */
459
+ function createTables($sql_file, $main_table_ms=false){
460
+ // prepare some parameters
461
+ $model_user=WYSIJA::get('user','model');
462
+ $prefix = $model_user->getPrefix();
463
+ $array_tables_to_test = array('user_list','user','list','campaign','campaign_list','email','user_field','queue','user_history','email_user_stat','url','email_user_url','url_mail');
464
+
465
+ // if we are running a ms file we use a different prefix and we verify different tables
466
+ if($main_