@@ -18,83 +18,83 @@ discard block |
||
18 | 18 | class CommandHandlerManager implements CommandHandlerManagerInterface |
19 | 19 | { |
20 | 20 | |
21 | - /** |
|
22 | - * @var CommandHandlerInterface[] $command_handlers |
|
23 | - */ |
|
24 | - protected $command_handlers; |
|
21 | + /** |
|
22 | + * @var CommandHandlerInterface[] $command_handlers |
|
23 | + */ |
|
24 | + protected $command_handlers; |
|
25 | 25 | |
26 | - /** |
|
27 | - * @type LoaderInterface $loader |
|
28 | - */ |
|
29 | - private $loader; |
|
26 | + /** |
|
27 | + * @type LoaderInterface $loader |
|
28 | + */ |
|
29 | + private $loader; |
|
30 | 30 | |
31 | 31 | |
32 | - /** |
|
33 | - * CommandHandlerManager constructor |
|
34 | - * |
|
35 | - * @param LoaderInterface $loader |
|
36 | - */ |
|
37 | - public function __construct(LoaderInterface $loader) |
|
38 | - { |
|
39 | - $this->loader = $loader; |
|
40 | - } |
|
32 | + /** |
|
33 | + * CommandHandlerManager constructor |
|
34 | + * |
|
35 | + * @param LoaderInterface $loader |
|
36 | + */ |
|
37 | + public function __construct(LoaderInterface $loader) |
|
38 | + { |
|
39 | + $this->loader = $loader; |
|
40 | + } |
|
41 | 41 | |
42 | 42 | |
43 | - /** |
|
44 | - * By default, Commands and CommandHandlers would normally |
|
45 | - * reside in the same folder under the same namespace, |
|
46 | - * and the names of the two classes would only differ in that |
|
47 | - * one ends in "Command" and the other ends in "CommandHandler". |
|
48 | - * However, if you wanted to utilize a CommandHandler from somewhere else, |
|
49 | - * then this method allows you to add that CommandHandler and specify the FQCN |
|
50 | - * (Fully Qualified ClassName) for the Command class that it should be used for. |
|
51 | - * For example: |
|
52 | - * by default the "Vendor\some\namespace\DoSomethingCommand" |
|
53 | - * would resolve to using "Vendor\some\namespace\DoSomethingCommandHandler" |
|
54 | - * but if you wanted to instead process that commend using: |
|
55 | - * "Vendor\a\totally\different\namespace\for\DoSomethingCommandHandler" |
|
56 | - * then the following code: |
|
57 | - * $CommandHandlerManager = $this->loader->getShared( 'CommandHandlerManagerInterface' ); |
|
58 | - * $CommandHandlerManager->addCommandHandler( |
|
59 | - * new Vendor\a\totally\different\namespace\for\DoSomethingCommandHandler(), |
|
60 | - * 'Vendor\some\namespace\DoSomethingCommand' |
|
61 | - * ); |
|
62 | - * would result in the alternate CommandHandler being used to process that Command |
|
63 | - * |
|
64 | - * !!! IMPORTANT !!! |
|
65 | - * If overriding the default CommandHandler for a Command, |
|
66 | - * be sure to also override CommandHandler::verify(), |
|
67 | - * or else an Exception will be thrown when the CommandBus |
|
68 | - * attempts to verify that the incoming Command matches the Handler |
|
69 | - * |
|
70 | - * @param CommandHandlerInterface $command_handler |
|
71 | - * @param string $fqcn_for_command Fully Qualified ClassName for Command |
|
72 | - * @return void |
|
73 | - * @throws InvalidCommandHandlerException |
|
74 | - */ |
|
75 | - public function addCommandHandler(CommandHandlerInterface $command_handler, $fqcn_for_command = '') |
|
76 | - { |
|
77 | - $command = ! empty($fqcn_for_command) |
|
78 | - ? $fqcn_for_command |
|
79 | - : str_replace('CommandHandler', 'Command', get_class($command_handler)); |
|
80 | - if (empty($command)) { |
|
81 | - throw new InvalidCommandHandlerException($command); |
|
82 | - } |
|
83 | - $this->command_handlers[ $command ] = $command_handler; |
|
84 | - } |
|
43 | + /** |
|
44 | + * By default, Commands and CommandHandlers would normally |
|
45 | + * reside in the same folder under the same namespace, |
|
46 | + * and the names of the two classes would only differ in that |
|
47 | + * one ends in "Command" and the other ends in "CommandHandler". |
|
48 | + * However, if you wanted to utilize a CommandHandler from somewhere else, |
|
49 | + * then this method allows you to add that CommandHandler and specify the FQCN |
|
50 | + * (Fully Qualified ClassName) for the Command class that it should be used for. |
|
51 | + * For example: |
|
52 | + * by default the "Vendor\some\namespace\DoSomethingCommand" |
|
53 | + * would resolve to using "Vendor\some\namespace\DoSomethingCommandHandler" |
|
54 | + * but if you wanted to instead process that commend using: |
|
55 | + * "Vendor\a\totally\different\namespace\for\DoSomethingCommandHandler" |
|
56 | + * then the following code: |
|
57 | + * $CommandHandlerManager = $this->loader->getShared( 'CommandHandlerManagerInterface' ); |
|
58 | + * $CommandHandlerManager->addCommandHandler( |
|
59 | + * new Vendor\a\totally\different\namespace\for\DoSomethingCommandHandler(), |
|
60 | + * 'Vendor\some\namespace\DoSomethingCommand' |
|
61 | + * ); |
|
62 | + * would result in the alternate CommandHandler being used to process that Command |
|
63 | + * |
|
64 | + * !!! IMPORTANT !!! |
|
65 | + * If overriding the default CommandHandler for a Command, |
|
66 | + * be sure to also override CommandHandler::verify(), |
|
67 | + * or else an Exception will be thrown when the CommandBus |
|
68 | + * attempts to verify that the incoming Command matches the Handler |
|
69 | + * |
|
70 | + * @param CommandHandlerInterface $command_handler |
|
71 | + * @param string $fqcn_for_command Fully Qualified ClassName for Command |
|
72 | + * @return void |
|
73 | + * @throws InvalidCommandHandlerException |
|
74 | + */ |
|
75 | + public function addCommandHandler(CommandHandlerInterface $command_handler, $fqcn_for_command = '') |
|
76 | + { |
|
77 | + $command = ! empty($fqcn_for_command) |
|
78 | + ? $fqcn_for_command |
|
79 | + : str_replace('CommandHandler', 'Command', get_class($command_handler)); |
|
80 | + if (empty($command)) { |
|
81 | + throw new InvalidCommandHandlerException($command); |
|
82 | + } |
|
83 | + $this->command_handlers[ $command ] = $command_handler; |
|
84 | + } |
|
85 | 85 | |
86 | 86 | |
87 | - /** |
|
88 | - * @param CommandInterface $command |
|
89 | - * @param CommandBusInterface $command_bus |
|
90 | - * @return mixed |
|
91 | - * @throws DomainException |
|
92 | - * @throws CommandHandlerNotFoundException |
|
93 | - */ |
|
94 | - public function getCommandHandler(CommandInterface $command, CommandBusInterface $command_bus = null) |
|
95 | - { |
|
96 | - $command_name = get_class($command); |
|
97 | - /* |
|
87 | + /** |
|
88 | + * @param CommandInterface $command |
|
89 | + * @param CommandBusInterface $command_bus |
|
90 | + * @return mixed |
|
91 | + * @throws DomainException |
|
92 | + * @throws CommandHandlerNotFoundException |
|
93 | + */ |
|
94 | + public function getCommandHandler(CommandInterface $command, CommandBusInterface $command_bus = null) |
|
95 | + { |
|
96 | + $command_name = get_class($command); |
|
97 | + /* |
|
98 | 98 | * Filters the Fully Qualified Class Name for the Command Handler |
99 | 99 | * that will be used to handle the incoming Command |
100 | 100 | * |
@@ -107,34 +107,34 @@ discard block |
||
107 | 107 | * @param string "CommandHandler::class" Fully Qualified Class Name for the Command Handler |
108 | 108 | * @param CommandInterface $command the actual Command instance |
109 | 109 | */ |
110 | - $command_handler = apply_filters( |
|
111 | - 'FHEE__EventEspresso_core_services_commands_CommandHandlerManager__getCommandHandler__command_handler', |
|
112 | - str_replace('Command', 'CommandHandler', $command_name), |
|
113 | - $command |
|
114 | - ); |
|
115 | - $handler = null; |
|
116 | - // has a command handler already been set for this class ? |
|
117 | - // if not, can we find one via the FQCN ? |
|
118 | - if (isset($this->command_handlers[ $command_name ])) { |
|
119 | - $handler = $this->command_handlers[ $command_name ]; |
|
120 | - } elseif (class_exists($command_handler)) { |
|
121 | - $handler = $this->loader->getShared($command_handler); |
|
122 | - } |
|
123 | - // if Handler requires an instance of the CommandBus, but that has not yet been set |
|
124 | - if ($handler instanceof CompositeCommandHandler && ! $handler->commandBus() instanceof CommandBusInterface) { |
|
125 | - if (! $command_bus instanceof CommandBusInterface) { |
|
126 | - throw new DomainException( |
|
127 | - esc_html__( |
|
128 | - 'CompositeCommandHandler classes require an instance of the CommandBus.', |
|
129 | - 'event_espresso' |
|
130 | - ) |
|
131 | - ); |
|
132 | - } |
|
133 | - $handler->setCommandBus($command_bus); |
|
134 | - } |
|
135 | - if ($handler instanceof CommandHandlerInterface) { |
|
136 | - return $handler; |
|
137 | - } |
|
138 | - throw new CommandHandlerNotFoundException($command_handler); |
|
139 | - } |
|
110 | + $command_handler = apply_filters( |
|
111 | + 'FHEE__EventEspresso_core_services_commands_CommandHandlerManager__getCommandHandler__command_handler', |
|
112 | + str_replace('Command', 'CommandHandler', $command_name), |
|
113 | + $command |
|
114 | + ); |
|
115 | + $handler = null; |
|
116 | + // has a command handler already been set for this class ? |
|
117 | + // if not, can we find one via the FQCN ? |
|
118 | + if (isset($this->command_handlers[ $command_name ])) { |
|
119 | + $handler = $this->command_handlers[ $command_name ]; |
|
120 | + } elseif (class_exists($command_handler)) { |
|
121 | + $handler = $this->loader->getShared($command_handler); |
|
122 | + } |
|
123 | + // if Handler requires an instance of the CommandBus, but that has not yet been set |
|
124 | + if ($handler instanceof CompositeCommandHandler && ! $handler->commandBus() instanceof CommandBusInterface) { |
|
125 | + if (! $command_bus instanceof CommandBusInterface) { |
|
126 | + throw new DomainException( |
|
127 | + esc_html__( |
|
128 | + 'CompositeCommandHandler classes require an instance of the CommandBus.', |
|
129 | + 'event_espresso' |
|
130 | + ) |
|
131 | + ); |
|
132 | + } |
|
133 | + $handler->setCommandBus($command_bus); |
|
134 | + } |
|
135 | + if ($handler instanceof CommandHandlerInterface) { |
|
136 | + return $handler; |
|
137 | + } |
|
138 | + throw new CommandHandlerNotFoundException($command_handler); |
|
139 | + } |
|
140 | 140 | } |
@@ -34,74 +34,74 @@ |
||
34 | 34 | class CommandBus implements CommandBusInterface |
35 | 35 | { |
36 | 36 | |
37 | - /** |
|
38 | - * @type CommandHandlerManagerInterface $command_handler_manager |
|
39 | - */ |
|
40 | - private $command_handler_manager; |
|
37 | + /** |
|
38 | + * @type CommandHandlerManagerInterface $command_handler_manager |
|
39 | + */ |
|
40 | + private $command_handler_manager; |
|
41 | 41 | |
42 | - /** |
|
43 | - * @type CommandBusMiddlewareInterface[] $command_bus_middleware |
|
44 | - */ |
|
45 | - private $command_bus_middleware; |
|
42 | + /** |
|
43 | + * @type CommandBusMiddlewareInterface[] $command_bus_middleware |
|
44 | + */ |
|
45 | + private $command_bus_middleware; |
|
46 | 46 | |
47 | 47 | |
48 | - /** |
|
49 | - * CommandBus constructor |
|
50 | - * |
|
51 | - * @param CommandHandlerManagerInterface $command_handler_manager |
|
52 | - * @param CommandBusMiddlewareInterface[] $command_bus_middleware |
|
53 | - */ |
|
54 | - public function __construct( |
|
55 | - CommandHandlerManagerInterface $command_handler_manager, |
|
56 | - array $command_bus_middleware = array() |
|
57 | - ) { |
|
58 | - $this->command_handler_manager = $command_handler_manager; |
|
59 | - $this->command_bus_middleware = is_array($command_bus_middleware) |
|
60 | - ? $command_bus_middleware |
|
61 | - : array($command_bus_middleware); |
|
62 | - } |
|
48 | + /** |
|
49 | + * CommandBus constructor |
|
50 | + * |
|
51 | + * @param CommandHandlerManagerInterface $command_handler_manager |
|
52 | + * @param CommandBusMiddlewareInterface[] $command_bus_middleware |
|
53 | + */ |
|
54 | + public function __construct( |
|
55 | + CommandHandlerManagerInterface $command_handler_manager, |
|
56 | + array $command_bus_middleware = array() |
|
57 | + ) { |
|
58 | + $this->command_handler_manager = $command_handler_manager; |
|
59 | + $this->command_bus_middleware = is_array($command_bus_middleware) |
|
60 | + ? $command_bus_middleware |
|
61 | + : array($command_bus_middleware); |
|
62 | + } |
|
63 | 63 | |
64 | 64 | |
65 | - /** |
|
66 | - * @return CommandHandlerManagerInterface |
|
67 | - */ |
|
68 | - public function getCommandHandlerManager() |
|
69 | - { |
|
70 | - return $this->command_handler_manager; |
|
71 | - } |
|
65 | + /** |
|
66 | + * @return CommandHandlerManagerInterface |
|
67 | + */ |
|
68 | + public function getCommandHandlerManager() |
|
69 | + { |
|
70 | + return $this->command_handler_manager; |
|
71 | + } |
|
72 | 72 | |
73 | 73 | |
74 | - /** |
|
75 | - * @param CommandInterface $command |
|
76 | - * @return mixed |
|
77 | - * @throws InvalidDataTypeException |
|
78 | - * @throws InvalidCommandBusMiddlewareException |
|
79 | - */ |
|
80 | - public function execute($command) |
|
81 | - { |
|
82 | - if (! $command instanceof CommandInterface) { |
|
83 | - throw new InvalidDataTypeException(__METHOD__ . '( $command )', $command, 'CommandInterface'); |
|
84 | - } |
|
85 | - // we're going to add the Command Handler as a callable |
|
86 | - // that will get run at the end of our middleware stack |
|
87 | - // can't pass $this to a Closure, so use a named variable |
|
88 | - $command_bus = $this; |
|
89 | - $middleware = static function ($command) use ($command_bus) { |
|
90 | - return $command_bus->getCommandHandlerManager() |
|
91 | - ->getCommandHandler($command, $command_bus) |
|
92 | - ->verify($command) |
|
93 | - ->handle($command); |
|
94 | - }; |
|
95 | - // now build the rest of the middleware stack |
|
96 | - while ($command_bus_middleware = array_pop($this->command_bus_middleware)) { |
|
97 | - if (! $command_bus_middleware instanceof CommandBusMiddlewareInterface) { |
|
98 | - throw new InvalidCommandBusMiddlewareException($command_bus_middleware); |
|
99 | - } |
|
100 | - $middleware = static function ($command) use ($command_bus_middleware, $middleware) { |
|
101 | - return $command_bus_middleware->handle($command, $middleware); |
|
102 | - }; |
|
103 | - } |
|
104 | - // and finally, pass the command into the stack and return the results |
|
105 | - return $middleware($command); |
|
106 | - } |
|
74 | + /** |
|
75 | + * @param CommandInterface $command |
|
76 | + * @return mixed |
|
77 | + * @throws InvalidDataTypeException |
|
78 | + * @throws InvalidCommandBusMiddlewareException |
|
79 | + */ |
|
80 | + public function execute($command) |
|
81 | + { |
|
82 | + if (! $command instanceof CommandInterface) { |
|
83 | + throw new InvalidDataTypeException(__METHOD__ . '( $command )', $command, 'CommandInterface'); |
|
84 | + } |
|
85 | + // we're going to add the Command Handler as a callable |
|
86 | + // that will get run at the end of our middleware stack |
|
87 | + // can't pass $this to a Closure, so use a named variable |
|
88 | + $command_bus = $this; |
|
89 | + $middleware = static function ($command) use ($command_bus) { |
|
90 | + return $command_bus->getCommandHandlerManager() |
|
91 | + ->getCommandHandler($command, $command_bus) |
|
92 | + ->verify($command) |
|
93 | + ->handle($command); |
|
94 | + }; |
|
95 | + // now build the rest of the middleware stack |
|
96 | + while ($command_bus_middleware = array_pop($this->command_bus_middleware)) { |
|
97 | + if (! $command_bus_middleware instanceof CommandBusMiddlewareInterface) { |
|
98 | + throw new InvalidCommandBusMiddlewareException($command_bus_middleware); |
|
99 | + } |
|
100 | + $middleware = static function ($command) use ($command_bus_middleware, $middleware) { |
|
101 | + return $command_bus_middleware->handle($command, $middleware); |
|
102 | + }; |
|
103 | + } |
|
104 | + // and finally, pass the command into the stack and return the results |
|
105 | + return $middleware($command); |
|
106 | + } |
|
107 | 107 | } |
@@ -79,14 +79,14 @@ discard block |
||
79 | 79 | */ |
80 | 80 | public function execute($command) |
81 | 81 | { |
82 | - if (! $command instanceof CommandInterface) { |
|
83 | - throw new InvalidDataTypeException(__METHOD__ . '( $command )', $command, 'CommandInterface'); |
|
82 | + if ( ! $command instanceof CommandInterface) { |
|
83 | + throw new InvalidDataTypeException(__METHOD__.'( $command )', $command, 'CommandInterface'); |
|
84 | 84 | } |
85 | 85 | // we're going to add the Command Handler as a callable |
86 | 86 | // that will get run at the end of our middleware stack |
87 | 87 | // can't pass $this to a Closure, so use a named variable |
88 | 88 | $command_bus = $this; |
89 | - $middleware = static function ($command) use ($command_bus) { |
|
89 | + $middleware = static function($command) use ($command_bus) { |
|
90 | 90 | return $command_bus->getCommandHandlerManager() |
91 | 91 | ->getCommandHandler($command, $command_bus) |
92 | 92 | ->verify($command) |
@@ -94,10 +94,10 @@ discard block |
||
94 | 94 | }; |
95 | 95 | // now build the rest of the middleware stack |
96 | 96 | while ($command_bus_middleware = array_pop($this->command_bus_middleware)) { |
97 | - if (! $command_bus_middleware instanceof CommandBusMiddlewareInterface) { |
|
97 | + if ( ! $command_bus_middleware instanceof CommandBusMiddlewareInterface) { |
|
98 | 98 | throw new InvalidCommandBusMiddlewareException($command_bus_middleware); |
99 | 99 | } |
100 | - $middleware = static function ($command) use ($command_bus_middleware, $middleware) { |
|
100 | + $middleware = static function($command) use ($command_bus_middleware, $middleware) { |
|
101 | 101 | return $command_bus_middleware->handle($command, $middleware); |
102 | 102 | }; |
103 | 103 | } |
@@ -12,2671 +12,2671 @@ |
||
12 | 12 | class Events_Admin_Page extends EE_Admin_Page_CPT |
13 | 13 | { |
14 | 14 | |
15 | - /** |
|
16 | - * This will hold the event object for event_details screen. |
|
17 | - * |
|
18 | - * @access protected |
|
19 | - * @var EE_Event $_event |
|
20 | - */ |
|
21 | - protected $_event; |
|
22 | - |
|
23 | - |
|
24 | - /** |
|
25 | - * This will hold the category object for category_details screen. |
|
26 | - * |
|
27 | - * @var stdClass $_category |
|
28 | - */ |
|
29 | - protected $_category; |
|
30 | - |
|
31 | - |
|
32 | - /** |
|
33 | - * This will hold the event model instance |
|
34 | - * |
|
35 | - * @var EEM_Event $_event_model |
|
36 | - */ |
|
37 | - protected $_event_model; |
|
38 | - |
|
39 | - |
|
40 | - /** |
|
41 | - * @var EE_Event |
|
42 | - */ |
|
43 | - protected $_cpt_model_obj = false; |
|
44 | - |
|
45 | - |
|
46 | - /** |
|
47 | - * Initialize page props for this admin page group. |
|
48 | - */ |
|
49 | - protected function _init_page_props() |
|
50 | - { |
|
51 | - $this->page_slug = EVENTS_PG_SLUG; |
|
52 | - $this->page_label = EVENTS_LABEL; |
|
53 | - $this->_admin_base_url = EVENTS_ADMIN_URL; |
|
54 | - $this->_admin_base_path = EVENTS_ADMIN; |
|
55 | - $this->_cpt_model_names = array( |
|
56 | - 'create_new' => 'EEM_Event', |
|
57 | - 'edit' => 'EEM_Event', |
|
58 | - ); |
|
59 | - $this->_cpt_edit_routes = array( |
|
60 | - 'espresso_events' => 'edit', |
|
61 | - ); |
|
62 | - add_action( |
|
63 | - 'AHEE__EE_Admin_Page_CPT__set_model_object__after_set_object', |
|
64 | - array($this, 'verify_event_edit'), |
|
65 | - 10, |
|
66 | - 2 |
|
67 | - ); |
|
68 | - } |
|
69 | - |
|
70 | - |
|
71 | - /** |
|
72 | - * Sets the ajax hooks used for this admin page group. |
|
73 | - */ |
|
74 | - protected function _ajax_hooks() |
|
75 | - { |
|
76 | - add_action('wp_ajax_ee_save_timezone_setting', array($this, 'save_timezonestring_setting')); |
|
77 | - } |
|
78 | - |
|
79 | - |
|
80 | - /** |
|
81 | - * Sets the page properties for this admin page group. |
|
82 | - */ |
|
83 | - protected function _define_page_props() |
|
84 | - { |
|
85 | - $this->_admin_page_title = EVENTS_LABEL; |
|
86 | - $this->_labels = array( |
|
87 | - 'buttons' => array( |
|
88 | - 'add' => esc_html__('Add New Event', 'event_espresso'), |
|
89 | - 'edit' => esc_html__('Edit Event', 'event_espresso'), |
|
90 | - 'delete' => esc_html__('Delete Event', 'event_espresso'), |
|
91 | - 'add_category' => esc_html__('Add New Category', 'event_espresso'), |
|
92 | - 'edit_category' => esc_html__('Edit Category', 'event_espresso'), |
|
93 | - 'delete_category' => esc_html__('Delete Category', 'event_espresso'), |
|
94 | - ), |
|
95 | - 'editor_title' => array( |
|
96 | - 'espresso_events' => esc_html__('Enter event title here', 'event_espresso'), |
|
97 | - ), |
|
98 | - 'publishbox' => array( |
|
99 | - 'create_new' => esc_html__('Save New Event', 'event_espresso'), |
|
100 | - 'edit' => esc_html__('Update Event', 'event_espresso'), |
|
101 | - 'add_category' => esc_html__('Save New Category', 'event_espresso'), |
|
102 | - 'edit_category' => esc_html__('Update Category', 'event_espresso'), |
|
103 | - 'template_settings' => esc_html__('Update Settings', 'event_espresso'), |
|
104 | - ), |
|
105 | - ); |
|
106 | - } |
|
107 | - |
|
108 | - |
|
109 | - /** |
|
110 | - * Sets the page routes property for this admin page group. |
|
111 | - */ |
|
112 | - protected function _set_page_routes() |
|
113 | - { |
|
114 | - // load formatter helper |
|
115 | - // load field generator helper |
|
116 | - // is there a evt_id in the request? |
|
117 | - $evt_id = ! empty($this->_req_data['EVT_ID']) && ! is_array($this->_req_data['EVT_ID']) |
|
118 | - ? $this->_req_data['EVT_ID'] |
|
119 | - : 0; |
|
120 | - $evt_id = ! empty($this->_req_data['post']) ? $this->_req_data['post'] : $evt_id; |
|
121 | - $this->_page_routes = array( |
|
122 | - 'default' => array( |
|
123 | - 'func' => '_events_overview_list_table', |
|
124 | - 'capability' => 'ee_read_events', |
|
125 | - ), |
|
126 | - 'create_new' => array( |
|
127 | - 'func' => '_create_new_cpt_item', |
|
128 | - 'capability' => 'ee_edit_events', |
|
129 | - ), |
|
130 | - 'edit' => array( |
|
131 | - 'func' => '_edit_cpt_item', |
|
132 | - 'capability' => 'ee_edit_event', |
|
133 | - 'obj_id' => $evt_id, |
|
134 | - ), |
|
135 | - 'copy_event' => array( |
|
136 | - 'func' => '_copy_events', |
|
137 | - 'capability' => 'ee_edit_event', |
|
138 | - 'obj_id' => $evt_id, |
|
139 | - 'noheader' => true, |
|
140 | - ), |
|
141 | - 'trash_event' => array( |
|
142 | - 'func' => '_trash_or_restore_event', |
|
143 | - 'args' => array('event_status' => 'trash'), |
|
144 | - 'capability' => 'ee_delete_event', |
|
145 | - 'obj_id' => $evt_id, |
|
146 | - 'noheader' => true, |
|
147 | - ), |
|
148 | - 'trash_events' => array( |
|
149 | - 'func' => '_trash_or_restore_events', |
|
150 | - 'args' => array('event_status' => 'trash'), |
|
151 | - 'capability' => 'ee_delete_events', |
|
152 | - 'noheader' => true, |
|
153 | - ), |
|
154 | - 'restore_event' => array( |
|
155 | - 'func' => '_trash_or_restore_event', |
|
156 | - 'args' => array('event_status' => 'draft'), |
|
157 | - 'capability' => 'ee_delete_event', |
|
158 | - 'obj_id' => $evt_id, |
|
159 | - 'noheader' => true, |
|
160 | - ), |
|
161 | - 'restore_events' => array( |
|
162 | - 'func' => '_trash_or_restore_events', |
|
163 | - 'args' => array('event_status' => 'draft'), |
|
164 | - 'capability' => 'ee_delete_events', |
|
165 | - 'noheader' => true, |
|
166 | - ), |
|
167 | - 'delete_event' => array( |
|
168 | - 'func' => '_delete_event', |
|
169 | - 'capability' => 'ee_delete_event', |
|
170 | - 'obj_id' => $evt_id, |
|
171 | - 'noheader' => true, |
|
172 | - ), |
|
173 | - 'delete_events' => array( |
|
174 | - 'func' => '_delete_events', |
|
175 | - 'capability' => 'ee_delete_events', |
|
176 | - 'noheader' => true, |
|
177 | - ), |
|
178 | - 'view_report' => array( |
|
179 | - 'func' => '_view_report', |
|
180 | - 'capablity' => 'ee_edit_events', |
|
181 | - ), |
|
182 | - 'default_event_settings' => array( |
|
183 | - 'func' => '_default_event_settings', |
|
184 | - 'capability' => 'manage_options', |
|
185 | - ), |
|
186 | - 'update_default_event_settings' => array( |
|
187 | - 'func' => '_update_default_event_settings', |
|
188 | - 'capability' => 'manage_options', |
|
189 | - 'noheader' => true, |
|
190 | - ), |
|
191 | - 'template_settings' => array( |
|
192 | - 'func' => '_template_settings', |
|
193 | - 'capability' => 'manage_options', |
|
194 | - ), |
|
195 | - // event category tab related |
|
196 | - 'add_category' => array( |
|
197 | - 'func' => '_category_details', |
|
198 | - 'capability' => 'ee_edit_event_category', |
|
199 | - 'args' => array('add'), |
|
200 | - ), |
|
201 | - 'edit_category' => array( |
|
202 | - 'func' => '_category_details', |
|
203 | - 'capability' => 'ee_edit_event_category', |
|
204 | - 'args' => array('edit'), |
|
205 | - ), |
|
206 | - 'delete_categories' => array( |
|
207 | - 'func' => '_delete_categories', |
|
208 | - 'capability' => 'ee_delete_event_category', |
|
209 | - 'noheader' => true, |
|
210 | - ), |
|
211 | - 'delete_category' => array( |
|
212 | - 'func' => '_delete_categories', |
|
213 | - 'capability' => 'ee_delete_event_category', |
|
214 | - 'noheader' => true, |
|
215 | - ), |
|
216 | - 'insert_category' => array( |
|
217 | - 'func' => '_insert_or_update_category', |
|
218 | - 'args' => array('new_category' => true), |
|
219 | - 'capability' => 'ee_edit_event_category', |
|
220 | - 'noheader' => true, |
|
221 | - ), |
|
222 | - 'update_category' => array( |
|
223 | - 'func' => '_insert_or_update_category', |
|
224 | - 'args' => array('new_category' => false), |
|
225 | - 'capability' => 'ee_edit_event_category', |
|
226 | - 'noheader' => true, |
|
227 | - ), |
|
228 | - 'category_list' => array( |
|
229 | - 'func' => '_category_list_table', |
|
230 | - 'capability' => 'ee_manage_event_categories', |
|
231 | - ), |
|
232 | - ); |
|
233 | - } |
|
234 | - |
|
235 | - |
|
236 | - /** |
|
237 | - * Set the _page_config property for this admin page group. |
|
238 | - */ |
|
239 | - protected function _set_page_config() |
|
240 | - { |
|
241 | - $this->_page_config = array( |
|
242 | - 'default' => array( |
|
243 | - 'nav' => array( |
|
244 | - 'label' => esc_html__('Overview', 'event_espresso'), |
|
245 | - 'order' => 10, |
|
246 | - ), |
|
247 | - 'list_table' => 'Events_Admin_List_Table', |
|
248 | - 'help_tabs' => array( |
|
249 | - 'events_overview_help_tab' => array( |
|
250 | - 'title' => esc_html__('Events Overview', 'event_espresso'), |
|
251 | - 'filename' => 'events_overview', |
|
252 | - ), |
|
253 | - 'events_overview_table_column_headings_help_tab' => array( |
|
254 | - 'title' => esc_html__('Events Overview Table Column Headings', 'event_espresso'), |
|
255 | - 'filename' => 'events_overview_table_column_headings', |
|
256 | - ), |
|
257 | - 'events_overview_filters_help_tab' => array( |
|
258 | - 'title' => esc_html__('Events Overview Filters', 'event_espresso'), |
|
259 | - 'filename' => 'events_overview_filters', |
|
260 | - ), |
|
261 | - 'events_overview_view_help_tab' => array( |
|
262 | - 'title' => esc_html__('Events Overview Views', 'event_espresso'), |
|
263 | - 'filename' => 'events_overview_views', |
|
264 | - ), |
|
265 | - 'events_overview_other_help_tab' => array( |
|
266 | - 'title' => esc_html__('Events Overview Other', 'event_espresso'), |
|
267 | - 'filename' => 'events_overview_other', |
|
268 | - ), |
|
269 | - ), |
|
270 | - 'help_tour' => array( |
|
271 | - 'Event_Overview_Help_Tour', |
|
272 | - // 'New_Features_Test_Help_Tour' for testing multiple help tour |
|
273 | - ), |
|
274 | - 'qtips' => array( |
|
275 | - 'EE_Event_List_Table_Tips', |
|
276 | - ), |
|
277 | - 'require_nonce' => false, |
|
278 | - ), |
|
279 | - 'create_new' => array( |
|
280 | - 'nav' => array( |
|
281 | - 'label' => esc_html__('Add Event', 'event_espresso'), |
|
282 | - 'order' => 5, |
|
283 | - 'persistent' => false, |
|
284 | - ), |
|
285 | - 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
286 | - 'help_tabs' => array( |
|
287 | - 'event_editor_help_tab' => array( |
|
288 | - 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
289 | - 'filename' => 'event_editor', |
|
290 | - ), |
|
291 | - 'event_editor_title_richtexteditor_help_tab' => array( |
|
292 | - 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
293 | - 'filename' => 'event_editor_title_richtexteditor', |
|
294 | - ), |
|
295 | - 'event_editor_venue_details_help_tab' => array( |
|
296 | - 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
297 | - 'filename' => 'event_editor_venue_details', |
|
298 | - ), |
|
299 | - 'event_editor_event_datetimes_help_tab' => array( |
|
300 | - 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
301 | - 'filename' => 'event_editor_event_datetimes', |
|
302 | - ), |
|
303 | - 'event_editor_event_tickets_help_tab' => array( |
|
304 | - 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
305 | - 'filename' => 'event_editor_event_tickets', |
|
306 | - ), |
|
307 | - 'event_editor_event_registration_options_help_tab' => array( |
|
308 | - 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
309 | - 'filename' => 'event_editor_event_registration_options', |
|
310 | - ), |
|
311 | - 'event_editor_tags_categories_help_tab' => array( |
|
312 | - 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
313 | - 'filename' => 'event_editor_tags_categories', |
|
314 | - ), |
|
315 | - 'event_editor_questions_registrants_help_tab' => array( |
|
316 | - 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
317 | - 'filename' => 'event_editor_questions_registrants', |
|
318 | - ), |
|
319 | - 'event_editor_save_new_event_help_tab' => array( |
|
320 | - 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
321 | - 'filename' => 'event_editor_save_new_event', |
|
322 | - ), |
|
323 | - 'event_editor_other_help_tab' => array( |
|
324 | - 'title' => esc_html__('Event Other', 'event_espresso'), |
|
325 | - 'filename' => 'event_editor_other', |
|
326 | - ), |
|
327 | - ), |
|
328 | - 'help_tour' => array( |
|
329 | - 'Event_Editor_Help_Tour', |
|
330 | - ), |
|
331 | - 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
332 | - 'require_nonce' => false, |
|
333 | - ), |
|
334 | - 'edit' => array( |
|
335 | - 'nav' => array( |
|
336 | - 'label' => esc_html__('Edit Event', 'event_espresso'), |
|
337 | - 'order' => 5, |
|
338 | - 'persistent' => false, |
|
339 | - 'url' => isset($this->_req_data['post']) |
|
340 | - ? EE_Admin_Page::add_query_args_and_nonce( |
|
341 | - array('post' => $this->_req_data['post'], 'action' => 'edit'), |
|
342 | - $this->_current_page_view_url |
|
343 | - ) |
|
344 | - : $this->_admin_base_url, |
|
345 | - ), |
|
346 | - 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
347 | - 'help_tabs' => array( |
|
348 | - 'event_editor_help_tab' => array( |
|
349 | - 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
350 | - 'filename' => 'event_editor', |
|
351 | - ), |
|
352 | - 'event_editor_title_richtexteditor_help_tab' => array( |
|
353 | - 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
354 | - 'filename' => 'event_editor_title_richtexteditor', |
|
355 | - ), |
|
356 | - 'event_editor_venue_details_help_tab' => array( |
|
357 | - 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
358 | - 'filename' => 'event_editor_venue_details', |
|
359 | - ), |
|
360 | - 'event_editor_event_datetimes_help_tab' => array( |
|
361 | - 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
362 | - 'filename' => 'event_editor_event_datetimes', |
|
363 | - ), |
|
364 | - 'event_editor_event_tickets_help_tab' => array( |
|
365 | - 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
366 | - 'filename' => 'event_editor_event_tickets', |
|
367 | - ), |
|
368 | - 'event_editor_event_registration_options_help_tab' => array( |
|
369 | - 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
370 | - 'filename' => 'event_editor_event_registration_options', |
|
371 | - ), |
|
372 | - 'event_editor_tags_categories_help_tab' => array( |
|
373 | - 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
374 | - 'filename' => 'event_editor_tags_categories', |
|
375 | - ), |
|
376 | - 'event_editor_questions_registrants_help_tab' => array( |
|
377 | - 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
378 | - 'filename' => 'event_editor_questions_registrants', |
|
379 | - ), |
|
380 | - 'event_editor_save_new_event_help_tab' => array( |
|
381 | - 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
382 | - 'filename' => 'event_editor_save_new_event', |
|
383 | - ), |
|
384 | - 'event_editor_other_help_tab' => array( |
|
385 | - 'title' => esc_html__('Event Other', 'event_espresso'), |
|
386 | - 'filename' => 'event_editor_other', |
|
387 | - ), |
|
388 | - ), |
|
389 | - 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
390 | - 'require_nonce' => false, |
|
391 | - ), |
|
392 | - 'default_event_settings' => array( |
|
393 | - 'nav' => array( |
|
394 | - 'label' => esc_html__('Default Settings', 'event_espresso'), |
|
395 | - 'order' => 40, |
|
396 | - ), |
|
397 | - 'metaboxes' => array_merge($this->_default_espresso_metaboxes, array('_publish_post_box')), |
|
398 | - 'labels' => array( |
|
399 | - 'publishbox' => esc_html__('Update Settings', 'event_espresso'), |
|
400 | - ), |
|
401 | - 'help_tabs' => array( |
|
402 | - 'default_settings_help_tab' => array( |
|
403 | - 'title' => esc_html__('Default Event Settings', 'event_espresso'), |
|
404 | - 'filename' => 'events_default_settings', |
|
405 | - ), |
|
406 | - 'default_settings_status_help_tab' => array( |
|
407 | - 'title' => esc_html__('Default Registration Status', 'event_espresso'), |
|
408 | - 'filename' => 'events_default_settings_status', |
|
409 | - ), |
|
410 | - 'default_maximum_tickets_help_tab' => array( |
|
411 | - 'title' => esc_html__('Default Maximum Tickets Per Order', 'event_espresso'), |
|
412 | - 'filename' => 'events_default_settings_max_tickets', |
|
413 | - ), |
|
414 | - ), |
|
415 | - 'help_tour' => array('Event_Default_Settings_Help_Tour'), |
|
416 | - 'require_nonce' => false, |
|
417 | - ), |
|
418 | - // template settings |
|
419 | - 'template_settings' => array( |
|
420 | - 'nav' => array( |
|
421 | - 'label' => esc_html__('Templates', 'event_espresso'), |
|
422 | - 'order' => 30, |
|
423 | - ), |
|
424 | - 'metaboxes' => $this->_default_espresso_metaboxes, |
|
425 | - 'help_tabs' => array( |
|
426 | - 'general_settings_templates_help_tab' => array( |
|
427 | - 'title' => esc_html__('Templates', 'event_espresso'), |
|
428 | - 'filename' => 'general_settings_templates', |
|
429 | - ), |
|
430 | - ), |
|
431 | - 'help_tour' => array('Templates_Help_Tour'), |
|
432 | - 'require_nonce' => false, |
|
433 | - ), |
|
434 | - // event category stuff |
|
435 | - 'add_category' => array( |
|
436 | - 'nav' => array( |
|
437 | - 'label' => esc_html__('Add Category', 'event_espresso'), |
|
438 | - 'order' => 15, |
|
439 | - 'persistent' => false, |
|
440 | - ), |
|
441 | - 'help_tabs' => array( |
|
442 | - 'add_category_help_tab' => array( |
|
443 | - 'title' => esc_html__('Add New Event Category', 'event_espresso'), |
|
444 | - 'filename' => 'events_add_category', |
|
445 | - ), |
|
446 | - ), |
|
447 | - 'help_tour' => array('Event_Add_Category_Help_Tour'), |
|
448 | - 'metaboxes' => array('_publish_post_box'), |
|
449 | - 'require_nonce' => false, |
|
450 | - ), |
|
451 | - 'edit_category' => array( |
|
452 | - 'nav' => array( |
|
453 | - 'label' => esc_html__('Edit Category', 'event_espresso'), |
|
454 | - 'order' => 15, |
|
455 | - 'persistent' => false, |
|
456 | - 'url' => isset($this->_req_data['EVT_CAT_ID']) |
|
457 | - ? add_query_arg( |
|
458 | - array('EVT_CAT_ID' => $this->_req_data['EVT_CAT_ID']), |
|
459 | - $this->_current_page_view_url |
|
460 | - ) |
|
461 | - : $this->_admin_base_url, |
|
462 | - ), |
|
463 | - 'help_tabs' => array( |
|
464 | - 'edit_category_help_tab' => array( |
|
465 | - 'title' => esc_html__('Edit Event Category', 'event_espresso'), |
|
466 | - 'filename' => 'events_edit_category', |
|
467 | - ), |
|
468 | - ), |
|
469 | - /*'help_tour' => array('Event_Edit_Category_Help_Tour'),*/ |
|
470 | - 'metaboxes' => array('_publish_post_box'), |
|
471 | - 'require_nonce' => false, |
|
472 | - ), |
|
473 | - 'category_list' => array( |
|
474 | - 'nav' => array( |
|
475 | - 'label' => esc_html__('Categories', 'event_espresso'), |
|
476 | - 'order' => 20, |
|
477 | - ), |
|
478 | - 'list_table' => 'Event_Categories_Admin_List_Table', |
|
479 | - 'help_tabs' => array( |
|
480 | - 'events_categories_help_tab' => array( |
|
481 | - 'title' => esc_html__('Event Categories', 'event_espresso'), |
|
482 | - 'filename' => 'events_categories', |
|
483 | - ), |
|
484 | - 'events_categories_table_column_headings_help_tab' => array( |
|
485 | - 'title' => esc_html__('Event Categories Table Column Headings', 'event_espresso'), |
|
486 | - 'filename' => 'events_categories_table_column_headings', |
|
487 | - ), |
|
488 | - 'events_categories_view_help_tab' => array( |
|
489 | - 'title' => esc_html__('Event Categories Views', 'event_espresso'), |
|
490 | - 'filename' => 'events_categories_views', |
|
491 | - ), |
|
492 | - 'events_categories_other_help_tab' => array( |
|
493 | - 'title' => esc_html__('Event Categories Other', 'event_espresso'), |
|
494 | - 'filename' => 'events_categories_other', |
|
495 | - ), |
|
496 | - ), |
|
497 | - 'help_tour' => array( |
|
498 | - 'Event_Categories_Help_Tour', |
|
499 | - ), |
|
500 | - 'metaboxes' => $this->_default_espresso_metaboxes, |
|
501 | - 'require_nonce' => false, |
|
502 | - ), |
|
503 | - ); |
|
504 | - } |
|
505 | - |
|
506 | - |
|
507 | - /** |
|
508 | - * Used to register any global screen options if necessary for every route in this admin page group. |
|
509 | - */ |
|
510 | - protected function _add_screen_options() |
|
511 | - { |
|
512 | - } |
|
513 | - |
|
514 | - |
|
515 | - /** |
|
516 | - * Implementing the screen options for the 'default' route. |
|
517 | - */ |
|
518 | - protected function _add_screen_options_default() |
|
519 | - { |
|
520 | - $this->_per_page_screen_option(); |
|
521 | - } |
|
522 | - |
|
523 | - |
|
524 | - /** |
|
525 | - * Implementing screen options for the category list route. |
|
526 | - */ |
|
527 | - protected function _add_screen_options_category_list() |
|
528 | - { |
|
529 | - $page_title = $this->_admin_page_title; |
|
530 | - $this->_admin_page_title = esc_html__('Categories', 'event_espresso'); |
|
531 | - $this->_per_page_screen_option(); |
|
532 | - $this->_admin_page_title = $page_title; |
|
533 | - } |
|
534 | - |
|
535 | - |
|
536 | - /** |
|
537 | - * Used to register any global feature pointers for the admin page group. |
|
538 | - */ |
|
539 | - protected function _add_feature_pointers() |
|
540 | - { |
|
541 | - } |
|
542 | - |
|
543 | - |
|
544 | - /** |
|
545 | - * Registers and enqueues any global scripts and styles for the entire admin page group. |
|
546 | - */ |
|
547 | - public function load_scripts_styles() |
|
548 | - { |
|
549 | - wp_register_style( |
|
550 | - 'events-admin-css', |
|
551 | - EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
552 | - array(), |
|
553 | - EVENT_ESPRESSO_VERSION |
|
554 | - ); |
|
555 | - wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
556 | - wp_enqueue_style('events-admin-css'); |
|
557 | - wp_enqueue_style('ee-cat-admin'); |
|
558 | - // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
|
559 | - // registers for all views |
|
560 | - // scripts |
|
561 | - wp_register_script( |
|
562 | - 'event_editor_js', |
|
563 | - EVENTS_ASSETS_URL . 'event_editor.js', |
|
564 | - array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
|
565 | - EVENT_ESPRESSO_VERSION, |
|
566 | - true |
|
567 | - ); |
|
568 | - } |
|
569 | - |
|
570 | - |
|
571 | - /** |
|
572 | - * Enqueuing scripts and styles specific to this view |
|
573 | - */ |
|
574 | - public function load_scripts_styles_create_new() |
|
575 | - { |
|
576 | - $this->load_scripts_styles_edit(); |
|
577 | - } |
|
578 | - |
|
579 | - |
|
580 | - /** |
|
581 | - * Enqueuing scripts and styles specific to this view |
|
582 | - */ |
|
583 | - public function load_scripts_styles_edit() |
|
584 | - { |
|
585 | - // styles |
|
586 | - wp_enqueue_style('espresso-ui-theme'); |
|
587 | - wp_register_style( |
|
588 | - 'event-editor-css', |
|
589 | - EVENTS_ASSETS_URL . 'event-editor.css', |
|
590 | - array('ee-admin-css'), |
|
591 | - EVENT_ESPRESSO_VERSION |
|
592 | - ); |
|
593 | - wp_enqueue_style('event-editor-css'); |
|
594 | - // scripts |
|
595 | - wp_register_script( |
|
596 | - 'event-datetime-metabox', |
|
597 | - EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
598 | - array('event_editor_js', 'ee-datepicker'), |
|
599 | - EVENT_ESPRESSO_VERSION |
|
600 | - ); |
|
601 | - wp_enqueue_script('event-datetime-metabox'); |
|
602 | - } |
|
603 | - |
|
604 | - |
|
605 | - /** |
|
606 | - * Populating the _views property for the category list table view. |
|
607 | - */ |
|
608 | - protected function _set_list_table_views_category_list() |
|
609 | - { |
|
610 | - $this->_views = array( |
|
611 | - 'all' => array( |
|
612 | - 'slug' => 'all', |
|
613 | - 'label' => esc_html__('All', 'event_espresso'), |
|
614 | - 'count' => 0, |
|
615 | - 'bulk_action' => array( |
|
616 | - 'delete_categories' => esc_html__('Delete Permanently', 'event_espresso'), |
|
617 | - ), |
|
618 | - ), |
|
619 | - ); |
|
620 | - } |
|
621 | - |
|
622 | - |
|
623 | - /** |
|
624 | - * For adding anything that fires on the admin_init hook for any route within this admin page group. |
|
625 | - */ |
|
626 | - public function admin_init() |
|
627 | - { |
|
628 | - EE_Registry::$i18n_js_strings['image_confirm'] = esc_html__( |
|
629 | - 'Do you really want to delete this image? Please remember to update your event to complete the removal.', |
|
630 | - 'event_espresso' |
|
631 | - ); |
|
632 | - } |
|
633 | - |
|
634 | - |
|
635 | - /** |
|
636 | - * For adding anything that should be triggered on the admin_notices hook for any route within this admin page |
|
637 | - * group. |
|
638 | - */ |
|
639 | - public function admin_notices() |
|
640 | - { |
|
641 | - } |
|
642 | - |
|
643 | - |
|
644 | - /** |
|
645 | - * For adding anything that should be triggered on the `admin_print_footer_scripts` hook for any route within |
|
646 | - * this admin page group. |
|
647 | - */ |
|
648 | - public function admin_footer_scripts() |
|
649 | - { |
|
650 | - } |
|
651 | - |
|
652 | - |
|
653 | - /** |
|
654 | - * Call this function to verify if an event is public and has tickets for sale. If it does, then we need to show a |
|
655 | - * warning (via EE_Error::add_error()); |
|
656 | - * |
|
657 | - * @param EE_Event $event Event object |
|
658 | - * @param string $req_type |
|
659 | - * @return void |
|
660 | - * @throws EE_Error |
|
661 | - * @access public |
|
662 | - */ |
|
663 | - public function verify_event_edit($event = null, $req_type = '') |
|
664 | - { |
|
665 | - // don't need to do this when processing |
|
666 | - if (! empty($req_type)) { |
|
667 | - return; |
|
668 | - } |
|
669 | - // no event? |
|
670 | - if (empty($event)) { |
|
671 | - // set event |
|
672 | - $event = $this->_cpt_model_obj; |
|
673 | - } |
|
674 | - // STILL no event? |
|
675 | - if (! $event instanceof EE_Event) { |
|
676 | - return; |
|
677 | - } |
|
678 | - $orig_status = $event->status(); |
|
679 | - // first check if event is active. |
|
680 | - if ($orig_status === EEM_Event::cancelled |
|
681 | - || $orig_status === EEM_Event::postponed |
|
682 | - || $event->is_expired() |
|
683 | - || $event->is_inactive() |
|
684 | - ) { |
|
685 | - return; |
|
686 | - } |
|
687 | - // made it here so it IS active... next check that any of the tickets are sold. |
|
688 | - if ($event->is_sold_out(true)) { |
|
689 | - if ($orig_status !== EEM_Event::sold_out && $event->status() !== $orig_status) { |
|
690 | - EE_Error::add_attention( |
|
691 | - sprintf( |
|
692 | - esc_html__( |
|
693 | - 'Please note that the Event Status has automatically been changed to %s because there are no more spaces available for this event. However, this change is not permanent until you update the event. You can change the status back to something else before updating if you wish.', |
|
694 | - 'event_espresso' |
|
695 | - ), |
|
696 | - EEH_Template::pretty_status(EEM_Event::sold_out, false, 'sentence') |
|
697 | - ) |
|
698 | - ); |
|
699 | - } |
|
700 | - return; |
|
701 | - } elseif ($orig_status === EEM_Event::sold_out) { |
|
702 | - EE_Error::add_attention( |
|
703 | - sprintf( |
|
704 | - esc_html__( |
|
705 | - 'Please note that the Event Status has automatically been changed to %s because more spaces have become available for this event, most likely due to abandoned transactions freeing up reserved tickets. However, this change is not permanent until you update the event. If you wish, you can change the status back to something else before updating.', |
|
706 | - 'event_espresso' |
|
707 | - ), |
|
708 | - EEH_Template::pretty_status($event->status(), false, 'sentence') |
|
709 | - ) |
|
710 | - ); |
|
711 | - } |
|
712 | - // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
|
713 | - if (! $event->tickets_on_sale()) { |
|
714 | - return; |
|
715 | - } |
|
716 | - // made it here so show warning |
|
717 | - $this->_edit_event_warning(); |
|
718 | - } |
|
719 | - |
|
720 | - |
|
721 | - /** |
|
722 | - * This is the text used for when an event is being edited that is public and has tickets for sale. |
|
723 | - * When needed, hook this into a EE_Error::add_error() notice. |
|
724 | - * |
|
725 | - * @access protected |
|
726 | - * @return void |
|
727 | - */ |
|
728 | - protected function _edit_event_warning() |
|
729 | - { |
|
730 | - // we don't want to add warnings during these requests |
|
731 | - if (isset($this->_req_data['action']) && $this->_req_data['action'] === 'editpost') { |
|
732 | - return; |
|
733 | - } |
|
734 | - EE_Error::add_attention( |
|
735 | - sprintf( |
|
736 | - esc_html__( |
|
737 | - 'Your event is open for registration. Making changes may disrupt any transactions in progress. %sLearn more%s', |
|
738 | - 'event_espresso' |
|
739 | - ), |
|
740 | - '<a class="espresso-help-tab-lnk">', |
|
741 | - '</a>' |
|
742 | - ) |
|
743 | - ); |
|
744 | - } |
|
745 | - |
|
746 | - |
|
747 | - /** |
|
748 | - * When a user is creating a new event, notify them if they haven't set their timezone. |
|
749 | - * Otherwise, do the normal logic |
|
750 | - * |
|
751 | - * @return string |
|
752 | - * @throws \EE_Error |
|
753 | - */ |
|
754 | - protected function _create_new_cpt_item() |
|
755 | - { |
|
756 | - $has_timezone_string = get_option('timezone_string'); |
|
757 | - // only nag them about setting their timezone if it's their first event, and they haven't already done it |
|
758 | - if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
759 | - EE_Error::add_attention( |
|
760 | - sprintf( |
|
761 | - __( |
|
762 | - 'Your website\'s timezone is currently set to a UTC offset. We recommend updating your timezone to a city or region near you before you create an event. Change your timezone now:%1$s%2$s%3$sChange Timezone%4$s', |
|
763 | - 'event_espresso' |
|
764 | - ), |
|
765 | - '<br>', |
|
766 | - '<select id="timezone_string" name="timezone_string" aria-describedby="timezone-description">' |
|
767 | - . EEH_DTT_Helper::wp_timezone_choice('', EEH_DTT_Helper::get_user_locale()) |
|
768 | - . '</select>', |
|
769 | - '<button class="button button-secondary timezone-submit">', |
|
770 | - '</button><span class="spinner"></span>' |
|
771 | - ), |
|
772 | - __FILE__, |
|
773 | - __FUNCTION__, |
|
774 | - __LINE__ |
|
775 | - ); |
|
776 | - } |
|
777 | - return parent::_create_new_cpt_item(); |
|
778 | - } |
|
779 | - |
|
780 | - |
|
781 | - /** |
|
782 | - * Sets the _views property for the default route in this admin page group. |
|
783 | - */ |
|
784 | - protected function _set_list_table_views_default() |
|
785 | - { |
|
786 | - $this->_views = array( |
|
787 | - 'all' => array( |
|
788 | - 'slug' => 'all', |
|
789 | - 'label' => esc_html__('View All Events', 'event_espresso'), |
|
790 | - 'count' => 0, |
|
791 | - 'bulk_action' => array( |
|
792 | - 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
793 | - ), |
|
794 | - ), |
|
795 | - 'draft' => array( |
|
796 | - 'slug' => 'draft', |
|
797 | - 'label' => esc_html__('Draft', 'event_espresso'), |
|
798 | - 'count' => 0, |
|
799 | - 'bulk_action' => array( |
|
800 | - 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
801 | - ), |
|
802 | - ), |
|
803 | - ); |
|
804 | - if (EE_Registry::instance()->CAP->current_user_can('ee_delete_events', 'espresso_events_trash_events')) { |
|
805 | - $this->_views['trash'] = array( |
|
806 | - 'slug' => 'trash', |
|
807 | - 'label' => esc_html__('Trash', 'event_espresso'), |
|
808 | - 'count' => 0, |
|
809 | - 'bulk_action' => array( |
|
810 | - 'restore_events' => esc_html__('Restore From Trash', 'event_espresso'), |
|
811 | - 'delete_events' => esc_html__('Delete Permanently', 'event_espresso'), |
|
812 | - ), |
|
813 | - ); |
|
814 | - } |
|
815 | - } |
|
816 | - |
|
817 | - |
|
818 | - /** |
|
819 | - * Provides the legend item array for the default list table view. |
|
820 | - * |
|
821 | - * @return array |
|
822 | - */ |
|
823 | - protected function _event_legend_items() |
|
824 | - { |
|
825 | - $items = array( |
|
826 | - 'view_details' => array( |
|
827 | - 'class' => 'dashicons dashicons-search', |
|
828 | - 'desc' => esc_html__('View Event', 'event_espresso'), |
|
829 | - ), |
|
830 | - 'edit_event' => array( |
|
831 | - 'class' => 'ee-icon ee-icon-calendar-edit', |
|
832 | - 'desc' => esc_html__('Edit Event Details', 'event_espresso'), |
|
833 | - ), |
|
834 | - 'view_attendees' => array( |
|
835 | - 'class' => 'dashicons dashicons-groups', |
|
836 | - 'desc' => esc_html__('View Registrations for Event', 'event_espresso'), |
|
837 | - ), |
|
838 | - ); |
|
839 | - $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
|
840 | - $statuses = array( |
|
841 | - 'sold_out_status' => array( |
|
842 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
843 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
|
844 | - ), |
|
845 | - 'active_status' => array( |
|
846 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
847 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
|
848 | - ), |
|
849 | - 'upcoming_status' => array( |
|
850 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
851 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
|
852 | - ), |
|
853 | - 'postponed_status' => array( |
|
854 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
855 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
|
856 | - ), |
|
857 | - 'cancelled_status' => array( |
|
858 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
859 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
|
860 | - ), |
|
861 | - 'expired_status' => array( |
|
862 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
863 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
|
864 | - ), |
|
865 | - 'inactive_status' => array( |
|
866 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
867 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
|
868 | - ), |
|
869 | - ); |
|
870 | - $statuses = apply_filters('FHEE__Events_Admin_Page__event_legend_items__statuses', $statuses); |
|
871 | - return array_merge($items, $statuses); |
|
872 | - } |
|
873 | - |
|
874 | - |
|
875 | - /** |
|
876 | - * @return EEM_Event |
|
877 | - */ |
|
878 | - private function _event_model() |
|
879 | - { |
|
880 | - if (! $this->_event_model instanceof EEM_Event) { |
|
881 | - $this->_event_model = EE_Registry::instance()->load_model('Event'); |
|
882 | - } |
|
883 | - return $this->_event_model; |
|
884 | - } |
|
885 | - |
|
886 | - |
|
887 | - /** |
|
888 | - * Adds extra buttons to the WP CPT permalink field row. |
|
889 | - * Method is called from parent and is hooked into the wp 'get_sample_permalink_html' filter. |
|
890 | - * |
|
891 | - * @param string $return the current html |
|
892 | - * @param int $id the post id for the page |
|
893 | - * @param string $new_title What the title is |
|
894 | - * @param string $new_slug what the slug is |
|
895 | - * @return string The new html string for the permalink area |
|
896 | - */ |
|
897 | - public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
|
898 | - { |
|
899 | - // make sure this is only when editing |
|
900 | - if (! empty($id)) { |
|
901 | - $post = get_post($id); |
|
902 | - $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
|
903 | - . esc_html__('Shortcode', 'event_espresso') |
|
904 | - . '</a> '; |
|
905 | - $return .= '<input id="shortcode" type="hidden" value="[ESPRESSO_TICKET_SELECTOR event_id=' |
|
906 | - . $post->ID |
|
907 | - . ']">'; |
|
908 | - } |
|
909 | - return $return; |
|
910 | - } |
|
911 | - |
|
912 | - |
|
913 | - /** |
|
914 | - * _events_overview_list_table |
|
915 | - * This contains the logic for showing the events_overview list |
|
916 | - * |
|
917 | - * @access protected |
|
918 | - * @return void |
|
919 | - * @throws \EE_Error |
|
920 | - */ |
|
921 | - protected function _events_overview_list_table() |
|
922 | - { |
|
923 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
924 | - $this->_template_args['after_list_table'] = ! empty($this->_template_args['after_list_table']) |
|
925 | - ? (array) $this->_template_args['after_list_table'] |
|
926 | - : array(); |
|
927 | - $this->_template_args['after_list_table']['view_event_list_button'] = EEH_HTML::br() |
|
928 | - . EEH_Template::get_button_or_link( |
|
929 | - get_post_type_archive_link('espresso_events'), |
|
930 | - esc_html__("View Event Archive Page", "event_espresso"), |
|
931 | - 'button' |
|
932 | - ); |
|
933 | - $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
|
934 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
935 | - 'create_new', |
|
936 | - 'add', |
|
937 | - array(), |
|
938 | - 'add-new-h2' |
|
939 | - ); |
|
940 | - $this->display_admin_list_table_page_with_no_sidebar(); |
|
941 | - } |
|
942 | - |
|
943 | - |
|
944 | - /** |
|
945 | - * this allows for extra misc actions in the default WP publish box |
|
946 | - * |
|
947 | - * @return void |
|
948 | - */ |
|
949 | - public function extra_misc_actions_publish_box() |
|
950 | - { |
|
951 | - $this->_generate_publish_box_extra_content(); |
|
952 | - } |
|
953 | - |
|
954 | - |
|
955 | - /** |
|
956 | - * This is hooked into the WordPress do_action('save_post') hook and runs after the custom post type has been |
|
957 | - * saved. |
|
958 | - * Typically you would use this to save any additional data. |
|
959 | - * Keep in mind also that "save_post" runs on EVERY post update to the database. |
|
960 | - * ALSO very important. When a post transitions from scheduled to published, |
|
961 | - * the save_post action is fired but you will NOT have any _POST data containing any extra info you may have from |
|
962 | - * other meta saves. So MAKE sure that you handle this accordingly. |
|
963 | - * |
|
964 | - * @access protected |
|
965 | - * @abstract |
|
966 | - * @param string $post_id The ID of the cpt that was saved (so you can link relationally) |
|
967 | - * @param object $post The post object of the cpt that was saved. |
|
968 | - * @return void |
|
969 | - * @throws \EE_Error |
|
970 | - */ |
|
971 | - protected function _insert_update_cpt_item($post_id, $post) |
|
972 | - { |
|
973 | - if ($post instanceof WP_Post && $post->post_type !== 'espresso_events') { |
|
974 | - // get out we're not processing an event save. |
|
975 | - return; |
|
976 | - } |
|
977 | - $event_values = array( |
|
978 | - 'EVT_display_desc' => ! empty($this->_req_data['display_desc']) ? 1 : 0, |
|
979 | - 'EVT_display_ticket_selector' => ! empty($this->_req_data['display_ticket_selector']) ? 1 : 0, |
|
980 | - 'EVT_additional_limit' => min( |
|
981 | - apply_filters('FHEE__EE_Events_Admin__insert_update_cpt_item__EVT_additional_limit_max', 255), |
|
982 | - ! empty($this->_req_data['additional_limit']) ? $this->_req_data['additional_limit'] : null |
|
983 | - ), |
|
984 | - 'EVT_default_registration_status' => ! empty($this->_req_data['EVT_default_registration_status']) |
|
985 | - ? $this->_req_data['EVT_default_registration_status'] |
|
986 | - : EE_Registry::instance()->CFG->registration->default_STS_ID, |
|
987 | - 'EVT_member_only' => ! empty($this->_req_data['member_only']) ? 1 : 0, |
|
988 | - 'EVT_allow_overflow' => ! empty($this->_req_data['EVT_allow_overflow']) ? 1 : 0, |
|
989 | - 'EVT_timezone_string' => ! empty($this->_req_data['timezone_string']) |
|
990 | - ? $this->_req_data['timezone_string'] : null, |
|
991 | - 'EVT_external_URL' => ! empty($this->_req_data['externalURL']) |
|
992 | - ? $this->_req_data['externalURL'] : null, |
|
993 | - 'EVT_phone' => ! empty($this->_req_data['event_phone']) |
|
994 | - ? $this->_req_data['event_phone'] : null, |
|
995 | - ); |
|
996 | - // update event |
|
997 | - $success = $this->_event_model()->update_by_ID($event_values, $post_id); |
|
998 | - // get event_object for other metaboxes... though it would seem to make sense to just use $this->_event_model()->get_one_by_ID( $post_id ).. i have to setup where conditions to override the filters in the model that filter out autodraft and inherit statuses so we GET the inherit id! |
|
999 | - $get_one_where = array( |
|
1000 | - $this->_event_model()->primary_key_name() => $post_id, |
|
1001 | - 'OR' => array( |
|
1002 | - 'status' => $post->post_status, |
|
1003 | - // if trying to "Publish" a sold out event, it's status will get switched back to "sold_out" in the db, |
|
1004 | - // but the returned object here has a status of "publish", so use the original post status as well |
|
1005 | - 'status*1' => $this->_req_data['original_post_status'], |
|
1006 | - ), |
|
1007 | - ); |
|
1008 | - $event = $this->_event_model()->get_one(array($get_one_where)); |
|
1009 | - // the following are default callbacks for event attachment updates that can be overridden by caffeinated functionality and/or addons. |
|
1010 | - $event_update_callbacks = apply_filters( |
|
1011 | - 'FHEE__Events_Admin_Page___insert_update_cpt_item__event_update_callbacks', |
|
1012 | - array( |
|
1013 | - array($this, '_default_venue_update'), |
|
1014 | - array($this, '_default_tickets_update'), |
|
1015 | - ) |
|
1016 | - ); |
|
1017 | - $att_success = true; |
|
1018 | - foreach ($event_update_callbacks as $e_callback) { |
|
1019 | - $_success = is_callable($e_callback) |
|
1020 | - ? call_user_func($e_callback, $event, $this->_req_data) |
|
1021 | - : false; |
|
1022 | - // if ANY of these updates fail then we want the appropriate global error message |
|
1023 | - $att_success = ! $att_success ? $att_success : $_success; |
|
1024 | - } |
|
1025 | - // any errors? |
|
1026 | - if ($success && false === $att_success) { |
|
1027 | - EE_Error::add_error( |
|
1028 | - esc_html__( |
|
1029 | - 'Event Details saved successfully but something went wrong with saving attachments.', |
|
1030 | - 'event_espresso' |
|
1031 | - ), |
|
1032 | - __FILE__, |
|
1033 | - __FUNCTION__, |
|
1034 | - __LINE__ |
|
1035 | - ); |
|
1036 | - } elseif ($success === false) { |
|
1037 | - EE_Error::add_error( |
|
1038 | - esc_html__('Event Details did not save successfully.', 'event_espresso'), |
|
1039 | - __FILE__, |
|
1040 | - __FUNCTION__, |
|
1041 | - __LINE__ |
|
1042 | - ); |
|
1043 | - } |
|
1044 | - } |
|
1045 | - |
|
1046 | - |
|
1047 | - /** |
|
1048 | - * @see parent::restore_item() |
|
1049 | - * @param int $post_id |
|
1050 | - * @param int $revision_id |
|
1051 | - */ |
|
1052 | - protected function _restore_cpt_item($post_id, $revision_id) |
|
1053 | - { |
|
1054 | - // copy existing event meta to new post |
|
1055 | - $post_evt = $this->_event_model()->get_one_by_ID($post_id); |
|
1056 | - if ($post_evt instanceof EE_Event) { |
|
1057 | - // meta revision restore |
|
1058 | - $post_evt->restore_revision($revision_id); |
|
1059 | - // related objs restore |
|
1060 | - $post_evt->restore_revision($revision_id, array('Venue', 'Datetime', 'Price')); |
|
1061 | - } |
|
1062 | - } |
|
1063 | - |
|
1064 | - |
|
1065 | - /** |
|
1066 | - * Attach the venue to the Event |
|
1067 | - * |
|
1068 | - * @param \EE_Event $evtobj Event Object to add the venue to |
|
1069 | - * @param array $data The request data from the form |
|
1070 | - * @return bool Success or fail. |
|
1071 | - */ |
|
1072 | - protected function _default_venue_update(\EE_Event $evtobj, $data) |
|
1073 | - { |
|
1074 | - require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
1075 | - $venue_model = EE_Registry::instance()->load_model('Venue'); |
|
1076 | - $rows_affected = null; |
|
1077 | - $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
|
1078 | - // very important. If we don't have a venue name... |
|
1079 | - // then we'll get out because not necessary to create empty venue |
|
1080 | - if (empty($data['venue_title'])) { |
|
1081 | - return false; |
|
1082 | - } |
|
1083 | - $venue_array = array( |
|
1084 | - 'VNU_wp_user' => $evtobj->get('EVT_wp_user'), |
|
1085 | - 'VNU_name' => ! empty($data['venue_title']) ? $data['venue_title'] : null, |
|
1086 | - 'VNU_desc' => ! empty($data['venue_description']) ? $data['venue_description'] : null, |
|
1087 | - 'VNU_identifier' => ! empty($data['venue_identifier']) ? $data['venue_identifier'] : null, |
|
1088 | - 'VNU_short_desc' => ! empty($data['venue_short_description']) ? $data['venue_short_description'] |
|
1089 | - : null, |
|
1090 | - 'VNU_address' => ! empty($data['address']) ? $data['address'] : null, |
|
1091 | - 'VNU_address2' => ! empty($data['address2']) ? $data['address2'] : null, |
|
1092 | - 'VNU_city' => ! empty($data['city']) ? $data['city'] : null, |
|
1093 | - 'STA_ID' => ! empty($data['state']) ? $data['state'] : null, |
|
1094 | - 'CNT_ISO' => ! empty($data['countries']) ? $data['countries'] : null, |
|
1095 | - 'VNU_zip' => ! empty($data['zip']) ? $data['zip'] : null, |
|
1096 | - 'VNU_phone' => ! empty($data['venue_phone']) ? $data['venue_phone'] : null, |
|
1097 | - 'VNU_capacity' => ! empty($data['venue_capacity']) ? $data['venue_capacity'] : null, |
|
1098 | - 'VNU_url' => ! empty($data['venue_url']) ? $data['venue_url'] : null, |
|
1099 | - 'VNU_virtual_phone' => ! empty($data['virtual_phone']) ? $data['virtual_phone'] : null, |
|
1100 | - 'VNU_virtual_url' => ! empty($data['virtual_url']) ? $data['virtual_url'] : null, |
|
1101 | - 'VNU_enable_for_gmap' => isset($data['enable_for_gmap']) ? 1 : 0, |
|
1102 | - 'status' => 'publish', |
|
1103 | - ); |
|
1104 | - // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
|
1105 | - if (! empty($venue_id)) { |
|
1106 | - $update_where = array($venue_model->primary_key_name() => $venue_id); |
|
1107 | - $rows_affected = $venue_model->update($venue_array, array($update_where)); |
|
1108 | - // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
|
1109 | - $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
1110 | - return $rows_affected > 0 ? true : false; |
|
1111 | - } else { |
|
1112 | - // we insert the venue |
|
1113 | - $venue_id = $venue_model->insert($venue_array); |
|
1114 | - $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
1115 | - return ! empty($venue_id) ? true : false; |
|
1116 | - } |
|
1117 | - // when we have the ancestor come in it's already been handled by the revision save. |
|
1118 | - } |
|
1119 | - |
|
1120 | - |
|
1121 | - /** |
|
1122 | - * Handles saving everything related to Tickets (datetimes, tickets, prices) |
|
1123 | - * |
|
1124 | - * @param EE_Event $evtobj The Event object we're attaching data to |
|
1125 | - * @param array $data The request data from the form |
|
1126 | - * @return array |
|
1127 | - */ |
|
1128 | - protected function _default_tickets_update(EE_Event $evtobj, $data) |
|
1129 | - { |
|
1130 | - $success = true; |
|
1131 | - $saved_dtt = null; |
|
1132 | - $saved_tickets = array(); |
|
1133 | - $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
1134 | - foreach ($data['edit_event_datetimes'] as $row => $dtt) { |
|
1135 | - // trim all values to ensure any excess whitespace is removed. |
|
1136 | - $dtt = array_map('trim', $dtt); |
|
1137 | - $dtt['DTT_EVT_end'] = isset($dtt['DTT_EVT_end']) && ! empty($dtt['DTT_EVT_end']) ? $dtt['DTT_EVT_end'] |
|
1138 | - : $dtt['DTT_EVT_start']; |
|
1139 | - $datetime_values = array( |
|
1140 | - 'DTT_ID' => ! empty($dtt['DTT_ID']) ? $dtt['DTT_ID'] : null, |
|
1141 | - 'DTT_EVT_start' => $dtt['DTT_EVT_start'], |
|
1142 | - 'DTT_EVT_end' => $dtt['DTT_EVT_end'], |
|
1143 | - 'DTT_reg_limit' => empty($dtt['DTT_reg_limit']) ? EE_INF : $dtt['DTT_reg_limit'], |
|
1144 | - 'DTT_order' => $row, |
|
1145 | - ); |
|
1146 | - // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
|
1147 | - if (! empty($dtt['DTT_ID'])) { |
|
1148 | - $DTM = EE_Registry::instance() |
|
1149 | - ->load_model('Datetime', array($evtobj->get_timezone())) |
|
1150 | - ->get_one_by_ID($dtt['DTT_ID']); |
|
1151 | - $DTM->set_date_format($incoming_date_formats[0]); |
|
1152 | - $DTM->set_time_format($incoming_date_formats[1]); |
|
1153 | - foreach ($datetime_values as $field => $value) { |
|
1154 | - $DTM->set($field, $value); |
|
1155 | - } |
|
1156 | - // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
|
1157 | - $saved_dtts[ $DTM->ID() ] = $DTM; |
|
1158 | - } else { |
|
1159 | - $DTM = EE_Registry::instance()->load_class( |
|
1160 | - 'Datetime', |
|
1161 | - array($datetime_values, $evtobj->get_timezone(), $incoming_date_formats), |
|
1162 | - false, |
|
1163 | - false |
|
1164 | - ); |
|
1165 | - foreach ($datetime_values as $field => $value) { |
|
1166 | - $DTM->set($field, $value); |
|
1167 | - } |
|
1168 | - } |
|
1169 | - $DTM->save(); |
|
1170 | - $DTT = $evtobj->_add_relation_to($DTM, 'Datetime'); |
|
1171 | - // load DTT helper |
|
1172 | - // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
1173 | - if ($DTT->get_raw('DTT_EVT_start') > $DTT->get_raw('DTT_EVT_end')) { |
|
1174 | - $DTT->set('DTT_EVT_end', $DTT->get('DTT_EVT_start')); |
|
1175 | - $DTT = EEH_DTT_Helper::date_time_add($DTT, 'DTT_EVT_end', 'days'); |
|
1176 | - $DTT->save(); |
|
1177 | - } |
|
1178 | - // now we got to make sure we add the new DTT_ID to the $saved_dtts array because it is possible there was a new one created for the autosave. |
|
1179 | - $saved_dtt = $DTT; |
|
1180 | - $success = ! $success ? $success : $DTT; |
|
1181 | - // if ANY of these updates fail then we want the appropriate global error message. |
|
1182 | - // //todo this is actually sucky we need a better error message but this is what it is for now. |
|
1183 | - } |
|
1184 | - // no dtts get deleted so we don't do any of that logic here. |
|
1185 | - // update tickets next |
|
1186 | - $old_tickets = isset($data['ticket_IDs']) ? explode(',', $data['ticket_IDs']) : array(); |
|
1187 | - foreach ($data['edit_tickets'] as $row => $tkt) { |
|
1188 | - $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
1189 | - $update_prices = false; |
|
1190 | - $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
1191 | - ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
1192 | - // trim inputs to ensure any excess whitespace is removed. |
|
1193 | - $tkt = array_map('trim', $tkt); |
|
1194 | - if (empty($tkt['TKT_start_date'])) { |
|
1195 | - // let's use now in the set timezone. |
|
1196 | - $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
|
1197 | - $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
1198 | - } |
|
1199 | - if (empty($tkt['TKT_end_date'])) { |
|
1200 | - // use the start date of the first datetime |
|
1201 | - $dtt = $evtobj->first_datetime(); |
|
1202 | - $tkt['TKT_end_date'] = $dtt->start_date_and_time( |
|
1203 | - $incoming_date_formats[0], |
|
1204 | - $incoming_date_formats[1] |
|
1205 | - ); |
|
1206 | - } |
|
1207 | - $TKT_values = array( |
|
1208 | - 'TKT_ID' => ! empty($tkt['TKT_ID']) ? $tkt['TKT_ID'] : null, |
|
1209 | - 'TTM_ID' => ! empty($tkt['TTM_ID']) ? $tkt['TTM_ID'] : 0, |
|
1210 | - 'TKT_name' => ! empty($tkt['TKT_name']) ? $tkt['TKT_name'] : '', |
|
1211 | - 'TKT_description' => ! empty($tkt['TKT_description']) ? $tkt['TKT_description'] : '', |
|
1212 | - 'TKT_start_date' => $tkt['TKT_start_date'], |
|
1213 | - 'TKT_end_date' => $tkt['TKT_end_date'], |
|
1214 | - 'TKT_qty' => ! isset($tkt['TKT_qty']) || $tkt['TKT_qty'] === '' ? EE_INF : $tkt['TKT_qty'], |
|
1215 | - 'TKT_uses' => ! isset($tkt['TKT_uses']) || $tkt['TKT_uses'] === '' ? EE_INF : $tkt['TKT_uses'], |
|
1216 | - 'TKT_min' => empty($tkt['TKT_min']) ? 0 : $tkt['TKT_min'], |
|
1217 | - 'TKT_max' => empty($tkt['TKT_max']) ? EE_INF : $tkt['TKT_max'], |
|
1218 | - 'TKT_row' => $row, |
|
1219 | - 'TKT_order' => isset($tkt['TKT_order']) ? $tkt['TKT_order'] : $row, |
|
1220 | - 'TKT_price' => $ticket_price, |
|
1221 | - ); |
|
1222 | - // if this is a default TKT, then we need to set the TKT_ID to 0 and update accordingly, which means in turn that the prices will become new prices as well. |
|
1223 | - if (isset($tkt['TKT_is_default']) && $tkt['TKT_is_default']) { |
|
1224 | - $TKT_values['TKT_ID'] = 0; |
|
1225 | - $TKT_values['TKT_is_default'] = 0; |
|
1226 | - $TKT_values['TKT_price'] = $ticket_price; |
|
1227 | - $update_prices = true; |
|
1228 | - } |
|
1229 | - // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
|
1230 | - // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
|
1231 | - // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
|
1232 | - if (! empty($tkt['TKT_ID'])) { |
|
1233 | - $TKT = EE_Registry::instance() |
|
1234 | - ->load_model('Ticket', array($evtobj->get_timezone())) |
|
1235 | - ->get_one_by_ID($tkt['TKT_ID']); |
|
1236 | - if ($TKT instanceof EE_Ticket) { |
|
1237 | - $ticket_sold = $TKT->count_related( |
|
1238 | - 'Registration', |
|
1239 | - array( |
|
1240 | - array( |
|
1241 | - 'STS_ID' => array( |
|
1242 | - 'NOT IN', |
|
1243 | - array(EEM_Registration::status_id_incomplete), |
|
1244 | - ), |
|
1245 | - ), |
|
1246 | - ) |
|
1247 | - ) > 0 ? true : false; |
|
1248 | - // let's just check the total price for the existing ticket and determine if it matches the new total price. if they are different then we create a new ticket (if tkts sold) if they aren't different then we go ahead and modify existing ticket. |
|
1249 | - $create_new_TKT = $ticket_sold && $ticket_price != $TKT->get('TKT_price') |
|
1250 | - && ! $TKT->get('TKT_deleted'); |
|
1251 | - $TKT->set_date_format($incoming_date_formats[0]); |
|
1252 | - $TKT->set_time_format($incoming_date_formats[1]); |
|
1253 | - // set new values |
|
1254 | - foreach ($TKT_values as $field => $value) { |
|
1255 | - if ($field == 'TKT_qty') { |
|
1256 | - $TKT->set_qty($value); |
|
1257 | - } else { |
|
1258 | - $TKT->set($field, $value); |
|
1259 | - } |
|
1260 | - } |
|
1261 | - // if $create_new_TKT is false then we can safely update the existing ticket. Otherwise we have to create a new ticket. |
|
1262 | - if ($create_new_TKT) { |
|
1263 | - // archive the old ticket first |
|
1264 | - $TKT->set('TKT_deleted', 1); |
|
1265 | - $TKT->save(); |
|
1266 | - // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
|
1267 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1268 | - // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
|
1269 | - $TKT = clone $TKT; |
|
1270 | - $TKT->set('TKT_ID', 0); |
|
1271 | - $TKT->set('TKT_deleted', 0); |
|
1272 | - $TKT->set('TKT_price', $ticket_price); |
|
1273 | - $TKT->set('TKT_sold', 0); |
|
1274 | - // now we need to make sure that $new prices are created as well and attached to new ticket. |
|
1275 | - $update_prices = true; |
|
1276 | - } |
|
1277 | - // make sure price is set if it hasn't been already |
|
1278 | - $TKT->set('TKT_price', $ticket_price); |
|
1279 | - } |
|
1280 | - } else { |
|
1281 | - // no TKT_id so a new TKT |
|
1282 | - $TKT_values['TKT_price'] = $ticket_price; |
|
1283 | - $TKT = EE_Registry::instance()->load_class('Ticket', array($TKT_values), false, false); |
|
1284 | - if ($TKT instanceof EE_Ticket) { |
|
1285 | - // need to reset values to properly account for the date formats |
|
1286 | - $TKT->set_date_format($incoming_date_formats[0]); |
|
1287 | - $TKT->set_time_format($incoming_date_formats[1]); |
|
1288 | - $TKT->set_timezone($evtobj->get_timezone()); |
|
1289 | - // set new values |
|
1290 | - foreach ($TKT_values as $field => $value) { |
|
1291 | - if ($field == 'TKT_qty') { |
|
1292 | - $TKT->set_qty($value); |
|
1293 | - } else { |
|
1294 | - $TKT->set($field, $value); |
|
1295 | - } |
|
1296 | - } |
|
1297 | - $update_prices = true; |
|
1298 | - } |
|
1299 | - } |
|
1300 | - // cap ticket qty by datetime reg limits |
|
1301 | - $TKT->set_qty(min($TKT->qty(), $TKT->qty('reg_limit'))); |
|
1302 | - // update ticket. |
|
1303 | - $TKT->save(); |
|
1304 | - // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
1305 | - if ($TKT->get_raw('TKT_start_date') > $TKT->get_raw('TKT_end_date')) { |
|
1306 | - $TKT->set('TKT_end_date', $TKT->get('TKT_start_date')); |
|
1307 | - $TKT = EEH_DTT_Helper::date_time_add($TKT, 'TKT_end_date', 'days'); |
|
1308 | - $TKT->save(); |
|
1309 | - } |
|
1310 | - // initially let's add the ticket to the dtt |
|
1311 | - $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
|
1312 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1313 | - // add prices to ticket |
|
1314 | - $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
1315 | - } |
|
1316 | - // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
|
1317 | - $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
|
1318 | - $tickets_removed = array_diff($old_tickets, array_keys($saved_tickets)); |
|
1319 | - foreach ($tickets_removed as $id) { |
|
1320 | - $id = absint($id); |
|
1321 | - // get the ticket for this id |
|
1322 | - $tkt_to_remove = EE_Registry::instance()->load_model('Ticket')->get_one_by_ID($id); |
|
1323 | - // need to get all the related datetimes on this ticket and remove from every single one of them (remember this process can ONLY kick off if there are NO tkts_sold) |
|
1324 | - $dtts = $tkt_to_remove->get_many_related('Datetime'); |
|
1325 | - foreach ($dtts as $dtt) { |
|
1326 | - $tkt_to_remove->_remove_relation_to($dtt, 'Datetime'); |
|
1327 | - } |
|
1328 | - // need to do the same for prices (except these prices can also be deleted because again, tickets can only be trashed if they don't have any TKTs sold (otherwise they are just archived)) |
|
1329 | - $tkt_to_remove->delete_related_permanently('Price'); |
|
1330 | - // finally let's delete this ticket (which should not be blocked at this point b/c we've removed all our relationships) |
|
1331 | - $tkt_to_remove->delete_permanently(); |
|
1332 | - } |
|
1333 | - return array($saved_dtt, $saved_tickets); |
|
1334 | - } |
|
1335 | - |
|
1336 | - |
|
1337 | - /** |
|
1338 | - * This attaches a list of given prices to a ticket. |
|
1339 | - * Note we dont' have to worry about ever removing relationships (or archiving prices) because if there is a change |
|
1340 | - * in price information on a ticket, a new ticket is created anyways so the archived ticket will retain the old |
|
1341 | - * price info and prices are automatically "archived" via the ticket. |
|
1342 | - * |
|
1343 | - * @access private |
|
1344 | - * @param array $prices Array of prices from the form. |
|
1345 | - * @param EE_Ticket $ticket EE_Ticket object that prices are being attached to. |
|
1346 | - * @param bool $new_prices Whether attach existing incoming prices or create new ones. |
|
1347 | - * @return void |
|
1348 | - */ |
|
1349 | - private function _add_prices_to_ticket($prices, EE_Ticket $ticket, $new_prices = false) |
|
1350 | - { |
|
1351 | - foreach ($prices as $row => $prc) { |
|
1352 | - $PRC_values = array( |
|
1353 | - 'PRC_ID' => ! empty($prc['PRC_ID']) ? $prc['PRC_ID'] : null, |
|
1354 | - 'PRT_ID' => ! empty($prc['PRT_ID']) ? $prc['PRT_ID'] : null, |
|
1355 | - 'PRC_amount' => ! empty($prc['PRC_amount']) ? $prc['PRC_amount'] : 0, |
|
1356 | - 'PRC_name' => ! empty($prc['PRC_name']) ? $prc['PRC_name'] : '', |
|
1357 | - 'PRC_desc' => ! empty($prc['PRC_desc']) ? $prc['PRC_desc'] : '', |
|
1358 | - 'PRC_is_default' => 0, // make sure prices are NOT set as default from this context |
|
1359 | - 'PRC_order' => $row, |
|
1360 | - ); |
|
1361 | - if ($new_prices || empty($PRC_values['PRC_ID'])) { |
|
1362 | - $PRC_values['PRC_ID'] = 0; |
|
1363 | - $PRC = EE_Registry::instance()->load_class('Price', array($PRC_values), false, false); |
|
1364 | - } else { |
|
1365 | - $PRC = EE_Registry::instance()->load_model('Price')->get_one_by_ID($prc['PRC_ID']); |
|
1366 | - // update this price with new values |
|
1367 | - foreach ($PRC_values as $field => $newprc) { |
|
1368 | - $PRC->set($field, $newprc); |
|
1369 | - } |
|
1370 | - $PRC->save(); |
|
1371 | - } |
|
1372 | - $ticket->_add_relation_to($PRC, 'Price'); |
|
1373 | - } |
|
1374 | - } |
|
1375 | - |
|
1376 | - |
|
1377 | - /** |
|
1378 | - * Add in our autosave ajax handlers |
|
1379 | - * |
|
1380 | - */ |
|
1381 | - protected function _ee_autosave_create_new() |
|
1382 | - { |
|
1383 | - } |
|
1384 | - |
|
1385 | - |
|
1386 | - /** |
|
1387 | - * More autosave handlers. |
|
1388 | - */ |
|
1389 | - protected function _ee_autosave_edit() |
|
1390 | - { |
|
1391 | - return; // TEMPORARILY EXITING CAUSE THIS IS A TODO |
|
1392 | - } |
|
1393 | - |
|
1394 | - |
|
1395 | - /** |
|
1396 | - * _generate_publish_box_extra_content |
|
1397 | - */ |
|
1398 | - private function _generate_publish_box_extra_content() |
|
1399 | - { |
|
1400 | - // load formatter helper |
|
1401 | - // args for getting related registrations |
|
1402 | - $approved_query_args = array( |
|
1403 | - array( |
|
1404 | - 'REG_deleted' => 0, |
|
1405 | - 'STS_ID' => EEM_Registration::status_id_approved, |
|
1406 | - ), |
|
1407 | - ); |
|
1408 | - $not_approved_query_args = array( |
|
1409 | - array( |
|
1410 | - 'REG_deleted' => 0, |
|
1411 | - 'STS_ID' => EEM_Registration::status_id_not_approved, |
|
1412 | - ), |
|
1413 | - ); |
|
1414 | - $pending_payment_query_args = array( |
|
1415 | - array( |
|
1416 | - 'REG_deleted' => 0, |
|
1417 | - 'STS_ID' => EEM_Registration::status_id_pending_payment, |
|
1418 | - ), |
|
1419 | - ); |
|
1420 | - // publish box |
|
1421 | - $publish_box_extra_args = array( |
|
1422 | - 'view_approved_reg_url' => add_query_arg( |
|
1423 | - array( |
|
1424 | - 'action' => 'default', |
|
1425 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
1426 | - '_reg_status' => EEM_Registration::status_id_approved, |
|
1427 | - ), |
|
1428 | - REG_ADMIN_URL |
|
1429 | - ), |
|
1430 | - 'view_not_approved_reg_url' => add_query_arg( |
|
1431 | - array( |
|
1432 | - 'action' => 'default', |
|
1433 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
1434 | - '_reg_status' => EEM_Registration::status_id_not_approved, |
|
1435 | - ), |
|
1436 | - REG_ADMIN_URL |
|
1437 | - ), |
|
1438 | - 'view_pending_payment_reg_url' => add_query_arg( |
|
1439 | - array( |
|
1440 | - 'action' => 'default', |
|
1441 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
1442 | - '_reg_status' => EEM_Registration::status_id_pending_payment, |
|
1443 | - ), |
|
1444 | - REG_ADMIN_URL |
|
1445 | - ), |
|
1446 | - 'approved_regs' => $this->_cpt_model_obj->count_related( |
|
1447 | - 'Registration', |
|
1448 | - $approved_query_args |
|
1449 | - ), |
|
1450 | - 'not_approved_regs' => $this->_cpt_model_obj->count_related( |
|
1451 | - 'Registration', |
|
1452 | - $not_approved_query_args |
|
1453 | - ), |
|
1454 | - 'pending_payment_regs' => $this->_cpt_model_obj->count_related( |
|
1455 | - 'Registration', |
|
1456 | - $pending_payment_query_args |
|
1457 | - ), |
|
1458 | - 'misc_pub_section_class' => apply_filters( |
|
1459 | - 'FHEE_Events_Admin_Page___generate_publish_box_extra_content__misc_pub_section_class', |
|
1460 | - 'misc-pub-section' |
|
1461 | - ), |
|
1462 | - ); |
|
1463 | - ob_start(); |
|
1464 | - do_action( |
|
1465 | - 'AHEE__Events_Admin_Page___generate_publish_box_extra_content__event_editor_overview_add', |
|
1466 | - $this->_cpt_model_obj |
|
1467 | - ); |
|
1468 | - $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
|
1469 | - // load template |
|
1470 | - EEH_Template::display_template( |
|
1471 | - EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
1472 | - $publish_box_extra_args |
|
1473 | - ); |
|
1474 | - } |
|
1475 | - |
|
1476 | - |
|
1477 | - /** |
|
1478 | - * @return EE_Event |
|
1479 | - */ |
|
1480 | - public function get_event_object() |
|
1481 | - { |
|
1482 | - return $this->_cpt_model_obj; |
|
1483 | - } |
|
1484 | - |
|
1485 | - |
|
1486 | - |
|
1487 | - |
|
1488 | - /** METABOXES * */ |
|
1489 | - /** |
|
1490 | - * _register_event_editor_meta_boxes |
|
1491 | - * add all metaboxes related to the event_editor |
|
1492 | - * |
|
1493 | - * @return void |
|
1494 | - */ |
|
1495 | - protected function _register_event_editor_meta_boxes() |
|
1496 | - { |
|
1497 | - $this->verify_cpt_object(); |
|
1498 | - add_meta_box( |
|
1499 | - 'espresso_event_editor_tickets', |
|
1500 | - esc_html__('Event Datetime & Ticket', 'event_espresso'), |
|
1501 | - array($this, 'ticket_metabox'), |
|
1502 | - $this->page_slug, |
|
1503 | - 'normal', |
|
1504 | - 'high' |
|
1505 | - ); |
|
1506 | - add_meta_box( |
|
1507 | - 'espresso_event_editor_event_options', |
|
1508 | - esc_html__('Event Registration Options', 'event_espresso'), |
|
1509 | - array($this, 'registration_options_meta_box'), |
|
1510 | - $this->page_slug, |
|
1511 | - 'side', |
|
1512 | - 'default' |
|
1513 | - ); |
|
1514 | - // NOTE: if you're looking for other metaboxes in here, |
|
1515 | - // where a metabox has a related management page in the admin |
|
1516 | - // you will find it setup in the related management page's "_Hooks" file. |
|
1517 | - // i.e. messages metabox is found in "espresso_events_Messages_Hooks.class.php". |
|
1518 | - } |
|
1519 | - |
|
1520 | - |
|
1521 | - /** |
|
1522 | - * @throws DomainException |
|
1523 | - * @throws EE_Error |
|
1524 | - */ |
|
1525 | - public function ticket_metabox() |
|
1526 | - { |
|
1527 | - $existing_datetime_ids = $existing_ticket_ids = array(); |
|
1528 | - // defaults for template args |
|
1529 | - $template_args = array( |
|
1530 | - 'existing_datetime_ids' => '', |
|
1531 | - 'event_datetime_help_link' => '', |
|
1532 | - 'ticket_options_help_link' => '', |
|
1533 | - 'time' => null, |
|
1534 | - 'ticket_rows' => '', |
|
1535 | - 'existing_ticket_ids' => '', |
|
1536 | - 'total_ticket_rows' => 1, |
|
1537 | - 'ticket_js_structure' => '', |
|
1538 | - 'trash_icon' => 'ee-lock-icon', |
|
1539 | - 'disabled' => '', |
|
1540 | - ); |
|
1541 | - $event_id = is_object($this->_cpt_model_obj) ? $this->_cpt_model_obj->ID() : null; |
|
1542 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
1543 | - /** |
|
1544 | - * 1. Start with retrieving Datetimes |
|
1545 | - * 2. Fore each datetime get related tickets |
|
1546 | - * 3. For each ticket get related prices |
|
1547 | - */ |
|
1548 | - $times = EE_Registry::instance()->load_model('Datetime')->get_all_event_dates($event_id); |
|
1549 | - /** @type EE_Datetime $first_datetime */ |
|
1550 | - $first_datetime = reset($times); |
|
1551 | - // do we get related tickets? |
|
1552 | - if ($first_datetime instanceof EE_Datetime |
|
1553 | - && $first_datetime->ID() !== 0 |
|
1554 | - ) { |
|
1555 | - $existing_datetime_ids[] = $first_datetime->get('DTT_ID'); |
|
1556 | - $template_args['time'] = $first_datetime; |
|
1557 | - $related_tickets = $first_datetime->tickets( |
|
1558 | - array( |
|
1559 | - array('OR' => array('TKT_deleted' => 1, 'TKT_deleted*' => 0)), |
|
1560 | - 'default_where_conditions' => 'none', |
|
1561 | - ) |
|
1562 | - ); |
|
1563 | - if (! empty($related_tickets)) { |
|
1564 | - $template_args['total_ticket_rows'] = count($related_tickets); |
|
1565 | - $row = 0; |
|
1566 | - foreach ($related_tickets as $ticket) { |
|
1567 | - $existing_ticket_ids[] = $ticket->get('TKT_ID'); |
|
1568 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket, false, $row); |
|
1569 | - $row++; |
|
1570 | - } |
|
1571 | - } else { |
|
1572 | - $template_args['total_ticket_rows'] = 1; |
|
1573 | - /** @type EE_Ticket $ticket */ |
|
1574 | - $ticket = EE_Registry::instance()->load_model('Ticket')->create_default_object(); |
|
1575 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket); |
|
1576 | - } |
|
1577 | - } else { |
|
1578 | - $template_args['time'] = $times[0]; |
|
1579 | - /** @type EE_Ticket $ticket */ |
|
1580 | - $ticket = EE_Registry::instance()->load_model('Ticket')->get_all_default_tickets(); |
|
1581 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket[1]); |
|
1582 | - // NOTE: we're just sending the first default row |
|
1583 | - // (decaf can't manage default tickets so this should be sufficient); |
|
1584 | - } |
|
1585 | - $template_args['event_datetime_help_link'] = $this->_get_help_tab_link( |
|
1586 | - 'event_editor_event_datetimes_help_tab' |
|
1587 | - ); |
|
1588 | - $template_args['ticket_options_help_link'] = $this->_get_help_tab_link('ticket_options_info'); |
|
1589 | - $template_args['existing_datetime_ids'] = implode(',', $existing_datetime_ids); |
|
1590 | - $template_args['existing_ticket_ids'] = implode(',', $existing_ticket_ids); |
|
1591 | - $template_args['ticket_js_structure'] = $this->_get_ticket_row( |
|
1592 | - EE_Registry::instance()->load_model('Ticket')->create_default_object(), |
|
1593 | - true |
|
1594 | - ); |
|
1595 | - $template = apply_filters( |
|
1596 | - 'FHEE__Events_Admin_Page__ticket_metabox__template', |
|
1597 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
1598 | - ); |
|
1599 | - EEH_Template::display_template($template, $template_args); |
|
1600 | - } |
|
1601 | - |
|
1602 | - |
|
1603 | - /** |
|
1604 | - * Setup an individual ticket form for the decaf event editor page |
|
1605 | - * |
|
1606 | - * @access private |
|
1607 | - * @param EE_Ticket $ticket the ticket object |
|
1608 | - * @param boolean $skeleton whether we're generating a skeleton for js manipulation |
|
1609 | - * @param int $row |
|
1610 | - * @return string generated html for the ticket row. |
|
1611 | - */ |
|
1612 | - private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
|
1613 | - { |
|
1614 | - $template_args = array( |
|
1615 | - 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
1616 | - 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
|
1617 | - : '', |
|
1618 | - 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
|
1619 | - 'TKT_ID' => $ticket->get('TKT_ID'), |
|
1620 | - 'TKT_name' => $ticket->get('TKT_name'), |
|
1621 | - 'TKT_start_date' => $skeleton ? '' : $ticket->get_date('TKT_start_date', 'Y-m-d h:i a'), |
|
1622 | - 'TKT_end_date' => $skeleton ? '' : $ticket->get_date('TKT_end_date', 'Y-m-d h:i a'), |
|
1623 | - 'TKT_is_default' => $ticket->get('TKT_is_default'), |
|
1624 | - 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
|
1625 | - 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
|
1626 | - 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
|
1627 | - 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
1628 | - && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
1629 | - ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
|
1630 | - 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
1631 | - : ' disabled=disabled', |
|
1632 | - ); |
|
1633 | - $price = $ticket->ID() !== 0 |
|
1634 | - ? $ticket->get_first_related('Price', array('default_where_conditions' => 'none')) |
|
1635 | - : EE_Registry::instance()->load_model('Price')->create_default_object(); |
|
1636 | - $price_args = array( |
|
1637 | - 'price_currency_symbol' => EE_Registry::instance()->CFG->currency->sign, |
|
1638 | - 'PRC_amount' => $price->get('PRC_amount'), |
|
1639 | - 'PRT_ID' => $price->get('PRT_ID'), |
|
1640 | - 'PRC_ID' => $price->get('PRC_ID'), |
|
1641 | - 'PRC_is_default' => $price->get('PRC_is_default'), |
|
1642 | - ); |
|
1643 | - // make sure we have default start and end dates if skeleton |
|
1644 | - // handle rows that should NOT be empty |
|
1645 | - if (empty($template_args['TKT_start_date'])) { |
|
1646 | - // if empty then the start date will be now. |
|
1647 | - $template_args['TKT_start_date'] = date('Y-m-d h:i a', current_time('timestamp')); |
|
1648 | - } |
|
1649 | - if (empty($template_args['TKT_end_date'])) { |
|
1650 | - // get the earliest datetime (if present); |
|
1651 | - $earliest_dtt = $this->_cpt_model_obj->ID() > 0 |
|
1652 | - ? $this->_cpt_model_obj->get_first_related( |
|
1653 | - 'Datetime', |
|
1654 | - array('order_by' => array('DTT_EVT_start' => 'ASC')) |
|
1655 | - ) |
|
1656 | - : null; |
|
1657 | - if (! empty($earliest_dtt)) { |
|
1658 | - $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
|
1659 | - } else { |
|
1660 | - $template_args['TKT_end_date'] = date( |
|
1661 | - 'Y-m-d h:i a', |
|
1662 | - mktime(0, 0, 0, date("m"), date("d") + 7, date("Y")) |
|
1663 | - ); |
|
1664 | - } |
|
1665 | - } |
|
1666 | - $template_args = array_merge($template_args, $price_args); |
|
1667 | - $template = apply_filters( |
|
1668 | - 'FHEE__Events_Admin_Page__get_ticket_row__template', |
|
1669 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
1670 | - $ticket |
|
1671 | - ); |
|
1672 | - return EEH_Template::display_template($template, $template_args, true); |
|
1673 | - } |
|
1674 | - |
|
1675 | - |
|
1676 | - /** |
|
1677 | - * @throws DomainException |
|
1678 | - */ |
|
1679 | - public function registration_options_meta_box() |
|
1680 | - { |
|
1681 | - $yes_no_values = array( |
|
1682 | - array('id' => true, 'text' => esc_html__('Yes', 'event_espresso')), |
|
1683 | - array('id' => false, 'text' => esc_html__('No', 'event_espresso')), |
|
1684 | - ); |
|
1685 | - $default_reg_status_values = EEM_Registration::reg_status_array( |
|
1686 | - array( |
|
1687 | - EEM_Registration::status_id_cancelled, |
|
1688 | - EEM_Registration::status_id_declined, |
|
1689 | - EEM_Registration::status_id_incomplete, |
|
1690 | - ), |
|
1691 | - true |
|
1692 | - ); |
|
1693 | - // $template_args['is_active_select'] = EEH_Form_Fields::select_input('is_active', $yes_no_values, $this->_cpt_model_obj->is_active()); |
|
1694 | - $template_args['_event'] = $this->_cpt_model_obj; |
|
1695 | - $template_args['active_status'] = $this->_cpt_model_obj->pretty_active_status(false); |
|
1696 | - $template_args['additional_limit'] = $this->_cpt_model_obj->additional_limit(); |
|
1697 | - $template_args['default_registration_status'] = EEH_Form_Fields::select_input( |
|
1698 | - 'default_reg_status', |
|
1699 | - $default_reg_status_values, |
|
1700 | - $this->_cpt_model_obj->default_registration_status() |
|
1701 | - ); |
|
1702 | - $template_args['display_description'] = EEH_Form_Fields::select_input( |
|
1703 | - 'display_desc', |
|
1704 | - $yes_no_values, |
|
1705 | - $this->_cpt_model_obj->display_description() |
|
1706 | - ); |
|
1707 | - $template_args['display_ticket_selector'] = EEH_Form_Fields::select_input( |
|
1708 | - 'display_ticket_selector', |
|
1709 | - $yes_no_values, |
|
1710 | - $this->_cpt_model_obj->display_ticket_selector(), |
|
1711 | - '', |
|
1712 | - '', |
|
1713 | - false |
|
1714 | - ); |
|
1715 | - $template_args['additional_registration_options'] = apply_filters( |
|
1716 | - 'FHEE__Events_Admin_Page__registration_options_meta_box__additional_registration_options', |
|
1717 | - '', |
|
1718 | - $template_args, |
|
1719 | - $yes_no_values, |
|
1720 | - $default_reg_status_values |
|
1721 | - ); |
|
1722 | - EEH_Template::display_template( |
|
1723 | - EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
1724 | - $template_args |
|
1725 | - ); |
|
1726 | - } |
|
1727 | - |
|
1728 | - |
|
1729 | - /** |
|
1730 | - * _get_events() |
|
1731 | - * This method simply returns all the events (for the given _view and paging) |
|
1732 | - * |
|
1733 | - * @access public |
|
1734 | - * @param int $per_page count of items per page (20 default); |
|
1735 | - * @param int $current_page what is the current page being viewed. |
|
1736 | - * @param bool $count if TRUE then we just return a count of ALL events matching the given _view. |
|
1737 | - * If FALSE then we return an array of event objects |
|
1738 | - * that match the given _view and paging parameters. |
|
1739 | - * @return array an array of event objects. |
|
1740 | - */ |
|
1741 | - public function get_events($per_page = 10, $current_page = 1, $count = false) |
|
1742 | - { |
|
1743 | - $EEME = $this->_event_model(); |
|
1744 | - $offset = ($current_page - 1) * $per_page; |
|
1745 | - $limit = $count ? null : $offset . ',' . $per_page; |
|
1746 | - $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
|
1747 | - $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
|
1748 | - if (isset($this->_req_data['month_range'])) { |
|
1749 | - $pieces = explode(' ', $this->_req_data['month_range'], 3); |
|
1750 | - // simulate the FIRST day of the month, that fixes issues for months like February |
|
1751 | - // where PHP doesn't know what to assume for date. |
|
1752 | - // @see https://events.codebasehq.com/projects/event-espresso/tickets/10437 |
|
1753 | - $month_r = ! empty($pieces[0]) ? date('m', \EEH_DTT_Helper::first_of_month_timestamp($pieces[0])) : ''; |
|
1754 | - $year_r = ! empty($pieces[1]) ? $pieces[1] : ''; |
|
1755 | - } |
|
1756 | - $where = array(); |
|
1757 | - $status = isset($this->_req_data['status']) ? $this->_req_data['status'] : null; |
|
1758 | - // determine what post_status our condition will have for the query. |
|
1759 | - switch ($status) { |
|
1760 | - case 'month': |
|
1761 | - case 'today': |
|
1762 | - case null: |
|
1763 | - case 'all': |
|
1764 | - break; |
|
1765 | - case 'draft': |
|
1766 | - $where['status'] = array('IN', array('draft', 'auto-draft')); |
|
1767 | - break; |
|
1768 | - default: |
|
1769 | - $where['status'] = $status; |
|
1770 | - } |
|
1771 | - // categories? |
|
1772 | - $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
|
1773 | - ? $this->_req_data['EVT_CAT'] : null; |
|
1774 | - if (! empty($category)) { |
|
1775 | - $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
|
1776 | - $where['Term_Taxonomy.term_id'] = $category; |
|
1777 | - } |
|
1778 | - // date where conditions |
|
1779 | - $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
|
1780 | - if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
|
1781 | - $DateTime = new DateTime( |
|
1782 | - $year_r . '-' . $month_r . '-01 00:00:00', |
|
1783 | - new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
|
1784 | - ); |
|
1785 | - $start = $DateTime->format(implode(' ', $start_formats)); |
|
1786 | - $end = $DateTime->setDate( |
|
1787 | - $year_r, |
|
1788 | - $month_r, |
|
1789 | - $DateTime |
|
1790 | - ->format('t') |
|
1791 | - )->setTime(23, 59, 59) |
|
1792 | - ->format(implode(' ', $start_formats)); |
|
1793 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1794 | - } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'today') { |
|
1795 | - $DateTime = new DateTime('now', new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
1796 | - $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
1797 | - $end = $DateTime->setTime(23, 59, 59)->format(implode(' ', $start_formats)); |
|
1798 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1799 | - } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'month') { |
|
1800 | - $now = date('Y-m-01'); |
|
1801 | - $DateTime = new DateTime($now, new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
1802 | - $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
1803 | - $end = $DateTime->setDate(date('Y'), date('m'), $DateTime->format('t')) |
|
1804 | - ->setTime(23, 59, 59) |
|
1805 | - ->format(implode(' ', $start_formats)); |
|
1806 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1807 | - } |
|
1808 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
1809 | - $where['EVT_wp_user'] = get_current_user_id(); |
|
1810 | - } else { |
|
1811 | - if (! isset($where['status'])) { |
|
1812 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
1813 | - $where['OR'] = array( |
|
1814 | - 'status*restrict_private' => array('!=', 'private'), |
|
1815 | - 'AND' => array( |
|
1816 | - 'status*inclusive' => array('=', 'private'), |
|
1817 | - 'EVT_wp_user' => get_current_user_id(), |
|
1818 | - ), |
|
1819 | - ); |
|
1820 | - } |
|
1821 | - } |
|
1822 | - } |
|
1823 | - if (isset($this->_req_data['EVT_wp_user'])) { |
|
1824 | - if ($this->_req_data['EVT_wp_user'] != get_current_user_id() |
|
1825 | - && EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events') |
|
1826 | - ) { |
|
1827 | - $where['EVT_wp_user'] = $this->_req_data['EVT_wp_user']; |
|
1828 | - } |
|
1829 | - } |
|
1830 | - // search query handling |
|
1831 | - if (isset($this->_req_data['s'])) { |
|
1832 | - $search_string = '%' . $this->_req_data['s'] . '%'; |
|
1833 | - $where['OR'] = array( |
|
1834 | - 'EVT_name' => array('LIKE', $search_string), |
|
1835 | - 'EVT_desc' => array('LIKE', $search_string), |
|
1836 | - 'EVT_short_desc' => array('LIKE', $search_string), |
|
1837 | - ); |
|
1838 | - } |
|
1839 | - $where = apply_filters('FHEE__Events_Admin_Page__get_events__where', $where, $this->_req_data); |
|
1840 | - $query_params = apply_filters( |
|
1841 | - 'FHEE__Events_Admin_Page__get_events__query_params', |
|
1842 | - array( |
|
1843 | - $where, |
|
1844 | - 'limit' => $limit, |
|
1845 | - 'order_by' => $orderby, |
|
1846 | - 'order' => $order, |
|
1847 | - 'group_by' => 'EVT_ID', |
|
1848 | - ), |
|
1849 | - $this->_req_data |
|
1850 | - ); |
|
1851 | - // let's first check if we have special requests coming in. |
|
1852 | - if (isset($this->_req_data['active_status'])) { |
|
1853 | - switch ($this->_req_data['active_status']) { |
|
1854 | - case 'upcoming': |
|
1855 | - return $EEME->get_upcoming_events($query_params, $count); |
|
1856 | - break; |
|
1857 | - case 'expired': |
|
1858 | - return $EEME->get_expired_events($query_params, $count); |
|
1859 | - break; |
|
1860 | - case 'active': |
|
1861 | - return $EEME->get_active_events($query_params, $count); |
|
1862 | - break; |
|
1863 | - case 'inactive': |
|
1864 | - return $EEME->get_inactive_events($query_params, $count); |
|
1865 | - break; |
|
1866 | - } |
|
1867 | - } |
|
1868 | - $events = $count ? $EEME->count(array($where), 'EVT_ID', true) : $EEME->get_all($query_params); |
|
1869 | - return $events; |
|
1870 | - } |
|
1871 | - |
|
1872 | - |
|
1873 | - /** |
|
1874 | - * handling for WordPress CPT actions (trash, restore, delete) |
|
1875 | - * |
|
1876 | - * @param string $post_id |
|
1877 | - */ |
|
1878 | - public function trash_cpt_item($post_id) |
|
1879 | - { |
|
1880 | - $this->_req_data['EVT_ID'] = $post_id; |
|
1881 | - $this->_trash_or_restore_event('trash', false); |
|
1882 | - } |
|
1883 | - |
|
1884 | - |
|
1885 | - /** |
|
1886 | - * @param string $post_id |
|
1887 | - */ |
|
1888 | - public function restore_cpt_item($post_id) |
|
1889 | - { |
|
1890 | - $this->_req_data['EVT_ID'] = $post_id; |
|
1891 | - $this->_trash_or_restore_event('draft', false); |
|
1892 | - } |
|
1893 | - |
|
1894 | - |
|
1895 | - /** |
|
1896 | - * @param string $post_id |
|
1897 | - */ |
|
1898 | - public function delete_cpt_item($post_id) |
|
1899 | - { |
|
1900 | - $this->_req_data['EVT_ID'] = $post_id; |
|
1901 | - $this->_delete_event(false); |
|
1902 | - } |
|
1903 | - |
|
1904 | - |
|
1905 | - /** |
|
1906 | - * _trash_or_restore_event |
|
1907 | - * |
|
1908 | - * @access protected |
|
1909 | - * @param string $event_status |
|
1910 | - * @param bool $redirect_after |
|
1911 | - */ |
|
1912 | - protected function _trash_or_restore_event($event_status = 'trash', $redirect_after = true) |
|
1913 | - { |
|
1914 | - // determine the event id and set to array. |
|
1915 | - $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : false; |
|
1916 | - // loop thru events |
|
1917 | - if ($EVT_ID) { |
|
1918 | - // clean status |
|
1919 | - $event_status = sanitize_key($event_status); |
|
1920 | - // grab status |
|
1921 | - if (! empty($event_status)) { |
|
1922 | - $success = $this->_change_event_status($EVT_ID, $event_status); |
|
1923 | - } else { |
|
1924 | - $success = false; |
|
1925 | - $msg = esc_html__( |
|
1926 | - 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
1927 | - 'event_espresso' |
|
1928 | - ); |
|
1929 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1930 | - } |
|
1931 | - } else { |
|
1932 | - $success = false; |
|
1933 | - $msg = esc_html__( |
|
1934 | - 'An error occurred. The event could not be moved to the trash because a valid event ID was not not supplied.', |
|
1935 | - 'event_espresso' |
|
1936 | - ); |
|
1937 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1938 | - } |
|
1939 | - $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
1940 | - if ($redirect_after) { |
|
1941 | - $this->_redirect_after_action($success, 'Event', $action, array('action' => 'default')); |
|
1942 | - } |
|
1943 | - } |
|
1944 | - |
|
1945 | - |
|
1946 | - /** |
|
1947 | - * _trash_or_restore_events |
|
1948 | - * |
|
1949 | - * @access protected |
|
1950 | - * @param string $event_status |
|
1951 | - * @return void |
|
1952 | - */ |
|
1953 | - protected function _trash_or_restore_events($event_status = 'trash') |
|
1954 | - { |
|
1955 | - // clean status |
|
1956 | - $event_status = sanitize_key($event_status); |
|
1957 | - // grab status |
|
1958 | - if (! empty($event_status)) { |
|
1959 | - $success = true; |
|
1960 | - // determine the event id and set to array. |
|
1961 | - $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
1962 | - // loop thru events |
|
1963 | - foreach ($EVT_IDs as $EVT_ID) { |
|
1964 | - if ($EVT_ID = absint($EVT_ID)) { |
|
1965 | - $results = $this->_change_event_status($EVT_ID, $event_status); |
|
1966 | - $success = $results !== false ? $success : false; |
|
1967 | - } else { |
|
1968 | - $msg = sprintf( |
|
1969 | - esc_html__( |
|
1970 | - 'An error occurred. Event #%d could not be moved to the trash because a valid event ID was not not supplied.', |
|
1971 | - 'event_espresso' |
|
1972 | - ), |
|
1973 | - $EVT_ID |
|
1974 | - ); |
|
1975 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1976 | - $success = false; |
|
1977 | - } |
|
1978 | - } |
|
1979 | - } else { |
|
1980 | - $success = false; |
|
1981 | - $msg = esc_html__( |
|
1982 | - 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
1983 | - 'event_espresso' |
|
1984 | - ); |
|
1985 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1986 | - } |
|
1987 | - // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
1988 | - $success = $success ? 2 : false; |
|
1989 | - $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
1990 | - $this->_redirect_after_action($success, 'Events', $action, array('action' => 'default')); |
|
1991 | - } |
|
1992 | - |
|
1993 | - |
|
1994 | - /** |
|
1995 | - * _trash_or_restore_events |
|
1996 | - * |
|
1997 | - * @access private |
|
1998 | - * @param int $EVT_ID |
|
1999 | - * @param string $event_status |
|
2000 | - * @return bool |
|
2001 | - */ |
|
2002 | - private function _change_event_status($EVT_ID = 0, $event_status = '') |
|
2003 | - { |
|
2004 | - // grab event id |
|
2005 | - if (! $EVT_ID) { |
|
2006 | - $msg = esc_html__( |
|
2007 | - 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
2008 | - 'event_espresso' |
|
2009 | - ); |
|
2010 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2011 | - return false; |
|
2012 | - } |
|
2013 | - $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
2014 | - // clean status |
|
2015 | - $event_status = sanitize_key($event_status); |
|
2016 | - // grab status |
|
2017 | - if (empty($event_status)) { |
|
2018 | - $msg = esc_html__( |
|
2019 | - 'An error occurred. No Event Status or an invalid Event Status was received.', |
|
2020 | - 'event_espresso' |
|
2021 | - ); |
|
2022 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2023 | - return false; |
|
2024 | - } |
|
2025 | - // was event trashed or restored ? |
|
2026 | - switch ($event_status) { |
|
2027 | - case 'draft': |
|
2028 | - $action = 'restored from the trash'; |
|
2029 | - $hook = 'AHEE_event_restored_from_trash'; |
|
2030 | - break; |
|
2031 | - case 'trash': |
|
2032 | - $action = 'moved to the trash'; |
|
2033 | - $hook = 'AHEE_event_moved_to_trash'; |
|
2034 | - break; |
|
2035 | - default: |
|
2036 | - $action = 'updated'; |
|
2037 | - $hook = false; |
|
2038 | - } |
|
2039 | - // use class to change status |
|
2040 | - $this->_cpt_model_obj->set_status($event_status); |
|
2041 | - $success = $this->_cpt_model_obj->save(); |
|
2042 | - if ($success === false) { |
|
2043 | - $msg = sprintf(esc_html__('An error occurred. The event could not be %s.', 'event_espresso'), $action); |
|
2044 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2045 | - return false; |
|
2046 | - } |
|
2047 | - if ($hook) { |
|
2048 | - do_action($hook); |
|
2049 | - } |
|
2050 | - return true; |
|
2051 | - } |
|
2052 | - |
|
2053 | - |
|
2054 | - /** |
|
2055 | - * _delete_event |
|
2056 | - * |
|
2057 | - * @access protected |
|
2058 | - * @param bool $redirect_after |
|
2059 | - */ |
|
2060 | - protected function _delete_event($redirect_after = true) |
|
2061 | - { |
|
2062 | - // determine the event id and set to array. |
|
2063 | - $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : null; |
|
2064 | - $EVT_ID = isset($this->_req_data['post']) ? absint($this->_req_data['post']) : $EVT_ID; |
|
2065 | - // loop thru events |
|
2066 | - if ($EVT_ID) { |
|
2067 | - $success = $this->_permanently_delete_event($EVT_ID); |
|
2068 | - // get list of events with no prices |
|
2069 | - $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
2070 | - // remove this event from the list of events with no prices |
|
2071 | - if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
2072 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2073 | - } |
|
2074 | - update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
2075 | - } else { |
|
2076 | - $success = false; |
|
2077 | - $msg = esc_html__( |
|
2078 | - 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
2079 | - 'event_espresso' |
|
2080 | - ); |
|
2081 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2082 | - } |
|
2083 | - if ($redirect_after) { |
|
2084 | - $this->_redirect_after_action( |
|
2085 | - $success, |
|
2086 | - 'Event', |
|
2087 | - 'deleted', |
|
2088 | - array('action' => 'default', 'status' => 'trash') |
|
2089 | - ); |
|
2090 | - } |
|
2091 | - } |
|
2092 | - |
|
2093 | - |
|
2094 | - /** |
|
2095 | - * _delete_events |
|
2096 | - * |
|
2097 | - * @access protected |
|
2098 | - * @return void |
|
2099 | - */ |
|
2100 | - protected function _delete_events() |
|
2101 | - { |
|
2102 | - $success = true; |
|
2103 | - // get list of events with no prices |
|
2104 | - $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
2105 | - // determine the event id and set to array. |
|
2106 | - $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
2107 | - // loop thru events |
|
2108 | - foreach ($EVT_IDs as $EVT_ID) { |
|
2109 | - $EVT_ID = absint($EVT_ID); |
|
2110 | - if ($EVT_ID) { |
|
2111 | - $results = $this->_permanently_delete_event($EVT_ID); |
|
2112 | - $success = $results !== false ? $success : false; |
|
2113 | - // remove this event from the list of events with no prices |
|
2114 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2115 | - } else { |
|
2116 | - $success = false; |
|
2117 | - $msg = esc_html__( |
|
2118 | - 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
2119 | - 'event_espresso' |
|
2120 | - ); |
|
2121 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2122 | - } |
|
2123 | - } |
|
2124 | - update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
2125 | - // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
2126 | - $success = $success ? 2 : false; |
|
2127 | - $this->_redirect_after_action($success, 'Events', 'deleted', array('action' => 'default')); |
|
2128 | - } |
|
2129 | - |
|
2130 | - |
|
2131 | - /** |
|
2132 | - * _permanently_delete_event |
|
2133 | - * |
|
2134 | - * @access private |
|
2135 | - * @param int $EVT_ID |
|
2136 | - * @return bool |
|
2137 | - */ |
|
2138 | - private function _permanently_delete_event($EVT_ID = 0) |
|
2139 | - { |
|
2140 | - // grab event id |
|
2141 | - if (! $EVT_ID) { |
|
2142 | - $msg = esc_html__( |
|
2143 | - 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
2144 | - 'event_espresso' |
|
2145 | - ); |
|
2146 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2147 | - return false; |
|
2148 | - } |
|
2149 | - if (! $this->_cpt_model_obj instanceof EE_Event |
|
2150 | - || $this->_cpt_model_obj->ID() !== $EVT_ID |
|
2151 | - ) { |
|
2152 | - $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
2153 | - } |
|
2154 | - if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
2155 | - return false; |
|
2156 | - } |
|
2157 | - // need to delete related tickets and prices first. |
|
2158 | - $datetimes = $this->_cpt_model_obj->get_many_related('Datetime'); |
|
2159 | - foreach ($datetimes as $datetime) { |
|
2160 | - $this->_cpt_model_obj->_remove_relation_to($datetime, 'Datetime'); |
|
2161 | - $tickets = $datetime->get_many_related('Ticket'); |
|
2162 | - foreach ($tickets as $ticket) { |
|
2163 | - $ticket->_remove_relation_to($datetime, 'Datetime'); |
|
2164 | - $ticket->delete_related_permanently('Price'); |
|
2165 | - $ticket->delete_permanently(); |
|
2166 | - } |
|
2167 | - $datetime->delete(); |
|
2168 | - } |
|
2169 | - // what about related venues or terms? |
|
2170 | - $venues = $this->_cpt_model_obj->get_many_related('Venue'); |
|
2171 | - foreach ($venues as $venue) { |
|
2172 | - $this->_cpt_model_obj->_remove_relation_to($venue, 'Venue'); |
|
2173 | - } |
|
2174 | - // any attached question groups? |
|
2175 | - $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
|
2176 | - if (! empty($question_groups)) { |
|
2177 | - foreach ($question_groups as $question_group) { |
|
2178 | - $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
|
2179 | - } |
|
2180 | - } |
|
2181 | - // Message Template Groups |
|
2182 | - $this->_cpt_model_obj->_remove_relations('Message_Template_Group'); |
|
2183 | - /** @type EE_Term_Taxonomy[] $term_taxonomies */ |
|
2184 | - $term_taxonomies = $this->_cpt_model_obj->term_taxonomies(); |
|
2185 | - foreach ($term_taxonomies as $term_taxonomy) { |
|
2186 | - $this->_cpt_model_obj->remove_relation_to_term_taxonomy($term_taxonomy); |
|
2187 | - } |
|
2188 | - $success = $this->_cpt_model_obj->delete_permanently(); |
|
2189 | - // did it all go as planned ? |
|
2190 | - if ($success) { |
|
2191 | - $msg = sprintf(esc_html__('Event ID # %d has been deleted.', 'event_espresso'), $EVT_ID); |
|
2192 | - EE_Error::add_success($msg); |
|
2193 | - } else { |
|
2194 | - $msg = sprintf( |
|
2195 | - esc_html__('An error occurred. Event ID # %d could not be deleted.', 'event_espresso'), |
|
2196 | - $EVT_ID |
|
2197 | - ); |
|
2198 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2199 | - return false; |
|
2200 | - } |
|
2201 | - do_action('AHEE__Events_Admin_Page___permanently_delete_event__after_event_deleted', $EVT_ID); |
|
2202 | - return true; |
|
2203 | - } |
|
2204 | - |
|
2205 | - |
|
2206 | - /** |
|
2207 | - * get total number of events |
|
2208 | - * |
|
2209 | - * @access public |
|
2210 | - * @return int |
|
2211 | - */ |
|
2212 | - public function total_events() |
|
2213 | - { |
|
2214 | - $count = EEM_Event::instance()->count(array('caps' => 'read_admin'), 'EVT_ID', true); |
|
2215 | - return $count; |
|
2216 | - } |
|
2217 | - |
|
2218 | - |
|
2219 | - /** |
|
2220 | - * get total number of draft events |
|
2221 | - * |
|
2222 | - * @access public |
|
2223 | - * @return int |
|
2224 | - */ |
|
2225 | - public function total_events_draft() |
|
2226 | - { |
|
2227 | - $where = array( |
|
2228 | - 'status' => array('IN', array('draft', 'auto-draft')), |
|
2229 | - ); |
|
2230 | - $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
2231 | - return $count; |
|
2232 | - } |
|
2233 | - |
|
2234 | - |
|
2235 | - /** |
|
2236 | - * get total number of trashed events |
|
2237 | - * |
|
2238 | - * @access public |
|
2239 | - * @return int |
|
2240 | - */ |
|
2241 | - public function total_trashed_events() |
|
2242 | - { |
|
2243 | - $where = array( |
|
2244 | - 'status' => 'trash', |
|
2245 | - ); |
|
2246 | - $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
2247 | - return $count; |
|
2248 | - } |
|
2249 | - |
|
2250 | - |
|
2251 | - /** |
|
2252 | - * _default_event_settings |
|
2253 | - * This generates the Default Settings Tab |
|
2254 | - * |
|
2255 | - * @return void |
|
2256 | - * @throws EE_Error |
|
2257 | - */ |
|
2258 | - protected function _default_event_settings() |
|
2259 | - { |
|
2260 | - $this->_set_add_edit_form_tags('update_default_event_settings'); |
|
2261 | - $this->_set_publish_post_box_vars(null, false, false, null, false); |
|
2262 | - $this->_template_args['admin_page_content'] = $this->_default_event_settings_form()->get_html(); |
|
2263 | - $this->display_admin_page_with_sidebar(); |
|
2264 | - } |
|
2265 | - |
|
2266 | - |
|
2267 | - /** |
|
2268 | - * Return the form for event settings. |
|
2269 | - * |
|
2270 | - * @return EE_Form_Section_Proper |
|
2271 | - * @throws EE_Error |
|
2272 | - */ |
|
2273 | - protected function _default_event_settings_form() |
|
2274 | - { |
|
2275 | - $registration_config = EE_Registry::instance()->CFG->registration; |
|
2276 | - $registration_stati_for_selection = EEM_Registration::reg_status_array( |
|
2277 | - // exclude |
|
2278 | - array( |
|
2279 | - EEM_Registration::status_id_cancelled, |
|
2280 | - EEM_Registration::status_id_declined, |
|
2281 | - EEM_Registration::status_id_incomplete, |
|
2282 | - EEM_Registration::status_id_wait_list, |
|
2283 | - ), |
|
2284 | - true |
|
2285 | - ); |
|
2286 | - return new EE_Form_Section_Proper( |
|
2287 | - array( |
|
2288 | - 'name' => 'update_default_event_settings', |
|
2289 | - 'html_id' => 'update_default_event_settings', |
|
2290 | - 'html_class' => 'form-table', |
|
2291 | - 'layout_strategy' => new EE_Admin_Two_Column_Layout(), |
|
2292 | - 'subsections' => apply_filters( |
|
2293 | - 'FHEE__Events_Admin_Page___default_event_settings_form__form_subsections', |
|
2294 | - array( |
|
2295 | - 'default_reg_status' => new EE_Select_Input( |
|
2296 | - $registration_stati_for_selection, |
|
2297 | - array( |
|
2298 | - 'default' => isset($registration_config->default_STS_ID) |
|
2299 | - && array_key_exists( |
|
2300 | - $registration_config->default_STS_ID, |
|
2301 | - $registration_stati_for_selection |
|
2302 | - ) |
|
2303 | - ? sanitize_text_field($registration_config->default_STS_ID) |
|
2304 | - : EEM_Registration::status_id_pending_payment, |
|
2305 | - 'html_label_text' => esc_html__('Default Registration Status', 'event_espresso') |
|
2306 | - . EEH_Template::get_help_tab_link( |
|
2307 | - 'default_settings_status_help_tab' |
|
2308 | - ), |
|
2309 | - 'html_help_text' => esc_html__( |
|
2310 | - 'This setting allows you to preselect what the default registration status setting is when creating an event. Note that changing this setting does NOT retroactively apply it to existing events.', |
|
2311 | - 'event_espresso' |
|
2312 | - ), |
|
2313 | - ) |
|
2314 | - ), |
|
2315 | - 'default_max_tickets' => new EE_Integer_Input( |
|
2316 | - array( |
|
2317 | - 'default' => isset($registration_config->default_maximum_number_of_tickets) |
|
2318 | - ? $registration_config->default_maximum_number_of_tickets |
|
2319 | - : EEM_Event::get_default_additional_limit(), |
|
2320 | - 'html_label_text' => esc_html__( |
|
2321 | - 'Default Maximum Tickets Allowed Per Order:', |
|
2322 | - 'event_espresso' |
|
2323 | - ) |
|
2324 | - . EEH_Template::get_help_tab_link( |
|
2325 | - 'default_maximum_tickets_help_tab"' |
|
2326 | - ), |
|
2327 | - 'html_help_text' => esc_html__( |
|
2328 | - 'This setting allows you to indicate what will be the default for the maximum number of tickets per order when creating new events.', |
|
2329 | - 'event_espresso' |
|
2330 | - ), |
|
2331 | - ) |
|
2332 | - ), |
|
2333 | - ) |
|
2334 | - ), |
|
2335 | - ) |
|
2336 | - ); |
|
2337 | - } |
|
2338 | - |
|
2339 | - |
|
2340 | - /** |
|
2341 | - * _update_default_event_settings |
|
2342 | - * |
|
2343 | - * @access protected |
|
2344 | - * @return void |
|
2345 | - * @throws EE_Error |
|
2346 | - */ |
|
2347 | - protected function _update_default_event_settings() |
|
2348 | - { |
|
2349 | - $registration_config = EE_Registry::instance()->CFG->registration; |
|
2350 | - $form = $this->_default_event_settings_form(); |
|
2351 | - if ($form->was_submitted()) { |
|
2352 | - $form->receive_form_submission(); |
|
2353 | - if ($form->is_valid()) { |
|
2354 | - $valid_data = $form->valid_data(); |
|
2355 | - if (isset($valid_data['default_reg_status'])) { |
|
2356 | - $registration_config->default_STS_ID = $valid_data['default_reg_status']; |
|
2357 | - } |
|
2358 | - if (isset($valid_data['default_max_tickets'])) { |
|
2359 | - $registration_config->default_maximum_number_of_tickets = $valid_data['default_max_tickets']; |
|
2360 | - } |
|
2361 | - // update because data was valid! |
|
2362 | - EE_Registry::instance()->CFG->update_espresso_config(); |
|
2363 | - EE_Error::overwrite_success(); |
|
2364 | - EE_Error::add_success( |
|
2365 | - __('Default Event Settings were updated', 'event_espresso') |
|
2366 | - ); |
|
2367 | - } |
|
2368 | - } |
|
2369 | - $this->_redirect_after_action(0, '', '', array('action' => 'default_event_settings'), true); |
|
2370 | - } |
|
2371 | - |
|
2372 | - |
|
2373 | - /************* Templates *************/ |
|
2374 | - protected function _template_settings() |
|
2375 | - { |
|
2376 | - $this->_admin_page_title = esc_html__('Template Settings (Preview)', 'event_espresso'); |
|
2377 | - $this->_template_args['preview_img'] = '<img src="' |
|
2378 | - . EVENTS_ASSETS_URL |
|
2379 | - . DS |
|
2380 | - . 'images' |
|
2381 | - . DS |
|
2382 | - . 'caffeinated_template_features.jpg" alt="' |
|
2383 | - . esc_attr__('Template Settings Preview screenshot', 'event_espresso') |
|
2384 | - . '" />'; |
|
2385 | - $this->_template_args['preview_text'] = '<strong>' |
|
2386 | - . esc_html__( |
|
2387 | - 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
|
2388 | - 'event_espresso' |
|
2389 | - ) . '</strong>'; |
|
2390 | - $this->display_admin_caf_preview_page('template_settings_tab'); |
|
2391 | - } |
|
2392 | - |
|
2393 | - |
|
2394 | - /** Event Category Stuff **/ |
|
2395 | - /** |
|
2396 | - * set the _category property with the category object for the loaded page. |
|
2397 | - * |
|
2398 | - * @access private |
|
2399 | - * @return void |
|
2400 | - */ |
|
2401 | - private function _set_category_object() |
|
2402 | - { |
|
2403 | - if (isset($this->_category->id) && ! empty($this->_category->id)) { |
|
2404 | - return; |
|
2405 | - } //already have the category object so get out. |
|
2406 | - // set default category object |
|
2407 | - $this->_set_empty_category_object(); |
|
2408 | - // only set if we've got an id |
|
2409 | - if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
2410 | - return; |
|
2411 | - } |
|
2412 | - $category_id = absint($this->_req_data['EVT_CAT_ID']); |
|
2413 | - $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2414 | - if (! empty($term)) { |
|
2415 | - $this->_category->category_name = $term->name; |
|
2416 | - $this->_category->category_identifier = $term->slug; |
|
2417 | - $this->_category->category_desc = $term->description; |
|
2418 | - $this->_category->id = $term->term_id; |
|
2419 | - $this->_category->parent = $term->parent; |
|
2420 | - } |
|
2421 | - } |
|
2422 | - |
|
2423 | - |
|
2424 | - /** |
|
2425 | - * Clears out category properties. |
|
2426 | - */ |
|
2427 | - private function _set_empty_category_object() |
|
2428 | - { |
|
2429 | - $this->_category = new stdClass(); |
|
2430 | - $this->_category->category_name = $this->_category->category_identifier = $this->_category->category_desc = ''; |
|
2431 | - $this->_category->id = $this->_category->parent = 0; |
|
2432 | - } |
|
2433 | - |
|
2434 | - |
|
2435 | - /** |
|
2436 | - * @throws EE_Error |
|
2437 | - */ |
|
2438 | - protected function _category_list_table() |
|
2439 | - { |
|
2440 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
2441 | - $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
|
2442 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
2443 | - 'add_category', |
|
2444 | - 'add_category', |
|
2445 | - array(), |
|
2446 | - 'add-new-h2' |
|
2447 | - ); |
|
2448 | - $this->display_admin_list_table_page_with_sidebar(); |
|
2449 | - } |
|
2450 | - |
|
2451 | - |
|
2452 | - /** |
|
2453 | - * Output category details view. |
|
2454 | - */ |
|
2455 | - protected function _category_details($view) |
|
2456 | - { |
|
2457 | - // load formatter helper |
|
2458 | - // load field generator helper |
|
2459 | - $route = $view == 'edit' ? 'update_category' : 'insert_category'; |
|
2460 | - $this->_set_add_edit_form_tags($route); |
|
2461 | - $this->_set_category_object(); |
|
2462 | - $id = ! empty($this->_category->id) ? $this->_category->id : ''; |
|
2463 | - $delete_action = 'delete_category'; |
|
2464 | - // custom redirect |
|
2465 | - $redirect = EE_Admin_Page::add_query_args_and_nonce( |
|
2466 | - array('action' => 'category_list'), |
|
2467 | - $this->_admin_base_url |
|
2468 | - ); |
|
2469 | - $this->_set_publish_post_box_vars('EVT_CAT_ID', $id, $delete_action, $redirect); |
|
2470 | - // take care of contents |
|
2471 | - $this->_template_args['admin_page_content'] = $this->_category_details_content(); |
|
2472 | - $this->display_admin_page_with_sidebar(); |
|
2473 | - } |
|
2474 | - |
|
2475 | - |
|
2476 | - /** |
|
2477 | - * Output category details content. |
|
2478 | - */ |
|
2479 | - protected function _category_details_content() |
|
2480 | - { |
|
2481 | - $editor_args['category_desc'] = array( |
|
2482 | - 'type' => 'wp_editor', |
|
2483 | - 'value' => EEH_Formatter::admin_format_content($this->_category->category_desc), |
|
2484 | - 'class' => 'my_editor_custom', |
|
2485 | - 'wpeditor_args' => array('media_buttons' => false), |
|
2486 | - ); |
|
2487 | - $_wp_editor = $this->_generate_admin_form_fields($editor_args, 'array'); |
|
2488 | - $all_terms = get_terms( |
|
2489 | - array(EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY), |
|
2490 | - array('hide_empty' => 0, 'exclude' => array($this->_category->id)) |
|
2491 | - ); |
|
2492 | - // setup category select for term parents. |
|
2493 | - $category_select_values[] = array( |
|
2494 | - 'text' => esc_html__('No Parent', 'event_espresso'), |
|
2495 | - 'id' => 0, |
|
2496 | - ); |
|
2497 | - foreach ($all_terms as $term) { |
|
2498 | - $category_select_values[] = array( |
|
2499 | - 'text' => $term->name, |
|
2500 | - 'id' => $term->term_id, |
|
2501 | - ); |
|
2502 | - } |
|
2503 | - $category_select = EEH_Form_Fields::select_input( |
|
2504 | - 'category_parent', |
|
2505 | - $category_select_values, |
|
2506 | - $this->_category->parent |
|
2507 | - ); |
|
2508 | - $template_args = array( |
|
2509 | - 'category' => $this->_category, |
|
2510 | - 'category_select' => $category_select, |
|
2511 | - 'unique_id_info_help_link' => $this->_get_help_tab_link('unique_id_info'), |
|
2512 | - 'category_desc_editor' => $_wp_editor['category_desc']['field'], |
|
2513 | - 'disable' => '', |
|
2514 | - 'disabled_message' => false, |
|
2515 | - ); |
|
2516 | - $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
2517 | - return EEH_Template::display_template($template, $template_args, true); |
|
2518 | - } |
|
2519 | - |
|
2520 | - |
|
2521 | - /** |
|
2522 | - * Handles deleting categories. |
|
2523 | - */ |
|
2524 | - protected function _delete_categories() |
|
2525 | - { |
|
2526 | - $cat_ids = isset($this->_req_data['EVT_CAT_ID']) ? (array) $this->_req_data['EVT_CAT_ID'] |
|
2527 | - : (array) $this->_req_data['category_id']; |
|
2528 | - foreach ($cat_ids as $cat_id) { |
|
2529 | - $this->_delete_category($cat_id); |
|
2530 | - } |
|
2531 | - // doesn't matter what page we're coming from... we're going to the same place after delete. |
|
2532 | - $query_args = array( |
|
2533 | - 'action' => 'category_list', |
|
2534 | - ); |
|
2535 | - $this->_redirect_after_action(0, '', '', $query_args); |
|
2536 | - } |
|
2537 | - |
|
2538 | - |
|
2539 | - /** |
|
2540 | - * Handles deleting specific category. |
|
2541 | - * |
|
2542 | - * @param int $cat_id |
|
2543 | - */ |
|
2544 | - protected function _delete_category($cat_id) |
|
2545 | - { |
|
2546 | - $cat_id = absint($cat_id); |
|
2547 | - wp_delete_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2548 | - } |
|
2549 | - |
|
2550 | - |
|
2551 | - /** |
|
2552 | - * Handles triggering the update or insertion of a new category. |
|
2553 | - * |
|
2554 | - * @param bool $new_category true means we're triggering the insert of a new category. |
|
2555 | - */ |
|
2556 | - protected function _insert_or_update_category($new_category) |
|
2557 | - { |
|
2558 | - $cat_id = $new_category ? $this->_insert_category() : $this->_insert_category(true); |
|
2559 | - $success = 0; // we already have a success message so lets not send another. |
|
2560 | - if ($cat_id) { |
|
2561 | - $query_args = array( |
|
2562 | - 'action' => 'edit_category', |
|
2563 | - 'EVT_CAT_ID' => $cat_id, |
|
2564 | - ); |
|
2565 | - } else { |
|
2566 | - $query_args = array('action' => 'add_category'); |
|
2567 | - } |
|
2568 | - $this->_redirect_after_action($success, '', '', $query_args, true); |
|
2569 | - } |
|
2570 | - |
|
2571 | - |
|
2572 | - /** |
|
2573 | - * Inserts or updates category |
|
2574 | - * |
|
2575 | - * @param bool $update (true indicates we're updating a category). |
|
2576 | - * @return bool|mixed|string |
|
2577 | - */ |
|
2578 | - private function _insert_category($update = false) |
|
2579 | - { |
|
2580 | - $cat_id = $update ? $this->_req_data['EVT_CAT_ID'] : ''; |
|
2581 | - $category_name = isset($this->_req_data['category_name']) ? $this->_req_data['category_name'] : ''; |
|
2582 | - $category_desc = isset($this->_req_data['category_desc']) ? $this->_req_data['category_desc'] : ''; |
|
2583 | - $category_parent = isset($this->_req_data['category_parent']) ? $this->_req_data['category_parent'] : 0; |
|
2584 | - if (empty($category_name)) { |
|
2585 | - $msg = esc_html__('You must add a name for the category.', 'event_espresso'); |
|
2586 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2587 | - return false; |
|
2588 | - } |
|
2589 | - $term_args = array( |
|
2590 | - 'name' => $category_name, |
|
2591 | - 'description' => $category_desc, |
|
2592 | - 'parent' => $category_parent, |
|
2593 | - ); |
|
2594 | - // was the category_identifier input disabled? |
|
2595 | - if (isset($this->_req_data['category_identifier'])) { |
|
2596 | - $term_args['slug'] = $this->_req_data['category_identifier']; |
|
2597 | - } |
|
2598 | - $insert_ids = $update |
|
2599 | - ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
|
2600 | - : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
|
2601 | - if (! is_array($insert_ids)) { |
|
2602 | - $msg = esc_html__( |
|
2603 | - 'An error occurred and the category has not been saved to the database.', |
|
2604 | - 'event_espresso' |
|
2605 | - ); |
|
2606 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2607 | - } else { |
|
2608 | - $cat_id = $insert_ids['term_id']; |
|
2609 | - $msg = sprintf(esc_html__('The category %s was successfully saved', 'event_espresso'), $category_name); |
|
2610 | - EE_Error::add_success($msg); |
|
2611 | - } |
|
2612 | - return $cat_id; |
|
2613 | - } |
|
2614 | - |
|
2615 | - |
|
2616 | - /** |
|
2617 | - * Gets categories or count of categories matching the arguments in the request. |
|
2618 | - * |
|
2619 | - * @param int $per_page |
|
2620 | - * @param int $current_page |
|
2621 | - * @param bool $count |
|
2622 | - * @return EE_Base_Class[]|EE_Term_Taxonomy[]|int |
|
2623 | - */ |
|
2624 | - public function get_categories($per_page = 10, $current_page = 1, $count = false) |
|
2625 | - { |
|
2626 | - // testing term stuff |
|
2627 | - $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'Term.term_id'; |
|
2628 | - $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : 'DESC'; |
|
2629 | - $limit = ($current_page - 1) * $per_page; |
|
2630 | - $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2631 | - if (isset($this->_req_data['s'])) { |
|
2632 | - $sstr = '%' . $this->_req_data['s'] . '%'; |
|
2633 | - $where['OR'] = array( |
|
2634 | - 'Term.name' => array('LIKE', $sstr), |
|
2635 | - 'description' => array('LIKE', $sstr), |
|
2636 | - ); |
|
2637 | - } |
|
2638 | - $query_params = array( |
|
2639 | - $where, |
|
2640 | - 'order_by' => array($orderby => $order), |
|
2641 | - 'limit' => $limit . ',' . $per_page, |
|
2642 | - 'force_join' => array('Term'), |
|
2643 | - ); |
|
2644 | - $categories = $count |
|
2645 | - ? EEM_Term_Taxonomy::instance()->count($query_params, 'term_id') |
|
2646 | - : EEM_Term_Taxonomy::instance()->get_all($query_params); |
|
2647 | - return $categories; |
|
2648 | - } |
|
2649 | - |
|
2650 | - /* end category stuff */ |
|
2651 | - /**************/ |
|
2652 | - |
|
2653 | - |
|
2654 | - /** |
|
2655 | - * Callback for the `ee_save_timezone_setting` ajax action. |
|
2656 | - * |
|
2657 | - * @throws EE_Error |
|
2658 | - */ |
|
2659 | - public function save_timezonestring_setting() |
|
2660 | - { |
|
2661 | - $timezone_string = isset($this->_req_data['timezone_selected']) |
|
2662 | - ? $this->_req_data['timezone_selected'] |
|
2663 | - : ''; |
|
2664 | - if (empty($timezone_string) || ! EEH_DTT_Helper::validate_timezone($timezone_string, false)) { |
|
2665 | - EE_Error::add_error( |
|
2666 | - esc_html__('An invalid timezone string submitted.', 'event_espresso'), |
|
2667 | - __FILE__, |
|
2668 | - __FUNCTION__, |
|
2669 | - __LINE__ |
|
2670 | - ); |
|
2671 | - $this->_template_args['error'] = true; |
|
2672 | - $this->_return_json(); |
|
2673 | - } |
|
2674 | - |
|
2675 | - update_option('timezone_string', $timezone_string); |
|
2676 | - EE_Error::add_success( |
|
2677 | - esc_html__('Your timezone string was updated.', 'event_espresso') |
|
2678 | - ); |
|
2679 | - $this->_template_args['success'] = true; |
|
2680 | - $this->_return_json(true, array('action' => 'create_new')); |
|
2681 | - } |
|
15 | + /** |
|
16 | + * This will hold the event object for event_details screen. |
|
17 | + * |
|
18 | + * @access protected |
|
19 | + * @var EE_Event $_event |
|
20 | + */ |
|
21 | + protected $_event; |
|
22 | + |
|
23 | + |
|
24 | + /** |
|
25 | + * This will hold the category object for category_details screen. |
|
26 | + * |
|
27 | + * @var stdClass $_category |
|
28 | + */ |
|
29 | + protected $_category; |
|
30 | + |
|
31 | + |
|
32 | + /** |
|
33 | + * This will hold the event model instance |
|
34 | + * |
|
35 | + * @var EEM_Event $_event_model |
|
36 | + */ |
|
37 | + protected $_event_model; |
|
38 | + |
|
39 | + |
|
40 | + /** |
|
41 | + * @var EE_Event |
|
42 | + */ |
|
43 | + protected $_cpt_model_obj = false; |
|
44 | + |
|
45 | + |
|
46 | + /** |
|
47 | + * Initialize page props for this admin page group. |
|
48 | + */ |
|
49 | + protected function _init_page_props() |
|
50 | + { |
|
51 | + $this->page_slug = EVENTS_PG_SLUG; |
|
52 | + $this->page_label = EVENTS_LABEL; |
|
53 | + $this->_admin_base_url = EVENTS_ADMIN_URL; |
|
54 | + $this->_admin_base_path = EVENTS_ADMIN; |
|
55 | + $this->_cpt_model_names = array( |
|
56 | + 'create_new' => 'EEM_Event', |
|
57 | + 'edit' => 'EEM_Event', |
|
58 | + ); |
|
59 | + $this->_cpt_edit_routes = array( |
|
60 | + 'espresso_events' => 'edit', |
|
61 | + ); |
|
62 | + add_action( |
|
63 | + 'AHEE__EE_Admin_Page_CPT__set_model_object__after_set_object', |
|
64 | + array($this, 'verify_event_edit'), |
|
65 | + 10, |
|
66 | + 2 |
|
67 | + ); |
|
68 | + } |
|
69 | + |
|
70 | + |
|
71 | + /** |
|
72 | + * Sets the ajax hooks used for this admin page group. |
|
73 | + */ |
|
74 | + protected function _ajax_hooks() |
|
75 | + { |
|
76 | + add_action('wp_ajax_ee_save_timezone_setting', array($this, 'save_timezonestring_setting')); |
|
77 | + } |
|
78 | + |
|
79 | + |
|
80 | + /** |
|
81 | + * Sets the page properties for this admin page group. |
|
82 | + */ |
|
83 | + protected function _define_page_props() |
|
84 | + { |
|
85 | + $this->_admin_page_title = EVENTS_LABEL; |
|
86 | + $this->_labels = array( |
|
87 | + 'buttons' => array( |
|
88 | + 'add' => esc_html__('Add New Event', 'event_espresso'), |
|
89 | + 'edit' => esc_html__('Edit Event', 'event_espresso'), |
|
90 | + 'delete' => esc_html__('Delete Event', 'event_espresso'), |
|
91 | + 'add_category' => esc_html__('Add New Category', 'event_espresso'), |
|
92 | + 'edit_category' => esc_html__('Edit Category', 'event_espresso'), |
|
93 | + 'delete_category' => esc_html__('Delete Category', 'event_espresso'), |
|
94 | + ), |
|
95 | + 'editor_title' => array( |
|
96 | + 'espresso_events' => esc_html__('Enter event title here', 'event_espresso'), |
|
97 | + ), |
|
98 | + 'publishbox' => array( |
|
99 | + 'create_new' => esc_html__('Save New Event', 'event_espresso'), |
|
100 | + 'edit' => esc_html__('Update Event', 'event_espresso'), |
|
101 | + 'add_category' => esc_html__('Save New Category', 'event_espresso'), |
|
102 | + 'edit_category' => esc_html__('Update Category', 'event_espresso'), |
|
103 | + 'template_settings' => esc_html__('Update Settings', 'event_espresso'), |
|
104 | + ), |
|
105 | + ); |
|
106 | + } |
|
107 | + |
|
108 | + |
|
109 | + /** |
|
110 | + * Sets the page routes property for this admin page group. |
|
111 | + */ |
|
112 | + protected function _set_page_routes() |
|
113 | + { |
|
114 | + // load formatter helper |
|
115 | + // load field generator helper |
|
116 | + // is there a evt_id in the request? |
|
117 | + $evt_id = ! empty($this->_req_data['EVT_ID']) && ! is_array($this->_req_data['EVT_ID']) |
|
118 | + ? $this->_req_data['EVT_ID'] |
|
119 | + : 0; |
|
120 | + $evt_id = ! empty($this->_req_data['post']) ? $this->_req_data['post'] : $evt_id; |
|
121 | + $this->_page_routes = array( |
|
122 | + 'default' => array( |
|
123 | + 'func' => '_events_overview_list_table', |
|
124 | + 'capability' => 'ee_read_events', |
|
125 | + ), |
|
126 | + 'create_new' => array( |
|
127 | + 'func' => '_create_new_cpt_item', |
|
128 | + 'capability' => 'ee_edit_events', |
|
129 | + ), |
|
130 | + 'edit' => array( |
|
131 | + 'func' => '_edit_cpt_item', |
|
132 | + 'capability' => 'ee_edit_event', |
|
133 | + 'obj_id' => $evt_id, |
|
134 | + ), |
|
135 | + 'copy_event' => array( |
|
136 | + 'func' => '_copy_events', |
|
137 | + 'capability' => 'ee_edit_event', |
|
138 | + 'obj_id' => $evt_id, |
|
139 | + 'noheader' => true, |
|
140 | + ), |
|
141 | + 'trash_event' => array( |
|
142 | + 'func' => '_trash_or_restore_event', |
|
143 | + 'args' => array('event_status' => 'trash'), |
|
144 | + 'capability' => 'ee_delete_event', |
|
145 | + 'obj_id' => $evt_id, |
|
146 | + 'noheader' => true, |
|
147 | + ), |
|
148 | + 'trash_events' => array( |
|
149 | + 'func' => '_trash_or_restore_events', |
|
150 | + 'args' => array('event_status' => 'trash'), |
|
151 | + 'capability' => 'ee_delete_events', |
|
152 | + 'noheader' => true, |
|
153 | + ), |
|
154 | + 'restore_event' => array( |
|
155 | + 'func' => '_trash_or_restore_event', |
|
156 | + 'args' => array('event_status' => 'draft'), |
|
157 | + 'capability' => 'ee_delete_event', |
|
158 | + 'obj_id' => $evt_id, |
|
159 | + 'noheader' => true, |
|
160 | + ), |
|
161 | + 'restore_events' => array( |
|
162 | + 'func' => '_trash_or_restore_events', |
|
163 | + 'args' => array('event_status' => 'draft'), |
|
164 | + 'capability' => 'ee_delete_events', |
|
165 | + 'noheader' => true, |
|
166 | + ), |
|
167 | + 'delete_event' => array( |
|
168 | + 'func' => '_delete_event', |
|
169 | + 'capability' => 'ee_delete_event', |
|
170 | + 'obj_id' => $evt_id, |
|
171 | + 'noheader' => true, |
|
172 | + ), |
|
173 | + 'delete_events' => array( |
|
174 | + 'func' => '_delete_events', |
|
175 | + 'capability' => 'ee_delete_events', |
|
176 | + 'noheader' => true, |
|
177 | + ), |
|
178 | + 'view_report' => array( |
|
179 | + 'func' => '_view_report', |
|
180 | + 'capablity' => 'ee_edit_events', |
|
181 | + ), |
|
182 | + 'default_event_settings' => array( |
|
183 | + 'func' => '_default_event_settings', |
|
184 | + 'capability' => 'manage_options', |
|
185 | + ), |
|
186 | + 'update_default_event_settings' => array( |
|
187 | + 'func' => '_update_default_event_settings', |
|
188 | + 'capability' => 'manage_options', |
|
189 | + 'noheader' => true, |
|
190 | + ), |
|
191 | + 'template_settings' => array( |
|
192 | + 'func' => '_template_settings', |
|
193 | + 'capability' => 'manage_options', |
|
194 | + ), |
|
195 | + // event category tab related |
|
196 | + 'add_category' => array( |
|
197 | + 'func' => '_category_details', |
|
198 | + 'capability' => 'ee_edit_event_category', |
|
199 | + 'args' => array('add'), |
|
200 | + ), |
|
201 | + 'edit_category' => array( |
|
202 | + 'func' => '_category_details', |
|
203 | + 'capability' => 'ee_edit_event_category', |
|
204 | + 'args' => array('edit'), |
|
205 | + ), |
|
206 | + 'delete_categories' => array( |
|
207 | + 'func' => '_delete_categories', |
|
208 | + 'capability' => 'ee_delete_event_category', |
|
209 | + 'noheader' => true, |
|
210 | + ), |
|
211 | + 'delete_category' => array( |
|
212 | + 'func' => '_delete_categories', |
|
213 | + 'capability' => 'ee_delete_event_category', |
|
214 | + 'noheader' => true, |
|
215 | + ), |
|
216 | + 'insert_category' => array( |
|
217 | + 'func' => '_insert_or_update_category', |
|
218 | + 'args' => array('new_category' => true), |
|
219 | + 'capability' => 'ee_edit_event_category', |
|
220 | + 'noheader' => true, |
|
221 | + ), |
|
222 | + 'update_category' => array( |
|
223 | + 'func' => '_insert_or_update_category', |
|
224 | + 'args' => array('new_category' => false), |
|
225 | + 'capability' => 'ee_edit_event_category', |
|
226 | + 'noheader' => true, |
|
227 | + ), |
|
228 | + 'category_list' => array( |
|
229 | + 'func' => '_category_list_table', |
|
230 | + 'capability' => 'ee_manage_event_categories', |
|
231 | + ), |
|
232 | + ); |
|
233 | + } |
|
234 | + |
|
235 | + |
|
236 | + /** |
|
237 | + * Set the _page_config property for this admin page group. |
|
238 | + */ |
|
239 | + protected function _set_page_config() |
|
240 | + { |
|
241 | + $this->_page_config = array( |
|
242 | + 'default' => array( |
|
243 | + 'nav' => array( |
|
244 | + 'label' => esc_html__('Overview', 'event_espresso'), |
|
245 | + 'order' => 10, |
|
246 | + ), |
|
247 | + 'list_table' => 'Events_Admin_List_Table', |
|
248 | + 'help_tabs' => array( |
|
249 | + 'events_overview_help_tab' => array( |
|
250 | + 'title' => esc_html__('Events Overview', 'event_espresso'), |
|
251 | + 'filename' => 'events_overview', |
|
252 | + ), |
|
253 | + 'events_overview_table_column_headings_help_tab' => array( |
|
254 | + 'title' => esc_html__('Events Overview Table Column Headings', 'event_espresso'), |
|
255 | + 'filename' => 'events_overview_table_column_headings', |
|
256 | + ), |
|
257 | + 'events_overview_filters_help_tab' => array( |
|
258 | + 'title' => esc_html__('Events Overview Filters', 'event_espresso'), |
|
259 | + 'filename' => 'events_overview_filters', |
|
260 | + ), |
|
261 | + 'events_overview_view_help_tab' => array( |
|
262 | + 'title' => esc_html__('Events Overview Views', 'event_espresso'), |
|
263 | + 'filename' => 'events_overview_views', |
|
264 | + ), |
|
265 | + 'events_overview_other_help_tab' => array( |
|
266 | + 'title' => esc_html__('Events Overview Other', 'event_espresso'), |
|
267 | + 'filename' => 'events_overview_other', |
|
268 | + ), |
|
269 | + ), |
|
270 | + 'help_tour' => array( |
|
271 | + 'Event_Overview_Help_Tour', |
|
272 | + // 'New_Features_Test_Help_Tour' for testing multiple help tour |
|
273 | + ), |
|
274 | + 'qtips' => array( |
|
275 | + 'EE_Event_List_Table_Tips', |
|
276 | + ), |
|
277 | + 'require_nonce' => false, |
|
278 | + ), |
|
279 | + 'create_new' => array( |
|
280 | + 'nav' => array( |
|
281 | + 'label' => esc_html__('Add Event', 'event_espresso'), |
|
282 | + 'order' => 5, |
|
283 | + 'persistent' => false, |
|
284 | + ), |
|
285 | + 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
286 | + 'help_tabs' => array( |
|
287 | + 'event_editor_help_tab' => array( |
|
288 | + 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
289 | + 'filename' => 'event_editor', |
|
290 | + ), |
|
291 | + 'event_editor_title_richtexteditor_help_tab' => array( |
|
292 | + 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
293 | + 'filename' => 'event_editor_title_richtexteditor', |
|
294 | + ), |
|
295 | + 'event_editor_venue_details_help_tab' => array( |
|
296 | + 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
297 | + 'filename' => 'event_editor_venue_details', |
|
298 | + ), |
|
299 | + 'event_editor_event_datetimes_help_tab' => array( |
|
300 | + 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
301 | + 'filename' => 'event_editor_event_datetimes', |
|
302 | + ), |
|
303 | + 'event_editor_event_tickets_help_tab' => array( |
|
304 | + 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
305 | + 'filename' => 'event_editor_event_tickets', |
|
306 | + ), |
|
307 | + 'event_editor_event_registration_options_help_tab' => array( |
|
308 | + 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
309 | + 'filename' => 'event_editor_event_registration_options', |
|
310 | + ), |
|
311 | + 'event_editor_tags_categories_help_tab' => array( |
|
312 | + 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
313 | + 'filename' => 'event_editor_tags_categories', |
|
314 | + ), |
|
315 | + 'event_editor_questions_registrants_help_tab' => array( |
|
316 | + 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
317 | + 'filename' => 'event_editor_questions_registrants', |
|
318 | + ), |
|
319 | + 'event_editor_save_new_event_help_tab' => array( |
|
320 | + 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
321 | + 'filename' => 'event_editor_save_new_event', |
|
322 | + ), |
|
323 | + 'event_editor_other_help_tab' => array( |
|
324 | + 'title' => esc_html__('Event Other', 'event_espresso'), |
|
325 | + 'filename' => 'event_editor_other', |
|
326 | + ), |
|
327 | + ), |
|
328 | + 'help_tour' => array( |
|
329 | + 'Event_Editor_Help_Tour', |
|
330 | + ), |
|
331 | + 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
332 | + 'require_nonce' => false, |
|
333 | + ), |
|
334 | + 'edit' => array( |
|
335 | + 'nav' => array( |
|
336 | + 'label' => esc_html__('Edit Event', 'event_espresso'), |
|
337 | + 'order' => 5, |
|
338 | + 'persistent' => false, |
|
339 | + 'url' => isset($this->_req_data['post']) |
|
340 | + ? EE_Admin_Page::add_query_args_and_nonce( |
|
341 | + array('post' => $this->_req_data['post'], 'action' => 'edit'), |
|
342 | + $this->_current_page_view_url |
|
343 | + ) |
|
344 | + : $this->_admin_base_url, |
|
345 | + ), |
|
346 | + 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
347 | + 'help_tabs' => array( |
|
348 | + 'event_editor_help_tab' => array( |
|
349 | + 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
350 | + 'filename' => 'event_editor', |
|
351 | + ), |
|
352 | + 'event_editor_title_richtexteditor_help_tab' => array( |
|
353 | + 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
354 | + 'filename' => 'event_editor_title_richtexteditor', |
|
355 | + ), |
|
356 | + 'event_editor_venue_details_help_tab' => array( |
|
357 | + 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
358 | + 'filename' => 'event_editor_venue_details', |
|
359 | + ), |
|
360 | + 'event_editor_event_datetimes_help_tab' => array( |
|
361 | + 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
362 | + 'filename' => 'event_editor_event_datetimes', |
|
363 | + ), |
|
364 | + 'event_editor_event_tickets_help_tab' => array( |
|
365 | + 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
366 | + 'filename' => 'event_editor_event_tickets', |
|
367 | + ), |
|
368 | + 'event_editor_event_registration_options_help_tab' => array( |
|
369 | + 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
370 | + 'filename' => 'event_editor_event_registration_options', |
|
371 | + ), |
|
372 | + 'event_editor_tags_categories_help_tab' => array( |
|
373 | + 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
374 | + 'filename' => 'event_editor_tags_categories', |
|
375 | + ), |
|
376 | + 'event_editor_questions_registrants_help_tab' => array( |
|
377 | + 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
378 | + 'filename' => 'event_editor_questions_registrants', |
|
379 | + ), |
|
380 | + 'event_editor_save_new_event_help_tab' => array( |
|
381 | + 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
382 | + 'filename' => 'event_editor_save_new_event', |
|
383 | + ), |
|
384 | + 'event_editor_other_help_tab' => array( |
|
385 | + 'title' => esc_html__('Event Other', 'event_espresso'), |
|
386 | + 'filename' => 'event_editor_other', |
|
387 | + ), |
|
388 | + ), |
|
389 | + 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
390 | + 'require_nonce' => false, |
|
391 | + ), |
|
392 | + 'default_event_settings' => array( |
|
393 | + 'nav' => array( |
|
394 | + 'label' => esc_html__('Default Settings', 'event_espresso'), |
|
395 | + 'order' => 40, |
|
396 | + ), |
|
397 | + 'metaboxes' => array_merge($this->_default_espresso_metaboxes, array('_publish_post_box')), |
|
398 | + 'labels' => array( |
|
399 | + 'publishbox' => esc_html__('Update Settings', 'event_espresso'), |
|
400 | + ), |
|
401 | + 'help_tabs' => array( |
|
402 | + 'default_settings_help_tab' => array( |
|
403 | + 'title' => esc_html__('Default Event Settings', 'event_espresso'), |
|
404 | + 'filename' => 'events_default_settings', |
|
405 | + ), |
|
406 | + 'default_settings_status_help_tab' => array( |
|
407 | + 'title' => esc_html__('Default Registration Status', 'event_espresso'), |
|
408 | + 'filename' => 'events_default_settings_status', |
|
409 | + ), |
|
410 | + 'default_maximum_tickets_help_tab' => array( |
|
411 | + 'title' => esc_html__('Default Maximum Tickets Per Order', 'event_espresso'), |
|
412 | + 'filename' => 'events_default_settings_max_tickets', |
|
413 | + ), |
|
414 | + ), |
|
415 | + 'help_tour' => array('Event_Default_Settings_Help_Tour'), |
|
416 | + 'require_nonce' => false, |
|
417 | + ), |
|
418 | + // template settings |
|
419 | + 'template_settings' => array( |
|
420 | + 'nav' => array( |
|
421 | + 'label' => esc_html__('Templates', 'event_espresso'), |
|
422 | + 'order' => 30, |
|
423 | + ), |
|
424 | + 'metaboxes' => $this->_default_espresso_metaboxes, |
|
425 | + 'help_tabs' => array( |
|
426 | + 'general_settings_templates_help_tab' => array( |
|
427 | + 'title' => esc_html__('Templates', 'event_espresso'), |
|
428 | + 'filename' => 'general_settings_templates', |
|
429 | + ), |
|
430 | + ), |
|
431 | + 'help_tour' => array('Templates_Help_Tour'), |
|
432 | + 'require_nonce' => false, |
|
433 | + ), |
|
434 | + // event category stuff |
|
435 | + 'add_category' => array( |
|
436 | + 'nav' => array( |
|
437 | + 'label' => esc_html__('Add Category', 'event_espresso'), |
|
438 | + 'order' => 15, |
|
439 | + 'persistent' => false, |
|
440 | + ), |
|
441 | + 'help_tabs' => array( |
|
442 | + 'add_category_help_tab' => array( |
|
443 | + 'title' => esc_html__('Add New Event Category', 'event_espresso'), |
|
444 | + 'filename' => 'events_add_category', |
|
445 | + ), |
|
446 | + ), |
|
447 | + 'help_tour' => array('Event_Add_Category_Help_Tour'), |
|
448 | + 'metaboxes' => array('_publish_post_box'), |
|
449 | + 'require_nonce' => false, |
|
450 | + ), |
|
451 | + 'edit_category' => array( |
|
452 | + 'nav' => array( |
|
453 | + 'label' => esc_html__('Edit Category', 'event_espresso'), |
|
454 | + 'order' => 15, |
|
455 | + 'persistent' => false, |
|
456 | + 'url' => isset($this->_req_data['EVT_CAT_ID']) |
|
457 | + ? add_query_arg( |
|
458 | + array('EVT_CAT_ID' => $this->_req_data['EVT_CAT_ID']), |
|
459 | + $this->_current_page_view_url |
|
460 | + ) |
|
461 | + : $this->_admin_base_url, |
|
462 | + ), |
|
463 | + 'help_tabs' => array( |
|
464 | + 'edit_category_help_tab' => array( |
|
465 | + 'title' => esc_html__('Edit Event Category', 'event_espresso'), |
|
466 | + 'filename' => 'events_edit_category', |
|
467 | + ), |
|
468 | + ), |
|
469 | + /*'help_tour' => array('Event_Edit_Category_Help_Tour'),*/ |
|
470 | + 'metaboxes' => array('_publish_post_box'), |
|
471 | + 'require_nonce' => false, |
|
472 | + ), |
|
473 | + 'category_list' => array( |
|
474 | + 'nav' => array( |
|
475 | + 'label' => esc_html__('Categories', 'event_espresso'), |
|
476 | + 'order' => 20, |
|
477 | + ), |
|
478 | + 'list_table' => 'Event_Categories_Admin_List_Table', |
|
479 | + 'help_tabs' => array( |
|
480 | + 'events_categories_help_tab' => array( |
|
481 | + 'title' => esc_html__('Event Categories', 'event_espresso'), |
|
482 | + 'filename' => 'events_categories', |
|
483 | + ), |
|
484 | + 'events_categories_table_column_headings_help_tab' => array( |
|
485 | + 'title' => esc_html__('Event Categories Table Column Headings', 'event_espresso'), |
|
486 | + 'filename' => 'events_categories_table_column_headings', |
|
487 | + ), |
|
488 | + 'events_categories_view_help_tab' => array( |
|
489 | + 'title' => esc_html__('Event Categories Views', 'event_espresso'), |
|
490 | + 'filename' => 'events_categories_views', |
|
491 | + ), |
|
492 | + 'events_categories_other_help_tab' => array( |
|
493 | + 'title' => esc_html__('Event Categories Other', 'event_espresso'), |
|
494 | + 'filename' => 'events_categories_other', |
|
495 | + ), |
|
496 | + ), |
|
497 | + 'help_tour' => array( |
|
498 | + 'Event_Categories_Help_Tour', |
|
499 | + ), |
|
500 | + 'metaboxes' => $this->_default_espresso_metaboxes, |
|
501 | + 'require_nonce' => false, |
|
502 | + ), |
|
503 | + ); |
|
504 | + } |
|
505 | + |
|
506 | + |
|
507 | + /** |
|
508 | + * Used to register any global screen options if necessary for every route in this admin page group. |
|
509 | + */ |
|
510 | + protected function _add_screen_options() |
|
511 | + { |
|
512 | + } |
|
513 | + |
|
514 | + |
|
515 | + /** |
|
516 | + * Implementing the screen options for the 'default' route. |
|
517 | + */ |
|
518 | + protected function _add_screen_options_default() |
|
519 | + { |
|
520 | + $this->_per_page_screen_option(); |
|
521 | + } |
|
522 | + |
|
523 | + |
|
524 | + /** |
|
525 | + * Implementing screen options for the category list route. |
|
526 | + */ |
|
527 | + protected function _add_screen_options_category_list() |
|
528 | + { |
|
529 | + $page_title = $this->_admin_page_title; |
|
530 | + $this->_admin_page_title = esc_html__('Categories', 'event_espresso'); |
|
531 | + $this->_per_page_screen_option(); |
|
532 | + $this->_admin_page_title = $page_title; |
|
533 | + } |
|
534 | + |
|
535 | + |
|
536 | + /** |
|
537 | + * Used to register any global feature pointers for the admin page group. |
|
538 | + */ |
|
539 | + protected function _add_feature_pointers() |
|
540 | + { |
|
541 | + } |
|
542 | + |
|
543 | + |
|
544 | + /** |
|
545 | + * Registers and enqueues any global scripts and styles for the entire admin page group. |
|
546 | + */ |
|
547 | + public function load_scripts_styles() |
|
548 | + { |
|
549 | + wp_register_style( |
|
550 | + 'events-admin-css', |
|
551 | + EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
552 | + array(), |
|
553 | + EVENT_ESPRESSO_VERSION |
|
554 | + ); |
|
555 | + wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
556 | + wp_enqueue_style('events-admin-css'); |
|
557 | + wp_enqueue_style('ee-cat-admin'); |
|
558 | + // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
|
559 | + // registers for all views |
|
560 | + // scripts |
|
561 | + wp_register_script( |
|
562 | + 'event_editor_js', |
|
563 | + EVENTS_ASSETS_URL . 'event_editor.js', |
|
564 | + array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
|
565 | + EVENT_ESPRESSO_VERSION, |
|
566 | + true |
|
567 | + ); |
|
568 | + } |
|
569 | + |
|
570 | + |
|
571 | + /** |
|
572 | + * Enqueuing scripts and styles specific to this view |
|
573 | + */ |
|
574 | + public function load_scripts_styles_create_new() |
|
575 | + { |
|
576 | + $this->load_scripts_styles_edit(); |
|
577 | + } |
|
578 | + |
|
579 | + |
|
580 | + /** |
|
581 | + * Enqueuing scripts and styles specific to this view |
|
582 | + */ |
|
583 | + public function load_scripts_styles_edit() |
|
584 | + { |
|
585 | + // styles |
|
586 | + wp_enqueue_style('espresso-ui-theme'); |
|
587 | + wp_register_style( |
|
588 | + 'event-editor-css', |
|
589 | + EVENTS_ASSETS_URL . 'event-editor.css', |
|
590 | + array('ee-admin-css'), |
|
591 | + EVENT_ESPRESSO_VERSION |
|
592 | + ); |
|
593 | + wp_enqueue_style('event-editor-css'); |
|
594 | + // scripts |
|
595 | + wp_register_script( |
|
596 | + 'event-datetime-metabox', |
|
597 | + EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
598 | + array('event_editor_js', 'ee-datepicker'), |
|
599 | + EVENT_ESPRESSO_VERSION |
|
600 | + ); |
|
601 | + wp_enqueue_script('event-datetime-metabox'); |
|
602 | + } |
|
603 | + |
|
604 | + |
|
605 | + /** |
|
606 | + * Populating the _views property for the category list table view. |
|
607 | + */ |
|
608 | + protected function _set_list_table_views_category_list() |
|
609 | + { |
|
610 | + $this->_views = array( |
|
611 | + 'all' => array( |
|
612 | + 'slug' => 'all', |
|
613 | + 'label' => esc_html__('All', 'event_espresso'), |
|
614 | + 'count' => 0, |
|
615 | + 'bulk_action' => array( |
|
616 | + 'delete_categories' => esc_html__('Delete Permanently', 'event_espresso'), |
|
617 | + ), |
|
618 | + ), |
|
619 | + ); |
|
620 | + } |
|
621 | + |
|
622 | + |
|
623 | + /** |
|
624 | + * For adding anything that fires on the admin_init hook for any route within this admin page group. |
|
625 | + */ |
|
626 | + public function admin_init() |
|
627 | + { |
|
628 | + EE_Registry::$i18n_js_strings['image_confirm'] = esc_html__( |
|
629 | + 'Do you really want to delete this image? Please remember to update your event to complete the removal.', |
|
630 | + 'event_espresso' |
|
631 | + ); |
|
632 | + } |
|
633 | + |
|
634 | + |
|
635 | + /** |
|
636 | + * For adding anything that should be triggered on the admin_notices hook for any route within this admin page |
|
637 | + * group. |
|
638 | + */ |
|
639 | + public function admin_notices() |
|
640 | + { |
|
641 | + } |
|
642 | + |
|
643 | + |
|
644 | + /** |
|
645 | + * For adding anything that should be triggered on the `admin_print_footer_scripts` hook for any route within |
|
646 | + * this admin page group. |
|
647 | + */ |
|
648 | + public function admin_footer_scripts() |
|
649 | + { |
|
650 | + } |
|
651 | + |
|
652 | + |
|
653 | + /** |
|
654 | + * Call this function to verify if an event is public and has tickets for sale. If it does, then we need to show a |
|
655 | + * warning (via EE_Error::add_error()); |
|
656 | + * |
|
657 | + * @param EE_Event $event Event object |
|
658 | + * @param string $req_type |
|
659 | + * @return void |
|
660 | + * @throws EE_Error |
|
661 | + * @access public |
|
662 | + */ |
|
663 | + public function verify_event_edit($event = null, $req_type = '') |
|
664 | + { |
|
665 | + // don't need to do this when processing |
|
666 | + if (! empty($req_type)) { |
|
667 | + return; |
|
668 | + } |
|
669 | + // no event? |
|
670 | + if (empty($event)) { |
|
671 | + // set event |
|
672 | + $event = $this->_cpt_model_obj; |
|
673 | + } |
|
674 | + // STILL no event? |
|
675 | + if (! $event instanceof EE_Event) { |
|
676 | + return; |
|
677 | + } |
|
678 | + $orig_status = $event->status(); |
|
679 | + // first check if event is active. |
|
680 | + if ($orig_status === EEM_Event::cancelled |
|
681 | + || $orig_status === EEM_Event::postponed |
|
682 | + || $event->is_expired() |
|
683 | + || $event->is_inactive() |
|
684 | + ) { |
|
685 | + return; |
|
686 | + } |
|
687 | + // made it here so it IS active... next check that any of the tickets are sold. |
|
688 | + if ($event->is_sold_out(true)) { |
|
689 | + if ($orig_status !== EEM_Event::sold_out && $event->status() !== $orig_status) { |
|
690 | + EE_Error::add_attention( |
|
691 | + sprintf( |
|
692 | + esc_html__( |
|
693 | + 'Please note that the Event Status has automatically been changed to %s because there are no more spaces available for this event. However, this change is not permanent until you update the event. You can change the status back to something else before updating if you wish.', |
|
694 | + 'event_espresso' |
|
695 | + ), |
|
696 | + EEH_Template::pretty_status(EEM_Event::sold_out, false, 'sentence') |
|
697 | + ) |
|
698 | + ); |
|
699 | + } |
|
700 | + return; |
|
701 | + } elseif ($orig_status === EEM_Event::sold_out) { |
|
702 | + EE_Error::add_attention( |
|
703 | + sprintf( |
|
704 | + esc_html__( |
|
705 | + 'Please note that the Event Status has automatically been changed to %s because more spaces have become available for this event, most likely due to abandoned transactions freeing up reserved tickets. However, this change is not permanent until you update the event. If you wish, you can change the status back to something else before updating.', |
|
706 | + 'event_espresso' |
|
707 | + ), |
|
708 | + EEH_Template::pretty_status($event->status(), false, 'sentence') |
|
709 | + ) |
|
710 | + ); |
|
711 | + } |
|
712 | + // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
|
713 | + if (! $event->tickets_on_sale()) { |
|
714 | + return; |
|
715 | + } |
|
716 | + // made it here so show warning |
|
717 | + $this->_edit_event_warning(); |
|
718 | + } |
|
719 | + |
|
720 | + |
|
721 | + /** |
|
722 | + * This is the text used for when an event is being edited that is public and has tickets for sale. |
|
723 | + * When needed, hook this into a EE_Error::add_error() notice. |
|
724 | + * |
|
725 | + * @access protected |
|
726 | + * @return void |
|
727 | + */ |
|
728 | + protected function _edit_event_warning() |
|
729 | + { |
|
730 | + // we don't want to add warnings during these requests |
|
731 | + if (isset($this->_req_data['action']) && $this->_req_data['action'] === 'editpost') { |
|
732 | + return; |
|
733 | + } |
|
734 | + EE_Error::add_attention( |
|
735 | + sprintf( |
|
736 | + esc_html__( |
|
737 | + 'Your event is open for registration. Making changes may disrupt any transactions in progress. %sLearn more%s', |
|
738 | + 'event_espresso' |
|
739 | + ), |
|
740 | + '<a class="espresso-help-tab-lnk">', |
|
741 | + '</a>' |
|
742 | + ) |
|
743 | + ); |
|
744 | + } |
|
745 | + |
|
746 | + |
|
747 | + /** |
|
748 | + * When a user is creating a new event, notify them if they haven't set their timezone. |
|
749 | + * Otherwise, do the normal logic |
|
750 | + * |
|
751 | + * @return string |
|
752 | + * @throws \EE_Error |
|
753 | + */ |
|
754 | + protected function _create_new_cpt_item() |
|
755 | + { |
|
756 | + $has_timezone_string = get_option('timezone_string'); |
|
757 | + // only nag them about setting their timezone if it's their first event, and they haven't already done it |
|
758 | + if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
759 | + EE_Error::add_attention( |
|
760 | + sprintf( |
|
761 | + __( |
|
762 | + 'Your website\'s timezone is currently set to a UTC offset. We recommend updating your timezone to a city or region near you before you create an event. Change your timezone now:%1$s%2$s%3$sChange Timezone%4$s', |
|
763 | + 'event_espresso' |
|
764 | + ), |
|
765 | + '<br>', |
|
766 | + '<select id="timezone_string" name="timezone_string" aria-describedby="timezone-description">' |
|
767 | + . EEH_DTT_Helper::wp_timezone_choice('', EEH_DTT_Helper::get_user_locale()) |
|
768 | + . '</select>', |
|
769 | + '<button class="button button-secondary timezone-submit">', |
|
770 | + '</button><span class="spinner"></span>' |
|
771 | + ), |
|
772 | + __FILE__, |
|
773 | + __FUNCTION__, |
|
774 | + __LINE__ |
|
775 | + ); |
|
776 | + } |
|
777 | + return parent::_create_new_cpt_item(); |
|
778 | + } |
|
779 | + |
|
780 | + |
|
781 | + /** |
|
782 | + * Sets the _views property for the default route in this admin page group. |
|
783 | + */ |
|
784 | + protected function _set_list_table_views_default() |
|
785 | + { |
|
786 | + $this->_views = array( |
|
787 | + 'all' => array( |
|
788 | + 'slug' => 'all', |
|
789 | + 'label' => esc_html__('View All Events', 'event_espresso'), |
|
790 | + 'count' => 0, |
|
791 | + 'bulk_action' => array( |
|
792 | + 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
793 | + ), |
|
794 | + ), |
|
795 | + 'draft' => array( |
|
796 | + 'slug' => 'draft', |
|
797 | + 'label' => esc_html__('Draft', 'event_espresso'), |
|
798 | + 'count' => 0, |
|
799 | + 'bulk_action' => array( |
|
800 | + 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
801 | + ), |
|
802 | + ), |
|
803 | + ); |
|
804 | + if (EE_Registry::instance()->CAP->current_user_can('ee_delete_events', 'espresso_events_trash_events')) { |
|
805 | + $this->_views['trash'] = array( |
|
806 | + 'slug' => 'trash', |
|
807 | + 'label' => esc_html__('Trash', 'event_espresso'), |
|
808 | + 'count' => 0, |
|
809 | + 'bulk_action' => array( |
|
810 | + 'restore_events' => esc_html__('Restore From Trash', 'event_espresso'), |
|
811 | + 'delete_events' => esc_html__('Delete Permanently', 'event_espresso'), |
|
812 | + ), |
|
813 | + ); |
|
814 | + } |
|
815 | + } |
|
816 | + |
|
817 | + |
|
818 | + /** |
|
819 | + * Provides the legend item array for the default list table view. |
|
820 | + * |
|
821 | + * @return array |
|
822 | + */ |
|
823 | + protected function _event_legend_items() |
|
824 | + { |
|
825 | + $items = array( |
|
826 | + 'view_details' => array( |
|
827 | + 'class' => 'dashicons dashicons-search', |
|
828 | + 'desc' => esc_html__('View Event', 'event_espresso'), |
|
829 | + ), |
|
830 | + 'edit_event' => array( |
|
831 | + 'class' => 'ee-icon ee-icon-calendar-edit', |
|
832 | + 'desc' => esc_html__('Edit Event Details', 'event_espresso'), |
|
833 | + ), |
|
834 | + 'view_attendees' => array( |
|
835 | + 'class' => 'dashicons dashicons-groups', |
|
836 | + 'desc' => esc_html__('View Registrations for Event', 'event_espresso'), |
|
837 | + ), |
|
838 | + ); |
|
839 | + $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
|
840 | + $statuses = array( |
|
841 | + 'sold_out_status' => array( |
|
842 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
843 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
|
844 | + ), |
|
845 | + 'active_status' => array( |
|
846 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
847 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
|
848 | + ), |
|
849 | + 'upcoming_status' => array( |
|
850 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
851 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
|
852 | + ), |
|
853 | + 'postponed_status' => array( |
|
854 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
855 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
|
856 | + ), |
|
857 | + 'cancelled_status' => array( |
|
858 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
859 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
|
860 | + ), |
|
861 | + 'expired_status' => array( |
|
862 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
863 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
|
864 | + ), |
|
865 | + 'inactive_status' => array( |
|
866 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
867 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
|
868 | + ), |
|
869 | + ); |
|
870 | + $statuses = apply_filters('FHEE__Events_Admin_Page__event_legend_items__statuses', $statuses); |
|
871 | + return array_merge($items, $statuses); |
|
872 | + } |
|
873 | + |
|
874 | + |
|
875 | + /** |
|
876 | + * @return EEM_Event |
|
877 | + */ |
|
878 | + private function _event_model() |
|
879 | + { |
|
880 | + if (! $this->_event_model instanceof EEM_Event) { |
|
881 | + $this->_event_model = EE_Registry::instance()->load_model('Event'); |
|
882 | + } |
|
883 | + return $this->_event_model; |
|
884 | + } |
|
885 | + |
|
886 | + |
|
887 | + /** |
|
888 | + * Adds extra buttons to the WP CPT permalink field row. |
|
889 | + * Method is called from parent and is hooked into the wp 'get_sample_permalink_html' filter. |
|
890 | + * |
|
891 | + * @param string $return the current html |
|
892 | + * @param int $id the post id for the page |
|
893 | + * @param string $new_title What the title is |
|
894 | + * @param string $new_slug what the slug is |
|
895 | + * @return string The new html string for the permalink area |
|
896 | + */ |
|
897 | + public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
|
898 | + { |
|
899 | + // make sure this is only when editing |
|
900 | + if (! empty($id)) { |
|
901 | + $post = get_post($id); |
|
902 | + $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
|
903 | + . esc_html__('Shortcode', 'event_espresso') |
|
904 | + . '</a> '; |
|
905 | + $return .= '<input id="shortcode" type="hidden" value="[ESPRESSO_TICKET_SELECTOR event_id=' |
|
906 | + . $post->ID |
|
907 | + . ']">'; |
|
908 | + } |
|
909 | + return $return; |
|
910 | + } |
|
911 | + |
|
912 | + |
|
913 | + /** |
|
914 | + * _events_overview_list_table |
|
915 | + * This contains the logic for showing the events_overview list |
|
916 | + * |
|
917 | + * @access protected |
|
918 | + * @return void |
|
919 | + * @throws \EE_Error |
|
920 | + */ |
|
921 | + protected function _events_overview_list_table() |
|
922 | + { |
|
923 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
924 | + $this->_template_args['after_list_table'] = ! empty($this->_template_args['after_list_table']) |
|
925 | + ? (array) $this->_template_args['after_list_table'] |
|
926 | + : array(); |
|
927 | + $this->_template_args['after_list_table']['view_event_list_button'] = EEH_HTML::br() |
|
928 | + . EEH_Template::get_button_or_link( |
|
929 | + get_post_type_archive_link('espresso_events'), |
|
930 | + esc_html__("View Event Archive Page", "event_espresso"), |
|
931 | + 'button' |
|
932 | + ); |
|
933 | + $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
|
934 | + $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
935 | + 'create_new', |
|
936 | + 'add', |
|
937 | + array(), |
|
938 | + 'add-new-h2' |
|
939 | + ); |
|
940 | + $this->display_admin_list_table_page_with_no_sidebar(); |
|
941 | + } |
|
942 | + |
|
943 | + |
|
944 | + /** |
|
945 | + * this allows for extra misc actions in the default WP publish box |
|
946 | + * |
|
947 | + * @return void |
|
948 | + */ |
|
949 | + public function extra_misc_actions_publish_box() |
|
950 | + { |
|
951 | + $this->_generate_publish_box_extra_content(); |
|
952 | + } |
|
953 | + |
|
954 | + |
|
955 | + /** |
|
956 | + * This is hooked into the WordPress do_action('save_post') hook and runs after the custom post type has been |
|
957 | + * saved. |
|
958 | + * Typically you would use this to save any additional data. |
|
959 | + * Keep in mind also that "save_post" runs on EVERY post update to the database. |
|
960 | + * ALSO very important. When a post transitions from scheduled to published, |
|
961 | + * the save_post action is fired but you will NOT have any _POST data containing any extra info you may have from |
|
962 | + * other meta saves. So MAKE sure that you handle this accordingly. |
|
963 | + * |
|
964 | + * @access protected |
|
965 | + * @abstract |
|
966 | + * @param string $post_id The ID of the cpt that was saved (so you can link relationally) |
|
967 | + * @param object $post The post object of the cpt that was saved. |
|
968 | + * @return void |
|
969 | + * @throws \EE_Error |
|
970 | + */ |
|
971 | + protected function _insert_update_cpt_item($post_id, $post) |
|
972 | + { |
|
973 | + if ($post instanceof WP_Post && $post->post_type !== 'espresso_events') { |
|
974 | + // get out we're not processing an event save. |
|
975 | + return; |
|
976 | + } |
|
977 | + $event_values = array( |
|
978 | + 'EVT_display_desc' => ! empty($this->_req_data['display_desc']) ? 1 : 0, |
|
979 | + 'EVT_display_ticket_selector' => ! empty($this->_req_data['display_ticket_selector']) ? 1 : 0, |
|
980 | + 'EVT_additional_limit' => min( |
|
981 | + apply_filters('FHEE__EE_Events_Admin__insert_update_cpt_item__EVT_additional_limit_max', 255), |
|
982 | + ! empty($this->_req_data['additional_limit']) ? $this->_req_data['additional_limit'] : null |
|
983 | + ), |
|
984 | + 'EVT_default_registration_status' => ! empty($this->_req_data['EVT_default_registration_status']) |
|
985 | + ? $this->_req_data['EVT_default_registration_status'] |
|
986 | + : EE_Registry::instance()->CFG->registration->default_STS_ID, |
|
987 | + 'EVT_member_only' => ! empty($this->_req_data['member_only']) ? 1 : 0, |
|
988 | + 'EVT_allow_overflow' => ! empty($this->_req_data['EVT_allow_overflow']) ? 1 : 0, |
|
989 | + 'EVT_timezone_string' => ! empty($this->_req_data['timezone_string']) |
|
990 | + ? $this->_req_data['timezone_string'] : null, |
|
991 | + 'EVT_external_URL' => ! empty($this->_req_data['externalURL']) |
|
992 | + ? $this->_req_data['externalURL'] : null, |
|
993 | + 'EVT_phone' => ! empty($this->_req_data['event_phone']) |
|
994 | + ? $this->_req_data['event_phone'] : null, |
|
995 | + ); |
|
996 | + // update event |
|
997 | + $success = $this->_event_model()->update_by_ID($event_values, $post_id); |
|
998 | + // get event_object for other metaboxes... though it would seem to make sense to just use $this->_event_model()->get_one_by_ID( $post_id ).. i have to setup where conditions to override the filters in the model that filter out autodraft and inherit statuses so we GET the inherit id! |
|
999 | + $get_one_where = array( |
|
1000 | + $this->_event_model()->primary_key_name() => $post_id, |
|
1001 | + 'OR' => array( |
|
1002 | + 'status' => $post->post_status, |
|
1003 | + // if trying to "Publish" a sold out event, it's status will get switched back to "sold_out" in the db, |
|
1004 | + // but the returned object here has a status of "publish", so use the original post status as well |
|
1005 | + 'status*1' => $this->_req_data['original_post_status'], |
|
1006 | + ), |
|
1007 | + ); |
|
1008 | + $event = $this->_event_model()->get_one(array($get_one_where)); |
|
1009 | + // the following are default callbacks for event attachment updates that can be overridden by caffeinated functionality and/or addons. |
|
1010 | + $event_update_callbacks = apply_filters( |
|
1011 | + 'FHEE__Events_Admin_Page___insert_update_cpt_item__event_update_callbacks', |
|
1012 | + array( |
|
1013 | + array($this, '_default_venue_update'), |
|
1014 | + array($this, '_default_tickets_update'), |
|
1015 | + ) |
|
1016 | + ); |
|
1017 | + $att_success = true; |
|
1018 | + foreach ($event_update_callbacks as $e_callback) { |
|
1019 | + $_success = is_callable($e_callback) |
|
1020 | + ? call_user_func($e_callback, $event, $this->_req_data) |
|
1021 | + : false; |
|
1022 | + // if ANY of these updates fail then we want the appropriate global error message |
|
1023 | + $att_success = ! $att_success ? $att_success : $_success; |
|
1024 | + } |
|
1025 | + // any errors? |
|
1026 | + if ($success && false === $att_success) { |
|
1027 | + EE_Error::add_error( |
|
1028 | + esc_html__( |
|
1029 | + 'Event Details saved successfully but something went wrong with saving attachments.', |
|
1030 | + 'event_espresso' |
|
1031 | + ), |
|
1032 | + __FILE__, |
|
1033 | + __FUNCTION__, |
|
1034 | + __LINE__ |
|
1035 | + ); |
|
1036 | + } elseif ($success === false) { |
|
1037 | + EE_Error::add_error( |
|
1038 | + esc_html__('Event Details did not save successfully.', 'event_espresso'), |
|
1039 | + __FILE__, |
|
1040 | + __FUNCTION__, |
|
1041 | + __LINE__ |
|
1042 | + ); |
|
1043 | + } |
|
1044 | + } |
|
1045 | + |
|
1046 | + |
|
1047 | + /** |
|
1048 | + * @see parent::restore_item() |
|
1049 | + * @param int $post_id |
|
1050 | + * @param int $revision_id |
|
1051 | + */ |
|
1052 | + protected function _restore_cpt_item($post_id, $revision_id) |
|
1053 | + { |
|
1054 | + // copy existing event meta to new post |
|
1055 | + $post_evt = $this->_event_model()->get_one_by_ID($post_id); |
|
1056 | + if ($post_evt instanceof EE_Event) { |
|
1057 | + // meta revision restore |
|
1058 | + $post_evt->restore_revision($revision_id); |
|
1059 | + // related objs restore |
|
1060 | + $post_evt->restore_revision($revision_id, array('Venue', 'Datetime', 'Price')); |
|
1061 | + } |
|
1062 | + } |
|
1063 | + |
|
1064 | + |
|
1065 | + /** |
|
1066 | + * Attach the venue to the Event |
|
1067 | + * |
|
1068 | + * @param \EE_Event $evtobj Event Object to add the venue to |
|
1069 | + * @param array $data The request data from the form |
|
1070 | + * @return bool Success or fail. |
|
1071 | + */ |
|
1072 | + protected function _default_venue_update(\EE_Event $evtobj, $data) |
|
1073 | + { |
|
1074 | + require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
1075 | + $venue_model = EE_Registry::instance()->load_model('Venue'); |
|
1076 | + $rows_affected = null; |
|
1077 | + $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
|
1078 | + // very important. If we don't have a venue name... |
|
1079 | + // then we'll get out because not necessary to create empty venue |
|
1080 | + if (empty($data['venue_title'])) { |
|
1081 | + return false; |
|
1082 | + } |
|
1083 | + $venue_array = array( |
|
1084 | + 'VNU_wp_user' => $evtobj->get('EVT_wp_user'), |
|
1085 | + 'VNU_name' => ! empty($data['venue_title']) ? $data['venue_title'] : null, |
|
1086 | + 'VNU_desc' => ! empty($data['venue_description']) ? $data['venue_description'] : null, |
|
1087 | + 'VNU_identifier' => ! empty($data['venue_identifier']) ? $data['venue_identifier'] : null, |
|
1088 | + 'VNU_short_desc' => ! empty($data['venue_short_description']) ? $data['venue_short_description'] |
|
1089 | + : null, |
|
1090 | + 'VNU_address' => ! empty($data['address']) ? $data['address'] : null, |
|
1091 | + 'VNU_address2' => ! empty($data['address2']) ? $data['address2'] : null, |
|
1092 | + 'VNU_city' => ! empty($data['city']) ? $data['city'] : null, |
|
1093 | + 'STA_ID' => ! empty($data['state']) ? $data['state'] : null, |
|
1094 | + 'CNT_ISO' => ! empty($data['countries']) ? $data['countries'] : null, |
|
1095 | + 'VNU_zip' => ! empty($data['zip']) ? $data['zip'] : null, |
|
1096 | + 'VNU_phone' => ! empty($data['venue_phone']) ? $data['venue_phone'] : null, |
|
1097 | + 'VNU_capacity' => ! empty($data['venue_capacity']) ? $data['venue_capacity'] : null, |
|
1098 | + 'VNU_url' => ! empty($data['venue_url']) ? $data['venue_url'] : null, |
|
1099 | + 'VNU_virtual_phone' => ! empty($data['virtual_phone']) ? $data['virtual_phone'] : null, |
|
1100 | + 'VNU_virtual_url' => ! empty($data['virtual_url']) ? $data['virtual_url'] : null, |
|
1101 | + 'VNU_enable_for_gmap' => isset($data['enable_for_gmap']) ? 1 : 0, |
|
1102 | + 'status' => 'publish', |
|
1103 | + ); |
|
1104 | + // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
|
1105 | + if (! empty($venue_id)) { |
|
1106 | + $update_where = array($venue_model->primary_key_name() => $venue_id); |
|
1107 | + $rows_affected = $venue_model->update($venue_array, array($update_where)); |
|
1108 | + // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
|
1109 | + $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
1110 | + return $rows_affected > 0 ? true : false; |
|
1111 | + } else { |
|
1112 | + // we insert the venue |
|
1113 | + $venue_id = $venue_model->insert($venue_array); |
|
1114 | + $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
1115 | + return ! empty($venue_id) ? true : false; |
|
1116 | + } |
|
1117 | + // when we have the ancestor come in it's already been handled by the revision save. |
|
1118 | + } |
|
1119 | + |
|
1120 | + |
|
1121 | + /** |
|
1122 | + * Handles saving everything related to Tickets (datetimes, tickets, prices) |
|
1123 | + * |
|
1124 | + * @param EE_Event $evtobj The Event object we're attaching data to |
|
1125 | + * @param array $data The request data from the form |
|
1126 | + * @return array |
|
1127 | + */ |
|
1128 | + protected function _default_tickets_update(EE_Event $evtobj, $data) |
|
1129 | + { |
|
1130 | + $success = true; |
|
1131 | + $saved_dtt = null; |
|
1132 | + $saved_tickets = array(); |
|
1133 | + $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
1134 | + foreach ($data['edit_event_datetimes'] as $row => $dtt) { |
|
1135 | + // trim all values to ensure any excess whitespace is removed. |
|
1136 | + $dtt = array_map('trim', $dtt); |
|
1137 | + $dtt['DTT_EVT_end'] = isset($dtt['DTT_EVT_end']) && ! empty($dtt['DTT_EVT_end']) ? $dtt['DTT_EVT_end'] |
|
1138 | + : $dtt['DTT_EVT_start']; |
|
1139 | + $datetime_values = array( |
|
1140 | + 'DTT_ID' => ! empty($dtt['DTT_ID']) ? $dtt['DTT_ID'] : null, |
|
1141 | + 'DTT_EVT_start' => $dtt['DTT_EVT_start'], |
|
1142 | + 'DTT_EVT_end' => $dtt['DTT_EVT_end'], |
|
1143 | + 'DTT_reg_limit' => empty($dtt['DTT_reg_limit']) ? EE_INF : $dtt['DTT_reg_limit'], |
|
1144 | + 'DTT_order' => $row, |
|
1145 | + ); |
|
1146 | + // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
|
1147 | + if (! empty($dtt['DTT_ID'])) { |
|
1148 | + $DTM = EE_Registry::instance() |
|
1149 | + ->load_model('Datetime', array($evtobj->get_timezone())) |
|
1150 | + ->get_one_by_ID($dtt['DTT_ID']); |
|
1151 | + $DTM->set_date_format($incoming_date_formats[0]); |
|
1152 | + $DTM->set_time_format($incoming_date_formats[1]); |
|
1153 | + foreach ($datetime_values as $field => $value) { |
|
1154 | + $DTM->set($field, $value); |
|
1155 | + } |
|
1156 | + // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
|
1157 | + $saved_dtts[ $DTM->ID() ] = $DTM; |
|
1158 | + } else { |
|
1159 | + $DTM = EE_Registry::instance()->load_class( |
|
1160 | + 'Datetime', |
|
1161 | + array($datetime_values, $evtobj->get_timezone(), $incoming_date_formats), |
|
1162 | + false, |
|
1163 | + false |
|
1164 | + ); |
|
1165 | + foreach ($datetime_values as $field => $value) { |
|
1166 | + $DTM->set($field, $value); |
|
1167 | + } |
|
1168 | + } |
|
1169 | + $DTM->save(); |
|
1170 | + $DTT = $evtobj->_add_relation_to($DTM, 'Datetime'); |
|
1171 | + // load DTT helper |
|
1172 | + // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
1173 | + if ($DTT->get_raw('DTT_EVT_start') > $DTT->get_raw('DTT_EVT_end')) { |
|
1174 | + $DTT->set('DTT_EVT_end', $DTT->get('DTT_EVT_start')); |
|
1175 | + $DTT = EEH_DTT_Helper::date_time_add($DTT, 'DTT_EVT_end', 'days'); |
|
1176 | + $DTT->save(); |
|
1177 | + } |
|
1178 | + // now we got to make sure we add the new DTT_ID to the $saved_dtts array because it is possible there was a new one created for the autosave. |
|
1179 | + $saved_dtt = $DTT; |
|
1180 | + $success = ! $success ? $success : $DTT; |
|
1181 | + // if ANY of these updates fail then we want the appropriate global error message. |
|
1182 | + // //todo this is actually sucky we need a better error message but this is what it is for now. |
|
1183 | + } |
|
1184 | + // no dtts get deleted so we don't do any of that logic here. |
|
1185 | + // update tickets next |
|
1186 | + $old_tickets = isset($data['ticket_IDs']) ? explode(',', $data['ticket_IDs']) : array(); |
|
1187 | + foreach ($data['edit_tickets'] as $row => $tkt) { |
|
1188 | + $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
1189 | + $update_prices = false; |
|
1190 | + $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
1191 | + ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
1192 | + // trim inputs to ensure any excess whitespace is removed. |
|
1193 | + $tkt = array_map('trim', $tkt); |
|
1194 | + if (empty($tkt['TKT_start_date'])) { |
|
1195 | + // let's use now in the set timezone. |
|
1196 | + $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
|
1197 | + $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
1198 | + } |
|
1199 | + if (empty($tkt['TKT_end_date'])) { |
|
1200 | + // use the start date of the first datetime |
|
1201 | + $dtt = $evtobj->first_datetime(); |
|
1202 | + $tkt['TKT_end_date'] = $dtt->start_date_and_time( |
|
1203 | + $incoming_date_formats[0], |
|
1204 | + $incoming_date_formats[1] |
|
1205 | + ); |
|
1206 | + } |
|
1207 | + $TKT_values = array( |
|
1208 | + 'TKT_ID' => ! empty($tkt['TKT_ID']) ? $tkt['TKT_ID'] : null, |
|
1209 | + 'TTM_ID' => ! empty($tkt['TTM_ID']) ? $tkt['TTM_ID'] : 0, |
|
1210 | + 'TKT_name' => ! empty($tkt['TKT_name']) ? $tkt['TKT_name'] : '', |
|
1211 | + 'TKT_description' => ! empty($tkt['TKT_description']) ? $tkt['TKT_description'] : '', |
|
1212 | + 'TKT_start_date' => $tkt['TKT_start_date'], |
|
1213 | + 'TKT_end_date' => $tkt['TKT_end_date'], |
|
1214 | + 'TKT_qty' => ! isset($tkt['TKT_qty']) || $tkt['TKT_qty'] === '' ? EE_INF : $tkt['TKT_qty'], |
|
1215 | + 'TKT_uses' => ! isset($tkt['TKT_uses']) || $tkt['TKT_uses'] === '' ? EE_INF : $tkt['TKT_uses'], |
|
1216 | + 'TKT_min' => empty($tkt['TKT_min']) ? 0 : $tkt['TKT_min'], |
|
1217 | + 'TKT_max' => empty($tkt['TKT_max']) ? EE_INF : $tkt['TKT_max'], |
|
1218 | + 'TKT_row' => $row, |
|
1219 | + 'TKT_order' => isset($tkt['TKT_order']) ? $tkt['TKT_order'] : $row, |
|
1220 | + 'TKT_price' => $ticket_price, |
|
1221 | + ); |
|
1222 | + // if this is a default TKT, then we need to set the TKT_ID to 0 and update accordingly, which means in turn that the prices will become new prices as well. |
|
1223 | + if (isset($tkt['TKT_is_default']) && $tkt['TKT_is_default']) { |
|
1224 | + $TKT_values['TKT_ID'] = 0; |
|
1225 | + $TKT_values['TKT_is_default'] = 0; |
|
1226 | + $TKT_values['TKT_price'] = $ticket_price; |
|
1227 | + $update_prices = true; |
|
1228 | + } |
|
1229 | + // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
|
1230 | + // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
|
1231 | + // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
|
1232 | + if (! empty($tkt['TKT_ID'])) { |
|
1233 | + $TKT = EE_Registry::instance() |
|
1234 | + ->load_model('Ticket', array($evtobj->get_timezone())) |
|
1235 | + ->get_one_by_ID($tkt['TKT_ID']); |
|
1236 | + if ($TKT instanceof EE_Ticket) { |
|
1237 | + $ticket_sold = $TKT->count_related( |
|
1238 | + 'Registration', |
|
1239 | + array( |
|
1240 | + array( |
|
1241 | + 'STS_ID' => array( |
|
1242 | + 'NOT IN', |
|
1243 | + array(EEM_Registration::status_id_incomplete), |
|
1244 | + ), |
|
1245 | + ), |
|
1246 | + ) |
|
1247 | + ) > 0 ? true : false; |
|
1248 | + // let's just check the total price for the existing ticket and determine if it matches the new total price. if they are different then we create a new ticket (if tkts sold) if they aren't different then we go ahead and modify existing ticket. |
|
1249 | + $create_new_TKT = $ticket_sold && $ticket_price != $TKT->get('TKT_price') |
|
1250 | + && ! $TKT->get('TKT_deleted'); |
|
1251 | + $TKT->set_date_format($incoming_date_formats[0]); |
|
1252 | + $TKT->set_time_format($incoming_date_formats[1]); |
|
1253 | + // set new values |
|
1254 | + foreach ($TKT_values as $field => $value) { |
|
1255 | + if ($field == 'TKT_qty') { |
|
1256 | + $TKT->set_qty($value); |
|
1257 | + } else { |
|
1258 | + $TKT->set($field, $value); |
|
1259 | + } |
|
1260 | + } |
|
1261 | + // if $create_new_TKT is false then we can safely update the existing ticket. Otherwise we have to create a new ticket. |
|
1262 | + if ($create_new_TKT) { |
|
1263 | + // archive the old ticket first |
|
1264 | + $TKT->set('TKT_deleted', 1); |
|
1265 | + $TKT->save(); |
|
1266 | + // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
|
1267 | + $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1268 | + // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
|
1269 | + $TKT = clone $TKT; |
|
1270 | + $TKT->set('TKT_ID', 0); |
|
1271 | + $TKT->set('TKT_deleted', 0); |
|
1272 | + $TKT->set('TKT_price', $ticket_price); |
|
1273 | + $TKT->set('TKT_sold', 0); |
|
1274 | + // now we need to make sure that $new prices are created as well and attached to new ticket. |
|
1275 | + $update_prices = true; |
|
1276 | + } |
|
1277 | + // make sure price is set if it hasn't been already |
|
1278 | + $TKT->set('TKT_price', $ticket_price); |
|
1279 | + } |
|
1280 | + } else { |
|
1281 | + // no TKT_id so a new TKT |
|
1282 | + $TKT_values['TKT_price'] = $ticket_price; |
|
1283 | + $TKT = EE_Registry::instance()->load_class('Ticket', array($TKT_values), false, false); |
|
1284 | + if ($TKT instanceof EE_Ticket) { |
|
1285 | + // need to reset values to properly account for the date formats |
|
1286 | + $TKT->set_date_format($incoming_date_formats[0]); |
|
1287 | + $TKT->set_time_format($incoming_date_formats[1]); |
|
1288 | + $TKT->set_timezone($evtobj->get_timezone()); |
|
1289 | + // set new values |
|
1290 | + foreach ($TKT_values as $field => $value) { |
|
1291 | + if ($field == 'TKT_qty') { |
|
1292 | + $TKT->set_qty($value); |
|
1293 | + } else { |
|
1294 | + $TKT->set($field, $value); |
|
1295 | + } |
|
1296 | + } |
|
1297 | + $update_prices = true; |
|
1298 | + } |
|
1299 | + } |
|
1300 | + // cap ticket qty by datetime reg limits |
|
1301 | + $TKT->set_qty(min($TKT->qty(), $TKT->qty('reg_limit'))); |
|
1302 | + // update ticket. |
|
1303 | + $TKT->save(); |
|
1304 | + // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
1305 | + if ($TKT->get_raw('TKT_start_date') > $TKT->get_raw('TKT_end_date')) { |
|
1306 | + $TKT->set('TKT_end_date', $TKT->get('TKT_start_date')); |
|
1307 | + $TKT = EEH_DTT_Helper::date_time_add($TKT, 'TKT_end_date', 'days'); |
|
1308 | + $TKT->save(); |
|
1309 | + } |
|
1310 | + // initially let's add the ticket to the dtt |
|
1311 | + $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
|
1312 | + $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1313 | + // add prices to ticket |
|
1314 | + $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
1315 | + } |
|
1316 | + // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
|
1317 | + $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
|
1318 | + $tickets_removed = array_diff($old_tickets, array_keys($saved_tickets)); |
|
1319 | + foreach ($tickets_removed as $id) { |
|
1320 | + $id = absint($id); |
|
1321 | + // get the ticket for this id |
|
1322 | + $tkt_to_remove = EE_Registry::instance()->load_model('Ticket')->get_one_by_ID($id); |
|
1323 | + // need to get all the related datetimes on this ticket and remove from every single one of them (remember this process can ONLY kick off if there are NO tkts_sold) |
|
1324 | + $dtts = $tkt_to_remove->get_many_related('Datetime'); |
|
1325 | + foreach ($dtts as $dtt) { |
|
1326 | + $tkt_to_remove->_remove_relation_to($dtt, 'Datetime'); |
|
1327 | + } |
|
1328 | + // need to do the same for prices (except these prices can also be deleted because again, tickets can only be trashed if they don't have any TKTs sold (otherwise they are just archived)) |
|
1329 | + $tkt_to_remove->delete_related_permanently('Price'); |
|
1330 | + // finally let's delete this ticket (which should not be blocked at this point b/c we've removed all our relationships) |
|
1331 | + $tkt_to_remove->delete_permanently(); |
|
1332 | + } |
|
1333 | + return array($saved_dtt, $saved_tickets); |
|
1334 | + } |
|
1335 | + |
|
1336 | + |
|
1337 | + /** |
|
1338 | + * This attaches a list of given prices to a ticket. |
|
1339 | + * Note we dont' have to worry about ever removing relationships (or archiving prices) because if there is a change |
|
1340 | + * in price information on a ticket, a new ticket is created anyways so the archived ticket will retain the old |
|
1341 | + * price info and prices are automatically "archived" via the ticket. |
|
1342 | + * |
|
1343 | + * @access private |
|
1344 | + * @param array $prices Array of prices from the form. |
|
1345 | + * @param EE_Ticket $ticket EE_Ticket object that prices are being attached to. |
|
1346 | + * @param bool $new_prices Whether attach existing incoming prices or create new ones. |
|
1347 | + * @return void |
|
1348 | + */ |
|
1349 | + private function _add_prices_to_ticket($prices, EE_Ticket $ticket, $new_prices = false) |
|
1350 | + { |
|
1351 | + foreach ($prices as $row => $prc) { |
|
1352 | + $PRC_values = array( |
|
1353 | + 'PRC_ID' => ! empty($prc['PRC_ID']) ? $prc['PRC_ID'] : null, |
|
1354 | + 'PRT_ID' => ! empty($prc['PRT_ID']) ? $prc['PRT_ID'] : null, |
|
1355 | + 'PRC_amount' => ! empty($prc['PRC_amount']) ? $prc['PRC_amount'] : 0, |
|
1356 | + 'PRC_name' => ! empty($prc['PRC_name']) ? $prc['PRC_name'] : '', |
|
1357 | + 'PRC_desc' => ! empty($prc['PRC_desc']) ? $prc['PRC_desc'] : '', |
|
1358 | + 'PRC_is_default' => 0, // make sure prices are NOT set as default from this context |
|
1359 | + 'PRC_order' => $row, |
|
1360 | + ); |
|
1361 | + if ($new_prices || empty($PRC_values['PRC_ID'])) { |
|
1362 | + $PRC_values['PRC_ID'] = 0; |
|
1363 | + $PRC = EE_Registry::instance()->load_class('Price', array($PRC_values), false, false); |
|
1364 | + } else { |
|
1365 | + $PRC = EE_Registry::instance()->load_model('Price')->get_one_by_ID($prc['PRC_ID']); |
|
1366 | + // update this price with new values |
|
1367 | + foreach ($PRC_values as $field => $newprc) { |
|
1368 | + $PRC->set($field, $newprc); |
|
1369 | + } |
|
1370 | + $PRC->save(); |
|
1371 | + } |
|
1372 | + $ticket->_add_relation_to($PRC, 'Price'); |
|
1373 | + } |
|
1374 | + } |
|
1375 | + |
|
1376 | + |
|
1377 | + /** |
|
1378 | + * Add in our autosave ajax handlers |
|
1379 | + * |
|
1380 | + */ |
|
1381 | + protected function _ee_autosave_create_new() |
|
1382 | + { |
|
1383 | + } |
|
1384 | + |
|
1385 | + |
|
1386 | + /** |
|
1387 | + * More autosave handlers. |
|
1388 | + */ |
|
1389 | + protected function _ee_autosave_edit() |
|
1390 | + { |
|
1391 | + return; // TEMPORARILY EXITING CAUSE THIS IS A TODO |
|
1392 | + } |
|
1393 | + |
|
1394 | + |
|
1395 | + /** |
|
1396 | + * _generate_publish_box_extra_content |
|
1397 | + */ |
|
1398 | + private function _generate_publish_box_extra_content() |
|
1399 | + { |
|
1400 | + // load formatter helper |
|
1401 | + // args for getting related registrations |
|
1402 | + $approved_query_args = array( |
|
1403 | + array( |
|
1404 | + 'REG_deleted' => 0, |
|
1405 | + 'STS_ID' => EEM_Registration::status_id_approved, |
|
1406 | + ), |
|
1407 | + ); |
|
1408 | + $not_approved_query_args = array( |
|
1409 | + array( |
|
1410 | + 'REG_deleted' => 0, |
|
1411 | + 'STS_ID' => EEM_Registration::status_id_not_approved, |
|
1412 | + ), |
|
1413 | + ); |
|
1414 | + $pending_payment_query_args = array( |
|
1415 | + array( |
|
1416 | + 'REG_deleted' => 0, |
|
1417 | + 'STS_ID' => EEM_Registration::status_id_pending_payment, |
|
1418 | + ), |
|
1419 | + ); |
|
1420 | + // publish box |
|
1421 | + $publish_box_extra_args = array( |
|
1422 | + 'view_approved_reg_url' => add_query_arg( |
|
1423 | + array( |
|
1424 | + 'action' => 'default', |
|
1425 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
1426 | + '_reg_status' => EEM_Registration::status_id_approved, |
|
1427 | + ), |
|
1428 | + REG_ADMIN_URL |
|
1429 | + ), |
|
1430 | + 'view_not_approved_reg_url' => add_query_arg( |
|
1431 | + array( |
|
1432 | + 'action' => 'default', |
|
1433 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
1434 | + '_reg_status' => EEM_Registration::status_id_not_approved, |
|
1435 | + ), |
|
1436 | + REG_ADMIN_URL |
|
1437 | + ), |
|
1438 | + 'view_pending_payment_reg_url' => add_query_arg( |
|
1439 | + array( |
|
1440 | + 'action' => 'default', |
|
1441 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
1442 | + '_reg_status' => EEM_Registration::status_id_pending_payment, |
|
1443 | + ), |
|
1444 | + REG_ADMIN_URL |
|
1445 | + ), |
|
1446 | + 'approved_regs' => $this->_cpt_model_obj->count_related( |
|
1447 | + 'Registration', |
|
1448 | + $approved_query_args |
|
1449 | + ), |
|
1450 | + 'not_approved_regs' => $this->_cpt_model_obj->count_related( |
|
1451 | + 'Registration', |
|
1452 | + $not_approved_query_args |
|
1453 | + ), |
|
1454 | + 'pending_payment_regs' => $this->_cpt_model_obj->count_related( |
|
1455 | + 'Registration', |
|
1456 | + $pending_payment_query_args |
|
1457 | + ), |
|
1458 | + 'misc_pub_section_class' => apply_filters( |
|
1459 | + 'FHEE_Events_Admin_Page___generate_publish_box_extra_content__misc_pub_section_class', |
|
1460 | + 'misc-pub-section' |
|
1461 | + ), |
|
1462 | + ); |
|
1463 | + ob_start(); |
|
1464 | + do_action( |
|
1465 | + 'AHEE__Events_Admin_Page___generate_publish_box_extra_content__event_editor_overview_add', |
|
1466 | + $this->_cpt_model_obj |
|
1467 | + ); |
|
1468 | + $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
|
1469 | + // load template |
|
1470 | + EEH_Template::display_template( |
|
1471 | + EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
1472 | + $publish_box_extra_args |
|
1473 | + ); |
|
1474 | + } |
|
1475 | + |
|
1476 | + |
|
1477 | + /** |
|
1478 | + * @return EE_Event |
|
1479 | + */ |
|
1480 | + public function get_event_object() |
|
1481 | + { |
|
1482 | + return $this->_cpt_model_obj; |
|
1483 | + } |
|
1484 | + |
|
1485 | + |
|
1486 | + |
|
1487 | + |
|
1488 | + /** METABOXES * */ |
|
1489 | + /** |
|
1490 | + * _register_event_editor_meta_boxes |
|
1491 | + * add all metaboxes related to the event_editor |
|
1492 | + * |
|
1493 | + * @return void |
|
1494 | + */ |
|
1495 | + protected function _register_event_editor_meta_boxes() |
|
1496 | + { |
|
1497 | + $this->verify_cpt_object(); |
|
1498 | + add_meta_box( |
|
1499 | + 'espresso_event_editor_tickets', |
|
1500 | + esc_html__('Event Datetime & Ticket', 'event_espresso'), |
|
1501 | + array($this, 'ticket_metabox'), |
|
1502 | + $this->page_slug, |
|
1503 | + 'normal', |
|
1504 | + 'high' |
|
1505 | + ); |
|
1506 | + add_meta_box( |
|
1507 | + 'espresso_event_editor_event_options', |
|
1508 | + esc_html__('Event Registration Options', 'event_espresso'), |
|
1509 | + array($this, 'registration_options_meta_box'), |
|
1510 | + $this->page_slug, |
|
1511 | + 'side', |
|
1512 | + 'default' |
|
1513 | + ); |
|
1514 | + // NOTE: if you're looking for other metaboxes in here, |
|
1515 | + // where a metabox has a related management page in the admin |
|
1516 | + // you will find it setup in the related management page's "_Hooks" file. |
|
1517 | + // i.e. messages metabox is found in "espresso_events_Messages_Hooks.class.php". |
|
1518 | + } |
|
1519 | + |
|
1520 | + |
|
1521 | + /** |
|
1522 | + * @throws DomainException |
|
1523 | + * @throws EE_Error |
|
1524 | + */ |
|
1525 | + public function ticket_metabox() |
|
1526 | + { |
|
1527 | + $existing_datetime_ids = $existing_ticket_ids = array(); |
|
1528 | + // defaults for template args |
|
1529 | + $template_args = array( |
|
1530 | + 'existing_datetime_ids' => '', |
|
1531 | + 'event_datetime_help_link' => '', |
|
1532 | + 'ticket_options_help_link' => '', |
|
1533 | + 'time' => null, |
|
1534 | + 'ticket_rows' => '', |
|
1535 | + 'existing_ticket_ids' => '', |
|
1536 | + 'total_ticket_rows' => 1, |
|
1537 | + 'ticket_js_structure' => '', |
|
1538 | + 'trash_icon' => 'ee-lock-icon', |
|
1539 | + 'disabled' => '', |
|
1540 | + ); |
|
1541 | + $event_id = is_object($this->_cpt_model_obj) ? $this->_cpt_model_obj->ID() : null; |
|
1542 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
1543 | + /** |
|
1544 | + * 1. Start with retrieving Datetimes |
|
1545 | + * 2. Fore each datetime get related tickets |
|
1546 | + * 3. For each ticket get related prices |
|
1547 | + */ |
|
1548 | + $times = EE_Registry::instance()->load_model('Datetime')->get_all_event_dates($event_id); |
|
1549 | + /** @type EE_Datetime $first_datetime */ |
|
1550 | + $first_datetime = reset($times); |
|
1551 | + // do we get related tickets? |
|
1552 | + if ($first_datetime instanceof EE_Datetime |
|
1553 | + && $first_datetime->ID() !== 0 |
|
1554 | + ) { |
|
1555 | + $existing_datetime_ids[] = $first_datetime->get('DTT_ID'); |
|
1556 | + $template_args['time'] = $first_datetime; |
|
1557 | + $related_tickets = $first_datetime->tickets( |
|
1558 | + array( |
|
1559 | + array('OR' => array('TKT_deleted' => 1, 'TKT_deleted*' => 0)), |
|
1560 | + 'default_where_conditions' => 'none', |
|
1561 | + ) |
|
1562 | + ); |
|
1563 | + if (! empty($related_tickets)) { |
|
1564 | + $template_args['total_ticket_rows'] = count($related_tickets); |
|
1565 | + $row = 0; |
|
1566 | + foreach ($related_tickets as $ticket) { |
|
1567 | + $existing_ticket_ids[] = $ticket->get('TKT_ID'); |
|
1568 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket, false, $row); |
|
1569 | + $row++; |
|
1570 | + } |
|
1571 | + } else { |
|
1572 | + $template_args['total_ticket_rows'] = 1; |
|
1573 | + /** @type EE_Ticket $ticket */ |
|
1574 | + $ticket = EE_Registry::instance()->load_model('Ticket')->create_default_object(); |
|
1575 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket); |
|
1576 | + } |
|
1577 | + } else { |
|
1578 | + $template_args['time'] = $times[0]; |
|
1579 | + /** @type EE_Ticket $ticket */ |
|
1580 | + $ticket = EE_Registry::instance()->load_model('Ticket')->get_all_default_tickets(); |
|
1581 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket[1]); |
|
1582 | + // NOTE: we're just sending the first default row |
|
1583 | + // (decaf can't manage default tickets so this should be sufficient); |
|
1584 | + } |
|
1585 | + $template_args['event_datetime_help_link'] = $this->_get_help_tab_link( |
|
1586 | + 'event_editor_event_datetimes_help_tab' |
|
1587 | + ); |
|
1588 | + $template_args['ticket_options_help_link'] = $this->_get_help_tab_link('ticket_options_info'); |
|
1589 | + $template_args['existing_datetime_ids'] = implode(',', $existing_datetime_ids); |
|
1590 | + $template_args['existing_ticket_ids'] = implode(',', $existing_ticket_ids); |
|
1591 | + $template_args['ticket_js_structure'] = $this->_get_ticket_row( |
|
1592 | + EE_Registry::instance()->load_model('Ticket')->create_default_object(), |
|
1593 | + true |
|
1594 | + ); |
|
1595 | + $template = apply_filters( |
|
1596 | + 'FHEE__Events_Admin_Page__ticket_metabox__template', |
|
1597 | + EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
1598 | + ); |
|
1599 | + EEH_Template::display_template($template, $template_args); |
|
1600 | + } |
|
1601 | + |
|
1602 | + |
|
1603 | + /** |
|
1604 | + * Setup an individual ticket form for the decaf event editor page |
|
1605 | + * |
|
1606 | + * @access private |
|
1607 | + * @param EE_Ticket $ticket the ticket object |
|
1608 | + * @param boolean $skeleton whether we're generating a skeleton for js manipulation |
|
1609 | + * @param int $row |
|
1610 | + * @return string generated html for the ticket row. |
|
1611 | + */ |
|
1612 | + private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
|
1613 | + { |
|
1614 | + $template_args = array( |
|
1615 | + 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
1616 | + 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
|
1617 | + : '', |
|
1618 | + 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
|
1619 | + 'TKT_ID' => $ticket->get('TKT_ID'), |
|
1620 | + 'TKT_name' => $ticket->get('TKT_name'), |
|
1621 | + 'TKT_start_date' => $skeleton ? '' : $ticket->get_date('TKT_start_date', 'Y-m-d h:i a'), |
|
1622 | + 'TKT_end_date' => $skeleton ? '' : $ticket->get_date('TKT_end_date', 'Y-m-d h:i a'), |
|
1623 | + 'TKT_is_default' => $ticket->get('TKT_is_default'), |
|
1624 | + 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
|
1625 | + 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
|
1626 | + 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
|
1627 | + 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
1628 | + && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
1629 | + ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
|
1630 | + 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
1631 | + : ' disabled=disabled', |
|
1632 | + ); |
|
1633 | + $price = $ticket->ID() !== 0 |
|
1634 | + ? $ticket->get_first_related('Price', array('default_where_conditions' => 'none')) |
|
1635 | + : EE_Registry::instance()->load_model('Price')->create_default_object(); |
|
1636 | + $price_args = array( |
|
1637 | + 'price_currency_symbol' => EE_Registry::instance()->CFG->currency->sign, |
|
1638 | + 'PRC_amount' => $price->get('PRC_amount'), |
|
1639 | + 'PRT_ID' => $price->get('PRT_ID'), |
|
1640 | + 'PRC_ID' => $price->get('PRC_ID'), |
|
1641 | + 'PRC_is_default' => $price->get('PRC_is_default'), |
|
1642 | + ); |
|
1643 | + // make sure we have default start and end dates if skeleton |
|
1644 | + // handle rows that should NOT be empty |
|
1645 | + if (empty($template_args['TKT_start_date'])) { |
|
1646 | + // if empty then the start date will be now. |
|
1647 | + $template_args['TKT_start_date'] = date('Y-m-d h:i a', current_time('timestamp')); |
|
1648 | + } |
|
1649 | + if (empty($template_args['TKT_end_date'])) { |
|
1650 | + // get the earliest datetime (if present); |
|
1651 | + $earliest_dtt = $this->_cpt_model_obj->ID() > 0 |
|
1652 | + ? $this->_cpt_model_obj->get_first_related( |
|
1653 | + 'Datetime', |
|
1654 | + array('order_by' => array('DTT_EVT_start' => 'ASC')) |
|
1655 | + ) |
|
1656 | + : null; |
|
1657 | + if (! empty($earliest_dtt)) { |
|
1658 | + $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
|
1659 | + } else { |
|
1660 | + $template_args['TKT_end_date'] = date( |
|
1661 | + 'Y-m-d h:i a', |
|
1662 | + mktime(0, 0, 0, date("m"), date("d") + 7, date("Y")) |
|
1663 | + ); |
|
1664 | + } |
|
1665 | + } |
|
1666 | + $template_args = array_merge($template_args, $price_args); |
|
1667 | + $template = apply_filters( |
|
1668 | + 'FHEE__Events_Admin_Page__get_ticket_row__template', |
|
1669 | + EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
1670 | + $ticket |
|
1671 | + ); |
|
1672 | + return EEH_Template::display_template($template, $template_args, true); |
|
1673 | + } |
|
1674 | + |
|
1675 | + |
|
1676 | + /** |
|
1677 | + * @throws DomainException |
|
1678 | + */ |
|
1679 | + public function registration_options_meta_box() |
|
1680 | + { |
|
1681 | + $yes_no_values = array( |
|
1682 | + array('id' => true, 'text' => esc_html__('Yes', 'event_espresso')), |
|
1683 | + array('id' => false, 'text' => esc_html__('No', 'event_espresso')), |
|
1684 | + ); |
|
1685 | + $default_reg_status_values = EEM_Registration::reg_status_array( |
|
1686 | + array( |
|
1687 | + EEM_Registration::status_id_cancelled, |
|
1688 | + EEM_Registration::status_id_declined, |
|
1689 | + EEM_Registration::status_id_incomplete, |
|
1690 | + ), |
|
1691 | + true |
|
1692 | + ); |
|
1693 | + // $template_args['is_active_select'] = EEH_Form_Fields::select_input('is_active', $yes_no_values, $this->_cpt_model_obj->is_active()); |
|
1694 | + $template_args['_event'] = $this->_cpt_model_obj; |
|
1695 | + $template_args['active_status'] = $this->_cpt_model_obj->pretty_active_status(false); |
|
1696 | + $template_args['additional_limit'] = $this->_cpt_model_obj->additional_limit(); |
|
1697 | + $template_args['default_registration_status'] = EEH_Form_Fields::select_input( |
|
1698 | + 'default_reg_status', |
|
1699 | + $default_reg_status_values, |
|
1700 | + $this->_cpt_model_obj->default_registration_status() |
|
1701 | + ); |
|
1702 | + $template_args['display_description'] = EEH_Form_Fields::select_input( |
|
1703 | + 'display_desc', |
|
1704 | + $yes_no_values, |
|
1705 | + $this->_cpt_model_obj->display_description() |
|
1706 | + ); |
|
1707 | + $template_args['display_ticket_selector'] = EEH_Form_Fields::select_input( |
|
1708 | + 'display_ticket_selector', |
|
1709 | + $yes_no_values, |
|
1710 | + $this->_cpt_model_obj->display_ticket_selector(), |
|
1711 | + '', |
|
1712 | + '', |
|
1713 | + false |
|
1714 | + ); |
|
1715 | + $template_args['additional_registration_options'] = apply_filters( |
|
1716 | + 'FHEE__Events_Admin_Page__registration_options_meta_box__additional_registration_options', |
|
1717 | + '', |
|
1718 | + $template_args, |
|
1719 | + $yes_no_values, |
|
1720 | + $default_reg_status_values |
|
1721 | + ); |
|
1722 | + EEH_Template::display_template( |
|
1723 | + EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
1724 | + $template_args |
|
1725 | + ); |
|
1726 | + } |
|
1727 | + |
|
1728 | + |
|
1729 | + /** |
|
1730 | + * _get_events() |
|
1731 | + * This method simply returns all the events (for the given _view and paging) |
|
1732 | + * |
|
1733 | + * @access public |
|
1734 | + * @param int $per_page count of items per page (20 default); |
|
1735 | + * @param int $current_page what is the current page being viewed. |
|
1736 | + * @param bool $count if TRUE then we just return a count of ALL events matching the given _view. |
|
1737 | + * If FALSE then we return an array of event objects |
|
1738 | + * that match the given _view and paging parameters. |
|
1739 | + * @return array an array of event objects. |
|
1740 | + */ |
|
1741 | + public function get_events($per_page = 10, $current_page = 1, $count = false) |
|
1742 | + { |
|
1743 | + $EEME = $this->_event_model(); |
|
1744 | + $offset = ($current_page - 1) * $per_page; |
|
1745 | + $limit = $count ? null : $offset . ',' . $per_page; |
|
1746 | + $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
|
1747 | + $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
|
1748 | + if (isset($this->_req_data['month_range'])) { |
|
1749 | + $pieces = explode(' ', $this->_req_data['month_range'], 3); |
|
1750 | + // simulate the FIRST day of the month, that fixes issues for months like February |
|
1751 | + // where PHP doesn't know what to assume for date. |
|
1752 | + // @see https://events.codebasehq.com/projects/event-espresso/tickets/10437 |
|
1753 | + $month_r = ! empty($pieces[0]) ? date('m', \EEH_DTT_Helper::first_of_month_timestamp($pieces[0])) : ''; |
|
1754 | + $year_r = ! empty($pieces[1]) ? $pieces[1] : ''; |
|
1755 | + } |
|
1756 | + $where = array(); |
|
1757 | + $status = isset($this->_req_data['status']) ? $this->_req_data['status'] : null; |
|
1758 | + // determine what post_status our condition will have for the query. |
|
1759 | + switch ($status) { |
|
1760 | + case 'month': |
|
1761 | + case 'today': |
|
1762 | + case null: |
|
1763 | + case 'all': |
|
1764 | + break; |
|
1765 | + case 'draft': |
|
1766 | + $where['status'] = array('IN', array('draft', 'auto-draft')); |
|
1767 | + break; |
|
1768 | + default: |
|
1769 | + $where['status'] = $status; |
|
1770 | + } |
|
1771 | + // categories? |
|
1772 | + $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
|
1773 | + ? $this->_req_data['EVT_CAT'] : null; |
|
1774 | + if (! empty($category)) { |
|
1775 | + $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
|
1776 | + $where['Term_Taxonomy.term_id'] = $category; |
|
1777 | + } |
|
1778 | + // date where conditions |
|
1779 | + $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
|
1780 | + if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
|
1781 | + $DateTime = new DateTime( |
|
1782 | + $year_r . '-' . $month_r . '-01 00:00:00', |
|
1783 | + new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
|
1784 | + ); |
|
1785 | + $start = $DateTime->format(implode(' ', $start_formats)); |
|
1786 | + $end = $DateTime->setDate( |
|
1787 | + $year_r, |
|
1788 | + $month_r, |
|
1789 | + $DateTime |
|
1790 | + ->format('t') |
|
1791 | + )->setTime(23, 59, 59) |
|
1792 | + ->format(implode(' ', $start_formats)); |
|
1793 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1794 | + } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'today') { |
|
1795 | + $DateTime = new DateTime('now', new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
1796 | + $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
1797 | + $end = $DateTime->setTime(23, 59, 59)->format(implode(' ', $start_formats)); |
|
1798 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1799 | + } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'month') { |
|
1800 | + $now = date('Y-m-01'); |
|
1801 | + $DateTime = new DateTime($now, new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
1802 | + $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
1803 | + $end = $DateTime->setDate(date('Y'), date('m'), $DateTime->format('t')) |
|
1804 | + ->setTime(23, 59, 59) |
|
1805 | + ->format(implode(' ', $start_formats)); |
|
1806 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
1807 | + } |
|
1808 | + if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
1809 | + $where['EVT_wp_user'] = get_current_user_id(); |
|
1810 | + } else { |
|
1811 | + if (! isset($where['status'])) { |
|
1812 | + if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
1813 | + $where['OR'] = array( |
|
1814 | + 'status*restrict_private' => array('!=', 'private'), |
|
1815 | + 'AND' => array( |
|
1816 | + 'status*inclusive' => array('=', 'private'), |
|
1817 | + 'EVT_wp_user' => get_current_user_id(), |
|
1818 | + ), |
|
1819 | + ); |
|
1820 | + } |
|
1821 | + } |
|
1822 | + } |
|
1823 | + if (isset($this->_req_data['EVT_wp_user'])) { |
|
1824 | + if ($this->_req_data['EVT_wp_user'] != get_current_user_id() |
|
1825 | + && EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events') |
|
1826 | + ) { |
|
1827 | + $where['EVT_wp_user'] = $this->_req_data['EVT_wp_user']; |
|
1828 | + } |
|
1829 | + } |
|
1830 | + // search query handling |
|
1831 | + if (isset($this->_req_data['s'])) { |
|
1832 | + $search_string = '%' . $this->_req_data['s'] . '%'; |
|
1833 | + $where['OR'] = array( |
|
1834 | + 'EVT_name' => array('LIKE', $search_string), |
|
1835 | + 'EVT_desc' => array('LIKE', $search_string), |
|
1836 | + 'EVT_short_desc' => array('LIKE', $search_string), |
|
1837 | + ); |
|
1838 | + } |
|
1839 | + $where = apply_filters('FHEE__Events_Admin_Page__get_events__where', $where, $this->_req_data); |
|
1840 | + $query_params = apply_filters( |
|
1841 | + 'FHEE__Events_Admin_Page__get_events__query_params', |
|
1842 | + array( |
|
1843 | + $where, |
|
1844 | + 'limit' => $limit, |
|
1845 | + 'order_by' => $orderby, |
|
1846 | + 'order' => $order, |
|
1847 | + 'group_by' => 'EVT_ID', |
|
1848 | + ), |
|
1849 | + $this->_req_data |
|
1850 | + ); |
|
1851 | + // let's first check if we have special requests coming in. |
|
1852 | + if (isset($this->_req_data['active_status'])) { |
|
1853 | + switch ($this->_req_data['active_status']) { |
|
1854 | + case 'upcoming': |
|
1855 | + return $EEME->get_upcoming_events($query_params, $count); |
|
1856 | + break; |
|
1857 | + case 'expired': |
|
1858 | + return $EEME->get_expired_events($query_params, $count); |
|
1859 | + break; |
|
1860 | + case 'active': |
|
1861 | + return $EEME->get_active_events($query_params, $count); |
|
1862 | + break; |
|
1863 | + case 'inactive': |
|
1864 | + return $EEME->get_inactive_events($query_params, $count); |
|
1865 | + break; |
|
1866 | + } |
|
1867 | + } |
|
1868 | + $events = $count ? $EEME->count(array($where), 'EVT_ID', true) : $EEME->get_all($query_params); |
|
1869 | + return $events; |
|
1870 | + } |
|
1871 | + |
|
1872 | + |
|
1873 | + /** |
|
1874 | + * handling for WordPress CPT actions (trash, restore, delete) |
|
1875 | + * |
|
1876 | + * @param string $post_id |
|
1877 | + */ |
|
1878 | + public function trash_cpt_item($post_id) |
|
1879 | + { |
|
1880 | + $this->_req_data['EVT_ID'] = $post_id; |
|
1881 | + $this->_trash_or_restore_event('trash', false); |
|
1882 | + } |
|
1883 | + |
|
1884 | + |
|
1885 | + /** |
|
1886 | + * @param string $post_id |
|
1887 | + */ |
|
1888 | + public function restore_cpt_item($post_id) |
|
1889 | + { |
|
1890 | + $this->_req_data['EVT_ID'] = $post_id; |
|
1891 | + $this->_trash_or_restore_event('draft', false); |
|
1892 | + } |
|
1893 | + |
|
1894 | + |
|
1895 | + /** |
|
1896 | + * @param string $post_id |
|
1897 | + */ |
|
1898 | + public function delete_cpt_item($post_id) |
|
1899 | + { |
|
1900 | + $this->_req_data['EVT_ID'] = $post_id; |
|
1901 | + $this->_delete_event(false); |
|
1902 | + } |
|
1903 | + |
|
1904 | + |
|
1905 | + /** |
|
1906 | + * _trash_or_restore_event |
|
1907 | + * |
|
1908 | + * @access protected |
|
1909 | + * @param string $event_status |
|
1910 | + * @param bool $redirect_after |
|
1911 | + */ |
|
1912 | + protected function _trash_or_restore_event($event_status = 'trash', $redirect_after = true) |
|
1913 | + { |
|
1914 | + // determine the event id and set to array. |
|
1915 | + $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : false; |
|
1916 | + // loop thru events |
|
1917 | + if ($EVT_ID) { |
|
1918 | + // clean status |
|
1919 | + $event_status = sanitize_key($event_status); |
|
1920 | + // grab status |
|
1921 | + if (! empty($event_status)) { |
|
1922 | + $success = $this->_change_event_status($EVT_ID, $event_status); |
|
1923 | + } else { |
|
1924 | + $success = false; |
|
1925 | + $msg = esc_html__( |
|
1926 | + 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
1927 | + 'event_espresso' |
|
1928 | + ); |
|
1929 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1930 | + } |
|
1931 | + } else { |
|
1932 | + $success = false; |
|
1933 | + $msg = esc_html__( |
|
1934 | + 'An error occurred. The event could not be moved to the trash because a valid event ID was not not supplied.', |
|
1935 | + 'event_espresso' |
|
1936 | + ); |
|
1937 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1938 | + } |
|
1939 | + $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
1940 | + if ($redirect_after) { |
|
1941 | + $this->_redirect_after_action($success, 'Event', $action, array('action' => 'default')); |
|
1942 | + } |
|
1943 | + } |
|
1944 | + |
|
1945 | + |
|
1946 | + /** |
|
1947 | + * _trash_or_restore_events |
|
1948 | + * |
|
1949 | + * @access protected |
|
1950 | + * @param string $event_status |
|
1951 | + * @return void |
|
1952 | + */ |
|
1953 | + protected function _trash_or_restore_events($event_status = 'trash') |
|
1954 | + { |
|
1955 | + // clean status |
|
1956 | + $event_status = sanitize_key($event_status); |
|
1957 | + // grab status |
|
1958 | + if (! empty($event_status)) { |
|
1959 | + $success = true; |
|
1960 | + // determine the event id and set to array. |
|
1961 | + $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
1962 | + // loop thru events |
|
1963 | + foreach ($EVT_IDs as $EVT_ID) { |
|
1964 | + if ($EVT_ID = absint($EVT_ID)) { |
|
1965 | + $results = $this->_change_event_status($EVT_ID, $event_status); |
|
1966 | + $success = $results !== false ? $success : false; |
|
1967 | + } else { |
|
1968 | + $msg = sprintf( |
|
1969 | + esc_html__( |
|
1970 | + 'An error occurred. Event #%d could not be moved to the trash because a valid event ID was not not supplied.', |
|
1971 | + 'event_espresso' |
|
1972 | + ), |
|
1973 | + $EVT_ID |
|
1974 | + ); |
|
1975 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1976 | + $success = false; |
|
1977 | + } |
|
1978 | + } |
|
1979 | + } else { |
|
1980 | + $success = false; |
|
1981 | + $msg = esc_html__( |
|
1982 | + 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
1983 | + 'event_espresso' |
|
1984 | + ); |
|
1985 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
1986 | + } |
|
1987 | + // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
1988 | + $success = $success ? 2 : false; |
|
1989 | + $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
1990 | + $this->_redirect_after_action($success, 'Events', $action, array('action' => 'default')); |
|
1991 | + } |
|
1992 | + |
|
1993 | + |
|
1994 | + /** |
|
1995 | + * _trash_or_restore_events |
|
1996 | + * |
|
1997 | + * @access private |
|
1998 | + * @param int $EVT_ID |
|
1999 | + * @param string $event_status |
|
2000 | + * @return bool |
|
2001 | + */ |
|
2002 | + private function _change_event_status($EVT_ID = 0, $event_status = '') |
|
2003 | + { |
|
2004 | + // grab event id |
|
2005 | + if (! $EVT_ID) { |
|
2006 | + $msg = esc_html__( |
|
2007 | + 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
2008 | + 'event_espresso' |
|
2009 | + ); |
|
2010 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2011 | + return false; |
|
2012 | + } |
|
2013 | + $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
2014 | + // clean status |
|
2015 | + $event_status = sanitize_key($event_status); |
|
2016 | + // grab status |
|
2017 | + if (empty($event_status)) { |
|
2018 | + $msg = esc_html__( |
|
2019 | + 'An error occurred. No Event Status or an invalid Event Status was received.', |
|
2020 | + 'event_espresso' |
|
2021 | + ); |
|
2022 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2023 | + return false; |
|
2024 | + } |
|
2025 | + // was event trashed or restored ? |
|
2026 | + switch ($event_status) { |
|
2027 | + case 'draft': |
|
2028 | + $action = 'restored from the trash'; |
|
2029 | + $hook = 'AHEE_event_restored_from_trash'; |
|
2030 | + break; |
|
2031 | + case 'trash': |
|
2032 | + $action = 'moved to the trash'; |
|
2033 | + $hook = 'AHEE_event_moved_to_trash'; |
|
2034 | + break; |
|
2035 | + default: |
|
2036 | + $action = 'updated'; |
|
2037 | + $hook = false; |
|
2038 | + } |
|
2039 | + // use class to change status |
|
2040 | + $this->_cpt_model_obj->set_status($event_status); |
|
2041 | + $success = $this->_cpt_model_obj->save(); |
|
2042 | + if ($success === false) { |
|
2043 | + $msg = sprintf(esc_html__('An error occurred. The event could not be %s.', 'event_espresso'), $action); |
|
2044 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2045 | + return false; |
|
2046 | + } |
|
2047 | + if ($hook) { |
|
2048 | + do_action($hook); |
|
2049 | + } |
|
2050 | + return true; |
|
2051 | + } |
|
2052 | + |
|
2053 | + |
|
2054 | + /** |
|
2055 | + * _delete_event |
|
2056 | + * |
|
2057 | + * @access protected |
|
2058 | + * @param bool $redirect_after |
|
2059 | + */ |
|
2060 | + protected function _delete_event($redirect_after = true) |
|
2061 | + { |
|
2062 | + // determine the event id and set to array. |
|
2063 | + $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : null; |
|
2064 | + $EVT_ID = isset($this->_req_data['post']) ? absint($this->_req_data['post']) : $EVT_ID; |
|
2065 | + // loop thru events |
|
2066 | + if ($EVT_ID) { |
|
2067 | + $success = $this->_permanently_delete_event($EVT_ID); |
|
2068 | + // get list of events with no prices |
|
2069 | + $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
2070 | + // remove this event from the list of events with no prices |
|
2071 | + if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
2072 | + unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2073 | + } |
|
2074 | + update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
2075 | + } else { |
|
2076 | + $success = false; |
|
2077 | + $msg = esc_html__( |
|
2078 | + 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
2079 | + 'event_espresso' |
|
2080 | + ); |
|
2081 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2082 | + } |
|
2083 | + if ($redirect_after) { |
|
2084 | + $this->_redirect_after_action( |
|
2085 | + $success, |
|
2086 | + 'Event', |
|
2087 | + 'deleted', |
|
2088 | + array('action' => 'default', 'status' => 'trash') |
|
2089 | + ); |
|
2090 | + } |
|
2091 | + } |
|
2092 | + |
|
2093 | + |
|
2094 | + /** |
|
2095 | + * _delete_events |
|
2096 | + * |
|
2097 | + * @access protected |
|
2098 | + * @return void |
|
2099 | + */ |
|
2100 | + protected function _delete_events() |
|
2101 | + { |
|
2102 | + $success = true; |
|
2103 | + // get list of events with no prices |
|
2104 | + $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
2105 | + // determine the event id and set to array. |
|
2106 | + $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
2107 | + // loop thru events |
|
2108 | + foreach ($EVT_IDs as $EVT_ID) { |
|
2109 | + $EVT_ID = absint($EVT_ID); |
|
2110 | + if ($EVT_ID) { |
|
2111 | + $results = $this->_permanently_delete_event($EVT_ID); |
|
2112 | + $success = $results !== false ? $success : false; |
|
2113 | + // remove this event from the list of events with no prices |
|
2114 | + unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2115 | + } else { |
|
2116 | + $success = false; |
|
2117 | + $msg = esc_html__( |
|
2118 | + 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
2119 | + 'event_espresso' |
|
2120 | + ); |
|
2121 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2122 | + } |
|
2123 | + } |
|
2124 | + update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
2125 | + // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
2126 | + $success = $success ? 2 : false; |
|
2127 | + $this->_redirect_after_action($success, 'Events', 'deleted', array('action' => 'default')); |
|
2128 | + } |
|
2129 | + |
|
2130 | + |
|
2131 | + /** |
|
2132 | + * _permanently_delete_event |
|
2133 | + * |
|
2134 | + * @access private |
|
2135 | + * @param int $EVT_ID |
|
2136 | + * @return bool |
|
2137 | + */ |
|
2138 | + private function _permanently_delete_event($EVT_ID = 0) |
|
2139 | + { |
|
2140 | + // grab event id |
|
2141 | + if (! $EVT_ID) { |
|
2142 | + $msg = esc_html__( |
|
2143 | + 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
2144 | + 'event_espresso' |
|
2145 | + ); |
|
2146 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2147 | + return false; |
|
2148 | + } |
|
2149 | + if (! $this->_cpt_model_obj instanceof EE_Event |
|
2150 | + || $this->_cpt_model_obj->ID() !== $EVT_ID |
|
2151 | + ) { |
|
2152 | + $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
2153 | + } |
|
2154 | + if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
2155 | + return false; |
|
2156 | + } |
|
2157 | + // need to delete related tickets and prices first. |
|
2158 | + $datetimes = $this->_cpt_model_obj->get_many_related('Datetime'); |
|
2159 | + foreach ($datetimes as $datetime) { |
|
2160 | + $this->_cpt_model_obj->_remove_relation_to($datetime, 'Datetime'); |
|
2161 | + $tickets = $datetime->get_many_related('Ticket'); |
|
2162 | + foreach ($tickets as $ticket) { |
|
2163 | + $ticket->_remove_relation_to($datetime, 'Datetime'); |
|
2164 | + $ticket->delete_related_permanently('Price'); |
|
2165 | + $ticket->delete_permanently(); |
|
2166 | + } |
|
2167 | + $datetime->delete(); |
|
2168 | + } |
|
2169 | + // what about related venues or terms? |
|
2170 | + $venues = $this->_cpt_model_obj->get_many_related('Venue'); |
|
2171 | + foreach ($venues as $venue) { |
|
2172 | + $this->_cpt_model_obj->_remove_relation_to($venue, 'Venue'); |
|
2173 | + } |
|
2174 | + // any attached question groups? |
|
2175 | + $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
|
2176 | + if (! empty($question_groups)) { |
|
2177 | + foreach ($question_groups as $question_group) { |
|
2178 | + $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
|
2179 | + } |
|
2180 | + } |
|
2181 | + // Message Template Groups |
|
2182 | + $this->_cpt_model_obj->_remove_relations('Message_Template_Group'); |
|
2183 | + /** @type EE_Term_Taxonomy[] $term_taxonomies */ |
|
2184 | + $term_taxonomies = $this->_cpt_model_obj->term_taxonomies(); |
|
2185 | + foreach ($term_taxonomies as $term_taxonomy) { |
|
2186 | + $this->_cpt_model_obj->remove_relation_to_term_taxonomy($term_taxonomy); |
|
2187 | + } |
|
2188 | + $success = $this->_cpt_model_obj->delete_permanently(); |
|
2189 | + // did it all go as planned ? |
|
2190 | + if ($success) { |
|
2191 | + $msg = sprintf(esc_html__('Event ID # %d has been deleted.', 'event_espresso'), $EVT_ID); |
|
2192 | + EE_Error::add_success($msg); |
|
2193 | + } else { |
|
2194 | + $msg = sprintf( |
|
2195 | + esc_html__('An error occurred. Event ID # %d could not be deleted.', 'event_espresso'), |
|
2196 | + $EVT_ID |
|
2197 | + ); |
|
2198 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2199 | + return false; |
|
2200 | + } |
|
2201 | + do_action('AHEE__Events_Admin_Page___permanently_delete_event__after_event_deleted', $EVT_ID); |
|
2202 | + return true; |
|
2203 | + } |
|
2204 | + |
|
2205 | + |
|
2206 | + /** |
|
2207 | + * get total number of events |
|
2208 | + * |
|
2209 | + * @access public |
|
2210 | + * @return int |
|
2211 | + */ |
|
2212 | + public function total_events() |
|
2213 | + { |
|
2214 | + $count = EEM_Event::instance()->count(array('caps' => 'read_admin'), 'EVT_ID', true); |
|
2215 | + return $count; |
|
2216 | + } |
|
2217 | + |
|
2218 | + |
|
2219 | + /** |
|
2220 | + * get total number of draft events |
|
2221 | + * |
|
2222 | + * @access public |
|
2223 | + * @return int |
|
2224 | + */ |
|
2225 | + public function total_events_draft() |
|
2226 | + { |
|
2227 | + $where = array( |
|
2228 | + 'status' => array('IN', array('draft', 'auto-draft')), |
|
2229 | + ); |
|
2230 | + $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
2231 | + return $count; |
|
2232 | + } |
|
2233 | + |
|
2234 | + |
|
2235 | + /** |
|
2236 | + * get total number of trashed events |
|
2237 | + * |
|
2238 | + * @access public |
|
2239 | + * @return int |
|
2240 | + */ |
|
2241 | + public function total_trashed_events() |
|
2242 | + { |
|
2243 | + $where = array( |
|
2244 | + 'status' => 'trash', |
|
2245 | + ); |
|
2246 | + $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
2247 | + return $count; |
|
2248 | + } |
|
2249 | + |
|
2250 | + |
|
2251 | + /** |
|
2252 | + * _default_event_settings |
|
2253 | + * This generates the Default Settings Tab |
|
2254 | + * |
|
2255 | + * @return void |
|
2256 | + * @throws EE_Error |
|
2257 | + */ |
|
2258 | + protected function _default_event_settings() |
|
2259 | + { |
|
2260 | + $this->_set_add_edit_form_tags('update_default_event_settings'); |
|
2261 | + $this->_set_publish_post_box_vars(null, false, false, null, false); |
|
2262 | + $this->_template_args['admin_page_content'] = $this->_default_event_settings_form()->get_html(); |
|
2263 | + $this->display_admin_page_with_sidebar(); |
|
2264 | + } |
|
2265 | + |
|
2266 | + |
|
2267 | + /** |
|
2268 | + * Return the form for event settings. |
|
2269 | + * |
|
2270 | + * @return EE_Form_Section_Proper |
|
2271 | + * @throws EE_Error |
|
2272 | + */ |
|
2273 | + protected function _default_event_settings_form() |
|
2274 | + { |
|
2275 | + $registration_config = EE_Registry::instance()->CFG->registration; |
|
2276 | + $registration_stati_for_selection = EEM_Registration::reg_status_array( |
|
2277 | + // exclude |
|
2278 | + array( |
|
2279 | + EEM_Registration::status_id_cancelled, |
|
2280 | + EEM_Registration::status_id_declined, |
|
2281 | + EEM_Registration::status_id_incomplete, |
|
2282 | + EEM_Registration::status_id_wait_list, |
|
2283 | + ), |
|
2284 | + true |
|
2285 | + ); |
|
2286 | + return new EE_Form_Section_Proper( |
|
2287 | + array( |
|
2288 | + 'name' => 'update_default_event_settings', |
|
2289 | + 'html_id' => 'update_default_event_settings', |
|
2290 | + 'html_class' => 'form-table', |
|
2291 | + 'layout_strategy' => new EE_Admin_Two_Column_Layout(), |
|
2292 | + 'subsections' => apply_filters( |
|
2293 | + 'FHEE__Events_Admin_Page___default_event_settings_form__form_subsections', |
|
2294 | + array( |
|
2295 | + 'default_reg_status' => new EE_Select_Input( |
|
2296 | + $registration_stati_for_selection, |
|
2297 | + array( |
|
2298 | + 'default' => isset($registration_config->default_STS_ID) |
|
2299 | + && array_key_exists( |
|
2300 | + $registration_config->default_STS_ID, |
|
2301 | + $registration_stati_for_selection |
|
2302 | + ) |
|
2303 | + ? sanitize_text_field($registration_config->default_STS_ID) |
|
2304 | + : EEM_Registration::status_id_pending_payment, |
|
2305 | + 'html_label_text' => esc_html__('Default Registration Status', 'event_espresso') |
|
2306 | + . EEH_Template::get_help_tab_link( |
|
2307 | + 'default_settings_status_help_tab' |
|
2308 | + ), |
|
2309 | + 'html_help_text' => esc_html__( |
|
2310 | + 'This setting allows you to preselect what the default registration status setting is when creating an event. Note that changing this setting does NOT retroactively apply it to existing events.', |
|
2311 | + 'event_espresso' |
|
2312 | + ), |
|
2313 | + ) |
|
2314 | + ), |
|
2315 | + 'default_max_tickets' => new EE_Integer_Input( |
|
2316 | + array( |
|
2317 | + 'default' => isset($registration_config->default_maximum_number_of_tickets) |
|
2318 | + ? $registration_config->default_maximum_number_of_tickets |
|
2319 | + : EEM_Event::get_default_additional_limit(), |
|
2320 | + 'html_label_text' => esc_html__( |
|
2321 | + 'Default Maximum Tickets Allowed Per Order:', |
|
2322 | + 'event_espresso' |
|
2323 | + ) |
|
2324 | + . EEH_Template::get_help_tab_link( |
|
2325 | + 'default_maximum_tickets_help_tab"' |
|
2326 | + ), |
|
2327 | + 'html_help_text' => esc_html__( |
|
2328 | + 'This setting allows you to indicate what will be the default for the maximum number of tickets per order when creating new events.', |
|
2329 | + 'event_espresso' |
|
2330 | + ), |
|
2331 | + ) |
|
2332 | + ), |
|
2333 | + ) |
|
2334 | + ), |
|
2335 | + ) |
|
2336 | + ); |
|
2337 | + } |
|
2338 | + |
|
2339 | + |
|
2340 | + /** |
|
2341 | + * _update_default_event_settings |
|
2342 | + * |
|
2343 | + * @access protected |
|
2344 | + * @return void |
|
2345 | + * @throws EE_Error |
|
2346 | + */ |
|
2347 | + protected function _update_default_event_settings() |
|
2348 | + { |
|
2349 | + $registration_config = EE_Registry::instance()->CFG->registration; |
|
2350 | + $form = $this->_default_event_settings_form(); |
|
2351 | + if ($form->was_submitted()) { |
|
2352 | + $form->receive_form_submission(); |
|
2353 | + if ($form->is_valid()) { |
|
2354 | + $valid_data = $form->valid_data(); |
|
2355 | + if (isset($valid_data['default_reg_status'])) { |
|
2356 | + $registration_config->default_STS_ID = $valid_data['default_reg_status']; |
|
2357 | + } |
|
2358 | + if (isset($valid_data['default_max_tickets'])) { |
|
2359 | + $registration_config->default_maximum_number_of_tickets = $valid_data['default_max_tickets']; |
|
2360 | + } |
|
2361 | + // update because data was valid! |
|
2362 | + EE_Registry::instance()->CFG->update_espresso_config(); |
|
2363 | + EE_Error::overwrite_success(); |
|
2364 | + EE_Error::add_success( |
|
2365 | + __('Default Event Settings were updated', 'event_espresso') |
|
2366 | + ); |
|
2367 | + } |
|
2368 | + } |
|
2369 | + $this->_redirect_after_action(0, '', '', array('action' => 'default_event_settings'), true); |
|
2370 | + } |
|
2371 | + |
|
2372 | + |
|
2373 | + /************* Templates *************/ |
|
2374 | + protected function _template_settings() |
|
2375 | + { |
|
2376 | + $this->_admin_page_title = esc_html__('Template Settings (Preview)', 'event_espresso'); |
|
2377 | + $this->_template_args['preview_img'] = '<img src="' |
|
2378 | + . EVENTS_ASSETS_URL |
|
2379 | + . DS |
|
2380 | + . 'images' |
|
2381 | + . DS |
|
2382 | + . 'caffeinated_template_features.jpg" alt="' |
|
2383 | + . esc_attr__('Template Settings Preview screenshot', 'event_espresso') |
|
2384 | + . '" />'; |
|
2385 | + $this->_template_args['preview_text'] = '<strong>' |
|
2386 | + . esc_html__( |
|
2387 | + 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
|
2388 | + 'event_espresso' |
|
2389 | + ) . '</strong>'; |
|
2390 | + $this->display_admin_caf_preview_page('template_settings_tab'); |
|
2391 | + } |
|
2392 | + |
|
2393 | + |
|
2394 | + /** Event Category Stuff **/ |
|
2395 | + /** |
|
2396 | + * set the _category property with the category object for the loaded page. |
|
2397 | + * |
|
2398 | + * @access private |
|
2399 | + * @return void |
|
2400 | + */ |
|
2401 | + private function _set_category_object() |
|
2402 | + { |
|
2403 | + if (isset($this->_category->id) && ! empty($this->_category->id)) { |
|
2404 | + return; |
|
2405 | + } //already have the category object so get out. |
|
2406 | + // set default category object |
|
2407 | + $this->_set_empty_category_object(); |
|
2408 | + // only set if we've got an id |
|
2409 | + if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
2410 | + return; |
|
2411 | + } |
|
2412 | + $category_id = absint($this->_req_data['EVT_CAT_ID']); |
|
2413 | + $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2414 | + if (! empty($term)) { |
|
2415 | + $this->_category->category_name = $term->name; |
|
2416 | + $this->_category->category_identifier = $term->slug; |
|
2417 | + $this->_category->category_desc = $term->description; |
|
2418 | + $this->_category->id = $term->term_id; |
|
2419 | + $this->_category->parent = $term->parent; |
|
2420 | + } |
|
2421 | + } |
|
2422 | + |
|
2423 | + |
|
2424 | + /** |
|
2425 | + * Clears out category properties. |
|
2426 | + */ |
|
2427 | + private function _set_empty_category_object() |
|
2428 | + { |
|
2429 | + $this->_category = new stdClass(); |
|
2430 | + $this->_category->category_name = $this->_category->category_identifier = $this->_category->category_desc = ''; |
|
2431 | + $this->_category->id = $this->_category->parent = 0; |
|
2432 | + } |
|
2433 | + |
|
2434 | + |
|
2435 | + /** |
|
2436 | + * @throws EE_Error |
|
2437 | + */ |
|
2438 | + protected function _category_list_table() |
|
2439 | + { |
|
2440 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
2441 | + $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
|
2442 | + $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
2443 | + 'add_category', |
|
2444 | + 'add_category', |
|
2445 | + array(), |
|
2446 | + 'add-new-h2' |
|
2447 | + ); |
|
2448 | + $this->display_admin_list_table_page_with_sidebar(); |
|
2449 | + } |
|
2450 | + |
|
2451 | + |
|
2452 | + /** |
|
2453 | + * Output category details view. |
|
2454 | + */ |
|
2455 | + protected function _category_details($view) |
|
2456 | + { |
|
2457 | + // load formatter helper |
|
2458 | + // load field generator helper |
|
2459 | + $route = $view == 'edit' ? 'update_category' : 'insert_category'; |
|
2460 | + $this->_set_add_edit_form_tags($route); |
|
2461 | + $this->_set_category_object(); |
|
2462 | + $id = ! empty($this->_category->id) ? $this->_category->id : ''; |
|
2463 | + $delete_action = 'delete_category'; |
|
2464 | + // custom redirect |
|
2465 | + $redirect = EE_Admin_Page::add_query_args_and_nonce( |
|
2466 | + array('action' => 'category_list'), |
|
2467 | + $this->_admin_base_url |
|
2468 | + ); |
|
2469 | + $this->_set_publish_post_box_vars('EVT_CAT_ID', $id, $delete_action, $redirect); |
|
2470 | + // take care of contents |
|
2471 | + $this->_template_args['admin_page_content'] = $this->_category_details_content(); |
|
2472 | + $this->display_admin_page_with_sidebar(); |
|
2473 | + } |
|
2474 | + |
|
2475 | + |
|
2476 | + /** |
|
2477 | + * Output category details content. |
|
2478 | + */ |
|
2479 | + protected function _category_details_content() |
|
2480 | + { |
|
2481 | + $editor_args['category_desc'] = array( |
|
2482 | + 'type' => 'wp_editor', |
|
2483 | + 'value' => EEH_Formatter::admin_format_content($this->_category->category_desc), |
|
2484 | + 'class' => 'my_editor_custom', |
|
2485 | + 'wpeditor_args' => array('media_buttons' => false), |
|
2486 | + ); |
|
2487 | + $_wp_editor = $this->_generate_admin_form_fields($editor_args, 'array'); |
|
2488 | + $all_terms = get_terms( |
|
2489 | + array(EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY), |
|
2490 | + array('hide_empty' => 0, 'exclude' => array($this->_category->id)) |
|
2491 | + ); |
|
2492 | + // setup category select for term parents. |
|
2493 | + $category_select_values[] = array( |
|
2494 | + 'text' => esc_html__('No Parent', 'event_espresso'), |
|
2495 | + 'id' => 0, |
|
2496 | + ); |
|
2497 | + foreach ($all_terms as $term) { |
|
2498 | + $category_select_values[] = array( |
|
2499 | + 'text' => $term->name, |
|
2500 | + 'id' => $term->term_id, |
|
2501 | + ); |
|
2502 | + } |
|
2503 | + $category_select = EEH_Form_Fields::select_input( |
|
2504 | + 'category_parent', |
|
2505 | + $category_select_values, |
|
2506 | + $this->_category->parent |
|
2507 | + ); |
|
2508 | + $template_args = array( |
|
2509 | + 'category' => $this->_category, |
|
2510 | + 'category_select' => $category_select, |
|
2511 | + 'unique_id_info_help_link' => $this->_get_help_tab_link('unique_id_info'), |
|
2512 | + 'category_desc_editor' => $_wp_editor['category_desc']['field'], |
|
2513 | + 'disable' => '', |
|
2514 | + 'disabled_message' => false, |
|
2515 | + ); |
|
2516 | + $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
2517 | + return EEH_Template::display_template($template, $template_args, true); |
|
2518 | + } |
|
2519 | + |
|
2520 | + |
|
2521 | + /** |
|
2522 | + * Handles deleting categories. |
|
2523 | + */ |
|
2524 | + protected function _delete_categories() |
|
2525 | + { |
|
2526 | + $cat_ids = isset($this->_req_data['EVT_CAT_ID']) ? (array) $this->_req_data['EVT_CAT_ID'] |
|
2527 | + : (array) $this->_req_data['category_id']; |
|
2528 | + foreach ($cat_ids as $cat_id) { |
|
2529 | + $this->_delete_category($cat_id); |
|
2530 | + } |
|
2531 | + // doesn't matter what page we're coming from... we're going to the same place after delete. |
|
2532 | + $query_args = array( |
|
2533 | + 'action' => 'category_list', |
|
2534 | + ); |
|
2535 | + $this->_redirect_after_action(0, '', '', $query_args); |
|
2536 | + } |
|
2537 | + |
|
2538 | + |
|
2539 | + /** |
|
2540 | + * Handles deleting specific category. |
|
2541 | + * |
|
2542 | + * @param int $cat_id |
|
2543 | + */ |
|
2544 | + protected function _delete_category($cat_id) |
|
2545 | + { |
|
2546 | + $cat_id = absint($cat_id); |
|
2547 | + wp_delete_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2548 | + } |
|
2549 | + |
|
2550 | + |
|
2551 | + /** |
|
2552 | + * Handles triggering the update or insertion of a new category. |
|
2553 | + * |
|
2554 | + * @param bool $new_category true means we're triggering the insert of a new category. |
|
2555 | + */ |
|
2556 | + protected function _insert_or_update_category($new_category) |
|
2557 | + { |
|
2558 | + $cat_id = $new_category ? $this->_insert_category() : $this->_insert_category(true); |
|
2559 | + $success = 0; // we already have a success message so lets not send another. |
|
2560 | + if ($cat_id) { |
|
2561 | + $query_args = array( |
|
2562 | + 'action' => 'edit_category', |
|
2563 | + 'EVT_CAT_ID' => $cat_id, |
|
2564 | + ); |
|
2565 | + } else { |
|
2566 | + $query_args = array('action' => 'add_category'); |
|
2567 | + } |
|
2568 | + $this->_redirect_after_action($success, '', '', $query_args, true); |
|
2569 | + } |
|
2570 | + |
|
2571 | + |
|
2572 | + /** |
|
2573 | + * Inserts or updates category |
|
2574 | + * |
|
2575 | + * @param bool $update (true indicates we're updating a category). |
|
2576 | + * @return bool|mixed|string |
|
2577 | + */ |
|
2578 | + private function _insert_category($update = false) |
|
2579 | + { |
|
2580 | + $cat_id = $update ? $this->_req_data['EVT_CAT_ID'] : ''; |
|
2581 | + $category_name = isset($this->_req_data['category_name']) ? $this->_req_data['category_name'] : ''; |
|
2582 | + $category_desc = isset($this->_req_data['category_desc']) ? $this->_req_data['category_desc'] : ''; |
|
2583 | + $category_parent = isset($this->_req_data['category_parent']) ? $this->_req_data['category_parent'] : 0; |
|
2584 | + if (empty($category_name)) { |
|
2585 | + $msg = esc_html__('You must add a name for the category.', 'event_espresso'); |
|
2586 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2587 | + return false; |
|
2588 | + } |
|
2589 | + $term_args = array( |
|
2590 | + 'name' => $category_name, |
|
2591 | + 'description' => $category_desc, |
|
2592 | + 'parent' => $category_parent, |
|
2593 | + ); |
|
2594 | + // was the category_identifier input disabled? |
|
2595 | + if (isset($this->_req_data['category_identifier'])) { |
|
2596 | + $term_args['slug'] = $this->_req_data['category_identifier']; |
|
2597 | + } |
|
2598 | + $insert_ids = $update |
|
2599 | + ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
|
2600 | + : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
|
2601 | + if (! is_array($insert_ids)) { |
|
2602 | + $msg = esc_html__( |
|
2603 | + 'An error occurred and the category has not been saved to the database.', |
|
2604 | + 'event_espresso' |
|
2605 | + ); |
|
2606 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
2607 | + } else { |
|
2608 | + $cat_id = $insert_ids['term_id']; |
|
2609 | + $msg = sprintf(esc_html__('The category %s was successfully saved', 'event_espresso'), $category_name); |
|
2610 | + EE_Error::add_success($msg); |
|
2611 | + } |
|
2612 | + return $cat_id; |
|
2613 | + } |
|
2614 | + |
|
2615 | + |
|
2616 | + /** |
|
2617 | + * Gets categories or count of categories matching the arguments in the request. |
|
2618 | + * |
|
2619 | + * @param int $per_page |
|
2620 | + * @param int $current_page |
|
2621 | + * @param bool $count |
|
2622 | + * @return EE_Base_Class[]|EE_Term_Taxonomy[]|int |
|
2623 | + */ |
|
2624 | + public function get_categories($per_page = 10, $current_page = 1, $count = false) |
|
2625 | + { |
|
2626 | + // testing term stuff |
|
2627 | + $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'Term.term_id'; |
|
2628 | + $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : 'DESC'; |
|
2629 | + $limit = ($current_page - 1) * $per_page; |
|
2630 | + $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
2631 | + if (isset($this->_req_data['s'])) { |
|
2632 | + $sstr = '%' . $this->_req_data['s'] . '%'; |
|
2633 | + $where['OR'] = array( |
|
2634 | + 'Term.name' => array('LIKE', $sstr), |
|
2635 | + 'description' => array('LIKE', $sstr), |
|
2636 | + ); |
|
2637 | + } |
|
2638 | + $query_params = array( |
|
2639 | + $where, |
|
2640 | + 'order_by' => array($orderby => $order), |
|
2641 | + 'limit' => $limit . ',' . $per_page, |
|
2642 | + 'force_join' => array('Term'), |
|
2643 | + ); |
|
2644 | + $categories = $count |
|
2645 | + ? EEM_Term_Taxonomy::instance()->count($query_params, 'term_id') |
|
2646 | + : EEM_Term_Taxonomy::instance()->get_all($query_params); |
|
2647 | + return $categories; |
|
2648 | + } |
|
2649 | + |
|
2650 | + /* end category stuff */ |
|
2651 | + /**************/ |
|
2652 | + |
|
2653 | + |
|
2654 | + /** |
|
2655 | + * Callback for the `ee_save_timezone_setting` ajax action. |
|
2656 | + * |
|
2657 | + * @throws EE_Error |
|
2658 | + */ |
|
2659 | + public function save_timezonestring_setting() |
|
2660 | + { |
|
2661 | + $timezone_string = isset($this->_req_data['timezone_selected']) |
|
2662 | + ? $this->_req_data['timezone_selected'] |
|
2663 | + : ''; |
|
2664 | + if (empty($timezone_string) || ! EEH_DTT_Helper::validate_timezone($timezone_string, false)) { |
|
2665 | + EE_Error::add_error( |
|
2666 | + esc_html__('An invalid timezone string submitted.', 'event_espresso'), |
|
2667 | + __FILE__, |
|
2668 | + __FUNCTION__, |
|
2669 | + __LINE__ |
|
2670 | + ); |
|
2671 | + $this->_template_args['error'] = true; |
|
2672 | + $this->_return_json(); |
|
2673 | + } |
|
2674 | + |
|
2675 | + update_option('timezone_string', $timezone_string); |
|
2676 | + EE_Error::add_success( |
|
2677 | + esc_html__('Your timezone string was updated.', 'event_espresso') |
|
2678 | + ); |
|
2679 | + $this->_template_args['success'] = true; |
|
2680 | + $this->_return_json(true, array('action' => 'create_new')); |
|
2681 | + } |
|
2682 | 2682 | } |
@@ -548,11 +548,11 @@ discard block |
||
548 | 548 | { |
549 | 549 | wp_register_style( |
550 | 550 | 'events-admin-css', |
551 | - EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
551 | + EVENTS_ASSETS_URL.'events-admin-page.css', |
|
552 | 552 | array(), |
553 | 553 | EVENT_ESPRESSO_VERSION |
554 | 554 | ); |
555 | - wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
555 | + wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL.'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
556 | 556 | wp_enqueue_style('events-admin-css'); |
557 | 557 | wp_enqueue_style('ee-cat-admin'); |
558 | 558 | // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
@@ -560,7 +560,7 @@ discard block |
||
560 | 560 | // scripts |
561 | 561 | wp_register_script( |
562 | 562 | 'event_editor_js', |
563 | - EVENTS_ASSETS_URL . 'event_editor.js', |
|
563 | + EVENTS_ASSETS_URL.'event_editor.js', |
|
564 | 564 | array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
565 | 565 | EVENT_ESPRESSO_VERSION, |
566 | 566 | true |
@@ -586,7 +586,7 @@ discard block |
||
586 | 586 | wp_enqueue_style('espresso-ui-theme'); |
587 | 587 | wp_register_style( |
588 | 588 | 'event-editor-css', |
589 | - EVENTS_ASSETS_URL . 'event-editor.css', |
|
589 | + EVENTS_ASSETS_URL.'event-editor.css', |
|
590 | 590 | array('ee-admin-css'), |
591 | 591 | EVENT_ESPRESSO_VERSION |
592 | 592 | ); |
@@ -594,7 +594,7 @@ discard block |
||
594 | 594 | // scripts |
595 | 595 | wp_register_script( |
596 | 596 | 'event-datetime-metabox', |
597 | - EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
597 | + EVENTS_ASSETS_URL.'event-datetime-metabox.js', |
|
598 | 598 | array('event_editor_js', 'ee-datepicker'), |
599 | 599 | EVENT_ESPRESSO_VERSION |
600 | 600 | ); |
@@ -663,7 +663,7 @@ discard block |
||
663 | 663 | public function verify_event_edit($event = null, $req_type = '') |
664 | 664 | { |
665 | 665 | // don't need to do this when processing |
666 | - if (! empty($req_type)) { |
|
666 | + if ( ! empty($req_type)) { |
|
667 | 667 | return; |
668 | 668 | } |
669 | 669 | // no event? |
@@ -672,7 +672,7 @@ discard block |
||
672 | 672 | $event = $this->_cpt_model_obj; |
673 | 673 | } |
674 | 674 | // STILL no event? |
675 | - if (! $event instanceof EE_Event) { |
|
675 | + if ( ! $event instanceof EE_Event) { |
|
676 | 676 | return; |
677 | 677 | } |
678 | 678 | $orig_status = $event->status(); |
@@ -710,7 +710,7 @@ discard block |
||
710 | 710 | ); |
711 | 711 | } |
712 | 712 | // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
713 | - if (! $event->tickets_on_sale()) { |
|
713 | + if ( ! $event->tickets_on_sale()) { |
|
714 | 714 | return; |
715 | 715 | } |
716 | 716 | // made it here so show warning |
@@ -755,7 +755,7 @@ discard block |
||
755 | 755 | { |
756 | 756 | $has_timezone_string = get_option('timezone_string'); |
757 | 757 | // only nag them about setting their timezone if it's their first event, and they haven't already done it |
758 | - if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
758 | + if ( ! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
759 | 759 | EE_Error::add_attention( |
760 | 760 | sprintf( |
761 | 761 | __( |
@@ -839,31 +839,31 @@ discard block |
||
839 | 839 | $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
840 | 840 | $statuses = array( |
841 | 841 | 'sold_out_status' => array( |
842 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
842 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::sold_out, |
|
843 | 843 | 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
844 | 844 | ), |
845 | 845 | 'active_status' => array( |
846 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
846 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::active, |
|
847 | 847 | 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
848 | 848 | ), |
849 | 849 | 'upcoming_status' => array( |
850 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
850 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::upcoming, |
|
851 | 851 | 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
852 | 852 | ), |
853 | 853 | 'postponed_status' => array( |
854 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
854 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::postponed, |
|
855 | 855 | 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
856 | 856 | ), |
857 | 857 | 'cancelled_status' => array( |
858 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
858 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::cancelled, |
|
859 | 859 | 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
860 | 860 | ), |
861 | 861 | 'expired_status' => array( |
862 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
862 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::expired, |
|
863 | 863 | 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
864 | 864 | ), |
865 | 865 | 'inactive_status' => array( |
866 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
866 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::inactive, |
|
867 | 867 | 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
868 | 868 | ), |
869 | 869 | ); |
@@ -877,7 +877,7 @@ discard block |
||
877 | 877 | */ |
878 | 878 | private function _event_model() |
879 | 879 | { |
880 | - if (! $this->_event_model instanceof EEM_Event) { |
|
880 | + if ( ! $this->_event_model instanceof EEM_Event) { |
|
881 | 881 | $this->_event_model = EE_Registry::instance()->load_model('Event'); |
882 | 882 | } |
883 | 883 | return $this->_event_model; |
@@ -897,7 +897,7 @@ discard block |
||
897 | 897 | public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
898 | 898 | { |
899 | 899 | // make sure this is only when editing |
900 | - if (! empty($id)) { |
|
900 | + if ( ! empty($id)) { |
|
901 | 901 | $post = get_post($id); |
902 | 902 | $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
903 | 903 | . esc_html__('Shortcode', 'event_espresso') |
@@ -931,7 +931,7 @@ discard block |
||
931 | 931 | 'button' |
932 | 932 | ); |
933 | 933 | $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
934 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
934 | + $this->_admin_page_title .= ' '.$this->get_action_link_or_button( |
|
935 | 935 | 'create_new', |
936 | 936 | 'add', |
937 | 937 | array(), |
@@ -1071,7 +1071,7 @@ discard block |
||
1071 | 1071 | */ |
1072 | 1072 | protected function _default_venue_update(\EE_Event $evtobj, $data) |
1073 | 1073 | { |
1074 | - require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
1074 | + require_once(EE_MODELS.'EEM_Venue.model.php'); |
|
1075 | 1075 | $venue_model = EE_Registry::instance()->load_model('Venue'); |
1076 | 1076 | $rows_affected = null; |
1077 | 1077 | $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
@@ -1102,7 +1102,7 @@ discard block |
||
1102 | 1102 | 'status' => 'publish', |
1103 | 1103 | ); |
1104 | 1104 | // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
1105 | - if (! empty($venue_id)) { |
|
1105 | + if ( ! empty($venue_id)) { |
|
1106 | 1106 | $update_where = array($venue_model->primary_key_name() => $venue_id); |
1107 | 1107 | $rows_affected = $venue_model->update($venue_array, array($update_where)); |
1108 | 1108 | // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
@@ -1144,7 +1144,7 @@ discard block |
||
1144 | 1144 | 'DTT_order' => $row, |
1145 | 1145 | ); |
1146 | 1146 | // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
1147 | - if (! empty($dtt['DTT_ID'])) { |
|
1147 | + if ( ! empty($dtt['DTT_ID'])) { |
|
1148 | 1148 | $DTM = EE_Registry::instance() |
1149 | 1149 | ->load_model('Datetime', array($evtobj->get_timezone())) |
1150 | 1150 | ->get_one_by_ID($dtt['DTT_ID']); |
@@ -1154,7 +1154,7 @@ discard block |
||
1154 | 1154 | $DTM->set($field, $value); |
1155 | 1155 | } |
1156 | 1156 | // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
1157 | - $saved_dtts[ $DTM->ID() ] = $DTM; |
|
1157 | + $saved_dtts[$DTM->ID()] = $DTM; |
|
1158 | 1158 | } else { |
1159 | 1159 | $DTM = EE_Registry::instance()->load_class( |
1160 | 1160 | 'Datetime', |
@@ -1187,14 +1187,14 @@ discard block |
||
1187 | 1187 | foreach ($data['edit_tickets'] as $row => $tkt) { |
1188 | 1188 | $incoming_date_formats = array('Y-m-d', 'h:i a'); |
1189 | 1189 | $update_prices = false; |
1190 | - $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
1191 | - ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
1190 | + $ticket_price = isset($data['edit_prices'][$row][1]['PRC_amount']) |
|
1191 | + ? $data['edit_prices'][$row][1]['PRC_amount'] : 0; |
|
1192 | 1192 | // trim inputs to ensure any excess whitespace is removed. |
1193 | 1193 | $tkt = array_map('trim', $tkt); |
1194 | 1194 | if (empty($tkt['TKT_start_date'])) { |
1195 | 1195 | // let's use now in the set timezone. |
1196 | 1196 | $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
1197 | - $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
1197 | + $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0].' '.$incoming_date_formats[1]); |
|
1198 | 1198 | } |
1199 | 1199 | if (empty($tkt['TKT_end_date'])) { |
1200 | 1200 | // use the start date of the first datetime |
@@ -1229,7 +1229,7 @@ discard block |
||
1229 | 1229 | // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
1230 | 1230 | // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
1231 | 1231 | // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
1232 | - if (! empty($tkt['TKT_ID'])) { |
|
1232 | + if ( ! empty($tkt['TKT_ID'])) { |
|
1233 | 1233 | $TKT = EE_Registry::instance() |
1234 | 1234 | ->load_model('Ticket', array($evtobj->get_timezone())) |
1235 | 1235 | ->get_one_by_ID($tkt['TKT_ID']); |
@@ -1264,7 +1264,7 @@ discard block |
||
1264 | 1264 | $TKT->set('TKT_deleted', 1); |
1265 | 1265 | $TKT->save(); |
1266 | 1266 | // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
1267 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1267 | + $saved_tickets[$TKT->ID()] = $TKT; |
|
1268 | 1268 | // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
1269 | 1269 | $TKT = clone $TKT; |
1270 | 1270 | $TKT->set('TKT_ID', 0); |
@@ -1309,9 +1309,9 @@ discard block |
||
1309 | 1309 | } |
1310 | 1310 | // initially let's add the ticket to the dtt |
1311 | 1311 | $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
1312 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
1312 | + $saved_tickets[$TKT->ID()] = $TKT; |
|
1313 | 1313 | // add prices to ticket |
1314 | - $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
1314 | + $this->_add_prices_to_ticket($data['edit_prices'][$row], $TKT, $update_prices); |
|
1315 | 1315 | } |
1316 | 1316 | // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
1317 | 1317 | $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
@@ -1468,7 +1468,7 @@ discard block |
||
1468 | 1468 | $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
1469 | 1469 | // load template |
1470 | 1470 | EEH_Template::display_template( |
1471 | - EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
1471 | + EVENTS_TEMPLATE_PATH.'event_publish_box_extras.template.php', |
|
1472 | 1472 | $publish_box_extra_args |
1473 | 1473 | ); |
1474 | 1474 | } |
@@ -1560,7 +1560,7 @@ discard block |
||
1560 | 1560 | 'default_where_conditions' => 'none', |
1561 | 1561 | ) |
1562 | 1562 | ); |
1563 | - if (! empty($related_tickets)) { |
|
1563 | + if ( ! empty($related_tickets)) { |
|
1564 | 1564 | $template_args['total_ticket_rows'] = count($related_tickets); |
1565 | 1565 | $row = 0; |
1566 | 1566 | foreach ($related_tickets as $ticket) { |
@@ -1594,7 +1594,7 @@ discard block |
||
1594 | 1594 | ); |
1595 | 1595 | $template = apply_filters( |
1596 | 1596 | 'FHEE__Events_Admin_Page__ticket_metabox__template', |
1597 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
1597 | + EVENTS_TEMPLATE_PATH.'event_tickets_metabox_main.template.php' |
|
1598 | 1598 | ); |
1599 | 1599 | EEH_Template::display_template($template, $template_args); |
1600 | 1600 | } |
@@ -1612,7 +1612,7 @@ discard block |
||
1612 | 1612 | private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
1613 | 1613 | { |
1614 | 1614 | $template_args = array( |
1615 | - 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
1615 | + 'tkt_status_class' => ' tkt-status-'.$ticket->ticket_status(), |
|
1616 | 1616 | 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
1617 | 1617 | : '', |
1618 | 1618 | 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
@@ -1624,10 +1624,10 @@ discard block |
||
1624 | 1624 | 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
1625 | 1625 | 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
1626 | 1626 | 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
1627 | - 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
1628 | - && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
1627 | + 'trash_icon' => ($skeleton || ( ! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
1628 | + && ( ! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
1629 | 1629 | ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
1630 | - 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
1630 | + 'disabled' => $skeleton || ( ! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
1631 | 1631 | : ' disabled=disabled', |
1632 | 1632 | ); |
1633 | 1633 | $price = $ticket->ID() !== 0 |
@@ -1654,7 +1654,7 @@ discard block |
||
1654 | 1654 | array('order_by' => array('DTT_EVT_start' => 'ASC')) |
1655 | 1655 | ) |
1656 | 1656 | : null; |
1657 | - if (! empty($earliest_dtt)) { |
|
1657 | + if ( ! empty($earliest_dtt)) { |
|
1658 | 1658 | $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
1659 | 1659 | } else { |
1660 | 1660 | $template_args['TKT_end_date'] = date( |
@@ -1666,7 +1666,7 @@ discard block |
||
1666 | 1666 | $template_args = array_merge($template_args, $price_args); |
1667 | 1667 | $template = apply_filters( |
1668 | 1668 | 'FHEE__Events_Admin_Page__get_ticket_row__template', |
1669 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
1669 | + EVENTS_TEMPLATE_PATH.'event_tickets_metabox_ticket_row.template.php', |
|
1670 | 1670 | $ticket |
1671 | 1671 | ); |
1672 | 1672 | return EEH_Template::display_template($template, $template_args, true); |
@@ -1720,7 +1720,7 @@ discard block |
||
1720 | 1720 | $default_reg_status_values |
1721 | 1721 | ); |
1722 | 1722 | EEH_Template::display_template( |
1723 | - EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
1723 | + EVENTS_TEMPLATE_PATH.'event_registration_options.template.php', |
|
1724 | 1724 | $template_args |
1725 | 1725 | ); |
1726 | 1726 | } |
@@ -1742,7 +1742,7 @@ discard block |
||
1742 | 1742 | { |
1743 | 1743 | $EEME = $this->_event_model(); |
1744 | 1744 | $offset = ($current_page - 1) * $per_page; |
1745 | - $limit = $count ? null : $offset . ',' . $per_page; |
|
1745 | + $limit = $count ? null : $offset.','.$per_page; |
|
1746 | 1746 | $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
1747 | 1747 | $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
1748 | 1748 | if (isset($this->_req_data['month_range'])) { |
@@ -1771,7 +1771,7 @@ discard block |
||
1771 | 1771 | // categories? |
1772 | 1772 | $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
1773 | 1773 | ? $this->_req_data['EVT_CAT'] : null; |
1774 | - if (! empty($category)) { |
|
1774 | + if ( ! empty($category)) { |
|
1775 | 1775 | $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
1776 | 1776 | $where['Term_Taxonomy.term_id'] = $category; |
1777 | 1777 | } |
@@ -1779,7 +1779,7 @@ discard block |
||
1779 | 1779 | $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
1780 | 1780 | if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
1781 | 1781 | $DateTime = new DateTime( |
1782 | - $year_r . '-' . $month_r . '-01 00:00:00', |
|
1782 | + $year_r.'-'.$month_r.'-01 00:00:00', |
|
1783 | 1783 | new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
1784 | 1784 | ); |
1785 | 1785 | $start = $DateTime->format(implode(' ', $start_formats)); |
@@ -1805,11 +1805,11 @@ discard block |
||
1805 | 1805 | ->format(implode(' ', $start_formats)); |
1806 | 1806 | $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
1807 | 1807 | } |
1808 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
1808 | + if ( ! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
1809 | 1809 | $where['EVT_wp_user'] = get_current_user_id(); |
1810 | 1810 | } else { |
1811 | - if (! isset($where['status'])) { |
|
1812 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
1811 | + if ( ! isset($where['status'])) { |
|
1812 | + if ( ! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
1813 | 1813 | $where['OR'] = array( |
1814 | 1814 | 'status*restrict_private' => array('!=', 'private'), |
1815 | 1815 | 'AND' => array( |
@@ -1829,7 +1829,7 @@ discard block |
||
1829 | 1829 | } |
1830 | 1830 | // search query handling |
1831 | 1831 | if (isset($this->_req_data['s'])) { |
1832 | - $search_string = '%' . $this->_req_data['s'] . '%'; |
|
1832 | + $search_string = '%'.$this->_req_data['s'].'%'; |
|
1833 | 1833 | $where['OR'] = array( |
1834 | 1834 | 'EVT_name' => array('LIKE', $search_string), |
1835 | 1835 | 'EVT_desc' => array('LIKE', $search_string), |
@@ -1918,7 +1918,7 @@ discard block |
||
1918 | 1918 | // clean status |
1919 | 1919 | $event_status = sanitize_key($event_status); |
1920 | 1920 | // grab status |
1921 | - if (! empty($event_status)) { |
|
1921 | + if ( ! empty($event_status)) { |
|
1922 | 1922 | $success = $this->_change_event_status($EVT_ID, $event_status); |
1923 | 1923 | } else { |
1924 | 1924 | $success = false; |
@@ -1955,7 +1955,7 @@ discard block |
||
1955 | 1955 | // clean status |
1956 | 1956 | $event_status = sanitize_key($event_status); |
1957 | 1957 | // grab status |
1958 | - if (! empty($event_status)) { |
|
1958 | + if ( ! empty($event_status)) { |
|
1959 | 1959 | $success = true; |
1960 | 1960 | // determine the event id and set to array. |
1961 | 1961 | $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
@@ -2002,7 +2002,7 @@ discard block |
||
2002 | 2002 | private function _change_event_status($EVT_ID = 0, $event_status = '') |
2003 | 2003 | { |
2004 | 2004 | // grab event id |
2005 | - if (! $EVT_ID) { |
|
2005 | + if ( ! $EVT_ID) { |
|
2006 | 2006 | $msg = esc_html__( |
2007 | 2007 | 'An error occurred. No Event ID or an invalid Event ID was received.', |
2008 | 2008 | 'event_espresso' |
@@ -2068,8 +2068,8 @@ discard block |
||
2068 | 2068 | // get list of events with no prices |
2069 | 2069 | $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
2070 | 2070 | // remove this event from the list of events with no prices |
2071 | - if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
2072 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2071 | + if (isset($espresso_no_ticket_prices[$EVT_ID])) { |
|
2072 | + unset($espresso_no_ticket_prices[$EVT_ID]); |
|
2073 | 2073 | } |
2074 | 2074 | update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
2075 | 2075 | } else { |
@@ -2111,7 +2111,7 @@ discard block |
||
2111 | 2111 | $results = $this->_permanently_delete_event($EVT_ID); |
2112 | 2112 | $success = $results !== false ? $success : false; |
2113 | 2113 | // remove this event from the list of events with no prices |
2114 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
2114 | + unset($espresso_no_ticket_prices[$EVT_ID]); |
|
2115 | 2115 | } else { |
2116 | 2116 | $success = false; |
2117 | 2117 | $msg = esc_html__( |
@@ -2138,7 +2138,7 @@ discard block |
||
2138 | 2138 | private function _permanently_delete_event($EVT_ID = 0) |
2139 | 2139 | { |
2140 | 2140 | // grab event id |
2141 | - if (! $EVT_ID) { |
|
2141 | + if ( ! $EVT_ID) { |
|
2142 | 2142 | $msg = esc_html__( |
2143 | 2143 | 'An error occurred. No Event ID or an invalid Event ID was received.', |
2144 | 2144 | 'event_espresso' |
@@ -2146,12 +2146,12 @@ discard block |
||
2146 | 2146 | EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
2147 | 2147 | return false; |
2148 | 2148 | } |
2149 | - if (! $this->_cpt_model_obj instanceof EE_Event |
|
2149 | + if ( ! $this->_cpt_model_obj instanceof EE_Event |
|
2150 | 2150 | || $this->_cpt_model_obj->ID() !== $EVT_ID |
2151 | 2151 | ) { |
2152 | 2152 | $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
2153 | 2153 | } |
2154 | - if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
2154 | + if ( ! $this->_cpt_model_obj instanceof EE_Event) { |
|
2155 | 2155 | return false; |
2156 | 2156 | } |
2157 | 2157 | // need to delete related tickets and prices first. |
@@ -2173,7 +2173,7 @@ discard block |
||
2173 | 2173 | } |
2174 | 2174 | // any attached question groups? |
2175 | 2175 | $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
2176 | - if (! empty($question_groups)) { |
|
2176 | + if ( ! empty($question_groups)) { |
|
2177 | 2177 | foreach ($question_groups as $question_group) { |
2178 | 2178 | $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
2179 | 2179 | } |
@@ -2386,7 +2386,7 @@ discard block |
||
2386 | 2386 | . esc_html__( |
2387 | 2387 | 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
2388 | 2388 | 'event_espresso' |
2389 | - ) . '</strong>'; |
|
2389 | + ).'</strong>'; |
|
2390 | 2390 | $this->display_admin_caf_preview_page('template_settings_tab'); |
2391 | 2391 | } |
2392 | 2392 | |
@@ -2406,12 +2406,12 @@ discard block |
||
2406 | 2406 | // set default category object |
2407 | 2407 | $this->_set_empty_category_object(); |
2408 | 2408 | // only set if we've got an id |
2409 | - if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
2409 | + if ( ! isset($this->_req_data['EVT_CAT_ID'])) { |
|
2410 | 2410 | return; |
2411 | 2411 | } |
2412 | 2412 | $category_id = absint($this->_req_data['EVT_CAT_ID']); |
2413 | 2413 | $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
2414 | - if (! empty($term)) { |
|
2414 | + if ( ! empty($term)) { |
|
2415 | 2415 | $this->_category->category_name = $term->name; |
2416 | 2416 | $this->_category->category_identifier = $term->slug; |
2417 | 2417 | $this->_category->category_desc = $term->description; |
@@ -2439,7 +2439,7 @@ discard block |
||
2439 | 2439 | { |
2440 | 2440 | do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
2441 | 2441 | $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
2442 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
2442 | + $this->_admin_page_title .= ' '.$this->get_action_link_or_button( |
|
2443 | 2443 | 'add_category', |
2444 | 2444 | 'add_category', |
2445 | 2445 | array(), |
@@ -2513,7 +2513,7 @@ discard block |
||
2513 | 2513 | 'disable' => '', |
2514 | 2514 | 'disabled_message' => false, |
2515 | 2515 | ); |
2516 | - $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
2516 | + $template = EVENTS_TEMPLATE_PATH.'event_category_details.template.php'; |
|
2517 | 2517 | return EEH_Template::display_template($template, $template_args, true); |
2518 | 2518 | } |
2519 | 2519 | |
@@ -2598,7 +2598,7 @@ discard block |
||
2598 | 2598 | $insert_ids = $update |
2599 | 2599 | ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
2600 | 2600 | : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
2601 | - if (! is_array($insert_ids)) { |
|
2601 | + if ( ! is_array($insert_ids)) { |
|
2602 | 2602 | $msg = esc_html__( |
2603 | 2603 | 'An error occurred and the category has not been saved to the database.', |
2604 | 2604 | 'event_espresso' |
@@ -2629,7 +2629,7 @@ discard block |
||
2629 | 2629 | $limit = ($current_page - 1) * $per_page; |
2630 | 2630 | $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
2631 | 2631 | if (isset($this->_req_data['s'])) { |
2632 | - $sstr = '%' . $this->_req_data['s'] . '%'; |
|
2632 | + $sstr = '%'.$this->_req_data['s'].'%'; |
|
2633 | 2633 | $where['OR'] = array( |
2634 | 2634 | 'Term.name' => array('LIKE', $sstr), |
2635 | 2635 | 'description' => array('LIKE', $sstr), |
@@ -2638,7 +2638,7 @@ discard block |
||
2638 | 2638 | $query_params = array( |
2639 | 2639 | $where, |
2640 | 2640 | 'order_by' => array($orderby => $order), |
2641 | - 'limit' => $limit . ',' . $per_page, |
|
2641 | + 'limit' => $limit.','.$per_page, |
|
2642 | 2642 | 'force_join' => array('Term'), |
2643 | 2643 | ); |
2644 | 2644 | $categories = $count |
@@ -23,1679 +23,1679 @@ |
||
23 | 23 | class EE_Registry implements ResettableInterface |
24 | 24 | { |
25 | 25 | |
26 | - /** |
|
27 | - * @var EE_Registry $_instance |
|
28 | - */ |
|
29 | - private static $_instance; |
|
30 | - |
|
31 | - /** |
|
32 | - * @var EE_Dependency_Map $_dependency_map |
|
33 | - */ |
|
34 | - protected $_dependency_map; |
|
35 | - |
|
36 | - /** |
|
37 | - * @var Mirror |
|
38 | - */ |
|
39 | - private $mirror; |
|
40 | - |
|
41 | - /** |
|
42 | - * @var ClassInterfaceCache $class_cache |
|
43 | - */ |
|
44 | - private $class_cache; |
|
45 | - |
|
46 | - /** |
|
47 | - * @var array $_class_abbreviations |
|
48 | - */ |
|
49 | - protected $_class_abbreviations = array(); |
|
50 | - |
|
51 | - /** |
|
52 | - * @var CommandBusInterface $BUS |
|
53 | - */ |
|
54 | - public $BUS; |
|
55 | - |
|
56 | - /** |
|
57 | - * @var EE_Cart $CART |
|
58 | - */ |
|
59 | - public $CART; |
|
60 | - |
|
61 | - /** |
|
62 | - * @var EE_Config $CFG |
|
63 | - */ |
|
64 | - public $CFG; |
|
65 | - |
|
66 | - /** |
|
67 | - * @var EE_Network_Config $NET_CFG |
|
68 | - */ |
|
69 | - public $NET_CFG; |
|
70 | - |
|
71 | - /** |
|
72 | - * StdClass object for storing library classes in |
|
73 | - * |
|
74 | - * @var RegistryContainer $LIB |
|
75 | - */ |
|
76 | - public $LIB; |
|
77 | - |
|
78 | - /** |
|
79 | - * @var EE_Request_Handler $REQ |
|
80 | - */ |
|
81 | - public $REQ; |
|
82 | - |
|
83 | - /** |
|
84 | - * @var EE_Session $SSN |
|
85 | - */ |
|
86 | - public $SSN; |
|
87 | - |
|
88 | - /** |
|
89 | - * @since 4.5.0 |
|
90 | - * @var EE_Capabilities $CAP |
|
91 | - */ |
|
92 | - public $CAP; |
|
93 | - |
|
94 | - /** |
|
95 | - * @since 4.9.0 |
|
96 | - * @var EE_Message_Resource_Manager $MRM |
|
97 | - */ |
|
98 | - public $MRM; |
|
99 | - |
|
100 | - /** |
|
101 | - * @var Registry $AssetsRegistry |
|
102 | - */ |
|
103 | - public $AssetsRegistry; |
|
104 | - |
|
105 | - /** |
|
106 | - * StdClass object for holding addons which have registered themselves to work with EE core |
|
107 | - * |
|
108 | - * @var EE_Addon[] $addons |
|
109 | - */ |
|
110 | - public $addons; |
|
111 | - |
|
112 | - /** |
|
113 | - * keys are 'short names' (eg Event), values are class names (eg 'EEM_Event') |
|
114 | - * |
|
115 | - * @var EEM_Base[] $models |
|
116 | - */ |
|
117 | - public $models = array(); |
|
118 | - |
|
119 | - /** |
|
120 | - * @var EED_Module[] $modules |
|
121 | - */ |
|
122 | - public $modules; |
|
123 | - |
|
124 | - /** |
|
125 | - * @var EES_Shortcode[] $shortcodes |
|
126 | - */ |
|
127 | - public $shortcodes; |
|
128 | - |
|
129 | - /** |
|
130 | - * @var WP_Widget[] $widgets |
|
131 | - */ |
|
132 | - public $widgets; |
|
133 | - |
|
134 | - /** |
|
135 | - * this is an array of all implemented model names (i.e. not the parent abstract models, or models |
|
136 | - * which don't actually fetch items from the DB in the normal way (ie, are not children of EEM_Base)). |
|
137 | - * Keys are model "short names" (eg "Event") as used in model relations, and values are |
|
138 | - * classnames (eg "EEM_Event") |
|
139 | - * |
|
140 | - * @var array $non_abstract_db_models |
|
141 | - */ |
|
142 | - public $non_abstract_db_models = array(); |
|
143 | - |
|
144 | - /** |
|
145 | - * internationalization for JS strings |
|
146 | - * usage: EE_Registry::i18n_js_strings['string_key'] = esc_html__( 'string to translate.', 'event_espresso' ); |
|
147 | - * in js file: var translatedString = eei18n.string_key; |
|
148 | - * |
|
149 | - * @var array $i18n_js_strings |
|
150 | - */ |
|
151 | - public static $i18n_js_strings = array(); |
|
152 | - |
|
153 | - /** |
|
154 | - * $main_file - path to espresso.php |
|
155 | - * |
|
156 | - * @var array $main_file |
|
157 | - */ |
|
158 | - public $main_file; |
|
159 | - |
|
160 | - /** |
|
161 | - * array of ReflectionClass objects where the key is the class name |
|
162 | - * |
|
163 | - * @deprecated 4.9.62.p |
|
164 | - * @var ReflectionClass[] $_reflectors |
|
165 | - */ |
|
166 | - public $_reflectors; |
|
167 | - |
|
168 | - /** |
|
169 | - * boolean flag to indicate whether or not to load/save dependencies from/to the cache |
|
170 | - * |
|
171 | - * @var boolean $_cache_on |
|
172 | - */ |
|
173 | - protected $_cache_on = true; |
|
174 | - |
|
175 | - /** |
|
176 | - * @var ObjectIdentifier |
|
177 | - */ |
|
178 | - private $object_identifier; |
|
179 | - |
|
180 | - |
|
181 | - /** |
|
182 | - * @singleton method used to instantiate class object |
|
183 | - * @param EE_Dependency_Map|null $dependency_map |
|
184 | - * @param Mirror|null $mirror |
|
185 | - * @param ClassInterfaceCache|null $class_cache |
|
186 | - * @param ObjectIdentifier|null $object_identifier |
|
187 | - * @return EE_Registry instance |
|
188 | - */ |
|
189 | - public static function instance( |
|
190 | - EE_Dependency_Map $dependency_map = null, |
|
191 | - Mirror $mirror = null, |
|
192 | - ClassInterfaceCache $class_cache = null, |
|
193 | - ObjectIdentifier $object_identifier = null |
|
194 | - ) { |
|
195 | - // check if class object is instantiated |
|
196 | - if (! self::$_instance instanceof EE_Registry |
|
197 | - && $dependency_map instanceof EE_Dependency_Map |
|
198 | - && $mirror instanceof Mirror |
|
199 | - && $class_cache instanceof ClassInterfaceCache |
|
200 | - && $object_identifier instanceof ObjectIdentifier |
|
201 | - ) { |
|
202 | - self::$_instance = new self( |
|
203 | - $dependency_map, |
|
204 | - $mirror, |
|
205 | - $class_cache, |
|
206 | - $object_identifier |
|
207 | - ); |
|
208 | - } |
|
209 | - return self::$_instance; |
|
210 | - } |
|
211 | - |
|
212 | - |
|
213 | - /** |
|
214 | - * protected constructor to prevent direct creation |
|
215 | - * |
|
216 | - * @Constructor |
|
217 | - * @param EE_Dependency_Map $dependency_map |
|
218 | - * @param Mirror $mirror |
|
219 | - * @param ClassInterfaceCache $class_cache |
|
220 | - * @param ObjectIdentifier $object_identifier |
|
221 | - */ |
|
222 | - protected function __construct( |
|
223 | - EE_Dependency_Map $dependency_map, |
|
224 | - Mirror $mirror, |
|
225 | - ClassInterfaceCache $class_cache, |
|
226 | - ObjectIdentifier $object_identifier |
|
227 | - ) { |
|
228 | - $this->_dependency_map = $dependency_map; |
|
229 | - $this->mirror = $mirror; |
|
230 | - $this->class_cache = $class_cache; |
|
231 | - $this->object_identifier = $object_identifier; |
|
232 | - // $registry_container = new RegistryContainer(); |
|
233 | - $this->LIB = new RegistryContainer(); |
|
234 | - $this->addons = new RegistryContainer(); |
|
235 | - $this->modules = new RegistryContainer(); |
|
236 | - $this->shortcodes = new RegistryContainer(); |
|
237 | - $this->widgets = new RegistryContainer(); |
|
238 | - add_action('EE_Load_Espresso_Core__handle_request__initialize_core_loading', array($this, 'initialize')); |
|
239 | - } |
|
240 | - |
|
241 | - |
|
242 | - /** |
|
243 | - * initialize |
|
244 | - * |
|
245 | - * @throws OutOfBoundsException |
|
246 | - * @throws InvalidArgumentException |
|
247 | - * @throws InvalidInterfaceException |
|
248 | - * @throws InvalidDataTypeException |
|
249 | - * @throws EE_Error |
|
250 | - * @throws ReflectionException |
|
251 | - */ |
|
252 | - public function initialize() |
|
253 | - { |
|
254 | - $this->_class_abbreviations = apply_filters( |
|
255 | - 'FHEE__EE_Registry____construct___class_abbreviations', |
|
256 | - array( |
|
257 | - 'EE_Config' => 'CFG', |
|
258 | - 'EE_Session' => 'SSN', |
|
259 | - 'EE_Capabilities' => 'CAP', |
|
260 | - 'EE_Cart' => 'CART', |
|
261 | - 'EE_Network_Config' => 'NET_CFG', |
|
262 | - 'EE_Request_Handler' => 'REQ', |
|
263 | - 'EE_Message_Resource_Manager' => 'MRM', |
|
264 | - 'EventEspresso\core\services\commands\CommandBus' => 'BUS', |
|
265 | - 'EventEspresso\core\services\assets\Registry' => 'AssetsRegistry', |
|
266 | - ) |
|
267 | - ); |
|
268 | - $this->load_core('Base', array(), true); |
|
269 | - // add our request and response objects to the cache |
|
270 | - $request_loader = $this->_dependency_map->class_loader( |
|
271 | - 'EventEspresso\core\services\request\Request' |
|
272 | - ); |
|
273 | - $this->_set_cached_class( |
|
274 | - $request_loader(), |
|
275 | - 'EventEspresso\core\services\request\Request' |
|
276 | - ); |
|
277 | - $response_loader = $this->_dependency_map->class_loader( |
|
278 | - 'EventEspresso\core\services\request\Response' |
|
279 | - ); |
|
280 | - $this->_set_cached_class( |
|
281 | - $response_loader(), |
|
282 | - 'EventEspresso\core\services\request\Response' |
|
283 | - ); |
|
284 | - add_action('AHEE__EE_System__set_hooks_for_core', array($this, 'init')); |
|
285 | - } |
|
286 | - |
|
287 | - |
|
288 | - /** |
|
289 | - * @return void |
|
290 | - */ |
|
291 | - public function init() |
|
292 | - { |
|
293 | - // Get current page protocol |
|
294 | - $protocol = isset($_SERVER['HTTPS']) ? 'https://' : 'http://'; |
|
295 | - // Output admin-ajax.php URL with same protocol as current page |
|
296 | - self::$i18n_js_strings['ajax_url'] = admin_url('admin-ajax.php', $protocol); |
|
297 | - self::$i18n_js_strings['wp_debug'] = defined('WP_DEBUG') ? WP_DEBUG : false; |
|
298 | - } |
|
299 | - |
|
300 | - |
|
301 | - /** |
|
302 | - * localize_i18n_js_strings |
|
303 | - * |
|
304 | - * @return string |
|
305 | - */ |
|
306 | - public static function localize_i18n_js_strings() |
|
307 | - { |
|
308 | - $i18n_js_strings = (array) self::$i18n_js_strings; |
|
309 | - foreach ($i18n_js_strings as $key => $value) { |
|
310 | - if (is_scalar($value)) { |
|
311 | - $i18n_js_strings[ $key ] = html_entity_decode((string) $value, ENT_QUOTES, 'UTF-8'); |
|
312 | - } |
|
313 | - } |
|
314 | - return '/* <![CDATA[ */ var eei18n = ' . wp_json_encode($i18n_js_strings) . '; /* ]]> */'; |
|
315 | - } |
|
316 | - |
|
317 | - |
|
318 | - /** |
|
319 | - * @param mixed string | EED_Module $module |
|
320 | - * @throws OutOfBoundsException |
|
321 | - * @throws InvalidArgumentException |
|
322 | - * @throws InvalidInterfaceException |
|
323 | - * @throws InvalidDataTypeException |
|
324 | - * @throws EE_Error |
|
325 | - * @throws ReflectionException |
|
326 | - */ |
|
327 | - public function add_module($module) |
|
328 | - { |
|
329 | - if ($module instanceof EED_Module) { |
|
330 | - $module_class = get_class($module); |
|
331 | - $this->modules->{$module_class} = $module; |
|
332 | - } else { |
|
333 | - if (! class_exists('EE_Module_Request_Router', false)) { |
|
334 | - $this->load_core('Module_Request_Router'); |
|
335 | - } |
|
336 | - EE_Module_Request_Router::module_factory($module); |
|
337 | - } |
|
338 | - } |
|
339 | - |
|
340 | - |
|
341 | - /** |
|
342 | - * @param string $module_name |
|
343 | - * @return mixed EED_Module | NULL |
|
344 | - */ |
|
345 | - public function get_module($module_name = '') |
|
346 | - { |
|
347 | - return isset($this->modules->{$module_name}) |
|
348 | - ? $this->modules->{$module_name} |
|
349 | - : null; |
|
350 | - } |
|
351 | - |
|
352 | - |
|
353 | - /** |
|
354 | - * loads core classes - must be singletons |
|
355 | - * |
|
356 | - * @param string $class_name - simple class name ie: session |
|
357 | - * @param mixed $arguments |
|
358 | - * @param bool $load_only |
|
359 | - * @return mixed |
|
360 | - * @throws InvalidInterfaceException |
|
361 | - * @throws InvalidDataTypeException |
|
362 | - * @throws EE_Error |
|
363 | - * @throws ReflectionException |
|
364 | - * @throws InvalidArgumentException |
|
365 | - */ |
|
366 | - public function load_core($class_name, $arguments = array(), $load_only = false) |
|
367 | - { |
|
368 | - $core_paths = apply_filters( |
|
369 | - 'FHEE__EE_Registry__load_core__core_paths', |
|
370 | - array( |
|
371 | - EE_CORE, |
|
372 | - EE_ADMIN, |
|
373 | - EE_CPTS, |
|
374 | - EE_CORE . 'data_migration_scripts' . DS, |
|
375 | - EE_CORE . 'capabilities' . DS, |
|
376 | - EE_CORE . 'request_stack' . DS, |
|
377 | - EE_CORE . 'middleware' . DS, |
|
378 | - ) |
|
379 | - ); |
|
380 | - // retrieve instantiated class |
|
381 | - return $this->_load( |
|
382 | - $core_paths, |
|
383 | - 'EE_', |
|
384 | - $class_name, |
|
385 | - 'core', |
|
386 | - $arguments, |
|
387 | - false, |
|
388 | - true, |
|
389 | - $load_only |
|
390 | - ); |
|
391 | - } |
|
392 | - |
|
393 | - |
|
394 | - /** |
|
395 | - * loads service classes |
|
396 | - * |
|
397 | - * @param string $class_name - simple class name ie: session |
|
398 | - * @param mixed $arguments |
|
399 | - * @param bool $load_only |
|
400 | - * @return mixed |
|
401 | - * @throws InvalidInterfaceException |
|
402 | - * @throws InvalidDataTypeException |
|
403 | - * @throws EE_Error |
|
404 | - * @throws ReflectionException |
|
405 | - * @throws InvalidArgumentException |
|
406 | - */ |
|
407 | - public function load_service($class_name, $arguments = array(), $load_only = false) |
|
408 | - { |
|
409 | - $service_paths = apply_filters( |
|
410 | - 'FHEE__EE_Registry__load_service__service_paths', |
|
411 | - array( |
|
412 | - EE_CORE . 'services' . DS, |
|
413 | - ) |
|
414 | - ); |
|
415 | - // retrieve instantiated class |
|
416 | - return $this->_load( |
|
417 | - $service_paths, |
|
418 | - 'EE_', |
|
419 | - $class_name, |
|
420 | - 'class', |
|
421 | - $arguments, |
|
422 | - false, |
|
423 | - true, |
|
424 | - $load_only |
|
425 | - ); |
|
426 | - } |
|
427 | - |
|
428 | - |
|
429 | - /** |
|
430 | - * loads data_migration_scripts |
|
431 | - * |
|
432 | - * @param string $class_name - class name for the DMS ie: EE_DMS_Core_4_2_0 |
|
433 | - * @param mixed $arguments |
|
434 | - * @return EE_Data_Migration_Script_Base|mixed |
|
435 | - * @throws InvalidInterfaceException |
|
436 | - * @throws InvalidDataTypeException |
|
437 | - * @throws EE_Error |
|
438 | - * @throws ReflectionException |
|
439 | - * @throws InvalidArgumentException |
|
440 | - */ |
|
441 | - public function load_dms($class_name, $arguments = array()) |
|
442 | - { |
|
443 | - // retrieve instantiated class |
|
444 | - return $this->_load( |
|
445 | - EE_Data_Migration_Manager::instance()->get_data_migration_script_folders(), |
|
446 | - 'EE_DMS_', |
|
447 | - $class_name, |
|
448 | - 'dms', |
|
449 | - $arguments, |
|
450 | - false, |
|
451 | - false |
|
452 | - ); |
|
453 | - } |
|
454 | - |
|
455 | - |
|
456 | - /** |
|
457 | - * loads object creating classes - must be singletons |
|
458 | - * |
|
459 | - * @param string $class_name - simple class name ie: attendee |
|
460 | - * @param mixed $arguments - an array of arguments to pass to the class |
|
461 | - * @param bool $from_db - some classes are instantiated from the db and thus call a different method to |
|
462 | - * instantiate |
|
463 | - * @param bool $cache if you don't want the class to be stored in the internal cache (non-persistent) then |
|
464 | - * set this to FALSE (ie. when instantiating model objects from client in a loop) |
|
465 | - * @param bool $load_only whether or not to just load the file and NOT instantiate, or load AND instantiate |
|
466 | - * (default) |
|
467 | - * @return EE_Base_Class | bool |
|
468 | - * @throws InvalidInterfaceException |
|
469 | - * @throws InvalidDataTypeException |
|
470 | - * @throws EE_Error |
|
471 | - * @throws ReflectionException |
|
472 | - * @throws InvalidArgumentException |
|
473 | - */ |
|
474 | - public function load_class($class_name, $arguments = array(), $from_db = false, $cache = true, $load_only = false) |
|
475 | - { |
|
476 | - $paths = apply_filters( |
|
477 | - 'FHEE__EE_Registry__load_class__paths', |
|
478 | - array( |
|
479 | - EE_CORE, |
|
480 | - EE_CLASSES, |
|
481 | - EE_BUSINESS, |
|
482 | - ) |
|
483 | - ); |
|
484 | - // retrieve instantiated class |
|
485 | - return $this->_load( |
|
486 | - $paths, |
|
487 | - 'EE_', |
|
488 | - $class_name, |
|
489 | - 'class', |
|
490 | - $arguments, |
|
491 | - $from_db, |
|
492 | - $cache, |
|
493 | - $load_only |
|
494 | - ); |
|
495 | - } |
|
496 | - |
|
497 | - |
|
498 | - /** |
|
499 | - * loads helper classes - must be singletons |
|
500 | - * |
|
501 | - * @param string $class_name - simple class name ie: price |
|
502 | - * @param mixed $arguments |
|
503 | - * @param bool $load_only |
|
504 | - * @return EEH_Base | bool |
|
505 | - * @throws InvalidInterfaceException |
|
506 | - * @throws InvalidDataTypeException |
|
507 | - * @throws EE_Error |
|
508 | - * @throws ReflectionException |
|
509 | - * @throws InvalidArgumentException |
|
510 | - */ |
|
511 | - public function load_helper($class_name, $arguments = array(), $load_only = true) |
|
512 | - { |
|
513 | - // todo: add doing_it_wrong() in a few versions after all addons have had calls to this method removed |
|
514 | - $helper_paths = apply_filters('FHEE__EE_Registry__load_helper__helper_paths', array(EE_HELPERS)); |
|
515 | - // retrieve instantiated class |
|
516 | - return $this->_load( |
|
517 | - $helper_paths, |
|
518 | - 'EEH_', |
|
519 | - $class_name, |
|
520 | - 'helper', |
|
521 | - $arguments, |
|
522 | - false, |
|
523 | - true, |
|
524 | - $load_only |
|
525 | - ); |
|
526 | - } |
|
527 | - |
|
528 | - |
|
529 | - /** |
|
530 | - * loads core classes - must be singletons |
|
531 | - * |
|
532 | - * @param string $class_name - simple class name ie: session |
|
533 | - * @param mixed $arguments |
|
534 | - * @param bool $load_only |
|
535 | - * @param bool $cache whether to cache the object or not. |
|
536 | - * @return mixed |
|
537 | - * @throws InvalidInterfaceException |
|
538 | - * @throws InvalidDataTypeException |
|
539 | - * @throws EE_Error |
|
540 | - * @throws ReflectionException |
|
541 | - * @throws InvalidArgumentException |
|
542 | - */ |
|
543 | - public function load_lib($class_name, $arguments = array(), $load_only = false, $cache = true) |
|
544 | - { |
|
545 | - $paths = array( |
|
546 | - EE_LIBRARIES, |
|
547 | - EE_LIBRARIES . 'messages' . DS, |
|
548 | - EE_LIBRARIES . 'shortcodes' . DS, |
|
549 | - EE_LIBRARIES . 'qtips' . DS, |
|
550 | - EE_LIBRARIES . 'payment_methods' . DS, |
|
551 | - ); |
|
552 | - // retrieve instantiated class |
|
553 | - return $this->_load( |
|
554 | - $paths, |
|
555 | - 'EE_', |
|
556 | - $class_name, |
|
557 | - 'lib', |
|
558 | - $arguments, |
|
559 | - false, |
|
560 | - $cache, |
|
561 | - $load_only |
|
562 | - ); |
|
563 | - } |
|
564 | - |
|
565 | - |
|
566 | - /** |
|
567 | - * loads model classes - must be singletons |
|
568 | - * |
|
569 | - * @param string $class_name - simple class name ie: price |
|
570 | - * @param mixed $arguments |
|
571 | - * @param bool $load_only |
|
572 | - * @return EEM_Base | bool |
|
573 | - * @throws InvalidInterfaceException |
|
574 | - * @throws InvalidDataTypeException |
|
575 | - * @throws EE_Error |
|
576 | - * @throws ReflectionException |
|
577 | - * @throws InvalidArgumentException |
|
578 | - */ |
|
579 | - public function load_model($class_name, $arguments = array(), $load_only = false) |
|
580 | - { |
|
581 | - $paths = apply_filters( |
|
582 | - 'FHEE__EE_Registry__load_model__paths', |
|
583 | - array( |
|
584 | - EE_MODELS, |
|
585 | - EE_CORE, |
|
586 | - ) |
|
587 | - ); |
|
588 | - // retrieve instantiated class |
|
589 | - return $this->_load( |
|
590 | - $paths, |
|
591 | - 'EEM_', |
|
592 | - $class_name, |
|
593 | - 'model', |
|
594 | - $arguments, |
|
595 | - false, |
|
596 | - true, |
|
597 | - $load_only |
|
598 | - ); |
|
599 | - } |
|
600 | - |
|
601 | - |
|
602 | - /** |
|
603 | - * loads model classes - must be singletons |
|
604 | - * |
|
605 | - * @param string $class_name - simple class name ie: price |
|
606 | - * @param mixed $arguments |
|
607 | - * @param bool $load_only |
|
608 | - * @return mixed | bool |
|
609 | - * @throws InvalidInterfaceException |
|
610 | - * @throws InvalidDataTypeException |
|
611 | - * @throws EE_Error |
|
612 | - * @throws ReflectionException |
|
613 | - * @throws InvalidArgumentException |
|
614 | - */ |
|
615 | - public function load_model_class($class_name, $arguments = array(), $load_only = true) |
|
616 | - { |
|
617 | - $paths = array( |
|
618 | - EE_MODELS . 'fields' . DS, |
|
619 | - EE_MODELS . 'helpers' . DS, |
|
620 | - EE_MODELS . 'relations' . DS, |
|
621 | - EE_MODELS . 'strategies' . DS, |
|
622 | - ); |
|
623 | - // retrieve instantiated class |
|
624 | - return $this->_load( |
|
625 | - $paths, |
|
626 | - 'EE_', |
|
627 | - $class_name, |
|
628 | - '', |
|
629 | - $arguments, |
|
630 | - false, |
|
631 | - true, |
|
632 | - $load_only |
|
633 | - ); |
|
634 | - } |
|
635 | - |
|
636 | - |
|
637 | - /** |
|
638 | - * Determines if $model_name is the name of an actual EE model. |
|
639 | - * |
|
640 | - * @param string $model_name like Event, Attendee, Question_Group_Question, etc. |
|
641 | - * @return boolean |
|
642 | - */ |
|
643 | - public function is_model_name($model_name) |
|
644 | - { |
|
645 | - return isset($this->models[ $model_name ]); |
|
646 | - } |
|
647 | - |
|
648 | - |
|
649 | - /** |
|
650 | - * generic class loader |
|
651 | - * |
|
652 | - * @param string $path_to_file - directory path to file location, not including filename |
|
653 | - * @param string $file_name - file name ie: my_file.php, including extension |
|
654 | - * @param string $type - file type - core? class? helper? model? |
|
655 | - * @param mixed $arguments |
|
656 | - * @param bool $load_only |
|
657 | - * @return mixed |
|
658 | - * @throws InvalidInterfaceException |
|
659 | - * @throws InvalidDataTypeException |
|
660 | - * @throws EE_Error |
|
661 | - * @throws ReflectionException |
|
662 | - * @throws InvalidArgumentException |
|
663 | - */ |
|
664 | - public function load_file($path_to_file, $file_name, $type = '', $arguments = array(), $load_only = true) |
|
665 | - { |
|
666 | - // retrieve instantiated class |
|
667 | - return $this->_load( |
|
668 | - $path_to_file, |
|
669 | - '', |
|
670 | - $file_name, |
|
671 | - $type, |
|
672 | - $arguments, |
|
673 | - false, |
|
674 | - true, |
|
675 | - $load_only |
|
676 | - ); |
|
677 | - } |
|
678 | - |
|
679 | - |
|
680 | - /** |
|
681 | - * @param string $path_to_file - directory path to file location, not including filename |
|
682 | - * @param string $class_name - full class name ie: My_Class |
|
683 | - * @param string $type - file type - core? class? helper? model? |
|
684 | - * @param mixed $arguments |
|
685 | - * @param bool $load_only |
|
686 | - * @return bool|EE_Addon|object |
|
687 | - * @throws InvalidInterfaceException |
|
688 | - * @throws InvalidDataTypeException |
|
689 | - * @throws EE_Error |
|
690 | - * @throws ReflectionException |
|
691 | - * @throws InvalidArgumentException |
|
692 | - */ |
|
693 | - public function load_addon($path_to_file, $class_name, $type = 'class', $arguments = array(), $load_only = false) |
|
694 | - { |
|
695 | - // retrieve instantiated class |
|
696 | - return $this->_load( |
|
697 | - $path_to_file, |
|
698 | - 'addon', |
|
699 | - $class_name, |
|
700 | - $type, |
|
701 | - $arguments, |
|
702 | - false, |
|
703 | - true, |
|
704 | - $load_only |
|
705 | - ); |
|
706 | - } |
|
707 | - |
|
708 | - |
|
709 | - /** |
|
710 | - * instantiates, caches, and automatically resolves dependencies |
|
711 | - * for classes that use a Fully Qualified Class Name. |
|
712 | - * if the class is not capable of being loaded using PSR-4 autoloading, |
|
713 | - * then you need to use one of the existing load_*() methods |
|
714 | - * which can resolve the classname and filepath from the passed arguments |
|
715 | - * |
|
716 | - * @param bool|string $class_name Fully Qualified Class Name |
|
717 | - * @param array $arguments an argument, or array of arguments to pass to the class upon instantiation |
|
718 | - * @param bool $cache whether to cache the instantiated object for reuse |
|
719 | - * @param bool $from_db some classes are instantiated from the db |
|
720 | - * and thus call a different method to instantiate |
|
721 | - * @param bool $load_only if true, will only load the file, but will NOT instantiate an object |
|
722 | - * @param bool|string $addon if true, will cache the object in the EE_Registry->$addons array |
|
723 | - * @return bool|null|mixed null = failure to load or instantiate class object. |
|
724 | - * object = class loaded and instantiated successfully. |
|
725 | - * bool = fail or success when $load_only is true |
|
726 | - * @throws InvalidInterfaceException |
|
727 | - * @throws InvalidDataTypeException |
|
728 | - * @throws EE_Error |
|
729 | - * @throws ReflectionException |
|
730 | - * @throws InvalidArgumentException |
|
731 | - */ |
|
732 | - public function create( |
|
733 | - $class_name = false, |
|
734 | - $arguments = array(), |
|
735 | - $cache = false, |
|
736 | - $from_db = false, |
|
737 | - $load_only = false, |
|
738 | - $addon = false |
|
739 | - ) { |
|
740 | - $class_name = ltrim($class_name, '\\'); |
|
741 | - $class_name = $this->class_cache->getFqnForAlias($class_name); |
|
742 | - $class_exists = $this->loadOrVerifyClassExists($class_name, $arguments); |
|
743 | - // if a non-FQCN was passed, then |
|
744 | - // verifyClassExists() might return an object |
|
745 | - // or it could return null if the class just could not be found anywhere |
|
746 | - if ($class_exists instanceof $class_name || $class_exists === null) { |
|
747 | - // either way, return the results |
|
748 | - return $class_exists; |
|
749 | - } |
|
750 | - $class_name = $class_exists; |
|
751 | - // if we're only loading the class and it already exists, then let's just return true immediately |
|
752 | - if ($load_only) { |
|
753 | - return true; |
|
754 | - } |
|
755 | - $addon = $addon ? 'addon' : ''; |
|
756 | - // $this->_cache_on is toggled during the recursive loading that can occur with dependency injection |
|
757 | - // $cache is controlled by individual calls to separate Registry loader methods like load_class() |
|
758 | - // $load_only is also controlled by individual calls to separate Registry loader methods like load_file() |
|
759 | - if ($this->_cache_on && $cache && ! $load_only) { |
|
760 | - // return object if it's already cached |
|
761 | - $cached_class = $this->_get_cached_class($class_name, $addon, $arguments); |
|
762 | - if ($cached_class !== null) { |
|
763 | - return $cached_class; |
|
764 | - } |
|
765 | - }// obtain the loader method from the dependency map |
|
766 | - $loader = $this->_dependency_map->class_loader($class_name);// instantiate the requested object |
|
767 | - if ($loader instanceof Closure) { |
|
768 | - $class_obj = $loader($arguments); |
|
769 | - } else { |
|
770 | - if ($loader && method_exists($this, $loader)) { |
|
771 | - $class_obj = $this->{$loader}($class_name, $arguments); |
|
772 | - } else { |
|
773 | - $class_obj = $this->_create_object($class_name, $arguments, $addon, $from_db); |
|
774 | - } |
|
775 | - } |
|
776 | - if (($this->_cache_on && $cache) || $this->get_class_abbreviation($class_name, '')) { |
|
777 | - // save it for later... kinda like gum { : $ |
|
778 | - $this->_set_cached_class( |
|
779 | - $class_obj, |
|
780 | - $class_name, |
|
781 | - $addon, |
|
782 | - $from_db, |
|
783 | - $arguments |
|
784 | - ); |
|
785 | - } |
|
786 | - $this->_cache_on = true; |
|
787 | - return $class_obj; |
|
788 | - } |
|
789 | - |
|
790 | - |
|
791 | - /** |
|
792 | - * Recursively checks that a class exists and potentially attempts to load classes with non-FQCNs |
|
793 | - * |
|
794 | - * @param string|object $class_name |
|
795 | - * @param array $arguments |
|
796 | - * @param int $attempt |
|
797 | - * @return mixed |
|
798 | - */ |
|
799 | - private function loadOrVerifyClassExists($class_name, array $arguments, $attempt = 1) |
|
800 | - { |
|
801 | - if (is_object($class_name) || class_exists($class_name)) { |
|
802 | - return $class_name; |
|
803 | - } |
|
804 | - switch ($attempt) { |
|
805 | - case 1: |
|
806 | - // if it's a FQCN then maybe the class is registered with a preceding \ |
|
807 | - $class_name = strpos($class_name, '\\') !== false |
|
808 | - ? '\\' . ltrim($class_name, '\\') |
|
809 | - : $class_name; |
|
810 | - break; |
|
811 | - case 2: |
|
812 | - // |
|
813 | - $loader = $this->_dependency_map->class_loader($class_name); |
|
814 | - if ($loader && method_exists($this, $loader)) { |
|
815 | - return $this->{$loader}($class_name, $arguments); |
|
816 | - } |
|
817 | - break; |
|
818 | - case 3: |
|
819 | - default: |
|
820 | - return null; |
|
821 | - } |
|
822 | - $attempt++; |
|
823 | - return $this->loadOrVerifyClassExists($class_name, $arguments, $attempt); |
|
824 | - } |
|
825 | - |
|
826 | - |
|
827 | - /** |
|
828 | - * instantiates, caches, and injects dependencies for classes |
|
829 | - * |
|
830 | - * @param array $file_paths an array of paths to folders to look in |
|
831 | - * @param string $class_prefix EE or EEM or... ??? |
|
832 | - * @param bool|string $class_name $class name |
|
833 | - * @param string $type file type - core? class? helper? model? |
|
834 | - * @param mixed $arguments an argument or array of arguments to pass to the class upon instantiation |
|
835 | - * @param bool $from_db some classes are instantiated from the db |
|
836 | - * and thus call a different method to instantiate |
|
837 | - * @param bool $cache whether to cache the instantiated object for reuse |
|
838 | - * @param bool $load_only if true, will only load the file, but will NOT instantiate an object |
|
839 | - * @return bool|null|object null = failure to load or instantiate class object. |
|
840 | - * object = class loaded and instantiated successfully. |
|
841 | - * bool = fail or success when $load_only is true |
|
842 | - * @throws EE_Error |
|
843 | - * @throws ReflectionException |
|
844 | - * @throws InvalidInterfaceException |
|
845 | - * @throws InvalidDataTypeException |
|
846 | - * @throws InvalidArgumentException |
|
847 | - */ |
|
848 | - protected function _load( |
|
849 | - $file_paths = array(), |
|
850 | - $class_prefix = 'EE_', |
|
851 | - $class_name = false, |
|
852 | - $type = 'class', |
|
853 | - $arguments = array(), |
|
854 | - $from_db = false, |
|
855 | - $cache = true, |
|
856 | - $load_only = false |
|
857 | - ) { |
|
858 | - $class_name = ltrim($class_name, '\\'); |
|
859 | - // strip php file extension |
|
860 | - $class_name = str_replace('.php', '', trim($class_name)); |
|
861 | - // does the class have a prefix ? |
|
862 | - if (! empty($class_prefix) && $class_prefix !== 'addon') { |
|
863 | - // make sure $class_prefix is uppercase |
|
864 | - $class_prefix = strtoupper(trim($class_prefix)); |
|
865 | - // add class prefix ONCE!!! |
|
866 | - $class_name = $class_prefix . str_replace($class_prefix, '', $class_name); |
|
867 | - } |
|
868 | - $class_name = $this->class_cache->getFqnForAlias($class_name); |
|
869 | - $class_exists = class_exists($class_name, false); |
|
870 | - // if we're only loading the class and it already exists, then let's just return true immediately |
|
871 | - if ($load_only && $class_exists) { |
|
872 | - return true; |
|
873 | - } |
|
874 | - $arguments = is_array($arguments) ? $arguments : array($arguments); |
|
875 | - // $this->_cache_on is toggled during the recursive loading that can occur with dependency injection |
|
876 | - // $cache is controlled by individual calls to separate Registry loader methods like load_class() |
|
877 | - // $load_only is also controlled by individual calls to separate Registry loader methods like load_file() |
|
878 | - if ($this->_cache_on && $cache && ! $load_only) { |
|
879 | - // return object if it's already cached |
|
880 | - $cached_class = $this->_get_cached_class($class_name, $class_prefix, $arguments); |
|
881 | - if ($cached_class !== null) { |
|
882 | - return $cached_class; |
|
883 | - } |
|
884 | - } |
|
885 | - // if the class doesn't already exist.. then we need to try and find the file and load it |
|
886 | - if (! $class_exists) { |
|
887 | - // get full path to file |
|
888 | - $path = $this->_resolve_path($class_name, $type, $file_paths); |
|
889 | - // load the file |
|
890 | - $loaded = $this->_require_file($path, $class_name, $type, $file_paths); |
|
891 | - // if we are only loading a file but NOT instantiating an object |
|
892 | - // then return boolean for whether class was loaded or not |
|
893 | - if ($load_only) { |
|
894 | - return $loaded; |
|
895 | - } |
|
896 | - // if an object was expected but loading failed, then return nothing |
|
897 | - if (! $loaded) { |
|
898 | - return null; |
|
899 | - } |
|
900 | - } |
|
901 | - // instantiate the requested object |
|
902 | - $class_obj = $this->_create_object($class_name, $arguments, $type, $from_db); |
|
903 | - if ($this->_cache_on && $cache) { |
|
904 | - // save it for later... kinda like gum { : $ |
|
905 | - $this->_set_cached_class( |
|
906 | - $class_obj, |
|
907 | - $class_name, |
|
908 | - $class_prefix, |
|
909 | - $from_db, |
|
910 | - $arguments |
|
911 | - ); |
|
912 | - } |
|
913 | - $this->_cache_on = true; |
|
914 | - return $class_obj; |
|
915 | - } |
|
916 | - |
|
917 | - |
|
918 | - /** |
|
919 | - * @param string $class_name |
|
920 | - * @param string $default have to specify something, but not anything that will conflict |
|
921 | - * @return mixed|string |
|
922 | - */ |
|
923 | - protected function get_class_abbreviation($class_name, $default = 'FANCY_BATMAN_PANTS') |
|
924 | - { |
|
925 | - return isset($this->_class_abbreviations[ $class_name ]) |
|
926 | - ? $this->_class_abbreviations[ $class_name ] |
|
927 | - : $default; |
|
928 | - } |
|
929 | - |
|
930 | - |
|
931 | - /** |
|
932 | - * attempts to find a cached version of the requested class |
|
933 | - * by looking in the following places: |
|
934 | - * $this->{$class_abbreviation} ie: $this->CART |
|
935 | - * $this->{$class_name} ie: $this->Some_Class |
|
936 | - * $this->LIB->{$class_name} ie: $this->LIB->Some_Class |
|
937 | - * $this->addon->{$class_name} ie: $this->addon->Some_Addon_Class |
|
938 | - * |
|
939 | - * @param string $class_name |
|
940 | - * @param string $class_prefix |
|
941 | - * @param array $arguments |
|
942 | - * @return mixed |
|
943 | - */ |
|
944 | - protected function _get_cached_class( |
|
945 | - $class_name, |
|
946 | - $class_prefix = '', |
|
947 | - $arguments = array() |
|
948 | - ) { |
|
949 | - if ($class_name === 'EE_Registry') { |
|
950 | - return $this; |
|
951 | - } |
|
952 | - $class_abbreviation = $this->get_class_abbreviation($class_name); |
|
953 | - // check if class has already been loaded, and return it if it has been |
|
954 | - if (isset($this->{$class_abbreviation})) { |
|
955 | - return $this->{$class_abbreviation}; |
|
956 | - } |
|
957 | - $class_name = str_replace('\\', '_', $class_name); |
|
958 | - if (isset($this->{$class_name})) { |
|
959 | - return $this->{$class_name}; |
|
960 | - } |
|
961 | - if ($class_prefix === 'addon' && isset($this->addons->{$class_name})) { |
|
962 | - return $this->addons->{$class_name}; |
|
963 | - } |
|
964 | - $object_identifier = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
965 | - if (isset($this->LIB->{$object_identifier})) { |
|
966 | - return $this->LIB->{$object_identifier}; |
|
967 | - } |
|
968 | - foreach ($this->LIB as $key => $object) { |
|
969 | - if (// request does not contain new arguments and therefore no args identifier |
|
970 | - ! $this->object_identifier->hasArguments($object_identifier) |
|
971 | - // but previously cached class with args was found |
|
972 | - && $this->object_identifier->fqcnMatchesObjectIdentifier($class_name, $key) |
|
973 | - ) { |
|
974 | - return $object; |
|
975 | - } |
|
976 | - } |
|
977 | - return null; |
|
978 | - } |
|
979 | - |
|
980 | - |
|
981 | - /** |
|
982 | - * removes a cached version of the requested class |
|
983 | - * |
|
984 | - * @param string $class_name |
|
985 | - * @param boolean $addon |
|
986 | - * @param array $arguments |
|
987 | - * @return boolean |
|
988 | - */ |
|
989 | - public function clear_cached_class( |
|
990 | - $class_name, |
|
991 | - $addon = false, |
|
992 | - $arguments = array() |
|
993 | - ) { |
|
994 | - $class_abbreviation = $this->get_class_abbreviation($class_name); |
|
995 | - // check if class has already been loaded, and return it if it has been |
|
996 | - if (isset($this->{$class_abbreviation})) { |
|
997 | - $this->{$class_abbreviation} = null; |
|
998 | - return true; |
|
999 | - } |
|
1000 | - $class_name = str_replace('\\', '_', $class_name); |
|
1001 | - if (isset($this->{$class_name})) { |
|
1002 | - $this->{$class_name} = null; |
|
1003 | - return true; |
|
1004 | - } |
|
1005 | - if ($addon && isset($this->addons->{$class_name})) { |
|
1006 | - unset($this->addons->{$class_name}); |
|
1007 | - return true; |
|
1008 | - } |
|
1009 | - $class_name = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
1010 | - if (isset($this->LIB->{$class_name})) { |
|
1011 | - unset($this->LIB->{$class_name}); |
|
1012 | - return true; |
|
1013 | - } |
|
1014 | - return false; |
|
1015 | - } |
|
1016 | - |
|
1017 | - |
|
1018 | - /** |
|
1019 | - * _set_cached_class |
|
1020 | - * attempts to cache the instantiated class locally |
|
1021 | - * in one of the following places, in the following order: |
|
1022 | - * $this->{class_abbreviation} ie: $this->CART |
|
1023 | - * $this->{$class_name} ie: $this->Some_Class |
|
1024 | - * $this->addon->{$$class_name} ie: $this->addon->Some_Addon_Class |
|
1025 | - * $this->LIB->{$class_name} ie: $this->LIB->Some_Class |
|
1026 | - * |
|
1027 | - * @param object $class_obj |
|
1028 | - * @param string $class_name |
|
1029 | - * @param string $class_prefix |
|
1030 | - * @param bool $from_db |
|
1031 | - * @param array $arguments |
|
1032 | - * @return void |
|
1033 | - */ |
|
1034 | - protected function _set_cached_class( |
|
1035 | - $class_obj, |
|
1036 | - $class_name, |
|
1037 | - $class_prefix = '', |
|
1038 | - $from_db = false, |
|
1039 | - $arguments = array() |
|
1040 | - ) { |
|
1041 | - if ($class_name === 'EE_Registry' || empty($class_obj)) { |
|
1042 | - return; |
|
1043 | - } |
|
1044 | - // return newly instantiated class |
|
1045 | - $class_abbreviation = $this->get_class_abbreviation($class_name, ''); |
|
1046 | - if ($class_abbreviation) { |
|
1047 | - $this->{$class_abbreviation} = $class_obj; |
|
1048 | - return; |
|
1049 | - } |
|
1050 | - $class_name = str_replace('\\', '_', $class_name); |
|
1051 | - if (property_exists($this, $class_name)) { |
|
1052 | - $this->{$class_name} = $class_obj; |
|
1053 | - return; |
|
1054 | - } |
|
1055 | - if ($class_prefix === 'addon') { |
|
1056 | - $this->addons->{$class_name} = $class_obj; |
|
1057 | - return; |
|
1058 | - } |
|
1059 | - if (! $from_db) { |
|
1060 | - $class_name = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
1061 | - $this->LIB->{$class_name} = $class_obj; |
|
1062 | - } |
|
1063 | - } |
|
1064 | - |
|
1065 | - |
|
1066 | - /** |
|
1067 | - * attempts to find a full valid filepath for the requested class. |
|
1068 | - * loops thru each of the base paths in the $file_paths array and appends : "{classname} . {file type} . php" |
|
1069 | - * then returns that path if the target file has been found and is readable |
|
1070 | - * |
|
1071 | - * @param string $class_name |
|
1072 | - * @param string $type |
|
1073 | - * @param array $file_paths |
|
1074 | - * @return string | bool |
|
1075 | - */ |
|
1076 | - protected function _resolve_path($class_name, $type = '', $file_paths = array()) |
|
1077 | - { |
|
1078 | - // make sure $file_paths is an array |
|
1079 | - $file_paths = is_array($file_paths) |
|
1080 | - ? $file_paths |
|
1081 | - : array($file_paths); |
|
1082 | - // cycle thru paths |
|
1083 | - foreach ($file_paths as $key => $file_path) { |
|
1084 | - // convert all separators to proper DS, if no filepath, then use EE_CLASSES |
|
1085 | - $file_path = $file_path |
|
1086 | - ? str_replace(array('/', '\\'), DS, $file_path) |
|
1087 | - : EE_CLASSES; |
|
1088 | - // prep file type |
|
1089 | - $type = ! empty($type) |
|
1090 | - ? trim($type, '.') . '.' |
|
1091 | - : ''; |
|
1092 | - // build full file path |
|
1093 | - $file_paths[ $key ] = rtrim($file_path, DS) . DS . $class_name . '.' . $type . 'php'; |
|
1094 | - // does the file exist and can be read ? |
|
1095 | - if (is_readable($file_paths[ $key ])) { |
|
1096 | - return $file_paths[ $key ]; |
|
1097 | - } |
|
1098 | - } |
|
1099 | - return false; |
|
1100 | - } |
|
1101 | - |
|
1102 | - |
|
1103 | - /** |
|
1104 | - * basically just performs a require_once() |
|
1105 | - * but with some error handling |
|
1106 | - * |
|
1107 | - * @param string $path |
|
1108 | - * @param string $class_name |
|
1109 | - * @param string $type |
|
1110 | - * @param array $file_paths |
|
1111 | - * @return bool |
|
1112 | - * @throws EE_Error |
|
1113 | - * @throws ReflectionException |
|
1114 | - */ |
|
1115 | - protected function _require_file($path, $class_name, $type = '', $file_paths = array()) |
|
1116 | - { |
|
1117 | - $this->resolve_legacy_class_parent($class_name); |
|
1118 | - // don't give up! you gotta... |
|
1119 | - try { |
|
1120 | - // does the file exist and can it be read ? |
|
1121 | - if (! $path) { |
|
1122 | - // just in case the file has already been autoloaded, |
|
1123 | - // but discrepancies in the naming schema are preventing it from |
|
1124 | - // being loaded via one of the EE_Registry::load_*() methods, |
|
1125 | - // then let's try one last hail mary before throwing an exception |
|
1126 | - // and call class_exists() again, but with autoloading turned ON |
|
1127 | - if (class_exists($class_name)) { |
|
1128 | - return true; |
|
1129 | - } |
|
1130 | - // so sorry, can't find the file |
|
1131 | - throw new EE_Error( |
|
1132 | - sprintf( |
|
1133 | - esc_html__( |
|
1134 | - 'The %1$s file %2$s could not be located or is not readable due to file permissions. Please ensure that the following filepath(s) are correct: %3$s', |
|
1135 | - 'event_espresso' |
|
1136 | - ), |
|
1137 | - trim($type, '.'), |
|
1138 | - $class_name, |
|
1139 | - '<br />' . implode(',<br />', $file_paths) |
|
1140 | - ) |
|
1141 | - ); |
|
1142 | - } |
|
1143 | - // get the file |
|
1144 | - require_once($path); |
|
1145 | - // if the class isn't already declared somewhere |
|
1146 | - if (class_exists($class_name, false) === false) { |
|
1147 | - // so sorry, not a class |
|
1148 | - throw new EE_Error( |
|
1149 | - sprintf( |
|
1150 | - esc_html__( |
|
1151 | - 'The %s file %s does not appear to contain the %s Class.', |
|
1152 | - 'event_espresso' |
|
1153 | - ), |
|
1154 | - $type, |
|
1155 | - $path, |
|
1156 | - $class_name |
|
1157 | - ) |
|
1158 | - ); |
|
1159 | - } |
|
1160 | - } catch (EE_Error $e) { |
|
1161 | - $e->get_error(); |
|
1162 | - return false; |
|
1163 | - } |
|
1164 | - return true; |
|
1165 | - } |
|
1166 | - |
|
1167 | - |
|
1168 | - /** |
|
1169 | - * Some of our legacy classes that extended a parent class would simply use a require() statement |
|
1170 | - * before their class declaration in order to ensure that the parent class was loaded. |
|
1171 | - * This is not ideal, but it's nearly impossible to determine the parent class of a non-namespaced class, |
|
1172 | - * without triggering a fatal error because the parent class has yet to be loaded and therefore doesn't exist. |
|
1173 | - * |
|
1174 | - * @param string $class_name |
|
1175 | - */ |
|
1176 | - protected function resolve_legacy_class_parent($class_name = '') |
|
1177 | - { |
|
1178 | - try { |
|
1179 | - $legacy_parent_class_map = array( |
|
1180 | - 'EE_Payment_Processor' => 'core/business/EE_Processor_Base.class.php', |
|
1181 | - ); |
|
1182 | - if (isset($legacy_parent_class_map[ $class_name ])) { |
|
1183 | - require_once EE_PLUGIN_DIR_PATH . $legacy_parent_class_map[ $class_name ]; |
|
1184 | - } |
|
1185 | - } catch (Exception $exception) { |
|
1186 | - } |
|
1187 | - } |
|
1188 | - |
|
1189 | - |
|
1190 | - /** |
|
1191 | - * _create_object |
|
1192 | - * Attempts to instantiate the requested class via any of the |
|
1193 | - * commonly used instantiation methods employed throughout EE. |
|
1194 | - * The priority for instantiation is as follows: |
|
1195 | - * - abstract classes or any class flagged as "load only" (no instantiation occurs) |
|
1196 | - * - model objects via their 'new_instance_from_db' method |
|
1197 | - * - model objects via their 'new_instance' method |
|
1198 | - * - "singleton" classes" via their 'instance' method |
|
1199 | - * - standard instantiable classes via their __constructor |
|
1200 | - * Prior to instantiation, if the classname exists in the dependency_map, |
|
1201 | - * then the constructor for the requested class will be examined to determine |
|
1202 | - * if any dependencies exist, and if they can be injected. |
|
1203 | - * If so, then those classes will be added to the array of arguments passed to the constructor |
|
1204 | - * |
|
1205 | - * @param string $class_name |
|
1206 | - * @param array $arguments |
|
1207 | - * @param string $type |
|
1208 | - * @param bool $from_db |
|
1209 | - * @return null|object|bool |
|
1210 | - * @throws InvalidArgumentException |
|
1211 | - * @throws InvalidInterfaceException |
|
1212 | - * @throws EE_Error |
|
1213 | - * @throws ReflectionException |
|
1214 | - * @throws InvalidDataTypeException |
|
1215 | - */ |
|
1216 | - protected function _create_object($class_name, $arguments = array(), $type = '', $from_db = false) |
|
1217 | - { |
|
1218 | - // create reflection |
|
1219 | - $reflector = $this->mirror->getReflectionClass($class_name); |
|
1220 | - // make sure arguments are an array |
|
1221 | - $arguments = is_array($arguments) |
|
1222 | - ? $arguments |
|
1223 | - : array($arguments); |
|
1224 | - // and if arguments array is numerically and sequentially indexed, then we want it to remain as is, |
|
1225 | - // else wrap it in an additional array so that it doesn't get split into multiple parameters |
|
1226 | - $arguments = $this->_array_is_numerically_and_sequentially_indexed($arguments) |
|
1227 | - ? $arguments |
|
1228 | - : array($arguments); |
|
1229 | - // attempt to inject dependencies ? |
|
1230 | - if ($this->_dependency_map->has($class_name)) { |
|
1231 | - $arguments = $this->_resolve_dependencies($reflector, $class_name, $arguments); |
|
1232 | - } |
|
1233 | - // instantiate the class if possible |
|
1234 | - if ($reflector->isAbstract()) { |
|
1235 | - // nothing to instantiate, loading file was enough |
|
1236 | - // does not throw an exception so $instantiation_mode is unused |
|
1237 | - // $instantiation_mode = "1) no constructor abstract class"; |
|
1238 | - return true; |
|
1239 | - } |
|
1240 | - if (empty($arguments) |
|
1241 | - && $this->mirror->getConstructorFromReflection($reflector) === null |
|
1242 | - && $reflector->isInstantiable() |
|
1243 | - ) { |
|
1244 | - // no constructor = static methods only... nothing to instantiate, loading file was enough |
|
1245 | - // $instantiation_mode = "2) no constructor but instantiable"; |
|
1246 | - return $reflector->newInstance(); |
|
1247 | - } |
|
1248 | - if ($from_db && method_exists($class_name, 'new_instance_from_db')) { |
|
1249 | - // $instantiation_mode = "3) new_instance_from_db()"; |
|
1250 | - return call_user_func_array(array($class_name, 'new_instance_from_db'), $arguments); |
|
1251 | - } |
|
1252 | - if (method_exists($class_name, 'new_instance')) { |
|
1253 | - // $instantiation_mode = "4) new_instance()"; |
|
1254 | - return call_user_func_array(array($class_name, 'new_instance'), $arguments); |
|
1255 | - } |
|
1256 | - if (method_exists($class_name, 'instance')) { |
|
1257 | - // $instantiation_mode = "5) instance()"; |
|
1258 | - return call_user_func_array(array($class_name, 'instance'), $arguments); |
|
1259 | - } |
|
1260 | - if ($reflector->isInstantiable()) { |
|
1261 | - // $instantiation_mode = "6) constructor"; |
|
1262 | - return $reflector->newInstanceArgs($arguments); |
|
1263 | - } |
|
1264 | - // heh ? something's not right ! |
|
1265 | - throw new EE_Error( |
|
1266 | - sprintf( |
|
1267 | - __('The %s file %s could not be instantiated.', 'event_espresso'), |
|
1268 | - $type, |
|
1269 | - $class_name |
|
1270 | - ) |
|
1271 | - ); |
|
1272 | - } |
|
1273 | - |
|
1274 | - |
|
1275 | - /** |
|
1276 | - * @see http://stackoverflow.com/questions/173400/how-to-check-if-php-array-is-associative-or-sequential |
|
1277 | - * @param array $array |
|
1278 | - * @return bool |
|
1279 | - */ |
|
1280 | - protected function _array_is_numerically_and_sequentially_indexed(array $array) |
|
1281 | - { |
|
1282 | - return ! empty($array) |
|
1283 | - ? array_keys($array) === range(0, count($array) - 1) |
|
1284 | - : true; |
|
1285 | - } |
|
1286 | - |
|
1287 | - |
|
1288 | - /** |
|
1289 | - * _resolve_dependencies |
|
1290 | - * examines the constructor for the requested class to determine |
|
1291 | - * if any dependencies exist, and if they can be injected. |
|
1292 | - * If so, then those classes will be added to the array of arguments passed to the constructor |
|
1293 | - * PLZ NOTE: this is achieved by type hinting the constructor params |
|
1294 | - * For example: |
|
1295 | - * if attempting to load a class "Foo" with the following constructor: |
|
1296 | - * __construct( Bar $bar_class, Fighter $grohl_class ) |
|
1297 | - * then $bar_class and $grohl_class will be added to the $arguments array, |
|
1298 | - * but only IF they are NOT already present in the incoming arguments array, |
|
1299 | - * and the correct classes can be loaded |
|
1300 | - * |
|
1301 | - * @param ReflectionClass $reflector |
|
1302 | - * @param string $class_name |
|
1303 | - * @param array $arguments |
|
1304 | - * @return array |
|
1305 | - * @throws InvalidArgumentException |
|
1306 | - * @throws InvalidDataTypeException |
|
1307 | - * @throws InvalidInterfaceException |
|
1308 | - * @throws ReflectionException |
|
1309 | - */ |
|
1310 | - protected function _resolve_dependencies(ReflectionClass $reflector, $class_name, array $arguments = array()) |
|
1311 | - { |
|
1312 | - // let's examine the constructor |
|
1313 | - $constructor = $this->mirror->getConstructorFromReflection($reflector); |
|
1314 | - // whu? huh? nothing? |
|
1315 | - if (! $constructor) { |
|
1316 | - return $arguments; |
|
1317 | - } |
|
1318 | - // get constructor parameters |
|
1319 | - $params = $this->mirror->getParametersFromReflection($reflector); |
|
1320 | - // and the keys for the incoming arguments array so that we can compare existing arguments with what is expected |
|
1321 | - $argument_keys = array_keys($arguments); |
|
1322 | - // now loop thru all of the constructors expected parameters |
|
1323 | - foreach ($params as $index => $param) { |
|
1324 | - try { |
|
1325 | - // is this a dependency for a specific class ? |
|
1326 | - $param_class = $this->mirror->getParameterClassName($param, $class_name, $index); |
|
1327 | - } catch (ReflectionException $exception) { |
|
1328 | - // uh-oh... most likely a legacy class that has not been autoloaded |
|
1329 | - // let's try to derive the classname from what we have now |
|
1330 | - // and hope that the property var name is close to the class name |
|
1331 | - $param_class = $param->getName(); |
|
1332 | - $param_class = str_replace('_', ' ', $param_class); |
|
1333 | - $param_class = ucwords($param_class); |
|
1334 | - $param_class = str_replace(' ', '_', $param_class); |
|
1335 | - } |
|
1336 | - // BUT WAIT !!! This class may be an alias for something else (or getting replaced at runtime) |
|
1337 | - $param_class = $this->class_cache->isAlias($param_class, $class_name) |
|
1338 | - ? $this->class_cache->getFqnForAlias($param_class, $class_name) |
|
1339 | - : $param_class; |
|
1340 | - if (// param is not even a class |
|
1341 | - $param_class === null |
|
1342 | - // and something already exists in the incoming arguments for this param |
|
1343 | - && array_key_exists($index, $argument_keys) |
|
1344 | - && array_key_exists($argument_keys[ $index ], $arguments) |
|
1345 | - ) { |
|
1346 | - // so let's skip this argument and move on to the next |
|
1347 | - continue; |
|
1348 | - } |
|
1349 | - if (// parameter is type hinted as a class, exists as an incoming argument, AND it's the correct class |
|
1350 | - $param_class !== null |
|
1351 | - && isset($argument_keys[ $index ], $arguments[ $argument_keys[ $index ] ]) |
|
1352 | - && $arguments[ $argument_keys[ $index ] ] instanceof $param_class |
|
1353 | - ) { |
|
1354 | - // skip this argument and move on to the next |
|
1355 | - continue; |
|
1356 | - } |
|
1357 | - if (// parameter is type hinted as a class, and should be injected |
|
1358 | - $param_class !== null |
|
1359 | - && $this->_dependency_map->has_dependency_for_class($class_name, $param_class) |
|
1360 | - ) { |
|
1361 | - $arguments = $this->_resolve_dependency( |
|
1362 | - $class_name, |
|
1363 | - $param_class, |
|
1364 | - $arguments, |
|
1365 | - $index |
|
1366 | - ); |
|
1367 | - } |
|
1368 | - if (empty($arguments[ $index ])) { |
|
1369 | - $arguments[ $index ] = $this->mirror->getParameterDefaultValue( |
|
1370 | - $param, |
|
1371 | - $class_name, |
|
1372 | - $index |
|
1373 | - ); |
|
1374 | - } |
|
1375 | - } |
|
1376 | - return $arguments; |
|
1377 | - } |
|
1378 | - |
|
1379 | - |
|
1380 | - /** |
|
1381 | - * @param string $class_name |
|
1382 | - * @param string $param_class |
|
1383 | - * @param array $arguments |
|
1384 | - * @param mixed $index |
|
1385 | - * @return array |
|
1386 | - * @throws InvalidArgumentException |
|
1387 | - * @throws InvalidInterfaceException |
|
1388 | - * @throws InvalidDataTypeException |
|
1389 | - */ |
|
1390 | - protected function _resolve_dependency($class_name, $param_class, $arguments, $index) |
|
1391 | - { |
|
1392 | - $dependency = null; |
|
1393 | - // should dependency be loaded from cache ? |
|
1394 | - $cache_on = $this->_dependency_map->loading_strategy_for_class_dependency( |
|
1395 | - $class_name, |
|
1396 | - $param_class |
|
1397 | - ); |
|
1398 | - $cache_on = $cache_on !== EE_Dependency_Map::load_new_object; |
|
1399 | - // we might have a dependency... |
|
1400 | - // let's MAYBE try and find it in our cache if that's what's been requested |
|
1401 | - $cached_class = $cache_on |
|
1402 | - ? $this->_get_cached_class($param_class) |
|
1403 | - : null; |
|
1404 | - // and grab it if it exists |
|
1405 | - if ($cached_class instanceof $param_class) { |
|
1406 | - $dependency = $cached_class; |
|
1407 | - } elseif ($param_class !== $class_name) { |
|
1408 | - // obtain the loader method from the dependency map |
|
1409 | - $loader = $this->_dependency_map->class_loader($param_class); |
|
1410 | - // is loader a custom closure ? |
|
1411 | - if ($loader instanceof Closure) { |
|
1412 | - $dependency = $loader($arguments); |
|
1413 | - } else { |
|
1414 | - // set the cache on property for the recursive loading call |
|
1415 | - $this->_cache_on = $cache_on; |
|
1416 | - // if not, then let's try and load it via the registry |
|
1417 | - if ($loader && method_exists($this, $loader)) { |
|
1418 | - $dependency = $this->{$loader}($param_class); |
|
1419 | - } else { |
|
1420 | - $dependency = LoaderFactory::getLoader()->load( |
|
1421 | - $param_class, |
|
1422 | - array(), |
|
1423 | - $cache_on |
|
1424 | - ); |
|
1425 | - } |
|
1426 | - } |
|
1427 | - } |
|
1428 | - // did we successfully find the correct dependency ? |
|
1429 | - if ($dependency instanceof $param_class) { |
|
1430 | - // then let's inject it into the incoming array of arguments at the correct location |
|
1431 | - $arguments[ $index ] = $dependency; |
|
1432 | - } |
|
1433 | - return $arguments; |
|
1434 | - } |
|
1435 | - |
|
1436 | - |
|
1437 | - /** |
|
1438 | - * call any loader that's been registered in the EE_Dependency_Map::$_class_loaders array |
|
1439 | - * |
|
1440 | - * @param string $classname PLEASE NOTE: the class name needs to match what's registered |
|
1441 | - * in the EE_Dependency_Map::$_class_loaders array, |
|
1442 | - * including the class prefix, ie: "EE_", "EEM_", "EEH_", etc |
|
1443 | - * @param array $arguments |
|
1444 | - * @return object |
|
1445 | - */ |
|
1446 | - public static function factory($classname, $arguments = array()) |
|
1447 | - { |
|
1448 | - $loader = self::instance()->_dependency_map->class_loader($classname); |
|
1449 | - if ($loader instanceof Closure) { |
|
1450 | - return $loader($arguments); |
|
1451 | - } |
|
1452 | - if (method_exists(self::instance(), $loader)) { |
|
1453 | - return self::instance()->{$loader}($classname, $arguments); |
|
1454 | - } |
|
1455 | - return null; |
|
1456 | - } |
|
1457 | - |
|
1458 | - |
|
1459 | - /** |
|
1460 | - * Gets the addon by its class name |
|
1461 | - * |
|
1462 | - * @param string $class_name |
|
1463 | - * @return EE_Addon |
|
1464 | - */ |
|
1465 | - public function getAddon($class_name) |
|
1466 | - { |
|
1467 | - $class_name = str_replace('\\', '_', $class_name); |
|
1468 | - if (isset($this->addons->{$class_name})) { |
|
1469 | - return $this->addons->{$class_name}; |
|
1470 | - } else { |
|
1471 | - return null; |
|
1472 | - } |
|
1473 | - } |
|
1474 | - |
|
1475 | - |
|
1476 | - /** |
|
1477 | - * removes the addon from the internal cache |
|
1478 | - * |
|
1479 | - * @param string $class_name |
|
1480 | - * @return void |
|
1481 | - */ |
|
1482 | - public function removeAddon($class_name) |
|
1483 | - { |
|
1484 | - $class_name = str_replace('\\', '_', $class_name); |
|
1485 | - unset($this->addons->{$class_name}); |
|
1486 | - } |
|
1487 | - |
|
1488 | - |
|
1489 | - /** |
|
1490 | - * Gets the addon by its name/slug (not classname. For that, just |
|
1491 | - * use the get_addon() method above |
|
1492 | - * |
|
1493 | - * @param string $name |
|
1494 | - * @return EE_Addon |
|
1495 | - */ |
|
1496 | - public function get_addon_by_name($name) |
|
1497 | - { |
|
1498 | - foreach ($this->addons as $addon) { |
|
1499 | - if ($addon->name() === $name) { |
|
1500 | - return $addon; |
|
1501 | - } |
|
1502 | - } |
|
1503 | - return null; |
|
1504 | - } |
|
1505 | - |
|
1506 | - |
|
1507 | - /** |
|
1508 | - * Gets an array of all the registered addons, where the keys are their names. |
|
1509 | - * (ie, what each returns for their name() function) |
|
1510 | - * They're already available on EE_Registry::instance()->addons as properties, |
|
1511 | - * where each property's name is the addon's classname, |
|
1512 | - * So if you just want to get the addon by classname, |
|
1513 | - * OR use the get_addon() method above. |
|
1514 | - * PLEASE NOTE: |
|
1515 | - * addons with Fully Qualified Class Names |
|
1516 | - * have had the namespace separators converted to underscores, |
|
1517 | - * so a classname like Fully\Qualified\ClassName |
|
1518 | - * would have been converted to Fully_Qualified_ClassName |
|
1519 | - * |
|
1520 | - * @return EE_Addon[] where the KEYS are the addon's name() |
|
1521 | - */ |
|
1522 | - public function get_addons_by_name() |
|
1523 | - { |
|
1524 | - $addons = array(); |
|
1525 | - foreach ($this->addons as $addon) { |
|
1526 | - $addons[ $addon->name() ] = $addon; |
|
1527 | - } |
|
1528 | - return $addons; |
|
1529 | - } |
|
1530 | - |
|
1531 | - |
|
1532 | - /** |
|
1533 | - * Resets the specified model's instance AND makes sure EE_Registry doesn't keep |
|
1534 | - * a stale copy of it around |
|
1535 | - * |
|
1536 | - * @param string $model_name |
|
1537 | - * @return \EEM_Base |
|
1538 | - * @throws \EE_Error |
|
1539 | - */ |
|
1540 | - public function reset_model($model_name) |
|
1541 | - { |
|
1542 | - $model_class_name = strpos($model_name, 'EEM_') !== 0 |
|
1543 | - ? "EEM_{$model_name}" |
|
1544 | - : $model_name; |
|
1545 | - if (! isset($this->LIB->{$model_class_name}) || ! $this->LIB->{$model_class_name} instanceof EEM_Base) { |
|
1546 | - return null; |
|
1547 | - } |
|
1548 | - // get that model reset it and make sure we nuke the old reference to it |
|
1549 | - if ($this->LIB->{$model_class_name} instanceof $model_class_name |
|
1550 | - && is_callable( |
|
1551 | - array($model_class_name, 'reset') |
|
1552 | - )) { |
|
1553 | - $this->LIB->{$model_class_name} = $this->LIB->{$model_class_name}->reset(); |
|
1554 | - } else { |
|
1555 | - throw new EE_Error( |
|
1556 | - sprintf( |
|
1557 | - esc_html__('Model %s does not have a method "reset"', 'event_espresso'), |
|
1558 | - $model_name |
|
1559 | - ) |
|
1560 | - ); |
|
1561 | - } |
|
1562 | - return $this->LIB->{$model_class_name}; |
|
1563 | - } |
|
1564 | - |
|
1565 | - |
|
1566 | - /** |
|
1567 | - * Resets the registry. |
|
1568 | - * The criteria for what gets reset is based on what can be shared between sites on the same request when |
|
1569 | - * switch_to_blog is used in a multisite install. Here is a list of things that are NOT reset. |
|
1570 | - * - $_dependency_map |
|
1571 | - * - $_class_abbreviations |
|
1572 | - * - $NET_CFG (EE_Network_Config): The config is shared network wide so no need to reset. |
|
1573 | - * - $REQ: Still on the same request so no need to change. |
|
1574 | - * - $CAP: There is no site specific state in the EE_Capability class. |
|
1575 | - * - $SSN: Although ideally, the session should not be shared between site switches, we can't reset it because only |
|
1576 | - * one Session can be active in a single request. Resetting could resolve in "headers already sent" errors. |
|
1577 | - * - $addons: In multisite, the state of the addons is something controlled via hooks etc in a normal request. So |
|
1578 | - * for now, we won't reset the addons because it could break calls to an add-ons class/methods in the |
|
1579 | - * switch or on the restore. |
|
1580 | - * - $modules |
|
1581 | - * - $shortcodes |
|
1582 | - * - $widgets |
|
1583 | - * |
|
1584 | - * @param boolean $hard [deprecated] |
|
1585 | - * @param boolean $reinstantiate whether to create new instances of EE_Registry's singletons too, |
|
1586 | - * or just reset without re-instantiating (handy to set to FALSE if you're not |
|
1587 | - * sure if you CAN currently reinstantiate the singletons at the moment) |
|
1588 | - * @param bool $reset_models Defaults to true. When false, then the models are not reset. This is so |
|
1589 | - * client |
|
1590 | - * code instead can just change the model context to a different blog id if |
|
1591 | - * necessary |
|
1592 | - * @return EE_Registry |
|
1593 | - * @throws InvalidInterfaceException |
|
1594 | - * @throws InvalidDataTypeException |
|
1595 | - * @throws EE_Error |
|
1596 | - * @throws ReflectionException |
|
1597 | - * @throws InvalidArgumentException |
|
1598 | - */ |
|
1599 | - public static function reset($hard = false, $reinstantiate = true, $reset_models = true) |
|
1600 | - { |
|
1601 | - $instance = self::instance(); |
|
1602 | - $instance->_cache_on = true; |
|
1603 | - // reset some "special" classes |
|
1604 | - EEH_Activation::reset(); |
|
1605 | - $hard = apply_filters('FHEE__EE_Registry__reset__hard', $hard); |
|
1606 | - $instance->CFG = EE_Config::reset($hard, $reinstantiate); |
|
1607 | - $instance->CART = null; |
|
1608 | - $instance->MRM = null; |
|
1609 | - $instance->AssetsRegistry = LoaderFactory::getLoader()->getShared( |
|
1610 | - 'EventEspresso\core\services\assets\Registry' |
|
1611 | - ); |
|
1612 | - // messages reset |
|
1613 | - EED_Messages::reset(); |
|
1614 | - // handle of objects cached on LIB |
|
1615 | - foreach (array('LIB', 'modules') as $cache) { |
|
1616 | - foreach ($instance->{$cache} as $class_name => $class) { |
|
1617 | - if (self::_reset_and_unset_object($class, $reset_models)) { |
|
1618 | - unset($instance->{$cache}->{$class_name}); |
|
1619 | - } |
|
1620 | - } |
|
1621 | - } |
|
1622 | - return $instance; |
|
1623 | - } |
|
1624 | - |
|
1625 | - |
|
1626 | - /** |
|
1627 | - * if passed object implements ResettableInterface, then call it's reset() method |
|
1628 | - * if passed object implements InterminableInterface, then return false, |
|
1629 | - * to indicate that it should NOT be cleared from the Registry cache |
|
1630 | - * |
|
1631 | - * @param $object |
|
1632 | - * @param bool $reset_models |
|
1633 | - * @return bool returns true if cached object should be unset |
|
1634 | - */ |
|
1635 | - private static function _reset_and_unset_object($object, $reset_models) |
|
1636 | - { |
|
1637 | - if (! is_object($object)) { |
|
1638 | - // don't unset anything that's not an object |
|
1639 | - return false; |
|
1640 | - } |
|
1641 | - if ($object instanceof EED_Module) { |
|
1642 | - $object::reset(); |
|
1643 | - // don't unset modules |
|
1644 | - return false; |
|
1645 | - } |
|
1646 | - if ($object instanceof ResettableInterface) { |
|
1647 | - if ($object instanceof EEM_Base) { |
|
1648 | - if ($reset_models) { |
|
1649 | - $object->reset(); |
|
1650 | - return true; |
|
1651 | - } |
|
1652 | - return false; |
|
1653 | - } |
|
1654 | - $object->reset(); |
|
1655 | - return true; |
|
1656 | - } |
|
1657 | - if (! $object instanceof InterminableInterface) { |
|
1658 | - return true; |
|
1659 | - } |
|
1660 | - return false; |
|
1661 | - } |
|
1662 | - |
|
1663 | - |
|
1664 | - /** |
|
1665 | - * Gets all the custom post type models defined |
|
1666 | - * |
|
1667 | - * @return array keys are model "short names" (Eg "Event") and keys are classnames (eg "EEM_Event") |
|
1668 | - */ |
|
1669 | - public function cpt_models() |
|
1670 | - { |
|
1671 | - $cpt_models = array(); |
|
1672 | - foreach ($this->non_abstract_db_models as $short_name => $classname) { |
|
1673 | - if (is_subclass_of($classname, 'EEM_CPT_Base')) { |
|
1674 | - $cpt_models[ $short_name ] = $classname; |
|
1675 | - } |
|
1676 | - } |
|
1677 | - return $cpt_models; |
|
1678 | - } |
|
1679 | - |
|
1680 | - |
|
1681 | - /** |
|
1682 | - * @return \EE_Config |
|
1683 | - */ |
|
1684 | - public static function CFG() |
|
1685 | - { |
|
1686 | - return self::instance()->CFG; |
|
1687 | - } |
|
1688 | - |
|
1689 | - |
|
1690 | - /** |
|
1691 | - * @deprecated 4.9.62.p |
|
1692 | - * @param string $class_name |
|
1693 | - * @return ReflectionClass |
|
1694 | - * @throws ReflectionException |
|
1695 | - * @throws InvalidDataTypeException |
|
1696 | - */ |
|
1697 | - public function get_ReflectionClass($class_name) |
|
1698 | - { |
|
1699 | - return $this->mirror->getReflectionClass($class_name); |
|
1700 | - } |
|
26 | + /** |
|
27 | + * @var EE_Registry $_instance |
|
28 | + */ |
|
29 | + private static $_instance; |
|
30 | + |
|
31 | + /** |
|
32 | + * @var EE_Dependency_Map $_dependency_map |
|
33 | + */ |
|
34 | + protected $_dependency_map; |
|
35 | + |
|
36 | + /** |
|
37 | + * @var Mirror |
|
38 | + */ |
|
39 | + private $mirror; |
|
40 | + |
|
41 | + /** |
|
42 | + * @var ClassInterfaceCache $class_cache |
|
43 | + */ |
|
44 | + private $class_cache; |
|
45 | + |
|
46 | + /** |
|
47 | + * @var array $_class_abbreviations |
|
48 | + */ |
|
49 | + protected $_class_abbreviations = array(); |
|
50 | + |
|
51 | + /** |
|
52 | + * @var CommandBusInterface $BUS |
|
53 | + */ |
|
54 | + public $BUS; |
|
55 | + |
|
56 | + /** |
|
57 | + * @var EE_Cart $CART |
|
58 | + */ |
|
59 | + public $CART; |
|
60 | + |
|
61 | + /** |
|
62 | + * @var EE_Config $CFG |
|
63 | + */ |
|
64 | + public $CFG; |
|
65 | + |
|
66 | + /** |
|
67 | + * @var EE_Network_Config $NET_CFG |
|
68 | + */ |
|
69 | + public $NET_CFG; |
|
70 | + |
|
71 | + /** |
|
72 | + * StdClass object for storing library classes in |
|
73 | + * |
|
74 | + * @var RegistryContainer $LIB |
|
75 | + */ |
|
76 | + public $LIB; |
|
77 | + |
|
78 | + /** |
|
79 | + * @var EE_Request_Handler $REQ |
|
80 | + */ |
|
81 | + public $REQ; |
|
82 | + |
|
83 | + /** |
|
84 | + * @var EE_Session $SSN |
|
85 | + */ |
|
86 | + public $SSN; |
|
87 | + |
|
88 | + /** |
|
89 | + * @since 4.5.0 |
|
90 | + * @var EE_Capabilities $CAP |
|
91 | + */ |
|
92 | + public $CAP; |
|
93 | + |
|
94 | + /** |
|
95 | + * @since 4.9.0 |
|
96 | + * @var EE_Message_Resource_Manager $MRM |
|
97 | + */ |
|
98 | + public $MRM; |
|
99 | + |
|
100 | + /** |
|
101 | + * @var Registry $AssetsRegistry |
|
102 | + */ |
|
103 | + public $AssetsRegistry; |
|
104 | + |
|
105 | + /** |
|
106 | + * StdClass object for holding addons which have registered themselves to work with EE core |
|
107 | + * |
|
108 | + * @var EE_Addon[] $addons |
|
109 | + */ |
|
110 | + public $addons; |
|
111 | + |
|
112 | + /** |
|
113 | + * keys are 'short names' (eg Event), values are class names (eg 'EEM_Event') |
|
114 | + * |
|
115 | + * @var EEM_Base[] $models |
|
116 | + */ |
|
117 | + public $models = array(); |
|
118 | + |
|
119 | + /** |
|
120 | + * @var EED_Module[] $modules |
|
121 | + */ |
|
122 | + public $modules; |
|
123 | + |
|
124 | + /** |
|
125 | + * @var EES_Shortcode[] $shortcodes |
|
126 | + */ |
|
127 | + public $shortcodes; |
|
128 | + |
|
129 | + /** |
|
130 | + * @var WP_Widget[] $widgets |
|
131 | + */ |
|
132 | + public $widgets; |
|
133 | + |
|
134 | + /** |
|
135 | + * this is an array of all implemented model names (i.e. not the parent abstract models, or models |
|
136 | + * which don't actually fetch items from the DB in the normal way (ie, are not children of EEM_Base)). |
|
137 | + * Keys are model "short names" (eg "Event") as used in model relations, and values are |
|
138 | + * classnames (eg "EEM_Event") |
|
139 | + * |
|
140 | + * @var array $non_abstract_db_models |
|
141 | + */ |
|
142 | + public $non_abstract_db_models = array(); |
|
143 | + |
|
144 | + /** |
|
145 | + * internationalization for JS strings |
|
146 | + * usage: EE_Registry::i18n_js_strings['string_key'] = esc_html__( 'string to translate.', 'event_espresso' ); |
|
147 | + * in js file: var translatedString = eei18n.string_key; |
|
148 | + * |
|
149 | + * @var array $i18n_js_strings |
|
150 | + */ |
|
151 | + public static $i18n_js_strings = array(); |
|
152 | + |
|
153 | + /** |
|
154 | + * $main_file - path to espresso.php |
|
155 | + * |
|
156 | + * @var array $main_file |
|
157 | + */ |
|
158 | + public $main_file; |
|
159 | + |
|
160 | + /** |
|
161 | + * array of ReflectionClass objects where the key is the class name |
|
162 | + * |
|
163 | + * @deprecated 4.9.62.p |
|
164 | + * @var ReflectionClass[] $_reflectors |
|
165 | + */ |
|
166 | + public $_reflectors; |
|
167 | + |
|
168 | + /** |
|
169 | + * boolean flag to indicate whether or not to load/save dependencies from/to the cache |
|
170 | + * |
|
171 | + * @var boolean $_cache_on |
|
172 | + */ |
|
173 | + protected $_cache_on = true; |
|
174 | + |
|
175 | + /** |
|
176 | + * @var ObjectIdentifier |
|
177 | + */ |
|
178 | + private $object_identifier; |
|
179 | + |
|
180 | + |
|
181 | + /** |
|
182 | + * @singleton method used to instantiate class object |
|
183 | + * @param EE_Dependency_Map|null $dependency_map |
|
184 | + * @param Mirror|null $mirror |
|
185 | + * @param ClassInterfaceCache|null $class_cache |
|
186 | + * @param ObjectIdentifier|null $object_identifier |
|
187 | + * @return EE_Registry instance |
|
188 | + */ |
|
189 | + public static function instance( |
|
190 | + EE_Dependency_Map $dependency_map = null, |
|
191 | + Mirror $mirror = null, |
|
192 | + ClassInterfaceCache $class_cache = null, |
|
193 | + ObjectIdentifier $object_identifier = null |
|
194 | + ) { |
|
195 | + // check if class object is instantiated |
|
196 | + if (! self::$_instance instanceof EE_Registry |
|
197 | + && $dependency_map instanceof EE_Dependency_Map |
|
198 | + && $mirror instanceof Mirror |
|
199 | + && $class_cache instanceof ClassInterfaceCache |
|
200 | + && $object_identifier instanceof ObjectIdentifier |
|
201 | + ) { |
|
202 | + self::$_instance = new self( |
|
203 | + $dependency_map, |
|
204 | + $mirror, |
|
205 | + $class_cache, |
|
206 | + $object_identifier |
|
207 | + ); |
|
208 | + } |
|
209 | + return self::$_instance; |
|
210 | + } |
|
211 | + |
|
212 | + |
|
213 | + /** |
|
214 | + * protected constructor to prevent direct creation |
|
215 | + * |
|
216 | + * @Constructor |
|
217 | + * @param EE_Dependency_Map $dependency_map |
|
218 | + * @param Mirror $mirror |
|
219 | + * @param ClassInterfaceCache $class_cache |
|
220 | + * @param ObjectIdentifier $object_identifier |
|
221 | + */ |
|
222 | + protected function __construct( |
|
223 | + EE_Dependency_Map $dependency_map, |
|
224 | + Mirror $mirror, |
|
225 | + ClassInterfaceCache $class_cache, |
|
226 | + ObjectIdentifier $object_identifier |
|
227 | + ) { |
|
228 | + $this->_dependency_map = $dependency_map; |
|
229 | + $this->mirror = $mirror; |
|
230 | + $this->class_cache = $class_cache; |
|
231 | + $this->object_identifier = $object_identifier; |
|
232 | + // $registry_container = new RegistryContainer(); |
|
233 | + $this->LIB = new RegistryContainer(); |
|
234 | + $this->addons = new RegistryContainer(); |
|
235 | + $this->modules = new RegistryContainer(); |
|
236 | + $this->shortcodes = new RegistryContainer(); |
|
237 | + $this->widgets = new RegistryContainer(); |
|
238 | + add_action('EE_Load_Espresso_Core__handle_request__initialize_core_loading', array($this, 'initialize')); |
|
239 | + } |
|
240 | + |
|
241 | + |
|
242 | + /** |
|
243 | + * initialize |
|
244 | + * |
|
245 | + * @throws OutOfBoundsException |
|
246 | + * @throws InvalidArgumentException |
|
247 | + * @throws InvalidInterfaceException |
|
248 | + * @throws InvalidDataTypeException |
|
249 | + * @throws EE_Error |
|
250 | + * @throws ReflectionException |
|
251 | + */ |
|
252 | + public function initialize() |
|
253 | + { |
|
254 | + $this->_class_abbreviations = apply_filters( |
|
255 | + 'FHEE__EE_Registry____construct___class_abbreviations', |
|
256 | + array( |
|
257 | + 'EE_Config' => 'CFG', |
|
258 | + 'EE_Session' => 'SSN', |
|
259 | + 'EE_Capabilities' => 'CAP', |
|
260 | + 'EE_Cart' => 'CART', |
|
261 | + 'EE_Network_Config' => 'NET_CFG', |
|
262 | + 'EE_Request_Handler' => 'REQ', |
|
263 | + 'EE_Message_Resource_Manager' => 'MRM', |
|
264 | + 'EventEspresso\core\services\commands\CommandBus' => 'BUS', |
|
265 | + 'EventEspresso\core\services\assets\Registry' => 'AssetsRegistry', |
|
266 | + ) |
|
267 | + ); |
|
268 | + $this->load_core('Base', array(), true); |
|
269 | + // add our request and response objects to the cache |
|
270 | + $request_loader = $this->_dependency_map->class_loader( |
|
271 | + 'EventEspresso\core\services\request\Request' |
|
272 | + ); |
|
273 | + $this->_set_cached_class( |
|
274 | + $request_loader(), |
|
275 | + 'EventEspresso\core\services\request\Request' |
|
276 | + ); |
|
277 | + $response_loader = $this->_dependency_map->class_loader( |
|
278 | + 'EventEspresso\core\services\request\Response' |
|
279 | + ); |
|
280 | + $this->_set_cached_class( |
|
281 | + $response_loader(), |
|
282 | + 'EventEspresso\core\services\request\Response' |
|
283 | + ); |
|
284 | + add_action('AHEE__EE_System__set_hooks_for_core', array($this, 'init')); |
|
285 | + } |
|
286 | + |
|
287 | + |
|
288 | + /** |
|
289 | + * @return void |
|
290 | + */ |
|
291 | + public function init() |
|
292 | + { |
|
293 | + // Get current page protocol |
|
294 | + $protocol = isset($_SERVER['HTTPS']) ? 'https://' : 'http://'; |
|
295 | + // Output admin-ajax.php URL with same protocol as current page |
|
296 | + self::$i18n_js_strings['ajax_url'] = admin_url('admin-ajax.php', $protocol); |
|
297 | + self::$i18n_js_strings['wp_debug'] = defined('WP_DEBUG') ? WP_DEBUG : false; |
|
298 | + } |
|
299 | + |
|
300 | + |
|
301 | + /** |
|
302 | + * localize_i18n_js_strings |
|
303 | + * |
|
304 | + * @return string |
|
305 | + */ |
|
306 | + public static function localize_i18n_js_strings() |
|
307 | + { |
|
308 | + $i18n_js_strings = (array) self::$i18n_js_strings; |
|
309 | + foreach ($i18n_js_strings as $key => $value) { |
|
310 | + if (is_scalar($value)) { |
|
311 | + $i18n_js_strings[ $key ] = html_entity_decode((string) $value, ENT_QUOTES, 'UTF-8'); |
|
312 | + } |
|
313 | + } |
|
314 | + return '/* <![CDATA[ */ var eei18n = ' . wp_json_encode($i18n_js_strings) . '; /* ]]> */'; |
|
315 | + } |
|
316 | + |
|
317 | + |
|
318 | + /** |
|
319 | + * @param mixed string | EED_Module $module |
|
320 | + * @throws OutOfBoundsException |
|
321 | + * @throws InvalidArgumentException |
|
322 | + * @throws InvalidInterfaceException |
|
323 | + * @throws InvalidDataTypeException |
|
324 | + * @throws EE_Error |
|
325 | + * @throws ReflectionException |
|
326 | + */ |
|
327 | + public function add_module($module) |
|
328 | + { |
|
329 | + if ($module instanceof EED_Module) { |
|
330 | + $module_class = get_class($module); |
|
331 | + $this->modules->{$module_class} = $module; |
|
332 | + } else { |
|
333 | + if (! class_exists('EE_Module_Request_Router', false)) { |
|
334 | + $this->load_core('Module_Request_Router'); |
|
335 | + } |
|
336 | + EE_Module_Request_Router::module_factory($module); |
|
337 | + } |
|
338 | + } |
|
339 | + |
|
340 | + |
|
341 | + /** |
|
342 | + * @param string $module_name |
|
343 | + * @return mixed EED_Module | NULL |
|
344 | + */ |
|
345 | + public function get_module($module_name = '') |
|
346 | + { |
|
347 | + return isset($this->modules->{$module_name}) |
|
348 | + ? $this->modules->{$module_name} |
|
349 | + : null; |
|
350 | + } |
|
351 | + |
|
352 | + |
|
353 | + /** |
|
354 | + * loads core classes - must be singletons |
|
355 | + * |
|
356 | + * @param string $class_name - simple class name ie: session |
|
357 | + * @param mixed $arguments |
|
358 | + * @param bool $load_only |
|
359 | + * @return mixed |
|
360 | + * @throws InvalidInterfaceException |
|
361 | + * @throws InvalidDataTypeException |
|
362 | + * @throws EE_Error |
|
363 | + * @throws ReflectionException |
|
364 | + * @throws InvalidArgumentException |
|
365 | + */ |
|
366 | + public function load_core($class_name, $arguments = array(), $load_only = false) |
|
367 | + { |
|
368 | + $core_paths = apply_filters( |
|
369 | + 'FHEE__EE_Registry__load_core__core_paths', |
|
370 | + array( |
|
371 | + EE_CORE, |
|
372 | + EE_ADMIN, |
|
373 | + EE_CPTS, |
|
374 | + EE_CORE . 'data_migration_scripts' . DS, |
|
375 | + EE_CORE . 'capabilities' . DS, |
|
376 | + EE_CORE . 'request_stack' . DS, |
|
377 | + EE_CORE . 'middleware' . DS, |
|
378 | + ) |
|
379 | + ); |
|
380 | + // retrieve instantiated class |
|
381 | + return $this->_load( |
|
382 | + $core_paths, |
|
383 | + 'EE_', |
|
384 | + $class_name, |
|
385 | + 'core', |
|
386 | + $arguments, |
|
387 | + false, |
|
388 | + true, |
|
389 | + $load_only |
|
390 | + ); |
|
391 | + } |
|
392 | + |
|
393 | + |
|
394 | + /** |
|
395 | + * loads service classes |
|
396 | + * |
|
397 | + * @param string $class_name - simple class name ie: session |
|
398 | + * @param mixed $arguments |
|
399 | + * @param bool $load_only |
|
400 | + * @return mixed |
|
401 | + * @throws InvalidInterfaceException |
|
402 | + * @throws InvalidDataTypeException |
|
403 | + * @throws EE_Error |
|
404 | + * @throws ReflectionException |
|
405 | + * @throws InvalidArgumentException |
|
406 | + */ |
|
407 | + public function load_service($class_name, $arguments = array(), $load_only = false) |
|
408 | + { |
|
409 | + $service_paths = apply_filters( |
|
410 | + 'FHEE__EE_Registry__load_service__service_paths', |
|
411 | + array( |
|
412 | + EE_CORE . 'services' . DS, |
|
413 | + ) |
|
414 | + ); |
|
415 | + // retrieve instantiated class |
|
416 | + return $this->_load( |
|
417 | + $service_paths, |
|
418 | + 'EE_', |
|
419 | + $class_name, |
|
420 | + 'class', |
|
421 | + $arguments, |
|
422 | + false, |
|
423 | + true, |
|
424 | + $load_only |
|
425 | + ); |
|
426 | + } |
|
427 | + |
|
428 | + |
|
429 | + /** |
|
430 | + * loads data_migration_scripts |
|
431 | + * |
|
432 | + * @param string $class_name - class name for the DMS ie: EE_DMS_Core_4_2_0 |
|
433 | + * @param mixed $arguments |
|
434 | + * @return EE_Data_Migration_Script_Base|mixed |
|
435 | + * @throws InvalidInterfaceException |
|
436 | + * @throws InvalidDataTypeException |
|
437 | + * @throws EE_Error |
|
438 | + * @throws ReflectionException |
|
439 | + * @throws InvalidArgumentException |
|
440 | + */ |
|
441 | + public function load_dms($class_name, $arguments = array()) |
|
442 | + { |
|
443 | + // retrieve instantiated class |
|
444 | + return $this->_load( |
|
445 | + EE_Data_Migration_Manager::instance()->get_data_migration_script_folders(), |
|
446 | + 'EE_DMS_', |
|
447 | + $class_name, |
|
448 | + 'dms', |
|
449 | + $arguments, |
|
450 | + false, |
|
451 | + false |
|
452 | + ); |
|
453 | + } |
|
454 | + |
|
455 | + |
|
456 | + /** |
|
457 | + * loads object creating classes - must be singletons |
|
458 | + * |
|
459 | + * @param string $class_name - simple class name ie: attendee |
|
460 | + * @param mixed $arguments - an array of arguments to pass to the class |
|
461 | + * @param bool $from_db - some classes are instantiated from the db and thus call a different method to |
|
462 | + * instantiate |
|
463 | + * @param bool $cache if you don't want the class to be stored in the internal cache (non-persistent) then |
|
464 | + * set this to FALSE (ie. when instantiating model objects from client in a loop) |
|
465 | + * @param bool $load_only whether or not to just load the file and NOT instantiate, or load AND instantiate |
|
466 | + * (default) |
|
467 | + * @return EE_Base_Class | bool |
|
468 | + * @throws InvalidInterfaceException |
|
469 | + * @throws InvalidDataTypeException |
|
470 | + * @throws EE_Error |
|
471 | + * @throws ReflectionException |
|
472 | + * @throws InvalidArgumentException |
|
473 | + */ |
|
474 | + public function load_class($class_name, $arguments = array(), $from_db = false, $cache = true, $load_only = false) |
|
475 | + { |
|
476 | + $paths = apply_filters( |
|
477 | + 'FHEE__EE_Registry__load_class__paths', |
|
478 | + array( |
|
479 | + EE_CORE, |
|
480 | + EE_CLASSES, |
|
481 | + EE_BUSINESS, |
|
482 | + ) |
|
483 | + ); |
|
484 | + // retrieve instantiated class |
|
485 | + return $this->_load( |
|
486 | + $paths, |
|
487 | + 'EE_', |
|
488 | + $class_name, |
|
489 | + 'class', |
|
490 | + $arguments, |
|
491 | + $from_db, |
|
492 | + $cache, |
|
493 | + $load_only |
|
494 | + ); |
|
495 | + } |
|
496 | + |
|
497 | + |
|
498 | + /** |
|
499 | + * loads helper classes - must be singletons |
|
500 | + * |
|
501 | + * @param string $class_name - simple class name ie: price |
|
502 | + * @param mixed $arguments |
|
503 | + * @param bool $load_only |
|
504 | + * @return EEH_Base | bool |
|
505 | + * @throws InvalidInterfaceException |
|
506 | + * @throws InvalidDataTypeException |
|
507 | + * @throws EE_Error |
|
508 | + * @throws ReflectionException |
|
509 | + * @throws InvalidArgumentException |
|
510 | + */ |
|
511 | + public function load_helper($class_name, $arguments = array(), $load_only = true) |
|
512 | + { |
|
513 | + // todo: add doing_it_wrong() in a few versions after all addons have had calls to this method removed |
|
514 | + $helper_paths = apply_filters('FHEE__EE_Registry__load_helper__helper_paths', array(EE_HELPERS)); |
|
515 | + // retrieve instantiated class |
|
516 | + return $this->_load( |
|
517 | + $helper_paths, |
|
518 | + 'EEH_', |
|
519 | + $class_name, |
|
520 | + 'helper', |
|
521 | + $arguments, |
|
522 | + false, |
|
523 | + true, |
|
524 | + $load_only |
|
525 | + ); |
|
526 | + } |
|
527 | + |
|
528 | + |
|
529 | + /** |
|
530 | + * loads core classes - must be singletons |
|
531 | + * |
|
532 | + * @param string $class_name - simple class name ie: session |
|
533 | + * @param mixed $arguments |
|
534 | + * @param bool $load_only |
|
535 | + * @param bool $cache whether to cache the object or not. |
|
536 | + * @return mixed |
|
537 | + * @throws InvalidInterfaceException |
|
538 | + * @throws InvalidDataTypeException |
|
539 | + * @throws EE_Error |
|
540 | + * @throws ReflectionException |
|
541 | + * @throws InvalidArgumentException |
|
542 | + */ |
|
543 | + public function load_lib($class_name, $arguments = array(), $load_only = false, $cache = true) |
|
544 | + { |
|
545 | + $paths = array( |
|
546 | + EE_LIBRARIES, |
|
547 | + EE_LIBRARIES . 'messages' . DS, |
|
548 | + EE_LIBRARIES . 'shortcodes' . DS, |
|
549 | + EE_LIBRARIES . 'qtips' . DS, |
|
550 | + EE_LIBRARIES . 'payment_methods' . DS, |
|
551 | + ); |
|
552 | + // retrieve instantiated class |
|
553 | + return $this->_load( |
|
554 | + $paths, |
|
555 | + 'EE_', |
|
556 | + $class_name, |
|
557 | + 'lib', |
|
558 | + $arguments, |
|
559 | + false, |
|
560 | + $cache, |
|
561 | + $load_only |
|
562 | + ); |
|
563 | + } |
|
564 | + |
|
565 | + |
|
566 | + /** |
|
567 | + * loads model classes - must be singletons |
|
568 | + * |
|
569 | + * @param string $class_name - simple class name ie: price |
|
570 | + * @param mixed $arguments |
|
571 | + * @param bool $load_only |
|
572 | + * @return EEM_Base | bool |
|
573 | + * @throws InvalidInterfaceException |
|
574 | + * @throws InvalidDataTypeException |
|
575 | + * @throws EE_Error |
|
576 | + * @throws ReflectionException |
|
577 | + * @throws InvalidArgumentException |
|
578 | + */ |
|
579 | + public function load_model($class_name, $arguments = array(), $load_only = false) |
|
580 | + { |
|
581 | + $paths = apply_filters( |
|
582 | + 'FHEE__EE_Registry__load_model__paths', |
|
583 | + array( |
|
584 | + EE_MODELS, |
|
585 | + EE_CORE, |
|
586 | + ) |
|
587 | + ); |
|
588 | + // retrieve instantiated class |
|
589 | + return $this->_load( |
|
590 | + $paths, |
|
591 | + 'EEM_', |
|
592 | + $class_name, |
|
593 | + 'model', |
|
594 | + $arguments, |
|
595 | + false, |
|
596 | + true, |
|
597 | + $load_only |
|
598 | + ); |
|
599 | + } |
|
600 | + |
|
601 | + |
|
602 | + /** |
|
603 | + * loads model classes - must be singletons |
|
604 | + * |
|
605 | + * @param string $class_name - simple class name ie: price |
|
606 | + * @param mixed $arguments |
|
607 | + * @param bool $load_only |
|
608 | + * @return mixed | bool |
|
609 | + * @throws InvalidInterfaceException |
|
610 | + * @throws InvalidDataTypeException |
|
611 | + * @throws EE_Error |
|
612 | + * @throws ReflectionException |
|
613 | + * @throws InvalidArgumentException |
|
614 | + */ |
|
615 | + public function load_model_class($class_name, $arguments = array(), $load_only = true) |
|
616 | + { |
|
617 | + $paths = array( |
|
618 | + EE_MODELS . 'fields' . DS, |
|
619 | + EE_MODELS . 'helpers' . DS, |
|
620 | + EE_MODELS . 'relations' . DS, |
|
621 | + EE_MODELS . 'strategies' . DS, |
|
622 | + ); |
|
623 | + // retrieve instantiated class |
|
624 | + return $this->_load( |
|
625 | + $paths, |
|
626 | + 'EE_', |
|
627 | + $class_name, |
|
628 | + '', |
|
629 | + $arguments, |
|
630 | + false, |
|
631 | + true, |
|
632 | + $load_only |
|
633 | + ); |
|
634 | + } |
|
635 | + |
|
636 | + |
|
637 | + /** |
|
638 | + * Determines if $model_name is the name of an actual EE model. |
|
639 | + * |
|
640 | + * @param string $model_name like Event, Attendee, Question_Group_Question, etc. |
|
641 | + * @return boolean |
|
642 | + */ |
|
643 | + public function is_model_name($model_name) |
|
644 | + { |
|
645 | + return isset($this->models[ $model_name ]); |
|
646 | + } |
|
647 | + |
|
648 | + |
|
649 | + /** |
|
650 | + * generic class loader |
|
651 | + * |
|
652 | + * @param string $path_to_file - directory path to file location, not including filename |
|
653 | + * @param string $file_name - file name ie: my_file.php, including extension |
|
654 | + * @param string $type - file type - core? class? helper? model? |
|
655 | + * @param mixed $arguments |
|
656 | + * @param bool $load_only |
|
657 | + * @return mixed |
|
658 | + * @throws InvalidInterfaceException |
|
659 | + * @throws InvalidDataTypeException |
|
660 | + * @throws EE_Error |
|
661 | + * @throws ReflectionException |
|
662 | + * @throws InvalidArgumentException |
|
663 | + */ |
|
664 | + public function load_file($path_to_file, $file_name, $type = '', $arguments = array(), $load_only = true) |
|
665 | + { |
|
666 | + // retrieve instantiated class |
|
667 | + return $this->_load( |
|
668 | + $path_to_file, |
|
669 | + '', |
|
670 | + $file_name, |
|
671 | + $type, |
|
672 | + $arguments, |
|
673 | + false, |
|
674 | + true, |
|
675 | + $load_only |
|
676 | + ); |
|
677 | + } |
|
678 | + |
|
679 | + |
|
680 | + /** |
|
681 | + * @param string $path_to_file - directory path to file location, not including filename |
|
682 | + * @param string $class_name - full class name ie: My_Class |
|
683 | + * @param string $type - file type - core? class? helper? model? |
|
684 | + * @param mixed $arguments |
|
685 | + * @param bool $load_only |
|
686 | + * @return bool|EE_Addon|object |
|
687 | + * @throws InvalidInterfaceException |
|
688 | + * @throws InvalidDataTypeException |
|
689 | + * @throws EE_Error |
|
690 | + * @throws ReflectionException |
|
691 | + * @throws InvalidArgumentException |
|
692 | + */ |
|
693 | + public function load_addon($path_to_file, $class_name, $type = 'class', $arguments = array(), $load_only = false) |
|
694 | + { |
|
695 | + // retrieve instantiated class |
|
696 | + return $this->_load( |
|
697 | + $path_to_file, |
|
698 | + 'addon', |
|
699 | + $class_name, |
|
700 | + $type, |
|
701 | + $arguments, |
|
702 | + false, |
|
703 | + true, |
|
704 | + $load_only |
|
705 | + ); |
|
706 | + } |
|
707 | + |
|
708 | + |
|
709 | + /** |
|
710 | + * instantiates, caches, and automatically resolves dependencies |
|
711 | + * for classes that use a Fully Qualified Class Name. |
|
712 | + * if the class is not capable of being loaded using PSR-4 autoloading, |
|
713 | + * then you need to use one of the existing load_*() methods |
|
714 | + * which can resolve the classname and filepath from the passed arguments |
|
715 | + * |
|
716 | + * @param bool|string $class_name Fully Qualified Class Name |
|
717 | + * @param array $arguments an argument, or array of arguments to pass to the class upon instantiation |
|
718 | + * @param bool $cache whether to cache the instantiated object for reuse |
|
719 | + * @param bool $from_db some classes are instantiated from the db |
|
720 | + * and thus call a different method to instantiate |
|
721 | + * @param bool $load_only if true, will only load the file, but will NOT instantiate an object |
|
722 | + * @param bool|string $addon if true, will cache the object in the EE_Registry->$addons array |
|
723 | + * @return bool|null|mixed null = failure to load or instantiate class object. |
|
724 | + * object = class loaded and instantiated successfully. |
|
725 | + * bool = fail or success when $load_only is true |
|
726 | + * @throws InvalidInterfaceException |
|
727 | + * @throws InvalidDataTypeException |
|
728 | + * @throws EE_Error |
|
729 | + * @throws ReflectionException |
|
730 | + * @throws InvalidArgumentException |
|
731 | + */ |
|
732 | + public function create( |
|
733 | + $class_name = false, |
|
734 | + $arguments = array(), |
|
735 | + $cache = false, |
|
736 | + $from_db = false, |
|
737 | + $load_only = false, |
|
738 | + $addon = false |
|
739 | + ) { |
|
740 | + $class_name = ltrim($class_name, '\\'); |
|
741 | + $class_name = $this->class_cache->getFqnForAlias($class_name); |
|
742 | + $class_exists = $this->loadOrVerifyClassExists($class_name, $arguments); |
|
743 | + // if a non-FQCN was passed, then |
|
744 | + // verifyClassExists() might return an object |
|
745 | + // or it could return null if the class just could not be found anywhere |
|
746 | + if ($class_exists instanceof $class_name || $class_exists === null) { |
|
747 | + // either way, return the results |
|
748 | + return $class_exists; |
|
749 | + } |
|
750 | + $class_name = $class_exists; |
|
751 | + // if we're only loading the class and it already exists, then let's just return true immediately |
|
752 | + if ($load_only) { |
|
753 | + return true; |
|
754 | + } |
|
755 | + $addon = $addon ? 'addon' : ''; |
|
756 | + // $this->_cache_on is toggled during the recursive loading that can occur with dependency injection |
|
757 | + // $cache is controlled by individual calls to separate Registry loader methods like load_class() |
|
758 | + // $load_only is also controlled by individual calls to separate Registry loader methods like load_file() |
|
759 | + if ($this->_cache_on && $cache && ! $load_only) { |
|
760 | + // return object if it's already cached |
|
761 | + $cached_class = $this->_get_cached_class($class_name, $addon, $arguments); |
|
762 | + if ($cached_class !== null) { |
|
763 | + return $cached_class; |
|
764 | + } |
|
765 | + }// obtain the loader method from the dependency map |
|
766 | + $loader = $this->_dependency_map->class_loader($class_name);// instantiate the requested object |
|
767 | + if ($loader instanceof Closure) { |
|
768 | + $class_obj = $loader($arguments); |
|
769 | + } else { |
|
770 | + if ($loader && method_exists($this, $loader)) { |
|
771 | + $class_obj = $this->{$loader}($class_name, $arguments); |
|
772 | + } else { |
|
773 | + $class_obj = $this->_create_object($class_name, $arguments, $addon, $from_db); |
|
774 | + } |
|
775 | + } |
|
776 | + if (($this->_cache_on && $cache) || $this->get_class_abbreviation($class_name, '')) { |
|
777 | + // save it for later... kinda like gum { : $ |
|
778 | + $this->_set_cached_class( |
|
779 | + $class_obj, |
|
780 | + $class_name, |
|
781 | + $addon, |
|
782 | + $from_db, |
|
783 | + $arguments |
|
784 | + ); |
|
785 | + } |
|
786 | + $this->_cache_on = true; |
|
787 | + return $class_obj; |
|
788 | + } |
|
789 | + |
|
790 | + |
|
791 | + /** |
|
792 | + * Recursively checks that a class exists and potentially attempts to load classes with non-FQCNs |
|
793 | + * |
|
794 | + * @param string|object $class_name |
|
795 | + * @param array $arguments |
|
796 | + * @param int $attempt |
|
797 | + * @return mixed |
|
798 | + */ |
|
799 | + private function loadOrVerifyClassExists($class_name, array $arguments, $attempt = 1) |
|
800 | + { |
|
801 | + if (is_object($class_name) || class_exists($class_name)) { |
|
802 | + return $class_name; |
|
803 | + } |
|
804 | + switch ($attempt) { |
|
805 | + case 1: |
|
806 | + // if it's a FQCN then maybe the class is registered with a preceding \ |
|
807 | + $class_name = strpos($class_name, '\\') !== false |
|
808 | + ? '\\' . ltrim($class_name, '\\') |
|
809 | + : $class_name; |
|
810 | + break; |
|
811 | + case 2: |
|
812 | + // |
|
813 | + $loader = $this->_dependency_map->class_loader($class_name); |
|
814 | + if ($loader && method_exists($this, $loader)) { |
|
815 | + return $this->{$loader}($class_name, $arguments); |
|
816 | + } |
|
817 | + break; |
|
818 | + case 3: |
|
819 | + default: |
|
820 | + return null; |
|
821 | + } |
|
822 | + $attempt++; |
|
823 | + return $this->loadOrVerifyClassExists($class_name, $arguments, $attempt); |
|
824 | + } |
|
825 | + |
|
826 | + |
|
827 | + /** |
|
828 | + * instantiates, caches, and injects dependencies for classes |
|
829 | + * |
|
830 | + * @param array $file_paths an array of paths to folders to look in |
|
831 | + * @param string $class_prefix EE or EEM or... ??? |
|
832 | + * @param bool|string $class_name $class name |
|
833 | + * @param string $type file type - core? class? helper? model? |
|
834 | + * @param mixed $arguments an argument or array of arguments to pass to the class upon instantiation |
|
835 | + * @param bool $from_db some classes are instantiated from the db |
|
836 | + * and thus call a different method to instantiate |
|
837 | + * @param bool $cache whether to cache the instantiated object for reuse |
|
838 | + * @param bool $load_only if true, will only load the file, but will NOT instantiate an object |
|
839 | + * @return bool|null|object null = failure to load or instantiate class object. |
|
840 | + * object = class loaded and instantiated successfully. |
|
841 | + * bool = fail or success when $load_only is true |
|
842 | + * @throws EE_Error |
|
843 | + * @throws ReflectionException |
|
844 | + * @throws InvalidInterfaceException |
|
845 | + * @throws InvalidDataTypeException |
|
846 | + * @throws InvalidArgumentException |
|
847 | + */ |
|
848 | + protected function _load( |
|
849 | + $file_paths = array(), |
|
850 | + $class_prefix = 'EE_', |
|
851 | + $class_name = false, |
|
852 | + $type = 'class', |
|
853 | + $arguments = array(), |
|
854 | + $from_db = false, |
|
855 | + $cache = true, |
|
856 | + $load_only = false |
|
857 | + ) { |
|
858 | + $class_name = ltrim($class_name, '\\'); |
|
859 | + // strip php file extension |
|
860 | + $class_name = str_replace('.php', '', trim($class_name)); |
|
861 | + // does the class have a prefix ? |
|
862 | + if (! empty($class_prefix) && $class_prefix !== 'addon') { |
|
863 | + // make sure $class_prefix is uppercase |
|
864 | + $class_prefix = strtoupper(trim($class_prefix)); |
|
865 | + // add class prefix ONCE!!! |
|
866 | + $class_name = $class_prefix . str_replace($class_prefix, '', $class_name); |
|
867 | + } |
|
868 | + $class_name = $this->class_cache->getFqnForAlias($class_name); |
|
869 | + $class_exists = class_exists($class_name, false); |
|
870 | + // if we're only loading the class and it already exists, then let's just return true immediately |
|
871 | + if ($load_only && $class_exists) { |
|
872 | + return true; |
|
873 | + } |
|
874 | + $arguments = is_array($arguments) ? $arguments : array($arguments); |
|
875 | + // $this->_cache_on is toggled during the recursive loading that can occur with dependency injection |
|
876 | + // $cache is controlled by individual calls to separate Registry loader methods like load_class() |
|
877 | + // $load_only is also controlled by individual calls to separate Registry loader methods like load_file() |
|
878 | + if ($this->_cache_on && $cache && ! $load_only) { |
|
879 | + // return object if it's already cached |
|
880 | + $cached_class = $this->_get_cached_class($class_name, $class_prefix, $arguments); |
|
881 | + if ($cached_class !== null) { |
|
882 | + return $cached_class; |
|
883 | + } |
|
884 | + } |
|
885 | + // if the class doesn't already exist.. then we need to try and find the file and load it |
|
886 | + if (! $class_exists) { |
|
887 | + // get full path to file |
|
888 | + $path = $this->_resolve_path($class_name, $type, $file_paths); |
|
889 | + // load the file |
|
890 | + $loaded = $this->_require_file($path, $class_name, $type, $file_paths); |
|
891 | + // if we are only loading a file but NOT instantiating an object |
|
892 | + // then return boolean for whether class was loaded or not |
|
893 | + if ($load_only) { |
|
894 | + return $loaded; |
|
895 | + } |
|
896 | + // if an object was expected but loading failed, then return nothing |
|
897 | + if (! $loaded) { |
|
898 | + return null; |
|
899 | + } |
|
900 | + } |
|
901 | + // instantiate the requested object |
|
902 | + $class_obj = $this->_create_object($class_name, $arguments, $type, $from_db); |
|
903 | + if ($this->_cache_on && $cache) { |
|
904 | + // save it for later... kinda like gum { : $ |
|
905 | + $this->_set_cached_class( |
|
906 | + $class_obj, |
|
907 | + $class_name, |
|
908 | + $class_prefix, |
|
909 | + $from_db, |
|
910 | + $arguments |
|
911 | + ); |
|
912 | + } |
|
913 | + $this->_cache_on = true; |
|
914 | + return $class_obj; |
|
915 | + } |
|
916 | + |
|
917 | + |
|
918 | + /** |
|
919 | + * @param string $class_name |
|
920 | + * @param string $default have to specify something, but not anything that will conflict |
|
921 | + * @return mixed|string |
|
922 | + */ |
|
923 | + protected function get_class_abbreviation($class_name, $default = 'FANCY_BATMAN_PANTS') |
|
924 | + { |
|
925 | + return isset($this->_class_abbreviations[ $class_name ]) |
|
926 | + ? $this->_class_abbreviations[ $class_name ] |
|
927 | + : $default; |
|
928 | + } |
|
929 | + |
|
930 | + |
|
931 | + /** |
|
932 | + * attempts to find a cached version of the requested class |
|
933 | + * by looking in the following places: |
|
934 | + * $this->{$class_abbreviation} ie: $this->CART |
|
935 | + * $this->{$class_name} ie: $this->Some_Class |
|
936 | + * $this->LIB->{$class_name} ie: $this->LIB->Some_Class |
|
937 | + * $this->addon->{$class_name} ie: $this->addon->Some_Addon_Class |
|
938 | + * |
|
939 | + * @param string $class_name |
|
940 | + * @param string $class_prefix |
|
941 | + * @param array $arguments |
|
942 | + * @return mixed |
|
943 | + */ |
|
944 | + protected function _get_cached_class( |
|
945 | + $class_name, |
|
946 | + $class_prefix = '', |
|
947 | + $arguments = array() |
|
948 | + ) { |
|
949 | + if ($class_name === 'EE_Registry') { |
|
950 | + return $this; |
|
951 | + } |
|
952 | + $class_abbreviation = $this->get_class_abbreviation($class_name); |
|
953 | + // check if class has already been loaded, and return it if it has been |
|
954 | + if (isset($this->{$class_abbreviation})) { |
|
955 | + return $this->{$class_abbreviation}; |
|
956 | + } |
|
957 | + $class_name = str_replace('\\', '_', $class_name); |
|
958 | + if (isset($this->{$class_name})) { |
|
959 | + return $this->{$class_name}; |
|
960 | + } |
|
961 | + if ($class_prefix === 'addon' && isset($this->addons->{$class_name})) { |
|
962 | + return $this->addons->{$class_name}; |
|
963 | + } |
|
964 | + $object_identifier = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
965 | + if (isset($this->LIB->{$object_identifier})) { |
|
966 | + return $this->LIB->{$object_identifier}; |
|
967 | + } |
|
968 | + foreach ($this->LIB as $key => $object) { |
|
969 | + if (// request does not contain new arguments and therefore no args identifier |
|
970 | + ! $this->object_identifier->hasArguments($object_identifier) |
|
971 | + // but previously cached class with args was found |
|
972 | + && $this->object_identifier->fqcnMatchesObjectIdentifier($class_name, $key) |
|
973 | + ) { |
|
974 | + return $object; |
|
975 | + } |
|
976 | + } |
|
977 | + return null; |
|
978 | + } |
|
979 | + |
|
980 | + |
|
981 | + /** |
|
982 | + * removes a cached version of the requested class |
|
983 | + * |
|
984 | + * @param string $class_name |
|
985 | + * @param boolean $addon |
|
986 | + * @param array $arguments |
|
987 | + * @return boolean |
|
988 | + */ |
|
989 | + public function clear_cached_class( |
|
990 | + $class_name, |
|
991 | + $addon = false, |
|
992 | + $arguments = array() |
|
993 | + ) { |
|
994 | + $class_abbreviation = $this->get_class_abbreviation($class_name); |
|
995 | + // check if class has already been loaded, and return it if it has been |
|
996 | + if (isset($this->{$class_abbreviation})) { |
|
997 | + $this->{$class_abbreviation} = null; |
|
998 | + return true; |
|
999 | + } |
|
1000 | + $class_name = str_replace('\\', '_', $class_name); |
|
1001 | + if (isset($this->{$class_name})) { |
|
1002 | + $this->{$class_name} = null; |
|
1003 | + return true; |
|
1004 | + } |
|
1005 | + if ($addon && isset($this->addons->{$class_name})) { |
|
1006 | + unset($this->addons->{$class_name}); |
|
1007 | + return true; |
|
1008 | + } |
|
1009 | + $class_name = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
1010 | + if (isset($this->LIB->{$class_name})) { |
|
1011 | + unset($this->LIB->{$class_name}); |
|
1012 | + return true; |
|
1013 | + } |
|
1014 | + return false; |
|
1015 | + } |
|
1016 | + |
|
1017 | + |
|
1018 | + /** |
|
1019 | + * _set_cached_class |
|
1020 | + * attempts to cache the instantiated class locally |
|
1021 | + * in one of the following places, in the following order: |
|
1022 | + * $this->{class_abbreviation} ie: $this->CART |
|
1023 | + * $this->{$class_name} ie: $this->Some_Class |
|
1024 | + * $this->addon->{$$class_name} ie: $this->addon->Some_Addon_Class |
|
1025 | + * $this->LIB->{$class_name} ie: $this->LIB->Some_Class |
|
1026 | + * |
|
1027 | + * @param object $class_obj |
|
1028 | + * @param string $class_name |
|
1029 | + * @param string $class_prefix |
|
1030 | + * @param bool $from_db |
|
1031 | + * @param array $arguments |
|
1032 | + * @return void |
|
1033 | + */ |
|
1034 | + protected function _set_cached_class( |
|
1035 | + $class_obj, |
|
1036 | + $class_name, |
|
1037 | + $class_prefix = '', |
|
1038 | + $from_db = false, |
|
1039 | + $arguments = array() |
|
1040 | + ) { |
|
1041 | + if ($class_name === 'EE_Registry' || empty($class_obj)) { |
|
1042 | + return; |
|
1043 | + } |
|
1044 | + // return newly instantiated class |
|
1045 | + $class_abbreviation = $this->get_class_abbreviation($class_name, ''); |
|
1046 | + if ($class_abbreviation) { |
|
1047 | + $this->{$class_abbreviation} = $class_obj; |
|
1048 | + return; |
|
1049 | + } |
|
1050 | + $class_name = str_replace('\\', '_', $class_name); |
|
1051 | + if (property_exists($this, $class_name)) { |
|
1052 | + $this->{$class_name} = $class_obj; |
|
1053 | + return; |
|
1054 | + } |
|
1055 | + if ($class_prefix === 'addon') { |
|
1056 | + $this->addons->{$class_name} = $class_obj; |
|
1057 | + return; |
|
1058 | + } |
|
1059 | + if (! $from_db) { |
|
1060 | + $class_name = $this->object_identifier->getIdentifier($class_name, $arguments); |
|
1061 | + $this->LIB->{$class_name} = $class_obj; |
|
1062 | + } |
|
1063 | + } |
|
1064 | + |
|
1065 | + |
|
1066 | + /** |
|
1067 | + * attempts to find a full valid filepath for the requested class. |
|
1068 | + * loops thru each of the base paths in the $file_paths array and appends : "{classname} . {file type} . php" |
|
1069 | + * then returns that path if the target file has been found and is readable |
|
1070 | + * |
|
1071 | + * @param string $class_name |
|
1072 | + * @param string $type |
|
1073 | + * @param array $file_paths |
|
1074 | + * @return string | bool |
|
1075 | + */ |
|
1076 | + protected function _resolve_path($class_name, $type = '', $file_paths = array()) |
|
1077 | + { |
|
1078 | + // make sure $file_paths is an array |
|
1079 | + $file_paths = is_array($file_paths) |
|
1080 | + ? $file_paths |
|
1081 | + : array($file_paths); |
|
1082 | + // cycle thru paths |
|
1083 | + foreach ($file_paths as $key => $file_path) { |
|
1084 | + // convert all separators to proper DS, if no filepath, then use EE_CLASSES |
|
1085 | + $file_path = $file_path |
|
1086 | + ? str_replace(array('/', '\\'), DS, $file_path) |
|
1087 | + : EE_CLASSES; |
|
1088 | + // prep file type |
|
1089 | + $type = ! empty($type) |
|
1090 | + ? trim($type, '.') . '.' |
|
1091 | + : ''; |
|
1092 | + // build full file path |
|
1093 | + $file_paths[ $key ] = rtrim($file_path, DS) . DS . $class_name . '.' . $type . 'php'; |
|
1094 | + // does the file exist and can be read ? |
|
1095 | + if (is_readable($file_paths[ $key ])) { |
|
1096 | + return $file_paths[ $key ]; |
|
1097 | + } |
|
1098 | + } |
|
1099 | + return false; |
|
1100 | + } |
|
1101 | + |
|
1102 | + |
|
1103 | + /** |
|
1104 | + * basically just performs a require_once() |
|
1105 | + * but with some error handling |
|
1106 | + * |
|
1107 | + * @param string $path |
|
1108 | + * @param string $class_name |
|
1109 | + * @param string $type |
|
1110 | + * @param array $file_paths |
|
1111 | + * @return bool |
|
1112 | + * @throws EE_Error |
|
1113 | + * @throws ReflectionException |
|
1114 | + */ |
|
1115 | + protected function _require_file($path, $class_name, $type = '', $file_paths = array()) |
|
1116 | + { |
|
1117 | + $this->resolve_legacy_class_parent($class_name); |
|
1118 | + // don't give up! you gotta... |
|
1119 | + try { |
|
1120 | + // does the file exist and can it be read ? |
|
1121 | + if (! $path) { |
|
1122 | + // just in case the file has already been autoloaded, |
|
1123 | + // but discrepancies in the naming schema are preventing it from |
|
1124 | + // being loaded via one of the EE_Registry::load_*() methods, |
|
1125 | + // then let's try one last hail mary before throwing an exception |
|
1126 | + // and call class_exists() again, but with autoloading turned ON |
|
1127 | + if (class_exists($class_name)) { |
|
1128 | + return true; |
|
1129 | + } |
|
1130 | + // so sorry, can't find the file |
|
1131 | + throw new EE_Error( |
|
1132 | + sprintf( |
|
1133 | + esc_html__( |
|
1134 | + 'The %1$s file %2$s could not be located or is not readable due to file permissions. Please ensure that the following filepath(s) are correct: %3$s', |
|
1135 | + 'event_espresso' |
|
1136 | + ), |
|
1137 | + trim($type, '.'), |
|
1138 | + $class_name, |
|
1139 | + '<br />' . implode(',<br />', $file_paths) |
|
1140 | + ) |
|
1141 | + ); |
|
1142 | + } |
|
1143 | + // get the file |
|
1144 | + require_once($path); |
|
1145 | + // if the class isn't already declared somewhere |
|
1146 | + if (class_exists($class_name, false) === false) { |
|
1147 | + // so sorry, not a class |
|
1148 | + throw new EE_Error( |
|
1149 | + sprintf( |
|
1150 | + esc_html__( |
|
1151 | + 'The %s file %s does not appear to contain the %s Class.', |
|
1152 | + 'event_espresso' |
|
1153 | + ), |
|
1154 | + $type, |
|
1155 | + $path, |
|
1156 | + $class_name |
|
1157 | + ) |
|
1158 | + ); |
|
1159 | + } |
|
1160 | + } catch (EE_Error $e) { |
|
1161 | + $e->get_error(); |
|
1162 | + return false; |
|
1163 | + } |
|
1164 | + return true; |
|
1165 | + } |
|
1166 | + |
|
1167 | + |
|
1168 | + /** |
|
1169 | + * Some of our legacy classes that extended a parent class would simply use a require() statement |
|
1170 | + * before their class declaration in order to ensure that the parent class was loaded. |
|
1171 | + * This is not ideal, but it's nearly impossible to determine the parent class of a non-namespaced class, |
|
1172 | + * without triggering a fatal error because the parent class has yet to be loaded and therefore doesn't exist. |
|
1173 | + * |
|
1174 | + * @param string $class_name |
|
1175 | + */ |
|
1176 | + protected function resolve_legacy_class_parent($class_name = '') |
|
1177 | + { |
|
1178 | + try { |
|
1179 | + $legacy_parent_class_map = array( |
|
1180 | + 'EE_Payment_Processor' => 'core/business/EE_Processor_Base.class.php', |
|
1181 | + ); |
|
1182 | + if (isset($legacy_parent_class_map[ $class_name ])) { |
|
1183 | + require_once EE_PLUGIN_DIR_PATH . $legacy_parent_class_map[ $class_name ]; |
|
1184 | + } |
|
1185 | + } catch (Exception $exception) { |
|
1186 | + } |
|
1187 | + } |
|
1188 | + |
|
1189 | + |
|
1190 | + /** |
|
1191 | + * _create_object |
|
1192 | + * Attempts to instantiate the requested class via any of the |
|
1193 | + * commonly used instantiation methods employed throughout EE. |
|
1194 | + * The priority for instantiation is as follows: |
|
1195 | + * - abstract classes or any class flagged as "load only" (no instantiation occurs) |
|
1196 | + * - model objects via their 'new_instance_from_db' method |
|
1197 | + * - model objects via their 'new_instance' method |
|
1198 | + * - "singleton" classes" via their 'instance' method |
|
1199 | + * - standard instantiable classes via their __constructor |
|
1200 | + * Prior to instantiation, if the classname exists in the dependency_map, |
|
1201 | + * then the constructor for the requested class will be examined to determine |
|
1202 | + * if any dependencies exist, and if they can be injected. |
|
1203 | + * If so, then those classes will be added to the array of arguments passed to the constructor |
|
1204 | + * |
|
1205 | + * @param string $class_name |
|
1206 | + * @param array $arguments |
|
1207 | + * @param string $type |
|
1208 | + * @param bool $from_db |
|
1209 | + * @return null|object|bool |
|
1210 | + * @throws InvalidArgumentException |
|
1211 | + * @throws InvalidInterfaceException |
|
1212 | + * @throws EE_Error |
|
1213 | + * @throws ReflectionException |
|
1214 | + * @throws InvalidDataTypeException |
|
1215 | + */ |
|
1216 | + protected function _create_object($class_name, $arguments = array(), $type = '', $from_db = false) |
|
1217 | + { |
|
1218 | + // create reflection |
|
1219 | + $reflector = $this->mirror->getReflectionClass($class_name); |
|
1220 | + // make sure arguments are an array |
|
1221 | + $arguments = is_array($arguments) |
|
1222 | + ? $arguments |
|
1223 | + : array($arguments); |
|
1224 | + // and if arguments array is numerically and sequentially indexed, then we want it to remain as is, |
|
1225 | + // else wrap it in an additional array so that it doesn't get split into multiple parameters |
|
1226 | + $arguments = $this->_array_is_numerically_and_sequentially_indexed($arguments) |
|
1227 | + ? $arguments |
|
1228 | + : array($arguments); |
|
1229 | + // attempt to inject dependencies ? |
|
1230 | + if ($this->_dependency_map->has($class_name)) { |
|
1231 | + $arguments = $this->_resolve_dependencies($reflector, $class_name, $arguments); |
|
1232 | + } |
|
1233 | + // instantiate the class if possible |
|
1234 | + if ($reflector->isAbstract()) { |
|
1235 | + // nothing to instantiate, loading file was enough |
|
1236 | + // does not throw an exception so $instantiation_mode is unused |
|
1237 | + // $instantiation_mode = "1) no constructor abstract class"; |
|
1238 | + return true; |
|
1239 | + } |
|
1240 | + if (empty($arguments) |
|
1241 | + && $this->mirror->getConstructorFromReflection($reflector) === null |
|
1242 | + && $reflector->isInstantiable() |
|
1243 | + ) { |
|
1244 | + // no constructor = static methods only... nothing to instantiate, loading file was enough |
|
1245 | + // $instantiation_mode = "2) no constructor but instantiable"; |
|
1246 | + return $reflector->newInstance(); |
|
1247 | + } |
|
1248 | + if ($from_db && method_exists($class_name, 'new_instance_from_db')) { |
|
1249 | + // $instantiation_mode = "3) new_instance_from_db()"; |
|
1250 | + return call_user_func_array(array($class_name, 'new_instance_from_db'), $arguments); |
|
1251 | + } |
|
1252 | + if (method_exists($class_name, 'new_instance')) { |
|
1253 | + // $instantiation_mode = "4) new_instance()"; |
|
1254 | + return call_user_func_array(array($class_name, 'new_instance'), $arguments); |
|
1255 | + } |
|
1256 | + if (method_exists($class_name, 'instance')) { |
|
1257 | + // $instantiation_mode = "5) instance()"; |
|
1258 | + return call_user_func_array(array($class_name, 'instance'), $arguments); |
|
1259 | + } |
|
1260 | + if ($reflector->isInstantiable()) { |
|
1261 | + // $instantiation_mode = "6) constructor"; |
|
1262 | + return $reflector->newInstanceArgs($arguments); |
|
1263 | + } |
|
1264 | + // heh ? something's not right ! |
|
1265 | + throw new EE_Error( |
|
1266 | + sprintf( |
|
1267 | + __('The %s file %s could not be instantiated.', 'event_espresso'), |
|
1268 | + $type, |
|
1269 | + $class_name |
|
1270 | + ) |
|
1271 | + ); |
|
1272 | + } |
|
1273 | + |
|
1274 | + |
|
1275 | + /** |
|
1276 | + * @see http://stackoverflow.com/questions/173400/how-to-check-if-php-array-is-associative-or-sequential |
|
1277 | + * @param array $array |
|
1278 | + * @return bool |
|
1279 | + */ |
|
1280 | + protected function _array_is_numerically_and_sequentially_indexed(array $array) |
|
1281 | + { |
|
1282 | + return ! empty($array) |
|
1283 | + ? array_keys($array) === range(0, count($array) - 1) |
|
1284 | + : true; |
|
1285 | + } |
|
1286 | + |
|
1287 | + |
|
1288 | + /** |
|
1289 | + * _resolve_dependencies |
|
1290 | + * examines the constructor for the requested class to determine |
|
1291 | + * if any dependencies exist, and if they can be injected. |
|
1292 | + * If so, then those classes will be added to the array of arguments passed to the constructor |
|
1293 | + * PLZ NOTE: this is achieved by type hinting the constructor params |
|
1294 | + * For example: |
|
1295 | + * if attempting to load a class "Foo" with the following constructor: |
|
1296 | + * __construct( Bar $bar_class, Fighter $grohl_class ) |
|
1297 | + * then $bar_class and $grohl_class will be added to the $arguments array, |
|
1298 | + * but only IF they are NOT already present in the incoming arguments array, |
|
1299 | + * and the correct classes can be loaded |
|
1300 | + * |
|
1301 | + * @param ReflectionClass $reflector |
|
1302 | + * @param string $class_name |
|
1303 | + * @param array $arguments |
|
1304 | + * @return array |
|
1305 | + * @throws InvalidArgumentException |
|
1306 | + * @throws InvalidDataTypeException |
|
1307 | + * @throws InvalidInterfaceException |
|
1308 | + * @throws ReflectionException |
|
1309 | + */ |
|
1310 | + protected function _resolve_dependencies(ReflectionClass $reflector, $class_name, array $arguments = array()) |
|
1311 | + { |
|
1312 | + // let's examine the constructor |
|
1313 | + $constructor = $this->mirror->getConstructorFromReflection($reflector); |
|
1314 | + // whu? huh? nothing? |
|
1315 | + if (! $constructor) { |
|
1316 | + return $arguments; |
|
1317 | + } |
|
1318 | + // get constructor parameters |
|
1319 | + $params = $this->mirror->getParametersFromReflection($reflector); |
|
1320 | + // and the keys for the incoming arguments array so that we can compare existing arguments with what is expected |
|
1321 | + $argument_keys = array_keys($arguments); |
|
1322 | + // now loop thru all of the constructors expected parameters |
|
1323 | + foreach ($params as $index => $param) { |
|
1324 | + try { |
|
1325 | + // is this a dependency for a specific class ? |
|
1326 | + $param_class = $this->mirror->getParameterClassName($param, $class_name, $index); |
|
1327 | + } catch (ReflectionException $exception) { |
|
1328 | + // uh-oh... most likely a legacy class that has not been autoloaded |
|
1329 | + // let's try to derive the classname from what we have now |
|
1330 | + // and hope that the property var name is close to the class name |
|
1331 | + $param_class = $param->getName(); |
|
1332 | + $param_class = str_replace('_', ' ', $param_class); |
|
1333 | + $param_class = ucwords($param_class); |
|
1334 | + $param_class = str_replace(' ', '_', $param_class); |
|
1335 | + } |
|
1336 | + // BUT WAIT !!! This class may be an alias for something else (or getting replaced at runtime) |
|
1337 | + $param_class = $this->class_cache->isAlias($param_class, $class_name) |
|
1338 | + ? $this->class_cache->getFqnForAlias($param_class, $class_name) |
|
1339 | + : $param_class; |
|
1340 | + if (// param is not even a class |
|
1341 | + $param_class === null |
|
1342 | + // and something already exists in the incoming arguments for this param |
|
1343 | + && array_key_exists($index, $argument_keys) |
|
1344 | + && array_key_exists($argument_keys[ $index ], $arguments) |
|
1345 | + ) { |
|
1346 | + // so let's skip this argument and move on to the next |
|
1347 | + continue; |
|
1348 | + } |
|
1349 | + if (// parameter is type hinted as a class, exists as an incoming argument, AND it's the correct class |
|
1350 | + $param_class !== null |
|
1351 | + && isset($argument_keys[ $index ], $arguments[ $argument_keys[ $index ] ]) |
|
1352 | + && $arguments[ $argument_keys[ $index ] ] instanceof $param_class |
|
1353 | + ) { |
|
1354 | + // skip this argument and move on to the next |
|
1355 | + continue; |
|
1356 | + } |
|
1357 | + if (// parameter is type hinted as a class, and should be injected |
|
1358 | + $param_class !== null |
|
1359 | + && $this->_dependency_map->has_dependency_for_class($class_name, $param_class) |
|
1360 | + ) { |
|
1361 | + $arguments = $this->_resolve_dependency( |
|
1362 | + $class_name, |
|
1363 | + $param_class, |
|
1364 | + $arguments, |
|
1365 | + $index |
|
1366 | + ); |
|
1367 | + } |
|
1368 | + if (empty($arguments[ $index ])) { |
|
1369 | + $arguments[ $index ] = $this->mirror->getParameterDefaultValue( |
|
1370 | + $param, |
|
1371 | + $class_name, |
|
1372 | + $index |
|
1373 | + ); |
|
1374 | + } |
|
1375 | + } |
|
1376 | + return $arguments; |
|
1377 | + } |
|
1378 | + |
|
1379 | + |
|
1380 | + /** |
|
1381 | + * @param string $class_name |
|
1382 | + * @param string $param_class |
|
1383 | + * @param array $arguments |
|
1384 | + * @param mixed $index |
|
1385 | + * @return array |
|
1386 | + * @throws InvalidArgumentException |
|
1387 | + * @throws InvalidInterfaceException |
|
1388 | + * @throws InvalidDataTypeException |
|
1389 | + */ |
|
1390 | + protected function _resolve_dependency($class_name, $param_class, $arguments, $index) |
|
1391 | + { |
|
1392 | + $dependency = null; |
|
1393 | + // should dependency be loaded from cache ? |
|
1394 | + $cache_on = $this->_dependency_map->loading_strategy_for_class_dependency( |
|
1395 | + $class_name, |
|
1396 | + $param_class |
|
1397 | + ); |
|
1398 | + $cache_on = $cache_on !== EE_Dependency_Map::load_new_object; |
|
1399 | + // we might have a dependency... |
|
1400 | + // let's MAYBE try and find it in our cache if that's what's been requested |
|
1401 | + $cached_class = $cache_on |
|
1402 | + ? $this->_get_cached_class($param_class) |
|
1403 | + : null; |
|
1404 | + // and grab it if it exists |
|
1405 | + if ($cached_class instanceof $param_class) { |
|
1406 | + $dependency = $cached_class; |
|
1407 | + } elseif ($param_class !== $class_name) { |
|
1408 | + // obtain the loader method from the dependency map |
|
1409 | + $loader = $this->_dependency_map->class_loader($param_class); |
|
1410 | + // is loader a custom closure ? |
|
1411 | + if ($loader instanceof Closure) { |
|
1412 | + $dependency = $loader($arguments); |
|
1413 | + } else { |
|
1414 | + // set the cache on property for the recursive loading call |
|
1415 | + $this->_cache_on = $cache_on; |
|
1416 | + // if not, then let's try and load it via the registry |
|
1417 | + if ($loader && method_exists($this, $loader)) { |
|
1418 | + $dependency = $this->{$loader}($param_class); |
|
1419 | + } else { |
|
1420 | + $dependency = LoaderFactory::getLoader()->load( |
|
1421 | + $param_class, |
|
1422 | + array(), |
|
1423 | + $cache_on |
|
1424 | + ); |
|
1425 | + } |
|
1426 | + } |
|
1427 | + } |
|
1428 | + // did we successfully find the correct dependency ? |
|
1429 | + if ($dependency instanceof $param_class) { |
|
1430 | + // then let's inject it into the incoming array of arguments at the correct location |
|
1431 | + $arguments[ $index ] = $dependency; |
|
1432 | + } |
|
1433 | + return $arguments; |
|
1434 | + } |
|
1435 | + |
|
1436 | + |
|
1437 | + /** |
|
1438 | + * call any loader that's been registered in the EE_Dependency_Map::$_class_loaders array |
|
1439 | + * |
|
1440 | + * @param string $classname PLEASE NOTE: the class name needs to match what's registered |
|
1441 | + * in the EE_Dependency_Map::$_class_loaders array, |
|
1442 | + * including the class prefix, ie: "EE_", "EEM_", "EEH_", etc |
|
1443 | + * @param array $arguments |
|
1444 | + * @return object |
|
1445 | + */ |
|
1446 | + public static function factory($classname, $arguments = array()) |
|
1447 | + { |
|
1448 | + $loader = self::instance()->_dependency_map->class_loader($classname); |
|
1449 | + if ($loader instanceof Closure) { |
|
1450 | + return $loader($arguments); |
|
1451 | + } |
|
1452 | + if (method_exists(self::instance(), $loader)) { |
|
1453 | + return self::instance()->{$loader}($classname, $arguments); |
|
1454 | + } |
|
1455 | + return null; |
|
1456 | + } |
|
1457 | + |
|
1458 | + |
|
1459 | + /** |
|
1460 | + * Gets the addon by its class name |
|
1461 | + * |
|
1462 | + * @param string $class_name |
|
1463 | + * @return EE_Addon |
|
1464 | + */ |
|
1465 | + public function getAddon($class_name) |
|
1466 | + { |
|
1467 | + $class_name = str_replace('\\', '_', $class_name); |
|
1468 | + if (isset($this->addons->{$class_name})) { |
|
1469 | + return $this->addons->{$class_name}; |
|
1470 | + } else { |
|
1471 | + return null; |
|
1472 | + } |
|
1473 | + } |
|
1474 | + |
|
1475 | + |
|
1476 | + /** |
|
1477 | + * removes the addon from the internal cache |
|
1478 | + * |
|
1479 | + * @param string $class_name |
|
1480 | + * @return void |
|
1481 | + */ |
|
1482 | + public function removeAddon($class_name) |
|
1483 | + { |
|
1484 | + $class_name = str_replace('\\', '_', $class_name); |
|
1485 | + unset($this->addons->{$class_name}); |
|
1486 | + } |
|
1487 | + |
|
1488 | + |
|
1489 | + /** |
|
1490 | + * Gets the addon by its name/slug (not classname. For that, just |
|
1491 | + * use the get_addon() method above |
|
1492 | + * |
|
1493 | + * @param string $name |
|
1494 | + * @return EE_Addon |
|
1495 | + */ |
|
1496 | + public function get_addon_by_name($name) |
|
1497 | + { |
|
1498 | + foreach ($this->addons as $addon) { |
|
1499 | + if ($addon->name() === $name) { |
|
1500 | + return $addon; |
|
1501 | + } |
|
1502 | + } |
|
1503 | + return null; |
|
1504 | + } |
|
1505 | + |
|
1506 | + |
|
1507 | + /** |
|
1508 | + * Gets an array of all the registered addons, where the keys are their names. |
|
1509 | + * (ie, what each returns for their name() function) |
|
1510 | + * They're already available on EE_Registry::instance()->addons as properties, |
|
1511 | + * where each property's name is the addon's classname, |
|
1512 | + * So if you just want to get the addon by classname, |
|
1513 | + * OR use the get_addon() method above. |
|
1514 | + * PLEASE NOTE: |
|
1515 | + * addons with Fully Qualified Class Names |
|
1516 | + * have had the namespace separators converted to underscores, |
|
1517 | + * so a classname like Fully\Qualified\ClassName |
|
1518 | + * would have been converted to Fully_Qualified_ClassName |
|
1519 | + * |
|
1520 | + * @return EE_Addon[] where the KEYS are the addon's name() |
|
1521 | + */ |
|
1522 | + public function get_addons_by_name() |
|
1523 | + { |
|
1524 | + $addons = array(); |
|
1525 | + foreach ($this->addons as $addon) { |
|
1526 | + $addons[ $addon->name() ] = $addon; |
|
1527 | + } |
|
1528 | + return $addons; |
|
1529 | + } |
|
1530 | + |
|
1531 | + |
|
1532 | + /** |
|
1533 | + * Resets the specified model's instance AND makes sure EE_Registry doesn't keep |
|
1534 | + * a stale copy of it around |
|
1535 | + * |
|
1536 | + * @param string $model_name |
|
1537 | + * @return \EEM_Base |
|
1538 | + * @throws \EE_Error |
|
1539 | + */ |
|
1540 | + public function reset_model($model_name) |
|
1541 | + { |
|
1542 | + $model_class_name = strpos($model_name, 'EEM_') !== 0 |
|
1543 | + ? "EEM_{$model_name}" |
|
1544 | + : $model_name; |
|
1545 | + if (! isset($this->LIB->{$model_class_name}) || ! $this->LIB->{$model_class_name} instanceof EEM_Base) { |
|
1546 | + return null; |
|
1547 | + } |
|
1548 | + // get that model reset it and make sure we nuke the old reference to it |
|
1549 | + if ($this->LIB->{$model_class_name} instanceof $model_class_name |
|
1550 | + && is_callable( |
|
1551 | + array($model_class_name, 'reset') |
|
1552 | + )) { |
|
1553 | + $this->LIB->{$model_class_name} = $this->LIB->{$model_class_name}->reset(); |
|
1554 | + } else { |
|
1555 | + throw new EE_Error( |
|
1556 | + sprintf( |
|
1557 | + esc_html__('Model %s does not have a method "reset"', 'event_espresso'), |
|
1558 | + $model_name |
|
1559 | + ) |
|
1560 | + ); |
|
1561 | + } |
|
1562 | + return $this->LIB->{$model_class_name}; |
|
1563 | + } |
|
1564 | + |
|
1565 | + |
|
1566 | + /** |
|
1567 | + * Resets the registry. |
|
1568 | + * The criteria for what gets reset is based on what can be shared between sites on the same request when |
|
1569 | + * switch_to_blog is used in a multisite install. Here is a list of things that are NOT reset. |
|
1570 | + * - $_dependency_map |
|
1571 | + * - $_class_abbreviations |
|
1572 | + * - $NET_CFG (EE_Network_Config): The config is shared network wide so no need to reset. |
|
1573 | + * - $REQ: Still on the same request so no need to change. |
|
1574 | + * - $CAP: There is no site specific state in the EE_Capability class. |
|
1575 | + * - $SSN: Although ideally, the session should not be shared between site switches, we can't reset it because only |
|
1576 | + * one Session can be active in a single request. Resetting could resolve in "headers already sent" errors. |
|
1577 | + * - $addons: In multisite, the state of the addons is something controlled via hooks etc in a normal request. So |
|
1578 | + * for now, we won't reset the addons because it could break calls to an add-ons class/methods in the |
|
1579 | + * switch or on the restore. |
|
1580 | + * - $modules |
|
1581 | + * - $shortcodes |
|
1582 | + * - $widgets |
|
1583 | + * |
|
1584 | + * @param boolean $hard [deprecated] |
|
1585 | + * @param boolean $reinstantiate whether to create new instances of EE_Registry's singletons too, |
|
1586 | + * or just reset without re-instantiating (handy to set to FALSE if you're not |
|
1587 | + * sure if you CAN currently reinstantiate the singletons at the moment) |
|
1588 | + * @param bool $reset_models Defaults to true. When false, then the models are not reset. This is so |
|
1589 | + * client |
|
1590 | + * code instead can just change the model context to a different blog id if |
|
1591 | + * necessary |
|
1592 | + * @return EE_Registry |
|
1593 | + * @throws InvalidInterfaceException |
|
1594 | + * @throws InvalidDataTypeException |
|
1595 | + * @throws EE_Error |
|
1596 | + * @throws ReflectionException |
|
1597 | + * @throws InvalidArgumentException |
|
1598 | + */ |
|
1599 | + public static function reset($hard = false, $reinstantiate = true, $reset_models = true) |
|
1600 | + { |
|
1601 | + $instance = self::instance(); |
|
1602 | + $instance->_cache_on = true; |
|
1603 | + // reset some "special" classes |
|
1604 | + EEH_Activation::reset(); |
|
1605 | + $hard = apply_filters('FHEE__EE_Registry__reset__hard', $hard); |
|
1606 | + $instance->CFG = EE_Config::reset($hard, $reinstantiate); |
|
1607 | + $instance->CART = null; |
|
1608 | + $instance->MRM = null; |
|
1609 | + $instance->AssetsRegistry = LoaderFactory::getLoader()->getShared( |
|
1610 | + 'EventEspresso\core\services\assets\Registry' |
|
1611 | + ); |
|
1612 | + // messages reset |
|
1613 | + EED_Messages::reset(); |
|
1614 | + // handle of objects cached on LIB |
|
1615 | + foreach (array('LIB', 'modules') as $cache) { |
|
1616 | + foreach ($instance->{$cache} as $class_name => $class) { |
|
1617 | + if (self::_reset_and_unset_object($class, $reset_models)) { |
|
1618 | + unset($instance->{$cache}->{$class_name}); |
|
1619 | + } |
|
1620 | + } |
|
1621 | + } |
|
1622 | + return $instance; |
|
1623 | + } |
|
1624 | + |
|
1625 | + |
|
1626 | + /** |
|
1627 | + * if passed object implements ResettableInterface, then call it's reset() method |
|
1628 | + * if passed object implements InterminableInterface, then return false, |
|
1629 | + * to indicate that it should NOT be cleared from the Registry cache |
|
1630 | + * |
|
1631 | + * @param $object |
|
1632 | + * @param bool $reset_models |
|
1633 | + * @return bool returns true if cached object should be unset |
|
1634 | + */ |
|
1635 | + private static function _reset_and_unset_object($object, $reset_models) |
|
1636 | + { |
|
1637 | + if (! is_object($object)) { |
|
1638 | + // don't unset anything that's not an object |
|
1639 | + return false; |
|
1640 | + } |
|
1641 | + if ($object instanceof EED_Module) { |
|
1642 | + $object::reset(); |
|
1643 | + // don't unset modules |
|
1644 | + return false; |
|
1645 | + } |
|
1646 | + if ($object instanceof ResettableInterface) { |
|
1647 | + if ($object instanceof EEM_Base) { |
|
1648 | + if ($reset_models) { |
|
1649 | + $object->reset(); |
|
1650 | + return true; |
|
1651 | + } |
|
1652 | + return false; |
|
1653 | + } |
|
1654 | + $object->reset(); |
|
1655 | + return true; |
|
1656 | + } |
|
1657 | + if (! $object instanceof InterminableInterface) { |
|
1658 | + return true; |
|
1659 | + } |
|
1660 | + return false; |
|
1661 | + } |
|
1662 | + |
|
1663 | + |
|
1664 | + /** |
|
1665 | + * Gets all the custom post type models defined |
|
1666 | + * |
|
1667 | + * @return array keys are model "short names" (Eg "Event") and keys are classnames (eg "EEM_Event") |
|
1668 | + */ |
|
1669 | + public function cpt_models() |
|
1670 | + { |
|
1671 | + $cpt_models = array(); |
|
1672 | + foreach ($this->non_abstract_db_models as $short_name => $classname) { |
|
1673 | + if (is_subclass_of($classname, 'EEM_CPT_Base')) { |
|
1674 | + $cpt_models[ $short_name ] = $classname; |
|
1675 | + } |
|
1676 | + } |
|
1677 | + return $cpt_models; |
|
1678 | + } |
|
1679 | + |
|
1680 | + |
|
1681 | + /** |
|
1682 | + * @return \EE_Config |
|
1683 | + */ |
|
1684 | + public static function CFG() |
|
1685 | + { |
|
1686 | + return self::instance()->CFG; |
|
1687 | + } |
|
1688 | + |
|
1689 | + |
|
1690 | + /** |
|
1691 | + * @deprecated 4.9.62.p |
|
1692 | + * @param string $class_name |
|
1693 | + * @return ReflectionClass |
|
1694 | + * @throws ReflectionException |
|
1695 | + * @throws InvalidDataTypeException |
|
1696 | + */ |
|
1697 | + public function get_ReflectionClass($class_name) |
|
1698 | + { |
|
1699 | + return $this->mirror->getReflectionClass($class_name); |
|
1700 | + } |
|
1701 | 1701 | } |
@@ -556,8 +556,8 @@ discard block |
||
556 | 556 | * that they WILL be added (because the attendees stage runs nearly last during |
557 | 557 | * the migration script) |
558 | 558 | * @param type $new_ticket_id |
559 | - * @param type $sold |
|
560 | - * @param type $STS_ID |
|
559 | + * @param string $STS_ID |
|
560 | + * @param integer $quantity_sold |
|
561 | 561 | * @return boolean whether they were successfully updated or not |
562 | 562 | */ |
563 | 563 | protected function _add_regs_to_ticket_and_datetimes($new_ticket_id, $quantity_sold, $STS_ID) |
@@ -757,6 +757,9 @@ discard block |
||
757 | 757 | return $count; |
758 | 758 | } |
759 | 759 | |
760 | + /** |
|
761 | + * @param integer $new_txn_id |
|
762 | + */ |
|
760 | 763 | private function _insert_new_payment($old_attendee, $new_txn_id) |
761 | 764 | { |
762 | 765 | global $wpdb; |
@@ -183,645 +183,645 @@ |
||
183 | 183 | */ |
184 | 184 | class EE_DMS_4_1_0_attendees extends EE_Data_Migration_Script_Stage_Table |
185 | 185 | { |
186 | - private $_new_attendee_cpt_table; |
|
187 | - private $_new_attendee_meta_table; |
|
188 | - private $_new_reg_table; |
|
189 | - private $_new_transaction_table; |
|
190 | - private $_new_payment_table; |
|
191 | - private $_new_line_table; |
|
192 | - private $_old_mer_table; |
|
193 | - private $_new_ticket_table; |
|
194 | - private $_new_ticket_datetime_table; |
|
195 | - private $_new_datetime_table; |
|
196 | - private $_new_datetime_ticket_table; |
|
197 | - private $_new_price_table; |
|
198 | - private $_new_ticket_price_table; |
|
199 | - /** |
|
200 | - * Rememebrs whether or not the mer table exists |
|
201 | - * @var boolean |
|
202 | - */ |
|
203 | - private $_mer_tables_exist = null; |
|
204 | - |
|
205 | - public function __construct() |
|
206 | - { |
|
207 | - global $wpdb; |
|
208 | - $this->_pretty_name = __("Attendees", "event_espresso"); |
|
209 | - $this->_old_table = $wpdb->prefix . "events_attendee"; |
|
210 | - // Only select the event status column from the event table. |
|
211 | - $this->select_expression = 'att.*, e.event_status'; |
|
212 | - // Only select attendees for events that aren't deleted. |
|
213 | - $this->_extra_where_sql = 'AS att |
|
186 | + private $_new_attendee_cpt_table; |
|
187 | + private $_new_attendee_meta_table; |
|
188 | + private $_new_reg_table; |
|
189 | + private $_new_transaction_table; |
|
190 | + private $_new_payment_table; |
|
191 | + private $_new_line_table; |
|
192 | + private $_old_mer_table; |
|
193 | + private $_new_ticket_table; |
|
194 | + private $_new_ticket_datetime_table; |
|
195 | + private $_new_datetime_table; |
|
196 | + private $_new_datetime_ticket_table; |
|
197 | + private $_new_price_table; |
|
198 | + private $_new_ticket_price_table; |
|
199 | + /** |
|
200 | + * Rememebrs whether or not the mer table exists |
|
201 | + * @var boolean |
|
202 | + */ |
|
203 | + private $_mer_tables_exist = null; |
|
204 | + |
|
205 | + public function __construct() |
|
206 | + { |
|
207 | + global $wpdb; |
|
208 | + $this->_pretty_name = __("Attendees", "event_espresso"); |
|
209 | + $this->_old_table = $wpdb->prefix . "events_attendee"; |
|
210 | + // Only select the event status column from the event table. |
|
211 | + $this->select_expression = 'att.*, e.event_status'; |
|
212 | + // Only select attendees for events that aren't deleted. |
|
213 | + $this->_extra_where_sql = 'AS att |
|
214 | 214 | INNER JOIN ' . $wpdb->prefix . 'events_detail AS e ON att.event_id=e.id |
215 | 215 | WHERE e.event_status!="D"'; |
216 | - $this->_old_mer_table = $wpdb->prefix . "events_multi_event_registration_id_group"; |
|
217 | - $this->_new_attendee_cpt_table = $wpdb->posts; |
|
218 | - $this->_new_attendee_meta_table = $wpdb->prefix . "esp_attendee_meta"; |
|
219 | - $this->_new_reg_table = $wpdb->prefix . "esp_registration"; |
|
220 | - $this->_new_transaction_table = $wpdb->prefix . "esp_transaction"; |
|
221 | - $this->_new_payment_table = $wpdb->prefix . "esp_payment"; |
|
222 | - $this->_new_line_table = $wpdb->prefix . "esp_line_item"; |
|
223 | - $this->_new_ticket_table = $wpdb->prefix . "esp_ticket"; |
|
224 | - $this->_new_ticket_datetime_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
225 | - $this->_new_datetime_table = $wpdb->prefix . "esp_datetime"; |
|
226 | - $this->_new_datetime_ticket_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
227 | - $this->_new_price_table = $wpdb->prefix . "esp_price"; |
|
228 | - $this->_new_ticket_price_table = $wpdb->prefix . "esp_ticket_price"; |
|
229 | - parent::__construct(); |
|
230 | - } |
|
231 | - |
|
232 | - protected function _migrate_old_row($old_row) |
|
233 | - { |
|
234 | - // first check if there's already a new attendee with similar characteristics |
|
235 | - $new_att_id = $this->_find_attendee_cpt_matching($old_row); |
|
236 | - if (!$new_att_id) { |
|
237 | - $new_att_id = $this->_insert_new_attendee_cpt($old_row); |
|
238 | - if (!$new_att_id) { |
|
239 | - // if we couldnt even make an attendee, abandon all hope |
|
240 | - return false; |
|
241 | - } |
|
242 | - $new_att_meta_id = $this->_insert_attendee_meta_row($old_row, $new_att_id); |
|
243 | - if ($new_att_meta_id) { |
|
244 | - $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_attendee_meta_table, $new_att_meta_id); |
|
245 | - } |
|
246 | - } |
|
247 | - $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_attendee_cpt_table, $new_att_id); |
|
248 | - |
|
249 | - $txn_id = $this->_insert_new_transaction($old_row); |
|
250 | - if (!$txn_id) { |
|
251 | - // if we couldnt make the transaction, also abandon all hope |
|
252 | - return false; |
|
253 | - } |
|
254 | - $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_transaction_table, $txn_id); |
|
255 | - $pay_id = $this->_insert_new_payment($old_row, $txn_id); |
|
256 | - if ($pay_id) { |
|
257 | - $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_payment_table, $pay_id); |
|
258 | - } |
|
259 | - |
|
260 | - |
|
261 | - // even if there was no payment, we can go ahead with adding the reg |
|
262 | - $new_regs = $this->_insert_new_registrations($old_row, $new_att_id, $txn_id); |
|
263 | - if ($new_regs) { |
|
264 | - $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_reg_table, $new_regs); |
|
265 | - } |
|
266 | - } |
|
267 | - |
|
268 | - /** |
|
269 | - * Checks if there's already an attendee CPT in the db that has the same |
|
270 | - * first and last name, and email. If so, returns its ID as an int. |
|
271 | - * @global type $wpdb |
|
272 | - * @param array $old_attendee |
|
273 | - * @return int |
|
274 | - */ |
|
275 | - private function _find_attendee_cpt_matching($old_attendee) |
|
276 | - { |
|
277 | - global $wpdb; |
|
278 | - $existing_attendee_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_new_attendee_cpt_table . " AS cpt INNER JOIN " . $this->_new_attendee_meta_table . " AS meta ON cpt.ID = meta.ATT_ID WHERE meta.ATT_fname = %s AND meta.ATT_lname = %s AND meta.ATT_email = %s LIMIT 1", $old_attendee['fname'], $old_attendee['lname'], $old_attendee['email'])); |
|
279 | - return intval($existing_attendee_id); |
|
280 | - } |
|
281 | - |
|
282 | - private function _insert_new_attendee_cpt($old_attendee) |
|
283 | - { |
|
284 | - global $wpdb; |
|
285 | - $cols_n_values = array( |
|
286 | - 'post_title' => stripslashes($old_attendee['fname'] . " " . $old_attendee['lname']),// ATT_full_name |
|
287 | - 'post_content' => '',// ATT_bio |
|
288 | - 'post_name' => sanitize_title($old_attendee['fname'] . "-" . $old_attendee['lname']),// ATT_slug |
|
289 | - 'post_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_created |
|
290 | - 'post_excerpt' => '',// ATT_short_bio |
|
291 | - 'post_modified' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_modified |
|
292 | - 'post_author' => 0,// ATT_author |
|
293 | - 'post_parent' => 0,// ATT_parent |
|
294 | - 'post_type' => 'espresso_attendees',// post_type |
|
295 | - 'post_status' => 'publish'// status |
|
296 | - ); |
|
297 | - $datatypes = array( |
|
298 | - '%s',// ATT_full_name |
|
299 | - '%s',// ATT_bio |
|
300 | - '%s',// ATT_slug |
|
301 | - '%s',// ATT_created |
|
302 | - '%s',// ATT_short_bio |
|
303 | - '%s',// ATT_modified |
|
304 | - '%d',// ATT_author |
|
305 | - '%d',// ATT_parent |
|
306 | - '%s',// post_type |
|
307 | - '%s',// status |
|
308 | - ); |
|
309 | - $success = $wpdb->insert($this->_new_attendee_cpt_table, $cols_n_values, $datatypes); |
|
310 | - if (!$success) { |
|
311 | - $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
|
312 | - return 0; |
|
313 | - } |
|
314 | - $new_id = $wpdb->insert_id; |
|
315 | - return $new_id; |
|
316 | - } |
|
317 | - |
|
318 | - private function _insert_attendee_meta_row($old_attendee, $new_attendee_cpt_id) |
|
319 | - { |
|
320 | - global $wpdb; |
|
321 | - // get the state and country ids from the old row |
|
322 | - try { |
|
323 | - $new_country = $this->get_migration_script()->get_or_create_country(stripslashes($old_attendee['country_id'])); |
|
324 | - $new_country_iso = $new_country['CNT_ISO']; |
|
325 | - } catch (EE_Error $exception) { |
|
326 | - $new_country_iso = $this->get_migration_script()->get_default_country_iso(); |
|
327 | - } |
|
328 | - try { |
|
329 | - $new_state = $this->get_migration_script()->get_or_create_state(stripslashes($old_attendee['state']), $new_country_iso); |
|
330 | - $new_state_id = $new_state['STA_ID']; |
|
331 | - } catch (EE_Error $exception) { |
|
332 | - $new_state_id = 0; |
|
333 | - } |
|
334 | - $cols_n_values = array( |
|
335 | - 'ATT_ID' => $new_attendee_cpt_id, |
|
336 | - 'ATT_fname' => stripslashes($old_attendee['fname']), |
|
337 | - 'ATT_lname' => stripslashes($old_attendee['lname']), |
|
338 | - 'ATT_address' => stripslashes($old_attendee['address']), |
|
339 | - 'ATT_address2' => stripslashes($old_attendee['address2']), |
|
340 | - 'ATT_city' => stripslashes($old_attendee['city']), |
|
341 | - 'STA_ID' => $new_state_id, |
|
342 | - 'CNT_ISO' => $new_country_iso, |
|
343 | - 'ATT_zip' => stripslashes($old_attendee['zip']), |
|
344 | - 'ATT_email' => stripslashes($old_attendee['email']), |
|
345 | - 'ATT_phone' => stripslashes($old_attendee['phone']), |
|
346 | - ); |
|
347 | - $datatypes = array( |
|
348 | - '%d',// ATT_ID |
|
349 | - '%s',// ATT_fname |
|
350 | - '%s',// ATT_lname |
|
351 | - '%s',// ATT_address |
|
352 | - '%s',// ATT_address2 |
|
353 | - '%s',// ATT_city |
|
354 | - '%d',// STA_ID |
|
355 | - '%s',// CNT_ISO |
|
356 | - '%s',// ATT_zip |
|
357 | - '%s',// ATT_email |
|
358 | - '%s',// ATT_phone |
|
359 | - ); |
|
360 | - $success = $wpdb->insert($this->_new_attendee_meta_table, $cols_n_values, $datatypes); |
|
361 | - if (!$success) { |
|
362 | - $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_meta_table, $cols_n_values, $datatypes)); |
|
363 | - return 0; |
|
364 | - } |
|
365 | - $new_id = $wpdb->insert_id; |
|
366 | - return $new_id; |
|
367 | - } |
|
368 | - |
|
369 | - /** |
|
370 | - * Note: we don't necessarily create a new transaction for each attendee row. |
|
371 | - * Only if the old attendee 'is_primary' is true; otherwise we find the old attendee row that |
|
372 | - * 'is_primary' and has the same 'txn_id', then we return ITS new transaction id |
|
373 | - * @global type $wpdb |
|
374 | - * @param type $old_attendee |
|
375 | - * @return int new transaction id |
|
376 | - */ |
|
377 | - private function _insert_new_transaction($old_attendee) |
|
378 | - { |
|
379 | - global $wpdb; |
|
380 | - |
|
381 | - // first: let's check for an existing transaction for this old attendee |
|
382 | - if (intval($old_attendee['is_primary'])) {// primary attendee, so create txn |
|
383 | - $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($old_attendee['id']), $this->_new_transaction_table); |
|
384 | - } else { // non-primary attendee, so find its primary attendee's transaction |
|
385 | - $primary_attendee_old_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_old_table . " WHERE is_primary=1 and registration_id=%s", $old_attendee['registration_id'])); |
|
386 | - if (!$primary_attendee_old_id) { |
|
387 | - $primary_attendee = $this->_find_mer_primary_attendee_using_mer_tables($old_attendee['registration_id']); |
|
388 | - $primary_attendee_old_id = is_array($primary_attendee) ? $primary_attendee['id'] : null; |
|
389 | - } |
|
390 | - $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($primary_attendee_old_id), $this->_new_transaction_table); |
|
391 | - if (!$txn_id) { |
|
392 | - $this->add_error(sprintf(__("Could not find primary attendee's new transaction. Current attendee is: %s, we think the 3.1 primary attendee for it has id %d, but there's no 4.1 transaction for that primary attendee id.", "event_espresso"), $this->_json_encode($old_attendee), $primary_attendee_old_id)); |
|
393 | - $txn_id = 0; |
|
394 | - } |
|
395 | - } |
|
396 | - // if there isn't yet a transaction row for this, create one |
|
397 | - // (so even if it was a non-primary attendee with no EE3 primary attendee, |
|
398 | - // it ought to have SOME transaction, so we'll make one) |
|
399 | - if (!$txn_id) { |
|
400 | - // maps 3.1 payment stati onto 4.1 transaction stati |
|
401 | - $txn_status_mapping = array( |
|
402 | - 'Completed' => 'TCM', |
|
403 | - 'Pending' => 'TIN', |
|
404 | - 'Payment Declined' => 'TIN', |
|
405 | - 'Incomplete' => 'TIN', |
|
406 | - 'Not Completed' => 'TIN', |
|
407 | - 'Cancelled' => 'TIN', |
|
408 | - 'Declined' => 'TIN' |
|
409 | - ); |
|
410 | - $STS_ID = isset($txn_status_mapping[ $old_attendee['payment_status'] ]) ? $txn_status_mapping[ $old_attendee['payment_status'] ] : 'TIN'; |
|
411 | - $cols_n_values = array( |
|
412 | - 'TXN_timestamp' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
413 | - 'TXN_total' => floatval($old_attendee['total_cost']), |
|
414 | - 'TXN_paid' => floatval($old_attendee['amount_pd']), |
|
415 | - 'STS_ID' => $STS_ID, |
|
416 | - 'TXN_hash_salt' => $old_attendee['hashSalt'] |
|
417 | - ); |
|
418 | - $datatypes = array( |
|
419 | - '%s',// TXN_timestamp |
|
420 | - '%f',// TXN_total |
|
421 | - '%f',// TXN_paid |
|
422 | - '%s',// STS_ID |
|
423 | - '%s',// TXN_hash_salt |
|
424 | - ); |
|
425 | - $success = $wpdb->insert($this->_new_transaction_table, $cols_n_values, $datatypes); |
|
426 | - if (!$success) { |
|
427 | - $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_transaction_table, $cols_n_values, $datatypes)); |
|
428 | - return 0; |
|
429 | - } |
|
430 | - $txn_id = $wpdb->insert_id; |
|
431 | - } |
|
432 | - |
|
433 | - return $txn_id; |
|
434 | - } |
|
435 | - |
|
436 | - /** |
|
437 | - * Detects if the MER tables exist |
|
438 | - * @global type $wpdb |
|
439 | - * @return boolean |
|
440 | - */ |
|
441 | - private function _mer_tables_exist() |
|
442 | - { |
|
443 | - if ($this->_mer_tables_exist === null) { |
|
444 | - global $wpdb; |
|
445 | - |
|
446 | - if ($wpdb->get_var("SHOW TABLES LIKE '{$this->_old_mer_table}'") != $this->_old_mer_table) { |
|
447 | - $this->_mer_tables_exist = false; |
|
448 | - } else { |
|
449 | - $this->_mer_tables_exist = true; |
|
450 | - } |
|
451 | - } |
|
452 | - return $this->_mer_tables_exist; |
|
453 | - } |
|
454 | - |
|
455 | - /** |
|
456 | - * Gets the 4.1 registration's status given the 3.1 attendee row. We consider |
|
457 | - * whether the event required pre-approval or not,a dn the 4.1 payment status. |
|
458 | - * @global type $wpdb |
|
459 | - * @param type $old_attendee_row |
|
460 | - * @return string |
|
461 | - */ |
|
462 | - private function _get_reg_status_for_old_payment_status($old_attendee_row) |
|
463 | - { |
|
464 | - // need event default reg status and if pre_approval was required |
|
465 | - global $wpdb; |
|
466 | - $event_required_pre_approval = $wpdb->get_var($wpdb->prepare("SELECT require_pre_approval FROM " . $wpdb->prefix . "events_detail WHERE id = %d", $old_attendee_row['event_id'])); |
|
467 | - return $this->get_migration_script()->convert_3_1_payment_status_to_4_1_STS_ID( |
|
468 | - $old_attendee_row['payment_status'], |
|
469 | - intval($event_required_pre_approval) && intval($old_attendee_row['pre_approve']) |
|
470 | - ); |
|
471 | - } |
|
472 | - |
|
473 | - /** |
|
474 | - * Adds however many rgistrations are indicated by the old attendee's QUANTITY field, |
|
475 | - * and returns an array of their IDs |
|
476 | - * @global type $wpdb |
|
477 | - * @param array $old_attendee |
|
478 | - * @param int $new_attendee_id |
|
479 | - * @param int $new_txn_id |
|
480 | - * @return array of new registratio ids |
|
481 | - */ |
|
482 | - private function _insert_new_registrations($old_attendee, $new_attendee_id, $new_txn_id) |
|
483 | - { |
|
484 | - global $wpdb; |
|
485 | - |
|
486 | - $STS_ID = $this->_get_reg_status_for_old_payment_status($old_attendee); |
|
487 | - $new_event_id = $this->get_migration_script()->get_mapping_new_pk($wpdb->prefix . 'events_detail', $old_attendee['event_id'], $wpdb->posts); |
|
488 | - if (!$new_event_id) { |
|
489 | - $this->add_error(sprintf(__("Could not find NEW event CPT ID for old event '%d' on old attendee %s", "event_espresso"), $old_attendee['event_id'], $this->_json_encode($old_attendee))); |
|
490 | - } |
|
491 | - |
|
492 | - $ticket_id = $this->_try_to_find_new_ticket_id($old_attendee, $new_event_id); |
|
493 | - if (!$ticket_id) { |
|
494 | - $ticket_id = $this->_insert_new_ticket_because_none_found($old_attendee, $new_event_id); |
|
495 | - $this->add_error(sprintf(__('Could not find a ticket for old attendee with id %d for new event %d, so created a new ticket with id %d', 'event_espresso'), $old_attendee['id'], $new_event_id, $ticket_id)); |
|
496 | - } |
|
497 | - $regs_on_this_row = intval($old_attendee['quantity']); |
|
498 | - $new_regs = array(); |
|
499 | - // 4 cases we need to account for: |
|
500 | - // 1 old attendee_details row with a quantity of X (no mer) |
|
501 | - // Y old attendee_details rows with a quantity of 1 (no mer) joined by their common registration_id |
|
502 | - // Y old attendee_details rows with a quantity of x (because of mer) |
|
503 | - // Y old attendee_details rows with a quantity of 1 (because of mer) joined by wp_events_multi_event_registration_id_group |
|
504 | - for ($count = 1; $count <= $regs_on_this_row; $count++) { |
|
505 | - // sum regs on older rows |
|
506 | - $regs_on_this_event_and_txn = $this->_sum_old_attendees_on_old_txn($old_attendee, true); |
|
507 | - $cols_n_values = array( |
|
508 | - 'EVT_ID' => $new_event_id, |
|
509 | - 'ATT_ID' => $new_attendee_id, |
|
510 | - 'TXN_ID' => $new_txn_id, |
|
511 | - 'TKT_ID' => $ticket_id, |
|
512 | - 'STS_ID' => $STS_ID, |
|
513 | - 'REG_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
514 | - 'REG_final_price' => $old_attendee['final_price'], |
|
515 | - 'REG_session' => substr($old_attendee['attendee_session'], 0, 44), |
|
516 | - 'REG_code' => sanitize_key($old_attendee['registration_id']), |
|
517 | - 'REG_url_link' => sanitize_key($old_attendee['registration_id'] . '-' . $count), |
|
518 | - 'REG_count' => $regs_on_this_event_and_txn + $count, |
|
519 | - 'REG_group_size' => $this->_sum_old_attendees_on_old_txn($old_attendee, false), |
|
520 | - 'REG_att_is_going' => true, |
|
521 | - 'REG_deleted' => false |
|
522 | - ); |
|
523 | - $datatypes = array( |
|
524 | - '%d',// EVT_ID |
|
525 | - '%d',// ATT_ID |
|
526 | - '%d',// TXN_ID |
|
527 | - '%d',// TKT_ID |
|
528 | - '%s',// STS_ID |
|
529 | - '%s',// REG_date |
|
530 | - '%f',// REG_final_price |
|
531 | - '%s',// REG_session |
|
532 | - '%s',// REG_code |
|
533 | - '%s',// REG_url_link |
|
534 | - '%d',// REG_count |
|
535 | - '%d',// REG_group_size |
|
536 | - '%d',// REG_att_is_going |
|
537 | - '%d',// REG_deleted |
|
538 | - ); |
|
539 | - $success = $wpdb->insert($this->_new_reg_table, $cols_n_values, $datatypes); |
|
540 | - if (!$success) { |
|
541 | - $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_reg_table, $cols_n_values, $datatypes)); |
|
542 | - return 0; |
|
543 | - } |
|
544 | - $cols_n_values['REG_ID'] = $wpdb->insert_id; |
|
545 | - $new_regs[] = $wpdb->insert_id; |
|
546 | - } |
|
547 | - $this->_add_regs_to_ticket_and_datetimes($ticket_id, count($new_regs), $STS_ID); |
|
548 | - return $new_regs; |
|
549 | - } |
|
550 | - |
|
551 | - /** |
|
552 | - * Increments the sold values on the ticket and its related datetimes by the amount sold, |
|
553 | - * which should be done directly after adding the rows. Yes this means we're constantly incrementing |
|
554 | - * the sold amounts as we go, and is less efficient than a single big query, |
|
555 | - * but its safer because we KNOW these regs have been added, rather than inferring |
|
556 | - * that they WILL be added (because the attendees stage runs nearly last during |
|
557 | - * the migration script) |
|
558 | - * @param type $new_ticket_id |
|
559 | - * @param type $sold |
|
560 | - * @param type $STS_ID |
|
561 | - * @return boolean whether they were successfully updated or not |
|
562 | - */ |
|
563 | - protected function _add_regs_to_ticket_and_datetimes($new_ticket_id, $quantity_sold, $STS_ID) |
|
564 | - { |
|
565 | - if ($STS_ID != 'RAP') { |
|
566 | - return true; |
|
567 | - } |
|
568 | - global $wpdb; |
|
569 | - $success = $wpdb->query($wpdb->prepare("UPDATE {$this->_new_ticket_table} SET TKT_sold=TKT_sold+%d WHERE TKT_ID=%d", $quantity_sold, $new_ticket_id)); |
|
570 | - if ($success) { |
|
571 | - // get the ticket's datetimes, and increment them too |
|
572 | - $success_update_dateimtes = $wpdb->query($wpdb->prepare("UPDATE {$this->_new_ticket_table} TKT |
|
216 | + $this->_old_mer_table = $wpdb->prefix . "events_multi_event_registration_id_group"; |
|
217 | + $this->_new_attendee_cpt_table = $wpdb->posts; |
|
218 | + $this->_new_attendee_meta_table = $wpdb->prefix . "esp_attendee_meta"; |
|
219 | + $this->_new_reg_table = $wpdb->prefix . "esp_registration"; |
|
220 | + $this->_new_transaction_table = $wpdb->prefix . "esp_transaction"; |
|
221 | + $this->_new_payment_table = $wpdb->prefix . "esp_payment"; |
|
222 | + $this->_new_line_table = $wpdb->prefix . "esp_line_item"; |
|
223 | + $this->_new_ticket_table = $wpdb->prefix . "esp_ticket"; |
|
224 | + $this->_new_ticket_datetime_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
225 | + $this->_new_datetime_table = $wpdb->prefix . "esp_datetime"; |
|
226 | + $this->_new_datetime_ticket_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
227 | + $this->_new_price_table = $wpdb->prefix . "esp_price"; |
|
228 | + $this->_new_ticket_price_table = $wpdb->prefix . "esp_ticket_price"; |
|
229 | + parent::__construct(); |
|
230 | + } |
|
231 | + |
|
232 | + protected function _migrate_old_row($old_row) |
|
233 | + { |
|
234 | + // first check if there's already a new attendee with similar characteristics |
|
235 | + $new_att_id = $this->_find_attendee_cpt_matching($old_row); |
|
236 | + if (!$new_att_id) { |
|
237 | + $new_att_id = $this->_insert_new_attendee_cpt($old_row); |
|
238 | + if (!$new_att_id) { |
|
239 | + // if we couldnt even make an attendee, abandon all hope |
|
240 | + return false; |
|
241 | + } |
|
242 | + $new_att_meta_id = $this->_insert_attendee_meta_row($old_row, $new_att_id); |
|
243 | + if ($new_att_meta_id) { |
|
244 | + $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_attendee_meta_table, $new_att_meta_id); |
|
245 | + } |
|
246 | + } |
|
247 | + $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_attendee_cpt_table, $new_att_id); |
|
248 | + |
|
249 | + $txn_id = $this->_insert_new_transaction($old_row); |
|
250 | + if (!$txn_id) { |
|
251 | + // if we couldnt make the transaction, also abandon all hope |
|
252 | + return false; |
|
253 | + } |
|
254 | + $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_transaction_table, $txn_id); |
|
255 | + $pay_id = $this->_insert_new_payment($old_row, $txn_id); |
|
256 | + if ($pay_id) { |
|
257 | + $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_payment_table, $pay_id); |
|
258 | + } |
|
259 | + |
|
260 | + |
|
261 | + // even if there was no payment, we can go ahead with adding the reg |
|
262 | + $new_regs = $this->_insert_new_registrations($old_row, $new_att_id, $txn_id); |
|
263 | + if ($new_regs) { |
|
264 | + $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_reg_table, $new_regs); |
|
265 | + } |
|
266 | + } |
|
267 | + |
|
268 | + /** |
|
269 | + * Checks if there's already an attendee CPT in the db that has the same |
|
270 | + * first and last name, and email. If so, returns its ID as an int. |
|
271 | + * @global type $wpdb |
|
272 | + * @param array $old_attendee |
|
273 | + * @return int |
|
274 | + */ |
|
275 | + private function _find_attendee_cpt_matching($old_attendee) |
|
276 | + { |
|
277 | + global $wpdb; |
|
278 | + $existing_attendee_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_new_attendee_cpt_table . " AS cpt INNER JOIN " . $this->_new_attendee_meta_table . " AS meta ON cpt.ID = meta.ATT_ID WHERE meta.ATT_fname = %s AND meta.ATT_lname = %s AND meta.ATT_email = %s LIMIT 1", $old_attendee['fname'], $old_attendee['lname'], $old_attendee['email'])); |
|
279 | + return intval($existing_attendee_id); |
|
280 | + } |
|
281 | + |
|
282 | + private function _insert_new_attendee_cpt($old_attendee) |
|
283 | + { |
|
284 | + global $wpdb; |
|
285 | + $cols_n_values = array( |
|
286 | + 'post_title' => stripslashes($old_attendee['fname'] . " " . $old_attendee['lname']),// ATT_full_name |
|
287 | + 'post_content' => '',// ATT_bio |
|
288 | + 'post_name' => sanitize_title($old_attendee['fname'] . "-" . $old_attendee['lname']),// ATT_slug |
|
289 | + 'post_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_created |
|
290 | + 'post_excerpt' => '',// ATT_short_bio |
|
291 | + 'post_modified' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_modified |
|
292 | + 'post_author' => 0,// ATT_author |
|
293 | + 'post_parent' => 0,// ATT_parent |
|
294 | + 'post_type' => 'espresso_attendees',// post_type |
|
295 | + 'post_status' => 'publish'// status |
|
296 | + ); |
|
297 | + $datatypes = array( |
|
298 | + '%s',// ATT_full_name |
|
299 | + '%s',// ATT_bio |
|
300 | + '%s',// ATT_slug |
|
301 | + '%s',// ATT_created |
|
302 | + '%s',// ATT_short_bio |
|
303 | + '%s',// ATT_modified |
|
304 | + '%d',// ATT_author |
|
305 | + '%d',// ATT_parent |
|
306 | + '%s',// post_type |
|
307 | + '%s',// status |
|
308 | + ); |
|
309 | + $success = $wpdb->insert($this->_new_attendee_cpt_table, $cols_n_values, $datatypes); |
|
310 | + if (!$success) { |
|
311 | + $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
|
312 | + return 0; |
|
313 | + } |
|
314 | + $new_id = $wpdb->insert_id; |
|
315 | + return $new_id; |
|
316 | + } |
|
317 | + |
|
318 | + private function _insert_attendee_meta_row($old_attendee, $new_attendee_cpt_id) |
|
319 | + { |
|
320 | + global $wpdb; |
|
321 | + // get the state and country ids from the old row |
|
322 | + try { |
|
323 | + $new_country = $this->get_migration_script()->get_or_create_country(stripslashes($old_attendee['country_id'])); |
|
324 | + $new_country_iso = $new_country['CNT_ISO']; |
|
325 | + } catch (EE_Error $exception) { |
|
326 | + $new_country_iso = $this->get_migration_script()->get_default_country_iso(); |
|
327 | + } |
|
328 | + try { |
|
329 | + $new_state = $this->get_migration_script()->get_or_create_state(stripslashes($old_attendee['state']), $new_country_iso); |
|
330 | + $new_state_id = $new_state['STA_ID']; |
|
331 | + } catch (EE_Error $exception) { |
|
332 | + $new_state_id = 0; |
|
333 | + } |
|
334 | + $cols_n_values = array( |
|
335 | + 'ATT_ID' => $new_attendee_cpt_id, |
|
336 | + 'ATT_fname' => stripslashes($old_attendee['fname']), |
|
337 | + 'ATT_lname' => stripslashes($old_attendee['lname']), |
|
338 | + 'ATT_address' => stripslashes($old_attendee['address']), |
|
339 | + 'ATT_address2' => stripslashes($old_attendee['address2']), |
|
340 | + 'ATT_city' => stripslashes($old_attendee['city']), |
|
341 | + 'STA_ID' => $new_state_id, |
|
342 | + 'CNT_ISO' => $new_country_iso, |
|
343 | + 'ATT_zip' => stripslashes($old_attendee['zip']), |
|
344 | + 'ATT_email' => stripslashes($old_attendee['email']), |
|
345 | + 'ATT_phone' => stripslashes($old_attendee['phone']), |
|
346 | + ); |
|
347 | + $datatypes = array( |
|
348 | + '%d',// ATT_ID |
|
349 | + '%s',// ATT_fname |
|
350 | + '%s',// ATT_lname |
|
351 | + '%s',// ATT_address |
|
352 | + '%s',// ATT_address2 |
|
353 | + '%s',// ATT_city |
|
354 | + '%d',// STA_ID |
|
355 | + '%s',// CNT_ISO |
|
356 | + '%s',// ATT_zip |
|
357 | + '%s',// ATT_email |
|
358 | + '%s',// ATT_phone |
|
359 | + ); |
|
360 | + $success = $wpdb->insert($this->_new_attendee_meta_table, $cols_n_values, $datatypes); |
|
361 | + if (!$success) { |
|
362 | + $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_meta_table, $cols_n_values, $datatypes)); |
|
363 | + return 0; |
|
364 | + } |
|
365 | + $new_id = $wpdb->insert_id; |
|
366 | + return $new_id; |
|
367 | + } |
|
368 | + |
|
369 | + /** |
|
370 | + * Note: we don't necessarily create a new transaction for each attendee row. |
|
371 | + * Only if the old attendee 'is_primary' is true; otherwise we find the old attendee row that |
|
372 | + * 'is_primary' and has the same 'txn_id', then we return ITS new transaction id |
|
373 | + * @global type $wpdb |
|
374 | + * @param type $old_attendee |
|
375 | + * @return int new transaction id |
|
376 | + */ |
|
377 | + private function _insert_new_transaction($old_attendee) |
|
378 | + { |
|
379 | + global $wpdb; |
|
380 | + |
|
381 | + // first: let's check for an existing transaction for this old attendee |
|
382 | + if (intval($old_attendee['is_primary'])) {// primary attendee, so create txn |
|
383 | + $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($old_attendee['id']), $this->_new_transaction_table); |
|
384 | + } else { // non-primary attendee, so find its primary attendee's transaction |
|
385 | + $primary_attendee_old_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_old_table . " WHERE is_primary=1 and registration_id=%s", $old_attendee['registration_id'])); |
|
386 | + if (!$primary_attendee_old_id) { |
|
387 | + $primary_attendee = $this->_find_mer_primary_attendee_using_mer_tables($old_attendee['registration_id']); |
|
388 | + $primary_attendee_old_id = is_array($primary_attendee) ? $primary_attendee['id'] : null; |
|
389 | + } |
|
390 | + $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($primary_attendee_old_id), $this->_new_transaction_table); |
|
391 | + if (!$txn_id) { |
|
392 | + $this->add_error(sprintf(__("Could not find primary attendee's new transaction. Current attendee is: %s, we think the 3.1 primary attendee for it has id %d, but there's no 4.1 transaction for that primary attendee id.", "event_espresso"), $this->_json_encode($old_attendee), $primary_attendee_old_id)); |
|
393 | + $txn_id = 0; |
|
394 | + } |
|
395 | + } |
|
396 | + // if there isn't yet a transaction row for this, create one |
|
397 | + // (so even if it was a non-primary attendee with no EE3 primary attendee, |
|
398 | + // it ought to have SOME transaction, so we'll make one) |
|
399 | + if (!$txn_id) { |
|
400 | + // maps 3.1 payment stati onto 4.1 transaction stati |
|
401 | + $txn_status_mapping = array( |
|
402 | + 'Completed' => 'TCM', |
|
403 | + 'Pending' => 'TIN', |
|
404 | + 'Payment Declined' => 'TIN', |
|
405 | + 'Incomplete' => 'TIN', |
|
406 | + 'Not Completed' => 'TIN', |
|
407 | + 'Cancelled' => 'TIN', |
|
408 | + 'Declined' => 'TIN' |
|
409 | + ); |
|
410 | + $STS_ID = isset($txn_status_mapping[ $old_attendee['payment_status'] ]) ? $txn_status_mapping[ $old_attendee['payment_status'] ] : 'TIN'; |
|
411 | + $cols_n_values = array( |
|
412 | + 'TXN_timestamp' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
413 | + 'TXN_total' => floatval($old_attendee['total_cost']), |
|
414 | + 'TXN_paid' => floatval($old_attendee['amount_pd']), |
|
415 | + 'STS_ID' => $STS_ID, |
|
416 | + 'TXN_hash_salt' => $old_attendee['hashSalt'] |
|
417 | + ); |
|
418 | + $datatypes = array( |
|
419 | + '%s',// TXN_timestamp |
|
420 | + '%f',// TXN_total |
|
421 | + '%f',// TXN_paid |
|
422 | + '%s',// STS_ID |
|
423 | + '%s',// TXN_hash_salt |
|
424 | + ); |
|
425 | + $success = $wpdb->insert($this->_new_transaction_table, $cols_n_values, $datatypes); |
|
426 | + if (!$success) { |
|
427 | + $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_transaction_table, $cols_n_values, $datatypes)); |
|
428 | + return 0; |
|
429 | + } |
|
430 | + $txn_id = $wpdb->insert_id; |
|
431 | + } |
|
432 | + |
|
433 | + return $txn_id; |
|
434 | + } |
|
435 | + |
|
436 | + /** |
|
437 | + * Detects if the MER tables exist |
|
438 | + * @global type $wpdb |
|
439 | + * @return boolean |
|
440 | + */ |
|
441 | + private function _mer_tables_exist() |
|
442 | + { |
|
443 | + if ($this->_mer_tables_exist === null) { |
|
444 | + global $wpdb; |
|
445 | + |
|
446 | + if ($wpdb->get_var("SHOW TABLES LIKE '{$this->_old_mer_table}'") != $this->_old_mer_table) { |
|
447 | + $this->_mer_tables_exist = false; |
|
448 | + } else { |
|
449 | + $this->_mer_tables_exist = true; |
|
450 | + } |
|
451 | + } |
|
452 | + return $this->_mer_tables_exist; |
|
453 | + } |
|
454 | + |
|
455 | + /** |
|
456 | + * Gets the 4.1 registration's status given the 3.1 attendee row. We consider |
|
457 | + * whether the event required pre-approval or not,a dn the 4.1 payment status. |
|
458 | + * @global type $wpdb |
|
459 | + * @param type $old_attendee_row |
|
460 | + * @return string |
|
461 | + */ |
|
462 | + private function _get_reg_status_for_old_payment_status($old_attendee_row) |
|
463 | + { |
|
464 | + // need event default reg status and if pre_approval was required |
|
465 | + global $wpdb; |
|
466 | + $event_required_pre_approval = $wpdb->get_var($wpdb->prepare("SELECT require_pre_approval FROM " . $wpdb->prefix . "events_detail WHERE id = %d", $old_attendee_row['event_id'])); |
|
467 | + return $this->get_migration_script()->convert_3_1_payment_status_to_4_1_STS_ID( |
|
468 | + $old_attendee_row['payment_status'], |
|
469 | + intval($event_required_pre_approval) && intval($old_attendee_row['pre_approve']) |
|
470 | + ); |
|
471 | + } |
|
472 | + |
|
473 | + /** |
|
474 | + * Adds however many rgistrations are indicated by the old attendee's QUANTITY field, |
|
475 | + * and returns an array of their IDs |
|
476 | + * @global type $wpdb |
|
477 | + * @param array $old_attendee |
|
478 | + * @param int $new_attendee_id |
|
479 | + * @param int $new_txn_id |
|
480 | + * @return array of new registratio ids |
|
481 | + */ |
|
482 | + private function _insert_new_registrations($old_attendee, $new_attendee_id, $new_txn_id) |
|
483 | + { |
|
484 | + global $wpdb; |
|
485 | + |
|
486 | + $STS_ID = $this->_get_reg_status_for_old_payment_status($old_attendee); |
|
487 | + $new_event_id = $this->get_migration_script()->get_mapping_new_pk($wpdb->prefix . 'events_detail', $old_attendee['event_id'], $wpdb->posts); |
|
488 | + if (!$new_event_id) { |
|
489 | + $this->add_error(sprintf(__("Could not find NEW event CPT ID for old event '%d' on old attendee %s", "event_espresso"), $old_attendee['event_id'], $this->_json_encode($old_attendee))); |
|
490 | + } |
|
491 | + |
|
492 | + $ticket_id = $this->_try_to_find_new_ticket_id($old_attendee, $new_event_id); |
|
493 | + if (!$ticket_id) { |
|
494 | + $ticket_id = $this->_insert_new_ticket_because_none_found($old_attendee, $new_event_id); |
|
495 | + $this->add_error(sprintf(__('Could not find a ticket for old attendee with id %d for new event %d, so created a new ticket with id %d', 'event_espresso'), $old_attendee['id'], $new_event_id, $ticket_id)); |
|
496 | + } |
|
497 | + $regs_on_this_row = intval($old_attendee['quantity']); |
|
498 | + $new_regs = array(); |
|
499 | + // 4 cases we need to account for: |
|
500 | + // 1 old attendee_details row with a quantity of X (no mer) |
|
501 | + // Y old attendee_details rows with a quantity of 1 (no mer) joined by their common registration_id |
|
502 | + // Y old attendee_details rows with a quantity of x (because of mer) |
|
503 | + // Y old attendee_details rows with a quantity of 1 (because of mer) joined by wp_events_multi_event_registration_id_group |
|
504 | + for ($count = 1; $count <= $regs_on_this_row; $count++) { |
|
505 | + // sum regs on older rows |
|
506 | + $regs_on_this_event_and_txn = $this->_sum_old_attendees_on_old_txn($old_attendee, true); |
|
507 | + $cols_n_values = array( |
|
508 | + 'EVT_ID' => $new_event_id, |
|
509 | + 'ATT_ID' => $new_attendee_id, |
|
510 | + 'TXN_ID' => $new_txn_id, |
|
511 | + 'TKT_ID' => $ticket_id, |
|
512 | + 'STS_ID' => $STS_ID, |
|
513 | + 'REG_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
514 | + 'REG_final_price' => $old_attendee['final_price'], |
|
515 | + 'REG_session' => substr($old_attendee['attendee_session'], 0, 44), |
|
516 | + 'REG_code' => sanitize_key($old_attendee['registration_id']), |
|
517 | + 'REG_url_link' => sanitize_key($old_attendee['registration_id'] . '-' . $count), |
|
518 | + 'REG_count' => $regs_on_this_event_and_txn + $count, |
|
519 | + 'REG_group_size' => $this->_sum_old_attendees_on_old_txn($old_attendee, false), |
|
520 | + 'REG_att_is_going' => true, |
|
521 | + 'REG_deleted' => false |
|
522 | + ); |
|
523 | + $datatypes = array( |
|
524 | + '%d',// EVT_ID |
|
525 | + '%d',// ATT_ID |
|
526 | + '%d',// TXN_ID |
|
527 | + '%d',// TKT_ID |
|
528 | + '%s',// STS_ID |
|
529 | + '%s',// REG_date |
|
530 | + '%f',// REG_final_price |
|
531 | + '%s',// REG_session |
|
532 | + '%s',// REG_code |
|
533 | + '%s',// REG_url_link |
|
534 | + '%d',// REG_count |
|
535 | + '%d',// REG_group_size |
|
536 | + '%d',// REG_att_is_going |
|
537 | + '%d',// REG_deleted |
|
538 | + ); |
|
539 | + $success = $wpdb->insert($this->_new_reg_table, $cols_n_values, $datatypes); |
|
540 | + if (!$success) { |
|
541 | + $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_reg_table, $cols_n_values, $datatypes)); |
|
542 | + return 0; |
|
543 | + } |
|
544 | + $cols_n_values['REG_ID'] = $wpdb->insert_id; |
|
545 | + $new_regs[] = $wpdb->insert_id; |
|
546 | + } |
|
547 | + $this->_add_regs_to_ticket_and_datetimes($ticket_id, count($new_regs), $STS_ID); |
|
548 | + return $new_regs; |
|
549 | + } |
|
550 | + |
|
551 | + /** |
|
552 | + * Increments the sold values on the ticket and its related datetimes by the amount sold, |
|
553 | + * which should be done directly after adding the rows. Yes this means we're constantly incrementing |
|
554 | + * the sold amounts as we go, and is less efficient than a single big query, |
|
555 | + * but its safer because we KNOW these regs have been added, rather than inferring |
|
556 | + * that they WILL be added (because the attendees stage runs nearly last during |
|
557 | + * the migration script) |
|
558 | + * @param type $new_ticket_id |
|
559 | + * @param type $sold |
|
560 | + * @param type $STS_ID |
|
561 | + * @return boolean whether they were successfully updated or not |
|
562 | + */ |
|
563 | + protected function _add_regs_to_ticket_and_datetimes($new_ticket_id, $quantity_sold, $STS_ID) |
|
564 | + { |
|
565 | + if ($STS_ID != 'RAP') { |
|
566 | + return true; |
|
567 | + } |
|
568 | + global $wpdb; |
|
569 | + $success = $wpdb->query($wpdb->prepare("UPDATE {$this->_new_ticket_table} SET TKT_sold=TKT_sold+%d WHERE TKT_ID=%d", $quantity_sold, $new_ticket_id)); |
|
570 | + if ($success) { |
|
571 | + // get the ticket's datetimes, and increment them too |
|
572 | + $success_update_dateimtes = $wpdb->query($wpdb->prepare("UPDATE {$this->_new_ticket_table} TKT |
|
573 | 573 | INNER JOIN {$this->_new_ticket_datetime_table} as DTK ON TKT.TKT_ID = DTK.TKT_ID |
574 | 574 | INNER JOIN {$this->_new_datetime_table} as DTT ON DTK.DTT_ID = DTT.DTT_ID |
575 | 575 | SET DTT.DTT_sold = DTT.DTT_sold + %d WHERE TKT.TKT_ID = %d", $quantity_sold, $new_ticket_id)); |
576 | - if (!$success_update_dateimtes) { |
|
577 | - $this->add_error(sprintf(__("Could not update datetimes related to ticket with ID %d's TKT_sold by %d because %s", "event_espresso"), $new_ticket_id, $quantity_sold, $wpdb->last_error)); |
|
578 | - } |
|
579 | - } else { |
|
580 | - $this->add_error(sprintf(__("Could not update ticket with ID %d's TKT_sold by %d because %s", "event_espresso"), $new_ticket_id, $quantity_sold, $wpdb->last_error)); |
|
581 | - } |
|
582 | - return true; |
|
583 | - } |
|
584 | - |
|
585 | - /** |
|
586 | - * Makes a best guess at which ticket is the one the attendee purchased. |
|
587 | - * Obviously, the old attendee's event_id narrows it down quite a bit; |
|
588 | - * then the old attendee's orig_price and event_time, and price_option can uniquely identify the ticket |
|
589 | - * however, if we don't find an exact match, see if any of those conditions match; |
|
590 | - * and lastly if none of that works, just use the first ticket for the event we find |
|
591 | - * @param array $old_attendee |
|
592 | - */ |
|
593 | - private function _try_to_find_new_ticket_id($old_attendee, $new_event_id) |
|
594 | - { |
|
595 | - global $wpdb; |
|
596 | - $tickets_table = $this->_new_ticket_table; |
|
597 | - $datetime_tickets_table = $this->_new_ticket_datetime_table; |
|
598 | - $datetime_table = $this->_new_datetime_table; |
|
599 | - |
|
600 | - $old_att_price_option = $old_attendee['price_option']; |
|
601 | - $old_att_price = floatval($old_attendee['orig_price']); |
|
602 | - |
|
603 | - $old_att_start_date = $old_attendee['start_date']; |
|
604 | - $old_att_start_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
605 | - $old_att_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_start_date $old_att_start_time:00"); |
|
606 | - // add all conditions to an array from which we can SHIFT conditions off in order to widen our search |
|
607 | - // the most important condition should be last, as it will be array_shift'ed off last |
|
608 | - $conditions = array( |
|
609 | - $wpdb->prepare("$datetime_table.DTT_EVT_start = %s", $old_att_datetime),// times match? |
|
610 | - $wpdb->prepare("$tickets_table.TKT_price = %f", $old_att_price),// prices match? |
|
611 | - $wpdb->prepare("$tickets_table.TKT_name = %s", $old_att_price_option),// names match? |
|
612 | - $wpdb->prepare("$datetime_table.EVT_ID = %d", $new_event_id),// events match? |
|
613 | - ); |
|
614 | - $select_and_join_part = "SELECT $tickets_table.TKT_ID FROM $tickets_table INNER JOIN |
|
576 | + if (!$success_update_dateimtes) { |
|
577 | + $this->add_error(sprintf(__("Could not update datetimes related to ticket with ID %d's TKT_sold by %d because %s", "event_espresso"), $new_ticket_id, $quantity_sold, $wpdb->last_error)); |
|
578 | + } |
|
579 | + } else { |
|
580 | + $this->add_error(sprintf(__("Could not update ticket with ID %d's TKT_sold by %d because %s", "event_espresso"), $new_ticket_id, $quantity_sold, $wpdb->last_error)); |
|
581 | + } |
|
582 | + return true; |
|
583 | + } |
|
584 | + |
|
585 | + /** |
|
586 | + * Makes a best guess at which ticket is the one the attendee purchased. |
|
587 | + * Obviously, the old attendee's event_id narrows it down quite a bit; |
|
588 | + * then the old attendee's orig_price and event_time, and price_option can uniquely identify the ticket |
|
589 | + * however, if we don't find an exact match, see if any of those conditions match; |
|
590 | + * and lastly if none of that works, just use the first ticket for the event we find |
|
591 | + * @param array $old_attendee |
|
592 | + */ |
|
593 | + private function _try_to_find_new_ticket_id($old_attendee, $new_event_id) |
|
594 | + { |
|
595 | + global $wpdb; |
|
596 | + $tickets_table = $this->_new_ticket_table; |
|
597 | + $datetime_tickets_table = $this->_new_ticket_datetime_table; |
|
598 | + $datetime_table = $this->_new_datetime_table; |
|
599 | + |
|
600 | + $old_att_price_option = $old_attendee['price_option']; |
|
601 | + $old_att_price = floatval($old_attendee['orig_price']); |
|
602 | + |
|
603 | + $old_att_start_date = $old_attendee['start_date']; |
|
604 | + $old_att_start_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
605 | + $old_att_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_start_date $old_att_start_time:00"); |
|
606 | + // add all conditions to an array from which we can SHIFT conditions off in order to widen our search |
|
607 | + // the most important condition should be last, as it will be array_shift'ed off last |
|
608 | + $conditions = array( |
|
609 | + $wpdb->prepare("$datetime_table.DTT_EVT_start = %s", $old_att_datetime),// times match? |
|
610 | + $wpdb->prepare("$tickets_table.TKT_price = %f", $old_att_price),// prices match? |
|
611 | + $wpdb->prepare("$tickets_table.TKT_name = %s", $old_att_price_option),// names match? |
|
612 | + $wpdb->prepare("$datetime_table.EVT_ID = %d", $new_event_id),// events match? |
|
613 | + ); |
|
614 | + $select_and_join_part = "SELECT $tickets_table.TKT_ID FROM $tickets_table INNER JOIN |
|
615 | 615 | $datetime_tickets_table ON $tickets_table.TKT_ID = $datetime_tickets_table.TKT_ID INNER JOIN |
616 | 616 | $datetime_table ON $datetime_tickets_table.DTT_ID = $datetime_table.DTT_ID"; |
617 | - // start running queries, widening search each time by removing a condition |
|
618 | - do { |
|
619 | - $full_query = $select_and_join_part . " WHERE " . implode(" AND ", $conditions) . " LIMIT 1"; |
|
620 | - $ticket_id_found = $wpdb->get_var($full_query); |
|
621 | - array_shift($conditions); |
|
622 | - } while (!$ticket_id_found && $conditions); |
|
623 | - return $ticket_id_found; |
|
624 | - } |
|
625 | - |
|
626 | - /** |
|
627 | - * If we couldn't find a 4.1 ticket for a 3.1 attendee row, this function creates one; |
|
628 | - * and it also tries to find a datetime that works, and a inserts a price, and associates |
|
629 | - * the new ticket to that datetime and price. |
|
630 | - * @return int ticket id |
|
631 | - */ |
|
632 | - private function _insert_new_ticket_because_none_found($old_attendee, $new_event_id) |
|
633 | - { |
|
634 | - global $wpdb; |
|
635 | - $old_att_price_option = $old_attendee['price_option']; |
|
636 | - $old_att_price = floatval($old_attendee['orig_price']); |
|
637 | - |
|
638 | - $old_att_start_date = $old_attendee['start_date']; |
|
639 | - $old_att_start_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
640 | - $old_att_start_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_start_date $old_att_start_time:00"); |
|
641 | - |
|
642 | - |
|
643 | - // insert new datetime unless we find one |
|
644 | - $datetime_id = $wpdb->get_var($wpdb->prepare("SELECT DTT_ID FROM " . $this->_new_datetime_table . " WHERE DTT_EVT_start=%s AND EVT_ID=%d LIMIT 1", $old_att_start_datetime, $new_event_id), ARRAY_A); |
|
645 | - if (!$datetime_id) { |
|
646 | - $old_att_end_date = $old_attendee['start_date']; |
|
647 | - $old_att_end_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
648 | - $old_att_end_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_end_date $old_att_end_time:00"); |
|
649 | - $wpdb->insert( |
|
650 | - $this->_new_datetime_table, |
|
651 | - array( |
|
652 | - 'EVT_ID' => $new_event_id, |
|
653 | - 'DTT_EVT_start' => $old_att_start_datetime, |
|
654 | - 'DTT_EVT_end' => $old_att_end_datetime, |
|
655 | - 'DTT_deleted' => true |
|
656 | - ), |
|
657 | - array( |
|
658 | - '%d',// EVT_ID |
|
659 | - '%s',// DTT_EVT_start |
|
660 | - '%s',// DTT_EVT_end |
|
661 | - '%d',// DTT_deleted |
|
662 | - ) |
|
663 | - ); |
|
664 | - $datetime_id = $wpdb->insert_id; |
|
665 | - } |
|
666 | - |
|
667 | - // insert new ticket |
|
668 | - $success = $wpdb->insert( |
|
669 | - $wpdb->prefix . 'esp_ticket', |
|
670 | - array( |
|
671 | - 'TKT_name' => $old_att_price_option, |
|
672 | - 'TKT_qty' => -1, |
|
673 | - 'TKT_price' => $old_att_price, |
|
674 | - 'TKT_start_date' => $old_att_start_datetime,// we really have no clue what the time should be, but at least it was available when they attended |
|
675 | - 'TKT_end_date' => $old_att_end_datetime, |
|
676 | - |
|
677 | - ), |
|
678 | - array( |
|
679 | - '%s',// name |
|
680 | - '%d',// qty |
|
681 | - '%d',// price |
|
682 | - '%s',// start_date |
|
683 | - '%s',// end_date |
|
684 | - ) |
|
685 | - ); |
|
686 | - $ticket_id = $wpdb->insert_id; |
|
687 | - // associate the ticket with the datetime we found earlier |
|
688 | - $wpdb->insert( |
|
689 | - $this->_new_datetime_ticket_table, |
|
690 | - array( |
|
691 | - 'DTT_ID' => $datetime_id, |
|
692 | - 'TKT_ID' => $ticket_id |
|
693 | - ), |
|
694 | - array( |
|
695 | - '%d',// DTT_ID |
|
696 | - '%d',// TKT_ID |
|
697 | - ) |
|
698 | - ); |
|
699 | - // insert new price |
|
700 | - $wpdb->insert( |
|
701 | - $this->_new_price_table, |
|
702 | - array( |
|
703 | - 'PRC_amount' => $old_att_price, |
|
704 | - 'PRT_ID' => EE_DMS_4_1_0_prices::price_type_base, |
|
705 | - 'PRC_name' => $old_att_price_option, |
|
706 | - 'PRC_deleted' => true |
|
707 | - ), |
|
708 | - array( |
|
709 | - '%d',// PRC_amount |
|
710 | - '%d',// PRT_ID |
|
711 | - '%s',// PRC_name |
|
712 | - '%d',// PRC_deleted |
|
713 | - ) |
|
714 | - ); |
|
715 | - $price_id = $wpdb->insert_id; |
|
716 | - // associate the price to the ticket |
|
717 | - $wpdb->insert( |
|
718 | - $this->_new_ticket_price_table, |
|
719 | - array( |
|
720 | - 'TKT_ID' => $ticket_id, |
|
721 | - 'PRC_ID' => $price_id |
|
722 | - ), |
|
723 | - array( |
|
724 | - '%d',// TKT_ID |
|
725 | - '%d',// PRC_ID |
|
726 | - ) |
|
727 | - ); |
|
728 | - return $ticket_id; |
|
729 | - } |
|
730 | - |
|
731 | - /** |
|
732 | - * Counts all the registrations on this transaction. If $count_only_older is TRUE then returns the number added SO FAR (ie, |
|
733 | - * only considers attendee rows with an ID less than this one's), but if $count_only_older is FALSe returns ALL |
|
734 | - * @global type $wpdb |
|
735 | - * @param array $old_attendee_row |
|
736 | - * @param boolean $count_only_older true if you want the running count (ie, the total up to this row), and false if you want ALL |
|
737 | - * @return int |
|
738 | - */ |
|
739 | - private function _sum_old_attendees_on_old_txn($old_attendee_row, $count_only_older = false) |
|
740 | - { |
|
741 | - global $wpdb; |
|
742 | - $count_only_older_sql = $count_only_older ? $wpdb->prepare(" AND id<%d", $old_attendee_row['id']) : ''; |
|
743 | - $count = intval($wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM " . $this->_old_table . " WHERE registration_id=%s $count_only_older_sql", $old_attendee_row['registration_id']))); |
|
744 | - |
|
745 | - if ($this->_mer_tables_exist()) { |
|
746 | - // if MER exists, then its a little tricky. |
|
747 | - // when users registered by adding items to the cart, and it was a |
|
748 | - // group registration requiring additional attendee INFO, then the attendee rows |
|
749 | - // DO NOT have the same registration_id (although they probably should have) |
|
750 | - // they are related just like MER attendee rows are related, through the MER group table |
|
751 | - // BUT we want to count all the MER attendee rows for the same registration |
|
752 | - $primary_attendee = $this->_find_mer_primary_attendee_using_mer_tables($old_attendee_row['registration_id']); |
|
753 | - |
|
754 | - $count_using_mer_table = $wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM {$this->_old_table} att INNER JOIN {$this->_old_mer_table} mer ON att.registration_id = mer.registration_id WHERE att.event_id=%d AND mer.primary_registration_id = %s $count_only_older_sql", $old_attendee_row['event_id'], $primary_attendee['registration_id'])); |
|
755 | - $count = max($count_using_mer_table, $count); |
|
756 | - } |
|
757 | - return $count; |
|
758 | - } |
|
759 | - |
|
760 | - private function _insert_new_payment($old_attendee, $new_txn_id) |
|
761 | - { |
|
762 | - global $wpdb; |
|
763 | - // only add a payment for primary attendees |
|
764 | - $old_pay_stati_indicating_no_payment = array('Pending', 'Incomplete', 'Not Completed'); |
|
765 | - // if this is for a primary 3.1 attendee which WASN'T free and has a completed, cancelled, or declined payment... |
|
766 | - if (intval($old_attendee['is_primary']) && floatval($old_attendee['total_cost']) && !in_array($old_attendee['payment_status'], $old_pay_stati_indicating_no_payment)) { |
|
767 | - $pay_status_mapping = array( |
|
768 | - 'Completed' => 'PAP', |
|
769 | - 'Payment Declined' => 'PDC', |
|
770 | - 'Cancelled' => 'PCN', |
|
771 | - 'Declined' => 'PDC' |
|
772 | - ); |
|
773 | - $by_admin = $old_attendee['payment'] == 'Admin'; |
|
774 | - $STS_ID = isset($pay_status_mapping[ $old_attendee['payment_status'] ]) ? $pay_status_mapping[ $old_attendee['payment_status'] ] : 'PFL';// IE, if we don't recognize the status, assume payment failed |
|
775 | - $cols_n_values = array( |
|
776 | - 'TXN_ID' => $new_txn_id, |
|
777 | - 'STS_ID' => $STS_ID, |
|
778 | - 'PAY_timestamp' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
779 | - 'PAY_method' => 'CART', |
|
780 | - 'PAY_amount' => $old_attendee['amount_pd'], |
|
781 | - 'PAY_gateway' => $old_attendee['txn_type'], |
|
782 | - 'PAY_gateway_response' => '', |
|
783 | - 'PAY_txn_id_chq_nmbr' => substr($old_attendee['txn_id'], 0, 32), |
|
784 | - 'PAY_via_admin' => $by_admin, |
|
785 | - 'PAY_details' => $old_attendee['transaction_details'] |
|
786 | - |
|
787 | - ); |
|
788 | - $datatypes = array( |
|
789 | - '%d',// TXN_Id |
|
790 | - '%s',// STS_ID |
|
791 | - '%s',// PAY_timestamp |
|
792 | - '%s',// PAY_method |
|
793 | - '%f',// PAY_amount |
|
794 | - '%s',// PAY_gateway |
|
795 | - '%s',// PAY_gateway_response |
|
796 | - '%s',// PAY_txn_id_chq_nmbr |
|
797 | - '%d',// PAY_via_admin |
|
798 | - '%s',// PAY_details |
|
799 | - ); |
|
800 | - $success = $wpdb->insert($this->_new_payment_table, $cols_n_values, $datatypes); |
|
801 | - if (!$success) { |
|
802 | - $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
|
803 | - return 0; |
|
804 | - } |
|
805 | - $new_id = $wpdb->insert_id; |
|
806 | - return $new_id; |
|
807 | - } else { |
|
808 | - return 0; |
|
809 | - } |
|
810 | - } |
|
811 | - |
|
812 | - /** |
|
813 | - * If MER is active, if you want ot fin dthe other registrations on that attendee row |
|
814 | - * @global type $wpdb |
|
815 | - * @param type $old_registration_id |
|
816 | - * @return array |
|
817 | - */ |
|
818 | - private function _find_mer_primary_attendee_using_mer_tables($old_registration_id) |
|
819 | - { |
|
820 | - if (!$this->_mer_tables_exist()) { |
|
821 | - return false; |
|
822 | - } |
|
823 | - global $wpdb; |
|
824 | - $old_att_for_primary_reg = $wpdb->get_row($wpdb->prepare("SELECT * FROM {$this->_old_mer_table} AS mer INNER JOIN {$this->_old_table} AS att ON mer.primary_registration_id = att.registration_id WHERE mer.registration_id=%s LIMIT 1", $old_registration_id), ARRAY_A); |
|
825 | - return $old_att_for_primary_reg; |
|
826 | - } |
|
617 | + // start running queries, widening search each time by removing a condition |
|
618 | + do { |
|
619 | + $full_query = $select_and_join_part . " WHERE " . implode(" AND ", $conditions) . " LIMIT 1"; |
|
620 | + $ticket_id_found = $wpdb->get_var($full_query); |
|
621 | + array_shift($conditions); |
|
622 | + } while (!$ticket_id_found && $conditions); |
|
623 | + return $ticket_id_found; |
|
624 | + } |
|
625 | + |
|
626 | + /** |
|
627 | + * If we couldn't find a 4.1 ticket for a 3.1 attendee row, this function creates one; |
|
628 | + * and it also tries to find a datetime that works, and a inserts a price, and associates |
|
629 | + * the new ticket to that datetime and price. |
|
630 | + * @return int ticket id |
|
631 | + */ |
|
632 | + private function _insert_new_ticket_because_none_found($old_attendee, $new_event_id) |
|
633 | + { |
|
634 | + global $wpdb; |
|
635 | + $old_att_price_option = $old_attendee['price_option']; |
|
636 | + $old_att_price = floatval($old_attendee['orig_price']); |
|
637 | + |
|
638 | + $old_att_start_date = $old_attendee['start_date']; |
|
639 | + $old_att_start_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
640 | + $old_att_start_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_start_date $old_att_start_time:00"); |
|
641 | + |
|
642 | + |
|
643 | + // insert new datetime unless we find one |
|
644 | + $datetime_id = $wpdb->get_var($wpdb->prepare("SELECT DTT_ID FROM " . $this->_new_datetime_table . " WHERE DTT_EVT_start=%s AND EVT_ID=%d LIMIT 1", $old_att_start_datetime, $new_event_id), ARRAY_A); |
|
645 | + if (!$datetime_id) { |
|
646 | + $old_att_end_date = $old_attendee['start_date']; |
|
647 | + $old_att_end_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
|
648 | + $old_att_end_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_end_date $old_att_end_time:00"); |
|
649 | + $wpdb->insert( |
|
650 | + $this->_new_datetime_table, |
|
651 | + array( |
|
652 | + 'EVT_ID' => $new_event_id, |
|
653 | + 'DTT_EVT_start' => $old_att_start_datetime, |
|
654 | + 'DTT_EVT_end' => $old_att_end_datetime, |
|
655 | + 'DTT_deleted' => true |
|
656 | + ), |
|
657 | + array( |
|
658 | + '%d',// EVT_ID |
|
659 | + '%s',// DTT_EVT_start |
|
660 | + '%s',// DTT_EVT_end |
|
661 | + '%d',// DTT_deleted |
|
662 | + ) |
|
663 | + ); |
|
664 | + $datetime_id = $wpdb->insert_id; |
|
665 | + } |
|
666 | + |
|
667 | + // insert new ticket |
|
668 | + $success = $wpdb->insert( |
|
669 | + $wpdb->prefix . 'esp_ticket', |
|
670 | + array( |
|
671 | + 'TKT_name' => $old_att_price_option, |
|
672 | + 'TKT_qty' => -1, |
|
673 | + 'TKT_price' => $old_att_price, |
|
674 | + 'TKT_start_date' => $old_att_start_datetime,// we really have no clue what the time should be, but at least it was available when they attended |
|
675 | + 'TKT_end_date' => $old_att_end_datetime, |
|
676 | + |
|
677 | + ), |
|
678 | + array( |
|
679 | + '%s',// name |
|
680 | + '%d',// qty |
|
681 | + '%d',// price |
|
682 | + '%s',// start_date |
|
683 | + '%s',// end_date |
|
684 | + ) |
|
685 | + ); |
|
686 | + $ticket_id = $wpdb->insert_id; |
|
687 | + // associate the ticket with the datetime we found earlier |
|
688 | + $wpdb->insert( |
|
689 | + $this->_new_datetime_ticket_table, |
|
690 | + array( |
|
691 | + 'DTT_ID' => $datetime_id, |
|
692 | + 'TKT_ID' => $ticket_id |
|
693 | + ), |
|
694 | + array( |
|
695 | + '%d',// DTT_ID |
|
696 | + '%d',// TKT_ID |
|
697 | + ) |
|
698 | + ); |
|
699 | + // insert new price |
|
700 | + $wpdb->insert( |
|
701 | + $this->_new_price_table, |
|
702 | + array( |
|
703 | + 'PRC_amount' => $old_att_price, |
|
704 | + 'PRT_ID' => EE_DMS_4_1_0_prices::price_type_base, |
|
705 | + 'PRC_name' => $old_att_price_option, |
|
706 | + 'PRC_deleted' => true |
|
707 | + ), |
|
708 | + array( |
|
709 | + '%d',// PRC_amount |
|
710 | + '%d',// PRT_ID |
|
711 | + '%s',// PRC_name |
|
712 | + '%d',// PRC_deleted |
|
713 | + ) |
|
714 | + ); |
|
715 | + $price_id = $wpdb->insert_id; |
|
716 | + // associate the price to the ticket |
|
717 | + $wpdb->insert( |
|
718 | + $this->_new_ticket_price_table, |
|
719 | + array( |
|
720 | + 'TKT_ID' => $ticket_id, |
|
721 | + 'PRC_ID' => $price_id |
|
722 | + ), |
|
723 | + array( |
|
724 | + '%d',// TKT_ID |
|
725 | + '%d',// PRC_ID |
|
726 | + ) |
|
727 | + ); |
|
728 | + return $ticket_id; |
|
729 | + } |
|
730 | + |
|
731 | + /** |
|
732 | + * Counts all the registrations on this transaction. If $count_only_older is TRUE then returns the number added SO FAR (ie, |
|
733 | + * only considers attendee rows with an ID less than this one's), but if $count_only_older is FALSe returns ALL |
|
734 | + * @global type $wpdb |
|
735 | + * @param array $old_attendee_row |
|
736 | + * @param boolean $count_only_older true if you want the running count (ie, the total up to this row), and false if you want ALL |
|
737 | + * @return int |
|
738 | + */ |
|
739 | + private function _sum_old_attendees_on_old_txn($old_attendee_row, $count_only_older = false) |
|
740 | + { |
|
741 | + global $wpdb; |
|
742 | + $count_only_older_sql = $count_only_older ? $wpdb->prepare(" AND id<%d", $old_attendee_row['id']) : ''; |
|
743 | + $count = intval($wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM " . $this->_old_table . " WHERE registration_id=%s $count_only_older_sql", $old_attendee_row['registration_id']))); |
|
744 | + |
|
745 | + if ($this->_mer_tables_exist()) { |
|
746 | + // if MER exists, then its a little tricky. |
|
747 | + // when users registered by adding items to the cart, and it was a |
|
748 | + // group registration requiring additional attendee INFO, then the attendee rows |
|
749 | + // DO NOT have the same registration_id (although they probably should have) |
|
750 | + // they are related just like MER attendee rows are related, through the MER group table |
|
751 | + // BUT we want to count all the MER attendee rows for the same registration |
|
752 | + $primary_attendee = $this->_find_mer_primary_attendee_using_mer_tables($old_attendee_row['registration_id']); |
|
753 | + |
|
754 | + $count_using_mer_table = $wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM {$this->_old_table} att INNER JOIN {$this->_old_mer_table} mer ON att.registration_id = mer.registration_id WHERE att.event_id=%d AND mer.primary_registration_id = %s $count_only_older_sql", $old_attendee_row['event_id'], $primary_attendee['registration_id'])); |
|
755 | + $count = max($count_using_mer_table, $count); |
|
756 | + } |
|
757 | + return $count; |
|
758 | + } |
|
759 | + |
|
760 | + private function _insert_new_payment($old_attendee, $new_txn_id) |
|
761 | + { |
|
762 | + global $wpdb; |
|
763 | + // only add a payment for primary attendees |
|
764 | + $old_pay_stati_indicating_no_payment = array('Pending', 'Incomplete', 'Not Completed'); |
|
765 | + // if this is for a primary 3.1 attendee which WASN'T free and has a completed, cancelled, or declined payment... |
|
766 | + if (intval($old_attendee['is_primary']) && floatval($old_attendee['total_cost']) && !in_array($old_attendee['payment_status'], $old_pay_stati_indicating_no_payment)) { |
|
767 | + $pay_status_mapping = array( |
|
768 | + 'Completed' => 'PAP', |
|
769 | + 'Payment Declined' => 'PDC', |
|
770 | + 'Cancelled' => 'PCN', |
|
771 | + 'Declined' => 'PDC' |
|
772 | + ); |
|
773 | + $by_admin = $old_attendee['payment'] == 'Admin'; |
|
774 | + $STS_ID = isset($pay_status_mapping[ $old_attendee['payment_status'] ]) ? $pay_status_mapping[ $old_attendee['payment_status'] ] : 'PFL';// IE, if we don't recognize the status, assume payment failed |
|
775 | + $cols_n_values = array( |
|
776 | + 'TXN_ID' => $new_txn_id, |
|
777 | + 'STS_ID' => $STS_ID, |
|
778 | + 'PAY_timestamp' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
|
779 | + 'PAY_method' => 'CART', |
|
780 | + 'PAY_amount' => $old_attendee['amount_pd'], |
|
781 | + 'PAY_gateway' => $old_attendee['txn_type'], |
|
782 | + 'PAY_gateway_response' => '', |
|
783 | + 'PAY_txn_id_chq_nmbr' => substr($old_attendee['txn_id'], 0, 32), |
|
784 | + 'PAY_via_admin' => $by_admin, |
|
785 | + 'PAY_details' => $old_attendee['transaction_details'] |
|
786 | + |
|
787 | + ); |
|
788 | + $datatypes = array( |
|
789 | + '%d',// TXN_Id |
|
790 | + '%s',// STS_ID |
|
791 | + '%s',// PAY_timestamp |
|
792 | + '%s',// PAY_method |
|
793 | + '%f',// PAY_amount |
|
794 | + '%s',// PAY_gateway |
|
795 | + '%s',// PAY_gateway_response |
|
796 | + '%s',// PAY_txn_id_chq_nmbr |
|
797 | + '%d',// PAY_via_admin |
|
798 | + '%s',// PAY_details |
|
799 | + ); |
|
800 | + $success = $wpdb->insert($this->_new_payment_table, $cols_n_values, $datatypes); |
|
801 | + if (!$success) { |
|
802 | + $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
|
803 | + return 0; |
|
804 | + } |
|
805 | + $new_id = $wpdb->insert_id; |
|
806 | + return $new_id; |
|
807 | + } else { |
|
808 | + return 0; |
|
809 | + } |
|
810 | + } |
|
811 | + |
|
812 | + /** |
|
813 | + * If MER is active, if you want ot fin dthe other registrations on that attendee row |
|
814 | + * @global type $wpdb |
|
815 | + * @param type $old_registration_id |
|
816 | + * @return array |
|
817 | + */ |
|
818 | + private function _find_mer_primary_attendee_using_mer_tables($old_registration_id) |
|
819 | + { |
|
820 | + if (!$this->_mer_tables_exist()) { |
|
821 | + return false; |
|
822 | + } |
|
823 | + global $wpdb; |
|
824 | + $old_att_for_primary_reg = $wpdb->get_row($wpdb->prepare("SELECT * FROM {$this->_old_mer_table} AS mer INNER JOIN {$this->_old_table} AS att ON mer.primary_registration_id = att.registration_id WHERE mer.registration_id=%s LIMIT 1", $old_registration_id), ARRAY_A); |
|
825 | + return $old_att_for_primary_reg; |
|
826 | + } |
|
827 | 827 | } |
@@ -206,26 +206,26 @@ discard block |
||
206 | 206 | { |
207 | 207 | global $wpdb; |
208 | 208 | $this->_pretty_name = __("Attendees", "event_espresso"); |
209 | - $this->_old_table = $wpdb->prefix . "events_attendee"; |
|
209 | + $this->_old_table = $wpdb->prefix."events_attendee"; |
|
210 | 210 | // Only select the event status column from the event table. |
211 | 211 | $this->select_expression = 'att.*, e.event_status'; |
212 | 212 | // Only select attendees for events that aren't deleted. |
213 | 213 | $this->_extra_where_sql = 'AS att |
214 | - INNER JOIN ' . $wpdb->prefix . 'events_detail AS e ON att.event_id=e.id |
|
214 | + INNER JOIN ' . $wpdb->prefix.'events_detail AS e ON att.event_id=e.id |
|
215 | 215 | WHERE e.event_status!="D"'; |
216 | - $this->_old_mer_table = $wpdb->prefix . "events_multi_event_registration_id_group"; |
|
216 | + $this->_old_mer_table = $wpdb->prefix."events_multi_event_registration_id_group"; |
|
217 | 217 | $this->_new_attendee_cpt_table = $wpdb->posts; |
218 | - $this->_new_attendee_meta_table = $wpdb->prefix . "esp_attendee_meta"; |
|
219 | - $this->_new_reg_table = $wpdb->prefix . "esp_registration"; |
|
220 | - $this->_new_transaction_table = $wpdb->prefix . "esp_transaction"; |
|
221 | - $this->_new_payment_table = $wpdb->prefix . "esp_payment"; |
|
222 | - $this->_new_line_table = $wpdb->prefix . "esp_line_item"; |
|
223 | - $this->_new_ticket_table = $wpdb->prefix . "esp_ticket"; |
|
224 | - $this->_new_ticket_datetime_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
225 | - $this->_new_datetime_table = $wpdb->prefix . "esp_datetime"; |
|
226 | - $this->_new_datetime_ticket_table = $wpdb->prefix . "esp_datetime_ticket"; |
|
227 | - $this->_new_price_table = $wpdb->prefix . "esp_price"; |
|
228 | - $this->_new_ticket_price_table = $wpdb->prefix . "esp_ticket_price"; |
|
218 | + $this->_new_attendee_meta_table = $wpdb->prefix."esp_attendee_meta"; |
|
219 | + $this->_new_reg_table = $wpdb->prefix."esp_registration"; |
|
220 | + $this->_new_transaction_table = $wpdb->prefix."esp_transaction"; |
|
221 | + $this->_new_payment_table = $wpdb->prefix."esp_payment"; |
|
222 | + $this->_new_line_table = $wpdb->prefix."esp_line_item"; |
|
223 | + $this->_new_ticket_table = $wpdb->prefix."esp_ticket"; |
|
224 | + $this->_new_ticket_datetime_table = $wpdb->prefix."esp_datetime_ticket"; |
|
225 | + $this->_new_datetime_table = $wpdb->prefix."esp_datetime"; |
|
226 | + $this->_new_datetime_ticket_table = $wpdb->prefix."esp_datetime_ticket"; |
|
227 | + $this->_new_price_table = $wpdb->prefix."esp_price"; |
|
228 | + $this->_new_ticket_price_table = $wpdb->prefix."esp_ticket_price"; |
|
229 | 229 | parent::__construct(); |
230 | 230 | } |
231 | 231 | |
@@ -233,9 +233,9 @@ discard block |
||
233 | 233 | { |
234 | 234 | // first check if there's already a new attendee with similar characteristics |
235 | 235 | $new_att_id = $this->_find_attendee_cpt_matching($old_row); |
236 | - if (!$new_att_id) { |
|
236 | + if ( ! $new_att_id) { |
|
237 | 237 | $new_att_id = $this->_insert_new_attendee_cpt($old_row); |
238 | - if (!$new_att_id) { |
|
238 | + if ( ! $new_att_id) { |
|
239 | 239 | // if we couldnt even make an attendee, abandon all hope |
240 | 240 | return false; |
241 | 241 | } |
@@ -247,7 +247,7 @@ discard block |
||
247 | 247 | $this->get_migration_script()->set_mapping($this->_old_table, $old_row['id'], $this->_new_attendee_cpt_table, $new_att_id); |
248 | 248 | |
249 | 249 | $txn_id = $this->_insert_new_transaction($old_row); |
250 | - if (!$txn_id) { |
|
250 | + if ( ! $txn_id) { |
|
251 | 251 | // if we couldnt make the transaction, also abandon all hope |
252 | 252 | return false; |
253 | 253 | } |
@@ -275,7 +275,7 @@ discard block |
||
275 | 275 | private function _find_attendee_cpt_matching($old_attendee) |
276 | 276 | { |
277 | 277 | global $wpdb; |
278 | - $existing_attendee_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_new_attendee_cpt_table . " AS cpt INNER JOIN " . $this->_new_attendee_meta_table . " AS meta ON cpt.ID = meta.ATT_ID WHERE meta.ATT_fname = %s AND meta.ATT_lname = %s AND meta.ATT_email = %s LIMIT 1", $old_attendee['fname'], $old_attendee['lname'], $old_attendee['email'])); |
|
278 | + $existing_attendee_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM ".$this->_new_attendee_cpt_table." AS cpt INNER JOIN ".$this->_new_attendee_meta_table." AS meta ON cpt.ID = meta.ATT_ID WHERE meta.ATT_fname = %s AND meta.ATT_lname = %s AND meta.ATT_email = %s LIMIT 1", $old_attendee['fname'], $old_attendee['lname'], $old_attendee['email'])); |
|
279 | 279 | return intval($existing_attendee_id); |
280 | 280 | } |
281 | 281 | |
@@ -283,31 +283,31 @@ discard block |
||
283 | 283 | { |
284 | 284 | global $wpdb; |
285 | 285 | $cols_n_values = array( |
286 | - 'post_title' => stripslashes($old_attendee['fname'] . " " . $old_attendee['lname']),// ATT_full_name |
|
287 | - 'post_content' => '',// ATT_bio |
|
288 | - 'post_name' => sanitize_title($old_attendee['fname'] . "-" . $old_attendee['lname']),// ATT_slug |
|
289 | - 'post_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_created |
|
290 | - 'post_excerpt' => '',// ATT_short_bio |
|
291 | - 'post_modified' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']),// ATT_modified |
|
292 | - 'post_author' => 0,// ATT_author |
|
293 | - 'post_parent' => 0,// ATT_parent |
|
294 | - 'post_type' => 'espresso_attendees',// post_type |
|
286 | + 'post_title' => stripslashes($old_attendee['fname']." ".$old_attendee['lname']), // ATT_full_name |
|
287 | + 'post_content' => '', // ATT_bio |
|
288 | + 'post_name' => sanitize_title($old_attendee['fname']."-".$old_attendee['lname']), // ATT_slug |
|
289 | + 'post_date' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), // ATT_created |
|
290 | + 'post_excerpt' => '', // ATT_short_bio |
|
291 | + 'post_modified' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), // ATT_modified |
|
292 | + 'post_author' => 0, // ATT_author |
|
293 | + 'post_parent' => 0, // ATT_parent |
|
294 | + 'post_type' => 'espresso_attendees', // post_type |
|
295 | 295 | 'post_status' => 'publish'// status |
296 | 296 | ); |
297 | 297 | $datatypes = array( |
298 | - '%s',// ATT_full_name |
|
299 | - '%s',// ATT_bio |
|
300 | - '%s',// ATT_slug |
|
301 | - '%s',// ATT_created |
|
302 | - '%s',// ATT_short_bio |
|
303 | - '%s',// ATT_modified |
|
304 | - '%d',// ATT_author |
|
305 | - '%d',// ATT_parent |
|
306 | - '%s',// post_type |
|
307 | - '%s',// status |
|
298 | + '%s', // ATT_full_name |
|
299 | + '%s', // ATT_bio |
|
300 | + '%s', // ATT_slug |
|
301 | + '%s', // ATT_created |
|
302 | + '%s', // ATT_short_bio |
|
303 | + '%s', // ATT_modified |
|
304 | + '%d', // ATT_author |
|
305 | + '%d', // ATT_parent |
|
306 | + '%s', // post_type |
|
307 | + '%s', // status |
|
308 | 308 | ); |
309 | 309 | $success = $wpdb->insert($this->_new_attendee_cpt_table, $cols_n_values, $datatypes); |
310 | - if (!$success) { |
|
310 | + if ( ! $success) { |
|
311 | 311 | $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
312 | 312 | return 0; |
313 | 313 | } |
@@ -345,20 +345,20 @@ discard block |
||
345 | 345 | 'ATT_phone' => stripslashes($old_attendee['phone']), |
346 | 346 | ); |
347 | 347 | $datatypes = array( |
348 | - '%d',// ATT_ID |
|
349 | - '%s',// ATT_fname |
|
350 | - '%s',// ATT_lname |
|
351 | - '%s',// ATT_address |
|
352 | - '%s',// ATT_address2 |
|
353 | - '%s',// ATT_city |
|
354 | - '%d',// STA_ID |
|
355 | - '%s',// CNT_ISO |
|
356 | - '%s',// ATT_zip |
|
357 | - '%s',// ATT_email |
|
358 | - '%s',// ATT_phone |
|
348 | + '%d', // ATT_ID |
|
349 | + '%s', // ATT_fname |
|
350 | + '%s', // ATT_lname |
|
351 | + '%s', // ATT_address |
|
352 | + '%s', // ATT_address2 |
|
353 | + '%s', // ATT_city |
|
354 | + '%d', // STA_ID |
|
355 | + '%s', // CNT_ISO |
|
356 | + '%s', // ATT_zip |
|
357 | + '%s', // ATT_email |
|
358 | + '%s', // ATT_phone |
|
359 | 359 | ); |
360 | 360 | $success = $wpdb->insert($this->_new_attendee_meta_table, $cols_n_values, $datatypes); |
361 | - if (!$success) { |
|
361 | + if ( ! $success) { |
|
362 | 362 | $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_meta_table, $cols_n_values, $datatypes)); |
363 | 363 | return 0; |
364 | 364 | } |
@@ -382,13 +382,13 @@ discard block |
||
382 | 382 | if (intval($old_attendee['is_primary'])) {// primary attendee, so create txn |
383 | 383 | $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($old_attendee['id']), $this->_new_transaction_table); |
384 | 384 | } else { // non-primary attendee, so find its primary attendee's transaction |
385 | - $primary_attendee_old_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM " . $this->_old_table . " WHERE is_primary=1 and registration_id=%s", $old_attendee['registration_id'])); |
|
386 | - if (!$primary_attendee_old_id) { |
|
385 | + $primary_attendee_old_id = $wpdb->get_var($wpdb->prepare("SELECT id FROM ".$this->_old_table." WHERE is_primary=1 and registration_id=%s", $old_attendee['registration_id'])); |
|
386 | + if ( ! $primary_attendee_old_id) { |
|
387 | 387 | $primary_attendee = $this->_find_mer_primary_attendee_using_mer_tables($old_attendee['registration_id']); |
388 | 388 | $primary_attendee_old_id = is_array($primary_attendee) ? $primary_attendee['id'] : null; |
389 | 389 | } |
390 | 390 | $txn_id = $this->get_migration_script()->get_mapping_new_pk($this->_old_table, intval($primary_attendee_old_id), $this->_new_transaction_table); |
391 | - if (!$txn_id) { |
|
391 | + if ( ! $txn_id) { |
|
392 | 392 | $this->add_error(sprintf(__("Could not find primary attendee's new transaction. Current attendee is: %s, we think the 3.1 primary attendee for it has id %d, but there's no 4.1 transaction for that primary attendee id.", "event_espresso"), $this->_json_encode($old_attendee), $primary_attendee_old_id)); |
393 | 393 | $txn_id = 0; |
394 | 394 | } |
@@ -396,7 +396,7 @@ discard block |
||
396 | 396 | // if there isn't yet a transaction row for this, create one |
397 | 397 | // (so even if it was a non-primary attendee with no EE3 primary attendee, |
398 | 398 | // it ought to have SOME transaction, so we'll make one) |
399 | - if (!$txn_id) { |
|
399 | + if ( ! $txn_id) { |
|
400 | 400 | // maps 3.1 payment stati onto 4.1 transaction stati |
401 | 401 | $txn_status_mapping = array( |
402 | 402 | 'Completed' => 'TCM', |
@@ -407,7 +407,7 @@ discard block |
||
407 | 407 | 'Cancelled' => 'TIN', |
408 | 408 | 'Declined' => 'TIN' |
409 | 409 | ); |
410 | - $STS_ID = isset($txn_status_mapping[ $old_attendee['payment_status'] ]) ? $txn_status_mapping[ $old_attendee['payment_status'] ] : 'TIN'; |
|
410 | + $STS_ID = isset($txn_status_mapping[$old_attendee['payment_status']]) ? $txn_status_mapping[$old_attendee['payment_status']] : 'TIN'; |
|
411 | 411 | $cols_n_values = array( |
412 | 412 | 'TXN_timestamp' => $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, $old_attendee['date']), |
413 | 413 | 'TXN_total' => floatval($old_attendee['total_cost']), |
@@ -416,14 +416,14 @@ discard block |
||
416 | 416 | 'TXN_hash_salt' => $old_attendee['hashSalt'] |
417 | 417 | ); |
418 | 418 | $datatypes = array( |
419 | - '%s',// TXN_timestamp |
|
420 | - '%f',// TXN_total |
|
421 | - '%f',// TXN_paid |
|
422 | - '%s',// STS_ID |
|
423 | - '%s',// TXN_hash_salt |
|
419 | + '%s', // TXN_timestamp |
|
420 | + '%f', // TXN_total |
|
421 | + '%f', // TXN_paid |
|
422 | + '%s', // STS_ID |
|
423 | + '%s', // TXN_hash_salt |
|
424 | 424 | ); |
425 | 425 | $success = $wpdb->insert($this->_new_transaction_table, $cols_n_values, $datatypes); |
426 | - if (!$success) { |
|
426 | + if ( ! $success) { |
|
427 | 427 | $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_transaction_table, $cols_n_values, $datatypes)); |
428 | 428 | return 0; |
429 | 429 | } |
@@ -463,7 +463,7 @@ discard block |
||
463 | 463 | { |
464 | 464 | // need event default reg status and if pre_approval was required |
465 | 465 | global $wpdb; |
466 | - $event_required_pre_approval = $wpdb->get_var($wpdb->prepare("SELECT require_pre_approval FROM " . $wpdb->prefix . "events_detail WHERE id = %d", $old_attendee_row['event_id'])); |
|
466 | + $event_required_pre_approval = $wpdb->get_var($wpdb->prepare("SELECT require_pre_approval FROM ".$wpdb->prefix."events_detail WHERE id = %d", $old_attendee_row['event_id'])); |
|
467 | 467 | return $this->get_migration_script()->convert_3_1_payment_status_to_4_1_STS_ID( |
468 | 468 | $old_attendee_row['payment_status'], |
469 | 469 | intval($event_required_pre_approval) && intval($old_attendee_row['pre_approve']) |
@@ -484,13 +484,13 @@ discard block |
||
484 | 484 | global $wpdb; |
485 | 485 | |
486 | 486 | $STS_ID = $this->_get_reg_status_for_old_payment_status($old_attendee); |
487 | - $new_event_id = $this->get_migration_script()->get_mapping_new_pk($wpdb->prefix . 'events_detail', $old_attendee['event_id'], $wpdb->posts); |
|
488 | - if (!$new_event_id) { |
|
487 | + $new_event_id = $this->get_migration_script()->get_mapping_new_pk($wpdb->prefix.'events_detail', $old_attendee['event_id'], $wpdb->posts); |
|
488 | + if ( ! $new_event_id) { |
|
489 | 489 | $this->add_error(sprintf(__("Could not find NEW event CPT ID for old event '%d' on old attendee %s", "event_espresso"), $old_attendee['event_id'], $this->_json_encode($old_attendee))); |
490 | 490 | } |
491 | 491 | |
492 | 492 | $ticket_id = $this->_try_to_find_new_ticket_id($old_attendee, $new_event_id); |
493 | - if (!$ticket_id) { |
|
493 | + if ( ! $ticket_id) { |
|
494 | 494 | $ticket_id = $this->_insert_new_ticket_because_none_found($old_attendee, $new_event_id); |
495 | 495 | $this->add_error(sprintf(__('Could not find a ticket for old attendee with id %d for new event %d, so created a new ticket with id %d', 'event_espresso'), $old_attendee['id'], $new_event_id, $ticket_id)); |
496 | 496 | } |
@@ -514,30 +514,30 @@ discard block |
||
514 | 514 | 'REG_final_price' => $old_attendee['final_price'], |
515 | 515 | 'REG_session' => substr($old_attendee['attendee_session'], 0, 44), |
516 | 516 | 'REG_code' => sanitize_key($old_attendee['registration_id']), |
517 | - 'REG_url_link' => sanitize_key($old_attendee['registration_id'] . '-' . $count), |
|
517 | + 'REG_url_link' => sanitize_key($old_attendee['registration_id'].'-'.$count), |
|
518 | 518 | 'REG_count' => $regs_on_this_event_and_txn + $count, |
519 | 519 | 'REG_group_size' => $this->_sum_old_attendees_on_old_txn($old_attendee, false), |
520 | 520 | 'REG_att_is_going' => true, |
521 | 521 | 'REG_deleted' => false |
522 | 522 | ); |
523 | 523 | $datatypes = array( |
524 | - '%d',// EVT_ID |
|
525 | - '%d',// ATT_ID |
|
526 | - '%d',// TXN_ID |
|
527 | - '%d',// TKT_ID |
|
528 | - '%s',// STS_ID |
|
529 | - '%s',// REG_date |
|
530 | - '%f',// REG_final_price |
|
531 | - '%s',// REG_session |
|
532 | - '%s',// REG_code |
|
533 | - '%s',// REG_url_link |
|
534 | - '%d',// REG_count |
|
535 | - '%d',// REG_group_size |
|
536 | - '%d',// REG_att_is_going |
|
537 | - '%d',// REG_deleted |
|
524 | + '%d', // EVT_ID |
|
525 | + '%d', // ATT_ID |
|
526 | + '%d', // TXN_ID |
|
527 | + '%d', // TKT_ID |
|
528 | + '%s', // STS_ID |
|
529 | + '%s', // REG_date |
|
530 | + '%f', // REG_final_price |
|
531 | + '%s', // REG_session |
|
532 | + '%s', // REG_code |
|
533 | + '%s', // REG_url_link |
|
534 | + '%d', // REG_count |
|
535 | + '%d', // REG_group_size |
|
536 | + '%d', // REG_att_is_going |
|
537 | + '%d', // REG_deleted |
|
538 | 538 | ); |
539 | 539 | $success = $wpdb->insert($this->_new_reg_table, $cols_n_values, $datatypes); |
540 | - if (!$success) { |
|
540 | + if ( ! $success) { |
|
541 | 541 | $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_reg_table, $cols_n_values, $datatypes)); |
542 | 542 | return 0; |
543 | 543 | } |
@@ -573,7 +573,7 @@ discard block |
||
573 | 573 | INNER JOIN {$this->_new_ticket_datetime_table} as DTK ON TKT.TKT_ID = DTK.TKT_ID |
574 | 574 | INNER JOIN {$this->_new_datetime_table} as DTT ON DTK.DTT_ID = DTT.DTT_ID |
575 | 575 | SET DTT.DTT_sold = DTT.DTT_sold + %d WHERE TKT.TKT_ID = %d", $quantity_sold, $new_ticket_id)); |
576 | - if (!$success_update_dateimtes) { |
|
576 | + if ( ! $success_update_dateimtes) { |
|
577 | 577 | $this->add_error(sprintf(__("Could not update datetimes related to ticket with ID %d's TKT_sold by %d because %s", "event_espresso"), $new_ticket_id, $quantity_sold, $wpdb->last_error)); |
578 | 578 | } |
579 | 579 | } else { |
@@ -606,20 +606,20 @@ discard block |
||
606 | 606 | // add all conditions to an array from which we can SHIFT conditions off in order to widen our search |
607 | 607 | // the most important condition should be last, as it will be array_shift'ed off last |
608 | 608 | $conditions = array( |
609 | - $wpdb->prepare("$datetime_table.DTT_EVT_start = %s", $old_att_datetime),// times match? |
|
610 | - $wpdb->prepare("$tickets_table.TKT_price = %f", $old_att_price),// prices match? |
|
611 | - $wpdb->prepare("$tickets_table.TKT_name = %s", $old_att_price_option),// names match? |
|
612 | - $wpdb->prepare("$datetime_table.EVT_ID = %d", $new_event_id),// events match? |
|
609 | + $wpdb->prepare("$datetime_table.DTT_EVT_start = %s", $old_att_datetime), // times match? |
|
610 | + $wpdb->prepare("$tickets_table.TKT_price = %f", $old_att_price), // prices match? |
|
611 | + $wpdb->prepare("$tickets_table.TKT_name = %s", $old_att_price_option), // names match? |
|
612 | + $wpdb->prepare("$datetime_table.EVT_ID = %d", $new_event_id), // events match? |
|
613 | 613 | ); |
614 | 614 | $select_and_join_part = "SELECT $tickets_table.TKT_ID FROM $tickets_table INNER JOIN |
615 | 615 | $datetime_tickets_table ON $tickets_table.TKT_ID = $datetime_tickets_table.TKT_ID INNER JOIN |
616 | 616 | $datetime_table ON $datetime_tickets_table.DTT_ID = $datetime_table.DTT_ID"; |
617 | 617 | // start running queries, widening search each time by removing a condition |
618 | 618 | do { |
619 | - $full_query = $select_and_join_part . " WHERE " . implode(" AND ", $conditions) . " LIMIT 1"; |
|
619 | + $full_query = $select_and_join_part." WHERE ".implode(" AND ", $conditions)." LIMIT 1"; |
|
620 | 620 | $ticket_id_found = $wpdb->get_var($full_query); |
621 | 621 | array_shift($conditions); |
622 | - } while (!$ticket_id_found && $conditions); |
|
622 | + }while ( ! $ticket_id_found && $conditions); |
|
623 | 623 | return $ticket_id_found; |
624 | 624 | } |
625 | 625 | |
@@ -641,8 +641,8 @@ discard block |
||
641 | 641 | |
642 | 642 | |
643 | 643 | // insert new datetime unless we find one |
644 | - $datetime_id = $wpdb->get_var($wpdb->prepare("SELECT DTT_ID FROM " . $this->_new_datetime_table . " WHERE DTT_EVT_start=%s AND EVT_ID=%d LIMIT 1", $old_att_start_datetime, $new_event_id), ARRAY_A); |
|
645 | - if (!$datetime_id) { |
|
644 | + $datetime_id = $wpdb->get_var($wpdb->prepare("SELECT DTT_ID FROM ".$this->_new_datetime_table." WHERE DTT_EVT_start=%s AND EVT_ID=%d LIMIT 1", $old_att_start_datetime, $new_event_id), ARRAY_A); |
|
645 | + if ( ! $datetime_id) { |
|
646 | 646 | $old_att_end_date = $old_attendee['start_date']; |
647 | 647 | $old_att_end_time = $this->get_migration_script()->convertTimeFromAMPM($old_attendee['event_time']); |
648 | 648 | $old_att_end_datetime = $this->get_migration_script()->convert_date_string_to_utc($this, $old_attendee, "$old_att_end_date $old_att_end_time:00"); |
@@ -655,10 +655,10 @@ discard block |
||
655 | 655 | 'DTT_deleted' => true |
656 | 656 | ), |
657 | 657 | array( |
658 | - '%d',// EVT_ID |
|
659 | - '%s',// DTT_EVT_start |
|
660 | - '%s',// DTT_EVT_end |
|
661 | - '%d',// DTT_deleted |
|
658 | + '%d', // EVT_ID |
|
659 | + '%s', // DTT_EVT_start |
|
660 | + '%s', // DTT_EVT_end |
|
661 | + '%d', // DTT_deleted |
|
662 | 662 | ) |
663 | 663 | ); |
664 | 664 | $datetime_id = $wpdb->insert_id; |
@@ -666,21 +666,21 @@ discard block |
||
666 | 666 | |
667 | 667 | // insert new ticket |
668 | 668 | $success = $wpdb->insert( |
669 | - $wpdb->prefix . 'esp_ticket', |
|
669 | + $wpdb->prefix.'esp_ticket', |
|
670 | 670 | array( |
671 | 671 | 'TKT_name' => $old_att_price_option, |
672 | 672 | 'TKT_qty' => -1, |
673 | 673 | 'TKT_price' => $old_att_price, |
674 | - 'TKT_start_date' => $old_att_start_datetime,// we really have no clue what the time should be, but at least it was available when they attended |
|
674 | + 'TKT_start_date' => $old_att_start_datetime, // we really have no clue what the time should be, but at least it was available when they attended |
|
675 | 675 | 'TKT_end_date' => $old_att_end_datetime, |
676 | 676 | |
677 | 677 | ), |
678 | 678 | array( |
679 | - '%s',// name |
|
680 | - '%d',// qty |
|
681 | - '%d',// price |
|
682 | - '%s',// start_date |
|
683 | - '%s',// end_date |
|
679 | + '%s', // name |
|
680 | + '%d', // qty |
|
681 | + '%d', // price |
|
682 | + '%s', // start_date |
|
683 | + '%s', // end_date |
|
684 | 684 | ) |
685 | 685 | ); |
686 | 686 | $ticket_id = $wpdb->insert_id; |
@@ -692,8 +692,8 @@ discard block |
||
692 | 692 | 'TKT_ID' => $ticket_id |
693 | 693 | ), |
694 | 694 | array( |
695 | - '%d',// DTT_ID |
|
696 | - '%d',// TKT_ID |
|
695 | + '%d', // DTT_ID |
|
696 | + '%d', // TKT_ID |
|
697 | 697 | ) |
698 | 698 | ); |
699 | 699 | // insert new price |
@@ -706,10 +706,10 @@ discard block |
||
706 | 706 | 'PRC_deleted' => true |
707 | 707 | ), |
708 | 708 | array( |
709 | - '%d',// PRC_amount |
|
710 | - '%d',// PRT_ID |
|
711 | - '%s',// PRC_name |
|
712 | - '%d',// PRC_deleted |
|
709 | + '%d', // PRC_amount |
|
710 | + '%d', // PRT_ID |
|
711 | + '%s', // PRC_name |
|
712 | + '%d', // PRC_deleted |
|
713 | 713 | ) |
714 | 714 | ); |
715 | 715 | $price_id = $wpdb->insert_id; |
@@ -721,8 +721,8 @@ discard block |
||
721 | 721 | 'PRC_ID' => $price_id |
722 | 722 | ), |
723 | 723 | array( |
724 | - '%d',// TKT_ID |
|
725 | - '%d',// PRC_ID |
|
724 | + '%d', // TKT_ID |
|
725 | + '%d', // PRC_ID |
|
726 | 726 | ) |
727 | 727 | ); |
728 | 728 | return $ticket_id; |
@@ -740,7 +740,7 @@ discard block |
||
740 | 740 | { |
741 | 741 | global $wpdb; |
742 | 742 | $count_only_older_sql = $count_only_older ? $wpdb->prepare(" AND id<%d", $old_attendee_row['id']) : ''; |
743 | - $count = intval($wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM " . $this->_old_table . " WHERE registration_id=%s $count_only_older_sql", $old_attendee_row['registration_id']))); |
|
743 | + $count = intval($wpdb->get_var($wpdb->prepare("SELECT SUM(quantity) FROM ".$this->_old_table." WHERE registration_id=%s $count_only_older_sql", $old_attendee_row['registration_id']))); |
|
744 | 744 | |
745 | 745 | if ($this->_mer_tables_exist()) { |
746 | 746 | // if MER exists, then its a little tricky. |
@@ -763,7 +763,7 @@ discard block |
||
763 | 763 | // only add a payment for primary attendees |
764 | 764 | $old_pay_stati_indicating_no_payment = array('Pending', 'Incomplete', 'Not Completed'); |
765 | 765 | // if this is for a primary 3.1 attendee which WASN'T free and has a completed, cancelled, or declined payment... |
766 | - if (intval($old_attendee['is_primary']) && floatval($old_attendee['total_cost']) && !in_array($old_attendee['payment_status'], $old_pay_stati_indicating_no_payment)) { |
|
766 | + if (intval($old_attendee['is_primary']) && floatval($old_attendee['total_cost']) && ! in_array($old_attendee['payment_status'], $old_pay_stati_indicating_no_payment)) { |
|
767 | 767 | $pay_status_mapping = array( |
768 | 768 | 'Completed' => 'PAP', |
769 | 769 | 'Payment Declined' => 'PDC', |
@@ -771,7 +771,7 @@ discard block |
||
771 | 771 | 'Declined' => 'PDC' |
772 | 772 | ); |
773 | 773 | $by_admin = $old_attendee['payment'] == 'Admin'; |
774 | - $STS_ID = isset($pay_status_mapping[ $old_attendee['payment_status'] ]) ? $pay_status_mapping[ $old_attendee['payment_status'] ] : 'PFL';// IE, if we don't recognize the status, assume payment failed |
|
774 | + $STS_ID = isset($pay_status_mapping[$old_attendee['payment_status']]) ? $pay_status_mapping[$old_attendee['payment_status']] : 'PFL'; // IE, if we don't recognize the status, assume payment failed |
|
775 | 775 | $cols_n_values = array( |
776 | 776 | 'TXN_ID' => $new_txn_id, |
777 | 777 | 'STS_ID' => $STS_ID, |
@@ -786,19 +786,19 @@ discard block |
||
786 | 786 | |
787 | 787 | ); |
788 | 788 | $datatypes = array( |
789 | - '%d',// TXN_Id |
|
790 | - '%s',// STS_ID |
|
791 | - '%s',// PAY_timestamp |
|
792 | - '%s',// PAY_method |
|
793 | - '%f',// PAY_amount |
|
794 | - '%s',// PAY_gateway |
|
795 | - '%s',// PAY_gateway_response |
|
796 | - '%s',// PAY_txn_id_chq_nmbr |
|
797 | - '%d',// PAY_via_admin |
|
798 | - '%s',// PAY_details |
|
789 | + '%d', // TXN_Id |
|
790 | + '%s', // STS_ID |
|
791 | + '%s', // PAY_timestamp |
|
792 | + '%s', // PAY_method |
|
793 | + '%f', // PAY_amount |
|
794 | + '%s', // PAY_gateway |
|
795 | + '%s', // PAY_gateway_response |
|
796 | + '%s', // PAY_txn_id_chq_nmbr |
|
797 | + '%d', // PAY_via_admin |
|
798 | + '%s', // PAY_details |
|
799 | 799 | ); |
800 | 800 | $success = $wpdb->insert($this->_new_payment_table, $cols_n_values, $datatypes); |
801 | - if (!$success) { |
|
801 | + if ( ! $success) { |
|
802 | 802 | $this->add_error($this->get_migration_script()->_create_error_message_for_db_insertion($this->_old_table, $old_attendee, $this->_new_attendee_cpt_table, $cols_n_values, $datatypes)); |
803 | 803 | return 0; |
804 | 804 | } |
@@ -817,7 +817,7 @@ discard block |
||
817 | 817 | */ |
818 | 818 | private function _find_mer_primary_attendee_using_mer_tables($old_registration_id) |
819 | 819 | { |
820 | - if (!$this->_mer_tables_exist()) { |
|
820 | + if ( ! $this->_mer_tables_exist()) { |
|
821 | 821 | return false; |
822 | 822 | } |
823 | 823 | global $wpdb; |
@@ -38,103 +38,103 @@ |
||
38 | 38 | * @since 4.0 |
39 | 39 | */ |
40 | 40 | if (function_exists('espresso_version')) { |
41 | - if (! function_exists('espresso_duplicate_plugin_error')) { |
|
42 | - /** |
|
43 | - * espresso_duplicate_plugin_error |
|
44 | - * displays if more than one version of EE is activated at the same time |
|
45 | - */ |
|
46 | - function espresso_duplicate_plugin_error() |
|
47 | - { |
|
48 | - ?> |
|
41 | + if (! function_exists('espresso_duplicate_plugin_error')) { |
|
42 | + /** |
|
43 | + * espresso_duplicate_plugin_error |
|
44 | + * displays if more than one version of EE is activated at the same time |
|
45 | + */ |
|
46 | + function espresso_duplicate_plugin_error() |
|
47 | + { |
|
48 | + ?> |
|
49 | 49 | <div class="error"> |
50 | 50 | <p> |
51 | 51 | <?php |
52 | - echo esc_html__( |
|
53 | - 'Can not run multiple versions of Event Espresso! One version has been automatically deactivated. Please verify that you have the correct version you want still active.', |
|
54 | - 'event_espresso' |
|
55 | - ); ?> |
|
52 | + echo esc_html__( |
|
53 | + 'Can not run multiple versions of Event Espresso! One version has been automatically deactivated. Please verify that you have the correct version you want still active.', |
|
54 | + 'event_espresso' |
|
55 | + ); ?> |
|
56 | 56 | </p> |
57 | 57 | </div> |
58 | 58 | <?php |
59 | - espresso_deactivate_plugin(plugin_basename(__FILE__)); |
|
60 | - } |
|
61 | - } |
|
62 | - add_action('admin_notices', 'espresso_duplicate_plugin_error', 1); |
|
59 | + espresso_deactivate_plugin(plugin_basename(__FILE__)); |
|
60 | + } |
|
61 | + } |
|
62 | + add_action('admin_notices', 'espresso_duplicate_plugin_error', 1); |
|
63 | 63 | } else { |
64 | - define('EE_MIN_PHP_VER_REQUIRED', '5.4.0'); |
|
65 | - if (! version_compare(PHP_VERSION, EE_MIN_PHP_VER_REQUIRED, '>=')) { |
|
66 | - /** |
|
67 | - * espresso_minimum_php_version_error |
|
68 | - * |
|
69 | - * @return void |
|
70 | - */ |
|
71 | - function espresso_minimum_php_version_error() |
|
72 | - { |
|
73 | - ?> |
|
64 | + define('EE_MIN_PHP_VER_REQUIRED', '5.4.0'); |
|
65 | + if (! version_compare(PHP_VERSION, EE_MIN_PHP_VER_REQUIRED, '>=')) { |
|
66 | + /** |
|
67 | + * espresso_minimum_php_version_error |
|
68 | + * |
|
69 | + * @return void |
|
70 | + */ |
|
71 | + function espresso_minimum_php_version_error() |
|
72 | + { |
|
73 | + ?> |
|
74 | 74 | <div class="error"> |
75 | 75 | <p> |
76 | 76 | <?php |
77 | - printf( |
|
78 | - esc_html__( |
|
79 | - 'We\'re sorry, but Event Espresso requires PHP version %1$s or greater in order to operate. You are currently running version %2$s.%3$sIn order to update your version of PHP, you will need to contact your current hosting provider.%3$sFor information on stable PHP versions, please go to %4$s.', |
|
80 | - 'event_espresso' |
|
81 | - ), |
|
82 | - EE_MIN_PHP_VER_REQUIRED, |
|
83 | - PHP_VERSION, |
|
84 | - '<br/>', |
|
85 | - '<a href="http://php.net/downloads.php">http://php.net/downloads.php</a>' |
|
86 | - ); |
|
87 | - ?> |
|
77 | + printf( |
|
78 | + esc_html__( |
|
79 | + 'We\'re sorry, but Event Espresso requires PHP version %1$s or greater in order to operate. You are currently running version %2$s.%3$sIn order to update your version of PHP, you will need to contact your current hosting provider.%3$sFor information on stable PHP versions, please go to %4$s.', |
|
80 | + 'event_espresso' |
|
81 | + ), |
|
82 | + EE_MIN_PHP_VER_REQUIRED, |
|
83 | + PHP_VERSION, |
|
84 | + '<br/>', |
|
85 | + '<a href="http://php.net/downloads.php">http://php.net/downloads.php</a>' |
|
86 | + ); |
|
87 | + ?> |
|
88 | 88 | </p> |
89 | 89 | </div> |
90 | 90 | <?php |
91 | - espresso_deactivate_plugin(plugin_basename(__FILE__)); |
|
92 | - } |
|
91 | + espresso_deactivate_plugin(plugin_basename(__FILE__)); |
|
92 | + } |
|
93 | 93 | |
94 | - add_action('admin_notices', 'espresso_minimum_php_version_error', 1); |
|
95 | - } else { |
|
96 | - define('EVENT_ESPRESSO_MAIN_FILE', __FILE__); |
|
97 | - /** |
|
98 | - * espresso_version |
|
99 | - * Returns the plugin version |
|
100 | - * |
|
101 | - * @return string |
|
102 | - */ |
|
103 | - function espresso_version() |
|
104 | - { |
|
105 | - return apply_filters('FHEE__espresso__espresso_version', '4.9.80.rc.075'); |
|
106 | - } |
|
94 | + add_action('admin_notices', 'espresso_minimum_php_version_error', 1); |
|
95 | + } else { |
|
96 | + define('EVENT_ESPRESSO_MAIN_FILE', __FILE__); |
|
97 | + /** |
|
98 | + * espresso_version |
|
99 | + * Returns the plugin version |
|
100 | + * |
|
101 | + * @return string |
|
102 | + */ |
|
103 | + function espresso_version() |
|
104 | + { |
|
105 | + return apply_filters('FHEE__espresso__espresso_version', '4.9.80.rc.075'); |
|
106 | + } |
|
107 | 107 | |
108 | - /** |
|
109 | - * espresso_plugin_activation |
|
110 | - * adds a wp-option to indicate that EE has been activated via the WP admin plugins page |
|
111 | - */ |
|
112 | - function espresso_plugin_activation() |
|
113 | - { |
|
114 | - update_option('ee_espresso_activation', true); |
|
115 | - } |
|
108 | + /** |
|
109 | + * espresso_plugin_activation |
|
110 | + * adds a wp-option to indicate that EE has been activated via the WP admin plugins page |
|
111 | + */ |
|
112 | + function espresso_plugin_activation() |
|
113 | + { |
|
114 | + update_option('ee_espresso_activation', true); |
|
115 | + } |
|
116 | 116 | |
117 | - register_activation_hook(EVENT_ESPRESSO_MAIN_FILE, 'espresso_plugin_activation'); |
|
117 | + register_activation_hook(EVENT_ESPRESSO_MAIN_FILE, 'espresso_plugin_activation'); |
|
118 | 118 | |
119 | - require_once __DIR__ . '/core/bootstrap_espresso.php'; |
|
120 | - bootstrap_espresso(); |
|
121 | - } |
|
119 | + require_once __DIR__ . '/core/bootstrap_espresso.php'; |
|
120 | + bootstrap_espresso(); |
|
121 | + } |
|
122 | 122 | } |
123 | 123 | if (! function_exists('espresso_deactivate_plugin')) { |
124 | - /** |
|
125 | - * deactivate_plugin |
|
126 | - * usage: espresso_deactivate_plugin( plugin_basename( __FILE__ )); |
|
127 | - * |
|
128 | - * @access public |
|
129 | - * @param string $plugin_basename - the results of plugin_basename( __FILE__ ) for the plugin's main file |
|
130 | - * @return void |
|
131 | - */ |
|
132 | - function espresso_deactivate_plugin($plugin_basename = '') |
|
133 | - { |
|
134 | - if (! function_exists('deactivate_plugins')) { |
|
135 | - require_once ABSPATH . 'wp-admin/includes/plugin.php'; |
|
136 | - } |
|
137 | - unset($_GET['activate'], $_REQUEST['activate']); |
|
138 | - deactivate_plugins($plugin_basename); |
|
139 | - } |
|
124 | + /** |
|
125 | + * deactivate_plugin |
|
126 | + * usage: espresso_deactivate_plugin( plugin_basename( __FILE__ )); |
|
127 | + * |
|
128 | + * @access public |
|
129 | + * @param string $plugin_basename - the results of plugin_basename( __FILE__ ) for the plugin's main file |
|
130 | + * @return void |
|
131 | + */ |
|
132 | + function espresso_deactivate_plugin($plugin_basename = '') |
|
133 | + { |
|
134 | + if (! function_exists('deactivate_plugins')) { |
|
135 | + require_once ABSPATH . 'wp-admin/includes/plugin.php'; |
|
136 | + } |
|
137 | + unset($_GET['activate'], $_REQUEST['activate']); |
|
138 | + deactivate_plugins($plugin_basename); |
|
139 | + } |
|
140 | 140 | } |