Calculate pauses in file operations.
[midnight-commander.git] / src / filemanager / info.c
blobc8570d8fb01e234b2eb4df32c4c94145c36499d1
1 /*
2 Panel managing.
4 Copyright (C) 1994-2024
5 Free Software Foundation, Inc.
7 Written by:
8 Slava Zanko <slavazanko@gmail.com>, 2013
9 Andrew Borodin <aborodin@vmail.ru>, 2013-2023
11 This file is part of the Midnight Commander.
13 The Midnight Commander is free software: you can redistribute it
14 and/or modify it under the terms of the GNU General Public License as
15 published by the Free Software Foundation, either version 3 of the License,
16 or (at your option) any later version.
18 The Midnight Commander is distributed in the hope that it will be useful,
19 but WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 GNU General Public License for more details.
23 You should have received a copy of the GNU General Public License
24 along with this program. If not, see <http://www.gnu.org/licenses/>.
27 /** \file info.c
28 * \brief Source: panel managing
31 #include <config.h>
33 #include <stdio.h>
34 #include <stdlib.h>
35 #include <sys/stat.h>
36 #include <inttypes.h> /* PRIuMAX */
38 #include "lib/global.h"
39 #include "lib/unixcompat.h"
40 #include "lib/tty/tty.h"
41 #include "lib/tty/key.h" /* is_idle() */
42 #include "lib/skin.h"
43 #include "lib/strutil.h"
44 #include "lib/timefmt.h" /* file_date() */
45 #include "lib/util.h"
46 #include "lib/widget.h"
48 #include "src/setup.h" /* panels_options */
50 #include "filemanager.h" /* the_menubar */
51 #include "layout.h"
52 #include "mountlist.h"
53 #ifdef ENABLE_EXT2FS_ATTR
54 #include "cmd.h" /* chattr_get_as_str() */
55 #endif
57 #include "info.h"
59 /*** global variables ****************************************************************************/
61 /*** file scope macro definitions ****************************************************************/
63 /*** file scope type declarations ****************************************************************/
65 struct WInfo
67 Widget widget;
68 gboolean ready;
71 /*** forward declarations (file scope functions) *************************************************/
73 /*** file scope variables ************************************************************************/
75 static struct my_statfs myfs_stats;
77 /* --------------------------------------------------------------------------------------------- */
78 /*** file scope functions ************************************************************************/
79 /* --------------------------------------------------------------------------------------------- */
81 static void
82 info_box (WInfo *info)
84 Widget *w = WIDGET (info);
86 const char *title = _("Information");
87 const int len = str_term_width1 (title);
89 tty_set_normal_attrs ();
90 tty_setcolor (NORMAL_COLOR);
91 widget_erase (w);
92 tty_draw_box (w->rect.y, w->rect.x, w->rect.lines, w->rect.cols, FALSE);
94 widget_gotoyx (w, 0, (w->rect.cols - len - 2) / 2);
95 tty_printf (" %s ", title);
97 widget_gotoyx (w, 2, 0);
98 tty_print_alt_char (ACS_LTEE, FALSE);
99 widget_gotoyx (w, 2, w->rect.cols - 1);
100 tty_print_alt_char (ACS_RTEE, FALSE);
101 tty_draw_hline (w->rect.y + 2, w->rect.x + 1, ACS_HLINE, w->rect.cols - 2);
104 /* --------------------------------------------------------------------------------------------- */
106 static void
107 info_show_info (WInfo *info)
109 const WRect *w = &CONST_WIDGET (info)->rect;
110 const file_entry_t *fe;
111 static int i18n_adjust = 0;
112 static const char *file_label;
113 GString *buff;
114 struct stat st;
115 char rp_cwd[PATH_MAX];
116 const char *p_rp_cwd;
118 if (!is_idle ())
119 return;
121 info_box (info);
123 tty_setcolor (MARKED_COLOR);
124 widget_gotoyx (w, 1, 3);
125 tty_printf (_("Midnight Commander %s"), mc_global.mc_version);
127 if (!info->ready)
128 return;
130 if (get_current_type () != view_listing)
131 return;
133 /* don't rely on vpath CWD when cd_symlinks enabled */
134 p_rp_cwd = mc_realpath (vfs_path_as_str (current_panel->cwd_vpath), rp_cwd);
135 if (p_rp_cwd == NULL)
136 p_rp_cwd = vfs_path_as_str (current_panel->cwd_vpath);
138 my_statfs (&myfs_stats, p_rp_cwd);
140 fe = panel_current_entry (current_panel);
141 if (fe == NULL)
142 return;
144 st = fe->st;
146 /* Print only lines which fit */
148 if (i18n_adjust == 0)
150 /* This printf pattern string is used as a reference for size */
151 file_label = _("File: %s");
152 i18n_adjust = str_term_width1 (file_label) + 2;
155 tty_setcolor (NORMAL_COLOR);
157 buff = g_string_new ("");
159 switch (w->lines - 2)
161 /* Note: all cases are fall-throughs */
163 default:
164 MC_FALLTHROUGH;
165 case 17:
166 widget_gotoyx (w, 17, 3);
167 if ((myfs_stats.nfree == 0 && myfs_stats.nodes == 0) ||
168 (myfs_stats.nfree == (uintmax_t) (-1) && myfs_stats.nodes == (uintmax_t) (-1)))
169 tty_print_string (_("No node information"));
170 else if (myfs_stats.nfree == (uintmax_t) (-1))
171 tty_printf ("%s - / %" PRIuMAX, _("Free nodes:"), myfs_stats.nodes);
172 else if (myfs_stats.nodes == (uintmax_t) (-1))
173 tty_printf ("%s %" PRIuMAX " / -", _("Free nodes:"), myfs_stats.nfree);
174 else
175 tty_printf ("%s %" PRIuMAX " / %" PRIuMAX " (%d%%)",
176 _("Free nodes:"),
177 myfs_stats.nfree, myfs_stats.nodes,
178 myfs_stats.nodes == 0 ? 0 :
179 (int) (100 * (long double) myfs_stats.nfree / myfs_stats.nodes));
180 MC_FALLTHROUGH;
181 case 16:
182 widget_gotoyx (w, 16, 3);
183 if (myfs_stats.avail == 0 && myfs_stats.total == 0)
184 tty_print_string (_("No space information"));
185 else
187 char buffer1[6], buffer2[6];
189 size_trunc_len (buffer1, 5, myfs_stats.avail, 1, panels_options.kilobyte_si);
190 size_trunc_len (buffer2, 5, myfs_stats.total, 1, panels_options.kilobyte_si);
191 tty_printf (_("Free space: %s / %s (%d%%)"), buffer1, buffer2,
192 myfs_stats.total == 0 ? 0 :
193 (int) (100 * (long double) myfs_stats.avail / myfs_stats.total));
195 MC_FALLTHROUGH;
196 case 15:
197 widget_gotoyx (w, 15, 3);
198 tty_printf (_("Type: %s"),
199 myfs_stats.typename ? myfs_stats.typename : _("non-local vfs"));
200 if (myfs_stats.type != 0xffff && myfs_stats.type != -1)
201 tty_printf (" (%Xh)", (unsigned int) myfs_stats.type);
202 MC_FALLTHROUGH;
203 case 14:
204 widget_gotoyx (w, 14, 3);
205 str_printf (buff, _("Device: %s"),
206 str_trunc (myfs_stats.device, w->cols - i18n_adjust));
207 tty_print_string (buff->str);
208 g_string_set_size (buff, 0);
209 MC_FALLTHROUGH;
210 case 13:
211 widget_gotoyx (w, 13, 3);
212 str_printf (buff, _("Filesystem: %s"),
213 str_trunc (myfs_stats.mpoint, w->cols - i18n_adjust));
214 tty_print_string (buff->str);
215 g_string_set_size (buff, 0);
216 MC_FALLTHROUGH;
217 case 12:
218 widget_gotoyx (w, 12, 3);
219 str_printf (buff, _("Accessed: %s"), file_date (st.st_atime));
220 tty_print_string (buff->str);
221 g_string_set_size (buff, 0);
222 MC_FALLTHROUGH;
223 case 11:
224 widget_gotoyx (w, 11, 3);
225 str_printf (buff, _("Modified: %s"), file_date (st.st_mtime));
226 tty_print_string (buff->str);
227 g_string_set_size (buff, 0);
228 MC_FALLTHROUGH;
229 case 10:
230 widget_gotoyx (w, 10, 3);
231 /* The field st_ctime is changed by writing or by setting inode
232 information (i.e., owner, group, link count, mode, etc.). */
233 /* TRANSLATORS: Time of last status change as in stat(2) man. */
234 str_printf (buff, _("Changed: %s"), file_date (st.st_ctime));
235 tty_print_string (buff->str);
236 g_string_set_size (buff, 0);
237 MC_FALLTHROUGH;
238 case 9:
239 widget_gotoyx (w, 9, 3);
240 #ifdef HAVE_STRUCT_STAT_ST_RDEV
241 if (S_ISCHR (st.st_mode) || S_ISBLK (st.st_mode))
242 tty_printf (_("Dev. type: major %lu, minor %lu"),
243 (unsigned long) major (st.st_rdev), (unsigned long) minor (st.st_rdev));
244 else
245 #endif
247 char buffer[10];
248 size_trunc_len (buffer, 9, st.st_size, 0, panels_options.kilobyte_si);
249 tty_printf (_("Size: %s"), buffer);
250 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
251 tty_printf (ngettext (" (%lu block)", " (%lu blocks)",
252 (unsigned long) st.st_blocks), (unsigned long) st.st_blocks);
253 #endif
255 MC_FALLTHROUGH;
256 case 8:
257 widget_gotoyx (w, 8, 3);
258 tty_printf (_("Owner: %s/%s"), get_owner (st.st_uid), get_group (st.st_gid));
259 MC_FALLTHROUGH;
260 case 7:
261 widget_gotoyx (w, 7, 3);
262 tty_printf (_("Links: %d"), (int) st.st_nlink);
263 MC_FALLTHROUGH;
264 case 6:
265 widget_gotoyx (w, 6, 3);
267 #ifdef ENABLE_EXT2FS_ATTR
269 vfs_path_t *vpath;
270 unsigned long attr;
272 vpath = vfs_path_from_str (fe->fname->str);
274 if (mc_fgetflags (vpath, &attr) == 0)
275 tty_printf (_("Attributes: %s"), chattr_get_as_str (attr));
276 else
277 tty_print_string (_("Attributes: unavailable"));
279 vfs_path_free (vpath, TRUE);
281 #else
282 tty_print_string (_("Attributes: unavailable"));
283 #endif /* ENABLE_EXT2FS_ATTR */
284 MC_FALLTHROUGH;
285 case 5:
286 widget_gotoyx (w, 5, 3);
287 tty_printf (_("Mode: %s (%04o)"),
288 string_perm (st.st_mode), (unsigned) st.st_mode & 07777);
289 MC_FALLTHROUGH;
290 case 4:
291 widget_gotoyx (w, 4, 3);
292 tty_printf (_("Location: %Xh:%Xh"), (unsigned int) st.st_dev, (unsigned int) st.st_ino);
293 MC_FALLTHROUGH;
294 case 3:
296 const char *fname;
298 widget_gotoyx (w, 3, 2);
299 fname = fe->fname->str;
300 str_printf (buff, file_label, str_trunc (fname, w->cols - i18n_adjust));
301 tty_print_string (buff->str);
303 MC_FALLTHROUGH;
304 case 2:
305 MC_FALLTHROUGH;
306 case 1:
307 MC_FALLTHROUGH;
308 case 0:
310 } /* switch */
311 g_string_free (buff, TRUE);
314 /* --------------------------------------------------------------------------------------------- */
316 static void
317 info_hook (void *data)
319 WInfo *info = (WInfo *) data;
320 Widget *other_widget;
322 other_widget = get_panel_widget (get_current_index ());
323 if (!other_widget)
324 return;
325 if (widget_overlapped (WIDGET (info), other_widget))
326 return;
328 info->ready = TRUE;
329 info_show_info (info);
332 /* --------------------------------------------------------------------------------------------- */
334 static cb_ret_t
335 info_callback (Widget *w, Widget *sender, widget_msg_t msg, int parm, void *data)
337 WInfo *info = (WInfo *) w;
339 switch (msg)
341 case MSG_INIT:
342 init_my_statfs ();
343 add_hook (&select_file_hook, info_hook, info);
344 info->ready = FALSE;
345 return MSG_HANDLED;
347 case MSG_DRAW:
348 info_hook (info);
349 return MSG_HANDLED;
351 case MSG_DESTROY:
352 delete_hook (&select_file_hook, info_hook);
353 free_my_statfs ();
354 return MSG_HANDLED;
356 default:
357 return widget_default_callback (w, sender, msg, parm, data);
361 /* --------------------------------------------------------------------------------------------- */
362 /*** public functions ****************************************************************************/
363 /* --------------------------------------------------------------------------------------------- */
365 WInfo *
366 info_new (const WRect *r)
368 WInfo *info;
369 Widget *w;
371 info = g_new (struct WInfo, 1);
372 w = WIDGET (info);
373 widget_init (w, r, info_callback, NULL);
375 return info;
378 /* --------------------------------------------------------------------------------------------- */