Weather 1.5 :

- Added auto-update 
- Fixed unicode 3byte symbol  ( 'ь' - BUG )

git-svn-id: svn://kolibrios.org@8566 a494cfbc-eb01-0410-851d-a64ba20cac60
This commit is contained in:
superturbocat2001 2021-02-02 17:17:20 +00:00
parent c42b0cb538
commit 8236f21f5b
2 changed files with 82 additions and 52 deletions

View File

@ -4,18 +4,29 @@
Info: App uses api from openweathermap.org. Info: App uses api from openweathermap.org.
The standard configuration uses my token and the city of Moscow. The standard configuration uses my token and the city of Moscow.
You can always change it in the weather.json file. 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! weather.json configuration example:
{
"Celsius": false, // Enabled fahrenheit (Optional)
"Location": "Berlin", // City Berlin
"Token": "19ffa14b3dc0e238175829461d1788b8", // OpenWeatherMap token
"Lang": "ru", // Language (Optional)
"AutoUpdate": 5 // In minutes. 0 - disabled (Optional)
}
*/ */
#include <stdio.h> #include <stdio.h>
#include <string.h> #include <string.h>
#include <time.h>
#include <stdbool.h>
#include "json/json.h" #include "json/json.h"
#include <kos32sys1.h> #include <kos32sys1.h>
#include <kolibrisys.h> #include <kolibrisys.h>
#include <clayer/http.h> #include <clayer/http.h>
#include <clayer/libimg.h> #include <clayer/libimg.h>
#define VERSION "Weather 1.4" #define VERSION "Weather 1.5"
enum BUTTONS{ enum BUTTONS{
BTN_QUIT = 1, BTN_QUIT = 1,
@ -33,9 +44,10 @@ unsigned WINDOW_W = 230;
#define API "api.openweathermap.org/data/2.5/weather?q=%s&appid=%s&units=%s&lang=%s" #define API "api.openweathermap.org/data/2.5/weather?q=%s&appid=%s&units=%s&lang=%s"
#define IMAGE_URL "openweathermap.org/img/w/%s.png" #define IMAGE_URL "openweathermap.org/img/w/%s.png"
Image *blend; Image *blend=NULL;
unsigned char char_size=1; unsigned char char_size=1;
uint64_t AutoUpdateTime = 0;
char *wind_speed_str, *pressure_str, *visibility_str, *humidity_str, *update_str, *wind_deg_str; char *wind_speed_str, *pressure_str, *visibility_str, *humidity_str, *update_str, *wind_deg_str;
@ -72,7 +84,7 @@ void notify_show(char *text)
void* safe_malloc(size_t size) void* safe_malloc(size_t size)
{ {
void *p=user_alloc(size); void *p=malloc(size);
if(p==NULL){ if(p==NULL){
notify_show("'Memory allocation error!' -E"); notify_show("'Memory allocation error!' -E");
exit(0); exit(0);
@ -81,10 +93,9 @@ void* safe_malloc(size_t size)
} }
} }
char tmp_buff[100]; void draw_format_text_sys(int x, int y, color_t color, const char *format_str, ... ) // Форматированный вывод в окно
void draw_format_text_sys(int x, int y, color_t color, const char *format_str, ... )
{ {
char tmp_buff[100];
va_list ap; va_list ap;
va_start (ap, format_str); va_start (ap, format_str);
vsnprintf(tmp_buff, sizeof tmp_buff ,format_str, ap); vsnprintf(tmp_buff, sizeof tmp_buff ,format_str, ap);
@ -92,7 +103,7 @@ void draw_format_text_sys(int x, int y, color_t color, const char *format_str, .
draw_text_sys(tmp_buff, x, y , 0, color); draw_text_sys(tmp_buff, x, y , 0, color);
} }
void find_and_set(json_value *value, struct open_weather_data* weather) void find_and_set(json_value *value, struct open_weather_data* weather) // Ищем значения в json и заполняем структуру "myw"
{ {
for(int i=0; i<value->u.object.length; i++){ for(int i=0; i<value->u.object.length; i++){
if(!strcmp(JSON_OBJ(i).name, "main")){ if(!strcmp(JSON_OBJ(i).name, "main")){
@ -106,8 +117,12 @@ void find_and_set(json_value *value, struct open_weather_data* weather)
weather->humidity = JSON_OBJ(i).value->u.object.values[5].value->u.integer; weather->humidity = JSON_OBJ(i).value->u.object.values[5].value->u.integer;
} }
if(!strcmp(JSON_OBJ(i).name, "name")){ if(!strcmp(JSON_OBJ(i).name, "name")){
if(!strcmp(&JSON_OBJ(i).value->u.string.ptr[JSON_OBJ(i).value->u.string.length-3], "")){
strncpy(weather->City, JSON_OBJ(i).value->u.string.ptr, JSON_OBJ(i).value->u.string.length-3);
}else{
strcpy(weather->City, JSON_OBJ(i).value->u.string.ptr); strcpy(weather->City, JSON_OBJ(i).value->u.string.ptr);
} }
}
if(!strcmp(JSON_OBJ(i).name, "weather")){ if(!strcmp(JSON_OBJ(i).name, "weather")){
strcpy(weather->weath_desc, JSON_OBJ(i).value->u.array.values[0]->u.object.values[2].value->u.string.ptr); 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); strcpy(weather->image_code, JSON_OBJ(i).value->u.array.values[0]->u.object.values[3].value->u.string.ptr);
@ -131,7 +146,7 @@ void find_and_set(json_value *value, struct open_weather_data* weather)
char *errmsg = safe_malloc(weather->timezone = JSON_OBJ(i).value->u.string.length+6); 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); sprintf(errmsg,"'%s!' -E", JSON_OBJ(i).value->u.string.ptr);
notify_show(errmsg); notify_show(errmsg);
user_free(errmsg); free(errmsg);
} }
} }
} }
@ -144,46 +159,45 @@ http_msg* get_json(char *City, char *Token, char* Units)
if (h->status == OK || h->status == 404) { if (h->status == OK || h->status == 404) {
return h; return h;
} else { } else {
user_free(h->content_ptr); http_free(h);
user_free(h);
return NULL; return NULL;
} }
} }
void get_image(){ // Функция загрузки изображения void get_image() // Функция загрузки изображения
{
sprintf(full_url_image, IMAGE_URL, myw.image_code); sprintf(full_url_image, IMAGE_URL, myw.image_code);
http_msg *h= http_get(full_url_image, 0, HTTP_FLAG_BLOCK, ""); http_msg *h= http_get(full_url_image, 0, HTTP_FLAG_BLOCK, "");
http_long_receive(h); http_long_receive(h);
if (h->status == OK) { if (h->status == OK) {
Image *image = img_decode(h->content_ptr, h->content_length, 0); // Decode RAW data to Image data Image *image = img_decode(h->content_ptr, h->content_length, 0); // Декодирование RAW данных в данные изображения
if (image->Type != IMAGE_BPP32) { if (image->Type != IMAGE_BPP32) {
image = img_convert(image, NULL, IMAGE_BPP32, 0, 0); // Convert image to format BPP32 image = img_convert(image, NULL, IMAGE_BPP32, 0, 0); // Конвертируем картику в BPP32
if (!image) { if (!image) {
notify_show("'Convetring image error!' -E"); notify_show("'Convetring image error!' -E");
exit(0); exit(0);
} }
} }
blend = img_create(64, 64, IMAGE_BPP32); // Create an empty layer blend = img_create(64, 64, IMAGE_BPP32); // Создаём фон для картинки
img_fill_color(blend, 64, 64, sys_color_table.work_area); // Fill the layer with one color img_fill_color(blend, 64, 64, sys_color_table.work_area); // Заливаем фон цветом окна
Image* image2 = img_scale(image, 0, 0, 50, 50, NULL, LIBIMG_SCALE_STRETCH , LIBIMG_INTER_BILINEAR, 64, 64); 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_blend(blend, image2, 0, 0, 0, 0, 64, 64); // Смешиваем растянутую картинку и фон для получения прозрачности
// Уничтожаем ненужные структуры изображений img_destroy(image); // Уничтожаем исходную картинку
img_destroy(image); img_destroy(image2); // Уничтажаем растянутую картинку
img_destroy(image2);
}else{ }else{
notify_show("'Image not loaded!!' -W"); notify_show("'Image not loaded!!' -W");
} }
user_free(h->content_ptr); if(h!=NULL){
user_free(h); http_free(h);
h=NULL; }
} }
void RedrawGUI() // Перересовываем интерфейс void RedrawGUI() // Перересовываем интерфейс
{ {
begin_draw(); // Начинам прорисовку begin_draw(); // Начинам прорисовку
int new_win_w = (strlen(myw.City)/char_size+10)*(UTF8_W+char_size-1); // Если название города не влезает int new_win_w = (strlen(myw.City)/char_size+10)*(UTF8_W+char_size-1); // Если название города не влезает в окно
if(new_win_w<WINDOW_W){ if(new_win_w<WINDOW_W){
new_win_w=WINDOW_W; new_win_w=WINDOW_W;
} }
@ -227,13 +241,13 @@ void get_config(char **City, char **Token, char **Units) // Загружаем
} }
json_value* value =json_parse (config_buff, size); // Парсим конфиг json_value* value =json_parse (config_buff, size); // Парсим конфиг
for(int i=0; i<value->u.object.length; i++){ for(int i=0; i<value->u.object.length; i++){
if(!strcmp(JSON_OBJ(i).name, "Location")){ if(!strcmp(JSON_OBJ(i).name, "Location") && JSON_OBJ(i).value->type==json_string){
*City = JSON_OBJ(i).value->u.string.ptr; // Получаем название города *City = JSON_OBJ(i).value->u.string.ptr; // Получаем название города
} }
if(!strcmp(JSON_OBJ(i).name, "Token")){ else if(!strcmp(JSON_OBJ(i).name, "Token") && JSON_OBJ(i).value->type==json_string){
*Token = JSON_OBJ(i).value->u.string.ptr; // Получаем токен *Token = JSON_OBJ(i).value->u.string.ptr; // Получаем токен
} }
if(!strcmp(JSON_OBJ(i).name, "Celsius")){ else if(!strcmp(JSON_OBJ(i).name, "Celsius") && JSON_OBJ(i).value->type==json_boolean){
if(JSON_OBJ(i).value->u.boolean){ if(JSON_OBJ(i).value->u.boolean){
*Units = "metric"; *Units = "metric";
temp_char = 'C'; temp_char = 'C';
@ -242,22 +256,25 @@ void get_config(char **City, char **Token, char **Units) // Загружаем
temp_char = 'F'; temp_char = 'F';
} }
} }
if(!strcmp(JSON_OBJ(i).name, "Lang")){ else if(!strcmp(JSON_OBJ(i).name, "Lang") && JSON_OBJ(i).value->type==json_string){
strncpy(lang, JSON_OBJ(i).value->u.string.ptr,2); // Получам язык strncpy(lang, JSON_OBJ(i).value->u.string.ptr,2); // Получаем язык
}
else if(!strcmp(JSON_OBJ(i).name, "AutoUpdate") && JSON_OBJ(i).value->type==json_integer){
AutoUpdateTime = JSON_OBJ(i).value->u.integer; // Получаем время автообновлений данных
} }
} }
if(*City==NULL || *Token ==NULL){ if(*City==NULL || *Token ==NULL){
notify_show("'Invalid config!' -E"); notify_show("'Invalid config!' -E");
exit(0); exit(0);
} }
user_free(config_buff); free(config_buff);
fclose(config_j); fclose(config_j);
} }
void Update(char* city, char* token, char* units) // Обновление данных void Update(char* city, char* token, char* units) // Обновление данных
{ {
if(blend!=NULL){ if(blend!=NULL){
img_destroy(blend); // Уничтожение картинки с прозрачностью img_destroy(blend); // Уничтожение картинику с прозрачностью
blend = NULL; blend = NULL;
} }
memset(&myw, 0, sizeof myw); // Обнуляем структуру memset(&myw, 0, sizeof myw); // Обнуляем структуру
@ -268,10 +285,9 @@ void Update(char* city, char* token, char* units) // Обновление дан
json_value* value=json_parse(json_file->content_ptr, json_file->content_length); // Парсим json файл json_value* value=json_parse(json_file->content_ptr, json_file->content_length); // Парсим json файл
find_and_set(value, &myw); // Ищем значения в json find_and_set(value, &myw); // Ищем значения в json
sprintf(format_temp_str, "%s°%c","%d",temp_char); // Формируем строку для вывода температуры sprintf(format_temp_str, "%s°%c","%d",temp_char); // Формируем строку для вывода температуры
get_image(); // Получаем изображение get_image(); // Получаем картинку
json_value_free(value); // Очищаем ненужные данные json_value_free(value); // Уничтожаем полученные json значения
user_free(json_file->content_ptr); http_free(json_file);
user_free(json_file);
}else{ }else{
notify_show("'Connection error!' -E"); notify_show("'Connection error!' -E");
} }
@ -306,30 +322,42 @@ void set_lang()
} }
} }
int main(){ int main()
{
win_pos = get_mouse_pos(0); // Получаем позицию курсора win_pos = get_mouse_pos(0); // Получаем позицию курсора
if(!kolibri_libimg_init()){ // Загружаем libimg.obj if(!kolibri_libimg_init()){ // Загружаем libimg.obj
notify_show("Libimg.obj not loaded!' -E"); notify_show("Libimg.obj not loaded!' -E");
exit(0); exit(0);
} }
get_system_colors(&sys_color_table); // Получаем таблица цветов get_system_colors(&sys_color_table); // Получаем таблица цветов
char *City, *Token, *Units; // Указатели на токен, название города, систему мер char *City=NULL, *Token=NULL, *Units=NULL; // Указатели на токен, название города, систему мер
get_config(&City, &Token, &Units); // Загружаем конфиг get_config(&City, &Token, &Units); // Загружаем конфиг
set_lang(); // Установить язык приложения set_lang(); // Установить язык приложения
Update(City,Token, Units); // Обновить данные Update(City, Token, Units);
uint32_t (*event)();
if(AutoUpdateTime<=0){
event = get_os_event;
}else{
event = wait_for_event;
}
while(1){ while(1){
switch(get_os_event()){ // Получаем системное событие switch(event(6000*AutoUpdateTime)){ // Получаем системное событие
case KOLIBRI_EVENT_NONE: // Нет события case KOLIBRI_EVENT_NONE: // Нет события
Update(City, Token, Units);
debug_printf("Weather: Update\n");
break; break;
case KOLIBRI_EVENT_REDRAW: // Событие перерисовки case KOLIBRI_EVENT_REDRAW: // Событие перерисовки
RedrawGUI(); RedrawGUI();
break; break;
case KOLIBRI_EVENT_BUTTON: // Событие кнопок case KOLIBRI_EVENT_BUTTON: // Событие кнопок
switch (get_os_button()){ switch (get_os_button()){
case BTN_UPDATE: // Кнопка обновить case BTN_UPDATE:
Update(City, Token, Units); Update(City, Token, Units);
RedrawGUI(); RedrawGUI();
break; break;
@ -337,6 +365,7 @@ int main(){
exit(0); exit(0);
break; break;
} }
break;
} }
} }
return 0; return 0;

View File

@ -2,5 +2,6 @@
"Celsius": true, "Celsius": true,
"Location": "Moscow", "Location": "Moscow",
"Token": "19ffa14b3dc0e238175829461d1788b8", "Token": "19ffa14b3dc0e238175829461d1788b8",
"Lang": "ru" "Lang": "ru",
"AutoUpdate": 1
} }