/* Panel managing. Copyright (C) 1994-2024 Free Software Foundation, Inc. Written by: Slava Zanko , 2013 Andrew Borodin , 2013-2023 This file is part of the Midnight Commander. The Midnight Commander is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. The Midnight Commander is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /** \file info.c * \brief Source: panel managing */ #include #include #include #include #include /* PRIuMAX */ #include "lib/global.h" #include "lib/unixcompat.h" #include "lib/tty/tty.h" #include "lib/tty/key.h" /* is_idle() */ #include "lib/skin.h" #include "lib/strutil.h" #include "lib/timefmt.h" /* file_date() */ #include "lib/util.h" #include "lib/widget.h" #include "src/setup.h" /* panels_options */ #include "filemanager.h" /* the_menubar */ #include "layout.h" #include "mountlist.h" #ifdef ENABLE_EXT2FS_ATTR #include "cmd.h" /* chattr_get_as_str() */ #endif #include "info.h" /*** global variables ****************************************************************************/ /*** file scope macro definitions ****************************************************************/ /*** file scope type declarations ****************************************************************/ struct WInfo { Widget widget; gboolean ready; }; /*** forward declarations (file scope functions) *************************************************/ /*** file scope variables ************************************************************************/ static struct my_statfs myfs_stats; /* --------------------------------------------------------------------------------------------- */ /*** file scope functions ************************************************************************/ /* --------------------------------------------------------------------------------------------- */ static void info_box (WInfo * info) { Widget *w = WIDGET (info); const char *title = _("Information"); const int len = str_term_width1 (title); tty_set_normal_attrs (); tty_setcolor (NORMAL_COLOR); widget_erase (w); tty_draw_box (w->rect.y, w->rect.x, w->rect.lines, w->rect.cols, FALSE); widget_gotoyx (w, 0, (w->rect.cols - len - 2) / 2); tty_printf (" %s ", title); widget_gotoyx (w, 2, 0); tty_print_alt_char (ACS_LTEE, FALSE); widget_gotoyx (w, 2, w->rect.cols - 1); tty_print_alt_char (ACS_RTEE, FALSE); tty_draw_hline (w->rect.y + 2, w->rect.x + 1, ACS_HLINE, w->rect.cols - 2); } /* --------------------------------------------------------------------------------------------- */ static void info_show_info (WInfo * info) { const WRect *w = &CONST_WIDGET (info)->rect; const file_entry_t *fe; static int i18n_adjust = 0; static const char *file_label; GString *buff; struct stat st; char rp_cwd[PATH_MAX]; const char *p_rp_cwd; if (!is_idle ()) return; info_box (info); tty_setcolor (MARKED_COLOR); widget_gotoyx (w, 1, 3); tty_printf (_("Midnight Commander %s"), mc_global.mc_version); if (!info->ready) return; if (get_current_type () != view_listing) return; /* don't rely on vpath CWD when cd_symlinks enabled */ p_rp_cwd = mc_realpath (vfs_path_as_str (current_panel->cwd_vpath), rp_cwd); if (p_rp_cwd == NULL) p_rp_cwd = vfs_path_as_str (current_panel->cwd_vpath); my_statfs (&myfs_stats, p_rp_cwd); fe = panel_current_entry (current_panel); st = fe->st; /* Print only lines which fit */ if (i18n_adjust == 0) { /* This printf pattern string is used as a reference for size */ file_label = _("File: %s"); i18n_adjust = str_term_width1 (file_label) + 2; } tty_setcolor (NORMAL_COLOR); buff = g_string_new (""); switch (w->lines - 2) { /* Note: all cases are fall-throughs */ default: MC_FALLTHROUGH; case 17: widget_gotoyx (w, 17, 3); if ((myfs_stats.nfree == 0 && myfs_stats.nodes == 0) || (myfs_stats.nfree == (uintmax_t) (-1) && myfs_stats.nodes == (uintmax_t) (-1))) tty_print_string (_("No node information")); else if (myfs_stats.nfree == (uintmax_t) (-1)) tty_printf ("%s - / %" PRIuMAX, _("Free nodes:"), myfs_stats.nodes); else if (myfs_stats.nodes == (uintmax_t) (-1)) tty_printf ("%s %" PRIuMAX " / -", _("Free nodes:"), myfs_stats.nfree); else tty_printf ("%s %" PRIuMAX " / %" PRIuMAX " (%d%%)", _("Free nodes:"), myfs_stats.nfree, myfs_stats.nodes, myfs_stats.nodes == 0 ? 0 : (int) (100 * (long double) myfs_stats.nfree / myfs_stats.nodes)); MC_FALLTHROUGH; case 16: widget_gotoyx (w, 16, 3); if (myfs_stats.avail == 0 && myfs_stats.total == 0) tty_print_string (_("No space information")); else { char buffer1[6], buffer2[6]; size_trunc_len (buffer1, 5, myfs_stats.avail, 1, panels_options.kilobyte_si); size_trunc_len (buffer2, 5, myfs_stats.total, 1, panels_options.kilobyte_si); tty_printf (_("Free space: %s / %s (%d%%)"), buffer1, buffer2, myfs_stats.total == 0 ? 0 : (int) (100 * (long double) myfs_stats.avail / myfs_stats.total)); } MC_FALLTHROUGH; case 15: widget_gotoyx (w, 15, 3); tty_printf (_("Type: %s"), myfs_stats.typename ? myfs_stats.typename : _("non-local vfs")); if (myfs_stats.type != 0xffff && myfs_stats.type != -1) tty_printf (" (%Xh)", (unsigned int) myfs_stats.type); MC_FALLTHROUGH; case 14: widget_gotoyx (w, 14, 3); str_printf (buff, _("Device: %s"), str_trunc (myfs_stats.device, w->cols - i18n_adjust)); tty_print_string (buff->str); g_string_set_size (buff, 0); MC_FALLTHROUGH; case 13: widget_gotoyx (w, 13, 3); str_printf (buff, _("Filesystem: %s"), str_trunc (myfs_stats.mpoint, w->cols - i18n_adjust)); tty_print_string (buff->str); g_string_set_size (buff, 0); MC_FALLTHROUGH; case 12: widget_gotoyx (w, 12, 3); str_printf (buff, _("Accessed: %s"), file_date (st.st_atime)); tty_print_string (buff->str); g_string_set_size (buff, 0); MC_FALLTHROUGH; case 11: widget_gotoyx (w, 11, 3); str_printf (buff, _("Modified: %s"), file_date (st.st_mtime)); tty_print_string (buff->str); g_string_set_size (buff, 0); MC_FALLTHROUGH; case 10: widget_gotoyx (w, 10, 3); /* The field st_ctime is changed by writing or by setting inode information (i.e., owner, group, link count, mode, etc.). */ /* TRANSLATORS: Time of last status change as in stat(2) man. */ str_printf (buff, _("Changed: %s"), file_date (st.st_ctime)); tty_print_string (buff->str); g_string_set_size (buff, 0); MC_FALLTHROUGH; case 9: widget_gotoyx (w, 9, 3); #ifdef HAVE_STRUCT_STAT_ST_RDEV if (S_ISCHR (st.st_mode) || S_ISBLK (st.st_mode)) tty_printf (_("Dev. type: major %lu, minor %lu"), (unsigned long) major (st.st_rdev), (unsigned long) minor (st.st_rdev)); else #endif { char buffer[10]; size_trunc_len (buffer, 9, st.st_size, 0, panels_options.kilobyte_si); tty_printf (_("Size: %s"), buffer); #ifdef HAVE_STRUCT_STAT_ST_BLOCKS tty_printf (ngettext (" (%lu block)", " (%lu blocks)", (unsigned long) st.st_blocks), (unsigned long) st.st_blocks); #endif } MC_FALLTHROUGH; case 8: widget_gotoyx (w, 8, 3); tty_printf (_("Owner: %s/%s"), get_owner (st.st_uid), get_group (st.st_gid)); MC_FALLTHROUGH; case 7: widget_gotoyx (w, 7, 3); tty_printf (_("Links: %d"), (int) st.st_nlink); MC_FALLTHROUGH; case 6: widget_gotoyx (w, 6, 3); { vfs_path_t *vpath; #ifdef ENABLE_EXT2FS_ATTR unsigned long attr; #endif vpath = vfs_path_from_str (fe->fname->str); #ifdef ENABLE_EXT2FS_ATTR if (mc_fgetflags (vpath, &attr) == 0) tty_printf (_("Attributes: %s"), chattr_get_as_str (attr)); else #endif tty_print_string (_("Attributes: unavailable")); vfs_path_free (vpath, TRUE); } MC_FALLTHROUGH; case 5: widget_gotoyx (w, 5, 3); tty_printf (_("Mode: %s (%04o)"), string_perm (st.st_mode), (unsigned) st.st_mode & 07777); MC_FALLTHROUGH; case 4: widget_gotoyx (w, 4, 3); tty_printf (_("Location: %Xh:%Xh"), (unsigned int) st.st_dev, (unsigned int) st.st_ino); MC_FALLTHROUGH; case 3: { const char *fname; widget_gotoyx (w, 3, 2); fname = fe->fname->str; str_printf (buff, file_label, str_trunc (fname, w->cols - i18n_adjust)); tty_print_string (buff->str); } MC_FALLTHROUGH; case 2: MC_FALLTHROUGH; case 1: MC_FALLTHROUGH; case 0: ; } /* switch */ g_string_free (buff, TRUE); } /* --------------------------------------------------------------------------------------------- */ static void info_hook (void *data) { WInfo *info = (WInfo *) data; Widget *other_widget; other_widget = get_panel_widget (get_current_index ()); if (!other_widget) return; if (widget_overlapped (WIDGET (info), other_widget)) return; info->ready = TRUE; info_show_info (info); } /* --------------------------------------------------------------------------------------------- */ static cb_ret_t info_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data) { WInfo *info = (WInfo *) w; switch (msg) { case MSG_INIT: init_my_statfs (); add_hook (&select_file_hook, info_hook, info); info->ready = FALSE; return MSG_HANDLED; case MSG_DRAW: info_hook (info); return MSG_HANDLED; case MSG_DESTROY: delete_hook (&select_file_hook, info_hook); free_my_statfs (); return MSG_HANDLED; default: return widget_default_callback (w, sender, msg, parm, data); } } /* --------------------------------------------------------------------------------------------- */ /*** public functions ****************************************************************************/ /* --------------------------------------------------------------------------------------------- */ WInfo * info_new (int y, int x, int lines, int cols) { WRect r = { y, x, lines, cols }; WInfo *info; Widget *w; info = g_new (struct WInfo, 1); w = WIDGET (info); widget_init (w, &r, info_callback, NULL); return info; } /* --------------------------------------------------------------------------------------------- */