2021-01-27 20:37:12 +01:00
|
|
|
|
/* Copyright (C) 2019-2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Info: App uses api from openweathermap.org.
|
|
|
|
|
The standard configuration uses my token and the city of Moscow.
|
|
|
|
|
You can always change it in the weather.json file.
|
|
|
|
|
If you use UTF-8 encoding, then city names can be entered in different languages!
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include "json/json.h"
|
|
|
|
|
#include <kos32sys1.h>
|
|
|
|
|
#include <kolibrisys.h>
|
|
|
|
|
#include <clayer/http.h>
|
|
|
|
|
#include <clayer/libimg.h>
|
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
#define VERSION "Weather 1.2b"
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
|
|
|
|
enum BUTTONS{
|
|
|
|
|
BTN_QUIT = 1,
|
|
|
|
|
BTN_UPDATE = 2
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#define JSON_OBJ(X) value->u.object.values[X]
|
|
|
|
|
#define OK 200
|
|
|
|
|
#define API "http://api.openweathermap.org/data/2.5/weather?q=%s&appid=%s&units=metric"
|
|
|
|
|
#define IMAGE_URL "http://openweathermap.org/img/w/%s.png"
|
|
|
|
|
#define START_YPOS 34
|
2021-01-28 00:59:58 +01:00
|
|
|
|
#define UTF8_W 8
|
|
|
|
|
#define CP866_W 6
|
|
|
|
|
|
|
|
|
|
#define WINDOW_W 200
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
|
|
|
|
Image *image;
|
2021-01-28 00:59:58 +01:00
|
|
|
|
Image *blend;
|
|
|
|
|
|
|
|
|
|
char full_url[512];
|
2021-01-27 20:37:12 +01:00
|
|
|
|
char full_url_image[256];
|
|
|
|
|
struct kolibri_system_colors sys_color_table;
|
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
pos_t win_pos;
|
|
|
|
|
|
2021-01-27 20:37:12 +01:00
|
|
|
|
#pragma pack(push,1)
|
|
|
|
|
typedef struct {
|
2021-01-28 00:59:58 +01:00
|
|
|
|
char City[256];
|
|
|
|
|
int wind_speed;
|
2021-01-27 20:37:12 +01:00
|
|
|
|
//int wind_deg;
|
|
|
|
|
int pressure;
|
|
|
|
|
int humidity;
|
2021-01-28 00:59:58 +01:00
|
|
|
|
//char weath_main[256];
|
|
|
|
|
char weath_desc[256];
|
2021-01-27 20:37:12 +01:00
|
|
|
|
int visibility;
|
|
|
|
|
int timezone;
|
2021-01-28 00:59:58 +01:00
|
|
|
|
char image_code[4];
|
|
|
|
|
int temp;
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}open_weather_data;
|
|
|
|
|
#pragma pack(pop)
|
|
|
|
|
|
|
|
|
|
open_weather_data myw;
|
|
|
|
|
|
|
|
|
|
void notify_show(char *text)
|
|
|
|
|
{
|
|
|
|
|
start_app("/sys/@notify", text);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void* safe_malloc(size_t size) // Безопасный malloc. Показывает уведомление об ошибке и закрывает программу если память не была выделена
|
|
|
|
|
{
|
|
|
|
|
void *p=malloc(size);
|
|
|
|
|
if(p==NULL){
|
|
|
|
|
notify_show("'Memory allocation error!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}else{
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
char tmp_buff[100];
|
|
|
|
|
|
2021-01-27 20:37:12 +01:00
|
|
|
|
static void draw_format_text_sys(int x, int y, color_t color, const char *format_str, ... )
|
|
|
|
|
{
|
|
|
|
|
va_list ap;
|
|
|
|
|
va_start (ap, format_str);
|
|
|
|
|
vsnprintf(tmp_buff, sizeof tmp_buff ,format_str, ap);
|
|
|
|
|
va_end(ap);
|
|
|
|
|
draw_text_sys(tmp_buff, x, y , 0, color);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void find_and_set(json_value *value, open_weather_data* weather)
|
|
|
|
|
{
|
|
|
|
|
for(int i=0; i<value->u.object.length; i++){
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "main")){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
if(JSON_OBJ(i).value->u.object.values[0].value->type==json_double)
|
|
|
|
|
{
|
|
|
|
|
weather->temp = (int)JSON_OBJ(i).value->u.object.values[0].value->u.dbl;
|
|
|
|
|
}else{
|
|
|
|
|
weather->temp = JSON_OBJ(i).value->u.object.values[0].value->u.integer;
|
|
|
|
|
}
|
2021-01-27 20:37:12 +01:00
|
|
|
|
weather->pressure = JSON_OBJ(i).value->u.object.values[4].value->u.integer;
|
|
|
|
|
weather->humidity = JSON_OBJ(i).value->u.object.values[5].value->u.integer;
|
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "name")){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
strcpy(weather->City,JSON_OBJ(i).value->u.string.ptr);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "weather")){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
strcpy(weather->weath_desc, JSON_OBJ(i).value->u.array.values[0]->u.object.values[2].value->u.string.ptr);
|
|
|
|
|
strcpy(weather->image_code, JSON_OBJ(i).value->u.array.values[0]->u.object.values[3].value->u.string.ptr);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "wind")){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
if(JSON_OBJ(i).value->u.object.values[0].value->type==json_double)
|
|
|
|
|
{
|
|
|
|
|
weather->wind_speed = (int)JSON_OBJ(i).value->u.object.values[0].value->u.dbl;
|
|
|
|
|
}else{
|
|
|
|
|
weather->wind_speed = JSON_OBJ(i).value->u.object.values[0].value->u.integer;
|
|
|
|
|
}
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "visibility")){
|
|
|
|
|
weather->visibility = JSON_OBJ(i).value->u.integer;
|
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "timezone")){
|
|
|
|
|
weather->timezone = JSON_OBJ(i).value->u.integer/60/60;
|
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "message")){
|
|
|
|
|
char *errmsg = safe_malloc(weather->timezone = JSON_OBJ(i).value->u.string.length+6);
|
|
|
|
|
sprintf(errmsg,"'%s!' -E", JSON_OBJ(i).value->u.string.ptr);
|
|
|
|
|
notify_show(errmsg);
|
2021-01-28 00:59:58 +01:00
|
|
|
|
free(errmsg);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
http_msg* get_json(char *City, char *Token)
|
|
|
|
|
{
|
|
|
|
|
sprintf(full_url, API, City, Token);
|
|
|
|
|
http_msg *h = http_get(full_url, 0, HTTP_FLAG_BLOCK, "");
|
|
|
|
|
http_long_receive(h);
|
|
|
|
|
if (h->status == OK || h->status == 404) {
|
2021-01-28 00:59:58 +01:00
|
|
|
|
return h;
|
2021-01-27 20:37:12 +01:00
|
|
|
|
} else {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void get_image(){
|
|
|
|
|
sprintf(full_url_image, IMAGE_URL, myw.image_code);
|
|
|
|
|
http_msg *h= http_get(full_url_image, 0, HTTP_FLAG_BLOCK, "");
|
|
|
|
|
http_long_receive(h);
|
|
|
|
|
|
|
|
|
|
if (h->status == OK) {
|
|
|
|
|
image = img_decode(h->content_ptr, h->content_length, 0); // Decode RAW data to Image data
|
|
|
|
|
if (image->Type != IMAGE_BPP32) {
|
|
|
|
|
image = img_convert(image, NULL, IMAGE_BPP32, 0, 0); // Convert image to format BPP32
|
|
|
|
|
if (!image) {
|
|
|
|
|
notify_show("'Convetring image error!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-01-28 00:59:58 +01:00
|
|
|
|
user_free(h->content_ptr);
|
|
|
|
|
user_free(h);
|
|
|
|
|
blend = img_create(64, 64, IMAGE_BPP32); // Create an empty layer
|
|
|
|
|
img_fill_color(blend, 64, 64, sys_color_table.work_area); // Fill the layer with one color
|
|
|
|
|
Image* image2 = img_scale(image, 0, 0, 50, 50, NULL, LIBIMG_SCALE_STRETCH , LIBIMG_INTER_BILINEAR, 64, 64);
|
|
|
|
|
img_blend(blend, image2, 0, 0, 0, 0, 64, 64); // Blending images to display the alpha channel.
|
|
|
|
|
img_destroy(image);
|
|
|
|
|
img_destroy(image2);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}else{
|
|
|
|
|
notify_show("'Image not loaded!!' -W");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RedrawGUI() //Рисуем окно
|
|
|
|
|
{
|
|
|
|
|
begin_draw(); //Начинаем рисование интерфейса )
|
2021-01-28 00:59:58 +01:00
|
|
|
|
|
|
|
|
|
int new_win_w = (strlen(myw.City)+11)*UTF8_W;
|
|
|
|
|
if(new_win_w<WINDOW_W){
|
|
|
|
|
new_win_w=WINDOW_W;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sys_create_window(win_pos.x, win_pos.y, new_win_w, START_YPOS+200, VERSION, sys_color_table.work_area, 0x14); // Создаём окно.
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
draw_format_text_sys(20, START_YPOS, 0xB0000000 | sys_color_table.work_text, "%s (UTC%+d)", myw.City, myw.timezone);
|
|
|
|
|
draw_format_text_sys(21, START_YPOS, 0xB0000000 | sys_color_table.work_text, "%s (UTC%+d)", myw.City, myw.timezone);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
img_draw(blend, 10, START_YPOS+30, 64,64,0,0);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
draw_format_text_sys(20, START_YPOS+20, 0xb0000000 | sys_color_table.work_text, myw.weath_desc);
|
|
|
|
|
draw_format_text_sys(21, START_YPOS+20, 0xb0000000 | sys_color_table.work_text, myw.weath_desc);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
draw_format_text_sys(100, START_YPOS+45, 0xB1000000 | sys_color_table.work_text, "%+d°C", myw.temp);
|
|
|
|
|
draw_format_text_sys(101, START_YPOS+46, 0xB1000000 | sys_color_table.work_text, "%+d°C", myw.temp);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
draw_format_text_sys(20, START_YPOS+80, 0x90000000 | sys_color_table.work_text, "Pressure: %d hPa",myw.pressure);
|
|
|
|
|
draw_format_text_sys(20, START_YPOS+100, 0x90000000 | sys_color_table.work_text, "Humidity: %d %s", myw.humidity, "%");
|
|
|
|
|
draw_format_text_sys(20, START_YPOS+120, 0x90000000 | sys_color_table.work_text, "Wind speed: %d m/s", myw.wind_speed);
|
|
|
|
|
draw_format_text_sys(20, START_YPOS+140, 0x90000000 | sys_color_table.work_text, "Visibility: %d m", myw.visibility);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
|
2021-01-28 00:59:58 +01:00
|
|
|
|
define_button(X_W(new_win_w/2-40,80), Y_H(START_YPOS+160,30), BTN_UPDATE, sys_color_table.work_button);
|
|
|
|
|
draw_text_sys("Update", (new_win_w/2)-(CP866_W*4), START_YPOS+170, 0, 0x90000000 | sys_color_table.work_button_text);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
end_draw();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void get_config(char **City, char **Token)
|
|
|
|
|
{
|
|
|
|
|
FILE *config_j = fopen("weather.json", "rb");
|
|
|
|
|
if(config_j==NULL){
|
|
|
|
|
notify_show("'Configuration file not found!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
|
|
|
|
size_t size = _ksys_get_filesize("weather.json");
|
|
|
|
|
char *config_buff = safe_malloc(size+1);
|
|
|
|
|
if(size != fread(config_buff, sizeof(char), size, config_j)){
|
|
|
|
|
notify_show("'The configuration file was not fully read!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
|
|
|
|
json_value* value =json_parse (config_buff, size);
|
|
|
|
|
for(int i=0; i<value->u.object.length; i++){
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "Location")){
|
|
|
|
|
*City = JSON_OBJ(i).value->u.string.ptr;
|
|
|
|
|
}
|
|
|
|
|
if(!strcmp(JSON_OBJ(i).name, "Token")){
|
|
|
|
|
*Token = JSON_OBJ(i).value->u.string.ptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if(*City==NULL || *Token ==NULL){
|
|
|
|
|
notify_show("'Invalid config!!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
|
|
|
|
free(config_buff);
|
2021-01-28 00:59:58 +01:00
|
|
|
|
fclose(config_j);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Update(char* city, char* token)
|
|
|
|
|
{
|
2021-01-28 00:59:58 +01:00
|
|
|
|
if(blend!=NULL){
|
|
|
|
|
img_destroy(blend);
|
|
|
|
|
}
|
2021-01-27 20:37:12 +01:00
|
|
|
|
memset(&myw, 0, sizeof myw);
|
2021-01-28 00:59:58 +01:00
|
|
|
|
strcpy(myw.City,"None");
|
|
|
|
|
strcpy(myw.weath_desc,"unknown");
|
2021-01-27 20:37:12 +01:00
|
|
|
|
http_msg *json_file = get_json(city, token);
|
|
|
|
|
if(json_file != NULL){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
json_value* value=json_parse (json_file->content_ptr, json_file->content_length);
|
2021-01-27 20:37:12 +01:00
|
|
|
|
find_and_set(value, &myw);
|
|
|
|
|
get_image();
|
2021-01-28 00:59:58 +01:00
|
|
|
|
json_value_free(value);
|
|
|
|
|
user_free(json_file->content_ptr);
|
|
|
|
|
user_free(json_file);
|
|
|
|
|
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}else{
|
2021-01-28 00:59:58 +01:00
|
|
|
|
notify_show("'Connection error!' -E");
|
2021-01-27 20:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int main(){
|
2021-01-28 00:59:58 +01:00
|
|
|
|
win_pos = get_mouse_pos(0);
|
|
|
|
|
if(!kolibri_libimg_init()){
|
|
|
|
|
notify_show("Libimg.obj not loaded!' -E");
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
2021-01-27 20:37:12 +01:00
|
|
|
|
get_system_colors(&sys_color_table);
|
|
|
|
|
char *City = NULL;
|
|
|
|
|
char *Token = NULL;
|
|
|
|
|
get_config(&City, &Token);
|
|
|
|
|
Update(City,Token);
|
2021-01-28 00:59:58 +01:00
|
|
|
|
|
2021-01-27 20:37:12 +01:00
|
|
|
|
while(1){
|
|
|
|
|
switch(get_os_event()){
|
|
|
|
|
case KOLIBRI_EVENT_NONE:
|
|
|
|
|
break;
|
|
|
|
|
case KOLIBRI_EVENT_REDRAW:
|
|
|
|
|
RedrawGUI();
|
|
|
|
|
break;
|
|
|
|
|
case KOLIBRI_EVENT_BUTTON: // Событие обработки кнопок
|
|
|
|
|
switch (get_os_button()){
|
|
|
|
|
case BTN_UPDATE:
|
|
|
|
|
Update(City, Token);
|
|
|
|
|
RedrawGUI();
|
|
|
|
|
break;
|
|
|
|
|
case BTN_QUIT:
|
|
|
|
|
exit(0);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|