get4cast_salesforecast - Version 1.1.0

Version Notes

pt_BR locale

Download this release

Release Info

Developer Get4Cast
Extension get4cast_salesforecast
Version 1.1.0
Comparing to
See all releases


Version 1.1.0

Files changed (29) hide show
  1. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Account.php +23 -0
  2. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast.php +29 -0
  3. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Edit.php +25 -0
  4. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Edit/Form.php +417 -0
  5. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Grid.php +111 -0
  6. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Link.php +19 -0
  7. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Translate.php +10 -0
  8. app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Url.php +10 -0
  9. app/code/community/Get4Cast/SalesForecast/Helper/Data.php +115 -0
  10. app/code/community/Get4Cast/SalesForecast/Model/Apiclient.php +200 -0
  11. app/code/community/Get4Cast/SalesForecast/Model/Collect.php +262 -0
  12. app/code/community/Get4Cast/SalesForecast/Model/Config.php +32 -0
  13. app/code/community/Get4Cast/SalesForecast/Model/Forecast.php +109 -0
  14. app/code/community/Get4Cast/SalesForecast/Model/Observer.php +106 -0
  15. app/code/community/Get4Cast/SalesForecast/Model/Resource/Config.php +9 -0
  16. app/code/community/Get4Cast/SalesForecast/Model/Resource/Config/Collection.php +14 -0
  17. app/code/community/Get4Cast/SalesForecast/Model/Resource/Forecast.php +9 -0
  18. app/code/community/Get4Cast/SalesForecast/Model/Resource/Forecast/Collection.php +14 -0
  19. app/code/community/Get4Cast/SalesForecast/controllers/Adminhtml/ForecastController.php +173 -0
  20. app/code/community/Get4Cast/SalesForecast/data/get4cast_salesforecast_setup/data-install-1.0.0.php +4 -0
  21. app/code/community/Get4Cast/SalesForecast/etc/adminhtml.xml +93 -0
  22. app/code/community/Get4Cast/SalesForecast/etc/config.xml +113 -0
  23. app/code/community/Get4Cast/SalesForecast/sql/get4cast_salesforecast_setup/install-1.0.0.php +195 -0
  24. app/design/adminhtml/default/default/layout/get4cast_salesforecast.xml +9 -0
  25. app/etc/modules/Get4Cast_SalesForecast.xml +9 -0
  26. app/locale/pt_BR/Get4Cast_SalesForecast.csv +114 -0
  27. js/get4cast/salesforecast/forecast.js +707 -0
  28. js/get4cast/salesforecast/jquery-1.12.0.min.js +5 -0
  29. package.xml +18 -0
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Account.php ADDED
@@ -0,0 +1,23 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Get4Cast_SalesForecast_Block_Adminhtml_Account
4
+ extends Mage_Adminhtml_Block_Widget
5
+ implements Varien_Data_Form_Element_Renderer_Interface
6
+ {
7
+ public function render(Varien_Data_Form_Element_Abstract $element) {
8
+ $account_info = $this->getData('account_info');
9
+ $account_balance = $account_info['account_balance'];
10
+ $html = '<td class="label">'.$element->getLabelHtml().'</td>';
11
+ $html .= '<td class = "value">';
12
+ $html .= '<ul>';
13
+ if($account_info['account_notifications']){
14
+ foreach($account_info['account_notifications'] as $notification){
15
+ $html .= '<li>'.$this->__($notification).'</li>';
16
+ }
17
+ }
18
+ $html .= '</ul>';
19
+ $html .= '</td>';
20
+
21
+ return $html;
22
+ }
23
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast.php ADDED
@@ -0,0 +1,29 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Block_Adminhtml_Forecast
3
+ extends Mage_Adminhtml_Block_Widget_Grid_Container
4
+ {
5
+ protected function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_blockGroup = 'get4cast_salesforecast_adminhtml';
9
+ $this->_controller = 'forecast';
10
+ $this->_headerText = Mage::helper('get4cast_salesforecast')
11
+ ->__('Forecast history');
12
+ }
13
+
14
+ protected function _prepareLayout() {
15
+ $admin_session = Mage::getSingleton('admin/session');
16
+ if(!$admin_session->isAllowed('admin/report/get4cast_salesforecast/sales_forecast/new_forecast')){
17
+ $this->_removeButton('add');
18
+ }
19
+
20
+ return parent::_prepareLayout();
21
+ }
22
+
23
+ public function getCreateUrl()
24
+ {
25
+ return $this->getUrl(
26
+ 'get4cast_salesforecast_admin/forecast/edit'
27
+ );
28
+ }
29
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Edit.php ADDED
@@ -0,0 +1,25 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Block_Adminhtml_Forecast_Edit
3
+ extends Mage_Adminhtml_Block_Widget_Form_Container
4
+ {
5
+ protected function _construct()
6
+ {
7
+ $this->_blockGroup = 'get4cast_salesforecast_adminhtml';
8
+ $this->_controller = 'forecast';
9
+
10
+ $this->_mode = 'edit';
11
+
12
+ $newOrEdit = $this->getRequest()->getParam('id')
13
+ ? $this->__('Edit')
14
+ : $this->__('New');
15
+ $this->_headerText = $newOrEdit . ' ' . $this->__('forecast report');
16
+ }
17
+
18
+ protected function _prepareLayout() {
19
+ $this->_removeButton('save');
20
+ $this->_removeButton('delete');
21
+ $this->_removeButton('reset');
22
+
23
+ return parent::_prepareLayout();
24
+ }
25
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Edit/Form.php ADDED
@@ -0,0 +1,417 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Block_Adminhtml_Forecast_Edit_Form
3
+ extends Mage_Adminhtml_Block_Widget_Form
4
+ {
5
+ protected function _prepareForm()
6
+ {
7
+
8
+ $helper = Mage::helper('get4cast_salesforecast/data');
9
+
10
+ // Instantiate a new form
11
+ $form = new Varien_Data_Form(array(
12
+ 'id' => 'edit_form',
13
+ 'action' => $this->getUrl(
14
+ 'get4cast_salesforecast_admin/forecast/edit',
15
+ array(
16
+ '_current' => true,
17
+ 'continue' => 0,
18
+ )
19
+ ),
20
+ 'method' => 'post',
21
+ ));
22
+ $form->setUseContainer(true);
23
+ $this->setForm($form);
24
+
25
+ // Define a new fieldset
26
+ $fieldset = $form->addFieldset(
27
+ 'general',
28
+ array(
29
+ 'legend' => $this->__('Forecast details')
30
+ )
31
+ );
32
+
33
+ $forecast = Mage::registry('current_forecast');
34
+ $disabled = $forecast->getId() ? true : false;
35
+
36
+ $enable_request_report = Mage::getStoreConfig('get4cast/default/enable_request_report');
37
+
38
+ $account_info = Mage::getSingleton('core/session')
39
+ ->getGet4CastAccountInfo();
40
+ if(!$account_info){
41
+ $account_info = array();
42
+ }
43
+
44
+ $account_info_json = htmlspecialchars(Mage::helper('core')
45
+ ->jsonEncode($account_info));
46
+ $account_info_json = str_replace('\'', '\\\'', $account_info_json);
47
+
48
+ if(isset($account_info['account_notifications'])
49
+ && count($account_info['account_notifications'])
50
+ && !$disabled
51
+ ){
52
+ // Field to show account information. This field uses a custom
53
+ // renderer Get4Cast_SalesForecast_Block_Adminhtml_Account
54
+ // file: app/code/community/Get4Cast/SalesForecast
55
+ // /Block/Adminhtml/Account.php
56
+ $fieldset->addField('account', 'text', array(
57
+ 'label' => $this->__('Account info'),
58
+ 'tabindex' => 1
59
+ ));
60
+ $form->getElement('account')->setRenderer( Mage::app()
61
+ ->getLayout()
62
+ ->createBlock('get4cast_salesforecast_adminhtml/account')
63
+ ->setData('account_info', $account_info)
64
+ );
65
+ }
66
+
67
+ if($forecast->getEntityId()){
68
+ $fieldset->addField('report_link', 'text', array(
69
+ 'label' => $this->__('Report link'),
70
+ 'tabindex' => 2
71
+ ));
72
+ $form->getElement('report_link')->setRenderer( Mage::app()
73
+ ->getLayout()
74
+ ->createBlock('get4cast_salesforecast_adminhtml/link')
75
+ ->setData('report_link', $forecast)
76
+ );
77
+ }
78
+
79
+ // Select box with store groups
80
+ $store_groups = $helper->getAllStoresToSelect();
81
+ reset($store_groups);
82
+ $first_store = $forecast->getStoreGroupId() ? $forecast->getStoreGroupId() : current($store_groups);
83
+ $fieldset->addField('store_group_id', 'select', array(
84
+ 'name' => 'store_group_id',
85
+ 'label' => $this->__('Store'),
86
+ 'class' => 'required-entry',
87
+ 'required' => true,
88
+ 'disabled' => false,
89
+ 'after_element_html' => '<small>'.$this->__('Forecast for the selected store').'</small>',
90
+ 'options' => $store_groups,
91
+ 'onchange' => "jQuery('#h_store_group_name').val(jQuery('#store_group_id option:selected').text());",
92
+ 'value' => $forecast->getStoreGroupId(),
93
+ 'disabled' => $disabled,
94
+ ));
95
+ $fieldset->addField('h_store_group_name', 'hidden', array(
96
+ 'name' => 'h_store_group_name',
97
+ 'value' => $first_store,
98
+ ));
99
+
100
+ $jstranslator_array = array();
101
+ $jstranslator_array[] = "Loading...";
102
+ $jstranslator_array[] = "Unable to get forecast price";
103
+ $jstranslator_array[] = "The payment link will be available";
104
+ $jstranslator_array[] = "when you open your report and in your email.";
105
+ $jstranslator_array[] = "You have a credit of:";
106
+ $jstranslator_array[] = "Forecast original price:";
107
+ $jstranslator_array[] = "Forecast final price";
108
+ $jstranslator_array[] = "You will still have<br>a credit of";
109
+ $jstranslator_array[] = "Forecast price";
110
+ $jstranslator_array[] = "Try again or";
111
+ $jstranslator_array[] = "contact us";
112
+ $jstranslator_array[] = "Could not retrieve account information";
113
+ $jstranslator_array[] = "Your email is required";
114
+ $jstranslator_array[] = "Your email is invalid";
115
+ $jstranslator_array[] = "Analyse from day is required";
116
+ $jstranslator_array[] = "Analyse to day is required";
117
+ $jstranslator_array[] = "Forecast until day is required";
118
+ $jstranslator_array[] = "Invalid date format";
119
+ $jstranslator_array[] = "Invalid date format...";
120
+ $jstranslator_array[] = "Analyse from day invalid date";
121
+ $jstranslator_array[] = "Analyse to day invalid date";
122
+ $jstranslator_array[] = "Analyse from day must be before Analyse to day";
123
+ $jstranslator_array[] = "Analyse to day must be in the past";
124
+ $jstranslator_array[] = "Forecast until day must be in the future";
125
+ $jstranslator_array[] = "To have a minimum acceptable precision you must:";
126
+ $jstranslator_array[] = "Select at least";
127
+ $jstranslator_array[] = "days of historical data to be analysed.";
128
+ $jstranslator_array[] = "See";
129
+ $jstranslator_array[] = "how to increase your forecast accuracy";
130
+ $jstranslator_array[] = "There is a limit of";
131
+ $jstranslator_array[] = "days of historical data that can be analysed.";
132
+ $jstranslator_array[] = "Please, select a smaller period.";
133
+ $jstranslator_array[] = "If you need to analyze more data";
134
+ $jstranslator_array[] = "for a custom service.";
135
+ $jstranslator_array[] = "There is a limit of";
136
+ $jstranslator_array[] = "days for forecast.";
137
+ $jstranslator_array[] = "Please, select a smaller period.";
138
+ $jstranslator_array[] = "If you need to process more data";
139
+ $jstranslator_array[] = "for a custom service.";
140
+ $jstranslator_array[] = "You are willing to analyse";
141
+ $jstranslator_array[] = "days";
142
+ $jstranslator_array[] = "of historical data and to forecast";
143
+ $jstranslator_array[] = "To have a minimum acceptable precision you can:";
144
+ $jstranslator_array[] = "Increase the period of historical data to be analysed";
145
+ $jstranslator_array[] = "Decrease the number of forecast days";
146
+ $jstranslator_array[] = "Request error";
147
+ $jstranslator_array[] = "Validating data...";
148
+ $jstranslator_array[] = "Validation error";
149
+ $jstranslator_array[] = "Collecting, splitting and sending data...";
150
+ $jstranslator_array[] = "Unable to request";
151
+ $jstranslator_array[] = "Your request was successfully received.";
152
+ $jstranslator_array[] = "We will process your data and notify you when everything is done.";
153
+ $jstranslator_array[] = "You can check your report status in Forecast history page.";
154
+ $jstranslator_array[] = "Error";
155
+ $jstranslator_array[] = "Info";
156
+ $jstranslator_array[] = "Ok!";
157
+ $jstranslator_array[] = "Click on Step 1: Check forecast price before";
158
+ $jstranslator_array[] = "Ready to go!";
159
+ $jstranslator_array[] = "Request forecast is disabled in this platform.";
160
+
161
+ $jstranslator = '<script>';
162
+ foreach($jstranslator_array as $value){
163
+ $jstranslator .= "Translator.add(\"".$value."\", \"".$this->__($value)."\");";
164
+ }
165
+ $jstranslator .= '</script>';
166
+
167
+
168
+ $notify_email_after = $this->__('You will be notified by email when the report is ready');
169
+ $fieldset->addField('email', 'text', array(
170
+ 'name' => 'email',
171
+ 'label' => $this->__('Your email'),
172
+ 'class' => 'required-entry',
173
+ 'required' => true,
174
+ 'disabled' => false,
175
+ 'image' => $this->getSkinUrl('images/grid-cal.gif'),
176
+ 'value' => $forecast->getEmail(),
177
+ 'disabled' => $disabled,
178
+ ))->setAfterElementHtml('<small>'.$notify_email_after.'</small>'.$jstranslator);
179
+
180
+ // Period start to analyse historical data
181
+ $fieldset->addField('historical_date_start', 'date', array(
182
+ 'name' => 'historical_date_start',
183
+ 'label' => $this->__('Analyse from day'),
184
+ 'class' => 'required-entry',
185
+ 'required' => true,
186
+ 'image' => $this->getSkinUrl('images/grid-cal.gif'),
187
+ 'after_element_html' => '<small>'.$this->__('Start period of historical data to be analysed').'</small>',
188
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_SHORT) ,
189
+ 'onchange' => 'disableRequestButton()',
190
+ 'value' => $forecast->getPeriodStart(),
191
+ 'disabled' => $disabled,
192
+ ));
193
+
194
+ // Period end to analyse historical data
195
+ $fieldset->addField('historical_date_end', 'date', array(
196
+ 'name' => 'historical_date_end',
197
+ 'label' => $this->__('Analyse to day'),
198
+ 'class' => 'required-entry',
199
+ 'required' => true,
200
+ 'disabled' => false,
201
+ 'image' => $this->getSkinUrl('images/grid-cal.gif'),
202
+ 'after_element_html' => '<small>'.$this->__('End period of historical data to be analysed').'</small>',
203
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_SHORT) ,
204
+ 'onchange' => 'disableRequestButton()',
205
+ 'value' => $forecast->getPeriodEnd(),
206
+ 'disabled' => $disabled,
207
+ ));
208
+
209
+ // Field to select the day to forecast
210
+ $fieldset->addField('forecast_date_end', 'date', array(
211
+ 'name' => 'forecast_date_end',
212
+ 'label' => $this->__('Forecast until day'),
213
+ 'class' => 'required-entry',
214
+ 'required' => true,
215
+ 'disabled' => false,
216
+ 'image' => $this->getSkinUrl('images/grid-cal.gif'),
217
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_SHORT) ,
218
+ 'onchange' => 'disableRequestButton()',
219
+ 'value' => $forecast->getForecastDateEnd(),
220
+ 'disabled' => $disabled,
221
+ ))->setAfterElementHtml("
222
+ <small>
223
+ <span id='forecast_ahead_info'>"
224
+ .$this->__('Forecast from today until the selected day').
225
+ "</span>
226
+ </small>
227
+ ");
228
+
229
+ $after_html = "";
230
+ $after_html .= "<small>
231
+ <span id='request_forecast_price_info' style='font-weight:bold'><br></span>
232
+ </small>";
233
+ if(!$disabled){
234
+ $after_html .= "<script type=\"text/javascript\">
235
+ </script>";
236
+ }
237
+ $fieldset->addField(
238
+ 'request_forecast_price', 'button', array(
239
+ 'name' => 'request_forecast_price',
240
+ 'label' => $this->__('Check forecast price'),
241
+ 'value' => $this->__('Step 1: Check forecast price'),
242
+ 'class' => 'form-button',
243
+ 'onclick' => 'callCheckReportDetails(
244
+ \''.$account_info_json.'\'
245
+ ,\''.$this->getUrl('get4cast_salesforecast_admin/forecast/getForecastPrice').'\')',
246
+ )
247
+ )->setAfterElementHtml($after_html);
248
+
249
+ // Button to call ajax functions to request forecast
250
+ $request_urls = array();
251
+ $request_urls['request_validate_url'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestValidate');
252
+ $request_urls['request_analyse_data_base_size_url'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestAnalyseDataBaseSize');
253
+ $request_urls['request_collect_url'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestCollect');
254
+ $request_urls['request_split_url'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestSplit');
255
+ $request_urls['request_send_url'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestSend');
256
+ $request_urls['request_best_fit'] = $this->getUrl('get4cast_salesforecast_admin/forecast/requestBestFit');
257
+
258
+ $request_urls = htmlspecialchars(Mage::helper('core')
259
+ ->jsonEncode($request_urls));
260
+
261
+ $after_html = "";
262
+ $after_html .= "<small>
263
+ <span id='request_forecast_info' style=''></span>
264
+ </small>";
265
+ $after_html .= "<script type=\"text/javascript\">
266
+ disableRequestButton();
267
+ </script>";
268
+ if($disabled){
269
+ $after_html .= "<script type=\"text/javascript\">
270
+ disableRequestButton(false, 'request_forecast_price');
271
+ </script>";
272
+ }
273
+ $fieldset->addField(
274
+ 'request_forecast', 'button', array(
275
+ 'name' => 'request_forecast',
276
+ 'label' => $this->__('Request forecast'),
277
+ 'value' => $this->__('Step 2: Request forecast'),
278
+ 'class' => 'form-button',
279
+ 'onclick' => 'requestValidate(\''.$request_urls.'\')',
280
+ 'disabled' => true,
281
+ )
282
+ )->setAfterElementHtml($after_html);
283
+
284
+ if($forecast->getStatus() == 'request result' || $forecast->getStatus() == 'result requested' || $forecast->getStatus() == 'complete'){
285
+
286
+ $after_html = "";
287
+ $after_html .= "<small>
288
+ <span id='request_result_info' style=''></span>
289
+ </small>";
290
+
291
+ $disable_result = $forecast->getStatus() == 'request result' ? false : true;
292
+ if($disable_result || $enable_request_report == 'false'){
293
+ $after_html .= "<script type=\"text/javascript\">
294
+ disableRequestButton(false, 'request_result');
295
+ </script>";
296
+ }
297
+
298
+
299
+ $fieldset->addField(
300
+ 'request_result', 'button', array(
301
+ 'name' => 'request_result',
302
+ 'label' => $this->__('Request result comparison'),
303
+ 'value' => $this->__('Step 3: Request result'),
304
+ 'class' => 'form-button',
305
+ 'onclick' => 'disableRequestButton(false, \'request_result\');requestValidate(\''.$request_urls.'\')',
306
+ 'disabled' => $disable_result,
307
+ )
308
+ )->setAfterElementHtml($after_html);
309
+ }
310
+
311
+ $fieldset->addField('h_report_key', 'hidden', array(
312
+ 'name' => 'h_report_key',
313
+ 'value' => $forecast->getReportKey(),
314
+ ));
315
+
316
+ $fieldset->addField('h_forecast_quote_key', 'hidden', array(
317
+ 'name' => 'h_forecast_quote_key',
318
+ 'value' => '',
319
+ ));
320
+
321
+ $h_data_type = $disabled ? 'result' : 'input';
322
+ $fieldset->addField('h_data_type', 'hidden', array(
323
+ 'name' => 'h_data_type',
324
+ 'value' => $h_data_type,
325
+ ));
326
+
327
+ $h_forecast_date_start = $forecast->getId() ? substr($forecast->getForecastDateStart(), 0, 10) : date('Y-m-d');
328
+ $h_forecast_date_end = $forecast->getId() ? substr($forecast->getForecastDateEnd(), 0, 10) : '';
329
+ $fieldset->addField('h_forecast_date_start', 'hidden', array(
330
+ 'name' => 'h_forecast_date_start',
331
+ 'value' => $h_forecast_date_start,
332
+ ));
333
+
334
+ $after_html = "";
335
+ if($disabled){
336
+ if($h_forecast_date_start
337
+ && $h_forecast_date_end
338
+ && isset($account_info['core_config']['limit_data_split']))
339
+ {
340
+ $collect = Mage::getModel('get4cast_salesforecast/collect');
341
+ $count_orders = $collect->getOrders(1 // store group ID
342
+ , $h_forecast_date_start
343
+ , $h_forecast_date_end
344
+ , false // limit
345
+ , false // page
346
+ , false // filters
347
+ , true); // get only count
348
+
349
+ $data_size = $count_orders;
350
+
351
+ $split_pages = round($data_size / $account_info['core_config']['limit_data_split']);
352
+ if ($split_pages == 0) {
353
+ $split_pages = 1;
354
+ }
355
+
356
+ $split_per_page = round($data_size / $split_pages);
357
+
358
+ $after_html .= "<script type=\"text/javascript\">
359
+ global_split_pages = ".$split_pages.";
360
+ global_split_per_page = ".$split_per_page.";
361
+ global_data_size = ".$data_size.";
362
+ </script>";
363
+ }
364
+ }
365
+
366
+ $after_html .= "<script type=\"text/javascript\">
367
+ global_enable_request_report = ".$enable_request_report.";
368
+ </script>";
369
+
370
+ $fieldset->addField('h_forecast_date_end', 'hidden', array(
371
+ 'name' => 'h_forecast_date_end',
372
+ 'value' => $h_forecast_date_end,
373
+ ))->setAfterElementHtml($after_html);
374
+
375
+
376
+ $fieldset->addField('h_historical_date_start', 'hidden', array(
377
+ 'name' => 'h_historical_date_start',
378
+ 'value' => '',
379
+ ));
380
+ $fieldset->addField('h_historical_date_end', 'hidden', array(
381
+ 'name' => 'h_historical_date_end',
382
+ 'value' => '',
383
+ ));
384
+ $fieldset->addField('h_historical_diff', 'hidden', array(
385
+ 'name' => 'h_historical_diff',
386
+ 'value' => '',
387
+ ));
388
+ $fieldset->addField('h_forecast_diff', 'hidden', array(
389
+ 'name' => 'h_forecast_diff',
390
+ 'value' => '',
391
+ ));
392
+
393
+ $fieldset->addField('h_date_format', 'hidden', array(
394
+ 'name' => 'h_date_format',
395
+ 'value' => $helper->getDateFormat(),
396
+ ));
397
+
398
+ $limit_best_fit_historical_days = '';
399
+ if(isset($account_info['core_config']['limit_best_fit_historical_days'])){
400
+ $limit_best_fit_historical_days = $account_info['core_config']['limit_best_fit_historical_days'];
401
+ }
402
+ $fieldset->addField('h_limit_best_fit_historical_days', 'hidden', array(
403
+ 'name' => 'h_limit_best_fit_historical_days',
404
+ 'value' => $limit_best_fit_historical_days,
405
+ ));
406
+
407
+
408
+ if (Mage::getSingleton('adminhtml/session')->getFondationData()) {
409
+ $form->setValues(Mage::getSingleton('adminhtml/session')->getFondationData());
410
+ Mage::getSingleton('adminhtml/session')->setFondationData(null);
411
+ } elseif (Mage::registry('fondation_data')) {
412
+ $form->setValues(Mage::registry('fondation_data')->getData());
413
+ }
414
+
415
+ return $this;
416
+ }
417
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Forecast/Grid.php ADDED
@@ -0,0 +1,111 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Block_Adminhtml_Forecast_Grid
3
+ extends Mage_Adminhtml_Block_Widget_Grid
4
+ {
5
+ protected function _prepareCollection()
6
+ {
7
+ $collection = Mage::getResourceModel(
8
+ 'get4cast_salesforecast/forecast_collection'
9
+ );
10
+ $this->setCollection($collection);
11
+
12
+ return parent::_prepareCollection();
13
+ }
14
+
15
+ public function getRowUrl($row)
16
+ {
17
+ $action_name = $this->getRequest()->getActionName();
18
+ $admin_session = Mage::getSingleton('admin/session');
19
+ if($admin_session->isAllowed('admin/report/get4cast_salesforecast/sales_forecast/new_forecast')){
20
+ return $this->getUrl(
21
+ 'get4cast_salesforecast_admin/forecast/edit',
22
+ array(
23
+ 'id' => $row->getId()
24
+ )
25
+ );
26
+ }
27
+ return null;
28
+
29
+ }
30
+
31
+ protected function _prepareColumns()
32
+ {
33
+ $this->addColumn('entity_id', array(
34
+ 'header' => $this->__('ID'),
35
+ 'type' => 'number',
36
+ 'index' => 'entity_id',
37
+ 'align' =>'center',
38
+ ));
39
+ $this->setDefaultSort('entity_id');
40
+ $this->setDefaultDir('desc');
41
+
42
+ $this->addColumn('created_at', array(
43
+ 'header' => $this->__('Created at'),
44
+ 'type' => 'datetime',
45
+ 'index' => 'created_at',
46
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_MEDIUM),
47
+ ));
48
+
49
+ $this->addColumn('updated_at', array(
50
+ 'header' => $this->__('Updated at'),
51
+ 'type' => 'date',
52
+ 'index' => 'updated_at',
53
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_MEDIUM),
54
+ ));
55
+
56
+ $this->addColumn('store_group_name', array(
57
+ 'header' => $this->__('Store'),
58
+ 'type' => 'text',
59
+ 'index' => 'store_group_name',
60
+ ));
61
+
62
+ $this->addColumn('period_start', array(
63
+ 'header' => $this->__('Period start'),
64
+ 'type' => 'date',
65
+ 'index' => 'period_start',
66
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_MEDIUM),
67
+ ));
68
+
69
+ $this->addColumn('period_end', array(
70
+ 'header' => $this->__('Period end'),
71
+ 'type' => 'date',
72
+ 'index' => 'period_end',
73
+ 'format' => Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_MEDIUM),
74
+ ));
75
+
76
+ $this->addColumn('forecast_date_end', array(
77
+ 'header' => $this->__('Forecast day'),
78
+ 'type' => 'date',
79
+ 'index' => 'forecast_date_end',
80
+ ));
81
+
82
+ $this->addColumn('price', array(
83
+ 'header' => $this->__('Report price'),
84
+ 'type' => 'text',
85
+ 'index' => 'price',
86
+ ));
87
+
88
+ $this->addColumn('payment_status', array(
89
+ 'header' => $this->__('Payment status'),
90
+ 'type' => 'text',
91
+ 'index' => 'payment_status',
92
+ 'renderer' => 'Get4Cast_SalesForecast_Block_Adminhtml_Translate'
93
+ ));
94
+
95
+ $this->addColumn('url', array(
96
+ 'header' => $this->__('Open report'),
97
+ 'type' => 'text',
98
+ 'index' => 'url',
99
+ 'renderer' => 'Get4Cast_SalesForecast_Block_Adminhtml_Url'
100
+ ));
101
+
102
+ $this->addColumn('status', array(
103
+ 'header' => $this->__('Report status'),
104
+ 'type' => 'text',
105
+ 'index' => 'status',
106
+ 'renderer' => 'Get4Cast_SalesForecast_Block_Adminhtml_Translate'
107
+ ));
108
+
109
+ return parent::_prepareColumns();
110
+ }
111
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Link.php ADDED
@@ -0,0 +1,19 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Get4Cast_SalesForecast_Block_Adminhtml_Link
4
+ extends Mage_Adminhtml_Block_Widget
5
+ implements Varien_Data_Form_Element_Renderer_Interface
6
+ {
7
+ public function render(Varien_Data_Form_Element_Abstract $element) {
8
+ $forecast = $this->getData('report_link');
9
+
10
+ $html = '<td class="label">'.$element->getLabelHtml().'</td>';
11
+ $html .= '<td class = "value">';
12
+ $html .= '<ul>';
13
+ $html .= "<a href='".$forecast->getUrl()."' target='_blank'>".$this->__('Click here to open your report')."</a>";
14
+ $html .= '</ul>';
15
+ $html .= '</td>';
16
+
17
+ return $html;
18
+ }
19
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Translate.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Get4Cast_SalesForecast_Block_Adminhtml_Translate
4
+ extends Mage_Adminhtml_Block_Widget_Grid_Column_Renderer_Abstract
5
+ {
6
+ public function render(Varien_Object $row) {
7
+ $value = $row->getData($this->getColumn()->getIndex());
8
+ return $this->__($value);
9
+ }
10
+ }
app/code/community/Get4Cast/SalesForecast/Block/Adminhtml/Url.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Get4Cast_SalesForecast_Block_Adminhtml_Url
4
+ extends Mage_Adminhtml_Block_Widget_Grid_Column_Renderer_Abstract
5
+ {
6
+ public function render(Varien_Object $row) {
7
+ $value = $row->getData($this->getColumn()->getIndex());
8
+ return '<a target=\'_blank\' href="'.$value.'">'.$this->__('Click here to open your report').'</a>';
9
+ }
10
+ }
app/code/community/Get4Cast/SalesForecast/Helper/Data.php ADDED
@@ -0,0 +1,115 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Helper_Data
3
+ extends Mage_Core_Helper_Abstract
4
+ {
5
+ public function todayPlus($plus){
6
+ $date_format = Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_SHORT);
7
+ $date_format = str_replace('yyyy', 'Y', $date_format);
8
+ $result = Date($date_format, strtotime('+'.$plus.' days'));
9
+ return $result;
10
+ }
11
+
12
+ public function getAllStores($ids_from_group = null){
13
+ $return = array();
14
+ $return_group = array();
15
+
16
+ $all_stores = Mage::app()->getStores();
17
+
18
+ foreach ($all_stores as $store_id => $store){
19
+ $_store = Mage::getModel('core/store')->load($store_id);
20
+ $website = Mage::getModel('core/website')->load($_store->getWebsiteId());
21
+
22
+ $store_info['group_id'] = $_store->getGroupId();
23
+
24
+ $store_info['website_id'] = $website->getId();
25
+ $store_info['website_name'] = $website->getName();
26
+
27
+ $store_info['store_id'] = $store_id;
28
+ $store_info['store_name'] = $store->getFrontendName();
29
+
30
+ $store_info['view_id'] = $store_id;
31
+ $store_info['view_unsecure_url'] = $store->getUrl();
32
+ $store_info['view_secure_url'] = $store->getUrl('',array('_forced_secure'=>true));
33
+ $store_info['view_code'] = Mage::app()->getStore($store_id)->getCode();
34
+ $store_info['view_name'] = Mage::app()->getStore($store_id)->getName();
35
+
36
+ if($ids_from_group == $store_info['group_id']){
37
+ $return_group[] = $store_info['store_id'];
38
+ }
39
+
40
+ $return[] = $store_info;
41
+ }
42
+ if($ids_from_group){
43
+ return $return_group;
44
+ }
45
+ return $return;
46
+ }
47
+
48
+ public function getAllStoresToSelect(){
49
+ $return = array();
50
+ $all_stores = $this->getAllStores();
51
+
52
+ foreach ($all_stores as $store_id => $store){
53
+ if(!array_key_exists($store['store_id'], $return)){
54
+ $return[$store['store_id']] = $store['website_name'].' ('.$store['view_code'].')';
55
+ }
56
+ }
57
+ return $return;
58
+ }
59
+
60
+ public function updateTimestamps($obj){
61
+ $timestamp = now();
62
+ if ($obj->isObjectNew()) {
63
+ $obj->setCreatedAt($timestamp);
64
+ } else {
65
+ $obj->setUpdatedAt($timestamp);
66
+ }
67
+ return $obj;
68
+ }
69
+
70
+ public function getModuleVersion(){
71
+ return Mage::getConfig()
72
+ ->getModuleConfig('Get4Cast_SalesForecast')
73
+ ->version
74
+ ->__toString();
75
+ }
76
+
77
+ public function getDateFormat(){
78
+ $date_format = Mage::app()->getLocale()->getDateFormat(Mage_Core_Model_Locale::FORMAT_TYPE_SHORT);
79
+ $date_format = strtolower($date_format);
80
+ $date_format = preg_replace('/(y)\\1+/', 'Y', $date_format);
81
+ $date_format = preg_replace('/(m)\\1+/', 'm', $date_format);
82
+ $date_format = preg_replace('/(d)\\1+/', 'd', $date_format);
83
+
84
+ return $date_format;
85
+ }
86
+
87
+ public function diffDates($start = 'today', $end){
88
+ $date_format = $this->getDateFormat();
89
+ $date_format = explode('/', $date_format);
90
+
91
+ if($start == 'today'){
92
+ $start = Mage::getModel('core/date')->date($this->getDateFormat());
93
+ }
94
+
95
+ $day_index = array_search('d', $date_format);
96
+ $month_index = array_search('m', $date_format);
97
+ $year_index = array_search('Y', $date_format);
98
+
99
+ $start = explode('/', $start);
100
+ $end = explode('/', $end);
101
+
102
+ $start = $start[$year_index].'-'
103
+ .$start[$month_index].'-'
104
+ .$start[$day_index];
105
+ $end = $end[$year_index].'-'
106
+ .$end[$month_index].'-'
107
+ .$end[$day_index];
108
+
109
+ $start = strtotime($start);
110
+ $end = strtotime($end);
111
+ $difference = $end - $start;
112
+ $diff_days = floor($difference/(60*60*24));
113
+ return $diff_days;
114
+ }
115
+ }
app/code/community/Get4Cast/SalesForecast/Model/Apiclient.php ADDED
@@ -0,0 +1,200 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Apiclient
3
+ extends Mage_Core_Model_Abstract
4
+ {
5
+ public $register_called = false;
6
+
7
+ protected function _construct()
8
+ {
9
+ $this->api_url = Mage::getStoreConfig('get4cast/default/api_url');
10
+ }
11
+
12
+ public function httpRequest($path, $method = 'post', $data = null){
13
+ try{
14
+ if($data === null){
15
+ $data = array();
16
+ }
17
+
18
+ $config_model = Mage::getModel('get4cast_salesforecast/config');
19
+
20
+ $_config_app = $config_model->getCollection()
21
+ ->addFilter('config','app')
22
+ ->getFirstItem();
23
+
24
+ $_config_app_value = $_config_app->getValue();
25
+ $_config_app_array = Mage::helper('core')->jsonDecode($_config_app_value);
26
+
27
+ if($this->register_called){
28
+ $data['_app_key'] = null;
29
+ } elseif(!isset($_config_app_array['app_key'])){
30
+ Throw new Exception('Could not get app key');
31
+ }
32
+
33
+ $data['_app_key'] = $_config_app_array['app_key'];
34
+
35
+ $helper = Mage::helper('get4cast_salesforecast/data');
36
+ $data['_module_version'] = $helper->getModuleVersion();
37
+
38
+ $url = $this->api_url.'/'.$path;
39
+
40
+ $data_json = Mage::helper('core')->jsonEncode($data);
41
+ $data_json = urlencode($data_json);
42
+ $send_data = '';
43
+ if($data_json){
44
+ $send_data = '&p='.$data_json;
45
+ }
46
+
47
+ Mage::log('..........................');
48
+ Mage::log($send_data);
49
+ Mage::log('..........................');
50
+
51
+ if(strtolower($method) == 'post'){
52
+ return $this->httpPost($url, $send_data);
53
+ } elseif(strtolower($method) == 'get'){
54
+ return $this->httpGet($url, $send_data);
55
+ }
56
+ } catch (Exception $e) {
57
+ Mage::log('G4C-MAG-21Pf: '.$e->getMessage());
58
+ return false;
59
+ }
60
+ }
61
+
62
+ public function httpPost($url, $data){
63
+ try{
64
+ $ch = curl_init($url);
65
+ curl_setopt($ch, CURLOPT_POST, 1);
66
+ curl_setopt($ch, CURLOPT_POSTFIELDS, $data);
67
+ curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
68
+ curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
69
+ curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
70
+ curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 30);
71
+
72
+ $response = curl_exec($ch);
73
+
74
+ $return_code = (int)curl_getinfo( $ch, CURLINFO_HTTP_CODE);
75
+ if($return_code != 200){
76
+ Throw new Exception('Could not get API response');
77
+ }
78
+
79
+ curl_close($ch);
80
+
81
+ return $response;
82
+
83
+ } catch (Exception $e) {
84
+ Mage::log('G4C-MAG-nlZP: '.$e->getMessage());
85
+ return false;
86
+ }
87
+ }
88
+
89
+ public function register(){
90
+ try{
91
+ //Check if app is already registered
92
+ $config_model = Mage::getModel('get4cast_salesforecast/config');
93
+
94
+ $_config_app = $config_model->getCollection()
95
+ ->addFilter('config','app')
96
+ ->getFirstItem();
97
+
98
+ $_config_app_value = $_config_app->getValue();
99
+ $_config_app_array = Mage::helper('core')->jsonDecode($_config_app_value);
100
+
101
+ if(isset($_config_app_array['app_key'])){
102
+ return $_config_app_value;
103
+ }
104
+
105
+ //Register app...
106
+ $this->register_called = true;
107
+
108
+ //POST data to API server for security reasons
109
+ $data = array();
110
+ $data['server'] = $_SERVER;
111
+ $data['store'] = Mage::helper('get4cast_salesforecast/data')
112
+ ->getAllStores();
113
+
114
+ $data['magento_version'] = Mage::getVersion();
115
+
116
+ //Request register
117
+ $response = $this->httpRequest('Install/register', 'POST', $data);
118
+ $response = Mage::helper('core')->jsonDecode($response);
119
+
120
+ if(isset($response['_error'])){
121
+ Throw new Exception($response['_error']);
122
+ }
123
+
124
+ if(!$response || !isset($response['app_key'])){
125
+ Throw new Exception('Request register returned wrong parameters');
126
+ }
127
+
128
+ //Save app register
129
+ $_config_app->setConfig('app');
130
+ $_config_app->setValue(Mage::helper('core')->jsonEncode($response));
131
+ $saved = $_config_app->save();
132
+
133
+ if(!$saved){
134
+ Throw new Exception('Register key received but could not save it');
135
+ }
136
+
137
+ return $response;
138
+
139
+ } catch (Exception $e) {
140
+ Mage::log('G4C-MAG-4yrx: '.$e->getMessage());
141
+ return false;
142
+ }
143
+ }
144
+
145
+ public function getAccountInfo(){
146
+ try{
147
+ $response = $this->httpRequest('install/getInfo', 'POST');
148
+ $response = Mage::helper('core')->jsonDecode($response);
149
+
150
+ if(isset($response['_error'])){
151
+ Throw new Exception($response['_error']);
152
+ }
153
+
154
+ return $response;
155
+ } catch (Exception $e) {
156
+ Mage::log('G4C-MAG-b2rq: '.$e->getMessage());
157
+ return false;
158
+ }
159
+ }
160
+
161
+ public function getForecastPrice($get_data){
162
+ try{
163
+
164
+ $response = $this->httpRequest('forecast/getPrice', 'POST', $get_data);
165
+
166
+ $response = Mage::helper('core')->jsonDecode($response);
167
+
168
+ if(isset($response['_error'])){
169
+ Throw new Exception($response['_error']);
170
+ }
171
+
172
+ return $response;
173
+
174
+ } catch (Exception $e) {
175
+ Mage::log('G4C-MAG-ZcKc: '.$e->getMessage());
176
+ return false;
177
+ }
178
+ }
179
+
180
+
181
+ public function requestSend($post_data){
182
+ try{
183
+
184
+ $response = $this->httpRequest('Forecast/requestForecast', 'POST', $post_data);
185
+
186
+ $response = Mage::helper('core')->jsonDecode($response);
187
+
188
+ if(isset($response['_error'])){
189
+ Throw new Exception($response['_error']);
190
+ }
191
+
192
+ } catch (Exception $e) {
193
+ Mage::log('G4C-MAG-fcnW: '.$e->getMessage());
194
+ }
195
+
196
+ return $response;
197
+ }
198
+
199
+
200
+ }
app/code/community/Get4Cast/SalesForecast/Model/Collect.php ADDED
@@ -0,0 +1,262 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Collect
3
+ extends Mage_Core_Model_Abstract
4
+ {
5
+ protected function _construct(){
6
+ $this->_init('get4cast_salesforecast/forecast');
7
+ }
8
+
9
+ public function getForecastPrice($get_data){
10
+ try{
11
+ $api_reponse = '';
12
+ if(!$get_data){
13
+ return Mage::helper('core')->jsonEncode($api_reponse);
14
+ }
15
+
16
+ $get_data['data_size'] = $this->getOrders($get_data['store_group_id']
17
+ , $get_data['h_historical_date_start']
18
+ , $get_data['h_historical_date_end']
19
+ , false // limit
20
+ , false // page
21
+ , false // filter
22
+ , true); // get only count
23
+
24
+ if(!$get_data['data_size']){
25
+ $api_reponse['error'] = 'There is no data in the selected period';
26
+ return Mage::helper('core')->jsonEncode($api_reponse);
27
+ }
28
+
29
+ $orders = $this->getOrders($get_data['store_group_id']
30
+ , $get_data['h_historical_date_start']
31
+ , $get_data['h_historical_date_end']
32
+ , false // limit
33
+ , false // page
34
+ , array('state'=>'complete'));
35
+
36
+ $ticket = 0;
37
+ foreach($orders as $order){
38
+ $ticket += $order->getGrandTotal();
39
+ }
40
+ $average_ticket = $ticket/count($orders);
41
+ $get_data['average_ticket'] = number_format($average_ticket, 2, '.', '');
42
+
43
+ $get_data['store_group_url'] = Mage::app()->getStore($get_data['store_group_id'])->getBaseUrl();
44
+
45
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
46
+ $api_reponse = $api_client->getForecastPrice($get_data);
47
+
48
+ return Mage::helper('core')->jsonEncode($api_reponse);
49
+
50
+ } catch (Exception $e) {
51
+ Mage::log('G4C-MAG-6CAo: '.$e->getMessage());
52
+ return false;
53
+ }
54
+ }
55
+
56
+ public function getOrders($store_group_id, $from_date, $to_date, $limit = false, $page = false, $field_filter = false, $only_get_count = false){
57
+
58
+ $helper = Mage::helper('get4cast_salesforecast/data');
59
+
60
+ $stores = $store_group_id;
61
+
62
+ $magento_core_date = Mage::getSingleton('core/date');
63
+ $date_format = 'Y-m-d H:i:s';
64
+
65
+ $from_date = str_replace('/', '-', $from_date);
66
+ $from_date .= ' 00:00:00';
67
+ $from_date = $magento_core_date->gmtDate($date_format, $from_date);
68
+
69
+ $to_date = str_replace('/', '-', $to_date);
70
+ $to_date .= ' 23:59:59';
71
+ $to_date = $magento_core_date->gmtDate($date_format, $to_date);
72
+
73
+ $orders = Mage::getSingleton('sales/order')
74
+ ->getCollection()
75
+ ->addAttributeToFilter('store_id', $stores)
76
+ ->addAttributeToFilter('created_at', array('from'=>$from_date, 'to'=>$to_date));
77
+
78
+ if($limit){
79
+ $orders->setPageSize($limit);
80
+ }
81
+
82
+ if($page){
83
+ $orders->setCurPage($page);
84
+ }
85
+
86
+ if(is_array($field_filter)){
87
+ foreach($field_filter as $key=>$value){
88
+ $orders->addFieldToFilter($key, $value);
89
+ }
90
+ }
91
+
92
+ if($only_get_count){
93
+ return $orders->getSize();
94
+ }
95
+
96
+ return $orders;
97
+ }
98
+
99
+ public function requestCollect($get_data){
100
+ try{
101
+
102
+ $enable_request_report = Mage::getStoreConfig('get4cast/default/enable_request_report');
103
+ if(!$enable_request_report){
104
+ throw new Exception('Request forecast is disabled in this platform');
105
+ }
106
+
107
+ $api_reponse = '';
108
+ if(!$get_data){
109
+ return Mage::helper('core')->jsonEncode($api_reponse);
110
+ }
111
+
112
+ $from_date = $get_data['h_historical_date_start'];
113
+ $to_date = $get_data['h_historical_date_end'];
114
+ if($get_data['h_data_type'] == 'result'){
115
+ $from_date = $get_data['h_forecast_date_start'];
116
+ $to_date = $get_data['h_forecast_date_end'];
117
+ }
118
+
119
+ $limit = $get_data['limit'];
120
+ $page = $get_data['page'];
121
+
122
+ $orders = $this->getOrders($get_data['store_group_id']
123
+ , $from_date
124
+ , $to_date
125
+ , $limit
126
+ , $page
127
+ );
128
+
129
+ $report_data = array();
130
+ $report_data['metadata'] = array();
131
+ $report_data['metadata']['limit'] = $limit;
132
+ $report_data['metadata']['page'] = $page;
133
+ $report_data['metadata']['pages'] = $get_data['pages'];
134
+ $report_data['metadata']['currency'] = Mage::app()->getStore()->getCurrentCurrencyCode();
135
+ $report_data['metadata']['type'] = $get_data['h_data_type'];
136
+ $report_data['metadata']['report_key'] = $get_data['h_report_key'];
137
+ $report_data['metadata']['forecast_quote_key'] = $get_data['h_forecast_quote_key'];
138
+
139
+ $store_timezone = Mage::getStoreConfig('general/locale/timezone');
140
+ $report_data['metadata']['timezone'] = $store_timezone;
141
+
142
+ $time_zone = new DateTimeZone($store_timezone);
143
+ $stock_qty = array();
144
+
145
+ foreach($orders as $order){
146
+ $order_info = array();
147
+ $get_data_order = $order->getData();
148
+ $order_info['id'] = $order->getId();
149
+
150
+ $date = new DateTime($order->getCreatedAt().' GMT');
151
+ $date->setTimezone($time_zone);
152
+ $order_info['created_at_gmt'] = $order->getCreatedAt();
153
+ $order_info['created_at_locale'] = $date->format('Y-m-d H:i:s');
154
+
155
+ $date = new DateTime($order->getUpdatedAt().' GMT');
156
+ $date->setTimezone($time_zone);
157
+ $order_info['updated_at_gmt'] = $order->getUpdatedAt();
158
+ $order_info['updated_at_locale'] = $date->format('Y-m-d H:i:s');
159
+
160
+ $order_info['cupom_code'] = $order->getCupomCode();
161
+ $order_info['state'] = $order->getState();
162
+ $order_info['store_id'] = $order->getStoreId();
163
+ $order_info['base_grand_total'] = number_format($get_data_order['base_grand_total'], 2, '.', '');
164
+ $order_info['payment_discount'] = number_format($order->getPaymentDiscount(), 2, '.', '');
165
+ $order_info['shipping_amount'] = $order->getShippingAmount();
166
+ $order_info['increment_id'] = $order->getIncrementId();
167
+ $order_info['base_currency_code'] = $order->getBaseCurrencyCode();
168
+ $order_info['total_item_count'] = $order->getTotalItemCount();
169
+
170
+ $payment_data = $order->getPayment()->getData();
171
+ $payment_info = array();
172
+ $payment_info['additional_data'] = $payment_data['additional_data'];
173
+ $payment_info['method'] = $payment_data['method'];
174
+ $order_info['payment_data'] = $payment_info;
175
+
176
+ $status_history = array();
177
+ $order_status_history = $order->getStatusHistoryCollection()->setOrder('created_at', 'DESC');
178
+
179
+ if($order->getState() == 'canceled' && $order_status_history){
180
+ foreach($order_status_history as $history){
181
+ $index = $history->getEntityId();
182
+ $status_history_aux[$index] = array();
183
+ $status_history_aux[$index]['status'] = $history->getStatus();
184
+ $status_history_aux[$index]['created_at'] = $history->getCreatedAt();
185
+ $status_history_aux[$index]['entity_id'] = $history->getEntityName();
186
+ }
187
+ ksort($status_history_aux);
188
+ reset($status_history_aux);
189
+ $status_history[0] = current($status_history_aux);
190
+ $status_history[1] = end($status_history_aux);
191
+ }
192
+ $order_info['status_history'] = $status_history;
193
+
194
+ $order_info['items'] = array();
195
+ $items = $order->getItemsCollection();
196
+ foreach($items as $item){
197
+ $item_info = array();
198
+ $get_data = $item->getData();
199
+
200
+ if(!isset($stock_qty[$item->getSku()])){
201
+ $_product = Mage::getModel('catalog/product')->loadByAttribute('sku', $item->getSku());
202
+ $_stock_item = Mage::getModel('cataloginventory/stock_item')->loadByProduct($_product);
203
+ $stock_qty[$item->getSku()] = (int)$_stock_item->getQty();
204
+ }
205
+
206
+ $item_info['id'] = $item->getId();
207
+ $item_info['name'] = $item->getName();
208
+ $item_info['sku'] = $item->getSku();
209
+ $item_info['qty_ordered'] = $get_data['qty_ordered'];
210
+ $item_info['price_incl_tax'] = number_format($item->getPriceInclTax(), 2, '.', '');
211
+ $item_info['weight'] = $item->getWeight();
212
+ $item_info['stock_qty'] = $stock_qty[$item->getSku()];
213
+ $order_info['items'][] = $item_info;
214
+ }
215
+
216
+ $customer_info = array();
217
+
218
+ if($order->getCustomerId()){
219
+ $_customer = Mage::getModel('customer/customer')->load($order->getCustomerId());
220
+ $customer_info['email'] = $order->getCustomerEmail();
221
+ $customer_info['id'] = $order->getCustomerId();
222
+ $customer_info['gender'] = $_customer->getResource()->getAttribute('gender')->getSource()->getOptionText($_customer->getData('gender'));
223
+ $customer_info['is_guest'] = 0;
224
+ $customer_info['day_of_birth'] = substr($_customer->getDob(), 0, 10);
225
+ }
226
+ else{
227
+ $customer_info['email'] = $order->getBillingAddress()->getEmail();
228
+ $customer_info['id'] = null;
229
+ $customer_info['gender'] = null;
230
+ $customer_info['is_guest'] = 1;
231
+ $customer_info['day_of_birth'] = null;
232
+ }
233
+
234
+ $customer_info['shipping_address'] = array();
235
+ $customer_info['shipping_address']['zipcode'] = $order->getShippingAddress()->getPostcode();
236
+ $customer_info['shipping_address']['country'] = $order->getShippingAddress()->getCountry();
237
+ $customer_info['shipping_address']['region'] = $order->getShippingAddress()->getRegion();
238
+ $customer_info['shipping_address']['city'] = $order->getShippingAddress()->getCity();
239
+
240
+ $customer_info['billing_address'] = array();
241
+ $customer_info['billing_address']['zipcode'] = $order->getBillingAddress()->getPostcode();
242
+ $customer_info['billing_address']['country'] = $order->getBillingAddress()->getCountry();
243
+ $customer_info['billing_address']['region'] = $order->getBillingAddress()->getRegion();
244
+ $customer_info['billing_address']['city'] = $order->getBillingAddress()->getCity();
245
+
246
+ $order_info['customer'] = $customer_info;
247
+
248
+ $report_data['data']['orders'][] = $order_info;
249
+ }
250
+
251
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
252
+ $api_reponse = $api_client->requestSend($report_data);
253
+
254
+ return Mage::helper('core')->jsonEncode($api_reponse);
255
+
256
+ } catch (Exception $e) {
257
+ Mage::log('G4C-MAG-X0j3: '.$e->getMessage());
258
+ return false;
259
+ }
260
+ }
261
+ }
262
+
app/code/community/Get4Cast/SalesForecast/Model/Config.php ADDED
@@ -0,0 +1,32 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Config
3
+ extends Mage_Core_Model_Abstract
4
+ {
5
+ protected function _construct()
6
+ {
7
+ $this->_init('get4cast_salesforecast/config');
8
+ }
9
+
10
+ public function getApiKey(){
11
+ $_config_app = $this->getCollection()
12
+ ->addFilter('config','app')
13
+ ->getFirstItem();
14
+
15
+ $_config_app_value = $_config_app->getValue();
16
+ $_config_app_json = Mage::helper('core')->jsonDecode($_config_app_value);
17
+
18
+ if(isset($_config_app_json['app_key'])){
19
+ return $_config_app_value;
20
+ }
21
+
22
+ return false;
23
+ }
24
+
25
+ protected function _beforeSave()
26
+ {
27
+ parent::_beforeSave();
28
+ Mage::helper('get4cast_salesforecast/data')
29
+ ->updateTimestamps($this);
30
+ return $this;
31
+ }
32
+ }
app/code/community/Get4Cast/SalesForecast/Model/Forecast.php ADDED
@@ -0,0 +1,109 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Forecast
3
+ extends Mage_Core_Model_Abstract
4
+ {
5
+ const VISIBILITY_HIDDEN = '0';
6
+ const VISIBILITY_DIRECTORY = '1';
7
+
8
+ protected function _construct()
9
+ {
10
+ $this->_init('get4cast_salesforecast/forecast');
11
+ }
12
+
13
+ public function getForecastDaysAhead()
14
+ {
15
+ $session_server_config = Mage::getSingleton('core/session')
16
+ ->getGet4CastServerConfig();
17
+
18
+ $days_available = $session_server_config['forecast_days_available'];
19
+ $price_per_day = (float)$session_server_config['price_per_day'];
20
+
21
+ $account_balance = Mage::getSingleton('core/session')
22
+ ->getGet4CastAccountBalance($account_balance);
23
+ if($account_balance){
24
+ // Flash session
25
+ Mage::getSingleton('core/session')
26
+ ->setGet4CastAccountBalance(null);
27
+ } else {
28
+ $api_client = Mage::getSingleton('get4cast_salesforecast/apiclient');
29
+ $account_balance = $api_client->getAccountBalance();
30
+ }
31
+
32
+ $limit = $account_balance['account_balance']['days_ahead'];
33
+ $helper = Mage::helper('get4cast_salesforecast/data');
34
+ $return = array();
35
+ $return['-1'] = $helper->__('Select...');
36
+ foreach($days_available as $day){
37
+ $price = '';
38
+ if($day > $limit ){
39
+ $sum = '';
40
+ $days_until = $helper->__('(insufficient balance)');
41
+ } else {
42
+ $sum = $helper->todayPlus($day);
43
+ $days_until = $helper->__('days, until');
44
+ $price = ' ( $'.number_format($day * $price_per_day, 2).' ) ';
45
+ }
46
+
47
+ $return[$day] = $day.' '.$days_until.' '.$sum.$price;
48
+ }
49
+
50
+ return $return;
51
+ }
52
+
53
+ public function processForecastHistory($forecast_history){
54
+ $session_forecast_history = Mage::getSingleton('core/session')
55
+ ->getGet4CastForecastHistory();
56
+ if($session_forecast_history && $forecast_history){
57
+ foreach($forecast_history as $data){
58
+ if($data->action != null && $data->action != 'nop'){
59
+ try{
60
+ $_forecast = Mage::getModel('get4cast_salesforecast/forecast');
61
+ $_forecast = $_forecast->load($data->entity_id);
62
+
63
+ if($data->action == 'delete'){
64
+ $_forecast->delete();
65
+ continue;
66
+ }
67
+
68
+ $_forecast->setCreatedAt($data->created_at);
69
+ $_forecast->setUpdatedAt($data->updated_at);
70
+ $_forecast->setPeriodStart($data->period_start);
71
+ $_forecast->setPeriodEnd($data->period_end);
72
+ $_forecast->setForecastAhead($data->forecast_ahead);
73
+ $_forecast->setUrl($data->url);
74
+ $_forecast->setStatus($data->status);
75
+ $_forecast->save();
76
+
77
+ } catch (Exception $e) {
78
+ Mage::log($e->getMessage());
79
+ }
80
+ }
81
+ }
82
+ Mage::getSingleton('core/session')->setGet4CastForecastHistory($forecast_history);
83
+ }
84
+ }
85
+
86
+ protected function _beforeSave()
87
+ {
88
+ parent::_beforeSave();
89
+ }
90
+
91
+ protected function _updateTimestamps()
92
+ {
93
+ if(!$this->getUpdatedAt()){
94
+ $timestamp = now();
95
+ $this->setUpdatedAt($timestamp);
96
+ }
97
+
98
+ if ($this->isObjectNew() && !$this->getCreatedAt()) {
99
+ $this->setCreatedAt($timestamp);
100
+ }
101
+
102
+ return $this;
103
+ }
104
+
105
+ protected function _prepareUrlKey()
106
+ {
107
+ return $this;
108
+ }
109
+ }
app/code/community/Get4Cast/SalesForecast/Model/Observer.php ADDED
@@ -0,0 +1,106 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Observer {
3
+
4
+ public function get4CastAddCustomHandle(Varien_Event_Observer $observer){
5
+ $controller_action = $observer->getEvent()->getAction();
6
+ $layout = $observer->getEvent()->getLayout();
7
+ $controller_module = $controller_action->getRequest()->getControllerModule();
8
+
9
+ if ($controller_action
10
+ && $layout
11
+ && $controller_module == 'Get4Cast_SalesForecast_Adminhtml')
12
+ {
13
+ $layout->getUpdate()->addHandle('get4cast_salesforecast');
14
+ }
15
+ return $this;
16
+ }
17
+
18
+ public function get4CastPreDispatch($observer) {
19
+ try {
20
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
21
+ $_response = $api_client->register();
22
+ $_response_array = Mage::helper('core')->jsonDecode($_response);
23
+ if(!isset($_response_array['app_key'])){
24
+ Throw new Exception('App not registered');
25
+ }
26
+
27
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
28
+ $account_info = $api_client->getAccountInfo();
29
+
30
+ $this->processForecastHistory($account_info['forecast_history']);
31
+
32
+ Mage::getSingleton('core/session')
33
+ ->setGet4CastAccountInfo($account_info);
34
+
35
+ } catch (Exception $e) {
36
+ Mage::log('G4C-MAG-YXiY: '.$e->getMessage());
37
+ return false;
38
+ }
39
+ }
40
+
41
+ public function processForecastHistory($forecast_history){
42
+ try{
43
+ $report_keys = array();
44
+ if($forecast_history){
45
+ foreach($forecast_history as $data){
46
+
47
+ if($data['action'] != null && $data['action'] != 'nop'){
48
+
49
+ $forecast = Mage::getModel('get4cast_salesforecast/forecast');
50
+ $forecast = $forecast->getCollection()
51
+ ->addFieldToFilter('report_key', $data['report_key']);
52
+ $_forecast = $forecast->getFirstItem();
53
+ $report_keys[] = $data['report_key'];
54
+
55
+ if(!$_forecast->getId()){
56
+ $_forecast = Mage::getModel('get4cast_salesforecast/forecast');
57
+ }
58
+
59
+ if($data['action'] == 'delete'){
60
+ $_forecast->delete();
61
+ continue;
62
+ }
63
+
64
+ $_forecast->setReportKey($data['report_key']);
65
+
66
+ $date = Mage::getModel('core/date')->gmtDate($data['created_at']);
67
+ $_forecast->setCreatedAt($date);
68
+
69
+ $date = Mage::getModel('core/date')->gmtDate($data['updated_at']);
70
+ $_forecast->setUpdatedAt($date);
71
+
72
+ $_forecast->setStoreGroupId($data['store_group_id']);
73
+ $_forecast->setStoreGroupName($data['store_group_name']);
74
+ $_forecast->setEmail($data['email']);
75
+ $_forecast->setPeriodStart($data['period_start']);
76
+ $_forecast->setPeriodEnd($data['period_end']);
77
+ $_forecast->setForecastDateStart($data['forecast_date_start']);
78
+ $_forecast->setForecastDateEnd($data['forecast_date_end']);
79
+ $_forecast->setPrice($data['price']);
80
+ $_forecast->setPaymentStatus($data['payment_status']);
81
+
82
+ $_forecast->setReportKey($data['report_key']);
83
+ $_forecast->setUrl($data['url']);
84
+ $_forecast->setStatus($data['status']);
85
+
86
+ $_forecast->save();
87
+ }
88
+ }
89
+ }
90
+
91
+ $forecast = Mage::getModel('get4cast_salesforecast/forecast');
92
+ $forecasts = $forecast->getCollection();
93
+ if($forecasts){
94
+ foreach($forecasts as $forecast){
95
+ if(array_search($forecast->getReportKey(), $report_keys) === false){
96
+ $forecast->delete();
97
+ }
98
+ }
99
+ }
100
+
101
+ } catch (Exception $e) {
102
+ Mage::log('G4C-MAG-b2uu: '.$e->getMessage());
103
+ return false;
104
+ }
105
+ }
106
+ }
app/code/community/Get4Cast/SalesForecast/Model/Resource/Config.php ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Resource_Config
3
+ extends Mage_Core_Model_Resource_Db_Abstract
4
+ {
5
+ protected function _construct()
6
+ {
7
+ $this->_init('get4cast_salesforecast/config', 'entity_id');
8
+ }
9
+ }
app/code/community/Get4Cast/SalesForecast/Model/Resource/Config/Collection.php ADDED
@@ -0,0 +1,14 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Resource_Config_Collection
3
+ extends Mage_Core_Model_Resource_Db_Collection_Abstract
4
+ {
5
+ protected function _construct()
6
+ {
7
+ parent::_construct();
8
+
9
+ $this->_init(
10
+ 'get4cast_salesforecast/config',
11
+ 'get4cast_salesforecast/config'
12
+ );
13
+ }
14
+ }
app/code/community/Get4Cast/SalesForecast/Model/Resource/Forecast.php ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Resource_Forecast
3
+ extends Mage_Core_Model_Resource_Db_Abstract
4
+ {
5
+ protected function _construct()
6
+ {
7
+ $this->_init('get4cast_salesforecast/forecast', 'entity_id');
8
+ }
9
+ }
app/code/community/Get4Cast/SalesForecast/Model/Resource/Forecast/Collection.php ADDED
@@ -0,0 +1,14 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Model_Resource_Forecast_Collection
3
+ extends Mage_Core_Model_Resource_Db_Collection_Abstract
4
+ {
5
+ protected function _construct()
6
+ {
7
+ parent::_construct();
8
+
9
+ $this->_init(
10
+ 'get4cast_salesforecast/forecast',
11
+ 'get4cast_salesforecast/forecast'
12
+ );
13
+ }
14
+ }
app/code/community/Get4Cast/SalesForecast/controllers/Adminhtml/ForecastController.php ADDED
@@ -0,0 +1,173 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Get4Cast_SalesForecast_Adminhtml_ForecastController
3
+ extends Mage_Adminhtml_Controller_Action
4
+ {
5
+ public function indexAction(){
6
+ $forecastBlock = $this->getLayout()
7
+ ->createBlock('get4cast_salesforecast_adminhtml/forecast');
8
+
9
+ $this->loadLayout()
10
+ ->_addContent($forecastBlock)
11
+ ->renderLayout();
12
+ }
13
+
14
+ public function editAction(){
15
+ try{
16
+
17
+ $forecast = Mage::getModel('get4cast_salesforecast/forecast');
18
+ if ($forecast_id = $this->getRequest()->getParam('id', false)) {
19
+ $forecast->load($forecast_id);
20
+
21
+ if ($forecast->getId() < 1) {
22
+ $this->_getSession()->addError(
23
+ $this->__('This forecast no longer exists.')
24
+ );
25
+ return $this->_redirect(
26
+ 'get4cast_salesforecast_admin/forecast/index'
27
+ );
28
+ }
29
+ }
30
+
31
+ Mage::register('current_forecast', $forecast);
32
+
33
+ $forecastEditBlock = $this->getLayout()->createBlock(
34
+ 'get4cast_salesforecast_adminhtml/forecast_edit'
35
+ );
36
+
37
+ $this->loadLayout()
38
+ ->_addContent($forecastEditBlock)
39
+ ->renderLayout();
40
+ } catch (Exception $e) {
41
+ Mage::log('G4C-MAG-4yrx: '.$e->getMessage());
42
+ return false;
43
+ }
44
+ }
45
+
46
+ public function requestValidateAction(){
47
+ try{
48
+ $get_data = $this->getRequest()->getParam('data');
49
+ $get_data = Mage::helper('core')->jsonDecode($get_data);
50
+ if($get_data){
51
+ $errors = array();
52
+ $helper = Mage::helper('get4cast_salesforecast/data');
53
+ $forecast = Mage::getModel('get4cast_salesforecast/forecast');
54
+
55
+ /*
56
+ * START: basic validation - required fields
57
+ * */
58
+ if(!$get_data['store_group_id']){
59
+ $errors[] = $this->__('\'Store\' is required');
60
+ }
61
+
62
+ if(!$get_data['email']){
63
+ $errors[] = $this->__('\'Your email\' is required');
64
+ }
65
+
66
+ if(!$get_data['historical_date_start']){
67
+ $errors[] = $this->__('\'Analyse from day\' is required');
68
+ }
69
+
70
+ if(!$get_data['historical_date_end']){
71
+ $errors[] = $this->__('\'Analyse to day\' is required');
72
+ }
73
+
74
+ if(!$get_data['forecast_date_end']){
75
+ $errors[] = $this->__('\'Forecast until day\' is required');
76
+ }
77
+ /*
78
+ * END: basic validation - required fields
79
+ * */
80
+
81
+ // Check if 'Analyse from day' is before 'Analyse to day'
82
+ $diff_days = $helper->diffDates($get_data['historical_date_start'], $get_data['historical_date_end']);
83
+ if($diff_days < 1){
84
+ $errors[] = $this->__('\'Analyse to day\' must be after \'Analyse from day\'');
85
+ }
86
+
87
+ // Check if 'Forecast until day' is a future date
88
+ if($diff_days < 1){
89
+ $errors[] = $this->__('\'Forecast until day\' must be in the future');
90
+ }
91
+
92
+ $forecast = Mage::getModel('get4cast_salesforecast/forecast');
93
+
94
+ $return['type'] = count($errors) ? 0 : 1;
95
+ $return['data'] = $errors;
96
+ echo Mage::helper('core')->jsonEncode($return);
97
+ return true;
98
+ }
99
+ echo Mage::helper('core')->jsonEncode($api_reponse);
100
+ return true;
101
+ } catch (Exception $e) {
102
+ Mage::log('G4C-MAG-eBGA: '.$e->getMessage());
103
+ return false;
104
+ }
105
+ }
106
+
107
+ public function getForecastPriceAction(){
108
+ try{
109
+ $get_data = $this->getRequest()->getParam('data');
110
+ $get_data = Mage::helper('core')->jsonDecode($get_data);
111
+ $collect = Mage::getModel('get4cast_salesforecast/collect');
112
+ $return_data = $collect->getForecastPrice($get_data);
113
+ echo $return_data;
114
+ return true;
115
+ } catch (Exception $e) {
116
+ Mage::log('G4C-MAG-kNRW: '.$e->getMessage());
117
+ return false;
118
+ }
119
+ }
120
+
121
+ public function requestCollectAction(){
122
+ try{
123
+
124
+ $get_data = $this->getRequest()->getParam('data');
125
+ $get_data = Mage::helper('core')->jsonDecode($get_data);
126
+ $get_data['limit'] = $this->getRequest()->getParam('limit');
127
+ $get_data['page'] = $this->getRequest()->getParam('page');
128
+ $get_data['data_size'] = $this->getRequest()->getParam('data_size');
129
+ $get_data['pages'] = $this->getRequest()->getParam('pages');
130
+ $collect = Mage::getModel('get4cast_salesforecast/collect');
131
+ $return_data = $collect->requestCollect($get_data);
132
+ echo $return_data;
133
+ return true;
134
+
135
+ } catch (Exception $e) {
136
+ Mage::log('G4C-MAG-P3vP: '.$e->getMessage());
137
+ return false;
138
+ }
139
+ }
140
+
141
+ public function requestSendAction(){
142
+ $get_data = $this->getRequest()->getParam('data');
143
+ $get_data = Mage::helper('core')->jsonDecode($get_data);
144
+ $api_reponse = '';
145
+ if($get_data){
146
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
147
+ $api_reponse = $api_client->getForecastPrice($get_data);
148
+ }
149
+ echo Mage::helper('core')->jsonEncode($api_reponse);
150
+ return true;
151
+ }
152
+
153
+ protected function _isAllowed()
154
+ {
155
+ $action_name = $this->getRequest()->getActionName();
156
+ $admin_session = Mage::getSingleton('admin/session');
157
+ $is_allowed = false;
158
+ switch ($action_name) {
159
+ case 'index':
160
+ $is_allowed = $admin_session->isAllowed('admin/report/get4cast_salesforecast/sales_forecast/forecast_history');
161
+ break;
162
+ case 'edit':
163
+ case 'requestValidate':
164
+ case 'getForecastPrice':
165
+ case 'requestCollect':
166
+ case 'requestSend':
167
+ $is_allowed = $admin_session->isAllowed('admin/report/get4cast_salesforecast/sales_forecast/new_forecast');
168
+ break;
169
+ }
170
+
171
+ return $is_allowed;
172
+ }
173
+ }
app/code/community/Get4Cast/SalesForecast/data/get4cast_salesforecast_setup/data-install-1.0.0.php ADDED
@@ -0,0 +1,4 @@
 
 
 
 
1
+ <?php
2
+ //Register app
3
+ $api_client = Mage::getModel('get4cast_salesforecast/apiclient');
4
+ $response = $api_client->register();
app/code/community/Get4Cast/SalesForecast/etc/adminhtml.xml ADDED
@@ -0,0 +1,93 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <config>
3
+ <menu>
4
+ <report>
5
+ <children>
6
+ <get4cast_salesforecast translate="title"
7
+ module="get4cast_salesforecast">
8
+ <title>Get4Cast</title>
9
+ <sort_order>0</sort_order>
10
+ <depends>
11
+ <module>Get4Cast_SalesForecast</module>
12
+ </depends>
13
+ <children>
14
+ <sales_forecast translate="title"
15
+ module="get4cast_salesforecast">
16
+ <title>Sales Forecast</title>
17
+ <sort_order>1</sort_order>
18
+ <action>get4cast_salesforecast_admin/forecast</action>
19
+ <children>
20
+ <forecast_history translate="title"
21
+ module="get4cast_salesforecast">
22
+ <title>Forecast history</title>
23
+ <sort_order>1</sort_order>
24
+ <action>get4cast_salesforecast_admin/forecast</action>
25
+ </forecast_history>
26
+ </children>
27
+ <children>
28
+ <new_forecast translate="title"
29
+ module="get4cast_salesforecast">
30
+ <title>New forecast</title>
31
+ <sort_order>2</sort_order>
32
+ <action>get4cast_salesforecast_admin/forecast/edit</action>
33
+ </new_forecast>
34
+ </children>
35
+ </sales_forecast>
36
+ </children>
37
+ </get4cast_salesforecast>
38
+ </children>
39
+ </report>
40
+ </menu>
41
+
42
+ <acl>
43
+ <resources>
44
+ <admin>
45
+ <children>
46
+ <report>
47
+ <children>
48
+ <get4cast_salesforecast translate="title"
49
+ module="get4cast_salesforecast">
50
+ <title>Get4Cast</title>
51
+ <sort_order>0</sort_order>
52
+ <depends>
53
+ <module>Get4Cast_SalesForecast</module>
54
+ </depends>
55
+ <children>
56
+ <sales_forecast translate="title"
57
+ module="get4cast_salesforecast">
58
+ <title>Sales Forecast</title>
59
+ <sort_order>1</sort_order>
60
+
61
+ <children>
62
+ <forecast_history translate="title"
63
+ module="get4cast_salesforecast">
64
+ <title>Forecast history</title>
65
+ <sort_order>1</sort_order>
66
+
67
+ </forecast_history>
68
+ </children>
69
+ <children>
70
+ <new_forecast translate="title"
71
+ module="get4cast_salesforecast">
72
+ <title>New forecast</title>
73
+ <sort_order>2</sort_order>
74
+
75
+ </new_forecast>
76
+ </children>
77
+ </sales_forecast>
78
+ </children>
79
+ </get4cast_salesforecast>
80
+ </children>
81
+ </report>
82
+ </children>
83
+ </admin>
84
+ </resources>
85
+ </acl>
86
+ <layout>
87
+ <updates>
88
+ <salesforecastadmin>
89
+ <file>get4cast_salesforecast_admin.xml</file>
90
+ </salesforecastadmin>
91
+ </updates>
92
+ </layout>
93
+ </config>
app/code/community/Get4Cast/SalesForecast/etc/config.xml ADDED
@@ -0,0 +1,113 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <config>
3
+ <modules>
4
+ <Get4Cast_SalesForecast>
5
+ <version>1.0.0</version>
6
+ </Get4Cast_SalesForecast>
7
+ </modules>
8
+ <default>
9
+ <get4cast>
10
+ <default>
11
+ <api_url>https://api-sfm.get4cast.com/api</api_url>
12
+ <enable_request_report>true</enable_request_report>
13
+ </default>
14
+ </get4cast>
15
+ </default>
16
+ <adminhtml>
17
+ <layout>
18
+ <updates>
19
+ <get4cast_salesforecast>
20
+ <file>get4cast_salesforecast.xml</file>
21
+ </get4cast_salesforecast>
22
+ </updates>
23
+ </layout>
24
+ <translate>
25
+ <modules>
26
+ <get4cast_salesforecast>
27
+ <files>
28
+ <default>Get4Cast_SalesForecast.csv</default>
29
+ </files>
30
+ </get4cast_salesforecast>
31
+ </modules>
32
+ </translate>
33
+ </adminhtml>
34
+ <global>
35
+ <blocks>
36
+ <get4cast_salesforecast_adminhtml>
37
+ <class>Get4Cast_SalesForecast_Block_Adminhtml</class>
38
+ </get4cast_salesforecast_adminhtml>
39
+ </blocks>
40
+ <helpers>
41
+ <get4cast_salesforecast>
42
+ <class>Get4Cast_SalesForecast_Helper</class>
43
+ </get4cast_salesforecast>
44
+ </helpers>
45
+ <models>
46
+ <get4cast_salesforecast>
47
+ <class>Get4Cast_SalesForecast_Model</class>
48
+ <resourceModel>get4cast_salesforecast_resource</resourceModel>
49
+ </get4cast_salesforecast>
50
+ <get4cast_salesforecast_resource>
51
+ <class>Get4Cast_SalesForecast_Model_Resource</class>
52
+ <entities>
53
+ <config>
54
+ <table>get4cast_salesforecast_config</table>
55
+ </config>
56
+ <forecast>
57
+ <table>get4cast_salesforecast_forecast</table>
58
+ </forecast>
59
+ </entities>
60
+ </get4cast_salesforecast_resource>
61
+ </models>
62
+ <resources>
63
+ <get4cast_salesforecast_setup>
64
+ <setup>
65
+ <module>Get4Cast_SalesForecast</module>
66
+ <class>Mage_Core_Model_Resource_Setup</class>
67
+ </setup>
68
+ <connection>
69
+ <use>core_setup</use>
70
+ </connection>
71
+ </get4cast_salesforecast_setup>
72
+ </resources>
73
+ <events>
74
+ <controller_action_layout_load_before>
75
+ <observers>
76
+ <get4cast_salesforecast_model_observer>
77
+ <class>Get4Cast_SalesForecast_Model_Observer</class>
78
+ <method>get4CastAddCustomHandle</method>
79
+ </get4cast_salesforecast_model_observer>
80
+ </observers>
81
+ </controller_action_layout_load_before>
82
+ <controller_action_predispatch_get4cast_salesforecast_admin_forecast_index>
83
+ <observers>
84
+ <get4cast_salesforecast_model_observer>
85
+ <type>singleton</type>
86
+ <class>Get4Cast_SalesForecast_Model_Observer</class>
87
+ <method>get4CastPreDispatch</method>
88
+ </get4cast_salesforecast_model_observer>
89
+ </observers>
90
+ </controller_action_predispatch_get4cast_salesforecast_admin_forecast_index>
91
+ <controller_action_predispatch_get4cast_salesforecast_admin_forecast_edit>
92
+ <observers>
93
+ <get4cast_salesforecast_model_observer>
94
+ <type>singleton</type>
95
+ <class>Get4Cast_SalesForecast_Model_Observer</class>
96
+ <method>get4CastPreDispatch</method>
97
+ </get4cast_salesforecast_model_observer>
98
+ </observers>
99
+ </controller_action_predispatch_get4cast_salesforecast_admin_forecast_edit>
100
+ </events>
101
+ </global>
102
+ <admin>
103
+ <routers>
104
+ <get4cast_salesforecast_admin>
105
+ <use>admin</use>
106
+ <args>
107
+ <module>Get4Cast_SalesForecast_Adminhtml</module>
108
+ <frontName>get4cast-salesforecast-admin</frontName>
109
+ </args>
110
+ </get4cast_salesforecast_admin>
111
+ </routers>
112
+ </admin>
113
+ </config>
app/code/community/Get4Cast/SalesForecast/sql/get4cast_salesforecast_setup/install-1.0.0.php ADDED
@@ -0,0 +1,195 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ $this->startSetup();
4
+
5
+ ########################################################################
6
+
7
+ $table = new Varien_Db_Ddl_Table();
8
+
9
+ $table->setName($this->getTable('get4cast_salesforecast/config'));
10
+ $table->setOption('type', 'InnoDB');
11
+ $table->setOption('charset', 'utf8');
12
+
13
+ $table->addColumn(
14
+ 'entity_id',
15
+ Varien_Db_Ddl_Table::TYPE_INTEGER,
16
+ 10,
17
+ array(
18
+ 'auto_increment' => true,
19
+ 'unsigned' => true,
20
+ 'nullable'=> false,
21
+ 'primary' => true
22
+ )
23
+ );
24
+ $table->addColumn(
25
+ 'created_at',
26
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
27
+ null,
28
+ array(
29
+ 'nullable' => false,
30
+ )
31
+ );
32
+ $table->addColumn(
33
+ 'updated_at',
34
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
35
+ null,
36
+ array(
37
+ 'nullable' => true,
38
+ )
39
+ );
40
+ $table->addColumn(
41
+ 'config',
42
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
43
+ 255,
44
+ array(
45
+ 'nullable' => false,
46
+ )
47
+ );
48
+ $table->addColumn(
49
+ 'value',
50
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
51
+ 255,
52
+ array(
53
+ 'nullable' => false,
54
+ )
55
+ );
56
+
57
+ $this->getConnection()->createTable($table);
58
+
59
+ ########################################################################
60
+
61
+ $table = new Varien_Db_Ddl_Table();
62
+
63
+ $table->setName($this->getTable('get4cast_salesforecast/forecast'));
64
+ $table->setOption('type', 'InnoDB');
65
+ $table->setOption('charset', 'utf8');
66
+
67
+ $table->addColumn(
68
+ 'entity_id',
69
+ Varien_Db_Ddl_Table::TYPE_INTEGER,
70
+ 10,
71
+ array(
72
+ 'auto_increment' => true,
73
+ 'unsigned' => true,
74
+ 'nullable'=> false,
75
+ 'primary' => true
76
+ )
77
+ );
78
+ $table->addColumn(
79
+ 'report_key',
80
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
81
+ 255,
82
+ array(
83
+ 'nullable' => false,
84
+ )
85
+ );
86
+ $table->addColumn(
87
+ 'created_at',
88
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
89
+ null,
90
+ array(
91
+ 'nullable' => false,
92
+ )
93
+ );
94
+ $table->addColumn(
95
+ 'updated_at',
96
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
97
+ null,
98
+ array(
99
+ 'nullable' => true,
100
+ )
101
+ );
102
+ $table->addColumn(
103
+ 'store_group_id',
104
+ Varien_Db_Ddl_Table::TYPE_INTEGER,
105
+ 10,
106
+ array(
107
+ 'nullable' => false,
108
+ )
109
+ );
110
+ $table->addColumn(
111
+ 'store_group_name',
112
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
113
+ 255,
114
+ array(
115
+ 'nullable' => false,
116
+ )
117
+ );
118
+ $table->addColumn(
119
+ 'email',
120
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
121
+ 255,
122
+ array(
123
+ 'nullable' => false,
124
+ )
125
+ );
126
+ $table->addColumn(
127
+ 'period_start',
128
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
129
+ null,
130
+ array(
131
+ 'nullable' => false,
132
+ )
133
+ );
134
+ $table->addColumn(
135
+ 'period_end',
136
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
137
+ null,
138
+ array(
139
+ 'nullable' => false,
140
+ )
141
+ );
142
+ $table->addColumn(
143
+ 'forecast_date_start',
144
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
145
+ null,
146
+ array(
147
+ 'nullable' => false,
148
+ )
149
+ );
150
+ $table->addColumn(
151
+ 'forecast_date_end',
152
+ Varien_Db_Ddl_Table::TYPE_DATETIME,
153
+ null,
154
+ array(
155
+ 'nullable' => false,
156
+ )
157
+ );
158
+ $table->addColumn(
159
+ 'price',
160
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
161
+ 255,
162
+ array(
163
+ 'nullable' => false,
164
+ )
165
+ );
166
+ $table->addColumn(
167
+ 'payment_status',
168
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
169
+ 255,
170
+ array(
171
+ 'nullable' => false,
172
+ )
173
+ );
174
+ $table->addColumn(
175
+ 'url',
176
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
177
+ 255,
178
+ array(
179
+ 'nullable' => false,
180
+ )
181
+ );
182
+ $table->addColumn(
183
+ 'status',
184
+ Varien_Db_Ddl_Table::TYPE_VARCHAR,
185
+ 255,
186
+ array(
187
+ 'nullable' => false,
188
+ )
189
+ );
190
+
191
+ $this->getConnection()->createTable($table);
192
+
193
+ ########################################################################
194
+
195
+ $this->endSetup();
app/design/adminhtml/default/default/layout/get4cast_salesforecast.xml ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <layout>
3
+ <get4cast_salesforecast>
4
+ <reference name="head">
5
+ <action method="addJs"><script>get4cast/salesforecast/jquery-1.12.0.min.js</script></action>
6
+ <action method="addJs"><script>get4cast/salesforecast/forecast.js</script></action>
7
+ </reference>
8
+ </get4cast_salesforecast>
9
+ </layout>
app/etc/modules/Get4Cast_SalesForecast.xml ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <config>
3
+ <modules>
4
+ <Get4Cast_SalesForecast>
5
+ <active>true</active>
6
+ <codePool>community</codePool>
7
+ </Get4Cast_SalesForecast>
8
+ </modules>
9
+ </config>
app/locale/pt_BR/Get4Cast_SalesForecast.csv ADDED
@@ -0,0 +1,114 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ "Reports", "Relatórios"
2
+ "Forecast history", "Histórico de Previsões"
3
+ "New forecast", "Nova previsão"
4
+ "Page", "Página"
5
+ "of %s pages","de %s páginas"
6
+ "View", "Visualizar"
7
+ "per page","por página"
8
+ "Total %d records found","Total %d registros encontrados"
9
+ "Reset Filter","Limpar filtro"
10
+ "Search", "Buscar"
11
+ "New", "Novo"
12
+ "Edit", "Editar"
13
+ "forecast", "previsão"
14
+ "Add New", "Nova previsão"
15
+ "Back", "Voltar"
16
+ "Sales Forecast", "Previsão de Vendas"
17
+ "ID", "ID"
18
+ "Created at", "Criado em"
19
+ "Updated at", "Atualizado em"
20
+ "Store", "Loja"
21
+ "Period start", "Início período analisado"
22
+ "Period end", "Fim período analisado"
23
+ "Forecast day", "Previsão até dia"
24
+ "Forecast day", "Previsão até dia"
25
+ "Report price", "Preço do relatório"
26
+ "Payment status", "Pagamento"
27
+ "Open report", "Abrir relatório"
28
+ "Report status", "Situação do relatório"
29
+ "Forecast details", "Detalhes da previsão"
30
+ "Account info", "Informações da conta"
31
+ "Forecast for the selected store", "Previsão para a loja selecionada"
32
+ "Your email", "Seu e-mail"
33
+ "You will be notified by email when the report is ready", "Você será notificado por email quando o relatório estiver pronto"
34
+ "Analyse from day", "Início do período analisado"
35
+ "Start period of historical data to be analysed", "Início do período de histórico de dados para ser analisado"
36
+ "Analyse to day", "Fim do período analisado"
37
+ "End period of historical data to be analysed", "Fim do período de histórico de dados para ser analisado"
38
+ "Forecast until day", "Previsão até o dia"
39
+ "Forecast from today until the selected day", "Previsão de hoje até o dia selecionado"
40
+ "Check forecast price", "Solicitar orçamento do relatório"
41
+ "Step 1: Check forecast price", "Etapa 1: Solicitar orçamento do relatório"
42
+ "Request forecast", "Solicitar relatório de previsão"
43
+ "Step 2: Request forecast", "Etapa 2: Solicitar relatório de previsão"
44
+ "Click on Step 1: Check forecast price before", "Primeiramente, clique em Etapa 1: Solicitar orçamento do relatório"
45
+ "Loading...", "Carregando..."
46
+ "Unable to get forecast price", "Não foi possível obter o preço do relatório"
47
+ "The payment link will be available", "O link para o pagamento estará disponível"
48
+ "when you open your report and in your email.", "quando você abrir o seu relatório e, também, no seu e-mail."
49
+ "Try again or", "Tente novamente ou"
50
+ "contact us", "entre em contato"
51
+ "Could not retrieve account information", "Não foi possível obter as informações da conta"
52
+ "Your email is required", "Informe o seu e-mail"
53
+ "Your email is invalid", "E-mail inválido"
54
+ "Analyse from day is required", "Informe o Início do período analisado"
55
+ "Analyse to day is required", "Informe o Fim do período analisado"
56
+ "Forecast until day is required", "Informe Previsão até o dia"
57
+ "Invalid date format", "Formato da data inválido"
58
+ "Invalid date format...", "Formato da data inválido..."
59
+ "Analyse from day invalid date", "Data inválida para Início do período analisado"
60
+ "Analyse to day invalid date", "Data inválida para Fim do período analisado"
61
+ "Analyse from day must be before Analyse to day", "Início do período analisado deve ser anterior ao Fim do período analisado"
62
+ "Analyse to day must be in the past", "A data do Fim do período analisado deve estar no passado"
63
+ "Forecast until day must be in the future", "A data de Previsão até o dia deve estar no futuro"
64
+ "To have a minimum acceptable precision you must:", "Para ter uma precisão mínima aceitável, você deve"
65
+ "Select at least", "Selecionar, no mínimo,"
66
+ "days of historical data to be analysed.", "dias do histórico de dados para ser analisado."
67
+ "See", "Veja"
68
+ "how to increase your forecast accuracy", "como aumentar a precisão da sua previsão"
69
+ "There is a limit of", "Há um limite de"
70
+ "days of historical data that can be analysed.", "dias do histórico de dados que pode ser analisado"
71
+ "Please, select a smaller period.", "Por favor, selecione um período menor."
72
+ "If you need to analyze more data", "Se você precisa analisar mais dados"
73
+ "for a custom service.", "para uma análise personalizada."
74
+ "You are willing to analyse", "Você está analisando"
75
+ "days", "dias"
76
+ "of historical data and to forecast", "do histórico de dados para prever"
77
+ "To have a minimum acceptable precision you can:", "Para ter uma precisão mínima aceitável, você pode:"
78
+ "Increase the period of historical data to be analysed", "Aumentar o período do histórico de dados para ser analisado"
79
+ "Decrease the number of forecast days", "Reduzir o número de dias da previsão"
80
+ "Request error", "Erro durante requisição"
81
+ "Validating data...", "Validando informações..."
82
+ "Validation error", "Erro de validação"
83
+ "Collecting, splitting and sending data...", "Coletando, particionando e enviando os dados..."
84
+ "Unable to request", "Não foi possível realizar a requisição"
85
+ "Your request was successfully received.", "Sua requisição foi enviada com sucesso"
86
+ "We will process your data and notify you when everything is done.", "Nós vamos processar os seus dados e, assim que estiver tudo pronto, enviaremos uma notificação por e-mail."
87
+ "You can check your report status in Forecast history page.", "Você pode acompanhar a situação do seu relatório na página Histórico de Previsões."
88
+ "Error", "Erro"
89
+ "Info", "Info"
90
+ "Ok!", "Ok!"
91
+ "You have a credit of:", "Você tem um crédito de:"
92
+ "Forecast original price:", "Preço original do relatório:"
93
+ "You will still have<br>a credit of", "Você ainda terá<br>um crédito de"
94
+ "Forecast price", "Preço do relatório"
95
+ "Forecast final price", "Preço final do relatório"
96
+ "Ready to go!", "Solicite o seu relatório!"
97
+ "Click here to open your report", "Clique aqui para abrir o seu relatório"
98
+ "Request result comparison", "Solicitar comparação com resultado"
99
+ "Step 3: Request result", "Etapa 3: Solicitar comparação com resultado"
100
+ "receiving data", "recebendo dados"
101
+ "requested", "solicitado"
102
+ "extracting raw data", "extraindo dados"
103
+ "forecast queue", "fila de previsão"
104
+ "forecasting", "prevendo dados"
105
+ "ready", "pronto"
106
+ "request result", "comparar resultado"
107
+ "result requested", "comparação solicitada"
108
+ "complete", "completo"
109
+ "waiting", "aguardando"
110
+ "fail", "falhou"
111
+ "pending", "pendente"
112
+ "succeeded", "finalizado"
113
+ "forecast report", "relatório de previsão"
114
+ "Request forecast is disabled in this platform.", "A solicitação de relatórios está desabilitada nesta plataforma."
js/get4cast/salesforecast/forecast.js ADDED
@@ -0,0 +1,707 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ jQuery.noConflict();
2
+ var first_validation = true;
3
+ var request_sent = false;
4
+ var count_dates_error = 0;
5
+
6
+ var global_account_info_json = null
7
+ var global_ajax_url = null
8
+ var global_request_urls = null;
9
+ var global_split_pages = null;
10
+ var global_split_per_page = null;
11
+ var global_request_collect_error_msg = 0;
12
+ var global_tmp_final_balance = null;
13
+
14
+ function callCheckReportDetails(account_info_json, ajax_url){
15
+ global_account_info_json = account_info_json;
16
+
17
+ account_info_json = JSON.parse(account_info_json);
18
+ if(global_tmp_final_balance != null){
19
+ account_info_json.account_balance = global_tmp_final_balance;
20
+ }
21
+
22
+ global_ajax_url = ajax_url;
23
+
24
+ disableRequestButton();
25
+ checked = checkReportDetails(account_info_json);
26
+ if(checked.type == 'ok'){
27
+ if(request_sent){
28
+ return true;
29
+ }
30
+ request_sent = true;
31
+ jQuery.ajax(ajax_url, {
32
+ type: 'GET',
33
+ data: {'data': formToJson()},
34
+ beforeSend: function() {
35
+ jQuery('#loading-mask').show();
36
+ updateRequestForecastInfo('info', Translator.translate('Loading...'), 'request_forecast_price_info');
37
+ },
38
+ success: function(data) {
39
+ jQuery('#loading-mask').hide();
40
+ request_sent = false;
41
+ data = JSON.parse(data);
42
+ if(typeof(data) != 'object' || !data.original_price || data.error){
43
+ msg = 'Unable to get forecast price';
44
+ if(data.error){
45
+ msg = data.error;
46
+ }
47
+ msg = Translator.translate(msg);
48
+ updateRequestForecastInfo('error', msg, 'request_forecast_price_info');
49
+ return false;
50
+ }
51
+ if(data.original_price){
52
+
53
+ payment_ling_msg_info = Translator.translate('The payment link will be available');
54
+ payment_ling_msg_info += '<br>';
55
+ payment_ling_msg_info += Translator.translate('when you open your report and in your email.');
56
+
57
+ msg = '<div style=\'color:black;font-weight:normal;font-size:12px\'>';
58
+ if(parseFloat(account_info_json.account_balance) > 0){
59
+ msg += '<table>';
60
+ msg += '<tr>';
61
+ msg += '<td>'+Translator.translate('You have a credit of:')+'</td>';
62
+ msg += '<td>+ $'+account_info_json.account_balance+'</td>';
63
+ msg += '</tr>';
64
+ msg += '<tr>';
65
+ msg += '<td style=\'border-bottom: solid 1px silver\'>'+Translator.translate('Forecast original price:')+'</td>';
66
+ msg += '<td style=\'border-bottom: solid 1px silver\'>- $'+data.original_price+'</td>';
67
+ msg += '</tr>';
68
+ msg += '<tr>';
69
+ msg += '<td style=\'font-weight:bold\'>'+Translator.translate('Forecast final price')+':</td>';
70
+ if(data.final_balance > 0 || data.final_price <= 0){
71
+ msg += '<td>';
72
+ msg += '<span style=\'font-weight:bold\'>FREE</span>';
73
+ if(data.final_balance > 0){
74
+ msg += '<br>';
75
+ msg += Translator.translate('You will still have<br>a credit of')+' $'+data.final_balance;
76
+ }
77
+ msg += '</td>';
78
+ } else {
79
+ msg += '<td>';
80
+ msg += '<span style=\'font-weight:bold\'>$'+data.final_price+'</span>';
81
+ msg += '<br>';
82
+ msg += payment_ling_msg_info;
83
+ msg += '</td>';
84
+ }
85
+ msg += '</tr>';
86
+ msg += '</table>';
87
+ } else {
88
+ msg += '<span style=\'font-weight:bold\'>'
89
+ msg += Translator.translate('Forecast price')+': $'+data.original_price;
90
+ msg += '</span>'
91
+ msg += '<br>';
92
+ msg += payment_ling_msg_info;
93
+ }
94
+
95
+ msg += '</div>';
96
+
97
+ jQuery('#h_forecast_quote_key').val(data.forecast_quote_key);
98
+
99
+ global_split_pages = data.split_pages;
100
+ global_split_per_page = data.split_per_page;
101
+ global_data_size = data.data_size;
102
+
103
+ updateRequestForecastInfo('ok', msg, 'request_forecast_price_info');
104
+
105
+ return true;
106
+ }
107
+ },
108
+ error: function() {
109
+ jQuery('#loading-mask').hide();
110
+ request_sent = false;
111
+ msg = Translator.translate('Unable to get forecast price');
112
+ msg += ' ';
113
+ msg += Translator.translate('Try again or');
114
+ msg += ' ';
115
+ msg += '<a target=\'_blank\' href=\''+account_info_json.core_config.url_contact_us_error+'\'>';
116
+ msg += Translator.translate('contact us');
117
+ msg += '</a>.';
118
+ updateRequestForecastInfo('error', msg, 'request_forecast_price_info');
119
+ return false;
120
+ }
121
+ });
122
+ } else {
123
+ updateRequestForecastInfo(checked.type, checked.msg, 'request_forecast_price_info')
124
+ }
125
+ }
126
+
127
+ function checkReportDetails(account_info_json){
128
+
129
+ first_validation = false;
130
+ historical_date_start = jQuery('#historical_date_start').val();
131
+ historical_date_end = jQuery('#historical_date_end').val();
132
+ forecast_date_end = jQuery('#forecast_date_end').val();
133
+ email = jQuery('#email').val();
134
+
135
+ checked = {}
136
+ checked.type = '';
137
+ checked.msg = '';
138
+
139
+ if( typeof(account_info_json.account_balance) == 'undefined'
140
+ || typeof(account_info_json.core_config) == 'undefined'
141
+ ){
142
+ checked.type = 'error';
143
+ checked.msg = Translator.translate('Could not retrieve account information');
144
+ return checked;
145
+ }
146
+
147
+ if(!email){
148
+ jQuery('#email').focus();
149
+ checked.type = 'error';
150
+ checked.msg = Translator.translate('Your email is required');
151
+ return checked;
152
+ }
153
+
154
+ pattern = /^([a-z\d!#$%&'*+\-\/=?^_`{|}~\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+(\.[a-z\d!#$%&'*+\-\/=?^_`{|}~\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+)*|"((([ \t]*\r\n)?[ \t]+)?([\x01-\x08\x0b\x0c\x0e-\x1f\x7f\x21\x23-\x5b\x5d-\x7e\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]|\\[\x01-\x09\x0b\x0c\x0d-\x7f\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))*(([ \t]*\r\n)?[ \t]+)?")@(([a-z\d\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]|[a-z\d\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF][a-z\d\-._~\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]*[a-z\d\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])\.)+([a-z\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]|[a-z\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF][a-z\d\-._~\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]*[a-z\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])\.?$/i;
155
+ validate_email = pattern.test(email);
156
+ if(!validate_email){
157
+ checked.type = 'error';
158
+ checked.msg = Translator.translate('Your email is invalid');
159
+ return checked;
160
+ }
161
+
162
+ if(!historical_date_start){
163
+ jQuery('#historical_date_start').focus();
164
+ checked.type = 'error';
165
+ checked.msg = Translator.translate('Analyse from day is required');
166
+ return checked;
167
+ }
168
+
169
+ if(!historical_date_end){
170
+ jQuery('#historical_date_end').focus();
171
+ checked.type = 'error';
172
+ if(!first_validation){
173
+ checked.type = 'error';
174
+ checked.msg = Translator.translate('Analyse to day is required');
175
+ }
176
+ return checked;
177
+ }
178
+
179
+ if(!forecast_date_end){
180
+ jQuery('#forecast_date_end').focus();
181
+ checked.type = 'error';
182
+ checked.msg = Translator.translate('Forecast until day is required');
183
+ return checked;
184
+ }
185
+
186
+ date_format = jQuery('#h_date_format').val();
187
+ date_format = date_format.split('/');
188
+ if(date_format.length != 3){
189
+ checked = {}
190
+ checked.type = 'error';
191
+ checked.msg = Translator.translate('Invalid date format');
192
+ return checked;
193
+ }
194
+
195
+ index_day = null;
196
+ index_month = null;
197
+ index_year = null;
198
+ for(i=0;i<=2;i++){
199
+ if(date_format[i].toLowerCase() == 'd' || date_format[i].toLowerCase() == 'dd'){
200
+ index_day = i;
201
+ } else if(date_format[i].toLowerCase() == 'm' || date_format[i].toLowerCase() == 'mm'){
202
+ index_month = i;
203
+ } else if(date_format[i].toLowerCase() == 'y'){
204
+ index_year = i;
205
+ }
206
+ }
207
+ required_info = (index_day != null
208
+ && index_month !=null
209
+ && index_year != null);
210
+
211
+ if(!required_info){
212
+ checked = {}
213
+ checked.type = 'error';
214
+ checked.msg = Translator.translate('Invalid date format...');
215
+ return checked;
216
+ }
217
+
218
+ historical_date_start = historical_date_start.split('/');
219
+
220
+ aux_two_digit_year = parseInt(historical_date_start[index_year]);
221
+ if(aux_two_digit_year <= 99){
222
+ historical_date_start[index_year] = twoDigitYearToFour(aux_two_digit_year);
223
+ }
224
+
225
+ aux_historical_date_start = parseInt(historical_date_start[index_year])
226
+ +'/'+parseInt(historical_date_start[index_month])
227
+ +'/'+parseInt(historical_date_start[index_day])
228
+ jQuery('#h_historical_date_start').val(aux_historical_date_start);
229
+
230
+ historical_date_end = historical_date_end.split('/');
231
+
232
+ aux_two_digit_year = parseInt(historical_date_end[index_year]);
233
+ if(aux_two_digit_year <= 99){
234
+ historical_date_end[index_year] = twoDigitYearToFour(aux_two_digit_year);
235
+ }
236
+
237
+ aux_historical_date_end = parseInt(historical_date_end[index_year]);
238
+ aux_historical_date_end += '/';
239
+ aux_historical_date_end += parseInt(historical_date_end[index_month]);
240
+ aux_historical_date_end += '/';
241
+ aux_historical_date_end += parseInt(historical_date_end[index_day]);
242
+
243
+ jQuery('#h_historical_date_end').val(aux_historical_date_end);
244
+ var aux_historical_date_start = new Date(aux_historical_date_start);
245
+ var aux_historical_date_end = new Date(aux_historical_date_end);
246
+
247
+ if(parseInt(historical_date_start[index_day]) > 31
248
+ || parseInt(historical_date_start[index_month]) > 12)
249
+ {
250
+ checked = {}
251
+ checked.type = 'error';
252
+ checked.msg = Translator.translate('Analyse from day invalid date');
253
+ return checked;
254
+ }
255
+
256
+ if(parseInt(historical_date_end[index_day]) > 31
257
+ || parseInt(historical_date_end[index_month]) > 12)
258
+ {
259
+ checked = {}
260
+ checked.type = 'error';
261
+ checked.msg = Translator.translate('Analyse to day invalid date');
262
+ return checked;
263
+ }
264
+
265
+ hist_days_diff = subtractDates(aux_historical_date_start, aux_historical_date_end);
266
+ jQuery('#h_historical_diff').val(hist_days_diff);
267
+
268
+ if(hist_days_diff <= 0){
269
+ count_dates_error++;
270
+ checked.type = 'error';
271
+ checked.msg = Translator.translate('Analyse from day must be before Analyse to day');
272
+ return checked;
273
+ }
274
+
275
+ today_date = new Date();
276
+ today = [];
277
+ today[index_day] = parseInt(today_date.getDate());
278
+ today[index_month] = parseInt(today_date.getMonth()+1);
279
+ today[index_year] = parseInt(today_date.getFullYear());
280
+ aux_today = parseInt(today[index_year])
281
+ +'/'+parseInt(today[index_month])
282
+ +'/'+parseInt(today[index_day])
283
+ var aux_today = new Date(aux_today);
284
+
285
+ forecast_date_end = forecast_date_end.split('/');
286
+
287
+ aux_two_digit_year = parseInt(forecast_date_end[index_year]);
288
+ if(aux_two_digit_year <= 99){
289
+ forecast_date_end[index_year] = twoDigitYearToFour(aux_two_digit_year);
290
+ }
291
+
292
+ aux_forecast_date_end = parseInt(forecast_date_end[index_year])
293
+ +'/'+parseInt(forecast_date_end[index_month])
294
+ +'/'+parseInt(forecast_date_end[index_day])
295
+ jQuery('#h_forecast_date_end').val(aux_forecast_date_end);
296
+ var aux_forecast_date_end = new Date(aux_forecast_date_end);
297
+
298
+ days_diff = subtractDates(aux_historical_date_end, aux_today);
299
+ if(days_diff <= 0){
300
+ count_dates_error++;
301
+ checked.type = 'error';
302
+ checked.msg = Translator.translate('Analyse to day must be in the past');
303
+ return checked;
304
+ }
305
+
306
+ forecast_days_diff = subtractDates(aux_today, aux_forecast_date_end);
307
+ jQuery('#h_forecast_diff').val(forecast_days_diff);
308
+ if(forecast_days_diff <= 0){
309
+ count_dates_error++;
310
+ checked.type = 'error';
311
+ checked.msg = Translator.translate('Forecast until day must be in the future');
312
+ return checked;
313
+ }
314
+
315
+ if(account_info_json.core_config.limit_min_historic_days != 0
316
+ && hist_days_diff < account_info_json.core_config.limit_min_historic_days)
317
+ {
318
+ count_dates_error++;
319
+ checked.type = 'error';
320
+
321
+ msg = '';
322
+ msg += Translator.translate('To have a minimum acceptable precision you must:');
323
+ msg += '<br>';
324
+ msg += '- '+Translator.translate('Select at least')+' '+account_info_json.core_config.limit_min_historic_days;
325
+ msg += ' ';
326
+ msg += Translator.translate('days of historical data to be analysed.');
327
+ msg += '<br>';
328
+ msg += Translator.translate('See');
329
+ msg += ' ';
330
+ msg += '<a target=\'_blank\' href=\''+account_info_json.core_config.url_help_tips_about_precision+'\'>';
331
+ msg += Translator.translate('how to increase your forecast accuracy');
332
+ msg += '</a>.';
333
+ checked.msg = msg;
334
+
335
+ return checked;
336
+ }
337
+
338
+ if(account_info_json.core_config.limit_historic_days != 0
339
+ && hist_days_diff > account_info_json.core_config.limit_historic_days)
340
+ {
341
+ count_dates_error++;
342
+ checked.type = 'error';
343
+ msg = Translator.translate('There is a limit of');
344
+ msg += ' '+account_info_json.core_config.limit_historic_days+' ';
345
+ msg += Translator.translate('days of historical data that can be analysed.');
346
+ msg += '<br>';
347
+ msg += Translator.translate('Please, select a smaller period.');
348
+ msg += '<br>';
349
+ msg += Translator.translate('If you need to analyze more data');
350
+ msg += ' ';
351
+ msg += '<a target=\'_blank\' href=\''+account_info_json.core_config.url_contact_us_custom+'\'>';
352
+ msg += Translator.translate('contact us');
353
+ msg += '</a>';
354
+ msg += ' ';
355
+ msg += Translator.translate('for a custom service.');
356
+ checked.msg = msg;
357
+ return checked;
358
+ }
359
+
360
+
361
+ if(account_info_json.core_config.limit_future_days != 0
362
+ && forecast_days_diff > account_info_json.core_config.limit_future_days)
363
+ {
364
+ count_dates_error++;
365
+ checked.type = 'error';
366
+ msg = Translator.translate('There is a limit of');
367
+ msg += ' '+account_info_json.core_config.limit_future_days+' ';
368
+ msg += Translator.translate('days for forecast.');
369
+ msg += '<br>';
370
+ msg += Translator.translate('Please, select a smaller period.');
371
+ msg += '<br>';
372
+ msg += Translator.translate('If you need to process more data');
373
+ msg += ' ';
374
+ msg += '<a target=\'_blank\' href=\''+account_info_json.core_config.url_contact_us_custom+'\'>';
375
+ msg += Translator.translate('contact us');
376
+ msg += '</a>';
377
+ msg += ' ';
378
+ msg += Translator.translate('for a custom service.');
379
+ checked.msg = msg;
380
+ return checked;
381
+ }
382
+
383
+ percent_historic_forecast = hist_days_diff/forecast_days_diff;
384
+ percent_historic_forecast = percent_historic_forecast * 100;
385
+ if(percent_historic_forecast < account_info_json.core_config.limit_percent_historic_forecast){
386
+ count_dates_error++;
387
+ checked.type = 'error';
388
+ msg = '';
389
+ msg += Translator.translate('You are willing to analyse');
390
+ msg += ' ';
391
+ msg += '<strong>';
392
+ msg += hist_days_diff;
393
+ msg += ' ';
394
+ msg += Translator.translate('days');
395
+ msg += '</strong>';
396
+ msg += ' ';
397
+ msg += Translator.translate('of historical data and to forecast');
398
+ msg += ' ';
399
+ msg += '<strong>';
400
+ msg += forecast_days_diff;
401
+ msg += ' ';
402
+ msg += Translator.translate('days');
403
+ msg += '</strong>.';
404
+ msg += ' ';
405
+ msg += '<br>';
406
+ msg += Translator.translate('To have a minimum acceptable precision you can:');
407
+ msg += '<br>';
408
+ msg += '- '+Translator.translate('Increase the period of historical data to be analysed');
409
+ msg += '<br>';
410
+ msg += '- '+Translator.translate('Decrease the number of forecast days');
411
+ msg += '<br>';
412
+ msg += '<br>';
413
+ msg += Translator.translate('See');
414
+ msg += ' ';
415
+ msg += '<a target=\'_blank\' href=\''+account_info_json.core_config.url_help_tips_about_precision+'\'>';
416
+ msg += Translator.translate('how to increase your forecast accuracy');
417
+ msg += '</a>.';
418
+ checked.msg = msg;
419
+ return checked;
420
+ }
421
+
422
+ if(historical_date_start
423
+ && historical_date_end
424
+ && forecast_date_end)
425
+ {
426
+ checked.type = 'ok';
427
+ }
428
+
429
+ return checked;
430
+ }
431
+
432
+ function bestFit(){
433
+ count_dates_error = 0;
434
+
435
+ if(request_sent){
436
+ return true;
437
+ }
438
+ request_sent = false;
439
+
440
+ request_urls = JSON.parse(global_request_urls);
441
+ ajax_url = request_urls.request_best_fit;
442
+
443
+ jQuery.ajax(ajax_url, {
444
+ type: 'GET',
445
+ async: false,
446
+ data: {'data': formToJson()},
447
+ beforeSend: function() {
448
+ jQuery('#loading-mask').show();
449
+ },
450
+ success: function(data) {
451
+ jQuery('#loading-mask').hide();
452
+ request_sent = false;
453
+ api_result = JSON.parse(data)
454
+ global_request_collect_error_msg = 0;
455
+
456
+ if(api_result["_error"] != undefined){
457
+ global_request_collect_error_msg = api_result["_error"];
458
+ }
459
+
460
+ },
461
+ error: function(data) {
462
+ jQuery('#loading-mask').hide();
463
+ request_sent = false;
464
+ global_request_collect_error_msg = data.result;
465
+ info_id = jQuery('#h_data_type').val() == 'input' ? 'request_forecast_info' : 'request_result_info';
466
+ updateRequestForecastInfo('error', Translator.translate('Request error'), info_id);
467
+ }
468
+ });
469
+
470
+ jQuery('#historical_date_start').val(formatDate(best['historic_start']));
471
+ jQuery('#historical_date_end').val(formatDate(best['historic_end']));
472
+ jQuery('#forecast_date_end').val(formatDate(best['forecast_date_end']));
473
+ callCheckReportDetails(global_account_info_json, global_ajax_url)
474
+ }
475
+
476
+
477
+ function requestValidate(request_urls){
478
+ global_request_urls = request_urls;
479
+
480
+ if(request_sent){
481
+ return true;
482
+ }
483
+ request_sent = true;
484
+ request_urls = JSON.parse(global_request_urls);
485
+ ajax_url = request_urls.request_validate_url;
486
+ jQuery.ajax(ajax_url, {
487
+ type: 'GET',
488
+ data: {'data': formToJson()},
489
+ beforeSend: function() {
490
+ disableRequestButton();
491
+ jQuery('#loading-mask').show();
492
+ info_id = jQuery('#h_data_type').val() == 'input' ? 'request_forecast_info' : 'request_result_info';
493
+ updateRequestForecastInfo('info', Translator.translate('Validating data...'), info_id);
494
+ },
495
+ success: function(data) {
496
+ data = JSON.parse(data);
497
+ info_id = jQuery('#h_data_type').val() == 'input' ? 'request_forecast_info' : 'request_result_info';
498
+ if(data.type == 0 || !global_split_pages){
499
+ disableRequestButton();
500
+ jQuery('#loading-mask').hide();
501
+ updateRequestForecastInfo('error', Translator.translate('Validation error'), info_id);
502
+ } else {
503
+ request_sent = false;
504
+ msg = Translator.translate('Collecting, splitting and sending data...');
505
+
506
+ for(i=1;i<=global_split_pages;i++){
507
+
508
+ msg_count = ' [ '+i+' / '+global_split_pages+' ]';
509
+
510
+ updateRequestForecastInfo('info', msg+msg_count, info_id);
511
+
512
+ requestCollect(global_split_per_page, i);
513
+
514
+ if(global_request_collect_error_msg != 0){
515
+ jQuery('#loading-mask').hide();
516
+ updateRequestForecastInfo('error', Translator.translate('Unable to request'), info_id);
517
+ return false;
518
+ }
519
+ }
520
+
521
+ if(global_tmp_final_balance >= 0){
522
+ jQuery('#you_have_a_credit_of').html(global_tmp_final_balance);
523
+ } else {
524
+ jQuery('#you_have_a_credit_of').html('0.00');
525
+ }
526
+
527
+ msg = Translator.translate('Your request was successfully received.');
528
+ msg += '<br>';
529
+ msg += Translator.translate('We will process your data and notify you when everything is done.');
530
+ msg += '<br>';
531
+ msg += Translator.translate('You can check your report status in Forecast history page.');
532
+
533
+ updateRequestForecastInfo('ok', msg, info_id, true);
534
+ }
535
+
536
+ },
537
+ error: function(data) {
538
+ jQuery('#loading-mask').hide();
539
+ request_sent = false;
540
+ info_id = jQuery('#h_data_type').val() == 'input' ? 'request_forecast_info' : 'request_result_info';
541
+ updateRequestForecastInfo('error', Translator.translate('Request error'), info_id);
542
+ }
543
+ });
544
+ }
545
+
546
+ function requestCollect(limit, page){
547
+ if(request_sent){
548
+ return true;
549
+ }
550
+ request_sent = false;
551
+
552
+ request_urls = JSON.parse(global_request_urls);
553
+ ajax_url = request_urls.request_collect_url;
554
+
555
+ jQuery.ajax(ajax_url, {
556
+ type: 'GET',
557
+ async: false,
558
+ data: {'data': formToJson(), 'limit':limit, 'page':page, 'data_size':global_data_size, 'pages':global_split_pages},
559
+ beforeSend: function() {
560
+ jQuery('#loading-mask').show();
561
+ },
562
+ success: function(data) {
563
+ jQuery('#loading-mask').hide();
564
+ request_sent = false;
565
+ api_result = JSON.parse(data)
566
+ global_request_collect_error_msg = 0;
567
+
568
+ if(api_result["_error"] != undefined){
569
+ global_request_collect_error_msg = api_result["_error"];
570
+ } else {
571
+ global_tmp_final_balance = api_result.final_balance;
572
+ }
573
+
574
+ },
575
+ error: function(data) {
576
+ jQuery('#loading-mask').hide();
577
+ request_sent = false;
578
+ global_request_collect_error_msg = data.result;
579
+ info_id = jQuery('#h_data_type').val() == 'input' ? 'request_forecast_info' : 'request_result_info';
580
+ updateRequestForecastInfo('error', Translator.translate('Request error'), info_id);
581
+ }
582
+ });
583
+ }
584
+
585
+ function formatDate(original_date){
586
+ date_format = jQuery('#h_date_format').val();
587
+ date_format = date_format.split('/');
588
+ if(date_format.length != 3){
589
+ return false;
590
+ }
591
+
592
+ original_date = original_date.split('/');
593
+ formated_date = '';
594
+ for(i=0;i<=2;i++){
595
+ if(date_format[i].toLowerCase() == 'd' || date_format[i].toLowerCase() == 'dd'){
596
+ formated_date += original_date[2]+'/';
597
+ } else if(date_format[i].toLowerCase() == 'm' || date_format[i].toLowerCase() == 'mm'){
598
+ formated_date += original_date[1]+'/';
599
+ } else if(date_format[i].toLowerCase() == 'y'){
600
+ formated_date += original_date[0];
601
+ }
602
+ }
603
+
604
+ return formated_date;
605
+ }
606
+
607
+ function subtractDates(date1, date2){
608
+
609
+ var diff = (date2 - date1)/1000;
610
+ var diff = Math.floor(diff);
611
+
612
+ var days = Math.floor(diff/(24*60*60));
613
+ var leftSec = diff - days * 24*60*60;
614
+
615
+ var hrs = Math.floor(leftSec/(60*60));
616
+ var leftSec = leftSec - hrs * 60*60;
617
+
618
+ var min = Math.floor(leftSec/(60));
619
+ var leftSec = leftSec - min * 60;
620
+
621
+ return days;
622
+ }
623
+
624
+ function formToJson(){
625
+ var formFields = {};
626
+ formFields = {}
627
+ var forms = document.getElementById('edit_form').elements
628
+ for(var i = 0; i < forms.length; i++){
629
+ formFields[forms[i].id] = forms[i].value;
630
+ }
631
+ return JSON.stringify(formFields)
632
+ }
633
+
634
+ function updateRequestForecastInfo(type, msg, element_id, disable_request_forecast = false){
635
+ title_color = '';
636
+ title = '';
637
+ if(type == 'error'){
638
+ title_color = 'red';
639
+ title = Translator.translate('Error');
640
+ disableRequestButton();
641
+ }
642
+ if(type == 'info'){
643
+ title_color = 'blue';
644
+ title = Translator.translate('Info');
645
+ }
646
+ if(type == 'ok'){
647
+ title_color = 'green';
648
+ title = Translator.translate('Ok!');
649
+ enableRequestButton();
650
+ }
651
+
652
+ if(disable_request_forecast){
653
+ disableRequestButton();
654
+ }
655
+
656
+ info_html = '';
657
+ info_html += '<br>';
658
+ info_html += '<span style="color:'+title_color+';font-size:12px;font-weight:bold;">';
659
+ info_html += title;
660
+ info_html += '</span>';
661
+ info_html += '<br>';
662
+ info_html += '<span style="color:black;font-weight:normal;font-size:12px">';
663
+ info_html += msg;
664
+ info_html += '</span>';
665
+
666
+ jQuery('#'+element_id).html(info_html);
667
+ }
668
+
669
+ function disableRequestButton(show_tip = true, button_id = 'request_forecast'){
670
+ jQuery('#'+button_id).attr("style", "border:none;background-color:silver;background-image:none");
671
+ jQuery('#'+button_id).prop("disabled",true);
672
+ if(show_tip){
673
+ jQuery('#request_forecast_info').html('<br>'+Translator.translate('Click on Step 1: Check forecast price before'));
674
+ } else {
675
+ jQuery('#request_forecast_info').hide();
676
+ }
677
+ }
678
+
679
+ function enableRequestButton(){
680
+ if(global_enable_request_report){
681
+ jQuery('#request_forecast').removeAttr("style");
682
+ jQuery('#request_forecast').prop("disabled",false);
683
+ info_html = '<br>';
684
+ info_html += '<span style="color:green;font-size:12px;font-weight:bold">';
685
+ info_html += Translator.translate('Ready to go!');
686
+ info_html += '</span>';
687
+ } else {
688
+ info_html = '<br>';
689
+ info_html += '<span style="color:red;font-size:12px;font-weight:bold">';
690
+ info_html += Translator.translate('Request forecast is disabled in this platform.');
691
+ info_html += '</span>';
692
+ }
693
+ jQuery('#request_forecast_info').html(info_html);
694
+ }
695
+
696
+ function twoDigitYearToFour(year){
697
+ aux_two_digit_year = parseInt(year);
698
+ // Magento behavior is to consider any 2-digit year <= 29
699
+ // in 2000's. And > 30 in 1900's
700
+ if(aux_two_digit_year <= 29){
701
+ aux_two_digit_year = aux_two_digit_year+2000;
702
+ } else {
703
+ aux_two_digit_year = aux_two_digit_year+1900;
704
+ }
705
+
706
+ return aux_two_digit_year;
707
+ }
js/get4cast/salesforecast/jquery-1.12.0.min.js ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ /*! jQuery v1.12.0 | (c) jQuery Foundation | jquery.org/license */
2
+ !function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=a.document,e=c.slice,f=c.concat,g=c.push,h=c.indexOf,i={},j=i.toString,k=i.hasOwnProperty,l={},m="1.12.0",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r=function(a,b){return b.toUpperCase()};n.fn=n.prototype={jquery:m,constructor:n,selector:"",length:0,toArray:function(){return e.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:e.call(this)},pushStack:function(a){var b=n.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a){return n.each(this,a)},map:function(a){return this.pushStack(n.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(e.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor()},push:g,sort:c.sort,splice:c.splice},n.extend=n.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||n.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(e=arguments[h]))for(d in e)a=g[d],c=e[d],g!==c&&(j&&c&&(n.isPlainObject(c)||(b=n.isArray(c)))?(b?(b=!1,f=a&&n.isArray(a)?a:[]):f=a&&n.isPlainObject(a)?a:{},g[d]=n.extend(j,f,c)):void 0!==c&&(g[d]=c));return g},n.extend({expando:"jQuery"+(m+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===n.type(a)},isArray:Array.isArray||function(a){return"array"===n.type(a)},isWindow:function(a){return null!=a&&a==a.window},isNumeric:function(a){var b=a&&a.toString();return!n.isArray(a)&&b-parseFloat(b)+1>=0},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},isPlainObject:function(a){var b;if(!a||"object"!==n.type(a)||a.nodeType||n.isWindow(a))return!1;try{if(a.constructor&&!k.call(a,"constructor")&&!k.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}if(!l.ownFirst)for(b in a)return k.call(a,b);for(b in a);return void 0===b||k.call(a,b)},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?i[j.call(a)]||"object":typeof a},globalEval:function(b){b&&n.trim(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(p,"ms-").replace(q,r)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b){var c,d=0;if(s(a)){for(c=a.length;c>d;d++)if(b.call(a[d],d,a[d])===!1)break}else for(d in a)if(b.call(a[d],d,a[d])===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(o,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(s(Object(a))?n.merge(c,"string"==typeof a?[a]:a):g.call(c,a)),c},inArray:function(a,b,c){var d;if(b){if(h)return h.call(b,a,c);for(d=b.length,c=c?0>c?Math.max(0,d+c):c:0;d>c;c++)if(c in b&&b[c]===a)return c}return-1},merge:function(a,b){var c=+b.length,d=0,e=a.length;while(c>d)a[e++]=b[d++];if(c!==c)while(void 0!==b[d])a[e++]=b[d++];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,e,g=0,h=[];if(s(a))for(d=a.length;d>g;g++)e=b(a[g],g,c),null!=e&&h.push(e);else for(g in a)e=b(a[g],g,c),null!=e&&h.push(e);return f.apply([],h)},guid:1,proxy:function(a,b){var c,d,f;return"string"==typeof b&&(f=a[b],b=a,a=f),n.isFunction(a)?(c=e.call(arguments,2),d=function(){return a.apply(b||this,c.concat(e.call(arguments)))},d.guid=a.guid=a.guid||n.guid++,d):void 0},now:function(){return+new Date},support:l}),"function"==typeof Symbol&&(n.fn[Symbol.iterator]=c[Symbol.iterator]),n.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(a,b){i["[object "+b+"]"]=b.toLowerCase()});function s(a){var b=!!a&&"length"in a&&a.length,c=n.type(a);return"function"===c||n.isWindow(a)?!1:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var t=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ga(),z=ga(),A=ga(),B=function(a,b){return a===b&&(l=!0),0},C=1<<31,D={}.hasOwnProperty,E=[],F=E.pop,G=E.push,H=E.push,I=E.slice,J=function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1},K="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",L="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",N="\\["+L+"*("+M+")(?:"+L+"*([*^$|!~]?=)"+L+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+M+"))|)"+L+"*\\]",O=":("+M+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+N+")*)|.*)\\)|)",P=new RegExp(L+"+","g"),Q=new RegExp("^"+L+"+|((?:^|[^\\\\])(?:\\\\.)*)"+L+"+$","g"),R=new RegExp("^"+L+"*,"+L+"*"),S=new RegExp("^"+L+"*([>+~]|"+L+")"+L+"*"),T=new RegExp("="+L+"*([^\\]'\"]*?)"+L+"*\\]","g"),U=new RegExp(O),V=new RegExp("^"+M+"$"),W={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),TAG:new RegExp("^("+M+"|[*])"),ATTR:new RegExp("^"+N),PSEUDO:new RegExp("^"+O),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+L+"*(even|odd|(([+-]|)(\\d*)n|)"+L+"*(?:([+-]|)"+L+"*(\\d+)|))"+L+"*\\)|)","i"),bool:new RegExp("^(?:"+K+")$","i"),needsContext:new RegExp("^"+L+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+L+"*((?:-\\d)?\\d*)"+L+"*\\)|)(?=[^-]|$)","i")},X=/^(?:input|select|textarea|button)$/i,Y=/^h\d$/i,Z=/^[^{]+\{\s*\[native \w/,$=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,_=/[+~]/,aa=/'|\\/g,ba=new RegExp("\\\\([\\da-f]{1,6}"+L+"?|("+L+")|.)","ig"),ca=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},da=function(){m()};try{H.apply(E=I.call(v.childNodes),v.childNodes),E[v.childNodes.length].nodeType}catch(ea){H={apply:E.length?function(a,b){G.apply(a,I.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function fa(a,b,d,e){var f,h,j,k,l,o,r,s,w=b&&b.ownerDocument,x=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==x&&9!==x&&11!==x)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==x&&(o=$.exec(a)))if(f=o[1]){if(9===x){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(w&&(j=w.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(o[2])return H.apply(d,b.getElementsByTagName(a)),d;if((f=o[3])&&c.getElementsByClassName&&b.getElementsByClassName)return H.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==x)w=b,s=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(aa,"\\$&"):b.setAttribute("id",k=u),r=g(a),h=r.length,l=V.test(k)?"#"+k:"[id='"+k+"']";while(h--)r[h]=l+" "+qa(r[h]);s=r.join(","),w=_.test(a)&&oa(b.parentNode)||b}if(s)try{return H.apply(d,w.querySelectorAll(s)),d}catch(y){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(Q,"$1"),b,d,e)}function ga(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ha(a){return a[u]=!0,a}function ia(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ja(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function ka(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||C)-(~a.sourceIndex||C);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function la(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function na(a){return ha(function(b){return b=+b,ha(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function oa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=fa.support={},f=fa.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=fa.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ia(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ia(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Z.test(n.getElementsByClassName),c.getById=ia(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}},d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return"undefined"!=typeof b.getElementsByClassName&&p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=Z.test(n.querySelectorAll))&&(ia(function(a){o.appendChild(a).innerHTML="<a id='"+u+"'></a><select id='"+u+"-\r\\' msallowcapture=''><option selected=''></option></select>",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+L+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+L+"*(?:value|"+K+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ia(function(a){var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+L+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Z.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ia(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",O)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Z.test(o.compareDocumentPosition),t=b||Z.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?J(k,a)-J(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?J(k,a)-J(k,b):0;if(e===f)return ka(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?ka(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},fa.matches=function(a,b){return fa(a,null,null,b)},fa.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(T,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return fa(b,n,null,[a]).length>0},fa.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},fa.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&D.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},fa.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},fa.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=fa.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=fa.selectors={cacheLength:50,createPseudo:ha,match:W,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(ba,ca),a[3]=(a[3]||a[4]||a[5]||"").replace(ba,ca),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||fa.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&fa.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return W.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&U.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(ba,ca).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+L+")"+a+"("+L+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=fa.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(P," ")+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||fa.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ha(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=J(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ha(function(a){var b=[],c=[],d=h(a.replace(Q,"$1"));return d[u]?ha(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ha(function(a){return function(b){return fa(a,b).length>0}}),contains:ha(function(a){return a=a.replace(ba,ca),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ha(function(a){return V.test(a||"")||fa.error("unsupported lang: "+a),a=a.replace(ba,ca).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Y.test(a.nodeName)},input:function(a){return X.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:na(function(){return[0]}),last:na(function(a,b){return[b-1]}),eq:na(function(a,b,c){return[0>c?c+b:c]}),even:na(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:na(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:na(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:na(function(a,b,c){for(var d=0>c?c+b:c;++d<b;)a.push(d);return a})}},d.pseudos.nth=d.pseudos.eq;for(b in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})d.pseudos[b]=la(b);for(b in{submit:!0,reset:!0})d.pseudos[b]=ma(b);function pa(){}pa.prototype=d.filters=d.pseudos,d.setFilters=new pa,g=fa.tokenize=function(a,b){var c,e,f,g,h,i,j,k=z[a+" "];if(k)return b?0:k.slice(0);h=a,i=[],j=d.preFilter;while(h){(!c||(e=R.exec(h)))&&(e&&(h=h.slice(e[0].length)||h),i.push(f=[])),c=!1,(e=S.exec(h))&&(c=e.shift(),f.push({value:c,type:e[0].replace(Q," ")}),h=h.slice(c.length));for(g in d.filter)!(e=W[g].exec(h))||j[g]&&!(e=j[g](e))||(c=e.shift(),f.push({value:c,type:g,matches:e}),h=h.slice(c.length));if(!c)break}return b?h.length:h?fa.error(a):z(a,i).slice(0)};function qa(a){for(var b=0,c=a.length,d="";c>b;b++)d+=a[b].value;return d}function ra(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j,k=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(j=b[u]||(b[u]={}),i=j[b.uniqueID]||(j[b.uniqueID]={}),(h=i[d])&&h[0]===w&&h[1]===f)return k[2]=h[2];if(i[d]=k,k[2]=a(b,c,g))return!0}}}function sa(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function ta(a,b,c){for(var d=0,e=b.length;e>d;d++)fa(a,b[d],c);return c}function ua(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(!c||c(f,d,e))&&(g.push(f),j&&b.push(h));return g}function va(a,b,c,d,e,f){return d&&!d[u]&&(d=va(d)),e&&!e[u]&&(e=va(e,f)),ha(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||ta(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:ua(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=ua(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?J(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=ua(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):H.apply(g,r)})}function wa(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ra(function(a){return a===b},h,!0),l=ra(function(a){return J(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];f>i;i++)if(c=d.relative[a[i].type])m=[ra(sa(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return va(i>1&&sa(m),i>1&&qa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(Q,"$1"),c,e>i&&wa(a.slice(i,e)),f>e&&wa(a=a.slice(e)),f>e&&qa(a))}m.push(c)}return sa(m)}function xa(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=F.call(i));u=ua(u)}H.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&fa.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ha(f):f}return h=fa.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=wa(b[c]),f[u]?d.push(f):e.push(f);f=A(a,xa(e,d)),f.selector=a}return f},i=fa.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(ba,ca),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=W.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(ba,ca),_.test(j[0].type)&&oa(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&qa(j),!a)return H.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,!b||_.test(a)&&oa(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ia(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ia(function(a){return a.innerHTML="<a href='#'></a>","#"===a.firstChild.getAttribute("href")})||ja("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ia(function(a){return a.innerHTML="<input/>",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ja("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ia(function(a){return null==a.getAttribute("disabled")})||ja(K,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),fa}(a);n.find=t,n.expr=t.selectors,n.expr[":"]=n.expr.pseudos,n.uniqueSort=n.unique=t.uniqueSort,n.text=t.getText,n.isXMLDoc=t.isXML,n.contains=t.contains;var u=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&n(a).is(c))break;d.push(a)}return d},v=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},w=n.expr.match.needsContext,x=/^<([\w-]+)\s*\/?>(?:<\/\1>|)$/,y=/^.[^:#\[\.,]*$/;function z(a,b,c){if(n.isFunction(b))return n.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return n.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(y.test(b))return n.filter(b,a,c);b=n.filter(b,a)}return n.grep(a,function(a){return n.inArray(a,b)>-1!==c})}n.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?n.find.matchesSelector(d,a)?[d]:[]:n.find.matches(a,n.grep(b,function(a){return 1===a.nodeType}))},n.fn.extend({find:function(a){var b,c=[],d=this,e=d.length;if("string"!=typeof a)return this.pushStack(n(a).filter(function(){for(b=0;e>b;b++)if(n.contains(d[b],this))return!0}));for(b=0;e>b;b++)n.find(a,d[b],c);return c=this.pushStack(e>1?n.unique(c):c),c.selector=this.selector?this.selector+" "+a:a,c},filter:function(a){return this.pushStack(z(this,a||[],!1))},not:function(a){return this.pushStack(z(this,a||[],!0))},is:function(a){return!!z(this,"string"==typeof a&&w.test(a)?n(a):a||[],!1).length}});var A,B=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,C=n.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||A,"string"==typeof a){if(e="<"===a.charAt(0)&&">"===a.charAt(a.length-1)&&a.length>=3?[null,a,null]:B.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof n?b[0]:b,n.merge(this,n.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),x.test(e[1])&&n.isPlainObject(b))for(e in b)n.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}if(f=d.getElementById(e[2]),f&&f.parentNode){if(f.id!==e[2])return A.find(a);this.length=1,this[0]=f}return this.context=d,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):n.isFunction(a)?"undefined"!=typeof c.ready?c.ready(a):a(n):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),n.makeArray(a,this))};C.prototype=n.fn,A=n(d);var D=/^(?:parents|prev(?:Until|All))/,E={children:!0,contents:!0,next:!0,prev:!0};n.fn.extend({has:function(a){var b,c=n(a,this),d=c.length;return this.filter(function(){for(b=0;d>b;b++)if(n.contains(this,c[b]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=w.test(a)||"string"!=typeof a?n(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&n.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?n.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?n.inArray(this[0],n(a)):n.inArray(a.jquery?a[0]:a,this):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(n.uniqueSort(n.merge(this.get(),n(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function F(a,b){do a=a[b];while(a&&1!==a.nodeType);return a}n.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return u(a,"parentNode")},parentsUntil:function(a,b,c){return u(a,"parentNode",c)},next:function(a){return F(a,"nextSibling")},prev:function(a){return F(a,"previousSibling")},nextAll:function(a){return u(a,"nextSibling")},prevAll:function(a){return u(a,"previousSibling")},nextUntil:function(a,b,c){return u(a,"nextSibling",c)},prevUntil:function(a,b,c){return u(a,"previousSibling",c)},siblings:function(a){return v((a.parentNode||{}).firstChild,a)},children:function(a){return v(a.firstChild)},contents:function(a){return n.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:n.merge([],a.childNodes)}},function(a,b){n.fn[a]=function(c,d){var e=n.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=n.filter(d,e)),this.length>1&&(E[a]||(e=n.uniqueSort(e)),D.test(a)&&(e=e.reverse())),this.pushStack(e)}});var G=/\S+/g;function H(a){var b={};return n.each(a.match(G)||[],function(a,c){b[c]=!0}),b}n.Callbacks=function(a){a="string"==typeof a?H(a):n.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h<f.length)f[h].apply(c[0],c[1])===!1&&a.stopOnFalse&&(h=f.length,c=!1)}a.memory||(c=!1),b=!1,e&&(f=c?[]:"")},j={add:function(){return f&&(c&&!b&&(h=f.length-1,g.push(c)),function d(b){n.each(b,function(b,c){n.isFunction(c)?a.unique&&j.has(c)||f.push(c):c&&c.length&&"string"!==n.type(c)&&d(c)})}(arguments),c&&!b&&i()),this},remove:function(){return n.each(arguments,function(a,b){var c;while((c=n.inArray(b,f,c))>-1)f.splice(c,1),h>=c&&h--}),this},has:function(a){return a?n.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=!0,c||j.disable(),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j},n.extend({Deferred:function(a){var b=[["resolve","done",n.Callbacks("once memory"),"resolved"],["reject","fail",n.Callbacks("once memory"),"rejected"],["notify","progress",n.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return n.Deferred(function(c){n.each(b,function(b,f){var g=n.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&n.isFunction(a.promise)?a.promise().progress(c.notify).done(c.resolve).fail(c.reject):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?n.extend(a,d):d}},e={};return d.pipe=d.then,n.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=e.call(arguments),d=c.length,f=1!==d||a&&n.isFunction(a.promise)?d:0,g=1===f?a:n.Deferred(),h=function(a,b,c){return function(d){b[a]=this,c[a]=arguments.length>1?e.call(arguments):d,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(d>1)for(i=new Array(d),j=new Array(d),k=new Array(d);d>b;b++)c[b]&&n.isFunction(c[b].promise)?c[b].promise().progress(h(b,j,i)).done(h(b,k,c)).fail(g.reject):--f;return f||g.resolveWith(k,c),g.promise()}});var I;n.fn.ready=function(a){return n.ready.promise().done(a),this},n.extend({isReady:!1,readyWait:1,holdReady:function(a){a?n.readyWait++:n.ready(!0)},ready:function(a){(a===!0?--n.readyWait:n.isReady)||(n.isReady=!0,a!==!0&&--n.readyWait>0||(I.resolveWith(d,[n]),n.fn.triggerHandler&&(n(d).triggerHandler("ready"),n(d).off("ready"))))}});function J(){d.addEventListener?(d.removeEventListener("DOMContentLoaded",K),a.removeEventListener("load",K)):(d.detachEvent("onreadystatechange",K),a.detachEvent("onload",K))}function K(){(d.addEventListener||"load"===a.event.type||"complete"===d.readyState)&&(J(),n.ready())}n.ready.promise=function(b){if(!I)if(I=n.Deferred(),"complete"===d.readyState)a.setTimeout(n.ready);else if(d.addEventListener)d.addEventListener("DOMContentLoaded",K),a.addEventListener("load",K);else{d.attachEvent("onreadystatechange",K),a.attachEvent("onload",K);var c=!1;try{c=null==a.frameElement&&d.documentElement}catch(e){}c&&c.doScroll&&!function f(){if(!n.isReady){try{c.doScroll("left")}catch(b){return a.setTimeout(f,50)}J(),n.ready()}}()}return I.promise(b)},n.ready.promise();var L;for(L in n(l))break;l.ownFirst="0"===L,l.inlineBlockNeedsLayout=!1,n(function(){var a,b,c,e;c=d.getElementsByTagName("body")[0],c&&c.style&&(b=d.createElement("div"),e=d.createElement("div"),e.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(e).appendChild(b),"undefined"!=typeof b.style.zoom&&(b.style.cssText="display:inline;margin:0;border:0;padding:1px;width:1px;zoom:1",l.inlineBlockNeedsLayout=a=3===b.offsetWidth,a&&(c.style.zoom=1)),c.removeChild(e))}),function(){var a=d.createElement("div");l.deleteExpando=!0;try{delete a.test}catch(b){l.deleteExpando=!1}a=null}();var M=function(a){var b=n.noData[(a.nodeName+" ").toLowerCase()],c=+a.nodeType||1;return 1!==c&&9!==c?!1:!b||b!==!0&&a.getAttribute("classid")===b},N=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,O=/([A-Z])/g;function P(a,b,c){if(void 0===c&&1===a.nodeType){var d="data-"+b.replace(O,"-$1").toLowerCase();if(c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:N.test(c)?n.parseJSON(c):c}catch(e){}n.data(a,b,c)}else c=void 0}return c}function Q(a){var b;for(b in a)if(("data"!==b||!n.isEmptyObject(a[b]))&&"toJSON"!==b)return!1;
3
+ return!0}function R(a,b,d,e){if(M(a)){var f,g,h=n.expando,i=a.nodeType,j=i?n.cache:a,k=i?a[h]:a[h]&&h;if(k&&j[k]&&(e||j[k].data)||void 0!==d||"string"!=typeof b)return k||(k=i?a[h]=c.pop()||n.guid++:h),j[k]||(j[k]=i?{}:{toJSON:n.noop}),("object"==typeof b||"function"==typeof b)&&(e?j[k]=n.extend(j[k],b):j[k].data=n.extend(j[k].data,b)),g=j[k],e||(g.data||(g.data={}),g=g.data),void 0!==d&&(g[n.camelCase(b)]=d),"string"==typeof b?(f=g[b],null==f&&(f=g[n.camelCase(b)])):f=g,f}}function S(a,b,c){if(M(a)){var d,e,f=a.nodeType,g=f?n.cache:a,h=f?a[n.expando]:n.expando;if(g[h]){if(b&&(d=c?g[h]:g[h].data)){n.isArray(b)?b=b.concat(n.map(b,n.camelCase)):b in d?b=[b]:(b=n.camelCase(b),b=b in d?[b]:b.split(" ")),e=b.length;while(e--)delete d[b[e]];if(c?!Q(d):!n.isEmptyObject(d))return}(c||(delete g[h].data,Q(g[h])))&&(f?n.cleanData([a],!0):l.deleteExpando||g!=g.window?delete g[h]:g[h]=void 0)}}}n.extend({cache:{},noData:{"applet ":!0,"embed ":!0,"object ":"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"},hasData:function(a){return a=a.nodeType?n.cache[a[n.expando]]:a[n.expando],!!a&&!Q(a)},data:function(a,b,c){return R(a,b,c)},removeData:function(a,b){return S(a,b)},_data:function(a,b,c){return R(a,b,c,!0)},_removeData:function(a,b){return S(a,b,!0)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=n.data(f),1===f.nodeType&&!n._data(f,"parsedAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),P(f,d,e[d])));n._data(f,"parsedAttrs",!0)}return e}return"object"==typeof a?this.each(function(){n.data(this,a)}):arguments.length>1?this.each(function(){n.data(this,a,b)}):f?P(f,a,n.data(f,a)):void 0},removeData:function(a){return this.each(function(){n.removeData(this,a)})}}),n.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=n._data(a,b),c&&(!d||n.isArray(c)?d=n._data(a,b,n.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=n.queue(a,b),d=c.length,e=c.shift(),f=n._queueHooks(a,b),g=function(){n.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return n._data(a,c)||n._data(a,c,{empty:n.Callbacks("once memory").add(function(){n._removeData(a,b+"queue"),n._removeData(a,c)})})}}),n.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.length<c?n.queue(this[0],a):void 0===b?this:this.each(function(){var c=n.queue(this,a,b);n._queueHooks(this,a),"fx"===a&&"inprogress"!==c[0]&&n.dequeue(this,a)})},dequeue:function(a){return this.each(function(){n.dequeue(this,a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,b){var c,d=1,e=n.Deferred(),f=this,g=this.length,h=function(){--d||e.resolveWith(f,[f])};"string"!=typeof a&&(b=a,a=void 0),a=a||"fx";while(g--)c=n._data(f[g],a+"queueHooks"),c&&c.empty&&(d++,c.empty.add(h));return h(),e.promise(b)}}),function(){var a;l.shrinkWrapBlocks=function(){if(null!=a)return a;a=!1;var b,c,e;return c=d.getElementsByTagName("body")[0],c&&c.style?(b=d.createElement("div"),e=d.createElement("div"),e.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(e).appendChild(b),"undefined"!=typeof b.style.zoom&&(b.style.cssText="-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:1px;width:1px;zoom:1",b.appendChild(d.createElement("div")).style.width="5px",a=3!==b.offsetWidth),c.removeChild(e),a):void 0}}();var T=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,U=new RegExp("^(?:([+-])=|)("+T+")([a-z%]*)$","i"),V=["Top","Right","Bottom","Left"],W=function(a,b){return a=b||a,"none"===n.css(a,"display")||!n.contains(a.ownerDocument,a)};function X(a,b,c,d){var e,f=1,g=20,h=d?function(){return d.cur()}:function(){return n.css(a,b,"")},i=h(),j=c&&c[3]||(n.cssNumber[b]?"":"px"),k=(n.cssNumber[b]||"px"!==j&&+i)&&U.exec(n.css(a,b));if(k&&k[3]!==j){j=j||k[3],c=c||[],k=+i||1;do f=f||".5",k/=f,n.style(a,b,k+j);while(f!==(f=h()/i)&&1!==f&&--g)}return c&&(k=+k||+i||0,e=c[1]?k+(c[1]+1)*c[2]:+c[2],d&&(d.unit=j,d.start=k,d.end=e)),e}var Y=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===n.type(c)){e=!0;for(h in c)Y(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,n.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(n(a),c)})),b))for(;i>h;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},Z=/^(?:checkbox|radio)$/i,$=/<([\w:-]+)/,_=/^$|\/(?:java|ecma)script/i,aa=/^\s+/,ba="abbr|article|aside|audio|bdi|canvas|data|datalist|details|dialog|figcaption|figure|footer|header|hgroup|main|mark|meter|nav|output|picture|progress|section|summary|template|time|video";function ca(a){var b=ba.split("|"),c=a.createDocumentFragment();if(c.createElement)while(b.length)c.createElement(b.pop());return c}!function(){var a=d.createElement("div"),b=d.createDocumentFragment(),c=d.createElement("input");a.innerHTML=" <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",l.leadingWhitespace=3===a.firstChild.nodeType,l.tbody=!a.getElementsByTagName("tbody").length,l.htmlSerialize=!!a.getElementsByTagName("link").length,l.html5Clone="<:nav></:nav>"!==d.createElement("nav").cloneNode(!0).outerHTML,c.type="checkbox",c.checked=!0,b.appendChild(c),l.appendChecked=c.checked,a.innerHTML="<textarea>x</textarea>",l.noCloneChecked=!!a.cloneNode(!0).lastChild.defaultValue,b.appendChild(a),c=d.createElement("input"),c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),a.appendChild(c),l.checkClone=a.cloneNode(!0).cloneNode(!0).lastChild.checked,l.noCloneEvent=!!a.addEventListener,a[n.expando]=1,l.attributes=!a.getAttribute(n.expando)}();var da={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],area:[1,"<map>","</map>"],param:[1,"<object>","</object>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:l.htmlSerialize?[0,"",""]:[1,"X<div>","</div>"]};da.optgroup=da.option,da.tbody=da.tfoot=da.colgroup=da.caption=da.thead,da.th=da.td;function ea(a,b){var c,d,e=0,f="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):void 0;if(!f)for(f=[],c=a.childNodes||a;null!=(d=c[e]);e++)!b||n.nodeName(d,b)?f.push(d):n.merge(f,ea(d,b));return void 0===b||b&&n.nodeName(a,b)?n.merge([a],f):f}function fa(a,b){for(var c,d=0;null!=(c=a[d]);d++)n._data(c,"globalEval",!b||n._data(b[d],"globalEval"))}var ga=/<|&#?\w+;/,ha=/<tbody/i;function ia(a){Z.test(a.type)&&(a.defaultChecked=a.checked)}function ja(a,b,c,d,e){for(var f,g,h,i,j,k,m,o=a.length,p=ca(b),q=[],r=0;o>r;r++)if(g=a[r],g||0===g)if("object"===n.type(g))n.merge(q,g.nodeType?[g]:g);else if(ga.test(g)){i=i||p.appendChild(b.createElement("div")),j=($.exec(g)||["",""])[1].toLowerCase(),m=da[j]||da._default,i.innerHTML=m[1]+n.htmlPrefilter(g)+m[2],f=m[0];while(f--)i=i.lastChild;if(!l.leadingWhitespace&&aa.test(g)&&q.push(b.createTextNode(aa.exec(g)[0])),!l.tbody){g="table"!==j||ha.test(g)?"<table>"!==m[1]||ha.test(g)?0:i:i.firstChild,f=g&&g.childNodes.length;while(f--)n.nodeName(k=g.childNodes[f],"tbody")&&!k.childNodes.length&&g.removeChild(k)}n.merge(q,i.childNodes),i.textContent="";while(i.firstChild)i.removeChild(i.firstChild);i=p.lastChild}else q.push(b.createTextNode(g));i&&p.removeChild(i),l.appendChecked||n.grep(ea(q,"input"),ia),r=0;while(g=q[r++])if(d&&n.inArray(g,d)>-1)e&&e.push(g);else if(h=n.contains(g.ownerDocument,g),i=ea(p.appendChild(g),"script"),h&&fa(i),c){f=0;while(g=i[f++])_.test(g.type||"")&&c.push(g)}return i=null,p}!function(){var b,c,e=d.createElement("div");for(b in{submit:!0,change:!0,focusin:!0})c="on"+b,(l[b]=c in a)||(e.setAttribute(c,"t"),l[b]=e.attributes[c].expando===!1);e=null}();var ka=/^(?:input|select|textarea)$/i,la=/^key/,ma=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,na=/^(?:focusinfocus|focusoutblur)$/,oa=/^([^.]*)(?:\.(.+)|)/;function pa(){return!0}function qa(){return!1}function ra(){try{return d.activeElement}catch(a){}}function sa(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)sa(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=qa;else if(!e)return a;return 1===f&&(g=e,e=function(a){return n().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=n.guid++)),a.each(function(){n.event.add(this,b,e,d,c)})}n.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n._data(a);if(r){c.handler&&(i=c,c=i.handler,e=i.selector),c.guid||(c.guid=n.guid++),(g=r.events)||(g=r.events={}),(k=r.handle)||(k=r.handle=function(a){return"undefined"==typeof n||a&&n.event.triggered===a.type?void 0:n.event.dispatch.apply(k.elem,arguments)},k.elem=a),b=(b||"").match(G)||[""],h=b.length;while(h--)f=oa.exec(b[h])||[],o=q=f[1],p=(f[2]||"").split(".").sort(),o&&(j=n.event.special[o]||{},o=(e?j.delegateType:j.bindType)||o,j=n.event.special[o]||{},l=n.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&n.expr.match.needsContext.test(e),namespace:p.join(".")},i),(m=g[o])||(m=g[o]=[],m.delegateCount=0,j.setup&&j.setup.call(a,d,p,k)!==!1||(a.addEventListener?a.addEventListener(o,k,!1):a.attachEvent&&a.attachEvent("on"+o,k))),j.add&&(j.add.call(a,l),l.handler.guid||(l.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,l):m.push(l),n.event.global[o]=!0);a=null}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n.hasData(a)&&n._data(a);if(r&&(k=r.events)){b=(b||"").match(G)||[""],j=b.length;while(j--)if(h=oa.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=n.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,m=k[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),i=f=m.length;while(f--)g=m[f],!e&&q!==g.origType||c&&c.guid!==g.guid||h&&!h.test(g.namespace)||d&&d!==g.selector&&("**"!==d||!g.selector)||(m.splice(f,1),g.selector&&m.delegateCount--,l.remove&&l.remove.call(a,g));i&&!m.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||n.removeEvent(a,o,r.handle),delete k[o])}else for(o in k)n.event.remove(a,o+b[j],c,d,!0);n.isEmptyObject(k)&&(delete r.handle,n._removeData(a,"events"))}},trigger:function(b,c,e,f){var g,h,i,j,l,m,o,p=[e||d],q=k.call(b,"type")?b.type:b,r=k.call(b,"namespace")?b.namespace.split("."):[];if(i=m=e=e||d,3!==e.nodeType&&8!==e.nodeType&&!na.test(q+n.event.triggered)&&(q.indexOf(".")>-1&&(r=q.split("."),q=r.shift(),r.sort()),h=q.indexOf(":")<0&&"on"+q,b=b[n.expando]?b:new n.Event(q,"object"==typeof b&&b),b.isTrigger=f?2:3,b.namespace=r.join("."),b.rnamespace=b.namespace?new RegExp("(^|\\.)"+r.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=e),c=null==c?[b]:n.makeArray(c,[b]),l=n.event.special[q]||{},f||!l.trigger||l.trigger.apply(e,c)!==!1)){if(!f&&!l.noBubble&&!n.isWindow(e)){for(j=l.delegateType||q,na.test(j+q)||(i=i.parentNode);i;i=i.parentNode)p.push(i),m=i;m===(e.ownerDocument||d)&&p.push(m.defaultView||m.parentWindow||a)}o=0;while((i=p[o++])&&!b.isPropagationStopped())b.type=o>1?j:l.bindType||q,g=(n._data(i,"events")||{})[b.type]&&n._data(i,"handle"),g&&g.apply(i,c),g=h&&i[h],g&&g.apply&&M(i)&&(b.result=g.apply(i,c),b.result===!1&&b.preventDefault());if(b.type=q,!f&&!b.isDefaultPrevented()&&(!l._default||l._default.apply(p.pop(),c)===!1)&&M(e)&&h&&e[q]&&!n.isWindow(e)){m=e[h],m&&(e[h]=null),n.event.triggered=q;try{e[q]()}catch(s){}n.event.triggered=void 0,m&&(e[h]=m)}return b.result}},dispatch:function(a){a=n.event.fix(a);var b,c,d,f,g,h=[],i=e.call(arguments),j=(n._data(this,"events")||{})[a.type]||[],k=n.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=n.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,c=0;while((g=f.handlers[c++])&&!a.isImmediatePropagationStopped())(!a.rnamespace||a.rnamespace.test(g.namespace))&&(a.handleObj=g,a.data=g.data,d=((n.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==d&&(a.result=d)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&("click"!==a.type||isNaN(a.button)||a.button<1))for(;i!=this;i=i.parentNode||this)if(1===i.nodeType&&(i.disabled!==!0||"click"!==a.type)){for(d=[],c=0;h>c;c++)f=b[c],e=f.selector+" ",void 0===d[e]&&(d[e]=f.needsContext?n(e,this).index(i)>-1:n.find(e,this,null,[i]).length),d[e]&&d.push(f);d.length&&g.push({elem:i,handlers:d})}return h<b.length&&g.push({elem:this,handlers:b.slice(h)}),g},fix:function(a){if(a[n.expando])return a;var b,c,e,f=a.type,g=a,h=this.fixHooks[f];h||(this.fixHooks[f]=h=ma.test(f)?this.mouseHooks:la.test(f)?this.keyHooks:{}),e=h.props?this.props.concat(h.props):this.props,a=new n.Event(g),b=e.length;while(b--)c=e[b],a[c]=g[c];return a.target||(a.target=g.srcElement||d),3===a.target.nodeType&&(a.target=a.target.parentNode),a.metaKey=!!a.metaKey,h.filter?h.filter(a,g):a},props:"altKey bubbles cancelable ctrlKey currentTarget detail eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(a,b){return null==a.which&&(a.which=null!=b.charCode?b.charCode:b.keyCode),a}},mouseHooks:{props:"button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(a,b){var c,e,f,g=b.button,h=b.fromElement;return null==a.pageX&&null!=b.clientX&&(e=a.target.ownerDocument||d,f=e.documentElement,c=e.body,a.pageX=b.clientX+(f&&f.scrollLeft||c&&c.scrollLeft||0)-(f&&f.clientLeft||c&&c.clientLeft||0),a.pageY=b.clientY+(f&&f.scrollTop||c&&c.scrollTop||0)-(f&&f.clientTop||c&&c.clientTop||0)),!a.relatedTarget&&h&&(a.relatedTarget=h===a.target?b.toElement:h),a.which||void 0===g||(a.which=1&g?1:2&g?3:4&g?2:0),a}},special:{load:{noBubble:!0},focus:{trigger:function(){if(this!==ra()&&this.focus)try{return this.focus(),!1}catch(a){}},delegateType:"focusin"},blur:{trigger:function(){return this===ra()&&this.blur?(this.blur(),!1):void 0},delegateType:"focusout"},click:{trigger:function(){return n.nodeName(this,"input")&&"checkbox"===this.type&&this.click?(this.click(),!1):void 0},_default:function(a){return n.nodeName(a.target,"a")}},beforeunload:{postDispatch:function(a){void 0!==a.result&&a.originalEvent&&(a.originalEvent.returnValue=a.result)}}},simulate:function(a,b,c){var d=n.extend(new n.Event,c,{type:a,isSimulated:!0});n.event.trigger(d,null,b),d.isDefaultPrevented()&&c.preventDefault()}},n.removeEvent=d.removeEventListener?function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c)}:function(a,b,c){var d="on"+b;a.detachEvent&&("undefined"==typeof a[d]&&(a[d]=null),a.detachEvent(d,c))},n.Event=function(a,b){return this instanceof n.Event?(a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||void 0===a.defaultPrevented&&a.returnValue===!1?pa:qa):this.type=a,b&&n.extend(this,b),this.timeStamp=a&&a.timeStamp||n.now(),void(this[n.expando]=!0)):new n.Event(a,b)},n.Event.prototype={constructor:n.Event,isDefaultPrevented:qa,isPropagationStopped:qa,isImmediatePropagationStopped:qa,preventDefault:function(){var a=this.originalEvent;this.isDefaultPrevented=pa,a&&(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){var a=this.originalEvent;this.isPropagationStopped=pa,a&&!this.isSimulated&&(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){var a=this.originalEvent;this.isImmediatePropagationStopped=pa,a&&a.stopImmediatePropagation&&a.stopImmediatePropagation(),this.stopPropagation()}},n.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(a,b){n.event.special[a]={delegateType:b,bindType:b,handle:function(a){var c,d=this,e=a.relatedTarget,f=a.handleObj;return(!e||e!==d&&!n.contains(d,e))&&(a.type=f.origType,c=f.handler.apply(this,arguments),a.type=b),c}}}),l.submit||(n.event.special.submit={setup:function(){return n.nodeName(this,"form")?!1:void n.event.add(this,"click._submit keypress._submit",function(a){var b=a.target,c=n.nodeName(b,"input")||n.nodeName(b,"button")?n.prop(b,"form"):void 0;c&&!n._data(c,"submit")&&(n.event.add(c,"submit._submit",function(a){a._submitBubble=!0}),n._data(c,"submit",!0))})},postDispatch:function(a){a._submitBubble&&(delete a._submitBubble,this.parentNode&&!a.isTrigger&&n.event.simulate("submit",this.parentNode,a))},teardown:function(){return n.nodeName(this,"form")?!1:void n.event.remove(this,"._submit")}}),l.change||(n.event.special.change={setup:function(){return ka.test(this.nodeName)?(("checkbox"===this.type||"radio"===this.type)&&(n.event.add(this,"propertychange._change",function(a){"checked"===a.originalEvent.propertyName&&(this._justChanged=!0)}),n.event.add(this,"click._change",function(a){this._justChanged&&!a.isTrigger&&(this._justChanged=!1),n.event.simulate("change",this,a)})),!1):void n.event.add(this,"beforeactivate._change",function(a){var b=a.target;ka.test(b.nodeName)&&!n._data(b,"change")&&(n.event.add(b,"change._change",function(a){!this.parentNode||a.isSimulated||a.isTrigger||n.event.simulate("change",this.parentNode,a)}),n._data(b,"change",!0))})},handle:function(a){var b=a.target;return this!==b||a.isSimulated||a.isTrigger||"radio"!==b.type&&"checkbox"!==b.type?a.handleObj.handler.apply(this,arguments):void 0},teardown:function(){return n.event.remove(this,"._change"),!ka.test(this.nodeName)}}),l.focusin||n.each({focus:"focusin",blur:"focusout"},function(a,b){var c=function(a){n.event.simulate(b,a.target,n.event.fix(a))};n.event.special[b]={setup:function(){var d=this.ownerDocument||this,e=n._data(d,b);e||d.addEventListener(a,c,!0),n._data(d,b,(e||0)+1)},teardown:function(){var d=this.ownerDocument||this,e=n._data(d,b)-1;e?n._data(d,b,e):(d.removeEventListener(a,c,!0),n._removeData(d,b))}}}),n.fn.extend({on:function(a,b,c,d){return sa(this,a,b,c,d)},one:function(a,b,c,d){return sa(this,a,b,c,d,1)},off:function(a,b,c){var d,e;if(a&&a.preventDefault&&a.handleObj)return d=a.handleObj,n(a.delegateTarget).off(d.namespace?d.origType+"."+d.namespace:d.origType,d.selector,d.handler),this;if("object"==typeof a){for(e in a)this.off(e,b,a[e]);return this}return(b===!1||"function"==typeof b)&&(c=b,b=void 0),c===!1&&(c=qa),this.each(function(){n.event.remove(this,a,c,b)})},trigger:function(a,b){return this.each(function(){n.event.trigger(a,b,this)})},triggerHandler:function(a,b){var c=this[0];return c?n.event.trigger(a,b,c,!0):void 0}});var ta=/ jQuery\d+="(?:null|\d+)"/g,ua=new RegExp("<(?:"+ba+")[\\s/>]","i"),va=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi,wa=/<script|<style|<link/i,xa=/checked\s*(?:[^=]|=\s*.checked.)/i,ya=/^true\/(.*)/,za=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g,Aa=ca(d),Ba=Aa.appendChild(d.createElement("div"));function Ca(a,b){return n.nodeName(a,"table")&&n.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function Da(a){return a.type=(null!==n.find.attr(a,"type"))+"/"+a.type,a}function Ea(a){var b=ya.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function Fa(a,b){if(1===b.nodeType&&n.hasData(a)){var c,d,e,f=n._data(a),g=n._data(b,f),h=f.events;if(h){delete g.handle,g.events={};for(c in h)for(d=0,e=h[c].length;e>d;d++)n.event.add(b,c,h[c][d])}g.data&&(g.data=n.extend({},g.data))}}function Ga(a,b){var c,d,e;if(1===b.nodeType){if(c=b.nodeName.toLowerCase(),!l.noCloneEvent&&b[n.expando]){e=n._data(b);for(d in e.events)n.removeEvent(b,d,e.handle);b.removeAttribute(n.expando)}"script"===c&&b.text!==a.text?(Da(b).text=a.text,Ea(b)):"object"===c?(b.parentNode&&(b.outerHTML=a.outerHTML),l.html5Clone&&a.innerHTML&&!n.trim(b.innerHTML)&&(b.innerHTML=a.innerHTML)):"input"===c&&Z.test(a.type)?(b.defaultChecked=b.checked=a.checked,b.value!==a.value&&(b.value=a.value)):"option"===c?b.defaultSelected=b.selected=a.defaultSelected:("input"===c||"textarea"===c)&&(b.defaultValue=a.defaultValue)}}function Ha(a,b,c,d){b=f.apply([],b);var e,g,h,i,j,k,m=0,o=a.length,p=o-1,q=b[0],r=n.isFunction(q);if(r||o>1&&"string"==typeof q&&!l.checkClone&&xa.test(q))return a.each(function(e){var f=a.eq(e);r&&(b[0]=q.call(this,e,f.html())),Ha(f,b,c,d)});if(o&&(k=ja(b,a[0].ownerDocument,!1,a,d),e=k.firstChild,1===k.childNodes.length&&(k=e),e||d)){for(i=n.map(ea(k,"script"),Da),h=i.length;o>m;m++)g=k,m!==p&&(g=n.clone(g,!0,!0),h&&n.merge(i,ea(g,"script"))),c.call(a[m],g,m);if(h)for(j=i[i.length-1].ownerDocument,n.map(i,Ea),m=0;h>m;m++)g=i[m],_.test(g.type||"")&&!n._data(g,"globalEval")&&n.contains(j,g)&&(g.src?n._evalUrl&&n._evalUrl(g.src):n.globalEval((g.text||g.textContent||g.innerHTML||"").replace(za,"")));k=e=null}return a}function Ia(a,b,c){for(var d,e=b?n.filter(b,a):a,f=0;null!=(d=e[f]);f++)c||1!==d.nodeType||n.cleanData(ea(d)),d.parentNode&&(c&&n.contains(d.ownerDocument,d)&&fa(ea(d,"script")),d.parentNode.removeChild(d));return a}n.extend({htmlPrefilter:function(a){return a.replace(va,"<$1></$2>")},clone:function(a,b,c){var d,e,f,g,h,i=n.contains(a.ownerDocument,a);if(l.html5Clone||n.isXMLDoc(a)||!ua.test("<"+a.nodeName+">")?f=a.cloneNode(!0):(Ba.innerHTML=a.outerHTML,Ba.removeChild(f=Ba.firstChild)),!(l.noCloneEvent&&l.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||n.isXMLDoc(a)))for(d=ea(f),h=ea(a),g=0;null!=(e=h[g]);++g)d[g]&&Ga(e,d[g]);if(b)if(c)for(h=h||ea(a),d=d||ea(f),g=0;null!=(e=h[g]);g++)Fa(e,d[g]);else Fa(a,f);return d=ea(f,"script"),d.length>0&&fa(d,!i&&ea(a,"script")),d=h=e=null,f},cleanData:function(a,b){for(var d,e,f,g,h=0,i=n.expando,j=n.cache,k=l.attributes,m=n.event.special;null!=(d=a[h]);h++)if((b||M(d))&&(f=d[i],g=f&&j[f])){if(g.events)for(e in g.events)m[e]?n.event.remove(d,e):n.removeEvent(d,e,g.handle);j[f]&&(delete j[f],k||"undefined"==typeof d.removeAttribute?d[i]=void 0:d.removeAttribute(i),c.push(f))}}}),n.fn.extend({domManip:Ha,detach:function(a){return Ia(this,a,!0)},remove:function(a){return Ia(this,a)},text:function(a){return Y(this,function(a){return void 0===a?n.text(this):this.empty().append((this[0]&&this[0].ownerDocument||d).createTextNode(a))},null,a,arguments.length)},append:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.appendChild(a)}})},prepend:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},empty:function(){for(var a,b=0;null!=(a=this[b]);b++){1===a.nodeType&&n.cleanData(ea(a,!1));while(a.firstChild)a.removeChild(a.firstChild);a.options&&n.nodeName(a,"select")&&(a.options.length=0)}return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return n.clone(this,a,b)})},html:function(a){return Y(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a)return 1===b.nodeType?b.innerHTML.replace(ta,""):void 0;if("string"==typeof a&&!wa.test(a)&&(l.htmlSerialize||!ua.test(a))&&(l.leadingWhitespace||!aa.test(a))&&!da[($.exec(a)||["",""])[1].toLowerCase()]){a=n.htmlPrefilter(a);try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(n.cleanData(ea(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=[];return Ha(this,arguments,function(b){var c=this.parentNode;n.inArray(this,a)<0&&(n.cleanData(ea(this)),c&&c.replaceChild(b,this))},a)}}),n.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){n.fn[a]=function(a){for(var c,d=0,e=[],f=n(a),h=f.length-1;h>=d;d++)c=d===h?this:this.clone(!0),n(f[d])[b](c),g.apply(e,c.get());return this.pushStack(e)}});var Ja,Ka={HTML:"block",BODY:"block"};function La(a,b){var c=n(b.createElement(a)).appendTo(b.body),d=n.css(c[0],"display");return c.detach(),d}function Ma(a){var b=d,c=Ka[a];return c||(c=La(a,b),"none"!==c&&c||(Ja=(Ja||n("<iframe frameborder='0' width='0' height='0'/>")).appendTo(b.documentElement),b=(Ja[0].contentWindow||Ja[0].contentDocument).document,b.write(),b.close(),c=La(a,b),Ja.detach()),Ka[a]=c),c}var Na=/^margin/,Oa=new RegExp("^("+T+")(?!px)[a-z%]+$","i"),Pa=function(a,b,c,d){var e,f,g={};for(f in b)g[f]=a.style[f],a.style[f]=b[f];e=c.apply(a,d||[]);for(f in b)a.style[f]=g[f];return e},Qa=d.documentElement;!function(){var b,c,e,f,g,h,i=d.createElement("div"),j=d.createElement("div");if(j.style){j.style.cssText="float:left;opacity:.5",l.opacity="0.5"===j.style.opacity,l.cssFloat=!!j.style.cssFloat,j.style.backgroundClip="content-box",j.cloneNode(!0).style.backgroundClip="",l.clearCloneStyle="content-box"===j.style.backgroundClip,i=d.createElement("div"),i.style.cssText="border:0;width:8px;height:0;top:0;left:-9999px;padding:0;margin-top:1px;position:absolute",j.innerHTML="",i.appendChild(j),l.boxSizing=""===j.style.boxSizing||""===j.style.MozBoxSizing||""===j.style.WebkitBoxSizing,n.extend(l,{reliableHiddenOffsets:function(){return null==b&&k(),f},boxSizingReliable:function(){return null==b&&k(),e},pixelMarginRight:function(){return null==b&&k(),c},pixelPosition:function(){return null==b&&k(),b},reliableMarginRight:function(){return null==b&&k(),g},reliableMarginLeft:function(){return null==b&&k(),h}});function k(){var k,l,m=d.documentElement;m.appendChild(i),j.style.cssText="-webkit-box-sizing:border-box;box-sizing:border-box;position:relative;display:block;margin:auto;border:1px;padding:1px;top:1%;width:50%",b=e=h=!1,c=g=!0,a.getComputedStyle&&(l=a.getComputedStyle(j),b="1%"!==(l||{}).top,h="2px"===(l||{}).marginLeft,e="4px"===(l||{width:"4px"}).width,j.style.marginRight="50%",c="4px"===(l||{marginRight:"4px"}).marginRight,k=j.appendChild(d.createElement("div")),k.style.cssText=j.style.cssText="-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:0",k.style.marginRight=k.style.width="0",j.style.width="1px",g=!parseFloat((a.getComputedStyle(k)||{}).marginRight),j.removeChild(k)),j.style.display="none",f=0===j.getClientRects().length,f&&(j.style.display="",j.innerHTML="<table><tr><td></td><td>t</td></tr></table>",k=j.getElementsByTagName("td"),k[0].style.cssText="margin:0;border:0;padding:0;display:none",f=0===k[0].offsetHeight,f&&(k[0].style.display="",k[1].style.display="none",f=0===k[0].offsetHeight)),m.removeChild(i)}}}();var Ra,Sa,Ta=/^(top|right|bottom|left)$/;a.getComputedStyle?(Ra=function(b){var c=b.ownerDocument.defaultView;return c.opener||(c=a),c.getComputedStyle(b)},Sa=function(a,b,c){var d,e,f,g,h=a.style;return c=c||Ra(a),g=c?c.getPropertyValue(b)||c[b]:void 0,c&&(""!==g||n.contains(a.ownerDocument,a)||(g=n.style(a,b)),!l.pixelMarginRight()&&Oa.test(g)&&Na.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=g,g=c.width,h.width=d,h.minWidth=e,h.maxWidth=f)),void 0===g?g:g+""}):Qa.currentStyle&&(Ra=function(a){return a.currentStyle},Sa=function(a,b,c){var d,e,f,g,h=a.style;return c=c||Ra(a),g=c?c[b]:void 0,null==g&&h&&h[b]&&(g=h[b]),Oa.test(g)&&!Ta.test(b)&&(d=h.left,e=a.runtimeStyle,f=e&&e.left,f&&(e.left=a.currentStyle.left),h.left="fontSize"===b?"1em":g,g=h.pixelLeft+"px",h.left=d,f&&(e.left=f)),void 0===g?g:g+""||"auto"});function Ua(a,b){return{get:function(){return a()?void delete this.get:(this.get=b).apply(this,arguments)}}}var Va=/alpha\([^)]*\)/i,Wa=/opacity\s*=\s*([^)]*)/i,Xa=/^(none|table(?!-c[ea]).+)/,Ya=new RegExp("^("+T+")(.*)$","i"),Za={position:"absolute",visibility:"hidden",display:"block"},$a={letterSpacing:"0",fontWeight:"400"},_a=["Webkit","O","Moz","ms"],ab=d.createElement("div").style;function bb(a){if(a in ab)return a;var b=a.charAt(0).toUpperCase()+a.slice(1),c=_a.length;while(c--)if(a=_a[c]+b,a in ab)return a}function cb(a,b){for(var c,d,e,f=[],g=0,h=a.length;h>g;g++)d=a[g],d.style&&(f[g]=n._data(d,"olddisplay"),c=d.style.display,b?(f[g]||"none"!==c||(d.style.display=""),""===d.style.display&&W(d)&&(f[g]=n._data(d,"olddisplay",Ma(d.nodeName)))):(e=W(d),(c&&"none"!==c||!e)&&n._data(d,"olddisplay",e?c:n.css(d,"display"))));for(g=0;h>g;g++)d=a[g],d.style&&(b&&"none"!==d.style.display&&""!==d.style.display||(d.style.display=b?f[g]||"":"none"));return a}function db(a,b,c){var d=Ya.exec(b);return d?Math.max(0,d[1]-(c||0))+(d[2]||"px"):b}function eb(a,b,c,d,e){for(var f=c===(d?"border":"content")?4:"width"===b?1:0,g=0;4>f;f+=2)"margin"===c&&(g+=n.css(a,c+V[f],!0,e)),d?("content"===c&&(g-=n.css(a,"padding"+V[f],!0,e)),"margin"!==c&&(g-=n.css(a,"border"+V[f]+"Width",!0,e))):(g+=n.css(a,"padding"+V[f],!0,e),"padding"!==c&&(g+=n.css(a,"border"+V[f]+"Width",!0,e)));return g}function fb(b,c,e){var f=!0,g="width"===c?b.offsetWidth:b.offsetHeight,h=Ra(b),i=l.boxSizing&&"border-box"===n.css(b,"boxSizing",!1,h);if(d.msFullscreenElement&&a.top!==a&&b.getClientRects().length&&(g=Math.round(100*b.getBoundingClientRect()[c])),0>=g||null==g){if(g=Sa(b,c,h),(0>g||null==g)&&(g=b.style[c]),Oa.test(g))return g;f=i&&(l.boxSizingReliable()||g===b.style[c]),g=parseFloat(g)||0}return g+eb(b,c,e||(i?"border":"content"),f,h)+"px"}n.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=Sa(a,"opacity");return""===c?"1":c}}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":l.cssFloat?"cssFloat":"styleFloat"},style:function(a,b,c,d){if(a&&3!==a.nodeType&&8!==a.nodeType&&a.style){var e,f,g,h=n.camelCase(b),i=a.style;if(b=n.cssProps[h]||(n.cssProps[h]=bb(h)||h),g=n.cssHooks[b]||n.cssHooks[h],void 0===c)return g&&"get"in g&&void 0!==(e=g.get(a,!1,d))?e:i[b];if(f=typeof c,"string"===f&&(e=U.exec(c))&&e[1]&&(c=X(a,b,e),f="number"),null!=c&&c===c&&("number"===f&&(c+=e&&e[3]||(n.cssNumber[h]?"":"px")),l.clearCloneStyle||""!==c||0!==b.indexOf("background")||(i[b]="inherit"),!(g&&"set"in g&&void 0===(c=g.set(a,c,d)))))try{i[b]=c}catch(j){}}},css:function(a,b,c,d){var e,f,g,h=n.camelCase(b);return b=n.cssProps[h]||(n.cssProps[h]=bb(h)||h),g=n.cssHooks[b]||n.cssHooks[h],g&&"get"in g&&(f=g.get(a,!0,c)),void 0===f&&(f=Sa(a,b,d)),"normal"===f&&b in $a&&(f=$a[b]),""===c||c?(e=parseFloat(f),c===!0||isFinite(e)?e||0:f):f}}),n.each(["height","width"],function(a,b){n.cssHooks[b]={get:function(a,c,d){return c?Xa.test(n.css(a,"display"))&&0===a.offsetWidth?Pa(a,Za,function(){return fb(a,b,d)}):fb(a,b,d):void 0},set:function(a,c,d){var e=d&&Ra(a);return db(a,c,d?eb(a,b,d,l.boxSizing&&"border-box"===n.css(a,"boxSizing",!1,e),e):0)}}}),l.opacity||(n.cssHooks.opacity={get:function(a,b){return Wa.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?.01*parseFloat(RegExp.$1)+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=n.isNumeric(b)?"alpha(opacity="+100*b+")":"",f=d&&d.filter||c.filter||"";c.zoom=1,(b>=1||""===b)&&""===n.trim(f.replace(Va,""))&&c.removeAttribute&&(c.removeAttribute("filter"),""===b||d&&!d.filter)||(c.filter=Va.test(f)?f.replace(Va,e):f+" "+e)}}),n.cssHooks.marginRight=Ua(l.reliableMarginRight,function(a,b){return b?Pa(a,{display:"inline-block"},Sa,[a,"marginRight"]):void 0}),n.cssHooks.marginLeft=Ua(l.reliableMarginLeft,function(a,b){return b?(parseFloat(Sa(a,"marginLeft"))||(n.contains(a.ownerDocument,a)?a.getBoundingClientRect().left-Pa(a,{
4
+ marginLeft:0},function(){return a.getBoundingClientRect().left}):0))+"px":void 0}),n.each({margin:"",padding:"",border:"Width"},function(a,b){n.cssHooks[a+b]={expand:function(c){for(var d=0,e={},f="string"==typeof c?c.split(" "):[c];4>d;d++)e[a+V[d]+b]=f[d]||f[d-2]||f[0];return e}},Na.test(a)||(n.cssHooks[a+b].set=db)}),n.fn.extend({css:function(a,b){return Y(this,function(a,b,c){var d,e,f={},g=0;if(n.isArray(b)){for(d=Ra(a),e=b.length;e>g;g++)f[b[g]]=n.css(a,b[g],!1,d);return f}return void 0!==c?n.style(a,b,c):n.css(a,b)},a,b,arguments.length>1)},show:function(){return cb(this,!0)},hide:function(){return cb(this)},toggle:function(a){return"boolean"==typeof a?a?this.show():this.hide():this.each(function(){W(this)?n(this).show():n(this).hide()})}});function gb(a,b,c,d,e){return new gb.prototype.init(a,b,c,d,e)}n.Tween=gb,gb.prototype={constructor:gb,init:function(a,b,c,d,e,f){this.elem=a,this.prop=c,this.easing=e||n.easing._default,this.options=b,this.start=this.now=this.cur(),this.end=d,this.unit=f||(n.cssNumber[c]?"":"px")},cur:function(){var a=gb.propHooks[this.prop];return a&&a.get?a.get(this):gb.propHooks._default.get(this)},run:function(a){var b,c=gb.propHooks[this.prop];return this.options.duration?this.pos=b=n.easing[this.easing](a,this.options.duration*a,0,1,this.options.duration):this.pos=b=a,this.now=(this.end-this.start)*b+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),c&&c.set?c.set(this):gb.propHooks._default.set(this),this}},gb.prototype.init.prototype=gb.prototype,gb.propHooks={_default:{get:function(a){var b;return 1!==a.elem.nodeType||null!=a.elem[a.prop]&&null==a.elem.style[a.prop]?a.elem[a.prop]:(b=n.css(a.elem,a.prop,""),b&&"auto"!==b?b:0)},set:function(a){n.fx.step[a.prop]?n.fx.step[a.prop](a):1!==a.elem.nodeType||null==a.elem.style[n.cssProps[a.prop]]&&!n.cssHooks[a.prop]?a.elem[a.prop]=a.now:n.style(a.elem,a.prop,a.now+a.unit)}}},gb.propHooks.scrollTop=gb.propHooks.scrollLeft={set:function(a){a.elem.nodeType&&a.elem.parentNode&&(a.elem[a.prop]=a.now)}},n.easing={linear:function(a){return a},swing:function(a){return.5-Math.cos(a*Math.PI)/2},_default:"swing"},n.fx=gb.prototype.init,n.fx.step={};var hb,ib,jb=/^(?:toggle|show|hide)$/,kb=/queueHooks$/;function lb(){return a.setTimeout(function(){hb=void 0}),hb=n.now()}function mb(a,b){var c,d={height:a},e=0;for(b=b?1:0;4>e;e+=2-b)c=V[e],d["margin"+c]=d["padding"+c]=a;return b&&(d.opacity=d.width=a),d}function nb(a,b,c){for(var d,e=(qb.tweeners[b]||[]).concat(qb.tweeners["*"]),f=0,g=e.length;g>f;f++)if(d=e[f].call(c,b,a))return d}function ob(a,b,c){var d,e,f,g,h,i,j,k,m=this,o={},p=a.style,q=a.nodeType&&W(a),r=n._data(a,"fxshow");c.queue||(h=n._queueHooks(a,"fx"),null==h.unqueued&&(h.unqueued=0,i=h.empty.fire,h.empty.fire=function(){h.unqueued||i()}),h.unqueued++,m.always(function(){m.always(function(){h.unqueued--,n.queue(a,"fx").length||h.empty.fire()})})),1===a.nodeType&&("height"in b||"width"in b)&&(c.overflow=[p.overflow,p.overflowX,p.overflowY],j=n.css(a,"display"),k="none"===j?n._data(a,"olddisplay")||Ma(a.nodeName):j,"inline"===k&&"none"===n.css(a,"float")&&(l.inlineBlockNeedsLayout&&"inline"!==Ma(a.nodeName)?p.zoom=1:p.display="inline-block")),c.overflow&&(p.overflow="hidden",l.shrinkWrapBlocks()||m.always(function(){p.overflow=c.overflow[0],p.overflowX=c.overflow[1],p.overflowY=c.overflow[2]}));for(d in b)if(e=b[d],jb.exec(e)){if(delete b[d],f=f||"toggle"===e,e===(q?"hide":"show")){if("show"!==e||!r||void 0===r[d])continue;q=!0}o[d]=r&&r[d]||n.style(a,d)}else j=void 0;if(n.isEmptyObject(o))"inline"===("none"===j?Ma(a.nodeName):j)&&(p.display=j);else{r?"hidden"in r&&(q=r.hidden):r=n._data(a,"fxshow",{}),f&&(r.hidden=!q),q?n(a).show():m.done(function(){n(a).hide()}),m.done(function(){var b;n._removeData(a,"fxshow");for(b in o)n.style(a,b,o[b])});for(d in o)g=nb(q?r[d]:0,d,m),d in r||(r[d]=g.start,q&&(g.end=g.start,g.start="width"===d||"height"===d?1:0))}}function pb(a,b){var c,d,e,f,g;for(c in a)if(d=n.camelCase(c),e=b[d],f=a[c],n.isArray(f)&&(e=f[1],f=a[c]=f[0]),c!==d&&(a[d]=f,delete a[c]),g=n.cssHooks[d],g&&"expand"in g){f=g.expand(f),delete a[d];for(c in f)c in a||(a[c]=f[c],b[c]=e)}else b[d]=e}function qb(a,b,c){var d,e,f=0,g=qb.prefilters.length,h=n.Deferred().always(function(){delete i.elem}),i=function(){if(e)return!1;for(var b=hb||lb(),c=Math.max(0,j.startTime+j.duration-b),d=c/j.duration||0,f=1-d,g=0,i=j.tweens.length;i>g;g++)j.tweens[g].run(f);return h.notifyWith(a,[j,f,c]),1>f&&i?c:(h.resolveWith(a,[j]),!1)},j=h.promise({elem:a,props:n.extend({},b),opts:n.extend(!0,{specialEasing:{},easing:n.easing._default},c),originalProperties:b,originalOptions:c,startTime:hb||lb(),duration:c.duration,tweens:[],createTween:function(b,c){var d=n.Tween(a,j.opts,b,c,j.opts.specialEasing[b]||j.opts.easing);return j.tweens.push(d),d},stop:function(b){var c=0,d=b?j.tweens.length:0;if(e)return this;for(e=!0;d>c;c++)j.tweens[c].run(1);return b?(h.notifyWith(a,[j,1,0]),h.resolveWith(a,[j,b])):h.rejectWith(a,[j,b]),this}}),k=j.props;for(pb(k,j.opts.specialEasing);g>f;f++)if(d=qb.prefilters[f].call(j,a,k,j.opts))return n.isFunction(d.stop)&&(n._queueHooks(j.elem,j.opts.queue).stop=n.proxy(d.stop,d)),d;return n.map(k,nb,j),n.isFunction(j.opts.start)&&j.opts.start.call(a,j),n.fx.timer(n.extend(i,{elem:a,anim:j,queue:j.opts.queue})),j.progress(j.opts.progress).done(j.opts.done,j.opts.complete).fail(j.opts.fail).always(j.opts.always)}n.Animation=n.extend(qb,{tweeners:{"*":[function(a,b){var c=this.createTween(a,b);return X(c.elem,a,U.exec(b),c),c}]},tweener:function(a,b){n.isFunction(a)?(b=a,a=["*"]):a=a.match(G);for(var c,d=0,e=a.length;e>d;d++)c=a[d],qb.tweeners[c]=qb.tweeners[c]||[],qb.tweeners[c].unshift(b)},prefilters:[ob],prefilter:function(a,b){b?qb.prefilters.unshift(a):qb.prefilters.push(a)}}),n.speed=function(a,b,c){var d=a&&"object"==typeof a?n.extend({},a):{complete:c||!c&&b||n.isFunction(a)&&a,duration:a,easing:c&&b||b&&!n.isFunction(b)&&b};return d.duration=n.fx.off?0:"number"==typeof d.duration?d.duration:d.duration in n.fx.speeds?n.fx.speeds[d.duration]:n.fx.speeds._default,(null==d.queue||d.queue===!0)&&(d.queue="fx"),d.old=d.complete,d.complete=function(){n.isFunction(d.old)&&d.old.call(this),d.queue&&n.dequeue(this,d.queue)},d},n.fn.extend({fadeTo:function(a,b,c,d){return this.filter(W).css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=n.isEmptyObject(a),f=n.speed(b,c,d),g=function(){var b=qb(this,n.extend({},a),f);(e||n._data(this,"finish"))&&b.stop(!0)};return g.finish=g,e||f.queue===!1?this.each(g):this.queue(f.queue,g)},stop:function(a,b,c){var d=function(a){var b=a.stop;delete a.stop,b(c)};return"string"!=typeof a&&(c=b,b=a,a=void 0),b&&a!==!1&&this.queue(a||"fx",[]),this.each(function(){var b=!0,e=null!=a&&a+"queueHooks",f=n.timers,g=n._data(this);if(e)g[e]&&g[e].stop&&d(g[e]);else for(e in g)g[e]&&g[e].stop&&kb.test(e)&&d(g[e]);for(e=f.length;e--;)f[e].elem!==this||null!=a&&f[e].queue!==a||(f[e].anim.stop(c),b=!1,f.splice(e,1));(b||!c)&&n.dequeue(this,a)})},finish:function(a){return a!==!1&&(a=a||"fx"),this.each(function(){var b,c=n._data(this),d=c[a+"queue"],e=c[a+"queueHooks"],f=n.timers,g=d?d.length:0;for(c.finish=!0,n.queue(this,a,[]),e&&e.stop&&e.stop.call(this,!0),b=f.length;b--;)f[b].elem===this&&f[b].queue===a&&(f[b].anim.stop(!0),f.splice(b,1));for(b=0;g>b;b++)d[b]&&d[b].finish&&d[b].finish.call(this);delete c.finish})}}),n.each(["toggle","show","hide"],function(a,b){var c=n.fn[b];n.fn[b]=function(a,d,e){return null==a||"boolean"==typeof a?c.apply(this,arguments):this.animate(mb(b,!0),a,d,e)}}),n.each({slideDown:mb("show"),slideUp:mb("hide"),slideToggle:mb("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){n.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),n.timers=[],n.fx.tick=function(){var a,b=n.timers,c=0;for(hb=n.now();c<b.length;c++)a=b[c],a()||b[c]!==a||b.splice(c--,1);b.length||n.fx.stop(),hb=void 0},n.fx.timer=function(a){n.timers.push(a),a()?n.fx.start():n.timers.pop()},n.fx.interval=13,n.fx.start=function(){ib||(ib=a.setInterval(n.fx.tick,n.fx.interval))},n.fx.stop=function(){a.clearInterval(ib),ib=null},n.fx.speeds={slow:600,fast:200,_default:400},n.fn.delay=function(b,c){return b=n.fx?n.fx.speeds[b]||b:b,c=c||"fx",this.queue(c,function(c,d){var e=a.setTimeout(c,b);d.stop=function(){a.clearTimeout(e)}})},function(){var a,b=d.createElement("input"),c=d.createElement("div"),e=d.createElement("select"),f=e.appendChild(d.createElement("option"));c=d.createElement("div"),c.setAttribute("className","t"),c.innerHTML=" <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",a=c.getElementsByTagName("a")[0],b.setAttribute("type","checkbox"),c.appendChild(b),a=c.getElementsByTagName("a")[0],a.style.cssText="top:1px",l.getSetAttribute="t"!==c.className,l.style=/top/.test(a.getAttribute("style")),l.hrefNormalized="/a"===a.getAttribute("href"),l.checkOn=!!b.value,l.optSelected=f.selected,l.enctype=!!d.createElement("form").enctype,e.disabled=!0,l.optDisabled=!f.disabled,b=d.createElement("input"),b.setAttribute("value",""),l.input=""===b.getAttribute("value"),b.value="t",b.setAttribute("type","radio"),l.radioValue="t"===b.value}();var rb=/\r/g;n.fn.extend({val:function(a){var b,c,d,e=this[0];{if(arguments.length)return d=n.isFunction(a),this.each(function(c){var e;1===this.nodeType&&(e=d?a.call(this,c,n(this).val()):a,null==e?e="":"number"==typeof e?e+="":n.isArray(e)&&(e=n.map(e,function(a){return null==a?"":a+""})),b=n.valHooks[this.type]||n.valHooks[this.nodeName.toLowerCase()],b&&"set"in b&&void 0!==b.set(this,e,"value")||(this.value=e))});if(e)return b=n.valHooks[e.type]||n.valHooks[e.nodeName.toLowerCase()],b&&"get"in b&&void 0!==(c=b.get(e,"value"))?c:(c=e.value,"string"==typeof c?c.replace(rb,""):null==c?"":c)}}}),n.extend({valHooks:{option:{get:function(a){var b=n.find.attr(a,"value");return null!=b?b:n.trim(n.text(a))}},select:{get:function(a){for(var b,c,d=a.options,e=a.selectedIndex,f="select-one"===a.type||0>e,g=f?null:[],h=f?e+1:d.length,i=0>e?h:f?e:0;h>i;i++)if(c=d[i],(c.selected||i===e)&&(l.optDisabled?!c.disabled:null===c.getAttribute("disabled"))&&(!c.parentNode.disabled||!n.nodeName(c.parentNode,"optgroup"))){if(b=n(c).val(),f)return b;g.push(b)}return g},set:function(a,b){var c,d,e=a.options,f=n.makeArray(b),g=e.length;while(g--)if(d=e[g],n.inArray(n.valHooks.option.get(d),f)>=0)try{d.selected=c=!0}catch(h){d.scrollHeight}else d.selected=!1;return c||(a.selectedIndex=-1),e}}}}),n.each(["radio","checkbox"],function(){n.valHooks[this]={set:function(a,b){return n.isArray(b)?a.checked=n.inArray(n(a).val(),b)>-1:void 0}},l.checkOn||(n.valHooks[this].get=function(a){return null===a.getAttribute("value")?"on":a.value})});var sb,tb,ub=n.expr.attrHandle,vb=/^(?:checked|selected)$/i,wb=l.getSetAttribute,xb=l.input;n.fn.extend({attr:function(a,b){return Y(this,n.attr,a,b,arguments.length>1)},removeAttr:function(a){return this.each(function(){n.removeAttr(this,a)})}}),n.extend({attr:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return"undefined"==typeof a.getAttribute?n.prop(a,b,c):(1===f&&n.isXMLDoc(a)||(b=b.toLowerCase(),e=n.attrHooks[b]||(n.expr.match.bool.test(b)?tb:sb)),void 0!==c?null===c?void n.removeAttr(a,b):e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:(a.setAttribute(b,c+""),c):e&&"get"in e&&null!==(d=e.get(a,b))?d:(d=n.find.attr(a,b),null==d?void 0:d))},attrHooks:{type:{set:function(a,b){if(!l.radioValue&&"radio"===b&&n.nodeName(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}},removeAttr:function(a,b){var c,d,e=0,f=b&&b.match(G);if(f&&1===a.nodeType)while(c=f[e++])d=n.propFix[c]||c,n.expr.match.bool.test(c)?xb&&wb||!vb.test(c)?a[d]=!1:a[n.camelCase("default-"+c)]=a[d]=!1:n.attr(a,c,""),a.removeAttribute(wb?c:d)}}),tb={set:function(a,b,c){return b===!1?n.removeAttr(a,c):xb&&wb||!vb.test(c)?a.setAttribute(!wb&&n.propFix[c]||c,c):a[n.camelCase("default-"+c)]=a[c]=!0,c}},n.each(n.expr.match.bool.source.match(/\w+/g),function(a,b){var c=ub[b]||n.find.attr;xb&&wb||!vb.test(b)?ub[b]=function(a,b,d){var e,f;return d||(f=ub[b],ub[b]=e,e=null!=c(a,b,d)?b.toLowerCase():null,ub[b]=f),e}:ub[b]=function(a,b,c){return c?void 0:a[n.camelCase("default-"+b)]?b.toLowerCase():null}}),xb&&wb||(n.attrHooks.value={set:function(a,b,c){return n.nodeName(a,"input")?void(a.defaultValue=b):sb&&sb.set(a,b,c)}}),wb||(sb={set:function(a,b,c){var d=a.getAttributeNode(c);return d||a.setAttributeNode(d=a.ownerDocument.createAttribute(c)),d.value=b+="","value"===c||b===a.getAttribute(c)?b:void 0}},ub.id=ub.name=ub.coords=function(a,b,c){var d;return c?void 0:(d=a.getAttributeNode(b))&&""!==d.value?d.value:null},n.valHooks.button={get:function(a,b){var c=a.getAttributeNode(b);return c&&c.specified?c.value:void 0},set:sb.set},n.attrHooks.contenteditable={set:function(a,b,c){sb.set(a,""===b?!1:b,c)}},n.each(["width","height"],function(a,b){n.attrHooks[b]={set:function(a,c){return""===c?(a.setAttribute(b,"auto"),c):void 0}}})),l.style||(n.attrHooks.style={get:function(a){return a.style.cssText||void 0},set:function(a,b){return a.style.cssText=b+""}});var yb=/^(?:input|select|textarea|button|object)$/i,zb=/^(?:a|area)$/i;n.fn.extend({prop:function(a,b){return Y(this,n.prop,a,b,arguments.length>1)},removeProp:function(a){return a=n.propFix[a]||a,this.each(function(){try{this[a]=void 0,delete this[a]}catch(b){}})}}),n.extend({prop:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return 1===f&&n.isXMLDoc(a)||(b=n.propFix[b]||b,e=n.propHooks[b]),void 0!==c?e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:a[b]=c:e&&"get"in e&&null!==(d=e.get(a,b))?d:a[b]},propHooks:{tabIndex:{get:function(a){var b=n.find.attr(a,"tabindex");return b?parseInt(b,10):yb.test(a.nodeName)||zb.test(a.nodeName)&&a.href?0:-1}}},propFix:{"for":"htmlFor","class":"className"}}),l.hrefNormalized||n.each(["href","src"],function(a,b){n.propHooks[b]={get:function(a){return a.getAttribute(b,4)}}}),l.optSelected||(n.propHooks.selected={get:function(a){var b=a.parentNode;return b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex),null}}),n.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){n.propFix[this.toLowerCase()]=this}),l.enctype||(n.propFix.enctype="encoding");var Ab=/[\t\r\n\f]/g;function Bb(a){return n.attr(a,"class")||""}n.fn.extend({addClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).addClass(a.call(this,b,Bb(this)))});if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=Bb(c),d=1===c.nodeType&&(" "+e+" ").replace(Ab," ")){g=0;while(f=b[g++])d.indexOf(" "+f+" ")<0&&(d+=f+" ");h=n.trim(d),e!==h&&n.attr(c,"class",h)}}return this},removeClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).removeClass(a.call(this,b,Bb(this)))});if(!arguments.length)return this.attr("class","");if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=Bb(c),d=1===c.nodeType&&(" "+e+" ").replace(Ab," ")){g=0;while(f=b[g++])while(d.indexOf(" "+f+" ")>-1)d=d.replace(" "+f+" "," ");h=n.trim(d),e!==h&&n.attr(c,"class",h)}}return this},toggleClass:function(a,b){var c=typeof a;return"boolean"==typeof b&&"string"===c?b?this.addClass(a):this.removeClass(a):n.isFunction(a)?this.each(function(c){n(this).toggleClass(a.call(this,c,Bb(this),b),b)}):this.each(function(){var b,d,e,f;if("string"===c){d=0,e=n(this),f=a.match(G)||[];while(b=f[d++])e.hasClass(b)?e.removeClass(b):e.addClass(b)}else(void 0===a||"boolean"===c)&&(b=Bb(this),b&&n._data(this,"__className__",b),n.attr(this,"class",b||a===!1?"":n._data(this,"__className__")||""))})},hasClass:function(a){var b,c,d=0;b=" "+a+" ";while(c=this[d++])if(1===c.nodeType&&(" "+Bb(c)+" ").replace(Ab," ").indexOf(b)>-1)return!0;return!1}}),n.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(a,b){n.fn[b]=function(a,c){return arguments.length>0?this.on(b,null,a,c):this.trigger(b)}}),n.fn.extend({hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}});var Cb=a.location,Db=n.now(),Eb=/\?/,Fb=/(,)|(\[|{)|(}|])|"(?:[^"\\\r\n]|\\["\\\/bfnrt]|\\u[\da-fA-F]{4})*"\s*:?|true|false|null|-?(?!0\d)\d+(?:\.\d+|)(?:[eE][+-]?\d+|)/g;n.parseJSON=function(b){if(a.JSON&&a.JSON.parse)return a.JSON.parse(b+"");var c,d=null,e=n.trim(b+"");return e&&!n.trim(e.replace(Fb,function(a,b,e,f){return c&&b&&(d=0),0===d?a:(c=e||b,d+=!f-!e,"")}))?Function("return "+e)():n.error("Invalid JSON: "+b)},n.parseXML=function(b){var c,d;if(!b||"string"!=typeof b)return null;try{a.DOMParser?(d=new a.DOMParser,c=d.parseFromString(b,"text/xml")):(c=new a.ActiveXObject("Microsoft.XMLDOM"),c.async="false",c.loadXML(b))}catch(e){c=void 0}return c&&c.documentElement&&!c.getElementsByTagName("parsererror").length||n.error("Invalid XML: "+b),c};var Gb=/#.*$/,Hb=/([?&])_=[^&]*/,Ib=/^(.*?):[ \t]*([^\r\n]*)\r?$/gm,Jb=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Kb=/^(?:GET|HEAD)$/,Lb=/^\/\//,Mb=/^([\w.+-]+:)(?:\/\/(?:[^\/?#]*@|)([^\/?#:]*)(?::(\d+)|)|)/,Nb={},Ob={},Pb="*/".concat("*"),Qb=Cb.href,Rb=Mb.exec(Qb.toLowerCase())||[];function Sb(a){return function(b,c){"string"!=typeof b&&(c=b,b="*");var d,e=0,f=b.toLowerCase().match(G)||[];if(n.isFunction(c))while(d=f[e++])"+"===d.charAt(0)?(d=d.slice(1)||"*",(a[d]=a[d]||[]).unshift(c)):(a[d]=a[d]||[]).push(c)}}function Tb(a,b,c,d){var e={},f=a===Ob;function g(h){var i;return e[h]=!0,n.each(a[h]||[],function(a,h){var j=h(b,c,d);return"string"!=typeof j||f||e[j]?f?!(i=j):void 0:(b.dataTypes.unshift(j),g(j),!1)}),i}return g(b.dataTypes[0])||!e["*"]&&g("*")}function Ub(a,b){var c,d,e=n.ajaxSettings.flatOptions||{};for(d in b)void 0!==b[d]&&((e[d]?a:c||(c={}))[d]=b[d]);return c&&n.extend(!0,a,c),a}function Vb(a,b,c){var d,e,f,g,h=a.contents,i=a.dataTypes;while("*"===i[0])i.shift(),void 0===e&&(e=a.mimeType||b.getResponseHeader("Content-Type"));if(e)for(g in h)if(h[g]&&h[g].test(e)){i.unshift(g);break}if(i[0]in c)f=i[0];else{for(g in c){if(!i[0]||a.converters[g+" "+i[0]]){f=g;break}d||(d=g)}f=f||d}return f?(f!==i[0]&&i.unshift(f),c[f]):void 0}function Wb(a,b,c,d){var e,f,g,h,i,j={},k=a.dataTypes.slice();if(k[1])for(g in a.converters)j[g.toLowerCase()]=a.converters[g];f=k.shift();while(f)if(a.responseFields[f]&&(c[a.responseFields[f]]=b),!i&&d&&a.dataFilter&&(b=a.dataFilter(b,a.dataType)),i=f,f=k.shift())if("*"===f)f=i;else if("*"!==i&&i!==f){if(g=j[i+" "+f]||j["* "+f],!g)for(e in j)if(h=e.split(" "),h[1]===f&&(g=j[i+" "+h[0]]||j["* "+h[0]])){g===!0?g=j[e]:j[e]!==!0&&(f=h[0],k.unshift(h[1]));break}if(g!==!0)if(g&&a["throws"])b=g(b);else try{b=g(b)}catch(l){return{state:"parsererror",error:g?l:"No conversion from "+i+" to "+f}}}return{state:"success",data:b}}n.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:Qb,type:"GET",isLocal:Jb.test(Rb[1]),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Pb,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":n.parseJSON,"text xml":n.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(a,b){return b?Ub(Ub(a,n.ajaxSettings),b):Ub(n.ajaxSettings,a)},ajaxPrefilter:Sb(Nb),ajaxTransport:Sb(Ob),ajax:function(b,c){"object"==typeof b&&(c=b,b=void 0),c=c||{};var d,e,f,g,h,i,j,k,l=n.ajaxSetup({},c),m=l.context||l,o=l.context&&(m.nodeType||m.jquery)?n(m):n.event,p=n.Deferred(),q=n.Callbacks("once memory"),r=l.statusCode||{},s={},t={},u=0,v="canceled",w={readyState:0,getResponseHeader:function(a){var b;if(2===u){if(!k){k={};while(b=Ib.exec(g))k[b[1].toLowerCase()]=b[2]}b=k[a.toLowerCase()]}return null==b?null:b},getAllResponseHeaders:function(){return 2===u?g:null},setRequestHeader:function(a,b){var c=a.toLowerCase();return u||(a=t[c]=t[c]||a,s[a]=b),this},overrideMimeType:function(a){return u||(l.mimeType=a),this},statusCode:function(a){var b;if(a)if(2>u)for(b in a)r[b]=[r[b],a[b]];else w.always(a[w.status]);return this},abort:function(a){var b=a||v;return j&&j.abort(b),y(0,b),this}};if(p.promise(w).complete=q.add,w.success=w.done,w.error=w.fail,l.url=((b||l.url||Qb)+"").replace(Gb,"").replace(Lb,Rb[1]+"//"),l.type=c.method||c.type||l.method||l.type,l.dataTypes=n.trim(l.dataType||"*").toLowerCase().match(G)||[""],null==l.crossDomain&&(d=Mb.exec(l.url.toLowerCase()),l.crossDomain=!(!d||d[1]===Rb[1]&&d[2]===Rb[2]&&(d[3]||("http:"===d[1]?"80":"443"))===(Rb[3]||("http:"===Rb[1]?"80":"443")))),l.data&&l.processData&&"string"!=typeof l.data&&(l.data=n.param(l.data,l.traditional)),Tb(Nb,l,c,w),2===u)return w;i=n.event&&l.global,i&&0===n.active++&&n.event.trigger("ajaxStart"),l.type=l.type.toUpperCase(),l.hasContent=!Kb.test(l.type),f=l.url,l.hasContent||(l.data&&(f=l.url+=(Eb.test(f)?"&":"?")+l.data,delete l.data),l.cache===!1&&(l.url=Hb.test(f)?f.replace(Hb,"$1_="+Db++):f+(Eb.test(f)?"&":"?")+"_="+Db++)),l.ifModified&&(n.lastModified[f]&&w.setRequestHeader("If-Modified-Since",n.lastModified[f]),n.etag[f]&&w.setRequestHeader("If-None-Match",n.etag[f])),(l.data&&l.hasContent&&l.contentType!==!1||c.contentType)&&w.setRequestHeader("Content-Type",l.contentType),w.setRequestHeader("Accept",l.dataTypes[0]&&l.accepts[l.dataTypes[0]]?l.accepts[l.dataTypes[0]]+("*"!==l.dataTypes[0]?", "+Pb+"; q=0.01":""):l.accepts["*"]);for(e in l.headers)w.setRequestHeader(e,l.headers[e]);if(l.beforeSend&&(l.beforeSend.call(m,w,l)===!1||2===u))return w.abort();v="abort";for(e in{success:1,error:1,complete:1})w[e](l[e]);if(j=Tb(Ob,l,c,w)){if(w.readyState=1,i&&o.trigger("ajaxSend",[w,l]),2===u)return w;l.async&&l.timeout>0&&(h=a.setTimeout(function(){w.abort("timeout")},l.timeout));try{u=1,j.send(s,y)}catch(x){if(!(2>u))throw x;y(-1,x)}}else y(-1,"No Transport");function y(b,c,d,e){var k,s,t,v,x,y=c;2!==u&&(u=2,h&&a.clearTimeout(h),j=void 0,g=e||"",w.readyState=b>0?4:0,k=b>=200&&300>b||304===b,d&&(v=Vb(l,w,d)),v=Wb(l,v,w,k),k?(l.ifModified&&(x=w.getResponseHeader("Last-Modified"),x&&(n.lastModified[f]=x),x=w.getResponseHeader("etag"),x&&(n.etag[f]=x)),204===b||"HEAD"===l.type?y="nocontent":304===b?y="notmodified":(y=v.state,s=v.data,t=v.error,k=!t)):(t=y,(b||!y)&&(y="error",0>b&&(b=0))),w.status=b,w.statusText=(c||y)+"",k?p.resolveWith(m,[s,y,w]):p.rejectWith(m,[w,y,t]),w.statusCode(r),r=void 0,i&&o.trigger(k?"ajaxSuccess":"ajaxError",[w,l,k?s:t]),q.fireWith(m,[w,y]),i&&(o.trigger("ajaxComplete",[w,l]),--n.active||n.event.trigger("ajaxStop")))}return w},getJSON:function(a,b,c){return n.get(a,b,c,"json")},getScript:function(a,b){return n.get(a,void 0,b,"script")}}),n.each(["get","post"],function(a,b){n[b]=function(a,c,d,e){return n.isFunction(c)&&(e=e||d,d=c,c=void 0),n.ajax(n.extend({url:a,type:b,dataType:e,data:c,success:d},n.isPlainObject(a)&&a))}}),n._evalUrl=function(a){return n.ajax({url:a,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,"throws":!0})},n.fn.extend({wrapAll:function(a){if(n.isFunction(a))return this.each(function(b){n(this).wrapAll(a.call(this,b))});if(this[0]){var b=n(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&1===a.firstChild.nodeType)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){return n.isFunction(a)?this.each(function(b){n(this).wrapInner(a.call(this,b))}):this.each(function(){var b=n(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=n.isFunction(a);return this.each(function(c){n(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){n.nodeName(this,"body")||n(this).replaceWith(this.childNodes)}).end()}});function Xb(a){return a.style&&a.style.display||n.css(a,"display")}function Yb(a){while(a&&1===a.nodeType){if("none"===Xb(a)||"hidden"===a.type)return!0;a=a.parentNode}return!1}n.expr.filters.hidden=function(a){return l.reliableHiddenOffsets()?a.offsetWidth<=0&&a.offsetHeight<=0&&!a.getClientRects().length:Yb(a)},n.expr.filters.visible=function(a){return!n.expr.filters.hidden(a)};var Zb=/%20/g,$b=/\[\]$/,_b=/\r?\n/g,ac=/^(?:submit|button|image|reset|file)$/i,bc=/^(?:input|select|textarea|keygen)/i;function cc(a,b,c,d){var e;if(n.isArray(b))n.each(b,function(b,e){c||$b.test(a)?d(a,e):cc(a+"["+("object"==typeof e&&null!=e?b:"")+"]",e,c,d)});else if(c||"object"!==n.type(b))d(a,b);else for(e in b)cc(a+"["+e+"]",b[e],c,d)}n.param=function(a,b){var c,d=[],e=function(a,b){b=n.isFunction(b)?b():null==b?"":b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};if(void 0===b&&(b=n.ajaxSettings&&n.ajaxSettings.traditional),n.isArray(a)||a.jquery&&!n.isPlainObject(a))n.each(a,function(){e(this.name,this.value)});else for(c in a)cc(c,a[c],b,e);return d.join("&").replace(Zb,"+")},n.fn.extend({serialize:function(){return n.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var a=n.prop(this,"elements");return a?n.makeArray(a):this}).filter(function(){var a=this.type;return this.name&&!n(this).is(":disabled")&&bc.test(this.nodeName)&&!ac.test(a)&&(this.checked||!Z.test(a))}).map(function(a,b){var c=n(this).val();return null==c?null:n.isArray(c)?n.map(c,function(a){return{name:b.name,value:a.replace(_b,"\r\n")}}):{name:b.name,value:c.replace(_b,"\r\n")}}).get()}}),n.ajaxSettings.xhr=void 0!==a.ActiveXObject?function(){return this.isLocal?hc():d.documentMode>8?gc():/^(get|post|head|put|delete|options)$/i.test(this.type)&&gc()||hc()}:gc;var dc=0,ec={},fc=n.ajaxSettings.xhr();a.attachEvent&&a.attachEvent("onunload",function(){for(var a in ec)ec[a](void 0,!0)}),l.cors=!!fc&&"withCredentials"in fc,fc=l.ajax=!!fc,fc&&n.ajaxTransport(function(b){if(!b.crossDomain||l.cors){var c;return{send:function(d,e){var f,g=b.xhr(),h=++dc;if(g.open(b.type,b.url,b.async,b.username,b.password),b.xhrFields)for(f in b.xhrFields)g[f]=b.xhrFields[f];b.mimeType&&g.overrideMimeType&&g.overrideMimeType(b.mimeType),b.crossDomain||d["X-Requested-With"]||(d["X-Requested-With"]="XMLHttpRequest");for(f in d)void 0!==d[f]&&g.setRequestHeader(f,d[f]+"");g.send(b.hasContent&&b.data||null),c=function(a,d){var f,i,j;if(c&&(d||4===g.readyState))if(delete ec[h],c=void 0,g.onreadystatechange=n.noop,d)4!==g.readyState&&g.abort();else{j={},f=g.status,"string"==typeof g.responseText&&(j.text=g.responseText);try{i=g.statusText}catch(k){i=""}f||!b.isLocal||b.crossDomain?1223===f&&(f=204):f=j.text?200:404}j&&e(f,i,j,g.getAllResponseHeaders())},b.async?4===g.readyState?a.setTimeout(c):g.onreadystatechange=ec[h]=c:c()},abort:function(){c&&c(void 0,!0)}}}});function gc(){try{return new a.XMLHttpRequest}catch(b){}}function hc(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}n.ajaxPrefilter(function(a){a.crossDomain&&(a.contents.script=!1)}),n.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(a){return n.globalEval(a),a}}}),n.ajaxPrefilter("script",function(a){void 0===a.cache&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),n.ajaxTransport("script",function(a){if(a.crossDomain){var b,c=d.head||n("head")[0]||d.documentElement;return{send:function(e,f){b=d.createElement("script"),b.async=!0,a.scriptCharset&&(b.charset=a.scriptCharset),b.src=a.url,b.onload=b.onreadystatechange=function(a,c){(c||!b.readyState||/loaded|complete/.test(b.readyState))&&(b.onload=b.onreadystatechange=null,b.parentNode&&b.parentNode.removeChild(b),b=null,c||f(200,"success"))},c.insertBefore(b,c.firstChild)},abort:function(){b&&b.onload(void 0,!0)}}}});var ic=[],jc=/(=)\?(?=&|$)|\?\?/;n.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var a=ic.pop()||n.expando+"_"+Db++;return this[a]=!0,a}}),n.ajaxPrefilter("json jsonp",function(b,c,d){var e,f,g,h=b.jsonp!==!1&&(jc.test(b.url)?"url":"string"==typeof b.data&&0===(b.contentType||"").indexOf("application/x-www-form-urlencoded")&&jc.test(b.data)&&"data");return h||"jsonp"===b.dataTypes[0]?(e=b.jsonpCallback=n.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,h?b[h]=b[h].replace(jc,"$1"+e):b.jsonp!==!1&&(b.url+=(Eb.test(b.url)?"&":"?")+b.jsonp+"="+e),b.converters["script json"]=function(){return g||n.error(e+" was not called"),g[0]},b.dataTypes[0]="json",f=a[e],a[e]=function(){g=arguments},d.always(function(){void 0===f?n(a).removeProp(e):a[e]=f,b[e]&&(b.jsonpCallback=c.jsonpCallback,ic.push(e)),g&&n.isFunction(f)&&f(g[0]),g=f=void 0}),"script"):void 0}),l.createHTMLDocument=function(){if(!d.implementation.createHTMLDocument)return!1;var a=d.implementation.createHTMLDocument("");return a.body.innerHTML="<form></form><form></form>",2===a.body.childNodes.length}(),n.parseHTML=function(a,b,c){if(!a||"string"!=typeof a)return null;"boolean"==typeof b&&(c=b,b=!1),b=b||(l.createHTMLDocument?d.implementation.createHTMLDocument(""):d);var e=x.exec(a),f=!c&&[];return e?[b.createElement(e[1])]:(e=ja([a],b,f),f&&f.length&&n(f).remove(),n.merge([],e.childNodes))};var kc=n.fn.load;n.fn.load=function(a,b,c){if("string"!=typeof a&&kc)return kc.apply(this,arguments);var d,e,f,g=this,h=a.indexOf(" ");return h>-1&&(d=n.trim(a.slice(h,a.length)),a=a.slice(0,h)),n.isFunction(b)?(c=b,b=void 0):b&&"object"==typeof b&&(e="POST"),g.length>0&&n.ajax({url:a,type:e||"GET",dataType:"html",data:b}).done(function(a){f=arguments,g.html(d?n("<div>").append(n.parseHTML(a)).find(d):a)}).always(c&&function(a,b){g.each(function(){c.apply(g,f||[a.responseText,b,a])})}),this},n.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(a,b){n.fn[b]=function(a){return this.on(b,a)}}),n.expr.filters.animated=function(a){return n.grep(n.timers,function(b){return a===b.elem}).length};function lc(a){return n.isWindow(a)?a:9===a.nodeType?a.defaultView||a.parentWindow:!1}n.offset={setOffset:function(a,b,c){var d,e,f,g,h,i,j,k=n.css(a,"position"),l=n(a),m={};"static"===k&&(a.style.position="relative"),h=l.offset(),f=n.css(a,"top"),i=n.css(a,"left"),j=("absolute"===k||"fixed"===k)&&n.inArray("auto",[f,i])>-1,j?(d=l.position(),g=d.top,e=d.left):(g=parseFloat(f)||0,e=parseFloat(i)||0),n.isFunction(b)&&(b=b.call(a,c,n.extend({},h))),null!=b.top&&(m.top=b.top-h.top+g),null!=b.left&&(m.left=b.left-h.left+e),"using"in b?b.using.call(a,m):l.css(m)}},n.fn.extend({offset:function(a){if(arguments.length)return void 0===a?this:this.each(function(b){n.offset.setOffset(this,a,b)});var b,c,d={top:0,left:0},e=this[0],f=e&&e.ownerDocument;if(f)return b=f.documentElement,n.contains(b,e)?("undefined"!=typeof e.getBoundingClientRect&&(d=e.getBoundingClientRect()),c=lc(f),{top:d.top+(c.pageYOffset||b.scrollTop)-(b.clientTop||0),left:d.left+(c.pageXOffset||b.scrollLeft)-(b.clientLeft||0)}):d},position:function(){if(this[0]){var a,b,c={top:0,left:0},d=this[0];return"fixed"===n.css(d,"position")?b=d.getBoundingClientRect():(a=this.offsetParent(),b=this.offset(),n.nodeName(a[0],"html")||(c=a.offset()),c.top+=n.css(a[0],"borderTopWidth",!0)-a.scrollTop(),c.left+=n.css(a[0],"borderLeftWidth",!0)-a.scrollLeft()),{top:b.top-c.top-n.css(d,"marginTop",!0),left:b.left-c.left-n.css(d,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var a=this.offsetParent;while(a&&!n.nodeName(a,"html")&&"static"===n.css(a,"position"))a=a.offsetParent;return a||Qa})}}),n.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(a,b){var c=/Y/.test(b);n.fn[a]=function(d){return Y(this,function(a,d,e){var f=lc(a);return void 0===e?f?b in f?f[b]:f.document.documentElement[d]:a[d]:void(f?f.scrollTo(c?n(f).scrollLeft():e,c?e:n(f).scrollTop()):a[d]=e)},a,d,arguments.length,null)}}),n.each(["top","left"],function(a,b){
5
+ n.cssHooks[b]=Ua(l.pixelPosition,function(a,c){return c?(c=Sa(a,b),Oa.test(c)?n(a).position()[b]+"px":c):void 0})}),n.each({Height:"height",Width:"width"},function(a,b){n.each({padding:"inner"+a,content:b,"":"outer"+a},function(c,d){n.fn[d]=function(d,e){var f=arguments.length&&(c||"boolean"!=typeof d),g=c||(d===!0||e===!0?"margin":"border");return Y(this,function(b,c,d){var e;return n.isWindow(b)?b.document.documentElement["client"+a]:9===b.nodeType?(e=b.documentElement,Math.max(b.body["scroll"+a],e["scroll"+a],b.body["offset"+a],e["offset"+a],e["client"+a])):void 0===d?n.css(b,c,g):n.style(b,c,d,g)},b,f?d:void 0,f,null)}})}),n.fn.extend({bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return 1===arguments.length?this.off(a,"**"):this.off(b,a||"**",c)}}),n.fn.size=function(){return this.length},n.fn.andSelf=n.fn.addBack,"function"==typeof define&&define.amd&&define("jquery",[],function(){return n});var mc=a.jQuery,nc=a.$;return n.noConflict=function(b){return a.$===n&&(a.$=nc),b&&a.jQuery===n&&(a.jQuery=mc),n},b||(a.jQuery=a.$=n),n});
package.xml ADDED
@@ -0,0 +1,18 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <package>
3
+ <name>get4cast_salesforecast</name>
4
+ <version>1.1.0</version>
5
+ <stability>stable</stability>
6
+ <license uri="https://www.get4cast.com/sales-forecast-for-magento/license">Get4Cast</license>
7
+ <channel>community</channel>
8
+ <extends/>
9
+ <summary>Sales Forecast by Get4Cast</summary>
10
+ <description>Anticipate outcomes and decision-making using Predictive Analytics. This extension forecasts sales, orders, inventory, revenue and cancellation probability. Be one step ahead.</description>
11
+ <notes>pt_BR locale</notes>
12
+ <authors><author><name>Get4Cast</name><user>Get4Cast</user><email>hello@get4cast.com</email></author></authors>
13
+ <date>2016-07-12</date>
14
+ <time>16:28:14</time>
15
+ <contents><target name="magecommunity"><dir name="Get4Cast"><dir name="SalesForecast"><dir name="Block"><dir name="Adminhtml"><file name="Account.php" hash="55882c10fb27f3f468e003e39b9e310a"/><dir name="Forecast"><dir name="Edit"><file name="Form.php" hash="1d6cb5ead427ec7c8132f3eec3d3a545"/></dir><file name="Edit.php" hash="339a7df28ab8b567c2cb9f84f2d32d48"/><file name="Grid.php" hash="20bfc2add945ef4ae85c3f1bfc574c24"/></dir><file name="Forecast.php" hash="305cf21ba3760fd9b67b0ab9dbbbb9ce"/><file name="Link.php" hash="ed7e2d77241da304ccc1d1dbfe26d0a6"/><file name="Translate.php" hash="902e087a6ae5a86f466e6f196778abff"/><file name="Url.php" hash="19fd9218cdfdfd5d5bd3c0b31bec0d65"/></dir></dir><dir name="Helper"><file name="Data.php" hash="35a3c85242db4d493c449ecedbfcf72f"/></dir><dir name="Model"><file name="Apiclient.php" hash="6b573c83d3a40f8f4d4628af23d6eac0"/><file name="Collect.php" hash="8ff7330fe7081312e6e05e1e1237a628"/><file name="Config.php" hash="7b6b632f01e216524cabe1ea9182c6a1"/><file name="Forecast.php" hash="b20a234cda4dac8a78a60a8bd61a883f"/><file name="Observer.php" hash="344dc7705f9b67f5dad2d32c49ccd0ce"/><dir name="Resource"><dir name="Config"><file name="Collection.php" hash="3da7570f50ec6576af694a31487b5818"/></dir><file name="Config.php" hash="d77d907aa40179a1c8733265b0990587"/><dir name="Forecast"><file name="Collection.php" hash="cb43b14d6f7d22d22848abfc271cc68b"/></dir><file name="Forecast.php" hash="ccbadf0663f3437a5f795f50c4668ae5"/></dir></dir><dir name="controllers"><dir name="Adminhtml"><file name="ForecastController.php" hash="74e329b19a241166730d2ce73696f436"/></dir></dir><dir name="data"><dir name="get4cast_salesforecast_setup"><file name="data-install-1.0.0.php" hash="c7460f229ffb7e2ff5c897ad909c5e17"/></dir></dir><dir name="etc"><file name="adminhtml.xml" hash="5413e7a5b2ce99d5738b808156159c41"/><file name="config.xml" hash="d780c7606386b8e8302b582702d5ea7c"/></dir><dir name="sql"><dir name="get4cast_salesforecast_setup"><file name="install-1.0.0.php" hash="c223b6c4b84e9213bffe535ba117ab2e"/></dir></dir></dir></dir></target><target name="magedesign"><dir name="adminhtml"><dir name="default"><dir name="default"><dir name="layout"><file name="get4cast_salesforecast.xml" hash="7c5b1db972b48a0f0042f69e640909f1"/></dir></dir></dir></dir></target><target name="mageetc"><dir name="modules"><file name="Get4Cast_SalesForecast.xml" hash="480957f25e9f09325c8b30d02dd10799"/></dir></target><target name="magelocale"><dir><dir name="pt_BR"><file name="Get4Cast_SalesForecast.csv" hash="59ef03bfafc95d4c8854b8dc52652b2f"/></dir></dir></target><target name="mageweb"><dir name="js"><dir name="get4cast"><dir name="salesforecast"><file name="forecast.js" hash="1f64d88f390543d71349b72512ff9ff4"/><file name="jquery-1.12.0.min.js" hash="cbb11b58473b2d672f4ed53abbb67336"/></dir></dir></dir></target></contents>
16
+ <compatible/>
17
+ <dependencies><required><php><min>5.1.0</min><max>6.0.0</max></php></required></dependencies>
18
+ </package>