Events Manager - Version 5.0.3

Version Description

  • searching from/to without one or another date works as intended
  • fixed various old-named properties (and refreshed old properties after object save)
  • fixed overriding front-end edit links from within admin area
  • fixed #_EDITEVENTURL
  • fixed location placeholder output/filter function
  • fixed missing php in opening tag of search template
Download this release

Release Info

Developer netweblogic
Plugin Icon 128x128 Events Manager
Version 5.0.3
Comparing to
See all releases

Code changes from version 3.0.98 to 5.0.3

Files changed (63) hide show
  1. admin/bookings/em-cancelled.php +136 -132
  2. admin/bookings/em-confirmed.php +143 -139
  3. admin/bookings/em-events.php +178 -178
  4. admin/bookings/em-pending.php +158 -154
  5. admin/bookings/em-person.php +151 -149
  6. admin/bookings/em-rejected.php +136 -132
  7. admin/em-admin.php +216 -284
  8. admin/em-bookings.php +421 -287
  9. admin/em-categories.php +0 -177
  10. admin/em-docs.php +285 -192
  11. admin/em-event.php +0 -627
  12. admin/em-events.php +0 -242
  13. admin/em-help.php +68 -51
  14. admin/em-locations-search.php +7 -3
  15. admin/em-locations.php +0 -268
  16. admin/em-ms-locations.php +55 -0
  17. admin/em-ms-options.php +213 -0
  18. admin/em-options.php +1468 -443
  19. admin/em-people.php +0 -192
  20. buddypress/bp-em-activity.php +126 -0
  21. buddypress/bp-em-core.php +227 -0
  22. buddypress/bp-em-groups.php +65 -0
  23. buddypress/bp-em-notifications.php +82 -0
  24. buddypress/bp-em-templatetags.php +21 -0
  25. buddypress/screens/attending.php +29 -0
  26. buddypress/screens/group-events.php +30 -0
  27. buddypress/screens/my-bookings.php +35 -0
  28. buddypress/screens/my-events.php +56 -0
  29. buddypress/screens/my-group-events.php +27 -0
  30. buddypress/screens/my-locations.php +52 -0
  31. buddypress/screens/profile.php +32 -0
  32. classes/em-booking.php +676 -378
  33. classes/em-bookings.php +585 -487
  34. classes/em-calendar.php +339 -366
  35. classes/em-categories-taxonomy.php +84 -0
  36. classes/em-categories.php +319 -156
  37. classes/em-category-taxonomy.php +67 -0
  38. classes/em-category.php +213 -151
  39. classes/em-event-post-admin.php +373 -0
  40. classes/em-event-post.php +256 -0
  41. classes/em-event-posts-admin.php +243 -0
  42. classes/em-event.php +1790 -1005
  43. classes/em-events.php +375 -253
  44. classes/em-location-post-admin.php +123 -0
  45. classes/em-location-post.php +67 -0
  46. classes/em-location-posts-admin.php +68 -0
  47. classes/em-location.php +662 -306
  48. classes/em-locations.php +305 -212
  49. classes/em-mailer.php +96 -79
  50. classes/em-map.php +0 -60
  51. classes/em-notices.php +196 -0
  52. classes/em-object.php +889 -485
  53. classes/em-people.php +33 -128
  54. classes/em-permalinks.php +200 -0
  55. classes/em-person.php +91 -153
  56. classes/em-recurrence.php +0 -89
  57. classes/em-tag-taxonomy.php +63 -0
  58. classes/em-tag.php +135 -0
  59. classes/em-ticket-booking.php +223 -0
  60. classes/em-ticket.php +340 -0
  61. classes/em-tickets-bookings.php +283 -0
  62. classes/em-tickets.php +241 -0
  63. classes/phpmailer/class.phpmailer.php +2128 -1426
admin/bookings/em-cancelled.php CHANGED
@@ -1,133 +1,137 @@
1
- <?php
2
- /**
3
- * Generates a "widget" table of confirmed bookings for a specific event.
4
- *
5
- * @param int $event_id
6
- */
7
- function em_bookings_cancelled_table(){
8
- global $EM_Event, $wpdb, $current_user;
9
-
10
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_cancelled_table' );
11
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
12
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
13
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
14
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
15
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
16
-
17
- if( is_object($EM_Event) ){
18
- $bookings = $EM_Event->get_bookings()->get_cancelled_bookings();
19
- }else{
20
- return false;
21
- }
22
- $bookings_count = (is_array($bookings)) ? count($bookings):0;
23
- ?>
24
- <div class='wrap em_bookings_pending_table em_obj'>
25
- <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
26
- <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
27
- <!--
28
- <ul class="subsubsub">
29
- <li>
30
- <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
31
- </li>
32
- </ul>
33
- <p class="search-box">
34
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search'); ?>:</label>
35
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
36
- <input type="submit" value="<?php _e('Search'); ?>" class="button" />
37
- </p>
38
- -->
39
- <?php if ( $bookings_count >= $limit ) : ?>
40
- <div class='tablenav'>
41
- <!--
42
- <div class="alignleft actions">
43
- <select name="action">
44
- <option value="-1" selected="selected">
45
- <?php _e('Bulk Actions'); ?>
46
- </option>
47
- <option value="approve">
48
- <?php _e('Approve', 'dbem'); ?>
49
- </option>
50
- <option value="decline">
51
- <?php _e('Decline', 'dbem'); ?>
52
- </option>
53
- </select>
54
- <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
55
- </div>
56
- -->
57
- <!--
58
- <div class="view-switch">
59
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
60
- </div>
61
- -->
62
- <?php
63
- if ( $bookings_count >= $limit ) {
64
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
65
- $bookings_nav = em_admin_paginate( $page_link_template, $bookings_count, $limit, $page, 5);
66
- echo $bookings_nav;
67
- }
68
- ?>
69
- <div class="clear"></div>
70
- </div>
71
- <?php endif; ?>
72
- <div class="clear"></div>
73
- <?php if( $bookings_count > 0 ): ?>
74
- <div class='table-wrap'>
75
- <table id='dbem-bookings-table' class='widefat post fixed'>
76
- <thead>
77
- <tr>
78
- <th class='manage-column column-cb check-column' scope='col'>
79
- <input class='select-all' type="checkbox" value='1' />
80
- </th>
81
- <th class='manage-column' scope='col'>Booker</th>
82
- <th class='manage-column' scope='col'>E-mail</th>
83
- <th class='manage-column' scope='col'>Phone number</th>
84
- <th class='manage-column' scope='col'>Spaces</th>
85
- <th class='manage-column' scope='col'>&nbsp;</th>
86
- </tr>
87
- </thead>
88
- <tbody>
89
- <?php
90
- $rowno = 0;
91
- $event_count = 0;
92
- foreach ($bookings as $EM_Booking) {
93
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
94
- $rowno++;
95
- ?>
96
- <tr>
97
- <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->id ?>' name='bookings[]'/></th>
98
- <td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->id; ?>"><?php echo $EM_Booking->person->name ?></a></td>
99
- <td><?php echo $EM_Booking->person->email ?></td>
100
- <td><?php echo $EM_Booking->person->phone ?></td>
101
- <td><?php echo $EM_Booking->seats ?></td>
102
- <td>
103
- <?php
104
- $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->id));
105
- $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->id));
106
- ?>
107
- <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Restore','dbem'); ?></a> |
108
- <a class="em-bookings-edit" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->id; ?>"><?php _e('Edit'); ?></a> |
109
- <a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a>
110
- </td>
111
- </tr>
112
- <?php
113
- }
114
- $event_count++;
115
- }
116
- ?>
117
- </tbody>
118
- </table>
119
- </div>
120
- <?php else: ?>
121
- <?php _e('No cancelled bookings.', 'dbem'); ?>
122
- <?php endif; ?>
123
- </form>
124
- <?php if( !empty($bookings_nav) ) : ?>
125
- <div class='tablenav'>
126
- <?php echo $bookings_nav; ?>
127
- <div class="clear"></div>
128
- </div>
129
- <?php endif; ?>
130
- </div>
131
- <?php
132
- }
 
 
 
 
133
  ?>
1
+ <?php
2
+ /**
3
+ * Generates a "widget" table of confirmed bookings for a specific event.
4
+ *
5
+ * @param int $event_id
6
+ */
7
+ function em_bookings_cancelled_table(){
8
+ global $EM_Event, $EM_Ticket, $wpdb, $current_user;
9
+
10
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_cancelled_table' );
11
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
12
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
13
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
14
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
15
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
16
+
17
+ if( is_object($EM_Ticket) ){
18
+ $EM_Bookings = $EM_Ticket->get_bookings()->get_cancelled_bookings();
19
+ }else{
20
+ if( is_object($EM_Event) ){
21
+ $EM_Bookings = $EM_Event->get_bookings()->get_cancelled_bookings();
22
+ }else{
23
+ return false;
24
+ }
25
+ }
26
+ $bookings_count = (is_array($EM_Bookings->bookings)) ? count($EM_Bookings->bookings):0;
27
+ ?>
28
+ <div class='wrap em_bookings_pending_table em_obj'>
29
+ <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
30
+ <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
31
+ <!--
32
+ <ul class="subsubsub">
33
+ <li>
34
+ <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
35
+ </li>
36
+ </ul>
37
+ <p class="search-box">
38
+ <label class="screen-reader-text" for="post-search-input"><?php _e('Search', 'dbem'); ?>:</label>
39
+ <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
40
+ <input type="submit" value="<?php _e('Search', 'dbem'); ?>" class="button" />
41
+ </p>
42
+ -->
43
+ <?php if ( $bookings_count >= $limit ) : ?>
44
+ <div class='tablenav'>
45
+ <!--
46
+ <div class="alignleft actions">
47
+ <select name="action">
48
+ <option value="-1" selected="selected">
49
+ <?php _e('Bulk Actions', 'dbem'); ?>
50
+ </option>
51
+ <option value="approve">
52
+ <?php _e('Approve', 'dbem'); ?>
53
+ </option>
54
+ <option value="decline">
55
+ <?php _e('Decline', 'dbem'); ?>
56
+ </option>
57
+ </select>
58
+ <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
59
+ </div>
60
+ -->
61
+ <!--
62
+ <div class="view-switch">
63
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
64
+ </div>
65
+ -->
66
+ <?php
67
+ if ( $bookings_count >= $limit ) {
68
+ $bookings_nav = em_admin_paginate( $bookings_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
69
+ echo $bookings_nav;
70
+ }
71
+ ?>
72
+ <div class="clear"></div>
73
+ </div>
74
+ <?php endif; ?>
75
+ <div class="clear"></div>
76
+ <?php if( $bookings_count > 0 ): ?>
77
+ <div class='table-wrap'>
78
+ <table id='dbem-bookings-table' class='widefat post '>
79
+ <thead>
80
+ <tr>
81
+ <th class='manage-column column-cb check-column' scope='col'>
82
+ <input class='select-all' type="checkbox" value='1' />
83
+ </th>
84
+ <th class='manage-column' scope='col'><?php _e('Booker', 'dbem'); ?></th>
85
+ <th class='manage-column' scope='col'><?php _e('E-mail', 'dbem'); ?></th>
86
+ <th class='manage-column' scope='col'><?php _e('Phone number', 'dbem'); ?></th>
87
+ <th class='manage-column' scope='col'><?php _e('Spaces', 'dbem'); ?></th>
88
+ <th class='manage-column' scope='col'>&nbsp;</th>
89
+ </tr>
90
+ </thead>
91
+ <tbody>
92
+ <?php
93
+ $rowno = 0;
94
+ $event_count = 0;
95
+ foreach ($EM_Bookings->bookings as $EM_Booking) {
96
+ if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
97
+ $rowno++;
98
+ ?>
99
+ <tr>
100
+ <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->booking_id ?>' name='bookings[]'/></th>
101
+ <td><a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->ID; ?>"><?php echo $EM_Booking->person->get_name() ?></a></td>
102
+ <td><?php echo $EM_Booking->person->user_email ?></td>
103
+ <td><?php echo $EM_Booking->person->phone ?></td>
104
+ <td><?php echo $EM_Booking->get_spaces() ?></td>
105
+ <td>
106
+ <?php
107
+ $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->booking_id));
108
+ $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->booking_id));
109
+ $edit_url = em_add_get_params($_SERVER['REQUEST_URI'], array('booking_id'=>$EM_Booking->booking_id, 'em_ajax'=>null, 'em_obj'=>null));
110
+ ?>
111
+ <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Restore','dbem'); ?></a> |
112
+ <a class="em-bookings-edit" href="<?php echo $edit_url; ?>"><?php _e('Edit/View','dbem'); ?></a> |
113
+ <a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a>
114
+ </td>
115
+ </tr>
116
+ <?php
117
+ }
118
+ $event_count++;
119
+ }
120
+ ?>
121
+ </tbody>
122
+ </table>
123
+ </div>
124
+ <?php else: ?>
125
+ <?php _e('No cancelled bookings.', 'dbem'); ?>
126
+ <?php endif; ?>
127
+ </form>
128
+ <?php if( !empty($bookings_nav) && $EM_Bookings >= $limit ) : ?>
129
+ <div class='tablenav'>
130
+ <?php echo $bookings_nav; ?>
131
+ <div class="clear"></div>
132
+ </div>
133
+ <?php endif; ?>
134
+ </div>
135
+ <?php
136
+ }
137
  ?>
admin/bookings/em-confirmed.php CHANGED
@@ -1,140 +1,144 @@
1
- <?php
2
-
3
- /**
4
- * Generates a "widget" table of confirmed bookings for a specific event.
5
- *
6
- * @param int $event_id
7
- */
8
- function em_bookings_confirmed_table(){
9
- global $EM_Event, $wpdb, $current_user;
10
-
11
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
12
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
13
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
14
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
15
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
16
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
17
-
18
- if( is_object($EM_Event) ){
19
- $bookings = $EM_Event->get_bookings()->get_bookings();
20
- }else{
21
- return false;
22
- }
23
- $bookings_count = (is_array($bookings)) ? count($bookings):0;
24
- ?>
25
- <div class='wrap em_bookings_pending_table em_obj'>
26
- <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
27
- <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
28
- <!--
29
- <ul class="subsubsub">
30
- <li>
31
- <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
32
- </li>
33
- </ul>
34
- <p class="search-box">
35
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search'); ?>:</label>
36
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
37
- <input type="submit" value="<?php _e('Search'); ?>" class="button" />
38
- </p>
39
- -->
40
- <?php if ( $bookings_count >= $limit ) : ?>
41
- <div class='tablenav'>
42
- <!--
43
- <div class="alignleft actions">
44
- <select name="action">
45
- <option value="-1" selected="selected">
46
- <?php _e('Bulk Actions'); ?>
47
- </option>
48
- <option value="approve">
49
- <?php _e('Approve', 'dbem'); ?>
50
- </option>
51
- <option value="decline">
52
- <?php _e('Decline', 'dbem'); ?>
53
- </option>
54
- </select>
55
- <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
56
- </div>
57
- -->
58
- <!--
59
- <div class="view-switch">
60
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
61
- </div>
62
- -->
63
- <?php
64
- if ( $bookings_count >= $limit ) {
65
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
66
- $bookings_nav = em_admin_paginate( $page_link_template, $bookings_count, $limit, $page, 5);
67
- echo $bookings_nav;
68
- }
69
- ?>
70
- <div class="clear"></div>
71
- </div>
72
- <?php endif; ?>
73
- <div class="clear"></div>
74
- <?php if( $bookings_count > 0 ): ?>
75
- <div class='table-wrap'>
76
- <table id='dbem-bookings-table' class='widefat post fixed'>
77
- <thead>
78
- <tr>
79
- <th class='manage-column column-cb check-column' scope='col'>
80
- <input class='select-all' type="checkbox" value='1' />
81
- </th>
82
- <th class='manage-column' scope='col'>Booker</th>
83
- <th class='manage-column' scope='col'>E-mail</th>
84
- <th class='manage-column' scope='col'>Phone number</th>
85
- <th class='manage-column' scope='col'>Spaces</th>
86
- <th class='manage-column' scope='col'>&nbsp;</th>
87
- </tr>
88
- </thead>
89
- <tbody>
90
- <?php
91
- $rowno = 0;
92
- $event_count = 0;
93
- foreach ($bookings as $EM_Booking) {
94
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
95
- $rowno++;
96
- ?>
97
- <tr>
98
- <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->id ?>' name='bookings[]'/></th>
99
- <td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person_id; ?>"><?php echo $EM_Booking->person->name ?></a></td>
100
- <td><?php echo $EM_Booking->person->email ?></td>
101
- <td><?php echo $EM_Booking->person->phone ?></td>
102
- <td><?php echo $EM_Booking->seats ?></td>
103
- <td>
104
- <?php
105
- $unapprove_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_unapprove', 'booking_id'=>$EM_Booking->id));
106
- $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'booking_id'=>$EM_Booking->id));
107
- $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->id));
108
- ?>
109
- <?php if( get_option('dbem_bookings_approval') ): ?>
110
- <a class="em-bookings-unapprove" href="<?php echo $unapprove_url ?>"><?php _e('Unapprove','dbem'); ?></a> |
111
- <?php else: ?>
112
- <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
113
- <?php endif; ?>
114
- <a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a> |
115
- <a class="em-bookings-edit" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->id; ?>"><?php _e('Edit'); ?></a>
116
- </td>
117
- </tr>
118
- <?php
119
- }
120
- $event_count++;
121
- }
122
- ?>
123
- </tbody>
124
- </table>
125
- </div>
126
- <?php else: ?>
127
- <?php _e('No confirmed bookings.', 'dbem'); ?>
128
- <?php endif; ?>
129
- </form>
130
- <?php if( !empty($bookings_nav) ) : ?>
131
- <div class='tablenav'>
132
- <?php echo $bookings_nav; ?>
133
- <div class="clear"></div>
134
- </div>
135
- <?php endif; ?>
136
- </div>
137
- <?php
138
-
139
- }
 
 
 
 
140
  ?>
1
+ <?php
2
+
3
+ /**
4
+ * Generates a "widget" table of confirmed bookings for a specific event.
5
+ *
6
+ * @param int $event_id
7
+ */
8
+ function em_bookings_confirmed_table(){
9
+ global $EM_Event, $EM_Ticket, $wpdb, $current_user;
10
+
11
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
12
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
13
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
14
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
15
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
16
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
17
+
18
+ if( is_object($EM_Ticket) ){
19
+ $EM_Bookings = $EM_Ticket->get_bookings()->get_bookings();
20
+ }else{
21
+ if( is_object($EM_Event) ){
22
+ $EM_Bookings = $EM_Event->get_bookings()->get_bookings();
23
+ }else{
24
+ return false;
25
+ }
26
+ }
27
+ $bookings_count = (is_array($EM_Bookings->bookings)) ? count($EM_Bookings->bookings):0;
28
+ ?>
29
+ <div class='wrap em_bookings_pending_table em_obj'>
30
+ <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
31
+ <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
32
+ <!--
33
+ <ul class="subsubsub">
34
+ <li>
35
+ <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
36
+ </li>
37
+ </ul>
38
+ <p class="search-box">
39
+ <label class="screen-reader-text" for="post-search-input"><?php _e('Search', 'dbem'); ?>:</label>
40
+ <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
41
+ <input type="submit" value="<?php _e('Search', 'dbem'); ?>" class="button" />
42
+ </p>
43
+ -->
44
+ <?php if ( $bookings_count >= $limit ) : ?>
45
+ <div class='tablenav'>
46
+ <!--
47
+ <div class="alignleft actions">
48
+ <select name="action">
49
+ <option value="-1" selected="selected">
50
+ <?php _e('Bulk Actions', 'dbem'); ?>
51
+ </option>
52
+ <option value="approve">
53
+ <?php _e('Approve', 'dbem'); ?>
54
+ </option>
55
+ <option value="decline">
56
+ <?php _e('Decline', 'dbem'); ?>
57
+ </option>
58
+ </select>
59
+ <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
60
+ </div>
61
+ -->
62
+ <!--
63
+ <div class="view-switch">
64
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
65
+ </div>
66
+ -->
67
+ <?php
68
+ if ( $bookings_count >= $limit ) {
69
+ $bookings_nav = em_admin_paginate($bookings_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
70
+ echo $bookings_nav;
71
+ }
72
+ ?>
73
+ <div class="clear"></div>
74
+ </div>
75
+ <?php endif; ?>
76
+ <div class="clear"></div>
77
+ <?php if( $bookings_count > 0 ): ?>
78
+ <div class='table-wrap'>
79
+ <table id='dbem-bookings-table' class='widefat post '>
80
+ <thead>
81
+ <tr>
82
+ <th class='manage-column column-cb check-column' scope='col'>
83
+ <input class='select-all' type="checkbox" value='1' />
84
+ </th>
85
+ <th class='manage-column' scope='col'><?php _e('Booker', 'dbem'); ?></th>
86
+ <th class='manage-column' scope='col'><?php _e('E-mail', 'dbem'); ?></th>
87
+ <th class='manage-column' scope='col'><?php _e('Phone number', 'dbem'); ?></th>
88
+ <th class='manage-column' scope='col'><?php _e('Spaces', 'dbem'); ?></th>
89
+ <th class='manage-column' scope='col'>&nbsp;</th>
90
+ </tr>
91
+ </thead>
92
+ <tbody>
93
+ <?php
94
+ $rowno = 0;
95
+ $event_count = 0;
96
+ foreach ($EM_Bookings->bookings as $EM_Booking) {
97
+ if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
98
+ $rowno++;
99
+ ?>
100
+ <tr>
101
+ <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->booking_id ?>' name='bookings[]'/></th>
102
+ <td><a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person_id; ?>"><?php echo $EM_Booking->person->get_name() ?></a></td>
103
+ <td><?php echo $EM_Booking->person->user_email ?></td>
104
+ <td><?php echo $EM_Booking->person->phone ?></td>
105
+ <td><?php echo $EM_Booking->get_spaces() ?></td>
106
+ <td>
107
+ <?php
108
+ $unapprove_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_unapprove', 'booking_id'=>$EM_Booking->booking_id));
109
+ $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'booking_id'=>$EM_Booking->booking_id));
110
+ $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->booking_id));
111
+ $edit_url = em_add_get_params($_SERVER['REQUEST_URI'], array('booking_id'=>$EM_Booking->booking_id, 'em_ajax'=>null, 'em_obj'=>null));
112
+ ?>
113
+ <?php if( current_user_can('manage_bookings') && get_option('dbem_bookings_approval') ): ?>
114
+ <a class="em-bookings-unapprove" href="<?php echo $unapprove_url ?>"><?php _e('Unapprove','dbem'); ?></a> |
115
+ <?php else: ?>
116
+ <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
117
+ <?php endif; ?>
118
+ <span class="trash"><a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a></span> |
119
+ <a class="em-bookings-edit" href="<?php echo $edit_url; ?>"><?php _e('Edit/View','dbem'); ?></a>
120
+ </td>
121
+ </tr>
122
+ <?php
123
+ }
124
+ $event_count++;
125
+ }
126
+ ?>
127
+ </tbody>
128
+ </table>
129
+ </div>
130
+ <?php else: ?>
131
+ <?php _e('No confirmed bookings.', 'dbem'); ?>
132
+ <?php endif; ?>
133
+ </form>
134
+ <?php if( !empty($bookings_nav) && $EM_Bookings >= $limit ) : ?>
135
+ <div class='tablenav'>
136
+ <?php echo $bookings_nav; ?>
137
+ <div class="clear"></div>
138
+ </div>
139
+ <?php endif; ?>
140
+ </div>
141
+ <?php
142
+
143
+ }
144
  ?>
admin/bookings/em-events.php CHANGED
@@ -1,179 +1,179 @@
1
- <?php
2
-
3
- /**
4
- * Determines whether to show event page or events page, and saves any updates to the event or events
5
- * @return null
6
- */
7
- function em_bookings_events_table() {
8
- //TODO Simplify panel for events, use form flags to detect certain actions (e.g. submitted, etc)
9
- global $wpdb;
10
- global $EM_Event;
11
-
12
- $scope_names = array (
13
- 'past' => __ ( 'Past events', 'dbem' ),
14
- 'all' => __ ( 'All events', 'dbem' ),
15
- 'future' => __ ( 'Future events', 'dbem' )
16
- );
17
-
18
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_events_table' );
19
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
20
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
21
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
22
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
23
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
24
- $scope = ( $action_scope && !empty($_GET ['scope']) && array_key_exists($_GET ['scope'], $scope_names) ) ? $_GET ['scope']:'future';
25
-
26
- // No action, only showing the events list
27
- switch ($scope) {
28
- case "past" :
29
- $title = __ ( 'Past Events', 'dbem' );
30
- break;
31
- case "all" :
32
- $title = __ ( 'All Events', 'dbem' );
33
- break;
34
- default :
35
- $title = __ ( 'Future Events', 'dbem' );
36
- $scope = "future";
37
- }
38
- $events = EM_Events::get( array('scope'=>$scope, 'limit'=>0, 'order'=>$order, 'rsvp'=>true ) );
39
- $events_count = count ( $events );
40
-
41
- $use_events_end = get_option ( 'dbem_use_event_end' );
42
- ?>
43
- <div class="wrap em_bookings_events_table em_obj">
44
- <form id="posts-filter" action="<?php echo $_SERVER['REQUEST_URI'] ?>" method="get">
45
- <input type="hidden" name="em_obj" value="em_bookings_events_table" />
46
- <?php if(!empty($_GET['page'])): ?>
47
- <input type='hidden' name='page' value='events-manager-bookings' />
48
- <?php endif; ?>
49
- <ul class="subsubsub">
50
- <li><a href='#' class="current"><?php _e ( 'Total', 'dbem' ); ?> <span class="count">(<?php echo (count ( $events )); ?>)</span></a></li>
51
- </ul>
52
- <div class="tablenav">
53
- <div class="alignleft actions">
54
- <!--
55
- <select name="action">
56
- <option value="-1" selected="selected"><?php _e ( 'Bulk Actions' ); ?></option>
57
- <option value="deleteEvents"><?php _e ( 'Delete selected','dbem' ); ?></option>
58
- </select>
59
- <input type="submit" value="<?php _e ( 'Apply' ); ?>" name="doaction2" id="doaction2" class="button-secondary action" />
60
- -->
61
- <select name="scope">
62
- <?php
63
- foreach ( $scope_names as $key => $value ) {
64
- $selected = "";
65
- if ($key == $scope)
66
- $selected = "selected='selected'";
67
- echo "<option value='$key' $selected>$value</option> ";
68
- }
69
- ?>
70
- </select>
71
- <input id="post-query-submit" class="button-secondary" type="submit" value="<?php _e ( 'Filter' )?>" />
72
- </div>
73
- <!--
74
- <div class="view-switch">
75
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
76
- </div>
77
- -->
78
- <?php
79
- if ( $events_count >= $limit ) {
80
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_events_table'));
81
- $events_nav = em_admin_paginate( $page_link_template, $events_count, $limit, $page, 5);
82
- echo $events_nav;
83
- }
84
- ?>
85
- <br class="clear" />
86
- </div>
87
-
88
- <?php
89
- if (empty ( $events )) {
90
- // TODO localize
91
- _e ( 'no events','dbem' );
92
- } else {
93
- ?>
94
- <div class='table-wrap'>
95
- <table class="widefat">
96
- <thead>
97
- <tr>
98
- <th class='manage-column column-cb check-column' scope='col'>
99
- <input class='select-all' type="checkbox" value='1' />
100
- </th>
101
- <th><?php _e ( 'Event', 'dbem' ); ?></th>
102
- <th><?php _e ( 'Date and time', 'dbem' ); ?></th>
103
- </tr>
104
- </thead>
105
- <tbody>
106
- <?php
107
- $rowno = 0;
108
- $event_count = 0;
109
- foreach ( $events as $event ) {
110
- /* @var $event EM_Event */
111
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
112
- $rowno++;
113
- $class = ($rowno % 2) ? ' class="alternate"' : '';
114
- // FIXME set to american
115
- $localised_start_date = date_i18n('D d M Y', $event->start);
116
- $localised_end_date = date_i18n('D d M Y', $event->end);
117
- $style = "";
118
- $today = date ( "Y-m-d" );
119
-
120
- if ($event->start_date < $today && $event->end_date < $today){
121
- $style = "style ='background-color: #FADDB7;'";
122
- }
123
- ?>
124
- <tr <?php echo "$class $style"; ?>>
125
-
126
- <td>
127
- <input type='checkbox' class='row-selector' value='<?php echo $event->id; ?>' name='events[]' />
128
- </td>
129
- <td>
130
- <strong>
131
- <a class="row-title" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;event_id=<?php echo $event->id ?>"><?php echo ($event->name); ?></a>
132
- </strong>
133
- &ndash;
134
- <?php _e("Booked Seats",'dbem') ?>: <?php echo $event->get_bookings()->get_booked_seats()."/".$event->seats ?>
135
- <?php if( get_option('dbem_bookings_approval') == 1 ) : ?>
136
- | <?php _e("Pending",'dbem') ?>: <?php echo $event->get_bookings()->get_pending_seats(); ?>
137
- <?php endif; ?>
138
- </td>
139
-
140
- <td>
141
- <?php echo $localised_start_date; ?>
142
- <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
143
- &ndash;
144
- <?php
145
- //TODO Should 00:00 - 00:00 be treated as an all day event?
146
- echo substr ( $event->start_time, 0, 5 ) . " - " . substr ( $event->end_time, 0, 5 );
147
- ?>
148
- </td>
149
- </tr>
150
- <?php
151
- }
152
- $event_count++;
153
- }
154
- ?>
155
- </tbody>
156
- </table>
157
- </div>
158
- <?php
159
- } // end of table
160
- ?>
161
- <div class='tablenav'>
162
- <div class="alignleft actions">
163
- <br class='clear' />
164
- </div>
165
- <?php if( !empty($events_nav) ) : ?>
166
- <div class="tablenav-pages">
167
- <?php
168
- echo $events_nav;
169
- ?>
170
- </div>
171
- <?php endif; ?>
172
- <br class='clear' />
173
- </div>
174
- </form>
175
- </div>
176
- <?php
177
- }
178
-
179
  ?>
1
+ <?php
2
+
3
+ /**
4
+ * Determines whether to show event page or events page, and saves any updates to the event or events
5
+ * @return null
6
+ */
7
+ function em_bookings_events_table() {
8
+ //TODO Simplify panel for events, use form flags to detect certain actions (e.g. submitted, etc)
9
+ global $wpdb;
10
+ global $EM_Event;
11
+
12
+ $scope_names = array (
13
+ 'past' => __ ( 'Past events', 'dbem' ),
14
+ 'all' => __ ( 'All events', 'dbem' ),
15
+ 'future' => __ ( 'Future events', 'dbem' )
16
+ );
17
+
18
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_events_table' );
19
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
20
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
21
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
22
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
23
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
24
+ $scope = ( $action_scope && !empty($_GET ['scope']) && array_key_exists($_GET ['scope'], $scope_names) ) ? $_GET ['scope']:'future';
25
+
26
+ // No action, only showing the events list
27
+ switch ($scope) {
28
+ case "past" :
29
+ $title = __ ( 'Past Events', 'dbem' );
30
+ break;
31
+ case "all" :
32
+ $title = __ ( 'All Events', 'dbem' );
33
+ break;
34
+ default :
35
+ $title = __ ( 'Future Events', 'dbem' );
36
+ $scope = "future";
37
+ }
38
+ $owner = !current_user_can('edit_others_events') ? get_current_user_id() : false;
39
+ $events = EM_Events::get( array('scope'=>$scope, 'limit'=>0, 'order'=>$order, 'bookings'=>true, 'owner' => $owner ) );
40
+ $events_count = count ( $events );
41
+
42
+ $use_events_end = get_option ( 'dbem_use_event_end' );
43
+ ?>
44
+ <div class="wrap em_bookings_events_table em_obj">
45
+ <form id="posts-filter" action="" method="get">
46
+ <input type="hidden" name="em_obj" value="em_bookings_events_table" />
47
+ <?php if(!empty($_GET['page'])): ?>
48
+ <input type='hidden' name='page' value='events-manager-bookings' />
49
+ <?php endif; ?>
50
+ <ul class="subsubsub">
51
+ <li><a href='#' class="current"><?php _e ( 'Total', 'dbem' ); ?> <span class="count">(<?php echo (count ( $events )); ?>)</span></a></li>
52
+ </ul>
53
+ <div class="tablenav">
54
+ <div class="alignleft actions">
55
+ <!--
56
+ <select name="action">
57
+ <option value="-1" selected="selected"><?php _e ( 'Bulk Actions' ); ?></option>
58
+ <option value="deleteEvents"><?php _e ( 'Delete selected','dbem' ); ?></option>
59
+ </select>
60
+ <input type="submit" value="<?php _e ( 'Apply' ); ?>" name="doaction2" id="doaction2" class="button-secondary action" />
61
+ -->
62
+ <select name="scope">
63
+ <?php
64
+ foreach ( $scope_names as $key => $value ) {
65
+ $selected = "";
66
+ if ($key == $scope)
67
+ $selected = "selected='selected'";
68
+ echo "<option value='$key' $selected>$value</option> ";
69
+ }
70
+ ?>
71
+ </select>
72
+ <input id="post-query-submit" class="button-secondary" type="submit" value="<?php _e ( 'Filter' )?>" />
73
+ </div>
74
+ <!--
75
+ <div class="view-switch">
76
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
77
+ </div>
78
+ -->
79
+ <?php
80
+ if ( $events_count >= $limit ) {
81
+ $events_nav = em_admin_paginate( $events_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_events_table'));
82
+ echo $events_nav;
83
+ }
84
+ ?>
85
+ <br class="clear" />
86
+ </div>
87
+
88
+ <?php
89
+ if (empty ( $events )) {
90
+ // TODO localize
91
+ _e ( 'no events','dbem' );
92
+ } else {
93
+ ?>
94
+ <div class='table-wrap'>
95
+ <table class="widefat">
96
+ <thead>
97
+ <tr>
98
+ <th class='manage-column column-cb check-column' scope='col'>
99
+ <input class='select-all' type="checkbox" value='1' />
100
+ </th>
101
+ <th><?php _e ( 'Event', 'dbem' ); ?></th>
102
+ <th><?php _e ( 'Date and time', 'dbem' ); ?></th>
103
+ </tr>
104
+ </thead>
105
+ <tbody>
106
+ <?php
107
+ $rowno = 0;
108
+ $event_count = 0;
109
+ foreach ( $events as $event ) {
110
+ /* @var $event EM_Event */
111
+ if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
112
+ $rowno++;
113
+ $class = ($rowno % 2) ? ' class="alternate"' : '';
114
+ // FIXME set to american
115
+ $localised_start_date = date_i18n(get_option('date_format'), $event->start);
116
+ $localised_end_date = date_i18n(get_option('date_format'), $event->end);
117
+ $style = "";
118
+ $today = date ( "Y-m-d" );
119
+
120
+ if ($event->start_date < $today && $event->end_date < $today){
121
+ $style = "style ='background-color: #FADDB7;'";
122
+ }
123
+ ?>
124
+ <tr <?php echo "$class $style"; ?>>
125
+
126
+ <td>
127
+ <input type='checkbox' class='row-selector' value='<?php echo $event->event_id; ?>' name='events[]' />
128
+ </td>
129
+ <td>
130
+ <strong>
131
+ <?php echo $event->output('#_BOOKINGSLINK'); ?>
132
+ </strong>
133
+ &ndash;
134
+ <?php _e("Booked Spaces",'dbem') ?>: <?php echo $event->get_bookings()->get_booked_spaces()."/".$event->get_spaces() ?>
135
+ <?php if( get_option('dbem_bookings_approval') == 1 ) : ?>
136
+ | <?php _e("Pending",'dbem') ?>: <?php echo $event->get_bookings()->get_pending_spaces(); ?>
137
+ <?php endif; ?>
138
+ </td>
139
+
140
+ <td>
141
+ <?php echo $localised_start_date; ?>
142
+ <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
143
+ &ndash;
144
+ <?php
145
+ //TODO Should 00:00 - 00:00 be treated as an all day event?
146
+ echo substr ( $event->start_time, 0, 5 ) . " - " . substr ( $event->end_time, 0, 5 );
147
+ ?>
148
+ </td>
149
+ </tr>
150
+ <?php
151
+ }
152
+ $event_count++;
153
+ }
154
+ ?>
155
+ </tbody>
156
+ </table>
157
+ </div>
158
+ <?php
159
+ } // end of table
160
+ ?>
161
+ <div class='tablenav'>
162
+ <div class="alignleft actions">
163
+ <br class='clear' />
164
+ </div>
165
+ <?php if (!empty($events_nav) && $events_count >= $limit ) : ?>
166
+ <div class="tablenav-pages">
167
+ <?php
168
+ echo $events_nav;
169
+ ?>
170
+ </div>
171
+ <?php endif; ?>
172
+ <br class='clear' />
173
+ </div>
174
+ </form>
175
+ </div>
176
+ <?php
177
+ }
178
+
179
  ?>
admin/bookings/em-pending.php CHANGED
@@ -1,155 +1,159 @@
1
- <?php
2
-
3
- /**
4
- * Generates a "widget" table of pending bookings with some quick admin operation options.
5
- * If event id supplied then only pending bookings for that event will show.
6
- *
7
- * @param int $event_id
8
- */
9
- function em_bookings_pending_table($event_id = false){
10
- global $EM_Event, $wpdb, $current_user;
11
-
12
- if( get_option('dbem_bookings_approval') == 0 ){
13
- return false;
14
- }
15
-
16
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_pending_table' );
17
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
18
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
19
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
20
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
21
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
22
-
23
- if( is_object($EM_Event) ){
24
- $bookings = $EM_Event->get_bookings()->get_pending();
25
- }else{
26
- //To optimize performance, we can do one query here for all pending bookings to show.
27
- $bookings = EM_Bookings::get(array('status'=>0));
28
- $events = array();
29
- //Now let's create events and bookings for this:
30
- foreach($bookings as $EM_Booking){
31
- //create event
32
- if( !array_key_exists($EM_Booking->event_id,$events) ){
33
- $events[$EM_Booking->event_id] = new EM_Event($EM_Booking->event_id);
34
- }
35
- }
36
- }
37
- $bookings_count = (is_array($bookings)) ? count($bookings):0;
38
- ?>
39
- <div class='wrap em_bookings_pending_table em_obj'>
40
- <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
41
- <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
42
- <!--
43
- <ul class="subsubsub">
44
- <li>
45
- <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
46
- </li>
47
- </ul>
48
- <p class="search-box">
49
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search'); ?>:</label>
50
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
51
- <input type="submit" value="<?php _e('Search'); ?>" class="button" />
52
- </p>
53
- -->
54
- <?php if ( $bookings_count >= $limit ) : ?>
55
- <div class='tablenav'>
56
- <!--
57
- <div class="alignleft actions">
58
- <select name="action">
59
- <option value="-1" selected="selected">
60
- <?php _e('Bulk Actions'); ?>
61
- </option>
62
- <option value="approve">
63
- <?php _e('Approve', 'dbem'); ?>
64
- </option>
65
- <option value="decline">
66
- <?php _e('Decline', 'dbem'); ?>
67
- </option>
68
- </select>
69
- <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
70
- </div>
71
- -->
72
- <!--
73
- <div class="view-switch">
74
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
75
- </div>
76
- -->
77
- <?php
78
- if ( $bookings_count >= $limit ) {
79
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_pending_table'));
80
- $bookings_nav = em_admin_paginate( $page_link_template, $bookings_count, $limit, $page, 5);
81
- echo $bookings_nav;
82
- }
83
- ?>
84
- <div class="clear"></div>
85
- </div>
86
- <?php endif; ?>
87
- <div class="clear"></div>
88
- <?php if( $bookings_count > 0 ): ?>
89
- <div class='table-wrap'>
90
- <table id='dbem-bookings-table' class='widefat post fixed'>
91
- <thead>
92
- <tr>
93
- <th class='manage-column column-cb check-column' scope='col'>
94
- <input class='select-all' type="checkbox" value='1' />
95
- </th>
96
- <th class='manage-column' scope='col'>Booker</th>
97
- <?php if( !is_object($EM_Event) ): ?>
98
- <th class='manage-column' scope="col">Event</th>
99
- <?php endif; ?>
100
- <th class='manage-column' scope='col'>E-mail</th>
101
- <th class='manage-column' scope='col'>Phone number</th>
102
- <th class='manage-column' scope='col'>Spaces</th>
103
- <th class='manage-column' scope='col'>&nbsp;</th>
104
- </tr>
105
- </thead>
106
- <tbody>
107
- <?php
108
- $rowno = 0;
109
- $event_count = 0;
110
- foreach ($bookings as $EM_Booking) {
111
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
112
- $rowno++;
113
- ?>
114
- <tr>
115
- <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->id ?>' name='bookings[]'/></th>
116
- <td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->id; ?>"><?php echo $EM_Booking->person->name ?></a></td>
117
- <?php if( !is_object($EM_Event) ): ?>
118
- <td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;event_id=<?php echo $EM_Booking->event_id; ?>"><?php echo $events[$EM_Booking->event_id]->name ?></a></td>
119
- <?php endif; ?>
120
- <td><?php echo $EM_Booking->person->email ?></td>
121
- <td><?php echo $EM_Booking->person->phone ?></td>
122
- <td><?php echo $EM_Booking->seats ?></td>
123
- <td>
124
- <?php
125
- $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->id));
126
- $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'booking_id'=>$EM_Booking->id));
127
- ?>
128
- <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
129
- <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
130
- <a class="em-bookings-edit" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->id; ?>"><?php _e('Edit'); ?></a>
131
- </td>
132
- </tr>
133
- <?php
134
- }
135
- $event_count++;
136
- }
137
- ?>
138
- </tbody>
139
- </table>
140
- </div>
141
- <?php else: ?>
142
- <?php _e('No pending bookings.', 'dbem'); ?>
143
- <?php endif; ?>
144
- </form>
145
- <?php if( !empty($bookings_nav) ) : ?>
146
- <div class='tablenav'>
147
- <?php echo $bookings_nav; ?>
148
- <div class="clear"></div>
149
- </div>
150
- <?php endif; ?>
151
- </div>
152
- <?php
153
-
154
- }
 
 
 
 
155
  ?>
1
+ <?php
2
+
3
+ /**
4
+ * Generates a "widget" table of pending bookings with some quick admin operation options.
5
+ * If event id supplied then only pending bookings for that event will show.
6
+ *
7
+ * @param int $event_id
8
+ */
9
+ function em_bookings_pending_table($event_id = false){
10
+ global $EM_Event, $EM_Ticket, $wpdb, $current_user;
11
+
12
+ if( get_option('dbem_bookings_approval') == 0 ){
13
+ return false;
14
+ }
15
+
16
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_pending_table' );
17
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
18
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
19
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
20
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
21
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
22
+
23
+ if( is_object($EM_Ticket) ){
24
+ $EM_Bookings = $EM_Ticket->get_bookings()->get_pending_bookings();
25
+ }else{
26
+ if( is_object($EM_Event) ){
27
+ $EM_Bookings = $EM_Event->get_bookings()->get_pending_bookings();
28
+ }else{
29
+ //To optimize performance, we can do one query here for all pending bookings to show.
30
+ $EM_Bookings = EM_Bookings::get(array('status'=>0));
31
+ $events = array();
32
+ //Now let's create events and bookings for this:
33
+ foreach($EM_Bookings->bookings as $EM_Booking){
34
+ //create event
35
+ if( !array_key_exists($EM_Booking->event_id,$events) ){
36
+ $events[$EM_Booking->event_id] = new EM_Event($EM_Booking->event_id);
37
+ }
38
+ }
39
+ }
40
+ }
41
+ $bookings_count = (is_array($EM_Bookings->bookings)) ? count($EM_Bookings->bookings):0;
42
+ ?>
43
+ <div class='wrap em_bookings_pending_table em_obj'>
44
+ <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
45
+ <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
46
+ <!--
47
+ <ul class="subsubsub">
48
+ <li>
49
+ <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
50
+ </li>
51
+ </ul>
52
+ <p class="search-box">
53
+ <label class="screen-reader-text" for="post-search-input"><?php _e('Search', 'dbem'); ?>:</label>
54
+ <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
55
+ <input type="submit" value="<?php _e('Search', 'dbem'); ?>" class="button" />
56
+ </p>
57
+ -->
58
+ <?php if ( $bookings_count >= $limit ) : ?>
59
+ <div class='tablenav'>
60
+ <!--
61
+ <div class="alignleft actions">
62
+ <select name="action">
63
+ <option value="-1" selected="selected">
64
+ <?php _e('Bulk Actions', 'dbem'); ?>
65
+ </option>
66
+ <option value="approve">
67
+ <?php _e('Approve', 'dbem'); ?>
68
+ </option>
69
+ <option value="decline">
70
+ <?php _e('Decline', 'dbem'); ?>
71
+ </option>
72
+ </select>
73
+ <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
74
+ </div>
75
+ -->
76
+ <!--
77
+ <div class="view-switch">
78
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
79
+ </div>
80
+ -->
81
+ <?php
82
+ if ( $bookings_count >= $limit ) {
83
+ $bookings_nav = em_admin_paginate( $bookings_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_pending_table'));
84
+ echo $bookings_nav;
85
+ }
86
+ ?>
87
+ <div class="clear"></div>
88
+ </div>
89
+ <?php endif; ?>
90
+ <div class="clear"></div>
91
+ <?php if( $bookings_count > 0 ): ?>
92
+ <div class='table-wrap'>
93
+ <table id='dbem-bookings-table' class='widefat post '>
94
+ <thead>
95
+ <tr>
96
+ <th class='manage-column column-cb check-column' scope='col'>
97
+ <input class='select-all' type="checkbox" value='1' />
98
+ </th>
99
+ <th class='manage-column' scope='col'><?php _e('Booker', 'dbem'); ?></th>
100
+ <?php if( !is_object($EM_Event) && !is_object($EM_Ticket) ): ?>
101
+ <th class='manage-column' scope="col"><?php _e('Event', 'dbem'); ?></th>
102
+ <?php endif; ?>
103
+ <th class='manage-column' scope='col'><?php _e('E-mail', 'dbem'); ?></th>
104
+ <th class='manage-column' scope='col'><?php _e('Phone number', 'dbem'); ?></th>
105
+ <th class='manage-column' scope='col'><?php _e('Spaces', 'dbem'); ?></th>
106
+ <th class='manage-column' scope='col'>&nbsp;</th>
107
+ </tr>
108
+ </thead>
109
+ <tbody>
110
+ <?php
111
+ $rowno = 0;
112
+ $event_count = 0;
113
+ foreach ($EM_Bookings->bookings as $EM_Booking) {
114
+ if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
115
+ $rowno++;
116
+ ?>
117
+ <tr>
118
+ <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->booking_id ?>' name='bookings[]'/></th>
119
+ <td><a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->ID; ?>"><?php echo $EM_Booking->person->get_name() ?></a></td>
120
+ <?php if( !is_object($EM_Event) && !is_object($EM_Ticket) ): ?>
121
+ <td><a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;event_id=<?php echo $EM_Booking->event_id; ?>"><?php echo $events[$EM_Booking->event_id]->name ?></a></td>
122
+ <?php endif; ?>
123
+ <td><?php echo $EM_Booking->person->user_email ?></td>
124
+ <td><?php echo $EM_Booking->person->phone ?></td>
125
+ <td><?php echo $EM_Booking->get_spaces() ?></td>
126
+ <td>
127
+ <?php
128
+ $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->booking_id));
129
+ $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'booking_id'=>$EM_Booking->booking_id));
130
+ $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->booking_id));
131
+ ?>
132
+ <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
133
+ <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
134
+ <span class="trash"><a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a></span> |
135
+ <a class="em-bookings-edit" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->booking_id; ?>"><?php _e('Edit/View','dbem'); ?></a>
136
+ </td>
137
+ </tr>
138
+ <?php
139
+ }
140
+ $event_count++;
141
+ }
142
+ ?>
143
+ </tbody>
144
+ </table>
145
+ </div>
146
+ <?php else: ?>
147
+ <?php _e('No pending bookings.', 'dbem'); ?>
148
+ <?php endif; ?>
149
+ </form>
150
+ <?php if( !empty($bookings_nav) && $EM_Bookings >= $limit ) : ?>
151
+ <div class='tablenav'>
152
+ <?php echo $bookings_nav; ?>
153
+ <div class="clear"></div>
154
+ </div>
155
+ <?php endif; ?>
156
+ </div>
157
+ <?php
158
+ }
159
  ?>
admin/bookings/em-person.php CHANGED
@@ -1,150 +1,152 @@
1
- <?php
2
-
3
- /**
4
- * Generates a "widget" table of confirmed bookings for a specific event.
5
- *
6
- * @param int $event_id
7
- */
8
- function em_bookings_person_table(){
9
- global $wpdb, $current_user,$EM_Person;
10
- if(!is_object($EM_Person)){
11
- return false;
12
- }
13
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
14
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
15
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
16
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
17
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
18
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
19
-
20
- $bookings = $EM_Person->get_bookings();
21
- $bookings_count = (is_array($bookings)) ? count($bookings):0;
22
- if($bookings_count > 0){
23
- //Get events here in one query to speed things up
24
- foreach($bookings as $EM_Booking){
25
- $event_ids[] = $EM_Booking->event_id;
26
- }
27
- $events = EM_Events::get($event_ids);
28
- }
29
- ?>
30
- <div class='wrap em_bookings_pending_table em_obj'>
31
- <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
32
- <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
33
- <!--
34
- <ul class="subsubsub">
35
- <li>
36
- <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
37
- </li>
38
- </ul>
39
- <p class="search-box">
40
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search'); ?>:</label>
41
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
42
- <input type="submit" value="<?php _e('Search'); ?>" class="button" />
43
- </p>
44
- -->
45
- <?php if ( $bookings_count >= $limit ) : ?>
46
- <div class='tablenav'>
47
- <!--
48
- <div class="alignleft actions">
49
- <select name="action">
50
- <option value="-1" selected="selected">
51
- <?php _e('Bulk Actions'); ?>
52
- </option>
53
- <option value="approve">
54
- <?php _e('Approve', 'dbem'); ?>
55
- </option>
56
- <option value="decline">
57
- <?php _e('Decline', 'dbem'); ?>
58
- </option>
59
- </select>
60
- <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
61
- </div>
62
- -->
63
- <!--
64
- <div class="view-switch">
65
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
66
- </div>
67
- -->
68
- <?php
69
- if ( $bookings_count >= $limit ) {
70
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
71
- $bookings_nav = em_admin_paginate( $page_link_template, $bookings_count, $limit, $page, 5);
72
- echo $bookings_nav;
73
- }
74
- ?>
75
- <div class="clear"></div>
76
- </div>
77
- <?php endif; ?>
78
- <div class="clear"></div>
79
- <?php if( $bookings_count > 0 ): ?>
80
- <div class='table-wrap'>
81
- <table id='dbem-bookings-table' class='widefat post fixed'>
82
- <thead>
83
- <tr>
84
- <th class='manage-column column-cb check-column' scope='col'>
85
- <input class='select-all' type="checkbox" value='1' />
86
- </th>
87
- <th class='manage-column' scope='col'>Event</th>
88
- <th class='manage-column' scope='col'>Spaces</th>
89
- <th class='manage-column' scope='col'>Status</th>
90
- <th class='manage-column' scope='col'>&nbsp;</th>
91
- </tr>
92
- </thead>
93
- <tbody>
94
- <?php
95
- $rowno = 0;
96
- $event_count = 0;
97
- foreach ($bookings as $EM_Booking) {
98
- $EM_Event = $events[$EM_Booking->event_id];
99
- if( $EM_Event->can_manage() && ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
100
- $rowno++;
101
- ?>
102
- <tr>
103
- <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->id ?>' name='bookings[]'/></th>
104
- <td><a class="row-title" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;event_id=<?php echo $EM_Event->id ?>"><?php echo ($EM_Event->name); ?></a></td>
105
- <td><?php echo $EM_Booking->seats ?></td>
106
- <td><?php echo $EM_Booking->status_array[$EM_Booking->status]; ?>
107
- </td>
108
- <td>
109
- <?php
110
- $unapprove_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_unapprove', 'bookings'=>$EM_Booking->id));
111
- $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'bookings'=>$EM_Booking->id));
112
- $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'bookings'=>$EM_Booking->id));
113
- $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'bookings'=>$EM_Booking->id));
114
- ?>
115
- <?php if( $EM_Booking->status == 0 || $EM_Booking->status == 2 ): ?>
116
- <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
117
- <?php endif; ?>
118
- <?php if( $EM_Booking->status == 2 ): ?>
119
- <a class="em-bookings-unapprove" href="<?php echo $unapprove_url ?>"><?php _e('Unapprove','dbem'); ?></a> |
120
- <?php endif; ?>
121
- <?php if( $EM_Booking->status == 0 || $EM_Booking->status == 1 ): ?>
122
- <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
123
- <?php endif; ?>
124
- <a class="em-bookings-edit" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->id; ?>"><?php _e('Edit'); ?></a> |
125
- <a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a>
126
- </td>
127
- </tr>
128
- <?php
129
- }
130
- $event_count++;
131
- }
132
- ?>
133
- </tbody>
134
- </table>
135
- </div>
136
- <?php else: ?>
137
- <?php _e('No confirmed bookings.', 'dbem'); ?>
138
- <?php endif; ?>
139
- </form>
140
- <?php if( !empty($bookings_nav) ) : ?>
141
- <div class='tablenav'>
142
- <?php echo $bookings_nav; ?>
143
- <div class="clear"></div>
144
- </div>
145
- <?php endif; ?>
146
- </div>
147
- <?php
148
-
149
- }
 
 
150
  ?>
1
+ <?php
2
+
3
+ /**
4
+ * Generates a "widget" table of confirmed bookings for a specific event.
5
+ *
6
+ * @param int $event_id
7
+ */
8
+ function em_bookings_person_table(){
9
+ global $wpdb, $current_user,$EM_Person;
10
+ if(!is_object($EM_Person)){
11
+ return false;
12
+ }
13
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
14
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
15
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
16
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
17
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
18
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
19
+
20
+ $bookings = $EM_Person->get_bookings();
21
+ $bookings_count = count($bookings);
22
+ if($bookings_count > 0){
23
+ //Get events here in one query to speed things up
24
+ foreach($bookings as $EM_Booking){
25
+ $event_ids[] = $EM_Booking->event_id;
26
+ }
27
+ $events = EM_Events::get($event_ids);
28
+ }
29
+ ?>
30
+ <div class='wrap em_bookings_pending_table em_obj'>
31
+ <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
32
+ <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
33
+ <!--
34
+ <ul class="subsubsub">
35
+ <li>
36
+ <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
37
+ </li>
38
+ </ul>
39
+ <p class="search-box">
40
+ <label class="screen-reader-text" for="post-search-input"><?php _e('Search', 'dbem'); ?>:</label>
41
+ <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
42
+ <input type="submit" value="<?php _e('Search', 'dbem'); ?>" class="button" />
43
+ </p>
44
+ -->
45
+ <?php if ( $bookings_count >= $limit ) : ?>
46
+ <div class='tablenav'>
47
+ <!--
48
+ <div class="alignleft actions">
49
+ <select name="action">
50
+ <option value="-1" selected="selected">
51
+ <?php _e('Bulk Actions', 'dbem'); ?>
52
+ </option>
53
+ <option value="approve">
54
+ <?php _e('Approve', 'dbem'); ?>
55
+ </option>
56
+ <option value="decline">
57
+ <?php _e('Decline', 'dbem'); ?>
58
+ </option>
59
+ </select>
60
+ <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
61
+ </div>
62
+ -->
63
+ <!--
64
+ <div class="view-switch">
65
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
66
+ </div>
67
+ -->
68
+ <?php
69
+ if ( $bookings_count >= $limit ) {
70
+ $bookings_nav = em_admin_paginate( $bookings_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
71
+ echo $bookings_nav;
72
+ }
73
+ ?>
74
+ <div class="clear"></div>
75
+ </div>
76
+ <?php endif; ?>
77
+ <div class="clear"></div>
78
+ <?php if( $bookings_count > 0 ): ?>
79
+ <div class='table-wrap'>
80
+ <table id='dbem-bookings-table' class='widefat post '>
81
+ <thead>
82
+ <tr>
83
+ <th class='manage-column column-cb check-column' scope='col'>
84
+ <input class='select-all' type="checkbox" value='1' />
85
+ </th>
86
+ <th class='manage-column' scope='col'><?php _e('Event', 'dbem'); ?></th>
87
+ <th class='manage-column' scope='col'><?php _e('Spaces', 'dbem'); ?></th>
88
+ <th class='manage-column' scope='col'><?php _e('Status', 'dbem'); ?></th>
89
+ <th class='manage-column' scope='col'>&nbsp;</th>
90
+ </tr>
91
+ </thead>
92
+ <tbody>
93
+ <?php
94
+ $rowno = 0;
95
+ $event_count = 0;
96
+ foreach ($bookings as $EM_Booking) {
97
+ $EM_Event = $events[$EM_Booking->event_id];
98
+ if( $EM_Event->can_manage('edit_events','edit_others_events') && ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
99
+ $rowno++;
100
+ ?>
101
+ <tr>
102
+ <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->booking_id ?>' name='bookings[]'/></th>
103
+ <td><a class="row-title" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;event_id=<?php echo $EM_Event->event_id ?>"><?php echo ($EM_Event->event_name); ?></a></td>
104
+ <td><?php echo $EM_Booking->get_spaces() ?></td>
105
+ <td><?php echo $EM_Booking->status_array[$EM_Booking->booking_status]; ?>
106
+ </td>
107
+ <td>
108
+ <?php
109
+ $unapprove_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_unapprove', 'booking_id'=>$EM_Booking->booking_id));
110
+ $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->booking_id));
111
+ $reject_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_reject', 'booking_id'=>$EM_Booking->booking_id));
112
+ $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->booking_id));
113
+ ?>
114
+ <?php if( get_option('dbem_bookings_approval') && ($EM_Booking->booking_status == 0 ) ): ?>
115
+ <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
116
+ <?php endif; ?>
117
+ <?php if( get_option('dbem_bookings_approval') && $EM_Booking->booking_status == 1 ): ?>
118
+ <a class="em-bookings-unapprove" href="<?php echo $unapprove_url ?>"><?php _e('Unapprove','dbem'); ?></a> |
119
+ <?php endif; ?>
120
+ <?php if( $EM_Booking->booking_status == 2 ): ?>
121
+ <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Restore','dbem'); ?></a> |
122
+ <?php endif; ?>
123
+ <?php if( $EM_Booking->booking_status == 0 || $EM_Booking->booking_status == 1 ): ?>
124
+ <a class="em-bookings-reject" href="<?php echo $reject_url ?>"><?php _e('Reject','dbem'); ?></a> |
125
+ <?php endif; ?>
126
+ <a class="em-bookings-edit" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->booking_id; ?>"><?php _e('Edit/View','dbem'); ?></a> |
127
+ <span class="trash"><a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a></span>
128
+ </td>
129
+ </tr>
130
+ <?php
131
+ }
132
+ $event_count++;
133
+ }
134
+ ?>
135
+ </tbody>
136
+ </table>
137
+ </div>
138
+ <?php else: ?>
139
+ <?php _e('No confirmed bookings.', 'dbem'); ?>
140
+ <?php endif; ?>
141
+ </form>
142
+ <?php if( !empty($bookings_nav) && $bookings >= $limit ) : ?>
143
+ <div class='tablenav'>
144
+ <?php echo $bookings_nav; ?>
145
+ <div class="clear"></div>
146
+ </div>
147
+ <?php endif; ?>
148
+ </div>
149
+ <?php
150
+
151
+ }
152
  ?>
admin/bookings/em-rejected.php CHANGED
@@ -1,133 +1,137 @@
1
- <?php
2
- /**
3
- * Generates a "widget" table of confirmed bookings for a specific event.
4
- *
5
- * @param int $event_id
6
- */
7
- function em_bookings_rejected_table(){
8
- global $EM_Event, $wpdb, $current_user;
9
-
10
- $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
11
- $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
12
- $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
13
- $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
14
- $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
15
- $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
16
-
17
- if( is_object($EM_Event) ){
18
- $bookings = $EM_Event->get_bookings()->get_rejected_bookings();
19
- }else{
20
- return false;
21
- }
22
- $bookings_count = (is_array($bookings)) ? count($bookings):0;
23
- ?>
24
- <div class='wrap em_bookings_pending_table em_obj'>
25
- <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
26
- <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
27
- <!--
28
- <ul class="subsubsub">
29
- <li>
30
- <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
31
- </li>
32
- </ul>
33
- <p class="search-box">
34
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search'); ?>:</label>
35
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
36
- <input type="submit" value="<?php _e('Search'); ?>" class="button" />
37
- </p>
38
- -->
39
- <?php if ( $bookings_count >= $limit ) : ?>
40
- <div class='tablenav'>
41
- <!--
42
- <div class="alignleft actions">
43
- <select name="action">
44
- <option value="-1" selected="selected">
45
- <?php _e('Bulk Actions'); ?>
46
- </option>
47
- <option value="approve">
48
- <?php _e('Approve', 'dbem'); ?>
49
- </option>
50
- <option value="decline">
51
- <?php _e('Decline', 'dbem'); ?>
52
- </option>
53
- </select>
54
- <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
55
- </div>
56
- -->
57
- <!--
58
- <div class="view-switch">
59
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
60
- </div>
61
- -->
62
- <?php
63
- if ( $bookings_count >= $limit ) {
64
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%', 'em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
65
- $bookings_nav = em_admin_paginate( $page_link_template, $bookings_count, $limit, $page, 5);
66
- echo $bookings_nav;
67
- }
68
- ?>
69
- <div class="clear"></div>
70
- </div>
71
- <?php endif; ?>
72
- <div class="clear"></div>
73
- <?php if( $bookings_count > 0 ): ?>
74
- <div class='table-wrap'>
75
- <table id='dbem-bookings-table' class='widefat post fixed'>
76
- <thead>
77
- <tr>
78
- <th class='manage-column column-cb check-column' scope='col'>
79
- <input class='select-all' type="checkbox" value='1' />
80
- </th>
81
- <th class='manage-column' scope='col'>Booker</th>
82
- <th class='manage-column' scope='col'>E-mail</th>
83
- <th class='manage-column' scope='col'>Phone number</th>
84
- <th class='manage-column' scope='col'>Spaces</th>
85
- <th class='manage-column' scope='col'>&nbsp;</th>
86
- </tr>
87
- </thead>
88
- <tbody>
89
- <?php
90
- $rowno = 0;
91
- $event_count = 0;
92
- foreach ($bookings as $EM_Booking) {
93
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
94
- $rowno++;
95
- ?>
96
- <tr>
97
- <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->id ?>' name='bookings[]'/></th>
98
- <td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->id; ?>"><?php echo $EM_Booking->person->name ?></a></td>
99
- <td><?php echo $EM_Booking->person->email ?></td>
100
- <td><?php echo $EM_Booking->person->phone ?></td>
101
- <td><?php echo $EM_Booking->seats ?></td>
102
- <td>
103
- <?php
104
- $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->id));
105
- $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->id));
106
- ?>
107
- <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
108
- <a class="em-bookings-edit" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;booking_id=<?php echo $EM_Booking->id; ?>"><?php _e('Edit'); ?></a> |
109
- <a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a>
110
- </td>
111
- </tr>
112
- <?php
113
- }
114
- $event_count++;
115
- }
116
- ?>
117
- </tbody>
118
- </table>
119
- </div>
120
- <?php else: ?>
121
- <?php _e('No rejected bookings.', 'dbem'); ?>
122
- <?php endif; ?>
123
- </form>
124
- <?php if( !empty($bookings_nav) ) : ?>
125
- <div class='tablenav'>
126
- <?php echo $bookings_nav; ?>
127
- <div class="clear"></div>
128
- </div>
129
- <?php endif; ?>
130
- </div>
131
- <?php
132
- }
 
 
 
 
133
  ?>
1
+ <?php
2
+ /**
3
+ * Generates a "widget" table of confirmed bookings for a specific event.
4
+ *
5
+ * @param int $event_id
6
+ */
7
+ function em_bookings_rejected_table(){
8
+ global $EM_Event, $EM_Ticket, $wpdb, $current_user;
9
+
10
+ $action_scope = ( !empty($_REQUEST['em_obj']) && $_REQUEST['em_obj'] == 'em_bookings_confirmed_table' );
11
+ $action = ( $action_scope && !empty($_GET ['action']) ) ? $_GET ['action']:'';
12
+ $order = ( $action_scope && !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
13
+ $limit = ( $action_scope && !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
14
+ $page = ( $action_scope && !empty($_GET['pno']) ) ? $_GET['pno']:1;
15
+ $offset = ( $action_scope && $page > 1 ) ? ($page-1)*$limit : 0;
16
+
17
+ if( is_object($EM_Ticket) ){
18
+ $EM_Bookings = $EM_Ticket->get_bookings()->get_rejected_bookings();
19
+ }else{
20
+ if( is_object($EM_Event) ){
21
+ $EM_Bookings = $EM_Event->get_bookings()->get_rejected_bookings();
22
+ }else{
23
+ return false;
24
+ }
25
+ }
26
+ $bookings_count = (is_array($EM_Bookings->bookings)) ? count($EM_Bookings->bookings):0;
27
+ ?>
28
+ <div class='wrap em_bookings_pending_table em_obj'>
29
+ <form id='bookings-filter' method='get' action='<?php bloginfo('wpurl') ?>/wp-admin/edit.php'>
30
+ <input type="hidden" name="em_obj" value="em_bookings_pending_table" />
31
+ <!--
32
+ <ul class="subsubsub">
33
+ <li>
34
+ <a href='edit.php?post_type=post' class="current">All <span class="count">(1)</span></a> |
35
+ </li>
36
+ </ul>
37
+ <p class="search-box">
38
+ <label class="screen-reader-text" for="post-search-input"><?php _e('Search', 'dbem'); ?>:</label>
39
+ <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
40
+ <input type="submit" value="<?php _e('Search', 'dbem'); ?>" class="button" />
41
+ </p>
42
+ -->
43
+ <?php if ( $bookings_count >= $limit ) : ?>
44
+ <div class='tablenav'>
45
+ <!--
46
+ <div class="alignleft actions">
47
+ <select name="action">
48
+ <option value="-1" selected="selected">
49
+ <?php _e('Bulk Actions', 'dbem'); ?>
50
+ </option>
51
+ <option value="approve">
52
+ <?php _e('Approve', 'dbem'); ?>
53
+ </option>
54
+ <option value="decline">
55
+ <?php _e('Decline', 'dbem'); ?>
56
+ </option>
57
+ </select>
58
+ <input type="submit" id="post-query-submit" value="Filter" class="button-secondary" />
59
+ </div>
60
+ -->
61
+ <!--
62
+ <div class="view-switch">
63
+ <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
64
+ </div>
65
+ -->
66
+ <?php
67
+ if ( $bookings_count >= $limit ) {
68
+ $bookings_nav = em_admin_paginate( $bookings_count, $limit, $page, array('em_ajax'=>0, 'em_obj'=>'em_bookings_confirmed_table'));
69
+ echo $bookings_nav;
70
+ }
71
+ ?>
72
+ <div class="clear"></div>
73
+ </div>
74
+ <?php endif; ?>
75
+ <div class="clear"></div>
76
+ <?php if( $bookings_count > 0 ): ?>
77
+ <div class='table-wrap'>
78
+ <table id='dbem-bookings-table' class='widefat post '>
79
+ <thead>
80
+ <tr>
81
+ <th class='manage-column column-cb check-column' scope='col'>
82
+ <input class='select-all' type="checkbox" value='1' />
83
+ </th>
84
+ <th class='manage-column' scope='col'><?php _e('Booker', 'dbem'); ?></th>
85
+ <th class='manage-column' scope='col'><?php _e('E-mail', 'dbem'); ?></th>
86
+ <th class='manage-column' scope='col'><?php _e('Phone number', 'dbem'); ?></th>
87
+ <th class='manage-column' scope='col'><?php _e('Spaces', 'dbem'); ?></th>
88
+ <th class='manage-column' scope='col'>&nbsp;</th>
89
+ </tr>
90
+ </thead>
91
+ <tbody>
92
+ <?php
93
+ $rowno = 0;
94
+ $event_count = 0;
95
+ foreach ($EM_Bookings->bookings as $EM_Booking) {
96
+ if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
97
+ $rowno++;
98
+ ?>
99
+ <tr>
100
+ <th scope="row" class="check-column" style="padding:7px 0px 7px;"><input type='checkbox' value='<?php echo $EM_Booking->booking_id ?>' name='bookings[]'/></th>
101
+ <td><a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person->ID; ?>"><?php echo $EM_Booking->person->get_name() ?></a></td>
102
+ <td><?php echo $EM_Booking->person->user_email ?></td>
103
+ <td><?php echo $EM_Booking->person->phone ?></td>
104
+ <td><?php echo $EM_Booking->get_spaces() ?></td>
105
+ <td>
106
+ <?php
107
+ $approve_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_approve', 'booking_id'=>$EM_Booking->booking_id));
108
+ $delete_url = em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'bookings_delete', 'booking_id'=>$EM_Booking->booking_id));
109
+ $edit_url = em_add_get_params($_SERVER['REQUEST_URI'], array('booking_id'=>$EM_Booking->booking_id, 'em_ajax'=>null, 'em_obj'=>null));
110
+ ?>
111
+ <a class="em-bookings-approve" href="<?php echo $approve_url ?>"><?php _e('Approve','dbem'); ?></a> |
112
+ <a class="em-bookings-edit" href="<?php echo $edit_url ?>"><?php _e('Edit/View','dbem'); ?></a> |
113
+ <span class="trash"><a class="em-bookings-delete" href="<?php echo $delete_url ?>"><?php _e('Delete','dbem'); ?></a></span>
114
+ </td>
115
+ </tr>
116
+ <?php
117
+ }
118
+ $event_count++;
119
+ }
120
+ ?>
121
+ </tbody>
122
+ </table>
123
+ </div>
124
+ <?php else: ?>
125
+ <?php _e('No rejected bookings.', 'dbem'); ?>
126
+ <?php endif; ?>
127
+ </form>
128
+ <?php if( !empty($bookings_nav) && $EM_Bookings >= $limit ) : ?>
129
+ <div class='tablenav'>
130
+ <?php echo $bookings_nav; ?>
131
+ <div class="clear"></div>
132
+ </div>
133
+ <?php endif; ?>
134
+ </div>
135
+ <?php
136
+ }
137
  ?>
admin/em-admin.php CHANGED
@@ -1,285 +1,217 @@
1
- <?php
2
- //Admin functions
3
-
4
- /**
5
- * Generate warnings and notices in the admin area
6
- */
7
- function em_admin_warnings() {
8
- //If we're editing the events page show hello to new user
9
- $events_page_id = get_option ( 'dbem_events_page' );
10
- $dismiss_link_joiner = ( count($_GET) > 0 ) ? '&amp;':'?';
11
-
12
- if( em_verify_admin() ){
13
- //New User Intro
14
- if (isset ( $_GET ['disable_hello_to_user'] ) && $_GET ['disable_hello_to_user'] == 'true'){
15
- // Disable Hello to new user if requested
16
- update_option ( 'dbem_hello_to_user', 0 );
17
- }elseif ( get_option ( 'dbem_hello_to_user' ) == 1 && !empty($_GET['page']) && $_GET['page'] == 'events-manager-events' ) {
18
- $current_user = wp_get_current_user ();
19
- //FIXME update welcome msg with good links
20
- $advice = sprintf ( __ ( "<p>Hey, <strong>%s</strong>, welcome to <strong>Events Manager</strong>! We hope you like it around here.</p>
21
- <p>Now it's time to insert events lists through <a href='%s' title='Widgets page'>widgets</a>, <a href='%s' title='Template tags documentation'>template tags</a> or <a href='%s' title='Shortcodes documentation'>shortcodes</a>.</p>
22
- <p>By the way, have you taken a look at the <a href='%s' title='Change settings'>Settings page</a>? That's where you customize the way events and locations are displayed.</p>
23
- <p>What? Tired of seeing this advice? I hear you, <a href='%s' title='Don't show this advice again'>click here</a> and you won't see this again!</p>", 'dbem' ), $current_user->display_name, get_bloginfo ( 'url' ) . '/wp-admin/widgets.php', 'http://wp-events-plugin.com/documentation/template-tags/', 'http://wp-events-plugin.com/documentation/shortcodes/', get_bloginfo ( 'url' ) . '/wp-admin/admin.php?page=events-manager-options', get_bloginfo ( 'url' ) . '/wp-admin/admin.php?page=events-manager&disable_hello_to_user=true' );
24
- ?>
25
- <div id="message" class="updated">
26
- <?php echo $advice; ?>
27
- </div>
28
- <?php
29
- }
30
-
31
- //If events page couldn't be created
32
- if( !empty($_GET['em_dismiss_admin_notice']) ){
33
- delete_option('dbem_admin_notice_'.$_GET['em_dismiss_admin_notice']);
34
- }else{
35
- if ( get_option('dbem_admin_notice_3.0.91') ){
36
- ?>
37
- <div class="updated">
38
- <p><?php echo sprintf ( __( '<strong>Events Manager has some new features!</strong><ul><li>Bookings can now be approved before they count towards your event\'s space allocations.</li><li>Events now have owners, and you can restrict users so they can only manage events/locations/categories they create.<br/><br/>These new permissions are enabled by default, but since you upgraded it has been disabled to maintain the previous plugin behaviour. You can re-enable it from the <a href="%s">settings page</a>. <a href="%s">Dismiss</a>', 'dbem'), get_bloginfo ( 'url' ) . '/wp-admin/admin.php?page=events-manager-options', $_SERVER['REQUEST_URI'].$dismiss_link_joiner.'em_dismiss_admin_notice=3.0.91' ); ?></p>
39
- </div>
40
- <?php
41
- }
42
- }
43
- //If events page couldn't be created
44
- if( !empty($_GET['em_dismiss_rc_notice']) ){
45
- delete_option('dbem_admin_notice_rc');
46
- }elseif( get_option('dbem_admin_notice_rc')){
47
- ?>
48
- <div class="updated">
49
- <p>
50
- <strong>Events Manager 4.0 on its way!</strong>
51
- We've just about finished with 4.0, which boasts TONS of long awaited features including advanced booking, permalinks, MultiSite and BuddPress integration, and much more!
52
- It is undergoing final testing to be released in the coming days, and the website has already had a documentation and tutorial overhaul in preparation.
53
- </p><p>
54
- To read up on the changes and get a hold of the Release Candidate
55
- <a href="http://wp-events-plugin.com/try-the-beta/?utm_source=plugin&utm_medium=html&utm_campaign=rc_notice">click here</a>.
56
- Please make sure to have a look at this page if you use EM for bookings or have made any mods to the plugin, as there are some considerations to make when upgrading.<br />
57
- <a href="http://wp-events-plugin.com/try-the-beta/?utm_source=plugin&utm_medium=html&utm_campaign=rc_notice">See what's new</a> | <a href="<?php echo $_SERVER['REQUEST_URI'].$dismiss_link_joiner.'em_dismiss_rc_notice=1' ?>">Dismiss this message.</a>
58
- </p>
59
- </div>
60
- <?php
61
- }
62
-
63
- //If events page couldn't be created
64
- if( !empty($_GET['em_dismiss_events_page']) ){
65
- update_option('dbem_dismiss_events_page',1);
66
- }else{
67
- if ( !get_page($events_page_id) && !get_option('dbem_dismiss_events_page') ){
68
- ?>
69
- <div id="em_page_error" class="updated">
70
- <p><?php echo sprintf ( __( 'Uh Oh! For some reason wordpress could not create an events page for you (or you just deleted it). Not to worry though, all you have to do is create an empty page, name it whatever you want, and select it as your events page in your <a href="%s">settings page</a>. Sorry for the extra step! If you know what you are doing, you may have done this on purpose, if so <a href="%s">ignore this message</a>', 'dbem'), get_bloginfo ( 'url' ) . '/wp-admin/admin.php?page=events-manager-options', $_SERVER['REQUEST_URI'].$dismiss_link_joiner.'em_dismiss_events_page=1' ); ?></p>
71
- </div>
72
- <?php
73
- }
74
- }
75
- }
76
- //Warn about EM page edit
77
- if ( preg_match( '/(post|page).php/', $_SERVER ['SCRIPT_NAME']) && isset ( $_GET ['action'] ) && $_GET ['action'] == 'edit' && isset ( $_GET ['post'] ) && $_GET ['post'] == "$events_page_id") {
78
- $message = sprintf ( __ ( "This page corresponds to <strong>Events Manager</strong> events page. Its content will be overriden by <strong>Events Manager</strong>. If you want to display your content, you can can assign another page to <strong>Events Manager</strong> in the the <a href='%s'>Settings</a>. ", 'dbem' ), 'admin.php?page=events-manager-options' );
79
- $notice = "<div class='error'><p>$message</p></div>";
80
- echo $notice;
81
- }
82
-
83
- }
84
- add_action ( 'admin_notices', 'em_admin_warnings' );
85
-
86
- /**
87
- * Creates a wp-admin style navigation. All this does is wrap some html around the em_paginate function result to make it style correctly in the admin area.
88
- * @param string $link
89
- * @param int $total
90
- * @param int $limit
91
- * @param int $page
92
- * @param int $pagesToShow
93
- * @return string
94
- * @uses em_paginate()
95
- */
96
- function em_admin_paginate($link, $total, $limit, $page=1, $pagesToShow=5){
97
- $return = '<div class="tablenav-pages">';
98
- $return .= sprintf( '<span class="displaying-num">' . __( 'Displaying %s&#8211;%s of %s' ) . '</span>',
99
- number_format_i18n( ( $page - 1 ) * $limit + 1 ),
100
- number_format_i18n( min( $page * $limit, $total ) ),
101
- number_format_i18n( $total )
102
- );
103
- $return .= em_paginate($link, $total, $limit, $page, $pagesToShow);
104
- $return .= '</div>';
105
- return $return;
106
- }
107
-
108
- /**
109
- * Called by admin_print_scripts-(hook|page) action, created when adding menu items in events-manager.php
110
- */
111
- function em_admin_load_scripts(){
112
- //Load the UI items, currently date picker and autocomplete plus dependencies
113
- //wp_enqueue_script('em-ui-js', WP_PLUGIN_URL.'/events-manager/includes/js/jquery-ui-1.8.5.custom.min.js', array('jquery', 'jquery-ui-core'));
114
- wp_enqueue_script('em-ui-js', WP_PLUGIN_URL.'/events-manager/includes/js/em_ui.js', array('jquery', 'jquery-ui-core'));
115
-
116
- //Add maps
117
- if( get_option('dbem_gmap_is_active') ){
118
- wp_enqueue_script('em-google-maps', 'http://maps.google.com/maps/api/js?sensor=false');
119
- }
120
- //Time Entry
121
- wp_enqueue_script('em-timeentry', WP_PLUGIN_URL.'/events-manager/includes/js/timeentry/jquery.timeentry.js', array('jquery'));
122
-
123
- //Date Picker Locale
124
- $locale_code = substr ( get_locale (), 0, 2 );
125
- $locale_file = "/events-manager/includes/js/i18n/jquery.ui.datepicker-$locale_code.js";
126
- if ( file_exists(WP_PLUGIN_DIR.$locale_file) ) {
127
- wp_enqueue_script("em-ui-datepicker-$locale_code", WP_PLUGIN_URL.$locale_file, array('em-ui-js'));
128
- }
129
- wp_enqueue_script('em-script', WP_PLUGIN_URL.'/events-manager/includes/js/em_admin.js', array('em-ui-js'));
130
-
131
- //TinyMCE Editor
132
- remove_filter('the_editor', 'qtrans_modifyRichEditor'); //qtranslate filter
133
- add_action( 'admin_print_footer_scripts', 'wp_tiny_mce', 25 );
134
- add_action( 'admin_print_footer_scripts', 'wp_tiny_mce_preload_dialogs', 30 );
135
- wp_enqueue_script('post');
136
- if ( user_can_richedit() )
137
- wp_enqueue_script('editor');
138
-
139
- add_thickbox();
140
- wp_enqueue_script('media-upload');
141
- wp_enqueue_script('word-count');
142
- wp_enqueue_script('quicktags');
143
- }
144
-
145
- /**
146
- * Called by admin_print_styles-(hook|page) action, created when adding menu items in events-manager.php
147
- */
148
- function em_admin_load_styles() {
149
- add_thickbox();
150
- wp_enqueue_style('em-ui-css', WP_PLUGIN_URL.'/events-manager/includes/css/jquery-ui-1.7.3.custom.css');
151
- wp_enqueue_style('events-manager-admin', WP_PLUGIN_URL.'/events-manager/includes/css/events_manager_admin.css');
152
- }
153
-
154
- /**
155
- * Loads script inline due to insertion of php values
156
- */
157
- function em_admin_general_script() {
158
- //TODO clean script up, remove dependency of php so it can be moved to js file.
159
- // Check if the locale is there and loads it
160
- $locale_code = substr ( get_locale (), 0, 2 );
161
- $show24Hours = 'true';
162
- // Setting 12 hours format for those countries using it
163
- if (preg_match ( "/en|sk|zh|us|uk/", $locale_code ))
164
- $show24Hours = 'false';
165
- ?>
166
- <script type="text/javascript">
167
- //<![CDATA[
168
- // TODO: make more general, to support also latitude and longitude (when added)
169
-
170
- jQuery(document).ready( function($) {
171
-
172
- function updateIntervalDescriptor () {
173
- $(".interval-desc").hide();
174
- var number = "-plural";
175
- if ($('input#recurrence-interval').val() == 1 || $('input#recurrence-interval').val() == "")
176
- number = "-singular"
177
- var descriptor = "span#interval-"+$("select#recurrence-frequency").val()+number;
178
- $(descriptor).show();
179
- }
180
- function updateIntervalSelectors () {
181
- $('p.alternate-selector').hide();
182
- $('p#'+ $('select#recurrence-frequency').val() + "-selector").show();
183
- }
184
- function updateShowHideRsvp () {
185
- if($('input#rsvp-checkbox').attr("checked")) {
186
- $("div#rsvp-data").fadeIn();
187
- } else {
188
- $("div#rsvp-data").hide();
189
- }
190
- }
191
- function updateShowHideRecurrence () {
192
- if( $('input#event-recurrence').attr("checked")) {
193
- $("#event_recurrence_pattern").fadeIn();
194
- $("#event-date-explanation").hide();
195
- $("#recurrence-dates-explanation").show();
196
- $("h3#recurrence-dates-title").show();
197
- $("h3#event-date-title").hide();
198
- } else {
199
- $("#event_recurrence_pattern").hide();
200
- $("#recurrence-dates-explanation").hide();
201
- $("#event-date-explanation").show();
202
- $("h3#recurrence-dates-title").hide();
203
- $("h3#event-date-title").show();
204
- }
205
- }
206
- $("#recurrence-dates-explanation").hide();
207
- $("#localised-date").show();
208
- $("#localised-end-date").show();
209
-
210
- $("#date-to-submit").hide();
211
- $("#end-date-to-submit").hide();
212
-
213
- $("#start-time").timeEntry({spinnerImage: '', show24Hours: <?php echo $show24Hours; ?> });
214
- $("#end-time").timeEntry({spinnerImage: '', show24Hours: <?php echo $show24Hours; ?>});
215
-
216
- $('input.select-all').change(function(){
217
- if($(this).is(':checked'))
218
- $('input.row-selector').attr('checked', true);
219
- else
220
- $('input.row-selector').attr('checked', false);
221
- });
222
-
223
- updateIntervalDescriptor();
224
- updateIntervalSelectors();
225
- updateShowHideRecurrence();
226
- updateShowHideRsvp();
227
- $('input#event-recurrence').change(updateShowHideRecurrence);
228
- $('input#rsvp-checkbox').change(updateShowHideRsvp);
229
-
230
- // recurrency elements
231
- $('input#recurrence-interval').keyup(updateIntervalDescriptor);
232
- $('select#recurrence-frequency').change(updateIntervalDescriptor);
233
- $('select#recurrence-frequency').change(updateIntervalSelectors);
234
-
235
- // hiding or showing notes according to their content
236
- $('.postbox h3').prepend('<a class="togbox">+</a> ');
237
- $('#event_notes h3').click( function() {
238
- $(this).parent().first().toggleClass('closed');
239
- });
240
-
241
- // users cannot submit the event form unless some fields are filled
242
- function validateEventForm(){
243
- errors = "";
244
- var recurring = $("input[@name=repeated_event]:checked").val();
245
- requiredFields= new Array('event_name', 'localised_event_date', 'location_name','location_address','location_town');
246
- var localisedRequiredFields = {
247
- 'event_name':"<?php _e ( 'Name', 'dbem' )?>",
248
- 'localised_event_date':"<?php _e ( 'Date', 'dbem' )?>",
249
- 'location_name':"<?php _e ( 'Location', 'dbem' )?>",
250
- 'location_address':"<?php _e ( 'Address', 'dbem' )?>",
251
- 'location_town':"<?php _e ( 'Town', 'dbem' )?>"
252
- };
253
- missingFields = new Array;
254
- for (var i in requiredFields) {
255
- if ($("input[@name=" + requiredFields[i]+ "]").val() == 0) {
256
- missingFields.push(localisedRequiredFields[requiredFields[i]]);
257
- $("input[@name=" + requiredFields[i]+ "]").css('border','2px solid red');
258
- } else {
259
- $("input[@name=" + requiredFields[i]+ "]").css('border','1px solid #DFDFDF');
260
- }
261
- }
262
-
263
- // alert('ciao ' + recurring+ " end: " + $("input[@name=localised_event_end_date]").val());
264
- if (missingFields.length > 0) {
265
- errors = "<?php _e ( 'Some required fields are missing:', 'dbem' )?> " + missingFields.join(", ") + ".\n";
266
- }
267
- if(recurring && $("input[@name=localised_event_end_date]").val() == "") {
268
- errors = errors + "<?php _e ( 'Since the event is repeated, you must specify an end date', 'dbem' )?>.";
269
- $("input[@name=localised_event_end_date]").css('border','2px solid red');
270
- } else {
271
- $("input[@name=localised_event_end_date]").css('border','1px solid #DFDFDF');
272
- }
273
- if(errors != "") {
274
- alert(errors);
275
- return false;
276
- }
277
- return true;
278
- }
279
- $('#event-form').bind("submit", validateEventForm);
280
- });
281
- //]]>
282
- </script>
283
- <?php
284
- }
285
  ?>
1
+ <?php
2
+ //Admin functions
3
+ function em_admin_menu(){
4
+ global $menu, $submenu, $pagenow;
5
+ //Count pending bookings
6
+ if( get_option('dbem_rsvp_enabled') ){
7
+ $bookings_num = '';
8
+ $bookings_pending_count = apply_filters('em_bookings_pending_count',0);
9
+ if( get_option('dbem_bookings_approval') == 1){
10
+ $bookings_pending_count += count(EM_Bookings::get(array('status'=>'0', 'blog'=>get_current_blog_id()))->bookings);
11
+ }
12
+ if($bookings_pending_count > 0){
13
+ $bookings_num = '<span class="update-plugins count-'.$bookings_pending_count.'"><span class="plugin-count">'.$bookings_pending_count.'</span></span>';
14
+ }
15
+ }else{
16
+ $bookings_num = '';
17
+ $bookings_pending_count = 0;
18
+ }
19
+ //Count pending events
20
+ $events_num = '';
21
+ $events_pending_count = EM_Events::count(array('status'=>0, 'scope'=>'all', 'blog'=>get_current_blog_id()));
22
+ //TODO Add flexible permissions
23
+ if($events_pending_count > 0){
24
+ $events_num = '<span class="update-plugins count-'.$events_pending_count.'"><span class="plugin-count">'.$events_pending_count.'</span></span>';
25
+ }
26
+ //Count pending recurring events
27
+ $events_recurring_num = '';
28
+ $events_recurring_pending_count = EM_Events::count(array('status'=>0, 'recurring'=>1, 'scope'=>'all', 'blog'=>get_current_blog_id()));
29
+ //TODO Add flexible permissions
30
+ if($events_recurring_pending_count > 0){
31
+ $events_recurring_num = '<span class="update-plugins count-'.$events_recurring_pending_count.'"><span class="plugin-count">'.$events_recurring_pending_count.'</span></span>';
32
+ }
33
+ $both_pending_count = apply_filters('em_items_pending_count', $events_pending_count + $bookings_pending_count + $events_recurring_pending_count);
34
+ $both_num = ($both_pending_count > 0) ? '<span class="update-plugins count-'.$both_pending_count.'"><span class="plugin-count">'.$both_pending_count.'</span></span>':'';
35
+ // Add a submenu to the custom top-level menu:
36
+ $plugin_pages = array();
37
+ if( get_option('dbem_rsvp_enabled') ){
38
+ $plugin_pages['bookings'] = add_submenu_page('edit.php?post_type='.EM_POST_TYPE_EVENT, __('Bookings', 'dbem'), __('Bookings', 'dbem').$bookings_num, 'manage_bookings', 'events-manager-bookings', "em_bookings_page");
39
+ }
40
+ $plugin_pages['options'] = add_submenu_page('edit.php?post_type='.EM_POST_TYPE_EVENT, __('Events Manager Settings','dbem'),__('Settings','dbem'), 'activate_plugins', "events-manager-options", 'em_admin_options_page');
41
+ $plugin_pages['help'] = add_submenu_page('edit.php?post_type='.EM_POST_TYPE_EVENT, __('Getting Help for Events Manager','dbem'),__('Help','dbem'), 'activate_plugins', "events-manager-help", 'em_admin_help_page');
42
+ //If multisite global with locations set to be saved in main blogs we can force locations to be created on the main blog only
43
+ if( EM_MS_GLOBAL && !is_main_site() && get_site_option('dbem_ms_mainblog_locations') ){
44
+ include( dirname(__FILE__)."/em-ms-locations.php" );
45
+ $plugin_pages['locations'] = add_submenu_page('edit.php?post_type='.EM_POST_TYPE_EVENT, __('Locations','dbem'),__('Locations','dbem'), 'read_others_locations', "locations", 'em_admin_ms_locations');
46
+ }
47
+ $plugin_pages = apply_filters('em_create_events_submenu',$plugin_pages);
48
+ //We have to modify the menus manually
49
+ if( !empty($both_num) ){ //Main Event Menu
50
+ //go through the menu array and modify the events menu if found
51
+ foreach ( (array)$menu as $key => $parent_menu ) {
52
+ if ( $parent_menu[2] == 'edit.php?post_type='.EM_POST_TYPE_EVENT ){
53
+ $menu[$key][0] = $menu[$key][0]. $both_num;
54
+ break;
55
+ }
56
+ }
57
+ }
58
+ if( !empty($events_num) && !empty($submenu['edit.php?post_type='.EM_POST_TYPE_EVENT]) ){ //Submenu Event Item
59
+ //go through the menu array and modify the events menu if found
60
+ foreach ( (array)$submenu['edit.php?post_type='.EM_POST_TYPE_EVENT] as $key => $submenu_item ) {
61
+ if ( $submenu_item[2] == 'edit.php?post_type='.EM_POST_TYPE_EVENT ){
62
+ $submenu['edit.php?post_type='.EM_POST_TYPE_EVENT][$key][0] = $submenu['edit.php?post_type='.EM_POST_TYPE_EVENT][$key][0]. $events_num;
63
+ break;
64
+ }
65
+ }
66
+ }
67
+ if( !empty($events_recurring_num) && !empty($submenu['edit.php?post_type='.EM_POST_TYPE_EVENT]) ){ //Submenu Recurring Event Item
68
+ //go through the menu array and modify the events menu if found
69
+ foreach ( (array)$submenu['edit.php?post_type='.EM_POST_TYPE_EVENT] as $key => $submenu_item ) {
70
+ if ( $submenu_item[2] == 'edit.php?post_type=event-recurring' ){
71
+ $submenu['edit.php?post_type='.EM_POST_TYPE_EVENT][$key][0] = $submenu['edit.php?post_type='.EM_POST_TYPE_EVENT][$key][0]. $events_recurring_num;
72
+ break;
73
+ }
74
+ }
75
+ }
76
+ /* Hack! Add location/recurrence isn't possible atm so this is a workaround */
77
+ global $_wp_submenu_nopriv;
78
+ if( $pagenow == 'post-new.php' && !empty($_REQUEST['post_type']) ){
79
+ if( $_REQUEST['post_type'] == EM_POST_TYPE_LOCATION && !empty($_wp_submenu_nopriv['edit.php']['post-new.php']) && current_user_can('edit_locations') ){
80
+ unset($_wp_submenu_nopriv['edit.php']['post-new.php']);
81
+ }
82
+ if( $_REQUEST['post_type'] == 'event-recurring' && !empty($_wp_submenu_nopriv['edit.php']['post-new.php']) && current_user_can('edit_recurring_events') ){
83
+ unset($_wp_submenu_nopriv['edit.php']['post-new.php']);
84
+ }
85
+ }
86
+ }
87
+ add_action('admin_menu','em_admin_menu');
88
+
89
+ function em_ms_admin_menu(){
90
+ add_menu_page( __('Events Manager','dbem'), __('Events Manager','dbem'), 'activate_plugins', 'events-manager-options', 'em_ms_admin_options_page', plugins_url('includes/images/calendar-16.png', dirname(dirname(__FILE__)).'/events-manager.php') );
91
+ add_submenu_page('events-manager-options', __('Update Blogs','dbem'),__('Update Blogs','dbem'), 'activate_plugins', "events-manager-update", 'em_ms_upgrade');
92
+ }
93
+ add_action('network_admin_menu','em_ms_admin_menu');
94
+
95
+ function em_admin_init(){
96
+ //in MS global mode
97
+ if( EM_MS_GLOBAL && is_user_logged_in() && !is_main_site() && get_site_option('dbem_ms_mainblog_locations') ){
98
+ EM_Object::ms_global_switch();
99
+ $user = new WP_User(get_current_user_id());
100
+ if( count($user->roles) == 0 ){
101
+ $user->set_role('subscriber');
102
+ }
103
+ EM_Object::ms_global_switch_back();
104
+ }
105
+ }
106
+ add_action('admin_init','em_admin_init');
107
+
108
+ /**
109
+ * Generate warnings and notices in the admin area
110
+ */
111
+ function em_admin_warnings() {
112
+ global $EM_Notices;
113
+ //If we're editing the events page show hello to new user
114
+ $events_page_id = get_option ( 'dbem_events_page' );
115
+ $dismiss_link_joiner = ( count($_GET) > 0 ) ? '&amp;':'?';
116
+
117
+ if( current_user_can('activate_plugins') ){
118
+ //New User Intro
119
+ if (isset ( $_GET ['disable_hello_to_user'] ) && $_GET ['disable_hello_to_user'] == 'true'){
120
+ // Disable Hello to new user if requested
121
+ update_option('dbem_hello_to_user',0);
122
+ }elseif ( get_option ( 'dbem_hello_to_user' ) ) {
123
+ //FIXME update welcome msg with good links
124
+ $advice = sprintf( __("<p>Events Manager is ready to go! It is highly recommended you read the <a href='%s'>Getting Started</a> guide on our site, as well as checking out the <a href='%s'>Settings Page</a>. <a href='%s' title='Don't show this advice again'>Dismiss</a></p>", 'dbem'), 'http://wp-events-plugin.com/documentation/getting-started/?utm_source=em&utm_medium=plugin&utm_content=installationlink&utm_campaign=plugin_links', EM_ADMIN_URL .'&amp;page=events-manager-options', $_SERVER['REQUEST_URI'].$dismiss_link_joiner.'disable_hello_to_user=true');
125
+ ?>
126
+ <div id="message" class="updated">
127
+ <?php echo $advice; ?>
128
+ </div>
129
+ <?php
130
+ }
131
+
132
+ //If events page couldn't be created or is missing
133
+ if( !empty($_GET['em_dismiss_events_page']) ){
134
+ update_option('dbem_dismiss_events_page',1);
135
+ }else{
136
+ if ( !get_page($events_page_id) && !get_option('dbem_dismiss_events_page') ){
137
+ ?>
138
+ <div id="em_page_error" class="updated">
139
+ <p><?php echo sprintf ( __( 'Uh Oh! For some reason WordPress could not create an events page for you (or you just deleted it). Not to worry though, all you have to do is create an empty page, name it whatever you want, and select it as your events page in your <a href="%s">settings page</a>. Sorry for the extra step! If you know what you are doing, you may have done this on purpose, if so <a href="%s">ignore this message</a>', 'dbem'), EM_ADMIN_URL .'&amp;page=events-manager-options', $_SERVER['REQUEST_URI'].$dismiss_link_joiner.'em_dismiss_events_page=1' ); ?></p>
140
+ </div>
141
+ <?php
142
+ }
143
+ }
144
+
145
+ if( defined('EMP_VERSION') && EMP_VERSION < EM_PRO_MIN_VERSION ){
146
+ ?>
147
+ <div id="em_page_error" class="updated">
148
+ <p><?php _e('There is a newer version of Events Manager Pro which is required for this current version of Events Manager. Please go to the plugin website and download the latest update.','dbem'); ?></p>
149
+ </div>
150
+ <?php
151
+ }
152
+
153
+ if( is_multisite() && !empty($_REQUEST['page']) && $_REQUEST['page']=='events-manager-options' && is_super_admin() && get_option('dbem_ms_update_nag') ){
154
+ if( !empty($_GET['disable_dbem_ms_update_nag']) ){
155
+ delete_site_option('dbem_ms_update_nag');
156
+ }else{
157
+ ?>
158
+ <div id="em_page_error" class="updated">
159
+ <p><?php echo sprintf(__('MultiSite options have moved <a href="%s">here</a>. <a href="%s">Dismiss message</a>','dbem'),admin_url().'network/admin.php?page=events-manager-options', $_SERVER['REQUEST_URI'].'&amp;disable_dbem_ms_update_nag=1'); ?></p>
160
+ </div>
161
+ <?php
162
+ }
163
+ }
164
+
165
+ if( is_super_admin() && get_option('dbem_migrate_images_nag') ){
166
+ if( !empty($_GET['disable_dbem_migrate_images_nag']) ){
167
+ delete_site_option('dbem_migrate_images_nag');
168
+ }else{
169
+ ?>
170
+ <div id="em_page_error" class="updated">
171
+ <p><?php echo sprintf(__('Whilst they will still appear using placeholders, you need to <a href="%s">migrate your location and event images</a> in order for them to appear in your edit forms and media library. <a href="%s">Dismiss message</a>','dbem'),admin_url().'edit.php?post_type=event&page=events-manager-options&em_migrate_images=1&_wpnonce='.wp_create_nonce('em_migrate_images'), em_add_get_params($_SERVER['REQUEST_URI'], array('disable_dbem_migrate_images_nag' => 1))); ?></p>
172
+ </div>
173
+ <?php
174
+ }
175
+ }
176
+ }
177
+ //Warn about EM page edit
178
+ if ( preg_match( '/(post|page).php/', $_SERVER ['SCRIPT_NAME']) && isset ( $_GET ['action'] ) && $_GET ['action'] == 'edit' && isset ( $_GET ['post'] ) && $_GET ['post'] == "$events_page_id") {
179
+ $message = sprintf ( __ ( "This page corresponds to <strong>Events Manager</strong> events page. Its content will be overriden by Events Manager, although if you include the word CONTENTS (exactly in capitals) and surround it with other text, only CONTENTS will be overwritten. If you want to change the way your events look, go to the <a href='%s'>settings</a> page. ", 'dbem' ), EM_ADMIN_URL .'&amp;page=events-manager-options' );
180
+ $notice = "<div class='error'><p>$message</p></div>";
181
+ echo $notice;
182
+ }
183
+ echo $EM_Notices;
184
+ }
185
+ add_action ( 'admin_notices', 'em_admin_warnings', 100 );
186
+
187
+ /**
188
+ * Settings link in the plugins page menu
189
+ * @param array $links
190
+ * @param string $file
191
+ * @return array
192
+ */
193
+ function em_set_plugin_meta($links, $file) {
194
+ $plugin = plugin_basename(__FILE__);
195
+ // create link
196
+ if ($file == $plugin) {
197
+ return array_merge(
198
+ $links,
199
+ array( sprintf( '<a href="'.EM_ADMIN_URL.'&amp;page=events-manager-options">%s</a>', __('Settings', 'dbem') ) )
200
+ );
201
+ }
202
+ return $links;
203
+ }
204
+ //add_filter( 'plugin_row_meta', 'em_set_plugin_meta', 10, 2 );
205
+
206
+ function em_plugin_action_links($actions, $file, $plugin_data) {
207
+ $actions['settings'] = sprintf( '<a href="'.EM_ADMIN_URL.'&amp;page=events-manager-options">%s</a>', __('Settings', 'dbem') );
208
+ if( is_multisite() ){
209
+ $uninstall_url = admin_url().'network/admin.php?page=events-manager-options&amp;action=uninstall&amp;_wpnonce='.wp_create_nonce('em_uninstall_'.get_current_user_id().'_wpnonce');
210
+ }else{
211
+ $uninstall_url = EM_ADMIN_URL.'&amp;page=events-manager-options&amp;action=uninstall&amp;_wpnonce='.wp_create_nonce('em_uninstall_'.get_current_user_id().'_wpnonce');
212
+ }
213
+ $actions['uninstall'] = '<span class="delete"><a href="'.$uninstall_url.'" class="delete">'.__('Uninstall','dbem').'</a></span>';
214
+ return $actions;
215
+ }
216
+ add_filter( 'plugin_action_links_events-manager/events-manager.php', 'em_plugin_action_links', 10, 3 );
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
217
  ?>
admin/em-bookings.php CHANGED
@@ -1,288 +1,422 @@
1
- <?php
2
- /**
3
- * Check if there's any admin-related actions to take for bookings. All actions are caught here.
4
- * @return null
5
- */
6
- function em_admin_actions_bookings() {
7
- global $dbem_form_add_message;
8
- global $dbem_form_delete_message;
9
- global $wpdb, $EM_Booking, $EM_Event;
10
-
11
- if( current_user_can(EM_MIN_CAPABILITY) && is_object($EM_Booking) && !empty($_REQUEST['action']) ) {
12
- if( $_REQUEST['action'] == 'bookings_delete' ){
13
- //Delete
14
- if( isset($_POST['booking_id']) ){
15
- $EM_Booking = new EM_Booking($_POST['booking_id']);
16
- $EM_Booking->delete();
17
- }
18
- }elseif( $_REQUEST['action'] == 'bookings_edit' ){
19
- //Edit Booking
20
- $validation = $EM_Booking->get_post();
21
- if ( $validation ) { //EM_Event gets the event if submitted via POST and validates it (safer than to depend on JS)
22
- //Save
23
- if( $EM_Booking->save() ) {
24
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="updated"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
25
- }else{
26
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="error"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
27
- }
28
- }else{
29
- //TODO make errors clearer when saving person
30
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="error"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
31
- }
32
- add_action ( 'admin_notices', 'em_booking_save_notification' );
33
- }elseif( $_REQUEST['action'] == 'bookings_approve' || $_REQUEST['action'] == 'bookings_reject' || $_REQUEST['action'] == 'bookings_unapprove' ){
34
- //Booking Approvals
35
- $status_array = array('bookings_unapprove' => 0,'bookings_approve' => 1,'bookings_reject' => 2, 'bookings_cancel' => 3);
36
- if( $EM_Booking->set_status( $status_array[$_REQUEST['action']] ) ) {
37
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="updated"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
38
- }else{
39
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="error"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
40
- }
41
- add_action ( 'admin_notices', 'em_booking_save_notification' );
42
- }elseif( $_REQUEST['action'] == 'bookings_add_note' ){
43
- $EM_Booking->add_note($_REQUEST['booking_note']);
44
- function em_booking_save_notification(){ global $EM_Booking; ?><div class="updated"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
45
- add_action ( 'admin_notices', 'em_booking_save_notification' );
46
- }
47
- }elseif( current_user_can(EM_MIN_CAPABILITY) && is_object($EM_Event) && !empty($_REQUEST['action']) ){
48
- if( $_REQUEST['action'] == 'export_csv' ){
49
- $EM_Event->get_bookings()->export_csv();
50
- exit();
51
- }
52
- }
53
- }
54
- add_action('admin_init','em_admin_actions_bookings',100);
55
-
56
- /**
57
- * Decide what content to show in the bookings section.
58
- */
59
- function em_bookings_page(){
60
- //First any actions take priority
61
- if( !empty($_REQUEST['booking_id']) ){
62
- em_bookings_single();
63
- }elseif( !empty($_REQUEST['person_id']) ){
64
- em_bookings_person();
65
- }elseif( !empty($_REQUEST['event_id']) ){
66
- em_bookings_event();
67
- }else{
68
- em_bookings_dashboard();
69
- }
70
- }
71
-
72
- /**
73
- * Generates the bookings dashboard, showing information on all events
74
- */
75
- function em_bookings_dashboard(){
76
- ?>
77
- <div class='wrap'>
78
- <div id='icon-users' class='icon32'>
79
- <br/>
80
- </div>
81
- <h2>
82
- <?php _e('Event Bookings Dashboard', 'dbem'); ?>
83
- </h2>
84
- <?php if( get_option('dbem_bookings_approval') ): ?>
85
- <h2><?php _e('Pending Bookings','dbem'); ?></h2>
86
- <?php em_bookings_pending_table(); ?>
87
- <?php endif; ?>
88
- <h2><?php _e('Events With Bookings Enabled','dbem'); ?></h2>
89
- <?php em_bookings_events_table(); ?>
90
- </div>
91
- <?php
92
- }
93
-
94
- /**
95
- * Shows all booking data for a single event
96
- */
97
- function em_bookings_event(){
98
- global $EM_Event,$EM_Person;
99
- //check that user can access this page
100
- if( is_object($EM_Event) && !$EM_Event->can_manage() ){
101
- ?>
102
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
103
- <?php
104
- return false;
105
- }
106
- $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
107
- $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
108
- ?>
109
- <div class='wrap'>
110
- <div id='icon-users' class='icon32'>
111
- <br/>
112
- </div>
113
- <h2>
114
- <?php echo sprintf(__('Manage %s Bookings', 'dbem'), "'{$EM_Event->name}'"); ?>
115
- <a href="admin.php?page=events-manager-event&event_id=<?php echo $EM_Event->id; ?>" class="button add-new-h2"><?php _e('View/Edit Event','dbem') ?></a>
116
- </h2>
117
- <div><a href='<?php echo get_bloginfo('wpurl') . "/wp-admin/admin.php?page=events-manager-bookings&action=export_csv&event_id=".$EM_Event->id ?>'><?php _e('export csv','dbem')?></a></div>
118
- <div>
119
- <p><strong><?php _e('Event Name','dbem'); ?></strong> : <?php echo ($EM_Event->name); ?></p>
120
- <p><strong>Availability :</strong> <?php echo $EM_Event->get_bookings()->get_booked_seats() . '/'. $EM_Event->seats ." ". __('Seats confirmed','dbem'); ?></p>
121
- <p>
122
- <strong><?php _e('Date','dbem'); ?></strong> :
123
- <?php echo $localised_start_date; ?>
124
- <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
125
- <?php echo substr ( $EM_Event->start_time, 0, 5 ) . " - " . substr ( $EM_Event->end_time, 0, 5 ); ?>
126
- </p>
127
- <p>
128
- <strong><?php _e('Location','dbem'); ?></strong> :
129
- <a class="row-title" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-locations&amp;location_id=<?php echo $EM_Event->location->id ?>"><?php echo ($EM_Event->location->name); ?></a>
130
- </p>
131
- </div>
132
- <?php if( get_option('dbem_bookings_approval') ): ?>
133
- <h2><?php _e('Pending Bookings','dbem'); ?></h2>
134
- <?php em_bookings_pending_table(); ?>
135
- <?php endif; ?>
136
- <h2><?php _e('Confirmed Bookings','dbem'); ?></h2>
137
- <?php em_bookings_confirmed_table(); ?>
138
- <h2><?php _e('Rejected Bookings','dbem'); ?></h2>
139
- <?php em_bookings_rejected_table(); ?>
140
- <h2><?php _e('Cancelled Bookings','dbem'); ?></h2>
141
- <?php em_bookings_cancelled_table(); ?>
142
- </div>
143
- <?php
144
- }
145
-
146
- /**
147
- * Shows a single booking for a single person.
148
- */
149
- function em_bookings_single(){
150
- global $EM_Booking;
151
- //check that user can access this page
152
- if( is_object($EM_Booking) && !$EM_Booking->can_manage() ){
153
- ?>
154
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
155
- <?php
156
- return false;
157
- }
158
- ?>
159
- <div class='wrap'>
160
- <div id='icon-users' class='icon32'>
161
- <br/>
162
- </div>
163
- <h2>
164
- <?php _e('Edit Booking', 'dbem'); ?>
165
- </h2>
166
- <div id="poststuff" class="metabox-holder has-right-sidebar">
167
- <div id="post-body">
168
- <div id="post-body-content">
169
- <div id="event_name" class="stuffbox">
170
- <h3>
171
- <?php _e ( 'Booking Details', 'dbem' ); ?>
172
- </h3>
173
- <div class="inside">
174
- <?php em_bookings_edit_form(); ?>
175
- </div>
176
- </div>
177
- </div>
178
- </div>
179
- </div>
180
- <br style="clear:both;" />
181
- <div id="poststuff" class="metabox-holder has-right-sidebar">
182
- <div id="post-body">
183
- <div id="post-body-content">
184
- <div id="event_name" class="stuffbox">
185
- <h3>
186
- <?php _e ( 'Booking Notes', 'dbem' ); ?>
187
- </h3>
188
- <div class="inside">
189
- <p><?php _e('You can add private notes below for internal reference that only event managers will see.','dbem'); ?></p>
190
- <?php foreach( $EM_Booking->notes as $note ):
191
- $user = get_userdata($note['author']);
192
- ?>
193
- <div>
194
- <?php echo date(get_option('date_format'), $note['timestamp']) .' - '. $user->display_name; ?> <?php _e('wrote','dbem'); ?>:
195
- <p style="background:#efefef; padding:5px;"><?php echo nl2br($note['note']); ?></p>
196
- </div>
197
- <?php endforeach; ?>
198
- <form method="post" action="" style="padding:5px;">
199
- <textarea class="widefat" rows="5" name="booking_note"></textarea>
200
- <input type="hidden" name="action" value="bookings_add_note" />
201
- <input type="submit" value="Add Note" />
202
- </form>
203
- </div>
204
- </div>
205
- </div>
206
- </div>
207
- </div>
208
- <br style="clear:both;" />
209
- <?php do_action('em_bookings_single_footer'); ?>
210
- </div>
211
- <?php
212
-
213
- }
214
-
215
- /**
216
- * Shows all bookings made by one person.
217
- */
218
- function em_bookings_person(){
219
- global $EM_Person;
220
- ?>
221
- <div class='wrap'>
222
- <div id='icon-users' class='icon32'>
223
- <br/>
224
- </div>
225
- <h2>
226
- <?php _e('Manage Person\'s Booking', 'dbem'); ?>
227
- <a href="admin.php?page=events-manager-bookings&action=person_delete&person_id=<?php echo $EM_Person->id; ?>" onclick="if( !confirm('<?php _e('Are you sure you want to delete this person? All bookings corresponding to this person will be deleted and this is not reversible.','dbem') ?>') ){ return false; }" class="button add-new-h2"><?php _e('Delete Person','dbem') ?></a>
228
- </h2>
229
- <div id="poststuff" class="metabox-holder has-right-sidebar">
230
- <div id="post-body">
231
- <div id="post-body-content">
232
- <div id="event_name" class="stuffbox">
233
- <h3>
234
- <?php _e ( 'Personal Details', 'dbem' ); ?>
235
- </h3>
236
- <div class="inside">
237
- <?php em_person_edit_form(); ?>
238
- </div>
239
- </div>
240
- </div>
241
- </div>
242
- </div>
243
- <br style="clear:both;" />
244
- <h3><?php _e('Past And Present Bookings','dbem'); ?></h3>
245
- <?php em_bookings_person_table(); ?>
246
- </div>
247
- <?php
248
- }
249
-
250
- function em_bookings_edit_form(){
251
- global $EM_Booking;
252
- $EM_Event = new EM_Event($EM_Booking->event_id);
253
- $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
254
- $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
255
- ?>
256
- <form action="" method="post" id="em-person-form">
257
- <h4>Event Details</h4>
258
- <table>
259
- <tr><td><strong><?php _e('Name','dbem'); ?></strong></td><td><a class="row-title" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;event_id=<?php echo $EM_Event->id ?>"><?php echo ($EM_Event->name); ?></a></td></tr>
260
- <tr>
261
- <td><strong><?php _e('Date/Time','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</strong></td>
262
- <td>
263
- <?php echo $localised_start_date; ?>
264
- <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
265
- <?php echo substr ( $EM_Event->start_time, 0, 5 ) . " - " . substr ( $EM_Event->end_time, 0, 5 ); ?>
266
- </td>
267
- </tr>
268
- </table>
269
- <h4>Personal Details</h4>
270
- <table>
271
- <tr><td><strong><?php _e('Name','dbem'); ?> </strong></td><td><a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;person_id=<?php echo $EM_Booking->person_id; ?>"><?php echo $EM_Booking->person->name ?></a></td></tr>
272
- <tr><td><strong><?php _e('Phone','dbem'); ?> </strong></td><td><?php echo $EM_Booking->person->phone; ?></td></tr>
273
- <tr><td><strong><?php _e('E-mail','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</strong></td><td><?php echo $EM_Booking->person->email; ?></td></tr>
274
- </table>
275
- <h4>Booking Details</h4>
276
- <table>
277
- <tr><td><strong><?php _e('Spaces','dbem'); ?> </strong></td><td><input type="text" name="booking_seats" value="<?php echo $EM_Booking->seats; ?>" /></td></tr>
278
- <tr><td><strong><?php _e('Comment','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</strong></td><td><textarea name="booking_comment"><?php echo $EM_Booking->comment; ?></textarea></td></tr>
279
- </table>
280
- <p class="submit">
281
- <input type="submit" name="events_update" value="<?php _e ( 'Save' ); ?> &raquo;" />
282
- </p>
283
- <input type="hidden" name="action" value="bookings_edit" />
284
- <input type="hidden" name="booking_id" value="<?php echo $EM_Booking->id; ?>" />
285
- </form>
286
- <?php
287
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
288
  ?>
1
+ <?php
2
+ /**
3
+ * Check if there's any admin-related actions to take for bookings. All actions are caught here.
4
+ * @return null
5
+ */
6
+ function em_admin_actions_bookings() {
7
+ global $dbem_form_add_message;
8
+ global $dbem_form_delete_message;
9
+ global $wpdb, $EM_Booking, $EM_Event, $EM_Notices;
10
+
11
+ if( is_object($EM_Booking) && !empty($_REQUEST['action']) && $EM_Booking->can_manage('manage_bookings','manage_others_bookings') ) {
12
+ if( $_REQUEST['action'] == 'bookings_add_note' ){
13
+ $EM_Booking->add_note($_REQUEST['booking_note']);
14
+ function em_booking_save_notification(){ global $EM_Booking; ?><div class="updated"><p><strong><?php echo $EM_Booking->feedback_message; ?></strong></p></div><?php }
15
+ add_action ( 'admin_notices', 'em_booking_save_notification' );
16
+ }
17
+ }
18
+ if( is_object($EM_Event) && !empty($_REQUEST['action']) ){
19
+ if( $_REQUEST['action'] == 'bookings_export_csv' && wp_verify_nonce($_REQUEST['_wpnonce'],'bookings_export_csv') ){
20
+ $EM_Event->get_bookings()->export_csv();
21
+ exit();
22
+ }
23
+ }
24
+ }
25
+ add_action('admin_init','em_admin_actions_bookings',100);
26
+
27
+ /**
28
+ * Decide what content to show in the bookings section.
29
+ */
30
+ function em_bookings_page(){
31
+ global $action;
32
+ //First any actions take priority
33
+ if( !empty($_REQUEST['action']) && substr($_REQUEST['action'],0,7) != 'booking' ){ //actions not starting with booking_
34
+ do_action('em_bookings_'.$action);
35
+ }elseif( !empty($_REQUEST['booking_id']) ){
36
+ em_bookings_single();
37
+ }elseif( !empty($_REQUEST['person_id']) ){
38
+ em_bookings_person();
39
+ }elseif( !empty($_REQUEST['event_id']) ){
40
+ em_bookings_event();
41
+ }elseif( !empty($_REQUEST['ticket_id']) ){
42
+ em_bookings_ticket();
43
+ }else{
44
+ em_bookings_dashboard();
45
+ }
46
+ }
47
+
48
+ /**
49
+ * Generates the bookings dashboard, showing information on all events
50
+ */
51
+ function em_bookings_dashboard(){
52
+ global $EM_Notices;
53
+ ?>
54
+ <div class='wrap'>
55
+ <div id='icon-users' class='icon32'>
56
+ <br/>
57
+ </div>
58
+ <h2>
59
+ <?php _e('Event Bookings Dashboard', 'dbem'); ?>
60
+ </h2>
61
+ <?php echo $EM_Notices; ?>
62
+ <?php if( get_option('dbem_bookings_approval')): ?>
63
+ <h2><?php _e('Pending Bookings','dbem'); ?></h2>
64
+ <?php em_bookings_pending_table(); ?>
65
+ <?php endif; ?>
66
+ <h2><?php _e('Events With Bookings Enabled','dbem'); ?></h2>
67
+ <?php em_bookings_events_table(); ?>
68
+ <?php do_action('em_bookings_dashboard'); ?>
69
+ </div>
70
+ <?php
71
+ }
72
+
73
+ /**
74
+ * Shows all booking data for a single event
75
+ */
76
+ function em_bookings_event(){
77
+ global $EM_Event,$EM_Person,$EM_Notices;
78
+ //check that user can access this page
79
+ if( is_object($EM_Event) && !$EM_Event->can_manage('manage_bookings','manage_others_bookings') ){
80
+ ?>
81
+ <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
82
+ <?php
83
+ return false;
84
+ }
85
+ $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
86
+ $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
87
+ ?>
88
+ <div class='wrap'>
89
+ <div id='icon-users' class='icon32'>
90
+ <br/>
91
+ </div>
92
+ <h2>
93
+ <?php echo sprintf(__('Manage %s Bookings', 'dbem'), "'{$EM_Event->event_name}'"); ?>
94
+ <a href="<?php echo $EM_Event->output('#_EDITEVENTURL'); ?>" class="button add-new-h2"><?php _e('View/Edit Event','dbem') ?></a>
95
+ <?php do_action('em_admin_event_booking_options_buttons'); ?>
96
+ </h2>
97
+ <?php echo $EM_Notices; ?>
98
+ <div><a href='<?php echo EM_ADMIN_URL ."&amp;page=events-manager-bookings&action=bookings_export_csv&_wpnonce=".wp_create_nonce('bookings_export_csv')."&event_id=".$EM_Event->event_id ?>'><?php _e('export csv','dbem')?></a></div>
99
+ <div>
100
+ <p><strong><?php _e('Event Name','dbem'); ?></strong> : <?php echo ($EM_Event->event_name); ?></p>
101
+ <p><strong><?php _e('Availability','dbem'); ?></strong> : <?php echo $EM_Event->get_bookings()->get_booked_spaces() . '/'. $EM_Event->get_spaces() ." ". __('Spaces confirmed','dbem'); ?></p>
102
+ <p>
103
+ <strong><?php _e('Date','dbem'); ?></strong> :
104
+ <?php echo $localised_start_date; ?>
105
+ <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
106
+ <?php echo substr ( $EM_Event->event_start_time, 0, 5 ) . " - " . substr ( $EM_Event->event_end_time, 0, 5 ); ?>
107
+ </p>
108
+ <p>
109
+ <strong><?php _e('Location','dbem'); ?></strong> :
110
+ <a class="row-title" href="<?php echo admin_url(); ?>post.php?action=edit&amp;post=<?php echo $EM_Event->get_location()->post_id ?>"><?php echo ($EM_Event->get_location()->location_name); ?></a>
111
+ </p>
112
+ </div>
113
+ <?php if( get_option('dbem_bookings_approval')): ?>
114
+ <h2><?php _e('Pending Bookings','dbem'); ?></h2>
115
+ <?php em_bookings_pending_table(); ?>
116
+ <?php endif; ?>
117
+ <h2><?php _e('Confirmed Bookings','dbem'); ?></h2>
118
+ <?php em_bookings_confirmed_table(); ?>
119
+ <h2><?php _e('Rejected Bookings','dbem'); ?></h2>
120
+ <?php em_bookings_rejected_table(); ?>
121
+ <h2><?php _e('Cancelled Bookings','dbem'); ?></h2>
122
+ <?php em_bookings_cancelled_table(); ?>
123
+ <?php do_action('em_bookings_event_footer', $EM_Event); ?>
124
+ </div>
125
+ <?php
126
+ }
127
+
128
+ /**
129
+ * Shows a ticket view
130
+ */
131
+ function em_bookings_ticket(){
132
+ global $EM_Ticket,$EM_Notices;
133
+ $EM_Event = $EM_Ticket->get_event();
134
+ //check that user can access this page
135
+ if( is_object($EM_Ticket) && !$EM_Ticket->can_manage() ){
136
+ ?>
137
+ <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this ticket.','dbem'); ?></p></div>
138
+ <?php
139
+ return false;
140
+ }
141
+ ?>
142
+ <div class='wrap'>
143
+ <div id='icon-users' class='icon32'>
144
+ <br/>
145
+ </div>
146
+ <h2>
147
+ <?php echo sprintf(__('Ticket for %s', 'dbem'), "'{$EM_Event->name}'"); ?>
148
+ <a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-event&event_id=<?php echo $EM_Event->event_id; ?>" class="button add-new-h2"><?php _e('View/Edit Event','dbem') ?></a>
149
+ <a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&event_id=<?php echo $EM_Event->event_id; ?>" class="button add-new-h2"><?php _e('View Event Bookings','dbem') ?></a>
150
+ </h2>
151
+ <?php echo $EM_Notices; ?>
152
+ <div>
153
+ <table>
154
+ <tr><td><?php echo __('Name','dbem'); ?></td><td></td><td><?php echo $EM_Ticket->ticket_name; ?></td></tr>
155
+ <tr><td><?php echo __('Description','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</td><td></td><td><?php echo ($EM_Ticket->ticket_description) ? $EM_Ticket->ticket_description : '-'; ?></td></tr>
156
+ <tr><td><?php echo __('Price','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_price) ? $EM_Ticket->ticket_price : '-'; ?></td></tr>
157
+ <tr><td><?php echo __('Spaces','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_spaces) ? $EM_Ticket->ticket_spaces : '-'; ?></td></tr>
158
+ <tr><td><?php echo __('Min','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_min) ? $EM_Ticket->ticket_min : '-'; ?></td></tr>
159
+ <tr><td><?php echo __('Max','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_max) ? $EM_Ticket->ticket_max : '-'; ?></td></tr>
160
+ <tr><td><?php echo __('Start','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_start) ? $EM_Ticket->ticket_start : '-'; ?></td></tr>
161
+ <tr><td><?php echo __('End','dbem'); ?></td><td></td><td><?php echo ($EM_Ticket->ticket_end) ? $EM_Ticket->ticket_end : '-'; ?></td></tr>
162
+ <?php do_action('em_booking_admin_ticket_row', $EM_Ticket); ?>
163
+ </table>
164
+ </div>
165
+ <?php if( get_option('dbem_bookings_approval')): ?>
166
+ <h2><?php _e('Pending Bookings','dbem'); ?></h2>
167
+ <?php em_bookings_pending_table(); ?>
168
+ <?php endif; ?>
169
+ <h2><?php _e('Confirmed Bookings','dbem'); ?></h2>
170
+ <?php em_bookings_confirmed_table(); ?>
171
+ <h2><?php _e('Rejected Bookings','dbem'); ?></h2>
172
+ <?php em_bookings_rejected_table(); ?>
173
+ <h2><?php _e('Cancelled Bookings','dbem'); ?></h2>
174
+ <?php em_bookings_cancelled_table(); ?>
175
+ <?php do_action('em_bookings_ticket_footer', $EM_Ticket); ?>
176
+ </div>
177
+ <?php
178
+ }
179
+
180
+ /**
181
+ * Shows a single booking for a single person.
182
+ */
183
+ function em_bookings_single(){
184
+ global $EM_Booking, $EM_Notices;
185
+ //check that user can access this page
186
+ if( is_object($EM_Booking) && !$EM_Booking->can_manage() ){
187
+ ?>
188
+ <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
189
+ <?php
190
+ return false;
191
+ }
192
+ ?>
193
+ <div class='wrap'>
194
+ <div id='icon-users' class='icon32'>
195
+ <br/>
196
+ </div>
197
+ <h2>
198
+ <?php _e('Edit Booking', 'dbem'); ?>
199
+ </h2>
200
+ <?php echo $EM_Notices; ?>
201
+ <div id="poststuff" class="metabox-holder">
202
+ <div id="post-body">
203
+ <div id="post-body-content">
204
+ <div id="em-booking-details" class="stuffbox">
205
+ <h3>
206
+ <?php _e ( 'Event Details', 'dbem' ); ?>
207
+ </h3>
208
+ <div class="inside">
209
+ <?php
210
+ $EM_Event = $EM_Booking->get_event();
211
+ $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
212
+ $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
213
+ ?>
214
+ <table>
215
+ <tr><td><strong><?php _e('Name','dbem'); ?></strong></td><td><a class="row-title" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;event_id=<?php echo $EM_Event->event_id ?>"><?php echo ($EM_Event->event_name); ?></a></td></tr>
216
+ <tr>
217
+ <td><strong><?php _e('Date/Time','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</strong></td>
218
+ <td>
219
+ <?php echo $localised_start_date; ?>
220
+ <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
221
+ <?php echo substr ( $EM_Event->start_time, 0, 5 ) . " - " . substr ( $EM_Event->end_time, 0, 5 ); ?>
222
+ </td>
223
+ </tr>
224
+ </table>
225
+ </div>
226
+ </div>
227
+ <div id="em-booking-details" class="stuffbox">
228
+ <h3>
229
+ <?php _e ( 'Personal Details', 'dbem' ); ?>
230
+ </h3>
231
+ <div class="inside">
232
+ <?php echo $EM_Booking->get_person()->display_summary(); ?>
233
+ </div>
234
+ </div>
235
+ <div id="em-booking-details" class="stuffbox">
236
+ <h3>
237
+ <?php _e ( 'Booking Details', 'dbem' ); ?>
238
+ </h3>
239
+ <div class="inside">
240
+ <?php
241
+ $EM_Event = $EM_Booking->get_event();
242
+ $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
243
+ $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
244
+ $shown_tickets = array();
245
+ ?>
246
+ <p><strong><?php _e('Status','dbem'); ?> : </strong><?php echo $EM_Booking->get_status(); ?></p>
247
+ <form action="" method="post">
248
+ <table class="em-tickets-bookings-table" cellspacing="0" cellpadding="0">
249
+ <thead>
250
+ <tr>
251
+ <th><?php _e('Ticket Type','dbem'); ?></th>
252
+ <th><?php _e('Spaces','dbem'); ?></th>
253
+ <th><?php _e('Price','dbem'); ?></th>
254
+ </tr>
255
+ </thead>
256
+ <tbody>
257
+ <?php foreach($EM_Booking->get_tickets_bookings()->tickets_bookings as $EM_Ticket_Booking): ?>
258
+ <tr>
259
+ <td class="ticket-type"><a class="row-title" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;ticket_id=<?php echo $EM_Ticket_Booking->get_ticket()->ticket_id ?>"><?php echo $EM_Ticket_Booking->get_ticket()->ticket_name ?></a></td>
260
+ <td>
261
+ <input name="em_tickets[<?php echo $EM_Ticket_Booking->get_ticket()->ticket_id; ?>][spaces]" class="em-ticket-select" value="<?php echo $EM_Ticket_Booking->get_spaces(); ?>" />
262
+ </td>
263
+ <td><?php echo $EM_Ticket_Booking->get_price(true,true); ?></td>
264
+ </tr>
265
+ <?php $shown_tickets[] = $EM_Ticket_Booking->ticket_id; ?>
266
+ <?php endforeach; ?>
267
+ <?php if( count($shown_tickets) < count($EM_Event->get_bookings()->get_tickets()->tickets)): ?><tr>
268
+ <?php foreach($EM_Event->get_bookings()->get_tickets()->tickets as $EM_Ticket): ?>
269
+ <?php if( !in_array($EM_Ticket->ticket_id, $shown_tickets) ): ?>
270
+ <tr>
271
+ <td class="ticket-type"><a class="row-title" href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;ticket_id=<?php echo $EM_Ticket->ticket_id ?>"><?php echo $EM_Ticket->ticket_name ?></a></td>
272
+ <td>
273
+ <input name="em_tickets[<?php echo $EM_Ticket->ticket_id; ?>][spaces]" class="em-ticket-select" value="0" />
274
+ </td>
275
+ <td><?php echo em_get_currency_symbol() ?>0.00</td>
276
+ </tr>
277
+ <?php endif; ?>
278
+ <?php endforeach; ?>
279
+ <?php endif; ?>
280
+ </tbody>
281
+ <tfoot>
282
+ <tr>
283
+ <th><?php _e('Totals','dbem'); ?></th>
284
+ <th><?php echo $EM_Booking->get_spaces(); ?></th>
285
+ <th><?php echo $EM_Booking->get_price(true, true); ?></th>
286
+ </tr>
287
+ <?php if( !get_option('dbem_bookings_tax_auto_add') && is_numeric(get_option('dbem_bookings_tax')) && get_option('dbem_bookings_tax') > 0 ): ?>
288
+ <tr>
289
+ <th><?php _e('Tax','dbem'); ?></th>
290
+ <th><?php echo get_option('dbem_bookings_tax') ?>%</th>
291
+ <th><?php echo em_get_currency_symbol().number_format($EM_Booking->get_price() * (get_option('dbem_bookings_tax')/100),2); ?></th>
292
+ </tr>
293
+ <tr>
294
+ <th><?php _e('Total (inc. tax)','dbem'); ?></th>
295
+ <th>&nbsp;</th>
296
+ <th><?php echo em_get_currency_symbol().number_format($EM_Booking->get_price()* (1 + get_option('dbem_bookings_tax')/100),2); ?></th>
297
+ </tr>
298
+ <?php endif; ?>
299
+ </tfoot>
300
+ </table>
301
+ <p>
302
+ <input type="submit" class="em-booking-submit" id="em-booking-submit" value="<?php _e('Modify Booking', 'dbem'); ?>" />
303
+ <input type='hidden' name='action' value='booking_save'/>
304
+ <input type='hidden' name='booking_id' value='<?php echo $EM_Booking->booking_id; ?>'/>
305
+ <input type='hidden' name='event_id' value='<?php echo $EM_Event->event_id; ?>'/>
306
+ <input type='hidden' name='_wpnonce' value='<?php echo wp_create_nonce('booking_save'); ?>'/>
307
+ <em><?php _e('<strong>Note:</strong> ticket availability not taken into account (i.e. you can overbook). Confirmation email is not resent automatically.','dbem'); ?></em>
308
+ </p>
309
+ <table cellspacing="0" cellpadding="0">
310
+ <?php if( !get_option('em_booking_form_custom') ): ?>
311
+ <tr><td><strong><?php _e('Comment','dbem'); ?>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</strong></td><td><?php echo $EM_Booking->booking_comment; ?></td></tr>
312
+ <?php foreach( $EM_Booking->get_custom() as $custom_option ){
313
+ ?><tr><td><strong><?php echo $custom_option['name'] ?></strong></td><td><?php echo esc_html($custom_option['value']); ?></td></tr><?php
314
+ } ?>
315
+ <?php else: do_action('em_bookings_single_custom',$EM_Booking); ?>
316
+ <?php endif; ?>
317
+ </table>
318
+ </form>
319
+ </div>
320
+ </div>
321
+ <div id="em-booking-notes" class="stuffbox">
322
+ <h3>
323
+ <?php _e ( 'Booking Notes', 'dbem' ); ?>
324
+ </h3>
325
+ <div class="inside">
326
+ <p><?php _e('You can add private notes below for internal reference that only event managers will see.','dbem'); ?></p>
327
+ <?php foreach( $EM_Booking->notes as $note ):
328
+ $user = new EM_Person($note['author']);
329
+ ?>
330
+ <div>
331
+ <?php echo date(get_option('date_format'), $note['timestamp']) .' - '. $user->get_name(); ?> <?php _e('wrote','dbem'); ?>:
332
+ <p style="background:#efefef; padding:5px;"><?php echo nl2br($note['note']); ?></p>
333
+ </div>
334
+ <?php endforeach; ?>
335
+ <form method="post" action="" style="padding:5px;">
336
+ <textarea class="widefat" rows="5" name="booking_note"></textarea>
337
+ <input type="hidden" name="action" value="bookings_add_note" />
338
+ <input type="submit" value="Add Note" />
339
+ </form>
340
+ </div>
341
+ </div>
342
+ <?php do_action('em_bookings_single_metabox_footer', $EM_Booking); ?>
343
+ </div>
344
+ </div>
345
+ </div>
346
+ <br style="clear:both;" />
347
+ <?php do_action('em_bookings_single_footer', $EM_Booking); ?>
348
+ </div>
349
+ <?php
350
+
351
+ }
352
+
353
+ /**
354
+ * Shows all bookings made by one person.
355
+ */
356
+ function em_bookings_person(){
357
+ global $EM_Person, $EM_Notices;
358
+ $EM_Person->get_bookings();
359
+ $has_booking = false;
360
+ foreach($EM_Person->get_bookings() as $EM_Booking){
361
+ if($EM_Booking->can_manage('manage_bookings','manage_others_bookings')){
362
+ $has_booking = true;
363
+ }
364
+ }
365
+ if( !$has_booking ){
366
+ ?>
367
+ <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
368
+ <?php
369
+ return false;
370
+ }
371
+ ?>
372
+ <div class='wrap'>
373
+ <div id='icon-users' class='icon32'>
374
+ <br/>
375
+ </div>
376
+ <h2>
377
+ <?php _e('Manage Person\'s Booking', 'dbem'); ?>
378
+ <?php if( current_user_can('edit_users') ) : ?>
379
+ <a href="user-edit.php?user_id=<?php echo $EM_Person->ID; ?>" class="button add-new-h2"><?php _e('Edit User','dbem') ?></a>
380
+ <?php endif; ?>
381
+ </h2>
382
+ <?php echo $EM_Notices; ?>
383
+ <?php do_action('em_bookings_person_header'); ?>
384
+ <div id="poststuff" class="metabox-holder has-right-sidebar">
385
+ <div id="post-body">
386
+ <div id="post-body-content">
387
+ <div id="event_name" class="stuffbox">
388
+ <h3>
389
+ <?php _e ( 'Personal Details', 'dbem' ); ?>
390
+ </h3>
391
+ <div class="inside">
392
+ <?php echo $EM_Person->display_summary(); ?>
393
+ </div>
394
+ </div>
395
+ </div>
396
+ </div>
397
+ </div>
398
+ <br style="clear:both;" />
399
+ <?php do_action('em_bookings_person_body_1'); ?>
400
+ <h3><?php _e('Past And Present Bookings','dbem'); ?></h3>
401
+ <?php em_bookings_person_table(); ?>
402
+ <?php do_action('em_bookings_person_footer', $EM_Person); ?>
403
+ </div>
404
+ <?php
405
+ }
406
+
407
+ function em_printable_booking_report() {
408
+ global $EM_Event;
409
+ //check that user can access this page
410
+ if( isset($_GET['page']) && $_GET['page']=='events-manager-bookings' && isset($_GET['action']) && $_GET['action'] == 'bookings_report' && is_object($EM_Event)){
411
+ if( is_object($EM_Event) && !$EM_Event->can_manage('edit_events','edit_others_events') ){
412
+ ?>
413
+ <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
414
+ <?php
415
+ return false;
416
+ }
417
+ em_locate_template('templates/bookings-event-printable.php', true);
418
+ die();
419
+ }
420
+ }
421
+ add_action('admin_init', 'em_printable_booking_report');
422
  ?>
admin/em-categories.php DELETED
@@ -1,177 +0,0 @@
1
- <?php
2
- function em_admin_categories_page() {
3
- global $wpdb, $EM_Category;
4
-
5
- if( !empty($_REQUEST['action']) ){
6
- if( $_REQUEST['action'] == "save") {
7
- // save (add/update) category
8
- if( empty($EM_Category) || !is_object($EM_Category) ){
9
- $EM_Category = new EM_Category(); //blank category
10
- $success_message = __('The category has been added.', 'dbem');
11
- }else{
12
- $success_message = __('The category has been updated.', 'dbem');
13
- }
14
- $EM_Category->get_post();
15
- if ( $EM_Category->validate() ) {
16
- $EM_Category->save(); //FIXME better handling of db write fails when saving category
17
- em_categories_table_layout($success_message);
18
- } else {
19
- ?>
20
- <div id='message' class='error '>
21
- <p>
22
- <strong><?php _e( "Ach, there's a problem here:", "dbem" ) ?></strong><br /><br /><?php echo implode('<br />', $EM_Category->errors); ?>
23
- </p>
24
- </div>
25
- <?php
26
- em_categories_edit_layout();
27
- }
28
- } elseif( $_REQUEST['action'] == "edit" ){
29
- em_categories_edit_layout();
30
- } elseif( $_REQUEST['action'] == "delete" ){
31
- //delelte category
32
- EM_Categories::delete($_REQUEST['categories']);
33
- //FIXME no result verification when deleting various categories
34
- $message = __('Categories Deleted', "dbem" );
35
- em_categories_table_layout($message);
36
- }
37
- }else{
38
- em_categories_table_layout($message);
39
- }
40
- }
41
-
42
- function em_categories_table_layout($message = "") {
43
- $categories = EM_Categories::get();
44
- $destination = get_bloginfo('url')."/wp-admin/admin.php";
45
- ?>
46
- <div class='wrap nosubsub'>
47
- <div id='icon-edit' class='icon32'>
48
- <br/>
49
- </div>
50
- <h2><?php echo __('Categories', 'dbem') ?></h2>
51
-
52
- <?php if($message != "") : ?>
53
- <div id='message' class='updated fade below-h2' style='background-color: rgb(255, 251, 204);'>
54
- <p><?php echo $message ?></p>
55
- </div>
56
- <?php endif; ?>
57
-
58
- <div id='col-container'>
59
- <!-- begin col-right -->
60
- <div id='col-right'>
61
- <div class='col-wrap'>
62
- <form id='bookings-filter' method='post' action='<?php echo get_bloginfo('wpurl') ?>/wp-admin/admin.php?page=events-manager-categories'>
63
- <input type='hidden' name='action' value='delete'/>
64
- <?php if (count($categories)>0) : ?>
65
- <table class='widefat'>
66
- <thead>
67
- <tr>
68
- <th class='manage-column column-cb check-column' scope='col'><input type='checkbox' class='select-all' value='1'/></th>
69
- <th><?php echo __('ID', 'dbem') ?></th>
70
- <th><?php echo __('Name', 'dbem') ?></th>
71
- </tr>
72
- </thead>
73
- <tfoot>
74
- <tr>
75
- <th class='manage-column column-cb check-column' scope='col'><input type='checkbox' class='select-all' value='1'/></th>
76
- <th><?php echo __('ID', 'dbem') ?></th>
77
- <th><?php echo __('Name', 'dbem') ?></th>
78
- </tr>
79
- </tfoot>
80
- <tbody>
81
- <?php foreach ($categories as $EM_Category) : ?>
82
- <tr>
83
- <td><input type='checkbox' class ='row-selector' value='<?php echo $EM_Category->id ?>' name='categories[]'/></td>
84
- <td><a href='<?php echo get_bloginfo('wpurl') ?>/wp-admin/admin.php?page=events-manager-categories&amp;action=edit&amp;category_id=<?php echo $EM_Category->id ?>'><?php echo htmlspecialchars($EM_Category->id, ENT_QUOTES); ?></a></td>
85
- <td><a href='<?php echo get_bloginfo('wpurl') ?>/wp-admin/admin.php?page=events-manager-categories&amp;action=edit&amp;category_id=<?php echo $EM_Category->id ?>'><?php echo htmlspecialchars($EM_Category->name, ENT_QUOTES); ?></a></td>
86
- </tr>
87
- <?php endforeach; ?>
88
- </tbody>
89
-
90
- </table>
91
-
92
- <div class='tablenav'>
93
- <div class='alignleft actions'>
94
- <input class='button-secondary action' type='submit' name='doaction2' value='Delete'/>
95
- <br class='clear'/>
96
- </div>
97
- <br class='clear'/>
98
- </div>
99
- <?php else: ?>
100
- <p><?php echo __('No categories have been inserted yet!', 'dbem'); ?></p>
101
- <?php endif; ?>
102
- </form>
103
- </div>
104
- </div>
105
- <!-- end col-right -->
106
-
107
- <!-- begin col-left -->
108
- <div id='col-left'>
109
- <div class='col-wrap'>
110
- <div class='form-wrap'>
111
- <div id='ajax-response'>
112
- <h3><?php echo __('Add category', 'dbem') ?></h3>
113
- <form name='add' id='add' method='post' action='admin.php?page=events-manager-categories' class='add:the-list: validate'>
114
- <input type='hidden' name='action' value='save' />
115
- <div class='form-field form-required'>
116
- <label for='category_name'><?php echo __('Category name', 'dbem') ?></label>
117
- <input id='category-name' name='category_name' id='category_name' type='text' size='40' />
118
- <p><?php echo __('The name of the category', 'dbem'); ?></p>
119
- </div>
120
- <p class='submit'><input type='submit' class='button' name='submit' value='<?php echo __('Add category', 'dbem') ?>' /></p>
121
- </form>
122
- </div>
123
- </div>
124
- </div>
125
- </div>
126
- <!-- end col-left -->
127
- </div>
128
- </div>
129
- <?php
130
- }
131
-
132
-
133
- function em_categories_edit_layout($message = "") {
134
- global $EM_Category;
135
- if( !is_object($EM_Category) ){
136
- $EM_Category = new EM_Category();
137
- }
138
- //check that user can access this page
139
- if( is_object($EM_Category) && !$EM_Category->can_manage() ){
140
- ?>
141
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
142
- <?php
143
- return;
144
- }
145
- ?>
146
- <div class='wrap'>
147
- <div id='icon-edit' class='icon32'>
148
- <br/>
149
- </div>
150
-
151
- <h2><?php echo __('Edit category', 'dbem') ?></h2>
152
-
153
- <?php if($message != "") : ?>
154
- <div id='message' class='updated fade below-h2' style='background-color: rgb(255, 251, 204);'>
155
- <p><?php echo $message ?></p>
156
- </div>
157
- <?php endif; ?>
158
-
159
- <div id='ajax-response'></div>
160
-
161
- <form name='editcat' id='editcat' method='post' action='admin.php?page=events-manager-categories' class='validate'>
162
- <input type='hidden' name='action' value='save' />
163
- <input type='hidden' name='category_id' value='<?php echo $EM_Category->id ?>'/>
164
-
165
- <table class='form-table'>
166
- <tr class='form-field form-required'>
167
- <th scope='row' valign='top'><label for='category_name'><?php echo __('Category name', 'dbem') ?></label></th>
168
- <td><input name='category_name' id='category-name' type='text' value='<?php echo $EM_Category->name ?>' size='40' /><br />
169
- <?php echo __('The name of the category', 'dbem') ?></td>
170
- </tr>
171
- </table>
172
- <p class='submit'><input type='submit' class='button-primary' name='submit' value='<?php echo __('Update category', 'dbem') ?>' /></p>
173
- </form>
174
- </div>
175
- <?php
176
- }
177
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
admin/em-docs.php CHANGED
@@ -1,193 +1,286 @@
1
- <?php
2
-
3
- function em_docs_init(){
4
- if( class_exists('EM_Event') ){
5
- add_action('wp_head', 'emd_head');
6
- //Generate the docs
7
- global $EM_Documentation;
8
- $EM_Event = new EM_Event();
9
- $event_fields = $EM_Event->get_fields(true);
10
- $EM_Location = new EM_Location();
11
- $location_fields = $EM_Location->get_fields(true);
12
- $EM_Documentation = array(
13
- 'arguments' => array(
14
- 'events' => array(
15
- 'limit' => array( 'default'=>'See the events lists limit option on the settings page.' ),
16
- 'orderby' => array( 'desc'=>'Choose what fields to order your results by. You can supply a single field or multiple comma-seperated fields (e.g. "start_date,name").', 'default'=>'See the event lists ordering option on the settings page.', 'args'=>'name, start_date, start_time, end_date, end_time'),
17
- 'order' => array( 'default'=>'See the event lists ordering option on the settings page.' )
18
- ),
19
- 'locations' => array(
20
- 'eventful' => array( 'desc'=>'If set to 1 will only show locations that have at least one event occurring during the scope.', 'default' => 0),
21
- 'eventless' => array( 'desc'=>'If set to 1 will only show locations that have no events occurring during the scope.', 'default' => 0),
22
- 'orderby' => array('desc'=>'Choose what fields to order your results by. You can supply a single field or multiple comma-seperated fields (e.g. "start_date,name").', 'default'=>'name', 'args' => 'name, address, town'),
23
- 'scope' => array( 'default' => 'all')
24
- ),
25
- 'calendar' => array(
26
- 'full' => array( 'desc'=>'If set to 1 it will display a full calendar that shows event names.', 'default' => 0),
27
- 'long_events' => array( 'desc'=>'If set to 1, will show events that last longer than a day.', 'default' => 0),
28
- 'order' => array( 'desc'=>'Same as for events.' ),
29
- 'scope' => array( 'default' => 'future')
30
- ),
31
- //The object is commonly shared by all, so entries above overwrite entries here
32
- 'general' => array(
33
- 'limit' => array( 'desc'=>'Limits the amount of values returned to this number.', 'default'=>'0 (no limit)'),
34
- 'scope' => array( 'desc'=>'Choose the time frame of events to show. Accepted values are "future", "past" or "all" events. Additionally you can supply dates (in format of YYYY-MM-DD), either single for events on a specific date or two dates seperated by a comma (e.g. 2010-12-25,2010-12-31) for events ocurring between these dates.', 'default'=>'future'),
35
- 'order' => array( 'desc'=>'Indicates the order of the events. Choose between ASC (ascending) and DESC (descending).', 'default'=>'ASC'),
36
- 'orderby' => array( 'desc'=>'Choose what fields to order your results by. You can supply a single field or multiple comma-seperated fields (e.g. "start_date,name"). See specific instances (e.g. events, locations, etc.) for field names.', 'default'=>0),
37
- 'format' => array( 'desc'=>'If you are displaying some information with the shortcode or function (e.g. listing events), you can supply the html and placeholders here.', 'default'=>'The relevant default format will be taken from the settings page.'),
38
- 'event' => array( 'desc'=>'Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to events with these id(s).', 'default'=>0),
39
- 'category' => array( 'desc'=>'Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to events in these categories.', 'default'=>0),
40
- 'location' => array( 'desc'=>'Supply a single id or comma-seperated ids to limit the search to these locations (or events in these locations).', 'default'=>0),
41
- 'offset' => array( 'desc'=>'For example, if you have ten results, if you set this to 5, only the last 5 results will be returned. Useful for pagination.', 'default'=>0),
42
- 'recurrence' => array( 'desc'=>'If set to 1, will show only events that are recurring (i.e. events that are repeated over different dates).', 'default'=>0),
43
- 'recurring' => array( 'desc'=>'If set to 1, will only show recurring event templates. Only useful if you know what you\'re doing, use recurrence if you want events that are recurrences.', 'default'=>0),
44
- 'month' => array( 'desc'=>'If set to a month (1 to 12) only events that start or end during this month/year will be retured. Must be used in conjunction with year', 'default'=>''),
45
- 'year' => array( 'desc'=>'If set to a year (e.g. 2010) only events that start or end during this year/month will be returned. Must be used in conjunction with year', 'default'=>''),
46
- 'array' => array( 'desc'=>'If you supply this as an argument, the returned data will be in an array, no objects (only useful wen using PHP, not shortcodes)', 'default'=>0),
47
- 'pagination' => array('desc'=>'When using a function or shortcode that outputs items (e.g. [events_list] for events, [locations_list] for locations), if the number of items supercede the limit of items to show, setting this to 1 will show page links under the list.', 'default'=>0)
48
- )
49
- ),
50
- 'placeholders' => array(
51
- 'events' => array(
52
- 'Event Details' => array(
53
- 'placeholders' => array(
54
- '#_NAME' => array( 'desc' => 'Displays the name of the event.' ),
55
- '#_NOTES' => array( 'desc' => 'Shows the description of the event.' ),
56
- '#_EXCERPT' => array( 'desc' => 'If you added a <a href="http://en.support.wordpress.com/splitting-content/more-tag/">more tag</a> to your event description, only the content before this tag will show (currently, no read more link is added).' ),
57
- '#_EVENTID' => array( 'desc' => 'Shows the event\'s corresponding ID number in the database table.' )
58
- )
59
- ),
60
- 'Category Details' => array(
61
- 'placeholders' => array(
62
- '#_CATEGORYNAME' => array( 'desc' => 'Shows the category name of the event.' ),
63
- '#_CATEGORYID' => array( 'desc' => 'Shows the category ID of the event.' )
64
- )
65
- ),
66
- 'Time' => array(
67
- 'desc' => '',
68
- 'placeholders' => array(
69
- '#_24HSTARTTIME' => array( 'desc' => 'Displays the start time in a 24 hours format (e.g. 16:30).' ),
70
- '#_24HENDTIME' => array( 'desc' => 'Displays the end time in a 24 hours format (e.g. 18:30).' ),
71
- '#_12HSTARTTIME' => array( 'desc' => 'Displays the start time in a 12 hours format (e.g. 4:30 PM).' ),
72
- '#_12HENDTIME' => array( 'desc' => 'Displays the end time in a 12 hours format (e.g. 6:30 PM).' )
73
- )
74
- ),
75
- 'Custom Date/Time Formatting' => array(
76
- 'desc' => 'Events Manager allows extremely flexible date formatting by using <a href="http://www.php.net/manual/en/function.date.php">PHP date syntax format characters</a> along with placeholders.',
77
- 'placeholders' => array(
78
- '# or #@' => array( 'desc' => 'Prepend <code>#</code> or <code>#@</code> before a valid PHP date syntax format character to show start and end date/time information respectively (e.g. <code>#F</code> will show the starting month name like "January", #@h shows the end hour).' ),
79
- '#{x} or #@{x}' => array( 'desc' => 'You can also create a date format without prepending # to each character by wrapping a valid php date() format in <code>#{}</code> or <code>#@{}</code> (e.g. <code>#_{d/m/Y}</code>). If there is no end date (or is same as start date), the value is not shown. This is useful if you want to show event end dates only on events that are longer than on day, e.g. <code>#j #M #Y #@_{ \u\n\t\i\l j M Y}</code>.' ),
80
- )
81
- ),
82
- 'Links' => array(
83
- 'placeholders' => array(
84
- '#_EVENTURL' => array( 'desc' => 'Simply prints the event URL. You can use this placeholder to build your own customised links.' ),
85
- '#_EVENTLINK' => array( 'desc' => 'Displays the event name with a link to the event page.' ),
86
- '#_EDITEVENTLINK' => array( 'desc' => 'Inserts a link to the edit event page, only if a user is logged in and is allowed to edit the event.' )
87
- )
88
- ),
89
- 'Custom Attributes' => array(
90
- 'desc' => 'Events Manager allows you to create dynamic attributes to your events, which act as extra information fields for your events (e.g. "Dress Code"). For more information see <a href="http://wp-events-plugin.com/documentation/categories-and-attributes/">our online documentation</a> for more info on attributes.',
91
- 'placeholders' => array(
92
- '#_ATT{key}{alternative text}' => array('desc'=>'This key will appear as an option when adding attributes to your event. The second braces are optional and will appear if the attribute is not defined or left blank for that event.')
93
- )
94
- ),
95
- 'Bookings/RSVP' => array(
96
- 'desc' => 'These placeholders will only show if RSVP is enabled for the given event and in the events manager settings page. Spaces placeholders will default to 0',
97
- 'placeholders' => array(
98
- '#_ADDBOOKINGFORM' => array( 'desc' => 'Adds a form which allows the visitors to register for an event.' ),
99
- '#_REMOVEBOOKINGFORM' => array( 'desc' => 'Adds a form which allows the visitors to remove their booking.' ),
100
- '#_BOOKINGFORM' => array( 'desc' => 'Adds a both booking forms (add and remove).' ),
101
- '#_AVAILABLESPACES' => array( 'desc' => 'Shows available spaces for the event.' ),
102
- '#_BOOKEDSPACES' => array( 'desc' => 'Shows the amount of currently booked seats for the event.' ),
103
- '#_SPACES' => array( 'desc' => 'Shows the total spaces for the event.' )
104
- )
105
- ),
106
- 'Contact Details' => array(
107
- 'desc' => 'The values here are taken from the chosen contact for the specific event, or the default contact in the settings page.',
108
- 'placeholders' => array(
109
- '#_CONTACTNAME' => array( 'desc' => 'Name of the contact person for this event (as shown in the dropdown when adding an event).' ),
110
- '#_CONTACTUSERNAME' => array( 'desc' => 'Contact person\'s username.' ),
111
- '#_CONTACTEMAIL' => array( 'desc' => 'E-mail of the contact person for this event.' ),
112
- '#_CONTACTPHONE' => array( 'desc' => 'Phone number of the contact person for this event. Can be set in the user profile page.' ),
113
- '#_CONTACTAVATAR' => array( 'desc' => 'Contact person\'s avatar.' ),
114
- '#_CONTACTPROFILELINK' => array( 'desc' => 'Contact person\'s "Profile" link. Only works with BuddyPress enabled.' ),
115
- '#_CONTACTPROFILEURL' => array( 'desc' => 'Contact person\'s profile url. Only works with BuddyPress enabled.' ),
116
- '#_CONTACTID' => array( 'desc' => 'Contact person\'s wordpress user ID.')
117
- )
118
- ),
119
- ),
120
- 'locations' => array(
121
- 'Location Details' => array(
122
- 'desc' => '',
123
- 'placeholders' => array(
124
- '#_LOCATIONNAME' => array( 'desc' => 'Displays the location name.' ),
125
- '#_LOCATIONADDRESS' => array( 'desc' => 'Displays the address.' ),
126
- '#_LOCATIONTOWN' => array( 'desc' => 'Displays the town.' ),
127
- '#_LOCATIONMAP' => array( 'desc' => 'Displays a google map showing where the event is located (Will not show if maps are disabled in the settings page)' ),
128
- '#_LOCATIONNOTES' => array( 'desc' => 'Shows the location description.' ),
129
- '#_LOCATIONEXCERPT' => array( 'desc' => 'If you added a <a href="http://en.support.wordpress.com/splitting-content/more-tag/">more tag</a> to your location description, only the content before this tag will show (currently, no read more link is added).' ),
130
- '#_LOCATIONIMAGE' => array( 'desc' => 'Shows the location image.' ),
131
- '#_LOCATIONID' => array( 'desc' => 'Displays the location ID number.' )
132
- )
133
- ),
134
- 'Links' => array(
135
- 'placeholders' => array(
136
- '#_LOCATIONURL' => array( 'desc' => 'Simply prints the location URL. You can use this placeholder to build your own customised links.' ),
137
- '#_LOCATIONLINK' => array( 'desc' => 'Displays the location name with a link to the location page.' )
138
- )
139
- ),
140
- 'Related Events' => array(
141
- 'desc' => 'You can show lists of other events that are being held at this location. The formatting of the list is the same as a normal events list.',
142
- 'placeholders' => array(
143
- '#_LOCATIONPASTEVENTS' => array( 'desc' => 'Will show a list of all past events at this location.' ),
144
- '#_LOCATIONNEXTEVENTS' => array( 'desc' => 'Will show a list of all future events at this location.' ),
145
- '#_LOCATIONALLEVENTS' => array( 'desc' => 'Will show a list of all events at this location.' )
146
- )
147
- ),
148
- ),
149
- 'bookings' => array(
150
- 'Booking Person Information' => array(
151
- 'desc' => 'When a specific booking is displayed (on screen and on email), you can use these placeholders to show specific information about the booking. For contact details of the contact of this event, see the events placeholders.',
152
- 'placeholders' => array(
153
- '#_BOOKINGNAME' => array( 'desc' => 'Name of person who made the booking.' ),
154
- '#_BOOKINGEMAIL' => array( 'desc' => 'Email of person who made the booking.' ),
155
- '#_BOOKINGPHONE' => array( 'desc' => 'Phone number of person who made the booking.' ),
156
- '#_BOOKINGSPACES' => array( 'desc' => 'Number of spaces the person has booked.' ),
157
- '#_BOOKINGCOMMENT' => array( 'desc' => 'Any specific comments made by the person who made the booking.' )
158
- )
159
- )
160
- ),
161
- )
162
- );
163
- }
164
- }
165
- add_action('init', 'em_docs_init');
166
-
167
- function em_docs_placeholders($atts){
168
- ob_start();
169
- ?>
170
- <div class="em-docs">
171
- <?php
172
- global $EM_Documentation;
173
- $type = $atts['type'];
174
- $data = $EM_Documentation['placeholders'][$type];
175
- foreach($data as $sectionTitle => $details) : ?>
176
- <div>
177
- <h3><?php echo $sectionTitle; ?></h3>
178
- <?php if($details['desc'] != ''): ?>
179
- <p><?php echo $details['desc']; ?></p>
180
- <?php endif; ?>
181
- <dl>
182
- <?php foreach($details['placeholders'] as $placeholder => $desc ): ?>
183
- <dt><b><?php echo $placeholder; ?></b></dt>
184
- <dd><?php echo $desc['desc']; ?></dd>
185
- <?php endforeach; ?>
186
- </dl>
187
- </div>
188
- <?php endforeach; ?>
189
- </div>
190
- <?php
191
- return ob_get_clean();
192
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
193
  ?>
1
+ <?php
2
+
3
+ function em_docs_init($force_init = false){
4
+ global $pagenow;
5
+ if( ($pagenow == 'edit.php' && !empty($_GET['page']) && $_GET['page']=='events-manager-help' && class_exists('EM_Event')) || $force_init){
6
+ add_action('wp_head', 'emd_head');
7
+ //Generate the docs
8
+ global $EM_Documentation;
9
+ $EM_Event = new EM_Event();
10
+ $event_fields = $EM_Event->get_fields(true);
11
+ $EM_Location = new EM_Location();
12
+ $location_fields = $EM_Location->get_fields(true);
13
+ $EM_Documentation = array(
14
+ 'arguments' => array(
15
+ 'events' => array(
16
+ 'blog' => array( 'desc' => sprintf('Limit search to %s created in a specific blog id (MultiSite only)','events')),
17
+ 'bookings' => array( 'desc'=> 'Include only events with bookings enabled. Use \'user\' to show events a logged in user has booked.'.'1 = yes, 0 = no'),
18
+ 'category' => array( 'desc'=> sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to events in these %s.','categories'), 'default'=>0),
19
+ 'event' => array( 'desc'=> sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to %s with the %s.','events', 'event_id(s)'), 'default'=>0),
20
+ 'group' => array( 'desc' => 'Limit search to events belonging to a specific group id (BuddyPress only). Using \'my\' will show events belonging to groups the logged in user is a member of.'),
21
+ 'post_id' => array( 'desc' => sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to %s with the %s.','events', 'post_id(s)')),
22
+ 'private' => array( 'desc' => sprintf('Display private %s within your list?','events'), 'args' => '1 = yes, 0 = no', 'default' => 'If user can view private events, 1, otherwise 0.'),
23
+ 'private_only' => array( 'desc' =>sprintf('Display only private %s ?','events'), 'args' => '1 = yes, 0 = no', 'default' => '0'),
24
+ 'recurrence' => array( 'desc'=> 'If set to 1, will show only events that are recurring (i.e. events that are repeated over different dates).', 'default'=>0),
25
+ 'recurring' => array( 'desc'=> 'If set to 1, will only show recurring event templates. Only useful if you know what you\'re doing, use recurrence if you want events that are recurrences.', 'default'=>0),
26
+ 'search' => array( 'desc'=> 'Do a search for this string within event name, details and location address.' ),
27
+ 'status' => array( 'desc' => sprintf('Limit search to %s with a spefic status (1 is active, 0 is pending approval)','events'), 'default'=>1),
28
+ 'scope' => array( 'desc'=> 'Choose the time frame of events to show. Additionally you can supply dates (in format of YYYY-MM-DD), either single for events on a specific date or two dates seperated by a comma (e.g. 2010-12-25,2010-12-31) for events ocurring between these dates.', 'default'=>'future', 'args'=>array("future", "past", "today", "tomorrow", "month", "next-month", "1-months", "2-months", "3-months", "6-months", "12-months","all")),
29
+ 'category' => array( 'desc'=> sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to events in these %s.','tags'), 'default'=>0),
30
+ 'year' => array( 'desc'=> 'If set to a year (e.g. 2010) only events that start or end during this year/month will be returned. Does not work as intended if used with scope.', 'default'=>''),
31
+ ),
32
+ 'locations' => array(
33
+ 'blog' => array( 'desc' => sprintf('Limit search to %s created in a specific blog id (MultiSite only)','locations')),
34
+ 'eventful' => array( 'desc'=> 'If set to 1 will only show locations that have at least one event occurring during the scope.', 'default' => 0),
35
+ 'eventless' => array( 'desc'=> 'If set to 1 will only show locations that have no events occurring during the scope.', 'default' => 0),
36
+ 'location' => array( 'desc'=> sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to %s with the %s.','locations', 'location_id(s)'), 'default'=>0),
37
+ 'post_id' => array( 'desc' => sprintf('Supply a single id or comma-seperated ids (e.g. "1,2,3") to limit the search to %s with the %s.','locations', 'post_id(s)')),
38
+ 'private' => array( 'desc' => sprintf('Display private %s within your list?','locations'), 'args' => '1 = yes, 0 = no', 'default' => 'If user can view private locations, 1, otherwise 0.'),
39
+ 'private_only' => array( 'desc' =>sprintf('Display only private %s ?','locations'), 'args' => '1 = yes, 0 = no', 'default' => '0'),
40
+ 'scope' => array( 'default' => 'all'),
41
+ 'status' => array( 'desc' => sprintf('Limit search to %s with a spefic status (1 is active, 0 is pending approval)','locations'), 'default'=>1),
42
+ 'country' => array( 'desc'=> sprintf('Search for %s in this %s (no partial matches, case sensitive).','locations','Country'), 'default' => 'none', 'args'=>'Use two-character country codes as defined in <a href="http://countrycode.org/">countrycode.org</a>, e.g. US,GB,ES'),
43
+ 'region' => array( 'desc'=> sprintf('Search for %s in this %s (no partial matches, case sensitive).','locations','Region'), 'default' => 'none'),
44
+ 'state' => array( 'desc'=> sprintf('Search for %s in this %s (no partial matches, case sensitive).','locations','State'), 'default' => 'none'),
45
+ 'town' => array( 'desc'=> sprintf('Search for %s in this %s (no partial matches, case sensitive).','locations','Town'), 'default' => 'none'),
46
+ 'postcode' => array( 'desc'=> sprintf('Search for %s in this %s (no partial matches, case sensitive).','locations','Postcode'), 'default' => 'none')
47
+ ),
48
+ 'categories' => array(
49
+ '' => array( 'desc' => 'See the <a href="http://codex.wordpress.org/Function_Reference/get_terms">WordPress get_terms() Codex</a> for a list of possible search attributes/arguments.'),
50
+ ),
51
+ 'calendar' => array(
52
+ 'full' => array( 'desc'=> 'If set to 1 it will display a full calendar that shows event names.', 'default' => 0),
53
+ 'long_events' => array( 'desc'=> 'If set to 1, will show events that last longer than a day.', 'default' => 0),
54
+ ),
55
+ //The object is commonly shared by all, so entries above overwrite entries here
56
+ 'general' => array(
57
+ 'array' => array( 'desc'=> 'If you supply this as an argument, the returned data will be in an array, not an object (only useful wen using PHP, not shortcodes)', 'default'=>0),
58
+ 'format_header' => array( 'desc'=> sprintf('If you are displaying lists (e.g. listing events), you can supply the %s html and placeholders here.','header'), 'default'=> 'The relevant default format will be taken from the settings page.'),
59
+ 'format' => array( 'desc'=> 'If you are displaying some information with the shortcode or function (e.g. listing events), you can supply the html and placeholders here.', 'default'=> 'The relevant default format will be taken from the settings page.'),
60
+ 'format_footer' => array( 'desc'=> sprintf('If you are displaying lists (e.g. listing events), you can supply the %s html and placeholders here.','footer'), 'default'=> 'The relevant default format will be taken from the settings page.'),
61
+ 'limit' => array( 'desc'=> 'Limits the amount of values returned to this number.', 'default'=>'0 (no limit)'),
62
+ 'offset' => array( 'desc'=> 'For example, if you have ten results, if you set this to 5, only the last 5 results will be returned. Useful for pagination.', 'default'=>0),
63
+ 'order' => array( 'desc'=> 'Indicates the alphabeitcal/numerical order of the lists. Choose between ASC (ascending) and DESC (descending).', 'default'=>'ASC'),
64
+ 'orderby' => array( 'desc'=> 'Choose what fields to order your results by. You can supply a single field or multiple comma-seperated fields (e.g. "event_start_date,event_name").', 'default'=>0, 'args'=>'Database table fields, e.g. <code>event_name</code> or <code>location_name</code>'),
65
+ 'owner' => array('desc'=> 'Limits returned results to a specific owner, identified by their user id (e.g. list events or locations owned by user)', 'default'=>0),
66
+ 'pagination' => array('desc'=> 'When using a function or shortcode that outputs items (e.g. [events_list] for events, [locations_list] for locations), if the number of items supercede the limit of items to show, setting this to 1 will show page links under the list.', 'default'=>0),
67
+ )
68
+ ),
69
+ 'placeholders' => array(
70
+ 'events' => array(
71
+ 'Event Details' => array(
72
+ 'placeholders' => array(
73
+ '#_EVENTID' => array( 'desc' => 'Shows the event ID number in the wp_em_events table.' ),
74
+ '#_EVENTPOSTID' => array( 'desc' => 'Shows the event corresponding Post ID in the wp_posts table.' ),
75
+ '#_EVENTNAME' => array( 'desc' => 'Displays the name of the event.' ),
76
+ '#_EVENTNOTES' => array( 'desc' => 'Shows the description of the event.' ),
77
+ '#_EVENTEXCERPT' => array( 'desc' => 'If you added a <a href="http://en.support.wordpress.com/splitting-content/more-tag/">more tag</a> to your event description, only the content before this tag will show (currently, no read more link is added).' ),
78
+ '#_EVENTIMAGE' => array( 'desc' => 'Shows the event image, if available.' ),
79
+ '#_EVENTIMAGE{x,y}' => array( 'desc' => 'Shows the event image thumbnail, x and y are width and height respectively, both being numbers e.g. <code>#_EVENTIMAGE{100,100}</code>' ),
80
+ '#_EVENTCATEGORIES' => array( 'desc' => 'Shows a list of category links this event belongs to.' ),
81
+ '#_EVENTTAGS' => array( 'desc' => 'Shows a list of tag links this event belongs to.' ),
82
+ )
83
+ ),
84
+ 'Date and Times' => array(
85
+ 'desc' => '',
86
+ 'placeholders' => array(
87
+ '#_24HSTARTTIME' => array( 'desc' => 'Displays the start time in a 24 hours format (e.g. 16:30).' ),
88
+ '#_24HENDTIME' => array( 'desc' => 'Displays the end time in a 24 hours format (e.g. 18:30).' ),
89
+ '#_12HSTARTTIME' => array( 'desc' => 'Displays the start time in a 12 hours format (e.g. 4:30 PM).' ),
90
+ '#_12HENDTIME' => array( 'desc' => 'Displays the end time in a 12 hours format (e.g. 6:30 PM).' ),
91
+ '#_EVENTTIMES' => array( 'desc' => 'Displays either a single time, time-span, or "All Day" depending on your event times. Format is taken from your Events Manager settings page.' ),
92
+ '#_EVENTDATES' => array( 'desc' => 'Displays either a single date or a date range depending on your event dates. Format is taken from your Events Manager settings page.' ),
93
+ )
94
+ ),
95
+ 'Custom Date/Time Formatting' => array(
96
+ 'desc' => 'Events Manager allows extremely flexible date formatting by using <a href="http://www.php.net/manual/en/function.date.php">PHP date syntax format characters</a> along with placeholders.',
97
+ 'placeholders' => array(
98
+ '# or #@' => array( 'desc' => 'Prepend <code>#</code> or <code>#@</code> before a valid PHP date syntax format character to show start and end date/time information respectively (e.g. <code>#F</code> will show the starting month name like "January", #@h shows the end hour).' ),
99
+ '#{x} or #@{x}' => array( 'desc' => 'You can also create a date format without prepending # to each character by wrapping a valid php date() format in <code>#{}</code> or <code>#@{}</code> (e.g. <code>#_{d/m/Y}</code>). If there is no end date (or is same as start date), the value is not shown. This is useful if you want to show event end dates only on events that are longer than on day, e.g. <code>#j #M #Y #@_{ \u\n\t\i\l j M Y}</code>.' ),
100
+ )
101
+ ),
102
+ 'Links/URLs' => array(
103
+ 'placeholders' => array(
104
+ '#_EVENTIMAGEURL' => array( 'desc' => 'Shows the event image url, if available.' ),
105
+ '#_EVENTURL' => array( 'desc' => 'Simply prints the event URL. You can use this placeholder to build your own customised links.' ),
106
+ '#_EVENTLINK' => array( 'desc' => 'Displays the event name with a link to the event page.' ),
107
+ '#_EDITEVENTLINK' => array( 'desc' => 'Inserts a link to the admin or buddypress (if activated) edit event page, only if a user is logged in and is allowed to edit the event.' ),
108
+ '#_EDITEVENTURL' => array( 'desc' => 'Inserts a url to the admin or buddypress (if activated) edit event page, only if a user is logged in and is allowed to edit the event.' )
109
+ )
110
+ ),
111
+ 'Custom Attributes' => array(
112
+ 'desc' => 'Events Manager allows you to create dynamic attributes to your events, which act as extra information fields for your events (e.g. "Dress Code"). For more information see <a href="http://wp-events-plugin.com/documentation/categories-and-attributes/">our online documentation</a> for more info on attributes.',
113
+ 'placeholders' => array(
114
+ '#_ATT{key}' => array('desc'=> 'This key will appear as an option when adding attributes to your event.'),
115
+ '#_ATT{key}{alternative text}' => array('desc'=> 'This key will appear as an option when adding attributes to your event. The text in the second braces will appear if the attribute is not defined or left blank for that event.'),
116
+ '#_ATT{key}{option 1|option 2|option 3|etc.}' => array('desc'=> 'This key will appear as an option when adding attributes to your event. The second braces are optional and will use a select box with these values as input. If no valid value is defined, the first option is used.'),
117
+ )
118
+ ),
119
+ 'Bookings' => array(
120
+ 'desc' => 'These placeholders will only show if bookings are enabled for the given event and in the events manager settings page. Spaces placeholders will default to 0',
121
+ 'placeholders' => array(
122
+ '#_BOOKINGFORM' => array( 'desc' => 'Adds a booking forms for this event.' ),
123
+ '#_BOOKINGBUTTON' => array( 'desc' => 'A single button that will appear to logged in users, if they click on it, they apply for a booking. This button will only display if there is one ticket.' ),
124
+ '#_AVAILABLESPACES' => array( 'desc' => 'Shows available spaces for the event.' ),
125
+ '#_BOOKEDSPACES' => array( 'desc' => 'Shows the amount of currently booked spaces for the event.' ),
126
+ '#_PENDINGSPACES' => array( 'desc' => 'Shows the amount of pending spaces for the event.' ),
127
+ '#_SPACES' => array( 'desc' => 'Shows the total spaces for the event.' ),
128
+ '#_ATTENDEES' => array( 'desc' => 'Shows the list of user avatars attending events.' ),
129
+ '#_BOOKINGSURL' => array( 'desc' => 'Shows the url to the admin, front-end or buddypress (if activated) bookings management page for this event. Only shown if user is logged in and able to manage bookings.' ),
130
+ '#_BOOKINGSLINK' => array( 'desc' => 'Shows a link to the admin, front-end or buddypress (if activated) bookings management page for this event. Only shown if user is logged in and able to manage bookings.' )
131
+ )
132
+ ),
133
+ 'Contact Details' => array(
134
+ 'desc' => 'The values here are taken from the chosen contact for the specific event, or the default contact in the settings page.',
135
+ 'placeholders' => array(
136
+ '#_CONTACTNAME' => array( 'desc' => 'Name of the contact person for this event (as shown in the dropdown when adding an event).' ),
137
+ '#_CONTACTUSERNAME' => array( 'desc' => 'Contact person\'s username.' ),
138
+ '#_CONTACTEMAIL' => array( 'desc' => 'E-mail of the contact person for this event.' ),
139
+ '#_CONTACTPHONE' => array( 'desc' => 'Phone number of the contact person for this event. Can be set in the user profile page.' ),
140
+ '#_CONTACTAVATAR' => array( 'desc' => 'Contact person\'s avatar.' ),
141
+ '#_CONTACTPROFILELINK' => array( 'desc' => 'Contact person\'s "Profile" link. Only works with BuddyPress enabled.' ),
142
+ '#_CONTACTPROFILEURL' => array( 'desc' => 'Contact person\'s profile url. Only works with BuddyPress enabled.' ),
143
+ '#_CONTACTID' => array( 'desc' => 'Contact person\'s WordPress user ID.')
144
+ )
145
+ ),
146
+ ),
147
+ 'categories' => array(
148
+ 'Category Details' => array(
149
+ 'placeholders' => array(
150
+ '#_CATEGORYNAME' => array( 'desc' => 'Shows the category name of the event.' ),
151
+ '#_CATEGORYID' => array( 'desc' => 'Shows the category ID of the event.' ),
152
+ '#_CATEGORYIMAGE' => array( 'desc' => 'Shows the event image, if available.' ),
153
+ '#_CATEGORYIMAGE{x,y}' => array( 'desc' => 'Shows the category image thumbnail, x and y are width and height respectively, both being numbers e.g. <code>#_CATEGORYIMAGE{100,100}</code>' ),
154
+ '#_CATEGORYIMAGEURL' => array( 'desc' => 'Shows the category image url, if available.' ),
155
+ '#_CATEGORYNOTES' => array( 'desc' => 'Shows the location description.' )
156
+ )
157
+ ),
158
+ 'Related Events' => array(
159
+ 'desc' => 'You can show lists of other events belonging to this category. The formatting of the list is the same as a normal events list.',
160
+ 'placeholders' => array(
161
+ '#_CATEGORYPASTEVENTS' => array( 'desc' => 'Will show a list of all past events at this category.' ),
162
+ '#_CATEGORYNEXTEVENTS' => array( 'desc' => 'Will show a list of all future events at this category.' ),
163
+ '#_CATEGORYALLEVENTS' => array( 'desc' => 'Will show a list of all events at this category.' )
164
+ )
165
+ )
166
+ ),
167
+ 'locations' => array(
168
+ 'Location Details' => array(
169
+ 'desc' => '',
170
+ 'placeholders' => array(
171
+ '#_LOCATIONID' => array( 'desc' => 'Shows the event ID number in the wp_em_locations table.' ),
172
+ '#_LOCATIONPOSTID' => array( 'desc' => 'Shows the location corresponding Post ID in the wp_posts table.' ),
173
+ '#_LOCATIONNAME' => array( 'desc' => 'Displays the location name.' ),
174
+ '#_LOCATIONADDRESS' => array( 'desc' => 'Displays the address.' ),
175
+ '#_LOCATIONTOWN' => array( 'desc' => 'Displays the town.' ),
176
+ '#_LOCATIONSTATE' => array( 'desc' => 'Displays the state/county.' ),
177
+ '#_LOCATIONPOSTCODE' => array( 'desc' => 'Displays the postcode.' ),
178
+ '#_LOCATIONREGION' => array( 'desc' => 'Displays the region.' ),
179
+ '#_LOCATIONCOUNTRY' => array( 'desc' => 'Displays the country.' ),
180
+ '#_LOCATIONMAP' => array( 'desc' => 'Displays a google map showing where the event is located (Will not show if maps are disabled in the settings page)' ),
181
+ '#_LOCATIONNOTES' => array( 'desc' => 'Shows the location description.' ),
182
+ '#_LOCATIONEXCERPT' => array( 'desc' => 'If you added a <a href="http://en.support.wordpress.com/splitting-content/more-tag/">more tag</a> to your location description, only the content before this tag will show (currently, no read more link is added).' ),
183
+ '#_LOCATIONIMAGE' => array( 'desc' => 'Shows the location image.' ),
184
+ '#_LOCATIONIMAGE{x,y}' => array( 'desc' => 'Shows the location image thumbnail, x and y are width and height respectively, both being numbers e.g. <code>#_LOCATIONIMAGE{100,100}</code>' ),
185
+ '#_LOCATIONIMAGEURL' => array( 'desc' => 'Shows the cattegory image url, if available.' ),
186
+ '#_LOCATIONFULLLINE' => array( 'desc' => 'Shows a comma-seperated line of location information, ommitting blanks (format of address, town, state, postcode, region' ),
187
+ '#_LOCATIONFULLBR' => array( 'desc' => 'Shows the cattegory image url, if available.' ),
188
+ '#_LOCATIONFULLBR' => array( 'desc' => 'Shows a line-break (br tag) seperated location information, ommitting blanks (format of address, town, state, postcode, region' ),
189
+ )
190
+ ),
191
+ 'Links' => array(
192
+ 'placeholders' => array(
193
+ '#_LOCATIONURL' => array( 'desc' => 'Simply prints the location URL. You can use this placeholder to build your own customised links.' ),
194
+ '#_LOCATIONLINK' => array( 'desc' => 'Displays the location name with a link to the location page.' ),
195
+ '#_EDITLOCATIONLINK' => array( 'desc' => 'Inserts a link to the admin or buddypress (if activated) edit location page, only if a user is logged in and is allowed to edit the location.' ),
196
+ '#_EDITLOCATIONURL' => array( 'desc' => 'Inserts a url to the admin or buddypress (if activated) edit location page, only if a user is logged in and is allowed to edit the location.' )
197
+ )
198
+ ),
199
+ 'Related Events' => array(
200
+ 'desc' => 'You can show lists of other events that are being held at this location. The formatting of the list is the same as a normal events list.',
201
+ 'placeholders' => array(
202
+ '#_LOCATIONPASTEVENTS' => array( 'desc' => 'Will show a list of all past events at this location.' ),
203
+ '#_LOCATIONNEXTEVENTS' => array( 'desc' => 'Will show a list of all future events at this location.' ),
204
+ '#_LOCATIONALLEVENTS' => array( 'desc' => 'Will show a list of all events at this location.' )
205
+ )
206
+ ),
207
+ ),
208
+ 'bookings' => array(
209
+ 'Booking Person Information' => array(
210
+ 'desc' => 'When a specific booking is displayed (on screen and on email), you can use these placeholders to show specific information about the booking. Event and Location placeholders are also available in these cases.',
211
+ 'placeholders' => array(
212
+ '#_BOOKINGNAME' => array( 'desc' => 'Name of person who made the booking.' ),
213
+ '#_BOOKINGEMAIL' => array( 'desc' => 'Email of person who made the booking.' ),
214
+ '#_BOOKINGPHONE' => array( 'desc' => 'Phone number of person who made the booking.' ),
215
+ '#_BOOKINGSPACES' => array( 'desc' => 'Number of spaces the person has booked.' ),
216
+ '#_BOOKINGCOMMENT' => array( 'desc' => 'Any specific comments made by the person who made the booking.' ),
217
+ '#_BOOKINGTICKETNAME' => array( 'desc' => 'Name of the ticket booked. Useful in single ticket mode, if multiple tickets are booked a random ticket is used.' ),
218
+ '#_BOOKINGTICKETDESCRIPTION' => array( 'desc' => 'Description of the ticket booked. Useful in single ticket mode, if multiple tickets are booked a random ticket is used.' ),
219
+ '#_BOOKINGTICKETPRICE' => array( 'desc' => 'Booked ticket price with currency symbol (e.g. $ 10.00). Useful in single ticket mode, if multiple tickets are booked a random ticket is used.' ),
220
+ '#_BOOKINGTICKETS' => array( 'desc' => 'A list of booked tickets. You can modify this by using template files and modifying templates/emails/bookingtickets.php' ),
221
+ '#_BOOKINGFORMCUSTOM{field_id}' => array( 'desc' => sprintf('(<a href="%s">pro only</a>) Shows booking form custom fields. The field_id value must match that of your custom booking form field.','http://wp-events-plugin.com/upgrade/') ),
222
+ '#_BOOKINGFORMCUSTOMREG{field_id}' => array( 'desc' => sprintf('(<a href="%s">pro only</a>) Shows booking form custom fields that are used for guest user registration. The field_id value must match that of your custom booking form field.','http://wp-events-plugin.com/upgrade/') )
223
+ )
224
+ ),
225
+ 'Pricing Information' => array(
226
+ 'desc' => '',
227
+ 'placeholders' => array(
228
+ '#_BOOKINGPRICE' => array( 'desc' => 'Displays booking total price (tax inclusion depends on your booking settings).' ),
229
+ '#_BOOKINGPRICETAX' => array( 'desc' => 'Displays booking total tax.' ),
230
+ '#_BOOKINGPRICEWITHOUTTAX' => array( 'desc' => 'Displays booking total without tax.' ),
231
+ '#_BOOKINGPRICEWITHTAX' => array( 'desc' => 'Displays booking total with tax.' ),
232
+ )
233
+ ),
234
+ 'Ticket Information' => array(
235
+ 'desc' => '',
236
+ 'placeholders' => array(
237
+ '#_BOOKINGTICKETS' => array( 'desc' => 'Shows a breakdown of tickets and pricing, defined in the <code>emails/bookingtickets.php</code> template. (See <a href="http://wp-events-plugin.com/documentation/using-template-files/">Using Template Files</a> for more information)' ),
238
+ '#_BOOKINGTICKETDESCRIPTION' => array( 'desc' => 'Shows the description of the first ticket booked (useful in single ticket mode/events).' ),
239
+ '#_BOOKINGTICKETPRICE' => array( 'desc' => 'Shows the price of the first ticket booked, tax inclusion depending on your booking settings (useful in single ticket mode/events).' ),
240
+ '#_BOOKINGTICKETTAX' => array( 'desc' => 'Shows the tax of the first ticket booked (useful in single ticket mode/events).' ),
241
+ '#_BOOKINGTICKETPRICEWITHTAX' => array( 'desc' => 'Shows the price including tax of the first ticket booked (useful in single ticket mode/events).' ),
242
+ '#_BOOKINGTICKETPRICEWITHOUTTAX' => array( 'desc' => 'Shows the price excluding tax of the first ticket booked (useful in single ticket mode/events).' ),
243
+ )
244
+ ),
245
+ 'Links' => array(
246
+ 'desc' => 'People are able to manage their bookings. Below are some placeholder which automatically provides correctly formatted urls',
247
+ 'placeholders' => array(
248
+ '#_BOOKINGLISTURL' => array( 'desc' => 'URL to page showing that users booked events.' )
249
+ )
250
+ )
251
+ ),
252
+ ),
253
+ //TODO add capabilites explanations
254
+ 'capabilities' => array()
255
+ );
256
+ }
257
+ }
258
+ add_action('init', 'em_docs_init');
259
+
260
+ function em_docs_placeholders($atts){
261
+ ob_start();
262
+ ?>
263
+ <div class="em-docs">
264
+ <?php
265
+ global $EM_Documentation;
266
+ $type = $atts['type'];
267
+ $data = $EM_Documentation['placeholders'][$type];
268
+ foreach($data as $sectionTitle => $details) : ?>
269
+ <div>
270
+ <h3><?php echo $sectionTitle; ?></h3>
271
+ <?php if( !empty($details['desc']) ): ?>
272
+ <p><?php echo $details['desc']; ?></p>
273
+ <?php endif; ?>
274
+ <dl>
275
+ <?php foreach($details['placeholders'] as $placeholder => $desc ): ?>
276
+ <dt><b><?php echo $placeholder; ?></b></dt>
277
+ <dd><?php echo $desc['desc']; ?></dd>
278
+ <?php endforeach; ?>
279
+ </dl>
280
+ </div>
281
+ <?php endforeach; ?>
282
+ </div>
283
+ <?php
284
+ return ob_get_clean();
285
+ }
286
  ?>
admin/em-event.php DELETED
@@ -1,627 +0,0 @@
1
- <?php
2
- function em_admin_event_actions(){
3
- if( current_user_can(EM_MIN_CAPABILITY) && !empty($_GET['page']) && $_GET['page'] == 'events-manager-event' && !empty($_REQUEST ['action']) ){
4
- global $wpdb;
5
- global $EM_Event;
6
-
7
- //if dealing with new event, we still want an event object
8
- if( !is_object($EM_Event) ){
9
- $EM_Event = new EM_Event();
10
- }
11
-
12
- // UPDATE or CREATE action
13
- if ($_REQUEST['action'] == 'save') {
14
- $validation = $EM_Event->get_post();
15
- if ( $validation ) { //EM_Event gets the event if submitted via POST and validates it (safer than to depend on JS)
16
- //Save
17
- if( $EM_Event->save() ) {
18
- $page = !empty($_REQUEST['pno']) ? $_REQUEST['pno']:'';
19
- $scope = !empty($_REQUEST['scope']) ? $_REQUEST['scope']:'';
20
- wp_redirect( get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager&p='.$page.'&scope='.$scope.'&message='.urlencode($EM_Event->feedback_message));
21
- }
22
- }//errors added automatically to event global object
23
- }
24
-
25
- //Copy the event
26
- if ($_REQUEST['action'] == 'duplicate') {
27
- global $EZSQL_ERROR;
28
- $EM_Event = $EM_Event->duplicate();
29
- if( $EM_Event === false ){
30
- $redirect_url = em_add_get_params($_SERVER['HTTP_REFERER'], array('error' => __('There was an error duplicating the event. Try again maybe?', 'dbem'), 'message'=>''), false);
31
- wp_redirect($redirect_url);
32
- }else{
33
- $page = !empty($_REQUEST['pno']) ? $_REQUEST['pno']:'';
34
- $scope = !empty($_REQUEST['scope']) ? $_REQUEST['scope']:'';
35
- wp_redirect( get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager-event&event_id='.$EM_Event->id.'&p='.$page.'&scope='.$scope.'&message='.urlencode($EM_Event->feedback_message));
36
- }
37
- }
38
- }
39
- }
40
- add_action('admin_init', 'em_admin_event_actions');
41
-
42
- /**
43
- * Generates Event Admin page, for adding and updating a single (or recurring) event.
44
- * @param $title
45
- * @return null
46
- */
47
- function em_admin_event_page() {
48
- global $EM_Event, $current_user;
49
- global $localised_date_formats;
50
-
51
- //check that user can access this page
52
- if( is_object($EM_Event) && !$EM_Event->can_manage() ){
53
- ?>
54
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
55
- <?php
56
- return false;
57
- }
58
-
59
- if( is_object($EM_Event) && $EM_Event->id > 0 ){
60
- if($EM_Event->is_recurring()){
61
- $title = __( "Reschedule", 'dbem' )." '{$EM_Event->name}'";
62
- }else{
63
- $title = __ ( "Edit Event", 'dbem' ) . " '" . $EM_Event->name . "'";
64
- }
65
- } else {
66
- $EM_Event = ( is_object($EM_Event) && get_class($EM_Event) == 'EM_Event') ? $EM_Event : new EM_Event();
67
- $title = __ ( "Insert New Event", 'dbem' );
68
- //Give a default location & category
69
- $default_cat = get_option('dbem_default_category');
70
- $default_loc = get_option('dbem_default_location');
71
- if( is_numeric($default_cat) && $default_cat > 0 ){
72
- $EM_Event->category_id = $default_cat;
73
- $EM_Event->category = new EM_Category($default_cat);
74
- }
75
- if( is_numeric($default_loc) && $default_loc > 0 && ( empty($EM_Event->location->id) && empty($EM_Event->location->name) && empty($EM_Event->location->address) && empty($EM_Event->location->town) ) ){
76
- $EM_Event->location_id = $default_loc;
77
- $EM_Event->location = new EM_Location($default_loc);
78
- }
79
- }
80
-
81
- $use_select_for_locations = get_option('dbem_use_select_for_locations');
82
- // change prefix according to event/recurrence
83
- $pref = "event_";
84
-
85
- $locale_code = substr ( get_locale (), 0, 2 );
86
- $localised_date_format = $localised_date_formats[$locale_code];
87
-
88
- //FIXME time useage is very flimsy imho
89
- $hours_locale_regexp = "H:i";
90
- // Setting 12 hours format for those countries using it
91
- if (preg_match ( "/en|sk|zh|us|uk/", $locale_code ))
92
- $hours_locale_regexp = "h:iA";
93
-
94
- $days_names = array (1 => __ ( 'Mon' ), 2 => __ ( 'Tue' ), 3 => __ ( 'Wed' ), 4 => __ ( 'Thu' ), 5 => __ ( 'Fri' ), 6 => __ ( 'Sat' ), 0 => __ ( 'Sun' ) );
95
- ?>
96
- <?php if ( count($EM_Event->errors) > 0 || !empty($_GET['error']) ) : ?>
97
- <div id='message' class='error '>
98
- <p>
99
- <?php if( count($EM_Event->errors) ){ ?>
100
- <strong><?php echo __( "Ach, there's a problem here:", "dbem" ) ?></strong><br /><br />
101
- <?php echo implode('<br />', $EM_Event->errors); ?>
102
- <?php } else { echo $_GET['error']; } ?>
103
- </p>
104
- </div>
105
- <?php endif; ?>
106
- <?php if ( !empty($EM_Event->feedback_message) || !empty($_GET['message']) ) : ?>
107
- <div id='message' class='updated fade'>
108
- <p><?php echo !empty($EM_Event->feedback_message) ? $EM_Event->feedback_message : $_GET['message']; ?></p>
109
- </div>
110
- <?php endif; ?>
111
- <form id="event-form" method="post" action="">
112
- <div class="wrap">
113
- <div id="icon-events" class="icon32"><br /></div>
114
- <h2><?php echo $title; ?></h2>
115
- <?php if ( $EM_Event->is_recurrence() || $EM_Event->is_recurring() ) : ?>
116
- <p id='recurrence_warning'>
117
- <?php
118
- //TODO better warning system when changing a recurring event (e.g. when removing recurrences).
119
- if ( $EM_Event->is_recurring() ) {
120
- _e ( 'WARNING: This is a recurring event.', 'dbem' );
121
- echo "<br />";
122
- _e ( 'Modifying these data all the events linked to this recurrence will be rescheduled', 'dbem' );
123
- echo " ";
124
- _e ( 'and all booking information will be deleted!', 'dbem' );
125
- } elseif ( $EM_Event->is_recurrence() ) {
126
- //TODO Terminology confusing with methods, maybe worth changing?
127
- _e ( 'WARNING: This is a recurrence.', 'dbem' );
128
- echo "<br />";
129
- _e ( 'If you change these data and save, this will become an independent event.', 'dbem' );
130
- }
131
- ?>
132
- </p>
133
- <?php endif; ?>
134
- <div id="poststuff" class="metabox-holder has-right-sidebar">
135
- <!-- SIDEBAR -->
136
- <div id="side-info-column" class='inner-sidebar'>
137
- <div id='side-sortables'>
138
- <?php if(get_option('dbem_recurrence_enabled')) : ?>
139
- <!-- START recurrence postbox -->
140
- <div class="postbox ">
141
- <div class="handlediv" title="Fare clic per cambiare."><br />
142
- </div>
143
- <h3 class='hndle'><span>
144
- <?php _e ( "Recurrence", 'dbem' ); ?>
145
- </span></h3>
146
- <div class="inside">
147
- <?php //TODO add js warning if rescheduling, since all bookings are deleted ?>
148
- <?php if ( !$EM_Event->id || $EM_Event->is_recurring() ) : ?>
149
- <p>
150
- <input id="event-recurrence" type="checkbox" name="repeated_event" value="1" <?php echo ( $EM_Event->is_recurring() ) ? 'checked="checked"':'' ; ?> />
151
- <?php _e ( 'Repeated event', 'dbem' ); ?>
152
- </p>
153
- <div id="event_recurrence_pattern">
154
- <p>
155
- Frequency:
156
- <select id="recurrence-frequency" name="recurrence_freq">
157
- <?php
158
- $freq_options = array ("daily" => __ ( 'Daily', 'dbem' ), "weekly" => __ ( 'Weekly', 'dbem' ), "monthly" => __ ( 'Monthly', 'dbem' ) );
159
- em_option_items ( $freq_options, $EM_Event->freq );
160
- ?>
161
- </select>
162
- </p>
163
- <p>
164
- <?php _e ( 'Every', 'dbem' )?>
165
- <input id="recurrence-interval" name='recurrence_interval' size='2' value='<?php echo $EM_Event->interval ; ?>' />
166
- <span class='interval-desc' id="interval-daily-singular">
167
- <?php _e ( 'day', 'dbem' )?>
168
- </span> <span class='interval-desc' id="interval-daily-plural">
169
- <?php _e ( 'days', 'dbem' ) ?>
170
- </span> <span class='interval-desc' id="interval-weekly-singular">
171
- <?php _e ( 'week', 'dbem' )?>
172
- </span> <span class='interval-desc' id="interval-weekly-plural">
173
- <?php _e ( 'weeks', 'dbem' )?>
174
- </span> <span class='interval-desc' id="interval-monthly-singular">
175
- <?php _e ( 'month', 'dbem' )?>
176
- </span> <span class='interval-desc' id="interval-monthly-plural">
177
- <?php _e ( 'months', 'dbem' )?>
178
- </span>
179
- </p>
180
- <p class="alternate-selector" id="weekly-selector">
181
- <?php
182
- $saved_bydays = ($EM_Event->is_recurring()) ? explode ( ",", $EM_Event->byday ) : array();
183
- em_checkbox_items ( 'recurrence_bydays[]', $days_names, $saved_bydays );
184
- ?>
185
- </p>
186
- <p class="alternate-selector" id="monthly-selector">
187
- <?php _e ( 'Every', 'dbem' )?>
188
- <select id="monthly-modifier" name="recurrence_byweekno">
189
- <?php
190
- $weekno_options = array ("1" => __ ( 'first', 'dbem' ), '2' => __ ( 'second', 'dbem' ), '3' => __ ( 'third', 'dbem' ), '4' => __ ( 'fourth', 'dbem' ), '-1' => __ ( 'last', 'dbem' ) );
191
- em_option_items ( $weekno_options, $EM_Event->byweekno );
192
- ?>
193
- </select>
194
- <select id="recurrence-weekday" name="recurrence_byday">
195
- <?php em_option_items ( $days_names, $EM_Event->byday ); ?>
196
- </select>
197
- &nbsp;
198
- </p>
199
- </div>
200
- <p id="recurrence-tip">
201
- <?php _e ( 'Check if your event happens more than once according to a regular pattern', 'dbem' )?>
202
- </p>
203
- <?php elseif( $EM_Event->is_recurrence() ) : ?>
204
- <p>
205
- <?php echo $EM_Event->get_recurrence_description(); ?>
206
- <br />
207
- <a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-event&amp;event_id=<?php echo $EM_Event->recurrence_id; ?>">
208
- <?php _e ( 'Reschedule', 'dbem' ); ?>
209
- </a>
210
- <input type="hidden" name="recurrence_id" value="<?php echo $EM_Event->recurrence_id; ?>" />
211
- </p>
212
- <?php else : ?>
213
- <p><?php _e ( 'This is\'t a recurrent event', 'dbem' ) ?></p>
214
- <?php endif; ?>
215
- </div>
216
- </div>
217
- <!-- END recurrence postbox -->
218
- <?php endif; ?>
219
- <?php if(get_option('dbem_rsvp_enabled')) : ?>
220
- <!-- START RSVP -->
221
- <?php if ( em_verify_admin() ): ?>
222
- <div class="postbox ">
223
- <div class="handlediv" title="Fare clic per cambiare."><br />
224
- </div>
225
- <h3 class='hndle'><span>
226
- <?php _e ( 'Contact Person', 'dbem' ); ?>
227
- </span></h3>
228
- <div class="inside">
229
- <p><?php _e('Contact','dbem'); ?>
230
- <?php
231
- wp_dropdown_users ( array ('name' => 'event_contactperson_id', 'show_option_none' => __ ( "Select...", 'dbem' ), 'selected' => $EM_Event->contactperson_id ) );
232
- ?>
233
- </p>
234
- </div>
235
- </div>
236
- <?php else: ?>
237
- <input type="hidden" name="event_contactperson_id" value="<?php get_current_user_id() ?>" />
238
- <?php endif; ?>
239
- <div class="postbox ">
240
- <div class="handlediv" title="Fare clic per cambiare."><br />
241
- </div>
242
- <h3 class='hndle'><span><?php _e('RSVP','dbem'); ?></span></h3>
243
- <div class="inside">
244
- <p>
245
- <input id="rsvp-checkbox" name='event_rsvp' value='1' type='checkbox' <?php echo ($EM_Event->rsvp) ? 'checked="checked"' : ''; ?> />
246
- <?php _e ( 'Enable registration for this event', 'dbem' )?>
247
- </p>
248
- <div id='rsvp-data'>
249
- <?php
250
- if ($EM_Event->contactperson_id != NULL){
251
- $selected = $EM_Event->contactperson_id;
252
- }else{
253
- $selected = '0';
254
- }
255
- ?>
256
- <p>
257
- <?php _e ( 'Spaces','dbem' ); ?> :
258
- <input id="seats-input" type="text" name="event_seats" size='5' value="<?php echo $EM_Event->seats ?>" />
259
- </p>
260
- <!-- START RSVP Stats -->
261
- <?php
262
- if ($EM_Event->rsvp ) {
263
- $available_seats = $EM_Event->get_bookings()->get_available_seats();
264
- $booked_seats = $EM_Event->get_bookings()->get_booked_seats();
265
-
266
- if ( count($EM_Event->get_bookings()->bookings) > 0 ) {
267
- ?>
268
- <div class='wrap'>
269
- <p><strong><?php echo __('Available Spaces','dbem').': '.$EM_Event->get_bookings()->get_available_seats(); ?></strong></p>
270
- <p><strong><?php echo __('Confirmed Spaces','dbem').': '.$EM_Event->get_bookings()->get_booked_seats(); ?></strong></p>
271
- <p><strong><?php echo __('Pending Spaces','dbem').': '.$EM_Event->get_bookings()->get_pending_seats(); ?></strong></p>
272
- </div>
273
-
274
- <br class='clear'/>
275
-
276
- <div id='major-publishing-actions'>
277
- <div id='publishing-action'>
278
- <a id='printable' href='<?php echo get_bloginfo('wpurl') . "/wp-admin/admin.php?page=events-manager-bookings&event_id=".$EM_Event->id ?>'><?php _e('manage bookings','dbem')?></a><br />
279
- <a target='_blank' href='<?php echo get_bloginfo('wpurl') . "/wp-admin/admin.php?page=events-manager-bookings&action=bookings_report&event_id=".$EM_Event->id ?>'><?php _e('printable view','dbem')?></a>
280
- <a href='<?php echo get_bloginfo('wpurl') . "/wp-admin/admin.php?page=events-manager-bookings&action=export_csv&event_id=".$EM_Event->id ?>'><?php _e('export csv','dbem')?></a>
281
- <br class='clear'/>
282
- </div>
283
- <br class='clear'/>
284
- </div>
285
- <?php
286
- } else {
287
- ?>
288
- <p><em><?php _e('No responses yet!')?></em></p>
289
- <?php
290
- }
291
- }
292
- ?>
293
- <!-- END RSVP Stats -->
294
- </div>
295
- </div>
296
- </div>
297
- <!-- END RSVP -->
298
- <?php endif; ?>
299
- <?php if(get_option('dbem_categories_enabled')) :?>
300
- <!-- START Categories -->
301
- <div class="postbox ">
302
- <div class="handlediv" title="Fare clic per cambiare."><br />
303
- </div>
304
- <h3 class='hndle'><span>
305
- <?php _e ( 'Category', 'dbem' ); ?>
306
- </span></h3>
307
- <div class="inside">
308
- <?php $categories = EM_Categories::get(array('orderby'=>'category_name')); ?>
309
- <?php if( count($categories) > 0 ): ?>
310
- <p><?php _e ( 'Category:', 'dbem' ); ?>
311
- <select name="event_category_id">
312
- <option value="" <?php echo ($EM_Event->category_id == '') ? "selected='selected'":'' ?>><?php _e('no category','dbem') ?></option>
313
- <?php
314
- foreach ( $categories as $EM_Category ){
315
- $selected = ($EM_Category->id == $EM_Event->category_id) ? "selected='selected'": '';
316
- ?>
317
- <option value="<?php echo $EM_Category->id ?>" <?php echo $selected ?>>
318
- <?php echo $EM_Category->name ?>
319
- </option>
320
- <?php
321
- }
322
- ?>
323
- </select>
324
- </p>
325
- <?php else: ?>
326
- <p><?php sprintf(__('No categories available, <a href="%s">create one here first</a>','dbem'), get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager-categories'); ?></p>
327
- <?php endif; ?>
328
- </div>
329
- </div>
330
- <!-- END Categories -->
331
- <?php endif; ?>
332
- </div>
333
- </div>
334
- <!-- END OF SIDEBAR -->
335
- <div id="post-body">
336
- <div id="post-body-content">
337
- <div id="event_name" class="stuffbox">
338
- <h3>
339
- <?php _e ( 'Name', 'dbem' ); ?>
340
- </h3>
341
- <div class="inside">
342
- <input type="text" name="event_name" id="event-name" value="<?php echo htmlspecialchars($EM_Event->name,ENT_QUOTES); ?>" />
343
- <br />
344
- <?php _e ( 'The event name. Example: Birthday party', 'dbem' )?>
345
- </div>
346
- </div>
347
- <div id="event_start_date" class="stuffbox">
348
- <h3 id='event-date-title'>
349
- <?php _e ( 'Event date', 'dbem' ); ?>
350
- </h3>
351
- <h3 id='recurrence-dates-title'>
352
- <?php _e ( 'Recurrence dates', 'dbem' ); ?>
353
- </h3>
354
- <div class="inside">
355
- <input id="localised-date" type="text" name="localised_event_date" style="display: none;" />
356
- <input id="date-to-submit" type="text" name="event_start_date" value="<?php echo $EM_Event->start_date ?>" style="background: #FCFFAA" />
357
- <input id="localised-end-date" type="text" name="localised_event_end_date" style="display: none;" />
358
- <input id="end-date-to-submit" type="text" name="event_end_date" value="<?php echo $EM_Event->end_date ?>" style="background: #FCFFAA" />
359
- <br />
360
- <span id='event-date-explanation'>
361
- <?php
362
- _e ( 'The event date.', 'dbem' );
363
- /* Marcus Begin Edit */
364
- echo " ";
365
- _e ( 'When not reoccurring, this event spans between the beginning and end date.', 'dbem' );
366
- /* Marcus End Edit */
367
- ?>
368
- </span>
369
- <span id='recurrence-dates-explanation'>
370
- <?php _e ( 'The recurrence beginning and end date.', 'dbem' ); ?>
371
- </span>
372
- </div>
373
- </div>
374
- <div id="event_end_day" class="stuffbox">
375
- <h3>
376
- <?php _e ( 'Event time', 'dbem' ); ?>
377
- </h3>
378
- <div class="inside">
379
- <input id="start-time" type="text" size="8" maxlength="8" name="event_start_time" value="<?php echo date( $hours_locale_regexp, strtotime($EM_Event->start_time) ); ?>" />
380
- -
381
- <input id="end-time" type="text" size="8" maxlength="8" name="event_end_time" value="<?php echo date( $hours_locale_regexp, strtotime($EM_Event->end_time) ); ?>" />
382
- <br />
383
- <?php _e ( 'The time of the event beginning and end', 'dbem' )?>.
384
- </div>
385
- </div>
386
- <div id="location_coordinates" class="stuffbox" style='display: none;'>
387
- <h3>
388
- <?php _e ( 'Coordinates', 'dbem' ); ?>
389
- </h3>
390
- <div class="inside">
391
- <input id='location-latitude' name='location_latitude' type='text' value='<?php echo $EM_Event->latitude; ?>' size='15' />
392
- -
393
- <input id='location-longitude' name='location_longitude' type='text' value='<?php echo $EM_Event->longitude; ?>' size='15' />
394
- </div>
395
- </div>
396
- <div id="location_info" class="stuffbox">
397
- <h3>
398
- <?php _e ( 'Location', 'dbem' ); ?>
399
- </h3>
400
- <div class="inside">
401
- <table id="dbem-location-data">
402
- <tr>
403
- <td style="padding-right:20px">
404
- <table>
405
- <?php if($use_select_for_locations) : ?>
406
- <tr>
407
- <th><?php _e('Location:','dbem') ?></th>
408
- <td>
409
- <select name="location-select-id" id='location-select-id' size="1">
410
- <?php
411
- $locations = EM_Locations::get();
412
- foreach($locations as $location) {
413
- $selected = "";
414
- if( is_object($EM_Event->location) ) {
415
- if ($EM_Event->location->id == $location->id)
416
- $selected = "selected='selected' ";
417
- }
418
- ?>
419
- <option value="<?php echo $location->id ?>" title="<?php echo "{$location->latitude},{$location->longitude}" ?>" <?php echo $selected ?>><?php echo $location->name; ?></option>
420
- <?php
421
- }
422
- ?>
423
- </select>
424
- <p><?php _e ( 'The name of the location where the event takes place. You can use the name of a venue, a square, etc', 'dbem' )?></p>
425
- </td>
426
- </tr>
427
- <?php else : ?>
428
- <tr>
429
- <th><?php _e ( 'Name:' )?></th>
430
- <td>
431
- <input id="location-name" type="text" name="location_name" value="<?php echo htmlspecialchars($EM_Event->location->name, ENT_QUOTES); ?>" />
432
- <p><?php _e ( 'Select a location for your event', 'dbem' )?></p>
433
- </td>
434
- </tr>
435
- <tr>
436
- <th><?php _e ( 'Address:' )?>&nbsp;</th>
437
- <td>
438
- <input id="location-address" type="text" name="location_address" value="<?php echo htmlspecialchars($EM_Event->location->address, ENT_QUOTES); ; ?>" />
439
- <p><?php _e ( 'The address of the location where the event takes place. Example: 21, Dominick Street', 'dbem' )?></p>
440
- </td>
441
- </tr>
442
- <tr>
443
- <th><?php _e ( 'Town:' )?>&nbsp;</th>
444
- <td>
445
- <input id="location-town" type="text" name="location_town" value="<?php echo htmlspecialchars($EM_Event->location->town, ENT_QUOTES); ?>" />
446
- <p><?php _e ( 'The town where the location is located. If you\'re using the Google Map integration and want to avoid geotagging ambiguities include the country in the town field. Example: Verona, Italy.', 'dbem' )?></p>
447
- </td>
448
- </tr>
449
- <?php endif; ?>
450
- </table>
451
- </td>
452
- <?php if ( get_option ( 'dbem_gmap_is_active' ) ) : ?>
453
- <td width="400">
454
- <div id='em-map-404' style='width: 400px; vertical-align:middle; text-align: center;'>
455
- <p><em><?php _e ( 'Location not found', 'dbem' ); ?></em></p>
456
- </div>
457
- <div id='em-map' style='width: 400px; height: 300px; display: none;'></div>
458
- </td>
459
- <?php endif; ?>
460
- </tr>
461
- </table>
462
- </div>
463
- </div>
464
- <div id="event_notes" class="postbox">
465
- <h3>
466
- <?php _e ( 'Details', 'dbem' ); ?>
467
- </h3>
468
- <div class="inside">
469
- <div id="<?php echo user_can_richedit() ? 'postdivrich' : 'postdiv'; ?>" class="postarea">
470
- <?php the_editor($EM_Event->notes ); ?>
471
- </div>
472
- <br />
473
- <?php _e ( 'Details about the event', 'dbem' )?>
474
- </div>
475
- </div>
476
-
477
- <?php if(get_option('dbem_attributes_enabled')) : ?>
478
- <div id="event_attributes" class="postbox">
479
- <h3>
480
- <?php _e ( 'Attributes', 'dbem' ); ?>
481
- </h3>
482
- <div class="inside">
483
- <?php
484
- //We also get a list of attribute names and create a ddm list (since placeholders are fixed)
485
- $formats =
486
- get_option ( 'dbem_event_list_item_format' ).
487
- get_option ( 'dbem_event_page_title_format' ).
488
- get_option ( 'dbem_full_calendar_event_format' ).
489
- get_option ( 'dbem_location_baloon_format' ).
490
- get_option ( 'dbem_location_event_list_item_format' ).
491
- get_option ( 'dbem_location_page_title_format' ).
492
- get_option ( 'dbem_map_text_format' ).
493
- get_option ( 'dbem_rss_description_format' ).
494
- get_option ( 'dbem_rss_title_format' ).
495
- get_option ( 'dbem_single_event_format' ).
496
- get_option ( 'dbem_single_location_format' ).
497
- get_option ( 'dbem_placeholders_custom' );
498
- //We now have one long string of formats, get all the attribute placeholders
499
- preg_match_all('/#_ATT\{.+?\}(\{.+?\})?/', $formats, $placeholders);
500
- //Now grab all the unique attributes we can use in our event.
501
- $attributes = array();
502
- foreach($placeholders[0] as $result) {
503
- $attribute = substr( substr($result, 0, strpos($result, '}')), 6 );
504
- if( !in_array($attribute, $attributes) ){
505
- $attributes[] = $attribute ;
506
- }
507
- }
508
- ?>
509
- <div class="wrap">
510
- <?php if( count( $attributes ) > 0 ) : ?>
511
- <h2>Attributes</h2>
512
- <p>Add attributes here</p>
513
- <table class="form-table">
514
- <thead>
515
- <tr valign="top">
516
- <td><strong>Attribute Name</strong></td>
517
- <td><strong>Value</strong></td>
518
- </tr>
519
- </thead>
520
- <tfoot>
521
- <tr valign="top">
522
- <td colspan="3"><a href="#" id="mtm_add_tag">Add new tag</a></td>
523
- </tr>
524
- </tfoot>
525
- <tbody id="mtm_body">
526
- <?php
527
- $count = 1;
528
- if( is_array($EM_Event->attributes) and count($EM_Event->attributes) > 0){
529
- foreach( $EM_Event->attributes as $name => $value){
530
- ?>
531
- <tr valign="top" id="mtm_<?php echo $count ?>">
532
- <td scope="row">
533
- <select name="mtm_<?php echo $count ?>_ref">
534
- <?php
535
- if( !in_array($name, $attributes) ){
536
- echo "<option value='$name'>$name (".__('Not defined in templates', 'dbem').")</option>";
537
- }
538
- foreach( $attributes as $attribute ){
539
- if( $attribute == $name ) {
540
- echo "<option selected='selected'>$attribute</option>";
541
- }else{
542
- echo "<option>$attribute</option>";
543
- }
544
- }
545
- ?>
546
- </select>
547
- <a href="#" rel="<?php echo $count ?>">Remove</a>
548
- </td>
549
- <td>
550
- <input type="text" name="mtm_<?php echo $count ?>_name" value="<?php echo htmlspecialchars($value, ENT_QUOTES); ?>" />
551
- </td>
552
- </tr>
553
- <?php
554
- $count++;
555
- }
556
- }else{
557
- ?>
558
- <tr valign="top" id="mtm_<?php echo $count ?>">
559
- <td scope="row">
560
- <select name="mtm_<?php echo $count ?>_ref">
561
- <?php
562
- foreach( $attributes as $attribute ){
563
- echo "<option>$attribute</option>";
564
- }
565
- ?>
566
- </select>
567
- <a href="#" rel="<?php echo $count ?>">Remove</a>
568
- </td>
569
- <td>
570
- <input type="text" name="mtm_<?php echo $count ?>_name" />
571
- </td>
572
- </tr>
573
- <?php
574
- }
575
- ?>
576
- </tbody>
577
- </table>
578
- <?php else : ?>
579
- <p>
580
- <?php _e('In order to use attributes, you must define some in your templates, otherwise they\'ll never show. Go to Events > Settings to add attribute placeholders.', 'dbem'); ?>
581
- </p>
582
- <script>
583
- jQuery(document).ready(function($){ $('#event_attributes').addClass('closed'); });
584
- </script>
585
- <?php endif; ?>
586
- </div>
587
- </div>
588
- </div>
589
- <?php endif; ?>
590
- </div>
591
- <p class="submit">
592
- <input type="submit" name="events_update" value="<?php _e ( 'Submit Event', 'dbem' ); ?> &raquo;" />
593
- </p>
594
- <input type="hidden" name="p" value="<?php echo ( !empty($_REQUEST['pno']) ) ? $_REQUEST['pno']:''; ?>" /><a>
595
- <input type="hidden" name="scope" value="<?php echo ( !empty($_REQUEST['scope']) ) ? $_REQUEST['scope']:'' ?>" /></a>
596
- <input type="hidden" name="action" value="save" />
597
- </div>
598
- </div>
599
- </div>
600
- </form>
601
- <script type="text/javascript">
602
- jQuery(document).ready( function($) {
603
- <?php if( $EM_Event->is_recurring() ): ?>
604
- //Recurrence Warnings
605
- $('#event_form').submit( function(event){
606
- confirmation = confirm('<?php _e('Are you sure you want to reschedule this recurring event? If you do this, you will lose all booking information and the old recurring events will be deleted.', 'dbem'); ?>');
607
- if( confirmation == false ){
608
- event.preventDefault();
609
- }
610
- });
611
- <?php endif; ?>
612
- <?php if( $EM_Event->rsvp == 1 ): ?>
613
- //RSVP Warning
614
- $('#rsvp-checkbox').click( function(event){
615
- if( !this.checked ){
616
- confirmation = confirm('<?php _e('Are you sure you want to disable bookings? If you do this and save, you will lose all previous bookings. If you wish to prevent further bookings, reduce the number of seats available to the amount of bookings you currently have', 'dbem'); ?>');
617
- if( confirmation == false ){
618
- event.preventDefault();
619
- }
620
- }
621
- });
622
- <?php endif; ?>
623
- });
624
- </script>
625
- <?php
626
- }
627
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
admin/em-events.php DELETED
@@ -1,242 +0,0 @@
1
- <?php
2
-
3
- /**
4
- * Determines whether to show event page or events page, and saves any updates to the event or events
5
- * @return null
6
- */
7
- function em_admin_events_page() {
8
- //TODO Simplify panel for events, use form flags to detect certain actions (e.g. submitted, etc)
9
- global $wpdb;
10
- global $EM_Event;
11
- $action = ( !empty($_GET ['action']) ) ? $_GET ['action']:'';
12
- $order = ( !empty($_GET ['order']) ) ? $_GET ['order']:'ASC';
13
- $limit = ( !empty($_GET['limit']) ) ? $_GET['limit'] : 20;//Default limit
14
- $page = ( !empty($_GET['pno']) ) ? $_GET['pno']:1;
15
- $offset = ( $page > 1 ) ? ($page-1)*$limit : 0;
16
- $scope_names = array (
17
- 'past' => __ ( 'Past events', 'dbem' ),
18
- 'all' => __ ( 'All events', 'dbem' ),
19
- 'future' => __ ( 'Future events', 'dbem' )
20
- );
21
- $scope = ( !empty($_GET ['scope']) && array_key_exists($_GET ['scope'], $scope_names) ) ? $_GET ['scope']:'future';
22
- $selectedEvents = ( !empty($_GET ['events']) ) ? $_GET ['events']:'';
23
-
24
- // DELETE action
25
- if ( $action == 'deleteEvents' && EM_Object::array_is_numeric($selectedEvents) ) {
26
- EM_Events::delete( $selectedEvents );
27
- }
28
-
29
- // No action, only showing the events list
30
- switch ($scope) {
31
- case "past" :
32
- $title = __ ( 'Past Events', 'dbem' );
33
- break;
34
- case "all" :
35
- $title = __ ( 'All Events', 'dbem' );
36
- break;
37
- default :
38
- $title = __ ( 'Future Events', 'dbem' );
39
- $scope = "future";
40
- }
41
- $args = array('scope'=>$scope, 'limit'=>0, 'order'=>$order );
42
-
43
- if( !get_option('dbem_permissions_events') && !em_verify_admin() ){
44
- $args['owner'] = get_current_user_id();
45
- }
46
-
47
- $events = EM_Events::get( $args );
48
- $events_count = count ( $events );
49
-
50
- $use_events_end = get_option ( 'dbem_use_event_end' );
51
- ?>
52
- <div class="wrap">
53
- <div id="icon-events" class="icon32"><br />
54
- </div>
55
- <h2>
56
- <?php echo $title; ?>
57
- <a href="admin.php?page=events-manager-event" class="button add-new-h2"><?php _e('Add New','dbem'); ?></a>
58
- </h2>
59
- <?php
60
- $link = array ();
61
- $link ['past'] = "<a href='" . get_bloginfo ( 'wpurl' ) . "/wp-admin/admin.php?page=events-manager&amp;scope=past&amp;order=desc'>" . __ ( 'Past events', 'dbem' ) . "</a>";
62
- $link ['all'] = " <a href='" . get_bloginfo ( 'wpurl' ) . "/wp-admin/admin.php?page=events-manager&amp;scope=all&amp;order=desc'>" . __ ( 'All events', 'dbem' ) . "</a>";
63
- $link ['future'] = " <a href='" . get_bloginfo ( 'wpurl' ) . "/wp-admin/admin.php?page=events-manager&amp;scope=future'>" . __ ( 'Future events', 'dbem' ) . "</a>";
64
- ?>
65
- <?php if ( !empty($_GET['error']) ) : ?>
66
- <div id='message' class='error'>
67
- <p><?php echo $_GET['error']; ?></p>
68
- </div>
69
- <?php endif; ?>
70
- <?php if ( !empty($_GET['message']) ) : ?>
71
- <div id='message' class='updated fade'>
72
- <p><?php echo $_GET['message']; ?></p>
73
- </div>
74
- <?php endif; ?>
75
- <form id="posts-filter" action="" method="get"><input type='hidden' name='page' value='events-manager' />
76
- <ul class="subsubsub">
77
- <li><a href='#' class="current"><?php _e ( 'Total', 'dbem' ); ?> <span class="count">(<?php echo (count ( $events )); ?>)</span></a></li>
78
- </ul>
79
- <p class="search-box">
80
- <label class="screen-reader-text" for="post-search-input"><?php _e('Search Events','dbem'); ?>:</label>
81
- <input type="text" id="post-search-input" name="em_search" value="<?php echo (!empty($_GET['em_search'])) ? $_GET['em_search']:''; ?>" />
82
- <input type="submit" value="<?php _e('Search Events','dbem'); ?>" class="button" />
83
- </p>
84
- <div class="tablenav">
85
-
86
- <div class="alignleft actions">
87
- <select name="action">
88
- <option value="-1" selected="selected"><?php _e ( 'Bulk Actions' ); ?></option>
89
- <option value="deleteEvents"><?php _e ( 'Delete selected','dbem' ); ?></option>
90
- </select>
91
- <input type="submit" value="<?php _e ( 'Apply' ); ?>" name="doaction2" id="doaction2" class="button-secondary action" />
92
- <select name="scope">
93
- <?php
94
- foreach ( $scope_names as $key => $value ) {
95
- $selected = "";
96
- if ($key == $scope)
97
- $selected = "selected='selected'";
98
- echo "<option value='$key' $selected>$value</option> ";
99
- }
100
- ?>
101
- </select>
102
- <input id="post-query-submit" class="button-secondary" type="submit" value="<?php _e ( 'Filter' )?>" />
103
- </div>
104
- <!--
105
- <div class="view-switch">
106
- <a href="/wp-admin/edit.php?mode=list"><img class="current" id="view-switch-list" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="List View" alt="List View" name="view-switch-list" /></a> <a href="/wp-admin/edit.php?mode=excerpt"><img id="view-switch-excerpt" src="http://wordpress.lan/wp-includes/images/blank.gif" width="20" height="20" title="Excerpt View" alt="Excerpt View" name="view-switch-excerpt" /></a>
107
- </div>
108
- -->
109
- <?php
110
- if ( $events_count >= $limit ) {
111
- $page_link_template = em_add_get_params($_SERVER['REQUEST_URI'], array('pno'=>'%PAGE%'));
112
- $events_nav = em_admin_paginate( $page_link_template, $events_count, $limit, $page, 5);
113
- echo $events_nav;
114
- }
115
- ?>
116
- <br class="clear" />
117
- </div>
118
-
119
- <?php
120
- if (empty ( $events )) {
121
- // TODO localize
122
- _e ( 'no events','dbem' );
123
- } else {
124
- ?>
125
-
126
- <table class="widefat">
127
- <thead>
128
- <tr>
129
- <th class='manage-column column-cb check-column' scope='col'>
130
- <input class='select-all' type="checkbox" value='1' />
131
- </th>
132
- <th><?php _e ( 'Name', 'dbem' ); ?></th>
133
- <th>&nbsp;</th>
134
- <th><?php _e ( 'Location', 'dbem' ); ?></th>
135
- <th colspan="2"><?php _e ( 'Date and time', 'dbem' ); ?></th>
136
- </tr>
137
- </thead>
138
- <tbody>
139
- <?php
140
- $rowno = 0;
141
- $event_count = 0;
142
- foreach ( $events as $event ) {
143
- /* @var $event EM_Event */
144
- if( ($rowno < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ) {
145
- $rowno++;
146
- $class = ($rowno % 2) ? ' class="alternate"' : '';
147
- // FIXME set to american
148
- $localised_start_date = date_i18n('D d M Y', $event->start);
149
- $localised_end_date = date_i18n('D d M Y', $event->end);
150
- $style = "";
151
- $today = date ( "Y-m-d" );
152
- $location_summary = "<b>" . $event->location->name . "</b><br/>" . $event->location->address . " - " . $event->location->town;
153
- $category = new EM_Category($event->category_id);
154
-
155
- if ($event->start_date < $today && $event->end_date < $today){
156
- $style = "style ='background-color: #FADDB7;'";
157
- }
158
- ?>
159
- <tr <?php echo "$class $style"; ?>>
160
-
161
- <td>
162
- <input type='checkbox' class='row-selector' value='<?php echo $event->id; ?>' name='events[]' />
163
- </td>
164
- <td>
165
- <strong>
166
- <a class="row-title" href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-event&amp;event_id=<?php echo $event->id ?>&amp;scope=<?php echo $scope ?>&amp;p=<?php echo $page ?>"><?php echo ($event->name); ?></a>
167
- </strong>
168
- <?php if( is_object($category) ) : ?>
169
- <br/><span title='<?php echo __( 'Category', 'dbem' ).": ".$category->name ?>'><?php echo $category->name ?></span>
170
- <?php endif; ?>
171
- <?php
172
- if( get_option('dbem_rsvp_enabled') == 1 && $event->rsvp == 1 ){
173
- ?>
174
- <br/>
175
- <a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-bookings&amp;event_id=<?php echo $event->id ?>"><?php echo __("Bookings",'dbem'); ?></a> &ndash;
176
- <?php _e("Booked",'dbem'); ?>: <?php echo $event->get_bookings()->get_booked_seats()."/".$event->seats; ?>
177
- <?php if( get_option('dbem_bookings_approval') == 1 ): ?>
178
- | <?php _e("Pending",'dbem') ?>: <?php echo $event->get_bookings()->get_pending_seats(); ?>
179
- <?php endif;
180
- }
181
- ?>
182
- </td>
183
- <td>
184
- <a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-event&amp;action=duplicate&amp;event_id=<?php echo $event->id; ?>&amp;scope=<?php echo $scope ?>&amp;p=<?php echo $page ?>" title="<?php _e ( 'Duplicate this event', 'dbem' ); ?>">
185
- <strong>+</strong>
186
- </a>
187
- </td>
188
- <td>
189
- <?php echo $location_summary; ?>
190
- </td>
191
-
192
- <td>
193
- <?php echo $localised_start_date; ?>
194
- <?php echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'' ?>
195
- <br />
196
- <?php
197
- //TODO Should 00:00 - 00:00 be treated as an all day event?
198
- echo substr ( $event->start_time, 0, 5 ) . " - " . substr ( $event->end_time, 0, 5 );
199
- ?>
200
- </td>
201
- <td>
202
- <?php
203
- if ( $event->is_recurrence() ) {
204
- ?>
205
- <strong>
206
- <?php echo $event->get_recurrence_description(); ?> <br />
207
- <a href="<?php bloginfo ( 'wpurl' )?>/wp-admin/admin.php?page=events-manager-event&amp;event_id=<?php echo $event->recurrence_id ?>&amp;scope=<?php echo $scope ?>&amp;p=<?php echo $page ?>"><?php _e ( 'Reschedule', 'dbem' ); ?></a>
208
- </strong>
209
- <?php
210
- }
211
- ?>
212
- </td>
213
- </tr>
214
- <?php
215
- }
216
- $event_count++;
217
- }
218
- ?>
219
- </tbody>
220
- </table>
221
- <?php
222
- } // end of table
223
- ?>
224
- <div class='tablenav'>
225
- <div class="alignleft actions">
226
- <br class='clear' />
227
- </div>
228
- <?php if ( $events_count >= $limit ) : ?>
229
- <div class="tablenav-pages">
230
- <?php
231
- echo $events_nav;
232
- ?>
233
- </div>
234
- <?php endif; ?>
235
- <br class='clear' />
236
- </div>
237
- </form>
238
- </div>
239
- <?php
240
- }
241
-
242
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
admin/em-help.php CHANGED
@@ -1,52 +1,69 @@
1
- <?php
2
- /**
3
- * Display function for the support page. here we can give links to forums and special upgrade instructions e.g. migration features
4
- */
5
- function em_admin_help_page(){
6
- global $wpdb;
7
- ?>
8
- <div class="wrap">
9
- <div id="icon-events" class="icon32"><br /></div>
10
- <h2><?php _e('Getting Help for Events Manager','dbem'); ?></h2>
11
- <p>
12
- For further information on using this plugin, please view the <a href="http://wp-events-plugin.com/documentation/">documentation pages</a>. If you can't find what you're looking for in the documentation, you may find help on our <a href="http://wp-events-plugin.com/forums/">support forums</a>.
13
- </p>
14
- <div class="em-docs">
15
- <h2><?php _e('Placeholders for customizing event pages','dbem'); ?></h2>
16
- <p><?php echo sprintf( __("In the <a href='%s'>settings page</a>, you'll find various textboxes where you can edit how event information looks, such as for event and location lists. Using the placeholders below, you can choose what information should be displayed.",'dbem'), 'admin.php?page=events-manager-options'); ?></p>
17
- <h3><a name="event-placeholders"><?php _e('Event Related Placeholders','dbem'); ?></a></h3>
18
- <?php echo em_docs_placeholders( array('type'=>'events') ); ?>
19
- <h3><a name="location-placeholders"><?php _e('Location Related Placeholders','dbem'); ?></a></h3>
20
- <?php echo em_docs_placeholders( array('type'=>'locations') ); ?>
21
- <h3><a name="booking-placeholders"><?php _e('Booking Related Placeholders','dbem'); ?></a></h3>
22
- <?php echo em_docs_placeholders( array('type'=>'bookings') ); ?>
23
- </div>
24
- <?php
25
- //Is this a previously imported installation?
26
- $old_table_name = $wpdb->prefix.'dbem_events';
27
- if( $wpdb->get_var("SHOW TABLES LIKE '$old_table_name'") == $old_table_name ){
28
- ?>
29
- <hr style="margin:30px 10px;" />
30
- <div class="updated">
31
- <h3>Troubleshooting upgrades from version 2.x to 3.x</h3>
32
- <p>We notice that you upgraded from version 2, as we are now using new database tables, and we do not delete the old tables in case something went wrong with this upgrade.</p>
33
- <p>If something went wrong with the update to version 3 read on:</p>
34
- <h4>Scenario 1: the plugin is working, but for some reason the old events weren't imported</h4>
35
- <p>You can safely reimport your old events from the previous tables without any risk of deleting them. However, if you click the link below <b>YOU WILL OVERWRITE ANY NEW EVENTS YOU CREATED IN VERSION 3</b></p>
36
- <p><a onclick="return confirm('Are you sure you want to do this? Any new changes made since updating will be overwritten by your old ones, and this cannot be undone');" href="<?php echo wp_nonce_url( get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager-help&em_reimport=1', 'em_reimport' ) ?>">Reimport Events from version 2</a></p>
37
- <h4>Scenario 2: the plugin is not working, I want to go back to version 2!</h4>
38
- <p>You can safely downgrade and will not lose any information.</p>
39
- <ol>
40
- <li>First of all, <a href='http://downloads.wordpress.org/plugin/events-manager.2.2.2.zip'>dowload a copy of version 2.2</a></li>
41
- <li>Deactivate and delete Events Manager in the plugin page</li>
42
- <li><a href="<?php bloginfo('wpurl'); ?>/wp-admin/plugin-install.php?tab=upload">Upload the zip file you just downloaded here</a></li>
43
- <li>Let the developers know, of any bugs you ran into while upgrading. We'll help you out if there is a simple solution, and will fix reported bugs within days, if not quicker!</li>
44
- </ol>
45
- </div>
46
- <?php
47
- }
48
- ?>
49
- </div>
50
- <?php
51
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
52
  ?>
1
+ <?php
2
+ /**
3
+ * Display function for the support page. here we can give links to forums and special upgrade instructions e.g. migration features
4
+ */
5
+ function em_admin_help_page(){
6
+ global $wpdb;
7
+ ?>
8
+ <div class="wrap">
9
+ <div id="icon-events" class="icon32"><br /></div>
10
+ <h2><?php _e('Getting Help for Events Manager','dbem'); ?></h2>
11
+ <div class="em-docs">
12
+ <h2>Where To Get Help</h3>
13
+ <p>
14
+ This page is only a small portion of the event documentation which is here for quick reference. If you're just starting out, we recommend you visit the following places for further support:
15
+ </p>
16
+ <ol>
17
+ <li>New users are strongly encouraged to have a look at our <a href="http://wp-events-plugin.com/documentation/getting-started/">getting started page</a>.</li>
18
+ <li>Browse the other documentation pages and <a href="http://wp-events-plugin.com/tutorials/">tutorials</a>.</li>
19
+ <li>View the <a href="http://wp-events-plugin.com/documentation/faq/">FAQ</a> for general questions and <a href="http://wp-events-plugin.com/documentation/troubleshooting/">Troubleshooting</a> for common issues. These are regularly updated with recent issues.</li>
20
+ <li>Rather than trying to contact us directly, we request you use the <a href="http://wordpress.org/tags/events-manager?forum_id=10">support forums</a> as others may be experiencing the same issues as you. For faster support via private member forums and extra features please consider <a href="http://wp-events-plugin.com/events-manager-pro/">upgrading to pro</a>.</li>
21
+ </ol>
22
+ <p>
23
+ If you can't find what you're looking for in the documentation, you may find help on our <a href="http://wp-events-plugin.com/forums/">support forums</a>.
24
+ </p>
25
+ <h2><?php _e('Placeholders for customizing event pages','dbem'); ?></h2>
26
+ <p><?php echo sprintf( __("In the <a href='%s'>settings page</a>, you'll find various textboxes where you can edit how event information looks, such as for event and location lists. Using the placeholders below, you can choose what information should be displayed.",'dbem'), EM_ADMIN_URL .'&amp;events-manager-options'); ?></p>
27
+ <a name="event-placeholders"></a>
28
+ <h3 style="margin-top:20px;"><?php _e('Event Related Placeholders','dbem'); ?></h3>
29
+ <?php echo em_docs_placeholders( array('type'=>'events') ); ?>
30
+ <a name="category-placeholders"></a>
31
+ <h3><?php _e('Category Related Placeholders','dbem'); ?></h3>
32
+ <?php echo em_docs_placeholders( array('type'=>'categories') ); ?>
33
+ <a name="location-placeholders"></a>
34
+ <h3><?php _e('Location Related Placeholders','dbem'); ?></h3>
35
+ <?php echo em_docs_placeholders( array('type'=>'locations') ); ?>
36
+ <a name="booking-placeholders"></a>
37
+ <h3><?php _e('Booking Related Placeholders','dbem'); ?></h3>
38
+ <?php echo em_docs_placeholders( array('type'=>'bookings') ); ?>
39
+ </div>
40
+ <?php
41
+
42
+ //Is this a previously imported installation?
43
+ $old_table_name = $wpdb->prefix.'dbem_events';
44
+ if( $wpdb->get_var("SHOW TABLES LIKE '$old_table_name'") == $old_table_name ){
45
+ ?>
46
+ <hr style="margin:30px 10px;" />
47
+ <div class="updated">
48
+ <h3>Troubleshooting upgrades from version 2.x to 3.x</h3>
49
+ <p>We notice that you upgraded from version 2, as we are now using new database tables, and we do not delete the old tables in case something went wrong with this upgrade.</p>
50
+ <p>If something went wrong with the update to version 3 read on:</p>
51
+ <h4>Scenario 1: the plugin is working, but for some reason the old events weren't imported</h4>
52
+ <p>You can safely reimport your old events from the previous tables without any risk of deleting them. However, if you click the link below <b>YOU WILL OVERWRITE ANY NEW EVENTS YOU CREATED IN VERSION 3</b></p>
53
+ <p><a onclick="return confirm('Are you sure you want to do this? Any new changes made since updating will be overwritten by your old ones, and this cannot be undone');" href="<?php echo wp_nonce_url( EM_ADMIN_URL .'&amp;events-manager-help&em_reimport=1', 'em_reimport' ) ?>">Reimport Events from version 2</a></p>
54
+ <h4>Scenario 2: the plugin is not working, I want to go back to version 2!</h4>
55
+ <p>You can safely downgrade and will not lose any information.</p>
56
+ <ol>
57
+ <li>First of all, <a href='http://downloads.wordpress.org/plugin/events-manager.2.2.2.zip'>dowload a copy of version 2.2</a></li>
58
+ <li>Deactivate and delete Events Manager in the plugin page</li>
59
+ <li><a href="<?php bloginfo('wpurl'); ?>/wp-admin/plugin-install.php?tab=upload">Upload the zip file you just downloaded here</a></li>
60
+ <li>Let the developers know, of any bugs you ran into while upgrading. We'll help you out if there is a simple solution, and will fix reported bugs within days, if not quicker!</li>
61
+ </ol>
62
+ </div>
63
+ <?php
64
+ }
65
+ ?>
66
+ </div>
67
+ <?php
68
+ }
69
  ?>
admin/em-locations-search.php CHANGED
@@ -3,20 +3,24 @@
3
  * This page will search for either a specific location via GET "id" variable
4
  * or will search for events by name via the GET "q" variable.
5
  */
 
6
  require_once('../../../../wp-load.php');
7
  global $wpdb;
8
 
9
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
10
- $location_cond = ( get_option('dbem_permissions_locations') < 1 && !em_verify_admin() ) ? "AND location_owner=".get_current_user_id() : '';
11
 
12
  $term = (isset($_GET['term'])) ? '%'.$_GET['term'].'%' : '%'.$_GET['q'].'%';
13
  $sql = $wpdb->prepare("
14
  SELECT
 
15
  Concat( location_name, ', ', location_address, ', ', location_town) AS `label`,
16
  location_name AS `value`,
17
  location_address AS `address`,
18
  location_town AS `town`,
19
- location_id AS `id`
 
 
20
  FROM $locations_table
21
  WHERE ( `location_name` LIKE %s ) $location_cond LIMIT 10
22
  ", $term);
3
  * This page will search for either a specific location via GET "id" variable
4
  * or will search for events by name via the GET "q" variable.
5
  */
6
+ //FIXME just plug loc search into ajax catcher
7
  require_once('../../../../wp-load.php');
8
  global $wpdb;
9
 
10
+ $locations_table = EM_LOCATIONS_TABLE;
11
+ $location_cond = ( !current_user_can('edit_others_locations') && !current_user_can('read_others_locations') ) ? "AND location_owner=".get_current_user_id() : '';
12
 
13
  $term = (isset($_GET['term'])) ? '%'.$_GET['term'].'%' : '%'.$_GET['q'].'%';
14
  $sql = $wpdb->prepare("
15
  SELECT
16
+ location_id AS `id`,
17
  Concat( location_name, ', ', location_address, ', ', location_town) AS `label`,
18
  location_name AS `value`,
19
  location_address AS `address`,
20
  location_town AS `town`,
21
+ location_state AS `state`,
22
+ location_postcode AS `postcode`,
23
+ location_country AS `country`
24
  FROM $locations_table
25
  WHERE ( `location_name` LIKE %s ) $location_cond LIMIT 10
26
  ", $term);
admin/em-locations.php DELETED
@@ -1,268 +0,0 @@
1
- <?php
2
- /**
3
- * Looks at the request values, saves/updates and then displays the right menu in the admin
4
- * @return null
5
- */
6
- function em_admin_locations_page() {
7
- //TODO EM_Location is globalized, use it fully here
8
- global $EM_Location;
9
- //Take actions
10
- if( !empty($_REQUEST['action']) || !empty($_REQUEST['location_id']) ){
11
- if( $_REQUEST['action'] == "edit" || $_REQUEST['action'] == "add" ) {
12
- //edit/add location
13
- em_admin_location();
14
- } elseif( $_REQUEST['action'] == "delete" ){
15
- //delelte location
16
- $locations = $_REQUEST['locations'];
17
- foreach($locations as $location_id) {
18
- $EM_Location = new EM_Location($location_id);
19
- $EM_Location->delete();
20
- }
21
- em_admin_locations(__('Locations Deleted', "dbem" ));
22
- } elseif( $_REQUEST['action'] == "save") {
23
- // save (add/update) location
24
- if( empty($EM_Location) || !is_object($EM_Location) ){
25
- $EM_Location = new EM_Location(); //blank location
26
- $success_message = __('The location has been added.', 'dbem');
27
- }else{
28
- $success_message = __('The location has been updated.', 'dbem');
29
- }
30
- $EM_Location->get_post();
31
- $validation_result = $EM_Location->validate();
32
- if ( $validation_result ) {
33
- $EM_Location->save(); //FIXME better handling of db write fails when saving location
34
- em_admin_locations($success_message);
35
- } else {
36
- ?>
37
- <div id='message' class='error '>
38
- <p>
39
- <strong><?php _e( "Ach, there's a problem here:", "dbem" ) ?></strong><br /><br /><?php echo implode('<br />', $EM_Location->errors); ?>
40
- </p>
41
- </div>
42
- <?php
43
- unset($EM_Location);
44
- em_admin_location();
45
- }
46
- }
47
- } else {
48
- // no action, just a locations list
49
- em_admin_locations();
50
- }
51
- }
52
-
53
- function em_admin_locations($message='', $fill_fields = false) {
54
- $limit = ( !empty($_REQUEST['limit']) ) ? $_REQUEST['limit'] : 20;//Default limit
55
- $page = ( !empty($_REQUEST['pno']) ) ? $_REQUEST['pno']:1;
56
- $offset = ( $page > 1 ) ? ($page-1)*$limit : 0;
57
- $locations = EM_Locations::get();
58
- $locations_count = count($locations);
59
- ?>
60
- <div class='wrap'>
61
- <div id='icon-edit' class='icon32'>
62
- <br/>
63
- </div>
64
- <h2>
65
- <?php _e('Locations', 'dbem'); ?>
66
- <a href="admin.php?page=events-manager-locations&action=add" class="button add-new-h2"><?php _e('Add New') ?></a>
67
- </h2>
68
-
69
- <?php if($message != "") : ?>
70
- <div id='message' class='updated fade below-h2'>
71
- <p><?php echo $message ?></p>
72
- </div>
73
- <?php endif; ?>
74
-
75
- <form id='bookings-filter' method='post' action=''>
76
- <input type='hidden' name='page' value='locations'/>
77
- <input type='hidden' name='limit' value='<?php echo $limit ?>' />
78
- <input type='hidden' name='p' value='<?php echo $page ?>' />
79
- <?php if ( $locations_count > 0 ) : ?>
80
- <div class='tablenav'>
81
- <div class="alignleft actions">
82
- <select name="action">
83
- <option value="" selected="selected"><?php _e ( 'Bulk Actions' ); ?></option>
84
- <option value="delete"><?php _e ( 'Delete selected','dbem' ); ?></option>
85
- </select>
86
- <input type="submit" value="<?php _e ( 'Apply' ); ?>" id="doaction2" class="button-secondary action" />
87
- <?php
88
- //Pagination (if needed/requested)
89
- if( $locations_count >= $limit ){
90
- //Show the pagination links (unless there's less than 10 events
91
- $page_link_template = preg_replace('/(&|\?)p=\d+/i','',$_SERVER['REQUEST_URI']);
92
- $page_link_template = em_add_get_params($page_link_template, array('pno'=>'%PAGE%'));
93
- $locations_nav = em_paginate( $page_link_template, $locations_count, $limit, $page);
94
- echo $locations_nav;
95
- }
96
- ?>
97
- </div>
98
- </div>
99
- <table class='widefat'>
100
- <thead>
101
- <tr>
102
- <th class='manage-column column-cb check-column' scope='col'><input type='checkbox' class='select-all' value='1'/></th>
103
- <th><?php _e('Name', 'dbem') ?></th>
104
- <th><?php _e('Address', 'dbem') ?></th>
105
- <th><?php _e('Town', 'dbem') ?></th>
106
- </tr>
107
- </thead>
108
- <tfoot>
109
- <tr>
110
- <th class='manage-column column-cb check-column' scope='col'><input type='checkbox' class='select-all' value='1'/></th>
111
- <th><?php _e('Name', 'dbem') ?></th>
112
- <th><?php _e('Address', 'dbem') ?></th>
113
- <th><?php _e('Town', 'dbem') ?></th>
114
- </tr>
115
- </tfoot>
116
- <tbody>
117
- <?php $i = 1; ?>
118
- <?php foreach ($locations as $EM_Location) : ?>
119
- <?php if( $i >= $offset && $i <= $offset+$limit ): ?>
120
- <tr>
121
- <td><input type='checkbox' class ='row-selector' value='<?php echo $EM_Location->id ?>' name='locations[]'/></td>
122
- <td><a href='admin.php?page=events-manager-locations&amp;action=edit&amp;location_id=<?php echo $EM_Location->id ?>'><?php echo $EM_Location->name ?></a></td>
123
- <td><?php echo $EM_Location->address ?></td>
124
- <td><?php echo $EM_Location->town ?></td>
125
- </tr>
126
- <?php endif; ?>
127
- <?php $i++; ?>
128
- <?php endforeach; ?>
129
- </tbody>
130
- </table>
131
- <?php else: ?>
132
- <p><?php _e('No venues have been inserted yet!', 'dbem') ?></p>
133
- <?php endif; ?>
134
- </form>
135
- </div>
136
- <?php
137
- }
138
-
139
- function em_admin_location($message = "") {
140
- global $EM_Location, $EM_Event;
141
- //check that user can access this page
142
- if( is_object($EM_Location) && !$EM_Location->can_manage() ){
143
- ?>
144
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this location.','dbem'); ?></p></div>
145
- <?php
146
- return false;
147
- }
148
- if( empty($EM_Location) || !is_object($EM_Location) ){
149
- $title = __('Add location', 'dbem');
150
- $EM_Location = new EM_Location();
151
- }else{
152
- $title = __('Edit location', 'dbem');
153
- }
154
- ?>
155
- <form enctype='multipart/form-data' name='editcat' id='locationForm' method='post' action='admin.php?page=events-manager-locations' class='validate'>
156
- <input type='hidden' name='action' value='save' />
157
- <input type='hidden' name='location_id' value='<?php echo $EM_Location->id ?>'/>
158
- <div class='wrap'>
159
- <div id='icon-edit' class='icon32'>
160
- <br/>
161
- </div>
162
- <h2><?php echo $title ?></h2>
163
-
164
- <?php if($message != "") : ?>
165
- <div id='message' class='updated fade below-h2' style='background-color: rgb(255, 251, 204);'>
166
- <p><?php echo $message ?></p>
167
- </div>
168
- <?php endif; ?>
169
- <div id='ajax-response'></div>
170
-
171
- <div id="poststuff" class="metabox-holder">
172
- <div id="post-body">
173
- <div id="post-body-content">
174
- <div id="location_name" class="stuffbox">
175
- <h3>
176
- <?php _e ( 'Location Name', 'dbem' ); ?>
177
- </h3>
178
- <div class="inside">
179
- <input name='location_name' id='location-name' type='text' value='<?php echo htmlspecialchars($EM_Location->name, ENT_QUOTES); ?>' size='40' />
180
- <br />
181
- <?php _e('The name of the location', 'dbem') ?>
182
- </div>
183
- </div>
184
- <div id="location_coordinates" class="stuffbox" style='display: none;'>
185
- <h3>
186
- <?php _e ( 'Coordinates', 'dbem' ); ?>
187
- </h3>
188
- <div class="inside">
189
- <input id='location-latitude' name='location_latitude' type='text' value='<?php echo $EM_Location->latitude; ?>' size='15' />
190
- -
191
- <input id='location-longitude' name='location_longitude' type='text' value='<?php echo $EM_Location->longitude; ?>' size='15' />
192
- </div>
193
- </div>
194
- <div id="location_info" class="stuffbox">
195
- <h3>
196
- <?php _e ( 'Location', 'dbem' ); ?>
197
- </h3>
198
- <div class="inside">
199
- <table id="dbem-location-data">
200
- <tr>
201
- <td style="padding-right:20px">
202
- <table>
203
- <tr>
204
- <th><?php _e ( 'Address:' )?>&nbsp;</th>
205
- <td>
206
- <input id="location-address" type="text" name="location_address" value="<?php echo htmlspecialchars($EM_Location->address, ENT_QUOTES); ; ?>" />
207
- <p><?php _e ( 'The address of the location where the event takes place. Example: 21, Dominick Street', 'dbem' )?></p>
208
- </td>
209
- </tr>
210
- <tr>
211
- <th><?php _e ( 'Town:' )?>&nbsp;</th>
212
- <td>
213
- <input id="location-town" type="text" name="location_town" value="<?php echo htmlspecialchars($EM_Location->town, ENT_QUOTES); ?>" />
214
- <p><?php _e ( 'The town where the location is located. If you\'re using the Google Map integration and want to avoid geotagging ambiguities include the country in the town field. Example: Verona, Italy.', 'dbem' )?></p>
215
- </td>
216
- </tr>
217
- </table>
218
- </td>
219
- <?php if ( get_option ( 'dbem_gmap_is_active' ) ) : ?>
220
- <td width="400">
221
- <div id='em-map-404' style='width: 400px; vertical-align:middle; text-align: center;'>
222
- <p><em><?php _e ( 'Location not found', 'dbem' ); ?></em></p>
223
- </div>
224
- <div id='em-map' style='width: 400px; height: 300px; display: none;'></div>
225
- </td>
226
- <?php endif; ?>
227
- </tr>
228
- </table>
229
- </div>
230
- </div>
231
-
232
- <div id="location_description" class="postbox">
233
- <h3>
234
- <?php _e ( 'Details', 'dbem' ); ?>
235
- </h3>
236
- <div class="inside">
237
- <div id="<?php echo user_can_richedit() ? 'postdivrich' : 'postdiv'; ?>" class="postarea">
238
- <?php the_editor($EM_Location->description ); ?>
239
- </div>
240
- <br />
241
- <?php _e ( 'Details about the location', 'dbem' )?>
242
- </div>
243
- </div>
244
-
245
- <div id="location_description" class="stuffbox">
246
- <h3>
247
- <?php _e ( 'Location image', 'dbem' ); ?>
248
- </h3>
249
- <div class="inside" style="padding:10px;">
250
- <?php if ($EM_Location->image_url != '') : ?>
251
- <img src='<?php echo $EM_Location->image_url; ?>' alt='<?php echo $EM_Location->name ?>'/>
252
- <?php else : ?>
253
- <?php _e('No image uploaded for this location yet', 'debm') ?>
254
- <?php endif; ?>
255
- <br /><br />
256
- <label for='location_image'><?php _e('Upload/change picture', 'dbem') ?></label> <input id='location-image' name='location_image' id='location_image' type='file' size='40' />
257
- </div>
258
- </div>
259
- </div>
260
- </div>
261
- </div>
262
- <p class='submit'><input type='submit' class='button-primary' name='submit' value='<?php _e('Update location', 'dbem') ?>' /></p>
263
- </div>
264
- </form>
265
- <?php
266
- }
267
-
268
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
admin/em-ms-locations.php ADDED
@@ -0,0 +1,55 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Looks at the request values, saves/updates and then displays the right menu in the admin
4
+ * @return null
5
+ */
6
+ function em_admin_ms_locations() {
7
+ //TODO EM_Location is globalized, use it fully here
8
+ global $EM_Location;
9
+ EM_Object::ms_global_switch();
10
+ //Take actions
11
+ if( !empty($_REQUEST['action']) && ($_REQUEST['action'] == "edit" || $_REQUEST['action'] == "location_save")) {
12
+ em_admin_location();
13
+ } else {
14
+ // no action, just a locations list
15
+ em_admin_locations();
16
+ }
17
+ EM_Object::ms_global_switch_back();
18
+ }
19
+
20
+ function em_admin_locations($message='', $fill_fields = false) {
21
+ global $current_site;
22
+ ?>
23
+ <div class='wrap'>
24
+ <div id='icon-edit' class='icon32'>
25
+ <br/>
26
+ </div>
27
+ <h2>
28
+ <?php _e('Locations', 'dbem'); ?>
29
+ <span><a href="<?php echo em_add_get_params($_SERVER['REQUEST_URI'], array('action'=>'edit')); ?>" class="button add-new-h2"><?php _e('Add New','dbem'); ?></a></span>
30
+ </h2>
31
+ <?php em_locate_template('tables/locations.php',true, array('url' => $_SERVER['REQUEST_URI'])); ?>
32
+ </div>
33
+ <?php
34
+ }
35
+
36
+ function em_admin_location($message = "") {
37
+ global $EM_Location, $EM_Notices;
38
+ if( empty($EM_Location) || !is_object($EM_Location) ){
39
+ $title = __('Add location', 'dbem');
40
+ $EM_Location = new EM_Location();
41
+ }else{
42
+ $title = __('Edit location', 'dbem');
43
+ }
44
+ ?>
45
+ <div class='wrap'>
46
+ <div id='icon-edit' class='icon32'>
47
+ <br/>
48
+ </div>
49
+ <h2><?php echo $title ?></h2>
50
+ <?php em_locate_template('forms/location-editor.php', true); ?>
51
+ </div>
52
+ <?php
53
+ }
54
+
55
+ ?>
admin/em-ms-options.php ADDED
@@ -0,0 +1,213 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ function em_ms_upgrade( $blog_id ){
3
+ ?>
4
+ <div class="wrap">
5
+ <div id='icon-options-general' class='icon32'><br /></div>
6
+ <h2><?php _e('Update Network'); ?></h2>
7
+ <?php
8
+ if( $_REQUEST['action'] == 'upgrade' && check_admin_referer('em_ms_ugrade_'.get_current_user_id()) ){
9
+ global $current_site,$wpdb;
10
+ $blog_ids = $wpdb->get_col('SELECT blog_id FROM '.$wpdb->blogs.' WHERE site_id='.$current_site->blog_id);
11
+ foreach($blog_ids as $blog_id){
12
+ switch_to_blog($blog_id);
13
+ if( EM_VERSION > get_option('dbem_version', 0) ){
14
+ require_once( dirname(__FILE__).'/../em-install.php');
15
+ em_install();
16
+ echo "<p>Upgraded - ".get_bloginfo('blogname')."</p>";
17
+ }
18
+ restore_current_blog();
19
+ }
20
+ echo "<p>Done Upgrading</p>";
21
+ }else{
22
+ ?>
23
+ <form action="" method="post">
24
+ <p><?php _e('To update your network blogs with the latest Events Manager automatically, click the update button below.'); ?></p>
25
+ <input type="hidden" name="_wpnonce" value="<?php echo wp_create_nonce('em_ms_ugrade_'.get_current_user_id()); ?>" />
26
+ <input type="hidden" name="action" value="upgrade" />
27
+ <input type="submit" value="<?php _e('Update','dbem'); ?>" />
28
+ </form>
29
+ <?php
30
+ }
31
+ ?>
32
+ </div>
33
+ <?php
34
+ }
35
+
36
+ /**
37
+ * Displays network-related options in the network admin section
38
+ * @uses em_options_save() to save settings
39
+ */
40
+ function em_ms_admin_options_page() {
41
+ global $wpdb,$EM_Notices;
42
+ //Check for uninstall/reset request
43
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'uninstall' ){
44
+ em_admin_options_uninstall_page();
45
+ return;
46
+ }
47
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'reset' ){
48
+ em_admin_options_reset_page();
49
+ return;
50
+ }
51
+ //TODO place all options into an array
52
+ $events_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;events-manager-help#event-placeholders">'. __('Event Related Placeholders','dbem') .'</a>';
53
+ $locations_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;events-manager-help#location-placeholders">'. __('Location Related Placeholders','dbem') .'</a>';
54
+ $bookings_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;events-manager-help#booking-placeholders">'. __('Booking Related Placeholders','dbem') .'</a>';
55
+ $categories_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;events-manager-help#category-placeholders">'. __('Category Related Placeholders','dbem') .'</a>';
56
+ $events_placeholder_tip = " ". sprintf(__('This accepts %s and %s placeholders.','dbem'),$events_placeholders, $locations_placeholders);
57
+ $locations_placeholder_tip = " ". sprintf(__('This accepts %s placeholders.','dbem'), $locations_placeholders);
58
+ $categories_placeholder_tip = " ". sprintf(__('This accepts %s placeholders.','dbem'), $categories_placeholders);
59
+ $bookings_placeholder_tip = " ". sprintf(__('This accepts %s, %s and %s placeholders.','dbem'), $bookings_placeholders, $events_placeholders, $locations_placeholders);
60
+
61
+ $save_button = '<tr><th>&nbsp;</th><td><p class="submit" style="margin:0px; padding:0px; text-align:right;"><input type="submit" id="dbem_options_submit" name="Submit" value="'. __( 'Save Changes', 'dbem') .' ('. __('All','dbem') .')" /></p></ts></td></tr>';
62
+ //Do some multisite checking here for reuse
63
+ ?>
64
+ <script type="text/javascript" charset="utf-8">
65
+ jQuery(document).ready(function($){
66
+ var close_text = '<?php _e('Collapse All','dbem'); ?>';
67
+ var open_text = '<?php _e('Expand All','dbem'); ?>';
68
+ var open_close = $('<a href="#" style="display:block; float:right; clear:right; margin:10px;">'+open_text+'</a>');
69
+ $('#em-options-title').before(open_close);
70
+ open_close.click( function(e){
71
+ e.preventDefault();
72
+ if($(this).text() == close_text){
73
+ $(".postbox").addClass('closed');
74
+ $(this).text(open_text);
75
+ }else{
76
+ $(".postbox").removeClass('closed');
77
+ $(this).text(close_text);
78
+ }
79
+ });
80
+ $(".postbox > h3").click(function(){ $(this).parent().toggleClass('closed'); });
81
+ $(".postbox").addClass('closed');
82
+ //MS Mode selection hiders
83
+ $('input[name="dbem_ms_global_table"]').change(function(){ //global
84
+ if( $('input:radio[name="dbem_ms_global_table"]:checked').val() == 1 ){
85
+ $("tbody.em-global-options").show();
86
+ $('input:radio[name="dbem_ms_mainblog_locations"]:checked').trigger('change');
87
+ }else{
88
+ $("tbody.em-global-options").hide();
89
+ }
90
+ }).first().trigger('change');
91
+ //events
92
+ $('input[name="dbem_ms_global_events"]').change(function(){
93
+ if( $('input:radio[name="dbem_ms_global_events"]:checked').val() == 1 ){
94
+ $("tr#dbem_ms_global_events_links_row").show();
95
+ $('input:radio[name="dbem_ms_global_events_links"]:checked').trigger('change');
96
+ }else{
97
+ $("tr#dbem_ms_global_events_links_row, tr#dbem_ms_events_slug_row").hide();
98
+ }
99
+ }).first().trigger('change');
100
+ $('input[name="dbem_ms_global_events_links"]').change(function(){
101
+ if( $('input:radio[name="dbem_ms_global_events_links"]:checked').val() == 1 ){
102
+ $("tr#dbem_ms_events_slug_row").hide();
103
+ }else{
104
+ $("tr#dbem_ms_events_slug_row").show();
105
+ }
106
+ }).first().trigger('change');
107
+ //locations
108
+ $('input[name="dbem_ms_mainblog_locations"]').change(function(){
109
+ if( $('input:radio[name="dbem_ms_mainblog_locations"]:checked').val() == 1 ){
110
+ $("tbody.em-global-locations").hide();
111
+ }else{
112
+ $("tbody.em-global-locations").show();
113
+ }
114
+ }).first().trigger('change');
115
+ $('input[name="dbem_ms_global_locations"]').change(function(){
116
+ if( $('input:radio[name="dbem_ms_global_locations"]:checked').val() == 1 ){
117
+ $("tr#dbem_ms_global_locations_links_row").show();
118
+ $('input:radio[name="dbem_ms_global_locations_links"]:checked').trigger('change');
119
+ }else{
120
+ $("tr#dbem_ms_global_locations_links_row, tr#dbem_ms_locations_slug_row").hide();
121
+ }
122
+ }).first().trigger('change');
123
+ $('input[name="dbem_ms_global_locations_links"]').change(function(){
124
+ if( $('input:radio[name="dbem_ms_global_locations_links"]:checked').val() == 1 ){
125
+ $("tr#dbem_ms_locations_slug_row").hide();
126
+ }else{
127
+ $("tr#dbem_ms_locations_slug_row").show();
128
+ }
129
+ });
130
+ });
131
+ </script>
132
+ <style type="text/css">.postbox h3 { cursor:pointer; }</style>
133
+ <div class="wrap">
134
+ <div id='icon-options-general' class='icon32'><br /></div>
135
+ <h2 class="nav-tab-wrapper">
136
+ <a href="#" id="em-menu-general" class="nav-tab nav-tab-active"><?php _e('General','dbem'); ?></a>
137
+ </h2>
138
+ <h3 id="em-options-title"><?php _e ( 'Event Manager Options', 'dbem' ); ?></h3>
139
+ <?php echo $EM_Notices; ?>
140
+ <form id="em-options-form" method="post" action="">
141
+ <div class="metabox-holder">
142
+ <!-- // TODO Move style in css -->
143
+ <div class='postbox-container' style='width: 99.5%'>
144
+ <div id="">
145
+
146
+ <div class="em-menu-general em-menu-group">
147
+ <div class="postbox " >
148
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Multi Site Options', 'dbem' ); ?></span></h3>
149
+ <div class="inside">
150
+ <table class="form-table">
151
+ <?php
152
+ em_options_radio_binary ( __( 'Enable global tables mode?', 'dbem'), 'dbem_ms_global_table', __( 'Setting this to yes will make all events save in the main site event tables (EM must also be activated). This allows you to share events across different blogs, such as showing events in your network whilst allowing users to display and manage their events within their own blog. Bear in mind that activating this will mean old events created on the sub-blogs will not be accessible anymore, and if you switch back they will be but new events created during global events mode will only remain on the main site.','dbem' ) );
153
+ ?>
154
+ <tbody class="em-global-options">
155
+ <?php
156
+ global $current_site;
157
+ $global_slug_tip = __('%s belonging to other sub-sites will have an extra slug preppended to it so that your main site can differentiate between its own %s and those belonging to other sites in your network.');
158
+ $global_link_tip = __( 'When displaying global %s on the main site you have the option of users viewing the %s details on the main site or being directed to the sub-site.','dbem' );
159
+ $global_post_tip = __( 'Displays %s from all sites on the network by default. You can still restrict %s by blog using shortcodes and template tags coupled with the <code>blog</code> attribute. Requires global tables to be turned on.','dbem');
160
+ $global_link_tip2 = __('You <strong>must</strong> have assigned a %s page in your <a href="%s">main blog settings</a> for this to work.');
161
+ $options_page_link = get_admin_url($current_site->blog_id, 'edit.php?post_type=event&page=events-manager-options#pages');
162
+ ?><tr><td><strong><?php echo sprintf(__('%s Options','dbem'),__('Event','dbem')); ?></strong></td></tr><?php
163
+ em_options_radio_binary ( sprintf(__( 'Display global events on main blog?', 'dbem'), __('events','dbem')), 'dbem_ms_global_events', sprintf($global_post_tip, __('events','dbem'), __('events','dbem')) );
164
+ em_options_radio_binary ( sprintf(__( 'Link sub-site %s directly to sub-site?', 'dbem'), __('events','dbem')), 'dbem_ms_global_events_links', sprintf($global_link_tip, __('events','dbem'), __('event','dbem')).sprintf($global_link_tip2, __('event','dbem'), $options_page_link) );
165
+ em_options_input_text ( sprintf(__( 'Global %s slug', 'dbem' ),__('event','dbem')), 'dbem_ms_events_slug', sprintf($global_slug_tip, __('Events','dbem'), __('events','dbem')).__('Example:','dbem').'<code>http://yoursite.com/events/<strong>event</strong>/subsite-event-slug/', EM_EVENT_SLUG );
166
+ ?><tr><td><strong><?php echo sprintf(__('%s Options','dbem'),__('Location','dbem')); ?></strong></td></tr><?php
167
+ em_options_radio_binary ( sprintf(__( 'Locations on main blog?', 'dbem'), __('locations','dbem')), 'dbem_ms_mainblog_locations', __('If you would prefer all your locations to belong to your main blog, users in sub-sites will still be able to create locations, but the actual locations are created and reside in the main blog.','dbem') );
168
+ ?>
169
+ </tbody>
170
+ <tbody class="em-global-options em-global-locations">
171
+ <?php
172
+ em_options_radio_binary ( sprintf(__( 'Display global %s on main blog?', 'dbem'), __('locations','dbem')), 'dbem_ms_global_locations', sprintf($global_post_tip, __('locations','dbem'), __('locations','dbem')) );
173
+ em_options_radio_binary ( sprintf(__( 'Link sub-site %s directly to sub-site?', 'dbem'), __('locations','dbem')), 'dbem_ms_global_locations_links', sprintf($global_link_tip, __('locations','dbem'), __('location','dbem')).sprintf($global_link_tip2, __('location','dbem'), $options_page_link) );
174
+ em_options_input_text ( sprintf(__( 'Global %s slug', 'dbem' ),__('event','dbem')), 'dbem_ms_locations_slug', sprintf($global_slug_tip, __('Locations','dbem'), __('locations','dbem')).__('Example:','dbem').'<code>http://yoursite.com/locations/<strong>location</strong>/subsite-location-slug/', EM_LOCATION_SLUG );
175
+ ?>
176
+ </tbody>
177
+ <?php echo $save_button; ?>
178
+ </table>
179
+
180
+ </div> <!-- . inside -->
181
+ </div> <!-- .postbox -->
182
+
183
+ <?php
184
+ //including shared MS/non-MS boxes
185
+ em_admin_option_box_caps();
186
+ em_admin_option_box_image_sizes();
187
+ em_admin_option_box_email();
188
+ em_admin_option_box_anon_events();
189
+ em_admin_option_box_uninstall();
190
+ ?>
191
+
192
+ <?php do_action('em_ms_options_page_footer'); ?>
193
+ </div> <!-- .em-menu-general -->
194
+
195
+ <div class="em-menu-pages em-menu-group" style="display:none;">
196
+
197
+ </div> <!-- .em-menu-pages -->
198
+
199
+ <p class="submit">
200
+ <input type="submit" id="dbem_options_submit" name="Submit" value="<?php _e ( 'Save Changes' )?>" />
201
+ <input type="hidden" name="em-submitted" value="1" />
202
+ <input type="hidden" name="_wpnonce" value="<?php echo wp_create_nonce('events-manager-options'); ?>" />
203
+ </p>
204
+
205
+ </div> <!-- .metabox-sortables -->
206
+ </div> <!-- .postbox-container -->
207
+
208
+ </div> <!-- .metabox-holder -->
209
+ </form>
210
+ </div>
211
+ <?php
212
+ }
213
+ ?>
admin/em-options.php CHANGED
@@ -1,444 +1,1469 @@
1
- <?php
2
-
3
- //Function composing the options subpanel
4
- function em_options_save(){
5
- if( current_user_can('activate_plugins') && !empty($_POST['em-submitted']) ){
6
- //Build the array of options here
7
- $post = $_POST;
8
- foreach ($_POST as $postKey => $postValue){
9
- if( substr($postKey, 0, 5) == 'dbem_' ){
10
- //TODO some more validation/reporting
11
- $numeric_options = array('dbem_locations_default_limit','dbem_events_default_limit');
12
- if( in_array($postKey,$numeric_options) && !is_numeric($postValue) ){
13
- //Do nothing, keep old setting.
14
- }else{
15
- //TODO slashes being added?
16
- //$postValue = EM_Object::sanitize($postValue)
17
- update_option($postKey, stripslashes($postValue));
18
- }
19
- }
20
- }
21
- function em_options_saved_notice(){
22
- ?>
23
- <div class="updated"><p><strong><?php _e('Changes saved.'); ?></strong></p></div>
24
- <?php
25
- }
26
- add_action ( 'admin_notices', 'em_options_saved_notice' );
27
- }
28
- }
29
- add_action('admin_head', 'em_options_save');
30
-
31
-
32
-
33
- function em_admin_options_page() {
34
- //TODO place all options into an array
35
- $events_placeholders = '<a href="admin.php?page=events-manager-help#event-placeholders">'. __('Event Related Placeholders','dbem') .'</a>';
36
- $locations_placeholders = '<a href="admin.php?page=events-manager-help#location-placeholders">'. __('Location Related Placeholders','dbem') .'</a>';
37
- $bookings_placeholders = '<a href="admin.php?page=events-manager-help#booking-placeholders">'. __('Booking Related Placeholders','dbem') .'</a>';
38
- $events_placeholder_tip = " ". sprintf(__('This accepts %s and %s placeholders.','dbem'),$events_placeholders, $locations_placeholders);
39
- $locations_placeholder_tip = " ". sprintf(__('This accepts %s placeholders.','dbem'), $locations_placeholders);
40
- $bookings_placeholder_tip = " ". sprintf(__('This accepts %s, %s and %s placeholders.','dbem'), $bookings_placeholders, $events_placeholders, $locations_placeholders);
41
-
42
- $save_button = '<tr><th>&nbsp;</th><td><p class="submit" style="margin:0px; padding:0px; text-align:right;"><input type="submit" id="dbem_options_submit" name="Submit" value="'. __( 'Save Changes' ) .' ('. __('All','dbem') .')" /></p></ts></td></tr>';
43
- ?>
44
- <script type="text/javascript" charset="utf-8">
45
- jQuery(document).ready(function($){
46
- var close_text = '<?php _e('Collapse All','dbem'); ?>';
47
- var open_text = '<?php _e('Expand All','dbem'); ?>';
48
- var open_close = $('<a href="#" style="display:block; float:right; clear:right; margin:10px;">'+close_text+'</a>');
49
- $('#icon-options-general').after(open_close);
50
- open_close.click( function(e){
51
- e.preventDefault();
52
- if($(this).text() == close_text){
53
- $(".postbox").addClass('closed');
54
- $(this).text(open_text);
55
- }else{
56
- $(".postbox").removeClass('closed');
57
- $(this).text(close_text);
58
- }
59
- });
60
- //For rewrite titles
61
- $('input:radio[name=dbem_disable_title_rewrites]').live('change',function(){
62
- checked_check = $('input:radio[name=dbem_disable_title_rewrites]:checked');
63
- if( checked_check.val() == 1 ){
64
- $('#dbem_title_html_row').show();
65
- }else{
66
- $('#dbem_title_html_row').hide();
67
- }
68
- });
69
- $('input:radio[name=dbem_disable_title_rewrites]').trigger('change');
70
-
71
- });
72
- </script>
73
- <div class="wrap">
74
- <div id='icon-options-general' class='icon32'><br />
75
- </div>
76
- <h2><?php _e ( 'Event Manager Options', 'dbem' ); ?></h2>
77
-
78
- <form id="dbem_options_form" method="post" action="">
79
-
80
- <div class="metabox-holder">
81
- <!-- // TODO Move style in css -->
82
- <div class='postbox-container' style='width: 99.5%'>
83
- <div id="" class="meta-box-sortables" >
84
-
85
- <div class="postbox " >
86
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'General options', 'dbem' ); ?> </span></h3>
87
- <div class="inside">
88
- <table class="form-table">
89
- <?php
90
- em_options_radio_binary ( __( 'Use dropdown for locations?' ), 'dbem_use_select_for_locations', __( 'Select yes to select location from a drow-down menu; location selection will be faster, but you will lose the ability to insert locations with events','dbem' ) );
91
- em_options_radio_binary ( __( 'Use recurrence?' ), 'dbem_recurrence_enabled', __( 'Select yes to enable the recurrence features feature','dbem' ) );
92
- em_options_radio_binary ( __( 'Use RSVP?' ), 'dbem_rsvp_enabled', __( 'Select yes to enable the RSVP feature','dbem' ) );
93
- em_options_radio_binary ( __( 'Use categories?' ), 'dbem_categories_enabled', __( 'Select yes to enable the category features','dbem' ) );
94
- em_options_radio_binary ( __( 'Use attributes?' ), 'dbem_attributes_enabled', __( 'Select yes to enable the attributes feature','dbem' ) );
95
-
96
- /*default category*/
97
- $category_options = array();
98
- $category_options[0] = __('no default category','dbem');
99
- $EM_Categories = EM_Categories::get();
100
- foreach($EM_Categories as $EM_Category){
101
- $category_options[$EM_Category->id] = $EM_Category->name;
102
- }
103
- em_options_select ( __( 'Default Category' ), 'dbem_default_category', $category_options, __( 'This option allows you to select the default category when adding an event.','dbem' )." ".__('(not applicable with event ownership on presently, coming soon!)','dbem') );
104
-
105
- /*default location*/
106
- $location_options = array();
107
- $location_options[0] = __('no default location','dbem');
108
- $EM_Locations = EM_Locations::get();
109
- foreach($EM_Locations as $EM_Location){
110
- $location_options[$EM_Location->id] = $EM_Location->name;
111
- }
112
- em_options_select ( __( 'Default Location' ), 'dbem_default_location', $location_options, __( 'This option allows you to select the default location when adding an event.','dbem' )." ".__('(not applicable with event ownership on presently, coming soon!)','dbem') );
113
-
114
- em_options_textarea ( __( 'Custom Placeholders', 'dbem' ), 'dbem_placeholders_custom', sprintf(__( "You can add custom placeholders here, one per line in this format <code>#_ATT{key}</code>. They will not appear on event pages unless you insert them into another template below, but you may want to store extra information about an event for other uses. <a href='%s'>More information on placeholders.</a>", 'dbem' ), 'wp-events-plugin.com/documentation/the-em-templates-syntax/') );
115
-
116
- echo $save_button;
117
- ?>
118
- </table>
119
-
120
- </div> <!-- . inside -->
121
- </div> <!-- .postbox -->
122
-
123
- <div class="postbox " >
124
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Events page', 'dbem' ); ?> </span></h3>
125
- <div class="inside">
126
- <table class="form-table">
127
- <?php
128
- //Wordpress Pages
129
- global $em_disable_filter; //Using a flag here instead
130
- $em_disable_filter = true;
131
- $get_pages = get_pages();
132
- $events_page_options = array();
133
- $events_page_options[0] = __('[No Events Page]');
134
- //TODO Add the hierarchy style ddm, like when choosing page parents
135
- foreach($get_pages as $page){
136
- $events_page_options[$page->ID] = $page->post_title;
137
- }
138
- em_options_select ( __( 'Events page' ), 'dbem_events_page', $events_page_options, __( 'This option allows you to select which page to use as an events page','dbem' ) );
139
- $em_disable_filter = false;
140
- //Rest
141
- em_options_radio_binary ( __( 'Show events page in lists?', 'dbem' ), 'dbem_list_events_page', __( 'Check this option if you want the events page to appear together with other pages in pages lists.', 'dbem' ) );
142
- em_options_radio_binary ( __( 'Display calendar in events page?', 'dbem' ), 'dbem_display_calendar_in_events_page', __( 'This options allows to display the calendar in the events page, instead of the default list. It is recommended not to display both the calendar widget and a calendar page.','dbem' ).' '.__('If you would like to show events that span over more than one day, see the Calendar section on this page.','dbem') );
143
- em_options_radio_binary ( __( 'Disable title rewriting?', 'dbem' ), 'dbem_disable_title_rewrites', __( "Some wordpress themes don't follow best practices when generating navigation menus, and so the automatic title rewriting feature may cause problems, if your menus aren't working correctly on the event pages, try setting this to 'Yes', and provide an appropriate HTML title format below.",'dbem' ) );
144
- em_options_input_text ( __( 'Event Manager titles', 'dbem' ), 'dbem_title_html', __( "This only setting only matters if you selected 'Yes' to above. You will notice the events page titles aren't being rewritten, and you have a new title underneath the default page name. This is where you control the HTML of this title. Make sure you keep the #_PAGETITLE placeholder here, as that's what is rewritten by events manager. To control what's rewritten in this title, see settings further down for page titles.", 'dbem' ) );
145
- em_options_input_text ( __( 'Event List Limits', 'dbem' ), 'dbem_events_default_limit', __( "This will control how many events are shown on one list by default.", 'dbem' ) );
146
- ?>
147
- <tr valign="top" id='dbem_events_default_orderby_row'>
148
- <th scope="row"><?php _e('Default event list ordering','dbem'); ?></th>
149
- <td>
150
- <select name="dbem_events_default_orderby" >
151
- <?php
152
- $orderby_options = apply_filters('em_settings_events_default_orderby_ddm', array(
153
- 'start_date,start_time,name' => __('Order by start date, start time, then event name','dbem'),
154
- 'name,start_date,start_time' => __('Order by name, start date, then start time','dbem'),
155
- 'name,end_date,end_time' => __('Order by name, end date, then end time','dbem'),
156
- 'end_date,end_time,name' => __('Order by end date, end time, then event name','dbem'),
157
- ));
158
- ?>
159
- <?php foreach($orderby_options as $key => $value) : ?>
160
- <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_orderby')) ? "selected='selected'" : ''; ?>>
161
- <?php echo $value; ?>
162
- </option>
163
- <?php endforeach; ?>
164
- </select>
165
- <select name="dbem_events_default_order" >
166
- <?php
167
- $ascending = __('Ascending','dbem');
168
- $descending = __('Descending','dbem');
169
- $order_options = apply_filters('em_settings_events_default_order_ddm', array(
170
- 'ASC' => __('All Ascending','dbem'),
171
- 'DESC,ASC,ASC' => __("$descending, $ascending, $ascending",'dbem'),
172
- 'DESC,DESC,ASC' => __("$descending, $descending, $ascending",'dbem'),
173
- 'DESC' => __('All Descending','dbem'),
174
- 'ASC,DESC,ASC' => __("$ascending, $descending, $ascending",'dbem'),
175
- 'ASC,DESC,DESC' => __("$ascending, $descending, $descending",'dbem'),
176
- 'ASC,ASC,DESC' => __("$ascending, $ascending, $descending",'dbem'),
177
- 'DESC,ASC,DESC' => __("$descending, $ascending, $descending",'dbem'),
178
- ));
179
- ?>
180
- <?php foreach( $order_options as $key => $value) : ?>
181
- <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_order')) ? "selected='selected'" : ''; ?>>
182
- <?php echo $value; ?>
183
- </option>
184
- <?php endforeach; ?>
185
- </select>
186
- <br/>
187
- <em><?php _e('When Events Manager displays lists of events the default behaviour is ordering by start date in ascending order. To change this, modify the values above.','dbem'); ?></em>
188
- </td>
189
- </tr>
190
- <tr valign="top" id='dbem_events_display_time_limit'>
191
- <th scope="row"><?php _e('Event list range limit','dbem'); ?></th>
192
- <td>
193
- <select name="dbem_events_page_time_limit" >
194
- <?php
195
- $limit_options = apply_filters('em_settings_events_page_time_limit_ddm', array(
196
- '0' => __('no limit','dbem'),
197
- '1' => __('This month','dbem'),
198
- '2' => __('Next two months','dbem'),
199
- '3' => __('Next three months','dbem'),
200
- '6' => __('Next six months','dbem'),
201
- '12' => __('Next twelve months','dbem')
202
- ));
203
- ?>
204
- <?php foreach( $limit_options as $key => $value) : ?>
205
- <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_page_time_limit')) ? "selected='selected'" : ''; ?>>
206
- <?php echo $value; ?>
207
- </option>
208
- <?php endforeach; ?>
209
- </select>
210
- <br />
211
- <em><?php _e('Only show events starting within a certain time limit on the events page. Default is no time limit is applied.','dbem'); ?></em>
212
- </td>
213
- </tr>
214
- <?php
215
- echo $save_button;
216
- ?>
217
- </table>
218
- </div> <!-- . inside -->
219
- </div> <!-- .postbox -->
220
-
221
- <div class="postbox " >
222
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Events format', 'dbem' ); ?> </span></h3>
223
- <div class="inside">
224
- <table class="form-table">
225
- <?php
226
- em_options_textarea ( __( 'Default event list format header', 'dbem' ), 'dbem_event_list_item_format_header', __( 'This content will appear just above your code for the default event list format. Default is blank', 'dbem' ) );
227
- em_options_textarea ( __( 'Default event list format', 'dbem' ), 'dbem_event_list_item_format', __( 'The format of any events in a list.', 'dbem' ).$events_placeholder_tip );
228
- em_options_textarea ( __( 'Default event list format footer', 'dbem' ), 'dbem_event_list_item_format_footer', __( 'This content will appear just below your code for the default event list format. Default is blank', 'dbem' ) );
229
- em_options_input_text ( __( 'Single event page title format', 'dbem' ), 'dbem_event_page_title_format', __( 'The format of a single event page title.', 'dbem' ).$events_placeholder_tip );
230
- em_options_textarea ( __( 'Default single event format', 'dbem' ), 'dbem_single_event_format', __( 'The format of a single event page.', 'dbem' ).$events_placeholder_tip );
231
- em_options_input_text ( __( 'Events page title', 'dbem' ), 'dbem_events_page_title', __( 'The title on the multiple events page.', 'dbem' ) );
232
- em_options_input_text ( __( 'No events message', 'dbem' ), 'dbem_no_events_message', __( 'The message displayed when no events are available.', 'dbem' ) );
233
- em_options_input_text ( __( 'List events by date title', 'dbem' ), 'dbem_list_date_title', __( 'If viewing a page for events on a specific date, this is the title that would show up. To insert date values, use <a href="http://www.php.net/manual/en/function.date.php">PHP time format characters</a> with a <code>#</code> symbol before them, i.e. <code>#m</code>, <code>#M</code>, <code>#j</code>, etc.<br/>', 'dbem' ) );
234
- echo $save_button;
235
- ?>
236
- </table>
237
- </div> <!-- . inside -->
238
- </div> <!-- .postbox -->
239
-
240
- <div class="postbox " >
241
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Calendar format', 'dbem' ); ?></span></h3>
242
- <div class="inside">
243
- <table class="form-table">
244
- <?php
245
- em_options_input_text ( __( 'Small calendar title', 'dbem' ), 'dbem_small_calendar_event_title_format', __( 'The format of the title, corresponding to the text that appears when hovering on an eventful calendar day.', 'dbem' ).$events_placeholder_tip );
246
- em_options_input_text ( __( 'Small calendar title separator', 'dbem' ), 'dbem_small_calendar_event_title_separator', __( 'The separator appearing on the above title when more than one events are taking place on the same day.', 'dbem' ) );
247
- em_options_input_text ( __( 'Full calendar events format', 'dbem' ), 'dbem_full_calendar_event_format', __( 'The format of each event when displayed in the full calendar. Remember to include <code>li</code> tags before and after the event.', 'dbem' ).$events_placeholder_tip );
248
- em_options_radio_binary ( __( 'Show long events on calendar pages?', 'dbem' ), 'dbem_full_calendar_long_events', __( "If you are showing a calendar on the events page (see Events format section on this page), you have the option of showing events that span over days on each day it occurs.",'dbem' ) );
249
- em_options_radio_binary ( __( 'Show list on day with single event?', 'dbem' ), 'dbem_display_calendar_day_single', __( "By default, if a calendar day only has one event, it display a single event when clicking on the link of that calendar date. If you select Yes here, you will get always see a list of events.",'dbem' ) );
250
- echo $save_button;
251
- ?>
252
- </table>
253
- </div> <!-- . inside -->
254
- </div> <!-- .postbox -->
255
-
256
- <div class="postbox " >
257
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Locations format', 'dbem' ); ?> </span></h3>
258
- <div class="inside">
259
- <table class="form-table">
260
- <?php
261
- em_options_input_text ( __( 'Single location page title format', 'dbem' ), 'dbem_location_page_title_format', __( 'The format of a single location page title.', 'dbem' ).$locations_placeholder_tip );
262
- em_options_textarea ( __( 'Default single location page format', 'dbem' ), 'dbem_single_location_format', __( 'The format of a single location page.', 'dbem' ).$locations_placeholder_tip );
263
- em_options_textarea ( __( 'Default location balloon format', 'dbem' ), 'dbem_location_baloon_format', __( 'The format of of the text appearing in the baloon describing the location in the map.', 'dbem' ).$locations_placeholder_tip );
264
- em_options_textarea ( __( 'Default location event list format', 'dbem' ), 'dbem_location_event_list_item_format', __( 'The format of the events the list inserted in the location page through the <code>#_NEXTEVENTS</code>, <code>#_PASTEVENTS</code> and <code>#_ALLEVENTS</code> element.', 'dbem' ).$locations_placeholder_tip );
265
- em_options_textarea ( __( 'Default no events message', 'dbem' ), 'dbem_location_no_events_message', __( 'The message to be displayed in the list generated by <code>#_NEXTEVENTS</code>, <code>#_PASTEVENTS</code> and <code>#_ALLEVENTS</code> when no events are available.', 'dbem' ) );
266
- echo $save_button;
267
- ?>
268
- </table>
269
- </div> <!-- . inside -->
270
- </div> <!-- .postbox -->
271
-
272
- <div class="postbox " >
273
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'RSS feed format', 'dbem' ); ?> </span></h3>
274
- <div class="inside">
275
- <table class="form-table">
276
- <?php
277
- em_options_input_text ( __( 'RSS main title', 'dbem' ), 'dbem_rss_main_title', __( 'The main title of your RSS events feed.', 'dbem' ).$events_placeholder_tip );
278
- em_options_input_text ( __( 'RSS main description', 'dbem' ), 'dbem_rss_main_description', __( 'The main description of your RSS events feed.', 'dbem' ) );
279
- em_options_input_text ( __( 'RSS title format', 'dbem' ), 'dbem_rss_title_format', __( 'The format of the title of each item in the events RSS feed.', 'dbem' ).$events_placeholder_tip );
280
- em_options_input_text ( __( 'RSS description format', 'dbem' ), 'dbem_rss_description_format', __( 'The format of the description of each item in the events RSS feed.', 'dbem' ).$events_placeholder_tip );
281
- echo $save_button;
282
- ?>
283
- </table>
284
- </div> <!-- . inside -->
285
- </div> <!-- .postbox -->
286
-
287
- <div class="postbox " >
288
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Maps and geotagging', 'dbem' ); ?> </span></h3>
289
- <div class="inside">
290
- <table class='form-table'>
291
- <?php $gmap_is_active = get_option ( 'dbem_gmap_is_active' ); ?>
292
- <tr valign="top">
293
- <th scope="row"><?php _e ( 'Enable Google Maps integration?', 'dbem' ); ?></th>
294
- <td>
295
- <?php _e ( 'Yes' ); ?> <input id="dbem_gmap_is_active_yes" name="dbem_gmap_is_active" type="radio" value="1" <?php echo ($gmap_is_active) ? "checked='checked'":''; ?> />
296
- <?php _e ( 'No' ); ?> <input name="dbem_gmap_is_active" type="radio" value="0" <?php echo ($gmap_is_active) ? '':"checked='checked'"; ?> /><br />
297
- <em><?php _e ( 'Check this option to enable Goggle Map integration.', 'dbem' )?></em>
298
- </td>
299
- </tr>
300
- <?php
301
- em_options_textarea ( __( 'Map text format', 'dbem' ), 'dbem_map_text_format', __( 'The text format inside the map balloons.', 'dbem' ).$events_placeholder_tip );
302
- echo $save_button;
303
- ?>
304
- </table>
305
- </div> <!-- . inside -->
306
- </div> <!-- .postbox -->
307
-
308
- <div class="postbox " >
309
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'RSVP and bookings', 'dbem' ); ?> </span></h3>
310
- <div class="inside">
311
- <table class='form-table'>
312
- <?php
313
- em_options_select ( __( 'Default contact person', 'dbem' ), 'dbem_default_contact_person', em_get_wp_users (), __( 'Select the default contact person. This user will be employed whenever a contact person is not explicitly specified for an event', 'dbem' ) );
314
- em_options_radio_binary ( __( 'Approval Required?', 'dbem' ), 'dbem_bookings_approval', __( 'Bookings will not be confirmed until the event administrator approves it.', 'dbem' ) );
315
- em_options_input_text ( __( 'Email events admin?', 'dbem' ), 'dbem_bookings_notify_admin', __( "If you would like every event booking confirmation email sent to an administrator write their email here (leave blank to not send an email).", 'dbem' ) );
316
- em_options_radio_binary ( __( 'Email contact person?', 'dbem' ), 'dbem_bookings_contact_email', __( 'Check this option if you want the event contact to receive an email when someone books places. An email will be sent when a booking is first made (regardless if confirmed or pending)', 'dbem' ) );
317
- ?>
318
- <tr><td colspan='2'><h4><?php _e('Contact person booking confirmed','dbem') ?></h4></td></tr>
319
- <tr><td colspan='2'><?php echo __('An email will be sent to the event contact when a booking is first made.','dbem').$bookings_placeholder_tip ?></td></tr>
320
- <?php
321
- em_options_input_text ( __( 'Contact person email subject', 'dbem' ), 'dbem_bookings_contact_email_subject', '' );
322
- em_options_textarea ( __( 'Contact person email', 'dbem' ), 'dbem_bookings_contact_email_body', '' );
323
- ?>
324
- <tr><td colspan='2'><h4><?php _e('Contact person booking cancelled','dbem') ?></h4></td></tr>
325
- <tr><td colspan='2'><?php echo __('An email will be sent to the event contact if someone cancels their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
326
- <?php
327
- em_options_input_text ( __( 'Contact person cancellation subject', 'dbem' ), 'dbem_contactperson_email_cancelled_subject', '' );
328
- em_options_textarea ( __( 'Contact person cancellation email', 'dbem' ), 'dbem_contactperson_email_cancelled_body', '' );
329
- ?>
330
- <tr><td colspan='2'><h4><?php _e('Confirmed booking email','dbem') ?></h4></td></tr>
331
- <tr><td colspan='2'><?php echo __('This is sent when a person\'s booking is confirmed. This will be sent automatically if approvals are required and the booking is approved. If approvals are disabled, this is sent out when a user first submits their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
332
- <?php
333
- em_options_input_text ( __( 'Booking confirmed email subject', 'dbem' ), 'dbem_bookings_email_confirmed_subject', '' );
334
- em_options_textarea ( __( 'Booking confirmed email', 'dbem' ), 'dbem_bookings_email_confirmed_body', '' );
335
- ?>
336
- <tr><td colspan='2'><h4><?php _e('Pending booking email','dbem') ?></h4></td></tr>
337
- <tr><td colspan='2'><?php echo __( 'This will be sent to the person when they first submit their booking. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip ?></td></tr>
338
- <?php
339
- em_options_input_text ( __( 'Booking pending email subject', 'dbem' ), 'dbem_bookings_email_pending_subject', '');
340
- em_options_textarea ( __( 'Booking pending email', 'dbem' ), 'dbem_bookings_email_pending_body','') ;
341
- ?>
342
- <tr><td colspan='2'><h4><?php _e('Rejected booking email','dbem') ?></h4></td></tr>
343
- <tr><td colspan='2'><?php echo __( 'This will be sent automatically when a booking is rejected. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip ?></td></tr>
344
- <?php
345
- em_options_input_text ( __( 'Booking rejected email subject', 'dbem' ), 'dbem_bookings_email_rejected_subject', __( "The subject of the email sent to the person making a booking that is awaiting administrator approval. Not relevant if bookings don't require approval.", 'dbem' ).$bookings_placeholder_tip );
346
- em_options_textarea ( __( 'Booking rejected email', 'dbem' ), 'dbem_bookings_email_rejected_body', __( 'The body of the email which will be sent to the person if the booking is rejected. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip );
347
- echo $save_button;
348
- ?>
349
- <tr><td colspan='2'><h4><?php _e('Booking cancelled','dbem') ?></h4></td></tr>
350
- <tr><td colspan='2'><?php echo __('This will be sent when a user cancels their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
351
- <?php
352
- em_options_input_text ( __( 'Booking cancelled email subject', 'dbem' ), 'dbem_bookings_email_cancelled_subject', '' );
353
- em_options_textarea ( __( 'Booking cancelled email', 'dbem' ), 'dbem_bookings_email_cancelled_body', '' );
354
- ?>
355
- </table>
356
- </div> <!-- . inside -->
357
- </div> <!-- .postbox -->
358
-
359
- <div class="postbox " >
360
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Email Settings', 'dbem' ); ?> </span></h3>
361
- <div class="inside">
362
- <table class='form-table'>
363
- <?php
364
- em_options_input_text ( __( 'Notification sender name', 'dbem' ), 'dbem_mail_sender_name', __( "Insert the display name of the notification sender.", 'dbem' ) );
365
- em_options_input_text ( __( 'Notification sender address', 'dbem' ), 'dbem_mail_sender_address', __( "Insert the address of the notification sender.", 'dbem' ) );
366
- em_options_input_text ( 'Mail sending port', 'dbem_rsvp_mail_port', __( "The port through which you e-mail notifications will be sent. Make sure the firewall doesn't block this port", 'dbem' ) );
367
- em_options_select ( __( 'Mail sending method', 'dbem' ), 'dbem_rsvp_mail_send_method', array ('smtp' => 'SMTP', 'mail' => __( 'PHP mail function', 'dbem' ), 'sendmail' => 'Sendmail', 'qmail' => 'Qmail', 'wp_mail' => 'WP Mail' ), __( 'Select the method to send email notification.', 'dbem' ) );
368
- em_options_radio_binary ( __( 'Use SMTP authentication?', 'dbem' ), 'dbem_rsvp_mail_SMTPAuth', __( 'SMTP authentication is often needed. If you use GMail, make sure to set this parameter to Yes', 'dbem' ) );
369
- em_options_input_text ( 'SMTP host', 'dbem_smtp_host', __( "The SMTP host. Usually it corresponds to 'localhost'. If you use GMail, set this value to 'ssl://smtp.gmail.com:465'.", 'dbem' ) );
370
- em_options_input_text ( __( 'SMTP username', 'dbem' ), 'dbem_smtp_username', __( "Insert the username to be used to access your SMTP server.", 'dbem' ) );
371
- em_options_input_password ( __( 'SMTP password', 'dbem' ), "dbem_smtp_password", __( "Insert the password to be used to access your SMTP server", 'dbem' ) );
372
- echo $save_button;
373
- ?>
374
- </table>
375
- </div> <!-- . inside -->
376
- </div> <!-- .postbox -->
377
-
378
- <div class="postbox " >
379
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Images size', 'dbem' ); ?> </span></h3>
380
- <div class="inside">
381
- <table class='form-table'>
382
- <?php
383
- em_options_input_text ( __( 'Maximum width (px)', 'dbem' ), 'dbem_image_max_width', __( 'The maximum allowed width for images uploades', 'dbem' ) );
384
- em_options_input_text ( __( 'Maximum height (px)', 'dbem' ), 'dbem_image_max_height', __( "The maximum allowed height for images uploaded, in pixels", 'dbem' ) );
385
- em_options_input_text ( __( 'Maximum size (bytes)', 'dbem' ), 'dbem_image_max_size', __( "The maximum allowed size for images uploaded, in pixels", 'dbem' ) );
386
- echo $save_button;
387
- ?>
388
- </table>
389
- </div> <!-- . inside -->
390
- </div> <!-- .postbox -->
391
-
392
- <div class="postbox " >
393
- <div class="handlediv" title="<?php __('Click to toggle'); ?>"><br /></div><h3 class='hndle'><span><?php _e ( 'Management Permission Options', 'dbem' ); ?> (Beta)</span></h3>
394
- <div class="inside">
395
- <table class="form-table">
396
- <tr><td colspan="2">
397
- <strong><?php _e('Warning: Changing these values may result in exposing previously hidden information to all users.')?></strong><br />
398
- <em><?php _e('Note that currently "users" are considered as wordpress contributor users upwards, as they can create and manage events (we\'re working on that). Wordpress administrators can control all events/locations/categories/etc. on Events Manager.','dbem'); ?></em>
399
- </td></tr>
400
- <tr><th colspan="2"><strong><?php _e('Event Permissions','dbem'); ?></strong></th></tr>
401
- <?php
402
- $location_privacy_options = array(
403
- '0' => __('Every user can create and manage their own events. Users can\'t view or modify each others\' events and booking data','dbem'),
404
- '1' => __('Every user can create/edit/delete any event on this site. (not recommended)','dbem')
405
- );
406
- em_options_radio ( 'dbem_permissions_events', $location_privacy_options );
407
- ?>
408
- <tr><th colspan="2"><strong><?php _e('Location Permissions','dbem'); ?></strong></th></tr>
409
- <?php
410
- $location_privacy_options = array(
411
- '0' => __('Every user can create and manage their own location.','dbem')." ".__('In future releases of Events Manager, sharing locations with more than one user will be possible in this option, as well as connecting different user defined locations to prevent duplicate listings.','dbem'),
412
- '1' => __('Only event administrators can create and edit locations. User must choose from these available locations.','dbem'),
413
- '2' => __('Everyone can create/edit/delete all locations on this site. (not recommended)','dbem')
414
- );
415
- em_options_radio ( 'dbem_permissions_locations', $location_privacy_options );
416
- ?>
417
- <tr><th colspan="2"><strong><?php _e('Category Permissions','dbem'); ?></strong></th></tr>
418
- <?php
419
- $category_privacy_options = array(
420
- '0' => __('Every user can create and manage their own category.','dbem'),
421
- '1' => __('Only event administrators can create and edit categories. User must choose from these available categories.','dbem'),
422
- '2' => __('Everyone can create/edit/delete all categories on the system. (not recommended)','dbem')
423
- );
424
- em_options_radio ( 'dbem_permissions_categories', $category_privacy_options );
425
- ?>
426
- </table>
427
-
428
- </div> <!-- . inside -->
429
- </div> <!-- .postbox -->
430
-
431
- <p class="submit">
432
- <input type="submit" id="dbem_options_submit" name="Submit" value="<?php _e ( 'Save Changes' )?>" />
433
- <input type="hidden" name="em-submitted" value="1" />
434
- </p>
435
-
436
- </div> <!-- .metabox-sortables -->
437
- </div> <!-- .postbox-container -->
438
-
439
- </div> <!-- .metabox-holder -->
440
- </form>
441
- </div>
442
- <?php
443
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
444
  ?>
1
+ <?php
2
+
3
+ //Function composing the options subpanel
4
+ function em_options_save(){
5
+ global $EM_Notices;
6
+ /*
7
+ * Here's the idea, we have an array of all options that need super admin approval if in multi-site mode
8
+ * since options are only updated here, its one place fit all
9
+ */
10
+ if( current_user_can('activate_plugins') && !empty($_POST['em-submitted']) && check_admin_referer('events-manager-options','_wpnonce') ){
11
+ //Build the array of options here
12
+ $post = $_POST;
13
+ foreach ($_POST as $postKey => $postValue){
14
+ if( substr($postKey, 0, 5) == 'dbem_' ){
15
+ //TODO some more validation/reporting
16
+ $numeric_options = array('dbem_locations_default_limit','dbem_events_default_limit');
17
+ if( in_array($postKey,$numeric_options) && !is_numeric($postValue) ){
18
+ //Do nothing, keep old setting.
19
+ }else{
20
+ //TODO slashes being added?
21
+ update_option($postKey, stripslashes($postValue));
22
+ }
23
+ }
24
+ }
25
+ //set capabilities
26
+ if( !empty($_POST['em_capabilities']) && is_array($_POST['em_capabilities']) && (!is_multisite() || is_multisite() && is_super_admin()) ){
27
+ global $em_capabilities_array, $wp_roles;
28
+ foreach( $wp_roles->role_objects as $role_name => $role ){
29
+ foreach( array_keys($em_capabilities_array) as $capability){
30
+ if( !empty($_POST['em_capabilities'][$role_name][$capability]) ){
31
+ $role->add_cap($capability);
32
+ }else{
33
+ $role->remove_cap($capability);
34
+ }
35
+ }
36
+ }
37
+ }
38
+ update_option('dbem_flush_needed',1);
39
+ do_action('em_options_save');
40
+ $EM_Notices->add_confirm('<strong>'.__('Changes saved.', 'dbem').'</strong>', true);
41
+ wp_redirect(wp_get_referer());
42
+ exit();
43
+ }
44
+ //Migration
45
+ if( !empty($_GET['em_migrate_images']) && check_admin_referer('em_migrate_images','_wpnonce') && get_option('dbem_migrate_images') ){
46
+ include(plugin_dir_path(__FILE__).'../em-install.php');
47
+ $result = em_migrate_uploads();
48
+ if($result){
49
+ $failed = ( $result['fail'] > 0 ) ? $result['fail'] . ' images failed to migrate.' : '';
50
+ $EM_Notices->add_confirm('<strong>'.$result['success'].' images migrated successfully. '.$failed.'</strong>');
51
+ }
52
+ wp_redirect(admin_url().'edit.php?post_type=event&page=events-manager-options&em_migrate_images');
53
+ }elseif( !empty($_GET['em_not_migrate_images']) && check_admin_referer('em_not_migrate_images','_wpnonce') ){
54
+ delete_option('dbem_migrate_images_nag');
55
+ delete_option('dbem_migrate_images');
56
+ }
57
+ //Uninstall
58
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'uninstall' && !empty($_REQUEST['confirmed']) && check_admin_referer('em_uninstall_'.get_current_user_id().'_wpnonce') && is_super_admin() ){
59
+ if( check_admin_referer('em_uninstall_'.get_current_user_id().'_confirmed','_wpnonce2') ){
60
+ //We have a go to uninstall
61
+ global $wpdb;
62
+ //delete EM posts
63
+ remove_action('before_delete_post',array('EM_Location_Post_Admin','before_delete_post'),10,1);
64
+ remove_action('before_delete_post',array('EM_Event_Post_Admin','before_delete_post'),10,1);
65
+ remove_action('before_delete_post',array('EM_Event_Recurring_Post_Admin','before_delete_post'),10,1);
66
+ $post_ids = $wpdb->get_col('SELECT ID FROM '.$wpdb->posts." WHERE post_type IN ('".EM_POST_TYPE_EVENT."','".EM_POST_TYPE_LOCATION."','event-recurring')");
67
+ foreach($post_ids as $post_id){
68
+ wp_delete_post($post_id);
69
+ }
70
+ //delete categories
71
+ $cat_terms = get_terms(EM_TAXONOMY_CATEGORY, array('hide_empty'=>false));
72
+ foreach($cat_terms as $cat_term){
73
+ wp_delete_term($cat_term->term_id, EM_TAXONOMY_CATEGORY);
74
+ }
75
+ $tag_terms = get_terms(EM_TAXONOMY_TAG, array('hide_empty'=>false));
76
+ foreach($tag_terms as $tag_term){
77
+ wp_delete_term($tag_term->term_id, EM_TAXONOMY_TAG);
78
+ }
79
+ //delete EM tables
80
+ $wpdb->query('DROP TABLE '.EM_EVENTS_TABLE);
81
+ $wpdb->query('DROP TABLE '.EM_BOOKINGS_TABLE);
82
+ $wpdb->query('DROP TABLE '.EM_LOCATIONS_TABLE);
83
+ $wpdb->query('DROP TABLE '.EM_TICKETS_TABLE);
84
+ $wpdb->query('DROP TABLE '.EM_TICKETS_BOOKINGS_TABLE);
85
+ $wpdb->query('DROP TABLE '.EM_RECURRENCE_TABLE);
86
+ $wpdb->query('DROP TABLE '.EM_CATEGORIES_TABLE);
87
+ $wpdb->query('DROP TABLE '.EM_META_TABLE);
88
+
89
+ //delete options
90
+ $wpdb->query('DELETE FROM '.$wpdb->options.' WHERE option_name LIKE \'em_%\' OR option_name LIKE \'dbem_%\'');
91
+ //deactivate and go!
92
+ deactivate_plugins(array('events-manager/events-manager.php','events-manager-pro/events-manager-pro.php'), true);
93
+ wp_redirect(admin_url('plugins.php?deactivate=true'));
94
+ exit();
95
+ }
96
+ }
97
+ //Reset
98
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'reset' && !empty($_REQUEST['confirmed']) && check_admin_referer('em_reset_'.get_current_user_id().'_wpnonce') && is_super_admin() ){
99
+ if( check_admin_referer('em_reset_'.get_current_user_id().'_confirmed','_wpnonce2') ){
100
+ //We have a go to uninstall
101
+ global $wpdb;
102
+ //delete options
103
+ $wpdb->query('DELETE FROM '.$wpdb->options.' WHERE option_name LIKE \'em_%\' OR option_name LIKE \'dbem_%\'');
104
+ //reset capabilities
105
+ global $em_capabilities_array, $wp_roles;
106
+ foreach( $wp_roles->role_objects as $role_name => $role ){
107
+ foreach( array_keys($em_capabilities_array) as $capability){
108
+ $role->remove_cap($capability);
109
+ }
110
+ }
111
+ //go back to plugin options page
112
+ $EM_Notices->add_confirm(__('Settings have been reset back to default. Your events, locations and categories have not been modified.','dbem'), true);
113
+ wp_redirect(EM_ADMIN_URL.'&page=events-manager-options');
114
+ exit();
115
+ }
116
+ }
117
+ }
118
+ add_action('admin_init', 'em_options_save');
119
+
120
+ function em_admin_options_reset_page(){
121
+ if( check_admin_referer('em_reset_'.get_current_user_id().'_wpnonce') && is_super_admin() ){
122
+ ?>
123
+ <div class="wrap">
124
+ <div id='icon-options-general' class='icon32'><br /></div>
125
+ <h2><?php _e('Reset Events Manager','dbem'); ?></h2>
126
+ <p style="color:red; font-weight:bold;"><?php _e('Are you sure you want to reset Events Manager?','dbem')?></p>
127
+ <p style="font-weight:bold;"><?php _e('All your settings, including email templates and template formats for Events Manager will be deleted.','dbem')?></p>
128
+ <p>
129
+ <a href="<?php echo add_query_arg(array('_wpnonce2' => wp_create_nonce('em_reset_'.get_current_user_id().'_confirmed'), 'confirmed'=>1)); ?>" class="button-primary"><?php _e('Reset Events Manager','dbem'); ?></a>
130
+ <a href="<?php echo wp_get_referer(); ?>" class="button-secondary"><?php _e('Cancel','dbem'); ?></a>
131
+ </p>
132
+ </div>
133
+ <?php
134
+ }
135
+ }
136
+ function em_admin_options_uninstall_page(){
137
+ if( check_admin_referer('em_uninstall_'.get_current_user_id().'_wpnonce') && is_super_admin() ){
138
+ ?>
139
+ <div class="wrap">
140
+ <div id='icon-options-general' class='icon32'><br /></div>
141
+ <h2><?php _e('Uninstall Events Manager','dbem'); ?></h2>
142
+ <p style="color:red; font-weight:bold;"><?php _e('Are you sure you want to uninstall Events Manager?','dbem')?></p>
143
+ <p style="font-weight:bold;"><?php _e('All your settings and events will be permanently deleted. This cannot be undone.','dbem')?></p>
144
+ <p><?php echo sprintf(__('If you just want to deactivate the plugin, <a href="%s">go to your plugins page</a>.','dbem'), wp_nonce_url(admin_url('plugins.php'))); ?></p>
145
+ <p>
146
+ <a href="<?php echo add_query_arg(array('_wpnonce2' => wp_create_nonce('em_uninstall_'.get_current_user_id().'_confirmed'), 'confirmed'=>1)); ?>" class="button-primary"><?php _e('Uninstall and Deactivate','dbem'); ?></a>
147
+ <a href="<?php echo wp_get_referer(); ?>" class="button-secondary"><?php _e('Cancel','dbem'); ?></a>
148
+ </p>
149
+ </div>
150
+ <?php
151
+ }
152
+ }
153
+
154
+ function em_admin_options_page() {
155
+ global $wpdb, $EM_Notices;
156
+ //Check for uninstall/reset request
157
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'uninstall' ){
158
+ em_admin_options_uninstall_page();
159
+ return;
160
+ }
161
+ if( !empty($_REQUEST['action']) && $_REQUEST['action'] == 'reset' ){
162
+ em_admin_options_reset_page();
163
+ return;
164
+ }
165
+ //TODO place all options into an array
166
+ global $events_placeholder_tip, $locations_placeholder_tip, $categories_placeholder_tip, $bookings_placeholder_tip;
167
+ $events_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;page=events-manager-help#event-placeholders">'. __('Event Related Placeholders','dbem') .'</a>';
168
+ $locations_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;page=events-manager-help#location-placeholders">'. __('Location Related Placeholders','dbem') .'</a>';
169
+ $bookings_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;page=events-manager-help#booking-placeholders">'. __('Booking Related Placeholders','dbem') .'</a>';
170
+ $categories_placeholders = '<a href="'.EM_ADMIN_URL .'&amp;page=events-manager-help#category-placeholders">'. __('Category Related Placeholders','dbem') .'</a>';
171
+ $events_placeholder_tip = " ". sprintf(__('This accepts %s and %s placeholders.','dbem'),$events_placeholders, $locations_placeholders);
172
+ $locations_placeholder_tip = " ". sprintf(__('This accepts %s placeholders.','dbem'), $locations_placeholders);
173
+ $categories_placeholder_tip = " ". sprintf(__('This accepts %s placeholders.','dbem'), $categories_placeholders);
174
+ $bookings_placeholder_tip = " ". sprintf(__('This accepts %s, %s and %s placeholders.','dbem'), $bookings_placeholders, $events_placeholders, $locations_placeholders);
175
+
176
+ global $save_button;
177
+ $save_button = '<tr><th>&nbsp;</th><td><p class="submit" style="margin:0px; padding:0px; text-align:right;"><input type="submit" id="dbem_options_submit" name="Submit" value="'. __( 'Save Changes', 'dbem') .' ('. __('All','dbem') .')" /></p></ts></td></tr>';
178
+ ?>
179
+ <script type="text/javascript" charset="utf-8">
180
+ jQuery(document).ready(function($){
181
+ //Meta Box Options
182
+ var close_text = '<?php _e('Collapse All','dbem'); ?>';
183
+ var open_text = '<?php _e('Expand All','dbem'); ?>';
184
+ var open_close = $('<a href="#" style="display:block; float:right; clear:right; margin:10px;">'+open_text+'</a>');
185
+ $('#em-options-title').before(open_close);
186
+ open_close.click( function(e){
187
+ e.preventDefault();
188
+ if($(this).text() == close_text){
189
+ $(".postbox").addClass('closed');
190
+ $(this).text(open_text);
191
+ }else{
192
+ $(".postbox").removeClass('closed');
193
+ $(this).text(close_text);
194
+ }
195
+ });
196
+ $(".postbox > h3").click(function(){ $(this).parent().toggleClass('closed'); });
197
+ $(".postbox").addClass('closed');
198
+ //Navigation Tabs
199
+ $('.nav-tab-wrapper .nav-tab').click(function(){
200
+ $('.nav-tab-wrapper .nav-tab').removeClass('nav-tab-active');
201
+ el = $(this);
202
+ elid = el.attr('id');
203
+ $('.em-menu-group').hide();
204
+ $('.'+elid).show();
205
+ el.addClass('nav-tab-active');
206
+ $(".postbox").addClass('closed');
207
+ open_close.text(open_text);
208
+ });
209
+ var navUrl = document.location.toString();
210
+ if (navUrl.match('#')) { //anchor-based navigation
211
+ var current_tab = 'a#em-menu-' + navUrl.split('#')[1];
212
+ $(current_tab).trigger('click');
213
+ }
214
+ $('.nav-tab-link').click(function(){ $($(this).attr('rel')).trigger('click'); }); //links to mimick tabs
215
+ //Page Options
216
+ $('input[name="dbem_cp_events_has_archive"]').change(function(){ //event archives
217
+ if( $('input:radio[name="dbem_cp_events_has_archive"]:checked').val() == 1 ){
218
+ $('tbody.em-event-archive-sub-options').show();
219
+ }else{
220
+ $('tbody.em-event-archive-sub-options').hide();
221
+ }
222
+ });
223
+ $('select[name="dbem_events_page"]').change(function(){
224
+ if( $('select[name="dbem_events_page"]').val() == 0 ){
225
+ $('tbody.em-event-page-options').hide();
226
+ $('tbody.em-event-archive-options').show();
227
+ $('input:radio[name="dbem_cp_events_has_archive"]:checked').trigger('change');
228
+ }else{
229
+ $('tbody.em-event-page-options').show();
230
+ $('tbody.em-event-archive-options').hide();
231
+ }
232
+ }).trigger('change');
233
+ $('input[name="dbem_cp_locations_has_archive"]').change(function(){ //location archives
234
+ console.log('changed!');
235
+ if( $('input:radio[name="dbem_cp_locations_has_archive"]:checked').val() == 1 ){
236
+ $('tbody.em-location-archive-sub-options').show();
237
+ }else{
238
+ $('tbody.em-location-archive-sub-options').hide();
239
+ }
240
+ });
241
+ $('select[name="dbem_locations_page"]').change(function(){
242
+ if( $('select[name="dbem_locations_page"]').val() == 0 ){
243
+ $('tbody.em-location-page-options').hide();
244
+ $('tbody.em-location-archive-options').show();
245
+ $('input:radio[name="dbem_cp_locations_has_archive"]:checked').trigger('change');
246
+ }else{
247
+ $('tbody.em-location-page-options').show();
248
+ $('tbody.em-location-archive-options').hide();
249
+ }
250
+ }).trigger('change');
251
+ //For rewrite titles
252
+ $('input:radio[name=dbem_disable_title_rewrites]').live('change',function(){
253
+ checked_check = $('input:radio[name=dbem_disable_title_rewrites]:checked');
254
+ if( checked_check.val() == 1 ){
255
+ $('#dbem_title_html_row').show();
256
+ }else{
257
+ $('#dbem_title_html_row').hide();
258
+ }
259
+ });
260
+ $('input:radio[name=dbem_disable_title_rewrites]').trigger('change');
261
+ });
262
+ </script>
263
+ <style type="text/css">.postbox h3 { cursor:pointer; }</style>
264
+ <div class="wrap">
265
+ <div id='icon-options-general' class='icon32'><br /></div>
266
+ <h2 class="nav-tab-wrapper">
267
+ <a href="#general" id="em-menu-general" class="nav-tab nav-tab-active"><?php _e('General','dbem'); ?></a>
268
+ <a href="#pages" id="em-menu-pages" class="nav-tab"><?php _e('Pages','dbem'); ?></a>
269
+ <a href="#formats" id="em-menu-formats" class="nav-tab"><?php _e('Formats/Layouts','dbem'); ?></a>
270
+ <?php if( get_option('dbem_rsvp_enabled') ): ?>
271
+ <a href="#bookings" id="em-menu-bookings" class="nav-tab"><?php _e('Booking Options','dbem'); ?></a>
272
+ <?php endif; ?>
273
+ <a href="#emails" id="em-menu-emails" class="nav-tab"><?php _e('Emails','dbem'); ?></a>
274
+ </h2>
275
+ <?php echo $EM_Notices; ?>
276
+ <h3 id="em-options-title"><?php _e ( 'Event Manager Options', 'dbem' ); ?></h3>
277
+ <form id="em-options-form" method="post" action="">
278
+ <div class="metabox-holder">
279
+ <!-- // TODO Move style in css -->
280
+ <div class='postbox-container' style='width: 99.5%'>
281
+ <div id="">
282
+
283
+ <div class="em-menu-general em-menu-group">
284
+
285
+ <!-- GENERAL OPTIONS -->
286
+ <div class="postbox " >
287
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'General Options', 'dbem' ); ?> </span></h3>
288
+ <div class="inside">
289
+ <table class="form-table">
290
+ <?php em_options_radio_binary ( __( 'Disable thumbnails?', 'dbem' ), 'dbem_thumbnails_enabled', __( 'Select yes to disable Events Manager from enabling thumbnails (some themes may already have this enabled, which we cannot be turned off here).','dbem' ) ); ?>
291
+ <tr>
292
+ <td colspan="2">
293
+ <h4><?php echo sprintf(__('%s Settings','dbem'),__('Event','dbem')); ?></h4>
294
+ </td>
295
+ </tr>
296
+ <?php
297
+ em_options_radio_binary ( __( 'Enable recurrence?', 'dbem' ), 'dbem_recurrence_enabled', __( 'Select yes to enable the recurrence features feature','dbem' ) );
298
+ em_options_radio_binary ( __( 'Enable bookings?', 'dbem' ), 'dbem_rsvp_enabled', __( 'Select yes to allow bookings and tickets for events.','dbem' ) );
299
+ em_options_radio_binary ( __( 'Enable tags?', 'dbem' ), 'dbem_tags_enabled', __( 'Select yes to enable the tag features','dbem' ) );
300
+ if( !(EM_MS_GLOBAL && !is_main_blog()) ){
301
+ em_options_radio_binary ( __( 'Enable categories?', 'dbem' ), 'dbem_categories_enabled', __( 'Select yes to enable the category features','dbem' ) );
302
+ if( get_option('dbem_categories_enabled') ){
303
+ /*default category*/
304
+ $category_options = array();
305
+ $category_options[0] = __('no default category','dbem');
306
+ $EM_Categories = EM_Categories::get();
307
+ foreach($EM_Categories as $EM_Category){
308
+ $category_options[$EM_Category->id] = $EM_Category->name;
309
+ }
310
+ em_options_select ( __( 'Default Category', 'dbem' ), 'dbem_default_category', $category_options, __( 'This option allows you to select the default category when adding an event.','dbem' ).' '.__('If an event does not have a category assigned when editing, this one will be assigned automatically.','dbem'));
311
+ }
312
+ }
313
+ em_options_radio_binary ( __( 'Enable event attributes?', 'dbem' ), 'dbem_attributes_enabled', __( 'Select yes to enable the attributes feature','dbem' ) );
314
+ em_options_radio_binary ( __( 'Enable event custom fields?', 'dbem' ), 'dbem_cp_events_custom_fields', __( 'Custom fields are the same as attributes, except you cannot restrict specific values, users can add any kind of custom field name/value pair. Only available in the WordPress admin area.','dbem' ) );
315
+ if( get_option('dbem_attributes_enabled') ){
316
+ em_options_textarea ( __( 'Event Attributes', 'dbem' ), 'dbem_placeholders_custom', sprintf(__( "You can also add event attributes here, one per line in this format <code>#_ATT{key}</code>. They will not appear on event pages unless you insert them into another template below, but you may want to store extra information about an event for other uses. <a href='%s'>More information on placeholders.</a>", 'dbem' ), EM_ADMIN_URL .'&amp;page=events-manager-help') );
317
+ }
318
+ if( get_option('dbem_locations_enabled') ){
319
+ /*default location*/
320
+ $location_options = array();
321
+ $location_options[0] = __('no default location','dbem');
322
+ $EM_Locations = EM_Locations::get();
323
+ foreach($EM_Locations as $EM_Location){
324
+ $location_options[$EM_Location->location_id] = $EM_Location->location_name;
325
+ }
326
+ em_options_select ( __( 'Default Location', 'dbem' ), 'dbem_default_location', $location_options, __( 'This option allows you to select the default location when adding an event.','dbem' )." ".__('(not applicable with event ownership on presently, coming soon!)','dbem') );
327
+
328
+ /*default location country*/
329
+ em_options_select ( __( 'Default Location Country', 'dbem' ), 'dbem_location_default_country', em_get_countries(__('no default country', 'dbem')), __('If you select a default country, that will be pre-selected when creating a new location.','dbem') );
330
+ }
331
+ ?>
332
+ <tr>
333
+ <td colspan="2">
334
+ <h4><?php echo sprintf(__('%s Settings','dbem'),__('Location','dbem')); ?></h4>
335
+ </td>
336
+ </tr>
337
+ <?php
338
+ em_options_radio_binary ( __( 'Enable locations?', 'dbem' ), 'dbem_locations_enabled', __( 'If you disable locations, bear in mind that you should remove your location page, shortcodes and related placeholders from your formats.','dbem' ) );
339
+ if( get_option('dbem_locations_enabled') ){
340
+ em_options_radio_binary ( __( 'Require locations for events?', 'dbem' ), 'dbem_require_location', __( 'Setting this to no will allow you to submit events without locations. You can use the <code>{no_location}...{/no_location}</code> or <code>{has_location}..{/has_location}</code> conditional placeholder to selectively display location information.','dbem' ) );
341
+ em_options_radio_binary ( __( 'Use dropdown for locations?', 'dbem' ), 'dbem_use_select_for_locations', __( 'Select yes to select location from a drow-down menu; location selection will be faster, but you will lose the ability to insert locations with events','dbem' ) );
342
+ }
343
+ ?>
344
+ <tr>
345
+ <td colspan="2">
346
+ <h4><?php echo sprintf(__('%s Settings','dbem'),__('Other','dbem')); ?></h4>
347
+ </td>
348
+ </tr>
349
+ <?php
350
+ em_options_radio_binary ( __('Show some love?','dbem'), 'dbem_credits', __( 'Hundreds of free hours have gone into making this free plugin, show your support and add a small link to the plugin website at the bottom of your event pages.','dbem' ) );
351
+ echo $save_button;
352
+ ?>
353
+ </table>
354
+
355
+ </div> <!-- . inside -->
356
+ </div> <!-- .postbox -->
357
+
358
+ <?php if ( !is_multisite() ){ em_admin_option_box_image_sizes(); } ?>
359
+
360
+ <?php if ( !is_multisite() ){ em_admin_option_box_caps(); } ?>
361
+
362
+ <?php if ( !is_multisite() ) { em_admin_option_box_anon_events(); } ?>
363
+
364
+ <?php do_action('em_options_page_footer'); ?>
365
+
366
+ <?php if ( !is_multisite() ) { em_admin_option_box_uninstall(); } ?>
367
+
368
+ <?php if( get_option('dbem_migrate_images') ): ?>
369
+ <div class="postbox " >
370
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span>Migrate Images From Version 4</span></h3>
371
+ <div class="inside">
372
+ <?php /* Not translating as it's temporary */ ?>
373
+ <p>You have the option of migrating images from version 4 so they become the equivalent of 'featured images' like with regular WordPress posts and pages and are also available in your media library.</p>
374
+ <p>Your event and location images will still display correctly on the front-end even if you don't migrate, but will not show up within your edit location/event pages in the admin area.</p>
375
+ <p>
376
+ <a href="<?php echo $_SERVER['REQUEST_URI'] ?>&amp;em_migrate_images=1&amp;_wpnonce=<?php echo wp_create_nonce('em_migrate_images'); ?>" />Migrate Images</a><br />
377
+ <a href="<?php echo $_SERVER['REQUEST_URI'] ?>&amp;em_not_migrate_images=1&amp;_wpnonce=<?php echo wp_create_nonce('em_not_migrate_images'); ?>" />Do Not Migrate Images</a>
378
+ </p>
379
+ </div> <!-- . inside -->
380
+ </div> <!-- .postbox -->
381
+ <?php endif; ?>
382
+ </div> <!-- .em-menu-general -->
383
+
384
+ <!-- PAGE OPTIONS -->
385
+ <div class="em-menu-pages em-menu-group" style="display:none;">
386
+ <?php
387
+ $template_page_tip = __( "Many themes display extra meta information on post pages such as 'posted by' or 'post date' information, which may not be desired. Usually, page templates contain less clutter.", 'dbem' );
388
+ $template_page_tip .= str_replace('#','http://codex.wordpress.org/Post_Types#Template_Files',__("Be aware that some themes will not work with this option, if so (or you want to make your own changes), you can create a file named <code>single-%s.php</code> <a href='#'>as shown on the wordpress codex</a>, and leave this set to Posts.", 'dbem'));
389
+ $format_override_tip = __("By using formats, you can control how your %s are displayed from within the Events Manager <a href='#formats' class='nav-tab-link' rel='#em-menu-formats'>Formats</a> tab above without having to edit your theme files.")
390
+ ?>
391
+ <div class="postbox " >
392
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('Permalink Slugs','dbem')); ?></span></h3>
393
+ <div class="inside">
394
+ <p><?php _e('You can change the permalink structure of your events, locations, categories and tags here. Be aware that you may want to set up redirects if you change your permalink structures to maintain SEO rankings.','dbem'); ?></p>
395
+ <table class="form-table">
396
+ <?php
397
+ em_options_input_text ( __( 'Events', 'dbem' ), 'dbem_cp_events_slug', sprintf(__('e.g. %s - you can use / seperators too', 'dbem' ), '<strong>'.home_url().'/<code>'.get_option('dbem_cp_events_slug',EM_POST_TYPE_EVENT_SLUG).'</code>/2012-olympics/</strong>'), EM_POST_TYPE_EVENT_SLUG );
398
+ if( get_option('dbem_locations_enabled') ){
399
+ em_options_input_text ( __( 'Locations', 'dbem' ), 'dbem_cp_locations_slug', sprintf(__('e.g. %s - you can use / seperators too', 'dbem' ), '<strong>'.home_url().'/<code>'.get_option('dbem_cp_locations_slug',EM_POST_TYPE_LOCATION_SLUG).'</code>/wembley-stadium/</strong>'), EM_POST_TYPE_LOCATION_SLUG );
400
+ }
401
+ if( get_option('dbem_categories_enabled') && !(EM_MS_GLOBAL && !is_main_blog()) ){
402
+ em_options_input_text ( __( 'Event Categories', 'dbem' ), 'dbem_taxonomy_category_slug', sprintf(__('e.g. %s - you can use / seperators too', 'dbem' ), '<strong>'.home_url().'/<code>'.get_option('dbem_taxonomy_category_slug',EM_TAXONOMY_CATEGORY_SLUG).'</code>/sports/</strong>'), EM_TAXONOMY_CATEGORY_SLUG );
403
+ }
404
+ if( get_option('dbem_tags_enabled') ){
405
+ em_options_input_text ( __( 'Event Tags', 'dbem' ), 'dbem_taxonomy_tag_slug', sprintf(__('e.g. %s - you can use / seperators too', 'dbem' ), '<strong>'.home_url().'/<code>'.get_option('dbem_taxonomy_tag_slug',EM_TAXONOMY_TAG_SLUG).'</code>/running/</strong>'), EM_TAXONOMY_TAG_SLUG );
406
+ }
407
+ echo $save_button;
408
+ ?>
409
+ </table>
410
+ </div> <!-- . inside -->
411
+ </div> <!-- .postbox -->
412
+
413
+ <div class="postbox " >
414
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('%s Pages','dbem'),__('Event','dbem')); ?></span></h3>
415
+ <div class="inside">
416
+ <table class="form-table">
417
+ <?php
418
+ em_options_radio_binary ( sprintf(__( 'Display %s as', 'dbem' ),__('events','dbem')), 'dbem_cp_events_template_page', sprintf($template_page_tip, EM_POST_TYPE_EVENT), array(__('Posts'),__('Pages')) );
419
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_events_formats', sprintf($format_override_tip,__('events','dbem')));
420
+ echo $save_button;
421
+ ?>
422
+ </table>
423
+ </div> <!-- . inside -->
424
+ </div> <!-- .postbox -->
425
+
426
+ <div class="postbox " >
427
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('%s List/Archives','dbem'),__('Event','dbem')); ?></span></h3>
428
+ <div class="inside">
429
+ <table class="form-table">
430
+ <?php //WordPress Pages
431
+ global $em_disable_filter; //Using a flag here instead
432
+ $em_disable_filter = true;
433
+ $get_pages = get_pages();
434
+ $events_page_options = array();
435
+ $events_page_options[0] = sprintf(__('[No %s Page]', 'dbem'),__('Events','dbem'));
436
+ //TODO Add the hierarchy style ddm, like when choosing page parents
437
+ foreach($get_pages as $page){
438
+ $events_page_options[$page->ID] = $page->post_title;
439
+ }
440
+ em_options_select ( __( 'Events page', 'dbem' ), 'dbem_events_page', $events_page_options, __( 'This option allows you to select which page to use as an events page. If you do not select an events page, to display event lists you can enable event archives or use the appropriate shortcodes and/or template tags.','dbem' ) );
441
+ $em_disable_filter = false;
442
+ ?>
443
+ <tbody class="em-event-page-options">
444
+ <?php
445
+ em_options_radio_binary ( __( 'Show events page in lists?', 'dbem' ), 'dbem_list_events_page', __( 'Check this option if you want the events page to appear together with other pages in pages lists.', 'dbem' ) );
446
+ em_options_radio_binary ( __( 'Display calendar in events page?', 'dbem' ), 'dbem_display_calendar_in_events_page', __( 'This options allows to display the calendar in the events page, instead of the default list. It is recommended not to display both the calendar widget and a calendar page.','dbem' ).' '.__('If you would like to show events that span over more than one day, see the Calendar section on this page.','dbem') );
447
+ em_options_radio_binary ( __( 'Disable title rewriting?', 'dbem' ), 'dbem_disable_title_rewrites', __( "Some WordPress themes don't follow best practices when generating navigation menus, and so the automatic title rewriting feature may cause problems, if your menus aren't working correctly on the event pages, try setting this to 'Yes', and provide an appropriate HTML title format below.",'dbem' ) );
448
+ em_options_input_text ( __( 'Event Manager titles', 'dbem' ), 'dbem_title_html', __( "This only setting only matters if you selected 'Yes' to above. You will notice the events page titles aren't being rewritten, and you have a new title underneath the default page name. This is where you control the HTML of this title. Make sure you keep the #_PAGETITLE placeholder here, as that's what is rewritten by events manager. To control what's rewritten in this title, see settings further down for page titles.", 'dbem' ) );
449
+ em_options_radio_binary ( __( 'Show events search?', 'dbem' ), 'dbem_events_page_search', __( "If set to yes, a search form will appear just above your list of events.", 'dbem' ) );
450
+ ?>
451
+ </tbody>
452
+ <tbody class="em-event-archive-options">
453
+ <?php
454
+ em_options_radio_binary ( __( 'Enable Archives?', 'dbem' ), 'dbem_cp_events_has_archive', __( "Allow WordPress post-style archives.", 'dbem' ) );
455
+ ?>
456
+ </tbody>
457
+ <tbody class="em-event-archive-options em-event-archive-sub-options">
458
+ <tr valign="top">
459
+ <th scope="row"><?php _e('Default event archive ordering','dbem'); ?></th>
460
+ <td>
461
+ <select name="dbem_events_default_archive_orderby" >
462
+ <?php
463
+ $event_archive_orderby_options = apply_filters('em_settings_events_default_archive_orderby_ddm', array(
464
+ '_start_ts' => __('Order by start date, start time','dbem'),
465
+ 'title' => __('Order by name','dbem')
466
+ ));
467
+ ?>
468
+ <?php foreach($event_archive_orderby_options as $key => $value) : ?>
469
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_archive_orderby')) ? "selected='selected'" : ''; ?>>
470
+ <?php echo $value; ?>
471
+ </option>
472
+ <?php endforeach; ?>
473
+ </select>
474
+ <select name="dbem_events_default_archive_order" >
475
+ <?php
476
+ $ascending = __('Ascending','dbem');
477
+ $descending = __('Descending','dbem');
478
+ $event_archive_order_options = apply_filters('em_settings_events_default_archive_order_ddm', array(
479
+ 'ASC' => __('Ascending','dbem'),
480
+ 'DESC' => __('Descending','dbem')
481
+ ));
482
+ ?>
483
+ <?php foreach( $event_archive_order_options as $key => $value) : ?>
484
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_archive_order')) ? "selected='selected'" : ''; ?>>
485
+ <?php echo $value; ?>
486
+ </option>
487
+ <?php endforeach; ?>
488
+ </select>
489
+ <br/>
490
+ <em><?php _e('When Events Manager displays lists of events the default behaviour is ordering by start date in ascending order. To change this, modify the values above.','dbem'); ?></em>
491
+ </td>
492
+ </tr>
493
+ </tbody>
494
+ <tr>
495
+ <td colspan="2">
496
+ <h4><?php echo _e('General settings','dbem'); ?></h4>
497
+ </td>
498
+ </tr>
499
+ <?php
500
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_events_archive_formats', sprintf($format_override_tip,__('events','dbem')));
501
+ em_options_radio_binary ( __( 'Are current events past events?', 'dbem' ), 'dbem_events_current_are_past', __( "By default, events that are have an end date later than today will be included in searches, set this to yes to consider events that started 'yesterday' as past.", 'dbem' ) );
502
+ em_options_radio_binary ( __( 'Include in WordPress Searches?', 'dbem' ), 'dbem_cp_events_search_results', sprintf(__( "Allow %s to appear in the built-in search results.", 'dbem' ),__('events','dbem')) );
503
+ ?>
504
+ <tr>
505
+ <td colspan="2">
506
+ <h4><?php echo sprintf(__('Default %s list options','dbem'), __('event','dbem')); ?></h4>
507
+ <p><?php _e('These can be overriden when using shortcode or template tags.','dbem'); ?></p>
508
+ </td>
509
+ </tr>
510
+ <tr valign="top" id='dbem_events_default_orderby_row'>
511
+ <th scope="row"><?php _e('Default event list ordering','dbem'); ?></th>
512
+ <td>
513
+ <select name="dbem_events_default_orderby" >
514
+ <?php
515
+ $orderby_options = apply_filters('em_settings_events_default_orderby_ddm', array(
516
+ 'event_start_date,event_start_time,event_name' => __('Order by start date, start time, then event name','dbem'),
517
+ 'event_name,event_start_date,event_start_time' => __('Order by name, start date, then start time','dbem'),
518
+ 'event_name,event_end_date,event_end_time' => __('Order by name, end date, then end time','dbem'),
519
+ 'event_end_date,event_end_time,event_name' => __('Order by end date, end time, then event name','dbem'),
520
+ ));
521
+ ?>
522
+ <?php foreach($orderby_options as $key => $value) : ?>
523
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_orderby')) ? "selected='selected'" : ''; ?>>
524
+ <?php echo $value; ?>
525
+ </option>
526
+ <?php endforeach; ?>
527
+ </select>
528
+ <select name="dbem_events_default_order" >
529
+ <?php
530
+ $ascending = __('Ascending','dbem');
531
+ $descending = __('Descending','dbem');
532
+ $order_options = apply_filters('em_settings_events_default_order_ddm', array(
533
+ 'ASC' => __('All Ascending','dbem'),
534
+ 'DESC,ASC,ASC' => __("$descending, $ascending, $ascending",'dbem'),
535
+ 'DESC,DESC,ASC' => __("$descending, $descending, $ascending",'dbem'),
536
+ 'DESC' => __('All Descending','dbem'),
537
+ 'ASC,DESC,ASC' => __("$ascending, $descending, $ascending",'dbem'),
538
+ 'ASC,DESC,DESC' => __("$ascending, $descending, $descending",'dbem'),
539
+ 'ASC,ASC,DESC' => __("$ascending, $ascending, $descending",'dbem'),
540
+ 'DESC,ASC,DESC' => __("$descending, $ascending, $descending",'dbem'),
541
+ ));
542
+ ?>
543
+ <?php foreach( $order_options as $key => $value) : ?>
544
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_default_order')) ? "selected='selected'" : ''; ?>>
545
+ <?php echo $value; ?>
546
+ </option>
547
+ <?php endforeach; ?>
548
+ </select>
549
+ <br/>
550
+ <em><?php _e('When Events Manager displays lists of events the default behaviour is ordering by start date in ascending order. To change this, modify the values above.','dbem'); ?></em>
551
+ </td>
552
+ </tr>
553
+ <tr valign="top" id='dbem_events_display_time_limit'>
554
+ <th scope="row"><?php _e('Event list scope','dbem'); ?></th>
555
+ <td>
556
+ <select name="dbem_events_page_scope" >
557
+ <?php foreach( em_get_scopes() as $key => $value) : ?>
558
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_events_page_scope')) ? "selected='selected'" : ''; ?>>
559
+ <?php echo $value; ?>
560
+ </option>
561
+ <?php endforeach; ?>
562
+ </select>
563
+ <br />
564
+ <em><?php _e('Only show events starting within a certain time limit on the events page. Default is future events with no end time limit.','dbem'); ?></em>
565
+ </td>
566
+ </tr>
567
+ <?php
568
+ em_options_input_text ( __( 'Event List Limits', 'dbem' ), 'dbem_events_default_limit', __( "This will control how many events are shown on one list by default.", 'dbem' ) );
569
+ echo $save_button;
570
+ ?>
571
+ </table>
572
+ </div> <!-- . inside -->
573
+ </div> <!-- .postbox -->
574
+
575
+ <?php if( get_option('dbem_locations_enabled') ): ?>
576
+ <div class="postbox " >
577
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('%s Pages','dbem'),__('Location','dbem')); ?></span></h3>
578
+ <div class="inside">
579
+ <table class="form-table">
580
+ <?php
581
+ em_options_radio_binary ( sprintf(__( 'Display %s as', 'dbem' ),__('locations','dbem')), 'dbem_cp_locations_template_page', sprintf($template_page_tip, EM_POST_TYPE_LOCATION), array(__('Posts'),__('Pages')) );
582
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_locations_formats', sprintf($format_override_tip,__('locations','dbem')));
583
+ echo $save_button;
584
+ ?>
585
+ </table>
586
+ </div> <!-- . inside -->
587
+ </div> <!-- .postbox -->
588
+
589
+ <div class="postbox " >
590
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('%s List/Archives','dbem'),__('Location','dbem')); ?></span></h3>
591
+ <div class="inside">
592
+ <table class="form-table">
593
+ <?php
594
+ $events_page_options[0] = sprintf(__('[No %s Page]', 'dbem'),__('Locations','dbem'));
595
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('Locations','dbem')), 'dbem_locations_page', $events_page_options, sprintf(__( 'This option allows you to select which page to use as the %s page. If you do not select no %s page, to display lists you can enable archives or use the appropriate shortcodes and/or template tags.','dbem' ),__('locations','dbem'),__('locations','dbem')) );
596
+ ?>
597
+ <tbody class="em-location-page-options">
598
+ <?php
599
+ em_options_radio_binary ( sprintf(__( 'Show %s page in lists?', 'dbem' ),__('locations','dbem')), 'dbem_list_locations_page', sprintf(__( 'Check this option if you want the %s page to appear together with other pages in pages lists.', 'dbem' ),__('locations','dbem')) );
600
+ ?>
601
+ </tbody>
602
+ <tbody class="em-location-archive-options">
603
+ <?php
604
+ em_options_radio_binary ( __( 'Enable Archives?', 'dbem' ), 'dbem_cp_locations_has_archive', __( "Allow WordPress post-style archives.", 'dbem' ) );
605
+ ?>
606
+ </tbody>
607
+ <tbody class="em-location-archive-options em-location-archive-sub-options">
608
+ <tr valign="top">
609
+ <th scope="row"><?php _e('Default archive ordering','dbem'); ?></th>
610
+ <td>
611
+ <select name="dbem_locations_default_archive_orderby" >
612
+ <?php
613
+ $orderby_options = apply_filters('em_settings_locations_default_archive_orderby_ddm', array(
614
+ '_country' => sprintf(__('Order by %s','dbem'),__('Country','dbem')),
615
+ '_town' => sprintf(__('Order by %s','dbem'),__('Town','dbem')),
616
+ 'title' => sprintf(__('Order by %s','dbem'),__('Name','dbem'))
617
+ ));
618
+ ?>
619
+ <?php foreach($orderby_options as $key => $value) : ?>
620
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_locations_default_archive_orderby')) ? "selected='selected'" : ''; ?>>
621
+ <?php echo $value; ?>
622
+ </option>
623
+ <?php endforeach; ?>
624
+ </select>
625
+ <select name="dbem_locations_default_archive_order" >
626
+ <?php
627
+ $ascending = __('Ascending','dbem');
628
+ $descending = __('Descending','dbem');
629
+ $order_options = apply_filters('em_settings_locations_default_archive_order_ddm', array(
630
+ 'ASC' => __('Ascending','dbem'),
631
+ 'DESC' => __('Descending','dbem')
632
+ ));
633
+ ?>
634
+ <?php foreach( $order_options as $key => $value) : ?>
635
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_locations_default_archive_order')) ? "selected='selected'" : ''; ?>>
636
+ <?php echo $value; ?>
637
+ </option>
638
+ <?php endforeach; ?>
639
+ </select>
640
+ </td>
641
+ </tr>
642
+ </tbody>
643
+ <tr>
644
+ <td colspan="2">
645
+ <h4><?php echo _e('General settings','dbem'); ?></h4>
646
+ </td>
647
+ </tr>
648
+ <?php
649
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_locations_archive_formats', sprintf($format_override_tip,__('locations','dbem')));
650
+ em_options_radio_binary ( __( 'Include in WordPress Searches?', 'dbem' ), 'dbem_cp_locations_search_results', sprintf(__( "Allow %s to appear in the built-in search results.", 'dbem' ),__('locations','dbem')) );
651
+ ?>
652
+ <tr>
653
+ <td colspan="2">
654
+ <h4><?php echo sprintf(__('Default %s list options','dbem'), __('location','dbem')); ?></h4>
655
+ <p><?php _e('These can be overriden when using shortcode or template tags.','dbem'); ?></p>
656
+ </td>
657
+ </tr>
658
+ <tr valign="top" id='dbem_locations_default_orderby_row'>
659
+ <th scope="row"><?php _e('Default list ordering','dbem'); ?></th>
660
+ <td>
661
+ <select name="dbem_locations_default_orderby" >
662
+ <?php
663
+ $orderby_options = apply_filters('em_settings_locations_default_orderby_ddm', array(
664
+ 'location_country' => sprintf(__('Order by %s','dbem'),__('Country','dbem')),
665
+ 'location_town' => sprintf(__('Order by %s','dbem'),__('Town','dbem')),
666
+ 'location_name' => sprintf(__('Order by %s','dbem'),__('Name','dbem'))
667
+ ));
668
+ ?>
669
+ <?php foreach($orderby_options as $key => $value) : ?>
670
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_locations_default_orderby')) ? "selected='selected'" : ''; ?>>
671
+ <?php echo $value; ?>
672
+ </option>
673
+ <?php endforeach; ?>
674
+ </select>
675
+ <select name="dbem_locations_default_order" >
676
+ <?php
677
+ $ascending = __('Ascending','dbem');
678
+ $descending = __('Descending','dbem');
679
+ $order_options = apply_filters('em_settings_locations_default_order_ddm', array(
680
+ 'ASC' => __('Ascending','dbem'),
681
+ 'DESC' => __('Descending','dbem')
682
+ ));
683
+ ?>
684
+ <?php foreach( $order_options as $key => $value) : ?>
685
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_locations_default_order')) ? "selected='selected'" : ''; ?>>
686
+ <?php echo $value; ?>
687
+ </option>
688
+ <?php endforeach; ?>
689
+ </select>
690
+ </td>
691
+ </tr>
692
+ <?php
693
+ em_options_input_text ( __( 'List Limits', 'dbem' ), 'dbem_locations_default_limit', sprintf(__( "This will control how many %s are shown on one list by default.", 'dbem' ),__('locations','dbem')) );
694
+ echo $save_button;
695
+ ?>
696
+ </table>
697
+ </div> <!-- . inside -->
698
+ </div> <!-- .postbox -->
699
+ <?php endif; ?>
700
+
701
+ <?php if( get_option('dbem_categories_enabled') && !(EM_MS_GLOBAL && !is_main_blog()) ): ?>
702
+ <div class="postbox " >
703
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo __('Event Categories','dbem'); ?></span></h3>
704
+ <div class="inside">
705
+ <table class="form-table">
706
+ <?php
707
+ $events_page_options[0] = sprintf(__('[No %s Page]', 'dbem'),__('Categories','dbem'));
708
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('Categories','dbem')), 'dbem_categories_page', $events_page_options, sprintf(__( 'This option allows you to select which page to use as the %s page.','dbem' ),__('categories','dbem'),__('categories','dbem')) );
709
+ ?>
710
+ <tbody class="em-category-page-options">
711
+ <?php
712
+ em_options_radio_binary ( sprintf(__( 'Show %s page in lists?', 'dbem' ),__('categories','dbem')), 'dbem_list_categories_page', sprintf(__( 'Check this option if you want the %s page to appear together with other pages in pages lists.', 'dbem' ),__('categories','dbem')) );
713
+ ?>
714
+ </tbody>
715
+ <tr>
716
+ <td colspan="2">
717
+ <h4><?php echo _e('General settings','dbem'); ?></h4>
718
+ </td>
719
+ </tr>
720
+ <?php
721
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_categories_formats', sprintf($format_override_tip,__('categories','dbem').' '. __('Setting this to yes will make categories display as a page rather than an archive.')));
722
+ ?>
723
+ <tr valign="top">
724
+ <th scope="row"><?php _e('Default archive ordering','dbem'); ?></th>
725
+ <td>
726
+ <select name="dbem_categories_default_archive_orderby" >
727
+ <?php foreach($event_archive_orderby_options as $key => $value) : ?>
728
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_categories_default_archive_orderby')) ? "selected='selected'" : ''; ?>>
729
+ <?php echo $value; ?>
730
+ </option>
731
+ <?php endforeach; ?>
732
+ </select>
733
+ <select name="dbem_categories_default_archive_order" >
734
+ <?php foreach( $event_archive_order_options as $key => $value) : ?>
735
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_categories_default_archive_order')) ? "selected='selected'" : ''; ?>>
736
+ <?php echo $value; ?>
737
+ </option>
738
+ <?php endforeach; ?>
739
+ </select>
740
+ </td>
741
+ </tr>
742
+ <tr>
743
+ <td colspan="2">
744
+ <h4><?php echo sprintf(__('Default %s list options','dbem'), __('category','dbem')); ?></h4>
745
+ <p><?php _e('These can be overriden when using shortcode or template tags.','dbem'); ?></p>
746
+ </td>
747
+ </tr>
748
+ <tr valign="top" id='dbem_categories_default_orderby_row'>
749
+ <th scope="row"><?php _e('Default list ordering','dbem'); ?></th>
750
+ <td>
751
+ <select name="dbem_categories_default_orderby" >
752
+ <?php
753
+ $orderby_options = apply_filters('em_settings_categories_default_orderby_ddm', array(
754
+ 'category_country' => sprintf(__('Order by %s','dbem'),__('Country','dbem')),
755
+ 'category_town' => sprintf(__('Order by %s','dbem'),__('Town','dbem')),
756
+ 'category_name' => sprintf(__('Order by %s','dbem'),__('Name','dbem'))
757
+ ));
758
+ ?>
759
+ <?php foreach($orderby_options as $key => $value) : ?>
760
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_categories_default_orderby')) ? "selected='selected'" : ''; ?>>
761
+ <?php echo $value; ?>
762
+ </option>
763
+ <?php endforeach; ?>
764
+ </select>
765
+ <select name="dbem_categories_default_order" >
766
+ <?php
767
+ $ascending = __('Ascending','dbem');
768
+ $descending = __('Descending','dbem');
769
+ $order_options = apply_filters('em_settings_categories_default_order_ddm', array(
770
+ 'ASC' => __('Ascending','dbem'),
771
+ 'DESC' => __('Descending','dbem')
772
+ ));
773
+ ?>
774
+ <?php foreach( $order_options as $key => $value) : ?>
775
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_categories_default_order')) ? "selected='selected'" : ''; ?>>
776
+ <?php echo $value; ?>
777
+ </option>
778
+ <?php endforeach; ?>
779
+ </select>
780
+ </td>
781
+ </tr>
782
+ <?php
783
+ em_options_input_text ( __( 'List Limits', 'dbem' ), 'dbem_categories_default_limit', sprintf(__( "This will control how many %s are shown on one list by default.", 'dbem' ),__('categories','dbem')) );
784
+ echo $save_button;
785
+ ?>
786
+ </table>
787
+ </div> <!-- . inside -->
788
+ </div> <!-- .postbox -->
789
+ <?php endif; ?>
790
+
791
+ <?php if( get_option('dbem_tags_enabled') ): //disabled for now, will add tag stuff later ?>
792
+ <div class="postbox " >
793
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo __('Event Tags','dbem'); ?></span></h3>
794
+ <div class="inside">
795
+ <table class="form-table">
796
+ <?php
797
+ em_options_radio_binary ( __( 'Override with Formats?', 'dbem' ), 'dbem_cp_tags_formats', sprintf($format_override_tip,__('tags','dbem')));
798
+ ?>
799
+ <tr valign="top">
800
+ <th scope="row"><?php _e('Default archive ordering','dbem'); ?></th>
801
+ <td>
802
+ <select name="dbem_tags_default_archive_orderby" >
803
+ <?php foreach($event_archive_orderby_options as $key => $value) : ?>
804
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_tags_default_archive_orderby')) ? "selected='selected'" : ''; ?>>
805
+ <?php echo $value; ?>
806
+ </option>
807
+ <?php endforeach; ?>
808
+ </select>
809
+ <select name="dbem_tags_default_archive_order" >
810
+ <?php foreach( $event_archive_order_options as $key => $value) : ?>
811
+ <option value='<?php echo $key ?>' <?php echo ($key == get_option('dbem_tags_default_archive_order')) ? "selected='selected'" : ''; ?>>
812
+ <?php echo $value; ?>
813
+ </option>
814
+ <?php endforeach; ?>
815
+ </select>
816
+ </td>
817
+ </tr>
818
+ <?php echo $save_button; ?>
819
+ </table>
820
+ </div> <!-- . inside -->
821
+ </div> <!-- .postbox -->
822
+ <?php endif; ?>
823
+
824
+ <div class="postbox " >
825
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__('%s Pages','dbem'),__('Other','dbem')); ?></span></h3>
826
+ <div class="inside">
827
+ <p><?php _e('These pages allow you to provide an event management interface outside the admin area on whatever page you want on your website. Bear in mind that this is overriden by BuddyPress if activated.'); ?></p>
828
+ <table class="form-table">
829
+ <?php
830
+ $other_pages_tip = 'Using the %s shortcode, you can allow users to manage %s outside the admin area.';
831
+ $events_page_options[0] = '['.__('None', 'dbem').']';
832
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('Edit events','dbem')), 'dbem_edit_events_page', $events_page_options, '' );
833
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('Edit locations','dbem')), 'dbem_edit_locations_page', $events_page_options, '' );
834
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('Manage bookings','dbem')), 'dbem_edit_bookings_page', $events_page_options, '' );
835
+ em_options_select ( sprintf(__( '%s page', 'dbem' ),__('My bookings','dbem')), 'dbem_my_bookings_page', $events_page_options, sprintf(__('Users can view their bookings for other events on this page.','dbem' ),'<code>[my_bookings]</code>',__('bookings','dbem')) );
836
+ echo $save_button;
837
+ ?>
838
+ </table>
839
+ </div> <!-- . inside -->
840
+ </div> <!-- .postbox -->
841
+
842
+ </div> <!-- .em-menu-pages -->
843
+
844
+ <!-- FORMAT OPTIONS -->
845
+ <div class="em-menu-formats em-menu-group" style="display:none;">
846
+ <div class="postbox " >
847
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Events format', 'dbem' ); ?> </span></h3>
848
+ <div class="inside">
849
+ <table class="form-table">
850
+ <tr><td><strong><?php echo sprintf(__('%s Page','dbem'),__('Events','dbem')); ?></strong></td></tr>
851
+ <?php
852
+ em_options_textarea ( __( 'Default event list format header', 'dbem' ), 'dbem_event_list_item_format_header', __( 'This content will appear just above your code for the default event list format. Default is blank', 'dbem' ) );
853
+ em_options_textarea ( __( 'Default event list format', 'dbem' ), 'dbem_event_list_item_format', __( 'The format of any events in a list.', 'dbem' ).$events_placeholder_tip );
854
+ em_options_textarea ( __( 'Default event list format footer', 'dbem' ), 'dbem_event_list_item_format_footer', __( 'This content will appear just below your code for the default event list format. Default is blank', 'dbem' ) );
855
+ em_options_input_text ( __( 'No events message', 'dbem' ), 'dbem_no_events_message', __( 'The message displayed when no events are available.', 'dbem' ) );
856
+ em_options_input_text ( __( 'List events by date title', 'dbem' ), 'dbem_list_date_title', __( 'If viewing a page for events on a specific date, this is the title that would show up. To insert date values, use <a href="http://www.php.net/manual/en/function.date.php">PHP time format characters</a> with a <code>#</code> symbol before them, i.e. <code>#m</code>, <code>#M</code>, <code>#j</code>, etc.<br/>', 'dbem' ) );
857
+ ?>
858
+ <tr><td><strong><?php echo sprintf(__('Single %s Page','dbem'),__('Event','dbem')); ?></strong></td></tr>
859
+ <?php
860
+ em_options_input_text ( __( 'Single event page title format', 'dbem' ), 'dbem_event_page_title_format', __( 'The format of a single event page title.', 'dbem' ).$events_placeholder_tip );
861
+ em_options_textarea ( __( 'Default single event format', 'dbem' ), 'dbem_single_event_format', __( 'The format of a single event page.', 'dbem' ).$events_placeholder_tip );
862
+ echo $save_button;
863
+ ?>
864
+ </table>
865
+ </div> <!-- . inside -->
866
+ </div> <!-- .postbox -->
867
+
868
+ <div class="postbox " >
869
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Search Form Options', 'dbem' ); ?> </span></h3>
870
+ <div class="inside">
871
+ <table class="form-table">
872
+ <?php
873
+ em_options_radio_binary ( __( 'Show text search?', 'dbem' ), 'dbem_search_form_text', '' );
874
+ em_options_input_text ( __( 'Text search label', 'dbem' ), 'dbem_search_form_text_label', __('Appears within the input box.','dbem') );
875
+ em_options_radio_binary ( __( 'Show date range?', 'dbem' ), 'dbem_search_form_dates', '' );
876
+ em_options_radio_binary ( __( 'Show categories?', 'dbem' ), 'dbem_search_form_categories', '' );
877
+ em_options_input_text ( __( 'Categories label', 'dbem' ), 'dbem_search_form_categories_label', __('Appears as the first default search option.','dbem') );
878
+ em_options_radio_binary ( __( 'Show countries?', 'dbem' ), 'dbem_search_form_countries', '' );
879
+ em_options_input_text ( __( 'All countries text', 'dbem' ), 'dbem_search_form_countries_label', __('Appears as the first default search option.','dbem') );
880
+ em_options_radio_binary ( __( 'Show regions?', 'dbem' ), 'dbem_search_form_regions', '' );
881
+ em_options_input_text ( __( 'All regions text', 'dbem' ), 'dbem_search_form_regions_label', __('Appears as the first default search option.','dbem') );
882
+ em_options_radio_binary ( __( 'Show states?', 'dbem' ), 'dbem_search_form_states', '' );
883
+ em_options_input_text ( __( 'All states text', 'dbem' ), 'dbem_search_form_states_label', __('Appears as the first default search option.','dbem') );
884
+ em_options_radio_binary ( __( 'Show towns/cities?', 'dbem' ), 'dbem_search_form_towns', '' );
885
+ em_options_input_text ( __( 'All towns/cities text', 'dbem' ), 'dbem_search_form_towns_label', __('Appears as the first default search option.','dbem') );
886
+ echo $save_button;
887
+ ?>
888
+ </table>
889
+ </div> <!-- . inside -->
890
+ </div> <!-- .postbox -->
891
+
892
+ <div class="postbox " >
893
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Date/Time formats', 'dbem' ); ?> </span></h3>
894
+ <div class="inside">
895
+ <p><?php echo sprintf(__('Date and Time formats follow the <a href="%s">WordPress time formatting conventions</a>', 'dbem'), 'http://codex.wordpress.org/Formatting_Date_and_Time'); ?></p>
896
+ <table class="form-table">
897
+ <?php
898
+ em_options_input_text ( __( 'Date Format', 'dbem' ), 'dbem_date_format', sprintf(__('For use with the %s placeholder'),'<code>#_EVENTDATES</code>') );
899
+ em_options_input_text ( __( 'Date Seperator', 'dbem' ), 'dbem_dates_seperator', sprintf(__( 'For when start/end %s are present, this will seperate the two (include spaces here if necessary).', 'dbem' ), __('dates','dbem')) );
900
+ em_options_input_text ( __( 'Time Format', 'dbem' ), 'dbem_time_format', sprintf(__('For use with the %s placeholder'),'<code>#_EVENTTIMES</code>') );
901
+ em_options_input_text ( __( 'Time Seperator', 'dbem' ), 'dbem_times_seperator', sprintf(__( 'For when start/end %s are present, this will seperate the two (include spaces here if necessary).', 'dbem' ), __('times','dbem')) );
902
+ echo $save_button;
903
+ ?>
904
+ </table>
905
+ </div> <!-- . inside -->
906
+ </div> <!-- .postbox -->
907
+
908
+ <div class="postbox " >
909
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Calendar format', 'dbem' ); ?></span></h3>
910
+ <div class="inside">
911
+ <table class="form-table">
912
+ <?php
913
+ em_options_input_text ( __( 'Small calendar title', 'dbem' ), 'dbem_small_calendar_event_title_format', __( 'The format of the title, corresponding to the text that appears when hovering on an eventful calendar day.', 'dbem' ).$events_placeholder_tip );
914
+ em_options_input_text ( __( 'Small calendar title separator', 'dbem' ), 'dbem_small_calendar_event_title_separator', __( 'The separator appearing on the above title when more than one events are taking place on the same day.', 'dbem' ) );
915
+ em_options_input_text ( __( 'Full calendar events format', 'dbem' ), 'dbem_full_calendar_event_format', __( 'The format of each event when displayed in the full calendar. Remember to include <code>li</code> tags before and after the event.', 'dbem' ).$events_placeholder_tip );
916
+ em_options_radio_binary ( __( 'Show long events on calendar pages?', 'dbem' ), 'dbem_full_calendar_long_events', __( "If you are showing a calendar on the events page (see Events format section on this page), you have the option of showing events that span over days on each day it occurs.",'dbem' ) );
917
+ em_options_radio_binary ( __( 'Show list on day with single event?', 'dbem' ), 'dbem_display_calendar_day_single', __( "By default, if a calendar day only has one event, it display a single event when clicking on the link of that calendar date. If you select Yes here, you will get always see a list of events.",'dbem' ) );
918
+ em_options_input_text ( __( 'Full calendar events/day limit', 'dbem' ), 'dbem_full_calendar_events_limit', __( 'To reduce the clutter on your calendar, you can add a limit to events displayed in one day on the calendar.', 'dbem' ) );
919
+ ?>
920
+ <tr><td><strong><?php echo sprintf(__('iCal Feed Settings','dbem'),__('Event','dbem')); ?></strong></td></tr>
921
+ <?php
922
+ em_options_input_text ( __( 'iCal Title', 'dbem' ), 'dbem_ical_description_format', __( 'The title that will appear in the calendar.', 'dbem' ).$events_placeholder_tip );
923
+ em_options_input_text ( __( 'iCal Limit', 'dbem' ), 'dbem_ical_limit', __( 'Limits the number of future events shown (0 = unlimited).', 'dbem' ) );
924
+ echo $save_button;
925
+ ?>
926
+ </table>
927
+ </div> <!-- . inside -->
928
+ </div> <!-- .postbox -->
929
+
930
+ <?php if( get_option('dbem_locations_enabled') ): ?>
931
+ <div class="postbox " >
932
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Locations format', 'dbem' ); ?> </span></h3>
933
+ <div class="inside">
934
+ <table class="form-table">
935
+ <tr><td><strong><?php echo sprintf(__('%s Page','dbem'),__('Locations','dbem')); ?></strong></td></tr>
936
+ <?php
937
+ em_options_textarea ( sprintf(__('%s list header format','dbem'),__('Locations','dbem')), 'dbem_location_list_item_format_header', sprintf(__( 'This content will appear just above your code for the %s list format below. Default is blank', 'dbem' ), __('locations','dbem')) );
938
+ em_options_textarea ( sprintf(__('%s list item format','dbem'),__('Locations','dbem')), 'dbem_location_list_item_format', sprintf(__( 'The format of a single %s in a list.', 'dbem' ), __('locations','dbem')).$locations_placeholder_tip );
939
+ em_options_textarea ( sprintf(__('%s list footer format','dbem'),__('Locations','dbem')), 'dbem_location_list_item_format_footer', sprintf(__( 'This content will appear just below your code for the %s list format above. Default is blank', 'dbem' ), __('locations','dbem')) );
940
+ em_options_input_text ( sprintf(__( 'No %s message', 'dbem' ),__('Locations','dbem')), 'dbem_no_locations_message', sprintf( __( 'The message displayed when no %s are available.', 'dbem' ), __('locations','dbem')) );
941
+ ?>
942
+ <tr><td><strong><?php echo sprintf(__('Single %s Page','dbem'),__('Location','dbem')); ?></strong></td></tr>
943
+ <?php
944
+ em_options_input_text (sprintf( __( 'Single %s title format', 'dbem' ),__('location','dbem')), 'dbem_location_page_title_format', __( 'The format of a single location page title.', 'dbem' ).$locationscategoriesder_tip );
945
+ em_options_textarea ( sprintf(__('Single %s page format', 'dbem' ),__('location','dbem')), 'dbem_single_location_format', __( 'The format of a single location page.', 'dbem' ).$locations_placeholder_tip );
946
+ em_options_textarea ( __( 'Default location balloon format', 'dbem' ), 'dbem_location_baloon_format', __( 'The format of of the text appearing in the baloon describing the location a single location map.', 'dbem' ).$locations_placeholder_tip );
947
+ ?>
948
+ <tr><td><strong><?php echo sprintf(__('%s List Formats','dbem'),__('Event','dbem')); ?></strong></td></tr>
949
+ <?php
950
+ em_options_textarea ( sprintf(__( 'Default %s list format', 'dbem' ),__('events','dbem')), 'dbem_location_event_list_item_format', __( 'The format of the events the list inserted in the location page through the <code>#_LOCATIONNEXTEVENTS</code>, <code>#_LOCATIONNEXTEVENTS</code> and <code>#_LOCATIONALLEVENTS</code> element.', 'dbem' ).$locations_placeholder_tip );
951
+ em_options_textarea ( sprintf(__( 'No %s message', 'dbem' ),__('events','dbem')), 'dbem_location_no_events_message', __( 'The message to be displayed in the list generated by <code>#_LOCATIONNEXTEVENTS</code>, <code>#_LOCATIONNEXTEVENTS</code> and <code>#_LOCATIONALLEVENTS</code> when no events are available.', 'dbem' ) );
952
+ echo $save_button;
953
+ ?>
954
+ </table>
955
+ </div> <!-- . inside -->
956
+ </div> <!-- .postbox -->
957
+ <?php endif; ?>
958
+
959
+ <?php if( get_option('dbem_categories_enabled') && !(EM_MS_GLOBAL && !is_main_blog()) ): ?>
960
+ <div class="postbox " >
961
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Categories format', 'dbem' ); ?> </span></h3>
962
+ <div class="inside">
963
+ <table class="form-table">
964
+ <tr><td><strong><?php echo sprintf(__('%s Page','dbem'),__('Categories','dbem')); ?></strong></td></tr>
965
+ <?php
966
+ em_options_textarea ( sprintf(__('%s list header format','dbem'),__('Categories','dbem')), 'dbem_categories_list_item_format_header', sprintf(__( 'This content will appear just above your code for the %s list format below. Default is blank', 'dbem' ), __('categories','dbem')) );
967
+ em_options_textarea ( sprintf(__('%s list item format','dbem'),__('Categories','dbem')), 'dbem_categories_list_item_format', sprintf(__( 'The format of a single %s in a list.', 'dbem' ), __('categories','dbem')).$categories_placeholder_tip );
968
+ em_options_textarea ( sprintf(__('%s list footer format','dbem'),__('Categories','dbem')), 'dbem_categories_list_item_format_footer', sprintf(__( 'This content will appear just below your code for the %s list format above. Default is blank', 'dbem' ), __('categories','dbem')) );
969
+ em_options_input_text ( sprintf(__( 'No %s message', 'dbem' ),__('Categories','dbem')), 'dbem_no_categories_message', sprintf( __( 'The message displayed when no %s are available.', 'dbem' ), __('categories','dbem')) );
970
+ ?>
971
+ <tr><td><strong><?php echo sprintf(__('Single %s Page','dbem'),__('Category','dbem')); ?></strong></td></tr>
972
+ <?php
973
+ em_options_input_text ( sprintf(__( 'Single %s title format', 'dbem' ),__('category','dbem')), 'dbem_category_page_title_format', __( 'The format of a single category page title.', 'dbem' ).$categories_placeholder_tip );
974
+ em_options_textarea ( sprintf(__('Single %s page format', 'dbem' ),__('category','dbem')), 'dbem_category_page_format', __( 'The format of a single category page.', 'dbem' ).$categories_placeholder_tip );
975
+ ?>
976
+ <tr><td><strong><?php echo sprintf(__('%s List Formats','dbem'),__('Event','dbem')); ?></strong></td></tr>
977
+ <?php
978
+ em_options_textarea ( sprintf(__( 'Default %s list format', 'dbem' ),__('events','dbem')), 'dbem_category_event_list_item_format', __( 'The format of the events the list inserted in the category page through the <code>#_CATEGORYNEXTEVENTS</code>, <code>#_CATEGORYNEXTEVENTS</code> and <code>#_CATEGORYALLEVENTS</code> element.', 'dbem' ).$categories_placeholder_tip );
979
+ em_options_textarea ( sprintf(__( 'No %s message', 'dbem' ),__('events','dbem')), 'dbem_category_no_events_message', __( 'The message to be displayed in the list generated by <code>#_CATEGORYNEXTEVENTS</code>, <code>#_CATEGORYNEXTEVENTS</code> and <code>#_CATEGORYALLEVENTS</code> when no events are available.', 'dbem' ) );
980
+ echo $save_button;
981
+ ?>
982
+ </table>
983
+ </div> <!-- . inside -->
984
+ </div> <!-- .postbox -->
985
+ <?php endif; ?>
986
+
987
+ <?php if( get_option('dbem_tags_enabled') ): ?>
988
+ <div class="postbox " >
989
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Tags format', 'dbem' ); ?> </span></h3>
990
+ <div class="inside">
991
+ <table class="form-table">
992
+ <tr><td><strong><?php echo sprintf(__('Single %s Page','dbem'),__('Tag','dbem')); ?></strong></td></tr>
993
+ <?php
994
+ em_options_input_text ( sprintf(__( 'Single %s title format', 'dbem' ),__('tag','dbem')), 'dbem_tag_page_title_format', __( 'The format of a single tag page title.', 'dbem' ).$categories_placeholder_tip );
995
+ em_options_textarea ( sprintf(__('Single %s page format', 'dbem' ),__('tag','dbem')), 'dbem_tag_page_format', __( 'The format of a single tag page.', 'dbem' ).$categories_placeholder_tip );
996
+ ?>
997
+ <tr><td><strong><?php echo sprintf(__('%s List Formats','dbem'),__('Event','dbem')); ?></strong></td></tr>
998
+ <?php
999
+ em_options_textarea ( sprintf(__( 'Default %s list format', 'dbem' ),__('events','dbem')), 'dbem_tag_event_list_item_format', __( 'The format of the events the list inserted in the tag page through the <code>#_TAGNEXTEVENTS</code>, <code>#_TAGNEXTEVENTS</code> and <code>#_TAGALLEVENTS</code> element.', 'dbem' ).$categories_placeholder_tip );
1000
+ em_options_textarea ( sprintf(__( 'No %s message', 'dbem' ),__('events','dbem')), 'dbem_tag_no_events_message', __( 'The message to be displayed in the list generated by <code>#_TAGNEXTEVENTS</code>, <code>#_TAGNEXTEVENTS</code> and <code>#_TAGALLEVENTS</code> when no events are available.', 'dbem' ) );
1001
+ echo $save_button;
1002
+ ?>
1003
+ </table>
1004
+ </div> <!-- . inside -->
1005
+ </div> <!-- .postbox -->
1006
+ <?php endif; ?>
1007
+
1008
+ <div class="postbox " >
1009
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'RSS feed format', 'dbem' ); ?> </span></h3>
1010
+ <div class="inside">
1011
+ <table class="form-table">
1012
+ <?php
1013
+ em_options_input_text ( __( 'RSS main title', 'dbem' ), 'dbem_rss_main_title', __( 'The main title of your RSS events feed.', 'dbem' ).$events_placeholder_tip );
1014
+ em_options_input_text ( __( 'RSS main description', 'dbem' ), 'dbem_rss_main_description', __( 'The main description of your RSS events feed.', 'dbem' ) );
1015
+ em_options_input_text ( __( 'RSS title format', 'dbem' ), 'dbem_rss_title_format', __( 'The format of the title of each item in the events RSS feed.', 'dbem' ).$events_placeholder_tip );
1016
+ em_options_input_text ( __( 'RSS description format', 'dbem' ), 'dbem_rss_description_format', __( 'The format of the description of each item in the events RSS feed.', 'dbem' ).$events_placeholder_tip );
1017
+ echo $save_button;
1018
+ ?>
1019
+ </table>
1020
+ </div> <!-- . inside -->
1021
+ </div> <!-- .postbox -->
1022
+
1023
+ <div class="postbox " >
1024
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Maps and geotagging', 'dbem' ); ?> </span></h3>
1025
+ <div class="inside">
1026
+ <table class='form-table'>
1027
+ <?php $gmap_is_active = get_option ( 'dbem_gmap_is_active' ); ?>
1028
+ <tr valign="top">
1029
+ <th scope="row"><?php _e ( 'Enable Google Maps integration?', 'dbem' ); ?></th>
1030
+ <td>
1031
+ <?php _e ( 'Yes' ); ?> <input id="dbem_gmap_is_active_yes" name="dbem_gmap_is_active" type="radio" value="1" <?php echo ($gmap_is_active) ? "checked='checked'":''; ?> />
1032
+ <?php _e ( 'No' ); ?> <input name="dbem_gmap_is_active" type="radio" value="0" <?php echo ($gmap_is_active) ? '':"checked='checked'"; ?> /><br />
1033
+ <em><?php _e ( 'Check this option to enable Goggle Map integration.', 'dbem' )?></em>
1034
+ </td>
1035
+ </tr>
1036
+ <?php
1037
+ em_options_textarea ( __( 'Map text format', 'dbem' ), 'dbem_map_text_format', __( 'The text format inside the map balloons.', 'dbem' ).$events_placeholder_tip );
1038
+ echo $save_button;
1039
+ ?>
1040
+ </table>
1041
+ </div> <!-- . inside -->
1042
+ </div> <!-- .postbox -->
1043
+
1044
+ </div> <!-- .em-menu-formats -->
1045
+
1046
+ <?php if( get_option('dbem_rsvp_enabled') ): ?>
1047
+ <!-- BOOKING OPTIONS -->
1048
+ <div class="em-menu-bookings em-menu-group" style="display:none;">
1049
+
1050
+ <div class="postbox " >
1051
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__( '%s Options', 'dbem' ),__('General','dbem')); ?> </span></h3>
1052
+ <div class="inside">
1053
+ <table class='form-table'>
1054
+ <?php
1055
+ em_options_radio_binary ( __( 'Approval Required?', 'dbem' ), 'dbem_bookings_approval', __( 'Bookings will not be confirmed until the event administrator approves it.', 'dbem' ) );
1056
+ em_options_radio_binary ( __( 'Reserved unconfirmed spaces?', 'dbem' ), 'dbem_bookings_approval_reserved', __( 'By default, event spaces become unavailable once there are enough CONFIRMED bookings. To reserve spaces even if unnapproved, choose yes.', 'dbem' ) );
1057
+ em_options_radio_binary ( __( 'Can users cancel their booking?', 'dbem' ), 'dbem_bookings_user_cancellation', __( 'If enabled, users can cancel their bookings themselves from their bookings page.', 'dbem' ) );
1058
+ em_options_radio_binary ( __( 'Allow overbooking when approving?', 'dbem' ), 'dbem_bookings_approval_overbooking', __( 'If you get a lot of pending bookings and you decide to allow more bookings than spaces allow, setting this to yes will allow you to override the event space limit when manually approving.', 'dbem' ) );
1059
+ em_options_radio_binary ( __( 'Allow double bookings?', 'dbem' ), 'dbem_bookings_double', __( 'If enabled, users can book an event more than once.', 'dbem' ) );
1060
+ echo $save_button;
1061
+ ?>
1062
+ </table>
1063
+ </div> <!-- . inside -->
1064
+ </div> <!-- .postbox -->
1065
+
1066
+ <div class="postbox " >
1067
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__( '%s Options', 'dbem' ),__('Pricing','dbem')); ?> </span></h3>
1068
+ <div class="inside">
1069
+ <table class='form-table'>
1070
+ <?php
1071
+ /* Tax & Currency */
1072
+ em_options_select ( __( 'Currency', 'dbem' ), 'dbem_bookings_currency', em_get_currencies()->names, __( 'Choose your currency for displaying event pricing.', 'dbem' ) );
1073
+ em_options_input_text ( __( 'Tax Rate', 'dbem' ), 'dbem_bookings_tax', __( 'Add a tax rate to your ticket prices (entering 10 will add 10% to the ticket price).', 'dbem' ) );
1074
+ em_options_radio_binary ( __( 'Add tax to ticket price?', 'dbem' ), 'dbem_bookings_tax_auto_add', __( 'When displaying ticket prices and booking totals, include the tax automatically?', 'dbem' ) );
1075
+
1076
+ echo $save_button;
1077
+ ?>
1078
+ </table>
1079
+ </div> <!-- . inside -->
1080
+ </div> <!-- .postbox -->
1081
+
1082
+ <div class="postbox " >
1083
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__( '%s Options', 'dbem' ),__('Booking Form','dbem')); ?> </span></h3>
1084
+ <div class="inside">
1085
+ <table class='form-table'>
1086
+ <?php
1087
+ em_options_radio_binary ( __( 'Allow guest bookings?', 'dbem' ), 'dbem_bookings_anonymous', __( 'If enabled, guest visitors can supply an email address and a user account will automatically be created for them along with their booking. They will be also be able to log back in with that newly created account.', 'dbem' ) );
1088
+ em_options_radio_binary ( __( 'Display login form?', 'dbem' ), 'dbem_bookings_login_form', __( 'Choose whether or not to display a login form in the booking form area to remind your members to log in before booking.', 'dbem' ) );
1089
+ ?>
1090
+ <tr><td colspan='2'><h4><?php _e('Booking form feedback messages','dbem') ?></h4></td></tr>
1091
+ <tr><td colspan='2'><?php _e('When a booking is made by a user, a feedback message is shown depending on the result, which can be customized below.','dbem'); ?></td></tr>
1092
+ <?php
1093
+ em_options_input_text ( __( 'Successful booking', 'dbem' ), 'dbem_booking_feedback', __( 'When a booking is registered and confirmed.', 'dbem' ) );
1094
+ em_options_input_text ( __( 'Successful pending booking', 'dbem' ), 'dbem_booking_feedback_pending', __( 'When a booking is registered but pending.', 'dbem' ) );
1095
+ em_options_input_text ( __( 'Not enough spaces', 'dbem' ), 'dbem_booking_feedback_full', __( 'When a booking cannot be made due to lack of spaces.', 'dbem' ) );
1096
+ em_options_input_text ( __( 'Errors', 'dbem' ), 'dbem_booking_feedback_error', __( 'When a booking cannot be made due to an error when filling the form. Below this, there will be a dynamic list of errors.', 'dbem' ) );
1097
+ em_options_input_text ( __( 'User must log in', 'dbem' ), 'dbem_booking_feedback_log_in', __( 'When a user must log in before making a booking.', 'dbem' ) );
1098
+ em_options_input_text ( __( 'Error mailing user', 'dbem' ), 'dbem_booking_feedback_nomail', __( 'If a booking is made and an email cannot be sent, this is added to the success message.', 'dbem' ) );
1099
+ em_options_input_text ( __( 'Already booked', 'dbem' ), 'dbem_booking_feedback_already_booked', __( 'If the user made a previous booking and cannot double-book.', 'dbem' ) );
1100
+ em_options_input_text ( __( 'No spaces booked', 'dbem' ), 'dbem_booking_feedback_min_space', __( 'If the user tries to make a booking without requesting any spaces.', 'dbem' ) );
1101
+ echo $save_button;
1102
+ ?>
1103
+ </table>
1104
+ </div> <!-- . inside -->
1105
+ </div> <!-- .postbox -->
1106
+
1107
+ <div class="postbox " >
1108
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php echo sprintf(__( '%s Options', 'dbem' ),__('Ticket','dbem')); ?> </span></h3>
1109
+ <div class="inside">
1110
+ <table class='form-table'>
1111
+ <?php
1112
+ em_options_radio_binary ( __( 'Single ticket mode?', 'dbem' ), 'dbem_bookings_tickets_single', __( 'In single ticket mode, users can only create one ticket per booking (and will not see options to add more tickets).', 'dbem' ) );
1113
+ em_options_radio_binary ( __( 'Show ticket table in single ticket mode?', 'dbem' ), 'dbem_bookings_tickets_single_form', __( 'If you prefer a ticket table like with multiple tickets, even for single ticket events, enable this.', 'dbem' ) );
1114
+ em_options_radio_binary ( __( 'Show unavailable tickets?', 'dbem' ), 'dbem_bookings_tickets_show_unavailable', __( 'You can choose whether or not to show unavailable tickets to visitors.', 'dbem' ) );
1115
+ em_options_radio_binary ( __( 'Show multiple tickets if logged out?', 'dbem' ), 'dbem_bookings_tickets_show_loggedout', __( 'If logged out, a user will be asked to register in order to book. However, we can show available tickets if you have more than one ticket.', 'dbem' ) );
1116
+ $ticket_orders = array(
1117
+ 'ticket_price DESC, ticket_name ASC'=>__('Ticket Price (Descending)','dbem'),
1118
+ 'ticket_price ASC, ticket_name ASC'=>__('Ticket Price (Ascending)','dbem'),
1119
+ 'ticket_name DESC, ticket_price DESC'=>__('Ticket Name (Ascending)','dbem'),
1120
+ 'ticket_name ASC, ticket_price DESC'=>__('Ticket Name (Descending)','dbem')
1121
+ );
1122
+ em_options_select ( __( 'Order Tickets By', 'dbem' ), 'dbem_bookings_tickets_orderby', $ticket_orders, __( 'Choose which order your tickets appear.', 'dbem' ) );
1123
+ echo $save_button;
1124
+ ?>
1125
+ </table>
1126
+ </div> <!-- . inside -->
1127
+ </div> <!-- .postbox -->
1128
+
1129
+ <div class="postbox " >
1130
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e('No-User Booking Mode','dbem'); ?> </span></h3>
1131
+ <div class="inside">
1132
+ <table class='form-table'>
1133
+ <tr><td colspan='2'>
1134
+ <p><?php _e('By default, when a booking is made by a user, this booking is tied to a user account, if the user is not registered nor logged in and guest bookings are enabled, an account will be created for them.','dbem'); ?></p>
1135
+ <p><?php _e('The option below allows you to disable user accounts and assign all bookings to a parent user, yet you will still see the supplied booking personal information for each booking. When this mode is enabled, extra booking information about the person is stored alongside the booking record rather than as a WordPress user.','dbem'); ?></p>
1136
+ <p><?php _e('<strong>Warning : </strong> Various features afforded to users with an account will not be available, e.g. viewing bookings. Once you enable this and select a user, modifying these values will prevent older non-user bookings from displaying the correct information.','dbem'); ?></p>
1137
+ </td></tr>
1138
+ <?php
1139
+ em_options_radio_binary ( __( 'Enable No-User Booking Mode?', 'dbem' ), 'dbem_bookings_registration_disable', __( 'This disables user registrations for bookings.', 'dbem' ) );
1140
+ $current_user = array();
1141
+ if( get_option('dbem_bookings_registration_user') ){
1142
+ $user = get_user_by('id',get_option('dbem_bookings_registration_user'));
1143
+ $current_user[$user->ID] = $user->display_name;
1144
+ }
1145
+ em_options_select ( __( 'Assign bookings to', 'dbem' ), 'dbem_bookings_registration_user', em_get_wp_users(array('role' => 'subscriber'), $current_user), __( 'Choose a parent user to assign bookings to. People making their booking will be unaware of this and will never have access to those user details. This should be a subscriber user you do not use to log in with yourself.', 'dbem' ) );
1146
+ echo $save_button;
1147
+ ?>
1148
+ </table>
1149
+ </div> <!-- . inside -->
1150
+ </div> <!-- .postbox -->
1151
+
1152
+ </div> <!-- .em-menu-bookings -->
1153
+ <?php endif; ?>
1154
+
1155
+ <!-- EMAIL OPTIONS -->
1156
+ <div class="em-menu-emails em-menu-group" style="display:none;">
1157
+
1158
+ <?php if ( !is_multisite() ) { em_admin_option_box_email(); } ?>
1159
+
1160
+ <?php if( get_option('dbem_rsvp_enabled') ): ?>
1161
+ <div class="postbox " >
1162
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Booking Email Templates', 'dbem' ); ?> </span></h3>
1163
+ <div class="inside">
1164
+ <table class='form-table'>
1165
+ <?php
1166
+ em_options_select ( __( 'Default contact person', 'dbem' ), 'dbem_default_contact_person', em_get_wp_users(), __( 'Select the default contact person. This user will be employed whenever a contact person is not explicitly specified for an event', 'dbem' ) );
1167
+ em_options_input_text ( __( 'Email events admin?', 'dbem' ), 'dbem_bookings_notify_admin', __( "If you would like every event booking confirmation email sent to an administrator write their email here (leave blank to not send an email).", 'dbem' ) );
1168
+ em_options_radio_binary ( __( 'Email contact person?', 'dbem' ), 'dbem_bookings_contact_email', __( 'Check this option if you want the event contact to receive an email when someone books places. An email will be sent when a booking is first made (regardless if confirmed or pending)', 'dbem' ) );
1169
+ em_options_radio_binary ( __( 'Disable new registration email?', 'dbem' ), 'dbem_email_disable_registration', __( 'Check this option if you want to prevent the WordPress registration email from going out when a user anonymously books an event.', 'dbem' ) );
1170
+ ?>
1171
+ <tr><td colspan='2'><h4><?php _e('Contact person booking confirmed','dbem') ?></h4></td></tr>
1172
+ <tr><td colspan='2'><?php echo __('An email will be sent to the event contact when a booking is first made.','dbem').$bookings_placeholder_tip ?></td></tr>
1173
+ <?php
1174
+ em_options_input_text ( __( 'Contact person email subject', 'dbem' ), 'dbem_bookings_contact_email_subject', '' );
1175
+ em_options_textarea ( __( 'Contact person email', 'dbem' ), 'dbem_bookings_contact_email_body', '' );
1176
+ ?>
1177
+ <tr><td colspan='2'><h4><?php _e('Contact person booking cancelled','dbem') ?></h4></td></tr>
1178
+ <tr><td colspan='2'><?php echo __('An email will be sent to the event contact if someone cancels their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
1179
+ <?php
1180
+ em_options_input_text ( __( 'Contact person cancellation subject', 'dbem' ), 'dbem_contactperson_email_cancelled_subject', '' );
1181
+ em_options_textarea ( __( 'Contact person cancellation email', 'dbem' ), 'dbem_contactperson_email_cancelled_body', '' );
1182
+ ?>
1183
+ <tr><td colspan='2'><h4><?php _e('Confirmed booking email','dbem') ?></h4></td></tr>
1184
+ <tr><td colspan='2'><?php echo __('This is sent when a person\'s booking is confirmed. This will be sent automatically if approvals are required and the booking is approved. If approvals are disabled, this is sent out when a user first submits their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
1185
+ <?php
1186
+ em_options_input_text ( __( 'Booking confirmed email subject', 'dbem' ), 'dbem_bookings_email_confirmed_subject', '' );
1187
+ em_options_textarea ( __( 'Booking confirmed email', 'dbem' ), 'dbem_bookings_email_confirmed_body', '' );
1188
+ ?>
1189
+ <tr><td colspan='2'><h4><?php _e('Pending booking email','dbem') ?></h4></td></tr>
1190
+ <tr><td colspan='2'><?php echo __( 'This will be sent to the person when they first submit their booking. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip ?></td></tr>
1191
+ <?php
1192
+ em_options_input_text ( __( 'Booking pending email subject', 'dbem' ), 'dbem_bookings_email_pending_subject', '');
1193
+ em_options_textarea ( __( 'Booking pending email', 'dbem' ), 'dbem_bookings_email_pending_body','') ;
1194
+ ?>
1195
+ <tr><td colspan='2'><h4><?php _e('Rejected booking email','dbem') ?></h4></td></tr>
1196
+ <tr><td colspan='2'><?php echo __( 'This will be sent automatically when a booking is rejected. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip ?></td></tr>
1197
+ <?php
1198
+ em_options_input_text ( __( 'Booking rejected email subject', 'dbem' ), 'dbem_bookings_email_rejected_subject', __( "The subject of the email sent to the person making a booking that is awaiting administrator approval. Not relevant if bookings don't require approval.", 'dbem' ).$bookings_placeholder_tip );
1199
+ em_options_textarea ( __( 'Booking rejected email', 'dbem' ), 'dbem_bookings_email_rejected_body', __( 'The body of the email which will be sent to the person if the booking is rejected. Not relevant if bookings don\'t require approval.', 'dbem' ).$bookings_placeholder_tip );
1200
+ ?>
1201
+ <tr><td colspan='2'><h4><?php _e('Booking cancelled','dbem') ?></h4></td></tr>
1202
+ <tr><td colspan='2'><?php echo __('This will be sent when a user cancels their booking.','dbem').$bookings_placeholder_tip ?></td></tr>
1203
+ <?php
1204
+ em_options_input_text ( __( 'Booking cancelled email subject', 'dbem' ), 'dbem_bookings_email_cancelled_subject', '' );
1205
+ em_options_textarea ( __( 'Booking cancelled email', 'dbem' ), 'dbem_bookings_email_cancelled_body', '' );
1206
+ ?>
1207
+ <?php echo $save_button; ?>
1208
+ </table>
1209
+ </div> <!-- . inside -->
1210
+ </div> <!-- .postbox -->
1211
+ <?php endif; ?>
1212
+
1213
+ <div class="postbox " >
1214
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Event Email Templates', 'dbem' ); ?> </span></h3>
1215
+ <div class="inside">
1216
+ <table class='form-table'>
1217
+ <tr><td colspan='2'><strong><?php _e('Event Submitted','dbem') ?></strong></td></tr>
1218
+ <tr><td colspan='2'><?php echo __('An email will be sent to the an administrator of your choice when an event is submitted and pending approval.','dbem').$bookings_placeholder_tip ?></td></tr>
1219
+ <?php
1220
+ em_options_input_text ( __( 'Administrator Email', 'dbem' ), 'dbem_event_submitted_email_admin', __('If left blank, no email will be sent. Seperate emails with commas for more than one email.','dbem') );
1221
+ em_options_input_text ( __( 'Event approved subject', 'dbem' ), 'dbem_event_submitted_email_subject', '' );
1222
+ em_options_textarea ( __( 'Event approved email', 'dbem' ), 'dbem_event_submitted_email_body', '' );
1223
+ ?>
1224
+ <tr><td colspan='2'><strong><?php _e('Event Approved','dbem') ?></strong></td></tr>
1225
+ <tr><td colspan='2'><?php echo __('An email will be sent to the event owner when their event is approved. Users requiring event approval do not have the <code>publish_events</code> capability.','dbem').$bookings_placeholder_tip ?></td></tr>
1226
+ <?php
1227
+ em_options_input_text ( __( 'Event approved subject', 'dbem' ), 'dbem_event_approved_email_subject', '' );
1228
+ em_options_textarea ( __( 'Event approved email', 'dbem' ), 'dbem_event_approved_email_body', '' );
1229
+ ?>
1230
+ <?php echo $save_button; ?>
1231
+ </table>
1232
+ </div> <!-- . inside -->
1233
+ </div> <!-- .postbox -->
1234
+
1235
+ </div><!-- .em-group-emails -->
1236
+ <?php /*
1237
+ <div class="postbox " >
1238
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Debug Modes', 'dbem' ); ?> </span></h3>
1239
+ <div class="inside">
1240
+ <table class='form-table'>
1241
+ <?php
1242
+ em_options_radio_binary ( __( 'EM Debug Mode?', 'dbem' ), 'dbem_debug', __( 'Setting this to yes will display different content to admins for event pages and emails so you can see all the available placeholders and their values.', 'dbem' ) );
1243
+ em_options_radio_binary ( __( 'WP Debug Mode?', 'dbem' ), 'dbem_wp_debug', __( 'This will turn WP_DEBUG mode on. Useful if you want to troubleshoot php errors without looking at your logs.', 'dbem' ) );
1244
+ ?>
1245
+ </table>
1246
+ </div> <!-- . inside -->
1247
+ </div> <!-- .postbox -->
1248
+ */ ?>
1249
+
1250
+ <p class="submit">
1251
+ <input type="submit" id="dbem_options_submit" name="Submit" value="<?php _e ( 'Save Changes' )?>" />
1252
+ <input type="hidden" name="em-submitted" value="1" />
1253
+ <input type="hidden" name="_wpnonce" value="<?php echo wp_create_nonce('events-manager-options'); ?>" />
1254
+ </p>
1255
+
1256
+ </div> <!-- .metabox-sortables -->
1257
+ </div> <!-- .postbox-container -->
1258
+
1259
+ </div> <!-- .metabox-holder -->
1260
+ </form>
1261
+ </div>
1262
+ <?php
1263
+ }
1264
+
1265
+ /**
1266
+ * Meta options box for anonymous events. Shared in both MS and Normal options page, hence it's own function
1267
+ */
1268
+ function em_admin_option_box_anon_events(){
1269
+ global $save_button, $events_placeholder_tip;
1270
+ ?>
1271
+ <div class="postbox" >
1272
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Event Submission Forms', 'dbem' ); ?></span></h3>
1273
+ <div class="inside">
1274
+ <table class="form-table">
1275
+ <tr><td colspan="2">
1276
+ <?php echo sprintf(__('You can allow users to publicly submit events on your blog by using the %s shortcode, and enabling anonymous submissions below.','dbem'), '<code>[event_form]</code>'); ?>
1277
+ </td></tr>
1278
+ <?php
1279
+ em_options_radio_binary ( __( 'Use Visual Editor?', 'dbem' ), 'dbem_events_form_editor', __( 'Users can now use the WordPress editor for easy HTML entry in the submission form.', 'dbem' ) );
1280
+ em_options_radio_binary ( __( 'Show form again?', 'dbem' ), 'dbem_events_form_reshow', __( 'When a user submits their event, you can display a new event form again.', 'dbem' ) );
1281
+ em_options_textarea ( __( 'Success Message', 'dbem' ), 'dbem_events_form_result_success', __( 'Customize the message your user sees when they submitted their event.', 'dbem' ).$events_placeholder_tip );
1282
+ ?>
1283
+ <tr><td colspan="2">
1284
+ <strong><?php echo sprintf(__('Anonymous event submissions','dbem'), '<code>[event_form]</code>'); ?></strong>
1285
+ </td></tr>
1286
+ <?php
1287
+ em_options_radio_binary ( __( 'Allow anonymous event submissions?', 'dbem' ), 'dbem_events_anonymous_submissions', __( 'Would you like to allow users to submit bookings anonymously? If so, you can use the new [event_form] shortcode or <code>em_event_form()</code> template tag with this enabled.', 'dbem' ) );
1288
+ em_options_select ( __('Guest Default User', 'dbem'), 'dbem_events_anonymous_user', em_get_wp_users (), __( 'Events require a user to own them. In order to allow events to be submitted anonymously you need to assign that event a specific user. We recommend you create a "Anonymous" subscriber with a very good password and use that. Guests will have the same event permissions as this user when submitting.', 'dbem' ) );
1289
+ em_options_textarea ( __( 'Success Message', 'dbem' ), 'dbem_events_anonymous_result_success', __( 'Anonymous submitters cannot see or modify their event once submitted. You can customize the success message they see here.', 'dbem' ).$events_placeholder_tip );
1290
+ ?>
1291
+ <?php echo $save_button; ?>
1292
+ </table>
1293
+ </div> <!-- . inside -->
1294
+ </div> <!-- .postbox -->
1295
+ <?php
1296
+ }
1297
+
1298
+ /**
1299
+ * Meta options box for image sizes. Shared in both MS and Normal options page, hence it's own function
1300
+ */
1301
+ function em_admin_option_box_image_sizes(){
1302
+ global $save_button;
1303
+ ?>
1304
+ <div class="postbox " >
1305
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Image Sizes', 'dbem' ); ?> </span></h3>
1306
+ <div class="inside">
1307
+ <table class='form-table'>
1308
+ <?php
1309
+ em_options_input_text ( __( 'Maximum width (px)', 'dbem' ), 'dbem_image_max_width', __( 'The maximum allowed width for images uploades', 'dbem' ) );
1310
+ em_options_input_text ( __( 'Maximum height (px)', 'dbem' ), 'dbem_image_max_height', __( "The maximum allowed height for images uploaded, in pixels", 'dbem' ) );
1311
+ em_options_input_text ( __( 'Maximum size (bytes)', 'dbem' ), 'dbem_image_max_size', __( "The maximum allowed size for images uploaded, in bytes", 'dbem' ) );
1312
+ echo $save_button;
1313
+ ?>
1314
+ </table>
1315
+ </div> <!-- . inside -->
1316
+ </div> <!-- .postbox -->
1317
+ <?php
1318
+ }
1319
+
1320
+ /**
1321
+ * Meta options box for email settings. Shared in both MS and Normal options page, hence it's own function
1322
+ */
1323
+ function em_admin_option_box_email(){
1324
+ global $save_button;
1325
+ ?>
1326
+ <div class="postbox " >
1327
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Email Settings', 'dbem' ); ?></span></h3>
1328
+ <div class="inside">
1329
+ <table class='form-table'>
1330
+ <?php
1331
+ em_options_radio_binary ( __( 'Send HTML Emails?', 'dbem' ), 'dbem_smtp_html', __( 'If set to yes, your emails will be sent in HTML format, otherwise plaintext.', 'dbem' ) );
1332
+ em_options_input_text ( __( 'Notification sender name', 'dbem' ), 'dbem_mail_sender_name', __( "Insert the display name of the notification sender.", 'dbem' ) );
1333
+ em_options_input_text ( __( 'Notification sender address', 'dbem' ), 'dbem_mail_sender_address', __( "Insert the address of the notification sender.", 'dbem' ) );
1334
+ em_options_input_text ( 'Mail sending port', 'dbem_rsvp_mail_port', __( "The port through which you e-mail notifications will be sent. Make sure the firewall doesn't block this port", 'dbem' ) );
1335
+ em_options_select ( __( 'Mail sending method', 'dbem' ), 'dbem_rsvp_mail_send_method', array ('smtp' => 'SMTP', 'mail' => __( 'PHP mail function', 'dbem' ), 'sendmail' => 'Sendmail', 'qmail' => 'Qmail', 'wp_mail' => 'WP Mail' ), __( 'Select the method to send email notification.', 'dbem' ) );
1336
+ em_options_radio_binary ( __( 'Use SMTP authentication?', 'dbem' ), 'dbem_rsvp_mail_SMTPAuth', __( 'SMTP authentication is often needed. If you use GMail, make sure to set this parameter to Yes', 'dbem' ) );
1337
+ em_options_input_text ( 'SMTP host', 'dbem_smtp_host', __( "The SMTP host. Usually it corresponds to 'localhost'. If you use GMail, set this value to 'ssl://smtp.gmail.com:465'.", 'dbem' ) );
1338
+ em_options_input_text ( __( 'SMTP username', 'dbem' ), 'dbem_smtp_username', __( "Insert the username to be used to access your SMTP server.", 'dbem' ) );
1339
+ em_options_input_password ( __( 'SMTP password', 'dbem' ), "dbem_smtp_password", __( "Insert the password to be used to access your SMTP server", 'dbem' ) );
1340
+ echo $save_button;
1341
+ ?>
1342
+ </table>
1343
+ </div> <!-- . inside -->
1344
+ </div> <!-- .postbox -->
1345
+ <?php
1346
+ }
1347
+
1348
+ /**
1349
+ * Meta options box for user capabilities. Shared in both MS and Normal options page, hence it's own function
1350
+ */
1351
+ function em_admin_option_box_caps(){
1352
+ global $save_button;
1353
+ ?>
1354
+ <div class="postbox" >
1355
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'User Capabilities', 'dbem' ); ?></span></h3>
1356
+ <div class="inside">
1357
+ <table class="form-table">
1358
+ <tr><td colspan="2">
1359
+ <strong><?php _e('Warning: Changing these values may result in exposing previously hidden information to all users.', 'dbem')?></strong><br />
1360
+ </td></tr>
1361
+ <?php
1362
+ global $wp_roles;
1363
+ $cap_docs = array(
1364
+ sprintf(__('%s Capabilities','dbem'),__('Event','dbem')) => array(
1365
+ /* Event Capabilities */
1366
+ 'publish_events' => sprintf(__('Users can publish %s and skip any admin approval','dbem'),__('events','dbem')),
1367
+ 'delete_others_events' => sprintf(__('User can delete other users %s','dbem'),__('events','dbem')),
1368
+ 'edit_others_events' => sprintf(__('User can edit other users %s','dbem'),__('events','dbem')),
1369
+ 'delete_events' => sprintf(__('User can delete their own %s','dbem'),__('events','dbem')),
1370
+ 'edit_events' => sprintf(__('User can create and edit %s','dbem'),__('events','dbem')),
1371
+ 'read_private_events' => sprintf(__('User can view private %s','dbem'),__('events','dbem')),
1372
+ /*'read_events' => sprintf(__('User can view %s','dbem'),__('events','dbem')),*/
1373
+ ),
1374
+ sprintf(__('%s Capabilities','dbem'),__('Recurring Event','dbem')) => array(
1375
+ /* Recurring Event Capabilties */
1376
+ 'publish_recurring_events' => sprintf(__('Users can publish %s and skip any admin approval','dbem'),__('recurring events','dbem')),
1377
+ 'delete_others_recurring_events' => sprintf(__('User can delete other users %s','dbem'),__('recurring events','dbem')),
1378
+ 'edit_others_recurring_events' => sprintf(__('User can edit other users %s','dbem'),__('recurring events','dbem')),
1379
+ 'delete_recurring_events' => sprintf(__('User can delete their own %s','dbem'),__('recurring events','dbem')),
1380
+ 'edit_recurring_events' => sprintf(__('User can create and edit %s','dbem'),__('recurring events','dbem'))
1381
+ ),
1382
+ sprintf(__('%s Capabilities','dbem'),__('Location','dbem')) => array(
1383
+ /* Location Capabilities */
1384
+ 'publish_locations' => sprintf(__('Users can publish %s and skip any admin approval','dbem'),__('locations','dbem')),
1385
+ 'delete_others_locations' => sprintf(__('User can delete other users %s','dbem'),__('locations','dbem')),
1386
+ 'edit_others_locations' => sprintf(__('User can edit other users %s','dbem'),__('locations','dbem')),
1387
+ 'delete_locations' => sprintf(__('User can delete their own %s','dbem'),__('locations','dbem')),
1388
+ 'edit_locations' => sprintf(__('User can create and edit %s','dbem'),__('locations','dbem')),
1389
+ 'read_private_locations' => sprintf(__('User can view private %s','dbem'),__('locations','dbem')),
1390
+ 'read_others_locations' => __('User can use other user locations for their events.','dbem'),
1391
+ /*'read_locations' => sprintf(__('User can view %s','dbem'),__('locations','dbem')),*/
1392
+ ),
1393
+ sprintf(__('%s Capabilities','dbem'),__('Other','dbem')) => array(
1394
+ /* Category Capabilities */
1395
+ 'delete_event_categories' => sprintf(__('User can delete %s categories and tags.','dbem'),__('event','dbem')),
1396
+ 'edit_event_categories' => sprintf(__('User can edit %s categories and tags.','dbem'),__('event','dbem')),
1397
+ /* Booking Capabilities */
1398
+ 'manage_others_bookings' => __('User can manage other users individual bookings and event booking settings.','dbem'),
1399
+ 'manage_bookings' => __('User can use and manage bookings with their events.','dbem'),
1400
+ 'upload_event_images' => __('User can upload images along with their events and locations.','dbem')
1401
+ )
1402
+ );
1403
+ ?>
1404
+ <tr><td colspan="2">
1405
+ <p><em><?php _e('You can now give fine grained control with regards to what your users can do with events. Each user role can have perform different sets of actions.','dbem'); ?></em></p>
1406
+ <table class="em-caps-table" style="width:auto;" cellspacing="0" cellpadding="0">
1407
+ <thead>
1408
+ <tr>
1409
+ <td>&nbsp;</td>
1410
+ <?php
1411
+ $odd = 0;
1412
+ foreach(array_keys($cap_docs) as $capability_group){
1413
+ ?><th class="<?php echo ( !is_int($odd/2) ) ? 'odd':''; ?>"><?php echo $capability_group ?></th><?php
1414
+ $odd++;
1415
+ }
1416
+ ?>
1417
+ </tr>
1418
+ </thead>
1419
+ <tbody>
1420
+ <?php foreach($wp_roles->role_objects as $role): ?>
1421
+ <tr>
1422
+ <td class="cap"><strong><?php echo $role->name; ?></strong></td>
1423
+ <?php
1424
+ $odd = 0;
1425
+ foreach($cap_docs as $capability_group){
1426
+ ?>
1427
+ <td class="<?php echo ( !is_int($odd/2) ) ? 'odd':''; ?>">
1428
+ <?php foreach($capability_group as $cap => $cap_help){ ?>
1429
+ <input type="checkbox" name="em_capabilities[<?php echo $role->name; ?>][<?php echo $cap ?>]" value="1" id="<?php echo $role->name.'_'.$cap; ?>" <?php echo $role->has_cap($cap) ? 'checked="checked"':''; ?> />
1430
+ &nbsp;<label for="<?php echo $role->name.'_'.$cap; ?>"><?php echo $cap; ?></label>&nbsp;<a href="#" title="<?php echo $cap_help; ?>">?</a>
1431
+ <br />
1432
+ <?php } ?>
1433
+ </td>
1434
+ <?php
1435
+ $odd++;
1436
+ }
1437
+ ?>
1438
+ </tr>
1439
+ <?php endforeach; ?>
1440
+ </tbody>
1441
+ </table>
1442
+ </td></tr>
1443
+ <?php echo $save_button; ?>
1444
+ </table>
1445
+ </div> <!-- . inside -->
1446
+ </div> <!-- .postbox -->
1447
+ <?php
1448
+ }
1449
+
1450
+ function em_admin_option_box_uninstall(){
1451
+ if( is_multisite() ){
1452
+ $uninstall_url = admin_url().'network/admin.php?page=events-manager-options&amp;action=uninstall&amp;_wpnonce='.wp_create_nonce('em_uninstall_'.get_current_user_id().'_wpnonce');
1453
+ $reset_url = admin_url().'network/admin.php?page=events-manager-options&amp;action=reset&amp;_wpnonce='.wp_create_nonce('em_reset_'.get_current_user_id().'_wpnonce');
1454
+ }else{
1455
+ $uninstall_url = EM_ADMIN_URL.'&amp;page=events-manager-options&amp;action=uninstall&amp;_wpnonce='.wp_create_nonce('em_uninstall_'.get_current_user_id().'_wpnonce');
1456
+ $reset_url = EM_ADMIN_URL.'&amp;page=events-manager-options&amp;action=reset&amp;_wpnonce='.wp_create_nonce('em_reset_'.get_current_user_id().'_wpnonce');
1457
+ }
1458
+ ?>
1459
+ <div class="postbox" >
1460
+ <div class="handlediv" title="<?php __('Click to toggle', 'dbem'); ?>"><br /></div><h3><span><?php _e ( 'Uninstall/Reset', 'dbem' ); ?></span></h3>
1461
+ <div class="inside">
1462
+ <p><?php _e('Use the buttons below to uninstall Events Manager completely from your system or reset Events Manager to original settings and keep your event data.','dbem'); ?></p>
1463
+ <a href="<?php echo $uninstall_url; ?>" class="button-secondary"><?php _e('Uninstall','dbem'); ?></a>
1464
+ <a href="<?php echo $reset_url; ?>" class="button-secondary"><?php _e('Reset','dbem'); ?></a>
1465
+ </div>
1466
+ </div>
1467
+ <?php
1468
+ }
1469
  ?>
admin/em-people.php DELETED
@@ -1,192 +0,0 @@
1
- <?php
2
- /**
3
- * Check for flags to save personal data
4
- */
5
- function em_person_actions(){
6
- global $EM_Person;
7
- if( !empty($_REQUEST['action']) && is_object($EM_Person) ){
8
- if( $_REQUEST['action'] == 'edit_person' ){
9
- $validation = $EM_Person->get_post();
10
- if ( $validation ) { //EM_Event gets the event if submitted via POST and validates it (safer than to depend on JS)
11
- //Save
12
- if( $EM_Person->save() ) {
13
- function em_person_save_notification(){
14
- global $EM_Person;
15
- ?><div class="updated"><p><strong><?php echo $EM_Person->feedback_message; ?></strong></p></div><?php
16
- }
17
- }else{
18
- function em_person_save_notification(){
19
- global $EM_Person;
20
- ?><div class="error"><p><strong><?php echo $EM_Person->feedback_message; ?></strong></p></div><?php
21
- }
22
- }
23
- }else{
24
- //TODO make errors clearer when saving person
25
- function em_person_save_notification(){
26
- global $EM_Person;
27
- ?><div class="error"><p><strong><?php echo $EM_Person->feedback_message; ?></strong></p></div><?php
28
- }
29
- }
30
- add_action ( 'admin_notices', 'em_person_save_notification' );
31
- }
32
- if( $_REQUEST['action'] == 'person_delete' ){
33
- if( $EM_Person->delete() ){
34
- //TODO delete person needs confirmation
35
- wp_redirect( get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager-bookings');
36
- exit();
37
- }else{
38
- function em_person_delete_notification(){
39
- global $EM_Person;
40
- ?><div class="error"><p><strong><?php echo $EM_Person->feedback_message; ?></strong></p></div><?php
41
- }
42
- }
43
- add_action ( 'admin_notices', 'em_person_delete_notification' );
44
- }
45
- }
46
- }
47
- add_action('admin_init','em_person_actions');
48
-
49
- /**
50
- * Generate an edit form for person details.
51
- */
52
- function em_person_edit_form(){
53
- global $EM_Person;
54
- ?>
55
- <form action="" method="post" id="em-person-form">
56
- <table>
57
- <tr><td><strong><?php _e('Name','dbem'); ?></strong></td><td><input type="text" name="person_name" size="60" value="<?php echo $EM_Person->name; ?>" /></td></tr>
58
- <tr><td><strong><?php _e('Phone','dbem'); ?></strong></td><td><input type="text" name="person_phone" size="60" value="<?php echo $EM_Person->phone; ?>" /></td></tr>
59
- <tr><td><strong><?php _e('E-mail','dbem'); ?></strong></td><td><input type="text" name="person_email" size="60" value="<?php echo $EM_Person->email; ?>" /></td></tr>
60
- </table>
61
- <p class="submit">
62
- <input type="submit" name="events_update" value="<?php _e ( 'Save' ); ?> &raquo;" />
63
- </p>
64
- <input type="hidden" name="action" value="person_edit" />
65
- <input type="hidden" name="person_id" value="<?php echo $EM_Person->id; ?>" />
66
- </form>
67
- <?php
68
- }
69
-
70
- /**
71
- * Depreciated page... for now at least.
72
- */
73
- function em_admin_people_page() {
74
- ?>
75
- <div class='wrap'>
76
- <div id="icon-users" class="icon32"><br/></div>
77
- <h2>People</h2>
78
- <?php
79
- $EM_People = EM_People::get();
80
- if (count($EM_People) < 1 ) {
81
- _e("No people have responded to your events yet!", 'dbem');
82
- } else {
83
- ?>
84
- <p><?php _e('This table collects the data about the people who responded to your events', 'dbem') ?></p>
85
- <table id='dbem-people-table' class='widefat post fixed'>
86
- <thead>
87
- <tr>
88
- <th class='manage-column column-cb check-column' scope='col'>&nbsp;</th>
89
- <th class='manage-column ' scope='col'>Name</th>
90
- <th scope='col'>E-mail</th>
91
- <th scope='col'>Phone number</th>
92
- </tr>
93
- </thead>
94
- <tfoot>
95
- <tr>
96
- <th class='manage-column column-cb check-column' scope='col'>&nbsp;</th>
97
- <th class='manage-column ' scope='col'>Name</th>
98
- <th scope='col'>E-mail</th>
99
- <th scope='col'>Phone number</th>
100
- </tr>
101
- </tfoot>
102
- <?php foreach ($EM_People as $EM_Person): ?>
103
- <tr>
104
- <td>&nbsp;</td>
105
- <td><?php echo $EM_Person->name ?></td>
106
- <td><?php echo $EM_Person->email ?></td>
107
- <td><?php echo $EM_Person->phone ?></td>
108
- </tr>
109
- <?php endforeach; ?>
110
- </table>
111
- <?php
112
- }
113
- ?>
114
- </div>
115
- <?php
116
- }
117
-
118
- function em_printable_booking_report() {
119
- global $EM_Event;
120
- //check that user can access this page
121
- if( isset($_GET['page']) && $_GET['page']=='events-manager-bookings' && isset($_GET['action']) && $_GET['action'] == 'bookings_report' && is_object($EM_Event)){
122
- if( is_object($EM_Event) && !$EM_Event->can_manage() ){
123
- ?>
124
- <div class="wrap"><h2><?php _e('Unauthorized Access','dbem'); ?></h2><p><?php _e('You do not have the rights to manage this event.','dbem'); ?></p></div>
125
- <?php
126
- return false;
127
- }
128
- ?>
129
- <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
130
- "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
131
- <html>
132
- <head>
133
- <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
134
- <title>Bookings for <?php echo $EM_Event->name; ?></title>
135
- <link rel="stylesheet" href="<?php echo bloginfo('wpurl') ?>/wp-content/plugins/events-manager/includes/css/events_manager.css" type="text/css" media="screen" />
136
- </head>
137
- <body id="printable">
138
- <div id="container">
139
- <h1>Bookings for <?php echo $EM_Event->name; ?></h1>
140
- <p><?php echo $EM_Event->output("#d #M #Y"); ?></p>
141
- <p><?php echo $EM_Event->output("#_LOCATION, #_ADDRESS, #_TOWN"); ?></p>
142
- <h2><?php _e('Bookings data', 'dbem');?></h2>
143
- <table id="bookings-table">
144
- <tr>
145
- <th scope='col'><?php _e('Name', 'dbem')?></th>
146
- <th scope='col'><?php _e('E-mail', 'dbem')?></th>
147
- <th scope='col'><?php _e('Phone number', 'dbem')?></th>
148
- <th scope='col'><?php _e('Seats', 'dbem')?></th>
149
- <th scope='col'><?php _e('Comment', 'dbem')?></th>
150
- </tr>
151
- <?php foreach($EM_Event->get_bookings()->bookings as $EM_Booking) { ?>
152
- <tr>
153
-
154
- <td><?php echo $EM_Booking->person->name ?></td>
155
- <td><?php echo $EM_Booking->person->email ?></td>
156
- <td><?php echo $EM_Booking->person->phone ?></td>
157
- <td class='seats-number'><?php echo $EM_Booking->seats ?></td>
158
- <td><?php echo $EM_Booking->comment ?></td>
159
- </tr>
160
- <?php } ?>
161
- <tr id='booked-seats'>
162
- <td colspan='3'>&nbsp;</td>
163
- <td class='total-label'><?php _e('Booked', 'dbem')?>:</td>
164
- <td class='seats-number'><?php echo $EM_Event->get_bookings()->get_booked_seats(); ?></td>
165
- </tr>
166
- <tr id='available-seats'>
167
- <td colspan='3'>&nbsp;</td>
168
- <td class='total-label'><?php _e('Available', 'dbem')?>:</td>
169
- <td class='seats-number'><?php echo $EM_Event->get_bookings()->get_available_seats(); ?></td>
170
- </tr>
171
- </table>
172
- </div>
173
- </body>
174
- </html>
175
- <?php
176
- die();
177
- }
178
- }
179
- add_action('admin_init', 'em_printable_booking_report');
180
-
181
- /**
182
- * Adds phone number to contact info of users, compatible with previous phone field method
183
- * @param $array
184
- * @return array
185
- */
186
- function em_contact_methods($array){
187
- $array['dbem_phone'] = __('Phone','dbem') . ' <span class="description">('. __('Events Manager','dbem') .')</span>';
188
- return $array;
189
- }
190
- add_filter( 'user_contactmethods' , 'em_contact_methods' , 10 , 1 );
191
-
192
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
buddypress/bp-em-activity.php ADDED
@@ -0,0 +1,126 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ // This file handles hooks/filter requiring activity stream publications
3
+
4
+ /**
5
+ * bp_em_record_activity()
6
+ *
7
+ * If the activity stream component is installed, this function will record activity items for your
8
+ * component.
9
+ */
10
+ function bp_em_record_activity( $args = '' ) {
11
+ if ( !function_exists( 'bp_activity_add' ) )
12
+ return false;
13
+
14
+ $defaults = array(
15
+ 'id' => false,
16
+ 'user_id' => '',
17
+ 'action' => '',
18
+ 'content' => '',
19
+ 'primary_link' => '',
20
+ 'component' => 'events-manager',
21
+ 'type' => false,
22
+ 'item_id' => false,
23
+ 'secondary_item_id' => false,
24
+ 'recorded_time' => gmdate( "Y-m-d H:i:s" ),
25
+ 'hide_sitewide' => false
26
+ );
27
+
28
+ $r = wp_parse_args( $args, $defaults );
29
+ extract( $r );
30
+
31
+ return bp_activity_add( array( 'id' => $id, 'user_id' => $user_id, 'action' => $action, 'content' => $content, 'primary_link' => $primary_link, 'component' => $component, 'type' => $type, 'item_id' => $item_id, 'secondary_item_id' => $secondary_item_id, 'recorded_time' => $recorded_time, 'hide_sitewide' => $hide_sitewide ) );
32
+ }
33
+
34
+ /**
35
+ * Records new events to the activity stream.
36
+ * @param unknown_type $result
37
+ * @param unknown_type $EM_Event
38
+ * @return unknown
39
+ */
40
+ function bp_em_record_activity_event_save( $result, $EM_Event ){
41
+ if( $result && $EM_Event->event_status == 1 && empty($EM_Event->previous_status) ){
42
+ $user = get_userdata($EM_Event->event_owner);
43
+ $member_slug = function_exists( 'bp_get_members_root_slug' ) ? bp_get_members_root_slug() : BP_MEMBERS_SLUG;
44
+ $member_link = trailingslashit(bp_get_root_domain()) . $member_slug . '/' . $user->user_login;
45
+ if( empty($EM_Event->group_id) ){
46
+ bp_em_record_activity( array(
47
+ 'user_id' => $user->ID,
48
+ 'action' => sprintf(__('%s added the event %s','dbem'), "<a href='".$member_link."/'>".$user->display_name."</a>", $EM_Event->output('#_EVENTLINK') ),
49
+ 'primary_link' => $EM_Event->output('#_EVENTURL'),
50
+ 'type' => 'new_event',
51
+ 'item_id' => $EM_Event->event_id,
52
+ ));
53
+ }else{
54
+ //tis a group event
55
+ $group = new BP_Groups_Group($EM_Event->group_id);
56
+ bp_em_record_activity( array(
57
+ 'user_id' => $user->ID,
58
+ 'action' => sprintf(__('%s added the event %s to %s.','dbem'), "<a href='".$member_link."/'>".$user->display_name."</a>", $EM_Event->output('#_EVENTLINK'), '<a href="'.bp_get_group_permalink($group).'">'.bp_get_group_name($group).'</a>' ),
59
+ 'component' => 'groups',
60
+ 'type' => 'new_event',
61
+ 'item_id' => $EM_Event->group_id,
62
+ ));
63
+ }
64
+ }
65
+ return $result;
66
+ }
67
+ add_filter('em_event_save','bp_em_record_activity_event_save', 10, 2);
68
+
69
+ /**
70
+ * @param boolean $result
71
+ * @param EM_Booking $EM_Booking
72
+ * @return boolean
73
+ */
74
+ function bp_em_record_activity_booking_save( $result, $EM_Booking ){
75
+ if( $result ){
76
+ $rejected_statuses = array(0,2,3); //these statuses apply to rejected/cancelled bookings
77
+ $user = $EM_Booking->person;
78
+ $member_slug = function_exists( 'bp_get_members_root_slug' ) ? bp_get_members_root_slug() : BP_MEMBERS_SLUG;
79
+ $member_link = trailingslashit(bp_get_root_domain()) . $member_slug . '/' . $user->user_login;
80
+ $user_link = "<a href='".$member_link."/'>".$user->display_name."</a>";
81
+ $event_link = $EM_Booking->get_event()->output('#_EVENTLINK');
82
+ $status = $EM_Booking->booking_status;
83
+ $EM_Event = $EM_Booking->get_event();
84
+ if( empty($EM_Event->group_id) ){
85
+ if( $status == 1 || (!get_option('dbem_bookings_approval') && $status < 2) ){
86
+ $action = sprintf(__('%s is attending %s.','dbem'), $user_link, $event_link );
87
+ }elseif( ($EM_Booking->previous_status == 1 || (!get_option('dbem_bookings_approval') && $EM_Booking->previous_status < 2)) && in_array($status, $rejected_statuses) ){
88
+ $action = sprintf(__('%s will not be attending %s anymore.','dbem'), $user_link, $event_link );
89
+ }
90
+ }else{
91
+ $group = new BP_Groups_Group($EM_Event->group_id);
92
+ $group_link = '<a href="'.bp_get_group_permalink($group).'">'.bp_get_group_name($group).'</a>';
93
+ if( $status == 1 || (!get_option('dbem_bookings_approval') && $status < 2) ){
94
+ $action = sprintf(__('%s is attending %s of the group %s.','dbem'), $user_link, $event_link, $group_link );
95
+ }elseif( ($EM_Booking->previous_status == 1 || (!get_option('dbem_bookings_approval') && $EM_Booking->previous_status < 2)) && in_array($status, $rejected_statuses) ){
96
+ $action = sprintf(__('%s will not be attending %s of group %s anymore.','dbem'), $user_link, $event_link, $group_link );
97
+ }
98
+ }
99
+ if( !empty($action) ){
100
+ bp_em_record_activity( array(
101
+ 'user_id' => $EM_Booking->person->ID,
102
+ 'action' => $action,
103
+ 'primary_link' => $EM_Event->output('#_EVENTURL'),
104
+ 'type' => 'new_booking',
105
+ 'item_id' => $EM_Event->event_id,
106
+ 'secondary_item_id' => $EM_Booking->booking_id
107
+ ));
108
+ //group activity
109
+ if( !empty($EM_Event->group_id) ){
110
+ //tis a group event
111
+ bp_em_record_activity( array(
112
+ 'component' => 'groups',
113
+ 'item_id' => $EM_Event->group_id,
114
+ 'user_id' => $EM_Booking->person->ID,
115
+ 'action' => $action,
116
+ 'primary_link' => $EM_Event->output('#_EVENTURL'),
117
+ 'type' => 'new_booking',
118
+ 'secondary_item_id' => $EM_Booking->booking_id
119
+ ));
120
+ }
121
+ }
122
+ }
123
+ return $result;
124
+ }
125
+ add_filter('em_booking_save','bp_em_record_activity_booking_save', 10, 2);
126
+ add_filter('em_booking_delete','bp_em_record_activity_booking_save', 10, 2);
buddypress/bp-em-core.php ADDED
@@ -0,0 +1,227 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ //Main loader for buddypress
3
+ /**
4
+ * Events Manager component for BuddyPress
5
+ * @author marcus
6
+ * @since 5.0
7
+ */
8
+ class BP_EM_Component extends BP_Component {
9
+
10
+ function __construct() {
11
+ global $bp;
12
+ parent::start('events', __('Events', 'dbem'), EM_DIR);
13
+ $this->includes();
14
+ //TODO make BP component optional
15
+ $bp->active_components[$this->id] = '1';
16
+ }
17
+
18
+ function includes() {
19
+ // Files to include
20
+ $includes = array(
21
+ 'buddypress/bp-em-activity.php',
22
+ 'buddypress/bp-em-templatetags.php',
23
+ 'buddypress/bp-em-notifications.php',
24
+ 'buddypress/screens/profile.php',
25
+ 'buddypress/screens/my-events.php',
26
+ 'buddypress/screens/my-locations.php',
27
+ 'buddypress/screens/attending.php',
28
+ 'buddypress/screens/my-bookings.php',
29
+ 'buddypress/screens/my-group-events.php',
30
+ 'buddypress/screens/group-events.php',
31
+ 'buddypress/bp-em-groups.php'
32
+ );
33
+ parent::includes( $includes );
34
+ //TODO add admin pages for extra BP specific settings
35
+ }
36
+
37
+ /**
38
+ * Sets up the global Events Manager BuddyPress Components
39
+ */
40
+ function setup_globals() {
41
+ global $bp, $wpdb;
42
+ // Define a slug constant that will be used to view this components pages
43
+ if ( !defined( 'BP_EM_SLUG' ) )
44
+ define ( 'BP_EM_SLUG', EM_POST_TYPE_EVENT_SLUG );
45
+
46
+ // Set up the $globals array to be passed along to parent::setup_globals()
47
+ $globals = array(
48
+ 'slug' => BP_EM_SLUG,
49
+ 'has_directory' => false, //already done by EM
50
+ 'notification_callback' => 'bp_em_format_notifications',
51
+ 'search_string' => sprintf(__( 'Search %s...', 'dbem' ),__('Events','dbem')),
52
+ );
53
+
54
+ // Let BP_Component::setup_globals() do its work.
55
+ parent::setup_globals( $globals );
56
+
57
+ //quick link shortcut - may need to revisit this
58
+ $bp->{$this->id}->link = trailingslashit($bp->loggedin_user->domain).BP_EM_SLUG.'/';
59
+ }
60
+
61
+ function setup_nav() {
62
+ global $blog_id;
63
+ //check multisite or normal mode for correct permission checking
64
+ if(is_multisite() && $blog_id != BP_ROOT_BLOG){
65
+ //FIXME MS mode doesn't seem to recognize cross subsite caps, using the proper functions, for now we use switch_blog.
66
+ $current_blog = $blog_id;
67
+ switch_to_blog(BP_ROOT_BLOG);
68
+ $can_manage_events = current_user_can_for_blog(BP_ROOT_BLOG, 'edit_events');
69
+ $can_manage_locations = current_user_can_for_blog(BP_ROOT_BLOG, 'edit_locations');
70
+ $can_manage_bookings = current_user_can_for_blog(BP_ROOT_BLOG, 'manage_bookings');
71
+ switch_to_blog($current_blog);
72
+ }else{
73
+ $can_manage_events = current_user_can('edit_events');
74
+ $can_manage_locations = current_user_can('edit_locations');
75
+ $can_manage_bookings = current_user_can('manage_bookings');
76
+ }
77
+ /* Add 'Events' to the main user profile navigation */
78
+ $main_nav = array(
79
+ 'name' => __( 'Events', 'dbem' ),
80
+ 'slug' => em_bp_get_slug(),
81
+ 'position' => 80,
82
+ 'screen_function' => 'bp_em_events',
83
+ 'default_subnav_slug' => 'profile'
84
+ );
85
+
86
+ $em_link = trailingslashit( bp_loggedin_user_domain() . em_bp_get_slug() );
87
+
88
+ /* Create SubNav Items */
89
+ $sub_nav[] = array(
90
+ 'name' => __( 'My Profile', 'dbem' ),
91
+ 'slug' => 'profile',
92
+ 'parent_slug' => em_bp_get_slug(),
93
+ 'parent_url' => $em_link,
94
+ 'screen_function' => 'bp_em_events',
95
+ 'position' => 10
96
+ );
97
+
98
+ $sub_nav[] = array(
99
+ 'name' => __( 'Events I\'m Attending', 'dbem' ),
100
+ 'slug' => 'attending',
101
+ 'parent_slug' => em_bp_get_slug(),
102
+ 'parent_url' => $em_link,
103
+ 'screen_function' => 'bp_em_attending',
104
+ 'position' => 20,
105
+ 'user_has_access' => bp_is_my_profile() // Only the logged in user can access this on his/her profile
106
+ );
107
+
108
+ if( $can_manage_events ){
109
+ $sub_nav[] = array(
110
+ 'name' => __( 'My Events', 'dbem' ),
111
+ 'slug' => 'my-events',
112
+ 'parent_slug' => em_bp_get_slug(),
113
+ 'parent_url' => $em_link,
114
+ 'screen_function' => 'bp_em_my_events',
115
+ 'position' => 30,
116
+ 'user_has_access' => bp_is_my_profile() // Only the logged in user can access this on his/her profile
117
+ );
118
+ }
119
+
120
+ if( $can_manage_locations && get_option('dbem_locations_enabled') ){
121
+ $sub_nav[] = array(
122
+ 'name' => __( 'My Locations', 'dbem' ),
123
+ 'slug' => 'my-locations',
124
+ 'parent_slug' => em_bp_get_slug(),
125
+ 'parent_url' => $em_link,
126
+ 'screen_function' => 'bp_em_my_locations',
127
+ 'position' => 40,
128
+ 'user_has_access' => bp_is_my_profile() // Only the logged in user can access this on his/her profile
129
+ );
130
+ }
131
+
132
+ if( $can_manage_bookings && get_option('dbem_rsvp_enabled') ){
133
+ $sub_nav[] = array(
134
+ 'name' => __( 'My Event Bookings', 'dbem' ),
135
+ 'slug' => 'my-bookings',
136
+ 'parent_slug' => em_bp_get_slug(),
137
+ 'parent_url' => $em_link,
138
+ 'screen_function' => 'bp_em_my_bookings',
139
+ 'position' => 50,
140
+ 'user_has_access' => bp_is_my_profile() // Only the logged in user can access this on his/her profile
141
+ );
142
+ }
143
+
144
+ if( bp_is_active('groups') ){
145
+ /* Create Profile Group Sub-Nav */
146
+ $sub_nav[] = array(
147
+ 'name' => __( 'Events', 'dbem' ),
148
+ 'slug' => 'group-events',
149
+ 'parent_slug' => bp_get_groups_slug(),
150
+ 'parent_url' =>trailingslashit( bp_loggedin_user_domain() . bp_get_groups_slug() ),
151
+ 'screen_function' => 'bp_em_my_group_events',
152
+ 'position' => 60,
153
+ 'user_has_access' => bp_is_my_profile() // Only the logged in user can access this on his/her profile
154
+ );
155
+ }
156
+
157
+ parent::setup_nav( $main_nav, $sub_nav );
158
+ add_action( 'bp_init', array(&$this, 'setup_group_nav') );
159
+ }
160
+
161
+ function setup_group_nav(){
162
+ global $bp;
163
+ /* Add some group subnav items */
164
+ $user_access = false;
165
+ $group_link = '';
166
+ if( bp_is_active('groups') && !empty($bp->groups->current_group) ){
167
+ $group_link = $bp->root_domain . '/' . bp_get_groups_root_slug() . '/' . $bp->groups->current_group->slug . '/';
168
+ $user_access = $bp->groups->current_group->user_has_access;
169
+ if( !empty($bp->current_component) && $bp->current_component == 'groups' ){
170
+ $count = EM_Events::count(array('group'=>$bp->groups->current_group->id));
171
+ if( empty($count) ) $count = 0;
172
+ }
173
+ bp_core_new_subnav_item( array(
174
+ 'name' => __( 'Events', 'dbem' ) . " <span>$count</span>",
175
+ 'slug' => 'events',
176
+ 'parent_url' => $group_link,
177
+ 'parent_slug' => $bp->groups->current_group->slug,
178
+ 'screen_function' => 'bp_em_group_events',
179
+ 'position' => 50,
180
+ 'user_has_access' => $user_access,
181
+ 'item_css_id' => 'forums'
182
+ ));
183
+ }
184
+ }
185
+ }
186
+ function bp_em_load_core_component() {
187
+ global $bp;
188
+ $bp->events = new BP_EM_Component();
189
+ }
190
+ add_action( 'bp_loaded', 'bp_em_load_core_component' );
191
+
192
+ if( !is_admin() ){
193
+ /*
194
+ * Links and URL Rewriting
195
+ */
196
+ function em_bp_rewrite_edit_url($url, $EM_Event){
197
+ global $bp;
198
+ return $bp->events->link.'my-events/?action=edit&event_id='.$EM_Event->event_id;
199
+ }
200
+ add_filter('em_event_get_edit_url','em_bp_rewrite_edit_url',10,2);
201
+
202
+
203
+ function em_bp_rewrite_bookings_url($url, $EM_Event){
204
+ global $bp;
205
+ return $bp->events->link.'my-bookings/?event_id='.$EM_Event->event_id;
206
+ }
207
+ add_filter('em_event_get_bookings_url','em_bp_rewrite_bookings_url',10,2);
208
+
209
+ function em_bp_rewrite_edit_location_url($url, $EM_Location){
210
+ global $bp;
211
+ return $bp->events->link.'my-locations/?location_id='.$EM_Location->location_id;
212
+ }
213
+ add_filter('em_location_get_edit_url','em_bp_rewrite_edit_location_url',10,2);
214
+ }
215
+
216
+ /**
217
+ * Delete events when you delete a user.
218
+ */
219
+ function bp_em_remove_data( $user_id ) {
220
+ $EM_Events = EM_Events::get(array('scope'=>'all','owner'=>$user_id, 'status'=>false));
221
+ EM_Events::delete($EM_Events);
222
+ }
223
+ add_action( 'wpmu_delete_user', 'bp_em_remove_data', 1 );
224
+ add_action( 'delete_user', 'bp_em_remove_data', 1 );
225
+
226
+ define('EM_BP_LOADED',true); //so we know
227
+ ?>
buddypress/bp-em-groups.php ADDED
@@ -0,0 +1,65 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @param EM_Event $EM_Event
4
+ */
5
+ function bp_em_group_event_save($EM_Event){
6
+ if( is_object($EM_Event) && empty($EM_Event->group_id) && !empty($_REQUEST['group_id']) && is_numeric($_REQUEST['group_id']) && bp_is_active('groups') ){
7
+ //we have been requested an event creation tied to a group, so does this group exist, and does this person have admin rights to it?
8
+ if( groups_is_user_admin(get_current_user_id(), $_REQUEST['group_id']) ){
9
+ $EM_Event->group_id = $_REQUEST['group_id'];
10
+ }
11
+ }
12
+ return $EM_Event;
13
+ }
14
+ add_action('em_event_save_pre','bp_em_group_event_save',1,1);
15
+
16
+ /**
17
+ * @param boolean $result
18
+ * @param EM_Event $EM_Event
19
+ */
20
+ function bp_em_group_event_can_manage( $result, $EM_Event){
21
+ if( !$result && !empty($EM_Event->group_id) && bp_is_active('groups') ){ //only override if already false, incase it's true
22
+ if( groups_is_user_admin(get_current_user_id(),$EM_Event->group_id) && current_user_can('edit_events') ){
23
+ //This user is an admin of the owner's group, so they can edit this event.
24
+ return true;
25
+ }
26
+ }
27
+ return $result;
28
+ }
29
+ add_filter('em_event_can_manage','bp_em_group_event_can_manage',1,2);
30
+
31
+
32
+ function bp_em_group_events_accepted_searches($searches){
33
+ if( bp_is_active('groups') ){
34
+ $searches[] = 'group';
35
+ }
36
+ return $searches;
37
+ }
38
+ add_filter('em_accepted_searches','bp_em_group_events_accepted_searches',1,1);
39
+
40
+ function bp_em_group_events_get_default_search($searches, $array){
41
+ if( !empty($array['group']) && (is_numeric($array['group']) || $array['group'] == 'my' || $array['group'] == 'this') && bp_is_active('groups') ){
42
+ if($array['group'] == 'this'){ //shows current group, if applicable
43
+ if( is_numeric(bp_get_current_group_id()) ){
44
+ $searches['group'] = bp_get_current_group_id();
45
+ }
46
+ }else{
47
+ $searches['group'] = $array['group'];
48
+ }
49
+ }
50
+ return $searches;
51
+ }
52
+ add_filter('em_events_get_default_search','bp_em_group_events_get_default_search',1,2);
53
+
54
+ function bp_em_group_events_build_sql_conditions( $conditions, $args ){
55
+ if( !empty($args['group']) && is_numeric($args['group']) ){
56
+ $conditions['group'] = "( `group_id`={$args['group']} )";
57
+ }elseif( !empty($args['group']) && $args['group'] == 'my' ){
58
+ $groups = groups_get_user_groups(get_current_user_id());
59
+ if( count($groups) > 0 ){
60
+ $conditions['group'] = "( `group_id` IN (".implode(',',$groups['groups']).") )";
61
+ }
62
+ }
63
+ return $conditions;
64
+ }
65
+ add_filter('em_events_build_sql_conditions','bp_em_group_events_build_sql_conditions',1,2);
buddypress/bp-em-notifications.php ADDED
@@ -0,0 +1,82 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ //This file handles hooks requiring notifications
3
+
4
+ /**
5
+ * bp_em_format_notifications()
6
+ *
7
+ * The format notification function will take DB entries for notifications and format them
8
+ * so that they can be displayed and read on the screen.
9
+ *
10
+ * Notifications are "screen" notifications, that is, they appear on the notifications menu
11
+ * in the site wide navigation bar. They are not for email notifications.
12
+ *
13
+ *
14
+ * The recording is done by using bp_core_add_notification() which you can search for in this file for
15
+ * ems of usage.
16
+ */
17
+ function bp_em_format_notifications( $action, $item_id, $secondary_item_id, $total_items ) {
18
+ global $bp;
19
+ switch ( $action ) {
20
+ case 'pending_booking':
21
+ //Count pending bookings
22
+ if( get_option('dbem_bookings_approval')){
23
+ if ( $total_items > 1 ) {
24
+ return '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . __('You have a pending booking','dbem'). '</a>';
25
+ } else {
26
+ return apply_filters( 'bp_em_format_new_booking_notification', '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . sprintf(__('You have %s pending bookings','dbem'), $total_items). '</a>' );
27
+ }
28
+ }
29
+ break;
30
+ case 'confirmed_booking':
31
+ //Count pending bookings
32
+ if ( $total_items > 1 ) {
33
+ return apply_filters( 'bp_em_format_confirmed_booking_notifications', '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . __('You have a confirmed booking','dbem'). '</a>' );
34
+ } else {
35
+ return apply_filters( 'bp_em_format_confirmed_booking_notification', '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . sprintf(__('You have %s confirmed bookings','dbem'), $total_items). '</a>' );
36
+ }
37
+ break;
38
+ case 'cancelled_booking':
39
+ //Count pending bookings
40
+ if ( $total_items > 1 ) {
41
+ return apply_filters( 'bp_em_format_cancelled_booking_notifications', '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . __('A user cancelled a booking','dbem'). '</a>' );
42
+ } else {
43
+ return apply_filters( 'bp_em_format_cancelled_booking_notification', '<a href="' . $bp->loggedin_user->domain . $bp->events->slug . '/my-bookings/" title="' . __( 'My Bookings', 'dbem' ) . '">' . sprintf(__('%s users cancelled bookings.','dbem'), $total_items). '</a>' );
44
+ }
45
+ break;
46
+ }
47
+ do_action( 'bp_em_format_notifications', $action, $item_id, $secondary_item_id, $total_items );
48
+
49
+ return false;
50
+ }
51
+
52
+ /**
53
+ * Remove a screen notification for a user.
54
+ */
55
+ function bp_em_remove_screen_notifications() {
56
+ global $bp;
57
+ bp_core_delete_notifications_by_type( $bp->loggedin_user->id, $bp->events->slug, 'attending' );
58
+ }
59
+ add_action( 'bp_em_my_events', 'bp_em_remove_screen_notifications' );
60
+ add_action( 'xprofile_screen_display_profile', 'bp_em_remove_screen_notifications' );
61
+
62
+ /**
63
+ * Catch booking saves and add a BP notification.
64
+ * @param boolean $result
65
+ * @param EM_Booking $EM_Booking
66
+ * @return boolean
67
+ */
68
+ function bp_em_add_booking_notification($result, $EM_Booking){
69
+ global $bp;
70
+ if( get_option('dbem_bookings_approval') && $EM_Booking->status == 0 ){
71
+ $action = 'pending_booking';
72
+ }elseif( $EM_Booking->status == 1 || (get_option('dbem_bookings_approval') && $EM_Booking->status == 0) ){
73
+ $action = 'confirmed_booking';
74
+ }elseif( $EM_Booking->status == 3 ){
75
+ $action = 'cancelled_booking';
76
+ }
77
+ if( !empty($action) ){
78
+ bp_core_add_notification( $EM_Booking->booking_id, $EM_Booking->get_event()->owner, 'events', $action );
79
+ }
80
+ return $result;
81
+ }
82
+ add_filter('em_booking_save','bp_em_add_booking_notification',1,2);
buddypress/bp-em-templatetags.php ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Echo the Events Manager BuddyPresss component's slug
4
+ * @since 5.0
5
+ */
6
+ function em_bp_slug() {
7
+ echo em_bp_get_slug();
8
+ }
9
+ /**
10
+ * Return the Events Manager BuddyPresss component's slug
11
+ *
12
+ * @since 5.0
13
+ * @uses apply_filters() Filter 'em_bp_get_slug' to change the output
14
+ * @return str $slug The slug from $bp->events->slug, if it exists
15
+ */
16
+ function em_bp_get_slug() {
17
+ global $bp;
18
+ // Avoid PHP warnings, in case the value is not set for some reason
19
+ $slug = !empty( $bp->evebts->slug ) ? $bp->events->slug : BP_EM_SLUG;
20
+ return apply_filters( 'em_bp_get_slug', $slug );
21
+ }
buddypress/screens/attending.php ADDED
@@ -0,0 +1,29 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * bp_em_screen_two()
4
+ *
5
+ * Sets up and displays the screen output for the sub nav item "em/screen-two"
6
+ */
7
+ function bp_em_attending() {
8
+ global $bp;
9
+ /**
10
+ * If the user has not Accepted or Rejected anything, then the code above will not run,
11
+ * we can continue and load the template.
12
+ */
13
+ do_action( 'bp_em_attending' );
14
+
15
+ add_action( 'bp_template_title', 'bp_em_attending_title' );
16
+ add_action( 'bp_template_content', 'bp_em_attending_content' );
17
+
18
+ /* Finally load the plugin template file. */
19
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
20
+ }
21
+
22
+ function bp_em_attending_title() {
23
+ _e( 'Events I\'m Attending', 'dbem' );
24
+ }
25
+
26
+ function bp_em_attending_content() {
27
+ //We can use the same template as the public user interface for non bp sites
28
+ em_locate_template('templates/my-bookings.php',true);
29
+ }
buddypress/screens/group-events.php ADDED
@@ -0,0 +1,30 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Controller for the event views in BP (using mvc terms here)
4
+ */
5
+ function bp_em_group_events() {
6
+ global $bp;
7
+ do_action( 'bp_em_group_events' );
8
+
9
+ //plug into EM admin code (at least for now)
10
+ include_once(EM_DIR.'/admin/em-admin.php');
11
+
12
+ add_action( 'bp_template_title', 'bp_em_group_events_title' );
13
+ add_action( 'bp_template_content', 'bp_em_group_events_content' );
14
+
15
+ /* Finally load the plugin template file. */
16
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'groups/single/plugins' ) );
17
+ }
18
+
19
+ function bp_em_group_events_title() {
20
+ _e( 'Group Events', 'dbem' );
21
+ }
22
+ /**
23
+ * Determines whether to show event page or events page, and saves any updates to the event or events
24
+ * @return null
25
+ */
26
+ function bp_em_group_events_content() {
27
+ em_locate_template('buddypress/group-events.php', true);
28
+ }
29
+
30
+ ?>
buddypress/screens/my-bookings.php ADDED
@@ -0,0 +1,35 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * bp_em_screen_two()
4
+ *
5
+ * Sets up and displays the screen output for the sub nav item "em/screen-two"
6
+ */
7
+ function bp_em_my_bookings() {
8
+ global $bp, $EM_Event;
9
+
10
+ //assume any notifications here are considered viewed via this page
11
+ bp_core_delete_notifications_by_type(get_current_user_id(), 'events','pending_booking');
12
+ bp_core_delete_notifications_by_type(get_current_user_id(), 'events','confirmed_booking');
13
+ bp_core_delete_notifications_by_type(get_current_user_id(), 'events','cancelled_booking');
14
+
15
+ em_load_event();
16
+ /**
17
+ * If the user has not Accepted or Rejected anything, then the code above will not run,
18
+ * we can continue and load the template.
19
+ */
20
+ do_action( 'bp_em_my_bookings' );
21
+
22
+ add_action( 'bp_template_title', 'bp_em_my_bookings_title' );
23
+ add_action( 'bp_template_content', 'bp_em_my_bookings_content' );
24
+
25
+ /* Finally load the plugin template file. */
26
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
27
+ }
28
+
29
+ function bp_em_my_bookings_title() {
30
+ _e( 'My Event Bookings', 'dbem' );
31
+ }
32
+
33
+ function bp_em_my_bookings_content() {
34
+ em_locate_template('buddypress/my-bookings.php',true);
35
+ }
buddypress/screens/my-events.php ADDED
@@ -0,0 +1,56 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Controller for the event views in BP (using mvc terms here)
4
+ */
5
+ function bp_em_my_events() {
6
+ global $bp, $EM_Event;
7
+ if( !is_object($EM_Event) && !empty($_REQUEST['event_id']) ){
8
+ $EM_Event = new EM_Event($_REQUEST['event_id']);
9
+ }
10
+
11
+ do_action( 'bp_em_my_events' );
12
+
13
+ $template_title = 'bp_em_my_events_title';
14
+ $template_content = 'bp_em_my_events_content';
15
+
16
+ if( !empty($_GET['action']) ){
17
+ switch($_GET['action']){
18
+ case 'edit':
19
+ $template_title = 'bp_em_my_events_editor_title';
20
+ break;
21
+ }
22
+ }
23
+
24
+ add_action( 'bp_template_title', $template_title );
25
+ add_action( 'bp_template_content', $template_content );
26
+
27
+ /* Finally load the plugin template file. */
28
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
29
+ }
30
+
31
+ function bp_em_my_events_title() {
32
+ _e( 'My Events', 'dbem' );
33
+ if(current_user_can('edit_events')): ?> <a href="<?php echo add_query_arg(array('action'=>'edit')); ?>" class="button add-new-h2"><?php _e('Add New','dbem'); ?></a><?php endif;
34
+ }
35
+
36
+ /**
37
+ * Determines whether to show event page or events page, and saves any updates to the event or events
38
+ * @return null
39
+ */
40
+ function bp_em_my_events_content() {
41
+ em_locate_template('buddypress/my-events.php', true);
42
+ }
43
+
44
+ function bp_em_my_events_editor_title() {
45
+ global $EM_Event;
46
+ if( is_object($EM_Event) ){
47
+ if($EM_Event->is_recurring()){
48
+ _e( "Reschedule Events", 'dbem' )." '{$EM_Event->event_name}'";
49
+ }else{
50
+ _e ( "Edit Event", 'dbem' ) . " '" . $EM_Event->event_name . "'";
51
+ }
52
+ }else{
53
+ _e( 'Add Event', 'dbem' );
54
+ }
55
+ }
56
+ ?>
buddypress/screens/my-group-events.php ADDED
@@ -0,0 +1,27 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Controller for the event views in BP (using mvc terms here)
4
+ */
5
+ function bp_em_my_group_events() {
6
+ global $bp;
7
+ do_action( 'bp_em_my_group_events' );
8
+
9
+ //plug into EM admin code (at least for now)
10
+ include_once(EM_DIR.'/admin/em-admin.php');
11
+
12
+ add_action( 'bp_template_title', 'bp_em_my_group_events_title' );
13
+ add_action( 'bp_template_content', 'bp_em_my_group_events_content' );
14
+
15
+ /* Finally load the plugin template file. */
16
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
17
+ }
18
+
19
+ function bp_em_my_group_events_title() {
20
+ _e( 'Group Events', 'dbem' );
21
+ }
22
+
23
+ function bp_em_my_group_events_content(){
24
+ em_locate_template('buddypress/my-group-events.php', true);
25
+ }
26
+
27
+ ?>
buddypress/screens/my-locations.php ADDED
@@ -0,0 +1,52 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Controller for the location views in BP (using mvc terms here)
4
+ */
5
+ function bp_em_my_locations() {
6
+ global $bp, $EM_Location;
7
+ if( !is_object($EM_Location) && !empty($_REQUEST['location_id']) ){
8
+ $EM_Location = new EM_Location($_REQUEST['location_id']);
9
+ }
10
+
11
+ do_action( 'bp_em_my_locations' );
12
+
13
+ $template_title = 'bp_em_my_locations_title';
14
+ $template_content = 'bp_em_my_locations_content';
15
+
16
+ if( !empty($_GET['action']) ){
17
+ switch($_GET['action']){
18
+ case 'edit':
19
+ $template_title = 'bp_em_my_locations_editor_title';
20
+ break;
21
+ }
22
+ }
23
+
24
+ add_action( 'bp_template_title', $template_title );
25
+ add_action( 'bp_template_content', $template_content );
26
+
27
+ /* Finally load the plugin template file. */
28
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
29
+ }
30
+
31
+ function bp_em_my_locations_title() {
32
+ _e( 'My Locations', 'dbem' );
33
+ if(current_user_can('edit_locations')): ?> <a href="<?php echo add_query_arg(array('action'=>'edit')); ?>" class="button add-new-h2"><?php _e('Add New','dbem'); ?></a><?php endif;
34
+ }
35
+
36
+ /**
37
+ * Determines whether to show location page or locations page, and saves any updates to the location or locations
38
+ * @return null
39
+ */
40
+ function bp_em_my_locations_content() {
41
+ em_locate_template('buddypress/my-locations.php', true);
42
+ }
43
+
44
+ function bp_em_my_locations_editor_title() {
45
+ global $EM_Location;
46
+ if( empty($EM_Location) || !is_object($EM_Location) ){
47
+ $title = __('Add Location', 'dbem');
48
+ }else{
49
+ $title = __('Edit Location', 'dbem');
50
+ }
51
+ }
52
+ ?>
buddypress/screens/profile.php ADDED
@@ -0,0 +1,32 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * bp_em_screen_one()
4
+ *
5
+ * Sets up and displays the screen output for the sub nav item "em/screen-one"
6
+ */
7
+ function bp_em_events() {
8
+ global $bp, $EM_Notices;
9
+
10
+ if( bp_is_my_profile() ){
11
+ $EM_Notices->add_info( __('You are currently viewing your public page, this is what other users will see.', 'dbem') );
12
+ }
13
+
14
+ /* Add a do action here, so your component can be extended by others. */
15
+ do_action( 'bp_em_events' );
16
+
17
+ add_action( 'bp_template_title', 'bp_em_events_title' );
18
+ add_action( 'bp_template_content', 'bp_em_events_content' );
19
+ bp_core_load_template( apply_filters( 'bp_core_template_plugin', 'members/single/plugins' ) );
20
+ //bp_core_load_template( apply_filters( 'bp_em_template_screen_one', 'em/screen-one' ) );
21
+ }
22
+ /***
23
+ * The second argument of each of the above add_action() calls is a function that will
24
+ * display the corresponding information. The functions are presented below:
25
+ */
26
+ function bp_em_events_title() {
27
+ _e( 'Events', 'dbem' );
28
+ }
29
+
30
+ function bp_em_events_content() {
31
+ em_locate_template('buddypress/profile.php',true);
32
+ }
classes/em-booking.php CHANGED
@@ -1,379 +1,677 @@
1
- <?php
2
- class EM_Booking extends EM_Object{
3
- //DB Fields
4
- var $id;
5
- var $event_id;
6
- var $person_id;
7
- var $seats;
8
- var $comment;
9
- var $status = 0;
10
- var $notes = array();
11
- var $fields = array(
12
- 'booking_id' => array('name'=>'id','type'=>'%d'),
13
- 'event_id' => array('name'=>'event_id','type'=>'%d'),
14
- 'person_id' => array('name'=>'person_id','type'=>'%d'),
15
- 'booking_seats' => array('name'=>'seats','type'=>'%d'),
16
- 'booking_comment' => array('name'=>'comment','type'=>'%s'),
17
- 'booking_status' => array('name'=>'status','type'=>'%d')
18
- );
19
- //Other Vars
20
- var $timestamp;
21
- var $person;
22
- var $required_fields = array('booking_id', 'event_id', 'person_id', 'booking_seats');
23
- var $feedback_message = "";
24
- var $errors = array();
25
- /**
26
- * If saved in this instance, you can see what previous approval status was.
27
- * @var int
28
- */
29
- var $previous_status = false;
30
- /**
31
- * The booking approval status number corresponds to a state in this array.
32
- * @var unknown_type
33
- */
34
- var $status_array = array();
35
-
36
- /**
37
- * Creates booking object and retreives booking data (default is a blank booking object). Accepts either array of booking data (from db) or a booking id.
38
- * @param mixed $booking_data
39
- * @return null
40
- */
41
- function EM_Booking( $booking_data = false ){
42
- if( !get_option('dbem_bookings_approval') ){
43
- $this->status = 1;
44
- }
45
- if( $booking_data !== false ){
46
- //Load booking data
47
- $booking = array();
48
- if( is_array($booking_data) ){
49
- $booking = $booking_data;
50
- //Also create a person out of this...
51
- $this->person = new EM_Person($booking_data);
52
- if( !empty($booking['person_id']) && $booking['person_id'] != $this->person->id){
53
- $this->person = new EM_Person($booking['booking_id']);
54
- }
55
- }elseif( is_numeric($booking_data) ){
56
- //Retreiving from the database
57
- global $wpdb;
58
- $sql = "SELECT * FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." WHERE booking_id ='$booking_data'";
59
- $booking = $wpdb->get_row($sql, ARRAY_A);
60
- //Get the person for this booking
61
- $this->person = new EM_Person($booking['person_id']);
62
- //Booking notes
63
- $notes = $wpdb->get_results("SELECT * FROM ". $wpdb->prefix . EM_META_TABLE ." WHERE meta_key='booking-note' AND object_id ='$booking_data'", ARRAY_A);
64
- foreach($notes as $note){
65
- $this->notes[] = unserialize($note['meta_value']);
66
- }
67
- }
68
- //Save into the object
69
- $this->to_object($booking);
70
- if( !empty($booking['booking_date']) ){
71
- $this->timestamp = strtotime($booking['booking_date']);
72
- }
73
- }
74
- //Do it here so things appear in the po file.
75
- $this->status_array = array(
76
- 0 => __('Pending','dbem'),
77
- 1 => __('Approved','dbem'),
78
- 2 => __('Rejected','dbem'),
79
- 3 => __('Cacncelled','dbem')
80
- );
81
- }
82
-
83
- /**
84
- * Saves the booking into the database, whether a new or existing booking
85
- * @return boolean
86
- */
87
- function save(){
88
- global $wpdb;
89
- $table = $wpdb->prefix.EM_BOOKINGS_TABLE;
90
- do_action('em_booking_save_pre',$this);
91
- //First the person
92
- if($this->validate()){
93
- //Does this person exist?
94
- $person_result = $this->person->save();
95
- if( $person_result === false ){
96
- $this->errors = array_merge($this->errors, $this->person->errors);
97
- return false;
98
- }
99
- $this->person_id = $this->person->id;
100
-
101
- //Now we save the booking
102
- $data = $this->to_array();
103
- if($this->id != ''){
104
- $where = array( 'booking_id' => $this->id );
105
- $result = $wpdb->update($table, $data, $where, $this->get_types($data));
106
- $this->feedback_message = __('Changes saved');
107
- }else{
108
- $result = $wpdb->insert($table, $data, $this->get_types($data));
109
- $this->id = $wpdb->insert_id;
110
- $this->feedback_message = __('Your booking has been recorded','dbem');
111
- }
112
- if( $result === false ){
113
- $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
114
- $this->errors[] = __('There was a problem saving the booking.', 'dbem');
115
- }
116
- return apply_filters('em_booking_save', ( count($this->errors) == 0 ), $this);
117
- }else{
118
- $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
119
- $this->errors[] = __('There was a problem saving the booking.', 'dbem');
120
- return apply_filters('em_booking_save', false, $this);
121
- }
122
- return true;
123
- }
124
-
125
- /**
126
- * Load an record into this object by passing an associative array of table criterie to search for.
127
- * Returns boolean depending on whether a record is found or not.
128
- * @param $search
129
- * @return boolean
130
- */
131
- function get($search) {
132
- global $wpdb;
133
- $conds = array();
134
- foreach($search as $key => $value) {
135
- if( array_key_exists($key, $this->fields) ){
136
- $value = $wpdb->escape($value);
137
- $conds[] = "`$key`='$value'";
138
- }
139
- }
140
- $sql = "SELECT * FROM ". $wpdb->EM_BOOKINGS_TABLE ." WHERE " . implode(' AND ', $conds) ;
141
- $result = $wpdb->get_row($sql, ARRAY_A);
142
- if($result){
143
- $this->to_object($result);
144
- return true;
145
- }else{
146
- return false;
147
- }
148
- }
149
-
150
- /**
151
- * Get posted data and save it into the object (not db)
152
- * @return boolean
153
- */
154
- function get_post(){
155
- //Currently, only editing allowed, so here we are.
156
- $this->comment = (!empty($_REQUEST['booking_comment'])) ? $_REQUEST['booking_comment']:'';
157
- $this->seats = (!empty($_REQUEST['booking_seats'])) ? $_REQUEST['booking_seats']:'';
158
- return apply_filters('em_booking_get_post',$this->validate(),$this);
159
- }
160
-
161
- function validate(){
162
- return (
163
- (empty($this->event_id) || is_numeric($this->event_id)) &&
164
- (empty($this->person_id) || is_numeric($this->person_id)) &&
165
- is_numeric($this->seats)
166
- );
167
- }
168
-
169
- /**
170
- * Smart event locator, saves a database read if possible.
171
- */
172
- function get_event(){
173
- global $EM_Event;
174
- if( !empty($this->event) && is_object($this->event) && get_class($this->event)=='EM_Event' && $this->event->id == $this->event_id ){
175
- return $this->event;
176
- }elseif( is_object($EM_Event) && $EM_Event->id == $this->event_id ){
177
- $this->event = $EM_Event;
178
- }else{
179
- $this->event = new EM_Event($this->event_id);
180
- }
181
- return $this->event;
182
- }
183
-
184
- /**
185
- * Returns a string representation of the booking's status
186
- * @return string
187
- */
188
- function get_status(){
189
- return $this->status_array[$this->status];
190
- }
191
- /**
192
- * I wonder what this does....
193
- * @return boolean
194
- */
195
- function delete(){
196
- global $wpdb;
197
- $sql = $wpdb->prepare("DELETE FROM ". $wpdb->prefix.EM_BOOKINGS_TABLE . " WHERE booking_id=%d", $this->id);
198
- $result = $wpdb->query( $sql );
199
- return ( $result !== false );
200
- }
201
-
202
- function cancel(){
203
- return $this->set_status(3);
204
- }
205
-
206
- /**
207
- * Approve a booking.
208
- * @return bool
209
- */
210
- function approve(){
211
- return $this->set_status(1);
212
- }
213
- /**
214
- * Reject a booking and save
215
- * @return bool
216
- */
217
- function reject(){
218
- return $this->set_status(2);
219
- }
220
- /**
221
- * Unpprove a booking.
222
- * @return bool
223
- */
224
- function unapprove(){
225
- return $this->set_status(0);
226
- }
227
-
228
- /**
229
- * Change the status of the booking. This will save to the Database too.
230
- * @param unknown_type $status
231
- * @return string|string|string
232
- */
233
- function set_status($status){
234
- $this->previous_status = $this->status;
235
- $this->status = $status;
236
- $result = $this->save();
237
- $action_string = strtolower($this->status_array[$status]);
238
- if($result){
239
- $this->feedback_message = sprintf(__('Booking %s.','dbem'), $action_string);
240
- if( $this->email() ){
241
- $this->feedback_message .= " ".__('Mail Sent.','dbem');
242
- }elseif( $this->previous_status == 0 ){
243
- //extra errors may be logged by email() in EM_Object
244
- $this->feedback_message .= ' <span style="color:red">'.__('ERROR : Mail Not Sent.','dbem').'</span>';
245
- return false;
246
- }
247
- return true;
248
- }else{
249
- //errors should be logged by save()
250
- $this->feedback_message = sprintf(__('Booking could not be %s.','dbem'), $action_string);
251
- return false;
252
- }
253
- }
254
-
255
- /**
256
- * Add a booking note to this booking. returns wpdb result or false if use can't manage this event.
257
- * @param string $note
258
- * @return mixed
259
- */
260
- function add_note( $note_text ){
261
- global $wpdb;
262
- if( $this->can_manage() ){
263
- $note = array('author'=>get_current_user_id(),'note'=>$note_text,'timestamp'=>current_time('timestamp'));
264
- $this->notes[] = $note;
265
- $this->feedback_message = __('Booking note successfully added.','dbem');
266
- return $wpdb->insert($wpdb->prefix.EM_META_TABLE, array('object_id'=>$this->id, 'meta_key'=>'booking-note', 'meta_value'=> serialize($note)),array('%d','%s','%s'));
267
- }
268
- return false;
269
- }
270
-
271
- /**
272
- * @param EM_Booking $EM_Booking
273
- * @param EM_Event $event
274
- * @return boolean
275
- */
276
- function email(){
277
- global $EM_Mailer;
278
- $EM_Event = $this->get_event(); //We NEED event details here.
279
- //Make sure event matches booking, and that booking used to be approved.
280
- if( $this->previous_status == 0 || $this->status == 3 ){
281
- $contact_id = ( $EM_Event->contactperson_id != "") ? $EM_Event->contactperson_id : get_option('dbem_default_contact_person');
282
-
283
- $contact_subject = get_option('dbem_bookings_contact_email_subject');
284
- $contact_body = get_option('dbem_bookings_contact_email_body');
285
-
286
- if( (get_option('dbem_bookings_approval') == 0 && $this->previous_status === false) || $this->status == 1 ){
287
- $booker_subject = get_option('dbem_bookings_email_confirmed_subject');
288
- $booker_body = get_option('dbem_bookings_email_confirmed_body');
289
- }elseif( $this->status == 0 ){
290
- $booker_subject = get_option('dbem_bookings_email_pending_subject');
291
- $booker_body = get_option('dbem_bookings_email_pending_body');
292
- }elseif( $this->status == 2 ){
293
- $booker_subject = get_option('dbem_bookings_email_rejected_subject');
294
- $booker_body = get_option('dbem_bookings_email_rejected_body');
295
- }elseif( $this->status == 3 ){
296
- $booker_subject = get_option('dbem_bookings_email_cancelled_subject');
297
- $booker_body = get_option('dbem_bookings_email_cancelled_body');
298
- $contact_subject = get_option('dbem_contactperson_email_cancelled_subject');
299
- $contact_body = get_option('dbem_contactperson_email_cancelled_body');
300
- }
301
-
302
- // email specific placeholders
303
- $placeholders = array(
304
- '#_RESPNAME' => '#_BOOKINGNAME',//Depreciated
305
- '#_RESPEMAIL' => '#_BOOKINGEMAIL',//Depreciated
306
- '#_RESPPHONE' => '#_BOOKINGPHONE',//Depreciated
307
- '#_COMMENT' => '#_BOOKINGCOMMENT',//Depreciated
308
- '#_RESERVEDSPACES' => '#_BOOKEDSPACES',//Depreciated
309
- '#_BOOKINGNAME' => $this->person->name,
310
- '#_BOOKINGEMAIL' => $this->person->email,
311
- '#_BOOKINGPHONE' => $this->person->phone,
312
- '#_BOOKINGSPACES' => $this->seats,
313
- '#_BOOKINGCOMMENT' => $this->comment,
314
- );
315
- foreach($placeholders as $key => $value) {
316
- $contact_subject = str_replace($key, $value, $contact_subject);
317
- $contact_body = str_replace($key, $value, $contact_body);
318
- $booker_subject = str_replace($key, $value, $booker_subject);
319
- $booker_body = str_replace($key, $value, $booker_body);
320
- }
321
-
322
- $booker_subject = $EM_Event->output($booker_subject, 'email');
323
- $booker_body = $EM_Event->output($booker_body, 'email');
324
-
325
- //Send to the person booking
326
- if( !$this->email_send( $booker_subject,$booker_body, $this->person->email) ){
327
- return false;
328
- }
329
-
330
- //Send admin emails
331
- if( (get_option('dbem_bookings_approval') == 0 || $this->status == 0) && (get_option('dbem_bookings_contact_email') == 1 || get_option('dbem_bookings_notify_admin') != '') ){
332
- //Only gets sent if this is a pending booking, unless approvals are disabled.
333
- $contact_subject = $EM_Event->output($contact_subject, 'email');
334
- $contact_body = $EM_Event->output($contact_body, 'email');
335
-
336
- if( get_option('dbem_bookings_contact_email') == 1 ){
337
- if( !$this->email_send( $contact_subject, $contact_body, $EM_Event->contact->user_email) && current_user_can('activate_plugins')){
338
- $this->errors[] = __('Confirmation email could not be sent to contact person. Registrant should have gotten their email (only admin see this warning).','dbem');
339
- return false;
340
- }
341
- }
342
-
343
- if( get_option('dbem_bookings_notify_admin') != '' && preg_match('/^[_\.0-9a-z-]+@([0-9a-z][0-9a-z-]+\.)+[a-z]{2,3}$/', get_option('dbem_bookings_notify_admin')) ){
344
- if( !$this->email_send( $contact_subject, $contact_body, get_option('dbem_bookings_notify_admin')) ){
345
- $this->errors[] = __('Confirmation email could not be sent to admin. Registrant should have gotten their email (only admin see this warning).','dbem');
346
- return false;
347
- }
348
- }
349
- }
350
- return true;
351
- }
352
- return false;
353
- //TODO need error checking for booking mail send
354
- }
355
-
356
- /**
357
- * Can the user manage this event?
358
- */
359
- function can_manage(){
360
- return ( get_option('dbem_permissions_events') || $this->get_event()->author == get_current_user_id() || em_verify_admin() );
361
- }
362
-
363
- /**
364
- * Returns this object in the form of an array
365
- * @return array
366
- */
367
- function to_array($person = false){
368
- $booking = array();
369
- //Core Data
370
- $booking = parent::to_array();
371
- //Person Data
372
- if($person && is_object($this->person)){
373
- $person = $this->person->to_array();
374
- $booking = array_merge($booking, $person);
375
- }
376
- return $booking;
377
- }
378
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
379
  ?>
1
+ <?php
2
+ class EM_Booking extends EM_Object{
3
+ //DB Fields
4
+ var $booking_id;
5
+ var $event_id;
6
+ var $person_id;
7
+ var $booking_price;
8
+ var $booking_spaces;
9
+ var $booking_comment;
10
+ var $booking_status = 0;
11
+ var $booking_meta = array();
12
+ var $fields = array(
13
+ 'booking_id' => array('name'=>'id','type'=>'%d'),
14
+ 'event_id' => array('name'=>'event_id','type'=>'%d'),
15
+ 'person_id' => array('name'=>'person_id','type'=>'%d'),
16
+ 'booking_price' => array('name'=>'price','type'=>'%d'),
17
+ 'booking_spaces' => array('name'=>'spaces','type'=>'%d'),
18
+ 'booking_comment' => array('name'=>'comment','type'=>'%s'),
19
+ 'booking_status' => array('name'=>'status','type'=>'%d'),
20
+ 'booking_meta' => array('name'=>'meta','type'=>'%s')
21
+ );
22
+ //Other Vars
23
+ var $notes = array(); //loaded from em_meta table in construct
24
+ var $timestamp;
25
+ var $person;
26
+ var $required_fields = array('booking_id', 'event_id', 'person_id', 'booking_spaces');
27
+ var $feedback_message = "";
28
+ var $errors = array();
29
+ /**
30
+ * Contains an array of custom fields for a booking. This is loaded from em_meta, where the booking_custom name contains arrays of data.
31
+ * @var array
32
+ */
33
+ var $custom = array();
34
+ /**
35
+ * If saved in this instance, you can see what previous approval status was.
36
+ * @var int
37
+ */
38
+ var $previous_status;
39
+ /**
40
+ * The booking approval status number corresponds to a state in this array.
41
+ * @var unknown_type
42
+ */
43
+ var $status_array = array();
44
+ /**
45
+ * @var EM_Tickets
46
+ */
47
+ var $tickets;
48
+ /**
49
+ * @var EM_Event
50
+ */
51
+ var $event;
52
+ /**
53
+ * @var EM_Tickets_Bookings
54
+ */
55
+ var $tickets_bookings;
56
+ /**
57
+ * If set to true, this booking can be managed by any logged in user.
58
+ * @var EM_Tickets_Bookings
59
+ */
60
+ var $manage_override;
61
+
62
+ /**
63
+ * Creates booking object and retreives booking data (default is a blank booking object). Accepts either array of booking data (from db) or a booking id.
64
+ * @param mixed $booking_data
65
+ * @return null
66
+ */
67
+ function EM_Booking( $booking_data = false ){
68
+ //Get the person for this booking
69
+ if( $booking_data !== false ){
70
+ //Load booking data
71
+ $booking = array();
72
+ if( is_array($booking_data) ){
73
+ $booking = $booking_data;
74
+ }elseif( is_numeric($booking_data) ){
75
+ //Retreiving from the database
76
+ global $wpdb;
77
+ $sql = "SELECT * FROM ". EM_BOOKINGS_TABLE ." LEFT JOIN ". EM_META_TABLE ." ON object_id=booking_id WHERE booking_id ='$booking_data'";
78
+ $booking = $wpdb->get_row($sql, ARRAY_A);
79
+ //Custom Fields
80
+ $custom = $wpdb->get_row("SELECT meta_key, meta_value FROM ". EM_BOOKINGS_TABLE ." LEFT JOIN ". EM_META_TABLE ." ON object_id=booking_id WHERE booking_id ='$booking_data' AND meta_key='booking_custom'");
81
+ //Booking notes
82
+ $notes = $wpdb->get_results("SELECT * FROM ". EM_META_TABLE ." WHERE meta_key='booking-note' AND object_id ='$booking_data'", ARRAY_A);
83
+ foreach($notes as $note){
84
+ $this->notes[] = unserialize($note['meta_value']);
85
+ }
86
+ }
87
+ //booking meta
88
+ $booking['booking_meta'] = (!empty($booking['booking_meta'])) ? unserialize($booking['booking_meta']):array();
89
+ //Save into the object
90
+ $this->to_object($booking);
91
+ $this->get_person();
92
+ $this->timestamp = !empty($booking['booking_date']) ? strtotime($booking['booking_date']):false;
93
+ //Add custom booking data
94
+ if( !empty($custom['meta_key']) && $custom['meta_key'] == 'booking_custom' && is_serialized($custom['meta_value']) ){
95
+ $this->custom = unserialize($custom['meta_value']);
96
+ }
97
+ }
98
+ //Do it here so things appear in the po file.
99
+ $this->status_array = array(
100
+ 0 => __('Pending','dbem'),
101
+ 1 => __('Approved','dbem'),
102
+ 2 => __('Rejected','dbem'),
103
+ 3 => __('Cancelled','dbem'),
104
+ 4 => __('Awaiting Online Payment','dbem'),
105
+ 5 => __('Awaiting Payment','dbem')
106
+ );
107
+ $this->compat_keys();
108
+ do_action('em_booking', $this, $booking_data);
109
+ }
110
+
111
+ /**
112
+ * Saves the booking into the database, whether a new or existing booking
113
+ * @param $mail whether or not to email the user and contact people
114
+ * @return boolean
115
+ */
116
+ function save($mail = true){
117
+ global $wpdb;
118
+ $table = EM_BOOKINGS_TABLE;
119
+ do_action('em_booking_save_pre',$this);
120
+ if( $this->validate() ){
121
+ if( $this->can_manage() ){
122
+ $this->person_id = (empty($this->person_id)) ? $this->get_person()->ID : $this->person_id;
123
+ //Step 1. Save the booking
124
+ $data = $this->to_array();
125
+ $data['booking_meta'] = serialize($data['booking_meta']);
126
+ if($this->booking_id != ''){
127
+ $update = true;
128
+ //update price and spaces
129
+ $this->get_spaces(true);
130
+ $this->get_price(true);
131
+ $where = array( 'booking_id' => $this->booking_id );
132
+ $result = $wpdb->update($table, $data, $where, $this->get_types($data));
133
+ $result = ($result !== false);
134
+ $this->feedback_message = __('Changes saved','dbem');
135
+ }else{
136
+ $update = false;
137
+ $result = $wpdb->insert($table, $data, $this->get_types($data));
138
+ $this->booking_id = $wpdb->insert_id;
139
+ $this->feedback_message = __('Your booking has been recorded','dbem');
140
+ }
141
+ //Step 2. Insert ticket bookings for this booking id if no errors so far
142
+ if( $result === false ){
143
+ $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
144
+ $this->errors[] = __('There was a problem saving the booking.', 'dbem');
145
+ }else{
146
+ $tickets_bookings_result = $this->get_tickets_bookings()->save();
147
+ if( !$tickets_bookings_result ){
148
+ if( !$update ){
149
+ //delete the booking and tickets, instead of a transaction
150
+ $this->delete();
151
+ }
152
+ $this->errors[] = __('There was a problem saving the booking.', 'dbem');
153
+ $this->add_error( $this->get_tickets_bookings()->get_errors() );
154
+ }
155
+ }
156
+ //Step 3. email if necessary
157
+ if ( count($this->errors) == 0 && $mail ) {
158
+ $this->email();
159
+ }
160
+ $this->compat_keys();
161
+ return apply_filters('em_booking_save', ( count($this->errors) == 0 ), $this);
162
+ }else{
163
+ $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
164
+ if( !$this->can_manage() ){
165
+ $this->feedback_message = sprintf(__('You cannot manage this %s.', 'dbem'),__('Booking','dbem'));
166
+ }
167
+ }
168
+ }else{
169
+ $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
170
+ if( !$this->can_manage() ){
171
+ $this->feedback_message = sprintf(__('You cannot manage this %s.', 'dbem'),__('Booking','dbem'));
172
+ }
173
+ }
174
+ return apply_filters('em_booking_save', false, $this);
175
+ }
176
+
177
+ /**
178
+ * Load an record into this object by passing an associative array of table criterie to search for.
179
+ * Returns boolean depending on whether a record is found or not.
180
+ * @param $search
181
+ * @return boolean
182
+ */
183
+ function get($search) {
184
+ global $wpdb;
185
+ $conds = array();
186
+ foreach($search as $key => $value) {
187
+ if( array_key_exists($key, $this->fields) ){
188
+ $value = $wpdb->escape($value);
189
+ $conds[] = "`$key`='$value'";
190
+ }
191
+ }
192
+ $sql = "SELECT * FROM ". $wpdb->EM_BOOKINGS_TABLE ." WHERE " . implode(' AND ', $conds) ;
193
+ $result = $wpdb->get_row($sql, ARRAY_A);
194
+ if($result){
195
+ $this->to_object($result);
196
+ $this->person = new EM_Person($this->person_id);
197
+ return true;
198
+ }else{
199
+ return false;
200
+ }
201
+ }
202
+
203
+ /**
204
+ * Get posted data and save it into the object (not db)
205
+ * @return boolean
206
+ */
207
+ function get_post( $override_availability = false ){
208
+ $this->tickets_bookings = new EM_Tickets_Bookings($this->booking_id);
209
+ do_action('em_booking_get_post_pre',$this);
210
+ $result = array();
211
+ $this->event_id = $_REQUEST['event_id'];
212
+ if( isset($_REQUEST['em_tickets']) && is_array($_REQUEST['em_tickets']) && ($_REQUEST['em_tickets'] || $override_availability) ){
213
+ foreach( $_REQUEST['em_tickets'] as $ticket_id => $values){
214
+ //make sure ticket exists
215
+ if( !empty($values['spaces']) || $override_availability ){
216
+ $args = array('ticket_id'=>$ticket_id, 'ticket_booking_spaces'=>$values['spaces'], 'booking_id'=>$this->booking_id);
217
+ if($this->get_event()->get_bookings()->ticket_exists($ticket_id)){
218
+ $EM_Ticket_Booking = new EM_Ticket_Booking($args);
219
+ $EM_Ticket_Booking->booking = $this;
220
+ $this->tickets_bookings->add( $EM_Ticket_Booking, $override_availability );
221
+ }else{
222
+ $this->errors[]=__('You are trying to book a non-existent ticket for this event.','dbem');
223
+ }
224
+ }
225
+ }
226
+ $this->booking_comment = (!empty($_REQUEST['booking_comment'])) ? wp_kses_data(stripslashes($_REQUEST['booking_comment'])):'';
227
+ $this->get_spaces(true);
228
+ $this->get_price(true, false, false);
229
+ $this->get_person();
230
+ $this->compat_keys();
231
+ }
232
+ return apply_filters('em_booking_get_post',$this->validate(),$this);
233
+ }
234
+
235
+ function validate(){
236
+ //step 1, basic info
237
+ $basic = (
238
+ (empty($this->event_id) || is_numeric($this->event_id)) &&
239
+ (empty($this->person_id) || is_numeric($this->person_id)) &&
240
+ is_numeric($this->booking_spaces) && $this->booking_spaces > 0
241
+ );
242
+ //give some errors in step 1
243
+ if( $this->booking_spaces == 0 ){
244
+ $this->add_error(get_option('dbem_booking_feedback_min_space'));
245
+ }
246
+ //step 2, tickets bookings info
247
+ if( count($this->get_tickets_bookings()) > 0 ){
248
+ $ticket_validation = array();
249
+ foreach($this->get_tickets_bookings()->tickets_bookings as $EM_Ticket_Booking){
250
+ if ( !$EM_Ticket_Booking->validate() ){
251
+ $ticket_validation[] = false;
252
+ $result = $basic && !in_array(false,$ticket_validation);
253
+ }
254
+ $this->errors = array_merge($this->errors, $EM_Ticket_Booking->get_errors());
255
+ }
256
+ $result = $basic && !in_array(false,$ticket_validation);
257
+ }else{
258
+ $result = false;
259
+ }
260
+
261
+ return apply_filters('em_booking_validate',$result,$this);
262
+ }
263
+
264
+ /**
265
+ * Get the total number of spaces booked in THIS booking. Seting $force_refresh to true will recheck spaces, even if previously done so.
266
+ * @param unknown_type $force_refresh
267
+ * @return mixed
268
+ */
269
+ function get_spaces( $force_refresh=false ){
270
+ if($this->booking_spaces == 0 || $force_refresh == true ){
271
+ $this->booking_spaces = $this->get_tickets_bookings()->get_spaces($force_refresh);
272
+ }
273
+ return apply_filters('em_booking_get_spaces',$this->booking_spaces,$this);
274
+ }
275
+
276
+ /**
277
+ * Gets the total price for this whole booking. Seting $force_reset to true will recheck spaces, even if previously done so.
278
+ * @param boolean $force_refresh
279
+ * @param boolean $format
280
+ * @param boolean $add_tax
281
+ * @return float
282
+ */
283
+ function get_price( $force_refresh=false, $format=false, $add_tax='x' ){
284
+ if($force_refresh || $this->booking_price == 0 || $add_tax !== 'x' || get_option('dbem_bookings_tax_auto_add')){
285
+ $this->booking_price = $this->get_tickets_bookings()->get_price($force_refresh, false, $add_tax);
286
+ }
287
+ if($format){
288
+ return apply_filters('em_booking_get_price', em_get_currency_symbol().number_format($this->booking_price,2),$this);
289
+ }
290
+ return apply_filters('em_booking_get_price',$this->booking_price,$this);
291
+ }
292
+
293
+ /**
294
+ * Gets the event this booking belongs to and saves a refernece in the event property
295
+ * @return EM_Event
296
+ */
297
+ function get_event(){
298
+ global $EM_Event;
299
+ if( is_object($this->event) && get_class($this->event)=='EM_Event' && $this->event->event_id == $this->event_id ){
300
+ return $this->event;
301
+ }elseif( is_object($EM_Event) && ( (is_object($this->event) && $this->event->event_id == $this->event_id) || empty($this->booking_id)) ){
302
+ $this->event = $EM_Event;
303
+ }else{
304
+ $this->event = new EM_Event($this->event_id, 'event_id');
305
+ }
306
+ return apply_filters('em_booking_get_event',$this->event);
307
+ }
308
+
309
+ /**
310
+ * Outdated, use booking meta array. Get custom fields for this booking.
311
+ * @return array
312
+ */
313
+ function get_custom(){
314
+ global $wpdb;
315
+ if( count($this->custom) == 0 ){
316
+ $sql = "SELECT * FROM ". EM_META_TABLE ." WHERE object_id ='{$this->booking_id}' AND (meta_key='booking_custom' OR meta_key IS NULL)";
317
+ $booking = $wpdb->get_row($sql, ARRAY_A);
318
+ //Add custom booking data
319
+ if( !empty($booking['meta_key']) && $booking['meta_key'] == 'booking_custom' && is_serialized($booking['meta_value']) ){
320
+ $this->custom = unserialize($booking['meta_value']);
321
+ }
322
+ }
323
+ return $this->custom;
324
+ }
325
+
326
+ /**
327
+ * Gets the ticket object this booking belongs to, saves a reference in ticket property
328
+ * @return EM_Tickets
329
+ */
330
+ function get_tickets(){
331
+ if( is_object($this->tickets) && get_class($this->tickets)=='EM_Tickets' ){
332
+ return apply_filters('em_booking_get_tickets', $this->tickets, $this);
333
+ }else{
334
+ $this->tickets = new EM_Tickets($this);
335
+ }
336
+ return apply_filters('em_booking_get_tickets', $this->tickets, $this);
337
+ }
338
+
339
+ /**
340
+ * Gets the ticket object this booking belongs to, saves a reference in ticket property
341
+ * @return EM_Tickets_Bookings
342
+ */
343
+ function get_tickets_bookings(){
344
+ global $wpdb;
345
+ if( !is_object($this->tickets_bookings) || get_class($this->tickets_bookings)!='EM_Tickets_Bookings'){
346
+ $this->tickets_bookings = new EM_Tickets_Bookings($this);
347
+ }
348
+ return apply_filters('em_booking_get_tickets_bookings', $this->tickets_bookings, $this);
349
+ }
350
+
351
+ function get_person(){
352
+ global $EM_Person;
353
+ if( is_object($this->person) && get_class($this->person)=='EM_Person' && ($this->person->ID == $this->person_id || empty($this->person_id) ) ){
354
+ //This person is already included, so don't do anything
355
+ }elseif( is_object($EM_Person) && ($EM_Person->ID === $this->person_id || $this->booking_id == '') ){
356
+ $this->person = $EM_Person;
357
+ }elseif( is_numeric($this->person_id) ){
358
+ $this->person = new EM_Person($this->person_id);
359
+ }else{
360
+ $this->person = new EM_Person(0);
361
+ }
362
+ //if this user is the parent user of disabled registrations, replace user details here:
363
+ if( get_option('dbem_bookings_registration_disable') && $this->person->ID == get_option('dbem_bookings_registration_user') ){
364
+ //override any registration data into the person objet
365
+ if( !empty($this->booking_meta['registration']) ){
366
+ foreach($this->booking_meta['registration'] as $key => $value){
367
+ $this->person->$key = $value;
368
+ }
369
+ }
370
+ $this->person->user_email = ( !empty($this->booking_meta['registration']['user_email']) ) ? $this->booking_meta['registration']['user_email']:$this->person->user_email;
371
+ if( !empty($this->booking_meta['registration']['user_name']) ){
372
+ $name_string = explode(' ',$this->booking_meta['registration']['user_name']);
373
+ $this->booking_meta['registration']['first_name'] = array_shift($name_string);
374
+ $this->booking_meta['registration']['last_name'] = implode(' ', $name_string);
375
+ }
376
+ $this->person->user_firstname = ( !empty($this->booking_meta['registration']['first_name']) ) ? $this->booking_meta['registration']['first_name']:__('Guest User','dbem');
377
+ $this->person->first_name = $this->person->user_firstname;
378
+ $this->person->user_lastname = ( !empty($this->booking_meta['registration']['last_name']) ) ? $this->booking_meta['registration']['last_name']:'';
379
+ $this->person->last_name = $this->person->user_lastname;
380
+ $this->person->phone = ( !empty($this->booking_meta['registration']['dbem_phone']) ) ? $this->booking_meta['registration']['dbem_phone']:__('Not Supplied','dbem');
381
+ //build display name
382
+ $full_name = $this->user_firstname . " " . $this->user_lastname ;
383
+ $full_name = trim($full_name);
384
+ $display_name = ( empty($full_name) ) ? __('Guest User','dbem'):$full_name;
385
+ $this->person->display_name = $display_name;
386
+ }
387
+ return apply_filters('em_booking_get_person', $this->person, $this);
388
+ }
389
+
390
+ /**
391
+ * Returns a string representation of the booking's status
392
+ * @return string
393
+ */
394
+ function get_status(){
395
+ $status = ($this->booking_status == 0 && !get_option('dbem_bookings_approval') ) ? 1:$this->booking_status;
396
+ return $this->status_array[$status];
397
+ }
398
+
399
+ /**
400
+ * I wonder what this does....
401
+ * @return boolean
402
+ */
403
+ function delete(){
404
+ global $wpdb;
405
+ $result = false;
406
+ if( $this->can_manage('manage_bookings','manage_others_bookings') ){
407
+ $sql = $wpdb->prepare("DELETE FROM ". EM_BOOKINGS_TABLE . " WHERE booking_id=%d", $this->booking_id);
408
+ $result = $wpdb->query( $sql );
409
+ if( $result !== false ){
410
+ //delete the tickets too
411
+ $this->get_tickets_bookings()->delete();
412
+ $this->previous_status = $this->booking_status;
413
+ $this->booking_status = false;
414
+ $this->feedback_message = sprintf(__('%s deleted', 'dbem'), __('Booking','dbem'));
415
+ }else{
416
+ $this->add_error(sprintf(__('%s could not be deleted', 'dbem'), __('Booking','dbem')));
417
+ }
418
+ }
419
+ return apply_filters('em_booking_delete',( $result !== false ), $this);
420
+ }
421
+
422
+ function cancel(){
423
+ if( $this->person->ID == get_current_user_id() ){
424
+ $this->manage_override = true; //normally, users can't manage a bookiing, only event owners, so we allow them to mod their booking status in this case only.
425
+ }
426
+ return $this->set_status(3);
427
+ }
428
+
429
+ /**
430
+ * Approve a booking.
431
+ * @return bool
432
+ */
433
+ function approve(){
434
+ return $this->set_status(1);
435
+ }
436
+ /**
437
+ * Reject a booking and save
438
+ * @return bool
439
+ */
440
+ function reject(){
441
+ return $this->set_status(2);
442
+ }
443
+ /**
444
+ * Unpprove a booking.
445
+ * @return bool
446
+ */
447
+ function unapprove(){
448
+ return $this->set_status(0);
449
+ }
450
+
451
+ /**
452
+ * Change the status of the booking. This will save to the Database too.
453
+ * @param int $status
454
+ * @return boolean
455
+ */
456
+ function set_status($status){
457
+ global $wpdb;
458
+ $action_string = strtolower($this->status_array[$status]);
459
+ //if we're approving we can't approve a booking if spaces are full, so check before it's approved.
460
+ if($status == 1){
461
+ if( $this->get_event()->get_bookings()->get_available_spaces() < $this->get_spaces() && !get_option('dbem_bookings_approval_overbooking') ){
462
+ $this->feedback_message = sprintf(__('Not approved, spaces full.','dbem'), $action_string);
463
+ return apply_filters('em_booking_set_status', false, $this);
464
+ }
465
+ }
466
+ $this->previous_status = $this->booking_status;
467
+ $this->booking_status = $status;
468
+ $result = $wpdb->query($wpdb->prepare('UPDATE '.EM_BOOKINGS_TABLE.' SET booking_status=%d WHERE booking_id=%d', array($status, $this->booking_id)));
469
+ if($result !== false){
470
+ $this->feedback_message = sprintf(__('Booking %s.','dbem'), $action_string);
471
+ if( !($this->booking_status == 0 && $this->previous_status > 0) || $this->previous_status == 4 ){
472
+ if( $this->email() ){
473
+ $this->feedback_message .= " ".__('Mail Sent.','dbem');
474
+ }elseif( $this->previous_status == 0 ){
475
+ //extra errors may be logged by email() in EM_Object
476
+ $this->feedback_message .= ' <span style="color:red">'.__('ERROR : Mail Not Sent.','dbem').'</span>';
477
+ $this->add_error(__('ERROR : Mail Not Sent.','dbem'));
478
+ $result = false;
479
+ }
480
+ }
481
+ }else{
482
+ //errors should be logged by save()
483
+ $this->feedback_message = sprintf(__('Booking could not be %s.','dbem'), $action_string);
484
+ $this->add_error(sprintf(__('Booking could not be %s.','dbem'), $action_string));
485
+ }
486
+ return apply_filters('em_booking_set_status', $result, $this);
487
+ }
488
+
489
+ /**
490
+ * Add a booking note to this booking. returns wpdb result or false if use can't manage this event.
491
+ * @param string $note
492
+ * @return mixed
493
+ */
494
+ function add_note( $note_text ){
495
+ global $wpdb;
496
+ if( $this->can_manage() ){
497
+ $note = array('author'=>get_current_user_id(),'note'=>$note_text,'timestamp'=>current_time('timestamp'));
498
+ $this->notes[] = $note;
499
+ $this->feedback_message = __('Booking note successfully added.','dbem');
500
+ return $wpdb->insert(EM_META_TABLE, array('object_id'=>$this->booking_id, 'meta_key'=>'booking-note', 'meta_value'=> serialize($note)),array('%d','%s','%s'));
501
+ }
502
+ return false;
503
+ }
504
+
505
+ function output($format, $target="html") {
506
+ preg_match_all("/(#@?_?[A-Za-z0-9]+)({([^}]+)})?/", $format, $placeholders);
507
+ foreach( $this->get_tickets() as $EM_Ticket){ break; } //Get first ticket for single ticket placeholders
508
+ $output_string = $format;
509
+ foreach($placeholders[1] as $key => $result) {
510
+ $replace = '';
511
+ $full_result = $placeholders[0][$key];
512
+ switch( $result ){
513
+ case '#_BOOKINGID':
514
+ $replace = $this->booking_id;
515
+ break;
516
+ case '#_RESPNAME' : //Depreciated
517
+ case '#_BOOKINGNAME':
518
+ $replace = $this->get_person()->get_name();
519
+ break;
520
+ case '#_RESPEMAIL' : //Depreciated
521
+ case '#_BOOKINGEMAIL':
522
+ $replace = $this->get_person()->user_email;
523
+ break;
524
+ case '#_RESPPHONE' : //Depreciated
525
+ case '#_BOOKINGPHONE':
526
+ $replace = $this->get_person()->phone;
527
+ break;
528
+ case '#_BOOKINGSPACES':
529
+ $replace = $this->get_spaces();
530
+ break;
531
+ case '#_BOOKINGLISTURL':
532
+ $replace = em_get_my_bookings_url();
533
+ break;
534
+ case '#_COMMENT' : //Depreciated
535
+ case '#_BOOKINGCOMMENT':
536
+ $replace = $this->booking_comment;
537
+ break;
538
+ case '#_BOOKINGPRICEWITHTAX':
539
+ $replace = em_get_currency_symbol(true)." ". number_format($this->get_price(false,false,true),2);
540
+ break;
541
+ case '#_BOOKINGPRICEWITHOUTTAX':
542
+ $replace = em_get_currency_symbol(true)." ". number_format($this->get_price(false,false,false),2);
543
+ break;
544
+ case '#_BOOKINGPRICETAX':
545
+ $replace = em_get_currency_symbol(true)." ". number_format($this->get_price(false,false,false)*(get_option('dbem_bookings_tax')/100),2);
546
+ break;
547
+ case '#_BOOKINGPRICE':
548
+ $replace = em_get_currency_symbol(true)." ". number_format($this->get_price(),2);
549
+ break;
550
+ case '#_BOOKINGTICKETDESCRIPTION':
551
+ $replace = $EM_Ticket->description;
552
+ break;
553
+ case '#_BOOKINGTICKETPRICEWITHTAX':
554
+ $replace = em_get_currency_symbol(true)." ". number_format($EM_Ticket->get_price(false,true),2);
555
+ break;
556
+ case '#_BOOKINGTICKETPRICEWITHOUTTAX':
557
+ $replace = em_get_currency_symbol(true)." ". number_format($EM_Ticket->get_price(false,false),2);
558
+ break;
559
+ case '#_BOOKINGTICKETTAX':
560
+ $replace = em_get_currency_symbol(true)." ". number_format($EM_Ticket->get_price(false,false)*(get_option('dbem_bookings_tax')/100),2);
561
+ break;
562
+ case '#_BOOKINGTICKETPRICE':
563
+ $replace = em_get_currency_symbol(true)." ". number_format($EM_Ticket->get_price(),2);
564
+ break;
565
+ case '#_BOOKINGTICKETS':
566
+ ob_start();
567
+ em_locate_template('emails/bookingtickets.php', true, array('EM_Booking'=>$this));
568
+ $replace = ob_get_clean();
569
+ break;
570
+ default:
571
+ $replace = $full_result;
572
+ break;
573
+ }
574
+ $replace = apply_filters('em_booking_output_placeholder', $replace, $this, $full_result, $target);
575
+ $output_string = str_replace($full_result, $replace , $output_string );
576
+ }
577
+ $output_string = $this->get_event()->output($output_string, $target);
578
+ return apply_filters('em_booking_output', $output_string, $this, $format, $target);
579
+ }
580
+
581
+ /**
582
+ * @param EM_Booking $EM_Booking
583
+ * @param EM_Event $event
584
+ * @return boolean
585
+ */
586
+ function email(){
587
+ global $EM_Mailer;
588
+ //FIXME ticket logic needed
589
+ $EM_Event = $this->get_event(); //We NEED event details here.
590
+ //Make sure event matches booking, and that booking used to be approved.
591
+ if( !($this->booking_status == 0 && $this->previous_status > 0) || $this->previous_status == 4 ){
592
+ $contact_id = ( $EM_Event->owner != "") ? $EM_Event->owner : get_option('dbem_default_contact_person');
593
+
594
+ $contact_subject = get_option('dbem_bookings_contact_email_subject');
595
+ $contact_body = get_option('dbem_bookings_contact_email_body');
596
+
597
+ if( (get_option('dbem_bookings_approval') == 0 && $this->booking_status < 2) || $this->booking_status == 1 ){
598
+ $booker_subject = get_option('dbem_bookings_email_confirmed_subject');
599
+ $booker_body = get_option('dbem_bookings_email_confirmed_body');
600
+ }elseif( $this->booking_status == 0 || $this->booking_status == 5 || ( $this->booking_status == 0 && ($this->previous_status == 4 || $this->previous_status == 5) ) ){
601
+ $booker_subject = get_option('dbem_bookings_email_pending_subject');
602
+ $booker_body = get_option('dbem_bookings_email_pending_body');
603
+ }elseif( $this->booking_status == 2 ){
604
+ $booker_subject = get_option('dbem_bookings_email_rejected_subject');
605
+ $booker_body = get_option('dbem_bookings_email_rejected_body');
606
+ }elseif( $this->booking_status == 3 ){
607
+ $booker_subject = get_option('dbem_bookings_email_cancelled_subject');
608
+ $booker_body = get_option('dbem_bookings_email_cancelled_body');
609
+ $contact_subject = get_option('dbem_contactperson_email_cancelled_subject');
610
+ $contact_body = get_option('dbem_contactperson_email_cancelled_body');
611
+ }else{
612
+ return true;
613
+ }
614
+
615
+ $booker_subject = $this->output($booker_subject, 'email');
616
+ $booker_body = $this->output($booker_body, 'email');
617
+ if( get_option('dbem_smtp_html') ){
618
+ $booker_body = nl2br($booker_body);
619
+ }
620
+ //Send to the person booking
621
+ if( !$this->email_send( $booker_subject,$booker_body, $this->get_person()->user_email) ){
622
+ return false;
623
+ }
624
+
625
+ //Send admin/contact emails
626
+ if( (get_option('dbem_bookings_approval') == 0 || in_array($this->booking_status, array(0,3,4,5)) || (in_array($this->previous_status, array(4)) && $this->booking_status == 1)) && (get_option('dbem_bookings_contact_email') == 1 || get_option('dbem_bookings_notify_admin') != '') ){
627
+ //Only gets sent if this is a pending booking, unless approvals are disabled.
628
+ $contact_subject = $this->output($contact_subject, 'email');
629
+ $contact_body = $this->output($contact_body, 'email');
630
+ if( get_option('dbem_smtp_html') ){
631
+ $contact_body = nl2br($contact_body);
632
+ }
633
+
634
+ if( get_option('dbem_bookings_contact_email') == 1 ){
635
+ if( !$this->email_send( $contact_subject, $contact_body, $EM_Event->get_contact()->user_email) && current_user_can('activate_plugins')){
636
+ $this->errors[] = __('Confirmation email could not be sent to contact person. Registrant should have gotten their email (only admin see this warning).','dbem');
637
+ return false;
638
+ }
639
+ }
640
+
641
+ if( get_option('dbem_bookings_notify_admin') != '' && preg_match('/^[_\.0-9a-z-]+@([0-9a-z][0-9a-z-]+\.)+[a-z]{2,3}$/', get_option('dbem_bookings_notify_admin')) ){
642
+ if( !$this->email_send( $contact_subject, $contact_body, get_option('dbem_bookings_notify_admin')) ){
643
+ $this->errors[] = __('Confirmation email could not be sent to admin. Registrant should have gotten their email (only admin see this warning).','dbem');
644
+ return false;
645
+ }
646
+ }
647
+ }
648
+ return true;
649
+ }
650
+ return false;
651
+ //TODO need error checking for booking mail send
652
+ }
653
+
654
+ /**
655
+ * Can the user manage this event?
656
+ */
657
+ function can_manage(){
658
+ return $this->get_event()->can_manage('manage_bookings','manage_others_bookings') || empty($this->booking_id) || !empty($this->manage_override);
659
+ }
660
+
661
+ /**
662
+ * Returns this object in the form of an array
663
+ * @return array
664
+ */
665
+ function to_array($person = false){
666
+ $booking = array();
667
+ //Core Data
668
+ $booking = parent::to_array();
669
+ //Person Data
670
+ if($person && is_object($this->person)){
671
+ $person = $this->person->to_array();
672
+ $booking = array_merge($booking, $person);
673
+ }
674
+ return $booking;
675
+ }
676
+ }
677
  ?>
classes/em-bookings.php CHANGED
@@ -1,488 +1,586 @@
1
- <?php
2
- /**
3
- * Deals with the booking info for an event
4
- * @author marcus
5
- *
6
- */
7
- class EM_Bookings extends EM_Object{
8
-
9
- /**
10
- * Array of EM_Booking objects for a specific event
11
- * @var array
12
- */
13
- var $bookings = array();
14
- /**
15
- * Event ID
16
- * @var int
17
- */
18
- var $event_id;
19
- /**
20
- * Number of seats for this event
21
- * @var int
22
- */
23
- var $seats;
24
-
25
- var $feedback_message = "";
26
- var $errors = array();
27
-
28
- /**
29
- * Creates an EM_Bookings instance,
30
- * @param EM_Event $event
31
- * @return null
32
- */
33
- function EM_Bookings( $event = false ){
34
- if( is_object($event) && get_class($event) == "EM_Event" ){ //Creates a blank bookings object if needed
35
- global $wpdb;
36
- $this->event_id = $event->id;
37
- $this->seats = $event->seats;
38
- $sql = "SELECT * FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." b, ". $wpdb->prefix . EM_PEOPLE_TABLE ." p WHERE event_id ='{$this->event_id}' AND p.person_id=b.person_id";
39
- $bookings = $wpdb->get_results($sql, ARRAY_A);
40
- foreach ($bookings as $booking){
41
- $this->bookings[] = new EM_Booking($booking);
42
- }
43
- }
44
- }
45
-
46
- /**
47
- * Add a booking into this event (or add seats if person already booked this), checking that there's enough space for the event
48
- * @param $EM_Booking
49
- * @return boolean
50
- */
51
- function add( $EM_Booking ){
52
- global $wpdb,$EM_Mailer;
53
- if ( $this->get_available_seats() >= $EM_Booking->seats ) {
54
- $EM_Booking->event_id = $this->event_id;
55
- // checking whether the booker has already booked places
56
- $previous_booking = $this->find_previous_booking( $EM_Booking );
57
- $email = false;
58
- if ( is_object($previous_booking) ) {
59
- //Previously booked, so we add these seats to the booking
60
- $new_seats = $EM_Booking->seats;
61
- $EM_Booking = $previous_booking;
62
- $EM_Booking->seats += $new_seats;
63
- $result = $EM_Booking->save();
64
- if($result){
65
- //remove old booking
66
- foreach($this->bookings as $key=>$booking){
67
- if($booking->id == $EM_Booking->id){ unset($this->bookings[$key]); }
68
- }
69
- $this->bookings[] = $EM_Booking;
70
- $email = $EM_Booking->email();
71
- }
72
- } else {
73
- //New booking, so let's save the booking
74
- $result = $EM_Booking->save();
75
- if($result){
76
- $this->bookings[] = $EM_Booking;
77
- $email = $EM_Booking->email();
78
- }
79
- }
80
- if($result){
81
- //Success
82
- if( get_option('dbem_bookings_approval') == 1 ){
83
- $this->feedback_message = __('Booking successful, pending confirmation (you will also receive an email once confirmed).', 'dbem');
84
- }else{
85
- $this->feedback_message = __('Booking successful.', 'dbem');
86
- }
87
- if(!$email){
88
- $this->feedback_message .= ' '.__('However, there were some problems whilst sending confirmation emails to you and/or the event contact person. You may want to contact them directly and letting them know of this error.', 'dbem');
89
- if( current_user_can('activate_plugins') ){
90
- if( is_array($this->errors) && count($this->errors) > 0 ){
91
- $this->feedback_message .= '<br/><strong>Errors:</strong> (only admins see this message)<br/><ul><li>'. implode('</li><li>', $EM_Mailer->errors).'</li></ul>';
92
- }else{
93
- $this->feedback_message .= '<br/><strong>No errors returned by mailer</strong> (only admins see this message)';
94
- }
95
- }
96
- }
97
- return true;
98
- }else{
99
- //Failure
100
- $this->errors[] = "<strong>".__('Booking could not be created','dbem').":</strong><br />". implode('<br />', $EM_Booking->errors);
101
- }
102
- } else {
103
- $this->errors[] = __('Booking cannot be made, not enough seats available!', 'dbem');
104
- return false;
105
- }
106
- }
107
-
108
- /**
109
- * Delete bookings on this id
110
- * @return boolean
111
- */
112
- function delete(){
113
- global $wpdb;
114
- $result = $wpdb->query("DELETE FROM ".$wpdb->prefix.EM_BOOKINGS_TABLE." WHERE event_id='{$this->event_id}'");
115
- return ($result);
116
- }
117
-
118
-
119
- /**
120
- * Will approve all supplied booking ids, which must be in the form of a numeric array or a single number.
121
- * @param array|int $booking_ids
122
- * @return boolean
123
- */
124
- function approve( $booking_ids ){
125
- $this->set_status(1, $booking_ids);
126
- return false;
127
- }
128
-
129
- /**
130
- * Will reject all supplied booking ids, which must be in the form of a numeric array or a single number.
131
- * @param array|int $booking_ids
132
- * @return boolean
133
- */
134
- function reject( $booking_ids ){
135
- return $this->set_status(2, $booking_ids);
136
- }
137
-
138
- /**
139
- * Will unapprove all supplied booking ids, which must be in the form of a numeric array or a single number.
140
- * @param array|int $booking_ids
141
- * @return boolean
142
- */
143
- function unapprove( $booking_ids ){
144
- return $this->set_status(0, $booking_ids);
145
- }
146
-
147
- /**
148
- * @param int $status
149
- * @param array|int $booking_ids
150
- * @return bool
151
- */
152
- function set_status($status, $booking_ids){
153
- //FIXME there is a vulnerability where any user can approve/reject bookings if they know the ID
154
- if( EM_Object::array_is_numeric($booking_ids) ){
155
- //Get all the bookings
156
- $results = array();
157
- $mails = array();
158
- foreach( $booking_ids as $booking_id ){
159
- $EM_Booking = new EM_Booking($booking_id);
160
- $results[] = $EM_Booking->set_status($status);
161
- }
162
- if( !in_array('false',$results) ){
163
- $this->feedback_message = __('Bookings %s. Mails Sent.', 'dbem');
164
- return true;
165
- }else{
166
- //TODO Better error handling needed if some bookings fail approval/failure
167
- $this->feedback_message = __('An error occurred.', 'dbem');
168
- return false;
169
- }
170
- }elseif( is_numeric($booking_ids) || is_object($booking_ids) ){
171
- $EM_Booking = ( is_object($booking_ids) && get_class($booking_ids) == 'EM_Booking') ? $booking_ids : new EM_Booking($booking_ids);
172
- $result = $EM_Booking->set_status($status);
173
- $this->feedback_message = $EM_Booking->feedback_message;
174
- return $result;
175
- }
176
- return false;
177
- }
178
-
179
- /**
180
- * Returns number of available seats for this event. If approval of bookings is on, will include pending bookings depending on em option.
181
- * @return int
182
- */
183
- function get_available_seats(){
184
- $booked_seats = 0;
185
- if( get_option('dbem_bookings_approval_reserved') == 1 ){
186
- return $this->seats - $this->get_booked_seats() - $this->get_pending_seats();
187
- }else{
188
- return $this->seats - $this->get_booked_seats();
189
- }
190
- }
191
-
192
- /**
193
- * Returns number of booked seats for this event. If approval of bookings is on, will return number of booked confirmed seats.
194
- * @return int
195
- */
196
- function get_booked_seats(){
197
- $booked_seats = 0;
198
- foreach ( $this->bookings as $booking ){
199
- if( $booking->status != 3 && (get_option('dbem_bookings_approval') == 0 || $booking->status == 1) ){
200
- $booked_seats += $booking->seats;
201
- }
202
- }
203
- return $booked_seats;
204
- }
205
-
206
- /**
207
- * Gets number of pending seats awaiting approval. Will return 0 if booking approval is not enabled.
208
- * @return int
209
- */
210
- function get_pending_seats(){
211
- if( get_option('dbem_bookings_approval') == 0 ){
212
- return 0;
213
- }
214
- $pending = 0;
215
- foreach ( $this->bookings as $booking ){
216
- if($booking->status == 0){
217
- $pending += $booking->seats;
218
- }
219
- }
220
- return $pending;
221
- }
222
-
223
- /**
224
- * Gets number of bookings (not seats). If booking approval is enabled, only the number of approved bookings will be shown.
225
- * @return array EM_Booking
226
- */
227
- function get_bookings(){
228
- $confirmed = array();
229
- foreach ( $this->bookings as $booking ){
230
- if( $booking->status == 1 || (get_option('dbem_bookings_approval') == 0 && $booking->status < 2) ){
231
- $confirmed[] = $booking;
232
- }
233
- }
234
- return $confirmed;
235
- }
236
-
237
- /**
238
- * Get pending bookings. If booking approval is disabled, will return no bookings.
239
- * @return array EM_Booking
240
- */
241
- function get_pending_bookings(){
242
- if( get_option('dbem_bookings_approval') == 0 ){
243
- return array();
244
- }
245
- $pending = array();
246
- foreach ( $this->bookings as $booking ){
247
- if($booking->status == 0){
248
- $pending[] = $booking;
249
- }
250
- }
251
- return $pending;
252
- }
253
-
254
- /**
255
- * Get rejected bookings.
256
- * @return array EM_Booking
257
- */
258
- function get_rejected_bookings(){
259
- $pending = array();
260
- foreach ( $this->bookings as $booking ){
261
- if($booking->status == 2){
262
- $pending[] = $booking;
263
- }
264
- }
265
- return $pending;
266
- }
267
-
268
- /**
269
- * Get cancelled bookings.
270
- * @return array EM_Booking
271
- */
272
- function get_cancelled_bookings(){
273
- $pending = array();
274
- foreach ( $this->bookings as $booking ){
275
- if($booking->status == 3){
276
- $pending[] = $booking;
277
- }
278
- }
279
- return $pending;
280
- }
281
-
282
- /**
283
- * Checks if a person with similar details has booked for this before
284
- * @param $person_id
285
- * @return EM_Booking
286
- */
287
- function find_previous_booking($EM_Booking){
288
- //First see if we have a similar person on record that's making this booking
289
- $EM_Booking->person->load_similar();
290
- //If person exists on record, see if they've booked this event before, if so return the booking.
291
- if( is_numeric($EM_Booking->person->id) && $EM_Booking->person->id > 0 ){
292
- $EM_Booking->person_id = $EM_Booking->person->id;
293
- foreach ($this->bookings as $booking){
294
- if( $booking->person_id == $EM_Booking->person->id ){
295
- return $booking;
296
- }
297
- }
298
- }
299
- return false;
300
- }
301
-
302
-
303
- /**
304
- * Get all pending bookings for this event
305
- */
306
- function get_pending(){
307
- $pending = array();
308
- foreach($this->bookings as $booking){
309
- if($booking->status == 0){
310
- $pending[] = $booking;
311
- }
312
- }
313
- return $pending;
314
- }
315
-
316
- /**
317
- * Gets the pending number of bookings as a raw associative array.
318
- * @return array
319
- */
320
- function get( $args = array() ){
321
- global $wpdb,$current_user;
322
- $bookings_table = $wpdb->prefix . EM_BOOKINGS_TABLE;
323
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
324
- $people_table = $wpdb->prefix . EM_PEOPLE_TABLE;
325
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
326
-
327
- //Quick version, we can accept an array of IDs, which is easy to retrieve
328
- if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
329
- //We can just get all the events here and return them
330
- $sql = "
331
- SELECT * FROM $bookings_table b
332
- LEFT JOIN $events_table e ON e.event_id=b.event_id
333
- LEFT JOIN $people_table p ON p.person_id=b.person_id
334
- WHERE booking_id".implode(" OR booking_id=", $args);
335
- $results = $wpdb->get_results(apply_filters('em_bookings_get_sql',$sql),ARRAY_A);
336
- $bookings = array();
337
- foreach($results as $result){
338
- $bookings[$result['event_id']] = new EM_Event($result);
339
- }
340
- return $bookings; //We return all the events matched as an EM_Event array.
341
- }
342
-
343
- //We assume it's either an empty array or array of search arguments to merge with defaults
344
- $args = self::get_default_search($args);
345
- $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
346
- $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
347
-
348
- //Get the default conditions
349
- $conditions = self::build_sql_conditions($args);
350
- //Put it all together
351
- $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
352
-
353
- //Get ordering instructions
354
- $EM_Booking = new EM_Booking();
355
- $accepted_fields = $EM_Booking->get_fields(true);
356
- $orderby = self::build_sql_orderby($args, $accepted_fields);
357
- //Now, build orderby sql
358
- $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
359
-
360
- //Create the SQL statement and execute
361
- $sql = "
362
- SELECT * FROM $bookings_table
363
- LEFT JOIN $events_table ON {$events_table}.event_id={$bookings_table}.event_id
364
- LEFT JOIN $people_table ON {$people_table}.person_id={$bookings_table}.person_id
365
- LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
366
- $where
367
- $orderby_sql
368
- $limit $offset
369
- ";
370
-
371
- $results = $wpdb->get_results( apply_filters('em_events_get_sql',$sql, $args), ARRAY_A);
372
-
373
- //If we want results directly in an array, why not have a shortcut here?
374
- if( $args['array'] == true ){
375
- return $results;
376
- }
377
-
378
- //Make returned results EM_Booking objects
379
- $results = (is_array($results)) ? $results:array();
380
- $bookings = array();
381
- foreach ( $results as $booking ){
382
- $bookings[] = new EM_Booking($booking);
383
- }
384
-
385
- return apply_filters('em_bookings_get', $bookings);
386
- }
387
-
388
-
389
- //List of patients in the patient database, that a user can choose and go on to edit any previous treatment data, or add a new admission.
390
- function export_csv() {
391
- global $EM_Event;
392
- if($EM_Event->id != $this->event_id ){
393
- $event = new EM_Event($this->event_id);
394
- $event_name = $event->name;
395
- }else{
396
- $event_name = $EM_Event->name;
397
- }
398
- // The name of the file on the user's pc
399
- $file_name = sanitize_title($event_name). "-bookings.csv";
400
-
401
- header("Content-Type: application/octet-stream");
402
- header("Content-Disposition: Attachment; filename=$file_name");
403
-
404
- //Headers
405
- $labels = array(
406
- 'ID',
407
- 'Name',
408
- 'Email',
409
- 'Phone',
410
- 'Date',
411
- 'Status',
412
- 'Spaces',
413
- 'Comment'
414
- );
415
- $file = sprintf(__('Booking details for "%s" as of %s','dbem'),$event_name, date_i18n('D d M Y h:i', current_time('timestamp'))) . "\n";
416
- $file = '"'. strtolower(implode('","', $labels)). '"' . "\n";
417
-
418
- //Rows
419
- foreach( $this->bookings as $EM_Booking ) {
420
- $row = array(
421
- $EM_Booking->id,
422
- $EM_Booking->person->name,
423
- $EM_Booking->person->email,
424
- $EM_Booking->person->phone,
425
- date('Y-m-d h:i', $EM_Booking->timestamp),
426
- $EM_Booking->seats,
427
- $EM_Booking->get_status(),
428
- $EM_Booking->comment
429
- );
430
- //Display all values
431
- foreach($row as $value){
432
- $value = str_replace('"', '""', $value);
433
- $value = str_replace("=", "", $value);
434
- $file .= '"' . preg_replace("/\n\r|\r\n|\n|\r/", ". ", $value) . '",';
435
- }
436
- $file .= "\n";
437
- }
438
-
439
- // $file holds the data
440
- echo $file;
441
- $file = "";
442
- }
443
-
444
- /* Overrides EM_Object method to apply a filter to result
445
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
446
- */
447
- function build_sql_conditions( $args = array() ){
448
- $conditions = apply_filters( 'em_bookings_build_sql_conditions', parent::build_sql_conditions($args), $args );
449
- if( is_numeric($args['status']) ){
450
- $conditions['status'] = 'booking_status='.$args['status'];
451
- }
452
- return apply_filters('em_bookings_build_sql_conditions', $conditions, $args);
453
- }
454
-
455
- /* Overrides EM_Object method to apply a filter to result
456
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
457
- */
458
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
459
- return apply_filters( 'em_bookings_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
460
- }
461
-
462
- /*
463
- * Adds custom Events search defaults
464
- * @param array $array
465
- * @return array
466
- * @uses EM_Object#get_default_search()
467
- */
468
- function get_default_search( $array = array() ){
469
- $defaults = array(
470
- 'status' => false,
471
- 'person' => true //to add later, search by person's bookings...
472
- );
473
- if( true || is_admin() ){
474
- //figure out default owning permissions
475
- switch( get_option('dbem_permissions_events') ){
476
- case 0:
477
- $defaults['owner'] = get_current_user_id();
478
- break;
479
- case 1:
480
- $defaults['owner'] = false;
481
- break;
482
- }
483
- $defaults['owner'] = ( em_verify_admin() ) ? false:$defaults['owner'];
484
- }
485
- return apply_filters('em_bookings_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
486
- }
487
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
488
  ?>
1
+ <?php
2
+ /**
3
+ * Deals with the booking info for an event
4
+ * @author marcus
5
+ *
6
+ */
7
+ class EM_Bookings extends EM_Object implements Iterator{
8
+
9
+ /**
10
+ * Array of EM_Booking objects for a specific event
11
+ * @var array
12
+ */
13
+ var $bookings = array();
14
+ /**
15
+ * @var EM_Tickets
16
+ */
17
+ var $tickets;
18
+ /**
19
+ * @var int
20
+ */
21
+ var $event_id;
22
+ /**
23
+ * How many spaces this event has
24
+ * @var int
25
+ */
26
+ var $spaces;
27
+
28
+ /**
29
+ * Creates an EM_Bookings instance, currently accepts an EM_Event object (gets all bookings for that event) or array of any EM_Booking objects, which can be manipulated in bulk with helper functions.
30
+ * @param EM_Event $event
31
+ * @return null
32
+ */
33
+ function EM_Bookings( $data = false ){
34
+ if( is_object($data) && get_class($data) == "EM_Event" ){ //Creates a blank bookings object if needed
35
+ global $wpdb;
36
+ $this->event_id = $data->event_id;
37
+ $sql = "SELECT * FROM ". EM_BOOKINGS_TABLE ." WHERE event_id ='{$this->event_id}'";
38
+ $bookings = $wpdb->get_results($sql, ARRAY_A);
39
+ foreach ($bookings as $booking){
40
+ $this->bookings[] = new EM_Booking($booking);
41
+ }
42
+ $this->spaces = $this->get_spaces();
43
+ }elseif( is_array($data) ){
44
+ foreach( $data as $EM_Booking ){
45
+ if( get_class($EM_Booking) == 'EM_Booking'){
46
+ $this->bookings[] = $EM_Booking;
47
+ }
48
+ }
49
+ }
50
+ }
51
+
52
+ /**
53
+ * Add a booking into this event (or add spaces if person already booked this), checking that there's enough space for the event
54
+ * @param EM_Booking $EM_Booking
55
+ * @return boolean
56
+ */
57
+ function add( $EM_Booking ){
58
+ global $wpdb,$EM_Mailer;
59
+ if ( $this->get_available_spaces() >= $EM_Booking->get_spaces(true) ) {
60
+ //Save the booking
61
+ $email = false;
62
+ if( !get_option('dbem_bookings_approval') && $EM_Booking->booking_status < 2 ){
63
+ $EM_Booking->booking_status = 1;
64
+ }
65
+ $result = $EM_Booking->save(false);
66
+ if($result){
67
+ //Success
68
+ $this->bookings[] = $EM_Booking;
69
+ $email = $EM_Booking->email();
70
+ if( get_option('dbem_bookings_approval') == 1 && $EM_Booking->booking_status == 0){
71
+ $this->feedback_message = get_option('dbem_booking_feedback_pending');
72
+ }else{
73
+ $this->feedback_message = get_option('dbem_booking_feedback');
74
+ }
75
+ if(!$email){
76
+ $this->feedback_message .= ' '.get_option('dbem_booking_feedback_nomail');
77
+ if( current_user_can('activate_plugins') ){
78
+ if( count($EM_Booking->get_errors()) > 0 ){
79
+ $this->feedback_message .= '<br/><strong>Errors:</strong> (only admins see this message)<br/><ul><li>'. implode('</li><li>', $EM_Booking->get_errors()).'</li></ul>';
80
+ }else{
81
+ $this->feedback_message .= '<br/><strong>No errors returned by mailer</strong> (only admins see this message)';
82
+ }
83
+ }
84
+ }
85
+ return true;
86
+ }else{
87
+ //Failure
88
+ $this->errors[] = "<strong>".get_option('dbem_booking_feedback_error')."</strong><br />". implode('<br />', $EM_Booking->errors);
89
+ }
90
+ } else {
91
+ $this->add_error(get_option('dbem_booking_feedback_full'));
92
+ }
93
+ return apply_filters('em_bookings_add', false, $EM_Booking);
94
+ }
95
+
96
+ /**
97
+ * Get POST data and create a booking for each ticket requested. If successful, a booking object is returned, false if not.
98
+ * @return false|object
99
+ */
100
+ function add_from_post(){
101
+ $EM_Booking = new EM_booking();
102
+ $result = $EM_Booking->get_post();
103
+ if($result){
104
+ $result = $this->add($EM_Booking);
105
+ if($result){
106
+ $result = $EM_Booking;
107
+ }
108
+ $this->feedback_message = sprintf(__('%s created.','dbem'),__('Booking','dbem'));
109
+ }else{
110
+ $this->errors = array_merge($this->errors, $EM_Booking->errors);
111
+ }
112
+ return apply_filters('em_bookings_add_from_post',$result,$EM_Booking,$this);
113
+ }
114
+
115
+ /**
116
+ * Smart event locator, saves a database read if possible. Note that if an event doesn't exist, a blank object will be created to prevent duplicates.
117
+ */
118
+ function get_event(){
119
+ global $EM_Event;
120
+ if( is_object($EM_Event) && $EM_Event->event_id == $this->event_id ){
121
+ return $EM_Event;
122
+ }else{
123
+ if( is_numeric($this->event_id) && $this->event_id > 0 ){
124
+ return new EM_Event($this->event_id, 'event_id');
125
+ }elseif( count($this->bookings) > 0 ){
126
+ foreach($this->bookings as $EM_Booking){
127
+ /* @var $EM_Booking EM_Booking */
128
+ return new EM_Event($EM_Booking->event_id, 'event_id');
129
+ }
130
+ }
131
+ }
132
+ return new EM_Event($this->event_id);
133
+ }
134
+
135
+ /**
136
+ * Retrieve and save the bookings belonging to instance. If called again will return cached version, set $force_reload to true to create a new EM_Tickets object.
137
+ * @param boolean $force_reload
138
+ * @return EM_Tickets
139
+ */
140
+ function get_tickets( $force_reload = false ){
141
+ if( !is_object($this->tickets) || $force_reload ){
142
+ $this->tickets = new EM_Tickets($this->event_id);
143
+ }else{
144
+ $this->tickets->event_id = $this->event_id;
145
+ }
146
+ return apply_filters('em_bookings_get_tickets', $this->tickets, $this);
147
+ }
148
+
149
+ /**
150
+ * Returns EM_Tickets object with available tickets
151
+ * @return EM_Tickets
152
+ */
153
+ function get_available_tickets(){
154
+ $tickets = array();
155
+ $timesamp = current_time('timestamp');
156
+ foreach ($this->get_tickets() as $EM_Ticket){
157
+ /* @var EM_Ticket $EM_Ticket */
158
+ if( $EM_Ticket->is_available() ){
159
+ //within time range
160
+ if( $EM_Ticket->get_available_spaces() > 0 ){
161
+ $tickets[] = $EM_Ticket;
162
+ }
163
+ }
164
+ }
165
+ $EM_Tickets = new EM_Tickets($tickets);
166
+ return apply_filters('em_bookings_get_tickets', $EM_Tickets, $this);
167
+ }
168
+
169
+ function get_user_list(){
170
+ $users = array();
171
+ foreach( $this->get_bookings()->bookings as $EM_Booking ){
172
+ $users[$EM_Booking->person->ID] = $EM_Booking->person;
173
+ }
174
+ return $users;
175
+ }
176
+
177
+ /**
178
+ * does this ticket exist?
179
+ * @return bool
180
+ */
181
+ function ticket_exists($ticket_id){
182
+ $EM_Tickets = $this->get_tickets();
183
+ foreach( $EM_Tickets->tickets as $EM_Ticket){
184
+ if($EM_Ticket->ticket_id == $ticket_id){
185
+ return apply_filters('em_bookings_ticket_exists',true, $EM_Ticket, $this);
186
+ }
187
+ }
188
+ return apply_filters('em_bookings_ticket_exists',false, false,$this);
189
+ }
190
+
191
+ /**
192
+ * Delete bookings on this id
193
+ * @return boolean
194
+ */
195
+ function delete(){
196
+ global $wpdb;
197
+ $booking_ids = array();
198
+ //get the booking ids tied to this event
199
+ foreach( $this->bookings as $EM_Booking ){
200
+ $booking_ids[] = $EM_Booking->booking_id;
201
+ }
202
+ $result_tickets = true;
203
+ $result = true;
204
+ if( count($booking_ids) > 0 ){
205
+ //Delete bookings and ticket bookings
206
+ $result_tickets = $wpdb->query("DELETE FROM ". EM_TICKETS_BOOKINGS_TABLE ." WHERE booking_id IN (".implode(',',$booking_ids).");");
207
+ $result = $wpdb->query("DELETE FROM ".EM_BOOKINGS_TABLE." WHERE event_id IN (".implode(',',$booking_ids).")");
208
+ }
209
+ return ($result !== false && $result_tickets !== false);
210
+ }
211
+
212
+
213
+ /**
214
+ * Will approve all supplied booking ids, which must be in the form of a numeric array or a single number.
215
+ * @param array|int $booking_ids
216
+ * @return boolean
217
+ */
218
+ function approve( $booking_ids ){
219
+ $this->set_status(1, $booking_ids);
220
+ return false;
221
+ }
222
+
223
+ /**
224
+ * Will reject all supplied booking ids, which must be in the form of a numeric array or a single number.
225
+ * @param array|int $booking_ids
226
+ * @return boolean
227
+ */
228
+ function reject( $booking_ids ){
229
+ return $this->set_status(2, $booking_ids);
230
+ }
231
+
232
+ /**
233
+ * Will unapprove all supplied booking ids, which must be in the form of a numeric array or a single number.
234
+ * @param array|int $booking_ids
235
+ * @return boolean
236
+ */
237
+ function unapprove( $booking_ids ){
238
+ return $this->set_status(0, $booking_ids);
239
+ }
240
+
241
+ /**
242
+ * @param int $status
243
+ * @param array|int $booking_ids
244
+ * @return bool
245
+ */
246
+ function set_status($status, $booking_ids){
247
+ //FIXME status should work with instantiated object
248
+ //FIXME there is a vulnerability where any user can approve/reject bookings if they know the ID
249
+ if( $this->array_is_numeric($booking_ids) ){
250
+ //Get all the bookings
251
+ $results = array();
252
+ $mails = array();
253
+ foreach( $booking_ids as $booking_id ){
254
+ $EM_Booking = new EM_Booking($booking_id);
255
+ if( !$EM_Booking->can_manage() ){
256
+ $this->feedback_message = __('Bookings %s. Mails Sent.', 'dbem');
257
+ return false;
258
+ }
259
+ $results[] = $EM_Booking->set_status($status);
260
+ }
261
+ if( !in_array('false',$results) ){
262
+ $this->feedback_message = __('Bookings %s. Mails Sent.', 'dbem');
263
+ return true;
264
+ }else{
265
+ //TODO Better error handling needed if some bookings fail approval/failure
266
+ $this->feedback_message = __('An error occurred.', 'dbem');
267
+ return false;
268
+ }
269
+ }elseif( is_numeric($booking_ids) || is_object($booking_ids) ){
270
+ $EM_Booking = ( is_object($booking_ids) && get_class($booking_ids) == 'EM_Booking') ? $booking_ids : new EM_Booking($booking_ids);
271
+ $result = $EM_Booking->set_status($status);
272
+ $this->feedback_message = $EM_Booking->feedback_message;
273
+ return $result;
274
+ }
275
+ return false;
276
+ }
277
+
278
+
279
+ /**
280
+ * Get the total number of spaces this event has. This will show the lower value of event global spaces limit or total ticket spaces. Setting $force_refresh to true will recheck spaces, even if previously done so.
281
+ * @param boolean $force_refresh
282
+ * @return int
283
+ */
284
+ function get_spaces( $force_refresh=false ){
285
+ if($force_refresh || $this->spaces == 0){
286
+ $this->spaces = $this->get_tickets()->get_spaces();
287
+ }
288
+ return apply_filters('em_booking_get_spaces',$this->spaces,$this);
289
+ }
290
+
291
+ /**
292
+ * Returns number of available spaces for this event. If approval of bookings is on, will include pending bookings depending on em option.
293
+ * @return int
294
+ */
295
+ function get_available_spaces(){
296
+ $available_spaces = $this->get_spaces() - $this->get_booked_spaces();
297
+ return apply_filters('em_booking_get_available_spaces', $available_spaces, $this);
298
+ }
299
+
300
+ /**
301
+ * Returns number of booked spaces for this event. If approval of bookings is on, will return number of booked confirmed spaces.
302
+ * @return int
303
+ */
304
+ function get_booked_spaces($force_refresh = false){
305
+ $booked_spaces = 0;
306
+ $EM_Bookings = $this->get_bookings(true);
307
+ $reserved_pending = get_option('dbem_bookings_approval_reserved');
308
+ $auto_approval = get_option('dbem_bookings_approval');
309
+ foreach ( $EM_Bookings->bookings as $EM_Booking ){
310
+ //never show cancelled status, nor pending if approvals required
311
+ if( $EM_Booking->booking_status == 1 || ((!$auto_approval || $reserved_pending) && $EM_Booking->booking_status == 0) ){
312
+ $booked_spaces += $EM_Booking->get_spaces($force_refresh);
313
+ }
314
+ }
315
+ return $booked_spaces;
316
+ }
317
+
318
+ /**
319
+ * Gets number of pending spaces awaiting approval. Will return 0 if booking approval is not enabled.
320
+ * @return int
321
+ */
322
+ function get_pending_spaces(){
323
+ if( get_option('dbem_bookings_approval') == 0 ){
324
+ return 0;
325
+ }
326
+ $pending = 0;
327
+ foreach ( $this->bookings as $booking ){
328
+ if($booking->booking_status == 0){
329
+ $pending += $booking->get_spaces();
330
+ }
331
+ }
332
+ return apply_filters('em_bookings_get_pending_spaces', $pending, $this);
333
+ }
334
+
335
+ /**
336
+ * Gets number of bookings (not spaces). If booking approval is enabled, only the number of approved bookings will be shown.
337
+ * @return EM_Bookings
338
+ */
339
+ function get_bookings( $all_bookings = false ){
340
+ $confirmed = array();
341
+ foreach ( $this->bookings as $booking ){
342
+ if( $booking->booking_status == 1 || (get_option('dbem_bookings_approval') == 0 && $booking->booking_status == 0) || $all_bookings ){
343
+ $confirmed[] = $booking;
344
+ }
345
+ }
346
+ $EM_Bookings = new EM_Bookings($confirmed);
347
+ return $EM_Bookings;
348
+ }
349
+
350
+ /**
351
+ * Get pending bookings. If booking approval is disabled, will return no bookings.
352
+ * @return EM_Bookings
353
+ */
354
+ function get_pending_bookings(){
355
+ if( get_option('dbem_bookings_approval') == 0 ){
356
+ return array();
357
+ }
358
+ $pending = array();
359
+ foreach ( $this->bookings as $booking ){
360
+ if($booking->booking_status == 0){
361
+ $pending[] = $booking;
362
+ }
363
+ }
364
+ $EM_Bookings = new EM_Bookings($pending);
365
+ return $EM_Bookings;
366
+ }
367
+
368
+ /**
369
+ * Get rejected bookings. If booking approval is disabled, will return no bookings.
370
+ * @return array EM_Bookings
371
+ */
372
+ function get_rejected_bookings(){
373
+ $rejected = array();
374
+ foreach ( $this->bookings as $booking ){
375
+ if($booking->booking_status == 2){
376
+ $rejected[] = $booking;
377
+ }
378
+ }
379
+ $EM_Bookings = new EM_Bookings($rejected);
380
+ return $EM_Bookings;
381
+ }
382
+
383
+ /**
384
+ * Get cancelled bookings.
385
+ * @return array EM_Booking
386
+ */
387
+ function get_cancelled_bookings(){
388
+ $cancelled = array();
389
+ foreach ( $this->bookings as $booking ){
390
+ if($booking->booking_status == 3){
391
+ $cancelled[] = $booking;
392
+ }
393
+ }
394
+ $EM_Bookings = new EM_Bookings($cancelled);
395
+ return $EM_Bookings;
396
+ }
397
+
398
+ /**
399
+ * Checks if a person with similar details has booked for this before
400
+ * @param $person_id
401
+ * @return EM_Booking
402
+ */
403
+ function find_previous_booking($EM_Booking){
404
+ //First see if we have a similar person on record that's making this booking
405
+ $EM_Booking->person->load_similar();
406
+ //If person exists on record, see if they've booked this event before, if so return the booking.
407
+ if( is_numeric($EM_Booking->person->ID) && $EM_Booking->person->ID > 0 ){
408
+ $EM_Booking->person_id = $EM_Booking->person->ID;
409
+ foreach ($this->bookings as $booking){
410
+ if( $booking->person_id == $EM_Booking->person->ID ){
411
+ return $booking;
412
+ }
413
+ }
414
+ }
415
+ return false;
416
+ }
417
+
418
+ /**
419
+ * Checks to see if user has a booking for this event
420
+ * @param unknown_type $user_id
421
+ */
422
+ function has_booking( $user_id = false ){
423
+ if( $user_id === false ){
424
+ $user_id = get_current_user_id();
425
+ }
426
+ if( is_numeric($user_id) && $user_id > 0 ){
427
+ foreach ($this->bookings as $EM_Booking){
428
+ if( $EM_Booking->person->ID == $user_id && $EM_Booking->booking_status != 3 ){
429
+ return apply_filters('em_bookings_has_booking', $EM_Booking, $this);
430
+ }
431
+ }
432
+ }
433
+ return apply_filters('em_bookings_has_booking', false, $this);
434
+ }
435
+
436
+ /**
437
+ * Get bookings that match the array of arguments passed.
438
+ * @return array
439
+ * @static
440
+ */
441
+ function get( $args = array() ){
442
+ global $wpdb,$current_user;
443
+ $bookings_table = EM_BOOKINGS_TABLE;
444
+ $events_table = EM_EVENTS_TABLE;
445
+ $locations_table = EM_LOCATIONS_TABLE;
446
+
447
+ //Quick version, we can accept an array of IDs, which is easy to retrieve
448
+ if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
449
+ //We can just get all the events here and return them
450
+ $sql = "
451
+ SELECT * FROM $bookings_table b
452
+ LEFT JOIN $events_table e ON e.event_id=b.event_id
453
+ WHERE booking_id".implode(" OR booking_id=", $args);
454
+ $results = $wpdb->get_results(apply_filters('em_bookings_get_sql',$sql),ARRAY_A);
455
+ $bookings = array();
456
+ foreach($results as $result){
457
+ $bookings[] = new EM_Booking($result);
458
+ }
459
+ return $bookings; //We return all the bookings matched as an EM_Booking array.
460
+ }
461
+
462
+ //We assume it's either an empty array or array of search arguments to merge with defaults
463
+ $args = self::get_default_search($args);
464
+ $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
465
+ $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
466
+
467
+ //Get the default conditions
468
+ $conditions = self::build_sql_conditions($args);
469
+ //Put it all together
470
+ $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
471
+
472
+ //Get ordering instructions
473
+ $EM_Booking = new EM_Booking();
474
+ $accepted_fields = $EM_Booking->get_fields(true);
475
+ $orderby = self::build_sql_orderby($args, $accepted_fields);
476
+ //Now, build orderby sql
477
+ $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
478
+
479
+ //Create the SQL statement and execute
480
+ $sql = "
481
+ SELECT * FROM $bookings_table
482
+ LEFT JOIN $events_table ON {$events_table}.event_id={$bookings_table}.event_id
483
+ LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
484
+ $where
485
+ $orderby_sql
486
+ $limit $offset
487
+ ";
488
+ $results = $wpdb->get_results( apply_filters('em_events_get_sql',$sql, $args), ARRAY_A);
489
+
490
+ //If we want results directly in an array, why not have a shortcut here?
491
+ if( $args['array'] == true ){
492
+ return $results;
493
+ }
494
+
495
+ //Make returned results EM_Booking objects
496
+ $results = (is_array($results)) ? $results:array();
497
+ $bookings = array();
498
+ foreach ( $results as $booking ){
499
+ $bookings[] = new EM_Booking($booking);
500
+ }
501
+ $EM_Bookings = new EM_Bookings($bookings);
502
+ return apply_filters('em_bookings_get', $EM_Bookings);
503
+ }
504
+
505
+
506
+ //List of patients in the patient database, that a user can choose and go on to edit any previous treatment data, or add a new admission.
507
+ function export_csv() {
508
+ global $EM_Event;
509
+ if($EM_Event->event_id != $this->event_id ){
510
+ $event = $this->get_event();
511
+ $event_name = $event->name;
512
+ }else{
513
+ $event_name = $EM_Event->name;
514
+ }
515
+ // The name of the file on the user's pc
516
+ $file_name = sanitize_title($event_name). "-bookings.csv";
517
+
518
+ header("Content-Type: application/octet-stream");
519
+ header("Content-Disposition: Attachment; filename=$file_name");
520
+ em_locate_template('templates/csv-event-bookings.php', true);
521
+ exit();
522
+ }
523
+
524
+ /* Overrides EM_Object method to apply a filter to result
525
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
526
+ */
527
+ function build_sql_conditions( $args = array() ){
528
+ $conditions = apply_filters( 'em_bookings_build_sql_conditions', parent::build_sql_conditions($args), $args );
529
+ if( is_numeric($args['status']) ){
530
+ $conditions['status'] = 'booking_status='.$args['status'];
531
+ }
532
+ return apply_filters('em_bookings_build_sql_conditions', $conditions, $args);
533
+ }
534
+
535
+ /* Overrides EM_Object method to apply a filter to result
536
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
537
+ */
538
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
539
+ return apply_filters( 'em_bookings_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
540
+ }
541
+
542
+ /*
543
+ * Adds custom Events search defaults
544
+ * @param array $array
545
+ * @return array
546
+ * @uses EM_Object#get_default_search()
547
+ */
548
+ function get_default_search( $array = array() ){
549
+ $defaults = array(
550
+ 'status' => false,
551
+ 'person' => true //to add later, search by person's bookings...
552
+ );
553
+ if( true || is_admin() ){
554
+ //figure out default owning permissions
555
+ if( !current_user_can('edit_others_events') ){
556
+ $defaults['owner'] = get_current_user_id();
557
+ }else{
558
+ $defaults['owner'] = false;
559
+ }
560
+ }
561
+ return apply_filters('em_bookings_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
562
+ }
563
+
564
+ //Iterator Implementation
565
+ public function rewind(){
566
+ reset($this->bookings);
567
+ }
568
+ public function current(){
569
+ $var = current($this->bookings);
570
+ return $var;
571
+ }
572
+ public function key(){
573
+ $var = key($this->bookings);
574
+ return $var;
575
+ }
576
+ public function next(){
577
+ $var = next($this->bookings);
578
+ return $var;
579
+ }
580
+ public function valid(){
581
+ $key = key($this->bookings);
582
+ $var = ($key !== NULL && $key !== FALSE);
583
+ return $var;
584
+ }
585
+ }
586
  ?>
classes/em-calendar.php CHANGED
@@ -1,366 +1,339 @@
1
- <?php
2
- class EM_Calendar extends EM_Object {
3
-
4
- function init(){
5
- add_action('wp_head', array('EM_Calendar', 'insert_js'));
6
- }
7
-
8
- function output($args = array()) {
9
- global $wpdb;
10
-
11
- $args = self::get_default_search($args);
12
- $full = $args['full']; //For ZDE, don't delete pls
13
- $month = $args['month'];
14
- $year = $args['year'];
15
- $long_events = $args['long_events'];
16
-
17
- $week_starts_on_sunday = get_option('dbem_week_starts_sunday');
18
- $start_of_week = get_option('start_of_week');
19
-
20
- if( !(is_numeric($month) && $month <= 12 && $month > 0) ) {
21
- $month = date('m');
22
- }
23
- if( !( is_numeric($year) ) ){
24
- $year = date('Y');
25
- }
26
-
27
- // Get the first day of the month
28
- $month_start = mktime(0,0,0,$month, 1, $year);
29
- // Get friendly month name
30
- $month_name = date('M',$month_start);
31
- // Figure out which day of the week
32
- // the month starts on.
33
- $month_start_day = date('D', $month_start);
34
-
35
- switch($month_start_day){
36
- case "Sun": $offset = 0; break;
37
- case "Mon": $offset = 1; break;
38
- case "Tue": $offset = 2; break;
39
- case "Wed": $offset = 3; break;
40
- case "Thu": $offset = 4; break;
41
- case "Fri": $offset = 5; break;
42
- case "Sat": $offset = 6; break;
43
- }
44
- //We need to go back to the WP defined day when the week started, in case the event day is near the end
45
- $offset -= $start_of_week;
46
- if($offset<0)
47
- $offset += 7;
48
-
49
- // determine how many days are in the last month.
50
- $month_last = $month-1;
51
- $month_next = $month+1;
52
- $year_last = $year;
53
- $year_next = $year;
54
- if($month == 1) {
55
- $month_last = 12;
56
- $year_last = $year -1;
57
- }
58
- $num_days_last = self::days_in_month($month_last, $year_last);
59
-
60
- // determine how many days are in the current month.
61
- $num_days_current = self::days_in_month($month, $year);
62
- // Build an array for the current days
63
- // in the month
64
- for($i = 1; $i <= $num_days_current; $i++){
65
- $num_days_array[] = mktime(0,0,0,$month, $i, $year);
66
- }
67
- // Build an array for the number of days
68
- // in last month
69
- for($i = 1; $i <= $num_days_last; $i++){
70
- $num_days_last_array[] = mktime(0,0,0,$month_last, $i, $year_last);
71
- }
72
- // If the $offset from the starting day of the
73
- // week happens to be Sunday, $offset would be 0,
74
- // so don't need an offset correction.
75
-
76
- if($offset > 0){
77
- $offset_correction = array_slice($num_days_last_array, -$offset, $offset);
78
- $new_count = array_merge($offset_correction, $num_days_array);
79
- $offset_count = count($offset_correction);
80
- } else { // The else statement is to prevent building the $offset array.
81
- $offset_count = 0;
82
- $new_count = $num_days_array;
83
- }
84
- // count how many days we have with the two
85
- // previous arrays merged together
86
- $current_num = count($new_count);
87
-
88
- // Since we will have 5 HTML table rows (TR)
89
- // with 7 table data entries (TD)
90
- // we need to fill in 35 TDs
91
- // so, we will have to figure out
92
- // how many days to appened to the end
93
- // of the final array to make it 35 days.
94
-
95
-
96
- if($current_num > 35){
97
- $num_weeks = 6;
98
- $outset = (42 - $current_num);
99
- } elseif($current_num < 35){
100
- $num_weeks = 5;
101
- $outset = (35 - $current_num);
102
- }
103
- if($current_num == 35){
104
- $num_weeks = 5;
105
- $outset = 0;
106
- }
107
- // Outset Correction
108
- for($i = 1; $i <= $outset; $i++){
109
- $new_count[] = mktime(0,0,0,$month_next, $i, $year_next);
110
- }
111
- // Now let's "chunk" the $all_days array
112
- // into weeks. Each week has 7 days
113
- // so we will array_chunk it into 7 days.
114
- $weeks = array_chunk($new_count, 7);
115
-
116
-
117
-
118
- // Build Previous and Next Links
119
- $base_link = "?".$_SERVER['QUERY_STRING']."&amp;";
120
-
121
- $full ? $link_extra_class = "full-link" : $link_extra_class = '';
122
- //Get an array of arguments that don't include default valued args
123
- $link_args = self::get_link_args($args);
124
-
125
- //Get the previous link
126
- if($month == 1){
127
- $back_month = 12;
128
- $back_year = $year-1;
129
- } else {
130
- $back_month = $month -1;
131
- $back_year = $year;
132
- }
133
- $previous_link = "<a class='em-calnav $link_extra_class' href='?ajaxCalendar=1&amp;month={$back_month}&amp;year={$back_year}&amp;{$link_args}'>&lt;&lt;</a>";
134
-
135
- //Now the next
136
- if($month == 12){
137
- $next_month = 1;
138
- $next_year = $year+1;
139
- } else {
140
- $next_month = $month + 1;
141
- $next_year = $year;
142
- }
143
- $next_link = "<a class='em-calnav $link_extra_class' href='?ajaxCalendar=1&amp;month={$next_month}&amp;year={$next_year}&amp;{$link_args}'>&gt;&gt;</a>";
144
-
145
-
146
- $class = ($full) ? 'dbem-calendar-full' : 'dbem-calendar';
147
- $calendar="<div class='$class'><div style='display:none' class='month_n'>$month</div><div class='year_n' style='display:none' >$year</div>";
148
-
149
- $weekdays = array('Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday');
150
- $n = 0 ;
151
- while( $n < $start_of_week ) {
152
- $last_day = array_shift($weekdays);
153
- $weekdays[]= $last_day;
154
- $n++;
155
- }
156
-
157
- $days_initials = "";
158
- foreach($weekdays as $weekday) {
159
- $days_initials .= "<td>".self::translate_and_trim($weekday)."</td>";
160
- }
161
- $full ? $fullclass = 'fullcalendar' : $fullclass='';
162
- // Build the heading portion of the calendar table
163
- $calendar .= "<table class='dbem-calendar-table $fullclass'>\n".
164
- "<thead>\n<tr>\n".
165
- "<td>$previous_link</td><td class='month_name' colspan='5'>". apply_filters('dbem_calendar_output_month', ucfirst(date_i18n('M', $month_start)), $month_start, $args)." $year</td><td>$next_link</td>\n".
166
- "</tr>\n</thead>\n".
167
- "<tr class='days-names'>\n".
168
- $days_initials.
169
- "</tr>\n";
170
-
171
- // Now we break each key of the array
172
- // into a week and create a new table row for each
173
- // week with the days of that week in the table data
174
-
175
- $i = 0;
176
- $current_date = date('Y-m-d', current_time('timestamp'));
177
- foreach ( $weeks as $week ) {
178
- $calendar .= "<tr>\n";
179
- foreach ( $week as $d ) {
180
- if ($i < $offset_count) { //if it is PREVIOUS month
181
- $calendar .= "<td class='eventless-pre'>" . date ( 'j', $d ) . "</td>\n";
182
- }
183
- if (($i >= $offset_count) && ($i < ($num_weeks * 7) - $outset)) { // if it is THIS month
184
- $day = date ( 'j', $d );
185
- if ( $current_date == date('Y-m-d', $d) ){
186
- $calendar .= "<td class='eventless-today'>$day</td>\n";
187
- } else {
188
- $calendar .= "<td class='eventless'>$day</td>\n";
189
- }
190
- } elseif (($outset > 0)) { //if it is NEXT month
191
- if (($i >= ($num_weeks * 7) - $outset)) {
192
- $calendar .= "<td class='eventless-post'>" . date ( 'j', $d ) . "</td>\n";
193
- }
194
- }
195
- $i ++;
196
- }
197
- $calendar .= "</tr>\n";
198
- }
199
-
200
- $calendar .= " </table>\n</div>";
201
-
202
- // query the database for events in this time span
203
- if ($month == 1) {
204
- $month_pre=12;
205
- $month_post=2;
206
- $year_pre=$year-1;
207
- $year_post=$year;
208
- } elseif($month == 12) {
209
- $month_pre=11;
210
- $month_post=1;
211
- $year_pre=$year;
212
- $year_post=$year+1;
213
- } else {
214
- $month_pre=$month-1;
215
- $month_post=$month+1;
216
- $year_pre=$year;
217
- $year_post=$year;
218
- }
219
- $args['year'] = array($year_pre, $year_post);
220
- $args['month'] = array($month_pre, $month_post);
221
- $events = EM_Events::get($args);
222
-
223
- $eventful_days= array();
224
- if($events){
225
- //Go through the events and slot them into the right d-m index
226
- foreach($events as $event) {
227
- $event = apply_filters('em_calendar_output_loop_start', $event);
228
- if( $long_events ){
229
- //If $long_events is set then show a date as eventful if there is an multi-day event which runs during that day
230
- $event_start_date = mktime(0,0,0,$month_pre,1,$year_pre);
231
- $event_end_date = mktime(0,0,0,$month_post,date('t', $event_start_date),$year_post );
232
- if( $event_end_date == '' ) $event_end_date = $event_start_date;
233
- while( $event_start_date <= $event->end ){
234
- //Ensure date is within event dates, if so add to eventful days array
235
- if( $event_start_date > $event->start - (86400) ){ //subtract a day since start may be later in day
236
- $event_eventful_date = date('Y-m-d', $event_start_date);
237
- if( array_key_exists($event_eventful_date, $eventful_days) && is_array($eventful_days[$event_eventful_date]) ){
238
- $eventful_days[$event_eventful_date][] = $event;
239
- } else {
240
- $eventful_days[$event_eventful_date] = array($event);
241
- }
242
- }
243
- $event_start_date += (86400); //add a day
244
- }
245
- }else{
246
- //Only show events on the day that they start
247
- if( isset($eventful_days[$event->start_date]) && is_array($eventful_days[$event->start_date]) ){
248
- $eventful_days[$event->start_date][] = $event;
249
- } else {
250
- $eventful_days[$event->start_date] = array($event);
251
- }
252
- }
253
- $event = apply_filters('em_calendar_output_loop_end', $event);
254
- }
255
- }
256
-
257
- $event_format = get_option('dbem_full_calendar_event_format');
258
- $event_title_format = get_option('dbem_small_calendar_event_title_format');
259
- $event_title_separator_format = get_option('dbem_small_calendar_event_title_separator');
260
- $cells = array() ;
261
- foreach($eventful_days as $day_key => $events) {
262
- //Set the date into the key
263
- $event_start_date = explode('-', $day_key);
264
- $cells[$day_key]['day'] = ltrim($event_start_date[2],'0');
265
- $cells[$day_key]['month'] = $event_start_date[1];
266
- $cells[$day_key]['year'] = $event_start_date[0];
267
- $events_titles = array();
268
- foreach($events as $event) {
269
- $events_titles[] = $event->output($event_title_format);
270
- }
271
- $link_title = implode( $event_title_separator_format, $events_titles);
272
-
273
- $events_page_id = get_option('dbem_events_page');
274
- $event_page_link = get_permalink($events_page_id);
275
- if (stristr($event_page_link, "?"))
276
- $joiner = "&amp;";
277
- else
278
- $joiner = "?";
279
-
280
-
281
- $cells[$day_key]['cell'] = "<a title='$link_title' href='".$event_page_link.$joiner."calendar_day={$day_key}'>{$cells[$day_key]['day']}</a>";
282
- if ($full) {
283
- $cells[$day_key]['cell'] .= "<ul>";
284
-
285
- foreach($events as $event) {
286
- $cells[$day_key]['cell'] .= $event->output($event_format);
287
- }
288
- $cells[$day_key]['cell'] .= "</ul>";
289
- }
290
- }
291
-
292
- if($events){
293
- foreach($cells as $cell) {
294
- if ($cell['month'] == $month_pre && $cell['year'] == $year_pre) {
295
- $calendar = str_replace("<td class='eventless-pre'>".$cell['day']."</td>","<td class='eventful-pre'>".$cell['cell']."</td>",$calendar);
296
- } elseif($cell['month'] == $month_post && $cell['year'] == $year_post) {
297
- $calendar = str_replace("<td class='eventless-post'>".$cell['day']."</td>","<td class='eventful-post'>".$cell['cell']."</td>",$calendar);
298
- } elseif( date('Y-m-j', current_time('timestamp')) == $cell['year']."-".$cell['month']."-".$cell['day'] ) {
299
- $calendar = str_replace("<td class='eventless-today'>".$cell['day']."</td>","<td class='eventful-today'>".$cell['cell']."</td>",$calendar);
300
- } elseif( $cell['month'] == $month && $cell['year'] == $year){
301
- $calendar = str_replace("<td class='eventless'>".$cell['day']."</td>","<td class='eventful'>".$cell['cell']."</td>",$calendar);
302
- }
303
- }
304
- }
305
- return apply_filters('em_calendar_output', '<div id="em-calendar-'.rand(100,200).'" class="em-calendar-wrapper">'.$calendar.'</div>');
306
- }
307
-
308
- /**
309
- * Echoes the calendar external JS contents directly into the head of the document
310
- * @return unknown_type
311
- */
312
- function insert_js() {
313
- ?>
314
- <script type='text/javascript'>
315
- <?php include(WP_PLUGIN_DIR.'/events-manager/includes/js/em_calendar_ajax.js'); ?>
316
- </script>
317
- <?php
318
- }
319
-
320
-
321
- function days_in_month($month, $year) {
322
- return date('t', mktime(0,0,0,$month,1,$year));
323
- }
324
-
325
- function translate_and_trim($string, $length = 1) {
326
- return substr(__($string), 0, $length);
327
- }
328
-
329
- /**
330
- * Helper function to create a link querystring from array which contains arguments with only values that aren't defuaults.
331
- */
332
- function get_link_args($args = array(), $html_entities=true){
333
- unset($args['month']); unset($args['year']);
334
- $default_args = self::get_default_search(array());
335
- foreach($default_args as $arg_key => $arg_value){
336
- if( !isset($args[$arg_key]) || $args[$arg_key] == $arg_value ){
337
- unset($args[$arg_key]);
338
- }
339
- }
340
- $qs_array = array();
341
- foreach($args as $key => $value){
342
- if(is_array($value)){
343
- $value = implode(',',$value);
344
- }
345
- $qs_array[] = "$key=".urlencode($value);
346
- }
347
- return ($html_entities) ? implode('&amp;', $qs_array) : implode('&', $qs_array);
348
- }
349
-
350
-
351
- function get_default_search($array=array()){
352
- //These defaults aren't for db queries, but flags for what to display in calendar output
353
- $defaults = array(
354
- 'full' => 0, //Will display a full calendar with event names
355
- 'long_events' => 0, //Events that last longer than a day
356
- 'scope' => 'all',
357
- 'owner' => false
358
- );
359
- $atts = parent::get_default_search($defaults, $array);
360
- $atts['full'] = ($atts['full']==true) ? 1:0;
361
- $atts['long_events'] = ($atts['long_events']==true) ? 1:0;
362
- return apply_filters('em_calendar_get_default_search', $atts, $array, $defaults);
363
- }
364
- }
365
- add_action('init', array('EM_Calendar', 'init'));
366
- ?>
1
+ <?php
2
+ class EM_Calendar extends EM_Object {
3
+
4
+ function init(){
5
+ //nothing to init anymore
6
+ }
7
+
8
+ function get( $args ){
9
+
10
+ global $wpdb;
11
+
12
+ $calendar_array = array();
13
+ $calendar_array['cells'] = array();
14
+
15
+ $args = self::get_default_search($args);
16
+ $full = $args['full']; //For ZDE, don't delete pls
17
+ $month = $args['month'];
18
+ $year = $args['year'];
19
+ $long_events = $args['long_events'];
20
+
21
+ $week_starts_on_sunday = get_option('dbem_week_starts_sunday');
22
+ $start_of_week = get_option('start_of_week');
23
+
24
+ if( !(is_numeric($month) && $month <= 12 && $month > 0) ) {
25
+ $month = date('m');
26
+ }
27
+ if( !( is_numeric($year) ) ){
28
+ $year = date('Y');
29
+ }
30
+
31
+ // Get the first day of the month
32
+ $month_start = mktime(0,0,0,$month, 1, $year);
33
+ $calendar_array['month_start'] = $month_start;
34
+
35
+ // Get friendly month name
36
+ $month_name = date('M',$month_start);
37
+ // Figure out which day of the week
38
+ // the month starts on.
39
+ $month_start_day = date('D', $month_start);
40
+
41
+ switch($month_start_day){
42
+ case "Sun": $offset = 0; break;
43
+ case "Mon": $offset = 1; break;
44
+ case "Tue": $offset = 2; break;
45
+ case "Wed": $offset = 3; break;
46
+ case "Thu": $offset = 4; break;
47
+ case "Fri": $offset = 5; break;
48
+ case "Sat": $offset = 6; break;
49
+ }
50
+ //We need to go back to the WP defined day when the week started, in case the event day is near the end
51
+ $offset -= $start_of_week;
52
+ if($offset<0)
53
+ $offset += 7;
54
+
55
+ // determine how many days are in the last month.
56
+ $month_last = $month-1;
57
+ $month_next = $month+1;
58
+ $calendar_array['month_next'] = $month_next;
59
+ $year_last = $year;
60
+ $year_next = $year;
61
+ $calendar_array['year_next'] = $year_next;
62
+
63
+ if($month == 1) {
64
+ $month_last = 12;
65
+ $year_last = $year -1;
66
+ }elseif($month == 12){
67
+ $month_next = 1;
68
+ $year_next = $year + 1;
69
+ }
70
+ $calendar_array['month_last'] = $month_last;
71
+ $calendar_array['year_last'] = $year_last;
72
+
73
+ $num_days_last = self::days_in_month($month_last, $year_last);
74
+
75
+ // determine how many days are in the current month.
76
+ $num_days_current = self::days_in_month($month, $year);
77
+ // Build an array for the current days
78
+ // in the month
79
+ for($i = 1; $i <= $num_days_current; $i++){
80
+ $num_days_array[] = mktime(0,0,0,$month, $i, $year);
81
+ }
82
+ // Build an array for the number of days
83
+ // in last month
84
+ for($i = 1; $i <= $num_days_last; $i++){
85
+ $num_days_last_array[] = mktime(0,0,0,$month_last, $i, $year_last);
86
+ }
87
+ // If the $offset from the starting day of the
88
+ // week happens to be Sunday, $offset would be 0,
89
+ // so don't need an offset correction.
90
+
91
+ if($offset > 0){
92
+ $offset_correction = array_slice($num_days_last_array, -$offset, $offset);
93
+ $new_count = array_merge($offset_correction, $num_days_array);
94
+ $offset_count = count($offset_correction);
95
+ } else { // The else statement is to prevent building the $offset array.
96
+ $offset_count = 0;
97
+ $new_count = $num_days_array;
98
+ }
99
+ // count how many days we have with the two
100
+ // previous arrays merged together
101
+ $current_num = count($new_count);
102
+
103
+ // Since we will have 5 HTML table rows (TR)
104
+ // with 7 table data entries (TD)
105
+ // we need to fill in 35 TDs
106
+ // so, we will have to figure out
107
+ // how many days to appened to the end
108
+ // of the final array to make it 35 days.
109
+ if($current_num > 35){
110
+ $num_weeks = 6;
111
+ $outset = (42 - $current_num);
112
+ } elseif($current_num < 35){
113
+ $num_weeks = 5;
114
+ $outset = (35 - $current_num);
115
+ }
116
+ if($current_num == 35){
117
+ $num_weeks = 5;
118
+ $outset = 0;
119
+ }
120
+ // Outset Correction
121
+ for($i = 1; $i <= $outset; $i++){
122
+ $new_count[] = mktime(0,0,0,$month_next, $i, $year_next);
123
+ }
124
+ // Now let's "chunk" the $all_days array
125
+ // into weeks. Each week has 7 days
126
+ // so we will array_chunk it into 7 days.
127
+ $weeks = array_chunk($new_count, 7);
128
+
129
+ //Get an array of arguments that don't include default valued args
130
+ $link_args = self::get_link_args($args);
131
+
132
+ $previous_url = "?ajaxCalendar=1&amp;month={$month_last}&amp;year={$year_last}&amp;{$link_args}";
133
+ $next_url = "?ajaxCalendar=1&amp;month={$month_next}&amp;year={$year_next}&amp;{$link_args}";
134
+
135
+ $weekdays = array('Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday');
136
+ $n = 0 ;
137
+ while( $n < $start_of_week ) {
138
+ $last_day = array_shift($weekdays);
139
+ $weekdays[]= $last_day;
140
+ $n++;
141
+ }
142
+
143
+ $days_initials_array = array();
144
+ foreach($weekdays as $weekday) {
145
+ $days_initials_array[] = self::translate_and_trim($weekday);
146
+ }
147
+
148
+ $calendar_array['links'] = array( 'previous_url'=>$previous_url, 'next_url'=>$next_url);
149
+ $calendar_array['row_headers'] = $days_initials_array;
150
+
151
+ // Now we break each key of the array
152
+ // into a week and create a new table row for each
153
+ // week with the days of that week in the table data
154
+
155
+ $i = 0;
156
+ $current_date = date('Y-m-d', current_time('timestamp'));
157
+ $week_count = 0;
158
+ foreach ( $weeks as $week ) {
159
+ foreach ( $week as $d ) {
160
+ $date = date('Y-m-d', $d);
161
+ $calendar_array['cells'][$date] = array('date'=>$d); //set it up so we have the exact array of dates to be filled
162
+ if ($i < $offset_count) { //if it is PREVIOUS month
163
+ $calendar_array['cells'][$date]['type'] = 'pre';
164
+ }
165
+ if (($i >= $offset_count) && ($i < ($num_weeks * 7) - $outset)) { // if it is THIS month
166
+ if ( $current_date == $date ){
167
+ $calendar_array['cells'][$date]['type'] = 'today';
168
+ }
169
+ } elseif (($outset > 0)) { //if it is NEXT month
170
+ if (($i >= ($num_weeks * 7) - $outset)) {
171
+ $calendar_array['cells'][$date]['type'] = 'post';
172
+ }
173
+ }
174
+ $i ++;
175
+ }
176
+ $week_count++;
177
+ }
178
+
179
+ // query the database for events in this time span
180
+ if ($month == 1) {
181
+ $month_pre=12;
182
+ $month_post=2;
183
+ $year_pre=$year-1;
184
+ $year_post=$year;
185
+ } elseif($month == 12) {
186
+ $month_pre=11;
187
+ $month_post=1;
188
+ $year_pre=$year;
189
+ $year_post=$year+1;
190
+ } else {
191
+ $month_pre=$month-1;
192
+ $month_post=$month+1;
193
+ $year_pre=$year;
194
+ $year_post=$year;
195
+ }
196
+ $args['year'] = array($year_pre, $year_post);
197
+ $args['month'] = array($month_pre, $month_post);
198
+ $events = EM_Events::get($args);
199
+
200
+ $event_format = get_option('dbem_full_calendar_event_format');
201
+ $event_title_format = get_option('dbem_small_calendar_event_title_format');
202
+ $event_title_separator_format = get_option('dbem_small_calendar_event_title_separator');
203
+
204
+ $eventful_days= array();
205
+ if($events){
206
+ //Go through the events and slot them into the right d-m index
207
+ foreach($events as $event) {
208
+ $event = apply_filters('em_calendar_output_loop_start', $event);
209
+ if( $long_events ){
210
+ //If $long_events is set then show a date as eventful if there is an multi-day event which runs during that day
211
+ $event_start_date = strtotime($event->start_date);
212
+ $event_end_date = mktime(0,0,0,$month_post,date('t', $event_start_date),$year_post );
213
+ if( $event_end_date == '' ) $event_end_date = $event_start_date;
214
+ while( $event_start_date <= $event->end ){
215
+ //Ensure date is within event dates, if so add to eventful days array
216
+ $event_eventful_date = date('Y-m-d', $event_start_date);
217
+ if( array_key_exists($event_eventful_date, $eventful_days) && is_array($eventful_days[$event_eventful_date]) ){
218
+ $eventful_days[$event_eventful_date][] = $event;
219
+ } else {
220
+ $eventful_days[$event_eventful_date] = array($event);
221
+ }
222
+ $event_start_date += (86400); //add a day
223
+ }
224
+ }else{
225
+ //Only show events on the day that they start
226
+ if( isset($eventful_days[$event->event_start_date]) && is_array($eventful_days[$event->event_start_date]) ){
227
+ $eventful_days[$event->event_start_date][] = $event;
228
+ } else {
229
+ $eventful_days[$event->event_start_date] = array($event);
230
+ }
231
+ }
232
+ }
233
+ }
234
+ foreach($eventful_days as $day_key => $events) {
235
+ if( array_key_exists($day_key, $calendar_array['cells']) ){
236
+ //Get link title for this date
237
+ $events_titles = array();
238
+ foreach($events as $event) {
239
+ $events_titles[] = $event->output($event_title_format);
240
+ }
241
+ $calendar_array['cells'][$day_key]['link_title'] = implode( $event_title_separator_format, $events_titles);
242
+
243
+ //Get the link to this calendar day
244
+ global $wp_rewrite;
245
+ if( get_option("dbem_events_page") > 0 ){
246
+ $event_page_link = trailingslashit(get_permalink(get_option("dbem_events_page"))); //PAGE URI OF EM
247
+ }else{
248
+ if( $wp_rewrite->using_permalinks() ){
249
+ $event_page_link = trailingslashit(home_url()).EM_POST_TYPE_EVENT_SLUG.'/'; //don't use EM_URI here, since ajax calls this before EM_URI is defined.
250
+ }else{
251
+ $event_page_link = trailingslashit(home_url()).'?post_type='.EM_POST_TYPE_EVENT; //don't use EM_URI here, since ajax calls this before EM_URI is defined.
252
+ }
253
+ }
254
+ if( $wp_rewrite->using_permalinks() && !defined('EM_DISABLE_PERMALINKS') ){
255
+ $calendar_array['cells'][$day_key]['link'] = $event_page_link.$day_key."/";
256
+ }else{
257
+ $joiner = (stristr($event_page_link, "?")) ? "&amp;" : "?";
258
+ $calendar_array['cells'][$day_key]['link'] = $event_page_link.$joiner."calendar_day=".$day_key;
259
+ }
260
+ //Add events to array
261
+ $calendar_array['cells'][$day_key]['events'] = $events;
262
+ }
263
+ }
264
+ return apply_filters('em_calendar_get',$calendar_array, $args);
265
+ }
266
+
267
+ function output($args = array()) {
268
+ $calendar_array = self::get($args);
269
+ $template = (!empty($args['full'])) ? 'templates/calendar-full.php':'templates/calendar-small.php';
270
+ ob_start();
271
+ em_locate_template($template, true, array('calendar'=>$calendar_array,'args'=>$args));
272
+ $calendar = '<div id="em-calendar-'.rand(100,200).'" class="em-calendar-wrapper">'.ob_get_clean().'</div>';
273
+ return apply_filters('em_calendar_output', $calendar, $args);
274
+ }
275
+
276
+
277
+ function days_in_month($month, $year) {
278
+ return date('t', mktime(0,0,0,$month,1,$year));
279
+ }
280
+
281
+ function translate_and_trim($string, $length = 1) {
282
+ if(function_exists('mb_substr')){ //fix for diacritic calendar names
283
+ return mb_substr(__($string), 0, $length);
284
+ }else{
285
+ return substr(__($string), 0, $length);
286
+ }
287
+ }
288
+
289
+ /**
290
+ * Helper function to create a link querystring from array which contains arguments with only values that aren't defuaults.
291
+ */
292
+ function get_link_args($args = array(), $html_entities=true){
293
+ unset($args['month']); unset($args['year']);
294
+ $default_args = self::get_default_search(array());
295
+ foreach($default_args as $arg_key => $arg_value){
296
+ if( !isset($args[$arg_key]) || $args[$arg_key] == $arg_value ){
297
+ unset($args[$arg_key]);
298
+ }
299
+ }
300
+ $qs_array = array();
301
+ foreach($args as $key => $value){
302
+ if(is_array($value)){
303
+ $value = implode(',',$value);
304
+ }
305
+ $qs_array[] = "$key=".urlencode($value);
306
+ }
307
+ return ($html_entities) ? implode('&amp;', $qs_array) : implode('&', $qs_array);
308
+ }
309
+
310
+
311
+ function get_default_search($array=array()){
312
+ //These defaults aren't for db queries, but flags for what to display in calendar output
313
+ $defaults = array(
314
+ 'full' => 0, //Will display a full calendar with event names
315
+ 'long_events' => 0, //Events that last longer than a day
316
+ 'scope' => 'future',
317
+ 'status' => 1, //approved events only
318
+ 'town' => false,
319
+ 'state' => false,
320
+ 'country' => false,
321
+ 'region' => false,
322
+ 'blog' => get_current_blog_id(),
323
+ );
324
+ if(is_multisite()){
325
+ global $bp;
326
+ if( !is_main_site() && !array_key_exists('blog',$array) ){
327
+ //not the main blog, force single blog search
328
+ $array['blog'] = get_current_blog_id();
329
+ }elseif( empty($array['blog']) && get_site_option('dbem_ms_global_events') ) {
330
+ $array['blog'] = false;
331
+ }
332
+ }
333
+ $atts = parent::get_default_search($defaults, $array);
334
+ $atts['full'] = ($atts['full']==true) ? 1:0;
335
+ $atts['long_events'] = ($atts['long_events']==true) ? 1:0;
336
+ return apply_filters('em_calendar_get_default_search', $atts, $array, $defaults);
337
+ }
338
+ }
339
+ add_action('init', array('EM_Calendar', 'init'));
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
classes/em-categories-taxonomy.php ADDED
@@ -0,0 +1,84 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Categories_Taxonomy{
3
+ function init(){
4
+ add_action( EM_TAXONOMY_CATEGORY.'_edit_form_fields', array('EM_Categories_Taxonomy','form'), 10, 1);
5
+ add_action( EM_TAXONOMY_CATEGORY.'_add_form_fields', array('EM_Categories_Taxonomy','form'), 10, 1);
6
+ add_action( 'edited_'.EM_TAXONOMY_CATEGORY, array('EM_Categories_Taxonomy','save'), 10, 2);
7
+ add_action( 'create_'.EM_TAXONOMY_CATEGORY, array('EM_Categories_Taxonomy','save'), 10, 2);
8
+ add_action( 'delete_'.EM_TAXONOMY_CATEGORY, array('EM_Categories_Taxonomy','delete'), 10, 2);
9
+ self::admin_init();
10
+ }
11
+
12
+ function admin_init(){
13
+ global $pagenow;
14
+ if($pagenow == 'edit-tags.php' && !empty($_GET['taxonomy']) && $_GET['taxonomy'] == EM_TAXONOMY_CATEGORY){
15
+ wp_enqueue_style( 'farbtastic' );
16
+ wp_enqueue_style( 'thickbox' );
17
+
18
+ wp_enqueue_script( 'em-categories-admin', plugins_url().'/events-manager/includes/js/categories-admin.js', array( 'jquery','media-upload','thickbox','farbtastic' ) );
19
+ }
20
+ }
21
+
22
+ function form($tag){
23
+ $category_color = '#FFFFFF';
24
+ $category_image = '';
25
+ if( $tag != EM_TAXONOMY_CATEGORY ){ //not an add new tag form
26
+ $EM_Category = new EM_Category($tag);
27
+ $category_color = $EM_Category->get_color();
28
+ $category_image = $EM_Category->get_image_url();
29
+ }
30
+ ?>
31
+ <tr class="form-field">
32
+ <th scope="row" valign="top"><label for="category-bgcolor">Calendar Color</label></th>
33
+ <td>
34
+ <input type="text" name="category_bgcolor" id="category-bgcolor" class="colorwell" value="<?php echo esc_attr($category_color); ?>" style="width:100px;"/><br />
35
+ <p class="description"><?php echo sprintf(__('Choose a color for your category. You can access this using the %s placeholder.','dbem'),'<code>#_CATEGORYCOLOR</code>'); ?></p>
36
+ <div id="picker" style="position:absolute; display:none; background:#DEDEDE"></div>
37
+ </td>
38
+ </tr>
39
+ <tr class="form-field">
40
+ <th scope="row" valign="top"><label for="product_package_unit_price">Image</label></th>
41
+ <td>
42
+ <?php if( !empty($category_image) ): ?>
43
+ <p><img src="<?php echo $category_image; ?>" /></p>
44
+ <?php endif; ?>
45
+ <input type="text" name="category_image" id="category-image" value="<?php echo esc_attr($category_image); ?>" style="width:300px;" />
46
+ <input id="upload_image_button" type="button" value="<?php _e('Choose/Upload Image','dbem'); ?>" class="button-secondary" style="width:auto;" /><br />
47
+ <p class="description"><?php echo sprintf(__('Choose an image for your category, which can be displayed using the %s placeholder.','dbem'),'<code>#_CATEGORYIMAGE</code>'); ?></p>
48
+ </td>
49
+ </tr>
50
+ <?php
51
+ }
52
+
53
+ function save($term_id, $tt_id){
54
+ global $wpdb;
55
+ if (!$term_id) return;
56
+ if( !empty($_POST['category_bgcolor']) && preg_match('/^#[a-zA-Z0-9]{6}$/', $_POST['category_bgcolor']) ){
57
+ //get results and save/update
58
+ $prev_settings = $wpdb->get_results('SELECT meta_value FROM '.EM_META_TABLE." WHERE object_id='{$term_id}' AND meta_key='category-bgcolor'");
59
+ if( count($prev_settings) > 0 ){
60
+ $wpdb->update(EM_META_TABLE, array('object_id'=>$term_id,'meta_value'=>$_POST['category_bgcolor']), array('object_id'=>$term_id,'meta_key'=>'category-bgcolor'));
61
+ }else{
62
+ $wpdb->insert(EM_META_TABLE, array('object_id'=>$term_id,'meta_key'=>'category-bgcolor','meta_value'=>$_POST['category_bgcolor']));
63
+ }
64
+ }
65
+ if( !empty($_POST['category_image']) ){
66
+ //get results and save/update
67
+ $prev_settings = $wpdb->get_results('SELECT meta_value FROM '.EM_META_TABLE." WHERE object_id='{$term_id}' AND meta_key='category-image'");
68
+ if( count($prev_settings) > 0 ){
69
+ $wpdb->update(EM_META_TABLE, array('object_id'=>$term_id,'meta_value'=>$_POST['category_image']), array('object_id'=>$term_id,'meta_key'=>'category-image'));
70
+ }else{
71
+ $wpdb->insert(EM_META_TABLE, array('object_id'=>$term_id,'meta_key'=>'category-image','meta_value'=>$_POST['category_image']));
72
+ }
73
+ }
74
+ }
75
+
76
+ function delete( $term_id ){
77
+ global $wpdb;
78
+ //delete category image and color
79
+ $wpdb->query('DELETE FROM '.EM_META_TABLE." WHERE object_id='$term_id' AND (meta_key='category-image' OR meta_key='category-bgcolor')");
80
+ //delete all events category relations
81
+ $wpdb->query('DELETE FROM '.EM_META_TABLE." WHERE meta_value='{$term_id}' AND meta_key='event-category'");
82
+ }
83
+ }
84
+ add_action('admin_init',array('EM_Categories_Taxonomy','init'));
classes/em-categories.php CHANGED
@@ -1,157 +1,320 @@
1
- <?php
2
- class EM_Categories extends EM_Object {
3
-
4
- function get( $args = array() ) {
5
- global $wpdb;
6
- $categories_table = $wpdb->prefix.EM_CATEGORIES_TABLE;
7
- $events_table = $wpdb->prefix.EM_EVENTS_TABLE;
8
-
9
- //Quick version, we can accept an array of IDs, which is easy to retrieve
10
- if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
11
- //We can just get all the events here and return them
12
- $sql = "SELECT * FROM $categories_table WHERE category_id=".implode(" OR category_id=", $args);
13
- $results = $wpdb->get_results(apply_filters('em_categories_get_sql',$sql),ARRAY_A);
14
- $categories = array();
15
- foreach($results as $result){
16
- $categories[$result['category_id']] = new EM_Category($result);
17
- }
18
- return $categories; //We return all the categories matched as an EM_Event array.
19
- }
20
-
21
- //We assume it's either an empty array or array of search arguments to merge with defaults
22
- $args = self::get_default_search($args);
23
- $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
24
- $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
25
-
26
- //Get the default conditions
27
- $conditions = self::build_sql_conditions($args);
28
- //Put it all together
29
- $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
30
-
31
- //Get ordering instructions
32
- $EM_Category = new EM_Category();
33
- $accepted_fields = $EM_Category->get_fields(true);
34
- $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_categories_default_order'));
35
- //Now, build orderby sql
36
- $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
37
-
38
- //Create the SQL statement and execute
39
- $sql = "
40
- SELECT * FROM $categories_table
41
- LEFT JOIN $events_table ON {$events_table}.event_category_id={$categories_table}.category_id
42
- $where
43
- GROUP BY category_id
44
- $orderby_sql
45
- $limit $offset
46
- ";
47
- $results = $wpdb->get_results( apply_filters('em_categories_get_sql',$sql, $args), ARRAY_A);
48
- //If we want results directly in an array, why not have a shortcut here?
49
- if( $args['array'] == true ){
50
- return $results;
51
- }
52
-
53
- //Make returned results EM_Event objects
54
- $results = (is_array($results)) ? $results:array();
55
- $categories = array();
56
- foreach ( $results as $category_array ){
57
- $categories[$category_array['category_id']] = new EM_Category($category_array);
58
- }
59
-
60
- return apply_filters('em_categories_get', $categories);
61
- }
62
-
63
- /**
64
- * Will delete given an array of category_ids or EM_Event objects
65
- * @param unknown_type $id_array
66
- */
67
- function delete( $array ){
68
- global $wpdb;
69
- //Detect array type and generate SQL for event IDs
70
- $category_ids = array();
71
- if( @get_class(current($array)) == 'EM_Category' ){
72
- foreach($array as $EM_Category){
73
- $category_ids[] = $EM_Category->id;
74
- }
75
- }else{
76
- $category_ids = $array;
77
- }
78
- if(self::array_is_numeric($category_ids)){
79
- apply_filters('em_categories_delete', $category_ids);
80
- $condition = implode(" OR category_id=", $category_ids);
81
- //Delete all the bookings
82
- $result_bookings = $wpdb->query("DELETE FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." WHERE category_id=$condition;");
83
- //Now delete the categories
84
- $result = $wpdb->query ( "DELETE FROM ". $wpdb->prefix . EM_CATEGORIES_TABLE ." WHERE category_id=$condition;" );
85
- do_action('em_categories_delete', $category_ids);
86
- }
87
- //TODO add error detection on categories delete fails
88
- return apply_filters('em_categories_delete', true, $category_ids);
89
- }
90
-
91
- /* Overrides EM_Object method to apply a filter to result
92
- * @see wp-content/plugins/categories-manager/classes/EM_Object#build_sql_conditions()
93
- */
94
- function build_sql_conditions( $args = array() ){
95
- global $wpdb;
96
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
97
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
98
-
99
- //FIXME EM_Categories doesn't build sql conditions in EM_Object
100
- $conditions = array();
101
- //eventful locations
102
- if( true == $args['eventful'] ){
103
- $conditions['eventful'] = "{$events_table}.event_id IS NOT NULL";
104
- }elseif( true == $args['eventless'] ){
105
- $conditions['eventless'] = "{$events_table}.event_id IS NULL";
106
- }
107
- //owner lookup
108
- if( is_numeric($args['owner']) ){
109
- $conditions['owner'] = "category_owner=".get_current_user_id();
110
- }elseif( preg_match('/^([0-9],?)+$/', $args['owner']) ){
111
- $conditions['owner'] = "category_owner IN (".explode(',', $args['owner']).")";
112
- }
113
- return apply_filters( 'em_categories_build_sql_conditions', $conditions, $args );
114
- }
115
-
116
- /* Overrides EM_Object method to apply a filter to result
117
- * @see wp-content/plugins/categories-manager/classes/EM_Object#build_sql_orderby()
118
- */
119
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
120
- return apply_filters( 'em_categories_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_categories_default_order')), $args, $accepted_fields, $default_order );
121
- }
122
-
123
- /*
124
- * Adds custom categories search defaults
125
- * @param array $array
126
- * @return array
127
- * @uses EM_Object#get_default_search()
128
- */
129
- function get_default_search( $array = array() ){
130
- $defaults = array(
131
- 'scope'=>false,
132
- 'eventful' => false, //cats that have an event (scope will also play a part here
133
- 'eventless' => false, //cats WITHOUT events, eventful takes precedence
134
- );
135
- if( is_admin() ){
136
- //by default, we only get categories the owner can manage
137
- switch( get_option('dbem_permissions_categories') ){
138
- case 0:
139
- $defaults['owner'] = get_current_user_id();
140
- break;
141
- case 1:
142
- $wp_user_search = new WP_User_Search(null, null, 'administrator');
143
- $users = $wp_user_search->get_results();
144
- $users[] = get_current_user_id();
145
- $users[] = 0;
146
- $defaults['owner'] = implode(',', $users);
147
- break;
148
- case 2:
149
- $defaults['owner'] = false;
150
- break;
151
- }
152
- $defaults['owner'] = ( em_verify_admin() ) ? false:$defaults['owner'];
153
- }
154
- return apply_filters('em_categories_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
155
- }
156
-
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
157
  }
1
+ <?php
2
+ class EM_Categories extends EM_Object implements Iterator{
3
+
4
+ /**
5
+ * Array of EM_Category objects for a specific event
6
+ * @var array
7
+ */
8
+ var $categories = array();
9
+ /**
10
+ * Event ID of this set of categories
11
+ * @var int
12
+ */
13
+ var $event_id;
14
+ /**
15
+ * Post ID of this set of categories
16
+ * @var int
17
+ */
18
+ var $post_id;
19
+
20
+ /**
21
+ * Creates an EM_Categories instance, currently accepts an EM_Event object (gets all Categories for that event) or array of any EM_Category objects, which can be manipulated in bulk with helper functions.
22
+ * @param mixed $data
23
+ * @return null
24
+ */
25
+ function EM_Categories( $data = false ){
26
+ global $wpdb;
27
+ $this->ms_global_switch();
28
+ if( is_object($data) && get_class($data) == "EM_Event" && !empty($data->post_id) ){ //Creates a blank categories object if needed
29
+ $this->event_id = $data->event_id;
30
+ $this->post_id = $data->post_id;
31
+ if( EM_MS_GLOBAL && !is_main_site() ){
32
+ $cat_ids = $wpdb->get_col('SELECT meta_value FROM '.EM_META_TABLE." WHERE object_id='{$this->event_id}' AND meta_key='event-category'");
33
+ foreach($cat_ids as $cat_id){
34
+ $this->categories[$cat_id] = new EM_Category($cat_id);
35
+ }
36
+ }else{
37
+ $results = get_the_terms( $data->post_id, EM_TAXONOMY_CATEGORY );
38
+ if( is_array($results) ){
39
+ foreach($results as $result){
40
+ $this->categories[$result->term_id] = new EM_Category($result);
41
+ }
42
+ }
43
+ }
44
+ }elseif( is_array($data) && $this->array_is_numeric($data) ){
45
+ foreach($data as $category_id){
46
+ $this->categories[$category_id] = new EM_Category($category_id);
47
+ }
48
+ }elseif( is_array($data) ){
49
+ foreach( $data as $EM_Category ){
50
+ if( get_class($EM_Category) == 'EM_Category'){
51
+ $this->categories[] = $EM_Category;
52
+ }
53
+ }
54
+ }
55
+ $this->ms_global_switch_back();
56
+ do_action('em_categories', $this);
57
+ }
58
+
59
+ function get_post(){
60
+ $this->ms_global_switch();
61
+ if(!empty($_POST['event_categories']) && $this->array_is_numeric($_POST['event_categories'])){
62
+ foreach( $_POST['event_categories'] as $term ){
63
+ $this->categories[$term] = new EM_Category($term);
64
+ }
65
+ }
66
+ $this->ms_global_switch_back();
67
+ do_action('em_categories_get_post', $this);
68
+ }
69
+
70
+ function save(){
71
+ $term_slugs = array();
72
+ foreach($this->categories as $EM_Category){
73
+ /* @var $EM_Category EM_Category */
74
+ if( !empty($EM_Category->slug) ) $term_slugs[] = $EM_Category->slug; //save of category will soft-fail if slug is empty
75
+ }
76
+ if( count($term_slugs) == 0 && get_option('dbem_default_category') ){
77
+ $default_term = get_term_by('id',get_option('dbem_default_category'), EM_TAXONOMY_CATEGORY);
78
+ if($default_term) $term_slugs[] = $default_term->slug;
79
+ }
80
+ if( count($term_slugs) > 0 ){
81
+ if( is_multisite() ){
82
+ //In MS Global mode, we also save category meta information for global lookups
83
+ if( EM_MS_GLOBAL && !empty($this->event_id) ){
84
+ //delete categories
85
+ $this->save_index();
86
+ }
87
+ if( !EM_MS_GLOBAL || is_main_site() ){
88
+ wp_set_object_terms($this->post_id, $term_slugs, EM_TAXONOMY_CATEGORY);
89
+ }
90
+ }else{
91
+ wp_set_object_terms($this->post_id, $term_slugs, EM_TAXONOMY_CATEGORY);
92
+ }
93
+ }
94
+ do_action('em_categories_save', $this);
95
+ }
96
+
97
+ function save_index(){
98
+ global $wpdb;
99
+ $wpdb->query('DELETE FROM '.EM_META_TABLE." WHERE object_id='{$this->event_id}' AND meta_key='event-category'");
100
+ foreach($this->categories as $EM_Category){
101
+ $wpdb->insert(EM_META_TABLE, array('meta_value'=>$EM_Category->term_id,'object_id'=>$this->event_id,'meta_key'=>'event-category'));
102
+ }
103
+ }
104
+
105
+ function get( $args = array() ) {
106
+ //Quick version, we can accept an array of IDs, which is easy to retrieve
107
+ self::ms_global_switch();
108
+ if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
109
+ $results = get_terms( EM_TAXONOMY_CATEGORY );
110
+ $categories = array();
111
+ foreach($results as $result){
112
+ if( in_array($result->term_id, $args) ){
113
+ $categories[$result->term_id] = new EM_Category($result);
114
+ }
115
+ }
116
+ }else{
117
+ //We assume it's either an empty array or array of search arguments to merge with defaults
118
+ $term_args = self::get_default_search($args);
119
+ $results = get_terms( EM_TAXONOMY_CATEGORY, $term_args);
120
+
121
+ //If we want results directly in an array, why not have a shortcut here? We don't use this in code, so if you're using it and filter the em_categories_get hook, you may want to do this one too.
122
+ if( !empty($args['array']) ){
123
+ return apply_filters('em_categories_get_array', $results, $args);
124
+ }
125
+
126
+ //Make returned results EM_Category objects
127
+ $results = (is_array($results)) ? $results:array();
128
+ $categories = array();
129
+ foreach ( $results as $category ){
130
+ $categories[$category->term_id] = new EM_Category($category);
131
+ }
132
+ }
133
+ self::ms_global_switch_back();
134
+ return apply_filters('em_categories_get', $categories, $args);
135
+ }
136
+
137
+ function output( $args ){
138
+ global $EM_Category;
139
+ $EM_Category_old = $EM_Category; //When looping, we can replace EM_Category global with the current event in the loop
140
+ //Can be either an array for the get search or an array of EM_Category objects
141
+ if( is_object(current($args)) && get_class((current($args))) == 'EM_Category' ){
142
+ $func_args = func_get_args();
143
+ $categories = $func_args[0];
144
+ $args = (!empty($func_args[1])) ? $func_args[1] : array();
145
+ $args = apply_filters('em_categories_output_args', self::get_default_search($args), $categories);
146
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
147
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
148
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
149
+ }else{
150
+ $args = apply_filters('em_categories_output_args', self::get_default_search($args) );
151
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
152
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
153
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
154
+ $args['limit'] = false;
155
+ $args['offset'] = false;
156
+ $args['page'] = false;
157
+ $categories = self::get( $args );
158
+ }
159
+ //What format shall we output this to, or use default
160
+ $format = ( $args['format'] == '' ) ? get_option( 'dbem_categories_list_item_format' ) : $args['format'] ;
161
+
162
+ $output = "";
163
+ $categories_count = count($categories);
164
+ $categories = apply_filters('em_categories_output_categories', $categories);
165
+ if ( count($categories) > 0 ) {
166
+ $category_count = 0;
167
+ $categories_shown = 0;
168
+ foreach ( $categories as $EM_Category ) {
169
+ if( ($categories_shown < $limit || empty($limit)) && ($category_count >= $offset || $offset === 0) ){
170
+ $output .= $EM_Category->output($format);
171
+ $categories_shown++;
172
+ }
173
+ $category_count++;
174
+ }
175
+ //Add headers and footers to output
176
+ if( $format == get_option ( 'dbem_categories_list_item_format' ) ){
177
+ $single_event_format_header = get_option ( 'dbem_categories_list_item_format_header' );
178
+ $single_event_format_header = ( $single_event_format_header != '' ) ? $single_event_format_header : "<ul class='em-categories-list'>";
179
+ $single_event_format_footer = get_option ( 'dbem_categories_list_item_format_footer' );
180
+ $single_event_format_footer = ( $single_event_format_footer != '' ) ? $single_event_format_footer : "</ul>";
181
+ $output = $single_event_format_header . $output . $single_event_format_footer;
182
+ }
183
+ //Pagination (if needed/requested)
184
+ if( !empty($args['pagination']) && !empty($limit) && $categories_count >= $limit ){
185
+ //Show the pagination links (unless there's less than 10 events, or the custom limit)
186
+ $page_link_template = preg_replace('/(&|\?)page=\d+/i','',$_SERVER['REQUEST_URI']);
187
+ $page_link_template = em_add_get_params($page_link_template, array('page'=>'%PAGE%'), false); //don't html encode, so em_paginate does its thing
188
+ $output .= apply_filters('em_events_output_pagination', em_paginate( $page_link_template, $categories_count, $limit, $page), $page_link_template, $categories_count, $limit, $page);
189
+ }
190
+ } else {
191
+ $output = get_option ( 'dbem_no_categories_message' );
192
+ }
193
+ //FIXME check if reference is ok when restoring object, due to changes in php5 v 4
194
+ $EM_Category_old= $EM_Category;
195
+ return apply_filters('em_categories_output', $output, $categories, $args);
196
+ }
197
+
198
+ function has( $search ){
199
+ if( is_numeric($search) ){
200
+ foreach($this->categories as $EM_Category){
201
+ if($EM_Category->term_id == $search) return apply_filters('em_categories_has', true, $search, $this);
202
+ }
203
+ }else{
204
+ foreach($this->categories as $EM_Category){
205
+ if($EM_Category->slug == $search) return apply_filters('em_categories_has', true, $search, $this);
206
+ }
207
+ }
208
+ return apply_filters('em_categories_has', false, $search, $this);
209
+ }
210
+
211
+ function get_first(){
212
+ foreach($this->categories as $EM_Category){
213
+ return $EM_Category;
214
+ }
215
+ return false;
216
+ }
217
+
218
+ function get_ids(){
219
+ $ids = array();
220
+ foreach($this->categories as $EM_Category){
221
+ if( !empty($EM_Category->term_id) ){
222
+ $ids[] = $EM_Category->term_id;
223
+ }
224
+ }
225
+ return $ids;
226
+ }
227
+
228
+ /**
229
+ * Gets the event for this object, or a blank event if none exists
230
+ * @return EM_Event
231
+ */
232
+ function get_event(){
233
+ if( is_numeric($this->event_id) ){
234
+ return em_get_event($this->event_id);
235
+ }else{
236
+ return new EM_Event();
237
+ }
238
+ }
239
+
240
+ /* Overrides EM_Object method to apply a filter to result. Categories won't accept many arguments as you tend to search with events for much else.
241
+ * @see wp-content/plugins/categories-manager/classes/EM_Object#build_sql_conditions()
242
+ */
243
+ function build_sql_conditions( $args = array() ){
244
+ global $wpdb;
245
+ $events_table = EM_EVENTS_TABLE;
246
+ $locations_table = EM_LOCATIONS_TABLE;
247
+
248
+ $temp_conditions = parent::build_sql_conditions($args);
249
+ $conditions = array();
250
+ if( !empty($temp_conditions['category']) ){
251
+ $conditions['category'] = $temp_conditions['category'];
252
+ }
253
+ return apply_filters( 'em_categories_build_sql_conditions', $conditions, $args );
254
+ }
255
+
256
+ /* Overrides EM_Object method to apply a filter to result
257
+ * @see wp-content/plugins/categories-manager/classes/EM_Object#build_sql_orderby()
258
+ */
259
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
260
+ return apply_filters( 'em_categories_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_categories_default_order')), $args, $accepted_fields, $default_order );
261
+ }
262
+
263
+ /*
264
+ * Adds custom categories search defaults
265
+ * @param array $array
266
+ * @return array
267
+ * @uses EM_Object#get_default_search()
268
+ */
269
+ function get_default_search( $array = array() ){
270
+ $defaults = array(
271
+ //added from get_terms, so they don't get filtered out
272
+ 'orderby' => 'name', 'order' => 'ASC',
273
+ 'hide_empty' => false, 'exclude' => array(), 'exclude_tree' => array(), 'include' => array(),
274
+ 'number' => '', 'fields' => 'all', 'slug' => '', 'parent' => '',
275
+ 'hierarchical' => true, 'child_of' => 0, 'get' => '', 'name__like' => '',
276
+ 'pad_counts' => false, 'offset' => '', 'search' => '', 'cache_domain' => 'core'
277
+ );
278
+ return apply_filters('em_categories_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
279
+ }
280
+
281
+ /**
282
+ * will return the default search parameter to use according to permission settings
283
+ * @return string
284
+ */
285
+ function get_default_search_owner(){
286
+ //by default, we only get categories the owner can manage
287
+ $defaults = array('owner'=>false);
288
+ //by default, we only get categories the owner can manage
289
+ if( !current_user_can('edit_categories') ){
290
+ $defaults['owner'] = get_current_user_id();
291
+ break;
292
+ }else{
293
+ $defaults['owner'] = false;
294
+ break;
295
+ }
296
+ return $defaults['owner'];
297
+ }
298
+
299
+ //Iterator Implementation
300
+ public function rewind(){
301
+ reset($this->categories);
302
+ }
303
+ public function current(){
304
+ $var = current($this->categories);
305
+ return $var;
306
+ }
307
+ public function key(){
308
+ $var = key($this->categories);
309
+ return $var;
310
+ }
311
+ public function next(){
312
+ $var = next($this->categories);
313
+ return $var;
314
+ }
315
+ public function valid(){
316
+ $key = key($this->categories);
317
+ $var = ($key !== NULL && $key !== FALSE);
318
+ return $var;
319
+ }
320
  }
classes/em-category-taxonomy.php ADDED
@@ -0,0 +1,67 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Category_Taxonomy{
3
+ function init(){
4
+ if( !is_admin() ){
5
+ add_filter('archive_template', array('EM_Category_Taxonomy','template'));
6
+ add_filter('category_template', array('EM_Category_Taxonomy','template'));
7
+ add_filter('parse_query', array('EM_Category_Taxonomy','parse_query'));
8
+ }
9
+ }
10
+ /**
11
+ * Overrides archive pages e.g. locations, events, event categories, event tags based on user settings
12
+ * @param string $template
13
+ * @return string
14
+ */
15
+ function template($template){
16
+ global $wp_query, $EM_Category;
17
+ if( is_archive() ){
18
+ if( !empty($wp_query->queried_object->taxonomy) && $wp_query->queried_object->taxonomy == EM_TAXONOMY_CATEGORY && get_option('dbem_cp_categories_formats', true)){
19
+ $EM_Category = em_get_category($wp_query->queried_object->term_id);
20
+ add_filter('the_content', array('EM_Category_Taxonomy','the_content'));
21
+ $wp_query->posts = array();
22
+ $wp_query->posts[0] = new stdClass();
23
+ $wp_query->posts[0]->post_title = $EM_Category->output(get_option('dbem_category_page_title_format'));
24
+ $post_array = array('ID', 'post_author', 'post_date','post_date_gmt','post_content','post_excerpt','post_status','comment_status','ping_status','post_password','post_name','to_ping','pinged','post_modified','post_modified_gmt','post_content_filtered','post_parent','guid','menu_order','post_type','post_mime_type','comment_count','filter');
25
+ foreach($post_array as $post_array_item){
26
+ $wp_query->posts[0]->$post_array_item = '';
27
+ }
28
+ $wp_query->post = $wp_query->posts[0];
29
+ $wp_query->post_count = 1;
30
+ $wp_query->found_posts = 1;
31
+ $wp_query->max_num_pages = 1;
32
+ //echo "<pre>"; print_r($wp_query); echo "</pre>";
33
+ $template = locate_template(array('page.php','index.php'),false); //category becomes a page
34
+ }
35
+ }
36
+ return $template;
37
+ }
38
+
39
+ function the_content($content){
40
+ global $wp_query, $EM_Category;
41
+ $EM_Category = new EM_Category($wp_query->queried_object);
42
+ ob_start();
43
+ em_locate_template('templates/category-single.php',true);
44
+ return ob_get_clean();
45
+ }
46
+
47
+ function parse_query( ){
48
+ global $wp_query;
49
+ if( !empty($wp_query->tax_query->queries[0]['taxonomy']) && $wp_query->tax_query->queries[0]['taxonomy'] == EM_TAXONOMY_CATEGORY) {
50
+ //Scope is future
51
+ $today = strtotime(date('Y-m-d', current_time('timestamp')));
52
+ if( get_option('dbem_events_current_are_past') ){
53
+ $wp_query->query_vars['meta_query'][] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '>=' );
54
+ }else{
55
+ $wp_query->query_vars['meta_query'][] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '>=' );
56
+ }
57
+ if( get_option('dbem_categories_default_archive_orderby') == 'title'){
58
+ $wp_query->query_vars['orderby'] = 'title';
59
+ }else{
60
+ $wp_query->query_vars['orderby'] = 'meta_value_num';
61
+ $wp_query->query_vars['meta_key'] = get_option('dbem_categories_default_archive_orderby','_start_ts');
62
+ }
63
+ $wp_query->query_vars['order'] = get_option('dbem_categories_default_archive_order','ASC');
64
+ }
65
+ }
66
+ }
67
+ EM_Category_Taxonomy::init();
classes/em-category.php CHANGED
@@ -1,152 +1,214 @@
1
- <?php
2
- //TODO expand em_category to be like other classes
3
- class EM_Category extends EM_Object {
4
- //DB Fields
5
- var $id = '';
6
- var $owner = '';
7
- var $name = '';
8
- //Other Vars
9
- var $fields = array(
10
- 'category_id' => array('name'=>'id','type'=>'%d'),
11
- 'category_owner' => array('name'=>'owner','type'=>'%d'),
12
- 'category_name' => array('name'=>'name','type'=>'%s')
13
- );
14
- var $required_fields;
15
- var $feedback_message = "";
16
- var $errors = array();
17
-
18
- /**
19
- * Gets data from POST (default), supplied array, or from the database if an ID is supplied
20
- * @param $location_data
21
- * @return null
22
- */
23
- function EM_Category( $category_data = false ) {
24
- //Initialize
25
- $this->required_fields = array("category_name" => __('The category name', 'dbem'));
26
- $category = array();
27
- if( $category_data != false ){
28
- //Load location data
29
- if( is_array($category_data) && isset($category_data['category_name']) ){
30
- $category = $category_data;
31
- }elseif( is_numeric($category_data) ){
32
- //Retreiving from the database
33
- global $wpdb;
34
- $sql = "SELECT * FROM ". $wpdb->prefix.EM_CATEGORIES_TABLE ." WHERE category_id ='{$category_data}'";
35
- $category = $wpdb->get_row($sql, ARRAY_A);
36
- }
37
- //Save into the object
38
- $this->to_object($category);
39
- }
40
- }
41
-
42
- function get_post(){
43
- //We are getting the values via POST or GET
44
- do_action('em_location_get_post_pre', $this);
45
- $category = array();
46
- $category['category_id'] = ( !empty($_POST['category_id']) ) ? $_POST['category_id']:'';
47
- $category['category_name'] = ( !empty($_POST['category_name']) ) ? stripslashes($_POST['category_name']):'';
48
- $category['category_owner'] = ( !empty($_POST['category_owner']) && is_numeric($_POST['category_owner']) ) ? $_POST['category_owner']:get_current_user_id();
49
- $this->to_object( apply_filters('em_category_get_post', $category, $this) );
50
- }
51
-
52
- function save(){
53
- global $wpdb;
54
- do_action('em_category_save_pre', $this);
55
- $table = $wpdb->prefix.EM_CATEGORIES_TABLE;
56
- $data = $this->to_array();
57
- unset($data['category_id']);
58
- if($this->id != ''){
59
- $where = array( 'category_id' => $this->id );
60
- $wpdb->update($table, $data, $where, $this->get_types($data));
61
- }else{
62
- $wpdb->insert($table, $data, $this->get_types($data));
63
- $this->id = $wpdb->insert_id;
64
- }
65
- return apply_filters('em_category_save', ( $this->id > 0 && $image_upload ), $this, $image_upload);
66
- }
67
-
68
- function delete(){
69
- global $wpdb;
70
- do_action('em_category_delete_pre', $this);
71
- $table_name = $wpdb->prefix.EM_CATEGORIES_TABLE;
72
- $sql = "DELETE FROM $table_name WHERE category_id = '{$this->id}';";
73
- $result = $wpdb->query($sql);
74
- return apply_filters('em_category_delete', $result, $this);
75
- }
76
-
77
- /**
78
- * Validates the category. Should be run during any form submission or saving operation.
79
- * @return boolean
80
- */
81
- function validate(){
82
- $missing_fields = Array ();
83
- foreach ( $this->required_fields as $key => $field ) {
84
- $true_field = $this->fields[$key]['name'];
85
- if ( $this->$true_field == "") {
86
- $missing_fields[] = $field;
87
- }
88
- }
89
- if ( count($missing_fields) > 0){
90
- // TODO Create friendly equivelant names for missing fields notice in validation
91
- $this->errors[] = __ ( 'Missing fields: ' ) . implode ( ", ", $missing_fields ) . ". ";
92
- }
93
- return apply_filters('em_category_validate', ( count($this->errors) == 0 ), $this);
94
- }
95
-
96
- function has_events(){
97
- global $wpdb;
98
- $events_table = $wpdb->prefix.EM_EVENTS_TABLE;
99
- $sql = "SELECT count(event_id) as events_no FROM $events_table WHERE category_id = {$this->id}";
100
- $affected_events = $wpdb->get_row($sql);
101
- return apply_filters('em_category_has_events', (count($affected_events) > 0), $this);
102
- }
103
-
104
- function output_single($target = 'html'){
105
- $format = get_option ( 'dbem_single_category_format' );
106
- return apply_filters('em_category_output_single', $this->output($format, $target), $this, $target);
107
- }
108
-
109
- function output($format, $target="html") {
110
- $category_string = $format;
111
- preg_match_all("/#_[A-Za-z]+/", $format, $placeholders);
112
- foreach($placeholders[0] as $result) {
113
- $match = true;
114
- $replace = '';
115
- switch( $result ){
116
- case '#_CATEGORYNAME':
117
- $replace = $this->name;
118
- break;
119
- case '#_CATEGORYID':
120
- $replace = $this->id;
121
- break;
122
- default:
123
- $match = false;
124
- break;
125
- }
126
- if($match){ //if true, we've got a placeholder that needs replacing
127
- //TODO FILTER - placeholder filter
128
- $replace = apply_filters('em_category_output_placeholder', $replace, $this, $result, $target); //USE WITH CAUTION! THIS MIGHT GET RENAMED
129
- $category_string = str_replace($result, $replace , $category_string );
130
- }
131
- }
132
- $name_filter = ($target == "html") ? 'dbem_general':'dbem_general_rss'; //TODO remove dbem_ filters
133
- $category_string = str_replace('#_CATEGORY', apply_filters($name_filter, $this->name) , $category_string ); //Depreciated
134
- return apply_filters('em_category_output', $category_string, $this, $format, $target);
135
- }
136
-
137
- function can_manage(){
138
- return ( get_option('dbem_permissions_categories') == 2 || $this->owner == get_current_user_id() || empty($this->id) || em_verify_admin() );
139
- }
140
-
141
- function can_use(){
142
- switch( get_option('dbem_permissions_locations') ){
143
- case 0:
144
- return $this->owner == get_current_user_id();
145
- case 1:
146
- return em_verify_admin($this->owner);
147
- case 2:
148
- return true;
149
- }
150
- }
151
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
152
  ?>
1
+ <?php
2
+ /**
3
+ * Get an category in a db friendly way, by checking globals and passed variables to avoid extra class instantiations
4
+ * @param mixed $id
5
+ * @return EM_Category
6
+ */
7
+ function em_get_category($id = false) {
8
+ global $EM_Category;
9
+ //check if it's not already global so we don't instantiate again
10
+ if( is_object($EM_Category) && get_class($EM_Category) == 'EM_Category' ){
11
+ if( $EM_Category->term_id == $id ){
12
+ return $EM_Category;
13
+ }elseif( is_object($id) && $EM_Category->term_id == $id->term_id ){
14
+ return $EM_Category;
15
+ }
16
+ }
17
+ if( is_object($id) && get_class($id) == 'EM_Category' ){
18
+ return $id;
19
+ }else{
20
+ return new EM_Category($id);
21
+ }
22
+ }
23
+ class EM_Category extends EM_Object {
24
+ //Taxonomy Fields
25
+ var $id = '';
26
+ var $term_id;
27
+ var $name;
28
+ var $slug;
29
+ var $term_group;
30
+ var $term_taxonomy_id;
31
+ var $taxonomy;
32
+ var $description = '';
33
+ var $parent = 0;
34
+ var $count;
35
+ //extra attributes imposed by EM_Category
36
+ var $image_url = '';
37
+ var $color;
38
+
39
+ /**
40
+ * Gets data from POST (default), supplied array, or from the database if an ID is supplied
41
+ * @param $category_data
42
+ * @return null
43
+ */
44
+ function EM_Category( $category_data = false ) {
45
+ global $wpdb;
46
+ $this->ms_global_switch();
47
+ //Initialize
48
+ $category = array();
49
+ if( !empty($category_data) ){
50
+ //Load category data
51
+ if( is_object($category_data) && !empty($category_data->taxonomy) && $category_data->taxonomy == EM_TAXONOMY_CATEGORY ){
52
+ $category = $category_data;
53
+ }elseif( !is_numeric($category_data) ){
54
+ $category = get_term_by('slug', $category_data, EM_TAXONOMY_CATEGORY);
55
+ }else{
56
+ $category = get_term_by('id', $category_data, EM_TAXONOMY_CATEGORY);
57
+ }
58
+ }
59
+ if( is_object($category) || is_array($category) ){
60
+ foreach($category as $key => $value){
61
+ $this->$key = $value;
62
+ }
63
+ }
64
+ $this->id = $this->term_id; //backward compatability
65
+ $this->ms_global_switch_back();
66
+ do_action('em_category',$this, $category_data);
67
+ }
68
+
69
+ function get_color(){
70
+ if( empty($this->color) ){
71
+ global $wpdb;
72
+ $color = $wpdb->get_var('SELECT meta_value FROM '.EM_META_TABLE." WHERE object_id='{$this->term_id}' AND meta_key='category-bgcolor' LIMIT 1");
73
+ $this->color = ($color != '') ? $color:'#FFFFFF';
74
+ }
75
+ return $this->color;
76
+ }
77
+
78
+ function get_image_url(){
79
+ if( empty($this->image_url) ){
80
+ global $wpdb;
81
+ $image_url = $wpdb->get_var('SELECT meta_value FROM '.EM_META_TABLE." WHERE object_id='{$this->term_id}' AND meta_key='category-image' LIMIT 1");
82
+ $this->image_url = ($image_url != '') ? $image_url:'';
83
+ }
84
+ return $this->image_url;
85
+ }
86
+
87
+ function get_url(){
88
+ if( empty($this->link) ){
89
+ $this->ms_global_switch();
90
+ $this->link = get_term_link($this->slug, EM_TAXONOMY_CATEGORY);
91
+ $this->ms_global_switch_back();
92
+ if ( is_wp_error($this->link) ) $this->link = '';
93
+ }
94
+ return $this->link;
95
+ }
96
+
97
+ function has_events(){
98
+ global $wpdb;
99
+ //FIXME old event category checking
100
+ $events_table = EM_EVENTS_TABLE;
101
+ $sql = "SELECT count(event_id) as events_no FROM $events_table WHERE category_id = {$this->id}";
102
+ $affected_events = $wpdb->get_row($sql);
103
+ return apply_filters('em_category_has_events', (count($affected_events) > 0), $this);
104
+ }
105
+
106
+ function output_single($target = 'html'){
107
+ $format = get_option ( 'dbem_category_page_format' );
108
+ return apply_filters('em_category_output_single', $this->output($format, $target), $this, $target);
109
+ }
110
+
111
+ function output($format, $target="html") {
112
+ preg_match_all('/\{([a-zA-Z0-9_]+)\}([^{]+)\{\/[a-zA-Z0-9_]+\}/', $format, $conditionals);
113
+ if( count($conditionals[0]) > 0 ){
114
+ //Check if the language we want exists, if not we take the first language there
115
+ foreach($conditionals[1] as $key => $condition){
116
+ $format = str_replace($conditionals[0][$key], apply_filters('em_category_output_condition', '', $condition, $conditionals[0][$key], $this), $format);
117
+ }
118
+ }
119
+ $category_string = $format;
120
+ preg_match_all("/(#@?_?[A-Za-z0-9]+)({([a-zA-Z0-9,]+)})?/", $format, $placeholders);
121
+ foreach($placeholders[1] as $key => $result) {
122
+ $replace = '';
123
+ $full_result = $placeholders[0][$key];
124
+ switch( $result ){
125
+ case '#_CATEGORYNAME':
126
+ $replace = $this->name;
127
+ break;
128
+ case '#_CATEGORYID':
129
+ $replace = $this->term_id;
130
+ break;
131
+ case '#_CATEGORYNOTES':
132
+ case '#_CATEGORYDESCRIPTION':
133
+ $replace = $this->description;
134
+ break;
135
+ case '#_CATEGORYIMAGE':
136
+ case '#_CATEGORYIMAGEURL':
137
+ if( $this->get_image_url() != ''){
138
+ if($result == '#_CATEGORYIMAGEURL'){
139
+ $replace = $this->get_image_url();
140
+ }else{
141
+ if( empty($placeholders[3][$key]) ){
142
+ $replace = "<img src='".esc_url($this->get_image_url())."' alt='".esc_attr($this->name)."'/>";
143
+ }else{
144
+ $image_size = explode(',', $placeholders[3][$key]);
145
+ if( $this->array_is_numeric($image_size) && count($image_size) > 1 ){
146
+ $replace = "<img src='".em_get_thumbnail_url($this->get_image_url(), $image_size[0], $image_size[1])."' alt='".esc_attr($this->name)."'/>";
147
+ }else{
148
+ $replace = "<img src='".esc_url($this->get_image_url())."' alt='".esc_attr($this->name)."'/>";
149
+ }
150
+ }
151
+ }
152
+ }
153
+ break;
154
+ case '#_CATEGORYCOLOR':
155
+ $replace = $this->get_color();
156
+ break;
157
+ case '#_CATEGORYLINK':
158
+ case '#_CATEGORYURL':
159
+ $link = $this->get_url();
160
+ $replace = ($result == '#_CATEGORYURL') ? $link : '<a href="'.$link.'">'.esc_html($this->name).'</a>';
161
+ break;
162
+ case '#_CATEGORYEVENTSPAST': //depreciated, erroneous documentation, left for compatability
163
+ case '#_CATEGORYEVENTSNEXT': //depreciated, erroneous documentation, left for compatability
164
+ case '#_CATEGORYEVENTSALL': //depreciated, erroneous documentation, left for compatability
165
+ case '#_CATEGORYPASTEVENTS':
166
+ case '#_CATEGORYNEXTEVENTS':
167
+ case '#_CATEGORYALLEVENTS':
168
+ //convert depreciated placeholders for compatability
169
+ $result = ($result == '#_CATEGORYEVENTSPAST') ? '#_CATEGORYPASTEVENTS':$result;
170
+ $result = ($result == '#_CATEGORYEVENTSNEXT') ? '#_CATEGORYNEXTEVENTS':$result;
171
+ $result = ($result == '#_CATEGORYEVENTSALL') ? '#_CATEGORYALLEVENTS':$result;
172
+ //forget it ever happened? :/
173
+ if ($result == '#_CATEGORYPASTEVENTS'){ $scope = 'past'; }
174
+ elseif ( $result == '#_CATEGORYNEXTEVENTS' ){ $scope = 'future'; }
175
+ else{ $scope = 'all'; }
176
+ $events = EM_Events::get( array('category'=>$this->term_id, 'scope'=>$scope) );
177
+ if ( count($events) > 0 ){
178
+ foreach($events as $EM_Event){
179
+ $replace .= $EM_Event->output(get_option('dbem_category_event_list_item_format'));
180
+ }
181
+ } else {
182
+ $replace = get_option('dbem_category_no_events_message');
183
+ }
184
+ break;
185
+ default:
186
+ $replace = $full_result;
187
+ break;
188
+ }
189
+ $replace = apply_filters('em_category_output_placeholder', $replace, $this, $full_result, $target); //USE WITH CAUTION! THIS MIGHT GET RENAMED
190
+ $category_string = str_replace($full_result, $replace , $category_string );
191
+ }
192
+ $name_filter = ($target == "html") ? 'dbem_general':'dbem_general_rss'; //TODO remove dbem_ filters
193
+ $category_string = str_replace('#_CATEGORY', apply_filters($name_filter, $this->name) , $category_string ); //Depreciated
194
+ return apply_filters('em_category_output', $category_string, $this, $format, $target);
195
+ }
196
+
197
+ function can_manage( $capability_owner = 'edit_categories', $capability_admin = false ){
198
+ global $em_capabilities_array;
199
+ //Figure out if this is multisite and require an extra bit of validation
200
+ $multisite_check = true;
201
+ $can_manage = current_user_can($capability_owner);
202
+ //if multisite and supoer admin, just return true
203
+ if( is_multisite() && is_super_admin() ){ return true; }
204
+ if( EM_MS_GLOBAL && !is_main_site() ){
205
+ //User can't admin this bit, as they're on a sub-blog
206
+ $can_manage = false;
207
+ if(array_key_exists($capability_owner, $em_capabilities_array) ){
208
+ $this->add_error( $em_capabilities_array[$capability_owner]);
209
+ }
210
+ }
211
+ return $can_manage;
212
+ }
213
+ }
214
  ?>
classes/em-event-post-admin.php ADDED
@@ -0,0 +1,373 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /*
3
+ * Events Edit Page
4
+ */
5
+ class EM_Event_Post_Admin{
6
+ function init(){
7
+ global $pagenow;
8
+ if($pagenow == 'post.php' || $pagenow == 'post-new.php' ){ //only needed if editing post
9
+ add_action('admin_head', array('EM_Event_Post_Admin','admin_head'));
10
+ //Meta Boxes
11
+ add_action('add_meta_boxes', array('EM_Event_Post_Admin','meta_boxes'));
12
+ //Notices
13
+ add_action('admin_notices',array('EM_Event_Post_Admin','admin_notices'));
14
+ }
15
+ //Save/Edit actions
16
+ add_action('save_post',array('EM_Event_Post_Admin','save_post'),10,1);
17
+ add_action('before_delete_post',array('EM_Event_Post_Admin','before_delete_post'),10,1);
18
+ add_action('trashed_post',array('EM_Event_Post_Admin','trashed_post'),10,1);
19
+ add_action('untrash_post',array('EM_Event_Post_Admin','untrash_post'),10,1);
20
+ add_action('untrashed_post',array('EM_Event_Post_Admin','untrashed_post'),10,1);
21
+ //Notices
22
+ add_action('post_updated_messages',array('EM_Event_Post_Admin','admin_notices_filter'),1,1);
23
+ }
24
+
25
+ function admin_head(){
26
+ global $post, $EM_Event;
27
+ if( !empty($post) && $post->post_type == EM_POST_TYPE_EVENT ){
28
+ $EM_Event = em_get_event($post->ID, 'post_id');
29
+ }
30
+ }
31
+
32
+ function admin_notices(){
33
+ //When editing
34
+ global $post, $EM_Event, $pagenow;
35
+ if( $pagenow == 'post.php' && ($post->post_type == EM_POST_TYPE_EVENT || $post->post_type == 'event-recurring') ){
36
+ if ( $EM_Event->is_recurring() ) {
37
+ $warning = "<p><strong>".__( 'WARNING: This is a recurring event.', 'dbem' )."</strong></p>";
38
+ $warning .= "<p>". __( 'Modifications to this event will cause all recurrences of this event to be deleted and recreated and previous bookings will be deleted! You can edit individual recurrences and disassociate them with this recurring event.', 'dbem' );
39
+ ?><div class="updated"><?php echo $warning; ?></div><?php
40
+ } elseif ( $EM_Event->is_recurrence() ) {
41
+ $warning = "<p><strong>".__('WARNING: This is a recurrence in a set of recurring events.', 'dbem')."</strong></p>";
42
+ $warning .= "<p>". sprintf(__('If you update this event data and save, it could get overwritten if you edit the recurring event template. To make it an independent, <a href="%s">detach it</a>.', 'dbem' ), $EM_Event->get_detach_url())."</p>";
43
+ $warning .= "<p>".sprintf(__('To manage the whole set, <a href="%s">edit the recurring event template</a>.', 'dbem'),admin_url().'post.php?action=edit&amp;post='.$EM_Event->recurrence_id)."</p>";
44
+ ?><div class="updated"><?php echo $warning; ?></div><?php
45
+ }
46
+ if( !empty($EM_Event->group_id) && function_exists('groups_get_group') ){
47
+ $group = groups_get_group(array('group_id'=>$EM_Event->group_id));
48
+ $warning = sprintf(__('WARNING: This is a event belonging to the group "%s". Other group admins can also modify this event.', 'dbem'), $group->name);
49
+ ?><div class="updated"><p><?php echo $warning; ?></p></div><?php
50
+ }
51
+ }
52
+ }
53
+
54
+ function admin_notices_filter($messages){
55
+ //When editing
56
+ global $post, $EM_Notices;
57
+ if( $post->post_type == EM_POST_TYPE_EVENT || $post->post_type == 'event-recurring' ){
58
+ if ( $EM_Notices->count_errors() > 0 ) {
59
+ unset($_GET['message']);
60
+ }
61
+ }
62
+ return $messages;
63
+ }
64
+
65
+ function save_post($post_id){
66
+ global $wpdb, $EM_Event, $EM_Location, $EM_Notices;
67
+ $post_type = get_post_type($post_id);
68
+ $is_post_type = $post_type == EM_POST_TYPE_EVENT || $post_type == 'event-recurring';
69
+ $saving_status = !in_array(get_post_status($post_id), array('trash','auto-draft')) && !defined('DOING_AUTOSAVE');
70
+ if(!defined('UNTRASHING_'.$post_id) && $is_post_type && $saving_status ){
71
+ if( wp_verify_nonce($_REQUEST['_emnonce'], 'edit_event') ){
72
+ //this is only run if we know form data was submitted, hence the nonce
73
+ do_action('em_event_save_pre', $EM_Event); //technically, the event is saved... but the meta isn't. wp doesn't give an pre-intervention action for this (or does it?)
74
+ $EM_Event = em_get_event($post_id, 'post_id');
75
+ //Handle Errors by making post draft
76
+ $get_meta = $EM_Event->get_post_meta();
77
+ $save_meta = $EM_Event->save_meta();
78
+ $EM_Event->get_categories()->save(); //save categories in case of default category
79
+ if( !$get_meta || !$save_meta ){
80
+ //failed somewhere, set to draft, don't publish
81
+ $EM_Event->set_status(null, true);
82
+ if( $EM_Event->is_recurring() ){
83
+ $EM_Notices->add_error( '<strong>'.__('Your event details are incorrect and recurrences cannot be created, please correct these errors first:','dbem').'</strong>', true); //Always seems to redirect, so we make it static
84
+ }else{
85
+ $EM_Notices->add_error( '<strong>'.sprintf(__('Your %s details are incorrect and cannot be published, please correct these errors first:','dbem'),__('event','dbem')).'</strong>', true); //Always seems to redirect, so we make it static
86
+ }
87
+ $EM_Notices->add_error($EM_Event->get_errors(), true); //Always seems to redirect, so we make it static
88
+ apply_filters('em_event_save', false, $EM_Event);
89
+ }else{
90
+ //if this is just published, we need to email the user about the publication, or send to pending mode again for review
91
+ if( (!$EM_Event->is_recurring() && !current_user_can('publish_events')) || ($EM_Event->is_recurring() && !current_user_can('publish_recurring_events')) ){
92
+ if( $EM_Event->is_published() ){ $EM_Event->set_status(0, true); } //no publishing and editing... security threat
93
+ }else{
94
+ $EM_Event->send_approval_notification();
95
+ }
96
+ apply_filters('em_event_save', true, $EM_Event);
97
+ }
98
+ }else{
99
+ //we're updating only the quick-edit style information, which is only post info saved into the index
100
+ $EM_Event = em_get_event($post_id, 'post_id'); //grab event, via post info
101
+ if( $EM_Event->validate() ){
102
+ //we can save the status now
103
+ $event_status = $EM_Event->get_status(true);
104
+ //if this is just published, we need to email the user about the publication, or send to pending mode again for review
105
+ if( (!$EM_Event->is_recurring() && !current_user_can('publish_events')) || ($EM_Event->is_recurring() && !current_user_can('publish_recurring_events')) ){
106
+ if( $EM_Event->is_published() ){ $EM_Event->set_status(0, true); } //no publishing and editing... security threat
107
+ }else{
108
+ $EM_Event->previous_status = $EM_Event->get_previous_status(); //save previous status into object before approving
109
+ $EM_Event->send_approval_notification();
110
+ }
111
+ //now update the db
112
+ $wpdb->query("UPDATE ".EM_EVENTS_TABLE." SET event_name='{$EM_Event->event_name}', event_slug='{$EM_Event->event_slug}', event_status={$event_status}, event_private={$EM_Event->event_private} WHERE event_id='{$EM_Event->event_id}'");
113
+ if( $EM_Event->is_recurring() && $EM_Event->is_published()){
114
+ //recurrences are (re)saved only if event is published
115
+ $EM_Event->save_events();
116
+ }
117
+ apply_filters('em_event_save', true, $EM_Event);
118
+ }else{
119
+ //Event doesn't validate, so set status to null
120
+ $EM_Event->set_status(null, true);
121
+ apply_filters('em_event_save', false, $EM_Event);
122
+ }
123
+ }
124
+ self::maybe_publish_location($EM_Event);
125
+ }
126
+ }
127
+
128
+ /**
129
+ * Publish the location if the event has just been approved and the location is pending. We assume an editor published the event and approves the location too.
130
+ * @param EM_Event $EM_Event
131
+ */
132
+ function maybe_publish_location($EM_Event){
133
+ //do a dirty update for location too if it's not published
134
+ if( $EM_Event->is_published() && !empty($EM_Event->location_id) ){
135
+ $EM_Location = $EM_Event->get_location();
136
+ if( $EM_Location->location_status !== 1 ){
137
+ //let's also publish the location
138
+ $EM_Location->set_status(1, true);
139
+ }
140
+ }
141
+ }
142
+
143
+ function before_delete_post($post_id){
144
+ if(get_post_type($post_id) == EM_POST_TYPE_EVENT){
145
+ $EM_Event = em_get_event($post_id,'post_id');
146
+ $EM_Event->delete_meta();
147
+ }
148
+ }
149
+
150
+ function trashed_post($post_id){
151
+ if(get_post_type($post_id) == EM_POST_TYPE_EVENT){
152
+ global $EM_Notices;
153
+ $EM_Event = em_get_event($post_id,'post_id');
154
+ $EM_Event->set_status(null);
155
+ $EM_Notices->remove_all(); //no validation/notices needed
156
+ }
157
+ }
158
+
159
+ function untrash_post($post_id){
160
+ if(get_post_type($post_id) == EM_POST_TYPE_EVENT){
161
+ //set a constant so we know this event doesn't need 'saving'
162
+ if(!defined('UNTRASHING_'.$post_id)) define('UNTRASHING_'.$post_id, true);
163
+ }
164
+ }
165
+
166
+ function untrashed_post($post_id){
167
+ if(get_post_type($post_id) == EM_POST_TYPE_EVENT){
168
+ global $EM_Notices;
169
+ $EM_Event = em_get_event($post_id,'post_id');
170
+ $EM_Event->set_status(1);
171
+ $EM_Notices->remove_all(); //no validation/notices needed
172
+ }
173
+ }
174
+
175
+ function meta_boxes(){
176
+ global $EM_Event;
177
+ add_meta_box('em-event-when', __('When','dbem'), array('EM_Event_Post_Admin','meta_box_date'),EM_POST_TYPE_EVENT, 'side','high');
178
+ if(get_option('dbem_locations_enabled', true)){
179
+ add_meta_box('em-event-where', __('Where','dbem'), array('EM_Event_Post_Admin','meta_box_location'),EM_POST_TYPE_EVENT, 'normal','high');
180
+ }
181
+ //add_meta_box('em-event-meta', 'Event Meta (debugging only)', array('EM_Event_Post_Admin','meta_box_metadump'),EM_POST_TYPE_EVENT, 'normal','high');
182
+ if(get_option('dbem_rsvp_enabled', true)){
183
+ add_meta_box('em-event-bookings', __('Bookings/Registration','dbem'), array('EM_Event_Post_Admin','meta_box_bookings'),EM_POST_TYPE_EVENT, 'normal','high');
184
+ if( !empty($EM_Event->event_id) && $EM_Event->event_rsvp ){
185
+ add_meta_box('em-event-bookings-stats', __('Bookings Stats','dbem'), array('EM_Event_Post_Admin','meta_box_bookings_stats'),EM_POST_TYPE_EVENT, 'side','core');
186
+ }
187
+ }
188
+ if( get_option('dbem_attributes_enabled', true) ){
189
+ add_meta_box('em-event-attributes', __('Attributes','dbem'), array('EM_Event_Post_Admin','meta_box_attributes'),EM_POST_TYPE_EVENT, 'normal','default');
190
+ }
191
+ if( (empty($EM_Event->event_id) || !empty($EM_Event->group_id)) && function_exists('groups_get_user_groups') ){
192
+ add_meta_box('em-event-group', __('Group Ownership','dbem'), array('EM_Event_Post_Admin','meta_box_group'),EM_POST_TYPE_EVENT, 'side','low');
193
+ }
194
+ if( EM_MS_GLOBAL && !is_main_site() && get_option('dbem_categories_enabled') ){
195
+ add_meta_box('em-event-categories', __('Site Categories','dbem'), array('EM_Event_Post_Admin','meta_box_ms_categories'),EM_POST_TYPE_EVENT, 'side','low');
196
+ }
197
+ }
198
+ function meta_box_metadump(){
199
+ global $post,$EM_Event;
200
+ echo "<pre>"; print_r($EM_Event); echo "</pre>";
201
+ }
202
+ function meta_box_date(){
203
+ //create meta box check of date nonce
204
+ ?><input type="hidden" name="_emnonce" value="<?php echo wp_create_nonce('edit_event'); ?>" /><?php
205
+ em_locate_template('forms/event/when.php', true);
206
+ }
207
+
208
+ function meta_box_bookings_stats(){
209
+ em_locate_template('forms/event/booking-stats.php',true);
210
+ }
211
+
212
+ function meta_box_bookings(){
213
+ em_locate_template('forms/event/bookings.php', true);
214
+ if( !get_option('dbem_bookings_tickets_single') ){
215
+ add_action('admin_footer',array('EM_Event_Post_Admin','meta_box_bookings_overlay'));
216
+ }
217
+ }
218
+
219
+ function meta_box_bookings_overlay(){
220
+ em_locate_template('forms/tickets-form.php', true); //put here as it can't be in the add event form
221
+ }
222
+
223
+ function meta_box_attributes(){
224
+ em_locate_template('forms/event/attributes.php',true);
225
+ }
226
+
227
+ function meta_box_group(){
228
+ em_locate_template('forms/event/group.php',true);
229
+ }
230
+
231
+ function meta_box_location(){
232
+ em_locate_template('forms/event/location.php',true);
233
+ }
234
+
235
+ function meta_box_ms_categories(){
236
+ global $EM_Event;
237
+ $categories = EM_Categories::get(array('orderby'=>'category_name','hide_empty'=>false));
238
+ ?>
239
+ <?php if( count($categories) > 0 ): ?>
240
+ <p>
241
+ <?php foreach( $categories as $EM_Category ):?>
242
+ <label><input type="checkbox" name="event_categories[]" value="<?php echo $EM_Category->id; ?>" <?php if($EM_Event->get_categories()->has($EM_Category->id)) echo 'checked="checked"'; ?> /> <?php echo $EM_Category->name ?></label><br />
243
+ <?php endforeach; ?>
244
+ </p>
245
+ <?php else: ?>
246
+ <p><?php sprintf(__('No categories available, <a href="%s">create one here first</a>','dbem'), get_bloginfo('wpurl').'/wp-admin/admin.php?page=events-manager-categories'); ?></p>
247
+ <?php endif; ?>
248
+ <!-- END Categories -->
249
+ <?php
250
+ }
251
+ }
252
+ add_action('admin_init',array('EM_Event_Post_Admin','init'));
253
+
254
+ /*
255
+ * Recurring Events
256
+ */
257
+ class EM_Event_Recurring_Post_Admin{
258
+ function init(){
259
+ global $pagenow;
260
+ if($pagenow == 'post.php' || $pagenow == 'post-new.php' ){ //only needed if editing post
261
+ add_action('admin_head', array('EM_Event_Recurring_Post_Admin','admin_head'));
262
+ //Meta Boxes
263
+ add_action('add_meta_boxes', array('EM_Event_Recurring_Post_Admin','meta_boxes'));
264
+ //Notices
265
+ add_action('admin_notices',array('EM_Event_Post_Admin','admin_notices')); //shared with posts
266
+ }
267
+ //Save/Edit actions
268
+ add_action('before_delete_post',array('EM_Event_Recurring_Post_Admin','before_delete_post'),10,1);
269
+ add_action('trashed_post',array('EM_Event_Recurring_Post_Admin','trashed_post'),10,1);
270
+ add_action('untrash_post',array('EM_Event_Recurring_Post_Admin','untrash_post'),10,1);
271
+ add_action('untrashed_post',array('EM_Event_Recurring_Post_Admin','untrashed_post'),10,1);
272
+ //Notices
273
+ add_action('post_updated_messages',array('EM_Event_Post_Admin','admin_notices_filter'),1,1); //shared with posts
274
+ }
275
+
276
+ function admin_head(){
277
+ global $post, $EM_Event;
278
+ if( !empty($post) && $post->post_type == 'event-recurring' ){
279
+ $EM_Event = em_get_event($post->ID, 'post_id');
280
+ //quick hacks to make event admin table make more sense for events
281
+ ?>
282
+ <script type="text/javascript">
283
+ jQuery(document).ready( function($){
284
+ if(!EM.recurrences_menu){
285
+ $('#menu-posts-'+EM.event_post_type+', #menu-posts-'+EM.event_post_type+' > a').addClass('wp-has-current-submenu');
286
+ }
287
+ });
288
+ </script>
289
+ <?php
290
+ }
291
+ }
292
+
293
+ function before_delete_post($post_id){
294
+ if(get_post_type($post_id) == 'event-recurring'){
295
+ $EM_Event = em_get_event($post_id,'post_id');
296
+ //now delete recurrences
297
+ $events_array = EM_Events::get( array('recurrence'=>$EM_Event->event_id, 'scope'=>'all', 'status'=>'all' ) );
298
+ foreach($events_array as $event){
299
+ /* @var $event EM_Event */
300
+ if($EM_Event->event_id == $event->recurrence_id ){ //double check the event is a recurrence of this event
301
+ wp_delete_post($event->post_id, true);
302
+ }
303
+ }
304
+ $EM_Event->post_type = EM_POST_TYPE_EVENT; //trick it into thinking it's one event.
305
+ $EM_Event->delete_meta();
306
+ }
307
+ }
308
+
309
+ function trashed_post($post_id){
310
+ if(get_post_type($post_id) == 'event-recurring'){
311
+ global $EM_Notices, $wpdb;
312
+ $EM_Event = em_get_event($post_id,'post_id');
313
+ $EM_Event->set_status(null);
314
+ //now trash recurrences
315
+ $events_array = EM_Events::get( array('recurrence_id'=>$EM_Event->event_id, 'scope'=>'all', 'status'=>'all' ) );
316
+ foreach($events_array as $event){
317
+ /* @var $event EM_Event */
318
+ if($EM_Event->event_id == $event->recurrence_id ){ //double check the event is a recurrence of this event
319
+ wp_trash_post($event->post_id);
320
+ }
321
+ }
322
+ $EM_Notices->remove_all(); //no validation/notices needed
323
+ }
324
+ }
325
+
326
+ function untrash_post($post_id){
327
+ if(get_post_type($post_id) == 'event-recurring'){
328
+ global $wpdb;
329
+ //set a constant so we know this event doesn't need 'saving'
330
+ if(!defined('UNTRASHING_'.$post_id)) define('UNTRASHING_'.$post_id, true);
331
+ $EM_Event = em_get_event($post_id,'post_id');
332
+ $wpdb->query('UPDATE '.EM_EVENTS_TABLE.' SET event_status='.$EM_Event->event_status.' WHERE event_id='.$EM_Event->event_id);
333
+ $events_array = EM_Events::get( array('recurrence_id'=>$EM_Event->event_id, 'scope'=>'all', 'status'=>'all' ) );
334
+ foreach($events_array as $event){
335
+ /* @var $event EM_Event */
336
+ if($EM_Event->event_id == $event->recurrence_id){
337
+ wp_untrash_post($event->post_id);
338
+ }
339
+ }
340
+ }
341
+ }
342
+
343
+ function untrashed_post($post_id){
344
+ if(get_post_type($post_id) == 'event-recurring'){
345
+ global $EM_Notices,$EM_Event;
346
+ $EM_Event->set_status(1);
347
+ $EM_Notices->remove_all(); //no validation/notices needed
348
+ }
349
+ }
350
+
351
+ function meta_boxes(){
352
+ add_meta_box('em-event-recurring', __('Recurrences','dbem'), array('EM_Event_Recurring_Post_Admin','meta_box_recurrence'),'event-recurring', 'normal','high');
353
+ //add_meta_box('em-event-meta', 'Event Meta (debugging only)', array('EM_Event_Post_Admin','meta_box_metadump'),'event-recurring', 'normal','high');
354
+ add_meta_box('em-event-where', __('Where','dbem'), array('EM_Event_Post_Admin','meta_box_location'),'event-recurring', 'normal','high');
355
+ if(get_option('dbem_rsvp_enabled')){
356
+ add_meta_box('em-event-bookings', __('Bookings/Registration','dbem'), array('EM_Event_Post_Admin','meta_box_bookings'),'event-recurring', 'normal','high');
357
+ }
358
+ if( empty($EM_Event->event_id) && function_exists('groups_get_user_groups') ){
359
+ add_meta_box('em-event-group', __('Group Ownership','dbem'), array('EM_Event_Post_Admin','meta_box_group'),'event-recurring', 'side','low');
360
+ }
361
+ if( get_option('dbem_attributes_enabled') ){
362
+ add_meta_box('em-event-attributes', __('Attributes','dbem'), array('EM_Event_Post_Admin','meta_box_attributes'),'event-recurring', 'normal','default');
363
+ }
364
+ if( (empty($EM_Event->event_id) || !empty($EM_Event->group_id)) && function_exists('groups_get_user_groups') ){
365
+ add_meta_box('em-event-group', __('Group Ownership','dbem'), array('EM_Event_Post_Admin','meta_box_group'),'event-recurring', 'side','low');
366
+ }
367
+ }
368
+
369
+ function meta_box_recurrence(){
370
+ em_locate_template('forms/event/recurring-when.php', true);
371
+ }
372
+ }
373
+ add_action('admin_init',array('EM_Event_Recurring_Post_Admin','init'));
classes/em-event-post.php ADDED
@@ -0,0 +1,256 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Controls how events are queried and displayed via the WordPress Custom Post APIs
4
+ * @author marcus
5
+ *
6
+ */
7
+ class EM_Event_Post {
8
+ function init(){
9
+ global $wp_query;
10
+ //Front Side Modifiers
11
+ if( !is_admin() ){
12
+ //override single page with formats?
13
+ add_filter('the_content', array('EM_Event_Post','the_content'));
14
+ //display as page template?
15
+ if( get_option('dbem_cp_events_template_page') ){
16
+ add_filter('single_template',array('EM_Event_Post','single_template'));
17
+ }
18
+ //Override post template tags
19
+ add_filter('the_date',array('EM_Event_Post','the_date'));
20
+ add_filter('get_the_date',array('EM_Event_Post','the_date'),10,2);
21
+ add_filter('the_category',array('EM_Event_Post','the_category'),10,3);
22
+ }
23
+ add_action('parse_query', array('EM_Event_Post','parse_query'));
24
+ }
25
+
26
+ /**
27
+ * Overrides the default post format of an event and can display an event as a page, which uses the page.php template.
28
+ * @param string $template
29
+ * @return string
30
+ */
31
+ function single_template($template){
32
+ global $post;
33
+ if( $post->post_type == EM_POST_TYPE_EVENT ){
34
+ $template = locate_template(array('page.php','index.php'),false);
35
+ }
36
+ return $template;
37
+ }
38
+
39
+ function the_content( $content ){
40
+ global $post, $EM_Event;
41
+ if( $post->post_type == EM_POST_TYPE_EVENT ){
42
+ if( is_archive() || is_search() ){
43
+ if(get_option('dbem_cp_events_archive_formats')){
44
+ $EM_Event = em_get_event($post);
45
+ $content = $EM_Event->output(get_option('dbem_event_list_item_format'));
46
+ }
47
+ }else{
48
+ if( get_option('dbem_cp_events_formats') && !post_password_required() ){
49
+ $EM_Event = em_get_event($post);
50
+ //general defaults
51
+ $args = array(
52
+ 'owner' => false,
53
+ 'pagination' => 1
54
+ );
55
+ ob_start();
56
+ em_locate_template('templates/event-single.php',true, array('args'=>$args));
57
+ $content = ob_get_clean();
58
+ }
59
+ }
60
+ }
61
+ return $content;
62
+ }
63
+
64
+ function the_date( $the_date, $d = '' ){
65
+ global $post;
66
+ if( $post->post_type == EM_POST_TYPE_EVENT ){
67
+ $EM_Event = em_get_event($post);
68
+ if ( '' == $d ){
69
+ $the_date = date(get_option('date_format'), $EM_Event->start);
70
+ }else{
71
+ $the_date = date($d, $EM_Event->start);
72
+ }
73
+ }
74
+ return $the_date;
75
+ }
76
+
77
+ function the_category( $thelist, $separator = '', $parents='' ){
78
+ global $post, $wp_rewrite;
79
+ if( $post->post_type == EM_POST_TYPE_EVENT ){
80
+ $EM_Event = em_get_event($post);
81
+ $categories = $EM_Event->get_categories();
82
+ if( empty($categories) ) return '';
83
+
84
+ /* Copied from get_the_category_list function, with a few minor edits to make urls work, and removing parent stuff (for now) */
85
+ $rel = ( is_object( $wp_rewrite ) && $wp_rewrite->using_permalinks() ) ? 'rel="category tag"' : 'rel="category"';
86
+
87
+ $thelist = '';
88
+ if ( '' == $separator ) {
89
+ $thelist .= '<ul class="post-categories">';
90
+ foreach ( $categories as $category ) {
91
+ $thelist .= "\n\t<li>";
92
+ switch ( strtolower( $parents ) ) {
93
+ case 'multiple':
94
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>' . $category->name.'</a></li>';
95
+ break;
96
+ case 'single':
97
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>';
98
+ $thelist .= $category->name.'</a></li>';
99
+ break;
100
+ case '':
101
+ default:
102
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>' . $category->name.'</a></li>';
103
+ }
104
+ }
105
+ $thelist .= '</ul>';
106
+ } else {
107
+ $i = 0;
108
+ foreach ( $categories as $category ) {
109
+ if ( 0 < $i )
110
+ $thelist .= $separator;
111
+ switch ( strtolower( $parents ) ) {
112
+ case 'multiple':
113
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>' . $category->name.'</a>';
114
+ break;
115
+ case 'single':
116
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>';
117
+ $thelist .= "$category->name</a>";
118
+ break;
119
+ case '':
120
+ default:
121
+ $thelist .= '<a href="' . $category->get_url() . '" title="' . esc_attr( sprintf( __( "View all posts in %s" ), $category->name ) ) . '" ' . $rel . '>' . $category->name.'</a>';
122
+ }
123
+ ++$i;
124
+ }
125
+ }
126
+ /* End copying */
127
+ }
128
+ return $thelist;
129
+ }
130
+
131
+ function parse_query( ){
132
+ global $wp_query;
133
+ //Search Query Filtering
134
+ if( !empty($wp_query->query_vars['s']) && !get_option('dbem_cp_events_search_results') ){
135
+ $wp_query->query_vars['post_type'] = array_diff( get_post_types(array('exclude_from_search' => false)), array(EM_POST_TYPE_EVENT));
136
+ }
137
+ //Scoping
138
+ if( !empty($wp_query->query_vars['post_type']) && ($wp_query->query_vars['post_type'] == EM_POST_TYPE_EVENT || $wp_query->query_vars['post_type'] == 'event-recurring') && (empty($wp_query->query_vars['post_status']) || !in_array($wp_query->query_vars['post_status'],array('trash','pending','draft'))) ) {
139
+ //Let's deal with the scope - default is future
140
+ if( is_admin() ){
141
+ $scope = $wp_query->query_vars['scope'] = (!empty($_REQUEST['scope'])) ? $_REQUEST['scope']:'future';
142
+ //TODO limit what a user can see admin side for events/locations/recurring events
143
+ }else{
144
+ if( !empty($wp_query->query_vars['calendar_day']) ) $wp_query->query_vars['scope'] = $wp_query->query_vars['calendar_day'];
145
+ if( empty($wp_query->query_vars['scope']) ){
146
+ if( is_archive() ){
147
+ $scope = $wp_query->query_vars['scope'] = get_option('dbem_events_page_scope');
148
+ }else{
149
+ $scope = $wp_query->query_vars['scope'] = 'all'; //otherwise we'll get 404s for past events
150
+ }
151
+ }else{
152
+ $scope = $wp_query->query_vars['scope'];
153
+ }
154
+ }
155
+ $query = array();
156
+ $time = current_time('timestamp');
157
+ if ( preg_match ( "/^[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $scope ) ) {
158
+ $today = strtotime($scope);
159
+ $tomorrow = $today + 60*60*24-1;
160
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
161
+ $query[] = array( 'key' => '_start_ts', 'value' => array($today,$tomorrow), 'compare' => 'BETWEEN' );
162
+ }else{
163
+ $query[] = array( 'key' => '_start_ts', 'value' => $tomorrow, 'compare' => '<=' );
164
+ $query[] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '>=' );
165
+ }
166
+ }elseif ($scope == "future"){
167
+ $today = strtotime(date('Y-m-d', $time));
168
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
169
+ $query[] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '>=' );
170
+ }else{
171
+ $query[] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '>=' );
172
+ }
173
+ }elseif ($scope == "past"){
174
+ $today = strtotime(date('Y-m-d', $time));
175
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
176
+ $query[] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '<' );
177
+ }else{
178
+ $query[] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '<' );
179
+ }
180
+ }elseif ($scope == "today"){
181
+ $today = strtotime(date('Y-m-d', $time));
182
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
183
+ //date must be only today
184
+ $query[] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '=');
185
+ }else{
186
+ $query[] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '<=' );
187
+ $query[] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '>=' );
188
+ }
189
+ }elseif ($scope == "tomorrow"){
190
+ $tomorrow = strtotime(date('Y-m-d',$time+60*60*24));
191
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
192
+ //date must be only tomorrow
193
+ $query[] = array( 'key' => '_start_ts', 'value' => $tomorrow, 'compare' => '=');
194
+ }else{
195
+ $query[] = array( 'key' => '_start_ts', 'value' => $tomorrow, 'compare' => '<=' );
196
+ $query[] = array( 'key' => '_end_ts', 'value' => $tomorrow, 'compare' => '>=' );
197
+ }
198
+ }elseif ($scope == "month"){
199
+ $start_month = strtotime(date('Y-m-d',$time));
200
+ $end_month = strtotime(date('Y-m-t',$time));
201
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
202
+ $query[] = array( 'key' => '_start_ts', 'value' => array($start_month,$end_month), 'type' => 'numeric', 'compare' => 'BETWEEN');
203
+ }else{
204
+ $query[] = array( 'key' => '_start_ts', 'value' => $end_month, 'compare' => '<=' );
205
+ $query[] = array( 'key' => '_end_ts', 'value' => $start_month, 'compare' => '>=' );
206
+ }
207
+ }elseif ($scope == "next-month"){
208
+ $start_month_timestamp = strtotime('+1 month', $time); //get the end of this month + 1 day
209
+ $start_month = strtotime(date('Y-m-1',$start_month_timestamp));
210
+ $end_month = strtotime(date('Y-m-t',$start_month_timestamp));
211
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
212
+ $query[] = array( 'key' => '_start_ts', 'value' => array($start_month,$end_month), 'type' => 'numeric', 'compare' => 'BETWEEN');
213
+ }else{
214
+ $query[] = array( 'key' => '_start_ts', 'value' => $end_month, 'compare' => '<=' );
215
+ $query[] = array( 'key' => '_end_ts', 'value' => $start_month, 'compare' => '>=' );
216
+ }
217
+ }elseif( preg_match('/(\d\d?)\-months/',$scope,$matches) ){ // next x months means this month (what's left of it), plus the following x months until the end of that month.
218
+ $months_to_add = $matches[1];
219
+ $start_month = strtotime(date('Y-m-d',$time));
220
+ $end_month = strtotime(date('Y-m-t',strtotime("+$months_to_add month", $time)));
221
+ if( get_option('dbem_events_current_are_past') && $wp_query->query_vars['post_type'] != 'event-recurring' ){
222
+ $query[] = array( 'key' => '_start_ts', 'value' => array($start_month,$end_month), 'type' => 'numeric', 'compare' => 'BETWEEN');
223
+ }else{
224
+ $query[] = array( 'key' => '_start_ts', 'value' => $end_month, 'compare' => '<=' );
225
+ $query[] = array( 'key' => '_end_ts', 'value' => $start_month, 'compare' => '>=' );
226
+ }
227
+ }
228
+ if( !empty($query) && is_array($query) ){
229
+ $wp_query->query_vars['meta_query'] = $query;
230
+ }
231
+ if( is_admin() ){
232
+ //admin areas don't need special ordering, so make it simple
233
+ $wp_query->query_vars['orderby'] = (!empty($_REQUEST['orderby'])) ? $_REQUEST['orderby']:'meta_value_num';
234
+ $wp_query->query_vars['meta_key'] = '_start_ts';
235
+ $wp_query->query_vars['order'] = (!empty($_REQUEST['order'])) ? $_REQUEST['order']:'ASC';
236
+ }else{
237
+ if( get_option('dbem_events_default_archive_orderby') == 'title'){
238
+ $wp_query->query_vars['orderby'] = 'title';
239
+ $wp_query->query_vars['order'] = get_option('dbem_events_default_archive_orderby','ASC');
240
+ }else{
241
+ $wp_query->query_vars['orderby'] = 'meta_value_num';
242
+ $wp_query->query_vars['meta_key'] = '_start_ts';
243
+ }
244
+ $wp_query->query_vars['order'] = get_option('dbem_events_default_archive_orderby','ASC');
245
+ }
246
+ }elseif( !empty($wp_query->query_vars['post_type']) && $wp_query->query_vars['post_type'] == EM_POST_TYPE_EVENT ){
247
+ $wp_query->query_vars['scope'] = 'all';
248
+ if( $wp_query->query_vars['post_status'] == 'pending' ){
249
+ $wp_query->query_vars['orderby'] = 'meta_value_num';
250
+ $wp_query->query_vars['order'] = 'ASC';
251
+ $wp_query->query_vars['meta_key'] = '_start_ts';
252
+ }
253
+ }
254
+ }
255
+ }
256
+ EM_Event_Post::init();
classes/em-event-posts-admin.php ADDED
@@ -0,0 +1,243 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Event_Posts_Admin{
3
+ function init(){
4
+ global $pagenow;
5
+ if( $pagenow == 'edit.php' && !empty($_REQUEST['post_type']) && $_REQUEST['post_type'] == EM_POST_TYPE_EVENT ){ //only needed for events list
6
+ $row_action_type = is_post_type_hierarchical( EM_POST_TYPE_EVENT ) ? 'page_row_actions' : 'post_row_actions';
7
+ add_filter($row_action_type, array('EM_Event_Posts_Admin','row_actions'),10,2);
8
+ add_action('admin_head', array('EM_Event_Posts_Admin','admin_head'));
9
+ //collumns
10
+ add_filter('manage_edit-'.EM_POST_TYPE_EVENT.'_columns' , array('EM_Event_Posts_Admin','columns_add'));
11
+ add_filter('manage_'.EM_POST_TYPE_EVENT.'_posts_custom_column' , array('EM_Event_Posts_Admin','columns_output'),10,2 );
12
+ //TODO alter views of locations, events and recurrences, specifically find a good way to alter the wp_count_posts method to force user owned posts only
13
+ //add_filter('views_edit-'.EM_POST_TYPE_EVENT, array('EM_Event_Posts_Admin','views'),10,1);
14
+ }
15
+ add_action('restrict_manage_posts', array('EM_Event_Posts_Admin','restrict_manage_posts'));
16
+ }
17
+
18
+ function admin_head(){
19
+ //quick hacks to make event admin table make more sense for events
20
+ ?>
21
+ <script type="text/javascript">
22
+ jQuery(document).ready( function($){
23
+ $('.inline-edit-date').prev().css('display','none').next().css('display','none').next().css('display','none');
24
+ $('.em-detach-link').click(function( event ){
25
+ if( !confirm(EM.event_detach_warning) ){
26
+ event.preventDefault();
27
+ return false;
28
+ }
29
+ });
30
+ $('.em-delete-recurrence-link').click(function( event ){
31
+ if( !confirm(EM.delete_recurrence_warning) ){
32
+ event.preventDefault();
33
+ return false;
34
+ }
35
+ });
36
+ });
37
+ </script>
38
+ <style>
39
+ table.fixed{ table-layout:auto !important; }
40
+ .tablenav select[name="m"] { display:none; }
41
+ </style>
42
+ <?php
43
+ }
44
+
45
+ function restrict_manage_posts(){
46
+ global $wp_query;
47
+ if( $wp_query->query_vars['post_type'] == EM_POST_TYPE_EVENT || $wp_query->query_vars['post_type'] == 'event-recurring' ){
48
+ ?>
49
+ <select name="scope">
50
+ <?php
51
+ $scope = (!empty($wp_query->query_vars['scope'])) ? $wp_query->query_vars['scope']:'future';
52
+ foreach ( em_get_scopes() as $key => $value ) {
53
+ $selected = "";
54
+ if ($key == $scope)
55
+ $selected = "selected='selected'";
56
+ echo "<option value='$key' $selected>$value</option> ";
57
+ }
58
+ ?>
59
+ </select>
60
+ <?php
61
+ if( get_option('dbem_categories_enabled') ){
62
+ //Categories
63
+ $terms = get_terms(EM_TAXONOMY_CATEGORY);
64
+ // output html for taxonomy dropdown filter
65
+ echo '<select name="'.EM_TAXONOMY_CATEGORY.'" id="'.EM_TAXONOMY_CATEGORY.'" class="postform">';
66
+ echo '<option value="">'.__('View all categories').'&nbsp;</option>';
67
+ foreach ($terms as $term) {
68
+ // output each select option line, check against the last $_GET to show the current option selected
69
+ $selected = (!empty($_GET[EM_TAXONOMY_CATEGORY]) && $_GET[EM_TAXONOMY_CATEGORY] == $term->slug) ? 'selected="selected"':'';
70
+ echo '<option value="'. $term->slug.'" '.$selected.'>'.$term->name.'</option>';
71
+ }
72
+ echo "</select>";
73
+ }
74
+ if( !empty($_REQUEST['author']) ){
75
+ ?>
76
+ <input type="hidden" name="author" value="<?php echo $_REQUEST['author'] ?>" />
77
+ <?php
78
+ }
79
+ }
80
+ }
81
+
82
+ function views($views){
83
+ if( !current_user_can('edit_others_events') ){
84
+ //alter the views to reflect correct numbering
85
+
86
+ }
87
+ return $views;
88
+ }
89
+
90
+ function columns_add($columns) {
91
+ unset($columns['comments']);
92
+ unset($columns['date']);
93
+ unset($columns['author']);
94
+ $columns = array_merge($columns, array(
95
+ 'location' => __('Location'),
96
+ 'date-time' => __('Date and Time'),
97
+ 'author' => __('Owner','dbem'),
98
+ 'extra' => ''
99
+ ));
100
+ if( !get_option('dbem_locations_enabled') ){
101
+ unset($columns['location']);
102
+ }
103
+ return $columns;
104
+ }
105
+
106
+ function columns_output( $column ) {
107
+ global $post, $EM_Event;
108
+ $EM_Event = em_get_event($post, 'post_id');
109
+ /* @var $post EM_Event */
110
+ switch ( $column ) {
111
+ case 'location':
112
+ //get meta value to see if post has location, otherwise
113
+ $EM_Location = $EM_Event->get_location();
114
+ if( !empty($EM_Location->location_id) ){
115
+ echo "<strong>" . $EM_Location->location_name . "</strong><br/>" . $EM_Location->location_address . " - " . $EM_Location->location_town;
116
+ }else{
117
+ echo __('None','dbem');
118
+ }
119
+ break;
120
+ case 'date-time':
121
+ //get meta value to see if post has location, otherwise
122
+ $localised_start_date = date_i18n('D d M Y', $EM_Event->start);
123
+ $localised_end_date = date_i18n('D d M Y', $EM_Event->end);
124
+ echo $localised_start_date;
125
+ echo ($localised_end_date != $localised_start_date) ? " - $localised_end_date":'';
126
+ echo "<br />";
127
+ if(!$EM_Event->event_all_day){
128
+ echo substr ( $EM_Event->start_time, 0, 5 ) . " - " . substr ( $EM_Event->end_time, 0, 5 );
129
+ }else{
130
+ echo get_option('dbem_event_all_day_message');
131
+ }
132
+ break;
133
+ case 'extra':
134
+ if( get_option('dbem_rsvp_enabled') == 1 && $EM_Event->rsvp == 1 && $EM_Event->can_manage('manage_bookings','manage_others_bookings')){
135
+ ?>
136
+ <a href="<?php echo EM_ADMIN_URL; ?>&amp;page=events-manager-bookings&amp;event_id=<?php echo $EM_Event->event_id ?>"><?php echo __("Bookings",'dbem'); ?></a> &ndash;
137
+ <?php _e("Booked",'dbem'); ?>: <?php echo $EM_Event->get_bookings()->get_booked_spaces()."/".$EM_Event->get_spaces(); ?>
138
+ <?php if( get_option('dbem_bookings_approval') == 1 ): ?>
139
+ | <?php _e("Pending",'dbem') ?>: <?php echo $EM_Event->get_bookings()->get_pending_spaces(); ?>
140
+ <?php endif;
141
+ echo ($EM_Event->is_recurrence()) ? '<br />':'';
142
+ }
143
+ if ( $EM_Event->is_recurrence() && $EM_Event->can_manage('edit_recurring_events','edit_others_recurring_events') ) {
144
+ $recurrence_delete_confirm = __('WARNING! You will delete ALL recurrences of this event, including booking history associated with any event in this recurrence. To keep booking information, go to the relevant single event and save it to detach it from this recurrence series.','dbem');
145
+ ?>
146
+ <strong>
147
+ <?php echo $EM_Event->get_recurrence_description(); ?> <br />
148
+ </strong>
149
+ <div class="row-actions">
150
+ <a href="<?php echo admin_url(); ?>post.php?action=edit&amp;post=<?php echo $EM_Event->get_event_recurrence()->post_id ?>"><?php _e ( 'Edit Recurring Events', 'dbem' ); ?></a> | <span class="trash"><a class="em-delete-recurrence-link" href="<?php echo get_delete_post_link($EM_Event->get_event_recurrence()->post_id); ?>"><?php _e('Delete','dbem'); ?></a></span> | <a class="em-detach-link" href="<?php echo $EM_Event->get_detach_url(); ?>"><?php _e('Detach', 'dbem'); ?></a>
151
+ </div>
152
+ <?php
153
+ }
154
+
155
+ break;
156
+ }
157
+ }
158
+
159
+ function row_actions($actions, $post){
160
+ if($post->post_type == EM_POST_TYPE_EVENT){
161
+ global $post, $EM_Event;
162
+ $EM_Event = em_get_event($post, 'post_id');
163
+ $actions['duplicate'] = '<a href="'.admin_url().'edit.php?action=event_duplicate&amp;event_id='.$EM_Event->event_id.'&amp;_wpnonce='.wp_create_nonce('event_duplicate_'.$EM_Event->event_id).'" title="'.sprintf(__('Duplicate %s','dbem'), __('Event','dbem')).'">'.__('Duplicate','dbem').'</a>';
164
+ }
165
+ return $actions;
166
+ }
167
+ }
168
+ add_action('admin_init', array('EM_Event_Posts_Admin','init'));
169
+
170
+ /*
171
+ * Recurring Events
172
+ */
173
+ class EM_Event_Recurring_Posts_Admin{
174
+ function init(){
175
+ global $pagenow;
176
+ if( $pagenow == 'edit.php' && !empty($_REQUEST['post_type']) && $_REQUEST['post_type'] == 'event-recurring' ){
177
+ add_action('admin_notices',array('EM_Event_Recurring_Posts_Admin','admin_notices'));
178
+ add_action('admin_head', array('EM_Event_Recurring_Posts_Admin','admin_head'));
179
+ //collumns
180
+ add_filter('manage_edit-event-recurring_columns' , array('EM_Event_Recurring_Posts_Admin','columns_add'));
181
+ add_filter('manage_posts_custom_column' , array('EM_Event_Recurring_Posts_Admin','columns_output'),10,1 );
182
+ add_action('restrict_manage_posts', array('EM_Event_Posts_Admin','restrict_manage_posts'));
183
+ }
184
+ }
185
+
186
+ function admin_notices(){
187
+ $warning = sprintf(__( 'Modifications to these events will cause all recurrences of each event to be deleted and recreated and previous bookings will be deleted! You can edit individual recurrences and detach them from recurring events by visiting the <a href="%s">events page</a>.', 'dbem' ), admin_url().'edit.php?post_type='.EM_POST_TYPE_EVENT);
188
+ ?><div class="updated"><p><?php echo $warning; ?></p></div><?php
189
+ }
190
+
191
+ function admin_head(){
192
+ //quick hacks to make event admin table make more sense for events
193
+ ?>
194
+ <script type="text/javascript">
195
+ jQuery(document).ready( function($){
196
+ $('.inline-edit-date').prev().css('display','none').next().css('display','none').next().css('display','none');
197
+ if(!EM.recurrences_menu){
198
+ $('#menu-posts-'+EM.event_post_type+', #menu-posts-'+EM.event_post_type+' > a').addClass('wp-has-current-submenu');
199
+ }
200
+ });
201
+ </script>
202
+ <style>
203
+ table.fixed{ table-layout:auto !important; }
204
+ .tablenav select[name="m"] { display:none; }
205
+ </style>
206
+ <?php
207
+ }
208
+
209
+ function columns_add($columns) {
210
+ unset($columns['comments']);
211
+ unset($columns['date']);
212
+ unset($columns['author']);
213
+ return array_merge($columns, array(
214
+ 'location' => __('Location'),
215
+ 'date-time' => __('Date and Time'),
216
+ 'author' => __('Owner','dbem'),
217
+ ));
218
+ }
219
+
220
+
221
+ function columns_output( $column ) {
222
+ global $post, $EM_Event;
223
+ if( $post->post_type == 'event-recurring' ){
224
+ $post = $EM_Event = em_get_event($post);
225
+ /* @var $post EM_Event */
226
+ switch ( $column ) {
227
+ case 'location':
228
+ //get meta value to see if post has location, otherwise
229
+ $EM_Location = $EM_Event->get_location();
230
+ if( !empty($EM_Location->location_id) ){
231
+ echo "<strong>" . $EM_Location->location_name . "</strong><br/>" . $EM_Location->location_address . " - " . $EM_Location->location_town;
232
+ }else{
233
+ echo __('None','dbem');
234
+ }
235
+ break;
236
+ case 'date-time':
237
+ echo $EM_Event->get_recurrence_description();
238
+ break;
239
+ }
240
+ }
241
+ }
242
+ }
243
+ add_action('admin_init', array('EM_Event_Recurring_Posts_Admin','init'));
classes/em-event.php CHANGED
@@ -1,1006 +1,1791 @@
1
- <?php
2
- /**
3
- * Event Object. This holds all the info pertaining to an event, including location and recurrence info.
4
- * An event object can be one of three "types" a recurring event, recurrence of a recurring event, or a single event.
5
- * The single event might be part of a set of recurring events, but if loaded by specific event id then any operations and saves are
6
- * specifically done on this event. However, if you edit the recurring group, any changes made to single events are overwritten.
7
- *
8
- * @author marcus
9
- */
10
- //TODO Can add more recurring functionality such as "also update all future recurring events" or "edit all events" like google calendar does.
11
- //TODO Integrate recurrences into events table
12
- //FIXME If you create a super long recurrence timespan, there could be thousands of events... need an upper limit here.
13
- class EM_Event extends EM_Object{
14
- /**
15
- * Assoc array where keys are names of database fields and values are array corresponding object property name, regex, data types, etc.
16
- * for use when importing/exporting event data between database and object
17
- * @var array
18
- */
19
- var $fields = array(
20
- 'event_id' => array( 'name'=>'id', 'type'=>'%d' ),
21
- 'event_author' => array( 'name'=>'author', 'type'=>'%d' ),
22
- 'event_name' => array( 'name'=>'name', 'type'=>'%s' ),
23
- 'event_start_time' => array( 'name'=>'start_time', 'type'=>'%s' ),
24
- 'event_end_time' => array( 'name'=>'end_time', 'type'=>'%s' ),
25
- 'event_start_date' => array( 'name'=>'start_date', 'type'=>'%s' ),
26
- 'event_end_date' => array( 'name'=>'end_date', 'type'=>'%s' ),
27
- 'event_notes' => array( 'name'=>'notes', 'type'=>'%s' ),
28
- 'event_rsvp' => array( 'name'=>'rsvp', 'type'=>'%d' ),
29
- 'event_seats' => array( 'name'=>'seats', 'type'=>'%d' ),
30
- 'event_contactperson_id' => array( 'name'=>'contactperson_id', 'type'=>'%d' ),
31
- 'location_id' => array( 'name'=>'location_id', 'type'=>'%d' ),
32
- 'recurrence_id' => array( 'name'=>'recurrence_id', 'type'=>'%d' ),
33
- 'event_category_id' => array( 'name'=>'category_id', 'type'=>'%d' ),
34
- 'event_attributes' => array( 'name'=>'attributes', 'type'=>'%s' ),
35
- 'recurrence' => array( 'name'=>'recurrence', 'type'=>'%d' ),
36
- 'recurrence_interval' => array( 'name'=>'interval', 'type'=>'%d' ), //every x day(s)/week(s)/month(s)
37
- 'recurrence_freq' => array( 'name'=>'freq', 'type'=>'%s' ), //daily,weekly,monthly?
38
- 'recurrence_byday' => array( 'name'=>'byday', 'type'=>'%s' ), //if weekly or monthly, what days of the week?
39
- 'recurrence_byweekno' => array( 'name'=>'byweekno', 'type'=>'%d' ), //if monthly which week (-1 is last)
40
- 'event_date_created' => array( 'name'=>'date_created', 'type'=>'%s' ),
41
- 'event_date_modified' => array( 'name'=>'date_modified', 'type'=>'%s' )
42
- );
43
- /* Field Names - see above for matching DB field names and other field meta data */
44
- var $id;
45
- var $author;
46
- var $name;
47
- var $start_time;
48
- var $end_time;
49
- var $start_date;
50
- var $end_date;
51
- var $notes;
52
- var $rsvp;
53
- var $seats;
54
- var $contactperson_id;
55
- var $location_id;
56
- var $recurrence_id;
57
- var $category_id;
58
- var $attributes;
59
- var $recurrence;
60
- var $interval;
61
- var $freq;
62
- var $byday;
63
- var $byweekno;
64
- var $date_created;
65
- var $date_modified;
66
-
67
- /**
68
- * Timestamp of start date/time
69
- * @var int
70
- */
71
- var $start;
72
- /**
73
- * Timestamp of end date/time
74
- * @var int
75
- */
76
- var $end;
77
- /**
78
- * Created on timestamp, taken from DB, converted to TS
79
- * @var int
80
- */
81
- var $created;
82
- /**
83
- * Created on timestamp, taken from DB, converted to TS
84
- * @var int
85
- */
86
- var $modified;
87
-
88
- /**
89
- * @var EM_Location
90
- */
91
- var $location;
92
- /**
93
- * @var EM_Bookings
94
- */
95
- var $bookings;
96
- /**
97
- * The contact person for this event
98
- * @var WP_User
99
- */
100
- var $contact;
101
- /**
102
- * The category object
103
- * @var EM_Category
104
- */
105
- var $category;
106
- /**
107
- * If there are any errors, they will be added here.
108
- * @var array
109
- */
110
- var $errors = array();
111
- /**
112
- * If something was successful, a feedback message might be supplied here.
113
- * @var string
114
- */
115
- var $feedback_message;
116
- /**
117
- * Array of dbem_event field names required to create an event
118
- * @var array
119
- */
120
- var $required_fields = array('event_name', 'event_start_date');
121
-
122
- /**
123
- * Initialize an event. You can provide event data in an associative array (using database table field names), an id number, or false (default) to create empty event.
124
- * @param mixed $event_data
125
- * @param boolean $recurrent
126
- * @return null
127
- */
128
- function EM_Event($event_data = false, $recurrent = false) {
129
- global $wpdb, $EM_Recurrences;
130
- //TODO Change the way we deal with time, maybe revert to timestamps for manipulation, and worry about output in html and db writes?
131
- if( $event_data !== false ){
132
- $event = array();
133
- if( is_array($event_data) ){
134
- //Accepts a raw array that'll just be imported directly into the object with no DB lookups (same for event and recurrence)
135
- $event = $event_data;
136
- $this->location = new EM_Location( $event );
137
- $this->category = new EM_Category( $event );
138
- }elseif( is_numeric($event_data) && $event_data > 0 ){
139
- //Retreiving from the database
140
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
141
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
142
- $categories_table = $wpdb->prefix . EM_CATEGORIES_TABLE;
143
- $sql = "
144
- SELECT * FROM $events_table
145
- LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
146
- LEFT JOIN $categories_table ON {$categories_table}.category_id={$events_table}.event_category_id
147
- WHERE event_id = $event_data
148
- "; //We get event and location data here to avoid extra queries
149
- $event = $wpdb->get_row ( $sql, ARRAY_A );
150
- //Sort Location
151
- $this->location = new EM_Location ( $event );
152
- $this->category = new EM_Category( $event );
153
- }
154
- //Sort out attributes
155
- if( !empty($event['event_attributes']) ){
156
- if( is_serialized($event['event_attributes']) ){
157
- $event['event_attributes'] = @unserialize($event['event_attributes']);
158
- }
159
- $event['event_attributes'] = (!is_array($event['event_attributes'])) ? array() : $event['event_attributes'] ;
160
- }
161
- $event['recurrence_byday'] = ( $event['recurrence_byday'] == 7 ) ? 0:$event['recurrence_byday']; //Backward compatibility (since 3.0.3), using 0 makes more sense due to date() function
162
- $this->to_object($event, true);
163
-
164
- //Start/End times should be available as timestamp
165
- $this->start = strtotime($this->start_date." ".$this->start_time);
166
- $this->end = strtotime($this->end_date." ".$this->end_time);
167
- $this->modified = strtotime($event['event_date_modified']);
168
- $this->created = strtotime($event['event_date_created']);
169
-
170
- //Add Contact Person
171
- if($this->contactperson_id){
172
- if($this->contactperson_id > 0){
173
- $this->contact = get_userdata($this->contactperson_id);
174
- }
175
- }
176
- if( !is_object($this->contact) ){
177
- $this->contactperson_id = get_option('dbem_default_contact_person');
178
- $this->contact = get_userdata($this->contactperson_id);
179
- }
180
- if( is_object($this->contact) ){
181
- $this->contact->phone = get_metadata('user', $this->contact->ID, 'dbem_phone', true);
182
- }
183
- //Now, if this is a recurrence, get the recurring for caching to the $EM_Recurrences
184
- if( $this->is_recurrence() && !array_key_exists($this->recurrence_id, $EM_Recurrences) ){
185
- $EM_Recurrences[$this->recurrence_id] = new EM_Event($this->recurrence_id);
186
- }
187
- }else{
188
- $this->location = new EM_Location(); //blank location
189
- $this->category = new EM_Category(); //blank category
190
- }
191
- }
192
-
193
- /**
194
- * Retrieve event, location and recurring information via POST
195
- * @return boolean
196
- */
197
- function get_post(){
198
- //Build Event Array
199
- do_action('em_event_get_post_pre', $this);
200
- $this->name = ( !empty($_POST['event_name']) ) ? stripslashes($_POST['event_name']) : '' ;
201
- $this->start_date = ( !empty($_POST['event_start_date']) ) ? $_POST['event_start_date'] : '';
202
- $this->end_date = ( !empty($_POST['event_end_date']) ) ? $_POST['event_end_date'] : $this->start_date;
203
- $this->rsvp = ( !empty($_POST['event_rsvp']) ) ? 1:0;
204
- $this->seats = ( !empty($_POST['event_seats']) && is_numeric($_POST['event_seats']) ) ? $_POST['event_seats']:0;
205
- $this->notes = ( !empty($_POST['content']) ) ? stripslashes($_POST['content']) : ''; //WP TinyMCE field
206
- //Sort out time
207
- //TODO make time handling less painful
208
- $match = array();
209
- if( !empty($_POST['event_start_time']) && preg_match ( '/^([01]\d|2[0-3]):([0-5]\d)(AM|PM)?$/', $_POST['event_start_time'], $match ) ){
210
- if( $match[3] == 'PM' && $match[1] != 12 ){
211
- $match[1] = 12+$match[1];
212
- }elseif( $match[3] == 'AM' && $match[1] == 12 ){
213
- $match[1] = '00';
214
- }
215
- $this->start_time = $match[1].":".$match[2].":00";
216
- }else{
217
- $this->start_time = "00:00:00";
218
- }
219
- if( !empty($_POST['event_end_time']) && preg_match ( '/^([01]\d|2[0-3]):([0-5]\d)(AM|PM)?$/', $_POST['event_end_time'], $match ) ){
220
- if( $match[3] == 'PM' && $match[1] != 12 ){
221
- $match[1] = 12+$match[1];
222
- }elseif( $match[3] == 'AM' && $match[1] == 12 ){
223
- $match[1] = '00';
224
- }
225
- $this->end_time = $match[1].":".$match[2].":00";
226
- }else{
227
- $this->end_time = $this->start_time;
228
- }
229
- //Start/End times should be available as timestamp
230
- $this->start = strtotime($this->start_date." ".$this->start_time);
231
- $this->end = strtotime($this->end_date." ".$this->end_time);
232
- //Contact Person
233
- if ( !empty($_POST['event_contactperson_id']) && is_numeric($_POST['event_contactperson_id']) ) {
234
- //TODO contactperson choices needs limiting depending on role
235
- $this->contactperson_id = $_POST['event_contactperson_id'];
236
- }
237
- //category
238
- if( !empty($_POST['event_category_id']) && is_numeric($_POST['event_category_id']) ){
239
- $this->category_id = $_POST['event_category_id'];
240
- }
241
- //Attributes
242
- $event_attributes = array();
243
- for($i=1 ; !empty($_POST["mtm_{$i}_ref"]) && trim($_POST["mtm_{$i}_ref"]) != '' ; $i++ ){
244
- if( !empty($_POST["mtm_{$i}_name"]) && trim($_POST["mtm_{$i}_name"]) != '' ){
245
- $event_attributes[$_POST["mtm_{$i}_ref"]] = stripslashes($_POST["mtm_{$i}_name"]);
246
- }
247
- }
248
- $this->attributes = $event_attributes;
249
- //Recurrence data
250
- $this->recurrence_id = ( !empty($_POST['recurrence_id']) && is_numeric($_POST['recurrence_id']) ) ? $_POST['recurrence_id'] : 0 ;
251
- if( !empty($_POST['repeated_event']) ){
252
- $this->recurrence = 1;
253
- $this->freq = ( !empty($_POST['recurrence_freq']) && in_array($_POST['recurrence_freq'], array('daily','weekly','monthly')) ) ? $_POST['recurrence_freq']:'daily';
254
- if( !empty($_POST['recurrence_bydays']) && $this->freq == 'weekly' && self::array_is_numeric($_POST['recurrence_bydays']) ){
255
- $this->byday = implode ( ",", $_POST['recurrence_bydays'] );
256
- }elseif( !empty($_POST['recurrence_byday']) && $this->freq == 'monthly' ){
257
- $this->byday = $_POST['recurrence_byday'];
258
- }
259
- $this->interval = ( !empty($_POST['recurrence_interval']) ) ? $_POST['recurrence_interval']:1;
260
- $this->byweekno = ( !empty($_POST['recurrence_byweekno']) ) ? $_POST['recurrence_byweekno']:'';
261
- }
262
-
263
- //Add location information, or just link to previous location, this is a requirement...
264
- if( !empty($_POST['location-select-id']) ) {
265
- $this->location = new EM_Location($_POST['location-select-id']);
266
- } else {
267
- $this->location = new EM_Location($_POST);
268
- $this->location->load_similar($_POST);
269
- }
270
- return apply_filters('em_event_get_post', $this->validate(), $this);
271
- }
272
-
273
- /**
274
- * Will save the current instance into the database, along with location information if a new one was created and return true if successful, false if not.
275
- * Will automatically detect what type of event it is (recurrent, recurrence or normal) and whether it's a new or existing event.
276
- * @return boolean
277
- */
278
- function save(){
279
- //FIXME Event doesn't save title when inserting first time
280
- global $wpdb, $current_user;
281
- if( !$this->can_manage(true) ){
282
- return apply_filters('em_event_save', false, $this);
283
- }
284
- do_action('em_event_save_pre', $this);
285
- get_currentuserinfo();
286
- $events_table = $wpdb->prefix.EM_EVENTS_TABLE;
287
- //First let's save the location, no location no event!
288
- if ( !$this->location->id && !$this->location->save() ){ //shouldn't try to save if location exists
289
- $this->errors[] = __ ( 'There was a problem saving the location so event was not saved.', 'dbem' );
290
- return apply_filters('em_event_save', false, $this);
291
- }
292
- $this->location_id = $this->location->id;
293
- //Contact person can be anyone the admin wants, but the creator if not.
294
- if( em_verify_admin() ){
295
- $this->contactperson_id = ( $this->contactperson_id > 0 ) ? $this->contactperson_id:0;
296
- }else{
297
- //force
298
- $this->contactperson_id = get_current_user_id();
299
- }
300
- //Now save the event
301
- if ( !$this->id ) {
302
- // Insert New Event
303
- $this->author = $current_user->ID; //Record creator of event
304
- $event = $this->to_array(false, true);
305
- $event['event_attributes'] = serialize($this->attributes);
306
- $event['recurrence_id'] = ( is_numeric($this->recurrence_id) ) ? $this->recurrence_id : 0;
307
- $event = apply_filters('em_event_save_pre',$event,$this);
308
- $result = $wpdb->insert ( $events_table, $event, $this->get_types($event) );
309
- if($result !== false){
310
- $this->id = $wpdb->insert_id;
311
- //Deal with recurrences
312
- if ( $this->is_recurring() ) {
313
- //Recurrence master event saved, now Save Events & check errors
314
- if( !$this->save_events() ){
315
- $this->errors[] = __ ( 'Something went wrong with the recurrence update...', 'dbem' ).
316
- __ ( 'There was a problem saving the recurring events.', 'dbem' );
317
- $this->delete();
318
- return apply_filters('em_event_save', false, $this);
319
- }
320
- //All good! Event Saved
321
- $this->feedback_message = __ ( 'New recurrent event inserted!', 'dbem' );
322
- return apply_filters('em_event_save', true, $this);
323
- }
324
- //Successful individual save
325
- $this->feedback_message = __ ( 'New event successfully inserted!', 'dbem' );
326
- return apply_filters('em_event_save', true, $this);
327
- }else{
328
- $this->errors[] = __ ( 'Could not save the event details due to a database error.', 'dbem' );
329
- }
330
- } else {
331
- // Update Event
332
- //TODO event privacy protection, only authors and authorized users can edit events
333
- //$this->author = $current_user->ID; //Record creator of event
334
- //FIXME Saving recurrence and disabling recurrence doesn't work
335
- $this->recurrence_id = 0; // If it's saved here, it becomes individual
336
- $event = $this->to_array();
337
- $event['event_attributes'] = serialize($event['event_attributes']);
338
- $event['event_date_modified'] = current_time('mysql');
339
- $event = apply_filters('em_event_save_pre',$event,$this);
340
- $result = $wpdb->update ( $events_table, $event, array('event_id' => $this->id), $this->get_types($event) );
341
- if($result !== false){ //Can't just do $result since if you don't make an actual record details change, it'll return 0 for no changes made
342
- //Deal with recurrences
343
- if ( $this->is_recurring() ) {
344
- if( !$this->save_events() ){
345
- $this->errors[] = __ ( 'Something went wrong with the recurrence update...', 'dbem' ).
346
- __ ( 'There was a problem saving the recurring events.', 'dbem' );
347
- return apply_filters('em_event_save', false, $this);
348
- }
349
- $this->feedback_message = __ ( 'Recurrence updated!', 'dbem' );
350
- return apply_filters('em_event_save', true, $this);
351
- }
352
- }else{
353
- $this->errors[] = __('Could not save the event details due to a database error.', 'dbem');
354
- return apply_filters('em_event_save', false, $this);
355
- }
356
- //Successful individual or recurrence save
357
- $this->feedback_message = "{$this->name} " . __ ( 'updated', 'dbem' ) . "!";
358
- if($this->rsvp == 0){
359
- $this->delete_bookings();
360
- }
361
- return apply_filters('em_event_save', true, $this);
362
- }
363
- }
364
-
365
- /**
366
- * Delete whole event, including recurrence and recurring data
367
- * @param $recurrence_id
368
- * @return boolean
369
- */
370
- function delete(){
371
- global $wpdb;
372
- do_action('em_event_delete_pre', $this);
373
- $result = false;
374
- if( $this->can_manage(true) ){
375
- if( $this->is_recurring() ){
376
- //Delete the recurrences then this recurrence event
377
- $this->delete_events();
378
- }
379
- $result = $wpdb->query ( $wpdb->prepare("DELETE FROM ". $wpdb->prefix . EM_EVENTS_TABLE ." WHERE event_id=%d", $this->id) );
380
- if($result !== false){
381
- $result = $this->get_bookings()->delete();
382
- }
383
- }
384
- return apply_filters('em_event_delete', $result, $this);
385
- }
386
-
387
- /**
388
- * Duplicates this event and returns the duplicated event. Will return false if there is a problem with duplication.
389
- * @return EM_Event
390
- */
391
- function duplicate(){
392
- global $wpdb, $EZSQL_ERROR;
393
- //First, duplicate.
394
- if( $this->can_manage(true) ){
395
- $event_table_name = $wpdb->prefix . EM_EVENTS_TABLE;
396
- $eventArray = $this->to_array(true);
397
- unset($eventArray['event_id']);
398
- $EM_Event = new EM_Event( $eventArray );
399
- if( $EM_Event->save() ){
400
- $EM_Event->feedback_message = __("You are now viewing the duplicated event", 'dbem');
401
- return apply_filters('em_event_duplicate', $EM_Event, $this);
402
- }
403
- }
404
- //TODO add error notifications for duplication failures.
405
- return apply_filters('em_event_duplicate', false, $this);;
406
- }
407
-
408
-
409
- /**
410
- * Validates the event. Should be run during any form submission or saving operation.
411
- * @return boolean
412
- */
413
- function validate() {
414
- $missing_fields = Array ();
415
- foreach ( $this->required_fields as $field ) {
416
- $true_field = $this->fields[$field]['name'];
417
- if ( $this->$true_field == "") {
418
- $missing_fields[] = $field;
419
- }
420
- }
421
- if ( count($missing_fields) > 0){
422
- // TODO Create friendly equivelant names for missing fields notice in validation
423
- $this->errors[] = __ ( 'Missing fields: ' ) . implode ( ", ", $missing_fields ) . ". ";
424
- }
425
- if ( !empty($_POST['repeated_event']) && $_POST['repeated_event'] == "1" && $this->end_date == "" ){
426
- $this->errors[] = __ ( 'Since the event is repeated, you must specify an event date.', 'dbem' );
427
- }
428
- if( preg_match('/\d{4}-\d{2}-\d{2}/', $this->start_date) && preg_match('/\d{4}-\d{2}-\d{2}/', $this->end_date) ){
429
- if( strtotime($this->start_date . $this->start_time) > strtotime($this->end_date . $this->end_time) ){
430
- $this->errors[] = __('Events cannot start after they end.','dbem');
431
- }
432
- }else{
433
- $this->errors[] = __('Dates must have correct formatting. Please use the date picker provided.','dbem');
434
- }
435
- if( !$this->location->validate() ){
436
- $this->errors = array_merge($this->errors, $this->location->errors);
437
- }
438
- //TODO validate recurrence during event validate
439
- return apply_filters('em_event_validate', count($this->errors) == 0, $this );
440
- }
441
-
442
-
443
- /**
444
- * Returns an array with category id and name (in that order) of the EM_Event instance.
445
- * @return array
446
- */
447
- function get_category() {
448
- global $EM_Category;
449
- if( is_object($this->category) && get_class($this->category)=='EM_Category' && $this->category_id == $this->category->id ){
450
- return $this->category;
451
- }elseif( is_object($EM_Category) && $EM_Category->id == $this->category_id ){
452
- $this->category = $EM_Category;
453
- }else{
454
- $this->category = new EM_Category($this->category_id);
455
- }
456
- return $this->event;
457
- global $wpdb;
458
- $sql = "SELECT category_id, category_name FROM ".$wpdb->prefix.EM_EVENTS_TABLE." LEFT JOIN ".$wpdb->prefix.EM_CATEGORIES_TABLE." ON category_id=event_category_id WHERE event_id ='".$this->id."'";
459
- $category = $wpdb->get_row($sql, ARRAY_A);
460
- return apply_filters('em_event_get_category', $category, $this);
461
- }
462
-
463
- /**
464
- * Shortcut function for $this->get_bookings()->delete(), because using the EM_Bookings requires loading previous bookings, which isn't neceesary.
465
- */
466
- function delete_bookings(){
467
- global $wpdb;
468
- do_action('em_event_delete_bookings_pre', $this);
469
- $result = false;
470
- if( $this->can_manage(true) ){
471
- $result = $wpdb->query( $wpdb->prepare("DELETE FROM ".$wpdb->prefix.EM_BOOKINGS_TABLE." WHERE event_id=%d", $this->id) );
472
- }
473
- return apply_filters('em_event_delete_bookings', $result, $this);
474
- }
475
-
476
- /**
477
- * Retrieve and save the bookings belonging to instance. If called again will return cached version, set $force_reload to true to create a new EM_Bookings object.
478
- * @param boolean $force_reload
479
- * @return EM_Bookings
480
- */
481
- function get_bookings( $force_reload = false ){
482
- if( get_option('dbem_rsvp_enabled') ){
483
- if( (!$this->bookings || $force_reload) ){
484
- $this->bookings = new EM_Bookings($this);
485
- }
486
- }
487
- return apply_filters('em_event_get_bookings', $this->bookings, $this);
488
- }
489
-
490
- /**
491
- * Will output a single event format of this event.
492
- * Equivalent of calling EM_Event::output( get_option ( 'dbem_single_event_format' ) )
493
- * @param string $target
494
- * @return string
495
- */
496
- function output_single($target='html'){
497
- $format = get_option ( 'dbem_single_event_format' );
498
- return apply_filters('em_event_output_single', $this->output($format, $target), $this, $target);
499
- }
500
-
501
- /**
502
- * Will output a event list item format of this event.
503
- * Equivalent of calling EM_Event::output( get_option ( 'dbem_event_list_item_format' ) )
504
- * @param string $target
505
- * @return string
506
- */
507
- function output_list($target='html'){
508
- $format = get_option ( 'dbem_event_list_item_format' );
509
- return apply_filters('em_event_output_list', $this->output($format, $target), $this, $target);
510
- }
511
-
512
- /**
513
- * Will output a event in the format passed in $format by replacing placeholders within the format.
514
- * @param string $format
515
- * @param string $target
516
- * @return string
517
- */
518
- function output($format, $target="html") {
519
- $event_string = $format;
520
- preg_match_all("/#@?_?[A-Za-z0-9]+/", $format, $placeholders);
521
- foreach($placeholders[0] as $result) {
522
- $match = true;
523
- $replace = '';
524
- switch( $result ){
525
- //Event Details
526
- case '#_EVENTID':
527
- $replace = $this->id;
528
- break;
529
- case '#_NAME':
530
- $replace = $this->name;
531
- break;
532
- case '#_NOTES':
533
- case '#_EXCERPT':
534
- //SEE AT BOTTOM OF FILE FOR OLD TARGET FILTERS FROM 2.x
535
- $replace = $this->notes;
536
- if($result == "#_EXCERPT"){
537
- $matches = explode('<!--more', $this->notes);
538
- $replace = $matches[0];
539
- }
540
- break;
541
- //Times
542
- case '#_24HSTARTTIME':
543
- case '#_24HENDTIME':
544
- $time = ($result == '#_24HSTARTTIME') ? $this->start_time:$this->end_time;
545
- $replace = substr($time, 0,5);
546
- break;
547
- case '#_12HSTARTTIME':
548
- case '#_12HENDTIME':
549
- $time = ($result == '#_12HSTARTTIME') ? $this->start_time:$this->end_time;
550
- $replace = date('g:i A', strtotime($time));
551
- break;
552
- //Links
553
- case '#_EVENTPAGEURL': //Depreciated
554
- case '#_LINKEDNAME': //Depreciated
555
- case '#_EVENTURL': //Just the URL
556
- case '#_EVENTLINK': //HTML Link
557
- $joiner = (stristr(EM_URI, "?")) ? "&amp;" : "?";
558
- $event_link = EM_URI.$joiner."event_id=".$this->id;
559
- if($result == '#_LINKEDNAME' || $result == '#_EVENTLINK'){
560
- $replace = "<a href='{$event_link}' title='{$this->name}'>{$this->name}</a>";
561
- }else{
562
- $replace = $event_link;
563
- }
564
- break;
565
- case '#_EDITEVENTLINK':
566
- if( $this->can_manage() ){
567
- //TODO user should have permission to edit the event
568
- $replace = "<a href='".get_bloginfo('wpurl')."/wp-admin/admin.php?page=events-manager-event&amp;event_id={$this->id}'>".__('Edit').' '.__('Event', 'dbem')."</a>";
569
- }
570
- break;
571
- //Bookings
572
- case '#_ADDBOOKINGFORM':
573
- case '#_REMOVEBOOKINGFORM':
574
- case '#_BOOKINGFORM':
575
- if ($this->rsvp && get_option('dbem_rsvp_enabled')){
576
- if($result == '#_BOOKINGFORM'){
577
- $replace = em_add_booking_form().em_delete_booking_form();
578
- }else{
579
- $replace = ($result == '#_ADDBOOKINGFORM') ? em_add_booking_form():em_delete_booking_form();
580
- }
581
- }
582
- break;
583
- case '#_AVAILABLESEATS': //Depreciated
584
- case '#_AVAILABLESPACES':
585
- if ($this->rsvp && get_option('dbem_rsvp_enabled')) {
586
- $replace = $this->get_bookings()->get_available_seats();
587
- } else {
588
- $replace = "0";
589
- }
590
- break;
591
- case '#_BOOKEDSEATS': //Depreciated
592
- case '#_BOOKEDSPACES':
593
- if ($this->rsvp && get_option('dbem_rsvp_enabled')) {
594
- $replace = $this->get_bookings()->get_booked_seats();
595
- } else {
596
- $replace = "0";
597
- }
598
- break;
599
- case '#_SEATS': //Depreciated
600
- case '#_SPACES':
601
- $replace = $this->seats;
602
- break;
603
- //Contact Person
604
- case '#_CONTACTNAME':
605
- case '#_CONTACTPERSON': //Depreciated (your call, I think name is better)
606
- $replace = $this->contact->display_name;
607
- break;
608
- case '#_CONTACTUSERNAME':
609
- $replace = $this->contact->user_login;
610
- break;
611
- case '#_CONTACTEMAIL':
612
- case '#_CONTACTMAIL': //Depreciated
613
- $replace = $this->contact->user_email;
614
- break;
615
- case '#_CONTACTID':
616
- $replace = $this->contact->ID;
617
- break;
618
- case '#_CONTACTPHONE':
619
- $replace = ( $this->contact->phone != '') ? $this->contact->phone : __('N/A', 'dbem');
620
- break;
621
- case '#_CONTACTAVATAR':
622
- $replace = get_avatar( $this->contact->ID, $size = '50' );
623
- break;
624
- case '#_CONTACTPROFILELINK':
625
- case '#_CONTACTPROFILEURL':
626
- if( function_exists('bp_core_get_user_domain') ){
627
- $replace = bp_core_get_user_domain($this->contact->ID);
628
- if( $result == '#_CONTACTPROFILELINK' ){
629
- $replace = '<a href="'.$replace.'">'.__('Profile').'</a>';
630
- }
631
- }
632
- break;
633
- default:
634
- $replace = $result;
635
- break;
636
- }
637
- $replace = apply_filters('em_event_output_placeholder', $replace, $this, $result, $target);
638
- $event_string = str_replace($result, $replace , $event_string );
639
- }
640
- //Time placeholders
641
- foreach($placeholders[0] as $result) {
642
- // matches all PHP START date and time placeholders
643
- if (preg_match('/^#[dDjlNSwzWFmMntLoYyaABgGhHisueIOPTZcrU]$/', $result)) {
644
- $replace = date_i18n(ltrim($result, "#"), $this->start);
645
- $replace = apply_filters('em_event_output_placeholder', $replace, $this, $result, $target);
646
- $event_string = str_replace($result, $replace, $event_string );
647
- }
648
- // matches all PHP END time placeholders for endtime
649
- if (preg_match('/^#@[dDjlNSwzWFmMntLoYyaABgGhHisueIOPTZcrU]$/', $result)) {
650
- $replace = date_i18n(ltrim($result, "#@"), $this->end);
651
- $replace = apply_filters('em_event_output_placeholder', $replace, $this, $result, $target);
652
- $event_string = str_replace($result, $replace, $event_string );
653
- }
654
- }
655
- //Time place holder that doesn't show if empty.
656
- //TODO add filter here too
657
- preg_match_all('/#@?_\{[A-Za-z0-9 -\/,\.\\\]+\}/', $format, $results);
658
- foreach($results[0] as $result) {
659
- if(substr($result, 0, 3 ) == "#@_"){
660
- $date = 'end_date';
661
- $offset = 4;
662
- }else{
663
- $date = 'start_date';
664
- $offset = 3;
665
- }
666
- if( $date == 'end_date' && $this->end_date == $this->start_date ){
667
- $replace = __( apply_filters('em_event_output_placeholder', '', $this, $result, $target) );
668
- }else{
669
- $replace = __( apply_filters('em_event_output_placeholder', mysql2date(substr($result, $offset, (strlen($result)-($offset+1)) ), $this->$date), $this, $result, $target) );
670
- }
671
- $event_string = str_replace($result,$replace,$event_string );
672
- }
673
- //This is for the custom attributes
674
- preg_match_all('/#_ATT\{.+?\}(\{(.+)\})?/', $format, $results);
675
- foreach($results[0] as $resultKey => $result) {
676
- //Strip string of placeholder and just leave the reference
677
- $attRef = substr( substr($result, 0, strpos($result, '}')), 6 );
678
- $attString = '';
679
- if( is_array($this->attributes) && array_key_exists($attRef, $this->attributes) ){
680
- $attString = $this->attributes[$attRef];
681
- }elseif( !empty($results[2][$resultKey]) ){
682
- //Check to see if we have a second set of braces;
683
- $attString = $results[2][$resultKey];
684
- }
685
- $attString = apply_filters('em_event_output_placeholder', $attString, $this, $result, $target);
686
- $event_string = str_replace($result, $attString ,$event_string );
687
- }
688
-
689
- //Now do dependent objects
690
- $event_string = $this->location->output($event_string, $target);
691
- $event_string = $this->category->output($event_string, $target);
692
- return apply_filters('em_event_output', $event_string, $this, $target);
693
- }
694
-
695
- /**********************************************************
696
- * RECURRENCE METHODS
697
- ***********************************************************/
698
-
699
- /**
700
- * Saves events and replaces old ones. Returns true if sucecssful or false if not.
701
- * @return boolean
702
- */
703
- function save_events() {
704
- if( $this->is_recurring() && $this->can_manage() ){
705
- do_action('em_event_save_events_pre', $this); //actions/filters only run if event is recurring
706
- global $wpdb;
707
- $event_saves = array();
708
- $matching_days = $this->get_recurrence_days(); //Get days where events recur
709
- $this->delete_events(); //Delete old events beforehand
710
- //Make template event (and we just change dates)
711
- $event = $this->to_array();
712
- unset($event['event_id']); //remove id and we have a event template to feed to wpdb insert
713
- $event['event_attributes'] = serialize($event['event_attributes']);
714
- foreach($event as $key => $value ){ //remove recurrence information
715
- if( substr($key, 0, 10) == 'recurrence' ){
716
- unset($event[$key]);
717
- }
718
- }
719
- $event['recurrence_id'] = $this->id;
720
- //Save event template with different dates
721
- foreach( $matching_days as $day ) {
722
- $event['event_start_date'] = date("Y-m-d", $day);
723
- $event['event_end_date'] = $event['event_start_date'];
724
- $event_saves[] = $wpdb->insert($wpdb->prefix.EM_EVENTS_TABLE, $event, $this->get_types($event));
725
- //TODO should be EM_DEBUG, and do we really need it?
726
- if( DEBUG ){ echo "Entering recurrence " . date("D d M Y", $day)."<br/>"; }
727
- }
728
- return apply_filters('em_event_save_events', !in_array(false, $event_saves), $this);
729
- }
730
- return apply_filters('em_event_save_events', false, $this);;
731
- }
732
-
733
- /**
734
- * Removes all reoccurring events.
735
- * @param $recurrence_id
736
- * @return null
737
- */
738
- function delete_events(){
739
- global $wpdb;
740
- do_action('em_event_delete_events_pre', $this);
741
- //So we don't do something we'll regret later, we could just supply the get directly into the delete, but this is safer
742
- $result = false;
743
- if( $this->can_manage(true) ){
744
- $EM_Events = EM_Events::get( array('recurrence_id'=>$this->id) );
745
- $event_ids = array();
746
- foreach($EM_Events as $EM_Event){
747
- if($EM_Event->recurrence_id == $this->id){
748
- $event_ids[] = $EM_Event->id; //ONLY ADD if id's match - hard coded
749
- }
750
- }
751
- $result = EM_Events::delete( $event_ids );
752
- }
753
- return apply_filters('delete_events', $result, $this);
754
- }
755
-
756
- /**
757
- * Returns true if this event is a recurring event, meaning that it's not an individual event,
758
- * but an event that defines many events that recur over a span of time.
759
- * For checking if a specific event is part of a greater set of recurring events, use is_recurrence()
760
- * @return boolean
761
- */
762
- function is_recurring(){
763
- return ( $this->recurrence );
764
- }
765
- /**
766
- * Will return true if this individual event is part of a set of events that recur
767
- * For checking if this is the "master recurring event", see is_recurring()
768
- * @return boolean
769
- */
770
- function is_recurrence(){
771
- return ( $this->id > 0 && $this->recurrence_id > 0 );
772
- }
773
- /**
774
- * Returns if this is an individual event and is not recurring or a recurrence
775
- * @return boolean
776
- */
777
- function is_individual(){
778
- return ( !$this->is_recurring() && !$this->is_recurrence() );
779
- }
780
-
781
- /**
782
- * Can the user manage this event?
783
- */
784
- function can_manage( $error_msg = false ){
785
- $can_manage = ( get_option('dbem_permissions_events') || $this->author == get_current_user_id() || empty($this->id) || em_verify_admin() );
786
- if($error_msg && !$can_manage){
787
- $this->errors[] = __('You do not have permission to manage this event.','dbem');
788
- }
789
- return apply_filters('em_event_can_manage', $can_manage, $this);
790
- }
791
-
792
- /**
793
- * Returns the days that match the recurrance array passed (unix timestamps)
794
- * @param array $recurrence
795
- * @return array
796
- */
797
- function get_recurrence_days(){
798
- if( $this->is_recurring() ){
799
-
800
- $start_date = strtotime($this->start_date);
801
- $end_date = strtotime($this->end_date);
802
-
803
- $weekdays = explode(",", $this->byday); //what days of the week (or if monthly, one value at index 0)
804
-
805
- $matching_days = array();
806
- $aDay = 86400; // a day in seconds
807
- $aWeek = $aDay * 7;
808
-
809
- //TODO can this be optimized?
810
- switch ( $this->freq ){
811
- case 'daily':
812
- //If daily, it's simple. Get start date, add interval timestamps to that and create matching day for each interval until end date.
813
- $current_date = $start_date;
814
- while( $current_date <= $end_date ){
815
- $matching_days[] = $current_date;
816
- $current_date = $current_date + ($aDay * $this->interval);
817
- }
818
- break;
819
- case 'weekly':
820
- //sort out week one, get starting days and then days that match time span of event (i.e. remove past events in week 1)
821
- $start_of_week = get_option('start_of_week'); //Start of week depends on wordpress
822
- //first, get the start of this week as timestamp
823
- $event_start_day = date('w', $start_date);
824
- $offset = 0;
825
- if( $event_start_day > $start_of_week ){
826
- $offset = $event_start_day - $start_of_week; //x days backwards
827
- }elseif( $event_start_day < $start_of_week ){
828
- $offset = $start_of_week;
829
- }
830
- $start_week_date = $start_date - ( ($event_start_day - $start_of_week) * $aDay );
831
- //then get the timestamps of weekdays during this first week, regardless if within event range
832
- $start_weekday_dates = array(); //Days in week 1 where there would events, regardless of event date range
833
- for($i = 0; $i < 7; $i++){
834
- $weekday_date = $start_week_date+($aDay*$i); //the date of the weekday we're currently checking
835
- $weekday_day = date('w',$weekday_date); //the day of the week we're checking, taking into account wp start of week setting
836
- if( in_array( $weekday_day, $weekdays) ){
837
- $start_weekday_dates[] = $weekday_date; //it's in our starting week day, so add it
838
- }
839
- }
840
- //for each day of eventful days in week 1, add 7 days * weekly intervals
841
- foreach ($start_weekday_dates as $weekday_date){
842
- //Loop weeks by interval until we reach or surpass end date
843
- while($weekday_date <= $end_date){
844
- if( $weekday_date >= $start_date && $weekday_date <= $end_date ){
845
- $matching_days[] = $weekday_date;
846
- }
847
- $weekday_date = $weekday_date + ($aWeek * $this->interval);
848
- }
849
- }//done!
850
- break;
851
- case 'monthly':
852
- //loop months starting this month by intervals
853
- $current_arr = getdate($start_date);
854
- $end_arr = getdate($end_date);
855
- $end_month_date = strtotime( date('Y-m-t', $end_date) ); //End date on last day of month
856
- $current_date = strtotime( date('Y-m-1', $start_date) ); //Start date on first day of month
857
- while( $current_date <= $end_month_date ){
858
- $last_day_of_month = date('t', $current_date);
859
- //Now find which day we're talking about
860
- $current_week_day = date('w',$current_date);
861
- $matching_month_days = array();
862
- //Loop through days of this years month and save matching days to temp array
863
- for($day = 1; $day <= $last_day_of_month; $day++){
864
- if($current_week_day == $this->byday){
865
- $matching_month_days[] = $day;
866
- }
867
- $current_week_day = ($current_week_day < 6) ? $current_week_day+1 : 0;
868
- }
869
- //Now grab from the array the x day of the month
870
- $matching_day = ($this->byweekno > 0) ? $matching_month_days[$this->byweekno-1] : array_pop($matching_month_days);
871
- $matching_date = strtotime(date('Y-m',$current_date).'-'.$matching_day);
872
- if($matching_date >= $start_date && $matching_date <= $end_date){
873
- $matching_days[] = $matching_date;
874
- }
875
- //add the number of days in this month to make start of next month
876
- $current_arr['mon'] += $this->interval;
877
- if($current_arr['mon'] > 12){
878
- //FIXME this won't work if interval is more than 12
879
- $current_arr['mon'] = $current_arr['mon'] - 12;
880
- $current_arr['year']++;
881
- }
882
- $current_date = strtotime("{$current_arr['year']}-{$current_arr['mon']}-1");
883
- }
884
- break;
885
- }
886
- sort($matching_days);
887
- //TODO delete this after testing
888
- /*Delete*/
889
- $test_dates = array();
890
- foreach($matching_days as $matching_day){
891
- $test_dates[] = date('d/m/Y', $matching_day);
892
- }
893
- /*end delete*/
894
- return $matching_days;
895
- }
896
- }
897
-
898
- /**
899
- * Returns a string representation of this recurrence. Will return false if not a recurrence
900
- * @return string
901
- */
902
- function get_recurrence_description() {
903
- //FIXME Recurrence description not working for recurrence
904
- global $EM_Recurrences;
905
- if( $this->is_individual() ) return false;
906
- $recurrence = $EM_Recurrences[$this->recurrence_id]->to_array();
907
- $weekdays_name = array(__('Sunday'),__('Monday'),__('Tuesday'),__('Wednesday'),__('Thursday'),__('Friday'),__('Saturday'));
908
- $monthweek_name = array('1' => __('the first %s of the month', 'dbem'),'2' => __('the second %s of the month', 'dbem'), '3' => __('the third %s of the month', 'dbem'), '4' => __('the fourth %s of the month', 'dbem'), '-1' => __('the last %s of the month', 'dbem'));
909
- $output = sprintf (__('From %1$s to %2$s', 'dbem'), $recurrence['event_start_date'], $recurrence['event_end_date']).", ";
910
- if ($recurrence['recurrence_freq'] == 'daily') {
911
- $freq_desc =__('everyday', 'dbem');
912
- if ($recurrence['recurrence_interval'] > 1 ) {
913
- $freq_desc = sprintf (__("every %s days", 'dbem'), $recurrence['recurrence_interval']);
914
- }
915
- }
916
- if ($recurrence['recurrence_freq'] == 'weekly') {
917
- $weekday_array = explode(",", $recurrence['recurrence_byday']);
918
- $natural_days = array();
919
- foreach($weekday_array as $day){
920
- array_push($natural_days, $weekdays_name[$day]);
921
- }
922
- $output .= implode(" and ", $natural_days);
923
- $freq_desc = ", " . __("every week", 'dbem');
924
- if ($recurrence['recurrence_interval'] > 1 ) {
925
- $freq_desc = ", ".sprintf (__("every %s weeks", 'dbem'), $recurrence['recurrence_interval']);
926
- }
927
-
928
- }
929
- if ($recurrence['recurrence_freq'] == 'monthly') {
930
- $weekday_array = explode(",", $recurrence['recurrence_byday']);
931
- $natural_days = array();
932
- foreach($weekday_array as $day){
933
- array_push($natural_days, $weekdays_name[$day]);
934
- }
935
- $freq_desc = sprintf (($monthweek_name[$recurrence['recurrence_byweekno']]), implode(" and ", $natural_days));
936
- if ($recurrence['recurrence_interval'] > 1 ) {
937
- $freq_desc .= ", ".sprintf (__("every %s months",'dbem'), $recurrence['recurrence_interval']);
938
- }
939
-
940
- }
941
- $output .= $freq_desc;
942
- return $output;
943
- }
944
-
945
- /**********************************************************
946
- * UTILITIES
947
- ***********************************************************/
948
-
949
- /**
950
- * Returns this object in the form of an array, useful for saving directly into the wp_dbem_events table.
951
- * @param boolean $location
952
- * @param boolean $for_database
953
- * @return array
954
- */
955
- function to_array($location = false, $for_database = false){
956
- $event = array();
957
- //Core Event Data
958
- foreach ( $this->fields as $key => $val ) {
959
- //TODO does it matter if it's for db or not... shouldn't it just not include blanks?
960
- if( !$for_database || $for_database && $this->$val['name'] ){
961
- $event[$key] = $this->$val['name'];
962
- }
963
- }
964
- //Location Data
965
- if($location && is_object($this->location)){
966
- $location = $this->location->to_array();
967
- $event = array_merge($event, $location);
968
- }
969
- return $event;
970
- }
971
- }
972
-
973
- //TODO placeholder targets filtering could be streamlined better
974
- /**
975
- * This is a temporary filter function which mimicks the old filters in the old 2.x placeholders function
976
- * @param string $result
977
- * @param EM_Event $event
978
- * @param string $placeholder
979
- * @param string $target
980
- * @return mixed
981
- */
982
- function em_event_output_placeholder($result,$event,$placeholder,$target='html'){
983
- if( ($placeholder == "#_EXCERPT" || $placeholder == "#_LOCATIONEXCERPT") && $target == 'html' ){
984
- $result = apply_filters('dbem_notes_excerpt', $result);
985
- }elseif( $placeholder == '#_CONTACTEMAIL' && $target == 'html' ){
986
- $result = em_ascii_encode($event->contact->user_email);
987
- }elseif( $placeholder == "#_NOTES" || $placeholder == "#_EXCERPT" || $placeholder == "#_LOCATIONEXCERPT" ){
988
- if($target == 'html'){
989
- $result = apply_filters('dbem_notes', $result);
990
- }elseif($target == 'map'){
991
- $result = apply_filters('dbem_notes_map', $result);
992
- }else{
993
- $result = apply_filters('dbem_notes_rss', $result);
994
- $result = apply_filters('the_content_rss', $result);
995
- }
996
- }elseif( in_array($placeholder, array("#_NAME",'#_ADDRESS','#_LOCATION','#_TOWN')) ){
997
- if ($target == "html"){
998
- $result = apply_filters('dbem_general', $result);
999
- }else{
1000
- $result = apply_filters('dbem_general_rss', $result);
1001
- }
1002
- }
1003
- return $result;
1004
- }
1005
- add_filter('em_event_output_placeholder','em_event_output_placeholder',1,4);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1006
  ?>
1
+ <?php
2
+ /**
3
+ * Get an event in a db friendly way, by checking globals and passed variables to avoid extra class instantiations
4
+ * @param mixed $id
5
+ * @param mixed $search_by
6
+ * @return EM_Event
7
+ */
8
+ function em_get_event($id = false, $search_by = 'event_id') {
9
+ global $EM_Event;
10
+ //check if it's not already global so we don't instantiate again
11
+ if( is_object($EM_Event) && get_class($EM_Event) == 'EM_Event' ){
12
+ if( $search_by == 'event_id' && $EM_Event->event_id == $id ){
13
+ return $EM_Event;
14
+ }elseif( $search_by == 'post_id' && $EM_Event->post_id == $id ){
15
+ return $EM_Event;
16
+ }elseif( is_object($id) && $EM_Event->post_id == $id->ID ){
17
+ return $EM_Event;
18
+ }
19
+ }
20
+ if( is_object($id) && get_class($id) == 'EM_Event' ){
21
+ return $id;
22
+ }else{
23
+ return new EM_Event($id,$search_by);
24
+ }
25
+ }
26
+ /**
27
+ * Event Object. This holds all the info pertaining to an event, including location and recurrence info.
28
+ * An event object can be one of three "types" a recurring event, recurrence of a recurring event, or a single event.
29
+ * The single event might be part of a set of recurring events, but if loaded by specific event id then any operations and saves are
30
+ * specifically done on this event. However, if you edit the recurring group, any changes made to single events are overwritten.
31
+ *
32
+ * @author marcus
33
+ */
34
+ //TODO Can add more recurring functionality such as "also update all future recurring events" or "edit all events" like google calendar does.
35
+ //TODO Integrate recurrences into events table
36
+ //FIXME If you create a super long recurrence timespan, there could be thousands of events... need an upper limit here.
37
+ class EM_Event extends EM_Object{
38
+ /* Field Names */
39
+ var $event_id;
40
+ var $event_slug;
41
+ var $event_owner;
42
+ var $event_name;
43
+ var $event_start_time;
44
+ var $event_end_time;
45
+ var $event_all_day;
46
+ var $event_start_date;
47
+ var $event_end_date;
48
+ var $post_content;
49
+ var $event_rsvp;
50
+ //var $spaces;
51
+ var $location_id;
52
+ var $recurrence_id;
53
+ var $event_status;
54
+ var $event_date_created;
55
+ var $event_date_modified;
56
+ var $blog_id;
57
+ var $group_id;
58
+ /**
59
+ * Populated with the non-hidden event post custom fields (i.e. not starting with _)
60
+ * @var array
61
+ */
62
+ var $event_attributes = array();
63
+ /* Recurring Specific Values */
64
+ var $recurrence;
65
+ var $recurrence_interval;
66
+ var $recurrence_freq;
67
+ var $recurrence_byday;
68
+ var $recurrence_days;
69
+ var $recurrence_byweekno;
70
+ /**
71
+ * Previously used to give this object shorter property names for db values (each key has a name) but this is now depreciated, use the db field names as properties. This propertey provides extra info about the db fields.
72
+ * @var array
73
+ */
74
+ var $fields = array(
75
+ 'event_id' => array( 'name'=>'id', 'type'=>'%d' ),
76
+ 'post_id' => array( 'name'=>'post_id', 'type'=>'%d' ),
77
+ 'event_slug' => array( 'name'=>'slug', 'type'=>'%s', 'null'=>true ),
78
+ 'event_owner' => array( 'name'=>'owner', 'type'=>'%d', 'null'=>true ),
79
+ 'event_name' => array( 'name'=>'name', 'type'=>'%s', 'null'=>true ),
80
+ 'event_start_time' => array( 'name'=>'start_time', 'type'=>'%s', 'null'=>true ),
81
+ 'event_end_time' => array( 'name'=>'end_time', 'type'=>'%s', 'null'=>true ),
82
+ 'event_all_day' => array( 'name'=>'all_day', 'type'=>'%d', 'null'=>true ),
83
+ 'event_start_date' => array( 'name'=>'start_date', 'type'=>'%s', 'null'=>true ),
84
+ 'event_end_date' => array( 'name'=>'end_date', 'type'=>'%s', 'null'=>true ),
85
+ 'post_content' => array( 'name'=>'notes', 'type'=>'%s', 'null'=>true ),
86
+ 'event_rsvp' => array( 'name'=>'rsvp', 'type'=>'%d', 'null'=>true ), //has a default, so can be null/excluded
87
+ //'event_spaces' => array( 'name'=>'spaces', 'type'=>'%d' ),
88
+ 'location_id' => array( 'name'=>'location_id', 'type'=>'%d', 'null'=>true ),
89
+ 'recurrence_id' => array( 'name'=>'recurrence_id', 'type'=>'%d', 'null'=>true ),
90
+ 'event_status' => array( 'name'=>'status', 'type'=>'%d', 'null'=>true ),
91
+ 'event_private' => array( 'name'=>'status', 'type'=>'%d', 'null'=>true ),
92
+ 'event_date_created' => array( 'name'=>'date_created', 'type'=>'%s', 'null'=>true ),
93
+ 'event_date_modified' => array( 'name'=>'date_modified', 'type'=>'%s', 'null'=>true ),
94
+ 'event_attributes' => array( 'name'=>'attributes', 'type'=>'%s', 'null'=>true ),
95
+ 'blog_id' => array( 'name'=>'blog_id', 'type'=>'%d', 'null'=>true ),
96
+ 'group_id' => array( 'name'=>'group_id', 'type'=>'%d', 'null'=>true ),
97
+ 'recurrence' => array( 'name'=>'recurrence', 'type'=>'%d', 'null'=>true ), //every x day(s)/week(s)/month(s)
98
+ 'recurrence_interval' => array( 'name'=>'interval', 'type'=>'%d', 'null'=>true ), //every x day(s)/week(s)/month(s)
99
+ 'recurrence_freq' => array( 'name'=>'freq', 'type'=>'%s', 'null'=>true ), //daily,weekly,monthly?
100
+ 'recurrence_days' => array( 'name'=>'days', 'type'=>'%d', 'null'=>true ), //daily,weekly,monthly?
101
+ 'recurrence_byday' => array( 'name'=>'byday', 'type'=>'%s', 'null'=>true ), //if weekly or monthly, what days of the week?
102
+ 'recurrence_byweekno' => array( 'name'=>'byweekno', 'type'=>'%d', 'null'=>true ), //if monthly which week (-1 is last)
103
+ );
104
+ var $post_fields = array('event_slug','event_owner','event_name','event_attributes','post_id','post_content'); //fields that won't be taken from the em_events table anymore
105
+ var $recurrence_fields = array('recurrence_interval', 'recurrence_freq', 'recurrence_days', 'recurrence_byday', 'recurrence_byweekno');
106
+
107
+ var $image_url = '';
108
+ /**
109
+ * Timestamp of start date/time
110
+ * @var int
111
+ */
112
+ var $start;
113
+ /**
114
+ * Timestamp of end date/time
115
+ * @var int
116
+ */
117
+ var $end;
118
+ /**
119
+ * Created on timestamp, taken from DB, converted to TS
120
+ * @var int
121
+ */
122
+ var $created;
123
+ /**
124
+ * Created on timestamp, taken from DB, converted to TS
125
+ * @var int
126
+ */
127
+ var $modified;
128
+
129
+ /**
130
+ * @var EM_Location
131
+ */
132
+ var $location;
133
+ /**
134
+ * @var EM_Bookings
135
+ */
136
+ var $bookings;
137
+ /**
138
+ * The contact person for this event
139
+ * @var WP_User
140
+ */
141
+ var $contact;
142
+ /**
143
+ * The category object
144
+ * @var EM_Category
145
+ */
146
+ var $category;
147
+ /**
148
+ * If there are any errors, they will be added here.
149
+ * @var array
150
+ */
151
+ var $errors = array();
152
+ /**
153
+ * If something was successful, a feedback message might be supplied here.
154
+ * @var string
155
+ */
156
+ var $feedback_message;
157
+ /**
158
+ * Any warnings about an event (e.g. bad data, recurrence, etc.)
159
+ * @var string
160
+ */
161
+ var $warnings;
162
+ /**
163
+ * Array of dbem_event field names required to create an event
164
+ * @var array
165
+ */
166
+ var $required_fields = array('event_name', 'event_start_date');
167
+ var $mime_types = array(1 => 'gif', 2 => 'jpg', 3 => 'png');
168
+ /**
169
+ * previous status of event when instantiated
170
+ * @access protected
171
+ * @var mixed
172
+ */
173
+ var $previous_status = 0;
174
+
175
+ /* Post Variables - copied out of post object for easy IDE reference */
176
+ var $ID;
177
+ var $post_author;
178
+ var $post_date;
179
+ var $post_date_gmt;
180
+ var $post_title;
181
+ var $post_excerpt;
182
+ var $post_status;
183
+ var $comment_status;
184
+ var $ping_status;
185
+ var $post_password;
186
+ var $post_name;
187
+ var $to_ping;
188
+ var $pinged;
189
+ var $post_modified;
190
+ var $post_modified_gmt;
191
+ var $post_content_filtered;
192
+ var $post_parent;
193
+ var $guid;
194
+ var $menu_order;
195
+ var $post_type;
196
+ var $post_mime_type;
197
+ var $comment_count;
198
+ var $ancestors;
199
+ var $filter;
200
+
201
+ /**
202
+ * Initialize an event. You can provide event data in an associative array (using database table field names), an id number, or false (default) to create empty event.
203
+ * @param mixed $event_data
204
+ * @param mixed $search_by default is post_id, otherwise it can be by event_id as well.
205
+ * @return null
206
+ */
207
+ function __construct($id = false, $search_by = 'event_id') {
208
+ global $wpdb;
209
+ if( is_array($id) ){
210
+ //deal with the old array style, but we can't supply arrays anymore
211
+ $id = (!empty($id['event_id'])) ? $id['event_id'] : $id['post_id'];
212
+ $search_by = (!empty($id['event_id'])) ? 'event_id':'post_id';
213
+ }
214
+ $is_post = !empty($id->ID) && ($id->post_type == EM_POST_TYPE_EVENT || $id->post_type == 'event-recurring');
215
+ if( is_numeric($id) || $is_post ){ //only load info if $id is a number
216
+ if($search_by == 'event_id' && !$is_post ){
217
+ //search by event_id, get post_id and blog_id (if in ms mode) and load the post
218
+ $results = $wpdb->get_row($wpdb->prepare("SELECT post_id, blog_id FROM ".EM_EVENTS_TABLE." WHERE event_id=%d",$id), ARRAY_A);
219
+ if( is_multisite() && is_numeric($results['blog_id']) ){
220
+ $event_post = get_blog_post($results['blog_id'], $results['post_id']);
221
+ $search_by = $results['blog_id'];
222
+ }else{
223
+ $event_post = get_post($results['post_id']);
224
+ }
225
+ }else{
226
+ if(!$is_post){
227
+ if( is_numeric($search_by) && is_multisite() ){
228
+ //we've been given a blog_id, so we're searching for a post id
229
+ $event_post = get_blog_post($search_by, $id);
230
+ }else{
231
+ //search for the post id only
232
+ $event_post = get_post($id);
233
+ }
234
+ }else{
235
+ $event_post = $id;
236
+ }
237
+ }
238
+ $this->load_postdata($event_post, $search_by);
239
+ }
240
+ $this->recurrence = $this->is_recurring() ? 1:0;
241
+ //if(defined('trashtest')){ print_r($this); die("got here");}
242
+ //Do it here so things appear in the po file.
243
+ $this->status_array = array(
244
+ 0 => __('Pending','dbem'),
245
+ 1 => __('Approved','dbem')
246
+ );
247
+ do_action('em_event', $this, $id, $search_by);
248
+ }
249
+
250
+ function load_postdata($event_post, $search_by = false){
251
+ if( is_object($event_post) ){
252
+ if( $event_post->post_status != 'auto-draft' ){
253
+ if( is_numeric($search_by) && is_multisite() ){
254
+ // if in multisite mode, switch blogs quickly to get the right post meta.
255
+ switch_to_blog($search_by);
256
+ $event_meta = get_post_custom($event_post->ID);
257
+ restore_current_blog();
258
+ $this->blog_id = $search_by;
259
+ }else{
260
+ $event_meta = get_post_custom($event_post->ID);
261
+ }
262
+ //Get custom fields and post meta
263
+ foreach($event_meta as $event_meta_key => $event_meta_val){
264
+ if($event_meta_key[0] != '_'){
265
+ $this->event_attributes[$event_meta_key] = ( count($event_meta_val) > 1 ) ? $event_meta_val:$event_meta_val[0];
266
+ }else{
267
+ foreach($this->fields as $field_name => $field_info){
268
+ if( $event_meta_key == '_'.$field_name && $event_meta_key != '_event_attributes'){
269
+ $this->$field_name = $event_meta_val[0];
270
+ }
271
+ }
272
+ }
273
+ }
274
+ //Start/End times should be available as timestamp
275
+ $this->start = strtotime($this->event_start_date." ".$this->event_start_time);
276
+ $this->end = strtotime($this->event_end_date." ".$this->event_end_time);
277
+ //quick compatability fix in case _event_id isn't loaded or somehow got erased in post meta
278
+ if( empty($this->event_id) && !$this->is_recurring() ){
279
+ global $wpdb;
280
+ $event_array = $wpdb->get_row('SELECT * FROM '.EM_EVENTS_TABLE. ' WHERE post_id='.$event_post->ID, ARRAY_A);
281
+ if( !empty($event_array['event_id']) ){
282
+ foreach($event_array as $key => $value){
283
+ if( !empty($value) && empty($this->$key) ){
284
+ update_post_meta($event_post->ID, '_'.$key, $value);
285
+ $this->$key = $value;
286
+ }
287
+ }
288
+ }
289
+ }
290
+ }
291
+ //load post data - regardless
292
+ $this->post_id = $event_post->ID;
293
+ $this->event_name = $event_post->post_title;
294
+ $this->event_owner = $event_post->post_author;
295
+ $this->post_content = $event_post->post_content;
296
+ $this->event_slug = $event_post->post_name;
297
+ $this->event_modified = $event_post->post_modified;
298
+ foreach( $event_post as $key => $value ){ //merge post object into this object
299
+ $this->$key = $value;
300
+ }
301
+ $this->previous_status = $this->event_status; //so we know about updates
302
+ $this->recurrence = $this->is_recurring() ? 1:0;
303
+ $this->get_status();
304
+ $this->compat_keys();
305
+ }
306
+ }
307
+
308
+ /**
309
+ * Retrieve event information via POST (only used in situations where posts aren't submitted via WP)
310
+ * @return boolean
311
+ */
312
+ function get_post($validate = true){
313
+ global $allowedposttags;
314
+ //we need to get the post/event name and content.... that's it.
315
+ $this->post_content = !empty($_POST['content']) ? wp_kses($_POST['content'], $allowedposttags):'';
316
+ $this->event_name = !empty($_POST['event_name']) ? wp_kses($_POST['event_name'], array()):'';
317
+ $this->post_type = ($this->is_recurring() || !empty($_POST['recurring'])) ? 'event-recurring':EM_POST_TYPE_EVENT;
318
+ //don't forget categories!
319
+ $this->get_categories()->get_post();
320
+ $this->get_post_meta(false);
321
+ $result = $validate ? $this->validate():true; //validate both post and meta, otherwise return true
322
+ return apply_filters('em_event_get_post', $result, $this);
323
+ }
324
+
325
+ /**
326
+ * Retrieve event post meta information via POST, which should be always be called when saving the event custom post via WP.
327
+ * @param boolean $validate whether or not to run validation, default is true
328
+ * @return boolean
329
+ */
330
+ function get_post_meta($validate = true){
331
+ //Grab POST data
332
+ $this->event_start_date = ( !empty($_POST['event_start_date']) ) ? $_POST['event_start_date'] : '';
333
+ $this->event_end_date = ( !empty($_POST['event_end_date']) ) ? $_POST['event_end_date'] : $this->event_start_date;
334
+ //check if this is recurring or not
335
+ if( $_REQUEST['recurring'] ){
336
+ $this->recurrence = 1;
337
+ $this->post_type = 'event-recurring';
338
+ }
339
+ //Get Location info
340
+ if( !get_option('dbem_locations_enabled') || (!empty($_POST['no_location']) && !get_option('dbem_require_location',true)) || (empty($_POST['location_id']) && !get_option('dbem_require_location',true) && get_option('dbem_use_select_for_locations')) ){
341
+ $this->location_id = 0;
342
+ }elseif( !empty($_POST['location_id']) && is_numeric($_POST['location_id']) ){
343
+ $this->location_id = $_POST['location_id'];
344
+ }else{
345
+ //we're adding a new location, so create an empty location and populate
346
+ $this->location_id = null;
347
+ $this->get_location()->get_post(false);
348
+ }
349
+ //Sort out time
350
+ $this->event_all_day = ( !empty($_POST['event_all_day']) ) ? 1 : 0;
351
+ if( !$this->event_all_day ){
352
+ $match = array();
353
+ foreach( array('event_start_time','event_end_time') as $timeName ){
354
+ if( !empty($_POST[$timeName]) && preg_match ( '/^([01]\d|2[0-3]):([0-5]\d) ?(AM|PM)?$/', $_POST[$timeName], $match ) ){
355
+ if( !empty($match[3]) && $match[3] == 'PM' && $match[1] != 12 ){
356
+ $match[1] = 12+$match[1];
357
+ }elseif( !empty($match[3]) && $match[3] == 'AM' && $match[1] == 12 ){
358
+ $match[1] = '00';
359
+ }
360
+ $this->$timeName = $match[1].":".$match[2].":00";
361
+ }else{
362
+ $this->$timeName = ($timeName == 'event_start_time') ? "00:00:00":$this->event_start_time;
363
+ }
364
+ }
365
+ }else{
366
+ $this->event_start_time = $this->event_end_time = '00:00:00';
367
+ }
368
+ //Start/End times should be available as timestamp
369
+ $this->start = strtotime($this->event_start_date." ".$this->event_start_time);
370
+ $this->end = strtotime($this->event_end_date." ".$this->event_end_time);
371
+ //Bookings
372
+ if( !empty($_REQUEST['event_rsvp']) && $_REQUEST['event_rsvp'] ){
373
+ $this->get_bookings()->get_tickets()->get_post();
374
+ $this->event_rsvp = 1;
375
+ }else{
376
+ $this->event_rsvp = 0;
377
+ }
378
+ //Sort out event attributes - note that custom post meta now also gets inserted here automatically (and is overwritten by these attributes)
379
+ if(get_option('dbem_attributes_enabled')){
380
+ global $allowedtags;
381
+ if( !is_array($this->event_attributes) ){ $this->event_attributes = array(); }
382
+ $event_available_attributes = em_get_attributes();
383
+ if( !empty($_POST['em_attributes']) && is_array($_POST['em_attributes']) ){
384
+ foreach($_POST['em_attributes'] as $att_key => $att_value ){
385
+ if( (in_array($att_key, $event_available_attributes['names']) || array_key_exists($att_key, $this->event_attributes) ) && trim($att_value) != '' ){
386
+ $att_vals = count($event_available_attributes['values'][$att_key]);
387
+ if( $att_vals == 0 || ($att_vals > 0 && in_array($att_value, $event_available_attributes['values'][$att_key])) ){
388
+ $this->event_attributes[$att_key] = stripslashes($att_value);
389
+ }elseif($att_vals > 0){
390
+ $this->event_attributes[$att_key] = stripslashes(wp_kses($event_available_attributes['values'][$att_key][0], $allowedtags));
391
+ }
392
+ }
393
+ }
394
+ }
395
+ }
396
+ //Set Blog ID
397
+ if( is_multisite() ){
398
+ $this->blog_id = get_current_blog_id();
399
+ }
400
+ //group id
401
+ $this->group_id = (!empty($_POST['group_id']) && is_numeric($_POST['group_id'])) ? $_POST['group_id']:$this->group_id;
402
+ //Recurrence data
403
+ if( $this->is_recurring() ){
404
+ $this->recurrence = 1; //just in case
405
+ $this->recurrence_freq = ( !empty($_REQUEST['recurrence_freq']) && in_array($_REQUEST['recurrence_freq'], array('daily','weekly','monthly')) ) ? $_REQUEST['recurrence_freq']:'daily';
406
+ if( !empty($_REQUEST['recurrence_bydays']) && $this->recurrence_freq == 'weekly' && self::array_is_numeric($_REQUEST['recurrence_bydays']) ){
407
+ $this->recurrence_byday = implode( ",", $_REQUEST['recurrence_bydays'] );
408
+ }elseif( !empty($_REQUEST['recurrence_byday']) && $this->recurrence_freq == 'monthly' ){
409
+ $this->recurrence_byday = $_REQUEST['recurrence_byday'];
410
+ }
411
+ $this->recurrence_interval = ( !empty($_REQUEST['recurrence_interval']) && is_numeric($_REQUEST['recurrence_interval']) ) ? $_REQUEST['recurrence_interval']:1;
412
+ $this->recurrence_byweekno = ( !empty($_REQUEST['recurrence_byweekno']) ) ? $_REQUEST['recurrence_byweekno']:'';
413
+ $this->recurrence_days = ( !empty($_REQUEST['recurrence_days']) && is_numeric($_REQUEST['recurrence_days']) ) ? $_REQUEST['recurrence_days']:1;
414
+ }
415
+ //categories in MS GLobal
416
+ if(EM_MS_GLOBAL && !is_main_site()){
417
+ $this->get_categories()->get_post(); //it'll know what to do
418
+ }
419
+ $result = $validate ? $this->validate_meta():true;
420
+ $this->compat_keys(); //compatability
421
+ return apply_filters('em_event_get_post', $result, $this);
422
+ }
423
+
424
+ function validate(){
425
+ $validate_post = true;
426
+ if( empty($this->event_name) ){
427
+ $validate_post = false;
428
+ $this->add_error( __('Event name').__(" is required.", "dbem") );
429
+ }
430
+ $validate_meta = $this->validate_meta();
431
+ return apply_filters('em_event_validate', $validate_post && $validate_meta, $this );
432
+ }
433
+ function validate_meta(){
434
+ $missing_fields = Array ();
435
+ foreach ( array('event_start_date') as $field ) {
436
+ if ( $this->$field == "") {
437
+ $missing_fields[$field] = $field;
438
+ }
439
+ }
440
+ if( preg_match('/\d{4}-\d{2}-\d{2}/', $this->event_start_date) && preg_match('/\d{4}-\d{2}-\d{2}/', $this->event_end_date) ){
441
+ if( strtotime($this->event_start_date . $this->event_start_time) > strtotime($this->event_end_date . $this->event_end_time) ){
442
+ $this->add_error(__('Events cannot start after they end.','dbem'));
443
+ }
444
+ }else{
445
+ if( !empty($missing_fields['event_start_date']) ) { unset($missing_fields['event_start_date']); }
446
+ if( !empty($missing_fields['event_end_date']) ) { unset($missing_fields['event_end_date']); }
447
+ $this->add_error(__('Dates must have correct formatting. Please use the date picker provided.','dbem'));
448
+ }
449
+ if( $this->event_rsvp && !$this->get_bookings()->get_tickets()->validate() ){
450
+ $this->add_error($this->get_bookings()->get_tickets()->get_errors());
451
+ }
452
+ if( get_option('dbem_locations_enabled') && empty($this->location_id) ){ //location ids don't need validating as we're not saving a location
453
+ if( get_option('dbem_require_location',true) || $this->location_id !== 0 ){
454
+ if( !$this->get_location()->validate() ){
455
+ $this->add_error($this->get_location()->get_errors());
456
+ }
457
+ }
458
+ }
459
+ if ( count($missing_fields) > 0){
460
+ // TODO Create friendly equivelant names for missing fields notice in validation
461
+ $this->add_error( __( 'Missing fields: ', 'dbem') . implode ( ", ", $missing_fields ) . ". " );
462
+ }
463
+ if ( $this->is_recurring() && ($this->event_end_date == "" || $this->event_end_date == $this->event_start_date) ){
464
+ $this->add_error( __( 'Since the event is repeated, you must specify an event end date greater than the start date.', 'dbem' ));
465
+ }
466
+ return apply_filters('em_event_validate_meta', count($this->errors) == 0, $this );
467
+ }
468
+
469
+ /**
470
+ * Will save the current instance into the database, along with location information if a new one was created and return true if successful, false if not.
471
+ * Will automatically detect whether it's a new or existing event.
472
+ * @return boolean
473
+ */
474
+ function save(){
475
+ global $wpdb, $current_user, $blog_id;
476
+ if( !$this->can_manage('edit_events', 'edit_others_events') && !( get_option('dbem_events_anonymous_submissions') && empty($this->event_id)) ){
477
+ //unless events can be submitted by an anonymous user (and this is a new event), user must have permissions.
478
+ return apply_filters('em_event_save', false, $this);
479
+ }
480
+ remove_action('save_post',array('EM_Event_Post_Admin','save_post'),10,1); //disable the default save post action, we'll do it manually this way
481
+ do_action('em_event_save_pre', $this);
482
+ $post_array = array();
483
+ //Deal with updates to an event
484
+ if( !empty($this->post_id) ){
485
+ //get the full array of post data so we don't overwrite anything.
486
+ if( !empty($this->blog_id) && is_multisite() ){
487
+ $post_array = (array) get_blog_post($this->blog_id, $this->post_id);
488
+ }else{
489
+ $post_array = (array) get_post($this->post_id);
490
+ }
491
+ }
492
+ //Overwrite new post info
493
+ $post_array['post_type'] = ($this->recurrence && get_option('dbem_recurrence_enabled')) ? 'event-recurring':EM_POST_TYPE_EVENT;
494
+ $post_array['post_title'] = $this->event_name;
495
+ $post_array['post_content'] = $this->post_content;
496
+ //decide on post status
497
+ if( count($this->errors) == 0 ){
498
+ $post_array['post_status'] = ( $this->can_manage('publish_events','publish_events') ) ? 'publish':'pending';
499
+ }else{
500
+ $post_array['post_status'] = 'draft';
501
+ }
502
+ //anonymous submission only
503
+ if( !is_user_logged_in() && get_option('dbem_events_anonymous_submissions') && empty($this->event_id) ){
504
+ $post_array['post_author'] = get_option('dbem_events_anonymous_user');
505
+ if( !is_numeric($post_array['post_author']) ) $post_array['post_author'] = 0;
506
+ }
507
+ //Save post and continue with meta
508
+ $post_id = wp_insert_post($post_array);
509
+ $post_save = false;
510
+ $meta_save = false;
511
+ if( !is_wp_error($post_id) && !empty($post_id) ){
512
+ $post_save = true;
513
+ //refresh this event with wp post info we'll put into the db
514
+ $post_data = get_post($post_id);
515
+ $this->post_id = $post_id;
516
+ $this->event_slug = $post_data->post_name;
517
+ $this->event_owner = $post_data->post_author;
518
+ $this->post_status = $post_data->post_status;
519
+ $this->get_status();
520
+ //Categories? note that categories will soft-fail, so no errors
521
+ $this->get_categories()->save();
522
+ //now save the meta
523
+ $meta_save = $this->save_meta();
524
+ //save the image
525
+ $this->image_upload();
526
+ $image_save = (count($this->errors) == 0); //whilst it might not be an image save that fails, we can know something went wrong
527
+ }
528
+ $result = $meta_save && $post_save && $image_save;
529
+ $previous_status = $this->previous_status;
530
+ if($result) $this->load_postdata($post_data, $blog_id); //reload post info
531
+ $this->previous_status = $previous_status;
532
+ return apply_filters('em_event_save', $result, $this);
533
+ }
534
+
535
+ function save_meta(){
536
+ global $wpdb;
537
+ if( ( get_option('dbem_events_anonymous_submissions') && empty($this->event_id)) || $this->can_manage('edit_events', 'edit_others_events') ){
538
+ do_action('em_event_save_meta_pre', $this);
539
+ //first save location
540
+ if( empty($this->location_id) && !($this->location_id === 0 && !get_option('dbem_require_location',true)) ){
541
+ if( get_site_option('dbem_ms_mainblog_locations') ){ $this->ms_global_switch(); }
542
+ if( !$this->get_location()->save() ){ //soft fail
543
+ global $EM_Notices;
544
+ $this->get_location()->set_status(null);
545
+ if( !empty($this->get_location()->location_id) ){
546
+ $EM_Notices->add_error( __('There were some errors saving your location.','dbem').' '.sprintf(__('It will not be displayed on the website listings, to correct this you must <a href="%s">edit your location</a> directly.'),$this->get_location()->output('#_LOCATIONEDITURL')), true);
547
+ }else{
548
+ $EM_Notices->add_error( sprintf(__('There were some errors saving your location.'),$this->get_location()->output('#_LOCATIONEDITURL')), true);
549
+ }
550
+ }
551
+ if( get_site_option('dbem_ms_mainblog_locations') ){ $this->ms_global_switch_back(); }
552
+ if( !empty($this->location->location_id) ){ //only case we don't use get_location(), since it will fail as location has an id, whereas location_id isn't set in this object
553
+ $this->location_id = $this->location->location_id;
554
+ }
555
+ }
556
+ //Update Post Meta
557
+ foreach($this->fields as $key => $field_info){
558
+ if( !in_array($key, $this->post_fields) && $key != 'event_attributes' ){
559
+ update_post_meta($this->post_id, '_'.$key, $this->$key);
560
+ }elseif($key == 'event_attributes'){
561
+ //attributes get saved as individual keys
562
+ foreach($this->event_attributes as $event_attribute_key => $event_attribute){
563
+ update_post_meta($this->post_id, $event_attribute_key, $event_attribute);
564
+ }
565
+ }
566
+ }
567
+ update_post_meta($this->post_id, '_start_ts', strtotime($this->event_start_date));
568
+ update_post_meta($this->post_id, '_end_ts', strtotime($this->event_end_date));
569
+
570
+ $result = count($this->errors) == 0;
571
+ $this->get_status();
572
+ $this->event_status = ($result) ? $this->event_status:null; //set status at this point, it's either the current status, or if validation fails, null
573
+ //Save to em_event table
574
+ $event_array = $this->to_array(true);
575
+ unset($event_array['event_id']);
576
+ if( $this->post_status == 'private' ) $event_array['event_private'] = 1;
577
+ $event_array['event_attributes'] = serialize($this->event_attributes); //might as well
578
+ if( empty($this->event_id) ){
579
+ $this->previous_status = 0; //for sure this was previously status 0
580
+ $this->event_date_created = current_time('mysql');
581
+ if ( !$wpdb->insert(EM_EVENTS_TABLE, $event_array) ){
582
+ $this->add_error( sprintf(__('Something went wrong saving your %s to the index table. Please inform a site administrator about this.','dbem'),__('event','dbem')));
583
+ }else{
584
+ //success, so link the event with the post via an event id meta value for easy retrieval
585
+ $this->event_id = $wpdb->insert_id;
586
+ update_post_meta($this->post_id, '_event_id', $this->event_id);
587
+ $this->feedback_message = sprintf(__('Successfully saved %s','dbem'),__('Event','dbem'));
588
+ $just_added_event = true; //make an easy hook
589
+ }
590
+ }else{
591
+ $this->previous_status = $this->get_previous_status();
592
+ $this->event_date_modified = current_time('mysql');
593
+ if ( $wpdb->update(EM_EVENTS_TABLE, $event_array, array('event_id'=>$this->event_id) ) === false ){
594
+ $this->add_error( sprintf(__('Something went wrong updating your %s to the index table. Please inform a site administrator about this.','dbem'),__('event','dbem')));
595
+ }else{
596
+ $this->feedback_message = sprintf(__('Successfully saved %s','dbem'),__('Event','dbem'));
597
+ }
598
+ }
599
+ //Add/Delete Tickets
600
+ if($this->event_rsvp == 0){
601
+ $this->get_bookings()->delete();
602
+ }else{
603
+ if( !$this->get_bookings()->get_tickets()->save() ){
604
+ $this->add_error( $this->get_bookings()->get_tickets()->get_errors() );
605
+ }
606
+ }
607
+ $result = count($this->errors) == 0;
608
+ //If we're saving event categories in MS Global mode, we'll add them here, saving by term id (cat ids are gone now)
609
+ if( EM_MS_GLOBAL && !is_main_site() ){
610
+ $this->get_categories()->save(); //it'll know what to do
611
+ }elseif( EM_MS_GLOBAL ){
612
+ $this->get_categories()->save_index(); //just save to index, we assume cats are saved in $this->save();
613
+ }
614
+ //build recurrences if needed
615
+ if( $this->is_recurring() && $result && $this->is_published() ){ //only save events if recurring event validates and is published
616
+ if( !$this->save_events() ){ //only save if post is 'published'
617
+ $this->add_error(__ ( 'Something went wrong with the recurrence update...', 'dbem' ). __ ( 'There was a problem saving the recurring events.', 'dbem' ));
618
+ }
619
+ }
620
+ if( !empty($just_added_event) ){
621
+ do_action('em_event_added', $this);
622
+ }
623
+ }
624
+ $this->compat_keys();
625
+ return apply_filters('em_event_save_meta', count($this->errors) == 0, $this);
626
+ }
627
+
628
+ /**
629
+ * Duplicates this event and returns the duplicated event. Will return false if there is a problem with duplication.
630
+ * @return EM_Event
631
+ */
632
+ function duplicate(){
633
+ global $wpdb, $EZSQL_ERROR;
634
+ //First, duplicate.
635
+ if( $this->can_manage('edit_events','edit_others_events') ){
636
+ $EM_Event = clone $this;
637
+ $EM_Event->event_id = null;
638
+ //Duplicate Post
639
+ $fields = $wpdb->get_col("DESCRIBE ".$wpdb->posts);
640
+ unset($fields[0]);
641
+ $fields = implode(',',$fields);
642
+ $sql = "INSERT INTO {$wpdb->posts} ($fields) SELECT $fields FROM {$wpdb->posts} WHERE ID={$this->post_id}";
643
+ $result = $wpdb->query($sql);
644
+ $EM_Event->post_id = $EM_Event->ID = $wpdb->insert_id;
645
+ //Duplicate Event Table index and tickets
646
+ if( $EM_Event->save_meta() ){
647
+ //duplicate tickets
648
+ $EM_Tickets = $this->get_bookings()->get_tickets();
649
+ foreach($EM_Tickets->tickets as $EM_Ticket){
650
+ $EM_Ticket->ticket_id = '';
651
+ $EM_Ticket->event_id = $EM_Event->event_id;
652
+ $EM_Ticket->save();
653
+ }
654
+ $EM_Event->get_bookings(true); //refresh booking tickets
655
+ $EM_Event->feedback_message = sprintf(__("%s successfully duplicated.", 'dbem'), __('Event','dbem'));
656
+ return apply_filters('em_event_duplicate', $EM_Event, $this);
657
+ }
658
+ }
659
+ //TODO add error notifications for duplication failures.
660
+ return apply_filters('em_event_duplicate', false, $this);;
661
+ }
662
+
663
+ /**
664
+ * Delete whole event, including bookings, tickets, etc.
665
+ * @return boolean
666
+ */
667
+ function delete($force_delete = true){ //atm wp seems to force cp deletions anyway
668
+ global $wpdb;
669
+ if( $this->can_manage('delete_events', 'delete_others_events') ){
670
+ remove_action('before_delete_post',array('EM_Event_Post_Admin','before_delete_post'),10,1); //since we're deleting directly, remove post actions
671
+ do_action('em_event_delete_pre', $this);
672
+ $result = wp_delete_post($this->post_id,$force_delete);
673
+ if( $force_delete ){
674
+ $result_meta = $this->delete_meta();
675
+ }
676
+ }else{
677
+ $result = $result_meta = false;
678
+ }
679
+ //print_r($result); echo "|"; print_r($result_meta); die('DELETING');
680
+ return apply_filters('em_event_delete', $result !== false && $result_meta, $this);
681
+ }
682
+
683
+ function delete_meta(){
684
+ global $wpdb;
685
+ $result = false;
686
+ if( $this->can_manage('delete_events', 'delete_others_events') ){
687
+ do_action('em_event_delete_meta_event_pre', $this);
688
+ $result = $wpdb->query ( $wpdb->prepare("DELETE FROM ". EM_EVENTS_TABLE ." WHERE event_id=%d", $this->event_id) );
689
+ if( $result !== false ){
690
+ $this->delete_bookings();
691
+ $this->delete_tickets();
692
+ //Delete the recurrences then this recurrence event
693
+ if( $this->is_recurring() ){
694
+ $result = $this->delete_events(); //was true at this point, so false if fails
695
+ }
696
+ //Delete categories from meta if in MS global mode
697
+ if( EM_MS_GLOBAL ){
698
+ $wpdb->query('DELETE FROM '.EM_META_TABLE.' WHERE object_id='.$this->event_id." AND meta_key='event-category'");
699
+ }
700
+ }
701
+ }
702
+ return apply_filters('em_event_delete_meta', $result !== false, $this);
703
+ }
704
+
705
+ /**
706
+ * Shortcut function for $this->get_bookings()->delete(), because using the EM_Bookings requires loading previous bookings, which isn't neceesary.
707
+ */
708
+ function delete_bookings(){
709
+ global $wpdb;
710
+ do_action('em_event_delete_bookings_pre', $this);
711
+ $result = false;
712
+ if( $this->can_manage('manage_bookings','manage_others_bookings') ){
713
+ $result_bt = $wpdb->query( $wpdb->prepare("DELETE FROM ".EM_TICKETS_BOOKINGS_TABLE." WHERE booking_id IN (SELECT booking_id FROM ".EM_BOOKINGS_TABLE." WHERE event_id=%d)", $this->event_id) );
714
+ $result = $wpdb->query( $wpdb->prepare("DELETE FROM ".EM_BOOKINGS_TABLE." WHERE event_id=%d", $this->event_id) );
715
+ }
716
+ return apply_filters('em_event_delete_bookings', $result !== false && $result_bt !== false, $this);
717
+ }
718
+
719
+ /**
720
+ * Shortcut function for $this->get_bookings()->delete(), because using the EM_Bookings requires loading previous bookings, which isn't neceesary.
721
+ */
722
+ function delete_tickets(){
723
+ global $wpdb;
724
+ do_action('em_event_delete_tickets_pre', $this);
725
+ $result = false;
726
+ if( $this->can_manage('manage_bookings','manage_others_bookings') ){
727
+ $result_bt = $wpdb->query( $wpdb->prepare("DELETE FROM ".EM_TICKETS_BOOKINGS_TABLE." WHERE ticket_id IN (SELECT ticket_id FROM ".EM_TICKETS_TABLE." WHERE event_id=%d)", $this->event_id) );
728
+ $result = $wpdb->query( $wpdb->prepare("DELETE FROM ".EM_TICKETS_TABLE." WHERE event_id=%d", $this->event_id) );
729
+ }
730
+ return apply_filters('em_event_delete_tickets', $result, $this);
731
+ }
732
+
733
+ /**
734
+ * Publish Events
735
+ * @return bool
736
+ */
737
+ function send_approval_notification(){
738
+ if( (!$this->is_recurring() && !user_can($this->event_owner, 'publish_events')) || ($this->is_recurring() && !user_can($this->event_owner, 'publish_recurring_events')) ){
739
+ //only send email to users that can't publish events themselves and that were previously unpublished
740
+ if( !$this->previous_status && $this->is_published() ){
741
+ //email
742
+ if( $this->event_owner == "" ) return true;
743
+ $subject = $this->output(get_option('dbem_event_approved_email_subject'), 'email');
744
+ $body = $this->output(get_option('dbem_event_approved_email_body'), 'email');
745
+
746
+ //Send to the person booking
747
+ if( !$this->email_send( $subject, $body, $this->get_contact()->user_email) ){
748
+ return false;
749
+ }
750
+ return true;
751
+ }
752
+ }
753
+ return false;
754
+ }
755
+
756
+ /**
757
+ * Change the status of the event. This will save to the Database too.
758
+ * @param int $status
759
+ * @param boolean $set_post_status
760
+ * @return string
761
+ */
762
+ function set_status($status, $set_post_status = false){
763
+ global $wpdb;
764
+ if($status === null){
765
+ $set_status='NULL';
766
+ if($set_post_status){
767
+ //if the post is trash, don't untrash it!
768
+ $wpdb->update( $wpdb->posts, array( 'post_status' => 'draft' ), array( 'ID' => $this->post_id ) );
769
+ $this->post_status = 'draft';
770
+ }
771
+ }else{
772
+ $set_status = $status ? 1:0;
773
+ if($set_post_status){
774
+ if($this->post_status == 'pending'){
775
+ $this->post_name = sanitize_title($this->post_title);
776
+ }
777
+ $this->post_status = $set_status ? 'publish':'pending';
778
+ $wpdb->update( $wpdb->posts, array( 'post_status' => $this->post_status, 'post_name' => $this->post_name ), array( 'ID' => $this->post_id ) );
779
+ }
780
+ }
781
+ $this->previous_status = $this->get_previous_status();
782
+ $result = $wpdb->query("UPDATE ".EM_EVENTS_TABLE." SET event_status=$set_status, event_slug='{$this->post_name}' WHERE event_id=".$this->event_id);
783
+ $this->get_status(); //reload status
784
+ return apply_filters('em_event_set_status', $result !== false, $status, $this);
785
+ }
786
+
787
+ function is_published(){
788
+ return apply_filters('em_event_is_published', ($this->post_status == 'publish' || $this->post_status == 'private'), $this);
789
+ }
790
+
791
+ function get_status($db = false){
792
+ switch( $this->post_status ){
793
+ case 'private':
794
+ $this->event_private = 1;
795
+ $this->event_status = $status = 1;
796
+ break;
797
+ case 'publish':
798
+ $this->event_private = 0;
799
+ $this->event_status = $status = 1;
800
+ break;
801
+ case 'pending':
802
+ $this->event_private = 0;
803
+ $this->event_status = $status = 0;
804
+ break;
805
+ default: //draft or unknown
806
+ $this->event_private = 0;
807
+ $status = $db ? 'NULL':null;
808
+ $this->event_status = null;
809
+ break;
810
+ }
811
+ return $status;
812
+ }
813
+
814
+ function get_previous_status(){
815
+ global $wpdb;
816
+ return $wpdb->get_var('SELECT event_status FROM '.EM_EVENTS_TABLE.' WHERE event_id='.$this->event_id); //get status from db, not post_status, as posts get saved quickly
817
+ }
818
+
819
+ /**
820
+ * Returns an EM_Categories object of the EM_Event instance.
821
+ * @return EM_Categories
822
+ */
823
+ function get_categories() {
824
+ if( empty($this->categories) ){
825
+ $this->categories = new EM_Categories($this);
826
+ }elseif(empty($this->categories->event_id)){
827
+ $this->categories->event_id = $this->event_id;
828
+ $this->categories->post_id = $this->post_id;
829
+ }
830
+ return apply_filters('em_event_get_categories', $this->categories, $this);
831
+ }
832
+
833
+ /**
834
+ * Returns the location object this event belongs to.
835
+ * @return EM_Location
836
+ */
837
+ function get_location() {
838
+ global $EM_Location;
839
+ if( is_object($EM_Location) && $EM_Location->location_id == $this->location_id ){
840
+ $this->location = $EM_Location;
841
+ }else{
842
+ if( !is_object($this->location) || $this->location->location_id != $this->location_id ){
843
+ $this->location = em_get_location($this->location_id);
844
+ }
845
+ }
846
+ return $this->location;
847
+ }
848
+
849
+ /**
850
+ * Returns the location object this event belongs to.
851
+ * @return EM_Person
852
+ */
853
+ function get_contact(){
854
+ if( !is_object($this->contact) ){
855
+ $this->contact = new EM_Person($this->event_owner);
856
+ }
857
+ return $this->contact;
858
+ }
859
+
860
+ /**
861
+ * Retrieve and save the bookings belonging to instance. If called again will return cached version, set $force_reload to true to create a new EM_Bookings object.
862
+ * @param boolean $force_reload
863
+ * @return EM_Bookings
864
+ */
865
+ function get_bookings( $force_reload = false ){
866
+ if( get_option('dbem_rsvp_enabled') ){
867
+ if( (!$this->bookings || $force_reload) ){
868
+ $this->bookings = new EM_Bookings($this);
869
+ }
870
+ $this->bookings->event_id = $this->event_id; //always refresh event_id
871
+ }else{
872
+ return new EM_Bookings();
873
+ }
874
+ return apply_filters('em_event_get_bookings', $this->bookings, $this);
875
+ }
876
+
877
+ /**
878
+ * Get the tickets related to this event.
879
+ * @param boolean $force_reload
880
+ * @return EM_Tickets
881
+ */
882
+ function get_tickets( $force_reload = false ){
883
+ return $this->get_bookings($force_reload)->get_tickets();
884
+ }
885
+
886
+ /**
887
+ * Gets number of spaces in this event, dependent on ticket spaces or hard limit, whichever is smaller.
888
+ * @param boolean $force_refresh
889
+ * @return int
890
+ */
891
+ function get_spaces($force_refresh=false){
892
+ return $this->get_bookings()->get_spaces($force_refresh);
893
+ }
894
+
895
+ function get_edit_reschedule_url(){
896
+ if( $this->is_recurrence() ){
897
+ $EM_Event = em_get_event($this->recurrence_id);
898
+ return $EM_Event->get_edit_url();
899
+ }
900
+ }
901
+
902
+ function get_edit_url(){
903
+ if( $this->can_manage('edit_events','edit_others_events') ){
904
+ if( EM_MS_GLOBAL && get_site_option('dbem_ms_global_events_links') && !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
905
+ if( get_blog_option($this->blog_id, 'dbem_edit_events_page') ){
906
+ $link = em_add_get_params(get_permalink(get_blog_option($this->blog_id, 'dbem_edit_events_page')), array('action'=>'edit','event_id'=>$this->event_id), false);
907
+ }
908
+ if( empty($link))
909
+ $link = get_admin_url($this->blog_id, "post.php?post={$this->post_id}&action=edit");
910
+ }else{
911
+ if( get_option('dbem_edit_events_page') && !is_admin() ){
912
+ $link = em_add_get_params(get_permalink(get_option('dbem_edit_events_page')), array('action'=>'edit','event_id'=>$this->event_id), false);
913
+ }
914
+ if( empty($link))
915
+ $link = admin_url()."post.php?post={$this->post_id}&action=edit";
916
+ }
917
+ return apply_filters('em_event_get_edit_url', $link, $this);
918
+ }
919
+ }
920
+
921
+ function get_bookings_url(){
922
+ if( get_option('dbem_edit_bookings_page') && !is_admin() ){
923
+ $my_bookings_page = get_permalink(get_option('dbem_edit_bookings_page'));
924
+ $bookings_link = em_add_get_params($my_bookings_page, array('event_id'=>$this->event_id), false);
925
+ }else{
926
+ $bookings_link = EM_ADMIN_URL ."&page=events-manager-bookings&event_id=".$this->event_id;
927
+ }
928
+ return apply_filters('em_event_get_bookings_url', $bookings_link, $this);
929
+ }
930
+
931
+ function get_permalink(){
932
+ if( EM_MS_GLOBAL ){
933
+ if( get_site_option('dbem_ms_global_events_links') && !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
934
+ //linking directly to the blog
935
+ $event_link = get_blog_permalink( $this->blog_id, $this->post_id);
936
+ }elseif( !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
937
+ if( get_option('dbem_events_page') ){
938
+ $event_link = trailingslashit(get_permalink(get_option('dbem_events_page')).get_site_option('dbem_ms_events_slug',EM_EVENT_SLUG).'/'.$this->event_slug.'-'.$this->event_id);
939
+ }else{
940
+ $event_link = trailingslashit(home_url()).EM_POST_TYPE_EVENT_SLUG.'/'.get_site_option('dbem_ms_events_slug',EM_EVENT_SLUG).'/'.$this->event_slug.'-'.$this->event_id;
941
+ }
942
+ }
943
+ }
944
+ if( empty($event_link) ){
945
+ $event_link = get_post_permalink($this->post_id);
946
+ }
947
+ return apply_filters('em_event_get_permalink', $event_link, $this);
948
+ }
949
+
950
+ function is_free(){
951
+ $free = true;
952
+ if( isset($this->free) ) return $this->free;
953
+ foreach($this->get_tickets() as $EM_Ticket){
954
+ if( $EM_Ticket->price > 0 ){
955
+ $free = false;
956
+ }
957
+ }
958
+ return apply_filters('em_event_is_free',$free,$this);
959
+ }
960
+
961
+ /**
962
+ * Will output a single event format of this event.
963
+ * Equivalent of calling EM_Event::output( get_option ( 'dbem_single_event_format' ) )
964
+ * @param string $target
965
+ * @return string
966
+ */
967
+ function output_single($target='html'){
968
+ $format = get_option ( 'dbem_single_event_format' );
969
+ return apply_filters('em_event_output_single', $this->output($format, $target), $this, $target);
970
+ }
971
+
972
+ /**
973
+ * Will output a event in the format passed in $format by replacing placeholders within the format.
974
+ * @param string $format
975
+ * @param string $target
976
+ * @return string
977
+ */
978
+ function output($format, $target="html") {
979
+ $event_string = $format;
980
+ //Time place holder that doesn't show if empty.
981
+ //TODO add filter here too
982
+ preg_match_all('/#@?_\{[A-Za-z0-9 -\/,\.\\\]+\}/', $format, $results);
983
+ foreach($results[0] as $result) {
984
+ if(substr($result, 0, 3 ) == "#@_"){
985
+ $date = 'end_date';
986
+ $offset = 4;
987
+ }else{
988
+ $date = 'start_date';
989
+ $offset = 3;
990
+ }
991
+ if( $date == 'end_date' && $this->event_end_date == $this->event_start_date ){
992
+ $replace = __( apply_filters('em_event_output_placeholder', '', $this, $result, $target) );
993
+ }else{
994
+ $replace = __( apply_filters('em_event_output_placeholder', mysql2date(substr($result, $offset, (strlen($result)-($offset+1)) ), $this->$date), $this, $result, $target) );
995
+ }
996
+ $event_string = str_replace($result,$replace,$event_string );
997
+ }
998
+ //This is for the custom attributes
999
+ preg_match_all('/#_ATT\{([^}]+)\}(\{([^}]+)\})?/', $format, $results);
1000
+ foreach($results[0] as $resultKey => $result) {
1001
+ //Strip string of placeholder and just leave the reference
1002
+ $attRef = substr( substr($result, 0, strpos($result, '}')), 6 );
1003
+ $attString = '';
1004
+ if( is_array($this->event_attributes) && array_key_exists($attRef, $this->event_attributes) ){
1005
+ $attString = $this->event_attributes[$attRef];
1006
+ }elseif( !empty($results[3][$resultKey]) ){
1007
+ //Check to see if we have a second set of braces;
1008
+ $attString = $results[3][$resultKey];
1009
+ }
1010
+ $attString = apply_filters('em_event_output_placeholder', $attString, $this, $result, $target);
1011
+ $event_string = str_replace($result, $attString ,$event_string );
1012
+ }
1013
+ //First let's do some conditional placeholder removals
1014
+ preg_match_all('/\{([a-zA-Z0-9_]+)\}([^{]+)\{\/\1\}/', $event_string, $conditionals);
1015
+ if( count($conditionals[0]) > 0 ){
1016
+ //Check if the language we want exists, if not we take the first language there
1017
+ foreach($conditionals[1] as $key => $condition){
1018
+ $show_condition = false;
1019
+ if ($condition == 'has_bookings') {
1020
+ //check if there's a booking, if not, remove this section of code.
1021
+ $show_condition = ($this->event_rsvp && get_option('dbem_rsvp_enabled'));
1022
+ }elseif ($condition == 'no_bookings') {
1023
+ //check if there's a booking, if not, remove this section of code.
1024
+ $show_condition = (!$this->event_rsvp && get_option('dbem_rsvp_enabled'));
1025
+ }elseif ($condition == 'no_location'){
1026
+ //does this event have a valid location?
1027
+ $show_condition = ( empty($this->location_id) || !$this->get_location()->location_status );
1028
+ }elseif ($condition == 'has_location'){
1029
+ //does this event have a valid location?
1030
+ $show_condition = ( !empty($this->location_id) && $this->get_location()->location_status );
1031
+ }elseif ($condition == 'has_image'){
1032
+ //does this event have an image?
1033
+ $show_condition = ( $this->get_image_url() != '' );
1034
+ }elseif ($condition == 'no_image'){
1035
+ //does this event have an image?
1036
+ $show_condition = ( $this->get_image_url() == '' );
1037
+ }elseif ($condition == 'has_time'){
1038
+ //are the booking times different and not an all-day event
1039
+ $show_condition = ( $this->start != $this->end && !$this->event_all_day );
1040
+ }elseif ($condition == 'no_time'){
1041
+ //are the booking times exactly the same and it's not an all-day event.
1042
+ $show_condition = ( $this->event_start_time == $this->event_end_time && !$this->event_all_day );
1043
+ }elseif ($condition == 'all_day'){
1044
+ //is it an all day event
1045
+ $show_condition = !empty($this->event_all_day);
1046
+ }
1047
+ $show_condition = apply_filters('em_event_output_show_condition', $show_condition, $condition, $conditionals[0][$key], $this);
1048
+ if($show_condition){
1049
+ //calculate lengths to delete placeholders
1050
+ $placeholder_length = strlen($condition)+2;
1051
+ $replacement = substr($conditionals[0][$key], $placeholder_length, strlen($conditionals[0][$key])-($placeholder_length *2 +1));
1052
+ }else{
1053
+ $replacement = '';
1054
+ }
1055
+ $event_string = str_replace($conditionals[0][$key], apply_filters('em_event_output_condition', $replacement, $condition, $conditionals[0][$key], $this), $event_string);
1056
+ }
1057
+ }
1058
+ //Now let's check out the placeholders.
1059
+ preg_match_all("/(#@?_?[A-Za-z0-9]+)({([a-zA-Z0-9,]+)})?/", $format, $placeholders);
1060
+ foreach($placeholders[1] as $key => $result) {
1061
+ $match = true;
1062
+ $replace = '';
1063
+ $full_result = $placeholders[0][$key];
1064
+ switch( $result ){
1065
+ //Event Details
1066
+ case '#_EVENTID':
1067
+ $replace = $this->event_id;
1068
+ break;
1069
+ case '#_EVENTPOSTID':
1070
+ $replace = $this->post_id;
1071
+ break;
1072
+ case '#_NAME': //depreciated
1073
+ case '#_EVENTNAME':
1074
+ $replace = $this->event_name;
1075
+ break;
1076
+ case '#_NOTES': //depreciated
1077
+ case '#_EXCERPT': //depreciated
1078
+ case '#_EVENTNOTES':
1079
+ case '#_EVENTEXCERPT':
1080
+ $replace = $this->post_content;
1081
+ if($result == "#_EXCERPT" || $result == "#_EVENTEXCERPT"){
1082
+ if( !empty($this->post_excerpt) ){
1083
+ $replace = $this->post_excerpt;
1084
+ }else{
1085
+ $matches = explode('<!--more', $this->post_content);
1086
+ $replace = $matches[0];
1087
+ }
1088
+ }
1089
+ break;
1090
+ case '#_EVENTIMAGEURL':
1091
+ case '#_EVENTIMAGE':
1092
+ if($this->get_image_url() != ''){
1093
+ if($result == '#_EVENTIMAGEURL'){
1094
+ $replace = esc_url($this->image_url);
1095
+ }else{
1096
+ if( empty($placeholders[3][$key]) ){
1097
+ $replace = "<img src='".esc_url($this->image_url)."' alt='".esc_attr($this->event_name)."'/>";
1098
+ }else{
1099
+ $image_size = explode(',', $placeholders[3][$key]);
1100
+ if( $this->array_is_numeric($image_size) && count($image_size) > 1 ){
1101
+ $replace = "<img src='".esc_url(em_get_thumbnail_url($this->image_url, $image_size[0], $image_size[1]))."' alt='".esc_attr($this->event_name)."'/>";
1102
+ }else{
1103
+ $replace = "<img src='".esc_url($this->image_url)."' alt='".esc_attr($this->event_name)."'/>";
1104
+ }
1105
+ }
1106
+ }
1107
+ }
1108
+ break;
1109
+ //Times & Dates
1110
+ case '#_24HSTARTTIME':
1111
+ case '#_24HENDTIME':
1112
+ $time = ($result == '#_24HSTARTTIME') ? $this->event_start_time:$this->event_end_time;
1113
+ $replace = substr($time, 0,5);
1114
+ break;
1115
+ case '#_12HSTARTTIME':
1116
+ case '#_12HENDTIME':
1117
+ $time = ($result == '#_12HSTARTTIME') ? $this->event_start_time:$this->event_end_time;
1118
+ $replace = date('g:i A', strtotime($time));
1119
+ break;
1120
+ case '#_EVENTTIMES':
1121
+ //get format of time to show
1122
+ if( !$this->event_all_day ){
1123
+ $time_format = ( get_option('dbem_time_format') ) ? get_option('dbem_time_format'):get_option('time_format');
1124
+ if($this->event_start_time != $this->event_end_time ){
1125
+ $replace = date($time_format, $this->start). get_option('dbem_times_seperator') . date($time_format, $this->end);
1126
+ }else{
1127
+ $replace = date($time_format, $this->start);
1128
+ }
1129
+ }else{
1130
+ $replace = get_option('dbem_event_all_day_message');
1131
+ }
1132
+ break;
1133
+ case '#_EVENTDATES':
1134
+ //get format of time to show
1135
+ $date_format = ( get_option('dbem_date_format') ) ? get_option('dbem_date_format'):get_option('date_format');
1136
+ if( $this->event_start_date != $this->event_end_date){
1137
+ $replace = date($date_format, $this->start). get_option('dbem_dates_seperator') . date($date_format, $this->end);
1138
+ }else{
1139
+ $replace = date($date_format, $this->start);
1140
+ }
1141
+ break;
1142
+ //Links
1143
+ case '#_EVENTPAGEURL': //Depreciated
1144
+ case '#_LINKEDNAME': //Depreciated
1145
+ case '#_EVENTURL': //Just the URL
1146
+ case '#_EVENTLINK': //HTML Link
1147
+ $event_link = esc_url($this->get_permalink());
1148
+ if($result == '#_LINKEDNAME' || $result == '#_EVENTLINK'){
1149
+ $replace = '<a href="'.$event_link.'" title="'.esc_attr($this->event_name).'">'.esc_attr($this->event_name).'</a>';
1150
+ }else{
1151
+ $replace = $event_link;
1152
+ }
1153
+ break;
1154
+ case '#_EDITEVENTURL':
1155
+ case '#_EDITEVENTLINK':
1156
+ if( $this->can_manage('edit_events','edit_others_events') ){
1157
+ $link = esc_url($this->get_edit_url());
1158
+ if( $result == '#_EDITEVENTLINK'){
1159
+ $replace = '<a href="'.$link.'">'.esc_html(sprintf(__('Edit %s','dbem'),__('Event','dbem'))).'</a>';
1160
+ }else{
1161
+ $replace = $link;
1162
+ }
1163
+ }
1164
+ break;
1165
+ //Bookings
1166
+ case '#_ADDBOOKINGFORM': //Depreciated
1167
+ case '#_REMOVEBOOKINGFORM': //Depreciated
1168
+ case '#_BOOKINGFORM':
1169
+ if( get_option('dbem_rsvp_enabled')){
1170
+ ob_start();
1171
+ $template = em_locate_template('placeholders/bookingform.php', true, array('EM_Event'=>$this));
1172
+ if( !defined('EM_BOOKING_JS_LOADED') ){
1173
+ //this kicks off the Javascript required by booking forms. This is fired once for all booking forms on a page load and appears at the bottom of the page
1174
+ //your theme must call the wp_footer() function for this to work (as required by many other plugins too)
1175
+ function em_booking_js_footer(){
1176
+ ?>
1177
+ <script type="text/javascript">
1178
+ jQuery(document).ready( function($){
1179
+ <?php
1180
+ //we call the segmented JS files and include them here
1181
+ include(WP_PLUGIN_DIR.'/events-manager/includes/js/bookingsform.js');
1182
+ do_action('em_gateway_js');
1183
+ ?>
1184
+ });
1185
+ </script>
1186
+ <?php
1187
+ }
1188
+ add_action('wp_footer','em_booking_js_footer');
1189
+ define('EM_BOOKING_JS_LOADED',true);
1190
+ }
1191
+ $replace = ob_get_clean();
1192
+ }
1193
+ break;
1194
+ case '#_BOOKINGBUTTON':
1195
+ if( get_option('dbem_rsvp_enabled')){
1196
+ ob_start();
1197
+ $template = em_locate_template('placeholders/bookingbutton.php', true, array('EM_Event'=>$this));
1198
+ $replace = ob_get_clean();
1199
+ }
1200
+ break;
1201
+ case '#_AVAILABLESEATS': //Depreciated
1202
+ case '#_AVAILABLESPACES':
1203
+ if ($this->event_rsvp && get_option('dbem_rsvp_enabled')) {
1204
+ $replace = $this->get_bookings()->get_available_spaces();
1205
+ } else {
1206
+ $replace = "0";
1207
+ }
1208
+ break;
1209
+ case '#_BOOKEDSEATS': //Depreciated
1210
+ case '#_BOOKEDSPACES':
1211
+ if ($this->event_rsvp && get_option('dbem_rsvp_enabled')) {
1212
+ $replace = $this->get_bookings()->get_booked_spaces();
1213
+ } else {
1214
+ $replace = "0";
1215
+ }
1216
+ break;
1217
+ case '#_PENDINGSPACES':
1218
+ if ($this->event_rsvp && get_option('dbem_rsvp_enabled')) {
1219
+ $replace = $this->get_bookings()->get_pending_spaces();
1220
+ } else {
1221
+ $replace = "0";
1222
+ }
1223
+ break;
1224
+ case '#_SEATS': //Depreciated
1225
+ case '#_SPACES':
1226
+ $replace = $this->get_spaces();
1227
+ break;
1228
+ case '#_BOOKINGSURL':
1229
+ case '#_BOOKINGSLINK':
1230
+ if( $this->can_manage('manage_bookings','manage_others_bookings') ){
1231
+ $bookings_link = esc_url($this->get_bookings_url());
1232
+ if($result == '#_BOOKINGSLINK'){
1233
+ $replace = '<a href="'.$bookings_link.'" title="'.esc_attr($this->event_name).'">'.esc_html($this->event_name).'</a>';
1234
+ }else{
1235
+ $replace = $bookings_link;
1236
+ }
1237
+ }
1238
+ break;
1239
+ //Contact Person
1240
+ case '#_CONTACTNAME':
1241
+ case '#_CONTACTPERSON': //Depreciated (your call, I think name is better)
1242
+ $replace = $this->get_contact()->display_name;
1243
+ break;
1244
+ case '#_CONTACTUSERNAME':
1245
+ $replace = $this->get_contact()->user_login;
1246
+ break;
1247
+ case '#_CONTACTEMAIL':
1248
+ case '#_CONTACTMAIL': //Depreciated
1249
+ $replace = $this->get_contact()->user_email;
1250
+ break;
1251
+ case '#_CONTACTID':
1252
+ $replace = $this->get_contact()->ID;
1253
+ break;
1254
+ case '#_CONTACTPHONE':
1255
+ $replace = ( $this->get_contact()->phone != '') ? $this->get_contact()->phone : __('N/A', 'dbem');
1256
+ break;
1257
+ case '#_CONTACTAVATAR':
1258
+ $replace = get_avatar( $this->get_contact()->ID, $size = '50' );
1259
+ break;
1260
+ case '#_CONTACTPROFILELINK':
1261
+ case '#_CONTACTPROFILEURL':
1262
+ if( function_exists('bp_core_get_user_domain') ){
1263
+ $replace = bp_core_get_user_domain($this->get_contact()->ID);
1264
+ if( $result == '#_CONTACTPROFILELINK' ){
1265
+ $replace = '<a href="'.esc_url($replace).'">'.__('Profile', 'dbem').'</a>';
1266
+ }
1267
+ }
1268
+ break;
1269
+ case '#_ATTENDEES':
1270
+ ob_start();
1271
+ $template = em_locate_template('placeholders/attendees.php', true, array('EM_Event'=>$this));
1272
+ $replace = ob_get_clean();
1273
+ break;
1274
+ case '#_CATEGORIES': //depreciated
1275
+ case '#_EVENTCATEGORIES':
1276
+ ob_start();
1277
+ $template = em_locate_template('placeholders/categories.php', true, array('EM_Event'=>$this));
1278
+ $replace = ob_get_clean();
1279
+ break;
1280
+ case '#_EVENTTAGS':
1281
+ ob_start();
1282
+ $template = em_locate_template('placeholders/eventtags.php', true, array('EM_Event'=>$this));
1283
+ $replace = ob_get_clean();
1284
+ break;
1285
+ default:
1286
+ $replace = $full_result;
1287
+ break;
1288
+ }
1289
+ $replace = apply_filters('em_event_output_placeholder', $replace, $this, $full_result, $target );
1290
+ $event_string = str_replace($full_result, $replace , $event_string );
1291
+ }
1292
+ //Time placeholders
1293
+ foreach($placeholders[1] as $result) {
1294
+ // matches all PHP START date and time placeholders
1295
+ if (preg_match('/^#[dDjlNSwzWFmMntLoYyaABgGhHisueIOPTZcrU]$/', $result)) {
1296
+ $replace = date_i18n(ltrim($result, "#"), $this->start);
1297
+ $replace = apply_filters('em_event_output_placeholder', $replace, $this, $result, $target);
1298
+ $event_string = str_replace($result, $replace, $event_string );
1299
+ }
1300
+ // matches all PHP END time placeholders for endtime
1301
+ if (preg_match('/^#@[dDjlNSwzWFmMntLoYyaABgGhHisueIOPTZcrU]$/', $result)) {
1302
+ $replace = date_i18n(ltrim($result, "#@"), $this->end);
1303
+ $replace = apply_filters('em_event_output_placeholder', $replace, $this, $result, $target);
1304
+ $event_string = str_replace($result, $replace, $event_string );
1305
+ }
1306
+ }
1307
+ //Now do dependent objects
1308
+ if( !empty($this->location_id) && $this->get_location()->location_status ){
1309
+ $event_string = $this->get_location()->output($event_string, $target);
1310
+ }else{
1311
+ $EM_Location = new EM_Location();
1312
+ $event_string = $EM_Location->output($event_string, $target);
1313
+ }
1314
+
1315
+ //for backwards compat and easy use, take over the individual category placeholders with the frirst cat in th elist.
1316
+ $EM_Categories = $this->get_categories();
1317
+ if( count($EM_Categories->categories) > 0 ){
1318
+ $EM_Category = $EM_Categories->get_first();
1319
+ }
1320
+
1321
+ if( empty($EM_Category) ) $EM_Category = new EM_Category();
1322
+ $event_string = $EM_Category->output($event_string, $target);
1323
+
1324
+ return apply_filters('em_event_output', $event_string, $this, $format, $target);
1325
+ }
1326
+
1327
+ /**********************************************************
1328
+ * RECURRENCE METHODS
1329
+ ***********************************************************/
1330
+
1331
+ /**
1332
+ * Returns true if this is a recurring event.
1333
+ * @return boolean
1334
+ */
1335
+ function is_recurring(){
1336
+ return $this->post_type == 'event-recurring' && get_option('dbem_recurrence_enabled');
1337
+ }
1338
+ /**
1339
+ * Will return true if this individual event is part of a set of events that recur
1340
+ * @return boolean
1341
+ */
1342
+ function is_recurrence(){
1343
+ return ( $this->event_id > 0 && $this->recurrence_id > 0 && get_option('dbem_recurrence_enabled') );
1344
+ }
1345
+ /**
1346
+ * Returns if this is an individual event and is not a recurrence
1347
+ * @return boolean
1348
+ */
1349
+ function is_individual(){
1350
+ return ( !$this->is_recurring() && !$this->is_recurrence() );
1351
+ }
1352
+
1353
+ /**
1354
+ * Gets the event recurrence template, which is an EM_Event_Recurrence object (based off an event-recurring post)
1355
+ * @return EM_Event_Recurrence
1356
+ */
1357
+ function get_event_recurrence(){
1358
+ if(!$this->is_recurring()){
1359
+ return new EM_Event($this->recurrence_id); //remember, recurrence_id is a post!
1360
+ }else{
1361
+ return $this;
1362
+ }
1363
+ }
1364
+
1365
+ function get_detach_url(){
1366
+ return admin_url().'admin.php?event_id='.$this->event_id.'&amp;action=event_detach&amp;_wpnonce='.wp_create_nonce('event_detach_'.get_current_user_id().'_'.$this->event_id);
1367
+ }
1368
+
1369
+ function get_attach_url($recurrence_id){
1370
+ return admin_url().'admin.php?undo_id='.$recurrence_id.'&amp;event_id='.$this->event_id.'&amp;action=event_attach&amp;_wpnonce='.wp_create_nonce('event_attach_'.get_current_user_id().'_'.$this->event_id);
1371
+ }
1372
+
1373
+ /**
1374
+ * Returns if this is an individual event and is not recurring or a recurrence
1375
+ * @return boolean
1376
+ */
1377
+ function detach(){
1378
+ global $wpdb;
1379
+ if( $this->is_recurrence() && !$this->is_recurring() && $this->can_manage('edit_recurring_events','edit_others_recurring_events') ){
1380
+ //remove recurrence id from post meta and index table
1381
+ $url = $this->get_attach_url($this->recurrence_id);
1382
+ $wpdb->update(EM_EVENTS_TABLE, array('recurrence_id'=>0), array('event_id' => $this->event_id));
1383
+ update_post_meta($this->post_id, '_recurrence_id', 0);
1384
+ $this->feedback_message = __('Event detached.','dbem') . ' <a href="'.$url.'">'.__('Undo','dbem').'</a>';
1385
+ $this->recurrence_id = 0;
1386
+ return true;
1387
+ }
1388
+ $this->add_error(__('Event could not be detached.','dbem'));
1389
+ return false;
1390
+ }
1391
+
1392
+ /**
1393
+ * Returns if this is an individual event and is not recurring or a recurrence
1394
+ * @return boolean
1395
+ */
1396
+ function attach($recurrence_id){
1397
+ global $wpdb;
1398
+ if( !$this->is_recurrence() && !$this->is_recurring() && is_numeric($recurrence_id) && $this->can_manage('edit_recurring_events','edit_others_recurring_events') ){
1399
+ //add recurrence id to post meta and index table
1400
+ $wpdb->update(EM_EVENTS_TABLE, array('recurrence_id'=>$recurrence_id), array('event_id' => $this->event_id));
1401
+ update_post_meta($this->post_id, '_recurrence_id', $recurrence_id);
1402
+ $this->feedback_message = __('Event re-attached to recurrence.','dbem');
1403
+ return true;
1404
+ }
1405
+ $this->add_error(__('Event could not be attached.','dbem'));
1406
+ return false;
1407
+ }
1408
+
1409
+ /**
1410
+ * Saves events and replaces old ones. Returns true if sucecssful or false if not.
1411
+ * @return boolean
1412
+ */
1413
+ function save_events() {
1414
+ global $wpdb;
1415
+ if( $this->can_manage('edit_events','edit_others_events') && $this->is_published() ){
1416
+ do_action('em_event_save_events_pre', $this); //actions/filters only run if event is recurring
1417
+ //Make template event index, post, and meta (and we just change event dates)
1418
+ $event = $this->to_array(true); //event template - for index
1419
+ $event['event_attributes'] = serialize($event['event_attributes']);
1420
+ $post_fields = $wpdb->get_row('SELECT * FROM '.$wpdb->posts.' WHERE ID='.$this->post_id, ARRAY_A); //post to copy
1421
+ $post_name = $post_fields['post_name']; //save post slug since we'll be using this
1422
+ $post_fields['post_type'] = 'event'; //make sure we'll save events, not recurrence templates
1423
+ $meta_fields_map = $wpdb->get_results('SELECT meta_key,meta_value FROM '.$wpdb->postmeta.' WHERE post_id='.$this->post_id, ARRAY_A);
1424
+ $meta_fields = array();
1425
+ //convert meta_fields into a cleaner array
1426
+ foreach($meta_fields_map as $meta_data){
1427
+ $meta_fields[$meta_data['meta_key']] = $meta_data['meta_value'];
1428
+ }
1429
+ //remove id and we have a event template to feed to wpdb insert
1430
+ unset($event['event_id']);
1431
+ unset($post_fields['ID']);
1432
+ //remove recurrence meta info we won't need in events
1433
+ foreach( array_keys($this->recurrence_fields) as $recurrence_field){
1434
+ unset($event[$recurrence_field]);
1435
+ unset($meta_fields['_'.$recurrence_field]);
1436
+ }
1437
+ $event['event_date_created'] = current_time('mysql'); //since the recurrences are recreated
1438
+ unset($event['event_date_modified']);
1439
+ //Set the recurrence ID
1440
+ $event['recurrence_id'] = $meta_fields['_recurrence_id'] = $this->event_id;
1441
+ $event['recurrence'] = $meta_fields['_recurrence'] = 0;
1442
+ //Let's start saving!
1443
+ $this->delete_events(); //Delete old events beforehand, this will change soon
1444
+ $event_saves = array();
1445
+ $event_ids = array();
1446
+ $post_ids = array();
1447
+ $matching_days = $this->get_recurrence_days(); //Get days where events recur
1448
+ /*
1449
+ echo 'tickets';
1450
+ echo "<pre>"; print_r($this->recurrence_tickets); echo "</pre>";
1451
+ echo '$meta_fields';
1452
+ echo "<pre>"; print_r($meta_fields); echo "</pre>";
1453
+ echo '$post_fields';
1454
+ echo "<pre>"; print_r($post_fields); echo "</pre>";
1455
+ echo '$event';
1456
+ echo "<pre>"; print_r($event); echo "</pre>";
1457
+ echo '$matching_days';
1458
+ echo "<pre>"; print_r($matching_days); echo "</pre>";
1459
+ die('i got here and we have '.count($this->errors).' errors');
1460
+ */
1461
+ if( count($matching_days) > 0 ){
1462
+ //first save event post data
1463
+ foreach( $matching_days as $day ) {
1464
+ //rewrite post fields if needed
1465
+ $post_fields['post_name'] = $event['event_slug'] = $meta_fields['_event_slug'] = $post_name.'-'.date("Y-m-d", $day);
1466
+ //adjust certain meta information
1467
+ $event['event_start_date'] = $meta_fields['_event_start_date'] = date("Y-m-d", $day);
1468
+ $meta_fields['_start_ts'] = $day;
1469
+ if($this->recurrence_days > 1){
1470
+ $meta_fields['_end_ts'] = $day + ($this->recurrence_days * 60*60*24);
1471
+ $event['event_end_date'] = $meta_fields['_event_end_date'] = date("Y-m-d", $meta_fields['_end_ts']);
1472
+ }else{
1473
+ $meta_fields['_end_ts'] = $day;
1474
+ $event['event_end_date'] = $meta_fields['_event_end_date'] = $event['event_start_date'];
1475
+ }
1476
+ //create the event
1477
+ if( $wpdb->insert($wpdb->posts, $post_fields ) ){
1478
+ $event['post_id'] = $meta_fields['_post_id'] = $post_id = $post_ids[] = $wpdb->insert_id; //post id saved into event and also as a var for later user
1479
+ // Set GUID and event slug as per wp_insert_post
1480
+ $wpdb->update( $wpdb->posts, array( 'guid' => get_permalink( $post_id ) ), array('ID'=>$post_id) );
1481
+ //insert into events index table
1482
+ $event_saves[] = $wpdb->insert(EM_EVENTS_TABLE, $event);
1483
+ $event_ids[] = $event_id = $wpdb->insert_id;
1484
+ //create the meta inserts for each event
1485
+ $meta_fields['_event_id'] = $event_id;
1486
+ foreach($meta_fields as $meta_key => $meta_val){
1487
+ $meta_inserts[] = $wpdb->prepare("(%d, '%s', '%s')", array($post_id, $meta_key, $meta_val));
1488
+ }
1489
+ }else{
1490
+ $event_saves[] = false;
1491
+ }
1492
+ //if( EM_DEBUG ){ echo "Entering recurrence " . date("D d M Y", $day)."<br/>"; }
1493
+ }
1494
+ //insert the metas in one go, faster than one by one
1495
+ if( count($meta_inserts) > 0 ){
1496
+ $result = $wpdb->query("INSERT INTO ".$wpdb->postmeta." (post_id,meta_key,meta_value) VALUES ".implode(',',$meta_inserts));
1497
+ if($result === false){
1498
+ $this->add_error('There was a problem adding custom fields to your recurring events.','dbem');
1499
+ }
1500
+ }
1501
+ //copy the event tags and categories
1502
+ $categories = get_the_terms( $this->post_id, EM_TAXONOMY_CATEGORY );
1503
+ $cat_slugs = array();
1504
+ if( is_array($categories) ){
1505
+ foreach($categories as $category){
1506
+ if( !empty($category->slug) ) $cat_slugs[] = $category->slug; //save of category will soft-fail if slug is empty
1507
+ }
1508
+ }
1509
+ $cat_slugs_count = count($cat_slugs);
1510
+ $tags = get_the_terms( $this->post_id, EM_TAXONOMY_TAG);
1511
+ $tax_slugs = array();
1512
+ if( is_array($tags) ){
1513
+ foreach($tags as $tag){
1514
+ if( !empty($tag->slug) ) $tax_slugs[] = $tag->slug; //save of category will soft-fail if slug is empty
1515
+ }
1516
+ }
1517
+ $tax_slugs_count = count($tax_slugs);
1518
+ foreach($post_ids as $post_id){
1519
+ if( $cat_slugs_count > 0 ){
1520
+ wp_set_object_terms($post_id, $cat_slugs, EM_TAXONOMY_CATEGORY);
1521
+ }
1522
+ if( $tax_slugs_count > 0 ){
1523
+ wp_set_object_terms($post_id, $tax_slugs, EM_TAXONOMY_TAG);
1524
+ }
1525
+ }
1526
+ //now, save booking info for each event
1527
+ if( $this->event_rsvp ){
1528
+ $meta_inserts = array();
1529
+ foreach($this->get_tickets() as $EM_Ticket){
1530
+ /* @var $EM_Ticket EM_Ticket */
1531
+ //get array, modify event id and insert
1532
+ $ticket = $EM_Ticket->to_array();
1533
+ unset($ticket['ticket_id']);
1534
+ //clean up ticket values
1535
+ foreach($ticket as $k => $v){
1536
+ if( empty($v) && $k != 'ticket_name' ){
1537
+ $ticket[$k] = 'NULL';
1538
+ }else{
1539
+ $ticket[$k] = "'$v'";
1540
+ }
1541
+ }
1542
+ foreach($event_ids as $event_id){
1543
+ $ticket['event_id'] = $event_id;
1544
+ $meta_inserts[] = "(".implode(",",$ticket).")";
1545
+ }
1546
+ }
1547
+ $keys = "(".implode(",",array_keys($ticket)).")";
1548
+ $values = implode(',',$meta_inserts);
1549
+ $sql = "INSERT INTO ".EM_TICKETS_TABLE." $keys VALUES $values";
1550
+ $result = $wpdb->query($sql);
1551
+ }
1552
+ }else{
1553
+ $this->add_error('You have not defined a date range long enough to create a recurrence.','dbem');
1554
+ $result = false;
1555
+ }
1556
+ return apply_filters('em_event_save_events', !in_array(false, $event_saves) && $result !== false, $this, $event_ids);
1557
+ }
1558
+ return apply_filters('em_event_save_events', false, $this, $event_ids);
1559
+ }
1560
+
1561
+ /**
1562
+ * Removes all reoccurring events.
1563
+ * @param $recurrence_id
1564
+ * @return null
1565
+ */
1566
+ function delete_events(){
1567
+ global $wpdb;
1568
+ do_action('em_event_delete_events_pre', $this);
1569
+ //So we don't do something we'll regret later, we could just supply the get directly into the delete, but this is safer
1570
+ $result = false;
1571
+ if( $this->can_manage('delete_events', 'delete_others_events') ){
1572
+ //delete events from em_events table
1573
+ $events_array = EM_Events::get( array('recurrence_id'=>$this->event_id, 'scope'=>'all', 'status'=>false ) );
1574
+ foreach($events_array as $EM_Event){
1575
+ /* @var $EM_Event EM_Event */
1576
+ if($EM_Event->recurrence_id == $this->event_id){
1577
+ $EM_Event->delete(true);
1578
+ }
1579
+ }
1580
+ }
1581
+ return apply_filters('delete_events', $result, $this, $events_array);
1582
+ }
1583
+
1584
+ /**
1585
+ * Returns the days that match the recurrance array passed (unix timestamps)
1586
+ * @param array $recurrence
1587
+ * @return array
1588
+ */
1589
+ function get_recurrence_days(){
1590
+ $start_date = strtotime($this->event_start_date);
1591
+ $end_date = strtotime($this->event_end_date);
1592
+
1593
+ $weekdays = explode(",", $this->recurrence_byday); //what days of the week (or if monthly, one value at index 0)
1594
+
1595
+ $matching_days = array();
1596
+ $aDay = 86400; // a day in seconds
1597
+ $aWeek = $aDay * 7;
1598
+
1599
+ //TODO can this be optimized?
1600
+ switch ( $this->recurrence_freq ){
1601
+ case 'daily':
1602
+ //If daily, it's simple. Get start date, add interval timestamps to that and create matching day for each interval until end date.
1603
+ $current_date = $start_date;
1604
+ while( $current_date <= $end_date ){
1605
+ $matching_days[] = $current_date;
1606
+ $current_date = $current_date + ($aDay * $this->recurrence_interval);
1607
+ }
1608
+ break;
1609
+ case 'weekly':
1610
+ //sort out week one, get starting days and then days that match time span of event (i.e. remove past events in week 1)
1611
+ $start_of_week = get_option('start_of_week'); //Start of week depends on WordPress
1612
+ //first, get the start of this week as timestamp
1613
+ $event_start_day = date('w', $start_date);
1614
+ $offset = 0;
1615
+ if( $event_start_day > $start_of_week ){
1616
+ $offset = $event_start_day - $start_of_week; //x days backwards
1617
+ }elseif( $event_start_day < $start_of_week ){
1618
+ $offset = $start_of_week;
1619
+ }
1620
+ $start_week_date = $start_date - ( ($event_start_day - $start_of_week) * $aDay );
1621
+ //then get the timestamps of weekdays during this first week, regardless if within event range
1622
+ $start_weekday_dates = array(); //Days in week 1 where there would events, regardless of event date range
1623
+ for($i = 0; $i < 7; $i++){
1624
+ $weekday_date = $start_week_date+($aDay*$i); //the date of the weekday we're currently checking
1625
+ $weekday_day = date('w',$weekday_date); //the day of the week we're checking, taking into account wp start of week setting
1626
+ if( in_array( $weekday_day, $weekdays) ){
1627
+ $start_weekday_dates[] = $weekday_date; //it's in our starting week day, so add it
1628
+ }
1629
+ }
1630
+ //for each day of eventful days in week 1, add 7 days * weekly intervals
1631
+ foreach ($start_weekday_dates as $weekday_date){
1632
+ //Loop weeks by interval until we reach or surpass end date
1633
+ while($weekday_date <= $end_date){
1634
+ if( $weekday_date >= $start_date && $weekday_date <= $end_date ){
1635
+ $matching_days[] = $weekday_date;
1636
+ }
1637
+ $weekday_date = $weekday_date + ($aWeek * $this->recurrence_interval);
1638
+ }
1639
+ }//done!
1640
+ break;
1641
+ case 'monthly':
1642
+ //loop months starting this month by intervals
1643
+ $current_arr = getdate($start_date);
1644
+ $end_arr = getdate($end_date);
1645
+ $end_month_date = strtotime( date('Y-m-t', $end_date) ); //End date on last day of month
1646
+ $current_date = strtotime( date('Y-m-1', $start_date) ); //Start date on first day of month
1647
+ while( $current_date <= $end_month_date ){
1648
+ $last_day_of_month = date('t', $current_date);
1649
+ //Now find which day we're talking about
1650
+ $current_week_day = date('w',$current_date);
1651
+ $matching_month_days = array();
1652
+ //Loop through days of this years month and save matching days to temp array
1653
+ for($day = 1; $day <= $last_day_of_month; $day++){
1654
+ if($current_week_day == $this->recurrence_byday){
1655
+ $matching_month_days[] = $day;
1656
+ }
1657
+ $current_week_day = ($current_week_day < 6) ? $current_week_day+1 : 0;
1658
+ }
1659
+ //Now grab from the array the x day of the month
1660
+ $matching_day = ($this->recurrence_byweekno > 0) ? $matching_month_days[$this->recurrence_byweekno-1] : array_pop($matching_month_days);
1661
+ $matching_date = strtotime(date('Y-m',$current_date).'-'.$matching_day);
1662
+ if($matching_date >= $start_date && $matching_date <= $end_date){
1663
+ $matching_days[] = $matching_date;
1664
+ }
1665
+ //add the number of days in this month to make start of next month
1666
+ $current_arr['mon'] += $this->recurrence_interval;
1667
+ if($current_arr['mon'] > 12){
1668
+ //FIXME this won't work if interval is more than 12
1669
+ $current_arr['mon'] = $current_arr['mon'] - 12;
1670
+ $current_arr['year']++;
1671
+ }
1672
+ $current_date = strtotime("{$current_arr['year']}-{$current_arr['mon']}-1");
1673
+ }
1674
+ break;
1675
+ }
1676
+ sort($matching_days);
1677
+ return apply_filters('em_events_get_recurrence_days', $matching_days, $this);
1678
+ }
1679
+
1680
+ /**
1681
+ * If event is recurring, set recurrences to same status as template
1682
+ * @param $status
1683
+ */
1684
+ function set_status_events($status){
1685
+ //give sub events same status
1686
+ $events_array = EM_Events::get( array('recurrence_id'=>$this->post_id, 'scope'=>'all', 'status'=>false ) );
1687
+ foreach($events_array as $EM_Event){
1688
+ /* @var $EM_Event EM_Event */
1689
+ if($EM_Event->recurrence_id == $this->event_id){
1690
+ $EM_Event->set_status($status);
1691
+ }
1692
+ }
1693
+ }
1694
+
1695
+ /**
1696
+ * Returns a string representation of this recurrence. Will return false if not a recurrence
1697
+ * @return string
1698
+ */
1699
+ function get_recurrence_description() {
1700
+ $EM_Event_Recurring = $this->get_event_recurrence();
1701
+ $recurrence = $this->to_array();
1702
+ $weekdays_name = array(__('Sunday', 'dbem'),__('Monday', 'dbem'),__('Tuesday', 'dbem'),__('Wednesday', 'dbem'),__('Thursday', 'dbem'),__('Friday', 'dbem'),__('Saturday', 'dbem'));
1703
+ $monthweek_name = array('1' => __('the first %s of the month', 'dbem'),'2' => __('the second %s of the month', 'dbem'), '3' => __('the third %s of the month', 'dbem'), '4' => __('the fourth %s of the month', 'dbem'), '-1' => __('the last %s of the month', 'dbem'));
1704
+ $output = sprintf (__('From %1$s to %2$s', 'dbem'), $EM_Event_Recurring->event_start_date, $EM_Event_Recurring->event_end_date).", ";
1705
+ if ($EM_Event_Recurring->recurrence_freq == 'daily') {
1706
+ $freq_desc =__('everyday', 'dbem');
1707
+ if ($EM_Event_Recurring->recurrence_interval > 1 ) {
1708
+ $freq_desc = sprintf (__("every %s days", 'dbem'), $EM_Event_Recurring->recurrence_interval);
1709
+ }
1710
+ }elseif ($EM_Event_Recurring->recurrence_freq == 'weekly') {
1711
+ $weekday_array = explode(",", $EM_Event_Recurring->recurrence_byday);
1712
+ $natural_days = array();
1713
+ foreach($weekday_array as $day){
1714
+ array_push($natural_days, $weekdays_name[$day]);
1715
+ }
1716
+ $output .= implode(" and ", $natural_days);
1717
+ $freq_desc = ", " . __("every week", 'dbem');
1718
+ if ($EM_Event_Recurring->recurrence_interval > 1 ) {
1719
+ $freq_desc = ", ".sprintf (__("every %s weeks", 'dbem'), $EM_Event_Recurring->recurrence_interval);
1720
+ }
1721
+
1722
+ }elseif ($EM_Event_Recurring->recurrence_freq == 'monthly') {
1723
+ $weekday_array = explode(",", $EM_Event_Recurring->recurrence_byday);
1724
+ $natural_days = array();
1725
+ foreach($weekday_array as $day){
1726
+ array_push($natural_days, $weekdays_name[$day]);
1727
+ }
1728
+ $freq_desc = sprintf (($monthweek_name[$EM_Event_Recurring->recurrence_byweekno]), implode(" and ", $natural_days));
1729
+ if ($EM_Event_Recurring->recurrence_interval > 1 ) {
1730
+ $freq_desc .= ", ".sprintf (__("every %s months",'dbem'), $EM_Event_Recurring->recurrence_interval);
1731
+ }
1732
+ }else{
1733
+ $freq_desc = "[ERROR: corrupted database record]";
1734
+ }
1735
+ $output .= $freq_desc;
1736
+ return $output;
1737
+ }
1738
+
1739
+ /**********************************************************
1740
+ * UTILITIES
1741
+ ***********************************************************/
1742
+
1743
+ /**
1744
+ * Can the user manage this?
1745
+ */
1746
+ function can_manage( $owner_capability = false, $admin_capability = false, $user_to_check = false ){
1747
+ if( $this->event_id == '' && !is_user_logged_in() && get_option('dbem_events_anonymous_submissions') ){
1748
+ $user_to_check = get_option('dbem_events_anonymous_user');
1749
+ }
1750
+ return apply_filters('em_event_can_manage', parent::can_manage($owner_capability, $admin_capability, $user_to_check), $this, $owner_capability, $admin_capability, $user_to_check);
1751
+ }
1752
+ }
1753
+
1754
+ //TODO placeholder targets filtering could be streamlined better
1755
+ /**
1756
+ * This is a temporary filter function which mimicks the old filters in the old 2.x placeholders function
1757
+ * @param string $result
1758
+ * @param EM_Event $event
1759
+ * @param string $placeholder
1760
+ * @param string $target
1761
+ * @return mixed
1762
+ */
1763
+ function em_event_output_placeholder($result,$event,$placeholder,$target='html'){
1764
+ if( ($placeholder == "#_EXCERPT" || $placeholder == "#_LOCATIONEXCERPT") && $target == 'html' ){
1765
+ $result = apply_filters('dbem_notes_excerpt', $result);
1766
+ }elseif( $placeholder == '#_CONTACTEMAIL' && $target == 'html' ){
1767
+ $result = em_ascii_encode($event->get_contact()->user_email);
1768
+ }elseif( $placeholder == "#_NOTES" || $placeholder == "#_EXCERPT" || $placeholder == "#_LOCATIONEXCERPT" ){
1769
+ if($target == 'html'){
1770
+ $result = apply_filters('dbem_notes', $result);
1771
+ }elseif($target == 'map'){
1772
+ $result = apply_filters('dbem_notes_map', $result);
1773
+ }elseif($target == 'ical'){
1774
+ $result = apply_filters('dbem_notes_ical', $result);
1775
+ }else{
1776
+ $result = apply_filters('dbem_notes_rss', $result);
1777
+ $result = apply_filters('the_content_rss', $result);
1778
+ }
1779
+ }elseif( in_array($placeholder, array("#_NAME",'#_ADDRESS','#_LOCATION','#_TOWN')) ){
1780
+ if ($target == "html"){
1781
+ $result = apply_filters('dbem_general', $result);
1782
+ }elseif ($target == "ical"){
1783
+ $result = apply_filters('dbem_general_ical', $result);
1784
+ }else{
1785
+ $result = apply_filters('dbem_general_rss', $result);
1786
+ }
1787
+ }
1788
+ return $result;
1789
+ }
1790
+ add_filter('em_event_output_placeholder','em_event_output_placeholder',1,4);
1791
  ?>
classes/em-events.php CHANGED
@@ -1,254 +1,376 @@
1
- <?php
2
- /**
3
- * Use this class to query and manipulate sets of events. If dealing with more than one event, you probably want to use this class in some way.
4
- *
5
- */
6
- class EM_Events extends EM_Object {
7
-
8
- /**
9
- * Returns an array of EM_Events that match the given specs in the argument, or returns a list of future evetnts in future
10
- * (see EM_Events::get_default_search() ) for explanation of possible search array values. You can also supply a numeric array
11
- * containing the ids of the events you'd like to obtain
12
- *
13
- * @param array $args
14
- * @return EM_Event array()
15
- */
16
- function get( $args = array() ) {
17
- global $wpdb;
18
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
19
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
20
- $categories_table = $wpdb->prefix . EM_CATEGORIES_TABLE;
21
-
22
- //Quick version, we can accept an array of IDs, which is easy to retrieve
23
- if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
24
- //We can just get all the events here and return them
25
- $sql = "
26
- SELECT * FROM $events_table
27
- LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
28
- LEFT JOIN $categories_table ON {$categories_table}.category_id={$events_table}.event_category_id
29
- WHERE event_id=".implode(" OR event_id=", $args)."
30
- ";
31
- $results = $wpdb->get_results(apply_filters('em_events_get_sql',$sql),ARRAY_A);
32
- $events = array();
33
- foreach($results as $result){
34
- $events[$result['event_id']] = new EM_Event($result);
35
- }
36
- return $events; //We return all the events matched as an EM_Event array.
37
- }
38
-
39
- //We assume it's either an empty array or array of search arguments to merge with defaults
40
- $args = self::get_default_search($args);
41
- $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
42
- $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
43
-
44
- //Get the default conditions
45
- $conditions = self::build_sql_conditions($args);
46
- //Put it all together
47
- $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
48
-
49
- //Get ordering instructions
50
- $EM_Event = new EM_Event();
51
- $accepted_fields = $EM_Event->get_fields(true);
52
- $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order'));
53
- //Now, build orderby sql
54
- $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
55
-
56
- //Create the SQL statement and execute
57
- $sql = "
58
- SELECT * FROM $events_table
59
- LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
60
- LEFT JOIN $categories_table ON {$categories_table}.category_id={$events_table}.event_category_id
61
- $where
62
- $orderby_sql
63
- $limit $offset
64
- ";
65
-
66
- $results = $wpdb->get_results( apply_filters('em_events_get_sql',$sql, $args), ARRAY_A);
67
-
68
- //If we want results directly in an array, why not have a shortcut here?
69
- if( $args['array'] == true ){
70
- return $results;
71
- }
72
-
73
- //Make returned results EM_Event objects
74
- $results = (is_array($results)) ? $results:array();
75
- $events = array();
76
- foreach ( $results as $event ){
77
- $events[] = new EM_Event($event);
78
- }
79
-
80
- return apply_filters('em_events_get', $events);
81
- }
82
-
83
- /**
84
- * Returns the number of events on a given date
85
- * @param $date
86
- * @return int
87
- */
88
- function count_date($date){
89
- global $wpdb;
90
- $table_name = $wpdb->prefix . EM_EVENTS_TABLE;
91
- $sql = "SELECT COUNT(*) FROM $table_name WHERE (event_start_date like '$date') OR (event_start_date <= '$date' AND event_end_date >= '$date');";
92
- return apply_filters('em_events_count_date', $wpdb->get_var($sql));
93
- }
94
-
95
- /**
96
- * Will delete given an array of event_ids or EM_Event objects
97
- * @param unknown_type $id_array
98
- */
99
- function delete( $array ){
100
- global $wpdb;
101
- //Detect array type and generate SQL for event IDs
102
- $event_ids = array();
103
- if( @get_class(current($array)) == 'EM_Event' ){
104
- foreach($array as $EM_Event){
105
- $event_ids[] = $EM_Event->id;
106
- }
107
- }else{
108
- $event_ids = $array;
109
- }
110
- if(self::array_is_numeric($event_ids)){
111
- //First we have to check that they're all deletable by this user.
112
- if ( self::can_manage($event_ids) ) {
113
- apply_filters('em_events_delete_pre', $event_ids);
114
- $condition = implode(" OR event_id=", $event_ids);
115
- //Delete all the bookings
116
- $result_bookings = $wpdb->query("DELETE FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." WHERE event_id=$condition;");
117
- //Now delete the events
118
- $result = $wpdb->query ( "DELETE FROM ". $wpdb->prefix . EM_EVENTS_TABLE ." WHERE event_id=$condition;" );
119
- return apply_filters('em_events_delete', true, $event_ids);
120
- }else{
121
- return apply_filters('em_events_delete', true, $event_ids);
122
- }
123
- }
124
- //TODO add better error feedback on events delete fails
125
- return apply_filters('em_events_delete', true, $event_ids);
126
- }
127
-
128
-
129
- /**
130
- * Output a set of matched of events. You can pass on an array of EM_Events as well, in this event you can pass args in second param.
131
- * Note that you can pass a 'pagination' boolean attribute to enable pagination, default is enabled (true).
132
- * @param array $args
133
- * @param array $secondary_args
134
- * @return string
135
- */
136
- function output( $args ){
137
- global $EM_Event;
138
- $EM_Event_old = $EM_Event; //When looping, we can replace EM_Event global with the current event in the loop
139
- //Can be either an array for the get search or an array of EM_Event objects
140
- if( is_object(current($args)) && get_class((current($args))) == 'EM_Event' ){
141
- $func_args = func_get_args();
142
- $events = $func_args[0];
143
- $args = apply_filters('em_events_output_args', self::get_default_search($func_args[1]), $events);
144
- $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
145
- $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
146
- $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
147
- }else{
148
- //Firstly, let's check for a limit/offset here, because if there is we need to remove it and manually do this
149
- $args = apply_filters('em_events_output_args', self::get_default_search($args) );
150
- $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
151
- $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
152
- $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
153
- $args['limit'] = false;
154
- $args['offset'] = false;
155
- $args['page'] = false;
156
- $events = self::get( $args );
157
- }
158
- //What format shall we output this to, or use default
159
- $format = ( empty($args['format']) ) ? get_option( 'dbem_event_list_item_format' ) : $args['format'] ;
160
-
161
- $output = "";
162
- $events_count = count($events);
163
- $events = apply_filters('em_events_output_events', $events);
164
-
165
- if ( $events_count > 0 ) {
166
- $event_count = 0;
167
- $events_shown = 0;
168
- foreach ( $events as $EM_Event ) {
169
- if( ($events_shown < $limit || empty($limit)) && ($event_count >= $offset || $offset === 0) ){
170
- $output .= $EM_Event->output($format);
171
- $events_shown++;
172
- }
173
- $event_count++;
174
- }
175
- //Add headers and footers to output
176
- if( $format == get_option ( 'dbem_event_list_item_format' ) ){
177
- $single_event_format_header = get_option ( 'dbem_event_list_item_format_header' );
178
- $single_event_format_header = ( $single_event_format_header != '' ) ? $single_event_format_header : "<ul class='dbem_events_list'>";
179
- $single_event_format_footer = get_option ( 'dbem_event_list_item_format_footer' );
180
- $single_event_format_footer = ( $single_event_format_footer != '' ) ? $single_event_format_footer : "</ul>";
181
- $output = $single_event_format_header . $output . $single_event_format_footer;
182
- }
183
- //Pagination (if needed/requested)
184
- if( !empty($args['pagination']) && !empty($limit) && $events_count > $limit ){
185
- //Show the pagination links (unless there's less than $limit events)
186
- $page_link_template = preg_replace('/(&|\?)page=\d+/i','',$_SERVER['REQUEST_URI']);
187
- $page_link_template = em_add_get_params($page_link_template, array('page'=>'%PAGE%'));
188
- $output .= apply_filters('em_events_output_pagination', em_paginate( $page_link_template, $events_count, $limit, $page), $page_link_template, $events_count, $limit, $page);
189
- }
190
- } else {
191
- $output = get_option ( 'dbem_no_events_message' );
192
- }
193
- //TODO check if reference is ok when restoring object, due to changes in php5 v 4
194
- $EM_Event = $EM_Event_old;
195
- $output = apply_filters('em_events_output', $output, $events, $args);
196
- return $output;
197
- }
198
-
199
- function can_manage($event_ids){
200
- global $wpdb;
201
- if( em_verify_admin() ){
202
- return apply_filters('em_events_can_manage', true, $event_ids);
203
- }
204
- if( EM_Object::array_is_numeric($event_ids) ){
205
- $condition = implode(" OR event_id=", $event_ids);
206
- //Delete all the bookings
207
- $results = $wpdb->query("SELECT event_author FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." WHERE author_id != '". get_current_user_id() ."' event_id=$condition;");
208
- return apply_filters('em_events_can_manage', (count($results) > 0), $event_ids);
209
- }
210
- return apply_filters('em_events_can_manage', false, $event_ids);
211
- }
212
-
213
- /* Overrides EM_Object method to apply a filter to result
214
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
215
- */
216
- function build_sql_conditions( $args = array() ){
217
- return apply_filters( 'em_events_build_sql_conditions', parent::build_sql_conditions($args), $args );
218
- }
219
-
220
- /* Overrides EM_Object method to apply a filter to result
221
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
222
- */
223
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
224
- return apply_filters( 'em_events_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
225
- }
226
-
227
- /*
228
- * Adds custom Events search defaults
229
- * @param array $array
230
- * @return array
231
- * @uses EM_Object#get_default_search()
232
- */
233
- function get_default_search( $array = array() ){
234
- $defaults = array(
235
- 'orderby' => get_option('dbem_events_default_orderby'),
236
- 'order' => get_option('dbem_events_default_order'),
237
- 'rsvp' => false //if set to true, only events with bookings enabled are returned
238
- );
239
- if( is_admin() ){
240
- //figure out default owning permissions
241
- switch( get_option('dbem_permissions_events') ){
242
- case 0:
243
- $defaults['owner'] = get_current_user_id();
244
- break;
245
- case 1:
246
- $defaults['owner'] = false;
247
- break;
248
- }
249
- $defaults['owner'] = ( em_verify_admin() ) ? false:$defaults['owner'];
250
- }
251
- return apply_filters('em_events_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
252
- }
253
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
254
  ?>
1
+ <?php
2
+ //TODO EM_Events is currently static, better we make this non-static so we can loop sets of events, and standardize with other objects.
3
+ /**
4
+ * Use this class to query and manipulate sets of events. If dealing with more than one event, you probably want to use this class in some way.
5
+ *
6
+ */
7
+ class EM_Events extends EM_Object implements Iterator {
8
+ /**
9
+ * Array of EM_Event objects
10
+ * @var array EM_Event
11
+ */
12
+ var $events = array();
13
+
14
+ function EM_Events( $args = array() ){
15
+ if( is_array($args) ){
16
+ if( is_object(current($args)) && get_class(current($args)) == 'EM_Event' ){
17
+ $this->events = $args;
18
+ }else{
19
+ $this->events = EM_Events::get($args);
20
+ }
21
+ }else{
22
+ $this->events = EM_Events::get();
23
+ }
24
+ do_action('em_events',$this);
25
+ }
26
+
27
+ /**
28
+ * Returns an array of EM_Events that match the given specs in the argument, or returns a list of future evetnts in future
29
+ * (see EM_Events::get_default_search() ) for explanation of possible search array values. You can also supply a numeric array
30
+ * containing the ids of the events you'd like to obtain
31
+ *
32
+ * @param array $args
33
+ * @return EM_Event array()
34
+ */
35
+ function get( $args = array(), $count=false ) {
36
+ global $wpdb;
37
+ $events_table = EM_EVENTS_TABLE;
38
+ $locations_table = EM_LOCATIONS_TABLE;
39
+
40
+ //Quick version, we can accept an array of IDs, which is easy to retrieve
41
+ if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
42
+ //We can just get all the events here and return them
43
+ $sql = "
44
+ SELECT * FROM $events_table
45
+ LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
46
+ WHERE event_id=".implode(" OR event_id=", $args)."
47
+ ";
48
+ $results = $wpdb->get_results(apply_filters('em_events_get_sql',$sql),ARRAY_A);
49
+ $events = array();
50
+ foreach($results as $result){
51
+ $events[$result['event_id']] = new EM_Event($result);
52
+ }
53
+ return $events; //We return all the events matched as an EM_Event array.
54
+ }
55
+
56
+ //We assume it's either an empty array or array of search arguments to merge with defaults
57
+ $args = self::get_default_search($args);
58
+ $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
59
+ $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
60
+
61
+ //Get the default conditions
62
+ $conditions = self::build_sql_conditions($args);
63
+ //Put it all together
64
+ $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
65
+
66
+ //Get ordering instructions
67
+ $EM_Event = new EM_Event();
68
+ $accepted_fields = $EM_Event->get_fields(true);
69
+ $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order'));
70
+ //Now, build orderby sql
71
+ $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
72
+
73
+ //Create the SQL statement and execute
74
+
75
+ if( EM_MS_GLOBAL ){
76
+ $selectors = ( $count ) ? 'COUNT(*)':$events_table.'.post_id, '.$events_table.'.blog_id';
77
+ }else{
78
+ $selectors = ( $count ) ? 'COUNT(*)':$events_table.'.post_id';
79
+ }
80
+
81
+ $sql = "
82
+ SELECT $selectors FROM $events_table
83
+ LEFT JOIN $locations_table ON {$locations_table}.location_id={$events_table}.location_id
84
+ $where
85
+ $orderby_sql
86
+ $limit $offset
87
+ ";
88
+
89
+ //If we're only counting results, return the number of results
90
+ if( $count ){
91
+ return apply_filters('em_events_get_count', $wpdb->get_var($sql), $args);
92
+ }
93
+ $results = $wpdb->get_results( apply_filters('em_events_get_sql',$sql, $args), ARRAY_A);
94
+
95
+ //If we want results directly in an array, why not have a shortcut here?
96
+ if( $args['array'] == true ){
97
+ return apply_filters('em_events_get_array',$results, $args);
98
+ }
99
+
100
+ //Make returned results EM_Event objects
101
+ $results = (is_array($results)) ? $results:array();
102
+ $events = array();
103
+
104
+ if( EM_MS_GLOBAL ){
105
+ foreach ( $results as $event ){
106
+ $events[] = em_get_event($event['post_id'], $event['blog_id']);
107
+ }
108
+ }else{
109
+ foreach ( $results as $event ){
110
+ $events[] = em_get_event($event['post_id'], 'post_id');
111
+ }
112
+ }
113
+
114
+ return apply_filters('em_events_get', $events, $args);
115
+ }
116
+
117
+ /**
118
+ * Returns the number of events on a given date
119
+ * @param $date
120
+ * @return int
121
+ */
122
+ function count_date($date){
123
+ global $wpdb;
124
+ $table_name = EM_EVENTS_TABLE;
125
+ $sql = "SELECT COUNT(*) FROM $table_name WHERE (event_start_date like '$date') OR (event_start_date <= '$date' AND event_end_date >= '$date');";
126
+ return apply_filters('em_events_count_date', $wpdb->get_var($sql));
127
+ }
128
+
129
+ function count( $args = array() ){
130
+ return apply_filters('em_events_count', self::get($args, true), $args);
131
+ }
132
+
133
+ /**
134
+ * Will delete given an array of event_ids or EM_Event objects
135
+ * @param unknown_type $id_array
136
+ */
137
+ function delete( $array ){
138
+ global $wpdb;
139
+ //Detect array type and generate SQL for event IDs
140
+ $results = array();
141
+ if( !empty($array) && @get_class(current($array)) != 'EM_Event' ){
142
+ $events = self::get($array);
143
+ }else{
144
+ $events = $array;
145
+ }
146
+ foreach ($events as $EM_Event){
147
+ $results[] = $EM_Event->delete();
148
+ }
149
+ //TODO add better error feedback on events delete fails
150
+ return apply_filters('em_events_delete', in_array(false, $results), $event_ids);
151
+ }
152
+
153
+
154
+ /**
155
+ * Output a set of matched of events. You can pass on an array of EM_Events as well, in this event you can pass args in second param.
156
+ * Note that you can pass a 'pagination' boolean attribute to enable pagination, default is enabled (true).
157
+ * @param array $args
158
+ * @param array $secondary_args
159
+ * @return string
160
+ */
161
+ function output( $args ){
162
+ global $EM_Event;
163
+ $EM_Event_old = $EM_Event; //When looping, we can replace EM_Event global with the current event in the loop
164
+ //Can be either an array for the get search or an array of EM_Event objects
165
+ $func_args = func_get_args();
166
+ if( is_object(current($args)) && get_class((current($args))) == 'EM_Event' ){
167
+ $func_args = func_get_args();
168
+ $events = $func_args[0];
169
+ $args = (!empty($func_args[1]) && is_array($func_args[1])) ? $func_args[1] : array();
170
+ $args = apply_filters('em_events_output_args', self::get_default_search($args), $events);
171
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
172
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
173
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
174
+ $events_count = count($events);
175
+ }else{
176
+ //Firstly, let's check for a limit/offset here, because if there is we need to remove it and manually do this
177
+ $args = apply_filters('em_events_output_args', self::get_default_search($args) );
178
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
179
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
180
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
181
+ $args_count = $args;
182
+ $args_count['limit'] = false;
183
+ $args_count['offset'] = false;
184
+ $args_count['page'] = false;
185
+ $events_count = self::count($args_count);
186
+ $events = self::get( $args );
187
+ }
188
+ //What format shall we output this to, or use default
189
+ $format = ( empty($args['format']) ) ? get_option( 'dbem_event_list_item_format' ) : $args['format'] ;
190
+
191
+ $output = "";
192
+ $events = apply_filters('em_events_output_events', $events);
193
+ if ( $events_count > 0 ) {
194
+ foreach ( $events as $EM_Event ) {
195
+ $output .= $EM_Event->output($format);
196
+ }
197
+ //Add headers and footers to output
198
+ if( $format == get_option ( 'dbem_event_list_item_format' ) ){
199
+ $format_header = ( get_option( 'dbem_event_list_item_format_header') == '' ) ? '':get_option ( 'dbem_event_list_item_format_header' );
200
+ $format_footer = ( get_option ( 'dbem_event_list_item_format_footer' ) == '' ) ? '':get_option ( 'dbem_event_list_item_format_footer' );
201
+ }else{
202
+ $format_header = ( !empty($args['format_header']) ) ? $args['format_header']:'';
203
+ $format_footer = ( !empty($args['format_footer']) ) ? $args['format_footer']:'';
204
+ }
205
+ $output = $format_header . $output . $format_footer;
206
+ //Pagination (if needed/requested)
207
+ if( !empty($args['pagination']) && !empty($limit) && $events_count > $limit ){
208
+ //Show the pagination links (unless there's less than $limit events)
209
+ $page_link_template = preg_replace('/(&|\?)page=\d+/i','',$_SERVER['REQUEST_URI']);
210
+ $page_link_template = em_add_get_params($page_link_template, array('page'=>'%PAGE%'), false); //don't html encode, so em_paginate does its thing;
211
+ $output .= apply_filters('em_events_output_pagination', em_paginate( $page_link_template, $events_count, $limit, $page), $page_link_template, $events_count, $limit, $page);
212
+ }
213
+ } else {
214
+ $output = get_option ( 'dbem_no_events_message' );
215
+ }
216
+
217
+ //TODO check if reference is ok when restoring object, due to changes in php5 v 4
218
+ $EM_Event = $EM_Event_old;
219
+ $output = apply_filters('em_events_output', $output, $events, $args);
220
+ return $output;
221
+ }
222
+
223
+ function can_manage($event_ids){
224
+ global $wpdb;
225
+ if( current_user_can('edit_others_events') ){
226
+ return apply_filters('em_events_can_manage', true, $event_ids);
227
+ }
228
+ if( EM_Object::array_is_numeric($event_ids) ){
229
+ $condition = implode(" OR event_id=", $event_ids);
230
+ //we try to find any of these events that don't belong to this user
231
+ $results = $wpdb->get_var("SELECT COUNT(*) FROM ". EM_EVENTS_TABLE ." WHERE event_owner != '". get_current_user_id() ."' event_id=$condition;");
232
+ return apply_filters('em_events_can_manage', ($results == 0), $event_ids);
233
+ }
234
+ return apply_filters('em_events_can_manage', false, $event_ids);
235
+ }
236
+
237
+ function get_post_search($args = array()){
238
+ if( !empty($args['em_search']) && empty($args['search']) ) $args['search'] = $args['em_search'];
239
+ $accepted_searches = apply_filters('em_accepted_searches', array('scope','search','category','country','state','region','town'), $args);
240
+ foreach($_REQUEST as $post_key => $post_value){
241
+ if( in_array($post_key, $accepted_searches) && !empty($post_value) ){
242
+ if(is_array($post_value)){
243
+ $post_value = implode(',',$post_value);
244
+ }
245
+ if($post_value != ',' ){
246
+ $args[$post_key] = $post_value;
247
+ }
248
+ }
249
+ }
250
+ return apply_filters('em_events_get_post_search', $args);
251
+ }
252
+
253
+ /* Overrides EM_Object method to apply a filter to result
254
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
255
+ */
256
+ function build_sql_conditions( $args = array() ){
257
+ $conditions = parent::build_sql_conditions($args);
258
+ if( !empty($args['search']) ){
259
+ $like_search = array('event_name',EM_EVENTS_TABLE.'.post_content','location_name','location_address','location_town','location_postcode','location_state','location_country');
260
+ $conditions['search'] = "(".implode(" LIKE '%{$args['search']}%' OR ", $like_search). " LIKE '%{$args['search']}%')";
261
+ }
262
+ if( array_key_exists('status',$args) && is_numeric($args['status']) ){
263
+ $null = ($args['status'] == 0) ? ' OR `event_status` = 0':'';
264
+ $conditions['status'] = "(`event_status`={$args['status']}{$null} )";
265
+ }elseif( empty($args['status']) || $args['status'] != 'all'){
266
+ $conditions['status'] = "(`event_status` IS NOT NULL )"; //by default, we don't show deleted items
267
+ }
268
+ //private events
269
+ if( !empty($args['private']) ){
270
+ $conditions['private'] = "(`event_private`=0)";
271
+ }elseif( !empty($args['private_only']) ){
272
+ $conditions['private_only'] = "(`event_private`=1)";
273
+ }
274
+ if( is_multisite() && array_key_exists('blog',$args) && is_numeric($args['blog']) ){
275
+ if( is_main_site($args['blog']) ){
276
+ $conditions['blog'] = "(".EM_EVENTS_TABLE.".blog_id={$args['blog']} OR ".EM_EVENTS_TABLE.".blog_id IS NULL)";
277
+ }else{
278
+ $conditions['blog'] = "(".EM_EVENTS_TABLE.".blog_id={$args['blog']})";
279
+ }
280
+ }
281
+ if( $args['bookings'] === 'user' && is_user_logged_in()){
282
+ //get bookings of user
283
+ $EM_Person = new EM_Person(get_current_user_id());
284
+ $booking_ids = $EM_Person->get_bookings(true);
285
+ if( count($booking_ids) > 0 ){
286
+ $conditions['bookings'] = "(event_id IN (SELECT event_id FROM ".EM_BOOKINGS_TABLE." WHERE booking_id IN (".implode(',',$booking_ids).")))";
287
+ }else{
288
+ $conditions['bookings'] = "(event_id = 0)";
289
+ }
290
+ }
291
+ //post search
292
+ if( !empty($args['post_id'])){
293
+ if( is_array($args['post_id']) ){
294
+ $conditions['post_id'] = "(".EM_EVENTS_TABLE.".post_id IN (".implode(',',$args['post_id'])."))";
295
+ }else{
296
+ $conditions['post_id'] = "(".EM_EVENTS_TABLE.".post_id={$args['post_id']})";
297
+ }
298
+ }
299
+ return apply_filters( 'em_events_build_sql_conditions', $conditions, $args );
300
+ }
301
+
302
+ /* Overrides EM_Object method to apply a filter to result
303
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
304
+ */
305
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
306
+ return apply_filters( 'em_events_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
307
+ }
308
+
309
+ /*
310
+ * Adds custom Events search defaults
311
+ * @param array $array
312
+ * @return array
313
+ * @uses EM_Object#get_default_search()
314
+ */
315
+ function get_default_search( $array = array() ){
316
+ $defaults = array(
317
+ 'orderby' => get_option('dbem_events_default_orderby'),
318
+ 'order' => get_option('dbem_events_default_order'),
319
+ 'bookings' => false, //if set to true, only events with bookings enabled are returned
320
+ 'status' => 1, //approved events only
321
+ 'format_header' => '', //events can have custom html above the list
322
+ 'format_footer' => '', //events can have custom html below the list
323
+ 'town' => false,
324
+ 'state' => false,
325
+ 'country' => false,
326
+ 'region' => false,
327
+ 'blog' => get_current_blog_id(),
328
+ 'private' => !current_user_can('read_private_events'),
329
+ 'private_only' => false,
330
+ 'post_id' => false
331
+ );
332
+ if(is_multisite()){
333
+ global $bp;
334
+ if( !is_main_site() && !array_key_exists('blog', $array) ){
335
+ $array['blog'] = get_current_blog_id();
336
+ }elseif( empty($array['blog']) && get_site_option('dbem_ms_global_events') ) {
337
+ $array['blog'] = false;
338
+ }
339
+ }
340
+ if( is_admin() ){
341
+ //figure out default owning permissions
342
+ $defaults['owner'] = !current_user_can('edit_others_events') ? get_current_user_id() : false;
343
+ if( !array_key_exists('status', $array) && current_user_can('edit_others_events') ){
344
+ $defaults['status'] = false; //by default, admins see pending and live events
345
+ }
346
+ }
347
+ return apply_filters('em_events_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
348
+ }
349
+
350
+ //TODO Implement object and interators for handling groups of events.
351
+ public function rewind(){
352
+ reset($this->events);
353
+ }
354
+
355
+ public function current(){
356
+ $var = current($this->events);
357
+ return $var;
358
+ }
359
+
360
+ public function key(){
361
+ $var = key($this->events);
362
+ return $var;
363
+ }
364
+
365
+ public function next(){
366
+ $var = next($this->events);
367
+ return $var;
368
+ }
369
+
370
+ public function valid(){
371
+ $key = key($this->events);
372
+ $var = ($key !== NULL && $key !== FALSE);
373
+ return $var;
374
+ }
375
+ }
376
  ?>
classes/em-location-post-admin.php ADDED
@@ -0,0 +1,123 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Location_Post_Admin{
3
+ function init(){
4
+ global $pagenow;
5
+ if($pagenow == 'post.php' || $pagenow == 'post-new.php' ){ //only needed if editing post
6
+ add_action('admin_head', array('EM_Location_Post_Admin','admin_head'));
7
+ }
8
+ //Meta Boxes
9
+ add_action('add_meta_boxes', array('EM_Location_Post_Admin','meta_boxes'));
10
+ //Save/Edit actions
11
+ add_action('save_post',array('EM_Location_Post_Admin','save_post'));
12
+ add_action('before_delete_post',array('EM_Location_Post_Admin','before_delete_post'),10,1);
13
+ add_action('trashed_post',array('EM_Location_Post_Admin','trashed_post'),10,1);
14
+ add_action('untrash_post',array('EM_Location_Post_Admin','untrash_post'),10,1);
15
+ add_action('untrashed_post',array('EM_Location_Post_Admin','untrashed_post'),10,1);
16
+ //Notices
17
+ add_action('admin_notices',array('EM_Location_Post_Admin','admin_notices'));
18
+ add_action('post_updated_messages',array('EM_Location_Post_Admin','admin_notices_filter'),1,1);
19
+ }
20
+
21
+ function admin_head(){
22
+ global $post, $EM_Location;
23
+ if( !empty($post) && $post->post_type == EM_POST_TYPE_LOCATION ){
24
+ $EM_Location = em_get_location($post);
25
+ }
26
+ }
27
+
28
+ function admin_notices(){
29
+ //When editing
30
+ global $post, $EM_Notices;
31
+ if( !empty($post) && $post->post_type == EM_POST_TYPE_LOCATION){
32
+ }
33
+ }
34
+
35
+ function admin_notices_filter($messages){
36
+ //When editing
37
+ global $post, $EM_Notices;
38
+ if( $post->post_type == EM_POST_TYPE_LOCATION ){
39
+ if ( $EM_Notices->count_errors() > 0 ) {
40
+ unset($_GET['message']);
41
+ }
42
+ }
43
+ return $messages;
44
+ }
45
+
46
+ function save_post($post_id){
47
+ global $wpdb, $EM_Location, $EM_Notices;
48
+ $saving_status = !in_array(get_post_status($post_id), array('trash','auto-draft')) && !defined('DOING_AUTOSAVE');
49
+ $is_post_type = get_post_type($post_id) == EM_POST_TYPE_LOCATION;
50
+ if(!defined('UNTRASHING_'.$post_id) && $is_post_type && $saving_status){
51
+ if( wp_verify_nonce($_REQUEST['_emnonce'], 'edit_location')){
52
+ do_action('em_location_save_pre', $this);
53
+ $EM_Location = em_get_location($post_id, 'post_id');
54
+ $get_meta = $EM_Location->get_post_meta();
55
+ $save_meta = $EM_Location->save_meta();
56
+ //Handle Errors by making post draft
57
+ if( !$get_meta || !$save_meta ){
58
+ $EM_Location->set_status(null, true);
59
+ $EM_Notices->add_error( '<strong>'.sprintf(__('Your %s details are incorrect and cannot be published, please correct these errors first:','dbem'),__('location','dbem')).'</strong>', true); //Always seems to redirect, so we make it static
60
+ $EM_Notices->add_error($EM_Location->get_errors(), true); //Always seems to redirect, so we make it static
61
+ apply_filters('em_location_save', false , $this);
62
+ }else{
63
+ apply_filters('em_location_save', true , $this);
64
+ }
65
+ }else{
66
+ //do a quick and dirty update
67
+ do_action('em_location_save_pre', $this);
68
+ $EM_Location = new EM_Location($post_id, 'post_id');
69
+ $location_status = ($EM_Location->is_published()) ? 1:0;
70
+ $wpdb->query("UPDATE ".EM_LOCATIONS_TABLE." SET location_name='{$EM_Location->location_name}', location_slug='{$EM_Location->location_slug}', location_private='{$EM_Location->location_private}',location_status={$location_status} WHERE location_id='{$EM_Location->location_id}'");
71
+ apply_filters('em_location_save', true , $this);
72
+ }
73
+ }
74
+ }
75
+
76
+ function before_delete_post($post_id){
77
+ if(get_post_type($post_id) == EM_POST_TYPE_LOCATION){
78
+ $EM_Location = em_get_location($post_id,'post_id');
79
+ $EM_Location->delete_meta();
80
+ }
81
+ }
82
+
83
+ function trashed_post($post_id){
84
+ if(get_post_type($post_id) == EM_POST_TYPE_LOCATION){
85
+ global $EM_Notices;
86
+ $EM_Location = em_get_location($post_id,'post_id');
87
+ $EM_Location->set_status(null);
88
+ $EM_Notices->remove_all(); //no validation/notices needed
89
+ }
90
+ }
91
+
92
+ function untrash_post($post_id){
93
+ if(get_post_type($post_id) == EM_POST_TYPE_LOCATION){
94
+ //set a constant so we know this event doesn't need 'saving'
95
+ if(!defined('UNTRASHING_'.$post_id)) define('UNTRASHING_'.$post_id, true);
96
+ }
97
+ }
98
+
99
+ function untrashed_post($post_id){
100
+ if(get_post_type($post_id) == EM_POST_TYPE_LOCATION){
101
+ global $EM_Notices;
102
+ $EM_Location = em_get_location($post_id,'post_id');
103
+ $EM_Location->set_status(1);
104
+ $EM_Notices->remove_all(); //no validation/notices needed
105
+ }
106
+ }
107
+
108
+ function meta_boxes(){
109
+ add_meta_box('em-location-where', __('Where','dbem'), array('EM_Location_Post_Admin','meta_box_where'),EM_POST_TYPE_LOCATION, 'normal','high');
110
+ //add_meta_box('em-location-metadump', __('EM_Location Meta Dump','dbem'), array('EM_Location_Post_Admin','meta_box_metadump'),EM_POST_TYPE_LOCATION, 'normal','high');
111
+ }
112
+
113
+ function meta_box_metadump(){
114
+ global $post,$EM_Location;
115
+ echo "<pre>"; print_r(get_post_custom($post->ID)); echo "</pre>";
116
+ echo "<pre>"; print_r($EM_Location); echo "</pre>";
117
+ }
118
+ function meta_box_where(){
119
+ ?><input type="hidden" name="_emnonce" value="<?php echo wp_create_nonce('edit_location'); ?>" /><?php
120
+ em_locate_template('forms/location/where.php',true);
121
+ }
122
+ }
123
+ add_action('admin_init',array('EM_Location_Post_Admin','init'));
classes/em-location-post.php ADDED
@@ -0,0 +1,67 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Location_Post {
3
+ function init(){
4
+ //Front Side Modifiers
5
+ if( !is_admin() ){
6
+ //override single page with formats?
7
+ add_filter('the_content', array('EM_Location_Post','the_content'));
8
+ //display as page template?
9
+ if( get_option('dbem_cp_locations_template_page') ){
10
+ add_filter('single_template',array('EM_Location_Post','single_template'));
11
+ }
12
+ }
13
+ add_action('parse_query', array('EM_Location_Post','parse_query'));
14
+ }
15
+
16
+ /**
17
+ * Overrides the default post format of a location and can display a location as a page, which uses the page.php template.
18
+ * @param string $template
19
+ * @return string
20
+ */
21
+ function single_template($template){
22
+ global $post;
23
+ if( $post->post_type == EM_POST_TYPE_LOCATION ){
24
+ $template = locate_template(array('page.php','index.php'),false);
25
+ }
26
+ return $template;
27
+ }
28
+
29
+ function the_content( $content ){
30
+ global $post;
31
+ if( $post->post_type == EM_POST_TYPE_LOCATION ){
32
+ if( is_archive() || is_search() ){
33
+ if( get_option('dbem_cp_locations_archive_formats') ){
34
+ $EM_Location = em_get_location($post);
35
+ $content = $EM_Location->output(get_option('dbem_location_list_item_format'));
36
+ }
37
+ }else{
38
+ if( get_option('dbem_cp_locations_formats') && !post_password_required() ){
39
+ $EM_Location = em_get_location($post);
40
+ $content = $EM_Location->output_single();
41
+ }
42
+ }
43
+ }
44
+ return $content;
45
+ }
46
+
47
+ function parse_query( ){
48
+ global $wp_query;
49
+ if( !empty($wp_query->query_vars['post_type']) && $wp_query->query_vars['post_type'] == EM_POST_TYPE_LOCATION ){
50
+ if( is_admin() ){
51
+ $wp_query->query_vars['orderby'] = (!empty($_REQUEST['orderby'])) ? $_REQUEST['orderby']:'title';
52
+ $wp_query->query_vars['order'] = (!empty($_REQUEST['order'])) ? $_REQUEST['order']:'ASC';
53
+ }else{
54
+ if( empty($wp_query->query_vars['location']) ) {
55
+ if( get_option('dbem_locations_default_archive_orderby') == 'title'){
56
+ $wp_query->query_vars['orderby'] = 'title';
57
+ }else{
58
+ $wp_query->query_vars['orderby'] = 'meta_value_num';
59
+ $wp_query->query_vars['meta_key'] = get_option('dbem_locations_default_archive_orderby','_location_country');
60
+ }
61
+ $wp_query->query_vars['order'] = get_option('dbem_locations_default_archive_orderby','ASC');
62
+ }
63
+ }
64
+ }
65
+ }
66
+ }
67
+ EM_Location_Post::init();
classes/em-location-posts-admin.php ADDED
@@ -0,0 +1,68 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Location_Posts_Admin{
3
+ function init(){
4
+ global $pagenow;
5
+ if($pagenow == 'edit.php' && !empty($_REQUEST['post_type']) && $_REQUEST['post_type'] == EM_POST_TYPE_LOCATION ){ //only needed if editing post
6
+ add_action('admin_head', array('EM_Location_Posts_Admin','admin_head'));
7
+ }
8
+ add_filter('manage_edit-location_columns' , array('EM_Location_Posts_Admin','columns_add'));
9
+ add_filter('manage_posts_custom_column' , array('EM_Location_Posts_Admin','columns_output'),10,1 );
10
+ }
11
+
12
+ function admin_head(){
13
+ //quick hacks to make event admin table make more sense for events
14
+ ?>
15
+ <script type="text/javascript">
16
+ jQuery(document).ready( function($){
17
+ $('.inline-edit-date').prev().css('display','none').next().css('display','none').next().css('display','none');
18
+ });
19
+ </script>
20
+ <style>
21
+ table.fixed{ table-layout:auto !important; }
22
+ .tablenav select[name="m"] { display:none; }
23
+ </style>
24
+ <?php
25
+ }
26
+
27
+ function admin_menu(){
28
+ global $menu, $submenu;
29
+ // Add a submenu to the custom top-level menu:
30
+ $plugin_pages = array();
31
+ $plugin_pages[] = add_submenu_page('edit.php?post_type='.EM_POST_TYPE_EVENT, __('Locations', 'dbem'), __('Locations', 'dbem'), 'edit_locations', 'events-manager-locations', "edit.php?post_type=event");
32
+ $plugin_pages = apply_filters('em_create_locationss_submenu',$plugin_pages);
33
+ }
34
+
35
+ function columns_add($columns) {
36
+ unset($columns['author']);
37
+ unset($columns['date']);
38
+ unset($columns['comments']);
39
+ return array_merge($columns, array(
40
+ 'address' => __('Address','dbem'),
41
+ 'town' => __('Town','dbem'),
42
+ 'state' => __('State','dbem'),
43
+ 'country' => __('Country','dbem')
44
+ ));
45
+ }
46
+
47
+ function columns_output( $column ) {
48
+ global $post;
49
+ if( $post->post_type == EM_POST_TYPE_LOCATION ){
50
+ $post = em_get_location($post);
51
+ switch ( $column ) {
52
+ case 'address':
53
+ echo $post->location_address;
54
+ break;
55
+ case 'town':
56
+ echo $post->location_town;
57
+ break;
58
+ case 'state':
59
+ echo $post->location_state;
60
+ break;
61
+ case 'country':
62
+ echo $post->location_country;
63
+ break;
64
+ }
65
+ }
66
+ }
67
+ }
68
+ EM_Location_Posts_Admin::init();
classes/em-location.php CHANGED
@@ -1,307 +1,663 @@
1
- <?php
2
- /**
3
- * Object that holds location info and related functions
4
- * @author marcus
5
- */
6
- class EM_Location extends EM_Object {
7
- //DB Fields
8
- var $id = '';
9
- var $name = '';
10
- var $address = '';
11
- var $town = '';
12
- var $latitude = '';
13
- var $longitude = '';
14
- var $description = '';
15
- var $image_url = '';
16
- var $owner = '';
17
- //Other Vars
18
- var $fields = array(
19
- 'location_id' => array('name'=>'id','type'=>'%d'),
20
- 'location_name' => array('name'=>'name','type'=>'%s'),
21
- 'location_address' => array('name'=>'address','type'=>'%s'),
22
- 'location_town' => array('name'=>'town','type'=>'%s'),
23
- //Not Used - 'location_province' => array('name'=>'province','type'=>'%s'),
24
- 'location_latitude' => array('name'=>'latitude','type'=>'%f'),
25
- 'location_longitude' => array('name'=>'longitude','type'=>'%f'),
26
- 'location_description' => array('name'=>'description','type'=>'%s'),
27
- 'location_owner' => array('name'=>'owner','type'=>'%d')
28
- );
29
- var $required_fields;
30
- var $feedback_message = "";
31
- var $mime_types = array(1 => 'gif', 2 => 'jpg', 3 => 'png');
32
- var $errors = array();
33
-
34
- /**
35
- * Gets data from POST (default), supplied array, or from the database if an ID is supplied
36
- * @param $location_data
37
- * @return null
38
- */
39
- function EM_Location( $location_data = 0 ) {
40
- //Initialize
41
- $this->required_fields = array("name" => __('The location name', 'dbem'), "address" => __('The location address', 'dbem'), "town" => __('The location town', 'dbem'));
42
- if( $location_data != 0 ){
43
- //Load location data
44
- if( is_array($location_data) && isset($location_data['location_name']) ){
45
- $location = $location_data;
46
- }elseif( $location_data > 0 ){
47
- //Retreiving from the database
48
- global $wpdb;
49
- $sql = "SELECT * FROM ". $wpdb->prefix.EM_LOCATIONS_TABLE ." WHERE location_id ='{$location_data}'";
50
- $location = $wpdb->get_row($sql, ARRAY_A);
51
- }
52
- //If gmap is turned off, values may not be returned and set, so we set it here
53
- if(empty($location['location_latitude'])) {
54
- $location['location_latitude'] = 0;
55
- $location['location_longitude'] = 0;
56
- }
57
- //Save into the object
58
- $this->to_object($location, true);
59
- $this->get_image_url();
60
- }
61
- }
62
-
63
- function get_post(){
64
- //We are getting the values via POST or GET
65
- do_action('em_location_get_post_pre', $this);
66
- $location = array();
67
- $location['location_id'] = ( !empty($_POST['location_id']) ) ? $_POST['location_id']:'';
68
- $location['location_name'] = ( !empty($_POST['location_name']) ) ? stripslashes($_POST['location_name']):'';
69
- $location['location_address'] = ( !empty($_POST['location_address']) ) ? stripslashes($_POST['location_address']):'';
70
- $location['location_town'] = ( !empty($_POST['location_town']) ) ? stripslashes($_POST['location_town']):'';
71
- $location['location_latitude'] = ( !empty($_POST['location_latitude']) ) ? $_POST['location_latitude']:'';
72
- $location['location_longitude'] = ( !empty($_POST['location_longitude']) ) ? $_POST['location_longitude']:'';
73
- $location['location_description'] = ( !empty($_POST['content']) ) ? stripslashes($_POST['content']):'';
74
- $this->to_object( apply_filters('em_location_get_post', $location, $this) );
75
- }
76
-
77
- function save(){
78
- global $wpdb, $current_user;
79
- get_currentuserinfo();
80
- do_action('em_location_save_pre', $this);
81
- $table = $wpdb->prefix.EM_LOCATIONS_TABLE;
82
- $data = $this->to_array();
83
- unset($data['location_id']);
84
- unset($data['location_image_url']);
85
- if($this->id != ''){
86
- $where = array( 'location_id' => $this->id );
87
- $wpdb->update($table, $data, $where, $this->get_types($data));
88
- }else{
89
- $this->owner = $current_user->ID; //Record creator of event
90
- $data['location_owner'] = $this->owner;
91
- $wpdb->insert($table, $data, $this->get_types($data));
92
- $this->id = $wpdb->insert_id;
93
- }
94
- $image_upload = $this->image_upload();
95
- return apply_filters('em_location_save', ( $this->id > 0 && $image_upload ), $this, $image_upload);
96
- }
97
-
98
- function delete(){
99
- global $wpdb;
100
- do_action('em_location_delete_pre', $this);
101
- $table_name = $wpdb->prefix.EM_LOCATIONS_TABLE;
102
- $sql = "DELETE FROM $table_name WHERE location_id = '{$this->id}';";
103
- $result = $wpdb->query($sql);
104
- $result = $this->image_delete() && $result;
105
- return apply_filters('em_location_delete', $result, $this);
106
- }
107
-
108
- function get_image_url(){
109
- if($this->image_url == ''){
110
- foreach($this->mime_types as $type) {
111
- $file_path = "/".EM_IMAGE_UPLOAD_DIR."/location-{$this->id}.$type";
112
- if( file_exists( ABSPATH . $file_path) ) {
113
- $result = get_bloginfo('wpurl').$file_path;
114
- $this->image_url = $result;
115
- }
116
- }
117
- }
118
- return apply_filters('em_location_get_image_url', $this->image_url, $this);
119
- }
120
-
121
- function image_delete() {
122
- $file_name= ABSPATH.EM_IMAGE_UPLOAD_DIR."/location-".$this->id;
123
- $result = false;
124
- foreach($this->mime_types as $type) {
125
- if (file_exists($file_name.".".$type))
126
- $result = unlink($file_name.".".$type);
127
- }
128
- return apply_filters('em_location_image_delete', $result, $this);
129
- }
130
-
131
- function image_upload(){
132
- //TODO better image upload error handling
133
- do_action('em_location_image_upload_pre', $this);
134
- $result = true;
135
- if ( !empty($_FILES['location_image']['size']) ) {
136
- if( !file_exists(ABSPATH.EM_IMAGE_UPLOAD_DIR) ){
137
- mkdir(ABSPATH.EM_IMAGE_UPLOAD_DIR, 0777);
138
- }
139
- $this->image_delete();
140
- list($width, $height, $type, $attr) = getimagesize($_FILES['location_image']['tmp_name']);
141
- $image_path = ABSPATH.EM_IMAGE_UPLOAD_DIR."/location-".$this->id.".".$this->mime_types[$type];
142
- if (!move_uploaded_file($_FILES['location_image']['tmp_name'], $image_path)){
143
- $this->errors = __('The image could not be loaded','dbem');
144
- $result = false;
145
- }else{
146
- $result = true;
147
- }
148
- }
149
- return apply_filters('em_location_image_upload', $result, $this);
150
- }
151
-
152
- function load_similar($criteria){
153
- global $wpdb;
154
- if( !empty($criteria['location_name']) && !empty($criteria['location_name']) && !empty($criteria['location_name']) ){
155
- $locations_table = $wpdb->prefix.EM_LOCATIONS_TABLE;
156
- $prepared_sql = $wpdb->prepare("SELECT * FROM $locations_table WHERE location_name = %s AND location_address = %s AND location_town = %s", stripcslashes($criteria['location_name']), stripcslashes($criteria['location_address']), stripcslashes($criteria['location_town']) );
157
- //$wpdb->show_errors(true);
158
- $location = $wpdb->get_row($prepared_sql, ARRAY_A);
159
- if( is_array($location) ){
160
- $this->to_object($location);
161
- }
162
- return apply_filters('em_location_load_similar', $location, $this);
163
- }
164
- return apply_filters('em_location_load_similar', false, $this);
165
- }
166
-
167
- /**
168
- * Validates the location. Should be run during any form submission or saving operation.
169
- * @return boolean
170
- */
171
- function validate(){
172
- foreach ( $this->required_fields as $field => $description) {
173
- if ( $this->$field == "" ) {
174
- $this->errors[] = $description.__(" is missing!", "dbem");
175
- }
176
- }
177
- if ( !empty($_FILES['location_image']) && $_FILES['location_image']['size'] > 0 ) {
178
- if (is_uploaded_file($_FILES['location_image']['tmp_name'])) {
179
- $mime_types = array(1 => 'gif', 2 => 'jpg', 3 => 'png');
180
- $maximum_size = get_option('dbem_image_max_size');
181
- if ($_FILES['location_image']['size'] > $maximum_size){
182
- $this->errors[] = __('The image file is too big! Maximum size:', 'dbem')." $maximum_size";
183
- }
184
- list($width, $height, $type, $attr) = getimagesize($_FILES['location_image']['tmp_name']);
185
- $maximum_width = get_option('dbem_image_max_width');
186
- $maximum_height = get_option('dbem_image_max_height');
187
- if (($width > $maximum_width) || ($height > $maximum_height)) {
188
- $this->errors[] = __('The image is too big! Maximum size allowed:')." $maximum_width x $maximum_height";
189
- }
190
- if (($type!=1) && ($type!=2) && ($type!=3)){
191
- $this->errors[] = __('The image is in a wrong format!');
192
- }
193
- }
194
- }
195
- return apply_filters('em_location_validate', ( count($this->errors) == 0 ), $this);
196
- }
197
-
198
- function has_events(){
199
- global $wpdb;
200
- $events_table = $wpdb->prefix.EM_EVENTS_TABLE;
201
- $sql = "SELECT count(event_id) as events_no FROM $events_table WHERE location_id = {$this->id}";
202
- $affected_events = $wpdb->get_row($sql);
203
- return apply_filters('em_location_has_events', (count($affected_events) > 0), $this);
204
- }
205
-
206
- function can_manage(){
207
- return ( get_option('dbem_permissions_locations') == 2 || $this->owner == get_current_user_id() || empty($this->id) || em_verify_admin() );
208
- }
209
-
210
- function can_use(){
211
- switch( get_option('dbem_permissions_locations') ){
212
- case 0:
213
- return $this->owner == get_current_user_id();
214
- case 1:
215
- return em_verify_admin($this->owner);
216
- case 2:
217
- return true;
218
- }
219
- }
220
-
221
- function output_single($target = 'html'){
222
- $format = get_option ( 'dbem_single_location_format' );
223
- return apply_filters('em_location_output_single', $this->output($format, $target), $this, $target);
224
- }
225
-
226
- function output($format, $target="html") {
227
- $location_string = $format;
228
- preg_match_all("/#_[A-Za-z]+/", $format, $placeholders);
229
- foreach($placeholders[0] as $result) {
230
- $match = true;
231
- $replace = '';
232
- switch( $result ){
233
- case '#_MAP': //Depreciated
234
- case '#_LOCATIONMAP':
235
- $replace = EM_Map::get_single( array('location' => $this) );
236
- break;
237
- case '#_DESCRIPTION': //Depreciated
238
- case '#_EXCERPT': //Depreciated
239
- case '#_LOCATIONNOTES':
240
- case '#_LOCATIONEXCERPT':
241
- $replace = $this->description;
242
- if($result == "#_EXCERPT" || $result == "#_LOCATIONEXCERPT"){
243
- $matches = explode('<!--more', $this->description);
244
- $replace = $matches[0];
245
- }
246
- break;
247
- case '#_LOCATIONURL':
248
- case '#_LOCATIONLINK':
249
- case '#_LOCATIONPAGEURL': //Depreciated
250
- $joiner = (stristr(EM_URI, "?")) ? "&amp;" : "?";
251
- $link = EM_URI.$joiner."location_id=".$this->id;
252
- $replace = ($result == '#_LOCATIONURL' || $result == '#_LOCATIONPAGEURL') ? $link : '<a href="'.$link.'">'.$this->name.'</a>';
253
- break;
254
- case '#_PASTEVENTS': //Depreciated
255
- case '#_LOCATIONPASTEVENTS':
256
- case '#_NEXTEVENTS': //Depreciated
257
- case '#_LOCATIONNEXTEVENTS':
258
- case '#_ALLEVENTS': //Depreciated
259
- case '#_LOCATIONALLEVENTS':
260
- if ($result == '#_PASTEVENTS' || $result == '#_LOCATIONPASTEVENTS'){ $scope = 'past'; }
261
- elseif ( $result == '#_NEXTEVENTS' || $result == '#_LOCATIONNEXTEVENTS' ){ $scope = 'future'; }
262
- else{ $scope = 'all'; }
263
- $events = EM_Events::get( array('location'=>$this->id, 'scope'=>$scope) );
264
- if ( count($events) > 0 ){
265
- foreach($events as $event){
266
- $replace .= $event->output(get_option('dbem_location_event_list_item_format'));
267
- }
268
- } else {
269
- $replace = get_option('dbem_location_no_events_message');
270
- }
271
- break;
272
- case '#_IMAGE': //Depreciated
273
- case '#_LOCATIONIMAGE':
274
- if($this->image_url != ''){
275
- $replace = "<img src='".$this->image_url."' alt='".$this->name."'/>";
276
- }
277
- break;
278
- case '#_NAME': //Depreciated
279
- case '#_LOCATIONNAME':
280
- $replace = $this->name;
281
- break;
282
- case '#_ADDRESS': //Depreciated
283
- case '#_LOCATIONADDRESS':
284
- $replace = $this->address;
285
- break;
286
- case '#_TOWN': //Depreciated
287
- case '#_LOCATIONTOWN':
288
- $replace = $this->town;
289
- break;
290
- case '#_LOCATIONID':
291
- $replace = $this->id;
292
- break;
293
- default:
294
- $match = false;
295
- break;
296
- }
297
- if($match){ //if true, we've got a placeholder that needs replacing
298
- //TODO FILTER - placeholder filter
299
- $replace = apply_filters('em_location_output_placeholder', $replace, $this, $result, $target); //USE WITH CAUTION! THIS MIGHT GET RENAMED
300
- $location_string = str_replace($result, $replace , $location_string );
301
- }
302
- }
303
- $name_filter = ($target == "html") ? 'dbem_general':'dbem_general_rss'; //TODO remove dbem_ filters
304
- $location_string = str_replace('#_LOCATION', apply_filters($name_filter, $this->name) , $location_string ); //Depreciated
305
- return apply_filters('em_location_output', $location_string, $this, $format, $target);
306
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
307
  }
1
+ <?php
2
+ /**
3
+ * gets a location
4
+ * @param mixed $id
5
+ * @param mixed $search_by
6
+ * @return EM_Location
7
+ */
8
+ function em_get_location($id = false, $search_by = 'location_id') {
9
+ if( is_object($id) && get_class($id) == 'EM_Location' ){
10
+ return $id;
11
+ }else{
12
+ return new EM_Location($id,$search_by);
13
+ }
14
+ }
15
+ /**
16
+ * Object that holds location info and related functions
17
+ * @author marcus
18
+ */
19
+ class EM_Location extends EM_Object {
20
+ //DB Fields
21
+ var $location_id = '';
22
+ var $post_id = '';
23
+ var $blog_id = '';
24
+ var $location_slug = '';
25
+ var $location_name = '';
26
+ var $location_address = '';
27
+ var $location_town = '';
28
+ var $location_state = '';
29
+ var $location_postcode = '';
30
+ var $location_region = '';
31
+ var $location_country = '';
32
+ var $location_latitude = 0;
33
+ var $location_longitude = 0;
34
+ var $post_content = '';
35
+ var $location_owner = '';
36
+ var $location_status = 0;
37
+ //Other Vars
38
+ var $fields = array(
39
+ 'location_id' => array('name'=>'id','type'=>'%d'),
40
+ 'post_id' => array('name'=>'post_id','type'=>'%d'),
41
+ 'blog_id' => array('name'=>'blog_id','type'=>'%d'),
42
+ 'location_slug' => array('name'=>'slug','type'=>'%s', 'null'=>true),
43
+ 'location_name' => array('name'=>'name','type'=>'%s', 'null'=>true),
44
+ 'location_address' => array('name'=>'address','type'=>'%s','null'=>true),
45
+ 'location_town' => array('name'=>'town','type'=>'%s','null'=>true),
46
+ 'location_state' => array('name'=>'state','type'=>'%s','null'=>true),
47
+ 'location_postcode' => array('name'=>'postcode','type'=>'%s','null'=>true),
48
+ 'location_region' => array('name'=>'region','type'=>'%s','null'=>true),
49
+ 'location_country' => array('name'=>'country','type'=>'%s','null'=>true),
50
+ 'location_latitude' => array('name'=>'latitude','type'=>'%f','null'=>true),
51
+ 'location_longitude' => array('name'=>'longitude','type'=>'%f','null'=>true),
52
+ 'post_content' => array('name'=>'description','type'=>'%s', 'null'=>true),
53
+ 'location_owner' => array('name'=>'owner','type'=>'%d', 'null'=>true),
54
+ 'location_status' => array('name'=>'status','type'=>'%d', 'null'=>true)
55
+ );
56
+ var $post_fields = array('post_id','location_slug','location_name','post_content','location_owner');
57
+ var $attributes = array();
58
+ var $image_url = '';
59
+ var $required_fields = array();
60
+ var $feedback_message = "";
61
+ var $mime_types = array(1 => 'gif', 2 => 'jpg', 3 => 'png');
62
+ var $errors = array();
63
+ /**
64
+ * previous status of location
65
+ * @access protected
66
+ * @var mixed
67
+ */
68
+ var $previous_status = 0;
69
+
70
+ /* Post Variables - copied out of post object for easy IDE reference */
71
+ var $ID;
72
+ var $post_author;
73
+ var $post_date;
74
+ var $post_date_gmt;
75
+ var $post_title;
76
+ var $post_excerpt;
77
+ var $post_status;
78
+ var $comment_status;
79
+ var $ping_status;
80
+ var $post_password;
81
+ var $post_name;
82
+ var $to_ping;
83
+ var $pinged;
84
+ var $post_modified;
85
+ var $post_modified_gmt;
86
+ var $post_content_filtered;
87
+ var $post_parent;
88
+ var $guid;
89
+ var $menu_order;
90
+ var $post_type;
91
+ var $post_mime_type;
92
+ var $comment_count;
93
+ var $ancestors;
94
+ var $filter;
95
+
96
+
97
+ /**
98
+ * Gets data from POST (default), supplied array, or from the database if an ID is supplied
99
+ * @param $location_data
100
+ * @param $search_by can be set to post_id or a number for a blog id if in ms mode with global tables, default is location_id
101
+ * @return null
102
+ */
103
+ function __construct($id = false, $search_by = 'location_id' ) {
104
+ global $wpdb;
105
+ //Initialize
106
+ $this->required_fields = array("location_address" => __('The location address', 'dbem'), "location_town" => __('The location town', 'dbem'), "location_country" => __('The country', 'dbem'));
107
+ //Get the post_id/location_id
108
+ $is_post = !empty($id->ID) && $id->post_type == EM_POST_TYPE_LOCATION;
109
+ if( $is_post || absint($id) > 0 ){ //only load info if $id is a number
110
+ $location_post = false;
111
+ if($search_by == 'location_id' && !$is_post){
112
+ //search by location_id, get post_id and blog_id (if in ms mode) and load the post
113
+ $results = $wpdb->get_row($wpdb->prepare("SELECT post_id, blog_id FROM ".EM_LOCATIONS_TABLE." WHERE location_id=%d",$id), ARRAY_A);
114
+ if( !empty($results['post_id']) ){
115
+ if( is_multisite() && is_numeric($results['blog_id']) ){
116
+ $location_post = get_blog_post($results['blog_id'], $results['post_id']);
117
+ $search_by = $results['blog_id'];
118
+ }else{
119
+ $location_post = get_post($results['post_id']);
120
+ }
121
+ }
122
+ }else{
123
+ if(!$is_post){
124
+ if( is_numeric($search_by) && is_multisite() ){
125
+ //we've been given a blog_id, so we're searching for a post id
126
+ $location_post = get_blog_post($search_by, $id);
127
+ }else{
128
+ //search for the post id only
129
+ $location_post = get_post($id);
130
+ }
131
+ }else{
132
+ $location_post = $id;
133
+ }
134
+ }
135
+ $this->load_postdata($location_post, $search_by);
136
+ }
137
+ $this->compat_keys();
138
+ do_action('em_location', $this, $id, $search_by);
139
+ }
140
+
141
+ function load_postdata($location_post, $search_by = false){
142
+ if( is_object($location_post) ){
143
+ if( $location_post->post_status != 'auto-draft' ){
144
+ if( is_numeric($search_by) && is_multisite() ){
145
+ // if in multisite mode, switch blogs quickly to get the right post meta.
146
+ switch_to_blog($search_by);
147
+ $location_meta = get_post_custom($location_post->ID);
148
+ restore_current_blog();
149
+ $this->blog_id = $search_by;
150
+ }else{
151
+ $location_meta = get_post_custom($location_post->ID);
152
+ }
153
+ //Get custom fields
154
+ foreach($location_meta as $location_meta_key => $location_meta_val){
155
+ $found = false;
156
+ foreach($this->fields as $field_name => $field_info){
157
+ if( $location_meta_key == '_'.$field_name){
158
+ $this->$field_name = $location_meta_val[0];
159
+ $found = true;
160
+ }
161
+ }
162
+ if(!$found && $location_meta_key[0] != '_'){
163
+ $this->attributes[$location_meta_key] = ( count($location_meta_val) > 1 ) ? $location_meta_val:$location_meta_val[0];
164
+ }
165
+ }
166
+ }
167
+ //load post data - regardless
168
+ $this->post_id = $location_post->ID;
169
+ $this->location_name = $location_post->post_title;
170
+ $this->location_slug = $location_post->post_name;
171
+ $this->location_owner = $location_post->post_author;
172
+ $this->post_content = $location_post->post_content;
173
+ foreach( $location_post as $key => $value ){ //merge the post data into location object
174
+ $this->$key = $value;
175
+ }
176
+ $this->previous_status = $this->location_status; //so we know about updates
177
+ $this->get_status();
178
+ }
179
+ }
180
+
181
+ /**
182
+ * Retrieve event information via POST (used in situations where posts aren't submitted via WP)
183
+ * @param boolean $validate whether or not to run validation, default is true
184
+ * @return boolean
185
+ */
186
+ function get_post($validate = true){
187
+ do_action('em_location_get_post_pre', $this);
188
+ $this->location_name = ( !empty($_POST['location_name']) ) ? wp_kses($_POST['location_name'], array()):'';
189
+ $this->post_content = ( !empty($_POST['content']) ) ? wp_kses($_POST['content'], array()):'';
190
+ $this->get_post_meta(false);
191
+ $result = $validate ? $this->validate():true; //validate both post and meta, otherwise return true
192
+ $this->compat_keys();
193
+ return apply_filters('em_location_get_post', $result, $this);
194
+ }
195
+ /**
196
+ * Retrieve event post meta information via POST, which should be always be called when saving the event custom post via WP.
197
+ * @param boolean $validate whether or not to run validation, default is true
198
+ * @return mixed
199
+ */
200
+ function get_post_meta($validate = true){
201
+ //We are getting the values via POST or GET
202
+ do_action('em_location_get_post_meta_pre', $this);
203
+ $this->location_address = ( !empty($_POST['location_address']) ) ? wp_kses($_POST['location_address'], array()):'';
204
+ $this->location_town = ( !empty($_POST['location_town']) ) ? wp_kses($_POST['location_town'], array()):'';
205
+ $this->location_state = ( !empty($_POST['location_state']) ) ? wp_kses($_POST['location_state'], array()):'';
206
+ $this->location_postcode = ( !empty($_POST['location_postcode']) ) ? wp_kses($_POST['location_postcode'], array()):'';
207
+ $this->location_region = ( !empty($_POST['location_region']) ) ? wp_kses($_POST['location_region'], array()):'';
208
+ $this->location_country = ( !empty($_POST['location_country']) ) ? wp_kses($_POST['location_country'], array()):'';
209
+ $this->location_latitude = ( !empty($_POST['location_latitude']) && is_numeric($_POST['location_latitude']) ) ? $_POST['location_latitude']:'';
210
+ $this->location_longitude = ( !empty($_POST['location_longitude']) && is_numeric($_POST['location_longitude']) ) ? $_POST['location_longitude']:'';
211
+ //Set Blog ID
212
+ if( is_multisite() && empty($this->blog_id) ){
213
+ $this->blog_id = get_current_blog_id();
214
+ }
215
+ $result = $validate ? $this->validate_meta():true; //post returns null
216
+ $this->compat_keys();
217
+ return apply_filters('em_location_get_post_meta',$result,$this);
218
+ }
219
+
220
+ function validate(){
221
+ $validate_post = true;
222
+ if( empty($this->location_name) ){
223
+ $validate_post = false;
224
+ $this->add_error( __('Location name').__(" is required.", "dbem") );
225
+ }
226
+ $validate_meta = $this->validate_meta();
227
+ return apply_filters('em_location_validate', $validate_post && $validate_meta, $this );
228
+ }
229
+
230
+ /**
231
+ * Validates the location. Should be run during any form submission or saving operation.
232
+ * @return boolean
233
+ */
234
+ function validate_meta(){
235
+ //check required fields
236
+ foreach ( $this->required_fields as $field => $description) {
237
+ if( $field == 'location_country' && !array_key_exists($this->location_country, em_get_countries()) ){
238
+ //country specific checking
239
+ $this->add_error( $this->required_fields['location_country'].__(" is required.", "dbem") );
240
+ }elseif ( $this->$field == "" ) {
241
+ $this->add_error( $description.__(" is required.", "dbem") );
242
+ }
243
+ }
244
+ return apply_filters('em_location_validate_meta', ( count($this->errors) == 0 ), $this);
245
+ }
246
+
247
+ function save(){
248
+ global $wpdb, $current_user, $blog_id;
249
+ //TODO shuffle filters into right place
250
+ if( !$this->can_manage('edit_locations', 'edit_others_locations') && !( get_option('dbem_events_anonymous_submissions') && empty($this->location_id)) ){
251
+ return apply_filters('em_location_save', false, $this);
252
+ }
253
+ remove_action('save_post',array('EM_Location_Post_Admin','save_post'),10,1); //disable the default save post action, we'll do it manually this way
254
+ do_action('em_location_save_pre', $this);
255
+ $post_array = array();
256
+ //Deal with updates to a location
257
+ if( !empty($this->post_id) ){
258
+ //get the full array of post data so we don't overwrite anything.
259
+ if( !empty($this->blog_id) && is_multisite() ){
260
+ $post_array = (array) get_blog_post($this->blog_id, $this->post_id);
261
+ }else{
262
+ $post_array = (array) get_post($this->post_id);
263
+ }
264
+ }
265
+ //Overwrite new post info
266
+ $post_array['post_type'] = EM_POST_TYPE_LOCATION;
267
+ $post_array['post_title'] = $this->location_name;
268
+ $post_array['post_content'] = $this->post_content;
269
+ //decide on post status
270
+ if( count($this->errors) == 0 ){
271
+ if( EM_MS_GLOBAL && !is_main_site() && get_site_option('dbem_ms_mainblog_locations') ){
272
+ //if in global ms mode and user is a valid role to publish on their blog, then we will publish the location on the main blog
273
+ restore_current_blog();
274
+ $switch_back = true;
275
+ }
276
+ $post_array['post_status'] = ( current_user_can('publish_locations') ) ? 'publish':'pending';
277
+ if(!empty($switch_back) && get_site_option('dbem_ms_mainblog_locations') ) EM_Object::ms_global_switch(); //switch 'back' to main blog
278
+ }else{
279
+ $post_array['post_status'] = 'draft';
280
+ }
281
+ //Anonymous submission
282
+ if( !is_user_logged_in() && get_option('dbem_events_anonymous_submissions') && empty($this->location_id) ){
283
+ $post_array['post_author'] = get_option('dbem_events_anonymous_user');
284
+ if( !is_numeric($post_array['post_author']) ) $post_array['post_author'] = 0;
285
+ }
286
+ //Save post and continue with meta
287
+ $post_id = wp_insert_post($post_array);
288
+ $post_save = false;
289
+ $meta_save = false;
290
+ if( !is_wp_error($post_id) && !empty($post_id) ){
291
+ $post_save = true;
292
+ //refresh this event with wp post
293
+ $post_data = get_post($post_id);
294
+ $this->post_id = $post_id;
295
+ $this->location_slug = $post_data->post_name;
296
+ $this->location_owner = $post_data->post_author;
297
+ $this->post_status = $post_data->post_status;
298
+ $this->get_status();
299
+ //now save the meta
300
+ $meta_save = $this->save_meta();
301
+ //save the image
302
+ $this->image_upload();
303
+ $image_save = (count($this->errors) == 0);
304
+ }elseif(is_wp_error($post_id)){
305
+ //location not saved, add an error
306
+ $this->add_error($post_id->get_error_message());
307
+ }
308
+ return apply_filters('em_location_save', $post_save && $meta_save && $image_save, $this);
309
+ }
310
+
311
+ function save_meta(){
312
+ //echo "<pre>"; print_r($this); echo "</pre>"; die();
313
+ global $wpdb, $current_user;
314
+ if( $this->can_manage('edit_locations','edit_others_locations') || ( get_option('dbem_events_anonymous_submissions') && empty($this->location_id)) ){
315
+ do_action('em_location_save_meta_pre', $this);
316
+ $data = $this->to_array();
317
+ //Update Post Meta
318
+ foreach( array_keys($this->fields) as $key ){
319
+ if( !in_array($key, $this->post_fields) ){
320
+ update_post_meta($this->post_id, '_'.$key, $this->$key);
321
+ }
322
+ }
323
+ $this->get_status();
324
+ $this->location_status = (count($this->errors) == 0) ? $this->location_status:null; //set status at this point, it's either the current status, or if validation fails, null
325
+ //Save to em_locations table
326
+ $location_array = $this->to_array(true);
327
+ if( $this->post_status == 'private' ) $location_array['location_private'] = 1;
328
+ unset($location_array['location_id']);
329
+ if( empty($this->location_id) ){
330
+ $this->previous_status = 0; //for sure this was previously status 0
331
+ if ( !$wpdb->insert(EM_LOCATIONS_TABLE, $location_array) ){
332
+ $this->add_error( sprintf(__('Something went wrong saving your %s to the index table. Please inform a site administrator about this.','dbem'),__('location','dbem')));
333
+ }else{
334
+ //success, so link the event with the post via an event id meta value for easy retrieval
335
+ $this->location_id = $wpdb->insert_id;
336
+ update_post_meta($this->post_id, '_location_id', $this->location_id);
337
+ $this->feedback_message = sprintf(__('Successfully saved %s','dbem'),__('Location','dbem'));
338
+ }
339
+ }else{
340
+ $this->previous_status = $wpdb->get_var('SELECT location_status FROM '.EM_LOCATIONS_TABLE.' WHERE location_id='.$this->location_id); //get status from db, not post_status
341
+ if ( $wpdb->update(EM_LOCATIONS_TABLE, $location_array, array('location_id'=>$this->location_id)) === false ){
342
+ $this->add_error( sprintf(__('Something went wrong updating your %s to the index table. Please inform a site administrator about this.','dbem'),__('location','dbem')));
343
+ }else{
344
+ $this->feedback_message = sprintf(__('Successfully saved %s','dbem'),__('Location','dbem'));
345
+ }
346
+ }
347
+ }else{
348
+ $this->add_error( sprintf(__('You do not have permission to create/edit %s.','dbem'), __('locations','dbem')) );
349
+ }
350
+ $this->compat_keys();
351
+ return apply_filters('em_location_save_meta', count($this->errors) == 0, $this);
352
+ }
353
+
354
+ function delete($force_delete = true){ //atm wp seems to force cp deletions anyway
355
+ global $wpdb;
356
+ $result = false;
357
+ if( $this->can_manage('delete_locations','delete_others_locations') ){
358
+ do_action('em_location_delete_pre', $this);
359
+ $result = wp_delete_post($this->post_id,$force_delete); //the post class will take care of the meta
360
+ if( $force_delete ){
361
+ $result_meta = $this->delete_meta();
362
+ }
363
+ }
364
+ return apply_filters('em_location_delete', $result !== false && $result_meta, $this);
365
+ }
366
+
367
+ function delete_meta(){
368
+ global $wpdb;
369
+ $result = false;
370
+ if( $this->can_manage('delete_locations','delete_others_locations') ){
371
+ do_action('em_location_delete_meta_pre', $this);
372
+ $result = $wpdb->query ( $wpdb->prepare("DELETE FROM ". EM_LOCATIONS_TABLE ." WHERE location_id=%d", $this->location_id) );
373
+ }
374
+ return apply_filters('em_location_delete_meta', $result !== false, $this);
375
+ }
376
+
377
+ function is_published(){
378
+ return apply_filters('em_location_is_published', ($this->post_status == 'publish' || $this->post_status == 'private'), $this);
379
+ }
380
+
381
+ /**
382
+ * Change the status of the location. This will save to the Database too.
383
+ * @param int $status
384
+ * @param boolean $set_post_status
385
+ * @return string
386
+ */
387
+ function set_status($status, $set_post_status = false){
388
+ global $wpdb;
389
+ if($status === null){
390
+ $set_status='NULL';
391
+ if($set_post_status){
392
+ //if the post is trash, don't untrash it!
393
+ $wpdb->update( $wpdb->posts, array( 'post_status' => 'draft' ), array( 'ID' => $this->post_id ) );
394
+ $this->post_status = 'draft';
395
+ }
396
+ }else{
397
+ $set_status = $status ? 1:0;
398
+ if($set_post_status){
399
+ if($this->post_status == 'pending'){
400
+ $this->post_name = sanitize_title($this->post_title);
401
+ }
402
+ $this->post_status = $set_status ? 'publish':'pending';
403
+ $wpdb->update( $wpdb->posts, array( 'post_status' => $this->post_status, 'post_name' => $this->post_name ), array( 'ID' => $this->post_id ) );
404
+ }
405
+ }
406
+ $this->previous_status = $wpdb->get_var('SELECT location_status FROM '.EM_LOCATIONS_TABLE.' WHERE location_id='.$this->location_id); //get status from db, not post_status, as posts get saved quickly
407
+ $result = $wpdb->query("UPDATE ".EM_LOCATIONS_TABLE." SET location_status=$set_status, location_slug='{$this->post_name}' WHERE location_id={$this->location_id}");
408
+ $this->get_status();
409
+ return apply_filters('em_location_set_status', $result !== false, $status, $this);
410
+ }
411
+
412
+ function get_status($db = false){
413
+ switch( $this->post_status ){
414
+ case 'private':
415
+ $this->location_private = 1;
416
+ $this->location_status = $status = 1;
417
+ break;
418
+ case 'publish':
419
+ $this->location_private = 0;
420
+ $this->location_status = $status = 1;
421
+ break;
422
+ case 'pending':
423
+ $this->location_private = 0;
424
+ $this->location_status = $status = 0;
425
+ break;
426
+ default: //draft or unknown
427
+ $this->location_private = 0;
428
+ $status = $db ? 'NULL':null;
429
+ $this->location_status = null;
430
+ break;
431
+ }
432
+ return $status;
433
+ }
434
+
435
+ function load_similar($criteria){
436
+ global $wpdb;
437
+ if( !empty($criteria['location_name']) && !empty($criteria['location_name']) && !empty($criteria['location_name']) ){
438
+ $locations_table = EM_LOCATIONS_TABLE;
439
+ $prepared_sql = $wpdb->prepare("SELECT * FROM $locations_table WHERE location_name = %s AND location_address = %s AND location_town = %s AND location_state = %s AND location_postcode = %s AND location_country = %s", stripcslashes($criteria['location_name']), stripcslashes($criteria['location_address']), stripcslashes($criteria['location_town']), stripcslashes($criteria['location_state']), stripcslashes($criteria['location_postcode']), stripcslashes($criteria['location_country']) );
440
+ //$wpdb->show_errors(true);
441
+ $location = $wpdb->get_row($prepared_sql, ARRAY_A);
442
+ if( is_array($location) ){
443
+ $this->to_object($location);
444
+ }
445
+ return apply_filters('em_location_load_similar', $location, $this);
446
+ }
447
+ return apply_filters('em_location_load_similar', false, $this);
448
+ }
449
+
450
+ function has_events(){
451
+ global $wpdb;
452
+ $events_table = EM_EVENTS_TABLE;
453
+ $sql = "SELECT count(event_id) as events_no FROM $events_table WHERE location_id = {$this->location_id}";
454
+ $affected_events = $wpdb->get_row($sql);
455
+ return apply_filters('em_location_has_events', (count($affected_events) > 0), $this);
456
+ }
457
+
458
+ /**
459
+ * Can the user manage this location?
460
+ */
461
+ function can_manage( $owner_capability = false, $admin_capability = false, $user_to_check = false ){
462
+ if( $this->location_id == '' && !is_user_logged_in() && get_option('dbem_events_anonymous_submissions') ){
463
+ $user_to_check = get_option('dbem_events_anonymous_user');
464
+ }
465
+ return apply_filters('em_location_can_manage', parent::can_manage($owner_capability, $admin_capability, $user_to_check), $this, $owner_capability, $admin_capability, $user_to_check);
466
+ }
467
+
468
+ function get_permalink(){
469
+ if( EM_MS_GLOBAL ){
470
+ if( get_site_option('dbem_ms_global_locations_links') && !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
471
+ //linking directly to the blog, we should be on the main blog here
472
+ $link = get_blog_permalink( $this->blog_id, $this->post_id);
473
+ }elseif( !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
474
+ if( get_option('dbem_locations_page') ){
475
+ $link = trailingslashit(get_permalink(get_option('dbem_locations_page')).get_site_option('dbem_ms_locations_slug',EM_LOCATION_SLUG).'/'.$this->location_slug.'-'.$this->location_id);
476
+ }else{
477
+ $link = trailingslashit(home_url()).EM_POST_TYPE_LOCATION_SLUG.'/'.get_site_option('dbem_ms_events_slug',EM_LOCATION_SLUG).'/'.$this->location_slug.'-'.$this->location_id;
478
+ }
479
+ }
480
+ }
481
+ if( empty($link) ){
482
+ $link = get_post_permalink($this->post_id);
483
+ }
484
+ return apply_filters('em_location_get_permalink', $link, $this); ;
485
+ }
486
+
487
+ function get_edit_url(){
488
+ if( $this->can_manage('edit_locations','edit_others_locations') ){
489
+ if( EM_MS_GLOBAL && get_site_option('dbem_ms_global_locations_links') && !empty($this->blog_id) && is_main_site() && $this->blog_id != get_current_blog_id() ){
490
+ if( get_blog_option($this->blog_id, 'dbem_edit_locations_page') ){
491
+ $link = em_add_get_params(get_permalink(get_blog_option($this->blog_id, 'dbem_edit_locations_page')), array('action'=>'edit','location_id'=>$this->location_id), false);
492
+ }
493
+ if( empty($link))
494
+ $link = get_admin_url($this->blog_id, "post.php?post={$this->post_id}&action=edit");
495
+ }else{
496
+ if( get_option('dbem_edit_locations_page') ){
497
+ $link = em_add_get_params(get_permalink(get_option('dbem_edit_locations_page')), array('action'=>'edit','location_id'=>$this->location_id), false);
498
+ }
499
+ if( empty($link))
500
+ $link = admin_url()."post.php?post={$this->post_id}&action=edit";
501
+ }
502
+ return apply_filters('em_location_get_edit_url', $link, $this);
503
+ }
504
+ }
505
+
506
+ function output_single($target = 'html'){
507
+ $format = get_option ( 'dbem_single_location_format' );
508
+ return apply_filters('em_location_output_single', $this->output($format, $target), $this, $target);
509
+ }
510
+
511
+ function output($format, $target="html") {
512
+ preg_match_all('/\{([a-zA-Z0-9_]+)\}([^{]+)\{\/[a-zA-Z0-9_]+\}/', $format, $conditionals);
513
+ if( count($conditionals[0]) > 0 ){
514
+ //Check if the language we want exists, if not we take the first language there
515
+ foreach($conditionals[1] as $key => $condition){
516
+ $format = str_replace($conditionals[0][$key], apply_filters('em_location_output_condition', '', $conditionals[0][$key], $condition, $this), $format);
517
+ }
518
+ }
519
+ $location_string = $format;
520
+ preg_match_all("/(#@?_?[A-Za-z0-9]+)({([a-zA-Z0-9,]+)})?/", $format, $placeholders);
521
+ foreach($placeholders[1] as $key => $result) {
522
+ $replace = '';
523
+ $full_result = $placeholders[0][$key];
524
+ switch( $result ){
525
+ case '#_LOCATIONID':
526
+ $replace = $this->location_id;
527
+ break;
528
+ case '#_LOCATIONPOSTID':
529
+ $replace = $this->location_id;
530
+ break;
531
+ case '#_NAME': //Depreciated
532
+ case '#_LOCATIONNAME':
533
+ $replace = $this->location_name;
534
+ break;
535
+ case '#_ADDRESS': //Depreciated
536
+ case '#_LOCATIONADDRESS':
537
+ $replace = $this->location_address;
538
+ break;
539
+ case '#_TOWN': //Depreciated
540
+ case '#_LOCATIONTOWN':
541
+ $replace = $this->location_town;
542
+ break;
543
+ case '#_LOCATIONSTATE':
544
+ $replace = $this->location_state;
545
+ break;
546
+ case '#_LOCATIONPOSTCODE':
547
+ $replace = $this->location_postcode;
548
+ break;
549
+ case '#_LOCATIONREGION':
550
+ $replace = $this->location_region;
551
+ break;
552
+ case '#_LOCATIONCOUNTRY':
553
+ $replace = $this->get_country();
554
+ break;
555
+ case '#_LOCATIONFULLLINE':
556
+ $replace = $this->location_address.', ';
557
+ $replace = empty($this->location_town) ? '':', '.$this->location_town;
558
+ $replace = empty($this->location_state) ? '':', '.$this->location_state;
559
+ $replace = empty($this->location_postcode) ? '':', '.$this->location_postcode;
560
+ $replace = empty($this->location_region) ? '':', '.$this->location_region;
561
+ break;
562
+ case '#_LOCATIONFULLBR':
563
+ $replace = $this->location_address.'<br /> ';
564
+ $replace = empty($this->location_town) ? '':'<br /> '.$this->location_town;
565
+ $replace = empty($this->location_state) ? '':'<br /> '.$this->location_state;
566
+ $replace = empty($this->location_postcode) ? '':'<br /> '.$this->location_postcode;
567
+ $replace = empty($this->location_region) ? '':'<br /> '.$this->location_region;
568
+ break;
569
+ case '#_MAP': //Depreciated
570
+ case '#_LOCATIONMAP':
571
+ ob_start();
572
+ $template = em_locate_template('placeholders/locationmap.php', true, array('EM_Location'=>$this));
573
+ $replace = ob_get_clean();
574
+ break;
575
+ case '#_DESCRIPTION': //Depreciated
576
+ case '#_EXCERPT': //Depreciated
577
+ case '#_LOCATIONNOTES':
578
+ case '#_LOCATIONEXCERPT':
579
+ $replace = $this->post_content;
580
+ if($result == "#_EXCERPT" || $result == "#_LOCATIONEXCERPT"){
581
+ if( !empty($this->post_excerpt) ){
582
+ $replace = $this->post_excerpt;
583
+ }else{
584
+ $matches = explode('<!--more', $this->post_content);
585
+ $replace = $matches[0];
586
+ }
587
+ }
588
+ break;
589
+ case '#_LOCATIONIMAGEURL':
590
+ case '#_LOCATIONIMAGE':
591
+ if($this->get_image_url() != ''){
592
+ $image_url = esc_url($this->get_image_url());
593
+ if($result == '#_LOCATIONIMAGEURL'){
594
+ $replace = $this->get_image_url();
595
+ }else{
596
+ if( empty($placeholders[3][$key]) ){
597
+ $replace = "<img src='".$image_url."' alt='".esc_attr($this->location_name)."'/>";
598
+ }else{
599
+ $image_size = explode(',', $placeholders[3][$key]);
600
+ if( $this->array_is_numeric($image_size) && count($image_size) > 1 ){
601
+ $replace = "<img src='".esc_url(em_get_thumbnail_url($image_url, $image_size[0], $image_size[1]))."' alt='".esc_attr($this->location_name)."'/>";
602
+ }else{
603
+ $replace = "<img src='".$image_url."' alt='".esc_attr($this->location_name)."'/>";
604
+ }
605
+ }
606
+ }
607
+ }
608
+ break;
609
+ case '#_LOCATIONURL':
610
+ case '#_LOCATIONLINK':
611
+ case '#_LOCATIONPAGEURL': //Depreciated
612
+ $link = esc_url($this->get_permalink());
613
+ $replace = ($result == '#_LOCATIONURL' || $result == '#_LOCATIONPAGEURL') ? $link : '<a href="'.$link.'" title="'.esc_attr($this->location_name).'">'.esc_html($this->location_name).'</a>';
614
+ break;
615
+ case '#_LOCATIONEDITURL':
616
+ case '#_LOCATIONEDITLINK':
617
+ $link = esc_url($this->get_edit_url());
618
+ $replace = ($result == '#_LOCATIONEDITURL') ? $link : '<a href="'.$link.'" title="'.esc_attr($this->location_name).'">'.esc_html(sprintf(__('Edit %s','dbem'),__('Location','dbem'))).'</a>';
619
+ break;
620
+ case '#_PASTEVENTS': //Depreciated
621
+ case '#_LOCATIONPASTEVENTS':
622
+ case '#_NEXTEVENTS': //Depreciated
623
+ case '#_LOCATIONNEXTEVENTS':
624
+ case '#_ALLEVENTS': //Depreciated
625
+ case '#_LOCATIONALLEVENTS':
626
+ //convert depreciated placeholders for compatability
627
+ $result = ($result == '#_PASTEVENTS') ? '#_LOCATIONPASTEVENTS':$result;
628
+ $result = ($result == '#_NEXTEVENTS') ? '#_LOCATIONNEXTEVENTS':$result;
629
+ $result = ($result == '#_ALLEVENTS') ? '#_LOCATIONALLEVENTS':$result;
630
+ //forget it ever happened? :/
631
+ if ( $result == '#_LOCATIONPASTEVENTS'){ $scope = 'past'; }
632
+ elseif ( $result == '#_LOCATIONNEXTEVENTS' ){ $scope = 'future'; }
633
+ else{ $scope = 'all'; }
634
+ $events = EM_Events::get( array('location'=>$this->location_id, 'scope'=>$scope) );
635
+ if ( count($events) > 0 ){
636
+ foreach($events as $event){
637
+ $replace .= $event->output(get_option('dbem_location_event_list_item_format'));
638
+ }
639
+ } else {
640
+ $replace = get_option('dbem_location_no_events_message');
641
+ }
642
+ break;
643
+ default:
644
+ $replace = $full_result;
645
+ break;
646
+ }
647
+ $replace = apply_filters('em_location_output_placeholder', $replace, $this, $full_result, $target);
648
+ $location_string = str_replace($full_result, $replace , $location_string );
649
+ }
650
+ $name_filter = ($target == "html") ? 'dbem_general':'dbem_general_rss'; //TODO remove dbem_ filters
651
+ $location_string = str_replace('#_LOCATION', apply_filters($name_filter, $this->location_name) , $location_string ); //Depreciated
652
+ return apply_filters('em_location_output', $location_string, $this, $format, $target);
653
+ }
654
+
655
+ function get_country(){
656
+ $countries = em_get_countries();
657
+ if( !empty($countries[$this->location_country]) ){
658
+ return apply_filters('em_location_get_country', $countries[$this->location_country], $this);
659
+ }
660
+ return apply_filters('em_location_get_country', false, $this);
661
+
662
+ }
663
  }
classes/em-locations.php CHANGED
@@ -1,213 +1,306 @@
1
- <?php
2
- /**
3
- * Static class which will help bulk add/edit/retrieve/manipulate arrays of EM_Location objects.
4
- * Optimized for specifically retreiving locations (whether eventful or not). If you want event data AND location information for each event, use EM_Events
5
- *
6
- */
7
- class EM_Locations extends EM_Object {
8
- /**
9
- * Returns an array of EM_Location objects
10
- * @param boolean $eventful
11
- * @param boolean $return_objects
12
- * @return array
13
- */
14
- function get( $args = array() ){
15
- global $wpdb;
16
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
17
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
18
-
19
- //Quick version, we can accept an array of IDs, which is easy to retrieve
20
- if( self::array_is_numeric($args) && count() ){ //Array of numbers, assume they are event IDs to retreive
21
- //We can just get all the events here and return them
22
- $sql = "SELECT * FROM $locations_table WHERE location_id=".implode(" OR location_id=", $args);
23
- $results = $wpdb->get_results($sql,ARRAY_A);
24
- $events = array();
25
- foreach($results as $result){
26
- $locations[$result['location_id']] = new EM_Location($result);
27
- }
28
- return $locations; //We return all the events matched as an EM_Event array.
29
- }
30
-
31
-
32
- //We assume it's either an empty array or array of search arguments to merge with defaults
33
- $args = self::get_default_search($args);
34
- $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
35
- $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
36
-
37
- //Get the default conditions
38
- $conditions = self::build_sql_conditions($args);
39
-
40
- //Put it all together
41
- $EM_Location = new EM_Location(0); //Empty class for strict message avoidance
42
- $fields = $locations_table .".". implode(", {$locations_table}.", array_keys($EM_Location->fields));
43
- $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
44
-
45
- //Get ordering instructions
46
- $EM_Event = new EM_Event(); //blank event for below
47
- $accepted_fields = $EM_Location->get_fields(true);
48
- $accepted_fields = array_merge($EM_Event->get_fields(true),$accepted_fields);
49
- $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order'));
50
- //Now, build orderby sql
51
- $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
52
-
53
-
54
- //Create the SQL statement and execute
55
- $sql = "
56
- SELECT $fields FROM $locations_table
57
- LEFT JOIN $events_table ON {$locations_table}.location_id={$events_table}.location_id
58
- $where
59
- GROUP BY location_id
60
- $orderby_sql
61
- $limit $offset
62
- ";
63
-
64
- $results = $wpdb->get_results($sql, ARRAY_A);
65
-
66
- //If we want results directly in an array, why not have a shortcut here?
67
- if( $args['array'] == true ){
68
- return $results;
69
- }
70
-
71
- $locations = array();
72
- foreach ($results as $location){
73
- $locations[] = new EM_Location($location);
74
- }
75
- return apply_filters('em_locations_get', $locations, $args);
76
- }
77
-
78
- /**
79
- * Output a set of matched of events
80
- * @param array $args
81
- * @return string
82
- */
83
- function output( $args ){
84
- global $EM_Location;
85
- $EM_Location_old = $EM_Location; //When looping, we can replace EM_Location global with the current event in the loop
86
- //Can be either an array for the get search or an array of EM_Location objects
87
- if( is_object(current($args)) && get_class((current($args))) == 'EM_Location' ){
88
- $func_args = func_get_args();
89
- $locations = $func_args[0];
90
- $args = apply_filters('em_locations_output_args', self::get_default_search($func_args[1]), $locations);
91
- $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
92
- $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
93
- $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
94
- }else{
95
- $args = apply_filters('em_locations_output_args', self::get_default_search($args) );
96
- $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
97
- $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
98
- $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
99
- $args['limit'] = false;
100
- $args['offset'] = false;
101
- $args['page'] = false;
102
- $locations = self::get( $args );
103
- }
104
- //What format shall we output this to, or use default
105
- $format = ( $args['format'] == '' ) ? get_option( 'dbem_location_list_item_format' ) : $args['format'] ;
106
-
107
- $output = "";
108
- $locations_count = count($locations);
109
- $locations = apply_filters('em_locations_output_locations', $locations);
110
-
111
- if ( count($locations) > 0 ) {
112
- $location_count = 0;
113
- $locations_shown = 0;
114
- foreach ( $locations as $EM_Location ) {
115
- if( ($locations_shown < $limit || empty($limit)) && ($location_count >= $offset || $offset === 0) ){
116
- $output .= $EM_Location->output($format);
117
- $locations_shown++;
118
- }
119
- $location_count++;
120
- }
121
- //Add headers and footers to output
122
- if( $format == get_option ( 'dbem_location_list_item_format' ) ){
123
- $single_event_format_header = get_option ( 'dbem_location_list_item_format_header' );
124
- $single_event_format_header = ( $single_event_format_header != '' ) ? $single_event_format_header : "<ul class='dbem_events_list'>";
125
- $single_event_format_footer = get_option ( 'dbem_location_list_item_format_footer' );
126
- $single_event_format_footer = ( $single_event_format_footer != '' ) ? $single_event_format_footer : "</ul>";
127
- $output = $single_event_format_header . $output . $single_event_format_footer;
128
- }
129
- //Pagination (if needed/requested)
130
- if( !empty($args['pagination']) && !empty($limit) && $locations_count >= $limit ){
131
- //Show the pagination links (unless there's less than 10 events
132
- $page_link_template = preg_replace('/(&|\?)page=\d+/i','',$_SERVER['REQUEST_URI']);
133
- $page_link_template = em_add_get_params($page_link_template, array('page'=>'%PAGE%'));
134
- $output .= apply_filters('em_events_output_pagination', em_paginate( $page_link_template, $locations_count, $limit, $page), $page_link_template, $locations_count, $limit, $page);
135
- }
136
- } else {
137
- $output = get_option ( 'dbem_no_events_message' );
138
- }
139
- //FIXME check if reference is ok when restoring object, due to changes in php5 v 4
140
- $EM_Location_old= $EM_Location;
141
- return apply_filters('em_locations_output', $output, $locations, $args);
142
- }
143
-
144
- /**
145
- * Builds an array of SQL query conditions based on regularly used arguments
146
- * @param array $args
147
- * @return array
148
- */
149
- function build_sql_conditions( $args = array() ){
150
- global $wpdb;
151
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
152
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
153
-
154
- $conditions = parent::build_sql_conditions($args);
155
- //eventful locations
156
- if( true == $args['eventful'] ){
157
- $conditions['eventful'] = "{$events_table}.event_id IS NOT NULL";
158
- }elseif( true == $args['eventless'] ){
159
- $conditions['eventless'] = "{$events_table}.event_id IS NULL";
160
- }
161
- //owner lookup
162
- if( !empty($args['owner']) ){
163
- if ( get_option('dbem_permissions_locations') < 1 && !em_verify_admin() ){
164
- $conditions['owner'] = "location_owner=".get_current_user_id();
165
- }
166
- }
167
- return apply_filters('em_locations_build_sql_conditions', $conditions, $args);
168
- }
169
-
170
- /* Overrides EM_Object method to apply a filter to result
171
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
172
- */
173
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
174
- return apply_filters( 'em_locations_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
175
- }
176
-
177
- /*
178
- * Generate a search arguments array from defalut and user-defined.
179
- * @see wp-content/plugins/events-manager/classes/EM_Object::get_default_search()
180
- */
181
- function get_default_search($array = array()){
182
- $defaults = array(
183
- 'eventful' => false, //Locations that have an event (scope will also play a part here
184
- 'eventless' => false, //Locations WITHOUT events, eventful takes precedence
185
- 'orderby' => 'name',
186
- 'scope' => 'all' //we probably want to search all locations by default, not like events
187
- );
188
- $array['eventful'] = ( !empty($array['eventful']) && $array['eventful'] == true );
189
- $array['eventless'] = ( !empty($array['eventless']) && $array['eventless'] == true );
190
- if( is_admin() ){
191
- //by default, we only get categories the owner can manage
192
- switch( get_option('dbem_permissions_locations') ){
193
- case 0:
194
- $defaults['owner'] = get_current_user_id();
195
- break;
196
- case 1:
197
- $wp_user_search = new WP_User_Search(null, null, 'administrator');
198
- $users = $wp_user_search->get_results();
199
- $users[] = get_current_user_id();
200
- $users[] = 0;
201
- $defaults['owner'] = implode(',', $users);
202
- break;
203
- case 2:
204
- $defaults['owner'] = false;
205
- break;
206
- }
207
- $defaults['owner'] = ( em_verify_admin() ) ? false:$defaults['owner'];
208
- }
209
- return apply_filters('em_locations_get_default_search', parent::get_default_search($defaults, $array), $array, $defaults);
210
- }
211
- //TODO for all the static plural classes like this one, we might benefit from bulk actions like delete/add/save etc.... just a random thought.
212
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
213
  ?>
1
+ <?php
2
+ /**
3
+ * Static class which will help bulk add/edit/retrieve/manipulate arrays of EM_Location objects.
4
+ * Optimized for specifically retreiving locations (whether eventful or not). If you want event data AND location information for each event, use EM_Events
5
+ *
6
+ */
7
+ class EM_Locations extends EM_Object implements Iterator {
8
+ /**
9
+ * Array of EM_Location objects
10
+ * @var array EM_Location
11
+ */
12
+ var $locations = array();
13
+
14
+ /**
15
+ * Returns an array of EM_Location objects
16
+ * @param boolean $eventful
17
+ * @param boolean $return_objects
18
+ * @return array
19
+ */
20
+ function get( $args = array(), $count=false ){
21
+ global $wpdb;
22
+ $events_table = EM_EVENTS_TABLE;
23
+ $locations_table = EM_LOCATIONS_TABLE;
24
+ $locations = array();
25
+
26
+ //Quick version, we can accept an array of IDs, which is easy to retrieve
27
+ if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
28
+ //We can just get all the events here and return them
29
+ $sql = "SELECT * FROM $locations_table WHERE location_id=".implode(" OR location_id=", $args);
30
+ $results = $wpdb->get_results($sql,ARRAY_A);
31
+ $events = array();
32
+ foreach($results as $result){
33
+ $locations[$result['location_id']] = new EM_Location($result);
34
+ }
35
+ return apply_filters('em_locations_get', $locations, $args); //We return all the events matched as an EM_Event array.
36
+ }elseif( is_numeric($args) ){
37
+ //return an event in the usual array format
38
+ return apply_filters('em_locations_get', array(new EM_Location($args)), $args);
39
+ }elseif( is_array($args) && is_object(current($args)) && get_class((current($args))) == 'EM_Location' ){
40
+ return apply_filters('em_locations_get', $args, $args);
41
+ }
42
+
43
+ //We assume it's either an empty array or array of search arguments to merge with defaults
44
+ $args = self::get_default_search($args);
45
+ $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
46
+ $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
47
+
48
+ //Get the default conditions
49
+ $conditions = self::build_sql_conditions($args);
50
+
51
+ //Put it all together
52
+ $EM_Location = new EM_Location(0); //Empty class for strict message avoidance
53
+ $fields = $locations_table .".". implode(", {$locations_table}.", array_keys($EM_Location->fields));
54
+ $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
55
+
56
+ //Get ordering instructions
57
+ $EM_Event = new EM_Event(); //blank event for below
58
+ $accepted_fields = $EM_Location->get_fields(true);
59
+ $accepted_fields = array_merge($EM_Event->get_fields(true),$accepted_fields);
60
+ $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order'));
61
+ //Now, build orderby sql
62
+ $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
63
+
64
+ $fields = ( $count ) ? $locations_table.'.location_id':$locations_table.'.post_id';
65
+ if( EM_MS_GLOBAL ){
66
+ $selectors = ( $count ) ? 'COUNT('.$locations_table.'.location_id)':$locations_table.'.post_id, '.$locations_table.'.blog_id';
67
+ }else{
68
+ $selectors = ( $count ) ? 'COUNT('.$locations_table.'.location_id)':$locations_table.'.post_id';
69
+ }
70
+ //Create the SQL statement and execute
71
+ $sql = "
72
+ SELECT $selectors FROM $locations_table
73
+ LEFT JOIN $events_table ON {$locations_table}.location_id={$events_table}.location_id
74
+ $where
75
+ GROUP BY {$locations_table}.location_id
76
+ $orderby_sql
77
+ $limit $offset
78
+ ";
79
+
80
+ //If we're only counting results, return the number of results
81
+ if( $count ){
82
+ return apply_filters('em_locations_get_array', count($wpdb->get_col($sql)), $args);
83
+ }
84
+ $results = $wpdb->get_results($sql, ARRAY_A);
85
+
86
+ //If we want results directly in an array, why not have a shortcut here?
87
+ if( $args['array'] == true ){
88
+ return apply_filters('em_locations_get_array', $results, $args);
89
+ }
90
+
91
+ if( EM_MS_GLOBAL ){
92
+ foreach ( $results as $location ){
93
+ $locations[] = em_get_location($location['post_id'], $location['blog_id']);
94
+ }
95
+ }else{
96
+ foreach ( $results as $location ){
97
+ $locations[] = em_get_location($location['post_id'], 'post_id');
98
+ }
99
+ }
100
+ return apply_filters('em_locations_get', $locations, $args);
101
+ }
102
+
103
+ function count( $args = array() ){
104
+ return apply_filters('em_locations_count', self::get($args, true), $args);
105
+ }
106
+
107
+ /**
108
+ * Output a set of matched of events
109
+ * @param array $args
110
+ * @return string
111
+ */
112
+ function output( $args ){
113
+ global $EM_Location;
114
+ $EM_Location_old = $EM_Location; //When looping, we can replace EM_Location global with the current event in the loop
115
+ //Can be either an array for the get search or an array of EM_Location objects
116
+ if( is_object(current($args)) && get_class((current($args))) == 'EM_Location' ){
117
+ $func_args = func_get_args();
118
+ $locations = $func_args[0];
119
+ $args = (!empty($func_args[1])) ? $func_args[1] : array();
120
+ $args = apply_filters('em_locations_output_args', self::get_default_search($args), $locations);
121
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
122
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
123
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
124
+ $locations_count = count($locations);
125
+ }else{
126
+ $args = apply_filters('em_locations_output_args', self::get_default_search($args) );
127
+ $limit = ( !empty($args['limit']) && is_numeric($args['limit']) ) ? $args['limit']:false;
128
+ $offset = ( !empty($args['offset']) && is_numeric($args['offset']) ) ? $args['offset']:0;
129
+ $page = ( !empty($args['page']) && is_numeric($args['page']) ) ? $args['page']:1;
130
+ $args_count = $args;
131
+ $args_count['limit'] = 0;
132
+ $args_count['offset'] = 0;
133
+ $args_count['page'] = 1;
134
+ $locations_count = self::count($args_count);
135
+ $locations = self::get( $args );
136
+ }
137
+ //What format shall we output this to, or use default
138
+ $format = ( $args['format'] == '' ) ? get_option( 'dbem_location_list_item_format' ) : $args['format'] ;
139
+
140
+ $output = "";
141
+ $locations = apply_filters('em_locations_output_locations', $locations);
142
+ if ( count($locations) > 0 ) {
143
+ foreach ( $locations as $EM_Location ) {
144
+ $output .= $EM_Location->output($format);
145
+ }
146
+ //Add headers and footers to output
147
+ if( $format == get_option ( 'dbem_location_list_item_format' ) ){
148
+ $single_event_format_header = get_option ( 'dbem_location_list_item_format_header' );
149
+ $single_event_format_header = ( $single_event_format_header != '' ) ? $single_event_format_header : "<ul class='em-locations-list'>";
150
+ $single_event_format_footer = get_option ( 'dbem_location_list_item_format_footer' );
151
+ $single_event_format_footer = ( $single_event_format_footer != '' ) ? $single_event_format_footer : "</ul>";
152
+ $output = $single_event_format_header . $output . $single_event_format_footer;
153
+ }
154
+ //Pagination (if needed/requested)
155
+ if( !empty($args['pagination']) && !empty($limit) && $locations_count >= $limit ){
156
+ //Show the pagination links (unless there's less than 10 events
157
+ $page_link_template = preg_replace('/(&|\?)page=\d+/i','',$_SERVER['REQUEST_URI']);
158
+ $page_link_template = em_add_get_params($page_link_template, array('page'=>'%PAGE%'), false); //don't html encode, so em_paginate does its thing
159
+ $output .= apply_filters('em_events_output_pagination', em_paginate( $page_link_template, $locations_count, $limit, $page), $page_link_template, $locations_count, $limit, $page);
160
+ }
161
+ } else {
162
+ $output = get_option ( 'dbem_no_locations_message' );
163
+ }
164
+ //FIXME check if reference is ok when restoring object, due to changes in php5 v 4
165
+ $EM_Location_old= $EM_Location;
166
+ return apply_filters('em_locations_output', $output, $locations, $args);
167
+ }
168
+
169
+ function delete( $args = array() ){
170
+ if( !is_object(current($args)) && get_class((current($args))) != 'EM_Location' ){
171
+ $locations = self::get($args);
172
+ }else{
173
+ $locations = $args;
174
+ }
175
+ $results = array();
176
+ foreach ( $locations as $EM_Location ){
177
+ $results[] = $EM_Location->delete();
178
+ }
179
+ return apply_filters('em_locations_delete', in_array(false, $results), $locations);
180
+ }
181
+
182
+ /**
183
+ * Builds an array of SQL query conditions based on regularly used arguments
184
+ * @param array $args
185
+ * @return array
186
+ */
187
+ function build_sql_conditions( $args = array(), $count=false ){
188
+ global $wpdb;
189
+ $events_table = EM_EVENTS_TABLE;
190
+ $locations_table = EM_LOCATIONS_TABLE;
191
+
192
+ $conditions = parent::build_sql_conditions($args);
193
+ //eventful locations
194
+ if( true == $args['eventful'] ){
195
+ $conditions['eventful'] = "{$events_table}.event_id IS NOT NULL";
196
+ }elseif( true == $args['eventless'] ){
197
+ $conditions['eventless'] = "{$events_table}.event_id IS NULL";
198
+ }
199
+ //owner lookup
200
+ if( !empty($args['owner']) && is_numeric($args['owner'])){
201
+ $conditions['owner'] = "location_owner=".$args['owner'];
202
+ }elseif( !empty($args['owner']) && $args['owner'] == 'me' && is_user_logged_in() ){
203
+ $conditions['owner'] = 'location_owner='.get_current_user_id();
204
+ }
205
+ //blog id in events table
206
+ if( is_multisite() && array_key_exists('blog',$args) && is_numeric($args['blog']) ){
207
+ if( is_main_site($args['blog']) ){
208
+ $conditions['blog'] = "($locations_table.blog_id={$args['blog']} OR $locations_table.blog_id IS NULL)";
209
+ }else{
210
+ $conditions['blog'] = "($locations_table.blog_id={$args['blog']})";
211
+ }
212
+ }
213
+ //status
214
+ if( array_key_exists('status',$args) && is_numeric($args['status']) ){
215
+ $null = ($args['status'] == 0) ? ' OR `location_status` = 0':'';
216
+ $conditions['status'] = "(`location_status`={$args['status']}{$null} )";
217
+ }else{
218
+ $conditions['status'] = "(`location_status` IS NOT NULL)";
219
+ }
220
+ //private locations
221
+ if( !empty($args['private']) ){
222
+ $conditions['private'] = "(`location_private`=0)";
223
+ }elseif( !empty($args['private_only']) ){
224
+ $conditions['private_only'] = "(`location_private`=1)";
225
+ }
226
+ //post search
227
+ if( !empty($args['post_id'])){
228
+ if( is_array($args['post_id']) ){
229
+ $conditions['post_id'] = "($locations_table.post_id IN (".implode(',',$args['post_id'])."))";
230
+ }else{
231
+ $conditions['post_id'] = "($locations_table.post_id={$args['post_id']})";
232
+ }
233
+ }
234
+ return apply_filters('em_locations_build_sql_conditions', $conditions, $args);
235
+ }
236
+
237
+ /* Overrides EM_Object method to apply a filter to result
238
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
239
+ */
240
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
241
+ return apply_filters( 'em_locations_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
242
+ }
243
+
244
+ /*
245
+ * Generate a search arguments array from defalut and user-defined.
246
+ * @see wp-content/plugins/events-manager/classes/EM_Object::get_default_search()
247
+ */
248
+ function get_default_search($array = array()){
249
+ $defaults = array(
250
+ 'eventful' => false, //Locations that have an event (scope will also play a part here
251
+ 'eventless' => false, //Locations WITHOUT events, eventful takes precedence
252
+ 'orderby' => 'location_name',
253
+ 'town' => false,
254
+ 'state' => false,
255
+ 'country' => false,
256
+ 'region' => false,
257
+ 'status' => 1, //approved locations only
258
+ 'scope' => 'all', //we probably want to search all locations by default, not like events
259
+ 'blog' => get_current_blog_id(),
260
+ 'private' => !current_user_can('read_private_locations'),
261
+ 'private_only' => false,
262
+ 'post_id' => false
263
+ );
264
+ if(is_multisite()){
265
+ global $bp;
266
+ if( !is_main_site() && !array_key_exists('blog', $array) ){
267
+ $array['blog'] = get_current_blog_id();
268
+ }elseif( empty($array['blog']) && get_site_option('dbem_ms_global_locations') ) {
269
+ $array['blog'] = false;
270
+ }
271
+ }
272
+ $array['eventful'] = ( !empty($array['eventful']) && $array['eventful'] == true );
273
+ $array['eventless'] = ( !empty($array['eventless']) && $array['eventless'] == true );
274
+ if( is_admin() ){
275
+ $defaults['owner'] = !current_user_can('read_others_locations') ? get_current_user_id():false;
276
+ }
277
+ return apply_filters('em_locations_get_default_search', parent::get_default_search($defaults, $array), $array, $defaults);
278
+ }
279
+
280
+ //Iteratior methods
281
+ public function rewind(){
282
+ reset($this->locations);
283
+ }
284
+
285
+ public function current(){
286
+ $var = current($this->locations);
287
+ return $var;
288
+ }
289
+
290
+ public function key(){
291
+ $var = key($this->locations);
292
+ return $var;
293
+ }
294
+
295
+ public function next(){
296
+ $var = next($this->locations);
297
+ return $var;
298
+ }
299
+
300
+ public function valid(){
301
+ $key = key($this->locations);
302
+ $var = ($key !== NULL && $key !== FALSE);
303
+ return $var;
304
+ }
305
+ }
306
  ?>
classes/em-mailer.php CHANGED
@@ -1,80 +1,97 @@
1
- <?php
2
- /**
3
- * phpmailer support
4
- *
5
- */
6
- class EM_Mailer {
7
-
8
- /**
9
- * if any errors crop up, here they are
10
- * @var array
11
- */
12
- var $errors = array();
13
-
14
- /**
15
- * @param $subject
16
- * @param $body
17
- * @param $receiver
18
- */
19
- function send($subject="no title",$body="No message specified", $receiver='') {
20
- //TODO add an EM_Error global object, for this sort of error reporting. (@marcus like StatusNotice)
21
- global $smtpsettings, $phpmailer, $cformsSettings;
22
-
23
- if( preg_match('/^[_.0-9a-z-]+@([0-9a-z][0-9a-z-]+.)+[a-z]{2,3}$/i', $receiver) ){
24
- $this->load_phpmailer();
25
- $mail = new EM_PHPMailer();
26
- //$mail->SMTPDebug = true;
27
- $mail->ClearAllRecipients();
28
- $mail->ClearAddresses();
29
- $mail->ClearAttachments();
30
- $mail->CharSet = 'utf-8';
31
- $mail->SetLanguage('en', dirname(__FILE__).'/');
32
- $mail->PluginDir = dirname(__FILE__).'/phpmailer/';
33
- $mail->Host = get_option('dbem_smtp_host');
34
- $mail->port = get_option('dbem_rsvp_mail_port');
35
- $mail->Username = get_option('dbem_smtp_username');
36
- $mail->Password = get_option('dbem_smtp_password');
37
- $mail->From = get_option('dbem_mail_sender_address');
38
- $mail->FromName = get_option('dbem_mail_sender_name'); // This is the from name in the email, you can put anything you like here
39
- $mail->Body = $body;
40
- $mail->Subject = $subject;
41
- $mail->AddAddress($receiver);
42
-
43
- //Protocols
44
- if ( get_option('dbem_rsvp_mail_send_method') == 'wp_mail' ){
45
- $mail->Mailer = 'wp_mail';
46
- $send = wp_mail($receiver, $subject, $body);
47
- if(!$send){
48
- global $phpmailer;
49
- $this->errors[] = $phpmailer->ErrorInfo;
50
- }
51
- }else{
52
- if( get_option('dbem_rsvp_mail_send_method') == 'qmail' ){
53
- $mail->IsQmail();
54
- }else {
55
- $mail->Mailer = get_option('dbem_rsvp_mail_send_method');
56
- }
57
- if(get_option('dbem_rsvp_mail_SMTPAuth') == '1'){
58
- $mail->SMTPAuth = TRUE;
59
- }
60
- $send = $mail->Send();
61
- if(!$send){
62
- $this->errors[] = $mail->ErrorInfo;
63
- }
64
- }
65
- return $send;
66
- }else{
67
- $this->errors = __('Please supply a valid email format.', 'dbem');
68
- return false;
69
- }
70
- }
71
-
72
- /**
73
- * load phpmailer classes
74
- */
75
- function load_phpmailer(){
76
- require_once(dirname(__FILE__) . '/phpmailer/class.phpmailer.php');
77
- require_once(dirname(__FILE__) . '/phpmailer/class.smtp.php');
78
- }
79
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
80
  ?>
1
+ <?php
2
+ /**
3
+ * phpmailer support
4
+ *
5
+ */
6
+ class EM_Mailer {
7
+
8
+ /**
9
+ * if any errors crop up, here they are
10
+ * @var array
11
+ */
12
+ var $errors = array();
13
+
14
+ /**
15
+ * @param $subject
16
+ * @param $body
17
+ * @param $receiver
18
+ */
19
+ function send($subject="no title",$body="No message specified", $receiver='') {
20
+ //TODO add an EM_Error global object, for this sort of error reporting. (@marcus like StatusNotice)
21
+ global $smtpsettings, $phpmailer, $cformsSettings;
22
+ if( is_array($receiver) ){
23
+ $receiver_emails = array();
24
+ foreach($receiver as $receiver_email){
25
+ $receiver_emails[] = is_email($receiver_email);
26
+ }
27
+ $emails_ok = !in_array(false, $receiver_emails);
28
+ }else{
29
+ $emails_ok = is_email($receiver);
30
+ }
31
+ if( $emails_ok ){
32
+ $this->load_phpmailer();
33
+ $mail = new EM_PHPMailer();
34
+ //$mail->SMTPDebug = true;
35
+ if( get_option('dbem_smtp_html') ){
36
+ $mail->isHTML();
37
+ }
38
+ $mail->ClearAllRecipients();
39
+ $mail->ClearAddresses();
40
+ $mail->ClearAttachments();
41
+ $mail->CharSet = 'utf-8';
42
+ $mail->SetLanguage('en', dirname(__FILE__).'/');
43
+ $mail->PluginDir = dirname(__FILE__).'/phpmailer/';
44
+ $mail->Host = get_option('dbem_smtp_host');
45
+ $mail->port = get_option('dbem_rsvp_mail_port');
46
+ $mail->Username = get_option('dbem_smtp_username');
47
+ $mail->Password = get_option('dbem_smtp_password');
48
+ $mail->From = get_option('dbem_mail_sender_address');
49
+ $mail->FromName = get_option('dbem_mail_sender_name'); // This is the from name in the email, you can put anything you like here
50
+ $mail->Body = $body;
51
+ $mail->Subject = $subject;
52
+ if(is_array($receiver)){
53
+ foreach($receiver as $receiver_email){
54
+ $mail->AddAddress($receiver_email);
55
+ }
56
+ }else{
57
+ $mail->AddAddress($receiver);
58
+ }
59
+
60
+ //Protocols
61
+ if ( get_option('dbem_rsvp_mail_send_method') == 'wp_mail' ){
62
+ $mail->Mailer = 'wp_mail';
63
+ $send = wp_mail($receiver, $subject, $body);
64
+ if(!$send){
65
+ global $phpmailer;
66
+ $this->errors[] = $phpmailer->ErrorInfo;
67
+ }
68
+ }else{
69
+ if( get_option('dbem_rsvp_mail_send_method') == 'qmail' ){
70
+ $mail->IsQmail();
71
+ }else {
72
+ $mail->Mailer = get_option('dbem_rsvp_mail_send_method');
73
+ }
74
+ if(get_option('dbem_rsvp_mail_SMTPAuth') == '1'){
75
+ $mail->SMTPAuth = TRUE;
76
+ }
77
+ $send = $mail->Send();
78
+ if(!$send){
79
+ $this->errors[] = $mail->ErrorInfo;
80
+ }
81
+ }
82
+ return $send;
83
+ }else{
84
+ $this->errors = __('Please supply a valid email format.', 'dbem');
85
+ return false;
86
+ }
87
+ }
88
+
89
+ /**
90
+ * load phpmailer classes
91
+ */
92
+ function load_phpmailer(){
93
+ require_once(dirname(__FILE__) . '/phpmailer/class.phpmailer.php');
94
+ require_once(dirname(__FILE__) . '/phpmailer/class.smtp.php');
95
+ }
96
+ }
97
  ?>
classes/em-map.php DELETED
@@ -1,60 +0,0 @@
1
- <?php
2
- /**
3
- * Obtains the html required to display a google map for given location(s)
4
- *
5
- */
6
- class EM_Map extends EM_Object {
7
- /**
8
- * Shortcode for producing a google map with all the locations. Unfinished and undocumented.
9
- * @param array $atts
10
- * @return string
11
- */
12
- function get_global($atts) {
13
- //TODO Finish and document this feature, need to add balloons here
14
- if (get_option('dbem_gmap_is_active') == '1') {
15
- ob_start();
16
- $atts['em_ajax'] = true;
17
- $atts['query'] = 'GlobalMapData';
18
- $rand = substr(md5(rand().rand()),0,5);
19
- //build js array of arguments to send to event query
20
- ?>
21
- <div class='em-locations-map' id='em-locations-map-<?php echo $rand; ?>' style='width:<?php echo $atts['width']; ?>px; height:<?php echo $atts['height']; ?>px'><em><?php _e('Loading Map....', 'dbem'); ?></em></div>
22
- <div class='em-locations-map-coords' id='em-locations-map-coords-<?php echo $rand; ?>' style="display:none; visibility:hidden;"><?php echo EM_Object::json_encode($atts); ?></div>
23
- <?php
24
- return apply_filters('em_map_get_global', ob_get_clean());
25
- }else{
26
- return '';
27
- }
28
- }
29
-
30
-
31
- /**
32
- * Returns th HTML and JS required to produce a google map in for this location.
33
- * @param EM_Location $location
34
- * @return string
35
- */
36
- function get_single($args) {
37
- //TODO do some validation here of defaults
38
- //FIXME change baloon to balloon for consistent spelling
39
- $location = $args['location'];
40
- if ( get_option('dbem_gmap_is_active') && ( is_object($location) && $location->latitude != 0 && $location->longitude != 0 ) ) {
41
- $width = (isset($args['width'])) ? $args['width']:'400';
42
- $height = (isset($args['height'])) ? $args['height']:'300';
43
- ob_start();
44
- $rand = substr(md5(rand().rand()),0,5);
45
- ?>
46
- <div class='em-location-map' id='em-location-map-<?php echo $rand ?>' style='background: #CDCDCD; width: <?php echo $width ?>px; height: <?php echo $height ?>px'><?php _e('Loading Map....', 'dbem'); ?></div>
47
- <div class='em-location-map-info' id='em-location-map-info-<?php echo $rand ?>' style="display:none; visibility:hidden;"><div class="em-map-balloon" style="font-size:12px;"><div class="em-map-balloon-content" ><?php echo $location->output(get_option('dbem_location_baloon_format')); ?></div></div></div>
48
- <div class='em-location-map-coords' id='em-location-map-coords-<?php echo $rand ?>' style="display:none; visibility:hidden;">
49
- <span class="lat"><?php echo $location->latitude; ?></span>
50
- <span class="lng"><?php echo $location->longitude; ?></span>
51
- </div>
52
- <?php
53
- return ob_get_clean();
54
- }elseif( is_object($location) && $location->latitude == 0 && $location->longitude == 0 ){
55
- return '<i>'. __('Map Unavailable', 'dbem') .'</i>';
56
- }else{
57
- return '';
58
- }
59
- }
60
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
classes/em-notices.php ADDED
@@ -0,0 +1,196 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ *
4
+ * @author marcus
5
+ *
6
+ */
7
+ class EM_Notices implements Iterator {
8
+ var $notices = array('errors'=>array(), 'infos'=>array(), 'alerts'=>array(), 'confirms'=>array());
9
+
10
+ function __construct(){
11
+ session_start();
12
+ //Grab from session
13
+ if( !empty($_SESSION['events-manager']['notices']) && is_serialized($_SESSION['events-manager']['notices']) ){
14
+ $this->notices = unserialize($_SESSION['events-manager']['notices']);
15
+ }
16
+ //Flush notices that weren't made to stay cross-requests, we can do this if initialized immediately.
17
+ foreach($this->notices as $notice_type => $notices){
18
+ foreach ($notices as $key => $notice){
19
+ if( empty($notice['static']) ){
20
+ unset($this->notices[$notice_type][$key]);
21
+ }else{
22
+ unset($this->notices[$notice_type][$key]['static']); //so it gets removed next request
23
+ }
24
+ }
25
+ }
26
+ add_action('shutdown', array(&$this,'destruct'));
27
+ add_filter('wp_redirect', array(&$this,'destruct'), 1,1);
28
+ }
29
+
30
+ function destruct($redirect = false){
31
+ //die('destructing');
32
+ $_SESSION['events-manager']['notices'] = serialize($this->notices);
33
+ return $redirect;
34
+ }
35
+
36
+ function __toString(){
37
+ $string = false;
38
+ if(count($this->notices['errors']) > 0){
39
+ $string .= "<div class='em-warning em-warning-errors error'>{$this->get_errors()}</div>";
40
+ }
41
+ if(count($this->notices['alerts']) > 0){
42
+ $string .= "<div class='em-warning em-warning-alerts updated'>{$this->get_alerts()}</div>";
43
+ }
44
+ if(count($this->notices['infos']) > 0){
45
+ $string .= "<div class='em-warning em-warning-infos updated'>{$this->get_infos()}</div>";
46
+ }
47
+ if(count($this->notices['confirms']) > 0){
48
+ $string .= "<div class='em-warning em-warning-confirms updated'>{$this->get_confirms()}</div>";
49
+ }
50
+ return ($string !== false) ? "<div class='statusnotice'>".$string."</div>" : '';
51
+ }
52
+
53
+ /* General */
54
+ function add($string, $type, $static = false){
55
+ if( is_array($string) ){
56
+ $result = true;
57
+ foreach($string as $string_item){
58
+ if( $this->add($string_item, $type, $static) === false ){ $result = false; }
59
+ }
60
+ return $result;
61
+ }
62
+ if($string != ''){
63
+ if( isset($this->notices[$type]) ){
64
+ $notice_key = 0;
65
+ foreach( $this->notices[$type] as $notice_key => $notice ){
66
+ if($string == $notice['string']){
67
+ return $notice_key;
68
+ }
69
+ }
70
+ $i = $notice_key+1;
71
+ $this->notices[$type][$i]['string'] = $string;
72
+ if( $static ){
73
+ $this->notices[$type][$i]['static'] = true;
74
+ }
75
+ return $i;
76
+ }else{
77
+ return false;
78
+ }
79
+ }else{
80
+ return false;
81
+ }
82
+ }
83
+ function remove($key, $type){
84
+ if( isset($this->notices[$type]) ){
85
+ unset($this->notices[$type][$key]);
86
+ return true;
87
+ }else{
88
+ return false;
89
+ }
90
+ }
91
+ function remove_all(){
92
+ $this->notices = array('errors'=>array(), 'infos'=>array(), 'alerts'=>array(), 'confirms'=>array());
93
+ }
94
+ function get($type){
95
+ if( isset($this->notices[$type]) ){
96
+ $string = '';
97
+ foreach ($this->notices[$type] as $key => $error){
98
+ $class = substr($type, 0, (strlen($type)-1));
99
+ $string .= "<p>{$error['string']}</p>";
100
+ if( empty($error['static']) || $error['static'] !== true){
101
+ $this->remove($key, $type);
102
+ }
103
+ }
104
+ return $string;
105
+ }
106
+ return false;
107
+ }
108
+ function count($type){
109
+ if( isset($this->notices[$type]) ){
110
+ return count($this->notices[$type]);
111
+ }
112
+ return 0;
113
+ }
114
+
115
+ /* Errors */
116
+ function add_error($string, $static=false){
117
+ return $this->add($string, 'errors', $static);
118
+ }
119
+ function remove_error($key){
120
+ return $this->remove($key, 'errors');
121
+ }
122
+ function get_errors(){
123
+ return $this->get('errors');
124
+ }
125
+ function count_errors(){
126
+ return $this->count('errors');
127
+ }
128
+
129
+ /* Alerts */
130
+ function add_alert($string, $static=false){
131
+ return $this->add($string, 'alerts', $static);
132
+ }
133
+ function remove_alert($key){
134
+ return $this->remove($key, 'alerts');
135
+ }
136
+ function get_alerts(){
137
+ return $this->get('alerts');
138
+ }
139
+ function count_alerts(){
140
+ return $this->count('alerts');
141
+ }
142
+
143
+ /* Info */
144
+ function add_info($string, $static=false){
145
+ return $this->add($string, 'infos', $static);
146
+ }
147
+ function remove_info($key){
148
+ return $this->remove($key, 'infos');
149
+ }
150
+ function get_infos(){
151
+ return $this->get('infos');
152
+ }
153
+ function count_infos(){
154
+ return $this->count('infos');
155
+ }
156
+
157
+ /* Confirms */
158
+ function add_confirm($string, $static=false){
159
+ return $this->add($string, 'confirms', $static);
160
+ }
161
+ function remove_confirm($key){
162
+ return $this->remove($key, 'confirms');
163
+ }
164
+ function get_confirms(){
165
+ return $this->get('confirms');
166
+ }
167
+ function count_confirms(){
168
+ return $this->count('confirms');
169
+ }
170
+
171
+ //Iterator Implementation
172
+ function rewind(){
173
+ reset($this->bookings);
174
+ }
175
+ function current(){
176
+ $var = current($this->bookings);
177
+ return $var;
178
+ }
179
+ function key(){
180
+ $var = key($this->bookings);
181
+ return $var;
182
+ }
183
+ function next(){
184
+ $var = next($this->bookings);
185
+ return $var;
186
+ }
187
+ function valid(){
188
+ $key = key($this->bookings);
189
+ $var = ($key !== NULL && $key !== FALSE);
190
+ return $var;
191
+ }
192
+
193
+ }
194
+ global $EM_Notices;
195
+ $EM_Notices = new EM_Notices();
196
+ ?>
classes/em-object.php CHANGED
@@ -1,486 +1,890 @@
1
- <?php
2
- /**
3
- * Base class which others extend on. Contains functions shared across all EM objects.
4
- *
5
- */
6
- class EM_Object {
7
-
8
- var $fields = array();
9
-
10
- /**
11
- * Takes the array and provides a clean array of search parameters, along with details
12
- * @param array $defaults
13
- * @param array $array
14
- * @return array
15
- */
16
- function get_default_search($defaults=array(), $array = array()){
17
- //TODO accept all objects as search options as well as ids (e.g. location vs. location_id, person vs. person_id)
18
- //Create minimal defaults array, merge it with supplied defaults array
19
- $super_defaults = array(
20
- 'limit' => false,
21
- 'scope' => 'future',
22
- 'order' => 'ASC', //hard-coded at end of this function
23
- 'orderby' => false,
24
- 'format' => '',
25
- 'category' => 0,
26
- 'location' => 0,
27
- 'event' => 0,
28
- 'offset'=>0,
29
- 'page'=>1,//basically, if greater than 0, calculates offset at end
30
- 'recurrence'=>0,
31
- 'recurring'=>false,
32
- 'month'=>'',
33
- 'year'=>'',
34
- 'pagination'=>false,
35
- 'array'=>false,
36
- 'owner'=>false,
37
- 'rsvp'=>false
38
- );
39
- //Return default if nothing passed
40
- if( empty($defaults) && empty($array) ){
41
- return $super_defaults;
42
- }
43
- //TODO decide on search defaults shared across all objects and then validate here
44
- $defaults = array_merge($super_defaults, $defaults);
45
-
46
- //We are still dealing with recurrence_id, location_id, category_id in some place, so we do a quick replace here just in case
47
- if( array_key_exists('recurrence_id', $array) && !array_key_exists('recurrence', $array) ) { $array['recurrence'] = $array['recurrence_id']; }
48
- if( array_key_exists('location_id', $array) && !array_key_exists('location', $array) ) { $array['location'] = $array['location_id']; }
49
- if( array_key_exists('category_id', $array) && !array_key_exists('category', $array) ) { $array['category'] = $array['category_id']; }
50
-
51
- if(is_array($array)){
52
- //Clean all id lists
53
- $array = self::clean_id_atts($array, array('location', 'event', 'category'));
54
-
55
- //OrderBy - can be a comma-seperated array of field names to order by (field names of object, not db)
56
- if( array_key_exists('orderby', $array)){
57
- if( !is_array($array['orderby']) && preg_match('/,/', $array['orderby']) ) {
58
- $array['orderby'] = explode(',', $array['orderby']);
59
- }
60
- }
61
- //TODO validate search query array
62
- //Clean the supplied array, so we only have allowed keys
63
- foreach( array_keys($array) as $key){
64
- if( !array_key_exists($key, $defaults) ) unset($array[$key]);
65
- }
66
- //return clean array
67
- $defaults = array_merge ( $defaults, $array ); //No point using WP's cleaning function, we're doing it already.
68
- }
69
- //Do some spring cleaning for known values
70
- //Month & Year - may be array or single number
71
- $month_regex = '/^[0-9]{1,2}$/';
72
- $year_regex = '/^[0-9]{4}$/';
73
- if( is_array($defaults['month']) ){
74
- $defaults['month'] = ( preg_match($month_regex, $defaults['month'][0]) && preg_match($month_regex, $defaults['month'][1]) ) ? $defaults['month']:'';
75
- }else{
76
- $defaults['month'] = preg_match($month_regex, $defaults['month']) ? $defaults['month']:'';
77
- }
78
- if( is_array($defaults['year']) ){
79
- $defaults['year'] = ( preg_match($year_regex, $defaults['year'][0]) && preg_match($year_regex, $defaults['year'][1]) ) ? $defaults['year']:'';
80
- }else{
81
- $defaults['year'] = preg_match($year_regex, $defaults['year']) ? $defaults['year']:'';
82
- }
83
- //Order - it's either ASC or DESC, so let's just validate
84
- if( !is_array($defaults['order']) && preg_match('/,/', $defaults['order']) ) {
85
- $defaults['order'] = explode(',', $defaults['order']);
86
- }elseif( !in_array($defaults['order'], array('ASC','DESC')) ){
87
- $defaults['order'] = $super_defaults['order'];
88
- }
89
- //ORDER BY, split if an array
90
- if( !is_array($defaults['orderby']) && preg_match('/,/', $defaults['orderby']) ) {
91
- $defaults['orderby'] = explode(',', $defaults['orderby']);
92
- }
93
- //TODO should we clean format of malicious code over here and run everything thorugh this?
94
- $defaults['array'] = ($defaults['array'] == true);
95
- $defaults['pagination'] = ($defaults['pagination'] == true);
96
- $defaults['limit'] = (is_numeric($defaults['limit'])) ? $defaults['limit']:$super_defaults['limit'];
97
- $defaults['offset'] = (is_numeric($defaults['offset'])) ? $defaults['offset']:$super_defaults['offset'];
98
- $defaults['recurring'] = ($defaults['recurring'] == true);
99
- $defaults['owner'] = (is_numeric($defaults['owner'])) ? $defaults['owner']:$super_defaults['owner'];
100
- //Calculate offset in event page is set
101
- if($defaults['page'] > 1){
102
- $defaults['offset'] = $defaults['limit'] * ($defaults['page']-1);
103
- }else{
104
- $defaults['page'] = ($defaults['limit'] > 0 ) ? floor($defaults['offset']/$defaults['limit']) + 1 : 1;
105
- }
106
- return apply_filters('em_object_get_default_search', $defaults, $array, $super_defaults);
107
- }
108
-
109
- /**
110
- * Builds an array of SQL query conditions based on regularly used arguments
111
- * @param array $args
112
- * @return array
113
- */
114
- function build_sql_conditions( $args = array() ){
115
- global $wpdb;
116
- $events_table = $wpdb->prefix . EM_EVENTS_TABLE;
117
- $locations_table = $wpdb->prefix . EM_LOCATIONS_TABLE;
118
-
119
- $args = apply_filters('em_object_build_sql_conditions_args',$args);
120
-
121
- //Format the arguments passed on
122
- $scope = $args['scope'];//undefined variable warnings in ZDE, could just delete this (but dont pls!)
123
- $recurring = $args['recurring'];
124
- $recurrence = $args['recurrence'];
125
- $category = $args['category'];
126
- $location = $args['location'];
127
- $rsvp = $args['rsvp'];
128
- $owner = $args['owner'];
129
- $event = $args['event'];
130
- $month = $args['month'];
131
- $year = $args['year'];
132
- $today = date('Y-m-d', current_time('timestamp'));
133
- //Create the WHERE statement
134
-
135
- //Recurrences
136
- $conditions = array();
137
- if( $recurring ){
138
- $conditions['recurring'] = "`recurrence`=1";
139
- }elseif( $recurrence > 0 ){
140
- $conditions['recurrence'] = "`recurrence_id`=$recurrence";
141
- }else{
142
- $conditions['recurring'] = "(`recurrence`!=1 OR `recurrence` IS NULL)";
143
- }
144
- //Dates - first check 'month', and 'year', and adjust scope if needed
145
- if( !($month=='' && $year=='') ){
146
- //Sort out month range, if supplied an array of array(month,month), it'll check between these two months
147
- if( self::array_is_numeric($month) ){
148
- $date_month_start = $month[0];
149
- $date_month_end = $month[1];
150
- }else{
151
- $date_month_start = $date_month_end = $month;
152
- }
153
- //Sort out year range, if supplied an array of array(year,year), it'll check between these two years
154
- if( self::array_is_numeric($year) ){
155
- $date_year_start = $year[0];
156
- $date_year_end = $year[1];
157
- }else{
158
- $date_year_start = $date_year_end = $year;
159
- }
160
- $date_start = $date_year_start."-".$date_month_start."-01";
161
- $date_end = date('Y-m-t', mktime(0,0,0,$date_month_end,1,$date_year_end));
162
- $scope = $date_start.",".$date_end; //just modify the scope here
163
- }
164
- //No date requested, so let's look at scope
165
- if ( preg_match ( "/^[0-9]{4}-[0-9]{2}-[0-9]{2},[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $scope ) ) {
166
- //This is an array, let's split it up
167
- $dates = explode(',', $scope);
168
- $date_start = $dates[0];
169
- $date_end = $dates[1];
170
- $conditions['scope'] = " ( ( event_start_date <= CAST('$date_end' AS DATE) AND event_end_date >= CAST('$date_start' AS DATE) ) OR (event_start_date BETWEEN CAST('$date_start' AS DATE) AND CAST('$date_end' AS DATE)) OR (event_end_date BETWEEN CAST('$date_start' AS DATE) AND CAST('$date_end' AS DATE)) )";
171
- } elseif ( preg_match ( "/^[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $scope ) ) {
172
- //Scope can also be a specific date. However, if 'day', 'month', or 'year' are set, that will take precedence
173
- $conditions['scope'] = " ( event_start_date = CAST('$scope' AS DATE) OR ( event_start_date <= CAST('$scope' AS DATE) AND event_end_date >= CAST('$scope' AS DATE) ) )";
174
- } else {
175
- if ($scope == "past"){
176
- $conditions['scope'] = " event_start_date < '$today'";
177
- }elseif ($scope == "today"){
178
- $conditions['scope'] = " ( (event_start_date = CAST('$today' AS DATE)) OR (event_start_date <= CAST('$today' AS DATE) AND event_end_date >= CAST('$today' AS DATE)) )";
179
- }elseif ($scope == "future"){
180
- $conditions['scope'] = " (event_start_date >= CAST('$today' AS DATE) OR (event_end_date >= CAST('$today' AS DATE) AND event_end_date != '0000-00-00' AND event_end_date IS NOT NULL))";
181
- }
182
- }
183
-
184
- //Filter by Location - can be object, array, or id
185
- if ( is_numeric($location) && $location > 0 ) { //Location ID takes precedence
186
- $conditions['location'] = " {$locations_table}.location_id = $location";
187
- }elseif ( self::array_is_numeric($location) ){
188
- $conditions['location'] = "( {$locations_table}.location_id = " . implode(" OR {$locations_table}.location_id = ", $location) .' )';
189
- }elseif ( is_object($location) && get_class($location)=='EM_Location' ){ //Now we deal with objects
190
- $conditions['location'] = " {$locations_table}.location_id = $location->id";
191
- }elseif ( is_array($location) && @get_class(current($location)=='EM_Location') ){ //we can accept array of ids or EM_Location objects
192
- foreach($location as $EM_Location){
193
- $location_ids[] = $EM_Location->id;
194
- }
195
- $conditions['location'] = "( {$locations_table}.location_id=". implode(" {$locations_table}.location_id=", $location_ids) ." )";
196
- }
197
-
198
- //Filter by Event - can be object, array, or id
199
- if ( is_numeric($event) && $event > 0 ) { //event ID takes precedence
200
- $conditions['event'] = " {$events_table}.event_id = $event";
201
- }elseif ( self::array_is_numeric($event) ){ //array of ids
202
- $conditions['event'] = "( {$events_table}.event_id = " . implode(" OR {$events_table}.event_id = ", $event) .' )';
203
- }elseif ( is_object($event) && get_class($event)=='EM_Event' ){ //Now we deal with objects
204
- $conditions['event'] = " {$events_table}.event_id = $event->id";
205
- }elseif ( is_array($event) && @get_class(current($event)=='EM_Event') ){ //we can accept array of ids or EM_event objects
206
- foreach($event as $EM_Event){
207
- $event_ids[] = $EM_Event->id;
208
- }
209
- $conditions['event'] = "( {$events_table}.event_id=". implode(" {$events_table}.event_id=", $event_ids) ." )";
210
- }
211
-
212
- //Add conditions for category selection
213
- //Filter by category, can be id or comma seperated ids
214
- //TODO create an exclude category option
215
- if ( is_numeric($category) && $category > 0 ){
216
- $conditions['category'] = " event_category_id = $category";
217
- }elseif( self::array_is_numeric($category) ){
218
- $conditions['category'] = "( event_category_id = ". implode(' OR event_category_id = ', $category).")";
219
- }
220
-
221
- //If we want rsvped items, we usually check the event
222
- if( $rsvp == 1 ){
223
- $conditions['rsvp'] = 'event_rsvp=1';
224
- }
225
- //Default ownership belongs to an event, child objects can just overwrite this if needed.
226
- if( is_numeric($owner) ){
227
- $conditions['owner'] = 'event_author='.$owner;
228
- }
229
- return apply_filters('em_object_build_sql_conditions', $conditions);
230
- }
231
-
232
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
233
- //First, ORDER BY
234
- $args = apply_filters('em_object_build_sql_orderby_args', $args);
235
- $orderby = array();
236
- if(is_array($args['orderby'])){
237
- //Clean orderby array so we only have accepted values
238
- foreach( $args['orderby'] as $key => $field ){
239
- if( array_key_exists($field, $accepted_fields) ){
240
- $orderby[] = $accepted_fields[$field];
241
- }elseif( in_array($field,$accepted_fields) ){
242
- $orderby[] = $field;
243
- }else{
244
- unset($args['orderby'][$key]);
245
- }
246
- }
247
- }elseif( $args['orderby'] != '' && array_key_exists($args['orderby'], $accepted_fields) ){
248
- $orderby[] = $accepted_fields[$args['orderby']];
249
- }elseif( $args['orderby'] != '' && in_array($args['orderby'], $accepted_fields) ){
250
- $orderby[] = $args['orderby'];
251
- }
252
- //ORDER
253
- //If order is an array, we'll go through the orderby array and match the order values (in order of array) with orderby values
254
- //If orders don't match up, or it's not ASC/DESC, the default events search in EM settings/options page will be used.
255
- foreach($orderby as $i => $field){
256
- $orderby[$i] .= ' ';
257
- if(is_array($args['order'])){
258
- if( in_array($args['order'][$i], array('ASC','DESC')) ){
259
- $orderby[$i] .= $args['order'][$i];
260
- }else{
261
- $orderby[$i] .= $default_order;
262
- }
263
- }else{
264
- $orderby[$i] .= ( in_array($args['order'], array('ASC','DESC')) ) ? $args['order'] : $default_order;
265
- }
266
- }
267
- return apply_filters('em_object_build_sql_orderby', $orderby);
268
- }
269
-
270
- /**
271
- * Save an array into this class.
272
- * If you provide a record from the database table corresponding to this class type it will add the data to this object.
273
- * @param array $array
274
- * @return null
275
- */
276
- function to_object( $array = array(), $addslashes = false ){
277
- //Save core data
278
- if( is_array($array) ){
279
- $array = apply_filters('em_to_object', $array);
280
- foreach ( $this->fields as $key => $val ) {
281
- if(array_key_exists($key, $array)){
282
- if( !is_object($array[$key]) && !is_array($array[$key]) ){
283
- $array[$key] = ($addslashes) ? stripslashes($array[$key]):$array[$key];
284
- }
285
- $this->$val['name'] = $array[$key];
286
- }
287
- }
288
- }
289
- }
290
-
291
- /**
292
- * Returns this object in the form of an array, useful for saving directly into a database table.
293
- * @return array
294
- */
295
- function to_array(){
296
- $array = array();
297
- foreach ( $this->fields as $key => $val ) {
298
- $array[$key] = $this->$val['name'];
299
- }
300
- return apply_filters('em_to_array', $array);
301
- }
302
-
303
-
304
- /**
305
- * Function to retreive wpdb types for all fields, or if you supply an assoc array with field names as keys it'll return an equivalent array of wpdb types
306
- * @param array $array
307
- * @return array:
308
- */
309
- function get_types($array = array()){
310
- $types = array();
311
- if( count($array)>0 ){
312
- //So we look at assoc array and find equivalents
313
- foreach ($array as $key => $val){
314
- $types[] = $this->fields[$key]['type'];
315
- }
316
- }else{
317
- //Blank array, let's assume we're getting a standard list of types
318
- foreach ($this->fields as $field){
319
- $types[] = $field['type'];
320
- }
321
- }
322
- return apply_filters('em_object_get_types', $types, $this, $array);
323
- }
324
-
325
- function get_fields( $inverted_array=false ){
326
- if( is_array($this->fields) ){
327
- $return = array();
328
- foreach($this->fields as $fieldName => $fieldArray){
329
- if($inverted_array){
330
- $return[$fieldArray['name']] = $fieldName;
331
- }else{
332
- $return[$fieldName] = $fieldArray['name'];
333
- }
334
- }
335
- return apply_filters('em_object_get_fields', $return, $this, $inverted_array);
336
- }
337
- return apply_filters('em_object_get_fields', array(), $this, $inverted_array);
338
- }
339
-
340
- /**
341
- * Sanitize text before inserting into database
342
- * @param string $value
343
- * @return string
344
- */
345
- function sanitize( $value ) {
346
- if( get_magic_quotes_gpc() )
347
- $value = stripslashes( $value );
348
-
349
- //check if this function exists
350
- if( function_exists( "mysql_real_escape_string" ) ) {
351
- $value = mysql_real_escape_string( $value );
352
- //for PHP version < 4.3.0 use addslashes
353
- } else {
354
- $value = addslashes( $value );
355
- }
356
- return apply_filters('em_object_sanitize', $value);
357
- }
358
-
359
- /**
360
- * Cleans arrays that contain id lists. Takes an array of items and will clean the keys passed in second argument so that if they keep numbers, explode comma-seperated numbers, and unsets the key if there's any other value
361
- * @param array $array
362
- * @param array $id_atts
363
- */
364
- function clean_id_atts( $array = array(), $id_atts = array() ){
365
- if( is_array($array) && is_array($id_atts) ){
366
- foreach( $array as $key => $string ){
367
- if( in_array($key, $id_atts) ){
368
- //This is in the list of atts we want cleaned
369
- if( is_numeric($string) ){
370
- $array[$key] = (int) $string;
371
- }elseif( self::array_is_numeric($string) ){
372
- $array[$key] = $string;
373
- }elseif( preg_match('/^([0-9],?)+$/', $string) ){
374
- $array[$key] = explode(',', $string);
375
- }else{
376
- //No format we accept
377
- unset($array[$key]);
378
- }
379
- }
380
- }
381
- }
382
- return $array;
383
- }
384
-
385
- /**
386
- * Send an email and log errors in this object
387
- * @param string $subject
388
- * @param string $body
389
- * @param string $email
390
- * @return string
391
- */
392
- function email_send($subject, $body, $email){
393
- global $EM_Mailer;
394
- if( !$EM_Mailer->send($subject,$body,$email) ){
395
- foreach($EM_Mailer->errors as $error){
396
- $this->errors[] = $error;
397
- }
398
- return false;
399
- }
400
- return true;
401
- }
402
-
403
- /**
404
- * Will return true if this is a simple (non-assoc) numeric array, meaning it has at one or more numeric entries and nothing else
405
- * @param mixed $array
406
- * @return boolean
407
- */
408
- function array_is_numeric($array){
409
- $results = array();
410
- if(is_array($array)){
411
- foreach($array as $key => $item){
412
- $results[] = (is_numeric($item)&&is_numeric($key));
413
- }
414
- }
415
- return (!in_array(false, $results) && count($results) > 0);
416
- }
417
-
418
- /**
419
- * Converts an array to JSON format, useful for outputting data for AJAX calls. Uses a PHP4 fallback function, given it doesn't support json_encode().
420
- * @param array $array
421
- * @return string
422
- */
423
- function json_encode($array){
424
- if( function_exists("json_encode") ){
425
- $return = json_encode($array);
426
- }else{
427
- $return = self::array_to_json($array);
428
- }
429
- if( isset($_GET['callback']) ){
430
- $return = $_GET['callback']."($return)";
431
- }
432
- return apply_filters('em_object_json_encode', $return, $array);
433
- }
434
-
435
- /**
436
- * Compatible json encoder function for PHP4
437
- * @param array $array
438
- * @return string
439
- */
440
- function array_to_json($array){
441
- //PHP4 Comapatability - This encodes the array into JSON. Thanks go to Andy - http://www.php.net/manual/en/function.json-encode.php#89908
442
- if( !is_array( $array ) ){
443
- $array = array();
444
- }
445
- $associative = count( array_diff( array_keys($array), array_keys( array_keys( $array )) ));
446
- if( $associative ){
447
- $construct = array();
448
- foreach( $array as $key => $value ){
449
- // We first copy each key/value pair into a staging array,
450
- // formatting each key and value properly as we go.
451
- // Format the key:
452
- if( is_numeric($key) ){
453
- $key = "key_$key";
454
- }
455
- $key = "'".addslashes($key)."'";
456
- // Format the value:
457
- if( is_array( $value )){
458
- $value = $this->array_to_json( $value );
459
- }else if( is_bool($value) ) {
460
- $value = ($value) ? "true" : "false";
461
- }else if( !is_numeric( $value ) || is_string( $value ) ){
462
- $value = "'".addslashes($value)."'";
463
- }
464
- // Add to staging array:
465
- $construct[] = "$key: $value";
466
- }
467
- // Then we collapse the staging array into the JSON form:
468
- $result = "{ " . implode( ", ", $construct ) . " }";
469
- } else { // If the array is a vector (not associative):
470
- $construct = array();
471
- foreach( $array as $value ){
472
- // Format the value:
473
- if( is_array( $value )){
474
- $value = $this->array_to_json( $value );
475
- } else if( !is_numeric( $value ) || is_string( $value ) ){
476
- $value = "'".addslashes($value)."'";
477
- }
478
- // Add to staging array:
479
- $construct[] = $value;
480
- }
481
- // Then we collapse the staging array into the JSON form:
482
- $result = "[ " . implode( ", ", $construct ) . " ]";
483
- }
484
- return $result;
485
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
486
  }
1
+ <?php
2
+ /**
3
+ * Base class which others extend on. Contains functions shared across all EM objects.
4
+ *
5
+ */
6
+ class EM_Object {
7
+ var $fields = array();
8
+ var $required_fields = array();
9
+ var $feedback_message = "";
10
+ var $errors = array();
11
+ var $mime_types = array(1 => 'gif', 2 => 'jpg', 3 => 'png');
12
+
13
+ /**
14
+ * Takes the array and provides a clean array of search parameters, along with details
15
+ * @param array $defaults
16
+ * @param array $array
17
+ * @return array
18
+ */
19
+ function get_default_search($defaults=array(), $array = array()){
20
+ global $wpdb;
21
+ //TODO accept all objects as search options as well as ids (e.g. location vs. location_id, person vs. person_id)
22
+ //Create minimal defaults array, merge it with supplied defaults array
23
+ $super_defaults = array(
24
+ 'limit' => false,
25
+ 'scope' => 'future',
26
+ 'order' => 'ASC', //hard-coded at end of this function
27
+ 'orderby' => false,
28
+ 'format' => '',
29
+ 'category' => 0,
30
+ 'tag' => 0,
31
+ 'location' => 0,
32
+ 'event' => 0,
33
+ 'offset'=>0,
34
+ 'page'=>1,//basically, if greater than 0, calculates offset at end
35
+ 'recurrence'=>0,
36
+ 'recurring'=>false,
37
+ 'month'=>'',
38
+ 'year'=>'',
39
+ 'pagination'=>false,
40
+ 'array'=>false,
41
+ 'owner'=>false,
42
+ 'rsvp'=>false, //depreciated for bookings
43
+ 'bookings'=>false,
44
+ 'search'=>false
45
+ );
46
+ //Return default if nothing passed
47
+ if( empty($defaults) && empty($array) ){
48
+ return $super_defaults;
49
+ }
50
+ //TODO decide on search defaults shared across all objects and then validate here
51
+ $defaults = array_merge($super_defaults, $defaults);
52
+
53
+ if(is_array($array)){
54
+ //We are still dealing with recurrence_id, location_id, category_id in some place, so we do a quick replace here just in case
55
+ if( array_key_exists('recurrence_id', $array) && !array_key_exists('recurrence', $array) ) { $array['recurrence'] = $array['recurrence_id']; }
56
+ if( array_key_exists('location_id', $array) && !array_key_exists('location', $array) ) { $array['location'] = $array['location_id']; }
57
+ if( array_key_exists('category_id', $array) && !array_key_exists('category', $array) ) { $array['category'] = $array['category_id']; }
58
+
59
+ //Clean all id lists
60
+ $array = self::clean_id_atts($array, array('location', 'event', 'category', 'tag', 'post_id'));
61
+
62
+ //OrderBy - can be a comma-seperated array of field names to order by (field names of object, not db)
63
+ if( array_key_exists('orderby', $array)){
64
+ if( !is_array($array['orderby']) && preg_match('/,/', $array['orderby']) ) {
65
+ $array['orderby'] = explode(',', $array['orderby']);
66
+ }
67
+ }
68
+ //TODO validate search query array
69
+ //Clean the supplied array, so we only have allowed keys
70
+ foreach( array_keys($array) as $key){
71
+ if( !array_key_exists($key, $defaults) ) unset($array[$key]);
72
+ }
73
+ //return clean array
74
+ $defaults = array_merge ( $defaults, $array ); //No point using WP's cleaning function, we're doing it already.
75
+
76
+ }
77
+
78
+ //Do some spring cleaning for known values
79
+ //Month & Year - may be array or single number
80
+ $month_regex = '/^[0-9]{1,2}$/';
81
+ $year_regex = '/^[0-9]{4}$/';
82
+ if( is_array($defaults['month']) ){
83
+ $defaults['month'] = ( preg_match($month_regex, $defaults['month'][0]) && preg_match($month_regex, $defaults['month'][1]) ) ? $defaults['month']:'';
84
+ }else{
85
+ $defaults['month'] = preg_match($month_regex, $defaults['month']) ? $defaults['month']:'';
86
+ }
87
+ if( is_array($defaults['year']) ){
88
+ $defaults['year'] = ( preg_match($year_regex, $defaults['year'][0]) && preg_match($year_regex, $defaults['year'][1]) ) ? $defaults['year']:'';
89
+ }else{
90
+ $defaults['year'] = preg_match($year_regex, $defaults['year']) ? $defaults['year']:'';
91
+ }
92
+ //Deal with scope and date searches
93
+ if ( !is_array($defaults['scope']) && preg_match ( "/^([0-9]{4}-[0-9]{2}-[0-9]{2})?,([0-9]{4}-[0-9]{2}-[0-9]{2})?$/", $defaults['scope'] ) ) {
94
+ //This is to become an array, so let's split it up
95
+ $defaults['scope'] = explode(',', $defaults['scope']);
96
+ }
97
+ if( is_array($defaults['scope']) ){
98
+ //looking for a date range here, so we'll verify the dates validate, if not get the default.
99
+ if ( !preg_match("/^[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $defaults['scope'][0]) ){
100
+ $defaults['scope'][0] = '';
101
+ }
102
+ if( !preg_match("/^[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $defaults['scope'][1]) ) {
103
+ $defaults['scope'][1] = '';
104
+ }
105
+ if( empty($defaults['scope'][0]) && empty($defaults['scope'][1]) ){
106
+ $defaults['scope'] = $super_defaults['scope'];
107
+ }
108
+ }
109
+ //Order - it's either ASC or DESC, so let's just validate
110
+ if( !is_array($defaults['order']) && preg_match('/,/', $defaults['order']) ) {
111
+ $defaults['order'] = explode(',', $defaults['order']);
112
+ }elseif( !in_array($defaults['order'], array('ASC','DESC')) ){
113
+ $defaults['order'] = $super_defaults['order'];
114
+ }
115
+ //ORDER BY, split if an array
116
+ if( !is_array($defaults['orderby']) && preg_match('/,/', $defaults['orderby']) ) {
117
+ $defaults['orderby'] = explode(',', $defaults['orderby']);
118
+ }
119
+ //TODO should we clean format of malicious code over here and run everything thorugh this?
120
+ $defaults['array'] = ($defaults['array'] == true);
121
+ $defaults['pagination'] = ($defaults['pagination'] == true);
122
+ $defaults['limit'] = (is_numeric($defaults['limit'])) ? $defaults['limit']:$super_defaults['limit'];
123
+ $defaults['offset'] = (is_numeric($defaults['offset'])) ? $defaults['offset']:$super_defaults['offset'];
124
+ $defaults['recurring'] = ($defaults['recurring'] == true);
125
+ $defaults['owner'] = (is_numeric($defaults['owner']) || $defaults['owner'] == 'me') ? $defaults['owner']:$super_defaults['owner'];
126
+ $defaults['search'] = ($defaults['search']) ? trim($wpdb->escape(like_escape($defaults['search']))):false;
127
+ //Calculate offset if event page is set
128
+ if($defaults['page'] > 1){
129
+ $defaults['offset'] = $defaults['limit'] * ($defaults['page']-1);
130
+ }else{
131
+ $defaults['page'] = ($defaults['limit'] > 0 ) ? floor($defaults['offset']/$defaults['limit']) + 1 : 1;
132
+ }
133
+ return apply_filters('em_object_get_default_search', $defaults, $array, $super_defaults);
134
+ }
135
+
136
+ /**
137
+ * Builds an array of SQL query conditions based on regularly used arguments
138
+ * @param array $args
139
+ * @return array
140
+ */
141
+ function build_sql_conditions( $args = array() ){
142
+ global $wpdb;
143
+ $events_table = EM_EVENTS_TABLE;
144
+ $locations_table = EM_LOCATIONS_TABLE;
145
+
146
+ $args = apply_filters('em_object_build_sql_conditions_args',$args);
147
+
148
+ //Format the arguments passed on
149
+ $scope = $args['scope'];//undefined variable warnings in ZDE, could just delete this (but dont pls!)
150
+ $recurring = $args['recurring'];
151
+ $recurrence = $args['recurrence'];
152
+ $category = $args['category'];
153
+ $tag = $args['tag'];
154
+ $location = $args['location'];
155
+ $bookings = $args['rsvp'];
156
+ $bookings = !empty($args['bookings']) ? $args['bookings']:$bookings;
157
+ $owner = $args['owner'];
158
+ $event = $args['event'];
159
+ $month = $args['month'];
160
+ $year = $args['year'];
161
+ $today = date('Y-m-d', current_time('timestamp'));
162
+ //Create the WHERE statement
163
+
164
+ //Recurrences
165
+ $conditions = array();
166
+ if( $recurring ){
167
+ $conditions['recurring'] = "`recurrence`=1";
168
+ }elseif( $recurrence > 0 ){
169
+ $conditions['recurrence'] = "`recurrence_id`=$recurrence";
170
+ }else{
171
+ $conditions['recurring'] = "(`recurrence`!=1 OR `recurrence` IS NULL)";
172
+ }
173
+ //Dates - first check 'month', and 'year', and adjust scope if needed
174
+ if( !($month=='' && $year=='') ){
175
+ //Sort out month range, if supplied an array of array(month,month), it'll check between these two months
176
+ if( self::array_is_numeric($month) ){
177
+ $date_month_start = $month[0];
178
+ $date_month_end = $month[1];
179
+ }else{
180
+ if( !empty($month) ){
181
+ $date_month_start = $date_month_end = $month;
182
+ }else{
183
+ $date_month_start = 1;
184
+ $date_month_end = 12;
185
+ }
186
+ }
187
+ //Sort out year range, if supplied an array of array(year,year), it'll check between these two years
188
+ if( self::array_is_numeric($year) ){
189
+ $date_year_start = $year[0];
190
+ $date_year_end = $year[1];
191
+ }else{
192
+ $date_year_start = $date_year_end = $year;
193
+ }
194
+ $date_start = $date_year_start."-".$date_month_start."-01";
195
+ $date_end = date('Y-m-t', mktime(0,0,0,$date_month_end,1,$date_year_end));
196
+ $scope = array($date_start,$date_end); //just modify the scope here
197
+ }
198
+ //No date requested, so let's look at scope
199
+ if ( is_array($scope) ) {
200
+ //This is an array, let's split it up
201
+ $date_start = $scope[0];
202
+ $date_end = $scope[1];
203
+ if( !empty($date_start) && empty($date_end) ){
204
+ //do a from till infinity
205
+ $conditions['scope'] = " event_start_date >= CAST('$date_start' AS DATE)";
206
+ }elseif( empty($date_start) && !empty($date_end) ){
207
+ //do past till $date_end
208
+ if( get_option('dbem_events_current_are_past') ){
209
+ $conditions['scope'] = " event_start_date <= CAST('$date_end' AS DATE)";
210
+ }else{
211
+ $conditions['scope'] = " event_end_date <= CAST('$date_end' AS DATE)";
212
+ }
213
+ }else{
214
+ //date range
215
+ $conditions['scope'] = " ( ( event_start_date <= CAST('$date_end' AS DATE) AND event_end_date >= CAST('$date_start' AS DATE) ) OR (event_start_date BETWEEN CAST('$date_start' AS DATE) AND CAST('$date_end' AS DATE)) OR (event_end_date BETWEEN CAST('$date_start' AS DATE) AND CAST('$date_end' AS DATE)) )";
216
+ }
217
+ } elseif ( preg_match ( "/^[0-9]{4}-[0-9]{2}-[0-9]{2}$/", $scope ) ) {
218
+ //Scope can also be a specific date. However, if 'day', 'month', or 'year' are set, that will take precedence
219
+ $conditions['scope'] = " ( event_start_date = CAST('$scope' AS DATE) OR ( event_start_date <= CAST('$scope' AS DATE) AND event_end_date >= CAST('$scope' AS DATE) ) )";
220
+ } else {
221
+ if ($scope == "past"){
222
+ if( get_option('dbem_events_current_are_past') ){
223
+ $conditions['scope'] = " event_start_date < '$today'";
224
+ }else{
225
+ $conditions['scope'] = " event_end_date < '$today'";
226
+ }
227
+ }elseif ($scope == "today"){
228
+ $conditions['scope'] = " (event_start_date = CAST('$today' AS DATE))";
229
+ if( !get_option('dbem_events_current_are_past') ){
230
+ $conditions['scope'] .= " OR (event_start_date <= CAST('$today' AS DATE) AND event_end_date >= CAST('$today' AS DATE))";
231
+ }
232
+ }elseif ($scope == "tomorrow"){
233
+ $tomorrow = date('Y-m-d',current_time('timestamp')+60*60*24);
234
+ $conditions['scope'] = "(event_start_date = CAST('$tomorrow' AS DATE))";
235
+ if( !get_option('dbem_events_current_are_past') ){
236
+ $conditions['scope'] .= " OR (event_start_date <= CAST('$tomorrow' AS DATE) AND event_end_date >= CAST('$tomorrow' AS DATE))";
237
+ }
238
+ }elseif ($scope == "month"){
239
+ $start_month = date('Y-m-d',current_time('timestamp'));
240
+ $end_month = date('Y-m-t',current_time('timestamp'));
241
+ $conditions['scope'] = " (event_start_date BETWEEN CAST('$start_month' AS DATE) AND CAST('$end_month' AS DATE))";
242
+ if( !get_option('dbem_events_current_are_past') ){
243
+ $conditions['scope'] .= " OR (event_start_date < CAST('$start_month' AS DATE) AND event_end_date >= CAST('$start_month' AS DATE))";
244
+ }
245
+ }elseif ($scope == "next-month"){
246
+ $start_month_timestamp = strtotime('+1 month', current_time('timestamp')); //get the end of this month + 1 day
247
+ $start_month = date('Y-m-1',$start_month_timestamp);
248
+ $end_month = date('Y-m-t',$start_month_timestamp);
249
+ $conditions['scope'] = " (event_start_date BETWEEN CAST('$start_month' AS DATE) AND CAST('$end_month' AS DATE))";
250
+ if( !get_option('dbem_events_current_are_past') ){
251
+ $conditions['scope'] .= " OR (event_start_date < CAST('$start_month' AS DATE) AND event_end_date >= CAST('$start_month' AS DATE))";
252
+ }
253
+ }elseif( preg_match('/(\d\d?)\-months/',$scope,$matches) ){ // next x months means this month (what's left of it), plus the following x months until the end of that month.
254
+ $months_to_add = $matches[1];
255
+ $start_month = date('Y-m-d',current_time('timestamp'));
256
+ $end_month = date('Y-m-t',strtotime("+$months_to_add month", current_time('timestamp')));
257
+ $conditions['scope'] = " (event_start_date BETWEEN CAST('$start_month' AS DATE) AND CAST('$end_month' AS DATE))";
258
+ if( !get_option('dbem_events_current_are_past') ){
259
+ $conditions['scope'] .= " OR (event_start_date < CAST('$start_month' AS DATE) AND event_end_date >= CAST('$start_month' AS DATE))";
260
+ }
261
+ }elseif ($scope == "future"){
262
+ $conditions['scope'] = " event_start_date >= CAST('$today' AS DATE)";
263
+ if( !get_option('dbem_events_current_are_past') ){
264
+ $conditions['scope'] .= " OR (event_end_date >= CAST('$today' AS DATE) AND event_end_date != '0000-00-00' AND event_end_date IS NOT NULL)";
265
+ }
266
+ }
267
+ if( !empty($conditions['scope']) ){
268
+ $conditions['scope'] = '('.$conditions['scope'].')';
269
+ }
270
+ }
271
+
272
+ //Filter by Location - can be object, array, or id
273
+ if ( is_numeric($location) && $location > 0 ) { //Location ID takes precedence
274
+ $conditions['location'] = " {$locations_table}.location_id = $location";
275
+ }elseif ( self::array_is_numeric($location) ){
276
+ $conditions['location'] = "( {$locations_table}.location_id = " . implode(" OR {$locations_table}.location_id = ", $location) .' )';
277
+ }elseif ( is_object($location) && get_class($location)=='EM_Location' ){ //Now we deal with objects
278
+ $conditions['location'] = " {$locations_table}.location_id = $location->location_id";
279
+ }elseif ( is_array($location) && @get_class(current($location)=='EM_Location') ){ //we can accept array of ids or EM_Location objects
280
+ foreach($location as $EM_Location){
281
+ $location_ids[] = $EM_Location->location_id;
282
+ }
283
+ $conditions['location'] = "( {$locations_table}.location_id=". implode(" {$locations_table}.location_id=", $location_ids) ." )";
284
+ }
285
+
286
+ //Filter by Event - can be object, array, or id
287
+ if ( is_numeric($event) && $event > 0 ) { //event ID takes precedence
288
+ $conditions['event'] = " {$events_table}.event_id = $event";
289
+ }elseif ( self::array_is_numeric($event) ){ //array of ids
290
+ $conditions['event'] = "( {$events_table}.event_id = " . implode(" OR {$events_table}.event_id = ", $event) .' )';
291
+ }elseif ( is_object($event) && get_class($event)=='EM_Event' ){ //Now we deal with objects
292
+ $conditions['event'] = " {$events_table}.event_id = $event->event_id";
293
+ }elseif ( is_array($event) && @get_class(current($event)=='EM_Event') ){ //we can accept array of ids or EM_event objects
294
+ foreach($event as $EM_Event){
295
+ $event_ids[] = $EM_Event->event_id;
296
+ }
297
+ $conditions['event'] = "( {$events_table}.event_id=". implode(" {$events_table}.event_id=", $event_ids) ." )";
298
+ }
299
+ //Location specific filters
300
+ //country lookup
301
+ if( !empty($args['country']) ){
302
+ $countries = em_get_countries();
303
+ //we can accept country codes or names
304
+ if( in_array($args['country'], $countries) ){
305
+ //we have a country name,
306
+ $conditions['country'] = "location_country='".array_search($args['country'])."'";
307
+ }elseif( array_key_exists($args['country'], $countries) ){
308
+ //we have a country code
309
+ $conditions['country'] = "location_country='".$args['country']."'";
310
+ }
311
+ }
312
+ //state lookup
313
+ if( !empty($args['state']) ){
314
+ $conditions['state'] = $wpdb->prepare('location_state=%s', $args['state']);
315
+ }
316
+ //state lookup
317
+ if( !empty($args['town']) ){
318
+ $conditions['town'] = $wpdb->prepare('location_town=%s', $args['town']);
319
+ }
320
+ //region lookup
321
+ if( !empty($args['region']) ){
322
+ $conditions['region'] = $wpdb->prepare('location_region=%s', $args['region']);
323
+ }
324
+ //Add conditions for category selection
325
+ //Filter by category, can be id or comma seperated ids
326
+ //TODO create an exclude category option
327
+ if ( is_numeric($category) && $category > 0 ){
328
+ //get the term id directly
329
+ $term = new EM_Category($category);
330
+ if( $term !== false && !is_wp_error($term) ){
331
+ if( EM_MS_GLOBAL ){
332
+ $conditions['category'] = " ".EM_EVENTS_TABLE.".event_id IN ( SELECT object_id FROM ".EM_META_TABLE." WHERE meta_value={$term->term_id} AND meta_key='event-category' ) ";
333
+ }else{
334
+ $conditions['category'] = " ".EM_EVENTS_TABLE.".post_id IN ( SELECT object_id FROM ".$wpdb->term_relationships." WHERE term_taxonomy_id={$term->term_taxonomy_id} ) ";
335
+ }
336
+ }
337
+ }elseif( self::array_is_numeric($category) ){
338
+ $term_ids = array();
339
+ foreach($category as $category_id){
340
+ $term = new EM_Category($category_id);
341
+ if( $term !== false && !is_wp_error($term) ){
342
+ $term_ids[] = $term->term_taxonomy_id;
343
+ }
344
+ }
345
+ if( count($term_ids) > 0 ){
346
+ if( EM_MS_GLOBAL ){
347
+ $conditions['category'] = " ".EM_EVENTS_TABLE.".event_id IN ( SELECT object_id FROM ".EM_META_TABLE." WHERE meta_value IN (".implode(',',$term_ids).") AND meta_name='event-category ) ";
348
+ }else{
349
+ $conditions['category'] = " ".EM_EVENTS_TABLE.".post_id IN ( SELECT object_id FROM ".$wpdb->term_relationships." WHERE term_taxonomy_id IN (".implode(',',$term_ids).") ) ";
350
+ }
351
+ }
352
+ }
353
+ //Add conditions for tags
354
+ //Filter by tag, can be id or comma seperated ids
355
+ if ( is_numeric($tag) && $tag > 0 ){
356
+ //get the term id directly
357
+ $term = new EM_Tag($tag);
358
+ if( $term !== false && !is_wp_error($term) ){
359
+ $conditions['tag'] = " ".EM_EVENTS_TABLE.".post_id IN ( SELECT object_id FROM ".$wpdb->term_relationships." WHERE term_taxonomy_id={$term->term_taxonomy_id} ) ";
360
+ }
361
+ }elseif( self::array_is_numeric($tag) ){
362
+ $term_ids = array();
363
+ foreach($tag as $tag_id){
364
+ $term = new EM_Tag($tag_id);
365
+ if( $term !== false && !is_wp_error($term) ){
366
+ $term_ids[] = $term->term_taxonomy_id;
367
+ }
368
+ }
369
+ if( count($term_ids) > 0 ){
370
+ $conditions['tag'] = " ".EM_EVENTS_TABLE.".post_id IN ( SELECT object_id FROM ".$wpdb->term_relationships." WHERE term_taxonomy_id IN (".implode(',',$term_ids).") ) ";
371
+ }
372
+ }
373
+
374
+ //If we want rsvped items, we usually check the event
375
+ if( $bookings == 1 ){
376
+ $conditions['bookings'] = 'event_rsvp=1';
377
+ }
378
+ //Default ownership belongs to an event, child objects can just overwrite this if needed.
379
+ if( is_numeric($owner) ){
380
+ $conditions['owner'] = 'event_owner='.$owner;
381
+ }elseif( $owner == 'me' && is_user_logged_in() ){
382
+ $conditions['owner'] = 'event_owner='.get_current_user_id();
383
+ }
384
+ return apply_filters('em_object_build_sql_conditions', $conditions);
385
+ }
386
+
387
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
388
+ //First, ORDER BY
389
+ $args = apply_filters('em_object_build_sql_orderby_args', $args);
390
+ $orderby = array();
391
+ if(is_array($args['orderby'])){
392
+ //Clean orderby array so we only have accepted values
393
+ foreach( $args['orderby'] as $key => $field ){
394
+ if( array_key_exists($field, $accepted_fields) ){
395
+ $orderby[] = $accepted_fields[$field];
396
+ }elseif( in_array($field,$accepted_fields) ){
397
+ $orderby[] = $field;
398
+ }else{
399
+ unset($args['orderby'][$key]);
400
+ }
401
+ }
402
+ }elseif( $args['orderby'] != '' && array_key_exists($args['orderby'], $accepted_fields) ){
403
+ $orderby[] = $accepted_fields[$args['orderby']];
404
+ }elseif( $args['orderby'] != '' && in_array($args['orderby'], $accepted_fields) ){
405
+ $orderby[] = $args['orderby'];
406
+ }
407
+ //ORDER
408
+ //If order is an array, we'll go through the orderby array and match the order values (in order of array) with orderby values
409
+ //If orders don't match up, or it's not ASC/DESC, the default events search in EM settings/options page will be used.
410
+ foreach($orderby as $i => $field){
411
+ $orderby[$i] .= ' ';
412
+ if(is_array($args['order'])){
413
+ if( in_array($args['order'][$i], array('ASC','DESC')) ){
414
+ $orderby[$i] .= $args['order'][$i];
415
+ }else{
416
+ $orderby[$i] .= $default_order;
417
+ }
418
+ }else{
419
+ $orderby[$i] .= ( in_array($args['order'], array('ASC','DESC')) ) ? $args['order'] : $default_order;
420
+ }
421
+ }
422
+ return apply_filters('em_object_build_sql_orderby', $orderby);
423
+ }
424
+
425
+ /**
426
+ * Used by "single" objects, e.g. bookings, events, locations to verify if they have the capability to edit this or someone else's object. Relies on the fact that the object has an owner property with id of user (or admin capability must pass).
427
+ * @param string $owner_capability If the object has an owner property and the user id matches that, this capability will be checked for.
428
+ * @param string $admin_capability If the user isn't the owner of the object, this capability will be checked for.
429
+ * @return boolean
430
+ */
431
+ function can_manage( $owner_capability = false, $admin_capability = false, $user_to_check = false ){
432
+ global $em_capabilities_array;
433
+ if( $user_to_check ){
434
+ $user = new WP_User($user_to_check);
435
+ if( empty($user->ID) ) $user = false;
436
+ }
437
+ //if multisite and supoer admin, just return true
438
+ if( is_multisite() && is_super_admin() ){ return true; }
439
+ //do they own this?
440
+ $is_owner = ( $this->owner == get_current_user_id() || empty($this->id) || (!empty($user) && $this->owner == $user->ID) );
441
+ //now check capability
442
+ $can_manage = false;
443
+ if( $is_owner && (current_user_can($owner_capability) || (!empty($user) && $user->has_cap($owner_capability))) ){
444
+ //user owns the object and can therefore manage it
445
+ $can_manage = true;
446
+ }elseif( array_key_exists($owner_capability, $em_capabilities_array) ){
447
+ //currently user is not able to manage as they aren't the owner
448
+ $error_msg = $em_capabilities_array[$owner_capability];
449
+ }
450
+ //admins have special rights
451
+ if( current_user_can($admin_capability) || (!empty($user) && $user->has_cap($admin_capability)) ){
452
+ $can_manage = true;
453
+ }elseif( array_key_exists($admin_capability, $em_capabilities_array) ){
454
+ $error_msg = $em_capabilities_array[$admin_capability];
455
+ }
456
+
457
+ if( !$can_manage && !$is_owner && !empty($error_msg) ){
458
+ $this->add_error($error_msg);
459
+ }
460
+ return $can_manage;
461
+ }
462
+
463
+
464
+ function ms_global_switch(){
465
+ if( EM_MS_GLOBAL && !is_main_site() ){
466
+ //If in multisite global, then get the main blog categories
467
+ global $current_site;
468
+ switch_to_blog($current_site->blog_id);
469
+ }
470
+ }
471
+
472
+ function ms_global_switch_back(){
473
+ if( EM_MS_GLOBAL && !is_main_site() ){
474
+ restore_current_blog();
475
+ }
476
+ }
477
+
478
+ /**
479
+ * Save an array into this class.
480
+ * If you provide a record from the database table corresponding to this class type it will add the data to this object.
481
+ * @param array $array
482
+ * @return null
483
+ */
484
+ function to_object( $array = array(), $addslashes = false ){
485
+ //Save core data
486
+ if( is_array($array) ){
487
+ $array = apply_filters('em_to_object', $array);
488
+ foreach ( array_keys($this->fields) as $key ) {
489
+ if(array_key_exists($key, $array)){
490
+ if( !is_object($array[$key]) && !is_array($array[$key]) ){
491
+ $array[$key] = ($addslashes) ? stripslashes($array[$key]):$array[$key];
492
+ }elseif( is_array($array[$key]) ){
493
+ $array[$key] = ($addslashes) ? stripslashes_deep($array[$key]):$array[$key];
494
+ }
495
+ $this->$key = $array[$key];
496
+ }
497
+ }
498
+ }
499
+ }
500
+
501
+ /**
502
+ * Copies all the properties to shorter property names for compatability, do not use the old properties.
503
+ */
504
+ function compat_keys(){
505
+ foreach($this->fields as $key => $fieldinfo){
506
+ $this->$fieldinfo['name'] = $this->$key;
507
+ }
508
+ }
509
+
510
+ /**
511
+ * Returns this object in the form of an array, useful for saving directly into a database table.
512
+ * @return array
513
+ */
514
+ function to_array($db = false){
515
+ $array = array();
516
+ foreach ( $this->fields as $key => $val ) {
517
+ if($db){
518
+ if( !empty($this->$key) || $this->$key === 0 || empty($val['null']) ){
519
+ $array[$key] = $this->$key;
520
+ }
521
+ }else{
522
+ $array[$key] = $this->$key;
523
+ }
524
+ }
525
+ return apply_filters('em_to_array', $array);
526
+ }
527
+
528
+
529
+ /**
530
+ * Function to retreive wpdb types for all fields, or if you supply an assoc array with field names as keys it'll return an equivalent array of wpdb types
531
+ * @param array $array
532
+ * @return array:
533
+ */
534
+ function get_types($array = array()){
535
+ $types = array();
536
+ if( count($array)>0 ){
537
+ //So we look at assoc array and find equivalents
538
+ foreach ($array as $key => $val){
539
+ $types[] = $this->fields[$key]['type'];
540
+ }
541
+ }else{
542
+ //Blank array, let's assume we're getting a standard list of types
543
+ foreach ($this->fields as $field){
544
+ $types[] = $field['type'];
545
+ }
546
+ }
547
+ return apply_filters('em_object_get_types', $types, $this, $array);
548
+ }
549
+
550
+ function get_fields( $inverted_array=false ){
551
+ if( is_array($this->fields) ){
552
+ $return = array();
553
+ foreach($this->fields as $fieldName => $fieldArray){
554
+ if($inverted_array){
555
+ $return[$fieldName] = $fieldName;
556
+ }else{
557
+ $return[$fieldName] = $fieldName;
558
+ }
559
+ }
560
+ return apply_filters('em_object_get_fields', $return, $this, $inverted_array);
561
+ }
562
+ return apply_filters('em_object_get_fields', array(), $this, $inverted_array);
563
+ }
564
+
565
+ /**
566
+ * Sanitize text before inserting into database
567
+ * @param string $value
568
+ * @return string
569
+ */
570
+ function sanitize( $value ) {
571
+ if( get_magic_quotes_gpc() )
572
+ $value = stripslashes( $value );
573
+
574
+ //check if this function exists
575
+ if( function_exists( "mysql_real_escape_string" ) ) {
576
+ $value = mysql_real_escape_string( $value );
577
+ //for PHP version < 4.3.0 use addslashes
578
+ } else {
579
+ $value = addslashes( $value );
580
+ }
581
+ return apply_filters('em_object_sanitize', $value);
582
+ }
583
+
584
+ /**
585
+ * Cleans arrays that contain id lists. Takes an array of items and will clean the keys passed in second argument so that if they keep numbers, explode comma-seperated numbers, and unsets the key if there's any other value
586
+ * @param array $array
587
+ * @param array $id_atts
588
+ */
589
+ function clean_id_atts( $array = array(), $id_atts = array() ){
590
+ if( is_array($array) && is_array($id_atts) ){
591
+ foreach( $array as $key => $string ){
592
+ if( in_array($key, $id_atts) ){
593
+ //This is in the list of atts we want cleaned
594
+ if( is_numeric($string) ){
595
+ $array[$key] = (int) $string;
596
+ }elseif( self::array_is_numeric($string) ){
597
+ $array[$key] = $string;
598
+ }elseif( !is_array($string) && preg_match('/^([0-9],?)+$/', $string) ){
599
+ $array[$key] = explode(',', $string);
600
+ }else{
601
+ //No format we accept
602
+ unset($array[$key]);
603
+ }
604
+ }
605
+ }
606
+ }
607
+ return $array;
608
+ }
609
+
610
+ /**
611
+ * Send an email and log errors in this object
612
+ * @param string $subject
613
+ * @param string $body
614
+ * @param string $email
615
+ * @return string
616
+ */
617
+ function email_send($subject, $body, $email){
618
+ global $EM_Mailer;
619
+ if( !$EM_Mailer->send($subject,$body,$email) ){
620
+ if( is_array($EM_Mailer->errors) ){
621
+ foreach($EM_Mailer->errors as $error){
622
+ $this->errors[] = $error;
623
+ }
624
+ }else{
625
+ $this->errors[] = $EM_Mailer->errors;
626
+ }
627
+ return false;
628
+ }
629
+ return true;
630
+ }
631
+
632
+ /**
633
+ * Will return true if this is a simple (non-assoc) numeric array, meaning it has at one or more numeric entries and nothing else
634
+ * @param mixed $array
635
+ * @return boolean
636
+ */
637
+ function array_is_numeric($array){
638
+ $results = array();
639
+ if(is_array($array)){
640
+ foreach($array as $key => $item){
641
+ $results[] = (is_numeric($item)&&is_numeric($key));
642
+ }
643
+ }
644
+ return (!in_array(false, $results) && count($results) > 0);
645
+ }
646
+
647
+ /**
648
+ * Returns an array of errors in this object
649
+ * @return array
650
+ */
651
+ function get_errors(){
652
+ if(is_array($this->errors)){
653
+ return $this->errors;
654
+ }else{
655
+ return array();
656
+ }
657
+ }
658
+
659
+ /**
660
+ * Adds an error to the object
661
+ */
662
+ function add_error($errors){
663
+ if(!is_array($errors)){ $errors = array($errors); } //make errors var an array if it isn't already
664
+ if(!is_array($this->errors)){ $this->errors = array(); } //create empty array if this isn't an array
665
+ foreach($errors as $error){
666
+ if( !in_array($error, $this->errors) ){
667
+ $this->errors[] = $error;
668
+ }
669
+ }
670
+ }
671
+
672
+ /**
673
+ * Converts an array to JSON format, useful for outputting data for AJAX calls. Uses a PHP4 fallback function, given it doesn't support json_encode().
674
+ * @param array $array
675
+ * @return string
676
+ */
677
+ function json_encode($array){
678
+ if( function_exists("json_encode") ){
679
+ $return = json_encode($array);
680
+ }else{
681
+ $return = self::array_to_json($array);
682
+ }
683
+ if( isset($_REQUEST['callback']) ){
684
+ $return = $_REQUEST['callback']."($return)";
685
+ }
686
+ return apply_filters('em_object_json_encode', $return, $array);
687
+ }
688
+
689
+ /**
690
+ * Compatible json encoder function for PHP4
691
+ * @param array $array
692
+ * @return string
693
+ */
694
+ function array_to_json($array){
695
+ //PHP4 Comapatability - This encodes the array into JSON. Thanks go to Andy - http://www.php.net/manual/en/function.json-encode.php#89908
696
+ if( !is_array( $array ) ){
697
+ $array = array();
698
+ }
699
+ $associative = count( array_diff( array_keys($array), array_keys( array_keys( $array )) ));
700
+ if( $associative ){
701
+ $construct = array();
702
+ foreach( $array as $key => $value ){
703
+ // We first copy each key/value pair into a staging array,
704
+ // formatting each key and value properly as we go.
705
+ // Format the key:
706
+ if( is_numeric($key) ){
707
+ $key = "key_$key";
708
+ }
709
+ $key = "'".addslashes($key)."'";
710
+ // Format the value:
711
+ if( is_array( $value )){
712
+ $value = $this->array_to_json( $value );
713
+ }else if( is_bool($value) ) {
714
+ $value = ($value) ? "true" : "false";
715
+ }else if( !is_numeric( $value ) || is_string( $value ) ){
716
+ $value = "'".addslashes($value)."'";
717
+ }
718
+ // Add to staging array:
719
+ $construct[] = "$key: $value";
720
+ }
721
+ // Then we collapse the staging array into the JSON form:
722
+ $result = "{ " . implode( ", ", $construct ) . " }";
723
+ } else { // If the array is a vector (not associative):
724
+ $construct = array();
725
+ foreach( $array as $value ){
726
+ // Format the value:
727
+ if( is_array( $value )){
728
+ $value = $this->array_to_json( $value );
729
+ } else if( !is_numeric( $value ) || is_string( $value ) ){
730
+ $value = "'".addslashes($value)."'";
731
+ }
732
+ // Add to staging array:
733
+ $construct[] = $value;
734
+ }
735
+ // Then we collapse the staging array into the JSON form:
736
+ $result = "[ " . implode( ", ", $construct ) . " ]";
737
+ }
738
+ return $result;
739
+ }
740
+
741
+ /*
742
+ * START IMAGE UPlOAD FUNCTIONS
743
+ * Used for various objects, so shared in one place
744
+ */
745
+ /**
746
+ * Returns the type of image in lowercase, if $path is true, a base filename is returned which indicates where to store the file from the root upload folder.
747
+ * @param unknown_type $path
748
+ * @return mixed|mixed
749
+ */
750
+ function get_image_type($path = false){
751
+ $type = false;
752
+ switch( get_class($this) ){
753
+ case 'EM_Event':
754
+ $dir = (EM_IMAGE_DS == '/') ? 'events/':'';
755
+ $type = 'event';
756
+ break;
757
+ case 'EM_Location':
758
+ $dir = (EM_IMAGE_DS == '/') ? 'locations/':'';
759
+ $type = 'location';
760
+ break;
761
+ case 'EM_Category':
762
+ $dir = (EM_IMAGE_DS == '/') ? 'categories/':'';
763
+ $type = 'category';
764
+ break;
765
+ }
766
+ if($path){
767
+ return apply_filters('em_object_get_image_type',$dir.$type, $path, $this);
768
+ }
769
+ return apply_filters('em_object_get_image_type',$type, $path, $this);
770
+ }
771
+
772
+ function get_image_url($size = 'full'){
773
+ $image_url = $this->image_url;
774
+ if( !empty($this->post_id) && (empty($this->image_url) || $size != 'full') ){
775
+ $post_thumbnail_id = get_post_thumbnail_id( $this->post_id );
776
+ $src = wp_get_attachment_image_src($post_thumbnail_id, $size);
777
+ if( !empty($src[0]) && $size == 'full' ){
778
+ $image_url = $this->image_url = $src[0];
779
+ }elseif(!empty($src[0])){
780
+ $image_url = $src[0];
781
+ }
782
+ //legacy image finder, annoying, but must be done
783
+ if( empty($image_url) ){
784
+ $type = $this->get_image_type();
785
+ $id = ( get_class($this) == "EM_Event" && $this->is_recurrence() ) ? $this->recurrence_id:$this->event_id; //quick fix for recurrences
786
+ if( $type ){
787
+ foreach($this->mime_types as $mime_type) {
788
+ $file_name = $this->get_image_type(true)."-{$id}.$mime_type";
789
+ if( file_exists( EM_IMAGE_UPLOAD_DIR . $file_name) ) {
790
+ $image_url = $this->image_url = EM_IMAGE_UPLOAD_URI.$file_name;
791
+ }
792
+ }
793
+ }
794
+ }
795
+ }
796
+ return apply_filters('em_object_get_image_url', $image_url, $this);
797
+ }
798
+
799
+ function image_delete($force_delete=true) {
800
+ $type = $this->get_image_type();
801
+ if( $type ){
802
+ if( $this->get_image_url() == '' ){
803
+ $result = true;
804
+ }else{
805
+ $post_thumbnail_id = get_post_thumbnail_id( $this->post_id );
806
+ $delete_attachment = wp_delete_attachment($post_thumbnail_id, $force_delete);
807
+ if( $delete_attachment !== false ){
808
+ //check legacy image
809
+ $type_id_name = $type.'_id';
810
+ $file_name= EM_IMAGE_UPLOAD_DIR.$this->get_image_type(true)."-".$this->$type_id_name;
811
+ $result = false;
812
+ foreach($this->mime_types as $mime_type) {
813
+ if (file_exists($file_name.".".$mime_type)){
814
+ $result = unlink($file_name.".".$mime_type);
815
+ $this->image_url = '';
816
+ }
817
+ }
818
+ }
819
+ }
820
+ }
821
+ return apply_filters('em_object_get_image_url', $result, $this);
822
+ }
823
+
824
+ function image_upload(){
825
+ $type = $this->get_image_type();
826
+ //Handle the attachment as a WP Post
827
+ $attachment = '';
828
+ if( $this->can_manage('upload_event_images','upload_event_images') ){
829
+ if ( !empty($_FILES[$type.'_image']['size']) && file_exists($_FILES[$type.'_image']['tmp_name']) && $this->image_validate() ) {
830
+ require_once(ABSPATH . "wp-admin" . '/includes/file.php');
831
+ require_once(ABSPATH . "wp-admin" . '/includes/image.php');
832
+
833
+ $attachment = wp_handle_upload($_FILES[$type.'_image'], array('test_form'=>false), current_time('mysql'));
834
+
835
+ if ( isset($attachment['error']) ){
836
+ $this->add_error('Image Error: ' . $attachment['error'] );
837
+ }
838
+
839
+ /* Attach file to ticket */
840
+ if ( count($this->errors) == 0 && $attachment ){
841
+ $attachment_data = array(
842
+ 'post_mime_type' => $attachment['type'],
843
+ 'post_title' => $this->post_title,
844
+ 'post_content' => '',
845
+ 'post_status' => 'inherit'
846
+ );
847
+ $attachment_id = wp_insert_attachment( $attachment_data, $attachment['file'], $this->post_id );
848
+ $attachment_metadata = wp_generate_attachment_metadata( $attachment_id, $attachment['file'] );
849
+ wp_update_attachment_metadata( $attachment_id, $attachment_metadata );
850
+ //delete the old attachment
851
+ $this->image_delete();
852
+ update_post_meta($this->post_id, '_thumbnail_id', $attachment_id);
853
+ return apply_filters('em_object_image_upload', true, $this);
854
+ }else{
855
+ return apply_filters('em_object_image_upload', false, $this);
856
+ }
857
+ }elseif( !empty($_REQUEST[$type.'_image_delete']) ){
858
+ $this->image_delete();
859
+ }
860
+ }
861
+ return apply_filters('em_object_image_upload', false, $this);
862
+ }
863
+
864
+ function image_validate(){
865
+ $type = $this->get_image_type();
866
+ if( $type ){
867
+ if ( !empty($_FILES[$type.'_image']) && $_FILES[$type.'_image']['size'] > 0 ) {
868
+ if (is_uploaded_file($_FILES[$type.'_image']['tmp_name'])) {
869
+ list($width, $height, $mime_type, $attr) = getimagesize($_FILES[$type.'_image']['tmp_name']);
870
+ $maximum_size = get_option('dbem_image_max_size');
871
+ if ($_FILES[$type.'_image']['size'] > $maximum_size){
872
+ $this->add_error( __('The image file is too big! Maximum size:', 'dbem')." $maximum_size");
873
+ }
874
+ $maximum_width = get_option('dbem_image_max_width');
875
+ $maximum_height = get_option('dbem_image_max_height');
876
+ if (($width > $maximum_width) || ($height > $maximum_height)) {
877
+ $this->add_error( __('The image is too big! Maximum size allowed:','dbem')." $maximum_width x $maximum_height");
878
+ }
879
+ if ( empty($mime_type) || !array_key_exists($mime_type, $this->mime_types) ){
880
+ $this->add_error(__('The image is in a wrong format!','dbem'));
881
+ }
882
+ }
883
+ }
884
+ }
885
+ return apply_filters('em_object_image_validate', count($this->errors) == 0, $this);
886
+ }
887
+ /*
888
+ * END IMAGE UPlOAD FUNCTIONS
889
+ */
890
  }
classes/em-people.php CHANGED
@@ -1,129 +1,34 @@
1
- <?php
2
- class EM_People extends EM_Object {
3
-
4
- /**
5
- * Gets all users, if $return_people false an array associative arrays will be returned. If $return_people is true this will return an array of EM_Person objects
6
- * @param $return_people
7
- * @return array
8
- */
9
- function get( $return_people = true ) {
10
- global $wpdb;
11
- $sql = "SELECT * FROM ". $wpdb->prefix.EM_PEOPLE_TABLE ;
12
- $result = $wpdb->get_results($sql, ARRAY_A);
13
- if( $return_people ){
14
- //Return people as EM_Person objects
15
- $people = array();
16
- foreach ($result as $person){
17
- $people[] = new EM_Person($person);
18
- }
19
- return $people;
20
- }
21
- return $result;
22
- }
23
-
24
- function get_new( $args = array() ) {
25
- global $wpdb;
26
- $people_table = $wpdb->prefix.EM_PEOPLE_TABLE;
27
- $bookings_table = $wpdb->prefix.EM_BOOKINGS_TABLE;
28
-
29
- //Quick version, we can accept an array of IDs, which is easy to retrieve
30
- if( self::array_is_numeric($args) ){ //Array of numbers, assume they are event IDs to retreive
31
- //We can just get all the events here and return them
32
- $sql = "SELECT * FROM $people_table WHERE person_id=".implode(" OR person_id=", $args);
33
- $results = $wpdb->get_results(apply_filters('em_people_get_sql',$sql),ARRAY_A);
34
- $people = array();
35
- foreach($results as $result){
36
- $people[$result['person_id']] = new EM_Person($result);
37
- }
38
- return $people; //We return all the people matched as an EM_Event array.
39
- }
40
-
41
- //We assume it's either an empty array or array of search arguments to merge with defaults
42
- $args = self::get_default_search($args);
43
- $limit = ( $args['limit'] && is_numeric($args['limit'])) ? "LIMIT {$args['limit']}" : '';
44
- $offset = ( $limit != "" && is_numeric($args['offset']) ) ? "OFFSET {$args['offset']}" : '';
45
-
46
- //Get the default conditions
47
- $conditions = self::build_sql_conditions($args);
48
- //Put it all together
49
- $where = ( count($conditions) > 0 ) ? " WHERE " . implode ( " AND ", $conditions ):'';
50
-
51
- //Get ordering instructions
52
- $EM_Person = new EM_Person();
53
- $accepted_fields = $EM_Person->get_fields(true);
54
- $orderby = self::build_sql_orderby($args, $accepted_fields, get_option('dbem_people_default_order'));
55
- //Now, build orderby sql
56
- $orderby_sql = ( count($orderby) > 0 ) ? 'ORDER BY '. implode(', ', $orderby) : '';
57
-
58
- //Create the SQL statement and execute
59
- $sql = "
60
- SELECT * FROM $people_table
61
- LEFT JOIN $bookings_table ON {$bookings_table}.person_id={$people_table}.person_id
62
- $where
63
- GROUP BY person_id
64
- $orderby_sql
65
- $limit $offset
66
- ";
67
- $results = $wpdb->get_results( apply_filters('em_people_get_sql',$sql, $args), ARRAY_A);
68
- //If we want results directly in an array, why not have a shortcut here?
69
- if( $args['array'] == true ){
70
- return $results;
71
- }
72
-
73
- //Make returned results EM_Event objects
74
- $results = (is_array($results)) ? $results:array();
75
- $people = array();
76
- foreach ( $results as $person_array ){
77
- $people[$person_array['person_id']] = new EM_Person($person_array);
78
- }
79
-
80
- return apply_filters('em_people_get', $people);
81
- }
82
-
83
- /* Overrides EM_Object method to apply a filter to result
84
- * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
85
- */
86
- function build_sql_conditions( $args = array() ){
87
- global $wpdb;
88
- //FIXME EM_People doesn't build sql conditions in EM_Object
89
- $conditions = array();
90
-
91
- return apply_filters( 'em_people_build_sql_conditions', $conditions, $args );
92
- }
93
-
94
- /* Overrides EM_Object method to apply a filter to result
95
- * @see wp-content/plugins/people-manager/classes/EM_Object#build_sql_orderby()
96
- */
97
- function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
98
- return apply_filters( 'em_people_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_people_default_order')), $args, $accepted_fields, $default_order );
99
- }
100
-
101
- /*
102
- * Adds custom people search defaults
103
- * @param array $array
104
- * @return array
105
- * @uses EM_Object#get_default_search()
106
- */
107
- function get_default_search( $array = array() ){
108
- $defaults = array(
109
- 'scope'=>false,
110
- 'eventful' => false, //cats that have an event (scope will also play a part here
111
- 'eventless' => false, //cats WITHOUT events, eventful takes precedence
112
- );
113
- //figure out default owning permissions, but since public is for viewing events, only impose limitations in admin area
114
- if( is_admin() ){
115
- switch( get_option('dbem_permissions_events') ){
116
- case 0:
117
- $defaults['owner'] = get_current_user_id();
118
- break;
119
- case 1:
120
- $defaults['owner'] = false;
121
- break;
122
- }
123
- $defaults['owner'] = ( em_verify_admin() ) ? false:$defaults['owner'];
124
- }
125
- return apply_filters('em_people_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
126
- }
127
-
128
- }
129
  ?>
1
+ <?php
2
+ class EM_People extends EM_Object {
3
+
4
+ /**
5
+ * Handles the action of someone being deleted on WordPress
6
+ * @param int $id
7
+ */
8
+ function delete_user( $id ){
9
+ global $wpdb;
10
+ if( current_user_can('delete_users') ){
11
+ if( $_REQUEST['delete_option'] == 'reassign' && is_numeric($_REQUEST['reassign_user']) ){
12
+ $wpdb->update(EM_EVENTS_TABLE, array('event_owner'=>$_REQUEST['reassign_user']), array('event_owner'=>$id));
13
+ }else{
14
+ //User is being deleted, so we delete their events and cancel their bookings.
15
+ $wpdb->query("DELETE FROM ".EM_EVENTS_TABLE." WHERE event_owner=$id");
16
+ }
17
+ }
18
+ //set bookings to cancelled
19
+ $wpdb->update(EM_BOOKINGS_TABLE, array('booking_status'=>3, 'person_id'=>0, 'booking_comment'=>__('User deleted by administrators','dbem')), array('person_id'=>$id));
20
+ }
21
+
22
+ /**
23
+ * Adds phone number to contact info of users, compatible with previous phone field method
24
+ * @param $array
25
+ * @return array
26
+ */
27
+ function user_contactmethods($array){
28
+ $array['dbem_phone'] = __('Phone','dbem') . ' <span class="description">('. __('Events Manager','dbem') .')</span>';
29
+ return $array;
30
+ }
31
+ }
32
+ add_action('delete_user', array('EM_People','delete_user'),10,1);
33
+ add_filter( 'user_contactmethods' , array('EM_People','user_contactmethods'),10,1);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
34
  ?>
classes/em-permalinks.php ADDED
@@ -0,0 +1,200 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ if( !class_exists('EM_Permalinks') ){
4
+ class EM_Permalinks {
5
+ static $em_queryvars = array(
6
+ 'event_id','event_slug', 'em_redirect',
7
+ 'location_id','location_slug',
8
+ 'person_id',
9
+ 'booking_id',
10
+ 'category_id', 'category_slug',
11
+ 'ticket_id',
12
+ 'calendar_day',
13
+ 'rss', 'page', 'bookings_page', 'payment_gateway','event_categories','event_locations'
14
+ );
15
+
16
+ function init(){
17
+ add_filter('pre_update_option_dbem_events_page', array('EM_Permalinks','option_update'));
18
+ if( get_option('dbem_flush_needed') ){
19
+ add_filter('init', array('EM_Permalinks','flush'));
20
+ }
21
+ add_filter('rewrite_rules_array',array('EM_Permalinks','rewrite_rules_array'));
22
+ add_filter('query_vars',array('EM_Permalinks','query_vars'));
23
+ add_action('parse_query',array('EM_Permalinks','init_objects'), 1);
24
+ add_action('parse_query',array('EM_Permalinks','redirection'), 10);
25
+ if( !defined('EM_EVENT_SLUG') ){ define('EM_EVENT_SLUG','event'); }
26
+ if( !defined('EM_LOCATION_SLUG') ){ define('EM_LOCATION_SLUG','location'); }
27
+ if( !defined('EM_LOCATIONS_SLUG') ){ define('EM_LOCATIONS_SLUG','locations'); }
28
+ if( !defined('EM_CATEGORY_SLUG') ){ define('EM_CATEGORY_SLUG','category'); }
29
+ if( !defined('EM_CATEGORIES_SLUG') ){ define('EM_CATEGORIES_SLUG','categories'); }
30
+ }
31
+
32
+ function flush(){
33
+ global $wp_rewrite;
34
+ $wp_rewrite->flush_rules();
35
+ delete_option('dbem_flush_needed');
36
+ }
37
+
38
+ /**
39
+ * will redirect old links to new link structures.
40
+ * @return mixed
41
+ */
42
+ function redirection(){
43
+ global $wpdb, $wp_rewrite, $post, $wp_query;
44
+ if( $wp_query->get('em_redirect') ){
45
+ //is this a querystring url?
46
+ if( $wp_query->get('event_slug') ){
47
+ $event = $wpdb->get_row('SELECT event_id, post_id FROM '.EM_EVENTS_TABLE." WHERE event_slug='".$wp_query->get('event_slug')."' AND (blog_id=".get_current_blog_id()." OR blog_id IS NULL)", ARRAY_A);
48
+ if( $event !== false ){
49
+ $EM_Event = em_get_event($event['event_id']);
50
+ $url = get_permalink($EM_Event->post_id);
51
+ }
52
+ }elseif( $wp_query->get('location_slug') ){
53
+ $location = $wpdb->get_row('SELECT location_id, post_id FROM '.EM_LOCATIONS_TABLE." WHERE location_slug='".$wp_query->get('location_slug')."' AND (blog_id=".get_current_blog_id()." OR blog_id IS NULL)", ARRAY_A);
54
+ if( $location !== false ){
55
+ $EM_Location = em_get_location($location['location_id']);
56
+ $url = get_permalink($EM_Location->post_id);
57
+ }
58
+ }elseif( $wp_query->get('category_slug') ){
59
+ $url = get_term_link($wp_query->get('category_slug'), EM_TAXONOMY_CATEGORY);
60
+ }
61
+ if(!empty($url)){
62
+ wp_redirect($url,301);
63
+ exit();
64
+ }
65
+ }
66
+ }
67
+ // Adding a new rule
68
+ function rewrite_rules_array($rules){
69
+ //get the slug of the event page
70
+ $events_page_id = get_option ( 'dbem_events_page' );
71
+ $events_page = get_post($events_page_id);
72
+ $em_rules = array();
73
+ if( is_object($events_page) ){
74
+ $events_slug = preg_replace('/\/$/', '', str_replace( trailingslashit(home_url()), '', get_permalink($events_page_id)) );
75
+ $events_slug = ( !empty($events_slug) ) ? trailingslashit($events_slug) : $events_slug;
76
+ $em_rules[$events_slug.'(\d{4}-\d{2}-\d{2})$'] = 'index.php?pagename='.$events_slug.'&calendar_day=$matches[1]'; //event calendar date search
77
+ if( !get_option( 'dbem_my_bookings_page') || !is_object(get_post(get_option( 'dbem_my_bookings_page'))) ){
78
+ $em_rules[$events_slug.'my\-bookings$'] = 'index.php?pagename='.$events_slug.'&bookings_page=1'; //page for users to manage bookings
79
+ }
80
+ $em_rules[$events_slug.'rss$'] = 'index.php?pagename='.$events_slug.'&rss=1'; //rss page
81
+ $em_rules[$events_slug.'feed$'] = 'index.php?pagename='.$events_slug.'&rss=1'; //compatible rss page
82
+ $em_rules[$events_slug.'payments/(.+)$'] = 'index.php?pagename='.$events_slug.'&payment_gateway=$matches[1]'; //single event booking form with slug
83
+ if( EM_POST_TYPE_EVENT_SLUG.'/' == $events_slug ){ //won't apply on homepage
84
+ //make sure we hard-code rewrites for child pages of events
85
+ $child_posts = get_posts(array('post_type'=>'page', 'post_parent'=>$events_page->ID));
86
+ foreach($child_posts as $child_post){
87
+ $em_rules[$events_slug.$child_post->post_name.'/?$'] = 'index.php?page_id='.$child_post->ID; //single event booking form with slug
88
+ }
89
+ }
90
+ if( EM_MS_GLOBAL && !get_site_option('dbem_ms_global_events_links', true) ){
91
+ //MS Mode has slug also for global links
92
+ $em_rules[$events_slug.get_site_option('dbem_ms_events_slug',EM_EVENT_SLUG).'/(.+)$'] = 'index.php?pagename='.$events_slug.'&em_redirect=1&event_slug=$matches[1]'; //single event from subsite
93
+ }
94
+ //add redirection for backwards compatability
95
+ $em_rules[$events_slug.EM_EVENT_SLUG.'/(.+)$'] = 'index.php?pagename='.$events_slug.'&em_redirect=1&event_slug=$matches[1]'; //single event
96
+ $em_rules[$events_slug.EM_LOCATION_SLUG.'/(.+)$'] = 'index.php?pagename='.$events_slug.'&em_redirect=1&location_slug=$matches[1]'; //single location page
97
+ $em_rules[$events_slug.EM_CATEGORY_SLUG.'/(.+)$'] = 'index.php?pagename='.$events_slug.'&em_redirect=1&category_slug=$matches[1]'; //single category page slug
98
+ }else{
99
+ $events_slug = EM_POST_TYPE_EVENT_SLUG;
100
+ $em_rules[$events_slug.'/(\d{4}-\d{2}-\d{2})$'] = 'index.php?post_type='.EM_POST_TYPE_EVENT.'&calendar_day=$matches[1]'; //event calendar date search
101
+ $em_rules[$events_slug.'/my\-bookings$'] = 'index.php?post_type='.EM_POST_TYPE_EVENT.'&bookings_page=1'; //page for users to manage bookings
102
+ $em_rules[$events_slug.'/rss$'] = 'index.php?post_type='.EM_POST_TYPE_EVENT.'&rss=1'; //rss page
103
+ $em_rules[$events_slug.'/payments/(.+)$'] = 'index.php?post_type='.EM_POST_TYPE_EVENT.'&payment_gateway=$matches[1]'; //single event booking form with slug
104
+ }
105
+ //If in MS global mode and locations are linked on same site
106
+ if( EM_MS_GLOBAL && !get_site_option('dbem_ms_global_locations_links', true) ){
107
+ $locations_page_id = get_option ( 'dbem_locations_page' );
108
+ $locations_page = get_post($locations_page_id);
109
+ if( is_object($locations_page) ){
110
+ $locations_slug = preg_replace('/\/$/', '', str_replace( trailingslashit(home_url()), '', get_permalink($locations_page_id) ));
111
+ $em_rules[$locations_slug.'/'.get_site_option('dbem_ms_locations_slug',EM_LOCATION_SLUG).'/(.+)$'] = 'index.php?pagename='.$locations_slug.'&location_slug=$matches[1]'; //single event booking form with slug
112
+ }
113
+ }
114
+ return $em_rules + $rules;
115
+ }
116
+
117
+ /**
118
+ * Depreciated, use get_post_permalink() from now on or the output function with a placeholder
119
+ * Generate a URL. Pass each section of a link as a parameter, e.g. EM_Permalinks::url('event',$event_id); will create an event link.
120
+ * @param mixed
121
+ */
122
+ function url(){
123
+ global $wp_rewrite;
124
+ $args = func_get_args();
125
+ $em_uri = get_permalink(get_option("dbem_events_page")); //PAGE URI OF EM
126
+ if ( $wp_rewrite->using_permalinks() /*&& !defined('EM_DISABLE_PERMALINKS')*/ ) {
127
+ $event_link = trailingslashit(trailingslashit($em_uri). implode('/',$args));
128
+ }
129
+ return $event_link;
130
+ }
131
+
132
+ /**
133
+ * checks if the events page has changed, and sets a flag to flush wp_rewrite.
134
+ * @param mixed $val
135
+ * @return mixed
136
+ */
137
+ function option_update( $val ){
138
+ if( get_option('dbem_events_page') != $val ){
139
+ update_option('dbem_flush_needed',1);
140
+ }
141
+ return $val;
142
+ }
143
+
144
+ // Adding the id var so that WP recognizes it
145
+ function query_vars($vars){
146
+ foreach(self::$em_queryvars as $em_queryvar){
147
+ array_push($vars, $em_queryvar);
148
+ }
149
+ return $vars;
150
+ }
151
+
152
+ /**
153
+ * Not the "WP way" but for now this'll do!
154
+ */
155
+ function init_objects(){
156
+ global $wp_query, $wp_rewrite;
157
+ //check some homepage conditions
158
+ $events_page_id = get_option ( 'dbem_events_page' );
159
+ if( $wp_query->is_home && 'page' == get_option('show_on_front') && get_option('page_on_front') == $events_page_id ){
160
+ $wp_query->is_page = true;
161
+ $wp_query->is_home = false;
162
+ $wp_query->query_vars['page_id'] = $events_page_id;
163
+ }
164
+ if ( $wp_rewrite->using_permalinks() ) {
165
+ foreach(self::$em_queryvars as $em_queryvar){
166
+ if( $wp_query->get($em_queryvar) ) {
167
+ $_REQUEST[$em_queryvar] = $wp_query->get($em_queryvar);
168
+ }
169
+ }
170
+ }
171
+ //dirty rss condition
172
+ if( !empty($_REQUEST['rss']) ){
173
+ $_REQUEST['rss_main'] = 'main';
174
+ }
175
+ }
176
+ }
177
+ EM_Permalinks::init();
178
+ }
179
+
180
+ //Specific links that aren't generated by objects
181
+
182
+ /**
183
+ * returns the url of the my bookings page, depending on the settings page and if BP is installed.
184
+ * @return string
185
+ */
186
+ function em_get_my_bookings_url(){
187
+ global $bp, $wp_rewrite;
188
+ if( is_object($bp) ){
189
+ //get member url
190
+ return $bp->events->link.'attending/';
191
+ }elseif( get_option('dbem_bookings_my_page') ){
192
+ return get_permalink(get_option('dbem_bookings_my_page'));
193
+ }else{
194
+ if( $wp_rewrite->using_permalinks() && !defined('EM_DISABLE_PERMALINKS') ){
195
+ return trailingslashit(EM_URI)."my-bookings/";
196
+ }else{
197
+ return preg_match('/\?/',EM_URI) ? EM_URI.'&bookings_page=1':EM_URI.'?bookings_page=1';
198
+ }
199
+ }
200
+ }
classes/em-person.php CHANGED
@@ -1,154 +1,92 @@
1
- <?php
2
- // TODO make person details more secure and integrate with WP user data
3
- class EM_Person extends EM_Object{
4
- //DB Fields
5
- var $id = '';
6
- var $owner = '';
7
- var $name = '';
8
- var $email = '';
9
- var $phone = '';
10
- //Other Vars
11
- var $fields = array(
12
- 'person_id' => array('name'=>'id','type'=>'%d'),
13
- 'person_name' => array('name'=>'name','type'=>'%s'),
14
- 'person_email' => array('name'=>'email','type'=>'%s'),
15
- 'person_phone' => array('name'=>'phone','type'=>'%s')
16
- );
17
- var $required_fields = array('person_id', 'person_name', 'person_email', 'person_phone');
18
- var $feedback_message = "";
19
- var $errors = array();
20
-
21
- function EM_Person( $person_data = false ){
22
- if( $person_data != 0 ){
23
- //Load person data
24
- $person = array();
25
- if( is_array($person_data) ){
26
- $person = $person_data;
27
- }elseif( is_numeric($person_data) ){
28
- //Retreiving from the database
29
- global $wpdb;
30
- $sql = "SELECT * FROM ". $wpdb->prefix . EM_PEOPLE_TABLE ." WHERE person_id ='$person_data'";
31
- $person = $wpdb->get_row($sql, ARRAY_A);
32
- }
33
- //Save into the object
34
- $this->to_object($person);
35
- }
36
- }
37
-
38
- /**
39
- * Load an record into this object by passing an associative array of table criterie to search for.
40
- * Returns boolean depending on whether a record is found or not.
41
- * @param $search
42
- * @return boolean
43
- */
44
- function get($search) {
45
- global $wpdb;
46
- $conds = array();
47
- foreach($search as $key => $value) {
48
- if( array_key_exists($key, $this->fields) ){
49
- $conds[] = "`$key`='$value'";
50
- }
51
- }
52
- $sql = "SELECT * FROM ". $wpdb->prefix.EM_PEOPLE_TABLE ." WHERE " . implode(' AND ', $conds) ;
53
- $result = $wpdb->get_row($sql, ARRAY_A);
54
- if($result){
55
- $this->to_object($result);
56
- return true;
57
- }else{
58
- return false;
59
- }
60
- }
61
-
62
- function get_post(){
63
- $this->name = ( !empty($_REQUEST['person_name']) ) ? stripslashes($_REQUEST['person_name']) : '' ;
64
- $this->email = ( !empty($_REQUEST['person_email']) ) ? $_REQUEST['person_email'] : '';
65
- $this->phone = ( !empty($_REQUEST['person_phone']) ) ? $_REQUEST['person_phone'] : '';
66
- return apply_filters('em_person_get_post', $this->validate(), $this);
67
- }
68
-
69
- function save(){
70
- global $wpdb;
71
- do_action('em_person_save_pre',$this);
72
- if($this->validate()){
73
- //Does this person already exist?
74
- $this->load_similar();
75
- $table = $wpdb->prefix.EM_PEOPLE_TABLE;
76
- $data = $this->to_array();
77
- unset($data['person_id']);
78
- if($this->id != ''){
79
- $where = array( 'person_id' => $this->id );
80
- $result = $wpdb->update($table, $data, $where);
81
- }else{
82
- $result = $wpdb->insert($table, $data);
83
- $this->id = $wpdb->insert_id;
84
- }
85
- $this->feedback_message = ($result) ? __('Changes saved.'): __('Could not edit person details.','dbem');
86
- return apply_filters('em_save_delete', $result, $this);
87
- }else{
88
- $this->feedback_message = __('Could not edit person details.','dbem');
89
- return apply_filters('em_save_delete', false, $this);
90
- }
91
- }
92
-
93
- function delete(){
94
- global $wpdb;
95
- do_action('em_person_delete_pre', $this);
96
- $results = array();
97
- $results[] = $wpdb->query ( $wpdb->prepare("DELETE FROM ". $wpdb->prefix . EM_PEOPLE_TABLE ." WHERE person_id=%d", $this->id) );
98
- $results[] = $wpdb->query ( $wpdb->prepare("DELETE FROM ". $wpdb->prefix . EM_BOOKINGS_TABLE ." WHERE person_id=%d", $this->id) );
99
- return apply_filters('em_event_delete', !in_array(false,$results), $this);
100
- }
101
-
102
- function validate(){
103
- if( !preg_match('/^[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Za-z]{2,4}$/', $this->email) ){
104
- $this->errors[] = __('Please provide a valid email address.', 'dbem');
105
- return false;
106
- }
107
- return true;
108
- }
109
-
110
- function get_bookings(){
111
- global $wpdb;
112
- $results = $wpdb->get_results("SELECT * FROM ".$wpdb->prefix.EM_BOOKINGS_TABLE." WHERE person_id={$this->id}",ARRAY_A);
113
- $bookings = array();
114
- foreach($results as $booking_data){
115
- $bookings[] = new EM_Booking($booking_data);
116
- }
117
- return $bookings;
118
- }
119
-
120
- /**
121
- * Checks agains the database to see if this user exists already
122
- * @return boolean|int
123
- */
124
- function find_similar(){
125
- global $wpdb;
126
- $sql = "SELECT * FROM ". $wpdb->prefix.EM_PEOPLE_TABLE ." WHERE person_name='%s' AND person_email='%s' AND person_phone='%s'";
127
- $row = $wpdb->get_row( $wpdb->prepare($sql, array($this->name, $this->email, $this->phone)), ARRAY_A );
128
- if( is_array($row) ){
129
- return $row['person_id'];
130
- }
131
- return false;
132
- }
133
-
134
- /**
135
- * Checks if a similar record exists (same name, email and phone) and if so it loads
136
- * @return boolean
137
- */
138
- function load_similar(){
139
- $return = $this->find_similar();
140
- if( is_numeric($return) ){
141
- $this->id = $return;
142
- return true;
143
- }
144
- return false;
145
- }
146
-
147
- /**
148
- * Can the user manage this event?
149
- */
150
- function can_manage(){
151
- return ( get_option('dbem_permissions_events') || $this->owner == get_current_user_id() || em_verify_admin() );
152
- }
153
- }
154
  ?>
1
+ <?php
2
+ // TODO make person details more secure and integrate with WP user data
3
+ class EM_Person extends WP_User{
4
+
5
+ function EM_Person( $person_id = false, $username = false ){
6
+ if( is_array($person_id) ){
7
+ if( array_key_exists('person_id',$person_id) ){
8
+ $person_id = $person_id['person_id'];
9
+ }elseif ( array_key_exists('user_id',$person_id) ){
10
+ $person_id = $person_id['user_id'];
11
+ }else{
12
+ $person_id = $person_id['ID'];
13
+ }
14
+ }elseif( is_object($person_id) && get_class($person_id) == 'WP_User'){
15
+ $person_id = $person_id->ID; //create new object if passed a wp_user
16
+ }
17
+ if($username){
18
+ parent::__construct($person_id, $username);
19
+ }elseif( is_numeric($person_id) && $person_id == 0 ){
20
+ $this->ID = 0;
21
+ $this->display_name = 'Non-Registered User';
22
+ $this->user_email = 'n/a';
23
+ }else{
24
+ parent::__construct($person_id);
25
+ }
26
+ $this->phone = get_metadata('user', $this->ID, 'dbem_phone', true); //extra field for EM
27
+ do_action('em_person',$this, $person_id, $username);
28
+ }
29
+
30
+ function get_bookings($ids_only = false){
31
+ global $wpdb;
32
+ $blog_condition = '';
33
+ if( is_multisite() ){
34
+ if( !is_main_site() ){
35
+ //not the main blog, force single blog search
36
+ $blog_condition = "AND e.blog_id=".get_current_blog_id();
37
+ }elseif(is_main_site() && !get_option('dbem_ms_global_events')){
38
+ $blog_condition = "AND (e.blog_id=".get_current_blog_id().' OR e.blog_id IS NULL)';
39
+ }
40
+ }
41
+ $EM_Booking = new EM_Booking(); //empty booking for fields
42
+ $results = $wpdb->get_results("SELECT b.".implode(', b.', array_keys($EM_Booking->fields))." FROM ".EM_BOOKINGS_TABLE." b, ".EM_EVENTS_TABLE." e WHERE e.event_id=b.event_id AND person_id={$this->ID} {$blog_condition} ORDER BY event_start_date DESC",ARRAY_A);
43
+ $bookings = array();
44
+ if($ids_only){
45
+ foreach($results as $booking_data){
46
+ $bookings[] = $booking_data['booking_id'];
47
+ }
48
+ return $bookings;
49
+ }else{
50
+ foreach($results as $booking_data){
51
+ $bookings[] = new EM_Booking($booking_data);
52
+ }
53
+ return new EM_Bookings($bookings);
54
+ }
55
+ }
56
+
57
+ /**
58
+ * @return EM_Events
59
+ */
60
+ function get_events(){
61
+ global $wpdb;
62
+ $events = array();
63
+ foreach( $this->get_bookings()->get_bookings() as $EM_Booking ){
64
+ $events[$EM_Booking->event_id] = $EM_Booking->get_event();
65
+ }
66
+ return $events;
67
+ }
68
+
69
+ function display_summary(){
70
+ ob_start();
71
+ ?>
72
+ <table>
73
+ <tr>
74
+ <td><?php echo get_avatar($this->ID); ?></td>
75
+ <td style="padding-left:10px; vertical-align: top;">
76
+ <strong><?php _e('Name','dbem'); ?></strong> : <a href="<?php echo EM_ADMIN_URL ?>&amp;page=events-manager-bookings&amp;person_id=<?php echo $this->ID; ?>"><?php echo $this->get_name() ?></a><br /><br />
77
+ <strong><?php _e('Email','dbem'); ?></strong> : <?php echo $this->user_email; ?><br /><br />
78
+ <strong><?php _e('Phone','dbem'); ?></strong> : <?php echo $this->phone; ?>
79
+ </td>
80
+ </tr>
81
+ </table>
82
+ <?php
83
+ return ob_get_clean();
84
+ }
85
+
86
+ function get_name(){
87
+ $full_name = $this->user_firstname . " " . $this->user_lastname ;
88
+ $full_name = trim($full_name);
89
+ return !empty($full_name) ? $full_name : $this->display_name;
90
+ }
91
+ }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
92
  ?>
classes/em-recurrence.php DELETED
@@ -1,89 +0,0 @@
1
- <?php
2
- /**
3
- * Object that holds event recurrence info
4
- * The EM_Event objects will handle adding / editing / deleting recurrences, this just holds the recurring info
5
- * and provides extra functions to obtain related events.
6
- * @author marcus
7
- */
8
- class EM_Recurrence extends EM_Object{
9
- //TODO We NEED to add an author and rsvp to this recurrence.....
10
- var $fields = array(
11
- 'recurrence_id' => 'id',
12
- 'recurrence_author' => 'author',
13
- 'recurrence_name' => 'name',
14
- 'recurrence_start_time' => 'start_time',
15
- 'recurrence_end_time' => 'end_time',
16
- 'recurrence_start_date' => 'start_date',
17
- 'recurrence_end_date' => 'end_date',
18
- 'recurrence_notes' => 'notes',
19
- 'recurrence_rsvp' => 'rsvp',
20
- 'recurrence_seats' => 'seats',
21
- 'recurrence_contactperson_id' => 'contactperson_id',
22
- 'location_id' => 'location_id',
23
- 'recurrence_id' => 'recurrence_id',
24
- 'recurrence_category_id' => 'category_id',
25
- 'recurrence_attributes' => 'attributes',
26
- 'recurrence_interval' => 'interval',
27
- 'recurrence_freq' => 'freq',
28
- 'recurrence_byday' => 'byday',
29
- 'recurrence_byweekno' => 'byweekno',
30
- );
31
- /**
32
- * Array of EM_Event objects
33
- * @var array
34
- */
35
- var $events = array();
36
-
37
- /**
38
- * Initialize object. You can provide event data in an associative array (using database table field names), an id number, or false (default) to create empty event.
39
- * If you provide an event array or object, it will convert it into a recurrence (useful if you want to change event data into recurrence).
40
- * @param mixed $event_data
41
- * @return null
42
- */
43
- function EM_Recurrence($event_data = false) {
44
- global $wpdb;
45
- if( $event_data !== false ){
46
- $recurrence = array();
47
- if( is_array($event_data) && isset($event_data['recurrence_name']) ){
48
- //Directly inserting array of recurrence data
49
- $recurrence = $event_data;
50
- }elseif( is_numeric($event_data) || isset($event_data['recurrence_id']) ){
51
- //$event_data is recurrence_id - Retreiving from the database
52
- $recurrence_id = (is_array($event_data)) ? $event_data['recurrence_id']:$event_data;
53
- $sql = "SELECT * FROM ". $wpdb->prefix . EM_RECURRENCE_TABLE ." WHERE recurrence_id = $recurrence_id";
54
- $result = $wpdb->get_row( $sql, ARRAY_A );
55
- if($result){
56
- $this->location = new EM_Location ( $recurrence ['location_id'] );
57
- $recurrence = $result;
58
- }
59
- }
60
- $this->to_object($recurrence);
61
- }
62
- }
63
-
64
- /**
65
- * Removes recurrence record.
66
- * @param $recurrence_id
67
- * @return null
68
- */
69
- function delete() {
70
- global $wpdb;
71
- $sql = "DELETE FROM ".$wpdb->prefix.EM_RECURRENCE_TABLE." WHERE recurrence_id = '{$this->id}';";
72
- $wpdb->query($sql);
73
- }
74
-
75
- /**
76
- * Save an array into this class
77
- * @param $array
78
- * @return null
79
- */
80
- function to_object( $array = array() ){
81
- //Save event core data
82
- parent::to_object($array);
83
- //Save location info
84
- $this->location = new EM_Location($array['location_id']);
85
- //Save contact person info
86
- }
87
-
88
- }
89
- ?>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
classes/em-tag-taxonomy.php ADDED
@@ -0,0 +1,63 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Tag_Taxonomy{
3
+ function init(){
4
+ if( !is_admin() ){
5
+ add_filter('archive_template', array('EM_Tag_Taxonomy','template'));
6
+ add_filter('category_template', array('EM_Tag_Taxonomy','template'));
7
+ add_filter('parse_query', array('EM_Tag_Taxonomy','parse_query'));
8
+ }
9
+ }
10
+ /**
11
+ * Overrides archive pages e.g. locations, events, event tags, event tags based on user settings
12
+ * @param string $template
13
+ * @return string
14
+ */
15
+ function template($template){
16
+ global $wp_query;
17
+ if( is_archive() ){
18
+ if( !empty($wp_query->queried_object->taxonomy) && $wp_query->queried_object->taxonomy == EM_TAXONOMY_TAG && get_option('dbem_cp_tags_formats', true)){
19
+ add_filter('the_content', array('EM_Tag_Taxonomy','the_content'));
20
+ $EM_Tag = em_get_tag($wp_query->queried_object->term_id);
21
+ $wp_query->posts = array();
22
+ $wp_query->posts[0] = new stdClass();
23
+ $wp_query->posts[0]->post_title = $EM_Tag->output(get_option('dbem_tag_page_title_format'));
24
+ $wp_query->posts[0]->post_content = '';
25
+ $wp_query->post = $wp_query->posts[0];
26
+ $wp_query->post_count = 1;
27
+ $wp_query->found_posts = 1;
28
+ $wp_query->max_num_pages = 1;
29
+ $template = locate_template(array('page.php','index.php'),false); //category becomes a page
30
+ }
31
+ }
32
+ return $template;
33
+ }
34
+
35
+ function the_content($content){
36
+ global $wp_query, $EM_Tag;
37
+ $EM_Tag = new EM_Tag($wp_query->queried_object);
38
+ ob_start();
39
+ em_locate_template('templates/tag-single.php',true);
40
+ return ob_get_clean();
41
+ }
42
+
43
+ function parse_query( ){
44
+ global $wp_query;
45
+ if( !empty($wp_query->tax_query->queries[0]['taxonomy']) && $wp_query->tax_query->queries[0]['taxonomy'] == EM_TAXONOMY_TAG) {
46
+ //Scope is future
47
+ $today = strtotime(date('Y-m-d', current_time('timestamp')));
48
+ if( get_option('dbem_events_current_are_past') ){
49
+ $wp_query->query_vars['meta_query'][] = array( 'key' => '_start_ts', 'value' => $today, 'compare' => '>=' );
50
+ }else{
51
+ $wp_query->query_vars['meta_query'][] = array( 'key' => '_end_ts', 'value' => $today, 'compare' => '>=' );
52
+ }
53
+ if( get_option('dbem_tags_default_archive_orderby') == 'title'){
54
+ $wp_query->query_vars['orderby'] = 'title';
55
+ }else{
56
+ $wp_query->query_vars['orderby'] = 'meta_value_num';
57
+ $wp_query->query_vars['meta_key'] = get_option('dbem_tags_default_archive_orderby','_start_ts');
58
+ }
59
+ $wp_query->query_vars['order'] = get_option('dbem_tags_default_archive_order','ASC');
60
+ }
61
+ }
62
+ }
63
+ EM_Tag_Taxonomy::init();
classes/em-tag.php ADDED
@@ -0,0 +1,135 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Get an tag in a db friendly way, by checking globals and passed variables to avoid extra class instantiations
4
+ * @param mixed $id
5
+ * @return EM_Tag
6
+ */
7
+ function em_get_tag($id = false) {
8
+ global $EM_Tag;
9
+ //check if it's not already global so we don't instantiate again
10
+ if( is_object($EM_Tag) && get_class($EM_Tag) == 'EM_Tag' ){
11
+ if( $EM_Tag->term_id == $id ){
12
+ return $EM_Tag;
13
+ }elseif( is_object($id) && $EM_Tag->term_id == $id->term_id ){
14
+ return $EM_Tag;
15
+ }
16
+ }
17
+ if( is_object($id) && get_class($id) == 'EM_Tag' ){
18
+ return $id;
19
+ }else{
20
+ return new EM_Tag($id);
21
+ }
22
+ }
23
+ class EM_Tag extends EM_Object {
24
+ //Taxonomy Fields
25
+ var $id = '';
26
+ var $term_id;
27
+ var $name;
28
+ var $slug;
29
+ var $term_group;
30
+ var $term_taxonomy_id;
31
+ var $taxonomy;
32
+ var $description = '';
33
+ var $parent = 0;
34
+ var $count;
35
+
36
+ /**
37
+ * Gets data from POST (default), supplied array, or from the database if an ID is supplied
38
+ * @param $tag_data
39
+ * @return null
40
+ */
41
+ function EM_Tag( $tag_data = false ) {
42
+ global $wpdb;
43
+ //Initialize
44
+ $tag = array();
45
+ if( !empty($tag_data) ){
46
+ //Load tag data
47
+ if( is_object($tag_data) && !empty($tag_data->taxonomy) && $tag_data->taxonomy == EM_TAXONOMY_TAG ){
48
+ $tag = $tag_data;
49
+ }elseif( !is_numeric($tag_data) ){
50
+ $tag = get_term_by('slug', $tag_data, EM_TAXONOMY_TAG);
51
+ }else{
52
+ $tag = get_term_by('id', $tag_data, EM_TAXONOMY_TAG);
53
+ }
54
+ }
55
+ foreach($tag as $key => $value){
56
+ $this->$key = $value;
57
+ }
58
+ $this->id = $this->term_id; //backward compatability
59
+ do_action('em_tag',$this, $tag_data);
60
+ }
61
+
62
+ function get_url(){
63
+ if( empty($this->link) ){
64
+ $this->ms_global_switch();
65
+ $this->link = get_term_link($this->slug, EM_TAXONOMY_TAG);
66
+ $this->ms_global_switch_back();
67
+ if ( is_wp_error($this->link) ) $this->link = '';
68
+ }
69
+ return $this->link;
70
+ }
71
+
72
+ function output_single($target = 'html'){
73
+ $format = get_option ( 'dbem_tag_page_format' );
74
+ return apply_filters('em_tag_output_single', $this->output($format, $target), $this, $target);
75
+ }
76
+
77
+ function output($format, $target="html") {
78
+ preg_match_all('/\{([a-zA-Z0-9_]+)\}([^{]+)\{\/[a-zA-Z0-9_]+\}/', $format, $conditionals);
79
+ if( count($conditionals[0]) > 0 ){
80
+ foreach($conditionals[1] as $key => $condition){
81
+ $format = str_replace($conditionals[0][$key], apply_filters('em_tag_output_condition', '', $condition, $conditionals[0][$key], $this), $format);
82
+ }
83
+ }
84
+ $tag_string = $format;
85
+ preg_match_all("/(#@?_?[A-Za-z0-9]+)({([a-zA-Z0-9,]+)})?/", $format, $placeholders);
86
+ foreach($placeholders[1] as $key => $result) {
87
+ $match = true;
88
+ $replace = '';
89
+ $full_result = $placeholders[0][$key];
90
+ switch( $result ){
91
+ case '#_TAGNAME':
92
+ $replace = $this->name;
93
+ break;
94
+ case '#_TAGID':
95
+ $replace = $this->term_id;
96
+ break;
97
+ case '#_TAGLINK':
98
+ case '#_TAGURL':
99
+ $link = $this->get_url();
100
+ $replace = ($result == '#_TAGURL') ? $link : '<a href="'.$link.'">'.esc_html($this->name).'</a>';
101
+ break;
102
+ case '#_TAGEVENTSPAST': //depreciated, erroneous documentation, left for compatability
103
+ case '#_TAGEVENTSNEXT': //depreciated, erroneous documentation, left for compatability
104
+ case '#_TAGEVENTSALL': //depreciated, erroneous documentation, left for compatability
105
+ case '#_TAGPASTEVENTS':
106
+ case '#_TAGNEXTEVENTS':
107
+ case '#_TAGALLEVENTS':
108
+ //convert depreciated placeholders for compatability
109
+ $result = ($result == '#_TAGEVENTSPAST') ? '#_TAGPASTEVENTS':$result;
110
+ $result = ($result == '#_TAGEVENTSNEXT') ? '#_TAGNEXTEVENTS':$result;
111
+ $result = ($result == '#_TAGEVENTSALL') ? '#_TAGALLEVENTS':$result;
112
+ //forget it ever happened? :/
113
+ if ($result == '#_TAGPASTEVENTS'){ $scope = 'past'; }
114
+ elseif ( $result == '#_TAGNEXTEVENTS' ){ $scope = 'future'; }
115
+ else{ $scope = 'all'; }
116
+ $events = EM_Events::get( array('tag'=>$this->term_id, 'scope'=>$scope) );
117
+ if ( count($events) > 0 ){
118
+ foreach($events as $EM_Event){
119
+ $replace .= $EM_Event->output(get_option('dbem_tag_event_list_item_format'));
120
+ }
121
+ } else {
122
+ $replace = get_option('dbem_tag_no_events_message');
123
+ }
124
+ break;
125
+ default:
126
+ $replace = $full_result;
127
+ break;
128
+ }
129
+ $replace = apply_filters('em_tag_output_placeholder', $replace, $this, $full_result, $target); //USE WITH CAUTION! THIS MIGHT GET RENAMED
130
+ $tag_string = str_replace($full_result, $replace , $tag_string );
131
+ }
132
+ return apply_filters('em_tag_output', $tag_string, $this, $format, $target);
133
+ }
134
+ }
135
+ ?>
classes/em-ticket-booking.php ADDED
@@ -0,0 +1,223 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Ticket_Booking extends EM_Object{
3
+ //DB Fields
4
+ var $ticket_booking_id;
5
+ var $booking_id;
6
+ var $ticket_id;
7
+ var $ticket_booking_price;
8
+ var $ticket_booking_spaces;
9
+ var $fields = array(
10
+ 'ticket_booking_id' => array('name'=>'id','type'=>'%d'),
11
+ 'ticket_id' => array('name'=>'ticket_id','type'=>'%d'),
12
+ 'booking_id' => array('name'=>'booking_id','type'=>'%d'),
13
+ 'ticket_booking_price' => array('name'=>'price','type'=>'%f'),
14
+ 'ticket_booking_spaces' => array('name'=>'spaces','type'=>'%d')
15
+ );
16
+ //Other Vars
17
+ /**
18
+ * Contains ticket object
19
+ * @var EM_Ticket
20
+ */
21
+ var $ticket;
22
+ /**
23
+ * Contains the booking object of this
24
+ * @var EM_Booking
25
+ */
26
+ var $booking;
27
+ var $required_fields = array( 'ticket_id', 'ticket_booking_spaces');
28
+
29
+ /**
30
+ * Creates ticket object and retreives ticket data (default is a blank ticket object). Accepts either array of ticket data (from db) or a ticket id.
31
+ * @param mixed $ticket_data
32
+ * @return null
33
+ */
34
+ function EM_Ticket_Booking( $ticket_data = false ){
35
+ if( $ticket_data !== false ){
36
+ //Load ticket data
37
+ $ticket = array();
38
+ if( is_array($ticket_data) ){
39
+ $ticket = $ticket_data;
40
+ }elseif( is_numeric($ticket_data) ){
41
+ //Retreiving from the database
42
+ global $wpdb;
43
+ $sql = "SELECT * FROM ". EM_TICKETS_BOOKINGS_TABLE ." WHERE ticket_booking_id ='$ticket_data'";
44
+ $ticket = $wpdb->get_row($sql, ARRAY_A);
45
+ }
46
+ //Save into the object
47
+ $this->to_object($ticket);
48
+ $this->compat_keys();
49
+ }
50
+ }
51
+
52
+ /**
53
+ * Saves the ticket into the database, whether a new or existing ticket
54
+ * @return boolean
55
+ */
56
+ function save(){
57
+ global $wpdb;
58
+ $table = EM_TICKETS_BOOKINGS_TABLE;
59
+ do_action('em_ticket_booking_save_pre',$this);
60
+ //First the person
61
+ if($this->validate()){
62
+ //Now we save the ticket
63
+ $this->booking_id = $this->get_booking()->booking_id; //event wouldn't exist before save, so refresh id
64
+ $data = $this->to_array(true); //add the true to remove the nulls
65
+ if($this->ticket_booking_id != ''){
66
+ if($this->get_spaces() > 0){
67
+ $where = array( 'ticket_booking_id' => $this->ticket_booking_id );
68
+ $result = $wpdb->update($table, $data, $where, $this->get_types($data));
69
+ $this->feedback_message = __('Changes saved','dbem');
70
+ }else{
71
+ $this->result = $this->delete();
72
+ }
73
+ }else{
74
+ if($this->get_spaces() > 0){
75
+ //TODO better error handling
76
+ $result = $wpdb->insert($table, $data, $this->get_types($data));
77
+ $this->ticket_booking_id = $wpdb->insert_id;
78
+ $this->feedback_message = __('Ticket booking created','dbem');
79
+ }else{
80
+ //no point saving a booking with no spaces
81
+ $result = false;
82
+ }
83
+ }
84
+ if( $result === false ){
85
+ $this->feedback_message = __('There was a problem saving the ticket booking.', 'dbem');
86
+ $this->errors[] = __('There was a problem saving the ticket booking.', 'dbem');
87
+ }
88
+ $this->compat_keys();
89
+ return apply_filters('em_ticket_booking_save', ( count($this->errors) == 0 ), $this);
90
+ }else{
91
+ $this->feedback_message = __('There was a problem saving the ticket booking.', 'dbem');
92
+ $this->errors[] = __('There was a problem saving the ticket booking.', 'dbem');
93
+ return apply_filters('em_ticket_booking_save', false, $this);
94
+ }
95
+ return true;
96
+ }
97
+
98
+
99
+ /**
100
+ * Validates the ticket for saving. Should be run during any form submission or saving operation.
101
+ * @return boolean
102
+ */
103
+ function validate(){
104
+ $missing_fields = Array ();
105
+ foreach ( $this->required_fields as $field ) {
106
+ $true_field = $this->fields[$field]['name'];
107
+ if ( $this->$true_field == "") {
108
+ $missing_fields[] = $field;
109
+ }
110
+ }
111
+ if ( count($missing_fields) > 0){
112
+ // TODO Create friendly equivelant names for missing fields notice in validation
113
+ $this->errors[] = __ ( 'Missing fields: ' ) . implode ( ", ", $missing_fields ) . ". ";
114
+ }
115
+ return apply_filters('em_event_validate', count($this->errors) == 0, $this );
116
+ }
117
+
118
+ /**
119
+ * Get the total number of spaces booked for this ticket within this booking.
120
+ * @return int
121
+ */
122
+ function get_spaces(){
123
+ return apply_filters('em_booking_get_spaces',$this->ticket_booking_spaces,$this);
124
+ }
125
+
126
+ /**
127
+ * Gets the total price for this whole booking. Seting $force_reset to true will recheck spaces, even if previously done so.
128
+ * @param boolean $force_refresh
129
+ * @return float
130
+ */
131
+ function get_price( $force_refresh=false, $format = false, $add_tax = 'x' ){
132
+ if( $force_refresh || $this->ticket_booking_price == 0 || $add_tax !== 'x' || get_option('dbem_bookings_tax_auto_add') ){
133
+ //get the ticket, calculate price on spaces
134
+ $this->ticket_booking_price = $this->get_ticket()->get_price(false, $add_tax) * $this->ticket_booking_spaces;
135
+ }
136
+ if($format){
137
+ return apply_filters('em_booking_get_prices', em_get_currency_symbol().number_format($this->ticket_booking_price,2),$this);
138
+ }
139
+ return apply_filters('em_booking_get_prices',$this->ticket_booking_price,$this);
140
+ }
141
+
142
+ /**
143
+ * Smart event locator, saves a database read if possible.
144
+ */
145
+ function get_booking(){
146
+ global $EM_Booking;
147
+ if( is_object($this->booking) && get_class($this->booking)=='EM_Booking' && ($this->booking->booking_id == $this->booking_id || (empty($this->ticket_booking_id) && empty($this->booking_id))) ){
148
+ return $this->booking;
149
+ }elseif( is_object($EM_Booking) && $EM_Booking->booking_id == $this->booking_id ){
150
+ $this->booking = $EM_Booking;
151
+ }else{
152
+ if(is_numeric($this->booking_id)){
153
+ $this->booking = new EM_Booking($this->booking_id);
154
+ }else{
155
+ $this->booking = new EM_Booking();
156
+ }
157
+ }
158
+ return apply_filters('em_ticket_booking_get_booking', $this->booking, $this);;
159
+ }
160
+
161
+ /**
162
+ * Gets the ticket object this booking belongs to, saves a reference in ticket property
163
+ * @return EM_Ticket
164
+ */
165
+ function get_ticket(){
166
+ global $EM_Ticket;
167
+ if( is_object($this->ticket) && get_class($this->ticket)=='EM_Ticket' && $this->ticket->ticket_id == $this->ticket_id ){
168
+ return $this->ticket;
169
+ }elseif( is_object($EM_Ticket) && $EM_Ticket->ticket_id == $this->ticket_id ){
170
+ $this->ticket = $EM_Ticket;
171
+ }else{
172
+ $this->ticket = new EM_Ticket($this->ticket_id);
173
+ }
174
+ return $this->ticket;
175
+ }
176
+
177
+ /**
178
+ * I wonder what this does....
179
+ * @return boolean
180
+ */
181
+ function delete(){
182
+ global $wpdb;
183
+ $sql = $wpdb->prepare("DELETE FROM ". EM_TICKETS_BOOKINGS_TABLE . " WHERE ticket_booking_id=%d", $this->ticket_booking_id);
184
+ $result = $wpdb->query( $sql );
185
+ return apply_filters('em_ticket_booking_delete', ($result !== false ), $this);
186
+ }
187
+
188
+
189
+ /**
190
+ * Get the html options for quantities to go within a <select> container
191
+ * @return string
192
+ */
193
+ function get_spaces_options($zero_value = true){
194
+ $available_spaces = $this->get_available_spaces();
195
+ if( $available_spaces >= $this->min || ( empty($this->min) && $available_spaces > 0) ) {
196
+ ob_start();
197
+ ?>
198
+ <select name="em_tickets[<?php echo $this->ticket_booking_id ?>][spaces]">
199
+ <?php
200
+ $min = ($this->min > 0) ? $this->min:1;
201
+ $max = ($this->max > 0) ? $this->max:get_option('dbem_bookings_form_max');
202
+ ?>
203
+ <?php if($zero_value) : ?><option>0</option><?php endif; ?>
204
+ <?php for( $i=$min; $i<=$max; $i++ ): ?>
205
+ <option><?php echo $i ?></option>
206
+ <?php endfor; ?>
207
+ </select>
208
+ <?php
209
+ return ob_get_clean();
210
+ }else{
211
+ return false;
212
+ }
213
+
214
+ }
215
+
216
+ /**
217
+ * Can the user manage this event?
218
+ */
219
+ function can_manage(){
220
+ return ( $this->get_booking()->can_manage() );
221
+ }
222
+ }
223
+ ?>
classes/em-ticket.php ADDED
@@ -0,0 +1,340 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class EM_Ticket extends EM_Object{
3
+ //DB Fields
4
+ var $ticket_id;
5
+ var $event_id;
6
+ var $ticket_name;
7
+ var $ticket_description;
8
+ var $ticket_price;
9
+ var $ticket_start;
10
+ var $ticket_end;
11
+ var $ticket_min;
12
+ var $ticket_max;
13
+ var $ticket_spaces = 10;
14
+ var $fields = array(
15
+ 'ticket_id' => array('name'=>'id','type'=>'%d'),
16
+ 'event_id' => array('name'=>'event_id','type'=>'%d'),
17
+ 'ticket_name' => array('name'=>'name','type'=>'%s'),
18
+ 'ticket_description' => array('name'=>'description','type'=>'%s','null'=>1),
19
+ 'ticket_price' => array('name'=>'price','type'=>'%f','null'=>1),
20
+ 'ticket_start' => array('name'=>'start','type'=>'%s','null'=>1),
21
+ 'ticket_end' => array('name'=>'end','type'=>'%s','null'=>1),
22
+ 'ticket_min' => array('name'=>'min','type'=>'%s','null'=>1),
23
+ 'ticket_max' => array('name'=>'max','type'=>'%s','null'=>1),
24
+ 'ticket_spaces' => array('name'=>'spaces','type'=>'%s','null'=>1)
25
+ );
26
+ //Other Vars
27
+ /**
28
+ * Contains only bookings belonging to this ticket.
29
+ * @var EM_Booking
30
+ */
31
+ var $bookings = array();
32
+ var $required_fields = array('ticket_name');
33
+ var $start_timestamp;
34
+ var $end_timestamp;
35
+ /**
36
+ * is this ticket limited by spaces allotted to this ticket? false if no limit (i.e. the events general limit of seats)
37
+ * @var unknown_type
38
+ */
39
+ var $spaces_limit = true;
40
+
41
+ /**
42
+ * Creates ticket object and retreives ticket data (default is a blank ticket object). Accepts either array of ticket data (from db) or a ticket id.
43
+ * @param mixed $ticket_data
44
+ * @return null
45
+ */
46
+ function EM_Ticket( $ticket_data = false ){
47
+ $this->ticket_name = __('Standard Ticket','dbem');
48
+ if( $ticket_data !== false ){
49
+ //Load ticket data
50
+ $ticket = array();
51
+ if( is_array($ticket_data) ){
52
+ $ticket = $ticket_data;
53
+ }elseif( is_numeric($ticket_data) ){
54
+ //Retreiving from the database
55
+ global $wpdb;
56
+ $sql = "SELECT * FROM ". EM_TICKETS_TABLE ." WHERE ticket_id ='$ticket_data'";
57
+ $ticket = $wpdb->get_row($sql, ARRAY_A);
58
+ //Ticket notes
59
+ $notes = $wpdb->get_results("SELECT * FROM ". EM_META_TABLE ." WHERE meta_key='ticket-note' AND object_id ='$ticket_data'", ARRAY_A);
60
+ foreach($notes as $note){
61
+ $this->notes[] = unserialize($note['meta_value']);
62
+ }
63
+ }
64
+ //Save into the object
65
+ $this->to_object($ticket);
66
+ $this->start_timestamp = (!empty($ticket['ticket_start'])) ? strtotime($ticket['ticket_start']):false;
67
+ $this->end_timestamp = (!empty($ticket['ticket_end'])) ? strtotime($ticket['ticket_end']):false;
68
+ }
69
+ $this->compat_keys();
70
+ do_action('em_ticket',$this, $ticket_data);
71
+ }
72
+
73
+ /**
74
+ * Saves the ticket into the database, whether a new or existing ticket
75
+ * @return boolean
76
+ */
77
+ function save(){
78
+ global $wpdb;
79
+ $table = EM_TICKETS_TABLE;
80
+ do_action('em_ticket_save_pre',$this);
81
+ //First the person
82
+ if($this->validate() && $this->can_manage() ){
83
+ //Now we save the ticket
84
+ $data = $this->to_array(true); //add the true to remove the nulls
85
+ if($this->ticket_id != ''){
86
+ //since currently wpdb calls don't accept null, let's build the sql ourselves.
87
+ $set_array = array();
88
+ foreach( $this->fields as $field_name => $field ){
89
+ if( empty($this->$field['name']) && $field['null'] ){
90
+ $set_array[] = "{$field_name}=NULL";
91
+ }else{
92
+ $set_array[] = "{$field_name}='".$wpdb->escape($this->$field['name'])."'";
93
+ }
94
+ }
95
+ $sql = "UPDATE $table SET ".implode(', ', $set_array)." WHERE ticket_id={$this->ticket_id}";
96
+ $result = $wpdb->query($sql);
97
+ $this->feedback_message = __('Changes saved','dbem');
98
+ }else{
99
+ //TODO better error handling
100
+ $result = $wpdb->insert($table, $data, $this->get_types($data));
101
+ $this->ticket_id = $wpdb->insert_id;
102
+ $this->feedback_message = __('Ticket created','dbem');
103
+ }
104
+ if( $result === false ){
105
+ $this->feedback_message = __('There was a problem saving the ticket.', 'dbem');
106
+ $this->errors[] = __('There was a problem saving the ticket.', 'dbem');
107
+ }
108
+ $this->compat_keys();
109
+ return apply_filters('em_ticket_save', ( count($this->errors) == 0 ), $this);
110
+ }else{
111
+ $this->feedback_message = __('There was a problem saving the ticket.', 'dbem');
112
+ $this->errors[] = __('There was a problem saving the ticket.', 'dbem');
113
+ return apply_filters('em_ticket_save', false, $this);
114
+ }
115
+ return true;
116
+ }
117
+
118
+ /**
119
+ * Get posted data and save it into the object (not db)
120
+ * @return boolean
121
+ */
122
+ function get_post(){
123
+ //We are getting the values via POST or GET
124
+ global $allowedposttags;
125
+ do_action('em_location_get_post_pre', $this);
126
+ $this->ticket_id = ( !empty($_POST['ticket_id']) ) ? $_POST['ticket_id']:'';
127
+ $this->event_id = ( !empty($_POST['event_id']) ) ? $_POST['event_id']:'';
128
+ $this->ticket_name = ( !empty($_POST['ticket_name']) ) ? wp_kses_data(stripslashes($_POST['ticket_name'])):'';
129
+ $this->ticket_description = ( !empty($_POST['ticket_description']) ) ? wp_kses(stripslashes($_POST['ticket_description'], $allowedposttags)):'';
130
+ $this->ticket_price = ( !empty($_POST['ticket_price']) ) ? $_POST['ticket_price']:'';
131
+ $this->ticket_start = ( !empty($_POST['ticket_start']) ) ? $_POST['ticket_start']:'';
132
+ $this->ticket_end = ( !empty($_POST['ticket_end']) ) ? $_POST['ticket_end']:'';
133
+ $this->start_timestamp = ( !empty($_POST['ticket_start']) ) ? strtotime($_POST['ticket_start']):'';
134
+ $this->end_timestamp = ( !empty($_POST['ticket_end']) ) ? strtotime($_POST['ticket_end']):'';
135
+ $this->ticket_min = ( !empty($_POST['ticket_min']) ) ? $_POST['ticket_min']:'';
136
+ $this->ticket_max = ( !empty($_POST['ticket_max']) ) ? $_POST['ticket_max']:'';
137
+ $this->ticket_spaces = ( !empty($_POST['ticket_spaces']) ) ? $_POST['ticket_spaces']:'';
138
+ $this->compat_keys();
139
+ do_action('em_ticket_get_post', $this);
140
+ }
141
+
142
+
143
+ /**
144
+ * Validates the ticket for saving. Should be run during any form submission or saving operation.
145
+ * @return boolean
146
+ */
147
+ function validate(){
148
+ $missing_fields = Array ();
149
+ $this->errors = array();
150
+ foreach ( $this->required_fields as $field ) {
151
+ $true_field = $this->fields[$field]['name'];
152
+ if ( $this->$true_field == "") {
153
+ $missing_fields[] = $field;
154
+ }
155
+ }
156
+ if( !empty($this->ticket_price) && !is_numeric($this->ticket_price) ){
157
+ $this->add_error(__('Please enter a valid ticket price e.g. 10.50 (no currency signs)','dbem'));
158
+ }
159
+ if ( count($missing_fields) > 0){
160
+ // TODO Create friendly equivelant names for missing fields notice in validation
161
+ $this->errors[] = __ ( 'Missing fields: ' ) . implode ( ", ", $missing_fields ) . ". ";
162
+ }
163
+ return apply_filters('em_event_validate', count($this->errors) == 0, $this );
164
+ }
165
+
166
+ function is_available(){
167
+ $timestamp = current_time('timestamp');
168
+ $available_spaces = $this->get_available_spaces();
169
+ $condition_1 = (empty($this->ticket_start) || $this->start_timestamp <= $timestamp);
170
+ $condition_2 = ($this->end_timestamp >= $timestamp || empty($this->ticket_end));
171
+ $condition_3 = $this->get_event()->end > $timestamp;
172
+ if( $condition_1 && $condition_2 && $condition_3 ){
173
+ //Time Constraints met, now quantities
174
+ if( $available_spaces > 0 && ($available_spaces >= $this->ticket_min || empty($this->ticket_min)) ){
175
+ return apply_filters('em_ticket_is_available',true,$this);
176
+ }
177
+ }
178
+ return apply_filters('em_ticket_is_available',false,$this);
179
+ }
180
+
181
+ /**
182
+ * Gets the total price for this ticket.
183
+ * @return float
184
+ */
185
+ function get_price($format = false, $add_tax = 'x' ){
186
+ $price = $this->ticket_price;
187
+ if( is_numeric(get_option('dbem_bookings_tax')) && get_option('dbem_bookings_tax') > 0 ){
188
+ //tax could be added here
189
+ if( $add_tax === true || ($add_tax !== false && get_option('dbem_bookings_tax_auto_add')) ){
190
+ $price = $price * (1 + get_option('dbem_bookings_tax')/100);
191
+ }
192
+ }
193
+ if($format){
194
+ return apply_filters('em_ticket_get_price', em_get_currency_symbol().number_format($price,2),$this);
195
+ }
196
+ return apply_filters('em_ticket_get_price',number_format($price,2),$this);
197
+ }
198
+
199
+ /**
200
+ * Get the total number of tickets (spaces) available, bearing in mind event-wide maxiumums and ticket priority settings.
201
+ * @return int
202
+ */
203
+ function get_spaces(){
204
+ return apply_filters('em_ticket_get_spaces',$this->ticket_spaces,$this);
205
+ }
206
+
207
+ /**
208
+ * Returns the number of available spaces left in this ticket, bearing in mind event-wide restrictions, previous bookings, approvals and other tickets.
209
+ * @return int
210
+ */
211
+ function get_available_spaces(){
212
+ $event_available_spaces = $this->get_event()->get_bookings()->get_available_spaces();
213
+ $ticket_available_spaces = $this->get_spaces() - $this->get_booked_spaces();
214
+ $return = ($ticket_available_spaces <= $event_available_spaces) ? $ticket_available_spaces:$event_available_spaces;
215
+ return apply_filters('em_ticket_get_available_spaces', $return, $this);
216
+ }
217
+
218
+ /**
219
+ * Returns the number of available spaces left in this ticket, bearing in mind event-wide restrictions, previous bookings, approvals and other tickets.
220
+ * @return int
221
+ */
222
+ function get_booked_spaces($force_reload=false){
223
+ //get all bookings for this event
224
+ $spaces = 0;
225
+ if( is_object($this->bookings) && $force_reload ){
226
+ return $this->bookings;
227
+ }
228
+ foreach( $this->get_bookings()->get_bookings()->bookings as $EM_Booking ){ //get_bookings() is used twice so we get the confirmed (or all if confirmation disabled) bookings of this ticket's total bookings.
229
+ //foreach booking, get this ticket booking info if found
230
+ foreach($EM_Booking->get_tickets_bookings()->tickets_bookings as $EM_Ticket_Booking){
231
+ if( $EM_Ticket_Booking->ticket_id == $this->ticket_id ){
232
+ $spaces += $EM_Ticket_Booking->get_spaces();
233
+ }
234
+ }
235
+ }
236
+ return apply_filters('em_ticket_get_available_spaces', $spaces, $this);
237
+ }
238
+
239
+ /**
240
+ * Smart event locator, saves a database read if possible.
241
+ * @return EM_Event
242
+ */
243
+ function get_event(){
244
+ return em_get_event($this->event_id);
245
+ }
246
+
247
+ /**
248
+ * returns array of EM_Booking objects that have this ticket
249
+ * @return EM_Bookings
250
+ */
251
+ function get_bookings(){
252
+ $bookings = array();
253
+ foreach( $this->get_event()->get_bookings()->bookings as $EM_Booking ){
254
+ foreach($EM_Booking->get_tickets_bookings()->tickets_bookings as $EM_Ticket_Booking){
255
+ if( $EM_Ticket_Booking->ticket_id == $this->ticket_id ){
256
+ $bookings[$EM_Booking->booking_id] = $EM_Booking;
257
+ }
258
+ }
259
+ }
260
+ $this->bookings = new EM_Bookings($bookings);
261
+ return $this->bookings;
262
+ }
263
+
264
+ /**
265
+ * I wonder what this does....
266
+ * @return boolean
267
+ */
268
+ function delete(){
269
+ global $wpdb;
270
+ $result = false;
271
+ if( $this->can_manage() ){
272
+ if( count($this->get_bookings()->bookings) == 0 ){
273
+ $sql = $wpdb->prepare("DELETE FROM ". EM_TICKETS_TABLE . " WHERE ticket_id=%d", $this->ticket_id);
274
+ $result = $wpdb->query( $sql );
275
+ }else{
276
+ $this->feedback_message = __('You cannot delete a ticket that has a booking on it.','dbem');
277
+ $this->add_error($this->feedback_message);
278
+ return false;
279
+ }
280
+ }
281
+ return ( $result !== false );
282
+ }
283
+
284
+ /**
285
+ * Get the html options for quantities to go within a <select> container
286
+ * @return string
287
+ */
288
+ function get_spaces_options($zero_value = true, $default_value = 0){
289
+ $available_spaces = $this->get_available_spaces();
290
+ if( $this->is_available() ) {
291
+ ob_start();
292
+ ?>
293
+ <select name="em_tickets[<?php echo $this->ticket_id ?>][spaces]" class="em-ticket-select">
294
+ <?php
295
+ $min = ($this->ticket_min > 0) ? $this->ticket_min:1;
296
+ $max = ($this->ticket_max > 0) ? $this->ticket_max:get_option('dbem_bookings_form_max');
297
+ ?>
298
+ <?php if($zero_value) : ?><option>0</option><?php endif; ?>
299
+ <?php for( $i=$min; $i<=$available_spaces && $i<=$max; $i++ ): ?>
300
+ <option <?php if($i == $default_value){ echo 'selected="selected"'; $shown_default = true; } ?>><?php echo $i ?></option>
301
+ <?php endfor; ?>
302
+ <?php if(empty($shown_default) && $default_value > 0 ): ?><option selected="selected"><?php echo $default_value; ?></option><?php endif; ?>
303
+ </select>
304
+ <?php
305
+ return apply_filters('em_ticket_get_spaces_options', ob_get_clean(), $zero_value, $default_value, $this);
306
+ }else{
307
+ return false;
308
+ }
309
+
310
+ }
311
+
312
+ /**
313
+ * Can the user manage this event?
314
+ */
315
+ function can_manage(){
316
+ return $this->get_event()->can_manage('manage_bookings','manage_others_bookings');
317
+ }
318
+
319
+ /**
320
+ * Outputs properties with formatting
321
+ * @param string $property
322
+ * @return string
323
+ */
324
+ function output_property($property){
325
+ switch($property){
326
+ case 'start':
327
+ $value = date_i18n( get_option('date_format'), $this->start_timestamp );
328
+ break;
329
+ case 'end':
330
+ $value = date_i18n( get_option('date_format'), $this->end_timestamp );
331
+ break;
332
+ break;
333
+ default:
334
+ $value = $this->$property;
335
+ break;
336
+ }
337
+ return apply_filters('em_ticket_output_property',$value,$this);
338
+ }
339
+ }
340
+ ?>
classes/em-tickets-bookings.php ADDED
@@ -0,0 +1,283 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Deals with the each ticket booked in a single booking
4
+ * @author marcus
5
+ *
6
+ */
7
+ class EM_Tickets_Bookings extends EM_Object implements Iterator{
8
+
9
+ /**
10
+ * Array of EM_Ticket_Booking objects for a specific event
11
+ * @var array
12
+ */
13
+ var $tickets_bookings = array();
14
+ /**
15
+ * This object belongs to this booking object
16
+ * @var EM_Booking
17
+ */
18
+ var $booking;
19
+ /**
20
+ * This object belongs to this booking object
21
+ * @var EM_Ticket
22
+ */
23
+ var $ticket;
24
+ var $spaces;
25
+ var $price;
26
+
27
+ /**
28
+ * Creates an EM_Tickets instance,
29
+ * @param EM_Event $event
30
+ * @return null
31
+ */
32
+ function EM_Tickets_Bookings( $object = false ){
33
+ global $wpdb;
34
+ if($object){
35
+ if( is_object($object) && get_class($object) == "EM_Booking"){
36
+ $this->booking = $object;
37
+ $sql = "SELECT * FROM ". EM_TICKETS_BOOKINGS_TABLE ." bt LEFT JOIN ". EM_BOOKINGS_TABLE ." b ON bt.booking_id=b.booking_id WHERE b.booking_id ='{$this->booking->booking_id}'";
38
+ }elseif( is_object($object) && get_class($object) == "EM_Ticket"){
39
+ $this->ticket = $object;
40
+ $sql = "SELECT * FROM ". EM_TICKETS_BOOKINGS_TABLE ." bt LEFT JOIN ". EM_TICKETS_TABLE ." t ON bt.ticket_id=t.ticket_id WHERE t.ticket_id ='{$this->ticket->ticket_id}'";
41
+ }elseif( is_numeric($object) ){
42
+ $sql = "SELECT * FROM ". EM_TICKETS_BOOKINGS_TABLE ." bt LEFT JOIN ". EM_BOOKINGS_TABLE ." t ON bt.booking_id=t.booking_id WHERE t.booking_id ='{$object}'";
43
+ }
44
+ $tickets_bookings = $wpdb->get_results($sql, ARRAY_A);
45
+ //Get tickets belonging to this tickets booking.
46
+ foreach ($tickets_bookings as $ticket_booking){
47
+ $EM_Ticket_Booking = new EM_Ticket_Booking($ticket_booking);
48
+ $EM_Ticket_Booking->booking = $this->booking; //save some calls
49
+ $this->tickets_bookings[] = $EM_Ticket_Booking;
50
+ }
51
+ }
52
+ do_action('em_tickets_bookings',$this, $object);
53
+ }
54
+
55
+ /**
56
+ * Saves the ticket bookings for this booking into the database, whether a new or existing booking
57
+ * @return boolean
58
+ */
59
+ function save(){
60
+ global $wpdb;
61
+ do_action('em_tickets_bookings_save_pre',$this);
62
+ foreach( $this->tickets_bookings as $EM_Ticket_Booking ){
63
+ $result = $EM_Ticket_Booking->save();
64
+ if(!$result){
65
+ $this->errors = array_merge($this->errors, $EM_Ticket_Booking->get_errors());
66
+ }
67
+ }
68
+ if( count($this->errors) > 0 ){
69
+ $this->feedback_message = __('There was a problem saving the booking.', 'dbem');
70
+ $this->errors[] = __('There was a problem saving the booking.', 'dbem');
71
+ return apply_filters('em_tickets_bookings_save', false, $this);
72
+ }
73
+ return apply_filters('em_tickets_bookings_save', true, $this);
74
+ }
75
+
76
+ /**
77
+ * Add a booking into this event object, checking that there's enough space for the event
78
+ * @param EM_Ticket_Booking $EM_Ticket_Booking
79
+ * @return boolean
80
+ */
81
+ function add( $EM_Ticket_Booking, $override = false ){ //note, $override was a quick fix, not necessarily permanent, so don't depend on it just yet
82
+ global $wpdb,$EM_Mailer;
83
+ //Does the ticket we want to book have enough spaeces?
84
+ if ( $override || $EM_Ticket_Booking->get_ticket()->get_available_spaces() >= $EM_Ticket_Booking->get_spaces() ) {
85
+ $ticket_booking_key = $this->has_ticket($EM_Ticket_Booking->ticket_id);
86
+ if( $ticket_booking_key !== false && is_object($this->tickets_bookings[$ticket_booking_key]) ){
87
+ //previously booked ticket, so let's just replace it
88
+ $this->tickets_bookings[$ticket_booking_key]->spaces = $EM_Ticket_Booking->get_spaces();
89
+ $this->tickets_bookings[$ticket_booking_key]->get_price(true);
90
+ return apply_filters('em_tickets_bookings_add',true,$this);
91
+ }elseif( $EM_Ticket_Booking->get_spaces() > 0 ){
92
+ //new ticket in booking
93
+ $this->tickets_bookings[] = $EM_Ticket_Booking;
94
+ $this->get_spaces(true);
95
+ $this->get_price(true);
96
+ return apply_filters('em_tickets_bookings_add',true,$this);
97
+ }
98
+ } else {
99
+ $this->errors[] = __('Booking cannot be made, not enough spaces available!', 'dbem');
100
+ return apply_filters('em_tickets_bookings_add',false,$this);
101
+ }
102
+ return apply_filters('em_tickets_bookings_add',false,$this);
103
+ }
104
+
105
+ /**
106
+ * Checks if this set has a specific ticket booked, returning the key of the ticket in the EM_Tickets_Bookings->ticket_bookings array
107
+ * @param int $ticket_id
108
+ * @return mixed
109
+ */
110
+ function has_ticket( $ticket_id ){
111
+ foreach ($this->tickets_bookings as $key => $EM_Ticket_Booking){
112
+ if( $EM_Ticket_Booking->ticket_id == $ticket_id ){
113
+ return apply_filters('em_tickets_has_ticket',$key,$this);
114
+ }
115
+ }
116
+ return apply_filters('em_tickets_has_ticket',false,$this);
117
+ }
118
+
119
+ /**
120
+ * Smart event locator, saves a database read if possible.
121
+ */
122
+ function get_booking(){
123
+ global $EM_Booking;
124
+ $booking_id = $this->get_booking_id();
125
+ if( is_object($this->booking) && get_class($this->booking)=='EM_Booking' && $this->booking->booking_id == $booking_id ){
126
+ return $this->booking;
127
+ }elseif( is_object($EM_Booking) && $EM_Booking->booking_id == $booking_id ){
128
+ $this->booking = $EM_Booking;
129
+ }else{
130
+ if(is_numeric($booking_id)){
131
+ $this->booking = new EM_Booking($booking_id);
132
+ }else{
133
+ $this->booking = new EM_Booking();
134
+ }
135
+ }
136
+ return apply_filters('em_tickets_bookings_get_booking', $this->booking, $this);;
137
+ }
138
+
139
+ function get_booking_id(){
140
+ if( count($this->tickets_bookings) > 0 ){
141
+ foreach($this->tickets_bookings as $EM_Ticket_Booking){ break; } //get first array item
142
+ return apply_filters('em_tickets_bookings_get_booking_id', $EM_Ticket_Booking->booking_id, $this);
143
+ }
144
+ return apply_filters('em_tickets_bookings_get_booking_id', false, $this);
145
+ }
146
+
147
+ /**
148
+ * Delete all ticket bookings
149
+ * @return boolean
150
+ */
151
+ function delete(){
152
+ global $wpdb;
153
+ if( $this->get_booking()->can_manage() ){
154
+ $result = $wpdb->query("DELETE FROM ".EM_TICKETS_BOOKINGS_TABLE." WHERE booking_id='{$this->get_booking()->booking_id}'");
155
+ //echo "<pre>";print_r($this->get_booking());echo "</pre>";
156
+ }else{
157
+ //FIXME ticket bookings
158
+ $ticket_ids = array();
159
+ foreach( $this->tickets_bookings as $EM_Ticket_Booking ){
160
+ if( $EM_Ticket_Booking->can_manage() ){
161
+ $tickets_bookings_ids[] = $EM_Ticket_Booking->booking_id;
162
+ }else{
163
+ $this->errors[] = sprintf(__('You do not have the rights to manage this %s.','dbem'),__('Booking','dbem'));
164
+ }
165
+ }
166
+ if(count($ticket_ids) > 0){
167
+ $result = $wpdb->query("DELETE FROM ".EM_TICKETS_BOOKINGS_TABLE." WHERE ticket_booking_id IN (".implode(',',$ticket_ids).")");
168
+ }
169
+ }
170
+ return apply_filters('em_tickets_bookings_get_booking_id', ($result == true), $this);
171
+ }
172
+
173
+ /**
174
+ * Go through the tickets in this object and validate them
175
+ */
176
+ function validate(){
177
+ $errors = array();
178
+ foreach($this->tickets_bookings as $EM_Ticket_Booking){
179
+ $errors[] = $EM_Ticket_Booking->validate();
180
+ }
181
+ return apply_filters('em_tickets_bookings_validate', !in_array(false, $errors), $this);
182
+ }
183
+
184
+ /**
185
+ * Get the total number of spaces booked in this booking. Seting $force_reset to true will recheck spaces, even if previously done so.
186
+ * @param unknown_type $force_refresh
187
+ * @return mixed
188
+ */
189
+ function get_spaces( $force_refresh=false ){
190
+ if($force_refresh || $this->spaces == 0){
191
+ $spaces = 0;
192
+ foreach($this->tickets_bookings as $EM_Ticket_Booking){
193
+ $spaces += $EM_Ticket_Booking->get_spaces();
194
+ }
195
+ $this->spaces = $spaces;
196
+ }
197
+ return apply_filters('em_booking_get_spaces',$this->spaces,$this);
198
+ }
199
+
200
+ /**
201
+ * Gets the total price for this whole booking by adding up subtotals of booked tickets. Seting $force_reset to true will recheck spaces, even if previously done so.
202
+ * @param boolean $force_refresh
203
+ * @return float
204
+ */
205
+ function get_price( $force_refresh=false, $format = false, $add_tax = 'x' ){
206
+ $price = 0;
207
+ if($force_refresh || $this->price == 0 || $add_tax !== 'x' || get_option('dbem_bookings_tax_auto_add')){
208
+ foreach($this->tickets_bookings as $EM_Ticket_Booking){
209
+ $price += $EM_Ticket_Booking->get_price($force_refresh, false, $add_tax);
210
+ }
211
+ $this->price = $price;
212
+ }
213
+ if($format){
214
+ return apply_filters('em_tickets_bookings_get_prices', em_get_currency_symbol().number_format($this->price,2),$this);
215
+ }
216
+ return apply_filters('em_tickets_bookings_get_prices',$this->price,$this);
217
+ }
218
+
219
+ /**
220
+ * Goes through each ticket and populates it with the bookings made
221
+ */
222
+ function get_ticket_bookings(){
223
+ foreach( $this->tickets as $EM_Ticket ){
224
+ $EM_Ticket->get_bookings();
225
+ }
226
+ }
227
+
228
+ /* Overrides EM_Object method to apply a filter to result
229
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
230
+ */
231
+ function build_sql_conditions( $args = array() ){
232
+ $conditions = apply_filters( 'em_tickets_build_sql_conditions', parent::build_sql_conditions($args), $args );
233
+ if( is_numeric($args['status']) ){
234
+ $conditions['status'] = 'ticket_status='.$args['status'];
235
+ }
236
+ return apply_filters('em_tickets_bookings_build_sql_conditions', $conditions, $args);
237
+ }
238
+
239
+ /* Overrides EM_Object method to apply a filter to result
240
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
241
+ */
242
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
243
+ return apply_filters( 'em_tickets_bookings_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
244
+ }
245
+
246
+ /*
247
+ * Adds custom Events search defaults
248
+ * @param array $array
249
+ * @return array
250
+ * @uses EM_Object#get_default_search()
251
+ */
252
+ function get_default_search( $array = array() ){
253
+ $defaults = array(
254
+ 'status' => false,
255
+ 'person' => true //to add later, search by person's tickets...
256
+ );
257
+ $defaults['owner'] = !current_user_can('manage_others_bookings') ? get_current_user_id():false;
258
+ return apply_filters('em_tickets_bookings_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
259
+ }
260
+
261
+ //Iterator Implementation
262
+ public function rewind(){
263
+ reset($this->tickets_bookings);
264
+ }
265
+ public function current(){
266
+ $var = current($this->tickets_bookings);
267
+ return $var;
268
+ }
269
+ public function key(){
270
+ $var = key($this->tickets_bookings);
271
+ return $var;
272
+ }
273
+ public function next(){
274
+ $var = next($this->tickets_bookings);
275
+ return $var;
276
+ }
277
+ public function valid(){
278
+ $key = key($this->tickets_bookings);
279
+ $var = ($key !== NULL && $key !== FALSE);
280
+ return $var;
281
+ }
282
+ }
283
+ ?>
classes/em-tickets.php ADDED
@@ -0,0 +1,241 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Deals with the ticket info for an event
4
+ * @author marcus
5
+ *
6
+ */
7
+ class EM_Tickets extends EM_Object implements Iterator{
8
+
9
+ /**
10
+ * Array of EM_Ticket objects for a specific event
11
+ * @var array
12
+ */
13
+ var $tickets = array();
14
+ /**
15
+ * @var int
16
+ */
17
+ var $event_id;
18
+ /**
19
+ * @var EM_Booking
20
+ */
21
+ var $booking;
22
+ var $spaces;
23
+
24
+
25
+ /**
26
+ * Creates an EM_Tickets instance,
27
+ * @param EM_Event $event
28
+ * @return null
29
+ */
30
+ function EM_Tickets( $object = false ){
31
+ global $wpdb;
32
+ if( is_numeric($object) || (is_object($object) && in_array(get_class($object), array("EM_Event","EM_Booking"))) ){
33
+ $this->event_id = (is_object($object)) ? $object->event_id:$object;
34
+ $sql = "SELECT * FROM ". EM_TICKETS_TABLE ." WHERE event_id ='{$this->event_id}' ORDER BY ".get_option('dbem_bookings_tickets_orderby');
35
+ $tickets = $wpdb->get_results($sql, ARRAY_A);
36
+ foreach ($tickets as $ticket){
37
+ $EM_Ticket = new EM_Ticket($ticket);
38
+ $EM_Ticket->event_id = $this->event_id;
39
+ $this->tickets[] = $EM_Ticket;
40
+ }
41
+ }elseif( is_array($object) ){ //expecting an array of EM_Ticket objects or ticket db array
42
+ if( is_object(current($object)) && get_class(current($object)) == 'EM_Ticket' ){
43
+ $this->tickets = $object;
44
+ }else{
45
+ foreach($object as $ticket){
46
+ $EM_Ticket = new EM_Ticket($ticket);
47
+ $EM_Ticket->event_id = $this->event_id;
48
+ $this->tickets[] = $EM_Ticket;
49
+ }
50
+ }
51
+ }
52
+ do_action('em_tickets', $this, $object);
53
+ }
54
+
55
+ /**
56
+ * @return EM_Event
57
+ */
58
+ function get_event(){
59
+ global $EM_Event;
60
+ if( is_object($EM_Event) && $EM_Event->event_id == $this->event_id ){
61
+ return $EM_Event;
62
+ }else{
63
+ return new EM_Event($this->event_id);
64
+ }
65
+ }
66
+
67
+ /**
68
+ * Get the first EM_Ticket object in this instance. Returns false if no tickets available.
69
+ * @return EM_Ticket
70
+ */
71
+ function get_first(){
72
+ if( count($this->tickets) > 0 ){
73
+ foreach($this->tickets as $EM_Ticket){
74
+ return $EM_Ticket;
75
+ }
76
+ }else{
77
+ return false;
78
+ }
79
+ }
80
+
81
+ /**
82
+ * Delete tickets in thie object
83
+ * @return boolean
84
+ */
85
+ function delete(){
86
+ global $wpdb;
87
+ //get all the ticket ids
88
+ $result = false;
89
+ $ticket_ids = array();
90
+ foreach( $this->tickets as $EM_Ticket ){
91
+ $ticket_ids[] = $EM_Ticket->ticket_id;
92
+ }
93
+ //check that tickets don't have bookings
94
+ if(count($ticket_ids) > 0){
95
+ $bookings = $wpdb->get_var("SELECT COUNT(*) FROM ". EM_TICKETS_BOOKINGS_TABLE." WHERE ticket_id IN (".implode(',',$ticket_ids).")");
96
+ if( $bookings > 0 ){
97
+ $result = false;
98
+ $this->add_error(__('You cannot delete tickets if there are any bookings associated with them. Please delete these bookings first.','dbem'));
99
+ }else{
100
+ $result = $wpdb->query("DELETE FROM ".EM_TICKETS_TABLE." WHERE event_id IN (".implode(',',$ticket_ids).")");
101
+ }
102
+ }
103
+ return ($result !== false);
104
+ }
105
+
106
+ /**
107
+ * Retrieve multiple ticket info via POST
108
+ * @return boolean
109
+ */
110
+ function get_post(){
111
+ //Build Event Array
112
+ do_action('em_tickets_get_post_pre', $this);
113
+ $this->tickets = array(); //clean current tickets out
114
+ if( !empty($_POST['em_tickets']) && is_array($_POST['em_tickets']) ){
115
+ //get all ticket data and create objects
116
+ foreach($_POST['em_tickets'] as $ticket_data){
117
+ $EM_Ticket = new EM_Ticket($ticket_data);
118
+ $this->tickets[] = $EM_Ticket;
119
+ }
120
+ }else{
121
+ //we create a blank standard ticket
122
+ $EM_Ticket = new EM_Ticket(array(
123
+ 'event_id' => $this->event_id,
124
+ 'ticket_name' => __('Standard','dbem')
125
+ ));
126
+ $this->tickets[] = $EM_Ticket;
127
+ }
128
+ return apply_filters('em_tickets_get_post', $this->validate(), $this);
129
+ }
130
+
131
+ /**
132
+ * Go through the tickets in this object and validate them
133
+ */
134
+ function validate(){
135
+ $this->errors = array();
136
+ foreach($this->tickets as $EM_Ticket){
137
+ if( !$EM_Ticket->validate() ){
138
+ $this->add_error($EM_Ticket->get_errors());
139
+ }
140
+ }
141
+ return apply_filters('em_tickets_validate', count($this->errors) == 0, $this);
142
+ }
143
+
144
+ /**
145
+ * Save tickets into DB
146
+ */
147
+ function save(){
148
+ $errors = array();
149
+ foreach( $this->tickets as $EM_Ticket ){
150
+ /* @var $EM_Ticket EM_Ticket */
151
+ $EM_Ticket->event_id = $this->event_id; //pass on saved event_data
152
+ $errors[] = $EM_Ticket->save();
153
+ }
154
+ return apply_filters('em_tickets_save', !in_array(false, $errors), $this);
155
+ }
156
+
157
+ /**
158
+ * Goes through each ticket and populates it with the bookings made
159
+ */
160
+ function get_ticket_bookings(){
161
+ foreach( $this->tickets as $EM_Ticket ){
162
+ $EM_Ticket->get_bookings();
163
+ }
164
+ }
165
+
166
+ /**
167
+ * Get the total number of spaces this event has. This will show the lower value of event global spaces limit or total ticket spaces. Setting $force_refresh to true will recheck spaces, even if previously done so.
168
+ * @param boolean $force_refresh
169
+ * @return int
170
+ */
171
+ function get_spaces( $force_refresh=false ){
172
+ $spaces = 0;
173
+ if($force_refresh || $this->spaces == 0){
174
+ foreach( $this->tickets as $EM_Ticket ){
175
+ /* @var $EM_Ticket EM_Ticket */
176
+ $spaces += $EM_Ticket->get_spaces();
177
+ }
178
+ $this->spaces = $spaces;
179
+ }
180
+ return apply_filters('em_booking_get_spaces',$this->spaces,$this);
181
+ }
182
+
183
+ /* Overrides EM_Object method to apply a filter to result
184
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_conditions()
185
+ */
186
+ function build_sql_conditions( $args = array() ){
187
+ $conditions = apply_filters( 'em_tickets_build_sql_conditions', parent::build_sql_conditions($args), $args );
188
+ if( is_numeric($args['ticket_status']) ){
189
+ $conditions['ticket_status'] = 'ticket_status='.$args['status'];
190
+ }
191
+ return apply_filters('em_tickets_build_sql_conditions', $conditions, $args);
192
+ }
193
+
194
+ /* Overrides EM_Object method to apply a filter to result
195
+ * @see wp-content/plugins/events-manager/classes/EM_Object#build_sql_orderby()
196
+ */
197
+ function build_sql_orderby( $args, $accepted_fields, $default_order = 'ASC' ){
198
+ return apply_filters( 'em_tickets_build_sql_orderby', parent::build_sql_orderby($args, $accepted_fields, get_option('dbem_events_default_order')), $args, $accepted_fields, $default_order );
199
+ }
200
+
201
+ /*
202
+ * Adds custom Events search defaults
203
+ * @param array $array
204
+ * @return array
205
+ * @uses EM_Object#get_default_search()
206
+ */
207
+ function get_default_search( $array = array() ){
208
+ $defaults = array(
209
+ 'ticket_status' => false,
210
+ 'person' => true //to add later, search by person's tickets...
211
+ );
212
+ if( is_admin() ){
213
+ //figure out default owning permissions
214
+ $defaults['owner'] = !current_user_can('manage_others_bookings') ? get_current_user_id():false;
215
+ }
216
+ return apply_filters('em_tickets_get_default_search', parent::get_default_search($defaults,$array), $array, $defaults);
217
+ }
218
+
219
+ //Iterator Implementation
220
+ public function rewind(){
221
+ reset($this->tickets);
222
+ }
223
+ public function current(){
224
+ $var = current($this->tickets);
225
+ return $var;
226
+ }
227
+ public function key(){
228
+ $var = key($this->tickets);
229
+ return $var;
230
+ }
231
+ public function next(){
232
+ $var = next($this->tickets);
233
+ return $var;
234
+ }
235
+ public function valid(){
236
+ $key = key($this->tickets);
237
+ $var = ($key !== NULL && $key !== FALSE);
238
+ return $var;
239
+ }
240
+ }
241
+ ?>
classes/phpmailer/class.phpmailer.php CHANGED
@@ -1,1506 +1,2473 @@
1
  <?php
2
- ////////////////////////////////////////////////////
3
- // PHPMailer - PHP email class
4
- //
5
- // Class for sending email using either
6
- // sendmail, PHP mail(), or SMTP. Methods are
7
- // based upon the standard AspEmail(tm) classes.
8
- //
9
- // Copyright (C) 2001 - 2003 Brent R. Matzelle
10
- //
11
- // License: LGPL, see LICENSE
12
- ////////////////////////////////////////////////////
 
 
 
 
 
 
 
 
 
 
 
13
 
14
  /**
15
  * PHPMailer - PHP email transport class
 
16
  * @package PHPMailer
17
- * @author Brent R. Matzelle
18
- * @copyright 2001 - 2003 Brent R. Matzelle
 
 
 
 
 
19
  */
20
- class EM_PHPMailer
21
- {
22
- /////////////////////////////////////////////////
23
- // PUBLIC VARIABLES
24
- /////////////////////////////////////////////////
25
-
26
- /**
27
- * Email priority (1 = High, 3 = Normal, 5 = low).
28
- * @var int
29
- */
30
- var $Priority = 3;
31
-
32
- /**
33
- * Sets the CharSet of the message.
34
- * @var string
35
- */
36
- var $CharSet = "iso-8859-1";
37
-
38
- /**
39
- * Sets the Content-type of the message.
40
- * @var string
41
- */
42
- var $ContentType = "text/plain";
43
-
44
- /**
45
- * Sets the Encoding of the message. Options for this are "8bit",
46
- * "7bit", "binary", "base64", and "quoted-printable".
47
- * @var string
48
- */
49
- var $Encoding = "8bit";
50
-
51
- /**
52
- * Holds the most recent mailer error message.
53
- * @var string
54
- */
55
- var $ErrorInfo = "";
56
-
57
- /**
58
- * Sets the From email address for the message.
59
- * @var string
60
- */
61
- var $From = "root@localhost";
62
-
63
- /**
64
- * Sets the From name of the message.
65
- * @var string
66
- */
67
- var $FromName = "Root User";
68
-
69
- /**
70
- * Sets the Sender email (Return-Path) of the message. If not empty,
71
- * will be sent via -f to sendmail or as 'MAIL FROM' in smtp mode.
72
- * @var string
73
- */
74
- var $Sender = "";
75
-
76
- /**
77
- * Sets the Subject of the message.
78
- * @var string
79
- */
80
- var $Subject = "";
81
-
82
- /**
83
- * Sets the Body of the message. This can be either an HTML or text body.
84
- * If HTML then run IsHTML(true).
85
- * @var string
86
- */
87
- var $Body = "";
88
-
89
- /**
90
- * Sets the text-only body of the message. This automatically sets the
91
- * email to multipart/alternative. This body can be read by mail
92
- * clients that do not have HTML email capability such as mutt. Clients
93
- * that can read HTML will view the normal Body.
94
- * @var string
95
- */
96
- var $AltBody = "";
97
-
98
- /**
99
- * Sets word wrapping on the body of the message to a given number of
100
- * characters.
101
- * @var int
102
- */
103
- var $WordWrap = 0;
104
-
105
- /**
106
- * Method to send mail: ("mail", "sendmail", or "smtp").
107
- * @var string
108
- */
109
- var $Mailer = "mail";
110
-
111
- /**
112
- * Sets the path of the sendmail program.
113
- * @var string
114
- */
115
- var $Sendmail = "/usr/sbin/sendmail";
116
-
117
- /**
118
- * Path to PHPMailer plugins. This is now only useful if the SMTP class
119
- * is in a different directory than the PHP include path.
120
- * @var string
121
- */
122
- var $PluginDir = "";
123
-
124
- /**
125
- * Holds PHPMailer version.
126
- * @var string
127
- */
128
- var $Version = "1.73";
129
-
130
- /**
131
- * Sets the email address that a reading confirmation will be sent.
132
- * @var string
133
- */
134
- var $ConfirmReadingTo = "";
135
-
136
- /**
137
- * Sets the hostname to use in Message-Id and Received headers
138
- * and as default HELO string. If empty, the value returned
139
- * by SERVER_NAME is used or 'localhost.localdomain'.
140
- * @var string
141
- */
142
- var $Hostname = "";
143
-
144
- /////////////////////////////////////////////////
145
- // SMTP VARIABLES
146
- /////////////////////////////////////////////////
147
-
148
- /**
149
- * Sets the SMTP hosts. All hosts must be separated by a
150
- * semicolon. You can also specify a different port
151
- * for each host by using this format: [hostname:port]
152
- * (e.g. "smtp1.example.com:25;smtp2.example.com").
153
- * Hosts will be tried in order.
154
- * @var string
155
- */
156
- var $Host = "localhost";
157
-
158
- /**
159
- * Sets the default SMTP server port.
160
- * @var int
161
- */
162
- var $Port = 25;
163
-
164
- /**
165
- * Sets the SMTP HELO of the message (Default is $Hostname).
166
- * @var string
167
- */
168
- var $Helo = "";
169
-
170
- /**
171
- * Sets SMTP authentication. Utilizes the Username and Password variables.
172
- * @var bool
173
- */
174
- var $SMTPAuth = false;
175
-
176
- /**
177
- * Sets SMTP username.
178
- * @var string
179
- */
180
- var $Username = "";
181
-
182
- /**
183
- * Sets SMTP password.
184
- * @var string
185
- */
186
- var $Password = "";
187
-
188
- /**
189
- * Sets the SMTP server timeout in seconds. This function will not
190
- * work with the win32 version.
191
- * @var int
192
- */
193
- var $Timeout = 10;
194
-
195
- /**
196
- * Sets SMTP class debugging on or off.
197
- * @var bool
198
- */
199
- var $SMTPDebug = false;
200
-
201
- /**
202
- * Prevents the SMTP connection from being closed after each mail
203
- * sending. If this is set to true then to close the connection
204
- * requires an explicit call to SmtpClose().
205
- * @var bool
206
- */
207
- var $SMTPKeepAlive = false;
208
-
209
- /**#@+
210
- * @access private
211
- */
212
- var $smtp = NULL;
213
- var $to = array();
214
- var $cc = array();
215
- var $bcc = array();
216
- var $ReplyTo = array();
217
- var $attachment = array();
218
- var $CustomHeader = array();
219
- var $message_type = "";
220
- var $boundary = array();
221
- var $language = array();
222
- var $error_count = 0;
223
- var $LE = "\n";
224
- /**#@-*/
225
-
226
- /////////////////////////////////////////////////
227
- // VARIABLE METHODS
228
- /////////////////////////////////////////////////
229
-
230
- /**
231
- * Sets message type to HTML.
232
- * @param bool $bool
233
- * @return void
234
- */
235
- function IsHTML($bool) {
236
- if($bool == true)
237
- $this->ContentType = "text/html";
238
- else
239
- $this->ContentType = "text/plain";
240
- }
241
-
242
- /**
243
- * Sets Mailer to send message using SMTP.
244
- * @return void
245
- */
246
- function IsSMTP() {
247
- $this->Mailer = "smtp";
248
- }
249
-
250
- /**
251
- * Sets Mailer to send message using PHP mail() function.
252
- * @return void
253
- */
254
- function IsMail() {
255
- $this->Mailer = "mail";
256
- }
257
-
258
- /**
259
- * Sets Mailer to send message using the $Sendmail program.
260
- * @return void
261
- */
262
- function IsSendmail() {
263
- $this->Mailer = "sendmail";
264
- }
265
-
266
- /**
267
- * Sets Mailer to send message using the qmail MTA.
268
- * @return void
269
- */
270
- function IsQmail() {
271
- $this->Sendmail = "/var/qmail/bin/sendmail";
272
- $this->Mailer = "sendmail";
273
- }
274
-
275
-
276
- /////////////////////////////////////////////////
277
- // RECIPIENT METHODS
278
- /////////////////////////////////////////////////
279
-
280
- /**
281
- * Adds a "To" address.
282
- * @param string $address
283
- * @param string $name
284
- * @return void
285
- */
286
- function AddAddress($address, $name = "") {
287
- $cur = count($this->to);
288
- $this->to[$cur][0] = trim($address);
289
- $this->to[$cur][1] = $name;
290
- }
291
-
292
- /**
293
- * Adds a "Cc" address. Note: this function works
294
- * with the SMTP mailer on win32, not with the "mail"
295
- * mailer.
296
- * @param string $address
297
- * @param string $name
298
- * @return void
299
- */
300
- function AddCC($address, $name = "") {
301
- $cur = count($this->cc);
302
- $this->cc[$cur][0] = trim($address);
303
- $this->cc[$cur][1] = $name;
304
- }
305
-
306
- /**
307
- * Adds a "Bcc" address. Note: this function works
308
- * with the SMTP mailer on win32, not with the "mail"
309
- * mailer.
310
- * @param string $address
311
- * @param string $name
312
- * @return void
313
- */
314
- function AddBCC($address, $name = "") {
315
- $cur = count($this->bcc);
316
- $this->bcc[$cur][0] = trim($address);
317
- $this->bcc[$cur][1] = $name;
318
- }
319
-
320
- /**
321
- * Adds a "Reply-to" address.
322
- * @param string $address
323
- * @param string $name
324
- * @return void
325
- */
326
- function AddReplyTo($address, $name = "") {
327
- $cur = count($this->ReplyTo);
328
- $this->ReplyTo[$cur][0] = trim($address);
329
- $this->ReplyTo[$cur][1] = $name;
330
- }
331
-
332
-
333
- /////////////////////////////////////////////////
334
- // MAIL SENDING METHODS
335
- /////////////////////////////////////////////////
336
-
337
- /**
338
- * Creates message and assigns Mailer. If the message is
339
- * not sent successfully then it returns false. Use the ErrorInfo
340
- * variable to view description of the error.
341
- * @return bool
342
- */
343
- function Send() {
344
- $header = "";
345
- $body = "";
346
- $result = true;
347
-
348
- if((count($this->to) + count($this->cc) + count($this->bcc)) < 1)
349
- {
350
- $this->SetError($this->Lang("provide_address"));
351
- return false;
352
- }
353
 
354
- // Set whether the message is multipart/alternative
355
- if(!empty($this->AltBody))
356
- $this->ContentType = "multipart/alternative";
357
-
358
- $this->error_count = 0; // reset errors
359
- $this->SetMessageType();
360
- $header .= $this->CreateHeader();
361
- $body = $this->CreateBody();
362
-
363
- if($body == "") { return false; }
364
-
365
- // Choose the mailer
366
- switch($this->Mailer)
367
- {
368
- case "sendmail":
369
- $result = $this->SendmailSend($header, $body);
370
- break;
371
- case "mail":
372
- $result = $this->MailSend($header, $body);
373
- break;
374
- case "smtp":
375
- $result = $this->SmtpSend($header, $body);
376
- break;
377
- default:
378
- $this->SetError($this->Mailer . $this->Lang("mailer_not_supported"));
379
- $result = false;
380
- break;
381
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
382
 
383
- return $result;
384
- }
385
-
386
- /**
387
- * Sends mail using the $Sendmail program.
388
- * @access private
389
- * @return bool
390
- */
391
- function SendmailSend($header, $body) {
392
- if ($this->Sender != "")
393
- $sendmail = sprintf("%s -oi -f %s -t", $this->Sendmail, $this->Sender);
394
- else
395
- $sendmail = sprintf("%s -oi -t", $this->Sendmail);
396
-
397
- if(!@$mail = popen($sendmail, "w"))
398
- {
399
- $this->SetError($this->Lang("execute") . $this->Sendmail);
400
- return false;
401
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
402
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
403
  fputs($mail, $header);
404
  fputs($mail, $body);
405
-
406
- $result = pclose($mail) >> 8 & 0xFF;
407
- if($result != 0)
408
- {
409
- $this->SetError($this->Lang("execute") . $this->Sendmail);
410
- return false;
411
  }
412
-
413
- return true;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
414
  }
 
415
 
416
- /**
417
- * Sends mail using the PHP mail() function.
418
- * @access private
419
- * @return bool
420
- */
421
- function MailSend($header, $body) {
422
- $to = "";
423
- for($i = 0; $i < count($this->to); $i++)
424
- {
425
- if($i != 0) { $to .= ", "; }
426
- $to .= $this->to[$i][0];
 
 
 
427
  }
428
-
429
- if ($this->Sender != "" && strlen(ini_get("safe_mode"))< 1)
430
- {
431
- $old_from = ini_get("sendmail_from");
432
- ini_set("sendmail_from", $this->Sender);
433
- $params = sprintf("-oi -f %s", $this->Sender);
434
- $rt = @mail($to, $this->EncodeHeader($this->Subject), $body,
435
- $header, $params);
 
 
 
 
 
436
  }
437
- else
438
- $rt = @mail($to, $this->EncodeHeader($this->Subject), $body, $header);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
439
 
440
- if (isset($old_from))
441
- ini_set("sendmail_from", $old_from);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
442
 
443
- if(!$rt)
444
- {
445
- $this->SetError($this->Lang("instantiate"));
446
- return false;
447
- }
448
 
449
- return true;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
450
  }
451
 
452
- /**
453
- * Sends mail via SMTP using PhpSMTP (Author:
454
- * Chris Ryan). Returns bool. Returns false if there is a
455
- * bad MAIL FROM, RCPT, or DATA input.
456
- * @access private
457
- * @return bool
458
- */
459
- function SmtpSend($header, $body) {
460
- include_once($this->PluginDir . "class.smtp.php");
461
- $error = "";
462
- $bad_rcpt = array();
463
-
464
- if(!$this->SmtpConnect())
465
- return false;
466
-
467
- $smtp_from = ($this->Sender == "") ? $this->From : $this->Sender;
468
- if(!$this->smtp->Mail($smtp_from))
469
- {
470
- $error = $this->Lang("from_failed") . $smtp_from;
471
- $this->SetError($error);
472
- $this->smtp->Reset();
473
- return false;
474
  }
475
 
476
- // Attempt to send attach all recipients
477
- for($i = 0; $i < count($this->to); $i++)
478
- {
479
- if(!$this->smtp->Recipient($this->to[$i][0]))
480
- $bad_rcpt[] = $this->to[$i][0];
481
- }
482
- for($i = 0; $i < count($this->cc); $i++)
483
- {
484
- if(!$this->smtp->Recipient($this->cc[$i][0]))
485
- $bad_rcpt[] = $this->cc[$i][0];
486
- }
487
- for($i = 0; $i < count($this->bcc); $i++)
488
- {
489
- if(!$this->smtp->Recipient($this->bcc[$i][0]))
490
- $bad_rcpt[] = $this->bcc[$i][0];
491
- }
492
 
493
- if(count($bad_rcpt) > 0) // Create error message
494
- {
495
- for($i = 0; $i < count($bad_rcpt); $i++)
496
- {
497
- if($i != 0) { $error .= ", "; }
498
- $error .= $bad_rcpt[$i];
499
  }
500
- $error = $this->Lang("recipients_failed") . $error;
501
- $this->SetError($error);
502
- $this->smtp->Reset();
503
- return false;
504
- }
505
 
506
- if(!$this->smtp->Data($header . $body))
507
- {
508
- $this->SetError($this->Lang("data_not_accepted"));
509
- $this->smtp->Reset();
510
- return false;
511
- }
512
- if($this->SMTPKeepAlive == true)
513
- $this->smtp->Reset();
514
- else
515
- $this->SmtpClose();
516
 
517
- return true;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
518
  }
519
 
520
- /**
521
- * Initiates a connection to an SMTP server. Returns false if the
522
- * operation failed.
523
- * @access private
524
- * @return bool
525
- */
526
- function SmtpConnect() {
527
- if($this->smtp == NULL) { $this->smtp = new EM_SMTP(); }
528
-
529
- $this->smtp->do_debug = $this->SMTPDebug;
530
- $hosts = explode(";", $this->Host);
531
- $index = 0;
532
- $connection = ($this->smtp->Connected());
533
-
534
- // Retry while there is no connection
535
- while($index < count($hosts) && $connection == false)
536
- {
537
- if(strstr($hosts[$index], ":"))
538
- {
539
- $hostA = explode(':', $hosts[$index]);
540
- if (is_numeric(end($hostA)))
541
- $port = array_pop($hostA);
542
- else
543
- $port = $this->Port;
544
- $host = implode(':', $hostA);
545
  }
546
- else
547
- {
548
- $host = $hosts[$index];
549
- $port = $this->Port;
 
 
 
 
 
 
550
  }
 
 
551
 
552
- if($this->smtp->Connect($host, $port, $this->Timeout))
553
- {
554
- if ($this->Helo != '')
555
- $this->smtp->Hello($this->Helo);
556
- else
557
- $this->smtp->Hello($this->ServerHostname());
558
-
559
- if($this->SMTPAuth)
560
- {
561
- if(!$this->smtp->Authenticate($this->Username,
562
- $this->Password))
563
- {
564
- $this->SetError($this->Lang("authenticate"));
565
- $this->smtp->Reset();
566
- $connection = false;
567
- }
568
- }
569
- $connection = true;
570
- }
571
- $index++;
572
- }
573
- if(!$connection)
574
- $this->SetError($this->Lang("connect_host"));
575
-
576
- return $connection;
577
- }
578
-
579
- /**
580
- * Closes the active SMTP session if one exists.
581
- * @return void
582
- */
583
- function SmtpClose() {
584
- if($this->smtp != NULL)
585
- {
586
- if($this->smtp->Connected())
587
- {
588
- $this->smtp->Quit();
589
- $this->smtp->Close();
590
  }
 
 
 
 
 
 
 
 
 
591
  }
 
 
592
  }
593
 
594
- /**
595
- * Sets the language for all class error messages. Returns false
596
- * if it cannot load the language file. The default language type
597
- * is English.
598
- * @param string $lang_type Type of language (e.g. Portuguese: "br")
599
- * @param string $lang_path Path to the language file directory
600
- * @access public
601
- * @return bool
602
- */
603
- function SetLanguage($lang_type, $lang_path = "language/") {
604
- if(file_exists($lang_path.'phpmailer.lang-'.$lang_type.'.php'))
605
- include($lang_path.'phpmailer.lang-'.$lang_type.'.php');
606
- else if(file_exists($lang_path.'phpmailer.lang-en.php'))
607
- include($lang_path.'phpmailer.lang-en.php');
608
- else
609
- {
610
- $this->SetError("Could not load language file");
611
- return false;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
612
  }
613
- $this->language = $PHPMAILER_LANG;
614
-
615
- return true;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
616
  }
617
 
618
- /////////////////////////////////////////////////
619
- // MESSAGE CREATION METHODS
620
- /////////////////////////////////////////////////
621
-
622
- /**
623
- * Creates recipient headers.
624
- * @access private
625
- * @return string
626
- */
627
- function AddrAppend($type, $addr) {
628
- $addr_str = $type . ": ";
629
- $addr_str .= $this->AddrFormat($addr[0]);
630
- if(count($addr) > 1)
631
- {
632
- for($i = 1; $i < count($addr); $i++)
633
- $addr_str .= ", " . $this->AddrFormat($addr[$i]);
634
  }
635
- $addr_str .= $this->LE;
636
-
637
- return $addr_str;
638
- }
639
-
640
- /**
641
- * Formats an address correctly.
642
- * @access private
643
- * @return string
644
- */
645
- function AddrFormat($addr) {
646
- if(empty($addr[1]))
647
- $formatted = $addr[0];
648
- else
649
- {
650
- $formatted = $this->EncodeHeader($addr[1], 'phrase') . " <" .
651
- $addr[0] . ">";
652
  }
 
 
653
 
654
- return $formatted;
655
- }
656
-
657
- /**
658
- * Wraps message for use with mailers that do not
659
- * automatically perform wrapping and for quoted-printable.
660
- * Original written by philippe.
661
- * @access private
662
- * @return string
663
- */
664
- function WrapText($message, $length, $qp_mode = false) {
665
- $soft_break = ($qp_mode) ? sprintf(" =%s", $this->LE) : $this->LE;
666
-
667
- $message = $this->FixEOL($message);
668
- if (substr($message, -1) == $this->LE)
669
- $message = substr($message, 0, -1);
670
-
671
- $line = explode($this->LE, $message);
672
- $message = "";
673
- for ($i=0 ;$i < count($line); $i++)
674
- {
675
- $line_part = explode(" ", $line[$i]);
676
- $buf = "";
677
- for ($e = 0; $e<count($line_part); $e++)
678
- {
679
- $word = $line_part[$e];
680
- if ($qp_mode and (strlen($word) > $length))
681
- {
682
- $space_left = $length - strlen($buf) - 1;
683
- if ($e != 0)
684
- {
685
- if ($space_left > 20)
686
- {
687
- $len = $space_left;
688
- if (substr($word, $len - 1, 1) == "=")
689
- $len--;
690
- elseif (substr($word, $len - 2, 1) == "=")
691
- $len -= 2;
692
- $part = substr($word, 0, $len);
693
- $word = substr($word, $len);
694
- $buf .= " " . $part;
695
- $message .= $buf . sprintf("=%s", $this->LE);
696
- }
697
- else
698
- {
699
- $message .= $buf . $soft_break;
700
- }
701
- $buf = "";
702
- }
703
- while (strlen($word) > 0)
704
- {
705
- $len = $length;
706
- if (substr($word, $len - 1, 1) == "=")
707
- $len--;
708
- elseif (substr($word, $len - 2, 1) == "=")
709
- $len -= 2;
710
- $part = substr($word, 0, $len);
711
- $word = substr($word, $len);
712
-
713
- if (strlen($word) > 0)
714
- $message .= $part . sprintf("=%s", $this->LE);
715
- else
716
- $buf = $part;
717
- }
718
- }
719
- else
720
- {
721
- $buf_o = $buf;
722
- $buf .= ($e == 0) ? $word : (" " . $word);
723
-
724
- if (strlen($buf) > $length and $buf_o != "")
725
- {
726
- $message .= $buf_o . $soft_break;
727
- $buf = $word;
728
- }
729
- }
730
- }
731
- $message .= $buf . $this->LE;
732
- }
733
 
734
- return $message;
735
- }
736
-
737
- /**
738
- * Set the body wrapping.
739
- * @access private
740
- * @return void
741
- */
742
- function SetWordWrap() {
743
- if($this->WordWrap < 1)
744
- return;
745
-
746
- switch($this->message_type)
747
- {
748
- case "alt":
749
- // fall through
750
- case "alt_attachments":
751
- $this->AltBody = $this->WrapText($this->AltBody, $this->WordWrap);
752
- break;
753
- default:
754
- $this->Body = $this->WrapText($this->Body, $this->WordWrap);
755
- break;
756
- }
757
  }
758
 
759
- /**
760
- * Assembles message header.
761
- * @access private
762
- * @return string
763
- */
764
- function CreateHeader() {
765
- $result = "";
766
-
767
- // Set the boundaries
768
- $uniq_id = md5(uniqid(time()));
769
- $this->boundary[1] = "b1_" . $uniq_id;
770
- $this->boundary[2] = "b2_" . $uniq_id;
771
-
772
- $result .= $this->HeaderLine("Date", $this->RFCDate());
773
- if($this->Sender == "")
774
- $result .= $this->HeaderLine("Return-Path", trim($this->From));
775
- else
776
- $result .= $this->HeaderLine("Return-Path", trim($this->Sender));
777
-
778
- // To be created automatically by mail()
779
- if($this->Mailer != "mail")
780
- {
781
- if(count($this->to) > 0)
782
- $result .= $this->AddrAppend("To", $this->to);
783
- else if (count($this->cc) == 0)
784
- $result .= $this->HeaderLine("To", "undisclosed-recipients:;");
785
- if(count($this->cc) > 0)
786
- $result .= $this->AddrAppend("Cc", $this->cc);
787
- }
788
 
789
- $from = array();
790
- $from[0][0] = trim($this->From);
791
- $from[0][1] = $this->FromName;
792
- $result .= $this->AddrAppend("From", $from);
793
-
794
- // sendmail and mail() extract Bcc from the header before sending
795
- if((($this->Mailer == "sendmail") || ($this->Mailer == "mail")) && (count($this->bcc) > 0))
796
- $result .= $this->AddrAppend("Bcc", $this->bcc);
797
-
798
- if(count($this->ReplyTo) > 0)
799
- $result .= $this->AddrAppend("Reply-to", $this->ReplyTo);
800
-
801
- // mail() sets the subject itself
802
- if($this->Mailer != "mail")
803
- $result .= $this->HeaderLine("Subject", $this->EncodeHeader(trim($this->Subject)));
804
-
805
- $result .= sprintf("Message-ID: <%s@%s>%s", $uniq_id, $this->ServerHostname(), $this->LE);
806
- $result .= $this->HeaderLine("X-Priority", $this->Priority);
807
- $result .= $this->HeaderLine("X-Mailer", "PHPMailer [version " . $this->Version . "]");
808
-
809
- if($this->ConfirmReadingTo != "")
810
- {
811
- $result .= $this->HeaderLine("Disposition-Notification-To",
812
- "<" . trim($this->ConfirmReadingTo) . ">");
813
- }
814
 
815
- // Add custom headers
816
- for($index = 0; $index < count($this->CustomHeader); $index++)
817
- {
818
- $result .= $this->HeaderLine(trim($this->CustomHeader[$index][0]),
819
- $this->EncodeHeader(trim($this->CustomHeader[$index][1])));
820
- }
821
- $result .= $this->HeaderLine("MIME-Version", "1.0");
822
-
823
- switch($this->message_type)
824
- {
825
- case "plain":
826
- $result .= $this->HeaderLine("Content-Transfer-Encoding", $this->Encoding);
827
- $result .= sprintf("Content-Type: %s; charset=\"%s\"",
828
- $this->ContentType, $this->CharSet);
829
- break;
830
- case "attachments":
831
- // fall through
832
- case "alt_attachments":
833
- if($this->InlineImageExists())
834
- {
835
- $result .= sprintf("Content-Type: %s;%s\ttype=\"text/html\";%s\tboundary=\"%s\"%s",
836
- "multipart/related", $this->LE, $this->LE,
837
- $this->boundary[1], $this->LE);
838
- }
839
- else
840
- {
841
- $result .= $this->HeaderLine("Content-Type", "multipart/mixed;");
842
- $result .= $this->TextLine("\tboundary=\"" . $this->boundary[1] . '"');
843
- }
844
- break;
845
- case "alt":
846
- $result .= $this->HeaderLine("Content-Type", "multipart/alternative;");
847
- $result .= $this->TextLine("\tboundary=\"" . $this->boundary[1] . '"');
848
- break;
849
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
850
 
851
- if($this->Mailer != "mail")
852
- $result .= $this->LE.$this->LE;
853
-
854
- return $result;
855
- }
856
-
857
- /**
858
- * Assembles the message body. Returns an empty string on failure.
859
- * @access private
860
- * @return string
861
- */
862
- function CreateBody() {
863
- $result = "";
864
-
865
- $this->SetWordWrap();
866
-
867
- switch($this->message_type)
868
- {
869
- case "alt":
870
- $result .= $this->GetBoundary($this->boundary[1], "",
871
- "text/plain", "");
872
- $result .= $this->EncodeString($this->AltBody, $this->Encoding);
873
- $result .= $this->LE.$this->LE;
874
- $result .= $this->GetBoundary($this->boundary[1], "",
875
- "text/html", "");
876
-
877
- $result .= $this->EncodeString($this->Body, $this->Encoding);
878
- $result .= $this->LE.$this->LE;
879
-
880
- $result .= $this->EndBoundary($this->boundary[1]);
881
- break;
882
- case "plain":
883
- $result .= $this->EncodeString($this->Body, $this->Encoding);
884
- break;
885
- case "attachments":
886
- $result .= $this->GetBoundary($this->boundary[1], "", "", "");
887
- $result .= $this->EncodeString($this->Body, $this->Encoding);
888
- $result .= $this->LE;
889
-
890
- $result .= $this->AttachAll();
891
- break;
892
- case "alt_attachments":
893
- $result .= sprintf("--%s%s", $this->boundary[1], $this->LE);
894
- $result .= sprintf("Content-Type: %s;%s" .
895
- "\tboundary=\"%s\"%s",
896
- "multipart/alternative", $this->LE,
897
- $this->boundary[2], $this->LE.$this->LE);
898
-
899
- // Create text body
900
- $result .= $this->GetBoundary($this->boundary[2], "",
901
- "text/plain", "") . $this->LE;
902
-
903
- $result .= $this->EncodeString($this->AltBody, $this->Encoding);
904
- $result .= $this->LE.$this->LE;
905
-
906
- // Create the HTML body
907
- $result .= $this->GetBoundary($this->boundary[2], "",
908
- "text/html", "") . $this->LE;
909
-
910
- $result .= $this->EncodeString($this->Body, $this->Encoding);
911
- $result .= $this->LE.$this->LE;
912
-
913
- $result .= $this->EndBoundary($this->boundary[2]);
914
-
915
- $result .= $this->AttachAll();
916
- break;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
917
  }
918
- if($this->IsError())
919
- $result = "";
920
-
921
- return $result;
922
- }
923
-
924
- /**
925
- * Returns the start of a message boundary.
926
- * @access private
927
- */
928
- function GetBoundary($boundary, $charSet, $contentType, $encoding) {
929
- $result = "";
930
- if($charSet == "") { $charSet = $this->CharSet; }
931
- if($contentType == "") { $contentType = $this->ContentType; }
932
- if($encoding == "") { $encoding = $this->Encoding; }
933
-
934
- $result .= $this->TextLine("--" . $boundary);
935
- $result .= sprintf("Content-Type: %s; charset = \"%s\"",
936
- $contentType, $charSet);
937
- $result .= $this->LE;
938
- $result .= $this->HeaderLine("Content-Transfer-Encoding", $encoding);
939
- $result .= $this->LE;
940
-
941
- return $result;
942
- }
943
-
944
- /**
945
- * Returns the end of a message boundary.
946
- * @access private
947
- */
948
- function EndBoundary($boundary) {
949
- return $this->LE . "--" . $boundary . "--" . $this->LE;
950
- }
951
-
952
- /**
953
- * Sets the message type.
954
- * @access private
955
- * @return void
956
- */
957
- function SetMessageType() {
958
- if(count($this->attachment) < 1 && strlen($this->AltBody) < 1)
959
- $this->message_type = "plain";
960
- else
961
- {
962
- if(count($this->attachment) > 0)
963
- $this->message_type = "attachments";
964
- if(strlen($this->AltBody) > 0 && count($this->attachment) < 1)
965
- $this->message_type = "alt";
966
- if(strlen($this->AltBody) > 0 && count($this->attachment) > 0)
967
- $this->message_type = "alt_attachments";
968
  }
 
969
  }
970
 
971
- /**
972
- * Returns a formatted header line.
973
- * @access private
974
- * @return string
975
- */
976
- function HeaderLine($name, $value) {
977
- return $name . ": " . $value . $this->LE;
978
- }
979
-
980
- /**
981
- * Returns a formatted mail line.
982
- * @access private
983
- * @return string
984
- */
985
- function TextLine($value) {
986
- return $value . $this->LE;
987
- }
988
-
989
- /////////////////////////////////////////////////
990
- // ATTACHMENT METHODS
991
- /////////////////////////////////////////////////
992
-
993
- /**
994
- * Adds an attachment from a path on the filesystem.
995
- * Returns false if the file could not be found
996
- * or accessed.
997
- * @param string $path Path to the attachment.
998
- * @param string $name Overrides the attachment name.
999
- * @param string $encoding File encoding (see $Encoding).
1000
- * @param string $type File extension (MIME) type.
1001
- * @return bool
1002
- */
1003
- function AddAttachment($path, $name = "", $encoding = "base64",
1004
- $type = "application/octet-stream") {
1005
- if(!@is_file($path))
1006
- {
1007
- $this->SetError($this->Lang("file_access") . $path);
1008
- return false;
1009
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1010
 
1011
- $filename = basename($path);
1012
- if($name == "")
1013
- $name = $filename;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1014
 
1015
- $cur = count($this->attachment);
1016
- $this->attachment[$cur][0] = $path;
1017
- $this->attachment[$cur][1] = $filename;
1018
- $this->attachment[$cur][2] = $name;
1019
- $this->attachment[$cur][3] = $encoding;
1020
- $this->attachment[$cur][4] = $type;
1021
- $this->attachment[$cur][5] = false; // isStringAttachment
1022
- $this->attachment[$cur][6] = "attachment";
1023
- $this->attachment[$cur][7] = 0;
 
 
 
 
 
 
 
 
 
 
1024
 
1025
- return true;
1026
- }
1027
 
1028
- /**
1029
- * Attaches all fs, string, and binary attachments to the message.
1030
- * Returns an empty string on failure.
1031
- * @access private
1032
- * @return string
1033
- */
1034
- function AttachAll() {
1035
- // Return text of body
1036
- $mime = array();
1037
-
1038
- // Add all attachments
1039
- for($i = 0; $i < count($this->attachment); $i++)
1040
- {
1041
- // Check for string attachment
1042
- $bString = $this->attachment[$i][5];
1043
- if ($bString)
1044
- $string = $this->attachment[$i][0];
1045
- else
1046
- $path = $this->attachment[$i][0];
1047
-
1048
- $filename = $this->attachment[$i][1];
1049
- $name = $this->attachment[$i][2];
1050
- $encoding = $this->attachment[$i][3];
1051
- $type = $this->attachment[$i][4];
1052
- $disposition = $this->attachment[$i][6];
1053
- $cid = $this->attachment[$i][7];
1054
-
1055
- $mime[] = sprintf("--%s%s", $this->boundary[1], $this->LE);
1056
- $mime[] = sprintf("Content-Type: %s; name=\"%s\"%s", $type, $name, $this->LE);
1057
- $mime[] = sprintf("Content-Transfer-Encoding: %s%s", $encoding, $this->LE);
1058
-
1059
- if($disposition == "inline")
1060
- $mime[] = sprintf("Content-ID: <%s>%s", $cid, $this->LE);
1061
-
1062
- $mime[] = sprintf("Content-Disposition: %s; filename=\"%s\"%s",
1063
- $disposition, $name, $this->LE.$this->LE);
1064
-
1065
- // Encode as string attachment
1066
- if($bString)
1067
- {
1068
- $mime[] = $this->EncodeString($string, $encoding);
1069
- if($this->IsError()) { return ""; }
1070
- $mime[] = $this->LE.$this->LE;
1071
- }
1072
- else
1073
- {
1074
- $mime[] = $this->EncodeFile($path, $encoding);
1075
- if($this->IsError()) { return ""; }
1076
- $mime[] = $this->LE.$this->LE;
1077
- }
1078
  }
 
 
1079
 
1080
- $mime[] = sprintf("--%s--%s", $this->boundary[1], $this->LE);
1081
-
1082
- return join("", $mime);
1083
- }
1084
-
1085
- /**
1086
- * Encodes attachment in requested format. Returns an
1087
- * empty string on failure.
1088
- * @access private
1089
- * @return string
1090
- */
1091
- function EncodeFile ($path, $encoding = "base64") {
1092
- if(!@$fd = fopen($path, "rb"))
1093
- {
1094
- $this->SetError($this->Lang("file_open") . $path);
1095
- return "";
 
 
 
 
 
 
1096
  }
 
 
1097
  $magic_quotes = get_magic_quotes_runtime();
1098
  set_magic_quotes_runtime(0);
1099
- $file_buffer = fread($fd, filesize($path));
1100
- $file_buffer = $this->EncodeString($file_buffer, $encoding);
1101
- fclose($fd);
 
1102
  set_magic_quotes_runtime($magic_quotes);
1103
-
1104
- return $file_buffer;
1105
- }
1106
-
1107
- /**
1108
- * Encodes string to requested format. Returns an
1109
- * empty string on failure.
1110
- * @access private
1111
- * @return string
1112
- */
1113
- function EncodeString ($str, $encoding = "base64") {
1114
- $encoded = "";
1115
- switch(strtolower($encoding)) {
1116
- case "base64":
1117
- // chunk_split is found in PHP >= 3.0.6
1118
- $encoded = chunk_split(base64_encode($str), 76, $this->LE);
1119
- break;
1120
- case "7bit":
1121
- case "8bit":
1122
- $encoded = $this->FixEOL($str);
1123
- if (substr($encoded, -(strlen($this->LE))) != $this->LE)
1124
- $encoded .= $this->LE;
1125
- break;
1126
- case "binary":
1127
- $encoded = $str;
1128
- break;
1129
- case "quoted-printable":
1130
- $encoded = $this->EncodeQP($str);
1131
- break;
1132
- default:
1133
- $this->SetError($this->Lang("encoding") . $encoding);
1134
- break;
1135
- }
1136
- return $encoded;
1137
- }
1138
-
1139
- /**
1140
- * Encode a header string to best of Q, B, quoted or none.
1141
- * @access private
1142
- * @return string
1143
- */
1144
- function EncodeHeader ($str, $position = 'text') {
1145
- $x = 0;
1146
-
1147
- switch (strtolower($position)) {
1148
- case 'phrase':
1149
- if (!preg_match('/[\200-\377]/', $str)) {
1150
- // Can't use addslashes as we don't know what value has magic_quotes_sybase.
1151
- $encoded = addcslashes($str, "\0..\37\177\\\"");
1152
-
1153
- if (($str == $encoded) && !preg_match('/[^A-Za-z0-9!#$%&\'*+\/=?^_`{|}~ -]/', $str))
1154
- return ($encoded);
1155
- else
1156
- return ("\"$encoded\"");
1157
- }
1158
- $x = preg_match_all('/[^\040\041\043-\133\135-\176]/', $str, $matches);
1159
- break;
1160
- case 'comment':
1161
- $x = preg_match_all('/[()"]/', $str, $matches);
1162
- // Fall-through
1163
- case 'text':
1164
- default:
1165
- $x += preg_match_all('/[\000-\010\013\014\016-\037\177-\377]/', $str, $matches);
1166
- break;
1167
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1168
 
1169
- if ($x == 0)
1170
- return ($str);
 
1171
 
1172
- $maxlen = 75 - 7 - strlen($this->CharSet);
1173
- // Try to select the encoding which should produce the shortest output
1174
- if (strlen($str)/3 < $x) {
1175
- $encoding = 'B';
 
 
 
 
 
1176
  $encoded = base64_encode($str);
1177
  $maxlen -= $maxlen % 4;
1178
  $encoded = trim(chunk_split($encoded, $maxlen, "\n"));
1179
- } else {
1180
- $encoding = 'Q';
1181
- $encoded = $this->EncodeQ($str, $position);
1182
- $encoded = $this->WrapText($encoded, $maxlen, true);
1183
- $encoded = str_replace("=".$this->LE, "\n", trim($encoded));
1184
  }
 
 
 
 
 
 
1185
 
1186
- $encoded = preg_replace('/^(.*)$/m', " =?".$this->CharSet."?$encoding?\\1?=", $encoded);
1187
- $encoded = trim(str_replace("\n", $this->LE, $encoded));
1188
-
1189
- return $encoded;
1190
- }
1191
-
1192
- /**
1193
- * Encode string to quoted-printable.
1194
- * @access private
1195
- * @return string
1196
- */
1197
- function EncodeQP ($str) {
1198
- $encoded = $this->FixEOL($str);
1199
- if (substr($encoded, -(strlen($this->LE))) != $this->LE)
1200
- $encoded .= $this->LE;
1201
-
1202
- // Replace every high ascii, control and = characters
1203
- $encoded = preg_replace('/([\000-\010\013\014\016-\037\075\177-\377])/e',
1204
- "'='.sprintf('%02X', ord('\\1'))", $encoded);
1205
- // Replace every spaces and tabs when it's the last character on a line
1206
- $encoded = preg_replace("/([\011\040])".$this->LE."/e",
1207
- "'='.sprintf('%02X', ord('\\1')).'".$this->LE."'", $encoded);
1208
-
1209
- // Maximum line length of 76 characters before CRLF (74 + space + '=')
1210
- $encoded = $this->WrapText($encoded, 74, true);
1211
-
1212
- return $encoded;
1213
- }
1214
-
1215
- /**
1216
- * Encode string to q encoding.
1217
- * @access private
1218
- * @return string
1219
- */
1220
- function EncodeQ ($str, $position = "text") {
1221
- // There should not be any EOL in the string
1222
- $encoded = preg_replace("[\r\n]", "", $str);
1223
-
1224
- switch (strtolower($position)) {
1225
- case "phrase":
1226
- $encoded = preg_replace("/([^A-Za-z0-9!*+\/ -])/e", "'='.sprintf('%02X', ord('\\1'))", $encoded);
1227
- break;
1228
- case "comment":
1229
- $encoded = preg_replace("/([\(\)\"])/e", "'='.sprintf('%02X', ord('\\1'))", $encoded);
1230
- case "text":
1231
- default:
1232
- // Replace every high ascii, control =, ? and _ characters
1233
- $encoded = preg_replace('/([\000-\011\013\014\016-\037\075\077\137\177-\377])/e',
1234
- "'='.sprintf('%02X', ord('\\1'))", $encoded);
1235
- break;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1236
  }
1237
-
1238
- // Replace every spaces to _ (more readable than =20)
1239
- $encoded = str_replace(" ", "_", $encoded);
1240
-
1241
- return $encoded;
1242
- }
1243
-
1244
- /**
1245
- * Adds a string or binary attachment (non-filesystem) to the list.
1246
- * This method can be used to attach ascii or binary data,
1247
- * such as a BLOB record from a database.
1248
- * @param string $string String attachment data.
1249
- * @param string $filename Name of the attachment.
1250
- * @param string $encoding File encoding (see $Encoding).
1251
- * @param string $type File extension (MIME) type.
1252
- * @return void
1253
- */
1254
- function AddStringAttachment($string, $filename, $encoding = "base64",
1255
- $type = "application/octet-stream") {
1256
- // Append to $attachment array
1257
- $cur = count($this->attachment);
1258
- $this->attachment[$cur][0] = $string;
1259
- $this->attachment[$cur][1] = $filename;
1260
- $this->attachment[$cur][2] = $filename;
1261
- $this->attachment[$cur][3] = $encoding;
1262
- $this->attachment[$cur][4] = $type;
1263
- $this->attachment[$cur][5] = true; // isString
1264
- $this->attachment[$cur][6] = "attachment";
1265
- $this->attachment[$cur][7] = 0;
1266
- }
1267
-
1268
- /**
1269
- * Adds an embedded attachment. This can include images, sounds, and
1270
- * just about any other document. Make sure to set the $type to an
1271
- * image type. For JPEG images use "image/jpeg" and for GIF images
1272
- * use "image/gif".
1273
- * @param string $path Path to the attachment.
1274
- * @param string $cid Content ID of the attachment. Use this to identify
1275
- * the Id for accessing the image in an HTML form.
1276
- * @param string $name Overrides the attachment name.
1277
- * @param string $encoding File encoding (see $Encoding).
1278
- * @param string $type File extension (MIME) type.
1279
- * @return bool
1280
- */
1281
- function AddEmbeddedImage($path, $cid, $name = "", $encoding = "base64",
1282
- $type = "application/octet-stream") {
1283
-
1284
- if(!@is_file($path))
1285
- {
1286
- $this->SetError($this->Lang("file_access") . $path);
1287
- return false;
1288
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1289
 
1290
- $filename = basename($path);
1291
- if($name == "")
1292
- $name = $filename;
1293
-
1294
- // Append to $attachment array
1295
- $cur = count($this->attachment);
1296
- $this->attachment[$cur][0] = $path;
1297
- $this->attachment[$cur][1] = $filename;
1298
- $this->attachment[$cur][2] = $name;
1299
- $this->attachment[$cur][3] = $encoding;
1300
- $this->attachment[$cur][4] = $type;
1301
- $this->attachment[$cur][5] = false; // isStringAttachment
1302
- $this->attachment[$cur][6] = "inline";
1303
- $this->attachment[$cur][7] = $cid;
1304
-
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1305
  return true;
 
1306
  }
1307
-
1308
- /**
1309
- * Returns true if an inline attachment is present.
1310
- * @access private
1311
- * @return bool
1312
- */
1313
- function InlineImageExists() {
1314
- $result = false;
1315
- for($i = 0; $i < count($this->attachment); $i++)
1316
- {
1317
- if($this->attachment[$i][6] == "inline")
1318
- {
1319
- $result = true;
1320
- break;
1321
- }
1322
- }
1323
-
1324
- return $result;
1325
- }
1326
-
1327
- /////////////////////////////////////////////////
1328
- // MESSAGE RESET METHODS
1329
- /////////////////////////////////////////////////
1330
-
1331
- /**
1332
- * Clears all recipients assigned in the TO array. Returns void.
1333
- * @return void
1334
- */
1335
- function ClearAddresses() {
1336
- $this->to = array();
1337
- }
1338
-
1339
- /**
1340
- * Clears all recipients assigned in the CC array. Returns void.
1341
- * @return void
1342
- */
1343
- function ClearCCs() {
1344
- $this->cc = array();
1345
- }
1346
-
1347
- /**
1348
- * Clears all recipients assigned in the BCC array. Returns void.
1349
- * @return void
1350
- */
1351
- function ClearBCCs() {
1352
- $this->bcc = array();
1353
- }
1354
-
1355
- /**
1356
- * Clears all recipients assigned in the ReplyTo array. Returns void.
1357
- * @return void
1358
- */
1359
- function ClearReplyTos() {
1360
- $this->ReplyTo = array();
1361
- }
1362
-
1363
- /**
1364
- * Clears all recipients assigned in the TO, CC and BCC
1365
- * array. Returns void.
1366
- * @return void
1367
- */
1368
- function ClearAllRecipients() {
1369
- $this->to = array();
1370
- $this->cc = array();
1371
- $this->bcc = array();
1372
- }
1373
-
1374
- /**
1375
- * Clears all previously set filesystem, string, and binary
1376
- * attachments. Returns void.
1377
- * @return void
1378
- */
1379
- function ClearAttachments() {
1380
- $this->attachment = array();
1381
- }
1382
-
1383
- /**
1384
- * Clears all custom headers. Returns void.
1385
- * @return void
1386
- */
1387
- function ClearCustomHeaders() {
1388
- $this->CustomHeader = array();
1389
- }
1390
-
1391
-
1392
- /////////////////////////////////////////////////
1393
- // MISCELLANEOUS METHODS
1394
- /////////////////////////////////////////////////
1395
-
1396
- /**
1397
- * Adds the error message to the error container.
1398
- * Returns void.
1399
- * @access private
1400
- * @return void
1401
- */
1402
- function SetError($msg) {
1403
- $this->error_count++;
1404
- $this->ErrorInfo = $msg;
1405
- }
1406
-
1407
- /**
1408
- * Returns the proper RFC 822 formatted date.
1409
- * @access private
1410
- * @return string
1411
- */
1412
- function RFCDate() {
1413
- $tz = date("Z");
1414
- $tzs = ($tz < 0) ? "-" : "+";
1415
- $tz = abs($tz);
1416
- $tz = ($tz/3600)*100 + ($tz%3600)/60;
1417
- $result = sprintf("%s %s%04d", date("D, j M Y H:i:s"), $tzs, $tz);
1418
-
1419
- return $result;
1420
- }
1421
-
1422
- /**
1423
- * Returns the appropriate server variable. Should work with both
1424
- * PHP 4.1.0+ as well as older versions. Returns an empty string
1425
- * if nothing is found.
1426
- * @access private
1427
- * @return mixed
1428
- */
1429
- function ServerVar($varName) {
1430
- global $HTTP_SERVER_VARS;
1431
- global $HTTP_ENV_VARS;
1432
-
1433
- if(!isset($_SERVER))
1434
- {
1435
- $_SERVER = $HTTP_SERVER_VARS;
1436
- if(!isset($_SERVER["REMOTE_ADDR"]))
1437
- $_SERVER = $HTTP_ENV_VARS; // must be Apache
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1438
  }
1439
-
1440
- if(isset($_SERVER[$varName]))
1441
- return $_SERVER[$varName];
1442
- else
1443
- return "";
1444
- }
1445
-
1446
- /**
1447
- * Returns the server hostname or 'localhost.localdomain' if unknown.
1448
- * @access private
1449
- * @return string
1450
- */
1451
- function ServerHostname() {
1452
- if ($this->Hostname != "")
1453
- $result = $this->Hostname;
1454
- elseif ($this->ServerVar('SERVER_NAME') != "")
1455
- $result = $this->ServerVar('SERVER_NAME');
1456
- else
1457
- $result = "localhost.localdomain";
1458
-
1459
- return $result;
1460
- }
1461
-
1462
- /**
1463
- * Returns a message in the appropriate language.
1464
- * @access private
1465
- * @return string
1466
- */
1467
- function Lang($key) {
1468
- if(count($this->language) < 1)
1469
- $this->SetLanguage("en"); // set the default language
1470
-
1471
- if(isset($this->language[$key]))
1472
- return $this->language[$key];
1473
- else
1474
- return "Language string failed to load: " . $key;
1475
- }
1476
-
1477
- /**
1478
- * Returns true if an error occurred.
1479
- * @return bool
1480
- */
1481
- function IsError() {
1482
- return ($this->error_count > 0);
1483
- }
1484
-
1485
- /**
1486
- * Changes every end of line from CR or LF to CRLF.
1487
- * @access private
1488
- * @return string
1489
- */
1490
- function FixEOL($str) {
1491
- $str = str_replace("\r\n", "\n", $str);
1492
- $str = str_replace("\r", "\n", $str);
1493
- $str = str_replace("\n", $this->LE, $str);
1494
- return $str;
1495
- }
1496
-
1497
- /**
1498
- * Adds a custom header.
1499
- * @return void
1500
- */
1501
- function AddCustomHeader($custom_header) {
1502
- $this->CustomHeader[] = explode(":", $custom_header, 2);
1503
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
  <?php
2
+ /*~ class.phpmailer.php
3
+ .---------------------------------------------------------------------------.
4
+ | Software: PHPMailer - PHP email class |
5
+ | Version: 5.2 |
6
+ | Site: https://code.google.com/a/apache-extras.org/p/phpmailer/ |
7
+ | ------------------------------------------------------------------------- |
8
+ | Admin: Jim Jagielski (project admininistrator) |
9
+ | Authors: Andy Prevost (codeworxtech) codeworxtech@users.sourceforge.net |
10
+ | : Marcus Bointon (coolbru) coolbru@users.sourceforge.net |
11
+ | : Jim Jagielski (jimjag) jimjag@gmail.com |
12
+ | Founder: Brent R. Matzelle (original founder) |
13
+ | Copyright (c) 2010-2011, Jim Jagielski. All Rights Reserved. |
14
+ | Copyright (c) 2004-2009, Andy Prevost. All Rights Reserved. |
15
+ | Copyright (c) 2001-2003, Brent R. Matzelle |
16
+ | ------------------------------------------------------------------------- |
17
+ | License: Distributed under the Lesser General Public License (LGPL) |
18
+ | http://www.gnu.org/copyleft/lesser.html |
19
+ | This program is distributed in the hope that it will be useful - WITHOUT |
20
+ | ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
21
+ | FITNESS FOR A PARTICULAR PURPOSE. |
22
+ '---------------------------------------------------------------------------'
23
+ */
24
 
25
  /**
26
  * PHPMailer - PHP email transport class
27
+ * NOTE: Requires PHP version 5 or later
28
  * @package PHPMailer
29
+ * @author Andy Prevost
30
+ * @author Marcus Bointon
31
+ * @author Jim Jagielski
32
+ * @copyright 2010 - 2011 Jim Jagielski
33
+ * @copyright 2004 - 2009 Andy Prevost
34
+ * @version $Id: class.phpmailer.php 450 2010-06-23 16:46:33Z coolbru $
35
+ * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License
36
  */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
37
 
38
+ if (version_compare(PHP_VERSION, '5.0.0', '<') ) exit("Sorry, this version of PHPMailer will only run on PHP version 5 or greater!\n");
39
+
40
+ class EM_PHPMailer {
41
+
42
+ /////////////////////////////////////////////////
43
+ // PROPERTIES, PUBLIC
44
+ /////////////////////////////////////////////////
45
+
46
+ /**
47
+ * Email priority (1 = High, 3 = Normal, 5 = low).
48
+ * @var int
49
+ */
50
+ public $Priority = 3;
51
+
52
+ /**
53
+ * Sets the CharSet of the message.
54
+ * @var string
55
+ */
56
+ public $CharSet = 'iso-8859-1';
57
+
58
+ /**
59
+ * Sets the Content-type of the message.
60
+ * @var string
61
+ */
62
+ public $ContentType = 'text/plain';
63
+
64
+ /**
65
+ * Sets the Encoding of the message. Options for this are
66
+ * "8bit", "7bit", "binary", "base64", and "quoted-printable".
67
+ * @var string
68
+ */
69
+ public $Encoding = '8bit';
70
+
71
+ /**
72
+ * Holds the most recent mailer error message.
73
+ * @var string
74
+ */
75
+ public $ErrorInfo = '';
76
+
77
+ /**
78
+ * Sets the From email address for the message.
79
+ * @var string
80
+ */
81
+ public $From = 'root@localhost';
82
+
83
+ /**
84
+ * Sets the From name of the message.
85
+ * @var string
86
+ */
87
+ public $FromName = 'Root User';
88
+
89
+ /**
90
+ * Sets the Sender email (Return-Path) of the message. If not empty,
91
+ * will be sent via -f to sendmail or as 'MAIL FROM' in smtp mode.
92
+ * @var string
93
+ */
94
+ public $Sender = '';
95
+
96
+ /**
97
+ * Sets the Subject of the message.
98
+ * @var string
99
+ */
100
+ public $Subject = '';
101
+
102
+ /**
103
+ * Sets the Body of the message. This can be either an HTML or text body.
104
+ * If HTML then run IsHTML(true).
105
+ * @var string
106
+ */
107
+ public $Body = '';
108
+
109
+ /**
110
+ * Sets the text-only body of the message. This automatically sets the
111
+ * email to multipart/alternative. This body can be read by mail
112
+ * clients that do not have HTML email capability such as mutt. Clients
113
+ * that can read HTML will view the normal Body.
114
+ * @var string
115
+ */
116
+ public $AltBody = '';
117
+
118
+ /**
119
+ * Stores the complete compiled MIME message body.
120
+ * @var string
121
+ * @access protected
122
+ */
123
+ protected $MIMEBody = '';
124
+
125
+ /**
126
+ * Stores the complete compiled MIME message headers.
127
+ * @var string
128
+ * @access protected
129
+ */
130
+ protected $MIMEHeader = '';
131
+
132
+ /**
133
+ * Sets word wrapping on the body of the message to a given number of
134
+ * characters.
135
+ * @var int
136
+ */
137
+ public $WordWrap = 0;
138
+
139
+ /**
140
+ * Method to send mail: ("mail", "sendmail", or "smtp").
141
+ * @var string
142
+ */
143
+ public $Mailer = 'mail';
144
+
145
+ /**
146
+ * Sets the path of the sendmail program.
147
+ * @var string
148
+ */
149
+ public $Sendmail = '/usr/sbin/sendmail';
150
+
151
+ /**
152
+ * Path to PHPMailer plugins. Useful if the SMTP class
153
+ * is in a different directory than the PHP include path.
154
+ * @var string
155
+ */
156
+ public $PluginDir = '';
157
+
158
+ /**
159
+ * Sets the email address that a reading confirmation will be sent.
160
+ * @var string
161
+ */
162
+ public $ConfirmReadingTo = '';
163
+
164
+ /**
165
+ * Sets the hostname to use in Message-Id and Received headers
166
+ * and as default HELO string. If empty, the value returned
167
+ * by SERVER_NAME is used or 'localhost.localdomain'.
168
+ * @var string
169
+ */
170
+ public $Hostname = '';
171
+
172
+ /**
173
+ * Sets the message ID to be used in the Message-Id header.
174
+ * If empty, a unique id will be generated.
175
+ * @var string
176
+ */
177
+ public $MessageID = '';
178
+
179
+ /////////////////////////////////////////////////
180
+ // PROPERTIES FOR SMTP
181
+ /////////////////////////////////////////////////
182
+
183
+ /**
184
+ * Sets the SMTP hosts. All hosts must be separated by a
185
+ * semicolon. You can also specify a different port
186
+ * for each host by using this format: [hostname:port]
187
+ * (e.g. "smtp1.example.com:25;smtp2.example.com").
188
+ * Hosts will be tried in order.
189
+ * @var string
190
+ */
191
+ public $Host = 'localhost';
192
+
193
+ /**
194
+ * Sets the default SMTP server port.
195
+ * @var int
196
+ */
197
+ public $Port = 25;
198
+
199
+ /**
200
+ * Sets the SMTP HELO of the message (Default is $Hostname).
201
+ * @var string
202
+ */
203
+ public $Helo = '';
204
+
205
+ /**
206
+ * Sets connection prefix.
207
+ * Options are "", "ssl" or "tls"
208
+ * @var string
209
+ */
210
+ public $SMTPSecure = '';
211
+
212
+ /**
213
+ * Sets SMTP authentication. Utilizes the Username and Password variables.
214
+ * @var bool
215
+ */
216
+ public $SMTPAuth = false;
217
+
218
+ /**
219
+ * Sets SMTP username.
220
+ * @var string
221
+ */
222
+ public $Username = '';
223
+
224
+ /**
225
+ * Sets SMTP password.
226
+ * @var string
227
+ */
228
+ public $Password = '';
229
+
230
+ /**
231
+ * Sets the SMTP server timeout in seconds.
232
+ * This function will not work with the win32 version.
233
+ * @var int
234
+ */
235
+ public $Timeout = 10;
236
+
237
+ /**
238
+ * Sets SMTP class debugging on or off.
239
+ * @var bool
240
+ */
241
+ public $SMTPDebug = false;
242
+
243
+ /**
244
+ * Prevents the SMTP connection from being closed after each mail
245
+ * sending. If this is set to true then to close the connection
246
+ * requires an explicit call to SmtpClose().
247
+ * @var bool
248
+ */
249
+ public $SMTPKeepAlive = false;
250
+
251
+ /**
252
+ * Provides the ability to have the TO field process individual
253
+ * emails, instead of sending to entire TO addresses
254
+ * @var bool
255
+ */
256
+ public $SingleTo = false;
257
+
258
+ /**
259
+ * If SingleTo is true, this provides the array to hold the email addresses
260
+ * @var bool
261
+ */
262
+ public $SingleToArray = array();
263
+
264
+ /**
265
+ * Provides the ability to change the line ending
266
+ * @var string
267
+ */
268
+ public $LE = "\n";
269
+
270
+ /**
271
+ * Used with DKIM DNS Resource Record
272
+ * @var string
273
+ */
274
+ public $DKIM_selector = 'phpmailer';
275
+
276
+ /**
277
+ * Used with DKIM DNS Resource Record
278
+ * optional, in format of email address 'you@yourdomain.com'
279
+ * @var string
280
+ */
281
+ public $DKIM_identity = '';
282
+
283
+ /**
284
+ * Used with DKIM DNS Resource Record
285
+ * @var string
286
+ */
287
+ public $DKIM_passphrase = '';
288
+
289
+ /**
290
+ * Used with DKIM DNS Resource Record
291
+ * optional, in format of email address 'you@yourdomain.com'
292
+ * @var string
293
+ */
294
+ public $DKIM_domain = '';
295
+
296
+ /**
297
+ * Used with DKIM DNS Resource Record
298
+ * optional, in format of email address 'you@yourdomain.com'
299
+ * @var string
300
+ */
301
+ public $DKIM_private = '';
302
+
303
+ /**
304
+ * Callback Action function name
305
+ * the function that handles the result of the send email action. Parameters:
306
+ * bool $result result of the send action
307
+ * string $to email address of the recipient
308
+ * string $cc cc email addresses
309
+ * string $bcc bcc email addresses
310
+ * string $subject the subject
311
+ * string $body the email body
312
+ * @var string
313
+ */
314
+ public $action_function = ''; //'callbackAction';
315
+
316
+ /**
317
+ * Sets the PHPMailer Version number
318
+ * @var string
319
+ */
320
+ public $Version = '5.2';
321
+
322
+ /**
323
+ * What to use in the X-Mailer header
324
+ * @var string
325
+ */
326
+ public $XMailer = '';
327
+
328
+ /////////////////////////////////////////////////
329
+ // PROPERTIES, PRIVATE AND PROTECTED
330
+ /////////////////////////////////////////////////
331
+
332
+ protected $smtp = NULL;
333
+ protected $to = array();
334
+ protected $cc = array();
335
+ protected $bcc = array();
336
+ protected $ReplyTo = array();
337
+ protected $all_recipients = array();
338
+ protected $attachment = array();
339
+ protected $CustomHeader = array();
340
+ protected $message_type = '';
341
+ protected $boundary = array();
342
+ protected $language = array();
343
+ protected $error_count = 0;
344
+ protected $sign_cert_file = '';
345
+ protected $sign_key_file = '';
346
+ protected $sign_key_pass = '';
347
+ protected $exceptions = false;
348
+
349
+ /////////////////////////////////////////////////
350
+ // CONSTANTS
351
+ /////////////////////////////////////////////////
352
+
353
+ const STOP_MESSAGE = 0; // message only, continue processing
354
+ const STOP_CONTINUE = 1; // message?, likely ok to continue processing
355
+ const STOP_CRITICAL = 2; // message, plus full stop, critical error reached
356
+
357
+ /////////////////////////////////////////////////
358
+ // METHODS, VARIABLES
359
+ /////////////////////////////////////////////////
360
+
361
+ /**
362
+ * Constructor
363
+ * @param boolean $exceptions Should we throw external exceptions?
364
+ */
365
+ public function __construct($exceptions = false) {
366
+ $this->exceptions = ($exceptions == true);
367
+ }
368
+
369
+ /**
370
+ * Sets message type to HTML.
371
+ * @param bool $ishtml
372
+ * @return void
373
+ */
374
+ public function IsHTML($ishtml = true) {
375
+ if ($ishtml) {
376
+ $this->ContentType = 'text/html';
377
+ } else {
378
+ $this->ContentType = 'text/plain';
379
+ }
380
+ }
381
+
382
+ /**
383
+ * Sets Mailer to send message using SMTP.
384
+ * @return void
385
+ */
386
+ public function IsSMTP() {
387
+ $this->Mailer = 'smtp';
388
+ }
389
+
390
+ /**
391
+ * Sets Mailer to send message using PHP mail() function.
392
+ * @return void
393
+ */
394
+ public function IsMail() {
395
+ $this->Mailer = 'mail';
396
+ }
397
+
398
+ /**
399
+ * Sets Mailer to send message using the $Sendmail program.
400
+ * @return void
401
+ */
402
+ public function IsSendmail() {
403
+ if (!stristr(ini_get('sendmail_path'), 'sendmail')) {
404
+ $this->Sendmail = '/var/qmail/bin/sendmail';
405
+ }
406
+ $this->Mailer = 'sendmail';
407
+ }
408
+
409
+ /**
410
+ * Sets Mailer to send message using the qmail MTA.
411
+ * @return void
412
+ */
413
+ public function IsQmail() {
414
+ if (stristr(ini_get('sendmail_path'), 'qmail')) {
415
+ $this->Sendmail = '/var/qmail/bin/sendmail';
416
+ }
417
+ $this->Mailer = 'sendmail';
418
+ }
419
+
420
+ /////////////////////////////////////////////////
421
+ // METHODS, RECIPIENTS
422
+ /////////////////////////////////////////////////
423
+
424
+ /**
425
+ * Adds a "To" address.
426
+ * @param string $address
427
+ * @param string $name
428
+ * @return boolean true on success, false if address already used
429
+ */
430
+ public function AddAddress($address, $name = '') {
431
+ return $this->AddAnAddress('to', $address, $name);
432
+ }
433
+
434
+ /**
435
+ * Adds a "Cc" address.
436
+ * Note: this function works with the SMTP mailer on win32, not with the "mail" mailer.
437
+ * @param string $address
438
+ * @param string $name
439
+ * @return boolean true on success, false if address already used
440
+ */
441
+ public function AddCC($address, $name = '') {
442
+ return $this->AddAnAddress('cc', $address, $name);
443
+ }
444
+
445
+ /**
446
+ * Adds a "Bcc" address.
447
+ * Note: this function works with the SMTP mailer on win32, not with the "mail" mailer.
448
+ * @param string $address
449
+ * @param string $name
450
+ * @return boolean true on success, false if address already used
451
+ */
452
+ public function AddBCC($address, $name = '') {
453
+ return $this->AddAnAddress('bcc', $address, $name);
454
+ }
455
+
456
+ /**
457
+ * Adds a "Reply-to" address.
458
+ * @param string $address
459
+ * @param string $name
460
+ * @return boolean
461
+ */
462
+ public function AddReplyTo($address, $name = '') {
463
+ return $this->AddAnAddress('ReplyTo', $address, $name);
464
+ }
465
+
466
+ /**
467
+ * Adds an address to one of the recipient arrays
468
+ * Addresses that have been added already return false, but do not throw exceptions
469
+ * @param string $kind One of 'to', 'cc', 'bcc', 'ReplyTo'
470
+ * @param string $address The email address to send to
471
+ * @param string $name
472
+ * @return boolean true on success, false if address already used or invalid in some way
473
+ * @access protected
474
+ */
475
+ protected function AddAnAddress($kind, $address, $name = '') {
476
+ if (!preg_match('/^(to|cc|bcc|ReplyTo)$/', $kind)) {
477
+ $this->SetError($this->Lang('Invalid recipient array').': '.$kind);
478
+ if ($this->exceptions) {
479
+ throw new EM_phpmailerException('Invalid recipient array: ' . $kind);
480
+ }
481
+ echo $this->Lang('Invalid recipient array').': '.$kind;
482
+ return false;
483
+ }
484
+ $address = trim($address);
485
+ $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim
486
+ if (!self::ValidateAddress($address)) {
487
+ $this->SetError($this->Lang('invalid_address').': '. $address);
488
+ if ($this->exceptions) {
489
+ throw new EM_phpmailerException($this->Lang('invalid_address').': '.$address);
490
+ }
491
+ echo $this->Lang('invalid_address').': '.$address;
492
+ return false;
493
+ }
494
+ if ($kind != 'ReplyTo') {
495
+ if (!isset($this->all_recipients[strtolower($address)])) {
496
+ array_push($this->$kind, array($address, $name));
497
+ $this->all_recipients[strtolower($address)] = true;
498
+ return true;
499
+ }
500
+ } else {
501
+ if (!array_key_exists(strtolower($address), $this->ReplyTo)) {
502
+ $this->ReplyTo[strtolower($address)] = array($address, $name);
503
+ return true;
504
+ }
505
+ }
506
+ return false;
507
+ }
508
 
509
+ /**
510
+ * Set the From and FromName properties
511
+ * @param string $address
512
+ * @param string $name
513
+ * @return boolean
514
+ */
515
+ public function SetFrom($address, $name = '', $auto = 1) {
516
+ $address = trim($address);
517
+ $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim
518
+ if (!self::ValidateAddress($address)) {
519
+ $this->SetError($this->Lang('invalid_address').': '. $address);
520
+ if ($this->exceptions) {
521
+ throw new EM_phpmailerException($this->Lang('invalid_address').': '.$address);
522
+ }
523
+ echo $this->Lang('invalid_address').': '.$address;
524
+ return false;
525
+ }
526
+ $this->From = $address;
527
+ $this->FromName = $name;
528
+ if ($auto) {
529
+ if (empty($this->ReplyTo)) {
530
+ $this->AddAnAddress('ReplyTo', $address, $name);
531
+ }
532
+ if (empty($this->Sender)) {
533
+ $this->Sender = $address;
534
+ }
535
+ }
536
+ return true;
537
+ }
538
+
539
+ /**
540
+ * Check that a string looks roughly like an email address should
541
+ * Static so it can be used without instantiation
542
+ * Tries to use PHP built-in validator in the filter extension (from PHP 5.2), falls back to a reasonably competent regex validator
543
+ * Conforms approximately to RFC2822
544
+ * @link http://www.hexillion.com/samples/#Regex Original pattern found here
545
+ * @param string $address The email address to check
546
+ * @return boolean
547
+ * @static
548
+ * @access public
549
+ */
550
+ public static function ValidateAddress($address) {
551
+ if (function_exists('filter_var')) { //Introduced in PHP 5.2
552
+ if(filter_var($address, FILTER_VALIDATE_EMAIL) === FALSE) {
553
+ return false;
554
+ } else {
555
+ return true;
556
+ }
557
+ } else {
558
+ return preg_match('/^(?:[\w\!\#\$\%\&\'\*\+\-\/\=\?\^\`\{\|\}\~]+\.)*[\w\!\#\$\%\&\'\*\+\-\/\=\?\^\`\{\|\}\~]+@(?:(?:(?:[a-zA-Z0-9_](?:[a-zA-Z0-9_\-](?!\.)){0,61}[a-zA-Z0-9_-]?\.)+[a-zA-Z0-9_](?:[a-zA-Z0-9_\-](?!$)){0,61}[a-zA-Z0-9_]?)|(?:\[(?:(?:[01]?\d{1,2}|2[0-4]\d|25[0-5])\.){3}(?:[01]?\d{1,2}|2[0-4]\d|25[0-5])\]))$/', $address);
559
+ }
560
+ }
561
+
562
+ /////////////////////////////////////////////////
563
+ // METHODS, MAIL SENDING
564
+ /////////////////////////////////////////////////
565
+
566
+ /**
567
+ * Creates message and assigns Mailer. If the message is
568
+ * not sent successfully then it returns false. Use the ErrorInfo
569
+ * variable to view description of the error.
570
+ * @return bool
571
+ */
572
+ public function Send() {
573
+ try {
574
+ if(!$this->PreSend()) return false;
575
+ return $this->PostSend();
576
+ } catch (EM_phpmailerException $e) {
577
+ $this->SetError($e->getMessage());
578
+ if ($this->exceptions) {
579
+ throw $e;
580
+ }
581
+ return false;
582
+ }
583
+ }
584
 
585
+ protected function PreSend() {
586
+ try {
587
+ if ((count($this->to) + count($this->cc) + count($this->bcc)) < 1) {
588
+ throw new EM_phpmailerException($this->Lang('provide_address'), self::STOP_CRITICAL);
589
+ }
590
+
591
+ // Set whether the message is multipart/alternative
592
+ if(!empty($this->AltBody)) {
593
+ $this->ContentType = 'multipart/alternative';
594
+ }
595
+
596
+ $this->error_count = 0; // reset errors
597
+ $this->SetMessageType();
598
+ //Refuse to send an empty message
599
+ if (empty($this->Body)) {
600
+ throw new EM_phpmailerException($this->Lang('empty_message'), self::STOP_CRITICAL);
601
+ }
602
+
603
+ $this->MIMEHeader = $this->CreateHeader();
604
+ $this->MIMEBody = $this->CreateBody();
605
+
606
+
607
+ // digitally sign with DKIM if enabled
608
+ if ($this->DKIM_domain && $this->DKIM_private) {
609
+ $header_dkim = $this->DKIM_Add($this->MIMEHeader, $this->EncodeHeader($this->SecureHeader($this->Subject)), $this->MIMEBody);
610
+ $this->MIMEHeader = str_replace("\r\n", "\n", $header_dkim) . $this->MIMEHeader;
611
+ }
612
+
613
+ return true;
614
+ } catch (EM_phpmailerException $e) {
615
+ $this->SetError($e->getMessage());
616
+ if ($this->exceptions) {
617
+ throw $e;
618
+ }
619
+ return false;
620
+ }
621
+ }
622
+
623
+ protected function PostSend() {
624
+ try {
625
+ // Choose the mailer and send through it
626
+ switch($this->Mailer) {
627
+ case 'sendmail':
628
+ return $this->SendmailSend($this->MIMEHeader, $this->MIMEBody);
629
+ case 'smtp':
630
+ return $this->SmtpSend($this->MIMEHeader, $this->MIMEBody);
631
+ default:
632
+ return $this->MailSend($this->MIMEHeader, $this->MIMEBody);
633
+ }
634
+
635
+ } catch (EM_phpmailerException $e) {
636
+ $this->SetError($e->getMessage());
637
+ if ($this->exceptions) {
638
+ throw $e;
639
+ }
640
+ echo $e->getMessage()."\n";
641
+ return false;
642
+ }
643
+ }
644
+
645
+ /**
646
+ * Sends mail using the $Sendmail program.
647
+ * @param string $header The message headers
648
+ * @param string $body The message body
649
+ * @access protected
650
+ * @return bool
651
+ */
652
+ protected function SendmailSend($header, $body) {
653
+ if ($this->Sender != '') {
654
+ $sendmail = sprintf("%s -oi -f %s -t", escapeshellcmd($this->Sendmail), escapeshellarg($this->Sender));
655
+ } else {
656
+ $sendmail = sprintf("%s -oi -t", escapeshellcmd($this->Sendmail));
657
+ }
658
+ if ($this->SingleTo === true) {
659
+ foreach ($this->SingleToArray as $key => $val) {
660
+ if(!@$mail = popen($sendmail, 'w')) {
661
+ throw new EM_phpmailerException($this->Lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
662
+ }
663
+ fputs($mail, "To: " . $val . "\n");
664
  fputs($mail, $header);
665
  fputs($mail, $body);
666
+ $result = pclose($mail);
667
+ // implement call back function if it exists
668
+ $isSent = ($result == 0) ? 1 : 0;
669
+ $this->doCallback($isSent, $val, $this->cc, $this->bcc, $this->Subject, $body);
670
+ if($result != 0) {
671
+ throw new EM_phpmailerException($this->Lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
672
  }
673
+ }
674
+ } else {
675
+ if(!@$mail = popen($sendmail, 'w')) {
676
+ throw new EM_phpmailerException($this->Lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
677
+ }
678
+ fputs($mail, $header);
679
+ fputs($mail, $body);
680
+ $result = pclose($mail);
681
+ // implement call back function if it exists
682
+ $isSent = ($result == 0) ? 1 : 0;
683
+ $this->doCallback($isSent, $this->to, $this->cc, $this->bcc, $this->Subject, $body);
684
+ if($result != 0) {
685
+ throw new EM_phpmailerException($this->Lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
686
+ }
687
+ }
688
+ return true;
689
+ }
690
+
691
+ /**
692
+ * Sends mail using the PHP mail() function.
693
+ * @param string $header The message headers
694
+ * @param string $body The message body
695
+ * @access protected
696
+ * @return bool
697
+ */
698
+ protected function MailSend($header, $body) {
699
+ $toArr = array();
700
+ foreach($this->to as $t) {
701
+ $toArr[] = $this->AddrFormat($t);
702
  }
703
+ $to = implode(', ', $toArr);
704
 
705
+ if (empty($this->Sender)) {
706
+ $params = "-oi -f %s";
707
+ } else {
708
+ $params = sprintf("-oi -f %s", $this->Sender);
709
+ }
710
+ if ($this->Sender != '' and !ini_get('safe_mode')) {
711
+ $old_from = ini_get('sendmail_from');
712
+ ini_set('sendmail_from', $this->Sender);
713
+ if ($this->SingleTo === true && count($toArr) > 1) {
714
+ foreach ($toArr as $key => $val) {
715
+ $rt = @mail($val, $this->EncodeHeader($this->SecureHeader($this->Subject)), $body, $header, $params);
716
+ // implement call back function if it exists
717
+ $isSent = ($rt == 1) ? 1 : 0;
718
+ $this->doCallback($isSent, $val, $this->cc, $this->bcc, $this->Subject, $body);
719
  }
720
+ } else {
721
+ $rt = @mail($to, $this->EncodeHeader($this->SecureHeader($this->Subject)), $body, $header, $params);
722
+ // implement call back function if it exists
723
+ $isSent = ($rt == 1) ? 1 : 0;
724
+ $this->doCallback($isSent, $to, $this->cc, $this->bcc, $this->Subject, $body);
725
+ }
726
+ } else {
727
+ if ($this->SingleTo === true && count($toArr) > 1) {
728
+ foreach ($toArr as $key => $val) {
729
+ $rt = @mail($val, $this->EncodeHeader($this->SecureHeader($this->Subject)), $body, $header, $params);
730
+ // implement call back function if it exists
731
+ $isSent = ($rt == 1) ? 1 : 0;
732
+ $this->doCallback($isSent, $val, $this->cc, $this->bcc, $this->Subject, $body);
733
  }
734
+ } else {
735
+ $rt = @mail($to, $this->EncodeHeader($this->SecureHeader($this->Subject)), $body, $header);
736
+ // implement call back function if it exists
737
+ $isSent = ($rt == 1) ? 1 : 0;
738
+ $this->doCallback($isSent, $to, $this->cc, $this->bcc, $this->Subject, $body);
739
+ }
740
+ }
741
+ if (isset($old_from)) {
742
+ ini_set('sendmail_from', $old_from);
743
+ }
744
+ if(!$rt) {
745
+ throw new EM_phpmailerException($this->Lang('instantiate'), self::STOP_CRITICAL);
746
+ }
747
+ return true;
748
+ }
749
+
750
+ /**
751
+ * Sends mail via SMTP using PhpSMTP
752
+ * Returns false if there is a bad MAIL FROM, RCPT, or DATA input.
753
+ * @param string $header The message headers
754
+ * @param string $body The message body
755
+ * @uses SMTP
756
+ * @access protected
757
+ * @return bool
758
+ */
759
+ protected function SmtpSend($header, $body) {
760
+ require_once $this->PluginDir . 'class.smtp.php';
761
+ $bad_rcpt = array();
762
+
763
+ if(!$this->SmtpConnect()) {
764
+ throw new EM_phpmailerException($this->Lang('smtp_connect_failed'), self::STOP_CRITICAL);
765
+ }
766
+ $smtp_from = ($this->Sender == '') ? $this->From : $this->Sender;
767
+ if(!$this->smtp->Mail($smtp_from)) {
768
+ throw new EM_phpmailerException($this->Lang('from_failed') . $smtp_from, self::STOP_CRITICAL);
769
+ }
770
 
771
+ // Attempt to send attach all recipients
772
+ foreach($this->to as $to) {
773
+ if (!$this->smtp->Recipient($to[0])) {
774
+ $bad_rcpt[] = $to[0];
775
+ // implement call back function if it exists
776
+ $isSent = 0;
777
+ $this->doCallback($isSent, $to[0], '', '', $this->Subject, $body);
778
+ } else {
779
+ // implement call back function if it exists
780
+ $isSent = 1;
781
+ $this->doCallback($isSent, $to[0], '', '', $this->Subject, $body);
782
+ }
783
+ }
784
+ foreach($this->cc as $cc) {
785
+ if (!$this->smtp->Recipient($cc[0])) {
786
+ $bad_rcpt[] = $cc[0];
787
+ // implement call back function if it exists
788
+ $isSent = 0;
789
+ $this->doCallback($isSent, '', $cc[0], '', $this->Subject, $body);
790
+ } else {
791
+ // implement call back function if it exists
792
+ $isSent = 1;
793
+ $this->doCallback($isSent, '', $cc[0], '', $this->Subject, $body);
794
+ }
795
+ }
796
+ foreach($this->bcc as $bcc) {
797
+ if (!$this->smtp->Recipient($bcc[0])) {
798
+ $bad_rcpt[] = $bcc[0];
799
+ // implement call back function if it exists
800
+ $isSent = 0;
801
+ $this->doCallback($isSent, '', '', $bcc[0], $this->Subject, $body);
802
+ } else {
803
+ // implement call back function if it exists
804
+ $isSent = 1;
805
+ $this->doCallback($isSent, '', '', $bcc[0], $this->Subject, $body);
806
+ }
807
+ }
808
 
 
 
 
 
 
809
 
810
+ if (count($bad_rcpt) > 0 ) { //Create error message for any bad addresses
811
+ $badaddresses = implode(', ', $bad_rcpt);
812
+ throw new EM_phpmailerException($this->Lang('recipients_failed') . $badaddresses);
813
+ }
814
+ if(!$this->smtp->Data($header . $body)) {
815
+ throw new EM_phpmailerException($this->Lang('data_not_accepted'), self::STOP_CRITICAL);
816
+ }
817
+ if($this->SMTPKeepAlive == true) {
818
+ $this->smtp->Reset();
819
+ }
820
+ return true;
821
+ }
822
+
823
+ /**
824
+ * Initiates a connection to an SMTP server.
825
+ * Returns false if the operation failed.
826
+ * @uses SMTP
827
+ * @access public
828
+ * @return bool
829
+ */
830
+ public function SmtpConnect() {
831
+ if(is_null($this->smtp)) {
832
+ $this->smtp = new EM_SMTP();
833
  }
834
 
835
+ $this->smtp->do_debug = $this->SMTPDebug;
836
+ $hosts = explode(';', $this->Host);
837
+ $index = 0;
838
+ $connection = $this->smtp->Connected();
839
+
840
+ // Retry while there is no connection
841
+ try {
842
+ while($index < count($hosts) && !$connection) {
843
+ $hostinfo = array();
844
+ if (preg_match('/^(.+):([0-9]+)$/', $hosts[$index], $hostinfo)) {
845
+ $host = $hostinfo[1];
846
+ $port = $hostinfo[2];
847
+ } else {
848
+ $host = $hosts[$index];
849
+ $port = $this->Port;
 
 
 
 
 
 
 
850
  }
851
 
852
+ $tls = ($this->SMTPSecure == 'tls');
853
+ $ssl = ($this->SMTPSecure == 'ssl');
854
+
855
+ if ($this->smtp->Connect(($ssl ? 'ssl://':'').$host, $port, $this->Timeout)) {
856
+
857
+ $hello = ($this->Helo != '' ? $this->Helo : $this->ServerHostname());
858
+ $this->smtp->Hello($hello);
 
 
 
 
 
 
 
 
 
859
 
860
+ if ($tls) {
861
+ if (!$this->smtp->StartTLS()) {
862
+ throw new EM_phpmailerException($this->Lang('tls'));
 
 
 
863
  }
 
 
 
 
 
864
 
865
+ //We must resend HELO after tls negotiation
866
+ $this->smtp->Hello($hello);
867
+ }
 
 
 
 
 
 
 
868
 
869
+ $connection = true;
870
+ if ($this->SMTPAuth) {
871
+ if (!$this->smtp->Authenticate($this->Username, $this->Password)) {
872
+ throw new EM_phpmailerException($this->Lang('authenticate'));
873
+ }
874
+ }
875
+ }
876
+ $index++;
877
+ if (!$connection) {
878
+ throw new EM_phpmailerException($this->Lang('connect_host'));
879
+ }
880
+ }
881
+ } catch (EM_phpmailerException $e) {
882
+ $this->smtp->Reset();
883
+ throw $e;
884
+ }
885
+ return true;
886
+ }
887
+
888
+ /**
889
+ * Closes the active SMTP session if one exists.
890
+ * @return void
891
+ */
892
+ public function SmtpClose() {
893
+ if(!is_null($this->smtp)) {
894
+ if($this->smtp->Connected()) {
895
+ $this->smtp->Quit();
896
+ $this->smtp->Close();
897
+ }
898
+ }
899
+ }
900
+
901
+ /**
902
+ * Sets the language for all class error messages.
903
+ * Returns false if it cannot load the language file. The default language is English.
904
+ * @param string $langcode ISO 639-1 2-character language code (e.g. Portuguese: "br")
905
+ * @param string $lang_path Path to the language file directory
906
+ * @access public
907
+ */
908
+ function SetLanguage($langcode = 'en', $lang_path = 'language/') {
909
+ //Define full set of translatable strings
910
+ $PHPMAILER_LANG = array(
911
+ 'provide_address' => 'You must provide at least one recipient email address.',
912
+ 'mailer_not_supported' => ' mailer is not supported.',
913
+ 'execute' => 'Could not execute: ',
914
+ 'instantiate' => 'Could not instantiate mail function.',
915
+ 'authenticate' => 'SMTP Error: Could not authenticate.',
916
+ 'from_failed' => 'The following From address failed: ',
917
+ 'recipients_failed' => 'SMTP Error: The following recipients failed: ',
918
+ 'data_not_accepted' => 'SMTP Error: Data not accepted.',
919
+ 'connect_host' => 'SMTP Error: Could not connect to SMTP host.',
920
+ 'file_access' => 'Could not access file: ',
921
+ 'file_open' => 'File Error: Could not open file: ',
922
+ 'encoding' => 'Unknown encoding: ',
923
+ 'signing' => 'Signing Error: ',
924
+ 'smtp_error' => 'SMTP server error: ',
925
+ 'empty_message' => 'Message body empty',
926
+ 'invalid_address' => 'Invalid address',
927
+ 'variable_set' => 'Cannot set or reset variable: '
928
+ );
929
+ //Overwrite language-specific strings. This way we'll never have missing translations - no more "language string failed to load"!
930
+ $l = true;
931
+ if ($langcode != 'en') { //There is no English translation file
932
+ $l = @include $lang_path.'phpmailer.lang-'.$langcode.'.php';
933
+ }
934
+ $this->language = $PHPMAILER_LANG;
935
+ return ($l == true); //Returns false if language not found
936
+ }
937
+
938
+ /**
939
+ * Return the current array of language strings
940
+ * @return array
941
+ */
942
+ public function GetTranslations() {
943
+ return $this->language;
944
+ }
945
+
946
+ /////////////////////////////////////////////////
947
+ // METHODS, MESSAGE CREATION
948
+ /////////////////////////////////////////////////
949
+
950
+ /**
951
+ * Creates recipient headers.
952
+ * @access public
953
+ * @return string
954
+ */
955
+ public function AddrAppend($type, $addr) {
956
+ $addr_str = $type . ': ';
957
+ $addresses = array();
958
+ foreach ($addr as $a) {
959
+ $addresses[] = $this->AddrFormat($a);
960
+ }
961
+ $addr_str .= implode(', ', $addresses);
962
+ $addr_str .= $this->LE;
963
+
964
+ return $addr_str;
965
+ }
966
+
967
+ /**
968
+ * Formats an address correctly.
969
+ * @access public
970
+ * @return string
971
+ */
972
+ public function AddrFormat($addr) {
973
+ if (empty($addr[1])) {
974
+ return $this->SecureHeader($addr[0]);
975
+ } else {
976
+ return $this->EncodeHeader($this->SecureHeader($addr[1]), 'phrase') . " <" . $this->SecureHeader($addr[0]) . ">";
977
+ }
978
+ }
979
+
980
+ /**
981
+ * Wraps message for use with mailers that do not
982
+ * automatically perform wrapping and for quoted-printable.
983
+ * Original written by philippe.
984
+ * @param string $message The message to wrap
985
+ * @param integer $length The line length to wrap to
986
+ * @param boolean $qp_mode Whether to run in Quoted-Printable mode
987
+ * @access public
988
+ * @return string
989
+ */
990
+ public function WrapText($message, $length, $qp_mode = false) {
991
+ $soft_break = ($qp_mode) ? sprintf(" =%s", $this->LE) : $this->LE;
992
+ // If utf-8 encoding is used, we will need to make sure we don't
993
+ // split multibyte characters when we wrap
994
+ $is_utf8 = (strtolower($this->CharSet) == "utf-8");
995
+
996
+ $message = $this->FixEOL($message);
997
+ if (substr($message, -1) == $this->LE) {
998
+ $message = substr($message, 0, -1);
999
  }
1000
 
1001
+ $line = explode($this->LE, $message);
1002
+ $message = '';
1003
+ for ($i = 0 ;$i < count($line); $i++) {
1004
+ $line_part = explode(' ', $line[$i]);
1005
+ $buf = '';
1006
+ for ($e = 0; $e<count($line_part); $e++) {
1007
+ $word = $line_part[$e];
1008
+ if ($qp_mode and (strlen($word) > $length)) {
1009
+ $space_left = $length - strlen($buf) - 1;
1010
+ if ($e != 0) {
1011
+ if ($space_left > 20) {
1012
+ $len = $space_left;
1013
+ if ($is_utf8) {
1014
+ $len = $this->UTF8CharBoundary($word, $len);
1015
+ } elseif (substr($word, $len - 1, 1) == "=") {
1016
+ $len--;
1017
+ } elseif (substr($word, $len - 2, 1) == "=") {
1018
+ $len -= 2;
1019
+ }
1020
+ $part = substr($word, 0, $len);
1021
+ $word = substr($word, $len);
1022
+ $buf .= ' ' . $part;
1023
+ $message .= $buf . sprintf("=%s", $this->LE);
1024
+ } else {
1025
+ $message .= $buf . $soft_break;
1026
  }
1027
+ $buf = '';
1028
+ }
1029
+ while (strlen($word) > 0) {
1030
+ $len = $length;
1031
+ if ($is_utf8) {
1032
+ $len = $this->UTF8CharBoundary($word, $len);
1033
+ } elseif (substr($word, $len - 1, 1) == "=") {
1034
+ $len--;
1035
+ } elseif (substr($word, $len - 2, 1) == "=") {
1036
+ $len -= 2;
1037
  }
1038
+ $part = substr($word, 0, $len);
1039
+ $word = substr($word, $len);
1040
 
1041
+ if (strlen($word) > 0) {
1042
+ $message .= $part . sprintf("=%s", $this->LE);
1043
+ } else {
1044
+ $buf = $part;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1045
  }
1046
+ }
1047
+ } else {
1048
+ $buf_o = $buf;
1049
+ $buf .= ($e == 0) ? $word : (' ' . $word);
1050
+
1051
+ if (strlen($buf) > $length and $buf_o != '') {
1052
+ $message .= $buf_o . $soft_break;
1053
+ $buf = $word;
1054
+ }
1055
  }
1056
+ }
1057
+ $message .= $buf . $this->LE;
1058
  }
1059
 
1060
+ return $message;
1061
+ }
1062
+
1063
+ /**
1064
+ * Finds last character boundary prior to maxLength in a utf-8
1065
+ * quoted (printable) encoded string.
1066
+ * Original written by Colin Brown.
1067
+ * @access public
1068
+ * @param string $encodedText utf-8 QP text
1069
+ * @param int $maxLength find last character boundary prior to this length
1070
+ * @return int
1071
+ */
1072
+ public function UTF8CharBoundary($encodedText, $maxLength) {
1073
+ $foundSplitPos = false;
1074
+ $lookBack = 3;
1075
+ while (!$foundSplitPos) {
1076
+ $lastChunk = substr($encodedText, $maxLength - $lookBack, $lookBack);
1077
+ $encodedCharPos = strpos($lastChunk, "=");
1078
+ if ($encodedCharPos !== false) {
1079
+ // Found start of encoded character byte within $lookBack block.
1080
+ // Check the encoded byte value (the 2 chars after the '=')
1081
+ $hex = substr($encodedText, $maxLength - $lookBack + $encodedCharPos + 1, 2);
1082
+ $dec = hexdec($hex);
1083
+ if ($dec < 128) { // Single byte character.
1084
+ // If the encoded char was found at pos 0, it will fit
1085
+ // otherwise reduce maxLength to start of the encoded char
1086
+ $maxLength = ($encodedCharPos == 0) ? $maxLength :
1087
+ $maxLength - ($lookBack - $encodedCharPos);
1088
+ $foundSplitPos = true;
1089
+ } elseif ($dec >= 192) { // First byte of a multi byte character
1090
+ // Reduce maxLength to split at start of character
1091
+ $maxLength = $maxLength - ($lookBack - $encodedCharPos);
1092
+ $foundSplitPos = true;
1093
+ } elseif ($dec < 192) { // Middle byte of a multi byte character, look further back
1094
+ $lookBack += 3;
1095
  }
1096
+ } else {
1097
+ // No encoded character found
1098
+ $foundSplitPos = true;
1099
+ }
1100
+ }
1101
+ return $maxLength;
1102
+ }
1103
+
1104
+
1105
+ /**
1106
+ * Set the body wrapping.
1107
+ * @access public
1108
+ * @return void
1109
+ */
1110
+ public function SetWordWrap() {
1111
+ if($this->WordWrap < 1) {
1112
+ return;
1113
+ }
1114
+
1115
+ switch($this->message_type) {
1116
+ case 'alt':
1117
+ case 'alt_inline':
1118
+ case 'alt_attach':
1119
+ case 'alt_inline_attach':
1120
+ $this->AltBody = $this->WrapText($this->AltBody, $this->WordWrap);
1121
+ break;
1122
+ default:
1123
+ $this->Body = $this->WrapText($this->Body, $this->WordWrap);
1124
+ break;
1125
+ }
1126
+ }
1127
+
1128
+ /**
1129
+ * Assembles message header.
1130
+ * @access public
1131
+ * @return string The assembled header
1132
+ */
1133
+ public function CreateHeader() {
1134
+ $result = '';
1135
+
1136
+ // Set the boundaries
1137
+ $uniq_id = md5(uniqid(time()));
1138
+ $this->boundary[1] = 'b1_' . $uniq_id;
1139
+ $this->boundary[2] = 'b2_' . $uniq_id;
1140
+ $this->boundary[3] = 'b3_' . $uniq_id;
1141
+
1142
+ $result .= $this->HeaderLine('Date', self::RFCDate());
1143
+ if($this->Sender == '') {
1144
+ $result .= $this->HeaderLine('Return-Path', trim($this->From));
1145
+ } else {
1146
+ $result .= $this->HeaderLine('Return-Path', trim($this->Sender));
1147
  }
1148
 
1149
+ // To be created automatically by mail()
1150
+ if($this->Mailer != 'mail') {
1151
+ if ($this->SingleTo === true) {
1152
+ foreach($this->to as $t) {
1153
+ $this->SingleToArray[] = $this->AddrFormat($t);
 
 
 
 
 
 
 
 
 
 
 
1154
  }
1155
+ } else {
1156
+ if(count($this->to) > 0) {
1157
+ $result .= $this->AddrAppend('To', $this->to);
1158
+ } elseif (count($this->cc) == 0) {
1159
+ $result .= $this->HeaderLine('To', 'undisclosed-recipients:;');
 
 
 
 
 
 
 
 
 
 
 
 
1160
  }
1161
+ }
1162
+ }
1163
 
1164
+ $from = array();
1165
+ $from[0][0] = trim($this->From);
1166
+ $from[0][1] = $this->FromName;
1167
+ $result .= $this->AddrAppend('From', $from);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1168
 
1169
+ // sendmail and mail() extract Cc from the header before sending
1170
+ if(count($this->cc) > 0) {
1171
+ $result .= $this->AddrAppend('Cc', $this->cc);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1172
  }
1173
 
1174
+ // sendmail and mail() extract Bcc from the header before sending
1175
+ if((($this->Mailer == 'sendmail') || ($this->Mailer == 'mail')) && (count($this->bcc) > 0)) {
1176
+ $result .= $this->AddrAppend('Bcc', $this->bcc);
1177
+ }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1178
 
1179
+ if(count($this->ReplyTo) > 0) {
1180
+ $result .= $this->AddrAppend('Reply-to', $this->ReplyTo);
1181
+ }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1182
 
1183
+ // mail() sets the subject itself
1184
+ if($this->Mailer != 'mail') {
1185
+ $result .= $this->HeaderLine('Subject', $this->EncodeHeader($this->SecureHeader($this->Subject)));
1186
+ }
1187
+
1188
+ if($this->MessageID != '') {
1189
+ $result .= $this->HeaderLine('Message-ID', $this->MessageID);
1190
+ } else {
1191
+ $result .= sprintf("Message-ID: <%s@%s>%s", $uniq_id, $this->ServerHostname(), $this->LE);
1192
+ }
1193
+ $result .= $this->HeaderLine('X-Priority', $this->Priority);
1194
+ if($this->XMailer) {
1195
+ $result .= $this->HeaderLine('X-Mailer', $this->XMailer);
1196
+ } else {
1197
+ $result .= $this->HeaderLine('X-Mailer', 'PHPMailer '.$this->Version.' (http://code.google.com/a/apache-extras.org/p/phpmailer/)');
1198
+ }
1199
+
1200
+ if($this->ConfirmReadingTo != '') {
1201
+ $result .= $this->HeaderLine('Disposition-Notification-To', '<' . trim($this->ConfirmReadingTo) . '>');
1202
+ }
1203
+
1204
+ // Add custom headers
1205
+ for($index = 0; $index < count($this->CustomHeader); $index++) {
1206
+ $result .= $this->HeaderLine(trim($this->CustomHeader[$index][0]), $this->EncodeHeader(trim($this->CustomHeader[$index][1])));
1207
+ }
1208
+ if (!$this->sign_key_file) {
1209
+ $result .= $this->HeaderLine('MIME-Version', '1.0');
1210
+ $result .= $this->GetMailMIME();
1211
+ }
1212
+
1213
+ return $result;
1214
+ }
1215
+
1216
+ /**
1217
+ * Returns the message MIME.
1218
+ * @access public
1219
+ * @return string
1220
+ */
1221
+ public function GetMailMIME() {
1222
+ $result = '';
1223
+ switch($this->message_type) {
1224
+ case 'plain':
1225
+ $result .= $this->HeaderLine('Content-Transfer-Encoding', $this->Encoding);
1226
+ $result .= $this->TextLine('Content-Type: '.$this->ContentType.'; charset="'.$this->CharSet.'"');
1227
+ break;
1228
+ case 'inline':
1229
+ $result .= $this->HeaderLine('Content-Type', 'multipart/related;');
1230
+ $result .= $this->TextLine("\tboundary=\"" . $this->boundary[1] . '"');
1231
+ break;
1232
+ case 'attach':
1233
+ case 'inline_attach':
1234
+ case 'alt_attach':
1235
+ case 'alt_inline_attach':
1236
+ $result .= $this->HeaderLine('Content-Type', 'multipart/mixed;');
1237
+ $result .= $this->TextLine("\tboundary=\"" . $this->boundary[1] . '"');
1238
+ break;
1239
+ case 'alt':
1240
+ case 'alt_inline':
1241
+ $result .= $this->HeaderLine('Content-Type', 'multipart/alternative;');
1242
+ $result .= $this->TextLine("\tboundary=\"" . $this->boundary[1] . '"');
1243
+ break;
1244
+ }
1245
+
1246
+ if($this->Mailer != 'mail') {
1247
+ $result .= $this->LE.$this->LE;
1248
+ }
1249
+
1250
+ return $result;
1251
+ }
1252
+
1253
+ /**
1254
+ * Assembles the message body. Returns an empty string on failure.
1255
+ * @access public
1256
+ * @return string The assembled message body
1257
+ */
1258
+ public function CreateBody() {
1259
+ $body = '';
1260
 
1261
+ if ($this->sign_key_file) {
1262
+ $body .= $this->GetMailMIME();
1263
+ }
1264
+
1265
+ $this->SetWordWrap();
1266
+
1267
+ switch($this->message_type) {
1268
+ case 'plain':
1269
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1270
+ break;
1271
+ case 'inline':
1272
+ $body .= $this->GetBoundary($this->boundary[1], '', '', '');
1273
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1274
+ $body .= $this->LE.$this->LE;
1275
+ $body .= $this->AttachAll("inline", $this->boundary[1]);
1276
+ break;
1277
+ case 'attach':
1278
+ $body .= $this->GetBoundary($this->boundary[1], '', '', '');
1279
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1280
+ $body .= $this->LE.$this->LE;
1281
+ $body .= $this->AttachAll("attachment", $this->boundary[1]);
1282
+ break;
1283
+ case 'inline_attach':
1284
+ $body .= $this->TextLine("--" . $this->boundary[1]);
1285
+ $body .= $this->HeaderLine('Content-Type', 'multipart/related;');
1286
+ $body .= $this->TextLine("\tboundary=\"" . $this->boundary[2] . '"');
1287
+ $body .= $this->LE;
1288
+ $body .= $this->GetBoundary($this->boundary[2], '', '', '');
1289
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1290
+ $body .= $this->LE.$this->LE;
1291
+ $body .= $this->AttachAll("inline", $this->boundary[2]);
1292
+ $body .= $this->LE;
1293
+ $body .= $this->AttachAll("attachment", $this->boundary[1]);
1294
+ break;
1295
+ case 'alt':
1296
+ $body .= $this->GetBoundary($this->boundary[1], '', 'text/plain', '');
1297
+ $body .= $this->EncodeString($this->AltBody, $this->Encoding);
1298
+ $body .= $this->LE.$this->LE;
1299
+ $body .= $this->GetBoundary($this->boundary[1], '', 'text/html', '');
1300
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1301
+ $body .= $this->LE.$this->LE;
1302
+ $body .= $this->EndBoundary($this->boundary[1]);
1303
+ break;
1304
+ case 'alt_inline':
1305
+ $body .= $this->GetBoundary($this->boundary[1], '', 'text/plain', '');
1306
+ $body .= $this->EncodeString($this->AltBody, $this->Encoding);
1307
+ $body .= $this->LE.$this->LE;
1308
+ $body .= $this->TextLine("--" . $this->boundary[1]);
1309
+ $body .= $this->HeaderLine('Content-Type', 'multipart/related;');
1310
+ $body .= $this->TextLine("\tboundary=\"" . $this->boundary[2] . '"');
1311
+ $body .= $this->LE;
1312
+ $body .= $this->GetBoundary($this->boundary[2], '', 'text/html', '');
1313
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1314
+ $body .= $this->LE.$this->LE;
1315
+ $body .= $this->AttachAll("inline", $this->boundary[2]);
1316
+ $body .= $this->LE;
1317
+ $body .= $this->EndBoundary($this->boundary[1]);
1318
+ break;
1319
+ case 'alt_attach':
1320
+ $body .= $this->TextLine("--" . $this->boundary[1]);
1321
+ $body .= $this->HeaderLine('Content-Type', 'multipart/alternative;');
1322
+ $body .= $this->TextLine("\tboundary=\"" . $this->boundary[2] . '"');
1323
+ $body .= $this->LE;
1324
+ $body .= $this->GetBoundary($this->boundary[2], '', 'text/plain', '');
1325
+ $body .= $this->EncodeString($this->AltBody, $this->Encoding);
1326
+ $body .= $this->LE.$this->LE;
1327
+ $body .= $this->GetBoundary($this->boundary[2], '', 'text/html', '');
1328
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1329
+ $body .= $this->LE.$this->LE;
1330
+ $body .= $this->EndBoundary($this->boundary[2]);
1331
+ $body .= $this->LE;
1332
+ $body .= $this->AttachAll("attachment", $this->boundary[1]);
1333
+ break;
1334
+ case 'alt_inline_attach':
1335
+ $body .= $this->TextLine("--" . $this->boundary[1]);
1336
+ $body .= $this->HeaderLine('Content-Type', 'multipart/alternative;');
1337
+ $body .= $this->TextLine("\tboundary=\"" . $this->boundary[2] . '"');
1338
+ $body .= $this->LE;
1339
+ $body .= $this->GetBoundary($this->boundary[2], '', 'text/plain', '');
1340
+ $body .= $this->EncodeString($this->AltBody, $this->Encoding);
1341
+ $body .= $this->LE.$this->LE;
1342
+ $body .= $this->TextLine("--" . $this->boundary[2]);
1343
+ $body .= $this->HeaderLine('Content-Type', 'multipart/related;');
1344
+ $body .= $this->TextLine("\tboundary=\"" . $this->boundary[3] . '"');
1345
+ $body .= $this->LE;
1346
+ $body .= $this->GetBoundary($this->boundary[3], '', 'text/html', '');
1347
+ $body .= $this->EncodeString($this->Body, $this->Encoding);
1348
+ $body .= $this->LE.$this->LE;
1349
+ $body .= $this->AttachAll("inline", $this->boundary[3]);
1350
+ $body .= $this->LE;
1351
+ $body .= $this->EndBoundary($this->boundary[2]);
1352
+ $body .= $this->LE;
1353
+ $body .= $this->AttachAll("attachment", $this->boundary[1]);
1354
+ break;
1355
+ }
1356
+
1357
+ if ($this->IsError()) {
1358
+ $body = '';
1359
+ } elseif ($this->sign_key_file) {
1360
+ try {
1361
+ $file = tempnam('', 'mail');
1362
+ file_put_contents($file, $body); //TODO check this worked
1363
+ $signed = tempnam("", "signed");
1364
+ if (@openssl_pkcs7_sign($file, $signed, "file://".$this->sign_cert_file, array("file://".$this->sign_key_file, $this->sign_key_pass), NULL)) {
1365
+ @unlink($file);
1366
+ @unlink($signed);
1367
+ $body = file_get_contents($signed);
1368
+ } else {
1369
+ @unlink($file);
1370
+ @unlink($signed);
1371
+ throw new EM_phpmailerException($this->Lang("signing").openssl_error_string());
1372
  }
1373
+ } catch (EM_phpmailerException $e) {
1374
+ $body = '';
1375
+ if ($this->exceptions) {
1376
+ throw $e;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1377
  }
1378
+ }
1379
  }
1380
 
1381
+ return $body;
1382
+ }
1383
+
1384
+ /**
1385
+ * Returns the start of a message boundary.
1386
+ * @access protected
1387
+ * @return string
1388
+ */
1389
+ protected function GetBoundary($boundary, $charSet, $contentType, $encoding) {
1390
+ $result = '';
1391
+ if($charSet == '') {
1392
+ $charSet = $this->CharSet;
1393
+ }
1394
+ if($contentType == '') {
1395
+ $contentType = $this->ContentType;
1396
+ }
1397
+ if($encoding == '') {
1398
+ $encoding = $this->Encoding;
1399
+ }
1400
+ $result .= $this->TextLine('--' . $boundary);
1401
+ $result .= sprintf("Content-Type: %s; charset=\"%s\"", $contentType, $charSet);
1402
+ $result .= $this->LE;
1403
+ $result .= $this->HeaderLine('Content-Transfer-Encoding', $encoding);
1404
+ $result .= $this->LE;
1405
+
1406
+ return $result;
1407
+ }
1408
+
1409
+ /**
1410
+ * Returns the end of a message boundary.
1411
+ * @access protected
1412
+ * @return string
1413
+ */
1414
+ protected function EndBoundary($boundary) {
1415
+ return $this->LE . '--' . $boundary . '--' . $this->LE;
1416
+ }
1417
+
1418
+ /**
1419
+ * Sets the message type.
1420
+ * @access protected
1421
+ * @return void
1422
+ */
1423
+ protected function SetMessageType() {
1424
+ $this->message_type = array();
1425
+ if($this->AlternativeExists()) $this->message_type[] = "alt";
1426
+ if($this->InlineImageExists()) $this->message_type[] = "inline";
1427
+ if($this->AttachmentExists()) $this->message_type[] = "attach";
1428
+ $this->message_type = implode("_", $this->message_type);
1429
+ if($this->message_type == "") $this->message_type = "plain";
1430
+ }
1431
+
1432
+ /**
1433
+ * Returns a formatted header line.
1434
+ * @access public
1435
+ * @return string
1436
+ */
1437
+ public function HeaderLine($name, $value) {
1438
+ return $name . ': ' . $value . $this->LE;
1439
+ }
1440
+
1441
+ /**
1442
+ * Returns a formatted mail line.
1443
+ * @access public
1444
+ * @return string
1445
+ */
1446
+ public function TextLine($value) {
1447
+ return $value . $this->LE;
1448
+ }
1449
+
1450
+ /////////////////////////////////////////////////
1451
+ // CLASS METHODS, ATTACHMENTS
1452
+ /////////////////////////////////////////////////
1453
+
1454
+ /**
1455
+ * Adds an attachment from a path on the filesystem.
1456
+ * Returns false if the file could not be found
1457
+ * or accessed.
1458
+ * @param string $path Path to the attachment.
1459
+ * @param string $name Overrides the attachment name.
1460
+ * @param string $encoding File encoding (see $Encoding).
1461
+ * @param string $type File extension (MIME) type.
1462
+ * @return bool
1463
+ */
1464
+ public function AddAttachment($path, $name = '', $encoding = 'base64', $type = 'application/octet-stream') {
1465
+ try {
1466
+ if ( !@is_file($path) ) {
1467
+ throw new EM_phpmailerException($this->Lang('file_access') . $path, self::STOP_CONTINUE);
1468
+ }
1469
+ $filename = basename($path);
1470
+ if ( $name == '' ) {
1471
+ $name = $filename;
1472
+ }
1473
 
1474
+ $this->attachment[] = array(
1475
+ 0 => $path,
1476
+ 1 => $filename,
1477
+ 2 => $name,
1478
+ 3 => $encoding,
1479
+ 4 => $type,
1480
+ 5 => false, // isStringAttachment
1481
+ 6 => 'attachment',
1482
+ 7 => 0
1483
+ );
1484
+
1485
+ } catch (EM_phpmailerException $e) {
1486
+ $this->SetError($e->getMessage());
1487
+ if ($this->exceptions) {
1488
+ throw $e;
1489
+ }
1490
+ echo $e->getMessage()."\n";
1491
+ if ( $e->getCode() == self::STOP_CRITICAL ) {
1492
+ return false;
1493
+ }
1494
+ }
1495
+ return true;
1496
+ }
1497
+
1498
+ /**
1499
+ * Return the current array of attachments
1500
+ * @return array
1501
+ */
1502
+ public function GetAttachments() {
1503
+ return $this->attachment;
1504
+ }
1505
+
1506
+ /**
1507
+ * Attaches all fs, string, and binary attachments to the message.
1508
+ * Returns an empty string on failure.
1509
+ * @access protected
1510
+ * @return string
1511
+ */
1512
+ protected function AttachAll($disposition_type, $boundary) {
1513
+ // Return text of body
1514
+ $mime = array();
1515
+ $cidUniq = array();
1516
+ $incl = array();
1517
+
1518
+ // Add all attachments
1519
+ foreach ($this->attachment as $attachment) {
1520
+ // CHECK IF IT IS A VALID DISPOSITION_FILTER
1521
+ if($attachment[6] == $disposition_type) {
1522
+ // Check for string attachment
1523
+ $bString = $attachment[5];
1524
+ if ($bString) {
1525
+ $string = $attachment[0];
1526
+ } else {
1527
+ $path = $attachment[0];
1528
+ }
1529
 
1530
+ $inclhash = md5(serialize($attachment));
1531
+ if (in_array($inclhash, $incl)) { continue; }
1532
+ $incl[] = $inclhash;
1533
+ $filename = $attachment[1];
1534
+ $name = $attachment[2];
1535
+ $encoding = $attachment[3];
1536
+ $type = $attachment[4];
1537
+ $disposition = $attachment[6];
1538
+ $cid = $attachment[7];
1539
+ if ( $disposition == 'inline' && isset($cidUniq[$cid]) ) { continue; }
1540
+ $cidUniq[$cid] = true;
1541
+
1542
+ $mime[] = sprintf("--%s%s", $boundary, $this->LE);
1543
+ $mime[] = sprintf("Content-Type: %s; name=\"%s\"%s", $type, $this->EncodeHeader($this->SecureHeader($name)), $this->LE);
1544
+ $mime[] = sprintf("Content-Transfer-Encoding: %s%s", $encoding, $this->LE);
1545
+
1546
+ if($disposition == 'inline') {
1547
+ $mime[] = sprintf("Content-ID: <%s>%s", $cid, $this->LE);
1548
+ }
1549
 
1550
+ $mime[] = sprintf("Content-Disposition: %s; filename=\"%s\"%s", $disposition, $this->EncodeHeader($this->SecureHeader($name)), $this->LE.$this->LE);
 
1551
 
1552
+ // Encode as string attachment
1553
+ if($bString) {
1554
+ $mime[] = $this->EncodeString($string, $encoding);
1555
+ if($this->IsError()) {
1556
+ return '';
1557
+ }
1558
+ $mime[] = $this->LE.$this->LE;
1559
+ } else {
1560
+ $mime[] = $this->EncodeFile($path, $encoding);
1561
+ if($this->IsError()) {
1562
+ return '';
1563
+ }
1564
+ $mime[] = $this->LE.$this->LE;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1565
  }
1566
+ }
1567
+ }
1568
 
1569
+ $mime[] = sprintf("--%s--%s", $boundary, $this->LE);
1570
+
1571
+ return implode("", $mime);
1572
+ }
1573
+
1574
+ /**
1575
+ * Encodes attachment in requested format.
1576
+ * Returns an empty string on failure.
1577
+ * @param string $path The full path to the file
1578
+ * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable'
1579
+ * @see EncodeFile()
1580
+ * @access protected
1581
+ * @return string
1582
+ */
1583
+ protected function EncodeFile($path, $encoding = 'base64') {
1584
+ try {
1585
+ if (!is_readable($path)) {
1586
+ throw new EM_phpmailerException($this->Lang('file_open') . $path, self::STOP_CONTINUE);
1587
+ }
1588
+ if (function_exists('get_magic_quotes')) {
1589
+ function get_magic_quotes() {
1590
+ return false;
1591
  }
1592
+ }
1593
+ if (version_compare(PHP_VERSION, '5.3.0', '<')) {
1594
  $magic_quotes = get_magic_quotes_runtime();
1595
  set_magic_quotes_runtime(0);
1596
+ }
1597
+ $file_buffer = file_get_contents($path);
1598
+ $file_buffer = $this->EncodeString($file_buffer, $encoding);
1599
+ if (version_compare(PHP_VERSION, '5.3.0', '<')) {
1600
  set_magic_quotes_runtime($magic_quotes);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1601
  }
1602
+ return $file_buffer;
1603
+ } catch (Exception $e) {
1604
+ $this->SetError($e->getMessage());
1605
+ return '';
1606
+ }
1607
+ }
1608
+
1609
+ /**
1610
+ * Encodes string to requested format.
1611
+ * Returns an empty string on failure.
1612
+ * @param string $str The text to encode
1613
+ * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable'
1614
+ * @access public
1615
+ * @return string
1616
+ */
1617
+ public function EncodeString($str, $encoding = 'base64') {
1618
+ $encoded = '';
1619
+ switch(strtolower($encoding)) {
1620
+ case 'base64':
1621
+ $encoded = chunk_split(base64_encode($str), 76, $this->LE);
1622
+ break;
1623
+ case '7bit':
1624
+ case '8bit':
1625
+ $encoded = $this->FixEOL($str);
1626
+ //Make sure it ends with a line break
1627
+ if (substr($encoded, -(strlen($this->LE))) != $this->LE)
1628
+ $encoded .= $this->LE;
1629
+ break;
1630
+ case 'binary':
1631
+ $encoded = $str;
1632
+ break;
1633
+ case 'quoted-printable':
1634
+ $encoded = $this->EncodeQP($str);
1635
+ break;
1636
+ default:
1637
+ $this->SetError($this->Lang('encoding') . $encoding);
1638
+ break;
1639
+ }
1640
+ return $encoded;
1641
+ }
1642
+
1643
+ /**
1644
+ * Encode a header string to best (shortest) of Q, B, quoted or none.
1645
+ * @access public
1646
+ * @return string
1647
+ */
1648
+ public function EncodeHeader($str, $position = 'text') {
1649
+ $x = 0;
1650
+
1651
+ switch (strtolower($position)) {
1652
+ case 'phrase':
1653
+ if (!preg_match('/[\200-\377]/', $str)) {
1654
+ // Can't use addslashes as we don't know what value has magic_quotes_sybase
1655
+ $encoded = addcslashes($str, "\0..\37\177\\\"");
1656
+ if (($str == $encoded) && !preg_match('/[^A-Za-z0-9!#$%&\'*+\/=?^_`{|}~ -]/', $str)) {
1657
+ return ($encoded);
1658
+ } else {
1659
+ return ("\"$encoded\"");
1660
+ }
1661
+ }
1662
+ $x = preg_match_all('/[^\040\041\043-\133\135-\176]/', $str, $matches);
1663
+ break;
1664
+ case 'comment':
1665
+ $x = preg_match_all('/[()"]/', $str, $matches);
1666
+ // Fall-through
1667
+ case 'text':
1668
+ default:
1669
+ $x += preg_match_all('/[\000-\010\013\014\016-\037\177-\377]/', $str, $matches);
1670
+ break;
1671
+ }
1672
 
1673
+ if ($x == 0) {
1674
+ return ($str);
1675
+ }
1676
 
1677
+ $maxlen = 75 - 7 - strlen($this->CharSet);
1678
+ // Try to select the encoding which should produce the shortest output
1679
+ if (strlen($str)/3 < $x) {
1680
+ $encoding = 'B';
1681
+ if (function_exists('mb_strlen') && $this->HasMultiBytes($str)) {
1682
+ // Use a custom function which correctly encodes and wraps long
1683
+ // multibyte strings without breaking lines within a character
1684
+ $encoded = $this->Base64EncodeWrapMB($str);
1685
+ } else {
1686
  $encoded = base64_encode($str);
1687
  $maxlen -= $maxlen % 4;
1688
  $encoded = trim(chunk_split($encoded, $maxlen, "\n"));
 
 
 
 
 
1689
  }
1690
+ } else {
1691
+ $encoding = 'Q';
1692
+ $encoded = $this->EncodeQ($str, $position);
1693
+ $encoded = $this->WrapText($encoded, $maxlen, true);
1694
+ $encoded = str_replace('='.$this->LE, "\n", trim($encoded));
1695
+ }
1696
 
1697
+ $encoded = preg_replace('/^(.*)$/m', " =?".$this->CharSet."?$encoding?\\1?=", $encoded);
1698
+ $encoded = trim(str_replace("\n", $this->LE, $encoded));
1699
+
1700
+ return $encoded;
1701
+ }
1702
+
1703
+ /**
1704
+ * Checks if a string contains multibyte characters.
1705
+ * @access public
1706
+ * @param string $str multi-byte text to wrap encode
1707
+ * @return bool
1708
+ */
1709
+ public function HasMultiBytes($str) {
1710
+ if (function_exists('mb_strlen')) {
1711
+ return (strlen($str) > mb_strlen($str, $this->CharSet));
1712
+ } else { // Assume no multibytes (we can't handle without mbstring functions anyway)
1713
+ return false;
1714
+ }
1715
+ }
1716
+
1717
+ /**
1718
+ * Correctly encodes and wraps long multibyte strings for mail headers
1719
+ * without breaking lines within a character.
1720
+ * Adapted from a function by paravoid at http://uk.php.net/manual/en/function.mb-encode-mimeheader.php
1721
+ * @access public
1722
+ * @param string $str multi-byte text to wrap encode
1723
+ * @return string
1724
+ */
1725
+ public function Base64EncodeWrapMB($str) {
1726
+ $start = "=?".$this->CharSet."?B?";
1727
+ $end = "?=";
1728
+ $encoded = "";
1729
+
1730
+ $mb_length = mb_strlen($str, $this->CharSet);
1731
+ // Each line must have length <= 75, including $start and $end
1732
+ $length = 75 - strlen($start) - strlen($end);
1733
+ // Average multi-byte ratio
1734
+ $ratio = $mb_length / strlen($str);
1735
+ // Base64 has a 4:3 ratio
1736
+ $offset = $avgLength = floor($length * $ratio * .75);
1737
+
1738
+ for ($i = 0; $i < $mb_length; $i += $offset) {
1739
+ $lookBack = 0;
1740
+
1741
+ do {
1742
+ $offset = $avgLength - $lookBack;
1743
+ $chunk = mb_substr($str, $i, $offset, $this->CharSet);
1744
+ $chunk = base64_encode($chunk);
1745
+ $lookBack++;
1746
+ }
1747
+ while (strlen($chunk) > $length);
1748
+
1749
+ $encoded .= $chunk . $this->LE;
1750
+ }
1751
+
1752
+ // Chomp the last linefeed
1753
+ $encoded = substr($encoded, 0, -strlen($this->LE));
1754
+ return $encoded;
1755
+ }
1756
+
1757
+ /**
1758
+ * Encode string to quoted-printable.
1759
+ * Only uses standard PHP, slow, but will always work
1760
+ * @access public
1761
+ * @param string $string the text to encode
1762
+ * @param integer $line_max Number of chars allowed on a line before wrapping
1763
+ * @return string
1764
+ */
1765
+ public function EncodeQPphp( $input = '', $line_max = 76, $space_conv = false) {
1766
+ $hex = array('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F');
1767
+ $lines = preg_split('/(?:\r\n|\r|\n)/', $input);
1768
+ $eol = "\r\n";
1769
+ $escape = '=';
1770
+ $output = '';
1771
+ while( list(, $line) = each($lines) ) {
1772
+ $linlen = strlen($line);
1773
+ $newline = '';
1774
+ for($i = 0; $i < $linlen; $i++) {
1775
+ $c = substr( $line, $i, 1 );
1776
+ $dec = ord( $c );
1777
+ if ( ( $i == 0 ) && ( $dec == 46 ) ) { // convert first point in the line into =2E
1778
+ $c = '=2E';
1779
+ }
1780
+ if ( $dec == 32 ) {
1781
+ if ( $i == ( $linlen - 1 ) ) { // convert space at eol only
1782
+ $c = '=20';
1783
+ } else if ( $space_conv ) {
1784
+ $c = '=20';
1785
+ }
1786
+ } elseif ( ($dec == 61) || ($dec < 32 ) || ($dec > 126) ) { // always encode "\t", which is *not* required
1787
+ $h2 = floor($dec/16);
1788
+ $h1 = floor($dec%16);
1789
+ $c = $escape.$hex[$h2].$hex[$h1];
1790
  }
1791
+ if ( (strlen($newline) + strlen($c)) >= $line_max ) { // CRLF is not counted
1792
+ $output .= $newline.$escape.$eol; // soft line break; " =\r\n" is okay
1793
+ $newline = '';
1794
+ // check if newline first character will be point or not
1795
+ if ( $dec == 46 ) {
1796
+ $c = '=2E';
1797
+ }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1798
  }
1799
+ $newline .= $c;
1800
+ } // end of for
1801
+ $output .= $newline.$eol;
1802
+ } // end of while
1803
+ return $output;
1804
+ }
1805
+
1806
+ /**
1807
+ * Encode string to RFC2045 (6.7) quoted-printable format
1808
+ * Uses a PHP5 stream filter to do the encoding about 64x faster than the old version
1809
+ * Also results in same content as you started with after decoding
1810
+ * @see EncodeQPphp()
1811
+ * @access public
1812
+ * @param string $string the text to encode
1813
+ * @param integer $line_max Number of chars allowed on a line before wrapping
1814
+ * @param boolean $space_conv Dummy param for compatibility with existing EncodeQP function
1815
+ * @return string
1816
+ * @author Marcus Bointon
1817
+ */
1818
+ public function EncodeQP($string, $line_max = 76, $space_conv = false) {
1819
+ if (function_exists('quoted_printable_encode')) { //Use native function if it's available (>= PHP5.3)
1820
+ return quoted_printable_encode($string);
1821
+ }
1822
+ $filters = stream_get_filters();
1823
+ if (!in_array('convert.*', $filters)) { //Got convert stream filter?
1824
+ return $this->EncodeQPphp($string, $line_max, $space_conv); //Fall back to old implementation
1825
+ }
1826
+ $fp = fopen('php://temp/', 'r+');
1827
+ $string = preg_replace('/\r\n?/', $this->LE, $string); //Normalise line breaks
1828
+ $params = array('line-length' => $line_max, 'line-break-chars' => $this->LE);
1829
+ $s = stream_filter_append($fp, 'convert.quoted-printable-encode', STREAM_FILTER_READ, $params);
1830
+ fputs($fp, $string);
1831
+ rewind($fp);
1832
+ $out = stream_get_contents($fp);
1833
+ stream_filter_remove($s);
1834
+ $out = preg_replace('/^\./m', '=2E', $out); //Encode . if it is first char on a line, workaround for bug in Exchange
1835
+ fclose($fp);
1836
+ return $out;
1837
+ }
1838
+
1839
+ /**
1840
+ * Encode string to q encoding.
1841
+ * @link http://tools.ietf.org/html/rfc2047
1842
+ * @param string $str the text to encode
1843
+ * @param string $position Where the text is going to be used, see the RFC for what that means
1844
+ * @access public
1845
+ * @return string
1846
+ */
1847
+ public function EncodeQ($str, $position = 'text') {
1848
+ // There should not be any EOL in the string
1849
+ $encoded = preg_replace('/[\r\n]*/', '', $str);
1850
+
1851
+ switch (strtolower($position)) {
1852
+ case 'phrase':
1853
+ $encoded = preg_replace("/([^A-Za-z0-9!*+\/ -])/e", "'='.sprintf('%02X', ord('\\1'))", $encoded);
1854
+ break;
1855
+ case 'comment':
1856
+ $encoded = preg_replace("/([\(\)\"])/e", "'='.sprintf('%02X', ord('\\1'))", $encoded);
1857
+ case 'text':
1858
+ default:
1859
+ // Replace every high ascii, control =, ? and _ characters
1860
+ //TODO using /e (equivalent to eval()) is probably not a good idea
1861
+ $encoded = preg_replace('/([\000-\011\013\014\016-\037\075\077\137\177-\377])/e',
1862
+ "'='.sprintf('%02X', ord(stripslashes('\\1')))", $encoded);
1863
+ break;
1864
+ }
1865
+
1866
+ // Replace every spaces to _ (more readable than =20)
1867
+ $encoded = str_replace(' ', '_', $encoded);
1868
+
1869
+ return $encoded;
1870
+ }
1871
+
1872
+ /**
1873
+ * Adds a string or binary attachment (non-filesystem) to the list.
1874
+ * This method can be used to attach ascii or binary data,
1875
+ * such as a BLOB record from a database.
1876
+ * @param string $string String attachment data.
1877
+ * @param string $filename Name of the attachment.
1878
+ * @param string $encoding File encoding (see $Encoding).
1879
+ * @param string $type File extension (MIME) type.
1880
+ * @return void
1881
+ */
1882
+ public function AddStringAttachment($string, $filename, $encoding = 'base64', $type = 'application/octet-stream') {
1883
+ // Append to $attachment array
1884
+ $this->attachment[] = array(
1885
+ 0 => $string,
1886
+ 1 => $filename,
1887
+ 2 => basename($filename),
1888
+ 3 => $encoding,
1889
+ 4 => $type,
1890
+ 5 => true, // isStringAttachment
1891
+ 6 => 'attachment',
1892
+ 7 => 0
1893
+ );
1894
+ }
1895
+
1896
+ /**
1897
+ * Adds an embedded attachment. This can include images, sounds, and
1898
+ * just about any other document. Make sure to set the $type to an
1899
+ * image type. For JPEG images use "image/jpeg" and for GIF images
1900
+ * use "image/gif".
1901
+ * @param string $path Path to the attachment.
1902
+ * @param string $cid Content ID of the attachment. Use this to identify
1903
+ * the Id for accessing the image in an HTML form.
1904
+ * @param string $name Overrides the attachment name.
1905
+ * @param string $encoding File encoding (see $Encoding).
1906
+ * @param string $type File extension (MIME) type.
1907
+ * @return bool
1908
+ */
1909
+ public function AddEmbeddedImage($path, $cid, $name = '', $encoding = 'base64', $type = 'application/octet-stream') {
1910
+
1911
+ if ( !@is_file($path) ) {
1912
+ $this->SetError($this->Lang('file_access') . $path);
1913
+ return false;
1914
+ }
1915
+
1916
+ $filename = basename($path);
1917
+ if ( $name == '' ) {
1918
+ $name = $filename;
1919
+ }
1920
 
1921
+ // Append to $attachment array
1922
+ $this->attachment[] = array(
1923
+ 0 => $path,
1924
+ 1 => $filename,
1925
+ 2 => $name,
1926
+ 3 => $encoding,
1927
+ 4 => $type,
1928
+ 5 => false, // isStringAttachment
1929
+ 6 => 'inline',
1930
+ 7 => $cid
1931
+ );
1932
+
1933
+ return true;
1934
+ }
1935
+
1936
+ public function AddStringEmbeddedImage($string, $cid, $filename = '', $encoding = 'base64', $type = 'application/octet-stream') {
1937
+ // Append to $attachment array
1938
+ $this->attachment[] = array(
1939
+ 0 => $string,
1940
+ 1 => $filename,
1941
+ 2 => basename($filename),
1942
+ 3 => $encoding,
1943
+ 4 => $type,
1944
+ 5 => true, // isStringAttachment
1945
+ 6 => 'inline',
1946
+ 7 => $cid
1947
+ );
1948
+ }
1949
+
1950
+ /**
1951
+ * Returns true if an inline attachment is present.
1952
+ * @access public
1953
+ * @return bool
1954
+ */
1955
+ public function InlineImageExists() {
1956
+ foreach($this->attachment as $attachment) {
1957
+ if ($attachment[6] == 'inline') {
1958
  return true;
1959
+ }
1960
  }
1961
+ return false;
1962
+ }
1963
+
1964
+ public function AttachmentExists() {
1965
+ foreach($this->attachment as $attachment) {
1966
+ if ($attachment[6] == 'attachment') {
1967
+ return true;
1968
+ }
1969
+ }
1970
+ return false;
1971
+ }
1972
+
1973
+ public function AlternativeExists() {
1974
+ return strlen($this->AltBody)>0;
1975
+ }
1976
+
1977
+ /////////////////////////////////////////////////
1978
+ // CLASS METHODS, MESSAGE RESET
1979
+ /////////////////////////////////////////////////
1980
+
1981
+ /**
1982
+ * Clears all recipients assigned in the TO array. Returns void.
1983
+ * @return void
1984
+ */
1985
+ public function ClearAddresses() {
1986
+ foreach($this->to as $to) {
1987
+ unset($this->all_recipients[strtolower($to[0])]);
1988
+ }
1989
+ $this->to = array();
1990
+ }
1991
+
1992
+ /**
1993
+ * Clears all recipients assigned in the CC array. Returns void.
1994
+ * @return void
1995
+ */
1996
+ public function ClearCCs() {
1997
+ foreach($this->cc as $cc) {
1998
+ unset($this->all_recipients[strtolower($cc[0])]);
1999
+ }
2000
+ $this->cc = array();
2001
+ }
2002
+
2003
+ /**
2004
+ * Clears all recipients assigned in the BCC array. Returns void.
2005
+ * @return void
2006
+ */
2007
+ public function ClearBCCs() {
2008
+ foreach($this->bcc as $bcc) {
2009
+ unset($this->all_recipients[strtolower($bcc[0])]);
2010
+ }
2011
+ $this->bcc = array();
2012
+ }
2013
+
2014
+ /**
2015
+ * Clears all recipients assigned in the ReplyTo array. Returns void.
2016
+ * @return void
2017
+ */
2018
+ public function ClearReplyTos() {
2019
+ $this->ReplyTo = array();
2020
+ }
2021
+
2022
+ /**
2023
+ * Clears all recipients assigned in the TO, CC and BCC
2024
+ * array. Returns void.
2025
+ * @return void
2026
+ */
2027
+ public function ClearAllRecipients() {
2028
+ $this->to = array();
2029
+ $this->cc = array();
2030
+ $this->bcc = array();
2031
+ $this->all_recipients = array();
2032
+ }
2033
+
2034
+ /**
2035
+ * Clears all previously set filesystem, string, and binary
2036
+ * attachments. Returns void.
2037
+ * @return void
2038
+ */
2039
+ public function ClearAttachments() {
2040
+ $this->attachment = array();
2041
+ }
2042
+
2043
+ /**
2044
+ * Clears all custom headers. Returns void.
2045
+ * @return void
2046
+ */
2047
+ public function ClearCustomHeaders() {
2048
+ $this->CustomHeader = array();
2049
+ }
2050
+
2051
+ /////////////////////////////////////////////////
2052
+ // CLASS METHODS, MISCELLANEOUS
2053
+ /////////////////////////////////////////////////
2054
+
2055
+ /**
2056
+ * Adds the error message to the error container.
2057
+ * @access protected
2058
+ * @return void
2059
+ */
2060
+ protected function SetError($msg) {
2061
+ $this->error_count++;
2062
+ if ($this->Mailer == 'smtp' and !is_null($this->smtp)) {
2063
+ $lasterror = $this->smtp->getError();
2064
+ if (!empty($lasterror) and array_key_exists('smtp_msg', $lasterror)) {
2065
+ $msg .= '<p>' . $this->Lang('smtp_error') . $lasterror['smtp_msg'] . "</p>\n";
2066
+ }
2067
+ }
2068
+ $this->ErrorInfo = $msg;
2069
+ }
2070
+
2071
+ /**
2072
+ * Returns the proper RFC 822 formatted date.
2073
+ * @access public
2074
+ * @return string
2075
+ * @static
2076
+ */
2077
+ public static function RFCDate() {
2078
+ $tz = date('Z');
2079
+ $tzs = ($tz < 0) ? '-' : '+';
2080
+ $tz = abs($tz);
2081
+ $tz = (int)($tz/3600)*100 + ($tz%3600)/60;
2082
+ $result = sprintf("%s %s%04d", date('D, j M Y H:i:s'), $tzs, $tz);
2083
+
2084
+ return $result;
2085
+ }
2086
+
2087
+ /**
2088
+ * Returns the server hostname or 'localhost.localdomain' if unknown.
2089
+ * @access protected
2090
+ * @return string
2091
+ */
2092
+ protected function ServerHostname() {
2093
+ if (!empty($this->Hostname)) {
2094
+ $result = $this->Hostname;
2095
+ } elseif (isset($_SERVER['SERVER_NAME'])) {
2096
+ $result = $_SERVER['SERVER_NAME'];
2097
+ } else {
2098
+ $result = 'localhost.localdomain';
2099
+ }
2100
+
2101
+ return $result;
2102
+ }
2103
+
2104
+ /**
2105
+ * Returns a message in the appropriate language.
2106
+ * @access protected
2107
+ * @return string
2108
+ */
2109
+ protected function Lang($key) {
2110
+ if(count($this->language) < 1) {
2111
+ $this->SetLanguage('en'); // set the default language
2112
+ }
2113
+
2114
+ if(isset($this->language[$key])) {
2115
+ return $this->language[$key];
2116
+ } else {
2117
+ return 'Language string failed to load: ' . $key;
2118
+ }
2119
+ }
2120
+
2121
+ /**
2122
+ * Returns true if an error occurred.
2123
+ * @access public
2124
+ * @return bool
2125
+ */
2126
+ public function IsError() {
2127
+ return ($this->error_count > 0);
2128
+ }
2129
+
2130
+ /**
2131
+ * Changes every end of line from CR or LF to CRLF.
2132
+ * @access public
2133
+ * @return string
2134
+ */
2135
+ public function FixEOL($str) {
2136
+ $str = str_replace("\r\n", "\n", $str);
2137
+ $str = str_replace("\r", "\n", $str);
2138
+ $str = str_replace("\n", $this->LE, $str);
2139
+ return $str;
2140
+ }
2141
+
2142
+ /**
2143
+ * Adds a custom header.
2144
+ * @access public
2145
+ * @return void
2146
+ */
2147
+ public function AddCustomHeader($custom_header) {
2148
+ $this->CustomHeader[] = explode(':', $custom_header, 2);
2149
+ }
2150
+
2151
+ /**
2152
+ * Evaluates the message and returns modifications for inline images and backgrounds
2153
+ * @access public
2154
+ * @return $message
2155
+ */
2156
+ public function MsgHTML($message, $basedir = '') {
2157
+ preg_match_all("/(src|background)=\"(.*)\"/Ui", $message, $images);
2158
+ if(isset($images[2])) {
2159
+ foreach($images[2] as $i => $url) {
2160
+ // do not change urls for absolute images (thanks to corvuscorax)
2161
+ if (!preg_match('#^[A-z]+://#', $url)) {
2162
+ $filename = basename($url);
2163
+ $directory = dirname($url);
2164
+ ($directory == '.') ? $directory='': '';
2165
+ $cid = 'cid:' . md5($filename);
2166
+ $ext = pathinfo($filename, PATHINFO_EXTENSION);
2167
+ $mimeType = self::_mime_types($ext);
2168
+ if ( strlen($basedir) > 1 && substr($basedir, -1) != '/') { $basedir .= '/'; }
2169
+ if ( strlen($directory) > 1 && substr($directory, -1) != '/') { $directory .= '/'; }
2170
+ if ( $this->AddEmbeddedImage($basedir.$directory.$filename, md5($filename), $filename, 'base64', $mimeType) ) {
2171
+ $message = preg_replace("/".$images[1][$i]."=\"".preg_quote($url, '/')."\"/Ui", $images[1][$i]."=\"".$cid."\"", $message);
2172
+ }
2173
  }
2174
+ }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
2175
  }
2176
+ $this->IsHTML(true);
2177
+ $this->Body = $message;
2178
+ $textMsg = trim(strip_tags(preg_replace('/<(head|title|style|script)[^>]*>.*?<\/\\1>/s', '', $message)));
2179
+ if (!empty($textMsg) && empty($this->AltBody)) {
2180
+ $this->AltBody = html_entity_decode($textMsg);
2181
+ }
2182
+ if (empty($this->AltBody)) {
2183
+ $this->AltBody = 'To view this email message, open it in a program that understands HTML!' . "\n\n";
2184
+ }
2185
+ }
2186
+
2187
+ /**
2188
+ * Gets the MIME type of the embedded or inline image
2189
+ * @param string File extension
2190
+ * @access public
2191
+ * @return string MIME type of ext
2192
+ * @static
2193
+ */
2194
+ public static function _mime_types($ext = '') {
2195
+ $mimes = array(
2196
+ 'hqx' => 'application/mac-binhex40',
2197
+ 'cpt' => 'application/mac-compactpro',
2198
+ 'doc' => 'application/msword',
2199
+ 'bin' => 'application/macbinary',
2200
+ 'dms' => 'application/octet-stream',
2201
+ 'lha' => 'application/octet-stream',
2202
+ 'lzh' => 'application/octet-stream',
2203
+ 'exe' => 'application/octet-stream',
2204
+ 'class' => 'application/octet-stream',
2205
+ 'psd' => 'application/octet-stream',