2018-08-14 11:13:38 +03:00
/* vim: tabstop=4 shiftwidth=4 noexpandtab
* This file is part of ToaruOS and is released under the terms
* of the NCSA / University of Illinois License - see LICENSE . md
* Copyright ( C ) 2018 K . Lange
*
* file - browser - Show directory listings .
*
* This is a basic graphical file navigator . It ' s based somewhat
* on the original Python implementation . There ' s still a lot
* of work to do here presentation - wise .
*/
2018-05-21 02:12:02 +03:00
# include <stdio.h>
2018-06-04 06:35:46 +03:00
# include <unistd.h>
2018-08-10 07:57:24 +03:00
# include <dirent.h>
# include <time.h>
2018-08-12 14:03:37 +03:00
# include <math.h>
2018-11-18 04:25:24 +03:00
# include <libgen.h>
2018-11-19 15:14:39 +03:00
# include <signal.h>
2018-08-10 07:57:24 +03:00
# include <sys/stat.h>
# include <sys/time.h>
2018-11-19 13:26:40 +03:00
# include <sys/wait.h>
2018-05-21 02:12:02 +03:00
# include <toaru/yutani.h>
# include <toaru/graphics.h>
# include <toaru/decorations.h>
# include <toaru/menu.h>
2018-08-10 07:57:24 +03:00
# include <toaru/icon_cache.h>
# include <toaru/list.h>
# include <toaru/sdf.h>
2018-05-21 02:12:02 +03:00
# define APPLICATION_TITLE "File Browser"
2018-11-18 12:59:29 +03:00
uint64_t precise_current_time ( void ) {
struct timeval t ;
gettimeofday ( & t , NULL ) ;
time_t sec_diff = t . tv_sec ;
suseconds_t usec_diff = t . tv_usec ;
return ( uint64_t ) ( sec_diff * 1000 + usec_diff / 1000 ) ;
}
uint64_t precise_time_since ( uint64_t start_time ) {
uint32_t now = precise_current_time ( ) ;
uint32_t diff = now - start_time ; /* Milliseconds */
return diff ;
}
2018-11-18 04:25:24 +03:00
char title [ 512 ] ;
2018-05-21 02:12:02 +03:00
static yutani_t * yctx ;
static yutani_window_t * main_window ;
static gfx_context_t * ctx ;
2018-11-19 13:26:40 +03:00
# define SCROLL_AMOUNT 120
2018-05-21 02:12:02 +03:00
static int application_running = 1 ;
2018-08-10 07:57:24 +03:00
static int show_hidden = 1 ;
2018-08-12 07:59:00 +03:00
static int scroll_offset = 0 ;
static int available_height = 0 ;
2018-11-19 15:14:39 +03:00
static int is_desktop_background = 0 ;
static sprite_t * wallpaper_buffer = NULL ;
static void draw_background ( int width , int height ) ;
static int menu_bar_height = MENU_BAR_HEIGHT ;
2018-05-21 02:12:02 +03:00
static struct menu_bar menu_bar = { 0 } ;
static struct menu_bar_entries menu_entries [ ] = {
{ " File " , " file " } ,
{ " Go " , " go " } ,
{ " Help " , " help " } ,
{ NULL , NULL } ,
} ;
2018-11-18 03:38:47 +03:00
static struct MenuList * context_menu = NULL ;
2018-05-21 02:12:02 +03:00
static void _menu_action_exit ( struct MenuEntry * entry ) {
application_running = 0 ;
}
2018-11-19 15:14:39 +03:00
static int _decor_get_bounds ( yutani_window_t * win , struct decor_bounds * bounds ) {
if ( is_desktop_background ) {
memset ( bounds , 0 , sizeof ( struct decor_bounds ) ) ;
bounds - > top_height = 30 ;
bounds - > left_width = 20 ;
return 0 ;
}
return decor_get_bounds ( win , bounds ) ;
}
2018-08-10 07:57:24 +03:00
struct File {
char name [ 256 ] ;
char icon [ 256 ] ;
2018-11-19 13:26:40 +03:00
//char date[256];
char link [ 256 ] ; /* why so big */
2018-11-19 15:14:39 +03:00
char launcher [ 256 ] ;
char filename [ 256 ] ;
int order ;
2018-08-12 14:03:37 +03:00
int type ;
2018-11-18 12:59:29 +03:00
int selected ;
2018-08-10 07:57:24 +03:00
} ;
2018-11-18 12:36:03 +03:00
static int FILE_HEIGHT = 80 ; /* Not a constant */
2018-11-18 13:27:35 +03:00
static int FILE_WIDTH = 100 ;
2018-11-18 12:36:03 +03:00
static int FILE_PTR_WIDTH = 1 ;
2018-08-10 07:57:24 +03:00
static gfx_context_t * contents = NULL ;
static sprite_t * contents_sprite = NULL ;
2018-08-12 14:03:37 +03:00
static char * last_directory = NULL ;
static int hilighted_offset = - 1 ;
static struct File * * file_pointers = NULL ;
static ssize_t file_pointers_len = 0 ;
static int _close_enough ( struct yutani_msg_window_mouse_event * me ) {
if ( me - > command = = YUTANI_MOUSE_EVENT_RAISE & & sqrt ( pow ( me - > new_x - me - > old_x , 2 ) + pow ( me - > new_y - me - > old_y , 2 ) ) < 10 ) {
return 1 ;
}
return 0 ;
}
2018-08-10 07:57:24 +03:00
2018-08-12 14:03:37 +03:00
static void clear_offset ( int offset ) {
2018-11-18 12:36:03 +03:00
int offset_y = offset / FILE_PTR_WIDTH ;
int offset_x = offset % FILE_PTR_WIDTH ;
2018-11-19 15:14:39 +03:00
draw_rectangle_solid ( contents , offset_x * FILE_WIDTH , offset_y * FILE_HEIGHT , FILE_WIDTH , FILE_HEIGHT , rgba ( 0 , 0 , 0 , 0 ) ) ;
2018-08-12 14:03:37 +03:00
}
2018-08-10 07:57:24 +03:00
2018-08-12 14:03:37 +03:00
static void draw_file ( struct File * f , int offset ) {
2018-11-18 12:36:03 +03:00
int offset_y = offset / FILE_PTR_WIDTH ;
int offset_x = offset % FILE_PTR_WIDTH ;
int x = offset_x * FILE_WIDTH ;
int y = offset_y * FILE_HEIGHT ;
sprite_t * icon = icon_get_48 ( f - > icon ) ;
2018-11-18 13:27:35 +03:00
int len = strlen ( f - > name ) ;
char * name = malloc ( len + 4 ) ;
memcpy ( name , f - > name , len + 1 ) ;
2018-11-18 12:36:03 +03:00
int name_width ;
while ( ( name_width = draw_sdf_string_width ( name , 16 , SDF_FONT_THIN ) ) > FILE_WIDTH - 8 /* Padding */ ) {
len - - ;
2018-11-18 13:27:35 +03:00
name [ len + 0 ] = ' . ' ;
name [ len + 1 ] = ' . ' ;
name [ len + 2 ] = ' . ' ;
name [ len + 3 ] = ' \0 ' ;
2018-11-18 12:36:03 +03:00
}
int center_x_icon = ( FILE_WIDTH - icon - > width ) / 2 ;
int center_x_text = ( FILE_WIDTH - name_width ) / 2 ;
draw_sprite ( contents , icon , center_x_icon + x , y + 2 ) ;
2018-11-18 12:59:29 +03:00
if ( f - > selected ) {
2018-11-18 12:36:03 +03:00
draw_sprite_alpha_paint ( contents , icon , center_x_icon + x , y + 2 , 0.5 , rgb ( 72 , 167 , 255 ) ) ;
draw_rounded_rectangle ( contents , center_x_text + x - 2 , y + 54 , name_width + 6 , 20 , 3 , rgb ( 72 , 167 , 255 ) ) ;
draw_sdf_string ( contents , center_x_text + x , y + 54 , name , 16 , rgb ( 255 , 255 , 255 ) , SDF_FONT_THIN ) ;
2018-08-12 14:03:37 +03:00
} else {
2018-11-19 15:14:39 +03:00
if ( is_desktop_background ) {
//draw_rounded_rectangle(contents, center_x_text + x - 2, y + 54, name_width + 6, 20, 3, rgba(255,255,255,120));
draw_sdf_string_stroke ( contents , center_x_text + x + 1 , y + 55 , name , 16 , rgba ( 0 , 0 , 0 , 120 ) , SDF_FONT_THIN , 1.7 , 0.5 ) ;
draw_sdf_string ( contents , center_x_text + x , y + 54 , name , 16 , rgb ( 255 , 255 , 255 ) , SDF_FONT_THIN ) ;
} else {
draw_sdf_string ( contents , center_x_text + x , y + 54 , name , 16 , rgb ( 0 , 0 , 0 ) , SDF_FONT_THIN ) ;
}
2018-08-12 14:03:37 +03:00
}
2018-11-18 12:36:03 +03:00
2018-11-18 12:59:29 +03:00
if ( offset = = hilighted_offset ) {
draw_sprite_alpha_paint ( contents , icon , center_x_icon + x , y + 2 , 0.3 , rgb ( 255 , 255 , 255 ) ) ;
}
2018-11-19 13:26:40 +03:00
if ( f - > link [ 0 ] ) {
sprite_t * arrow = icon_get_16 ( " forward " ) ;
draw_sprite ( contents , arrow , center_x_icon + 32 + x , y + 32 ) ;
}
2018-11-18 12:36:03 +03:00
free ( name ) ;
2018-08-12 14:03:37 +03:00
}
2018-08-10 07:57:24 +03:00
2018-08-12 14:03:37 +03:00
static struct File * get_file_at_offset ( int offset ) {
2018-11-18 12:36:03 +03:00
int offset_y = offset / FILE_PTR_WIDTH ;
int offset_x = offset % FILE_PTR_WIDTH ;
if ( offset_y > = 0 & & offset_x > = 0 ) {
int off_calc = offset_y * FILE_PTR_WIDTH + offset_x ;
if ( off_calc < file_pointers_len ) {
return file_pointers [ off_calc ] ;
}
2018-08-12 14:03:37 +03:00
}
return NULL ;
}
static void redraw_files ( void ) {
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
draw_file ( file_pointers [ i ] , i ) ;
2018-08-10 07:57:24 +03:00
}
}
2018-11-18 04:25:24 +03:00
static void set_title ( char * directory ) {
2018-11-19 15:14:39 +03:00
if ( is_desktop_background ) return ;
2018-11-18 04:25:24 +03:00
if ( directory ) {
if ( ! strcmp ( directory , " / " ) ) {
directory = " File System " ;
}
sprintf ( title , " %s - " APPLICATION_TITLE , directory ) ;
} else {
sprintf ( title , APPLICATION_TITLE ) ;
}
yutani_window_advertise_icon ( yctx , main_window , title , " folder " ) ;
}
2018-11-19 13:26:40 +03:00
static int has_extension ( struct File * f , char * extension ) {
int i = strlen ( f - > name ) ;
int j = strlen ( extension ) ;
do {
if ( f - > name [ i ] ! = ( extension ) [ j ] ) break ;
if ( j = = 0 ) return 1 ;
if ( i = = 0 ) break ;
i - - ;
j - - ;
} while ( 1 ) ;
return 0 ;
}
2018-08-12 07:59:00 +03:00
static void load_directory ( const char * path ) {
2018-08-12 14:03:37 +03:00
if ( file_pointers ) {
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
free ( file_pointers [ i ] ) ;
}
free ( file_pointers ) ;
2018-08-10 07:57:24 +03:00
}
DIR * dirp = opendir ( path ) ;
if ( ! dirp ) {
/* Failed to open directory. Throw up a warning? */
2018-08-12 14:03:37 +03:00
file_pointers = NULL ;
file_pointers_len = 0 ;
2018-08-10 07:57:24 +03:00
return ;
}
2018-08-12 14:03:37 +03:00
if ( last_directory ) {
free ( last_directory ) ;
}
2018-11-19 15:14:39 +03:00
char * home = getenv ( " HOME " ) ;
if ( home & & ! strcmp ( path , home ) ) {
set_title ( " Home " ) ;
} else {
char * tmp = strdup ( path ) ;
char * base = basename ( tmp ) ;
set_title ( base ) ;
free ( tmp ) ;
}
2018-11-18 04:25:24 +03:00
2018-11-18 13:27:35 +03:00
if ( path [ 0 ] = = ' / ' & & path [ 1 ] = = ' / ' ) {
last_directory = strdup ( path + 1 ) ;
} else {
last_directory = strdup ( path ) ;
}
2018-08-12 14:03:37 +03:00
2018-08-10 07:57:24 +03:00
/* Get the current time */
#if 0
struct tm * timeinfo ;
struct timeval now ;
gettimeofday ( & now , NULL ) ; //time(NULL);
timeinfo = localtime ( ( time_t * ) & now . tv_sec ) ;
int this_year = timeinfo - > tm_year ;
# endif
2018-08-12 14:03:37 +03:00
list_t * file_list = list_create ( ) ;
2018-08-10 07:57:24 +03:00
struct dirent * ent = readdir ( dirp ) ;
while ( ent ! = NULL ) {
2018-08-12 14:14:07 +03:00
if ( ent - > d_name [ 0 ] = = ' . ' & &
( ent - > d_name [ 1 ] = = ' \0 ' | |
( ent - > d_name [ 1 ] = = ' . ' & &
ent - > d_name [ 2 ] = = ' \0 ' ) ) ) {
/* skip . and .. */
ent = readdir ( dirp ) ;
continue ;
}
2018-08-10 07:57:24 +03:00
if ( show_hidden | | ( ent - > d_name [ 0 ] ! = ' . ' ) ) {
2018-08-12 14:14:07 +03:00
2018-08-10 07:57:24 +03:00
struct File * f = malloc ( sizeof ( struct File ) ) ;
sprintf ( f - > name , " %s " , ent - > d_name ) ; /* snprintf? copy min()? */
struct stat statbuf ;
2018-11-19 13:26:40 +03:00
struct stat statbufl ;
2018-08-10 07:57:24 +03:00
char tmp [ strlen ( path ) + strlen ( ent - > d_name ) + 2 ] ;
sprintf ( tmp , " %s/%s " , path , ent - > d_name ) ;
lstat ( tmp , & statbuf ) ;
2018-11-19 13:26:40 +03:00
2018-08-10 07:57:24 +03:00
if ( S_ISLNK ( statbuf . st_mode ) ) {
2018-11-19 13:26:40 +03:00
memcpy ( & statbufl , & statbuf , sizeof ( struct stat ) ) ;
stat ( tmp , & statbuf ) ;
readlink ( tmp , f - > link , 256 ) ;
} else {
f - > link [ 0 ] = ' \0 ' ;
2018-08-10 07:57:24 +03:00
}
2018-11-19 13:26:40 +03:00
2018-11-19 15:14:39 +03:00
f - > launcher [ 0 ] = ' \0 ' ;
2018-08-10 07:57:24 +03:00
if ( S_ISDIR ( statbuf . st_mode ) ) {
sprintf ( f - > icon , " folder " ) ;
2018-08-12 14:03:37 +03:00
f - > type = 1 ;
2018-08-10 07:57:24 +03:00
} else {
2018-11-19 15:14:39 +03:00
sprintf ( f - > launcher , " exec terminal bim " ) ;
if ( is_desktop_background & & has_extension ( f , " .launcher " ) ) {
FILE * file = fopen ( tmp , " r " ) ;
char tbuf [ 1024 ] ;
while ( ! feof ( file ) ) {
fgets ( tbuf , 1024 , file ) ;
char * nl = strchr ( tbuf , ' \n ' ) ;
if ( nl ) * nl = ' \0 ' ;
char * eq = strchr ( tbuf , ' = ' ) ;
if ( ! eq ) continue ;
* eq = ' \0 ' ; eq + + ;
if ( ! strcmp ( tbuf , " icon " ) ) {
sprintf ( f - > icon , " %s " , eq ) ;
} else if ( ! strcmp ( tbuf , " run " ) ) {
sprintf ( f - > launcher , " %s # " , eq ) ;
} else if ( ! strcmp ( tbuf , " title " ) ) {
sprintf ( f - > name , eq ) ;
}
}
sprintf ( f - > filename , " %s " , tmp ) ;
f - > type = 2 ;
2018-11-19 13:26:40 +03:00
} else {
2018-11-19 15:14:39 +03:00
if ( has_extension ( f , " .c " ) ) {
sprintf ( f - > icon , " c " ) ;
} else if ( has_extension ( f , " .h " ) ) {
sprintf ( f - > icon , " h " ) ;
} else if ( has_extension ( f , " .bmp " ) ) {
sprintf ( f - > icon , " image " ) ;
sprintf ( f - > launcher , " exec imgviewer " ) ;
} else if ( statbuf . st_mode & 0111 ) {
sprintf ( f - > icon , " %s " , f - > name ) ;
sprintf ( f - > launcher , " SELF " ) ;
} else {
sprintf ( f - > icon , " file " ) ;
}
f - > type = 0 ;
2018-11-19 13:26:40 +03:00
}
2018-08-10 07:57:24 +03:00
}
2018-11-18 12:59:29 +03:00
f - > selected = 0 ;
2018-08-10 07:57:24 +03:00
list_insert ( file_list , f ) ;
}
ent = readdir ( dirp ) ;
}
closedir ( dirp ) ;
2018-08-12 07:59:00 +03:00
2018-08-12 14:03:37 +03:00
/* create a an array to hold the files */
file_pointers = malloc ( sizeof ( struct File * ) * file_list - > length ) ;
file_pointers_len = file_list - > length ;
int i = 0 ;
foreach ( node , file_list ) {
file_pointers [ i ] = node - > value ;
i + + ;
}
2018-11-18 12:36:03 +03:00
2018-08-12 14:03:37 +03:00
list_free ( file_list ) ;
free ( file_list ) ;
/* Sort files */
int comparator ( const void * c1 , const void * c2 ) {
const struct File * f1 = * ( const struct File * * ) ( c1 ) ;
const struct File * f2 = * ( const struct File * * ) ( c2 ) ;
2018-11-19 15:14:39 +03:00
if ( f1 - > type > f2 - > type ) return - 1 ;
if ( f2 - > type > f1 - > type ) return 1 ;
if ( f1 - > type = = 2 & & f2 - > type = = 2 ) {
return strcmp ( f1 - > filename , f2 - > filename ) ;
}
2018-08-12 14:03:37 +03:00
return strcmp ( f1 - > name , f2 - > name ) ;
}
qsort ( file_pointers , file_pointers_len , sizeof ( struct File * ) , comparator ) ;
2018-08-12 07:59:00 +03:00
scroll_offset = 0 ;
2018-08-10 07:57:24 +03:00
}
static void reinitialize_contents ( void ) {
if ( contents ) {
free ( contents ) ;
}
if ( contents_sprite ) {
sprite_free ( contents_sprite ) ;
}
2018-09-12 06:53:08 +03:00
struct decor_bounds bounds ;
2018-11-19 15:14:39 +03:00
_decor_get_bounds ( main_window , & bounds ) ;
2018-09-12 06:53:08 +03:00
2018-11-19 15:14:39 +03:00
if ( is_desktop_background ) {
FILE_PTR_WIDTH = 1 ;
} else {
FILE_PTR_WIDTH = ( ctx - > width - bounds . width ) / FILE_WIDTH ;
}
2018-11-18 12:59:29 +03:00
int calculated_height = ( file_pointers_len / FILE_PTR_WIDTH + 1 ) * FILE_HEIGHT ;
2018-11-18 12:36:03 +03:00
2018-09-12 06:53:08 +03:00
contents_sprite = create_sprite ( main_window - > width - bounds . width , calculated_height , ALPHA_EMBEDDED ) ;
2018-08-10 07:57:24 +03:00
contents = init_graphics_sprite ( contents_sprite ) ;
2018-11-19 15:14:39 +03:00
draw_fill ( contents , rgba ( 0 , 0 , 0 , 0 ) ) ;
2018-08-10 07:57:24 +03:00
/* Draw file entries */
redraw_files ( ) ;
}
2018-11-19 15:14:39 +03:00
static void sig_usr2 ( int sig ) {
yutani_set_stack ( yctx , main_window , YUTANI_ZORDER_BOTTOM ) ;
yutani_flip ( yctx , main_window ) ;
signal ( SIGUSR2 , sig_usr2 ) ;
}
2018-05-21 02:12:02 +03:00
2018-11-19 15:14:39 +03:00
static void redraw_window ( void ) {
if ( ! is_desktop_background ) {
draw_fill ( ctx , rgb ( 255 , 255 , 255 ) ) ;
render_decorations ( main_window , ctx , title ) ;
} else {
/* Draw wallpaper */
draw_sprite ( ctx , wallpaper_buffer , 0 , 0 ) ;
}
2018-05-21 02:12:02 +03:00
2018-09-12 06:53:08 +03:00
struct decor_bounds bounds ;
2018-11-19 15:14:39 +03:00
_decor_get_bounds ( main_window , & bounds ) ;
if ( ! is_desktop_background ) {
menu_bar . x = bounds . left_width ;
menu_bar . y = bounds . top_height ;
menu_bar . width = ctx - > width - bounds . width ;
menu_bar . window = main_window ;
menu_bar_render ( & menu_bar , ctx ) ;
}
2018-05-21 02:12:02 +03:00
2018-08-10 07:57:24 +03:00
gfx_clear_clip ( ctx ) ;
2018-11-19 15:14:39 +03:00
gfx_add_clip ( ctx , bounds . left_width , bounds . top_height + menu_bar_height , ctx - > width - bounds . width , available_height ) ;
draw_sprite ( ctx , contents_sprite , bounds . left_width , bounds . top_height + menu_bar_height - scroll_offset ) ;
2018-08-10 07:57:24 +03:00
gfx_clear_clip ( ctx ) ;
gfx_add_clip ( ctx , 0 , 0 , ctx - > width , ctx - > height ) ;
2018-05-21 02:12:02 +03:00
flip ( ctx ) ;
yutani_flip ( yctx , main_window ) ;
}
static void resize_finish ( int w , int h ) {
2018-08-12 07:59:00 +03:00
int width_changed = ( main_window - > width ! = ( unsigned int ) w ) ;
2018-08-10 07:57:24 +03:00
2018-05-21 02:12:02 +03:00
yutani_window_resize_accept ( yctx , main_window , w , h ) ;
reinit_graphics_yutani ( ctx , main_window ) ;
2018-09-12 06:53:08 +03:00
struct decor_bounds bounds ;
2018-11-19 15:14:39 +03:00
_decor_get_bounds ( main_window , & bounds ) ;
2018-09-12 06:53:08 +03:00
2018-11-19 15:14:39 +03:00
available_height = ctx - > height - menu_bar_height - bounds . height ;
2018-08-12 07:59:00 +03:00
if ( width_changed ) {
2018-08-10 07:57:24 +03:00
reinitialize_contents ( ) ;
}
2018-08-12 07:59:00 +03:00
if ( available_height > contents - > height ) {
scroll_offset = 0 ;
} else {
if ( scroll_offset > contents - > height - available_height ) {
scroll_offset = contents - > height - available_height ;
}
}
2018-11-19 15:14:39 +03:00
if ( is_desktop_background ) {
draw_background ( w , h ) ;
}
2018-05-21 02:12:02 +03:00
redraw_window ( ) ;
yutani_window_resize_done ( yctx , main_window ) ;
yutani_flip ( yctx , main_window ) ;
}
2018-08-14 11:13:38 +03:00
/* TODO */
#if 0
2018-05-21 02:12:02 +03:00
static void _menu_action_input_path ( struct MenuEntry * entry ) {
}
2018-08-14 11:13:38 +03:00
# endif
2018-05-21 02:12:02 +03:00
static void _menu_action_navigate ( struct MenuEntry * entry ) {
/* go to entry->action */
2018-08-12 07:59:00 +03:00
struct MenuEntry_Normal * _entry = ( void * ) entry ;
load_directory ( _entry - > action ) ;
reinitialize_contents ( ) ;
redraw_window ( ) ;
2018-05-21 02:12:02 +03:00
}
static void _menu_action_up ( struct MenuEntry * entry ) {
/* go up */
2018-11-18 04:25:24 +03:00
char * tmp = strdup ( last_directory ) ;
char * dir = dirname ( tmp ) ;
load_directory ( dir ) ;
2018-08-12 14:03:37 +03:00
reinitialize_contents ( ) ;
redraw_window ( ) ;
2018-05-21 02:12:02 +03:00
}
2018-11-19 15:14:39 +03:00
static void _menu_action_refresh ( struct MenuEntry * entry ) {
char * tmp = strdup ( last_directory ) ;
load_directory ( tmp ) ;
reinitialize_contents ( ) ;
redraw_window ( ) ;
}
2018-05-21 02:12:02 +03:00
static void _menu_action_help ( struct MenuEntry * entry ) {
/* show help documentation */
2018-08-12 07:24:34 +03:00
system ( " help-browser file-browser.trt & " ) ;
2018-06-04 12:50:36 +03:00
redraw_window ( ) ;
2018-05-21 02:12:02 +03:00
}
2018-11-18 13:16:22 +03:00
static void _menu_action_copy ( struct MenuEntry * entry ) {
size_t output_size = 0 ;
int base_is_root = ! strcmp ( last_directory , " / " ) ; /* avoid redundant slash */
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
if ( file_pointers [ i ] - > selected ) {
2018-11-19 15:14:39 +03:00
output_size + = strlen ( last_directory ) + ! base_is_root + strlen ( file_pointers [ i ] - > type = = 2 ? file_pointers [ i ] - > filename : file_pointers [ i ] - > name ) + 1 ; /* base / file \n */
2018-11-18 13:16:22 +03:00
}
}
2018-11-18 13:42:31 +03:00
if ( ! output_size ) return ;
2018-11-18 13:16:22 +03:00
char * clipboard = malloc ( output_size ) ;
clipboard [ 0 ] = ' \0 ' ;
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
if ( file_pointers [ i ] - > selected ) {
strcat ( clipboard , last_directory ) ;
if ( ! base_is_root ) { strcat ( clipboard , " / " ) ; }
2018-11-19 15:14:39 +03:00
strcat ( clipboard , file_pointers [ i ] - > type = = 2 ? file_pointers [ i ] - > filename : file_pointers [ i ] - > name ) ;
2018-11-18 13:16:22 +03:00
strcat ( clipboard , " \n " ) ;
}
}
2018-11-18 13:42:31 +03:00
if ( clipboard [ output_size - 1 ] = = ' \n ' ) {
/* Remove trailing line feed */
clipboard [ output_size - 1 ] = ' \0 ' ;
}
2018-11-18 13:16:22 +03:00
yutani_set_clipboard ( yctx , clipboard ) ;
free ( clipboard ) ;
}
2018-05-21 02:12:02 +03:00
static void _menu_action_about ( struct MenuEntry * entry ) {
/* Show About dialog */
2018-08-12 07:24:34 +03:00
char about_cmd [ 1024 ] = " \0 " ;
strcat ( about_cmd , " about \" About File Browser \" /usr/share/icons/48/folder.bmp \" ToaruOS File Browser \" \" (C) 2018 K. Lange \n - \n Part of ToaruOS, which is free software \n released under the NCSA/University of Illinois \n license. \n - \n %https://toaruos.org \n %https://gitlab.com/toaruos \" " ) ;
char coords [ 100 ] ;
sprintf ( coords , " %d %d & " , ( int ) main_window - > x + ( int ) main_window - > width / 2 , ( int ) main_window - > y + ( int ) main_window - > height / 2 ) ;
strcat ( about_cmd , coords ) ;
system ( about_cmd ) ;
2018-06-04 12:50:36 +03:00
redraw_window ( ) ;
2018-05-21 02:12:02 +03:00
}
2018-11-19 13:26:40 +03:00
static void launch_application ( char * app ) {
if ( ! fork ( ) ) {
2018-11-19 15:14:39 +03:00
if ( last_directory ) chdir ( last_directory ) ;
2018-11-19 13:26:40 +03:00
char * tmp = malloc ( strlen ( app ) + 10 ) ;
2018-11-19 15:14:39 +03:00
sprintf ( tmp , " %s " , app ) ;
2018-11-19 13:26:40 +03:00
char * args [ ] = { " /bin/sh " , " -c " , tmp , NULL } ;
execvp ( args [ 0 ] , args ) ;
exit ( 1 ) ;
}
}
static void launch_application_menu ( struct MenuEntry * self ) {
struct MenuEntry_Normal * _self = ( void * ) self ;
launch_application ( ( char * ) _self - > action ) ;
}
static void open_file ( struct File * f ) {
if ( f - > type = = 1 ) {
char tmp [ 1024 ] ;
2018-11-19 15:14:39 +03:00
if ( is_desktop_background ) {
sprintf ( tmp , " file-browser \" %s/%s \" " , last_directory , f - > name ) ;
launch_application ( tmp ) ;
} else {
sprintf ( tmp , " %s/%s " , last_directory , f - > name ) ;
load_directory ( tmp ) ;
reinitialize_contents ( ) ;
redraw_window ( ) ;
}
} else if ( f - > launcher [ 0 ] ) {
2018-11-19 13:26:40 +03:00
char tmp [ 4096 ] ;
if ( ! strcmp ( f - > launcher , " SELF " ) ) {
2018-11-19 15:14:39 +03:00
sprintf ( tmp , " exec ./%s " , f - > name ) ;
2018-11-19 13:26:40 +03:00
} else {
sprintf ( tmp , " %s \" %s \" " , f - > launcher , f - > name ) ;
}
launch_application ( tmp ) ;
}
}
static void _menu_action_open ( struct MenuEntry * self ) {
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
if ( file_pointers [ i ] - > selected ) {
open_file ( file_pointers [ i ] ) ;
}
}
}
static void toggle_selected ( int hilighted_offset , int modifiers ) {
struct File * f = get_file_at_offset ( hilighted_offset ) ;
if ( ! f ) return ;
f - > selected = ! f - > selected ;
if ( ! ( modifiers & KEY_MOD_LEFT_CTRL ) ) {
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
if ( file_pointers [ i ] ! = f & & file_pointers [ i ] - > selected ) {
file_pointers [ i ] - > selected = 0 ;
clear_offset ( i ) ;
draw_file ( file_pointers [ i ] , i ) ;
}
}
}
clear_offset ( hilighted_offset ) ;
draw_file ( f , hilighted_offset ) ;
redraw_window ( ) ;
}
2018-11-19 15:14:39 +03:00
static void draw_background ( int width , int height ) {
if ( wallpaper_buffer ) {
sprite_free ( wallpaper_buffer ) ;
}
sprite_t * wallpaper = malloc ( sizeof ( sprite_t ) ) ;
load_sprite ( wallpaper , " /usr/share/wallpaper.bmp " ) ;
wallpaper - > alpha = 0 ;
wallpaper_buffer = create_sprite ( width , height , 0 ) ;
gfx_context_t * ctx = init_graphics_sprite ( wallpaper_buffer ) ;
float x = ( float ) width / ( float ) wallpaper - > width ;
float y = ( float ) height / ( float ) wallpaper - > height ;
int nh = ( int ) ( x * ( float ) wallpaper - > height ) ;
int nw = ( int ) ( y * ( float ) wallpaper - > width ) ;
draw_fill ( ctx , rgb ( 0 , 0 , 0 ) ) ;
if ( nw = = wallpaper - > width & & nh = = wallpaper - > height ) {
// special case
draw_sprite ( ctx , wallpaper , 0 , 0 ) ;
} else if ( nw > = width ) {
draw_sprite_scaled ( ctx , wallpaper , ( ( int ) width - nw ) / 2 , 0 , nw + 2 , height ) ;
} else {
draw_sprite_scaled ( ctx , wallpaper , 0 , ( ( int ) height - nh ) / 2 , width + 2 , nh ) ;
}
sprite_free ( wallpaper ) ;
free ( ctx ) ;
}
2018-05-21 02:12:02 +03:00
int main ( int argc , char * argv [ ] ) {
yctx = yutani_init ( ) ;
init_decorations ( ) ;
2018-11-19 15:14:39 +03:00
if ( argc > 1 & & ! strcmp ( argv [ 1 ] , " --wallpaper " ) ) {
is_desktop_background = 1 ;
signal ( SIGUSR2 , sig_usr2 ) ;
draw_background ( yctx - > display_width , yctx - > display_height ) ;
main_window = yutani_window_create ( yctx , yctx - > display_width , yctx - > display_height ) ;
yutani_window_move ( yctx , main_window , 0 , 0 ) ;
yutani_set_stack ( yctx , main_window , YUTANI_ZORDER_BOTTOM ) ;
} else {
main_window = yutani_window_create ( yctx , 800 , 600 ) ;
yutani_window_move ( yctx , main_window , yctx - > display_width / 2 - main_window - > width / 2 , yctx - > display_height / 2 - main_window - > height / 2 ) ;
}
2018-05-21 02:12:02 +03:00
ctx = init_graphics_yutani_double_buffer ( main_window ) ;
2018-09-12 06:53:08 +03:00
struct decor_bounds bounds ;
2018-11-19 15:14:39 +03:00
_decor_get_bounds ( main_window , & bounds ) ;
2018-09-12 06:53:08 +03:00
2018-11-18 04:25:24 +03:00
set_title ( NULL ) ;
2018-05-21 02:12:02 +03:00
menu_bar . entries = menu_entries ;
menu_bar . redraw_callback = redraw_window ;
menu_bar . set = menu_set_create ( ) ;
struct MenuList * m = menu_create ( ) ; /* File */
menu_insert ( m , menu_create_normal ( " exit " , NULL , " Exit " , _menu_action_exit ) ) ;
menu_set_insert ( menu_bar . set , " file " , m ) ;
m = menu_create ( ) ; /* Go */
2018-08-14 11:13:38 +03:00
/* TODO implement input dialog for Path... */
#if 0
2018-05-21 02:12:02 +03:00
menu_insert ( m , menu_create_normal ( " open " , NULL , " Path... " , _menu_action_input_path ) ) ;
menu_insert ( m , menu_create_separator ( ) ) ;
2018-08-14 11:13:38 +03:00
# endif
2018-05-21 02:12:02 +03:00
menu_insert ( m , menu_create_normal ( " home " , getenv ( " HOME " ) , " Home " , _menu_action_navigate ) ) ;
menu_insert ( m , menu_create_normal ( NULL , " / " , " File System " , _menu_action_navigate ) ) ;
menu_insert ( m , menu_create_normal ( " up " , NULL , " Up " , _menu_action_up ) ) ;
menu_set_insert ( menu_bar . set , " go " , m ) ;
m = menu_create ( ) ;
menu_insert ( m , menu_create_normal ( " help " , NULL , " Contents " , _menu_action_help ) ) ;
menu_insert ( m , menu_create_separator ( ) ) ;
menu_insert ( m , menu_create_normal ( " star " , NULL , " About " APPLICATION_TITLE , _menu_action_about ) ) ;
menu_set_insert ( menu_bar . set , " help " , m ) ;
2018-11-19 15:14:39 +03:00
available_height = ctx - > height - menu_bar_height - bounds . height ;
2018-11-18 03:38:47 +03:00
context_menu = menu_create ( ) ; /* Right-click menu */
2018-11-19 13:26:40 +03:00
menu_insert ( context_menu , menu_create_normal ( NULL , NULL , " Open " , _menu_action_open ) ) ;
2018-11-18 13:16:22 +03:00
menu_insert ( context_menu , menu_create_separator ( ) ) ;
menu_insert ( context_menu , menu_create_normal ( NULL , NULL , " Copy " , _menu_action_copy ) ) ;
2018-11-19 13:26:40 +03:00
menu_insert ( context_menu , menu_create_separator ( ) ) ;
2018-11-19 15:14:39 +03:00
if ( ! is_desktop_background ) {
menu_insert ( context_menu , menu_create_normal ( " up " , NULL , " Up " , _menu_action_up ) ) ;
}
menu_insert ( context_menu , menu_create_normal ( " refresh " , NULL , " Refresh " , _menu_action_refresh ) ) ;
2018-11-19 13:26:40 +03:00
menu_insert ( context_menu , menu_create_normal ( " utilities-terminal " , " terminal " , " Open Terminal " , launch_application_menu ) ) ;
2018-11-18 03:38:47 +03:00
2018-11-19 13:26:40 +03:00
char tmp [ 1024 ] ;
getcwd ( tmp , 1024 ) ;
load_directory ( tmp ) ;
2018-08-10 07:57:24 +03:00
reinitialize_contents ( ) ;
2018-05-21 02:12:02 +03:00
redraw_window ( ) ;
2018-11-18 12:59:29 +03:00
uint64_t last_click = 0 ; /* For double click */
2018-11-18 13:02:25 +03:00
int last_click_offset = - 1 ;
2018-11-18 13:18:35 +03:00
int modifiers = 0 ; /* For ctrl-click */
2018-11-18 12:59:29 +03:00
2018-05-21 02:12:02 +03:00
while ( application_running ) {
2018-11-19 13:26:40 +03:00
waitpid ( - 1 , NULL , WNOHANG ) ;
2018-05-21 02:12:02 +03:00
yutani_msg_t * m = yutani_poll ( yctx ) ;
while ( m ) {
2018-07-21 19:24:22 +03:00
if ( menu_process_event ( yctx , m ) ) {
redraw_window ( ) ;
}
2018-05-21 02:12:02 +03:00
switch ( m - > type ) {
2018-11-19 15:14:39 +03:00
case YUTANI_MSG_WELCOME :
if ( is_desktop_background ) {
yutani_window_resize_offer ( yctx , main_window , yctx - > display_width , yctx - > display_height ) ;
}
break ;
2018-05-21 02:12:02 +03:00
case YUTANI_MSG_KEY_EVENT :
{
struct yutani_msg_key_event * ke = ( void * ) m - > data ;
2018-11-18 12:59:29 +03:00
modifiers = ke - > event . modifiers ;
2018-05-21 02:12:02 +03:00
if ( ke - > event . action = = KEY_ACTION_DOWN & & ke - > event . keycode = = ' q ' ) {
_menu_action_exit ( NULL ) ;
}
}
break ;
case YUTANI_MSG_WINDOW_FOCUS_CHANGE :
{
struct yutani_msg_window_focus_change * wf = ( void * ) m - > data ;
yutani_window_t * win = hashmap_get ( yctx - > windows , ( void * ) wf - > wid ) ;
2018-07-21 19:24:22 +03:00
if ( win = = main_window ) {
2018-07-24 03:20:11 +03:00
win - > focused = wf - > focused ;
redraw_window ( ) ;
2018-05-21 02:12:02 +03:00
}
}
break ;
case YUTANI_MSG_RESIZE_OFFER :
{
struct yutani_msg_window_resize * wr = ( void * ) m - > data ;
if ( wr - > wid = = main_window - > wid ) {
resize_finish ( wr - > width , wr - > height ) ;
}
}
break ;
case YUTANI_MSG_WINDOW_MOUSE_EVENT :
{
struct yutani_msg_window_mouse_event * me = ( void * ) m - > data ;
yutani_window_t * win = hashmap_get ( yctx - > windows , ( void * ) me - > wid ) ;
2018-09-12 06:53:08 +03:00
struct decor_bounds bounds ;
2018-11-19 15:14:39 +03:00
_decor_get_bounds ( win , & bounds ) ;
2018-05-21 02:12:02 +03:00
if ( win = = main_window ) {
int result = decor_handle_event ( yctx , m ) ;
switch ( result ) {
case DECOR_CLOSE :
_menu_action_exit ( NULL ) ;
break ;
case DECOR_RIGHT :
/* right click in decoration, show appropriate menu */
decor_show_default_menu ( main_window , main_window - > x + me - > new_x , main_window - > y + me - > new_y ) ;
break ;
default :
/* Other actions */
break ;
}
/* Menu bar */
menu_bar_mouse_event ( yctx , main_window , & menu_bar , me , me - > new_x , me - > new_y ) ;
2018-08-12 07:59:00 +03:00
2018-11-19 15:14:39 +03:00
if ( me - > new_y > ( int ) ( bounds . top_height + menu_bar_height ) & &
2018-09-12 06:53:08 +03:00
me - > new_y < ( int ) ( main_window - > height - bounds . bottom_height ) & &
me - > new_x > ( int ) ( bounds . left_width ) & &
me - > new_x < ( int ) ( main_window - > width - bounds . right_width ) ) {
2018-08-12 07:59:00 +03:00
if ( me - > buttons & YUTANI_MOUSE_SCROLL_UP ) {
/* Scroll up */
2018-08-12 14:03:37 +03:00
scroll_offset - = SCROLL_AMOUNT ;
2018-08-12 07:59:00 +03:00
if ( scroll_offset < 0 ) {
scroll_offset = 0 ;
}
redraw_window ( ) ;
} else if ( me - > buttons & YUTANI_MOUSE_SCROLL_DOWN ) {
if ( available_height > contents - > height ) {
scroll_offset = 0 ;
} else {
2018-08-12 14:03:37 +03:00
scroll_offset + = SCROLL_AMOUNT ;
2018-08-12 07:59:00 +03:00
if ( scroll_offset > contents - > height - available_height ) {
scroll_offset = contents - > height - available_height ;
}
}
redraw_window ( ) ;
}
2018-08-12 14:03:37 +03:00
/* Get offset into contents */
2018-11-19 15:14:39 +03:00
int y_into = me - > new_y - bounds . top_height - menu_bar_height + scroll_offset ;
2018-11-18 12:36:03 +03:00
int x_into = me - > new_x - bounds . left_width ;
int offset = ( y_into / FILE_HEIGHT ) * FILE_PTR_WIDTH + x_into / FILE_WIDTH ;
if ( x_into > FILE_PTR_WIDTH * FILE_WIDTH ) {
offset = - 1 ;
}
2018-08-12 14:03:37 +03:00
if ( offset ! = hilighted_offset ) {
int old_offset = hilighted_offset ;
hilighted_offset = offset ;
if ( old_offset ! = - 1 ) {
clear_offset ( old_offset ) ;
struct File * f = get_file_at_offset ( old_offset ) ;
if ( f ) {
clear_offset ( old_offset ) ;
draw_file ( f , old_offset ) ;
}
}
struct File * f = get_file_at_offset ( hilighted_offset ) ;
if ( f ) {
clear_offset ( hilighted_offset ) ;
draw_file ( f , hilighted_offset ) ;
}
redraw_window ( ) ;
}
if ( me - > command = = YUTANI_MOUSE_EVENT_CLICK | | _close_enough ( me ) ) {
struct File * f = get_file_at_offset ( hilighted_offset ) ;
2018-11-18 12:59:29 +03:00
if ( f ) {
2018-11-18 13:02:25 +03:00
if ( last_click_offset = = hilighted_offset & & precise_time_since ( last_click ) < 400 ) {
2018-11-19 13:26:40 +03:00
open_file ( f ) ;
2018-11-18 12:59:29 +03:00
last_click = 0 ;
} else {
last_click = precise_current_time ( ) ;
2018-11-18 13:02:25 +03:00
last_click_offset = hilighted_offset ;
2018-11-19 13:26:40 +03:00
toggle_selected ( hilighted_offset , modifiers ) ;
2018-11-18 12:59:29 +03:00
}
} else {
2018-11-18 13:18:35 +03:00
if ( ! ( modifiers & KEY_MOD_LEFT_CTRL ) ) {
2018-11-18 12:59:29 +03:00
for ( int i = 0 ; i < file_pointers_len ; + + i ) {
if ( file_pointers [ i ] - > selected ) {
file_pointers [ i ] - > selected = 0 ;
clear_offset ( i ) ;
draw_file ( file_pointers [ i ] , i ) ;
}
}
redraw_window ( ) ;
}
2018-08-12 14:03:37 +03:00
}
2018-11-18 03:38:47 +03:00
} else if ( me - > buttons & YUTANI_MOUSE_BUTTON_RIGHT ) {
if ( ! context_menu - > window ) {
2018-11-19 13:26:40 +03:00
struct File * f = get_file_at_offset ( hilighted_offset ) ;
if ( f & & ! f - > selected ) {
toggle_selected ( hilighted_offset , modifiers ) ;
}
2018-11-18 03:38:47 +03:00
menu_show ( context_menu , main_window - > ctx ) ;
yutani_window_move ( main_window - > ctx , context_menu - > window , me - > new_x + main_window - > x , me - > new_y + main_window - > y ) ;
}
2018-08-12 14:03:37 +03:00
}
} else {
int old_offset = hilighted_offset ;
hilighted_offset = - 1 ;
if ( old_offset ! = - 1 ) {
clear_offset ( old_offset ) ;
struct File * f = get_file_at_offset ( old_offset ) ;
if ( f ) {
clear_offset ( old_offset ) ;
draw_file ( f , old_offset ) ;
}
redraw_window ( ) ;
}
2018-08-12 07:59:00 +03:00
}
2018-05-21 02:12:02 +03:00
}
}
break ;
case YUTANI_MSG_WINDOW_CLOSE :
case YUTANI_MSG_SESSION_END :
_menu_action_exit ( NULL ) ;
break ;
default :
break ;
}
free ( m ) ;
m = yutani_poll_async ( yctx ) ;
}
}
}