2 if(!defined('sugarEntry') || !sugarEntry) die('Not A Valid Entry Point');
3 /*********************************************************************************
4 * SugarCRM Community Edition is a customer relationship management program developed by
5 * SugarCRM, Inc. Copyright (C) 2004-2013 SugarCRM Inc.
7 * This program is free software; you can redistribute it and/or modify it under
8 * the terms of the GNU Affero General Public License version 3 as published by the
9 * Free Software Foundation with the addition of the following permission added
10 * to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
11 * IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
12 * OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
14 * This program is distributed in the hope that it will be useful, but WITHOUT
15 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16 * FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
19 * You should have received a copy of the GNU Affero General Public License along with
20 * this program; if not, see http://www.gnu.org/licenses or write to the Free
21 * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
24 * You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
25 * SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
27 * The interactive user interfaces in modified source and object code versions
28 * of this program must display Appropriate Legal Notices, as required under
29 * Section 5 of the GNU Affero General Public License version 3.
31 * In accordance with Section 7(b) of the GNU Affero General Public License version 3,
32 * these Appropriate Legal Notices must retain the display of the "Powered by
33 * SugarCRM" logo. If the display of the logo is not reasonably feasible for
34 * technical reasons, the Appropriate Legal Notices must display the words
35 * "Powered by SugarCRM".
36 ********************************************************************************/
42 require_once('include/utils/activity_utils.php');
43 require_once('modules/Calendar/CalendarUtils.php');
44 require_once('modules/Calendar/CalendarActivity.php');
49 public $view = 'week'; // current view
50 public $style; // calendar style (basic or advanced)
51 public $dashlet = false; // if is displayed in dashlet
52 public $date_time; // current date
54 public $show_tasks = true;
55 public $show_calls = true;
56 public $enable_repeat = true;
58 public $time_step = 60; // time step of each slot in minutes
60 public $acts_arr = array(); // Array of activities objects
61 public $items = array(); // Array of activities data to be displayed
62 public $shared_ids = array(); // ids of users for shared view
65 public $cells_per_day; // entire 24h day count of slots
66 public $grid_start_ts; // start timestamp of calendar grid
68 public $day_start_time; // working day start time in format '11:00'
69 public $day_end_time; // working day end time in format '11:00'
70 public $scroll_slot; // first slot of working day
71 public $celcount; // count of slots in a working day
74 * @var bool $print Whether is print mode.
76 private $print = false;
81 * @param array $time_arr
83 function __construct($view = "day", $time_arr = array()){
84 global $current_user, $timedate;
88 if(!in_array($this->view,array('day','week','month','year','shared')))
92 if(!empty($_REQUEST['day']))
93 $_REQUEST['day'] = intval($_REQUEST['day']);
94 if(!empty($_REQUEST['month']))
95 $_REQUEST['month'] = intval($_REQUEST['month']);
97 if (!empty($_REQUEST['day']))
98 $date_arr['day'] = $_REQUEST['day'];
99 if (!empty($_REQUEST['month']))
100 $date_arr['month'] = $_REQUEST['month'];
101 if (!empty($_REQUEST['week']))
102 $date_arr['week'] = $_REQUEST['week'];
104 if (!empty($_REQUEST['year'])){
105 if ($_REQUEST['year'] > 2037 || $_REQUEST['year'] < 1970){
106 print("Sorry, calendar cannot handle the year you requested");
107 print("<br>Year must be between 1970 and 2037");
110 $date_arr['year'] = $_REQUEST['year'];
113 if(empty($_REQUEST['day']))
114 $_REQUEST['day'] = "";
115 if(empty($_REQUEST['week']))
116 $_REQUEST['week'] = "";
117 if(empty($_REQUEST['month']))
118 $_REQUEST['month'] = "";
119 if(empty($_REQUEST['year']))
120 $_REQUEST['year'] = "";
122 // if date is not set in request use current date
123 if(empty($date_arr) || !isset($date_arr['year']) || !isset($date_arr['month']) || !isset($date_arr['day']) ){
124 $today = $timedate->getNow(true);
126 'year' => $today->year,
127 'month' => $today->month,
128 'day' => $today->day,
132 $current_date_db = $date_arr['year']."-".str_pad($date_arr['month'],2,"0",STR_PAD_LEFT)."-".str_pad($date_arr['day'],2,"0",STR_PAD_LEFT);
133 $this->date_time = $GLOBALS['timedate']->fromString($current_date_db);
135 $this->show_tasks = $current_user->getPreference('show_tasks');
136 if(is_null($this->show_tasks))
137 $this->show_tasks = SugarConfig::getInstance()->get('calendar.show_tasks_by_default',true);
138 $this->show_calls = $current_user->getPreference('show_calls');
139 if(is_null($this->show_calls))
140 $this->show_calls = SugarConfig::getInstance()->get('calendar.show_calls_by_default',true);
142 $this->enable_repeat = SugarConfig::getInstance()->get('calendar.enable_repeat',true);
144 if(in_array($this->view,array('month','year'))){
145 $this->style = "basic";
147 $displayTimeslots = $GLOBALS['current_user']->getPreference('calendar_display_timeslots');
148 if (is_null($displayTimeslots)) {
149 $displayTimeslots = SugarConfig::getInstance()->get('calendar.display_timeslots', true);
151 if ($displayTimeslots) {
152 $this->style = "advanced";
154 $this->style = "basic";
158 $this->day_start_time = $current_user->getPreference('day_start_time');
159 if(is_null($this->day_start_time))
160 $this->day_start_time = SugarConfig::getInstance()->get('calendar.default_day_start',"08:00");
161 $this->day_end_time = $current_user->getPreference('day_end_time');
162 if(is_null($this->day_end_time))
163 $this->day_end_time = SugarConfig::getInstance()->get('calendar.default_day_end',"19:00");
165 if($this->view == "day"){
166 $this->time_step = SugarConfig::getInstance()->get('calendar.day_timestep',15);
167 }else if($this->view == "week" || $this->view == "shared"){
168 $this->time_step = SugarConfig::getInstance()->get('calendar.week_timestep',30);
169 }else if($this->view == "month"){
170 $this->time_step = SugarConfig::getInstance()->get('calendar.month_timestep',60);
172 $this->time_step = 60;
174 $this->cells_per_day = 24 * (60 / $this->time_step);
175 $this->calculate_grid_start_ts();
176 $this->calculate_day_range();
180 * Load activities data to array
182 public function load_activities(){
183 $field_list = CalendarUtils::get_fields();
185 foreach($this->acts_arr as $user_id => $acts){
186 foreach($acts as $act){
189 $item['user_id'] = $user_id;
190 $item['module_name'] = $act->sugar_bean->module_dir;
191 $item['type'] = strtolower($act->sugar_bean->object_name);
192 $item['assigned_user_id'] = $act->sugar_bean->assigned_user_id;
193 $item['record'] = $act->sugar_bean->id;
194 $item['name'] = $act->sugar_bean->name;
196 if(isset($act->sugar_bean->duration_hours)){
197 $item['duration_hours'] = $act->sugar_bean->duration_hours;
198 $item['duration_minutes'] = $act->sugar_bean->duration_minutes;
204 if($act->sugar_bean->ACLAccess('DetailView'))
206 if($act->sugar_bean->ACLAccess('Save'))
209 if(empty($act->sugar_bean->id)){
214 if(!empty($act->sugar_bean->repeat_parent_id))
215 $item['repeat_parent_id'] = $act->sugar_bean->repeat_parent_id;
217 if($item['detail'] == 1){
218 if(isset($field_list[$item['module_name']])){
219 foreach($field_list[$item['module_name']] as $field){
220 if(!isset($item[$field]) && isset($act->sugar_bean->$field)){
221 $item[$field] = $act->sugar_bean->$field;
222 if(empty($item[$field]))
229 if (!empty($act->sugar_bean->parent_type) && !empty($act->sugar_bean->parent_id)) {
230 $focus = BeanFactory::getBean($act->sugar_bean->parent_type, $act->sugar_bean->parent_id);
231 $item['related_to'] = $focus->name;
234 if(!isset($item['duration_hours']) || empty($item['duration_hours']))
235 $item['duration_hours'] = 0;
236 if(!isset($item['duration_minutes']) || empty($item['duration_minutes']))
237 $item['duration_minutes'] = 0;
239 $item = array_merge($item,CalendarUtils::get_time_data($act->sugar_bean));
241 $this->items[] = $item;
247 * initialize ids of shared users
249 public function init_shared(){
250 global $current_user;
253 $user_ids = $current_user->getPreference('shared_ids');
254 if(!empty($user_ids) && count($user_ids) != 0 && !isset($_REQUEST['shared_ids'])) {
255 $this->shared_ids = $user_ids;
256 }else if(isset($_REQUEST['shared_ids']) && count($_REQUEST['shared_ids']) > 0){
257 $this->shared_ids = $_REQUEST['shared_ids'];
258 $current_user->setPreference('shared_ids', $_REQUEST['shared_ids']);
260 $this->shared_ids = array($current_user->id);
265 * Calculate timestamp the calendar grid should be started from
267 protected function calculate_grid_start_ts(){
269 if($this->view == "week" || $this->view == "shared"){
270 $week_start = CalendarUtils::get_first_day_of_week($this->date_time);
271 $this->grid_start_ts = $week_start->format('U') + $week_start->getOffset();
272 }else if($this->view == "month"){
273 $month_start = $this->date_time->get_day_by_index_this_month(0);
274 $week_start = CalendarUtils::get_first_day_of_week($month_start);
275 $this->grid_start_ts = $week_start->format('U') + $week_start->getOffset(); // convert to timestamp, ignore tz
276 }else if($this->view == "day"){
277 $this->grid_start_ts = $this->date_time->format('U') + $this->date_time->getOffset();
282 * calculate count of timeslots per visible day, calculates day start and day end in minutes
284 function calculate_day_range(){
286 list($hour_start,$minute_start) = explode(":",$this->day_start_time);
287 list($hour_end,$minute_end) = explode(":",$this->day_end_time);
288 $this->scroll_slot = intval($hour_start * (60 / $this->time_step) + ($minute_start / $this->time_step));
289 $this->celcount = (($hour_end * 60 + $minute_end) - ($hour_start * 60 + $minute_start)) / $this->time_step;
293 * loads array of objects
294 * @param User $user user object
295 * @param string $type
297 public function add_activities($user,$type='sugar'){
299 $start_date_time = $this->date_time;
300 if($this->view == 'week' || $this->view == 'shared'){
301 $start_date_time = CalendarUtils::get_first_day_of_week($this->date_time);
302 $end_date_time = $start_date_time->get("+7 days");
303 }else if($this->view == 'month'){
304 $start_date_time = $this->date_time->get_day_by_index_this_month(0);
305 $end_date_time = $start_date_time->get("+".$start_date_time->format('t')." days");
306 $start_date_time = CalendarUtils::get_first_day_of_week($start_date_time);
307 $end_date_time = CalendarUtils::get_first_day_of_week($end_date_time)->get("+7 days");
309 $end_date_time = $this->date_time->get("+1 day");
312 $start_date_time = $start_date_time->get("-5 days"); // 5 days step back to fetch multi-day activities that
316 $acts_arr = CalendarActivity::get_freebusy_activities($user, $start_date_time, $end_date_time);
318 $acts_arr = CalendarActivity::get_activities($user->id, $this->show_tasks, $start_date_time, $end_date_time, $this->view,$this->show_calls);
321 $this->acts_arr[$user->id] = $acts_arr;
325 * Get date string of next or previous calendar grid
326 * @param string $direction next or previous
329 public function get_neighbor_date_str($direction){
330 if($direction == "previous")
335 if($this->view == 'month'){
336 $day = $this->date_time->get_day_by_index_this_month(0)->get($sign."1 month")->get_day_begin(1);
337 }else if($this->view == 'week' || $this->view == 'shared'){
338 $day = CalendarUtils::get_first_day_of_week($this->date_time);
339 $day = $day->get($sign."7 days");
340 }else if($this->view == 'day'){
341 $day = $this->date_time->get($sign."1 day")->get_day_begin();
342 }else if($this->view == 'year'){
343 $day = $this->date_time->get($sign."1 year")->get_day_begin();
345 return "get_neighbor_date_str: notdefined for this view";
347 return $day->get_date_str();
350 public function setPrint($print)
352 $this->print = $print;
355 public function isPrint()