mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-01 11:38:54 +01:00
314 lines
9.9 KiB
PHP
314 lines
9.9 KiB
PHP
<?php
|
|
/**
|
|
* Egroupware Weather widget
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
* @package home
|
|
* @subpackage portlet
|
|
* @link http://www.egroupware.org
|
|
* @author Nathan Gray
|
|
* @version $Id$
|
|
*/
|
|
|
|
use EGroupware\Api;
|
|
use EGroupware\Api\Framework;
|
|
use EGroupware\Api\Etemplate;
|
|
|
|
/**
|
|
* Widget displaying the weather
|
|
*
|
|
* This widget displays more or less data depending on the portlet size using
|
|
* a combination of the disabled attribute in the template, and unsetting
|
|
* things to fit. It also uses some CSS to make sure things fit according to
|
|
* the grid size.
|
|
*
|
|
* We use openweathermap.org as a data source.
|
|
*/
|
|
class home_weather_portlet extends home_portlet
|
|
{
|
|
|
|
const API_URL = "http://api.openweathermap.org/data/2.5/";
|
|
const API_KEY = '45484f039c5caa14d31aefe7f5514292';
|
|
const CACHE_TIME = 3600; // Cache weather for an hour
|
|
|
|
/**
|
|
* Context for this portlet
|
|
*/
|
|
public function __construct(Array &$context = array(), &$need_reload = false)
|
|
{
|
|
// City not set for new widgets created via context menu
|
|
if(!$context['city'])
|
|
{
|
|
// Set initial size to 3x2, default is too small
|
|
$context['width'] = 3;
|
|
$context['height'] = 2;
|
|
}
|
|
|
|
$need_reload = true;
|
|
|
|
$this->context = $context;
|
|
}
|
|
|
|
public function get_value()
|
|
{
|
|
$id = $this->context['id'];
|
|
$content = array();
|
|
$request = array(
|
|
'units' => $this->context['units'] ?: 'metric',
|
|
'lang' => $GLOBALS['egw_info']['user']['preferences']['common']['lang'],
|
|
// Always get (& cache) 10 days, we'll cut down later
|
|
'cnt' => 10
|
|
);
|
|
if($this->context['city_id'])
|
|
{
|
|
$request['id'] = $this->context['city_id'];
|
|
$content += $this->get_weather($request);
|
|
}
|
|
elseif($this->context['city'])
|
|
{
|
|
$request['q'] = $this->context['city'];
|
|
$content += $this->get_weather($request);
|
|
}
|
|
elseif($this->context['position'])
|
|
{
|
|
list($request['lat'], $request['lon']) = explode(',', $this->context['position']);
|
|
$content += $this->get_weather($request);
|
|
}
|
|
|
|
// Caching is best done by city ID, so store that
|
|
if($content['city_id'] && (!$this->context['city_id'] || $content['city_id'] != $this->context['city_id']))
|
|
{
|
|
|
|
$arr = $GLOBALS['egw']->preferences->read_repository();
|
|
$portlets = $arr['home'];
|
|
|
|
// Save updated Api\Preferences
|
|
$portlets[$id]['city_id'] = $content['city_id'];
|
|
$this->context['city'] = $portlets[$id]['city'] = $content['settings']['city'] =
|
|
$content['settings']['title'] = $content['city'] = is_array($content['city']) ? $content['city']['name'] : $content['city'];
|
|
unset($portlets[$id]['position']);
|
|
$GLOBALS['egw']->preferences->add('home', $id, $portlets[$id]);
|
|
$GLOBALS['egw']->preferences->save_repository(True);
|
|
}
|
|
|
|
// Direct to full forecast page
|
|
$content['attribution'] = 'http://openweathermap.org/city/' . $content['city_id'];
|
|
|
|
return [
|
|
'color' => $this->context['color'],
|
|
'city' => $this->context['city'],
|
|
'display' => $this->context['display'],
|
|
'weather' => $content
|
|
];
|
|
}
|
|
|
|
public function exec($id = null, Etemplate &$etemplate = null)
|
|
{
|
|
return ['settings' => $this->get_value()];
|
|
}
|
|
|
|
/**
|
|
* Fetch weather data from provider openweathermap.org
|
|
*
|
|
* @see http://openweathermap.org/api
|
|
* @param array $query
|
|
*/
|
|
public function get_weather(Array $query, $api_url = '')
|
|
{
|
|
static $debug = false;
|
|
if(!$api_url)
|
|
{
|
|
$api_url = self::API_URL . '/weather?';
|
|
}
|
|
if(self::API_KEY)
|
|
{
|
|
$query['APPID'] = self::API_KEY;
|
|
}
|
|
$data = Api\Cache::getTree('home', json_encode($query), function($query)
|
|
{
|
|
$debug = false;
|
|
if($debug) error_log('Fetching fresh data from ' . static::API_URL);
|
|
|
|
$url = static::API_URL.'forecast/daily?'. http_build_query($query);
|
|
$forecast = file_get_contents($url);
|
|
|
|
$url = static::API_URL.'weather?'. http_build_query($query);
|
|
$current = file_get_contents($url);
|
|
if($debug) error_log(__METHOD__ . ' current: ' . $current);
|
|
|
|
return array_merge(array('current' => json_decode($current, true)), (array)json_decode($forecast, true));
|
|
}, array($query), self::CACHE_TIME);
|
|
|
|
// Some sample data, if you need to test
|
|
//error_log('Using hardcoded data instead of ' . $api_url . http_build_query($query));
|
|
//$weather = '{"coord":{"lon":-114.05,"lat":53.23},"sys":{"message":0.3098,"country":"Canada","sunrise":1420559329,"sunset":1420587344},"weather":[{"id":802,"main":"Clouds","description":"scattered clouds","icon":"03n"}],"base":"cmc stations","main":{"temp":-21.414,"temp_min":-21.414,"temp_max":-21.414,"pressure":947.79,"sea_level":1050.73,"grnd_level":947.79,"humidity":69},"wind":{"speed":3,"deg":273.5},"clouds":{"all":32},"dt":1420502430,"id":0,"name":"Thorsby","cod":200}';
|
|
//$weather = '{"cod":"200","message":0.1743,"city":{"id":"5978233","name":"Thorsby","coord":{"lon":-114.051,"lat":53.2285},"country":"Canada","population":0},"cnt":6,"list":[{"dt":1420743600,"temp":{"day":-17.49,"min":-27.86,"max":-16.38,"night":-27.86,"eve":-19.91,"morn":-16.77},"pressure":966.21,"humidity":66,"weather":[{"id":800,"main":"Clear","description":"sky is clear","icon":"01d"}],"speed":6.91,"deg":312,"clouds":0,"snow":0.02},{"dt":1420830000,"temp":{"day":-24.86,"min":-29.71,"max":-17.98,"night":-18.31,"eve":-18.32,"morn":-29.51},"pressure":948.46,"humidity":54,"weather":[{"id":801,"main":"Clouds","description":"few clouds","icon":"02d"}],"speed":3.21,"deg":166,"clouds":20},{"dt":1420916400,"temp":{"day":-18.51,"min":-25.57,"max":-17.86,"night":-23.83,"eve":-23.91,"morn":-19.28},"pressure":947.22,"humidity":74,"weather":[{"id":802,"main":"Clouds","description":"scattered clouds","icon":"03d"}],"speed":1.97,"deg":314,"clouds":48},{"dt":1421002800,"temp":{"day":-26.69,"min":-29.86,"max":-20.19,"night":-21.82,"eve":-24.66,"morn":-28.85},"pressure":951.93,"humidity":22,"weather":[{"id":800,"main":"Clear","description":"sky is clear","icon":"02d"}],"speed":1.36,"deg":196,"clouds":8},{"dt":1421089200,"temp":{"day":0.9,"min":-8.24,"max":0.9,"night":-4.99,"eve":-0.21,"morn":-8.24},"pressure":929.31,"humidity":0,"weather":[{"id":800,"main":"Clear","description":"sky is clear","icon":"01d"}],"speed":6.01,"deg":302,"clouds":5,"snow":0},{"dt":1421175600,"temp":{"day":-1.53,"min":-6.7,"max":2.23,"night":-3.65,"eve":2.23,"morn":-6.7},"pressure":934.51,"humidity":0,"weather":[{"id":800,"main":"Clear","description":"sky is clear","icon":"01d"}],"speed":3.9,"deg":201,"clouds":78}]}';
|
|
|
|
if($debug)
|
|
{
|
|
error_log(__METHOD__ .' weather info:');
|
|
foreach(array_keys($data) as $key)
|
|
{
|
|
error_log($key . ': ' .array2string($data[$key]));
|
|
}
|
|
}
|
|
if(is_string($data['message']))
|
|
{
|
|
$desc = $this->get_description();
|
|
Framework::message($desc['displayName'] . ': ' . $desc['title'] . "\n".$data['message'], 'warning');
|
|
return array();
|
|
}
|
|
|
|
if(array_key_exists('city', $data))
|
|
{
|
|
$data['city_id'] = $data['city']['id'];
|
|
}
|
|
elseif ($data['city'])
|
|
{
|
|
$data['city_id'] = $data['id'];
|
|
}
|
|
if($data['list'])
|
|
{
|
|
$massage =& $data['list'];
|
|
|
|
for($i = 0; $i < count($data['list']); $i++)
|
|
{
|
|
$forecast =& $massage[$i];
|
|
$forecast['day'] = Api\DateTime::to($forecast['dt'], 'l');
|
|
self::format_forecast($forecast);
|
|
}
|
|
// Chop data to fit into portlet
|
|
for($i; $i < count($massage); $i++)
|
|
{
|
|
unset($massage[$i]);
|
|
}
|
|
}
|
|
if($data['current'] && is_array($data['current']))
|
|
{
|
|
// Current weather
|
|
$data['current']['temp'] = $data['current']['main'];
|
|
self::format_forecast($data['current']);
|
|
}
|
|
|
|
|
|
if ($data['list'])
|
|
{
|
|
$data['current']['temp'] = array_merge($data['current']['temp'],$data['list'][0]['temp']);
|
|
}
|
|
return $data;
|
|
}
|
|
|
|
/**
|
|
* Format weather to our liking
|
|
*/
|
|
protected static function format_forecast(&$data)
|
|
{
|
|
$weather =& $data['weather'] ? $data['weather'] : $data;
|
|
$temp =& $data['temp'] ? $data['temp'] : $data;
|
|
|
|
// Find icon
|
|
if(is_array($weather))
|
|
{
|
|
foreach($weather as &$w)
|
|
{
|
|
$w['icon'] = static::get_icon($w);
|
|
}
|
|
}
|
|
|
|
// Round
|
|
foreach(array('temp', 'temp_min', 'temp_max', 'min', 'max') as $temp_name)
|
|
{
|
|
if(array_key_exists($temp_name, $temp))
|
|
{
|
|
$temp[$temp_name] = '' . round($temp[$temp_name]);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get an icon to represent the forecast
|
|
*
|
|
* We use icon names from shoelace
|
|
* @param $weather
|
|
* @return string
|
|
*/
|
|
protected static function get_icon(&$weather)
|
|
{
|
|
$icon = "question-lg";
|
|
switch(strtolower($weather['main']))
|
|
{
|
|
case 'clear' :
|
|
$icon = 'sun';
|
|
break;
|
|
case 'clouds':
|
|
$icon = strtolower($weather['main']);
|
|
if($weather['description'] == 'broken clouds')
|
|
{
|
|
$icon = 'cloud-sun';
|
|
}
|
|
break;
|
|
case 'rain':
|
|
$icon = 'cloud-rain';
|
|
if(str_contains($weather['description'], 'heavy'))
|
|
{
|
|
$icon = 'cloud-rain-heavy';
|
|
}
|
|
elseif(str_contains($weather['description'], 'light'))
|
|
{
|
|
$icon = 'cloud-drizzle';
|
|
|
|
}
|
|
break;
|
|
default:
|
|
$icon = strtolower($weather['main']);
|
|
}
|
|
return $icon;
|
|
}
|
|
|
|
public function get_actions()
|
|
{
|
|
$actions = array();
|
|
return $actions;
|
|
}
|
|
|
|
/**
|
|
* Return a list of settings to customize the portlet.
|
|
*
|
|
* Settings should be in the same style as for preferences. It is OK to return an empty array
|
|
* for no customizable settings.
|
|
*
|
|
* These should be already translated, no further translation will be done.
|
|
*
|
|
* @see preferences/inc/class.preferences_settings.inc.php
|
|
* @return Array of settings. Each setting should have the following keys:
|
|
* - name: Internal reference
|
|
* - type: Widget type for editing
|
|
* - label: Human name
|
|
* - help: Description of the setting, and what it does
|
|
* - default: Default value, for when it's not set yet
|
|
*/
|
|
public function get_properties()
|
|
{
|
|
$properties = parent::get_properties();
|
|
|
|
$properties[] = array(
|
|
'name' => 'city',
|
|
'type' => 'textbox',
|
|
'label' => lang('Location'),
|
|
);
|
|
return $properties;
|
|
}
|
|
|
|
public function get_description()
|
|
{
|
|
return array(
|
|
'displayName' => lang('Weather'),
|
|
'title' => $this->context['city'],
|
|
'description' => lang('Weather')
|
|
);
|
|
}
|
|
|
|
public function get_type()
|
|
{
|
|
return 'et2-portlet-weather';
|
|
}
|
|
} |